Compare commits
34 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
fc26189fe1 | ||
|
a40c90e7dd | ||
|
f864a49014 | ||
|
ecbf303266 | ||
|
b3bf05356b | ||
|
cb4b58052f | ||
|
f8cdd5f484 | ||
|
22202be394 | ||
|
17ba217940 | ||
|
aae4595bdb | ||
|
880fd3cfcb | ||
|
f679f25e08 | ||
|
c2709b3bdd | ||
|
2b6e81deea | ||
|
7271f1b18e | ||
|
5fda543f84 | ||
|
95c06de4c1 | ||
|
49c63ea077 | ||
|
531da8a1c0 | ||
|
5cbdfbc7a4 | ||
|
e0544dd9c7 | ||
|
aa784c3e5e | ||
|
9205077590 | ||
|
0ed40c7175 | ||
|
40d47b7aa2 | ||
|
ec0bb74968 | ||
|
42f7f98666 | ||
|
95bad6995c | ||
|
3d42995822 | ||
|
9095941fd1 | ||
|
ba71141bdc | ||
|
0a0675a7f6 | ||
|
a7c6e6a8cf | ||
|
0bc8151c7e |
90
.github/workflows/build.yml
vendored
90
.github/workflows/build.yml
vendored
@@ -18,10 +18,16 @@ on:
|
|||||||
- '*.yml'
|
- '*.yml'
|
||||||
- 'README.md'
|
- 'README.md'
|
||||||
|
|
||||||
|
env:
|
||||||
|
POWERSHELL_TELEMETRY_OPTOUT: 1
|
||||||
|
DOTNET_CLI_TELEMETRY_OPTOUT: 1
|
||||||
|
RYUJINX_BASE_VERSION: "1.1.0"
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
name: ${{ matrix.os }} (${{ matrix.configuration }})
|
name: ${{ matrix.OS_NAME }} (${{ matrix.configuration }})
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
|
timeout-minutes: 35
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ubuntu-latest, macOS-latest, windows-latest]
|
os: [ubuntu-latest, macOS-latest, windows-latest]
|
||||||
@@ -33,7 +39,7 @@ jobs:
|
|||||||
RELEASE_ZIP_OS_NAME: linux_x64
|
RELEASE_ZIP_OS_NAME: linux_x64
|
||||||
|
|
||||||
- os: macOS-latest
|
- os: macOS-latest
|
||||||
OS_NAME: MacOS x64
|
OS_NAME: macOS x64
|
||||||
DOTNET_RUNTIME_IDENTIFIER: osx-x64
|
DOTNET_RUNTIME_IDENTIFIER: osx-x64
|
||||||
RELEASE_ZIP_OS_NAME: osx_x64
|
RELEASE_ZIP_OS_NAME: osx_x64
|
||||||
|
|
||||||
@@ -43,47 +49,107 @@ jobs:
|
|||||||
RELEASE_ZIP_OS_NAME: win_x64
|
RELEASE_ZIP_OS_NAME: win_x64
|
||||||
|
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
env:
|
|
||||||
POWERSHELL_TELEMETRY_OPTOUT: 1
|
|
||||||
DOTNET_CLI_TELEMETRY_OPTOUT: 1
|
|
||||||
RYUJINX_BASE_VERSION: "1.1.0"
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
- uses: actions/setup-dotnet@v3
|
- uses: actions/setup-dotnet@v3
|
||||||
with:
|
with:
|
||||||
dotnet-version: 7.0.x
|
global-json-file: global.json
|
||||||
|
|
||||||
- name: Get git short hash
|
- name: Get git short hash
|
||||||
id: git_short_hash
|
id: git_short_hash
|
||||||
run: echo "result=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
run: echo "result=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
||||||
shell: bash
|
shell: bash
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: dotnet build -c "${{ matrix.configuration }}" -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER
|
run: dotnet build -c "${{ matrix.configuration }}" -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER
|
||||||
|
|
||||||
- name: Test
|
- name: Test
|
||||||
run: dotnet test --no-build -c "${{ matrix.configuration }}"
|
run: dotnet test --no-build -c "${{ matrix.configuration }}"
|
||||||
|
|
||||||
- name: Publish Ryujinx
|
- name: Publish Ryujinx
|
||||||
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx --self-contained true
|
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx --self-contained true
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
- name: Publish Ryujinx.Headless.SDL2
|
- name: Publish Ryujinx.Headless.SDL2
|
||||||
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_sdl2_headless -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx.Headless.SDL2 --self-contained true
|
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_sdl2_headless -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx.Headless.SDL2 --self-contained true
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
- name: Publish Ryujinx.Ava
|
- name: Publish Ryujinx.Ava
|
||||||
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_ava -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx.Ava --self-contained true
|
run: dotnet publish -c "${{ matrix.configuration }}" -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_ava -p:Version="${{ env.RYUJINX_BASE_VERSION }}" -p:DebugType=embedded -p:SourceRevisionId="${{ steps.git_short_hash.outputs.result }}" -p:ExtraDefineConstants=DISABLE_UPDATER src/Ryujinx.Ava --self-contained true
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
|
- name: Set executable bit
|
||||||
|
run: |
|
||||||
|
chmod +x ./publish/Ryujinx ./publish/Ryujinx.sh
|
||||||
|
chmod +x ./publish_sdl2_headless/Ryujinx.Headless.SDL2 ./publish_sdl2_headless/Ryujinx.sh
|
||||||
|
chmod +x ./publish_ava/Ryujinx.Ava ./publish_ava/Ryujinx.sh
|
||||||
|
if: github.event_name == 'pull_request' && matrix.os == 'ubuntu-latest'
|
||||||
|
|
||||||
- name: Upload Ryujinx artifact
|
- name: Upload Ryujinx artifact
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: ryujinx-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
name: ryujinx-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
||||||
path: publish
|
path: publish
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
- name: Upload Ryujinx.Headless.SDL2 artifact
|
- name: Upload Ryujinx.Headless.SDL2 artifact
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: sdl2-ryujinx-headless-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
name: sdl2-ryujinx-headless-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
||||||
path: publish_sdl2_headless
|
path: publish_sdl2_headless
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
- name: Upload Ryujinx.Ava artifact
|
- name: Upload Ryujinx.Ava artifact
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: ava-ryujinx-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
name: ava-ryujinx-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-${{ matrix.RELEASE_ZIP_OS_NAME }}
|
||||||
path: publish_ava
|
path: publish_ava
|
||||||
|
if: github.event_name == 'pull_request' && matrix.os != 'macOS-latest'
|
||||||
|
|
||||||
|
build_macos:
|
||||||
|
name: macOS Universal (${{ matrix.configuration }})
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 35
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
configuration: [ Debug, Release ]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- uses: actions/setup-dotnet@v3
|
||||||
|
with:
|
||||||
|
global-json-file: global.json
|
||||||
|
|
||||||
|
- name: Setup LLVM 14
|
||||||
|
run: |
|
||||||
|
wget https://apt.llvm.org/llvm.sh
|
||||||
|
chmod +x llvm.sh
|
||||||
|
sudo ./llvm.sh 14
|
||||||
|
|
||||||
|
- name: Install rcodesign
|
||||||
|
run: |
|
||||||
|
mkdir -p $HOME/.bin
|
||||||
|
gh release download -R indygreg/apple-platform-rs -O apple-codesign.tar.gz -p 'apple-codesign-*-x86_64-unknown-linux-musl.tar.gz'
|
||||||
|
tar -xzvf apple-codesign.tar.gz --wildcards '*/rcodesign' --strip-components=1
|
||||||
|
rm apple-codesign.tar.gz
|
||||||
|
mv rcodesign $HOME/.bin/
|
||||||
|
echo "$HOME/.bin" >> $GITHUB_PATH
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Get git short hash
|
||||||
|
id: git_short_hash
|
||||||
|
run: echo "result=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Publish macOS
|
||||||
|
run: |
|
||||||
|
./distribution/macos/create_macos_build.sh . publish_tmp publish_ava ./distribution/macos/entitlements.xml "${{ env.RYUJINX_BASE_VERSION }}" "${{ steps.git_short_hash.outputs.result }}" "${{ matrix.configuration }}" "-p:ExtraDefineConstants=DISABLE_UPDATER"
|
||||||
|
|
||||||
|
- name: Upload Ryujinx.Ava artifact
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: ava-ryujinx-${{ matrix.configuration }}-${{ env.RYUJINX_BASE_VERSION }}+${{ steps.git_short_hash.outputs.result }}-macos_universal
|
||||||
|
path: "publish_ava/*.tar.gz"
|
||||||
if: github.event_name == 'pull_request'
|
if: github.event_name == 'pull_request'
|
1
.github/workflows/flatpak.yml
vendored
1
.github/workflows/flatpak.yml
vendored
@@ -12,6 +12,7 @@ concurrency: flatpak-release
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
release:
|
||||||
|
timeout-minutes: 35
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
1
.github/workflows/nightly_pr_comment.yml
vendored
1
.github/workflows/nightly_pr_comment.yml
vendored
@@ -7,6 +7,7 @@ jobs:
|
|||||||
pr_comment:
|
pr_comment:
|
||||||
if: github.event.workflow_run.event == 'pull_request' && github.event.workflow_run.conclusion == 'success'
|
if: github.event.workflow_run.event == 'pull_request' && github.event.workflow_run.conclusion == 'success'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 35
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/github-script@v6
|
- uses: actions/github-script@v6
|
||||||
with:
|
with:
|
||||||
|
245
.github/workflows/release.yml
vendored
245
.github/workflows/release.yml
vendored
@@ -22,95 +22,15 @@ env:
|
|||||||
RYUJINX_TARGET_RELEASE_CHANNEL_REPO: "release-channel-master"
|
RYUJINX_TARGET_RELEASE_CHANNEL_REPO: "release-channel-master"
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
tag:
|
||||||
runs-on: windows-latest
|
name: Create tag
|
||||||
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-dotnet@v3
|
|
||||||
with:
|
|
||||||
global-json-file: global.json
|
|
||||||
- name: Get version info
|
- name: Get version info
|
||||||
id: version_info
|
id: version_info
|
||||||
run: |
|
run: |
|
||||||
echo "build_version=${{ env.RYUJINX_BASE_VERSION }}.${{ github.run_number }}" >> $GITHUB_OUTPUT
|
echo "build_version=${{ env.RYUJINX_BASE_VERSION }}.${{ github.run_number }}" >> $GITHUB_OUTPUT
|
||||||
echo "git_short_hash=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
|
||||||
shell: bash
|
shell: bash
|
||||||
- name: Configure for release
|
|
||||||
run: |
|
|
||||||
sed -r --in-place 's/\%\%RYUJINX_BUILD_VERSION\%\%/${{ steps.version_info.outputs.build_version }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
|
||||||
sed -r --in-place 's/\%\%RYUJINX_BUILD_GIT_HASH\%\%/${{ steps.version_info.outputs.git_short_hash }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
|
||||||
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_NAME\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_NAME }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
|
||||||
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_OWNER\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
|
||||||
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_REPO\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
|
||||||
shell: bash
|
|
||||||
- name: Create output dir
|
|
||||||
run: "mkdir release_output"
|
|
||||||
- name: Publish Windows
|
|
||||||
run: |
|
|
||||||
dotnet publish -c Release -r win10-x64 -o ./publish_windows/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx --self-contained true
|
|
||||||
dotnet publish -c Release -r win10-x64 -o ./publish_windows_sdl2_headless/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Headless.SDL2 --self-contained true
|
|
||||||
dotnet publish -c Release -r win10-x64 -o ./publish_windows_ava/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Ava --self-contained true
|
|
||||||
- name: Packing Windows builds
|
|
||||||
run: |
|
|
||||||
pushd publish_windows
|
|
||||||
7z a ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
|
||||||
popd
|
|
||||||
|
|
||||||
pushd publish_windows_sdl2_headless
|
|
||||||
7z a ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
|
||||||
popd
|
|
||||||
|
|
||||||
pushd publish_windows_ava
|
|
||||||
7z a ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
|
||||||
popd
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: Publish Linux
|
|
||||||
run: |
|
|
||||||
dotnet publish -c Release -r linux-x64 -o ./publish_linux/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx --self-contained true
|
|
||||||
dotnet publish -c Release -r linux-x64 -o ./publish_linux_sdl2_headless/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Headless.SDL2 --self-contained true
|
|
||||||
dotnet publish -c Release -r linux-x64 -o ./publish_linux_ava/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Ava --self-contained true
|
|
||||||
|
|
||||||
- name: Packing Linux builds
|
|
||||||
run: |
|
|
||||||
pushd publish_linux
|
|
||||||
tar --exclude "publish/Ryujinx" --exclude "publish/Ryujinx.sh" -cvf ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar publish
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx" "publish/Ryujinx"
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx.sh" "publish/Ryujinx.sh"
|
|
||||||
gzip -9 < ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar > ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz
|
|
||||||
rm ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar
|
|
||||||
popd
|
|
||||||
|
|
||||||
pushd publish_linux_sdl2_headless
|
|
||||||
tar --exclude "publish/Ryujinx.Headless.SDL2" --exclude "publish/Ryujinx.sh" -cvf ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar publish
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx.Headless.SDL2" "publish/Ryujinx.Headless.SDL2"
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx.sh" "publish/Ryujinx.sh"
|
|
||||||
gzip -9 < ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar > ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz
|
|
||||||
rm ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar
|
|
||||||
popd
|
|
||||||
|
|
||||||
pushd publish_linux_ava
|
|
||||||
tar --exclude "publish/Ryujinx.Ava" --exclude "publish/Ryujinx.sh" -cvf ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar publish
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx.Ava" "publish/Ryujinx.Ava"
|
|
||||||
python3 ../distribution/misc/add_tar_exec.py ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar "publish/Ryujinx.sh" "publish/Ryujinx.sh"
|
|
||||||
gzip -9 < ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar > ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz
|
|
||||||
rm ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar
|
|
||||||
popd
|
|
||||||
shell: bash
|
|
||||||
|
|
||||||
- name: Pushing new release
|
|
||||||
uses: ncipollo/release-action@v1
|
|
||||||
with:
|
|
||||||
name: ${{ steps.version_info.outputs.build_version }}
|
|
||||||
artifacts: "release_output/*.tar.gz,release_output/*.zip"
|
|
||||||
tag: ${{ steps.version_info.outputs.build_version }}
|
|
||||||
body: "For more information about this release please check out the official [Changelog](https://github.com/Ryujinx/Ryujinx/wiki/Changelog)."
|
|
||||||
allowUpdates: true
|
|
||||||
removeArtifacts: true
|
|
||||||
replacesArtifacts: true
|
|
||||||
owner: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}
|
|
||||||
repo: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}
|
|
||||||
token: ${{ secrets.RELEASE_TOKEN }}
|
|
||||||
|
|
||||||
- name: Create tag
|
- name: Create tag
|
||||||
uses: actions/github-script@v5
|
uses: actions/github-script@v5
|
||||||
@@ -123,6 +43,165 @@ jobs:
|
|||||||
sha: context.sha
|
sha: context.sha
|
||||||
})
|
})
|
||||||
|
|
||||||
|
release:
|
||||||
|
name: Release ${{ matrix.OS_NAME }}
|
||||||
|
runs-on: ${{ matrix.os }}
|
||||||
|
timeout-minutes: 35
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
os: [ ubuntu-latest, windows-latest ]
|
||||||
|
include:
|
||||||
|
- os: ubuntu-latest
|
||||||
|
OS_NAME: Linux x64
|
||||||
|
DOTNET_RUNTIME_IDENTIFIER: linux-x64
|
||||||
|
RELEASE_ZIP_OS_NAME: linux_x64
|
||||||
|
|
||||||
|
- os: windows-latest
|
||||||
|
OS_NAME: Windows x64
|
||||||
|
DOTNET_RUNTIME_IDENTIFIER: win10-x64
|
||||||
|
RELEASE_ZIP_OS_NAME: win_x64
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- uses: actions/setup-dotnet@v3
|
||||||
|
with:
|
||||||
|
global-json-file: global.json
|
||||||
|
|
||||||
|
- name: Get version info
|
||||||
|
id: version_info
|
||||||
|
run: |
|
||||||
|
echo "build_version=${{ env.RYUJINX_BASE_VERSION }}.${{ github.run_number }}" >> $GITHUB_OUTPUT
|
||||||
|
echo "git_short_hash=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Configure for release
|
||||||
|
run: |
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_BUILD_VERSION\%\%/${{ steps.version_info.outputs.build_version }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_BUILD_GIT_HASH\%\%/${{ steps.version_info.outputs.git_short_hash }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_NAME\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_NAME }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_OWNER\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_REPO\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Create output dir
|
||||||
|
run: "mkdir release_output"
|
||||||
|
|
||||||
|
- name: Publish
|
||||||
|
run: |
|
||||||
|
dotnet publish -c Release -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_gtk/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx --self-contained true
|
||||||
|
dotnet publish -c Release -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_sdl2_headless/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Headless.SDL2 --self-contained true
|
||||||
|
dotnet publish -c Release -r "${{ matrix.DOTNET_RUNTIME_IDENTIFIER }}" -o ./publish_ava/publish -p:Version="${{ steps.version_info.outputs.build_version }}" -p:SourceRevisionId="${{ steps.version_info.outputs.git_short_hash }}" -p:DebugType=embedded src/Ryujinx.Ava --self-contained true
|
||||||
|
|
||||||
|
- name: Packing Windows builds
|
||||||
|
if: matrix.os == 'windows-latest'
|
||||||
|
run: |
|
||||||
|
pushd publish_gtk
|
||||||
|
7z a ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
||||||
|
popd
|
||||||
|
|
||||||
|
pushd publish_sdl2_headless
|
||||||
|
7z a ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
||||||
|
popd
|
||||||
|
|
||||||
|
pushd publish_ava
|
||||||
|
7z a ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-win_x64.zip publish
|
||||||
|
popd
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Packing Linux builds
|
||||||
|
if: matrix.os == 'ubuntu-latest'
|
||||||
|
run: |
|
||||||
|
pushd publish_gtk
|
||||||
|
chmod +x publish/Ryujinx.sh publish/Ryujinx
|
||||||
|
tar -czvf ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz publish
|
||||||
|
popd
|
||||||
|
|
||||||
|
pushd publish_sdl2_headless
|
||||||
|
chmod +x publish/Ryujinx.sh publish/Ryujinx.Headless.SDL2
|
||||||
|
tar -czvf ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz publish
|
||||||
|
popd
|
||||||
|
|
||||||
|
pushd publish_ava
|
||||||
|
chmod +x publish/Ryujinx.sh publish/Ryujinx.Ava
|
||||||
|
tar -czvf ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-linux_x64.tar.gz publish
|
||||||
|
popd
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Pushing new release
|
||||||
|
uses: ncipollo/release-action@v1
|
||||||
|
with:
|
||||||
|
name: ${{ steps.version_info.outputs.build_version }}
|
||||||
|
artifacts: "release_output/*.tar.gz,release_output/*.zip"
|
||||||
|
tag: ${{ steps.version_info.outputs.build_version }}
|
||||||
|
body: "For more information about this release please check out the official [Changelog](https://github.com/Ryujinx/Ryujinx/wiki/Changelog)."
|
||||||
|
omitBodyDuringUpdate: true
|
||||||
|
allowUpdates: true
|
||||||
|
replacesArtifacts: true
|
||||||
|
owner: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}
|
||||||
|
repo: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}
|
||||||
|
token: ${{ secrets.RELEASE_TOKEN }}
|
||||||
|
|
||||||
|
macos_release:
|
||||||
|
name: Release MacOS universal
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 35
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- uses: actions/setup-dotnet@v3
|
||||||
|
with:
|
||||||
|
global-json-file: global.json
|
||||||
|
|
||||||
|
- name: Setup LLVM 14
|
||||||
|
run: |
|
||||||
|
wget https://apt.llvm.org/llvm.sh
|
||||||
|
chmod +x llvm.sh
|
||||||
|
sudo ./llvm.sh 14
|
||||||
|
|
||||||
|
- name: Install rcodesign
|
||||||
|
run: |
|
||||||
|
mkdir -p $HOME/.bin
|
||||||
|
gh release download -R indygreg/apple-platform-rs -O apple-codesign.tar.gz -p 'apple-codesign-*-x86_64-unknown-linux-musl.tar.gz'
|
||||||
|
tar -xzvf apple-codesign.tar.gz --wildcards '*/rcodesign' --strip-components=1
|
||||||
|
rm apple-codesign.tar.gz
|
||||||
|
mv rcodesign $HOME/.bin/
|
||||||
|
echo "$HOME/.bin" >> $GITHUB_PATH
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Get version info
|
||||||
|
id: version_info
|
||||||
|
run: |
|
||||||
|
echo "build_version=${{ env.RYUJINX_BASE_VERSION }}.${{ github.run_number }}" >> $GITHUB_OUTPUT
|
||||||
|
echo "git_short_hash=$(git rev-parse --short "${{ github.sha }}")" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Configure for release
|
||||||
|
run: |
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_BUILD_VERSION\%\%/${{ steps.version_info.outputs.build_version }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_BUILD_GIT_HASH\%\%/${{ steps.version_info.outputs.git_short_hash }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_NAME\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_NAME }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_OWNER\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
sed -r --in-place 's/\%\%RYUJINX_TARGET_RELEASE_CHANNEL_REPO\%\%/${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}/g;' src/Ryujinx.Common/ReleaseInformation.cs
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Publish macOS
|
||||||
|
run: |
|
||||||
|
./distribution/macos/create_macos_build.sh . publish_tmp publish_ava ./distribution/macos/entitlements.xml "${{ steps.version_info.outputs.build_version }}" "${{ steps.version_info.outputs.git_short_hash }}" Release
|
||||||
|
|
||||||
|
- name: Pushing new release
|
||||||
|
uses: ncipollo/release-action@v1
|
||||||
|
with:
|
||||||
|
name: ${{ steps.version_info.outputs.build_version }}
|
||||||
|
artifacts: "publish_ava/*.tar.gz"
|
||||||
|
tag: ${{ steps.version_info.outputs.build_version }}
|
||||||
|
body: "For more information about this release please check out the official [Changelog](https://github.com/Ryujinx/Ryujinx/wiki/Changelog)."
|
||||||
|
omitBodyDuringUpdate: true
|
||||||
|
allowUpdates: true
|
||||||
|
replacesArtifacts: true
|
||||||
|
owner: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_OWNER }}
|
||||||
|
repo: ${{ env.RYUJINX_TARGET_RELEASE_CHANNEL_REPO }}
|
||||||
|
token: ${{ secrets.RELEASE_TOKEN }}
|
||||||
|
|
||||||
flatpak_release:
|
flatpak_release:
|
||||||
uses: ./.github/workflows/flatpak.yml
|
uses: ./.github/workflows/flatpak.yml
|
||||||
needs: release
|
needs: release
|
||||||
|
@@ -13,7 +13,7 @@
|
|||||||
<PackageVersion Include="CommandLineParser" Version="2.9.1" />
|
<PackageVersion Include="CommandLineParser" Version="2.9.1" />
|
||||||
<PackageVersion Include="Concentus" Version="1.1.7" />
|
<PackageVersion Include="Concentus" Version="1.1.7" />
|
||||||
<PackageVersion Include="DiscordRichPresence" Version="1.1.3.18" />
|
<PackageVersion Include="DiscordRichPresence" Version="1.1.3.18" />
|
||||||
<PackageVersion Include="DynamicData" Version="7.13.5" />
|
<PackageVersion Include="DynamicData" Version="7.13.8" />
|
||||||
<PackageVersion Include="FluentAvaloniaUI" Version="1.4.5" />
|
<PackageVersion Include="FluentAvaloniaUI" Version="1.4.5" />
|
||||||
<PackageVersion Include="GtkSharp.Dependencies" Version="1.1.1" />
|
<PackageVersion Include="GtkSharp.Dependencies" Version="1.1.1" />
|
||||||
<PackageVersion Include="GtkSharp.Dependencies.osx" Version="0.0.5" />
|
<PackageVersion Include="GtkSharp.Dependencies.osx" Version="0.0.5" />
|
||||||
@@ -44,7 +44,7 @@
|
|||||||
<PackageVersion Include="SixLabors.ImageSharp.Drawing" Version="1.0.0-beta11" />
|
<PackageVersion Include="SixLabors.ImageSharp.Drawing" Version="1.0.0-beta11" />
|
||||||
<PackageVersion Include="SPB" Version="0.0.4-build28" />
|
<PackageVersion Include="SPB" Version="0.0.4-build28" />
|
||||||
<PackageVersion Include="System.Drawing.Common" Version="7.0.0" />
|
<PackageVersion Include="System.Drawing.Common" Version="7.0.0" />
|
||||||
<PackageVersion Include="System.IdentityModel.Tokens.Jwt" Version="6.30.0" />
|
<PackageVersion Include="System.IdentityModel.Tokens.Jwt" Version="6.30.1" />
|
||||||
<PackageVersion Include="System.IO.Hashing" Version="7.0.0" />
|
<PackageVersion Include="System.IO.Hashing" Version="7.0.0" />
|
||||||
<PackageVersion Include="System.Management" Version="7.0.1" />
|
<PackageVersion Include="System.Management" Version="7.0.1" />
|
||||||
<PackageVersion Include="UnicornEngine.Unicorn" Version="2.0.2-rc1-fb78016" />
|
<PackageVersion Include="UnicornEngine.Unicorn" Version="2.0.2-rc1-fb78016" />
|
||||||
|
@@ -2,8 +2,8 @@
|
|||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
if [ "$#" -ne 6 ]; then
|
if [ "$#" -lt 7 ]; then
|
||||||
echo "usage <BASE_DIR> <TEMP_DIRECTORY> <OUTPUT_DIRECTORY> <ENTITLEMENTS_FILE_PATH> <VERSION> <SOURCE_REVISION_ID>"
|
echo "usage <BASE_DIR> <TEMP_DIRECTORY> <OUTPUT_DIRECTORY> <ENTITLEMENTS_FILE_PATH> <VERSION> <SOURCE_REVISION_ID> <CONFIGURATION> <EXTRA_ARGS>"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -17,8 +17,16 @@ OUTPUT_DIRECTORY=$(readlink -f "$3")
|
|||||||
ENTITLEMENTS_FILE_PATH=$(readlink -f "$4")
|
ENTITLEMENTS_FILE_PATH=$(readlink -f "$4")
|
||||||
VERSION=$5
|
VERSION=$5
|
||||||
SOURCE_REVISION_ID=$6
|
SOURCE_REVISION_ID=$6
|
||||||
|
CONFIGURATION=$7
|
||||||
|
EXTRA_ARGS=$8
|
||||||
|
|
||||||
|
if [ "$VERSION" == "1.1.0" ];
|
||||||
|
then
|
||||||
|
RELEASE_TAR_FILE_NAME=test-ava-ryujinx-$CONFIGURATION-$VERSION+$SOURCE_REVISION_ID-macos_universal.app.tar
|
||||||
|
else
|
||||||
|
RELEASE_TAR_FILE_NAME=test-ava-ryujinx-$VERSION-macos_universal.app.tar
|
||||||
|
fi
|
||||||
|
|
||||||
RELEASE_TAR_FILE_NAME=Ryujinx-$VERSION-macos_universal.app.tar
|
|
||||||
ARM64_APP_BUNDLE="$TEMP_DIRECTORY/output_arm64/Ryujinx.app"
|
ARM64_APP_BUNDLE="$TEMP_DIRECTORY/output_arm64/Ryujinx.app"
|
||||||
X64_APP_BUNDLE="$TEMP_DIRECTORY/output_x64/Ryujinx.app"
|
X64_APP_BUNDLE="$TEMP_DIRECTORY/output_x64/Ryujinx.app"
|
||||||
UNIVERSAL_APP_BUNDLE="$OUTPUT_DIRECTORY/Ryujinx.app"
|
UNIVERSAL_APP_BUNDLE="$OUTPUT_DIRECTORY/Ryujinx.app"
|
||||||
@@ -27,12 +35,12 @@ EXECUTABLE_SUB_PATH=Contents/MacOS/Ryujinx
|
|||||||
rm -rf "$TEMP_DIRECTORY"
|
rm -rf "$TEMP_DIRECTORY"
|
||||||
mkdir -p "$TEMP_DIRECTORY"
|
mkdir -p "$TEMP_DIRECTORY"
|
||||||
|
|
||||||
DOTNET_COMMON_ARGS="-p:DebugType=embedded -p:Version=$VERSION -p:SourceRevisionId=$SOURCE_REVISION_ID --self-contained true"
|
DOTNET_COMMON_ARGS="-p:DebugType=embedded -p:Version=$VERSION -p:SourceRevisionId=$SOURCE_REVISION_ID --self-contained true $EXTRA_ARGS"
|
||||||
|
|
||||||
dotnet restore
|
dotnet restore
|
||||||
dotnet build -c Release src/Ryujinx.Ava
|
dotnet build -c $CONFIGURATION src/Ryujinx.Ava
|
||||||
dotnet publish -c Release -r osx-arm64 -o "$TEMP_DIRECTORY/publish_arm64" $DOTNET_COMMON_ARGS src/Ryujinx.Ava
|
dotnet publish -c $CONFIGURATION -r osx-arm64 -o "$TEMP_DIRECTORY/publish_arm64" $DOTNET_COMMON_ARGS src/Ryujinx.Ava
|
||||||
dotnet publish -c Release -r osx-x64 -o "$TEMP_DIRECTORY/publish_x64" $DOTNET_COMMON_ARGS src/Ryujinx.Ava
|
dotnet publish -c $CONFIGURATION -r osx-x64 -o "$TEMP_DIRECTORY/publish_x64" $DOTNET_COMMON_ARGS src/Ryujinx.Ava
|
||||||
|
|
||||||
# Get rid of the support library for ARMeilleure for x64 (that's only for arm64)
|
# Get rid of the support library for ARMeilleure for x64 (that's only for arm64)
|
||||||
rm -rf "$TEMP_DIRECTORY/publish_x64/libarmeilleure-jitsupport.dylib"
|
rm -rf "$TEMP_DIRECTORY/publish_x64/libarmeilleure-jitsupport.dylib"
|
||||||
@@ -68,7 +76,7 @@ else
|
|||||||
LIPO=lipo
|
LIPO=lipo
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Make it the executable universal
|
# Make the executable universal
|
||||||
$LIPO "$ARM64_APP_BUNDLE/$EXECUTABLE_SUB_PATH" "$X64_APP_BUNDLE/$EXECUTABLE_SUB_PATH" -output "$UNIVERSAL_APP_BUNDLE/$EXECUTABLE_SUB_PATH" -create
|
$LIPO "$ARM64_APP_BUNDLE/$EXECUTABLE_SUB_PATH" "$X64_APP_BUNDLE/$EXECUTABLE_SUB_PATH" -output "$UNIVERSAL_APP_BUNDLE/$EXECUTABLE_SUB_PATH" -create
|
||||||
|
|
||||||
# Patch up the Info.plist to have appropriate version
|
# Patch up the Info.plist to have appropriate version
|
||||||
@@ -87,10 +95,10 @@ then
|
|||||||
|
|
||||||
# NOTE: Currently require https://github.com/indygreg/apple-platform-rs/pull/44 to work on other OSes.
|
# NOTE: Currently require https://github.com/indygreg/apple-platform-rs/pull/44 to work on other OSes.
|
||||||
# cargo install --git "https://github.com/marysaka/apple-platform-rs" --branch "fix/adhoc-app-bundle" apple-codesign --bin "rcodesign"
|
# cargo install --git "https://github.com/marysaka/apple-platform-rs" --branch "fix/adhoc-app-bundle" apple-codesign --bin "rcodesign"
|
||||||
echo "Usign rcodesign for ad-hoc signing"
|
echo "Using rcodesign for ad-hoc signing"
|
||||||
rcodesign sign --entitlements-xml-path "$ENTITLEMENTS_FILE_PATH" "$UNIVERSAL_APP_BUNDLE"
|
rcodesign sign --entitlements-xml-path "$ENTITLEMENTS_FILE_PATH" "$UNIVERSAL_APP_BUNDLE"
|
||||||
else
|
else
|
||||||
echo "Usign codesign for ad-hoc signing"
|
echo "Using codesign for ad-hoc signing"
|
||||||
codesign --entitlements "$ENTITLEMENTS_FILE_PATH" -f --deep -s - "$UNIVERSAL_APP_BUNDLE"
|
codesign --entitlements "$ENTITLEMENTS_FILE_PATH" -f --deep -s - "$UNIVERSAL_APP_BUNDLE"
|
||||||
fi
|
fi
|
||||||
|
|
@@ -36,4 +36,9 @@ sleep 1
|
|||||||
# Now replace and reopen.
|
# Now replace and reopen.
|
||||||
rm -rf "$INSTALL_DIRECTORY"
|
rm -rf "$INSTALL_DIRECTORY"
|
||||||
mv "$NEW_APP_DIRECTORY" "$INSTALL_DIRECTORY"
|
mv "$NEW_APP_DIRECTORY" "$INSTALL_DIRECTORY"
|
||||||
|
|
||||||
|
if [ "$#" -le 3 ]; then
|
||||||
|
open -a "$INSTALL_DIRECTORY"
|
||||||
|
else
|
||||||
open -a "$INSTALL_DIRECTORY" --args "$APP_ARGUMENTS"
|
open -a "$INSTALL_DIRECTORY" --args "$APP_ARGUMENTS"
|
||||||
|
fi
|
||||||
|
@@ -54,6 +54,7 @@ namespace ARMeilleure.Translation
|
|||||||
internal TranslatorQueue Queue { get; }
|
internal TranslatorQueue Queue { get; }
|
||||||
internal IMemoryManager Memory { get; }
|
internal IMemoryManager Memory { get; }
|
||||||
|
|
||||||
|
private Thread[] _backgroundTranslationThreads;
|
||||||
private volatile int _threadCount;
|
private volatile int _threadCount;
|
||||||
|
|
||||||
// FIXME: Remove this once the init logic of the emulator will be redone.
|
// FIXME: Remove this once the init logic of the emulator will be redone.
|
||||||
@@ -127,18 +128,22 @@ namespace ARMeilleure.Translation
|
|||||||
int unboundedThreadCount = Math.Max(1, (Environment.ProcessorCount - 6) / 3);
|
int unboundedThreadCount = Math.Max(1, (Environment.ProcessorCount - 6) / 3);
|
||||||
int threadCount = Math.Min(4, unboundedThreadCount);
|
int threadCount = Math.Min(4, unboundedThreadCount);
|
||||||
|
|
||||||
|
Thread[] backgroundTranslationThreads = new Thread[threadCount];
|
||||||
|
|
||||||
for (int i = 0; i < threadCount; i++)
|
for (int i = 0; i < threadCount; i++)
|
||||||
{
|
{
|
||||||
bool last = i != 0 && i == unboundedThreadCount - 1;
|
bool last = i != 0 && i == unboundedThreadCount - 1;
|
||||||
|
|
||||||
Thread backgroundTranslatorThread = new Thread(BackgroundTranslate)
|
backgroundTranslationThreads[i] = new Thread(BackgroundTranslate)
|
||||||
{
|
{
|
||||||
Name = "CPU.BackgroundTranslatorThread." + i,
|
Name = "CPU.BackgroundTranslatorThread." + i,
|
||||||
Priority = last ? ThreadPriority.Lowest : ThreadPriority.Normal
|
Priority = last ? ThreadPriority.Lowest : ThreadPriority.Normal
|
||||||
};
|
};
|
||||||
|
|
||||||
backgroundTranslatorThread.Start();
|
backgroundTranslationThreads[i].Start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Interlocked.Exchange(ref _backgroundTranslationThreads, backgroundTranslationThreads);
|
||||||
}
|
}
|
||||||
|
|
||||||
Statistics.InitializeTimer();
|
Statistics.InitializeTimer();
|
||||||
@@ -162,9 +167,20 @@ namespace ARMeilleure.Translation
|
|||||||
|
|
||||||
if (Interlocked.Decrement(ref _threadCount) == 0)
|
if (Interlocked.Decrement(ref _threadCount) == 0)
|
||||||
{
|
{
|
||||||
|
Queue.Dispose();
|
||||||
|
|
||||||
|
Thread[] backgroundTranslationThreads = Interlocked.Exchange(ref _backgroundTranslationThreads, null);
|
||||||
|
|
||||||
|
if (backgroundTranslationThreads != null)
|
||||||
|
{
|
||||||
|
foreach (Thread thread in backgroundTranslationThreads)
|
||||||
|
{
|
||||||
|
thread.Join();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ClearJitCache();
|
ClearJitCache();
|
||||||
|
|
||||||
Queue.Dispose();
|
|
||||||
Stubs.Dispose();
|
Stubs.Dispose();
|
||||||
FunctionTable.Dispose();
|
FunctionTable.Dispose();
|
||||||
CountTable.Dispose();
|
CountTable.Dispose();
|
||||||
|
@@ -5,6 +5,7 @@ using Ryujinx.Memory;
|
|||||||
using Ryujinx.SDL2.Common;
|
using Ryujinx.SDL2.Common;
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Concurrent;
|
using System.Collections.Concurrent;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
|
|
||||||
using static Ryujinx.Audio.Integration.IHardwareDeviceDriver;
|
using static Ryujinx.Audio.Integration.IHardwareDeviceDriver;
|
||||||
@@ -18,6 +19,13 @@ namespace Ryujinx.Audio.Backends.SDL2
|
|||||||
private readonly ManualResetEvent _pauseEvent;
|
private readonly ManualResetEvent _pauseEvent;
|
||||||
private readonly ConcurrentDictionary<SDL2HardwareDeviceSession, byte> _sessions;
|
private readonly ConcurrentDictionary<SDL2HardwareDeviceSession, byte> _sessions;
|
||||||
|
|
||||||
|
private bool _supportSurroundConfiguration;
|
||||||
|
|
||||||
|
// TODO: Add this to SDL2-CS
|
||||||
|
// NOTE: We use a DllImport here because of marshaling issue for spec.
|
||||||
|
[DllImport("SDL2")]
|
||||||
|
private static extern int SDL_GetDefaultAudioInfo(IntPtr name, out SDL_AudioSpec spec, int isCapture);
|
||||||
|
|
||||||
public SDL2HardwareDeviceDriver()
|
public SDL2HardwareDeviceDriver()
|
||||||
{
|
{
|
||||||
_updateRequiredEvent = new ManualResetEvent(false);
|
_updateRequiredEvent = new ManualResetEvent(false);
|
||||||
@@ -25,6 +33,20 @@ namespace Ryujinx.Audio.Backends.SDL2
|
|||||||
_sessions = new ConcurrentDictionary<SDL2HardwareDeviceSession, byte>();
|
_sessions = new ConcurrentDictionary<SDL2HardwareDeviceSession, byte>();
|
||||||
|
|
||||||
SDL2Driver.Instance.Initialize();
|
SDL2Driver.Instance.Initialize();
|
||||||
|
|
||||||
|
int res = SDL_GetDefaultAudioInfo(IntPtr.Zero, out var spec, 0);
|
||||||
|
|
||||||
|
if (res != 0)
|
||||||
|
{
|
||||||
|
Logger.Error?.Print(LogClass.Application,
|
||||||
|
$"SDL_GetDefaultAudioInfo failed with error \"{SDL_GetError()}\"");
|
||||||
|
|
||||||
|
_supportSurroundConfiguration = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_supportSurroundConfiguration = spec.channels == 6;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool IsSupported => IsSupportedInternal();
|
public static bool IsSupported => IsSupportedInternal();
|
||||||
@@ -164,6 +186,11 @@ namespace Ryujinx.Audio.Backends.SDL2
|
|||||||
|
|
||||||
public bool SupportsChannelCount(uint channelCount)
|
public bool SupportsChannelCount(uint channelCount)
|
||||||
{
|
{
|
||||||
|
if (channelCount == 6)
|
||||||
|
{
|
||||||
|
return _supportSurroundConfiguration;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -45,7 +45,7 @@ namespace Ryujinx.Audio.Renderer.Device
|
|||||||
/// <param name="name">The name of the <see cref="VirtualDevice"/>.</param>
|
/// <param name="name">The name of the <see cref="VirtualDevice"/>.</param>
|
||||||
/// <param name="channelCount">The count of channels supported by the <see cref="VirtualDevice"/>.</param>
|
/// <param name="channelCount">The count of channels supported by the <see cref="VirtualDevice"/>.</param>
|
||||||
/// <param name="isExternalOutput">Indicate if the <see cref="VirtualDevice"/> is provided by an external interface.</param>
|
/// <param name="isExternalOutput">Indicate if the <see cref="VirtualDevice"/> is provided by an external interface.</param>
|
||||||
private VirtualDevice(string name, uint channelCount, bool isExternalOutput)
|
public VirtualDevice(string name, uint channelCount, bool isExternalOutput)
|
||||||
{
|
{
|
||||||
Name = name;
|
Name = name;
|
||||||
ChannelCount = channelCount;
|
ChannelCount = channelCount;
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
using Ryujinx.Audio.Integration;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
|
||||||
namespace Ryujinx.Audio.Renderer.Device
|
namespace Ryujinx.Audio.Renderer.Device
|
||||||
@@ -22,7 +23,23 @@ namespace Ryujinx.Audio.Renderer.Device
|
|||||||
/// The current active <see cref="VirtualDevice"/>.
|
/// The current active <see cref="VirtualDevice"/>.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
// TODO: make this configurable
|
// TODO: make this configurable
|
||||||
public VirtualDevice ActiveDevice = VirtualDevice.Devices[2];
|
public VirtualDevice ActiveDevice { get; }
|
||||||
|
|
||||||
|
public VirtualDeviceSessionRegistry(IHardwareDeviceDriver driver)
|
||||||
|
{
|
||||||
|
uint channelCount;
|
||||||
|
|
||||||
|
if (driver.GetRealDeviceDriver().SupportsChannelCount(6))
|
||||||
|
{
|
||||||
|
channelCount = 6;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
channelCount = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
ActiveDevice = new VirtualDevice("AudioTvOutput", channelCount, false);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Get the associated <see cref="T:VirtualDeviceSession[]"/> from an AppletResourceId.
|
/// Get the associated <see cref="T:VirtualDeviceSession[]"/> from an AppletResourceId.
|
||||||
|
@@ -65,9 +65,7 @@ namespace Ryujinx.Audio.Renderer.Dsp
|
|||||||
{
|
{
|
||||||
OutputDevices = new IHardwareDevice[Constants.AudioRendererSessionCountMax];
|
OutputDevices = new IHardwareDevice[Constants.AudioRendererSessionCountMax];
|
||||||
|
|
||||||
// TODO: Before enabling this, we need up-mixing from stereo to 5.1.
|
uint channelCount = GetHardwareChannelCount(deviceDriver);
|
||||||
// uint channelCount = GetHardwareChannelCount(deviceDriver);
|
|
||||||
uint channelCount = 2;
|
|
||||||
|
|
||||||
for (int i = 0; i < OutputDevices.Length; i++)
|
for (int i = 0; i < OutputDevices.Length; i++)
|
||||||
{
|
{
|
||||||
|
@@ -49,8 +49,8 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
OutputBufferIndices[i] = (ushort)(bufferOffset + Parameter.Output[i]);
|
OutputBufferIndices[i] = (ushort)(bufferOffset + Parameter.Output[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
DataSourceHelper.RemapLegacyChannelEffectMappingToChannelResourceMapping(newEffectChannelMappingSupported, InputBufferIndices);
|
DataSourceHelper.RemapLegacyChannelEffectMappingToChannelResourceMapping(newEffectChannelMappingSupported, InputBufferIndices, Parameter.ChannelCount);
|
||||||
DataSourceHelper.RemapLegacyChannelEffectMappingToChannelResourceMapping(newEffectChannelMappingSupported, OutputBufferIndices);
|
DataSourceHelper.RemapLegacyChannelEffectMappingToChannelResourceMapping(newEffectChannelMappingSupported, OutputBufferIndices, Parameter.ChannelCount);
|
||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining | MethodImplOptions.AggressiveOptimization)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining | MethodImplOptions.AggressiveOptimization)]
|
||||||
|
@@ -67,7 +67,19 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
|
|
||||||
const int sampleCount = Constants.TargetSampleCount;
|
const int sampleCount = Constants.TargetSampleCount;
|
||||||
|
|
||||||
short[] outputBuffer = new short[bufferCount * sampleCount];
|
uint inputCount;
|
||||||
|
|
||||||
|
// In case of upmixing to 5.1, we allocate the right amount.
|
||||||
|
if (bufferCount != channelCount && channelCount == 6)
|
||||||
|
{
|
||||||
|
inputCount = (uint)channelCount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
inputCount = bufferCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
short[] outputBuffer = new short[inputCount * sampleCount];
|
||||||
|
|
||||||
for (int i = 0; i < bufferCount; i++)
|
for (int i = 0; i < bufferCount; i++)
|
||||||
{
|
{
|
||||||
@@ -79,7 +91,7 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
device.AppendBuffer(outputBuffer, InputCount);
|
device.AppendBuffer(outputBuffer, inputCount);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@@ -66,8 +66,8 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
|
|
||||||
// NOTE: We do the opposite as Nintendo here for now to restore previous behaviour
|
// NOTE: We do the opposite as Nintendo here for now to restore previous behaviour
|
||||||
// TODO: Update reverb 3d processing and remove this to use RemapLegacyChannelEffectMappingToChannelResourceMapping.
|
// TODO: Update reverb 3d processing and remove this to use RemapLegacyChannelEffectMappingToChannelResourceMapping.
|
||||||
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, InputBufferIndices);
|
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, InputBufferIndices, Parameter.ChannelCount);
|
||||||
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, OutputBufferIndices);
|
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, OutputBufferIndices, Parameter.ChannelCount);
|
||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
@@ -116,7 +116,7 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
for (int i = 0; i < targetEarlyDelayLineIndicesTable.Length; i++)
|
for (int i = 0; i < targetEarlyDelayLineIndicesTable.Length; i++)
|
||||||
{
|
{
|
||||||
int earlyDelayIndex = targetEarlyDelayLineIndicesTable[i];
|
int earlyDelayIndex = targetEarlyDelayLineIndicesTable[i];
|
||||||
int outputIndex = outputEarlyIndicesTable[i];
|
int outputIndex = outputEarlyIndicesTable[earlyDelayIndex];
|
||||||
|
|
||||||
float tempTapOut = state.PreDelayLine.TapUnsafe(state.EarlyDelayTime[earlyDelayIndex], delayLineSampleIndexOffset);
|
float tempTapOut = state.PreDelayLine.TapUnsafe(state.EarlyDelayTime[earlyDelayIndex], delayLineSampleIndexOffset);
|
||||||
|
|
||||||
|
@@ -71,8 +71,8 @@ namespace Ryujinx.Audio.Renderer.Dsp.Command
|
|||||||
|
|
||||||
// NOTE: We do the opposite as Nintendo here for now to restore previous behaviour
|
// NOTE: We do the opposite as Nintendo here for now to restore previous behaviour
|
||||||
// TODO: Update reverb processing and remove this to use RemapLegacyChannelEffectMappingToChannelResourceMapping.
|
// TODO: Update reverb processing and remove this to use RemapLegacyChannelEffectMappingToChannelResourceMapping.
|
||||||
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, InputBufferIndices);
|
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, InputBufferIndices, Parameter.ChannelCount);
|
||||||
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, OutputBufferIndices);
|
DataSourceHelper.RemapChannelResourceMappingToLegacy(newEffectChannelMappingSupported, OutputBufferIndices, Parameter.ChannelCount);
|
||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
@@ -430,9 +430,9 @@ namespace Ryujinx.Audio.Renderer.Dsp
|
|||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public static void RemapLegacyChannelEffectMappingToChannelResourceMapping(bool isSupported, Span<ushort> bufferIndices)
|
public static void RemapLegacyChannelEffectMappingToChannelResourceMapping(bool isSupported, Span<ushort> bufferIndices, uint channelCount)
|
||||||
{
|
{
|
||||||
if (!isSupported && bufferIndices.Length == 6)
|
if (!isSupported && channelCount == 6)
|
||||||
{
|
{
|
||||||
ushort backLeft = bufferIndices[2];
|
ushort backLeft = bufferIndices[2];
|
||||||
ushort backRight = bufferIndices[3];
|
ushort backRight = bufferIndices[3];
|
||||||
@@ -447,9 +447,9 @@ namespace Ryujinx.Audio.Renderer.Dsp
|
|||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
public static void RemapChannelResourceMappingToLegacy(bool isSupported, Span<ushort> bufferIndices)
|
public static void RemapChannelResourceMappingToLegacy(bool isSupported, Span<ushort> bufferIndices, uint channelCount)
|
||||||
{
|
{
|
||||||
if (isSupported && bufferIndices.Length == 6)
|
if (isSupported && channelCount == 6)
|
||||||
{
|
{
|
||||||
ushort frontCenter = bufferIndices[2];
|
ushort frontCenter = bufferIndices[2];
|
||||||
ushort lowFrequency = bufferIndices[3];
|
ushort lowFrequency = bufferIndices[3];
|
||||||
|
@@ -86,7 +86,7 @@ namespace Ryujinx.Ava
|
|||||||
private KeyboardHotkeyState _prevHotkeyState;
|
private KeyboardHotkeyState _prevHotkeyState;
|
||||||
|
|
||||||
private long _lastCursorMoveTime;
|
private long _lastCursorMoveTime;
|
||||||
private bool _isCursorInRenderer;
|
private bool _isCursorInRenderer = true;
|
||||||
|
|
||||||
private bool _isStopped;
|
private bool _isStopped;
|
||||||
private bool _isActive;
|
private bool _isActive;
|
||||||
@@ -160,7 +160,9 @@ namespace Ryujinx.Ava
|
|||||||
|
|
||||||
ConfigurationState.Instance.HideCursor.Event += HideCursorState_Changed;
|
ConfigurationState.Instance.HideCursor.Event += HideCursorState_Changed;
|
||||||
|
|
||||||
_topLevel.PointerMoved += TopLevel_PointerMoved;
|
_topLevel.PointerMoved += TopLevel_PointerEnterOrMoved;
|
||||||
|
_topLevel.PointerEnter += TopLevel_PointerEnterOrMoved;
|
||||||
|
_topLevel.PointerLeave += TopLevel_PointerLeave;
|
||||||
|
|
||||||
if (OperatingSystem.IsWindows())
|
if (OperatingSystem.IsWindows())
|
||||||
{
|
{
|
||||||
@@ -183,7 +185,7 @@ namespace Ryujinx.Ava
|
|||||||
_gpuCancellationTokenSource = new CancellationTokenSource();
|
_gpuCancellationTokenSource = new CancellationTokenSource();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void TopLevel_PointerMoved(object sender, PointerEventArgs e)
|
private void TopLevel_PointerEnterOrMoved(object sender, PointerEventArgs e)
|
||||||
{
|
{
|
||||||
if (sender is MainWindow window)
|
if (sender is MainWindow window)
|
||||||
{
|
{
|
||||||
@@ -201,6 +203,12 @@ namespace Ryujinx.Ava
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void TopLevel_PointerLeave(object sender, PointerEventArgs e)
|
||||||
|
{
|
||||||
|
_isCursorInRenderer = false;
|
||||||
|
}
|
||||||
|
|
||||||
private void UpdateScalingFilterLevel(object sender, ReactiveEventArgs<int> e)
|
private void UpdateScalingFilterLevel(object sender, ReactiveEventArgs<int> e)
|
||||||
{
|
{
|
||||||
_renderer.Window?.SetScalingFilter((Graphics.GAL.ScalingFilter)ConfigurationState.Instance.Graphics.ScalingFilter.Value);
|
_renderer.Window?.SetScalingFilter((Graphics.GAL.ScalingFilter)ConfigurationState.Instance.Graphics.ScalingFilter.Value);
|
||||||
@@ -446,7 +454,9 @@ namespace Ryujinx.Ava
|
|||||||
ConfigurationState.Instance.Graphics.ScalingFilterLevel.Event -= UpdateScalingFilterLevel;
|
ConfigurationState.Instance.Graphics.ScalingFilterLevel.Event -= UpdateScalingFilterLevel;
|
||||||
ConfigurationState.Instance.Graphics.AntiAliasing.Event -= UpdateAntiAliasing;
|
ConfigurationState.Instance.Graphics.AntiAliasing.Event -= UpdateAntiAliasing;
|
||||||
|
|
||||||
_topLevel.PointerMoved -= TopLevel_PointerMoved;
|
_topLevel.PointerMoved -= TopLevel_PointerEnterOrMoved;
|
||||||
|
_topLevel.PointerEnter -= TopLevel_PointerEnterOrMoved;
|
||||||
|
_topLevel.PointerLeave -= TopLevel_PointerLeave;
|
||||||
|
|
||||||
_gpuCancellationTokenSource.Cancel();
|
_gpuCancellationTokenSource.Cancel();
|
||||||
_gpuCancellationTokenSource.Dispose();
|
_gpuCancellationTokenSource.Dispose();
|
||||||
@@ -671,7 +681,7 @@ namespace Ryujinx.Ava
|
|||||||
|
|
||||||
_viewModel.ApplicationLibrary.LoadAndSaveMetaData(Device.Processes.ActiveApplication.ProgramIdText, appMetadata =>
|
_viewModel.ApplicationLibrary.LoadAndSaveMetaData(Device.Processes.ActiveApplication.ProgramIdText, appMetadata =>
|
||||||
{
|
{
|
||||||
appMetadata.LastPlayed = DateTime.UtcNow.ToString();
|
appMetadata.LastPlayed = DateTime.UtcNow;
|
||||||
});
|
});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@@ -193,7 +193,7 @@ namespace Ryujinx.Ava.Common
|
|||||||
if (nca.Header.ContentType == NcaContentType.Program)
|
if (nca.Header.ContentType == NcaContentType.Program)
|
||||||
{
|
{
|
||||||
int dataIndex = Nca.GetSectionIndexFromType(NcaSectionType.Data, NcaContentType.Program);
|
int dataIndex = Nca.GetSectionIndexFromType(NcaSectionType.Data, NcaContentType.Program);
|
||||||
if (nca.Header.GetFsHeader(dataIndex).IsPatchSection())
|
if (nca.SectionExists(NcaSectionType.Data) && nca.Header.GetFsHeader(dataIndex).IsPatchSection())
|
||||||
{
|
{
|
||||||
patchNca = nca;
|
patchNca = nca;
|
||||||
}
|
}
|
||||||
|
@@ -70,13 +70,16 @@ namespace Ryujinx.Ava.Input
|
|||||||
|
|
||||||
private void Parent_PointerReleaseEvent(object o, PointerReleasedEventArgs args)
|
private void Parent_PointerReleaseEvent(object o, PointerReleasedEventArgs args)
|
||||||
{
|
{
|
||||||
int button = (int)args.InitialPressMouseButton - 1;
|
if (args.InitialPressMouseButton != Avalonia.Input.MouseButton.None)
|
||||||
|
{
|
||||||
|
int button = (int)args.InitialPressMouseButton;
|
||||||
|
|
||||||
if (PressedButtons.Count() >= button)
|
if (PressedButtons.Count() >= button)
|
||||||
{
|
{
|
||||||
PressedButtons[button] = false;
|
PressedButtons[button] = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void Parent_PointerPressEvent(object o, PointerPressedEventArgs args)
|
private void Parent_PointerPressEvent(object o, PointerPressedEventArgs args)
|
||||||
{
|
{
|
||||||
|
@@ -295,14 +295,7 @@ namespace Ryujinx.Modules
|
|||||||
if (shouldRestart)
|
if (shouldRestart)
|
||||||
{
|
{
|
||||||
List<string> arguments = CommandLineState.Arguments.ToList();
|
List<string> arguments = CommandLineState.Arguments.ToList();
|
||||||
string ryuName = Path.GetFileName(Environment.ProcessPath);
|
|
||||||
string executableDirectory = AppDomain.CurrentDomain.BaseDirectory;
|
string executableDirectory = AppDomain.CurrentDomain.BaseDirectory;
|
||||||
string executablePath = Path.Combine(executableDirectory, ryuName);
|
|
||||||
|
|
||||||
if (!Path.Exists(executablePath))
|
|
||||||
{
|
|
||||||
executablePath = Path.Combine(executableDirectory, OperatingSystem.IsWindows() ? "Ryujinx.exe" : "Ryujinx");
|
|
||||||
}
|
|
||||||
|
|
||||||
// On macOS we perform the update at relaunch.
|
// On macOS we perform the update at relaunch.
|
||||||
if (OperatingSystem.IsMacOS())
|
if (OperatingSystem.IsMacOS())
|
||||||
@@ -310,13 +303,42 @@ namespace Ryujinx.Modules
|
|||||||
string baseBundlePath = Path.GetFullPath(Path.Combine(executableDirectory, "..", ".."));
|
string baseBundlePath = Path.GetFullPath(Path.Combine(executableDirectory, "..", ".."));
|
||||||
string newBundlePath = Path.Combine(UpdateDir, "Ryujinx.app");
|
string newBundlePath = Path.Combine(UpdateDir, "Ryujinx.app");
|
||||||
string updaterScriptPath = Path.Combine(newBundlePath, "Contents", "Resources", "updater.sh");
|
string updaterScriptPath = Path.Combine(newBundlePath, "Contents", "Resources", "updater.sh");
|
||||||
string currentPid = Process.GetCurrentProcess().Id.ToString();
|
string currentPid = Environment.ProcessId.ToString();
|
||||||
|
|
||||||
executablePath = "/bin/bash";
|
|
||||||
arguments.InsertRange(0, new List<string> { updaterScriptPath, baseBundlePath, newBundlePath, currentPid });
|
arguments.InsertRange(0, new List<string> { updaterScriptPath, baseBundlePath, newBundlePath, currentPid });
|
||||||
|
Process.Start("/bin/bash", arguments);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Find the process name.
|
||||||
|
string ryuName = Path.GetFileName(Environment.ProcessPath);
|
||||||
|
|
||||||
|
// Some operating systems can see the renamed executable, so strip off the .ryuold if found.
|
||||||
|
if (ryuName.EndsWith(".ryuold"))
|
||||||
|
{
|
||||||
|
ryuName = ryuName[..^7];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback if the executable could not be found.
|
||||||
|
if (!Path.Exists(Path.Combine(executableDirectory, ryuName)))
|
||||||
|
{
|
||||||
|
ryuName = OperatingSystem.IsWindows() ? "Ryujinx.Ava.exe" : "Ryujinx.Ava";
|
||||||
|
}
|
||||||
|
|
||||||
|
ProcessStartInfo processStart = new(ryuName)
|
||||||
|
{
|
||||||
|
UseShellExecute = true,
|
||||||
|
WorkingDirectory = executableDirectory
|
||||||
|
};
|
||||||
|
|
||||||
|
foreach (string argument in CommandLineState.Arguments)
|
||||||
|
{
|
||||||
|
processStart.ArgumentList.Add(argument);
|
||||||
|
}
|
||||||
|
|
||||||
|
Process.Start(processStart);
|
||||||
}
|
}
|
||||||
|
|
||||||
Process.Start(executablePath, arguments);
|
|
||||||
Environment.Exit(0);
|
Environment.Exit(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -129,7 +129,7 @@
|
|||||||
TextWrapping="Wrap" />
|
TextWrapping="Wrap" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
Text="{Binding LastPlayed}"
|
Text="{Binding LastPlayed, Converter={helpers:NullableDateTimeConverter}}"
|
||||||
TextAlignment="Right"
|
TextAlignment="Right"
|
||||||
TextWrapping="Wrap" />
|
TextWrapping="Wrap" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
|
38
src/Ryujinx.Ava/UI/Helpers/NullableDateTimeConverter.cs
Normal file
38
src/Ryujinx.Ava/UI/Helpers/NullableDateTimeConverter.cs
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
using Avalonia.Data.Converters;
|
||||||
|
using Avalonia.Markup.Xaml;
|
||||||
|
using Ryujinx.Ava.Common.Locale;
|
||||||
|
using System;
|
||||||
|
using System.Globalization;
|
||||||
|
|
||||||
|
namespace Ryujinx.Ava.UI.Helpers
|
||||||
|
{
|
||||||
|
internal class NullableDateTimeConverter : MarkupExtension, IValueConverter
|
||||||
|
{
|
||||||
|
private static readonly NullableDateTimeConverter _instance = new();
|
||||||
|
|
||||||
|
public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
|
||||||
|
{
|
||||||
|
if (value == null)
|
||||||
|
{
|
||||||
|
return LocaleManager.Instance[LocaleKeys.Never];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value is DateTime dateTime)
|
||||||
|
{
|
||||||
|
return dateTime.ToLocalTime().ToString(culture);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new NotSupportedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
|
||||||
|
{
|
||||||
|
throw new NotSupportedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public override object ProvideValue(IServiceProvider serviceProvider)
|
||||||
|
{
|
||||||
|
return _instance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -1,4 +1,3 @@
|
|||||||
using Ryujinx.Ava.Common.Locale;
|
|
||||||
using Ryujinx.Ui.App.Common;
|
using Ryujinx.Ui.App.Common;
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
@@ -14,20 +13,20 @@ namespace Ryujinx.Ava.UI.Models.Generic
|
|||||||
|
|
||||||
public int Compare(ApplicationData x, ApplicationData y)
|
public int Compare(ApplicationData x, ApplicationData y)
|
||||||
{
|
{
|
||||||
string aValue = x.LastPlayed;
|
var aValue = x.LastPlayed;
|
||||||
string bValue = y.LastPlayed;
|
var bValue = y.LastPlayed;
|
||||||
|
|
||||||
if (aValue == LocaleManager.Instance[LocaleKeys.Never])
|
if (!aValue.HasValue)
|
||||||
{
|
{
|
||||||
aValue = DateTime.UnixEpoch.ToString();
|
aValue = DateTime.UnixEpoch;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bValue == LocaleManager.Instance[LocaleKeys.Never])
|
if (!bValue.HasValue)
|
||||||
{
|
{
|
||||||
bValue = DateTime.UnixEpoch.ToString();
|
bValue = DateTime.UnixEpoch;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (IsAscending ? 1 : -1) * DateTime.Compare(DateTime.Parse(bValue), DateTime.Parse(aValue));
|
return (IsAscending ? 1 : -1) * DateTime.Compare(bValue.Value, aValue.Value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -370,7 +370,7 @@ namespace Ryujinx.Ava.UI.ViewModels
|
|||||||
|
|
||||||
if (response.IsSuccessStatusCode)
|
if (response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
return response.Content.Headers.LastModified != oldLastModified;
|
return response.Content.Headers.LastModified != new DateTimeOffset(oldLastModified.Ticks - (oldLastModified.Ticks % TimeSpan.TicksPerSecond), TimeSpan.Zero);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@@ -1524,10 +1524,9 @@ namespace Ryujinx.Ava.UI.ViewModels
|
|||||||
{
|
{
|
||||||
ApplicationLibrary.LoadAndSaveMetaData(titleId, appMetadata =>
|
ApplicationLibrary.LoadAndSaveMetaData(titleId, appMetadata =>
|
||||||
{
|
{
|
||||||
if (DateTime.TryParse(appMetadata.LastPlayed, out DateTime lastPlayedDateTime))
|
if (appMetadata.LastPlayed.HasValue)
|
||||||
{
|
{
|
||||||
double sessionTimePlayed = DateTime.UtcNow.Subtract(lastPlayedDateTime).TotalSeconds;
|
double sessionTimePlayed = DateTime.UtcNow.Subtract(appMetadata.LastPlayed.Value).TotalSeconds;
|
||||||
|
|
||||||
appMetadata.TimePlayed += Math.Round(sessionTimePlayed, MidpointRounding.AwayFromZero);
|
appMetadata.TimePlayed += Math.Round(sessionTimePlayed, MidpointRounding.AwayFromZero);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@@ -21,35 +21,54 @@
|
|||||||
<UserControl.Resources>
|
<UserControl.Resources>
|
||||||
<helpers:KeyValueConverter x:Key="Key" />
|
<helpers:KeyValueConverter x:Key="Key" />
|
||||||
</UserControl.Resources>
|
</UserControl.Resources>
|
||||||
|
<UserControl.Styles>
|
||||||
|
<Style Selector="ToggleButton">
|
||||||
|
<Setter Property="Width" Value="90" />
|
||||||
|
<Setter Property="Height" Value="27" />
|
||||||
|
<Setter Property="HorizontalAlignment" Value="Stretch" />
|
||||||
|
</Style>
|
||||||
|
</UserControl.Styles>
|
||||||
|
|
||||||
<StackPanel
|
<StackPanel
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
VerticalAlignment="Stretch"
|
VerticalAlignment="Stretch"
|
||||||
Orientation="Vertical">
|
Orientation="Vertical">
|
||||||
<Grid Margin="2,2,2,5" HorizontalAlignment="Stretch">
|
<Grid Margin="0,2,0,5" HorizontalAlignment="Stretch">
|
||||||
|
<Grid.ColumnDefinitions>
|
||||||
|
<ColumnDefinition Width="*" />
|
||||||
|
<ColumnDefinition Width="*" />
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
|
<Grid.RowDefinitions>
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
</Grid.RowDefinitions>
|
||||||
|
|
||||||
|
<!-- Player selection -->
|
||||||
|
<Border
|
||||||
|
Grid.Row="0"
|
||||||
|
Grid.Column="0"
|
||||||
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
|
BorderThickness="1"
|
||||||
|
Padding="2">
|
||||||
|
<Grid
|
||||||
|
Margin="2"
|
||||||
|
HorizontalAlignment="Stretch"
|
||||||
|
VerticalAlignment="Center">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="Auto"/>
|
<ColumnDefinition Width="Auto"/>
|
||||||
<ColumnDefinition Width="*" />
|
<ColumnDefinition Width="*" />
|
||||||
<ColumnDefinition Width="0.5*" />
|
|
||||||
<ColumnDefinition Width="Auto" />
|
|
||||||
</Grid.ColumnDefinitions>
|
</Grid.ColumnDefinitions>
|
||||||
<Border
|
|
||||||
Grid.Column="0"
|
|
||||||
Margin="0,0,2,0"
|
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
|
||||||
BorderThickness="1"
|
|
||||||
Padding="2,0">
|
|
||||||
<StackPanel
|
|
||||||
Margin="2"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Orientation="Vertical">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,4"
|
Margin="5,0,10,0"
|
||||||
HorizontalAlignment="Center"
|
Width="90"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsPlayer}" />
|
Text="{locale:Locale ControllerSettingsPlayer}" />
|
||||||
<ComboBox
|
<ComboBox
|
||||||
|
Grid.Column="1"
|
||||||
Name="PlayerIndexBox"
|
Name="PlayerIndexBox"
|
||||||
Width="150"
|
HorizontalAlignment="Stretch"
|
||||||
|
VerticalAlignment="Center"
|
||||||
SelectionChanged="PlayerIndexBox_OnSelectionChanged"
|
SelectionChanged="PlayerIndexBox_OnSelectionChanged"
|
||||||
Items="{Binding PlayerIndexes}"
|
Items="{Binding PlayerIndexes}"
|
||||||
SelectedIndex="{Binding PlayerId}">
|
SelectedIndex="{Binding PlayerId}">
|
||||||
@@ -59,103 +78,44 @@
|
|||||||
</DataTemplate>
|
</DataTemplate>
|
||||||
</ComboBox.ItemTemplate>
|
</ComboBox.ItemTemplate>
|
||||||
</ComboBox>
|
</ComboBox>
|
||||||
</StackPanel>
|
|
||||||
</Border>
|
|
||||||
|
|
||||||
<!-- Main Controller Settings -->
|
|
||||||
<Border
|
|
||||||
Grid.Column="1"
|
|
||||||
Margin="0,0,2,0"
|
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
|
||||||
BorderThickness="1"
|
|
||||||
Padding="2,0">
|
|
||||||
<StackPanel
|
|
||||||
Margin="2"
|
|
||||||
HorizontalAlignment="Stretch"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Orientation="Vertical">
|
|
||||||
<TextBlock
|
|
||||||
Margin="0,0,0,5"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
Text="{locale:Locale ControllerSettingsInputDevice}" />
|
|
||||||
<Grid HorizontalAlignment="Stretch">
|
|
||||||
<Grid.ColumnDefinitions>
|
|
||||||
<ColumnDefinition />
|
|
||||||
<ColumnDefinition Width="Auto" />
|
|
||||||
</Grid.ColumnDefinitions>
|
|
||||||
<ComboBox
|
|
||||||
Name="DeviceBox"
|
|
||||||
HorizontalAlignment="Stretch"
|
|
||||||
Items="{Binding DeviceList}"
|
|
||||||
SelectedIndex="{Binding Device}" />
|
|
||||||
<Button
|
|
||||||
Grid.Column="1"
|
|
||||||
MinWidth="0"
|
|
||||||
Margin="5,0,0,0"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Command="{Binding LoadDevices}">
|
|
||||||
<ui:SymbolIcon
|
|
||||||
Symbol="Refresh"
|
|
||||||
FontSize="15"
|
|
||||||
Height="20" />
|
|
||||||
</Button>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
</StackPanel>
|
|
||||||
</Border>
|
</Border>
|
||||||
|
<!-- Profile selection -->
|
||||||
<Border
|
<Border
|
||||||
Grid.Column="2"
|
Grid.Row="1"
|
||||||
Margin="0,0,2,0"
|
Grid.Column="0"
|
||||||
|
Margin="0,-1,0,0"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
Padding="2,0">
|
Padding="2">
|
||||||
<Grid
|
<Grid
|
||||||
Margin="2"
|
Margin="2"
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
VerticalAlignment="Center">
|
VerticalAlignment="Center">
|
||||||
<Grid.RowDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<RowDefinition />
|
<ColumnDefinition Width="Auto"/>
|
||||||
<RowDefinition />
|
<ColumnDefinition Width="*" />
|
||||||
</Grid.RowDefinitions>
|
<ColumnDefinition Width="Auto"/>
|
||||||
|
<ColumnDefinition Width="Auto"/>
|
||||||
|
<ColumnDefinition Width="Auto"/>
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,4"
|
Margin="5,0,10,0"
|
||||||
HorizontalAlignment="Center"
|
Width="90"
|
||||||
Text="{locale:Locale ControllerSettingsControllerType}" />
|
HorizontalAlignment="Left"
|
||||||
<ComboBox
|
|
||||||
Grid.Row="1"
|
|
||||||
HorizontalAlignment="Stretch"
|
|
||||||
Items="{Binding Controllers}"
|
|
||||||
SelectedIndex="{Binding Controller}">
|
|
||||||
<ComboBox.ItemTemplate>
|
|
||||||
<DataTemplate>
|
|
||||||
<TextBlock Text="{Binding Name}" />
|
|
||||||
</DataTemplate>
|
|
||||||
</ComboBox.ItemTemplate>
|
|
||||||
</ComboBox>
|
|
||||||
</Grid>
|
|
||||||
</Border>
|
|
||||||
<Border
|
|
||||||
Grid.Column="3"
|
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
|
||||||
BorderThickness="1"
|
|
||||||
Padding="2,0" >
|
|
||||||
<StackPanel
|
|
||||||
Margin="2"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Orientation="Vertical">
|
|
||||||
<TextBlock
|
|
||||||
Margin="0,0,0,4"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
Text="{locale:Locale ControllerSettingsProfile}" />
|
Text="{locale:Locale ControllerSettingsProfile}" />
|
||||||
<StackPanel Orientation="Horizontal">
|
|
||||||
<ui:ComboBox
|
<ui:ComboBox
|
||||||
|
Grid.Column="1"
|
||||||
IsEditable="True"
|
IsEditable="True"
|
||||||
Name="ProfileBox"
|
Name="ProfileBox"
|
||||||
Width="100"
|
HorizontalAlignment="Stretch"
|
||||||
|
VerticalAlignment="Center"
|
||||||
SelectedIndex="0"
|
SelectedIndex="0"
|
||||||
Items="{Binding ProfilesList}"
|
Items="{Binding ProfilesList}"
|
||||||
Text="{Binding ProfileName}" />
|
Text="{Binding ProfileName}" />
|
||||||
<Button
|
<Button
|
||||||
|
Grid.Column="2"
|
||||||
MinWidth="0"
|
MinWidth="0"
|
||||||
Margin="5,0,0,0"
|
Margin="5,0,0,0"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -167,6 +127,7 @@
|
|||||||
Height="20" />
|
Height="20" />
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
|
Grid.Column="3"
|
||||||
MinWidth="0"
|
MinWidth="0"
|
||||||
Margin="5,0,0,0"
|
Margin="5,0,0,0"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -178,6 +139,7 @@
|
|||||||
Height="20" />
|
Height="20" />
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
|
Grid.Column="4"
|
||||||
MinWidth="0"
|
MinWidth="0"
|
||||||
Margin="5,0,0,0"
|
Margin="5,0,0,0"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -188,8 +150,85 @@
|
|||||||
FontSize="15"
|
FontSize="15"
|
||||||
Height="20" />
|
Height="20" />
|
||||||
</Button>
|
</Button>
|
||||||
</StackPanel>
|
</Grid>
|
||||||
</StackPanel>
|
</Border>
|
||||||
|
|
||||||
|
<!-- Input device -->
|
||||||
|
<Border
|
||||||
|
Grid.Row="0"
|
||||||
|
Grid.Column="1"
|
||||||
|
Margin="-1,0,0,0"
|
||||||
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
|
BorderThickness="1"
|
||||||
|
Padding="2">
|
||||||
|
<Grid Margin="2" HorizontalAlignment="Stretch">
|
||||||
|
<Grid.ColumnDefinitions>
|
||||||
|
<ColumnDefinition Width="Auto"/>
|
||||||
|
<ColumnDefinition Width="*"/>
|
||||||
|
<ColumnDefinition Width="Auto" />
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
|
<TextBlock
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="5,0,10,0"
|
||||||
|
Width="90"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsInputDevice}" />
|
||||||
|
<ComboBox
|
||||||
|
Grid.Column="1"
|
||||||
|
Name="DeviceBox"
|
||||||
|
HorizontalAlignment="Stretch"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Items="{Binding DeviceList}"
|
||||||
|
SelectedIndex="{Binding Device}" />
|
||||||
|
<Button
|
||||||
|
Grid.Column="2"
|
||||||
|
MinWidth="0"
|
||||||
|
Margin="5,0,0,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Command="{Binding LoadDevices}">
|
||||||
|
<ui:SymbolIcon
|
||||||
|
Symbol="Refresh"
|
||||||
|
FontSize="15"
|
||||||
|
Height="20"/>
|
||||||
|
</Button>
|
||||||
|
</Grid>
|
||||||
|
</Border>
|
||||||
|
|
||||||
|
<!-- Controler type -->
|
||||||
|
<Border
|
||||||
|
Grid.Row="1"
|
||||||
|
Grid.Column="1"
|
||||||
|
Margin="-1,-1,0,0"
|
||||||
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
|
BorderThickness="1"
|
||||||
|
Padding="2">
|
||||||
|
<Grid
|
||||||
|
Margin="2"
|
||||||
|
HorizontalAlignment="Stretch"
|
||||||
|
VerticalAlignment="Center">
|
||||||
|
<Grid.ColumnDefinitions>
|
||||||
|
<ColumnDefinition Width="Auto"/>
|
||||||
|
<ColumnDefinition Width="*" />
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
|
<TextBlock
|
||||||
|
Margin="5,0,10,0"
|
||||||
|
Width="90"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsControllerType}" />
|
||||||
|
<ComboBox
|
||||||
|
Grid.Column="1"
|
||||||
|
HorizontalAlignment="Stretch"
|
||||||
|
Items="{Binding Controllers}"
|
||||||
|
SelectedIndex="{Binding Controller}">
|
||||||
|
<ComboBox.ItemTemplate>
|
||||||
|
<DataTemplate>
|
||||||
|
<TextBlock Text="{Binding Name}" />
|
||||||
|
</DataTemplate>
|
||||||
|
</ComboBox.ItemTemplate>
|
||||||
|
</ComboBox>
|
||||||
|
</Grid>
|
||||||
</Border>
|
</Border>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
@@ -206,7 +245,7 @@
|
|||||||
|
|
||||||
<!-- Left -->
|
<!-- Left -->
|
||||||
<Grid
|
<Grid
|
||||||
Margin="0,0,10,0"
|
Margin="0,0,5,0"
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
VerticalAlignment="Stretch"
|
VerticalAlignment="Stretch"
|
||||||
DockPanel.Dock="Left">
|
DockPanel.Dock="Left">
|
||||||
@@ -221,7 +260,8 @@
|
|||||||
Grid.Row="0"
|
Grid.Row="0"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
IsVisible="{Binding IsLeft}">
|
IsVisible="{Binding IsLeft}"
|
||||||
|
MinHeight="90">
|
||||||
<Grid Margin="10" HorizontalAlignment="Stretch">
|
<Grid Margin="10" HorizontalAlignment="Stretch">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition />
|
<ColumnDefinition />
|
||||||
@@ -232,7 +272,6 @@
|
|||||||
<RowDefinition />
|
<RowDefinition />
|
||||||
</Grid.RowDefinitions>
|
</Grid.RowDefinitions>
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,0,0,4"
|
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
Grid.Row="0"
|
Grid.Row="0"
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
@@ -243,10 +282,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsTriggerZL}"
|
Text="{locale:Locale ControllerSettingsTriggerZL}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonZl, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonZl, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -263,19 +299,15 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsTriggerL}"
|
Text="{locale:Locale ControllerSettingsTriggerL}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonL, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonL, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,0,0,4"
|
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Grid.Row="0"
|
Grid.Row="1"
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
Orientation="Horizontal">
|
Orientation="Horizontal">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
@@ -284,10 +316,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsButtonMinus}"
|
Text="{locale:Locale ControllerSettingsButtonMinus}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonMinus, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonMinus, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -301,7 +330,8 @@
|
|||||||
Grid.Row="1"
|
Grid.Row="1"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
IsVisible="{Binding IsLeft}">
|
IsVisible="{Binding IsLeft}"
|
||||||
|
Margin="0,5,0,0">
|
||||||
<StackPanel Margin="10" Orientation="Vertical">
|
<StackPanel Margin="10" Orientation="Vertical">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,10"
|
Margin="0,0,0,10"
|
||||||
@@ -320,10 +350,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickButton}"
|
Text="{locale:Locale ControllerSettingsLStickButton}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftKeyboardStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftKeyboardStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -339,10 +366,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickUp}"
|
Text="{locale:Locale ControllerSettingsLStickUp}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftStickUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftStickUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -358,10 +382,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickDown}"
|
Text="{locale:Locale ControllerSettingsLStickDown}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftStickDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftStickDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -377,10 +398,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickLeft}"
|
Text="{locale:Locale ControllerSettingsLStickLeft}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftStickLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftStickLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -396,10 +414,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickRight}"
|
Text="{locale:Locale ControllerSettingsLStickRight}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftStickRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftStickRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -419,10 +434,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickButton}"
|
Text="{locale:Locale ControllerSettingsLStickButton}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftControllerStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftControllerStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -438,16 +450,13 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLStickStick}"
|
Text="{locale:Locale ControllerSettingsLStickStick}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton Tag="stick">
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch"
|
|
||||||
Tag="stick">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftJoystick, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftJoystick, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
|
<Separator Margin="0,8,0,8" Height="1" />
|
||||||
<CheckBox IsChecked="{Binding Configuration.LeftInvertStickX}">
|
<CheckBox IsChecked="{Binding Configuration.LeftInvertStickX}">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsLStickInvertXAxis}" />
|
<TextBlock Text="{locale:Locale ControllerSettingsLStickInvertXAxis}" />
|
||||||
</CheckBox>
|
</CheckBox>
|
||||||
@@ -457,9 +466,11 @@
|
|||||||
<CheckBox IsChecked="{Binding Configuration.LeftRotate90}">
|
<CheckBox IsChecked="{Binding Configuration.LeftRotate90}">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsRotate90}" />
|
<TextBlock Text="{locale:Locale ControllerSettingsRotate90}" />
|
||||||
</CheckBox>
|
</CheckBox>
|
||||||
<Separator Margin="0,4,0,4" Height="1" />
|
<Separator Margin="0,8,0,8" Height="1" />
|
||||||
<StackPanel Orientation="Vertical">
|
<StackPanel Orientation="Vertical">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsLStickDeadzone}" />
|
<TextBlock
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsLStickDeadzone}" />
|
||||||
<StackPanel
|
<StackPanel
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -473,9 +484,12 @@
|
|||||||
Value="{Binding Configuration.DeadzoneLeft, Mode=TwoWay}" />
|
Value="{Binding Configuration.DeadzoneLeft, Mode=TwoWay}" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
|
Width="25"
|
||||||
Text="{Binding Configuration.DeadzoneLeft, StringFormat=\{0:0.00\}}" />
|
Text="{Binding Configuration.DeadzoneLeft, StringFormat=\{0:0.00\}}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsStickRange}" />
|
<TextBlock
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsStickRange}" />
|
||||||
<StackPanel
|
<StackPanel
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -489,6 +503,7 @@
|
|||||||
Value="{Binding Configuration.RangeLeft, Mode=TwoWay}" />
|
Value="{Binding Configuration.RangeLeft, Mode=TwoWay}" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
|
Width="25"
|
||||||
Text="{Binding Configuration.RangeLeft, StringFormat=\{0:0.00\}}" />
|
Text="{Binding Configuration.RangeLeft, StringFormat=\{0:0.00\}}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
@@ -502,7 +517,8 @@
|
|||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
VerticalAlignment="Top"
|
VerticalAlignment="Top"
|
||||||
IsVisible="{Binding IsLeft}">
|
IsVisible="{Binding IsLeft}"
|
||||||
|
Margin="0,5,0,0">
|
||||||
<StackPanel Margin="10" Orientation="Vertical">
|
<StackPanel Margin="10" Orientation="Vertical">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,10"
|
Margin="0,0,0,10"
|
||||||
@@ -519,8 +535,6 @@
|
|||||||
Text="{locale:Locale ControllerSettingsDPadUp}"
|
Text="{locale:Locale ControllerSettingsDPadUp}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
HorizontalAlignment="Stretch">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.DpadUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.DpadUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
@@ -538,8 +552,6 @@
|
|||||||
Text="{locale:Locale ControllerSettingsDPadDown}"
|
Text="{locale:Locale ControllerSettingsDPadDown}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
HorizontalAlignment="Stretch">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.DpadDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.DpadDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
@@ -557,8 +569,6 @@
|
|||||||
Text="{locale:Locale ControllerSettingsDPadLeft}"
|
Text="{locale:Locale ControllerSettingsDPadLeft}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
HorizontalAlignment="Stretch">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.DpadLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.DpadLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
@@ -576,8 +586,6 @@
|
|||||||
Text="{locale:Locale ControllerSettingsDPadRight}"
|
Text="{locale:Locale ControllerSettingsDPadRight}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
HorizontalAlignment="Stretch">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.DpadRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.DpadRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
@@ -590,11 +598,11 @@
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<!-- Triggers And Side Buttons-->
|
<!-- Triggers And Side Buttons-->
|
||||||
<StackPanel Grid.Column="1" HorizontalAlignment="Stretch">
|
<StackPanel Grid.Column="1" HorizontalAlignment="Stretch" VerticalAlignment="Stretch">
|
||||||
<Border
|
<Border
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1">
|
BorderThickness="1" MinHeight="90">
|
||||||
<StackPanel Margin="10" Orientation="Vertical">
|
<StackPanel Margin="8" Orientation="Vertical">
|
||||||
<TextBlock HorizontalAlignment="Center" Text="{locale:Locale ControllerSettingsTriggerThreshold}" />
|
<TextBlock HorizontalAlignment="Center" Text="{locale:Locale ControllerSettingsTriggerThreshold}" />
|
||||||
<StackPanel HorizontalAlignment="Center" Orientation="Horizontal">
|
<StackPanel HorizontalAlignment="Center" Orientation="Horizontal">
|
||||||
<Slider
|
<Slider
|
||||||
@@ -604,7 +612,9 @@
|
|||||||
IsSnapToTickEnabled="True"
|
IsSnapToTickEnabled="True"
|
||||||
Minimum="0"
|
Minimum="0"
|
||||||
Value="{Binding Configuration.TriggerThreshold, Mode=TwoWay}" />
|
Value="{Binding Configuration.TriggerThreshold, Mode=TwoWay}" />
|
||||||
<TextBlock Text="{Binding Configuration.TriggerThreshold, StringFormat=\{0:0.00\}}" />
|
<TextBlock
|
||||||
|
Width="25"
|
||||||
|
Text="{Binding Configuration.TriggerThreshold, StringFormat=\{0:0.00\}}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,4,0,0"
|
Margin="0,4,0,0"
|
||||||
@@ -619,10 +629,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLeftSR}"
|
Text="{locale:Locale ControllerSettingsLeftSR}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftButtonSr, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftButtonSr, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -641,10 +648,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsLeftSL}"
|
Text="{locale:Locale ControllerSettingsLeftSL}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.LeftButtonSl, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.LeftButtonSl, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -663,10 +667,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRightSR}"
|
Text="{locale:Locale ControllerSettingsRightSR}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightButtonSr, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightButtonSr, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -685,10 +686,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRightSL}"
|
Text="{locale:Locale ControllerSettingsRightSL}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightButtonSl, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightButtonSl, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -702,14 +700,15 @@
|
|||||||
Margin="0,10,0,0"
|
Margin="0,10,0,0"
|
||||||
MaxHeight="250"
|
MaxHeight="250"
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
VerticalAlignment="Top"
|
VerticalAlignment="Stretch"
|
||||||
Source="{Binding Image}" />
|
Source="{Binding Image}" />
|
||||||
|
|
||||||
<!-- Motion+Rumble -->
|
<!-- Motion+Rumble -->
|
||||||
<StackPanel Margin="0,10,0,0" Orientation="Vertical" >
|
<StackPanel Margin="0,10,0,0" Orientation="Vertical" VerticalAlignment="Bottom">
|
||||||
<Border
|
<Border
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
|
VerticalAlignment="Bottom"
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
IsVisible="{Binding IsController}">
|
IsVisible="{Binding IsController}">
|
||||||
<Grid>
|
<Grid>
|
||||||
@@ -733,7 +732,8 @@
|
|||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
HorizontalAlignment="Stretch"
|
HorizontalAlignment="Stretch"
|
||||||
IsVisible="{Binding IsController}">
|
IsVisible="{Binding IsController}"
|
||||||
|
Margin="0,-1,0,0">
|
||||||
<Grid>
|
<Grid>
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="*" />
|
<ColumnDefinition Width="*" />
|
||||||
@@ -756,7 +756,7 @@
|
|||||||
|
|
||||||
<!--Right Controls-->
|
<!--Right Controls-->
|
||||||
<Grid
|
<Grid
|
||||||
Margin="10,0,0,0"
|
Margin="5,0,0,0"
|
||||||
Grid.Column="2"
|
Grid.Column="2"
|
||||||
VerticalAlignment="Top"
|
VerticalAlignment="Top"
|
||||||
HorizontalAlignment="Stretch" >
|
HorizontalAlignment="Stretch" >
|
||||||
@@ -770,9 +770,9 @@
|
|||||||
Grid.Row="0"
|
Grid.Row="0"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
IsVisible="{Binding IsRight}">
|
IsVisible="{Binding IsRight}"
|
||||||
<StackPanel Margin="10" Orientation="Vertical">
|
MinHeight="90">
|
||||||
<Grid HorizontalAlignment="Stretch">
|
<Grid Margin="10" HorizontalAlignment="Stretch">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition />
|
<ColumnDefinition />
|
||||||
<ColumnDefinition />
|
<ColumnDefinition />
|
||||||
@@ -782,7 +782,6 @@
|
|||||||
<RowDefinition />
|
<RowDefinition />
|
||||||
</Grid.RowDefinitions>
|
</Grid.RowDefinitions>
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,0,0,4"
|
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Grid.Row="0"
|
Grid.Row="0"
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
@@ -793,10 +792,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsTriggerZR}"
|
Text="{locale:Locale ControllerSettingsTriggerZR}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonZr, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonZr, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -815,19 +811,15 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsTriggerR}"
|
Text="{locale:Locale ControllerSettingsTriggerR}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonR, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonR, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,0,8,4"
|
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
Grid.Row="0"
|
Grid.Row="1"
|
||||||
HorizontalAlignment="Right"
|
HorizontalAlignment="Right"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
@@ -838,97 +830,39 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsButtonPlus}"
|
Text="{locale:Locale ControllerSettingsButtonPlus}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonPlus, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonPlus, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Grid>
|
</Grid>
|
||||||
</StackPanel>
|
|
||||||
</Border>
|
</Border>
|
||||||
<Border
|
<Border
|
||||||
Grid.Row="1"
|
Grid.Row="1"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
IsVisible="{Binding IsRight}">
|
IsVisible="{Binding IsRight}"
|
||||||
|
Margin="0,5,0,0">
|
||||||
<StackPanel Margin="10" Orientation="Vertical">
|
<StackPanel Margin="10" Orientation="Vertical">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,10"
|
Margin="0,0,0,10"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsButtons}" />
|
Text="{locale:Locale ControllerSettingsButtons}" />
|
||||||
<Grid HorizontalAlignment="Stretch">
|
<StackPanel Orientation="Vertical">
|
||||||
<Grid.ColumnDefinitions>
|
|
||||||
<ColumnDefinition />
|
|
||||||
<ColumnDefinition />
|
|
||||||
</Grid.ColumnDefinitions>
|
|
||||||
<Grid.RowDefinitions>
|
|
||||||
<RowDefinition />
|
|
||||||
<RowDefinition />
|
|
||||||
</Grid.RowDefinitions>
|
|
||||||
<!-- Right Buttons X -->
|
|
||||||
<StackPanel
|
|
||||||
Margin="0,0,0,4"
|
|
||||||
Grid.Column="0"
|
|
||||||
Grid.Row="0"
|
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
|
||||||
Orientation="Horizontal">
|
|
||||||
<TextBlock
|
|
||||||
Width="20"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Text="{locale:Locale ControllerSettingsButtonX}"
|
|
||||||
TextAlignment="Center" />
|
|
||||||
<ToggleButton
|
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
|
||||||
Text="{Binding Configuration.ButtonX, Mode=TwoWay, Converter={StaticResource Key}}"
|
|
||||||
TextAlignment="Center" />
|
|
||||||
</ToggleButton>
|
|
||||||
</StackPanel>
|
|
||||||
<!-- Right Buttons Y -->
|
|
||||||
<StackPanel
|
|
||||||
Grid.Column="0"
|
|
||||||
Grid.Row="1"
|
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
|
||||||
Orientation="Horizontal">
|
|
||||||
<TextBlock
|
|
||||||
Width="20"
|
|
||||||
HorizontalAlignment="Center"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Text="{locale:Locale ControllerSettingsButtonY}"
|
|
||||||
TextAlignment="Center" />
|
|
||||||
<ToggleButton
|
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
|
||||||
Text="{Binding Configuration.ButtonY, Mode=TwoWay, Converter={StaticResource Key}}"
|
|
||||||
TextAlignment="Center" />
|
|
||||||
</ToggleButton>
|
|
||||||
</StackPanel>
|
|
||||||
<!-- Right Buttons A -->
|
<!-- Right Buttons A -->
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Margin="0,0,0,4"
|
Margin="0,0,0,4"
|
||||||
Grid.Column="1"
|
|
||||||
Grid.Row="0"
|
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
Orientation="Horizontal">
|
Orientation="Horizontal">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Width="20"
|
Width="120"
|
||||||
|
Margin="0,0,10,0"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsButtonA}"
|
Text="{locale:Locale ControllerSettingsButtonA}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonA, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonA, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -936,26 +870,59 @@
|
|||||||
</StackPanel>
|
</StackPanel>
|
||||||
<!-- Right Buttons B -->
|
<!-- Right Buttons B -->
|
||||||
<StackPanel
|
<StackPanel
|
||||||
Grid.Column="1"
|
Margin="0,0,0,4"
|
||||||
Grid.Row="1"
|
|
||||||
Background="{DynamicResource ThemeDarkColor}"
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
Orientation="Horizontal">
|
Orientation="Horizontal">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Width="20"
|
Width="120"
|
||||||
|
Margin="0,0,10,0"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsButtonB}"
|
Text="{locale:Locale ControllerSettingsButtonB}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.ButtonB, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.ButtonB, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Grid>
|
<!-- Right Buttons X -->
|
||||||
|
<StackPanel
|
||||||
|
Margin="0,0,0,4"
|
||||||
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
|
Orientation="Horizontal">
|
||||||
|
<TextBlock
|
||||||
|
Width="120"
|
||||||
|
Margin="0,0,10,0"
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsButtonX}"
|
||||||
|
TextAlignment="Center" />
|
||||||
|
<ToggleButton>
|
||||||
|
<TextBlock
|
||||||
|
Text="{Binding Configuration.ButtonX, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
|
TextAlignment="Center" />
|
||||||
|
</ToggleButton>
|
||||||
|
</StackPanel>
|
||||||
|
<!-- Right Buttons Y -->
|
||||||
|
<StackPanel
|
||||||
|
Margin="0,0,0,4"
|
||||||
|
Background="{DynamicResource ThemeDarkColor}"
|
||||||
|
Orientation="Horizontal">
|
||||||
|
<TextBlock
|
||||||
|
Width="120"
|
||||||
|
Margin="0,0,10,0"
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsButtonY}"
|
||||||
|
TextAlignment="Center" />
|
||||||
|
<ToggleButton>
|
||||||
|
<TextBlock
|
||||||
|
Text="{Binding Configuration.ButtonY, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
|
TextAlignment="Center" />
|
||||||
|
</ToggleButton>
|
||||||
|
</StackPanel>
|
||||||
|
</StackPanel>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Border>
|
</Border>
|
||||||
<Border
|
<Border
|
||||||
@@ -963,7 +930,8 @@
|
|||||||
Padding="10"
|
Padding="10"
|
||||||
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
BorderBrush="{DynamicResource ThemeControlBorderColor}"
|
||||||
BorderThickness="1"
|
BorderThickness="1"
|
||||||
IsVisible="{Binding IsRight}">
|
IsVisible="{Binding IsRight}"
|
||||||
|
Margin="0,5,0,0">
|
||||||
<StackPanel Orientation="Vertical">
|
<StackPanel Orientation="Vertical">
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Margin="0,0,0,10"
|
Margin="0,0,0,10"
|
||||||
@@ -982,10 +950,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickButton}"
|
Text="{locale:Locale ControllerSettingsRStickButton}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightKeyboardStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightKeyboardStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1001,10 +966,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickUp}"
|
Text="{locale:Locale ControllerSettingsRStickUp}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightStickUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightStickUp, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1020,10 +982,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickDown}"
|
Text="{locale:Locale ControllerSettingsRStickDown}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightStickDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightStickDown, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1039,10 +998,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickLeft}"
|
Text="{locale:Locale ControllerSettingsRStickLeft}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightStickLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightStickLeft, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1058,10 +1014,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickRight}"
|
Text="{locale:Locale ControllerSettingsRStickRight}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightStickRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightStickRight, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1081,10 +1034,7 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickButton}"
|
Text="{locale:Locale ControllerSettingsRStickButton}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton>
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightControllerStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightControllerStickButton, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
@@ -1101,16 +1051,13 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Text="{locale:Locale ControllerSettingsRStickStick}"
|
Text="{locale:Locale ControllerSettingsRStickStick}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
<ToggleButton
|
<ToggleButton Tag="stick">
|
||||||
Width="90"
|
|
||||||
Height="27"
|
|
||||||
HorizontalAlignment="Stretch"
|
|
||||||
Tag="stick">
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Text="{Binding Configuration.RightJoystick, Mode=TwoWay, Converter={StaticResource Key}}"
|
Text="{Binding Configuration.RightJoystick, Mode=TwoWay, Converter={StaticResource Key}}"
|
||||||
TextAlignment="Center" />
|
TextAlignment="Center" />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
|
<Separator Margin="0,8,0,8" Height="1" />
|
||||||
<CheckBox IsChecked="{Binding Configuration.RightInvertStickX}">
|
<CheckBox IsChecked="{Binding Configuration.RightInvertStickX}">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsRStickInvertXAxis}" />
|
<TextBlock Text="{locale:Locale ControllerSettingsRStickInvertXAxis}" />
|
||||||
</CheckBox>
|
</CheckBox>
|
||||||
@@ -1120,9 +1067,11 @@
|
|||||||
<CheckBox IsChecked="{Binding Configuration.RightRotate90}">
|
<CheckBox IsChecked="{Binding Configuration.RightRotate90}">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsRotate90}" />
|
<TextBlock Text="{locale:Locale ControllerSettingsRotate90}" />
|
||||||
</CheckBox>
|
</CheckBox>
|
||||||
<Separator Margin="0,4,0,4" Height="1" />
|
<Separator Margin="0,8,0,8" Height="1" />
|
||||||
<StackPanel Orientation="Vertical">
|
<StackPanel Orientation="Vertical">
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsRStickDeadzone}" />
|
<TextBlock
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsRStickDeadzone}" />
|
||||||
<StackPanel
|
<StackPanel
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -1138,9 +1087,12 @@
|
|||||||
Value="{Binding Configuration.DeadzoneRight, Mode=TwoWay}" />
|
Value="{Binding Configuration.DeadzoneRight, Mode=TwoWay}" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
|
Width="25"
|
||||||
Text="{Binding Configuration.DeadzoneRight, StringFormat=\{0:0.00\}}" />
|
Text="{Binding Configuration.DeadzoneRight, StringFormat=\{0:0.00\}}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
<TextBlock Text="{locale:Locale ControllerSettingsStickRange}" />
|
<TextBlock
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
Text="{locale:Locale ControllerSettingsStickRange}" />
|
||||||
<StackPanel
|
<StackPanel
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -1154,6 +1106,7 @@
|
|||||||
Value="{Binding Configuration.RangeRight, Mode=TwoWay}" />
|
Value="{Binding Configuration.RangeRight, Mode=TwoWay}" />
|
||||||
<TextBlock
|
<TextBlock
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
|
Width="25"
|
||||||
Text="{Binding Configuration.RangeRight, StringFormat=\{0:0.00\}}" />
|
Text="{Binding Configuration.RangeRight, StringFormat=\{0:0.00\}}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
|
@@ -23,6 +23,8 @@ namespace Ryujinx.Graphics.Gpu.Engine.Threed
|
|||||||
public const int PrimitiveRestartStateIndex = 12;
|
public const int PrimitiveRestartStateIndex = 12;
|
||||||
public const int RenderTargetStateIndex = 27;
|
public const int RenderTargetStateIndex = 27;
|
||||||
|
|
||||||
|
private const ulong MaxUnknownStorageSize = 0x100000;
|
||||||
|
|
||||||
private readonly GpuContext _context;
|
private readonly GpuContext _context;
|
||||||
private readonly GpuChannel _channel;
|
private readonly GpuChannel _channel;
|
||||||
private readonly DeviceStateWithShadow<ThreedClassState> _state;
|
private readonly DeviceStateWithShadow<ThreedClassState> _state;
|
||||||
@@ -356,7 +358,19 @@ namespace Ryujinx.Graphics.Gpu.Engine.Threed
|
|||||||
|
|
||||||
SbDescriptor sbDescriptor = _channel.MemoryManager.Physical.Read<SbDescriptor>(sbDescAddress);
|
SbDescriptor sbDescriptor = _channel.MemoryManager.Physical.Read<SbDescriptor>(sbDescAddress);
|
||||||
|
|
||||||
_channel.BufferManager.SetGraphicsStorageBuffer(stage, sb.Slot, sbDescriptor.PackAddress(), (uint)sbDescriptor.Size, sb.Flags);
|
uint size;
|
||||||
|
if (sb.SbCbSlot == 0)
|
||||||
|
{
|
||||||
|
// Only trust the SbDescriptor size if it comes from slot 0.
|
||||||
|
size = (uint)sbDescriptor.Size;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// TODO: Use full mapped size and somehow speed up buffer sync.
|
||||||
|
size = (uint)_channel.MemoryManager.GetMappedSize(sbDescriptor.PackAddress(), MaxUnknownStorageSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
_channel.BufferManager.SetGraphicsStorageBuffer(stage, sb.Slot, sbDescriptor.PackAddress(), size, sb.Flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -98,6 +98,9 @@ namespace Ryujinx.Graphics.Gpu
|
|||||||
private Thread _gpuThread;
|
private Thread _gpuThread;
|
||||||
private bool _pendingSync;
|
private bool _pendingSync;
|
||||||
|
|
||||||
|
private long _modifiedSequence;
|
||||||
|
private ulong _firstTimestamp;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Creates a new instance of the GPU emulation context.
|
/// Creates a new instance of the GPU emulation context.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -121,6 +124,8 @@ namespace Ryujinx.Graphics.Gpu
|
|||||||
DeferredActions = new Queue<Action>();
|
DeferredActions = new Queue<Action>();
|
||||||
|
|
||||||
PhysicalMemoryRegistry = new ConcurrentDictionary<ulong, PhysicalMemory>();
|
PhysicalMemoryRegistry = new ConcurrentDictionary<ulong, PhysicalMemory>();
|
||||||
|
|
||||||
|
_firstTimestamp = ConvertNanosecondsToTicks((ulong)PerformanceCounter.ElapsedNanoseconds);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -200,13 +205,23 @@ namespace Ryujinx.Graphics.Gpu
|
|||||||
return divided * NsToTicksFractionNumerator + errorBias;
|
return divided * NsToTicksFractionNumerator + errorBias;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets a sequence number for resource modification ordering. This increments on each call.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>A sequence number for resource modification ordering</returns>
|
||||||
|
public long GetModifiedSequence()
|
||||||
|
{
|
||||||
|
return _modifiedSequence++;
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the value of the GPU timer.
|
/// Gets the value of the GPU timer.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <returns>The current GPU timestamp</returns>
|
/// <returns>The current GPU timestamp</returns>
|
||||||
public ulong GetTimestamp()
|
public ulong GetTimestamp()
|
||||||
{
|
{
|
||||||
ulong ticks = ConvertNanosecondsToTicks((ulong)PerformanceCounter.ElapsedNanoseconds);
|
// Guest timestamp will start at 0, instead of host value.
|
||||||
|
ulong ticks = ConvertNanosecondsToTicks((ulong)PerformanceCounter.ElapsedNanoseconds) - _firstTimestamp;
|
||||||
|
|
||||||
if (GraphicsConfig.FastGpuTime)
|
if (GraphicsConfig.FastGpuTime)
|
||||||
{
|
{
|
||||||
|
@@ -1170,6 +1170,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// <param name="caps">Host GPU capabilities</param>
|
/// <param name="caps">Host GPU capabilities</param>
|
||||||
/// <param name="firstLayer">Texture view initial layer on this texture</param>
|
/// <param name="firstLayer">Texture view initial layer on this texture</param>
|
||||||
/// <param name="firstLevel">Texture view first mipmap level on this texture</param>
|
/// <param name="firstLevel">Texture view first mipmap level on this texture</param>
|
||||||
|
/// <param name="flags">Texture search flags</param>
|
||||||
/// <returns>The level of compatiblilty a view with the given parameters created from this texture has</returns>
|
/// <returns>The level of compatiblilty a view with the given parameters created from this texture has</returns>
|
||||||
public TextureViewCompatibility IsViewCompatible(
|
public TextureViewCompatibility IsViewCompatible(
|
||||||
TextureInfo info,
|
TextureInfo info,
|
||||||
@@ -1178,11 +1179,12 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
int layerSize,
|
int layerSize,
|
||||||
Capabilities caps,
|
Capabilities caps,
|
||||||
out int firstLayer,
|
out int firstLayer,
|
||||||
out int firstLevel)
|
out int firstLevel,
|
||||||
|
TextureSearchFlags flags = TextureSearchFlags.None)
|
||||||
{
|
{
|
||||||
TextureViewCompatibility result = TextureViewCompatibility.Full;
|
TextureViewCompatibility result = TextureViewCompatibility.Full;
|
||||||
|
|
||||||
result = TextureCompatibility.PropagateViewCompatibility(result, TextureCompatibility.ViewFormatCompatible(Info, info, caps));
|
result = TextureCompatibility.PropagateViewCompatibility(result, TextureCompatibility.ViewFormatCompatible(Info, info, caps, flags));
|
||||||
if (result != TextureViewCompatibility.Incompatible)
|
if (result != TextureViewCompatibility.Incompatible)
|
||||||
{
|
{
|
||||||
result = TextureCompatibility.PropagateViewCompatibility(result, TextureCompatibility.ViewTargetCompatible(Info, info, ref caps));
|
result = TextureCompatibility.PropagateViewCompatibility(result, TextureCompatibility.ViewTargetCompatible(Info, info, ref caps));
|
||||||
|
@@ -569,7 +569,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
|
|
||||||
Texture texture = null;
|
Texture texture = null;
|
||||||
|
|
||||||
TextureMatchQuality bestQuality = TextureMatchQuality.NoMatch;
|
long bestSequence = 0;
|
||||||
|
|
||||||
for (int index = 0; index < sameAddressOverlapsCount; index++)
|
for (int index = 0; index < sameAddressOverlapsCount; index++)
|
||||||
{
|
{
|
||||||
@@ -601,17 +601,12 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (matchQuality == TextureMatchQuality.Perfect)
|
if (texture == null || overlap.Group.ModifiedSequence - bestSequence > 0)
|
||||||
{
|
{
|
||||||
texture = overlap;
|
texture = overlap;
|
||||||
break;
|
bestSequence = overlap.Group.ModifiedSequence;
|
||||||
}
|
}
|
||||||
else if (matchQuality > bestQuality)
|
|
||||||
{
|
|
||||||
texture = overlap;
|
|
||||||
bestQuality = matchQuality;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -664,6 +659,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
int fullyCompatible = 0;
|
int fullyCompatible = 0;
|
||||||
|
|
||||||
// Evaluate compatibility of overlaps, add temporary references
|
// Evaluate compatibility of overlaps, add temporary references
|
||||||
|
int preferredOverlap = -1;
|
||||||
|
|
||||||
for (int index = 0; index < overlapsCount; index++)
|
for (int index = 0; index < overlapsCount; index++)
|
||||||
{
|
{
|
||||||
@@ -675,17 +671,26 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
sizeInfo.LayerSize,
|
sizeInfo.LayerSize,
|
||||||
_context.Capabilities,
|
_context.Capabilities,
|
||||||
out int firstLayer,
|
out int firstLayer,
|
||||||
out int firstLevel);
|
out int firstLevel,
|
||||||
|
flags);
|
||||||
|
|
||||||
if (overlapCompatibility == TextureViewCompatibility.Full)
|
if (overlapCompatibility >= TextureViewCompatibility.FormatAlias)
|
||||||
{
|
{
|
||||||
if (overlap.IsView)
|
if (overlap.IsView)
|
||||||
{
|
{
|
||||||
overlapCompatibility = TextureViewCompatibility.CopyOnly;
|
overlapCompatibility = overlapCompatibility == TextureViewCompatibility.FormatAlias ?
|
||||||
|
TextureViewCompatibility.Incompatible :
|
||||||
|
TextureViewCompatibility.CopyOnly;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
fullyCompatible++;
|
fullyCompatible++;
|
||||||
|
|
||||||
|
if (preferredOverlap == -1 || overlap.Group.ModifiedSequence - bestSequence > 0)
|
||||||
|
{
|
||||||
|
preferredOverlap = index;
|
||||||
|
bestSequence = overlap.Group.ModifiedSequence;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -695,26 +700,38 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
|
|
||||||
// Search through the overlaps to find a compatible view and establish any copy dependencies.
|
// Search through the overlaps to find a compatible view and establish any copy dependencies.
|
||||||
|
|
||||||
for (int index = 0; index < overlapsCount; index++)
|
if (preferredOverlap != -1)
|
||||||
{
|
{
|
||||||
Texture overlap = _textureOverlaps[index];
|
Texture overlap = _textureOverlaps[preferredOverlap];
|
||||||
OverlapInfo oInfo = _overlapInfo[index];
|
OverlapInfo oInfo = _overlapInfo[preferredOverlap];
|
||||||
|
|
||||||
|
bool aliased = oInfo.Compatibility == TextureViewCompatibility.FormatAlias;
|
||||||
|
|
||||||
if (oInfo.Compatibility == TextureViewCompatibility.Full)
|
|
||||||
{
|
|
||||||
if (!isSamplerTexture)
|
if (!isSamplerTexture)
|
||||||
{
|
{
|
||||||
// If this is not a sampler texture, the size might be different from the requested size,
|
// If this is not a sampler texture, the size might be different from the requested size,
|
||||||
// so we need to make sure the texture information has the correct size for this base texture,
|
// so we need to make sure the texture information has the correct size for this base texture,
|
||||||
// before creating the view.
|
// before creating the view.
|
||||||
info = info.CreateInfoForLevelView(overlap, oInfo.FirstLevel);
|
|
||||||
|
info = info.CreateInfoForLevelView(overlap, oInfo.FirstLevel, aliased);
|
||||||
|
}
|
||||||
|
else if (aliased)
|
||||||
|
{
|
||||||
|
// The format must be changed to match the parent.
|
||||||
|
info = info.CreateInfoWithFormat(overlap.Info.FormatInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
texture = overlap.CreateView(info, sizeInfo, range.Value, oInfo.FirstLayer, oInfo.FirstLevel);
|
texture = overlap.CreateView(info, sizeInfo, range.Value, oInfo.FirstLayer, oInfo.FirstLevel);
|
||||||
texture.SynchronizeMemory();
|
texture.SynchronizeMemory();
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
else if (oInfo.Compatibility == TextureViewCompatibility.CopyOnly && fullyCompatible == 0)
|
else
|
||||||
|
{
|
||||||
|
for (int index = 0; index < overlapsCount; index++)
|
||||||
|
{
|
||||||
|
Texture overlap = _textureOverlaps[index];
|
||||||
|
OverlapInfo oInfo = _overlapInfo[index];
|
||||||
|
|
||||||
|
if (oInfo.Compatibility == TextureViewCompatibility.CopyOnly && fullyCompatible == 0)
|
||||||
{
|
{
|
||||||
// Only copy compatible. If there's another choice for a FULLY compatible texture, choose that instead.
|
// Only copy compatible. If there's another choice for a FULLY compatible texture, choose that instead.
|
||||||
|
|
||||||
@@ -728,6 +745,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (texture != null)
|
if (texture != null)
|
||||||
{
|
{
|
||||||
@@ -740,7 +758,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
Texture overlap = _textureOverlaps[index];
|
Texture overlap = _textureOverlaps[index];
|
||||||
OverlapInfo oInfo = _overlapInfo[index];
|
OverlapInfo oInfo = _overlapInfo[index];
|
||||||
|
|
||||||
if (oInfo.Compatibility <= TextureViewCompatibility.LayoutIncompatible)
|
if (oInfo.Compatibility <= TextureViewCompatibility.LayoutIncompatible || oInfo.Compatibility == TextureViewCompatibility.FormatAlias)
|
||||||
{
|
{
|
||||||
if (!overlap.IsView && texture.DataOverlaps(overlap, oInfo.Compatibility))
|
if (!overlap.IsView && texture.DataOverlaps(overlap, oInfo.Compatibility))
|
||||||
{
|
{
|
||||||
|
@@ -291,22 +291,7 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// <returns>The minimum compatibility level of two provided view compatibility results</returns>
|
/// <returns>The minimum compatibility level of two provided view compatibility results</returns>
|
||||||
public static TextureViewCompatibility PropagateViewCompatibility(TextureViewCompatibility first, TextureViewCompatibility second)
|
public static TextureViewCompatibility PropagateViewCompatibility(TextureViewCompatibility first, TextureViewCompatibility second)
|
||||||
{
|
{
|
||||||
if (first == TextureViewCompatibility.Incompatible || second == TextureViewCompatibility.Incompatible)
|
return (TextureViewCompatibility)Math.Min((int)first, (int)second);
|
||||||
{
|
|
||||||
return TextureViewCompatibility.Incompatible;
|
|
||||||
}
|
|
||||||
else if (first == TextureViewCompatibility.LayoutIncompatible || second == TextureViewCompatibility.LayoutIncompatible)
|
|
||||||
{
|
|
||||||
return TextureViewCompatibility.LayoutIncompatible;
|
|
||||||
}
|
|
||||||
else if (first == TextureViewCompatibility.CopyOnly || second == TextureViewCompatibility.CopyOnly)
|
|
||||||
{
|
|
||||||
return TextureViewCompatibility.CopyOnly;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return TextureViewCompatibility.Full;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -628,15 +613,21 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// <param name="lhs">Texture information of the texture view</param>
|
/// <param name="lhs">Texture information of the texture view</param>
|
||||||
/// <param name="rhs">Texture information of the texture view</param>
|
/// <param name="rhs">Texture information of the texture view</param>
|
||||||
/// <param name="caps">Host GPU capabilities</param>
|
/// <param name="caps">Host GPU capabilities</param>
|
||||||
|
/// <param name="flags">Texture search flags</param>
|
||||||
/// <returns>The view compatibility level of the texture formats</returns>
|
/// <returns>The view compatibility level of the texture formats</returns>
|
||||||
public static TextureViewCompatibility ViewFormatCompatible(TextureInfo lhs, TextureInfo rhs, Capabilities caps)
|
public static TextureViewCompatibility ViewFormatCompatible(TextureInfo lhs, TextureInfo rhs, Capabilities caps, TextureSearchFlags flags)
|
||||||
{
|
{
|
||||||
FormatInfo lhsFormat = lhs.FormatInfo;
|
FormatInfo lhsFormat = lhs.FormatInfo;
|
||||||
FormatInfo rhsFormat = rhs.FormatInfo;
|
FormatInfo rhsFormat = rhs.FormatInfo;
|
||||||
|
|
||||||
if (lhsFormat.Format.IsDepthOrStencil() || rhsFormat.Format.IsDepthOrStencil())
|
if (lhsFormat.Format.IsDepthOrStencil() || rhsFormat.Format.IsDepthOrStencil())
|
||||||
{
|
{
|
||||||
return lhsFormat.Format == rhsFormat.Format ? TextureViewCompatibility.Full : TextureViewCompatibility.Incompatible;
|
return FormatMatches(lhs, rhs, flags.HasFlag(TextureSearchFlags.ForSampler), flags.HasFlag(TextureSearchFlags.DepthAlias)) switch
|
||||||
|
{
|
||||||
|
TextureMatchQuality.Perfect => TextureViewCompatibility.Full,
|
||||||
|
TextureMatchQuality.FormatAlias => TextureViewCompatibility.FormatAlias,
|
||||||
|
_ => TextureViewCompatibility.Incompatible
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IsFormatHostIncompatible(lhs, caps) || IsFormatHostIncompatible(rhs, caps))
|
if (IsFormatHostIncompatible(lhs, caps) || IsFormatHostIncompatible(rhs, caps))
|
||||||
@@ -754,49 +745,6 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
return result ? TextureViewCompatibility.Full : TextureViewCompatibility.Incompatible;
|
return result ? TextureViewCompatibility.Full : TextureViewCompatibility.Incompatible;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks if a swizzle component in two textures functionally match, taking into account if the components are defined.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="lhs">Texture information to compare</param>
|
|
||||||
/// <param name="rhs">Texture information to compare with</param>
|
|
||||||
/// <param name="swizzleLhs">Swizzle component for the first texture</param>
|
|
||||||
/// <param name="swizzleRhs">Swizzle component for the second texture</param>
|
|
||||||
/// <param name="component">Component index, starting at 0 for red</param>
|
|
||||||
/// <returns>True if the swizzle components functionally match, false othersize</returns>
|
|
||||||
private static bool SwizzleComponentMatches(TextureInfo lhs, TextureInfo rhs, SwizzleComponent swizzleLhs, SwizzleComponent swizzleRhs, int component)
|
|
||||||
{
|
|
||||||
int lhsComponents = lhs.FormatInfo.Components;
|
|
||||||
int rhsComponents = rhs.FormatInfo.Components;
|
|
||||||
|
|
||||||
if (lhsComponents == 4 && rhsComponents == 4)
|
|
||||||
{
|
|
||||||
return swizzleLhs == swizzleRhs;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Swizzles after the number of components a format defines are "undefined".
|
|
||||||
// We allow these to not be equal under certain circumstances.
|
|
||||||
// This can only happen when there are less than 4 components in a format.
|
|
||||||
// It tends to happen when float depth textures are sampled.
|
|
||||||
|
|
||||||
bool lhsDefined = (swizzleLhs - SwizzleComponent.Red) < lhsComponents;
|
|
||||||
bool rhsDefined = (swizzleRhs - SwizzleComponent.Red) < rhsComponents;
|
|
||||||
|
|
||||||
if (lhsDefined == rhsDefined)
|
|
||||||
{
|
|
||||||
// If both are undefined, return true. Otherwise just check if they're equal.
|
|
||||||
return lhsDefined ? swizzleLhs == swizzleRhs : true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
SwizzleComponent defined = lhsDefined ? swizzleLhs : swizzleRhs;
|
|
||||||
SwizzleComponent undefined = lhsDefined ? swizzleRhs : swizzleLhs;
|
|
||||||
|
|
||||||
// Undefined swizzle can be matched by a forced value (0, 1), exact equality, or expected value.
|
|
||||||
// For example, R___ matches R001, RGBA but not RBGA.
|
|
||||||
return defined == undefined || defined < SwizzleComponent.Red || defined == SwizzleComponent.Red + component;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Checks if the texture shader sampling parameters of two texture informations match.
|
/// Checks if the texture shader sampling parameters of two texture informations match.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -806,10 +754,10 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
public static bool SamplerParamsMatches(TextureInfo lhs, TextureInfo rhs)
|
public static bool SamplerParamsMatches(TextureInfo lhs, TextureInfo rhs)
|
||||||
{
|
{
|
||||||
return lhs.DepthStencilMode == rhs.DepthStencilMode &&
|
return lhs.DepthStencilMode == rhs.DepthStencilMode &&
|
||||||
SwizzleComponentMatches(lhs, rhs, lhs.SwizzleR, rhs.SwizzleR, 0) &&
|
lhs.SwizzleR == rhs.SwizzleR &&
|
||||||
SwizzleComponentMatches(lhs, rhs, lhs.SwizzleG, rhs.SwizzleG, 1) &&
|
lhs.SwizzleG == rhs.SwizzleG &&
|
||||||
SwizzleComponentMatches(lhs, rhs, lhs.SwizzleB, rhs.SwizzleB, 2) &&
|
lhs.SwizzleB == rhs.SwizzleB &&
|
||||||
SwizzleComponentMatches(lhs, rhs, lhs.SwizzleA, rhs.SwizzleA, 3);
|
lhs.SwizzleA == rhs.SwizzleA;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@@ -68,6 +68,11 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public bool HasIncompatibleOverlaps => _incompatibleOverlaps.Count > 0;
|
public bool HasIncompatibleOverlaps => _incompatibleOverlaps.Count > 0;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Number indicating the order this texture group was modified relative to others.
|
||||||
|
/// </summary>
|
||||||
|
public long ModifiedSequence { get; private set; }
|
||||||
|
|
||||||
private readonly GpuContext _context;
|
private readonly GpuContext _context;
|
||||||
private readonly PhysicalMemory _physicalMemory;
|
private readonly PhysicalMemory _physicalMemory;
|
||||||
|
|
||||||
@@ -664,6 +669,8 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// <param name="texture">The texture that has been modified</param>
|
/// <param name="texture">The texture that has been modified</param>
|
||||||
public void SignalModified(Texture texture)
|
public void SignalModified(Texture texture)
|
||||||
{
|
{
|
||||||
|
ModifiedSequence = _context.GetModifiedSequence();
|
||||||
|
|
||||||
ClearIncompatibleOverlaps(texture);
|
ClearIncompatibleOverlaps(texture);
|
||||||
|
|
||||||
EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
|
EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
|
||||||
@@ -684,6 +691,8 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// <param name="bound">True if this texture is being bound, false if unbound</param>
|
/// <param name="bound">True if this texture is being bound, false if unbound</param>
|
||||||
public void SignalModifying(Texture texture, bool bound)
|
public void SignalModifying(Texture texture, bool bound)
|
||||||
{
|
{
|
||||||
|
ModifiedSequence = _context.GetModifiedSequence();
|
||||||
|
|
||||||
ClearIncompatibleOverlaps(texture);
|
ClearIncompatibleOverlaps(texture);
|
||||||
|
|
||||||
EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
|
EvaluateRelevantHandles(texture, (baseHandle, regionCount, split) =>
|
||||||
|
@@ -300,8 +300,9 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="parent">The parent texture</param>
|
/// <param name="parent">The parent texture</param>
|
||||||
/// <param name="firstLevel">The first level of the texture view</param>
|
/// <param name="firstLevel">The first level of the texture view</param>
|
||||||
|
/// <param name="parentFormat">True if the parent format should be inherited</param>
|
||||||
/// <returns>The adjusted texture information with the new size</returns>
|
/// <returns>The adjusted texture information with the new size</returns>
|
||||||
public TextureInfo CreateInfoForLevelView(Texture parent, int firstLevel)
|
public TextureInfo CreateInfoForLevelView(Texture parent, int firstLevel, bool parentFormat)
|
||||||
{
|
{
|
||||||
// When the texture is used as view of another texture, we must
|
// When the texture is used as view of another texture, we must
|
||||||
// ensure that the sizes are valid, otherwise data uploads would fail
|
// ensure that the sizes are valid, otherwise data uploads would fail
|
||||||
@@ -370,7 +371,36 @@ namespace Ryujinx.Graphics.Gpu.Image
|
|||||||
GobBlocksInZ,
|
GobBlocksInZ,
|
||||||
GobBlocksInTileX,
|
GobBlocksInTileX,
|
||||||
target,
|
target,
|
||||||
FormatInfo,
|
parentFormat ? parent.Info.FormatInfo : FormatInfo,
|
||||||
|
DepthStencilMode,
|
||||||
|
SwizzleR,
|
||||||
|
SwizzleG,
|
||||||
|
SwizzleB,
|
||||||
|
SwizzleA);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates texture information for a given format and this information.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="formatInfo">Format for the new texture info</param>
|
||||||
|
/// <returns>New info with the specified format</returns>
|
||||||
|
public TextureInfo CreateInfoWithFormat(FormatInfo formatInfo)
|
||||||
|
{
|
||||||
|
return new TextureInfo(
|
||||||
|
GpuAddress,
|
||||||
|
Width,
|
||||||
|
Height,
|
||||||
|
DepthOrLayers,
|
||||||
|
Levels,
|
||||||
|
SamplesInX,
|
||||||
|
SamplesInY,
|
||||||
|
Stride,
|
||||||
|
IsLinear,
|
||||||
|
GobBlocksInY,
|
||||||
|
GobBlocksInZ,
|
||||||
|
GobBlocksInTileX,
|
||||||
|
Target,
|
||||||
|
formatInfo,
|
||||||
DepthStencilMode,
|
DepthStencilMode,
|
||||||
SwizzleR,
|
SwizzleR,
|
||||||
SwizzleG,
|
SwizzleG,
|
||||||
|
@@ -9,6 +9,7 @@
|
|||||||
Incompatible = 0,
|
Incompatible = 0,
|
||||||
LayoutIncompatible,
|
LayoutIncompatible,
|
||||||
CopyOnly,
|
CopyOnly,
|
||||||
|
FormatAlias,
|
||||||
Full
|
Full
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -637,6 +637,33 @@ namespace Ryujinx.Graphics.Gpu.Memory
|
|||||||
return UnpackPaFromPte(pte) + (va & PageMask);
|
return UnpackPaFromPte(pte) + (va & PageMask);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Translates a GPU virtual address and returns the number of bytes that are mapped after it.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="va">GPU virtual address to be translated</param>
|
||||||
|
/// <param name="maxSize">Maximum size in bytes to scan</param>
|
||||||
|
/// <returns>Number of bytes, 0 if unmapped</returns>
|
||||||
|
public ulong GetMappedSize(ulong va, ulong maxSize)
|
||||||
|
{
|
||||||
|
if (!ValidateAddress(va))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ulong startVa = va;
|
||||||
|
ulong endVa = va + maxSize;
|
||||||
|
|
||||||
|
ulong pte = GetPte(va);
|
||||||
|
|
||||||
|
while (pte != PteUnmapped && va < endVa)
|
||||||
|
{
|
||||||
|
va += PageSize - (va & PageMask);
|
||||||
|
pte = GetPte(va);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Math.Min(maxSize, va - startVa);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Gets the kind of a given memory page.
|
/// Gets the kind of a given memory page.
|
||||||
/// This might indicate the type of resource that can be allocated on the page, and also texture tiling.
|
/// This might indicate the type of resource that can be allocated on the page, and also texture tiling.
|
||||||
|
@@ -22,7 +22,7 @@ namespace Ryujinx.Graphics.Gpu.Shader.DiskCache
|
|||||||
private const ushort FileFormatVersionMajor = 1;
|
private const ushort FileFormatVersionMajor = 1;
|
||||||
private const ushort FileFormatVersionMinor = 2;
|
private const ushort FileFormatVersionMinor = 2;
|
||||||
private const uint FileFormatVersionPacked = ((uint)FileFormatVersionMajor << 16) | FileFormatVersionMinor;
|
private const uint FileFormatVersionPacked = ((uint)FileFormatVersionMajor << 16) | FileFormatVersionMinor;
|
||||||
private const uint CodeGenVersion = 4821;
|
private const uint CodeGenVersion = 4578;
|
||||||
|
|
||||||
private const string SharedTocFileName = "shared.toc";
|
private const string SharedTocFileName = "shared.toc";
|
||||||
private const string SharedDataFileName = "shared.data";
|
private const string SharedDataFileName = "shared.data";
|
||||||
|
@@ -591,7 +591,7 @@ namespace Ryujinx.Graphics.Gpu.Shader
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
ReadOnlySpan<byte> memoryCode = memoryManager.GetSpan(gpuVa, shader.Code.Length);
|
ReadOnlySpan<byte> memoryCode = memoryManager.GetSpanMapped(gpuVa, shader.Code.Length);
|
||||||
|
|
||||||
return memoryCode.SequenceEqual(shader.Code);
|
return memoryCode.SequenceEqual(shader.Code);
|
||||||
}
|
}
|
||||||
|
@@ -306,6 +306,8 @@ namespace Ryujinx.Graphics.OpenGL.Image
|
|||||||
int offset = WriteToPbo2D(range.Offset, layer, level);
|
int offset = WriteToPbo2D(range.Offset, layer, level);
|
||||||
|
|
||||||
Debug.Assert(offset == 0);
|
Debug.Assert(offset == 0);
|
||||||
|
|
||||||
|
GL.BindBuffer(BufferTarget.PixelPackBuffer, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void WriteToPbo(int offset, bool forceBgra)
|
public void WriteToPbo(int offset, bool forceBgra)
|
||||||
|
@@ -1442,14 +1442,6 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Spirv
|
|||||||
return GetZeroOperationResult(context, texOp, AggregateType.FP32, colorIsVector);
|
return GetZeroOperationResult(context, texOp, AggregateType.FP32, colorIsVector);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This combination is valid, but not available on GLSL.
|
|
||||||
// For now, ignore the LOD level and do a normal sample.
|
|
||||||
// TODO: How to implement it properly?
|
|
||||||
if (hasLodLevel && isArray && isShadow)
|
|
||||||
{
|
|
||||||
hasLodLevel = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
int srcIndex = isBindless ? 1 : 0;
|
int srcIndex = isBindless ? 1 : 0;
|
||||||
|
|
||||||
SpvInstruction Src(AggregateType type)
|
SpvInstruction Src(AggregateType type)
|
||||||
|
@@ -20,6 +20,12 @@ namespace Ryujinx.Graphics.Shader.Translation.Optimizations
|
|||||||
GlobalToStorage.RunPass(blocks[blkIndex], config, ref sbUseMask, ref ubeUseMask);
|
GlobalToStorage.RunPass(blocks[blkIndex], config, ref sbUseMask, ref ubeUseMask);
|
||||||
BindlessToIndexed.RunPass(blocks[blkIndex], config);
|
BindlessToIndexed.RunPass(blocks[blkIndex], config);
|
||||||
BindlessElimination.RunPass(blocks[blkIndex], config);
|
BindlessElimination.RunPass(blocks[blkIndex], config);
|
||||||
|
|
||||||
|
// FragmentCoord only exists on fragment shaders, so we don't need to check other stages.
|
||||||
|
if (config.Stage == ShaderStage.Fragment)
|
||||||
|
{
|
||||||
|
EliminateMultiplyByFragmentCoordW(blocks[blkIndex]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
config.SetAccessibleBufferMasks(sbUseMask, ubeUseMask);
|
config.SetAccessibleBufferMasks(sbUseMask, ubeUseMask);
|
||||||
@@ -281,6 +287,75 @@ namespace Ryujinx.Graphics.Shader.Translation.Optimizations
|
|||||||
return modified;
|
return modified;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void EliminateMultiplyByFragmentCoordW(BasicBlock block)
|
||||||
|
{
|
||||||
|
foreach (INode node in block.Operations)
|
||||||
|
{
|
||||||
|
if (node is Operation operation)
|
||||||
|
{
|
||||||
|
EliminateMultiplyByFragmentCoordW(operation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void EliminateMultiplyByFragmentCoordW(Operation operation)
|
||||||
|
{
|
||||||
|
// We're looking for the pattern:
|
||||||
|
// y = x * gl_FragCoord.w
|
||||||
|
// v = y * (1.0 / gl_FragCoord.w)
|
||||||
|
// Then we transform it into:
|
||||||
|
// v = x
|
||||||
|
// This pattern is common on fragment shaders due to the way how perspective correction is done.
|
||||||
|
|
||||||
|
// We are expecting a multiplication by the reciprocal of gl_FragCoord.w.
|
||||||
|
if (operation.Inst != (Instruction.FP32 | Instruction.Multiply))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Operand lhs = operation.GetSource(0);
|
||||||
|
Operand rhs = operation.GetSource(1);
|
||||||
|
|
||||||
|
// Check LHS of the the main multiplication operation. We expect an input being multiplied by gl_FragCoord.w.
|
||||||
|
if (!(lhs.AsgOp is Operation attrMulOp) || attrMulOp.Inst != (Instruction.FP32 | Instruction.Multiply))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Operand attrMulLhs = attrMulOp.GetSource(0);
|
||||||
|
Operand attrMulRhs = attrMulOp.GetSource(1);
|
||||||
|
|
||||||
|
// LHS should be any input, RHS should be exactly gl_FragCoord.w.
|
||||||
|
if (!Utils.IsInputLoad(attrMulLhs.AsgOp) || !Utils.IsInputLoad(attrMulRhs.AsgOp, IoVariable.FragmentCoord, 3))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// RHS of the main multiplication should be a reciprocal operation (1.0 / x).
|
||||||
|
if (!(rhs.AsgOp is Operation reciprocalOp) || reciprocalOp.Inst != (Instruction.FP32 | Instruction.Divide))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Operand reciprocalLhs = reciprocalOp.GetSource(0);
|
||||||
|
Operand reciprocalRhs = reciprocalOp.GetSource(1);
|
||||||
|
|
||||||
|
// Check if the divisor is a constant equal to 1.0.
|
||||||
|
if (reciprocalLhs.Type != OperandType.Constant || reciprocalLhs.AsFloat() != 1.0f)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the dividend is gl_FragCoord.w.
|
||||||
|
if (!Utils.IsInputLoad(reciprocalRhs.AsgOp, IoVariable.FragmentCoord, 3))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If everything matches, we can replace the operation with the input load result.
|
||||||
|
operation.TurnIntoCopy(attrMulLhs);
|
||||||
|
}
|
||||||
|
|
||||||
private static void RemoveNode(BasicBlock block, LinkedListNode<INode> llNode)
|
private static void RemoveNode(BasicBlock block, LinkedListNode<INode> llNode)
|
||||||
{
|
{
|
||||||
// Remove a node from the nodes list, and also remove itself
|
// Remove a node from the nodes list, and also remove itself
|
||||||
|
@@ -4,6 +4,35 @@ namespace Ryujinx.Graphics.Shader.Translation.Optimizations
|
|||||||
{
|
{
|
||||||
static class Utils
|
static class Utils
|
||||||
{
|
{
|
||||||
|
public static bool IsInputLoad(INode node)
|
||||||
|
{
|
||||||
|
return (node is Operation operation) &&
|
||||||
|
operation.Inst == Instruction.Load &&
|
||||||
|
operation.StorageKind == StorageKind.Input;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool IsInputLoad(INode node, IoVariable ioVariable, int elemIndex)
|
||||||
|
{
|
||||||
|
if (!(node is Operation operation) ||
|
||||||
|
operation.Inst != Instruction.Load ||
|
||||||
|
operation.StorageKind != StorageKind.Input ||
|
||||||
|
operation.SourcesCount != 2)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Operand ioVariableSrc = operation.GetSource(0);
|
||||||
|
|
||||||
|
if (ioVariableSrc.Type != OperandType.Constant || (IoVariable)ioVariableSrc.Value != ioVariable)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Operand elemIndexSrc = operation.GetSource(1);
|
||||||
|
|
||||||
|
return elemIndexSrc.Type == OperandType.Constant && elemIndexSrc.Value == elemIndex;
|
||||||
|
}
|
||||||
|
|
||||||
private static Operation FindBranchSource(BasicBlock block)
|
private static Operation FindBranchSource(BasicBlock block)
|
||||||
{
|
{
|
||||||
foreach (BasicBlock sourceBlock in block.Predecessors)
|
foreach (BasicBlock sourceBlock in block.Predecessors)
|
||||||
|
@@ -956,7 +956,7 @@ namespace Ryujinx.Graphics.Texture.Astc
|
|||||||
{
|
{
|
||||||
Span<uint> val = ReadUintColorValues(2, colorValues, ref colorValuesPosition);
|
Span<uint> val = ReadUintColorValues(2, colorValues, ref colorValuesPosition);
|
||||||
int l0 = (int)((val[0] >> 2) | (val[1] & 0xC0));
|
int l0 = (int)((val[0] >> 2) | (val[1] & 0xC0));
|
||||||
int l1 = (int)Math.Max(l0 + (val[1] & 0x3F), 0xFFU);
|
int l1 = (int)Math.Min(l0 + (val[1] & 0x3F), 0xFFU);
|
||||||
|
|
||||||
endPoints[0] = new AstcPixel(0xFF, (short)l0, (short)l0, (short)l0);
|
endPoints[0] = new AstcPixel(0xFF, (short)l0, (short)l0, (short)l0);
|
||||||
endPoints[1] = new AstcPixel(0xFF, (short)l1, (short)l1, (short)l1);
|
endPoints[1] = new AstcPixel(0xFF, (short)l1, (short)l1, (short)l1);
|
||||||
|
@@ -56,6 +56,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
private int _writeCount;
|
private int _writeCount;
|
||||||
private int _flushCount;
|
private int _flushCount;
|
||||||
private int _flushTemp;
|
private int _flushTemp;
|
||||||
|
private int _lastFlushWrite = -1;
|
||||||
|
|
||||||
private ReaderWriterLock _flushLock;
|
private ReaderWriterLock _flushLock;
|
||||||
private FenceHolder _flushFence;
|
private FenceHolder _flushFence;
|
||||||
@@ -200,14 +201,21 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
{
|
{
|
||||||
if (_baseType == BufferAllocationType.Auto)
|
if (_baseType == BufferAllocationType.Auto)
|
||||||
{
|
{
|
||||||
if (_writeCount >= WriteCountThreshold || _setCount >= SetCountThreshold || _flushCount >= FlushCountThreshold)
|
// When flushed, wait for a bit more info to make a decision.
|
||||||
|
bool wasFlushed = _flushTemp > 0;
|
||||||
|
int multiplier = wasFlushed ? 2 : 0;
|
||||||
|
if (_writeCount >= (WriteCountThreshold << multiplier) || _setCount >= (SetCountThreshold << multiplier) || _flushCount >= (FlushCountThreshold << multiplier))
|
||||||
{
|
{
|
||||||
if (_flushCount > 0 || _flushTemp-- > 0)
|
if (_flushCount > 0 || _flushTemp-- > 0)
|
||||||
{
|
{
|
||||||
// Buffers that flush should ideally be mapped in host address space for easy copies.
|
// Buffers that flush should ideally be mapped in host address space for easy copies.
|
||||||
// If the buffer is large it will do better on GPU memory, as there will be more writes than data flushes (typically individual pages).
|
// If the buffer is large it will do better on GPU memory, as there will be more writes than data flushes (typically individual pages).
|
||||||
// If it is small, then it's likely most of the buffer will be flushed so we want it on host memory, as access is cached.
|
// If it is small, then it's likely most of the buffer will be flushed so we want it on host memory, as access is cached.
|
||||||
DesiredType = Size > DeviceLocalSizeThreshold ? BufferAllocationType.DeviceLocalMapped : BufferAllocationType.HostMapped;
|
|
||||||
|
bool hostMappingSensitive = _gd.Vendor == Vendor.Nvidia;
|
||||||
|
bool deviceLocalMapped = Size > DeviceLocalSizeThreshold || (wasFlushed && _writeCount > _flushCount * 10 && hostMappingSensitive) || _currentType == BufferAllocationType.DeviceLocalMapped;
|
||||||
|
|
||||||
|
DesiredType = deviceLocalMapped ? BufferAllocationType.DeviceLocalMapped : BufferAllocationType.HostMapped;
|
||||||
|
|
||||||
// It's harder for a buffer that is flushed to revert to another type of mapping.
|
// It's harder for a buffer that is flushed to revert to another type of mapping.
|
||||||
if (_flushCount > 0)
|
if (_flushCount > 0)
|
||||||
@@ -215,17 +223,18 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
_flushTemp = 1000;
|
_flushTemp = 1000;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (_writeCount >= WriteCountThreshold)
|
else if (_writeCount >= (WriteCountThreshold << multiplier))
|
||||||
{
|
{
|
||||||
// Buffers that are written often should ideally be in the device local heap. (Storage buffers)
|
// Buffers that are written often should ideally be in the device local heap. (Storage buffers)
|
||||||
DesiredType = BufferAllocationType.DeviceLocal;
|
DesiredType = BufferAllocationType.DeviceLocal;
|
||||||
}
|
}
|
||||||
else if (_setCount > SetCountThreshold)
|
else if (_setCount > (SetCountThreshold << multiplier))
|
||||||
{
|
{
|
||||||
// Buffers that have their data set often should ideally be host mapped. (Constant buffers)
|
// Buffers that have their data set often should ideally be host mapped. (Constant buffers)
|
||||||
DesiredType = BufferAllocationType.HostMapped;
|
DesiredType = BufferAllocationType.HostMapped;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_lastFlushWrite = -1;
|
||||||
_flushCount = 0;
|
_flushCount = 0;
|
||||||
_writeCount = 0;
|
_writeCount = 0;
|
||||||
_setCount = 0;
|
_setCount = 0;
|
||||||
@@ -418,7 +427,12 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
|
|
||||||
WaitForFlushFence();
|
WaitForFlushFence();
|
||||||
|
|
||||||
|
if (_lastFlushWrite != _writeCount)
|
||||||
|
{
|
||||||
|
// If it's on the same page as the last flush, ignore it.
|
||||||
|
_lastFlushWrite = _writeCount;
|
||||||
_flushCount++;
|
_flushCount++;
|
||||||
|
}
|
||||||
|
|
||||||
Span<byte> result;
|
Span<byte> result;
|
||||||
|
|
||||||
|
@@ -228,7 +228,12 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
SignalDirty(DirtyFlags.Storage);
|
SignalDirty(DirtyFlags.Storage);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SetTextureAndSampler(CommandBufferScoped cbs, ShaderStage stage, int binding, ITexture texture, ISampler sampler)
|
public void SetTextureAndSampler(
|
||||||
|
CommandBufferScoped cbs,
|
||||||
|
ShaderStage stage,
|
||||||
|
int binding,
|
||||||
|
ITexture texture,
|
||||||
|
ISampler sampler)
|
||||||
{
|
{
|
||||||
if (texture is TextureBuffer textureBuffer)
|
if (texture is TextureBuffer textureBuffer)
|
||||||
{
|
{
|
||||||
@@ -251,6 +256,28 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
SignalDirty(DirtyFlags.Texture);
|
SignalDirty(DirtyFlags.Texture);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void SetTextureAndSamplerIdentitySwizzle(
|
||||||
|
CommandBufferScoped cbs,
|
||||||
|
ShaderStage stage,
|
||||||
|
int binding,
|
||||||
|
ITexture texture,
|
||||||
|
ISampler sampler)
|
||||||
|
{
|
||||||
|
if (texture is TextureView view)
|
||||||
|
{
|
||||||
|
view.Storage.InsertWriteToReadBarrier(cbs, AccessFlags.ShaderReadBit, stage.ConvertToPipelineStageFlags());
|
||||||
|
|
||||||
|
_textureRefs[binding] = view.GetIdentityImageView();
|
||||||
|
_samplerRefs[binding] = ((SamplerHolder)sampler)?.GetSampler();
|
||||||
|
|
||||||
|
SignalDirty(DirtyFlags.Texture);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
SetTextureAndSampler(cbs, stage, binding, texture, sampler);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void SetUniformBuffers(CommandBuffer commandBuffer, ReadOnlySpan<BufferAssignment> buffers)
|
public void SetUniformBuffers(CommandBuffer commandBuffer, ReadOnlySpan<BufferAssignment> buffers)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < buffers.Length; i++)
|
for (int i = 0; i < buffers.Length; i++)
|
||||||
|
@@ -415,7 +415,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
|
|
||||||
var sampler = linearFilter ? _samplerLinear : _samplerNearest;
|
var sampler = linearFilter ? _samplerLinear : _samplerNearest;
|
||||||
|
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Fragment, 0, src, sampler);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Fragment, 0, src, sampler);
|
||||||
|
|
||||||
Span<float> region = stackalloc float[RegionBufferSize / sizeof(float)];
|
Span<float> region = stackalloc float[RegionBufferSize / sizeof(float)];
|
||||||
|
|
||||||
@@ -625,7 +625,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
|
|
||||||
private void BlitDepthStencilDraw(TextureView src, bool isDepth)
|
private void BlitDepthStencilDraw(TextureView src, bool isDepth)
|
||||||
{
|
{
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Fragment, 0, src, _samplerNearest);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Fragment, 0, src, _samplerNearest);
|
||||||
|
|
||||||
if (isDepth)
|
if (isDepth)
|
||||||
{
|
{
|
||||||
@@ -1037,7 +1037,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
var srcView = Create2DLayerView(src, srcLayer + z, srcLevel + l, srcFormat);
|
var srcView = Create2DLayerView(src, srcLayer + z, srcLevel + l, srcFormat);
|
||||||
var dstView = Create2DLayerView(dst, dstLayer + z, dstLevel + l);
|
var dstView = Create2DLayerView(dst, dstLayer + z, dstLevel + l);
|
||||||
|
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Compute, 0, srcView, null);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Compute, 0, srcView, null);
|
||||||
_pipeline.SetImage(0, dstView, dstFormat);
|
_pipeline.SetImage(0, dstView, dstFormat);
|
||||||
|
|
||||||
int dispatchX = (Math.Min(srcView.Info.Width, dstView.Info.Width) + 31) / 32;
|
int dispatchX = (Math.Min(srcView.Info.Width, dstView.Info.Width) + 31) / 32;
|
||||||
@@ -1177,7 +1177,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
var srcView = Create2DLayerView(src, srcLayer + z, 0, format);
|
var srcView = Create2DLayerView(src, srcLayer + z, 0, format);
|
||||||
var dstView = Create2DLayerView(dst, dstLayer + z, 0);
|
var dstView = Create2DLayerView(dst, dstLayer + z, 0);
|
||||||
|
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Compute, 0, srcView, null);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Compute, 0, srcView, null);
|
||||||
_pipeline.SetImage(0, dstView, format);
|
_pipeline.SetImage(0, dstView, format);
|
||||||
|
|
||||||
_pipeline.DispatchCompute(dispatchX, dispatchY, 1);
|
_pipeline.DispatchCompute(dispatchX, dispatchY, 1);
|
||||||
@@ -1313,7 +1313,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
var srcView = Create2DLayerView(src, srcLayer + z, 0, format);
|
var srcView = Create2DLayerView(src, srcLayer + z, 0, format);
|
||||||
var dstView = Create2DLayerView(dst, dstLayer + z, 0);
|
var dstView = Create2DLayerView(dst, dstLayer + z, 0);
|
||||||
|
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Fragment, 0, srcView, null);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Fragment, 0, srcView, null);
|
||||||
_pipeline.SetRenderTarget(
|
_pipeline.SetRenderTarget(
|
||||||
((TextureView)dstView).GetView(format).GetImageViewForAttachment(),
|
((TextureView)dstView).GetView(format).GetImageViewForAttachment(),
|
||||||
(uint)dst.Width,
|
(uint)dst.Width,
|
||||||
@@ -1384,7 +1384,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
|
|
||||||
private void CopyMSAspectDraw(TextureView src, bool fromMS, bool isDepth)
|
private void CopyMSAspectDraw(TextureView src, bool fromMS, bool isDepth)
|
||||||
{
|
{
|
||||||
_pipeline.SetTextureAndSampler(ShaderStage.Fragment, 0, src, _samplerNearest);
|
_pipeline.SetTextureAndSamplerIdentitySwizzle(ShaderStage.Fragment, 0, src, _samplerNearest);
|
||||||
|
|
||||||
if (isDepth)
|
if (isDepth)
|
||||||
{
|
{
|
||||||
|
@@ -25,6 +25,8 @@ namespace Ryujinx.Graphics.Vulkan.MoltenVK
|
|||||||
config.SemaphoreSupportStyle = MVKVkSemaphoreSupportStyle.MVK_CONFIG_VK_SEMAPHORE_SUPPORT_STYLE_SINGLE_QUEUE;
|
config.SemaphoreSupportStyle = MVKVkSemaphoreSupportStyle.MVK_CONFIG_VK_SEMAPHORE_SUPPORT_STYLE_SINGLE_QUEUE;
|
||||||
config.SynchronousQueueSubmits = false;
|
config.SynchronousQueueSubmits = false;
|
||||||
|
|
||||||
|
config.ResumeLostDevice = true;
|
||||||
|
|
||||||
vkSetMoltenVKConfigurationMVK(IntPtr.Zero, config, configSize);
|
vkSetMoltenVKConfigurationMVK(IntPtr.Zero, config, configSize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1098,6 +1098,11 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
_descriptorSetUpdater.SetTextureAndSampler(Cbs, stage, binding, texture, sampler);
|
_descriptorSetUpdater.SetTextureAndSampler(Cbs, stage, binding, texture, sampler);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void SetTextureAndSamplerIdentitySwizzle(ShaderStage stage, int binding, ITexture texture, ISampler sampler)
|
||||||
|
{
|
||||||
|
_descriptorSetUpdater.SetTextureAndSamplerIdentitySwizzle(Cbs, stage, binding, texture, sampler);
|
||||||
|
}
|
||||||
|
|
||||||
public void SetTransformFeedbackBuffers(ReadOnlySpan<BufferRange> buffers)
|
public void SetTransformFeedbackBuffers(ReadOnlySpan<BufferRange> buffers)
|
||||||
{
|
{
|
||||||
PauseTransformFeedbackInternal();
|
PauseTransformFeedbackInternal();
|
||||||
|
@@ -12,6 +12,28 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
ShaderStageFlags.FragmentBit |
|
ShaderStageFlags.FragmentBit |
|
||||||
ShaderStageFlags.ComputeBit;
|
ShaderStageFlags.ComputeBit;
|
||||||
|
|
||||||
|
private static ShaderStageFlags ActiveStages(uint stages)
|
||||||
|
{
|
||||||
|
ShaderStageFlags stageFlags = 0;
|
||||||
|
|
||||||
|
while (stages != 0)
|
||||||
|
{
|
||||||
|
int stage = BitOperations.TrailingZeroCount(stages);
|
||||||
|
stages &= ~(1u << stage);
|
||||||
|
|
||||||
|
stageFlags |= stage switch
|
||||||
|
{
|
||||||
|
1 => ShaderStageFlags.FragmentBit,
|
||||||
|
2 => ShaderStageFlags.GeometryBit,
|
||||||
|
3 => ShaderStageFlags.TessellationControlBit,
|
||||||
|
4 => ShaderStageFlags.TessellationEvaluationBit,
|
||||||
|
_ => ShaderStageFlags.VertexBit | ShaderStageFlags.ComputeBit
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return stageFlags;
|
||||||
|
}
|
||||||
|
|
||||||
public static unsafe DescriptorSetLayout[] Create(VulkanRenderer gd, Device device, uint stages, bool usePd, out PipelineLayout layout)
|
public static unsafe DescriptorSetLayout[] Create(VulkanRenderer gd, Device device, uint stages, bool usePd, out PipelineLayout layout)
|
||||||
{
|
{
|
||||||
int stagesCount = BitOperations.PopCount(stages);
|
int stagesCount = BitOperations.PopCount(stages);
|
||||||
@@ -34,6 +56,7 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
};
|
};
|
||||||
|
|
||||||
int iter = 0;
|
int iter = 0;
|
||||||
|
var activeStages = ActiveStages(stages);
|
||||||
|
|
||||||
while (stages != 0)
|
while (stages != 0)
|
||||||
{
|
{
|
||||||
@@ -67,12 +90,16 @@ namespace Ryujinx.Graphics.Vulkan
|
|||||||
|
|
||||||
void SetStorage(DescriptorSetLayoutBinding* bindings, int maxPerStage, int start = 0)
|
void SetStorage(DescriptorSetLayoutBinding* bindings, int maxPerStage, int start = 0)
|
||||||
{
|
{
|
||||||
|
// There's a bug on MoltenVK where using the same buffer across different stages
|
||||||
|
// causes invalid resource errors, allow the binding on all active stages as workaround.
|
||||||
|
var flags = gd.IsMoltenVk ? activeStages : stageFlags;
|
||||||
|
|
||||||
bindings[start + iter] = new DescriptorSetLayoutBinding
|
bindings[start + iter] = new DescriptorSetLayoutBinding
|
||||||
{
|
{
|
||||||
Binding = (uint)(start + stage * maxPerStage),
|
Binding = (uint)(start + stage * maxPerStage),
|
||||||
DescriptorType = DescriptorType.StorageBuffer,
|
DescriptorType = DescriptorType.StorageBuffer,
|
||||||
DescriptorCount = (uint)maxPerStage,
|
DescriptorCount = (uint)maxPerStage,
|
||||||
StageFlags = stageFlags
|
StageFlags = flags
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -261,7 +261,7 @@ namespace Ryujinx.HLE.HOS
|
|||||||
AudioInputManager = new AudioInputManager();
|
AudioInputManager = new AudioInputManager();
|
||||||
AudioRendererManager = new AudioRendererManager(tickSource);
|
AudioRendererManager = new AudioRendererManager(tickSource);
|
||||||
AudioRendererManager.SetVolume(Device.Configuration.AudioVolume);
|
AudioRendererManager.SetVolume(Device.Configuration.AudioVolume);
|
||||||
AudioDeviceSessionRegistry = new VirtualDeviceSessionRegistry();
|
AudioDeviceSessionRegistry = new VirtualDeviceSessionRegistry(Device.AudioDeviceDriver);
|
||||||
|
|
||||||
IWritableEvent[] audioOutputRegisterBufferEvents = new IWritableEvent[Constants.AudioOutSessionCountMax];
|
IWritableEvent[] audioOutputRegisterBufferEvents = new IWritableEvent[Constants.AudioOutSessionCountMax];
|
||||||
|
|
||||||
@@ -328,7 +328,7 @@ namespace Ryujinx.HLE.HOS
|
|||||||
private void StartNewServices()
|
private void StartNewServices()
|
||||||
{
|
{
|
||||||
ServiceTable = new ServiceTable();
|
ServiceTable = new ServiceTable();
|
||||||
var services = ServiceTable.GetServices(new HorizonOptions(Device.Configuration.IgnoreMissingServices));
|
var services = ServiceTable.GetServices(new HorizonOptions(Device.Configuration.IgnoreMissingServices, LibHacHorizonManager.BcatClient));
|
||||||
|
|
||||||
foreach (var service in services)
|
foreach (var service in services)
|
||||||
{
|
{
|
||||||
|
@@ -1,85 +0,0 @@
|
|||||||
using LibHac;
|
|
||||||
using LibHac.Common;
|
|
||||||
using Ryujinx.Common;
|
|
||||||
using Ryujinx.HLE.HOS.Services.Arp;
|
|
||||||
using Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat
|
|
||||||
{
|
|
||||||
[Service("bcat:a", "bcat:a")]
|
|
||||||
[Service("bcat:m", "bcat:m")]
|
|
||||||
[Service("bcat:u", "bcat:u")]
|
|
||||||
[Service("bcat:s", "bcat:s")]
|
|
||||||
class IServiceCreator : DisposableIpcService
|
|
||||||
{
|
|
||||||
private SharedRef<LibHac.Bcat.Impl.Ipc.IServiceCreator> _base;
|
|
||||||
|
|
||||||
public IServiceCreator(ServiceCtx context, string serviceName)
|
|
||||||
{
|
|
||||||
var applicationClient = context.Device.System.LibHacHorizonManager.ApplicationClient;
|
|
||||||
applicationClient.Sm.GetService(ref _base, serviceName).ThrowIfFailure();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Dispose(bool isDisposing)
|
|
||||||
{
|
|
||||||
if (isDisposing)
|
|
||||||
{
|
|
||||||
_base.Destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(0)]
|
|
||||||
// CreateBcatService(pid) -> object<nn::bcat::detail::ipc::IBcatService>
|
|
||||||
public ResultCode CreateBcatService(ServiceCtx context)
|
|
||||||
{
|
|
||||||
// TODO: Call arp:r GetApplicationLaunchProperty with the pid to get the TitleId.
|
|
||||||
// Add an instance of nn::bcat::detail::service::core::PassphraseManager.
|
|
||||||
// Add an instance of nn::bcat::detail::service::ServiceMemoryManager.
|
|
||||||
// Add an instance of nn::bcat::detail::service::core::TaskManager who load "bcat-sys:/" system save data and open "dc/task.bin".
|
|
||||||
// If the file don't exist, create a new one (size of 0x800) and write 2 empty struct with a size of 0x400.
|
|
||||||
|
|
||||||
MakeObject(context, new IBcatService(ApplicationLaunchProperty.GetByPid(context)));
|
|
||||||
|
|
||||||
// NOTE: If the IBcatService is null this error is returned, Doesn't occur in our case.
|
|
||||||
// return ResultCode.NullObject;
|
|
||||||
|
|
||||||
return ResultCode.Success;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(1)]
|
|
||||||
// CreateDeliveryCacheStorageService(pid) -> object<nn::bcat::detail::ipc::IDeliveryCacheStorageService>
|
|
||||||
public ResultCode CreateDeliveryCacheStorageService(ServiceCtx context)
|
|
||||||
{
|
|
||||||
ulong pid = context.RequestData.ReadUInt64();
|
|
||||||
|
|
||||||
using var serv = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>();
|
|
||||||
|
|
||||||
Result rc = _base.Get.CreateDeliveryCacheStorageService(ref serv.Ref, pid);
|
|
||||||
|
|
||||||
if (rc.IsSuccess())
|
|
||||||
{
|
|
||||||
MakeObject(context, new IDeliveryCacheStorageService(context, ref serv.Ref));
|
|
||||||
}
|
|
||||||
|
|
||||||
return (ResultCode)rc.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(2)]
|
|
||||||
// CreateDeliveryCacheStorageServiceWithApplicationId(nn::ApplicationId) -> object<nn::bcat::detail::ipc::IDeliveryCacheStorageService>
|
|
||||||
public ResultCode CreateDeliveryCacheStorageServiceWithApplicationId(ServiceCtx context)
|
|
||||||
{
|
|
||||||
ApplicationId applicationId = context.RequestData.ReadStruct<ApplicationId>();
|
|
||||||
|
|
||||||
using var service = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>();
|
|
||||||
|
|
||||||
Result rc = _base.Get.CreateDeliveryCacheStorageServiceWithApplicationId(ref service.Ref, applicationId);
|
|
||||||
|
|
||||||
if (rc.IsSuccess())
|
|
||||||
{
|
|
||||||
MakeObject(context, new IDeliveryCacheStorageService(context, ref service.Ref));
|
|
||||||
}
|
|
||||||
|
|
||||||
return (ResultCode)rc.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,29 +0,0 @@
|
|||||||
namespace Ryujinx.HLE.HOS.Services.Bcat
|
|
||||||
{
|
|
||||||
enum ResultCode
|
|
||||||
{
|
|
||||||
ModuleId = 122,
|
|
||||||
ErrorCodeShift = 9,
|
|
||||||
|
|
||||||
Success = 0,
|
|
||||||
|
|
||||||
InvalidArgument = (1 << ErrorCodeShift) | ModuleId,
|
|
||||||
NotFound = (2 << ErrorCodeShift) | ModuleId,
|
|
||||||
TargetLocked = (3 << ErrorCodeShift) | ModuleId,
|
|
||||||
TargetAlreadyMounted = (4 << ErrorCodeShift) | ModuleId,
|
|
||||||
TargetNotMounted = (5 << ErrorCodeShift) | ModuleId,
|
|
||||||
AlreadyOpen = (6 << ErrorCodeShift) | ModuleId,
|
|
||||||
NotOpen = (7 << ErrorCodeShift) | ModuleId,
|
|
||||||
InternetRequestDenied = (8 << ErrorCodeShift) | ModuleId,
|
|
||||||
ServiceOpenLimitReached = (9 << ErrorCodeShift) | ModuleId,
|
|
||||||
SaveDataNotFound = (10 << ErrorCodeShift) | ModuleId,
|
|
||||||
NetworkServiceAccountNotAvailable = (31 << ErrorCodeShift) | ModuleId,
|
|
||||||
PassphrasePathNotFound = (80 << ErrorCodeShift) | ModuleId,
|
|
||||||
DataVerificationFailed = (81 << ErrorCodeShift) | ModuleId,
|
|
||||||
PermissionDenied = (90 << ErrorCodeShift) | ModuleId,
|
|
||||||
AllocationFailed = (91 << ErrorCodeShift) | ModuleId,
|
|
||||||
InvalidOperation = (98 << ErrorCodeShift) | ModuleId,
|
|
||||||
InvalidDeliveryCacheStorageFile = (204 << ErrorCodeShift) | ModuleId,
|
|
||||||
StorageOpenLimitReached = (205 << ErrorCodeShift) | ModuleId
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,18 +0,0 @@
|
|||||||
using Ryujinx.HLE.HOS.Services.Arp;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator
|
|
||||||
{
|
|
||||||
class IBcatService : IpcService
|
|
||||||
{
|
|
||||||
public IBcatService(ApplicationLaunchProperty applicationLaunchProperty) { }
|
|
||||||
|
|
||||||
[CommandCmif(10100)]
|
|
||||||
// RequestSyncDeliveryCache() -> object<nn::bcat::detail::ipc::IDeliveryCacheProgressService>
|
|
||||||
public ResultCode RequestSyncDeliveryCache(ServiceCtx context)
|
|
||||||
{
|
|
||||||
MakeObject(context, new IDeliveryCacheProgressService(context));
|
|
||||||
|
|
||||||
return ResultCode.Success;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,65 +0,0 @@
|
|||||||
using LibHac;
|
|
||||||
using LibHac.Bcat;
|
|
||||||
using LibHac.Common;
|
|
||||||
using Ryujinx.Common;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator
|
|
||||||
{
|
|
||||||
class IDeliveryCacheDirectoryService : DisposableIpcService
|
|
||||||
{
|
|
||||||
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService> _base;
|
|
||||||
|
|
||||||
public IDeliveryCacheDirectoryService(ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService> baseService)
|
|
||||||
{
|
|
||||||
_base = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService>.CreateMove(ref baseService);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Dispose(bool isDisposing)
|
|
||||||
{
|
|
||||||
if (isDisposing)
|
|
||||||
{
|
|
||||||
_base.Destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(0)]
|
|
||||||
// Open(nn::bcat::DirectoryName)
|
|
||||||
public ResultCode Open(ServiceCtx context)
|
|
||||||
{
|
|
||||||
DirectoryName directoryName = context.RequestData.ReadStruct<DirectoryName>();
|
|
||||||
|
|
||||||
Result result = _base.Get.Open(ref directoryName);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(1)]
|
|
||||||
// Read() -> (u32, buffer<nn::bcat::DeliveryCacheDirectoryEntry, 6>)
|
|
||||||
public ResultCode Read(ServiceCtx context)
|
|
||||||
{
|
|
||||||
ulong bufferAddress = context.Request.ReceiveBuff[0].Position;
|
|
||||||
ulong bufferLen = context.Request.ReceiveBuff[0].Size;
|
|
||||||
|
|
||||||
using (var region = context.Memory.GetWritableRegion(bufferAddress, (int)bufferLen, true))
|
|
||||||
{
|
|
||||||
Result result = _base.Get.Read(out int entriesRead, MemoryMarshal.Cast<byte, DeliveryCacheDirectoryEntry>(region.Memory.Span));
|
|
||||||
|
|
||||||
context.ResponseData.Write(entriesRead);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(2)]
|
|
||||||
// GetCount() -> u32
|
|
||||||
public ResultCode GetCount(ServiceCtx context)
|
|
||||||
{
|
|
||||||
Result result = _base.Get.GetCount(out int count);
|
|
||||||
|
|
||||||
context.ResponseData.Write(count);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,78 +0,0 @@
|
|||||||
using LibHac;
|
|
||||||
using LibHac.Bcat;
|
|
||||||
using LibHac.Common;
|
|
||||||
using Ryujinx.Common;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator
|
|
||||||
{
|
|
||||||
class IDeliveryCacheFileService : DisposableIpcService
|
|
||||||
{
|
|
||||||
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService> _base;
|
|
||||||
|
|
||||||
public IDeliveryCacheFileService(ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService> baseService)
|
|
||||||
{
|
|
||||||
_base = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService>.CreateMove(ref baseService);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Dispose(bool isDisposing)
|
|
||||||
{
|
|
||||||
if (isDisposing)
|
|
||||||
{
|
|
||||||
_base.Destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(0)]
|
|
||||||
// Open(nn::bcat::DirectoryName, nn::bcat::FileName)
|
|
||||||
public ResultCode Open(ServiceCtx context)
|
|
||||||
{
|
|
||||||
DirectoryName directoryName = context.RequestData.ReadStruct<DirectoryName>();
|
|
||||||
FileName fileName = context.RequestData.ReadStruct<FileName>();
|
|
||||||
|
|
||||||
Result result = _base.Get.Open(ref directoryName, ref fileName);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(1)]
|
|
||||||
// Read(u64) -> (u64, buffer<bytes, 6>)
|
|
||||||
public ResultCode Read(ServiceCtx context)
|
|
||||||
{
|
|
||||||
ulong bufferAddress = context.Request.ReceiveBuff[0].Position;
|
|
||||||
ulong bufferLen = context.Request.ReceiveBuff[0].Size;
|
|
||||||
|
|
||||||
long offset = context.RequestData.ReadInt64();
|
|
||||||
|
|
||||||
using (var region = context.Memory.GetWritableRegion(bufferAddress, (int)bufferLen, true))
|
|
||||||
{
|
|
||||||
Result result = _base.Get.Read(out long bytesRead, offset, region.Memory.Span);
|
|
||||||
|
|
||||||
context.ResponseData.Write(bytesRead);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(2)]
|
|
||||||
// GetSize() -> u64
|
|
||||||
public ResultCode GetSize(ServiceCtx context)
|
|
||||||
{
|
|
||||||
Result result = _base.Get.GetSize(out long size);
|
|
||||||
|
|
||||||
context.ResponseData.Write(size);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(3)]
|
|
||||||
// GetDigest() -> nn::bcat::Digest
|
|
||||||
public ResultCode GetDigest(ServiceCtx context)
|
|
||||||
{
|
|
||||||
Result result = _base.Get.GetDigest(out Digest digest);
|
|
||||||
|
|
||||||
context.ResponseData.WriteStruct(digest);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,63 +0,0 @@
|
|||||||
using Ryujinx.Common.Logging;
|
|
||||||
using Ryujinx.Cpu;
|
|
||||||
using Ryujinx.HLE.HOS.Ipc;
|
|
||||||
using Ryujinx.HLE.HOS.Kernel.Threading;
|
|
||||||
using Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator.Types;
|
|
||||||
using Ryujinx.Horizon.Common;
|
|
||||||
using System;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator
|
|
||||||
{
|
|
||||||
class IDeliveryCacheProgressService : IpcService
|
|
||||||
{
|
|
||||||
private KEvent _event;
|
|
||||||
private int _eventHandle;
|
|
||||||
|
|
||||||
public IDeliveryCacheProgressService(ServiceCtx context)
|
|
||||||
{
|
|
||||||
_event = new KEvent(context.Device.System.KernelContext);
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(0)]
|
|
||||||
// GetEvent() -> handle<copy>
|
|
||||||
public ResultCode GetEvent(ServiceCtx context)
|
|
||||||
{
|
|
||||||
if (_eventHandle == 0)
|
|
||||||
{
|
|
||||||
if (context.Process.HandleTable.GenerateHandle(_event.ReadableEvent, out _eventHandle) != Result.Success)
|
|
||||||
{
|
|
||||||
throw new InvalidOperationException("Out of handles!");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
context.Response.HandleDesc = IpcHandleDesc.MakeCopy(_eventHandle);
|
|
||||||
|
|
||||||
Logger.Stub?.PrintStub(LogClass.ServiceBcat);
|
|
||||||
|
|
||||||
return ResultCode.Success;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(1)]
|
|
||||||
// GetImpl() -> buffer<nn::bcat::detail::DeliveryCacheProgressImpl, 0x1a>
|
|
||||||
public ResultCode GetImpl(ServiceCtx context)
|
|
||||||
{
|
|
||||||
DeliveryCacheProgressImpl deliveryCacheProgress = new DeliveryCacheProgressImpl
|
|
||||||
{
|
|
||||||
State = DeliveryCacheProgressImpl.Status.Done,
|
|
||||||
Result = 0
|
|
||||||
};
|
|
||||||
|
|
||||||
ulong dcpSize = WriteDeliveryCacheProgressImpl(context, context.Request.RecvListBuff[0], deliveryCacheProgress);
|
|
||||||
context.Response.PtrBuff[0] = context.Response.PtrBuff[0].WithSize(dcpSize);
|
|
||||||
|
|
||||||
Logger.Stub?.PrintStub(LogClass.ServiceBcat);
|
|
||||||
|
|
||||||
return ResultCode.Success;
|
|
||||||
}
|
|
||||||
|
|
||||||
private ulong WriteDeliveryCacheProgressImpl(ServiceCtx context, IpcRecvListBuffDesc ipcDesc, DeliveryCacheProgressImpl deliveryCacheProgress)
|
|
||||||
{
|
|
||||||
return MemoryHelper.Write(context.Memory, ipcDesc.Position, deliveryCacheProgress);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,74 +0,0 @@
|
|||||||
using LibHac;
|
|
||||||
using LibHac.Bcat;
|
|
||||||
using LibHac.Common;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator
|
|
||||||
{
|
|
||||||
class IDeliveryCacheStorageService : DisposableIpcService
|
|
||||||
{
|
|
||||||
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService> _base;
|
|
||||||
|
|
||||||
public IDeliveryCacheStorageService(ServiceCtx context, ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService> baseService)
|
|
||||||
{
|
|
||||||
_base = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>.CreateMove(ref baseService);
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(0)]
|
|
||||||
// CreateFileService() -> object<nn::bcat::detail::ipc::IDeliveryCacheFileService>
|
|
||||||
public ResultCode CreateFileService(ServiceCtx context)
|
|
||||||
{
|
|
||||||
using var service = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService>();
|
|
||||||
|
|
||||||
Result result = _base.Get.CreateFileService(ref service.Ref);
|
|
||||||
|
|
||||||
if (result.IsSuccess())
|
|
||||||
{
|
|
||||||
MakeObject(context, new IDeliveryCacheFileService(ref service.Ref));
|
|
||||||
}
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(1)]
|
|
||||||
// CreateDirectoryService() -> object<nn::bcat::detail::ipc::IDeliveryCacheDirectoryService>
|
|
||||||
public ResultCode CreateDirectoryService(ServiceCtx context)
|
|
||||||
{
|
|
||||||
using var service = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService>();
|
|
||||||
|
|
||||||
Result result = _base.Get.CreateDirectoryService(ref service.Ref);
|
|
||||||
|
|
||||||
if (result.IsSuccess())
|
|
||||||
{
|
|
||||||
MakeObject(context, new IDeliveryCacheDirectoryService(ref service.Ref));
|
|
||||||
}
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[CommandCmif(10)]
|
|
||||||
// EnumerateDeliveryCacheDirectory() -> (u32, buffer<nn::bcat::DirectoryName, 6>)
|
|
||||||
public ResultCode EnumerateDeliveryCacheDirectory(ServiceCtx context)
|
|
||||||
{
|
|
||||||
ulong bufferAddress = context.Request.ReceiveBuff[0].Position;
|
|
||||||
ulong bufferLen = context.Request.ReceiveBuff[0].Size;
|
|
||||||
|
|
||||||
using (var region = context.Memory.GetWritableRegion(bufferAddress, (int)bufferLen, true))
|
|
||||||
{
|
|
||||||
Result result = _base.Get.EnumerateDeliveryCacheDirectory(out int count, MemoryMarshal.Cast<byte, DirectoryName>(region.Memory.Span));
|
|
||||||
|
|
||||||
context.ResponseData.Write(count);
|
|
||||||
|
|
||||||
return (ResultCode)result.Value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override void Dispose(bool isDisposing)
|
|
||||||
{
|
|
||||||
if (isDisposing)
|
|
||||||
{
|
|
||||||
_base.Destroy();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -145,7 +145,7 @@ namespace Ryujinx.Horizon.Generators.Hipc
|
|||||||
|
|
||||||
if (bufferFixedSize != null)
|
if (bufferFixedSize != null)
|
||||||
{
|
{
|
||||||
arg = $"new CommandArg({bufferFlags}, {bufferFixedSize})";
|
arg = $"new CommandArg({bufferFlags} | HipcBufferFlags.FixedSize, {bufferFixedSize})";
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
48
src/Ryujinx.Horizon/Bcat/BcatIpcServer.cs
Normal file
48
src/Ryujinx.Horizon/Bcat/BcatIpcServer.cs
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
using Ryujinx.Horizon.Bcat.Ipc;
|
||||||
|
using Ryujinx.Horizon.Bcat.Types;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sm;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat
|
||||||
|
{
|
||||||
|
internal class BcatIpcServer
|
||||||
|
{
|
||||||
|
private const int BcatMaxSessionsCount = 8;
|
||||||
|
private const int BcatTotalMaxSessionsCount = BcatMaxSessionsCount * 4;
|
||||||
|
|
||||||
|
private const int PointerBufferSize = 0x400;
|
||||||
|
private const int MaxDomains = 64;
|
||||||
|
private const int MaxDomainObjects = 64;
|
||||||
|
private const int MaxPortsCount = 4;
|
||||||
|
|
||||||
|
private SmApi _sm;
|
||||||
|
private BcatServerManager _serverManager;
|
||||||
|
|
||||||
|
private static readonly ManagerOptions _bcatManagerOptions = new(PointerBufferSize, MaxDomains, MaxDomainObjects, false);
|
||||||
|
|
||||||
|
internal void Initialize()
|
||||||
|
{
|
||||||
|
HeapAllocator allocator = new();
|
||||||
|
|
||||||
|
_sm = new SmApi();
|
||||||
|
_sm.Initialize().AbortOnFailure();
|
||||||
|
|
||||||
|
_serverManager = new BcatServerManager(allocator, _sm, MaxPortsCount, _bcatManagerOptions, BcatTotalMaxSessionsCount);
|
||||||
|
|
||||||
|
_serverManager.RegisterServer((int)BcatPortIndex.Admin, ServiceName.Encode("bcat:a"), BcatMaxSessionsCount);
|
||||||
|
_serverManager.RegisterServer((int)BcatPortIndex.Manager, ServiceName.Encode("bcat:m"), BcatMaxSessionsCount);
|
||||||
|
_serverManager.RegisterServer((int)BcatPortIndex.User, ServiceName.Encode("bcat:u"), BcatMaxSessionsCount);
|
||||||
|
_serverManager.RegisterServer((int)BcatPortIndex.System, ServiceName.Encode("bcat:s"), BcatMaxSessionsCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ServiceRequests()
|
||||||
|
{
|
||||||
|
_serverManager.ServiceRequests();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Shutdown()
|
||||||
|
{
|
||||||
|
_serverManager.Dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
24
src/Ryujinx.Horizon/Bcat/BcatMain.cs
Normal file
24
src/Ryujinx.Horizon/Bcat/BcatMain.cs
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
using Ryujinx.Horizon.LogManager;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat
|
||||||
|
{
|
||||||
|
internal class BcatMain : IService
|
||||||
|
{
|
||||||
|
public static void Main(ServiceTable serviceTable)
|
||||||
|
{
|
||||||
|
BcatIpcServer ipcServer = new();
|
||||||
|
|
||||||
|
ipcServer.Initialize();
|
||||||
|
|
||||||
|
serviceTable.SignalServiceReady();
|
||||||
|
|
||||||
|
ipcServer.ServiceRequests();
|
||||||
|
ipcServer.Shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
29
src/Ryujinx.Horizon/Bcat/BcatResult.cs
Normal file
29
src/Ryujinx.Horizon/Bcat/BcatResult.cs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat
|
||||||
|
{
|
||||||
|
class BcatResult
|
||||||
|
{
|
||||||
|
private const int ModuleId = 122;
|
||||||
|
|
||||||
|
public static Result Success => new(ModuleId, 0);
|
||||||
|
public static Result InvalidArgument => new(ModuleId, 1);
|
||||||
|
public static Result NotFound => new(ModuleId, 2);
|
||||||
|
public static Result TargetLocked => new(ModuleId, 3);
|
||||||
|
public static Result TargetAlreadyMounted => new(ModuleId, 4);
|
||||||
|
public static Result TargetNotMounted => new(ModuleId, 5);
|
||||||
|
public static Result AlreadyOpen => new(ModuleId, 6);
|
||||||
|
public static Result NotOpen => new(ModuleId, 7);
|
||||||
|
public static Result InternetRequestDenied => new(ModuleId, 8);
|
||||||
|
public static Result ServiceOpenLimitReached => new(ModuleId, 9);
|
||||||
|
public static Result SaveDataNotFound => new(ModuleId, 10);
|
||||||
|
public static Result NetworkServiceAccountNotAvailable => new(ModuleId, 31);
|
||||||
|
public static Result PassphrasePathNotFound => new(ModuleId, 80);
|
||||||
|
public static Result DataVerificationFailed => new(ModuleId, 81);
|
||||||
|
public static Result PermissionDenied => new(ModuleId, 90);
|
||||||
|
public static Result AllocationFailed => new(ModuleId, 91);
|
||||||
|
public static Result InvalidOperation => new(ModuleId, 98);
|
||||||
|
public static Result InvalidDeliveryCacheStorageFile => new(ModuleId, 204);
|
||||||
|
public static Result StorageOpenLimitReached => new(ModuleId, 205);
|
||||||
|
}
|
||||||
|
}
|
28
src/Ryujinx.Horizon/Bcat/BcatServerManager.cs
Normal file
28
src/Ryujinx.Horizon/Bcat/BcatServerManager.cs
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
using Ryujinx.Horizon.Bcat.Ipc;
|
||||||
|
using Ryujinx.Horizon.Bcat.Types;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sm;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat
|
||||||
|
{
|
||||||
|
class BcatServerManager : ServerManager
|
||||||
|
{
|
||||||
|
public BcatServerManager(HeapAllocator allocator, SmApi sm, int maxPorts, ManagerOptions options, int maxSessions) : base(allocator, sm, maxPorts, options, maxSessions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override Result OnNeedsToAccept(int portIndex, Server server)
|
||||||
|
{
|
||||||
|
return (BcatPortIndex)portIndex switch
|
||||||
|
{
|
||||||
|
BcatPortIndex.Admin => AcceptImpl(server, new ServiceCreator("bcat:a", BcatServicePermissionLevel.Admin)),
|
||||||
|
BcatPortIndex.Manager => AcceptImpl(server, new ServiceCreator("bcat:m", BcatServicePermissionLevel.Manager)),
|
||||||
|
BcatPortIndex.User => AcceptImpl(server, new ServiceCreator("bcat:u", BcatServicePermissionLevel.User)),
|
||||||
|
BcatPortIndex.System => AcceptImpl(server, new ServiceCreator("bcat:s", BcatServicePermissionLevel.System)),
|
||||||
|
_ => throw new ArgumentOutOfRangeException(nameof(portIndex)),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
85
src/Ryujinx.Horizon/Bcat/Ipc/ServiceCreator.cs
Normal file
85
src/Ryujinx.Horizon/Bcat/Ipc/ServiceCreator.cs
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
using LibHac.Common;
|
||||||
|
using Ryujinx.Horizon.Bcat.Types;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
using ApplicationId = Ryujinx.Horizon.Sdk.Ncm.ApplicationId;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class ServiceCreator : IServiceCreator, IDisposable
|
||||||
|
{
|
||||||
|
private readonly BcatServicePermissionLevel _permissionLevel;
|
||||||
|
private SharedRef<LibHac.Bcat.Impl.Ipc.IServiceCreator> _libHacService;
|
||||||
|
|
||||||
|
private int _disposalState;
|
||||||
|
|
||||||
|
public ServiceCreator(string serviceName, BcatServicePermissionLevel permissionLevel)
|
||||||
|
{
|
||||||
|
HorizonStatic.Options.BcatClient.Sm.GetService(ref _libHacService, serviceName).ThrowIfFailure();
|
||||||
|
_permissionLevel = permissionLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(0)]
|
||||||
|
public Result CreateBcatService(out IBcatService bcatService, [ClientProcessId] ulong pid)
|
||||||
|
{
|
||||||
|
// TODO: Call arp:r GetApplicationLaunchProperty with the pid to get the TitleId.
|
||||||
|
// Add an instance of nn::bcat::detail::service::core::PassphraseManager.
|
||||||
|
// Add an instance of nn::bcat::detail::service::ServiceMemoryManager.
|
||||||
|
// Add an instance of nn::bcat::detail::service::core::TaskManager who loads "bcat-sys:/" system save data and opens "dc/task.bin".
|
||||||
|
// If the file don't exist, create a new one (with a size of 0x800 bytes) and write 2 empty structs with a size of 0x400 bytes.
|
||||||
|
|
||||||
|
bcatService = new BcatService(_permissionLevel);
|
||||||
|
|
||||||
|
return Result.Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(1)]
|
||||||
|
public Result CreateDeliveryCacheStorageService(out IDeliveryCacheStorageService service, [ClientProcessId] ulong pid)
|
||||||
|
{
|
||||||
|
using var libHacService = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>();
|
||||||
|
|
||||||
|
var resultCode = _libHacService.Get.CreateDeliveryCacheStorageService(ref libHacService.Ref, pid);
|
||||||
|
|
||||||
|
if (resultCode.IsSuccess())
|
||||||
|
{
|
||||||
|
service = new DeliveryCacheStorageService(ref libHacService.Ref);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
service = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resultCode.ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(2)]
|
||||||
|
public Result CreateDeliveryCacheStorageServiceWithApplicationId(out IDeliveryCacheStorageService service, ApplicationId applicationId)
|
||||||
|
{
|
||||||
|
using var libHacService = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>();
|
||||||
|
|
||||||
|
var resultCode = _libHacService.Get.CreateDeliveryCacheStorageServiceWithApplicationId(ref libHacService.Ref, new LibHac.ApplicationId(applicationId.Id));
|
||||||
|
|
||||||
|
if (resultCode.IsSuccess())
|
||||||
|
{
|
||||||
|
service = new DeliveryCacheStorageService(ref libHacService.Ref);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
service = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resultCode.ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (Interlocked.Exchange(ref _disposalState, 1) == 0)
|
||||||
|
{
|
||||||
|
_libHacService.Destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
25
src/Ryujinx.Horizon/Bcat/Ipc/ServiceCreator/BcatService.cs
Normal file
25
src/Ryujinx.Horizon/Bcat/Ipc/ServiceCreator/BcatService.cs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
using Ryujinx.Horizon.Bcat.Types;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class BcatService : IBcatService
|
||||||
|
{
|
||||||
|
private readonly BcatServicePermissionLevel _permissionLevel;
|
||||||
|
|
||||||
|
public BcatService(BcatServicePermissionLevel permissionLevel)
|
||||||
|
{
|
||||||
|
_permissionLevel = permissionLevel;
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(10100)]
|
||||||
|
public Result RequestSyncDeliveryCache(out IDeliveryCacheProgressService deliveryCacheProgressService)
|
||||||
|
{
|
||||||
|
deliveryCacheProgressService = new DeliveryCacheProgressService();
|
||||||
|
|
||||||
|
return Result.Success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,48 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using LibHac.Common;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class DeliveryCacheDirectoryService : IDeliveryCacheDirectoryService, IDisposable
|
||||||
|
{
|
||||||
|
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService> _libHacService;
|
||||||
|
private int _disposalState;
|
||||||
|
|
||||||
|
public DeliveryCacheDirectoryService(ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService> libHacService)
|
||||||
|
{
|
||||||
|
_libHacService = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService>.CreateMove(ref libHacService);
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(0)]
|
||||||
|
public Result Open(DirectoryName directoryName)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.Open(ref directoryName).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(1)]
|
||||||
|
public Result Read(out int entriesRead, [Buffer(Sdk.Sf.Hipc.HipcBufferFlags.Out | HipcBufferFlags.MapAlias)] Span<DeliveryCacheDirectoryEntry> entriesBuffer)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.Read(out entriesRead, entriesBuffer).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(2)]
|
||||||
|
public Result GetCount(out int count)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.GetCount(out count).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (Interlocked.Exchange(ref _disposalState, 1) == 0)
|
||||||
|
{
|
||||||
|
_libHacService.Destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,54 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using LibHac.Common;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class DeliveryCacheFileService : IDeliveryCacheFileService, IDisposable
|
||||||
|
{
|
||||||
|
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService> _libHacService;
|
||||||
|
private int _disposalState;
|
||||||
|
|
||||||
|
public DeliveryCacheFileService(ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService> libHacService)
|
||||||
|
{
|
||||||
|
_libHacService = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService>.CreateMove(ref libHacService);
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(0)]
|
||||||
|
public Result Open(DirectoryName directoryName, FileName fileName)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.Open(ref directoryName, ref fileName).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(1)]
|
||||||
|
public Result Read(long offset, out long bytesRead, [Buffer(HipcBufferFlags.Out | HipcBufferFlags.MapAlias)] Span<byte> data)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.Read(out bytesRead, offset, data).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(2)]
|
||||||
|
public Result GetSize(out long size)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.GetSize(out size).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(3)]
|
||||||
|
public Result GetDigest(out Digest digest)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.GetDigest(out digest).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (Interlocked.Exchange(ref _disposalState, 1) == 0)
|
||||||
|
{
|
||||||
|
_libHacService.Destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,58 @@
|
|||||||
|
using Ryujinx.Common.Logging;
|
||||||
|
using Ryujinx.Horizon.Bcat.Ipc.Types;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.OsTypes;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class DeliveryCacheProgressService : IDeliveryCacheProgressService, IDisposable
|
||||||
|
{
|
||||||
|
private int _handle;
|
||||||
|
private SystemEventType _systemEvent;
|
||||||
|
private int _disposalState;
|
||||||
|
|
||||||
|
[CmifCommand(0)]
|
||||||
|
public Result GetEvent([CopyHandle] out int handle)
|
||||||
|
{
|
||||||
|
if (_handle == 0)
|
||||||
|
{
|
||||||
|
Os.CreateSystemEvent(out _systemEvent, EventClearMode.ManualClear, true).AbortOnFailure();
|
||||||
|
|
||||||
|
_handle = Os.GetReadableHandleOfSystemEvent(ref _systemEvent);
|
||||||
|
}
|
||||||
|
|
||||||
|
handle = _handle;
|
||||||
|
|
||||||
|
Logger.Stub?.PrintStub(LogClass.ServiceBcat);
|
||||||
|
|
||||||
|
return Result.Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(1)]
|
||||||
|
public Result GetImpl([Buffer(HipcBufferFlags.Out | HipcBufferFlags.Pointer, 0x200)] out DeliveryCacheProgressImpl deliveryCacheProgressImpl)
|
||||||
|
{
|
||||||
|
deliveryCacheProgressImpl = new DeliveryCacheProgressImpl
|
||||||
|
{
|
||||||
|
State = DeliveryCacheProgressImpl.Status.Done,
|
||||||
|
Result = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
Logger.Stub?.PrintStub(LogClass.ServiceBcat);
|
||||||
|
|
||||||
|
return Result.Success;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (_handle != 0 && Interlocked.Exchange(ref _disposalState, 1) == 0)
|
||||||
|
{
|
||||||
|
Os.DestroySystemEvent(ref _systemEvent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,74 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using LibHac.Common;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Bcat;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf.Hipc;
|
||||||
|
using System;
|
||||||
|
using System.Threading;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Bcat.Ipc
|
||||||
|
{
|
||||||
|
partial class DeliveryCacheStorageService : IDeliveryCacheStorageService, IDisposable
|
||||||
|
{
|
||||||
|
private SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService> _libHacService;
|
||||||
|
private int _disposalState;
|
||||||
|
|
||||||
|
public DeliveryCacheStorageService(ref SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService> libHacService)
|
||||||
|
{
|
||||||
|
_libHacService = SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheStorageService>.CreateMove(ref libHacService);
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(0)]
|
||||||
|
public Result CreateFileService(out IDeliveryCacheFileService service)
|
||||||
|
{
|
||||||
|
using var libHacService = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheFileService>();
|
||||||
|
|
||||||
|
var resultCode = _libHacService.Get.CreateFileService(ref libHacService.Ref);
|
||||||
|
|
||||||
|
if (resultCode.IsSuccess())
|
||||||
|
{
|
||||||
|
service = new DeliveryCacheFileService(ref libHacService.Ref);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
service = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resultCode.ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(1)]
|
||||||
|
public Result CreateDirectoryService(out IDeliveryCacheDirectoryService service)
|
||||||
|
{
|
||||||
|
using var libHacService = new SharedRef<LibHac.Bcat.Impl.Ipc.IDeliveryCacheDirectoryService>();
|
||||||
|
|
||||||
|
var resultCode = _libHacService.Get.CreateDirectoryService(ref libHacService.Ref);
|
||||||
|
|
||||||
|
if (resultCode.IsSuccess())
|
||||||
|
{
|
||||||
|
service = new DeliveryCacheDirectoryService(ref libHacService.Ref);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
service = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resultCode.ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
[CmifCommand(10)]
|
||||||
|
public Result EnumerateDeliveryCacheDirectory(out int count, [Buffer(HipcBufferFlags.Out | HipcBufferFlags.MapAlias)] Span<DirectoryName> directoryNames)
|
||||||
|
{
|
||||||
|
return _libHacService.Get.EnumerateDeliveryCacheDirectory(out count, directoryNames).ToHorizonResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (Interlocked.Exchange(ref _disposalState, 1) == 0)
|
||||||
|
{
|
||||||
|
_libHacService.Destroy();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -1,6 +1,6 @@
|
|||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
|
|
||||||
namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator.Types
|
namespace Ryujinx.Horizon.Bcat.Ipc.Types
|
||||||
{
|
{
|
||||||
[StructLayout(LayoutKind.Sequential, Size = 0x200)]
|
[StructLayout(LayoutKind.Sequential, Size = 0x200)]
|
||||||
public struct DeliveryCacheProgressImpl
|
public struct DeliveryCacheProgressImpl
|
10
src/Ryujinx.Horizon/Bcat/Types/BcatPortIndex.cs
Normal file
10
src/Ryujinx.Horizon/Bcat/Types/BcatPortIndex.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Ryujinx.Horizon.Bcat.Types
|
||||||
|
{
|
||||||
|
enum BcatPortIndex
|
||||||
|
{
|
||||||
|
Admin,
|
||||||
|
Manager,
|
||||||
|
User,
|
||||||
|
System
|
||||||
|
}
|
||||||
|
}
|
10
src/Ryujinx.Horizon/Bcat/Types/BcatServicePermissionLevel.cs
Normal file
10
src/Ryujinx.Horizon/Bcat/Types/BcatServicePermissionLevel.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Ryujinx.Horizon.Bcat.Types
|
||||||
|
{
|
||||||
|
enum BcatServicePermissionLevel
|
||||||
|
{
|
||||||
|
Admin = -1,
|
||||||
|
User = 1,
|
||||||
|
System = 2,
|
||||||
|
Manager = 6
|
||||||
|
}
|
||||||
|
}
|
@@ -1,3 +1,5 @@
|
|||||||
|
using LibHac;
|
||||||
|
|
||||||
namespace Ryujinx.Horizon
|
namespace Ryujinx.Horizon
|
||||||
{
|
{
|
||||||
public struct HorizonOptions
|
public struct HorizonOptions
|
||||||
@@ -5,10 +7,13 @@ namespace Ryujinx.Horizon
|
|||||||
public bool IgnoreMissingServices { get; }
|
public bool IgnoreMissingServices { get; }
|
||||||
public bool ThrowOnInvalidCommandIds { get; }
|
public bool ThrowOnInvalidCommandIds { get; }
|
||||||
|
|
||||||
public HorizonOptions(bool ignoreMissingServices)
|
public HorizonClient BcatClient { get; }
|
||||||
|
|
||||||
|
public HorizonOptions(bool ignoreMissingServices, HorizonClient bcatClient)
|
||||||
{
|
{
|
||||||
IgnoreMissingServices = ignoreMissingServices;
|
IgnoreMissingServices = ignoreMissingServices;
|
||||||
ThrowOnInvalidCommandIds = true;
|
ThrowOnInvalidCommandIds = true;
|
||||||
|
BcatClient = bcatClient;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
12
src/Ryujinx.Horizon/LibHacResultExtensions.cs
Normal file
12
src/Ryujinx.Horizon/LibHacResultExtensions.cs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon
|
||||||
|
{
|
||||||
|
internal static class LibHacResultExtensions
|
||||||
|
{
|
||||||
|
public static Result ToHorizonResult(this LibHac.Result result)
|
||||||
|
{
|
||||||
|
return new Result((int)result.Module, (int)result.Description);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -9,12 +9,12 @@ namespace Ryujinx.Horizon.Prepo
|
|||||||
private const int PrepoMaxSessionsCount = 12;
|
private const int PrepoMaxSessionsCount = 12;
|
||||||
private const int PrepoTotalMaxSessionsCount = PrepoMaxSessionsCount * 6;
|
private const int PrepoTotalMaxSessionsCount = PrepoMaxSessionsCount * 6;
|
||||||
|
|
||||||
private const int PointerBufferSize = 0x3800;
|
private const int PointerBufferSize = 0x80;
|
||||||
private const int MaxDomains = 64;
|
private const int MaxDomains = 64;
|
||||||
private const int MaxDomainObjects = 16;
|
private const int MaxDomainObjects = 16;
|
||||||
private const int MaxPortsCount = 6;
|
private const int MaxPortsCount = 6;
|
||||||
|
|
||||||
private static readonly ManagerOptions _logManagerOptions = new(PointerBufferSize, MaxDomains, MaxDomainObjects, false);
|
private static readonly ManagerOptions _prepoManagerOptions = new(PointerBufferSize, MaxDomains, MaxDomainObjects, false);
|
||||||
|
|
||||||
private SmApi _sm;
|
private SmApi _sm;
|
||||||
private PrepoServerManager _serverManager;
|
private PrepoServerManager _serverManager;
|
||||||
@@ -26,7 +26,7 @@ namespace Ryujinx.Horizon.Prepo
|
|||||||
_sm = new SmApi();
|
_sm = new SmApi();
|
||||||
_sm.Initialize().AbortOnFailure();
|
_sm.Initialize().AbortOnFailure();
|
||||||
|
|
||||||
_serverManager = new PrepoServerManager(allocator, _sm, MaxPortsCount, _logManagerOptions, PrepoTotalMaxSessionsCount);
|
_serverManager = new PrepoServerManager(allocator, _sm, MaxPortsCount, _prepoManagerOptions, PrepoTotalMaxSessionsCount);
|
||||||
|
|
||||||
_serverManager.RegisterServer((int)PrepoPortIndex.Admin, ServiceName.Encode("prepo:a"), PrepoMaxSessionsCount); // 1.0.0-5.1.0
|
_serverManager.RegisterServer((int)PrepoPortIndex.Admin, ServiceName.Encode("prepo:a"), PrepoMaxSessionsCount); // 1.0.0-5.1.0
|
||||||
_serverManager.RegisterServer((int)PrepoPortIndex.Admin2, ServiceName.Encode("prepo:a2"), PrepoMaxSessionsCount); // 6.0.0+
|
_serverManager.RegisterServer((int)PrepoPortIndex.Admin2, ServiceName.Encode("prepo:a2"), PrepoMaxSessionsCount); // 6.0.0+
|
||||||
|
10
src/Ryujinx.Horizon/Sdk/Bcat/IBcatService.cs
Normal file
10
src/Ryujinx.Horizon/Sdk/Bcat/IBcatService.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IBcatService : IServiceObject
|
||||||
|
{
|
||||||
|
Result RequestSyncDeliveryCache(out IDeliveryCacheProgressService deliveryCacheProgressService);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,14 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IDeliveryCacheDirectoryService : IServiceObject
|
||||||
|
{
|
||||||
|
Result GetCount(out int count);
|
||||||
|
Result Open(DirectoryName directoryName);
|
||||||
|
Result Read(out int entriesRead, Span<DeliveryCacheDirectoryEntry> entriesBuffer);
|
||||||
|
}
|
||||||
|
}
|
15
src/Ryujinx.Horizon/Sdk/Bcat/IDeliveryCacheFileService.cs
Normal file
15
src/Ryujinx.Horizon/Sdk/Bcat/IDeliveryCacheFileService.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IDeliveryCacheFileService : IServiceObject
|
||||||
|
{
|
||||||
|
Result GetDigest(out Digest digest);
|
||||||
|
Result GetSize(out long size);
|
||||||
|
Result Open(DirectoryName directoryName, FileName fileName);
|
||||||
|
Result Read(long offset, out long bytesRead, Span<byte> data);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,12 @@
|
|||||||
|
using Ryujinx.Horizon.Bcat.Ipc.Types;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IDeliveryCacheProgressService : IServiceObject
|
||||||
|
{
|
||||||
|
Result GetEvent(out int handle);
|
||||||
|
Result GetImpl(out DeliveryCacheProgressImpl deliveryCacheProgressImpl);
|
||||||
|
}
|
||||||
|
}
|
14
src/Ryujinx.Horizon/Sdk/Bcat/IDeliveryCacheStorageService.cs
Normal file
14
src/Ryujinx.Horizon/Sdk/Bcat/IDeliveryCacheStorageService.cs
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
using LibHac.Bcat;
|
||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IDeliveryCacheStorageService : IServiceObject
|
||||||
|
{
|
||||||
|
Result CreateDirectoryService(out IDeliveryCacheDirectoryService service);
|
||||||
|
Result CreateFileService(out IDeliveryCacheFileService service);
|
||||||
|
Result EnumerateDeliveryCacheDirectory(out int count, Span<DirectoryName> directoryNames);
|
||||||
|
}
|
||||||
|
}
|
12
src/Ryujinx.Horizon/Sdk/Bcat/IServiceCreator.cs
Normal file
12
src/Ryujinx.Horizon/Sdk/Bcat/IServiceCreator.cs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
using Ryujinx.Horizon.Common;
|
||||||
|
using Ryujinx.Horizon.Sdk.Sf;
|
||||||
|
|
||||||
|
namespace Ryujinx.Horizon.Sdk.Bcat
|
||||||
|
{
|
||||||
|
internal interface IServiceCreator : IServiceObject
|
||||||
|
{
|
||||||
|
Result CreateBcatService(out IBcatService service, ulong pid);
|
||||||
|
Result CreateDeliveryCacheStorageService(out IDeliveryCacheStorageService service, ulong pid);
|
||||||
|
Result CreateDeliveryCacheStorageServiceWithApplicationId(out IDeliveryCacheStorageService service, Ncm.ApplicationId applicationId);
|
||||||
|
}
|
||||||
|
}
|
@@ -165,6 +165,12 @@ namespace Ryujinx.Horizon.Sdk.Sf.Cmif
|
|||||||
|
|
||||||
entry.Owner = null;
|
entry.Owner = null;
|
||||||
obj = entry.Obj;
|
obj = entry.Obj;
|
||||||
|
|
||||||
|
if (obj.ServiceObject is IDisposable disposableObj)
|
||||||
|
{
|
||||||
|
disposableObj.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
entry.Obj = null;
|
entry.Obj = null;
|
||||||
_entries.Remove(entry.Node);
|
_entries.Remove(entry.Node);
|
||||||
entry.Node = null;
|
entry.Node = null;
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
using Ryujinx.Horizon.Bcat;
|
||||||
using Ryujinx.Horizon.LogManager;
|
using Ryujinx.Horizon.LogManager;
|
||||||
using Ryujinx.Horizon.Prepo;
|
using Ryujinx.Horizon.Prepo;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
@@ -23,6 +24,7 @@ namespace Ryujinx.Horizon
|
|||||||
|
|
||||||
RegisterService<LmMain>();
|
RegisterService<LmMain>();
|
||||||
RegisterService<PrepoMain>();
|
RegisterService<PrepoMain>();
|
||||||
|
RegisterService<BcatMain>();
|
||||||
|
|
||||||
_totalServices = entries.Count;
|
_totalServices = entries.Count;
|
||||||
|
|
||||||
|
@@ -63,6 +63,7 @@ namespace Ryujinx.SDL2.Common
|
|||||||
SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS, "1");
|
SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS, "1");
|
||||||
SDL_SetHint(SDL_HINT_JOYSTICK_HIDAPI_SWITCH_HOME_LED, "0");
|
SDL_SetHint(SDL_HINT_JOYSTICK_HIDAPI_SWITCH_HOME_LED, "0");
|
||||||
SDL_SetHint(SDL_HINT_JOYSTICK_HIDAPI_JOY_CONS, "1");
|
SDL_SetHint(SDL_HINT_JOYSTICK_HIDAPI_JOY_CONS, "1");
|
||||||
|
SDL_SetHint(SDL_HINT_VIDEO_ALLOW_SCREENSAVER, "1");
|
||||||
|
|
||||||
|
|
||||||
// NOTE: As of SDL2 2.24.0, joycons are combined by default but the motion source only come from one of them.
|
// NOTE: As of SDL2 2.24.0, joycons are combined by default but the motion source only come from one of them.
|
||||||
|
@@ -10,7 +10,9 @@ using LibHac.Tools.FsSystem.NcaUtils;
|
|||||||
using Ryujinx.Common.Logging;
|
using Ryujinx.Common.Logging;
|
||||||
using Ryujinx.HLE.FileSystem;
|
using Ryujinx.HLE.FileSystem;
|
||||||
using System;
|
using System;
|
||||||
|
using System.Globalization;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
namespace Ryujinx.Ui.App.Common
|
namespace Ryujinx.Ui.App.Common
|
||||||
{
|
{
|
||||||
@@ -24,13 +26,28 @@ namespace Ryujinx.Ui.App.Common
|
|||||||
public string Version { get; set; }
|
public string Version { get; set; }
|
||||||
public string TimePlayed { get; set; }
|
public string TimePlayed { get; set; }
|
||||||
public double TimePlayedNum { get; set; }
|
public double TimePlayedNum { get; set; }
|
||||||
public string LastPlayed { get; set; }
|
public DateTime? LastPlayed { get; set; }
|
||||||
public string FileExtension { get; set; }
|
public string FileExtension { get; set; }
|
||||||
public string FileSize { get; set; }
|
public string FileSize { get; set; }
|
||||||
public double FileSizeBytes { get; set; }
|
public double FileSizeBytes { get; set; }
|
||||||
public string Path { get; set; }
|
public string Path { get; set; }
|
||||||
public BlitStruct<ApplicationControlProperty> ControlHolder { get; set; }
|
public BlitStruct<ApplicationControlProperty> ControlHolder { get; set; }
|
||||||
|
|
||||||
|
[JsonIgnore]
|
||||||
|
public string LastPlayedString
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (!LastPlayed.HasValue)
|
||||||
|
{
|
||||||
|
// TODO: maybe put localized string here instead of just "Never"
|
||||||
|
return "Never";
|
||||||
|
}
|
||||||
|
|
||||||
|
return LastPlayed.Value.ToLocalTime().ToString(CultureInfo.CurrentCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static string GetApplicationBuildId(VirtualFileSystem virtualFileSystem, string titleFilePath)
|
public static string GetApplicationBuildId(VirtualFileSystem virtualFileSystem, string titleFilePath)
|
||||||
{
|
{
|
||||||
using FileStream file = new(titleFilePath, FileMode.Open, FileAccess.Read);
|
using FileStream file = new(titleFilePath, FileMode.Open, FileAccess.Read);
|
||||||
|
@@ -414,21 +414,28 @@ namespace Ryujinx.Ui.App.Common
|
|||||||
ApplicationMetadata appMetadata = LoadAndSaveMetaData(titleId, appMetadata =>
|
ApplicationMetadata appMetadata = LoadAndSaveMetaData(titleId, appMetadata =>
|
||||||
{
|
{
|
||||||
appMetadata.Title = titleName;
|
appMetadata.Title = titleName;
|
||||||
});
|
|
||||||
|
|
||||||
if (appMetadata.LastPlayed != "Never")
|
if (appMetadata.LastPlayedOld == default || appMetadata.LastPlayed.HasValue)
|
||||||
{
|
{
|
||||||
if (!DateTime.TryParse(appMetadata.LastPlayed, out _))
|
// Don't do the migration if last_played doesn't exist or last_played_utc already has a value.
|
||||||
{
|
return;
|
||||||
Logger.Warning?.Print(LogClass.Application, $"Last played datetime \"{appMetadata.LastPlayed}\" is invalid for current system culture, skipping (did current culture change?)");
|
}
|
||||||
|
|
||||||
appMetadata.LastPlayed = "Never";
|
// Migrate from string-based last_played to DateTime-based last_played_utc.
|
||||||
|
if (DateTime.TryParse(appMetadata.LastPlayedOld, out DateTime lastPlayedOldParsed))
|
||||||
|
{
|
||||||
|
Logger.Info?.Print(LogClass.Application, $"last_played found: \"{appMetadata.LastPlayedOld}\", migrating to last_played_utc");
|
||||||
|
appMetadata.LastPlayed = lastPlayedOldParsed;
|
||||||
|
|
||||||
|
// Migration successful: deleting last_played from the metadata file.
|
||||||
|
appMetadata.LastPlayedOld = default;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
appMetadata.LastPlayed = appMetadata.LastPlayed[..^3];
|
// Migration failed: emitting warning but leaving the unparsable value in the metadata file so the user can fix it.
|
||||||
}
|
Logger.Warning?.Print(LogClass.Application, $"Last played string \"{appMetadata.LastPlayedOld}\" is invalid for current system culture, skipping (did current culture change?)");
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
|
||||||
ApplicationData data = new()
|
ApplicationData data = new()
|
||||||
{
|
{
|
||||||
|
@@ -1,10 +1,19 @@
|
|||||||
namespace Ryujinx.Ui.App.Common
|
using System;
|
||||||
|
using System.Text.Json.Serialization;
|
||||||
|
|
||||||
|
namespace Ryujinx.Ui.App.Common
|
||||||
{
|
{
|
||||||
public class ApplicationMetadata
|
public class ApplicationMetadata
|
||||||
{
|
{
|
||||||
public string Title { get; set; }
|
public string Title { get; set; }
|
||||||
public bool Favorite { get; set; }
|
public bool Favorite { get; set; }
|
||||||
public double TimePlayed { get; set; }
|
public double TimePlayed { get; set; }
|
||||||
public string LastPlayed { get; set; } = "Never";
|
|
||||||
|
[JsonPropertyName("last_played_utc")]
|
||||||
|
public DateTime? LastPlayed { get; set; } = null;
|
||||||
|
|
||||||
|
[JsonPropertyName("last_played")]
|
||||||
|
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingDefault)]
|
||||||
|
public string LastPlayedOld { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -1,5 +1,6 @@
|
|||||||
using Gdk;
|
using Gdk;
|
||||||
using Gtk;
|
using Gtk;
|
||||||
|
using Ryujinx.Common;
|
||||||
using Ryujinx.Ui;
|
using Ryujinx.Ui;
|
||||||
using Ryujinx.Ui.Common.Configuration;
|
using Ryujinx.Ui.Common.Configuration;
|
||||||
using Ryujinx.Ui.Common.Helper;
|
using Ryujinx.Ui.Common.Helper;
|
||||||
@@ -47,9 +48,19 @@ namespace Ryujinx.Modules
|
|||||||
if (_restartQuery)
|
if (_restartQuery)
|
||||||
{
|
{
|
||||||
string ryuName = OperatingSystem.IsWindows() ? "Ryujinx.exe" : "Ryujinx";
|
string ryuName = OperatingSystem.IsWindows() ? "Ryujinx.exe" : "Ryujinx";
|
||||||
string ryuExe = System.IO.Path.Combine(AppDomain.CurrentDomain.BaseDirectory, ryuName);
|
|
||||||
|
|
||||||
Process.Start(ryuExe, CommandLineState.Arguments);
|
ProcessStartInfo processStart = new(ryuName)
|
||||||
|
{
|
||||||
|
UseShellExecute = true,
|
||||||
|
WorkingDirectory = ReleaseInformation.GetBaseApplicationDirectory()
|
||||||
|
};
|
||||||
|
|
||||||
|
foreach (string argument in CommandLineState.Arguments)
|
||||||
|
{
|
||||||
|
processStart.ArgumentList.Add(argument);
|
||||||
|
}
|
||||||
|
|
||||||
|
Process.Start(processStart);
|
||||||
|
|
||||||
Environment.Exit(0);
|
Environment.Exit(0);
|
||||||
}
|
}
|
||||||
|
@@ -876,7 +876,7 @@ namespace Ryujinx.Ui
|
|||||||
|
|
||||||
_applicationLibrary.LoadAndSaveMetaData(_emulationContext.Processes.ActiveApplication.ProgramIdText, appMetadata =>
|
_applicationLibrary.LoadAndSaveMetaData(_emulationContext.Processes.ActiveApplication.ProgramIdText, appMetadata =>
|
||||||
{
|
{
|
||||||
appMetadata.LastPlayed = DateTime.UtcNow.ToString();
|
appMetadata.LastPlayed = DateTime.UtcNow;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1019,10 +1019,11 @@ namespace Ryujinx.Ui
|
|||||||
{
|
{
|
||||||
_applicationLibrary.LoadAndSaveMetaData(titleId, appMetadata =>
|
_applicationLibrary.LoadAndSaveMetaData(titleId, appMetadata =>
|
||||||
{
|
{
|
||||||
DateTime lastPlayedDateTime = DateTime.Parse(appMetadata.LastPlayed);
|
if (appMetadata.LastPlayed.HasValue)
|
||||||
double sessionTimePlayed = DateTime.UtcNow.Subtract(lastPlayedDateTime).TotalSeconds;
|
{
|
||||||
|
double sessionTimePlayed = DateTime.UtcNow.Subtract(appMetadata.LastPlayed.Value).TotalSeconds;
|
||||||
appMetadata.TimePlayed += Math.Round(sessionTimePlayed, MidpointRounding.AwayFromZero);
|
appMetadata.TimePlayed += Math.Round(sessionTimePlayed, MidpointRounding.AwayFromZero);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1089,7 +1090,7 @@ namespace Ryujinx.Ui
|
|||||||
args.AppData.Developer,
|
args.AppData.Developer,
|
||||||
args.AppData.Version,
|
args.AppData.Version,
|
||||||
args.AppData.TimePlayed,
|
args.AppData.TimePlayed,
|
||||||
args.AppData.LastPlayed,
|
args.AppData.LastPlayedString,
|
||||||
args.AppData.FileExtension,
|
args.AppData.FileExtension,
|
||||||
args.AppData.FileSize,
|
args.AppData.FileSize,
|
||||||
args.AppData.Path,
|
args.AppData.Path,
|
||||||
|
@@ -321,13 +321,12 @@ namespace Ryujinx.Ui
|
|||||||
|
|
||||||
_toggleDockedMode = toggleDockedMode;
|
_toggleDockedMode = toggleDockedMode;
|
||||||
|
|
||||||
if (ConfigurationState.Instance.Hid.EnableMouse.Value)
|
|
||||||
{
|
|
||||||
if (_isMouseInClient)
|
if (_isMouseInClient)
|
||||||
|
{
|
||||||
|
if (ConfigurationState.Instance.Hid.EnableMouse.Value)
|
||||||
{
|
{
|
||||||
Window.Cursor = _invisibleCursor;
|
Window.Cursor = _invisibleCursor;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
switch (_hideCursorMode)
|
switch (_hideCursorMode)
|
||||||
@@ -345,6 +344,7 @@ namespace Ryujinx.Ui
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void Initialize(Switch device)
|
public void Initialize(Switch device)
|
||||||
{
|
{
|
||||||
|
@@ -180,7 +180,7 @@ namespace Ryujinx.Ui.Windows
|
|||||||
|
|
||||||
if (response.IsSuccessStatusCode)
|
if (response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
return response.Content.Headers.LastModified != oldLastModified;
|
return response.Content.Headers.LastModified != new DateTimeOffset(oldLastModified.Ticks - (oldLastModified.Ticks % TimeSpan.TicksPerSecond), TimeSpan.Zero);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
Reference in New Issue
Block a user