Compare commits
4 Commits
1.11.1
...
gcc15_supp
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4859bcc0fe | ||
|
|
0ecd822c26 | ||
|
|
b1e0dad506 | ||
|
|
c55606cfdb |
@@ -4,7 +4,6 @@
|
|||||||
|
|
||||||

|

|
||||||
[](https://discordapp.com/invite/ZdqEhed)
|
[](https://discordapp.com/invite/ZdqEhed)
|
||||||

|
|
||||||
|
|
||||||
Atmosphère is a work-in-progress customized firmware for the Nintendo Switch.
|
Atmosphère is a work-in-progress customized firmware for the Nintendo Switch.
|
||||||
|
|
||||||
@@ -14,6 +13,7 @@ Components
|
|||||||
Atmosphère consists of multiple components, each of which replaces/modifies a different component of the system:
|
Atmosphère consists of multiple components, each of which replaces/modifies a different component of the system:
|
||||||
|
|
||||||
* Fusée: First-stage Loader, responsible for loading and validating stage 2 (custom TrustZone) plus package2 (Kernel/FIRM sysmodules), and patching them as needed. This replaces all functionality normally in Package1loader/NX Bootloader.
|
* Fusée: First-stage Loader, responsible for loading and validating stage 2 (custom TrustZone) plus package2 (Kernel/FIRM sysmodules), and patching them as needed. This replaces all functionality normally in Package1loader/NX Bootloader.
|
||||||
|
* Sept: Payload used to enable support for runtime key derivation on 7.0.0.
|
||||||
* Exosphère: Customized TrustZone, to run a customized Secure Monitor
|
* Exosphère: Customized TrustZone, to run a customized Secure Monitor
|
||||||
* Thermosphère: EL2 EmuNAND support, i.e. backing up and using virtualized/redirected NAND images
|
* Thermosphère: EL2 EmuNAND support, i.e. backing up and using virtualized/redirected NAND images
|
||||||
* Stratosphère: Custom Sysmodule(s), both Rosalina style to extend the kernel/provide new features, and of the loader reimplementation style to hook important system actions
|
* Stratosphère: Custom Sysmodule(s), both Rosalina style to extend the kernel/provide new features, and of the loader reimplementation style to hook important system actions
|
||||||
|
|||||||
@@ -50,7 +50,6 @@ dist: dist-no-debug
|
|||||||
cp $(CURRENT_DIRECTORY)/stratosphere/sm/$(ATMOSPHERE_OUT_DIR)/sm.elf $(DIST_DIR)/sm.elf
|
cp $(CURRENT_DIRECTORY)/stratosphere/sm/$(ATMOSPHERE_OUT_DIR)/sm.elf $(DIST_DIR)/sm.elf
|
||||||
cp $(CURRENT_DIRECTORY)/stratosphere/spl/$(ATMOSPHERE_OUT_DIR)/spl.elf $(DIST_DIR)/spl.elf
|
cp $(CURRENT_DIRECTORY)/stratosphere/spl/$(ATMOSPHERE_OUT_DIR)/spl.elf $(DIST_DIR)/spl.elf
|
||||||
cp $(CURRENT_DIRECTORY)/stratosphere/TioServer/$(ATMOSPHERE_OUT_DIR)/TioServer.elf $(DIST_DIR)/TioServer.elf
|
cp $(CURRENT_DIRECTORY)/stratosphere/TioServer/$(ATMOSPHERE_OUT_DIR)/TioServer.elf $(DIST_DIR)/TioServer.elf
|
||||||
cp $(CURRENT_DIRECTORY)/stratosphere/memlet/$(ATMOSPHERE_OUT_DIR)/memlet.elf $(DIST_DIR)/memlet.elf
|
|
||||||
cp $(CURRENT_DIRECTORY)/troposphere/daybreak/daybreak.elf $(DIST_DIR)/daybreak.elf
|
cp $(CURRENT_DIRECTORY)/troposphere/daybreak/daybreak.elf $(DIST_DIR)/daybreak.elf
|
||||||
cp $(CURRENT_DIRECTORY)/troposphere/haze/haze.elf $(DIST_DIR)/haze.elf
|
cp $(CURRENT_DIRECTORY)/troposphere/haze/haze.elf $(DIST_DIR)/haze.elf
|
||||||
cp $(CURRENT_DIRECTORY)/troposphere/reboot_to_payload/reboot_to_payload.elf $(DIST_DIR)/reboot_to_payload.elf
|
cp $(CURRENT_DIRECTORY)/troposphere/reboot_to_payload/reboot_to_payload.elf $(DIST_DIR)/reboot_to_payload.elf
|
||||||
@@ -88,7 +87,6 @@ dist-no-debug: package3 $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
|||||||
#mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000003c
|
#mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000003c
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000042
|
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000042
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000420
|
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000420
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000421
|
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000b240
|
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000b240
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d609
|
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d609
|
||||||
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d623
|
mkdir -p $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d623
|
||||||
@@ -106,8 +104,7 @@ dist-no-debug: package3 $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
|||||||
cp stratosphere/htc/$(ATMOSPHERE_OUT_DIR)/htc.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000b240/exefs.nsp
|
cp stratosphere/htc/$(ATMOSPHERE_OUT_DIR)/htc.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000b240/exefs.nsp
|
||||||
cp stratosphere/dmnt.gen2/$(ATMOSPHERE_OUT_DIR)/dmnt.gen2.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d609/exefs.nsp
|
cp stratosphere/dmnt.gen2/$(ATMOSPHERE_OUT_DIR)/dmnt.gen2.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d609/exefs.nsp
|
||||||
cp stratosphere/TioServer/$(ATMOSPHERE_OUT_DIR)/TioServer.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d623/exefs.nsp
|
cp stratosphere/TioServer/$(ATMOSPHERE_OUT_DIR)/TioServer.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/010000000000d623/exefs.nsp
|
||||||
cp stratosphere/memlet/$(ATMOSPHERE_OUT_DIR)/memlet.nsp $(DIST_DIR)/stratosphere_romfs/atmosphere/contents/0100000000000421/exefs.nsp
|
@build_romfs $(DIST_DIR)/stratosphere_romfs $(DIST_DIR)/atmosphere/stratosphere.romfs
|
||||||
@PATH="$(DEVKITPRO)/tools/bin:$$PATH" build_romfs $(DIST_DIR)/stratosphere_romfs $(DIST_DIR)/atmosphere/stratosphere.romfs
|
|
||||||
rm -r $(DIST_DIR)/stratosphere_romfs
|
rm -r $(DIST_DIR)/stratosphere_romfs
|
||||||
cp troposphere/reboot_to_payload/reboot_to_payload.nro $(DIST_DIR)/switch/reboot_to_payload.nro
|
cp troposphere/reboot_to_payload/reboot_to_payload.nro $(DIST_DIR)/switch/reboot_to_payload.nro
|
||||||
cp troposphere/daybreak/daybreak.nro $(DIST_DIR)/switch/daybreak.nro
|
cp troposphere/daybreak/daybreak.nro $(DIST_DIR)/switch/daybreak.nro
|
||||||
@@ -117,7 +114,7 @@ dist-no-debug: package3 $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
|||||||
cp fusee/$(ATMOSPHERE_BOOT_OUT_DIR)/fusee.bin $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)/fusee.bin
|
cp fusee/$(ATMOSPHERE_BOOT_OUT_DIR)/fusee.bin $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)/fusee.bin
|
||||||
|
|
||||||
package3: emummc fusee stratosphere mesosphere exosphere troposphere
|
package3: emummc fusee stratosphere mesosphere exosphere troposphere
|
||||||
$(SILENTCMD)$(PYTHON) fusee/build_package3.py $(CURRENT_DIRECTORY) $(ATMOSPHERE_OUT_DIR) $(ATMOSPHERE_BOOT_OUT_DIR) $(ATMOSPHERE_GIT_HASH) $(ATMOSPHERE_MAJOR_VERSION) $(ATMOSPHERE_MINOR_VERSION) $(ATMOSPHERE_MICRO_VERSION) 0 $(ATMOSPHERE_SUPPORTED_HOS_MAJOR_VERSION) $(ATMOSPHERE_SUPPORTED_HOS_MINOR_VERSION) $(ATMOSPHERE_SUPPORTED_HOS_MICRO_VERSION) 0
|
@python fusee/build_package3.py $(CURRENT_DIRECTORY) $(ATMOSPHERE_OUT_DIR) $(ATMOSPHERE_BOOT_OUT_DIR) $(ATMOSPHERE_GIT_HASH) $(ATMOSPHERE_MAJOR_VERSION) $(ATMOSPHERE_MINOR_VERSION) $(ATMOSPHERE_MICRO_VERSION) 0 $(ATMOSPHERE_SUPPORTED_HOS_MAJOR_VERSION) $(ATMOSPHERE_SUPPORTED_HOS_MINOR_VERSION) $(ATMOSPHERE_SUPPORTED_HOS_MICRO_VERSION) 0
|
||||||
@echo "Built package3!"
|
@echo "Built package3!"
|
||||||
|
|
||||||
emummc:
|
emummc:
|
||||||
|
|||||||
@@ -15,13 +15,6 @@
|
|||||||
# Desc: Controls whether userland has access to the PMU registers.
|
# Desc: Controls whether userland has access to the PMU registers.
|
||||||
# NOTE: It is unknown what effects this has on official code.
|
# NOTE: It is unknown what effects this has on official code.
|
||||||
|
|
||||||
# Key: enable_mem_mode, default: 0.
|
|
||||||
# Desc: Controls whether boot config memory mode is taken into account
|
|
||||||
# for retail units. This does not affect development units.
|
|
||||||
# NOTE: On retail units max ram size is capped to 4GB.
|
|
||||||
# Enabling this will use the boot config memory mode parameter,
|
|
||||||
# which by default is auto and size gets set based on physical size.
|
|
||||||
|
|
||||||
# Key: blank_prodinfo_sysmmc, default: 0.
|
# Key: blank_prodinfo_sysmmc, default: 0.
|
||||||
# Desc: Controls whether PRODINFO should be blanked in sysmmc.
|
# Desc: Controls whether PRODINFO should be blanked in sysmmc.
|
||||||
# This will cause the system to see dummied out keys and
|
# This will cause the system to see dummied out keys and
|
||||||
@@ -58,7 +51,6 @@ debugmode=1
|
|||||||
debugmode_user=0
|
debugmode_user=0
|
||||||
disable_user_exception_handlers=0
|
disable_user_exception_handlers=0
|
||||||
enable_user_pmu_access=0
|
enable_user_pmu_access=0
|
||||||
enable_mem_mode=0
|
|
||||||
blank_prodinfo_sysmmc=0
|
blank_prodinfo_sysmmc=0
|
||||||
blank_prodinfo_emummc=0
|
blank_prodinfo_emummc=0
|
||||||
allow_writing_to_cal_sysmmc=0
|
allow_writing_to_cal_sysmmc=0
|
||||||
|
|||||||
@@ -1,99 +1,4 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
## 1.11.0
|
|
||||||
+ Support was added for 22.0.0.
|
|
||||||
+ Special thanks to @alula for handling a large chunk of the necessary kernel, loader and erpt changes.
|
|
||||||
+ The console should boot and atmosphère should be fully functional.
|
|
||||||
+ **Please note**: A change in how applications/applets' lifespan is managed broke the homebrew ecosystem.
|
|
||||||
+ All applications and applets are expected to perform a clean exit by calling the relevant IPC commands.
|
|
||||||
+ Homebrew compiled with libnx and hbmenu itself explicitly avoid exiting so it can keep using the same process.
|
|
||||||
+ Properly fixing this would require a re-design of how homebrew is launched and terminated.
|
|
||||||
+ As a temporary solution, patches to the `am` sysmodule are now included which allow restoring the previous behavior and regain homebrew compatibility without any further changes.
|
|
||||||
+ Nonetheless, please report any issues you may face when testing homebrew to ensure no edge cases have been missed.
|
|
||||||
+ `exosphère` was updated to reflect the latest official secure monitor behavior.
|
|
||||||
+ `mesosphère` was updated to reflect the latest official kernel behavior.
|
|
||||||
+ `loader` was updated to reflect the latest official behavior.
|
|
||||||
+ `erpt` was updated to reflect the latest official behavior.
|
|
||||||
+ `pgl` was updated to fix a few issues.
|
|
||||||
+ Support was added for memory modes above 4GB (thanks @CTCaer).
|
|
||||||
+ A build time option to configure Joy-Con rails as UART for debugging was added (thanks @alula).
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.10.2
|
|
||||||
+ Basic support was added for 21.2.0.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.10.1
|
|
||||||
+ Basic support was added for 21.1.0.
|
|
||||||
+ A bug was fixed that caused some games (e.g. Tomb Raider definitive edition) to fail to launch.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.10.0
|
|
||||||
+ Basic support was added for 21.0.0.
|
|
||||||
+ The console should boot and atmosphère should be fully functional.
|
|
||||||
+ **Please note**: All homebrew software may need to be re-compiled with the latest libnx (>= 4.10.0), or else it may crash/experience memory corruption.
|
|
||||||
+ Nintendo broke the userland<->kernel TLS ABI in 21.0.0, by writing to previously reserved space.
|
|
||||||
+ Homebrew used this reserved space for its TLS slots, which means any homebrew software using TLS slots will experience memory corruption when running under Atmosphere 1.10.0.
|
|
||||||
+ This doesn't appear to impact everything, but a large portion of tested homebrew crashes (often on exit), and so will need re-compile for the new ABI.
|
|
||||||
+ For those technically inclined, while TLS slots are rarely used by developers, they're used to implement features like e.g. C++ exceptions under the hood, and so anything using those crashes, etc.
|
|
||||||
+ To help make this transition easier, hbmenu now shows a warning when selecting homebrew compiled with an older, incompatible ABI version.
|
|
||||||
+ I apologize for the hassle in general.
|
|
||||||
+ libnx has been updated so that its reserved space matches Nintendo's now -- this particular issue can never occur again, even if Nintendo touches more reserved space.
|
|
||||||
+ `exosphère` was updated to reflect the latest official secure monitor behavior.
|
|
||||||
+ `mesosphère` was updated to reflect the latest official kernel behavior.
|
|
||||||
+ `loader` was updated to reflect the latest official behavior.
|
|
||||||
+ `pm` was updated to reflect the latest official behavior.
|
|
||||||
+ `erpt` was updated to reflect the latest official behavior.
|
|
||||||
+ `pgl` was updated to reflect the latest official behavior.
|
|
||||||
+ `fatal` was updated to reflect the latest official behavior.
|
|
||||||
+ Support was added for launching another game-which-has-too-many-files with romfs mods.
|
|
||||||
+ I rely on user reports for adding support/fixing these, and some of these games can be pretty obscure!
|
|
||||||
+ If you are affected by this, you will see "Data abort (0x101)" when trying to launch the game with mods.
|
|
||||||
+ Please reach out to `sciresm` on discord if this occurs to share your error report binary.
|
|
||||||
+ Although some games may be impossible to fix, I believe I can get almost everything working, so please let me try to help you (and improve atmosphère's support!) if you run into this!
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.5
|
|
||||||
+ Basic support was added for 20.5.0.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.4
|
|
||||||
+ Basic support was added for 20.4.0.
|
|
||||||
+ An issue was fixed in `exosphère`'s register accessilibity tables (thanks @CTCaer).
|
|
||||||
+ I believe this had no impact on official code, though it would have prevented some homebrew from interacting correctly with the MC0/MC1 registers.
|
|
||||||
+ An issue was fixed that could cause a deadlock when building multiple romfs images simultaneously (thanks @Ereza).
|
|
||||||
+ This fixes support for certain mods, e.g. system language translations overriding content for both overlayDisp and qlaunch.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.3
|
|
||||||
+ Basic support was added for 20.3.0.
|
|
||||||
+ Compatibility was fixed for loading mods with KOTOR 2 (star wars).
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.2
|
|
||||||
+ Basic support was added for 20.2.0.
|
|
||||||
+ USB 3.0 support force-enable was fixed for 20.1.0+.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.1
|
|
||||||
+ Basic support was added for 20.1.0.
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.9.0
|
|
||||||
+ Basic support was added for 20.0.0.
|
|
||||||
+ The console should boot and atmosphère should be fully functional. However, not all modules have been fully updated to reflect the latest changes.
|
|
||||||
+ There shouldn't be anything user visible resulting from this, but it will be addressed in a future atmosphère update.
|
|
||||||
+ The same action item from 18.0.0 remains, and I believe in my heart of hearts that it will be addressed eventually. Someone has told me they're working on it.
|
|
||||||
+ There aren't (to my knowledge) outstanding 19.0.0 items any more.
|
|
||||||
+ **Please note**: As a result of changes made to nintendo's software in 20.0.0, there is roughly 10MB less memory available for custom system modules.
|
|
||||||
+ We can only steal a maximum of 14MB from the applet pool, down from 40MB.
|
|
||||||
+ To compensate for this, `ams.mitm`'s heap usage has been reduced by 20MB.
|
|
||||||
+ To facilitate this, a new helper module (`memlet`) was added, so that memory may be temporarily stolen during the romfs building process.
|
|
||||||
+ Hopefully, this results in relatively little breakage, however it is possible that user mods which replace extremely large numbers of files in The Legend of Zelda: Tears of the Kingdom may no longer function.
|
|
||||||
+ If you are affected by this, you will see "Data abort (0x101)" when trying to launch the game with mods.
|
|
||||||
+ Please reach out to `sciresm` on discord if this occurs to share your error report binary. However, some issues may be impossible to fix.
|
|
||||||
+ I apologize sincerely if the issue is impossible to resolve, but I have been forced unavoidably to make compromises here, and I think this is the best balance to be struck.
|
|
||||||
+ `exosphère` was updated to reflect the latest official secure monitor behavior.
|
|
||||||
+ `mesosphère` was updated to reflect the latest official kernel behavior.
|
|
||||||
+ `loader` was updated to reflect the latest official behavior.
|
|
||||||
+ `pm` was updated to reflect the latest official behavior.
|
|
||||||
+ `ncm` was partially updated to reflect the latest official behavior.
|
|
||||||
+ `erpt` was updated to reflect the latest official behavior.
|
|
||||||
+ Atmosphère was updated to use GCC 15/newlib (latest devkitA64/devkitARM releases).
|
|
||||||
+ A number of improvements were made to the dmnt cheat engine.
|
|
||||||
+ New instructions were added, and instructions were updated for improved/new functionality.
|
|
||||||
+ Please see the documents for details -- thanks @tomvita!
|
|
||||||
+ General system stability improvements to enhance the user's experience.
|
|
||||||
## 1.8.0
|
## 1.8.0
|
||||||
+ Basic support was added for 19.0.0.
|
+ Basic support was added for 19.0.0.
|
||||||
+ The console should boot and atmosphère should be fully functional. However, not all modules have been fully updated to reflect the latest changes.
|
+ The console should boot and atmosphère should be fully functional. However, not all modules have been fully updated to reflect the latest changes.
|
||||||
|
|||||||
@@ -423,41 +423,41 @@ Note that for multiple button combinations, the bitmasks should be OR'd together
|
|||||||
#### Keypad Values
|
#### Keypad Values
|
||||||
Note: This is the direct output of `hidKeysDown()`.
|
Note: This is the direct output of `hidKeysDown()`.
|
||||||
|
|
||||||
+ 00000000 00000001: A
|
+ 000000001: A
|
||||||
+ 00000000 00000002: B
|
+ 000000002: B
|
||||||
+ 00000000 00000004: X
|
+ 000000004: X
|
||||||
+ 00000000 00000008: Y
|
+ 000000008: Y
|
||||||
+ 00000000 00000010: Left Stick Pressed
|
+ 000000010: Left Stick Pressed
|
||||||
+ 00000000 00000020: Right Stick Pressed
|
+ 000000020: Right Stick Pressed
|
||||||
+ 00000000 00000040: L
|
+ 000000040: L
|
||||||
+ 00000000 00000080: R
|
+ 000000080: R
|
||||||
+ 00000000 00000100: ZL
|
+ 000000100: ZL
|
||||||
+ 00000000 00000200: ZR
|
+ 000000200: ZR
|
||||||
+ 00000000 00000400: Plus
|
+ 000000400: Plus
|
||||||
+ 00000000 00000800: Minus
|
+ 000000800: Minus
|
||||||
+ 00000000 00001000: Left
|
+ 000001000: Left
|
||||||
+ 00000000 00002000: Up
|
+ 000002000: Up
|
||||||
+ 00000000 00004000: Right
|
+ 000004000: Right
|
||||||
+ 00000000 00008000: Down
|
+ 000008000: Down
|
||||||
+ 00000000 00010000: Left Stick Left
|
+ 000010000: Left Stick Left
|
||||||
+ 00000000 00020000: Left Stick Up
|
+ 000020000: Left Stick Up
|
||||||
+ 00000000 00040000: Left Stick Right
|
+ 000040000: Left Stick Right
|
||||||
+ 00000000 00080000: Left Stick Down
|
+ 000080000: Left Stick Down
|
||||||
+ 00000000 00100000: Right Stick Left
|
+ 000100000: Right Stick Left
|
||||||
+ 00000000 00200000: Right Stick Up
|
+ 000200000: Right Stick Up
|
||||||
+ 00000000 00400000: Right Stick Right
|
+ 000400000: Right Stick Right
|
||||||
+ 00000000 00800000: Right Stick Down
|
+ 000800000: Right Stick Down
|
||||||
+ 00000000 01000000: SL Left Joy-Con
|
+ 001000000: SL Left Joy-Con
|
||||||
+ 00000000 02000000: SR Left Joy-Con
|
+ 002000000: SR Left Joy-Con
|
||||||
+ 00000000 04000000: SL Right Joy-Con
|
+ 004000000: SL Right Joy-Con
|
||||||
+ 00000000 08000000: SR Right Joy-Con
|
+ 008000000: SR Right Joy-Con
|
||||||
+ 00000000 10000000: Top button on Poké Ball Plus (Palma) controller
|
+ 010000000: Top button on Poké Ball Plus (Palma) controller
|
||||||
+ 00000000 20000000: Verification
|
+ 020000000: Verification
|
||||||
+ 00000000 40000000: B button on Left NES/HVC controller in Handheld mode
|
+ 040000000: B button on Left NES/HVC controller in Handheld mode
|
||||||
+ 00000000 80000000: Left C button in N64 controller
|
+ 080000000: Left C button in N64 controller
|
||||||
+ 00000001 00000000: Up C button in N64 controller
|
+ 100000000: Up C button in N64 controller
|
||||||
+ 00000002 00000000: Right C button in N64 controller
|
+ 200000000: Right C button in N64 controller
|
||||||
+ 00000004 00000000: Down C button in N64 controller
|
+ 400000000: Down C button in N64 controller
|
||||||
|
|
||||||
### Code Type 0xF0: Double Extended-Width Instruction
|
### Code Type 0xF0: Double Extended-Width Instruction
|
||||||
Code Type 0xF0 signals to the VM to treat the upper three nybbles of the first dword as instruction type, instead of just the upper nybble.
|
Code Type 0xF0 signals to the VM to treat the upper three nybbles of the first dword as instruction type, instead of just the upper nybble.
|
||||||
|
|||||||
6
emummc/.gitrepo
vendored
6
emummc/.gitrepo
vendored
@@ -6,7 +6,7 @@
|
|||||||
[subrepo]
|
[subrepo]
|
||||||
remote = https://github.com/m4xw/emummc
|
remote = https://github.com/m4xw/emummc
|
||||||
branch = develop
|
branch = develop
|
||||||
commit = 3726bfd659600cdafd138277054568a3edba60a6
|
commit = d248ea6f700c3e6987549e30a1e2eeb609ce9f8c
|
||||||
parent = 80bd459516e813fc6f10268ca31dd72a053a4ef3
|
parent = 9112461620330ba73a74926edd4c08b3cc0310f0
|
||||||
method = merge
|
method = merge
|
||||||
cmdver = 0.4.9
|
cmdver = 0.4.1
|
||||||
|
|||||||
2
emummc/README.md
vendored
2
emummc/README.md
vendored
@@ -2,7 +2,7 @@
|
|||||||
*A SDMMC driver replacement for Nintendo's Filesystem Services, by **m4xw***
|
*A SDMMC driver replacement for Nintendo's Filesystem Services, by **m4xw***
|
||||||
|
|
||||||
### Supported Horizon Versions
|
### Supported Horizon Versions
|
||||||
**1.0.0 - 21.2.0**
|
**1.0.0 - 19.0.0**
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
* Arbitrary SDMMC backend selection
|
* Arbitrary SDMMC backend selection
|
||||||
|
|||||||
40
emummc/source/FS/FS_offsets.c
vendored
40
emummc/source/FS/FS_offsets.c
vendored
@@ -75,16 +75,6 @@
|
|||||||
#include "offsets/1810_exfat.h"
|
#include "offsets/1810_exfat.h"
|
||||||
#include "offsets/1900.h"
|
#include "offsets/1900.h"
|
||||||
#include "offsets/1900_exfat.h"
|
#include "offsets/1900_exfat.h"
|
||||||
#include "offsets/2000.h"
|
|
||||||
#include "offsets/2000_exfat.h"
|
|
||||||
#include "offsets/2010.h"
|
|
||||||
#include "offsets/2010_exfat.h"
|
|
||||||
#include "offsets/2100.h"
|
|
||||||
#include "offsets/2100_exfat.h"
|
|
||||||
#include "offsets/2120.h"
|
|
||||||
#include "offsets/2120_exfat.h"
|
|
||||||
#include "offsets/2200.h"
|
|
||||||
#include "offsets/2200_exfat.h"
|
|
||||||
#include "../utils/fatal.h"
|
#include "../utils/fatal.h"
|
||||||
|
|
||||||
#define GET_OFFSET_STRUCT_NAME(vers) g_offsets##vers
|
#define GET_OFFSET_STRUCT_NAME(vers) g_offsets##vers
|
||||||
@@ -171,16 +161,6 @@ DEFINE_OFFSET_STRUCT(_1810);
|
|||||||
DEFINE_OFFSET_STRUCT(_1810_EXFAT);
|
DEFINE_OFFSET_STRUCT(_1810_EXFAT);
|
||||||
DEFINE_OFFSET_STRUCT(_1900);
|
DEFINE_OFFSET_STRUCT(_1900);
|
||||||
DEFINE_OFFSET_STRUCT(_1900_EXFAT);
|
DEFINE_OFFSET_STRUCT(_1900_EXFAT);
|
||||||
DEFINE_OFFSET_STRUCT(_2000);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2000_EXFAT);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2010);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2010_EXFAT);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2100);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2100_EXFAT);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2120);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2120_EXFAT);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2200);
|
|
||||||
DEFINE_OFFSET_STRUCT(_2200_EXFAT);
|
|
||||||
|
|
||||||
const fs_offsets_t *get_fs_offsets(enum FS_VER version) {
|
const fs_offsets_t *get_fs_offsets(enum FS_VER version) {
|
||||||
switch (version) {
|
switch (version) {
|
||||||
@@ -302,26 +282,6 @@ const fs_offsets_t *get_fs_offsets(enum FS_VER version) {
|
|||||||
return &(GET_OFFSET_STRUCT_NAME(_1900));
|
return &(GET_OFFSET_STRUCT_NAME(_1900));
|
||||||
case FS_VER_19_0_0_EXFAT:
|
case FS_VER_19_0_0_EXFAT:
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_1900_EXFAT));
|
return &(GET_OFFSET_STRUCT_NAME(_1900_EXFAT));
|
||||||
case FS_VER_20_0_0:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2000));
|
|
||||||
case FS_VER_20_0_0_EXFAT:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2000_EXFAT));
|
|
||||||
case FS_VER_20_1_0:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2010));
|
|
||||||
case FS_VER_20_1_0_EXFAT:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2010_EXFAT));
|
|
||||||
case FS_VER_21_0_0:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2100));
|
|
||||||
case FS_VER_21_0_0_EXFAT:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2100_EXFAT));
|
|
||||||
case FS_VER_21_2_0:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2120));
|
|
||||||
case FS_VER_21_2_0_EXFAT:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2120_EXFAT));
|
|
||||||
case FS_VER_22_0_0:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2200));
|
|
||||||
case FS_VER_22_0_0_EXFAT:
|
|
||||||
return &(GET_OFFSET_STRUCT_NAME(_2200_EXFAT));
|
|
||||||
default:
|
default:
|
||||||
fatal_abort(Fatal_UnknownVersion);
|
fatal_abort(Fatal_UnknownVersion);
|
||||||
}
|
}
|
||||||
|
|||||||
15
emummc/source/FS/FS_versions.h
vendored
15
emummc/source/FS/FS_versions.h
vendored
@@ -110,21 +110,6 @@ enum FS_VER
|
|||||||
FS_VER_19_0_0,
|
FS_VER_19_0_0,
|
||||||
FS_VER_19_0_0_EXFAT,
|
FS_VER_19_0_0_EXFAT,
|
||||||
|
|
||||||
FS_VER_20_0_0,
|
|
||||||
FS_VER_20_0_0_EXFAT,
|
|
||||||
|
|
||||||
FS_VER_20_1_0,
|
|
||||||
FS_VER_20_1_0_EXFAT,
|
|
||||||
|
|
||||||
FS_VER_21_0_0,
|
|
||||||
FS_VER_21_0_0_EXFAT,
|
|
||||||
|
|
||||||
FS_VER_21_2_0,
|
|
||||||
FS_VER_21_2_0_EXFAT,
|
|
||||||
|
|
||||||
FS_VER_22_0_0,
|
|
||||||
FS_VER_22_0_0_EXFAT,
|
|
||||||
|
|
||||||
FS_VER_MAX,
|
FS_VER_MAX,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
59
emummc/source/FS/offsets/2000.h
vendored
59
emummc/source/FS/offsets/2000.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2000_H__
|
|
||||||
#define __FS_2000_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2000_SDMMC_ACCESSOR_GC 0x1A7DB0
|
|
||||||
#define FS_OFFSET_2000_SDMMC_ACCESSOR_SD 0x1AA130
|
|
||||||
#define FS_OFFSET_2000_SDMMC_ACCESSOR_NAND 0x1A8560
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2000_SDMMC_WRAPPER_READ 0x1A3C20
|
|
||||||
#define FS_OFFSET_2000_SDMMC_WRAPPER_WRITE 0x1A3C80
|
|
||||||
#define FS_OFFSET_2000_RTLD 0x2B594
|
|
||||||
#define FS_OFFSET_2000_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2000_CLKRST_SET_MIN_V_CLK_RATE 0x1C6150
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2000_LOCK_MUTEX 0x19CD80
|
|
||||||
#define FS_OFFSET_2000_UNLOCK_MUTEX 0x19CDD0
|
|
||||||
|
|
||||||
#define FS_OFFSET_2000_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1A3BE0
|
|
||||||
#define FS_OFFSET_2000_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1A3C00
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2000_SD_MUTEX 0xFF5408
|
|
||||||
#define FS_OFFSET_2000_NAND_MUTEX 0xFF0CF0
|
|
||||||
#define FS_OFFSET_2000_ACTIVE_PARTITION 0xFF0D30
|
|
||||||
#define FS_OFFSET_2000_SDMMC_DAS_HANDLE 0xFD2B08
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2000_SD_DAS_INIT 0x289F4
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2000_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0006DB14, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0007CE1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x00084A08, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0009AE48, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2000_H__
|
|
||||||
59
emummc/source/FS/offsets/2000_exfat.h
vendored
59
emummc/source/FS/offsets/2000_exfat.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2000_EXFAT_H__
|
|
||||||
#define __FS_2000_EXFAT_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_ACCESSOR_GC 0x1B36D0
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_ACCESSOR_SD 0x1B5A50
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_ACCESSOR_NAND 0x1B3E80
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_WRAPPER_READ 0x1AF540
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_WRAPPER_WRITE 0x1AF5A0
|
|
||||||
#define FS_OFFSET_2000_EXFAT_RTLD 0x2B594
|
|
||||||
#define FS_OFFSET_2000_EXFAT_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2000_EXFAT_CLKRST_SET_MIN_V_CLK_RATE 0x1D1A70
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2000_EXFAT_LOCK_MUTEX 0x1A86A0
|
|
||||||
#define FS_OFFSET_2000_EXFAT_UNLOCK_MUTEX 0x1A86F0
|
|
||||||
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1AF500
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1AF520
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SD_MUTEX 0x1006408
|
|
||||||
#define FS_OFFSET_2000_EXFAT_NAND_MUTEX 0x1001CF0
|
|
||||||
#define FS_OFFSET_2000_EXFAT_ACTIVE_PARTITION 0x1001D30
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SDMMC_DAS_HANDLE 0xFDFB08
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2000_EXFAT_SD_DAS_INIT 0x289F4
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2000_EXFAT_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0006DB14, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0007CE1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x00084A08, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0009AE48, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2000_EXFAT_H__
|
|
||||||
59
emummc/source/FS/offsets/2010.h
vendored
59
emummc/source/FS/offsets/2010.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2010_H__
|
|
||||||
#define __FS_2010_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2010_SDMMC_ACCESSOR_GC 0x1A7DB0
|
|
||||||
#define FS_OFFSET_2010_SDMMC_ACCESSOR_SD 0x1AA130
|
|
||||||
#define FS_OFFSET_2010_SDMMC_ACCESSOR_NAND 0x1A8560
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2010_SDMMC_WRAPPER_READ 0x1A3C20
|
|
||||||
#define FS_OFFSET_2010_SDMMC_WRAPPER_WRITE 0x1A3C80
|
|
||||||
#define FS_OFFSET_2010_RTLD 0x2B594
|
|
||||||
#define FS_OFFSET_2010_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2010_CLKRST_SET_MIN_V_CLK_RATE 0x1C6150
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2010_LOCK_MUTEX 0x19CD80
|
|
||||||
#define FS_OFFSET_2010_UNLOCK_MUTEX 0x19CDD0
|
|
||||||
|
|
||||||
#define FS_OFFSET_2010_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1A3BE0
|
|
||||||
#define FS_OFFSET_2010_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1A3C00
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2010_SD_MUTEX 0xFF5408
|
|
||||||
#define FS_OFFSET_2010_NAND_MUTEX 0xFF0CF0
|
|
||||||
#define FS_OFFSET_2010_ACTIVE_PARTITION 0xFF0D30
|
|
||||||
#define FS_OFFSET_2010_SDMMC_DAS_HANDLE 0xFD2B08
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2010_SD_DAS_INIT 0x289F4
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2010_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0006DB14, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0007CE1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x00084A08, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0009AE48, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2010_H__
|
|
||||||
59
emummc/source/FS/offsets/2010_exfat.h
vendored
59
emummc/source/FS/offsets/2010_exfat.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2010_EXFAT_H__
|
|
||||||
#define __FS_2010_EXFAT_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_ACCESSOR_GC 0x1B36D0
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_ACCESSOR_SD 0x1B5A50
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_ACCESSOR_NAND 0x1B3E80
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_WRAPPER_READ 0x1AF540
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_WRAPPER_WRITE 0x1AF5A0
|
|
||||||
#define FS_OFFSET_2010_EXFAT_RTLD 0x2B594
|
|
||||||
#define FS_OFFSET_2010_EXFAT_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2010_EXFAT_CLKRST_SET_MIN_V_CLK_RATE 0x1D1A70
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2010_EXFAT_LOCK_MUTEX 0x1A86A0
|
|
||||||
#define FS_OFFSET_2010_EXFAT_UNLOCK_MUTEX 0x1A86F0
|
|
||||||
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1AF500
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1AF520
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SD_MUTEX 0x1006408
|
|
||||||
#define FS_OFFSET_2010_EXFAT_NAND_MUTEX 0x1001CF0
|
|
||||||
#define FS_OFFSET_2010_EXFAT_ACTIVE_PARTITION 0x1001D30
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SDMMC_DAS_HANDLE 0xFDFB08
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2010_EXFAT_SD_DAS_INIT 0x289F4
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2010_EXFAT_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0006DB14, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x0007CE1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x00084A08, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0009AE48, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2010_EXFAT_H__
|
|
||||||
59
emummc/source/FS/offsets/2100.h
vendored
59
emummc/source/FS/offsets/2100.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2100_H__
|
|
||||||
#define __FS_2100_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2100_SDMMC_ACCESSOR_GC 0x1AC970
|
|
||||||
#define FS_OFFSET_2100_SDMMC_ACCESSOR_SD 0x1AE980
|
|
||||||
#define FS_OFFSET_2100_SDMMC_ACCESSOR_NAND 0x1ACFA0
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2100_SDMMC_WRAPPER_READ 0x1A8850
|
|
||||||
#define FS_OFFSET_2100_SDMMC_WRAPPER_WRITE 0x1A88B0
|
|
||||||
#define FS_OFFSET_2100_RTLD 0x2E1C0
|
|
||||||
#define FS_OFFSET_2100_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2100_CLKRST_SET_MIN_V_CLK_RATE 0x1CB9B0
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2100_LOCK_MUTEX 0x1A17D0
|
|
||||||
#define FS_OFFSET_2100_UNLOCK_MUTEX 0x1A1830
|
|
||||||
|
|
||||||
#define FS_OFFSET_2100_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1A8810
|
|
||||||
#define FS_OFFSET_2100_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1A8830
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2100_SD_MUTEX 0xFEE408
|
|
||||||
#define FS_OFFSET_2100_NAND_MUTEX 0xFE9CF0
|
|
||||||
#define FS_OFFSET_2100_ACTIVE_PARTITION 0xFE9D30
|
|
||||||
#define FS_OFFSET_2100_SDMMC_DAS_HANDLE 0xFCBB18
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2100_SD_DAS_INIT 0x2B5C8
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2100_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000718CC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000824F4, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008AF18, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A0B8C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2100_H__
|
|
||||||
59
emummc/source/FS/offsets/2100_exfat.h
vendored
59
emummc/source/FS/offsets/2100_exfat.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2100_EXFAT_EXFAT_H__
|
|
||||||
#define __FS_2100_EXFAT_EXFAT_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_ACCESSOR_GC 0x1B7AD0
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_ACCESSOR_SD 0x1B9AE0
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_ACCESSOR_NAND 0x1B8100
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_WRAPPER_READ 0x1B39B0
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_WRAPPER_WRITE 0x1B3A10
|
|
||||||
#define FS_OFFSET_2100_EXFAT_RTLD 0x2E1C0
|
|
||||||
#define FS_OFFSET_2100_EXFAT_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2100_EXFAT_CLKRST_SET_MIN_V_CLK_RATE 0x1D6B10
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2100_EXFAT_LOCK_MUTEX 0x1AC930
|
|
||||||
#define FS_OFFSET_2100_EXFAT_UNLOCK_MUTEX 0x1AC990
|
|
||||||
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1B3970
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1B3990
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SD_MUTEX 0xFFF408
|
|
||||||
#define FS_OFFSET_2100_EXFAT_NAND_MUTEX 0xFFACF0
|
|
||||||
#define FS_OFFSET_2100_EXFAT_ACTIVE_PARTITION 0xFFAD30
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SDMMC_DAS_HANDLE 0xFD8B18
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2100_EXFAT_SD_DAS_INIT 0x2B5C8
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2100_EXFAT_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000718CC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000824F4, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008AF18, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A0B8C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2100_EXFAT_EXFAT_H__
|
|
||||||
59
emummc/source/FS/offsets/2120.h
vendored
59
emummc/source/FS/offsets/2120.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2120_H__
|
|
||||||
#define __FS_2120_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2120_SDMMC_ACCESSOR_GC 0x1AC970
|
|
||||||
#define FS_OFFSET_2120_SDMMC_ACCESSOR_SD 0x1AE980
|
|
||||||
#define FS_OFFSET_2120_SDMMC_ACCESSOR_NAND 0x1ACFA0
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2120_SDMMC_WRAPPER_READ 0x1A8850
|
|
||||||
#define FS_OFFSET_2120_SDMMC_WRAPPER_WRITE 0x1A88B0
|
|
||||||
#define FS_OFFSET_2120_RTLD 0x2E1C0
|
|
||||||
#define FS_OFFSET_2120_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2120_CLKRST_SET_MIN_V_CLK_RATE 0x1CB9B0
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2120_LOCK_MUTEX 0x1A17D0
|
|
||||||
#define FS_OFFSET_2120_UNLOCK_MUTEX 0x1A1830
|
|
||||||
|
|
||||||
#define FS_OFFSET_2120_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1A8810
|
|
||||||
#define FS_OFFSET_2120_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1A8830
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2120_SD_MUTEX 0xFEE408
|
|
||||||
#define FS_OFFSET_2120_NAND_MUTEX 0xFE9CF0
|
|
||||||
#define FS_OFFSET_2120_ACTIVE_PARTITION 0xFE9D30
|
|
||||||
#define FS_OFFSET_2120_SDMMC_DAS_HANDLE 0xFCBB18
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2120_SD_DAS_INIT 0x2B5C8
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2120_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000718CC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000824F4, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008AF18, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A0B8C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2120_H__
|
|
||||||
59
emummc/source/FS/offsets/2120_exfat.h
vendored
59
emummc/source/FS/offsets/2120_exfat.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2120_EXFAT_EXFAT_H__
|
|
||||||
#define __FS_2120_EXFAT_EXFAT_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_ACCESSOR_GC 0x1B7AD0
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_ACCESSOR_SD 0x1B9AE0
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_ACCESSOR_NAND 0x1B8100
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_WRAPPER_READ 0x1B39B0
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_WRAPPER_WRITE 0x1B3A10
|
|
||||||
#define FS_OFFSET_2120_EXFAT_RTLD 0x2E1C0
|
|
||||||
#define FS_OFFSET_2120_EXFAT_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2120_EXFAT_CLKRST_SET_MIN_V_CLK_RATE 0x1D6B10
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2120_EXFAT_LOCK_MUTEX 0x1AC930
|
|
||||||
#define FS_OFFSET_2120_EXFAT_UNLOCK_MUTEX 0x1AC990
|
|
||||||
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1B3970
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1B3990
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SD_MUTEX 0xFFF408
|
|
||||||
#define FS_OFFSET_2120_EXFAT_NAND_MUTEX 0xFFACF0
|
|
||||||
#define FS_OFFSET_2120_EXFAT_ACTIVE_PARTITION 0xFFAD30
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SDMMC_DAS_HANDLE 0xFD8B18
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2120_EXFAT_SD_DAS_INIT 0x2B5C8
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2120_EXFAT_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000718CC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000824F4, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008AF18, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A0B8C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2120_EXFAT_EXFAT_H__
|
|
||||||
59
emummc/source/FS/offsets/2200.h
vendored
59
emummc/source/FS/offsets/2200.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2200_H__
|
|
||||||
#define __FS_2200_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2200_SDMMC_ACCESSOR_GC 0x1B01C0
|
|
||||||
#define FS_OFFSET_2200_SDMMC_ACCESSOR_SD 0x1B21C0
|
|
||||||
#define FS_OFFSET_2200_SDMMC_ACCESSOR_NAND 0x1B07F0
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2200_SDMMC_WRAPPER_READ 0x1AC0F0
|
|
||||||
#define FS_OFFSET_2200_SDMMC_WRAPPER_WRITE 0x1AC150
|
|
||||||
#define FS_OFFSET_2200_RTLD 0x2DED0
|
|
||||||
#define FS_OFFSET_2200_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2200_CLKRST_SET_MIN_V_CLK_RATE 0x1CF260
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2200_LOCK_MUTEX 0x1A4EF0
|
|
||||||
#define FS_OFFSET_2200_UNLOCK_MUTEX 0x1A4F40
|
|
||||||
|
|
||||||
#define FS_OFFSET_2200_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1AC0B0
|
|
||||||
#define FS_OFFSET_2200_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1AC0D0
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2200_SD_MUTEX 0xFF1408
|
|
||||||
#define FS_OFFSET_2200_NAND_MUTEX 0xFECD00
|
|
||||||
#define FS_OFFSET_2200_ACTIVE_PARTITION 0xFECD40
|
|
||||||
#define FS_OFFSET_2200_SDMMC_DAS_HANDLE 0xFCEB98
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2200_SD_DAS_INIT 0x2B438
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2200_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000715EC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x00082E64, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008B888, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A1B1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2200_H__
|
|
||||||
59
emummc/source/FS/offsets/2200_exfat.h
vendored
59
emummc/source/FS/offsets/2200_exfat.h
vendored
@@ -1,59 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 m4xw <m4x@m4xw.net>
|
|
||||||
* Copyright (c) 2019 Atmosphere-NX
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms and conditions of the GNU General Public License,
|
|
||||||
* version 2, as published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
||||||
* more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
#ifndef __FS_2200_EXFAT_H__
|
|
||||||
#define __FS_2200_EXFAT_H__
|
|
||||||
|
|
||||||
// Accessor vtable getters
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_ACCESSOR_GC 0x1BB3B0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_ACCESSOR_SD 0x1BD3B0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_ACCESSOR_NAND 0x1BB9E0
|
|
||||||
|
|
||||||
// Hooks
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_WRAPPER_READ 0x1B72E0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_WRAPPER_WRITE 0x1B7340
|
|
||||||
#define FS_OFFSET_2200_EXFAT_RTLD 0x2DED0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_RTLD_DESTINATION ((uintptr_t)(INT64_C(-0x4C)))
|
|
||||||
|
|
||||||
#define FS_OFFSET_2200_EXFAT_CLKRST_SET_MIN_V_CLK_RATE 0x1DA450
|
|
||||||
|
|
||||||
// Misc funcs
|
|
||||||
#define FS_OFFSET_2200_EXFAT_LOCK_MUTEX 0x1B00E0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_UNLOCK_MUTEX 0x1B0130
|
|
||||||
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_WRAPPER_CONTROLLER_OPEN 0x1B72A0
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_WRAPPER_CONTROLLER_CLOSE 0x1B72C0
|
|
||||||
|
|
||||||
// Misc Data
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SD_MUTEX 0x1002408
|
|
||||||
#define FS_OFFSET_2200_EXFAT_NAND_MUTEX 0xFFDD00
|
|
||||||
#define FS_OFFSET_2200_EXFAT_ACTIVE_PARTITION 0xFFDD40
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SDMMC_DAS_HANDLE 0xFDBB98
|
|
||||||
|
|
||||||
// NOPs
|
|
||||||
#define FS_OFFSET_2200_EXFAT_SD_DAS_INIT 0x2B438
|
|
||||||
|
|
||||||
// Nintendo Paths
|
|
||||||
#define FS_OFFSET_2200_EXFAT_NINTENDO_PATHS \
|
|
||||||
{ \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x000715EC, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 3, .adrp_offset = 0x00082E64, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x0008B888, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 4, .adrp_offset = 0x000A1B1C, .add_rel_offset = 0x00000004}, \
|
|
||||||
{.opcode_reg = 0, .adrp_offset = 0, .add_rel_offset = 0}, \
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif // __FS_2200_EXFAT_H__
|
|
||||||
@@ -70,6 +70,6 @@ clean:
|
|||||||
@$(MAKE) --no-print-directory -C $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere -f $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere/libexosphere.mk clean
|
@$(MAKE) --no-print-directory -C $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere -f $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere/libexosphere.mk clean
|
||||||
@$(MAKE) --no-print-directory -C $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere -f $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere/libexosphere.mk clean ATMOSPHERE_CPU="$(strip $(ATMOSPHERE_BOOT_CPU))"
|
@$(MAKE) --no-print-directory -C $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere -f $(ATMOSPHERE_LIBRARIES_DIR)/libexosphere/libexosphere.mk clean ATMOSPHERE_CPU="$(strip $(ATMOSPHERE_BOOT_CPU))"
|
||||||
@rm -fr $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
@rm -fr $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
||||||
@for i in $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR); do [ -d $$i ] && rmdir $$i 2>/dev/null || true; done
|
@for i in $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR); do [ -d $$i ] && rmdir --ignore-fail-on-non-empty $$i || true; done
|
||||||
|
|
||||||
.PHONY: all clean check_lib check_boot_lib check_loader_stub check_program check_mariko_fatal check_warmboot
|
.PHONY: all clean check_lib check_boot_lib check_loader_stub check_program check_mariko_fatal check_warmboot
|
||||||
@@ -119,7 +119,7 @@ DEPENDS := $(OFILES:.o=.d)
|
|||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
$(OUTPUT).lz4 : $(OUTPUT).bin
|
$(OUTPUT).lz4 : $(OUTPUT).bin
|
||||||
$(SILENTCMD)$(PYTHON) $(CURRENT_DIRECTORY)/split_program.py $(OUTPUT).bin $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
@python $(CURRENT_DIRECTORY)/split_program.py $(OUTPUT).bin $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
||||||
@echo built ... $(notdir $@)
|
@echo built ... $(notdir $@)
|
||||||
|
|
||||||
$(OUTPUT).bin : $(OUTPUT).elf
|
$(OUTPUT).bin : $(OUTPUT).elf
|
||||||
|
|||||||
@@ -85,10 +85,10 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
/* We can get away with only including latest because exosphere supports newer-than-expected master key in engine. */
|
/* We can get away with only including latest because exosphere supports newer-than-expected master key in engine. */
|
||||||
/* TODO: Update on next change of keys. */
|
/* TODO: Update on next change of keys. */
|
||||||
/* Mariko Development Master Kek Source. */
|
/* Mariko Development Master Kek Source. */
|
||||||
.byte 0x2E, 0x27, 0x44, 0xEA, 0x32, 0xF8, 0x2C, 0xF0, 0x6F, 0xCA, 0xCD, 0x77, 0xAE, 0xAE, 0x1A, 0x1B
|
.byte 0x65, 0x7B, 0x11, 0x46, 0x0E, 0xC2, 0x22, 0x5D, 0xB9, 0xF1, 0xF5, 0x00, 0xF9, 0x3E, 0x1F, 0x70
|
||||||
|
|
||||||
/* Mariko Production Master Kek Source. */
|
/* Mariko Production Master Kek Source. */
|
||||||
.byte 0x82, 0xE2, 0x0A, 0x59, 0x67, 0xDF, 0xBF, 0x51, 0x47, 0x62, 0x11, 0xF2, 0x41, 0xD3, 0xEE, 0x13
|
.byte 0x31, 0xBE, 0x25, 0xFB, 0xDB, 0xB4, 0xEE, 0x49, 0x5C, 0x77, 0x05, 0xC2, 0x36, 0x9F, 0x34, 0x80
|
||||||
|
|
||||||
/* Development Master Key Vectors. */
|
/* Development Master Key Vectors. */
|
||||||
.byte 0x46, 0x22, 0xB4, 0x51, 0x9A, 0x7E, 0xA7, 0x7F, 0x62, 0xA1, 0x1F, 0x8F, 0xC5, 0x3A, 0xDB, 0xFE /* Zeroes encrypted with Master Key 00. */
|
.byte 0x46, 0x22, 0xB4, 0x51, 0x9A, 0x7E, 0xA7, 0x7F, 0x62, 0xA1, 0x1F, 0x8F, 0xC5, 0x3A, 0xDB, 0xFE /* Zeroes encrypted with Master Key 00. */
|
||||||
@@ -110,9 +110,6 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
.byte 0x39, 0x1E, 0x7E, 0xF8, 0x7E, 0x73, 0xEA, 0x6F, 0xAF, 0x00, 0x3A, 0xB4, 0xAA, 0xB8, 0xB7, 0x59 /* Master key 0F encrypted with Master key 10. */
|
.byte 0x39, 0x1E, 0x7E, 0xF8, 0x7E, 0x73, 0xEA, 0x6F, 0xAF, 0x00, 0x3A, 0xB4, 0xAA, 0xB8, 0xB7, 0x59 /* Master key 0F encrypted with Master key 10. */
|
||||||
.byte 0x0C, 0x75, 0x39, 0x15, 0x53, 0xEA, 0x81, 0x11, 0xA3, 0xE0, 0xDC, 0x3D, 0x0E, 0x76, 0xC6, 0xB8 /* Master key 10 encrypted with Master key 11. */
|
.byte 0x0C, 0x75, 0x39, 0x15, 0x53, 0xEA, 0x81, 0x11, 0xA3, 0xE0, 0xDC, 0x3D, 0x0E, 0x76, 0xC6, 0xB8 /* Master key 10 encrypted with Master key 11. */
|
||||||
.byte 0x90, 0x64, 0xF9, 0x08, 0x29, 0x88, 0xD4, 0xDC, 0x73, 0xA4, 0xA1, 0x13, 0x9E, 0x59, 0x85, 0xA0 /* Master key 11 encrypted with Master key 12. */
|
.byte 0x90, 0x64, 0xF9, 0x08, 0x29, 0x88, 0xD4, 0xDC, 0x73, 0xA4, 0xA1, 0x13, 0x9E, 0x59, 0x85, 0xA0 /* Master key 11 encrypted with Master key 12. */
|
||||||
.byte 0x94, 0x46, 0x3B, 0xFA, 0x7D, 0xB9, 0xE2, 0x94, 0xC2, 0x9D, 0xB9, 0xA4, 0xB2, 0x56, 0xCA, 0xFE /* Master key 12 encrypted with Master key 13. */
|
|
||||||
.byte 0x74, 0xB2, 0x5F, 0xA0, 0x4B, 0x74, 0x6D, 0x47, 0x5B, 0xA9, 0xF5, 0x26, 0x46, 0xD7, 0x4B, 0x6E /* Master key 13 encrypted with Master key 14. */
|
|
||||||
.byte 0x97, 0xB3, 0x61, 0x88, 0x5C, 0x0D, 0xA1, 0x38, 0x73, 0xA4, 0x2F, 0x1A, 0x46, 0xA1, 0x09, 0xBF /* Master key 14 encrypted with Master key 15. */
|
|
||||||
|
|
||||||
/* Production Master Key Vectors. */
|
/* Production Master Key Vectors. */
|
||||||
.byte 0x0C, 0xF0, 0x59, 0xAC, 0x85, 0xF6, 0x26, 0x65, 0xE1, 0xE9, 0x19, 0x55, 0xE6, 0xF2, 0x67, 0x3D /* Zeroes encrypted with Master Key 00. */
|
.byte 0x0C, 0xF0, 0x59, 0xAC, 0x85, 0xF6, 0x26, 0x65, 0xE1, 0xE9, 0x19, 0x55, 0xE6, 0xF2, 0x67, 0x3D /* Zeroes encrypted with Master Key 00. */
|
||||||
@@ -134,9 +131,6 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
.byte 0x25, 0x12, 0x8B, 0xCB, 0xB5, 0x46, 0xA1, 0xF8, 0xE0, 0x52, 0x15, 0xB7, 0x0B, 0x57, 0x00, 0xBD /* Master key 0F encrypted with Master key 10. */
|
.byte 0x25, 0x12, 0x8B, 0xCB, 0xB5, 0x46, 0xA1, 0xF8, 0xE0, 0x52, 0x15, 0xB7, 0x0B, 0x57, 0x00, 0xBD /* Master key 0F encrypted with Master key 10. */
|
||||||
.byte 0x58, 0x15, 0xD2, 0xF6, 0x8A, 0xE8, 0x19, 0xAB, 0xFB, 0x2D, 0x52, 0x9D, 0xE7, 0x55, 0xF3, 0x93 /* Master key 10 encrypted with Master key 11. */
|
.byte 0x58, 0x15, 0xD2, 0xF6, 0x8A, 0xE8, 0x19, 0xAB, 0xFB, 0x2D, 0x52, 0x9D, 0xE7, 0x55, 0xF3, 0x93 /* Master key 10 encrypted with Master key 11. */
|
||||||
.byte 0x4A, 0x01, 0x3B, 0xC7, 0x44, 0x6E, 0x45, 0xBD, 0xE6, 0x5E, 0x2B, 0xEC, 0x07, 0x37, 0x52, 0x86 /* Master key 11 encrypted with Master key 12. */
|
.byte 0x4A, 0x01, 0x3B, 0xC7, 0x44, 0x6E, 0x45, 0xBD, 0xE6, 0x5E, 0x2B, 0xEC, 0x07, 0x37, 0x52, 0x86 /* Master key 11 encrypted with Master key 12. */
|
||||||
.byte 0x97, 0xE4, 0x11, 0xAB, 0x22, 0x72, 0x1A, 0x1F, 0x70, 0x5C, 0x00, 0xB3, 0x96, 0x30, 0x05, 0x28 /* Master key 12 encrypted with Master key 13. */
|
|
||||||
.byte 0xF7, 0x92, 0xC0, 0xEC, 0xF3, 0xA4, 0x8C, 0xB7, 0x0D, 0xB3, 0xF3, 0xAB, 0x10, 0x9B, 0x18, 0xBA /* Master key 13 encrypted with Master key 14. */
|
|
||||||
.byte 0x14, 0xCB, 0x60, 0x29, 0x3D, 0xE0, 0xFB, 0xF2, 0x5B, 0x60, 0xB6, 0xC5, 0x2E, 0x77, 0x8F, 0x98 /* Master key 14 encrypted with Master key 15. */
|
|
||||||
|
|
||||||
/* Device Master Key Source Sources. */
|
/* Device Master Key Source Sources. */
|
||||||
.byte 0x8B, 0x4E, 0x1C, 0x22, 0x42, 0x07, 0xC8, 0x73, 0x56, 0x94, 0x08, 0x8B, 0xCC, 0x47, 0x0F, 0x5D /* 4.0.0 Device Master Key Source Source. */
|
.byte 0x8B, 0x4E, 0x1C, 0x22, 0x42, 0x07, 0xC8, 0x73, 0x56, 0x94, 0x08, 0x8B, 0xCC, 0x47, 0x0F, 0x5D /* 4.0.0 Device Master Key Source Source. */
|
||||||
@@ -155,9 +149,6 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
.byte 0xDA, 0xB9, 0xD6, 0x77, 0x52, 0x2D, 0x1F, 0x78, 0x73, 0xC9, 0x98, 0x5B, 0x06, 0xFE, 0xA0, 0x52 /* 17.0.0 Device Master Key Source Source. */
|
.byte 0xDA, 0xB9, 0xD6, 0x77, 0x52, 0x2D, 0x1F, 0x78, 0x73, 0xC9, 0x98, 0x5B, 0x06, 0xFE, 0xA0, 0x52 /* 17.0.0 Device Master Key Source Source. */
|
||||||
.byte 0x14, 0xF5, 0xA5, 0xD0, 0x73, 0x6D, 0x44, 0x80, 0x5F, 0x31, 0x5A, 0x8F, 0x1E, 0xD4, 0x0D, 0x63 /* 18.0.0 Device Master Key Source Source. */
|
.byte 0x14, 0xF5, 0xA5, 0xD0, 0x73, 0x6D, 0x44, 0x80, 0x5F, 0x31, 0x5A, 0x8F, 0x1E, 0xD4, 0x0D, 0x63 /* 18.0.0 Device Master Key Source Source. */
|
||||||
.byte 0x07, 0x38, 0x9A, 0xEC, 0x9C, 0xBD, 0x50, 0x4A, 0x4C, 0x1F, 0x04, 0xDA, 0x40, 0x68, 0x29, 0xE3 /* 19.0.0 Device Master Key Source Source. */
|
.byte 0x07, 0x38, 0x9A, 0xEC, 0x9C, 0xBD, 0x50, 0x4A, 0x4C, 0x1F, 0x04, 0xDA, 0x40, 0x68, 0x29, 0xE3 /* 19.0.0 Device Master Key Source Source. */
|
||||||
.byte 0xA3, 0x6B, 0x0A, 0xB5, 0x6F, 0x57, 0x4C, 0x5E, 0x00, 0xFD, 0x56, 0x21, 0xF5, 0x06, 0x6B, 0xD1 /* 20.0.0 Device Master Key Source Source. */
|
|
||||||
.byte 0xF9, 0x62, 0x05, 0x99, 0xE0, 0xB9, 0xA6, 0x9B, 0x9D, 0xAA, 0xB4, 0x12, 0x0B, 0x0F, 0xF5, 0x8F /* 21.0.0 Device Master Key Source Source. */
|
|
||||||
.byte 0xF8, 0xF4, 0x22, 0xA4, 0x34, 0xAE, 0x0E, 0x0C, 0x4D, 0x5C, 0x5B, 0xA1, 0x1B, 0x46, 0x1C, 0x78 /* 22.0.0 Device Master Key Source Source. */
|
|
||||||
|
|
||||||
/* Development Device Master Kek Sources. */
|
/* Development Device Master Kek Sources. */
|
||||||
.byte 0xD6, 0xBD, 0x9F, 0xC6, 0x18, 0x09, 0xE1, 0x96, 0x20, 0x39, 0x60, 0xD2, 0x89, 0x83, 0x31, 0x34 /* 4.0.0 Device Master Kek Source. */
|
.byte 0xD6, 0xBD, 0x9F, 0xC6, 0x18, 0x09, 0xE1, 0x96, 0x20, 0x39, 0x60, 0xD2, 0x89, 0x83, 0x31, 0x34 /* 4.0.0 Device Master Kek Source. */
|
||||||
@@ -176,9 +167,6 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
.byte 0x4E, 0xCE, 0x7B, 0x2A, 0xEA, 0x2E, 0x3D, 0x16, 0xD5, 0x2A, 0xDE, 0xF6, 0xF8, 0x6A, 0x7D, 0x43 /* 17.0.0 Device Master Kek Source. */
|
.byte 0x4E, 0xCE, 0x7B, 0x2A, 0xEA, 0x2E, 0x3D, 0x16, 0xD5, 0x2A, 0xDE, 0xF6, 0xF8, 0x6A, 0x7D, 0x43 /* 17.0.0 Device Master Kek Source. */
|
||||||
.byte 0x3B, 0x00, 0x89, 0xD7, 0xA9, 0x9E, 0xB7, 0x70, 0x86, 0x00, 0xC3, 0x49, 0x52, 0x8C, 0xA4, 0xAF /* 18.0.0 Device Master Kek Source. */
|
.byte 0x3B, 0x00, 0x89, 0xD7, 0xA9, 0x9E, 0xB7, 0x70, 0x86, 0x00, 0xC3, 0x49, 0x52, 0x8C, 0xA4, 0xAF /* 18.0.0 Device Master Kek Source. */
|
||||||
.byte 0xAE, 0x78, 0x36, 0xB6, 0x91, 0xEB, 0xAF, 0x9C, 0x18, 0xF1, 0xC0, 0xD5, 0x8A, 0x0C, 0x7C, 0xA1 /* 19.0.0 Device Master Kek Source. */
|
.byte 0xAE, 0x78, 0x36, 0xB6, 0x91, 0xEB, 0xAF, 0x9C, 0x18, 0xF1, 0xC0, 0xD5, 0x8A, 0x0C, 0x7C, 0xA1 /* 19.0.0 Device Master Kek Source. */
|
||||||
.byte 0x09, 0x12, 0x4F, 0x26, 0x90, 0xB9, 0xA6, 0xF5, 0xA5, 0x18, 0x74, 0xB6, 0x8D, 0x80, 0x59, 0x3D /* 20.0.0 Device Master Kek Source. */
|
|
||||||
.byte 0x7A, 0x4C, 0x38, 0xB7, 0x03, 0x6B, 0x1E, 0x81, 0x20, 0x53, 0x14, 0x99, 0xA4, 0x21, 0x92, 0x9F /* 21.0.0 Device Master Kek Source. */
|
|
||||||
.byte 0xF3, 0xBC, 0xB5, 0xB5, 0x5F, 0x01, 0x50, 0x2B, 0x69, 0x69, 0x3A, 0x6B, 0xF9, 0x2C, 0x11, 0x9F /* 22.0.0 Device Master Kek Source. */
|
|
||||||
|
|
||||||
/* Production Device Master Kek Sources. */
|
/* Production Device Master Kek Sources. */
|
||||||
.byte 0x88, 0x62, 0x34, 0x6E, 0xFA, 0xF7, 0xD8, 0x3F, 0xE1, 0x30, 0x39, 0x50, 0xF0, 0xB7, 0x5D, 0x5D /* 4.0.0 Device Master Kek Source. */
|
.byte 0x88, 0x62, 0x34, 0x6E, 0xFA, 0xF7, 0xD8, 0x3F, 0xE1, 0x30, 0x39, 0x50, 0xF0, 0xB7, 0x5D, 0x5D /* 4.0.0 Device Master Kek Source. */
|
||||||
@@ -197,6 +185,3 @@ _ZN3ams6secmon4boot15VolatileKeyDataE:
|
|||||||
.byte 0x21, 0xD6, 0x35, 0xF1, 0x0F, 0x7A, 0xF0, 0x5D, 0xDF, 0x79, 0x1C, 0x7A, 0xE4, 0x32, 0x82, 0x9E /* 17.0.0 Device Master Kek Source. */
|
.byte 0x21, 0xD6, 0x35, 0xF1, 0x0F, 0x7A, 0xF0, 0x5D, 0xDF, 0x79, 0x1C, 0x7A, 0xE4, 0x32, 0x82, 0x9E /* 17.0.0 Device Master Kek Source. */
|
||||||
.byte 0xE7, 0x85, 0x8C, 0xA2, 0xF4, 0x49, 0xCB, 0x07, 0xD1, 0x8E, 0x48, 0x1B, 0xE8, 0x1E, 0x28, 0x3B /* 18.0.0 Device Master Kek Source. */
|
.byte 0xE7, 0x85, 0x8C, 0xA2, 0xF4, 0x49, 0xCB, 0x07, 0xD1, 0x8E, 0x48, 0x1B, 0xE8, 0x1E, 0x28, 0x3B /* 18.0.0 Device Master Kek Source. */
|
||||||
.byte 0x9B, 0xA5, 0xFD, 0x74, 0x7F, 0xCD, 0x23, 0xD1, 0xD9, 0xBD, 0x6C, 0x51, 0x72, 0x5F, 0x3D, 0x1F /* 19.0.0 Device Master Kek Source. */
|
.byte 0x9B, 0xA5, 0xFD, 0x74, 0x7F, 0xCD, 0x23, 0xD1, 0xD9, 0xBD, 0x6C, 0x51, 0x72, 0x5F, 0x3D, 0x1F /* 19.0.0 Device Master Kek Source. */
|
||||||
.byte 0xDA, 0xFB, 0x61, 0x39, 0x48, 0x2D, 0xC2, 0x7E, 0x0D, 0x8E, 0x8F, 0x98, 0x57, 0x20, 0xB8, 0x15 /* 20.0.0 Device Master Kek Source. */
|
|
||||||
.byte 0x92, 0xBF, 0x37, 0x80, 0x0E, 0x79, 0x56, 0x8C, 0x57, 0x75, 0x72, 0x0A, 0x48, 0xD8, 0x15, 0x39 /* 21.0.0 Device Master Kek Source. */
|
|
||||||
.byte 0xC4, 0x6F, 0x0E, 0x72, 0x43, 0xCE, 0x87, 0xFC, 0x38, 0x95, 0x9B, 0xC9, 0x31, 0x44, 0x97, 0x63 /* 22.0.0 Device Master Kek Source. */
|
|
||||||
@@ -94,7 +94,7 @@ namespace ams::secmon::boot {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Check that the key generation is one that we can use. */
|
/* Check that the key generation is one that we can use. */
|
||||||
static_assert(pkg1::KeyGeneration_Count == 22);
|
static_assert(pkg1::KeyGeneration_Count == 19);
|
||||||
if (key_generation >= pkg1::KeyGeneration_Count) {
|
if (key_generation >= pkg1::KeyGeneration_Count) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -132,13 +132,10 @@ namespace ams::secmon::smc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
u32 GetMemoryMode() {
|
u32 GetMemoryMode() {
|
||||||
/* Unless development function or forced boot config memory size is enabled, we're 4 GB. */
|
/* Unless development function is enabled, we're 4 GB. */
|
||||||
u32 memory_mode = pkg1::MemoryMode_4GB;
|
u32 memory_mode = pkg1::MemoryMode_4GB;
|
||||||
|
|
||||||
const auto &bcd = GetBootConfig().data;
|
if (const auto &bcd = GetBootConfig().data; bcd.IsDevelopmentFunctionEnabled()) {
|
||||||
const auto &sc = GetSecmonConfiguration(); /* Exosphere extensions */
|
|
||||||
|
|
||||||
if (bcd.IsDevelopmentFunctionEnabled() || sc.IsBootConfigMemoryModeEnabled()) {
|
|
||||||
memory_mode = GetMemoryMode(bcd.GetMemoryMode());
|
memory_mode = GetMemoryMode(bcd.GetMemoryMode());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -149,20 +146,18 @@ namespace ams::secmon::smc {
|
|||||||
pkg1::MemorySize memory_size = pkg1::MemorySize_4GB;
|
pkg1::MemorySize memory_size = pkg1::MemorySize_4GB;
|
||||||
util::BitPack32 value = {};
|
util::BitPack32 value = {};
|
||||||
|
|
||||||
const auto &bcd = GetBootConfig().data;
|
if (const auto &bcd = GetBootConfig().data; bcd.IsDevelopmentFunctionEnabled()) {
|
||||||
const auto &sc = GetSecmonConfiguration(); /* Exosphere extensions */
|
memory_size = GetMemorySize(GetMemoryMode(bcd.GetMemoryMode()));
|
||||||
|
|
||||||
if (bcd.IsDevelopmentFunctionEnabled()) {
|
|
||||||
value.Set<KernelConfiguration::Flags1>(bcd.GetKernelFlags1());
|
value.Set<KernelConfiguration::Flags1>(bcd.GetKernelFlags1());
|
||||||
value.Set<KernelConfiguration::Flags0>(bcd.GetKernelFlags0());
|
value.Set<KernelConfiguration::Flags0>(bcd.GetKernelFlags0());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bcd.IsDevelopmentFunctionEnabled() || sc.IsBootConfigMemoryModeEnabled()) {
|
|
||||||
memory_size = GetMemorySize(GetMemoryMode(bcd.GetMemoryMode()));
|
|
||||||
}
|
|
||||||
|
|
||||||
value.Set<KernelConfiguration::PhysicalMemorySize>(memory_size);
|
value.Set<KernelConfiguration::PhysicalMemorySize>(memory_size);
|
||||||
|
|
||||||
|
/* Exosphere extensions. */
|
||||||
|
const auto &sc = GetSecmonConfiguration();
|
||||||
|
|
||||||
if (!sc.DisableUserModeExceptionHandlers()) {
|
if (!sc.DisableUserModeExceptionHandlers()) {
|
||||||
value.Set<KernelConfiguration::EnableUserExceptionHandlers>(true);
|
value.Set<KernelConfiguration::EnableUserExceptionHandlers>(true);
|
||||||
}
|
}
|
||||||
@@ -174,27 +169,6 @@ namespace ams::secmon::smc {
|
|||||||
return value.value;
|
return value.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
fuse::DramId GetDramIdAdjusted() {
|
|
||||||
const auto dram_id = fuse::GetDramId();
|
|
||||||
AMS_ABORT_UNLESS(dram_id < fuse::DramId_Count);
|
|
||||||
|
|
||||||
const auto fuse_mem_size = DramIdToMemorySize[dram_id];
|
|
||||||
const auto phys_mem_size = GetPhysicalMemorySize();
|
|
||||||
|
|
||||||
AMS_ABORT_UNLESS(fuse_mem_size <= phys_mem_size);
|
|
||||||
|
|
||||||
if (fuse_mem_size == phys_mem_size) {
|
|
||||||
return dram_id;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Adjust Dram ID to match density/ranks. */
|
|
||||||
if (GetSocType() == fuse::SocType_Erista) {
|
|
||||||
return fuse::DramId_IcosaSamsung6GB;
|
|
||||||
} else { /* fuse::SocType_Mariko */
|
|
||||||
return fuse::DramId_IowaSamsung1y8GBX;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
constinit u64 g_payload_address = 0;
|
constinit u64 g_payload_address = 0;
|
||||||
constinit bool g_set_true_target_firmware = false;
|
constinit bool g_set_true_target_firmware = false;
|
||||||
|
|
||||||
@@ -204,7 +178,7 @@ namespace ams::secmon::smc {
|
|||||||
args.r[1] = GetBootConfig().signed_data.IsProgramVerificationDisabled();
|
args.r[1] = GetBootConfig().signed_data.IsProgramVerificationDisabled();
|
||||||
break;
|
break;
|
||||||
case ConfigItem::DramId:
|
case ConfigItem::DramId:
|
||||||
args.r[1] = GetDramIdAdjusted(); /* Nintendo: fuse::GetDramId() */
|
args.r[1] = fuse::GetDramId();
|
||||||
break;
|
break;
|
||||||
case ConfigItem::SecurityEngineInterruptNumber:
|
case ConfigItem::SecurityEngineInterruptNumber:
|
||||||
args.r[1] = SecurityEngineUserInterruptId;
|
args.r[1] = SecurityEngineUserInterruptId;
|
||||||
@@ -497,18 +471,9 @@ namespace ams::secmon::smc {
|
|||||||
|
|
||||||
/* For exosphere's usage. */
|
/* For exosphere's usage. */
|
||||||
pkg1::MemorySize GetPhysicalMemorySize() {
|
pkg1::MemorySize GetPhysicalMemorySize() {
|
||||||
const uintptr_t MC = secmon::MemoryRegionVirtualDeviceMemoryController.GetAddress();
|
const auto dram_id = fuse::GetDramId();
|
||||||
const u32 mem_size = reg::Read(MC + MC_EMEM_CFG) & 0x3FFF;
|
AMS_ABORT_UNLESS(dram_id < fuse::DramId_Count);
|
||||||
|
return DramIdToMemorySize[dram_id];
|
||||||
switch (mem_size >> 10) {
|
|
||||||
case 4:
|
|
||||||
default:
|
|
||||||
return pkg1::MemorySize_4GB;
|
|
||||||
case 6:
|
|
||||||
return pkg1::MemorySize_6GB;
|
|
||||||
case 8:
|
|
||||||
return pkg1::MemorySize_8GB;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -101,10 +101,10 @@ namespace ams::secmon::smc {
|
|||||||
#include "secmon_define_mc01_access_table.inc"
|
#include "secmon_define_mc01_access_table.inc"
|
||||||
|
|
||||||
constexpr const AccessTableEntry AccessTables[] = {
|
constexpr const AccessTableEntry AccessTables[] = {
|
||||||
{ PmcAccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDevicePmc.GetAddress(), PmcAccessTable::Address, PmcAccessTable::Size, },
|
{ PmcAccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDevicePmc.GetAddress(), PmcAccessTable::Address, PmcAccessTable::Size, },
|
||||||
{ McAccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDeviceMemoryController.GetAddress(), McAccessTable::Address, McAccessTable::Size, },
|
{ McAccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDeviceMemoryController.GetAddress(), McAccessTable::Address, McAccessTable::Size, },
|
||||||
{ Mc01AccessTable::ReducedAccessTable.data(), Mc01AccessTable::Address + MemoryRegionVirtualDeviceMemoryController0.GetAddress(), Mc01AccessTable::Address + MemoryRegionPhysicalDeviceMemoryController0.GetAddress(), Mc01AccessTable::Size, },
|
{ Mc01AccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDeviceMemoryController0.GetAddress(), Mc01AccessTable::Address + MemoryRegionPhysicalDeviceMemoryController0.GetAddress(), Mc01AccessTable::Size, },
|
||||||
{ Mc01AccessTable::ReducedAccessTable.data(), Mc01AccessTable::Address + MemoryRegionVirtualDeviceMemoryController1.GetAddress(), Mc01AccessTable::Address + MemoryRegionPhysicalDeviceMemoryController1.GetAddress(), Mc01AccessTable::Size, },
|
{ Mc01AccessTable::ReducedAccessTable.data(), MemoryRegionVirtualDeviceMemoryController1.GetAddress(), Mc01AccessTable::Address + MemoryRegionPhysicalDeviceMemoryController1.GetAddress(), Mc01AccessTable::Size, },
|
||||||
};
|
};
|
||||||
|
|
||||||
constexpr bool IsAccessAllowed(const AccessTableEntry &entry, uintptr_t address) {
|
constexpr bool IsAccessAllowed(const AccessTableEntry &entry, uintptr_t address) {
|
||||||
|
|||||||
@@ -43,6 +43,6 @@ clean:
|
|||||||
@$(MAKE) --no-print-directory -C $(CURRENT_DIRECTORY)/loader_stub -f $(CURRENT_DIRECTORY)/loader_stub/loader_stub.mk clean
|
@$(MAKE) --no-print-directory -C $(CURRENT_DIRECTORY)/loader_stub -f $(CURRENT_DIRECTORY)/loader_stub/loader_stub.mk clean
|
||||||
@$(MAKE) --no-print-directory -C $(CURRENT_DIRECTORY)/program -f $(CURRENT_DIRECTORY)/program/program.mk clean
|
@$(MAKE) --no-print-directory -C $(CURRENT_DIRECTORY)/program -f $(CURRENT_DIRECTORY)/program/program.mk clean
|
||||||
@rm -fr $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
@rm -fr $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR)
|
||||||
@for i in $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR); do [ -d $$i ] && rmdir $$i 2>/dev/null || true; done
|
@for i in $(CURRENT_DIRECTORY)/$(ATMOSPHERE_OUT_DIR); do [ -d $$i ] && rmdir --ignore-fail-on-non-empty $$i || true; done
|
||||||
|
|
||||||
.PHONY: all clean check_lib check_loader_stub check_program
|
.PHONY: all clean check_lib check_loader_stub check_program
|
||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -84,7 +84,7 @@ DEPENDS := $(OFILES:.o=.d)
|
|||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
$(OUTPUT).lz4 : $(OUTPUT).bin
|
$(OUTPUT).lz4 : $(OUTPUT).bin
|
||||||
$(SILENTCMD)$(PYTHON) $(CURRENT_DIRECTORY)/lz4_compress.py $(OUTPUT).bin $(OUTPUT).lz4
|
@python $(CURRENT_DIRECTORY)/lz4_compress.py $(OUTPUT).bin $(OUTPUT).lz4
|
||||||
@echo built ... $(notdir $@)
|
@echo built ... $(notdir $@)
|
||||||
|
|
||||||
$(OUTPUT).bin : $(OUTPUT).elf
|
$(OUTPUT).bin : $(OUTPUT).elf
|
||||||
|
|||||||
@@ -602,7 +602,7 @@ namespace ams::nxboot {
|
|||||||
Print("\n");
|
Print("\n");
|
||||||
|
|
||||||
if (R_SUCCEEDED(save_result)) {
|
if (R_SUCCEEDED(save_result)) {
|
||||||
Print("Report saved to /atmosphere/fatal_errors/report_%016" PRIx64 ".bin\n", f_ctx->report_identifier);
|
Print("Report saved to /atmosphere/fatal_errors/report_%016" PRIx64 ".bin", f_ctx->report_identifier);
|
||||||
} else {
|
} else {
|
||||||
Print("Failed to save report to the SD card! (%08" PRIx32 ")\n", save_result.GetValue());
|
Print("Failed to save report to the SD card! (%08" PRIx32 ")\n", save_result.GetValue());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,17 +23,17 @@ namespace ams::nxboot {
|
|||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 MarikoMasterKekSource[se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 MarikoMasterKekSource[se::AesBlockSize] = {
|
||||||
/* TODO: Update on next change of keys. */
|
/* TODO: Update on next change of keys. */
|
||||||
0x82, 0xE2, 0x0A, 0x59, 0x67, 0xDF, 0xBF, 0x51, 0x47, 0x62, 0x11, 0xF2, 0x41, 0xD3, 0xEE, 0x13
|
0x31, 0xBE, 0x25, 0xFB, 0xDB, 0xB4, 0xEE, 0x49, 0x5C, 0x77, 0x05, 0xC2, 0x36, 0x9F, 0x34, 0x80
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 MarikoMasterKekSourceDev[se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 MarikoMasterKekSourceDev[se::AesBlockSize] = {
|
||||||
/* TODO: Update on next change of keys. */
|
/* TODO: Update on next change of keys. */
|
||||||
0x2E, 0x27, 0x44, 0xEA, 0x32, 0xF8, 0x2C, 0xF0, 0x6F, 0xCA, 0xCD, 0x77, 0xAE, 0xAE, 0x1A, 0x1B
|
0x65, 0x7B, 0x11, 0x46, 0x0E, 0xC2, 0x22, 0x5D, 0xB9, 0xF1, 0xF5, 0x00, 0xF9, 0x3E, 0x1F, 0x70
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 EristaMasterKekSource[se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 EristaMasterKekSource[se::AesBlockSize] = {
|
||||||
/* TODO: Update on next change of keys. */
|
/* TODO: Update on next change of keys. */
|
||||||
0x15, 0xAC, 0x96, 0x34, 0xF5, 0x32, 0x56, 0x68, 0xFE, 0x5B, 0x9D, 0xD7, 0xED, 0x19, 0xB7, 0x8E
|
0xD7, 0x63, 0x74, 0x46, 0x4E, 0xBA, 0x78, 0x0A, 0x7C, 0x9D, 0xB3, 0xE8, 0x7A, 0x3D, 0x71, 0xE3
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 KeyblobKeySource[se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 KeyblobKeySource[se::AesBlockSize] = {
|
||||||
@@ -73,9 +73,6 @@ namespace ams::nxboot {
|
|||||||
{ 0xDA, 0xB9, 0xD6, 0x77, 0x52, 0x2D, 0x1F, 0x78, 0x73, 0xC9, 0x98, 0x5B, 0x06, 0xFE, 0xA0, 0x52 }, /* 17.0.0 Device Master Key Source Source. */
|
{ 0xDA, 0xB9, 0xD6, 0x77, 0x52, 0x2D, 0x1F, 0x78, 0x73, 0xC9, 0x98, 0x5B, 0x06, 0xFE, 0xA0, 0x52 }, /* 17.0.0 Device Master Key Source Source. */
|
||||||
{ 0x14, 0xF5, 0xA5, 0xD0, 0x73, 0x6D, 0x44, 0x80, 0x5F, 0x31, 0x5A, 0x8F, 0x1E, 0xD4, 0x0D, 0x63 }, /* 18.0.0 Device Master Key Source Source. */
|
{ 0x14, 0xF5, 0xA5, 0xD0, 0x73, 0x6D, 0x44, 0x80, 0x5F, 0x31, 0x5A, 0x8F, 0x1E, 0xD4, 0x0D, 0x63 }, /* 18.0.0 Device Master Key Source Source. */
|
||||||
{ 0x07, 0x38, 0x9A, 0xEC, 0x9C, 0xBD, 0x50, 0x4A, 0x4C, 0x1F, 0x04, 0xDA, 0x40, 0x68, 0x29, 0xE3 }, /* 19.0.0 Device Master Key Source Source. */
|
{ 0x07, 0x38, 0x9A, 0xEC, 0x9C, 0xBD, 0x50, 0x4A, 0x4C, 0x1F, 0x04, 0xDA, 0x40, 0x68, 0x29, 0xE3 }, /* 19.0.0 Device Master Key Source Source. */
|
||||||
{ 0xA3, 0x6B, 0x0A, 0xB5, 0x6F, 0x57, 0x4C, 0x5E, 0x00, 0xFD, 0x56, 0x21, 0xF5, 0x06, 0x6B, 0xD1 }, /* 20.0.0 Device Master Key Source Source. */
|
|
||||||
{ 0xF9, 0x62, 0x05, 0x99, 0xE0, 0xB9, 0xA6, 0x9B, 0x9D, 0xAA, 0xB4, 0x12, 0x0B, 0x0F, 0xF5, 0x8F }, /* 21.0.0 Device Master Key Source Source. */
|
|
||||||
{ 0xF8, 0xF4, 0x22, 0xA4, 0x34, 0xAE, 0x0E, 0x0C, 0x4D, 0x5C, 0x5B, 0xA1, 0x1B, 0x46, 0x1C, 0x78 }, /* 22.0.0 Device Master Key Source Source. */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 DeviceMasterKekSources[pkg1::OldDeviceMasterKeyCount][se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 DeviceMasterKekSources[pkg1::OldDeviceMasterKeyCount][se::AesBlockSize] = {
|
||||||
@@ -95,9 +92,6 @@ namespace ams::nxboot {
|
|||||||
{ 0x21, 0xD6, 0x35, 0xF1, 0x0F, 0x7A, 0xF0, 0x5D, 0xDF, 0x79, 0x1C, 0x7A, 0xE4, 0x32, 0x82, 0x9E }, /* 17.0.0 Device Master Kek Source. */
|
{ 0x21, 0xD6, 0x35, 0xF1, 0x0F, 0x7A, 0xF0, 0x5D, 0xDF, 0x79, 0x1C, 0x7A, 0xE4, 0x32, 0x82, 0x9E }, /* 17.0.0 Device Master Kek Source. */
|
||||||
{ 0xE7, 0x85, 0x8C, 0xA2, 0xF4, 0x49, 0xCB, 0x07, 0xD1, 0x8E, 0x48, 0x1B, 0xE8, 0x1E, 0x28, 0x3B }, /* 18.0.0 Device Master Kek Source. */
|
{ 0xE7, 0x85, 0x8C, 0xA2, 0xF4, 0x49, 0xCB, 0x07, 0xD1, 0x8E, 0x48, 0x1B, 0xE8, 0x1E, 0x28, 0x3B }, /* 18.0.0 Device Master Kek Source. */
|
||||||
{ 0x9B, 0xA5, 0xFD, 0x74, 0x7F, 0xCD, 0x23, 0xD1, 0xD9, 0xBD, 0x6C, 0x51, 0x72, 0x5F, 0x3D, 0x1F }, /* 19.0.0 Device Master Kek Source. */
|
{ 0x9B, 0xA5, 0xFD, 0x74, 0x7F, 0xCD, 0x23, 0xD1, 0xD9, 0xBD, 0x6C, 0x51, 0x72, 0x5F, 0x3D, 0x1F }, /* 19.0.0 Device Master Kek Source. */
|
||||||
{ 0xDA, 0xFB, 0x61, 0x39, 0x48, 0x2D, 0xC2, 0x7E, 0x0D, 0x8E, 0x8F, 0x98, 0x57, 0x20, 0xB8, 0x15 }, /* 20.0.0 Device Master Kek Source. */
|
|
||||||
{ 0x92, 0xBF, 0x37, 0x80, 0x0E, 0x79, 0x56, 0x8C, 0x57, 0x75, 0x72, 0x0A, 0x48, 0xD8, 0x15, 0x39 }, /* 21.0.0 Device Master Kek Source. */
|
|
||||||
{ 0xC4, 0x6F, 0x0E, 0x72, 0x43, 0xCE, 0x87, 0xFC, 0x38, 0x95, 0x9B, 0xC9, 0x31, 0x44, 0x97, 0x63 }, /* 22.0.0 Device Master Kek Source. */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 DeviceMasterKekSourcesDev[pkg1::OldDeviceMasterKeyCount][se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 DeviceMasterKekSourcesDev[pkg1::OldDeviceMasterKeyCount][se::AesBlockSize] = {
|
||||||
@@ -117,9 +111,6 @@ namespace ams::nxboot {
|
|||||||
{ 0x4E, 0xCE, 0x7B, 0x2A, 0xEA, 0x2E, 0x3D, 0x16, 0xD5, 0x2A, 0xDE, 0xF6, 0xF8, 0x6A, 0x7D, 0x43 }, /* 17.0.0 Device Master Kek Source. */
|
{ 0x4E, 0xCE, 0x7B, 0x2A, 0xEA, 0x2E, 0x3D, 0x16, 0xD5, 0x2A, 0xDE, 0xF6, 0xF8, 0x6A, 0x7D, 0x43 }, /* 17.0.0 Device Master Kek Source. */
|
||||||
{ 0x3B, 0x00, 0x89, 0xD7, 0xA9, 0x9E, 0xB7, 0x70, 0x86, 0x00, 0xC3, 0x49, 0x52, 0x8C, 0xA4, 0xAF }, /* 18.0.0 Device Master Kek Source. */
|
{ 0x3B, 0x00, 0x89, 0xD7, 0xA9, 0x9E, 0xB7, 0x70, 0x86, 0x00, 0xC3, 0x49, 0x52, 0x8C, 0xA4, 0xAF }, /* 18.0.0 Device Master Kek Source. */
|
||||||
{ 0xAE, 0x78, 0x36, 0xB6, 0x91, 0xEB, 0xAF, 0x9C, 0x18, 0xF1, 0xC0, 0xD5, 0x8A, 0x0C, 0x7C, 0xA1 }, /* 19.0.0 Device Master Kek Source. */
|
{ 0xAE, 0x78, 0x36, 0xB6, 0x91, 0xEB, 0xAF, 0x9C, 0x18, 0xF1, 0xC0, 0xD5, 0x8A, 0x0C, 0x7C, 0xA1 }, /* 19.0.0 Device Master Kek Source. */
|
||||||
{ 0x09, 0x12, 0x4F, 0x26, 0x90, 0xB9, 0xA6, 0xF5, 0xA5, 0x18, 0x74, 0xB6, 0x8D, 0x80, 0x59, 0x3D }, /* 20.0.0 Device Master Kek Source. */
|
|
||||||
{ 0x7A, 0x4C, 0x38, 0xB7, 0x03, 0x6B, 0x1E, 0x81, 0x20, 0x53, 0x14, 0x99, 0xA4, 0x21, 0x92, 0x9F }, /* 21.0.0 Device Master Kek Source. */
|
|
||||||
{ 0xF3, 0xBC, 0xB5, 0xB5, 0x5F, 0x01, 0x50, 0x2B, 0x69, 0x69, 0x3A, 0x6B, 0xF9, 0x2C, 0x11, 0x9F }, /* 22.0.0 Device Master Kek Source. */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 MasterKeySources[pkg1::KeyGeneration_Count][se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 MasterKeySources[pkg1::KeyGeneration_Count][se::AesBlockSize] = {
|
||||||
@@ -142,9 +133,6 @@ namespace ams::nxboot {
|
|||||||
{ 0x25, 0x12, 0x8B, 0xCB, 0xB5, 0x46, 0xA1, 0xF8, 0xE0, 0x52, 0x15, 0xB7, 0x0B, 0x57, 0x00, 0xBD }, /* Master key 0F encrypted with Master key 10. */
|
{ 0x25, 0x12, 0x8B, 0xCB, 0xB5, 0x46, 0xA1, 0xF8, 0xE0, 0x52, 0x15, 0xB7, 0x0B, 0x57, 0x00, 0xBD }, /* Master key 0F encrypted with Master key 10. */
|
||||||
{ 0x58, 0x15, 0xD2, 0xF6, 0x8A, 0xE8, 0x19, 0xAB, 0xFB, 0x2D, 0x52, 0x9D, 0xE7, 0x55, 0xF3, 0x93 }, /* Master key 10 encrypted with Master key 11. */
|
{ 0x58, 0x15, 0xD2, 0xF6, 0x8A, 0xE8, 0x19, 0xAB, 0xFB, 0x2D, 0x52, 0x9D, 0xE7, 0x55, 0xF3, 0x93 }, /* Master key 10 encrypted with Master key 11. */
|
||||||
{ 0x4A, 0x01, 0x3B, 0xC7, 0x44, 0x6E, 0x45, 0xBD, 0xE6, 0x5E, 0x2B, 0xEC, 0x07, 0x37, 0x52, 0x86 }, /* Master key 11 encrypted with Master key 12. */
|
{ 0x4A, 0x01, 0x3B, 0xC7, 0x44, 0x6E, 0x45, 0xBD, 0xE6, 0x5E, 0x2B, 0xEC, 0x07, 0x37, 0x52, 0x86 }, /* Master key 11 encrypted with Master key 12. */
|
||||||
{ 0x97, 0xE4, 0x11, 0xAB, 0x22, 0x72, 0x1A, 0x1F, 0x70, 0x5C, 0x00, 0xB3, 0x96, 0x30, 0x05, 0x28 }, /* Master key 12 encrypted with Master key 13. */
|
|
||||||
{ 0xF7, 0x92, 0xC0, 0xEC, 0xF3, 0xA4, 0x8C, 0xB7, 0x0D, 0xB3, 0xF3, 0xAB, 0x10, 0x9B, 0x18, 0xBA }, /* Master key 13 encrypted with Master key 14. */
|
|
||||||
{ 0x14, 0xCB, 0x60, 0x29, 0x3D, 0xE0, 0xFB, 0xF2, 0x5B, 0x60, 0xB6, 0xC5, 0x2E, 0x77, 0x8F, 0x98 }, /* Master key 14 encrypted with Master key 15. */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constexpr inline const u8 MasterKeySourcesDev[pkg1::KeyGeneration_Count][se::AesBlockSize] = {
|
alignas(se::AesBlockSize) constexpr inline const u8 MasterKeySourcesDev[pkg1::KeyGeneration_Count][se::AesBlockSize] = {
|
||||||
@@ -167,9 +155,6 @@ namespace ams::nxboot {
|
|||||||
{ 0x39, 0x1E, 0x7E, 0xF8, 0x7E, 0x73, 0xEA, 0x6F, 0xAF, 0x00, 0x3A, 0xB4, 0xAA, 0xB8, 0xB7, 0x59 }, /* Master key 0F encrypted with Master key 10. */
|
{ 0x39, 0x1E, 0x7E, 0xF8, 0x7E, 0x73, 0xEA, 0x6F, 0xAF, 0x00, 0x3A, 0xB4, 0xAA, 0xB8, 0xB7, 0x59 }, /* Master key 0F encrypted with Master key 10. */
|
||||||
{ 0x0C, 0x75, 0x39, 0x15, 0x53, 0xEA, 0x81, 0x11, 0xA3, 0xE0, 0xDC, 0x3D, 0x0E, 0x76, 0xC6, 0xB8 }, /* Master key 10 encrypted with Master key 11. */
|
{ 0x0C, 0x75, 0x39, 0x15, 0x53, 0xEA, 0x81, 0x11, 0xA3, 0xE0, 0xDC, 0x3D, 0x0E, 0x76, 0xC6, 0xB8 }, /* Master key 10 encrypted with Master key 11. */
|
||||||
{ 0x90, 0x64, 0xF9, 0x08, 0x29, 0x88, 0xD4, 0xDC, 0x73, 0xA4, 0xA1, 0x13, 0x9E, 0x59, 0x85, 0xA0 }, /* Master key 11 encrypted with Master key 12. */
|
{ 0x90, 0x64, 0xF9, 0x08, 0x29, 0x88, 0xD4, 0xDC, 0x73, 0xA4, 0xA1, 0x13, 0x9E, 0x59, 0x85, 0xA0 }, /* Master key 11 encrypted with Master key 12. */
|
||||||
{ 0x94, 0x46, 0x3B, 0xFA, 0x7D, 0xB9, 0xE2, 0x94, 0xC2, 0x9D, 0xB9, 0xA4, 0xB2, 0x56, 0xCA, 0xFE }, /* Master key 12 encrypted with Master key 13. */
|
|
||||||
{ 0x74, 0xB2, 0x5F, 0xA0, 0x4B, 0x74, 0x6D, 0x47, 0x5B, 0xA9, 0xF5, 0x26, 0x46, 0xD7, 0x4B, 0x6E }, /* Master key 13 encrypted with Master key 14. */
|
|
||||||
{ 0x97, 0xB3, 0x61, 0x88, 0x5C, 0x0D, 0xA1, 0x38, 0x73, 0xA4, 0x2F, 0x1A, 0x46, 0xA1, 0x09, 0xBF }, /* Master key 14 encrypted with Master key 15. */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
alignas(se::AesBlockSize) constinit u8 MasterKeys[pkg1::OldMasterKeyCount][se::AesBlockSize] = {};
|
alignas(se::AesBlockSize) constinit u8 MasterKeys[pkg1::OldMasterKeyCount][se::AesBlockSize] = {};
|
||||||
|
|||||||
@@ -80,7 +80,7 @@ namespace ams::nxboot {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Check that the key generation is one that we can use. */
|
/* Check that the key generation is one that we can use. */
|
||||||
static_assert(pkg1::KeyGeneration_Count == 22);
|
static_assert(pkg1::KeyGeneration_Count == 19);
|
||||||
if (key_generation >= pkg1::KeyGeneration_Count) {
|
if (key_generation >= pkg1::KeyGeneration_Count) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -263,12 +263,6 @@ namespace ams::nxboot {
|
|||||||
return ams::TargetFirmware_18_0_0;
|
return ams::TargetFirmware_18_0_0;
|
||||||
} else if (std::memcmp(package1 + 0x10, "20240808", 8) == 0) {
|
} else if (std::memcmp(package1 + 0x10, "20240808", 8) == 0) {
|
||||||
return ams::TargetFirmware_19_0_0;
|
return ams::TargetFirmware_19_0_0;
|
||||||
} else if (std::memcmp(package1 + 0x10, "20250206", 8) == 0) {
|
|
||||||
return ams::TargetFirmware_20_0_0;
|
|
||||||
} else if (std::memcmp(package1 + 0x10, "20251009", 8) == 0) {
|
|
||||||
return ams::TargetFirmware_21_0_0;
|
|
||||||
} else if (std::memcmp(package1 + 0x10, "20260123", 8) == 0) {
|
|
||||||
return ams::TargetFirmware_22_0_0;
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
@@ -413,9 +407,8 @@ namespace ams::nxboot {
|
|||||||
/* If we should, save the current warmboot firmware. */
|
/* If we should, save the current warmboot firmware. */
|
||||||
UpdateWarmbootPath(expected_fuses);
|
UpdateWarmbootPath(expected_fuses);
|
||||||
if (!IsFileExist(warmboot_path)) {
|
if (!IsFileExist(warmboot_path)) {
|
||||||
/* Try to create the directory/file, allowing them to fail (if already exist). */
|
fs::CreateDirectory("sdmc:/warmboot_mariko");
|
||||||
static_cast<void>(fs::CreateDirectory("sdmc:/warmboot_mariko"));
|
fs::CreateFile(warmboot_path, warmboot_src_size);
|
||||||
static_cast<void>(fs::CreateFile(warmboot_path, warmboot_src_size));
|
|
||||||
|
|
||||||
Result result;
|
Result result;
|
||||||
fs::FileHandle file;
|
fs::FileHandle file;
|
||||||
@@ -545,12 +538,6 @@ namespace ams::nxboot {
|
|||||||
} else {
|
} else {
|
||||||
storage_ctx.flags[0] &= ~secmon::SecureMonitorConfigurationFlag_EnableUserModePerformanceCounterAccess;
|
storage_ctx.flags[0] &= ~secmon::SecureMonitorConfigurationFlag_EnableUserModePerformanceCounterAccess;
|
||||||
}
|
}
|
||||||
} else if (std::strcmp(entry.key, "enable_mem_mode") == 0) {
|
|
||||||
if (entry.value[0] == '1') {
|
|
||||||
storage_ctx.flags[0] |= secmon::SecureMonitorConfigurationFlag_BootConfigMemoryModeEnabled;
|
|
||||||
} else {
|
|
||||||
storage_ctx.flags[0] &= ~secmon::SecureMonitorConfigurationFlag_BootConfigMemoryModeEnabled;
|
|
||||||
}
|
|
||||||
} else if (std::strcmp(entry.key, "blank_prodinfo_sysmmc") == 0) {
|
} else if (std::strcmp(entry.key, "blank_prodinfo_sysmmc") == 0) {
|
||||||
if (!emummc_enabled) {
|
if (!emummc_enabled) {
|
||||||
if (entry.value[0] == '1') {
|
if (entry.value[0] == '1') {
|
||||||
|
|||||||
@@ -180,21 +180,6 @@ namespace ams::nxboot {
|
|||||||
FsVersion_19_0_0,
|
FsVersion_19_0_0,
|
||||||
FsVersion_19_0_0_Exfat,
|
FsVersion_19_0_0_Exfat,
|
||||||
|
|
||||||
FsVersion_20_0_0,
|
|
||||||
FsVersion_20_0_0_Exfat,
|
|
||||||
|
|
||||||
FsVersion_20_1_0,
|
|
||||||
FsVersion_20_1_0_Exfat,
|
|
||||||
|
|
||||||
FsVersion_21_0_0,
|
|
||||||
FsVersion_21_0_0_Exfat,
|
|
||||||
|
|
||||||
FsVersion_21_2_0,
|
|
||||||
FsVersion_21_2_0_Exfat,
|
|
||||||
|
|
||||||
FsVersion_22_0_0,
|
|
||||||
FsVersion_22_0_0_Exfat,
|
|
||||||
|
|
||||||
FsVersion_Count,
|
FsVersion_Count,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -287,21 +272,6 @@ namespace ams::nxboot {
|
|||||||
|
|
||||||
{ 0xD9, 0x4C, 0x68, 0x15, 0xF8, 0xF5, 0x0A, 0x20 }, /* FsVersion_19_0_0 */
|
{ 0xD9, 0x4C, 0x68, 0x15, 0xF8, 0xF5, 0x0A, 0x20 }, /* FsVersion_19_0_0 */
|
||||||
{ 0xED, 0xA8, 0x78, 0x68, 0xA4, 0x49, 0x07, 0x50 }, /* FsVersion_19_0_0_Exfat */
|
{ 0xED, 0xA8, 0x78, 0x68, 0xA4, 0x49, 0x07, 0x50 }, /* FsVersion_19_0_0_Exfat */
|
||||||
|
|
||||||
{ 0x63, 0x54, 0x96, 0x9E, 0x60, 0xA7, 0x97, 0x7B }, /* FsVersion_20_0_0 */
|
|
||||||
{ 0x47, 0x41, 0x07, 0x10, 0x65, 0x4F, 0xA4, 0x3F }, /* FsVersion_20_0_0_Exfat */
|
|
||||||
|
|
||||||
{ 0xED, 0x34, 0xB4, 0x50, 0x58, 0x4A, 0x5B, 0x43 }, /* FsVersion_20_1_0 */
|
|
||||||
{ 0xA5, 0x1A, 0xA4, 0x92, 0x6C, 0x41, 0x87, 0x59 }, /* FsVersion_20_1_0_Exfat */
|
|
||||||
|
|
||||||
{ 0xEE, 0x4B, 0x30, 0x12, 0xA6, 0x84, 0x02, 0x25 }, /* FsVersion_21_0_0 */
|
|
||||||
{ 0x6E, 0x2B, 0xD9, 0xBA, 0xA3, 0xB9, 0x10, 0xF1 }, /* FsVersion_21_0_0_Exfat */
|
|
||||||
|
|
||||||
{ 0xAF, 0x1D, 0xBD, 0xC7, 0x82, 0x98, 0x3C, 0xBD }, /* FsVersion_21_2_0 */
|
|
||||||
{ 0x56, 0x25, 0x17, 0xA1, 0x92, 0xC3, 0xC8, 0xF0 }, /* FsVersion_21_2_0_Exfat */
|
|
||||||
|
|
||||||
{ 0xB7, 0xA2, 0x97, 0x39, 0xB7, 0xED, 0xDE, 0xFC }, /* FsVersion_22_0_0 */
|
|
||||||
{ 0xFB, 0x0B, 0x68, 0xDB, 0x24, 0x03, 0xD1, 0x19 }, /* FsVersion_22_0_0_Exfat */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const InitialProcessBinaryHeader *FindInitialProcessBinary(const pkg2::Package2Header *header, const u8 *data, ams::TargetFirmware target_firmware) {
|
const InitialProcessBinaryHeader *FindInitialProcessBinary(const pkg2::Package2Header *header, const u8 *data, ams::TargetFirmware target_firmware) {
|
||||||
@@ -691,40 +661,6 @@ namespace ams::nxboot {
|
|||||||
AddPatch(fs_meta, 0x1A16A5, NogcPatch0, sizeof(NogcPatch0));
|
AddPatch(fs_meta, 0x1A16A5, NogcPatch0, sizeof(NogcPatch0));
|
||||||
AddPatch(fs_meta, 0x17A9A0, NogcPatch1, sizeof(NogcPatch1));
|
AddPatch(fs_meta, 0x17A9A0, NogcPatch1, sizeof(NogcPatch1));
|
||||||
break;
|
break;
|
||||||
case FsVersion_20_0_0:
|
|
||||||
case FsVersion_20_1_0:
|
|
||||||
AddPatch(fs_meta, 0x1A7E25, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1A8025, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x17C250, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
case FsVersion_20_0_0_Exfat:
|
|
||||||
case FsVersion_20_1_0_Exfat:
|
|
||||||
AddPatch(fs_meta, 0x1B3745, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1B3945, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x187B70, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
case FsVersion_21_0_0:
|
|
||||||
case FsVersion_21_2_0:
|
|
||||||
AddPatch(fs_meta, 0x1AC9ED, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1ACA05, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x17FBE0, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
case FsVersion_21_0_0_Exfat:
|
|
||||||
case FsVersion_21_2_0_Exfat:
|
|
||||||
AddPatch(fs_meta, 0x1B7B4D, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1B7B65, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x18AD40, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
case FsVersion_22_0_0:
|
|
||||||
AddPatch(fs_meta, 0x1B023D, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1B0255, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x183060, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
case FsVersion_22_0_0_Exfat:
|
|
||||||
AddPatch(fs_meta, 0x1BB42D, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x1BB445, NogcPatch0, sizeof(NogcPatch0));
|
|
||||||
AddPatch(fs_meta, 0x18E250, NogcPatch1, sizeof(NogcPatch1));
|
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|||||||
BIN
img/np++.png
BIN
img/np++.png
Binary file not shown.
|
Before Width: | Height: | Size: 2.2 KiB |
@@ -6,7 +6,7 @@
|
|||||||
[subrepo]
|
[subrepo]
|
||||||
remote = https://github.com/Atmosphere-NX/Atmosphere-libs
|
remote = https://github.com/Atmosphere-NX/Atmosphere-libs
|
||||||
branch = master
|
branch = master
|
||||||
commit = 82f1553c4c7e68364f7e630b1c68f2aee8681dee
|
commit = 989fb7be0c68bf229fe6789428b6c448b6de142a
|
||||||
parent = 252f8685b493d0dfd428e9439b0296109776b935
|
parent = be19749841e581de4cc5d38f39f4de5fa4046c52
|
||||||
method = merge
|
method = merge
|
||||||
cmdver = 0.4.9
|
cmdver = 0.4.1
|
||||||
|
|||||||
@@ -297,23 +297,6 @@ FIND_SOURCE_FILES=$(foreach dir,$1,$(filter-out $(notdir $(wildcard $(dir)/*.arc
|
|||||||
|
|
||||||
ATMOSPHERE_GCH_IDENTIFIER := $(ATMOSPHERE_FULL_NAME)
|
ATMOSPHERE_GCH_IDENTIFIER := $(ATMOSPHERE_FULL_NAME)
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
# Python. The scripts should work with Python 2 or 3, but 2 is preferred.
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
PYTHON = $(shell command -v python >/dev/null && echo python || echo python3)
|
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
# Export MAKE:
|
|
||||||
# GCC's LTO driver invokes Make internally. This invocation respects both $(MAKE)
|
|
||||||
# and $(MAKEFLAGS), but only if they're in the environment. By default, MAKEFLAGS
|
|
||||||
# is in the environment while MAKE isn't, so GCC will always use the default
|
|
||||||
# `make` command, yet it inherits MAKEFLAGS from the copy of Make the user
|
|
||||||
# invoked, which might have incompatible flags. In practice this is an issue on
|
|
||||||
# macOS when running e.g. `gmake -j32 -Otarget`. This behavior is arguably a bug
|
|
||||||
# in GCC and/or Make, but we can work around it by exporting MAKE.
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
export MAKE
|
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# Rules for compiling pre-compiled headers
|
# Rules for compiling pre-compiled headers
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ ATMOSPHERE_OPTIMIZATION_FLAG := -O2
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
export DEFINES = $(ATMOSPHERE_DEFINES) -DATMOSPHERE_IS_STRATOSPHERE -D_GNU_SOURCE
|
export DEFINES = $(ATMOSPHERE_DEFINES) -DATMOSPHERE_IS_STRATOSPHERE -D_GNU_SOURCE
|
||||||
export SETTINGS = $(ATMOSPHERE_SETTINGS) $(ATMOSPHERE_OPTIMIZATION_FLAG) -Wextra -Werror -Wno-missing-field-initializers -Wno-error=unused-result
|
export SETTINGS = $(ATMOSPHERE_SETTINGS) $(ATMOSPHERE_OPTIMIZATION_FLAG) -Wextra -Werror -Wno-missing-field-initializers
|
||||||
export CFLAGS = $(ATMOSPHERE_CFLAGS) $(SETTINGS) $(DEFINES) $(INCLUDE)
|
export CFLAGS = $(ATMOSPHERE_CFLAGS) $(SETTINGS) $(DEFINES) $(INCLUDE)
|
||||||
export CXXFLAGS = $(CFLAGS) $(ATMOSPHERE_CXXFLAGS)
|
export CXXFLAGS = $(CFLAGS) $(ATMOSPHERE_CXXFLAGS)
|
||||||
export ASFLAGS = $(ATMOSPHERE_ASFLAGS) $(SETTINGS) $(DEFINES)
|
export ASFLAGS = $(ATMOSPHERE_ASFLAGS) $(SETTINGS) $(DEFINES)
|
||||||
|
|||||||
@@ -39,9 +39,6 @@ namespace ams::pkg1 {
|
|||||||
KeyGeneration_17_0_0 = 0x10,
|
KeyGeneration_17_0_0 = 0x10,
|
||||||
KeyGeneration_18_0_0 = 0x11,
|
KeyGeneration_18_0_0 = 0x11,
|
||||||
KeyGeneration_19_0_0 = 0x12,
|
KeyGeneration_19_0_0 = 0x12,
|
||||||
KeyGeneration_20_0_0 = 0x13,
|
|
||||||
KeyGeneration_21_0_0 = 0x14,
|
|
||||||
KeyGeneration_22_0_0 = 0x15,
|
|
||||||
|
|
||||||
KeyGeneration_Count,
|
KeyGeneration_Count,
|
||||||
|
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ namespace ams::pkg2 {
|
|||||||
constexpr inline int PayloadCount = 3;
|
constexpr inline int PayloadCount = 3;
|
||||||
|
|
||||||
constexpr inline int MinimumValidDataVersion = 0; /* We allow older package2 to load; this value is currently 0x18 in Nintendo's code. */
|
constexpr inline int MinimumValidDataVersion = 0; /* We allow older package2 to load; this value is currently 0x18 in Nintendo's code. */
|
||||||
constexpr inline int CurrentBootloaderVersion = 0x17;
|
constexpr inline int CurrentBootloaderVersion = 0x16;
|
||||||
|
|
||||||
struct Package2Meta {
|
struct Package2Meta {
|
||||||
using Magic = util::FourCC<'P','K','2','1'>;
|
using Magic = util::FourCC<'P','K','2','1'>;
|
||||||
|
|||||||
@@ -30,7 +30,6 @@ namespace ams::secmon {
|
|||||||
SecureMonitorConfigurationFlag_ShouldUseBlankCalibrationBinary = (1u << 5),
|
SecureMonitorConfigurationFlag_ShouldUseBlankCalibrationBinary = (1u << 5),
|
||||||
SecureMonitorConfigurationFlag_AllowWritingToCalibrationBinarySysmmc = (1u << 6),
|
SecureMonitorConfigurationFlag_AllowWritingToCalibrationBinarySysmmc = (1u << 6),
|
||||||
SecureMonitorConfigurationFlag_ForceEnableUsb30 = (1u << 7),
|
SecureMonitorConfigurationFlag_ForceEnableUsb30 = (1u << 7),
|
||||||
SecureMonitorConfigurationFlag_BootConfigMemoryModeEnabled = (1u << 8),
|
|
||||||
|
|
||||||
SecureMonitorConfigurationFlag_Default = SecureMonitorConfigurationFlag_IsDevelopmentFunctionEnabledForKernel,
|
SecureMonitorConfigurationFlag_Default = SecureMonitorConfigurationFlag_IsDevelopmentFunctionEnabledForKernel,
|
||||||
};
|
};
|
||||||
@@ -104,7 +103,6 @@ namespace ams::secmon {
|
|||||||
constexpr bool ShouldUseBlankCalibrationBinary() const { return (this->flags[0] & SecureMonitorConfigurationFlag_ShouldUseBlankCalibrationBinary) != 0; }
|
constexpr bool ShouldUseBlankCalibrationBinary() const { return (this->flags[0] & SecureMonitorConfigurationFlag_ShouldUseBlankCalibrationBinary) != 0; }
|
||||||
constexpr bool AllowWritingToCalibrationBinarySysmmc() const { return (this->flags[0] & SecureMonitorConfigurationFlag_AllowWritingToCalibrationBinarySysmmc) != 0; }
|
constexpr bool AllowWritingToCalibrationBinarySysmmc() const { return (this->flags[0] & SecureMonitorConfigurationFlag_AllowWritingToCalibrationBinarySysmmc) != 0; }
|
||||||
constexpr bool IsUsb30ForceEnabled() const { return (this->flags[0] & SecureMonitorConfigurationFlag_ForceEnableUsb30) != 0; }
|
constexpr bool IsUsb30ForceEnabled() const { return (this->flags[0] & SecureMonitorConfigurationFlag_ForceEnableUsb30) != 0; }
|
||||||
constexpr bool IsBootConfigMemoryModeEnabled() const { return (this->flags[0] & SecureMonitorConfigurationFlag_BootConfigMemoryModeEnabled) != 0; }
|
|
||||||
|
|
||||||
constexpr bool IsDevelopmentFunctionEnabled(bool for_kern) const { return for_kern ? this->IsDevelopmentFunctionEnabledForKernel() : this->IsDevelopmentFunctionEnabledForUser(); }
|
constexpr bool IsDevelopmentFunctionEnabled(bool for_kern) const { return for_kern ? this->IsDevelopmentFunctionEnabledForKernel() : this->IsDevelopmentFunctionEnabledForUser(); }
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -104,7 +104,7 @@ clean:
|
|||||||
@echo clean $(ATMOSPHERE_BUILD_NAME) ...
|
@echo clean $(ATMOSPHERE_BUILD_NAME) ...
|
||||||
@rm -fr $(ATMOSPHERE_BUILD_DIR) $(ATMOSPHERE_OUT_DIR)
|
@rm -fr $(ATMOSPHERE_BUILD_DIR) $(ATMOSPHERE_OUT_DIR)
|
||||||
@rm -fr $(foreach hdr,$(GCH_DIRS),$(hdr)/$(ATMOSPHERE_GCH_IDENTIFIER))
|
@rm -fr $(foreach hdr,$(GCH_DIRS),$(hdr)/$(ATMOSPHERE_GCH_IDENTIFIER))
|
||||||
@for i in $(GCH_DIRS); do [ -d $$i ] && rmdir $$i 2>/dev/null || true; done
|
@for i in $(GCH_DIRS); do [ -d $$i ] && rmdir --ignore-fail-on-non-empty $$i || true; done
|
||||||
|
|
||||||
$(ATMOSPHERE_LIBRARY_DIR) $(ATMOSPHERE_BUILD_DIR) $(GCH_DIRS):
|
$(ATMOSPHERE_LIBRARY_DIR) $(ATMOSPHERE_BUILD_DIR) $(GCH_DIRS):
|
||||||
@[ -d $@ ] || mkdir -p $@
|
@[ -d $@ ] || mkdir -p $@
|
||||||
|
|||||||
@@ -177,9 +177,6 @@ namespace ams::fuse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
constexpr const TargetFirmware FuseVersionIncrementFirmwares[] = {
|
constexpr const TargetFirmware FuseVersionIncrementFirmwares[] = {
|
||||||
TargetFirmware_22_0_0,
|
|
||||||
TargetFirmware_21_0_0,
|
|
||||||
TargetFirmware_20_0_0,
|
|
||||||
TargetFirmware_19_0_0,
|
TargetFirmware_19_0_0,
|
||||||
TargetFirmware_17_0_0,
|
TargetFirmware_17_0_0,
|
||||||
TargetFirmware_16_0_0,
|
TargetFirmware_16_0_0,
|
||||||
|
|||||||
@@ -116,9 +116,7 @@ namespace ams::kern::arch::arm {
|
|||||||
u32 ipriorityr[NumLocalInterrupts / 4];
|
u32 ipriorityr[NumLocalInterrupts / 4];
|
||||||
u32 itargetsr[NumLocalInterrupts / 4];
|
u32 itargetsr[NumLocalInterrupts / 4];
|
||||||
u32 icfgr[NumLocalInterrupts / 16];
|
u32 icfgr[NumLocalInterrupts / 16];
|
||||||
u32 spendsgir[4];
|
|
||||||
};
|
};
|
||||||
static_assert(sizeof(LocalState{}.spendsgir) == sizeof(GicDistributor{}.spendsgir));
|
|
||||||
|
|
||||||
struct GlobalState {
|
struct GlobalState {
|
||||||
u32 isenabler[NumGlobalInterrupts / 32];
|
u32 isenabler[NumGlobalInterrupts / 32];
|
||||||
|
|||||||
@@ -83,9 +83,9 @@ namespace ams::kern::arch::arm64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
NOINLINE Result BindHandler(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
|
NOINLINE Result BindHandler(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
|
||||||
NOINLINE void UnbindHandler(s32 irq, s32 core);
|
NOINLINE Result UnbindHandler(s32 irq, s32 core);
|
||||||
|
|
||||||
NOINLINE void ClearInterrupt(s32 irq, s32 core_id);
|
NOINLINE Result ClearInterrupt(s32 irq, s32 core_id);
|
||||||
|
|
||||||
ALWAYS_INLINE void SendInterProcessorInterrupt(s32 irq, u64 core_mask) {
|
ALWAYS_INLINE void SendInterProcessorInterrupt(s32 irq, u64 core_mask) {
|
||||||
m_interrupt_controller.SendInterProcessorInterrupt(irq, core_mask);
|
m_interrupt_controller.SendInterProcessorInterrupt(irq, core_mask);
|
||||||
@@ -99,10 +99,10 @@ namespace ams::kern::arch::arm64 {
|
|||||||
private:
|
private:
|
||||||
Result BindGlobal(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
|
Result BindGlobal(KInterruptHandler *handler, s32 irq, s32 core_id, s32 priority, bool manual_clear, bool level);
|
||||||
Result BindLocal(KInterruptHandler *handler, s32 irq, s32 priority, bool manual_clear);
|
Result BindLocal(KInterruptHandler *handler, s32 irq, s32 priority, bool manual_clear);
|
||||||
void UnbindGlobal(s32 irq);
|
Result UnbindGlobal(s32 irq);
|
||||||
void UnbindLocal(s32 irq);
|
Result UnbindLocal(s32 irq);
|
||||||
void ClearGlobal(s32 irq);
|
Result ClearGlobal(s32 irq);
|
||||||
void ClearLocal(s32 irq);
|
Result ClearLocal(s32 irq);
|
||||||
private:
|
private:
|
||||||
[[nodiscard]] static ALWAYS_INLINE u32 GetInterruptsEnabledState() {
|
[[nodiscard]] static ALWAYS_INLINE u32 GetInterruptsEnabledState() {
|
||||||
u64 intr_state;
|
u64 intr_state;
|
||||||
|
|||||||
@@ -197,14 +197,9 @@ namespace ams::kern::arch::arm64 {
|
|||||||
cpu::SwitchProcess(s_ttbr0_entries[proc_idx + 1], proc_id);
|
cpu::SwitchProcess(s_ttbr0_entries[proc_idx + 1], proc_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
NOINLINE void InitializeForKernel(void *table, KVirtualAddress start, KVirtualAddress end);
|
NOINLINE Result InitializeForKernel(void *table, KVirtualAddress start, KVirtualAddress end);
|
||||||
NOINLINE Result InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit, size_t process_index);
|
NOINLINE Result InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit, size_t process_index);
|
||||||
void Finalize();
|
Result Finalize();
|
||||||
|
|
||||||
static void NoteUpdatedCallback(const void *pt) {
|
|
||||||
/* Note the update. */
|
|
||||||
static_cast<const KPageTable *>(pt)->NoteUpdated();
|
|
||||||
}
|
|
||||||
private:
|
private:
|
||||||
Result Unmap(KProcessAddress virt_addr, size_t num_pages, PageLinkedList *page_list, bool force, bool reuse_ll);
|
Result Unmap(KProcessAddress virt_addr, size_t num_pages, PageLinkedList *page_list, bool force, bool reuse_ll);
|
||||||
|
|
||||||
|
|||||||
@@ -49,8 +49,6 @@ namespace ams::kern::arch::arm64 {
|
|||||||
EntryLevel level;
|
EntryLevel level;
|
||||||
bool is_contiguous;
|
bool is_contiguous;
|
||||||
};
|
};
|
||||||
|
|
||||||
using EntryUpdatedCallback = void (*)(const void *);
|
|
||||||
private:
|
private:
|
||||||
static constexpr size_t PageBits = util::CountTrailingZeros(PageSize);
|
static constexpr size_t PageBits = util::CountTrailingZeros(PageSize);
|
||||||
static constexpr size_t NumLevels = 3;
|
static constexpr size_t NumLevels = 3;
|
||||||
@@ -146,8 +144,8 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
bool GetPhysicalAddress(KPhysicalAddress *out, KProcessAddress virt_addr) const;
|
bool GetPhysicalAddress(KPhysicalAddress *out, KProcessAddress virt_addr) const;
|
||||||
|
|
||||||
static bool MergePages(KVirtualAddress *out, TraversalContext *context, EntryUpdatedCallback on_entry_updated, const void *pt);
|
static bool MergePages(KVirtualAddress *out, TraversalContext *context);
|
||||||
void SeparatePages(TraversalEntry *entry, TraversalContext *context, KProcessAddress address, PageTableEntry *pte, EntryUpdatedCallback on_entry_updated, const void *pt) const;
|
void SeparatePages(TraversalEntry *entry, TraversalContext *context, KProcessAddress address, PageTableEntry *pte) const;
|
||||||
|
|
||||||
KProcessAddress GetAddressForContext(const TraversalContext *context) const {
|
KProcessAddress GetAddressForContext(const TraversalContext *context) const {
|
||||||
KProcessAddress addr = m_is_kernel ? static_cast<uintptr_t>(-GetBlockSize(EntryLevel_L1)) * m_num_entries : 0;
|
KProcessAddress addr = m_is_kernel ? static_cast<uintptr_t>(-GetBlockSize(EntryLevel_L1)) * m_num_entries : 0;
|
||||||
|
|||||||
@@ -165,7 +165,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
constexpr explicit KThreadContext(util::ConstantInitializeTag) : m_callee_saved(), m_lr(), m_sp(), m_fpcr(), m_fpsr(), m_callee_saved_fpu(), m_locked() { /* ... */ }
|
constexpr explicit KThreadContext(util::ConstantInitializeTag) : m_callee_saved(), m_lr(), m_sp(), m_fpcr(), m_fpsr(), m_callee_saved_fpu(), m_locked() { /* ... */ }
|
||||||
explicit KThreadContext() { /* ... */ }
|
explicit KThreadContext() { /* ... */ }
|
||||||
|
|
||||||
void Initialize(KVirtualAddress u_pc, KVirtualAddress k_sp, KVirtualAddress u_sp, uintptr_t arg, bool is_user, bool is_64_bit, bool is_main);
|
Result Initialize(KVirtualAddress u_pc, KVirtualAddress k_sp, KVirtualAddress u_sp, uintptr_t arg, bool is_user, bool is_64_bit, bool is_main);
|
||||||
|
|
||||||
void SetArguments(uintptr_t arg0, uintptr_t arg1);
|
void SetArguments(uintptr_t arg0, uintptr_t arg1);
|
||||||
|
|
||||||
|
|||||||
@@ -22,139 +22,49 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
class UserspaceAccess {
|
class UserspaceAccess {
|
||||||
private:
|
private:
|
||||||
class Impl {
|
static bool CopyMemoryFromUserSize32BitWithSupervisorAccessImpl(void *dst, const void *src);
|
||||||
public:
|
|
||||||
static bool CopyMemoryFromUser(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryFromUserAligned32Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryFromUserAligned64Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryFromUserSize64Bit(void *dst, const void *src);
|
|
||||||
static bool CopyMemoryFromUserSize32Bit(void *dst, const void *src);
|
|
||||||
static bool CopyMemoryFromUserSize32BitWithSupervisorAccess(void *dst, const void *src);
|
|
||||||
static s32 CopyStringFromUser(void *dst, const void *src, size_t size);
|
|
||||||
|
|
||||||
static bool CopyMemoryToUser(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryToUserAligned32Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryToUserAligned64Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool CopyMemoryToUserSize32Bit(void *dst, u32 value);
|
|
||||||
static s32 CopyStringToUser(void *dst, const void *src, size_t size);
|
|
||||||
|
|
||||||
static bool UpdateLockAtomic(u32 *out, u32 *address, u32 if_zero, u32 new_orr_mask);
|
|
||||||
static bool UpdateIfEqualAtomic(s32 *out, s32 *address, s32 compare_value, s32 new_value);
|
|
||||||
static bool DecrementIfLessThanAtomic(s32 *out, s32 *address, s32 compare);
|
|
||||||
|
|
||||||
static bool StoreDataCache(uintptr_t start, uintptr_t end);
|
|
||||||
static bool FlushDataCache(uintptr_t start, uintptr_t end);
|
|
||||||
static bool InvalidateDataCache(uintptr_t start, uintptr_t end);
|
|
||||||
|
|
||||||
static bool ReadIoMemory32Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool ReadIoMemory16Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool ReadIoMemory8Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool WriteIoMemory32Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool WriteIoMemory16Bit(void *dst, const void *src, size_t size);
|
|
||||||
static bool WriteIoMemory8Bit(void *dst, const void *src, size_t size);
|
|
||||||
};
|
|
||||||
public:
|
public:
|
||||||
static bool CopyMemoryFromUser(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::CopyMemoryFromUser(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool CopyMemoryFromUserAligned32Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::CopyMemoryFromUserAligned32Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool CopyMemoryFromUserAligned64Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::CopyMemoryFromUserAligned64Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool CopyMemoryFromUserSize64Bit(void *dst, const void *src) {
|
|
||||||
return Impl::CopyMemoryFromUserSize64Bit(dst, src);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool CopyMemoryFromUserSize32Bit(void *dst, const void *src) {
|
|
||||||
return Impl::CopyMemoryFromUserSize32Bit(dst, src);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static bool CopyMemoryFromUserSize32BitWithSupervisorAccess(void *dst, const void *src) {
|
static bool CopyMemoryFromUserSize32BitWithSupervisorAccess(void *dst, const void *src) {
|
||||||
/* Check that the address is within the valid userspace range. */
|
/* Check that the address is within the valid userspace range. */
|
||||||
if (const uintptr_t src_uptr = reinterpret_cast<uintptr_t>(src); src_uptr < ams::svc::AddressNullGuard32Size || (src_uptr + sizeof(u32) - 1) >= ams::svc::AddressMemoryRegion39Size) {
|
if (const uintptr_t src_uptr = reinterpret_cast<uintptr_t>(src); src_uptr < ams::svc::AddressNullGuard32Size || (src_uptr + sizeof(u32) - 1) >= ams::svc::AddressMemoryRegion39Size) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return Impl::CopyMemoryFromUserSize32BitWithSupervisorAccess(dst, src);
|
return CopyMemoryFromUserSize32BitWithSupervisorAccessImpl(dst, src);
|
||||||
}
|
}
|
||||||
|
|
||||||
static s32 CopyStringFromUser(void *dst, const void *src, size_t size) {
|
static bool CopyMemoryFromUser(void *dst, const void *src, size_t size);
|
||||||
return Impl::CopyStringFromUser(dst, src, size);
|
static bool CopyMemoryFromUserAligned32Bit(void *dst, const void *src, size_t size);
|
||||||
}
|
static bool CopyMemoryFromUserAligned64Bit(void *dst, const void *src, size_t size);
|
||||||
|
static bool CopyMemoryFromUserSize64Bit(void *dst, const void *src);
|
||||||
|
static bool CopyMemoryFromUserSize32Bit(void *dst, const void *src);
|
||||||
|
static s32 CopyStringFromUser(void *dst, const void *src, size_t size);
|
||||||
|
|
||||||
static bool CopyMemoryToUser(void *dst, const void *src, size_t size) {
|
static bool CopyMemoryToUser(void *dst, const void *src, size_t size);
|
||||||
return Impl::CopyMemoryToUser(dst, src, size);
|
static bool CopyMemoryToUserAligned32Bit(void *dst, const void *src, size_t size);
|
||||||
}
|
static bool CopyMemoryToUserAligned64Bit(void *dst, const void *src, size_t size);
|
||||||
|
static bool CopyMemoryToUserSize32Bit(void *dst, const void *src);
|
||||||
|
static s32 CopyStringToUser(void *dst, const void *src, size_t size);
|
||||||
|
|
||||||
static bool CopyMemoryToUserAligned32Bit(void *dst, const void *src, size_t size) {
|
static bool ClearMemory(void *dst, size_t size);
|
||||||
return Impl::CopyMemoryToUserAligned32Bit(dst, src, size);
|
static bool ClearMemoryAligned32Bit(void *dst, size_t size);
|
||||||
}
|
static bool ClearMemoryAligned64Bit(void *dst, size_t size);
|
||||||
|
static bool ClearMemorySize32Bit(void *dst);
|
||||||
|
|
||||||
static bool CopyMemoryToUserAligned64Bit(void *dst, const void *src, size_t size) {
|
static bool UpdateLockAtomic(u32 *out, u32 *address, u32 if_zero, u32 new_orr_mask);
|
||||||
return Impl::CopyMemoryToUserAligned64Bit(dst, src, size);
|
static bool UpdateIfEqualAtomic(s32 *out, s32 *address, s32 compare_value, s32 new_value);
|
||||||
}
|
static bool DecrementIfLessThanAtomic(s32 *out, s32 *address, s32 compare);
|
||||||
|
|
||||||
static bool CopyMemoryToUserSize32Bit(void *dst, u32 value) {
|
static bool StoreDataCache(uintptr_t start, uintptr_t end);
|
||||||
return Impl::CopyMemoryToUserSize32Bit(dst, value);
|
static bool FlushDataCache(uintptr_t start, uintptr_t end);
|
||||||
}
|
static bool InvalidateDataCache(uintptr_t start, uintptr_t end);
|
||||||
|
|
||||||
static s32 CopyStringToUser(void *dst, const void *src, size_t size) {
|
static bool ReadIoMemory32Bit(void *dst, const void *src, size_t size);
|
||||||
return Impl::CopyStringToUser(dst, src, size);
|
static bool ReadIoMemory16Bit(void *dst, const void *src, size_t size);
|
||||||
}
|
static bool ReadIoMemory8Bit(void *dst, const void *src, size_t size);
|
||||||
|
static bool WriteIoMemory32Bit(void *dst, const void *src, size_t size);
|
||||||
static bool UpdateLockAtomic(u32 *out, u32 *address, u32 if_zero, u32 new_orr_mask) {
|
static bool WriteIoMemory16Bit(void *dst, const void *src, size_t size);
|
||||||
return Impl::UpdateLockAtomic(out, address, if_zero, new_orr_mask);
|
static bool WriteIoMemory8Bit(void *dst, const void *src, size_t size);
|
||||||
}
|
|
||||||
|
|
||||||
static bool UpdateIfEqualAtomic(s32 *out, s32 *address, s32 compare_value, s32 new_value) {
|
|
||||||
return Impl::UpdateIfEqualAtomic(out, address, compare_value, new_value);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool DecrementIfLessThanAtomic(s32 *out, s32 *address, s32 compare) {
|
|
||||||
return Impl::DecrementIfLessThanAtomic(out, address, compare);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool StoreDataCache(uintptr_t start, uintptr_t end) {
|
|
||||||
return Impl::StoreDataCache(start, end);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool FlushDataCache(uintptr_t start, uintptr_t end) {
|
|
||||||
return Impl::FlushDataCache(start, end);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool InvalidateDataCache(uintptr_t start, uintptr_t end) {
|
|
||||||
return Impl::InvalidateDataCache(start, end);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ReadIoMemory32Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::ReadIoMemory32Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ReadIoMemory16Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::ReadIoMemory16Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool ReadIoMemory8Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::ReadIoMemory8Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool WriteIoMemory32Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::WriteIoMemory32Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool WriteIoMemory16Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::WriteIoMemory16Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool WriteIoMemory8Bit(void *dst, const void *src, size_t size) {
|
|
||||||
return Impl::WriteIoMemory8Bit(dst, src, size);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -47,6 +47,13 @@
|
|||||||
/* re-enabled by toggling this define. */
|
/* re-enabled by toggling this define. */
|
||||||
//#define MESOSPHERE_ENABLE_PROCESS_CREATION_TIME
|
//#define MESOSPHERE_ENABLE_PROCESS_CREATION_TIME
|
||||||
|
|
||||||
|
/* NOTE: This enables fast class token storage using a class member. */
|
||||||
|
/* This saves a virtual call when doing KAutoObject->DynCast<>(), */
|
||||||
|
/* at the cost of storing class tokens inside the class object. */
|
||||||
|
/* However, as of (10/16/2021) KAutoObject has an unused class member */
|
||||||
|
/* of the right side, and so this does not actually cost any space. */
|
||||||
|
#define MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST
|
||||||
|
|
||||||
/* NOTE: This enables usage of KDebug handles as parameter for svc::GetInfo */
|
/* NOTE: This enables usage of KDebug handles as parameter for svc::GetInfo */
|
||||||
/* calls which require a process parameter. This enables a debugger to obtain */
|
/* calls which require a process parameter. This enables a debugger to obtain */
|
||||||
/* address space/layout information, for example. However, it changes abi, and so */
|
/* address space/layout information, for example. However, it changes abi, and so */
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ namespace ams::kern {
|
|||||||
size_t m_size;
|
size_t m_size;
|
||||||
Type m_type;
|
Type m_type;
|
||||||
public:
|
public:
|
||||||
static uintptr_t GetAddressSpaceStart(ams::svc::CreateProcessFlag flags, Type type, size_t code_size);
|
static uintptr_t GetAddressSpaceStart(ams::svc::CreateProcessFlag flags, Type type);
|
||||||
static size_t GetAddressSpaceSize(ams::svc::CreateProcessFlag flags, Type type);
|
static size_t GetAddressSpaceSize(ams::svc::CreateProcessFlag flags, Type type);
|
||||||
|
|
||||||
static void SetAddressSpaceSize(size_t width, Type type, size_t size);
|
static void SetAddressSpaceSize(size_t width, Type type, size_t size);
|
||||||
|
|||||||
@@ -121,9 +121,14 @@ namespace ams::kern {
|
|||||||
private:
|
private:
|
||||||
KAutoObject *m_next_closed_object;
|
KAutoObject *m_next_closed_object;
|
||||||
ReferenceCount m_ref_count;
|
ReferenceCount m_ref_count;
|
||||||
|
#if defined(MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST)
|
||||||
ClassTokenType m_class_token;
|
ClassTokenType m_class_token;
|
||||||
|
#endif
|
||||||
public:
|
public:
|
||||||
constexpr ALWAYS_INLINE explicit KAutoObject(util::ConstantInitializeTag) : m_next_closed_object(nullptr), m_ref_count(0), m_class_token(0)
|
constexpr ALWAYS_INLINE explicit KAutoObject(util::ConstantInitializeTag) : m_next_closed_object(nullptr), m_ref_count(0)
|
||||||
|
#if defined(MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST)
|
||||||
|
, m_class_token(0)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
}
|
}
|
||||||
@@ -146,11 +151,19 @@ namespace ams::kern {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE bool IsDerivedFrom(const TypeObj &rhs) const {
|
ALWAYS_INLINE bool IsDerivedFrom(const TypeObj &rhs) const {
|
||||||
return TypeObj::IsClassTokenDerivedFrom(m_class_token, rhs.GetClassToken());
|
#if defined(MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST)
|
||||||
|
return TypeObj::IsClassTokenDerivedFrom(m_class_token, rhs.GetClassToken());
|
||||||
|
#else
|
||||||
|
return this->GetTypeObj().IsDerivedFrom(rhs);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE bool IsDerivedFrom(const KAutoObject &rhs) const {
|
ALWAYS_INLINE bool IsDerivedFrom(const KAutoObject &rhs) const {
|
||||||
return TypeObj::IsClassTokenDerivedFrom(m_class_token, rhs.m_class_token);
|
#if defined(MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST)
|
||||||
|
return TypeObj::IsClassTokenDerivedFrom(m_class_token, rhs.m_class_token);
|
||||||
|
#else
|
||||||
|
return this->IsDerivedFrom(rhs.GetTypeObj());
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Derived>
|
template<typename Derived>
|
||||||
@@ -205,10 +218,12 @@ namespace ams::kern {
|
|||||||
KAutoObject &auto_object = *static_cast<KAutoObject *>(obj);
|
KAutoObject &auto_object = *static_cast<KAutoObject *>(obj);
|
||||||
|
|
||||||
/* If we should, set our class token. */
|
/* If we should, set our class token. */
|
||||||
|
#if defined(MESOSPHERE_ENABLE_DEVIRTUALIZED_DYNAMIC_CAST)
|
||||||
{
|
{
|
||||||
constexpr auto Token = Derived::GetStaticTypeObj().GetClassToken();
|
constexpr auto Token = Derived::GetStaticTypeObj().GetClassToken();
|
||||||
auto_object.m_class_token = Token;
|
auto_object.m_class_token = Token;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Initialize reference count to 1. */
|
/* Initialize reference count to 1. */
|
||||||
auto_object.m_ref_count = 1;
|
auto_object.m_ref_count = 1;
|
||||||
|
|||||||
@@ -93,9 +93,9 @@ namespace ams::kern {
|
|||||||
static Result ProcessDebugEvent(ams::svc::DebugEvent event, const uintptr_t *params, size_t num_params);
|
static Result ProcessDebugEvent(ams::svc::DebugEvent event, const uintptr_t *params, size_t num_params);
|
||||||
public:
|
public:
|
||||||
static Result OnDebugEvent(ams::svc::DebugEvent event, const uintptr_t *params, size_t num_params);
|
static Result OnDebugEvent(ams::svc::DebugEvent event, const uintptr_t *params, size_t num_params);
|
||||||
static void OnExitProcess(KProcess *process);
|
static Result OnExitProcess(KProcess *process);
|
||||||
static void OnTerminateProcess(KProcess *process);
|
static Result OnTerminateProcess(KProcess *process);
|
||||||
static void OnExitThread(KThread *thread);
|
static Result OnExitThread(KThread *thread);
|
||||||
static KEventInfo *CreateDebugEvent(ams::svc::DebugEvent event, u64 thread_id, const uintptr_t *params, size_t num_params);
|
static KEventInfo *CreateDebugEvent(ams::svc::DebugEvent event, u64 thread_id, const uintptr_t *params, size_t num_params);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -50,8 +50,8 @@ namespace ams::kern {
|
|||||||
|
|
||||||
KReadableEvent &GetReadableEvent() { return m_readable_event; }
|
KReadableEvent &GetReadableEvent() { return m_readable_event; }
|
||||||
|
|
||||||
void Signal();
|
Result Signal();
|
||||||
void Clear();
|
Result Clear();
|
||||||
|
|
||||||
ALWAYS_INLINE void OnReadableEventDestroyed() { m_readable_event_destroyed = true; }
|
ALWAYS_INLINE void OnReadableEventDestroyed() { m_readable_event_destroyed = true; }
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -105,7 +105,7 @@ namespace ams::kern {
|
|||||||
constexpr ALWAYS_INLINE size_t GetCount() const { return m_count; }
|
constexpr ALWAYS_INLINE size_t GetCount() const { return m_count; }
|
||||||
constexpr ALWAYS_INLINE size_t GetMaxCount() const { return m_max_count; }
|
constexpr ALWAYS_INLINE size_t GetMaxCount() const { return m_max_count; }
|
||||||
|
|
||||||
MESOSPHERE_NOINLINE_IF_DEBUG void Finalize();
|
MESOSPHERE_NOINLINE_IF_DEBUG Result Finalize();
|
||||||
MESOSPHERE_NOINLINE_IF_DEBUG bool Remove(ams::svc::Handle handle);
|
MESOSPHERE_NOINLINE_IF_DEBUG bool Remove(ams::svc::Handle handle);
|
||||||
|
|
||||||
template<typename T = KAutoObject>
|
template<typename T = KAutoObject>
|
||||||
|
|||||||
@@ -38,11 +38,13 @@ namespace ams::kern {
|
|||||||
|
|
||||||
Result Reset();
|
Result Reset();
|
||||||
|
|
||||||
void Clear() {
|
Result Clear() {
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
|
|
||||||
/* Try to perform a reset, ignoring whether it succeeds. */
|
/* Try to perform a reset, succeeding unconditionally. */
|
||||||
static_cast<void>(this->Reset());
|
this->Reset();
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsInitialized() const { return m_is_initialized; }
|
bool IsInitialized() const { return m_is_initialized; }
|
||||||
|
|||||||
@@ -99,7 +99,6 @@ namespace ams::kern {
|
|||||||
Result Initialize(KProcessAddress st, KProcessAddress nd, KMemoryBlockSlabManager *slab_manager);
|
Result Initialize(KProcessAddress st, KProcessAddress nd, KMemoryBlockSlabManager *slab_manager);
|
||||||
void Finalize(KMemoryBlockSlabManager *slab_manager);
|
void Finalize(KMemoryBlockSlabManager *slab_manager);
|
||||||
|
|
||||||
static bool GetRegionForFindFreeArea(KProcessAddress *out_start, KProcessAddress *out_end, KProcessAddress region_start, size_t region_num_pages, size_t num_pages, size_t alignment, size_t offset, size_t guard_pages);
|
|
||||||
KProcessAddress FindFreeArea(KProcessAddress region_start, size_t region_num_pages, size_t num_pages, size_t alignment, size_t offset, size_t guard_pages) const;
|
KProcessAddress FindFreeArea(KProcessAddress region_start, size_t region_num_pages, size_t num_pages, size_t alignment, size_t offset, size_t guard_pages) const;
|
||||||
|
|
||||||
void Update(KMemoryBlockManagerUpdateAllocator *allocator, KProcessAddress address, size_t num_pages, KMemoryState state, KMemoryPermission perm, KMemoryAttribute attr, KMemoryBlockDisableMergeAttribute set_disable_attr, KMemoryBlockDisableMergeAttribute clear_disable_attr);
|
void Update(KMemoryBlockManagerUpdateAllocator *allocator, KProcessAddress address, size_t num_pages, KMemoryState state, KMemoryPermission perm, KMemoryAttribute attr, KMemoryBlockDisableMergeAttribute set_disable_attr, KMemoryBlockDisableMergeAttribute clear_disable_attr);
|
||||||
|
|||||||
@@ -200,7 +200,6 @@ namespace ams::kern {
|
|||||||
bool m_is_kernel;
|
bool m_is_kernel;
|
||||||
bool m_enable_aslr;
|
bool m_enable_aslr;
|
||||||
bool m_enable_device_address_space_merge;
|
bool m_enable_device_address_space_merge;
|
||||||
bool m_allowed_exec_device_mapping;
|
|
||||||
KMemoryBlockSlabManager *m_memory_block_slab_manager;
|
KMemoryBlockSlabManager *m_memory_block_slab_manager;
|
||||||
KBlockInfoManager *m_block_info_manager;
|
KBlockInfoManager *m_block_info_manager;
|
||||||
KResourceLimit *m_resource_limit;
|
KResourceLimit *m_resource_limit;
|
||||||
@@ -218,7 +217,7 @@ namespace ams::kern {
|
|||||||
m_alias_code_region_end(Null<KProcessAddress>), m_code_region_start(Null<KProcessAddress>), m_code_region_end(Null<KProcessAddress>),
|
m_alias_code_region_end(Null<KProcessAddress>), m_code_region_start(Null<KProcessAddress>), m_code_region_end(Null<KProcessAddress>),
|
||||||
m_max_heap_size(), m_mapped_physical_memory_size(), m_mapped_unsafe_physical_memory(), m_mapped_insecure_memory(), m_mapped_ipc_server_memory(), m_alias_region_extra_size(),
|
m_max_heap_size(), m_mapped_physical_memory_size(), m_mapped_unsafe_physical_memory(), m_mapped_insecure_memory(), m_mapped_ipc_server_memory(), m_alias_region_extra_size(),
|
||||||
m_general_lock(), m_map_physical_memory_lock(), m_device_map_lock(), m_impl(util::ConstantInitialize), m_memory_block_manager(util::ConstantInitialize),
|
m_general_lock(), m_map_physical_memory_lock(), m_device_map_lock(), m_impl(util::ConstantInitialize), m_memory_block_manager(util::ConstantInitialize),
|
||||||
m_allocate_option(), m_address_space_width(), m_is_kernel(), m_enable_aslr(), m_enable_device_address_space_merge(), m_allowed_exec_device_mapping(),
|
m_allocate_option(), m_address_space_width(), m_is_kernel(), m_enable_aslr(), m_enable_device_address_space_merge(),
|
||||||
m_memory_block_slab_manager(), m_block_info_manager(), m_resource_limit(), m_cached_physical_linear_region(), m_cached_physical_heap_region(),
|
m_memory_block_slab_manager(), m_block_info_manager(), m_resource_limit(), m_cached_physical_linear_region(), m_cached_physical_heap_region(),
|
||||||
m_heap_fill_value(), m_ipc_fill_value(), m_stack_fill_value()
|
m_heap_fill_value(), m_ipc_fill_value(), m_stack_fill_value()
|
||||||
{
|
{
|
||||||
@@ -227,7 +226,7 @@ namespace ams::kern {
|
|||||||
|
|
||||||
explicit KPageTableBase() { /* ... */ }
|
explicit KPageTableBase() { /* ... */ }
|
||||||
|
|
||||||
NOINLINE void InitializeForKernel(bool is_64_bit, void *table, KVirtualAddress start, KVirtualAddress end);
|
NOINLINE Result InitializeForKernel(bool is_64_bit, void *table, KVirtualAddress start, KVirtualAddress end);
|
||||||
NOINLINE Result InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, void *table, KProcessAddress start, KProcessAddress end, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit);
|
NOINLINE Result InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, void *table, KProcessAddress start, KProcessAddress end, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit);
|
||||||
|
|
||||||
void Finalize();
|
void Finalize();
|
||||||
@@ -521,8 +520,6 @@ namespace ams::kern {
|
|||||||
size_t GetAliasCodeDataSize() const;
|
size_t GetAliasCodeDataSize() const;
|
||||||
|
|
||||||
u32 GetAllocateOption() const { return m_allocate_option; }
|
u32 GetAllocateOption() const { return m_allocate_option; }
|
||||||
|
|
||||||
void AllowDeviceMappingOfExecPages() { m_allowed_exec_device_mapping = true; }
|
|
||||||
public:
|
public:
|
||||||
static ALWAYS_INLINE KVirtualAddress GetLinearMappedVirtualAddress(KPhysicalAddress addr) {
|
static ALWAYS_INLINE KVirtualAddress GetLinearMappedVirtualAddress(KPhysicalAddress addr) {
|
||||||
return KMemoryLayout::GetLinearVirtualAddress(addr);
|
return KMemoryLayout::GetLinearVirtualAddress(addr);
|
||||||
|
|||||||
@@ -102,8 +102,8 @@ namespace ams::kern {
|
|||||||
IoRegionList m_io_region_list;
|
IoRegionList m_io_region_list;
|
||||||
bool m_is_suspended;
|
bool m_is_suspended;
|
||||||
bool m_is_immortal;
|
bool m_is_immortal;
|
||||||
bool m_is_handle_table_initialized;
|
|
||||||
bool m_is_jit_debug;
|
bool m_is_jit_debug;
|
||||||
|
bool m_is_handle_table_initialized;
|
||||||
ams::svc::DebugEvent m_jit_debug_event_type;
|
ams::svc::DebugEvent m_jit_debug_event_type;
|
||||||
ams::svc::DebugException m_jit_debug_exception_type;
|
ams::svc::DebugException m_jit_debug_exception_type;
|
||||||
uintptr_t m_jit_debug_params[4];
|
uintptr_t m_jit_debug_params[4];
|
||||||
@@ -269,7 +269,7 @@ namespace ams::kern {
|
|||||||
void RemoveIoRegion(KIoRegion *io_region);
|
void RemoveIoRegion(KIoRegion *io_region);
|
||||||
|
|
||||||
Result CreateThreadLocalRegion(KProcessAddress *out);
|
Result CreateThreadLocalRegion(KProcessAddress *out);
|
||||||
void DeleteThreadLocalRegion(KProcessAddress addr);
|
Result DeleteThreadLocalRegion(KProcessAddress addr);
|
||||||
void *GetThreadLocalRegionPointer(KProcessAddress addr);
|
void *GetThreadLocalRegionPointer(KProcessAddress addr);
|
||||||
|
|
||||||
constexpr KProcessAddress GetProcessLocalRegionAddress() const { return m_plr_address; }
|
constexpr KProcessAddress GetProcessLocalRegionAddress() const { return m_plr_address; }
|
||||||
|
|||||||
@@ -35,14 +35,16 @@ namespace ams::kern {
|
|||||||
|
|
||||||
constexpr KEvent *GetParent() const { return m_parent; }
|
constexpr KEvent *GetParent() const { return m_parent; }
|
||||||
|
|
||||||
void Signal();
|
Result Signal();
|
||||||
Result Reset();
|
Result Reset();
|
||||||
|
|
||||||
void Clear() {
|
Result Clear() {
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
|
|
||||||
/* Try to perform a reset, ignoring whether it succeeds. */
|
/* Try to perform a reset, succeeding unconditionally. */
|
||||||
static_cast<void>(this->Reset());
|
this->Reset();
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual bool IsSignaled() const override;
|
virtual bool IsSignaled() const override;
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ namespace ams::kern {
|
|||||||
/* Nintendo uses std::mt19937_t for randomness. */
|
/* Nintendo uses std::mt19937_t for randomness. */
|
||||||
/* To save space (and because mt19337_t isn't secure anyway), */
|
/* To save space (and because mt19337_t isn't secure anyway), */
|
||||||
/* We will use TinyMT. */
|
/* We will use TinyMT. */
|
||||||
static constinit inline bool s_uninitialized_random_generator{true};
|
static constinit inline bool s_initialized_random_generator;
|
||||||
static constinit inline util::TinyMT s_random_generator{util::ConstantInitialize};
|
static constinit inline util::TinyMT s_random_generator{util::ConstantInitialize};
|
||||||
static constinit inline KSpinLock s_random_lock;
|
static constinit inline KSpinLock s_random_lock;
|
||||||
public:
|
public:
|
||||||
|
|||||||
@@ -94,11 +94,7 @@ namespace ams::kern {
|
|||||||
static void PostDestroy(uintptr_t arg) { MESOSPHERE_UNUSED(arg); /* ... */ }
|
static void PostDestroy(uintptr_t arg) { MESOSPHERE_UNUSED(arg); /* ... */ }
|
||||||
|
|
||||||
ALWAYS_INLINE size_t CalculateRequiredSecureMemorySize() const {
|
ALWAYS_INLINE size_t CalculateRequiredSecureMemorySize() const {
|
||||||
if (m_resource_limit != nullptr) {
|
return CalculateRequiredSecureMemorySize(m_resource_size, m_resource_pool);
|
||||||
return CalculateRequiredSecureMemorySize(m_resource_size, m_resource_pool);
|
|
||||||
} else {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE size_t GetSize() const { return m_resource_size; }
|
ALWAYS_INLINE size_t GetSize() const { return m_resource_size; }
|
||||||
|
|||||||
@@ -25,35 +25,35 @@ namespace ams::kern {
|
|||||||
friend class KSystemControl;
|
friend class KSystemControl;
|
||||||
private:
|
private:
|
||||||
struct KTargetSystemData {
|
struct KTargetSystemData {
|
||||||
bool is_not_debug_mode;
|
bool is_debug_mode;
|
||||||
bool disable_debug_logging;
|
bool enable_debug_logging;
|
||||||
bool disable_user_exception_handlers;
|
bool enable_user_exception_handlers;
|
||||||
bool disable_debug_memory_fill;
|
bool enable_debug_memory_fill;
|
||||||
bool disable_user_pmu_access;
|
bool enable_user_pmu_access;
|
||||||
bool disable_kernel_debugging;
|
bool enable_kernel_debugging;
|
||||||
bool disable_dynamic_resource_limits;
|
bool enable_dynamic_resource_limits;
|
||||||
};
|
};
|
||||||
private:
|
private:
|
||||||
static inline constinit bool s_is_uninitialized = true;
|
static inline constinit bool s_is_initialized = false;
|
||||||
static inline constinit const volatile KTargetSystemData s_data = {
|
static inline constinit const volatile KTargetSystemData s_data = {
|
||||||
.is_not_debug_mode = false,
|
.is_debug_mode = true,
|
||||||
.disable_debug_logging = false,
|
.enable_debug_logging = true,
|
||||||
.disable_user_exception_handlers = false,
|
.enable_user_exception_handlers = true,
|
||||||
.disable_debug_memory_fill = false,
|
.enable_debug_memory_fill = true,
|
||||||
.disable_user_pmu_access = false,
|
.enable_user_pmu_access = true,
|
||||||
.disable_kernel_debugging = false,
|
.enable_kernel_debugging = true,
|
||||||
.disable_dynamic_resource_limits = true,
|
.enable_dynamic_resource_limits = false,
|
||||||
};
|
};
|
||||||
private:
|
private:
|
||||||
static ALWAYS_INLINE void SetInitialized() { s_is_uninitialized = false; }
|
static ALWAYS_INLINE void SetInitialized() { s_is_initialized = true; }
|
||||||
public:
|
public:
|
||||||
static ALWAYS_INLINE bool IsDebugMode() { return !(s_is_uninitialized | s_data.is_not_debug_mode); }
|
static ALWAYS_INLINE bool IsDebugMode() { return s_is_initialized && s_data.is_debug_mode; }
|
||||||
static ALWAYS_INLINE bool IsDebugLoggingEnabled() { return !(s_is_uninitialized | s_data.disable_debug_logging); }
|
static ALWAYS_INLINE bool IsDebugLoggingEnabled() { return s_is_initialized && s_data.enable_debug_logging; }
|
||||||
static ALWAYS_INLINE bool IsUserExceptionHandlersEnabled() { return !(s_is_uninitialized | s_data.disable_user_exception_handlers); }
|
static ALWAYS_INLINE bool IsUserExceptionHandlersEnabled() { return s_is_initialized && s_data.enable_user_exception_handlers; }
|
||||||
static ALWAYS_INLINE bool IsDebugMemoryFillEnabled() { return !(s_is_uninitialized | s_data.disable_debug_memory_fill); }
|
static ALWAYS_INLINE bool IsDebugMemoryFillEnabled() { return s_is_initialized && s_data.enable_debug_memory_fill; }
|
||||||
static ALWAYS_INLINE bool IsUserPmuAccessEnabled() { return !(s_is_uninitialized | s_data.disable_user_pmu_access); }
|
static ALWAYS_INLINE bool IsUserPmuAccessEnabled() { return s_is_initialized && s_data.enable_user_pmu_access; }
|
||||||
static ALWAYS_INLINE bool IsKernelDebuggingEnabled() { return !(s_is_uninitialized | s_data.disable_kernel_debugging); }
|
static ALWAYS_INLINE bool IsKernelDebuggingEnabled() { return s_is_initialized && s_data.enable_kernel_debugging; }
|
||||||
static ALWAYS_INLINE bool IsDynamicResourceLimitsEnabled() { return !(s_is_uninitialized | s_data.disable_dynamic_resource_limits); }
|
static ALWAYS_INLINE bool IsDynamicResourceLimitsEnabled() { return s_is_initialized && s_data.enable_dynamic_resource_limits; }
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -105,7 +105,7 @@ namespace ams::kern {
|
|||||||
util::Atomic<u8> dpc_flags;
|
util::Atomic<u8> dpc_flags;
|
||||||
u8 current_svc_id;
|
u8 current_svc_id;
|
||||||
u8 reserved_2c;
|
u8 reserved_2c;
|
||||||
util::Atomic<u8> exception_flags;
|
u8 exception_flags;
|
||||||
bool is_pinned;
|
bool is_pinned;
|
||||||
u8 reserved_2f;
|
u8 reserved_2f;
|
||||||
u8 reserved_30[0x10];
|
u8 reserved_30[0x10];
|
||||||
@@ -417,17 +417,17 @@ namespace ams::kern {
|
|||||||
private:
|
private:
|
||||||
ALWAYS_INLINE void SetExceptionFlag(ExceptionFlag flag) {
|
ALWAYS_INLINE void SetExceptionFlag(ExceptionFlag flag) {
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
this->GetStackParameters().exception_flags.FetchOr<std::memory_order_relaxed>(flag);
|
this->GetStackParameters().exception_flags |= flag;
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE void ClearExceptionFlag(ExceptionFlag flag) {
|
ALWAYS_INLINE void ClearExceptionFlag(ExceptionFlag flag) {
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
this->GetStackParameters().exception_flags.FetchAnd<std::memory_order_relaxed>(~flag);
|
this->GetStackParameters().exception_flags &= ~flag;
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE bool IsExceptionFlagSet(ExceptionFlag flag) const {
|
ALWAYS_INLINE bool IsExceptionFlagSet(ExceptionFlag flag) const {
|
||||||
MESOSPHERE_ASSERT_THIS();
|
MESOSPHERE_ASSERT_THIS();
|
||||||
return this->GetStackParameters().exception_flags.Load<std::memory_order_relaxed>() & flag;
|
return this->GetStackParameters().exception_flags & flag;
|
||||||
}
|
}
|
||||||
public:
|
public:
|
||||||
/* ALWAYS_INLINE void SetCallingSvc() { return this->SetExceptionFlag(ExceptionFlag_IsCallingSvc); } */
|
/* ALWAYS_INLINE void SetCallingSvc() { return this->SetExceptionFlag(ExceptionFlag_IsCallingSvc); } */
|
||||||
@@ -523,7 +523,7 @@ namespace ams::kern {
|
|||||||
Result GetCoreMask(int32_t *out_ideal_core, u64 *out_affinity_mask);
|
Result GetCoreMask(int32_t *out_ideal_core, u64 *out_affinity_mask);
|
||||||
Result SetCoreMask(int32_t ideal_core, u64 affinity_mask);
|
Result SetCoreMask(int32_t ideal_core, u64 affinity_mask);
|
||||||
|
|
||||||
void GetPhysicalCoreMask(int32_t *out_ideal_core, u64 *out_affinity_mask);
|
Result GetPhysicalCoreMask(int32_t *out_ideal_core, u64 *out_affinity_mask);
|
||||||
|
|
||||||
constexpr ThreadState GetState() const { return static_cast<ThreadState>(m_thread_state & ThreadState_Mask); }
|
constexpr ThreadState GetState() const { return static_cast<ThreadState>(m_thread_state & ThreadState_Mask); }
|
||||||
constexpr ThreadState GetRawState() const { return m_thread_state; }
|
constexpr ThreadState GetRawState() const { return m_thread_state; }
|
||||||
@@ -717,7 +717,7 @@ namespace ams::kern {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void SetBasePriority(s32 priority);
|
void SetBasePriority(s32 priority);
|
||||||
void SetPriorityToIdle();
|
Result SetPriorityToIdle();
|
||||||
|
|
||||||
Result Run();
|
Result Run();
|
||||||
void Exit();
|
void Exit();
|
||||||
@@ -725,7 +725,7 @@ namespace ams::kern {
|
|||||||
Result Terminate();
|
Result Terminate();
|
||||||
ThreadState RequestTerminate();
|
ThreadState RequestTerminate();
|
||||||
|
|
||||||
void Sleep(s64 timeout);
|
Result Sleep(s64 timeout);
|
||||||
|
|
||||||
ALWAYS_INLINE void *GetStackTop() const { return reinterpret_cast<StackParameters *>(m_kernel_stack_top) - 1; }
|
ALWAYS_INLINE void *GetStackTop() const { return reinterpret_cast<StackParameters *>(m_kernel_stack_top) - 1; }
|
||||||
ALWAYS_INLINE void *GetKernelStackTop() const { return m_kernel_stack_top; }
|
ALWAYS_INLINE void *GetKernelStackTop() const { return m_kernel_stack_top; }
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ namespace ams::kern {
|
|||||||
}
|
}
|
||||||
public:
|
public:
|
||||||
Result Initialize(KProcess *process);
|
Result Initialize(KProcess *process);
|
||||||
void Finalize();
|
Result Finalize();
|
||||||
|
|
||||||
KProcessAddress Reserve();
|
KProcessAddress Reserve();
|
||||||
void Release(KProcessAddress addr);
|
void Release(KProcessAddress addr);
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ clean:
|
|||||||
@echo clean $(ATMOSPHERE_BUILD_NAME) ...
|
@echo clean $(ATMOSPHERE_BUILD_NAME) ...
|
||||||
@rm -fr $(ATMOSPHERE_BUILD_DIR) $(ATMOSPHERE_OUT_DIR)
|
@rm -fr $(ATMOSPHERE_BUILD_DIR) $(ATMOSPHERE_OUT_DIR)
|
||||||
@rm -fr $(foreach hdr,$(GCH_DIRS),$(hdr)/$(ATMOSPHERE_GCH_IDENTIFIER))
|
@rm -fr $(foreach hdr,$(GCH_DIRS),$(hdr)/$(ATMOSPHERE_GCH_IDENTIFIER))
|
||||||
@for i in $(GCH_DIRS); do [ -d $$i ] && rmdir $$i 2>/dev/null || true; done
|
@for i in $(GCH_DIRS); do [ -d $$i ] && rmdir --ignore-fail-on-non-empty $$i || true; done
|
||||||
|
|
||||||
$(ATMOSPHERE_LIBRARY_DIR) $(ATMOSPHERE_BUILD_DIR) $(GCH_DIRS):
|
$(ATMOSPHERE_LIBRARY_DIR) $(ATMOSPHERE_BUILD_DIR) $(GCH_DIRS):
|
||||||
@[ -d $@ ] || mkdir -p $@
|
@[ -d $@ ] || mkdir -p $@
|
||||||
|
|||||||
@@ -113,11 +113,6 @@ namespace ams::kern::arch::arm {
|
|||||||
constexpr size_t Offset = 0;
|
constexpr size_t Offset = 0;
|
||||||
state->icfgr[i] = m_gicd->icfgr[i + Offset];
|
state->icfgr[i] = m_gicd->icfgr[i + Offset];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Save spendsgir. */
|
|
||||||
for (size_t i = 0; i < util::size(state->spendsgir); ++i) {
|
|
||||||
state->spendsgir[i] = m_gicd->spendsgir[i];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptController::SaveGlobal(GlobalState *state) const {
|
void KInterruptController::SaveGlobal(GlobalState *state) const {
|
||||||
@@ -173,11 +168,6 @@ namespace ams::kern::arch::arm {
|
|||||||
m_gicd->icenabler[i + Offset] = 0xFFFFFFFF;
|
m_gicd->icenabler[i + Offset] = 0xFFFFFFFF;
|
||||||
m_gicd->isenabler[i + Offset] = state->isenabler[i];
|
m_gicd->isenabler[i + Offset] = state->isenabler[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Restore spendsgir. */
|
|
||||||
for (size_t i = 0; i < util::size(state->spendsgir); ++i) {
|
|
||||||
m_gicd->spendsgir[i] = state->spendsgir[i];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptController::RestoreGlobal(const GlobalState *state) const {
|
void KInterruptController::RestoreGlobal(const GlobalState *state) const {
|
||||||
|
|||||||
@@ -112,7 +112,7 @@ namespace ams::kern::arch::arm64::cpu {
|
|||||||
class KCoreBarrierInterruptHandler : public KInterruptHandler {
|
class KCoreBarrierInterruptHandler : public KInterruptHandler {
|
||||||
private:
|
private:
|
||||||
util::Atomic<u64> m_target_cores;
|
util::Atomic<u64> m_target_cores;
|
||||||
KLightLock m_lock;
|
KSpinLock m_lock;
|
||||||
public:
|
public:
|
||||||
constexpr KCoreBarrierInterruptHandler() : KInterruptHandler(), m_target_cores(0), m_lock() { /* ... */ }
|
constexpr KCoreBarrierInterruptHandler() : KInterruptHandler(), m_target_cores(0), m_lock() { /* ... */ }
|
||||||
|
|
||||||
@@ -123,8 +123,11 @@ namespace ams::kern::arch::arm64::cpu {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void SynchronizeCores(u64 core_mask) {
|
void SynchronizeCores(u64 core_mask) {
|
||||||
|
/* Disable dispatch while we synchronize. */
|
||||||
|
KScopedDisableDispatch dd;
|
||||||
|
|
||||||
/* Acquire exclusive access to ourselves. */
|
/* Acquire exclusive access to ourselves. */
|
||||||
KScopedLightLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
/* If necessary, force synchronization with other cores. */
|
/* If necessary, force synchronization with other cores. */
|
||||||
if (const u64 other_cores_mask = core_mask & ~(1ul << GetCurrentCoreId()); other_cores_mask != 0) {
|
if (const u64 other_cores_mask = core_mask & ~(1ul << GetCurrentCoreId()); other_cores_mask != 0) {
|
||||||
@@ -215,7 +218,7 @@ namespace ams::kern::arch::arm64::cpu {
|
|||||||
KThread::Register(new_thread);
|
KThread::Register(new_thread);
|
||||||
|
|
||||||
/* Run the thread. */
|
/* Run the thread. */
|
||||||
MESOSPHERE_R_ABORT_UNLESS(new_thread->Run());
|
new_thread->Run();
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual KInterruptTask *OnInterrupt(s32 interrupt_id) override {
|
virtual KInterruptTask *OnInterrupt(s32 interrupt_id) override {
|
||||||
@@ -508,16 +511,16 @@ namespace ams::kern::arch::arm64::cpu {
|
|||||||
g_cache_operation_handler.Initialize(core_id);
|
g_cache_operation_handler.Initialize(core_id);
|
||||||
|
|
||||||
/* Bind all handlers to the relevant interrupts. */
|
/* Bind all handlers to the relevant interrupts. */
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(std::addressof(g_cache_operation_handler), KInterruptName_CacheOperation, core_id, KInterruptController::PriorityLevel_High, false, false));
|
Kernel::GetInterruptManager().BindHandler(std::addressof(g_cache_operation_handler), KInterruptName_CacheOperation, core_id, KInterruptController::PriorityLevel_High, false, false);
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(std::addressof(g_thread_termination_handler), KInterruptName_ThreadTerminate, core_id, KInterruptController::PriorityLevel_Scheduler, false, false));
|
Kernel::GetInterruptManager().BindHandler(std::addressof(g_thread_termination_handler), KInterruptName_ThreadTerminate, core_id, KInterruptController::PriorityLevel_Scheduler, false, false);
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(std::addressof(g_core_barrier_handler), KInterruptName_CoreBarrier, core_id, KInterruptController::PriorityLevel_Scheduler, false, false));
|
Kernel::GetInterruptManager().BindHandler(std::addressof(g_core_barrier_handler), KInterruptName_CoreBarrier, core_id, KInterruptController::PriorityLevel_Scheduler, false, false);
|
||||||
|
|
||||||
/* If we should, enable user access to the performance counter registers. */
|
/* If we should, enable user access to the performance counter registers. */
|
||||||
if (KTargetSystem::IsUserPmuAccessEnabled()) { SetPmUserEnrEl0(1ul); }
|
if (KTargetSystem::IsUserPmuAccessEnabled()) { SetPmUserEnrEl0(1ul); }
|
||||||
|
|
||||||
/* If we should, enable the kernel performance counter interrupt handler. */
|
/* If we should, enable the kernel performance counter interrupt handler. */
|
||||||
#if defined(MESOSPHERE_ENABLE_PERFORMANCE_COUNTER)
|
#if defined(MESOSPHERE_ENABLE_PERFORMANCE_COUNTER)
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(std::addressof(g_performance_counter_handler[core_id]), KInterruptName_PerformanceCounter, core_id, KInterruptController::PriorityLevel_Timer, false, false));
|
Kernel::GetInterruptManager().BindHandler(std::addressof(g_performance_counter_handler[core_id]), KInterruptName_PerformanceCounter, core_id, KInterruptController::PriorityLevel_Timer, false, false);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -109,31 +109,12 @@ namespace ams::kern::arch::arm64 {
|
|||||||
return insn;
|
return insn;
|
||||||
}
|
}
|
||||||
|
|
||||||
void HandleUserException(KExceptionContext *context, u64 raw_esr, u64 raw_far, u64 afsr0, u64 afsr1, u32 data) {
|
void HandleUserException(KExceptionContext *context, u64 esr, u64 far, u64 afsr0, u64 afsr1, u32 data) {
|
||||||
/* Pre-process exception registers as needed. */
|
|
||||||
u64 esr = raw_esr;
|
|
||||||
u64 far = raw_far;
|
|
||||||
const u64 ec = (esr >> 26) & 0x3F;
|
|
||||||
if (ec == EsrEc_InstructionAbortEl0 || ec == EsrEc_DataAbortEl0) {
|
|
||||||
/* Adjust registers if a synchronous external abort has occurred with far not valid. */
|
|
||||||
/* Mask 0x03F = Low 6 bits IFSC == 0x10: "Synchronous External abort, */
|
|
||||||
/* not on translation table walk or hardware update of translation table. */
|
|
||||||
/* Mask 0x400 = FnV = "FAR Not Valid" */
|
|
||||||
/* TODO: How would we perform this check using named register accesses? */
|
|
||||||
if ((esr & 0x43F) == 0x410) {
|
|
||||||
/* Clear the faulting register on memory tagging exception. */
|
|
||||||
far = 0;
|
|
||||||
} else {
|
|
||||||
/* If the faulting address is a kernel address, set ISFC = 4. */
|
|
||||||
if (far >= ams::svc::AddressMemoryRegion39Size) {
|
|
||||||
esr = (esr & 0xFFFFFFC0) | 4;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
KProcess &cur_process = GetCurrentProcess();
|
KProcess &cur_process = GetCurrentProcess();
|
||||||
bool should_process_user_exception = KTargetSystem::IsUserExceptionHandlersEnabled();
|
bool should_process_user_exception = KTargetSystem::IsUserExceptionHandlersEnabled();
|
||||||
|
|
||||||
|
const u64 ec = (esr >> 26) & 0x3F;
|
||||||
|
|
||||||
/* In the event that we return from this exception, we want SPSR.SS set so that we advance an instruction if single-stepping. */
|
/* In the event that we return from this exception, we want SPSR.SS set so that we advance an instruction if single-stepping. */
|
||||||
#if defined(MESOSPHERE_ENABLE_HARDWARE_SINGLE_STEP)
|
#if defined(MESOSPHERE_ENABLE_HARDWARE_SINGLE_STEP)
|
||||||
context->psr |= (1ul << 21);
|
context->psr |= (1ul << 21);
|
||||||
@@ -184,7 +165,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Save the debug parameters to the current thread. */
|
/* Save the debug parameters to the current thread. */
|
||||||
GetCurrentThread().SaveDebugParams(raw_far, raw_esr, data);
|
GetCurrentThread().SaveDebugParams(far, esr, data);
|
||||||
|
|
||||||
/* Get the exception type. */
|
/* Get the exception type. */
|
||||||
u32 type;
|
u32 type;
|
||||||
@@ -406,6 +387,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
ams::svc::aarch32::ExceptionInfo info32;
|
ams::svc::aarch32::ExceptionInfo info32;
|
||||||
} info = {};
|
} info = {};
|
||||||
|
|
||||||
|
|
||||||
const bool is_aarch64 = (e_ctx->psr & 0x10) == 0;
|
const bool is_aarch64 = (e_ctx->psr & 0x10) == 0;
|
||||||
if (is_aarch64) {
|
if (is_aarch64) {
|
||||||
/* We're 64-bit. */
|
/* We're 64-bit. */
|
||||||
@@ -450,28 +432,9 @@ namespace ams::kern::arch::arm64 {
|
|||||||
uintptr_t far, esr, data;
|
uintptr_t far, esr, data;
|
||||||
GetCurrentThread().RestoreDebugParams(std::addressof(far), std::addressof(esr), std::addressof(data));
|
GetCurrentThread().RestoreDebugParams(std::addressof(far), std::addressof(esr), std::addressof(data));
|
||||||
|
|
||||||
/* Pre-process exception registers as needed. */
|
|
||||||
const u64 ec = (esr >> 26) & 0x3F;
|
|
||||||
if (ec == EsrEc_InstructionAbortEl0 || ec == EsrEc_DataAbortEl0) {
|
|
||||||
/* Adjust registers if a synchronous external abort has occurred with far not valid. */
|
|
||||||
/* Mask 0x03F = Low 6 bits IFSC == 0x10: "Synchronous External abort, */
|
|
||||||
/* not on translation table walk or hardware update of translation table. */
|
|
||||||
/* Mask 0x400 = FnV = "FAR Not Valid" */
|
|
||||||
/* TODO: How would we perform this check using named register accesses? */
|
|
||||||
if ((esr & 0x43F) == 0x410) {
|
|
||||||
/* Clear the faulting register on memory tagging exception. */
|
|
||||||
far = 0;
|
|
||||||
} else {
|
|
||||||
/* If the faulting address is a kernel address, set ISFC = 4. */
|
|
||||||
if (far >= ams::svc::AddressMemoryRegion39Size) {
|
|
||||||
esr = (esr & 0xFFFFFFC0) | 4;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Collect additional information based on the ec. */
|
/* Collect additional information based on the ec. */
|
||||||
uintptr_t params[3] = {};
|
uintptr_t params[3] = {};
|
||||||
switch (ec) {
|
switch ((esr >> 26) & 0x3F) {
|
||||||
case EsrEc_Unknown:
|
case EsrEc_Unknown:
|
||||||
case EsrEc_IllegalExecution:
|
case EsrEc_IllegalExecution:
|
||||||
case EsrEc_BkptInstruction:
|
case EsrEc_BkptInstruction:
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
m_maximum_time = static_cast<s64>(std::min<u64>(std::numeric_limits<s64>::max(), cpu::CounterTimerPhysicalTimerCompareValueRegisterAccessor().GetCompareValue()));
|
m_maximum_time = static_cast<s64>(std::min<u64>(std::numeric_limits<s64>::max(), cpu::CounterTimerPhysicalTimerCompareValueRegisterAccessor().GetCompareValue()));
|
||||||
|
|
||||||
/* Bind the interrupt task for this core. */
|
/* Bind the interrupt task for this core. */
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(this, KInterruptName_NonSecurePhysicalTimer, GetCurrentCoreId(), KInterruptController::PriorityLevel_Timer, true, true));
|
Kernel::GetInterruptManager().BindHandler(this, KInterruptName_NonSecurePhysicalTimer, GetCurrentCoreId(), KInterruptController::PriorityLevel_Timer, true, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void KHardwareTimer::Finalize() {
|
void KHardwareTimer::Finalize() {
|
||||||
|
|||||||
@@ -126,7 +126,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
if (entry.handler != nullptr) {
|
if (entry.handler != nullptr) {
|
||||||
/* Set manual clear needed if relevant. */
|
/* Set manual clear needed if relevant. */
|
||||||
if (entry.manually_cleared) {
|
if (entry.manually_cleared) {
|
||||||
m_interrupt_controller.Disable(irq);
|
m_interrupt_controller.SetPriorityLevel(irq, KInterruptController::PriorityLevel_Low);
|
||||||
entry.needs_clear = true;
|
entry.needs_clear = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -242,40 +242,40 @@ namespace ams::kern::arch::arm64 {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::UnbindHandler(s32 irq, s32 core_id) {
|
Result KInterruptManager::UnbindHandler(s32 irq, s32 core_id) {
|
||||||
MESOSPHERE_UNUSED(core_id);
|
MESOSPHERE_UNUSED(core_id);
|
||||||
|
|
||||||
MESOSPHERE_ASSERT(KInterruptController::IsGlobal(irq) || KInterruptController::IsLocal(irq));
|
R_UNLESS(KInterruptController::IsGlobal(irq) || KInterruptController::IsLocal(irq), svc::ResultOutOfRange());
|
||||||
|
|
||||||
|
|
||||||
if (KInterruptController::IsGlobal(irq)) {
|
if (KInterruptController::IsGlobal(irq)) {
|
||||||
KScopedInterruptDisable di;
|
KScopedInterruptDisable di;
|
||||||
|
|
||||||
KScopedSpinLock lk(this->GetGlobalInterruptLock());
|
KScopedSpinLock lk(this->GetGlobalInterruptLock());
|
||||||
return this->UnbindGlobal(irq);
|
R_RETURN(this->UnbindGlobal(irq));
|
||||||
} else if (KInterruptController::IsLocal(irq)) {
|
} else {
|
||||||
MESOSPHERE_ASSERT(core_id == GetCurrentCoreId());
|
MESOSPHERE_ASSERT(core_id == GetCurrentCoreId());
|
||||||
|
|
||||||
KScopedInterruptDisable di;
|
KScopedInterruptDisable di;
|
||||||
return this->UnbindLocal(irq);
|
R_RETURN(this->UnbindLocal(irq));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::ClearInterrupt(s32 irq, s32 core_id) {
|
Result KInterruptManager::ClearInterrupt(s32 irq, s32 core_id) {
|
||||||
MESOSPHERE_UNUSED(core_id);
|
MESOSPHERE_UNUSED(core_id);
|
||||||
|
|
||||||
MESOSPHERE_ASSERT(KInterruptController::IsGlobal(irq) || KInterruptController::IsLocal(irq));
|
R_UNLESS(KInterruptController::IsGlobal(irq) || KInterruptController::IsLocal(irq), svc::ResultOutOfRange());
|
||||||
|
|
||||||
|
|
||||||
if (KInterruptController::IsGlobal(irq)) {
|
if (KInterruptController::IsGlobal(irq)) {
|
||||||
KScopedInterruptDisable di;
|
KScopedInterruptDisable di;
|
||||||
KScopedSpinLock lk(this->GetGlobalInterruptLock());
|
KScopedSpinLock lk(this->GetGlobalInterruptLock());
|
||||||
return this->ClearGlobal(irq);
|
R_RETURN(this->ClearGlobal(irq));
|
||||||
} else if (KInterruptController::IsLocal(irq)) {
|
} else {
|
||||||
MESOSPHERE_ASSERT(core_id == GetCurrentCoreId());
|
MESOSPHERE_ASSERT(core_id == GetCurrentCoreId());
|
||||||
|
|
||||||
KScopedInterruptDisable di;
|
KScopedInterruptDisable di;
|
||||||
return this->ClearLocal(irq);
|
R_RETURN(this->ClearLocal(irq));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -332,7 +332,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::UnbindGlobal(s32 irq) {
|
Result KInterruptManager::UnbindGlobal(s32 irq) {
|
||||||
for (size_t core_id = 0; core_id < cpu::NumCores; core_id++) {
|
for (size_t core_id = 0; core_id < cpu::NumCores; core_id++) {
|
||||||
m_interrupt_controller.ClearTarget(irq, static_cast<s32>(core_id));
|
m_interrupt_controller.ClearTarget(irq, static_cast<s32>(core_id));
|
||||||
}
|
}
|
||||||
@@ -340,35 +340,50 @@ namespace ams::kern::arch::arm64 {
|
|||||||
m_interrupt_controller.Disable(irq);
|
m_interrupt_controller.Disable(irq);
|
||||||
|
|
||||||
GetGlobalInterruptEntry(irq).handler = nullptr;
|
GetGlobalInterruptEntry(irq).handler = nullptr;
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::UnbindLocal(s32 irq) {
|
Result KInterruptManager::UnbindLocal(s32 irq) {
|
||||||
|
auto &entry = this->GetLocalInterruptEntry(irq);
|
||||||
|
R_UNLESS(entry.handler != nullptr, svc::ResultInvalidState());
|
||||||
|
|
||||||
m_interrupt_controller.SetPriorityLevel(irq, KInterruptController::PriorityLevel_Low);
|
m_interrupt_controller.SetPriorityLevel(irq, KInterruptController::PriorityLevel_Low);
|
||||||
m_interrupt_controller.Disable(irq);
|
m_interrupt_controller.Disable(irq);
|
||||||
|
|
||||||
this->GetLocalInterruptEntry(irq).handler = nullptr;
|
entry.handler = nullptr;
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::ClearGlobal(s32 irq) {
|
Result KInterruptManager::ClearGlobal(s32 irq) {
|
||||||
/* Get the entry. */
|
/* We can't clear an entry with no handler. */
|
||||||
auto &entry = GetGlobalInterruptEntry(irq);
|
auto &entry = GetGlobalInterruptEntry(irq);
|
||||||
|
R_UNLESS(entry.handler != nullptr, svc::ResultInvalidState());
|
||||||
|
|
||||||
/* If not auto-cleared, clear and enable. */
|
/* If auto-cleared, we can succeed immediately. */
|
||||||
if (entry.manually_cleared && entry.needs_clear) {
|
R_SUCCEED_IF(!entry.manually_cleared);
|
||||||
entry.needs_clear = false;
|
R_SUCCEED_IF(!entry.needs_clear);
|
||||||
m_interrupt_controller.Enable(irq);
|
|
||||||
}
|
/* Clear and enable. */
|
||||||
|
entry.needs_clear = false;
|
||||||
|
m_interrupt_controller.Enable(irq);
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KInterruptManager::ClearLocal(s32 irq) {
|
Result KInterruptManager::ClearLocal(s32 irq) {
|
||||||
/* Get the entry. */
|
/* We can't clear an entry with no handler. */
|
||||||
auto &entry = this->GetLocalInterruptEntry(irq);
|
auto &entry = this->GetLocalInterruptEntry(irq);
|
||||||
|
R_UNLESS(entry.handler != nullptr, svc::ResultInvalidState());
|
||||||
|
|
||||||
/* If not auto-cleared, clear and enable. */
|
/* If auto-cleared, we can succeed immediately. */
|
||||||
if (entry.manually_cleared && entry.needs_clear) {
|
R_SUCCEED_IF(!entry.manually_cleared);
|
||||||
entry.needs_clear = false;
|
R_SUCCEED_IF(!entry.needs_clear);
|
||||||
m_interrupt_controller.Enable(irq);
|
|
||||||
}
|
/* Clear and set priority. */
|
||||||
|
entry.needs_clear = false;
|
||||||
|
m_interrupt_controller.SetPriorityLevel(irq, entry.priority);
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -119,13 +119,15 @@ namespace ams::kern::arch::arm64 {
|
|||||||
MESOSPHERE_UNUSED(core_id);
|
MESOSPHERE_UNUSED(core_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void KPageTable::InitializeForKernel(void *table, KVirtualAddress start, KVirtualAddress end) {
|
Result KPageTable::InitializeForKernel(void *table, KVirtualAddress start, KVirtualAddress end) {
|
||||||
/* Initialize basic fields. */
|
/* Initialize basic fields. */
|
||||||
m_asid = 0;
|
m_asid = 0;
|
||||||
m_manager = Kernel::GetSystemSystemResource().GetPageTableManagerPointer();
|
m_manager = Kernel::GetSystemSystemResource().GetPageTableManagerPointer();
|
||||||
|
|
||||||
/* Initialize the base page table. */
|
/* Initialize the base page table. */
|
||||||
KPageTableBase::InitializeForKernel(true, table, start, end);
|
MESOSPHERE_R_ABORT_UNLESS(KPageTableBase::InitializeForKernel(true, table, start, end));
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KPageTable::InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit, size_t process_index) {
|
Result KPageTable::InitializeForProcess(ams::svc::CreateProcessFlag flags, bool from_back, KMemoryManager::Pool pool, KProcessAddress code_address, size_t code_size, KSystemResource *system_resource, KResourceLimit *resource_limit, size_t process_index) {
|
||||||
@@ -151,7 +153,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KPageTable::Finalize() {
|
Result KPageTable::Finalize() {
|
||||||
/* Only process tables should be finalized. */
|
/* Only process tables should be finalized. */
|
||||||
MESOSPHERE_ASSERT(!this->IsKernel());
|
MESOSPHERE_ASSERT(!this->IsKernel());
|
||||||
|
|
||||||
@@ -269,6 +271,8 @@ namespace ams::kern::arch::arm64 {
|
|||||||
/* Perform inherited finalization. */
|
/* Perform inherited finalization. */
|
||||||
KPageTableBase::Finalize();
|
KPageTableBase::Finalize();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KPageTable::OperateImpl(PageLinkedList *page_list, KProcessAddress virt_addr, size_t num_pages, KPhysicalAddress phys_addr, bool is_pa_valid, const KPageProperties properties, OperationType operation, bool reuse_ll) {
|
Result KPageTable::OperateImpl(PageLinkedList *page_list, KProcessAddress virt_addr, size_t num_pages, KPhysicalAddress phys_addr, bool is_pa_valid, const KPageProperties properties, OperationType operation, bool reuse_ll) {
|
||||||
@@ -748,10 +752,13 @@ namespace ams::kern::arch::arm64 {
|
|||||||
while (true) {
|
while (true) {
|
||||||
/* Try to merge. */
|
/* Try to merge. */
|
||||||
KVirtualAddress freed_table = Null<KVirtualAddress>;
|
KVirtualAddress freed_table = Null<KVirtualAddress>;
|
||||||
if (!impl.MergePages(std::addressof(freed_table), context, &KPageTable::NoteUpdatedCallback, this)) {
|
if (!impl.MergePages(std::addressof(freed_table), context)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Note that we updated. */
|
||||||
|
this->NoteUpdated();
|
||||||
|
|
||||||
/* Free the page. */
|
/* Free the page. */
|
||||||
if (freed_table != Null<KVirtualAddress>) {
|
if (freed_table != Null<KVirtualAddress>) {
|
||||||
ClearPageTable(freed_table);
|
ClearPageTable(freed_table);
|
||||||
@@ -809,7 +816,8 @@ namespace ams::kern::arch::arm64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Separate. */
|
/* Separate. */
|
||||||
impl.SeparatePages(entry, context, virt_addr, GetPointer<PageTableEntry>(table), &KPageTable::NoteUpdatedCallback, this);
|
impl.SeparatePages(entry, context, virt_addr, GetPointer<PageTableEntry>(table));
|
||||||
|
this->NoteUpdated();
|
||||||
}
|
}
|
||||||
|
|
||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
@@ -938,7 +946,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
/* If we should flush entries, do so. */
|
/* If we should flush entries, do so. */
|
||||||
if ((apply_option & ApplyOption_FlushDataCache) != 0) {
|
if ((apply_option & ApplyOption_FlushDataCache) != 0) {
|
||||||
if (IsHeapPhysicalAddress(next_entry.phys_addr)) {
|
if (IsHeapPhysicalAddress(next_entry.phys_addr)) {
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::FlushDataCache(GetVoidPointer(GetHeapVirtualAddress(next_entry.phys_addr)), next_entry.block_size));
|
cpu::FlushDataCache(GetVoidPointer(GetHeapVirtualAddress(next_entry.phys_addr)), next_entry.block_size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1017,9 +1025,6 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
/* Finally, apply the changes as directed, flushing the mappings before they're applied (if we should). */
|
/* Finally, apply the changes as directed, flushing the mappings before they're applied (if we should). */
|
||||||
ApplyEntryTemplate(entry_template, flush_mapping ? ApplyOption_FlushDataCache : ApplyOption_None);
|
ApplyEntryTemplate(entry_template, flush_mapping ? ApplyOption_FlushDataCache : ApplyOption_None);
|
||||||
|
|
||||||
/* Wait for pending stores to complete. */
|
|
||||||
cpu::DataSynchronizationBarrierInnerShareableStore();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* We've succeeded, now perform what coalescing we can. */
|
/* We've succeeded, now perform what coalescing we can. */
|
||||||
|
|||||||
@@ -219,7 +219,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
return is_block;
|
return is_block;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool KPageTableImpl::MergePages(KVirtualAddress *out, TraversalContext *context, EntryUpdatedCallback on_entry_updated, const void *pt) {
|
bool KPageTableImpl::MergePages(KVirtualAddress *out, TraversalContext *context) {
|
||||||
/* We want to upgrade the pages by one step. */
|
/* We want to upgrade the pages by one step. */
|
||||||
if (context->is_contiguous) {
|
if (context->is_contiguous) {
|
||||||
/* We can't merge an L1 table. */
|
/* We can't merge an L1 table. */
|
||||||
@@ -251,7 +251,6 @@ namespace ams::kern::arch::arm64 {
|
|||||||
const auto sw_reserved_bits = PageTableEntry::EncodeSoftwareReservedBits(head_pte->IsHeadMergeDisabled(), head_pte->IsHeadAndBodyMergeDisabled(), tail_pte->IsTailMergeDisabled());
|
const auto sw_reserved_bits = PageTableEntry::EncodeSoftwareReservedBits(head_pte->IsHeadMergeDisabled(), head_pte->IsHeadAndBodyMergeDisabled(), tail_pte->IsTailMergeDisabled());
|
||||||
|
|
||||||
*context->level_entries[context->level + 1] = PageTableEntry(PageTableEntry::BlockTag{}, phys_addr, PageTableEntry(entry_template), sw_reserved_bits, false, false);
|
*context->level_entries[context->level + 1] = PageTableEntry(PageTableEntry::BlockTag{}, phys_addr, PageTableEntry(entry_template), sw_reserved_bits, false, false);
|
||||||
on_entry_updated(pt);
|
|
||||||
|
|
||||||
/* Update our context. */
|
/* Update our context. */
|
||||||
context->is_contiguous = false;
|
context->is_contiguous = false;
|
||||||
@@ -286,7 +285,6 @@ namespace ams::kern::arch::arm64 {
|
|||||||
for (size_t i = 0; i < BlocksPerContiguousBlock; ++i) {
|
for (size_t i = 0; i < BlocksPerContiguousBlock; ++i) {
|
||||||
pte[i] = PageTableEntry(PageTableEntry::BlockTag{}, phys_addr + (i << (PageBits + LevelBits * context->level)), PageTableEntry(entry_template), sw_reserved_bits, true, context->level == EntryLevel_L3);
|
pte[i] = PageTableEntry(PageTableEntry::BlockTag{}, phys_addr + (i << (PageBits + LevelBits * context->level)), PageTableEntry(entry_template), sw_reserved_bits, true, context->level == EntryLevel_L3);
|
||||||
}
|
}
|
||||||
on_entry_updated(pt);
|
|
||||||
|
|
||||||
/* Update our context. */
|
/* Update our context. */
|
||||||
context->level_entries[context->level] = pte;
|
context->level_entries[context->level] = pte;
|
||||||
@@ -296,7 +294,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void KPageTableImpl::SeparatePages(TraversalEntry *entry, TraversalContext *context, KProcessAddress address, PageTableEntry *pte, EntryUpdatedCallback on_entry_updated, const void *pt) const {
|
void KPageTableImpl::SeparatePages(TraversalEntry *entry, TraversalContext *context, KProcessAddress address, PageTableEntry *pte) const {
|
||||||
/* We want to downgrade the pages by one step. */
|
/* We want to downgrade the pages by one step. */
|
||||||
if (context->is_contiguous) {
|
if (context->is_contiguous) {
|
||||||
/* We want to downgrade a contiguous mapping to a non-contiguous mapping. */
|
/* We want to downgrade a contiguous mapping to a non-contiguous mapping. */
|
||||||
@@ -307,7 +305,6 @@ namespace ams::kern::arch::arm64 {
|
|||||||
for (size_t i = 0; i < BlocksPerContiguousBlock; ++i) {
|
for (size_t i = 0; i < BlocksPerContiguousBlock; ++i) {
|
||||||
pte[i] = PageTableEntry(PageTableEntry::BlockTag{}, block + (i << (PageBits + LevelBits * context->level)), PageTableEntry(first->GetEntryTemplateForSeparateContiguous(i)), PageTableEntry::SeparateContiguousTag{});
|
pte[i] = PageTableEntry(PageTableEntry::BlockTag{}, block + (i << (PageBits + LevelBits * context->level)), PageTableEntry(first->GetEntryTemplateForSeparateContiguous(i)), PageTableEntry::SeparateContiguousTag{});
|
||||||
}
|
}
|
||||||
on_entry_updated(pt);
|
|
||||||
|
|
||||||
context->is_contiguous = false;
|
context->is_contiguous = false;
|
||||||
|
|
||||||
@@ -328,12 +325,12 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
/* Update the block entry to be a table entry. */
|
/* Update the block entry to be a table entry. */
|
||||||
*context->level_entries[context->level + 1] = PageTableEntry(PageTableEntry::TableTag{}, KPageTable::GetPageTablePhysicalAddress(KVirtualAddress(pte)), m_is_kernel, true, BlocksPerTable);
|
*context->level_entries[context->level + 1] = PageTableEntry(PageTableEntry::TableTag{}, KPageTable::GetPageTablePhysicalAddress(KVirtualAddress(pte)), m_is_kernel, true, BlocksPerTable);
|
||||||
on_entry_updated(pt);
|
|
||||||
|
|
||||||
context->level_entries[context->level] = pte + this->GetLevelIndex(address, context->level);
|
context->level_entries[context->level] = pte + this->GetLevelIndex(address, context->level);
|
||||||
}
|
}
|
||||||
|
|
||||||
entry->sw_reserved_bits = context->level_entries[context->level]->GetSoftwareReservedBits();
|
entry->sw_reserved_bits = 0;
|
||||||
entry->attr = 0;
|
entry->attr = 0;
|
||||||
entry->phys_addr = this->GetBlock(context->level_entries[context->level], context->level) + this->GetOffset(address, context->level);
|
entry->phys_addr = this->GetBlock(context->level_entries[context->level], context->level) + this->GetOffset(address, context->level);
|
||||||
entry->block_size = static_cast<size_t>(1) << (PageBits + LevelBits * context->level + 4 * context->is_contiguous);
|
entry->block_size = static_cast<size_t>(1) << (PageBits + LevelBits * context->level + 4 * context->is_contiguous);
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
KScopedInterruptEnable ei;
|
KScopedInterruptEnable ei;
|
||||||
|
|
||||||
const uintptr_t params[2] = { GetCurrentThread().GetId(), GetInteger(GetCurrentThread().GetThreadLocalRegionAddress()) };
|
const uintptr_t params[2] = { GetCurrentThread().GetId(), GetInteger(GetCurrentThread().GetThreadLocalRegionAddress()) };
|
||||||
static_cast<void>(KDebug::OnDebugEvent(ams::svc::DebugEvent_CreateThread, params, util::size(params)));
|
KDebug::OnDebugEvent(ams::svc::DebugEvent_CreateThread, params, util::size(params));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Handle any pending dpc. */
|
/* Handle any pending dpc. */
|
||||||
@@ -116,7 +116,7 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KThreadContext::Initialize(KVirtualAddress u_pc, KVirtualAddress k_sp, KVirtualAddress u_sp, uintptr_t arg, bool is_user, bool is_64_bit, bool is_main) {
|
Result KThreadContext::Initialize(KVirtualAddress u_pc, KVirtualAddress k_sp, KVirtualAddress u_sp, uintptr_t arg, bool is_user, bool is_64_bit, bool is_main) {
|
||||||
MESOSPHERE_ASSERT(k_sp != Null<KVirtualAddress>);
|
MESOSPHERE_ASSERT(k_sp != Null<KVirtualAddress>);
|
||||||
|
|
||||||
/* Ensure that the stack pointers are aligned. */
|
/* Ensure that the stack pointers are aligned. */
|
||||||
@@ -157,6 +157,8 @@ namespace ams::kern::arch::arm64 {
|
|||||||
|
|
||||||
/* Lock the context, if we're a main thread. */
|
/* Lock the context, if we're a main thread. */
|
||||||
m_locked = is_main;
|
m_locked = is_main;
|
||||||
|
|
||||||
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KThreadContext::SetArguments(uintptr_t arg0, uintptr_t arg1) {
|
void KThreadContext::SetArguments(uintptr_t arg0, uintptr_t arg1) {
|
||||||
|
|||||||
@@ -24,12 +24,12 @@ _ZN3ams4kern4arch5arm6432UserspaceAccessFunctionAreaBeginEv:
|
|||||||
|
|
||||||
/* ================ All Userspace Access Functions after this line. ================ */
|
/* ================ All Userspace Access Functions after this line. ================ */
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUser(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUser(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyMemoryFromUserEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess18CopyMemoryFromUserEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyMemoryFromUserEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess18CopyMemoryFromUserEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyMemoryFromUserEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess18CopyMemoryFromUserEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyMemoryFromUserEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess18CopyMemoryFromUserEPvPKvm:
|
||||||
/* Check if there's anything to copy. */
|
/* Check if there's anything to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 2f
|
b.eq 2f
|
||||||
@@ -48,12 +48,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyMemoryFromUserEPvPKvm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUserAligned32Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUserAligned32Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned32BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned32BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned32BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned32BitEPvPKvm:
|
||||||
/* Check if there are 0x40 bytes to copy */
|
/* Check if there are 0x40 bytes to copy */
|
||||||
cmp x2, #0x3F
|
cmp x2, #0x3F
|
||||||
b.ls 1f
|
b.ls 1f
|
||||||
@@ -72,7 +72,7 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPv
|
|||||||
add x0, x0, #0x40
|
add x0, x0, #0x40
|
||||||
add x1, x1, #0x40
|
add x1, x1, #0x40
|
||||||
sub x2, x2, #0x40
|
sub x2, x2, #0x40
|
||||||
b _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPvPKvm
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned32BitEPvPKvm
|
||||||
|
|
||||||
1: /* We have less than 0x40 bytes to copy. */
|
1: /* We have less than 0x40 bytes to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
@@ -87,12 +87,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned32BitEPv
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUserAligned64Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUserAligned64Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned64BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned64BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned64BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned64BitEPvPKvm:
|
||||||
/* Check if there are 0x40 bytes to copy */
|
/* Check if there are 0x40 bytes to copy */
|
||||||
cmp x2, #0x3F
|
cmp x2, #0x3F
|
||||||
b.ls 1f
|
b.ls 1f
|
||||||
@@ -111,7 +111,7 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPv
|
|||||||
add x0, x0, #0x40
|
add x0, x0, #0x40
|
||||||
add x1, x1, #0x40
|
add x1, x1, #0x40
|
||||||
sub x2, x2, #0x40
|
sub x2, x2, #0x40
|
||||||
b _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPvPKvm
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess30CopyMemoryFromUserAligned64BitEPvPKvm
|
||||||
|
|
||||||
1: /* We have less than 0x40 bytes to copy. */
|
1: /* We have less than 0x40 bytes to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
@@ -126,12 +126,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl30CopyMemoryFromUserAligned64BitEPv
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUserSize64Bit(void *dst, const void *src) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUserSize64Bit(void *dst, const void *src) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize64BitEPvPKv, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize64BitEPvPKv, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize64BitEPvPKv
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize64BitEPvPKv
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize64BitEPvPKv, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize64BitEPvPKv, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize64BitEPvPKv:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize64BitEPvPKv:
|
||||||
/* Just load and store a u64. */
|
/* Just load and store a u64. */
|
||||||
ldtr x2, [x1]
|
ldtr x2, [x1]
|
||||||
str x2, [x0]
|
str x2, [x0]
|
||||||
@@ -140,12 +140,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize64BitEPvPKv
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUserSize32Bit(void *dst, const void *src) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUserSize32Bit(void *dst, const void *src) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize32BitEPvPKv, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize32BitEPvPKv, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize32BitEPvPKv
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize32BitEPvPKv
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize32BitEPvPKv, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize32BitEPvPKv, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize32BitEPvPKv:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess27CopyMemoryFromUserSize32BitEPvPKv:
|
||||||
/* Just load and store a u32. */
|
/* Just load and store a u32. */
|
||||||
ldtr w2, [x1]
|
ldtr w2, [x1]
|
||||||
str w2, [x0]
|
str w2, [x0]
|
||||||
@@ -154,12 +154,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl27CopyMemoryFromUserSize32BitEPvPKv
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryFromUserSize32BitWithSupervisorAccess(void *dst, const void *src) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryFromUserSize32BitWithSupervisorAccessImpl(void *dst, const void *src) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl47CopyMemoryFromUserSize32BitWithSupervisorAccessEPvPKv, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess51CopyMemoryFromUserSize32BitWithSupervisorAccessImplEPvPKv, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl47CopyMemoryFromUserSize32BitWithSupervisorAccessEPvPKv
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess51CopyMemoryFromUserSize32BitWithSupervisorAccessImplEPvPKv
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl47CopyMemoryFromUserSize32BitWithSupervisorAccessEPvPKv, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess51CopyMemoryFromUserSize32BitWithSupervisorAccessImplEPvPKv, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl47CopyMemoryFromUserSize32BitWithSupervisorAccessEPvPKv:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess51CopyMemoryFromUserSize32BitWithSupervisorAccessImplEPvPKv:
|
||||||
/* Just load and store a u32. */
|
/* Just load and store a u32. */
|
||||||
/* NOTE: This is done with supervisor access permissions. */
|
/* NOTE: This is done with supervisor access permissions. */
|
||||||
ldr w2, [x1]
|
ldr w2, [x1]
|
||||||
@@ -169,12 +169,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl47CopyMemoryFromUserSize32BitWithSu
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyStringFromUser(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyStringFromUser(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyStringFromUserEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess18CopyStringFromUserEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyStringFromUserEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess18CopyStringFromUserEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyStringFromUserEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess18CopyStringFromUserEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyStringFromUserEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess18CopyStringFromUserEPvPKvm:
|
||||||
/* Check if there's anything to copy. */
|
/* Check if there's anything to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -204,12 +204,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18CopyStringFromUserEPvPKvm:
|
|||||||
mov x0, #0
|
mov x0, #0
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryToUser(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryToUser(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyMemoryToUserEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess16CopyMemoryToUserEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyMemoryToUserEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess16CopyMemoryToUserEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyMemoryToUserEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess16CopyMemoryToUserEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyMemoryToUserEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess16CopyMemoryToUserEPvPKvm:
|
||||||
/* Check if there's anything to copy. */
|
/* Check if there's anything to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 2f
|
b.eq 2f
|
||||||
@@ -228,12 +228,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyMemoryToUserEPvPKvm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryToUserAligned32Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryToUserAligned32Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned32BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned32BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned32BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned32BitEPvPKvm:
|
||||||
/* Check if there are 0x40 bytes to copy */
|
/* Check if there are 0x40 bytes to copy */
|
||||||
cmp x2, #0x3F
|
cmp x2, #0x3F
|
||||||
b.ls 1f
|
b.ls 1f
|
||||||
@@ -252,7 +252,7 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPK
|
|||||||
add x0, x0, #0x40
|
add x0, x0, #0x40
|
||||||
add x1, x1, #0x40
|
add x1, x1, #0x40
|
||||||
sub x2, x2, #0x40
|
sub x2, x2, #0x40
|
||||||
b _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPKvm
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned32BitEPvPKvm
|
||||||
|
|
||||||
1: /* We have less than 0x40 bytes to copy. */
|
1: /* We have less than 0x40 bytes to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
@@ -267,12 +267,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned32BitEPvPK
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryToUserAligned64Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryToUserAligned64Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned64BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned64BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned64BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned64BitEPvPKvm:
|
||||||
/* Check if there are 0x40 bytes to copy */
|
/* Check if there are 0x40 bytes to copy */
|
||||||
cmp x2, #0x3F
|
cmp x2, #0x3F
|
||||||
b.ls 1f
|
b.ls 1f
|
||||||
@@ -291,7 +291,7 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPK
|
|||||||
add x0, x0, #0x40
|
add x0, x0, #0x40
|
||||||
add x1, x1, #0x40
|
add x1, x1, #0x40
|
||||||
sub x2, x2, #0x40
|
sub x2, x2, #0x40
|
||||||
b _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPKvm
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess28CopyMemoryToUserAligned64BitEPvPKvm
|
||||||
|
|
||||||
1: /* We have less than 0x40 bytes to copy. */
|
1: /* We have less than 0x40 bytes to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
@@ -306,25 +306,26 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl28CopyMemoryToUserAligned64BitEPvPK
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyMemoryToUserSize32Bit(void *dst, u32 value) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyMemoryToUserSize32Bit(void *dst, const void *src) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25CopyMemoryToUserSize32BitEPvj, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess25CopyMemoryToUserSize32BitEPvPKv, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25CopyMemoryToUserSize32BitEPvj
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess25CopyMemoryToUserSize32BitEPvPKv
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25CopyMemoryToUserSize32BitEPvj, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess25CopyMemoryToUserSize32BitEPvPKv, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25CopyMemoryToUserSize32BitEPvj:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess25CopyMemoryToUserSize32BitEPvPKv:
|
||||||
/* Just store a u32. */
|
/* Just load and store a u32. */
|
||||||
sttr w1, [x0]
|
ldr w2, [x1]
|
||||||
|
sttr w2, [x0]
|
||||||
|
|
||||||
/* We're done. */
|
/* We're done. */
|
||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::CopyStringToUser(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::CopyStringToUser(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyStringToUserEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess16CopyStringToUserEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyStringToUserEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess16CopyStringToUserEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyStringToUserEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess16CopyStringToUserEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyStringToUserEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess16CopyStringToUserEPvPKvm:
|
||||||
/* Check if there's anything to copy. */
|
/* Check if there's anything to copy. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -354,12 +355,114 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16CopyStringToUserEPvPKvm:
|
|||||||
mov x0, #0
|
mov x0, #0
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::UpdateLockAtomic(u32 *out, u32 *address, u32 if_zero, u32 new_orr_mask) */
|
/* ams::kern::arch::arm64::UserspaceAccess::ClearMemory(void *dst, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess11ClearMemoryEPvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess11ClearMemoryEPvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess11ClearMemoryEPvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess11ClearMemoryEPvm:
|
||||||
|
/* Check if there's anything to clear. */
|
||||||
|
cmp x1, #0
|
||||||
|
b.eq 2f
|
||||||
|
|
||||||
|
/* Keep track of the last address. */
|
||||||
|
add x2, x0, x1
|
||||||
|
|
||||||
|
1: /* We're copying memory byte-by-byte. */
|
||||||
|
sttrb wzr, [x0]
|
||||||
|
add x0, x0, #1
|
||||||
|
cmp x0, x2
|
||||||
|
b.ne 1b
|
||||||
|
|
||||||
|
2: /* We're done. */
|
||||||
|
mov x0, #1
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ams::kern::arch::arm64::UserspaceAccess::ClearMemoryAligned32Bit(void *dst, size_t size) */
|
||||||
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned32BitEPvm, "ax", %progbits
|
||||||
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned32BitEPvm
|
||||||
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned32BitEPvm, %function
|
||||||
|
.balign 0x10
|
||||||
|
_ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned32BitEPvm:
|
||||||
|
/* Check if there are 0x40 bytes to clear. */
|
||||||
|
cmp x1, #0x3F
|
||||||
|
b.ls 2f
|
||||||
|
sttr xzr, [x0, #0x00]
|
||||||
|
sttr xzr, [x0, #0x08]
|
||||||
|
sttr xzr, [x0, #0x10]
|
||||||
|
sttr xzr, [x0, #0x18]
|
||||||
|
sttr xzr, [x0, #0x20]
|
||||||
|
sttr xzr, [x0, #0x28]
|
||||||
|
sttr xzr, [x0, #0x30]
|
||||||
|
sttr xzr, [x0, #0x38]
|
||||||
|
add x0, x0, #0x40
|
||||||
|
sub x1, x1, #0x40
|
||||||
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned32BitEPvm
|
||||||
|
|
||||||
|
1: /* We have less than 0x40 bytes to clear. */
|
||||||
|
cmp x1, #0
|
||||||
|
b.eq 2f
|
||||||
|
sttr wzr, [x0]
|
||||||
|
add x0, x0, #4
|
||||||
|
sub x1, x1, #4
|
||||||
|
b 1b
|
||||||
|
|
||||||
|
2: /* We're done. */
|
||||||
|
mov x0, #1
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ams::kern::arch::arm64::UserspaceAccess::ClearMemoryAligned64Bit(void *dst, size_t size) */
|
||||||
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned64BitEPvm, "ax", %progbits
|
||||||
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned64BitEPvm
|
||||||
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned64BitEPvm, %function
|
||||||
|
.balign 0x10
|
||||||
|
_ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned64BitEPvm:
|
||||||
|
/* Check if there are 0x40 bytes to clear. */
|
||||||
|
cmp x1, #0x3F
|
||||||
|
b.ls 2f
|
||||||
|
sttr xzr, [x0, #0x00]
|
||||||
|
sttr xzr, [x0, #0x08]
|
||||||
|
sttr xzr, [x0, #0x10]
|
||||||
|
sttr xzr, [x0, #0x18]
|
||||||
|
sttr xzr, [x0, #0x20]
|
||||||
|
sttr xzr, [x0, #0x28]
|
||||||
|
sttr xzr, [x0, #0x30]
|
||||||
|
sttr xzr, [x0, #0x38]
|
||||||
|
add x0, x0, #0x40
|
||||||
|
sub x1, x1, #0x40
|
||||||
|
b _ZN3ams4kern4arch5arm6415UserspaceAccess23ClearMemoryAligned64BitEPvm
|
||||||
|
|
||||||
|
1: /* We have less than 0x40 bytes to clear. */
|
||||||
|
cmp x1, #0
|
||||||
|
b.eq 2f
|
||||||
|
sttr xzr, [x0]
|
||||||
|
add x0, x0, #8
|
||||||
|
sub x1, x1, #8
|
||||||
|
b 1b
|
||||||
|
|
||||||
|
2: /* We're done. */
|
||||||
|
mov x0, #1
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ams::kern::arch::arm64::UserspaceAccess::ClearMemorySize32Bit(void *dst) */
|
||||||
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess20ClearMemorySize32BitEPv, "ax", %progbits
|
||||||
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess20ClearMemorySize32BitEPv
|
||||||
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess20ClearMemorySize32BitEPv, %function
|
||||||
|
.balign 0x10
|
||||||
|
_ZN3ams4kern4arch5arm6415UserspaceAccess20ClearMemorySize32BitEPv:
|
||||||
|
/* Just store a zero. */
|
||||||
|
sttr wzr, [x0]
|
||||||
|
|
||||||
|
/* We're done. */
|
||||||
|
mov x0, #1
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ams::kern::arch::arm64::UserspaceAccess::UpdateLockAtomic(u32 *out, u32 *address, u32 if_zero, u32 new_orr_mask) */
|
||||||
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess16UpdateLockAtomicEPjS4_jj, "ax", %progbits
|
||||||
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess16UpdateLockAtomicEPjS4_jj
|
||||||
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess16UpdateLockAtomicEPjS4_jj, %function
|
||||||
|
.balign 0x10
|
||||||
|
_ZN3ams4kern4arch5arm6415UserspaceAccess16UpdateLockAtomicEPjS4_jj:
|
||||||
/* Load the value from the address. */
|
/* Load the value from the address. */
|
||||||
ldaxr w4, [x1]
|
ldaxr w4, [x1]
|
||||||
|
|
||||||
@@ -374,7 +477,7 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj:
|
|||||||
stlxr w6, w5, [x1]
|
stlxr w6, w5, [x1]
|
||||||
|
|
||||||
/* If we failed to store, try again. */
|
/* If we failed to store, try again. */
|
||||||
cbnz w6, _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj
|
cbnz w6, _ZN3ams4kern4arch5arm6415UserspaceAccess16UpdateLockAtomicEPjS4_jj
|
||||||
|
|
||||||
/* We're done. */
|
/* We're done. */
|
||||||
str w4, [x0]
|
str w4, [x0]
|
||||||
@@ -382,12 +485,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16UpdateLockAtomicEPjS5_jj:
|
|||||||
ret
|
ret
|
||||||
|
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::UpdateIfEqualAtomic(s32 *out, s32 *address, s32 compare_value, s32 new_value) */
|
/* ams::kern::arch::arm64::UserspaceAccess::UpdateIfEqualAtomic(s32 *out, s32 *address, s32 compare_value, s32 new_value) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess19UpdateIfEqualAtomicEPiS4_ii, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess19UpdateIfEqualAtomicEPiS4_ii
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess19UpdateIfEqualAtomicEPiS4_ii, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess19UpdateIfEqualAtomicEPiS4_ii:
|
||||||
/* Load the value from the address. */
|
/* Load the value from the address. */
|
||||||
ldaxr w4, [x1]
|
ldaxr w4, [x1]
|
||||||
|
|
||||||
@@ -405,19 +508,19 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii:
|
|||||||
stlxr w5, w3, [x1]
|
stlxr w5, w3, [x1]
|
||||||
|
|
||||||
/* If we failed to store, try again. */
|
/* If we failed to store, try again. */
|
||||||
cbnz w5, _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19UpdateIfEqualAtomicEPiS5_ii
|
cbnz w5, _ZN3ams4kern4arch5arm6415UserspaceAccess19UpdateIfEqualAtomicEPiS4_ii
|
||||||
|
|
||||||
2: /* We're done. */
|
2: /* We're done. */
|
||||||
str w4, [x0]
|
str w4, [x0]
|
||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::DecrementIfLessThanAtomic(s32 *out, s32 *address, s32 compare) */
|
/* ams::kern::arch::arm64::UserspaceAccess::DecrementIfLessThanAtomic(s32 *out, s32 *address, s32 compare) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess25DecrementIfLessThanAtomicEPiS4_i, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess25DecrementIfLessThanAtomicEPiS4_i
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess25DecrementIfLessThanAtomicEPiS4_i, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess25DecrementIfLessThanAtomicEPiS4_i:
|
||||||
/* Load the value from the address. */
|
/* Load the value from the address. */
|
||||||
ldaxr w3, [x1]
|
ldaxr w3, [x1]
|
||||||
|
|
||||||
@@ -436,19 +539,19 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i:
|
|||||||
stlxr w5, w4, [x1]
|
stlxr w5, w4, [x1]
|
||||||
|
|
||||||
/* If we failed to store, try again. */
|
/* If we failed to store, try again. */
|
||||||
cbnz w5, _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl25DecrementIfLessThanAtomicEPiS5_i
|
cbnz w5, _ZN3ams4kern4arch5arm6415UserspaceAccess25DecrementIfLessThanAtomicEPiS4_i
|
||||||
|
|
||||||
2: /* We're done. */
|
2: /* We're done. */
|
||||||
str w3, [x0]
|
str w3, [x0]
|
||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::StoreDataCache(uintptr_t start, uintptr_t end) */
|
/* ams::kern::arch::arm64::UserspaceAccess::StoreDataCache(uintptr_t start, uintptr_t end) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14StoreDataCacheEmm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess14StoreDataCacheEmm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14StoreDataCacheEmm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess14StoreDataCacheEmm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14StoreDataCacheEmm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess14StoreDataCacheEmm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14StoreDataCacheEmm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess14StoreDataCacheEmm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x1, x0
|
cmp x1, x0
|
||||||
b.eq 2f
|
b.eq 2f
|
||||||
@@ -463,12 +566,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14StoreDataCacheEmm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::FlushDataCache(uintptr_t start, uintptr_t end) */
|
/* ams::kern::arch::arm64::UserspaceAccess::FlushDataCache(uintptr_t start, uintptr_t end) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14FlushDataCacheEmm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess14FlushDataCacheEmm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14FlushDataCacheEmm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess14FlushDataCacheEmm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14FlushDataCacheEmm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess14FlushDataCacheEmm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14FlushDataCacheEmm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess14FlushDataCacheEmm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x1, x0
|
cmp x1, x0
|
||||||
b.eq 2f
|
b.eq 2f
|
||||||
@@ -483,12 +586,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl14FlushDataCacheEmm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::InvalidateDataCache(uintptr_t start, uintptr_t end) */
|
/* ams::kern::arch::arm64::UserspaceAccess::InvalidateDataCache(uintptr_t start, uintptr_t end) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19InvalidateDataCacheEmm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess19InvalidateDataCacheEmm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19InvalidateDataCacheEmm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess19InvalidateDataCacheEmm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19InvalidateDataCacheEmm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess19InvalidateDataCacheEmm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19InvalidateDataCacheEmm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess19InvalidateDataCacheEmm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x1, x0
|
cmp x1, x0
|
||||||
b.eq 2f
|
b.eq 2f
|
||||||
@@ -503,12 +606,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl19InvalidateDataCacheEmm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::ReadIoMemory32Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::ReadIoMemory32Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory32BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory32BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory32BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory32BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory32BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory32BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory32BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory32BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -553,12 +656,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory32BitEPvPKvm:
|
|||||||
mov w9, #0xFFFFFFFF
|
mov w9, #0xFFFFFFFF
|
||||||
b 2b
|
b 2b
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::ReadIoMemory16Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::ReadIoMemory16Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory16BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory16BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory16BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory16BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory16BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory16BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory16BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess17ReadIoMemory16BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -603,12 +706,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17ReadIoMemory16BitEPvPKvm:
|
|||||||
mov w9, #0xFFFFFFFF
|
mov w9, #0xFFFFFFFF
|
||||||
b 2b
|
b 2b
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::ReadIoMemory8Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::ReadIoMemory8Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16ReadIoMemory8BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess16ReadIoMemory8BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16ReadIoMemory8BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess16ReadIoMemory8BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16ReadIoMemory8BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess16ReadIoMemory8BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16ReadIoMemory8BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess16ReadIoMemory8BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -653,12 +756,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl16ReadIoMemory8BitEPvPKvm:
|
|||||||
mov w9, #0xFFFFFFFF
|
mov w9, #0xFFFFFFFF
|
||||||
b 2b
|
b 2b
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::WriteIoMemory32Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::WriteIoMemory32Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory32BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory32BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory32BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory32BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory32BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory32BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory32BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory32BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -698,12 +801,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory32BitEPvPKvm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::WriteIoMemory16Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::WriteIoMemory16Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory16BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory16BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory16BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory16BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory16BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory16BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory16BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess18WriteIoMemory16BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
@@ -743,12 +846,12 @@ _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl18WriteIoMemory16BitEPvPKvm:
|
|||||||
mov x0, #1
|
mov x0, #1
|
||||||
ret
|
ret
|
||||||
|
|
||||||
/* ams::kern::arch::arm64::UserspaceAccess::Impl::WriteIoMemory8Bit(void *dst, const void *src, size_t size) */
|
/* ams::kern::arch::arm64::UserspaceAccess::WriteIoMemory8Bit(void *dst, const void *src, size_t size) */
|
||||||
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17WriteIoMemory8BitEPvPKvm, "ax", %progbits
|
.section .text._ZN3ams4kern4arch5arm6415UserspaceAccess17WriteIoMemory8BitEPvPKvm, "ax", %progbits
|
||||||
.global _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17WriteIoMemory8BitEPvPKvm
|
.global _ZN3ams4kern4arch5arm6415UserspaceAccess17WriteIoMemory8BitEPvPKvm
|
||||||
.type _ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17WriteIoMemory8BitEPvPKvm, %function
|
.type _ZN3ams4kern4arch5arm6415UserspaceAccess17WriteIoMemory8BitEPvPKvm, %function
|
||||||
.balign 0x10
|
.balign 0x10
|
||||||
_ZN3ams4kern4arch5arm6415UserspaceAccess4Impl17WriteIoMemory8BitEPvPKvm:
|
_ZN3ams4kern4arch5arm6415UserspaceAccess17WriteIoMemory8BitEPvPKvm:
|
||||||
/* Check if we have any work to do. */
|
/* Check if we have any work to do. */
|
||||||
cmp x2, #0
|
cmp x2, #0
|
||||||
b.eq 3f
|
b.eq 3f
|
||||||
|
|||||||
@@ -660,7 +660,8 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
ptm.Open(table_virt_addr, 1);
|
ptm.Open(table_virt_addr, 1);
|
||||||
|
|
||||||
/* Save the page. Note that it is a pre-condition that the page is cleared, when allocated from the system page table manager. */
|
/* Save the page. Note that it is a pre-condition that the page is cleared, when allocated from the system page table manager. */
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(GetVoidPointer(table_virt_addr), PageDirectorySize));
|
/* NOTE: Nintendo does not check the result of StoreDataCache. */
|
||||||
|
cpu::StoreDataCache(GetVoidPointer(table_virt_addr), PageDirectorySize);
|
||||||
g_reserved_table_phys_addr = table_phys_addr;
|
g_reserved_table_phys_addr = table_phys_addr;
|
||||||
|
|
||||||
/* Reserve an asid to correspond to no device. */
|
/* Reserve an asid to correspond to no device. */
|
||||||
@@ -709,7 +710,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
/* Install interrupt handler. */
|
/* Install interrupt handler. */
|
||||||
#if defined(MESOSPHERE_ENABLE_MEMORY_CONTROLLER_INTERRUPT)
|
#if defined(MESOSPHERE_ENABLE_MEMORY_CONTROLLER_INTERRUPT)
|
||||||
{
|
{
|
||||||
MESOSPHERE_R_ABORT_UNLESS(Kernel::GetInterruptManager().BindHandler(std::addressof(g_mc_interrupt_task), KInterruptName_MemoryController, GetCurrentCoreId(), KInterruptController::PriorityLevel_High, true, true));
|
Kernel::GetInterruptManager().BindHandler(std::addressof(g_mc_interrupt_task), KInterruptName_MemoryController, GetCurrentCoreId(), KInterruptController::PriorityLevel_High, true, true);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@@ -805,7 +806,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
MESOSPHERE_ASSERT(IsValidPhysicalAddress(GetPageTablePhysicalAddress(table_vaddr)));
|
MESOSPHERE_ASSERT(IsValidPhysicalAddress(GetPageTablePhysicalAddress(table_vaddr)));
|
||||||
|
|
||||||
ptm.Open(table_vaddr, 1);
|
ptm.Open(table_vaddr, 1);
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(GetVoidPointer(table_vaddr), PageDirectorySize));
|
cpu::StoreDataCache(GetVoidPointer(table_vaddr), PageDirectorySize);
|
||||||
m_tables[i] = table_vaddr;
|
m_tables[i] = table_vaddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1041,7 +1042,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
if (l2_index == 0 && util::IsAligned(GetInteger(phys_addr), DeviceLargePageSize) && remaining >= DeviceLargePageSize) {
|
if (l2_index == 0 && util::IsAligned(GetInteger(phys_addr), DeviceLargePageSize) && remaining >= DeviceLargePageSize) {
|
||||||
/* Set the large page. */
|
/* Set the large page. */
|
||||||
l1[l1_index].SetLargePage(read, write, true, phys_addr);
|
l1[l1_index].SetLargePage(read, write, true, phys_addr);
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry)));
|
cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry));
|
||||||
|
|
||||||
/* Synchronize. */
|
/* Synchronize. */
|
||||||
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
||||||
@@ -1061,11 +1062,11 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
const KVirtualAddress table_vaddr = ptm.Allocate();
|
const KVirtualAddress table_vaddr = ptm.Allocate();
|
||||||
R_UNLESS(table_vaddr != Null<KVirtualAddress>, svc::ResultOutOfMemory());
|
R_UNLESS(table_vaddr != Null<KVirtualAddress>, svc::ResultOutOfMemory());
|
||||||
MESOSPHERE_ASSERT(IsValidPhysicalAddress(GetPageTablePhysicalAddress(table_vaddr)));
|
MESOSPHERE_ASSERT(IsValidPhysicalAddress(GetPageTablePhysicalAddress(table_vaddr)));
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(GetVoidPointer(table_vaddr), PageTableSize));
|
cpu::StoreDataCache(GetVoidPointer(table_vaddr), PageTableSize);
|
||||||
|
|
||||||
/* Set the l1 table. */
|
/* Set the l1 table. */
|
||||||
l1[l1_index].SetTable(true, true, true, GetPageTablePhysicalAddress(table_vaddr));
|
l1[l1_index].SetTable(true, true, true, GetPageTablePhysicalAddress(table_vaddr));
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry)));
|
cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry));
|
||||||
|
|
||||||
/* Synchronize. */
|
/* Synchronize. */
|
||||||
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
||||||
@@ -1092,7 +1093,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
/* Add a reference to the l2 page (from the l2 entry page). */
|
/* Add a reference to the l2 page (from the l2 entry page). */
|
||||||
ptm.Open(KVirtualAddress(l2), 1);
|
ptm.Open(KVirtualAddress(l2), 1);
|
||||||
}
|
}
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l2[l2_index]), map_count * sizeof(PageTableEntry)));
|
cpu::StoreDataCache(std::addressof(l2[l2_index]), map_count * sizeof(PageTableEntry));
|
||||||
|
|
||||||
/* Invalidate the page table cache. */
|
/* Invalidate the page table cache. */
|
||||||
for (size_t i = util::AlignDown(l2_index, 4); i <= util::AlignDown(l2_index + map_count - 1, 4); i += 4) {
|
for (size_t i = util::AlignDown(l2_index, 4); i <= util::AlignDown(l2_index + map_count - 1, 4); i += 4) {
|
||||||
@@ -1198,7 +1199,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
++num_closed;
|
++num_closed;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l2[l2_index]), map_count * sizeof(PageTableEntry)));
|
cpu::StoreDataCache(std::addressof(l2[l2_index]), map_count * sizeof(PageTableEntry));
|
||||||
|
|
||||||
/* Invalidate the page table cache. */
|
/* Invalidate the page table cache. */
|
||||||
for (size_t i = util::AlignDown(l2_index, 4); i <= util::AlignDown(l2_index + map_count - 1, 4); i += 4) {
|
for (size_t i = util::AlignDown(l2_index, 4); i <= util::AlignDown(l2_index + map_count - 1, 4); i += 4) {
|
||||||
@@ -1242,7 +1243,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
if (ptm.Close(KVirtualAddress(l2), num_closed)) {
|
if (ptm.Close(KVirtualAddress(l2), num_closed)) {
|
||||||
/* Invalidate the l1 entry. */
|
/* Invalidate the l1 entry. */
|
||||||
l1[l1_index].Invalidate();
|
l1[l1_index].Invalidate();
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry)));
|
cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry));
|
||||||
|
|
||||||
/* Synchronize. */
|
/* Synchronize. */
|
||||||
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
||||||
@@ -1265,7 +1266,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
|
|
||||||
/* Invalidate the entry. */
|
/* Invalidate the entry. */
|
||||||
l1[l1_index].Invalidate();
|
l1[l1_index].Invalidate();
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry)));
|
cpu::StoreDataCache(std::addressof(l1[l1_index]), sizeof(PageDirectoryEntry));
|
||||||
|
|
||||||
/* Synchronize. */
|
/* Synchronize. */
|
||||||
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
InvalidatePtc(GetPageTablePhysicalAddress(KVirtualAddress(std::addressof(l1[l1_index]))));
|
||||||
|
|||||||
@@ -441,7 +441,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
KThread::Register(new_thread);
|
KThread::Register(new_thread);
|
||||||
|
|
||||||
/* Run the thread. */
|
/* Run the thread. */
|
||||||
MESOSPHERE_R_ABORT_UNLESS(new_thread->Run());
|
new_thread->Run();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -524,14 +524,6 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
/* Ensure that all cores get to this point before continuing. */
|
/* Ensure that all cores get to this point before continuing. */
|
||||||
cpu::SynchronizeAllCores();
|
cpu::SynchronizeAllCores();
|
||||||
|
|
||||||
/* Wait 100us before continuing. */
|
|
||||||
{
|
|
||||||
const s64 timeout = KHardwareTimer::GetTick() + ams::svc::Tick(TimeSpan::FromMicroSeconds(100));
|
|
||||||
while (KHardwareTimer::GetTick() < timeout) {
|
|
||||||
__asm__ __volatile__("" ::: "memory");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Save the interrupt manager's state. */
|
/* Save the interrupt manager's state. */
|
||||||
Kernel::GetInterruptManager().Save(core_id);
|
Kernel::GetInterruptManager().Save(core_id);
|
||||||
|
|
||||||
|
|||||||
@@ -361,18 +361,8 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
}();
|
}();
|
||||||
|
|
||||||
/* Return (possibly) adjusted size. */
|
/* Return (possibly) adjusted size. */
|
||||||
/* NOTE: On 20.0.0+ (and even more-so 21.0.0+) the browser requires much more memory in the applet pool in order to function. */
|
constexpr size_t ExtraSystemMemoryForAtmosphere = 40_MB;
|
||||||
/* Thus, we have to reduce our extra system memory size by 26 MB to compensate. */
|
return base_pool_size - ExtraSystemMemoryForAtmosphere - KTraceBufferSize;
|
||||||
if (kern::GetTargetFirmware() >= ams::TargetFirmware_21_0_0) {
|
|
||||||
constexpr size_t ExtraSystemMemoryForAtmosphere_21_0_0 = 7_MB;
|
|
||||||
return base_pool_size - ExtraSystemMemoryForAtmosphere_21_0_0 - KTraceBufferSize;
|
|
||||||
} else if (kern::GetTargetFirmware() >= ams::TargetFirmware_20_0_0) {
|
|
||||||
constexpr size_t ExtraSystemMemoryForAtmosphere_20_0_0 = 14_MB;
|
|
||||||
return base_pool_size - ExtraSystemMemoryForAtmosphere_20_0_0 - KTraceBufferSize;
|
|
||||||
} else {
|
|
||||||
constexpr size_t ExtraSystemMemoryForAtmosphere = 40_MB;
|
|
||||||
return base_pool_size - ExtraSystemMemoryForAtmosphere - KTraceBufferSize;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t KSystemControl::Init::GetMinimumNonSecureSystemPoolSize() {
|
size_t KSystemControl::Init::GetMinimumNonSecureSystemPoolSize() {
|
||||||
@@ -415,22 +405,22 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
/* Configure KTargetSystem. */
|
/* Configure KTargetSystem. */
|
||||||
volatile auto *ts = const_cast<volatile KTargetSystem::KTargetSystemData *>(std::addressof(KTargetSystem::s_data));
|
volatile auto *ts = const_cast<volatile KTargetSystem::KTargetSystemData *>(std::addressof(KTargetSystem::s_data));
|
||||||
{
|
{
|
||||||
/* Set whether we're in debug mode. */
|
/* Set IsDebugMode. */
|
||||||
{
|
{
|
||||||
ts->is_not_debug_mode = !GetConfigBool(smc::ConfigItem::IsDebugMode);
|
ts->is_debug_mode = GetConfigBool(smc::ConfigItem::IsDebugMode);
|
||||||
|
|
||||||
/* If we're not in debug mode, we don't want to initialize uart logging. */
|
/* If debug mode, we want to initialize uart logging. */
|
||||||
ts->disable_debug_logging = ts->is_not_debug_mode;
|
ts->enable_debug_logging = ts->is_debug_mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set Kernel Configuration. */
|
/* Set Kernel Configuration. */
|
||||||
{
|
{
|
||||||
const auto kernel_config = util::BitPack32{GetConfigU32(smc::ConfigItem::KernelConfiguration)};
|
const auto kernel_config = util::BitPack32{GetConfigU32(smc::ConfigItem::KernelConfiguration)};
|
||||||
|
|
||||||
ts->disable_debug_memory_fill = !kernel_config.Get<smc::KernelConfiguration::DebugFillMemory>();
|
ts->enable_debug_memory_fill = kernel_config.Get<smc::KernelConfiguration::DebugFillMemory>();
|
||||||
ts->disable_user_exception_handlers = !kernel_config.Get<smc::KernelConfiguration::EnableUserExceptionHandlers>();
|
ts->enable_user_exception_handlers = kernel_config.Get<smc::KernelConfiguration::EnableUserExceptionHandlers>();
|
||||||
ts->disable_dynamic_resource_limits = kernel_config.Get<smc::KernelConfiguration::DisableDynamicResourceLimits>();
|
ts->enable_dynamic_resource_limits = !kernel_config.Get<smc::KernelConfiguration::DisableDynamicResourceLimits>();
|
||||||
ts->disable_user_pmu_access = !kernel_config.Get<smc::KernelConfiguration::EnableUserPmuAccess>();
|
ts->enable_user_pmu_access = kernel_config.Get<smc::KernelConfiguration::EnableUserPmuAccess>();
|
||||||
|
|
||||||
/* Configure call smc on panic. */
|
/* Configure call smc on panic. */
|
||||||
*const_cast<volatile bool *>(std::addressof(g_call_smc_on_panic)) = kernel_config.Get<smc::KernelConfiguration::UseSecureMonitorPanicCall>();
|
*const_cast<volatile bool *>(std::addressof(g_call_smc_on_panic)) = kernel_config.Get<smc::KernelConfiguration::UseSecureMonitorPanicCall>();
|
||||||
@@ -440,7 +430,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
{
|
{
|
||||||
/* NOTE: This is used to restrict access to SvcKernelDebug/SvcChangeKernelTraceState. */
|
/* NOTE: This is used to restrict access to SvcKernelDebug/SvcChangeKernelTraceState. */
|
||||||
/* Mesosphere may wish to not require this, as we'd ideally keep ProgramVerification enabled for userland. */
|
/* Mesosphere may wish to not require this, as we'd ideally keep ProgramVerification enabled for userland. */
|
||||||
ts->disable_kernel_debugging = !GetConfigBool(smc::ConfigItem::DisableProgramVerification);
|
ts->enable_kernel_debugging = GetConfigBool(smc::ConfigItem::DisableProgramVerification);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -534,7 +524,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
KScopedSpinLock lk(s_random_lock);
|
KScopedSpinLock lk(s_random_lock);
|
||||||
|
|
||||||
|
|
||||||
if (AMS_LIKELY(!s_uninitialized_random_generator)) {
|
if (AMS_LIKELY(s_initialized_random_generator)) {
|
||||||
return KSystemControlBase::GenerateUniformRange(min, max, []() ALWAYS_INLINE_LAMBDA -> u64 { return s_random_generator.GenerateRandomU64(); });
|
return KSystemControlBase::GenerateUniformRange(min, max, []() ALWAYS_INLINE_LAMBDA -> u64 { return s_random_generator.GenerateRandomU64(); });
|
||||||
} else {
|
} else {
|
||||||
return KSystemControlBase::GenerateUniformRange(min, max, GenerateRandomU64FromSmc);
|
return KSystemControlBase::GenerateUniformRange(min, max, GenerateRandomU64FromSmc);
|
||||||
@@ -545,7 +535,7 @@ namespace ams::kern::board::nintendo::nx {
|
|||||||
KScopedInterruptDisable intr_disable;
|
KScopedInterruptDisable intr_disable;
|
||||||
KScopedSpinLock lk(s_random_lock);
|
KScopedSpinLock lk(s_random_lock);
|
||||||
|
|
||||||
if (AMS_LIKELY(!s_uninitialized_random_generator)) {
|
if (AMS_LIKELY(s_initialized_random_generator)) {
|
||||||
return s_random_generator.GenerateRandomU64();
|
return s_random_generator.GenerateRandomU64();
|
||||||
} else {
|
} else {
|
||||||
return GenerateRandomU64FromSmc();
|
return GenerateRandomU64FromSmc();
|
||||||
|
|||||||
@@ -140,12 +140,14 @@ namespace ams::kern {
|
|||||||
|
|
||||||
/* Add the previously reserved pages. */
|
/* Add the previously reserved pages. */
|
||||||
if (src_pool == dst_pool && binary_pages != 0) {
|
if (src_pool == dst_pool && binary_pages != 0) {
|
||||||
MESOSPHERE_R_ABORT_UNLESS(pg.AddBlock(KMemoryLayout::GetLinearPhysicalAddress(data), binary_pages));
|
/* NOTE: Nintendo does not check the result of this operation. */
|
||||||
|
pg.AddBlock(KMemoryLayout::GetLinearPhysicalAddress(data), binary_pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Add the previously unreserved pages. */
|
/* Add the previously unreserved pages. */
|
||||||
for (const auto &block : unreserve_pg) {
|
for (const auto &block : unreserve_pg) {
|
||||||
MESOSPHERE_R_ABORT_UNLESS(pg.AddBlock(block.GetAddress(), block.GetNumPages()));
|
/* NOTE: Nintendo does not check the result of this operation. */
|
||||||
|
pg.AddBlock(block.GetAddress(), block.GetNumPages());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
MESOSPHERE_ABORT_UNLESS(pg.GetNumPages() == static_cast<size_t>(params.code_num_pages));
|
MESOSPHERE_ABORT_UNLESS(pg.GetNumPages() == static_cast<size_t>(params.code_num_pages));
|
||||||
@@ -285,7 +287,7 @@ namespace ams::kern {
|
|||||||
MESOSPHERE_INIT_ABORT_UNLESS(expected_size != 0);
|
MESOSPHERE_INIT_ABORT_UNLESS(expected_size != 0);
|
||||||
|
|
||||||
/* Ensure that the size we need to reserve is as we expect it to be. */
|
/* Ensure that the size we need to reserve is as we expect it to be. */
|
||||||
const u32 total_size = util::AlignUp(g_initial_process_binary_header.size, PageSize);
|
const size_t total_size = util::AlignUp(g_initial_process_binary_header.size, PageSize);
|
||||||
MESOSPHERE_ABORT_UNLESS(total_size == expected_size);
|
MESOSPHERE_ABORT_UNLESS(total_size == expected_size);
|
||||||
MESOSPHERE_ABORT_UNLESS(total_size <= InitialProcessBinarySizeMax);
|
MESOSPHERE_ABORT_UNLESS(total_size <= InitialProcessBinarySizeMax);
|
||||||
|
|
||||||
|
|||||||
@@ -66,39 +66,12 @@ namespace ams::kern {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uintptr_t KAddressSpaceInfo::GetAddressSpaceStart(ams::svc::CreateProcessFlag flags, KAddressSpaceInfo::Type type, size_t code_size) {
|
uintptr_t KAddressSpaceInfo::GetAddressSpaceStart(ams::svc::CreateProcessFlag flags, KAddressSpaceInfo::Type type) {
|
||||||
MESOSPHERE_UNUSED(code_size);
|
|
||||||
return GetAddressSpaceInfo(GetAddressSpaceWidth(flags), type).GetAddress();
|
return GetAddressSpaceInfo(GetAddressSpaceWidth(flags), type).GetAddress();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t KAddressSpaceInfo::GetAddressSpaceSize(ams::svc::CreateProcessFlag flags, KAddressSpaceInfo::Type type) {
|
size_t KAddressSpaceInfo::GetAddressSpaceSize(ams::svc::CreateProcessFlag flags, KAddressSpaceInfo::Type type) {
|
||||||
/* Extract the address space from the create process flags. */
|
return GetAddressSpaceInfo(GetAddressSpaceWidth(flags), type).GetSize();
|
||||||
const auto as_flags = (flags & ams::svc::CreateProcessFlag_AddressSpaceMask);
|
|
||||||
|
|
||||||
/* Get the address space width. */
|
|
||||||
const auto as_width = GetAddressSpaceWidth(flags);
|
|
||||||
|
|
||||||
/* Get the size. */
|
|
||||||
size_t as_size = GetAddressSpaceInfo(as_width, type).GetSize();
|
|
||||||
|
|
||||||
/* If we're getting size for 32-bit without alias, adjust the sizes accordingly. */
|
|
||||||
if (as_flags == ams::svc::CreateProcessFlag_AddressSpace32BitWithoutAlias) {
|
|
||||||
switch (type) {
|
|
||||||
/* The heap space receives space that would otherwise go to the alias space. */
|
|
||||||
case KAddressSpaceInfo::Type_Heap:
|
|
||||||
as_size += GetAddressSpaceInfo(as_width, KAddressSpaceInfo::Type_Alias).GetSize();
|
|
||||||
break;
|
|
||||||
/* The alias space doesn't exist. */
|
|
||||||
case KAddressSpaceInfo::Type_Alias:
|
|
||||||
as_size = 0;
|
|
||||||
break;
|
|
||||||
/* Nothing to do by default. */
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return as_size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KAddressSpaceInfo::SetAddressSpaceSize(size_t width, Type type, size_t size) {
|
void KAddressSpaceInfo::SetAddressSpaceSize(size_t width, Type type, size_t size) {
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ namespace ams::kern {
|
|||||||
/* Clear and store cache. */
|
/* Clear and store cache. */
|
||||||
void * const block_address = GetVoidPointer(KMemoryLayout::GetLinearVirtualAddress(block.GetAddress()));
|
void * const block_address = GetVoidPointer(KMemoryLayout::GetLinearVirtualAddress(block.GetAddress()));
|
||||||
std::memset(block_address, 0xFF, block.GetSize());
|
std::memset(block_address, 0xFF, block.GetSize());
|
||||||
MESOSPHERE_R_ABORT_UNLESS(cpu::StoreDataCache(block_address, block.GetSize()));
|
cpu::StoreDataCache(block_address, block.GetSize());
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set remaining tracking members. */
|
/* Set remaining tracking members. */
|
||||||
|
|||||||
@@ -23,8 +23,8 @@ namespace ams::kern {
|
|||||||
return UserspaceAccess::CopyMemoryFromUserSize32Bit(out, GetVoidPointer(address));
|
return UserspaceAccess::CopyMemoryFromUserSize32Bit(out, GetVoidPointer(address));
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE bool WriteToUser(KProcessAddress address, u32 val) {
|
ALWAYS_INLINE bool WriteToUser(KProcessAddress address, const u32 *p) {
|
||||||
return UserspaceAccess::CopyMemoryToUserSize32Bit(GetVoidPointer(address), val);
|
return UserspaceAccess::CopyMemoryToUserSize32Bit(GetVoidPointer(address), p);
|
||||||
}
|
}
|
||||||
|
|
||||||
ALWAYS_INLINE bool UpdateLockAtomic(u32 *out, KProcessAddress address, u32 if_zero, u32 new_orr_mask) {
|
ALWAYS_INLINE bool UpdateLockAtomic(u32 *out, KProcessAddress address, u32 if_zero, u32 new_orr_mask) {
|
||||||
@@ -94,7 +94,7 @@ namespace ams::kern {
|
|||||||
|
|
||||||
/* Write the value to userspace. */
|
/* Write the value to userspace. */
|
||||||
Result result;
|
Result result;
|
||||||
if (AMS_LIKELY(WriteToUser(addr, next_value))) {
|
if (AMS_LIKELY(WriteToUser(addr, std::addressof(next_value)))) {
|
||||||
result = ResultSuccess();
|
result = ResultSuccess();
|
||||||
} else {
|
} else {
|
||||||
result = svc::ResultInvalidCurrentMemory();
|
result = svc::ResultInvalidCurrentMemory();
|
||||||
@@ -210,8 +210,8 @@ namespace ams::kern {
|
|||||||
|
|
||||||
/* If we have no waiters, clear the has waiter flag. */
|
/* If we have no waiters, clear the has waiter flag. */
|
||||||
if (it == m_tree.end() || it->GetConditionVariableKey() != cv_key) {
|
if (it == m_tree.end() || it->GetConditionVariableKey() != cv_key) {
|
||||||
constexpr u32 HasNoWaiterFlag = 0;
|
const u32 has_waiter_flag = 0;
|
||||||
WriteToUser(cv_key, HasNoWaiterFlag);
|
WriteToUser(cv_key, std::addressof(has_waiter_flag));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -252,13 +252,13 @@ namespace ams::kern {
|
|||||||
|
|
||||||
/* Write to the cv key. */
|
/* Write to the cv key. */
|
||||||
{
|
{
|
||||||
constexpr u32 HasWaiterFlag = 1;
|
const u32 has_waiter_flag = 1;
|
||||||
WriteToUser(key, HasWaiterFlag);
|
WriteToUser(key, std::addressof(has_waiter_flag));
|
||||||
cpu::DataMemoryBarrierInnerShareable();
|
cpu::DataMemoryBarrierInnerShareable();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Write the value to userspace. */
|
/* Write the value to userspace. */
|
||||||
if (!WriteToUser(addr, next_value)) {
|
if (!WriteToUser(addr, std::addressof(next_value))) {
|
||||||
slp.CancelSleep();
|
slp.CancelSleep();
|
||||||
R_THROW(svc::ResultInvalidCurrentMemory());
|
R_THROW(svc::ResultInvalidCurrentMemory());
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user