#ifndef _SAVE_H #define _SAVE_H #include <stddef.h> #include <stdint.h> #include <libs/fatfs/ff.h> #define SAVE_HEADER_SIZE 0x4000 #define SAVE_FAT_ENTRY_SIZE 8 #define SAVE_FS_LIST_MAX_NAME_LENGTH 0x40 #define SAVE_FS_LIST_ENTRY_SIZE 0x60 #define IVFC_MAX_LEVEL 6 #define MAGIC_DISF 0x46534944 #define MAGIC_DPFS 0x53465044 #define MAGIC_JNGL 0x4C474E4A #define MAGIC_SAVE 0x45564153 #define MAGIC_RMAP 0x50414D52 #define MAGIC_IVFC 0x43465649 typedef enum { VALIDITY_UNCHECKED = 0, VALIDITY_INVALID, VALIDITY_VALID } validity_t; typedef struct save_ctx_t save_ctx_t; typedef struct { uint32_t magic; /* DISF */ uint32_t version; uint8_t hash[0x20]; uint64_t file_map_entry_offset; uint64_t file_map_entry_size; uint64_t meta_map_entry_offset; uint64_t meta_map_entry_size; uint64_t file_map_data_offset; uint64_t file_map_data_size; uint64_t duplex_l1_offset_a; uint64_t duplex_l1_offset_b; uint64_t duplex_l1_size; uint64_t duplex_data_offset_a; uint64_t duplex_data_offset_b; uint64_t duplex_data_size; uint64_t journal_data_offset; uint64_t journal_data_size_a; uint64_t journal_data_size_b; uint64_t journal_size; uint64_t duplex_master_offset_a; uint64_t duplex_master_offset_b; uint64_t duplex_master_size; uint64_t ivfc_master_hash_offset_a; uint64_t ivfc_master_hash_offset_b; uint64_t ivfc_master_hash_size; uint64_t journal_map_table_offset; uint64_t journal_map_table_size; uint64_t journal_physical_bitmap_offset; uint64_t journal_physical_bitmap_size; uint64_t journal_virtual_bitmap_offset; uint64_t journal_virtual_bitmap_size; uint64_t journal_free_bitmap_offset; uint64_t journal_free_bitmap_size; uint64_t ivfc_l1_offset; uint64_t ivfc_l1_size; uint64_t ivfc_l2_offset; uint64_t ivfc_l2_size; uint64_t ivfc_l3_offset; uint64_t ivfc_l3_size; uint64_t fat_offset; uint64_t fat_size; uint64_t duplex_index; uint64_t fat_ivfc_master_hash_a; uint64_t fat_ivfc_master_hash_b; uint64_t fat_ivfc_l1_offset; uint64_t fat_ivfc_l1_size; uint64_t fat_ivfc_l2_offset; uint64_t fat_ivfc_l2_size; uint8_t _0x190[0x70]; } fs_layout_t; #pragma pack(push, 1) typedef struct { uint64_t offset; uint64_t length; uint32_t block_size_power; } duplex_info_t; #pragma pack(pop) typedef struct { uint32_t magic; /* DPFS */ uint32_t version; duplex_info_t layers[3]; } duplex_header_t; typedef struct { uint32_t version; uint32_t main_data_block_count; uint32_t journal_block_count; uint32_t _0x0C; } journal_map_header_t; typedef struct { uint32_t magic; /* JNGL */ uint32_t version; uint64_t total_size; uint64_t journal_size; uint64_t block_size; } journal_header_t; typedef struct { uint32_t magic; /* SAVE */ uint32_t version; uint64_t block_count; uint64_t block_size; } save_fs_header_t; typedef struct { uint64_t block_size; uint64_t allocation_table_offset; uint32_t allocation_table_block_count; uint32_t _0x14; uint64_t data_offset; uint32_t data_block_count; uint32_t _0x24; uint32_t directory_table_block; uint32_t file_table_block; } fat_header_t; typedef struct { uint32_t magic; /* RMAP */ uint32_t version; uint32_t map_entry_count; uint32_t map_segment_count; uint32_t segment_bits; uint8_t _0x14[0x2C]; } remap_header_t; typedef struct remap_segment_ctx_t remap_segment_ctx_t; typedef struct remap_entry_ctx_t remap_entry_ctx_t; #pragma pack(push, 1) struct remap_entry_ctx_t { uint64_t virtual_offset; uint64_t physical_offset; uint64_t size; uint32_t alignment; uint32_t _0x1C; uint64_t virtual_offset_end; uint64_t physical_offset_end; remap_segment_ctx_t *segment; remap_entry_ctx_t *next; }; #pragma pack(pop) struct remap_segment_ctx_t{ uint64_t offset; uint64_t length; remap_entry_ctx_t **entries; uint64_t entry_count; }; typedef struct { uint8_t *data; uint8_t *bitmap; } duplex_bitmap_t; typedef struct { uint32_t block_size; uint8_t *bitmap_storage; uint8_t *data_a; uint8_t *data_b; duplex_bitmap_t bitmap; uint64_t _length; } duplex_storage_ctx_t; enum base_storage_type { STORAGE_BYTES = 0, STORAGE_DUPLEX = 1, STORAGE_REMAP = 2, STORAGE_JOURNAL = 3 }; typedef struct { remap_header_t *header; remap_entry_ctx_t *map_entries; remap_segment_ctx_t *segments; enum base_storage_type type; uint64_t base_storage_offset; duplex_storage_ctx_t *duplex; FIL *file; } remap_storage_ctx_t; typedef struct { uint64_t program_id; uint8_t user_id[0x10]; uint64_t save_id; uint8_t save_data_type; uint8_t _0x21[0x1F]; uint64_t save_owner_id; uint64_t timestamp; uint64_t _0x50; uint64_t data_size; uint64_t journal_size; uint64_t commit_id; } extra_data_t; typedef struct { uint64_t logical_offset; uint64_t hash_data_size; uint32_t block_size; uint32_t reserved; } ivfc_level_hdr_t; typedef struct { uint32_t magic; uint32_t id; uint32_t master_hash_size; uint32_t num_levels; ivfc_level_hdr_t level_headers[IVFC_MAX_LEVEL]; uint8_t salt_source[0x20]; } ivfc_save_hdr_t; #pragma pack(push, 1) typedef struct { uint8_t cmac[0x10]; uint8_t _0x10[0xF0]; fs_layout_t layout; duplex_header_t duplex_header; ivfc_save_hdr_t data_ivfc_header; uint32_t _0x404; journal_header_t journal_header; journal_map_header_t map_header; uint8_t _0x438[0x1D0]; save_fs_header_t save_header; fat_header_t fat_header; remap_header_t main_remap_header, meta_remap_header; uint64_t _0x6D0; extra_data_t extra_data; uint8_t _0x748[0x390]; ivfc_save_hdr_t fat_ivfc_header; uint8_t _0xB98[0x3468]; } save_header_t; #pragma pack(pop) typedef struct { duplex_storage_ctx_t layers[2]; duplex_storage_ctx_t data_layer; uint64_t _length; } hierarchical_duplex_storage_ctx_t; typedef struct { uint8_t *data_a; uint8_t *data_b; duplex_info_t info; } duplex_fs_layer_info_t; typedef struct { uint8_t *map_storage; uint8_t *physical_block_bitmap; uint8_t *virtual_block_bitmap; uint8_t *free_block_bitmap; } journal_map_params_t; typedef struct { uint32_t physical_index; uint32_t virtual_index; } journal_map_entry_t; typedef struct { journal_map_header_t *header; journal_map_entry_t *entries; uint8_t *map_storage; } journal_map_ctx_t; typedef struct { journal_map_ctx_t map; journal_header_t *header; uint32_t block_size; uint64_t journal_data_offset; uint64_t _length; FIL *file; } journal_storage_ctx_t; typedef struct { uint64_t data_offset; uint64_t data_size; uint64_t hash_offset; uint32_t hash_block_size; validity_t hash_validity; enum base_storage_type type; save_ctx_t *save_ctx; } ivfc_level_save_ctx_t; typedef struct { ivfc_level_save_ctx_t *data; uint32_t block_size; uint8_t salt[0x20]; } integrity_verification_info_ctx_t; typedef struct integrity_verification_storage_ctx_t integrity_verification_storage_ctx_t; struct integrity_verification_storage_ctx_t { ivfc_level_save_ctx_t *hash_storage; ivfc_level_save_ctx_t *base_storage; validity_t *block_validities; uint8_t salt[0x20]; uint32_t sector_size; uint32_t sector_count; uint64_t _length; integrity_verification_storage_ctx_t *next_level; }; typedef struct { ivfc_level_save_ctx_t levels[5]; ivfc_level_save_ctx_t *data_level; validity_t **level_validities; uint64_t _length; integrity_verification_storage_ctx_t integrity_storages[4]; } hierarchical_integrity_verification_storage_ctx_t; typedef struct { uint32_t prev; uint32_t next; } allocation_table_entry_t; typedef struct { uint32_t free_list_entry_index; void *base_storage; fat_header_t *header; } allocation_table_ctx_t; typedef struct { hierarchical_integrity_verification_storage_ctx_t *base_storage; uint32_t block_size; uint32_t initial_block; allocation_table_ctx_t *fat; uint64_t _length; } allocation_table_storage_ctx_t; typedef struct { allocation_table_ctx_t *fat; uint32_t virtual_block; uint32_t physical_block; uint32_t current_segment_size; uint32_t next_block; uint32_t prev_block; } allocation_table_iterator_ctx_t; typedef struct { char name[SAVE_FS_LIST_MAX_NAME_LENGTH]; uint32_t parent; } save_entry_key_t; #pragma pack(push, 1) typedef struct { uint32_t start_block; uint64_t length; uint32_t _0xC[2]; } save_file_info_t; #pragma pack(pop) #pragma pack(push, 1) typedef struct { uint32_t next_directory; uint32_t next_file; uint32_t _0x8[3]; } save_find_position_t; #pragma pack(pop) #pragma pack(push, 1) typedef struct { uint32_t next_sibling; union { /* Save table entry type. Size = 0x14. */ save_file_info_t save_file_info; save_find_position_t save_find_position; }; } save_table_entry_t; #pragma pack(pop) #pragma pack(push, 1) typedef struct { uint32_t parent; char name[SAVE_FS_LIST_MAX_NAME_LENGTH]; save_table_entry_t value; uint32_t next; } save_fs_list_entry_t; #pragma pack(pop) typedef struct { uint32_t free_list_head_index; uint32_t used_list_head_index; allocation_table_storage_ctx_t storage; uint32_t capacity; } save_filesystem_list_ctx_t; typedef struct { save_filesystem_list_ctx_t file_table; save_filesystem_list_ctx_t directory_table; } hierarchical_save_file_table_ctx_t; typedef struct { hierarchical_integrity_verification_storage_ctx_t *base_storage; allocation_table_ctx_t allocation_table; save_fs_header_t *header; hierarchical_save_file_table_ctx_t file_table; } save_filesystem_ctx_t; #define ACTION_VERIFY (1<<2) struct save_ctx_t { save_header_t header; FIL *file; struct { FIL *file; uint32_t action; } tool_ctx; validity_t header_cmac_validity; validity_t header_hash_validity; uint8_t *data_ivfc_master; uint8_t *fat_ivfc_master; remap_storage_ctx_t data_remap_storage; remap_storage_ctx_t meta_remap_storage; duplex_fs_layer_info_t duplex_layers[3]; hierarchical_duplex_storage_ctx_t duplex_storage; journal_storage_ctx_t journal_storage; journal_map_params_t journal_map_info; hierarchical_integrity_verification_storage_ctx_t core_data_ivfc_storage; hierarchical_integrity_verification_storage_ctx_t fat_ivfc_storage; uint8_t *fat_storage; save_filesystem_ctx_t save_filesystem_core; uint8_t save_mac_key[0x10]; }; static inline uint32_t allocation_table_entry_index_to_block(uint32_t entry_index) { return entry_index - 1; } static inline uint32_t allocation_table_block_to_entry_index(uint32_t block_index) { return block_index + 1; } static inline int allocation_table_get_prev(allocation_table_entry_t *entry) { return entry->prev & 0x7FFFFFFF; } static inline int allocation_table_get_next(allocation_table_entry_t *entry) { return entry->next & 0x7FFFFFFF; } static inline int allocation_table_is_list_start(allocation_table_entry_t *entry) { return entry->prev == 0x80000000; } static inline int allocation_table_is_list_end(allocation_table_entry_t *entry) { return (entry->next & 0x7FFFFFFF) == 0; } static inline bool allocation_table_is_multi_block_segment(allocation_table_entry_t *entry) { return entry->next & 0x80000000; } static inline void allocation_table_make_multi_block_segment(allocation_table_entry_t *entry) { entry->next |= 0x80000000; } static inline void allocation_table_make_single_block_segment(allocation_table_entry_t *entry) { entry->next &= 0x7FFFFFFF; } static inline bool allocation_table_is_single_block_segment(allocation_table_entry_t *entry) { return (entry->next & 0x80000000) == 0; } static inline void allocation_table_make_list_start(allocation_table_entry_t *entry) { entry->prev = 0x80000000; } static inline bool allocation_table_is_range_entry(allocation_table_entry_t *entry) { return (entry->prev & 0x80000000) != 0 && entry->prev != 0x80000000; } static inline void allocation_table_make_range_entry(allocation_table_entry_t *entry) { entry->prev |= 0x80000000; } static inline void allocation_table_set_next(allocation_table_entry_t *entry, int val) { entry->next = (entry->next & 0x80000000) | val; } static inline void allocation_table_set_prev(allocation_table_entry_t *entry, int val) { entry->prev = val; } static inline void allocation_table_set_range(allocation_table_entry_t *entry, int start_index, int end_index) { entry->next = end_index; entry->prev = start_index; allocation_table_make_range_entry(entry); } static inline allocation_table_entry_t *save_allocation_table_read_entry(allocation_table_ctx_t *ctx, uint32_t entry_index) { return (allocation_table_entry_t *)((uint8_t *)ctx->base_storage + entry_index * SAVE_FAT_ENTRY_SIZE); } static inline uint32_t save_allocation_table_get_free_list_entry_index(allocation_table_ctx_t *ctx) { return allocation_table_get_next(save_allocation_table_read_entry(ctx, ctx->free_list_entry_index)); } static inline uint32_t save_allocation_table_get_free_list_block_index(allocation_table_ctx_t *ctx) { return allocation_table_entry_index_to_block(save_allocation_table_get_free_list_entry_index(ctx)); } bool save_process(save_ctx_t *ctx); bool save_process_header(save_ctx_t *ctx); void save_free_contexts(save_ctx_t *ctx); void save_open_fat_storage(save_filesystem_ctx_t *ctx, allocation_table_storage_ctx_t *storage_ctx, uint32_t block_index); uint32_t save_allocation_table_storage_read(allocation_table_storage_ctx_t *ctx, void *buffer, uint64_t offset, size_t count); int save_fs_list_get_value(save_filesystem_list_ctx_t *ctx, uint32_t index, save_fs_list_entry_t *value); int save_hierarchical_file_table_get_file_entry_by_path(hierarchical_save_file_table_ctx_t *ctx, const char *path, save_fs_list_entry_t *entry); #endif