Compare commits
12 Commits
Author | SHA1 | Date |
---|---|---|
georgemoralis | 1dfe491b7d | |
georgemoralis | 3f987b375f | |
georgemoralis | b1a66807a2 | |
georgemoralis | 24194e738f | |
georgemoralis | 20bd6b4c4b | |
georgemoralis | 2bf8eea333 | |
georgemoralis | 1613e22d99 | |
georgemoralis | 5c7545198e | |
georgemoralis | ebf3a27998 | |
georgemoralis | 2c1d455942 | |
georgemoralis | 27f6f6bfd6 | |
georgemoralis | 7ef3fe608a |
|
@ -10,7 +10,7 @@ if grep -nrI '\s$' src *.yml *.txt *.md Doxyfile .gitignore .gitmodules .ci* dis
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Default clang-format points to default 3.5 version one
|
# Default clang-format points to default 3.5 version one
|
||||||
CLANG_FORMAT=clang-format-17
|
CLANG_FORMAT=clang-format-15
|
||||||
$CLANG_FORMAT --version
|
$CLANG_FORMAT --version
|
||||||
|
|
||||||
if [ "$GITHUB_EVENT_NAME" = "pull_request" ]; then
|
if [ "$GITHUB_EVENT_NAME" = "pull_request" ]; then
|
||||||
|
|
|
@ -7,7 +7,6 @@ if [[ -z $GITHUB_WORKSPACE ]]; then
|
||||||
GITHUB_WORKSPACE="${PWD%/*}"
|
GITHUB_WORKSPACE="${PWD%/*}"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
export Qt6_DIR="/usr/lib/qt6"
|
|
||||||
export PATH="$Qt6_DIR/bin:$PATH"
|
export PATH="$Qt6_DIR/bin:$PATH"
|
||||||
|
|
||||||
# Prepare Tools for building the AppImage
|
# Prepare Tools for building the AppImage
|
||||||
|
@ -19,12 +18,8 @@ chmod a+x linuxdeploy-x86_64.AppImage
|
||||||
chmod a+x linuxdeploy-plugin-qt-x86_64.AppImage
|
chmod a+x linuxdeploy-plugin-qt-x86_64.AppImage
|
||||||
chmod a+x linuxdeploy-plugin-checkrt-x86_64.sh
|
chmod a+x linuxdeploy-plugin-checkrt-x86_64.sh
|
||||||
|
|
||||||
|
|
||||||
# Build AppImage
|
# Build AppImage
|
||||||
./linuxdeploy-x86_64.AppImage --appdir AppDir
|
./linuxdeploy-x86_64.AppImage --appdir AppDir
|
||||||
./linuxdeploy-plugin-checkrt-x86_64.sh --appdir AppDir
|
./linuxdeploy-plugin-checkrt-x86_64.sh --appdir AppDir
|
||||||
|
|
||||||
cp -a "$GITHUB_WORKSPACE/build/translations" AppDir/usr/bin
|
|
||||||
|
|
||||||
./linuxdeploy-x86_64.AppImage --appdir AppDir -d "$GITHUB_WORKSPACE"/.github/shadps4.desktop -e "$GITHUB_WORKSPACE"/build/shadps4 -i "$GITHUB_WORKSPACE"/.github/shadps4.png --plugin qt --output appimage
|
./linuxdeploy-x86_64.AppImage --appdir AppDir -d "$GITHUB_WORKSPACE"/.github/shadps4.desktop -e "$GITHUB_WORKSPACE"/build/shadps4 -i "$GITHUB_WORKSPACE"/.github/shadps4.png --plugin qt --output appimage
|
||||||
mv Shadps4-x86_64.AppImage Shadps4-qt.AppImage
|
mv Shadps4-x86_64.AppImage Shadps4-qt.AppImage
|
||||||
|
|
Before Width: | Height: | Size: 130 KiB After Width: | Height: | Size: 141 KiB |
|
@ -1,3 +1,4 @@
|
||||||
|
|
||||||
# SPDX-FileCopyrightText: 2021 yuzu Emulator Project
|
# SPDX-FileCopyrightText: 2021 yuzu Emulator Project
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
# SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
@ -14,4 +15,4 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: fsfe/reuse-action@v4
|
- uses: fsfe/reuse-action@v1
|
|
@ -17,11 +17,7 @@ jobs:
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Install
|
- name: Install
|
||||||
run: |
|
run: sudo apt-get install clang-format-15
|
||||||
wget -O - https://apt.llvm.org/llvm-snapshot.gpg.key | sudo apt-key add -
|
|
||||||
sudo add-apt-repository 'deb http://apt.llvm.org/jammy/ llvm-toolchain-jammy-17 main'
|
|
||||||
sudo apt update
|
|
||||||
sudo apt install clang-format-17
|
|
||||||
- name: Build
|
- name: Build
|
||||||
env:
|
env:
|
||||||
COMMIT_RANGE: ${{ github.event.pull_request.base.sha }}..${{ github.event.pull_request.head.sha }}
|
COMMIT_RANGE: ${{ github.event.pull_request.base.sha }}..${{ github.event.pull_request.head.sha }}
|
||||||
|
|
|
@ -10,23 +10,44 @@ on:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
# Customize the CMake build type here (Release, Debug, RelWithDebInfo, etc.)
|
||||||
BUILD_TYPE: Release
|
BUILD_TYPE: Release
|
||||||
|
CLANG_VER: 17
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
runs-on: ubuntu-24.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
with:
|
- name: Fetch submodules
|
||||||
submodules: recursive
|
run: git submodule update --init --recursive
|
||||||
|
|
||||||
- name: Install misc packages
|
- name: Install misc packages
|
||||||
run: >
|
run: >
|
||||||
sudo apt-get update && sudo apt install libx11-dev libxext-dev libwayland-dev libfuse2 clang build-essential qt6-base-dev qt6-tools-dev
|
sudo apt-get update && sudo apt install libx11-dev libgl1-mesa-glx mesa-common-dev libfuse2
|
||||||
|
libwayland-dev libxcb-keysyms1 libxcb-randr0 libxcb-render-util0 libxcb-icccm4 libxcb-image0-dev
|
||||||
|
libxcb-cursor-dev libxxhash-dev libvulkan-dev
|
||||||
|
|
||||||
|
- name: Install newer Clang
|
||||||
|
run: |
|
||||||
|
wget https://apt.llvm.org/llvm.sh
|
||||||
|
chmod +x ./llvm.sh
|
||||||
|
sudo ./llvm.sh ${{env.CLANG_VER}}
|
||||||
|
|
||||||
|
- name: Install Qt
|
||||||
|
uses: jurplel/install-qt-action@v3
|
||||||
|
with:
|
||||||
|
version: 6.6.1
|
||||||
|
host: linux
|
||||||
|
target: desktop
|
||||||
|
#arch: clang++-17
|
||||||
|
dir: ${{ runner.temp }}
|
||||||
|
#modules: qtcharts qt3d
|
||||||
|
setup-python: false
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure CMake
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++ -DENABLE_QT_GUI=ON
|
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_C_COMPILER=clang-${{env.CLANG_VER}} -DCMAKE_CXX_COMPILER=clang++-${{env.CLANG_VER}} -DENABLE_QT_GUI=ON
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
||||||
|
|
|
@ -8,13 +8,16 @@ on:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
# Customize the CMake build type here (Release, Debug, RelWithDebInfo, etc.)
|
||||||
BUILD_TYPE: Release
|
BUILD_TYPE: Release
|
||||||
|
CLANG_VER: 17
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
runs-on: ubuntu-24.04
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
@ -23,10 +26,16 @@ jobs:
|
||||||
|
|
||||||
- name: Install misc packages
|
- name: Install misc packages
|
||||||
run: >
|
run: >
|
||||||
sudo apt-get update && sudo apt install libx11-dev libxext-dev libwayland-dev libfuse2 clang build-essential
|
sudo apt-get update && sudo apt install libx11-dev libgl1-mesa-glx mesa-common-dev libfuse2 libwayland-dev libxxhash-dev libvulkan-dev
|
||||||
|
|
||||||
|
- name: Install newer Clang
|
||||||
|
run: |
|
||||||
|
wget https://apt.llvm.org/llvm.sh
|
||||||
|
chmod +x ./llvm.sh
|
||||||
|
sudo ./llvm.sh ${{env.CLANG_VER}}
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure CMake
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++
|
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_C_COMPILER=clang-${{env.CLANG_VER}} -DCMAKE_CXX_COMPILER=clang++-${{env.CLANG_VER}}
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
||||||
|
@ -37,6 +46,7 @@ jobs:
|
||||||
name: shadps4-ubuntu64
|
name: shadps4-ubuntu64
|
||||||
path: |
|
path: |
|
||||||
${{github.workspace}}/build/shadps4
|
${{github.workspace}}/build/shadps4
|
||||||
|
${{github.workspace}}/build/libSDL3.so.0.0.0
|
||||||
|
|
||||||
- name: Run AppImage packaging script
|
- name: Run AppImage packaging script
|
||||||
run: ./.github/linux-appimage-sdl.sh
|
run: ./.github/linux-appimage-sdl.sh
|
||||||
|
|
|
@ -1,61 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
name: macOS-Qt
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [ "main" ]
|
|
||||||
pull_request:
|
|
||||||
branches: [ "main" ]
|
|
||||||
|
|
||||||
env:
|
|
||||||
BUILD_TYPE: Release
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: macos-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
submodules: recursive
|
|
||||||
|
|
||||||
- name: Setup latest Xcode
|
|
||||||
uses: maxim-lobanov/setup-xcode@v1
|
|
||||||
with:
|
|
||||||
xcode-version: latest
|
|
||||||
|
|
||||||
- name: Install MoltenVK
|
|
||||||
run: |
|
|
||||||
arch -x86_64 /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
|
||||||
arch -x86_64 /usr/local/bin/brew install molten-vk
|
|
||||||
|
|
||||||
- name: Setup Qt
|
|
||||||
uses: jurplel/install-qt-action@v4
|
|
||||||
with:
|
|
||||||
version: 6.7.2
|
|
||||||
host: mac
|
|
||||||
target: desktop
|
|
||||||
arch: clang_64
|
|
||||||
archives: qtbase qttools
|
|
||||||
|
|
||||||
- name: Configure CMake
|
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_OSX_ARCHITECTURES=x86_64 -DENABLE_QT_GUI=ON
|
|
||||||
|
|
||||||
- name: Build
|
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel $(sysctl -n hw.ncpu)
|
|
||||||
|
|
||||||
- name: Package
|
|
||||||
run: |
|
|
||||||
mkdir upload
|
|
||||||
mv ${{github.workspace}}/build/shadps4.app upload
|
|
||||||
mv ${{github.workspace}}/build/translations upload
|
|
||||||
macdeployqt upload/shadps4.app
|
|
||||||
tar cf shadps4-macos-qt.tar.gz -C upload .
|
|
||||||
|
|
||||||
- name: Upload executable
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: shadps4-macos-qt
|
|
||||||
path: shadps4-macos-qt.tar.gz
|
|
|
@ -1,52 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
name: macOS
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches: [ "main" ]
|
|
||||||
pull_request:
|
|
||||||
branches: [ "main" ]
|
|
||||||
|
|
||||||
env:
|
|
||||||
BUILD_TYPE: Release
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
build:
|
|
||||||
runs-on: macos-latest
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
submodules: recursive
|
|
||||||
|
|
||||||
- name: Setup latest Xcode
|
|
||||||
uses: maxim-lobanov/setup-xcode@v1
|
|
||||||
with:
|
|
||||||
xcode-version: latest
|
|
||||||
|
|
||||||
- name: Install MoltenVK
|
|
||||||
run: |
|
|
||||||
arch -x86_64 /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
|
||||||
arch -x86_64 /usr/local/bin/brew install molten-vk
|
|
||||||
|
|
||||||
- name: Configure CMake
|
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_OSX_ARCHITECTURES=x86_64
|
|
||||||
|
|
||||||
- name: Build
|
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel $(sysctl -n hw.ncpu)
|
|
||||||
|
|
||||||
- name: Package
|
|
||||||
run: |
|
|
||||||
mkdir upload
|
|
||||||
mv ${{github.workspace}}/build/shadps4 upload
|
|
||||||
cp $(arch -x86_64 /usr/local/bin/brew --prefix)/opt/molten-vk/lib/libMoltenVK.dylib upload
|
|
||||||
install_name_tool -add_rpath "@loader_path" upload/shadps4
|
|
||||||
tar cf shadps4-macos-sdl.tar.gz -C upload .
|
|
||||||
|
|
||||||
- name: Upload executable
|
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
|
||||||
name: shadps4-macos-sdl
|
|
||||||
path: shadps4-macos-sdl.tar.gz
|
|
|
@ -10,41 +10,46 @@ on:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
# Customize the CMake build type here (Release, Debug, RelWithDebInfo, etc.)
|
||||||
BUILD_TYPE: Release
|
BUILD_TYPE: Release
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
runs-on: windows-latest
|
runs-on: windows-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
|
|
||||||
- name: Setup Qt
|
- name: Setup Qt
|
||||||
uses: jurplel/install-qt-action@v4
|
uses: jurplel/install-qt-action@v3
|
||||||
with:
|
with:
|
||||||
version: 6.7.2
|
|
||||||
host: windows
|
|
||||||
target: desktop
|
|
||||||
arch: win64_msvc2019_64
|
arch: win64_msvc2019_64
|
||||||
archives: qtbase qttools
|
version: 6.6.1
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure CMake
|
||||||
|
# Configure CMake in a 'build' subdirectory. `CMAKE_BUILD_TYPE` is only required if you are using a single-configuration generator such as make.
|
||||||
|
# See https://cmake.org/cmake/help/latest/variable/CMAKE_BUILD_TYPE.html?highlight=cmake_build_type
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -T ClangCL -DENABLE_QT_GUI=ON
|
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -T ClangCL -DENABLE_QT_GUI=ON
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
|
# Build your program with the given configuration
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
||||||
|
|
||||||
- name: Deploy
|
- name: Deploy
|
||||||
run: |
|
run: |
|
||||||
mkdir upload
|
mkdir upload
|
||||||
move build/Release/shadPS4.exe upload
|
move build/Release/shadps4.exe upload
|
||||||
move build/translations upload
|
move build/Release/zlib-ng2.dll upload
|
||||||
windeployqt --dir upload upload/shadPS4.exe
|
move build/Release/libwinpthread-1.dll upload
|
||||||
|
windeployqt --dir upload upload/shadps4.exe
|
||||||
|
|
||||||
- name: Upload executable
|
- name: Upload executable
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v2
|
||||||
with:
|
with:
|
||||||
name: shadps4-win64-qt
|
name: shadps4-win64-qt
|
||||||
path: upload
|
path: upload
|
||||||
|
|
|
@ -10,25 +10,32 @@ on:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
# Customize the CMake build type here (Release, Debug, RelWithDebInfo, etc.)
|
||||||
BUILD_TYPE: Release
|
BUILD_TYPE: Release
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
runs-on: windows-latest
|
runs-on: windows-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure CMake
|
||||||
|
# Configure CMake in a 'build' subdirectory. `CMAKE_BUILD_TYPE` is only required if you are using a single-configuration generator such as make.
|
||||||
|
# See https://cmake.org/cmake/help/latest/variable/CMAKE_BUILD_TYPE.html?highlight=cmake_build_type
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -T ClangCL
|
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -T ClangCL
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
|
# Build your program with the given configuration
|
||||||
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
run: cmake --build ${{github.workspace}}/build --config ${{env.BUILD_TYPE}} --parallel
|
||||||
|
- name: Upload a Build Artifact
|
||||||
- name: Upload executable
|
uses: actions/upload-artifact@v3.1.2
|
||||||
uses: actions/upload-artifact@v4
|
|
||||||
with:
|
with:
|
||||||
name: shadps4-win64-sdl
|
name: shadps4-win64
|
||||||
|
# A file, directory or wildcard pattern that describes what to upload
|
||||||
path: |
|
path: |
|
||||||
${{github.workspace}}/build/Release/shadPS4.exe
|
${{github.workspace}}/build/Release/shadps4.exe
|
||||||
|
${{github.workspace}}/build/Release/SDL3.dll
|
||||||
|
${{github.workspace}}/build/Release/libwinpthread-1.dll
|
||||||
|
|
|
@ -408,4 +408,3 @@ FodyWeavers.xsd
|
||||||
/emulator/eboot.bin
|
/emulator/eboot.bin
|
||||||
/out/*
|
/out/*
|
||||||
/third-party/out/*
|
/third-party/out/*
|
||||||
/src/common/scm_rev.cpp
|
|
||||||
|
|
|
@ -1,87 +1,60 @@
|
||||||
|
[submodule "third-party/SDL"]
|
||||||
|
path = third-party/SDL
|
||||||
|
url = https://github.com/libsdl-org/SDL
|
||||||
|
shallow = true
|
||||||
|
[submodule "third-party/fmt"]
|
||||||
|
path = third-party/fmt
|
||||||
|
url = https://github.com/fmtlib/fmt.git
|
||||||
|
shallow = true
|
||||||
|
[submodule "third-party/magic_enum"]
|
||||||
|
path = third-party/magic_enum
|
||||||
|
url = https://github.com/Neargye/magic_enum.git
|
||||||
|
shallow = true
|
||||||
|
[submodule "third-party/zydis"]
|
||||||
|
path = third-party/zydis
|
||||||
|
url = https://github.com/zyantific/zydis.git
|
||||||
|
shallow = true
|
||||||
|
[submodule "third-party/winpthread"]
|
||||||
|
path = third-party/winpthread
|
||||||
|
url = https://github.com/shadps4/winpthread.git
|
||||||
|
branch = main
|
||||||
|
[submodule "third-party/toml11"]
|
||||||
|
path = third-party/toml11
|
||||||
|
url = https://github.com/ToruNiina/toml11
|
||||||
|
branch = master
|
||||||
|
[submodule "third-party/xxHash"]
|
||||||
|
path = third-party/xxHash
|
||||||
|
url = https://github.com/Cyan4973/xxHash.git
|
||||||
|
branch = dev
|
||||||
|
[submodule "externals/ps4libdoc"]
|
||||||
|
path = externals/ps4libdoc
|
||||||
|
url = https://github.com/idc/ps4libdoc.git
|
||||||
|
branch = 9.00
|
||||||
|
[submodule "third-party/vulkan"]
|
||||||
|
path = third-party/vulkan
|
||||||
|
url = https://github.com/GPUCode/vulkan
|
||||||
|
[submodule "externals/discord-rpc"]
|
||||||
|
path = externals/discord-rpc
|
||||||
|
url = https://github.com/shadps4-emu/ext-discord-rpc.git
|
||||||
|
branch = master
|
||||||
[submodule "externals/cryptopp-cmake"]
|
[submodule "externals/cryptopp-cmake"]
|
||||||
path = externals/cryptopp-cmake
|
path = externals/cryptopp-cmake
|
||||||
url = https://github.com/shadps4-emu/ext-cryptopp-cmake.git
|
url = https://github.com/shadps4-emu/ext-cryptopp-cmake.git
|
||||||
shallow = true
|
branch = master
|
||||||
[submodule "externals/cryptopp"]
|
[submodule "externals/cryptopp"]
|
||||||
path = externals/cryptopp
|
path = externals/cryptopp
|
||||||
url = https://github.com/shadps4-emu/ext-cryptopp.git
|
url = https://github.com/shadps4-emu/ext-cryptopp.git
|
||||||
shallow = true
|
branch = master
|
||||||
[submodule "externals/cryptoppwin"]
|
[submodule "externals/cryptoppwin"]
|
||||||
path = externals/cryptoppwin
|
path = externals/cryptoppwin
|
||||||
url = https://github.com/shadps4-emu/ext-cryptoppwin.git
|
url = https://github.com/shadps4-emu/ext-cryptoppwin.git
|
||||||
shallow = true
|
branch = main
|
||||||
[submodule "externals/zlib-ng"]
|
[submodule "externals/zlib-ng"]
|
||||||
path = externals/zlib-ng
|
path = externals/zlib-ng
|
||||||
url = https://github.com/shadps4-emu/ext-zlib-ng.git
|
url = https://github.com/shadps4-emu/ext-zlib-ng.git
|
||||||
shallow = true
|
[submodule "externals/zlib-ng-win"]
|
||||||
[submodule "externals/sdl3"]
|
path = externals/zlib-ng-win
|
||||||
path = externals/sdl3
|
url = https://github.com/shadps4-emu/ext-zlib-ng-win.git
|
||||||
url = https://github.com/shadps4-emu/ext-SDL.git
|
[submodule "third-party/spdlog"]
|
||||||
shallow = true
|
path = third-party/spdlog
|
||||||
[submodule "externals/fmt"]
|
url = https://github.com/gabime/spdlog
|
||||||
path = externals/fmt
|
|
||||||
url = https://github.com/shadps4-emu/ext-fmt.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/vulkan-headers"]
|
|
||||||
path = externals/vulkan-headers
|
|
||||||
url = https://github.com/KhronosGroup/Vulkan-Headers.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/vma"]
|
|
||||||
path = externals/vma
|
|
||||||
url = https://github.com/GPUOpen-LibrariesAndSDKs/VulkanMemoryAllocator.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/glslang"]
|
|
||||||
path = externals/glslang
|
|
||||||
url = https://github.com/KhronosGroup/glslang.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/robin-map"]
|
|
||||||
path = externals/robin-map
|
|
||||||
url = https://github.com/Tessil/robin-map.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/xbyak"]
|
|
||||||
path = externals/xbyak
|
|
||||||
url = https://github.com/herumi/xbyak.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/winpthreads"]
|
|
||||||
path = externals/winpthreads
|
|
||||||
url = https://github.com/shadps4-emu/winpthreads.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/magic_enum"]
|
|
||||||
path = externals/magic_enum
|
|
||||||
url = https://github.com/Neargye/magic_enum.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/toml11"]
|
|
||||||
path = externals/toml11
|
|
||||||
url = https://github.com/ToruNiina/toml11.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/zydis"]
|
|
||||||
path = externals/zydis
|
|
||||||
url = https://github.com/zyantific/zydis.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/sirit"]
|
|
||||||
path = externals/sirit
|
|
||||||
url = https://github.com/shadps4-emu/sirit.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/xxhash"]
|
|
||||||
path = externals/xxhash
|
|
||||||
url = https://github.com/Cyan4973/xxHash.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/tracy"]
|
|
||||||
path = externals/tracy
|
|
||||||
url = https://github.com/shadps4-emu/tracy.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/ext-boost"]
|
|
||||||
path = externals/ext-boost
|
|
||||||
url = https://github.com/shadps4-emu/ext-boost.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/date"]
|
|
||||||
path = externals/date
|
|
||||||
url = https://github.com/HowardHinnant/date.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/ffmpeg-core"]
|
|
||||||
path = externals/ffmpeg-core
|
|
||||||
url = https://github.com/shadps4-emu/ext-ffmpeg-core.git
|
|
||||||
shallow = true
|
|
||||||
[submodule "externals/half"]
|
|
||||||
path = externals/half
|
|
||||||
url = https://github.com/ROCm/half.git
|
|
||||||
|
|
41
.reuse/dep5
|
@ -3,28 +3,17 @@ Comment: It is best to use this file to record copyright information about
|
||||||
generated, binary and third party files
|
generated, binary and third party files
|
||||||
|
|
||||||
Files: CMakeSettings.json
|
Files: CMakeSettings.json
|
||||||
|
scripts/ps4_names.txt
|
||||||
|
documents/changelog.txt
|
||||||
|
documents/readme.txt
|
||||||
.github/shadps4.desktop
|
.github/shadps4.desktop
|
||||||
.github/shadps4.png
|
.github/shadps4.png
|
||||||
.gitmodules
|
.gitmodules
|
||||||
documents/changelog.txt
|
src/images/shadps4.ico
|
||||||
documents/readme.txt
|
screenshots/screenshot.png
|
||||||
documents/Quickstart/1.png
|
|
||||||
documents/Quickstart/2.png
|
|
||||||
documents/Screenshots/Bloodborne.png
|
|
||||||
documents/Screenshots/Sonic Mania.png
|
|
||||||
documents/Screenshots/Undertale.png
|
|
||||||
documents/Screenshots/We are DOOMED.png
|
|
||||||
scripts/ps4_names.txt
|
|
||||||
src/images/about_icon.png
|
|
||||||
src/images/controller_icon.png
|
src/images/controller_icon.png
|
||||||
src/images/exit_icon.png
|
src/images/exit_icon.png
|
||||||
src/images/file_icon.png
|
src/images/file_icon.png
|
||||||
src/images/flag_china.png
|
|
||||||
src/images/flag_eu.png
|
|
||||||
src/images/flag_jp.png
|
|
||||||
src/images/flag_unk.png
|
|
||||||
src/images/flag_us.png
|
|
||||||
src/images/flag_world.png
|
|
||||||
src/images/folder_icon.png
|
src/images/folder_icon.png
|
||||||
src/images/grid_icon.png
|
src/images/grid_icon.png
|
||||||
src/images/iconsize_icon.png
|
src/images/iconsize_icon.png
|
||||||
|
@ -35,26 +24,8 @@ Files: CMakeSettings.json
|
||||||
src/images/refresh_icon.png
|
src/images/refresh_icon.png
|
||||||
src/images/settings_icon.png
|
src/images/settings_icon.png
|
||||||
src/images/stop_icon.png
|
src/images/stop_icon.png
|
||||||
src/images/shadPS4.icns
|
|
||||||
src/images/shadps4.ico
|
|
||||||
src/images/themes_icon.png
|
src/images/themes_icon.png
|
||||||
src/shadps4.qrc
|
|
||||||
src/shadps4.rc
|
src/shadps4.rc
|
||||||
|
src/shadps4.qrc
|
||||||
Copyright: shadPS4 Emulator Project
|
Copyright: shadPS4 Emulator Project
|
||||||
License: GPL-2.0-or-later
|
License: GPL-2.0-or-later
|
||||||
|
|
||||||
Files: externals/cmake-modules/*
|
|
||||||
Copyright: 2009-2010 Iowa State University
|
|
||||||
License: BSL-1.0
|
|
||||||
|
|
||||||
Files: externals/renderdoc/*
|
|
||||||
Copyright: 2019-2024 Baldur Karlsson
|
|
||||||
License: MIT
|
|
||||||
|
|
||||||
Files: externals/stb_image.h
|
|
||||||
Copyright: 2017 Sean Barrett
|
|
||||||
License: MIT
|
|
||||||
|
|
||||||
Files: externals/tracy/*
|
|
||||||
Copyright: 2017-2024 Bartosz Taudul <wolf@nereid.pl>
|
|
||||||
License: BSD-3-Clause
|
|
||||||
|
|
838
CMakeLists.txt
|
@ -6,24 +6,26 @@ cmake_minimum_required(VERSION 3.16.3)
|
||||||
set(CMAKE_CXX_STANDARD 23)
|
set(CMAKE_CXX_STANDARD 23)
|
||||||
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
set(CMAKE_CXX_STANDARD_REQUIRED True)
|
||||||
|
|
||||||
if(APPLE)
|
|
||||||
enable_language(OBJC)
|
|
||||||
set(CMAKE_OSX_DEPLOYMENT_TARGET 11)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (NOT CMAKE_BUILD_TYPE)
|
if (NOT CMAKE_BUILD_TYPE)
|
||||||
set(CMAKE_BUILD_TYPE Release)
|
set(CMAKE_BUILD_TYPE Release)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
project(shadPS4)
|
project(shadps4)
|
||||||
|
|
||||||
option(ENABLE_QT_GUI "Enable the Qt GUI. If not selected then the emulator uses a minimal SDL-based UI instead" OFF)
|
option(ENABLE_QT_GUI "Enable the Qt GUI. If not selected then the emulator uses a minimal SDL-based UI instead" OFF)
|
||||||
|
|
||||||
# This function should be passed a list of all files in a target. It will automatically generate file groups
|
if(ENABLE_QT_GUI)
|
||||||
# following the directory hierarchy, so that the layout of the files in IDEs matches the one in the filesystem.
|
find_package(Qt6 REQUIRED COMPONENTS Widgets Concurrent)
|
||||||
function(create_target_directory_groups target_name)
|
qt_standard_project_setup()
|
||||||
|
set(CMAKE_AUTORCC ON)
|
||||||
|
endif()
|
||||||
|
|
||||||
# Place any files that aren't in the source list in a separate group so that they don't get in the way.
|
# This function should be passed a list of all files in a target. It will automatically generate
|
||||||
|
# file groups following the directory hierarchy, so that the layout of the files in IDEs matches the
|
||||||
|
# one in the filesystem.
|
||||||
|
function(create_target_directory_groups target_name)
|
||||||
|
# Place any files that aren't in the source list in a separate group so that they don't get in
|
||||||
|
# the way.
|
||||||
source_group("Other Files" REGULAR_EXPRESSION ".")
|
source_group("Other Files" REGULAR_EXPRESSION ".")
|
||||||
|
|
||||||
get_target_property(target_sources "${target_name}" SOURCES)
|
get_target_property(target_sources "${target_name}" SOURCES)
|
||||||
|
@ -36,15 +38,48 @@ function(create_target_directory_groups target_name)
|
||||||
endforeach()
|
endforeach()
|
||||||
endfunction()
|
endfunction()
|
||||||
|
|
||||||
|
|
||||||
# Setup a custom clang-format target (if clang-format can be found) that will run
|
# Setup a custom clang-format target (if clang-format can be found) that will run
|
||||||
# against all the src files. This should be used before making a pull request.
|
# against all the src files. This should be used before making a pull request.
|
||||||
|
# =======================================================================
|
||||||
|
|
||||||
|
set(CLANG_FORMAT_POSTFIX "-15")
|
||||||
|
find_program(CLANG_FORMAT
|
||||||
|
NAMES clang-format${CLANG_FORMAT_POSTFIX}
|
||||||
|
clang-format
|
||||||
|
PATHS ${PROJECT_BINARY_DIR}/externals)
|
||||||
|
# if find_program doesn't find it, try to download from externals
|
||||||
|
if (NOT CLANG_FORMAT)
|
||||||
|
if (WIN32)
|
||||||
|
message(STATUS "Clang format not found! Downloading...")
|
||||||
|
set(CLANG_FORMAT "${PROJECT_BINARY_DIR}/externals/clang-format${CLANG_FORMAT_POSTFIX}.exe")
|
||||||
|
file(DOWNLOAD
|
||||||
|
https://github.com/citra-emu/ext-windows-bin/raw/master/clang-format${CLANG_FORMAT_POSTFIX}.exe
|
||||||
|
"${CLANG_FORMAT}" SHOW_PROGRESS
|
||||||
|
STATUS DOWNLOAD_SUCCESS)
|
||||||
|
if (NOT DOWNLOAD_SUCCESS EQUAL 0)
|
||||||
|
message(WARNING "Could not download clang format! Disabling the clang format target")
|
||||||
|
file(REMOVE ${CLANG_FORMAT})
|
||||||
|
unset(CLANG_FORMAT)
|
||||||
|
endif()
|
||||||
|
else()
|
||||||
|
message(WARNING "Clang format not found! Disabling the clang format target")
|
||||||
|
endif()
|
||||||
|
endif()
|
||||||
|
|
||||||
if (CLANG_FORMAT)
|
if (CLANG_FORMAT)
|
||||||
set(SRCS ${PROJECT_SOURCE_DIR}/src)
|
set(SRCS ${PROJECT_SOURCE_DIR}/src)
|
||||||
set(CCOMMENT "Running clang format against all the .h and .cpp files in src/")
|
set(CCOMMENT "Running clang format against all the .h and .cpp files in src/")
|
||||||
if (WIN32)
|
if (WIN32)
|
||||||
|
if(MINGW)
|
||||||
|
add_custom_target(clang-format
|
||||||
|
COMMAND find `cygpath -u ${SRCS}` -iname *.h -o -iname *.cpp -o -iname *.mm | xargs `cygpath -u ${CLANG_FORMAT}` -i
|
||||||
|
COMMENT ${CCOMMENT})
|
||||||
|
else()
|
||||||
add_custom_target(clang-format
|
add_custom_target(clang-format
|
||||||
COMMAND powershell.exe -Command "Get-ChildItem '${SRCS}/*' -Include *.cpp,*.h,*.mm -Recurse | Foreach {&'${CLANG_FORMAT}' -i $_.fullname}"
|
COMMAND powershell.exe -Command "Get-ChildItem '${SRCS}/*' -Include *.cpp,*.h,*.mm -Recurse | Foreach {&'${CLANG_FORMAT}' -i $_.fullname}"
|
||||||
COMMENT ${CCOMMENT})
|
COMMENT ${CCOMMENT})
|
||||||
|
endif()
|
||||||
else()
|
else()
|
||||||
add_custom_target(clang-format
|
add_custom_target(clang-format
|
||||||
COMMAND find ${SRCS} -iname *.h -o -iname *.cpp -o -iname *.mm | xargs ${CLANG_FORMAT} -i
|
COMMAND find ${SRCS} -iname *.h -o -iname *.cpp -o -iname *.mm | xargs ${CLANG_FORMAT} -i
|
||||||
|
@ -54,674 +89,269 @@ if (CLANG_FORMAT)
|
||||||
unset(CCOMMENT)
|
unset(CCOMMENT)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake")
|
|
||||||
|
|
||||||
# generate git revision information
|
|
||||||
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/externals/cmake-modules/")
|
|
||||||
include(GetGitRevisionDescription)
|
|
||||||
get_git_head_revision(GIT_REF_SPEC GIT_REV)
|
|
||||||
git_describe(GIT_DESC --always --long --dirty)
|
|
||||||
git_branch_name(GIT_BRANCH)
|
|
||||||
|
|
||||||
configure_file("${CMAKE_CURRENT_SOURCE_DIR}/src/common/scm_rev.cpp.in" "${CMAKE_CURRENT_SOURCE_DIR}/src/common/scm_rev.cpp" @ONLY)
|
|
||||||
|
|
||||||
find_package(Boost 1.84.0 CONFIG)
|
|
||||||
find_package(FFmpeg 5.1.2 MODULE)
|
|
||||||
find_package(fmt 10.2.0 CONFIG)
|
|
||||||
find_package(glslang 14.2.0 CONFIG)
|
|
||||||
find_package(magic_enum 0.9.6 CONFIG)
|
|
||||||
find_package(RenderDoc 1.6.0 MODULE)
|
|
||||||
find_package(SDL3 3.1.2 CONFIG)
|
|
||||||
find_package(toml11 4.2.0 CONFIG)
|
|
||||||
find_package(tsl-robin-map 1.3.0 CONFIG)
|
|
||||||
find_package(VulkanHeaders 1.3.289 CONFIG)
|
|
||||||
find_package(VulkanMemoryAllocator 3.1.0 CONFIG)
|
|
||||||
find_package(xbyak 7.07 CONFIG)
|
|
||||||
find_package(xxHash 0.8.2 MODULE)
|
|
||||||
find_package(zlib-ng 2.1.7 MODULE)
|
|
||||||
find_package(Zydis 5.0.0 CONFIG)
|
|
||||||
|
|
||||||
if (NOT CMAKE_CXX_COMPILER_ID STREQUAL "Clang" OR NOT MSVC)
|
|
||||||
find_package(cryptopp 8.9.0 MODULE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (APPLE)
|
|
||||||
find_package(date 3.0.1 CONFIG)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Note: Windows always has these functions through winpthreads
|
|
||||||
include(CheckSymbolExists)
|
|
||||||
check_symbol_exists(pthread_mutex_timedlock "pthread.h" HAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
|
||||||
if(HAVE_PTHREAD_MUTEX_TIMEDLOCK OR WIN32)
|
|
||||||
add_compile_options(-DHAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" OR CMAKE_CXX_COMPILER_ID STREQUAL "AppleClang")
|
|
||||||
# libc++ requires -fexperimental-library to enable std::jthread and std::stop_token support.
|
|
||||||
include(CheckCXXSymbolExists)
|
|
||||||
check_cxx_symbol_exists(_LIBCPP_VERSION version LIBCPP)
|
|
||||||
if(LIBCPP)
|
|
||||||
add_compile_options(-fexperimental-library)
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
add_subdirectory(externals)
|
add_subdirectory(externals)
|
||||||
|
add_subdirectory(third-party)
|
||||||
include_directories(src)
|
include_directories(src)
|
||||||
|
|
||||||
|
set(LIBRARIES src/core/libraries/library_common.h
|
||||||
|
src/core/libraries/error_codes.h
|
||||||
|
src/core/libraries/libscecommondialog.cpp
|
||||||
|
src/core/libraries/libscecommondialog.h
|
||||||
|
src/core/libraries/libscegnmdriver.cpp
|
||||||
|
src/core/libraries/libscegnmdriver.h
|
||||||
|
src/core/libraries/libscemsgdialog.cpp
|
||||||
|
src/core/libraries/libscemsgdialog.h
|
||||||
|
src/core/libraries/libscesystemservice.cpp
|
||||||
|
src/core/libraries/libscesystemservice.h
|
||||||
|
src/core/libraries/libsceuserservice.cpp
|
||||||
|
src/core/libraries/libsceuserservice.h
|
||||||
|
)
|
||||||
|
set(LIBC_SOURCES src/core/hle/libraries/libc/libc.cpp
|
||||||
|
src/core/hle/libraries/libc/libc.h
|
||||||
|
src/core/hle/libraries/libc/printf.h
|
||||||
|
src/core/hle/libraries/libc/va_ctx.h
|
||||||
|
src/core/hle/libraries/libc/libc_cxa.cpp
|
||||||
|
src/core/hle/libraries/libc/libc_cxa.h
|
||||||
|
src/core/hle/libraries/libc/libc_stdio.cpp
|
||||||
|
src/core/hle/libraries/libc/libc_stdio.h
|
||||||
|
src/core/hle/libraries/libc/libc_math.cpp
|
||||||
|
src/core/hle/libraries/libc/libc_math.h
|
||||||
|
src/core/hle/libraries/libc/libc_string.cpp
|
||||||
|
src/core/hle/libraries/libc/libc_string.h
|
||||||
|
src/core/hle/libraries/libc/libc_stdlib.cpp
|
||||||
|
src/core/hle/libraries/libc/libc_stdlib.h
|
||||||
|
)
|
||||||
|
|
||||||
|
set(PAD_SOURCES src/core/hle/libraries/libpad/pad.cpp
|
||||||
|
src/core/hle/libraries/libpad/pad.h
|
||||||
|
)
|
||||||
|
|
||||||
|
set(FILESYSTEM_SOURCES src/core/hle/libraries/libkernel/file_system.cpp
|
||||||
|
src/core/hle/libraries/libkernel/file_system.h
|
||||||
|
src/core/file_sys/fs.cpp
|
||||||
|
src/core/file_sys/fs.h
|
||||||
|
)
|
||||||
|
|
||||||
|
set(HOST_SOURCES src/Emulator/Host/controller.cpp
|
||||||
|
src/Emulator/Host/controller.h
|
||||||
|
)
|
||||||
|
|
||||||
|
# the above is shared in sdl and qt version (TODO share them all)
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
find_package(Qt6 REQUIRED COMPONENTS Widgets Concurrent LinguistTools Network)
|
qt_add_resources(RESOURCE_FILES src/shadps4.qrc)
|
||||||
qt_standard_project_setup()
|
set(QT_GUI
|
||||||
set(CMAKE_AUTORCC ON)
|
src/qt_gui/main_window_ui.h
|
||||||
set(CMAKE_AUTOMOC ON)
|
src/qt_gui/main_window.cpp
|
||||||
set(CMAKE_AUTOUIC ON)
|
src/qt_gui/main_window.h
|
||||||
|
src/qt_gui/gui_settings.cpp
|
||||||
set(QT_TRANSLATIONS "${PROJECT_SOURCE_DIR}/src/qt_gui/translations")
|
src/qt_gui/gui_settings.h
|
||||||
file(GLOB_RECURSE TRANSLATIONS_TS ${QT_TRANSLATIONS}/*.ts)
|
src/qt_gui/settings.cpp
|
||||||
|
src/qt_gui/settings.h
|
||||||
set_source_files_properties(${TRANSLATIONS_TS} PROPERTIES OUTPUT_LOCATION "${CMAKE_CURRENT_BINARY_DIR}/translations")
|
src/qt_gui/gui_save.h
|
||||||
qt_add_translation(TRANSLATIONS_QM ${TRANSLATIONS_TS})
|
src/qt_gui/custom_dock_widget.h
|
||||||
|
src/qt_gui/custom_table_widget_item.cpp
|
||||||
set(TRANSLATIONS_QRC ${CMAKE_CURRENT_BINARY_DIR}/translations/translations.qrc)
|
src/qt_gui/custom_table_widget_item.h
|
||||||
file(WRITE ${TRANSLATIONS_QRC} "<RCC><qresource prefix=\"translations\">\n")
|
src/qt_gui/game_list_item.h
|
||||||
foreach (QM ${TRANSLATIONS_QM})
|
src/qt_gui/game_list_table.cpp
|
||||||
get_filename_component(QM_FILE ${QM} NAME)
|
src/qt_gui/game_list_table.h
|
||||||
file(APPEND ${TRANSLATIONS_QRC} "<file>${QM_FILE}</file>\n")
|
src/qt_gui/game_list_utils.h
|
||||||
endforeach (QM)
|
src/qt_gui/game_info.h
|
||||||
file(APPEND ${TRANSLATIONS_QRC} "</qresource></RCC>")
|
src/qt_gui/game_list_grid.cpp
|
||||||
|
src/qt_gui/game_list_grid.h
|
||||||
qt_add_resources(TRANSLATIONS ${TRANSLATIONS_QRC})
|
src/qt_gui/game_list_grid_delegate.cpp
|
||||||
|
src/qt_gui/game_list_grid_delegate.h
|
||||||
|
src/qt_gui/game_list_frame.cpp
|
||||||
|
src/qt_gui/game_list_frame.h
|
||||||
|
src/qt_gui/qt_utils.h
|
||||||
|
src/qt_gui/game_install_dialog.cpp
|
||||||
|
src/qt_gui/game_install_dialog.h
|
||||||
|
src/qt_gui/main_window_themes.cpp
|
||||||
|
src/qt_gui/main_window_themes.h
|
||||||
|
src/qt_gui/main.cpp
|
||||||
|
${RESOURCE_FILES}
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(AUDIO_CORE src/audio_core/sdl_audio.cpp
|
set(COMMON src/common/debug.h
|
||||||
src/audio_core/sdl_audio.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(AUDIO_LIB src/core/libraries/audio/audioin.cpp
|
|
||||||
src/core/libraries/audio/audioin.h
|
|
||||||
src/core/libraries/audio/audioout.cpp
|
|
||||||
src/core/libraries/audio/audioout.h
|
|
||||||
src/core/libraries/ajm/ajm.cpp
|
|
||||||
src/core/libraries/ajm/ajm.h
|
|
||||||
src/core/libraries/ngs2/ngs2.cpp
|
|
||||||
src/core/libraries/ngs2/ngs2.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(GNM_LIB src/core/libraries/gnmdriver/gnmdriver.cpp
|
|
||||||
src/core/libraries/gnmdriver/gnmdriver.h
|
|
||||||
src/core/libraries/gnmdriver/gnm_error.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(KERNEL_LIB
|
|
||||||
src/core/libraries/kernel/event_flag/event_flag.cpp
|
|
||||||
src/core/libraries/kernel/event_flag/event_flag.h
|
|
||||||
src/core/libraries/kernel/event_flag/event_flag_obj.cpp
|
|
||||||
src/core/libraries/kernel/event_flag/event_flag_obj.h
|
|
||||||
src/core/libraries/kernel/threads/rwlock.cpp
|
|
||||||
src/core/libraries/kernel/threads/semaphore.cpp
|
|
||||||
src/core/libraries/kernel/threads/keys.cpp
|
|
||||||
src/core/libraries/kernel/threads/threads.h
|
|
||||||
src/core/libraries/kernel/cpu_management.cpp
|
|
||||||
src/core/libraries/kernel/cpu_management.h
|
|
||||||
src/core/libraries/kernel/event_queue.cpp
|
|
||||||
src/core/libraries/kernel/event_queue.h
|
|
||||||
src/core/libraries/kernel/event_queues.cpp
|
|
||||||
src/core/libraries/kernel/event_queues.h
|
|
||||||
src/core/libraries/kernel/file_system.cpp
|
|
||||||
src/core/libraries/kernel/file_system.h
|
|
||||||
src/core/libraries/kernel/libkernel.cpp
|
|
||||||
src/core/libraries/kernel/libkernel.h
|
|
||||||
src/core/libraries/kernel/memory_management.cpp
|
|
||||||
src/core/libraries/kernel/memory_management.h
|
|
||||||
src/core/libraries/kernel/thread_management.cpp
|
|
||||||
src/core/libraries/kernel/thread_management.h
|
|
||||||
src/core/libraries/kernel/time_management.cpp
|
|
||||||
src/core/libraries/kernel/time_management.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(NETWORK_LIBS src/core/libraries/network/http.cpp
|
|
||||||
src/core/libraries/network/http.h
|
|
||||||
src/core/libraries/network/net.cpp
|
|
||||||
src/core/libraries/network/netctl.cpp
|
|
||||||
src/core/libraries/network/netctl.h
|
|
||||||
src/core/libraries/network/net.h
|
|
||||||
src/core/libraries/network/ssl.cpp
|
|
||||||
src/core/libraries/network/ssl.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(SYSTEM_LIBS src/core/libraries/system/commondialog.cpp
|
|
||||||
src/core/libraries/system/commondialog.h
|
|
||||||
src/core/libraries/system/msgdialog.cpp
|
|
||||||
src/core/libraries/system/msgdialog.h
|
|
||||||
src/core/libraries/system/posix.cpp
|
|
||||||
src/core/libraries/system/posix.h
|
|
||||||
src/core/libraries/save_data/error_codes.h
|
|
||||||
src/core/libraries/save_data/savedata.cpp
|
|
||||||
src/core/libraries/save_data/savedata.h
|
|
||||||
src/core/libraries/system/savedatadialog.cpp
|
|
||||||
src/core/libraries/system/savedatadialog.h
|
|
||||||
src/core/libraries/system/sysmodule.cpp
|
|
||||||
src/core/libraries/system/sysmodule.h
|
|
||||||
src/core/libraries/system/systemservice.cpp
|
|
||||||
src/core/libraries/system/systemservice.h
|
|
||||||
src/core/libraries/system/userservice.cpp
|
|
||||||
src/core/libraries/system/userservice.h
|
|
||||||
src/core/libraries/app_content/app_content.cpp
|
|
||||||
src/core/libraries/app_content/app_content.h
|
|
||||||
src/core/libraries/rtc/rtc.cpp
|
|
||||||
src/core/libraries/rtc/rtc.h
|
|
||||||
src/core/libraries/rtc/rtc_error.h
|
|
||||||
src/core/libraries/disc_map/disc_map.cpp
|
|
||||||
src/core/libraries/disc_map/disc_map.h
|
|
||||||
src/core/libraries/disc_map/disc_map_codes.h
|
|
||||||
src/core/libraries/avplayer/avplayer_common.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer_common.h
|
|
||||||
src/core/libraries/avplayer/avplayer_file_streamer.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer_file_streamer.h
|
|
||||||
src/core/libraries/avplayer/avplayer_impl.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer_impl.h
|
|
||||||
src/core/libraries/avplayer/avplayer_source.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer_source.h
|
|
||||||
src/core/libraries/avplayer/avplayer_state.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer_state.h
|
|
||||||
src/core/libraries/avplayer/avplayer.cpp
|
|
||||||
src/core/libraries/avplayer/avplayer.h
|
|
||||||
src/core/libraries/ngs2/ngs2.cpp
|
|
||||||
src/core/libraries/ngs2/ngs2.h
|
|
||||||
src/core/libraries/ngs2/ngs2_error.h
|
|
||||||
src/core/libraries/ngs2/ngs2_impl.cpp
|
|
||||||
src/core/libraries/ngs2/ngs2_impl.h
|
|
||||||
src/core/libraries/ajm/ajm_error.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(VIDEOOUT_LIB src/core/libraries/videoout/buffer.h
|
|
||||||
src/core/libraries/videoout/driver.cpp
|
|
||||||
src/core/libraries/videoout/driver.h
|
|
||||||
src/core/libraries/videoout/video_out.cpp
|
|
||||||
src/core/libraries/videoout/video_out.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(LIBC_SOURCES src/core/libraries/libc_internal/libc_internal.cpp
|
|
||||||
src/core/libraries/libc_internal/libc_internal.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(DIALOGS_LIB src/core/libraries/dialogs/error_dialog.cpp
|
|
||||||
src/core/libraries/dialogs/error_dialog.h
|
|
||||||
src/core/libraries/dialogs/ime_dialog.cpp
|
|
||||||
src/core/libraries/dialogs/ime_dialog.h
|
|
||||||
src/core/libraries/dialogs/error_codes.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(PAD_LIB src/core/libraries/pad/pad.cpp
|
|
||||||
src/core/libraries/pad/pad.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(PNG_LIB src/core/libraries/libpng/pngdec.cpp
|
|
||||||
src/core/libraries/libpng/pngdec.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(PLAYGO_LIB src/core/libraries/playgo/playgo.cpp
|
|
||||||
src/core/libraries/playgo/playgo.h
|
|
||||||
src/core/libraries/playgo/playgo_types.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(RANDOM_LIB src/core/libraries/random/random.cpp
|
|
||||||
src/core/libraries/random/random.h
|
|
||||||
src/core/libraries/random/random_error.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(USBD_LIB src/core/libraries/usbd/usbd.cpp
|
|
||||||
src/core/libraries/usbd/usbd.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(NP_LIBS src/core/libraries/np_manager/np_manager.cpp
|
|
||||||
src/core/libraries/np_manager/np_manager.h
|
|
||||||
src/core/libraries/np_score/np_score.cpp
|
|
||||||
src/core/libraries/np_score/np_score.h
|
|
||||||
src/core/libraries/np_trophy/np_trophy.cpp
|
|
||||||
src/core/libraries/np_trophy/np_trophy.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(MISC_LIBS src/core/libraries/screenshot/screenshot.cpp
|
|
||||||
src/core/libraries/screenshot/screenshot.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(COMMON src/common/logging/backend.cpp
|
|
||||||
src/common/logging/backend.h
|
|
||||||
src/common/logging/filter.cpp
|
|
||||||
src/common/logging/filter.h
|
|
||||||
src/common/logging/formatter.h
|
|
||||||
src/common/logging/log_entry.h
|
|
||||||
src/common/logging/log.h
|
|
||||||
src/common/logging/text_formatter.cpp
|
|
||||||
src/common/logging/text_formatter.h
|
|
||||||
src/common/logging/types.h
|
|
||||||
src/common/alignment.h
|
|
||||||
src/common/assert.cpp
|
|
||||||
src/common/assert.h
|
|
||||||
src/common/bit_field.h
|
|
||||||
src/common/bounded_threadsafe_queue.h
|
|
||||||
src/common/concepts.h
|
|
||||||
src/common/config.cpp
|
|
||||||
src/common/config.h
|
|
||||||
src/common/debug.h
|
|
||||||
src/common/disassembler.cpp
|
src/common/disassembler.cpp
|
||||||
src/common/disassembler.h
|
src/common/disassembler.h
|
||||||
|
src/common/discord.cpp
|
||||||
|
src/common/discord.h
|
||||||
src/common/endian.h
|
src/common/endian.h
|
||||||
src/common/enum.h
|
|
||||||
src/common/io_file.cpp
|
src/common/io_file.cpp
|
||||||
src/common/io_file.h
|
src/common/io_file.h
|
||||||
src/common/error.cpp
|
|
||||||
src/common/error.h
|
|
||||||
src/common/scope_exit.h
|
|
||||||
src/common/func_traits.h
|
|
||||||
src/common/native_clock.cpp
|
src/common/native_clock.cpp
|
||||||
src/common/native_clock.h
|
src/common/native_clock.h
|
||||||
src/common/path_util.cpp
|
|
||||||
src/common/path_util.h
|
|
||||||
src/common/object_pool.h
|
|
||||||
src/common/polyfill_thread.h
|
|
||||||
src/common/rdtsc.cpp
|
src/common/rdtsc.cpp
|
||||||
src/common/rdtsc.h
|
src/common/rdtsc.h
|
||||||
src/common/singleton.h
|
src/common/singleton.h
|
||||||
src/common/slot_vector.h
|
|
||||||
src/common/string_util.cpp
|
src/common/string_util.cpp
|
||||||
src/common/string_util.h
|
src/common/string_util.h
|
||||||
src/common/thread.cpp
|
|
||||||
src/common/thread.h
|
|
||||||
src/common/types.h
|
src/common/types.h
|
||||||
src/common/uint128.h
|
src/common/uint128.h
|
||||||
src/common/unique_function.h
|
|
||||||
src/common/version.h
|
src/common/version.h
|
||||||
src/common/ntapi.h
|
src/common/fs_file.cpp
|
||||||
src/common/ntapi.cpp
|
src/common/fs_file.h
|
||||||
src/common/scm_rev.cpp
|
src/common/log.cpp
|
||||||
src/common/scm_rev.h
|
src/common/log.h
|
||||||
|
src/common/io_file.cpp
|
||||||
|
src/common/io_file.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(CORE src/core/aerolib/stubs.cpp
|
set(CORE src/core/loader.cpp
|
||||||
src/core/aerolib/stubs.h
|
src/core/loader.h
|
||||||
src/core/aerolib/aerolib.cpp
|
)
|
||||||
src/core/aerolib/aerolib.h
|
|
||||||
src/core/address_space.cpp
|
set(CRYPTO src/core/crypto/crypto.cpp
|
||||||
src/core/address_space.h
|
|
||||||
src/core/cpu_patches.cpp
|
|
||||||
src/core/cpu_patches.h
|
|
||||||
src/core/crypto/crypto.cpp
|
|
||||||
src/core/crypto/crypto.h
|
src/core/crypto/crypto.h
|
||||||
src/core/crypto/keys.h
|
src/core/crypto/keys.h
|
||||||
src/core/file_format/pfs.h
|
)
|
||||||
|
set(FILE_FORMAT src/core/file_format/pfs.h
|
||||||
src/core/file_format/pkg.cpp
|
src/core/file_format/pkg.cpp
|
||||||
src/core/file_format/pkg.h
|
src/core/file_format/pkg.h
|
||||||
src/core/file_format/pkg_type.cpp
|
src/core/file_format/pkg_type.cpp
|
||||||
src/core/file_format/pkg_type.h
|
src/core/file_format/pkg_type.h
|
||||||
src/core/file_format/psf.cpp
|
src/core/file_format/psf.cpp
|
||||||
src/core/file_format/psf.h
|
src/core/file_format/psf.h
|
||||||
src/core/file_format/playgo_chunk.cpp
|
|
||||||
src/core/file_format/playgo_chunk.h
|
|
||||||
src/core/file_format/trp.cpp
|
|
||||||
src/core/file_format/trp.h
|
|
||||||
src/core/file_format/splash.h
|
|
||||||
src/core/file_format/splash.cpp
|
|
||||||
src/core/file_sys/fs.cpp
|
|
||||||
src/core/file_sys/fs.h
|
|
||||||
src/core/loader.cpp
|
|
||||||
src/core/loader.h
|
|
||||||
src/core/loader/dwarf.cpp
|
|
||||||
src/core/loader/dwarf.h
|
|
||||||
src/core/loader/elf.cpp
|
|
||||||
src/core/loader/elf.h
|
|
||||||
src/core/loader/symbols_resolver.h
|
|
||||||
src/core/loader/symbols_resolver.cpp
|
|
||||||
src/core/libraries/error_codes.h
|
|
||||||
src/core/libraries/libs.h
|
|
||||||
src/core/libraries/libs.cpp
|
|
||||||
${AUDIO_LIB}
|
|
||||||
${GNM_LIB}
|
|
||||||
${KERNEL_LIB}
|
|
||||||
${NETWORK_LIBS}
|
|
||||||
${SYSTEM_LIBS}
|
|
||||||
${LIBC_SOURCES}
|
|
||||||
${PAD_LIB}
|
|
||||||
${VIDEOOUT_LIB}
|
|
||||||
${NP_LIBS}
|
|
||||||
${PNG_LIB}
|
|
||||||
${PLAYGO_LIB}
|
|
||||||
${RANDOM_LIB}
|
|
||||||
${USBD_LIB}
|
|
||||||
${MISC_LIBS}
|
|
||||||
${DIALOGS_LIB}
|
|
||||||
src/core/linker.cpp
|
|
||||||
src/core/linker.h
|
|
||||||
src/core/memory.cpp
|
|
||||||
src/core/memory.h
|
|
||||||
src/core/module.cpp
|
|
||||||
src/core/module.h
|
|
||||||
src/core/platform.h
|
|
||||||
src/core/tls.cpp
|
|
||||||
src/core/tls.h
|
|
||||||
src/core/virtual_memory.cpp
|
|
||||||
src/core/virtual_memory.h
|
|
||||||
)
|
)
|
||||||
|
|
||||||
set(SHADER_RECOMPILER src/shader_recompiler/exception.h
|
set(UTILITIES src/Util/config.cpp
|
||||||
src/shader_recompiler/profile.h
|
src/Util/config.h
|
||||||
src/shader_recompiler/recompiler.cpp
|
|
||||||
src/shader_recompiler/recompiler.h
|
|
||||||
src/shader_recompiler/runtime_info.h
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv.h
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_atomic.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_barriers.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_bitwise_conversion.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_composite.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_context_get_set.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_convert.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_floating_point.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_image.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_instructions.h
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_integer.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_logical.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_select.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_shared_memory.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_special.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_undefined.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/emit_spirv_warp.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/spirv_emit_context.cpp
|
|
||||||
src/shader_recompiler/backend/spirv/spirv_emit_context.h
|
|
||||||
src/shader_recompiler/frontend/translate/data_share.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/export.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/scalar_alu.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/scalar_memory.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/translate.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/translate.h
|
|
||||||
src/shader_recompiler/frontend/translate/vector_alu.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/vector_interpolation.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/vector_memory.cpp
|
|
||||||
src/shader_recompiler/frontend/control_flow_graph.cpp
|
|
||||||
src/shader_recompiler/frontend/control_flow_graph.h
|
|
||||||
src/shader_recompiler/frontend/decode.cpp
|
|
||||||
src/shader_recompiler/frontend/decode.h
|
|
||||||
src/shader_recompiler/frontend/fetch_shader.cpp
|
|
||||||
src/shader_recompiler/frontend/fetch_shader.h
|
|
||||||
src/shader_recompiler/frontend/format.cpp
|
|
||||||
src/shader_recompiler/frontend/instruction.cpp
|
|
||||||
src/shader_recompiler/frontend/instruction.h
|
|
||||||
src/shader_recompiler/frontend/opcodes.h
|
|
||||||
src/shader_recompiler/frontend/structured_control_flow.cpp
|
|
||||||
src/shader_recompiler/frontend/structured_control_flow.h
|
|
||||||
src/shader_recompiler/ir/passes/constant_propagation_pass.cpp
|
|
||||||
src/shader_recompiler/ir/passes/dead_code_elimination_pass.cpp
|
|
||||||
src/shader_recompiler/ir/passes/identity_removal_pass.cpp
|
|
||||||
src/shader_recompiler/ir/passes/ir_passes.h
|
|
||||||
src/shader_recompiler/ir/passes/lower_shared_mem_to_registers.cpp
|
|
||||||
src/shader_recompiler/ir/passes/resource_tracking_pass.cpp
|
|
||||||
src/shader_recompiler/ir/passes/shader_info_collection_pass.cpp
|
|
||||||
src/shader_recompiler/ir/passes/ssa_rewrite_pass.cpp
|
|
||||||
src/shader_recompiler/ir/abstract_syntax_list.h
|
|
||||||
src/shader_recompiler/ir/attribute.cpp
|
|
||||||
src/shader_recompiler/ir/attribute.h
|
|
||||||
src/shader_recompiler/ir/basic_block.cpp
|
|
||||||
src/shader_recompiler/ir/basic_block.h
|
|
||||||
src/shader_recompiler/ir/condition.h
|
|
||||||
src/shader_recompiler/ir/ir_emitter.cpp
|
|
||||||
src/shader_recompiler/ir/ir_emitter.h
|
|
||||||
src/shader_recompiler/ir/microinstruction.cpp
|
|
||||||
src/shader_recompiler/ir/opcodes.cpp
|
|
||||||
src/shader_recompiler/ir/opcodes.h
|
|
||||||
src/shader_recompiler/ir/opcodes.inc
|
|
||||||
src/shader_recompiler/ir/post_order.cpp
|
|
||||||
src/shader_recompiler/ir/post_order.h
|
|
||||||
src/shader_recompiler/ir/program.cpp
|
|
||||||
src/shader_recompiler/ir/program.h
|
|
||||||
src/shader_recompiler/ir/reg.h
|
|
||||||
src/shader_recompiler/ir/type.cpp
|
|
||||||
src/shader_recompiler/ir/type.h
|
|
||||||
src/shader_recompiler/ir/value.cpp
|
|
||||||
src/shader_recompiler/ir/value.h
|
|
||||||
)
|
)
|
||||||
|
|
||||||
set(VIDEO_CORE src/video_core/amdgpu/liverpool.cpp
|
|
||||||
src/video_core/amdgpu/liverpool.h
|
|
||||||
src/video_core/amdgpu/pixel_format.cpp
|
|
||||||
src/video_core/amdgpu/pixel_format.h
|
|
||||||
src/video_core/amdgpu/pm4_cmds.h
|
|
||||||
src/video_core/amdgpu/pm4_opcodes.h
|
|
||||||
src/video_core/amdgpu/resource.h
|
|
||||||
src/video_core/amdgpu/default_context.cpp
|
|
||||||
src/video_core/buffer_cache/buffer.cpp
|
|
||||||
src/video_core/buffer_cache/buffer.h
|
|
||||||
src/video_core/buffer_cache/buffer_cache.cpp
|
|
||||||
src/video_core/buffer_cache/buffer_cache.h
|
|
||||||
src/video_core/buffer_cache/memory_tracker_base.h
|
|
||||||
src/video_core/buffer_cache/range_set.h
|
|
||||||
src/video_core/buffer_cache/word_manager.h
|
|
||||||
src/video_core/renderer_vulkan/liverpool_to_vk.cpp
|
|
||||||
src/video_core/renderer_vulkan/liverpool_to_vk.h
|
|
||||||
src/video_core/renderer_vulkan/renderer_vulkan.cpp
|
|
||||||
src/video_core/renderer_vulkan/renderer_vulkan.h
|
|
||||||
src/video_core/renderer_vulkan/vk_common.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_common.h
|
|
||||||
src/video_core/renderer_vulkan/vk_compute_pipeline.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_compute_pipeline.h
|
|
||||||
src/video_core/renderer_vulkan/vk_descriptor_update_queue.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_descriptor_update_queue.h
|
|
||||||
src/video_core/renderer_vulkan/vk_graphics_pipeline.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_graphics_pipeline.h
|
|
||||||
src/video_core/renderer_vulkan/vk_instance.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_instance.h
|
|
||||||
src/video_core/renderer_vulkan/vk_master_semaphore.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_master_semaphore.h
|
|
||||||
src/video_core/renderer_vulkan/vk_pipeline_cache.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_pipeline_cache.h
|
|
||||||
src/video_core/renderer_vulkan/vk_platform.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_platform.h
|
|
||||||
src/video_core/renderer_vulkan/vk_rasterizer.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_rasterizer.h
|
|
||||||
src/video_core/renderer_vulkan/vk_resource_pool.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_resource_pool.h
|
|
||||||
src/video_core/renderer_vulkan/vk_scheduler.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_scheduler.h
|
|
||||||
src/video_core/renderer_vulkan/vk_shader_cache.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_shader_cache.h
|
|
||||||
src/video_core/renderer_vulkan/vk_shader_util.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_shader_util.h
|
|
||||||
src/video_core/renderer_vulkan/vk_swapchain.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_swapchain.h
|
|
||||||
src/video_core/texture_cache/image.cpp
|
|
||||||
src/video_core/texture_cache/image.h
|
|
||||||
src/video_core/texture_cache/image_info.cpp
|
|
||||||
src/video_core/texture_cache/image_info.h
|
|
||||||
src/video_core/texture_cache/image_view.cpp
|
|
||||||
src/video_core/texture_cache/image_view.h
|
|
||||||
src/video_core/texture_cache/sampler.cpp
|
|
||||||
src/video_core/texture_cache/sampler.h
|
|
||||||
src/video_core/texture_cache/texture_cache.cpp
|
|
||||||
src/video_core/texture_cache/texture_cache.h
|
|
||||||
src/video_core/texture_cache/tile_manager.cpp
|
|
||||||
src/video_core/texture_cache/tile_manager.h
|
|
||||||
src/video_core/texture_cache/types.h
|
|
||||||
src/video_core/page_manager.cpp
|
|
||||||
src/video_core/page_manager.h
|
|
||||||
src/video_core/multi_level_page_table.h
|
|
||||||
src/video_core/renderdoc.cpp
|
|
||||||
src/video_core/renderdoc.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(INPUT src/input/controller.cpp
|
|
||||||
src/input/controller.h
|
|
||||||
)
|
|
||||||
|
|
||||||
set(EMULATOR src/emulator.cpp
|
|
||||||
src/emulator.h
|
|
||||||
src/sdl_window.h
|
|
||||||
src/sdl_window.cpp
|
|
||||||
)
|
|
||||||
|
|
||||||
# The above is shared in SDL and Qt version (TODO share them all)
|
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
|
||||||
qt_add_resources(RESOURCE_FILES src/shadps4.qrc)
|
|
||||||
|
|
||||||
set(QT_GUI src/qt_gui/about_dialog.cpp
|
|
||||||
src/qt_gui/about_dialog.h
|
|
||||||
src/qt_gui/about_dialog.ui
|
|
||||||
src/qt_gui/cheats_patches.cpp
|
|
||||||
src/qt_gui/cheats_patches.h
|
|
||||||
src/qt_gui/memory_patcher.cpp
|
|
||||||
src/qt_gui/memory_patcher.h
|
|
||||||
src/qt_gui/main_window_ui.h
|
|
||||||
src/qt_gui/main_window.cpp
|
|
||||||
src/qt_gui/main_window.h
|
|
||||||
src/qt_gui/gui_context_menus.h
|
|
||||||
src/qt_gui/game_list_utils.h
|
|
||||||
src/qt_gui/game_info.cpp
|
|
||||||
src/qt_gui/game_info.h
|
|
||||||
src/qt_gui/game_list_frame.cpp
|
|
||||||
src/qt_gui/game_list_frame.h
|
|
||||||
src/qt_gui/game_grid_frame.cpp
|
|
||||||
src/qt_gui/game_grid_frame.h
|
|
||||||
src/qt_gui/game_install_dialog.cpp
|
|
||||||
src/qt_gui/game_install_dialog.h
|
|
||||||
src/qt_gui/pkg_viewer.cpp
|
|
||||||
src/qt_gui/pkg_viewer.h
|
|
||||||
src/qt_gui/trophy_viewer.cpp
|
|
||||||
src/qt_gui/trophy_viewer.h
|
|
||||||
src/qt_gui/elf_viewer.cpp
|
|
||||||
src/qt_gui/elf_viewer.h
|
|
||||||
src/qt_gui/main_window_themes.cpp
|
|
||||||
src/qt_gui/main_window_themes.h
|
|
||||||
src/qt_gui/settings_dialog.cpp
|
|
||||||
src/qt_gui/settings_dialog.h
|
|
||||||
src/qt_gui/settings_dialog.ui
|
|
||||||
src/qt_gui/main.cpp
|
|
||||||
${EMULATOR}
|
|
||||||
${RESOURCE_FILES}
|
|
||||||
${TRANSLATIONS}
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
qt_add_executable(shadps4
|
qt_add_executable(shadps4
|
||||||
${AUDIO_CORE}
|
|
||||||
${INPUT}
|
|
||||||
${QT_GUI}
|
${QT_GUI}
|
||||||
${COMMON}
|
${COMMON}
|
||||||
${CORE}
|
${CORE}
|
||||||
${SHADER_RECOMPILER}
|
${CRYPTO}
|
||||||
${VIDEO_CORE}
|
${FILE_FORMAT}
|
||||||
${EMULATOR}
|
${UTILITIES}
|
||||||
src/images/shadPS4.icns
|
|
||||||
)
|
)
|
||||||
else()
|
else()
|
||||||
add_executable(shadps4
|
add_executable(shadps4
|
||||||
${AUDIO_CORE}
|
${LIBC_SOURCES}
|
||||||
${INPUT}
|
${PAD_SOURCES}
|
||||||
${COMMON}
|
${FILESYSTEM_SOURCES}
|
||||||
${CORE}
|
${HOST_SOURCES}
|
||||||
${SHADER_RECOMPILER}
|
${LIBRARIES}
|
||||||
${VIDEO_CORE}
|
|
||||||
${EMULATOR}
|
|
||||||
src/main.cpp
|
src/main.cpp
|
||||||
|
src/core/loader/elf.cpp
|
||||||
|
src/core/loader/elf.h
|
||||||
|
src/core/virtual_memory.cpp
|
||||||
|
src/core/virtual_memory.h
|
||||||
|
src/core/linker.cpp
|
||||||
|
src/core/linker.h
|
||||||
|
src/core/aerolib/stubs.cpp
|
||||||
|
src/core/aerolib/stubs.h
|
||||||
|
src/core/aerolib/aerolib.cpp
|
||||||
|
src/core/aerolib/aerolib.h
|
||||||
|
src/core/hle/kernel/Objects/physical_memory.h
|
||||||
|
src/core/hle/kernel/Objects/physical_memory.cpp
|
||||||
|
src/core/PS4/HLE/Graphics/video_out.cpp
|
||||||
|
src/core/PS4/HLE/Graphics/video_out.h
|
||||||
|
src/core/hle/kernel/event_queues.cpp
|
||||||
|
src/core/hle/kernel/event_queues.h
|
||||||
|
src/core/hle/kernel/cpu_management.cpp
|
||||||
|
src/core/hle/kernel/cpu_management.h
|
||||||
|
src/core/loader/symbols_resolver.h
|
||||||
|
src/core/loader/symbols_resolver.cpp
|
||||||
|
src/core/hle/libraries/libs.cpp
|
||||||
|
src/core/hle/libraries/libs.h
|
||||||
|
src/core/hle/libraries/libkernel/libkernel.cpp
|
||||||
|
src/core/hle/libraries/libkernel/libkernel.h
|
||||||
|
src/core/hle/libraries/libscegnmdriver/libscegnmdriver.cpp
|
||||||
|
src/core/hle/libraries/libscegnmdriver/libscegnmdriver.h
|
||||||
|
src/core/hle/libraries/libkernel/thread_management.cpp
|
||||||
|
src/core/hle/libraries/libkernel/thread_management.h
|
||||||
|
src/core/hle/kernel/memory_management.cpp
|
||||||
|
src/core/hle/kernel/memory_management.h
|
||||||
|
src/core/hle/error_codes.h
|
||||||
|
src/core/PS4/GPU/gpu_memory.cpp
|
||||||
|
src/core/PS4/GPU/gpu_memory.h
|
||||||
src/emulator.cpp
|
src/emulator.cpp
|
||||||
src/emulator.h
|
src/emulator.h
|
||||||
src/sdl_window.h
|
src/core/hle/kernel/Objects/event_queue.h
|
||||||
src/sdl_window.cpp
|
src/core/hle/kernel/Objects/event_queue.cpp
|
||||||
|
src/core/PS4/HLE/Graphics/Objects/video_out_ctx.cpp
|
||||||
|
src/core/PS4/HLE/Graphics/Objects/video_out_ctx.h
|
||||||
|
src/core/PS4/HLE/Graphics/graphics_ctx.h
|
||||||
|
src/vulkan_util.cpp
|
||||||
|
src/vulkan_util.h
|
||||||
|
src/core/PS4/GPU/video_out_buffer.cpp
|
||||||
|
src/core/PS4/GPU/video_out_buffer.h
|
||||||
|
src/core/PS4/HLE/Graphics/graphics_render.cpp
|
||||||
|
src/core/PS4/HLE/Graphics/graphics_render.h
|
||||||
|
src/core/PS4/GPU/tile_manager.cpp
|
||||||
|
src/core/PS4/GPU/tile_manager.h
|
||||||
|
src/core/hle/libraries/libkernel/time_management.cpp
|
||||||
|
src/core/hle/libraries/libkernel/time_management.h
|
||||||
|
${COMMON}
|
||||||
|
${CORE}
|
||||||
|
${CRYPTO}
|
||||||
|
${FILE_FORMAT}
|
||||||
|
${UTILITIES}
|
||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
create_target_directory_groups(shadps4)
|
create_target_directory_groups(shadps4)
|
||||||
|
|
||||||
target_link_libraries(shadps4 PRIVATE magic_enum::magic_enum fmt::fmt toml11::toml11 tsl::robin_map xbyak::xbyak Tracy::TracyClient RenderDoc::API FFmpeg::ffmpeg)
|
target_link_libraries(shadps4 PRIVATE magic_enum::magic_enum fmt::fmt toml11::toml11 spdlog::spdlog)
|
||||||
target_link_libraries(shadps4 PRIVATE Boost::headers GPUOpen::VulkanMemoryAllocator sirit Vulkan::Headers xxHash::xxhash Zydis::Zydis glslang::SPIRV glslang::glslang SDL3::SDL3)
|
target_link_libraries(shadps4 PRIVATE discord-rpc vulkan-1 xxhash Zydis)
|
||||||
|
|
||||||
if (APPLE)
|
|
||||||
# Reserve system-managed memory space.
|
|
||||||
target_link_options(shadps4 PRIVATE -Wl,-no_pie,-no_fixup_chains,-no_huge,-pagezero_size,0x400000,-segaddr,GUEST_SYSTEM,0x400000,-image_base,0x10000000000)
|
|
||||||
|
|
||||||
# Link MoltenVK for Vulkan support
|
|
||||||
find_library(MOLTENVK MoltenVK REQUIRED)
|
|
||||||
target_link_libraries(shadps4 PRIVATE ${MOLTENVK})
|
|
||||||
|
|
||||||
# Replacement for std::chrono::time_zone
|
|
||||||
target_link_libraries(shadps4 PRIVATE date::date-tz)
|
|
||||||
|
|
||||||
# Half float conversions for F16C patches
|
|
||||||
target_link_libraries(shadps4 PRIVATE half)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT ENABLE_QT_GUI)
|
if(NOT ENABLE_QT_GUI)
|
||||||
target_link_libraries(shadps4 PRIVATE SDL3::SDL3)
|
target_link_libraries(shadps4 PRIVATE SDL3-shared)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND MSVC)
|
if(CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND MSVC)
|
||||||
target_link_libraries(shadps4 PRIVATE cryptoppwin zlib-ng::zlib)
|
target_link_libraries(shadps4 PRIVATE cryptoppwin zlib)
|
||||||
else()
|
else()
|
||||||
target_link_libraries(shadps4 PRIVATE cryptopp::cryptopp zlib-ng::zlib)
|
target_link_libraries(shadps4 PRIVATE cryptopp::cryptopp zlib)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
target_link_libraries(shadps4 PRIVATE Qt6::Widgets Qt6::Concurrent Qt6::Network)
|
target_link_libraries(shadps4 PRIVATE Qt6::Widgets Qt6::Concurrent)
|
||||||
add_definitions(-DENABLE_QT_GUI)
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (WIN32)
|
if (WIN32)
|
||||||
target_link_libraries(shadps4 PRIVATE mincore winpthreads)
|
target_link_libraries(shadps4 PRIVATE mincore winpthread clang_rt.builtins-x86_64.lib)
|
||||||
|
|
||||||
if (MSVC)
|
|
||||||
# MSVC likes putting opinions on what people can use, disable:
|
|
||||||
add_definitions(-D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS)
|
add_definitions(-D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS)
|
||||||
endif()
|
|
||||||
|
|
||||||
add_definitions(-DNOMINMAX -DWIN32_LEAN_AND_MEAN)
|
add_definitions(-DNOMINMAX -DWIN32_LEAN_AND_MEAN)
|
||||||
|
|
||||||
if (MSVC)
|
|
||||||
# Needed for conflicts with time.h of windows.h
|
|
||||||
add_definitions(-D_TIMESPEC_DEFINED)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Target Windows 10 RS5
|
|
||||||
add_definitions(-DNTDDI_VERSION=0x0A000006 -D_WIN32_WINNT=0x0A00 -DWINVER=0x0A00)
|
|
||||||
|
|
||||||
if (MSVC)
|
|
||||||
target_link_libraries(shadps4 PRIVATE clang_rt.builtins-x86_64.lib)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Disable ASLR so we can reserve the user area
|
|
||||||
if (MSVC)
|
|
||||||
target_link_options(shadps4 PRIVATE /DYNAMICBASE:NO)
|
|
||||||
else()
|
|
||||||
target_link_options(shadps4 PRIVATE -Wl,--disable-dynamicbase)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Increase stack commit area (Needed, otherwise there are crashes)
|
|
||||||
if (MSVC)
|
|
||||||
target_link_options(shadps4 PRIVATE /STACK:0x200000,0x200000)
|
|
||||||
else()
|
|
||||||
target_link_options(shadps4 PRIVATE -Wl,--stack,2097152)
|
|
||||||
endif()
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WIN32)
|
if(WIN32)
|
||||||
target_sources(shadps4 PRIVATE src/shadps4.rc)
|
target_sources(shadps4 PRIVATE src/shadps4.rc)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_definitions(-DBOOST_ASIO_STANDALONE)
|
|
||||||
|
|
||||||
target_include_directories(shadps4 PRIVATE ${CMAKE_CURRENT_SOURCE_DIR})
|
target_include_directories(shadps4 PRIVATE ${CMAKE_CURRENT_SOURCE_DIR})
|
||||||
|
|
||||||
# Shaders sources
|
|
||||||
set(HOST_SHADERS_INCLUDE ${CMAKE_CURRENT_SOURCE_DIR}/src/video_core/host_shaders)
|
|
||||||
|
|
||||||
add_subdirectory(${HOST_SHADERS_INCLUDE})
|
|
||||||
add_dependencies(shadps4 host_shaders)
|
|
||||||
target_include_directories(shadps4 PRIVATE ${HOST_SHADERS_INCLUDE})
|
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
set_target_properties(shadps4 PROPERTIES
|
set_target_properties(shadps4 PROPERTIES
|
||||||
# WIN32_EXECUTABLE ON
|
WIN32_EXECUTABLE ON
|
||||||
MACOSX_BUNDLE ON
|
MACOSX_BUNDLE ON
|
||||||
MACOSX_BUNDLE_ICON_FILE shadPS4.icns)
|
)
|
||||||
|
endif()
|
||||||
set_source_files_properties(src/images/shadPS4.icns PROPERTIES
|
|
||||||
MACOSX_PACKAGE_LOCATION Resources)
|
if(CMAKE_BUILD_TYPE STREQUAL "Debug")
|
||||||
|
add_custom_command(TARGET shadps4 POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
|
"${PROJECT_SOURCE_DIR}/externals/zlib-ng-win/bin/zlib-ngd2.dll" $<TARGET_FILE_DIR:shadps4>)
|
||||||
|
else()
|
||||||
|
add_custom_command(TARGET shadps4 POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
|
"${PROJECT_SOURCE_DIR}/externals/zlib-ng-win/bin/zlib-ng2.dll" $<TARGET_FILE_DIR:shadps4>)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if(NOT ENABLE_QT_GUI)
|
||||||
|
add_custom_command(TARGET shadps4 POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
|
$<TARGET_FILE:SDL3-shared>
|
||||||
|
$<TARGET_FILE_DIR:shadps4>)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
if (WIN32)
|
||||||
|
add_custom_command(TARGET shadps4 POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
|
"${PROJECT_SOURCE_DIR}/third-party/winpthread/bin/libwinpthread-1.dll" $<TARGET_FILE_DIR:shadps4>)
|
||||||
endif()
|
endif()
|
||||||
|
|
|
@ -1,33 +1,22 @@
|
||||||
{
|
{
|
||||||
"configurations": [
|
"configurations": [
|
||||||
{
|
|
||||||
"name": "x64-Clang-Release",
|
|
||||||
"generator": "Ninja",
|
|
||||||
"configurationType": "Release",
|
|
||||||
"buildRoot": "${projectDir}\\Build\\${name}",
|
|
||||||
"installRoot": "${projectDir}\\Install\\${name}",
|
|
||||||
"cmakeCommandArgs": "",
|
|
||||||
"buildCommandArgs": "",
|
|
||||||
"ctestCommandArgs": "",
|
|
||||||
"inheritEnvironments": [ "clang_cl_x64_x64" ]
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"name": "x64-Clang-Debug",
|
"name": "x64-Clang-Debug",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "Debug",
|
"configurationType": "Debug",
|
||||||
"buildRoot": "${projectDir}\\Build\\${name}",
|
"buildRoot": "${projectDir}\\out\\build\\${name}",
|
||||||
"installRoot": "${projectDir}\\Install\\${name}",
|
"installRoot": "${projectDir}\\out\\install\\${name}",
|
||||||
"cmakeCommandArgs": "",
|
"cmakeCommandArgs": "",
|
||||||
"buildCommandArgs": "",
|
"buildCommandArgs": "",
|
||||||
"ctestCommandArgs": "",
|
"ctestCommandArgs": "",
|
||||||
"inheritEnvironments": [ "clang_cl_x64_x64" ]
|
"inheritEnvironments": [ "clang_cl_x64_x64" ]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "x64-Clang-RelWithDebInfo",
|
"name": "x64-Clang-Release",
|
||||||
"generator": "Ninja",
|
"generator": "Ninja",
|
||||||
"configurationType": "RelWithDebInfo",
|
"configurationType": "RelWithDebInfo",
|
||||||
"buildRoot": "${projectDir}\\Build\\${name}",
|
"buildRoot": "${projectDir}\\out\\build\\${name}",
|
||||||
"installRoot": "${projectDir}\\Install\\${name}",
|
"installRoot": "${projectDir}\\out\\install\\${name}",
|
||||||
"cmakeCommandArgs": "",
|
"cmakeCommandArgs": "",
|
||||||
"buildCommandArgs": "",
|
"buildCommandArgs": "",
|
||||||
"ctestCommandArgs": "",
|
"ctestCommandArgs": "",
|
||||||
|
|
|
@ -31,7 +31,7 @@ Follow the indentation/whitespace style shown below. Do not use tabs, use 4-spac
|
||||||
* For doc-comments (Doxygen comments), use /// if it's a single line, else use the /** */ style featured in the example. Start the text on the second line, not the first containing /**.
|
* For doc-comments (Doxygen comments), use /// if it's a single line, else use the /** */ style featured in the example. Start the text on the second line, not the first containing /**.
|
||||||
* For items that are both defined and declared in two separate files, put the doc-comment only next to the associated declaration. (In a header file, usually.) Otherwise, put it next to the implementation. Never duplicate doc-comments in both places.
|
* For items that are both defined and declared in two separate files, put the doc-comment only next to the associated declaration. (In a header file, usually.) Otherwise, put it next to the implementation. Never duplicate doc-comments in both places.
|
||||||
|
|
||||||
```c++
|
```
|
||||||
// Includes should be sorted lexicographically
|
// Includes should be sorted lexicographically
|
||||||
// STD includes first
|
// STD includes first
|
||||||
#include <array>
|
#include <array>
|
||||||
|
@ -69,7 +69,7 @@ enum class SomeEnum {
|
||||||
* Note that the asterisks are indented by one space to align to the first line.
|
* Note that the asterisks are indented by one space to align to the first line.
|
||||||
*/
|
*/
|
||||||
struct Position {
|
struct Position {
|
||||||
// Always initialize member variables!
|
// Always intitialize member variables!
|
||||||
int x{};
|
int x{};
|
||||||
int y{};
|
int y{};
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,11 +0,0 @@
|
||||||
Copyright (c) <year> <owner>.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
|
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
|
|
||||||
|
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
|
|
||||||
|
|
||||||
3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
@ -1,7 +0,0 @@
|
||||||
Boost Software License - Version 1.0 - August 17th, 2003
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person or organization obtaining a copy of the software and accompanying documentation covered by this license (the "Software") to use, reproduce, display, distribute, execute, and transmit the Software, and to prepare derivative works of the Software, and to permit third-parties to whom the Software is furnished to do so, all subject to the following:
|
|
||||||
|
|
||||||
The copyright notices in the Software and this entire statement, including the above license grant, this restriction and the following disclaimer, must be included in all copies of the Software, in whole or in part, and all derivative works of the Software, unless such copies or derivative works are solely in the form of machine-executable object code generated by a source language processor.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
@ -1,121 +0,0 @@
|
||||||
Creative Commons Legal Code
|
|
||||||
|
|
||||||
CC0 1.0 Universal
|
|
||||||
|
|
||||||
CREATIVE COMMONS CORPORATION IS NOT A LAW FIRM AND DOES NOT PROVIDE
|
|
||||||
LEGAL SERVICES. DISTRIBUTION OF THIS DOCUMENT DOES NOT CREATE AN
|
|
||||||
ATTORNEY-CLIENT RELATIONSHIP. CREATIVE COMMONS PROVIDES THIS
|
|
||||||
INFORMATION ON AN "AS-IS" BASIS. CREATIVE COMMONS MAKES NO WARRANTIES
|
|
||||||
REGARDING THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS
|
|
||||||
PROVIDED HEREUNDER, AND DISCLAIMS LIABILITY FOR DAMAGES RESULTING FROM
|
|
||||||
THE USE OF THIS DOCUMENT OR THE INFORMATION OR WORKS PROVIDED
|
|
||||||
HEREUNDER.
|
|
||||||
|
|
||||||
Statement of Purpose
|
|
||||||
|
|
||||||
The laws of most jurisdictions throughout the world automatically confer
|
|
||||||
exclusive Copyright and Related Rights (defined below) upon the creator
|
|
||||||
and subsequent owner(s) (each and all, an "owner") of an original work of
|
|
||||||
authorship and/or a database (each, a "Work").
|
|
||||||
|
|
||||||
Certain owners wish to permanently relinquish those rights to a Work for
|
|
||||||
the purpose of contributing to a commons of creative, cultural and
|
|
||||||
scientific works ("Commons") that the public can reliably and without fear
|
|
||||||
of later claims of infringement build upon, modify, incorporate in other
|
|
||||||
works, reuse and redistribute as freely as possible in any form whatsoever
|
|
||||||
and for any purposes, including without limitation commercial purposes.
|
|
||||||
These owners may contribute to the Commons to promote the ideal of a free
|
|
||||||
culture and the further production of creative, cultural and scientific
|
|
||||||
works, or to gain reputation or greater distribution for their Work in
|
|
||||||
part through the use and efforts of others.
|
|
||||||
|
|
||||||
For these and/or other purposes and motivations, and without any
|
|
||||||
expectation of additional consideration or compensation, the person
|
|
||||||
associating CC0 with a Work (the "Affirmer"), to the extent that he or she
|
|
||||||
is an owner of Copyright and Related Rights in the Work, voluntarily
|
|
||||||
elects to apply CC0 to the Work and publicly distribute the Work under its
|
|
||||||
terms, with knowledge of his or her Copyright and Related Rights in the
|
|
||||||
Work and the meaning and intended legal effect of CC0 on those rights.
|
|
||||||
|
|
||||||
1. Copyright and Related Rights. A Work made available under CC0 may be
|
|
||||||
protected by copyright and related or neighboring rights ("Copyright and
|
|
||||||
Related Rights"). Copyright and Related Rights include, but are not
|
|
||||||
limited to, the following:
|
|
||||||
|
|
||||||
i. the right to reproduce, adapt, distribute, perform, display,
|
|
||||||
communicate, and translate a Work;
|
|
||||||
ii. moral rights retained by the original author(s) and/or performer(s);
|
|
||||||
iii. publicity and privacy rights pertaining to a person's image or
|
|
||||||
likeness depicted in a Work;
|
|
||||||
iv. rights protecting against unfair competition in regards to a Work,
|
|
||||||
subject to the limitations in paragraph 4(a), below;
|
|
||||||
v. rights protecting the extraction, dissemination, use and reuse of data
|
|
||||||
in a Work;
|
|
||||||
vi. database rights (such as those arising under Directive 96/9/EC of the
|
|
||||||
European Parliament and of the Council of 11 March 1996 on the legal
|
|
||||||
protection of databases, and under any national implementation
|
|
||||||
thereof, including any amended or successor version of such
|
|
||||||
directive); and
|
|
||||||
vii. other similar, equivalent or corresponding rights throughout the
|
|
||||||
world based on applicable law or treaty, and any national
|
|
||||||
implementations thereof.
|
|
||||||
|
|
||||||
2. Waiver. To the greatest extent permitted by, but not in contravention
|
|
||||||
of, applicable law, Affirmer hereby overtly, fully, permanently,
|
|
||||||
irrevocably and unconditionally waives, abandons, and surrenders all of
|
|
||||||
Affirmer's Copyright and Related Rights and associated claims and causes
|
|
||||||
of action, whether now known or unknown (including existing as well as
|
|
||||||
future claims and causes of action), in the Work (i) in all territories
|
|
||||||
worldwide, (ii) for the maximum duration provided by applicable law or
|
|
||||||
treaty (including future time extensions), (iii) in any current or future
|
|
||||||
medium and for any number of copies, and (iv) for any purpose whatsoever,
|
|
||||||
including without limitation commercial, advertising or promotional
|
|
||||||
purposes (the "Waiver"). Affirmer makes the Waiver for the benefit of each
|
|
||||||
member of the public at large and to the detriment of Affirmer's heirs and
|
|
||||||
successors, fully intending that such Waiver shall not be subject to
|
|
||||||
revocation, rescission, cancellation, termination, or any other legal or
|
|
||||||
equitable action to disrupt the quiet enjoyment of the Work by the public
|
|
||||||
as contemplated by Affirmer's express Statement of Purpose.
|
|
||||||
|
|
||||||
3. Public License Fallback. Should any part of the Waiver for any reason
|
|
||||||
be judged legally invalid or ineffective under applicable law, then the
|
|
||||||
Waiver shall be preserved to the maximum extent permitted taking into
|
|
||||||
account Affirmer's express Statement of Purpose. In addition, to the
|
|
||||||
extent the Waiver is so judged Affirmer hereby grants to each affected
|
|
||||||
person a royalty-free, non transferable, non sublicensable, non exclusive,
|
|
||||||
irrevocable and unconditional license to exercise Affirmer's Copyright and
|
|
||||||
Related Rights in the Work (i) in all territories worldwide, (ii) for the
|
|
||||||
maximum duration provided by applicable law or treaty (including future
|
|
||||||
time extensions), (iii) in any current or future medium and for any number
|
|
||||||
of copies, and (iv) for any purpose whatsoever, including without
|
|
||||||
limitation commercial, advertising or promotional purposes (the
|
|
||||||
"License"). The License shall be deemed effective as of the date CC0 was
|
|
||||||
applied by Affirmer to the Work. Should any part of the License for any
|
|
||||||
reason be judged legally invalid or ineffective under applicable law, such
|
|
||||||
partial invalidity or ineffectiveness shall not invalidate the remainder
|
|
||||||
of the License, and in such case Affirmer hereby affirms that he or she
|
|
||||||
will not (i) exercise any of his or her remaining Copyright and Related
|
|
||||||
Rights in the Work or (ii) assert any associated claims and causes of
|
|
||||||
action with respect to the Work, in either case contrary to Affirmer's
|
|
||||||
express Statement of Purpose.
|
|
||||||
|
|
||||||
4. Limitations and Disclaimers.
|
|
||||||
|
|
||||||
a. No trademark or patent rights held by Affirmer are waived, abandoned,
|
|
||||||
surrendered, licensed or otherwise affected by this document.
|
|
||||||
b. Affirmer offers the Work as-is and makes no representations or
|
|
||||||
warranties of any kind concerning the Work, express, implied,
|
|
||||||
statutory or otherwise, including without limitation warranties of
|
|
||||||
title, merchantability, fitness for a particular purpose, non
|
|
||||||
infringement, or the absence of latent or other defects, accuracy, or
|
|
||||||
the present or absence of errors, whether or not discoverable, all to
|
|
||||||
the greatest extent permissible under applicable law.
|
|
||||||
c. Affirmer disclaims responsibility for clearing rights of other persons
|
|
||||||
that may apply to the Work or any use thereof, including without
|
|
||||||
limitation any person's Copyright and Related Rights in the Work.
|
|
||||||
Further, Affirmer disclaims responsibility for obtaining any necessary
|
|
||||||
consents, permissions or other rights required for any use of the
|
|
||||||
Work.
|
|
||||||
d. Affirmer understands and acknowledges that Creative Commons is not a
|
|
||||||
party to this document and has no duty or obligation with respect to
|
|
||||||
this CC0 or use of the Work.
|
|
165
README.md
|
@ -3,161 +3,64 @@ SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
||||||
SPDX-License-Identifier: GPL-2.0-or-later
|
SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
-->
|
-->
|
||||||
|
|
||||||
<h1 align="center">
|
# shadPS4
|
||||||
<br>
|
|
||||||
<a href="https://shadps4.net/"><img src="https://github.com/shadps4-emu/shadPS4/blob/main/.github/shadps4.png" width="220"></a>
|
|
||||||
<br>
|
|
||||||
<b>shadPS4</b>
|
|
||||||
<br>
|
|
||||||
</h1>
|
|
||||||
|
|
||||||
<h1 align="center">
|
An early PS4 emulator for Windows and Linux written in C++
|
||||||
<a href="https://discord.gg/MyZRaBngxA">
|
by shadow , skmp , wheremyfoodat , GPUCode
|
||||||
<img src="https://img.shields.io/discord/1080089157554155590?color=5865F2&label=shadPS4 Discord&logo=Discord&logoColor=white" width="240">
|
|
||||||
<a href="https://github.com/shadps4-emu/shadPS4/releases/latest">
|
|
||||||
<img src="https://img.shields.io/github/downloads/shadps4-emu/shadPS4/total.svg" width="140">
|
|
||||||
<a href="https://shadps4.net/">
|
|
||||||
<img src="https://img.shields.io/badge/shadPS4-website-8A2BE2" width="150">
|
|
||||||
<a href="https://x.com/shadps4">
|
|
||||||
<img src="https://img.shields.io/badge/-Join%20us-black?logo=X&logoColor=white" width="100">
|
|
||||||
<a href="https://github.com/shadps4-emu/shadPS4/stargazers">
|
|
||||||
<img src="https://img.shields.io/github/stars/shadps4-emu/shadPS4" width="120">
|
|
||||||
</h1>
|
|
||||||
|
|
||||||
<p align="center">
|
[Check us on twitter](https://twitter.com/shadps4 "Check us on twitter")
|
||||||
<a href="https://shadps4.net/">
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/Sonic Mania.png" width="400">
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/Bloodborne.png" width="400">
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/Undertale.png" width="400">
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/We are DOOMED.png" width="400">
|
|
||||||
</p>
|
|
||||||
|
|
||||||
# General information
|
|
||||||
|
|
||||||
shadPS4 is an early **PlayStation 4** emulator for **Windows**, **Linux** and **macOS** written in C++.
|
|
||||||
|
|
||||||
If you encounter problems or have doubts, do not hesitate to look at the [**Quickstart**](https://github.com/shadps4-emu/shadPS4/blob/main/documents/Quickstart/Quickstart.md).
|
|
||||||
|
|
||||||
To verify that a game works, you can look at [**shadPS4 Game Compatibility**](https://github.com/shadps4-emu/shadps4-game-compatibility).
|
|
||||||
|
|
||||||
To discuss shadPS4 development, suggest ideas or to ask for help, join our [**Discord server**](https://discord.gg/MyZRaBngxA).
|
|
||||||
|
|
||||||
To get the latest news, go to our [**X (Twitter)**](https://x.com/shadps4) or our [**website**](https://shadps4.net/).
|
|
||||||
|
|
||||||
# Status
|
# Status
|
||||||
|
|
||||||
> [!IMPORTANT]
|
Early progress , a small amount of ps4 sdk demos and homebrew games working
|
||||||
> shadPS4 is early in developement, don't expect a flawless experience.
|
|
||||||
|
|
||||||
Currently, the emulator successfully runs small games like [**Sonic Mania**](https://www.youtube.com/watch?v=AAHoNzhHyCU), [**Undertale**](https://youtu.be/5zIvdy65Ro4) and it can even *somewhat* run [**Bloodborne**](https://www.youtube.com/watch?v=wC6s0avpQRE).
|
# Why?
|
||||||
|
|
||||||
# Why
|
The project started as a fun project. Due to short free time, it will probably take a while since it will be able to run something decent, but I am trying to do regular small commits.
|
||||||
|
|
||||||
This project began as a fun project. Given our limited free time, it may take some time before shadPS4 can run more complex games, but we're committed to making small, regular updates.
|
# Build
|
||||||
|
|
||||||
# Building
|
|
||||||
|
|
||||||
## Windows
|
## Windows
|
||||||
|
|
||||||
Check the build instructions for [**Windows**](https://github.com/shadps4-emu/shadPS4/blob/main/documents/building-windows.md).
|
Check building instructions in [windows build](https://github.com/georgemoralis/shadPS4/blob/main/documents/building-windows.md)
|
||||||
|
|
||||||
## Linux
|
## Linux
|
||||||
|
|
||||||
Check the build instructions for [**Linux**](https://github.com/shadps4-emu/shadPS4/blob/main/documents/building-linux.md).
|
Check building instructions in [linux build](https://github.com/georgemoralis/shadPS4/blob/main/documents/linux_building.md)
|
||||||
|
|
||||||
## Building status
|
## Build status
|
||||||
|
|
||||||
<details>
|
|Platform|Build status|
|
||||||
<summary><b>Windows</b></summary>
|
|--------|------------|
|
||||||
|
|Windows SDL build|[![Windows-sdl](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows.yml)
|
||||||
|
|Linux SDL build|[![Linux-sdl](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux.yml)
|
||||||
|
|
||||||
| Windows | Build status |
|
To discuss the development of this emulator, please join our Discord server: [![Discord](https://img.shields.io/discord/1080089157554155590)](https://discord.gg/MyZRaBngxA)
|
||||||
|--------|--------|
|
|
||||||
|Windows SDL Build|[![Windows-sdl](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows.yml)
|
|
||||||
|Windows Qt Build|[![Windows-qt](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows-qt.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/windows-qt.yml)
|
|
||||||
</details>
|
|
||||||
|
|
||||||
<details>
|
# Who are you?
|
||||||
<summary><b>Linux</b></summary>
|
|
||||||
|
|
||||||
| Linux | Build status |
|
Old emulator fans and devs can recognize me as "shadow". I was the founder and coder for a lot of emulation projects:
|
||||||
|--------|--------|
|
* PCSX
|
||||||
|Linux SDL Build|[![Linux-sdl](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux.yml)
|
* PCSX2
|
||||||
|Linux Qt Build|[![Linux-qt](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux-qt.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/linux-qt.yml)
|
* PCSP
|
||||||
</details>
|
* JPCSP
|
||||||
|
* arcadeflex
|
||||||
|
* rpcs3 contributor
|
||||||
|
|
||||||
<details>
|
# Contribution
|
||||||
<summary><b>macOS</b></summary>
|
|
||||||
|
|
||||||
| macOS | Build status |
|
I currently accept any contribution, here is a list of some items that may be useful:
|
||||||
|--------|--------|
|
|
||||||
|macOS SDL Build|[![macOS-sdl](https://github.com/shadps4-emu/shadPS4/actions/workflows/macos.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/macos.yml)
|
|
||||||
|macOS Qt Build|[![macOS-qt](https://github.com/shadps4-emu/shadPS4/actions/workflows/macos-qt.yml/badge.svg)](https://github.com/shadps4-emu/shadPS4/actions/workflows/macos-qt.yml)
|
|
||||||
</details>
|
|
||||||
|
|
||||||
# Debugging and reporting issues
|
* PKG extractor (there was initial work on this, search project history commits).
|
||||||
|
* Initial GUI with imgui, SDL3 and Vulkan.
|
||||||
|
* to be filled...
|
||||||
|
|
||||||
For more information on how to test, debug and report issues with the emulator or games, read the [Debugging documentation](https://github.com/shadps4-emu/shadPS4/blob/main/documents/Debugging/Debugging.md).
|
# Documentation
|
||||||
|
|
||||||
# Keyboard mapping
|
Wiki has some documentation for PS4 PKG format
|
||||||
|
|
||||||
> [!NOTE]
|
[PKG PS4 File Format](https://github.com/georgemoralis/shadPS4/wiki/PKG-Information "PKG PS4 File Format")
|
||||||
> Xbox and DualShock controllers work out of the box.
|
|
||||||
|
|
||||||
| Controller button | Keyboard equivelant |
|
|
||||||
|-------------|-------------|
|
|
||||||
LEFT AXIS UP | W |
|
|
||||||
LEFT AXIS DOWN | S |
|
|
||||||
LEFT AXIS LEFT | A |
|
|
||||||
LEFT AXIS RIGHT | D |
|
|
||||||
RIGHT AXIS UP | I |
|
|
||||||
RIGHT AXIS DOWN | K |
|
|
||||||
RIGHT AXIS LEFT | J |
|
|
||||||
RIGHT AXIS RIGHT | L |
|
|
||||||
TRIANGLE | Numpad 8 |
|
|
||||||
CIRCLE | Numpad 6 |
|
|
||||||
CROSS | Numpad 2 |
|
|
||||||
SQUARE | Numpad 4 |
|
|
||||||
PAD UP | UP |
|
|
||||||
PAD DOWN | DOWN |
|
|
||||||
PAD LEFT | LEFT |
|
|
||||||
PAD RIGHT | RIGHT |
|
|
||||||
OPTIONS | RETURN |
|
|
||||||
TOUCH PAD | SPACE |
|
|
||||||
L1 | Q |
|
|
||||||
R1 | U |
|
|
||||||
L2 | E |
|
|
||||||
R2 | O |
|
|
||||||
L3 | X |
|
|
||||||
R3 | M |
|
|
||||||
|
|
||||||
# Main team
|
|
||||||
|
|
||||||
- [**georgemoralis**](https://github.com/georgemoralis)
|
|
||||||
- [**raphaelthegreat**](https://github.com/raphaelthegreat)
|
|
||||||
- [**psucien**](https://github.com/psucien)
|
|
||||||
- [**skmp**](https://github.com/skmp)
|
|
||||||
- [**wheremyfoodat**](https://github.com/wheremyfoodat)
|
|
||||||
- [**raziel1000**](https://github.com/raziel1000)
|
|
||||||
|
|
||||||
Logo is done by [**Xphalnos**](https://github.com/Xphalnos)
|
|
||||||
|
|
||||||
# Contributing
|
|
||||||
|
|
||||||
If you want to contribute, please look the [**CONTRIBUTING.md**](https://github.com/shadps4-emu/shadPS4/blob/main/CONTRIBUTING.md) file.
|
|
||||||
|
|
||||||
Open a PR and we'll check it :)
|
|
||||||
|
|
||||||
# Contributors
|
|
||||||
|
|
||||||
<a href="https://github.com/shadps4-emu/shadPS4/graphs/contributors">
|
|
||||||
<img src="https://contrib.rocks/image?repo=shadps4-emu/shadPS4&max=15">
|
|
||||||
</a>
|
|
||||||
|
|
||||||
# Sister Projects
|
# Sister Projects
|
||||||
|
- [Panda3DS](https://github.com/wheremyfoodat/Panda3DS): An multiplatform 3DS emulator from our co-author wheremyfoodat
|
||||||
- [**Panda3DS**](https://github.com/wheremyfoodat/Panda3DS): A multiplatform 3DS emulator from our co-author wheremyfoodat.
|
- [hydra](https://github.com/hydra-emu/hydra): An multisystem , multiplatform emulator (chip-8,GB,NES,N64) from Paris
|
||||||
- [**hydra**](https://github.com/hydra-emu/hydra): A multisystem, multiplatform emulator (chip-8, GB, NES, N64) from Paris.
|
|
||||||
|
|
||||||
# License
|
|
||||||
|
|
||||||
- [**GPL-2.0 license**](https://github.com/shadps4-emu/shadPS4/blob/main/LICENSE)
|
|
||||||
|
|
|
@ -1,23 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
find_package(PkgConfig QUIET)
|
|
||||||
pkg_check_modules(FFMPEG QUIET IMPORTED_TARGET libavcodec libavfilter libavformat libavutil libswresample libswscale)
|
|
||||||
|
|
||||||
find_file(FFMPEG_VERSION_FILE libavutil/ffversion.h HINTS "${FFMPEG_libavutil_INCLUDEDIR}")
|
|
||||||
if (FFMPEG_VERSION_FILE)
|
|
||||||
file(STRINGS "${FFMPEG_VERSION_FILE}" FFMPEG_VERSION_LINE REGEX "FFMPEG_VERSION")
|
|
||||||
string(REGEX MATCH "[0-9.]+" FFMPEG_VERSION "${FFMPEG_VERSION_LINE}")
|
|
||||||
unset(FFMPEG_VERSION_LINE)
|
|
||||||
unset(FFMPEG_VERSION_FILE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(FFmpeg
|
|
||||||
REQUIRED_VARS FFMPEG_LINK_LIBRARIES
|
|
||||||
VERSION_VAR FFMPEG_VERSION
|
|
||||||
)
|
|
||||||
|
|
||||||
if (FFmpeg_FOUND AND NOT TARGET FFmpeg::ffmpeg)
|
|
||||||
add_library(FFmpeg::ffmpeg ALIAS PkgConfig::FFMPEG)
|
|
||||||
endif()
|
|
|
@ -1,25 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
find_path(RENDERDOC_INCLUDE_DIR renderdoc_app.h)
|
|
||||||
|
|
||||||
if (RENDERDOC_INCLUDE_DIR AND EXISTS "${RENDERDOC_INCLUDE_DIR}/renderdoc_app.h")
|
|
||||||
file(STRINGS "${RENDERDOC_INCLUDE_DIR}/renderdoc_app.h" RENDERDOC_VERSION_LINE REGEX "typedef struct RENDERDOC_API")
|
|
||||||
string(REGEX REPLACE ".*typedef struct RENDERDOC_API_([0-9]+)_([0-9]+)_([0-9]+).*" "\\1.\\2.\\3" RENDERDOC_VERSION "${RENDERDOC_VERSION_LINE}")
|
|
||||||
unset(RENDERDOC_VERSION_LINE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(RenderDoc
|
|
||||||
REQUIRED_VARS RENDERDOC_INCLUDE_DIR
|
|
||||||
VERSION_VAR RENDERDOC_VERSION
|
|
||||||
)
|
|
||||||
|
|
||||||
if (RenderDoc_FOUND AND NOT TARGET RenderDoc::API)
|
|
||||||
add_library(RenderDoc::API INTERFACE IMPORTED)
|
|
||||||
set_target_properties(RenderDoc::API PROPERTIES
|
|
||||||
INTERFACE_INCLUDE_DIRECTORIES "${RENDERDOC_INCLUDE_DIR}"
|
|
||||||
)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
mark_as_advanced(RENDERDOC_INCLUDE_DIR)
|
|
|
@ -1,15 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
find_package(PkgConfig QUIET)
|
|
||||||
pkg_search_module(CRYPTOPP QUIET IMPORTED_TARGET libcryptopp)
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(cryptopp
|
|
||||||
REQUIRED_VARS CRYPTOPP_LINK_LIBRARIES
|
|
||||||
VERSION_VAR CRYPTOPP_VERSION
|
|
||||||
)
|
|
||||||
|
|
||||||
if (cryptopp_FOUND AND NOT TARGET cryptopp::cryptopp)
|
|
||||||
add_library(cryptopp::cryptopp ALIAS PkgConfig::CRYPTOPP)
|
|
||||||
endif()
|
|
|
@ -1,15 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
find_package(PkgConfig QUIET)
|
|
||||||
pkg_search_module(XXHASH QUIET IMPORTED_TARGET libxxhash)
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(xxHash
|
|
||||||
REQUIRED_VARS XXHASH_LINK_LIBRARIES
|
|
||||||
VERSION_VAR XXHASH_VERSION
|
|
||||||
)
|
|
||||||
|
|
||||||
if (xxHash_FOUND AND NOT TARGET xxHash::xxhash)
|
|
||||||
add_library(xxHash::xxhash ALIAS PkgConfig::XXHASH)
|
|
||||||
endif()
|
|
|
@ -1,15 +0,0 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
find_package(PkgConfig QUIET)
|
|
||||||
pkg_search_module(ZLIB_NG QUIET IMPORTED_TARGET zlib-ng)
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(zlib-ng
|
|
||||||
REQUIRED_VARS ZLIB_NG_LINK_LIBRARIES
|
|
||||||
VERSION_VAR ZLIB_NG_VERSION
|
|
||||||
)
|
|
||||||
|
|
||||||
if (zlib-ng_FOUND AND NOT TARGET zlib-ng::zlib)
|
|
||||||
add_library(zlib-ng::zlib ALIAS PkgConfig::ZLIB_NG)
|
|
||||||
endif()
|
|
|
@ -1,156 +0,0 @@
|
||||||
<!--
|
|
||||||
SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
|
||||||
SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
-->
|
|
||||||
|
|
||||||
# Debugging and reporting issues about shadPS4 and games
|
|
||||||
|
|
||||||
This document covers information about debugging, troubleshooting and reporting developer-side issues related to shadPS4 and games.
|
|
||||||
|
|
||||||
## Setup
|
|
||||||
|
|
||||||
This section will guide you through setting up tools for debugging the emulator. This list will likely expand as more tools and platforms receive consistent setups.
|
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary>Windows and Visual Studio</summary>
|
|
||||||
|
|
||||||
Make sure you have the project set up for building on Windows with Visual Studio and CMake: [Build shadPS4 for Windows
|
|
||||||
](https://github.com/shadps4-emu/shadPS4/blob/main/documents/building-windows.md)
|
|
||||||
|
|
||||||
1. Open the project folder in Visual Studio **as a folder**. _Do not run `cmake ..` or other commands that set up the project._
|
|
||||||
|
|
||||||
2. In the Solution Explorer, click the **Switch between solutions and available views** button.\
|
|
||||||
![image](https://github.com/user-attachments/assets/4e2be2b1-ba5a-4451-9ab2-f4ecf246213d)
|
|
||||||
|
|
||||||
3. Double-click on **CMake Targets View**.\
|
|
||||||
![image](https://github.com/user-attachments/assets/5ce7cf90-cd61-4cfa-bef5-645909827290)
|
|
||||||
|
|
||||||
4. Under **shadPS4 Project**, right-click on the **shadps4 (executable)** solution and click **Set as Startup Item**. This will let you start and debug shadPS4 using the VS debug buttons, as well as the default F5 shortcut.\
|
|
||||||
![image](https://github.com/user-attachments/assets/34c7c047-28a3-499f-be8f-df781134d104)
|
|
||||||
|
|
||||||
5. Right-click the **shadps4 (executable)** solution once more and click **Add debug configuration**.
|
|
||||||
|
|
||||||
6. Add an `"args: []"` section into the first `configurations` entry.\
|
|
||||||
List your game path as an argument, as if you were launching the non-GUI emulator from the command line.
|
|
||||||
![image](https://github.com/user-attachments/assets/8c7c3e69-f38f-4d6b-bdfd-4f1c41c50be7)
|
|
||||||
|
|
||||||
7. Set the appropriate CMake configuration for debugging or testing.
|
|
||||||
- For debugging the emulator and games within it, select `x64-Clang-Debug`.
|
|
||||||
- For testing the emulator with compiler optimizations as a release build, it is recommended to select `x64-Clang-RelWithDebInfo`,
|
|
||||||
as debug symbols will still be generated in case you encounter release configuration-exclusive bugs/errors.
|
|
||||||
![image](https://github.com/user-attachments/assets/0d975f7a-7bea-4f89-87ef-5d685bea4381)
|
|
||||||
|
|
||||||
Launch and debug the emulator through **Debug > Start Debugging** (F5 by default), or **Debug > Start Without Debugging** (Ctrl+F5 by default) when testing games for performance.
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## Configuration
|
|
||||||
|
|
||||||
You can configure the emulator by editing the `config.toml` file found in the `user` folder created after starting the application.
|
|
||||||
|
|
||||||
<details>
|
|
||||||
<summary>Some configuration entries worth changing</summary>
|
|
||||||
|
|
||||||
- `[General]`
|
|
||||||
|
|
||||||
- `logType`: Configures logging synchronization (`sync`/`async`)
|
|
||||||
- By default, the emulator logs messages asynchronously for better performance. Some log messages may end up being received out-of-order.
|
|
||||||
- It can be beneficial to set this to `sync` in order for the log to accurately maintain message order, at the cost of performance.
|
|
||||||
- When communicating about issues with games and the log messages aren't clear due to potentially confusing order, set this to `sync` and send that log as well.
|
|
||||||
- `logFilter`: Sets the logging category for various logging classes.
|
|
||||||
- Format: `<class>:<level> ...`
|
|
||||||
- Multiple classes can be set by separating them with a space. (example: `Render:Warning Debug:Critical Lib.Pad:Error`)
|
|
||||||
- Sub-classes can be specified in the same format as seen in the console/log (such as `Core.Linker`).
|
|
||||||
- All classes and sub-classes can be set by specifying a `*` symbol. (example: `Kernel.*:Critical`)
|
|
||||||
- Valid log levels: `Trace, Debug, Info, Warning, Error, Critical` - in this order, setting a level silences all levels preceding it and logs every level after it.
|
|
||||||
- Examples:
|
|
||||||
- If the log is being spammed with messages coming from Lib.Pad, you can use `Lib.Pad:Critical` to only log critical-level messages.
|
|
||||||
- If you'd like to mute everything, but still want to receive messages from Vulkan rendering: `*:Critical Render.Vulkan:Info`
|
|
||||||
|
|
||||||
- `Fullscreen`: Display the game in a full screen borderless window.
|
|
||||||
|
|
||||||
- `[GPU]`
|
|
||||||
- `dumpShaders`: Dump shaders that are loaded by the emulator. Dump path: `../user/shader/dumps`
|
|
||||||
- `nullGpu`: Disables rendering.
|
|
||||||
- `screenWidth` and `screenHeight`: Configures the game window width and height.
|
|
||||||
|
|
||||||
- `[Vulkan]`
|
|
||||||
- `validation`-related settings: Use when debugging Vulkan.
|
|
||||||
- `rdocEnable`: Automatically hook RenderDoc when installed. Useful for debugging shaders and game rendering.
|
|
||||||
- `rdocMarkersEnable`: Enable automatic RenderDoc event annotation
|
|
||||||
|
|
||||||
- `[LLE]`
|
|
||||||
- `libc`: Use LLE with `libc`.
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## Quick analysis
|
|
||||||
|
|
||||||
This section will provide some preliminary steps to take and tips on what to do when you encounter scenarios that require debugging.
|
|
||||||
|
|
||||||
<details open>
|
|
||||||
<summary>When a game crashes and breaks in the debugger</summary>
|
|
||||||
|
|
||||||
1. Analyze the log
|
|
||||||
- A console will open by default when you launch the emulator. It shows the same log messages that go into the log file found at `<emulator executable>/user/log/shad_log.txt`.
|
|
||||||
|
|
||||||
- It is recommended that you start analyzing the log bottom-up first:
|
|
||||||
- Are there any critical or error-level messages at the end of the log that would point to a reason for the game crashing?
|
|
||||||
- Do any of the last few messages contain information about the game loading files?
|
|
||||||
- Did the game window draw anything on-screen?
|
|
||||||
|
|
||||||
- Continue analyzing the log from the start to see other errors (such as with initialization, memory mapping, linker errors etc.)
|
|
||||||
|
|
||||||
2. Analyze the stack trace
|
|
||||||
- When the emulator is launched through a debugger, it will **break** when an exception or violation is encountered.\
|
|
||||||
_(**breaking** in this context means pausing execution of the program before it continues or stops altogether.
|
|
||||||
Breaks can be intentional as well - these are set with various kinds of **breakpoints**.)_
|
|
||||||
|
|
||||||
- Default setups of most debuggers include a **Stack trace** window/panel that lists the functions the program has called before breaking.
|
|
||||||
|
|
||||||
- The stack trace entries can be navigated to and will show the relevant function, as well as switch to the state that the program was in at the time of execution.\
|
|
||||||
Use the **Locals** and **Watch** windows to investigate variables and other code in these contexts.
|
|
||||||
|
|
||||||
3. Identify the reason for the crash
|
|
||||||
- **Logs aren't always accurate in determining the reason for a crash.**\
|
|
||||||
Some log entries are reported as errors but may not be fatal for the execution to stop. `Critical` entries are most likely to be the cause for crashes.
|
|
||||||
|
|
||||||
- Pinpoint the area of the emulator where the crash occured\
|
|
||||||
If the stack trace ends with functions that are relevant to rendering, it is safe to assume that the issue is with **rendering**.\
|
|
||||||
Similarly, if a crash is in a library responsible for playing videos, your issue can be narrowed down to the scope of video playback in the emulator.
|
|
||||||
|
|
||||||
- **⚠ Some crashes are intentional**
|
|
||||||
- If you identify **Access violations for writing operations** where the function is (or in cases of game libraries, _looks like_ it is) copying memory,
|
|
||||||
it most likely is an **intentional exception** meant to catch game data being written by the game.
|
|
||||||
This is used by the emulator developers to identify procedures that have to do with game data changing.
|
|
||||||
- Debugging tools usually include an option to not break on certain types of exceptions. **Exclude access violations and other intentional exceptions when debugging to skip these exceptions.**
|
|
||||||
- You can also identify such cases if the game works in Release builds of the emulator. These intentional exceptions are development-time only.
|
|
||||||
- Attempt to **Continue** and observe whether the stack trace and/or variables and registers change when you encounter exceptions.
|
|
||||||
|
|
||||||
</details>
|
|
||||||
|
|
||||||
## Reporting and communicating about issues
|
|
||||||
|
|
||||||
When communicating with the project about game-specific issues, specify an **uniquely identifable game name** along with its `CUSA-xxxxx` code that is specific to the region/variant of the game you're testing.\
|
|
||||||
The version number is also important to add at least in the description, especially if you can verify that the game behaves differently across versions.\
|
|
||||||
Accurately identifying games will help other developers that own that game recognize your issue by its title and jump in to help test and debug it.
|
|
||||||
|
|
||||||
- Examples of good naming schemes:
|
|
||||||
- Amplitude (2016) `CUSA02480`
|
|
||||||
- Rock Band 4 (`CUSA02084`) v1.0
|
|
||||||
- inFamous: Second Son \[`CUSA-00004`\]
|
|
||||||
- Examples of unideal naming schemes:
|
|
||||||
- _The Witness_
|
|
||||||
- _GTA 5_
|
|
||||||
- _Watch Dogs_
|
|
||||||
|
|
||||||
- If your issue is small or you aren't sure whether you have properly identified something, [join the Discord server](https://discord.gg/MyZRaBngxA) and use the #development channel
|
|
||||||
to concisely explain the issue, as well as any findings you currently have.
|
|
||||||
|
|
||||||
- It is recommended that you check the [game compatibility issue tracker](https://github.com/shadps4-emu/shadps4-game-compatibility/issues) and post very short summaries of progress changes there,
|
|
||||||
(such as the game now booting into the menu or getting in-game) for organizational and status update purposes.
|
|
||||||
|
|
||||||
- ⚠ **Do not post theoretical, unproven game-specific issues in the emulator issue tracker that you cannot verify and locate in the emulator source code as being a bug.**\
|
|
||||||
Do, however, add information about the game you experienced the issue in, so that it can be tested in a reproducible environment.
|
|
||||||
- Good example: "_Crash in `Shader::Gcn::CFG::EmitBlocks()`, out of bounds list access_" -> _issue description shares stack trace, points to code in the repository and provides relevant information_
|
|
||||||
- Bad example: "_Amplitude crashes on boot, access violation_" -> _issue description reiterates title, focuses on the game instead of the emulator and refuses to elaborate_
|
|
Before Width: | Height: | Size: 658 KiB |
|
@ -1,81 +0,0 @@
|
||||||
<!--
|
|
||||||
SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
|
||||||
SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
-->
|
|
||||||
|
|
||||||
# shadPS4 Quickstart
|
|
||||||
|
|
||||||
## Summary
|
|
||||||
|
|
||||||
- [PC Requirements](#pc-requirements)
|
|
||||||
- [CPU](#cpu)
|
|
||||||
- [GPU](#gpu)
|
|
||||||
- [RAM](#ram)
|
|
||||||
- [OS](#os)
|
|
||||||
- [Have the latest WIP version](#have-the-latest-wip-version)
|
|
||||||
- [Install PKG files (Games and Updates)](#install-pkg-files)
|
|
||||||
- [Configure the emulator](#configure-the-emulator)
|
|
||||||
|
|
||||||
## PC Requirements
|
|
||||||
|
|
||||||
### CPU
|
|
||||||
|
|
||||||
- A processor with at least 4 cores and 6 threads
|
|
||||||
- Above 2.5 GHz frequency
|
|
||||||
|
|
||||||
### GPU
|
|
||||||
|
|
||||||
- A graphics card with at least 1GB of VRAM
|
|
||||||
- Keep your graphics drivers up to date
|
|
||||||
- Vulkan 1.3 support (required)
|
|
||||||
|
|
||||||
### RAM
|
|
||||||
|
|
||||||
- 8GB of RAM or more
|
|
||||||
|
|
||||||
### OS
|
|
||||||
|
|
||||||
- Windows 10 or Ubuntu 22.04
|
|
||||||
|
|
||||||
## How to run the latest Work-in-Progress builds of ShadPS4
|
|
||||||
|
|
||||||
1. Go to <https://github.com/shadps4-emu/shadPS4/actions> and make sure you are logged into your GitHub account (important!)
|
|
||||||
2. On the left side of the page, select your operating system of choice (the "**qt**" versions have a user interface, which is probably the one you want. The others are SDL versions, which can only be run via command line). ![image](https://github.com/user-attachments/assets/43f01bbf-236c-4d6d-98ac-f5a5badd4ce8)
|
|
||||||
|
|
||||||
3. In the workflow list, select the latest entry with a green :white_check_mark: icon in front of it. (or the latest entry for whatever pull request you wish to test). ![image](https://github.com/user-attachments/assets/6365f407-867c-44ae-bf00-944f8d84a349)
|
|
||||||
|
|
||||||
4. On the bottom of this page, select the name of the file, and it should start downloading. (If there is no file here, double check that you are indeed logged into a GitHub account, and that there is a green :white_check_mark: icon. ![image](https://github.com/user-attachments/assets/97924500-3911-4f90-ab63-ffae7e52700b)
|
|
||||||
|
|
||||||
5. Once downloaded, extract to its own folder, and run ShadPS4's executable from the extracted folder.
|
|
||||||
|
|
||||||
6. Upon first launch, ShadPS4 will prompt you to select a folder to store your installed games in. Select "Browse" and then select a folder that ShadPS4 can use to install your PKG files to.
|
|
||||||
|
|
||||||
## Install PKG files
|
|
||||||
|
|
||||||
To install PKG files (game and updates), you will need the Qt application (with UI). You will have to go to "File" then to "Install Packages (PKG)", a window will open then you will have to select the files. You can install multiple PKG files at once. Once finished, the game should appear in the application.
|
|
||||||
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Quickstart/2.png" width="800"></a>
|
|
||||||
|
|
||||||
## Configure the emulator
|
|
||||||
|
|
||||||
You can configure the emulator by editing the `config.toml` file found in the `user` folder created after starting the application.\
|
|
||||||
Some settings may be related to more technical development and debugging. For more information on those, see [Debugging](https://github.com/shadps4-emu/shadPS4/blob/main/documents/Debugging/Debugging.md#configuration).
|
|
||||||
|
|
||||||
Here's a list of configuration entries that are worth changing:
|
|
||||||
|
|
||||||
- `[General]`
|
|
||||||
|
|
||||||
- `Fullscreen`: Display the game in a full screen borderless window.
|
|
||||||
|
|
||||||
- `logType`: Configures logging synchronization (`sync`/`async`)
|
|
||||||
- It can be beneficial to set this to `sync` in order for the log to accurately maintain message order, at the cost of performance.
|
|
||||||
- Use when sending logs to developers. See more about [reporting issues](https://github.com/shadps4-emu/shadPS4/blob/main/documents/Debugging/Debugging.md#reporting-and-communicating-about-issues).
|
|
||||||
- `logFilter`: Sets the logging category for various logging classes.
|
|
||||||
- Format: `<class>:<level> ...`, `<class.*>:<level> <*:level> ...`
|
|
||||||
- Valid log levels: `Trace, Debug, Info, Warning, Error, Critical` - in this order, setting a level silences all levels preceding it and logs every level after it.
|
|
||||||
- Examples:
|
|
||||||
- If the log is being spammed with messages coming from Lib.Pad, you can use `Lib.Pad:Critical` to only log critical-level messages.
|
|
||||||
- If you'd like to mute everything, but still want to receive messages from Vulkan rendering: `*:Error Render.Vulkan:Info`
|
|
||||||
|
|
||||||
- `[GPU]`
|
|
||||||
- `screenWidth` and `screenHeight`: Configures the game window width and height.
|
|
Before Width: | Height: | Size: 350 KiB |
Before Width: | Height: | Size: 850 KiB |
Before Width: | Height: | Size: 34 KiB |
Before Width: | Height: | Size: 175 KiB |
|
@ -1,56 +0,0 @@
|
||||||
<!--
|
|
||||||
SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
|
||||||
SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
-->
|
|
||||||
|
|
||||||
## Build shadPS4 for Linux
|
|
||||||
|
|
||||||
### Install the necessary tools to build shadPS4:
|
|
||||||
|
|
||||||
#### Debian & Ubuntu
|
|
||||||
```
|
|
||||||
sudo apt-get install build-essential libasound2-dev libpulse-dev libopenal-dev zlib1g-dev libedit-dev libvulkan-dev libudev-dev git libevdev-dev libsdl2-2.0 libsdl2-dev libjack-dev libsndio-dev
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Fedora
|
|
||||||
```
|
|
||||||
sudo dnf install alsa-lib-devel cmake libatomic libevdev-devel libudev-devel openal-devel qt6-qtbase-devel qt6-qtbase-private-devel vulkan-devel pipewire-jack-audio-connection-kit-devel qt6-qtmultimedia-devel qt6-qtsvg-devel
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Arch Linux
|
|
||||||
```
|
|
||||||
sudo pacman -S openal cmake vulkan-validation-layers qt6-base qt6-declarative qt6-multimedia sdl2 sndio jack2 base-devel
|
|
||||||
```
|
|
||||||
|
|
||||||
#### OpenSUSE
|
|
||||||
```
|
|
||||||
sudo zypper install git cmake libasound2 libpulse-devel openal-soft-devel zlib-devel libedit-devel vulkan-devel libudev-devel libqt6-qtbase-devel libqt6-qtmultimedia-devel libqt6-qtsvg-devel libQt6Gui-private-headers-devel libevdev-devel libsndio7_1 libjack-devel
|
|
||||||
```
|
|
||||||
### Cloning and compiling:
|
|
||||||
|
|
||||||
Clone the repository recursively:
|
|
||||||
```
|
|
||||||
git clone --recursive https://github.com/shadps4-emu/shadPS4.git
|
|
||||||
cd shadPS4
|
|
||||||
```
|
|
||||||
|
|
||||||
Generate the build directory in the shadPS4 directory:
|
|
||||||
```
|
|
||||||
cmake -S . -B build/
|
|
||||||
```
|
|
||||||
|
|
||||||
Enter the directory:
|
|
||||||
```
|
|
||||||
cd build/
|
|
||||||
```
|
|
||||||
|
|
||||||
Use make to build the project:
|
|
||||||
```
|
|
||||||
cmake --build . --parallel$(nproc)
|
|
||||||
```
|
|
||||||
|
|
||||||
Now run the emulator:
|
|
||||||
|
|
||||||
```
|
|
||||||
./shadps4 /"PATH"/"TO"/"GAME"/"FOLDER"/eboot.bin
|
|
||||||
```
|
|
|
@ -3,98 +3,23 @@ SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
||||||
SPDX-License-Identifier: GPL-2.0-or-later
|
SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
-->
|
-->
|
||||||
|
|
||||||
# Build shadPS4 for Windows
|
# How to build shadps4 in windows
|
||||||
|
|
||||||
This tutorial reads as if you have none of the prerequisites already installed. If you do, just ignore the steps regarding installation.
|
## Download VStudio Community 2022 17.7.4
|
||||||
If you are building to contribute to the project, please omit `--depth 1` from the git invokations.
|
|
||||||
|
|
||||||
Note: **ARM64 is not supported!** As of writing, it will not build nor run. The instructions with respect to ARM64 are for developers only.
|
[VStudio 2022](https://visualstudio.microsoft.com/vs/)
|
||||||
|
|
||||||
## Option 1: Visual Studio 2022
|
## Requirements
|
||||||
|
|
||||||
### (Prerequisite) Download the Community edition from [**Visual Studio 2022**](https://visualstudio.microsoft.com/vs/)
|
Install the following
|
||||||
|
|
||||||
Once you are within the installer:
|
- Desktop development with c++
|
||||||
1. Select `Desktop development with C++`
|
|
||||||
2. Go to "Individual Components" tab
|
|
||||||
3. Search and select `C++ Clang Compiler for Windows` and `MSBuild support for LLVM`
|
|
||||||
4. Continue the installation
|
|
||||||
|
|
||||||
### (Prerequisite) Download [**Qt**](https://doc.qt.io/qt-6/get-and-install-qt.html)
|
### From Individual components tab install
|
||||||
|
|
||||||
Beware, this requires you to create a Qt account. If you do not want to do this, please follow the MSYS2/MinGW compilation method instead.
|
- C++ Clang Compiler for Windows (16.0.5)
|
||||||
|
- MSBuild support for LLVM (clang-cl) toolset
|
||||||
|
|
||||||
1. Under the current, non beta version of Qt (at the time of writing 6.7.2), select the option `MSVC 2019 64-bit` or similar.
|
- ## Compiling
|
||||||
If you are on Windows on ARM / Qualcomm Snapdragon Elite X, select `MSVC 2019 ARM64` instead.
|
|
||||||
|
|
||||||
Go through the installation normally. If you know what you are doing, you may unselect individual components that eat up too much disk space.
|
- Open vstudio and select the clang debug or clang release . It should compile just fine
|
||||||
|
|
||||||
2. Download and install [Qt Visual Studio Tools](https://marketplace.visualstudio.com/items?itemName=TheQtCompany.QtVisualStudioTools2022)
|
|
||||||
|
|
||||||
Once you are finished, you will have to configure Qt within Visual Studio:
|
|
||||||
1. Tools -> Options -> Qt -> Versions
|
|
||||||
2. Add a new Qt version and navigate it to the correct folder. Should look like so: `C:\Qt\6.7.2\msvc2019_64`
|
|
||||||
3. Enable the default checkmark on the new version you just created.
|
|
||||||
|
|
||||||
### (Prerequisite) Download [**Git for Windows**](https://git-scm.com/download/win)
|
|
||||||
|
|
||||||
Go through the Git for Windows installation as normal
|
|
||||||
|
|
||||||
### Cloning the source code
|
|
||||||
|
|
||||||
1. Open Git for Windows, navigate to a place where you want to store the shadPS4 source code folder
|
|
||||||
2. Clone the repository by running
|
|
||||||
`git clone --depth 1 --recursive https://github.com/shadps4-emu/shadPS4`
|
|
||||||
|
|
||||||
### Compiling with Visual Studio GUI
|
|
||||||
|
|
||||||
1. Open up Visual Studio, select `Open a local folder` and select the folder with the shadPS4 source code. The folder should contain `CMakeLists.txt`
|
|
||||||
2. Change x64-Clang-Debug to x64-Clang-Release if you want a regular, non-debug build.
|
|
||||||
3. If you want to build shadPS4 with the Qt Gui:
|
|
||||||
1. Click x64-Clang-Release and select "Manage Configurations"
|
|
||||||
2. Look for "CMake command arguments" and add to the text field
|
|
||||||
`-DENABLE_QT_GUI=ON -DCMAKE_PREFIX_PATH=C:\Qt\6.7.2\msvc2019_64`
|
|
||||||
(Change Qt path if you've installed it to non-default path)
|
|
||||||
3. Press CTRL+S to save and wait a moment for CMake generation
|
|
||||||
4. Change the project to build to shadps4.exe
|
|
||||||
5. Build -> Build All
|
|
||||||
|
|
||||||
Your shadps4.exe will be in `c:\path\to\source\Build\x64-Clang-Release\`
|
|
||||||
|
|
||||||
To automatically populate the necessary files to run shadPS4.exe, run in a command prompt or terminal:
|
|
||||||
`C:\Qt\6.7.2\msvc2019_64\bin\windeployqt.exe c:\path\to\shadps4.exe`
|
|
||||||
(Change Qt path if you've installed it to non-default path)
|
|
||||||
|
|
||||||
## Option 2: MSYS2/MinGW
|
|
||||||
|
|
||||||
### (Prerequisite) Download [**MSYS2**](https://www.msys2.org/)
|
|
||||||
|
|
||||||
Go through the MSYS2 installation as normal
|
|
||||||
|
|
||||||
If you are building to distribute, please omit `-DCMAKE_CXX_FLAGS="-O2 -march=native"` within the build configuration step.
|
|
||||||
|
|
||||||
Normal x86-based computers, follow:
|
|
||||||
1. Open "MSYS2 MINGW64" from your new applications
|
|
||||||
2. Run `pacman -Syu`, let it complete;
|
|
||||||
3. Run `pacman -S --needed git mingw-w64-x86_64-binutils mingw-w64-x86_64-clang mingw-w64-x86_64-cmake mingw-w64-x86_64-ninja mingw-w64-x86_64-qt6-base`
|
|
||||||
4. Run `git clone --depth 1 --recursive https://github.com/shadps4-emu/shadPS4`
|
|
||||||
5. Run `cd shadPS4`
|
|
||||||
6. Run `cmake -S . -B build -DCMAKE_CXX_COMPILER="clang++.exe" -DCMAKE_C_COMPILER="clang.exe" -DCMAKE_CXX_FLAGS="-O2 -march=native"`
|
|
||||||
7. Run `cmake --build build`
|
|
||||||
8. To run the finished product, run `./build/shadPS4.exe`
|
|
||||||
|
|
||||||
ARM64-based computers, follow:
|
|
||||||
1. Open "MSYS2 CLANGARM64" from your new applications
|
|
||||||
2. Run `pacman -Syu`, let it complete;
|
|
||||||
3. Run `pacman -S --needed git mingw-w64-clang-aarch64-binutils mingw-w64-clang-aarch64-clang mingw-w64-clang-aarch64-cmake mingw-w64-clang-aarch64-ninja mingw-w64-clang-aarch64-qt6-base`
|
|
||||||
4. Run `git clone --depth 1 --recursive https://github.com/shadps4-emu/shadPS4`
|
|
||||||
5. Run `cd shadPS4`
|
|
||||||
6. Run `cmake -S . -B build -DCMAKE_CXX_COMPILER="clang++.exe" -DCMAKE_C_COMPILER="clang.exe" -DCMAKE_CXX_FLAGS="-O2 -march=native"`
|
|
||||||
7. Run `cmake --build build`
|
|
||||||
8. To run the finished product, run `./build/shadPS4.exe`
|
|
||||||
|
|
||||||
## Note on MSYS2 builds
|
|
||||||
|
|
||||||
These builds may not be easily copyable to people who do not also have a MSYS2 installation.
|
|
||||||
If you want to distribute these builds, you need to copy over the correct DLLs into a distribution folder.
|
|
||||||
In order to run them, you must be within the MSYS2 shell environment.
|
|
||||||
|
|
|
@ -1,44 +1,16 @@
|
||||||
v0.2.0 15/08/2024 - codename validptr
|
|
||||||
=================
|
|
||||||
- Adding macOS support
|
|
||||||
- Big shader recompiler improvements
|
|
||||||
- Core improvements
|
|
||||||
- GUI improvements
|
|
||||||
|
|
||||||
v0.1.0 01/07/2024 - codename madturtle
|
|
||||||
=================
|
|
||||||
- Added a shader recompiler, with this we have a lot of games that starts to work
|
|
||||||
- Rewrote a big part of core
|
|
||||||
|
|
||||||
v0.0.3 23/03/2024 - codename salad
|
|
||||||
=================
|
|
||||||
- Switching to std::thread
|
|
||||||
- Use unique_ptr where possible
|
|
||||||
- Replace printf/scanf with type safe fmt
|
|
||||||
- Implemented sceKernelGetProcessTime
|
|
||||||
- Implemented sceKernelGetProcessTimeCounter, sceKernelGetProcessTimeCounterFrequency
|
|
||||||
- Pause emu with P button
|
|
||||||
- Timers rewrote with std::chrono
|
|
||||||
- Added sceSystemServiceGetStatus
|
|
||||||
- Initial FileSystem implementation
|
|
||||||
- Initial TLS work
|
|
||||||
- New logging implementation
|
|
||||||
- Some functions implemented for userService, systemService
|
|
||||||
- Added sceAudioOut module and output using SDL audio
|
|
||||||
|
|
||||||
v0.0.2 21/10/2023
|
v0.0.2 21/10/2023
|
||||||
=================
|
=================
|
||||||
- Using cstdint header in variable types
|
-using cstdint header in variable types
|
||||||
-run_main_entry: Rewrite in asm for stack setup
|
-run_main_entry: Rewrite in asm for stack setup
|
||||||
- Printf libc implementation for work with sysv_abi
|
-printf libc implementation for work with sysv_abi
|
||||||
- Initial pad emulation (only digital pad atm)
|
-initial pad emulation (only digital pad atm)
|
||||||
-Implemented sceVideoOutIsFlipPending
|
-Implemented sceVideoOutIsFlipPending
|
||||||
-Added auto stubs , now unsupported hle function will resolve as empty stubs
|
-Added auto stubs , now unsupported hle function will resolve as empty stubs
|
||||||
-Rewrote libc_cxa functions
|
-Rewrote libc_cxa functions
|
||||||
-Libc implementations ( _ZdlPv,_Znwm,rand,_Fsin,qsort,free,strncpy,memmove,atan2f,pow,_Sin)
|
-Libc implementations ( _ZdlPv,_Znwm,rand,_Fsin,qsort,free,strncpy,memmove,atan2f,pow,_Sin)
|
||||||
- ET_SCE_DYNAMIC behaves as valid for execution now
|
-ET_SCE_DYNAMIC behaves as valid for execution now.
|
||||||
- Initial FileSystem work (not yet usable)
|
-Initial FileSystem work (not yet usable).
|
||||||
|
|
||||||
v0.0.1 29/09/2023
|
v0.0.1 29/09/2023
|
||||||
=================
|
=================
|
||||||
First public release. Everything is new.
|
First public release . Everything is new
|
|
@ -0,0 +1,21 @@
|
||||||
|
<!--
|
||||||
|
SPDX-FileCopyrightText: 2024 shadPS4 Emulator Project
|
||||||
|
SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
-->
|
||||||
|
|
||||||
|
## Linux
|
||||||
|
|
||||||
|
Generate the build directory in the shadPS4 directory:
|
||||||
|
```
|
||||||
|
cmake -S . -B build/
|
||||||
|
```
|
||||||
|
|
||||||
|
Enter the directory:
|
||||||
|
```
|
||||||
|
cd build/
|
||||||
|
```
|
||||||
|
|
||||||
|
Use make to build the project:
|
||||||
|
```
|
||||||
|
make -j$(nproc)
|
||||||
|
```
|
|
@ -0,0 +1,38 @@
|
||||||
|
shadps4 - An ps4 emulator
|
||||||
|
=========================
|
||||||
|
|
||||||
|
1. Intro
|
||||||
|
2. Current status
|
||||||
|
3. Contributors
|
||||||
|
4. Greetings
|
||||||
|
|
||||||
|
1.Intro
|
||||||
|
=======
|
||||||
|
shadPS4 is a ps4 emulator for windows. Although atm it can't run a lot of stuff , it can emulate perfectly videoout_basic demo from ps4 sdk. Since developing focused on that , don't expect to run anything else atm (ok maybe if you are extreme lucky , you might find something else that works)
|
||||||
|
|
||||||
|
2.Current status
|
||||||
|
================
|
||||||
|
shadps4 is a HLE emulator . Currently on a small amount of functions is emulated , which is one of the reasons compatibility is low.
|
||||||
|
|
||||||
|
|
||||||
|
3.Contributors
|
||||||
|
==============
|
||||||
|
Although atm project is mostly 1 man's work there are several commits from other devs (ironically there are all greeks)
|
||||||
|
|
||||||
|
- wheremyfoodat - (pcsx-redux dev and Panda3DS founder and dev)
|
||||||
|
- skmp - (nulldc , reincast and some many others emus founder and dev)
|
||||||
|
- OFFTKP - (paris... what to tell about paris , ok hydra is one of his best effords :) )
|
||||||
|
|
||||||
|
4.Greetings
|
||||||
|
===========
|
||||||
|
I would like to thank the following people for helping me so far , with coding or moral support.
|
||||||
|
|
||||||
|
- wheremyfoodat - or @rodakinos for believed me.
|
||||||
|
- paris - or OFFTKP for not believing me and that made me a better coder :D
|
||||||
|
- skmp - or kornilios for being good old friend
|
||||||
|
- PandaBad - our beloved stalker
|
||||||
|
- emufan4568 - for advices
|
||||||
|
- velocity - for talking 1-2 times per year on discord server . We miss you velocity
|
||||||
|
|
||||||
|
- probably more , will include in the next readme :D
|
||||||
|
|
|
@ -1,170 +1,37 @@
|
||||||
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
# SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
# SPDX-License-Identifier: GPL-2.0-or-later
|
# SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
set(BUILD_SHARED_LIBS OFF)
|
|
||||||
set(BUILD_TESTING OFF)
|
|
||||||
set_property(DIRECTORY PROPERTY EXCLUDE_FROM_ALL ON)
|
|
||||||
|
|
||||||
if (MSVC)
|
if (MSVC)
|
||||||
# Silence "deprecation" warnings
|
# Silence "deprecation" warnings
|
||||||
add_definitions(-D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS)
|
add_definitions(-D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Boost
|
# Discord-RPC
|
||||||
if (NOT TARGET Boost::headers)
|
set(BUILD_EXAMPLES OFF CACHE BOOL "")
|
||||||
set(BOOST_ROOT "${CMAKE_SOURCE_DIR}/externals/ext-boost" CACHE STRING "")
|
add_subdirectory(discord-rpc EXCLUDE_FROM_ALL)
|
||||||
set(Boost_INCLUDE_DIR "${CMAKE_SOURCE_DIR}/externals/ext-boost" CACHE STRING "")
|
target_include_directories(discord-rpc INTERFACE ./discord-rpc/include)
|
||||||
set(Boost_NO_SYSTEM_PATHS ON CACHE BOOL "")
|
|
||||||
add_library(boost INTERFACE)
|
|
||||||
target_include_directories(boost SYSTEM INTERFACE ${Boost_INCLUDE_DIR})
|
|
||||||
add_library(Boost::headers ALIAS boost)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# fmtlib
|
|
||||||
if (NOT TARGET fmt::fmt)
|
|
||||||
add_subdirectory(fmt)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND MSVC)
|
if (CMAKE_CXX_COMPILER_ID STREQUAL "Clang" AND MSVC)
|
||||||
# If it is clang and MSVC we will add a static lib
|
# If it is clang and MSVC we will add a static lib
|
||||||
# CryptoPP
|
# CryptoPP
|
||||||
add_subdirectory(cryptoppwin)
|
add_subdirectory(cryptoppwin EXCLUDE_FROM_ALL)
|
||||||
target_include_directories(cryptoppwin INTERFACE cryptoppwin/include)
|
target_include_directories(cryptoppwin INTERFACE cryptoppwin/include)
|
||||||
else()
|
|
||||||
# CryptoPP
|
|
||||||
if (NOT TARGET cryptopp::cryptopp)
|
|
||||||
set(CRYPTOPP_INSTALL OFF)
|
|
||||||
set(CRYPTOPP_BUILD_TESTING OFF)
|
|
||||||
set(CRYPTOPP_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/cryptopp/)
|
|
||||||
add_subdirectory(cryptopp-cmake)
|
|
||||||
file(COPY cryptopp DESTINATION cryptopp FILES_MATCHING PATTERN "*.h")
|
|
||||||
target_include_directories(cryptopp INTERFACE "${CMAKE_CURRENT_BINARY_DIR}/cryptopp")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (NOT TARGET FFmpeg::ffmpeg)
|
|
||||||
set(ARCHITECTURE "x86_64")
|
|
||||||
add_subdirectory(ffmpeg-core)
|
|
||||||
add_library(FFmpeg::ffmpeg ALIAS ffmpeg)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Zlib-Ng
|
# Zlib-Ng
|
||||||
if (NOT TARGET zlib-ng::zlib)
|
add_subdirectory(zlib-ng-win EXCLUDE_FROM_ALL)
|
||||||
|
target_include_directories(zlib INTERFACE zlib-ng-win/include)
|
||||||
|
else()
|
||||||
|
# CryptoPP
|
||||||
|
set(CRYPTOPP_BUILD_TESTING OFF)
|
||||||
|
set(CRYPTOPP_SOURCES ${CMAKE_CURRENT_SOURCE_DIR}/cryptopp/)
|
||||||
|
add_subdirectory(cryptopp-cmake EXCLUDE_FROM_ALL)
|
||||||
|
|
||||||
|
# Zlib-Ng
|
||||||
set(ZLIB_ENABLE_TESTS OFF)
|
set(ZLIB_ENABLE_TESTS OFF)
|
||||||
set(WITH_GTEST OFF)
|
set(WITH_GTEST OFF)
|
||||||
set(WITH_NEW_STRATEGIES ON)
|
set(WITH_NEW_STRATEGIES ON)
|
||||||
set(WITH_NATIVE_INSTRUCTIONS ON)
|
set(WITH_NATIVE_INSTRUCTIONS ON)
|
||||||
add_subdirectory(zlib-ng)
|
add_subdirectory(zlib-ng)
|
||||||
add_library(zlib-ng::zlib ALIAS zlib)
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# SDL3
|
|
||||||
if (NOT TARGET SDL3::SDL3)
|
|
||||||
set(SDL_PIPEWIRE OFF)
|
|
||||||
add_subdirectory(sdl3)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# vulkan-headers
|
|
||||||
if (NOT TARGET Vulkan::Headers)
|
|
||||||
set(VULKAN_HEADERS_ENABLE_MODULE OFF)
|
|
||||||
add_subdirectory(vulkan-headers)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# VMA
|
|
||||||
if (NOT TARGET GPUOpen::VulkanMemoryAllocator)
|
|
||||||
add_subdirectory(vma)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# RenderDoc
|
|
||||||
if (NOT TARGET RenderDoc::API)
|
|
||||||
add_library(renderdoc INTERFACE)
|
|
||||||
target_include_directories(renderdoc SYSTEM INTERFACE ./renderdoc)
|
|
||||||
add_library(RenderDoc::API ALIAS renderdoc)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# glslang
|
|
||||||
if (NOT TARGET glslang::glslang)
|
|
||||||
set(SKIP_GLSLANG_INSTALL ON CACHE BOOL "")
|
|
||||||
set(ENABLE_GLSLANG_BINARIES OFF CACHE BOOL "")
|
|
||||||
set(ENABLE_SPVREMAPPER OFF CACHE BOOL "")
|
|
||||||
set(ENABLE_CTEST OFF CACHE BOOL "")
|
|
||||||
set(ENABLE_HLSL OFF CACHE BOOL "")
|
|
||||||
set(BUILD_EXTERNAL OFF CACHE BOOL "")
|
|
||||||
set(ENABLE_OPT OFF CACHE BOOL "")
|
|
||||||
add_subdirectory(glslang)
|
|
||||||
file(COPY glslang/SPIRV DESTINATION glslang/glslang FILES_MATCHING PATTERN "*.h")
|
|
||||||
target_include_directories(SPIRV INTERFACE "${CMAKE_CURRENT_BINARY_DIR}/glslang")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Robin-map
|
|
||||||
if (NOT TARGET tsl::robin_map)
|
|
||||||
add_subdirectory(robin-map)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Xbyak
|
|
||||||
if (NOT TARGET xbyak::xbyak)
|
|
||||||
add_subdirectory(xbyak)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# MagicEnum
|
|
||||||
if (NOT TARGET magic_enum::magic_enum)
|
|
||||||
add_subdirectory(magic_enum)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Toml11
|
|
||||||
if (NOT TARGET toml11::toml11)
|
|
||||||
add_subdirectory(toml11)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# xxHash
|
|
||||||
if (NOT TARGET xxHash::xxhash)
|
|
||||||
add_library(xxhash xxhash/xxhash.h xxhash/xxhash.c)
|
|
||||||
target_include_directories(xxhash PUBLIC xxhash)
|
|
||||||
add_library(xxHash::xxhash ALIAS xxhash)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Zydis
|
|
||||||
if (NOT TARGET Zydis::Zydis)
|
|
||||||
option(ZYDIS_BUILD_TOOLS "" OFF)
|
|
||||||
option(ZYDIS_BUILD_EXAMPLES "" OFF)
|
|
||||||
add_subdirectory(zydis)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Winpthreads
|
|
||||||
if (WIN32)
|
|
||||||
add_subdirectory(winpthreads)
|
|
||||||
target_include_directories(winpthreads INTERFACE winpthreads/include)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# sirit
|
|
||||||
add_subdirectory(sirit)
|
|
||||||
if (WIN32)
|
|
||||||
target_compile_options(sirit PUBLIC "-Wno-error=unused-command-line-argument")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (APPLE)
|
|
||||||
# half
|
|
||||||
add_library(half INTERFACE)
|
|
||||||
target_include_directories(half INTERFACE half/include)
|
|
||||||
|
|
||||||
# date
|
|
||||||
if (NOT TARGET date::date-tz)
|
|
||||||
option(BUILD_TZ_LIB "" ON)
|
|
||||||
option(USE_SYSTEM_TZ_DB "" ON)
|
|
||||||
add_subdirectory(date)
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
# Tracy
|
|
||||||
option(TRACY_ENABLE "" ON)
|
|
||||||
option(TRACY_NO_CRASH_HANDLER "" ON) # Otherwise texture cache exceptions will be treaten as a crash
|
|
||||||
option(TRACY_ON_DEMAND "" ON)
|
|
||||||
option(TRACY_NO_FRAME_IMAGE "" ON)
|
|
||||||
option(TRACY_FIBERS "" ON) # For AmdGpu frontend profiling
|
|
||||||
option(TRACY_NO_SYSTEM_TRACING "" ON)
|
|
||||||
option(TRACY_NO_CALLSTACK "" ON)
|
|
||||||
option(TRACY_NO_CODE_TRANSFER "" ON)
|
|
||||||
option(TRACY_NO_SAMPLING "" ON)
|
|
||||||
option(TRACY_ONLY_LOCALHOST "" ON)
|
|
||||||
option(TRACY_NO_CONTEXT_SWITCH "" ON)
|
|
||||||
add_subdirectory(tracy)
|
|
||||||
|
|
|
@ -1,158 +0,0 @@
|
||||||
# - Returns a version string from Git
|
|
||||||
#
|
|
||||||
# These functions force a re-configure on each git commit so that you can
|
|
||||||
# trust the values of the variables in your build system.
|
|
||||||
#
|
|
||||||
# get_git_head_revision(<refspecvar> <hashvar> [<additional arguments to git describe> ...])
|
|
||||||
#
|
|
||||||
# Returns the refspec and sha hash of the current head revision
|
|
||||||
#
|
|
||||||
# git_describe(<var> [<additional arguments to git describe> ...])
|
|
||||||
#
|
|
||||||
# Returns the results of git describe on the source tree, and adjusting
|
|
||||||
# the output so that it tests false if an error occurs.
|
|
||||||
#
|
|
||||||
# git_get_exact_tag(<var> [<additional arguments to git describe> ...])
|
|
||||||
#
|
|
||||||
# Returns the results of git describe --exact-match on the source tree,
|
|
||||||
# and adjusting the output so that it tests false if there was no exact
|
|
||||||
# matching tag.
|
|
||||||
#
|
|
||||||
# Requires CMake 2.6 or newer (uses the 'function' command)
|
|
||||||
#
|
|
||||||
# Original Author:
|
|
||||||
# 2009-2010 Ryan Pavlik <rpavlik@iastate.edu> <abiryan@ryand.net>
|
|
||||||
# http://academic.cleardefinition.com
|
|
||||||
# Iowa State University HCI Graduate Program/VRAC
|
|
||||||
#
|
|
||||||
# Copyright Iowa State University 2009-2010.
|
|
||||||
# Distributed under the Boost Software License, Version 1.0.
|
|
||||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
|
||||||
# http://www.boost.org/LICENSE_1_0.txt)
|
|
||||||
|
|
||||||
if(__get_git_revision_description)
|
|
||||||
return()
|
|
||||||
endif()
|
|
||||||
set(__get_git_revision_description YES)
|
|
||||||
|
|
||||||
# We must run the following at "include" time, not at function call time,
|
|
||||||
# to find the path to this module rather than the path to a calling list file
|
|
||||||
get_filename_component(_gitdescmoddir ${CMAKE_CURRENT_LIST_FILE} PATH)
|
|
||||||
|
|
||||||
function(get_git_head_revision _refspecvar _hashvar)
|
|
||||||
set(GIT_PARENT_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
|
|
||||||
set(GIT_DIR "${GIT_PARENT_DIR}/.git")
|
|
||||||
while(NOT EXISTS "${GIT_DIR}") # .git dir not found, search parent directories
|
|
||||||
set(GIT_PREVIOUS_PARENT "${GIT_PARENT_DIR}")
|
|
||||||
get_filename_component(GIT_PARENT_DIR ${GIT_PARENT_DIR} PATH)
|
|
||||||
if(GIT_PARENT_DIR STREQUAL GIT_PREVIOUS_PARENT)
|
|
||||||
# We have reached the root directory, we are not in git
|
|
||||||
set(${_refspecvar} "GITDIR-NOTFOUND" PARENT_SCOPE)
|
|
||||||
set(${_hashvar} "GITDIR-NOTFOUND" PARENT_SCOPE)
|
|
||||||
return()
|
|
||||||
endif()
|
|
||||||
set(GIT_DIR "${GIT_PARENT_DIR}/.git")
|
|
||||||
endwhile()
|
|
||||||
# check if this is a submodule
|
|
||||||
if(NOT IS_DIRECTORY ${GIT_DIR})
|
|
||||||
file(READ ${GIT_DIR} submodule)
|
|
||||||
string(REGEX REPLACE "gitdir: (.*)\n$" "\\1" GIT_DIR_RELATIVE ${submodule})
|
|
||||||
get_filename_component(SUBMODULE_DIR ${GIT_DIR} PATH)
|
|
||||||
get_filename_component(GIT_DIR ${SUBMODULE_DIR}/${GIT_DIR_RELATIVE} ABSOLUTE)
|
|
||||||
endif()
|
|
||||||
set(GIT_DATA "${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/git-data")
|
|
||||||
if(NOT EXISTS "${GIT_DATA}")
|
|
||||||
file(MAKE_DIRECTORY "${GIT_DATA}")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT EXISTS "${GIT_DIR}/HEAD")
|
|
||||||
return()
|
|
||||||
endif()
|
|
||||||
set(HEAD_FILE "${GIT_DATA}/HEAD")
|
|
||||||
configure_file("${GIT_DIR}/HEAD" "${HEAD_FILE}" COPYONLY)
|
|
||||||
|
|
||||||
configure_file("${_gitdescmoddir}/GetGitRevisionDescription.cmake.in"
|
|
||||||
"${GIT_DATA}/grabRef.cmake"
|
|
||||||
@ONLY)
|
|
||||||
include("${GIT_DATA}/grabRef.cmake")
|
|
||||||
|
|
||||||
set(${_refspecvar} "${HEAD_REF}" PARENT_SCOPE)
|
|
||||||
set(${_hashvar} "${HEAD_HASH}" PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
function(git_branch_name _var)
|
|
||||||
if(NOT GIT_FOUND)
|
|
||||||
find_package(Git QUIET)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT GIT_FOUND)
|
|
||||||
set(${_var} "GIT-NOTFOUND" PARENT_SCOPE)
|
|
||||||
return()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
execute_process(COMMAND
|
|
||||||
"${GIT_EXECUTABLE}"
|
|
||||||
rev-parse --abbrev-ref HEAD
|
|
||||||
WORKING_DIRECTORY
|
|
||||||
"${CMAKE_SOURCE_DIR}"
|
|
||||||
RESULT_VARIABLE
|
|
||||||
res
|
|
||||||
OUTPUT_VARIABLE
|
|
||||||
out
|
|
||||||
ERROR_QUIET
|
|
||||||
OUTPUT_STRIP_TRAILING_WHITESPACE)
|
|
||||||
if(NOT res EQUAL 0)
|
|
||||||
set(out "${out}-${res}-NOTFOUND")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(${_var} "${out}" PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
function(git_describe _var)
|
|
||||||
if(NOT GIT_FOUND)
|
|
||||||
find_package(Git QUIET)
|
|
||||||
endif()
|
|
||||||
#get_git_head_revision(refspec hash)
|
|
||||||
if(NOT GIT_FOUND)
|
|
||||||
set(${_var} "GIT-NOTFOUND" PARENT_SCOPE)
|
|
||||||
return()
|
|
||||||
endif()
|
|
||||||
#if(NOT hash)
|
|
||||||
# set(${_var} "HEAD-HASH-NOTFOUND" PARENT_SCOPE)
|
|
||||||
# return()
|
|
||||||
#endif()
|
|
||||||
|
|
||||||
# TODO sanitize
|
|
||||||
#if((${ARGN}" MATCHES "&&") OR
|
|
||||||
# (ARGN MATCHES "||") OR
|
|
||||||
# (ARGN MATCHES "\\;"))
|
|
||||||
# message("Please report the following error to the project!")
|
|
||||||
# message(FATAL_ERROR "Looks like someone's doing something nefarious with git_describe! Passed arguments ${ARGN}")
|
|
||||||
#endif()
|
|
||||||
|
|
||||||
#message(STATUS "Arguments to execute_process: ${ARGN}")
|
|
||||||
|
|
||||||
execute_process(COMMAND
|
|
||||||
"${GIT_EXECUTABLE}"
|
|
||||||
describe
|
|
||||||
${hash}
|
|
||||||
${ARGN}
|
|
||||||
WORKING_DIRECTORY
|
|
||||||
"${CMAKE_SOURCE_DIR}"
|
|
||||||
RESULT_VARIABLE
|
|
||||||
res
|
|
||||||
OUTPUT_VARIABLE
|
|
||||||
out
|
|
||||||
ERROR_QUIET
|
|
||||||
OUTPUT_STRIP_TRAILING_WHITESPACE)
|
|
||||||
if(NOT res EQUAL 0)
|
|
||||||
set(out "${out}-${res}-NOTFOUND")
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(${_var} "${out}" PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
||||||
|
|
||||||
function(git_get_exact_tag _var)
|
|
||||||
git_describe(out --exact-match ${ARGN})
|
|
||||||
set(${_var} "${out}" PARENT_SCOPE)
|
|
||||||
endfunction()
|
|
|
@ -1,42 +0,0 @@
|
||||||
#
|
|
||||||
# Internal file for GetGitRevisionDescription.cmake
|
|
||||||
#
|
|
||||||
# Requires CMake 2.6 or newer (uses the 'function' command)
|
|
||||||
#
|
|
||||||
# Original Author:
|
|
||||||
# 2009-2010 Ryan Pavlik <rpavlik@iastate.edu> <abiryan@ryand.net>
|
|
||||||
# http://academic.cleardefinition.com
|
|
||||||
# Iowa State University HCI Graduate Program/VRAC
|
|
||||||
#
|
|
||||||
# Copyright Iowa State University 2009-2010.
|
|
||||||
# Distributed under the Boost Software License, Version 1.0.
|
|
||||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
|
||||||
# http://www.boost.org/LICENSE_1_0.txt)
|
|
||||||
|
|
||||||
set(HEAD_HASH)
|
|
||||||
|
|
||||||
file(READ "@HEAD_FILE@" HEAD_CONTENTS LIMIT 1024)
|
|
||||||
|
|
||||||
string(STRIP "${HEAD_CONTENTS}" HEAD_CONTENTS)
|
|
||||||
if(HEAD_CONTENTS MATCHES "ref")
|
|
||||||
# named branch
|
|
||||||
string(REPLACE "ref: " "" HEAD_REF "${HEAD_CONTENTS}")
|
|
||||||
if(EXISTS "@GIT_DIR@/${HEAD_REF}")
|
|
||||||
configure_file("@GIT_DIR@/${HEAD_REF}" "@GIT_DATA@/head-ref" COPYONLY)
|
|
||||||
elseif(EXISTS "@GIT_DIR@/logs/${HEAD_REF}")
|
|
||||||
configure_file("@GIT_DIR@/logs/${HEAD_REF}" "@GIT_DATA@/head-ref" COPYONLY)
|
|
||||||
set(HEAD_HASH "${HEAD_REF}")
|
|
||||||
endif()
|
|
||||||
else()
|
|
||||||
# detached HEAD
|
|
||||||
configure_file("@GIT_DIR@/HEAD" "@GIT_DATA@/head-ref" COPYONLY)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if(NOT HEAD_HASH)
|
|
||||||
if(EXISTS "@GIT_DATA@/head-ref")
|
|
||||||
file(READ "@GIT_DATA@/head-ref" HEAD_HASH LIMIT 1024)
|
|
||||||
string(STRIP "${HEAD_HASH}" HEAD_HASH)
|
|
||||||
else()
|
|
||||||
set(HEAD_HASH "Unknown")
|
|
||||||
endif()
|
|
||||||
endif()
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 60f81a77e0c9a0e7ffc1ca1bc438ddfa2e43b78e
|
Subproject commit 782057f5f18fbdad2bd2b291fb1ec558a8ab8225
|
|
@ -1 +1 @@
|
||||||
Subproject commit 2c384c28265a93358a2455e610e76393358794df
|
Subproject commit a99c80c26686e44eddf0432140ae397f3efbd0b3
|
|
@ -1 +1 @@
|
||||||
Subproject commit bc3441dd2d6a9728e747dc0180bc8b9065a2923c
|
Subproject commit 0dd224c6812cd038e9f4691d2f869dd57b547ff1
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 1ead6715dec030d340a316c927c877a3c4e5a00c
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 4ec218155d73bcb8022f8f7ca72305d801f84beb
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit a04136add1e469f46d8ae8d3e8307779240a5c53
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit e30b7d7fe228bfb3f6e41ce1040b44a15eb7d5e0
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit c98518351efd5a46f5d448e947e0b7242d197d07
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 12cbda959b6df2af119a76a73ff906c2bed36884
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 1ddada225144cac0de8f6b5c0dd9acffd99a2e68
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit dae6bbf16c363e9ead4e628a47fdb02956a634f3
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit a71315e7f36e312ae71e9e3a92982e9ffbfc725f
|
|
@ -1,741 +0,0 @@
|
||||||
/******************************************************************************
|
|
||||||
* The MIT License (MIT)
|
|
||||||
*
|
|
||||||
* Copyright (c) 2019-2024 Baldur Karlsson
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in
|
|
||||||
* all copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
||||||
* THE SOFTWARE.
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Documentation for the API is available at https://renderdoc.org/docs/in_application_api.html
|
|
||||||
//
|
|
||||||
|
|
||||||
#if !defined(RENDERDOC_NO_STDINT)
|
|
||||||
#include <stdint.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(WIN32) || defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER)
|
|
||||||
#define RENDERDOC_CC __cdecl
|
|
||||||
#elif defined(__linux__) || defined(__FreeBSD__)
|
|
||||||
#define RENDERDOC_CC
|
|
||||||
#elif defined(__APPLE__)
|
|
||||||
#define RENDERDOC_CC
|
|
||||||
#else
|
|
||||||
#error "Unknown platform"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
extern "C" {
|
|
||||||
#endif
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// Constants not used directly in below API
|
|
||||||
|
|
||||||
// This is a GUID/magic value used for when applications pass a path where shader debug
|
|
||||||
// information can be found to match up with a stripped shader.
|
|
||||||
// the define can be used like so: const GUID RENDERDOC_ShaderDebugMagicValue =
|
|
||||||
// RENDERDOC_ShaderDebugMagicValue_value
|
|
||||||
#define RENDERDOC_ShaderDebugMagicValue_struct \
|
|
||||||
{ \
|
|
||||||
0xeab25520, 0x6670, 0x4865, 0x84, 0x29, 0x6c, 0x8, 0x51, 0x54, 0x00, 0xff \
|
|
||||||
}
|
|
||||||
|
|
||||||
// as an alternative when you want a byte array (assuming x86 endianness):
|
|
||||||
#define RENDERDOC_ShaderDebugMagicValue_bytearray \
|
|
||||||
{ \
|
|
||||||
0x20, 0x55, 0xb2, 0xea, 0x70, 0x66, 0x65, 0x48, 0x84, 0x29, 0x6c, 0x8, 0x51, 0x54, 0x00, 0xff \
|
|
||||||
}
|
|
||||||
|
|
||||||
// truncated version when only a uint64_t is available (e.g. Vulkan tags):
|
|
||||||
#define RENDERDOC_ShaderDebugMagicValue_truncated 0x48656670eab25520ULL
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// RenderDoc capture options
|
|
||||||
//
|
|
||||||
|
|
||||||
typedef enum RENDERDOC_CaptureOption
|
|
||||||
{
|
|
||||||
// Allow the application to enable vsync
|
|
||||||
//
|
|
||||||
// Default - enabled
|
|
||||||
//
|
|
||||||
// 1 - The application can enable or disable vsync at will
|
|
||||||
// 0 - vsync is force disabled
|
|
||||||
eRENDERDOC_Option_AllowVSync = 0,
|
|
||||||
|
|
||||||
// Allow the application to enable fullscreen
|
|
||||||
//
|
|
||||||
// Default - enabled
|
|
||||||
//
|
|
||||||
// 1 - The application can enable or disable fullscreen at will
|
|
||||||
// 0 - fullscreen is force disabled
|
|
||||||
eRENDERDOC_Option_AllowFullscreen = 1,
|
|
||||||
|
|
||||||
// Record API debugging events and messages
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - Enable built-in API debugging features and records the results into
|
|
||||||
// the capture, which is matched up with events on replay
|
|
||||||
// 0 - no API debugging is forcibly enabled
|
|
||||||
eRENDERDOC_Option_APIValidation = 2,
|
|
||||||
eRENDERDOC_Option_DebugDeviceMode = 2, // deprecated name of this enum
|
|
||||||
|
|
||||||
// Capture CPU callstacks for API events
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - Enables capturing of callstacks
|
|
||||||
// 0 - no callstacks are captured
|
|
||||||
eRENDERDOC_Option_CaptureCallstacks = 3,
|
|
||||||
|
|
||||||
// When capturing CPU callstacks, only capture them from actions.
|
|
||||||
// This option does nothing without the above option being enabled
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - Only captures callstacks for actions.
|
|
||||||
// Ignored if CaptureCallstacks is disabled
|
|
||||||
// 0 - Callstacks, if enabled, are captured for every event.
|
|
||||||
eRENDERDOC_Option_CaptureCallstacksOnlyDraws = 4,
|
|
||||||
eRENDERDOC_Option_CaptureCallstacksOnlyActions = 4,
|
|
||||||
|
|
||||||
// Specify a delay in seconds to wait for a debugger to attach, after
|
|
||||||
// creating or injecting into a process, before continuing to allow it to run.
|
|
||||||
//
|
|
||||||
// 0 indicates no delay, and the process will run immediately after injection
|
|
||||||
//
|
|
||||||
// Default - 0 seconds
|
|
||||||
//
|
|
||||||
eRENDERDOC_Option_DelayForDebugger = 5,
|
|
||||||
|
|
||||||
// Verify buffer access. This includes checking the memory returned by a Map() call to
|
|
||||||
// detect any out-of-bounds modification, as well as initialising buffers with undefined contents
|
|
||||||
// to a marker value to catch use of uninitialised memory.
|
|
||||||
//
|
|
||||||
// NOTE: This option is only valid for OpenGL and D3D11. Explicit APIs such as D3D12 and Vulkan do
|
|
||||||
// not do the same kind of interception & checking and undefined contents are really undefined.
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - Verify buffer access
|
|
||||||
// 0 - No verification is performed, and overwriting bounds may cause crashes or corruption in
|
|
||||||
// RenderDoc.
|
|
||||||
eRENDERDOC_Option_VerifyBufferAccess = 6,
|
|
||||||
|
|
||||||
// The old name for eRENDERDOC_Option_VerifyBufferAccess was eRENDERDOC_Option_VerifyMapWrites.
|
|
||||||
// This option now controls the filling of uninitialised buffers with 0xdddddddd which was
|
|
||||||
// previously always enabled
|
|
||||||
eRENDERDOC_Option_VerifyMapWrites = eRENDERDOC_Option_VerifyBufferAccess,
|
|
||||||
|
|
||||||
// Hooks any system API calls that create child processes, and injects
|
|
||||||
// RenderDoc into them recursively with the same options.
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - Hooks into spawned child processes
|
|
||||||
// 0 - Child processes are not hooked by RenderDoc
|
|
||||||
eRENDERDOC_Option_HookIntoChildren = 7,
|
|
||||||
|
|
||||||
// By default RenderDoc only includes resources in the final capture necessary
|
|
||||||
// for that frame, this allows you to override that behaviour.
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - all live resources at the time of capture are included in the capture
|
|
||||||
// and available for inspection
|
|
||||||
// 0 - only the resources referenced by the captured frame are included
|
|
||||||
eRENDERDOC_Option_RefAllResources = 8,
|
|
||||||
|
|
||||||
// **NOTE**: As of RenderDoc v1.1 this option has been deprecated. Setting or
|
|
||||||
// getting it will be ignored, to allow compatibility with older versions.
|
|
||||||
// In v1.1 the option acts as if it's always enabled.
|
|
||||||
//
|
|
||||||
// By default RenderDoc skips saving initial states for resources where the
|
|
||||||
// previous contents don't appear to be used, assuming that writes before
|
|
||||||
// reads indicate previous contents aren't used.
|
|
||||||
//
|
|
||||||
// Default - disabled
|
|
||||||
//
|
|
||||||
// 1 - initial contents at the start of each captured frame are saved, even if
|
|
||||||
// they are later overwritten or cleared before being used.
|
|
||||||
// 0 - unless a read is detected, initial contents will not be saved and will
|
|
||||||
// appear as black or empty data.
|
|
||||||
eRENDERDOC_Option_SaveAllInitials = 9,
|
|
||||||
|
|
||||||
// In APIs that allow for the recording of command lists to be replayed later,
|
|
||||||
// RenderDoc may choose to not capture command lists before a frame capture is
|
|
||||||
// triggered, to reduce overheads. This means any command lists recorded once
|
|
||||||
// and replayed many times will not be available and may cause a failure to
|
|
||||||
// capture.
|
|
||||||
//
|
|
||||||
// NOTE: This is only true for APIs where multithreading is difficult or
|
|
||||||
// discouraged. Newer APIs like Vulkan and D3D12 will ignore this option
|
|
||||||
// and always capture all command lists since the API is heavily oriented
|
|
||||||
// around it and the overheads have been reduced by API design.
|
|
||||||
//
|
|
||||||
// 1 - All command lists are captured from the start of the application
|
|
||||||
// 0 - Command lists are only captured if their recording begins during
|
|
||||||
// the period when a frame capture is in progress.
|
|
||||||
eRENDERDOC_Option_CaptureAllCmdLists = 10,
|
|
||||||
|
|
||||||
// Mute API debugging output when the API validation mode option is enabled
|
|
||||||
//
|
|
||||||
// Default - enabled
|
|
||||||
//
|
|
||||||
// 1 - Mute any API debug messages from being displayed or passed through
|
|
||||||
// 0 - API debugging is displayed as normal
|
|
||||||
eRENDERDOC_Option_DebugOutputMute = 11,
|
|
||||||
|
|
||||||
// Option to allow vendor extensions to be used even when they may be
|
|
||||||
// incompatible with RenderDoc and cause corrupted replays or crashes.
|
|
||||||
//
|
|
||||||
// Default - inactive
|
|
||||||
//
|
|
||||||
// No values are documented, this option should only be used when absolutely
|
|
||||||
// necessary as directed by a RenderDoc developer.
|
|
||||||
eRENDERDOC_Option_AllowUnsupportedVendorExtensions = 12,
|
|
||||||
|
|
||||||
// Define a soft memory limit which some APIs may aim to keep overhead under where
|
|
||||||
// possible. Anything above this limit will where possible be saved directly to disk during
|
|
||||||
// capture.
|
|
||||||
// This will cause increased disk space use (which may cause a capture to fail if disk space is
|
|
||||||
// exhausted) as well as slower capture times.
|
|
||||||
//
|
|
||||||
// Not all memory allocations may be deferred like this so it is not a guarantee of a memory
|
|
||||||
// limit.
|
|
||||||
//
|
|
||||||
// Units are in MBs, suggested values would range from 200MB to 1000MB.
|
|
||||||
//
|
|
||||||
// Default - 0 Megabytes
|
|
||||||
eRENDERDOC_Option_SoftMemoryLimit = 13,
|
|
||||||
} RENDERDOC_CaptureOption;
|
|
||||||
|
|
||||||
// Sets an option that controls how RenderDoc behaves on capture.
|
|
||||||
//
|
|
||||||
// Returns 1 if the option and value are valid
|
|
||||||
// Returns 0 if either is invalid and the option is unchanged
|
|
||||||
typedef int(RENDERDOC_CC *pRENDERDOC_SetCaptureOptionU32)(RENDERDOC_CaptureOption opt, uint32_t val);
|
|
||||||
typedef int(RENDERDOC_CC *pRENDERDOC_SetCaptureOptionF32)(RENDERDOC_CaptureOption opt, float val);
|
|
||||||
|
|
||||||
// Gets the current value of an option as a uint32_t
|
|
||||||
//
|
|
||||||
// If the option is invalid, 0xffffffff is returned
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_GetCaptureOptionU32)(RENDERDOC_CaptureOption opt);
|
|
||||||
|
|
||||||
// Gets the current value of an option as a float
|
|
||||||
//
|
|
||||||
// If the option is invalid, -FLT_MAX is returned
|
|
||||||
typedef float(RENDERDOC_CC *pRENDERDOC_GetCaptureOptionF32)(RENDERDOC_CaptureOption opt);
|
|
||||||
|
|
||||||
typedef enum RENDERDOC_InputButton
|
|
||||||
{
|
|
||||||
// '0' - '9' matches ASCII values
|
|
||||||
eRENDERDOC_Key_0 = 0x30,
|
|
||||||
eRENDERDOC_Key_1 = 0x31,
|
|
||||||
eRENDERDOC_Key_2 = 0x32,
|
|
||||||
eRENDERDOC_Key_3 = 0x33,
|
|
||||||
eRENDERDOC_Key_4 = 0x34,
|
|
||||||
eRENDERDOC_Key_5 = 0x35,
|
|
||||||
eRENDERDOC_Key_6 = 0x36,
|
|
||||||
eRENDERDOC_Key_7 = 0x37,
|
|
||||||
eRENDERDOC_Key_8 = 0x38,
|
|
||||||
eRENDERDOC_Key_9 = 0x39,
|
|
||||||
|
|
||||||
// 'A' - 'Z' matches ASCII values
|
|
||||||
eRENDERDOC_Key_A = 0x41,
|
|
||||||
eRENDERDOC_Key_B = 0x42,
|
|
||||||
eRENDERDOC_Key_C = 0x43,
|
|
||||||
eRENDERDOC_Key_D = 0x44,
|
|
||||||
eRENDERDOC_Key_E = 0x45,
|
|
||||||
eRENDERDOC_Key_F = 0x46,
|
|
||||||
eRENDERDOC_Key_G = 0x47,
|
|
||||||
eRENDERDOC_Key_H = 0x48,
|
|
||||||
eRENDERDOC_Key_I = 0x49,
|
|
||||||
eRENDERDOC_Key_J = 0x4A,
|
|
||||||
eRENDERDOC_Key_K = 0x4B,
|
|
||||||
eRENDERDOC_Key_L = 0x4C,
|
|
||||||
eRENDERDOC_Key_M = 0x4D,
|
|
||||||
eRENDERDOC_Key_N = 0x4E,
|
|
||||||
eRENDERDOC_Key_O = 0x4F,
|
|
||||||
eRENDERDOC_Key_P = 0x50,
|
|
||||||
eRENDERDOC_Key_Q = 0x51,
|
|
||||||
eRENDERDOC_Key_R = 0x52,
|
|
||||||
eRENDERDOC_Key_S = 0x53,
|
|
||||||
eRENDERDOC_Key_T = 0x54,
|
|
||||||
eRENDERDOC_Key_U = 0x55,
|
|
||||||
eRENDERDOC_Key_V = 0x56,
|
|
||||||
eRENDERDOC_Key_W = 0x57,
|
|
||||||
eRENDERDOC_Key_X = 0x58,
|
|
||||||
eRENDERDOC_Key_Y = 0x59,
|
|
||||||
eRENDERDOC_Key_Z = 0x5A,
|
|
||||||
|
|
||||||
// leave the rest of the ASCII range free
|
|
||||||
// in case we want to use it later
|
|
||||||
eRENDERDOC_Key_NonPrintable = 0x100,
|
|
||||||
|
|
||||||
eRENDERDOC_Key_Divide,
|
|
||||||
eRENDERDOC_Key_Multiply,
|
|
||||||
eRENDERDOC_Key_Subtract,
|
|
||||||
eRENDERDOC_Key_Plus,
|
|
||||||
|
|
||||||
eRENDERDOC_Key_F1,
|
|
||||||
eRENDERDOC_Key_F2,
|
|
||||||
eRENDERDOC_Key_F3,
|
|
||||||
eRENDERDOC_Key_F4,
|
|
||||||
eRENDERDOC_Key_F5,
|
|
||||||
eRENDERDOC_Key_F6,
|
|
||||||
eRENDERDOC_Key_F7,
|
|
||||||
eRENDERDOC_Key_F8,
|
|
||||||
eRENDERDOC_Key_F9,
|
|
||||||
eRENDERDOC_Key_F10,
|
|
||||||
eRENDERDOC_Key_F11,
|
|
||||||
eRENDERDOC_Key_F12,
|
|
||||||
|
|
||||||
eRENDERDOC_Key_Home,
|
|
||||||
eRENDERDOC_Key_End,
|
|
||||||
eRENDERDOC_Key_Insert,
|
|
||||||
eRENDERDOC_Key_Delete,
|
|
||||||
eRENDERDOC_Key_PageUp,
|
|
||||||
eRENDERDOC_Key_PageDn,
|
|
||||||
|
|
||||||
eRENDERDOC_Key_Backspace,
|
|
||||||
eRENDERDOC_Key_Tab,
|
|
||||||
eRENDERDOC_Key_PrtScrn,
|
|
||||||
eRENDERDOC_Key_Pause,
|
|
||||||
|
|
||||||
eRENDERDOC_Key_Max,
|
|
||||||
} RENDERDOC_InputButton;
|
|
||||||
|
|
||||||
// Sets which key or keys can be used to toggle focus between multiple windows
|
|
||||||
//
|
|
||||||
// If keys is NULL or num is 0, toggle keys will be disabled
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetFocusToggleKeys)(RENDERDOC_InputButton *keys, int num);
|
|
||||||
|
|
||||||
// Sets which key or keys can be used to capture the next frame
|
|
||||||
//
|
|
||||||
// If keys is NULL or num is 0, captures keys will be disabled
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetCaptureKeys)(RENDERDOC_InputButton *keys, int num);
|
|
||||||
|
|
||||||
typedef enum RENDERDOC_OverlayBits
|
|
||||||
{
|
|
||||||
// This single bit controls whether the overlay is enabled or disabled globally
|
|
||||||
eRENDERDOC_Overlay_Enabled = 0x1,
|
|
||||||
|
|
||||||
// Show the average framerate over several seconds as well as min/max
|
|
||||||
eRENDERDOC_Overlay_FrameRate = 0x2,
|
|
||||||
|
|
||||||
// Show the current frame number
|
|
||||||
eRENDERDOC_Overlay_FrameNumber = 0x4,
|
|
||||||
|
|
||||||
// Show a list of recent captures, and how many captures have been made
|
|
||||||
eRENDERDOC_Overlay_CaptureList = 0x8,
|
|
||||||
|
|
||||||
// Default values for the overlay mask
|
|
||||||
eRENDERDOC_Overlay_Default = (eRENDERDOC_Overlay_Enabled | eRENDERDOC_Overlay_FrameRate |
|
|
||||||
eRENDERDOC_Overlay_FrameNumber | eRENDERDOC_Overlay_CaptureList),
|
|
||||||
|
|
||||||
// Enable all bits
|
|
||||||
eRENDERDOC_Overlay_All = ~0U,
|
|
||||||
|
|
||||||
// Disable all bits
|
|
||||||
eRENDERDOC_Overlay_None = 0,
|
|
||||||
} RENDERDOC_OverlayBits;
|
|
||||||
|
|
||||||
// returns the overlay bits that have been set
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_GetOverlayBits)();
|
|
||||||
// sets the overlay bits with an and & or mask
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_MaskOverlayBits)(uint32_t And, uint32_t Or);
|
|
||||||
|
|
||||||
// this function will attempt to remove RenderDoc's hooks in the application.
|
|
||||||
//
|
|
||||||
// Note: that this can only work correctly if done immediately after
|
|
||||||
// the module is loaded, before any API work happens. RenderDoc will remove its
|
|
||||||
// injected hooks and shut down. Behaviour is undefined if this is called
|
|
||||||
// after any API functions have been called, and there is still no guarantee of
|
|
||||||
// success.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_RemoveHooks)();
|
|
||||||
|
|
||||||
// DEPRECATED: compatibility for code compiled against pre-1.4.1 headers.
|
|
||||||
typedef pRENDERDOC_RemoveHooks pRENDERDOC_Shutdown;
|
|
||||||
|
|
||||||
// This function will unload RenderDoc's crash handler.
|
|
||||||
//
|
|
||||||
// If you use your own crash handler and don't want RenderDoc's handler to
|
|
||||||
// intercede, you can call this function to unload it and any unhandled
|
|
||||||
// exceptions will pass to the next handler.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_UnloadCrashHandler)();
|
|
||||||
|
|
||||||
// Sets the capture file path template
|
|
||||||
//
|
|
||||||
// pathtemplate is a UTF-8 string that gives a template for how captures will be named
|
|
||||||
// and where they will be saved.
|
|
||||||
//
|
|
||||||
// Any extension is stripped off the path, and captures are saved in the directory
|
|
||||||
// specified, and named with the filename and the frame number appended. If the
|
|
||||||
// directory does not exist it will be created, including any parent directories.
|
|
||||||
//
|
|
||||||
// If pathtemplate is NULL, the template will remain unchanged
|
|
||||||
//
|
|
||||||
// Example:
|
|
||||||
//
|
|
||||||
// SetCaptureFilePathTemplate("my_captures/example");
|
|
||||||
//
|
|
||||||
// Capture #1 -> my_captures/example_frame123.rdc
|
|
||||||
// Capture #2 -> my_captures/example_frame456.rdc
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetCaptureFilePathTemplate)(const char *pathtemplate);
|
|
||||||
|
|
||||||
// returns the current capture path template, see SetCaptureFileTemplate above, as a UTF-8 string
|
|
||||||
typedef const char *(RENDERDOC_CC *pRENDERDOC_GetCaptureFilePathTemplate)();
|
|
||||||
|
|
||||||
// DEPRECATED: compatibility for code compiled against pre-1.1.2 headers.
|
|
||||||
typedef pRENDERDOC_SetCaptureFilePathTemplate pRENDERDOC_SetLogFilePathTemplate;
|
|
||||||
typedef pRENDERDOC_GetCaptureFilePathTemplate pRENDERDOC_GetLogFilePathTemplate;
|
|
||||||
|
|
||||||
// returns the number of captures that have been made
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_GetNumCaptures)();
|
|
||||||
|
|
||||||
// This function returns the details of a capture, by index. New captures are added
|
|
||||||
// to the end of the list.
|
|
||||||
//
|
|
||||||
// filename will be filled with the absolute path to the capture file, as a UTF-8 string
|
|
||||||
// pathlength will be written with the length in bytes of the filename string
|
|
||||||
// timestamp will be written with the time of the capture, in seconds since the Unix epoch
|
|
||||||
//
|
|
||||||
// Any of the parameters can be NULL and they'll be skipped.
|
|
||||||
//
|
|
||||||
// The function will return 1 if the capture index is valid, or 0 if the index is invalid
|
|
||||||
// If the index is invalid, the values will be unchanged
|
|
||||||
//
|
|
||||||
// Note: when captures are deleted in the UI they will remain in this list, so the
|
|
||||||
// capture path may not exist anymore.
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_GetCapture)(uint32_t idx, char *filename,
|
|
||||||
uint32_t *pathlength, uint64_t *timestamp);
|
|
||||||
|
|
||||||
// Sets the comments associated with a capture file. These comments are displayed in the
|
|
||||||
// UI program when opening.
|
|
||||||
//
|
|
||||||
// filePath should be a path to the capture file to add comments to. If set to NULL or ""
|
|
||||||
// the most recent capture file created made will be used instead.
|
|
||||||
// comments should be a NULL-terminated UTF-8 string to add as comments.
|
|
||||||
//
|
|
||||||
// Any existing comments will be overwritten.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetCaptureFileComments)(const char *filePath,
|
|
||||||
const char *comments);
|
|
||||||
|
|
||||||
// returns 1 if the RenderDoc UI is connected to this application, 0 otherwise
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_IsTargetControlConnected)();
|
|
||||||
|
|
||||||
// DEPRECATED: compatibility for code compiled against pre-1.1.1 headers.
|
|
||||||
// This was renamed to IsTargetControlConnected in API 1.1.1, the old typedef is kept here for
|
|
||||||
// backwards compatibility with old code, it is castable either way since it's ABI compatible
|
|
||||||
// as the same function pointer type.
|
|
||||||
typedef pRENDERDOC_IsTargetControlConnected pRENDERDOC_IsRemoteAccessConnected;
|
|
||||||
|
|
||||||
// This function will launch the Replay UI associated with the RenderDoc library injected
|
|
||||||
// into the running application.
|
|
||||||
//
|
|
||||||
// if connectTargetControl is 1, the Replay UI will be launched with a command line parameter
|
|
||||||
// to connect to this application
|
|
||||||
// cmdline is the rest of the command line, as a UTF-8 string. E.g. a captures to open
|
|
||||||
// if cmdline is NULL, the command line will be empty.
|
|
||||||
//
|
|
||||||
// returns the PID of the replay UI if successful, 0 if not successful.
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_LaunchReplayUI)(uint32_t connectTargetControl,
|
|
||||||
const char *cmdline);
|
|
||||||
|
|
||||||
// RenderDoc can return a higher version than requested if it's backwards compatible,
|
|
||||||
// this function returns the actual version returned. If a parameter is NULL, it will be
|
|
||||||
// ignored and the others will be filled out.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_GetAPIVersion)(int *major, int *minor, int *patch);
|
|
||||||
|
|
||||||
// Requests that the replay UI show itself (if hidden or not the current top window). This can be
|
|
||||||
// used in conjunction with IsTargetControlConnected and LaunchReplayUI to intelligently handle
|
|
||||||
// showing the UI after making a capture.
|
|
||||||
//
|
|
||||||
// This will return 1 if the request was successfully passed on, though it's not guaranteed that
|
|
||||||
// the UI will be on top in all cases depending on OS rules. It will return 0 if there is no current
|
|
||||||
// target control connection to make such a request, or if there was another error
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_ShowReplayUI)();
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
// Capturing functions
|
|
||||||
//
|
|
||||||
|
|
||||||
// A device pointer is a pointer to the API's root handle.
|
|
||||||
//
|
|
||||||
// This would be an ID3D11Device, HGLRC/GLXContext, ID3D12Device, etc
|
|
||||||
typedef void *RENDERDOC_DevicePointer;
|
|
||||||
|
|
||||||
// A window handle is the OS's native window handle
|
|
||||||
//
|
|
||||||
// This would be an HWND, GLXDrawable, etc
|
|
||||||
typedef void *RENDERDOC_WindowHandle;
|
|
||||||
|
|
||||||
// A helper macro for Vulkan, where the device handle cannot be used directly.
|
|
||||||
//
|
|
||||||
// Passing the VkInstance to this macro will return the RENDERDOC_DevicePointer to use.
|
|
||||||
//
|
|
||||||
// Specifically, the value needed is the dispatch table pointer, which sits as the first
|
|
||||||
// pointer-sized object in the memory pointed to by the VkInstance. Thus we cast to a void** and
|
|
||||||
// indirect once.
|
|
||||||
#define RENDERDOC_DEVICEPOINTER_FROM_VKINSTANCE(inst) (*((void **)(inst)))
|
|
||||||
|
|
||||||
// This sets the RenderDoc in-app overlay in the API/window pair as 'active' and it will
|
|
||||||
// respond to keypresses. Neither parameter can be NULL
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetActiveWindow)(RENDERDOC_DevicePointer device,
|
|
||||||
RENDERDOC_WindowHandle wndHandle);
|
|
||||||
|
|
||||||
// capture the next frame on whichever window and API is currently considered active
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_TriggerCapture)();
|
|
||||||
|
|
||||||
// capture the next N frames on whichever window and API is currently considered active
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_TriggerMultiFrameCapture)(uint32_t numFrames);
|
|
||||||
|
|
||||||
// When choosing either a device pointer or a window handle to capture, you can pass NULL.
|
|
||||||
// Passing NULL specifies a 'wildcard' match against anything. This allows you to specify
|
|
||||||
// any API rendering to a specific window, or a specific API instance rendering to any window,
|
|
||||||
// or in the simplest case of one window and one API, you can just pass NULL for both.
|
|
||||||
//
|
|
||||||
// In either case, if there are two or more possible matching (device,window) pairs it
|
|
||||||
// is undefined which one will be captured.
|
|
||||||
//
|
|
||||||
// Note: for headless rendering you can pass NULL for the window handle and either specify
|
|
||||||
// a device pointer or leave it NULL as above.
|
|
||||||
|
|
||||||
// Immediately starts capturing API calls on the specified device pointer and window handle.
|
|
||||||
//
|
|
||||||
// If there is no matching thing to capture (e.g. no supported API has been initialised),
|
|
||||||
// this will do nothing.
|
|
||||||
//
|
|
||||||
// The results are undefined (including crashes) if two captures are started overlapping,
|
|
||||||
// even on separate devices and/oror windows.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_StartFrameCapture)(RENDERDOC_DevicePointer device,
|
|
||||||
RENDERDOC_WindowHandle wndHandle);
|
|
||||||
|
|
||||||
// Returns whether or not a frame capture is currently ongoing anywhere.
|
|
||||||
//
|
|
||||||
// This will return 1 if a capture is ongoing, and 0 if there is no capture running
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_IsFrameCapturing)();
|
|
||||||
|
|
||||||
// Ends capturing immediately.
|
|
||||||
//
|
|
||||||
// This will return 1 if the capture succeeded, and 0 if there was an error capturing.
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_EndFrameCapture)(RENDERDOC_DevicePointer device,
|
|
||||||
RENDERDOC_WindowHandle wndHandle);
|
|
||||||
|
|
||||||
// Ends capturing immediately and discard any data stored without saving to disk.
|
|
||||||
//
|
|
||||||
// This will return 1 if the capture was discarded, and 0 if there was an error or no capture
|
|
||||||
// was in progress
|
|
||||||
typedef uint32_t(RENDERDOC_CC *pRENDERDOC_DiscardFrameCapture)(RENDERDOC_DevicePointer device,
|
|
||||||
RENDERDOC_WindowHandle wndHandle);
|
|
||||||
|
|
||||||
// Only valid to be called between a call to StartFrameCapture and EndFrameCapture. Gives a custom
|
|
||||||
// title to the capture produced which will be displayed in the UI.
|
|
||||||
//
|
|
||||||
// If multiple captures are ongoing, this title will be applied to the first capture to end after
|
|
||||||
// this call. The second capture to end will have no title, unless this function is called again.
|
|
||||||
//
|
|
||||||
// Calling this function has no effect if no capture is currently running, and if it is called
|
|
||||||
// multiple times only the last title will be used.
|
|
||||||
typedef void(RENDERDOC_CC *pRENDERDOC_SetCaptureTitle)(const char *title);
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// RenderDoc API versions
|
|
||||||
//
|
|
||||||
|
|
||||||
// RenderDoc uses semantic versioning (http://semver.org/).
|
|
||||||
//
|
|
||||||
// MAJOR version is incremented when incompatible API changes happen.
|
|
||||||
// MINOR version is incremented when functionality is added in a backwards-compatible manner.
|
|
||||||
// PATCH version is incremented when backwards-compatible bug fixes happen.
|
|
||||||
//
|
|
||||||
// Note that this means the API returned can be higher than the one you might have requested.
|
|
||||||
// e.g. if you are running against a newer RenderDoc that supports 1.0.1, it will be returned
|
|
||||||
// instead of 1.0.0. You can check this with the GetAPIVersion entry point
|
|
||||||
typedef enum RENDERDOC_Version
|
|
||||||
{
|
|
||||||
eRENDERDOC_API_Version_1_0_0 = 10000, // RENDERDOC_API_1_0_0 = 1 00 00
|
|
||||||
eRENDERDOC_API_Version_1_0_1 = 10001, // RENDERDOC_API_1_0_1 = 1 00 01
|
|
||||||
eRENDERDOC_API_Version_1_0_2 = 10002, // RENDERDOC_API_1_0_2 = 1 00 02
|
|
||||||
eRENDERDOC_API_Version_1_1_0 = 10100, // RENDERDOC_API_1_1_0 = 1 01 00
|
|
||||||
eRENDERDOC_API_Version_1_1_1 = 10101, // RENDERDOC_API_1_1_1 = 1 01 01
|
|
||||||
eRENDERDOC_API_Version_1_1_2 = 10102, // RENDERDOC_API_1_1_2 = 1 01 02
|
|
||||||
eRENDERDOC_API_Version_1_2_0 = 10200, // RENDERDOC_API_1_2_0 = 1 02 00
|
|
||||||
eRENDERDOC_API_Version_1_3_0 = 10300, // RENDERDOC_API_1_3_0 = 1 03 00
|
|
||||||
eRENDERDOC_API_Version_1_4_0 = 10400, // RENDERDOC_API_1_4_0 = 1 04 00
|
|
||||||
eRENDERDOC_API_Version_1_4_1 = 10401, // RENDERDOC_API_1_4_1 = 1 04 01
|
|
||||||
eRENDERDOC_API_Version_1_4_2 = 10402, // RENDERDOC_API_1_4_2 = 1 04 02
|
|
||||||
eRENDERDOC_API_Version_1_5_0 = 10500, // RENDERDOC_API_1_5_0 = 1 05 00
|
|
||||||
eRENDERDOC_API_Version_1_6_0 = 10600, // RENDERDOC_API_1_6_0 = 1 06 00
|
|
||||||
} RENDERDOC_Version;
|
|
||||||
|
|
||||||
// API version changelog:
|
|
||||||
//
|
|
||||||
// 1.0.0 - initial release
|
|
||||||
// 1.0.1 - Bugfix: IsFrameCapturing() was returning false for captures that were triggered
|
|
||||||
// by keypress or TriggerCapture, instead of Start/EndFrameCapture.
|
|
||||||
// 1.0.2 - Refactor: Renamed eRENDERDOC_Option_DebugDeviceMode to eRENDERDOC_Option_APIValidation
|
|
||||||
// 1.1.0 - Add feature: TriggerMultiFrameCapture(). Backwards compatible with 1.0.x since the new
|
|
||||||
// function pointer is added to the end of the struct, the original layout is identical
|
|
||||||
// 1.1.1 - Refactor: Renamed remote access to target control (to better disambiguate from remote
|
|
||||||
// replay/remote server concept in replay UI)
|
|
||||||
// 1.1.2 - Refactor: Renamed "log file" in function names to just capture, to clarify that these
|
|
||||||
// are captures and not debug logging files. This is the first API version in the v1.0
|
|
||||||
// branch.
|
|
||||||
// 1.2.0 - Added feature: SetCaptureFileComments() to add comments to a capture file that will be
|
|
||||||
// displayed in the UI program on load.
|
|
||||||
// 1.3.0 - Added feature: New capture option eRENDERDOC_Option_AllowUnsupportedVendorExtensions
|
|
||||||
// which allows users to opt-in to allowing unsupported vendor extensions to function.
|
|
||||||
// Should be used at the user's own risk.
|
|
||||||
// Refactor: Renamed eRENDERDOC_Option_VerifyMapWrites to
|
|
||||||
// eRENDERDOC_Option_VerifyBufferAccess, which now also controls initialisation to
|
|
||||||
// 0xdddddddd of uninitialised buffer contents.
|
|
||||||
// 1.4.0 - Added feature: DiscardFrameCapture() to discard a frame capture in progress and stop
|
|
||||||
// capturing without saving anything to disk.
|
|
||||||
// 1.4.1 - Refactor: Renamed Shutdown to RemoveHooks to better clarify what is happening
|
|
||||||
// 1.4.2 - Refactor: Renamed 'draws' to 'actions' in callstack capture option.
|
|
||||||
// 1.5.0 - Added feature: ShowReplayUI() to request that the replay UI show itself if connected
|
|
||||||
// 1.6.0 - Added feature: SetCaptureTitle() which can be used to set a title for a
|
|
||||||
// capture made with StartFrameCapture() or EndFrameCapture()
|
|
||||||
|
|
||||||
typedef struct RENDERDOC_API_1_6_0
|
|
||||||
{
|
|
||||||
pRENDERDOC_GetAPIVersion GetAPIVersion;
|
|
||||||
|
|
||||||
pRENDERDOC_SetCaptureOptionU32 SetCaptureOptionU32;
|
|
||||||
pRENDERDOC_SetCaptureOptionF32 SetCaptureOptionF32;
|
|
||||||
|
|
||||||
pRENDERDOC_GetCaptureOptionU32 GetCaptureOptionU32;
|
|
||||||
pRENDERDOC_GetCaptureOptionF32 GetCaptureOptionF32;
|
|
||||||
|
|
||||||
pRENDERDOC_SetFocusToggleKeys SetFocusToggleKeys;
|
|
||||||
pRENDERDOC_SetCaptureKeys SetCaptureKeys;
|
|
||||||
|
|
||||||
pRENDERDOC_GetOverlayBits GetOverlayBits;
|
|
||||||
pRENDERDOC_MaskOverlayBits MaskOverlayBits;
|
|
||||||
|
|
||||||
// Shutdown was renamed to RemoveHooks in 1.4.1.
|
|
||||||
// These unions allow old code to continue compiling without changes
|
|
||||||
union
|
|
||||||
{
|
|
||||||
pRENDERDOC_Shutdown Shutdown;
|
|
||||||
pRENDERDOC_RemoveHooks RemoveHooks;
|
|
||||||
};
|
|
||||||
pRENDERDOC_UnloadCrashHandler UnloadCrashHandler;
|
|
||||||
|
|
||||||
// Get/SetLogFilePathTemplate was renamed to Get/SetCaptureFilePathTemplate in 1.1.2.
|
|
||||||
// These unions allow old code to continue compiling without changes
|
|
||||||
union
|
|
||||||
{
|
|
||||||
// deprecated name
|
|
||||||
pRENDERDOC_SetLogFilePathTemplate SetLogFilePathTemplate;
|
|
||||||
// current name
|
|
||||||
pRENDERDOC_SetCaptureFilePathTemplate SetCaptureFilePathTemplate;
|
|
||||||
};
|
|
||||||
union
|
|
||||||
{
|
|
||||||
// deprecated name
|
|
||||||
pRENDERDOC_GetLogFilePathTemplate GetLogFilePathTemplate;
|
|
||||||
// current name
|
|
||||||
pRENDERDOC_GetCaptureFilePathTemplate GetCaptureFilePathTemplate;
|
|
||||||
};
|
|
||||||
|
|
||||||
pRENDERDOC_GetNumCaptures GetNumCaptures;
|
|
||||||
pRENDERDOC_GetCapture GetCapture;
|
|
||||||
|
|
||||||
pRENDERDOC_TriggerCapture TriggerCapture;
|
|
||||||
|
|
||||||
// IsRemoteAccessConnected was renamed to IsTargetControlConnected in 1.1.1.
|
|
||||||
// This union allows old code to continue compiling without changes
|
|
||||||
union
|
|
||||||
{
|
|
||||||
// deprecated name
|
|
||||||
pRENDERDOC_IsRemoteAccessConnected IsRemoteAccessConnected;
|
|
||||||
// current name
|
|
||||||
pRENDERDOC_IsTargetControlConnected IsTargetControlConnected;
|
|
||||||
};
|
|
||||||
pRENDERDOC_LaunchReplayUI LaunchReplayUI;
|
|
||||||
|
|
||||||
pRENDERDOC_SetActiveWindow SetActiveWindow;
|
|
||||||
|
|
||||||
pRENDERDOC_StartFrameCapture StartFrameCapture;
|
|
||||||
pRENDERDOC_IsFrameCapturing IsFrameCapturing;
|
|
||||||
pRENDERDOC_EndFrameCapture EndFrameCapture;
|
|
||||||
|
|
||||||
// new function in 1.1.0
|
|
||||||
pRENDERDOC_TriggerMultiFrameCapture TriggerMultiFrameCapture;
|
|
||||||
|
|
||||||
// new function in 1.2.0
|
|
||||||
pRENDERDOC_SetCaptureFileComments SetCaptureFileComments;
|
|
||||||
|
|
||||||
// new function in 1.4.0
|
|
||||||
pRENDERDOC_DiscardFrameCapture DiscardFrameCapture;
|
|
||||||
|
|
||||||
// new function in 1.5.0
|
|
||||||
pRENDERDOC_ShowReplayUI ShowReplayUI;
|
|
||||||
|
|
||||||
// new function in 1.6.0
|
|
||||||
pRENDERDOC_SetCaptureTitle SetCaptureTitle;
|
|
||||||
} RENDERDOC_API_1_6_0;
|
|
||||||
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_0_0;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_0_1;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_0_2;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_1_0;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_1_1;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_1_2;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_2_0;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_3_0;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_4_0;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_4_1;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_4_2;
|
|
||||||
typedef RENDERDOC_API_1_6_0 RENDERDOC_API_1_5_0;
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
// RenderDoc API entry point
|
|
||||||
//
|
|
||||||
// This entry point can be obtained via GetProcAddress/dlsym if RenderDoc is available.
|
|
||||||
//
|
|
||||||
// The name is the same as the typedef - "RENDERDOC_GetAPI"
|
|
||||||
//
|
|
||||||
// This function is not thread safe, and should not be called on multiple threads at once.
|
|
||||||
// Ideally, call this once as early as possible in your application's startup, before doing
|
|
||||||
// any API work, since some configuration functionality etc has to be done also before
|
|
||||||
// initialising any APIs.
|
|
||||||
//
|
|
||||||
// Parameters:
|
|
||||||
// version is a single value from the RENDERDOC_Version above.
|
|
||||||
//
|
|
||||||
// outAPIPointers will be filled out with a pointer to the corresponding struct of function
|
|
||||||
// pointers.
|
|
||||||
//
|
|
||||||
// Returns:
|
|
||||||
// 1 - if the outAPIPointers has been filled with a pointer to the API struct requested
|
|
||||||
// 0 - if the requested version is not supported or the arguments are invalid.
|
|
||||||
//
|
|
||||||
typedef int(RENDERDOC_CC *pRENDERDOC_GetAPI)(RENDERDOC_Version version, void **outAPIPointers);
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
|
||||||
} // extern "C"
|
|
||||||
#endif
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 2c48a1a50203bbaf1e3d0d64c5d726d56f8d3bb3
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 4cc3410dce50cefce98d3cf3cf1bc8eca83b862a
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 37090c74cc6e680f2bc334cac8fd182f7634a1f6
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit 4b740127230472779c4a4d71e1a75aaa3a367a2d
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit b8061982cad0210b649541016c88ff5faa90733c
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit e1bdbca9baf4d682fb6066b380f4aa4a7bdbb58a
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit d205aff40b4e15d4c568523ee6a26f85138126d9
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit f00c973a6ab2a23573708568b8ef4acc20a9d36b
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit ccdf68421bc8eb85693f573080fc0a5faad862db
|
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit dbea33e47e7c0fe0b7c8592cd931c7430c1f130d
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit d54e3769be0c522015b784eca2af258b1c026107
|
Subproject commit 3ac71d9e2f860ac3634b43a99892255eb39179cf
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 6ad2df3e3c2e58c009f70936a8c510b939bca1d9
|
|
@ -1 +0,0 @@
|
||||||
Subproject commit bd73bc03b0aacaa89c9c203b9b43cd08f1b1843b
|
|
After Width: | Height: | Size: 810 KiB |
171520
scripts/aerolib.inl
105475
scripts/ps4_names.txt
|
@ -0,0 +1,59 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "Emulator/Host/controller.h"
|
||||||
|
#include "core/hle/libraries/libkernel/time_management.h"
|
||||||
|
|
||||||
|
namespace Emulator::Host::Controller {
|
||||||
|
|
||||||
|
GameController::GameController() {
|
||||||
|
m_states_num = 0;
|
||||||
|
m_last_state = State();
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameController::readState(State* state, bool* isConnected, int* connectedCount) {
|
||||||
|
std::scoped_lock lock{m_mutex};
|
||||||
|
|
||||||
|
*isConnected = m_connected;
|
||||||
|
*connectedCount = m_connected_count;
|
||||||
|
*state = getLastState();
|
||||||
|
}
|
||||||
|
|
||||||
|
State GameController::getLastState() const {
|
||||||
|
if (m_states_num == 0) {
|
||||||
|
return m_last_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto last = (m_first_state + m_states_num - 1) % MAX_STATES;
|
||||||
|
|
||||||
|
return m_states[last];
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameController::addState(const State& state) {
|
||||||
|
if (m_states_num >= MAX_STATES) {
|
||||||
|
m_states_num = MAX_STATES - 1;
|
||||||
|
m_first_state = (m_first_state + 1) % MAX_STATES;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto index = (m_first_state + m_states_num) % MAX_STATES;
|
||||||
|
|
||||||
|
m_states[index] = state;
|
||||||
|
m_last_state = state;
|
||||||
|
|
||||||
|
m_states_num++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void GameController::checKButton(int id, u32 button, bool isPressed) {
|
||||||
|
std::scoped_lock lock{m_mutex};
|
||||||
|
auto state = getLastState();
|
||||||
|
state.time = Core::Libraries::LibKernel::sceKernelGetProcessTime();
|
||||||
|
if (isPressed) {
|
||||||
|
state.buttonsState |= button;
|
||||||
|
} else {
|
||||||
|
state.buttonsState &= ~button;
|
||||||
|
}
|
||||||
|
|
||||||
|
addState(state);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Emulator::Host::Controller
|
|
@ -0,0 +1,37 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
#include "common/types.h"
|
||||||
|
|
||||||
|
namespace Emulator::Host::Controller {
|
||||||
|
struct State {
|
||||||
|
u32 buttonsState = 0;
|
||||||
|
u64 time = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr u32 MAX_STATES = 64;
|
||||||
|
|
||||||
|
class GameController {
|
||||||
|
public:
|
||||||
|
GameController();
|
||||||
|
virtual ~GameController() = default;
|
||||||
|
|
||||||
|
void readState(State* state, bool* isConnected, int* connectedCount);
|
||||||
|
State getLastState() const;
|
||||||
|
void checKButton(int id, u32 button, bool isPressed);
|
||||||
|
void addState(const State& state);
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::mutex m_mutex;
|
||||||
|
bool m_connected = false;
|
||||||
|
State m_last_state;
|
||||||
|
int m_connected_count = 0;
|
||||||
|
u32 m_states_num = 0;
|
||||||
|
u32 m_first_state = 0;
|
||||||
|
State m_states[MAX_STATES];
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Emulator::Host::Controller
|
|
@ -0,0 +1,95 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include <string>
|
||||||
|
#include <fmt/core.h>
|
||||||
|
#include <toml.hpp>
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
namespace Config {
|
||||||
|
|
||||||
|
bool isNeo = false;
|
||||||
|
u32 screenWidth = 1280;
|
||||||
|
u32 screenHeight = 720;
|
||||||
|
u32 logLevel = 0; // TRACE = 0 , DEBUG = 1 , INFO = 2 , WARN = 3 , ERROR = 4 , CRITICAL = 5, OFF = 6
|
||||||
|
|
||||||
|
bool isNeoMode() {
|
||||||
|
return isNeo;
|
||||||
|
}
|
||||||
|
u32 getScreenWidth() {
|
||||||
|
return screenWidth;
|
||||||
|
}
|
||||||
|
u32 getScreenHeight() {
|
||||||
|
return screenHeight;
|
||||||
|
}
|
||||||
|
u32 getLogLevel() {
|
||||||
|
return logLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
void load(const std::filesystem::path& path) {
|
||||||
|
// If the configuration file does not exist, create it and return
|
||||||
|
std::error_code error;
|
||||||
|
if (!std::filesystem::exists(path, error)) {
|
||||||
|
save(path);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toml::value data;
|
||||||
|
|
||||||
|
try {
|
||||||
|
data = toml::parse(path);
|
||||||
|
} catch (std::exception& ex) {
|
||||||
|
fmt::print("Got exception trying to load config file. Exception: {}\n", ex.what());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data.contains("General")) {
|
||||||
|
auto generalResult = toml::expect<toml::value>(data.at("General"));
|
||||||
|
if (generalResult.is_ok()) {
|
||||||
|
auto general = generalResult.unwrap();
|
||||||
|
|
||||||
|
isNeo = toml::find_or<toml::boolean>(general, "isPS4Pro", false);
|
||||||
|
logLevel = toml::find_or<toml::integer>(general, "logLevel", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (data.contains("GPU")) {
|
||||||
|
auto generalResult = toml::expect<toml::value>(data.at("GPU"));
|
||||||
|
if (generalResult.is_ok()) {
|
||||||
|
auto general = generalResult.unwrap();
|
||||||
|
|
||||||
|
screenWidth = toml::find_or<toml::integer>(general, "screenWidth", false);
|
||||||
|
screenHeight = toml::find_or<toml::integer>(general, "screenHeight", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
int k = 0;
|
||||||
|
}
|
||||||
|
void save(const std::filesystem::path& path) {
|
||||||
|
toml::basic_value<toml::preserve_comments> data;
|
||||||
|
|
||||||
|
std::error_code error;
|
||||||
|
if (std::filesystem::exists(path, error)) {
|
||||||
|
try {
|
||||||
|
data = toml::parse<toml::preserve_comments>(path);
|
||||||
|
} catch (const std::exception& ex) {
|
||||||
|
fmt::print("Exception trying to parse config file. Exception: {}\n", ex.what());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (error) {
|
||||||
|
fmt::print("Filesystem error accessing {} (error: {})\n", path.string(),
|
||||||
|
error.message().c_str());
|
||||||
|
}
|
||||||
|
fmt::print("Saving new configuration file {}\n", path.string());
|
||||||
|
}
|
||||||
|
|
||||||
|
data["General"]["isPS4Pro"] = isNeo;
|
||||||
|
data["General"]["logLevel"] = logLevel;
|
||||||
|
data["GPU"]["screenWidth"] = screenWidth;
|
||||||
|
data["GPU"]["screenHeight"] = screenHeight;
|
||||||
|
|
||||||
|
std::ofstream file(path, std::ios::out);
|
||||||
|
file << data;
|
||||||
|
file.close();
|
||||||
|
}
|
||||||
|
} // namespace Config
|
|
@ -0,0 +1,19 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <filesystem>
|
||||||
|
#include "common/types.h"
|
||||||
|
|
||||||
|
namespace Config {
|
||||||
|
void load(const std::filesystem::path& path);
|
||||||
|
void save(const std::filesystem::path& path);
|
||||||
|
|
||||||
|
bool isNeoMode();
|
||||||
|
u32 getLogLevel();
|
||||||
|
|
||||||
|
u32 getScreenWidth();
|
||||||
|
u32 getScreenHeight();
|
||||||
|
|
||||||
|
}; // namespace Config
|
|
@ -1,162 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include "sdl_audio.h"
|
|
||||||
|
|
||||||
#include "common/assert.h"
|
|
||||||
#include "core/libraries/error_codes.h"
|
|
||||||
|
|
||||||
#include <SDL3/SDL_audio.h>
|
|
||||||
#include <SDL3/SDL_init.h>
|
|
||||||
#include <SDL3/SDL_timer.h>
|
|
||||||
|
|
||||||
#include <mutex> // std::unique_lock
|
|
||||||
|
|
||||||
namespace Audio {
|
|
||||||
|
|
||||||
int SDLAudio::AudioOutOpen(int type, u32 samples_num, u32 freq,
|
|
||||||
Libraries::AudioOut::OrbisAudioOutParamFormat format) {
|
|
||||||
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
|
||||||
std::unique_lock lock{m_mutex};
|
|
||||||
for (int id = 0; id < portsOut.size(); id++) {
|
|
||||||
auto& port = portsOut[id];
|
|
||||||
if (!port.isOpen) {
|
|
||||||
port.isOpen = true;
|
|
||||||
port.type = type;
|
|
||||||
port.samples_num = samples_num;
|
|
||||||
port.freq = freq;
|
|
||||||
port.format = format;
|
|
||||||
SDL_AudioFormat sampleFormat;
|
|
||||||
switch (format) {
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_S16_MONO:
|
|
||||||
sampleFormat = SDL_AUDIO_S16;
|
|
||||||
port.channels_num = 1;
|
|
||||||
port.sample_size = 2;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_FLOAT_MONO:
|
|
||||||
sampleFormat = SDL_AUDIO_F32;
|
|
||||||
port.channels_num = 1;
|
|
||||||
port.sample_size = 4;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_S16_STEREO:
|
|
||||||
sampleFormat = SDL_AUDIO_S16;
|
|
||||||
port.channels_num = 2;
|
|
||||||
port.sample_size = 2;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_FLOAT_STEREO:
|
|
||||||
sampleFormat = SDL_AUDIO_F32;
|
|
||||||
port.channels_num = 2;
|
|
||||||
port.sample_size = 4;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_S16_8CH:
|
|
||||||
sampleFormat = SDL_AUDIO_S16;
|
|
||||||
port.channels_num = 8;
|
|
||||||
port.sample_size = 2;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_FLOAT_8CH:
|
|
||||||
sampleFormat = SDL_AUDIO_F32;
|
|
||||||
port.channels_num = 8;
|
|
||||||
port.sample_size = 4;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_S16_8CH_STD:
|
|
||||||
sampleFormat = SDL_AUDIO_S16;
|
|
||||||
port.channels_num = 8;
|
|
||||||
port.sample_size = 2;
|
|
||||||
break;
|
|
||||||
case OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_FLOAT_8CH_STD:
|
|
||||||
sampleFormat = SDL_AUDIO_F32;
|
|
||||||
port.channels_num = 8;
|
|
||||||
port.sample_size = 4;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
UNREACHABLE_MSG("Unknown format");
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < port.channels_num; i++) {
|
|
||||||
port.volume[i] = Libraries::AudioOut::SCE_AUDIO_OUT_VOLUME_0DB;
|
|
||||||
}
|
|
||||||
|
|
||||||
SDL_AudioSpec fmt;
|
|
||||||
SDL_zero(fmt);
|
|
||||||
fmt.format = sampleFormat;
|
|
||||||
fmt.channels = port.channels_num;
|
|
||||||
fmt.freq = 48000;
|
|
||||||
port.stream =
|
|
||||||
SDL_OpenAudioDeviceStream(SDL_AUDIO_DEVICE_DEFAULT_PLAYBACK, &fmt, NULL, NULL);
|
|
||||||
SDL_ResumeAudioDevice(SDL_GetAudioStreamDevice(port.stream));
|
|
||||||
return id + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1; // all ports are used
|
|
||||||
}
|
|
||||||
|
|
||||||
s32 SDLAudio::AudioOutOutput(s32 handle, const void* ptr) {
|
|
||||||
std::shared_lock lock{m_mutex};
|
|
||||||
auto& port = portsOut[handle - 1];
|
|
||||||
if (!port.isOpen) {
|
|
||||||
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
|
||||||
}
|
|
||||||
if (ptr == nullptr) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
// TODO mixing channels
|
|
||||||
int result = SDL_PutAudioStreamData(port.stream, ptr,
|
|
||||||
port.samples_num * port.sample_size * port.channels_num);
|
|
||||||
// TODO find a correct value 8192 is estimated
|
|
||||||
while (SDL_GetAudioStreamAvailable(port.stream) > 65536) {
|
|
||||||
SDL_Delay(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SDLAudio::AudioOutSetVolume(s32 handle, s32 bitflag, s32* volume) {
|
|
||||||
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
|
||||||
std::shared_lock lock{m_mutex};
|
|
||||||
auto& port = portsOut[handle - 1];
|
|
||||||
if (!port.isOpen) {
|
|
||||||
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
|
||||||
}
|
|
||||||
for (int i = 0; i < port.channels_num; i++, bitflag >>= 1u) {
|
|
||||||
auto bit = bitflag & 0x1u;
|
|
||||||
|
|
||||||
if (bit == 1) {
|
|
||||||
int src_index = i;
|
|
||||||
if (port.format ==
|
|
||||||
OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_FLOAT_8CH_STD ||
|
|
||||||
port.format == OrbisAudioOutParamFormat::ORBIS_AUDIO_OUT_PARAM_FORMAT_S16_8CH_STD) {
|
|
||||||
switch (i) {
|
|
||||||
case 4:
|
|
||||||
src_index = 6;
|
|
||||||
break;
|
|
||||||
case 5:
|
|
||||||
src_index = 7;
|
|
||||||
break;
|
|
||||||
case 6:
|
|
||||||
src_index = 4;
|
|
||||||
break;
|
|
||||||
case 7:
|
|
||||||
src_index = 5;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
port.volume[i] = volume[src_index];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SDLAudio::AudioOutGetStatus(s32 handle, int* type, int* channels_num) {
|
|
||||||
std::shared_lock lock{m_mutex};
|
|
||||||
auto& port = portsOut[handle - 1];
|
|
||||||
*type = port.type;
|
|
||||||
*channels_num = port.channels_num;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Audio
|
|
|
@ -1,40 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <shared_mutex>
|
|
||||||
#include <SDL3/SDL_audio.h>
|
|
||||||
#include "core/libraries/audio/audioout.h"
|
|
||||||
|
|
||||||
namespace Audio {
|
|
||||||
|
|
||||||
class SDLAudio {
|
|
||||||
public:
|
|
||||||
SDLAudio() = default;
|
|
||||||
virtual ~SDLAudio() = default;
|
|
||||||
|
|
||||||
int AudioOutOpen(int type, u32 samples_num, u32 freq,
|
|
||||||
Libraries::AudioOut::OrbisAudioOutParamFormat format);
|
|
||||||
s32 AudioOutOutput(s32 handle, const void* ptr);
|
|
||||||
bool AudioOutSetVolume(s32 handle, s32 bitflag, s32* volume);
|
|
||||||
bool AudioOutGetStatus(s32 handle, int* type, int* channels_num);
|
|
||||||
|
|
||||||
private:
|
|
||||||
struct PortOut {
|
|
||||||
SDL_AudioStream* stream = nullptr;
|
|
||||||
u32 samples_num = 0;
|
|
||||||
u32 freq = 0;
|
|
||||||
u32 format = -1;
|
|
||||||
int type = 0;
|
|
||||||
int channels_num = 0;
|
|
||||||
int volume[8] = {};
|
|
||||||
u8 sample_size = 0;
|
|
||||||
bool isOpen = false;
|
|
||||||
};
|
|
||||||
std::shared_mutex m_mutex;
|
|
||||||
std::array<PortOut, 22> portsOut; // main up to 8 ports , BGM 1 port , voice up to 4 ports ,
|
|
||||||
// personal up to 4 ports , padspk up to 5 ports , aux 1 port
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Audio
|
|
|
@ -1,31 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: 2014 Jannik Vogel <email@jannikvogel.de>
|
|
||||||
// SPDX-License-Identifier: CC0-1.0
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstddef>
|
|
||||||
#include <type_traits>
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
[[nodiscard]] constexpr T AlignUp(T value, std::size_t size) {
|
|
||||||
static_assert(std::is_unsigned_v<T>, "T must be an unsigned value.");
|
|
||||||
auto mod{static_cast<T>(value % size)};
|
|
||||||
value -= mod;
|
|
||||||
return static_cast<T>(mod == T{0} ? value : value + size);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
[[nodiscard]] constexpr T AlignDown(T value, std::size_t size) {
|
|
||||||
static_assert(std::is_unsigned_v<T>, "T must be an unsigned value.");
|
|
||||||
return static_cast<T>(value - value % size);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
requires std::is_integral_v<T>
|
|
||||||
[[nodiscard]] constexpr bool Is16KBAligned(T value) {
|
|
||||||
return (value & 0x3FFF) == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -1,20 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include "common/assert.h"
|
|
||||||
#include "common/logging/backend.h"
|
|
||||||
|
|
||||||
#define Crash() __asm__ __volatile__("int $3")
|
|
||||||
|
|
||||||
void assert_fail_impl() {
|
|
||||||
Common::Log::Stop();
|
|
||||||
std::fflush(stdout);
|
|
||||||
Crash();
|
|
||||||
}
|
|
||||||
|
|
||||||
[[noreturn]] void unreachable_impl() {
|
|
||||||
Common::Log::Stop();
|
|
||||||
std::fflush(stdout);
|
|
||||||
Crash();
|
|
||||||
throw std::runtime_error("Unreachable code");
|
|
||||||
}
|
|
|
@ -1,84 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: 2013 Dolphin Emulator Project
|
|
||||||
// SPDX-FileCopyrightText: 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "common/logging/log.h"
|
|
||||||
|
|
||||||
// Sometimes we want to try to continue even after hitting an assert.
|
|
||||||
// However touching this file yields a global recompilation as this header is included almost
|
|
||||||
// everywhere. So let's just move the handling of the failed assert to a single cpp file.
|
|
||||||
|
|
||||||
void assert_fail_impl();
|
|
||||||
[[noreturn]] void unreachable_impl();
|
|
||||||
|
|
||||||
#ifdef _MSC_VER
|
|
||||||
#define SHAD_NO_INLINE __declspec(noinline)
|
|
||||||
#else
|
|
||||||
#define SHAD_NO_INLINE __attribute__((noinline))
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define ASSERT(_a_) \
|
|
||||||
([&]() SHAD_NO_INLINE { \
|
|
||||||
if (!(_a_)) [[unlikely]] { \
|
|
||||||
LOG_CRITICAL(Debug, "Assertion Failed!"); \
|
|
||||||
assert_fail_impl(); \
|
|
||||||
} \
|
|
||||||
}())
|
|
||||||
|
|
||||||
#define ASSERT_MSG(_a_, ...) \
|
|
||||||
([&]() SHAD_NO_INLINE { \
|
|
||||||
if (!(_a_)) [[unlikely]] { \
|
|
||||||
LOG_CRITICAL(Debug, "Assertion Failed!\n" __VA_ARGS__); \
|
|
||||||
assert_fail_impl(); \
|
|
||||||
} \
|
|
||||||
}())
|
|
||||||
|
|
||||||
#define UNREACHABLE() \
|
|
||||||
do { \
|
|
||||||
LOG_CRITICAL(Debug, "Unreachable code!"); \
|
|
||||||
unreachable_impl(); \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define UNREACHABLE_MSG(...) \
|
|
||||||
do { \
|
|
||||||
LOG_CRITICAL(Debug, "Unreachable code!\n" __VA_ARGS__); \
|
|
||||||
unreachable_impl(); \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#ifdef _DEBUG
|
|
||||||
#define DEBUG_ASSERT(_a_) ASSERT(_a_)
|
|
||||||
#define DEBUG_ASSERT_MSG(_a_, ...) ASSERT_MSG(_a_, __VA_ARGS__)
|
|
||||||
#else // not debug
|
|
||||||
#define DEBUG_ASSERT(_a_) \
|
|
||||||
do { \
|
|
||||||
} while (0)
|
|
||||||
#define DEBUG_ASSERT_MSG(_a_, _desc_, ...) \
|
|
||||||
do { \
|
|
||||||
} while (0)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#define UNIMPLEMENTED() ASSERT_MSG(false, "Unimplemented code!")
|
|
||||||
#define UNIMPLEMENTED_MSG(...) ASSERT_MSG(false, __VA_ARGS__)
|
|
||||||
|
|
||||||
#define UNIMPLEMENTED_IF(cond) ASSERT_MSG(!(cond), "Unimplemented code!")
|
|
||||||
#define UNIMPLEMENTED_IF_MSG(cond, ...) ASSERT_MSG(!(cond), __VA_ARGS__)
|
|
||||||
|
|
||||||
// If the assert is ignored, execute _b_
|
|
||||||
#define ASSERT_OR_EXECUTE(_a_, _b_) \
|
|
||||||
do { \
|
|
||||||
ASSERT(_a_); \
|
|
||||||
if (!(_a_)) [[unlikely]] { \
|
|
||||||
_b_ \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
// If the assert is ignored, execute _b_
|
|
||||||
#define ASSERT_OR_EXECUTE_MSG(_a_, _b_, ...) \
|
|
||||||
do { \
|
|
||||||
ASSERT_MSG(_a_, __VA_ARGS__); \
|
|
||||||
if (!(_a_)) [[unlikely]] { \
|
|
||||||
_b_ \
|
|
||||||
} \
|
|
||||||
} while (0)
|
|
|
@ -1,166 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: 2014 Tony Wasserka
|
|
||||||
// SPDX-FileCopyrightText: 2014 Dolphin Emulator Project
|
|
||||||
// SPDX-License-Identifier: BSD-3-Clause AND GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstddef>
|
|
||||||
#include <limits>
|
|
||||||
#include <type_traits>
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Abstract bitfield class
|
|
||||||
*
|
|
||||||
* Allows endianness-independent access to individual bitfields within some raw
|
|
||||||
* integer value. The assembly generated by this class is identical to the
|
|
||||||
* usage of raw bitfields, so it's a perfectly fine replacement.
|
|
||||||
*
|
|
||||||
* For BitField<X,Y,Z>, X is the distance of the bitfield to the LSB of the
|
|
||||||
* raw value, Y is the length in bits of the bitfield. Z is an integer type
|
|
||||||
* which determines the sign of the bitfield. Z must have the same size as the
|
|
||||||
* raw integer.
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* General usage:
|
|
||||||
*
|
|
||||||
* Create a new union with the raw integer value as a member.
|
|
||||||
* Then for each bitfield you want to expose, add a BitField member
|
|
||||||
* in the union. The template parameters are the bit offset and the number
|
|
||||||
* of desired bits.
|
|
||||||
*
|
|
||||||
* Changes in the bitfield members will then get reflected in the raw integer
|
|
||||||
* value and vice-versa.
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* Sample usage:
|
|
||||||
*
|
|
||||||
* union SomeRegister
|
|
||||||
* {
|
|
||||||
* u32 hex;
|
|
||||||
*
|
|
||||||
* BitField<0,7,u32> first_seven_bits; // unsigned
|
|
||||||
* BitField<7,8,u32> next_eight_bits; // unsigned
|
|
||||||
* BitField<3,15,s32> some_signed_fields; // signed
|
|
||||||
* };
|
|
||||||
*
|
|
||||||
* This is equivalent to the little-endian specific code:
|
|
||||||
*
|
|
||||||
* union SomeRegister
|
|
||||||
* {
|
|
||||||
* u32 hex;
|
|
||||||
*
|
|
||||||
* struct
|
|
||||||
* {
|
|
||||||
* u32 first_seven_bits : 7;
|
|
||||||
* u32 next_eight_bits : 8;
|
|
||||||
* };
|
|
||||||
* struct
|
|
||||||
* {
|
|
||||||
* u32 : 3; // padding
|
|
||||||
* s32 some_signed_fields : 15;
|
|
||||||
* };
|
|
||||||
* };
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* Caveats:
|
|
||||||
*
|
|
||||||
* 1)
|
|
||||||
* BitField provides automatic casting from and to the storage type where
|
|
||||||
* appropriate. However, when using non-typesafe functions like printf, an
|
|
||||||
* explicit cast must be performed on the BitField object to make sure it gets
|
|
||||||
* passed correctly, e.g.:
|
|
||||||
* printf("Value: %d", (s32)some_register.some_signed_fields);
|
|
||||||
*
|
|
||||||
* 2)
|
|
||||||
* Not really a caveat, but potentially irritating: This class is used in some
|
|
||||||
* packed structures that do not guarantee proper alignment. Therefore we have
|
|
||||||
* to use #pragma pack here not to pack the members of the class, but instead
|
|
||||||
* to break GCC's assumption that the members of the class are aligned on
|
|
||||||
* sizeof(StorageType).
|
|
||||||
*/
|
|
||||||
#pragma pack(1)
|
|
||||||
template <std::size_t Position, std::size_t Bits, typename T>
|
|
||||||
struct BitField {
|
|
||||||
private:
|
|
||||||
// UnderlyingType is T for non-enum types and the underlying type of T if
|
|
||||||
// T is an enumeration. Note that T is wrapped within an enable_if in the
|
|
||||||
// former case to workaround compile errors which arise when using
|
|
||||||
// std::underlying_type<T>::type directly.
|
|
||||||
using UnderlyingType = typename std::conditional_t<std::is_enum_v<T>, std::underlying_type<T>,
|
|
||||||
std::enable_if<true, T>>::type;
|
|
||||||
|
|
||||||
// We store the value as the unsigned type to avoid undefined behaviour on value shifting
|
|
||||||
using StorageType = std::make_unsigned_t<UnderlyingType>;
|
|
||||||
|
|
||||||
public:
|
|
||||||
/// Constants to allow limited introspection of fields if needed
|
|
||||||
static constexpr std::size_t position = Position;
|
|
||||||
static constexpr std::size_t bits = Bits;
|
|
||||||
static constexpr StorageType mask = (((StorageType)~0) >> (8 * sizeof(T) - bits)) << position;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Formats a value by masking and shifting it according to the field parameters. A value
|
|
||||||
* containing several bitfields can be assembled by formatting each of their values and ORing
|
|
||||||
* the results together.
|
|
||||||
*/
|
|
||||||
[[nodiscard]] static constexpr StorageType FormatValue(const T& value) {
|
|
||||||
return (static_cast<StorageType>(value) << position) & mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extracts a value from the passed storage. In most situations prefer use the member functions
|
|
||||||
* (such as Value() or operator T), but this can be used to extract a value from a bitfield
|
|
||||||
* union in a constexpr context.
|
|
||||||
*/
|
|
||||||
[[nodiscard]] static constexpr T ExtractValue(const StorageType& storage) {
|
|
||||||
if constexpr (std::numeric_limits<UnderlyingType>::is_signed) {
|
|
||||||
std::size_t shift = 8 * sizeof(T) - bits;
|
|
||||||
return static_cast<T>(static_cast<UnderlyingType>(storage << (shift - position)) >>
|
|
||||||
shift);
|
|
||||||
} else {
|
|
||||||
return static_cast<T>((storage & mask) >> position);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// This constructor and assignment operator might be considered ambiguous:
|
|
||||||
// Would they initialize the storage or just the bitfield?
|
|
||||||
// Hence, delete them. Use the Assign method to set bitfield values!
|
|
||||||
BitField(T val) = delete;
|
|
||||||
BitField& operator=(T val) = delete;
|
|
||||||
|
|
||||||
constexpr BitField() noexcept = default;
|
|
||||||
|
|
||||||
constexpr BitField(const BitField&) noexcept = default;
|
|
||||||
constexpr BitField& operator=(const BitField&) noexcept = default;
|
|
||||||
|
|
||||||
constexpr BitField(BitField&&) noexcept = default;
|
|
||||||
constexpr BitField& operator=(BitField&&) noexcept = default;
|
|
||||||
|
|
||||||
[[nodiscard]] constexpr operator T() const {
|
|
||||||
return Value();
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr void Assign(const T& value) {
|
|
||||||
storage = (static_cast<StorageType>(storage) & ~mask) | FormatValue(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
[[nodiscard]] constexpr T Value() const {
|
|
||||||
return ExtractValue(storage);
|
|
||||||
}
|
|
||||||
|
|
||||||
[[nodiscard]] constexpr explicit operator bool() const {
|
|
||||||
return Value() != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
StorageType storage;
|
|
||||||
|
|
||||||
static_assert(bits + position <= 8 * sizeof(T), "Bitfield out of range");
|
|
||||||
|
|
||||||
// And, you know, just in case people specify something stupid like bits=position=0x80000000
|
|
||||||
static_assert(position < 8 * sizeof(T), "Invalid position");
|
|
||||||
static_assert(bits <= 8 * sizeof(T), "Invalid number of bits");
|
|
||||||
static_assert(bits > 0, "Invalid number of bits");
|
|
||||||
static_assert(std::is_trivially_copyable_v<T>, "T must be trivially copyable in a BitField");
|
|
||||||
};
|
|
||||||
#pragma pack()
|
|
|
@ -1,248 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <condition_variable>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include "common/polyfill_thread.h"
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
namespace detail {
|
|
||||||
constexpr std::size_t DefaultCapacity = 0x1000;
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
template <typename T, std::size_t Capacity = detail::DefaultCapacity>
|
|
||||||
class SPSCQueue {
|
|
||||||
static_assert((Capacity & (Capacity - 1)) == 0, "Capacity must be a power of two.");
|
|
||||||
|
|
||||||
public:
|
|
||||||
template <typename... Args>
|
|
||||||
bool TryEmplace(Args&&... args) {
|
|
||||||
return Emplace<PushMode::Try>(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
void EmplaceWait(Args&&... args) {
|
|
||||||
Emplace<PushMode::Wait>(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryPop(T& t) {
|
|
||||||
return Pop<PopMode::Try>(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t) {
|
|
||||||
Pop<PopMode::Wait>(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t, std::stop_token stop_token) {
|
|
||||||
Pop<PopMode::WaitWithStopToken>(t, stop_token);
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait() {
|
|
||||||
T t;
|
|
||||||
Pop<PopMode::Wait>(t);
|
|
||||||
return t;
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait(std::stop_token stop_token) {
|
|
||||||
T t;
|
|
||||||
Pop<PopMode::WaitWithStopToken>(t, stop_token);
|
|
||||||
return t;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
enum class PushMode {
|
|
||||||
Try,
|
|
||||||
Wait,
|
|
||||||
Count,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class PopMode {
|
|
||||||
Try,
|
|
||||||
Wait,
|
|
||||||
WaitWithStopToken,
|
|
||||||
Count,
|
|
||||||
};
|
|
||||||
|
|
||||||
template <PushMode Mode, typename... Args>
|
|
||||||
bool Emplace(Args&&... args) {
|
|
||||||
const std::size_t write_index = m_write_index.load(std::memory_order::relaxed);
|
|
||||||
|
|
||||||
if constexpr (Mode == PushMode::Try) {
|
|
||||||
// Check if we have free slots to write to.
|
|
||||||
if ((write_index - m_read_index.load(std::memory_order::acquire)) == Capacity) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else if constexpr (Mode == PushMode::Wait) {
|
|
||||||
// Wait until we have free slots to write to.
|
|
||||||
std::unique_lock lock{producer_cv_mutex};
|
|
||||||
producer_cv.wait(lock, [this, write_index] {
|
|
||||||
return (write_index - m_read_index.load(std::memory_order::acquire)) < Capacity;
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
static_assert(Mode < PushMode::Count, "Invalid PushMode.");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determine the position to write to.
|
|
||||||
const std::size_t pos = write_index % Capacity;
|
|
||||||
|
|
||||||
// Emplace into the queue.
|
|
||||||
new (std::addressof(m_data[pos])) T(std::forward<Args>(args)...);
|
|
||||||
|
|
||||||
// Increment the write index.
|
|
||||||
++m_write_index;
|
|
||||||
|
|
||||||
// Notify the consumer that we have pushed into the queue.
|
|
||||||
std::scoped_lock lock{consumer_cv_mutex};
|
|
||||||
consumer_cv.notify_one();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <PopMode Mode>
|
|
||||||
bool Pop(T& t, [[maybe_unused]] std::stop_token stop_token = {}) {
|
|
||||||
const std::size_t read_index = m_read_index.load(std::memory_order::relaxed);
|
|
||||||
|
|
||||||
if constexpr (Mode == PopMode::Try) {
|
|
||||||
// Check if the queue is empty.
|
|
||||||
if (read_index == m_write_index.load(std::memory_order::acquire)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else if constexpr (Mode == PopMode::Wait) {
|
|
||||||
// Wait until the queue is not empty.
|
|
||||||
std::unique_lock lock{consumer_cv_mutex};
|
|
||||||
consumer_cv.wait(lock, [this, read_index] {
|
|
||||||
return read_index != m_write_index.load(std::memory_order::acquire);
|
|
||||||
});
|
|
||||||
} else if constexpr (Mode == PopMode::WaitWithStopToken) {
|
|
||||||
// Wait until the queue is not empty.
|
|
||||||
std::unique_lock lock{consumer_cv_mutex};
|
|
||||||
Common::CondvarWait(consumer_cv, lock, stop_token, [this, read_index] {
|
|
||||||
return read_index != m_write_index.load(std::memory_order::acquire);
|
|
||||||
});
|
|
||||||
if (stop_token.stop_requested()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
static_assert(Mode < PopMode::Count, "Invalid PopMode.");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Determine the position to read from.
|
|
||||||
const std::size_t pos = read_index % Capacity;
|
|
||||||
|
|
||||||
// Pop the data off the queue, moving it.
|
|
||||||
t = std::move(m_data[pos]);
|
|
||||||
|
|
||||||
// Increment the read index.
|
|
||||||
++m_read_index;
|
|
||||||
|
|
||||||
// Notify the producer that we have popped off the queue.
|
|
||||||
std::scoped_lock lock{producer_cv_mutex};
|
|
||||||
producer_cv.notify_one();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
alignas(128) std::atomic_size_t m_read_index{0};
|
|
||||||
alignas(128) std::atomic_size_t m_write_index{0};
|
|
||||||
|
|
||||||
std::array<T, Capacity> m_data;
|
|
||||||
|
|
||||||
std::condition_variable_any producer_cv;
|
|
||||||
std::mutex producer_cv_mutex;
|
|
||||||
std::condition_variable_any consumer_cv;
|
|
||||||
std::mutex consumer_cv_mutex;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T, std::size_t Capacity = detail::DefaultCapacity>
|
|
||||||
class MPSCQueue {
|
|
||||||
public:
|
|
||||||
template <typename... Args>
|
|
||||||
bool TryEmplace(Args&&... args) {
|
|
||||||
std::scoped_lock lock{write_mutex};
|
|
||||||
return spsc_queue.TryEmplace(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
void EmplaceWait(Args&&... args) {
|
|
||||||
std::scoped_lock lock{write_mutex};
|
|
||||||
spsc_queue.EmplaceWait(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryPop(T& t) {
|
|
||||||
return spsc_queue.TryPop(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t) {
|
|
||||||
spsc_queue.PopWait(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t, std::stop_token stop_token) {
|
|
||||||
spsc_queue.PopWait(t, stop_token);
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait() {
|
|
||||||
return spsc_queue.PopWait();
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait(std::stop_token stop_token) {
|
|
||||||
return spsc_queue.PopWait(stop_token);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
SPSCQueue<T, Capacity> spsc_queue;
|
|
||||||
std::mutex write_mutex;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename T, std::size_t Capacity = detail::DefaultCapacity>
|
|
||||||
class MPMCQueue {
|
|
||||||
public:
|
|
||||||
template <typename... Args>
|
|
||||||
bool TryEmplace(Args&&... args) {
|
|
||||||
std::scoped_lock lock{write_mutex};
|
|
||||||
return spsc_queue.TryEmplace(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
void EmplaceWait(Args&&... args) {
|
|
||||||
std::scoped_lock lock{write_mutex};
|
|
||||||
spsc_queue.EmplaceWait(std::forward<Args>(args)...);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TryPop(T& t) {
|
|
||||||
std::scoped_lock lock{read_mutex};
|
|
||||||
return spsc_queue.TryPop(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t) {
|
|
||||||
std::scoped_lock lock{read_mutex};
|
|
||||||
spsc_queue.PopWait(t);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PopWait(T& t, std::stop_token stop_token) {
|
|
||||||
std::scoped_lock lock{read_mutex};
|
|
||||||
spsc_queue.PopWait(t, stop_token);
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait() {
|
|
||||||
std::scoped_lock lock{read_mutex};
|
|
||||||
return spsc_queue.PopWait();
|
|
||||||
}
|
|
||||||
|
|
||||||
T PopWait(std::stop_token stop_token) {
|
|
||||||
std::scoped_lock lock{read_mutex};
|
|
||||||
return spsc_queue.PopWait(stop_token);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
SPSCQueue<T, Capacity> spsc_queue;
|
|
||||||
std::mutex write_mutex;
|
|
||||||
std::mutex read_mutex;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -1,513 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include <fstream>
|
|
||||||
#include <string>
|
|
||||||
#include <fmt/core.h>
|
|
||||||
#include <toml.hpp>
|
|
||||||
#include "config.h"
|
|
||||||
|
|
||||||
namespace Config {
|
|
||||||
|
|
||||||
static bool isNeo = false;
|
|
||||||
static bool isFullscreen = false;
|
|
||||||
static u32 screenWidth = 1280;
|
|
||||||
static u32 screenHeight = 720;
|
|
||||||
static s32 gpuId = -1; // Vulkan physical device index. Set to negative for auto select
|
|
||||||
static std::string logFilter;
|
|
||||||
static std::string logType = "async";
|
|
||||||
static std::string userName = "shadPS4";
|
|
||||||
static bool useSpecialPad = false;
|
|
||||||
static int specialPadClass = 1;
|
|
||||||
static bool isDebugDump = false;
|
|
||||||
static bool isShowSplash = false;
|
|
||||||
static bool isNullGpu = false;
|
|
||||||
static bool shouldCopyGPUBuffers = false;
|
|
||||||
static bool shouldDumpShaders = false;
|
|
||||||
static bool shouldDumpPM4 = false;
|
|
||||||
static u32 vblankDivider = 1;
|
|
||||||
static bool vkValidation = false;
|
|
||||||
static bool vkValidationSync = false;
|
|
||||||
static bool vkValidationGpu = false;
|
|
||||||
static bool rdocEnable = false;
|
|
||||||
static bool rdocMarkersEnable = false;
|
|
||||||
// Gui
|
|
||||||
std::string settings_install_dir = "";
|
|
||||||
u32 main_window_geometry_x = 400;
|
|
||||||
u32 main_window_geometry_y = 400;
|
|
||||||
u32 main_window_geometry_w = 1280;
|
|
||||||
u32 main_window_geometry_h = 720;
|
|
||||||
u32 mw_themes = 0;
|
|
||||||
u32 m_icon_size = 36;
|
|
||||||
u32 m_icon_size_grid = 69;
|
|
||||||
u32 m_slider_pos = 0;
|
|
||||||
u32 m_slider_pos_grid = 0;
|
|
||||||
u32 m_table_mode = 0;
|
|
||||||
u32 m_window_size_W = 1280;
|
|
||||||
u32 m_window_size_H = 720;
|
|
||||||
std::vector<std::string> m_pkg_viewer;
|
|
||||||
std::vector<std::string> m_elf_viewer;
|
|
||||||
std::vector<std::string> m_recent_files;
|
|
||||||
std::string emulator_language = "en";
|
|
||||||
// Settings
|
|
||||||
u32 m_language = 1; // english
|
|
||||||
|
|
||||||
bool isNeoMode() {
|
|
||||||
return isNeo;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isFullscreenMode() {
|
|
||||||
return isFullscreen;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 getScreenWidth() {
|
|
||||||
return screenWidth;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 getScreenHeight() {
|
|
||||||
return screenHeight;
|
|
||||||
}
|
|
||||||
|
|
||||||
s32 getGpuId() {
|
|
||||||
return gpuId;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getLogFilter() {
|
|
||||||
return logFilter;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getLogType() {
|
|
||||||
return logType;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getUserName() {
|
|
||||||
return userName;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool getUseSpecialPad() {
|
|
||||||
return useSpecialPad;
|
|
||||||
}
|
|
||||||
|
|
||||||
int getSpecialPadClass() {
|
|
||||||
return specialPadClass;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool debugDump() {
|
|
||||||
return isDebugDump;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool showSplash() {
|
|
||||||
return isShowSplash;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool nullGpu() {
|
|
||||||
return isNullGpu;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool copyGPUCmdBuffers() {
|
|
||||||
return shouldCopyGPUBuffers;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool dumpShaders() {
|
|
||||||
return shouldDumpShaders;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool dumpPM4() {
|
|
||||||
return shouldDumpPM4;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isRdocEnabled() {
|
|
||||||
return rdocEnable;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isMarkersEnabled() {
|
|
||||||
return rdocMarkersEnable;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 vblankDiv() {
|
|
||||||
return vblankDivider;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool vkValidationEnabled() {
|
|
||||||
return vkValidation;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool vkValidationSyncEnabled() {
|
|
||||||
return vkValidationSync;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool vkValidationGpuEnabled() {
|
|
||||||
return vkValidationGpu;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setGpuId(s32 selectedGpuId) {
|
|
||||||
gpuId = selectedGpuId;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setScreenWidth(u32 width) {
|
|
||||||
screenWidth = width;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setScreenHeight(u32 height) {
|
|
||||||
screenHeight = height;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDebugDump(bool enable) {
|
|
||||||
isDebugDump = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setShowSplash(bool enable) {
|
|
||||||
isShowSplash = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setNullGpu(bool enable) {
|
|
||||||
isNullGpu = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setCopyGPUCmdBuffers(bool enable) {
|
|
||||||
shouldCopyGPUBuffers = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDumpShaders(bool enable) {
|
|
||||||
shouldDumpShaders = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDumpPM4(bool enable) {
|
|
||||||
shouldDumpPM4 = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setVkValidation(bool enable) {
|
|
||||||
vkValidation = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setVkSyncValidation(bool enable) {
|
|
||||||
vkValidationSync = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setRdocEnabled(bool enable) {
|
|
||||||
rdocEnable = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setVblankDiv(u32 value) {
|
|
||||||
vblankDivider = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setFullscreenMode(bool enable) {
|
|
||||||
isFullscreen = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setLanguage(u32 language) {
|
|
||||||
m_language = language;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setNeoMode(bool enable) {
|
|
||||||
isNeo = enable;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setLogType(const std::string& type) {
|
|
||||||
logType = type;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setLogFilter(const std::string& type) {
|
|
||||||
logFilter = type;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setUserName(const std::string& type) {
|
|
||||||
userName = type;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setUseSpecialPad(bool use) {
|
|
||||||
useSpecialPad = use;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setSpecialPadClass(int type) {
|
|
||||||
specialPadClass = type;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setMainWindowGeometry(u32 x, u32 y, u32 w, u32 h) {
|
|
||||||
main_window_geometry_x = x;
|
|
||||||
main_window_geometry_y = y;
|
|
||||||
main_window_geometry_w = w;
|
|
||||||
main_window_geometry_h = h;
|
|
||||||
}
|
|
||||||
void setGameInstallDir(const std::string& dir) {
|
|
||||||
settings_install_dir = dir;
|
|
||||||
}
|
|
||||||
void setMainWindowTheme(u32 theme) {
|
|
||||||
mw_themes = theme;
|
|
||||||
}
|
|
||||||
void setIconSize(u32 size) {
|
|
||||||
m_icon_size = size;
|
|
||||||
}
|
|
||||||
void setIconSizeGrid(u32 size) {
|
|
||||||
m_icon_size_grid = size;
|
|
||||||
}
|
|
||||||
void setSliderPosition(u32 pos) {
|
|
||||||
m_slider_pos = pos;
|
|
||||||
}
|
|
||||||
void setSliderPositionGrid(u32 pos) {
|
|
||||||
m_slider_pos_grid = pos;
|
|
||||||
}
|
|
||||||
void setTableMode(u32 mode) {
|
|
||||||
m_table_mode = mode;
|
|
||||||
}
|
|
||||||
void setMainWindowWidth(u32 width) {
|
|
||||||
m_window_size_W = width;
|
|
||||||
}
|
|
||||||
void setMainWindowHeight(u32 height) {
|
|
||||||
m_window_size_H = height;
|
|
||||||
}
|
|
||||||
void setPkgViewer(const std::vector<std::string>& pkgList) {
|
|
||||||
m_pkg_viewer.resize(pkgList.size());
|
|
||||||
m_pkg_viewer = pkgList;
|
|
||||||
}
|
|
||||||
void setElfViewer(const std::vector<std::string>& elfList) {
|
|
||||||
m_elf_viewer.resize(elfList.size());
|
|
||||||
m_elf_viewer = elfList;
|
|
||||||
}
|
|
||||||
void setRecentFiles(const std::vector<std::string>& recentFiles) {
|
|
||||||
m_recent_files.resize(recentFiles.size());
|
|
||||||
m_recent_files = recentFiles;
|
|
||||||
}
|
|
||||||
|
|
||||||
void setEmulatorLanguage(std::string language) {
|
|
||||||
emulator_language = language;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 getMainWindowGeometryX() {
|
|
||||||
return main_window_geometry_x;
|
|
||||||
}
|
|
||||||
u32 getMainWindowGeometryY() {
|
|
||||||
return main_window_geometry_y;
|
|
||||||
}
|
|
||||||
u32 getMainWindowGeometryW() {
|
|
||||||
return main_window_geometry_w;
|
|
||||||
}
|
|
||||||
u32 getMainWindowGeometryH() {
|
|
||||||
return main_window_geometry_h;
|
|
||||||
}
|
|
||||||
std::string getGameInstallDir() {
|
|
||||||
return settings_install_dir;
|
|
||||||
}
|
|
||||||
u32 getMainWindowTheme() {
|
|
||||||
return mw_themes;
|
|
||||||
}
|
|
||||||
u32 getIconSize() {
|
|
||||||
return m_icon_size;
|
|
||||||
}
|
|
||||||
u32 getIconSizeGrid() {
|
|
||||||
return m_icon_size_grid;
|
|
||||||
}
|
|
||||||
u32 getSliderPosition() {
|
|
||||||
return m_slider_pos;
|
|
||||||
}
|
|
||||||
u32 getSliderPositionGrid() {
|
|
||||||
return m_slider_pos_grid;
|
|
||||||
}
|
|
||||||
u32 getTableMode() {
|
|
||||||
return m_table_mode;
|
|
||||||
}
|
|
||||||
u32 getMainWindowWidth() {
|
|
||||||
return m_window_size_W;
|
|
||||||
}
|
|
||||||
u32 getMainWindowHeight() {
|
|
||||||
return m_window_size_H;
|
|
||||||
}
|
|
||||||
std::vector<std::string> getPkgViewer() {
|
|
||||||
return m_pkg_viewer;
|
|
||||||
}
|
|
||||||
std::vector<std::string> getElfViewer() {
|
|
||||||
return m_elf_viewer;
|
|
||||||
}
|
|
||||||
std::vector<std::string> getRecentFiles() {
|
|
||||||
return m_recent_files;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string getEmulatorLanguage() {
|
|
||||||
return emulator_language;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 GetLanguage() {
|
|
||||||
return m_language;
|
|
||||||
}
|
|
||||||
void load(const std::filesystem::path& path) {
|
|
||||||
// If the configuration file does not exist, create it and return
|
|
||||||
std::error_code error;
|
|
||||||
if (!std::filesystem::exists(path, error)) {
|
|
||||||
save(path);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
toml::value data;
|
|
||||||
|
|
||||||
try {
|
|
||||||
data = toml::parse(path);
|
|
||||||
} catch (std::exception& ex) {
|
|
||||||
fmt::print("Got exception trying to load config file. Exception: {}\n", ex.what());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (data.contains("General")) {
|
|
||||||
const toml::value& general = data.at("General");
|
|
||||||
|
|
||||||
isNeo = toml::find_or<bool>(general, "isPS4Pro", false);
|
|
||||||
isFullscreen = toml::find_or<bool>(general, "Fullscreen", false);
|
|
||||||
logFilter = toml::find_or<std::string>(general, "logFilter", "");
|
|
||||||
logType = toml::find_or<std::string>(general, "logType", "sync");
|
|
||||||
userName = toml::find_or<std::string>(general, "userName", "shadPS4");
|
|
||||||
isShowSplash = toml::find_or<bool>(general, "showSplash", true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("Input")) {
|
|
||||||
const toml::value& input = data.at("Input");
|
|
||||||
|
|
||||||
useSpecialPad = toml::find_or<bool>(input, "useSpecialPad", false);
|
|
||||||
specialPadClass = toml::find_or<int>(input, "specialPadClass", 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("GPU")) {
|
|
||||||
const toml::value& gpu = data.at("GPU");
|
|
||||||
|
|
||||||
screenWidth = toml::find_or<int>(gpu, "screenWidth", screenWidth);
|
|
||||||
screenHeight = toml::find_or<int>(gpu, "screenHeight", screenHeight);
|
|
||||||
isNullGpu = toml::find_or<bool>(gpu, "nullGpu", false);
|
|
||||||
shouldCopyGPUBuffers = toml::find_or<bool>(gpu, "copyGPUBuffers", false);
|
|
||||||
shouldDumpShaders = toml::find_or<bool>(gpu, "dumpShaders", false);
|
|
||||||
shouldDumpPM4 = toml::find_or<bool>(gpu, "dumpPM4", false);
|
|
||||||
vblankDivider = toml::find_or<int>(gpu, "vblankDivider", 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("Vulkan")) {
|
|
||||||
const toml::value& vk = data.at("Vulkan");
|
|
||||||
|
|
||||||
gpuId = toml::find_or<int>(vk, "gpuId", -1);
|
|
||||||
vkValidation = toml::find_or<bool>(vk, "validation", false);
|
|
||||||
vkValidationSync = toml::find_or<bool>(vk, "validation_sync", false);
|
|
||||||
vkValidationGpu = toml::find_or<bool>(vk, "validation_gpu", true);
|
|
||||||
rdocEnable = toml::find_or<bool>(vk, "rdocEnable", false);
|
|
||||||
rdocMarkersEnable = toml::find_or<bool>(vk, "rdocMarkersEnable", false);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("Debug")) {
|
|
||||||
const toml::value& debug = data.at("Debug");
|
|
||||||
|
|
||||||
isDebugDump = toml::find_or<bool>(debug, "DebugDump", false);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("GUI")) {
|
|
||||||
const toml::value& gui = data.at("GUI");
|
|
||||||
|
|
||||||
m_icon_size = toml::find_or<int>(gui, "iconSize", 0);
|
|
||||||
m_icon_size_grid = toml::find_or<int>(gui, "iconSizeGrid", 0);
|
|
||||||
m_slider_pos = toml::find_or<int>(gui, "sliderPos", 0);
|
|
||||||
m_slider_pos_grid = toml::find_or<int>(gui, "sliderPosGrid", 0);
|
|
||||||
mw_themes = toml::find_or<int>(gui, "theme", 0);
|
|
||||||
m_window_size_W = toml::find_or<int>(gui, "mw_width", 0);
|
|
||||||
m_window_size_H = toml::find_or<int>(gui, "mw_height", 0);
|
|
||||||
settings_install_dir = toml::find_or<std::string>(gui, "installDir", "");
|
|
||||||
main_window_geometry_x = toml::find_or<int>(gui, "geometry_x", 0);
|
|
||||||
main_window_geometry_y = toml::find_or<int>(gui, "geometry_y", 0);
|
|
||||||
main_window_geometry_w = toml::find_or<int>(gui, "geometry_w", 0);
|
|
||||||
main_window_geometry_h = toml::find_or<int>(gui, "geometry_h", 0);
|
|
||||||
m_pkg_viewer = toml::find_or<std::vector<std::string>>(gui, "pkgDirs", {});
|
|
||||||
m_elf_viewer = toml::find_or<std::vector<std::string>>(gui, "elfDirs", {});
|
|
||||||
m_recent_files = toml::find_or<std::vector<std::string>>(gui, "recentFiles", {});
|
|
||||||
m_table_mode = toml::find_or<int>(gui, "gameTableMode", 0);
|
|
||||||
emulator_language = toml::find_or<std::string>(gui, "emulatorLanguage", "en");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (data.contains("Settings")) {
|
|
||||||
const toml::value& settings = data.at("Settings");
|
|
||||||
|
|
||||||
m_language = toml::find_or<int>(settings, "consoleLanguage", 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void save(const std::filesystem::path& path) {
|
|
||||||
toml::value data;
|
|
||||||
|
|
||||||
std::error_code error;
|
|
||||||
if (std::filesystem::exists(path, error)) {
|
|
||||||
try {
|
|
||||||
data = toml::parse(path);
|
|
||||||
} catch (const std::exception& ex) {
|
|
||||||
fmt::print("Exception trying to parse config file. Exception: {}\n", ex.what());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (error) {
|
|
||||||
fmt::print("Filesystem error accessing {} (error: {})\n", path.string(),
|
|
||||||
error.message().c_str());
|
|
||||||
}
|
|
||||||
fmt::print("Saving new configuration file {}\n", path.string());
|
|
||||||
}
|
|
||||||
|
|
||||||
data["General"]["isPS4Pro"] = isNeo;
|
|
||||||
data["General"]["Fullscreen"] = isFullscreen;
|
|
||||||
data["General"]["logFilter"] = logFilter;
|
|
||||||
data["General"]["logType"] = logType;
|
|
||||||
data["General"]["userName"] = userName;
|
|
||||||
data["General"]["showSplash"] = isShowSplash;
|
|
||||||
data["Input"]["useSpecialPad"] = useSpecialPad;
|
|
||||||
data["Input"]["specialPadClass"] = specialPadClass;
|
|
||||||
data["GPU"]["screenWidth"] = screenWidth;
|
|
||||||
data["GPU"]["screenHeight"] = screenHeight;
|
|
||||||
data["GPU"]["nullGpu"] = isNullGpu;
|
|
||||||
data["GPU"]["copyGPUBuffers"] = shouldCopyGPUBuffers;
|
|
||||||
data["GPU"]["dumpShaders"] = shouldDumpShaders;
|
|
||||||
data["GPU"]["dumpPM4"] = shouldDumpPM4;
|
|
||||||
data["GPU"]["vblankDivider"] = vblankDivider;
|
|
||||||
data["Vulkan"]["gpuId"] = gpuId;
|
|
||||||
data["Vulkan"]["validation"] = vkValidation;
|
|
||||||
data["Vulkan"]["validation_sync"] = vkValidationSync;
|
|
||||||
data["Vulkan"]["validation_gpu"] = vkValidationGpu;
|
|
||||||
data["Vulkan"]["rdocEnable"] = rdocEnable;
|
|
||||||
data["Vulkan"]["rdocMarkersEnable"] = rdocMarkersEnable;
|
|
||||||
data["Debug"]["DebugDump"] = isDebugDump;
|
|
||||||
data["GUI"]["theme"] = mw_themes;
|
|
||||||
data["GUI"]["iconSize"] = m_icon_size;
|
|
||||||
data["GUI"]["sliderPos"] = m_slider_pos;
|
|
||||||
data["GUI"]["iconSizeGrid"] = m_icon_size_grid;
|
|
||||||
data["GUI"]["sliderPosGrid"] = m_slider_pos_grid;
|
|
||||||
data["GUI"]["gameTableMode"] = m_table_mode;
|
|
||||||
data["GUI"]["mw_width"] = m_window_size_W;
|
|
||||||
data["GUI"]["mw_height"] = m_window_size_H;
|
|
||||||
data["GUI"]["installDir"] = settings_install_dir;
|
|
||||||
data["GUI"]["geometry_x"] = main_window_geometry_x;
|
|
||||||
data["GUI"]["geometry_y"] = main_window_geometry_y;
|
|
||||||
data["GUI"]["geometry_w"] = main_window_geometry_w;
|
|
||||||
data["GUI"]["geometry_h"] = main_window_geometry_h;
|
|
||||||
data["GUI"]["pkgDirs"] = m_pkg_viewer;
|
|
||||||
data["GUI"]["elfDirs"] = m_elf_viewer;
|
|
||||||
data["GUI"]["recentFiles"] = m_recent_files;
|
|
||||||
data["GUI"]["emulatorLanguage"] = emulator_language;
|
|
||||||
|
|
||||||
data["Settings"]["consoleLanguage"] = m_language;
|
|
||||||
|
|
||||||
std::ofstream file(path, std::ios::out);
|
|
||||||
file << data;
|
|
||||||
file.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
void setDefaultValues() {
|
|
||||||
isNeo = false;
|
|
||||||
isFullscreen = false;
|
|
||||||
screenWidth = 1280;
|
|
||||||
screenHeight = 720;
|
|
||||||
logFilter = "";
|
|
||||||
logType = "async";
|
|
||||||
userName = "shadPS4";
|
|
||||||
useSpecialPad = false;
|
|
||||||
specialPadClass = 1;
|
|
||||||
isDebugDump = false;
|
|
||||||
isShowSplash = false;
|
|
||||||
isNullGpu = false;
|
|
||||||
shouldDumpShaders = false;
|
|
||||||
shouldDumpPM4 = false;
|
|
||||||
vblankDivider = 1;
|
|
||||||
vkValidation = false;
|
|
||||||
rdocEnable = false;
|
|
||||||
emulator_language = "en";
|
|
||||||
m_language = 1;
|
|
||||||
gpuId = -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Config
|
|
|
@ -1,104 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <filesystem>
|
|
||||||
#include <vector>
|
|
||||||
#include "types.h"
|
|
||||||
|
|
||||||
namespace Config {
|
|
||||||
void load(const std::filesystem::path& path);
|
|
||||||
void save(const std::filesystem::path& path);
|
|
||||||
|
|
||||||
bool isNeoMode();
|
|
||||||
bool isFullscreenMode();
|
|
||||||
std::string getLogFilter();
|
|
||||||
std::string getLogType();
|
|
||||||
std::string getUserName();
|
|
||||||
|
|
||||||
bool getUseSpecialPad();
|
|
||||||
int getSpecialPadClass();
|
|
||||||
|
|
||||||
u32 getScreenWidth();
|
|
||||||
u32 getScreenHeight();
|
|
||||||
s32 getGpuId();
|
|
||||||
|
|
||||||
bool debugDump();
|
|
||||||
bool showSplash();
|
|
||||||
bool nullGpu();
|
|
||||||
bool copyGPUCmdBuffers();
|
|
||||||
bool dumpShaders();
|
|
||||||
bool dumpPM4();
|
|
||||||
bool isRdocEnabled();
|
|
||||||
bool isMarkersEnabled();
|
|
||||||
u32 vblankDiv();
|
|
||||||
|
|
||||||
void setDebugDump(bool enable);
|
|
||||||
void setShowSplash(bool enable);
|
|
||||||
void setNullGpu(bool enable);
|
|
||||||
void setCopyGPUCmdBuffers(bool enable);
|
|
||||||
void setDumpShaders(bool enable);
|
|
||||||
void setDumpPM4(bool enable);
|
|
||||||
void setVblankDiv(u32 value);
|
|
||||||
void setGpuId(s32 selectedGpuId);
|
|
||||||
void setScreenWidth(u32 width);
|
|
||||||
void setScreenHeight(u32 height);
|
|
||||||
void setFullscreenMode(bool enable);
|
|
||||||
void setLanguage(u32 language);
|
|
||||||
void setNeoMode(bool enable);
|
|
||||||
void setUserName(const std::string& type);
|
|
||||||
|
|
||||||
void setUseSpecialPad(bool use);
|
|
||||||
void setSpecialPadClass(int type);
|
|
||||||
|
|
||||||
void setLogType(const std::string& type);
|
|
||||||
void setLogFilter(const std::string& type);
|
|
||||||
|
|
||||||
void setVkValidation(bool enable);
|
|
||||||
void setVkSyncValidation(bool enable);
|
|
||||||
void setRdocEnabled(bool enable);
|
|
||||||
|
|
||||||
bool vkValidationEnabled();
|
|
||||||
bool vkValidationSyncEnabled();
|
|
||||||
bool vkValidationGpuEnabled();
|
|
||||||
|
|
||||||
// Gui
|
|
||||||
void setMainWindowGeometry(u32 x, u32 y, u32 w, u32 h);
|
|
||||||
void setGameInstallDir(const std::string& dir);
|
|
||||||
void setMainWindowTheme(u32 theme);
|
|
||||||
void setIconSize(u32 size);
|
|
||||||
void setIconSizeGrid(u32 size);
|
|
||||||
void setSliderPosition(u32 pos);
|
|
||||||
void setSliderPositionGrid(u32 pos);
|
|
||||||
void setTableMode(u32 mode);
|
|
||||||
void setMainWindowWidth(u32 width);
|
|
||||||
void setMainWindowHeight(u32 height);
|
|
||||||
void setPkgViewer(const std::vector<std::string>& pkgList);
|
|
||||||
void setElfViewer(const std::vector<std::string>& elfList);
|
|
||||||
void setRecentFiles(const std::vector<std::string>& recentFiles);
|
|
||||||
void setEmulatorLanguage(std::string language);
|
|
||||||
|
|
||||||
u32 getMainWindowGeometryX();
|
|
||||||
u32 getMainWindowGeometryY();
|
|
||||||
u32 getMainWindowGeometryW();
|
|
||||||
u32 getMainWindowGeometryH();
|
|
||||||
std::string getGameInstallDir();
|
|
||||||
u32 getMainWindowTheme();
|
|
||||||
u32 getIconSize();
|
|
||||||
u32 getIconSizeGrid();
|
|
||||||
u32 getSliderPosition();
|
|
||||||
u32 getSliderPositionGrid();
|
|
||||||
u32 getTableMode();
|
|
||||||
u32 getMainWindowWidth();
|
|
||||||
u32 getMainWindowHeight();
|
|
||||||
std::vector<std::string> getPkgViewer();
|
|
||||||
std::vector<std::string> getElfViewer();
|
|
||||||
std::vector<std::string> getRecentFiles();
|
|
||||||
std::string getEmulatorLanguage();
|
|
||||||
|
|
||||||
void setDefaultValues();
|
|
||||||
|
|
||||||
// settings
|
|
||||||
u32 GetLanguage();
|
|
||||||
}; // namespace Config
|
|
|
@ -10,50 +10,3 @@
|
||||||
#else
|
#else
|
||||||
#error What the fuck is this compiler
|
#error What the fuck is this compiler
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <tracy/Tracy.hpp>
|
|
||||||
|
|
||||||
static inline bool IsProfilerConnected() {
|
|
||||||
return tracy::GetProfiler().IsConnected();
|
|
||||||
}
|
|
||||||
|
|
||||||
#define CUSTOM_LOCK(type, varname) \
|
|
||||||
tracy::LockableCtx varname { \
|
|
||||||
[]() -> const tracy::SourceLocationData* { \
|
|
||||||
static constexpr tracy::SourceLocationData srcloc{nullptr, #type " " #varname, \
|
|
||||||
TracyFile, TracyLine, 0}; \
|
|
||||||
return &srcloc; \
|
|
||||||
}() \
|
|
||||||
}
|
|
||||||
|
|
||||||
#define TRACK_ALLOC(ptr, size, pool) TracyAllocN(std::bit_cast<void*>(ptr), (size), (pool))
|
|
||||||
#define TRACK_FREE(ptr, pool) TracyFreeN(std::bit_cast<void*>(ptr), (pool))
|
|
||||||
|
|
||||||
enum MarkersPalette : int {
|
|
||||||
EmulatorMarkerColor = 0x264653,
|
|
||||||
RendererMarkerColor = 0x2a9d8f,
|
|
||||||
HleMarkerColor = 0xe9c46a,
|
|
||||||
GpuMarkerColor = 0xf4a261,
|
|
||||||
Reserved1 = 0xe76f51,
|
|
||||||
};
|
|
||||||
|
|
||||||
#define EMULATOR_TRACE ZoneScopedC(EmulatorMarkerColor)
|
|
||||||
#define RENDERER_TRACE ZoneScopedC(RendererMarkerColor)
|
|
||||||
#define HLE_TRACE ZoneScopedC(HleMarkerColor)
|
|
||||||
|
|
||||||
#define TRACE_HINT(str) ZoneText(str.c_str(), str.size())
|
|
||||||
|
|
||||||
#define TRACE_WARN(msg) \
|
|
||||||
[](const auto& msg) { TracyMessageC(msg.c_str(), msg.size(), tracy::Color::DarkOrange); }(msg);
|
|
||||||
#define TRACE_ERROR(msg) \
|
|
||||||
[](const auto& msg) { TracyMessageC(msg.c_str(), msg.size(), tracy::Color::Red); }(msg)
|
|
||||||
#define TRACE_CRIT(msg) \
|
|
||||||
[](const auto& msg) { TracyMessageC(msg.c_str(), msg.size(), tracy::Color::HotPink); }(msg)
|
|
||||||
|
|
||||||
#define GPU_SCOPE_LOCATION(name, color) \
|
|
||||||
tracy::SourceLocationData{name, TracyFunction, TracyFile, (uint32_t)TracyLine, color};
|
|
||||||
|
|
||||||
#define MUTEX_LOCATION(name) \
|
|
||||||
tracy::SourceLocationData{nullptr, name, TracyFile, (uint32_t)TracyLine, 0};
|
|
||||||
|
|
||||||
#define FRAME_END FrameMark
|
|
||||||
|
|
|
@ -1,25 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstddef>
|
|
||||||
#include <type_traits>
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
/// Ceiled integer division.
|
|
||||||
template <typename N, typename D>
|
|
||||||
requires std::is_integral_v<N> && std::is_unsigned_v<D>
|
|
||||||
[[nodiscard]] constexpr N DivCeil(N number, D divisor) {
|
|
||||||
return static_cast<N>((static_cast<D>(number) + divisor - 1) / divisor);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Ceiled integer division with logarithmic divisor in base 2
|
|
||||||
template <typename N, typename D>
|
|
||||||
requires std::is_integral_v<N> && std::is_unsigned_v<D>
|
|
||||||
[[nodiscard]] constexpr N DivCeilLog2(N value, D alignment_log2) {
|
|
||||||
return static_cast<N>((static_cast<D>(value) + (D(1) << alignment_log2) - 1) >> alignment_log2);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -1,161 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <type_traits>
|
|
||||||
#include "common/types.h"
|
|
||||||
|
|
||||||
#define DECLARE_ENUM_FLAG_OPERATORS(type) \
|
|
||||||
[[nodiscard]] constexpr type operator|(type a, type b) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(static_cast<T>(a) | static_cast<T>(b)); \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr type operator&(type a, type b) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr type operator^(type a, type b) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(static_cast<T>(a) ^ static_cast<T>(b)); \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr type operator<<(type a, type b) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(static_cast<T>(a) << static_cast<T>(b)); \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr type operator>>(type a, type b) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(static_cast<T>(a) >> static_cast<T>(b)); \
|
|
||||||
} \
|
|
||||||
constexpr type& operator|=(type& a, type b) noexcept { \
|
|
||||||
a = a | b; \
|
|
||||||
return a; \
|
|
||||||
} \
|
|
||||||
constexpr type& operator&=(type& a, type b) noexcept { \
|
|
||||||
a = a & b; \
|
|
||||||
return a; \
|
|
||||||
} \
|
|
||||||
constexpr type& operator^=(type& a, type b) noexcept { \
|
|
||||||
a = a ^ b; \
|
|
||||||
return a; \
|
|
||||||
} \
|
|
||||||
constexpr type& operator<<=(type& a, type b) noexcept { \
|
|
||||||
a = a << b; \
|
|
||||||
return a; \
|
|
||||||
} \
|
|
||||||
constexpr type& operator>>=(type& a, type b) noexcept { \
|
|
||||||
a = a >> b; \
|
|
||||||
return a; \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr type operator~(type key) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<type>(~static_cast<T>(key)); \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr bool True(type key) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<T>(key) != 0; \
|
|
||||||
} \
|
|
||||||
[[nodiscard]] constexpr bool False(type key) noexcept { \
|
|
||||||
using T = std::underlying_type_t<type>; \
|
|
||||||
return static_cast<T>(key) == 0; \
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
template <typename T>
|
|
||||||
class Flags {
|
|
||||||
public:
|
|
||||||
using IntType = std::underlying_type_t<T>;
|
|
||||||
|
|
||||||
Flags() {}
|
|
||||||
|
|
||||||
Flags(IntType t) : m_bits(t) {}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
Flags(T f, Tx... fx) {
|
|
||||||
this->set(f, fx...);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
void set(Tx... fx) {
|
|
||||||
m_bits |= bits(fx...);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set(Flags flags) {
|
|
||||||
m_bits |= flags.m_bits;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
void clr(Tx... fx) {
|
|
||||||
m_bits &= ~bits(fx...);
|
|
||||||
}
|
|
||||||
|
|
||||||
void clr(Flags flags) {
|
|
||||||
m_bits &= ~flags.m_bits;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
bool any(Tx... fx) const {
|
|
||||||
return (m_bits & bits(fx...)) != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
bool all(Tx... fx) const {
|
|
||||||
const IntType mask = bits(fx...);
|
|
||||||
return (m_bits & mask) == mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool test(T f) const {
|
|
||||||
return this->any(f);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isClear() const {
|
|
||||||
return m_bits == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void clrAll() {
|
|
||||||
m_bits = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
u32 raw() const {
|
|
||||||
return m_bits;
|
|
||||||
}
|
|
||||||
|
|
||||||
Flags operator&(const Flags& other) const {
|
|
||||||
return Flags(m_bits & other.m_bits);
|
|
||||||
}
|
|
||||||
|
|
||||||
Flags operator|(const Flags& other) const {
|
|
||||||
return Flags(m_bits | other.m_bits);
|
|
||||||
}
|
|
||||||
|
|
||||||
Flags operator^(const Flags& other) const {
|
|
||||||
return Flags(m_bits ^ other.m_bits);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator==(const Flags& other) const {
|
|
||||||
return m_bits == other.m_bits;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator!=(const Flags& other) const {
|
|
||||||
return m_bits != other.m_bits;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
IntType m_bits = 0;
|
|
||||||
|
|
||||||
static IntType bit(T f) {
|
|
||||||
return IntType(1) << static_cast<IntType>(f);
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename... Tx>
|
|
||||||
static IntType bits(T f, Tx... fx) {
|
|
||||||
return bit(f) | bits(fx...);
|
|
||||||
}
|
|
||||||
|
|
||||||
static IntType bits() {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -1,57 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: 2013 Dolphin Emulator Project
|
|
||||||
// SPDX-FileCopyrightText: 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include <cstddef>
|
|
||||||
#ifdef _WIN32
|
|
||||||
#include <windows.h>
|
|
||||||
#else
|
|
||||||
#include <cerrno>
|
|
||||||
#include <cstring>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "common/error.h"
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
std::string NativeErrorToString(int e) {
|
|
||||||
#ifdef _WIN32
|
|
||||||
LPSTR err_str;
|
|
||||||
|
|
||||||
DWORD res = FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
||||||
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
||||||
nullptr, e, MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US),
|
|
||||||
reinterpret_cast<LPSTR>(&err_str), 1, nullptr);
|
|
||||||
if (!res) {
|
|
||||||
return "(FormatMessageA failed to format error)";
|
|
||||||
}
|
|
||||||
std::string ret(err_str);
|
|
||||||
LocalFree(err_str);
|
|
||||||
return ret;
|
|
||||||
#else
|
|
||||||
char err_str[255];
|
|
||||||
#if defined(__GLIBC__) && (_GNU_SOURCE || (_POSIX_C_SOURCE < 200112L && _XOPEN_SOURCE < 600)) || \
|
|
||||||
defined(ANDROID)
|
|
||||||
// Thread safe (GNU-specific)
|
|
||||||
const char* str = strerror_r(e, err_str, sizeof(err_str));
|
|
||||||
return std::string(str);
|
|
||||||
#else
|
|
||||||
// Thread safe (XSI-compliant)
|
|
||||||
int second_err = strerror_r(e, err_str, sizeof(err_str));
|
|
||||||
if (second_err != 0) {
|
|
||||||
return "(strerror_r failed to format error)";
|
|
||||||
}
|
|
||||||
return std::string(err_str);
|
|
||||||
#endif // GLIBC etc.
|
|
||||||
#endif // _WIN32
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string GetLastErrorMsg() {
|
|
||||||
#ifdef _WIN32
|
|
||||||
return NativeErrorToString(GetLastError());
|
|
||||||
#else
|
|
||||||
return NativeErrorToString(errno);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -1,21 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: 2013 Dolphin Emulator Project
|
|
||||||
// SPDX-FileCopyrightText: 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
// Generic function to get last error message.
|
|
||||||
// Call directly after the command or use the error num.
|
|
||||||
// This function might change the error code.
|
|
||||||
// Defined in error.cpp.
|
|
||||||
[[nodiscard]] std::string GetLastErrorMsg();
|
|
||||||
|
|
||||||
// Like GetLastErrorMsg(), but passing an explicit error code.
|
|
||||||
// Defined in error.cpp.
|
|
||||||
[[nodiscard]] std::string NativeErrorToString(int e);
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -0,0 +1,96 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "common/fs_file.h"
|
||||||
|
|
||||||
|
namespace Common::FS {
|
||||||
|
|
||||||
|
File::File() = default;
|
||||||
|
|
||||||
|
File::File(const std::string& path, OpenMode mode) {
|
||||||
|
open(path, mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
File::~File() {
|
||||||
|
close();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::open(const std::string& path, OpenMode mode) {
|
||||||
|
close();
|
||||||
|
#ifdef _WIN64
|
||||||
|
fopen_s(&m_file, path.c_str(), getOpenMode(mode));
|
||||||
|
#else
|
||||||
|
m_file = std::fopen(path.c_str(), getOpenMode(mode));
|
||||||
|
#endif
|
||||||
|
return isOpen();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::close() {
|
||||||
|
if (!isOpen() || std::fclose(m_file) != 0) [[unlikely]] {
|
||||||
|
m_file = nullptr;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_file = nullptr;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::write(std::span<const u8> data) {
|
||||||
|
return isOpen() && std::fwrite(data.data(), 1, data.size(), m_file) == data.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::read(void* data, u64 size) const {
|
||||||
|
return isOpen() && std::fread(data, 1, size, m_file) == size;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool File::seek(s64 offset, SeekMode mode) {
|
||||||
|
#ifdef _WIN64
|
||||||
|
if (!isOpen() || _fseeki64(m_file, offset, getSeekMode(mode)) != 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
if (!isOpen() || fseeko64(m_file, offset, getSeekMode(mode)) != 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 File::tell() const {
|
||||||
|
if (isOpen()) [[likely]] {
|
||||||
|
#ifdef _WIN64
|
||||||
|
return _ftelli64(m_file);
|
||||||
|
#else
|
||||||
|
return ftello64(m_file);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 File::getFileSize() {
|
||||||
|
#ifdef _WIN64
|
||||||
|
const u64 pos = _ftelli64(m_file);
|
||||||
|
if (_fseeki64(m_file, 0, SEEK_END) != 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const u64 size = _ftelli64(m_file);
|
||||||
|
if (_fseeki64(m_file, pos, SEEK_SET) != 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
const u64 pos = ftello64(m_file);
|
||||||
|
if (fseeko64(m_file, 0, SEEK_END) != 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const u64 size = ftello64(m_file);
|
||||||
|
if (fseeko64(m_file, pos, SEEK_SET) != 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return size;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Common::FS
|
|
@ -0,0 +1,86 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <bit>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <span>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "common/types.h"
|
||||||
|
|
||||||
|
namespace Common::FS {
|
||||||
|
|
||||||
|
enum class OpenMode : u32 { Read = 0x1, Write = 0x2, ReadWrite = Read | Write };
|
||||||
|
|
||||||
|
enum class SeekMode : u32 {
|
||||||
|
Set,
|
||||||
|
Cur,
|
||||||
|
End,
|
||||||
|
};
|
||||||
|
|
||||||
|
class File {
|
||||||
|
public:
|
||||||
|
File();
|
||||||
|
explicit File(const std::string& path, OpenMode mode = OpenMode::Read);
|
||||||
|
~File();
|
||||||
|
|
||||||
|
bool open(const std::string& path, OpenMode mode = OpenMode::Read);
|
||||||
|
bool close();
|
||||||
|
bool read(void* data, u64 size) const;
|
||||||
|
bool write(std::span<const u8> data);
|
||||||
|
bool seek(s64 offset, SeekMode mode);
|
||||||
|
u64 getFileSize();
|
||||||
|
u64 tell() const;
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
bool read(T& data) const {
|
||||||
|
return read(&data, sizeof(T));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
bool read(std::vector<T>& data) const {
|
||||||
|
return read(data.data(), data.size() * sizeof(T));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool isOpen() const {
|
||||||
|
return m_file != nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* getOpenMode(OpenMode mode) const {
|
||||||
|
switch (mode) {
|
||||||
|
case OpenMode::Read:
|
||||||
|
return "rb";
|
||||||
|
case OpenMode::Write:
|
||||||
|
return "wb";
|
||||||
|
case OpenMode::ReadWrite:
|
||||||
|
return "r+b";
|
||||||
|
default:
|
||||||
|
return "r";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int getSeekMode(SeekMode mode) const {
|
||||||
|
switch (mode) {
|
||||||
|
case SeekMode::Set:
|
||||||
|
return SEEK_SET;
|
||||||
|
case SeekMode::Cur:
|
||||||
|
return SEEK_CUR;
|
||||||
|
case SeekMode::End:
|
||||||
|
return SEEK_END;
|
||||||
|
default:
|
||||||
|
return SEEK_SET;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] std::FILE* fileDescr() const {
|
||||||
|
return m_file;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::FILE* m_file{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Common::FS
|
|
@ -1,34 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <tuple>
|
|
||||||
|
|
||||||
namespace Common {
|
|
||||||
|
|
||||||
template <class Func>
|
|
||||||
struct FuncTraits {};
|
|
||||||
|
|
||||||
template <class ReturnType_, class... Args>
|
|
||||||
struct FuncTraits<ReturnType_ (*)(Args...)> {
|
|
||||||
using ReturnType = ReturnType_;
|
|
||||||
|
|
||||||
static constexpr size_t NUM_ARGS = sizeof...(Args);
|
|
||||||
|
|
||||||
template <size_t I>
|
|
||||||
using ArgType = std::tuple_element_t<I, std::tuple<Args...>>;
|
|
||||||
};
|
|
||||||
|
|
||||||
template <typename Func>
|
|
||||||
struct LambdaTraits : LambdaTraits<decltype(&std::remove_reference_t<Func>::operator())> {};
|
|
||||||
|
|
||||||
template <typename ReturnType, typename LambdaType, typename... Args>
|
|
||||||
struct LambdaTraits<ReturnType (LambdaType::*)(Args...) const> {
|
|
||||||
template <size_t I>
|
|
||||||
using ArgType = std::tuple_element_t<I, std::tuple<Args...>>;
|
|
||||||
|
|
||||||
static constexpr size_t NUM_ARGS{sizeof...(Args)};
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Common
|
|
|
@ -3,19 +3,13 @@
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "common/alignment.h"
|
|
||||||
#include "common/assert.h"
|
|
||||||
#include "common/error.h"
|
|
||||||
#include "common/io_file.h"
|
#include "common/io_file.h"
|
||||||
#include "common/logging/log.h"
|
// #include "common/logging/log.h"
|
||||||
#include "common/path_util.h"
|
// #include "common/path_util.h"
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#include "common/ntapi.h"
|
|
||||||
|
|
||||||
#include <io.h>
|
#include <io.h>
|
||||||
#include <share.h>
|
#include <share.h>
|
||||||
#include <windows.h>
|
|
||||||
#else
|
#else
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#endif
|
#endif
|
||||||
|
@ -169,7 +163,7 @@ IOFile& IOFile::operator=(IOFile&& other) noexcept {
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
int IOFile::Open(const fs::path& path, FileAccessMode mode, FileType type, FileShareFlag flag) {
|
void IOFile::Open(const fs::path& path, FileAccessMode mode, FileType type, FileShareFlag flag) {
|
||||||
Close();
|
Close();
|
||||||
|
|
||||||
file_path = path;
|
file_path = path;
|
||||||
|
@ -177,26 +171,22 @@ int IOFile::Open(const fs::path& path, FileAccessMode mode, FileType type, FileS
|
||||||
file_type = type;
|
file_type = type;
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
int result = 0;
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
if (flag != FileShareFlag::ShareNone) {
|
if (flag != FileShareFlag::ShareNone) {
|
||||||
file = _wfsopen(path.c_str(), AccessModeToWStr(mode, type), ToWindowsFileShareFlag(flag));
|
file = _wfsopen(path.c_str(), AccessModeToWStr(mode, type), ToWindowsFileShareFlag(flag));
|
||||||
result = errno;
|
|
||||||
} else {
|
} else {
|
||||||
result = _wfopen_s(&file, path.c_str(), AccessModeToWStr(mode, type));
|
_wfopen_s(&file, path.c_str(), AccessModeToWStr(mode, type));
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
file = std::fopen(path.c_str(), AccessModeToStr(mode, type));
|
file = std::fopen(path.c_str(), AccessModeToStr(mode, type));
|
||||||
result = errno;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!IsOpen()) {
|
if (!IsOpen()) {
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to open the file at path={}",
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
PathToUTF8String(file_path));
|
// LOG_ERROR(Common_Filesystem, "Failed to open the file at path={}, ec_message={}",
|
||||||
|
// PathToUTF8String(file_path), ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void IOFile::Close() {
|
void IOFile::Close() {
|
||||||
|
@ -210,65 +200,11 @@ void IOFile::Close() {
|
||||||
|
|
||||||
if (!close_result) {
|
if (!close_result) {
|
||||||
const auto ec = std::error_code{errno, std::generic_category()};
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to close the file at path={}, ec_message={}",
|
// LOG_ERROR(Common_Filesystem, "Failed to close the file at path={}, ec_message={}",
|
||||||
PathToUTF8String(file_path), ec.message());
|
// PathToUTF8String(file_path), ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
file = nullptr;
|
file = nullptr;
|
||||||
|
|
||||||
#ifdef _WIN64
|
|
||||||
if (file_mapping && file_access_mode == FileAccessMode::ReadWrite) {
|
|
||||||
CloseHandle(std::bit_cast<HANDLE>(file_mapping));
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void IOFile::Unlink() {
|
|
||||||
if (!IsOpen()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Mark the file for deletion
|
|
||||||
// TODO: Also remove the file path?
|
|
||||||
#if _WIN64
|
|
||||||
FILE_DISPOSITION_INFORMATION disposition;
|
|
||||||
IO_STATUS_BLOCK iosb;
|
|
||||||
|
|
||||||
const int fd = fileno(file);
|
|
||||||
HANDLE hfile = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
|
|
||||||
|
|
||||||
disposition.DeleteFile = TRUE;
|
|
||||||
NtSetInformationFile(hfile, &iosb, &disposition, sizeof(disposition),
|
|
||||||
FileDispositionInformation);
|
|
||||||
#else
|
|
||||||
UNREACHABLE_MSG("Missing Linux implementation");
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
uintptr_t IOFile::GetFileMapping() {
|
|
||||||
if (file_mapping) {
|
|
||||||
return file_mapping;
|
|
||||||
}
|
|
||||||
#ifdef _WIN64
|
|
||||||
const int fd = fileno(file);
|
|
||||||
|
|
||||||
HANDLE hfile = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
|
|
||||||
HANDLE mapping = nullptr;
|
|
||||||
|
|
||||||
if (file_access_mode == FileAccessMode::ReadWrite) {
|
|
||||||
mapping = CreateFileMapping2(hfile, NULL, FILE_MAP_WRITE, PAGE_READWRITE, SEC_COMMIT, 0,
|
|
||||||
NULL, NULL, 0);
|
|
||||||
} else {
|
|
||||||
mapping = hfile;
|
|
||||||
}
|
|
||||||
|
|
||||||
file_mapping = std::bit_cast<uintptr_t>(mapping);
|
|
||||||
ASSERT_MSG(file_mapping, "{}", Common::GetLastErrorMsg());
|
|
||||||
return file_mapping;
|
|
||||||
#else
|
|
||||||
file_mapping = fileno(file);
|
|
||||||
return file_mapping;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string IOFile::ReadString(size_t length) const {
|
std::string IOFile::ReadString(size_t length) const {
|
||||||
|
@ -295,8 +231,8 @@ bool IOFile::Flush() const {
|
||||||
|
|
||||||
if (!flush_result) {
|
if (!flush_result) {
|
||||||
const auto ec = std::error_code{errno, std::generic_category()};
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to flush the file at path={}, ec_message={}",
|
// LOG_ERROR(Common_Filesystem, "Failed to flush the file at path={}, ec_message={}",
|
||||||
PathToUTF8String(file_path), ec.message());
|
// PathToUTF8String(file_path), ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
return flush_result;
|
return flush_result;
|
||||||
|
@ -317,8 +253,8 @@ bool IOFile::Commit() const {
|
||||||
|
|
||||||
if (!commit_result) {
|
if (!commit_result) {
|
||||||
const auto ec = std::error_code{errno, std::generic_category()};
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to commit the file at path={}, ec_message={}",
|
// LOG_ERROR(Common_Filesystem, "Failed to commit the file at path={}, ec_message={}",
|
||||||
PathToUTF8String(file_path), ec.message());
|
// PathToUTF8String(file_path), ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
return commit_result;
|
return commit_result;
|
||||||
|
@ -339,8 +275,9 @@ bool IOFile::SetSize(u64 size) const {
|
||||||
|
|
||||||
if (!set_size_result) {
|
if (!set_size_result) {
|
||||||
const auto ec = std::error_code{errno, std::generic_category()};
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to resize the file at path={}, size={}, ec_message={}",
|
// LOG_ERROR(Common_Filesystem, "Failed to resize the file at path={}, size={},
|
||||||
PathToUTF8String(file_path), size, ec.message());
|
// ec_message={}",
|
||||||
|
// PathToUTF8String(file_path), size, ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
return set_size_result;
|
return set_size_result;
|
||||||
|
@ -359,8 +296,9 @@ u64 IOFile::GetSize() const {
|
||||||
const auto file_size = fs::file_size(file_path, ec);
|
const auto file_size = fs::file_size(file_path, ec);
|
||||||
|
|
||||||
if (ec) {
|
if (ec) {
|
||||||
LOG_ERROR(Common_Filesystem, "Failed to retrieve the file size of path={}, ec_message={}",
|
// LOG_ERROR(Common_Filesystem, "Failed to retrieve the file size of path={},
|
||||||
PathToUTF8String(file_path), ec.message());
|
// ec_message={}",
|
||||||
|
// PathToUTF8String(file_path), ec.message());
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -378,9 +316,9 @@ bool IOFile::Seek(s64 offset, SeekOrigin origin) const {
|
||||||
|
|
||||||
if (!seek_result) {
|
if (!seek_result) {
|
||||||
const auto ec = std::error_code{errno, std::generic_category()};
|
const auto ec = std::error_code{errno, std::generic_category()};
|
||||||
LOG_ERROR(Common_Filesystem,
|
// LOG_ERROR(Common_Filesystem,
|
||||||
"Failed to seek the file at path={}, offset={}, origin={}, ec_message={}",
|
// "Failed to seek the file at path={}, offset={}, origin={}, ec_message={}",
|
||||||
PathToUTF8String(file_path), offset, static_cast<u32>(origin), ec.message());
|
// PathToUTF8String(file_path), offset, static_cast<u32>(origin), ec.message());
|
||||||
}
|
}
|
||||||
|
|
||||||
return seek_result;
|
return seek_result;
|
||||||
|
|
|
@ -100,15 +100,11 @@ public:
|
||||||
return file != nullptr;
|
return file != nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
uintptr_t GetFileMapping();
|
void Open(const std::filesystem::path& path, FileAccessMode mode,
|
||||||
|
|
||||||
int Open(const std::filesystem::path& path, FileAccessMode mode,
|
|
||||||
FileType type = FileType::BinaryFile,
|
FileType type = FileType::BinaryFile,
|
||||||
FileShareFlag flag = FileShareFlag::ShareReadOnly);
|
FileShareFlag flag = FileShareFlag::ShareReadOnly);
|
||||||
void Close();
|
void Close();
|
||||||
|
|
||||||
void Unlink();
|
|
||||||
|
|
||||||
bool Flush() const;
|
bool Flush() const;
|
||||||
bool Commit() const;
|
bool Commit() const;
|
||||||
|
|
||||||
|
@ -183,7 +179,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
size_t WriteRaw(const void* data, size_t size) const {
|
size_t WriteRaw(void* data, size_t size) const {
|
||||||
return std::fwrite(data, sizeof(T), size, file);
|
return std::fwrite(data, sizeof(T), size, file);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -205,18 +201,12 @@ public:
|
||||||
return WriteSpan(string);
|
return WriteSpan(string);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void WriteBytes(const std::filesystem::path path, std::span<const u8> data) {
|
|
||||||
IOFile out(path, FileAccessMode::Write);
|
|
||||||
out.Write(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::filesystem::path file_path;
|
std::filesystem::path file_path;
|
||||||
FileAccessMode file_access_mode{};
|
FileAccessMode file_access_mode{};
|
||||||
FileType file_type{};
|
FileType file_type{};
|
||||||
|
|
||||||
std::FILE* file = nullptr;
|
std::FILE* file = nullptr;
|
||||||
uintptr_t file_mapping = 0;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Common::FS
|
} // namespace Common::FS
|
||||||
|
|
|
@ -0,0 +1,159 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <Util/config.h>
|
||||||
|
#include <spdlog/common.h>
|
||||||
|
#include <spdlog/pattern_formatter.h>
|
||||||
|
#include <spdlog/sinks/basic_file_sink.h>
|
||||||
|
#include <spdlog/sinks/stdout_color_sinks.h>
|
||||||
|
#include <spdlog/spdlog.h>
|
||||||
|
#include "common/log.h"
|
||||||
|
#ifdef _WIN64
|
||||||
|
#include <windows.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace Common::Log {
|
||||||
|
|
||||||
|
std::vector<spdlog::sink_ptr> sinks;
|
||||||
|
constexpr bool log_file_exceptions = true;
|
||||||
|
|
||||||
|
void Flush() {
|
||||||
|
spdlog::details::registry::instance().flush_all();
|
||||||
|
}
|
||||||
|
|
||||||
|
thread_local uint8_t TLS[1024];
|
||||||
|
|
||||||
|
uint64_t tls_access(int64_t tls_offset) {
|
||||||
|
if (tls_offset == 0) {
|
||||||
|
return (uint64_t)TLS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef _WIN64
|
||||||
|
static LONG WINAPI ExceptionHandler(PEXCEPTION_POINTERS pExp) noexcept {
|
||||||
|
auto orig_rip = pExp->ContextRecord->Rip;
|
||||||
|
while (*(uint8_t*)pExp->ContextRecord->Rip == 0x66)
|
||||||
|
pExp->ContextRecord->Rip++;
|
||||||
|
|
||||||
|
if (*(uint8_t*)pExp->ContextRecord->Rip == 0xcd) {
|
||||||
|
int reg = *(uint8_t*)(pExp->ContextRecord->Rip + 1) - 0x80;
|
||||||
|
int sizes = *(uint8_t*)(pExp->ContextRecord->Rip + 2);
|
||||||
|
int pattern_size = sizes & 0xF;
|
||||||
|
int imm_size = sizes >> 4;
|
||||||
|
|
||||||
|
int64_t tls_offset;
|
||||||
|
if (imm_size == 4)
|
||||||
|
tls_offset = *(int32_t*)(pExp->ContextRecord->Rip + pattern_size);
|
||||||
|
else
|
||||||
|
tls_offset = *(int64_t*)(pExp->ContextRecord->Rip + pattern_size);
|
||||||
|
|
||||||
|
(&pExp->ContextRecord->Rax)[reg] = tls_access(tls_offset); /* TLS_ACCESS */
|
||||||
|
pExp->ContextRecord->Rip += pattern_size + imm_size;
|
||||||
|
|
||||||
|
return EXCEPTION_CONTINUE_EXECUTION;
|
||||||
|
}
|
||||||
|
|
||||||
|
pExp->ContextRecord->Rip = orig_rip;
|
||||||
|
const u32 ec = pExp->ExceptionRecord->ExceptionCode;
|
||||||
|
switch (ec) {
|
||||||
|
case EXCEPTION_ACCESS_VIOLATION: {
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_ACCESS_VIOLATION ({:#x}). ", ec);
|
||||||
|
const auto info = pExp->ExceptionRecord->ExceptionInformation;
|
||||||
|
switch (info[0]) {
|
||||||
|
case 0:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Read violation at address {:#x}.", info[1]);
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Write violation at address {:#x}.", info[1]);
|
||||||
|
break;
|
||||||
|
case 8:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "DEP violation at address {:#x}.", info[1]);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case EXCEPTION_ARRAY_BOUNDS_EXCEEDED:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_ARRAY_BOUNDS_EXCEEDED ({:#x}). ",
|
||||||
|
ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_DATATYPE_MISALIGNMENT:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_DATATYPE_MISALIGNMENT ({:#x}). ",
|
||||||
|
ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_FLT_DIVIDE_BY_ZERO:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_FLT_DIVIDE_BY_ZERO ({:#x}). ",
|
||||||
|
ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_ILLEGAL_INSTRUCTION:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_ILLEGAL_INSTRUCTION ({:#x}). ",
|
||||||
|
ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_IN_PAGE_ERROR:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_IN_PAGE_ERROR ({:#x}). ", ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_INT_DIVIDE_BY_ZERO:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_INT_DIVIDE_BY_ZERO ({:#x}). ",
|
||||||
|
ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_PRIV_INSTRUCTION:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_PRIV_INSTRUCTION ({:#x}). ", ec);
|
||||||
|
break;
|
||||||
|
case EXCEPTION_STACK_OVERFLOW:
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Exception EXCEPTION_STACK_OVERFLOW ({:#x}). ", ec);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return EXCEPTION_CONTINUE_SEARCH;
|
||||||
|
}
|
||||||
|
Flush();
|
||||||
|
return EXCEPTION_CONTINUE_SEARCH;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int Init(bool use_stdout) {
|
||||||
|
sinks.clear();
|
||||||
|
if (use_stdout) {
|
||||||
|
sinks.push_back(std::make_shared<spdlog::sinks::stdout_color_sink_mt>());
|
||||||
|
}
|
||||||
|
#ifdef _WIN64
|
||||||
|
sinks.push_back(std::make_shared<spdlog::sinks::basic_file_sink_mt>(L"shadps4.txt", true));
|
||||||
|
#else
|
||||||
|
sinks.push_back(std::make_shared<spdlog::sinks::basic_file_sink_mt>("shadps4.txt", true));
|
||||||
|
#endif
|
||||||
|
spdlog::set_default_logger(
|
||||||
|
std::make_shared<spdlog::logger>("shadps4 logger", begin(sinks), end(sinks)));
|
||||||
|
auto f = std::make_unique<spdlog::pattern_formatter>(
|
||||||
|
"%^|%L|: %v%$", spdlog::pattern_time_type::local, std::string("")); // disable eol
|
||||||
|
spdlog::set_formatter(std::move(f));
|
||||||
|
spdlog::set_level(static_cast<spdlog::level::level_enum>(Config::getLogLevel()));
|
||||||
|
|
||||||
|
#ifdef _WIN64
|
||||||
|
if (!AddVectoredExceptionHandler(0, ExceptionHandler)) {
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Failed to register an exception handler");
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static std::terminate_handler old_terminate = nullptr;
|
||||||
|
old_terminate = std::set_terminate([]() {
|
||||||
|
try {
|
||||||
|
std::rethrow_exception(std::current_exception());
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Unhandled C++ exception. {}", e.what());
|
||||||
|
} catch (...) {
|
||||||
|
LOG_CRITICAL_IF(log_file_exceptions, "Unhandled C++ exception. UNKNOWN");
|
||||||
|
}
|
||||||
|
Flush();
|
||||||
|
if (old_terminate)
|
||||||
|
old_terminate();
|
||||||
|
});
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetLevel(spdlog::level::level_enum log_level) {
|
||||||
|
spdlog::set_level(log_level);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Common::Log
|
|
@ -0,0 +1,42 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#define SPDLOG_ACTIVE_LEVEL SPDLOG_LEVEL_TRACE
|
||||||
|
|
||||||
|
#include <spdlog/spdlog.h>
|
||||||
|
|
||||||
|
namespace Common::Log {
|
||||||
|
|
||||||
|
#define LOG_TRACE SPDLOG_TRACE
|
||||||
|
#define LOG_DEBUG SPDLOG_DEBUG
|
||||||
|
#define LOG_INFO SPDLOG_INFO
|
||||||
|
#define LOG_WARN SPDLOG_WARN
|
||||||
|
#define LOG_ERROR SPDLOG_ERROR
|
||||||
|
#define LOG_CRITICAL SPDLOG_CRITICAL
|
||||||
|
|
||||||
|
#define LOG_TRACE_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_TRACE(__VA_ARGS__)
|
||||||
|
#define LOG_DEBUG_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_DEBUG(__VA_ARGS__)
|
||||||
|
#define LOG_INFO_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_INFO(__VA_ARGS__)
|
||||||
|
#define LOG_WARN_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_WARN(__VA_ARGS__)
|
||||||
|
#define LOG_ERROR_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_ERROR(__VA_ARGS__)
|
||||||
|
#define LOG_CRITICAL_IF(flag, ...) \
|
||||||
|
if (flag) \
|
||||||
|
LOG_CRITICAL(__VA_ARGS__)
|
||||||
|
|
||||||
|
int Init(bool use_stdout);
|
||||||
|
|
||||||
|
void SetLevel(spdlog::level::level_enum log_level);
|
||||||
|
|
||||||
|
} // namespace Common::Log
|
|
@ -1,302 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <filesystem>
|
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
#include <fmt/format.h>
|
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
#include <windows.h> // For OutputDebugStringW
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "common/bounded_threadsafe_queue.h"
|
|
||||||
#include "common/config.h"
|
|
||||||
#include "common/debug.h"
|
|
||||||
#include "common/io_file.h"
|
|
||||||
#include "common/logging/backend.h"
|
|
||||||
#include "common/logging/log.h"
|
|
||||||
#include "common/logging/log_entry.h"
|
|
||||||
#include "common/logging/text_formatter.h"
|
|
||||||
#include "common/path_util.h"
|
|
||||||
#include "common/string_util.h"
|
|
||||||
#include "common/thread.h"
|
|
||||||
|
|
||||||
namespace Common::Log {
|
|
||||||
|
|
||||||
using namespace Common::FS;
|
|
||||||
|
|
||||||
namespace {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Backend that writes to stderr and with color
|
|
||||||
*/
|
|
||||||
class ColorConsoleBackend {
|
|
||||||
public:
|
|
||||||
explicit ColorConsoleBackend() = default;
|
|
||||||
|
|
||||||
~ColorConsoleBackend() = default;
|
|
||||||
|
|
||||||
void Write(const Entry& entry) {
|
|
||||||
if (enabled.load(std::memory_order_relaxed)) {
|
|
||||||
PrintColoredMessage(entry);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Flush() {
|
|
||||||
// stderr shouldn't be buffered
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetEnabled(bool enabled_) {
|
|
||||||
enabled = enabled_;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::atomic_bool enabled{true};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Backend that writes to a file passed into the constructor
|
|
||||||
*/
|
|
||||||
class FileBackend {
|
|
||||||
public:
|
|
||||||
explicit FileBackend(const std::filesystem::path& filename)
|
|
||||||
: file{filename, FS::FileAccessMode::Write, FS::FileType::TextFile} {}
|
|
||||||
|
|
||||||
~FileBackend() = default;
|
|
||||||
|
|
||||||
void Write(const Entry& entry) {
|
|
||||||
if (!enabled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bytes_written += file.WriteString(FormatLogMessage(entry).append(1, '\n'));
|
|
||||||
|
|
||||||
// Prevent logs from exceeding a set maximum size in the event that log entries are spammed.
|
|
||||||
const auto write_limit = 100_MB;
|
|
||||||
const bool write_limit_exceeded = bytes_written > write_limit;
|
|
||||||
if (entry.log_level >= Level::Error || write_limit_exceeded) {
|
|
||||||
if (write_limit_exceeded) {
|
|
||||||
// Stop writing after the write limit is exceeded.
|
|
||||||
// Don't close the file so we can print a stacktrace if necessary
|
|
||||||
enabled = false;
|
|
||||||
}
|
|
||||||
file.Flush();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Flush() {
|
|
||||||
file.Flush();
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
Common::FS::IOFile file;
|
|
||||||
bool enabled = true;
|
|
||||||
std::size_t bytes_written = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Backend that writes to Visual Studio's output window
|
|
||||||
*/
|
|
||||||
class DebuggerBackend {
|
|
||||||
public:
|
|
||||||
explicit DebuggerBackend() = default;
|
|
||||||
|
|
||||||
~DebuggerBackend() = default;
|
|
||||||
|
|
||||||
void Write(const Entry& entry) {
|
|
||||||
#ifdef _WIN32
|
|
||||||
::OutputDebugStringW(UTF8ToUTF16W(FormatLogMessage(entry).append(1, '\n')).c_str());
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void Flush() {}
|
|
||||||
|
|
||||||
void EnableForStacktrace() {}
|
|
||||||
};
|
|
||||||
|
|
||||||
bool initialization_in_progress_suppress_logging = true;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Static state as a singleton.
|
|
||||||
*/
|
|
||||||
class Impl {
|
|
||||||
public:
|
|
||||||
static Impl& Instance() {
|
|
||||||
if (!instance) {
|
|
||||||
throw std::runtime_error("Using Logging instance before its initialization");
|
|
||||||
}
|
|
||||||
return *instance;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Initialize(std::string_view log_file) {
|
|
||||||
if (instance) {
|
|
||||||
LOG_WARNING(Log, "Reinitializing logging backend");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const auto& log_dir = GetUserPath(PathType::LogDir);
|
|
||||||
std::filesystem::create_directory(log_dir);
|
|
||||||
Filter filter;
|
|
||||||
filter.ParseFilterString(Config::getLogFilter());
|
|
||||||
instance = std::unique_ptr<Impl, decltype(&Deleter)>(new Impl(log_dir / LOG_FILE, filter),
|
|
||||||
Deleter);
|
|
||||||
initialization_in_progress_suppress_logging = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Start() {
|
|
||||||
instance->StartBackendThread();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Stop() {
|
|
||||||
instance->StopBackendThread();
|
|
||||||
}
|
|
||||||
|
|
||||||
Impl(const Impl&) = delete;
|
|
||||||
Impl& operator=(const Impl&) = delete;
|
|
||||||
|
|
||||||
Impl(Impl&&) = delete;
|
|
||||||
Impl& operator=(Impl&&) = delete;
|
|
||||||
|
|
||||||
void SetGlobalFilter(const Filter& f) {
|
|
||||||
filter = f;
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetColorConsoleBackendEnabled(bool enabled) {
|
|
||||||
color_console_backend.SetEnabled(enabled);
|
|
||||||
}
|
|
||||||
|
|
||||||
void PushEntry(Class log_class, Level log_level, const char* filename, unsigned int line_num,
|
|
||||||
const char* function, std::string message) {
|
|
||||||
// Propagate important log messages to the profiler
|
|
||||||
if (IsProfilerConnected()) {
|
|
||||||
const auto& msg_str = fmt::format("[{}] {}", GetLogClassName(log_class), message);
|
|
||||||
switch (log_level) {
|
|
||||||
case Level::Warning:
|
|
||||||
TRACE_WARN(msg_str);
|
|
||||||
break;
|
|
||||||
case Level::Error:
|
|
||||||
TRACE_ERROR(msg_str);
|
|
||||||
break;
|
|
||||||
case Level::Critical:
|
|
||||||
TRACE_CRIT(msg_str);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!filter.CheckMessage(log_class, log_level)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
using std::chrono::duration_cast;
|
|
||||||
using std::chrono::microseconds;
|
|
||||||
using std::chrono::steady_clock;
|
|
||||||
|
|
||||||
const Entry entry = {
|
|
||||||
.timestamp = duration_cast<microseconds>(steady_clock::now() - time_origin),
|
|
||||||
.log_class = log_class,
|
|
||||||
.log_level = log_level,
|
|
||||||
.filename = filename,
|
|
||||||
.line_num = line_num,
|
|
||||||
.function = function,
|
|
||||||
.message = std::move(message),
|
|
||||||
};
|
|
||||||
if (Config::getLogType() == "async") {
|
|
||||||
message_queue.EmplaceWait(entry);
|
|
||||||
} else {
|
|
||||||
ForEachBackend([&entry](auto& backend) { backend.Write(entry); });
|
|
||||||
std::fflush(stdout);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
Impl(const std::filesystem::path& file_backend_filename, const Filter& filter_)
|
|
||||||
: filter{filter_}, file_backend{file_backend_filename} {}
|
|
||||||
|
|
||||||
~Impl() = default;
|
|
||||||
|
|
||||||
void StartBackendThread() {
|
|
||||||
backend_thread = std::jthread([this](std::stop_token stop_token) {
|
|
||||||
Common::SetCurrentThreadName("shadPS4:Log");
|
|
||||||
Entry entry;
|
|
||||||
const auto write_logs = [this, &entry]() {
|
|
||||||
ForEachBackend([&entry](auto& backend) { backend.Write(entry); });
|
|
||||||
};
|
|
||||||
while (!stop_token.stop_requested()) {
|
|
||||||
message_queue.PopWait(entry, stop_token);
|
|
||||||
if (entry.filename != nullptr) {
|
|
||||||
write_logs();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Drain the logging queue. Only writes out up to MAX_LOGS_TO_WRITE to prevent a
|
|
||||||
// case where a system is repeatedly spamming logs even on close.
|
|
||||||
int max_logs_to_write = filter.IsDebug() ? std::numeric_limits<s32>::max() : 100;
|
|
||||||
while (max_logs_to_write-- && message_queue.TryPop(entry)) {
|
|
||||||
write_logs();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
void StopBackendThread() {
|
|
||||||
backend_thread.request_stop();
|
|
||||||
if (backend_thread.joinable()) {
|
|
||||||
backend_thread.join();
|
|
||||||
}
|
|
||||||
|
|
||||||
ForEachBackend([](auto& backend) { backend.Flush(); });
|
|
||||||
}
|
|
||||||
|
|
||||||
void ForEachBackend(auto lambda) {
|
|
||||||
// lambda(debugger_backend);
|
|
||||||
lambda(color_console_backend);
|
|
||||||
lambda(file_backend);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void Deleter(Impl* ptr) {
|
|
||||||
delete ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline std::unique_ptr<Impl, decltype(&Deleter)> instance{nullptr, Deleter};
|
|
||||||
|
|
||||||
Filter filter;
|
|
||||||
DebuggerBackend debugger_backend{};
|
|
||||||
ColorConsoleBackend color_console_backend{};
|
|
||||||
FileBackend file_backend;
|
|
||||||
|
|
||||||
MPSCQueue<Entry> message_queue{};
|
|
||||||
std::chrono::steady_clock::time_point time_origin{std::chrono::steady_clock::now()};
|
|
||||||
std::jthread backend_thread;
|
|
||||||
};
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
void Initialize(std::string_view log_file) {
|
|
||||||
Impl::Initialize(log_file.empty() ? LOG_FILE : log_file);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Start() {
|
|
||||||
Impl::Start();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Stop() {
|
|
||||||
Impl::Stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetGlobalFilter(const Filter& filter) {
|
|
||||||
Impl::Instance().SetGlobalFilter(filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
void SetColorConsoleBackendEnabled(bool enabled) {
|
|
||||||
Impl::Instance().SetColorConsoleBackendEnabled(enabled);
|
|
||||||
}
|
|
||||||
|
|
||||||
void FmtLogMessageImpl(Class log_class, Level log_level, const char* filename,
|
|
||||||
unsigned int line_num, const char* function, const char* format,
|
|
||||||
const fmt::format_args& args) {
|
|
||||||
if (!initialization_in_progress_suppress_logging) [[likely]] {
|
|
||||||
Impl::Instance().PushEntry(log_class, log_level, filename, line_num, function,
|
|
||||||
fmt::vformat(format, args));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} // namespace Common::Log
|
|
|
@ -1,27 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string_view>
|
|
||||||
#include "common/logging/filter.h"
|
|
||||||
|
|
||||||
namespace Common::Log {
|
|
||||||
|
|
||||||
class Filter;
|
|
||||||
|
|
||||||
/// Initializes the logging system. This should be the first thing called in main.
|
|
||||||
void Initialize(std::string_view log_file = "");
|
|
||||||
|
|
||||||
/// Starts the logging threads.
|
|
||||||
void Start();
|
|
||||||
|
|
||||||
/// Explictily stops the logger thread and flushes the buffers
|
|
||||||
void Stop();
|
|
||||||
|
|
||||||
/// The global filter will prevent any messages from even being processed if they are filtered.
|
|
||||||
void SetGlobalFilter(const Filter& filter);
|
|
||||||
|
|
||||||
void SetColorConsoleBackendEnabled(bool enabled);
|
|
||||||
|
|
||||||
} // namespace Common::Log
|
|
|
@ -1,203 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include "common/assert.h"
|
|
||||||
#include "common/logging/filter.h"
|
|
||||||
|
|
||||||
namespace Common::Log {
|
|
||||||
namespace {
|
|
||||||
template <typename It>
|
|
||||||
Level GetLevelByName(const It begin, const It end) {
|
|
||||||
for (u8 i = 0; i < static_cast<u8>(Level::Count); ++i) {
|
|
||||||
const char* level_name = GetLevelName(static_cast<Level>(i));
|
|
||||||
if (std::string_view(begin, end).compare(level_name) == 0) {
|
|
||||||
return static_cast<Level>(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Level::Count;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename It>
|
|
||||||
Class GetClassByName(const It begin, const It end) {
|
|
||||||
for (u8 i = 0; i < static_cast<u8>(Class::Count); ++i) {
|
|
||||||
const char* level_name = GetLogClassName(static_cast<Class>(i));
|
|
||||||
if (std::string_view(begin, end).compare(level_name) == 0) {
|
|
||||||
return static_cast<Class>(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Class::Count;
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename Iterator>
|
|
||||||
bool ParseFilterRule(Filter& instance, Iterator begin, Iterator end) {
|
|
||||||
auto level_separator = std::find(begin, end, ':');
|
|
||||||
if (level_separator == end) {
|
|
||||||
LOG_ERROR(Log, "Invalid log filter. Must specify a log level after `:`: {}",
|
|
||||||
std::string_view(begin, end));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Level level = GetLevelByName(level_separator + 1, end);
|
|
||||||
if (level == Level::Count) {
|
|
||||||
LOG_ERROR(Log, "Unknown log level in filter: {}", std::string_view(begin, end));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (std::string_view(begin, level_separator).compare("*") == 0) {
|
|
||||||
instance.ResetAll(level);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Class log_class = GetClassByName(begin, level_separator);
|
|
||||||
if (log_class == Class::Count) {
|
|
||||||
LOG_ERROR(Log, "Unknown log class in filter: {}", std::string(begin, end));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
instance.SetClassLevel(log_class, level);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} // Anonymous namespace
|
|
||||||
|
|
||||||
/// Macro listing all log classes. Code should define CLS and SUB as desired before invoking this.
|
|
||||||
#define ALL_LOG_CLASSES() \
|
|
||||||
CLS(Log) \
|
|
||||||
CLS(Common) \
|
|
||||||
SUB(Common, Filesystem) \
|
|
||||||
SUB(Common, Memory) \
|
|
||||||
CLS(Core) \
|
|
||||||
SUB(Core, Linker) \
|
|
||||||
CLS(Config) \
|
|
||||||
CLS(Debug) \
|
|
||||||
CLS(Kernel) \
|
|
||||||
SUB(Kernel, Pthread) \
|
|
||||||
SUB(Kernel, Vmm) \
|
|
||||||
SUB(Kernel, Fs) \
|
|
||||||
SUB(Kernel, Event) \
|
|
||||||
SUB(Kernel, Sce) \
|
|
||||||
CLS(Lib) \
|
|
||||||
SUB(Lib, LibC) \
|
|
||||||
SUB(Lib, Kernel) \
|
|
||||||
SUB(Lib, Pad) \
|
|
||||||
SUB(Lib, GnmDriver) \
|
|
||||||
SUB(Lib, SystemService) \
|
|
||||||
SUB(Lib, UserService) \
|
|
||||||
SUB(Lib, VideoOut) \
|
|
||||||
SUB(Lib, CommonDlg) \
|
|
||||||
SUB(Lib, MsgDlg) \
|
|
||||||
SUB(Lib, AudioOut) \
|
|
||||||
SUB(Lib, AudioIn) \
|
|
||||||
SUB(Lib, Net) \
|
|
||||||
SUB(Lib, NetCtl) \
|
|
||||||
SUB(Lib, SaveData) \
|
|
||||||
SUB(Lib, SaveDataDialog) \
|
|
||||||
SUB(Lib, Http) \
|
|
||||||
SUB(Lib, Ssl) \
|
|
||||||
SUB(Lib, SysModule) \
|
|
||||||
SUB(Lib, NpManager) \
|
|
||||||
SUB(Lib, NpScore) \
|
|
||||||
SUB(Lib, NpTrophy) \
|
|
||||||
SUB(Lib, Screenshot) \
|
|
||||||
SUB(Lib, LibCInternal) \
|
|
||||||
SUB(Lib, AppContent) \
|
|
||||||
SUB(Lib, Rtc) \
|
|
||||||
SUB(Lib, DiscMap) \
|
|
||||||
SUB(Lib, Png) \
|
|
||||||
SUB(Lib, PlayGo) \
|
|
||||||
SUB(Lib, Random) \
|
|
||||||
SUB(Lib, Usbd) \
|
|
||||||
SUB(Lib, Ajm) \
|
|
||||||
SUB(Lib, ErrorDialog) \
|
|
||||||
SUB(Lib, ImeDialog) \
|
|
||||||
SUB(Lib, AvPlayer) \
|
|
||||||
SUB(Lib, Ngs2) \
|
|
||||||
CLS(Frontend) \
|
|
||||||
CLS(Render) \
|
|
||||||
SUB(Render, Vulkan) \
|
|
||||||
SUB(Render, Recompiler) \
|
|
||||||
CLS(Input) \
|
|
||||||
CLS(Tty) \
|
|
||||||
CLS(Loader)
|
|
||||||
|
|
||||||
// GetClassName is a macro defined by Windows.h, grrr...
|
|
||||||
const char* GetLogClassName(Class log_class) {
|
|
||||||
switch (log_class) {
|
|
||||||
#define CLS(x) \
|
|
||||||
case Class::x: \
|
|
||||||
return #x;
|
|
||||||
#define SUB(x, y) \
|
|
||||||
case Class::x##_##y: \
|
|
||||||
return #x "." #y;
|
|
||||||
ALL_LOG_CLASSES()
|
|
||||||
#undef CLS
|
|
||||||
#undef SUB
|
|
||||||
case Class::Count:
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* GetLevelName(Level log_level) {
|
|
||||||
#define LVL(x) \
|
|
||||||
case Level::x: \
|
|
||||||
return #x
|
|
||||||
switch (log_level) {
|
|
||||||
LVL(Trace);
|
|
||||||
LVL(Debug);
|
|
||||||
LVL(Info);
|
|
||||||
LVL(Warning);
|
|
||||||
LVL(Error);
|
|
||||||
LVL(Critical);
|
|
||||||
case Level::Count:
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#undef LVL
|
|
||||||
UNREACHABLE();
|
|
||||||
}
|
|
||||||
|
|
||||||
Filter::Filter(Level default_level) {
|
|
||||||
ResetAll(default_level);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Filter::ResetAll(Level level) {
|
|
||||||
class_levels.fill(level);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Filter::SetClassLevel(Class log_class, Level level) {
|
|
||||||
class_levels[static_cast<std::size_t>(log_class)] = level;
|
|
||||||
}
|
|
||||||
|
|
||||||
void Filter::ParseFilterString(std::string_view filter_view) {
|
|
||||||
auto clause_begin = filter_view.cbegin();
|
|
||||||
while (clause_begin != filter_view.cend()) {
|
|
||||||
auto clause_end = std::find(clause_begin, filter_view.cend(), ' ');
|
|
||||||
|
|
||||||
// If clause isn't empty
|
|
||||||
if (clause_end != clause_begin) {
|
|
||||||
ParseFilterRule(*this, clause_begin, clause_end);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clause_end != filter_view.cend()) {
|
|
||||||
// Skip over the whitespace
|
|
||||||
++clause_end;
|
|
||||||
}
|
|
||||||
clause_begin = clause_end;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Filter::CheckMessage(Class log_class, Level level) const {
|
|
||||||
return static_cast<u8>(level) >=
|
|
||||||
static_cast<u8>(class_levels[static_cast<std::size_t>(log_class)]);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Filter::IsDebug() const {
|
|
||||||
return std::any_of(class_levels.begin(), class_levels.end(), [](const Level& l) {
|
|
||||||
return static_cast<u8>(l) <= static_cast<u8>(Level::Debug);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Common::Log
|
|
|
@ -1,66 +0,0 @@
|
||||||
// SPDX-FileCopyrightText: Copyright 2014 Citra Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <cstddef>
|
|
||||||
#include <string_view>
|
|
||||||
#include "common/logging/types.h"
|
|
||||||
|
|
||||||
namespace Common::Log {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the name of the passed log class as a C-string. Subclasses are separated by periods
|
|
||||||
* instead of underscores as in the enumeration.
|
|
||||||
*/
|
|
||||||
const char* GetLogClassName(Class log_class);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the name of the passed log level as a C-string.
|
|
||||||
*/
|
|
||||||
const char* GetLevelName(Level log_level);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Implements a log message filter which allows different log classes to have different minimum
|
|
||||||
* severity levels. The filter can be changed at runtime and can be parsed from a string to allow
|
|
||||||
* editing via the interface or loading from a configuration file.
|
|
||||||
*/
|
|
||||||
class Filter {
|
|
||||||
public:
|
|
||||||
/// Initializes the filter with all classes having `default_level` as the minimum level.
|
|
||||||
explicit Filter(Level default_level = Level::Info);
|
|
||||||
|
|
||||||
/// Resets the filter so that all classes have `level` as the minimum displayed level.
|
|
||||||
void ResetAll(Level level);
|
|
||||||
|
|
||||||
/// Sets the minimum level of `log_class` (and not of its subclasses) to `level`.
|
|
||||||
void SetClassLevel(Class log_class, Level level);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Parses a filter string and applies it to this filter.
|
|
||||||
*
|
|
||||||
* A filter string consists of a space-separated list of filter rules, each of the format
|
|
||||||
* `<class>:<level>`. `<class>` is a log class name, with subclasses separated using periods.
|
|
||||||
* `*` is allowed as a class name and will reset all filters to the specified level. `<level>`
|
|
||||||
* a severity level name which will be set as the minimum logging level of the matched classes.
|
|
||||||
* Rules are applied left to right, with each rule overriding previous ones in the sequence.
|
|
||||||
*
|
|
||||||
* A few examples of filter rules:
|
|
||||||
* - `*:Info` -- Resets the level of all classes to Info.
|
|
||||||
* - `Service:Info` -- Sets the level of Service to Info.
|
|
||||||
* - `Service.FS:Trace` -- Sets the level of the Service.FS class to Trace.
|
|
||||||
*/
|
|
||||||
void ParseFilterString(std::string_view filter_view);
|
|
||||||
|
|
||||||
/// Matches class/level combination against the filter, returning true if it passed.
|
|
||||||
bool CheckMessage(Class log_class, Level level) const;
|
|
||||||
|
|
||||||
/// Returns true if any logging classes are set to debug
|
|
||||||
bool IsDebug() const;
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::array<Level, static_cast<std::size_t>(Class::Count)> class_levels;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Common::Log
|
|