Compare commits
316 Commits
miscFixes8
...
main
Author | SHA1 | Date |
---|---|---|
psucien | 4182740384 | |
psucien | ca1613258f | |
georgemoralis | 3d375a28eb | |
jnack | 69d4fecdfe | |
Xphalnos | 7886761476 | |
squidbus | 6080066f75 | |
georgemoralis | f1fe6b9f96 | |
squidbus | 6e552aac6a | |
adjonesey | 0f87d1e3d4 | |
georgemoralis | 68fa5e292f | |
bigol83 | bdfff5e8ea | |
georgemoralis | cdd193d5b1 | |
georgemoralis | 30ab2b7f71 | |
DanielSvoboda | f2b6843f9d | |
squidbus | a17150960f | |
Grégoire Hage | 1651db24fe | |
IndecisiveTurtle | 6bf42aa985 | |
georgemoralis | 8d41695e74 | |
IndecisiveTurtle | cb5190e31a | |
IndecisiveTurtle | cf706f8cc7 | |
IndecisiveTurtle | 6fbbe3d79b | |
IndecisiveTurtle | fab390b860 | |
Plínio Larrubia | 5a96ac1a4f | |
georgemoralis | ac2fa103fa | |
georgemoralis | 7e7854346d | |
DanielSvoboda | 751f6f9bab | |
Flamy | 1c835a1aa4 | |
georgemoralis | 07c8c28000 | |
TheTurtle | 66e96dd944 | |
DanielSvoboda | aef7498c49 | |
georgemoralis | 790d19e59b | |
georgemoralis | 3ae18c9073 | |
georgemoralis | 83e343f77e | |
kiwipuppeh | 81eb3a4d14 | |
Marcin Mitura | 37786e06c2 | |
psucien | 9d349a1308 | |
bigol83 | c997490645 | |
bigol83 | 600c5079ce | |
Dzmitry Dubrova | 8827c72a1c | |
¥IGA | e1382b43c8 | |
DanielSvoboda | 614a23b369 | |
georgemoralis | a6a9fff666 | |
psucien | 3fbb68048e | |
CyntexMore | 55be97bca3 | |
georgemoralis | d2e4a200fb | |
georgemoralis | a291800418 | |
bigol83 | 365e17f64e | |
Emulator-Team-2 | af18728962 | |
Emulator-Team-2 | 6f938a10ce | |
Emulator-Team-2 | 08036d2b49 | |
georgemoralis | e57d55e6e9 | |
georgemoralis | 163019e3dc | |
counter185 | bc661039c5 | |
counter185 | 2e2f3bb2c6 | |
counter185 | 89bdd3bba2 | |
counter185 | 15fc267f0d | |
counter185 | 0b24ac6991 | |
counter185 | 698dade864 | |
counter185 | 4e6e90dfb9 | |
georgemoralis | be49871c68 | |
squidbus | 905d49fd96 | |
georgemoralis | dcc4762c7e | |
squidbus | 4a6ce1e0d6 | |
georgemoralis | 878fac7f94 | |
squidbus | da966bedfd | |
Vinicius Rangel | 8192eaa668 | |
0xsegf4ult | 9f4e55a8e7 | |
InvoxiPlayGames | 9d59ea06c7 | |
tGecko | 990da7edcc | |
InvoxiPlayGames | 550cf0188d | |
InvoxiPlayGames | 79e86a39fc | |
Anton Kovalev | dfb30ea955 | |
Random | c37679154e | |
Anton Kovalev | 87ccfdfbbd | |
Exhigh | ed42db47af | |
Anton Kovalev | 1a02efbd15 | |
georgemoralis | e4254ebdaa | |
georgemoralis | e65bf19118 | |
Semenov Herman | 860b935054 | |
Anton Kovalev | 3842993a43 | |
Anton Kovalev | 3d46a5d492 | |
Anton Kovalev | 595b845df0 | |
Anton Kovalev | 659e7a4675 | |
psucien | 3eb47226d9 | |
psucien | 371d1d009a | |
Lizardy | 59b651be07 | |
Grégoire Hage | 288db9a0cf | |
georgemoralis | 62f165a7e4 | |
Vladislav Mikhalin | 2d354a095a | |
SolidStateDj | 7ecc1d3be0 | |
psucien | af4356bfe1 | |
psucien | 3e94b533b6 | |
georgemoralis | 287c3a4ab1 | |
Exhigh | 7c7d04133e | |
Plínio Larrubia | ad8373095a | |
Lizardy | aae6e5be73 | |
Batu | cd6b5abccf | |
Plínio Larrubia | 6520f3ca17 | |
Cristóbal Martí Méndez | d3b6b17c75 | |
Marcin Mitura | 0b5616c493 | |
georgemoralis | 3627393707 | |
nishinji | 6e340bcdc1 | |
psucien | 78e24852f8 | |
georgemoralis | 7f6d08a3c8 | |
nishinji | 7aa8cf992f | |
¥IGA | e16aa2a012 | |
greggameplayer | 86870e7c8d | |
Daniel R. | fae0c0ae85 | |
Yury | 5d7407dc7d | |
bigol83 | bcc3a10557 | |
psucien | 6fd4264b5d | |
DanielSvoboda | 2a737d0800 | |
psucien | b687ae5e34 | |
georgemoralis | c2ddfe51e1 | |
georgemoralis | abc7f55c06 | |
Dzmitry Dubrova | 26b5661dea | |
SaynedBread | 6a87c37aeb | |
SaynedBread | 5664d45305 | |
Dzmitry Dubrova | 21ce86003b | |
Dzmitry Dubrova | 83b43363fa | |
¥IGA | e603165e2b | |
Dzmitry Dubrova | 7766a653d5 | |
Dzmitry Dubrova | 3f445eca40 | |
SaynedBread | 41a93dcae4 | |
SaynedBread | 1e90f45941 | |
TheTurtle | c79b10edc1 | |
Vinicius Rangel | 9e4fc17e6c | |
georgemoralis | 208575d392 | |
georgemoralis | 8b0fd59149 | |
Vladislav Mikhalin | 41dec15869 | |
Exhigh | 6a84f6e188 | |
Herman Semenov | 243fd0be78 | |
Random | fc745ee767 | |
psucien | 2c540fbecb | |
georgemoralis | 0b3a88cf22 | |
Matthew Wells | 0ed4614464 | |
georgemoralis | 9852f95c58 | |
georgemoralis | df03783948 | |
¥IGA | 0c5b91e1fb | |
Herman Semenov | a0774c0e87 | |
Alexandre Bouvier | a71d0d8dbb | |
georgemoralis | fda2fdae69 | |
Exhigh | 6545b09b74 | |
georgemoralis | bce3a9c9e7 | |
georgemoralis | 3310fd9a1f | |
georgemoralis | 3e8d7c2040 | |
Xphalnos | d4be3dbb31 | |
georgemoralis | 83bcced16c | |
georgemoralis | 746559658c | |
georgemoralis | 12a65e3fb8 | |
georgemoralis | 834a25fa2b | |
Herman Semenov | aed9a737d6 | |
Aiden Turner | ca4b520272 | |
Vladislav Mikhalin | 79680c50c0 | |
georgemoralis | dfd305ff77 | |
Sebastian Kassai | ba0a6ab038 | |
georgemoralis | 132ca9c5a8 | |
georgemoralis | 3426ad8cc0 | |
Dzmitry Dubrova | 6d0d2eaa59 | |
squidbus | 9275b0966e | |
Borchev | fc300b5265 | |
Borchev | 6596fe091c | |
georgemoralis | 958db559c6 | |
TheTurtle | 3f9c86ad33 | |
Lizardy | 32cb3649d3 | |
xezrunner | 42c4d8353a | |
jdp_ | 6080649b7c | |
jdp_ | 34a1339a2b | |
jdp_ | e070dab2f0 | |
kotn3l | c60bfbe2a5 | |
georgemoralis | 95c89ca6a9 | |
georgemoralis | 8d90e6d997 | |
bax-cz | 0b3356bd1a | |
bax-cz | 8f7b3c2e8c | |
Random | 09da94b7b2 | |
¥IGA | 9d45b99171 | |
Dzmitry Dubrova | 1f416134e7 | |
bax-cz | 516a3e7104 | |
Lizardy | 138c9ce787 | |
georgemoralis | 360b7e1d71 | |
DanielSvoboda | d5e978c6f2 | |
georgemoralis | 09b71a9a3f | |
DanielSvoboda | c58ad6d3b5 | |
DanielSvoboda | c9a502b31a | |
georgemoralis | 5551c61000 | |
DanielSvoboda | ed96a9fb8e | |
DanielSvoboda | 5891900c6e | |
Vladislav Mikhalin | 1c898d0842 | |
Xphalnos | 70576035b0 | |
Lizardy | 63938ba8dd | |
¥IGA | 3be2e4b2b8 | |
georgemoralis | ecf7f36763 | |
georgemoralis | 43e92bb987 | |
Xphalnos | 73adc3ed1b | |
Stephen Miller | 2935ca0fef | |
Stephen Miller | 9fce6f7c01 | |
Stephen Miller | 93f14e8ae9 | |
Stephen Miller | 6510af90be | |
Stephen Miller | f36440dc09 | |
Stephen Miller | 786db80742 | |
Xphalnos | 558fcf6597 | |
Dzmitry Dubrova | dcb057dd7f | |
Dzmitry Dubrova | 9e810b7524 | |
TheTurtle | 1d1c88ad31 | |
Vinicius Rangel | ff33b00c3a | |
Alexandre Bouvier | 154771cca5 | |
Vinicius Rangel | 444cdfbba5 | |
Alexandre Bouvier | ad60ae1d40 | |
Herman Semenov | c1fb5d5bca | |
georgemoralis | 3197ad336e | |
Xphalnos | 910e96c420 | |
Vladislav Mikhalin | 23dddca1f0 | |
georgemoralis | 5c70adc6ab | |
squidbus | 62741434db | |
Vladislav Mikhalin | b3ef959b25 | |
Vladislav Mikhalin | 5c4ac98d49 | |
Vladislav Mikhalin | 0d6e8e227a | |
Vladislav Mikhalin | e33ff10212 | |
Vladislav Mikhalin | b5c69189e5 | |
georgemoralis | e96e66eedd | |
georgemoralis | da9b26fa1e | |
psucien | d32e584839 | |
¥IGA | 0b1d7839a3 | |
Vinicius Rangel | d45563f92c | |
¥IGA | a0fb47b0ab | |
¥IGA | 8c77d4dde6 | |
psucien | 9adc638220 | |
Dzmitry Dubrova | 6f4e1a47b9 | |
Stephen Miller | 5f963772a0 | |
TheTurtle | d332a5e611 | |
georgemoralis | 4b11dabd9e | |
Daniel R. | 6cc4a682fd | |
psucien | 27cb218584 | |
Samuel Fontes | ad3b6c793c | |
TheTurtle | d8b9d82ffa | |
counter185 | bb159eafb9 | |
squidbus | d1a033b6af | |
TheTurtle | 1fb0da9b89 | |
Vinicius Rangel | dfcfd62d4f | |
Lizardy | 5eecd089ab | |
Borchev | 18f1799280 | |
psucien | a8b6a55559 | |
squidbus | 284035d3e2 | |
psucien | a15a93997c | |
psucien | 2ba3221fc9 | |
psucien | 3d0fdf11f0 | |
psucien | ace39957ef | |
psucien | 3e2d4d6b79 | |
georgemoralis | 834e3a500e | |
georgemoralis | 924e9de97a | |
Dzmitry Dubrova | 3163cd135b | |
Dzmitry Dubrova | 96fb00d411 | |
Leonardo | 14947232a7 | |
georgemoralis | ea4ae56f4d | |
georgemoralis | 54b20e2938 | |
georgemoralis | 61a6f633fd | |
georgemoralis | d81dbc5b5b | |
georgemoralis | 425e5491a8 | |
Leonardo | 0d56be629b | |
Leonardo | 4375e6fa3a | |
georgemoralis | 250b2e4969 | |
georgemoralis | 6a2e09a1df | |
georgemoralis | 3ef69cae5e | |
georgemoralis | 7b7d1cb26f | |
georgemoralis | a83ac4c05e | |
georgemoralis | 816700d34d | |
georgemoralis | 4ffb812e94 | |
georgemoralis | 13331cdda9 | |
georgemoralis | 5a68224a13 | |
SleepingSnakezzz | ab56665d4b | |
Stephen Miller | 48c58d5ce0 | |
georgemoralis | 3943d13871 | |
georgemoralis | c96854f1f1 | |
georgemoralis | e5087877ae | |
georgemoralis | 14e56e267a | |
SamuelFontes | 564b2f6310 | |
SamuelFontes | 7e5cc6162c | |
georgemoralis | 81be8f270f | |
ElBread3 | 351f2e1073 | |
Xphalnos | 254b9ffb50 | |
IndecisiveTurtle | 3fd2abdd5b | |
TheTurtle | 341034fc30 | |
TheTurtle | 381ba8c7a5 | |
Leonardo | ddec111da6 | |
TheTurtle | 159be2c7f4 | |
DanielSvoboda | cdff4af38d | |
DanielSvoboda | 7308864537 | |
Xphalnos | 21462523de | |
Alexandre Bouvier | 6d0a763145 | |
georgemoralis | 9f2e73c1fd | |
georgemoralis | 305224650a | |
squidbus | be829389ea | |
squidbus | 56b362bb24 | |
georgemoralis | 54e831746d | |
georgemoralis | bd48e24c32 | |
georgemoralis | 62e501f31f | |
Xphalnos | a4e46c4362 | |
Xphalnos | 9968d03b90 | |
georgemoralis | 0ac9f8ffa4 | |
georgemoralis | a121e8e76a | |
Xphalnos | 205c0b961b | |
Dzmitry Dubrova | 0fe766db6c | |
Xphalnos | a7f2f09a44 | |
raziel1000 | 413402600b | |
raziel1000 | ec1335911b | |
raziel1000 | 51c89a9958 | |
raziel1000 | d017bab21e | |
raziel1000 | d7acb93d6f | |
TheTurtle | a7c9bfa5c5 | |
georgemoralis | ac6dc20c3b | |
Xphalnos | b3525d7f79 | |
Borchev | 361271826e | |
Borchev | 867f38fe13 | |
georgemoralis | c1d01709be | |
georgemoralis | 9b1092c995 | |
squidbus | 43d60a8ac9 |
|
@ -19,8 +19,12 @@ 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
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 197 KiB After Width: | Height: | Size: 130 KiB |
|
@ -10,7 +10,6 @@ on:
|
||||||
branches: [ "main" ]
|
branches: [ "main" ]
|
||||||
|
|
||||||
env:
|
env:
|
||||||
# Customize the CMake build type here (Release, Debug, RelWithDebInfo, etc.)
|
|
||||||
BUILD_TYPE: Release
|
BUILD_TYPE: Release
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
@ -19,12 +18,12 @@ jobs:
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Fetch submodules
|
with:
|
||||||
run: git submodule update --init --recursive
|
submodules: 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
|
sudo apt-get update && sudo apt install libx11-dev libxext-dev libwayland-dev libfuse2 clang build-essential qt6-base-dev qt6-tools-dev
|
||||||
|
|
||||||
- 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 -DCMAKE_CXX_COMPILER=clang++ -DENABLE_QT_GUI=ON
|
||||||
|
|
|
@ -8,10 +8,8 @@ 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
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
|
@ -8,10 +8,8 @@ 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
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
@ -36,10 +34,11 @@ jobs:
|
||||||
- name: Setup Qt
|
- name: Setup Qt
|
||||||
uses: jurplel/install-qt-action@v4
|
uses: jurplel/install-qt-action@v4
|
||||||
with:
|
with:
|
||||||
|
version: 6.7.2
|
||||||
host: mac
|
host: mac
|
||||||
target: desktop
|
target: desktop
|
||||||
arch: clang_64
|
arch: clang_64
|
||||||
version: 6.7.2
|
archives: qtbase qttools
|
||||||
|
|
||||||
- name: Configure CMake
|
- name: Configure CMake
|
||||||
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_OSX_ARCHITECTURES=x86_64 -DENABLE_QT_GUI=ON
|
run: cmake -B ${{github.workspace}}/build -DCMAKE_BUILD_TYPE=${{env.BUILD_TYPE}} -DCMAKE_OSX_ARCHITECTURES=x86_64 -DENABLE_QT_GUI=ON
|
||||||
|
@ -51,6 +50,7 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
mkdir upload
|
mkdir upload
|
||||||
mv ${{github.workspace}}/build/shadps4.app upload
|
mv ${{github.workspace}}/build/shadps4.app upload
|
||||||
|
mv ${{github.workspace}}/build/translations upload
|
||||||
macdeployqt upload/shadps4.app
|
macdeployqt upload/shadps4.app
|
||||||
tar cf shadps4-macos-qt.tar.gz -C upload .
|
tar cf shadps4-macos-qt.tar.gz -C upload .
|
||||||
|
|
||||||
|
|
|
@ -8,10 +8,8 @@ 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
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
@ -45,10 +43,10 @@ jobs:
|
||||||
mv ${{github.workspace}}/build/shadps4 upload
|
mv ${{github.workspace}}/build/shadps4 upload
|
||||||
cp $(arch -x86_64 /usr/local/bin/brew --prefix)/opt/molten-vk/lib/libMoltenVK.dylib 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
|
install_name_tool -add_rpath "@loader_path" upload/shadps4
|
||||||
tar cf shadps4-macos.tar.gz -C upload .
|
tar cf shadps4-macos-sdl.tar.gz -C upload .
|
||||||
|
|
||||||
- name: Upload executable
|
- name: Upload executable
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: shadps4-macos
|
name: shadps4-macos-sdl
|
||||||
path: shadps4-macos.tar.gz
|
path: shadps4-macos-sdl.tar.gz
|
||||||
|
|
|
@ -10,12 +10,8 @@ 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
|
||||||
|
@ -28,22 +24,23 @@ jobs:
|
||||||
- name: Setup Qt
|
- name: Setup Qt
|
||||||
uses: jurplel/install-qt-action@v4
|
uses: jurplel/install-qt-action@v4
|
||||||
with:
|
with:
|
||||||
arch: win64_msvc2019_64
|
|
||||||
version: 6.7.2
|
version: 6.7.2
|
||||||
|
host: windows
|
||||||
|
target: desktop
|
||||||
|
arch: win64_msvc2019_64
|
||||||
|
archives: qtbase qttools
|
||||||
|
|
||||||
- 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
|
||||||
windeployqt --dir upload upload/shadPS4.exe
|
windeployqt --dir upload upload/shadPS4.exe
|
||||||
|
|
||||||
- name: Upload executable
|
- name: Upload executable
|
||||||
|
|
|
@ -10,12 +10,8 @@ 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
|
||||||
|
@ -25,16 +21,14 @@ jobs:
|
||||||
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@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: shadps4-win64
|
name: shadps4-win64-sdl
|
||||||
# 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
|
||||||
|
|
|
@ -408,3 +408,4 @@ FodyWeavers.xsd
|
||||||
/emulator/eboot.bin
|
/emulator/eboot.bin
|
||||||
/out/*
|
/out/*
|
||||||
/third-party/out/*
|
/third-party/out/*
|
||||||
|
/src/common/scm_rev.cpp
|
||||||
|
|
|
@ -1,63 +1,87 @@
|
||||||
[submodule "externals/discord-rpc"]
|
|
||||||
path = externals/discord-rpc
|
|
||||||
url = https://github.com/shadps4-emu/ext-discord-rpc.git
|
|
||||||
[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
|
||||||
[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
|
||||||
[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
|
||||||
[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/sdl3"]
|
[submodule "externals/sdl3"]
|
||||||
path = externals/sdl3
|
path = externals/sdl3
|
||||||
url = https://github.com/shadps4-emu/ext-SDL.git
|
url = https://github.com/shadps4-emu/ext-SDL.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/fmt"]
|
[submodule "externals/fmt"]
|
||||||
path = externals/fmt
|
path = externals/fmt
|
||||||
url = https://github.com/shadps4-emu/ext-fmt.git
|
url = https://github.com/shadps4-emu/ext-fmt.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/vulkan-headers"]
|
[submodule "externals/vulkan-headers"]
|
||||||
path = externals/vulkan-headers
|
path = externals/vulkan-headers
|
||||||
url = https://github.com/KhronosGroup/Vulkan-Headers.git
|
url = https://github.com/KhronosGroup/Vulkan-Headers.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/vma"]
|
[submodule "externals/vma"]
|
||||||
path = externals/vma
|
path = externals/vma
|
||||||
url = https://github.com/GPUOpen-LibrariesAndSDKs/VulkanMemoryAllocator.git
|
url = https://github.com/GPUOpen-LibrariesAndSDKs/VulkanMemoryAllocator.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/glslang"]
|
[submodule "externals/glslang"]
|
||||||
path = externals/glslang
|
path = externals/glslang
|
||||||
url = https://github.com/KhronosGroup/glslang.git
|
url = https://github.com/KhronosGroup/glslang.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/robin-map"]
|
[submodule "externals/robin-map"]
|
||||||
path = externals/robin-map
|
path = externals/robin-map
|
||||||
url = https://github.com/Tessil/robin-map.git
|
url = https://github.com/Tessil/robin-map.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/xbyak"]
|
[submodule "externals/xbyak"]
|
||||||
path = externals/xbyak
|
path = externals/xbyak
|
||||||
url = https://github.com/herumi/xbyak.git
|
url = https://github.com/herumi/xbyak.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/winpthreads"]
|
[submodule "externals/winpthreads"]
|
||||||
path = externals/winpthreads
|
path = externals/winpthreads
|
||||||
url = https://github.com/shadps4-emu/winpthreads.git
|
url = https://github.com/shadps4-emu/winpthreads.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/magic_enum"]
|
[submodule "externals/magic_enum"]
|
||||||
path = externals/magic_enum
|
path = externals/magic_enum
|
||||||
url = https://github.com/Neargye/magic_enum.git
|
url = https://github.com/Neargye/magic_enum.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/toml11"]
|
[submodule "externals/toml11"]
|
||||||
path = externals/toml11
|
path = externals/toml11
|
||||||
url = https://github.com/ToruNiina/toml11.git
|
url = https://github.com/ToruNiina/toml11.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/zydis"]
|
[submodule "externals/zydis"]
|
||||||
path = externals/zydis
|
path = externals/zydis
|
||||||
url = https://github.com/zyantific/zydis.git
|
url = https://github.com/zyantific/zydis.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/sirit"]
|
[submodule "externals/sirit"]
|
||||||
path = externals/sirit
|
path = externals/sirit
|
||||||
url = https://github.com/shadps4-emu/sirit.git
|
url = https://github.com/shadps4-emu/sirit.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/xxhash"]
|
[submodule "externals/xxhash"]
|
||||||
path = externals/xxhash
|
path = externals/xxhash
|
||||||
url = https://github.com/Cyan4973/xxHash.git
|
url = https://github.com/Cyan4973/xxHash.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/tracy"]
|
[submodule "externals/tracy"]
|
||||||
path = externals/tracy
|
path = externals/tracy
|
||||||
url = https://github.com/shadps4-emu/tracy.git
|
url = https://github.com/shadps4-emu/tracy.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/ext-boost"]
|
[submodule "externals/ext-boost"]
|
||||||
path = externals/ext-boost
|
path = externals/ext-boost
|
||||||
url = https://github.com/shadps4-emu/ext-boost.git
|
url = https://github.com/shadps4-emu/ext-boost.git
|
||||||
|
shallow = true
|
||||||
[submodule "externals/date"]
|
[submodule "externals/date"]
|
||||||
path = externals/date
|
path = externals/date
|
||||||
url = https://github.com/HowardHinnant/date.git
|
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
|
||||||
|
|
16
.reuse/dep5
16
.reuse/dep5
|
@ -14,9 +14,8 @@ Files: CMakeSettings.json
|
||||||
documents/Screenshots/Sonic Mania.png
|
documents/Screenshots/Sonic Mania.png
|
||||||
documents/Screenshots/Undertale.png
|
documents/Screenshots/Undertale.png
|
||||||
documents/Screenshots/We are DOOMED.png
|
documents/Screenshots/We are DOOMED.png
|
||||||
externals/stb_image.h
|
|
||||||
externals/tracy/*
|
|
||||||
scripts/ps4_names.txt
|
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
|
||||||
|
@ -36,6 +35,7 @@ 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/shadps4.ico
|
||||||
src/images/themes_icon.png
|
src/images/themes_icon.png
|
||||||
src/shadps4.qrc
|
src/shadps4.qrc
|
||||||
|
@ -43,6 +43,18 @@ Files: CMakeSettings.json
|
||||||
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/*
|
Files: externals/renderdoc/*
|
||||||
Copyright: 2019-2024 Baldur Karlsson
|
Copyright: 2019-2024 Baldur Karlsson
|
||||||
License: MIT
|
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
|
||||||
|
|
246
CMakeLists.txt
246
CMakeLists.txt
|
@ -19,12 +19,11 @@ 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
|
# This function should be passed a list of all files in a target. It will automatically generate file groups
|
||||||
# file groups following the directory hierarchy, so that the layout of the files in IDEs matches the
|
# following the directory hierarchy, so that the layout of the files in IDEs matches the one in the filesystem.
|
||||||
# one in the filesystem.
|
|
||||||
function(create_target_directory_groups target_name)
|
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.
|
# 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)
|
||||||
|
@ -39,14 +38,6 @@ 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 "-17")
|
|
||||||
find_program(CLANG_FORMAT
|
|
||||||
NAMES clang-format${CLANG_FORMAT_POSTFIX}
|
|
||||||
clang-format
|
|
||||||
PATHS ${PROJECT_BINARY_DIR}/externals)
|
|
||||||
|
|
||||||
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/")
|
||||||
|
@ -65,40 +56,80 @@ endif()
|
||||||
|
|
||||||
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake")
|
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(Boost 1.84.0 CONFIG)
|
||||||
find_package(cryptopp 8.9.0 MODULE)
|
find_package(FFmpeg 5.1.2 MODULE)
|
||||||
find_package(fmt 10.2.1 CONFIG)
|
find_package(fmt 10.2.0 CONFIG)
|
||||||
find_package(glslang 14.2.0 CONFIG)
|
find_package(glslang 14.2.0 CONFIG)
|
||||||
find_package(magic_enum 0.9.6 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(SDL3 3.1.2 CONFIG)
|
||||||
find_package(toml11 3.8.1 CONFIG)
|
find_package(toml11 4.2.0 CONFIG)
|
||||||
find_package(tsl-robin-map 1.3.0 CONFIG)
|
find_package(tsl-robin-map 1.3.0 CONFIG)
|
||||||
find_package(VulkanHeaders 1.3.289 CONFIG)
|
find_package(VulkanHeaders 1.3.289 CONFIG)
|
||||||
find_package(VulkanMemoryAllocator 3.1.0 CONFIG)
|
find_package(VulkanMemoryAllocator 3.1.0 CONFIG)
|
||||||
find_package(xbyak 7.07 CONFIG)
|
find_package(xbyak 7.07 CONFIG)
|
||||||
find_package(xxHash 0.8.2 MODULE)
|
find_package(xxHash 0.8.2 MODULE)
|
||||||
find_package(zlib-ng 2.2.0 MODULE)
|
find_package(zlib-ng 2.1.7 MODULE)
|
||||||
find_package(Zydis 4.1.0 CONFIG)
|
find_package(Zydis 5.0.0 CONFIG)
|
||||||
find_package(RenderDoc MODULE)
|
|
||||||
|
if (NOT CMAKE_CXX_COMPILER_ID STREQUAL "Clang" OR NOT MSVC)
|
||||||
|
find_package(cryptopp 8.9.0 MODULE)
|
||||||
|
endif()
|
||||||
|
|
||||||
if (APPLE)
|
if (APPLE)
|
||||||
find_package(date 3.0.1 CONFIG)
|
find_package(date 3.0.1 CONFIG)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
# Note: Windows always has these functions through winpthreads
|
||||||
include(CheckSymbolExists)
|
include(CheckSymbolExists)
|
||||||
check_symbol_exists(pthread_mutex_timedlock "pthread.h" HAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
check_symbol_exists(pthread_mutex_timedlock "pthread.h" HAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
||||||
# Windows always has the function through winpthreads
|
|
||||||
if(HAVE_PTHREAD_MUTEX_TIMEDLOCK OR WIN32)
|
if(HAVE_PTHREAD_MUTEX_TIMEDLOCK OR WIN32)
|
||||||
add_compile_options(-DHAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
add_compile_options(-DHAVE_PTHREAD_MUTEX_TIMEDLOCK)
|
||||||
endif()
|
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)
|
||||||
include_directories(src)
|
include_directories(src)
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
find_package(Qt6 REQUIRED COMPONENTS Widgets Concurrent)
|
find_package(Qt6 REQUIRED COMPONENTS Widgets Concurrent LinguistTools Network)
|
||||||
qt_standard_project_setup()
|
qt_standard_project_setup()
|
||||||
set(CMAKE_AUTORCC ON)
|
set(CMAKE_AUTORCC ON)
|
||||||
|
set(CMAKE_AUTOMOC ON)
|
||||||
|
set(CMAKE_AUTOUIC ON)
|
||||||
|
|
||||||
|
set(QT_TRANSLATIONS "${PROJECT_SOURCE_DIR}/src/qt_gui/translations")
|
||||||
|
file(GLOB_RECURSE TRANSLATIONS_TS ${QT_TRANSLATIONS}/*.ts)
|
||||||
|
|
||||||
|
set_source_files_properties(${TRANSLATIONS_TS} PROPERTIES OUTPUT_LOCATION "${CMAKE_CURRENT_BINARY_DIR}/translations")
|
||||||
|
qt_add_translation(TRANSLATIONS_QM ${TRANSLATIONS_TS})
|
||||||
|
|
||||||
|
set(TRANSLATIONS_QRC ${CMAKE_CURRENT_BINARY_DIR}/translations/translations.qrc)
|
||||||
|
file(WRITE ${TRANSLATIONS_QRC} "<RCC><qresource prefix=\"translations\">\n")
|
||||||
|
foreach (QM ${TRANSLATIONS_QM})
|
||||||
|
get_filename_component(QM_FILE ${QM} NAME)
|
||||||
|
file(APPEND ${TRANSLATIONS_QRC} "<file>${QM_FILE}</file>\n")
|
||||||
|
endforeach (QM)
|
||||||
|
file(APPEND ${TRANSLATIONS_QRC} "</qresource></RCC>")
|
||||||
|
|
||||||
|
qt_add_resources(TRANSLATIONS ${TRANSLATIONS_QRC})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set(AUDIO_CORE src/audio_core/sdl_audio.cpp
|
set(AUDIO_CORE src/audio_core/sdl_audio.cpp
|
||||||
|
@ -111,10 +142,13 @@ set(AUDIO_LIB src/core/libraries/audio/audioin.cpp
|
||||||
src/core/libraries/audio/audioout.h
|
src/core/libraries/audio/audioout.h
|
||||||
src/core/libraries/ajm/ajm.cpp
|
src/core/libraries/ajm/ajm.cpp
|
||||||
src/core/libraries/ajm/ajm.h
|
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
|
set(GNM_LIB src/core/libraries/gnmdriver/gnmdriver.cpp
|
||||||
src/core/libraries/gnmdriver/gnmdriver.h
|
src/core/libraries/gnmdriver/gnmdriver.h
|
||||||
|
src/core/libraries/gnmdriver/gnm_error.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(KERNEL_LIB
|
set(KERNEL_LIB
|
||||||
|
@ -175,11 +209,28 @@ set(SYSTEM_LIBS src/core/libraries/system/commondialog.cpp
|
||||||
src/core/libraries/app_content/app_content.h
|
src/core/libraries/app_content/app_content.h
|
||||||
src/core/libraries/rtc/rtc.cpp
|
src/core/libraries/rtc/rtc.cpp
|
||||||
src/core/libraries/rtc/rtc.h
|
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.cpp
|
||||||
src/core/libraries/disc_map/disc_map.h
|
src/core/libraries/disc_map/disc_map.h
|
||||||
src/core/libraries/disc_map/disc_map_codes.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.cpp
|
||||||
src/core/libraries/avplayer/avplayer.h
|
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
|
set(VIDEOOUT_LIB src/core/libraries/videoout/buffer.h
|
||||||
|
@ -189,21 +240,7 @@ set(VIDEOOUT_LIB src/core/libraries/videoout/buffer.h
|
||||||
src/core/libraries/videoout/video_out.h
|
src/core/libraries/videoout/video_out.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(LIBC_SOURCES src/core/libraries/libc/libc.cpp
|
set(LIBC_SOURCES src/core/libraries/libc_internal/libc_internal.cpp
|
||||||
src/core/libraries/libc/libc.h
|
|
||||||
src/core/libraries/libc/printf.h
|
|
||||||
src/core/libraries/libc/va_ctx.h
|
|
||||||
src/core/libraries/libc/libc_cxa.cpp
|
|
||||||
src/core/libraries/libc/libc_cxa.h
|
|
||||||
src/core/libraries/libc/libc_stdio.cpp
|
|
||||||
src/core/libraries/libc/libc_stdio.h
|
|
||||||
src/core/libraries/libc/libc_math.cpp
|
|
||||||
src/core/libraries/libc/libc_math.h
|
|
||||||
src/core/libraries/libc/libc_string.cpp
|
|
||||||
src/core/libraries/libc/libc_string.h
|
|
||||||
src/core/libraries/libc/libc_stdlib.cpp
|
|
||||||
src/core/libraries/libc/libc_stdlib.h
|
|
||||||
src/core/libraries/libc_internal/libc_internal.cpp
|
|
||||||
src/core/libraries/libc_internal/libc_internal.h
|
src/core/libraries/libc_internal/libc_internal.h
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -227,6 +264,11 @@ set(PLAYGO_LIB src/core/libraries/playgo/playgo.cpp
|
||||||
src/core/libraries/playgo/playgo_types.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
|
set(USBD_LIB src/core/libraries/usbd/usbd.cpp
|
||||||
src/core/libraries/usbd/usbd.h
|
src/core/libraries/usbd/usbd.h
|
||||||
)
|
)
|
||||||
|
@ -238,6 +280,7 @@ set(NP_LIBS src/core/libraries/np_manager/np_manager.cpp
|
||||||
src/core/libraries/np_trophy/np_trophy.cpp
|
src/core/libraries/np_trophy/np_trophy.cpp
|
||||||
src/core/libraries/np_trophy/np_trophy.h
|
src/core/libraries/np_trophy/np_trophy.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(MISC_LIBS src/core/libraries/screenshot/screenshot.cpp
|
set(MISC_LIBS src/core/libraries/screenshot/screenshot.cpp
|
||||||
src/core/libraries/screenshot/screenshot.h
|
src/core/libraries/screenshot/screenshot.h
|
||||||
)
|
)
|
||||||
|
@ -275,6 +318,7 @@ set(COMMON src/common/logging/backend.cpp
|
||||||
src/common/native_clock.h
|
src/common/native_clock.h
|
||||||
src/common/path_util.cpp
|
src/common/path_util.cpp
|
||||||
src/common/path_util.h
|
src/common/path_util.h
|
||||||
|
src/common/object_pool.h
|
||||||
src/common/polyfill_thread.h
|
src/common/polyfill_thread.h
|
||||||
src/common/rdtsc.cpp
|
src/common/rdtsc.cpp
|
||||||
src/common/rdtsc.h
|
src/common/rdtsc.h
|
||||||
|
@ -286,9 +330,12 @@ set(COMMON src/common/logging/backend.cpp
|
||||||
src/common/thread.h
|
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/ntapi.h
|
||||||
src/common/ntapi.cpp
|
src/common/ntapi.cpp
|
||||||
|
src/common/scm_rev.cpp
|
||||||
|
src/common/scm_rev.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(CORE src/core/aerolib/stubs.cpp
|
set(CORE src/core/aerolib/stubs.cpp
|
||||||
|
@ -339,6 +386,7 @@ set(CORE src/core/aerolib/stubs.cpp
|
||||||
${NP_LIBS}
|
${NP_LIBS}
|
||||||
${PNG_LIB}
|
${PNG_LIB}
|
||||||
${PLAYGO_LIB}
|
${PLAYGO_LIB}
|
||||||
|
${RANDOM_LIB}
|
||||||
${USBD_LIB}
|
${USBD_LIB}
|
||||||
${MISC_LIBS}
|
${MISC_LIBS}
|
||||||
${DIALOGS_LIB}
|
${DIALOGS_LIB}
|
||||||
|
@ -356,7 +404,6 @@ set(CORE src/core/aerolib/stubs.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
set(SHADER_RECOMPILER src/shader_recompiler/exception.h
|
set(SHADER_RECOMPILER src/shader_recompiler/exception.h
|
||||||
src/shader_recompiler/object_pool.h
|
|
||||||
src/shader_recompiler/profile.h
|
src/shader_recompiler/profile.h
|
||||||
src/shader_recompiler/recompiler.cpp
|
src/shader_recompiler/recompiler.cpp
|
||||||
src/shader_recompiler/recompiler.h
|
src/shader_recompiler/recompiler.h
|
||||||
|
@ -383,7 +430,6 @@ set(SHADER_RECOMPILER src/shader_recompiler/exception.h
|
||||||
src/shader_recompiler/backend/spirv/spirv_emit_context.h
|
src/shader_recompiler/backend/spirv/spirv_emit_context.h
|
||||||
src/shader_recompiler/frontend/translate/data_share.cpp
|
src/shader_recompiler/frontend/translate/data_share.cpp
|
||||||
src/shader_recompiler/frontend/translate/export.cpp
|
src/shader_recompiler/frontend/translate/export.cpp
|
||||||
src/shader_recompiler/frontend/translate/flat_memory.cpp
|
|
||||||
src/shader_recompiler/frontend/translate/scalar_alu.cpp
|
src/shader_recompiler/frontend/translate/scalar_alu.cpp
|
||||||
src/shader_recompiler/frontend/translate/scalar_memory.cpp
|
src/shader_recompiler/frontend/translate/scalar_memory.cpp
|
||||||
src/shader_recompiler/frontend/translate/translate.cpp
|
src/shader_recompiler/frontend/translate/translate.cpp
|
||||||
|
@ -403,10 +449,11 @@ set(SHADER_RECOMPILER src/shader_recompiler/exception.h
|
||||||
src/shader_recompiler/frontend/opcodes.h
|
src/shader_recompiler/frontend/opcodes.h
|
||||||
src/shader_recompiler/frontend/structured_control_flow.cpp
|
src/shader_recompiler/frontend/structured_control_flow.cpp
|
||||||
src/shader_recompiler/frontend/structured_control_flow.h
|
src/shader_recompiler/frontend/structured_control_flow.h
|
||||||
src/shader_recompiler/ir/passes/constant_propogation_pass.cpp
|
src/shader_recompiler/ir/passes/constant_propagation_pass.cpp
|
||||||
src/shader_recompiler/ir/passes/dead_code_elimination_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/identity_removal_pass.cpp
|
||||||
src/shader_recompiler/ir/passes/ir_passes.h
|
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/resource_tracking_pass.cpp
|
||||||
src/shader_recompiler/ir/passes/shader_info_collection_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/passes/ssa_rewrite_pass.cpp
|
||||||
|
@ -440,6 +487,14 @@ set(VIDEO_CORE src/video_core/amdgpu/liverpool.cpp
|
||||||
src/video_core/amdgpu/pm4_cmds.h
|
src/video_core/amdgpu/pm4_cmds.h
|
||||||
src/video_core/amdgpu/pm4_opcodes.h
|
src/video_core/amdgpu/pm4_opcodes.h
|
||||||
src/video_core/amdgpu/resource.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.cpp
|
||||||
src/video_core/renderer_vulkan/liverpool_to_vk.h
|
src/video_core/renderer_vulkan/liverpool_to_vk.h
|
||||||
src/video_core/renderer_vulkan/renderer_vulkan.cpp
|
src/video_core/renderer_vulkan/renderer_vulkan.cpp
|
||||||
|
@ -466,10 +521,10 @@ set(VIDEO_CORE src/video_core/amdgpu/liverpool.cpp
|
||||||
src/video_core/renderer_vulkan/vk_resource_pool.h
|
src/video_core/renderer_vulkan/vk_resource_pool.h
|
||||||
src/video_core/renderer_vulkan/vk_scheduler.cpp
|
src/video_core/renderer_vulkan/vk_scheduler.cpp
|
||||||
src/video_core/renderer_vulkan/vk_scheduler.h
|
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.cpp
|
||||||
src/video_core/renderer_vulkan/vk_shader_util.h
|
src/video_core/renderer_vulkan/vk_shader_util.h
|
||||||
src/video_core/renderer_vulkan/vk_stream_buffer.cpp
|
|
||||||
src/video_core/renderer_vulkan/vk_stream_buffer.h
|
|
||||||
src/video_core/renderer_vulkan/vk_swapchain.cpp
|
src/video_core/renderer_vulkan/vk_swapchain.cpp
|
||||||
src/video_core/renderer_vulkan/vk_swapchain.h
|
src/video_core/renderer_vulkan/vk_swapchain.h
|
||||||
src/video_core/texture_cache/image.cpp
|
src/video_core/texture_cache/image.cpp
|
||||||
|
@ -485,6 +540,9 @@ set(VIDEO_CORE src/video_core/amdgpu/liverpool.cpp
|
||||||
src/video_core/texture_cache/tile_manager.cpp
|
src/video_core/texture_cache/tile_manager.cpp
|
||||||
src/video_core/texture_cache/tile_manager.h
|
src/video_core/texture_cache/tile_manager.h
|
||||||
src/video_core/texture_cache/types.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.cpp
|
||||||
src/video_core/renderdoc.h
|
src/video_core/renderdoc.h
|
||||||
)
|
)
|
||||||
|
@ -499,37 +557,47 @@ set(EMULATOR src/emulator.cpp
|
||||||
src/sdl_window.cpp
|
src/sdl_window.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
# the above is shared in sdl and qt version (TODO share them all)
|
# The above is shared in SDL and Qt version (TODO share them all)
|
||||||
|
|
||||||
if(ENABLE_QT_GUI)
|
if(ENABLE_QT_GUI)
|
||||||
qt_add_resources(RESOURCE_FILES src/shadps4.qrc)
|
qt_add_resources(RESOURCE_FILES src/shadps4.qrc)
|
||||||
|
|
||||||
set(QT_GUI
|
set(QT_GUI src/qt_gui/about_dialog.cpp
|
||||||
src/qt_gui/main_window_ui.h
|
src/qt_gui/about_dialog.h
|
||||||
src/qt_gui/main_window.cpp
|
src/qt_gui/about_dialog.ui
|
||||||
src/qt_gui/main_window.h
|
src/qt_gui/cheats_patches.cpp
|
||||||
src/qt_gui/gui_context_menus.h
|
src/qt_gui/cheats_patches.h
|
||||||
src/qt_gui/game_list_utils.h
|
src/qt_gui/memory_patcher.cpp
|
||||||
src/qt_gui/game_info.cpp
|
src/qt_gui/memory_patcher.h
|
||||||
src/qt_gui/game_info.h
|
src/qt_gui/main_window_ui.h
|
||||||
src/qt_gui/game_list_frame.cpp
|
src/qt_gui/main_window.cpp
|
||||||
src/qt_gui/game_list_frame.h
|
src/qt_gui/main_window.h
|
||||||
src/qt_gui/game_grid_frame.cpp
|
src/qt_gui/gui_context_menus.h
|
||||||
src/qt_gui/game_grid_frame.h
|
src/qt_gui/game_list_utils.h
|
||||||
src/qt_gui/game_install_dialog.cpp
|
src/qt_gui/game_info.cpp
|
||||||
src/qt_gui/game_install_dialog.h
|
src/qt_gui/game_info.h
|
||||||
src/qt_gui/pkg_viewer.cpp
|
src/qt_gui/game_list_frame.cpp
|
||||||
src/qt_gui/pkg_viewer.h
|
src/qt_gui/game_list_frame.h
|
||||||
src/qt_gui/trophy_viewer.cpp
|
src/qt_gui/game_grid_frame.cpp
|
||||||
src/qt_gui/trophy_viewer.h
|
src/qt_gui/game_grid_frame.h
|
||||||
src/qt_gui/elf_viewer.cpp
|
src/qt_gui/game_install_dialog.cpp
|
||||||
src/qt_gui/elf_viewer.h
|
src/qt_gui/game_install_dialog.h
|
||||||
src/qt_gui/main_window_themes.cpp
|
src/qt_gui/pkg_viewer.cpp
|
||||||
src/qt_gui/main_window_themes.h
|
src/qt_gui/pkg_viewer.h
|
||||||
src/qt_gui/main.cpp
|
src/qt_gui/trophy_viewer.cpp
|
||||||
${EMULATOR}
|
src/qt_gui/trophy_viewer.h
|
||||||
${RESOURCE_FILES}
|
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()
|
endif()
|
||||||
|
|
||||||
if (ENABLE_QT_GUI)
|
if (ENABLE_QT_GUI)
|
||||||
|
@ -542,6 +610,7 @@ if (ENABLE_QT_GUI)
|
||||||
${SHADER_RECOMPILER}
|
${SHADER_RECOMPILER}
|
||||||
${VIDEO_CORE}
|
${VIDEO_CORE}
|
||||||
${EMULATOR}
|
${EMULATOR}
|
||||||
|
src/images/shadPS4.icns
|
||||||
)
|
)
|
||||||
else()
|
else()
|
||||||
add_executable(shadps4
|
add_executable(shadps4
|
||||||
|
@ -562,7 +631,7 @@ 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)
|
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 Boost::headers GPUOpen::VulkanMemoryAllocator sirit Vulkan::Headers xxHash::xxhash Zydis::Zydis glslang::SPIRV glslang::glslang SDL3::SDL3)
|
target_link_libraries(shadps4 PRIVATE Boost::headers GPUOpen::VulkanMemoryAllocator sirit Vulkan::Headers xxHash::xxhash Zydis::Zydis glslang::SPIRV glslang::glslang SDL3::SDL3)
|
||||||
|
|
||||||
if (APPLE)
|
if (APPLE)
|
||||||
|
@ -575,6 +644,9 @@ if (APPLE)
|
||||||
|
|
||||||
# Replacement for std::chrono::time_zone
|
# Replacement for std::chrono::time_zone
|
||||||
target_link_libraries(shadps4 PRIVATE date::date-tz)
|
target_link_libraries(shadps4 PRIVATE date::date-tz)
|
||||||
|
|
||||||
|
# Half float conversions for F16C patches
|
||||||
|
target_link_libraries(shadps4 PRIVATE half)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (NOT ENABLE_QT_GUI)
|
if (NOT ENABLE_QT_GUI)
|
||||||
|
@ -588,27 +660,45 @@ else()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (ENABLE_QT_GUI)
|
if (ENABLE_QT_GUI)
|
||||||
target_link_libraries(shadps4 PRIVATE Qt6::Widgets Qt6::Concurrent)
|
target_link_libraries(shadps4 PRIVATE Qt6::Widgets Qt6::Concurrent Qt6::Network)
|
||||||
|
add_definitions(-DENABLE_QT_GUI)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (WIN32)
|
if (WIN32)
|
||||||
target_link_libraries(shadps4 PRIVATE mincore winpthreads clang_rt.builtins-x86_64.lib)
|
target_link_libraries(shadps4 PRIVATE mincore winpthreads)
|
||||||
add_definitions(-D_CRT_SECURE_NO_WARNINGS -D_CRT_NONSTDC_NO_DEPRECATE -D_SCL_SECURE_NO_WARNINGS)
|
|
||||||
|
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)
|
||||||
|
endif()
|
||||||
|
|
||||||
add_definitions(-DNOMINMAX -DWIN32_LEAN_AND_MEAN)
|
add_definitions(-DNOMINMAX -DWIN32_LEAN_AND_MEAN)
|
||||||
|
|
||||||
if (MSVC)
|
if (MSVC)
|
||||||
# Needed for conflicts with time.h of windows.h
|
# Needed for conflicts with time.h of windows.h
|
||||||
add_definitions(-D_TIMESPEC_DEFINED)
|
add_definitions(-D_TIMESPEC_DEFINED)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Target Windows 10 RS5
|
# Target Windows 10 RS5
|
||||||
add_definitions(-DNTDDI_VERSION=0x0A000006 -D_WIN32_WINNT=0x0A00 -DWINVER=0x0A00)
|
add_definitions(-DNTDDI_VERSION=0x0A000006 -D_WIN32_WINNT=0x0A00 -DWINVER=0x0A00)
|
||||||
# Increase stack commit area
|
|
||||||
target_link_options(shadps4 PRIVATE /STACK:0x200000,0x200000)
|
if (MSVC)
|
||||||
|
target_link_libraries(shadps4 PRIVATE clang_rt.builtins-x86_64.lib)
|
||||||
|
endif()
|
||||||
|
|
||||||
# Disable ASLR so we can reserve the user area
|
# Disable ASLR so we can reserve the user area
|
||||||
if (MSVC)
|
if (MSVC)
|
||||||
target_link_options(shadps4 PRIVATE /DYNAMICBASE:NO)
|
target_link_options(shadps4 PRIVATE /DYNAMICBASE:NO)
|
||||||
else()
|
else()
|
||||||
target_link_options(shadps4 PRIVATE -Wl,--disable-dynamicbase)
|
target_link_options(shadps4 PRIVATE -Wl,--disable-dynamicbase)
|
||||||
endif()
|
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)
|
||||||
|
@ -628,6 +718,10 @@ 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)
|
||||||
|
|
||||||
|
set_source_files_properties(src/images/shadPS4.icns PROPERTIES
|
||||||
|
MACOSX_PACKAGE_LOCATION Resources)
|
||||||
endif()
|
endif()
|
||||||
|
|
|
@ -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 intitialize member variables!
|
// Always initialize member variables!
|
||||||
int x{};
|
int x{};
|
||||||
int y{};
|
int y{};
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,7 @@
|
||||||
|
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.
|
61
README.md
61
README.md
|
@ -32,27 +32,30 @@ SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/We are DOOMED.png" width="400">
|
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Screenshots/We are DOOMED.png" width="400">
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
# shadPS4
|
# General information
|
||||||
|
|
||||||
shadPS4 is an early PS4 emulator for Windows and Linux written in C++
|
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).
|
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 verify that a game works, you can look at [**shadPS4 Game Compatibility**](https://github.com/shadps4-emu/shadps4-game-compatibility).
|
||||||
|
|
||||||
To discuss shadPS4 development or suggest ideas, join the [**Discord server**](https://discord.gg/MyZRaBngxA).
|
To discuss shadPS4 development, suggest ideas or to ask for help, join our [**Discord server**](https://discord.gg/MyZRaBngxA).
|
||||||
|
|
||||||
Check us on [**X (twitter)**](https://x.com/shadps4) or on our [**website**](https://shadps4.net/).
|
To get the latest news, go to our [**X (Twitter)**](https://x.com/shadps4) or our [**website**](https://shadps4.net/).
|
||||||
|
|
||||||
# Status
|
# Status
|
||||||
|
|
||||||
In development, small games are working like [**Sonic Mania**](https://www.youtube.com/watch?v=AAHoNzhHyCU), [**Undertale**](https://youtu.be/5zIvdy65Ro4), [**Dysmantle**](https://youtu.be/b9xzhLBdESE) and others...
|
> [!IMPORTANT]
|
||||||
|
> shadPS4 is early in developement, don't expect a flawless experience.
|
||||||
|
|
||||||
# Why?
|
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).
|
||||||
|
|
||||||
The project started as a fun project. Due to limited free time, it will probably take a while before shadPS4 is able to run anything decent, but we're trying to make small, regular commits.
|
# Why
|
||||||
|
|
||||||
# Build
|
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.
|
||||||
|
|
||||||
|
# Building
|
||||||
|
|
||||||
## Windows
|
## Windows
|
||||||
|
|
||||||
|
@ -62,22 +65,46 @@ Check the build instructions for [**Windows**](https://github.com/shadps4-emu/sh
|
||||||
|
|
||||||
Check the build instructions for [**Linux**](https://github.com/shadps4-emu/shadPS4/blob/main/documents/building-linux.md).
|
Check the build instructions for [**Linux**](https://github.com/shadps4-emu/shadPS4/blob/main/documents/building-linux.md).
|
||||||
|
|
||||||
## Build status
|
## Building status
|
||||||
|
|
||||||
|Windows|Build status|
|
<details>
|
||||||
|--------|------------|
|
<summary><b>Windows</b></summary>
|
||||||
|
|
||||||
|
| Windows | Build status |
|
||||||
|
|--------|--------|
|
||||||
|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 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)
|
|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>
|
||||||
|
|
||||||
|Linux|Build status|
|
<details>
|
||||||
|--------|------------|
|
<summary><b>Linux</b></summary>
|
||||||
|
|
||||||
|
| Linux | Build status |
|
||||||
|
|--------|--------|
|
||||||
|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)
|
|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)
|
||||||
|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)
|
|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)
|
||||||
|
</details>
|
||||||
|
|
||||||
# Keyboard Mapping
|
<details>
|
||||||
|
<summary><b>macOS</b></summary>
|
||||||
|
|
||||||
| Controller button | Keyboard |
|
| macOS | Build status |
|
||||||
| ------------- | ------------- |
|
|--------|--------|
|
||||||
|
|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
|
||||||
|
|
||||||
|
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).
|
||||||
|
|
||||||
|
# Keyboard mapping
|
||||||
|
|
||||||
|
> [!NOTE]
|
||||||
|
> Xbox and DualShock controllers work out of the box.
|
||||||
|
|
||||||
|
| Controller button | Keyboard equivelant |
|
||||||
|
|-------------|-------------|
|
||||||
LEFT AXIS UP | W |
|
LEFT AXIS UP | W |
|
||||||
LEFT AXIS DOWN | S |
|
LEFT AXIS DOWN | S |
|
||||||
LEFT AXIS LEFT | A |
|
LEFT AXIS LEFT | A |
|
||||||
|
@ -123,7 +150,7 @@ Open a PR and we'll check it :)
|
||||||
# Contributors
|
# Contributors
|
||||||
|
|
||||||
<a href="https://github.com/shadps4-emu/shadPS4/graphs/contributors">
|
<a href="https://github.com/shadps4-emu/shadPS4/graphs/contributors">
|
||||||
<img src="https://contrib.rocks/image?repo=shadps4-emu/shadPS4&max=15" />
|
<img src="https://contrib.rocks/image?repo=shadps4-emu/shadPS4&max=15">
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
# Sister Projects
|
# Sister Projects
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
# 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()
|
|
@ -0,0 +1,25 @@
|
||||||
|
# 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)
|
|
@ -0,0 +1,156 @@
|
||||||
|
<!--
|
||||||
|
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_
|
Binary file not shown.
Before Width: | Height: | Size: 51 KiB |
|
@ -37,13 +37,18 @@ SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
- Windows 10 or Ubuntu 22.04
|
- Windows 10 or Ubuntu 22.04
|
||||||
|
|
||||||
## Have the latest WIP version
|
## How to run the latest Work-in-Progress builds of ShadPS4
|
||||||
|
|
||||||
When you go to Github Release, you have the latest major versions (e.g. v0.0.3), but if you want to have the latest Work-In-Progress version, you can go to Actions on Github to download it (Please note a Github account is required to be able to download).
|
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)
|
||||||
|
|
||||||
<img src="https://github.com/shadps4-emu/shadPS4/blob/main/documents/Quickstart/1.png" width="800"></a>
|
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)
|
||||||
|
|
||||||
After downloading the version suitable for you (Windows or Linux), you must unzip the file and then you can run it. Please note, there are two versions for each platform, a Qt version with user interface and one without (SDL Builds).
|
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
|
## Install PKG files
|
||||||
|
|
||||||
|
@ -53,4 +58,24 @@ To install PKG files (game and updates), you will need the Qt application (with
|
||||||
|
|
||||||
## Configure the emulator
|
## Configure the emulator
|
||||||
|
|
||||||
You can configure the emulator in the "user" folder (created after the first start of the application) then in the "config.toml" file. Here you can find lots of parameters to set with True or False.
|
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.
|
|
@ -5,21 +5,96 @@ SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
# Build shadPS4 for Windows
|
# Build shadPS4 for Windows
|
||||||
|
|
||||||
## Download Visual Studio Community 2022
|
This tutorial reads as if you have none of the prerequisites already installed. If you do, just ignore the steps regarding installation.
|
||||||
|
If you are building to contribute to the project, please omit `--depth 1` from the git invokations.
|
||||||
|
|
||||||
Download link: [**Visual Studio 2022**](https://visualstudio.microsoft.com/vs/)
|
Note: **ARM64 is not supported!** As of writing, it will not build nor run. The instructions with respect to ARM64 are for developers only.
|
||||||
|
|
||||||
## Requirements
|
## Option 1: Visual Studio 2022
|
||||||
|
|
||||||
### From Visual Studio Community
|
### (Prerequisite) Download the Community edition from [**Visual Studio 2022**](https://visualstudio.microsoft.com/vs/)
|
||||||
|
|
||||||
- Desktop development with C++
|
Once you are within the installer:
|
||||||
|
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
|
||||||
|
|
||||||
### From individual components tab install
|
### (Prerequisite) Download [**Qt**](https://doc.qt.io/qt-6/get-and-install-qt.html)
|
||||||
|
|
||||||
- C++ Clang Compiler for Windows (17.0.3)
|
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.
|
||||||
- MSBuild support for LLVM (Clang-cl) toolset
|
|
||||||
|
|
||||||
- ## Compiling
|
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.
|
||||||
|
If you are on Windows on ARM / Qualcomm Snapdragon Elite X, select `MSVC 2019 ARM64` instead.
|
||||||
|
|
||||||
- Open Visual Studio Community and select the **x64-Clang-Release**, **x64-Clang-Debug** or **x64-Clang-RelWithDebInfo**. It should compile just fine.
|
Go through the installation normally. If you know what you are doing, you may unselect individual components that eat up too much disk space.
|
||||||
|
|
||||||
|
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,32 +1,44 @@
|
||||||
|
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
|
v0.0.3 23/03/2024 - codename salad
|
||||||
=================
|
=================
|
||||||
-Switching to std::thread
|
- Switching to std::thread
|
||||||
-Use unique_ptr where possible
|
- Use unique_ptr where possible
|
||||||
-Replace printf/scanf with type safe fmt
|
- Replace printf/scanf with type safe fmt
|
||||||
-Implemented sceKernelGetProcessTime
|
- Implemented sceKernelGetProcessTime
|
||||||
-Implemented sceKernelGetProcessTimeCounter , sceKernelGetProcessTimeCounterFrequency
|
- Implemented sceKernelGetProcessTimeCounter, sceKernelGetProcessTimeCounterFrequency
|
||||||
-Pause emu with P button
|
- Pause emu with P button
|
||||||
-Timers rewrote with std::chrono
|
- Timers rewrote with std::chrono
|
||||||
-Added sceSystemServiceGetStatus
|
- Added sceSystemServiceGetStatus
|
||||||
-Initial FileSystem implementation
|
- Initial FileSystem implementation
|
||||||
-Initial TLS work
|
- Initial TLS work
|
||||||
-New logging implementation
|
- New logging implementation
|
||||||
-Some functions implemented for userService,systemService
|
- Some functions implemented for userService, systemService
|
||||||
-Added sceAudioOut module and output using sdl audio
|
- 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.
|
|
@ -1,35 +0,0 @@
|
||||||
shadPS4 - A PS4 emulator
|
|
||||||
=========================
|
|
||||||
|
|
||||||
1. Intro
|
|
||||||
2. Current status
|
|
||||||
3. Contributors
|
|
||||||
4. Greetings
|
|
||||||
|
|
||||||
1.Intro
|
|
||||||
=======
|
|
||||||
shadPS4 is a Play Station 4 emulator for Windows and Linux. Although atm it can't run a lot of stuff, we are working torwards to make it more compatible.
|
|
||||||
|
|
||||||
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
|
|
||||||
==============
|
|
||||||
- georgemoralis
|
|
||||||
- raphaelthegreat
|
|
||||||
- skmp
|
|
||||||
- wheremyfoodat
|
|
||||||
|
|
||||||
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
|
|
|
@ -25,11 +25,6 @@ if (NOT TARGET fmt::fmt)
|
||||||
add_subdirectory(fmt)
|
add_subdirectory(fmt)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Discord-RPC
|
|
||||||
set(BUILD_EXAMPLES OFF CACHE BOOL "")
|
|
||||||
add_subdirectory(discord-rpc)
|
|
||||||
target_include_directories(discord-rpc INTERFACE ./discord-rpc/include)
|
|
||||||
|
|
||||||
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
|
||||||
|
@ -47,6 +42,12 @@ else()
|
||||||
endif()
|
endif()
|
||||||
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)
|
if (NOT TARGET zlib-ng::zlib)
|
||||||
set(ZLIB_ENABLE_TESTS OFF)
|
set(ZLIB_ENABLE_TESTS OFF)
|
||||||
|
@ -141,11 +142,17 @@ if (WIN32)
|
||||||
target_compile_options(sirit PUBLIC "-Wno-error=unused-command-line-argument")
|
target_compile_options(sirit PUBLIC "-Wno-error=unused-command-line-argument")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# date
|
if (APPLE)
|
||||||
if (APPLE AND NOT TARGET date::date-tz)
|
# half
|
||||||
option(BUILD_TZ_LIB "" ON)
|
add_library(half INTERFACE)
|
||||||
option(USE_SYSTEM_TZ_DB "" ON)
|
target_include_directories(half INTERFACE half/include)
|
||||||
add_subdirectory(date)
|
|
||||||
|
# date
|
||||||
|
if (NOT TARGET date::date-tz)
|
||||||
|
option(BUILD_TZ_LIB "" ON)
|
||||||
|
option(USE_SYSTEM_TZ_DB "" ON)
|
||||||
|
add_subdirectory(date)
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Tracy
|
# Tracy
|
||||||
|
|
|
@ -0,0 +1,158 @@
|
||||||
|
# - 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()
|
|
@ -0,0 +1,42 @@
|
||||||
|
#
|
||||||
|
# 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 +0,0 @@
|
||||||
Subproject commit 4ec218155d73bcb8022f8f7ca72305d801f84beb
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 147b2de7734f5dc3b9aeb1f4135ae15fcd44b9d7
|
Subproject commit a04136add1e469f46d8ae8d3e8307779240a5c53
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit e30b7d7fe228bfb3f6e41ce1040b44a15eb7d5e0
|
|
@ -1 +1 @@
|
||||||
Subproject commit bc8d32e9643d2be5fc070abf2a50bc021544545d
|
Subproject commit c98518351efd5a46f5d448e947e0b7242d197d07
|
|
@ -1 +1 @@
|
||||||
Subproject commit 52f68dc6b2a9d017b43161f31f13a6f44636ee7c
|
Subproject commit 12cbda959b6df2af119a76a73ff906c2bed36884
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 1ddada225144cac0de8f6b5c0dd9acffd99a2e68
|
|
@ -1 +1 @@
|
||||||
Subproject commit 664ee62c12570948b0e025d15b42d641fba8d54a
|
Subproject commit dae6bbf16c363e9ead4e628a47fdb02956a634f3
|
|
@ -1 +1 @@
|
||||||
Subproject commit 1115dad3ffa0994e3f43b693d9b9cc99944c64c1
|
Subproject commit 2c48a1a50203bbaf1e3d0d64c5d726d56f8d3bb3
|
|
@ -1 +1 @@
|
||||||
Subproject commit f9a06c20ed85fb1d6754fc2280d6183382217910
|
Subproject commit 4cc3410dce50cefce98d3cf3cf1bc8eca83b862a
|
|
@ -1 +1 @@
|
||||||
Subproject commit 8db09231c448b913ae905d5237ce2eca46e3fe87
|
Subproject commit 37090c74cc6e680f2bc334cac8fd182f7634a1f6
|
|
@ -1 +1 @@
|
||||||
Subproject commit b389bbc4ebf90fa2fe7651de3046fb19f661ba3c
|
Subproject commit 4b740127230472779c4a4d71e1a75aaa3a367a2d
|
|
@ -1 +1 @@
|
||||||
Subproject commit 871913da6a4b132b567d7b65c509600363c0041e
|
Subproject commit e1bdbca9baf4d682fb6066b380f4aa4a7bdbb58a
|
|
@ -1 +1 @@
|
||||||
Subproject commit b379292b2ab6df5771ba9870d53cf8b2c9295daf
|
Subproject commit d205aff40b4e15d4c568523ee6a26f85138126d9
|
|
@ -1 +1 @@
|
||||||
Subproject commit aabb091ae37068498751fd58202a9854408ecb0e
|
Subproject commit ccdf68421bc8eb85693f573080fc0a5faad862db
|
|
@ -1 +1 @@
|
||||||
Subproject commit a57f6cce2698049863af8c25787084ae0489d849
|
Subproject commit dbea33e47e7c0fe0b7c8592cd931c7430c1f130d
|
|
@ -1 +1 @@
|
||||||
Subproject commit 16c6a369c193981e9cf314126589eaa8763f92c3
|
Subproject commit bd73bc03b0aacaa89c9c203b9b43cd08f1b1843b
|
|
@ -114108,7 +114108,7 @@ STUB(
|
||||||
_ZN3sce2Np9CppWebApi6Common12IntrusivePtrINS1_7Matches2V124RequestCompetitiveResultEE7add_refEv)
|
_ZN3sce2Np9CppWebApi6Common12IntrusivePtrINS1_7Matches2V124RequestCompetitiveResultEE7add_refEv)
|
||||||
STUB("efPahl2FufA",
|
STUB("efPahl2FufA",
|
||||||
_ZN3sce2Np9CppWebApi30CommunicationRestrictionStatus2V35Error8fromJsonERKNS_4Json5ValueE)
|
_ZN3sce2Np9CppWebApi30CommunicationRestrictionStatus2V35Error8fromJsonERKNS_4Json5ValueE)
|
||||||
STUB("efX3lrPwdKA", sceAppContentAddcontMountByEntitlemetId)
|
STUB("efX3lrPwdKA", sceAppContentAddcontMountByEntitlementId)
|
||||||
STUB("efXnxYFN5oE", _ZNSt11range_errorD0Ev)
|
STUB("efXnxYFN5oE", _ZNSt11range_errorD0Ev)
|
||||||
STUB("efcwuDLsAM0", _ZThn120_NK7WebCore16HTMLMediaElement5mutedEv)
|
STUB("efcwuDLsAM0", _ZThn120_NK7WebCore16HTMLMediaElement5mutedEv)
|
||||||
STUB("efhGArzWdxE", _ZN7bmalloc6IsoTLS15s_didInitializeE)
|
STUB("efhGArzWdxE", _ZN7bmalloc6IsoTLS15s_didInitializeE)
|
||||||
|
@ -129493,7 +129493,7 @@ STUB(
|
||||||
STUB("kJlYH5uMAWI", sceNetResolverDestroy)
|
STUB("kJlYH5uMAWI", sceNetResolverDestroy)
|
||||||
STUB("kJmdxo4uM+8",
|
STUB("kJmdxo4uM+8",
|
||||||
_ZNSt9money_putIwSt19ostreambuf_iteratorIwSt11char_traitsIwEEE5_InitERKSt8_Locinfo)
|
_ZNSt9money_putIwSt19ostreambuf_iteratorIwSt11char_traitsIwEEE5_InitERKSt8_Locinfo)
|
||||||
STUB("kJmjt81mXKQ", sceAppContentAddcontEnqueueDownloadByEntitlemetId)
|
STUB("kJmjt81mXKQ", sceAppContentAddcontEnqueueDownloadByEntitlementId)
|
||||||
STUB(
|
STUB(
|
||||||
"kJoY9lMIFzY",
|
"kJoY9lMIFzY",
|
||||||
_ZN3sce2Np9CppWebApi6Common8IteratorINS2_12IntrusivePtrINS1_21AdvancedPlayerProfile2V138MatchCompletionRateDisconnectedMetricsEEEEmmEi)
|
_ZN3sce2Np9CppWebApi6Common8IteratorINS2_12IntrusivePtrINS1_21AdvancedPlayerProfile2V138MatchCompletionRateDisconnectedMetricsEEEEmmEi)
|
||||||
|
|
|
@ -80897,10 +80897,10 @@ sceAppCheckerExecute
|
||||||
sceAppCheckerExecuteEx
|
sceAppCheckerExecuteEx
|
||||||
sceAppContentAddcontDelete
|
sceAppContentAddcontDelete
|
||||||
sceAppContentAddcontEnqueueDownload
|
sceAppContentAddcontEnqueueDownload
|
||||||
sceAppContentAddcontEnqueueDownloadByEntitlemetId
|
sceAppContentAddcontEnqueueDownloadByEntitlementId
|
||||||
sceAppContentAddcontEnqueueDownloadSp
|
sceAppContentAddcontEnqueueDownloadSp
|
||||||
sceAppContentAddcontMount
|
sceAppContentAddcontMount
|
||||||
sceAppContentAddcontMountByEntitlemetId
|
sceAppContentAddcontMountByEntitlementId
|
||||||
sceAppContentAddcontShrink
|
sceAppContentAddcontShrink
|
||||||
sceAppContentAddcontUnmount
|
sceAppContentAddcontUnmount
|
||||||
sceAppContentAppParamGetInt
|
sceAppContentAppParamGetInt
|
||||||
|
|
|
@ -1,19 +1,23 @@
|
||||||
// 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
|
||||||
|
|
||||||
|
#include "sdl_audio.h"
|
||||||
|
|
||||||
|
#include "common/assert.h"
|
||||||
|
#include "core/libraries/error_codes.h"
|
||||||
|
|
||||||
#include <SDL3/SDL_audio.h>
|
#include <SDL3/SDL_audio.h>
|
||||||
#include <SDL3/SDL_init.h>
|
#include <SDL3/SDL_init.h>
|
||||||
#include <SDL3/SDL_timer.h>
|
#include <SDL3/SDL_timer.h>
|
||||||
#include <common/assert.h>
|
|
||||||
#include <core/libraries/error_codes.h>
|
#include <mutex> // std::unique_lock
|
||||||
#include "sdl_audio.h"
|
|
||||||
|
|
||||||
namespace Audio {
|
namespace Audio {
|
||||||
|
|
||||||
int SDLAudio::AudioOutOpen(int type, u32 samples_num, u32 freq,
|
int SDLAudio::AudioOutOpen(int type, u32 samples_num, u32 freq,
|
||||||
Libraries::AudioOut::OrbisAudioOutParamFormat format) {
|
Libraries::AudioOut::OrbisAudioOutParamFormat format) {
|
||||||
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
||||||
std::scoped_lock lock{m_mutex};
|
std::unique_lock lock{m_mutex};
|
||||||
for (int id = 0; id < portsOut.size(); id++) {
|
for (int id = 0; id < portsOut.size(); id++) {
|
||||||
auto& port = portsOut[id];
|
auto& port = portsOut[id];
|
||||||
if (!port.isOpen) {
|
if (!port.isOpen) {
|
||||||
|
@ -88,7 +92,7 @@ int SDLAudio::AudioOutOpen(int type, u32 samples_num, u32 freq,
|
||||||
}
|
}
|
||||||
|
|
||||||
s32 SDLAudio::AudioOutOutput(s32 handle, const void* ptr) {
|
s32 SDLAudio::AudioOutOutput(s32 handle, const void* ptr) {
|
||||||
std::scoped_lock lock{m_mutex};
|
std::shared_lock lock{m_mutex};
|
||||||
auto& port = portsOut[handle - 1];
|
auto& port = portsOut[handle - 1];
|
||||||
if (!port.isOpen) {
|
if (!port.isOpen) {
|
||||||
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
||||||
|
@ -100,7 +104,7 @@ s32 SDLAudio::AudioOutOutput(s32 handle, const void* ptr) {
|
||||||
int result = SDL_PutAudioStreamData(port.stream, ptr,
|
int result = SDL_PutAudioStreamData(port.stream, ptr,
|
||||||
port.samples_num * port.sample_size * port.channels_num);
|
port.samples_num * port.sample_size * port.channels_num);
|
||||||
// TODO find a correct value 8192 is estimated
|
// TODO find a correct value 8192 is estimated
|
||||||
while (SDL_GetAudioStreamAvailable(port.stream) > 8192) {
|
while (SDL_GetAudioStreamAvailable(port.stream) > 65536) {
|
||||||
SDL_Delay(0);
|
SDL_Delay(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,7 +113,7 @@ s32 SDLAudio::AudioOutOutput(s32 handle, const void* ptr) {
|
||||||
|
|
||||||
bool SDLAudio::AudioOutSetVolume(s32 handle, s32 bitflag, s32* volume) {
|
bool SDLAudio::AudioOutSetVolume(s32 handle, s32 bitflag, s32* volume) {
|
||||||
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
using Libraries::AudioOut::OrbisAudioOutParamFormat;
|
||||||
std::scoped_lock lock{m_mutex};
|
std::shared_lock lock{m_mutex};
|
||||||
auto& port = portsOut[handle - 1];
|
auto& port = portsOut[handle - 1];
|
||||||
if (!port.isOpen) {
|
if (!port.isOpen) {
|
||||||
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
return ORBIS_AUDIO_OUT_ERROR_INVALID_PORT;
|
||||||
|
@ -147,7 +151,7 @@ bool SDLAudio::AudioOutSetVolume(s32 handle, s32 bitflag, s32* volume) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SDLAudio::AudioOutGetStatus(s32 handle, int* type, int* channels_num) {
|
bool SDLAudio::AudioOutGetStatus(s32 handle, int* type, int* channels_num) {
|
||||||
std::scoped_lock lock{m_mutex};
|
std::shared_lock lock{m_mutex};
|
||||||
auto& port = portsOut[handle - 1];
|
auto& port = portsOut[handle - 1];
|
||||||
*type = port.type;
|
*type = port.type;
|
||||||
*channels_num = port.channels_num;
|
*channels_num = port.channels_num;
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <mutex>
|
#include <shared_mutex>
|
||||||
#include <SDL3/SDL_audio.h>
|
#include <SDL3/SDL_audio.h>
|
||||||
#include "core/libraries/audio/audioout.h"
|
#include "core/libraries/audio/audioout.h"
|
||||||
|
|
||||||
|
@ -22,17 +22,17 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct PortOut {
|
struct PortOut {
|
||||||
bool isOpen = false;
|
SDL_AudioStream* stream = nullptr;
|
||||||
int type = 0;
|
|
||||||
u32 samples_num = 0;
|
u32 samples_num = 0;
|
||||||
u8 sample_size = 0;
|
|
||||||
u32 freq = 0;
|
u32 freq = 0;
|
||||||
u32 format = -1;
|
u32 format = -1;
|
||||||
|
int type = 0;
|
||||||
int channels_num = 0;
|
int channels_num = 0;
|
||||||
int volume[8] = {};
|
int volume[8] = {};
|
||||||
SDL_AudioStream* stream = nullptr;
|
u8 sample_size = 0;
|
||||||
|
bool isOpen = false;
|
||||||
};
|
};
|
||||||
std::mutex m_mutex;
|
std::shared_mutex m_mutex;
|
||||||
std::array<PortOut, 22> portsOut; // main up to 8 ports , BGM 1 port , voice up to 4 ports ,
|
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
|
// personal up to 4 ports , padspk up to 5 ports , aux 1 port
|
||||||
};
|
};
|
||||||
|
|
|
@ -15,17 +15,22 @@ static u32 screenWidth = 1280;
|
||||||
static u32 screenHeight = 720;
|
static u32 screenHeight = 720;
|
||||||
static s32 gpuId = -1; // Vulkan physical device index. Set to negative for auto select
|
static s32 gpuId = -1; // Vulkan physical device index. Set to negative for auto select
|
||||||
static std::string logFilter;
|
static std::string logFilter;
|
||||||
static std::string logType = "sync";
|
static std::string logType = "async";
|
||||||
|
static std::string userName = "shadPS4";
|
||||||
|
static bool useSpecialPad = false;
|
||||||
|
static int specialPadClass = 1;
|
||||||
static bool isDebugDump = false;
|
static bool isDebugDump = false;
|
||||||
static bool isLibc = true;
|
|
||||||
static bool isShowSplash = false;
|
static bool isShowSplash = false;
|
||||||
static bool isNullGpu = false;
|
static bool isNullGpu = false;
|
||||||
|
static bool shouldCopyGPUBuffers = false;
|
||||||
static bool shouldDumpShaders = false;
|
static bool shouldDumpShaders = false;
|
||||||
static bool shouldDumpPM4 = false;
|
static bool shouldDumpPM4 = false;
|
||||||
static u32 vblankDivider = 1;
|
static u32 vblankDivider = 1;
|
||||||
static bool vkValidation = false;
|
static bool vkValidation = false;
|
||||||
static bool vkValidationSync = false;
|
static bool vkValidationSync = false;
|
||||||
|
static bool vkValidationGpu = false;
|
||||||
static bool rdocEnable = false;
|
static bool rdocEnable = false;
|
||||||
|
static bool rdocMarkersEnable = false;
|
||||||
// Gui
|
// Gui
|
||||||
std::string settings_install_dir = "";
|
std::string settings_install_dir = "";
|
||||||
u32 main_window_geometry_x = 400;
|
u32 main_window_geometry_x = 400;
|
||||||
|
@ -43,10 +48,9 @@ u32 m_window_size_H = 720;
|
||||||
std::vector<std::string> m_pkg_viewer;
|
std::vector<std::string> m_pkg_viewer;
|
||||||
std::vector<std::string> m_elf_viewer;
|
std::vector<std::string> m_elf_viewer;
|
||||||
std::vector<std::string> m_recent_files;
|
std::vector<std::string> m_recent_files;
|
||||||
|
std::string emulator_language = "en";
|
||||||
bool isLleLibc() {
|
// Settings
|
||||||
return isLibc;
|
u32 m_language = 1; // english
|
||||||
}
|
|
||||||
|
|
||||||
bool isNeoMode() {
|
bool isNeoMode() {
|
||||||
return isNeo;
|
return isNeo;
|
||||||
|
@ -76,6 +80,18 @@ std::string getLogType() {
|
||||||
return logType;
|
return logType;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string getUserName() {
|
||||||
|
return userName;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool getUseSpecialPad() {
|
||||||
|
return useSpecialPad;
|
||||||
|
}
|
||||||
|
|
||||||
|
int getSpecialPadClass() {
|
||||||
|
return specialPadClass;
|
||||||
|
}
|
||||||
|
|
||||||
bool debugDump() {
|
bool debugDump() {
|
||||||
return isDebugDump;
|
return isDebugDump;
|
||||||
}
|
}
|
||||||
|
@ -88,6 +104,10 @@ bool nullGpu() {
|
||||||
return isNullGpu;
|
return isNullGpu;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool copyGPUCmdBuffers() {
|
||||||
|
return shouldCopyGPUBuffers;
|
||||||
|
}
|
||||||
|
|
||||||
bool dumpShaders() {
|
bool dumpShaders() {
|
||||||
return shouldDumpShaders;
|
return shouldDumpShaders;
|
||||||
}
|
}
|
||||||
|
@ -100,6 +120,10 @@ bool isRdocEnabled() {
|
||||||
return rdocEnable;
|
return rdocEnable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool isMarkersEnabled() {
|
||||||
|
return rdocMarkersEnable;
|
||||||
|
}
|
||||||
|
|
||||||
u32 vblankDiv() {
|
u32 vblankDiv() {
|
||||||
return vblankDivider;
|
return vblankDivider;
|
||||||
}
|
}
|
||||||
|
@ -112,6 +136,94 @@ bool vkValidationSyncEnabled() {
|
||||||
return vkValidationSync;
|
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) {
|
void setMainWindowGeometry(u32 x, u32 y, u32 w, u32 h) {
|
||||||
main_window_geometry_x = x;
|
main_window_geometry_x = x;
|
||||||
main_window_geometry_y = y;
|
main_window_geometry_y = y;
|
||||||
|
@ -130,10 +242,10 @@ void setIconSize(u32 size) {
|
||||||
void setIconSizeGrid(u32 size) {
|
void setIconSizeGrid(u32 size) {
|
||||||
m_icon_size_grid = size;
|
m_icon_size_grid = size;
|
||||||
}
|
}
|
||||||
void setSliderPositon(u32 pos) {
|
void setSliderPosition(u32 pos) {
|
||||||
m_slider_pos = pos;
|
m_slider_pos = pos;
|
||||||
}
|
}
|
||||||
void setSliderPositonGrid(u32 pos) {
|
void setSliderPositionGrid(u32 pos) {
|
||||||
m_slider_pos_grid = pos;
|
m_slider_pos_grid = pos;
|
||||||
}
|
}
|
||||||
void setTableMode(u32 mode) {
|
void setTableMode(u32 mode) {
|
||||||
|
@ -145,19 +257,23 @@ void setMainWindowWidth(u32 width) {
|
||||||
void setMainWindowHeight(u32 height) {
|
void setMainWindowHeight(u32 height) {
|
||||||
m_window_size_H = height;
|
m_window_size_H = height;
|
||||||
}
|
}
|
||||||
void setPkgViewer(std::vector<std::string> pkgList) {
|
void setPkgViewer(const std::vector<std::string>& pkgList) {
|
||||||
m_pkg_viewer.resize(pkgList.size());
|
m_pkg_viewer.resize(pkgList.size());
|
||||||
m_pkg_viewer = pkgList;
|
m_pkg_viewer = pkgList;
|
||||||
}
|
}
|
||||||
void setElfViewer(std::vector<std::string> elfList) {
|
void setElfViewer(const std::vector<std::string>& elfList) {
|
||||||
m_elf_viewer.resize(elfList.size());
|
m_elf_viewer.resize(elfList.size());
|
||||||
m_elf_viewer = elfList;
|
m_elf_viewer = elfList;
|
||||||
}
|
}
|
||||||
void setRecentFiles(std::vector<std::string> recentFiles) {
|
void setRecentFiles(const std::vector<std::string>& recentFiles) {
|
||||||
m_recent_files.resize(recentFiles.size());
|
m_recent_files.resize(recentFiles.size());
|
||||||
m_recent_files = recentFiles;
|
m_recent_files = recentFiles;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setEmulatorLanguage(std::string language) {
|
||||||
|
emulator_language = language;
|
||||||
|
}
|
||||||
|
|
||||||
u32 getMainWindowGeometryX() {
|
u32 getMainWindowGeometryX() {
|
||||||
return main_window_geometry_x;
|
return main_window_geometry_x;
|
||||||
}
|
}
|
||||||
|
@ -182,10 +298,10 @@ u32 getIconSize() {
|
||||||
u32 getIconSizeGrid() {
|
u32 getIconSizeGrid() {
|
||||||
return m_icon_size_grid;
|
return m_icon_size_grid;
|
||||||
}
|
}
|
||||||
u32 getSliderPositon() {
|
u32 getSliderPosition() {
|
||||||
return m_slider_pos;
|
return m_slider_pos;
|
||||||
}
|
}
|
||||||
u32 getSliderPositonGrid() {
|
u32 getSliderPositionGrid() {
|
||||||
return m_slider_pos_grid;
|
return m_slider_pos_grid;
|
||||||
}
|
}
|
||||||
u32 getTableMode() {
|
u32 getTableMode() {
|
||||||
|
@ -207,6 +323,13 @@ std::vector<std::string> getRecentFiles() {
|
||||||
return m_recent_files;
|
return m_recent_files;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string getEmulatorLanguage() {
|
||||||
|
return emulator_language;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 GetLanguage() {
|
||||||
|
return m_language;
|
||||||
|
}
|
||||||
void load(const std::filesystem::path& path) {
|
void load(const std::filesystem::path& path) {
|
||||||
// If the configuration file does not exist, create it and return
|
// If the configuration file does not exist, create it and return
|
||||||
std::error_code error;
|
std::error_code error;
|
||||||
|
@ -223,90 +346,88 @@ void load(const std::filesystem::path& path) {
|
||||||
fmt::print("Got exception trying to load config file. Exception: {}\n", ex.what());
|
fmt::print("Got exception trying to load config file. Exception: {}\n", ex.what());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.contains("General")) {
|
if (data.contains("General")) {
|
||||||
auto generalResult = toml::expect<toml::value>(data.at("General"));
|
const toml::value& general = data.at("General");
|
||||||
if (generalResult.is_ok()) {
|
|
||||||
auto general = generalResult.unwrap();
|
|
||||||
|
|
||||||
isNeo = toml::find_or<toml::boolean>(general, "isPS4Pro", false);
|
isNeo = toml::find_or<bool>(general, "isPS4Pro", false);
|
||||||
isFullscreen = toml::find_or<toml::boolean>(general, "Fullscreen", false);
|
isFullscreen = toml::find_or<bool>(general, "Fullscreen", false);
|
||||||
logFilter = toml::find_or<toml::string>(general, "logFilter", "");
|
logFilter = toml::find_or<std::string>(general, "logFilter", "");
|
||||||
logType = toml::find_or<toml::string>(general, "logType", "sync");
|
logType = toml::find_or<std::string>(general, "logType", "sync");
|
||||||
isShowSplash = toml::find_or<toml::boolean>(general, "showSplash", true);
|
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")) {
|
if (data.contains("GPU")) {
|
||||||
auto gpuResult = toml::expect<toml::value>(data.at("GPU"));
|
const toml::value& gpu = data.at("GPU");
|
||||||
if (gpuResult.is_ok()) {
|
|
||||||
auto gpu = gpuResult.unwrap();
|
|
||||||
|
|
||||||
screenWidth = toml::find_or<toml::integer>(gpu, "screenWidth", screenWidth);
|
screenWidth = toml::find_or<int>(gpu, "screenWidth", screenWidth);
|
||||||
screenHeight = toml::find_or<toml::integer>(gpu, "screenHeight", screenHeight);
|
screenHeight = toml::find_or<int>(gpu, "screenHeight", screenHeight);
|
||||||
isNullGpu = toml::find_or<toml::boolean>(gpu, "nullGpu", false);
|
isNullGpu = toml::find_or<bool>(gpu, "nullGpu", false);
|
||||||
shouldDumpShaders = toml::find_or<toml::boolean>(gpu, "dumpShaders", false);
|
shouldCopyGPUBuffers = toml::find_or<bool>(gpu, "copyGPUBuffers", false);
|
||||||
shouldDumpPM4 = toml::find_or<toml::boolean>(gpu, "dumpPM4", false);
|
shouldDumpShaders = toml::find_or<bool>(gpu, "dumpShaders", false);
|
||||||
vblankDivider = toml::find_or<toml::integer>(gpu, "vblankDivider", 1);
|
shouldDumpPM4 = toml::find_or<bool>(gpu, "dumpPM4", false);
|
||||||
}
|
vblankDivider = toml::find_or<int>(gpu, "vblankDivider", 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.contains("Vulkan")) {
|
if (data.contains("Vulkan")) {
|
||||||
const auto vkResult = toml::expect<toml::value>(data.at("Vulkan"));
|
const toml::value& vk = data.at("Vulkan");
|
||||||
if (vkResult.is_ok()) {
|
|
||||||
auto vk = vkResult.unwrap();
|
|
||||||
|
|
||||||
gpuId = toml::find_or<toml::integer>(vk, "gpuId", 0);
|
gpuId = toml::find_or<int>(vk, "gpuId", -1);
|
||||||
vkValidation = toml::find_or<toml::boolean>(vk, "validation", true);
|
vkValidation = toml::find_or<bool>(vk, "validation", false);
|
||||||
vkValidationSync = toml::find_or<toml::boolean>(vk, "validation_sync", true);
|
vkValidationSync = toml::find_or<bool>(vk, "validation_sync", false);
|
||||||
rdocEnable = toml::find_or<toml::boolean>(vk, "rdocEnable", 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")) {
|
if (data.contains("Debug")) {
|
||||||
auto debugResult = toml::expect<toml::value>(data.at("Debug"));
|
const toml::value& debug = data.at("Debug");
|
||||||
if (debugResult.is_ok()) {
|
|
||||||
auto debug = debugResult.unwrap();
|
|
||||||
|
|
||||||
isDebugDump = toml::find_or<toml::boolean>(debug, "DebugDump", false);
|
isDebugDump = toml::find_or<bool>(debug, "DebugDump", false);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (data.contains("LLE")) {
|
|
||||||
auto lleResult = toml::expect<toml::value>(data.at("LLE"));
|
|
||||||
if (lleResult.is_ok()) {
|
|
||||||
auto lle = lleResult.unwrap();
|
|
||||||
|
|
||||||
isLibc = toml::find_or<toml::boolean>(lle, "libc", true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (data.contains("GUI")) {
|
if (data.contains("GUI")) {
|
||||||
auto guiResult = toml::expect<toml::value>(data.at("GUI"));
|
const toml::value& gui = data.at("GUI");
|
||||||
if (guiResult.is_ok()) {
|
|
||||||
auto gui = guiResult.unwrap();
|
|
||||||
|
|
||||||
m_icon_size = toml::find_or<toml::integer>(gui, "iconSize", 0);
|
m_icon_size = toml::find_or<int>(gui, "iconSize", 0);
|
||||||
m_icon_size_grid = toml::find_or<toml::integer>(gui, "iconSizeGrid", 0);
|
m_icon_size_grid = toml::find_or<int>(gui, "iconSizeGrid", 0);
|
||||||
m_slider_pos = toml::find_or<toml::integer>(gui, "sliderPos", 0);
|
m_slider_pos = toml::find_or<int>(gui, "sliderPos", 0);
|
||||||
m_slider_pos_grid = toml::find_or<toml::integer>(gui, "sliderPosGrid", 0);
|
m_slider_pos_grid = toml::find_or<int>(gui, "sliderPosGrid", 0);
|
||||||
mw_themes = toml::find_or<toml::integer>(gui, "theme", 0);
|
mw_themes = toml::find_or<int>(gui, "theme", 0);
|
||||||
m_window_size_W = toml::find_or<toml::integer>(gui, "mw_width", 0);
|
m_window_size_W = toml::find_or<int>(gui, "mw_width", 0);
|
||||||
m_window_size_H = toml::find_or<toml::integer>(gui, "mw_height", 0);
|
m_window_size_H = toml::find_or<int>(gui, "mw_height", 0);
|
||||||
settings_install_dir = toml::find_or<toml::string>(gui, "installDir", "");
|
settings_install_dir = toml::find_or<std::string>(gui, "installDir", "");
|
||||||
main_window_geometry_x = toml::find_or<toml::integer>(gui, "geometry_x", 0);
|
main_window_geometry_x = toml::find_or<int>(gui, "geometry_x", 0);
|
||||||
main_window_geometry_y = toml::find_or<toml::integer>(gui, "geometry_y", 0);
|
main_window_geometry_y = toml::find_or<int>(gui, "geometry_y", 0);
|
||||||
main_window_geometry_w = toml::find_or<toml::integer>(gui, "geometry_w", 0);
|
main_window_geometry_w = toml::find_or<int>(gui, "geometry_w", 0);
|
||||||
main_window_geometry_h = toml::find_or<toml::integer>(gui, "geometry_h", 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_pkg_viewer = toml::find_or<std::vector<std::string>>(gui, "pkgDirs", {});
|
||||||
m_elf_viewer = toml::find_or<std::vector<std::string>>(gui, "elfDirs", {});
|
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_recent_files = toml::find_or<std::vector<std::string>>(gui, "recentFiles", {});
|
||||||
m_table_mode = toml::find_or<toml::integer>(gui, "gameTableMode", 0);
|
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) {
|
void save(const std::filesystem::path& path) {
|
||||||
toml::basic_value<toml::preserve_comments> data;
|
toml::value data;
|
||||||
|
|
||||||
std::error_code error;
|
std::error_code error;
|
||||||
if (std::filesystem::exists(path, error)) {
|
if (std::filesystem::exists(path, error)) {
|
||||||
try {
|
try {
|
||||||
data = toml::parse<toml::preserve_comments>(path);
|
data = toml::parse(path);
|
||||||
} catch (const std::exception& ex) {
|
} catch (const std::exception& ex) {
|
||||||
fmt::print("Exception trying to parse config file. Exception: {}\n", ex.what());
|
fmt::print("Exception trying to parse config file. Exception: {}\n", ex.what());
|
||||||
return;
|
return;
|
||||||
|
@ -323,19 +444,24 @@ void save(const std::filesystem::path& path) {
|
||||||
data["General"]["Fullscreen"] = isFullscreen;
|
data["General"]["Fullscreen"] = isFullscreen;
|
||||||
data["General"]["logFilter"] = logFilter;
|
data["General"]["logFilter"] = logFilter;
|
||||||
data["General"]["logType"] = logType;
|
data["General"]["logType"] = logType;
|
||||||
|
data["General"]["userName"] = userName;
|
||||||
data["General"]["showSplash"] = isShowSplash;
|
data["General"]["showSplash"] = isShowSplash;
|
||||||
|
data["Input"]["useSpecialPad"] = useSpecialPad;
|
||||||
|
data["Input"]["specialPadClass"] = specialPadClass;
|
||||||
data["GPU"]["screenWidth"] = screenWidth;
|
data["GPU"]["screenWidth"] = screenWidth;
|
||||||
data["GPU"]["screenHeight"] = screenHeight;
|
data["GPU"]["screenHeight"] = screenHeight;
|
||||||
data["GPU"]["nullGpu"] = isNullGpu;
|
data["GPU"]["nullGpu"] = isNullGpu;
|
||||||
|
data["GPU"]["copyGPUBuffers"] = shouldCopyGPUBuffers;
|
||||||
data["GPU"]["dumpShaders"] = shouldDumpShaders;
|
data["GPU"]["dumpShaders"] = shouldDumpShaders;
|
||||||
data["GPU"]["dumpPM4"] = shouldDumpPM4;
|
data["GPU"]["dumpPM4"] = shouldDumpPM4;
|
||||||
data["GPU"]["vblankDivider"] = vblankDivider;
|
data["GPU"]["vblankDivider"] = vblankDivider;
|
||||||
data["Vulkan"]["gpuId"] = gpuId;
|
data["Vulkan"]["gpuId"] = gpuId;
|
||||||
data["Vulkan"]["validation"] = vkValidation;
|
data["Vulkan"]["validation"] = vkValidation;
|
||||||
data["Vulkan"]["validation_sync"] = vkValidationSync;
|
data["Vulkan"]["validation_sync"] = vkValidationSync;
|
||||||
|
data["Vulkan"]["validation_gpu"] = vkValidationGpu;
|
||||||
data["Vulkan"]["rdocEnable"] = rdocEnable;
|
data["Vulkan"]["rdocEnable"] = rdocEnable;
|
||||||
|
data["Vulkan"]["rdocMarkersEnable"] = rdocMarkersEnable;
|
||||||
data["Debug"]["DebugDump"] = isDebugDump;
|
data["Debug"]["DebugDump"] = isDebugDump;
|
||||||
data["LLE"]["libc"] = isLibc;
|
|
||||||
data["GUI"]["theme"] = mw_themes;
|
data["GUI"]["theme"] = mw_themes;
|
||||||
data["GUI"]["iconSize"] = m_icon_size;
|
data["GUI"]["iconSize"] = m_icon_size;
|
||||||
data["GUI"]["sliderPos"] = m_slider_pos;
|
data["GUI"]["sliderPos"] = m_slider_pos;
|
||||||
|
@ -352,9 +478,36 @@ void save(const std::filesystem::path& path) {
|
||||||
data["GUI"]["pkgDirs"] = m_pkg_viewer;
|
data["GUI"]["pkgDirs"] = m_pkg_viewer;
|
||||||
data["GUI"]["elfDirs"] = m_elf_viewer;
|
data["GUI"]["elfDirs"] = m_elf_viewer;
|
||||||
data["GUI"]["recentFiles"] = m_recent_files;
|
data["GUI"]["recentFiles"] = m_recent_files;
|
||||||
|
data["GUI"]["emulatorLanguage"] = emulator_language;
|
||||||
|
|
||||||
|
data["Settings"]["consoleLanguage"] = m_language;
|
||||||
|
|
||||||
std::ofstream file(path, std::ios::out);
|
std::ofstream file(path, std::ios::out);
|
||||||
file << data;
|
file << data;
|
||||||
file.close();
|
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
|
} // namespace Config
|
||||||
|
|
|
@ -15,22 +15,53 @@ bool isNeoMode();
|
||||||
bool isFullscreenMode();
|
bool isFullscreenMode();
|
||||||
std::string getLogFilter();
|
std::string getLogFilter();
|
||||||
std::string getLogType();
|
std::string getLogType();
|
||||||
|
std::string getUserName();
|
||||||
|
|
||||||
|
bool getUseSpecialPad();
|
||||||
|
int getSpecialPadClass();
|
||||||
|
|
||||||
u32 getScreenWidth();
|
u32 getScreenWidth();
|
||||||
u32 getScreenHeight();
|
u32 getScreenHeight();
|
||||||
s32 getGpuId();
|
s32 getGpuId();
|
||||||
|
|
||||||
bool debugDump();
|
bool debugDump();
|
||||||
bool isLleLibc();
|
|
||||||
bool showSplash();
|
bool showSplash();
|
||||||
bool nullGpu();
|
bool nullGpu();
|
||||||
|
bool copyGPUCmdBuffers();
|
||||||
bool dumpShaders();
|
bool dumpShaders();
|
||||||
bool dumpPM4();
|
bool dumpPM4();
|
||||||
bool isRdocEnabled();
|
bool isRdocEnabled();
|
||||||
|
bool isMarkersEnabled();
|
||||||
u32 vblankDiv();
|
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 vkValidationEnabled();
|
||||||
bool vkValidationSyncEnabled();
|
bool vkValidationSyncEnabled();
|
||||||
|
bool vkValidationGpuEnabled();
|
||||||
|
|
||||||
// Gui
|
// Gui
|
||||||
void setMainWindowGeometry(u32 x, u32 y, u32 w, u32 h);
|
void setMainWindowGeometry(u32 x, u32 y, u32 w, u32 h);
|
||||||
|
@ -38,14 +69,15 @@ void setGameInstallDir(const std::string& dir);
|
||||||
void setMainWindowTheme(u32 theme);
|
void setMainWindowTheme(u32 theme);
|
||||||
void setIconSize(u32 size);
|
void setIconSize(u32 size);
|
||||||
void setIconSizeGrid(u32 size);
|
void setIconSizeGrid(u32 size);
|
||||||
void setSliderPositon(u32 pos);
|
void setSliderPosition(u32 pos);
|
||||||
void setSliderPositonGrid(u32 pos);
|
void setSliderPositionGrid(u32 pos);
|
||||||
void setTableMode(u32 mode);
|
void setTableMode(u32 mode);
|
||||||
void setMainWindowWidth(u32 width);
|
void setMainWindowWidth(u32 width);
|
||||||
void setMainWindowHeight(u32 height);
|
void setMainWindowHeight(u32 height);
|
||||||
void setPkgViewer(std::vector<std::string> pkgList);
|
void setPkgViewer(const std::vector<std::string>& pkgList);
|
||||||
void setElfViewer(std::vector<std::string> elfList);
|
void setElfViewer(const std::vector<std::string>& elfList);
|
||||||
void setRecentFiles(std::vector<std::string> recentFiles);
|
void setRecentFiles(const std::vector<std::string>& recentFiles);
|
||||||
|
void setEmulatorLanguage(std::string language);
|
||||||
|
|
||||||
u32 getMainWindowGeometryX();
|
u32 getMainWindowGeometryX();
|
||||||
u32 getMainWindowGeometryY();
|
u32 getMainWindowGeometryY();
|
||||||
|
@ -55,13 +87,18 @@ std::string getGameInstallDir();
|
||||||
u32 getMainWindowTheme();
|
u32 getMainWindowTheme();
|
||||||
u32 getIconSize();
|
u32 getIconSize();
|
||||||
u32 getIconSizeGrid();
|
u32 getIconSizeGrid();
|
||||||
u32 getSliderPositon();
|
u32 getSliderPosition();
|
||||||
u32 getSliderPositonGrid();
|
u32 getSliderPositionGrid();
|
||||||
u32 getTableMode();
|
u32 getTableMode();
|
||||||
u32 getMainWindowWidth();
|
u32 getMainWindowWidth();
|
||||||
u32 getMainWindowHeight();
|
u32 getMainWindowHeight();
|
||||||
std::vector<std::string> getPkgViewer();
|
std::vector<std::string> getPkgViewer();
|
||||||
std::vector<std::string> getElfViewer();
|
std::vector<std::string> getElfViewer();
|
||||||
std::vector<std::string> getRecentFiles();
|
std::vector<std::string> getRecentFiles();
|
||||||
|
std::string getEmulatorLanguage();
|
||||||
|
|
||||||
|
void setDefaultValues();
|
||||||
|
|
||||||
|
// settings
|
||||||
|
u32 GetLanguage();
|
||||||
}; // namespace Config
|
}; // namespace Config
|
||||||
|
|
|
@ -29,7 +29,7 @@ static inline bool IsProfilerConnected() {
|
||||||
#define TRACK_ALLOC(ptr, size, pool) TracyAllocN(std::bit_cast<void*>(ptr), (size), (pool))
|
#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))
|
#define TRACK_FREE(ptr, pool) TracyFreeN(std::bit_cast<void*>(ptr), (pool))
|
||||||
|
|
||||||
enum MarkersPallete : int {
|
enum MarkersPalette : int {
|
||||||
EmulatorMarkerColor = 0x264653,
|
EmulatorMarkerColor = 0x264653,
|
||||||
RendererMarkerColor = 0x2a9d8f,
|
RendererMarkerColor = 0x2a9d8f,
|
||||||
HleMarkerColor = 0xe9c46a,
|
HleMarkerColor = 0xe9c46a,
|
||||||
|
|
|
@ -217,7 +217,7 @@ void IOFile::Close() {
|
||||||
file = nullptr;
|
file = nullptr;
|
||||||
|
|
||||||
#ifdef _WIN64
|
#ifdef _WIN64
|
||||||
if (file_mapping) {
|
if (file_mapping && file_access_mode == FileAccessMode::ReadWrite) {
|
||||||
CloseHandle(std::bit_cast<HANDLE>(file_mapping));
|
CloseHandle(std::bit_cast<HANDLE>(file_mapping));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -259,8 +259,7 @@ uintptr_t IOFile::GetFileMapping() {
|
||||||
mapping = CreateFileMapping2(hfile, NULL, FILE_MAP_WRITE, PAGE_READWRITE, SEC_COMMIT, 0,
|
mapping = CreateFileMapping2(hfile, NULL, FILE_MAP_WRITE, PAGE_READWRITE, SEC_COMMIT, 0,
|
||||||
NULL, NULL, 0);
|
NULL, NULL, 0);
|
||||||
} else {
|
} else {
|
||||||
mapping = CreateFileMapping2(hfile, NULL, FILE_MAP_READ, PAGE_READONLY, SEC_COMMIT, 0, NULL,
|
mapping = hfile;
|
||||||
NULL, 0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
file_mapping = std::bit_cast<uintptr_t>(mapping);
|
file_mapping = std::bit_cast<uintptr_t>(mapping);
|
||||||
|
|
|
@ -207,8 +207,8 @@ public:
|
||||||
message_queue.EmplaceWait(entry);
|
message_queue.EmplaceWait(entry);
|
||||||
} else {
|
} else {
|
||||||
ForEachBackend([&entry](auto& backend) { backend.Write(entry); });
|
ForEachBackend([&entry](auto& backend) { backend.Write(entry); });
|
||||||
|
std::fflush(stdout);
|
||||||
}
|
}
|
||||||
std::fflush(stdout);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
|
@ -106,11 +106,13 @@ bool ParseFilterRule(Filter& instance, Iterator begin, Iterator end) {
|
||||||
SUB(Lib, DiscMap) \
|
SUB(Lib, DiscMap) \
|
||||||
SUB(Lib, Png) \
|
SUB(Lib, Png) \
|
||||||
SUB(Lib, PlayGo) \
|
SUB(Lib, PlayGo) \
|
||||||
|
SUB(Lib, Random) \
|
||||||
SUB(Lib, Usbd) \
|
SUB(Lib, Usbd) \
|
||||||
SUB(Lib, Ajm) \
|
SUB(Lib, Ajm) \
|
||||||
SUB(Lib, ErrorDialog) \
|
SUB(Lib, ErrorDialog) \
|
||||||
SUB(Lib, ImeDialog) \
|
SUB(Lib, ImeDialog) \
|
||||||
SUB(Lib, AvPlayer) \
|
SUB(Lib, AvPlayer) \
|
||||||
|
SUB(Lib, Ngs2) \
|
||||||
CLS(Frontend) \
|
CLS(Frontend) \
|
||||||
CLS(Render) \
|
CLS(Render) \
|
||||||
SUB(Render, Vulkan) \
|
SUB(Render, Vulkan) \
|
||||||
|
|
|
@ -73,11 +73,13 @@ enum class Class : u8 {
|
||||||
Lib_DiscMap, ///< The LibSceDiscMap implementation.
|
Lib_DiscMap, ///< The LibSceDiscMap implementation.
|
||||||
Lib_Png, ///< The LibScePng implementation.
|
Lib_Png, ///< The LibScePng implementation.
|
||||||
Lib_PlayGo, ///< The LibScePlayGo implementation.
|
Lib_PlayGo, ///< The LibScePlayGo implementation.
|
||||||
|
Lib_Random, ///< The libSceRandom implementation.
|
||||||
Lib_Usbd, ///< The LibSceUsbd implementation.
|
Lib_Usbd, ///< The LibSceUsbd implementation.
|
||||||
Lib_Ajm, ///< The LibSceAjm implementation.
|
Lib_Ajm, ///< The LibSceAjm implementation.
|
||||||
Lib_ErrorDialog, ///< The LibSceErrorDialog implementation.
|
Lib_ErrorDialog, ///< The LibSceErrorDialog implementation.
|
||||||
Lib_ImeDialog, ///< The LibSceImeDialog implementation.
|
Lib_ImeDialog, ///< The LibSceImeDialog implementation.
|
||||||
Lib_AvPlayer, ///< The LibSceAvPlayer implementation.
|
Lib_AvPlayer, ///< The LibSceAvPlayer implementation.
|
||||||
|
Lib_Ngs2, ///< The LibSceNgs2 implementation.
|
||||||
Frontend, ///< Emulator UI
|
Frontend, ///< Emulator UI
|
||||||
Render, ///< Video Core
|
Render, ///< Video Core
|
||||||
Render_Vulkan, ///< Vulkan backend
|
Render_Vulkan, ///< Vulkan backend
|
||||||
|
|
|
@ -18,16 +18,16 @@ NativeClock::NativeClock()
|
||||||
us_rdtsc_factor{GetFixedPoint64Factor(std::micro::den, rdtsc_frequency)},
|
us_rdtsc_factor{GetFixedPoint64Factor(std::micro::den, rdtsc_frequency)},
|
||||||
ms_rdtsc_factor{GetFixedPoint64Factor(std::milli::den, rdtsc_frequency)} {}
|
ms_rdtsc_factor{GetFixedPoint64Factor(std::milli::den, rdtsc_frequency)} {}
|
||||||
|
|
||||||
u64 NativeClock::GetTimeNS() const {
|
u64 NativeClock::GetTimeNS(u64 base_ptc /*= 0*/) const {
|
||||||
return MultiplyHigh(GetUptime(), ns_rdtsc_factor);
|
return MultiplyHigh(GetUptime() - base_ptc, ns_rdtsc_factor);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 NativeClock::GetTimeUS() const {
|
u64 NativeClock::GetTimeUS(u64 base_ptc /*= 0*/) const {
|
||||||
return MultiplyHigh(GetUptime(), us_rdtsc_factor);
|
return MultiplyHigh(GetUptime() - base_ptc, us_rdtsc_factor);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 NativeClock::GetTimeMS() const {
|
u64 NativeClock::GetTimeMS(u64 base_ptc /*= 0*/) const {
|
||||||
return MultiplyHigh(GetUptime(), ms_rdtsc_factor);
|
return MultiplyHigh(GetUptime() - base_ptc, ms_rdtsc_factor);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 NativeClock::GetUptime() const {
|
u64 NativeClock::GetUptime() const {
|
||||||
|
|
|
@ -16,9 +16,9 @@ public:
|
||||||
return rdtsc_frequency;
|
return rdtsc_frequency;
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 GetTimeNS() const;
|
u64 GetTimeNS(u64 base_ptc = 0) const;
|
||||||
u64 GetTimeUS() const;
|
u64 GetTimeUS(u64 base_ptc = 0) const;
|
||||||
u64 GetTimeMS() const;
|
u64 GetTimeMS(u64 base_ptc = 0) const;
|
||||||
u64 GetUptime() const;
|
u64 GetUptime() const;
|
||||||
u64 GetProcessTimeUS() const;
|
u64 GetProcessTimeUS() const;
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace Shader {
|
namespace Common {
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
requires std::is_destructible_v<T>
|
requires std::is_destructible_v<T>
|
||||||
|
@ -104,4 +104,4 @@ private:
|
||||||
size_t new_chunk_size{};
|
size_t new_chunk_size{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Shader
|
} // namespace Common
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
#include <CoreFoundation/CFBundle.h>
|
#include <CoreFoundation/CFBundle.h>
|
||||||
|
#include <dlfcn.h>
|
||||||
#include <sys/param.h>
|
#include <sys/param.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -26,23 +27,52 @@ namespace Common::FS {
|
||||||
namespace fs = std::filesystem;
|
namespace fs = std::filesystem;
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
|
using IsTranslocatedURLFunc = Boolean (*)(CFURLRef path, bool* isTranslocated,
|
||||||
|
CFErrorRef* __nullable error);
|
||||||
|
using CreateOriginalPathForURLFunc = CFURLRef __nullable (*)(CFURLRef translocatedPath,
|
||||||
|
CFErrorRef* __nullable error);
|
||||||
|
|
||||||
|
static CFURLRef UntranslocateBundlePath(const CFURLRef bundle_path) {
|
||||||
|
if (void* security_handle =
|
||||||
|
dlopen("/System/Library/Frameworks/Security.framework/Security", RTLD_LAZY)) {
|
||||||
|
SCOPE_EXIT {
|
||||||
|
dlclose(security_handle);
|
||||||
|
};
|
||||||
|
|
||||||
|
const auto IsTranslocatedURL = reinterpret_cast<IsTranslocatedURLFunc>(
|
||||||
|
dlsym(security_handle, "SecTranslocateIsTranslocatedURL"));
|
||||||
|
const auto CreateOriginalPathForURL = reinterpret_cast<CreateOriginalPathForURLFunc>(
|
||||||
|
dlsym(security_handle, "SecTranslocateCreateOriginalPathForURL"));
|
||||||
|
|
||||||
|
bool is_translocated = false;
|
||||||
|
if (IsTranslocatedURL && CreateOriginalPathForURL &&
|
||||||
|
IsTranslocatedURL(bundle_path, &is_translocated, nullptr) && is_translocated) {
|
||||||
|
return CreateOriginalPathForURL(bundle_path, nullptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
static std::filesystem::path GetBundleParentDirectory() {
|
static std::filesystem::path GetBundleParentDirectory() {
|
||||||
if (CFBundleRef bundle_ref = CFBundleGetMainBundle()) {
|
if (CFBundleRef bundle_ref = CFBundleGetMainBundle()) {
|
||||||
if (CFURLRef bundle_url_ref = CFBundleCopyBundleURL(bundle_ref)) {
|
if (CFURLRef bundle_url_ref = CFBundleCopyBundleURL(bundle_ref)) {
|
||||||
SCOPE_EXIT {
|
SCOPE_EXIT {
|
||||||
CFRelease(bundle_url_ref);
|
CFRelease(bundle_url_ref);
|
||||||
};
|
};
|
||||||
if (CFStringRef bundle_path_ref =
|
|
||||||
CFURLCopyFileSystemPath(bundle_url_ref, kCFURLPOSIXPathStyle)) {
|
CFURLRef untranslocated_url_ref = UntranslocateBundlePath(bundle_url_ref);
|
||||||
SCOPE_EXIT {
|
SCOPE_EXIT {
|
||||||
CFRelease(bundle_path_ref);
|
if (untranslocated_url_ref) {
|
||||||
};
|
CFRelease(untranslocated_url_ref);
|
||||||
char app_bundle_path[MAXPATHLEN];
|
|
||||||
if (CFStringGetFileSystemRepresentation(bundle_path_ref, app_bundle_path,
|
|
||||||
sizeof(app_bundle_path))) {
|
|
||||||
std::filesystem::path bundle_path{app_bundle_path};
|
|
||||||
return bundle_path.parent_path();
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
char app_bundle_path[MAXPATHLEN];
|
||||||
|
if (CFURLGetFileSystemRepresentation(
|
||||||
|
untranslocated_url_ref ? untranslocated_url_ref : bundle_url_ref, true,
|
||||||
|
reinterpret_cast<u8*>(app_bundle_path), sizeof(app_bundle_path))) {
|
||||||
|
std::filesystem::path bundle_path{app_bundle_path};
|
||||||
|
return bundle_path.parent_path();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -74,6 +104,9 @@ static auto UserPaths = [] {
|
||||||
create_path(PathType::SysModuleDir, user_dir / SYSMODULES_DIR);
|
create_path(PathType::SysModuleDir, user_dir / SYSMODULES_DIR);
|
||||||
create_path(PathType::DownloadDir, user_dir / DOWNLOAD_DIR);
|
create_path(PathType::DownloadDir, user_dir / DOWNLOAD_DIR);
|
||||||
create_path(PathType::CapturesDir, user_dir / CAPTURES_DIR);
|
create_path(PathType::CapturesDir, user_dir / CAPTURES_DIR);
|
||||||
|
create_path(PathType::CheatsDir, user_dir / CHEATS_DIR);
|
||||||
|
create_path(PathType::PatchesDir, user_dir / PATCHES_DIR);
|
||||||
|
create_path(PathType::AddonsDir, user_dir / ADDONS_DIR);
|
||||||
|
|
||||||
return paths;
|
return paths;
|
||||||
}();
|
}();
|
||||||
|
|
|
@ -20,6 +20,9 @@ enum class PathType {
|
||||||
SysModuleDir, // Where system modules are stored.
|
SysModuleDir, // Where system modules are stored.
|
||||||
DownloadDir, // Where downloads/temp files are stored.
|
DownloadDir, // Where downloads/temp files are stored.
|
||||||
CapturesDir, // Where rdoc captures are stored.
|
CapturesDir, // Where rdoc captures are stored.
|
||||||
|
CheatsDir, // Where cheats are stored.
|
||||||
|
PatchesDir, // Where patches are stored.
|
||||||
|
AddonsDir, // Where additional content is stored.
|
||||||
};
|
};
|
||||||
|
|
||||||
constexpr auto PORTABLE_DIR = "user";
|
constexpr auto PORTABLE_DIR = "user";
|
||||||
|
@ -35,6 +38,9 @@ constexpr auto TEMPDATA_DIR = "temp";
|
||||||
constexpr auto SYSMODULES_DIR = "sys_modules";
|
constexpr auto SYSMODULES_DIR = "sys_modules";
|
||||||
constexpr auto DOWNLOAD_DIR = "download";
|
constexpr auto DOWNLOAD_DIR = "download";
|
||||||
constexpr auto CAPTURES_DIR = "captures";
|
constexpr auto CAPTURES_DIR = "captures";
|
||||||
|
constexpr auto CHEATS_DIR = "cheats";
|
||||||
|
constexpr auto PATCHES_DIR = "patches";
|
||||||
|
constexpr auto ADDONS_DIR = "addcont";
|
||||||
|
|
||||||
// Filenames
|
// Filenames
|
||||||
constexpr auto LOG_FILE = "shad_log.txt";
|
constexpr auto LOG_FILE = "shad_log.txt";
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "common/scm_rev.h"
|
||||||
|
|
||||||
|
#define GIT_REV "@GIT_REV@"
|
||||||
|
#define GIT_BRANCH "@GIT_BRANCH@"
|
||||||
|
#define GIT_DESC "@GIT_DESC@"
|
||||||
|
|
||||||
|
namespace Common {
|
||||||
|
|
||||||
|
const char g_scm_rev[] = GIT_REV;
|
||||||
|
const char g_scm_branch[] = GIT_BRANCH;
|
||||||
|
const char g_scm_desc[] = GIT_DESC;
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace Common {
|
||||||
|
|
||||||
|
extern const char g_scm_rev[];
|
||||||
|
extern const char g_scm_branch[];
|
||||||
|
extern const char g_scm_desc[];
|
||||||
|
|
||||||
|
} // namespace Common
|
|
@ -28,9 +28,13 @@ struct SlotId {
|
||||||
|
|
||||||
template <class T>
|
template <class T>
|
||||||
class SlotVector {
|
class SlotVector {
|
||||||
constexpr static std::size_t InitialCapacity = 1024;
|
constexpr static std::size_t InitialCapacity = 2048;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
SlotVector() {
|
||||||
|
Reserve(InitialCapacity);
|
||||||
|
}
|
||||||
|
|
||||||
~SlotVector() noexcept {
|
~SlotVector() noexcept {
|
||||||
std::size_t index = 0;
|
std::size_t index = 0;
|
||||||
for (u64 bits : stored_bitset) {
|
for (u64 bits : stored_bitset) {
|
||||||
|
@ -67,19 +71,6 @@ public:
|
||||||
return SlotId{index};
|
return SlotId{index};
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... Args>
|
|
||||||
[[nodiscard]] SlotId swap_and_insert(SlotId existing_id, Args&&... args) noexcept {
|
|
||||||
const u32 index = FreeValueIndex();
|
|
||||||
T& existing_value = values[existing_id.index].object;
|
|
||||||
|
|
||||||
new (&values[index].object) T(std::move(existing_value));
|
|
||||||
existing_value.~T();
|
|
||||||
new (&values[existing_id.index].object) T(std::forward<Args>(args)...);
|
|
||||||
SetStorageBit(index);
|
|
||||||
|
|
||||||
return SlotId{index};
|
|
||||||
}
|
|
||||||
|
|
||||||
void erase(SlotId id) noexcept {
|
void erase(SlotId id) noexcept {
|
||||||
values[id.index].object.~T();
|
values[id.index].object.~T();
|
||||||
free_list.push_back(id.index);
|
free_list.push_back(id.index);
|
||||||
|
@ -151,7 +142,8 @@ private:
|
||||||
|
|
||||||
const std::size_t old_free_size = free_list.size();
|
const std::size_t old_free_size = free_list.size();
|
||||||
free_list.resize(old_free_size + (new_capacity - values_capacity));
|
free_list.resize(old_free_size + (new_capacity - values_capacity));
|
||||||
std::iota(free_list.begin() + old_free_size, free_list.end(),
|
const std::size_t new_free_size = free_list.size();
|
||||||
|
std::iota(free_list.rbegin(), free_list.rbegin() + new_free_size - old_free_size,
|
||||||
static_cast<u32>(values_capacity));
|
static_cast<u32>(values_capacity));
|
||||||
|
|
||||||
delete[] values;
|
delete[] values;
|
||||||
|
|
|
@ -94,7 +94,7 @@ namespace Common {
|
||||||
|
|
||||||
// This function divides a u128 by a u32 value and produces two u64 values:
|
// This function divides a u128 by a u32 value and produces two u64 values:
|
||||||
// the result of division and the remainder
|
// the result of division and the remainder
|
||||||
[[nodiscard]] static inline std::pair<u64, u64> Divide128On32(u128 dividend, u32 divisor) {
|
[[nodiscard]] static inline std::pair<u64, u64> Divide128On32(const u128& dividend, u32 divisor) {
|
||||||
u64 remainder = dividend[0] % divisor;
|
u64 remainder = dividend[0] % divisor;
|
||||||
u64 accum = dividend[0] / divisor;
|
u64 accum = dividend[0] / divisor;
|
||||||
if (dividend[1] == 0)
|
if (dividend[1] == 0)
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
namespace Common {
|
||||||
|
|
||||||
|
/// General purpose function wrapper similar to std::function.
|
||||||
|
/// Unlike std::function, the captured values don't have to be copyable.
|
||||||
|
/// This class can be moved but not copied.
|
||||||
|
template <typename ResultType, typename... Args>
|
||||||
|
class UniqueFunction {
|
||||||
|
class CallableBase {
|
||||||
|
public:
|
||||||
|
virtual ~CallableBase() = default;
|
||||||
|
virtual ResultType operator()(Args&&...) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <typename Functor>
|
||||||
|
class Callable final : public CallableBase {
|
||||||
|
public:
|
||||||
|
Callable(Functor&& functor_) : functor{std::move(functor_)} {}
|
||||||
|
~Callable() override = default;
|
||||||
|
|
||||||
|
ResultType operator()(Args&&... args) override {
|
||||||
|
return functor(std::forward<Args>(args)...);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Functor functor;
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
UniqueFunction() = default;
|
||||||
|
|
||||||
|
template <typename Functor>
|
||||||
|
UniqueFunction(Functor&& functor)
|
||||||
|
: callable{std::make_unique<Callable<Functor>>(std::move(functor))} {}
|
||||||
|
|
||||||
|
UniqueFunction& operator=(UniqueFunction&& rhs) noexcept = default;
|
||||||
|
UniqueFunction(UniqueFunction&& rhs) noexcept = default;
|
||||||
|
|
||||||
|
UniqueFunction& operator=(const UniqueFunction&) = delete;
|
||||||
|
UniqueFunction(const UniqueFunction&) = delete;
|
||||||
|
|
||||||
|
ResultType operator()(Args&&... args) const {
|
||||||
|
return (*callable)(std::forward<Args>(args)...);
|
||||||
|
}
|
||||||
|
|
||||||
|
explicit operator bool() const noexcept {
|
||||||
|
return static_cast<bool>(callable);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::unique_ptr<CallableBase> callable;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Common
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
namespace Common {
|
namespace Common {
|
||||||
|
|
||||||
constexpr char VERSION[] = "0.1.1 WIP";
|
constexpr char VERSION[] = "0.2.1 WIP";
|
||||||
|
constexpr bool isRelease = false;
|
||||||
|
|
||||||
} // namespace Common
|
} // namespace Common
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
#include <boost/icl/separate_interval_set.hpp>
|
#include <boost/icl/separate_interval_set.hpp>
|
||||||
|
#include "common/alignment.h"
|
||||||
#include "common/assert.h"
|
#include "common/assert.h"
|
||||||
#include "common/error.h"
|
#include "common/error.h"
|
||||||
#include "core/address_space.h"
|
#include "core/address_space.h"
|
||||||
|
@ -98,7 +99,7 @@ struct AddressSpace::Impl {
|
||||||
backing_handle =
|
backing_handle =
|
||||||
CreateFileMapping2(INVALID_HANDLE_VALUE, nullptr, FILE_MAP_WRITE | FILE_MAP_READ,
|
CreateFileMapping2(INVALID_HANDLE_VALUE, nullptr, FILE_MAP_WRITE | FILE_MAP_READ,
|
||||||
PAGE_READWRITE, SEC_COMMIT, BackingSize, nullptr, nullptr, 0);
|
PAGE_READWRITE, SEC_COMMIT, BackingSize, nullptr, nullptr, 0);
|
||||||
ASSERT(backing_handle);
|
ASSERT_MSG(backing_handle, "{}", Common::GetLastErrorMsg());
|
||||||
// Allocate a virtual memory for the backing file map as placeholder
|
// Allocate a virtual memory for the backing file map as placeholder
|
||||||
backing_base = static_cast<u8*>(VirtualAlloc2(process, nullptr, BackingSize,
|
backing_base = static_cast<u8*>(VirtualAlloc2(process, nullptr, BackingSize,
|
||||||
MEM_RESERVE | MEM_RESERVE_PLACEHOLDER,
|
MEM_RESERVE | MEM_RESERVE_PLACEHOLDER,
|
||||||
|
@ -106,7 +107,7 @@ struct AddressSpace::Impl {
|
||||||
// Map backing placeholder. This will commit the pages
|
// Map backing placeholder. This will commit the pages
|
||||||
void* const ret = MapViewOfFile3(backing_handle, process, backing_base, 0, BackingSize,
|
void* const ret = MapViewOfFile3(backing_handle, process, backing_base, 0, BackingSize,
|
||||||
MEM_REPLACE_PLACEHOLDER, PAGE_READWRITE, nullptr, 0);
|
MEM_REPLACE_PLACEHOLDER, PAGE_READWRITE, nullptr, 0);
|
||||||
ASSERT(ret == backing_base);
|
ASSERT_MSG(ret == backing_base, "{}", Common::GetLastErrorMsg());
|
||||||
}
|
}
|
||||||
|
|
||||||
~Impl() {
|
~Impl() {
|
||||||
|
@ -129,9 +130,10 @@ struct AddressSpace::Impl {
|
||||||
}
|
}
|
||||||
|
|
||||||
void* Map(VAddr virtual_addr, PAddr phys_addr, size_t size, ULONG prot, uintptr_t fd = 0) {
|
void* Map(VAddr virtual_addr, PAddr phys_addr, size_t size, ULONG prot, uintptr_t fd = 0) {
|
||||||
|
const size_t aligned_size = Common::AlignUp(size, 16_KB);
|
||||||
const auto it = placeholders.find(virtual_addr);
|
const auto it = placeholders.find(virtual_addr);
|
||||||
ASSERT_MSG(it != placeholders.end(), "Cannot map already mapped region");
|
ASSERT_MSG(it != placeholders.end(), "Cannot map already mapped region");
|
||||||
ASSERT_MSG(virtual_addr >= it->lower() && virtual_addr + size <= it->upper(),
|
ASSERT_MSG(virtual_addr >= it->lower() && virtual_addr + aligned_size <= it->upper(),
|
||||||
"Map range must be fully contained in a placeholder");
|
"Map range must be fully contained in a placeholder");
|
||||||
|
|
||||||
// Windows only allows splitting a placeholder into two.
|
// Windows only allows splitting a placeholder into two.
|
||||||
|
@ -140,7 +142,7 @@ struct AddressSpace::Impl {
|
||||||
// one at the start and at the end.
|
// one at the start and at the end.
|
||||||
const VAddr placeholder_start = it->lower();
|
const VAddr placeholder_start = it->lower();
|
||||||
const VAddr placeholder_end = it->upper();
|
const VAddr placeholder_end = it->upper();
|
||||||
const VAddr virtual_end = virtual_addr + size;
|
const VAddr virtual_end = virtual_addr + aligned_size;
|
||||||
|
|
||||||
// If the placeholder doesn't exactly start at virtual_addr, split it at the start.
|
// If the placeholder doesn't exactly start at virtual_addr, split it at the start.
|
||||||
if (placeholder_start != virtual_addr) {
|
if (placeholder_start != virtual_addr) {
|
||||||
|
@ -161,11 +163,23 @@ struct AddressSpace::Impl {
|
||||||
void* ptr = nullptr;
|
void* ptr = nullptr;
|
||||||
if (phys_addr != -1) {
|
if (phys_addr != -1) {
|
||||||
HANDLE backing = fd ? reinterpret_cast<HANDLE>(fd) : backing_handle;
|
HANDLE backing = fd ? reinterpret_cast<HANDLE>(fd) : backing_handle;
|
||||||
ptr = MapViewOfFile3(backing, process, reinterpret_cast<PVOID>(virtual_addr), phys_addr,
|
if (fd && prot == PAGE_READONLY) {
|
||||||
size, MEM_REPLACE_PLACEHOLDER, prot, nullptr, 0);
|
DWORD resultvar;
|
||||||
|
ptr = VirtualAlloc2(process, reinterpret_cast<PVOID>(virtual_addr), aligned_size,
|
||||||
|
MEM_RESERVE | MEM_COMMIT | MEM_REPLACE_PLACEHOLDER,
|
||||||
|
PAGE_READWRITE, nullptr, 0);
|
||||||
|
bool ret = ReadFile(backing, ptr, size, &resultvar, NULL);
|
||||||
|
ASSERT_MSG(ret, "ReadFile failed. {}", Common::GetLastErrorMsg());
|
||||||
|
ret = VirtualProtect(ptr, size, prot, &resultvar);
|
||||||
|
ASSERT_MSG(ret, "VirtualProtect failed. {}", Common::GetLastErrorMsg());
|
||||||
|
} else {
|
||||||
|
ptr = MapViewOfFile3(backing, process, reinterpret_cast<PVOID>(virtual_addr),
|
||||||
|
phys_addr, aligned_size, MEM_REPLACE_PLACEHOLDER, prot,
|
||||||
|
nullptr, 0);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
ptr =
|
ptr =
|
||||||
VirtualAlloc2(process, reinterpret_cast<PVOID>(virtual_addr), size,
|
VirtualAlloc2(process, reinterpret_cast<PVOID>(virtual_addr), aligned_size,
|
||||||
MEM_RESERVE | MEM_COMMIT | MEM_REPLACE_PLACEHOLDER, prot, nullptr, 0);
|
MEM_RESERVE | MEM_COMMIT | MEM_REPLACE_PLACEHOLDER, prot, nullptr, 0);
|
||||||
}
|
}
|
||||||
ASSERT_MSG(ptr, "{}", Common::GetLastErrorMsg());
|
ASSERT_MSG(ptr, "{}", Common::GetLastErrorMsg());
|
||||||
|
@ -454,8 +468,28 @@ void* AddressSpace::MapFile(VAddr virtual_addr, size_t size, size_t offset, u32
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void AddressSpace::Unmap(VAddr virtual_addr, size_t size, bool has_backing) {
|
void AddressSpace::Unmap(VAddr virtual_addr, size_t size, VAddr start_in_vma, VAddr end_in_vma,
|
||||||
return impl->Unmap(virtual_addr, size, has_backing);
|
PAddr phys_base, bool is_exec, bool has_backing, bool readonly_file) {
|
||||||
|
#ifdef _WIN32
|
||||||
|
// There does not appear to be comparable support for partial unmapping on Windows.
|
||||||
|
// Unfortunately, a least one title was found to require this. The workaround is to unmap
|
||||||
|
// the entire allocation and remap the portions outside of the requested unmapping range.
|
||||||
|
impl->Unmap(virtual_addr, size, has_backing && !readonly_file);
|
||||||
|
|
||||||
|
// TODO: Determine if any titles require partial unmapping support for flexible allocations.
|
||||||
|
ASSERT_MSG(has_backing || (start_in_vma == 0 && end_in_vma == size),
|
||||||
|
"Partial unmapping of flexible allocations is not supported");
|
||||||
|
|
||||||
|
if (start_in_vma != 0) {
|
||||||
|
Map(virtual_addr, start_in_vma, 0, phys_base, is_exec);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (end_in_vma != size) {
|
||||||
|
Map(virtual_addr + end_in_vma, size - end_in_vma, 0, phys_base + end_in_vma, is_exec);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
impl->Unmap(virtual_addr + start_in_vma, end_in_vma - start_in_vma, has_backing);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void AddressSpace::Protect(VAddr virtual_addr, size_t size, MemoryPermission perms) {
|
void AddressSpace::Protect(VAddr virtual_addr, size_t size, MemoryPermission perms) {
|
||||||
|
|
|
@ -34,10 +34,7 @@ constexpr VAddr USER_MAX = 0xFBFFFFFFFFULL;
|
||||||
|
|
||||||
static constexpr size_t SystemManagedSize = SYSTEM_MANAGED_MAX - SYSTEM_MANAGED_MIN + 1;
|
static constexpr size_t SystemManagedSize = SYSTEM_MANAGED_MAX - SYSTEM_MANAGED_MIN + 1;
|
||||||
static constexpr size_t SystemReservedSize = SYSTEM_RESERVED_MAX - SYSTEM_RESERVED_MIN + 1;
|
static constexpr size_t SystemReservedSize = SYSTEM_RESERVED_MAX - SYSTEM_RESERVED_MIN + 1;
|
||||||
// User area size is normally larger than this. However games are unlikely to map to high
|
static constexpr size_t UserSize = 1ULL << 40;
|
||||||
// regions of that area, so by default we allocate a smaller virtual address space (about 1/4th).
|
|
||||||
// to save space on page tables.
|
|
||||||
static constexpr size_t UserSize = 1ULL << 39;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents the user virtual address space backed by a dmem memory block
|
* Represents the user virtual address space backed by a dmem memory block
|
||||||
|
@ -94,7 +91,8 @@ public:
|
||||||
void* MapFile(VAddr virtual_addr, size_t size, size_t offset, u32 prot, uintptr_t fd);
|
void* MapFile(VAddr virtual_addr, size_t size, size_t offset, u32 prot, uintptr_t fd);
|
||||||
|
|
||||||
/// Unmaps specified virtual memory area.
|
/// Unmaps specified virtual memory area.
|
||||||
void Unmap(VAddr virtual_addr, size_t size, bool has_backing);
|
void Unmap(VAddr virtual_addr, size_t size, VAddr start_in_vma, VAddr end_in_vma,
|
||||||
|
PAddr phys_base, bool is_exec, bool has_backing, bool readonly_file);
|
||||||
|
|
||||||
void Protect(VAddr virtual_addr, size_t size, MemoryPermission perms);
|
void Protect(VAddr virtual_addr, size_t size, MemoryPermission perms);
|
||||||
|
|
||||||
|
|
|
@ -114108,7 +114108,7 @@ STUB(
|
||||||
_ZN3sce2Np9CppWebApi6Common12IntrusivePtrINS1_7Matches2V124RequestCompetitiveResultEE7add_refEv)
|
_ZN3sce2Np9CppWebApi6Common12IntrusivePtrINS1_7Matches2V124RequestCompetitiveResultEE7add_refEv)
|
||||||
STUB("efPahl2FufA",
|
STUB("efPahl2FufA",
|
||||||
_ZN3sce2Np9CppWebApi30CommunicationRestrictionStatus2V35Error8fromJsonERKNS_4Json5ValueE)
|
_ZN3sce2Np9CppWebApi30CommunicationRestrictionStatus2V35Error8fromJsonERKNS_4Json5ValueE)
|
||||||
STUB("efX3lrPwdKA", sceAppContentAddcontMountByEntitlemetId)
|
STUB("efX3lrPwdKA", sceAppContentAddcontMountByEntitlementId)
|
||||||
STUB("efXnxYFN5oE", _ZNSt11range_errorD0Ev)
|
STUB("efXnxYFN5oE", _ZNSt11range_errorD0Ev)
|
||||||
STUB("efcwuDLsAM0", _ZThn120_NK7WebCore16HTMLMediaElement5mutedEv)
|
STUB("efcwuDLsAM0", _ZThn120_NK7WebCore16HTMLMediaElement5mutedEv)
|
||||||
STUB("efhGArzWdxE", _ZN7bmalloc6IsoTLS15s_didInitializeE)
|
STUB("efhGArzWdxE", _ZN7bmalloc6IsoTLS15s_didInitializeE)
|
||||||
|
@ -129493,7 +129493,7 @@ STUB(
|
||||||
STUB("kJlYH5uMAWI", sceNetResolverDestroy)
|
STUB("kJlYH5uMAWI", sceNetResolverDestroy)
|
||||||
STUB("kJmdxo4uM+8",
|
STUB("kJmdxo4uM+8",
|
||||||
_ZNSt9money_putIwSt19ostreambuf_iteratorIwSt11char_traitsIwEEE5_InitERKSt8_Locinfo)
|
_ZNSt9money_putIwSt19ostreambuf_iteratorIwSt11char_traitsIwEEE5_InitERKSt8_Locinfo)
|
||||||
STUB("kJmjt81mXKQ", sceAppContentAddcontEnqueueDownloadByEntitlemetId)
|
STUB("kJmjt81mXKQ", sceAppContentAddcontEnqueueDownloadByEntitlementId)
|
||||||
STUB(
|
STUB(
|
||||||
"kJoY9lMIFzY",
|
"kJoY9lMIFzY",
|
||||||
_ZN3sce2Np9CppWebApi6Common8IteratorINS2_12IntrusivePtrINS1_21AdvancedPlayerProfile2V138MatchCompletionRateDisconnectedMetricsEEEEmmEi)
|
_ZN3sce2Np9CppWebApi6Common8IteratorINS2_12IntrusivePtrINS1_21AdvancedPlayerProfile2V138MatchCompletionRateDisconnectedMetricsEEEEmmEi)
|
||||||
|
|
|
@ -13,7 +13,7 @@ namespace Core::AeroLib {
|
||||||
// on lookup, setting up the nid_entry they are matched with
|
// on lookup, setting up the nid_entry they are matched with
|
||||||
//
|
//
|
||||||
// If it runs out of stubs with name information, it will return
|
// If it runs out of stubs with name information, it will return
|
||||||
// a default implemetnation without function name details
|
// a default implementation without function name details
|
||||||
|
|
||||||
// Up to 512, larger values lead to more resolve stub slots
|
// Up to 512, larger values lead to more resolve stub slots
|
||||||
// and to longer compile / CI times
|
// and to longer compile / CI times
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
#else
|
#else
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
|
#include <half.hpp>
|
||||||
#include <sys/sysctl.h>
|
#include <sys/sysctl.h>
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
@ -30,6 +31,12 @@ static Xbyak::Reg ZydisToXbyakRegister(const ZydisRegister reg) {
|
||||||
if (reg >= ZYDIS_REGISTER_RAX && reg <= ZYDIS_REGISTER_R15) {
|
if (reg >= ZYDIS_REGISTER_RAX && reg <= ZYDIS_REGISTER_R15) {
|
||||||
return Xbyak::Reg64(reg - ZYDIS_REGISTER_RAX + Xbyak::Operand::RAX);
|
return Xbyak::Reg64(reg - ZYDIS_REGISTER_RAX + Xbyak::Operand::RAX);
|
||||||
}
|
}
|
||||||
|
if (reg >= ZYDIS_REGISTER_XMM0 && reg <= ZYDIS_REGISTER_XMM31) {
|
||||||
|
return Xbyak::Xmm(reg - ZYDIS_REGISTER_XMM0 + xmm0.getIdx());
|
||||||
|
}
|
||||||
|
if (reg >= ZYDIS_REGISTER_YMM0 && reg <= ZYDIS_REGISTER_YMM31) {
|
||||||
|
return Xbyak::Ymm(reg - ZYDIS_REGISTER_YMM0 + ymm0.getIdx());
|
||||||
|
}
|
||||||
UNREACHABLE_MSG("Unsupported register: {}", static_cast<u32>(reg));
|
UNREACHABLE_MSG("Unsupported register: {}", static_cast<u32>(reg));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -66,6 +73,12 @@ static Xbyak::Address ZydisToXbyakMemoryOperand(const ZydisDecodedOperand& opera
|
||||||
return ptr[expression];
|
return ptr[expression];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u64 ZydisToXbyakImmediateOperand(const ZydisDecodedOperand& operand) {
|
||||||
|
ASSERT_MSG(operand.type == ZYDIS_OPERAND_TYPE_IMMEDIATE,
|
||||||
|
"Expected immediate operand, got type: {}", static_cast<u32>(operand.type));
|
||||||
|
return operand.imm.value.u;
|
||||||
|
}
|
||||||
|
|
||||||
static std::unique_ptr<Xbyak::Operand> ZydisToXbyakOperand(const ZydisDecodedOperand& operand) {
|
static std::unique_ptr<Xbyak::Operand> ZydisToXbyakOperand(const ZydisDecodedOperand& operand) {
|
||||||
switch (operand.type) {
|
switch (operand.type) {
|
||||||
case ZYDIS_OPERAND_TYPE_REGISTER: {
|
case ZYDIS_OPERAND_TYPE_REGISTER: {
|
||||||
|
@ -110,51 +123,135 @@ static Xbyak::Reg AllocateScratchRegister(
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
|
|
||||||
static constexpr u32 MaxSavedRegisters = 3;
|
static pthread_key_t stack_pointer_slot;
|
||||||
static pthread_key_t register_save_slots[MaxSavedRegisters];
|
static pthread_key_t patch_stack_slot;
|
||||||
static std::once_flag register_save_init_flag;
|
static std::once_flag patch_context_slots_init_flag;
|
||||||
|
|
||||||
static_assert(sizeof(void*) == sizeof(u64),
|
static_assert(sizeof(void*) == sizeof(u64),
|
||||||
"Cannot fit a register inside a thread local storage slot.");
|
"Cannot fit a register inside a thread local storage slot.");
|
||||||
|
|
||||||
static void InitializeRegisterSaveSlots() {
|
static void InitializePatchContextSlots() {
|
||||||
for (u32 i = 0; i < MaxSavedRegisters; i++) {
|
ASSERT_MSG(pthread_key_create(&stack_pointer_slot, nullptr) == 0,
|
||||||
ASSERT_MSG(pthread_key_create(®ister_save_slots[i], nullptr) == 0,
|
"Unable to allocate thread-local register for stack pointer.");
|
||||||
"Unable to allocate thread-local register save slot {}", i);
|
ASSERT_MSG(pthread_key_create(&patch_stack_slot, nullptr) == 0,
|
||||||
|
"Unable to allocate thread-local register for patch stack.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void InitializeThreadPatchStack() {
|
||||||
|
std::call_once(patch_context_slots_init_flag, InitializePatchContextSlots);
|
||||||
|
|
||||||
|
const auto* patch_stack = std::malloc(0x1000);
|
||||||
|
pthread_setspecific(patch_stack_slot, patch_stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CleanupThreadPatchStack() {
|
||||||
|
std::call_once(patch_context_slots_init_flag, InitializePatchContextSlots);
|
||||||
|
|
||||||
|
auto* patch_stack = pthread_getspecific(patch_stack_slot);
|
||||||
|
if (patch_stack != nullptr) {
|
||||||
|
std::free(patch_stack);
|
||||||
|
pthread_setspecific(patch_stack_slot, nullptr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Saves the stack pointer to thread local storage and loads the patch stack.
|
||||||
|
static void SaveStack(Xbyak::CodeGenerator& c) {
|
||||||
|
std::call_once(patch_context_slots_init_flag, InitializePatchContextSlots);
|
||||||
|
|
||||||
|
// Save stack pointer and load patch stack.
|
||||||
|
c.putSeg(gs);
|
||||||
|
c.mov(qword[reinterpret_cast<void*>(stack_pointer_slot * sizeof(void*))], rsp);
|
||||||
|
c.putSeg(gs);
|
||||||
|
c.mov(rsp, qword[reinterpret_cast<void*>(patch_stack_slot * sizeof(void*))]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Restores the stack pointer from thread local storage.
|
||||||
|
static void RestoreStack(Xbyak::CodeGenerator& c) {
|
||||||
|
std::call_once(patch_context_slots_init_flag, InitializePatchContextSlots);
|
||||||
|
|
||||||
|
// Save patch stack pointer and load original stack.
|
||||||
|
c.putSeg(gs);
|
||||||
|
c.mov(qword[reinterpret_cast<void*>(patch_stack_slot * sizeof(void*))], rsp);
|
||||||
|
c.putSeg(gs);
|
||||||
|
c.mov(rsp, qword[reinterpret_cast<void*>(stack_pointer_slot * sizeof(void*))]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
// These utilities are not implemented as we can't save anything to thread local storage without
|
||||||
|
// temporary registers.
|
||||||
|
void InitializeThreadPatchStack() {
|
||||||
|
// No-op
|
||||||
|
}
|
||||||
|
|
||||||
|
void CleanupThreadPatchStack() {
|
||||||
|
// No-op
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Saves the stack pointer to thread local storage and loads the patch stack.
|
||||||
|
static void SaveStack(Xbyak::CodeGenerator& c) {
|
||||||
|
UNIMPLEMENTED();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Restores the stack pointer from thread local storage.
|
||||||
|
static void RestoreStack(Xbyak::CodeGenerator& c) {
|
||||||
|
UNIMPLEMENTED();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/// Switches to the patch stack, saves registers, and restores the original stack.
|
||||||
static void SaveRegisters(Xbyak::CodeGenerator& c, const std::initializer_list<Xbyak::Reg> regs) {
|
static void SaveRegisters(Xbyak::CodeGenerator& c, const std::initializer_list<Xbyak::Reg> regs) {
|
||||||
ASSERT_MSG(regs.size() <= MaxSavedRegisters, "Not enough space to save {} registers.",
|
SaveStack(c);
|
||||||
regs.size());
|
|
||||||
|
|
||||||
std::call_once(register_save_init_flag, &InitializeRegisterSaveSlots);
|
|
||||||
|
|
||||||
u32 index = 0;
|
|
||||||
for (const auto& reg : regs) {
|
for (const auto& reg : regs) {
|
||||||
const auto offset = reinterpret_cast<void*>(register_save_slots[index++] * sizeof(void*));
|
c.push(reg.cvt64());
|
||||||
|
|
||||||
c.putSeg(gs);
|
|
||||||
c.mov(qword[offset], reg.cvt64());
|
|
||||||
}
|
}
|
||||||
|
RestoreStack(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Switches to the patch stack, restores registers, and restores the original stack.
|
||||||
static void RestoreRegisters(Xbyak::CodeGenerator& c,
|
static void RestoreRegisters(Xbyak::CodeGenerator& c,
|
||||||
const std::initializer_list<Xbyak::Reg> regs) {
|
const std::initializer_list<Xbyak::Reg> regs) {
|
||||||
ASSERT_MSG(regs.size() <= MaxSavedRegisters, "Not enough space to restore {} registers.",
|
SaveStack(c);
|
||||||
regs.size());
|
|
||||||
|
|
||||||
std::call_once(register_save_init_flag, &InitializeRegisterSaveSlots);
|
|
||||||
|
|
||||||
u32 index = 0;
|
|
||||||
for (const auto& reg : regs) {
|
for (const auto& reg : regs) {
|
||||||
const auto offset = reinterpret_cast<void*>(register_save_slots[index++] * sizeof(void*));
|
c.pop(reg.cvt64());
|
||||||
|
}
|
||||||
|
RestoreStack(c);
|
||||||
|
}
|
||||||
|
|
||||||
c.putSeg(gs);
|
/// Switches to the patch stack and stores all registers.
|
||||||
c.mov(reg.cvt64(), qword[offset]);
|
static void SaveContext(Xbyak::CodeGenerator& c) {
|
||||||
|
SaveStack(c);
|
||||||
|
for (int reg = Xbyak::Operand::RAX; reg <= Xbyak::Operand::R15; reg++) {
|
||||||
|
c.push(Xbyak::Reg64(reg));
|
||||||
|
}
|
||||||
|
for (int reg = 0; reg <= 7; reg++) {
|
||||||
|
c.sub(rsp, 32);
|
||||||
|
c.vmovdqu(ptr[rsp], Xbyak::Ymm(reg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Restores all registers and restores the original stack.
|
||||||
|
/// If the destination is a register, it is not restored to preserve the output.
|
||||||
|
static void RestoreContext(Xbyak::CodeGenerator& c, const Xbyak::Operand& dst) {
|
||||||
|
for (int reg = 7; reg >= 0; reg--) {
|
||||||
|
if ((!dst.isXMM() && !dst.isYMM()) || dst.getIdx() != reg) {
|
||||||
|
c.vmovdqu(Xbyak::Ymm(reg), ptr[rsp]);
|
||||||
|
}
|
||||||
|
c.add(rsp, 32);
|
||||||
|
}
|
||||||
|
for (int reg = Xbyak::Operand::R15; reg >= Xbyak::Operand::RAX; reg--) {
|
||||||
|
if (!dst.isREG() || dst.getIdx() != reg) {
|
||||||
|
c.pop(Xbyak::Reg64(reg));
|
||||||
|
} else {
|
||||||
|
c.add(rsp, 4);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
RestoreStack(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef __APPLE__
|
||||||
|
|
||||||
static void GenerateANDN(const ZydisDecodedOperand* operands, Xbyak::CodeGenerator& c) {
|
static void GenerateANDN(const ZydisDecodedOperand* operands, Xbyak::CodeGenerator& c) {
|
||||||
const auto dst = ZydisToXbyakRegisterOperand(operands[0]);
|
const auto dst = ZydisToXbyakRegisterOperand(operands[0]);
|
||||||
const auto src1 = ZydisToXbyakRegisterOperand(operands[1]);
|
const auto src1 = ZydisToXbyakRegisterOperand(operands[1]);
|
||||||
|
@ -204,9 +301,9 @@ static void GenerateBEXTR(const ZydisDecodedOperand* operands, Xbyak::CodeGenera
|
||||||
c.and_(dst, scratch2);
|
c.and_(dst, scratch2);
|
||||||
|
|
||||||
if (dst.getIdx() == shift.getIdx()) {
|
if (dst.getIdx() == shift.getIdx()) {
|
||||||
RestoreRegisters(c, {scratch1, scratch2});
|
RestoreRegisters(c, {scratch2, scratch1});
|
||||||
} else {
|
} else {
|
||||||
RestoreRegisters(c, {scratch1, scratch2, shift});
|
RestoreRegisters(c, {shift, scratch2, scratch1});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -258,10 +355,138 @@ static void GenerateBLSR(const ZydisDecodedOperand* operands, Xbyak::CodeGenerat
|
||||||
RestoreRegisters(c, {scratch});
|
RestoreRegisters(c, {scratch});
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FilterRosetta2Only(const ZydisDecodedOperand*) {
|
static __attribute__((sysv_abi)) void PerformVCVTPH2PS(float* out, const half_float::half* in,
|
||||||
|
const u32 count) {
|
||||||
|
for (u32 i = 0; i < count; i++) {
|
||||||
|
out[i] = half_float::half_cast<float>(in[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void GenerateVCVTPH2PS(const ZydisDecodedOperand* operands, Xbyak::CodeGenerator& c) {
|
||||||
|
const auto dst = ZydisToXbyakRegisterOperand(operands[0]);
|
||||||
|
const auto src = ZydisToXbyakOperand(operands[1]);
|
||||||
|
|
||||||
|
const auto float_count = dst.getBit() / 32;
|
||||||
|
const auto byte_count = float_count * 4;
|
||||||
|
|
||||||
|
SaveContext(c);
|
||||||
|
|
||||||
|
// Allocate stack space for outputs and load into first parameter.
|
||||||
|
c.sub(rsp, byte_count);
|
||||||
|
c.mov(rdi, rsp);
|
||||||
|
|
||||||
|
if (src->isXMM()) {
|
||||||
|
// Allocate stack space for inputs and load into second parameter.
|
||||||
|
c.sub(rsp, byte_count);
|
||||||
|
c.mov(rsi, rsp);
|
||||||
|
|
||||||
|
// Move input to the allocated space.
|
||||||
|
c.movdqu(ptr[rsp], *reinterpret_cast<Xbyak::Xmm*>(src.get()));
|
||||||
|
} else {
|
||||||
|
c.lea(rsi, src->getAddress());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load float count into third parameter.
|
||||||
|
c.mov(rdx, float_count);
|
||||||
|
|
||||||
|
c.mov(rax, reinterpret_cast<u64>(PerformVCVTPH2PS));
|
||||||
|
c.call(rax);
|
||||||
|
|
||||||
|
if (src->isXMM()) {
|
||||||
|
// Clean up after inputs space.
|
||||||
|
c.add(rsp, byte_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load outputs into destination register and clean up space.
|
||||||
|
if (dst.isYMM()) {
|
||||||
|
c.vmovdqu(*reinterpret_cast<const Xbyak::Ymm*>(&dst), ptr[rsp]);
|
||||||
|
} else {
|
||||||
|
c.movdqu(*reinterpret_cast<const Xbyak::Xmm*>(&dst), ptr[rsp]);
|
||||||
|
}
|
||||||
|
c.add(rsp, byte_count);
|
||||||
|
|
||||||
|
RestoreContext(c, dst);
|
||||||
|
}
|
||||||
|
|
||||||
|
using SingleToHalfFloatConverter = half_float::half (*)(float);
|
||||||
|
static const SingleToHalfFloatConverter SingleToHalfFloatConverters[4] = {
|
||||||
|
half_float::half_cast<half_float::half, std::round_to_nearest, float>,
|
||||||
|
half_float::half_cast<half_float::half, std::round_toward_neg_infinity, float>,
|
||||||
|
half_float::half_cast<half_float::half, std::round_toward_infinity, float>,
|
||||||
|
half_float::half_cast<half_float::half, std::round_toward_zero, float>,
|
||||||
|
};
|
||||||
|
|
||||||
|
static __attribute__((sysv_abi)) void PerformVCVTPS2PH(half_float::half* out, const float* in,
|
||||||
|
const u32 count, const u8 rounding_mode) {
|
||||||
|
const auto conversion_func = SingleToHalfFloatConverters[rounding_mode];
|
||||||
|
|
||||||
|
for (u32 i = 0; i < count; i++) {
|
||||||
|
out[i] = conversion_func(in[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void GenerateVCVTPS2PH(const ZydisDecodedOperand* operands, Xbyak::CodeGenerator& c) {
|
||||||
|
const auto dst = ZydisToXbyakOperand(operands[0]);
|
||||||
|
const auto src = ZydisToXbyakRegisterOperand(operands[1]);
|
||||||
|
const auto ctrl = ZydisToXbyakImmediateOperand(operands[2]);
|
||||||
|
|
||||||
|
const auto float_count = src.getBit() / 32;
|
||||||
|
const auto byte_count = float_count * 4;
|
||||||
|
|
||||||
|
SaveContext(c);
|
||||||
|
|
||||||
|
if (dst->isXMM()) {
|
||||||
|
// Allocate stack space for outputs and load into first parameter.
|
||||||
|
c.sub(rsp, byte_count);
|
||||||
|
c.mov(rdi, rsp);
|
||||||
|
} else {
|
||||||
|
c.lea(rdi, dst->getAddress());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Allocate stack space for inputs and load into second parameter.
|
||||||
|
c.sub(rsp, byte_count);
|
||||||
|
c.mov(rsi, rsp);
|
||||||
|
|
||||||
|
// Move input to the allocated space.
|
||||||
|
if (src.isYMM()) {
|
||||||
|
c.vmovdqu(ptr[rsp], *reinterpret_cast<const Xbyak::Ymm*>(&src));
|
||||||
|
} else {
|
||||||
|
c.movdqu(ptr[rsp], *reinterpret_cast<const Xbyak::Xmm*>(&src));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load float count into third parameter.
|
||||||
|
c.mov(rdx, float_count);
|
||||||
|
|
||||||
|
// Load rounding mode into fourth parameter.
|
||||||
|
if (ctrl & 4) {
|
||||||
|
// Load from MXCSR.RC.
|
||||||
|
c.stmxcsr(ptr[rsp - 4]);
|
||||||
|
c.mov(rcx, ptr[rsp - 4]);
|
||||||
|
c.shr(rcx, 13);
|
||||||
|
c.and_(rcx, 3);
|
||||||
|
} else {
|
||||||
|
c.mov(rcx, ctrl & 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
c.mov(rax, reinterpret_cast<u64>(PerformVCVTPS2PH));
|
||||||
|
c.call(rax);
|
||||||
|
|
||||||
|
// Clean up after inputs space.
|
||||||
|
c.add(rsp, byte_count);
|
||||||
|
|
||||||
|
if (dst->isXMM()) {
|
||||||
|
// Load outputs into destination register and clean up space.
|
||||||
|
c.movdqu(*reinterpret_cast<Xbyak::Xmm*>(dst.get()), ptr[rsp]);
|
||||||
|
c.add(rsp, byte_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
RestoreContext(c, *dst);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool FilterRosetta2Only(const ZydisDecodedOperand*) {
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
size_t size = sizeof(ret);
|
size_t size = sizeof(ret);
|
||||||
if (sysctlbyname("sysctl.proc_translated", &ret, &size, NULL, 0) != 0) {
|
if (sysctlbyname("sysctl.proc_translated", &ret, &size, nullptr, 0) != 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
|
@ -339,12 +564,16 @@ static const std::unordered_map<ZydisMnemonic, PatchInfo> Patches = {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef __APPLE__
|
#ifdef __APPLE__
|
||||||
// BMI1 instructions that are not supported by Rosetta 2 on Apple Silicon.
|
// Patches for instruction sets not supported by Rosetta 2.
|
||||||
|
// BMI1
|
||||||
{ZYDIS_MNEMONIC_ANDN, {FilterRosetta2Only, GenerateANDN, true}},
|
{ZYDIS_MNEMONIC_ANDN, {FilterRosetta2Only, GenerateANDN, true}},
|
||||||
{ZYDIS_MNEMONIC_BEXTR, {FilterRosetta2Only, GenerateBEXTR, true}},
|
{ZYDIS_MNEMONIC_BEXTR, {FilterRosetta2Only, GenerateBEXTR, true}},
|
||||||
{ZYDIS_MNEMONIC_BLSI, {FilterRosetta2Only, GenerateBLSI, true}},
|
{ZYDIS_MNEMONIC_BLSI, {FilterRosetta2Only, GenerateBLSI, true}},
|
||||||
{ZYDIS_MNEMONIC_BLSMSK, {FilterRosetta2Only, GenerateBLSMSK, true}},
|
{ZYDIS_MNEMONIC_BLSMSK, {FilterRosetta2Only, GenerateBLSMSK, true}},
|
||||||
{ZYDIS_MNEMONIC_BLSR, {FilterRosetta2Only, GenerateBLSR, true}},
|
{ZYDIS_MNEMONIC_BLSR, {FilterRosetta2Only, GenerateBLSR, true}},
|
||||||
|
// F16C
|
||||||
|
{ZYDIS_MNEMONIC_VCVTPH2PS, {FilterRosetta2Only, GenerateVCVTPH2PS, true}},
|
||||||
|
{ZYDIS_MNEMONIC_VCVTPS2PH, {FilterRosetta2Only, GenerateVCVTPS2PH, true}},
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,12 @@ class CodeGenerator;
|
||||||
|
|
||||||
namespace Core {
|
namespace Core {
|
||||||
|
|
||||||
|
/// Initializes a stack for the current thread for use by patch implementations.
|
||||||
|
void InitializeThreadPatchStack();
|
||||||
|
|
||||||
|
/// Cleans up the patch stack for the current thread.
|
||||||
|
void CleanupThreadPatchStack();
|
||||||
|
|
||||||
/// Patches CPU instructions that cannot run as-is on the host.
|
/// Patches CPU instructions that cannot run as-is on the host.
|
||||||
void PatchInstructions(u64 segment_addr, u64 segment_size, Xbyak::CodeGenerator& c);
|
void PatchInstructions(u64 segment_addr, u64 segment_size, Xbyak::CodeGenerator& c);
|
||||||
|
|
||||||
|
|
|
@ -6,18 +6,18 @@
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey Crypto::key_pkg_derived_key3_keyset_init() {
|
CryptoPP::RSA::PrivateKey Crypto::key_pkg_derived_key3_keyset_init() {
|
||||||
CryptoPP::InvertibleRSAFunction params;
|
CryptoPP::InvertibleRSAFunction params;
|
||||||
params.SetPrime1(CryptoPP::Integer(pkg_derived_key3_keyset.Prime1, 0x80));
|
params.SetPrime1(CryptoPP::Integer(PkgDerivedKey3Keyset::Prime1, 0x80));
|
||||||
params.SetPrime2(CryptoPP::Integer(pkg_derived_key3_keyset.Prime2, 0x80));
|
params.SetPrime2(CryptoPP::Integer(PkgDerivedKey3Keyset::Prime2, 0x80));
|
||||||
|
|
||||||
params.SetPublicExponent(CryptoPP::Integer(pkg_derived_key3_keyset.PublicExponent, 4));
|
params.SetPublicExponent(CryptoPP::Integer(PkgDerivedKey3Keyset::PublicExponent, 4));
|
||||||
params.SetPrivateExponent(CryptoPP::Integer(pkg_derived_key3_keyset.PrivateExponent, 0x100));
|
params.SetPrivateExponent(CryptoPP::Integer(PkgDerivedKey3Keyset::PrivateExponent, 0x100));
|
||||||
|
|
||||||
params.SetModPrime1PrivateExponent(CryptoPP::Integer(pkg_derived_key3_keyset.Exponent1, 0x80));
|
params.SetModPrime1PrivateExponent(CryptoPP::Integer(PkgDerivedKey3Keyset::Exponent1, 0x80));
|
||||||
params.SetModPrime2PrivateExponent(CryptoPP::Integer(pkg_derived_key3_keyset.Exponent2, 0x80));
|
params.SetModPrime2PrivateExponent(CryptoPP::Integer(PkgDerivedKey3Keyset::Exponent2, 0x80));
|
||||||
|
|
||||||
params.SetModulus(CryptoPP::Integer(pkg_derived_key3_keyset.Modulus, 0x100));
|
params.SetModulus(CryptoPP::Integer(PkgDerivedKey3Keyset::Modulus, 0x100));
|
||||||
params.SetMultiplicativeInverseOfPrime2ModPrime1(
|
params.SetMultiplicativeInverseOfPrime2ModPrime1(
|
||||||
CryptoPP::Integer(pkg_derived_key3_keyset.Coefficient, 0x80));
|
CryptoPP::Integer(PkgDerivedKey3Keyset::Coefficient, 0x80));
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey privateKey(params);
|
CryptoPP::RSA::PrivateKey privateKey(params);
|
||||||
|
|
||||||
|
@ -26,18 +26,18 @@ CryptoPP::RSA::PrivateKey Crypto::key_pkg_derived_key3_keyset_init() {
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey Crypto::FakeKeyset_keyset_init() {
|
CryptoPP::RSA::PrivateKey Crypto::FakeKeyset_keyset_init() {
|
||||||
CryptoPP::InvertibleRSAFunction params;
|
CryptoPP::InvertibleRSAFunction params;
|
||||||
params.SetPrime1(CryptoPP::Integer(FakeKeyset_keyset.Prime1, 0x80));
|
params.SetPrime1(CryptoPP::Integer(FakeKeyset::Prime1, 0x80));
|
||||||
params.SetPrime2(CryptoPP::Integer(FakeKeyset_keyset.Prime2, 0x80));
|
params.SetPrime2(CryptoPP::Integer(FakeKeyset::Prime2, 0x80));
|
||||||
|
|
||||||
params.SetPublicExponent(CryptoPP::Integer(FakeKeyset_keyset.PublicExponent, 4));
|
params.SetPublicExponent(CryptoPP::Integer(FakeKeyset::PublicExponent, 4));
|
||||||
params.SetPrivateExponent(CryptoPP::Integer(FakeKeyset_keyset.PrivateExponent, 0x100));
|
params.SetPrivateExponent(CryptoPP::Integer(FakeKeyset::PrivateExponent, 0x100));
|
||||||
|
|
||||||
params.SetModPrime1PrivateExponent(CryptoPP::Integer(FakeKeyset_keyset.Exponent1, 0x80));
|
params.SetModPrime1PrivateExponent(CryptoPP::Integer(FakeKeyset::Exponent1, 0x80));
|
||||||
params.SetModPrime2PrivateExponent(CryptoPP::Integer(FakeKeyset_keyset.Exponent2, 0x80));
|
params.SetModPrime2PrivateExponent(CryptoPP::Integer(FakeKeyset::Exponent2, 0x80));
|
||||||
|
|
||||||
params.SetModulus(CryptoPP::Integer(FakeKeyset_keyset.Modulus, 0x100));
|
params.SetModulus(CryptoPP::Integer(FakeKeyset::Modulus, 0x100));
|
||||||
params.SetMultiplicativeInverseOfPrime2ModPrime1(
|
params.SetMultiplicativeInverseOfPrime2ModPrime1(
|
||||||
CryptoPP::Integer(FakeKeyset_keyset.Coefficient, 0x80));
|
CryptoPP::Integer(FakeKeyset::Coefficient, 0x80));
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey privateKey(params);
|
CryptoPP::RSA::PrivateKey privateKey(params);
|
||||||
|
|
||||||
|
@ -46,25 +46,22 @@ CryptoPP::RSA::PrivateKey Crypto::FakeKeyset_keyset_init() {
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey Crypto::DebugRifKeyset_init() {
|
CryptoPP::RSA::PrivateKey Crypto::DebugRifKeyset_init() {
|
||||||
CryptoPP::InvertibleRSAFunction params;
|
CryptoPP::InvertibleRSAFunction params;
|
||||||
params.SetPrime1(
|
params.SetPrime1(CryptoPP::Integer(DebugRifKeyset::Prime1, sizeof(DebugRifKeyset::Prime1)));
|
||||||
CryptoPP::Integer(DebugRifKeyset_keyset.Prime1, sizeof(DebugRifKeyset_keyset.Prime1)));
|
params.SetPrime2(CryptoPP::Integer(DebugRifKeyset::Prime2, sizeof(DebugRifKeyset::Prime2)));
|
||||||
params.SetPrime2(
|
|
||||||
CryptoPP::Integer(DebugRifKeyset_keyset.Prime2, sizeof(DebugRifKeyset_keyset.Prime2)));
|
|
||||||
|
|
||||||
params.SetPublicExponent(CryptoPP::Integer(DebugRifKeyset_keyset.PrivateExponent,
|
params.SetPublicExponent(
|
||||||
sizeof(DebugRifKeyset_keyset.PrivateExponent)));
|
CryptoPP::Integer(DebugRifKeyset::PublicExponent, sizeof(DebugRifKeyset::PublicExponent)));
|
||||||
params.SetPrivateExponent(CryptoPP::Integer(DebugRifKeyset_keyset.PrivateExponent,
|
params.SetPrivateExponent(CryptoPP::Integer(DebugRifKeyset::PrivateExponent,
|
||||||
sizeof(DebugRifKeyset_keyset.PrivateExponent)));
|
sizeof(DebugRifKeyset::PrivateExponent)));
|
||||||
|
|
||||||
params.SetModPrime1PrivateExponent(CryptoPP::Integer(DebugRifKeyset_keyset.Exponent1,
|
params.SetModPrime1PrivateExponent(
|
||||||
sizeof(DebugRifKeyset_keyset.Exponent1)));
|
CryptoPP::Integer(DebugRifKeyset::Exponent1, sizeof(DebugRifKeyset::Exponent1)));
|
||||||
params.SetModPrime2PrivateExponent(CryptoPP::Integer(DebugRifKeyset_keyset.Exponent2,
|
params.SetModPrime2PrivateExponent(
|
||||||
sizeof(DebugRifKeyset_keyset.Exponent2)));
|
CryptoPP::Integer(DebugRifKeyset::Exponent2, sizeof(DebugRifKeyset::Exponent2)));
|
||||||
|
|
||||||
params.SetModulus(
|
params.SetModulus(CryptoPP::Integer(DebugRifKeyset::Modulus, sizeof(DebugRifKeyset::Modulus)));
|
||||||
CryptoPP::Integer(DebugRifKeyset_keyset.Modulus, sizeof(DebugRifKeyset_keyset.Modulus)));
|
params.SetMultiplicativeInverseOfPrime2ModPrime1(
|
||||||
params.SetMultiplicativeInverseOfPrime2ModPrime1(CryptoPP::Integer(
|
CryptoPP::Integer(DebugRifKeyset::Coefficient, sizeof(DebugRifKeyset::Coefficient)));
|
||||||
DebugRifKeyset_keyset.Coefficient, sizeof(DebugRifKeyset_keyset.Coefficient)));
|
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey privateKey(params);
|
CryptoPP::RSA::PrivateKey privateKey(params);
|
||||||
|
|
||||||
|
|
|
@ -17,10 +17,6 @@
|
||||||
|
|
||||||
class Crypto {
|
class Crypto {
|
||||||
public:
|
public:
|
||||||
PkgDerivedKey3Keyset pkg_derived_key3_keyset;
|
|
||||||
FakeKeyset FakeKeyset_keyset;
|
|
||||||
DebugRifKeyset DebugRifKeyset_keyset;
|
|
||||||
|
|
||||||
CryptoPP::RSA::PrivateKey key_pkg_derived_key3_keyset_init();
|
CryptoPP::RSA::PrivateKey key_pkg_derived_key3_keyset_init();
|
||||||
CryptoPP::RSA::PrivateKey FakeKeyset_keyset_init();
|
CryptoPP::RSA::PrivateKey FakeKeyset_keyset_init();
|
||||||
CryptoPP::RSA::PrivateKey DebugRifKeyset_init();
|
CryptoPP::RSA::PrivateKey DebugRifKeyset_init();
|
||||||
|
|
|
@ -7,384 +7,299 @@
|
||||||
class FakeKeyset {
|
class FakeKeyset {
|
||||||
public:
|
public:
|
||||||
// Constructor
|
// Constructor
|
||||||
const CryptoPP::byte* Exponent1;
|
static constexpr CryptoPP::byte Exponent1[] = {
|
||||||
|
0x6D, 0x48, 0xE0, 0x54, 0x40, 0x25, 0xC8, 0x41, 0x29, 0x52, 0x42, 0x27, 0xEB, 0xD2, 0xC7,
|
||||||
|
0xAB, 0x6B, 0x9C, 0x27, 0x0A, 0xB4, 0x1F, 0x94, 0x4E, 0xFA, 0x42, 0x1D, 0xB7, 0xBC, 0xB9,
|
||||||
|
0xAE, 0xBC, 0x04, 0x6F, 0x75, 0x8F, 0x10, 0x5F, 0x89, 0xAC, 0xAB, 0x9C, 0xD2, 0xFA, 0xE6,
|
||||||
|
0xA4, 0x13, 0x83, 0x68, 0xD4, 0x56, 0x38, 0xFE, 0xE5, 0x2B, 0x78, 0x44, 0x9C, 0x34, 0xE6,
|
||||||
|
0x5A, 0xA0, 0xBE, 0x05, 0x70, 0xAD, 0x15, 0xC3, 0x2D, 0x31, 0xAC, 0x97, 0x5D, 0x88, 0xFC,
|
||||||
|
0xC1, 0x62, 0x3D, 0xE2, 0xED, 0x11, 0xDB, 0xB6, 0x9E, 0xFC, 0x5A, 0x5A, 0x03, 0xF6, 0xCF,
|
||||||
|
0x08, 0xD4, 0x5D, 0x90, 0xC9, 0x2A, 0xB9, 0x9B, 0xCF, 0xC8, 0x1A, 0x65, 0xF3, 0x5B, 0xE8,
|
||||||
|
0x7F, 0xCF, 0xA5, 0xA6, 0x4C, 0x5C, 0x2A, 0x12, 0x0F, 0x92, 0xA5, 0xE3, 0xF0, 0x17, 0x1E,
|
||||||
|
0x9A, 0x97, 0x45, 0x86, 0xFD, 0xDB, 0x54, 0x25};
|
||||||
// exponent2 = d mod (q - 1)
|
// exponent2 = d mod (q - 1)
|
||||||
const CryptoPP::byte* Exponent2;
|
static constexpr CryptoPP::byte Exponent2[] = {
|
||||||
|
0x2A, 0x51, 0xCE, 0x02, 0x44, 0x28, 0x50, 0xE8, 0x30, 0x20, 0x7C, 0x9C, 0x55, 0xBF, 0x60,
|
||||||
|
0x39, 0xBC, 0xD1, 0xF0, 0xE7, 0x68, 0xF8, 0x08, 0x5B, 0x61, 0x1F, 0xA7, 0xBF, 0xD0, 0xE8,
|
||||||
|
0x8B, 0xB5, 0xB1, 0xD5, 0xD9, 0x16, 0xAC, 0x75, 0x0C, 0x6D, 0xF2, 0xE0, 0xB5, 0x97, 0x75,
|
||||||
|
0xD2, 0x68, 0x16, 0x1F, 0x00, 0x7D, 0x8B, 0x17, 0xE8, 0x78, 0x48, 0x41, 0x71, 0x2B, 0x18,
|
||||||
|
0x96, 0x80, 0x11, 0xDB, 0x68, 0x39, 0x9C, 0xD6, 0xE0, 0x72, 0x42, 0x86, 0xF0, 0x1B, 0x16,
|
||||||
|
0x0D, 0x3E, 0x12, 0x94, 0x3D, 0x25, 0xA8, 0xA9, 0x30, 0x9E, 0x54, 0x5A, 0xD6, 0x36, 0x6C,
|
||||||
|
0xD6, 0x8C, 0x20, 0x62, 0x8F, 0xA1, 0x6B, 0x1F, 0x7C, 0x6D, 0xB2, 0xB1, 0xC1, 0x2E, 0xAD,
|
||||||
|
0x36, 0x02, 0x9C, 0x3A, 0xCA, 0x2F, 0x09, 0xD2, 0x45, 0x9E, 0xEB, 0xF2, 0xBC, 0x6C, 0xAA,
|
||||||
|
0x3B, 0x3E, 0x90, 0xBC, 0x38, 0x67, 0x35, 0x4D};
|
||||||
// e
|
// e
|
||||||
const CryptoPP::byte* PublicExponent;
|
static constexpr CryptoPP::byte PublicExponent[] = {0, 1, 0, 1};
|
||||||
// (InverseQ)(q) = 1 mod p
|
// (InverseQ)(q) = 1 mod p
|
||||||
const CryptoPP::byte* Coefficient;
|
static constexpr CryptoPP::byte Coefficient[] = {
|
||||||
|
0x0B, 0x67, 0x1C, 0x0D, 0x6C, 0x57, 0xD3, 0xE7, 0x05, 0x65, 0x94, 0x31, 0x56, 0x55, 0xFD,
|
||||||
|
0x28, 0x08, 0xFA, 0x05, 0x8A, 0xCC, 0x55, 0x39, 0x61, 0x97, 0x63, 0xA0, 0x16, 0x27, 0x3D,
|
||||||
|
0xED, 0xC1, 0x16, 0x40, 0x2A, 0x12, 0xEA, 0x6F, 0xD9, 0xD8, 0x58, 0x56, 0xA8, 0x56, 0x8B,
|
||||||
|
0x0D, 0x38, 0x5E, 0x1E, 0x80, 0x3B, 0x5F, 0x40, 0x80, 0x6F, 0x62, 0x4F, 0x28, 0xA2, 0x69,
|
||||||
|
0xF3, 0xD3, 0xF7, 0xFD, 0xB2, 0xC3, 0x52, 0x43, 0x20, 0x92, 0x9D, 0x97, 0x8D, 0xA0, 0x15,
|
||||||
|
0x07, 0x15, 0x6E, 0xA4, 0x0D, 0x56, 0xD3, 0x37, 0x1A, 0xC4, 0x9E, 0xDF, 0x02, 0x49, 0xB8,
|
||||||
|
0x0A, 0x84, 0x62, 0xF5, 0xFA, 0xB9, 0x3F, 0xA4, 0x09, 0x76, 0xCC, 0xAA, 0xB9, 0x9B, 0xA6,
|
||||||
|
0x4F, 0xC1, 0x6A, 0x64, 0xCE, 0xD8, 0x77, 0xAB, 0x4B, 0xF9, 0xA0, 0xAE, 0xDA, 0xF1, 0x67,
|
||||||
|
0x87, 0x7C, 0x98, 0x5C, 0x7E, 0xB8, 0x73, 0xF5};
|
||||||
// n = p * q
|
// n = p * q
|
||||||
const CryptoPP::byte* Modulus;
|
static constexpr CryptoPP::byte Modulus[] = {
|
||||||
|
0xC6, 0xCF, 0x71, 0xE7, 0xE5, 0x9A, 0xF0, 0xD1, 0x2A, 0x2C, 0x45, 0x8B, 0xF9, 0x2A, 0x0E,
|
||||||
|
0xC1, 0x43, 0x05, 0x8B, 0xC3, 0x71, 0x17, 0x80, 0x1D, 0xCD, 0x49, 0x7D, 0xDE, 0x35, 0x9D,
|
||||||
|
0x25, 0x9B, 0xA0, 0xD7, 0xA0, 0xF2, 0x7D, 0x6C, 0x08, 0x7E, 0xAA, 0x55, 0x02, 0x68, 0x2B,
|
||||||
|
0x23, 0xC6, 0x44, 0xB8, 0x44, 0x18, 0xEB, 0x56, 0xCF, 0x16, 0xA2, 0x48, 0x03, 0xC9, 0xE7,
|
||||||
|
0x4F, 0x87, 0xEB, 0x3D, 0x30, 0xC3, 0x15, 0x88, 0xBF, 0x20, 0xE7, 0x9D, 0xFF, 0x77, 0x0C,
|
||||||
|
0xDE, 0x1D, 0x24, 0x1E, 0x63, 0xA9, 0x4F, 0x8A, 0xBF, 0x5B, 0xBE, 0x60, 0x19, 0x68, 0x33,
|
||||||
|
0x3B, 0xFC, 0xED, 0x9F, 0x47, 0x4E, 0x5F, 0xF8, 0xEA, 0xCB, 0x3D, 0x00, 0xBD, 0x67, 0x01,
|
||||||
|
0xF9, 0x2C, 0x6D, 0xC6, 0xAC, 0x13, 0x64, 0xE7, 0x67, 0x14, 0xF3, 0xDC, 0x52, 0x69, 0x6A,
|
||||||
|
0xB9, 0x83, 0x2C, 0x42, 0x30, 0x13, 0x1B, 0xB2, 0xD8, 0xA5, 0x02, 0x0D, 0x79, 0xED, 0x96,
|
||||||
|
0xB1, 0x0D, 0xF8, 0xCC, 0x0C, 0xDF, 0x81, 0x95, 0x4F, 0x03, 0x58, 0x09, 0x57, 0x0E, 0x80,
|
||||||
|
0x69, 0x2E, 0xFE, 0xFF, 0x52, 0x77, 0xEA, 0x75, 0x28, 0xA8, 0xFB, 0xC9, 0xBE, 0xBF, 0x9F,
|
||||||
|
0xBB, 0xB7, 0x79, 0x8E, 0x18, 0x05, 0xE1, 0x80, 0xBD, 0x50, 0x34, 0x94, 0x81, 0xD3, 0x53,
|
||||||
|
0xC2, 0x69, 0xA2, 0xD2, 0x4C, 0xCF, 0x6C, 0xF4, 0x57, 0x2C, 0x10, 0x4A, 0x3F, 0xFB, 0x22,
|
||||||
|
0xFD, 0x8B, 0x97, 0xE2, 0xC9, 0x5B, 0xA6, 0x2B, 0xCD, 0xD6, 0x1B, 0x6B, 0xDB, 0x68, 0x7F,
|
||||||
|
0x4B, 0xC2, 0xA0, 0x50, 0x34, 0xC0, 0x05, 0xE5, 0x8D, 0xEF, 0x24, 0x67, 0xFF, 0x93, 0x40,
|
||||||
|
0xCF, 0x2D, 0x62, 0xA2, 0xA0, 0x50, 0xB1, 0xF1, 0x3A, 0xA8, 0x3D, 0xFD, 0x80, 0xD1, 0xF9,
|
||||||
|
0xB8, 0x05, 0x22, 0xAF, 0xC8, 0x35, 0x45, 0x90, 0x58, 0x8E, 0xE3, 0x3A, 0x7C, 0xBD, 0x3E,
|
||||||
|
0x27};
|
||||||
// p
|
// p
|
||||||
const CryptoPP::byte* Prime1;
|
static constexpr CryptoPP::byte Prime1[] = {
|
||||||
|
0xFE, 0xF6, 0xBF, 0x1D, 0x69, 0xAB, 0x16, 0x25, 0x08, 0x47, 0x55, 0x6B, 0x86, 0xE4, 0x35,
|
||||||
|
0x88, 0x72, 0x2A, 0xB1, 0x3D, 0xF8, 0xB6, 0x44, 0xCA, 0xB3, 0xAB, 0x19, 0xD1, 0x04, 0x24,
|
||||||
|
0x28, 0x0A, 0x74, 0x55, 0xB8, 0x15, 0x45, 0x09, 0xCC, 0x13, 0x1C, 0xF2, 0xBA, 0x37, 0xA9,
|
||||||
|
0x03, 0x90, 0x8F, 0x02, 0x10, 0xFF, 0x25, 0x79, 0x86, 0xCC, 0x18, 0x50, 0x9A, 0x10, 0x5F,
|
||||||
|
0x5B, 0x4C, 0x1C, 0x4E, 0xB0, 0xA7, 0xE3, 0x59, 0xB1, 0x2D, 0xA0, 0xC6, 0xB0, 0x20, 0x2C,
|
||||||
|
0x21, 0x33, 0x12, 0xB3, 0xAF, 0x72, 0x34, 0x83, 0xCD, 0x52, 0x2F, 0xAF, 0x0F, 0x20, 0x5A,
|
||||||
|
0x1B, 0xC0, 0xE2, 0xA3, 0x76, 0x34, 0x0F, 0xD7, 0xFC, 0xC1, 0x41, 0xC9, 0xF9, 0x79, 0x40,
|
||||||
|
0x17, 0x42, 0x21, 0x3E, 0x9D, 0xFD, 0xC7, 0xC1, 0x50, 0xDE, 0x44, 0x5A, 0xC9, 0x31, 0x89,
|
||||||
|
0x6A, 0x78, 0x05, 0xBE, 0x65, 0xB4, 0xE8, 0x2D};
|
||||||
// q
|
// q
|
||||||
const CryptoPP::byte* Prime2;
|
static constexpr CryptoPP::byte Prime2[] = {
|
||||||
const CryptoPP::byte* PrivateExponent;
|
0xC7, 0x9E, 0x47, 0x58, 0x00, 0x7D, 0x62, 0x82, 0xB0, 0xD2, 0x22, 0x81, 0xD4, 0xA8, 0x97,
|
||||||
|
0x1B, 0x79, 0x0C, 0x3A, 0xB0, 0xD7, 0xC9, 0x30, 0xE3, 0xC3, 0x53, 0x8E, 0x57, 0xEF, 0xF0,
|
||||||
// Constructor
|
0x9B, 0x9F, 0xB3, 0x90, 0x52, 0xC6, 0x94, 0x22, 0x36, 0xAA, 0xE6, 0x4A, 0x5F, 0x72, 0x1D,
|
||||||
FakeKeyset() {
|
0x70, 0xE8, 0x76, 0x58, 0xC8, 0xB2, 0x91, 0xCE, 0x9C, 0xC3, 0xE9, 0x09, 0x7F, 0x2E, 0x47,
|
||||||
// Initialize PrivateExponent
|
0x97, 0xCC, 0x90, 0x39, 0x15, 0x35, 0x31, 0xDE, 0x1F, 0x0C, 0x8C, 0x0D, 0xC1, 0xC2, 0x92,
|
||||||
PrivateExponent = new CryptoPP::byte[0x100]{
|
0xBE, 0x97, 0xBF, 0x2F, 0x91, 0xA1, 0x8C, 0x7D, 0x50, 0xA8, 0x21, 0x2F, 0xD7, 0xA2, 0x9A,
|
||||||
0x7F, 0x76, 0xCD, 0x0E, 0xE2, 0xD4, 0xDE, 0x05, 0x1C, 0xC6, 0xD9, 0xA8, 0x0E, 0x8D,
|
0x7E, 0xB5, 0xA7, 0x2A, 0x90, 0x02, 0xD9, 0xF3, 0x3D, 0xD1, 0xEB, 0xB8, 0xE0, 0x5A, 0x79,
|
||||||
0xFA, 0x7B, 0xCA, 0x1E, 0xAA, 0x27, 0x1A, 0x40, 0xF8, 0xF1, 0x22, 0x87, 0x35, 0xDD,
|
0x9E, 0x7D, 0x8D, 0xCA, 0x18, 0x6D, 0xBD, 0x9E, 0xA1, 0x80, 0x28, 0x6B, 0x2A, 0xFE, 0x51,
|
||||||
0xDB, 0xFD, 0xEE, 0xF8, 0xC2, 0xBC, 0xBD, 0x01, 0xFB, 0x8B, 0xE2, 0x3E, 0x63, 0xB2,
|
0x24, 0x9B, 0x6F, 0x4D, 0x84, 0x77, 0x80, 0x23};
|
||||||
0xB1, 0x22, 0x5C, 0x56, 0x49, 0x6E, 0x11, 0xBE, 0x07, 0x44, 0x0B, 0x9A, 0x26, 0x66,
|
static constexpr CryptoPP::byte PrivateExponent[] = {
|
||||||
0xD1, 0x49, 0x2C, 0x8F, 0xD3, 0x1B, 0xCF, 0xA4, 0xA1, 0xB8, 0xD1, 0xFB, 0xA4, 0x9E,
|
0x7F, 0x76, 0xCD, 0x0E, 0xE2, 0xD4, 0xDE, 0x05, 0x1C, 0xC6, 0xD9, 0xA8, 0x0E, 0x8D, 0xFA,
|
||||||
0xD2, 0x21, 0x28, 0x83, 0x09, 0x8A, 0xF6, 0xA0, 0x0B, 0xA3, 0xD6, 0x0F, 0x9B, 0x63,
|
0x7B, 0xCA, 0x1E, 0xAA, 0x27, 0x1A, 0x40, 0xF8, 0xF1, 0x22, 0x87, 0x35, 0xDD, 0xDB, 0xFD,
|
||||||
0x68, 0xCC, 0xBC, 0x0C, 0x4E, 0x14, 0x5B, 0x27, 0xA4, 0xA9, 0xF4, 0x2B, 0xB9, 0xB8,
|
0xEE, 0xF8, 0xC2, 0xBC, 0xBD, 0x01, 0xFB, 0x8B, 0xE2, 0x3E, 0x63, 0xB2, 0xB1, 0x22, 0x5C,
|
||||||
0x7B, 0xC0, 0xE6, 0x51, 0xAD, 0x1D, 0x77, 0xD4, 0x6B, 0xB9, 0xCE, 0x20, 0xD1, 0x26,
|
0x56, 0x49, 0x6E, 0x11, 0xBE, 0x07, 0x44, 0x0B, 0x9A, 0x26, 0x66, 0xD1, 0x49, 0x2C, 0x8F,
|
||||||
0x66, 0x7E, 0x5E, 0x9E, 0xA2, 0xE9, 0x6B, 0x90, 0xF3, 0x73, 0xB8, 0x52, 0x8F, 0x44,
|
0xD3, 0x1B, 0xCF, 0xA4, 0xA1, 0xB8, 0xD1, 0xFB, 0xA4, 0x9E, 0xD2, 0x21, 0x28, 0x83, 0x09,
|
||||||
0x11, 0x03, 0x0C, 0x13, 0x97, 0x39, 0x3D, 0x13, 0x22, 0x58, 0xD5, 0x43, 0x82, 0x49,
|
0x8A, 0xF6, 0xA0, 0x0B, 0xA3, 0xD6, 0x0F, 0x9B, 0x63, 0x68, 0xCC, 0xBC, 0x0C, 0x4E, 0x14,
|
||||||
0xDA, 0x6E, 0x7C, 0xA1, 0xC5, 0x8C, 0xA5, 0xB0, 0x09, 0xE0, 0xCE, 0x3D, 0xDF, 0xF4,
|
0x5B, 0x27, 0xA4, 0xA9, 0xF4, 0x2B, 0xB9, 0xB8, 0x7B, 0xC0, 0xE6, 0x51, 0xAD, 0x1D, 0x77,
|
||||||
0x9D, 0x3C, 0x97, 0x15, 0xE2, 0x6A, 0xC7, 0x2B, 0x3C, 0x50, 0x93, 0x23, 0xDB, 0xBA,
|
0xD4, 0x6B, 0xB9, 0xCE, 0x20, 0xD1, 0x26, 0x66, 0x7E, 0x5E, 0x9E, 0xA2, 0xE9, 0x6B, 0x90,
|
||||||
0x4A, 0x22, 0x66, 0x44, 0xAC, 0x78, 0xBB, 0x0E, 0x1A, 0x27, 0x43, 0xB5, 0x71, 0x67,
|
0xF3, 0x73, 0xB8, 0x52, 0x8F, 0x44, 0x11, 0x03, 0x0C, 0x13, 0x97, 0x39, 0x3D, 0x13, 0x22,
|
||||||
0xAF, 0xF4, 0xAB, 0x48, 0x46, 0x93, 0x73, 0xD0, 0x42, 0xAB, 0x93, 0x63, 0xE5, 0x6C,
|
0x58, 0xD5, 0x43, 0x82, 0x49, 0xDA, 0x6E, 0x7C, 0xA1, 0xC5, 0x8C, 0xA5, 0xB0, 0x09, 0xE0,
|
||||||
0x9A, 0xDE, 0x50, 0x24, 0xC0, 0x23, 0x7D, 0x99, 0x79, 0x3F, 0x22, 0x07, 0xE0, 0xC1,
|
0xCE, 0x3D, 0xDF, 0xF4, 0x9D, 0x3C, 0x97, 0x15, 0xE2, 0x6A, 0xC7, 0x2B, 0x3C, 0x50, 0x93,
|
||||||
0x48, 0x56, 0x1B, 0xDF, 0x83, 0x09, 0x12, 0xB4, 0x2D, 0x45, 0x6B, 0xC9, 0xC0, 0x68,
|
0x23, 0xDB, 0xBA, 0x4A, 0x22, 0x66, 0x44, 0xAC, 0x78, 0xBB, 0x0E, 0x1A, 0x27, 0x43, 0xB5,
|
||||||
0x85, 0x99, 0x90, 0x79, 0x96, 0x1A, 0xD7, 0xF5, 0x4D, 0x1F, 0x37, 0x83, 0x40, 0x4A,
|
0x71, 0x67, 0xAF, 0xF4, 0xAB, 0x48, 0x46, 0x93, 0x73, 0xD0, 0x42, 0xAB, 0x93, 0x63, 0xE5,
|
||||||
0xEC, 0x39, 0x37, 0xA6, 0x80, 0x92, 0x7D, 0xC5, 0x80, 0xC7, 0xD6, 0x6F, 0xFE, 0x8A,
|
0x6C, 0x9A, 0xDE, 0x50, 0x24, 0xC0, 0x23, 0x7D, 0x99, 0x79, 0x3F, 0x22, 0x07, 0xE0, 0xC1,
|
||||||
0x79, 0x89, 0xC6, 0xB1};
|
0x48, 0x56, 0x1B, 0xDF, 0x83, 0x09, 0x12, 0xB4, 0x2D, 0x45, 0x6B, 0xC9, 0xC0, 0x68, 0x85,
|
||||||
|
0x99, 0x90, 0x79, 0x96, 0x1A, 0xD7, 0xF5, 0x4D, 0x1F, 0x37, 0x83, 0x40, 0x4A, 0xEC, 0x39,
|
||||||
// Initialize Exponent1
|
0x37, 0xA6, 0x80, 0x92, 0x7D, 0xC5, 0x80, 0xC7, 0xD6, 0x6F, 0xFE, 0x8A, 0x79, 0x89, 0xC6,
|
||||||
Exponent1 = new CryptoPP::byte[0x80]{
|
0xB1};
|
||||||
0x6D, 0x48, 0xE0, 0x54, 0x40, 0x25, 0xC8, 0x41, 0x29, 0x52, 0x42, 0x27, 0xEB,
|
|
||||||
0xD2, 0xC7, 0xAB, 0x6B, 0x9C, 0x27, 0x0A, 0xB4, 0x1F, 0x94, 0x4E, 0xFA, 0x42,
|
|
||||||
0x1D, 0xB7, 0xBC, 0xB9, 0xAE, 0xBC, 0x04, 0x6F, 0x75, 0x8F, 0x10, 0x5F, 0x89,
|
|
||||||
0xAC, 0xAB, 0x9C, 0xD2, 0xFA, 0xE6, 0xA4, 0x13, 0x83, 0x68, 0xD4, 0x56, 0x38,
|
|
||||||
0xFE, 0xE5, 0x2B, 0x78, 0x44, 0x9C, 0x34, 0xE6, 0x5A, 0xA0, 0xBE, 0x05, 0x70,
|
|
||||||
0xAD, 0x15, 0xC3, 0x2D, 0x31, 0xAC, 0x97, 0x5D, 0x88, 0xFC, 0xC1, 0x62, 0x3D,
|
|
||||||
0xE2, 0xED, 0x11, 0xDB, 0xB6, 0x9E, 0xFC, 0x5A, 0x5A, 0x03, 0xF6, 0xCF, 0x08,
|
|
||||||
0xD4, 0x5D, 0x90, 0xC9, 0x2A, 0xB9, 0x9B, 0xCF, 0xC8, 0x1A, 0x65, 0xF3, 0x5B,
|
|
||||||
0xE8, 0x7F, 0xCF, 0xA5, 0xA6, 0x4C, 0x5C, 0x2A, 0x12, 0x0F, 0x92, 0xA5, 0xE3,
|
|
||||||
0xF0, 0x17, 0x1E, 0x9A, 0x97, 0x45, 0x86, 0xFD, 0xDB, 0x54, 0x25
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
Exponent2 = new CryptoPP::byte[0x80]{
|
|
||||||
0x2A, 0x51, 0xCE, 0x02, 0x44, 0x28, 0x50, 0xE8, 0x30, 0x20, 0x7C, 0x9C, 0x55,
|
|
||||||
0xBF, 0x60, 0x39, 0xBC, 0xD1, 0xF0, 0xE7, 0x68, 0xF8, 0x08, 0x5B, 0x61, 0x1F,
|
|
||||||
0xA7, 0xBF, 0xD0, 0xE8, 0x8B, 0xB5, 0xB1, 0xD5, 0xD9, 0x16, 0xAC, 0x75, 0x0C,
|
|
||||||
0x6D, 0xF2, 0xE0, 0xB5, 0x97, 0x75, 0xD2, 0x68, 0x16, 0x1F, 0x00, 0x7D, 0x8B,
|
|
||||||
0x17, 0xE8, 0x78, 0x48, 0x41, 0x71, 0x2B, 0x18, 0x96, 0x80, 0x11, 0xDB, 0x68,
|
|
||||||
0x39, 0x9C, 0xD6, 0xE0, 0x72, 0x42, 0x86, 0xF0, 0x1B, 0x16, 0x0D, 0x3E, 0x12,
|
|
||||||
0x94, 0x3D, 0x25, 0xA8, 0xA9, 0x30, 0x9E, 0x54, 0x5A, 0xD6, 0x36, 0x6C, 0xD6,
|
|
||||||
0x8C, 0x20, 0x62, 0x8F, 0xA1, 0x6B, 0x1F, 0x7C, 0x6D, 0xB2, 0xB1, 0xC1, 0x2E,
|
|
||||||
0xAD, 0x36, 0x02, 0x9C, 0x3A, 0xCA, 0x2F, 0x09, 0xD2, 0x45, 0x9E, 0xEB, 0xF2,
|
|
||||||
0xBC, 0x6C, 0xAA, 0x3B, 0x3E, 0x90, 0xBC, 0x38, 0x67, 0x35, 0x4D};
|
|
||||||
|
|
||||||
PublicExponent = new CryptoPP::byte[4]{0, 1, 0, 1};
|
|
||||||
|
|
||||||
Coefficient = new CryptoPP::byte[0x80]{
|
|
||||||
0x0B, 0x67, 0x1C, 0x0D, 0x6C, 0x57, 0xD3, 0xE7, 0x05, 0x65, 0x94, 0x31, 0x56,
|
|
||||||
0x55, 0xFD, 0x28, 0x08, 0xFA, 0x05, 0x8A, 0xCC, 0x55, 0x39, 0x61, 0x97, 0x63,
|
|
||||||
0xA0, 0x16, 0x27, 0x3D, 0xED, 0xC1, 0x16, 0x40, 0x2A, 0x12, 0xEA, 0x6F, 0xD9,
|
|
||||||
0xD8, 0x58, 0x56, 0xA8, 0x56, 0x8B, 0x0D, 0x38, 0x5E, 0x1E, 0x80, 0x3B, 0x5F,
|
|
||||||
0x40, 0x80, 0x6F, 0x62, 0x4F, 0x28, 0xA2, 0x69, 0xF3, 0xD3, 0xF7, 0xFD, 0xB2,
|
|
||||||
0xC3, 0x52, 0x43, 0x20, 0x92, 0x9D, 0x97, 0x8D, 0xA0, 0x15, 0x07, 0x15, 0x6E,
|
|
||||||
0xA4, 0x0D, 0x56, 0xD3, 0x37, 0x1A, 0xC4, 0x9E, 0xDF, 0x02, 0x49, 0xB8, 0x0A,
|
|
||||||
0x84, 0x62, 0xF5, 0xFA, 0xB9, 0x3F, 0xA4, 0x09, 0x76, 0xCC, 0xAA, 0xB9, 0x9B,
|
|
||||||
0xA6, 0x4F, 0xC1, 0x6A, 0x64, 0xCE, 0xD8, 0x77, 0xAB, 0x4B, 0xF9, 0xA0, 0xAE,
|
|
||||||
0xDA, 0xF1, 0x67, 0x87, 0x7C, 0x98, 0x5C, 0x7E, 0xB8, 0x73, 0xF5};
|
|
||||||
|
|
||||||
Modulus = new CryptoPP::byte[0x100]{
|
|
||||||
0xC6, 0xCF, 0x71, 0xE7, 0xE5, 0x9A, 0xF0, 0xD1, 0x2A, 0x2C, 0x45, 0x8B, 0xF9, 0x2A,
|
|
||||||
0x0E, 0xC1, 0x43, 0x05, 0x8B, 0xC3, 0x71, 0x17, 0x80, 0x1D, 0xCD, 0x49, 0x7D, 0xDE,
|
|
||||||
0x35, 0x9D, 0x25, 0x9B, 0xA0, 0xD7, 0xA0, 0xF2, 0x7D, 0x6C, 0x08, 0x7E, 0xAA, 0x55,
|
|
||||||
0x02, 0x68, 0x2B, 0x23, 0xC6, 0x44, 0xB8, 0x44, 0x18, 0xEB, 0x56, 0xCF, 0x16, 0xA2,
|
|
||||||
0x48, 0x03, 0xC9, 0xE7, 0x4F, 0x87, 0xEB, 0x3D, 0x30, 0xC3, 0x15, 0x88, 0xBF, 0x20,
|
|
||||||
0xE7, 0x9D, 0xFF, 0x77, 0x0C, 0xDE, 0x1D, 0x24, 0x1E, 0x63, 0xA9, 0x4F, 0x8A, 0xBF,
|
|
||||||
0x5B, 0xBE, 0x60, 0x19, 0x68, 0x33, 0x3B, 0xFC, 0xED, 0x9F, 0x47, 0x4E, 0x5F, 0xF8,
|
|
||||||
0xEA, 0xCB, 0x3D, 0x00, 0xBD, 0x67, 0x01, 0xF9, 0x2C, 0x6D, 0xC6, 0xAC, 0x13, 0x64,
|
|
||||||
0xE7, 0x67, 0x14, 0xF3, 0xDC, 0x52, 0x69, 0x6A, 0xB9, 0x83, 0x2C, 0x42, 0x30, 0x13,
|
|
||||||
0x1B, 0xB2, 0xD8, 0xA5, 0x02, 0x0D, 0x79, 0xED, 0x96, 0xB1, 0x0D, 0xF8, 0xCC, 0x0C,
|
|
||||||
0xDF, 0x81, 0x95, 0x4F, 0x03, 0x58, 0x09, 0x57, 0x0E, 0x80, 0x69, 0x2E, 0xFE, 0xFF,
|
|
||||||
0x52, 0x77, 0xEA, 0x75, 0x28, 0xA8, 0xFB, 0xC9, 0xBE, 0xBF, 0x9F, 0xBB, 0xB7, 0x79,
|
|
||||||
0x8E, 0x18, 0x05, 0xE1, 0x80, 0xBD, 0x50, 0x34, 0x94, 0x81, 0xD3, 0x53, 0xC2, 0x69,
|
|
||||||
0xA2, 0xD2, 0x4C, 0xCF, 0x6C, 0xF4, 0x57, 0x2C, 0x10, 0x4A, 0x3F, 0xFB, 0x22, 0xFD,
|
|
||||||
0x8B, 0x97, 0xE2, 0xC9, 0x5B, 0xA6, 0x2B, 0xCD, 0xD6, 0x1B, 0x6B, 0xDB, 0x68, 0x7F,
|
|
||||||
0x4B, 0xC2, 0xA0, 0x50, 0x34, 0xC0, 0x05, 0xE5, 0x8D, 0xEF, 0x24, 0x67, 0xFF, 0x93,
|
|
||||||
0x40, 0xCF, 0x2D, 0x62, 0xA2, 0xA0, 0x50, 0xB1, 0xF1, 0x3A, 0xA8, 0x3D, 0xFD, 0x80,
|
|
||||||
0xD1, 0xF9, 0xB8, 0x05, 0x22, 0xAF, 0xC8, 0x35, 0x45, 0x90, 0x58, 0x8E, 0xE3, 0x3A,
|
|
||||||
0x7C, 0xBD, 0x3E, 0x27};
|
|
||||||
|
|
||||||
Prime1 = new CryptoPP::byte[0x80]{
|
|
||||||
0xFE, 0xF6, 0xBF, 0x1D, 0x69, 0xAB, 0x16, 0x25, 0x08, 0x47, 0x55, 0x6B, 0x86,
|
|
||||||
0xE4, 0x35, 0x88, 0x72, 0x2A, 0xB1, 0x3D, 0xF8, 0xB6, 0x44, 0xCA, 0xB3, 0xAB,
|
|
||||||
0x19, 0xD1, 0x04, 0x24, 0x28, 0x0A, 0x74, 0x55, 0xB8, 0x15, 0x45, 0x09, 0xCC,
|
|
||||||
0x13, 0x1C, 0xF2, 0xBA, 0x37, 0xA9, 0x03, 0x90, 0x8F, 0x02, 0x10, 0xFF, 0x25,
|
|
||||||
0x79, 0x86, 0xCC, 0x18, 0x50, 0x9A, 0x10, 0x5F, 0x5B, 0x4C, 0x1C, 0x4E, 0xB0,
|
|
||||||
0xA7, 0xE3, 0x59, 0xB1, 0x2D, 0xA0, 0xC6, 0xB0, 0x20, 0x2C, 0x21, 0x33, 0x12,
|
|
||||||
0xB3, 0xAF, 0x72, 0x34, 0x83, 0xCD, 0x52, 0x2F, 0xAF, 0x0F, 0x20, 0x5A, 0x1B,
|
|
||||||
0xC0, 0xE2, 0xA3, 0x76, 0x34, 0x0F, 0xD7, 0xFC, 0xC1, 0x41, 0xC9, 0xF9, 0x79,
|
|
||||||
0x40, 0x17, 0x42, 0x21, 0x3E, 0x9D, 0xFD, 0xC7, 0xC1, 0x50, 0xDE, 0x44, 0x5A,
|
|
||||||
0xC9, 0x31, 0x89, 0x6A, 0x78, 0x05, 0xBE, 0x65, 0xB4, 0xE8, 0x2D};
|
|
||||||
|
|
||||||
Prime2 = new CryptoPP::byte[0x80]{
|
|
||||||
0xC7, 0x9E, 0x47, 0x58, 0x00, 0x7D, 0x62, 0x82, 0xB0, 0xD2, 0x22, 0x81, 0xD4,
|
|
||||||
0xA8, 0x97, 0x1B, 0x79, 0x0C, 0x3A, 0xB0, 0xD7, 0xC9, 0x30, 0xE3, 0xC3, 0x53,
|
|
||||||
0x8E, 0x57, 0xEF, 0xF0, 0x9B, 0x9F, 0xB3, 0x90, 0x52, 0xC6, 0x94, 0x22, 0x36,
|
|
||||||
0xAA, 0xE6, 0x4A, 0x5F, 0x72, 0x1D, 0x70, 0xE8, 0x76, 0x58, 0xC8, 0xB2, 0x91,
|
|
||||||
0xCE, 0x9C, 0xC3, 0xE9, 0x09, 0x7F, 0x2E, 0x47, 0x97, 0xCC, 0x90, 0x39, 0x15,
|
|
||||||
0x35, 0x31, 0xDE, 0x1F, 0x0C, 0x8C, 0x0D, 0xC1, 0xC2, 0x92, 0xBE, 0x97, 0xBF,
|
|
||||||
0x2F, 0x91, 0xA1, 0x8C, 0x7D, 0x50, 0xA8, 0x21, 0x2F, 0xD7, 0xA2, 0x9A, 0x7E,
|
|
||||||
0xB5, 0xA7, 0x2A, 0x90, 0x02, 0xD9, 0xF3, 0x3D, 0xD1, 0xEB, 0xB8, 0xE0, 0x5A,
|
|
||||||
0x79, 0x9E, 0x7D, 0x8D, 0xCA, 0x18, 0x6D, 0xBD, 0x9E, 0xA1, 0x80, 0x28, 0x6B,
|
|
||||||
0x2A, 0xFE, 0x51, 0x24, 0x9B, 0x6F, 0x4D, 0x84, 0x77, 0x80, 0x23};
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class DebugRifKeyset {
|
class DebugRifKeyset {
|
||||||
public:
|
public:
|
||||||
// Constructor
|
|
||||||
// std::uint8_t* PrivateExponent;
|
// std::uint8_t* PrivateExponent;
|
||||||
const CryptoPP::byte* Exponent1;
|
static constexpr CryptoPP::byte Exponent1[] = {
|
||||||
|
0xCD, 0x9A, 0x61, 0xB0, 0xB8, 0xD5, 0xB4, 0xE4, 0xE4, 0xF6, 0xAB, 0xF7, 0x27, 0xB7, 0x56,
|
||||||
|
0x59, 0x6B, 0xB9, 0x11, 0xE7, 0xF4, 0x83, 0xAF, 0xB9, 0x73, 0x99, 0x7F, 0x49, 0xA2, 0x9C,
|
||||||
|
0xF0, 0xB5, 0x6D, 0x37, 0x82, 0x14, 0x15, 0xF1, 0x04, 0x8A, 0xD4, 0x8E, 0xEB, 0x2E, 0x1F,
|
||||||
|
0xE2, 0x81, 0xA9, 0x62, 0x6E, 0xB1, 0x68, 0x75, 0x62, 0xF3, 0x0F, 0xFE, 0xD4, 0x91, 0x87,
|
||||||
|
0x98, 0x78, 0xBF, 0x26, 0xB5, 0x07, 0x58, 0xD0, 0xEE, 0x3F, 0x21, 0xE8, 0xC8, 0x0F, 0x5F,
|
||||||
|
0xFA, 0x1C, 0x64, 0x74, 0x49, 0x52, 0xEB, 0xE7, 0xEE, 0xDE, 0xBA, 0x23, 0x26, 0x4A, 0xF6,
|
||||||
|
0x9C, 0x1A, 0x09, 0x3F, 0xB9, 0x0B, 0x36, 0x26, 0x1A, 0xBE, 0xA9, 0x76, 0xE6, 0xF2, 0x69,
|
||||||
|
0xDE, 0xFF, 0xAF, 0xCC, 0x0C, 0x9A, 0x66, 0x03, 0x86, 0x0A, 0x1F, 0x49, 0xA4, 0x10, 0xB6,
|
||||||
|
0xBC, 0xC3, 0x7C, 0x88, 0xE8, 0xCE, 0x4B, 0xD9};
|
||||||
// exponent2 = d mod (q - 1)
|
// exponent2 = d mod (q - 1)
|
||||||
const CryptoPP::byte* Exponent2;
|
static constexpr CryptoPP::byte Exponent2[] = {
|
||||||
|
0xB3, 0x73, 0xA3, 0x59, 0xE6, 0x97, 0xC0, 0xAB, 0x3B, 0x68, 0xFC, 0x39, 0xAC, 0xDB, 0x44,
|
||||||
|
0xB1, 0xB4, 0x9E, 0x35, 0x4D, 0xBE, 0xC5, 0x36, 0x69, 0x6C, 0x3D, 0xC5, 0xFC, 0xFE, 0x4B,
|
||||||
|
0x2F, 0xDC, 0x86, 0x80, 0x46, 0x96, 0x40, 0x1A, 0x0D, 0x6E, 0xFA, 0x8C, 0xE0, 0x47, 0x91,
|
||||||
|
0xAC, 0xAD, 0x95, 0x2B, 0x8E, 0x1F, 0xF2, 0x0A, 0x45, 0xF8, 0x29, 0x95, 0x70, 0xC6, 0x88,
|
||||||
|
0x5F, 0x71, 0x03, 0x99, 0x79, 0xBC, 0x84, 0x71, 0xBD, 0xE8, 0x84, 0x8C, 0x0E, 0xD4, 0x7B,
|
||||||
|
0x30, 0x74, 0x57, 0x1A, 0x95, 0xE7, 0x90, 0x19, 0x8D, 0xAD, 0x8B, 0x4C, 0x4E, 0xC3, 0xE7,
|
||||||
|
0x6B, 0x23, 0x86, 0x01, 0xEE, 0x9B, 0xE0, 0x2F, 0x15, 0xA2, 0x2C, 0x4C, 0x39, 0xD3, 0xDF,
|
||||||
|
0x9C, 0x39, 0x01, 0xF1, 0x8C, 0x44, 0x4A, 0x15, 0x44, 0xDC, 0x51, 0xF7, 0x22, 0xD7, 0x7F,
|
||||||
|
0x41, 0x7F, 0x68, 0xFA, 0xEE, 0x56, 0xE8, 0x05};
|
||||||
// e
|
// e
|
||||||
const CryptoPP::byte* PublicExponent;
|
static constexpr CryptoPP::byte PublicExponent[] = {0x00, 0x01, 0x00, 0x01};
|
||||||
// (InverseQ)(q) = 1 mod p
|
// (InverseQ)(q) = 1 mod p
|
||||||
const CryptoPP::byte* Coefficient;
|
static constexpr CryptoPP::byte Coefficient[] = {
|
||||||
|
0xC0, 0x32, 0x43, 0xD3, 0x8C, 0x3D, 0xB4, 0xD2, 0x48, 0x8C, 0x42, 0x41, 0x24, 0x94, 0x6C,
|
||||||
|
0x80, 0xC9, 0xC1, 0x79, 0x36, 0x7F, 0xAC, 0xC3, 0xFF, 0x6A, 0x25, 0xEB, 0x2C, 0xFB, 0xD4,
|
||||||
|
0x2B, 0xA0, 0xEB, 0xFE, 0x25, 0xE9, 0xC6, 0x77, 0xCE, 0xFE, 0x2D, 0x23, 0xFE, 0xD0, 0xF4,
|
||||||
|
0x0F, 0xD9, 0x7E, 0xD5, 0xA5, 0x7D, 0x1F, 0xC0, 0xE8, 0xE8, 0xEC, 0x80, 0x5B, 0xC7, 0xFD,
|
||||||
|
0xE2, 0xBD, 0x94, 0xA6, 0x2B, 0xDD, 0x6A, 0x60, 0x45, 0x54, 0xAB, 0xCA, 0x42, 0x9C, 0x6A,
|
||||||
|
0x6C, 0xBF, 0x3C, 0x84, 0xF9, 0xA5, 0x0E, 0x63, 0x0C, 0x51, 0x58, 0x62, 0x6D, 0x5A, 0xB7,
|
||||||
|
0x3C, 0x3F, 0x49, 0x1A, 0xD0, 0x93, 0xB8, 0x4F, 0x1A, 0x6C, 0x5F, 0xC5, 0xE5, 0xA9, 0x75,
|
||||||
|
0xD4, 0x86, 0x9E, 0xDF, 0x87, 0x0F, 0x27, 0xB0, 0x26, 0x78, 0x4E, 0xFB, 0xC1, 0x8A, 0x4A,
|
||||||
|
0x24, 0x3F, 0x7F, 0x8F, 0x9A, 0x12, 0x51, 0xCB};
|
||||||
// n = p * q
|
// n = p * q
|
||||||
const CryptoPP::byte* Modulus;
|
static constexpr CryptoPP::byte Modulus[] = {
|
||||||
|
0xC2, 0xD2, 0x44, 0xBC, 0xDD, 0x84, 0x3F, 0xD9, 0xC5, 0x22, 0xAF, 0xF7, 0xFC, 0x88, 0x8A,
|
||||||
|
0x33, 0x80, 0xED, 0x8E, 0xE2, 0xCC, 0x81, 0xF7, 0xEC, 0xF8, 0x1C, 0x79, 0xBF, 0x02, 0xBB,
|
||||||
|
0x12, 0x8E, 0x61, 0x68, 0x29, 0x1B, 0x15, 0xB6, 0x5E, 0xC6, 0xF8, 0xBF, 0x5A, 0xE0, 0x3B,
|
||||||
|
0x6A, 0x6C, 0xD9, 0xD6, 0xF5, 0x75, 0xAB, 0xA0, 0x6F, 0x34, 0x81, 0x34, 0x9A, 0x5B, 0xAD,
|
||||||
|
0xED, 0x31, 0xE3, 0xC6, 0xEA, 0x1A, 0xD1, 0x13, 0x22, 0xBB, 0xB3, 0xDA, 0xB3, 0xB2, 0x53,
|
||||||
|
0xBD, 0x45, 0x79, 0x87, 0xAD, 0x0A, 0x01, 0x72, 0x18, 0x10, 0x29, 0x49, 0xF4, 0x41, 0x7F,
|
||||||
|
0xD6, 0x47, 0x0C, 0x72, 0x92, 0x9E, 0xE9, 0xBB, 0x95, 0xA9, 0x5D, 0x79, 0xEB, 0xE4, 0x30,
|
||||||
|
0x76, 0x90, 0x45, 0x4B, 0x9D, 0x9C, 0xCF, 0x92, 0x03, 0x60, 0x8C, 0x4B, 0x6C, 0xB3, 0x7A,
|
||||||
|
0x3A, 0x05, 0x39, 0xA0, 0x66, 0xA9, 0x35, 0xCF, 0xB9, 0xFA, 0xAD, 0x9C, 0xAB, 0xEB, 0xE4,
|
||||||
|
0x6A, 0x8C, 0xE9, 0x3B, 0xCC, 0x72, 0x12, 0x62, 0x63, 0xBD, 0x80, 0xC4, 0xEE, 0x37, 0x2B,
|
||||||
|
0x32, 0x03, 0xA3, 0x09, 0xF7, 0xA0, 0x61, 0x57, 0xAD, 0x0D, 0xCF, 0x15, 0x98, 0x9E, 0x4E,
|
||||||
|
0x49, 0xF8, 0xB5, 0xA3, 0x5C, 0x27, 0xEE, 0x45, 0x04, 0xEA, 0xE4, 0x4B, 0xBC, 0x8F, 0x87,
|
||||||
|
0xED, 0x19, 0x1E, 0x46, 0x75, 0x63, 0xC4, 0x5B, 0xD5, 0xBC, 0x09, 0x2F, 0x02, 0x73, 0x19,
|
||||||
|
0x3C, 0x58, 0x55, 0x49, 0x66, 0x4C, 0x11, 0xEC, 0x0F, 0x09, 0xFA, 0xA5, 0x56, 0x0A, 0x5A,
|
||||||
|
0x63, 0x56, 0xAD, 0xA0, 0x0D, 0x86, 0x08, 0xC1, 0xE6, 0xB6, 0x13, 0x22, 0x49, 0x2F, 0x7C,
|
||||||
|
0xDB, 0x4C, 0x56, 0x97, 0x0E, 0xC2, 0xD9, 0x2E, 0x87, 0xBC, 0x0E, 0x67, 0xC0, 0x1B, 0x58,
|
||||||
|
0xBC, 0x64, 0x2B, 0xC2, 0x6E, 0xE2, 0x93, 0x2E, 0xB5, 0x6B, 0x70, 0xA4, 0x42, 0x9F, 0x64,
|
||||||
|
0xC1};
|
||||||
// p
|
// p
|
||||||
const CryptoPP::byte* Prime1;
|
static constexpr CryptoPP::byte Prime1[] = {
|
||||||
|
0xE5, 0x62, 0xE1, 0x7F, 0x9F, 0x86, 0x08, 0xE2, 0x61, 0xD3, 0xD0, 0x42, 0xE2, 0xC4, 0xB6,
|
||||||
|
0xA8, 0x51, 0x09, 0x19, 0x14, 0xA4, 0x3A, 0x11, 0x4C, 0x33, 0xA5, 0x9C, 0x01, 0x5E, 0x34,
|
||||||
|
0xB6, 0x3F, 0x02, 0x1A, 0xCA, 0x47, 0xF1, 0x4F, 0x3B, 0x35, 0x2A, 0x07, 0x20, 0xEC, 0xD8,
|
||||||
|
0xC1, 0x15, 0xD9, 0xCA, 0x03, 0x4F, 0xB8, 0xE8, 0x09, 0x73, 0x3F, 0x85, 0xB7, 0x41, 0xD5,
|
||||||
|
0x51, 0x3E, 0x7B, 0xE3, 0x53, 0x2B, 0x48, 0x8B, 0x8E, 0xCB, 0xBA, 0xF7, 0xE0, 0x60, 0xF5,
|
||||||
|
0x35, 0x0E, 0x6F, 0xB0, 0xD9, 0x2A, 0x99, 0xD0, 0xFF, 0x60, 0x14, 0xED, 0x40, 0xEA, 0xF8,
|
||||||
|
0xD7, 0x0B, 0xC3, 0x8D, 0x8C, 0xE8, 0x81, 0xB3, 0x75, 0x93, 0x15, 0xB3, 0x7D, 0xF6, 0x39,
|
||||||
|
0x60, 0x1A, 0x00, 0xE7, 0xC3, 0x27, 0xAD, 0xA4, 0x33, 0xD5, 0x3E, 0xA4, 0x35, 0x48, 0x6F,
|
||||||
|
0x22, 0xEF, 0x5D, 0xDD, 0x7D, 0x7B, 0x61, 0x05};
|
||||||
// q
|
// q
|
||||||
const CryptoPP::byte* Prime2;
|
static constexpr CryptoPP::byte Prime2[] = {
|
||||||
const CryptoPP::byte* PrivateExponent;
|
0xD9, 0x6C, 0xC2, 0x0C, 0xF7, 0xAE, 0xD1, 0xF3, 0x3B, 0x3B, 0x49, 0x1E, 0x9F, 0x12, 0x9C,
|
||||||
|
0xA1, 0x78, 0x1F, 0x35, 0x1D, 0x98, 0x26, 0x13, 0x71, 0xF9, 0x09, 0xFD, 0xF0, 0xAD, 0x38,
|
||||||
// Constructor
|
0x55, 0xB7, 0xEE, 0x61, 0x04, 0x72, 0x51, 0x87, 0x2E, 0x05, 0x84, 0xB1, 0x1D, 0x0C, 0x0D,
|
||||||
DebugRifKeyset() {
|
0xDB, 0xD4, 0x25, 0x3E, 0x26, 0xED, 0xEA, 0xB8, 0xF7, 0x49, 0xFE, 0xA2, 0x94, 0xE6, 0xF2,
|
||||||
// Initialize PrivateExponent
|
0x08, 0x92, 0xA7, 0x85, 0xF5, 0x30, 0xB9, 0x84, 0x22, 0xBF, 0xCA, 0xF0, 0x5F, 0xCB, 0x31,
|
||||||
PrivateExponent = new CryptoPP::byte[0x100]{
|
0x20, 0x34, 0x49, 0x16, 0x76, 0x34, 0xCC, 0x7A, 0xCB, 0x96, 0xFE, 0x78, 0x7A, 0x41, 0xFE,
|
||||||
0x01, 0x61, 0xAD, 0xD8, 0x9C, 0x06, 0x89, 0xD0, 0x60, 0xC8, 0x41, 0xF0, 0xB3, 0x83,
|
0x9A, 0xA2, 0x23, 0xF7, 0x68, 0x80, 0xD6, 0xCE, 0x4A, 0x78, 0xA5, 0xB7, 0x05, 0x77, 0x81,
|
||||||
0x01, 0x5D, 0xE3, 0xA2, 0x6B, 0xA2, 0xBA, 0x9A, 0x0A, 0x58, 0xCD, 0x1A, 0xA0, 0x97,
|
0x1F, 0xDE, 0x5E, 0xA8, 0x6E, 0x3E, 0x87, 0xEC, 0x44, 0xD2, 0x69, 0xC6, 0x54, 0x91, 0x6B,
|
||||||
0x64, 0xEC, 0xD0, 0x31, 0x1F, 0xCA, 0x36, 0x0E, 0x69, 0xDD, 0x40, 0xF7, 0x4E, 0xC0,
|
0x5E, 0x13, 0x8A, 0x03, 0x87, 0x05, 0x31, 0x8D};
|
||||||
0xC6, 0xA3, 0x73, 0xF0, 0x69, 0x84, 0xB2, 0xF4, 0x4B, 0x29, 0x14, 0x2A, 0x6D, 0xB8,
|
static constexpr CryptoPP::byte PrivateExponent[] = {
|
||||||
0x23, 0xD8, 0x1B, 0x61, 0xD4, 0x9E, 0x87, 0xB3, 0xBB, 0xA9, 0xC4, 0x85, 0x4A, 0xF8,
|
0x01, 0x61, 0xAD, 0xD8, 0x9C, 0x06, 0x89, 0xD0, 0x60, 0xC8, 0x41, 0xF0, 0xB3, 0x83, 0x01,
|
||||||
0x03, 0x4A, 0xBF, 0xFE, 0xF9, 0xFE, 0x8B, 0xDD, 0x54, 0x83, 0xBA, 0xE0, 0x2F, 0x3F,
|
0x5D, 0xE3, 0xA2, 0x6B, 0xA2, 0xBA, 0x9A, 0x0A, 0x58, 0xCD, 0x1A, 0xA0, 0x97, 0x64, 0xEC,
|
||||||
0xB1, 0xEF, 0xA5, 0x05, 0x5D, 0x28, 0x8B, 0xAB, 0xB5, 0xD0, 0x23, 0x2F, 0x8A, 0xCF,
|
0xD0, 0x31, 0x1F, 0xCA, 0x36, 0x0E, 0x69, 0xDD, 0x40, 0xF7, 0x4E, 0xC0, 0xC6, 0xA3, 0x73,
|
||||||
0x48, 0x7C, 0xAA, 0xBB, 0xC8, 0x5B, 0x36, 0x27, 0xC5, 0x16, 0xA4, 0xB6, 0x61, 0xAC,
|
0xF0, 0x69, 0x84, 0xB2, 0xF4, 0x4B, 0x29, 0x14, 0x2A, 0x6D, 0xB8, 0x23, 0xD8, 0x1B, 0x61,
|
||||||
0x0C, 0x28, 0x47, 0x79, 0x3F, 0x38, 0xAE, 0x5E, 0x25, 0xC6, 0xAF, 0x35, 0xAE, 0xBC,
|
0xD4, 0x9E, 0x87, 0xB3, 0xBB, 0xA9, 0xC4, 0x85, 0x4A, 0xF8, 0x03, 0x4A, 0xBF, 0xFE, 0xF9,
|
||||||
0xB0, 0xF3, 0xBC, 0xBD, 0xFD, 0xA4, 0x87, 0x0D, 0x14, 0x3D, 0x90, 0xE4, 0xDE, 0x5D,
|
0xFE, 0x8B, 0xDD, 0x54, 0x83, 0xBA, 0xE0, 0x2F, 0x3F, 0xB1, 0xEF, 0xA5, 0x05, 0x5D, 0x28,
|
||||||
0x1D, 0x46, 0x81, 0xF1, 0x28, 0x6D, 0x2F, 0x2C, 0x5E, 0x97, 0x2D, 0x89, 0x2A, 0x51,
|
0x8B, 0xAB, 0xB5, 0xD0, 0x23, 0x2F, 0x8A, 0xCF, 0x48, 0x7C, 0xAA, 0xBB, 0xC8, 0x5B, 0x36,
|
||||||
0x72, 0x3C, 0x20, 0x02, 0x59, 0xB1, 0x98, 0x93, 0x05, 0x1E, 0x3F, 0xA1, 0x8A, 0x69,
|
0x27, 0xC5, 0x16, 0xA4, 0xB6, 0x61, 0xAC, 0x0C, 0x28, 0x47, 0x79, 0x3F, 0x38, 0xAE, 0x5E,
|
||||||
0x30, 0x0E, 0x70, 0x84, 0x8B, 0xAE, 0x97, 0xA1, 0x08, 0x95, 0x63, 0x4C, 0xC7, 0xE8,
|
0x25, 0xC6, 0xAF, 0x35, 0xAE, 0xBC, 0xB0, 0xF3, 0xBC, 0xBD, 0xFD, 0xA4, 0x87, 0x0D, 0x14,
|
||||||
0x5D, 0x59, 0xCA, 0x78, 0x2A, 0x23, 0x87, 0xAC, 0x6F, 0x04, 0x33, 0xB1, 0x61, 0xB9,
|
0x3D, 0x90, 0xE4, 0xDE, 0x5D, 0x1D, 0x46, 0x81, 0xF1, 0x28, 0x6D, 0x2F, 0x2C, 0x5E, 0x97,
|
||||||
0xF0, 0x95, 0xDA, 0x33, 0xCC, 0xE0, 0x4C, 0x82, 0x68, 0x82, 0x14, 0x51, 0xBE, 0x49,
|
0x2D, 0x89, 0x2A, 0x51, 0x72, 0x3C, 0x20, 0x02, 0x59, 0xB1, 0x98, 0x93, 0x05, 0x1E, 0x3F,
|
||||||
0x1C, 0x58, 0xA2, 0x8B, 0x05, 0x4E, 0x98, 0x37, 0xEB, 0x94, 0x0B, 0x01, 0x22, 0xDC,
|
0xA1, 0x8A, 0x69, 0x30, 0x0E, 0x70, 0x84, 0x8B, 0xAE, 0x97, 0xA1, 0x08, 0x95, 0x63, 0x4C,
|
||||||
0xB3, 0x19, 0xCA, 0x77, 0xA6, 0x6E, 0x97, 0xFF, 0x8A, 0x53, 0x5A, 0xC5, 0x24, 0xE4,
|
0xC7, 0xE8, 0x5D, 0x59, 0xCA, 0x78, 0x2A, 0x23, 0x87, 0xAC, 0x6F, 0x04, 0x33, 0xB1, 0x61,
|
||||||
0xAF, 0x6E, 0xA8, 0x2B, 0x53, 0xA4, 0xBE, 0x96, 0xA5, 0x7B, 0xCE, 0x22, 0x56, 0xA3,
|
0xB9, 0xF0, 0x95, 0xDA, 0x33, 0xCC, 0xE0, 0x4C, 0x82, 0x68, 0x82, 0x14, 0x51, 0xBE, 0x49,
|
||||||
0xF1, 0xCF, 0x14, 0xA5};
|
0x1C, 0x58, 0xA2, 0x8B, 0x05, 0x4E, 0x98, 0x37, 0xEB, 0x94, 0x0B, 0x01, 0x22, 0xDC, 0xB3,
|
||||||
|
0x19, 0xCA, 0x77, 0xA6, 0x6E, 0x97, 0xFF, 0x8A, 0x53, 0x5A, 0xC5, 0x24, 0xE4, 0xAF, 0x6E,
|
||||||
// Initialize Exponent1
|
0xA8, 0x2B, 0x53, 0xA4, 0xBE, 0x96, 0xA5, 0x7B, 0xCE, 0x22, 0x56, 0xA3, 0xF1, 0xCF, 0x14,
|
||||||
Exponent1 = new CryptoPP::byte[0x80]{
|
0xA5};
|
||||||
0xCD, 0x9A, 0x61, 0xB0, 0xB8, 0xD5, 0xB4, 0xE4, 0xE4, 0xF6, 0xAB, 0xF7, 0x27,
|
|
||||||
0xB7, 0x56, 0x59, 0x6B, 0xB9, 0x11, 0xE7, 0xF4, 0x83, 0xAF, 0xB9, 0x73, 0x99,
|
|
||||||
0x7F, 0x49, 0xA2, 0x9C, 0xF0, 0xB5, 0x6D, 0x37, 0x82, 0x14, 0x15, 0xF1, 0x04,
|
|
||||||
0x8A, 0xD4, 0x8E, 0xEB, 0x2E, 0x1F, 0xE2, 0x81, 0xA9, 0x62, 0x6E, 0xB1, 0x68,
|
|
||||||
0x75, 0x62, 0xF3, 0x0F, 0xFE, 0xD4, 0x91, 0x87, 0x98, 0x78, 0xBF, 0x26, 0xB5,
|
|
||||||
0x07, 0x58, 0xD0, 0xEE, 0x3F, 0x21, 0xE8, 0xC8, 0x0F, 0x5F, 0xFA, 0x1C, 0x64,
|
|
||||||
0x74, 0x49, 0x52, 0xEB, 0xE7, 0xEE, 0xDE, 0xBA, 0x23, 0x26, 0x4A, 0xF6, 0x9C,
|
|
||||||
0x1A, 0x09, 0x3F, 0xB9, 0x0B, 0x36, 0x26, 0x1A, 0xBE, 0xA9, 0x76, 0xE6, 0xF2,
|
|
||||||
0x69, 0xDE, 0xFF, 0xAF, 0xCC, 0x0C, 0x9A, 0x66, 0x03, 0x86, 0x0A, 0x1F, 0x49,
|
|
||||||
0xA4, 0x10, 0xB6, 0xBC, 0xC3, 0x7C, 0x88, 0xE8, 0xCE, 0x4B, 0xD9
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
Exponent2 = new CryptoPP::byte[0x80]{
|
|
||||||
0xB3, 0x73, 0xA3, 0x59, 0xE6, 0x97, 0xC0, 0xAB, 0x3B, 0x68, 0xFC, 0x39, 0xAC,
|
|
||||||
0xDB, 0x44, 0xB1, 0xB4, 0x9E, 0x35, 0x4D, 0xBE, 0xC5, 0x36, 0x69, 0x6C, 0x3D,
|
|
||||||
0xC5, 0xFC, 0xFE, 0x4B, 0x2F, 0xDC, 0x86, 0x80, 0x46, 0x96, 0x40, 0x1A, 0x0D,
|
|
||||||
0x6E, 0xFA, 0x8C, 0xE0, 0x47, 0x91, 0xAC, 0xAD, 0x95, 0x2B, 0x8E, 0x1F, 0xF2,
|
|
||||||
0x0A, 0x45, 0xF8, 0x29, 0x95, 0x70, 0xC6, 0x88, 0x5F, 0x71, 0x03, 0x99, 0x79,
|
|
||||||
0xBC, 0x84, 0x71, 0xBD, 0xE8, 0x84, 0x8C, 0x0E, 0xD4, 0x7B, 0x30, 0x74, 0x57,
|
|
||||||
0x1A, 0x95, 0xE7, 0x90, 0x19, 0x8D, 0xAD, 0x8B, 0x4C, 0x4E, 0xC3, 0xE7, 0x6B,
|
|
||||||
0x23, 0x86, 0x01, 0xEE, 0x9B, 0xE0, 0x2F, 0x15, 0xA2, 0x2C, 0x4C, 0x39, 0xD3,
|
|
||||||
0xDF, 0x9C, 0x39, 0x01, 0xF1, 0x8C, 0x44, 0x4A, 0x15, 0x44, 0xDC, 0x51, 0xF7,
|
|
||||||
0x22, 0xD7, 0x7F, 0x41, 0x7F, 0x68, 0xFA, 0xEE, 0x56, 0xE8, 0x05};
|
|
||||||
|
|
||||||
PublicExponent = new CryptoPP::byte[4]{0x00, 0x01, 0x00, 0x01};
|
|
||||||
|
|
||||||
Coefficient = new CryptoPP::byte[0x80]{
|
|
||||||
0xC0, 0x32, 0x43, 0xD3, 0x8C, 0x3D, 0xB4, 0xD2, 0x48, 0x8C, 0x42, 0x41, 0x24,
|
|
||||||
0x94, 0x6C, 0x80, 0xC9, 0xC1, 0x79, 0x36, 0x7F, 0xAC, 0xC3, 0xFF, 0x6A, 0x25,
|
|
||||||
0xEB, 0x2C, 0xFB, 0xD4, 0x2B, 0xA0, 0xEB, 0xFE, 0x25, 0xE9, 0xC6, 0x77, 0xCE,
|
|
||||||
0xFE, 0x2D, 0x23, 0xFE, 0xD0, 0xF4, 0x0F, 0xD9, 0x7E, 0xD5, 0xA5, 0x7D, 0x1F,
|
|
||||||
0xC0, 0xE8, 0xE8, 0xEC, 0x80, 0x5B, 0xC7, 0xFD, 0xE2, 0xBD, 0x94, 0xA6, 0x2B,
|
|
||||||
0xDD, 0x6A, 0x60, 0x45, 0x54, 0xAB, 0xCA, 0x42, 0x9C, 0x6A, 0x6C, 0xBF, 0x3C,
|
|
||||||
0x84, 0xF9, 0xA5, 0x0E, 0x63, 0x0C, 0x51, 0x58, 0x62, 0x6D, 0x5A, 0xB7, 0x3C,
|
|
||||||
0x3F, 0x49, 0x1A, 0xD0, 0x93, 0xB8, 0x4F, 0x1A, 0x6C, 0x5F, 0xC5, 0xE5, 0xA9,
|
|
||||||
0x75, 0xD4, 0x86, 0x9E, 0xDF, 0x87, 0x0F, 0x27, 0xB0, 0x26, 0x78, 0x4E, 0xFB,
|
|
||||||
0xC1, 0x8A, 0x4A, 0x24, 0x3F, 0x7F, 0x8F, 0x9A, 0x12, 0x51, 0xCB};
|
|
||||||
|
|
||||||
Modulus = new CryptoPP::byte[0x100]{
|
|
||||||
0xC2, 0xD2, 0x44, 0xBC, 0xDD, 0x84, 0x3F, 0xD9, 0xC5, 0x22, 0xAF, 0xF7, 0xFC, 0x88,
|
|
||||||
0x8A, 0x33, 0x80, 0xED, 0x8E, 0xE2, 0xCC, 0x81, 0xF7, 0xEC, 0xF8, 0x1C, 0x79, 0xBF,
|
|
||||||
0x02, 0xBB, 0x12, 0x8E, 0x61, 0x68, 0x29, 0x1B, 0x15, 0xB6, 0x5E, 0xC6, 0xF8, 0xBF,
|
|
||||||
0x5A, 0xE0, 0x3B, 0x6A, 0x6C, 0xD9, 0xD6, 0xF5, 0x75, 0xAB, 0xA0, 0x6F, 0x34, 0x81,
|
|
||||||
0x34, 0x9A, 0x5B, 0xAD, 0xED, 0x31, 0xE3, 0xC6, 0xEA, 0x1A, 0xD1, 0x13, 0x22, 0xBB,
|
|
||||||
0xB3, 0xDA, 0xB3, 0xB2, 0x53, 0xBD, 0x45, 0x79, 0x87, 0xAD, 0x0A, 0x01, 0x72, 0x18,
|
|
||||||
0x10, 0x29, 0x49, 0xF4, 0x41, 0x7F, 0xD6, 0x47, 0x0C, 0x72, 0x92, 0x9E, 0xE9, 0xBB,
|
|
||||||
0x95, 0xA9, 0x5D, 0x79, 0xEB, 0xE4, 0x30, 0x76, 0x90, 0x45, 0x4B, 0x9D, 0x9C, 0xCF,
|
|
||||||
0x92, 0x03, 0x60, 0x8C, 0x4B, 0x6C, 0xB3, 0x7A, 0x3A, 0x05, 0x39, 0xA0, 0x66, 0xA9,
|
|
||||||
0x35, 0xCF, 0xB9, 0xFA, 0xAD, 0x9C, 0xAB, 0xEB, 0xE4, 0x6A, 0x8C, 0xE9, 0x3B, 0xCC,
|
|
||||||
0x72, 0x12, 0x62, 0x63, 0xBD, 0x80, 0xC4, 0xEE, 0x37, 0x2B, 0x32, 0x03, 0xA3, 0x09,
|
|
||||||
0xF7, 0xA0, 0x61, 0x57, 0xAD, 0x0D, 0xCF, 0x15, 0x98, 0x9E, 0x4E, 0x49, 0xF8, 0xB5,
|
|
||||||
0xA3, 0x5C, 0x27, 0xEE, 0x45, 0x04, 0xEA, 0xE4, 0x4B, 0xBC, 0x8F, 0x87, 0xED, 0x19,
|
|
||||||
0x1E, 0x46, 0x75, 0x63, 0xC4, 0x5B, 0xD5, 0xBC, 0x09, 0x2F, 0x02, 0x73, 0x19, 0x3C,
|
|
||||||
0x58, 0x55, 0x49, 0x66, 0x4C, 0x11, 0xEC, 0x0F, 0x09, 0xFA, 0xA5, 0x56, 0x0A, 0x5A,
|
|
||||||
0x63, 0x56, 0xAD, 0xA0, 0x0D, 0x86, 0x08, 0xC1, 0xE6, 0xB6, 0x13, 0x22, 0x49, 0x2F,
|
|
||||||
0x7C, 0xDB, 0x4C, 0x56, 0x97, 0x0E, 0xC2, 0xD9, 0x2E, 0x87, 0xBC, 0x0E, 0x67, 0xC0,
|
|
||||||
0x1B, 0x58, 0xBC, 0x64, 0x2B, 0xC2, 0x6E, 0xE2, 0x93, 0x2E, 0xB5, 0x6B, 0x70, 0xA4,
|
|
||||||
0x42, 0x9F, 0x64, 0xC1};
|
|
||||||
|
|
||||||
Prime1 = new CryptoPP::byte[0x80]{
|
|
||||||
0xE5, 0x62, 0xE1, 0x7F, 0x9F, 0x86, 0x08, 0xE2, 0x61, 0xD3, 0xD0, 0x42, 0xE2,
|
|
||||||
0xC4, 0xB6, 0xA8, 0x51, 0x09, 0x19, 0x14, 0xA4, 0x3A, 0x11, 0x4C, 0x33, 0xA5,
|
|
||||||
0x9C, 0x01, 0x5E, 0x34, 0xB6, 0x3F, 0x02, 0x1A, 0xCA, 0x47, 0xF1, 0x4F, 0x3B,
|
|
||||||
0x35, 0x2A, 0x07, 0x20, 0xEC, 0xD8, 0xC1, 0x15, 0xD9, 0xCA, 0x03, 0x4F, 0xB8,
|
|
||||||
0xE8, 0x09, 0x73, 0x3F, 0x85, 0xB7, 0x41, 0xD5, 0x51, 0x3E, 0x7B, 0xE3, 0x53,
|
|
||||||
0x2B, 0x48, 0x8B, 0x8E, 0xCB, 0xBA, 0xF7, 0xE0, 0x60, 0xF5, 0x35, 0x0E, 0x6F,
|
|
||||||
0xB0, 0xD9, 0x2A, 0x99, 0xD0, 0xFF, 0x60, 0x14, 0xED, 0x40, 0xEA, 0xF8, 0xD7,
|
|
||||||
0x0B, 0xC3, 0x8D, 0x8C, 0xE8, 0x81, 0xB3, 0x75, 0x93, 0x15, 0xB3, 0x7D, 0xF6,
|
|
||||||
0x39, 0x60, 0x1A, 0x00, 0xE7, 0xC3, 0x27, 0xAD, 0xA4, 0x33, 0xD5, 0x3E, 0xA4,
|
|
||||||
0x35, 0x48, 0x6F, 0x22, 0xEF, 0x5D, 0xDD, 0x7D, 0x7B, 0x61, 0x05};
|
|
||||||
|
|
||||||
Prime2 = new CryptoPP::byte[0x80]{
|
|
||||||
0xD9, 0x6C, 0xC2, 0x0C, 0xF7, 0xAE, 0xD1, 0xF3, 0x3B, 0x3B, 0x49, 0x1E, 0x9F,
|
|
||||||
0x12, 0x9C, 0xA1, 0x78, 0x1F, 0x35, 0x1D, 0x98, 0x26, 0x13, 0x71, 0xF9, 0x09,
|
|
||||||
0xFD, 0xF0, 0xAD, 0x38, 0x55, 0xB7, 0xEE, 0x61, 0x04, 0x72, 0x51, 0x87, 0x2E,
|
|
||||||
0x05, 0x84, 0xB1, 0x1D, 0x0C, 0x0D, 0xDB, 0xD4, 0x25, 0x3E, 0x26, 0xED, 0xEA,
|
|
||||||
0xB8, 0xF7, 0x49, 0xFE, 0xA2, 0x94, 0xE6, 0xF2, 0x08, 0x92, 0xA7, 0x85, 0xF5,
|
|
||||||
0x30, 0xB9, 0x84, 0x22, 0xBF, 0xCA, 0xF0, 0x5F, 0xCB, 0x31, 0x20, 0x34, 0x49,
|
|
||||||
0x16, 0x76, 0x34, 0xCC, 0x7A, 0xCB, 0x96, 0xFE, 0x78, 0x7A, 0x41, 0xFE, 0x9A,
|
|
||||||
0xA2, 0x23, 0xF7, 0x68, 0x80, 0xD6, 0xCE, 0x4A, 0x78, 0xA5, 0xB7, 0x05, 0x77,
|
|
||||||
0x81, 0x1F, 0xDE, 0x5E, 0xA8, 0x6E, 0x3E, 0x87, 0xEC, 0x44, 0xD2, 0x69, 0xC6,
|
|
||||||
0x54, 0x91, 0x6B, 0x5E, 0x13, 0x8A, 0x03, 0x87, 0x05, 0x31, 0x8D};
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class PkgDerivedKey3Keyset {
|
class PkgDerivedKey3Keyset {
|
||||||
public:
|
public:
|
||||||
// PkgDerivedKey3Keyset();
|
|
||||||
//~PkgDerivedKey3Keyset();
|
|
||||||
|
|
||||||
// Constructor
|
|
||||||
// std::uint8_t* PrivateExponent;
|
// std::uint8_t* PrivateExponent;
|
||||||
const CryptoPP::byte* Exponent1;
|
static constexpr CryptoPP::byte Exponent1[] = {
|
||||||
|
0x52, 0xCC, 0x2D, 0xA0, 0x9C, 0x9E, 0x75, 0xE7, 0x28, 0xEE, 0x3D, 0xDE, 0xE3, 0x45, 0xD1,
|
||||||
|
0x4F, 0x94, 0x1C, 0xCC, 0xC8, 0x87, 0x29, 0x45, 0x3B, 0x8D, 0x6E, 0xAB, 0x6E, 0x2A, 0xA7,
|
||||||
|
0xC7, 0x15, 0x43, 0xA3, 0x04, 0x8F, 0x90, 0x5F, 0xEB, 0xF3, 0x38, 0x4A, 0x77, 0xFA, 0x36,
|
||||||
|
0xB7, 0x15, 0x76, 0xB6, 0x01, 0x1A, 0x8E, 0x25, 0x87, 0x82, 0xF1, 0x55, 0xD8, 0xC6, 0x43,
|
||||||
|
0x2A, 0xC0, 0xE5, 0x98, 0xC9, 0x32, 0xD1, 0x94, 0x6F, 0xD9, 0x01, 0xBA, 0x06, 0x81, 0xE0,
|
||||||
|
0x6D, 0x88, 0xF2, 0x24, 0x2A, 0x25, 0x01, 0x64, 0x5C, 0xBF, 0xF2, 0xD9, 0x99, 0x67, 0x3E,
|
||||||
|
0xF6, 0x72, 0xEE, 0xE4, 0xE2, 0x33, 0x5C, 0xF8, 0x00, 0x40, 0xE3, 0x2A, 0x9A, 0xF4, 0x3D,
|
||||||
|
0x22, 0x86, 0x44, 0x3C, 0xFB, 0x0A, 0xA5, 0x7C, 0x3F, 0xCC, 0xF5, 0xF1, 0x16, 0xC4, 0xAC,
|
||||||
|
0x88, 0xB4, 0xDE, 0x62, 0x94, 0x92, 0x6A, 0x13};
|
||||||
// exponent2 = d mod (q - 1)
|
// exponent2 = d mod (q - 1)
|
||||||
const CryptoPP::byte* Exponent2;
|
static constexpr CryptoPP::byte Exponent2[] = {
|
||||||
|
0x7C, 0x9D, 0xAD, 0x39, 0xE0, 0xD5, 0x60, 0x14, 0x94, 0x48, 0x19, 0x7F, 0x88, 0x95, 0xD5,
|
||||||
|
0x8B, 0x80, 0xAD, 0x85, 0x8A, 0x4B, 0x77, 0x37, 0x85, 0xD0, 0x77, 0xBB, 0xBF, 0x89, 0x71,
|
||||||
|
0x4A, 0x72, 0xCB, 0x72, 0x68, 0x38, 0xEC, 0x02, 0xC6, 0x7D, 0xC6, 0x44, 0x06, 0x33, 0x51,
|
||||||
|
0x1C, 0xC0, 0xFF, 0x95, 0x8F, 0x0D, 0x75, 0xDC, 0x25, 0xBB, 0x0B, 0x73, 0x91, 0xA9, 0x6D,
|
||||||
|
0x42, 0xD8, 0x03, 0xB7, 0x68, 0xD4, 0x1E, 0x75, 0x62, 0xA3, 0x70, 0x35, 0x79, 0x78, 0x00,
|
||||||
|
0xC8, 0xF5, 0xEF, 0x15, 0xB9, 0xFC, 0x4E, 0x47, 0x5A, 0xC8, 0x70, 0x70, 0x5B, 0x52, 0x98,
|
||||||
|
0xC0, 0xC2, 0x58, 0x4A, 0x70, 0x96, 0xCC, 0xB8, 0x10, 0xE1, 0x2F, 0x78, 0x8B, 0x2B, 0xA1,
|
||||||
|
0x7F, 0xF9, 0xAC, 0xDE, 0xF0, 0xBB, 0x2B, 0xE2, 0x66, 0xE3, 0x22, 0x92, 0x31, 0x21, 0x57,
|
||||||
|
0x92, 0xC4, 0xB8, 0xF2, 0x3E, 0x76, 0x20, 0x37};
|
||||||
// e
|
// e
|
||||||
const CryptoPP::byte* PublicExponent;
|
static constexpr CryptoPP::byte PublicExponent[] = {0, 1, 0, 1};
|
||||||
// (InverseQ)(q) = 1 mod p
|
// (InverseQ)(q) = 1 mod p
|
||||||
const CryptoPP::byte* Coefficient;
|
static constexpr CryptoPP::byte Coefficient[] = {
|
||||||
|
0x45, 0x97, 0x55, 0xD4, 0x22, 0x08, 0x5E, 0xF3, 0x5C, 0xB4, 0x05, 0x7A, 0xFD, 0xAA, 0x42,
|
||||||
|
0x42, 0xAD, 0x9A, 0x8C, 0xA0, 0x6C, 0xBB, 0x1D, 0x68, 0x54, 0x54, 0x6E, 0x3E, 0x32, 0xE3,
|
||||||
|
0x53, 0x73, 0x76, 0xF1, 0x3E, 0x01, 0xEA, 0xD3, 0xCF, 0xEB, 0xEB, 0x23, 0x3E, 0xC0, 0xBE,
|
||||||
|
0xCE, 0xEC, 0x2C, 0x89, 0x5F, 0xA8, 0x27, 0x3A, 0x4C, 0xB7, 0xE6, 0x74, 0xBC, 0x45, 0x4C,
|
||||||
|
0x26, 0xC8, 0x25, 0xFF, 0x34, 0x63, 0x25, 0x37, 0xE1, 0x48, 0x10, 0xC1, 0x93, 0xA6, 0xAF,
|
||||||
|
0xEB, 0xBA, 0xE3, 0xA2, 0xF1, 0x3D, 0xEF, 0x63, 0xD8, 0xF4, 0xFD, 0xD3, 0xEE, 0xE2, 0x5D,
|
||||||
|
0xE9, 0x33, 0xCC, 0xAD, 0xBA, 0x75, 0x5C, 0x85, 0xAF, 0xCE, 0xA9, 0x3D, 0xD1, 0xA2, 0x17,
|
||||||
|
0xF3, 0xF6, 0x98, 0xB3, 0x50, 0x8E, 0x5E, 0xF6, 0xEB, 0x02, 0x8E, 0xA1, 0x62, 0xA7, 0xD6,
|
||||||
|
0x2C, 0xEC, 0x91, 0xFF, 0x15, 0x40, 0xD2, 0xE3};
|
||||||
// n = p * q
|
// n = p * q
|
||||||
const CryptoPP::byte* Modulus;
|
static constexpr CryptoPP::byte Modulus[] = {
|
||||||
|
0xd2, 0x12, 0xfc, 0x33, 0x5f, 0x6d, 0xdb, 0x83, 0x16, 0x09, 0x62, 0x8b, 0x03, 0x56, 0x27,
|
||||||
|
0x37, 0x82, 0xd4, 0x77, 0x85, 0x35, 0x29, 0x39, 0x2d, 0x52, 0x6b, 0x8c, 0x4c, 0x8c, 0xfb,
|
||||||
|
0x06, 0xc1, 0x84, 0x5b, 0xe7, 0xd4, 0xf7, 0xbc, 0xd2, 0x4e, 0x62, 0x45, 0xcd, 0x2a, 0xbb,
|
||||||
|
0xd7, 0x77, 0x76, 0x45, 0x36, 0x55, 0x27, 0x3f, 0xb3, 0xf5, 0xf9, 0x8e, 0xda, 0x4b, 0xef,
|
||||||
|
0xaa, 0x59, 0xae, 0xb3, 0x9b, 0xea, 0x54, 0x98, 0xd2, 0x06, 0x32, 0x6a, 0x58, 0x31, 0x2a,
|
||||||
|
0xe0, 0xd4, 0x4f, 0x90, 0xb5, 0x0a, 0x7d, 0xec, 0xf4, 0x3a, 0x9c, 0x52, 0x67, 0x2d, 0x99,
|
||||||
|
0x31, 0x8e, 0x0c, 0x43, 0xe6, 0x82, 0xfe, 0x07, 0x46, 0xe1, 0x2e, 0x50, 0xd4, 0x1f, 0x2d,
|
||||||
|
0x2f, 0x7e, 0xd9, 0x08, 0xba, 0x06, 0xb3, 0xbf, 0x2e, 0x20, 0x3f, 0x4e, 0x3f, 0xfe, 0x44,
|
||||||
|
0xff, 0xaa, 0x50, 0x43, 0x57, 0x91, 0x69, 0x94, 0x49, 0x15, 0x82, 0x82, 0xe4, 0x0f, 0x4c,
|
||||||
|
0x8d, 0x9d, 0x2c, 0xc9, 0x5b, 0x1d, 0x64, 0xbf, 0x88, 0x8b, 0xd4, 0xc5, 0x94, 0xe7, 0x65,
|
||||||
|
0x47, 0x84, 0x1e, 0xe5, 0x79, 0x10, 0xfb, 0x98, 0x93, 0x47, 0xb9, 0x7d, 0x85, 0x12, 0xa6,
|
||||||
|
0x40, 0x98, 0x2c, 0xf7, 0x92, 0xbc, 0x95, 0x19, 0x32, 0xed, 0xe8, 0x90, 0x56, 0x0d, 0x65,
|
||||||
|
0xc1, 0xaa, 0x78, 0xc6, 0x2e, 0x54, 0xfd, 0x5f, 0x54, 0xa1, 0xf6, 0x7e, 0xe5, 0xe0, 0x5f,
|
||||||
|
0x61, 0xc1, 0x20, 0xb4, 0xb9, 0xb4, 0x33, 0x08, 0x70, 0xe4, 0xdf, 0x89, 0x56, 0xed, 0x01,
|
||||||
|
0x29, 0x46, 0x77, 0x5f, 0x8c, 0xb8, 0xa9, 0xf5, 0x1e, 0x2e, 0xb3, 0xb9, 0xbf, 0xe0, 0x09,
|
||||||
|
0xb7, 0x8d, 0x28, 0xd4, 0xa6, 0xc3, 0xb8, 0x1e, 0x1f, 0x07, 0xeb, 0xb4, 0x12, 0x0b, 0x95,
|
||||||
|
0xb8, 0x85, 0x30, 0xfd, 0xdc, 0x39, 0x13, 0xd0, 0x7c, 0xdc, 0x8f, 0xed, 0xf9, 0xc9, 0xa3,
|
||||||
|
0xc1};
|
||||||
// p
|
// p
|
||||||
const CryptoPP::byte* Prime1;
|
static constexpr CryptoPP::byte Prime1[] = {
|
||||||
|
0xF9, 0x67, 0xAD, 0x99, 0x12, 0x31, 0x0C, 0x56, 0xA2, 0x2E, 0x16, 0x1C, 0x46, 0xB3, 0x4D,
|
||||||
|
0x5B, 0x43, 0xBE, 0x42, 0xA2, 0xF6, 0x86, 0x96, 0x80, 0x42, 0xC3, 0xC7, 0x3F, 0xC3, 0x42,
|
||||||
|
0xF5, 0x87, 0x49, 0x33, 0x9F, 0x07, 0x5D, 0x6E, 0x2C, 0x04, 0xFD, 0xE3, 0xE1, 0xB2, 0xAE,
|
||||||
|
0x0A, 0x0C, 0xF0, 0xC7, 0xA6, 0x1C, 0xA1, 0x63, 0x50, 0xC8, 0x09, 0x9C, 0x51, 0x24, 0x52,
|
||||||
|
0x6C, 0x5E, 0x5E, 0xBD, 0x1E, 0x27, 0x06, 0xBB, 0xBC, 0x9E, 0x94, 0xE1, 0x35, 0xD4, 0x6D,
|
||||||
|
0xB3, 0xCB, 0x3C, 0x68, 0xDD, 0x68, 0xB3, 0xFE, 0x6C, 0xCB, 0x8D, 0x82, 0x20, 0x76, 0x23,
|
||||||
|
0x63, 0xB7, 0xE9, 0x68, 0x10, 0x01, 0x4E, 0xDC, 0xBA, 0x27, 0x5D, 0x01, 0xC1, 0x2D, 0x80,
|
||||||
|
0x5E, 0x2B, 0xAF, 0x82, 0x6B, 0xD8, 0x84, 0xB6, 0x10, 0x52, 0x86, 0xA7, 0x89, 0x8E, 0xAE,
|
||||||
|
0x9A, 0xE2, 0x89, 0xC6, 0xF7, 0xD5, 0x87, 0xFB};
|
||||||
// q
|
// q
|
||||||
const CryptoPP::byte* Prime2;
|
static constexpr CryptoPP::byte Prime2[] = {
|
||||||
const CryptoPP::byte* PrivateExponent;
|
0xD7, 0xA1, 0x0F, 0x9A, 0x8B, 0xF2, 0xC9, 0x11, 0x95, 0x32, 0x9A, 0x8C, 0xF0, 0xD9, 0x40,
|
||||||
|
0x47, 0xF5, 0x68, 0xA0, 0x0D, 0xBD, 0xC1, 0xFC, 0x43, 0x2F, 0x65, 0xF9, 0xC3, 0x61, 0x0F,
|
||||||
PkgDerivedKey3Keyset() {
|
0x25, 0x77, 0x54, 0xAD, 0xD7, 0x58, 0xAC, 0x84, 0x40, 0x60, 0x8D, 0x3F, 0xF3, 0x65, 0x89,
|
||||||
|
0x75, 0xB5, 0xC6, 0x2C, 0x51, 0x1A, 0x2F, 0x1F, 0x22, 0xE4, 0x43, 0x11, 0x54, 0xBE, 0xC9,
|
||||||
Prime1 = new CryptoPP::byte[0x80]{
|
0xB4, 0xC7, 0xB5, 0x1B, 0x05, 0x0B, 0xBC, 0x56, 0x9A, 0xCD, 0x4A, 0xD9, 0x73, 0x68, 0x5E,
|
||||||
0xF9, 0x67, 0xAD, 0x99, 0x12, 0x31, 0x0C, 0x56, 0xA2, 0x2E, 0x16, 0x1C, 0x46,
|
0x5C, 0xFB, 0x92, 0xB7, 0x8B, 0x0D, 0xFF, 0xF5, 0x07, 0xCA, 0xB4, 0xC8, 0x9B, 0x96, 0x3C,
|
||||||
0xB3, 0x4D, 0x5B, 0x43, 0xBE, 0x42, 0xA2, 0xF6, 0x86, 0x96, 0x80, 0x42, 0xC3,
|
0x07, 0x9E, 0x3E, 0x6B, 0x2A, 0x11, 0xF2, 0x8A, 0xB1, 0x8A, 0xD7, 0x2E, 0x1B, 0xA5, 0x53,
|
||||||
0xC7, 0x3F, 0xC3, 0x42, 0xF5, 0x87, 0x49, 0x33, 0x9F, 0x07, 0x5D, 0x6E, 0x2C,
|
0x24, 0x06, 0xED, 0x50, 0xB8, 0x90, 0x67, 0xB1, 0xE2, 0x41, 0xC6, 0x92, 0x01, 0xEE, 0x10,
|
||||||
0x04, 0xFD, 0xE3, 0xE1, 0xB2, 0xAE, 0x0A, 0x0C, 0xF0, 0xC7, 0xA6, 0x1C, 0xA1,
|
0xF0, 0x61, 0xBB, 0xFB, 0xB2, 0x7D, 0x4A, 0x73};
|
||||||
0x63, 0x50, 0xC8, 0x09, 0x9C, 0x51, 0x24, 0x52, 0x6C, 0x5E, 0x5E, 0xBD, 0x1E,
|
static constexpr CryptoPP::byte PrivateExponent[] = {
|
||||||
0x27, 0x06, 0xBB, 0xBC, 0x9E, 0x94, 0xE1, 0x35, 0xD4, 0x6D, 0xB3, 0xCB, 0x3C,
|
0x32, 0xD9, 0x03, 0x90, 0x8F, 0xBD, 0xB0, 0x8F, 0x57, 0x2B, 0x28, 0x5E, 0x0B, 0x8D, 0xB3,
|
||||||
0x68, 0xDD, 0x68, 0xB3, 0xFE, 0x6C, 0xCB, 0x8D, 0x82, 0x20, 0x76, 0x23, 0x63,
|
0xEA, 0x5C, 0xD1, 0x7E, 0xA8, 0x90, 0x88, 0x8C, 0xDD, 0x6A, 0x80, 0xBB, 0xB1, 0xDF, 0xC1,
|
||||||
0xB7, 0xE9, 0x68, 0x10, 0x01, 0x4E, 0xDC, 0xBA, 0x27, 0x5D, 0x01, 0xC1, 0x2D,
|
0xF7, 0x0D, 0xAA, 0x32, 0xF0, 0xB7, 0x7C, 0xCB, 0x88, 0x80, 0x0E, 0x8B, 0x64, 0xB0, 0xBE,
|
||||||
0x80, 0x5E, 0x2B, 0xAF, 0x82, 0x6B, 0xD8, 0x84, 0xB6, 0x10, 0x52, 0x86, 0xA7,
|
0x4C, 0xD6, 0x0E, 0x9B, 0x8C, 0x1E, 0x2A, 0x64, 0xE1, 0xF3, 0x5C, 0xD7, 0x76, 0x01, 0x41,
|
||||||
0x89, 0x8E, 0xAE, 0x9A, 0xE2, 0x89, 0xC6, 0xF7, 0xD5, 0x87, 0xFB};
|
0x5E, 0x93, 0x5C, 0x94, 0xFE, 0xDD, 0x46, 0x62, 0xC3, 0x1B, 0x5A, 0xE2, 0xA0, 0xBC, 0x2D,
|
||||||
|
0xEB, 0xC3, 0x98, 0x0A, 0xA7, 0xB7, 0x85, 0x69, 0x70, 0x68, 0x2B, 0x64, 0x4A, 0xB3, 0x1F,
|
||||||
Prime2 = new CryptoPP::byte[0x80]{
|
0xCC, 0x7D, 0xDC, 0x7C, 0x26, 0xF4, 0x77, 0xF6, 0x5C, 0xF2, 0xAE, 0x5A, 0x44, 0x2D, 0xD3,
|
||||||
0xD7, 0xA1, 0x0F, 0x9A, 0x8B, 0xF2, 0xC9, 0x11, 0x95, 0x32, 0x9A, 0x8C, 0xF0,
|
0xAB, 0x16, 0x62, 0x04, 0x19, 0xBA, 0xFB, 0x90, 0xFF, 0xE2, 0x30, 0x50, 0x89, 0x6E, 0xCB,
|
||||||
0xD9, 0x40, 0x47, 0xF5, 0x68, 0xA0, 0x0D, 0xBD, 0xC1, 0xFC, 0x43, 0x2F, 0x65,
|
0x56, 0xB2, 0xEB, 0xC0, 0x91, 0x16, 0x92, 0x5E, 0x30, 0x8E, 0xAE, 0xC7, 0x94, 0x5D, 0xFD,
|
||||||
0xF9, 0xC3, 0x61, 0x0F, 0x25, 0x77, 0x54, 0xAD, 0xD7, 0x58, 0xAC, 0x84, 0x40,
|
0x35, 0xE1, 0x20, 0xF8, 0xAD, 0x3E, 0xBC, 0x08, 0xBF, 0xC0, 0x36, 0x74, 0x9F, 0xD5, 0xBB,
|
||||||
0x60, 0x8D, 0x3F, 0xF3, 0x65, 0x89, 0x75, 0xB5, 0xC6, 0x2C, 0x51, 0x1A, 0x2F,
|
0x52, 0x08, 0xFD, 0x06, 0x66, 0xF3, 0x7A, 0xB3, 0x04, 0xF4, 0x75, 0x29, 0x5D, 0xE9, 0x5F,
|
||||||
0x1F, 0x22, 0xE4, 0x43, 0x11, 0x54, 0xBE, 0xC9, 0xB4, 0xC7, 0xB5, 0x1B, 0x05,
|
0xAA, 0x10, 0x30, 0xB2, 0x0F, 0x5A, 0x1A, 0xC1, 0x2A, 0xB3, 0xFE, 0xCB, 0x21, 0xAD, 0x80,
|
||||||
0x0B, 0xBC, 0x56, 0x9A, 0xCD, 0x4A, 0xD9, 0x73, 0x68, 0x5E, 0x5C, 0xFB, 0x92,
|
0xEC, 0x8F, 0x20, 0x09, 0x1C, 0xDB, 0xC5, 0x58, 0x94, 0xC2, 0x9C, 0xC6, 0xCE, 0x82, 0x65,
|
||||||
0xB7, 0x8B, 0x0D, 0xFF, 0xF5, 0x07, 0xCA, 0xB4, 0xC8, 0x9B, 0x96, 0x3C, 0x07,
|
0x3E, 0x57, 0x90, 0xBC, 0xA9, 0x8B, 0x06, 0xB4, 0xF0, 0x72, 0xF6, 0x77, 0xDF, 0x98, 0x64,
|
||||||
0x9E, 0x3E, 0x6B, 0x2A, 0x11, 0xF2, 0x8A, 0xB1, 0x8A, 0xD7, 0x2E, 0x1B, 0xA5,
|
0xF1, 0xEC, 0xFE, 0x37, 0x2D, 0xBC, 0xAE, 0x8C, 0x08, 0x81, 0x1F, 0xC3, 0xC9, 0x89, 0x1A,
|
||||||
0x53, 0x24, 0x06, 0xED, 0x50, 0xB8, 0x90, 0x67, 0xB1, 0xE2, 0x41, 0xC6, 0x92,
|
0xC7, 0x42, 0x82, 0x4B, 0x2E, 0xDC, 0x8E, 0x8D, 0x73, 0xCE, 0xB1, 0xCC, 0x01, 0xD9, 0x08,
|
||||||
0x01, 0xEE, 0x10, 0xF0, 0x61, 0xBB, 0xFB, 0xB2, 0x7D, 0x4A, 0x73};
|
0x70, 0x87, 0x3C, 0x44, 0x08, 0xEC, 0x49, 0x8F, 0x81, 0x5A, 0xE2, 0x40, 0xFF, 0x77, 0xFC,
|
||||||
PrivateExponent = new CryptoPP::byte[0x100]{
|
0x0D};
|
||||||
0x32, 0xD9, 0x03, 0x90, 0x8F, 0xBD, 0xB0, 0x8F, 0x57, 0x2B, 0x28, 0x5E, 0x0B, 0x8D,
|
|
||||||
0xB3, 0xEA, 0x5C, 0xD1, 0x7E, 0xA8, 0x90, 0x88, 0x8C, 0xDD, 0x6A, 0x80, 0xBB, 0xB1,
|
|
||||||
0xDF, 0xC1, 0xF7, 0x0D, 0xAA, 0x32, 0xF0, 0xB7, 0x7C, 0xCB, 0x88, 0x80, 0x0E, 0x8B,
|
|
||||||
0x64, 0xB0, 0xBE, 0x4C, 0xD6, 0x0E, 0x9B, 0x8C, 0x1E, 0x2A, 0x64, 0xE1, 0xF3, 0x5C,
|
|
||||||
0xD7, 0x76, 0x01, 0x41, 0x5E, 0x93, 0x5C, 0x94, 0xFE, 0xDD, 0x46, 0x62, 0xC3, 0x1B,
|
|
||||||
0x5A, 0xE2, 0xA0, 0xBC, 0x2D, 0xEB, 0xC3, 0x98, 0x0A, 0xA7, 0xB7, 0x85, 0x69, 0x70,
|
|
||||||
0x68, 0x2B, 0x64, 0x4A, 0xB3, 0x1F, 0xCC, 0x7D, 0xDC, 0x7C, 0x26, 0xF4, 0x77, 0xF6,
|
|
||||||
0x5C, 0xF2, 0xAE, 0x5A, 0x44, 0x2D, 0xD3, 0xAB, 0x16, 0x62, 0x04, 0x19, 0xBA, 0xFB,
|
|
||||||
0x90, 0xFF, 0xE2, 0x30, 0x50, 0x89, 0x6E, 0xCB, 0x56, 0xB2, 0xEB, 0xC0, 0x91, 0x16,
|
|
||||||
0x92, 0x5E, 0x30, 0x8E, 0xAE, 0xC7, 0x94, 0x5D, 0xFD, 0x35, 0xE1, 0x20, 0xF8, 0xAD,
|
|
||||||
0x3E, 0xBC, 0x08, 0xBF, 0xC0, 0x36, 0x74, 0x9F, 0xD5, 0xBB, 0x52, 0x08, 0xFD, 0x06,
|
|
||||||
0x66, 0xF3, 0x7A, 0xB3, 0x04, 0xF4, 0x75, 0x29, 0x5D, 0xE9, 0x5F, 0xAA, 0x10, 0x30,
|
|
||||||
0xB2, 0x0F, 0x5A, 0x1A, 0xC1, 0x2A, 0xB3, 0xFE, 0xCB, 0x21, 0xAD, 0x80, 0xEC, 0x8F,
|
|
||||||
0x20, 0x09, 0x1C, 0xDB, 0xC5, 0x58, 0x94, 0xC2, 0x9C, 0xC6, 0xCE, 0x82, 0x65, 0x3E,
|
|
||||||
0x57, 0x90, 0xBC, 0xA9, 0x8B, 0x06, 0xB4, 0xF0, 0x72, 0xF6, 0x77, 0xDF, 0x98, 0x64,
|
|
||||||
0xF1, 0xEC, 0xFE, 0x37, 0x2D, 0xBC, 0xAE, 0x8C, 0x08, 0x81, 0x1F, 0xC3, 0xC9, 0x89,
|
|
||||||
0x1A, 0xC7, 0x42, 0x82, 0x4B, 0x2E, 0xDC, 0x8E, 0x8D, 0x73, 0xCE, 0xB1, 0xCC, 0x01,
|
|
||||||
0xD9, 0x08, 0x70, 0x87, 0x3C, 0x44, 0x08, 0xEC, 0x49, 0x8F, 0x81, 0x5A, 0xE2, 0x40,
|
|
||||||
0xFF, 0x77, 0xFC, 0x0D};
|
|
||||||
Exponent1 = new CryptoPP::byte[0x80]{
|
|
||||||
0x52, 0xCC, 0x2D, 0xA0, 0x9C, 0x9E, 0x75, 0xE7, 0x28, 0xEE, 0x3D, 0xDE, 0xE3,
|
|
||||||
0x45, 0xD1, 0x4F, 0x94, 0x1C, 0xCC, 0xC8, 0x87, 0x29, 0x45, 0x3B, 0x8D, 0x6E,
|
|
||||||
0xAB, 0x6E, 0x2A, 0xA7, 0xC7, 0x15, 0x43, 0xA3, 0x04, 0x8F, 0x90, 0x5F, 0xEB,
|
|
||||||
0xF3, 0x38, 0x4A, 0x77, 0xFA, 0x36, 0xB7, 0x15, 0x76, 0xB6, 0x01, 0x1A, 0x8E,
|
|
||||||
0x25, 0x87, 0x82, 0xF1, 0x55, 0xD8, 0xC6, 0x43, 0x2A, 0xC0, 0xE5, 0x98, 0xC9,
|
|
||||||
0x32, 0xD1, 0x94, 0x6F, 0xD9, 0x01, 0xBA, 0x06, 0x81, 0xE0, 0x6D, 0x88, 0xF2,
|
|
||||||
0x24, 0x2A, 0x25, 0x01, 0x64, 0x5C, 0xBF, 0xF2, 0xD9, 0x99, 0x67, 0x3E, 0xF6,
|
|
||||||
0x72, 0xEE, 0xE4, 0xE2, 0x33, 0x5C, 0xF8, 0x00, 0x40, 0xE3, 0x2A, 0x9A, 0xF4,
|
|
||||||
0x3D, 0x22, 0x86, 0x44, 0x3C, 0xFB, 0x0A, 0xA5, 0x7C, 0x3F, 0xCC, 0xF5, 0xF1,
|
|
||||||
0x16, 0xC4, 0xAC, 0x88, 0xB4, 0xDE, 0x62, 0x94, 0x92, 0x6A, 0x13};
|
|
||||||
Exponent2 = new CryptoPP::byte[0x80]{
|
|
||||||
0x7C, 0x9D, 0xAD, 0x39, 0xE0, 0xD5, 0x60, 0x14, 0x94, 0x48, 0x19, 0x7F, 0x88,
|
|
||||||
0x95, 0xD5, 0x8B, 0x80, 0xAD, 0x85, 0x8A, 0x4B, 0x77, 0x37, 0x85, 0xD0, 0x77,
|
|
||||||
0xBB, 0xBF, 0x89, 0x71, 0x4A, 0x72, 0xCB, 0x72, 0x68, 0x38, 0xEC, 0x02, 0xC6,
|
|
||||||
0x7D, 0xC6, 0x44, 0x06, 0x33, 0x51, 0x1C, 0xC0, 0xFF, 0x95, 0x8F, 0x0D, 0x75,
|
|
||||||
0xDC, 0x25, 0xBB, 0x0B, 0x73, 0x91, 0xA9, 0x6D, 0x42, 0xD8, 0x03, 0xB7, 0x68,
|
|
||||||
0xD4, 0x1E, 0x75, 0x62, 0xA3, 0x70, 0x35, 0x79, 0x78, 0x00, 0xC8, 0xF5, 0xEF,
|
|
||||||
0x15, 0xB9, 0xFC, 0x4E, 0x47, 0x5A, 0xC8, 0x70, 0x70, 0x5B, 0x52, 0x98, 0xC0,
|
|
||||||
0xC2, 0x58, 0x4A, 0x70, 0x96, 0xCC, 0xB8, 0x10, 0xE1, 0x2F, 0x78, 0x8B, 0x2B,
|
|
||||||
0xA1, 0x7F, 0xF9, 0xAC, 0xDE, 0xF0, 0xBB, 0x2B, 0xE2, 0x66, 0xE3, 0x22, 0x92,
|
|
||||||
0x31, 0x21, 0x57, 0x92, 0xC4, 0xB8, 0xF2, 0x3E, 0x76, 0x20, 0x37};
|
|
||||||
Coefficient = new CryptoPP::byte[0x80]{
|
|
||||||
0x45, 0x97, 0x55, 0xD4, 0x22, 0x08, 0x5E, 0xF3, 0x5C, 0xB4, 0x05, 0x7A, 0xFD,
|
|
||||||
0xAA, 0x42, 0x42, 0xAD, 0x9A, 0x8C, 0xA0, 0x6C, 0xBB, 0x1D, 0x68, 0x54, 0x54,
|
|
||||||
0x6E, 0x3E, 0x32, 0xE3, 0x53, 0x73, 0x76, 0xF1, 0x3E, 0x01, 0xEA, 0xD3, 0xCF,
|
|
||||||
0xEB, 0xEB, 0x23, 0x3E, 0xC0, 0xBE, 0xCE, 0xEC, 0x2C, 0x89, 0x5F, 0xA8, 0x27,
|
|
||||||
0x3A, 0x4C, 0xB7, 0xE6, 0x74, 0xBC, 0x45, 0x4C, 0x26, 0xC8, 0x25, 0xFF, 0x34,
|
|
||||||
0x63, 0x25, 0x37, 0xE1, 0x48, 0x10, 0xC1, 0x93, 0xA6, 0xAF, 0xEB, 0xBA, 0xE3,
|
|
||||||
0xA2, 0xF1, 0x3D, 0xEF, 0x63, 0xD8, 0xF4, 0xFD, 0xD3, 0xEE, 0xE2, 0x5D, 0xE9,
|
|
||||||
0x33, 0xCC, 0xAD, 0xBA, 0x75, 0x5C, 0x85, 0xAF, 0xCE, 0xA9, 0x3D, 0xD1, 0xA2,
|
|
||||||
0x17, 0xF3, 0xF6, 0x98, 0xB3, 0x50, 0x8E, 0x5E, 0xF6, 0xEB, 0x02, 0x8E, 0xA1,
|
|
||||||
0x62, 0xA7, 0xD6, 0x2C, 0xEC, 0x91, 0xFF, 0x15, 0x40, 0xD2, 0xE3};
|
|
||||||
Modulus = new CryptoPP::byte[0x100]{
|
|
||||||
0xd2, 0x12, 0xfc, 0x33, 0x5f, 0x6d, 0xdb, 0x83, 0x16, 0x09, 0x62, 0x8b, 0x03, 0x56,
|
|
||||||
0x27, 0x37, 0x82, 0xd4, 0x77, 0x85, 0x35, 0x29, 0x39, 0x2d, 0x52, 0x6b, 0x8c, 0x4c,
|
|
||||||
0x8c, 0xfb, 0x06, 0xc1, 0x84, 0x5b, 0xe7, 0xd4, 0xf7, 0xbc, 0xd2, 0x4e, 0x62, 0x45,
|
|
||||||
0xcd, 0x2a, 0xbb, 0xd7, 0x77, 0x76, 0x45, 0x36, 0x55, 0x27, 0x3f, 0xb3, 0xf5, 0xf9,
|
|
||||||
0x8e, 0xda, 0x4b, 0xef, 0xaa, 0x59, 0xae, 0xb3, 0x9b, 0xea, 0x54, 0x98, 0xd2, 0x06,
|
|
||||||
0x32, 0x6a, 0x58, 0x31, 0x2a, 0xe0, 0xd4, 0x4f, 0x90, 0xb5, 0x0a, 0x7d, 0xec, 0xf4,
|
|
||||||
0x3a, 0x9c, 0x52, 0x67, 0x2d, 0x99, 0x31, 0x8e, 0x0c, 0x43, 0xe6, 0x82, 0xfe, 0x07,
|
|
||||||
0x46, 0xe1, 0x2e, 0x50, 0xd4, 0x1f, 0x2d, 0x2f, 0x7e, 0xd9, 0x08, 0xba, 0x06, 0xb3,
|
|
||||||
0xbf, 0x2e, 0x20, 0x3f, 0x4e, 0x3f, 0xfe, 0x44, 0xff, 0xaa, 0x50, 0x43, 0x57, 0x91,
|
|
||||||
0x69, 0x94, 0x49, 0x15, 0x82, 0x82, 0xe4, 0x0f, 0x4c, 0x8d, 0x9d, 0x2c, 0xc9, 0x5b,
|
|
||||||
0x1d, 0x64, 0xbf, 0x88, 0x8b, 0xd4, 0xc5, 0x94, 0xe7, 0x65, 0x47, 0x84, 0x1e, 0xe5,
|
|
||||||
0x79, 0x10, 0xfb, 0x98, 0x93, 0x47, 0xb9, 0x7d, 0x85, 0x12, 0xa6, 0x40, 0x98, 0x2c,
|
|
||||||
0xf7, 0x92, 0xbc, 0x95, 0x19, 0x32, 0xed, 0xe8, 0x90, 0x56, 0x0d, 0x65, 0xc1, 0xaa,
|
|
||||||
0x78, 0xc6, 0x2e, 0x54, 0xfd, 0x5f, 0x54, 0xa1, 0xf6, 0x7e, 0xe5, 0xe0, 0x5f, 0x61,
|
|
||||||
0xc1, 0x20, 0xb4, 0xb9, 0xb4, 0x33, 0x08, 0x70, 0xe4, 0xdf, 0x89, 0x56, 0xed, 0x01,
|
|
||||||
0x29, 0x46, 0x77, 0x5f, 0x8c, 0xb8, 0xa9, 0xf5, 0x1e, 0x2e, 0xb3, 0xb9, 0xbf, 0xe0,
|
|
||||||
0x09, 0xb7, 0x8d, 0x28, 0xd4, 0xa6, 0xc3, 0xb8, 0x1e, 0x1f, 0x07, 0xeb, 0xb4, 0x12,
|
|
||||||
0x0b, 0x95, 0xb8, 0x85, 0x30, 0xfd, 0xdc, 0x39, 0x13, 0xd0, 0x7c, 0xdc, 0x8f, 0xed,
|
|
||||||
0xf9, 0xc9, 0xa3, 0xc1};
|
|
||||||
PublicExponent = new CryptoPP::byte[4]{0, 1, 0, 1};
|
|
||||||
};
|
|
||||||
};
|
};
|
|
@ -67,15 +67,19 @@ bool PKG::Open(const std::filesystem::path& filepath) {
|
||||||
file.Seek(0x47); // skip first 7 characters of content_id
|
file.Seek(0x47); // skip first 7 characters of content_id
|
||||||
file.Read(pkgTitleID);
|
file.Read(pkgTitleID);
|
||||||
|
|
||||||
file.Seek(0);
|
|
||||||
pkg.resize(pkgheader.pkg_promote_size);
|
|
||||||
file.Read(pkg);
|
|
||||||
|
|
||||||
u32 offset = pkgheader.pkg_table_entry_offset;
|
u32 offset = pkgheader.pkg_table_entry_offset;
|
||||||
u32 n_files = pkgheader.pkg_table_entry_count;
|
u32 n_files = pkgheader.pkg_table_entry_count;
|
||||||
|
|
||||||
|
file.Seek(offset);
|
||||||
for (int i = 0; i < n_files; i++) {
|
for (int i = 0; i < n_files; i++) {
|
||||||
PKGEntry entry;
|
PKGEntry entry{};
|
||||||
std::memcpy(&entry, &pkg[offset + i * 0x20], sizeof(entry));
|
file.Read(entry.id);
|
||||||
|
file.Read(entry.filename_offset);
|
||||||
|
file.Read(entry.flags1);
|
||||||
|
file.Read(entry.flags2);
|
||||||
|
file.Read(entry.offset);
|
||||||
|
file.Read(entry.size);
|
||||||
|
file.Seek(8, Common::FS::SeekOrigin::CurrentPosition);
|
||||||
|
|
||||||
// Try to figure out the name
|
// Try to figure out the name
|
||||||
const auto name = GetEntryNameByType(entry.id);
|
const auto name = GetEntryNameByType(entry.id);
|
||||||
|
@ -113,9 +117,6 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
failreason = "Content size is bigger than pkg size";
|
failreason = "Content size is bigger than pkg size";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
file.Seek(0);
|
|
||||||
pkg.resize(pkgheader.pkg_promote_size);
|
|
||||||
file.Read(pkg);
|
|
||||||
|
|
||||||
u32 offset = pkgheader.pkg_table_entry_offset;
|
u32 offset = pkgheader.pkg_table_entry_offset;
|
||||||
u32 n_files = pkgheader.pkg_table_entry_count;
|
u32 n_files = pkgheader.pkg_table_entry_count;
|
||||||
|
@ -126,9 +127,18 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
std::array<std::array<u8, 256>, 7> key1;
|
std::array<std::array<u8, 256>, 7> key1;
|
||||||
std::array<u8, 256> imgkeydata;
|
std::array<u8, 256> imgkeydata;
|
||||||
|
|
||||||
|
file.Seek(offset);
|
||||||
for (int i = 0; i < n_files; i++) {
|
for (int i = 0; i < n_files; i++) {
|
||||||
PKGEntry entry;
|
PKGEntry entry{};
|
||||||
std::memcpy(&entry, &pkg[offset + i * 0x20], sizeof(entry));
|
file.Read(entry.id);
|
||||||
|
file.Read(entry.filename_offset);
|
||||||
|
file.Read(entry.flags1);
|
||||||
|
file.Read(entry.flags2);
|
||||||
|
file.Read(entry.offset);
|
||||||
|
file.Read(entry.size);
|
||||||
|
file.Seek(8, Common::FS::SeekOrigin::CurrentPosition);
|
||||||
|
|
||||||
|
auto currentPos = file.Tell();
|
||||||
|
|
||||||
// Try to figure out the name
|
// Try to figure out the name
|
||||||
const auto name = GetEntryNameByType(entry.id);
|
const auto name = GetEntryNameByType(entry.id);
|
||||||
|
@ -139,8 +149,15 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
// Just print with id
|
// Just print with id
|
||||||
Common::FS::IOFile out(extract_path / "sce_sys" / std::to_string(entry.id),
|
Common::FS::IOFile out(extract_path / "sce_sys" / std::to_string(entry.id),
|
||||||
Common::FS::FileAccessMode::Write);
|
Common::FS::FileAccessMode::Write);
|
||||||
out.WriteRaw<u8>(pkg.data() + entry.offset, entry.size);
|
file.Seek(entry.offset);
|
||||||
|
|
||||||
|
std::vector<u8> data;
|
||||||
|
data.resize(entry.size);
|
||||||
|
file.ReadRaw<u8>(data.data(), entry.size);
|
||||||
|
out.WriteRaw<u8>(data.data(), entry.size);
|
||||||
out.Close();
|
out.Close();
|
||||||
|
|
||||||
|
file.Seek(currentPos);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -178,14 +195,25 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
}
|
}
|
||||||
|
|
||||||
Common::FS::IOFile out(extract_path / "sce_sys" / name, Common::FS::FileAccessMode::Write);
|
Common::FS::IOFile out(extract_path / "sce_sys" / name, Common::FS::FileAccessMode::Write);
|
||||||
out.WriteRaw<u8>(pkg.data() + entry.offset, entry.size);
|
file.Seek(entry.offset);
|
||||||
|
|
||||||
|
std::vector<u8> data;
|
||||||
|
data.resize(entry.size);
|
||||||
|
file.ReadRaw<u8>(data.data(), entry.size);
|
||||||
|
out.WriteRaw<u8>(data.data(), entry.size);
|
||||||
out.Close();
|
out.Close();
|
||||||
|
|
||||||
// Decrypt Np stuff and overwrite.
|
// Decrypt Np stuff and overwrite.
|
||||||
if (entry.id == 0x400 || entry.id == 0x401 || entry.id == 0x402 ||
|
if (entry.id == 0x400 || entry.id == 0x401 || entry.id == 0x402 ||
|
||||||
entry.id == 0x403) { // somehow 0x401 is not decrypting
|
entry.id == 0x403) { // somehow 0x401 is not decrypting
|
||||||
decNp.resize(entry.size);
|
decNp.resize(entry.size);
|
||||||
std::span<u8> cipherNp(pkg.data() + entry.offset, entry.size);
|
file.Seek(entry.offset);
|
||||||
|
|
||||||
|
std::vector<u8> data;
|
||||||
|
data.resize(entry.size);
|
||||||
|
file.ReadRaw<u8>(data.data(), entry.size);
|
||||||
|
|
||||||
|
std::span<u8> cipherNp(data.data(), entry.size);
|
||||||
std::array<u8, 64> concatenated_ivkey_dk3_;
|
std::array<u8, 64> concatenated_ivkey_dk3_;
|
||||||
std::memcpy(concatenated_ivkey_dk3_.data(), &entry, sizeof(entry));
|
std::memcpy(concatenated_ivkey_dk3_.data(), &entry, sizeof(entry));
|
||||||
std::memcpy(concatenated_ivkey_dk3_.data() + sizeof(entry), dk3_.data(), sizeof(dk3_));
|
std::memcpy(concatenated_ivkey_dk3_.data() + sizeof(entry), dk3_.data(), sizeof(dk3_));
|
||||||
|
@ -197,6 +225,8 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
out.Write(decNp);
|
out.Write(decNp);
|
||||||
out.Close();
|
out.Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
file.Seek(currentPos);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extract trophy files
|
// Extract trophy files
|
||||||
|
@ -214,28 +244,31 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
PKG::crypto.PfsGenCryptoKey(ekpfsKey, seed, dataKey, tweakKey);
|
PKG::crypto.PfsGenCryptoKey(ekpfsKey, seed, dataKey, tweakKey);
|
||||||
const u32 length = pkgheader.pfs_cache_size * 0x2; // Seems to be ok.
|
const u32 length = pkgheader.pfs_cache_size * 0x2; // Seems to be ok.
|
||||||
|
|
||||||
// Read encrypted pfs_image
|
int num_blocks = 0;
|
||||||
std::vector<u8> pfs_encrypted(length);
|
|
||||||
file.Seek(pkgheader.pfs_image_offset);
|
|
||||||
file.Read(pfs_encrypted);
|
|
||||||
file.Close();
|
|
||||||
// Decrypt the pfs_image.
|
|
||||||
std::vector<u8> pfs_decrypted(length);
|
|
||||||
PKG::crypto.decryptPFS(dataKey, tweakKey, pfs_encrypted, pfs_decrypted, 0);
|
|
||||||
|
|
||||||
// Retrieve PFSC from decrypted pfs_image.
|
|
||||||
pfsc_offset = GetPFSCOffset(pfs_decrypted);
|
|
||||||
std::vector<u8> pfsc(length);
|
std::vector<u8> pfsc(length);
|
||||||
std::memcpy(pfsc.data(), pfs_decrypted.data() + pfsc_offset, length - pfsc_offset);
|
if (length != 0) {
|
||||||
|
// Read encrypted pfs_image
|
||||||
|
std::vector<u8> pfs_encrypted(length);
|
||||||
|
file.Seek(pkgheader.pfs_image_offset);
|
||||||
|
file.Read(pfs_encrypted);
|
||||||
|
file.Close();
|
||||||
|
// Decrypt the pfs_image.
|
||||||
|
std::vector<u8> pfs_decrypted(length);
|
||||||
|
PKG::crypto.decryptPFS(dataKey, tweakKey, pfs_encrypted, pfs_decrypted, 0);
|
||||||
|
|
||||||
PFSCHdr pfsChdr;
|
// Retrieve PFSC from decrypted pfs_image.
|
||||||
std::memcpy(&pfsChdr, pfsc.data(), sizeof(pfsChdr));
|
pfsc_offset = GetPFSCOffset(pfs_decrypted);
|
||||||
|
std::memcpy(pfsc.data(), pfs_decrypted.data() + pfsc_offset, length - pfsc_offset);
|
||||||
|
|
||||||
const int num_blocks = (int)(pfsChdr.data_length / pfsChdr.block_sz2);
|
PFSCHdr pfsChdr;
|
||||||
sectorMap.resize(num_blocks + 1); // 8 bytes, need extra 1 to get the last offset.
|
std::memcpy(&pfsChdr, pfsc.data(), sizeof(pfsChdr));
|
||||||
|
|
||||||
for (int i = 0; i < num_blocks + 1; i++) {
|
num_blocks = (int)(pfsChdr.data_length / pfsChdr.block_sz2);
|
||||||
std::memcpy(§orMap[i], pfsc.data() + pfsChdr.block_offsets + i * 8, 8);
|
sectorMap.resize(num_blocks + 1); // 8 bytes, need extra 1 to get the last offset.
|
||||||
|
|
||||||
|
for (int i = 0; i < num_blocks + 1; i++) {
|
||||||
|
std::memcpy(§orMap[i], pfsc.data() + pfsChdr.block_offsets + i * 8, 8);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 ent_size = 0;
|
u32 ent_size = 0;
|
||||||
|
@ -279,8 +312,8 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// let's deal with the root/uroot enteries here.
|
// let's deal with the root/uroot entries here.
|
||||||
// Sometimes it's more than 2 enteries (Tomb Raider Remastered)
|
// Sometimes it's more than 2 entries (Tomb Raider Remastered)
|
||||||
const std::string_view flat_path_table(&decompressedData[0x10], 15);
|
const std::string_view flat_path_table(&decompressedData[0x10], 15);
|
||||||
if (flat_path_table == "flat_path_table") {
|
if (flat_path_table == "flat_path_table") {
|
||||||
uroot_reached = true;
|
uroot_reached = true;
|
||||||
|
@ -296,7 +329,15 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
} else {
|
} else {
|
||||||
// Set the the folder according to the current inode.
|
// Set the the folder according to the current inode.
|
||||||
// Can be 2 or more (rarely)
|
// Can be 2 or more (rarely)
|
||||||
extractPaths[ndinode_counter] = extract_path.parent_path() / GetTitleID();
|
auto parent_path = extract_path.parent_path();
|
||||||
|
auto title_id = GetTitleID();
|
||||||
|
|
||||||
|
if (parent_path.filename() != title_id) {
|
||||||
|
extractPaths[ndinode_counter] = parent_path / title_id;
|
||||||
|
} else {
|
||||||
|
// DLCs path has different structure
|
||||||
|
extractPaths[ndinode_counter] = extract_path;
|
||||||
|
}
|
||||||
uroot_reached = false;
|
uroot_reached = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -350,7 +391,7 @@ bool PKG::Extract(const std::filesystem::path& filepath, const std::filesystem::
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PKG::ExtractFiles(const int& index) {
|
void PKG::ExtractFiles(const int index) {
|
||||||
int inode_number = fsTable[index].inode;
|
int inode_number = fsTable[index].inode;
|
||||||
int inode_type = fsTable[index].type;
|
int inode_type = fsTable[index].type;
|
||||||
std::string inode_name = fsTable[index].name;
|
std::string inode_name = fsTable[index].name;
|
||||||
|
|
|
@ -104,7 +104,7 @@ public:
|
||||||
~PKG();
|
~PKG();
|
||||||
|
|
||||||
bool Open(const std::filesystem::path& filepath);
|
bool Open(const std::filesystem::path& filepath);
|
||||||
void ExtractFiles(const int& index);
|
void ExtractFiles(const int index);
|
||||||
bool Extract(const std::filesystem::path& filepath, const std::filesystem::path& extract,
|
bool Extract(const std::filesystem::path& filepath, const std::filesystem::path& extract,
|
||||||
std::string& failreason);
|
std::string& failreason);
|
||||||
|
|
||||||
|
@ -149,7 +149,6 @@ public:
|
||||||
private:
|
private:
|
||||||
Crypto crypto;
|
Crypto crypto;
|
||||||
TRP trp;
|
TRP trp;
|
||||||
std::vector<u8> pkg;
|
|
||||||
u64 pkgSize = 0;
|
u64 pkgSize = 0;
|
||||||
char pkgTitleID[9];
|
char pkgTitleID[9];
|
||||||
PKGHeader pkgheader;
|
PKGHeader pkgheader;
|
||||||
|
|
|
@ -1,16 +1,75 @@
|
||||||
// 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
|
||||||
|
|
||||||
#include "common/io_file.h"
|
|
||||||
|
|
||||||
#include "playgo_chunk.h"
|
#include "playgo_chunk.h"
|
||||||
|
|
||||||
bool PlaygoChunk::Open(const std::filesystem::path& filepath) {
|
bool PlaygoFile::Open(const std::filesystem::path& filepath) {
|
||||||
Common::FS::IOFile file(filepath, Common::FS::FileAccessMode::Read);
|
Common::FS::IOFile file(filepath, Common::FS::FileAccessMode::Read);
|
||||||
if (!file.IsOpen()) {
|
if (file.IsOpen()) {
|
||||||
return false;
|
file.Read(playgoHeader);
|
||||||
|
if (LoadChunks(file)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
file.Read(playgoHeader);
|
return false;
|
||||||
|
}
|
||||||
return true;
|
|
||||||
|
bool PlaygoFile::LoadChunks(const Common::FS::IOFile& file) {
|
||||||
|
if (file.IsOpen()) {
|
||||||
|
if (playgoHeader.magic == PLAYGO_MAGIC) {
|
||||||
|
bool ret = true;
|
||||||
|
|
||||||
|
std::string chunk_attrs_data, chunk_mchunks_data, chunk_labels_data, mchunk_attrs_data;
|
||||||
|
ret = ret && load_chunk_data(file, playgoHeader.chunk_attrs, chunk_attrs_data);
|
||||||
|
ret = ret && load_chunk_data(file, playgoHeader.chunk_mchunks, chunk_mchunks_data);
|
||||||
|
ret = ret && load_chunk_data(file, playgoHeader.chunk_labels, chunk_labels_data);
|
||||||
|
ret = ret && load_chunk_data(file, playgoHeader.mchunk_attrs, mchunk_attrs_data);
|
||||||
|
|
||||||
|
if (ret) {
|
||||||
|
chunks.resize(playgoHeader.chunk_count);
|
||||||
|
|
||||||
|
auto chunk_attrs =
|
||||||
|
reinterpret_cast<playgo_chunk_attr_entry_t*>(&chunk_attrs_data[0]);
|
||||||
|
auto chunk_mchunks = reinterpret_cast<u16*>(&chunk_mchunks_data[0]);
|
||||||
|
auto chunk_labels = reinterpret_cast<char*>(&chunk_labels_data[0]);
|
||||||
|
auto mchunk_attrs =
|
||||||
|
reinterpret_cast<playgo_mchunk_attr_entry_t*>(&mchunk_attrs_data[0]);
|
||||||
|
|
||||||
|
for (u16 i = 0; i < playgoHeader.chunk_count; i++) {
|
||||||
|
chunks[i].req_locus = chunk_attrs[i].req_locus;
|
||||||
|
chunks[i].language_mask = chunk_attrs[i].language_mask;
|
||||||
|
chunks[i].label_name = std::string(chunk_labels + chunk_attrs[i].label_offset);
|
||||||
|
|
||||||
|
u64 total_size = 0;
|
||||||
|
u16 mchunk_count = chunk_attrs[i].mchunk_count;
|
||||||
|
if (mchunk_count != 0) {
|
||||||
|
auto mchunks = reinterpret_cast<u16*>(
|
||||||
|
((u8*)chunk_mchunks + chunk_attrs[i].mchunks_offset));
|
||||||
|
for (u16 j = 0; j < mchunk_count; j++) {
|
||||||
|
u16 mchunk_id = mchunks[j];
|
||||||
|
total_size += mchunk_attrs[mchunk_id].size.size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
chunks[i].total_size = total_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PlaygoFile::load_chunk_data(const Common::FS::IOFile& file, const chunk_t chunk,
|
||||||
|
std::string& data) {
|
||||||
|
if (file.IsOpen()) {
|
||||||
|
if (file.Seek(chunk.offset)) {
|
||||||
|
data.resize(chunk.length);
|
||||||
|
if (data.size() == chunk.length) {
|
||||||
|
file.ReadRaw<char>(&data[0], chunk.length);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
|
@ -3,29 +3,129 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
#include "common/types.h"
|
#include <mutex>
|
||||||
|
#include <vector>
|
||||||
|
#include "common/io_file.h"
|
||||||
|
#include "core/libraries/playgo/playgo_types.h"
|
||||||
|
|
||||||
|
constexpr u32 PLAYGO_MAGIC = 0x6F676C70;
|
||||||
|
|
||||||
|
struct chunk_t {
|
||||||
|
u32 offset;
|
||||||
|
u32 length;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
struct PlaygoHeader {
|
struct PlaygoHeader {
|
||||||
u32 magic;
|
u32 magic;
|
||||||
|
|
||||||
u16 version_major;
|
u16 version_major;
|
||||||
u16 version_minor;
|
u16 version_minor;
|
||||||
u16 image_count;
|
u16 image_count; // [0;1]
|
||||||
|
u16 chunk_count; // [0;1000]
|
||||||
|
u16 mchunk_count; // [0;8000]
|
||||||
|
u16 scenario_count; // [0;32]
|
||||||
|
|
||||||
|
u32 file_size;
|
||||||
|
u16 default_scenario_id;
|
||||||
|
u16 attrib;
|
||||||
|
u32 sdk_version;
|
||||||
|
u16 disc_count; // [0;2] (if equals to 0 then disc count = 1)
|
||||||
|
u16 layer_bmp;
|
||||||
|
|
||||||
|
u8 reserved[32];
|
||||||
|
char content_id[128];
|
||||||
|
|
||||||
|
chunk_t chunk_attrs; // [0;32000]
|
||||||
|
chunk_t chunk_mchunks;
|
||||||
|
chunk_t chunk_labels; // [0;16000]
|
||||||
|
chunk_t mchunk_attrs; // [0;12800]
|
||||||
|
chunk_t scenario_attrs; // [0;1024]
|
||||||
|
chunk_t scenario_chunks;
|
||||||
|
chunk_t scenario_labels;
|
||||||
|
chunk_t inner_mchunk_attrs; // [0;12800]
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct playgo_scenario_attr_entry_t {
|
||||||
|
u8 _type;
|
||||||
|
u8 _unk[19];
|
||||||
|
u16 initial_chunk_count;
|
||||||
u16 chunk_count;
|
u16 chunk_count;
|
||||||
|
u32 chunks_offset; //<-scenario_chunks
|
||||||
|
u32 label_offset; //<-scenario_labels
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct image_disc_layer_no_t {
|
||||||
|
u8 layer_no : 2;
|
||||||
|
u8 disc_no : 2;
|
||||||
|
u8 image_no : 4;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct playgo_chunk_attr_entry_t {
|
||||||
|
u8 flag;
|
||||||
|
image_disc_layer_no_t image_disc_layer_no;
|
||||||
|
u8 req_locus;
|
||||||
|
u8 unk[11];
|
||||||
u16 mchunk_count;
|
u16 mchunk_count;
|
||||||
u16 scenario_count;
|
u64 language_mask;
|
||||||
// TODO fill the rest
|
u32 mchunks_offset; //<-chunk_mchunks
|
||||||
|
u32 label_offset; //<-chunk_labels
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct playgo_chunk_loc_t {
|
||||||
|
u64 offset : 48;
|
||||||
|
u64 _align1 : 8;
|
||||||
|
u64 image_no : 4;
|
||||||
|
u64 _align2 : 4;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct playgo_chunk_size_t {
|
||||||
|
u64 size : 48;
|
||||||
|
u64 _align : 16;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct playgo_mchunk_attr_entry_t {
|
||||||
|
playgo_chunk_loc_t loc;
|
||||||
|
playgo_chunk_size_t size;
|
||||||
|
} __attribute__((packed));
|
||||||
|
|
||||||
|
struct PlaygoChunk {
|
||||||
|
u64 req_locus;
|
||||||
|
u64 language_mask;
|
||||||
|
u64 total_size;
|
||||||
|
std::string label_name;
|
||||||
};
|
};
|
||||||
class PlaygoChunk {
|
|
||||||
|
class PlaygoFile {
|
||||||
public:
|
public:
|
||||||
PlaygoChunk() = default;
|
bool initialized;
|
||||||
~PlaygoChunk() = default;
|
OrbisPlayGoHandle handle;
|
||||||
|
OrbisPlayGoChunkId id;
|
||||||
|
OrbisPlayGoLocus locus;
|
||||||
|
OrbisPlayGoInstallSpeed speed;
|
||||||
|
s64 speed_tick;
|
||||||
|
OrbisPlayGoEta eta;
|
||||||
|
OrbisPlayGoLanguageMask langMask;
|
||||||
|
std::vector<PlaygoChunk> chunks;
|
||||||
|
|
||||||
|
public:
|
||||||
|
PlaygoFile()
|
||||||
|
: initialized(false), handle(0), id(0), locus(0), speed(ORBIS_PLAYGO_INSTALL_SPEED_TRICKLE),
|
||||||
|
speed_tick(0), eta(0), langMask(0), playgoHeader{0} {}
|
||||||
|
~PlaygoFile() = default;
|
||||||
|
|
||||||
bool Open(const std::filesystem::path& filepath);
|
bool Open(const std::filesystem::path& filepath);
|
||||||
PlaygoHeader GetPlaygoHeader() {
|
bool LoadChunks(const Common::FS::IOFile& file);
|
||||||
|
PlaygoHeader& GetPlaygoHeader() {
|
||||||
return playgoHeader;
|
return playgoHeader;
|
||||||
}
|
}
|
||||||
|
std::mutex& GetSpeedMutex() {
|
||||||
|
return speed_mutex;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool load_chunk_data(const Common::FS::IOFile& file, const chunk_t chunk, std::string& data);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
PlaygoHeader playgoHeader;
|
PlaygoHeader playgoHeader;
|
||||||
|
std::mutex speed_mutex;
|
||||||
};
|
};
|
|
@ -9,7 +9,7 @@ PSF::PSF() = default;
|
||||||
|
|
||||||
PSF::~PSF() = default;
|
PSF::~PSF() = default;
|
||||||
|
|
||||||
bool PSF::open(const std::string& filepath, std::vector<u8> psfBuffer) {
|
bool PSF::open(const std::string& filepath, const std::vector<u8>& psfBuffer) {
|
||||||
if (!psfBuffer.empty()) {
|
if (!psfBuffer.empty()) {
|
||||||
psf.resize(psfBuffer.size());
|
psf.resize(psfBuffer.size());
|
||||||
psf = psfBuffer;
|
psf = psfBuffer;
|
||||||
|
|
|
@ -35,7 +35,7 @@ public:
|
||||||
PSF();
|
PSF();
|
||||||
~PSF();
|
~PSF();
|
||||||
|
|
||||||
bool open(const std::string& filepath, std::vector<u8> psfBuffer);
|
bool open(const std::string& filepath, const std::vector<u8>& psfBuffer);
|
||||||
|
|
||||||
std::string GetString(const std::string& key);
|
std::string GetString(const std::string& key);
|
||||||
u32 GetInteger(const std::string& key);
|
u32 GetInteger(const std::string& key);
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
TRP::TRP() = default;
|
TRP::TRP() = default;
|
||||||
TRP::~TRP() = default;
|
TRP::~TRP() = default;
|
||||||
|
|
||||||
void TRP::GetNPcommID(std::filesystem::path trophyPath, int index) {
|
void TRP::GetNPcommID(const std::filesystem::path& trophyPath, int index) {
|
||||||
std::filesystem::path trpPath = trophyPath / "sce_sys/npbind.dat";
|
std::filesystem::path trpPath = trophyPath / "sce_sys/npbind.dat";
|
||||||
Common::FS::IOFile npbindFile(trpPath, Common::FS::FileAccessMode::Read);
|
Common::FS::IOFile npbindFile(trpPath, Common::FS::FileAccessMode::Read);
|
||||||
if (!npbindFile.IsOpen()) {
|
if (!npbindFile.IsOpen()) {
|
||||||
|
@ -27,7 +27,7 @@ static void removePadding(std::vector<u8>& vec) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TRP::Extract(std::filesystem::path trophyPath) {
|
bool TRP::Extract(const std::filesystem::path& trophyPath) {
|
||||||
std::string title = trophyPath.filename().string();
|
std::string title = trophyPath.filename().string();
|
||||||
std::filesystem::path gameSysDir = trophyPath / "sce_sys/trophy/";
|
std::filesystem::path gameSysDir = trophyPath / "sce_sys/trophy/";
|
||||||
if (!std::filesystem::exists(gameSysDir)) {
|
if (!std::filesystem::exists(gameSysDir)) {
|
||||||
|
|
|
@ -33,8 +33,8 @@ class TRP {
|
||||||
public:
|
public:
|
||||||
TRP();
|
TRP();
|
||||||
~TRP();
|
~TRP();
|
||||||
bool Extract(std::filesystem::path trophyPath);
|
bool Extract(const std::filesystem::path& trophyPath);
|
||||||
void GetNPcommID(std::filesystem::path trophyPath, int index);
|
void GetNPcommID(const std::filesystem::path& trophyPath, int index);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Crypto crypto;
|
Crypto crypto;
|
||||||
|
|
|
@ -25,9 +25,9 @@ void MntPoints::UnmountAll() {
|
||||||
m_mnt_pairs.clear();
|
m_mnt_pairs.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::filesystem::path MntPoints::GetHostPath(const std::string& guest_directory) {
|
std::filesystem::path MntPoints::GetHostPath(std::string_view guest_directory) {
|
||||||
// Evil games like Turok2 pass double slashes e.g /app0//game.kpf
|
// Evil games like Turok2 pass double slashes e.g /app0//game.kpf
|
||||||
auto corrected_path = guest_directory;
|
std::string corrected_path(guest_directory);
|
||||||
size_t pos = corrected_path.find("//");
|
size_t pos = corrected_path.find("//");
|
||||||
while (pos != std::string::npos) {
|
while (pos != std::string::npos) {
|
||||||
corrected_path.replace(pos, 2, "/");
|
corrected_path.replace(pos, 2, "/");
|
||||||
|
@ -48,12 +48,13 @@ std::filesystem::path MntPoints::GetHostPath(const std::string& guest_directory)
|
||||||
pos = mount->mount.size() + 1;
|
pos = mount->mount.size() + 1;
|
||||||
const auto rel_path = std::string_view(corrected_path).substr(pos);
|
const auto rel_path = std::string_view(corrected_path).substr(pos);
|
||||||
const auto host_path = mount->host_path / rel_path;
|
const auto host_path = mount->host_path / rel_path;
|
||||||
if (!NeedsCaseInsensiveSearch) {
|
if (!NeedsCaseInsensitiveSearch) {
|
||||||
return host_path;
|
return host_path;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the path does not exist attempt to verify this.
|
// If the path does not exist attempt to verify this.
|
||||||
// Retrieve parent path until we find one that exists.
|
// Retrieve parent path until we find one that exists.
|
||||||
|
std::scoped_lock lk{m_mutex};
|
||||||
path_parts.clear();
|
path_parts.clear();
|
||||||
auto current_path = host_path;
|
auto current_path = host_path;
|
||||||
while (!std::filesystem::exists(current_path)) {
|
while (!std::filesystem::exists(current_path)) {
|
||||||
|
@ -70,7 +71,7 @@ std::filesystem::path MntPoints::GetHostPath(const std::string& guest_directory)
|
||||||
// exist in filesystem but in different case.
|
// exist in filesystem but in different case.
|
||||||
auto guest_path = current_path;
|
auto guest_path = current_path;
|
||||||
while (!path_parts.empty()) {
|
while (!path_parts.empty()) {
|
||||||
const auto& part = path_parts.back();
|
const auto part = path_parts.back();
|
||||||
const auto add_match = [&](const auto& host_part) {
|
const auto add_match = [&](const auto& host_part) {
|
||||||
current_path /= host_part;
|
current_path /= host_part;
|
||||||
guest_path /= part;
|
guest_path /= part;
|
||||||
|
|
|
@ -14,9 +14,9 @@ namespace Core::FileSys {
|
||||||
|
|
||||||
class MntPoints {
|
class MntPoints {
|
||||||
#ifdef _WIN64
|
#ifdef _WIN64
|
||||||
static constexpr bool NeedsCaseInsensiveSearch = false;
|
static constexpr bool NeedsCaseInsensitiveSearch = false;
|
||||||
#else
|
#else
|
||||||
static constexpr bool NeedsCaseInsensiveSearch = true;
|
static constexpr bool NeedsCaseInsensitiveSearch = true;
|
||||||
#endif
|
#endif
|
||||||
public:
|
public:
|
||||||
struct MntPair {
|
struct MntPair {
|
||||||
|
@ -31,7 +31,7 @@ public:
|
||||||
void Unmount(const std::filesystem::path& host_folder, const std::string& guest_folder);
|
void Unmount(const std::filesystem::path& host_folder, const std::string& guest_folder);
|
||||||
void UnmountAll();
|
void UnmountAll();
|
||||||
|
|
||||||
std::filesystem::path GetHostPath(const std::string& guest_directory);
|
std::filesystem::path GetHostPath(std::string_view guest_directory);
|
||||||
|
|
||||||
const MntPair* GetMount(const std::string& guest_path) {
|
const MntPair* GetMount(const std::string& guest_path) {
|
||||||
const auto it = std::ranges::find_if(
|
const auto it = std::ranges::find_if(
|
||||||
|
|
|
@ -1,9 +1,10 @@
|
||||||
// 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
|
||||||
|
|
||||||
// Generated By moduleGenerator
|
#include "ajm.h"
|
||||||
|
#include "ajm_error.h"
|
||||||
|
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
#include "core/libraries/ajm/ajm.h"
|
|
||||||
#include "core/libraries/error_codes.h"
|
#include "core/libraries/error_codes.h"
|
||||||
#include "core/libraries/libs.h"
|
#include "core/libraries/libs.h"
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
constexpr int ORBIS_AJM_ERROR_UNKNOWN = 0x80930001;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_CONTEXT = 0x80930002;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_INSTANCE = 0x80930003;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_BATCH = 0x80930004;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_PARAMETER = 0x80930005;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_OUT_OF_MEMORY = 0x80930006;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_OUT_OF_RESOURCES = 0x80930007;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_CODEC_NOT_SUPPORTED = 0x80930008;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_CODEC_ALREADY_REGISTERED = 0x80930009;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_CODEC_NOT_REGISTERED = 0x8093000A;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_WRONG_REVISION_FLAG = 0x8093000B;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_FLAG_NOT_SUPPORTED = 0x8093000C;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_BUSY = 0x8093000D;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_BAD_PRIORITY = 0x8093000E;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_IN_PROGRESS = 0x8093000F;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_RETRY = 0x80930010;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_MALFORMED_BATCH = 0x80930011;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_JOB_CREATION = 0x80930012;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_OPCODE = 0x80930013;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_PRIORITY_VIOLATION = 0x80930014;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_BUFFER_TOO_BIG = 0x80930015;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_INVALID_ADDRESS = 0x80930016;
|
||||||
|
constexpr int ORBIS_AJM_ERROR_CANCELLED = 0x80930017;
|
|
@ -1,20 +1,38 @@
|
||||||
// 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
|
||||||
|
|
||||||
// Generated By moduleGenerator
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <common/path_util.h>
|
|
||||||
#include <common/singleton.h>
|
|
||||||
#include <core/file_format/psf.h>
|
|
||||||
#include <core/file_sys/fs.h>
|
|
||||||
#include "app_content.h"
|
#include "app_content.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/singleton.h"
|
||||||
|
#include "common/string_util.h"
|
||||||
|
#include "core/file_format/psf.h"
|
||||||
|
#include "core/file_sys/fs.h"
|
||||||
#include "core/libraries/error_codes.h"
|
#include "core/libraries/error_codes.h"
|
||||||
#include "core/libraries/libs.h"
|
#include "core/libraries/libs.h"
|
||||||
|
|
||||||
namespace Libraries::AppContent {
|
namespace Libraries::AppContent {
|
||||||
|
|
||||||
|
int32_t addcont_count = 0;
|
||||||
|
|
||||||
|
struct AddContInfo {
|
||||||
|
char entitlementLabel[ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE];
|
||||||
|
OrbisAppContentAddcontDownloadStatus status;
|
||||||
|
OrbisAppContentGetEntitlementKey key;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::array<AddContInfo, ORBIS_APP_CONTENT_INFO_LIST_MAX_SIZE> addcont_info = {{
|
||||||
|
{"0000000000000000",
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_INSTALLED,
|
||||||
|
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00}},
|
||||||
|
}};
|
||||||
|
|
||||||
|
std::string title_id;
|
||||||
|
|
||||||
int PS4_SYSV_ABI _Z5dummyv() {
|
int PS4_SYSV_ABI _Z5dummyv() {
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
|
@ -35,9 +53,31 @@ int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadSp() {
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontMount() {
|
int PS4_SYSV_ABI sceAppContentAddcontMount(u32 service_label,
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
const OrbisNpUnifiedEntitlementLabel* entitlement_label,
|
||||||
return ORBIS_OK;
|
OrbisAppContentMountPoint* mount_point) {
|
||||||
|
LOG_INFO(Lib_AppContent, "called");
|
||||||
|
|
||||||
|
const auto& mount_dir = Common::FS::GetUserPath(Common::FS::PathType::AddonsDir) / title_id /
|
||||||
|
entitlement_label->data;
|
||||||
|
auto* mnt = Common::Singleton<Core::FileSys::MntPoints>::Instance();
|
||||||
|
|
||||||
|
for (int i = 0; i < addcont_count; i++) {
|
||||||
|
if (strncmp(entitlement_label->data, addcont_info[i].entitlementLabel,
|
||||||
|
ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE - 1) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (addcont_info[i].status != ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_INSTALLED) {
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_NOT_FOUND;
|
||||||
|
}
|
||||||
|
|
||||||
|
snprintf(mount_point->data, ORBIS_APP_CONTENT_MOUNTPOINT_DATA_MAXSIZE, "/addcont%d", i);
|
||||||
|
mnt->Mount(mount_dir, mount_point->data);
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontShrink() {
|
int PS4_SYSV_ABI sceAppContentAddcontShrink() {
|
||||||
|
@ -124,22 +164,80 @@ int PS4_SYSV_ABI sceAppContentGetAddcontDownloadProgress() {
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfo() {
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfo(u32 service_label,
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
const OrbisNpUnifiedEntitlementLabel* entitlementLabel,
|
||||||
return ORBIS_OK;
|
OrbisAppContentAddcontInfo* info) {
|
||||||
|
LOG_INFO(Lib_AppContent, "called");
|
||||||
|
|
||||||
|
if (entitlementLabel == nullptr || info == nullptr) {
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_PARAMETER;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto i = 0; i < addcont_count; i++) {
|
||||||
|
if (strncmp(entitlementLabel->data, addcont_info[i].entitlementLabel,
|
||||||
|
ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE - 1) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Lib_AppContent, "found DLC {}", entitlementLabel->data);
|
||||||
|
|
||||||
|
strncpy(info->entitlement_label.data, addcont_info[i].entitlementLabel,
|
||||||
|
ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE);
|
||||||
|
info->status = addcont_info[i].status;
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_DRM_NO_ENTITLEMENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfoList(u32 service_label,
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfoList(u32 service_label,
|
||||||
OrbisAppContentAddcontInfo* list, u32 list_num,
|
OrbisAppContentAddcontInfo* list, u32 list_num,
|
||||||
u32* hit_num) {
|
u32* hit_num) {
|
||||||
*hit_num = 0;
|
LOG_INFO(Lib_AppContent, "called");
|
||||||
LOG_ERROR(Lib_AppContent, "(DUMMY) called");
|
|
||||||
|
if (list_num == 0 || list == nullptr) {
|
||||||
|
if (hit_num == nullptr) {
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_PARAMETER;
|
||||||
|
}
|
||||||
|
|
||||||
|
*hit_num = addcont_count;
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int dlcs_to_list = addcont_count < list_num ? addcont_count : list_num;
|
||||||
|
for (int i = 0; i < dlcs_to_list; i++) {
|
||||||
|
strncpy(list[i].entitlement_label.data, addcont_info[i].entitlementLabel,
|
||||||
|
ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE);
|
||||||
|
list[i].status = addcont_info[i].status;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hit_num != nullptr) {
|
||||||
|
*hit_num = dlcs_to_list;
|
||||||
|
}
|
||||||
|
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentGetEntitlementKey() {
|
int PS4_SYSV_ABI sceAppContentGetEntitlementKey(
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
u32 service_label, const OrbisNpUnifiedEntitlementLabel* entitlement_label,
|
||||||
return ORBIS_OK;
|
OrbisAppContentGetEntitlementKey* key) {
|
||||||
|
LOG_ERROR(Lib_AppContent, "called");
|
||||||
|
|
||||||
|
if (entitlement_label == nullptr || key == nullptr) {
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_PARAMETER;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < addcont_count; i++) {
|
||||||
|
if (strncmp(entitlement_label->data, addcont_info[i].entitlementLabel,
|
||||||
|
ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE - 1) != 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(key->data, addcont_info[i].key.data, ORBIS_APP_CONTENT_ENTITLEMENT_KEY_SIZE);
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ORBIS_APP_CONTENT_ERROR_DRM_NO_ENTITLEMENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentGetRegion() {
|
int PS4_SYSV_ABI sceAppContentGetRegion() {
|
||||||
|
@ -150,7 +248,25 @@ int PS4_SYSV_ABI sceAppContentGetRegion() {
|
||||||
int PS4_SYSV_ABI sceAppContentInitialize(const OrbisAppContentInitParam* initParam,
|
int PS4_SYSV_ABI sceAppContentInitialize(const OrbisAppContentInitParam* initParam,
|
||||||
OrbisAppContentBootParam* bootParam) {
|
OrbisAppContentBootParam* bootParam) {
|
||||||
LOG_ERROR(Lib_AppContent, "(DUMMY) called");
|
LOG_ERROR(Lib_AppContent, "(DUMMY) called");
|
||||||
bootParam->attr = 0; // always 0
|
auto* param_sfo = Common::Singleton<PSF>::Instance();
|
||||||
|
|
||||||
|
const auto addons_dir = Common::FS::GetUserPath(Common::FS::PathType::AddonsDir);
|
||||||
|
title_id = param_sfo->GetString("TITLE_ID");
|
||||||
|
auto addon_path = addons_dir / title_id;
|
||||||
|
if (std::filesystem::exists(addon_path)) {
|
||||||
|
for (const auto& entry : std::filesystem::directory_iterator(addon_path)) {
|
||||||
|
if (entry.is_directory()) {
|
||||||
|
auto entitlement_label = entry.path().filename().string();
|
||||||
|
|
||||||
|
AddContInfo info{};
|
||||||
|
info.status = ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_INSTALLED;
|
||||||
|
strcpy(info.entitlementLabel, entitlement_label.c_str());
|
||||||
|
|
||||||
|
addcont_info[addcont_count++] = info;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -221,12 +337,12 @@ int PS4_SYSV_ABI Func_C59A36FF8D7C59DA() {
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadByEntitlemetId() {
|
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadByEntitlementId() {
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontMountByEntitlemetId() {
|
int PS4_SYSV_ABI sceAppContentAddcontMountByEntitlementId() {
|
||||||
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
LOG_ERROR(Lib_AppContent, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
@ -313,9 +429,9 @@ void RegisterlibSceAppContent(Core::Loader::SymbolsResolver* sym) {
|
||||||
LIB_FUNCTION("xZo2-418Wdo", "libSceAppContentBundle", 1, "libSceAppContent", 1, 1,
|
LIB_FUNCTION("xZo2-418Wdo", "libSceAppContentBundle", 1, "libSceAppContent", 1, 1,
|
||||||
Func_C59A36FF8D7C59DA);
|
Func_C59A36FF8D7C59DA);
|
||||||
LIB_FUNCTION("kJmjt81mXKQ", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
LIB_FUNCTION("kJmjt81mXKQ", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
||||||
sceAppContentAddcontEnqueueDownloadByEntitlemetId);
|
sceAppContentAddcontEnqueueDownloadByEntitlementId);
|
||||||
LIB_FUNCTION("efX3lrPwdKA", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
LIB_FUNCTION("efX3lrPwdKA", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
||||||
sceAppContentAddcontMountByEntitlemetId);
|
sceAppContentAddcontMountByEntitlementId);
|
||||||
LIB_FUNCTION("z9hgjLd1SGA", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
LIB_FUNCTION("z9hgjLd1SGA", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
||||||
sceAppContentGetAddcontInfoByEntitlementId);
|
sceAppContentGetAddcontInfoByEntitlementId);
|
||||||
LIB_FUNCTION("3wUaDTGmjcQ", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
LIB_FUNCTION("3wUaDTGmjcQ", "libSceAppContentIro", 1, "libSceAppContent", 1, 1,
|
||||||
|
|
|
@ -41,6 +41,16 @@ struct OrbisAppContentMountPoint {
|
||||||
constexpr int ORBIS_APP_CONTENT_TEMPORARY_DATA_OPTION_NONE = 0;
|
constexpr int ORBIS_APP_CONTENT_TEMPORARY_DATA_OPTION_NONE = 0;
|
||||||
constexpr int ORBIS_APP_CONTENT_TEMPORARY_DATA_OPTION_FORMAT = (1 << 0);
|
constexpr int ORBIS_APP_CONTENT_TEMPORARY_DATA_OPTION_FORMAT = (1 << 0);
|
||||||
constexpr int ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE = 17;
|
constexpr int ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE = 17;
|
||||||
|
constexpr int ORBIS_APP_CONTENT_ENTITLEMENT_KEY_SIZE = 16;
|
||||||
|
constexpr int ORBIS_APP_CONTENT_INFO_LIST_MAX_SIZE = 2500;
|
||||||
|
|
||||||
|
enum OrbisAppContentAddcontDownloadStatus : u32 {
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_NO_EXTRA_DATA = 0,
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_NO_IN_QUEUE = 1,
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_DOWNLOADING = 2,
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_DOWNLOAD_SUSPENDED = 3,
|
||||||
|
ORBIS_APP_CONTENT_ADDCONT_DOWNLOAD_STATUS_INSTALLED = 4
|
||||||
|
};
|
||||||
|
|
||||||
struct OrbisNpUnifiedEntitlementLabel {
|
struct OrbisNpUnifiedEntitlementLabel {
|
||||||
char data[ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE];
|
char data[ORBIS_NP_UNIFIED_ENTITLEMENT_LABEL_SIZE];
|
||||||
|
@ -54,11 +64,17 @@ struct OrbisAppContentAddcontInfo {
|
||||||
u32 status;
|
u32 status;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct OrbisAppContentGetEntitlementKey {
|
||||||
|
char data[ORBIS_APP_CONTENT_ENTITLEMENT_KEY_SIZE];
|
||||||
|
};
|
||||||
|
|
||||||
int PS4_SYSV_ABI _Z5dummyv();
|
int PS4_SYSV_ABI _Z5dummyv();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontDelete();
|
int PS4_SYSV_ABI sceAppContentAddcontDelete();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownload();
|
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownload();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadSp();
|
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadSp();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontMount();
|
int PS4_SYSV_ABI sceAppContentAddcontMount(u32 service_label,
|
||||||
|
const OrbisNpUnifiedEntitlementLabel* entitlement_label,
|
||||||
|
OrbisAppContentMountPoint* mount_point);
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontShrink();
|
int PS4_SYSV_ABI sceAppContentAddcontShrink();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontUnmount();
|
int PS4_SYSV_ABI sceAppContentAddcontUnmount();
|
||||||
int PS4_SYSV_ABI sceAppContentAppParamGetInt(OrbisAppContentAppParamId paramId, s32* value);
|
int PS4_SYSV_ABI sceAppContentAppParamGetInt(OrbisAppContentAppParamId paramId, s32* value);
|
||||||
|
@ -70,11 +86,15 @@ int PS4_SYSV_ABI sceAppContentDownload1Shrink();
|
||||||
int PS4_SYSV_ABI sceAppContentDownloadDataFormat();
|
int PS4_SYSV_ABI sceAppContentDownloadDataFormat();
|
||||||
int PS4_SYSV_ABI sceAppContentDownloadDataGetAvailableSpaceKb();
|
int PS4_SYSV_ABI sceAppContentDownloadDataGetAvailableSpaceKb();
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontDownloadProgress();
|
int PS4_SYSV_ABI sceAppContentGetAddcontDownloadProgress();
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfo();
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfo(u32 service_label,
|
||||||
|
const OrbisNpUnifiedEntitlementLabel* entitlementLabel,
|
||||||
|
OrbisAppContentAddcontInfo* info);
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfoList(u32 service_label,
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfoList(u32 service_label,
|
||||||
OrbisAppContentAddcontInfo* list, u32 list_num,
|
OrbisAppContentAddcontInfo* list, u32 list_num,
|
||||||
u32* hit_num);
|
u32* hit_num);
|
||||||
int PS4_SYSV_ABI sceAppContentGetEntitlementKey();
|
int PS4_SYSV_ABI sceAppContentGetEntitlementKey(
|
||||||
|
u32 service_label, const OrbisNpUnifiedEntitlementLabel* entitlement_label,
|
||||||
|
OrbisAppContentGetEntitlementKey* key);
|
||||||
int PS4_SYSV_ABI sceAppContentGetRegion();
|
int PS4_SYSV_ABI sceAppContentGetRegion();
|
||||||
int PS4_SYSV_ABI sceAppContentInitialize(const OrbisAppContentInitParam* initParam,
|
int PS4_SYSV_ABI sceAppContentInitialize(const OrbisAppContentInitParam* initParam,
|
||||||
OrbisAppContentBootParam* bootParam);
|
OrbisAppContentBootParam* bootParam);
|
||||||
|
@ -92,8 +112,8 @@ int PS4_SYSV_ABI sceAppContentTemporaryDataMount2(OrbisAppContentTemporaryDataOp
|
||||||
int PS4_SYSV_ABI sceAppContentTemporaryDataUnmount();
|
int PS4_SYSV_ABI sceAppContentTemporaryDataUnmount();
|
||||||
int PS4_SYSV_ABI sceAppContentGetPftFlag();
|
int PS4_SYSV_ABI sceAppContentGetPftFlag();
|
||||||
int PS4_SYSV_ABI Func_C59A36FF8D7C59DA();
|
int PS4_SYSV_ABI Func_C59A36FF8D7C59DA();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadByEntitlemetId();
|
int PS4_SYSV_ABI sceAppContentAddcontEnqueueDownloadByEntitlementId();
|
||||||
int PS4_SYSV_ABI sceAppContentAddcontMountByEntitlemetId();
|
int PS4_SYSV_ABI sceAppContentAddcontMountByEntitlementId();
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfoByEntitlementId();
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfoByEntitlementId();
|
||||||
int PS4_SYSV_ABI sceAppContentGetAddcontInfoListByIroTag();
|
int PS4_SYSV_ABI sceAppContentGetAddcontInfoListByIroTag();
|
||||||
int PS4_SYSV_ABI sceAppContentGetDownloadedStoreCountry();
|
int PS4_SYSV_ABI sceAppContentGetDownloadedStoreCountry();
|
||||||
|
|
|
@ -2,9 +2,10 @@
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <common/assert.h>
|
|
||||||
#include <magic_enum.hpp>
|
#include <magic_enum.hpp>
|
||||||
|
|
||||||
#include "audio_core/sdl_audio.h"
|
#include "audio_core/sdl_audio.h"
|
||||||
|
#include "common/assert.h"
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
#include "core/libraries/audio/audioout.h"
|
#include "core/libraries/audio/audioout.h"
|
||||||
#include "core/libraries/error_codes.h"
|
#include "core/libraries/error_codes.h"
|
||||||
|
@ -175,7 +176,6 @@ int PS4_SYSV_ABI sceAudioOutGetLastOutputTime() {
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAudioOutGetPortState(s32 handle, OrbisAudioOutPortState* state) {
|
int PS4_SYSV_ABI sceAudioOutGetPortState(s32 handle, OrbisAudioOutPortState* state) {
|
||||||
|
|
||||||
int type = 0;
|
int type = 0;
|
||||||
int channels_num = 0;
|
int channels_num = 0;
|
||||||
|
|
||||||
|
@ -235,8 +235,11 @@ int PS4_SYSV_ABI sceAudioOutGetSystemState() {
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAudioOutInit() {
|
int PS4_SYSV_ABI sceAudioOutInit() {
|
||||||
|
LOG_TRACE(Lib_AudioOut, "called");
|
||||||
|
if (audio != nullptr) {
|
||||||
|
return ORBIS_AUDIO_OUT_ERROR_ALREADY_INIT;
|
||||||
|
}
|
||||||
audio = std::make_unique<Audio::SDLAudio>();
|
audio = std::make_unique<Audio::SDLAudio>();
|
||||||
LOG_INFO(Lib_AudioOut, "called");
|
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,21 +1,34 @@
|
||||||
// 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
|
||||||
|
|
||||||
// Generated By moduleGenerator
|
|
||||||
#include "avplayer.h"
|
#include "avplayer.h"
|
||||||
|
|
||||||
|
#include "avplayer_impl.h"
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
#include "core/libraries/error_codes.h"
|
#include "core/libraries/error_codes.h"
|
||||||
|
#include "core/libraries/kernel/thread_management.h"
|
||||||
#include "core/libraries/libs.h"
|
#include "core/libraries/libs.h"
|
||||||
|
|
||||||
namespace Libraries::AvPlayer {
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerAddSource() {
|
using namespace Kernel;
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
s32 PS4_SYSV_ABI sceAvPlayerAddSource(SceAvPlayerHandle handle, const char* filename) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "filename = {}", filename);
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->AddSource(filename);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerAddSourceEx() {
|
s32 PS4_SYSV_ABI sceAvPlayerAddSourceEx(SceAvPlayerHandle handle, SceAvPlayerUriType uriType,
|
||||||
|
SceAvPlayerSourceDetails* sourceDetails) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,122 +37,247 @@ int PS4_SYSV_ABI sceAvPlayerChangeStream() {
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerClose() {
|
s32 PS4_SYSV_ABI sceAvPlayerClose(SceAvPlayerHandle handle) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning ORBIS_AVPLAYER_ERROR_INVALID_PARAMS");
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
delete handle;
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning ORBIS_OK");
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 PS4_SYSV_ABI sceAvPlayerCurrentTime(SceAvPlayerHandle handle) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->CurrentTime();
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerDisableStream(SceAvPlayerHandle handle, u32 stream_id) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerEnableStream(SceAvPlayerHandle handle, u32 stream_id) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "stream_id = {}", stream_id);
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->EnableStream(stream_id);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PS4_SYSV_ABI sceAvPlayerGetAudioData(SceAvPlayerHandle handle, SceAvPlayerFrameInfo* p_info) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr || p_info == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->GetAudioData(*p_info);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerGetStreamInfo(SceAvPlayerHandle handle, u32 stream_id,
|
||||||
|
SceAvPlayerStreamInfo* p_info) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "stream_id = {}", stream_id);
|
||||||
|
if (handle == nullptr || p_info == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->GetStreamInfo(stream_id, *p_info);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PS4_SYSV_ABI sceAvPlayerGetVideoData(SceAvPlayerHandle handle,
|
||||||
|
SceAvPlayerFrameInfo* video_info) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr || video_info == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->GetVideoData(*video_info);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PS4_SYSV_ABI sceAvPlayerGetVideoDataEx(SceAvPlayerHandle handle,
|
||||||
|
SceAvPlayerFrameInfoEx* video_info) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr || video_info == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->GetVideoData(*video_info);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
SceAvPlayerHandle PS4_SYSV_ABI sceAvPlayerInit(SceAvPlayerInitData* data) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (data == nullptr) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data->memory_replacement.allocate == nullptr ||
|
||||||
|
data->memory_replacement.allocate_texture == nullptr ||
|
||||||
|
data->memory_replacement.deallocate == nullptr ||
|
||||||
|
data->memory_replacement.deallocate_texture == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "All allocators are required for AVPlayer Initialisation.");
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new AvPlayer(*data);
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerInitEx(const SceAvPlayerInitDataEx* p_data,
|
||||||
|
SceAvPlayerHandle* p_player) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (p_data == nullptr || p_player == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p_data->memory_replacement.allocate == nullptr ||
|
||||||
|
p_data->memory_replacement.allocate_texture == nullptr ||
|
||||||
|
p_data->memory_replacement.deallocate == nullptr ||
|
||||||
|
p_data->memory_replacement.deallocate_texture == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "All allocators are required for AVPlayer Initialisation.");
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
|
||||||
|
SceAvPlayerInitData data = {};
|
||||||
|
data.memory_replacement = p_data->memory_replacement;
|
||||||
|
data.file_replacement = p_data->file_replacement;
|
||||||
|
data.event_replacement = p_data->event_replacement;
|
||||||
|
data.default_language = p_data->default_language;
|
||||||
|
data.num_output_video_framebuffers = p_data->num_output_video_framebuffers;
|
||||||
|
data.auto_start = p_data->auto_start;
|
||||||
|
|
||||||
|
*p_player = new AvPlayer(data);
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool PS4_SYSV_ABI sceAvPlayerIsActive(SceAvPlayerHandle handle) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning ORBIS_AVPLAYER_ERROR_INVALID_PARAMS");
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->IsActive();
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerJumpToTime(SceAvPlayerHandle handle, uint64_t time) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called, time (msec) = {}", time);
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerPause(SceAvPlayerHandle handle) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerPostInit(SceAvPlayerHandle handle, SceAvPlayerPostInitData* data) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
|
if (handle == nullptr || data == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->PostInit(*data);
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerPrintf(const char* format, ...) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerCurrentTime() {
|
s32 PS4_SYSV_ABI sceAvPlayerResume(SceAvPlayerHandle handle) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerSetAvSyncMode(SceAvPlayerHandle handle,
|
||||||
|
SceAvPlayerAvSyncMode sync_mode) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 PS4_SYSV_ABI sceAvPlayerSetLogCallback(SceAvPlayerLogCallback log_cb, void* user_data) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerDisableStream() {
|
s32 PS4_SYSV_ABI sceAvPlayerSetLooping(SceAvPlayerHandle handle, bool loop_flag) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_TRACE(Lib_AvPlayer, "called, looping = {}", loop_flag);
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
if (!handle->SetLooping(loop_flag)) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerEnableStream() {
|
s32 PS4_SYSV_ABI sceAvPlayerSetTrickSpeed(SceAvPlayerHandle handle, s32 trick_speed) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetAudioData() {
|
s32 PS4_SYSV_ABI sceAvPlayerStart(SceAvPlayerHandle handle) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
return ORBIS_OK;
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->Start();
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetStreamInfo() {
|
s32 PS4_SYSV_ABI sceAvPlayerStop(SceAvPlayerHandle handle) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
return ORBIS_OK;
|
if (handle == nullptr) {
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning ORBIS_AVPLAYER_ERROR_INVALID_PARAMS");
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->Stop();
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetVideoData() {
|
s32 PS4_SYSV_ABI sceAvPlayerStreamCount(SceAvPlayerHandle handle) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_TRACE(Lib_AvPlayer, "called");
|
||||||
return ORBIS_OK;
|
if (handle == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
const auto res = handle->GetStreamCount();
|
||||||
|
LOG_TRACE(Lib_AvPlayer, "returning {}", res);
|
||||||
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetVideoDataEx() {
|
s32 PS4_SYSV_ABI sceAvPlayerVprintf(const char* format, va_list args) {
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerInit() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerInitEx() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerIsActive() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerJumpToTime() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerPause() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerPostInit() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerPrintf() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerResume() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetAvSyncMode() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetLogCallback() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetLooping() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetTrickSpeed() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerStart() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerStop() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerStreamCount() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
|
||||||
return ORBIS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerVprintf() {
|
|
||||||
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
LOG_ERROR(Lib_AvPlayer, "(STUBBED) called");
|
||||||
return ORBIS_OK;
|
return ORBIS_OK;
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,39 +5,288 @@
|
||||||
|
|
||||||
#include "common/types.h"
|
#include "common/types.h"
|
||||||
|
|
||||||
|
#include <stdarg.h> // va_list
|
||||||
|
#include <stddef.h> // size_t
|
||||||
|
|
||||||
namespace Core::Loader {
|
namespace Core::Loader {
|
||||||
class SymbolsResolver;
|
class SymbolsResolver;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace Libraries::AvPlayer {
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerAddSource();
|
class AvPlayer;
|
||||||
int PS4_SYSV_ABI sceAvPlayerAddSourceEx();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerChangeStream();
|
using SceAvPlayerHandle = AvPlayer*;
|
||||||
int PS4_SYSV_ABI sceAvPlayerClose();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerCurrentTime();
|
enum SceAvPlayerUriType { SCE_AVPLAYER_URI_TYPE_SOURCE = 0 };
|
||||||
int PS4_SYSV_ABI sceAvPlayerDisableStream();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerEnableStream();
|
struct SceAvPlayerUri {
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetAudioData();
|
const char* name;
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetStreamInfo();
|
u32 length;
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetVideoData();
|
};
|
||||||
int PS4_SYSV_ABI sceAvPlayerGetVideoDataEx();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerInit();
|
enum SceAvPlayerSourceType {
|
||||||
int PS4_SYSV_ABI sceAvPlayerInitEx();
|
SCE_AVPLAYER_SOURCE_TYPE_UNKNOWN = 0,
|
||||||
int PS4_SYSV_ABI sceAvPlayerIsActive();
|
SCE_AVPLAYER_SOURCE_TYPE_FILE_MP4 = 1,
|
||||||
int PS4_SYSV_ABI sceAvPlayerJumpToTime();
|
SCE_AVPLAYER_SOURCE_TYPE_HLS = 8
|
||||||
int PS4_SYSV_ABI sceAvPlayerPause();
|
};
|
||||||
int PS4_SYSV_ABI sceAvPlayerPostInit();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerPrintf();
|
struct SceAvPlayerSourceDetails {
|
||||||
int PS4_SYSV_ABI sceAvPlayerResume();
|
SceAvPlayerUri uri;
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetAvSyncMode();
|
u8 reserved1[64];
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetLogCallback();
|
SceAvPlayerSourceType source_type;
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetLooping();
|
u8 reserved2[44];
|
||||||
int PS4_SYSV_ABI sceAvPlayerSetTrickSpeed();
|
};
|
||||||
int PS4_SYSV_ABI sceAvPlayerStart();
|
|
||||||
int PS4_SYSV_ABI sceAvPlayerStop();
|
struct SceAvPlayerAudio {
|
||||||
int PS4_SYSV_ABI sceAvPlayerStreamCount();
|
u16 channel_count;
|
||||||
int PS4_SYSV_ABI sceAvPlayerVprintf();
|
u8 reserved1[2];
|
||||||
|
u32 sample_rate;
|
||||||
|
u32 size;
|
||||||
|
u8 language_code[4];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerVideo {
|
||||||
|
u32 width;
|
||||||
|
u32 height;
|
||||||
|
f32 aspect_ratio;
|
||||||
|
u8 language_code[4];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerTextPosition {
|
||||||
|
u16 top;
|
||||||
|
u16 left;
|
||||||
|
u16 bottom;
|
||||||
|
u16 right;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerTimedText {
|
||||||
|
u8 language_code[4];
|
||||||
|
u16 text_size;
|
||||||
|
u16 font_size;
|
||||||
|
SceAvPlayerTextPosition position;
|
||||||
|
};
|
||||||
|
|
||||||
|
union SceAvPlayerStreamDetails {
|
||||||
|
u8 reserved[16];
|
||||||
|
SceAvPlayerAudio audio;
|
||||||
|
SceAvPlayerVideo video;
|
||||||
|
SceAvPlayerTimedText subs;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerFrameInfo {
|
||||||
|
u8* pData;
|
||||||
|
u8 reserved[4];
|
||||||
|
u64 timestamp;
|
||||||
|
SceAvPlayerStreamDetails details;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerStreamInfo {
|
||||||
|
u32 type;
|
||||||
|
u8 reserved[4];
|
||||||
|
SceAvPlayerStreamDetails details;
|
||||||
|
u64 duration;
|
||||||
|
u64 start_time;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerAudioEx {
|
||||||
|
u16 channel_count;
|
||||||
|
u8 reserved[2];
|
||||||
|
u32 sample_rate;
|
||||||
|
u32 size;
|
||||||
|
u8 language_code[4];
|
||||||
|
u8 reserved1[64];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerVideoEx {
|
||||||
|
u32 width;
|
||||||
|
u32 height;
|
||||||
|
f32 aspect_ratio;
|
||||||
|
u8 language_code[4];
|
||||||
|
u32 framerate;
|
||||||
|
u32 crop_left_offset;
|
||||||
|
u32 crop_right_offset;
|
||||||
|
u32 crop_top_offset;
|
||||||
|
u32 crop_bottom_offset;
|
||||||
|
u32 pitch;
|
||||||
|
u8 luma_bit_depth;
|
||||||
|
u8 chroma_bit_depth;
|
||||||
|
bool video_full_range_flag;
|
||||||
|
u8 reserved1[37];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerTimedTextEx {
|
||||||
|
u8 language_code[4];
|
||||||
|
u8 reserved[12];
|
||||||
|
u8 reserved1[64];
|
||||||
|
};
|
||||||
|
|
||||||
|
union SceAvPlayerStreamDetailsEx {
|
||||||
|
SceAvPlayerAudioEx audio;
|
||||||
|
SceAvPlayerVideoEx video;
|
||||||
|
SceAvPlayerTimedTextEx subs;
|
||||||
|
u8 reserved1[80];
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerFrameInfoEx {
|
||||||
|
void* pData;
|
||||||
|
u8 reserved[4];
|
||||||
|
u64 timestamp;
|
||||||
|
SceAvPlayerStreamDetailsEx details;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef void* PS4_SYSV_ABI (*SceAvPlayerAllocate)(void* p, u32 align, u32 size);
|
||||||
|
typedef void PS4_SYSV_ABI (*SceAvPlayerDeallocate)(void* p, void* mem);
|
||||||
|
typedef void* PS4_SYSV_ABI (*SceAvPlayerAllocateTexture)(void* p, u32 align, u32 size);
|
||||||
|
typedef void PS4_SYSV_ABI (*SceAvPlayerDeallocateTexture)(void* p, void* mem);
|
||||||
|
|
||||||
|
struct SceAvPlayerMemAllocator {
|
||||||
|
void* object_ptr;
|
||||||
|
SceAvPlayerAllocate allocate;
|
||||||
|
SceAvPlayerDeallocate deallocate;
|
||||||
|
SceAvPlayerAllocateTexture allocate_texture;
|
||||||
|
SceAvPlayerDeallocateTexture deallocate_texture;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef s32 PS4_SYSV_ABI (*SceAvPlayerOpenFile)(void* p, const char* name);
|
||||||
|
typedef s32 PS4_SYSV_ABI (*SceAvPlayerCloseFile)(void* p);
|
||||||
|
typedef s32 PS4_SYSV_ABI (*SceAvPlayerReadOffsetFile)(void* p, u8* buf, u64 pos, u32 len);
|
||||||
|
typedef u64 PS4_SYSV_ABI (*SceAvPlayerSizeFile)(void* p);
|
||||||
|
|
||||||
|
struct SceAvPlayerFileReplacement {
|
||||||
|
void* object_ptr;
|
||||||
|
SceAvPlayerOpenFile open;
|
||||||
|
SceAvPlayerCloseFile close;
|
||||||
|
SceAvPlayerReadOffsetFile readOffset;
|
||||||
|
SceAvPlayerSizeFile size;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef void PS4_SYSV_ABI (*SceAvPlayerEventCallback)(void* p, s32 event, s32 src_id, void* data);
|
||||||
|
|
||||||
|
struct SceAvPlayerEventReplacement {
|
||||||
|
void* object_ptr;
|
||||||
|
SceAvPlayerEventCallback event_callback;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SceAvPlayerDebuglevels {
|
||||||
|
SCE_AVPLAYER_DBG_NONE,
|
||||||
|
SCE_AVPLAYER_DBG_INFO,
|
||||||
|
SCE_AVPLAYER_DBG_WARNINGS,
|
||||||
|
SCE_AVPLAYER_DBG_ALL
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerInitData {
|
||||||
|
SceAvPlayerMemAllocator memory_replacement;
|
||||||
|
SceAvPlayerFileReplacement file_replacement;
|
||||||
|
SceAvPlayerEventReplacement event_replacement;
|
||||||
|
SceAvPlayerDebuglevels debug_level;
|
||||||
|
u32 base_priority;
|
||||||
|
s32 num_output_video_framebuffers;
|
||||||
|
bool auto_start;
|
||||||
|
u8 reserved[3];
|
||||||
|
const char* default_language;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerInitDataEx {
|
||||||
|
size_t this_size;
|
||||||
|
SceAvPlayerMemAllocator memory_replacement;
|
||||||
|
SceAvPlayerFileReplacement file_replacement;
|
||||||
|
SceAvPlayerEventReplacement event_replacement;
|
||||||
|
const char* default_language;
|
||||||
|
SceAvPlayerDebuglevels debug_level;
|
||||||
|
u32 audio_decoder_priority;
|
||||||
|
u32 audio_decoder_affinity;
|
||||||
|
u32 video_decoder_priority;
|
||||||
|
u32 video_decoder_affinity;
|
||||||
|
u32 demuxer_priority;
|
||||||
|
u32 demuxer_affinity;
|
||||||
|
u32 controller_priority;
|
||||||
|
u32 controller_affinity;
|
||||||
|
u32 http_streaming_priority;
|
||||||
|
u32 http_streaming_affinity;
|
||||||
|
u32 file_streaming_priority;
|
||||||
|
u32 file_streaming_affinity;
|
||||||
|
s32 num_output_video_framebuffers;
|
||||||
|
bool auto_start;
|
||||||
|
u8 reserved[3];
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SceAvPlayerStreamType {
|
||||||
|
SCE_AVPLAYER_VIDEO,
|
||||||
|
SCE_AVPLAYER_AUDIO,
|
||||||
|
SCE_AVPLAYER_TIMEDTEXT,
|
||||||
|
SCE_AVPLAYER_UNKNOWN
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SceAvPlayerVideoDecoderType {
|
||||||
|
SCE_AVPLAYER_VIDEO_DECODER_TYPE_DEFAULT = 0,
|
||||||
|
SCE_AVPLAYER_VIDEO_DECODER_TYPE_RESERVED1,
|
||||||
|
SCE_AVPLAYER_VIDEO_DECODER_TYPE_SOFTWARE,
|
||||||
|
SCE_AVPLAYER_VIDEO_DECODER_TYPE_SOFTWARE2
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SceAvPlayerAudioDecoderType {
|
||||||
|
SCE_AVPLAYER_AUDIO_DECODER_TYPE_DEFAULT = 0,
|
||||||
|
SCE_AVPLAYER_AUDIO_DECODER_TYPE_RESERVED1,
|
||||||
|
SCE_AVPLAYER_AUDIO_DECODER_TYPE_RESERVED2
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerDecoderInit {
|
||||||
|
union {
|
||||||
|
SceAvPlayerVideoDecoderType video_type;
|
||||||
|
SceAvPlayerAudioDecoderType audio_type;
|
||||||
|
u8 reserved[4];
|
||||||
|
} decoderType;
|
||||||
|
union {
|
||||||
|
struct {
|
||||||
|
s32 cpu_affinity_mask;
|
||||||
|
s32 cpu_thread_priority;
|
||||||
|
u8 decode_pipeline_depth;
|
||||||
|
u8 compute_pipe_id;
|
||||||
|
u8 compute_queue_id;
|
||||||
|
u8 enable_interlaced;
|
||||||
|
u8 reserved[16];
|
||||||
|
} avcSw2;
|
||||||
|
struct {
|
||||||
|
u8 audio_channel_order;
|
||||||
|
u8 reserved[27];
|
||||||
|
} aac;
|
||||||
|
u8 reserved[28];
|
||||||
|
} decoderParams;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerHTTPCtx {
|
||||||
|
u32 http_context_id;
|
||||||
|
u32 ssl_context_id;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SceAvPlayerPostInitData {
|
||||||
|
u32 demux_video_buffer_size;
|
||||||
|
SceAvPlayerDecoderInit video_decoder_init;
|
||||||
|
SceAvPlayerDecoderInit audio_decoder_init;
|
||||||
|
SceAvPlayerHTTPCtx http_context;
|
||||||
|
u8 reserved[56];
|
||||||
|
};
|
||||||
|
|
||||||
|
enum SceAvPlayerAvSyncMode {
|
||||||
|
SCE_AVPLAYER_AV_SYNC_MODE_DEFAULT = 0,
|
||||||
|
SCE_AVPLAYER_AV_SYNC_MODE_NONE
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef int PS4_SYSV_ABI (*SceAvPlayerLogCallback)(void* p, const char* format, va_list args);
|
||||||
|
|
||||||
|
enum SceAvPlayerEvents {
|
||||||
|
SCE_AVPLAYER_STATE_STOP = 0x01,
|
||||||
|
SCE_AVPLAYER_STATE_READY = 0x02,
|
||||||
|
SCE_AVPLAYER_STATE_PLAY = 0x03,
|
||||||
|
SCE_AVPLAYER_STATE_PAUSE = 0x04,
|
||||||
|
SCE_AVPLAYER_STATE_BUFFERING = 0x05,
|
||||||
|
SCE_AVPLAYER_TIMED_TEXT_DELIVERY = 0x10,
|
||||||
|
SCE_AVPLAYER_WARNING_ID = 0x20,
|
||||||
|
SCE_AVPLAYER_ENCRYPTION = 0x30,
|
||||||
|
SCE_AVPLAYER_DRM_ERROR = 0x40
|
||||||
|
};
|
||||||
|
|
||||||
void RegisterlibSceAvPlayer(Core::Loader::SymbolsResolver* sym);
|
void RegisterlibSceAvPlayer(Core::Loader::SymbolsResolver* sym);
|
||||||
|
|
||||||
} // namespace Libraries::AvPlayer
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,61 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
#include "avplayer_common.h"
|
||||||
|
|
||||||
|
#include <algorithm> // std::equal
|
||||||
|
#include <cctype> // std::tolower
|
||||||
|
#include <string_view> // std::string_view
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
using namespace Kernel;
|
||||||
|
|
||||||
|
static bool ichar_equals(char a, char b) {
|
||||||
|
return std::tolower(static_cast<unsigned char>(a)) ==
|
||||||
|
std::tolower(static_cast<unsigned char>(b));
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool iequals(std::string_view l, std::string_view r) {
|
||||||
|
return std::ranges::equal(l, r, ichar_equals);
|
||||||
|
}
|
||||||
|
|
||||||
|
SceAvPlayerSourceType GetSourceType(std::string_view path) {
|
||||||
|
if (path.empty()) {
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string_view name = path;
|
||||||
|
if (path.find("://") != std::string_view::npos) {
|
||||||
|
// This path is a URI. Strip HTTP parameters from it.
|
||||||
|
// schema://server.domain/path/file.ext/and/beyond?param=value#paragraph ->
|
||||||
|
// -> schema://server.domain/path/to/file.ext/and/beyond
|
||||||
|
name = path.substr(0, path.find_first_of("?#"));
|
||||||
|
if (name.empty()) {
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// schema://server.domain/path/to/file.ext/and/beyond -> .ext/and/beyond
|
||||||
|
auto ext = name.substr(name.rfind('.'));
|
||||||
|
if (ext.empty()) {
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
// .ext/and/beyond -> .ext
|
||||||
|
ext = ext.substr(0, ext.find('/'));
|
||||||
|
|
||||||
|
if (iequals(ext, ".mp4") || iequals(ext, ".m4v") || iequals(ext, ".m3d") ||
|
||||||
|
iequals(ext, ".m4a") || iequals(ext, ".mov")) {
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_FILE_MP4;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iequals(ext, ".m3u8")) {
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_HLS;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SCE_AVPLAYER_SOURCE_TYPE_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,91 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
|
||||||
|
#include "common/assert.h"
|
||||||
|
#include "common/logging/log.h"
|
||||||
|
#include "core/libraries/kernel/thread_management.h"
|
||||||
|
|
||||||
|
#include <optional>
|
||||||
|
#include <utility>
|
||||||
|
#include <queue>
|
||||||
|
|
||||||
|
#define AVPLAYER_IS_ERROR(x) ((x) < 0)
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
enum class AvState {
|
||||||
|
Initial,
|
||||||
|
AddingSource,
|
||||||
|
Ready,
|
||||||
|
Play,
|
||||||
|
Stop,
|
||||||
|
EndOfFile,
|
||||||
|
Pause,
|
||||||
|
C0x08,
|
||||||
|
Jump,
|
||||||
|
TrickMode,
|
||||||
|
C0x0B,
|
||||||
|
Buffering,
|
||||||
|
Starting,
|
||||||
|
Error,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class AvEventType {
|
||||||
|
ChangeFlowState = 21,
|
||||||
|
WarningId = 22,
|
||||||
|
RevertState = 30,
|
||||||
|
AddSource = 40,
|
||||||
|
Error = 255,
|
||||||
|
};
|
||||||
|
|
||||||
|
union AvPlayerEventData {
|
||||||
|
u32 num_frames; // 20
|
||||||
|
AvState state; // AvEventType::ChangeFlowState
|
||||||
|
s32 error; // AvEventType::WarningId
|
||||||
|
u32 attempt; // AvEventType::AddSource
|
||||||
|
};
|
||||||
|
|
||||||
|
struct AvPlayerEvent {
|
||||||
|
AvEventType event;
|
||||||
|
AvPlayerEventData payload;
|
||||||
|
};
|
||||||
|
|
||||||
|
template <class T>
|
||||||
|
class AvPlayerQueue {
|
||||||
|
public:
|
||||||
|
size_t Size() {
|
||||||
|
return m_queue.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Push(T&& value) {
|
||||||
|
std::lock_guard guard(m_mutex);
|
||||||
|
m_queue.emplace(std::forward<T>(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<T> Pop() {
|
||||||
|
if (Size() == 0) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
std::lock_guard guard(m_mutex);
|
||||||
|
auto result = std::move(m_queue.front());
|
||||||
|
m_queue.pop();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Clear() {
|
||||||
|
std::lock_guard guard(m_mutex);
|
||||||
|
m_queue = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::mutex m_mutex{};
|
||||||
|
std::queue<T> m_queue{};
|
||||||
|
};
|
||||||
|
|
||||||
|
SceAvPlayerSourceType GetSourceType(std::string_view path);
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,23 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
|
||||||
|
#include "common/types.h"
|
||||||
|
|
||||||
|
#include <string_view>
|
||||||
|
|
||||||
|
struct AVIOContext;
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
class IDataStreamer {
|
||||||
|
public:
|
||||||
|
virtual ~IDataStreamer() = default;
|
||||||
|
virtual bool Init(std::string_view path) = 0;
|
||||||
|
virtual AVIOContext* GetContext() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,90 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "avplayer_file_streamer.h"
|
||||||
|
|
||||||
|
#include "avplayer_common.h"
|
||||||
|
|
||||||
|
#include <magic_enum.hpp>
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include <libavformat/avformat.h>
|
||||||
|
#include <libavformat/avio.h>
|
||||||
|
}
|
||||||
|
|
||||||
|
#include <algorithm> // std::max, std::min
|
||||||
|
|
||||||
|
#define AVPLAYER_AVIO_BUFFER_SIZE 4096
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
AvPlayerFileStreamer::AvPlayerFileStreamer(const SceAvPlayerFileReplacement& file_replacement)
|
||||||
|
: m_file_replacement(file_replacement) {}
|
||||||
|
|
||||||
|
AvPlayerFileStreamer::~AvPlayerFileStreamer() {
|
||||||
|
if (m_avio_context != nullptr) {
|
||||||
|
avio_context_free(&m_avio_context);
|
||||||
|
}
|
||||||
|
if (m_file_replacement.close != nullptr && m_fd >= 0) {
|
||||||
|
const auto close = m_file_replacement.close;
|
||||||
|
const auto ptr = m_file_replacement.object_ptr;
|
||||||
|
close(ptr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerFileStreamer::Init(std::string_view path) {
|
||||||
|
const auto ptr = m_file_replacement.object_ptr;
|
||||||
|
m_fd = m_file_replacement.open(ptr, path.data());
|
||||||
|
if (m_fd < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
m_file_size = m_file_replacement.size(ptr);
|
||||||
|
// avio_buffer is deallocated in `avio_context_free`
|
||||||
|
const auto avio_buffer = reinterpret_cast<u8*>(av_malloc(AVPLAYER_AVIO_BUFFER_SIZE));
|
||||||
|
m_avio_context =
|
||||||
|
avio_alloc_context(avio_buffer, AVPLAYER_AVIO_BUFFER_SIZE, 0, this,
|
||||||
|
&AvPlayerFileStreamer::ReadPacket, nullptr, &AvPlayerFileStreamer::Seek);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayerFileStreamer::ReadPacket(void* opaque, u8* buffer, s32 size) {
|
||||||
|
const auto self = reinterpret_cast<AvPlayerFileStreamer*>(opaque);
|
||||||
|
if (self->m_position >= self->m_file_size) {
|
||||||
|
return AVERROR_EOF;
|
||||||
|
}
|
||||||
|
if (self->m_position + size > self->m_file_size) {
|
||||||
|
size = self->m_file_size - self->m_position;
|
||||||
|
}
|
||||||
|
const auto read_offset = self->m_file_replacement.readOffset;
|
||||||
|
const auto ptr = self->m_file_replacement.object_ptr;
|
||||||
|
const auto bytes_read = read_offset(ptr, buffer, self->m_position, size);
|
||||||
|
if (bytes_read == 0 && size != 0) {
|
||||||
|
return AVERROR_EOF;
|
||||||
|
}
|
||||||
|
self->m_position += bytes_read;
|
||||||
|
return bytes_read;
|
||||||
|
}
|
||||||
|
|
||||||
|
s64 AvPlayerFileStreamer::Seek(void* opaque, s64 offset, int whence) {
|
||||||
|
const auto self = reinterpret_cast<AvPlayerFileStreamer*>(opaque);
|
||||||
|
if (whence & AVSEEK_SIZE) {
|
||||||
|
return self->m_file_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (whence == SEEK_CUR) {
|
||||||
|
self->m_position =
|
||||||
|
std::min(u64(std::max(s64(0), s64(self->m_position) + offset)), self->m_file_size);
|
||||||
|
return self->m_position;
|
||||||
|
} else if (whence == SEEK_SET) {
|
||||||
|
self->m_position = std::min(u64(std::max(s64(0), offset)), self->m_file_size);
|
||||||
|
return self->m_position;
|
||||||
|
} else if (whence == SEEK_END) {
|
||||||
|
self->m_position =
|
||||||
|
std::min(u64(std::max(s64(0), s64(self->m_file_size) + offset)), self->m_file_size);
|
||||||
|
return self->m_position;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,39 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
#include "avplayer_data_streamer.h"
|
||||||
|
|
||||||
|
#include <string_view>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
struct AVIOContext;
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
class AvPlayerFileStreamer : public IDataStreamer {
|
||||||
|
public:
|
||||||
|
AvPlayerFileStreamer(const SceAvPlayerFileReplacement& file_replacement);
|
||||||
|
~AvPlayerFileStreamer();
|
||||||
|
|
||||||
|
bool Init(std::string_view path) override;
|
||||||
|
|
||||||
|
AVIOContext* GetContext() override {
|
||||||
|
return m_avio_context;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static s32 ReadPacket(void* opaque, u8* buffer, s32 size);
|
||||||
|
static s64 Seek(void* opaque, s64 buffer, int whence);
|
||||||
|
|
||||||
|
SceAvPlayerFileReplacement m_file_replacement;
|
||||||
|
|
||||||
|
int m_fd = -1;
|
||||||
|
u64 m_position{};
|
||||||
|
u64 m_file_size{};
|
||||||
|
AVIOContext* m_avio_context{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,203 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "avplayer_common.h"
|
||||||
|
#include "avplayer_file_streamer.h"
|
||||||
|
#include "avplayer_impl.h"
|
||||||
|
|
||||||
|
#include "common/logging/log.h"
|
||||||
|
#include "core/libraries/error_codes.h"
|
||||||
|
#include "core/libraries/kernel/libkernel.h"
|
||||||
|
|
||||||
|
using namespace Libraries::Kernel;
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
void* PS4_SYSV_ABI AvPlayer::Allocate(void* handle, u32 alignment, u32 size) {
|
||||||
|
const auto* const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
const auto allocate = self->m_init_data_original.memory_replacement.allocate;
|
||||||
|
const auto ptr = self->m_init_data_original.memory_replacement.object_ptr;
|
||||||
|
return allocate(ptr, alignment, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PS4_SYSV_ABI AvPlayer::Deallocate(void* handle, void* memory) {
|
||||||
|
const auto* const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
const auto deallocate = self->m_init_data_original.memory_replacement.deallocate;
|
||||||
|
const auto ptr = self->m_init_data_original.memory_replacement.object_ptr;
|
||||||
|
return deallocate(ptr, memory);
|
||||||
|
}
|
||||||
|
|
||||||
|
void* PS4_SYSV_ABI AvPlayer::AllocateTexture(void* handle, u32 alignment, u32 size) {
|
||||||
|
const auto* const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
const auto allocate = self->m_init_data_original.memory_replacement.allocate_texture;
|
||||||
|
const auto ptr = self->m_init_data_original.memory_replacement.object_ptr;
|
||||||
|
return allocate(ptr, alignment, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PS4_SYSV_ABI AvPlayer::DeallocateTexture(void* handle, void* memory) {
|
||||||
|
const auto* const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
const auto deallocate = self->m_init_data_original.memory_replacement.deallocate_texture;
|
||||||
|
const auto ptr = self->m_init_data_original.memory_replacement.object_ptr;
|
||||||
|
return deallocate(ptr, memory);
|
||||||
|
}
|
||||||
|
|
||||||
|
int PS4_SYSV_ABI AvPlayer::OpenFile(void* handle, const char* filename) {
|
||||||
|
auto const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
std::lock_guard guard(self->m_file_io_mutex);
|
||||||
|
|
||||||
|
const auto open = self->m_init_data_original.file_replacement.open;
|
||||||
|
const auto ptr = self->m_init_data_original.file_replacement.object_ptr;
|
||||||
|
return open(ptr, filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
int PS4_SYSV_ABI AvPlayer::CloseFile(void* handle) {
|
||||||
|
auto const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
std::lock_guard guard(self->m_file_io_mutex);
|
||||||
|
|
||||||
|
const auto close = self->m_init_data_original.file_replacement.close;
|
||||||
|
const auto ptr = self->m_init_data_original.file_replacement.object_ptr;
|
||||||
|
return close(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
int PS4_SYSV_ABI AvPlayer::ReadOffsetFile(void* handle, u8* buffer, u64 position, u32 length) {
|
||||||
|
auto const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
std::lock_guard guard(self->m_file_io_mutex);
|
||||||
|
|
||||||
|
const auto read_offset = self->m_init_data_original.file_replacement.readOffset;
|
||||||
|
const auto ptr = self->m_init_data_original.file_replacement.object_ptr;
|
||||||
|
return read_offset(ptr, buffer, position, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 PS4_SYSV_ABI AvPlayer::SizeFile(void* handle) {
|
||||||
|
auto const self = reinterpret_cast<AvPlayer*>(handle);
|
||||||
|
std::lock_guard guard(self->m_file_io_mutex);
|
||||||
|
|
||||||
|
const auto size = self->m_init_data_original.file_replacement.size;
|
||||||
|
const auto ptr = self->m_init_data_original.file_replacement.object_ptr;
|
||||||
|
return size(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
SceAvPlayerInitData AvPlayer::StubInitData(const SceAvPlayerInitData& data) {
|
||||||
|
SceAvPlayerInitData result = data;
|
||||||
|
result.memory_replacement.object_ptr = this;
|
||||||
|
result.memory_replacement.allocate = &AvPlayer::Allocate;
|
||||||
|
result.memory_replacement.deallocate = &AvPlayer::Deallocate;
|
||||||
|
result.memory_replacement.allocate_texture = &AvPlayer::AllocateTexture;
|
||||||
|
result.memory_replacement.deallocate_texture = &AvPlayer::DeallocateTexture;
|
||||||
|
if (data.file_replacement.open == nullptr || data.file_replacement.close == nullptr ||
|
||||||
|
data.file_replacement.readOffset == nullptr || data.file_replacement.size == nullptr) {
|
||||||
|
result.file_replacement = {};
|
||||||
|
} else {
|
||||||
|
result.file_replacement.object_ptr = this;
|
||||||
|
result.file_replacement.open = &AvPlayer::OpenFile;
|
||||||
|
result.file_replacement.close = &AvPlayer::CloseFile;
|
||||||
|
result.file_replacement.readOffset = &AvPlayer::ReadOffsetFile;
|
||||||
|
result.file_replacement.size = &AvPlayer::SizeFile;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
AvPlayer::AvPlayer(const SceAvPlayerInitData& data)
|
||||||
|
: m_init_data(StubInitData(data)), m_init_data_original(data),
|
||||||
|
m_state(std::make_unique<AvPlayerState>(m_init_data)) {}
|
||||||
|
|
||||||
|
s32 AvPlayer::PostInit(const SceAvPlayerPostInitData& data) {
|
||||||
|
m_post_init_data = data;
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::AddSource(std::string_view path) {
|
||||||
|
if (path.empty()) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_INVALID_PARAMS;
|
||||||
|
}
|
||||||
|
if (!m_state->AddSource(path, GetSourceType(path))) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::GetStreamCount() {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
const auto res = m_state->GetStreamCount();
|
||||||
|
if (AVPLAYER_IS_ERROR(res)) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::GetStreamInfo(u32 stream_index, SceAvPlayerStreamInfo& info) {
|
||||||
|
if (!m_state->GetStreamInfo(stream_index, info)) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::EnableStream(u32 stream_index) {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
if (!m_state->EnableStream(stream_index)) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::Start() {
|
||||||
|
if (!m_state->Start()) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayer::GetVideoData(SceAvPlayerFrameInfo& video_info) {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_state->GetVideoData(video_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayer::GetVideoData(SceAvPlayerFrameInfoEx& video_info) {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_state->GetVideoData(video_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayer::GetAudioData(SceAvPlayerFrameInfo& audio_info) {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_state->GetAudioData(audio_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayer::IsActive() {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_state->IsActive();
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 AvPlayer::CurrentTime() {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return m_state->CurrentTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayer::Stop() {
|
||||||
|
if (m_state == nullptr || !m_state->Stop()) {
|
||||||
|
return ORBIS_AVPLAYER_ERROR_OPERATION_FAILED;
|
||||||
|
}
|
||||||
|
return ORBIS_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayer::SetLooping(bool is_looping) {
|
||||||
|
if (m_state == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_state->SetLooping(is_looping);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,68 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
#include "avplayer_data_streamer.h"
|
||||||
|
#include "avplayer_state.h"
|
||||||
|
|
||||||
|
#include "core/libraries/kernel/thread_management.h"
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include <libavcodec/avcodec.h>
|
||||||
|
#include <libavformat/avformat.h>
|
||||||
|
}
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
class AvPlayer {
|
||||||
|
public:
|
||||||
|
AvPlayer(const SceAvPlayerInitData& data);
|
||||||
|
|
||||||
|
s32 PostInit(const SceAvPlayerPostInitData& data);
|
||||||
|
s32 AddSource(std::string_view filename);
|
||||||
|
s32 GetStreamCount();
|
||||||
|
s32 GetStreamInfo(u32 stream_index, SceAvPlayerStreamInfo& info);
|
||||||
|
s32 EnableStream(u32 stream_index);
|
||||||
|
s32 Start();
|
||||||
|
bool GetAudioData(SceAvPlayerFrameInfo& audio_info);
|
||||||
|
bool GetVideoData(SceAvPlayerFrameInfo& video_info);
|
||||||
|
bool GetVideoData(SceAvPlayerFrameInfoEx& video_info);
|
||||||
|
bool IsActive();
|
||||||
|
u64 CurrentTime();
|
||||||
|
s32 Stop();
|
||||||
|
bool SetLooping(bool is_looping);
|
||||||
|
|
||||||
|
private:
|
||||||
|
using ScePthreadMutex = Kernel::ScePthreadMutex;
|
||||||
|
|
||||||
|
// Memory Replacement
|
||||||
|
static void* PS4_SYSV_ABI Allocate(void* handle, u32 alignment, u32 size);
|
||||||
|
static void PS4_SYSV_ABI Deallocate(void* handle, void* memory);
|
||||||
|
static void* PS4_SYSV_ABI AllocateTexture(void* handle, u32 alignment, u32 size);
|
||||||
|
static void PS4_SYSV_ABI DeallocateTexture(void* handle, void* memory);
|
||||||
|
|
||||||
|
// File Replacement
|
||||||
|
static int PS4_SYSV_ABI OpenFile(void* handle, const char* filename);
|
||||||
|
static int PS4_SYSV_ABI CloseFile(void* handle);
|
||||||
|
static int PS4_SYSV_ABI ReadOffsetFile(void* handle, u8* buffer, u64 position, u32 length);
|
||||||
|
static u64 PS4_SYSV_ABI SizeFile(void* handle);
|
||||||
|
|
||||||
|
SceAvPlayerInitData StubInitData(const SceAvPlayerInitData& data);
|
||||||
|
|
||||||
|
SceAvPlayerInitData m_init_data{};
|
||||||
|
SceAvPlayerInitData m_init_data_original{};
|
||||||
|
SceAvPlayerPostInitData m_post_init_data{};
|
||||||
|
std::mutex m_file_io_mutex{};
|
||||||
|
|
||||||
|
std::atomic_bool m_has_source{};
|
||||||
|
std::unique_ptr<AvPlayerState> m_state{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,772 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "avplayer_source.h"
|
||||||
|
|
||||||
|
#include "avplayer_file_streamer.h"
|
||||||
|
|
||||||
|
#include "common/alignment.h"
|
||||||
|
#include "common/singleton.h"
|
||||||
|
#include "core/file_sys/fs.h"
|
||||||
|
#include "core/libraries/kernel/time_management.h"
|
||||||
|
|
||||||
|
#include <magic_enum.hpp>
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include <libavcodec/avcodec.h>
|
||||||
|
#include <libavformat/avformat.h>
|
||||||
|
#include <libavformat/avio.h>
|
||||||
|
#include <libswresample/swresample.h>
|
||||||
|
#include <libswscale/swscale.h>
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
using namespace Kernel;
|
||||||
|
|
||||||
|
AvPlayerSource::AvPlayerSource(AvPlayerStateCallback& state) : m_state(state) {}
|
||||||
|
|
||||||
|
AvPlayerSource::~AvPlayerSource() {
|
||||||
|
Stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::Init(const SceAvPlayerInitData& init_data, std::string_view path) {
|
||||||
|
m_memory_replacement = init_data.memory_replacement,
|
||||||
|
m_num_output_video_framebuffers =
|
||||||
|
std::min(std::max(2, init_data.num_output_video_framebuffers), 16);
|
||||||
|
|
||||||
|
AVFormatContext* context = avformat_alloc_context();
|
||||||
|
if (init_data.file_replacement.open != nullptr) {
|
||||||
|
m_up_data_streamer = std::make_unique<AvPlayerFileStreamer>(init_data.file_replacement);
|
||||||
|
if (!m_up_data_streamer->Init(path)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
context->pb = m_up_data_streamer->GetContext();
|
||||||
|
if (AVPLAYER_IS_ERROR(avformat_open_input(&context, nullptr, nullptr, nullptr))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const auto mnt = Common::Singleton<Core::FileSys::MntPoints>::Instance();
|
||||||
|
const auto filepath = mnt->GetHostPath(path);
|
||||||
|
if (AVPLAYER_IS_ERROR(
|
||||||
|
avformat_open_input(&context, filepath.string().c_str(), nullptr, nullptr))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
m_avformat_context = AVFormatContextPtr(context, &ReleaseAVFormatContext);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::FindStreamInfo() {
|
||||||
|
if (m_avformat_context == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not find stream info. NULL context.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (m_avformat_context->nb_streams > 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return avformat_find_stream_info(m_avformat_context.get(), nullptr) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
s32 AvPlayerSource::GetStreamCount() {
|
||||||
|
if (m_avformat_context == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get stream count. NULL context.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Stream Count: {}", m_avformat_context->nb_streams);
|
||||||
|
return m_avformat_context->nb_streams;
|
||||||
|
}
|
||||||
|
|
||||||
|
static s32 CodecTypeToStreamType(AVMediaType codec_type) {
|
||||||
|
switch (codec_type) {
|
||||||
|
case AVMediaType::AVMEDIA_TYPE_VIDEO:
|
||||||
|
return SCE_AVPLAYER_VIDEO;
|
||||||
|
case AVMediaType::AVMEDIA_TYPE_AUDIO:
|
||||||
|
return SCE_AVPLAYER_AUDIO;
|
||||||
|
case AVMediaType::AVMEDIA_TYPE_SUBTITLE:
|
||||||
|
return SCE_AVPLAYER_TIMEDTEXT;
|
||||||
|
default:
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Unexpected AVMediaType {}", magic_enum::enum_name(codec_type));
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static f32 AVRationalToF32(const AVRational rational) {
|
||||||
|
return f32(rational.num) / rational.den;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::GetStreamInfo(u32 stream_index, SceAvPlayerStreamInfo& info) {
|
||||||
|
info = {};
|
||||||
|
if (m_avformat_context == nullptr || stream_index >= m_avformat_context->nb_streams) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get stream {} info.", stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const auto p_stream = m_avformat_context->streams[stream_index];
|
||||||
|
if (p_stream == nullptr || p_stream->codecpar == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get stream {} info. NULL stream.", stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
info.type = CodecTypeToStreamType(p_stream->codecpar->codec_type);
|
||||||
|
info.start_time = p_stream->start_time;
|
||||||
|
info.duration = p_stream->duration;
|
||||||
|
const auto p_lang_node = av_dict_get(p_stream->metadata, "language", nullptr, 0);
|
||||||
|
if (p_lang_node != nullptr) {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Stream {} language = {}", stream_index, p_lang_node->value);
|
||||||
|
} else {
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Stream {} language is unknown", stream_index);
|
||||||
|
}
|
||||||
|
switch (info.type) {
|
||||||
|
case SCE_AVPLAYER_VIDEO:
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Stream {} is a video stream.", stream_index);
|
||||||
|
info.details.video.aspect_ratio =
|
||||||
|
f32(p_stream->codecpar->width) / p_stream->codecpar->height;
|
||||||
|
info.details.video.width = Common::AlignUp(u32(p_stream->codecpar->width), 16);
|
||||||
|
info.details.video.height = Common::AlignUp(u32(p_stream->codecpar->height), 16);
|
||||||
|
if (p_lang_node != nullptr) {
|
||||||
|
std::memcpy(info.details.video.language_code, p_lang_node->value,
|
||||||
|
std::min(strlen(p_lang_node->value), size_t(3)));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case SCE_AVPLAYER_AUDIO:
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Stream {} is an audio stream.", stream_index);
|
||||||
|
info.details.audio.channel_count = p_stream->codecpar->ch_layout.nb_channels;
|
||||||
|
info.details.audio.sample_rate = p_stream->codecpar->sample_rate;
|
||||||
|
info.details.audio.size = 0; // sceAvPlayerGetStreamInfo() is expected to set this to 0
|
||||||
|
if (p_lang_node != nullptr) {
|
||||||
|
std::memcpy(info.details.audio.language_code, p_lang_node->value,
|
||||||
|
std::min(strlen(p_lang_node->value), size_t(3)));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case SCE_AVPLAYER_TIMEDTEXT:
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Stream {} is a timedtext stream.", stream_index);
|
||||||
|
info.details.subs.font_size = 12;
|
||||||
|
info.details.subs.text_size = 12;
|
||||||
|
if (p_lang_node != nullptr) {
|
||||||
|
std::memcpy(info.details.subs.language_code, p_lang_node->value,
|
||||||
|
std::min(strlen(p_lang_node->value), size_t(3)));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Stream {} type is unknown: {}.", stream_index, info.type);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::EnableStream(u32 stream_index) {
|
||||||
|
if (m_avformat_context == nullptr || stream_index >= m_avformat_context->nb_streams) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const auto stream = m_avformat_context->streams[stream_index];
|
||||||
|
const auto decoder = avcodec_find_decoder(stream->codecpar->codec_id);
|
||||||
|
if (decoder == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
switch (stream->codecpar->codec_type) {
|
||||||
|
case AVMediaType::AVMEDIA_TYPE_VIDEO: {
|
||||||
|
m_video_stream_index = stream_index;
|
||||||
|
m_video_codec_context =
|
||||||
|
AVCodecContextPtr(avcodec_alloc_context3(decoder), &ReleaseAVCodecContext);
|
||||||
|
if (avcodec_parameters_to_context(m_video_codec_context.get(), stream->codecpar) < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not copy stream {} avcodec parameters to context.",
|
||||||
|
stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (avcodec_open2(m_video_codec_context.get(), decoder, nullptr) < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not open avcodec for video stream {}.", stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const auto width = Common::AlignUp(u32(m_video_codec_context->width), 16);
|
||||||
|
const auto height = Common::AlignUp(u32(m_video_codec_context->height), 16);
|
||||||
|
const auto size = (width * height * 3) / 2;
|
||||||
|
for (u64 index = 0; index < m_num_output_video_framebuffers; ++index) {
|
||||||
|
m_video_buffers.Push(FrameBuffer(m_memory_replacement, 0x100, size));
|
||||||
|
}
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Video stream {} enabled", stream_index);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AVMediaType::AVMEDIA_TYPE_AUDIO: {
|
||||||
|
m_audio_stream_index = stream_index;
|
||||||
|
m_audio_codec_context =
|
||||||
|
AVCodecContextPtr(avcodec_alloc_context3(decoder), &ReleaseAVCodecContext);
|
||||||
|
if (avcodec_parameters_to_context(m_audio_codec_context.get(), stream->codecpar) < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not copy stream {} avcodec parameters to context.",
|
||||||
|
stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (avcodec_open2(m_audio_codec_context.get(), decoder, nullptr) < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not open avcodec for audio stream {}.", stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const auto num_channels = m_audio_codec_context->ch_layout.nb_channels;
|
||||||
|
const auto align = num_channels * sizeof(u16);
|
||||||
|
const auto size = num_channels * sizeof(u16) * 1024;
|
||||||
|
for (u64 index = 0; index < 4; ++index) {
|
||||||
|
m_audio_buffers.Push(FrameBuffer(m_memory_replacement, 0x100, size));
|
||||||
|
}
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Audio stream {} enabled", stream_index);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Unknown stream type {} for stream {}",
|
||||||
|
magic_enum::enum_name(stream->codecpar->codec_type), stream_index);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::SetLooping(bool is_looping) {
|
||||||
|
m_is_looping = is_looping;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<bool> AvPlayerSource::HasFrames(u32 num_frames) {
|
||||||
|
return m_video_packets.Size() > num_frames || m_is_eof;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::Start() {
|
||||||
|
std::unique_lock lock(m_state_mutex);
|
||||||
|
|
||||||
|
if (m_audio_codec_context == nullptr && m_video_codec_context == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not start playback. NULL context.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
m_demuxer_thread = std::jthread([this](std::stop_token stop) { this->DemuxerThread(stop); });
|
||||||
|
m_video_decoder_thread =
|
||||||
|
std::jthread([this](std::stop_token stop) { this->VideoDecoderThread(stop); });
|
||||||
|
m_audio_decoder_thread =
|
||||||
|
std::jthread([this](std::stop_token stop) { this->AudioDecoderThread(stop); });
|
||||||
|
m_start_time = std::chrono::high_resolution_clock::now();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::Stop() {
|
||||||
|
std::unique_lock lock(m_state_mutex);
|
||||||
|
|
||||||
|
if (!HasRunningThreads()) {
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Could not stop playback: already stopped.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_video_decoder_thread.request_stop();
|
||||||
|
m_audio_decoder_thread.request_stop();
|
||||||
|
m_demuxer_thread.request_stop();
|
||||||
|
if (m_demuxer_thread.joinable()) {
|
||||||
|
m_demuxer_thread.join();
|
||||||
|
}
|
||||||
|
if (m_video_decoder_thread.joinable()) {
|
||||||
|
m_video_decoder_thread.join();
|
||||||
|
}
|
||||||
|
if (m_audio_decoder_thread.joinable()) {
|
||||||
|
m_audio_decoder_thread.join();
|
||||||
|
}
|
||||||
|
if (m_current_audio_frame.has_value()) {
|
||||||
|
m_audio_buffers.Push(std::move(m_current_audio_frame.value()));
|
||||||
|
m_current_audio_frame.reset();
|
||||||
|
}
|
||||||
|
if (m_current_video_frame.has_value()) {
|
||||||
|
m_video_buffers.Push(std::move(m_current_video_frame.value()));
|
||||||
|
m_current_video_frame.reset();
|
||||||
|
}
|
||||||
|
m_stop_cv.Notify();
|
||||||
|
|
||||||
|
m_audio_packets.Clear();
|
||||||
|
m_video_packets.Clear();
|
||||||
|
m_audio_frames.Clear();
|
||||||
|
m_video_frames.Clear();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::GetVideoData(SceAvPlayerFrameInfo& video_info) {
|
||||||
|
if (!IsActive()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
SceAvPlayerFrameInfoEx info{};
|
||||||
|
if (!GetVideoData(info)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
video_info = {};
|
||||||
|
video_info.timestamp = u64(info.timestamp);
|
||||||
|
video_info.pData = reinterpret_cast<u8*>(info.pData);
|
||||||
|
video_info.details.video.aspect_ratio = info.details.video.aspect_ratio;
|
||||||
|
video_info.details.video.width = info.details.video.width;
|
||||||
|
video_info.details.video.height = info.details.video.height;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::GetVideoData(SceAvPlayerFrameInfoEx& video_info) {
|
||||||
|
if (!IsActive()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_video_frames_cv.Wait([this] { return m_video_frames.Size() != 0 || m_is_eof; });
|
||||||
|
|
||||||
|
auto frame = m_video_frames.Pop();
|
||||||
|
if (!frame.has_value()) {
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Could get video frame. EOF reached.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using namespace std::chrono;
|
||||||
|
auto elapsed_time =
|
||||||
|
duration_cast<milliseconds>(high_resolution_clock::now() - m_start_time).count();
|
||||||
|
if (elapsed_time < frame->info.timestamp) {
|
||||||
|
if (m_stop_cv.WaitFor(milliseconds(frame->info.timestamp - elapsed_time),
|
||||||
|
[&] { return elapsed_time >= frame->info.timestamp; })) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// return the buffer to the queue
|
||||||
|
if (m_current_video_frame.has_value()) {
|
||||||
|
m_video_buffers.Push(std::move(m_current_video_frame.value()));
|
||||||
|
m_video_buffers_cv.Notify();
|
||||||
|
}
|
||||||
|
m_current_video_frame = std::move(frame->buffer);
|
||||||
|
video_info = frame->info;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::GetAudioData(SceAvPlayerFrameInfo& audio_info) {
|
||||||
|
if (!IsActive()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_audio_frames_cv.Wait([this] { return m_audio_frames.Size() != 0 || m_is_eof; });
|
||||||
|
|
||||||
|
auto frame = m_audio_frames.Pop();
|
||||||
|
if (!frame.has_value()) {
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Could get audio frame. EOF reached.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
using namespace std::chrono;
|
||||||
|
auto elapsed_time =
|
||||||
|
duration_cast<milliseconds>(high_resolution_clock::now() - m_start_time).count();
|
||||||
|
if (elapsed_time < frame->info.timestamp) {
|
||||||
|
if (m_stop_cv.WaitFor(milliseconds(frame->info.timestamp - elapsed_time),
|
||||||
|
[&] { return elapsed_time >= frame->info.timestamp; })) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// return the buffer to the queue
|
||||||
|
if (m_current_audio_frame.has_value()) {
|
||||||
|
m_audio_buffers.Push(std::move(m_current_audio_frame.value()));
|
||||||
|
m_audio_buffers_cv.Notify();
|
||||||
|
}
|
||||||
|
m_current_audio_frame = std::move(frame->buffer);
|
||||||
|
|
||||||
|
audio_info = {};
|
||||||
|
audio_info.timestamp = frame->info.timestamp;
|
||||||
|
audio_info.pData = reinterpret_cast<u8*>(frame->info.pData);
|
||||||
|
audio_info.details.audio.sample_rate = frame->info.details.audio.sample_rate;
|
||||||
|
audio_info.details.audio.size = frame->info.details.audio.size;
|
||||||
|
audio_info.details.audio.channel_count = frame->info.details.audio.channel_count;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 AvPlayerSource::CurrentTime() {
|
||||||
|
if (!IsActive()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
using namespace std::chrono;
|
||||||
|
return duration_cast<milliseconds>(high_resolution_clock::now() - m_start_time).count();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::IsActive() {
|
||||||
|
return !m_is_eof || m_audio_packets.Size() != 0 || m_video_packets.Size() != 0 ||
|
||||||
|
m_video_frames.Size() != 0 || m_audio_frames.Size() != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseAVPacket(AVPacket* packet) {
|
||||||
|
if (packet != nullptr) {
|
||||||
|
av_packet_free(&packet);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseAVFrame(AVFrame* frame) {
|
||||||
|
if (frame != nullptr) {
|
||||||
|
av_frame_free(&frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseAVCodecContext(AVCodecContext* context) {
|
||||||
|
if (context != nullptr) {
|
||||||
|
avcodec_free_context(&context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseSWRContext(SwrContext* context) {
|
||||||
|
if (context != nullptr) {
|
||||||
|
swr_free(&context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseSWSContext(SwsContext* context) {
|
||||||
|
if (context != nullptr) {
|
||||||
|
sws_freeContext(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::ReleaseAVFormatContext(AVFormatContext* context) {
|
||||||
|
if (context != nullptr) {
|
||||||
|
avformat_close_input(&context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::DemuxerThread(std::stop_token stop) {
|
||||||
|
using namespace std::chrono;
|
||||||
|
if (!m_audio_stream_index.has_value() && !m_video_stream_index.has_value()) {
|
||||||
|
LOG_WARNING(Lib_AvPlayer, "Could not start DEMUXER thread. No streams enabled.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Demuxer Thread started");
|
||||||
|
|
||||||
|
while (!stop.stop_requested()) {
|
||||||
|
if (m_video_packets.Size() > 30 && m_audio_packets.Size() > 8) {
|
||||||
|
std::this_thread::sleep_for(milliseconds(5));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
AVPacketPtr up_packet(av_packet_alloc(), &ReleaseAVPacket);
|
||||||
|
const auto res = av_read_frame(m_avformat_context.get(), up_packet.get());
|
||||||
|
if (res < 0) {
|
||||||
|
if (res == AVERROR_EOF) {
|
||||||
|
if (m_is_looping) {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "EOF reached in demuxer. Looping the source...");
|
||||||
|
avio_seek(m_avformat_context->pb, 0, SEEK_SET);
|
||||||
|
if (m_video_stream_index.has_value()) {
|
||||||
|
const auto index = m_video_stream_index.value();
|
||||||
|
const auto stream = m_avformat_context->streams[index];
|
||||||
|
avformat_seek_file(m_avformat_context.get(), index, 0, 0, stream->duration,
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
if (m_audio_stream_index.has_value()) {
|
||||||
|
const auto index = m_audio_stream_index.value();
|
||||||
|
const auto stream = m_avformat_context->streams[index];
|
||||||
|
avformat_seek_file(m_avformat_context.get(), index, 0, 0, stream->duration,
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "EOF reached in demuxer. Exiting.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not read AV frame: error = {}", res);
|
||||||
|
m_state.OnError();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (up_packet->stream_index == m_video_stream_index) {
|
||||||
|
m_video_packets.Push(std::move(up_packet));
|
||||||
|
m_video_packets_cv.Notify();
|
||||||
|
} else if (up_packet->stream_index == m_audio_stream_index) {
|
||||||
|
m_audio_packets.Push(std::move(up_packet));
|
||||||
|
m_audio_packets_cv.Notify();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m_is_eof = true;
|
||||||
|
|
||||||
|
m_video_packets_cv.Notify();
|
||||||
|
m_audio_packets_cv.Notify();
|
||||||
|
m_video_frames_cv.Notify();
|
||||||
|
m_audio_frames_cv.Notify();
|
||||||
|
|
||||||
|
if (m_video_decoder_thread.joinable()) {
|
||||||
|
m_video_decoder_thread.join();
|
||||||
|
}
|
||||||
|
if (m_audio_decoder_thread.joinable()) {
|
||||||
|
m_audio_decoder_thread.join();
|
||||||
|
}
|
||||||
|
m_state.OnEOF();
|
||||||
|
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Demuxer Thread exited normaly");
|
||||||
|
}
|
||||||
|
|
||||||
|
AvPlayerSource::AVFramePtr AvPlayerSource::ConvertVideoFrame(const AVFrame& frame) {
|
||||||
|
auto nv12_frame = AVFramePtr{av_frame_alloc(), &ReleaseAVFrame};
|
||||||
|
nv12_frame->pts = frame.pts;
|
||||||
|
nv12_frame->pkt_dts = frame.pkt_dts < 0 ? 0 : frame.pkt_dts;
|
||||||
|
nv12_frame->format = AV_PIX_FMT_NV12;
|
||||||
|
nv12_frame->width = frame.width;
|
||||||
|
nv12_frame->height = frame.height;
|
||||||
|
nv12_frame->sample_aspect_ratio = frame.sample_aspect_ratio;
|
||||||
|
nv12_frame->crop_top = frame.crop_top;
|
||||||
|
nv12_frame->crop_bottom = frame.crop_bottom;
|
||||||
|
nv12_frame->crop_left = frame.crop_left;
|
||||||
|
nv12_frame->crop_right = frame.crop_right;
|
||||||
|
|
||||||
|
av_frame_get_buffer(nv12_frame.get(), 0);
|
||||||
|
|
||||||
|
if (m_sws_context == nullptr) {
|
||||||
|
m_sws_context =
|
||||||
|
SWSContextPtr(sws_getContext(frame.width, frame.height, AVPixelFormat(frame.format),
|
||||||
|
nv12_frame->width, nv12_frame->height, AV_PIX_FMT_NV12,
|
||||||
|
SWS_FAST_BILINEAR, nullptr, nullptr, nullptr),
|
||||||
|
&ReleaseSWSContext);
|
||||||
|
}
|
||||||
|
const auto res = sws_scale(m_sws_context.get(), frame.data, frame.linesize, 0, frame.height,
|
||||||
|
nv12_frame->data, nv12_frame->linesize);
|
||||||
|
if (res < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not convert to NV12: {}", av_err2str(res));
|
||||||
|
return AVFramePtr{nullptr, &ReleaseAVFrame};
|
||||||
|
}
|
||||||
|
return nv12_frame;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void CopyNV12Data(u8* dst, const AVFrame& src) {
|
||||||
|
const auto width = Common::AlignUp(u32(src.width), 16);
|
||||||
|
const auto height = Common::AlignUp(u32(src.height), 16);
|
||||||
|
|
||||||
|
if (src.width == width) {
|
||||||
|
std::memcpy(dst, src.data[0], src.width * src.height);
|
||||||
|
std::memcpy(dst + src.width * height, src.data[1], (src.width * src.height) / 2);
|
||||||
|
} else {
|
||||||
|
const auto luma_dst = dst;
|
||||||
|
for (u32 y = 0; y < src.height; ++y) {
|
||||||
|
std::memcpy(luma_dst + y * width, src.data[0] + y * src.width, src.width);
|
||||||
|
}
|
||||||
|
const auto chroma_dst = dst + width * height;
|
||||||
|
for (u32 y = 0; y < src.height / 2; ++y) {
|
||||||
|
std::memcpy(chroma_dst + y * (width / 2), src.data[0] + y * (src.width / 2),
|
||||||
|
src.width / 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Frame AvPlayerSource::PrepareVideoFrame(FrameBuffer buffer, const AVFrame& frame) {
|
||||||
|
ASSERT(frame.format == AV_PIX_FMT_NV12);
|
||||||
|
|
||||||
|
auto p_buffer = buffer.GetBuffer();
|
||||||
|
CopyNV12Data(p_buffer, frame);
|
||||||
|
|
||||||
|
const auto pkt_dts = u64(frame.pkt_dts) * 1000;
|
||||||
|
const auto stream = m_avformat_context->streams[m_video_stream_index.value()];
|
||||||
|
const auto time_base = stream->time_base;
|
||||||
|
const auto den = time_base.den;
|
||||||
|
const auto num = time_base.num;
|
||||||
|
const auto timestamp = (num != 0 && den > 1) ? (pkt_dts * num) / den : pkt_dts;
|
||||||
|
|
||||||
|
const auto width = Common::AlignUp(u32(frame.width), 16);
|
||||||
|
const auto height = Common::AlignUp(u32(frame.height), 16);
|
||||||
|
|
||||||
|
return Frame{
|
||||||
|
.buffer = std::move(buffer),
|
||||||
|
.info =
|
||||||
|
{
|
||||||
|
.pData = p_buffer,
|
||||||
|
.timestamp = timestamp,
|
||||||
|
.details =
|
||||||
|
{
|
||||||
|
.video =
|
||||||
|
{
|
||||||
|
.width = u32(width),
|
||||||
|
.height = u32(height),
|
||||||
|
.aspect_ratio = AVRationalToF32(frame.sample_aspect_ratio),
|
||||||
|
.crop_left_offset = u32(frame.crop_left),
|
||||||
|
.crop_right_offset = u32(frame.crop_right + (width - frame.width)),
|
||||||
|
.crop_top_offset = u32(frame.crop_top),
|
||||||
|
.crop_bottom_offset =
|
||||||
|
u32(frame.crop_bottom + (height - frame.height)),
|
||||||
|
.pitch = u32(frame.linesize[0]),
|
||||||
|
.luma_bit_depth = 8,
|
||||||
|
.chroma_bit_depth = 8,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::VideoDecoderThread(std::stop_token stop) {
|
||||||
|
using namespace std::chrono;
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Video Decoder Thread started");
|
||||||
|
while ((!m_is_eof || m_video_packets.Size() != 0) && !stop.stop_requested()) {
|
||||||
|
if (!m_video_packets_cv.Wait(stop,
|
||||||
|
[this] { return m_video_packets.Size() != 0 || m_is_eof; })) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const auto packet = m_video_packets.Pop();
|
||||||
|
if (!packet.has_value()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto res = avcodec_send_packet(m_video_codec_context.get(), packet->get());
|
||||||
|
if (res < 0 && res != AVERROR(EAGAIN)) {
|
||||||
|
m_state.OnError();
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not send packet to the video codec. Error = {}",
|
||||||
|
av_err2str(res));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while (res >= 0) {
|
||||||
|
if (!m_video_buffers_cv.Wait(stop, [this] { return m_video_buffers.Size() != 0; })) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (m_video_buffers.Size() == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
auto up_frame = AVFramePtr(av_frame_alloc(), &ReleaseAVFrame);
|
||||||
|
res = avcodec_receive_frame(m_video_codec_context.get(), up_frame.get());
|
||||||
|
if (res < 0) {
|
||||||
|
if (res == AVERROR_EOF) {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "EOF reached in video decoder");
|
||||||
|
return;
|
||||||
|
} else if (res != AVERROR(EAGAIN)) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer,
|
||||||
|
"Could not receive frame from the video codec. Error = {}",
|
||||||
|
av_err2str(res));
|
||||||
|
m_state.OnError();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
auto buffer = m_video_buffers.Pop();
|
||||||
|
if (!buffer.has_value()) {
|
||||||
|
// Video buffers queue was cleared. This means that player was stopped.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (up_frame->format != AV_PIX_FMT_NV12) {
|
||||||
|
const auto nv12_frame = ConvertVideoFrame(*up_frame);
|
||||||
|
m_video_frames.Push(PrepareVideoFrame(std::move(buffer.value()), *nv12_frame));
|
||||||
|
} else {
|
||||||
|
m_video_frames.Push(PrepareVideoFrame(std::move(buffer.value()), *up_frame));
|
||||||
|
}
|
||||||
|
m_video_frames_cv.Notify();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Video Decoder Thread exited normaly");
|
||||||
|
}
|
||||||
|
|
||||||
|
AvPlayerSource::AVFramePtr AvPlayerSource::ConvertAudioFrame(const AVFrame& frame) {
|
||||||
|
auto pcm16_frame = AVFramePtr{av_frame_alloc(), &ReleaseAVFrame};
|
||||||
|
pcm16_frame->pts = frame.pts;
|
||||||
|
pcm16_frame->pkt_dts = frame.pkt_dts < 0 ? 0 : frame.pkt_dts;
|
||||||
|
pcm16_frame->format = AV_SAMPLE_FMT_S16;
|
||||||
|
pcm16_frame->ch_layout = frame.ch_layout;
|
||||||
|
pcm16_frame->sample_rate = frame.sample_rate;
|
||||||
|
|
||||||
|
if (m_swr_context == nullptr) {
|
||||||
|
SwrContext* swr_context = nullptr;
|
||||||
|
AVChannelLayout in_ch_layout = frame.ch_layout;
|
||||||
|
AVChannelLayout out_ch_layout = frame.ch_layout;
|
||||||
|
swr_alloc_set_opts2(&swr_context, &out_ch_layout, AV_SAMPLE_FMT_S16, frame.sample_rate,
|
||||||
|
&in_ch_layout, AVSampleFormat(frame.format), frame.sample_rate, 0,
|
||||||
|
nullptr);
|
||||||
|
m_swr_context = SWRContextPtr(swr_context, &ReleaseSWRContext);
|
||||||
|
swr_init(m_swr_context.get());
|
||||||
|
}
|
||||||
|
const auto res = swr_convert_frame(m_swr_context.get(), pcm16_frame.get(), &frame);
|
||||||
|
if (res < 0) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not convert to NV12: {}", av_err2str(res));
|
||||||
|
return AVFramePtr{nullptr, &ReleaseAVFrame};
|
||||||
|
}
|
||||||
|
return pcm16_frame;
|
||||||
|
}
|
||||||
|
|
||||||
|
Frame AvPlayerSource::PrepareAudioFrame(FrameBuffer buffer, const AVFrame& frame) {
|
||||||
|
ASSERT(frame.format == AV_SAMPLE_FMT_S16);
|
||||||
|
ASSERT(frame.nb_samples <= 1024);
|
||||||
|
|
||||||
|
auto p_buffer = buffer.GetBuffer();
|
||||||
|
const auto size = frame.ch_layout.nb_channels * frame.nb_samples * sizeof(u16);
|
||||||
|
std::memcpy(p_buffer, frame.data[0], size);
|
||||||
|
|
||||||
|
const auto pkt_dts = u64(frame.pkt_dts) * 1000;
|
||||||
|
const auto stream = m_avformat_context->streams[m_audio_stream_index.value()];
|
||||||
|
const auto time_base = stream->time_base;
|
||||||
|
const auto den = time_base.den;
|
||||||
|
const auto num = time_base.num;
|
||||||
|
const auto timestamp = (num != 0 && den > 1) ? (pkt_dts * num) / den : pkt_dts;
|
||||||
|
|
||||||
|
return Frame{
|
||||||
|
.buffer = std::move(buffer),
|
||||||
|
.info =
|
||||||
|
{
|
||||||
|
.pData = p_buffer,
|
||||||
|
.timestamp = timestamp,
|
||||||
|
.details =
|
||||||
|
{
|
||||||
|
.audio =
|
||||||
|
{
|
||||||
|
.channel_count = u16(frame.ch_layout.nb_channels),
|
||||||
|
.sample_rate = u32(frame.sample_rate),
|
||||||
|
.size = u32(size),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerSource::AudioDecoderThread(std::stop_token stop) {
|
||||||
|
using namespace std::chrono;
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Audio Decoder Thread started");
|
||||||
|
while ((!m_is_eof || m_audio_packets.Size() != 0) && !stop.stop_requested()) {
|
||||||
|
if (!m_audio_packets_cv.Wait(stop,
|
||||||
|
[this] { return m_audio_packets.Size() != 0 || m_is_eof; })) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const auto packet = m_audio_packets.Pop();
|
||||||
|
if (!packet.has_value()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
auto res = avcodec_send_packet(m_audio_codec_context.get(), packet->get());
|
||||||
|
if (res < 0 && res != AVERROR(EAGAIN)) {
|
||||||
|
m_state.OnError();
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not send packet to the audio codec. Error = {}",
|
||||||
|
av_err2str(res));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while (res >= 0) {
|
||||||
|
if (!m_audio_buffers_cv.Wait(stop, [this] { return m_audio_buffers.Size() != 0; })) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (m_audio_buffers.Size() == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto up_frame = AVFramePtr(av_frame_alloc(), &ReleaseAVFrame);
|
||||||
|
res = avcodec_receive_frame(m_audio_codec_context.get(), up_frame.get());
|
||||||
|
if (res < 0) {
|
||||||
|
if (res == AVERROR_EOF) {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "EOF reached in audio decoder");
|
||||||
|
return;
|
||||||
|
} else if (res != AVERROR(EAGAIN)) {
|
||||||
|
m_state.OnError();
|
||||||
|
LOG_ERROR(Lib_AvPlayer,
|
||||||
|
"Could not receive frame from the audio codec. Error = {}",
|
||||||
|
av_err2str(res));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
auto buffer = m_audio_buffers.Pop();
|
||||||
|
if (!buffer.has_value()) {
|
||||||
|
// Audio buffers queue was cleared. This means that player was stopped.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (up_frame->format != AV_SAMPLE_FMT_S16) {
|
||||||
|
const auto pcm16_frame = ConvertAudioFrame(*up_frame);
|
||||||
|
m_audio_frames.Push(PrepareAudioFrame(std::move(buffer.value()), *pcm16_frame));
|
||||||
|
} else {
|
||||||
|
m_audio_frames.Push(PrepareAudioFrame(std::move(buffer.value()), *up_frame));
|
||||||
|
}
|
||||||
|
m_audio_frames_cv.Notify();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Audio Decoder Thread exited normaly");
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerSource::HasRunningThreads() const {
|
||||||
|
return m_demuxer_thread.joinable() || m_video_decoder_thread.joinable() ||
|
||||||
|
m_audio_decoder_thread.joinable();
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,219 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "avplayer.h"
|
||||||
|
#include "avplayer_common.h"
|
||||||
|
#include "avplayer_data_streamer.h"
|
||||||
|
|
||||||
|
#include "common/polyfill_thread.h"
|
||||||
|
#include "common/types.h"
|
||||||
|
#include "core/libraries/kernel/thread_management.h"
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
#include <chrono>
|
||||||
|
#include <condition_variable>
|
||||||
|
#include <mutex>
|
||||||
|
#include <optional>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
struct AVCodecContext;
|
||||||
|
struct AVFormatContext;
|
||||||
|
struct AVFrame;
|
||||||
|
struct AVIOContext;
|
||||||
|
struct AVPacket;
|
||||||
|
struct SwrContext;
|
||||||
|
struct SwsContext;
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
class AvPlayerStateCallback {
|
||||||
|
public:
|
||||||
|
virtual ~AvPlayerStateCallback() = default;
|
||||||
|
|
||||||
|
virtual void OnWarning(u32 id) = 0;
|
||||||
|
virtual void OnError() = 0;
|
||||||
|
virtual void OnEOF() = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class FrameBuffer {
|
||||||
|
public:
|
||||||
|
FrameBuffer(const SceAvPlayerMemAllocator& memory_replacement, u32 align, u32 size) noexcept
|
||||||
|
: m_memory_replacement(memory_replacement),
|
||||||
|
m_data(Allocate(memory_replacement, align, size)) {
|
||||||
|
ASSERT_MSG(m_data, "Could not allocated frame buffer.");
|
||||||
|
}
|
||||||
|
|
||||||
|
~FrameBuffer() {
|
||||||
|
if (m_data != nullptr) {
|
||||||
|
Deallocate(m_memory_replacement, m_data);
|
||||||
|
m_data = {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
FrameBuffer(const FrameBuffer&) noexcept = delete;
|
||||||
|
FrameBuffer& operator=(const FrameBuffer&) noexcept = delete;
|
||||||
|
|
||||||
|
FrameBuffer(FrameBuffer&& r) noexcept
|
||||||
|
: m_memory_replacement(r.m_memory_replacement), m_data(r.m_data) {
|
||||||
|
r.m_data = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
FrameBuffer& operator=(FrameBuffer&& r) noexcept {
|
||||||
|
std::swap(m_data, r.m_data);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
u8* GetBuffer() const noexcept {
|
||||||
|
return m_data;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static u8* Allocate(const SceAvPlayerMemAllocator& memory_replacement, u32 align, u32 size) {
|
||||||
|
return reinterpret_cast<u8*>(
|
||||||
|
memory_replacement.allocate(memory_replacement.object_ptr, align, size));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void Deallocate(const SceAvPlayerMemAllocator& memory_replacement, void* ptr) {
|
||||||
|
memory_replacement.deallocate(memory_replacement.object_ptr, ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
const SceAvPlayerMemAllocator& m_memory_replacement;
|
||||||
|
u8* m_data = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Frame {
|
||||||
|
FrameBuffer buffer;
|
||||||
|
SceAvPlayerFrameInfoEx info;
|
||||||
|
};
|
||||||
|
|
||||||
|
class EventCV {
|
||||||
|
public:
|
||||||
|
template <class Pred>
|
||||||
|
void Wait(Pred pred) {
|
||||||
|
std::unique_lock lock(m_mutex);
|
||||||
|
m_cv.wait(lock, std::move(pred));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class Pred>
|
||||||
|
bool Wait(std::stop_token stop, Pred pred) {
|
||||||
|
std::unique_lock lock(m_mutex);
|
||||||
|
return m_cv.wait(lock, std::move(stop), std::move(pred));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <class Pred, class Rep, class Period>
|
||||||
|
bool WaitFor(std::chrono::duration<Rep, Period> timeout, Pred pred) {
|
||||||
|
std::unique_lock lock(m_mutex);
|
||||||
|
return m_cv.wait_for(lock, timeout, std::move(pred));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Notify() {
|
||||||
|
std::unique_lock lock(m_mutex);
|
||||||
|
m_cv.notify_all();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::mutex m_mutex{};
|
||||||
|
std::condition_variable_any m_cv{};
|
||||||
|
};
|
||||||
|
|
||||||
|
class AvPlayerSource {
|
||||||
|
public:
|
||||||
|
AvPlayerSource(AvPlayerStateCallback& state);
|
||||||
|
~AvPlayerSource();
|
||||||
|
|
||||||
|
bool Init(const SceAvPlayerInitData& init_data, std::string_view path);
|
||||||
|
bool FindStreamInfo();
|
||||||
|
s32 GetStreamCount();
|
||||||
|
bool GetStreamInfo(u32 stream_index, SceAvPlayerStreamInfo& info);
|
||||||
|
bool EnableStream(u32 stream_index);
|
||||||
|
void SetLooping(bool is_looping);
|
||||||
|
std::optional<bool> HasFrames(u32 num_frames);
|
||||||
|
bool Start();
|
||||||
|
bool Stop();
|
||||||
|
bool GetAudioData(SceAvPlayerFrameInfo& audio_info);
|
||||||
|
bool GetVideoData(SceAvPlayerFrameInfo& video_info);
|
||||||
|
bool GetVideoData(SceAvPlayerFrameInfoEx& video_info);
|
||||||
|
u64 CurrentTime();
|
||||||
|
bool IsActive();
|
||||||
|
|
||||||
|
private:
|
||||||
|
using ScePthread = Kernel::ScePthread;
|
||||||
|
|
||||||
|
static void ReleaseAVPacket(AVPacket* packet);
|
||||||
|
static void ReleaseAVFrame(AVFrame* frame);
|
||||||
|
static void ReleaseAVCodecContext(AVCodecContext* context);
|
||||||
|
static void ReleaseSWRContext(SwrContext* context);
|
||||||
|
static void ReleaseSWSContext(SwsContext* context);
|
||||||
|
static void ReleaseAVFormatContext(AVFormatContext* context);
|
||||||
|
|
||||||
|
using AVPacketPtr = std::unique_ptr<AVPacket, decltype(&ReleaseAVPacket)>;
|
||||||
|
using AVFramePtr = std::unique_ptr<AVFrame, decltype(&ReleaseAVFrame)>;
|
||||||
|
using AVCodecContextPtr = std::unique_ptr<AVCodecContext, decltype(&ReleaseAVCodecContext)>;
|
||||||
|
using SWRContextPtr = std::unique_ptr<SwrContext, decltype(&ReleaseSWRContext)>;
|
||||||
|
using SWSContextPtr = std::unique_ptr<SwsContext, decltype(&ReleaseSWSContext)>;
|
||||||
|
using AVFormatContextPtr = std::unique_ptr<AVFormatContext, decltype(&ReleaseAVFormatContext)>;
|
||||||
|
|
||||||
|
void DemuxerThread(std::stop_token stop);
|
||||||
|
void VideoDecoderThread(std::stop_token stop);
|
||||||
|
void AudioDecoderThread(std::stop_token stop);
|
||||||
|
|
||||||
|
bool HasRunningThreads() const;
|
||||||
|
|
||||||
|
AVFramePtr ConvertAudioFrame(const AVFrame& frame);
|
||||||
|
AVFramePtr ConvertVideoFrame(const AVFrame& frame);
|
||||||
|
|
||||||
|
Frame PrepareAudioFrame(FrameBuffer buffer, const AVFrame& frame);
|
||||||
|
Frame PrepareVideoFrame(FrameBuffer buffer, const AVFrame& frame);
|
||||||
|
|
||||||
|
AvPlayerStateCallback& m_state;
|
||||||
|
|
||||||
|
SceAvPlayerMemAllocator m_memory_replacement{};
|
||||||
|
u32 m_num_output_video_framebuffers{};
|
||||||
|
|
||||||
|
std::atomic_bool m_is_looping = false;
|
||||||
|
std::atomic_bool m_is_eof = false;
|
||||||
|
|
||||||
|
std::unique_ptr<IDataStreamer> m_up_data_streamer;
|
||||||
|
|
||||||
|
AvPlayerQueue<FrameBuffer> m_audio_buffers;
|
||||||
|
AvPlayerQueue<FrameBuffer> m_video_buffers;
|
||||||
|
|
||||||
|
AvPlayerQueue<AVPacketPtr> m_audio_packets;
|
||||||
|
AvPlayerQueue<AVPacketPtr> m_video_packets;
|
||||||
|
|
||||||
|
AvPlayerQueue<Frame> m_audio_frames;
|
||||||
|
AvPlayerQueue<Frame> m_video_frames;
|
||||||
|
|
||||||
|
std::optional<FrameBuffer> m_current_video_frame;
|
||||||
|
std::optional<FrameBuffer> m_current_audio_frame;
|
||||||
|
|
||||||
|
std::optional<s32> m_video_stream_index{};
|
||||||
|
std::optional<s32> m_audio_stream_index{};
|
||||||
|
|
||||||
|
EventCV m_audio_packets_cv{};
|
||||||
|
EventCV m_audio_frames_cv{};
|
||||||
|
EventCV m_audio_buffers_cv{};
|
||||||
|
|
||||||
|
EventCV m_video_packets_cv{};
|
||||||
|
EventCV m_video_frames_cv{};
|
||||||
|
EventCV m_video_buffers_cv{};
|
||||||
|
|
||||||
|
EventCV m_stop_cv{};
|
||||||
|
|
||||||
|
std::mutex m_state_mutex{};
|
||||||
|
std::jthread m_demuxer_thread{};
|
||||||
|
std::jthread m_video_decoder_thread{};
|
||||||
|
std::jthread m_audio_decoder_thread{};
|
||||||
|
|
||||||
|
AVFormatContextPtr m_avformat_context{nullptr, &ReleaseAVFormatContext};
|
||||||
|
AVCodecContextPtr m_video_codec_context{nullptr, &ReleaseAVCodecContext};
|
||||||
|
AVCodecContextPtr m_audio_codec_context{nullptr, &ReleaseAVCodecContext};
|
||||||
|
SWRContextPtr m_swr_context{nullptr, &ReleaseSWRContext};
|
||||||
|
SWSContextPtr m_sws_context{nullptr, &ReleaseSWSContext};
|
||||||
|
|
||||||
|
std::chrono::high_resolution_clock::time_point m_start_time{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
|
@ -0,0 +1,505 @@
|
||||||
|
// SPDX-FileCopyrightText: Copyright 2024 shadPS4 Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "avplayer_file_streamer.h"
|
||||||
|
#include "avplayer_source.h"
|
||||||
|
#include "avplayer_state.h"
|
||||||
|
|
||||||
|
#include "core/libraries/error_codes.h"
|
||||||
|
#include "core/libraries/kernel/time_management.h"
|
||||||
|
|
||||||
|
#include <magic_enum.hpp>
|
||||||
|
|
||||||
|
namespace Libraries::AvPlayer {
|
||||||
|
|
||||||
|
using namespace Kernel;
|
||||||
|
|
||||||
|
void PS4_SYSV_ABI AvPlayerState::AutoPlayEventCallback(void* opaque, s32 event_id, s32 source_id,
|
||||||
|
void* event_data) {
|
||||||
|
auto const self = reinterpret_cast<AvPlayerState*>(opaque);
|
||||||
|
|
||||||
|
if (event_id == SCE_AVPLAYER_STATE_READY) {
|
||||||
|
s32 video_stream_index = -1;
|
||||||
|
s32 audio_stream_index = -1;
|
||||||
|
s32 timedtext_stream_index = -1;
|
||||||
|
const s32 stream_count = self->GetStreamCount();
|
||||||
|
if (AVPLAYER_IS_ERROR(stream_count)) {
|
||||||
|
self->Stop();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (stream_count == 0) {
|
||||||
|
self->Stop();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (u32 stream_index = 0; stream_index < stream_count; ++stream_index) {
|
||||||
|
SceAvPlayerStreamInfo info{};
|
||||||
|
if (!self->GetStreamInfo(stream_index, info)) {
|
||||||
|
self->Stop();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string_view default_language(
|
||||||
|
reinterpret_cast<char*>(self->m_default_language));
|
||||||
|
switch (info.type) {
|
||||||
|
case SCE_AVPLAYER_VIDEO:
|
||||||
|
if (video_stream_index == -1) {
|
||||||
|
video_stream_index = stream_index;
|
||||||
|
}
|
||||||
|
if (!default_language.empty() &&
|
||||||
|
default_language == reinterpret_cast<char*>(info.details.video.language_code)) {
|
||||||
|
video_stream_index = stream_index;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case SCE_AVPLAYER_AUDIO:
|
||||||
|
if (audio_stream_index == -1) {
|
||||||
|
audio_stream_index = stream_index;
|
||||||
|
}
|
||||||
|
if (!default_language.empty() &&
|
||||||
|
default_language == reinterpret_cast<char*>(info.details.video.language_code)) {
|
||||||
|
audio_stream_index = stream_index;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case SCE_AVPLAYER_TIMEDTEXT:
|
||||||
|
if (default_language.empty()) {
|
||||||
|
timedtext_stream_index = stream_index;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (default_language == reinterpret_cast<char*>(info.details.video.language_code)) {
|
||||||
|
timedtext_stream_index = stream_index;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_stream_index != -1) {
|
||||||
|
self->EnableStream(video_stream_index);
|
||||||
|
}
|
||||||
|
if (audio_stream_index != -1) {
|
||||||
|
self->EnableStream(audio_stream_index);
|
||||||
|
}
|
||||||
|
if (timedtext_stream_index != -1) {
|
||||||
|
self->EnableStream(timedtext_stream_index);
|
||||||
|
}
|
||||||
|
self->Start();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pass other events to the game
|
||||||
|
const auto callback = self->m_event_replacement.event_callback;
|
||||||
|
const auto ptr = self->m_event_replacement.object_ptr;
|
||||||
|
if (callback != nullptr) {
|
||||||
|
callback(ptr, event_id, 0, event_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
AvPlayerState::AvPlayerState(const SceAvPlayerInitData& init_data)
|
||||||
|
: m_init_data(init_data), m_event_replacement(init_data.event_replacement) {
|
||||||
|
if (m_event_replacement.event_callback == nullptr || init_data.auto_start) {
|
||||||
|
m_auto_start = true;
|
||||||
|
m_init_data.event_replacement.event_callback = &AvPlayerState::AutoPlayEventCallback;
|
||||||
|
m_init_data.event_replacement.object_ptr = this;
|
||||||
|
}
|
||||||
|
if (init_data.default_language != nullptr) {
|
||||||
|
std::memcpy(m_default_language, init_data.default_language, sizeof(m_default_language));
|
||||||
|
}
|
||||||
|
SetState(AvState::Initial);
|
||||||
|
StartControllerThread();
|
||||||
|
}
|
||||||
|
|
||||||
|
AvPlayerState::~AvPlayerState() {
|
||||||
|
{
|
||||||
|
std::unique_lock lock(m_source_mutex);
|
||||||
|
m_up_source.reset();
|
||||||
|
}
|
||||||
|
if (m_controller_thread.joinable()) {
|
||||||
|
m_controller_thread.request_stop();
|
||||||
|
m_controller_thread.join();
|
||||||
|
}
|
||||||
|
m_event_queue.Clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
bool AvPlayerState::AddSource(std::string_view path, SceAvPlayerSourceType source_type) {
|
||||||
|
if (path.empty()) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "File path is empty.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::unique_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source != nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Only one source is supported.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_up_source = std::make_unique<AvPlayerSource>(*this);
|
||||||
|
if (!m_up_source->Init(m_init_data, path)) {
|
||||||
|
SetState(AvState::Error);
|
||||||
|
m_up_source.reset();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
AddSourceEvent();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
s32 AvPlayerState::GetStreamCount() {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get stream count. No source.");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return m_up_source->GetStreamCount();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
bool AvPlayerState::GetStreamInfo(u32 stream_index, SceAvPlayerStreamInfo& info) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get stream {} info. No source.", stream_index);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_up_source->GetStreamInfo(stream_index, info);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
bool AvPlayerState::Start() {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr || !m_up_source->Start()) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not start playback.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
SetState(AvState::Play);
|
||||||
|
OnPlaybackStateChanged(AvState::Play);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerState::AvControllerThread(std::stop_token stop) {
|
||||||
|
using std::chrono::milliseconds;
|
||||||
|
|
||||||
|
while (!stop.stop_requested()) {
|
||||||
|
if (m_event_queue.Size() != 0) {
|
||||||
|
ProcessEvent();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
std::this_thread::sleep_for(milliseconds(5));
|
||||||
|
UpdateBufferingState();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
void AvPlayerState::AddSourceEvent() {
|
||||||
|
SetState(AvState::AddingSource);
|
||||||
|
m_event_queue.Push(AvPlayerEvent{
|
||||||
|
.event = AvEventType::AddSource,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerState::WarningEvent(s32 id) {
|
||||||
|
m_event_queue.Push(AvPlayerEvent{
|
||||||
|
.event = AvEventType::WarningId,
|
||||||
|
.payload =
|
||||||
|
{
|
||||||
|
.error = id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
void AvPlayerState::StartControllerThread() {
|
||||||
|
m_controller_thread =
|
||||||
|
std::jthread([this](std::stop_token stop) { this->AvControllerThread(stop); });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
bool AvPlayerState::EnableStream(u32 stream_index) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_up_source->EnableStream(stream_index);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside GAME thread
|
||||||
|
bool AvPlayerState::Stop() {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr || m_current_state == AvState::Stop) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!m_up_source->Stop()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!SetState(AvState::Stop)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
OnPlaybackStateChanged(AvState::Stop);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::GetVideoData(SceAvPlayerFrameInfo& video_info) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_up_source->GetVideoData(video_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::GetVideoData(SceAvPlayerFrameInfoEx& video_info) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_up_source->GetVideoData(video_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::GetAudioData(SceAvPlayerFrameInfo& audio_info) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_up_source->GetAudioData(audio_info);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::IsActive() {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return m_current_state != AvState::Stop && m_current_state != AvState::Error &&
|
||||||
|
m_current_state != AvState::EndOfFile && m_up_source->IsActive();
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 AvPlayerState::CurrentTime() {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not get current time. No source.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return m_up_source->CurrentTime();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::SetLooping(bool is_looping) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source == nullptr) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Could not set loop flag. No source.");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
m_up_source->SetLooping(is_looping);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// May be called from different threads
|
||||||
|
void AvPlayerState::OnWarning(u32 id) {
|
||||||
|
// Forward to CONTROLLER thread
|
||||||
|
WarningEvent(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerState::OnError() {
|
||||||
|
SetState(AvState::Error);
|
||||||
|
OnPlaybackStateChanged(AvState::Error);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AvPlayerState::OnEOF() {
|
||||||
|
SetState(AvState::EndOfFile);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER thread
|
||||||
|
void AvPlayerState::OnPlaybackStateChanged(AvState state) {
|
||||||
|
switch (state) {
|
||||||
|
case AvState::Ready: {
|
||||||
|
EmitEvent(SCE_AVPLAYER_STATE_READY);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvState::Play: {
|
||||||
|
EmitEvent(SCE_AVPLAYER_STATE_PLAY);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvState::Stop: {
|
||||||
|
EmitEvent(SCE_AVPLAYER_STATE_STOP);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvState::Pause: {
|
||||||
|
EmitEvent(SCE_AVPLAYER_STATE_PAUSE);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvState::Buffering: {
|
||||||
|
EmitEvent(SCE_AVPLAYER_STATE_BUFFERING);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER and GAME threads
|
||||||
|
bool AvPlayerState::SetState(AvState state) {
|
||||||
|
std::lock_guard guard(m_state_machine_mutex);
|
||||||
|
|
||||||
|
if (!IsStateTransitionValid(state)) {
|
||||||
|
LOG_ERROR(Lib_AvPlayer, "Invalid state transition: {} -> {}",
|
||||||
|
magic_enum::enum_name(m_current_state.load()), magic_enum::enum_name(state));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
m_previous_state.store(m_current_state);
|
||||||
|
m_current_state.store(state);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER thread
|
||||||
|
std::optional<bool> AvPlayerState::OnBufferingCheckEvent(u32 num_frames) {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (!m_up_source) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
return m_up_source->HasFrames(num_frames);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER thread
|
||||||
|
void AvPlayerState::EmitEvent(SceAvPlayerEvents event_id, void* event_data) {
|
||||||
|
LOG_INFO(Lib_AvPlayer, "Sending event to the game: id = {}", magic_enum::enum_name(event_id));
|
||||||
|
const auto callback = m_init_data.event_replacement.event_callback;
|
||||||
|
if (callback) {
|
||||||
|
const auto ptr = m_init_data.event_replacement.object_ptr;
|
||||||
|
callback(ptr, event_id, 0, event_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER thread
|
||||||
|
void AvPlayerState::ProcessEvent() {
|
||||||
|
if (m_current_state == AvState::Jump) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::lock_guard guard(m_event_handler_mutex);
|
||||||
|
|
||||||
|
auto event = m_event_queue.Pop();
|
||||||
|
if (!event.has_value()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
switch (event->event) {
|
||||||
|
case AvEventType::WarningId: {
|
||||||
|
OnWarning(event->payload.error);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvEventType::RevertState: {
|
||||||
|
SetState(m_previous_state.load());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvEventType::AddSource: {
|
||||||
|
std::shared_lock lock(m_source_mutex);
|
||||||
|
if (m_up_source->FindStreamInfo()) {
|
||||||
|
SetState(AvState::Ready);
|
||||||
|
OnPlaybackStateChanged(AvState::Ready);
|
||||||
|
} else {
|
||||||
|
OnWarning(ORBIS_AVPLAYER_ERROR_NOT_SUPPORTED);
|
||||||
|
SetState(AvState::Error);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AvEventType::Error: {
|
||||||
|
OnWarning(event->payload.error);
|
||||||
|
SetState(AvState::Error);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Called inside CONTROLLER thread
|
||||||
|
void AvPlayerState::UpdateBufferingState() {
|
||||||
|
if (m_current_state == AvState::Buffering) {
|
||||||
|
const auto has_frames = OnBufferingCheckEvent(10);
|
||||||
|
if (!has_frames.has_value()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (has_frames.value()) {
|
||||||
|
const auto state =
|
||||||
|
m_previous_state >= AvState::C0x0B ? m_previous_state.load() : AvState::Play;
|
||||||
|
SetState(state);
|
||||||
|
OnPlaybackStateChanged(state);
|
||||||
|
}
|
||||||
|
} else if (m_current_state == AvState::Play) {
|
||||||
|
const auto has_frames = OnBufferingCheckEvent(0);
|
||||||
|
if (!has_frames.has_value()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!has_frames.value()) {
|
||||||
|
SetState(AvState::Buffering);
|
||||||
|
OnPlaybackStateChanged(AvState::Buffering);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AvPlayerState::IsStateTransitionValid(AvState state) {
|
||||||
|
switch (state) {
|
||||||
|
case AvState::Play: {
|
||||||
|
switch (m_current_state.load()) {
|
||||||
|
case AvState::Stop:
|
||||||
|
case AvState::EndOfFile:
|
||||||
|
// case AvState::C0x08:
|
||||||
|
case AvState::Error:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case AvState::Pause: {
|
||||||
|
switch (m_current_state.load()) {
|
||||||
|
case AvState::Stop:
|
||||||
|
case AvState::EndOfFile:
|
||||||
|
// case AvState::C0x08:
|
||||||
|
case AvState::Starting:
|
||||||
|
case AvState::Error:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case AvState::Jump: {
|
||||||
|
switch (m_current_state.load()) {
|
||||||
|
case AvState::Stop:
|
||||||
|
case AvState::EndOfFile:
|
||||||
|
// case AvState::C0x08:
|
||||||
|
case AvState::TrickMode:
|
||||||
|
case AvState::Starting:
|
||||||
|
case AvState::Error:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case AvState::TrickMode: {
|
||||||
|
switch (m_current_state.load()) {
|
||||||
|
case AvState::Stop:
|
||||||
|
case AvState::EndOfFile:
|
||||||
|
// case AvState::C0x08:
|
||||||
|
case AvState::Jump:
|
||||||
|
case AvState::Starting:
|
||||||
|
case AvState::Error:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case AvState::Buffering: {
|
||||||
|
switch (m_current_state.load()) {
|
||||||
|
case AvState::Stop:
|
||||||
|
case AvState::EndOfFile:
|
||||||
|
case AvState::Pause:
|
||||||
|
// case AvState::C0x08:
|
||||||
|
case AvState::Starting:
|
||||||
|
case AvState::Error:
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Libraries::AvPlayer
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue