Update to v1.1.1.
8
Makefile
|
@ -33,15 +33,15 @@ include $(DEVKITPRO)/libnx/switch_rules
|
|||
|
||||
VERSION_MAJOR := 1
|
||||
VERSION_MINOR := 1
|
||||
VERSION_MICRO := 0
|
||||
VERSION_MICRO := 1
|
||||
|
||||
APP_TITLE := gcdumptool
|
||||
APP_TITLE := nxdumptool
|
||||
APP_AUTHOR := MCMrARM, DarkMatterCore
|
||||
APP_VERSION := ${VERSION_MAJOR}.${VERSION_MINOR}.${VERSION_MICRO}
|
||||
|
||||
TARGET := gcdumptool
|
||||
TARGET := nxdumptool
|
||||
BUILD := build
|
||||
SOURCES := source
|
||||
SOURCES := source source/fatfs
|
||||
DATA := data
|
||||
INCLUDES := include
|
||||
EXEFS_SRC := exefs_src
|
||||
|
|
85
README.md
|
@ -1,11 +1,11 @@
|
|||
# gcdumptool
|
||||
Nintendo Switch Game Card Dump Tool
|
||||
# nxdumptool
|
||||
Nintendo Switch Dump Tool
|
||||
|
||||
Main features
|
||||
--------------
|
||||
|
||||
* Generates full Cartridge Image dumps (XCI) with optional certificate removal and optional trimming.
|
||||
* Generates installable Nintendo Submission Packages (NSP) from base applications, updates and DLCs stored in the inserted game card.
|
||||
* Generates installable Nintendo Submission Packages (NSP) from base applications, updates and DLCs stored in the inserted game card, SD card and eMMC storage devices.
|
||||
* Compatible with multigame carts.
|
||||
* CRC32 checksum calculation for XCI/NSP dumps.
|
||||
* Full XCI dump verification using XML database from NSWDB.COM (NSWreleases.xml).
|
||||
|
@ -13,33 +13,86 @@ Main features
|
|||
* Precise HFS0 raw partition dumping, using the root HFS0 header from the game card.
|
||||
* HFS0 partition file data dumping.
|
||||
* HFS0 partition file browser with manual file dump support.
|
||||
* RomFS section file data dumping.
|
||||
* RomFS section file browser with manual file dump support.
|
||||
* Program NCA ExeFS section file data dumping.
|
||||
* Program NCA ExeFS section file browser with manual file dump support.
|
||||
* Program NCA RomFS section file data dumping.
|
||||
* Program NCA RomFS section file browser with manual file dump support.
|
||||
* Manual game card certificate dump.
|
||||
* Free SD card space checks in place.
|
||||
* File splitting support for all operations.
|
||||
* Game card metadata retrieval using NCM and NS services.
|
||||
* Dump speed calculation, ETA calculation and progress bar.
|
||||
|
||||
Operations related to installed SD/eMMC titles require a keys file located at "sdmc:/switch/prod.keys". Use [Lockpick_RCM](https://github.com/shchmue/Lockpick_RCM) to generate it.
|
||||
|
||||
Thanks to
|
||||
--------------
|
||||
|
||||
* MCMrARM, for creating the original application.
|
||||
* RSDuck, for vba-next-switch port. It's UI menu code was taken as a basis for this application.
|
||||
* Foen, for giving me some pretty good hints about how to use the NCM service.
|
||||
* Yellows8, for helping me fix a silly bug in my implementation of some NCM service IPC calls.
|
||||
* SciresM, for hactool. It's NCA content handling procedure is reproduced during the NSP dump process.
|
||||
* The-4n, for 4NXCI and hacPack. The NCA content patching procedure used in 4NXCI is replicated in the application, as well as the NACP XML generation from hacPack.
|
||||
* shchmue, for Lockpick. It was used as a reference for the key-collection algorithm needed for the NSP dump and RomFS dump/browse procedures.
|
||||
* Björn Samuelsson, for his public domain CRC32 checksum calculation code for C (crc32_fast.c).
|
||||
* AnalogMan, for his constant support and ideas.
|
||||
* RattletraPM, for the awesome icon used in the application.
|
||||
* [MCMrARM](https://github.com/MCMrARM), for creating the original application.
|
||||
* [RSDuck](https://github.com/RSDuck), for vba-next-switch port. It's UI menu code was taken as a basis for this application.
|
||||
* [foen](https://github.com/foen), for giving me some pretty good hints about how to use the NCM service.
|
||||
* [yellows8](https://github.com/yellows8), for helping me fix a silly bug in my implementation of some NCM service IPC calls.
|
||||
* [SciresM](https://github.com/SciresM), for [hactool](https://github.com/SciresM/hactool) (licensed under [ISC](https://github.com/SciresM/hactool/blob/master/LICENSE)). It's NCA content handling procedure is reproduced during the NSP dump process.
|
||||
* [The-4n](https://github.com/The-4n), for [4NXCI](https://github.com/The-4n/4NXCI) (licensed under [ISC](https://github.com/The-4n/4NXCI/blob/master/LICENSE)) and [hacPack](https://github.com/The-4n/hacPack) (licensed under [GPLv2](https://github.com/The-4n/hacPack/blob/master/LICENSE)). The NCA content patching procedure used in 4NXCI is replicated in the application, as well as the NACP XML generation from hacPack.
|
||||
* [shchmue](https://github.com/shchmue), for [Lockpick](https://github.com/shchmue/Lockpick) (licensed under [GPLv2](https://github.com/shchmue/Lockpick/blob/master/LICENSE)). It is used as a reference for the runtime key-collection algorithm needed for the NSP dump, ExeFS dump/browse and RomFS dump/browse procedures.
|
||||
* Björn Samuelsson, for his [public domain CRC32 checksum calculation C-code](http://home.thep.lu.se/~bjorn/crc).
|
||||
* [Adubbz](https://github.com/Adubbz), for [Tinfoil](https://github.com/Adubbz/Tinfoil) (licensed under [MIT](https://github.com/Adubbz/Tinfoil/blob/master/LICENSE)). Its wrappers for ES service IPC calls are used in the application.
|
||||
* ChaN, for the [FatFs module](http://elm-chan.org/fsw/ff/00index_e.html) (licensed under [FatFs license](http://elm-chan.org/fsw/ff/doc/appnote.html#license)). It is used to read ES savedata files from the BIS System partition.
|
||||
* [AnalogMan](https://github.com/AnalogMan151), for his constant support and ideas.
|
||||
* [RattletraPM](https://github.com/RattletraPM), for the awesome icon used in the application.
|
||||
* The GNOME project, from which the high contrast directory/file icons for the filebrowser modes were retrieved.
|
||||
* The folks from ReSwitched, for working towards the creation of a good homebrew ecosystem.
|
||||
* The Comfy Boyes, for being both awesome and supportive. You know who you are.
|
||||
|
||||
Donate
|
||||
--------------
|
||||
|
||||
If you like my work and you'd like to support me in any way, it's not necessary, but a donation would be greatly appreciated!
|
||||
|
||||
[![Donate](https://img.shields.io/static/v1.svg?label=PayPal&message=Donate&color=blue&style=flat&logo=paypal)](https://paypal.me/DarkMatterCore)
|
||||
|
||||
Changelog
|
||||
--------------
|
||||
|
||||
**v1.1.1:**
|
||||
* Project name changed to `nxdumptool`. This is no longer a gamecard-only tool.
|
||||
* Added ExeFS dumping/browsing support. This feature, along with the already available RomFS options, makes the application an excellent tool for modders!
|
||||
* Added compatibility with FS process memory layout in the key retrieval procedure while using emuMMC. Thanks to [shchmue](https://github.com/shchmue)!
|
||||
* Due to public demand, NSP dumping, ExeFS dumping/browsing and RomFS dumping/browsing support has been added for base applications, updates and DLCs available in both SD card and eMMC!
|
||||
* Now it's possible to select the source storage device (gamecard, SD card / eMMC) for any operation right after launching the application.
|
||||
* The gamecard submenu works exactly like the main menu has worked up to this point (except for the update options, which are now displayed in the new main menu).
|
||||
* The SD card / eMMC submenu shows installed base applications along with their icons. Upon selecting a title, a submenu with NSP and RomFS options will show up. It's also possible to dump updates/DLCs for an installed base application this way.
|
||||
* If there's installed content (updates/DLCs) with missing base application titles, pressing Y on the SD card / eMMC submenu will display this "orphan" content list and let you dump titles from it nonetheless.
|
||||
* It is possible to generate console-specific NSP dumps, dumps with modified tickets to remove console-specific data, and ticket-less dumps with standard NCA key area crypto.
|
||||
* Two new options are available in the NSP dump submenus for SD/eMMC titles: "Remove console specific data" and "Generate ticket-less dump". The latter won't appear if the former isn't enabled.
|
||||
* "Remove console specific data" cleans console specific data fields from a "personalized" ticket and replaces its RSA titlekey block with a 16-byte encrypted titlekey, essentially converting it to a "common" ticket. This option has no effect if the title already uses a "common" ticket.
|
||||
* "Generate ticket-less dump" goes another step ahead by cleaning up the Rights ID field in every NCA content file that includes it, stores the decrypted titlekey in the NCA key area and then encrypts this area using standard crypto, removing the need for a tik/cert combination.
|
||||
* Console-specific NSP dumps and dumps with modified tickets include both `tik` and `cert` files.
|
||||
* All NSP dumps generated from installed SD/eMMC titles include both `.cnmt.xml` and `.nacp.xml` files whenever possible.
|
||||
* Sadly, due to limitations in the methods currently used to perform key retrieval/derivation at runtime, NSP dumping, ExeFS dumping/browsing and RomFS dumping/browsing for SD/eMMC titles require the "sdmc:/switch/prod.keys" file. Specifically, these are the needed keys:
|
||||
* `eticket_rsa_kek`.
|
||||
* `titlekek_##` (varies from `00` to `1F`).
|
||||
* Additionally, ticket-less NSP dumps for SD/eMMC titles also require the following keys:
|
||||
* `key_area_key_application_##` (varies from `00` to `1F`).
|
||||
* `key_area_key_ocean_##` (varies from `00` to `1F`).
|
||||
* `key_area_key_system_##` (varies from `00` to `1F`).
|
||||
* All gamecard-related operations can still be performed without the need for a keys file!
|
||||
* Output data generated by the application will now be saved to its corresponding subdirectory in "sdmc:/nxdumptool/":
|
||||
* XCI dumps: "sdmc:/nxdumptool/XCI/".
|
||||
* NSP dumps: "sdmc:/nxdumptool/NSP/".
|
||||
* HFS0 data: "sdmc:/nxdumptool/HFS0/".
|
||||
* ExeFS data: "sdmc:/nxdumptool/ExeFS/".
|
||||
* RomFS data: "sdmc:/nxdumptool/RomFS/".
|
||||
* Certificate dumps: "sdmc:/nxdumptool/Certificate/".
|
||||
* The location for the NSWDB.COM XML database has been moved to "sdmc:/nxdumptool/NSWreleases.xml".
|
||||
* Tickets from updates with titlekey crypto dumped from gamecards are now converted to regular "common" tickets before being written to the output NSP dump.
|
||||
* The content distribution type for updates dumped from custom XCIs mounted through SX OS is now set to "download".
|
||||
* Fixed a NCM service handle exhaustion bug if an error ocurred while reading the RomFS section entry from the Program NCA for any base application.
|
||||
* Changed the application icon yet again. Big thanks to RattletraPM!
|
||||
* Minor changes and other various general fixes.
|
||||
|
||||
Thanks to [simontime](https://github.com/simontime) for helping me out with the RSA certificate chain retrieval process! Also thanks to MUXI from PSXTools forums for providing with testing!
|
||||
|
||||
**v1.1.0:**
|
||||
* Replaced the application icon with a new, stylish one made by RattletraPM. Thanks a lot!
|
||||
* Gamecard base application icons are now retrieved and displayed in the menu.
|
||||
|
@ -75,7 +128,7 @@ Changelog
|
|||
- If only the base application is included, like most gamecards, choosing the NSP dump option in the main menu will take you right to the base application dump menu.
|
||||
- Once you enter a submenu, you'll be able to choose exactly which title to dump belonging to that category.
|
||||
- Output update NSPs will not be modified in any way. Thus, unlike NSPs from base applications and DLCs, their CRC32 checksums will always be the same.
|
||||
* Fixed the minimum system version field size in the extended CNMT header struct. Thanks to @0Liam !
|
||||
* Fixed the minimum system version field size in the extended CNMT header struct. Thanks to [0Liam](https://github.com/0Liam)!
|
||||
* Changed the naming convention for output NSP dumps:
|
||||
- Base application: "sdmc:/[GameName] v[GameVersion] ([TitleID]) (BASE).nsp".
|
||||
- Update: "sdmc:/[GameName] v[UpdateVersion] ([UpdateTitleID]) (UPD).nsp".
|
||||
|
|
BIN
icon.jpg
Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 31 KiB |
Before Width: | Height: | Size: 629 B After Width: | Height: | Size: 629 B |
Before Width: | Height: | Size: 434 B After Width: | Height: | Size: 434 B |
Before Width: | Height: | Size: 630 B After Width: | Height: | Size: 630 B |
Before Width: | Height: | Size: 435 B After Width: | Height: | Size: 435 B |
BIN
romfs/certificate/CA00000003
Normal file
BIN
romfs/certificate/XS00000020
Normal file
BIN
romfs/certificate/XS00000021
Normal file
1173
source/dumper.c
|
@ -20,14 +20,16 @@
|
|||
|
||||
#define SMOOTHING_FACTOR (double)0.05
|
||||
|
||||
void workaroundPartitionZeroAccess(FsDeviceOperator* fsOperator);
|
||||
bool dumpCartridgeImage(FsDeviceOperator* fsOperator, bool isFat32, bool setXciArchiveBit, bool dumpCert, bool trimDump, bool calcCrc);
|
||||
bool dumpNintendoSubmissionPackage(FsDeviceOperator* fsOperator, nspDumpType selectedNspDumpType, u32 titleIndex, bool isFat32, bool calcCrc);
|
||||
bool dumpRawHfs0Partition(FsDeviceOperator* fsOperator, u32 partition, bool doSplitting);
|
||||
bool dumpHfs0PartitionData(FsDeviceOperator* fsOperator, u32 partition);
|
||||
bool dumpFileFromHfs0Partition(FsDeviceOperator* fsOperator, u32 partition, u32 file, char *filename);
|
||||
bool dumpRomFsSectionData(FsDeviceOperator* fsOperator, u32 appIndex);
|
||||
void workaroundPartitionZeroAccess();
|
||||
bool dumpCartridgeImage(bool isFat32, bool setXciArchiveBit, bool dumpCert, bool trimDump, bool calcCrc);
|
||||
bool dumpNintendoSubmissionPackage(nspDumpType selectedNspDumpType, u32 titleIndex, bool isFat32, bool calcCrc, bool removeConsoleData, bool tiklessDump);
|
||||
bool dumpRawHfs0Partition(u32 partition, bool doSplitting);
|
||||
bool dumpHfs0PartitionData(u32 partition);
|
||||
bool dumpFileFromHfs0Partition(u32 partition, u32 file, char *filename);
|
||||
bool dumpExeFsSectionData(u32 appIndex, bool doSplitting);
|
||||
bool dumpFileFromExeFsSection(u32 appIndex, u32 fileIndex, bool doSplitting);
|
||||
bool dumpRomFsSectionData(u32 appIndex);
|
||||
bool dumpFileFromRomFsSection(u32 appIndex, u32 file_offset, bool doSplitting);
|
||||
bool dumpGameCertificate(FsDeviceOperator *fsOperator);
|
||||
bool dumpGameCardCertificate();
|
||||
|
||||
#endif
|
||||
|
|
175
source/es.c
Normal file
|
@ -0,0 +1,175 @@
|
|||
#include <switch/arm/atomics.h>
|
||||
#include <switch/kernel/ipc.h>
|
||||
#include <switch/services/sm.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "es.h"
|
||||
|
||||
static Service g_esSrv;
|
||||
static u64 g_esRefCnt;
|
||||
|
||||
Result esInitialize()
|
||||
{
|
||||
atomicIncrement64(&g_esRefCnt);
|
||||
|
||||
if (serviceIsActive(&g_esSrv)) return MAKERESULT(Module_Libnx, LibnxError_AlreadyInitialized);
|
||||
|
||||
return smGetService(&g_esSrv, "es");
|
||||
}
|
||||
|
||||
void esExit()
|
||||
{
|
||||
if (atomicDecrement64(&g_esRefCnt) == 0) serviceClose(&g_esSrv);
|
||||
}
|
||||
|
||||
Result esCountCommonTicket(u32 *num_tickets)
|
||||
{
|
||||
IpcCommand c;
|
||||
ipcInitialize(&c);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 cmd_id;
|
||||
} *raw;
|
||||
|
||||
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||
|
||||
raw->magic = SFCI_MAGIC;
|
||||
raw->cmd_id = 9;
|
||||
|
||||
Result rc = serviceIpcDispatch(&g_esSrv);
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
IpcParsedCommand r;
|
||||
ipcParse(&r);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 result;
|
||||
u32 num_tickets;
|
||||
} *resp = r.Raw;
|
||||
|
||||
rc = resp->result;
|
||||
|
||||
if (R_SUCCEEDED(rc)) *num_tickets = resp->num_tickets;
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
Result esCountPersonalizedTicket(u32 *num_tickets)
|
||||
{
|
||||
IpcCommand c;
|
||||
ipcInitialize(&c);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 cmd_id;
|
||||
} *raw;
|
||||
|
||||
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||
|
||||
raw->magic = SFCI_MAGIC;
|
||||
raw->cmd_id = 10;
|
||||
|
||||
Result rc = serviceIpcDispatch(&g_esSrv);
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
IpcParsedCommand r;
|
||||
ipcParse(&r);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 result;
|
||||
u32 num_tickets;
|
||||
} *resp = r.Raw;
|
||||
|
||||
rc = resp->result;
|
||||
|
||||
if (R_SUCCEEDED(rc)) *num_tickets = resp->num_tickets;
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
Result esListCommonTicket(u32 *numRightsIdsWritten, NcmRightsId *outBuf, size_t bufSize)
|
||||
{
|
||||
IpcCommand c;
|
||||
ipcInitialize(&c);
|
||||
ipcAddRecvBuffer(&c, outBuf, bufSize, BufferType_Normal);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 cmd_id;
|
||||
} *raw;
|
||||
|
||||
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||
|
||||
raw->magic = SFCI_MAGIC;
|
||||
raw->cmd_id = 11;
|
||||
|
||||
Result rc = serviceIpcDispatch(&g_esSrv);
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
IpcParsedCommand r;
|
||||
ipcParse(&r);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 result;
|
||||
u32 num_rights_ids_written;
|
||||
} *resp = r.Raw;
|
||||
|
||||
rc = resp->result;
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
if (numRightsIdsWritten) *numRightsIdsWritten = resp->num_rights_ids_written;
|
||||
}
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
Result esListPersonalizedTicket(u32 *numRightsIdsWritten, NcmRightsId *outBuf, size_t bufSize)
|
||||
{
|
||||
IpcCommand c;
|
||||
ipcInitialize(&c);
|
||||
ipcAddRecvBuffer(&c, outBuf, bufSize, BufferType_Normal);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 cmd_id;
|
||||
} *raw;
|
||||
|
||||
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||
raw->magic = SFCI_MAGIC;
|
||||
raw->cmd_id = 12;
|
||||
|
||||
Result rc = serviceIpcDispatch(&g_esSrv);
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
IpcParsedCommand r;
|
||||
ipcParse(&r);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 result;
|
||||
u32 num_rights_ids_written;
|
||||
} *resp = r.Raw;
|
||||
|
||||
rc = resp->result;
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
if (numRightsIdsWritten) *numRightsIdsWritten = resp->num_rights_ids_written;
|
||||
}
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
16
source/es.h
Normal file
|
@ -0,0 +1,16 @@
|
|||
#pragma once
|
||||
|
||||
#ifndef __ES_H__
|
||||
#define __ES_H__
|
||||
|
||||
#include <switch.h>
|
||||
|
||||
Result esInitialize();
|
||||
void esExit();
|
||||
|
||||
Result esCountCommonTicket(u32 *num_tickets);
|
||||
Result esCountPersonalizedTicket(u32 *num_tickets);
|
||||
Result esListCommonTicket(u32 *numRightsIdsWritten, NcmRightsId *outBuf, size_t bufSize);
|
||||
Result esListPersonalizedTicket(u32 *numRightsIdsWritten, NcmRightsId *outBuf, size_t bufSize);
|
||||
|
||||
#endif
|
24
source/fatfs/LICENSE.txt
Normal file
|
@ -0,0 +1,24 @@
|
|||
FatFs License
|
||||
|
||||
FatFs has being developped as a personal project of the author, ChaN. It is free from the code anyone else wrote at current release. Following code block shows a copy of the FatFs license document that heading the source files.
|
||||
|
||||
/*----------------------------------------------------------------------------/
|
||||
/ FatFs - Generic FAT Filesystem Module Rx.xx /
|
||||
/-----------------------------------------------------------------------------/
|
||||
/
|
||||
/ Copyright (C) 20xx, ChaN, all right reserved.
|
||||
/
|
||||
/ FatFs module is an open source software. Redistribution and use of FatFs in
|
||||
/ source and binary forms, with or without modification, are permitted provided
|
||||
/ that the following condition is met:
|
||||
/
|
||||
/ 1. Redistributions of source code must retain the above copyright notice,
|
||||
/ this condition and the following disclaimer.
|
||||
/
|
||||
/ This software is provided by the copyright holder and contributors "AS IS"
|
||||
/ and any warranties related to this software are DISCLAIMED.
|
||||
/ The copyright owner or contributors be NOT LIABLE for any damages caused
|
||||
/ by use of this software.
|
||||
/----------------------------------------------------------------------------*/
|
||||
|
||||
Therefore FatFs license is one of the BSD-style licenses but there is a significant feature. FatFs is mainly intended for embedded systems. In order to extend the usability for commercial products, the redistributions of FatFs in binary form, such as embedded code, binary library and any forms without source code, does not need to include about FatFs in the documentations. This is equivalent to the 1-clause BSD license. Of course FatFs is compatible with the most of open source software licenses including GNU GPL. When you redistribute the FatFs source code with any changes or create a fork, the license can also be changed to GNU GPL, BSD-style license or any open source software license that not conflict with FatFs license.
|
92
source/fatfs/diskio.c
Normal file
|
@ -0,0 +1,92 @@
|
|||
/*-----------------------------------------------------------------------*/
|
||||
/* Low level disk I/O module skeleton for FatFs (C)ChaN, 2016 */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* If a working storage control module is available, it should be */
|
||||
/* attached to the FatFs via a glue function rather than modifying it. */
|
||||
/* This is an example of glue functions to attach various exsisting */
|
||||
/* storage control modules to the FatFs module with a defined API. */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
#include "ff.h" /* Obtains integer types */
|
||||
#include "diskio.h" /* Declarations of disk functions */
|
||||
|
||||
#include <switch.h>
|
||||
|
||||
extern FsStorage fatFsStorage;
|
||||
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* Get Drive Status */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
DSTATUS disk_status (
|
||||
BYTE pdrv /* Physical drive nmuber to identify the drive */
|
||||
)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* Inidialize a Drive */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
DSTATUS disk_initialize (
|
||||
BYTE pdrv /* Physical drive nmuber to identify the drive */
|
||||
)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* Read Sector(s) */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
DRESULT disk_read (
|
||||
BYTE pdrv, /* Physical drive nmuber to identify the drive */
|
||||
BYTE *buff, /* Data buffer to store read data */
|
||||
DWORD sector, /* Start sector in LBA */
|
||||
UINT count /* Number of sectors to read */
|
||||
)
|
||||
{
|
||||
if (R_SUCCEEDED(fsStorageRead(&fatFsStorage, FF_MAX_SS * sector, buff, FF_MAX_SS * count)))
|
||||
return RES_OK;
|
||||
return RES_ERROR;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* Write Sector(s) */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
#if FF_FS_READONLY == 0
|
||||
|
||||
DRESULT disk_write (
|
||||
BYTE pdrv, /* Physical drive nmuber to identify the drive */
|
||||
const BYTE *buff, /* Data to be written */
|
||||
DWORD sector, /* Start sector in LBA */
|
||||
UINT count /* Number of sectors to write */
|
||||
)
|
||||
{
|
||||
return RES_PARERR;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
/*-----------------------------------------------------------------------*/
|
||||
/* Miscellaneous Functions */
|
||||
/*-----------------------------------------------------------------------*/
|
||||
|
||||
DRESULT disk_ioctl (
|
||||
BYTE pdrv, /* Physical drive nmuber (0..) */
|
||||
BYTE cmd, /* Control code */
|
||||
void *buff /* Buffer to send/receive control data */
|
||||
)
|
||||
{
|
||||
return RES_OK;
|
||||
}
|
||||
|
77
source/fatfs/diskio.h
Normal file
|
@ -0,0 +1,77 @@
|
|||
/*-----------------------------------------------------------------------/
|
||||
/ Low level disk interface modlue include file (C)ChaN, 2014 /
|
||||
/-----------------------------------------------------------------------*/
|
||||
|
||||
#ifndef _DISKIO_DEFINED
|
||||
#define _DISKIO_DEFINED
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
/* Status of Disk Functions */
|
||||
typedef BYTE DSTATUS;
|
||||
|
||||
/* Results of Disk Functions */
|
||||
typedef enum {
|
||||
RES_OK = 0, /* 0: Successful */
|
||||
RES_ERROR, /* 1: R/W Error */
|
||||
RES_WRPRT, /* 2: Write Protected */
|
||||
RES_NOTRDY, /* 3: Not Ready */
|
||||
RES_PARERR /* 4: Invalid Parameter */
|
||||
} DRESULT;
|
||||
|
||||
|
||||
/*---------------------------------------*/
|
||||
/* Prototypes for disk control functions */
|
||||
|
||||
|
||||
DSTATUS disk_initialize (BYTE pdrv);
|
||||
DSTATUS disk_status (BYTE pdrv);
|
||||
DRESULT disk_read (BYTE pdrv, BYTE* buff, DWORD sector, UINT count);
|
||||
DRESULT disk_write (BYTE pdrv, const BYTE* buff, DWORD sector, UINT count);
|
||||
DRESULT disk_ioctl (BYTE pdrv, BYTE cmd, void* buff);
|
||||
|
||||
|
||||
/* Disk Status Bits (DSTATUS) */
|
||||
|
||||
#define STA_NOINIT 0x01 /* Drive not initialized */
|
||||
#define STA_NODISK 0x02 /* No medium in the drive */
|
||||
#define STA_PROTECT 0x04 /* Write protected */
|
||||
|
||||
|
||||
/* Command code for disk_ioctrl fucntion */
|
||||
|
||||
/* Generic command (Used by FatFs) */
|
||||
#define CTRL_SYNC 0 /* Complete pending write process (needed at FF_FS_READONLY == 0) */
|
||||
#define GET_SECTOR_COUNT 1 /* Get media size (needed at FF_USE_MKFS == 1) */
|
||||
#define GET_SECTOR_SIZE 2 /* Get sector size (needed at FF_MAX_SS != FF_MIN_SS) */
|
||||
#define GET_BLOCK_SIZE 3 /* Get erase block size (needed at FF_USE_MKFS == 1) */
|
||||
#define CTRL_TRIM 4 /* Inform device that the data on the block of sectors is no longer used (needed at FF_USE_TRIM == 1) */
|
||||
|
||||
/* Generic command (Not used by FatFs) */
|
||||
#define CTRL_POWER 5 /* Get/Set power status */
|
||||
#define CTRL_LOCK 6 /* Lock/Unlock media removal */
|
||||
#define CTRL_EJECT 7 /* Eject media */
|
||||
#define CTRL_FORMAT 8 /* Create physical format on the media */
|
||||
|
||||
/* MMC/SDC specific ioctl command */
|
||||
#define MMC_GET_TYPE 10 /* Get card type */
|
||||
#define MMC_GET_CSD 11 /* Get CSD */
|
||||
#define MMC_GET_CID 12 /* Get CID */
|
||||
#define MMC_GET_OCR 13 /* Get OCR */
|
||||
#define MMC_GET_SDSTAT 14 /* Get SD status */
|
||||
#define ISDIO_READ 55 /* Read data form SD iSDIO register */
|
||||
#define ISDIO_WRITE 56 /* Write data to SD iSDIO register */
|
||||
#define ISDIO_MRITE 57 /* Masked write data to SD iSDIO register */
|
||||
|
||||
/* ATA/CF specific ioctl command */
|
||||
#define ATA_GET_REV 20 /* Get F/W revision */
|
||||
#define ATA_GET_MODEL 21 /* Get model name */
|
||||
#define ATA_GET_SN 22 /* Get serial number */
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif
|
6554
source/fatfs/ff.c
Normal file
405
source/fatfs/ff.h
Normal file
|
@ -0,0 +1,405 @@
|
|||
/*----------------------------------------------------------------------------/
|
||||
/ FatFs - Generic FAT Filesystem module R0.13c /
|
||||
/-----------------------------------------------------------------------------/
|
||||
/
|
||||
/ Copyright (C) 2018, ChaN, all right reserved.
|
||||
/
|
||||
/ FatFs module is an open source software. Redistribution and use of FatFs in
|
||||
/ source and binary forms, with or without modification, are permitted provided
|
||||
/ that the following condition is met:
|
||||
|
||||
/ 1. Redistributions of source code must retain the above copyright notice,
|
||||
/ this condition and the following disclaimer.
|
||||
/
|
||||
/ This software is provided by the copyright holder and contributors "AS IS"
|
||||
/ and any warranties related to this software are DISCLAIMED.
|
||||
/ The copyright owner or contributors be NOT LIABLE for any damages caused
|
||||
/ by use of this software.
|
||||
/
|
||||
/----------------------------------------------------------------------------*/
|
||||
|
||||
|
||||
#ifndef FF_DEFINED
|
||||
#define FF_DEFINED 86604 /* Revision ID */
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#include "ffconf.h" /* FatFs configuration options */
|
||||
|
||||
#if FF_DEFINED != FFCONF_DEF
|
||||
#error Wrong configuration file (ffconf.h).
|
||||
#endif
|
||||
|
||||
|
||||
/* Integer types used for FatFs API */
|
||||
|
||||
#if defined(_WIN32) /* Main development platform */
|
||||
#define FF_INTDEF 2
|
||||
#include <windows.h>
|
||||
typedef unsigned __int64 QWORD;
|
||||
#elif (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__cplusplus) /* C99 or later */
|
||||
#define FF_INTDEF 2
|
||||
#include <stdint.h>
|
||||
typedef unsigned int UINT; /* int must be 16-bit or 32-bit */
|
||||
typedef unsigned char BYTE; /* char must be 8-bit */
|
||||
typedef uint16_t WORD; /* 16-bit unsigned integer */
|
||||
typedef uint16_t WCHAR; /* 16-bit unsigned integer */
|
||||
typedef uint32_t DWORD; /* 32-bit unsigned integer */
|
||||
typedef uint64_t QWORD; /* 64-bit unsigned integer */
|
||||
#else /* Earlier than C99 */
|
||||
#define FF_INTDEF 1
|
||||
typedef unsigned int UINT; /* int must be 16-bit or 32-bit */
|
||||
typedef unsigned char BYTE; /* char must be 8-bit */
|
||||
typedef unsigned short WORD; /* 16-bit unsigned integer */
|
||||
typedef unsigned short WCHAR; /* 16-bit unsigned integer */
|
||||
typedef unsigned long DWORD; /* 32-bit unsigned integer */
|
||||
#endif
|
||||
|
||||
|
||||
/* Definitions of volume management */
|
||||
|
||||
#if FF_MULTI_PARTITION /* Multiple partition configuration */
|
||||
typedef struct {
|
||||
BYTE pd; /* Physical drive number */
|
||||
BYTE pt; /* Partition: 0:Auto detect, 1-4:Forced partition) */
|
||||
} PARTITION;
|
||||
extern PARTITION VolToPart[]; /* Volume - Partition resolution table */
|
||||
#endif
|
||||
|
||||
#if FF_STR_VOLUME_ID
|
||||
#ifndef FF_VOLUME_STRS
|
||||
extern const char* VolumeStr[FF_VOLUMES]; /* User defied volume ID */
|
||||
#endif
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
/* Type of path name strings on FatFs API */
|
||||
|
||||
#ifndef _INC_TCHAR
|
||||
#define _INC_TCHAR
|
||||
|
||||
#if FF_USE_LFN && FF_LFN_UNICODE == 1 /* Unicode in UTF-16 encoding */
|
||||
typedef WCHAR TCHAR;
|
||||
#define _T(x) L ## x
|
||||
#define _TEXT(x) L ## x
|
||||
#elif FF_USE_LFN && FF_LFN_UNICODE == 2 /* Unicode in UTF-8 encoding */
|
||||
typedef char TCHAR;
|
||||
#define _T(x) u8 ## x
|
||||
#define _TEXT(x) u8 ## x
|
||||
#elif FF_USE_LFN && FF_LFN_UNICODE == 3 /* Unicode in UTF-32 encoding */
|
||||
typedef DWORD TCHAR;
|
||||
#define _T(x) U ## x
|
||||
#define _TEXT(x) U ## x
|
||||
#elif FF_USE_LFN && (FF_LFN_UNICODE < 0 || FF_LFN_UNICODE > 3)
|
||||
#error Wrong FF_LFN_UNICODE setting
|
||||
#else /* ANSI/OEM code in SBCS/DBCS */
|
||||
typedef char TCHAR;
|
||||
#define _T(x) x
|
||||
#define _TEXT(x) x
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
/* Type of file size variables */
|
||||
|
||||
#if FF_FS_EXFAT
|
||||
#if FF_INTDEF != 2
|
||||
#error exFAT feature wants C99 or later
|
||||
#endif
|
||||
typedef QWORD FSIZE_t;
|
||||
#else
|
||||
typedef DWORD FSIZE_t;
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
/* Filesystem object structure (FATFS) */
|
||||
|
||||
typedef struct {
|
||||
BYTE fs_type; /* Filesystem type (0:not mounted) */
|
||||
BYTE pdrv; /* Associated physical drive */
|
||||
BYTE n_fats; /* Number of FATs (1 or 2) */
|
||||
BYTE wflag; /* win[] flag (b0:dirty) */
|
||||
BYTE fsi_flag; /* FSINFO flags (b7:disabled, b0:dirty) */
|
||||
WORD id; /* Volume mount ID */
|
||||
WORD n_rootdir; /* Number of root directory entries (FAT12/16) */
|
||||
WORD csize; /* Cluster size [sectors] */
|
||||
#if FF_MAX_SS != FF_MIN_SS
|
||||
WORD ssize; /* Sector size (512, 1024, 2048 or 4096) */
|
||||
#endif
|
||||
#if FF_USE_LFN
|
||||
WCHAR* lfnbuf; /* LFN working buffer */
|
||||
#endif
|
||||
#if FF_FS_EXFAT
|
||||
BYTE* dirbuf; /* Directory entry block scratchpad buffer for exFAT */
|
||||
#endif
|
||||
#if FF_FS_REENTRANT
|
||||
FF_SYNC_t sobj; /* Identifier of sync object */
|
||||
#endif
|
||||
#if !FF_FS_READONLY
|
||||
DWORD last_clst; /* Last allocated cluster */
|
||||
DWORD free_clst; /* Number of free clusters */
|
||||
#endif
|
||||
#if FF_FS_RPATH
|
||||
DWORD cdir; /* Current directory start cluster (0:root) */
|
||||
#if FF_FS_EXFAT
|
||||
DWORD cdc_scl; /* Containing directory start cluster (invalid when cdir is 0) */
|
||||
DWORD cdc_size; /* b31-b8:Size of containing directory, b7-b0: Chain status */
|
||||
DWORD cdc_ofs; /* Offset in the containing directory (invalid when cdir is 0) */
|
||||
#endif
|
||||
#endif
|
||||
DWORD n_fatent; /* Number of FAT entries (number of clusters + 2) */
|
||||
DWORD fsize; /* Size of an FAT [sectors] */
|
||||
DWORD volbase; /* Volume base sector */
|
||||
DWORD fatbase; /* FAT base sector */
|
||||
DWORD dirbase; /* Root directory base sector/cluster */
|
||||
DWORD database; /* Data base sector */
|
||||
#if FF_FS_EXFAT
|
||||
DWORD bitbase; /* Allocation bitmap base sector */
|
||||
#endif
|
||||
DWORD winsect; /* Current sector appearing in the win[] */
|
||||
BYTE win[FF_MAX_SS]; /* Disk access window for Directory, FAT (and file data at tiny cfg) */
|
||||
} FATFS;
|
||||
|
||||
|
||||
|
||||
/* Object ID and allocation information (FFOBJID) */
|
||||
|
||||
typedef struct {
|
||||
FATFS* fs; /* Pointer to the hosting volume of this object */
|
||||
WORD id; /* Hosting volume mount ID */
|
||||
BYTE attr; /* Object attribute */
|
||||
BYTE stat; /* Object chain status (b1-0: =0:not contiguous, =2:contiguous, =3:fragmented in this session, b2:sub-directory stretched) */
|
||||
DWORD sclust; /* Object data start cluster (0:no cluster or root directory) */
|
||||
FSIZE_t objsize; /* Object size (valid when sclust != 0) */
|
||||
#if FF_FS_EXFAT
|
||||
DWORD n_cont; /* Size of first fragment - 1 (valid when stat == 3) */
|
||||
DWORD n_frag; /* Size of last fragment needs to be written to FAT (valid when not zero) */
|
||||
DWORD c_scl; /* Containing directory start cluster (valid when sclust != 0) */
|
||||
DWORD c_size; /* b31-b8:Size of containing directory, b7-b0: Chain status (valid when c_scl != 0) */
|
||||
DWORD c_ofs; /* Offset in the containing directory (valid when file object and sclust != 0) */
|
||||
#endif
|
||||
#if FF_FS_LOCK
|
||||
UINT lockid; /* File lock ID origin from 1 (index of file semaphore table Files[]) */
|
||||
#endif
|
||||
} FFOBJID;
|
||||
|
||||
|
||||
|
||||
/* File object structure (FIL) */
|
||||
|
||||
typedef struct {
|
||||
FFOBJID obj; /* Object identifier (must be the 1st member to detect invalid object pointer) */
|
||||
BYTE flag; /* File status flags */
|
||||
BYTE err; /* Abort flag (error code) */
|
||||
FSIZE_t fptr; /* File read/write pointer (Zeroed on file open) */
|
||||
DWORD clust; /* Current cluster of fpter (invalid when fptr is 0) */
|
||||
DWORD sect; /* Sector number appearing in buf[] (0:invalid) */
|
||||
#if !FF_FS_READONLY
|
||||
DWORD dir_sect; /* Sector number containing the directory entry (not used at exFAT) */
|
||||
BYTE* dir_ptr; /* Pointer to the directory entry in the win[] (not used at exFAT) */
|
||||
#endif
|
||||
#if FF_USE_FASTSEEK
|
||||
DWORD* cltbl; /* Pointer to the cluster link map table (nulled on open, set by application) */
|
||||
#endif
|
||||
#if !FF_FS_TINY
|
||||
BYTE buf[FF_MAX_SS]; /* File private data read/write window */
|
||||
#endif
|
||||
} FIL;
|
||||
|
||||
|
||||
|
||||
/* Directory object structure (DIR) */
|
||||
|
||||
typedef struct {
|
||||
FFOBJID obj; /* Object identifier */
|
||||
DWORD dptr; /* Current read/write offset */
|
||||
DWORD clust; /* Current cluster */
|
||||
DWORD sect; /* Current sector (0:Read operation has terminated) */
|
||||
BYTE* dir; /* Pointer to the directory item in the win[] */
|
||||
BYTE fn[12]; /* SFN (in/out) {body[8],ext[3],status[1]} */
|
||||
#if FF_USE_LFN
|
||||
DWORD blk_ofs; /* Offset of current entry block being processed (0xFFFFFFFF:Invalid) */
|
||||
#endif
|
||||
#if FF_USE_FIND
|
||||
const TCHAR* pat; /* Pointer to the name matching pattern */
|
||||
#endif
|
||||
} FDIR;
|
||||
|
||||
|
||||
|
||||
/* File information structure (FILINFO) */
|
||||
|
||||
typedef struct {
|
||||
FSIZE_t fsize; /* File size */
|
||||
WORD fdate; /* Modified date */
|
||||
WORD ftime; /* Modified time */
|
||||
BYTE fattrib; /* File attribute */
|
||||
#if FF_USE_LFN
|
||||
TCHAR altname[FF_SFN_BUF + 1];/* Altenative file name */
|
||||
TCHAR fname[FF_LFN_BUF + 1]; /* Primary file name */
|
||||
#else
|
||||
TCHAR fname[12 + 1]; /* File name */
|
||||
#endif
|
||||
} FILINFO;
|
||||
|
||||
|
||||
|
||||
/* File function return code (FRESULT) */
|
||||
|
||||
typedef enum {
|
||||
FR_OK = 0, /* (0) Succeeded */
|
||||
FR_DISK_ERR, /* (1) A hard error occurred in the low level disk I/O layer */
|
||||
FR_INT_ERR, /* (2) Assertion failed */
|
||||
FR_NOT_READY, /* (3) The physical drive cannot work */
|
||||
FR_NO_FILE, /* (4) Could not find the file */
|
||||
FR_NO_PATH, /* (5) Could not find the path */
|
||||
FR_INVALID_NAME, /* (6) The path name format is invalid */
|
||||
FR_DENIED, /* (7) Access denied due to prohibited access or directory full */
|
||||
FR_EXIST, /* (8) Access denied due to prohibited access */
|
||||
FR_INVALID_OBJECT, /* (9) The file/directory object is invalid */
|
||||
FR_WRITE_PROTECTED, /* (10) The physical drive is write protected */
|
||||
FR_INVALID_DRIVE, /* (11) The logical drive number is invalid */
|
||||
FR_NOT_ENABLED, /* (12) The volume has no work area */
|
||||
FR_NO_FILESYSTEM, /* (13) There is no valid FAT volume */
|
||||
FR_MKFS_ABORTED, /* (14) The f_mkfs() aborted due to any problem */
|
||||
FR_TIMEOUT, /* (15) Could not get a grant to access the volume within defined period */
|
||||
FR_LOCKED, /* (16) The operation is rejected according to the file sharing policy */
|
||||
FR_NOT_ENOUGH_CORE, /* (17) LFN working buffer could not be allocated */
|
||||
FR_TOO_MANY_OPEN_FILES, /* (18) Number of open files > FF_FS_LOCK */
|
||||
FR_INVALID_PARAMETER /* (19) Given parameter is invalid */
|
||||
} FRESULT;
|
||||
|
||||
|
||||
|
||||
/*--------------------------------------------------------------*/
|
||||
/* FatFs module application interface */
|
||||
|
||||
FRESULT f_open (FIL* fp, const TCHAR* path, BYTE mode); /* Open or create a file */
|
||||
FRESULT f_close (FIL* fp); /* Close an open file object */
|
||||
FRESULT f_read (FIL* fp, void* buff, UINT btr, UINT* br); /* Read data from the file */
|
||||
FRESULT f_write (FIL* fp, const void* buff, UINT btw, UINT* bw); /* Write data to the file */
|
||||
FRESULT f_lseek (FIL* fp, FSIZE_t ofs); /* Move file pointer of the file object */
|
||||
FRESULT f_truncate (FIL* fp); /* Truncate the file */
|
||||
FRESULT f_sync (FIL* fp); /* Flush cached data of the writing file */
|
||||
FRESULT f_opendir (FDIR* dp, const TCHAR* path); /* Open a directory */
|
||||
FRESULT f_closedir (FDIR* dp); /* Close an open directory */
|
||||
FRESULT f_readdir (FDIR* dp, FILINFO* fno); /* Read a directory item */
|
||||
FRESULT f_findfirst (FDIR* dp, FILINFO* fno, const TCHAR* path, const TCHAR* pattern); /* Find first file */
|
||||
FRESULT f_findnext (FDIR* dp, FILINFO* fno); /* Find next file */
|
||||
FRESULT f_mkdir (const TCHAR* path); /* Create a sub directory */
|
||||
FRESULT f_unlink (const TCHAR* path); /* Delete an existing file or directory */
|
||||
FRESULT f_rename (const TCHAR* path_old, const TCHAR* path_new); /* Rename/Move a file or directory */
|
||||
FRESULT f_stat (const TCHAR* path, FILINFO* fno); /* Get file status */
|
||||
FRESULT f_chmod (const TCHAR* path, BYTE attr, BYTE mask); /* Change attribute of a file/dir */
|
||||
FRESULT f_utime (const TCHAR* path, const FILINFO* fno); /* Change timestamp of a file/dir */
|
||||
FRESULT f_chdir (const TCHAR* path); /* Change current directory */
|
||||
FRESULT f_chdrive (const TCHAR* path); /* Change current drive */
|
||||
FRESULT f_getcwd (TCHAR* buff, UINT len); /* Get current directory */
|
||||
FRESULT f_getfree (const TCHAR* path, DWORD* nclst, FATFS** fatfs); /* Get number of free clusters on the drive */
|
||||
FRESULT f_getlabel (const TCHAR* path, TCHAR* label, DWORD* vsn); /* Get volume label */
|
||||
FRESULT f_setlabel (const TCHAR* label); /* Set volume label */
|
||||
FRESULT f_forward (FIL* fp, UINT(*func)(const BYTE*,UINT), UINT btf, UINT* bf); /* Forward data to the stream */
|
||||
FRESULT f_expand (FIL* fp, FSIZE_t szf, BYTE opt); /* Allocate a contiguous block to the file */
|
||||
FRESULT f_mount (FATFS* fs, const TCHAR* path, BYTE opt); /* Mount/Unmount a logical drive */
|
||||
FRESULT f_mkfs (const TCHAR* path, BYTE opt, DWORD au, void* work, UINT len); /* Create a FAT volume */
|
||||
FRESULT f_fdisk (BYTE pdrv, const DWORD* szt, void* work); /* Divide a physical drive into some partitions */
|
||||
FRESULT f_setcp (WORD cp); /* Set current code page */
|
||||
int f_putc (TCHAR c, FIL* fp); /* Put a character to the file */
|
||||
int f_puts (const TCHAR* str, FIL* cp); /* Put a string to the file */
|
||||
int f_printf (FIL* fp, const TCHAR* str, ...); /* Put a formatted string to the file */
|
||||
TCHAR* f_gets (TCHAR* buff, int len, FIL* fp); /* Get a string from the file */
|
||||
|
||||
#define f_eof(fp) ((int)((fp)->fptr == (fp)->obj.objsize))
|
||||
#define f_error(fp) ((fp)->err)
|
||||
#define f_tell(fp) ((fp)->fptr)
|
||||
#define f_size(fp) ((fp)->obj.objsize)
|
||||
#define f_rewind(fp) f_lseek((fp), 0)
|
||||
#define f_rewinddir(dp) f_readdir((dp), 0)
|
||||
#define f_rmdir(path) f_unlink(path)
|
||||
#define f_unmount(path) f_mount(0, path, 0)
|
||||
|
||||
#ifndef EOF
|
||||
#define EOF (-1)
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
|
||||
/*--------------------------------------------------------------*/
|
||||
/* Additional user defined functions */
|
||||
|
||||
/* RTC function */
|
||||
#if !FF_FS_READONLY && !FF_FS_NORTC
|
||||
DWORD get_fattime (void);
|
||||
#endif
|
||||
|
||||
/* LFN support functions */
|
||||
#if FF_USE_LFN >= 1 /* Code conversion (defined in unicode.c) */
|
||||
WCHAR ff_oem2uni (WCHAR oem, WORD cp); /* OEM code to Unicode conversion */
|
||||
WCHAR ff_uni2oem (DWORD uni, WORD cp); /* Unicode to OEM code conversion */
|
||||
DWORD ff_wtoupper (DWORD uni); /* Unicode upper-case conversion */
|
||||
#endif
|
||||
#if FF_USE_LFN == 3 /* Dynamic memory allocation */
|
||||
void* ff_memalloc (UINT msize); /* Allocate memory block */
|
||||
void ff_memfree (void* mblock); /* Free memory block */
|
||||
#endif
|
||||
|
||||
/* Sync functions */
|
||||
#if FF_FS_REENTRANT
|
||||
int ff_cre_syncobj (BYTE vol, FF_SYNC_t* sobj); /* Create a sync object */
|
||||
int ff_req_grant (FF_SYNC_t sobj); /* Lock sync object */
|
||||
void ff_rel_grant (FF_SYNC_t sobj); /* Unlock sync object */
|
||||
int ff_del_syncobj (FF_SYNC_t sobj); /* Delete a sync object */
|
||||
#endif
|
||||
|
||||
|
||||
|
||||
|
||||
/*--------------------------------------------------------------*/
|
||||
/* Flags and offset address */
|
||||
|
||||
|
||||
/* File access mode and open method flags (3rd argument of f_open) */
|
||||
#define FA_READ 0x01
|
||||
#define FA_WRITE 0x02
|
||||
#define FA_OPEN_EXISTING 0x00
|
||||
#define FA_CREATE_NEW 0x04
|
||||
#define FA_CREATE_ALWAYS 0x08
|
||||
#define FA_OPEN_ALWAYS 0x10
|
||||
#define FA_OPEN_APPEND 0x30
|
||||
|
||||
/* Fast seek controls (2nd argument of f_lseek) */
|
||||
#define CREATE_LINKMAP ((FSIZE_t)0 - 1)
|
||||
|
||||
/* Format options (2nd argument of f_mkfs) */
|
||||
#define FM_FAT 0x01
|
||||
#define FM_FAT32 0x02
|
||||
#define FM_EXFAT 0x04
|
||||
#define FM_ANY 0x07
|
||||
#define FM_SFD 0x08
|
||||
|
||||
/* Filesystem type (FATFS.fs_type) */
|
||||
#define FS_FAT12 1
|
||||
#define FS_FAT16 2
|
||||
#define FS_FAT32 3
|
||||
#define FS_EXFAT 4
|
||||
|
||||
/* File attribute bits for directory entry (FILINFO.fattrib) */
|
||||
#define AM_RDO 0x01 /* Read only */
|
||||
#define AM_HID 0x02 /* Hidden */
|
||||
#define AM_SYS 0x04 /* System */
|
||||
#define AM_DIR 0x10 /* Directory */
|
||||
#define AM_ARC 0x20 /* Archive */
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* FF_DEFINED */
|
288
source/fatfs/ffconf.h
Normal file
|
@ -0,0 +1,288 @@
|
|||
/*---------------------------------------------------------------------------/
|
||||
/ FatFs Functional Configurations
|
||||
/---------------------------------------------------------------------------*/
|
||||
|
||||
#define FFCONF_DEF 86604 /* Revision ID */
|
||||
|
||||
/*---------------------------------------------------------------------------/
|
||||
/ Function Configurations
|
||||
/---------------------------------------------------------------------------*/
|
||||
|
||||
#define FF_FS_READONLY 1
|
||||
/* This option switches read-only configuration. (0:Read/Write or 1:Read-only)
|
||||
/ Read-only configuration removes writing API functions, f_write(), f_sync(),
|
||||
/ f_unlink(), f_mkdir(), f_chmod(), f_rename(), f_truncate(), f_getfree()
|
||||
/ and optional writing functions as well. */
|
||||
|
||||
|
||||
#define FF_FS_MINIMIZE 1
|
||||
/* This option defines minimization level to remove some basic API functions.
|
||||
/
|
||||
/ 0: Basic functions are fully enabled.
|
||||
/ 1: f_stat(), f_getfree(), f_unlink(), f_mkdir(), f_truncate() and f_rename()
|
||||
/ are removed.
|
||||
/ 2: f_opendir(), f_readdir() and f_closedir() are removed in addition to 1.
|
||||
/ 3: f_lseek() function is removed in addition to 2. */
|
||||
|
||||
|
||||
#define FF_USE_STRFUNC 0
|
||||
/* This option switches string functions, f_gets(), f_putc(), f_puts() and f_printf().
|
||||
/
|
||||
/ 0: Disable string functions.
|
||||
/ 1: Enable without LF-CRLF conversion.
|
||||
/ 2: Enable with LF-CRLF conversion. */
|
||||
|
||||
|
||||
#define FF_USE_FIND 2
|
||||
/* This option switches filtered directory read functions, f_findfirst() and
|
||||
/ f_findnext(). (0:Disable, 1:Enable 2:Enable with matching altname[] too) */
|
||||
|
||||
|
||||
#define FF_USE_MKFS 0
|
||||
/* This option switches f_mkfs() function. (0:Disable or 1:Enable) */
|
||||
|
||||
|
||||
#define FF_USE_FASTSEEK 0
|
||||
/* This option switches fast seek function. (0:Disable or 1:Enable) */
|
||||
|
||||
|
||||
#define FF_USE_EXPAND 0
|
||||
/* This option switches f_expand function. (0:Disable or 1:Enable) */
|
||||
|
||||
|
||||
#define FF_USE_CHMOD 0
|
||||
/* This option switches attribute manipulation functions, f_chmod() and f_utime().
|
||||
/ (0:Disable or 1:Enable) Also FF_FS_READONLY needs to be 0 to enable this option. */
|
||||
|
||||
|
||||
#define FF_USE_LABEL 0
|
||||
/* This option switches volume label functions, f_getlabel() and f_setlabel().
|
||||
/ (0:Disable or 1:Enable) */
|
||||
|
||||
|
||||
#define FF_USE_FORWARD 0
|
||||
/* This option switches f_forward() function. (0:Disable or 1:Enable) */
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------/
|
||||
/ Locale and Namespace Configurations
|
||||
/---------------------------------------------------------------------------*/
|
||||
|
||||
#define FF_CODE_PAGE 932
|
||||
/* This option specifies the OEM code page to be used on the target system.
|
||||
/ Incorrect code page setting can cause a file open failure.
|
||||
/
|
||||
/ 437 - U.S.
|
||||
/ 720 - Arabic
|
||||
/ 737 - Greek
|
||||
/ 771 - KBL
|
||||
/ 775 - Baltic
|
||||
/ 850 - Latin 1
|
||||
/ 852 - Latin 2
|
||||
/ 855 - Cyrillic
|
||||
/ 857 - Turkish
|
||||
/ 860 - Portuguese
|
||||
/ 861 - Icelandic
|
||||
/ 862 - Hebrew
|
||||
/ 863 - Canadian French
|
||||
/ 864 - Arabic
|
||||
/ 865 - Nordic
|
||||
/ 866 - Russian
|
||||
/ 869 - Greek 2
|
||||
/ 932 - Japanese (DBCS)
|
||||
/ 936 - Simplified Chinese (DBCS)
|
||||
/ 949 - Korean (DBCS)
|
||||
/ 950 - Traditional Chinese (DBCS)
|
||||
/ 0 - Include all code pages above and configured by f_setcp()
|
||||
*/
|
||||
|
||||
|
||||
#define FF_USE_LFN 1
|
||||
#define FF_MAX_LFN 255
|
||||
/* The FF_USE_LFN switches the support for LFN (long file name).
|
||||
/
|
||||
/ 0: Disable LFN. FF_MAX_LFN has no effect.
|
||||
/ 1: Enable LFN with static working buffer on the BSS. Always NOT thread-safe.
|
||||
/ 2: Enable LFN with dynamic working buffer on the STACK.
|
||||
/ 3: Enable LFN with dynamic working buffer on the HEAP.
|
||||
/
|
||||
/ To enable the LFN, ffunicode.c needs to be added to the project. The LFN function
|
||||
/ requiers certain internal working buffer occupies (FF_MAX_LFN + 1) * 2 bytes and
|
||||
/ additional (FF_MAX_LFN + 44) / 15 * 32 bytes when exFAT is enabled.
|
||||
/ The FF_MAX_LFN defines size of the working buffer in UTF-16 code unit and it can
|
||||
/ be in range of 12 to 255. It is recommended to be set 255 to fully support LFN
|
||||
/ specification.
|
||||
/ When use stack for the working buffer, take care on stack overflow. When use heap
|
||||
/ memory for the working buffer, memory management functions, ff_memalloc() and
|
||||
/ ff_memfree() in ffsystem.c, need to be added to the project. */
|
||||
|
||||
|
||||
#define FF_LFN_UNICODE 0
|
||||
/* This option switches the character encoding on the API when LFN is enabled.
|
||||
/
|
||||
/ 0: ANSI/OEM in current CP (TCHAR = char)
|
||||
/ 1: Unicode in UTF-16 (TCHAR = WCHAR)
|
||||
/ 2: Unicode in UTF-8 (TCHAR = char)
|
||||
/ 3: Unicode in UTF-32 (TCHAR = DWORD)
|
||||
/
|
||||
/ Also behavior of string I/O functions will be affected by this option.
|
||||
/ When LFN is not enabled, this option has no effect. */
|
||||
|
||||
|
||||
#define FF_LFN_BUF 255
|
||||
#define FF_SFN_BUF 12
|
||||
/* This set of options defines size of file name members in the FILINFO structure
|
||||
/ which is used to read out directory items. These values should be suffcient for
|
||||
/ the file names to read. The maximum possible length of the read file name depends
|
||||
/ on character encoding. When LFN is not enabled, these options have no effect. */
|
||||
|
||||
|
||||
#define FF_STRF_ENCODE 3
|
||||
/* When FF_LFN_UNICODE >= 1 with LFN enabled, string I/O functions, f_gets(),
|
||||
/ f_putc(), f_puts and f_printf() convert the character encoding in it.
|
||||
/ This option selects assumption of character encoding ON THE FILE to be
|
||||
/ read/written via those functions.
|
||||
/
|
||||
/ 0: ANSI/OEM in current CP
|
||||
/ 1: Unicode in UTF-16LE
|
||||
/ 2: Unicode in UTF-16BE
|
||||
/ 3: Unicode in UTF-8
|
||||
*/
|
||||
|
||||
|
||||
#define FF_FS_RPATH 1
|
||||
/* This option configures support for relative path.
|
||||
/
|
||||
/ 0: Disable relative path and remove related functions.
|
||||
/ 1: Enable relative path. f_chdir() and f_chdrive() are available.
|
||||
/ 2: f_getcwd() function is available in addition to 1.
|
||||
*/
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------/
|
||||
/ Drive/Volume Configurations
|
||||
/---------------------------------------------------------------------------*/
|
||||
|
||||
#define FF_VOLUMES 1
|
||||
/* Number of volumes (logical drives) to be used. (1-10) */
|
||||
|
||||
|
||||
#define FF_STR_VOLUME_ID 0
|
||||
#define FF_VOLUME_STRS "RAM","NAND","CF","SD","SD2","USB","USB2","USB3"
|
||||
/* FF_STR_VOLUME_ID switches support for volume ID in arbitrary strings.
|
||||
/ When FF_STR_VOLUME_ID is set to 1 or 2, arbitrary strings can be used as drive
|
||||
/ number in the path name. FF_VOLUME_STRS defines the volume ID strings for each
|
||||
/ logical drives. Number of items must not be less than FF_VOLUMES. Valid
|
||||
/ characters for the volume ID strings are A-Z, a-z and 0-9, however, they are
|
||||
/ compared in case-insensitive. If FF_STR_VOLUME_ID >= 1 and FF_VOLUME_STRS is
|
||||
/ not defined, a user defined volume string table needs to be defined as:
|
||||
/
|
||||
/ const char* VolumeStr[FF_VOLUMES] = {"ram","flash","sd","usb",...
|
||||
*/
|
||||
|
||||
|
||||
#define FF_MULTI_PARTITION 0
|
||||
/* This option switches support for multiple volumes on the physical drive.
|
||||
/ By default (0), each logical drive number is bound to the same physical drive
|
||||
/ number and only an FAT volume found on the physical drive will be mounted.
|
||||
/ When this function is enabled (1), each logical drive number can be bound to
|
||||
/ arbitrary physical drive and partition listed in the VolToPart[]. Also f_fdisk()
|
||||
/ funciton will be available. */
|
||||
|
||||
|
||||
#define FF_MIN_SS 512
|
||||
#define FF_MAX_SS 512
|
||||
/* This set of options configures the range of sector size to be supported. (512,
|
||||
/ 1024, 2048 or 4096) Always set both 512 for most systems, generic memory card and
|
||||
/ harddisk. But a larger value may be required for on-board flash memory and some
|
||||
/ type of optical media. When FF_MAX_SS is larger than FF_MIN_SS, FatFs is configured
|
||||
/ for variable sector size mode and disk_ioctl() function needs to implement
|
||||
/ GET_SECTOR_SIZE command. */
|
||||
|
||||
|
||||
#define FF_USE_TRIM 0
|
||||
/* This option switches support for ATA-TRIM. (0:Disable or 1:Enable)
|
||||
/ To enable Trim function, also CTRL_TRIM command should be implemented to the
|
||||
/ disk_ioctl() function. */
|
||||
|
||||
|
||||
#define FF_FS_NOFSINFO 0
|
||||
/* If you need to know correct free space on the FAT32 volume, set bit 0 of this
|
||||
/ option, and f_getfree() function at first time after volume mount will force
|
||||
/ a full FAT scan. Bit 1 controls the use of last allocated cluster number.
|
||||
/
|
||||
/ bit0=0: Use free cluster count in the FSINFO if available.
|
||||
/ bit0=1: Do not trust free cluster count in the FSINFO.
|
||||
/ bit1=0: Use last allocated cluster number in the FSINFO if available.
|
||||
/ bit1=1: Do not trust last allocated cluster number in the FSINFO.
|
||||
*/
|
||||
|
||||
|
||||
|
||||
/*---------------------------------------------------------------------------/
|
||||
/ System Configurations
|
||||
/---------------------------------------------------------------------------*/
|
||||
|
||||
#define FF_FS_TINY 0
|
||||
/* This option switches tiny buffer configuration. (0:Normal or 1:Tiny)
|
||||
/ At the tiny configuration, size of file object (FIL) is shrinked FF_MAX_SS bytes.
|
||||
/ Instead of private sector buffer eliminated from the file object, common sector
|
||||
/ buffer in the filesystem object (FATFS) is used for the file data transfer. */
|
||||
|
||||
|
||||
#define FF_FS_EXFAT 0
|
||||
/* This option switches support for exFAT filesystem. (0:Disable or 1:Enable)
|
||||
/ To enable exFAT, also LFN needs to be enabled. (FF_USE_LFN >= 1)
|
||||
/ Note that enabling exFAT discards ANSI C (C89) compatibility. */
|
||||
|
||||
|
||||
#define FF_FS_NORTC 0
|
||||
#define FF_NORTC_MON 1
|
||||
#define FF_NORTC_MDAY 1
|
||||
#define FF_NORTC_YEAR 2018
|
||||
/* The option FF_FS_NORTC switches timestamp functiton. If the system does not have
|
||||
/ any RTC function or valid timestamp is not needed, set FF_FS_NORTC = 1 to disable
|
||||
/ the timestamp function. Every object modified by FatFs will have a fixed timestamp
|
||||
/ defined by FF_NORTC_MON, FF_NORTC_MDAY and FF_NORTC_YEAR in local time.
|
||||
/ To enable timestamp function (FF_FS_NORTC = 0), get_fattime() function need to be
|
||||
/ added to the project to read current time form real-time clock. FF_NORTC_MON,
|
||||
/ FF_NORTC_MDAY and FF_NORTC_YEAR have no effect.
|
||||
/ These options have no effect at read-only configuration (FF_FS_READONLY = 1). */
|
||||
|
||||
|
||||
#define FF_FS_LOCK 0
|
||||
/* The option FF_FS_LOCK switches file lock function to control duplicated file open
|
||||
/ and illegal operation to open objects. This option must be 0 when FF_FS_READONLY
|
||||
/ is 1.
|
||||
/
|
||||
/ 0: Disable file lock function. To avoid volume corruption, application program
|
||||
/ should avoid illegal open, remove and rename to the open objects.
|
||||
/ >0: Enable file lock function. The value defines how many files/sub-directories
|
||||
/ can be opened simultaneously under file lock control. Note that the file
|
||||
/ lock control is independent of re-entrancy. */
|
||||
|
||||
|
||||
/* #include <somertos.h> // O/S definitions */
|
||||
#define FF_FS_REENTRANT 0
|
||||
#define FF_FS_TIMEOUT 1000
|
||||
#define FF_SYNC_t HANDLE
|
||||
/* The option FF_FS_REENTRANT switches the re-entrancy (thread safe) of the FatFs
|
||||
/ module itself. Note that regardless of this option, file access to different
|
||||
/ volume is always re-entrant and volume control functions, f_mount(), f_mkfs()
|
||||
/ and f_fdisk() function, are always not re-entrant. Only file/directory access
|
||||
/ to the same volume is under control of this function.
|
||||
/
|
||||
/ 0: Disable re-entrancy. FF_FS_TIMEOUT and FF_SYNC_t have no effect.
|
||||
/ 1: Enable re-entrancy. Also user provided synchronization handlers,
|
||||
/ ff_req_grant(), ff_rel_grant(), ff_del_syncobj() and ff_cre_syncobj()
|
||||
/ function, must be added to the project. Samples are available in
|
||||
/ option/syscall.c.
|
||||
/
|
||||
/ The FF_FS_TIMEOUT defines timeout period in unit of time tick.
|
||||
/ The FF_SYNC_t defines O/S dependent sync object type. e.g. HANDLE, ID, OS_EVENT*,
|
||||
/ SemaphoreHandle_t and etc. A header file for O/S definitions needs to be
|
||||
/ included somewhere in the scope of ff.h. */
|
||||
|
||||
|
||||
|
||||
/*--- End of configuration options ---*/
|
15597
source/fatfs/ffunicode.c
Normal file
922
source/keys.c
|
@ -4,7 +4,6 @@
|
|||
#define __KEYS_H__
|
||||
|
||||
#include <switch.h>
|
||||
|
||||
#include "nca.h"
|
||||
|
||||
#define FS_TID (u64)0x0100000000000000
|
||||
|
@ -15,6 +14,19 @@
|
|||
|
||||
#define SHA256_HASH_LENGTH 0x20
|
||||
|
||||
#define ETICKET_DEVKEY_DATA_SIZE 0x244
|
||||
#define ETICKET_DEVKEY_CTR_OFFSET 0x4
|
||||
#define ETICKET_DEVKEY_RSA_OFFSET 0x14
|
||||
#define ETICKET_DEVKEY_RSA_SIZE (ETICKET_DEVKEY_DATA_SIZE - ETICKET_DEVKEY_RSA_OFFSET)
|
||||
|
||||
#define SIGTYPE_RSA2048_SHA1 (u32)0x10001
|
||||
#define SIGTYPE_RSA2048_SHA256 (u32)0x10004
|
||||
|
||||
#define BIS_SYSTEM_PARTITION 31
|
||||
|
||||
#define COMMON_ETICKET_FILENAME "80000000000000e1"
|
||||
#define PERSONALIZED_ETICKET_FILENAME "80000000000000e2"
|
||||
|
||||
typedef struct {
|
||||
u64 titleID;
|
||||
u8 mask;
|
||||
|
@ -29,16 +41,34 @@ typedef struct {
|
|||
} PACKED keyInfo;
|
||||
|
||||
typedef struct {
|
||||
u32 key_cnt; /* Should be equal to 6. */
|
||||
u16 memory_key_cnt; /* Key counter for keys retrieved from memory. */
|
||||
u16 ext_key_cnt; /* Key counter for keys retrieved from keysfile. */
|
||||
u32 total_key_cnt; /* Total key counter. */
|
||||
|
||||
// Needed to decrypt the NCA header using AES-128-XTS
|
||||
u8 header_kek_source[0x10]; /* Seed for header kek. */
|
||||
u8 header_key_source[0x20]; /* Seed for NCA header key. */
|
||||
u8 header_kek[0x10]; /* NCA header kek. */
|
||||
u8 header_key[0x20]; /* NCA header key. */
|
||||
|
||||
// Needed to derive the KAEK needed to decrypt the NCA key area
|
||||
u8 key_area_key_application_source[0x10]; /* Seed for kaek 0. */
|
||||
u8 key_area_key_ocean_source[0x10]; /* Seed for kaek 1. */
|
||||
u8 key_area_key_system_source[0x10]; /* Seed for kaek 2. */
|
||||
u8 header_kek_source[0x10]; /* Seed for header kek. */
|
||||
u8 header_key_source[0x20]; /* Seed for NCA header key. */
|
||||
u8 header_key[0x20]; /* NCA header key. */
|
||||
|
||||
// Needed to decrypt the title key block from an eTicket
|
||||
u8 eticket_rsa_kek[0x10]; /* eTicket RSA kek. */
|
||||
u8 titlekeks[0x20][0x10]; /* Title key encryption keys. */
|
||||
|
||||
// Needed to reencrypt the NCA key area for tik-less NSP dumps
|
||||
u8 key_area_keys[0x20][3][0x10]; /* Key area encryption keys. */
|
||||
} PACKED nca_keyset_t;
|
||||
|
||||
bool getNcaKeys();
|
||||
bool decryptNcaKeyArea(nca_header_t *dec_nca_header, u8 *out);
|
||||
bool retrieveNcaTikTitleKey(nca_header_t *dec_nca_header, u8 *out_tik, u8 *out_enc_key, u8 *out_dec_key);
|
||||
bool generateEncryptedNcaKeyAreaWithTitlekey(nca_header_t *dec_nca_header, u8 *decrypted_nca_keys);
|
||||
bool readCertsFromApplicationRomFs();
|
||||
bool retrieveCertData(u8 *out_cert, bool personalized);
|
||||
|
||||
#endif
|
||||
|
|
|
@ -102,9 +102,15 @@ int main(int argc, char *argv[])
|
|||
/* Zero out NCA keyset */
|
||||
memset(&nca_keyset, 0, sizeof(nca_keyset_t));
|
||||
|
||||
/* Init ExeFS context */
|
||||
initExeFsContext();
|
||||
|
||||
/* Init RomFS context */
|
||||
initRomFsContext();
|
||||
|
||||
/* Make sure output directories exist */
|
||||
createOutputDirectories();
|
||||
|
||||
/* Main application loop */
|
||||
bool exitLoop = false;
|
||||
while(appletMainLoop())
|
||||
|
@ -115,6 +121,9 @@ int main(int argc, char *argv[])
|
|||
case resultShowMainMenu:
|
||||
uiSetState(stateMainMenu);
|
||||
break;
|
||||
case resultShowGameCardMenu:
|
||||
uiSetState(stateGameCardMenu);
|
||||
break;
|
||||
case resultShowXciDumpMenu:
|
||||
uiSetState(stateXciDumpMenu);
|
||||
break;
|
||||
|
@ -163,6 +172,27 @@ int main(int argc, char *argv[])
|
|||
case resultHfs0BrowserCopyFile:
|
||||
uiSetState(stateHfs0BrowserCopyFile);
|
||||
break;
|
||||
case resultShowExeFsMenu:
|
||||
uiSetState(stateExeFsMenu);
|
||||
break;
|
||||
case resultShowExeFsSectionDataDumpMenu:
|
||||
uiSetState(stateExeFsSectionDataDumpMenu);
|
||||
break;
|
||||
case resultDumpExeFsSectionData:
|
||||
uiSetState(stateDumpExeFsSectionData);
|
||||
break;
|
||||
case resultShowExeFsSectionBrowserMenu:
|
||||
uiSetState(stateExeFsSectionBrowserMenu);
|
||||
break;
|
||||
case resultExeFsSectionBrowserGetList:
|
||||
uiSetState(stateExeFsSectionBrowserGetList);
|
||||
break;
|
||||
case resultShowExeFsSectionBrowser:
|
||||
uiSetState(stateExeFsSectionBrowser);
|
||||
break;
|
||||
case resultExeFsSectionBrowserCopyFile:
|
||||
uiSetState(stateExeFsSectionBrowserCopyFile);
|
||||
break;
|
||||
case resultShowRomFsMenu:
|
||||
uiSetState(stateRomFsMenu);
|
||||
break;
|
||||
|
@ -190,6 +220,15 @@ int main(int argc, char *argv[])
|
|||
case resultDumpGameCardCertificate:
|
||||
uiSetState(stateDumpGameCardCertificate);
|
||||
break;
|
||||
case resultShowSdCardEmmcMenu:
|
||||
uiSetState(stateSdCardEmmcMenu);
|
||||
break;
|
||||
case resultShowSdCardEmmcTitleMenu:
|
||||
uiSetState(stateSdCardEmmcTitleMenu);
|
||||
break;
|
||||
case resultShowSdCardEmmcOrphanPatchAddOnMenu:
|
||||
uiSetState(stateSdCardEmmcOrphanPatchAddOnMenu);
|
||||
break;
|
||||
case resultShowUpdateMenu:
|
||||
uiSetState(stateUpdateMenu);
|
||||
break;
|
||||
|
@ -312,8 +351,8 @@ int main(int argc, char *argv[])
|
|||
ret = -2;
|
||||
}
|
||||
|
||||
/* Free gamecard resources */
|
||||
freeGameCardInfo();
|
||||
/* Free global resources */
|
||||
freeGlobalData();
|
||||
|
||||
/* Deinitialize UI */
|
||||
uiDeinit();
|
||||
|
|
230
source/nca.c
|
@ -2,10 +2,9 @@
|
|||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "nca.h"
|
||||
#include "keys.h"
|
||||
#include "util.h"
|
||||
#include "ui.h"
|
||||
#include "keys.h"
|
||||
#include "rsa.h"
|
||||
|
||||
/* Extern variables */
|
||||
|
@ -13,6 +12,7 @@
|
|||
extern int breaks;
|
||||
extern int font_height;
|
||||
|
||||
extern exefs_ctx_t exeFsContext;
|
||||
extern romfs_ctx_t romFsContext;
|
||||
|
||||
extern char strbuf[NAME_BUF_LEN * 4];
|
||||
|
@ -209,7 +209,7 @@ void convertU64ToNcaSize(const u64 size, u8 out[0x6])
|
|||
bool loadNcaKeyset()
|
||||
{
|
||||
// Keyset already loaded
|
||||
if (nca_keyset.key_cnt > 0) return true;
|
||||
if (nca_keyset.total_key_cnt > 0) return true;
|
||||
|
||||
if (!(envIsSyscallHinted(0x60) && // svcDebugActiveProcess
|
||||
envIsSyscallHinted(0x63) && // svcGetDebugEvent
|
||||
|
@ -386,7 +386,7 @@ bool encryptNcaHeader(nca_header_t *input, u8 *outBuf, u64 outBufSize)
|
|||
return true;
|
||||
}
|
||||
|
||||
bool decryptNcaHeader(const u8 *ncaBuf, u64 ncaBufSize, nca_header_t *out, title_rights_ctx *rights_info, u8 *decrypted_nca_keys)
|
||||
bool decryptNcaHeader(const u8 *ncaBuf, u64 ncaBufSize, nca_header_t *out, title_rights_ctx *rights_info, u8 *decrypted_nca_keys, bool retrieveTitleKeyData)
|
||||
{
|
||||
if (!ncaBuf || !ncaBufSize || ncaBufSize < NCA_FULL_HEADER_LENGTH || !out || !decrypted_nca_keys)
|
||||
{
|
||||
|
@ -462,13 +462,45 @@ bool decryptNcaHeader(const u8 *ncaBuf, u64 ncaBufSize, nca_header_t *out, title
|
|||
|
||||
if (has_rights_id)
|
||||
{
|
||||
if (rights_info != NULL && !rights_info->has_rights_id)
|
||||
if (rights_info != NULL)
|
||||
{
|
||||
// If we're dealing with a rights info context, retrieve the ticket for the current title
|
||||
|
||||
if (!rights_info->has_rights_id)
|
||||
{
|
||||
rights_info->has_rights_id = true;
|
||||
|
||||
memcpy(rights_info->rights_id, out->rights_id, 16);
|
||||
convertDataToHexString(out->rights_id, 16, rights_info->rights_id_str, 33);
|
||||
sprintf(rights_info->tik_filename, "%s.tik", rights_info->rights_id_str);
|
||||
sprintf(rights_info->cert_filename, "%s.cert", rights_info->rights_id_str);
|
||||
|
||||
if (retrieveTitleKeyData)
|
||||
{
|
||||
if (!retrieveNcaTikTitleKey(out, (u8*)(&(rights_info->tik_data)), rights_info->enc_titlekey, rights_info->dec_titlekey)) return false;
|
||||
|
||||
memset(decrypted_nca_keys, 0, NCA_KEY_AREA_SIZE);
|
||||
memcpy(decrypted_nca_keys + (NCA_KEY_AREA_KEY_SIZE * 2), rights_info->dec_titlekey, 0x10);
|
||||
}
|
||||
} else {
|
||||
// Copy what we already have
|
||||
if (retrieveTitleKeyData)
|
||||
{
|
||||
memset(decrypted_nca_keys, 0, NCA_KEY_AREA_SIZE);
|
||||
memcpy(decrypted_nca_keys + (NCA_KEY_AREA_KEY_SIZE * 2), rights_info->dec_titlekey, 0x10);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// Otherwise, only retrieve the decrypted titlekey. This is used with ExeFS/RomFS section parsing for SD/eMMC titles
|
||||
if (retrieveTitleKeyData)
|
||||
{
|
||||
u8 tmp_dec_titlekey[0x10];
|
||||
|
||||
if (!retrieveNcaTikTitleKey(out, NULL, NULL, tmp_dec_titlekey)) return false;
|
||||
|
||||
memset(decrypted_nca_keys, 0, NCA_KEY_AREA_SIZE);
|
||||
memcpy(decrypted_nca_keys + (NCA_KEY_AREA_KEY_SIZE * 2), tmp_dec_titlekey, 0x10);
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (!decryptNcaKeyArea(out, decrypted_nca_keys)) return false;
|
||||
|
@ -799,7 +831,7 @@ bool processProgramNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca
|
|||
return true;
|
||||
}
|
||||
|
||||
bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, nca_cnmt_mod_data *output, title_rights_ctx *rights_info)
|
||||
bool retrieveCnmtNcaData(FsStorageId curStorageId, nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, nca_cnmt_mod_data *output, title_rights_ctx *rights_info, bool replaceKeyArea)
|
||||
{
|
||||
if (!ncaBuf || !xml_program_info || !xml_content_info || !output || !rights_info)
|
||||
{
|
||||
|
@ -809,6 +841,8 @@ bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_p
|
|||
|
||||
nca_header_t dec_header;
|
||||
|
||||
u32 i;
|
||||
|
||||
u64 section_offset;
|
||||
u64 section_size;
|
||||
u8 *section_data = NULL;
|
||||
|
@ -835,7 +869,9 @@ bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_p
|
|||
char cnmtFileName[50] = {'\0'};
|
||||
snprintf(cnmtFileName, sizeof(cnmtFileName) / sizeof(cnmtFileName[0]), "%s_%016lx.cnmt", getTitleType(xml_program_info->type), xml_program_info->title_id);
|
||||
|
||||
if (!decryptNcaHeader(ncaBuf, xml_content_info->size, &dec_header, rights_info, xml_content_info->decrypted_nca_keys)) return false;
|
||||
// Decrypt the NCA header
|
||||
// Don't retrieve the ticket and/or titlekey if we're dealing with a Patch with titlekey crypto bundled with the inserted gamecard
|
||||
if (!decryptNcaHeader(ncaBuf, xml_content_info->size, &dec_header, rights_info, xml_content_info->decrypted_nca_keys, (curStorageId != FsStorageId_GameCard))) return false;
|
||||
|
||||
if (dec_header.fs_headers[0].partition_type != NCA_FS_HEADER_PARTITION_PFS0 || dec_header.fs_headers[0].fs_type != NCA_FS_HEADER_FSTYPE_PFS0)
|
||||
{
|
||||
|
@ -856,8 +892,39 @@ bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_p
|
|||
return false;
|
||||
}
|
||||
|
||||
bool has_rights_id = false;
|
||||
|
||||
for(i = 0; i < 0x10; i++)
|
||||
{
|
||||
if (dec_header.rights_id[i] != 0)
|
||||
{
|
||||
has_rights_id = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (curStorageId == FsStorageId_GameCard)
|
||||
{
|
||||
if (has_rights_id)
|
||||
{
|
||||
uiDrawString("Error: Rights ID field in NCA header not empty!", 8, (breaks * (font_height + (font_height / 4))) + (font_height / 8), 255, 0, 0);
|
||||
return false;
|
||||
}
|
||||
|
||||
// Modify distribution type
|
||||
if (selectedNspDumpType != DUMP_PATCH_NSP) dec_header.distribution = 0;
|
||||
} else
|
||||
if (curStorageId == FsStorageId_SdCard || curStorageId == FsStorageId_NandUser)
|
||||
{
|
||||
if (has_rights_id && replaceKeyArea)
|
||||
{
|
||||
// Generate new encrypted NCA key area using titlekey
|
||||
if (!generateEncryptedNcaKeyAreaWithTitlekey(&dec_header, xml_content_info->decrypted_nca_keys)) return false;
|
||||
|
||||
// Remove rights ID from NCA
|
||||
memset(dec_header.rights_id, 0, 0x10);
|
||||
}
|
||||
}
|
||||
|
||||
section_offset = ((u64)dec_header.section_entries[0].media_start_offset * (u64)MEDIA_UNIT_SIZE);
|
||||
section_size = (((u64)dec_header.section_entries[0].media_end_offset * (u64)MEDIA_UNIT_SIZE) - section_offset);
|
||||
|
@ -870,7 +937,6 @@ bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_p
|
|||
|
||||
// Generate initial CTR
|
||||
unsigned char ctr[0x10];
|
||||
unsigned int i;
|
||||
u64 ofs = (section_offset >> 4);
|
||||
|
||||
for(i = 0; i < 0x8; i++)
|
||||
|
@ -1083,7 +1149,147 @@ bool patchCnmtNca(u8 *ncaBuf, u64 ncaBufSize, cnmt_xml_program_info *xml_program
|
|||
return true;
|
||||
}
|
||||
|
||||
bool readRomFsEntriesFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys)
|
||||
bool readExeFsEntryFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys)
|
||||
{
|
||||
if (!ncmStorage || !ncaId || !dec_nca_header || !decrypted_nca_keys)
|
||||
{
|
||||
uiDrawString("Error: invalid parameters to read RomFS section from NCA!", 8, (breaks * (font_height + (font_height / 4))) + (font_height / 8), 255, 0, 0);
|
||||
return false;
|
||||
}
|
||||
|
||||
u8 exefs_index;
|
||||
bool found_exefs = false;
|
||||
|
||||
u32 i;
|
||||
|
||||
u64 section_offset;
|
||||
u64 section_size;
|
||||
|
||||
unsigned char ctr[0x10];
|
||||
memset(ctr, 0, 0x10);
|
||||
|
||||
u64 ofs;
|
||||
|
||||
u8 ctr_key[NCA_KEY_AREA_KEY_SIZE];
|
||||
memcpy(ctr_key, decrypted_nca_keys + (NCA_KEY_AREA_KEY_SIZE * 2), NCA_KEY_AREA_KEY_SIZE);
|
||||
|
||||
Aes128CtrContext aes_ctx;
|
||||
aes128CtrContextCreate(&aes_ctx, ctr_key, ctr);
|
||||
|
||||
u64 nca_pfs0_offset;
|
||||
pfs0_header nca_pfs0_header;
|
||||
|
||||
u64 nca_pfs0_entries_offset;
|
||||
pfs0_entry_table *nca_pfs0_entries = NULL;
|
||||
|
||||
u64 nca_pfs0_str_table_offset;
|
||||
char *nca_pfs0_str_table = NULL;
|
||||
|
||||
u64 nca_pfs0_data_offset;
|
||||
|
||||
initExeFsContext();
|
||||
|
||||
for(exefs_index = 0; exefs_index < 4; exefs_index++)
|
||||
{
|
||||
if (dec_nca_header->fs_headers[exefs_index].partition_type != NCA_FS_HEADER_PARTITION_PFS0 || dec_nca_header->fs_headers[exefs_index].fs_type != NCA_FS_HEADER_FSTYPE_PFS0 || !dec_nca_header->fs_headers[exefs_index].pfs0_superblock.pfs0_size || dec_nca_header->fs_headers[exefs_index].crypt_type != NCA_FS_HEADER_CRYPT_CTR) continue;
|
||||
|
||||
section_offset = ((u64)dec_nca_header->section_entries[exefs_index].media_start_offset * (u64)MEDIA_UNIT_SIZE);
|
||||
section_size = (((u64)dec_nca_header->section_entries[exefs_index].media_end_offset * (u64)MEDIA_UNIT_SIZE) - section_offset);
|
||||
|
||||
if (!section_offset || section_offset < NCA_FULL_HEADER_LENGTH || !section_size) continue;
|
||||
|
||||
// Generate initial CTR
|
||||
ofs = (section_offset >> 4);
|
||||
|
||||
for(i = 0; i < 0x8; i++)
|
||||
{
|
||||
ctr[i] = dec_nca_header->fs_headers[exefs_index].section_ctr[0x08 - i - 1];
|
||||
ctr[0x10 - i - 1] = (unsigned char)(ofs & 0xFF);
|
||||
ofs >>= 8;
|
||||
}
|
||||
|
||||
aes128CtrContextResetCtr(&aes_ctx, ctr);
|
||||
|
||||
nca_pfs0_offset = (section_offset + dec_nca_header->fs_headers[exefs_index].pfs0_superblock.pfs0_offset);
|
||||
|
||||
if (!processNcaCtrSectionBlock(ncmStorage, ncaId, nca_pfs0_offset, &nca_pfs0_header, sizeof(pfs0_header), &aes_ctx, false)) return false;
|
||||
|
||||
if (bswap_32(nca_pfs0_header.magic) != PFS0_MAGIC || !nca_pfs0_header.file_cnt || !nca_pfs0_header.str_table_size) continue;
|
||||
|
||||
nca_pfs0_entries_offset = (nca_pfs0_offset + sizeof(pfs0_header));
|
||||
|
||||
nca_pfs0_entries = calloc(nca_pfs0_header.file_cnt, sizeof(pfs0_entry_table));
|
||||
if (!nca_pfs0_entries) continue;
|
||||
|
||||
if (!processNcaCtrSectionBlock(ncmStorage, ncaId, nca_pfs0_entries_offset, nca_pfs0_entries, (u64)nca_pfs0_header.file_cnt * sizeof(pfs0_entry_table), &aes_ctx, false))
|
||||
{
|
||||
free(nca_pfs0_entries);
|
||||
return false;
|
||||
}
|
||||
|
||||
nca_pfs0_str_table_offset = (nca_pfs0_entries_offset + ((u64)nca_pfs0_header.file_cnt * sizeof(pfs0_entry_table)));
|
||||
|
||||
nca_pfs0_str_table = calloc(nca_pfs0_header.str_table_size, sizeof(char));
|
||||
if (!nca_pfs0_str_table)
|
||||
{
|
||||
free(nca_pfs0_entries);
|
||||
nca_pfs0_entries = NULL;
|
||||
continue;
|
||||
}
|
||||
|
||||
if (!processNcaCtrSectionBlock(ncmStorage, ncaId, nca_pfs0_str_table_offset, nca_pfs0_str_table, (u64)nca_pfs0_header.str_table_size, &aes_ctx, false))
|
||||
{
|
||||
free(nca_pfs0_str_table);
|
||||
free(nca_pfs0_entries);
|
||||
return false;
|
||||
}
|
||||
|
||||
for(i = 0; i < nca_pfs0_header.file_cnt; i++)
|
||||
{
|
||||
char *cur_filename = (nca_pfs0_str_table + nca_pfs0_entries[i].filename_offset);
|
||||
|
||||
if (!strncasecmp(cur_filename, "main.npdm", 9))
|
||||
{
|
||||
found_exefs = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (found_exefs) break;
|
||||
|
||||
free(nca_pfs0_str_table);
|
||||
nca_pfs0_str_table = NULL;
|
||||
|
||||
free(nca_pfs0_entries);
|
||||
nca_pfs0_entries = NULL;
|
||||
}
|
||||
|
||||
if (!found_exefs)
|
||||
{
|
||||
uiDrawString("Error: NCA doesn't hold an ExeFS section!", 8, (breaks * (font_height + (font_height / 4))) + (font_height / 8), 255, 0, 0);
|
||||
return false;
|
||||
}
|
||||
|
||||
nca_pfs0_data_offset = (nca_pfs0_str_table_offset + (u64)nca_pfs0_header.str_table_size);
|
||||
|
||||
// Save data to output struct
|
||||
// The caller function must free these data pointers
|
||||
memcpy(&(exeFsContext.ncmStorage), ncmStorage, sizeof(NcmContentStorage));
|
||||
memcpy(&(exeFsContext.ncaId), ncaId, sizeof(NcmNcaId));
|
||||
memcpy(&(exeFsContext.aes_ctx), &aes_ctx, sizeof(Aes128CtrContext));
|
||||
exeFsContext.exefs_offset = nca_pfs0_offset;
|
||||
exeFsContext.exefs_size = dec_nca_header->fs_headers[exefs_index].pfs0_superblock.pfs0_size;
|
||||
memcpy(&(exeFsContext.exefs_header), &nca_pfs0_header, sizeof(pfs0_header));
|
||||
exeFsContext.exefs_entries_offset = nca_pfs0_entries_offset;
|
||||
exeFsContext.exefs_entries = nca_pfs0_entries;
|
||||
exeFsContext.exefs_str_table_offset = nca_pfs0_str_table_offset;
|
||||
exeFsContext.exefs_str_table = nca_pfs0_str_table;
|
||||
exeFsContext.exefs_data_offset = nca_pfs0_data_offset;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool readRomFsEntryFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys)
|
||||
{
|
||||
if (!ncmStorage || !ncaId || !dec_nca_header || !decrypted_nca_keys)
|
||||
{
|
||||
|
@ -1116,6 +1322,8 @@ bool readRomFsEntriesFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaI
|
|||
romfs_dir *romfs_dir_entries = NULL;
|
||||
romfs_file *romfs_file_entries = NULL;
|
||||
|
||||
initRomFsContext();
|
||||
|
||||
for(romfs_index = 0; romfs_index < 4; romfs_index++)
|
||||
{
|
||||
if (dec_nca_header->fs_headers[romfs_index].partition_type == NCA_FS_HEADER_PARTITION_ROMFS && dec_nca_header->fs_headers[romfs_index].fs_type == NCA_FS_HEADER_FSTYPE_ROMFS)
|
||||
|
@ -1767,9 +1975,7 @@ bool generateNacpXmlFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId
|
|||
u8 i;
|
||||
char tmp[NAME_BUF_LEN] = {'\0'};
|
||||
|
||||
initRomFsContext();
|
||||
|
||||
if (!readRomFsEntriesFromNca(ncmStorage, ncaId, dec_nca_header, decrypted_nca_keys)) return false;
|
||||
if (!readRomFsEntryFromNca(ncmStorage, ncaId, dec_nca_header, decrypted_nca_keys)) return false;
|
||||
|
||||
// Look for the control.nacp file
|
||||
while(entryOffset < romFsContext.romfs_filetable_size)
|
||||
|
|
115
source/nca.h
|
@ -57,6 +57,21 @@
|
|||
#define NSP_TIK_FILENAME_LENGTH 0x25 // Rights ID + ".tik" + NULL terminator
|
||||
#define NSP_CERT_FILENAME_LENGTH 0x26 // Rights ID + ".cert" + NULL terminator
|
||||
|
||||
#define ETICKET_ENTRY_SIZE 0x400
|
||||
#define ETICKET_TITLEKEY_OFFSET 0x180
|
||||
#define ETICKET_RIGHTSID_OFFSET 0x2A0
|
||||
#define ETICKET_UNKNOWN_FIELD_SIZE 0x140
|
||||
#define ETICKET_DATA_OFFSET 0x140
|
||||
|
||||
#define ETICKET_CA_CERT_SIZE 0x400
|
||||
#define ETICKET_XS_CERT_SIZE 0x300
|
||||
|
||||
#define ETICKET_TIK_FILE_SIZE (ETICKET_ENTRY_SIZE - 0x140)
|
||||
#define ETICKET_CERT_FILE_SIZE (ETICKET_CA_CERT_SIZE + ETICKET_XS_CERT_SIZE)
|
||||
|
||||
#define ETICKET_TITLEKEY_COMMON 0
|
||||
#define ETICKET_TITLEKEY_PERSONALIZED 1
|
||||
|
||||
typedef enum {
|
||||
DUMP_APP_NSP = 0,
|
||||
DUMP_PATCH_NSP,
|
||||
|
@ -256,6 +271,70 @@ typedef struct {
|
|||
u64 title_cnmt_size;
|
||||
} PACKED nca_cnmt_mod_data;
|
||||
|
||||
typedef struct {
|
||||
u32 sig_type;
|
||||
u8 signature[0x100];
|
||||
u8 padding[0x3C];
|
||||
char sig_issuer[0x40];
|
||||
u8 titlekey_block[0x100];
|
||||
u8 unk1;
|
||||
u8 titlekey_type;
|
||||
u8 unk2[0x03];
|
||||
u8 master_key_rev;
|
||||
u8 unk3[0x0A];
|
||||
u64 ticket_id;
|
||||
u64 device_id;
|
||||
u8 rights_id[0x10];
|
||||
u32 account_id;
|
||||
u8 unk4[0x0C];
|
||||
} PACKED rsa2048_sha256_ticket;
|
||||
|
||||
typedef struct {
|
||||
bool has_rights_id;
|
||||
u8 rights_id[0x10];
|
||||
char rights_id_str[33];
|
||||
char tik_filename[37];
|
||||
char cert_filename[38];
|
||||
u8 enc_titlekey[0x10];
|
||||
u8 dec_titlekey[0x10];
|
||||
u8 cert_data[ETICKET_CERT_FILE_SIZE];
|
||||
rsa2048_sha256_ticket tik_data;
|
||||
} PACKED title_rights_ctx;
|
||||
|
||||
typedef struct {
|
||||
NcmContentStorage ncmStorage;
|
||||
NcmNcaId ncaId;
|
||||
Aes128CtrContext aes_ctx;
|
||||
u64 exefs_offset; // Relative to NCA start
|
||||
u64 exefs_size;
|
||||
pfs0_header exefs_header;
|
||||
u64 exefs_entries_offset; // Relative to NCA start
|
||||
pfs0_entry_table *exefs_entries;
|
||||
u64 exefs_str_table_offset; // Relative to NCA start
|
||||
char *exefs_str_table;
|
||||
u64 exefs_data_offset; // Relative to NCA start
|
||||
} PACKED exefs_ctx_t;
|
||||
|
||||
typedef struct {
|
||||
NcmContentStorage ncmStorage;
|
||||
NcmNcaId ncaId;
|
||||
Aes128CtrContext aes_ctx;
|
||||
u64 romfs_offset; // Relative to NCA start
|
||||
u64 romfs_size;
|
||||
u64 romfs_dirtable_offset; // Relative to NCA start
|
||||
u64 romfs_dirtable_size;
|
||||
romfs_dir *romfs_dir_entries;
|
||||
u64 romfs_filetable_offset; // Relative to NCA start
|
||||
u64 romfs_filetable_size;
|
||||
romfs_file *romfs_file_entries;
|
||||
u64 romfs_filedata_offset; // Relative to NCA start
|
||||
} PACKED romfs_ctx_t;
|
||||
|
||||
typedef struct {
|
||||
u8 type; // 1 = Dir, 2 = File
|
||||
u64 offset; // Relative to directory/file table, depending on type
|
||||
} PACKED romfs_browser_entry;
|
||||
|
||||
typedef struct {
|
||||
NacpLanguageEntry lang[16];
|
||||
char Isbn[0x25];
|
||||
|
@ -309,34 +388,6 @@ typedef struct {
|
|||
u8 Reserved[0xDEC];
|
||||
} PACKED nacp_t;
|
||||
|
||||
typedef struct {
|
||||
bool has_rights_id;
|
||||
u8 rights_id[0x10];
|
||||
char rights_id_str[33];
|
||||
char tik_filename[37];
|
||||
char cert_filename[38];
|
||||
} PACKED title_rights_ctx;
|
||||
|
||||
typedef struct {
|
||||
NcmContentStorage ncmStorage;
|
||||
NcmNcaId ncaId;
|
||||
Aes128CtrContext aes_ctx;
|
||||
u64 romfs_offset; // Relative to NCA start
|
||||
u64 romfs_size;
|
||||
u64 romfs_dirtable_offset; // Relative to NCA start
|
||||
u64 romfs_dirtable_size;
|
||||
romfs_dir *romfs_dir_entries;
|
||||
u64 romfs_filetable_offset; // Relative to NCA start
|
||||
u64 romfs_filetable_size;
|
||||
romfs_file *romfs_file_entries;
|
||||
u64 romfs_filedata_offset; // Relative to NCA start
|
||||
} PACKED romfs_ctx_t;
|
||||
|
||||
typedef struct {
|
||||
u8 type; // 1 = Dir, 2 = File
|
||||
u64 offset; // Relative to directory/file table, depending on type
|
||||
} PACKED romfs_browser_entry;
|
||||
|
||||
void generateCnmtXml(cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, char *out);
|
||||
|
||||
void convertNcaSizeToU64(const u8 size[0x6], u64 *out);
|
||||
|
@ -347,15 +398,17 @@ bool processNcaCtrSectionBlock(NcmContentStorage *ncmStorage, const NcmNcaId *nc
|
|||
|
||||
bool encryptNcaHeader(nca_header_t *input, u8 *outBuf, u64 outBufSize);
|
||||
|
||||
bool decryptNcaHeader(const u8 *ncaBuf, u64 ncaBufSize, nca_header_t *out, title_rights_ctx *rights_info, u8 *decrypted_nca_keys);
|
||||
bool decryptNcaHeader(const u8 *ncaBuf, u64 ncaBufSize, nca_header_t *out, title_rights_ctx *rights_info, u8 *decrypted_nca_keys, bool retrieveTitleKeyData);
|
||||
|
||||
bool processProgramNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, cnmt_xml_content_info *xml_content_info, nca_program_mod_data *output);
|
||||
|
||||
bool retrieveCnmtNcaData(nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, nca_cnmt_mod_data *output, title_rights_ctx *rights_info);
|
||||
bool retrieveCnmtNcaData(FsStorageId curStorageId, nspDumpType selectedNspDumpType, u8 *ncaBuf, cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, nca_cnmt_mod_data *output, title_rights_ctx *rights_info, bool replaceKeyArea);
|
||||
|
||||
bool patchCnmtNca(u8 *ncaBuf, u64 ncaBufSize, cnmt_xml_program_info *xml_program_info, cnmt_xml_content_info *xml_content_info, nca_cnmt_mod_data *cnmt_mod);
|
||||
|
||||
bool readRomFsEntriesFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys);
|
||||
bool readExeFsEntryFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys);
|
||||
|
||||
bool readRomFsEntryFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys);
|
||||
|
||||
bool generateNacpXmlFromNca(NcmContentStorage *ncmStorage, const NcmNcaId *ncaId, nca_header_t *dec_nca_header, u8 *decrypted_nca_keys, char **outBuf);
|
||||
|
||||
|
|
58
source/set_ext.c
Normal file
|
@ -0,0 +1,58 @@
|
|||
#include <switch/arm/atomics.h>
|
||||
#include <switch/services/sm.h>
|
||||
#include <switch/types.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "set_ext.h"
|
||||
|
||||
static Service g_setcalSrv;
|
||||
static u64 g_refCntCal;
|
||||
|
||||
Result setcalInitialize(void)
|
||||
{
|
||||
atomicIncrement64(&g_refCntCal);
|
||||
|
||||
if (serviceIsActive(&g_setcalSrv)) return MAKERESULT(Module_Libnx, LibnxError_AlreadyInitialized);
|
||||
|
||||
return smGetService(&g_setcalSrv, "set:cal");
|
||||
}
|
||||
|
||||
void setcalExit(void)
|
||||
{
|
||||
if (atomicDecrement64(&g_refCntCal) == 0) serviceClose(&g_setcalSrv);
|
||||
}
|
||||
|
||||
Result setcalGetEticketDeviceKey(void *key)
|
||||
{
|
||||
IpcCommand c;
|
||||
ipcInitialize(&c);
|
||||
ipcAddRecvBuffer(&c, key, 0x244, 0);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 cmd_id;
|
||||
} *raw;
|
||||
|
||||
raw = ipcPrepareHeader(&c, sizeof(*raw));
|
||||
|
||||
raw->magic = SFCI_MAGIC;
|
||||
raw->cmd_id = 21;
|
||||
|
||||
Result rc = serviceIpcDispatch(&g_setcalSrv);
|
||||
|
||||
if (R_SUCCEEDED(rc))
|
||||
{
|
||||
IpcParsedCommand r;
|
||||
ipcParse(&r);
|
||||
|
||||
struct {
|
||||
u64 magic;
|
||||
u64 result;
|
||||
} *resp = r.Raw;
|
||||
|
||||
rc = resp->result;
|
||||
}
|
||||
|
||||
return rc;
|
||||
}
|
17
source/set_ext.h
Normal file
|
@ -0,0 +1,17 @@
|
|||
#pragma once
|
||||
|
||||
#ifndef __SET_EXT_H__
|
||||
#define __SET_EXT_H__
|
||||
|
||||
#include <switch.h>
|
||||
|
||||
Result setcalInitialize(void);
|
||||
void setcalExit(void);
|
||||
|
||||
/**
|
||||
* @brief Gets the extended ETicket RSA-2048 Key from CAL0
|
||||
* @param key Pointer to 0x244-byte output buffer.
|
||||
*/
|
||||
Result setcalGetEticketDeviceKey(void *key);
|
||||
|
||||
#endif
|
1183
source/ui.c
33
source/ui.h
|
@ -22,6 +22,8 @@
|
|||
#define COMMON_MAX_ELEMENTS 8
|
||||
#define HFS0_MAX_ELEMENTS 14
|
||||
#define ROMFS_MAX_ELEMENTS 12
|
||||
#define SDCARD_MAX_ELEMENTS 4
|
||||
#define ORPHAN_MAX_ELEMENTS 12
|
||||
|
||||
#define OPTIONS_X_POS (35 * CHAR_PT_SIZE)
|
||||
|
||||
|
@ -31,6 +33,7 @@
|
|||
|
||||
#define NINTENDO_FONT_A "\xE0\xA0"
|
||||
#define NINTENDO_FONT_B "\xE0\xA1"
|
||||
#define NINTENDO_FONT_Y "\xE0\xA3"
|
||||
#define NINTENDO_FONT_L "\xE0\xA4"
|
||||
#define NINTENDO_FONT_R "\xE0\xA5"
|
||||
#define NINTENDO_FONT_ZL "\xE0\xA6"
|
||||
|
@ -44,6 +47,7 @@
|
|||
typedef enum {
|
||||
resultNone,
|
||||
resultShowMainMenu,
|
||||
resultShowGameCardMenu,
|
||||
resultShowXciDumpMenu,
|
||||
resultDumpXci,
|
||||
resultShowNspDumpMenu,
|
||||
|
@ -60,6 +64,13 @@ typedef enum {
|
|||
resultHfs0BrowserGetList,
|
||||
resultShowHfs0Browser,
|
||||
resultHfs0BrowserCopyFile,
|
||||
resultShowExeFsMenu,
|
||||
resultShowExeFsSectionDataDumpMenu,
|
||||
resultDumpExeFsSectionData,
|
||||
resultShowExeFsSectionBrowserMenu,
|
||||
resultExeFsSectionBrowserGetList,
|
||||
resultShowExeFsSectionBrowser,
|
||||
resultExeFsSectionBrowserCopyFile,
|
||||
resultShowRomFsMenu,
|
||||
resultShowRomFsSectionDataDumpMenu,
|
||||
resultDumpRomFsSectionData,
|
||||
|
@ -69,6 +80,9 @@ typedef enum {
|
|||
resultRomFsSectionBrowserChangeDir,
|
||||
resultRomFsSectionBrowserCopyFile,
|
||||
resultDumpGameCardCertificate,
|
||||
resultShowSdCardEmmcMenu,
|
||||
resultShowSdCardEmmcTitleMenu,
|
||||
resultShowSdCardEmmcOrphanPatchAddOnMenu,
|
||||
resultShowUpdateMenu,
|
||||
resultUpdateNSWDBXml,
|
||||
resultUpdateApplication,
|
||||
|
@ -77,6 +91,7 @@ typedef enum {
|
|||
|
||||
typedef enum {
|
||||
stateMainMenu,
|
||||
stateGameCardMenu,
|
||||
stateXciDumpMenu,
|
||||
stateDumpXci,
|
||||
stateNspDumpMenu,
|
||||
|
@ -93,6 +108,13 @@ typedef enum {
|
|||
stateHfs0BrowserGetList,
|
||||
stateHfs0Browser,
|
||||
stateHfs0BrowserCopyFile,
|
||||
stateExeFsMenu,
|
||||
stateExeFsSectionDataDumpMenu,
|
||||
stateDumpExeFsSectionData,
|
||||
stateExeFsSectionBrowserMenu,
|
||||
stateExeFsSectionBrowserGetList,
|
||||
stateExeFsSectionBrowser,
|
||||
stateExeFsSectionBrowserCopyFile,
|
||||
stateRomFsMenu,
|
||||
stateRomFsSectionDataDumpMenu,
|
||||
stateDumpRomFsSectionData,
|
||||
|
@ -102,11 +124,20 @@ typedef enum {
|
|||
stateRomFsSectionBrowserChangeDir,
|
||||
stateRomFsSectionBrowserCopyFile,
|
||||
stateDumpGameCardCertificate,
|
||||
stateSdCardEmmcMenu,
|
||||
stateSdCardEmmcTitleMenu,
|
||||
stateSdCardEmmcOrphanPatchAddOnMenu,
|
||||
stateUpdateMenu,
|
||||
stateUpdateNSWDBXml,
|
||||
stateUpdateApplication
|
||||
} UIState;
|
||||
|
||||
typedef enum {
|
||||
MENUTYPE_MAIN = 0,
|
||||
MENUTYPE_GAMECARD,
|
||||
MENUTYPE_SDCARD_EMMC
|
||||
} curMenuType;
|
||||
|
||||
void uiFill(int x, int y, int width, int height, u8 r, u8 g, u8 b);
|
||||
|
||||
void uiDrawIcon(const u8 *icon, int width, int height, int x, int y);
|
||||
|
@ -117,6 +148,8 @@ bool uiLoadJpgFromFile(const char *filename, int expectedWidth, int expectedHeig
|
|||
|
||||
void uiDrawString(const char *string, int x, int y, u8 r, u8 g, u8 b);
|
||||
|
||||
u32 uiGetStrWidth(char *string);
|
||||
|
||||
void uiRefreshDisplay();
|
||||
|
||||
void uiStatusMsg(const char *fmt, ...);
|
||||
|
|
1482
source/util.c
|
@ -6,6 +6,14 @@
|
|||
#include <switch.h>
|
||||
#include "nca.h"
|
||||
|
||||
#define OUTPUT_DUMP_BASE_PATH "sdmc:/nxdumptool/"
|
||||
#define XCI_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "XCI/")
|
||||
#define NSP_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "NSP/")
|
||||
#define HFS0_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "HFS0/")
|
||||
#define EXEFS_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "ExeFS/")
|
||||
#define ROMFS_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "RomFS/")
|
||||
#define CERT_DUMP_PATH (OUTPUT_DUMP_BASE_PATH "Certificate/")
|
||||
|
||||
#define KiB (1024.0)
|
||||
#define MiB (1024.0 * KiB)
|
||||
#define GiB (1024.0 * MiB)
|
||||
|
@ -21,8 +29,9 @@
|
|||
#define APPLICATION_PATCH_BITMASK (u64)0x800
|
||||
#define APPLICATION_ADDON_BITMASK (u64)0xFFFFFFFFFFFF0000
|
||||
|
||||
#define FILENAME_LENGTH 512
|
||||
#define FILENAME_MAX_CNT 20000
|
||||
#define FILENAME_BUFFER_SIZE (512 * FILENAME_MAX_CNT) // 10000 KiB
|
||||
#define FILENAME_BUFFER_SIZE (FILENAME_LENGTH * FILENAME_MAX_CNT) // 10000 KiB
|
||||
|
||||
#define NACP_APPNAME_LEN 0x200
|
||||
#define NACP_AUTHOR_LEN 0x100
|
||||
|
@ -91,6 +100,9 @@
|
|||
#define bswap_32(a) ((((a) << 24) & 0xff000000) | (((a) << 8) & 0xff0000) | (((a) >> 8) & 0xff00) | (((a) >> 24) & 0xff))
|
||||
#define round_up(x, y) ((x) + (((y) - ((x) % (y))) % (y))) // Aligns 'x' bytes to a 'y' bytes boundary
|
||||
|
||||
#define ORPHAN_ENTRY_TYPE_PATCH 1
|
||||
#define ORPHAN_ENTRY_TYPE_ADDON 2
|
||||
|
||||
typedef struct {
|
||||
u64 file_offset;
|
||||
u64 file_size;
|
||||
|
@ -115,6 +127,11 @@ typedef struct {
|
|||
double averageSpeed;
|
||||
} PACKED progress_ctx_t;
|
||||
|
||||
typedef struct {
|
||||
u32 index;
|
||||
u8 type; // 1 = Patch, 2 = AddOn
|
||||
} PACKED orphan_patch_addon_entry;
|
||||
|
||||
bool isGameCardInserted();
|
||||
|
||||
void fsGameCardDetectionThreadFunc(void *arg);
|
||||
|
@ -123,21 +140,25 @@ void delay(u8 seconds);
|
|||
|
||||
void formatETAString(u64 curTime, char *output, u32 outSize);
|
||||
|
||||
void convertTitleVersionToDecimal(u32 version, char *versionBuf, size_t versionBufSize);
|
||||
void initExeFsContext();
|
||||
|
||||
void removeIllegalCharacters(char *name);
|
||||
|
||||
void strtrim(char *str);
|
||||
|
||||
void freeStringsPtr(char **var);
|
||||
void freeExeFsContext();
|
||||
|
||||
void initRomFsContext();
|
||||
|
||||
void freeRomFsContext();
|
||||
|
||||
void freeGameCardInfo();
|
||||
void freeGlobalData();
|
||||
|
||||
void loadGameCardInfo();
|
||||
void convertTitleVersionToDecimal(u32 version, char *versionBuf, size_t versionBufSize);
|
||||
|
||||
void removeIllegalCharacters(char *name);
|
||||
|
||||
void createOutputDirectories();
|
||||
|
||||
void strtrim(char *str);
|
||||
|
||||
void loadTitleInfo();
|
||||
|
||||
bool getHfs0EntryDetails(u8 *hfs0Header, u64 hfs0HeaderOffset, u64 hfs0HeaderSize, u32 num_entries, u32 entry_idx, bool isRoot, u32 partitionIndex, u64 *out_offset, u64 *out_size);
|
||||
|
||||
|
@ -145,11 +166,15 @@ bool getPartitionHfs0Header(u32 partition);
|
|||
|
||||
bool getHfs0FileList(u32 partition);
|
||||
|
||||
u8 *getPartitionHfs0FileByName(FsStorage *gameCardStorage, const char *filename, u64 *outSize);
|
||||
bool getPartitionHfs0FileByName(FsStorage *gameCardStorage, const char *filename, u8 *outBuf, u64 outBufSize);
|
||||
|
||||
bool calculateExeFsExtractedDataSize(u64 *out);
|
||||
|
||||
bool calculateRomFsExtractedDataSize(u64 *out);
|
||||
|
||||
bool readProgramNcaRomFs(u32 appIndex);
|
||||
bool readProgramNcaExeFsOrRomFs(u32 appIndex, bool readRomFs);
|
||||
|
||||
bool getExeFsFileList();
|
||||
|
||||
bool getRomFsFileList(u32 dir_offset);
|
||||
|
||||
|
@ -157,12 +182,24 @@ int getSdCardFreeSpace(u64 *out);
|
|||
|
||||
void convertSize(u64 size, char *out, int bufsize);
|
||||
|
||||
void addStringToFilenameBuffer(const char *string, char **nextFilename);
|
||||
|
||||
char *generateDumpFullName();
|
||||
|
||||
char *generateNSPDumpName(nspDumpType selectedNspDumpType, u32 titleIndex);
|
||||
|
||||
void retrieveDescriptionForPatchOrAddOn(u64 titleID, u32 version, bool addOn, const char *prefix);
|
||||
|
||||
bool checkOrphanPatchOrAddOn(bool addOn);
|
||||
|
||||
void generateOrphanPatchOrAddOnList();
|
||||
|
||||
bool checkIfBaseApplicationHasPatchOrAddOn(u32 appIndex, bool addOn);
|
||||
|
||||
bool checkIfPatchOrAddOnBelongToBaseApplication(u32 titleIndex, u32 appIndex, bool addOn);
|
||||
|
||||
u32 retrieveFirstPatchOrAddOnIndexFromBaseApplication(u32 appIndex, bool addOn);
|
||||
|
||||
void waitForButtonPress();
|
||||
|
||||
void printProgressBar(progress_ctx_t *progressCtx, bool calcData, u64 chunkSize);
|
||||
|
@ -171,8 +208,6 @@ void setProgressBarError(progress_ctx_t *progressCtx);
|
|||
|
||||
void convertDataToHexString(const u8 *data, const u32 dataSize, char *outBuf, const u32 outBufSize);
|
||||
|
||||
void addStringToFilenameBuffer(const char *string, char **nextFilename);
|
||||
|
||||
void removeDirectory(const char *path);
|
||||
|
||||
void gameCardDumpNSWDBCheck(u32 crc);
|
||||
|
|