mirror of
https://github.com/DarkMatterCore/nxdumptool.git
synced 2024-11-22 18:26:39 +00:00
Remove NcaEncryptionType_Nca0 + BKTR start.
This commit is contained in:
parent
7aa72dc618
commit
7e8fa96829
5 changed files with 681 additions and 22 deletions
537
source/bktr.c
Normal file
537
source/bktr.c
Normal file
|
@ -0,0 +1,537 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020 DarkMatterCore
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "romfs.h"
|
||||||
|
#include "utils.h"
|
||||||
|
|
||||||
|
/* Function prototypes. */
|
||||||
|
|
||||||
|
static RomFileSystemDirectoryEntry *romfsGetChildDirectoryEntryByName(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, const char *name);
|
||||||
|
static RomFileSystemFileEntry *romfsGetChildFileEntryByName(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, const char *name);
|
||||||
|
|
||||||
|
bool romfsInitializeContext(RomFileSystemContext *out, NcaFsSectionContext *nca_fs_ctx)
|
||||||
|
{
|
||||||
|
NcaContext *nca_ctx = NULL;
|
||||||
|
u64 dir_table_offset = 0, file_table_offset = 0;
|
||||||
|
|
||||||
|
if (!out || !nca_fs_ctx || nca_fs_ctx->section_type != NcaFsSectionType_RomFs || !nca_fs_ctx->header || !(nca_ctx = (NcaContext*)nca_fs_ctx->nca_ctx) || \
|
||||||
|
(nca_ctx->format_version == NcaVersion_Nca0 && (nca_fs_ctx->section_type != NcaFsSectionType_Nca0RomFs || nca_fs_ctx->header->hash_type != NcaHashType_HierarchicalSha256)) || \
|
||||||
|
(nca_ctx->format_version != NcaVersion_Nca0 && (nca_fs_ctx->section_type != NcaFsSectionType_RomFs || nca_fs_ctx->header->hash_type != NcaHashType_HierarchicalIntegrity)))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Fill context */
|
||||||
|
out->nca_fs_ctx = nca_fs_ctx;
|
||||||
|
out->offset = 0;
|
||||||
|
out->size = 0;
|
||||||
|
out->dir_table_size = 0;
|
||||||
|
out->dir_table = NULL;
|
||||||
|
out->file_table_size = 0;
|
||||||
|
out->file_table = NULL;
|
||||||
|
out->body_offset = 0;
|
||||||
|
|
||||||
|
if (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs)
|
||||||
|
{
|
||||||
|
if (!ncaValidateHierarchicalSha256Offsets(&(nca_fs_ctx->header->hash_info.hierarchical_sha256), nca_fs_ctx->section_size))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid HierarchicalSha256 block!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
out->offset = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.offset;
|
||||||
|
out->size = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.size;
|
||||||
|
} else {
|
||||||
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && !ncaValidateHierarchicalIntegrityOffsets(&(nca_fs_ctx->header->hash_info.hierarchical_integrity), nca_fs_ctx->section_size))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid HierarchicalIntegrity block!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
out->offset = nca_fs_ctx->header->hash_info.hierarchical_integrity.hash_target_layer_info.offset;
|
||||||
|
out->size = nca_fs_ctx->header->hash_info.hierarchical_integrity.hash_target_layer_info.size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read RomFS header */
|
||||||
|
if (!ncaReadFsSection(nca_fs_ctx, &(out->header), sizeof(RomFileSystemHeader), out->offset))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to read RomFS header!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs && out->header.old_format.header_size != ROMFS_OLD_HEADER_SIZE) || \
|
||||||
|
(nca_fs_ctx->section_type == NcaFsSectionType_RomFs && out->header.cur_format.header_size != ROMFS_HEADER_SIZE))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid RomFS header size!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read directory entries table */
|
||||||
|
dir_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_offset : out->header.cur_format.directory_entry_offset);
|
||||||
|
out->dir_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_size : out->header.cur_format.directory_entry_size);
|
||||||
|
|
||||||
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (!out->dir_table_size || (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && (dir_table_offset >= out->size || (dir_table_offset + out->dir_table_size) > out->size)))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid RomFS directory entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
out->dir_table = malloc(out->dir_table_size);
|
||||||
|
if (!out->dir_table)
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to allocate memory for RomFS directory entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ncaReadFsSection(nca_fs_ctx, out->dir_table, out->dir_table_size, out->offset + dir_table_offset))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to read RomFS directory entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read file entries table */
|
||||||
|
file_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_offset : out->header.cur_format.file_entry_offset);
|
||||||
|
out->file_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_size : out->header.cur_format.file_entry_size);
|
||||||
|
|
||||||
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (!out->file_table_size || (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && (file_table_offset >= out->size || (file_table_offset + out->file_table_size) > out->size)))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid RomFS file entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
out->file_table = malloc(out->file_table_size);
|
||||||
|
if (!out->file_table)
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to allocate memory for RomFS file entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!ncaReadFsSection(nca_fs_ctx, out->file_table, out->file_table_size, out->offset + file_table_offset))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to read RomFS file entries table!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Get file data body offset */
|
||||||
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
out->body_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.body_offset : out->header.cur_format.body_offset);
|
||||||
|
if (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && out->body_offset >= out->size)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid RomFS file data body!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsReadFileSystemData(RomFileSystemContext *ctx, void *out, u64 read_size, u64 offset)
|
||||||
|
{
|
||||||
|
if (!ctx || !ctx->nca_fs_ctx || !ctx->size || !out || !read_size || offset >= ctx->size || (offset + read_size) > ctx->size)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read filesystem data */
|
||||||
|
if (!ncaReadFsSection(ctx->nca_fs_ctx, out, read_size, ctx->offset + offset))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to read RomFS data!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsReadFileEntryData(RomFileSystemContext *ctx, RomFileSystemFileEntry *file_entry, void *out, u64 read_size, u64 offset)
|
||||||
|
{
|
||||||
|
if (!ctx || !ctx->body_offset || !file_entry || !file_entry->size || file_entry->offset >= ctx->size || (file_entry->offset + file_entry->size) > ctx->size || \
|
||||||
|
!out || !read_size || offset >= file_entry->size || (offset + read_size) > file_entry->size)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Read entry data */
|
||||||
|
if (!romfsReadFileSystemData(ctx, out, read_size, ctx->body_offset + file_entry->offset + offset))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to read RomFS file entry data!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsGetTotalDataSize(RomFileSystemContext *ctx, u64 *out_size)
|
||||||
|
{
|
||||||
|
if (!ctx || !ctx->file_table_size || !ctx->file_table || !out_size)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 offset = 0, total_size = 0;
|
||||||
|
RomFileSystemFileEntry *file_entry = NULL;
|
||||||
|
|
||||||
|
while(offset < ctx->file_table_size)
|
||||||
|
{
|
||||||
|
if (!(file_entry = romfsGetFileEntryByOffset(ctx, offset)))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve file entry!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
total_size += file_entry->size;
|
||||||
|
offset += ALIGN_UP(sizeof(RomFileSystemFileEntry) + file_entry->name_length, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
*out_size = total_size;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsGetDirectoryDataSize(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, u64 *out_size)
|
||||||
|
{
|
||||||
|
if (!ctx || !ctx->dir_table_size || !ctx->dir_table || !ctx->file_table_size || !ctx->file_table || !dir_entry || (dir_entry->file_offset == ROMFS_VOID_ENTRY && \
|
||||||
|
dir_entry->directory_offset == ROMFS_VOID_ENTRY) || !out_size)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 total_size = 0, child_dir_size = 0;
|
||||||
|
u32 cur_file_offset = 0, cur_dir_offset = 0;
|
||||||
|
RomFileSystemFileEntry *cur_file_entry = NULL;
|
||||||
|
RomFileSystemDirectoryEntry *cur_dir_entry = NULL;
|
||||||
|
|
||||||
|
cur_file_offset = dir_entry->file_offset;
|
||||||
|
while(cur_file_offset != ROMFS_VOID_ENTRY)
|
||||||
|
{
|
||||||
|
if (!(cur_file_entry = romfsGetFileEntryByOffset(ctx, cur_file_offset)))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve file entry!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
total_size += cur_file_entry->size;
|
||||||
|
cur_file_offset = cur_file_entry->next_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
cur_dir_offset = dir_entry->directory_offset;
|
||||||
|
while(cur_dir_offset != ROMFS_VOID_ENTRY)
|
||||||
|
{
|
||||||
|
if (!(cur_dir_entry = romfsGetDirectoryEntryByOffset(ctx, cur_dir_offset)) || !romfsGetDirectoryDataSize(ctx, cur_dir_entry, &child_dir_size))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve directory entry/size!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
total_size += child_dir_size;
|
||||||
|
cur_dir_offset = cur_dir_entry->next_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
*out_size = total_size;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
RomFileSystemDirectoryEntry *romfsGetDirectoryEntryByPath(RomFileSystemContext *ctx, const char *path)
|
||||||
|
{
|
||||||
|
size_t path_len = 0;
|
||||||
|
char *path_dup = NULL, *pch = NULL;
|
||||||
|
RomFileSystemDirectoryEntry *dir_entry = NULL;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->dir_table || !ctx->dir_table_size || !path || *path != '/' || !(path_len = strlen(path)) || !(dir_entry = romfsGetDirectoryEntryByOffset(ctx, 0)))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check if the root directory was requested */
|
||||||
|
if (path_len == 1) return dir_entry;
|
||||||
|
|
||||||
|
/* Duplicate path to avoid problems with strtok() */
|
||||||
|
if (!(path_dup = strdup(path)))
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to duplicate input path!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
pch = strtok(path_dup, "/");
|
||||||
|
if (!pch)
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to tokenize input path!");
|
||||||
|
dir_entry = NULL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
while(pch)
|
||||||
|
{
|
||||||
|
if (!(dir_entry = romfsGetChildDirectoryEntryByName(ctx, dir_entry, pch)))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve directory entry by name!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
pch = strtok(NULL, "/");
|
||||||
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
|
if (path_dup) free(path_dup);
|
||||||
|
|
||||||
|
return dir_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
RomFileSystemFileEntry *romfsGetFileEntryByPath(RomFileSystemContext *ctx, const char *path)
|
||||||
|
{
|
||||||
|
size_t path_len = 0;
|
||||||
|
char *path_dup = NULL, *filename = NULL;
|
||||||
|
RomFileSystemFileEntry *file_entry = NULL;
|
||||||
|
RomFileSystemDirectoryEntry *dir_entry = NULL;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->file_table || !ctx->file_table_size || !path || *path != '/' || (path_len = strlen(path)) <= 1)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Duplicate path */
|
||||||
|
if (!(path_dup = strdup(path)))
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to duplicate input path!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove any trailing slashes */
|
||||||
|
while(path_dup[path_len - 1] == '/')
|
||||||
|
{
|
||||||
|
path_dup[path_len - 1] = '\0';
|
||||||
|
path_len--;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Safety check */
|
||||||
|
if (!path_len || !(filename = strrchr(path_dup, '/')))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid input path!");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Remove leading slash and adjust filename string pointer */
|
||||||
|
*filename++ = '\0';
|
||||||
|
|
||||||
|
/* Retrieve directory entry */
|
||||||
|
/* If the first character is NULL, then just retrieve the root directory entry */
|
||||||
|
if (!(dir_entry = (*path_dup ? romfsGetDirectoryEntryByPath(ctx, path_dup) : romfsGetDirectoryEntryByOffset(ctx, 0))))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve directory entry!");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Retrieve file entry */
|
||||||
|
if (!(file_entry = romfsGetChildFileEntryByName(ctx, dir_entry, filename))) LOGFILE("Failed to retrieve file entry by name!");
|
||||||
|
|
||||||
|
out:
|
||||||
|
if (path_dup) free(path_dup);
|
||||||
|
|
||||||
|
return file_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsGeneratePathFromDirectoryEntry(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, char *out_path, size_t out_path_size)
|
||||||
|
{
|
||||||
|
size_t path_len = 0;
|
||||||
|
u32 dir_offset = ROMFS_VOID_ENTRY, dir_entries_count = 0;
|
||||||
|
RomFileSystemDirectoryEntry **dir_entries = NULL, **tmp_dir_entries = NULL;
|
||||||
|
bool success = false;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->dir_table || !ctx->dir_table_size || !dir_entry || (!dir_entry->name_length && dir_entry->parent_offset) || !out_path || out_path_size < 2)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check if we're dealing with the root directory entry */
|
||||||
|
if (!dir_entry->name_length)
|
||||||
|
{
|
||||||
|
sprintf(out_path, "/");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Allocate memory for our directory entries */
|
||||||
|
dir_entries = calloc(1, sizeof(RomFileSystemDirectoryEntry*));
|
||||||
|
if (!dir_entries)
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to allocate memory for directory entries!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
path_len = (1 + dir_entry->name_length);
|
||||||
|
*dir_entries = dir_entry;
|
||||||
|
dir_entries_count++;
|
||||||
|
|
||||||
|
while(true)
|
||||||
|
{
|
||||||
|
dir_offset = dir_entries[dir_entries_count - 1]->parent_offset;
|
||||||
|
if (!dir_offset) break;
|
||||||
|
|
||||||
|
/* Reallocate directory entries */
|
||||||
|
if (!(tmp_dir_entries = realloc(dir_entries, (dir_entries_count + 1) * sizeof(RomFileSystemDirectoryEntry*))))
|
||||||
|
{
|
||||||
|
LOGFILE("Unable to reallocate directory entries buffer!");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
dir_entries = tmp_dir_entries;
|
||||||
|
tmp_dir_entries = NULL;
|
||||||
|
|
||||||
|
if (!(dir_entries[dir_entries_count] = romfsGetDirectoryEntryByOffset(ctx, dir_offset)) || !dir_entries[dir_entries_count]->name_length)
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve directory entry!");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
path_len += (1 + dir_entries[dir_entries_count]->name_length);
|
||||||
|
dir_entries_count++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path_len >= out_path_size)
|
||||||
|
{
|
||||||
|
LOGFILE("Output path length exceeds output buffer size!");
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Generate output path */
|
||||||
|
*out_path = '\0';
|
||||||
|
for(u32 i = dir_entries_count; i > 0; i--)
|
||||||
|
{
|
||||||
|
strcat(out_path, "/");
|
||||||
|
strncat(out_path, dir_entries[i - 1]->name, dir_entries[i - 1]->name_length);
|
||||||
|
}
|
||||||
|
|
||||||
|
success = true;
|
||||||
|
|
||||||
|
out:
|
||||||
|
if (dir_entries) free(dir_entries);
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsGeneratePathFromFileEntry(RomFileSystemContext *ctx, RomFileSystemFileEntry *file_entry, char *out_path, size_t out_path_size)
|
||||||
|
{
|
||||||
|
size_t path_len = 0;
|
||||||
|
RomFileSystemDirectoryEntry *dir_entry = NULL;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->file_table || !ctx->file_table_size || !file_entry || !file_entry->name_length || !out_path || out_path_size < 2 || \
|
||||||
|
!(dir_entry = romfsGetDirectoryEntryByOffset(ctx, file_entry->parent_offset)))
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Retrieve full RomFS path up to the file entry name */
|
||||||
|
if (!romfsGeneratePathFromDirectoryEntry(ctx, dir_entry, out_path, out_path_size))
|
||||||
|
{
|
||||||
|
LOGFILE("Failed to retrieve RomFS directory path!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check path length */
|
||||||
|
path_len = strlen(out_path);
|
||||||
|
if ((1 + file_entry->name_length) >= (out_path_size - path_len))
|
||||||
|
{
|
||||||
|
LOGFILE("Output path length exceeds output buffer size!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Concatenate file entry name */
|
||||||
|
strcat(out_path, "/");
|
||||||
|
strncat(out_path, file_entry->name, file_entry->name_length);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool romfsGenerateFileEntryPatch(RomFileSystemContext *ctx, RomFileSystemFileEntry *file_entry, const void *data, u64 data_size, u64 data_offset, RomFileSystemFileEntryPatch *out)
|
||||||
|
{
|
||||||
|
if (!ctx || !ctx->nca_fs_ctx || !ctx->body_offset || (ctx->nca_fs_ctx->section_type != NcaFsSectionType_Nca0RomFs && ctx->nca_fs_ctx->section_type != NcaFsSectionType_RomFs) || !file_entry || \
|
||||||
|
!file_entry->size || file_entry->offset >= ctx->size || (file_entry->offset + file_entry->size) > ctx->size || !data || !data_size || data_offset >= file_entry->size || \
|
||||||
|
(data_offset + data_size) > file_entry->size || !out)
|
||||||
|
{
|
||||||
|
LOGFILE("Invalid parameters!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool success = false;
|
||||||
|
u64 fs_offset = (ctx->body_offset + file_entry->offset + data_offset);
|
||||||
|
|
||||||
|
memset(&(out->old_format_patch), 0, sizeof(NcaHierarchicalSha256Patch));
|
||||||
|
memset(&(out->cur_format_patch), 0, sizeof(NcaHierarchicalIntegrityPatch));
|
||||||
|
|
||||||
|
if (ctx->nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs)
|
||||||
|
{
|
||||||
|
out->use_old_format_patch = true;
|
||||||
|
success = ncaGenerateHierarchicalSha256Patch(ctx->nca_fs_ctx, data, data_size, fs_offset, &(out->old_format_patch));
|
||||||
|
if (!success) LOGFILE("Failed to generate 0x%lX bytes HierarchicalSha256 patch at offset 0x%lX for RomFS file entry!", data_size, fs_offset);
|
||||||
|
} else {
|
||||||
|
out->use_old_format_patch = false;
|
||||||
|
success = ncaGenerateHierarchicalIntegrityPatch(ctx->nca_fs_ctx, data, data_size, fs_offset, &(out->cur_format_patch));
|
||||||
|
if (!success) LOGFILE("Failed to generate 0x%lX bytes HierarchicalIntegrity patch at offset 0x%lX for RomFS file entry!", data_size, fs_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
||||||
|
|
||||||
|
static RomFileSystemDirectoryEntry *romfsGetChildDirectoryEntryByName(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, const char *name)
|
||||||
|
{
|
||||||
|
u64 dir_offset = 0;
|
||||||
|
size_t name_len = 0;
|
||||||
|
RomFileSystemDirectoryEntry *child_dir_entry = NULL;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->dir_table || !ctx->dir_table_size || !dir_entry || (dir_offset = dir_entry->directory_offset) == ROMFS_VOID_ENTRY || !name || !(name_len = strlen(name))) return NULL;
|
||||||
|
|
||||||
|
while(dir_offset != ROMFS_VOID_ENTRY)
|
||||||
|
{
|
||||||
|
if (!(child_dir_entry = romfsGetDirectoryEntryByOffset(ctx, dir_offset)) || !child_dir_entry->name_length) return NULL;
|
||||||
|
if (!strncmp(child_dir_entry->name, name, name_len)) return child_dir_entry;
|
||||||
|
dir_offset = child_dir_entry->next_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static RomFileSystemFileEntry *romfsGetChildFileEntryByName(RomFileSystemContext *ctx, RomFileSystemDirectoryEntry *dir_entry, const char *name)
|
||||||
|
{
|
||||||
|
u64 file_offset = 0;
|
||||||
|
size_t name_len = 0;
|
||||||
|
RomFileSystemFileEntry *child_file_entry = NULL;
|
||||||
|
|
||||||
|
if (!ctx || !ctx->dir_table || !ctx->dir_table_size || !ctx->file_table || !ctx->file_table_size || !dir_entry || (file_offset = dir_entry->file_offset) == ROMFS_VOID_ENTRY || !name || \
|
||||||
|
!(name_len = strlen(name))) return NULL;
|
||||||
|
|
||||||
|
while(file_offset != ROMFS_VOID_ENTRY)
|
||||||
|
{
|
||||||
|
if (!(child_file_entry = romfsGetFileEntryByOffset(ctx, file_offset)) || !child_file_entry->name_length) return NULL;
|
||||||
|
if (!strncmp(child_file_entry->name, name, name_len)) return child_file_entry;
|
||||||
|
file_offset = child_file_entry->next_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
119
source/bktr.h
Normal file
119
source/bktr.h
Normal file
|
@ -0,0 +1,119 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020 DarkMatterCore
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms and conditions of the GNU General Public License,
|
||||||
|
* version 2, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||||
|
* more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef __BKTR_H__
|
||||||
|
#define __BKTR_H__
|
||||||
|
|
||||||
|
#include <switch.h>
|
||||||
|
#include "romfs.h"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
BktrIndirectStorageIndex_Original = 0,
|
||||||
|
BktrIndirectStorageIndex_Patch = 1
|
||||||
|
} BktrIndirectStorageIndex;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u64 virtual_offset;
|
||||||
|
u64 physical_offset;
|
||||||
|
u32 indirect_storage_index; ///< BktrIndirectStorageIndex.
|
||||||
|
} BktrIndirectStorageEntry;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u32 index;
|
||||||
|
u32 entry_count;
|
||||||
|
u64 end_offset;
|
||||||
|
BktrIndirectStorageEntry indirect_storage_entries[0x3FF0 / sizeof(BktrIndirectStorageEntry)];
|
||||||
|
u8 reserved[0x3FF0 % sizeof(BktrIndirectStorageEntry)];
|
||||||
|
} BktrIndirectStorageBucket;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u32 index;
|
||||||
|
u32 bucket_count;
|
||||||
|
u64 virtual_size;
|
||||||
|
u64 virtual_offsets[0x3FF0 / sizeof(u64)];
|
||||||
|
BktrIndirectStorageBucket indirect_storage_buckets[];
|
||||||
|
} BktrIndirectStorageBlock;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u64 offset;
|
||||||
|
u32 size;
|
||||||
|
u32 generation;
|
||||||
|
} BktrAesCtrExStorageEntry;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u32 index;
|
||||||
|
u32 entry_count;
|
||||||
|
u64 end_offset;
|
||||||
|
BktrAesCtrExStorageEntry aes_ctr_ex_storage_entries[0x3FF];
|
||||||
|
} BktrAesCtrExStorageBucket;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
u32 index;
|
||||||
|
u32 bucket_count;
|
||||||
|
u64 physical_size;
|
||||||
|
u64 physical_offsets[0x3FF0 / sizeof(u64)];
|
||||||
|
BktrAesCtrExStorageBucket aes_ctr_ex_storage_buckets[];
|
||||||
|
} BktrAesCtrExStorageBlock;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
RomFileSystemContext base_romfs; ///< Base NCA RomFS context.
|
||||||
|
RomFileSystemContext patch_romfs; ///< Update NCA RomFS context. Must be used with RomFS directory/file entry functions, because it holds the updated directory/file tables.
|
||||||
|
NcaPatchInfo *patch_info;
|
||||||
|
BktrIndirectStorageBlock *indirect_block;
|
||||||
|
BktrAesCtrExStorageBlock *aes_ctr_ex_block;
|
||||||
|
u64 virtual_seek; ///< Relative to the start of the NCA FS section.
|
||||||
|
u64 base_seek; ///< Relative to the start of the NCA FS section (base NCA RomFS).
|
||||||
|
u64 patch_seek; ///< Relative to the start of the NCA FS section (update NCA BKTR).
|
||||||
|
} BktrContext;
|
||||||
|
|
||||||
|
/// Initializes a BKTR context.
|
||||||
|
bool bktrInitializeContext(BktrContext *out, NcaFsSectionContext *base_nca_fs_ctx, NcaFsSectionContext *update_nca_fs_ctx);
|
||||||
|
|
||||||
|
/// Cleanups a previously initialized BKTR context.
|
||||||
|
NX_INLINE void bktrFreeContext(BktrContext *ctx)
|
||||||
|
{
|
||||||
|
if (!ctx) return;
|
||||||
|
romfsFreeContext(&(ctx->base_romfs));
|
||||||
|
romfsFreeContext(&(ctx->update_romfs));
|
||||||
|
if (ctx->indirect_block) free(ctx->indirect_block);
|
||||||
|
if (ctx->aes_ctr_ex_block) free(ctx->aes_ctr_ex_block);
|
||||||
|
memset(ctx, 0, sizeof(BktrContext));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/// Reads raw filesystem data using a RomFS context.
|
||||||
|
/// Input offset must be relative to the start of the RomFS.
|
||||||
|
//bool romfsReadFileSystemData(RomFileSystemContext *ctx, void *out, u64 read_size, u64 offset);
|
||||||
|
|
||||||
|
/// Reads data from a previously retrieved RomFileSystemFileEntry using a RomFS context.
|
||||||
|
/// Input offset must be relative to the start of the RomFS file entry data.
|
||||||
|
//bool romfsReadFileEntryData(RomFileSystemContext *ctx, RomFileSystemFileEntry *file_entry, void *out, u64 read_size, u64 offset);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#endif /* __BKTR_H__ */
|
32
source/nca.c
32
source/nca.c
|
@ -294,7 +294,7 @@ bool ncaInitializeContext(NcaContext *out, u8 storage_id, NcmContentStorage *ncm
|
||||||
memset(&(out->fs_contexts[i].xts_encrypt_ctx), 0, sizeof(Aes128XtsContext));
|
memset(&(out->fs_contexts[i].xts_encrypt_ctx), 0, sizeof(Aes128XtsContext));
|
||||||
|
|
||||||
/* Determine encryption type */
|
/* Determine encryption type */
|
||||||
out->fs_contexts[i].encryption_type = (out->format_version == NcaVersion_Nca0 ? NcaEncryptionType_Nca0 : out->header.fs_headers[i].encryption_type);
|
out->fs_contexts[i].encryption_type = (out->format_version == NcaVersion_Nca0 ? NcaEncryptionType_AesXts : out->header.fs_headers[i].encryption_type);
|
||||||
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_Auto)
|
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_Auto)
|
||||||
{
|
{
|
||||||
switch(out->fs_contexts[i].section_num)
|
switch(out->fs_contexts[i].section_num)
|
||||||
|
@ -312,7 +312,7 @@ bool ncaInitializeContext(NcaContext *out, u8 storage_id, NcmContentStorage *ncm
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if we're dealing with an invalid encryption type value */
|
/* Check if we're dealing with an invalid encryption type value */
|
||||||
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_Auto || out->fs_contexts[i].encryption_type > NcaEncryptionType_Nca0) continue;
|
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_Auto || out->fs_contexts[i].encryption_type > NcaEncryptionType_AesCtrEx) continue;
|
||||||
|
|
||||||
/* Determine FS section type */
|
/* Determine FS section type */
|
||||||
if (out->fs_contexts[i].header->fs_type == NcaFsType_PartitionFs && out->fs_contexts[i].header->hash_type == NcaHashType_HierarchicalSha256)
|
if (out->fs_contexts[i].header->fs_type == NcaFsType_PartitionFs && out->fs_contexts[i].header->hash_type == NcaHashType_HierarchicalSha256)
|
||||||
|
@ -332,7 +332,7 @@ bool ncaInitializeContext(NcaContext *out, u8 storage_id, NcmContentStorage *ncm
|
||||||
if (out->fs_contexts[i].section_type >= NcaFsSectionType_Invalid) continue;
|
if (out->fs_contexts[i].section_type >= NcaFsSectionType_Invalid) continue;
|
||||||
|
|
||||||
/* Initialize crypto related fields */
|
/* Initialize crypto related fields */
|
||||||
if (out->fs_contexts[i].encryption_type > NcaEncryptionType_None && out->fs_contexts[i].encryption_type <= NcaEncryptionType_Nca0)
|
if (out->fs_contexts[i].encryption_type > NcaEncryptionType_None && out->fs_contexts[i].encryption_type <= NcaEncryptionType_AesCtrEx)
|
||||||
{
|
{
|
||||||
/* Initialize section CTR */
|
/* Initialize section CTR */
|
||||||
ncaInitializeAesCtrIv(out->fs_contexts[i].ctr, out->fs_contexts[i].header->section_ctr, out->fs_contexts[i].section_offset);
|
ncaInitializeAesCtrIv(out->fs_contexts[i].ctr, out->fs_contexts[i].header->section_ctr, out->fs_contexts[i].section_offset);
|
||||||
|
@ -346,7 +346,7 @@ bool ncaInitializeContext(NcaContext *out, u8 storage_id, NcmContentStorage *ncm
|
||||||
{
|
{
|
||||||
aes128CtrContextCreate(&(out->fs_contexts[i].ctr_ctx), out->decrypted_keys[2].key, out->fs_contexts[i].ctr);
|
aes128CtrContextCreate(&(out->fs_contexts[i].ctr_ctx), out->decrypted_keys[2].key, out->fs_contexts[i].ctr);
|
||||||
} else
|
} else
|
||||||
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_AesXts || out->fs_contexts[i].encryption_type == NcaEncryptionType_Nca0)
|
if (out->fs_contexts[i].encryption_type == NcaEncryptionType_AesXts)
|
||||||
{
|
{
|
||||||
/* We need to create two different contexts: one for decryption and another one for encryption */
|
/* We need to create two different contexts: one for decryption and another one for encryption */
|
||||||
aes128XtsContextCreate(&(out->fs_contexts[i].xts_decrypt_ctx), out->decrypted_keys[0].key, out->decrypted_keys[1].key, false);
|
aes128XtsContextCreate(&(out->fs_contexts[i].xts_decrypt_ctx), out->decrypted_keys[0].key, out->decrypted_keys[1].key, false);
|
||||||
|
@ -942,7 +942,7 @@ static bool _ncaReadFsSection(NcaFsSectionContext *ctx, void *out, u64 read_size
|
||||||
bool ret = false;
|
bool ret = false;
|
||||||
|
|
||||||
if (!g_ncaCryptoBuffer || !ctx || !ctx->nca_ctx || ctx->section_num >= NCA_FS_HEADER_COUNT || ctx->section_offset < NCA_FULL_HEADER_LENGTH || ctx->section_type >= NcaFsSectionType_Invalid || \
|
if (!g_ncaCryptoBuffer || !ctx || !ctx->nca_ctx || ctx->section_num >= NCA_FS_HEADER_COUNT || ctx->section_offset < NCA_FULL_HEADER_LENGTH || ctx->section_type >= NcaFsSectionType_Invalid || \
|
||||||
ctx->encryption_type == NcaEncryptionType_Auto || ctx->encryption_type > NcaEncryptionType_Nca0 || !ctx->header || !out || !read_size || offset >= ctx->section_size || \
|
ctx->encryption_type == NcaEncryptionType_Auto || ctx->encryption_type > NcaEncryptionType_AesCtrEx || !ctx->header || !out || !read_size || offset >= ctx->section_size || \
|
||||||
(offset + read_size) > ctx->section_size)
|
(offset + read_size) > ctx->section_size)
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid NCA FS section header parameters!");
|
LOGFILE("Invalid NCA FS section header parameters!");
|
||||||
|
@ -967,7 +967,7 @@ static bool _ncaReadFsSection(NcaFsSectionContext *ctx, void *out, u64 read_size
|
||||||
|
|
||||||
/* Optimization for reads from plaintext FS sections or reads that are aligned to the AES-CTR / AES-XTS sector size */
|
/* Optimization for reads from plaintext FS sections or reads that are aligned to the AES-CTR / AES-XTS sector size */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_None || \
|
if (ctx->encryption_type == NcaEncryptionType_None || \
|
||||||
((ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) && !(content_offset % NCA_AES_XTS_SECTOR_SIZE) && !(read_size % NCA_AES_XTS_SECTOR_SIZE)) || \
|
(ctx->encryption_type == NcaEncryptionType_AesXts && !(content_offset % NCA_AES_XTS_SECTOR_SIZE) && !(read_size % NCA_AES_XTS_SECTOR_SIZE)) || \
|
||||||
((ctx->encryption_type == NcaEncryptionType_AesCtr || ctx->encryption_type == NcaEncryptionType_AesCtrEx) && !(content_offset % AES_BLOCK_SIZE) && !(read_size % AES_BLOCK_SIZE)))
|
((ctx->encryption_type == NcaEncryptionType_AesCtr || ctx->encryption_type == NcaEncryptionType_AesCtrEx) && !(content_offset % AES_BLOCK_SIZE) && !(read_size % AES_BLOCK_SIZE)))
|
||||||
{
|
{
|
||||||
/* Read data */
|
/* Read data */
|
||||||
|
@ -985,7 +985,7 @@ static bool _ncaReadFsSection(NcaFsSectionContext *ctx, void *out, u64 read_size
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Decrypt data */
|
/* Decrypt data */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0)
|
if (ctx->encryption_type == NcaEncryptionType_AesXts)
|
||||||
{
|
{
|
||||||
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
||||||
|
|
||||||
|
@ -1008,8 +1008,8 @@ static bool _ncaReadFsSection(NcaFsSectionContext *ctx, void *out, u64 read_size
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Calculate offsets and block sizes */
|
/* Calculate offsets and block sizes */
|
||||||
block_start_offset = ALIGN_DOWN(content_offset, (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
block_start_offset = ALIGN_DOWN(content_offset, ctx->encryption_type == NcaEncryptionType_AesXts ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
||||||
block_end_offset = ALIGN_UP(content_offset + read_size, (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
block_end_offset = ALIGN_UP(content_offset + read_size, ctx->encryption_type == NcaEncryptionType_AesXts ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
||||||
block_size = (block_end_offset - block_start_offset);
|
block_size = (block_end_offset - block_start_offset);
|
||||||
|
|
||||||
data_start_offset = (content_offset - block_start_offset);
|
data_start_offset = (content_offset - block_start_offset);
|
||||||
|
@ -1024,7 +1024,7 @@ static bool _ncaReadFsSection(NcaFsSectionContext *ctx, void *out, u64 read_size
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Decrypt data */
|
/* Decrypt data */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0)
|
if (ctx->encryption_type == NcaEncryptionType_AesXts)
|
||||||
{
|
{
|
||||||
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
||||||
|
|
||||||
|
@ -1061,7 +1061,7 @@ static void *_ncaGenerateEncryptedFsSectionBlock(NcaFsSectionContext *ctx, const
|
||||||
bool success = false;
|
bool success = false;
|
||||||
|
|
||||||
if (!g_ncaCryptoBuffer || !ctx || !ctx->nca_ctx || ctx->section_num >= NCA_FS_HEADER_COUNT || ctx->section_offset < NCA_FULL_HEADER_LENGTH || ctx->section_type >= NcaFsSectionType_Invalid || \
|
if (!g_ncaCryptoBuffer || !ctx || !ctx->nca_ctx || ctx->section_num >= NCA_FS_HEADER_COUNT || ctx->section_offset < NCA_FULL_HEADER_LENGTH || ctx->section_type >= NcaFsSectionType_Invalid || \
|
||||||
ctx->encryption_type == NcaEncryptionType_Auto || ctx->encryption_type > NcaEncryptionType_Nca0 || !ctx->header || !data || !data_size || data_offset >= ctx->section_size || \
|
ctx->encryption_type == NcaEncryptionType_Auto || ctx->encryption_type > NcaEncryptionType_AesCtrEx || !ctx->header || !data || !data_size || data_offset >= ctx->section_size || \
|
||||||
(data_offset + data_size) > ctx->section_size || !out_block_size || !out_block_offset)
|
(data_offset + data_size) > ctx->section_size || !out_block_size || !out_block_offset)
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid NCA FS section header parameters!");
|
LOGFILE("Invalid NCA FS section header parameters!");
|
||||||
|
@ -1086,7 +1086,7 @@ static void *_ncaGenerateEncryptedFsSectionBlock(NcaFsSectionContext *ctx, const
|
||||||
|
|
||||||
/* Optimization for blocks from plaintext FS sections or blocks that are aligned to the AES-CTR / AES-XTS sector size */
|
/* Optimization for blocks from plaintext FS sections or blocks that are aligned to the AES-CTR / AES-XTS sector size */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_None || \
|
if (ctx->encryption_type == NcaEncryptionType_None || \
|
||||||
((ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) && !(content_offset % NCA_AES_XTS_SECTOR_SIZE) && !(data_size % NCA_AES_XTS_SECTOR_SIZE)) || \
|
(ctx->encryption_type == NcaEncryptionType_AesXts && !(content_offset % NCA_AES_XTS_SECTOR_SIZE) && !(data_size % NCA_AES_XTS_SECTOR_SIZE)) || \
|
||||||
((ctx->encryption_type == NcaEncryptionType_AesCtr || ctx->encryption_type == NcaEncryptionType_AesCtrEx) && !(content_offset % AES_BLOCK_SIZE) && !(data_size % AES_BLOCK_SIZE)))
|
((ctx->encryption_type == NcaEncryptionType_AesCtr || ctx->encryption_type == NcaEncryptionType_AesCtrEx) && !(content_offset % AES_BLOCK_SIZE) && !(data_size % AES_BLOCK_SIZE)))
|
||||||
{
|
{
|
||||||
/* Allocate memory */
|
/* Allocate memory */
|
||||||
|
@ -1101,7 +1101,7 @@ static void *_ncaGenerateEncryptedFsSectionBlock(NcaFsSectionContext *ctx, const
|
||||||
memcpy(out, data, data_size);
|
memcpy(out, data, data_size);
|
||||||
|
|
||||||
/* Encrypt data */
|
/* Encrypt data */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0)
|
if (ctx->encryption_type == NcaEncryptionType_AesXts)
|
||||||
{
|
{
|
||||||
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? data_offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? data_offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
||||||
|
|
||||||
|
@ -1127,8 +1127,8 @@ static void *_ncaGenerateEncryptedFsSectionBlock(NcaFsSectionContext *ctx, const
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Calculate block offsets and size */
|
/* Calculate block offsets and size */
|
||||||
block_start_offset = ALIGN_DOWN(data_offset, (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
block_start_offset = ALIGN_DOWN(data_offset, ctx->encryption_type == NcaEncryptionType_AesXts ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
||||||
block_end_offset = ALIGN_UP(data_offset + data_size, (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0) ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
block_end_offset = ALIGN_UP(data_offset + data_size, ctx->encryption_type == NcaEncryptionType_AesXts ? NCA_AES_XTS_SECTOR_SIZE : AES_BLOCK_SIZE);
|
||||||
block_size = (block_end_offset - block_start_offset);
|
block_size = (block_end_offset - block_start_offset);
|
||||||
|
|
||||||
plain_chunk_offset = (data_offset - block_start_offset);
|
plain_chunk_offset = (data_offset - block_start_offset);
|
||||||
|
@ -1153,7 +1153,7 @@ static void *_ncaGenerateEncryptedFsSectionBlock(NcaFsSectionContext *ctx, const
|
||||||
memcpy(out + plain_chunk_offset, data, data_size);
|
memcpy(out + plain_chunk_offset, data, data_size);
|
||||||
|
|
||||||
/* Reencrypt data */
|
/* Reencrypt data */
|
||||||
if (ctx->encryption_type == NcaEncryptionType_AesXts || ctx->encryption_type == NcaEncryptionType_Nca0)
|
if (ctx->encryption_type == NcaEncryptionType_AesXts)
|
||||||
{
|
{
|
||||||
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? block_start_offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
sector_num = ((ctx->encryption_type == NcaEncryptionType_AesXts ? block_start_offset : (content_offset - NCA_HEADER_LENGTH)) / NCA_AES_XTS_SECTOR_SIZE);
|
||||||
|
|
||||||
|
|
|
@ -124,8 +124,7 @@ typedef enum {
|
||||||
NcaEncryptionType_None = 1,
|
NcaEncryptionType_None = 1,
|
||||||
NcaEncryptionType_AesXts = 2,
|
NcaEncryptionType_AesXts = 2,
|
||||||
NcaEncryptionType_AesCtr = 3,
|
NcaEncryptionType_AesCtr = 3,
|
||||||
NcaEncryptionType_AesCtrEx = 4,
|
NcaEncryptionType_AesCtrEx = 4
|
||||||
NcaEncryptionType_Nca0 = 5 ///< Only used to represent NCA0 AES-XTS FS section crypto - not actually used as a possible value for this field.
|
|
||||||
} NcaEncryptionType;
|
} NcaEncryptionType;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
|
|
@ -60,7 +60,8 @@ bool romfsInitializeContext(RomFileSystemContext *out, NcaFsSectionContext *nca_
|
||||||
out->offset = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.offset;
|
out->offset = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.offset;
|
||||||
out->size = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.size;
|
out->size = nca_fs_ctx->header->hash_info.hierarchical_sha256.hash_target_layer_info.size;
|
||||||
} else {
|
} else {
|
||||||
if (!ncaValidateHierarchicalIntegrityOffsets(&(nca_fs_ctx->header->hash_info.hierarchical_integrity), nca_fs_ctx->section_size))
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && !ncaValidateHierarchicalIntegrityOffsets(&(nca_fs_ctx->header->hash_info.hierarchical_integrity), nca_fs_ctx->section_size))
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid HierarchicalIntegrity block!");
|
LOGFILE("Invalid HierarchicalIntegrity block!");
|
||||||
return false;
|
return false;
|
||||||
|
@ -88,7 +89,8 @@ bool romfsInitializeContext(RomFileSystemContext *out, NcaFsSectionContext *nca_
|
||||||
dir_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_offset : out->header.cur_format.directory_entry_offset);
|
dir_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_offset : out->header.cur_format.directory_entry_offset);
|
||||||
out->dir_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_size : out->header.cur_format.directory_entry_size);
|
out->dir_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.directory_entry_size : out->header.cur_format.directory_entry_size);
|
||||||
|
|
||||||
if (dir_table_offset >= out->size || !out->dir_table_size || (dir_table_offset + out->dir_table_size) > out->size)
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (!out->dir_table_size || (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && (dir_table_offset >= out->size || (dir_table_offset + out->dir_table_size) > out->size)))
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid RomFS directory entries table!");
|
LOGFILE("Invalid RomFS directory entries table!");
|
||||||
return false;
|
return false;
|
||||||
|
@ -111,7 +113,8 @@ bool romfsInitializeContext(RomFileSystemContext *out, NcaFsSectionContext *nca_
|
||||||
file_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_offset : out->header.cur_format.file_entry_offset);
|
file_table_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_offset : out->header.cur_format.file_entry_offset);
|
||||||
out->file_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_size : out->header.cur_format.file_entry_size);
|
out->file_table_size = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.file_entry_size : out->header.cur_format.file_entry_size);
|
||||||
|
|
||||||
if (file_table_offset >= out->size || !out->file_table_size || (file_table_offset + out->file_table_size) > out->size)
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
|
if (!out->file_table_size || (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && (file_table_offset >= out->size || (file_table_offset + out->file_table_size) > out->size)))
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid RomFS file entries table!");
|
LOGFILE("Invalid RomFS file entries table!");
|
||||||
return false;
|
return false;
|
||||||
|
@ -131,8 +134,9 @@ bool romfsInitializeContext(RomFileSystemContext *out, NcaFsSectionContext *nca_
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get file data body offset */
|
/* Get file data body offset */
|
||||||
|
/* Don't verify offsets from Patch RomFS sections, because they reflect the full, patched RomFS image */
|
||||||
out->body_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.body_offset : out->header.cur_format.body_offset);
|
out->body_offset = (nca_fs_ctx->section_type == NcaFsSectionType_Nca0RomFs ? (u64)out->header.old_format.body_offset : out->header.cur_format.body_offset);
|
||||||
if (out->body_offset >= out->size)
|
if (nca_fs_ctx->encryption_type != NcaEncryptionType_AesCtrEx && out->body_offset >= out->size)
|
||||||
{
|
{
|
||||||
LOGFILE("Invalid RomFS file data body!");
|
LOGFILE("Invalid RomFS file data body!");
|
||||||
return false;
|
return false;
|
||||||
|
|
Loading…
Reference in a new issue