2024-12-08 19:16:34 +01:00
# include "fs_common.hpp"
2024-03-07 22:05:16 +01:00
# include "pxi.hpp"
# include "os.hpp"
# include "platform/crypto.hpp"
# include "platform/file_formats/ncch.hpp"
# include "platform/pxi.hpp"
# include <framework/exceptions.hpp>
# include "pxi_fs.hpp"
# include "pxi_fs_file_buffer_emu.hpp"
# include "range/v3/algorithm/transform.hpp"
# include <cryptopp/aes.h>
# include <cryptopp/modes.h>
# include <range/v3/algorithm/copy.hpp>
# include <range/v3/algorithm/equal.hpp>
# include <range/v3/algorithm/fill.hpp>
# include <range/v3/view/counted.hpp>
# include <range/v3/view/reverse.hpp>
# include <range/v3/view/zip.hpp>
# include <sstream>
# include <iomanip>
// Hardcoded size for PXI buffers.
// TODO: These buffers should only be 0x1000 bytes large, but our PXI tables currently may span more than that because they map individual pages as entries
const auto pxi_static_buffer_size = 0x2000 ;
namespace HLE {
// namespace OS {
namespace PXI {
std : : array < uint8_t , 16 > GenerateAESKey ( const std : : array < uint8_t , 16 > & key_x , const std : : array < uint8_t , 16 > & key_y ) {
std : : array < uint8_t , 16 > key_gen_constant = { 0x1f , 0xf9 , 0xe9 , 0xaa , 0xc5 , 0xfe , 0x04 , 0x08 , 0x02 , 0x45 , 0x91 , 0xdc , 0x5d , 0x52 , 0x76 , 0x8a } ;
std : : array < uint8_t , 16 > key = key_x ;
// ROL 2
for ( unsigned i = 0 , overflow = ( key [ 0 ] > > 6 ) ; i < key . size ( ) ; + + i ) {
auto new_overflow = key [ 15 - i ] > > 6 ;
key [ 15 - i ] < < = 2 ;
key [ 15 - i ] | = overflow ;
overflow = new_overflow ;
}
std : : transform ( key . begin ( ) , key . end ( ) , key_y . begin ( ) , key . begin ( ) , std : : bit_xor < > { } ) ; // (X ROL 2) XOR Y
// Add constant
for ( int i = key . size ( ) - 1 , carry = 0 ; i > = 0 ; - - i ) {
auto result = uint16_t { key [ i ] } + key_gen_constant [ i ] + carry ;
key [ i ] = static_cast < uint8_t > ( result ) ;
carry = result > > 8 ;
}
// ... ROR 41 == ROL 88 and ROR 1
std : : rotate ( key . begin ( ) , key . begin ( ) + 11 , key . end ( ) ) ;
for ( unsigned i = 0 , overflow = ( key . back ( ) & 1 ) ; i < key . size ( ) ; + + i ) {
auto new_overflow = key [ i ] & 1 ;
key [ i ] > > = 1 ;
key [ i ] | = overflow < < 7 ;
overflow = new_overflow ;
}
return key ;
}
using PAddr = OS : : PAddr ;
using Thread = OS : : Thread ;
PXIBuffer : : PXIBuffer ( const IPC : : StaticBuffer & other ) : IPC : : StaticBuffer ( other ) {
}
template < typename DataType >
DataType PXIBuffer : : Read ( Thread & thread , uint32_t offset ) const {
static_assert ( Meta : : is_same_v < DataType , uint8_t > | | Meta : : is_same_v < DataType , uint32_t > | | Meta : : is_same_v < DataType , uint64_t > , " " ) ;
if ( offset ! = ( offset & uint32_t { ~ uint32_t { sizeof ( DataType ) - 1 } } ) )
throw std : : runtime_error ( " Improper address alignment: Might cross the page boundary " ) ;
auto & process = thread . GetParentProcess ( ) ;
auto address = LookupAddress ( thread , offset ) ;
if ( std : : is_same < DataType , uint8_t > : : value )
return process . ReadPhysicalMemory ( address ) ;
else if ( std : : is_same < DataType , uint32_t > : : value )
return process . ReadPhysicalMemory32 ( address ) ;
else if ( std : : is_same < DataType , uint64_t > : : value )
return uint64_t { process . ReadPhysicalMemory32 ( address ) } | ( uint64_t { process . ReadPhysicalMemory32 ( address + 4 ) } < < uint64_t { 32 } ) ;
// TODO: Other sizes...
}
template uint8_t PXIBuffer : : Read ( Thread & , uint32_t ) const ;
template uint32_t PXIBuffer : : Read ( Thread & , uint32_t ) const ;
template uint64_t PXIBuffer : : Read ( Thread & , uint32_t ) const ;
template < typename DataType >
void PXIBuffer : : Write ( Thread & thread , uint32_t offset , DataType value ) const {
static_assert ( Meta : : is_same_v < DataType , uint8_t > | | Meta : : is_same_v < DataType , uint32_t > | | Meta : : is_same_v < DataType , uint64_t > , " " ) ;
if ( offset ! = ( offset & uint32_t { ~ uint32_t { sizeof ( DataType ) - 1 } } ) )
throw std : : runtime_error ( " Improper address alignment: Might cross the page boundary " ) ;
auto & process = thread . GetParentProcess ( ) ;
auto address = LookupAddress ( thread , offset ) ;
if ( std : : is_same < DataType , uint8_t > : : value )
return process . WritePhysicalMemory ( address , value ) ;
else if ( std : : is_same < DataType , uint32_t > : : value )
return process . WritePhysicalMemory32 ( address , value ) ;
else if ( std : : is_same < DataType , uint64_t > : : value ) {
process . WritePhysicalMemory32 ( address + 0 , value & 0xFFFFFFFF ) ;
process . WritePhysicalMemory32 ( address + 4 , value > > 32 ) ;
return ;
}
// TODO: Other sizes...
}
uint32_t PXIBuffer : : LookupAddress ( Thread & thread , uint32_t offset ) const {
auto & process = thread . GetParentProcess ( ) ;
if ( chunks . empty ( ) ) {
// Cache static buffer data
uint32_t table_addr = addr ;
while ( table_addr - addr < size ) {
if ( table_addr - addr > = pxi_static_buffer_size )
throw std : : runtime_error ( " Given PXI table exceeds PXI static buffer size " ) ;
// TODO: This currently assumes the page table is contiguous in memory!
// This is normally not an issue (since the PXI page table only occupies one page),
// but as an internal workaround we currently have our PXI services use two pages for the tables,
// so this assumption might break...
PAddr chunk_addr = process . ReadPhysicalMemory32 ( table_addr ) ;
PAddr current_chunk_size = process . ReadPhysicalMemory32 ( table_addr + 4 ) ;
// NOTE: The current PXI buffer memory access logic is optimized
// for the assumption that all memory chunks (other than the
// first and last one) are exactly 0x1000 bytes large. Since
// we currently control the generation of all PXI buffer
// descriptors, this is not an issue, but this assumption
// will need to be dropped once we low-level emulate the
// official ARM11 PXI module. To make sure this will not be
// forgotten, we add a safety check here.
if ( ! chunks . empty ( ) & & ! ( current_chunk_size = = 0x1000 | | table_addr - addr + 8 = = size ) )
throw std : : runtime_error ( fmt : : format ( " Invalid chunk of size {:#x} at address {:#x}: Apart from the first and last chunk, all chunks must be page-sized " ,
current_chunk_size , table_addr ) ) ;
if ( chunks . size ( ) < 2 )
chunk_size = current_chunk_size ;
chunks . emplace_back ( ChunkDescriptor { chunk_addr , current_chunk_size } ) ;
table_addr + = 8 ;
}
if ( chunks . empty ( ) )
throw std : : runtime_error ( " Malformed PXI buffer descriptor " ) ;
}
// Special-case for the first address since that one may be smaller than a page
PAddr first_chunk_size = chunks [ 0 ] . size_in_bytes ;
if ( offset < first_chunk_size )
return chunks [ 0 ] . start_addr + offset ;
// All chunks other than the first and last one are page-sized, so we
// rebase the offset to the second chunk address so that we can find the
// chunk index easily by dividing the new offset by the page size
offset - = first_chunk_size ;
auto chunk_index = 1 + ( offset / chunk_size ) ;
if ( chunk_index > = chunks . size ( ) )
throw std : : runtime_error ( fmt : : format ( " Trying to access PXI buffer at invalid offset {:#x} " , offset ) ) ;
auto & chunk = chunks [ chunk_index ] ;
PAddr current_chunk_size = chunk . size_in_bytes ;
if ( current_chunk_size < ( offset & ( chunk_size - 1 ) ) )
throw std : : runtime_error ( fmt : : format ( " Trying to access PXI buffer at invalid offset {:#x} (selected chunk size is {:#x}) " , offset , current_chunk_size ) ) ;
return chunk . start_addr + ( offset & ( chunk_size - 1 ) ) ;
}
template void PXIBuffer : : Write ( Thread & , uint32_t , uint8_t ) const ;
template void PXIBuffer : : Write ( Thread & , uint32_t , uint32_t ) const ;
template void PXIBuffer : : Write ( Thread & , uint32_t , uint64_t ) const ;
} // namespace PXI
// } // namespace OS
namespace PXI {
namespace FS {
using OS : : Result ;
using OS : : FakeThread ;
using OS : : Thread ;
using OS : : RESULT_OK ;
using OS : : ThreadPrinter ;
using OSImpl = OS : : OS ;
std : : string PrintEntirePXIBuffer ( Thread & thread , const PXIBuffer & buf , size_t size , const char * prefix ) {
// TODO: Re-enable, or something
std : : stringstream ss ;
// ss << '\n'; // whatever, just make it easy to filter out the mess that spdlog adds at the beginning
// for (size_t i = 0; i < size; ++i) {
// if ((i % 4) == 0)
// ss << prefix << " ";
// ss << std::hex << std::setw(2) << std::setfill('0') << +buf.Read<uint8_t>(thread, i);
// if ((i % 4) == 3)
// ss << '\n';
// }
return ss . str ( ) ;
}
void FileBufferInHostMemory : : Write ( char * source , uint32_t num_bytes ) {
if ( num_bytes > size ) {
throw std : : runtime_error ( " Writing out of FileBuffer bounds " ) ;
}
memcpy ( memory , source , num_bytes ) ;
}
void File : : Fail ( ) {
throw std : : runtime_error ( " Unspecified error in PXI file operation " ) ;
}
2024-03-23 17:05:55 +01:00
OS : : ResultAnd < > File : : Open ( FileContext & context , OpenFlags flags ) {
return Open ( context , flags ) ;
2024-03-07 22:05:16 +01:00
}
OS : : ResultAnd < uint32_t > File : : Read ( FileContext & , uint64_t offset , uint32_t num_bytes , FileBuffer & & dest ) {
Fail ( ) ;
}
OS : : ResultAnd < uint32_t > File : : Overwrite ( FakeThread & thread , uint64_t offset , uint32_t num_bytes , const PXIBuffer & data ) {
Fail ( ) ;
}
std : : pair < Result , std : : unique_ptr < File > > Archive : : OpenFile ( FakeThread & thread , const HLE : : CommonPath & path ) {
auto handler = [ & , this ] ( const auto & path ) { return OpenFile ( thread , path ) ; } ;
return path . Visit ( handler , handler ) ; ;
}
std : : pair < Result , std : : unique_ptr < File > > Archive : : OpenFile ( FakeThread & thread , const HLE : : Utf8PathType & path ) {
thread . GetLogger ( ) - > info ( " OpenFile on dummy archive: Stubbed " ) ;
return { 0 , nullptr } ;
}
std : : pair < Result , std : : unique_ptr < File > > Archive : : OpenFile ( FakeThread & thread , const HLE : : BinaryPathType & path ) {
thread . GetLogger ( ) - > info ( " OpenFile on dummy archive: Stubbed " ) ;
return { 0 , nullptr } ;
}
class Archive0x567890b0 final : public Archive {
// Unimplemented
} ;
class ArchiveSystemSaveData final : public Archive {
std : : string base_path ;
std : : pair < Result , std : : unique_ptr < File > > OpenFile ( FakeThread & thread , const HLE : : BinaryPathType & path ) override {
if ( path . size ( ) ! = 8 )
throw std : : runtime_error ( " Invalid file path for SystemSaveData archive: Expected 8 bytes describing the save id " ) ;
auto file_path = base_path ;
boost : : endian : : little_uint64_buf_t saveid_buf ;
memcpy ( & saveid_buf , path . data ( ) , path . size ( ) ) ;
uint64_t saveid = saveid_buf . value ( ) ;
file_path + = fmt : : format ( " {:08x} " , saveid & 0xFFFFFFFF ) ;
file_path + = " / " ;
file_path + = fmt : : format ( " {:08x} " , saveid > > 32 ) ;
// TODO: Drop now unused and unimplemented HostFile::PatchHash
auto file = std : : make_unique < HostFile > ( file_path , HostFile : : PatchHash ) ;
return std : : make_pair ( RESULT_OK , std : : move ( file ) ) ;
}
public :
2024-03-24 10:53:38 +01:00
ArchiveSystemSaveData ( Settings : : Settings & settings , uint32_t media_type ) {
2024-03-07 22:05:16 +01:00
switch ( media_type ) {
case 0 :
// TODO: This ID is actually generated from data in moveable.sed
std : : array < uint8_t , 0x10 > id0 ;
ranges : : fill ( id0 , 0 ) ;
2024-03-24 10:53:38 +01:00
base_path = ( GetRootDataDirectory ( settings ) / " data " / fmt : : format ( " {:02x} " , fmt : : join ( id0 , " " ) ) / " sysdata " ) . string ( ) ;
2024-03-07 22:05:16 +01:00
break ;
default :
throw std : : runtime_error ( " Unknown media type " ) ;
}
}
} ;
class AESDecryptingFileBufferView : public FileBuffer {
FileBuffer & buffer ;
CryptoPP : : CTR_Mode < CryptoPP : : AES > : : Decryption & dec ;
public :
AESDecryptingFileBufferView ( FileBuffer & buffer , CryptoPP : : CTR_Mode < CryptoPP : : AES > : : Decryption & dec ) : buffer ( buffer ) , dec ( dec ) {
}
void Write ( char * source , uint32_t num_bytes ) override {
// TODO: Use a CryptoPP pipeline instead to avoid the heap allocation?
std : : vector < CryptoPP : : byte > data ( num_bytes ) ;
dec . ProcessData ( data . data ( ) , reinterpret_cast < const CryptoPP : : byte * > ( source ) , num_bytes ) ;
buffer . Write ( reinterpret_cast < char * > ( data . data ( ) ) , num_bytes ) ;
}
} ;
class AESEncryptedFile final : public File {
std : : unique_ptr < File > file ;
uint64_t aes_offset ;
std : : array < uint8_t , 16 > key ;
std : : array < uint8_t , 16 > iv ;
public :
AESEncryptedFile ( std : : unique_ptr < File > file , uint64_t aes_offset , const std : : array < uint8_t , 16 > & key , const std : : array < uint8_t , 16 > & iv )
: file ( std : : move ( file ) ) , aes_offset ( aes_offset ) , key ( key ) , iv ( iv ) {
}
2024-03-23 17:05:55 +01:00
OS : : ResultAnd < > Open ( FileContext & context , OpenFlags flags ) override {
return file - > Open ( context , flags ) ;
2024-03-07 22:05:16 +01:00
}
OS : : ResultAnd < uint64_t > GetSize ( FileContext & context ) override {
return file - > GetSize ( context ) ;
}
OS : : ResultAnd < uint32_t > Read ( FileContext & context , uint64_t offset , uint32_t num_bytes , FileBuffer & & dest ) override {
if ( num_bytes = = 0 ) {
return std : : make_pair ( OS : : RESULT_OK , uint32_t { 0 } ) ;
}
CryptoPP : : CTR_Mode < CryptoPP : : AES > : : Decryption dec ;
dec . SetKeyWithIV ( key . data ( ) , sizeof ( key ) , iv . data ( ) ) ;
dec . Seek ( aes_offset + offset ) ;
AESDecryptingFileBufferView dest_view { dest , dec } ;
return file - > Read ( context , offset , num_bytes , std : : move ( dest_view ) ) ;
}
} ;
class DummyExeFSFile final : public File {
public :
DummyExeFSFile ( ) { }
2024-03-23 17:05:55 +01:00
OS : : ResultAnd < > Open ( FileContext & , OpenFlags ) override {
2024-03-07 22:05:16 +01:00
return { RESULT_OK } ;
}
OS : : ResultAnd < uint64_t > GetSize ( FileContext & ) override {
return { RESULT_OK , uint64_t { 1 } } ;
}
OS : : ResultAnd < uint32_t > Read ( FileContext & , uint64_t off , uint32_t num_bytes , FileBuffer & & buf ) override {
if ( off ! = 0 ) {
throw std : : runtime_error ( " ERROR " ) ;
}
char null = 0 ;
for ( ; off ! = num_bytes ; + + off ) {
buf . Write ( & null , 1 ) ;
}
return { RESULT_OK , uint32_t { num_bytes } } ;
}
void Close ( ) override {
}
} ;
// @pre "Open" must have been called on ncch before using this function
std : : unique_ptr < File > NCCHOpenExeFSSection ( spdlog : : logger & logger , FileContext & file_context , const KeyDatabase & keydb , std : : unique_ptr < File > ncch , uint8_t sub_file_type , std : : basic_string_view < uint8_t > requested_exefs_section_name ) {
auto ncch_offset = uint64_t { 0 } ;
auto read_host_file = [ & ] ( char * dest , size_t size ) {
ncch - > Read ( file_context , ncch_offset , static_cast < uint32_t > ( size ) , FileBufferInHostMemory { dest , static_cast < uint32_t > ( size ) } ) ;
ncch_offset + = size ;
} ;
auto ncch_header = FileFormat : : SerializationInterface < FileFormat : : NCCHHeader > : : Load ( read_host_file ) ;
if ( memcmp ( ncch_header . magic . data ( ) , " NCCH " , sizeof ( ncch_header . magic ) ) ! = 0 ) {
throw std : : runtime_error ( " NCCH header word not found. Corrupt ROM? " ) ;
}
bool is_encrypted = ! ( ncch_header . flags & 4 ) ;
std : : array < uint8_t , 16 > keys [ 2 ] { } ; // 0: Secure1 (system version < 7.x), 1: Secure2 (system version >= 7.x)
std : : array < uint8_t , 16 > iv { } ;
if ( ! is_encrypted ) {
// No encryption
} else if ( ( ncch_header . flags & 1 ) & & ( ncch_header . program_id > > 32 ) ! = 0x40130 ) {
// Encrypt with an all-zeroes key
} else if ( is_encrypted ) {
if ( ncch_header . crypto_method > 1 ) {
throw Mikage : : Exceptions : : Invalid ( " Invalid NCCH encryption method " ) ;
}
const std : : array < uint8_t , 16 > & key_x_0x25 = keydb . aes_slots [ 0x25 ] . x . value ( ) ;
const std : : array < uint8_t , 16 > & key_x_0x2c = keydb . aes_slots [ 0x2c ] . x . value ( ) ;
std : : array < uint8_t , 16 > key_y ;
memcpy ( key_y . data ( ) , & ncch_header , sizeof ( key_y ) ) ;
keys [ 0 ] = GenerateAESKey ( key_x_0x2c , key_y ) ;
keys [ 1 ] = GenerateAESKey ( key_x_0x25 , key_y ) ;
// First 8 bytes are the partition id interpreted as big-endian
// TODO: Should be version dependent? version==1 has different behavior, apparently
memcpy ( iv . data ( ) , & ncch_header . partition_id , sizeof ( ncch_header . partition_id ) ) ;
std : : reverse ( iv . begin ( ) , iv . begin ( ) + 8 ) ;
}
// TODO: Check if seed encryption is enabled
switch ( sub_file_type ) {
case 0 : // RomFS
{
iv [ 8 ] = 3 ;
// TODO: Cleanup. Currently adapted from FS-HLE code
// TODO: Verify the given archive even has a RomFS!
auto offset = ncch_header . romfs_offset . ToBytes ( ) ;
auto romfs_size = ncch_header . romfs_size . ToBytes ( ) ;
auto ncch_start = 0 ;
auto ivfc_start = ncch_start + static_cast < std : : ifstream : : off_type > ( offset ) ;
// TODO: Apply decryption to this check
char ivfc_header [ 5 ] { } ;
ncch - > Read ( file_context , ivfc_start , 4 , FileBufferInHostMemory { ivfc_header , 4 } ) ;
if ( is_encrypted & & ivfc_header = = std : : string_view { " IVFC " } ) {
// Signature present even though encryption flags are set => override
is_encrypted = false ;
}
if ( ivfc_header ! = std : : string_view { " IVFC " } & & ! is_encrypted ) {
throw Mikage : : Exceptions : : Invalid ( " Invalid RomFS " ) ;
}
/* std::cerr << "ivfc offset: 0x" << std::hex << ivfc_start-ncch_start << std::endl;
ncch . seekg ( ivfc_start + static_cast < std : : ifstream : : off_type > ( 0x3c ) ) ; // offset in IVFC to level 3 entry offset
boost : : endian : : little_uint32_t level3_offset ;
ncch . read ( reinterpret_cast < char * > ( & level3_offset ) , sizeof ( level3_offset ) ) ;
std : : cerr < < " level3 offset: 0x " < < std : : hex < < level3_offset < < std : : endl ; */
// Close so that FileView can open it again (TODO: Can we design this less stupidly?)
ncch - > Close ( /*thread*/ ) ;
std : : unique_ptr < File > ret = std : : make_unique < FileView > ( std : : move ( ncch ) , ivfc_start , romfs_size ) ;
if ( is_encrypted ) {
ret = std : : make_unique < AESEncryptedFile > ( std : : move ( ret ) , 0 , keys [ ncch_header . crypto_method ] , iv ) ;
}
return ret ;
}
// Open the ExeFS section specified by the remainder of the file path
case 1 :
case 2 :
{
// Titles built for firmware 5.0.0 or newer don't store the logo in the
// ExeFS but instead in the plaintext NCCH region.
const char logo_name [ 8 ] = { 0x6c , 0x6f , 0x67 , 0x6f } ;
if ( ranges : : equal ( logo_name , requested_exefs_section_name ) & &
ncch_header . logo_offset . ToBytes ( ) & & ncch_header . logo_size . ToBytes ( ) ) {
return std : : make_unique < FileView > ( std : : move ( ncch ) , ncch_header . logo_offset . ToBytes ( ) , ncch_header . logo_size . ToBytes ( ) , ncch_header . logo_sha256 ) ;
}
iv [ 8 ] = 2 ;
// TODO: Verify the given archive even has an ExeFS!
ncch_offset = ncch_header . exefs_offset . ToBytes ( ) ;
FileFormat : : ExeFSHeader exefs_header ; // = FileFormat::SerializationInterface<FileFormat::ExeFSHeader>::Load(read_host_file);
read_host_file ( reinterpret_cast < char * > ( & exefs_header ) , sizeof ( exefs_header ) ) ; // TODO: Use SerializationInterface
// TODO: Look for common section strings instead
if ( is_encrypted & & exefs_header . files [ 0 ] . offset = = 0 ) {
// Looks like a decrypted ExeFS => override encryption flags
is_encrypted = false ;
}
if ( is_encrypted ) {
// ExeFS header always uses Secure1 for encryption
CryptoPP : : CTR_Mode < CryptoPP : : AES > : : Decryption dec ;
dec . SetKeyWithIV ( keys [ 0 ] . data ( ) , sizeof ( keys [ 0 ] ) , iv . data ( ) ) ;
dec . ProcessData ( reinterpret_cast < CryptoPP : : byte * > ( & exefs_header ) , reinterpret_cast < const CryptoPP : : byte * > ( & exefs_header ) , sizeof ( exefs_header ) ) ;
}
// SerializationInterface not used currently because it triggers subcomplete reads on 3DSX files
// auto exefs_header = FileFormat::SerializationInterface<FileFormat::ExeFSHeader>::Load(read_host_file);
for ( auto exefs_section_and_hash : ranges : : views : : zip ( exefs_header . files , exefs_header . hashes | ranges : : views : : reverse ) ) {
auto & exefs_section = std : : get < 0 > ( exefs_section_and_hash ) ;
auto & section_hash = std : : get < 1 > ( exefs_section_and_hash ) ;
if ( ranges : : equal ( exefs_section . name , requested_exefs_section_name ) ) {
// TODO: Print the actual section name in the log
logger . info ( " Found section at post-exefs-header offset {:#x} with {:#x} bytes (ExeFS at {:#x}) " ,
exefs_section . offset , exefs_section . size_bytes , ncch_offset ) ;
auto data_offset = ncch_header . exefs_offset . ToBytes ( ) + FileFormat : : ExeFSHeader : : Tags : : expected_serialized_size + exefs_section . offset ;
// Close so that FileView can open it again (TODO: Can we design this less stupidly?)
ncch - > Close ( /*thread*/ ) ;
std : : unique_ptr < File > ret = std : : make_unique < FileView > ( std : : move ( ncch ) , data_offset , exefs_section . size_bytes , section_hash ) ;
if ( is_encrypted ) {
// TODO: The former check doesn't work!
// const auto& key = (!ranges::equal(requested_exefs_section_name, "icon") && !ranges::equal(requested_exefs_section_name, "banner")) ? keys[ncch_header.crypto_method] : keys[0];
// const auto& key = (requested_exefs_section_name[0] != 'i' && requested_exefs_section_name[0] != 'l' && requested_exefs_section_name[0] != 'b') ? keys[ncch_header.crypto_method] : keys[0];
const auto & key = keys [ ncch_header . crypto_method ? ( requested_exefs_section_name [ 0 ] = = ' . ' ) : 0 ] ;
ret = std : : make_unique < AESEncryptedFile > ( std : : move ( ret ) , FileFormat : : ExeFSHeader : : Tags : : expected_serialized_size + exefs_section . offset , key , iv ) ;
}
return ret ;
}
}
// No match => Panic
throw std : : runtime_error ( fmt : : format ( " Couldn't find section {} in ExeFS " , reinterpret_cast < const char * > ( requested_exefs_section_name . data ( ) ) ) ) ;
}
default :
throw std : : runtime_error ( fmt : : format ( " Unknown ArchiveProgramDataFromTitleId sub file type {:#x} " , sub_file_type ) ) ;
}
}
std : : unique_ptr < File > OpenNCCHSubFile ( Thread & thread , Platform : : FS : : ProgramInfo program_info , uint32_t content_id , uint32_t sub_file_type , std : : basic_string_view < uint8_t > file_path , Loader : : GameCard * gamecard ) {
std : : unique_ptr < File > ncch ;
if ( program_info . media_type = = 0 ) {
// If this title is HLEed, return a dummy ExeFS
auto hle_title_name = GetHLEModuleName ( program_info . program_id ) ;
if ( hle_title_name & & thread . GetOS ( ) . ShouldHLEProcess ( * hle_title_name ) ) {
if ( content_id ! = Meta : : to_underlying ( Loader : : NCSDPartitionId : : Executable ) | |
sub_file_type ! = 1 ) {
throw std : : runtime_error ( " Reading this type of NCCH data is not supported for HLEed titles " ) ;
}
return std : : make_unique < DummyExeFSFile > ( ) ;
}
2024-03-24 10:53:38 +01:00
auto ncch_filename = GetRootDataDirectory ( thread . GetOS ( ) . settings ) /
fmt : : format ( " {:08x}/{:08x}/content/{:08x}.cxi " ,
program_info . program_id > > 32 , program_info . program_id & 0xFFFFFFFF , content_id ) ;
ncch = std : : make_unique < HostFile > ( ncch_filename . string ( ) , HostFile : : Default ) ;
2024-03-07 22:05:16 +01:00
} else if ( gamecard & & program_info . media_type = = 2 ) {
if ( content_id > Meta : : to_underlying ( Loader : : NCSDPartitionId : : UpdateData ) ) {
throw Mikage : : Exceptions : : Invalid ( " Invalid NCSD partition index {} " , content_id ) ;
}
auto ncch_opt = gamecard - > GetPartitionFromId ( static_cast < Loader : : NCSDPartitionId > ( content_id ) ) ;
if ( ! ncch_opt )
throw std : : runtime_error ( " Couldn't open gamecard image " ) ;
ncch = * std : : move ( ncch_opt ) ;
} else {
ValidateContract ( false ) ;
}
FileContext file_context { * thread . GetLogger ( ) } ;
2024-03-23 17:05:55 +01:00
auto result = ncch - > OpenReadOnly ( file_context ) ;
2024-03-07 22:05:16 +01:00
if ( std : : get < 0 > ( result ) ! = RESULT_OK ) {
throw std : : runtime_error ( fmt : : format ( " Tried to access non-existing title {:#x} from emulated NAND. \n \n Please dump the title from your 3DS and install it manually to this path: \n {} " ,
program_info . program_id , ( char * ) file_path . data ( ) ) ) ;
}
return NCCHOpenExeFSSection ( * thread . GetLogger ( ) , file_context ,
thread . GetParentProcess ( ) . interpreter_setup . keydb ,
std : : move ( ncch ) , sub_file_type , file_path ) ;
}
namespace {
/**
* Interface for the application to access RomFS and ExeFS data from any
* program ' s NCCH . The FS interface takes a Process Manager program handle
* to identify different programs , which we translate to a ProgramInfo .
*/
class ArchiveProgramDataFromTitleId : public Archive {
Platform : : FS : : ProgramInfo program_info ;
Loader : : GameCard * gamecard ; // nullptr if none present
protected :
std : : pair < Result , std : : unique_ptr < File > > OpenFile ( FakeThread & thread , const HLE : : BinaryPathType & path ) override {
if ( path . size ( ) ! = 20 )
throw std : : runtime_error ( " Invalid file path for ProgramDataInternal archive: Expected 20 bytes " ) ;
// File path structure (according to archive 0x234567a):
// * NCCH (0) or save data (1)
// * TMD content index or NCSD partition index
// * 0 for romfs (and for save data), 1 for exefs code section, 2 for exefs non-code section
// * ExeFS section name (two words)
std : : array < boost : : endian : : little_uint32_buf_t , 3 > file_path_u32 ;
memcpy ( file_path_u32 . data ( ) , path . data ( ) , sizeof ( file_path_u32 ) ) ;
uint32_t is_save_data = file_path_u32 [ 0 ] . value ( ) ;
uint32_t content_id = file_path_u32 [ 1 ] . value ( ) ; // NCSD partition index for game cards
uint32_t sub_file_type = file_path_u32 [ 2 ] . value ( ) ;
if ( is_save_data ) {
throw Mikage : : Exceptions : : NotImplemented ( " Cannot open save data with this archive yet " ) ;
}
if ( sub_file_type > 2 ) {
// TODO: Value 5 has been observed when booting IronFall: Invasion
throw Mikage : : Exceptions : : Invalid ( " Invalid sub file type {} " , sub_file_type ) ;
}
auto sub_file = OpenNCCHSubFile ( thread , program_info , content_id , sub_file_type , std : : basic_string_view < uint8_t > { path . data ( ) + 0xc , 8 } , gamecard ) ;
return std : : make_pair ( RESULT_OK , std : : move ( sub_file ) ) ;
}
public :
ArchiveProgramDataFromTitleId ( const Platform : : FS : : ProgramInfo & program_info , Loader : : GameCard * gamecard ) : program_info ( program_info ) , gamecard ( gamecard ) {
if ( program_info . media_type = = 2 ) {
if ( ! gamecard ) {
throw Mikage : : Exceptions : : Invalid ( " Attempted to open game card archive without a game card inserted " ) ;
}
} else if ( program_info . media_type ! = 0 ) {
throw Mikage : : Exceptions : : NotImplemented ( " Unknown media type {} (SD is not supported yet) " , program_info . media_type ) ;
}
}
virtual ~ ArchiveProgramDataFromTitleId ( ) = default ;
} ;
class ArchiveProgramDataFromProgramId final : public ArchiveProgramDataFromTitleId {
std : : pair < Result , std : : unique_ptr < File > > OpenFile ( FakeThread & thread , const HLE : : BinaryPathType & path ) override {
if ( path . size ( ) ! = 12 )
throw std : : runtime_error ( " Invalid file path for ProgramData archive: Expected 12 bytes " ) ;
// Forward this call to ArchiveProgramDataFromTitleId::OpenFile using
// a new path:
// * Word 0: NCCH (0) or save data (1)
// * Word 1: TMD content index or NCSD partition index
// * Word 2: 0 for romfs (and for save data), 1 for exefs code section, 2 for exefs non-code section
// * Words 3+4: ExeFS section name
HLE : : BinaryPathType new_file_path ;
new_file_path . resize ( 20 ) ;
// First word is zero (=NCCH access)
// Second word is zero (=first content)
// The remaining data is copied verbatimly
ranges : : copy ( ranges : : views : : counted ( path . data ( ) , static_cast < std : : ptrdiff_t > ( path . size ( ) ) ) , new_file_path . begin ( ) + 8 ) ;
return ArchiveProgramDataFromTitleId : : OpenFile ( thread , new_file_path ) ;
}
public :
ArchiveProgramDataFromProgramId ( const Platform : : FS : : ProgramInfo & program_info , Loader : : GameCard * gamecard ) : ArchiveProgramDataFromTitleId ( program_info , gamecard ) {
}
} ;
} // anonymous namespace
static std : : tuple < Result , uint64_t > OpenFile ( FakeThread & thread , Context & context , uint32_t transaction , uint64_t archive_handle ,
uint32_t path_type , uint32_t path_size , uint32_t flags ,
uint32_t attributes , const PXIBuffer & path ) {
thread . GetLogger ( ) - > info ( " {}received OpenFile with archive_handle={:#x}, transaction={:#x}, path_type={:#x}, path_size={:#x}, flags={:#x}, attributes={:#x} " ,
ThreadPrinter { thread } , archive_handle , transaction , path_type , path . size , flags , attributes ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , path , path_size , " PXIPXI recv " ) ) ;
auto archive_it = context . archives . find ( archive_handle ) ;
if ( archive_it = = context . archives . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto & archive = archive_it - > second ;
// TODO: Support attributes other than zero! (Currently, this restricts support to non-hidden and writeable files, and excludes support for directories and archives)
if ( attributes ! = 0 ) {
throw Mikage : : Exceptions : : NotImplemented ( " Attribute combination {:#x} not supported " , attributes ) ;
}
HLE : : CommonPath common_path ( thread , path_type , path_size , path ) ;
Result result ;
std : : unique_ptr < File > file ;
std : : tie ( result , file ) = archive - > OpenFile ( thread , common_path ) ;
if ( result ! = RESULT_OK ) {
thread . GetLogger ( ) - > error ( " Failed to get file open handler " ) ;
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
}
// TODO: Respect flags & ~0x4 ... in particular, write/read-only!
if ( file ) { // TODO: Remove this check! We only do this so that we don't need to implement archive 0x56789a0b0 properly...
FileContext file_context { * thread . GetLogger ( ) } ;
2024-09-12 19:25:33 +02:00
std : : tie ( result ) = file - > Open ( file_context , { . create = ( flags & 0x4 ) ! = 0 } ) ;
2024-03-07 22:05:16 +01:00
}
if ( result ! = RESULT_OK ) {
thread . GetLogger ( ) - > warn ( " Failed to open file " ) ;
return std : : make_tuple ( result , 0 ) ;
}
auto file_handle = context . next_file_handle + + ;
context . files . emplace ( std : : make_pair ( file_handle , std : : move ( file ) ) ) ;
return std : : make_tuple ( result , file_handle ) ;
}
static std : : tuple < Result > DeleteFile ( FakeThread & thread , Context & context , uint32_t transaction , uint64_t archive_handle ,
uint32_t path_type , uint32_t path_size , const PXIBuffer & path ) {
thread . GetLogger ( ) - > info ( " {}received STUBBED DeleteFile with archive_handle={:#x}, transaction={:#x}, path_type={:#x}, path_size={:#x} " ,
ThreadPrinter { thread } , archive_handle , path_type , path_size , path . size ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , path , path_size , " PXIPXI recv " ) ) ;
// TODO: Implement!
std : : string binary_path ;
for ( uint32_t offset = 0 ; offset < path_size ; + + offset )
binary_path + = fmt : : format ( " {:02x} " , + path . Read < uint8_t > ( thread , offset ) ) ;
thread . GetLogger ( ) - > info ( binary_path ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static std : : tuple < Result , uint32_t > ReadFile ( FakeThread & thread , Context & context , FileHandle file_handle , uint64_t offset , uint32_t num_bytes , const PXIBuffer & dest ) {
thread . GetLogger ( ) - > info ( " {}received ReadFile with file_handle={:#x}: {:#x} bytes from offset={:#x} " ,
ThreadPrinter { thread } , file_handle , num_bytes , offset ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto & file = file_it - > second ;
thread . GetLogger ( ) - > info ( " File kind: {} " , typeid ( * file ) . name ( ) ) ;
FileContext file_context { * thread . GetLogger ( ) } ;
auto result_and_bytesread = file - > Read ( file_context , offset , num_bytes , FileBufferInEmulatedMemory { thread , dest } ) ;
if ( std : : get < 0 > ( result_and_bytesread ) ! = RESULT_OK /*|| std::get<1>(result_and_bytesread) != num_bytes*/ /* TODO: TESTING ONLY: Works around lack of promo video */ )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , dest , num_bytes , " PXIPXI send " ) ) ;
return result_and_bytesread ;
}
static std : : tuple < Result > GetFileSHA256 ( FakeThread & thread , Context & context , FileHandle file_handle , uint32_t buffer_size , const PXIBuffer & dest ) {
thread . GetLogger ( ) - > info ( " {}received GetFileSHA256 with file_handle={:#x} " ,
ThreadPrinter { thread } , file_handle ) ;
if ( buffer_size ! = 0x20 )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto * file_view = dynamic_cast < FileView * > ( file_it - > second . get ( ) ) ;
auto hash = file_view - > GetFileHash ( thread ) ;
std : : string hash_string ;
for ( auto i = 0 ; i < hash . size ( ) ; + + i ) {
dest . Write < uint8_t > ( thread , i , hash [ i ] ) ;
hash_string + = fmt : : format ( " {:02x} " , hash [ i ] ) ;
}
thread . GetLogger ( ) - > info ( " Returning hash {} " , hash_string ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , dest , buffer_size , " PXIPXI send " ) ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static std : : tuple < Result , uint32_t > WriteFile ( FakeThread & thread , Context & context , FileHandle file_handle ,
uint64_t offset , uint32_t num_bytes , uint32_t flags ,
const PXIBuffer & input ) {
thread . GetLogger ( ) - > info ( " {}received WriteFile with file_handle={:#x}: {:#x} bytes to offset={:#x}, flags={:#x} " ,
ThreadPrinter { thread } , file_handle , num_bytes , offset , flags ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , input , num_bytes , " PXIPXI recv " ) ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto & file = file_it - > second ;
// TODO: Respect the flush flags
auto result_and_byteswritten = file - > Overwrite ( thread , offset , num_bytes , input ) ;
if ( std : : get < 0 > ( result_and_byteswritten ) ! = RESULT_OK )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
return result_and_byteswritten ;
}
static std : : tuple < Result > CalcSavegameMAC ( FakeThread & thread , Context & context , FileHandle file_handle ,
uint32_t output_size , uint32_t input_size ,
const PXIBuffer & input , const PXIBuffer & output ) {
thread . GetLogger ( ) - > info ( " {}received CalcSavegameMAC with file_handle={:#x}, input_size={:#x}, output_size={:#x}, input_paddr={:#x}, output_paddr={:#x} " ,
ThreadPrinter { thread } , file_handle , input_size , output_size , input . addr , output . addr ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , input , input_size , " PXIPXI recv " ) ) ;
// TODO: Properly compute this... Currently, we just return the first 0x10 bytes from the input, which should usually be the expected MAC
for ( unsigned offset = 0 ; offset < output_size ; + + offset ) {
// output.Write<uint8_t>(thread, offset, (offset < 0x10) ? input.Read<uint8_t>(thread, offset) : 0x00);
// Copying the XDS hack!
output . Write < uint8_t > ( thread , offset , 0x11 ) ;
}
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , output , output_size , " PXIPXI send " ) ) ;
return std : : make_tuple ( 0 ) ;
}
static std : : tuple < Result , uint64_t > GetFileSize ( FakeThread & thread , Context & context , uint64_t file_handle ) {
thread . GetLogger ( ) - > info ( " {}received GetFileSize with file_handle={:#x} " ,
ThreadPrinter { thread } , file_handle ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto & file = file_it - > second ;
auto file_context = FileContext { * thread . GetLogger ( ) } ;
auto result_and_size = file - > GetSize ( file_context ) ;
if ( std : : get < 0 > ( result_and_size ) ! = RESULT_OK )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
return result_and_size ;
}
static std : : tuple < Result > SetFileSize ( FakeThread & thread , Context & context , uint64_t new_size , uint64_t file_handle ) {
thread . GetLogger ( ) - > info ( " {}received SetFileSize with file_handle={:#x} and new_size={:#x} " ,
ThreadPrinter { thread } , file_handle , new_size ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
auto & file = file_it - > second ;
Result result ;
std : : tie ( result ) = file - > SetSize ( thread , new_size ) ;
if ( result ! = RESULT_OK )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
return std : : make_tuple ( result ) ;
}
static std : : tuple < Result > CloseFile ( FakeThread & thread , Context & context , uint64_t file_handle ) {
thread . GetLogger ( ) - > info ( " {}received CloseFile with file_handle={:#x} " ,
ThreadPrinter { thread } , file_handle ) ;
auto file_it = context . files . find ( file_handle ) ;
if ( file_it = = context . files . end ( ) )
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
context . files . erase ( file_it ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static std : : tuple < Result , uint64_t > OpenArchive ( FakeThread & thread , Context & context ,
uint32_t archive_id , uint32_t path_type ,
uint32_t path_size , const PXIBuffer & path ) {
thread . GetLogger ( ) - > info ( " {}received OpenArchive with archive_id={:#x}, path_type={:#x}, path_size={:#x} " ,
ThreadPrinter { thread } , archive_id , path_type , path_size ) ;
thread . GetLogger ( ) - > info ( " {} " , PrintEntirePXIBuffer ( thread , path , path_size , " PXIPXI recv " ) ) ;
// TODO: Shouldn't check the PXIBuffer table size but instead its actual size!
// if (path_size > path.size) {
// thread.GetLogger()->error("{}invalid parameters: Expected at least {:#x} bytes of data, got {:#x}",
// ThreadPrinter{thread}, path_size, path.size);
// thread.CallSVC(&OSImpl::SVCBreak, OSImpl::BreakReason::Panic);
// }
std : : string binary_path = " Archive path: " ;
for ( uint32_t offset = 0 ; offset < path_size ; + + offset )
binary_path + = fmt : : format ( " {:02x} " , + path . Read < uint8_t > ( thread , offset ) ) ;
thread . GetLogger ( ) - > info ( binary_path ) ;
2024-03-24 10:53:38 +01:00
auto & settings = thread . GetOS ( ) . settings ;
2024-03-07 22:05:16 +01:00
switch ( archive_id ) {
case 0x567890b0 :
{
// Purpose unknown. This is opened during startup of the FS module
// but not immediately used.
auto archive = std : : make_unique < Archive0x567890b0 > ( ) ;
auto archive_handle = context . next_archive_handle + + ;
context . archives . emplace ( std : : make_pair ( archive_handle , std : : move ( archive ) ) ) ;
return std : : make_tuple ( RESULT_OK , archive_handle ) ;
}
case 0x1234567c :
{
// System SaveData stored on NAND
// TODO: Should we verify that not more than 4 bytes have been given?
2024-12-08 20:16:52 +01:00
// NOTE: Media type only considers the lowest byte. Services like CFG pass in garbage in the upper bytes.
auto media_type = path . Read < uint8_t > ( thread , 0 ) ;
2024-03-24 10:53:38 +01:00
auto archive = std : : make_unique < ArchiveSystemSaveData > ( settings , media_type ) ;
2024-03-07 22:05:16 +01:00
auto archive_handle = context . next_archive_handle + + ;
context . archives . emplace ( std : : make_pair ( archive_handle , std : : move ( archive ) ) ) ;
return std : : make_tuple ( RESULT_OK , archive_handle ) ;
}
// TODO: Verify this works the same way for both FS and PXIFS
case 0x2345678a :
{
auto program_info = Platform : : PXI : : PM : : ProgramInfo { path . Read < uint64_t > ( thread , 0 ) , static_cast < uint8_t > ( path . Read < uint32_t > ( thread , 0x8 ) & 0xff ) } ;
// This has actually been observed to be used when menu tries reading the logos of titles returned from AM::GetTitleList
// TODO: 3dbrew has a few recent notes about this...
// if (path.Read<uint32_t>(thread, 0xc) != 0)
// thread.CallSVC(&OSImpl::SVCBreak, OSImpl::BreakReason::Panic);
auto archive = std : : make_unique < ArchiveProgramDataFromTitleId > ( program_info , thread . GetOS ( ) . setup . gamecard . get ( ) ) ;
auto archive_handle = context . next_archive_handle + + ;
context . archives . emplace ( std : : make_pair ( archive_handle , std : : move ( archive ) ) ) ;
return std : : make_tuple ( RESULT_OK , archive_handle ) ;
}
case 0x2345678e :
{
auto program_handle = Platform : : PXI : : PM : : ProgramHandle { path . Read < uint64_t > ( thread , 0 ) } ;
auto program_info_it = context . programs . find ( program_handle ) ;
if ( program_info_it = = context . programs . end ( ) ) {
thread . GetLogger ( ) - > error ( " Couldn't find handle {:#x} in program handle table " , program_handle . value ) ;
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
}
auto & program_info = program_info_it - > second . first ;
thread . GetLogger ( ) - > info ( " Opening ArchiveProgramDataFromProgramId for title id {:#x}, media_type {:#x} " , program_info . program_id , program_info . media_type ) ;
auto archive = std : : unique_ptr < Archive > ( new ArchiveProgramDataFromProgramId ( program_info , thread . GetOS ( ) . setup . gamecard . get ( ) ) ) ;
auto archive_handle = context . next_archive_handle + + ;
context . archives . emplace ( std : : make_pair ( archive_handle , std : : move ( archive ) ) ) ;
return std : : make_tuple ( RESULT_OK , archive_handle ) ;
}
default :
thread . GetLogger ( ) - > error ( " Unknown archive id {:#x} " , archive_id ) ;
thread . CallSVC ( & OSImpl : : SVCBreak , OSImpl : : BreakReason : : Panic ) ;
throw nullptr ;
}
}
static std : : tuple < Result > CloseArchive ( FakeThread & thread , Context & context ,
uint64_t archive_handle ) {
thread . GetLogger ( ) - > info ( " {}received CloseArchive with archive_handle={:#x} " ,
ThreadPrinter { thread } , archive_handle ) ;
// TODO: Does anything happen to files from this archive that are still opened?
context . archives . erase ( archive_handle ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static std : : tuple < Result > Unknown0x4f ( FakeThread & thread , Context & context , uint32_t param1 , uint32_t param2 ) {
thread . GetLogger ( ) - > info ( " {}received Unknown0x4f with param1={:#x}, param2={:#x} " ,
ThreadPrinter { thread } , param1 , param2 ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static std : : tuple < Result > SetPriority ( FakeThread & thread , Context & context , uint32_t priority ) {
thread . GetLogger ( ) - > info ( " {}received SetPriority with priority={:#x} " , ThreadPrinter { thread } , priority ) ;
return std : : make_tuple ( RESULT_OK ) ;
}
static void CommandHandler ( FakeThread & thread , Context & context , const IPC : : CommandHeader & header ) try {
namespace FS = Platform : : PXI : : FS ;
thread . GetLogger ( ) - > info ( " \n PXIPXI cmd: {:08x} " , header . command_id . Value ( ) ) ;
switch ( header . command_id ) {
case FS : : OpenFile : : id :
return IPC : : HandleIPCCommand < FS : : OpenFile > ( OpenFile , thread , thread , context ) ;
case FS : : DeleteFile : : id :
return IPC : : HandleIPCCommand < FS : : DeleteFile > ( DeleteFile , thread , thread , context ) ;
case FS : : ReadFile : : id :
return IPC : : HandleIPCCommand < FS : : ReadFile > ( ReadFile , thread , thread , context ) ;
case FS : : GetFileSHA256 : : id :
return IPC : : HandleIPCCommand < FS : : GetFileSHA256 > ( GetFileSHA256 , thread , thread , context ) ;
case FS : : WriteFile : : id :
return IPC : : HandleIPCCommand < FS : : WriteFile > ( WriteFile , thread , thread , context ) ;
case FS : : CalcSavegameMAC : : id :
return IPC : : HandleIPCCommand < FS : : CalcSavegameMAC > ( CalcSavegameMAC , thread , thread , context ) ;
case FS : : GetFileSize : : id :
return IPC : : HandleIPCCommand < FS : : GetFileSize > ( GetFileSize , thread , thread , context ) ;
case FS : : SetFileSize : : id :
return IPC : : HandleIPCCommand < FS : : SetFileSize > ( SetFileSize , thread , thread , context ) ;
case FS : : CloseFile : : id :
return IPC : : HandleIPCCommand < FS : : CloseFile > ( CloseFile , thread , thread , context ) ;
case FS : : OpenArchive : : id :
return IPC : : HandleIPCCommand < FS : : OpenArchive > ( OpenArchive , thread , thread , context ) ;
case FS : : CloseArchive : : id :
return IPC : : HandleIPCCommand < FS : : CloseArchive > ( CloseArchive , thread , thread , context ) ;
case FS : : SetPriority : : id :
return IPC : : HandleIPCCommand < FS : : SetPriority > ( SetPriority , thread , thread , context ) ;
case FS : : Unknown0x4f : : id :
return IPC : : HandleIPCCommand < FS : : Unknown0x4f > ( Unknown0x4f , thread , thread , context ) ;
default :
throw std : : runtime_error ( fmt : : format ( " Unknown PxiFSx service command with header {:#010x} " , header . raw ) ) ;
}
} catch ( const IPC : : IPCError & err ) {
auto response_header = IPC : : CommandHeader : : Make ( 0 , 1 , 0 ) ;
response_header . command_id = ( err . header > > 16 ) ;
thread . WriteTLS ( 0x80 , response_header . raw ) ;
thread . WriteTLS ( 0x84 , err . result ) ;
}
} // namespace FS
} // namespace PXI
using namespace PXI : : FS ;
namespace PXI {
using OSImpl = OS : : OS ;
void FakePXI : : FSThread ( FakeThread & thread , Context & context , const char * service_name ) {
// NOTE: Actually, there should be four of these buffers!
for ( unsigned buffer_index = 0 ; buffer_index < 4 ; + + buffer_index ) {
const auto buffer_size = 0x1000 ;
thread . WriteTLS ( 0x180 + 8 * buffer_index , IPC : : TranslationDescriptor : : MakeStaticBuffer ( 0 , buffer_size ) . raw ) ;
Result result ;
uint32_t buffer_addr ;
std : : tie ( result , buffer_addr ) = thread . CallSVC ( & OSImpl : : SVCControlMemory , 0 , 0 , pxi_static_buffer_size , 3 /* COMMIT */ , 3 /* RW */ ) ;
// std::tie(result, buffer_addr) = thread.CallSVC(&OSImpl::SVCControlMemory, 0, 0, 0x1000, 3 /* COMMIT */, 3 /* RW */);
thread . WriteTLS ( 0x184 + 8 * buffer_index , buffer_addr ) ;
}
OS : : ServiceUtil service ( thread , service_name , 1 ) ;
OS : : Handle last_signalled = OS : : HANDLE_INVALID ;
for ( ; ; ) {
Result result ;
int32_t index ;
std : : tie ( result , index ) = service . ReplyAndReceive ( thread , last_signalled ) ;
last_signalled = OS : : HANDLE_INVALID ;
if ( result ! = RESULT_OK )
os . SVCBreak ( thread , OSImpl : : BreakReason : : Panic ) ;
if ( index = = 0 ) {
// ServerPort: Incoming client connection
int32_t session_index ;
std : : tie ( result , session_index ) = service . AcceptSession ( thread , index ) ;
if ( result ! = RESULT_OK ) {
auto session = service . GetObject < OS : : ServerSession > ( session_index ) ;
if ( ! session ) {
logger . error ( " {}Failed to accept session. " , ThreadPrinter { thread } ) ;
os . SVCBreak ( thread , OSImpl : : BreakReason : : Panic ) ;
}
auto session_handle = service . GetHandle ( session_index ) ;
logger . warn ( " {}Failed to accept session. Maximal number of sessions exhausted? Closing session handle {} " ,
ThreadPrinter { thread } , OS : : HandlePrinter { thread , session_handle } ) ;
os . SVCCloseHandle ( thread , session_handle ) ;
}
} else {
// server_session: Incoming IPC command from the indexed client
logger . info ( " {}received IPC request " , ThreadPrinter { thread } ) ;
Platform : : IPC : : CommandHeader header = { thread . ReadTLS ( 0x80 ) } ;
auto signalled_handle = service . GetHandle ( index ) ;
CommandHandler ( thread , context , header ) ;
last_signalled = signalled_handle ;
}
}
}
} // namespace PXI
} // namespace HLE