2020-10-10 11:35:14 -04:00
/*
* nso . h
*
2020-12-23 13:48:57 -04:00
* Copyright ( c ) 2020 - 2021 , DarkMatterCore < pabloacurielz @ gmail . com > .
2020-10-10 11:35:14 -04:00
*
* This file is part of nxdumptool ( https : //github.com/DarkMatterCore/nxdumptool).
*
2021-03-25 15:26:58 -04:00
* nxdumptool is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
2020-10-10 11:35:14 -04:00
*
2021-03-25 15:26:58 -04:00
* nxdumptool is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
2020-10-10 11:35:14 -04:00
*
* You should have received a copy of the GNU General Public License
2021-03-25 15:26:58 -04:00
* along with this program . If not , see < https : //www.gnu.org/licenses/>.
2020-10-10 11:35:14 -04:00
*/
# pragma once
# ifndef __NSO_H__
# define __NSO_H__
2020-10-11 11:22:26 -04:00
# include "pfs.h"
2021-03-24 13:25:19 -04:00
# ifdef __cplusplus
extern " C " {
# endif
2020-10-10 11:35:14 -04:00
# define NSO_HEADER_MAGIC 0x4E534F30 /* "NSO0". */
# define NSO_MOD_MAGIC 0x4D4F4430 /* "MOD0". */
typedef enum {
NsoFlags_TextCompress = BIT ( 0 ) , ///< Determines if .text segment is LZ4-compressed.
NsoFlags_RoCompress = BIT ( 1 ) , ///< Determines if .rodata segment is LZ4-compressed.
NsoFlags_DataCompress = BIT ( 2 ) , ///< Determines if .data segment is LZ4-compressed.
NsoFlags_TextHash = BIT ( 3 ) , ///< Determines if .text segment hash must be checked during load.
NsoFlags_RoHash = BIT ( 4 ) , ///< Determines if .rodata segment hash must be checked during load.
NsoFlags_DataHash = BIT ( 5 ) ///< Determines if .data segment hash must be checked during load.
} NsoFlags ;
typedef struct {
u32 file_offset ; ///< NSO segment offset.
u32 memory_offset ; ///< Memory segment offset.
u32 size ; ///< Decompressed segment size.
2020-11-08 15:08:30 -04:00
} NsoSegmentInfo ;
2020-10-10 11:35:14 -04:00
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoSegmentInfo , 0xC ) ;
2020-10-10 11:35:14 -04:00
typedef struct {
u32 offset ; ///< Relative to the .rodata segment start.
u32 size ;
2020-11-08 15:08:30 -04:00
} NsoSectionInfo ;
2020-10-10 11:35:14 -04:00
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoSectionInfo , 0x8 ) ;
2020-10-10 11:35:14 -04:00
/// This is the start of every NSO.
/// This is always followed by a NsoModuleName block.
typedef struct {
u32 magic ; ///< "NSO0".
u32 version ; ///< Always set to 0.
u8 reserved_1 [ 0x4 ] ;
u32 flags ; ///< NsoFlags.
2020-11-08 15:08:30 -04:00
NsoSegmentInfo text_segment_info ;
2020-10-10 11:35:14 -04:00
u32 module_name_offset ; ///< NsoModuleName block offset.
2020-11-08 15:08:30 -04:00
NsoSegmentInfo rodata_segment_info ;
2020-10-10 11:35:14 -04:00
u32 module_name_size ; ///< NsoModuleName block size.
2020-11-08 15:08:30 -04:00
NsoSegmentInfo data_segment_info ;
2020-10-10 11:35:14 -04:00
u32 bss_size ;
u8 module_id [ 0x20 ] ; ///< Also known as build ID.
u32 text_file_size ; ///< .text segment compressed size (if NsoFlags_TextCompress is enabled).
u32 rodata_file_size ; ///< .rodata segment compressed size (if NsoFlags_RoCompress is enabled).
u32 data_file_size ; ///< .data segment compressed size (if NsoFlags_DataCompress is enabled).
u8 reserved_2 [ 0x1C ] ;
2020-11-08 15:08:30 -04:00
NsoSectionInfo api_info_section_info ;
NsoSectionInfo dynstr_section_info ;
NsoSectionInfo dynsym_section_info ;
2021-03-24 13:25:19 -04:00
u8 text_segment_hash [ SHA256_HASH_SIZE ] ; ///< Decompressed .text segment SHA-256 checksum.
u8 rodata_segment_hash [ SHA256_HASH_SIZE ] ; ///< Decompressed .rodata segment SHA-256 checksum.
u8 data_segment_hash [ SHA256_HASH_SIZE ] ; ///< Decompressed .data segment SHA-256 checksum.
2020-10-10 11:35:14 -04:00
} NsoHeader ;
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoHeader , 0x100 ) ;
2020-10-10 11:35:14 -04:00
/// Usually placed right after NsoHeader, but it's actual offset may vary.
/// If the 'module_name_size' member from NsoHeader is greater than 1 and the 'name_length' element from NsoModuleName is greater than 0, 'name' will hold the module name.
typedef struct {
u8 name_length ;
char name [ ] ;
} NsoModuleName ;
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoModuleName , 0x1 ) ;
2020-10-10 11:35:14 -04:00
/// Placed at the very start of the decompressed .text segment.
typedef struct {
u32 entry_point ;
u32 mod_offset ; ///< NsoModHeader block offset (relative to the start of this header). Almost always set to 0x8 (the size of this struct).
} NsoModStart ;
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoModStart , 0x8 ) ;
2020-10-10 11:35:14 -04:00
/// This is essentially a replacement for the PT_DYNAMIC program header available in ELF binaries.
/// All offsets are signed 32-bit values relative to the start of this header.
/// This is usually placed at the start of the decompressed .text segment, right after a NsoModStart block.
2020-10-11 11:22:26 -04:00
/// However, in some NSOs, it can instead be placed at the start of the decompressed .rodata segment, right after its NsoModuleInfo block.
2020-10-10 11:35:14 -04:00
/// In these cases, the 'mod_offset' value from the NsoModStart block will point to an offset within the .rodata segment.
typedef struct {
u32 magic ; ///< "MOD0".
s32 dynamic_offset ;
s32 bss_start_offset ;
s32 bss_end_offset ;
s32 eh_frame_hdr_start_offset ;
s32 eh_frame_hdr_end_offset ;
2020-10-11 11:22:26 -04:00
s32 module_object_offset ; ///< Typically equal to bss_start_offset.
2020-10-10 11:35:14 -04:00
} NsoModHeader ;
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoModHeader , 0x1C ) ;
2020-10-10 11:35:14 -04:00
/// Placed at the start of the decompressed .rodata segment + 0x4.
/// If the 'name_length' element is greater than 0, 'name' will hold the module name.
typedef struct {
u32 name_length ;
char name [ ] ;
} NsoModuleInfo ;
2021-03-24 13:25:19 -04:00
NXDT_ASSERT ( NsoModuleInfo , 0x4 ) ;
2020-10-10 17:08:17 -04:00
typedef struct {
2020-10-11 11:22:26 -04:00
PartitionFileSystemContext * pfs_ctx ; ///< PartitionFileSystemContext for the Program NCA FS section #0, which is where this NSO is stored.
PartitionFileSystemEntry * pfs_entry ; ///< PartitionFileSystemEntry for this NSO in the Program NCA FS section #0. Used to read NSO data.
char * nso_filename ; ///< Pointer to the NSO filename in the Program NCA FS section #0.
NsoHeader nso_header ; ///< NSO header.
char * module_name ; ///< Pointer to a dynamically allocated buffer that holds the NSO module name, if available. Otherwise, this is set to NULL.
char * module_info_name ; ///< Pointer to a dynamically allocated buffer that holds the .rodata module info module name, if available. Otherwise, this is set to NULL.
2020-10-11 14:13:09 -04:00
char * rodata_api_info_section ; ///< Pointer to a dynamically allocated buffer that holds the .rodata API info section data, if available. Otherwise, this is set to NULL.
///< Middleware and GuidelineApi entries are retrieved from this section.
u64 rodata_api_info_section_size ; ///< .rodata API info section size, if available. Otherwise, this is set to 0. Kept here for convenience - this is part of 'nso_header'.
2020-10-11 11:22:26 -04:00
char * rodata_dynstr_section ; ///< Pointer to a dynamically allocated buffer that holds the .rodata dynamic string section data. UnresolvedApi data is retrieved from this section.
u64 rodata_dynstr_section_size ; ///< .rodata dynamic string section size. Kept here for convenience - this is part of 'nso_header'.
u8 * rodata_dynsym_section ; ///< Pointer to a dynamically allocated buffer that holds the .rodata dynamic symbol section data. Used to retrieve pointers to symbol strings within dynstr.
u64 rodata_dynsym_section_size ; ///< .rodata dynamic symbol section size. Kept here for convenience - this is part of 'nso_header'.
2020-10-10 17:08:17 -04:00
} NsoContext ;
2020-10-11 11:22:26 -04:00
/// Initializes a NsoContext using a previously initialized PartitionFileSystemContext (which must belong to the ExeFS from a Program NCA) and a PartitionFileSystemEntry belonging to an underlying NSO.
bool nsoInitializeContext ( NsoContext * out , PartitionFileSystemContext * pfs_ctx , PartitionFileSystemEntry * pfs_entry ) ;
/// Helper inline functions.
NX_INLINE void nsoFreeContext ( NsoContext * nso_ctx )
{
if ( ! nso_ctx ) return ;
if ( nso_ctx - > module_name ) free ( nso_ctx - > module_name ) ;
if ( nso_ctx - > module_info_name ) free ( nso_ctx - > module_info_name ) ;
if ( nso_ctx - > rodata_api_info_section ) free ( nso_ctx - > rodata_api_info_section ) ;
if ( nso_ctx - > rodata_dynstr_section ) free ( nso_ctx - > rodata_dynstr_section ) ;
if ( nso_ctx - > rodata_dynsym_section ) free ( nso_ctx - > rodata_dynsym_section ) ;
memset ( nso_ctx , 0 , sizeof ( NsoContext ) ) ;
}
2020-10-10 17:08:17 -04:00
2021-03-23 15:06:52 +01:00
# ifdef __cplusplus
}
2021-03-24 13:25:19 -04:00
# endif
# endif /* __NSO_H__ */