mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-25 16:32:55 +00:00
109 lines
No EOL
4.2 KiB
C
109 lines
No EOL
4.2 KiB
C
#include "utils.h"
|
|
#include "package2.h"
|
|
#include "stratosphere.h"
|
|
#include "sd_utils.h"
|
|
#include "lib/printk.h"
|
|
#include "lib/vsprintf.h"
|
|
|
|
unsigned char g_stratosphere_ini1[PACKAGE2_SIZE_MAX];
|
|
static bool g_initialized_stratosphere_ini1 = false;
|
|
|
|
unsigned char g_ini1_buffer[PACKAGE2_SIZE_MAX];
|
|
|
|
ini1_header_t *stratosphere_get_ini1(void) {
|
|
ini1_header_t *ini1_header = (ini1_header_t *)g_stratosphere_ini1;
|
|
if (g_initialized_stratosphere_ini1) {
|
|
return ini1_header;
|
|
}
|
|
ini1_header->magic = MAGIC_INI1;
|
|
ini1_header->size = sizeof(ini1_header_t);
|
|
ini1_header->num_processes = 0;
|
|
ini1_header->_0xC = 0;
|
|
|
|
/* TODO: When we have processes, copy them into ini1_header->kip_data here. */
|
|
|
|
g_initialized_stratosphere_ini1 = true;
|
|
return ini1_header;
|
|
}
|
|
|
|
/* Merges some number of INI1s into a single INI1. It's assumed that the INIs are in order of preference. */
|
|
size_t stratosphere_merge_inis(void *dst, ini1_header_t **inis, unsigned int num_inis) {
|
|
char sd_path[0x300] = {0};
|
|
/* Validate all ini headers. */
|
|
for (unsigned int i = 0; i < num_inis; i++) {
|
|
if (inis[i] == NULL || inis[i]->magic != MAGIC_INI1 || inis[i]->num_processes > INI1_MAX_KIPS) {
|
|
printk("Error: INI1s[%d] section appears to not contain an INI1!\n", i);
|
|
generic_panic();
|
|
}
|
|
}
|
|
|
|
uint64_t process_list[INI1_MAX_KIPS] = {0};
|
|
|
|
memset(g_ini1_buffer, 0, sizeof(g_ini1_buffer));
|
|
ini1_header_t *merged = (ini1_header_t *)g_ini1_buffer;
|
|
merged->magic = MAGIC_INI1;
|
|
merged->num_processes = 0;
|
|
merged->_0xC = 0;
|
|
size_t remaining_size = PACKAGE2_SIZE_MAX - sizeof(ini1_header_t);
|
|
|
|
unsigned char *current_dst_kip = merged->kip_data;
|
|
|
|
/* Actually merge into the inis. */
|
|
for (unsigned int i = 0; i < num_inis; i++) {
|
|
uint64_t offset = 0;
|
|
for (unsigned int p = 0; p < inis[i]->num_processes; p++) {
|
|
kip1_header_t *current_kip = (kip1_header_t *)(inis[i]->kip_data + offset);
|
|
if (current_kip->magic != MAGIC_KIP1) {
|
|
printk("Error: INI1s[%d][%d] appears not to be a KIP1!\n", i, p);
|
|
generic_panic();
|
|
}
|
|
|
|
|
|
bool already_loaded = false;
|
|
for (unsigned int j = 0; j < merged->num_processes; j++) {
|
|
if (process_list[j] == current_kip->title_id) {
|
|
already_loaded = true;
|
|
break;
|
|
}
|
|
}
|
|
if (already_loaded) {
|
|
continue;
|
|
}
|
|
|
|
/* TODO: What folder should these be read out of? */
|
|
snprintf(sd_path, sizeof(sd_path), "atmosph\xe8re/titles/%016llx/%016llx.kip", current_kip->title_id, current_kip->title_id);
|
|
|
|
/* Try to load an override KIP from SD, if possible. */
|
|
if (read_sd_file(current_dst_kip, remaining_size, sd_path)) {
|
|
kip1_header_t *sd_kip = (kip1_header_t *)(current_dst_kip);
|
|
if (sd_kip->magic != MAGIC_KIP1) {
|
|
printk("Error: %s is not a KIP1?\n", sd_path);
|
|
generic_panic();
|
|
} else if (sd_kip->title_id != current_kip->title_id) {
|
|
printk("Error: %s has wrong Title ID!\n", sd_path);
|
|
generic_panic();
|
|
}
|
|
uint64_t sd_kip_size = kip1_get_size_from_header(sd_kip);
|
|
remaining_size -= sd_kip_size;
|
|
current_dst_kip += sd_kip_size;
|
|
} else {
|
|
uint64_t current_kip_size = kip1_get_size_from_header(current_kip);
|
|
if (current_kip_size > remaining_size) {
|
|
printk("Error: Not enough space for all the KIP1s!\n");
|
|
generic_panic();
|
|
}
|
|
memcpy(current_dst_kip, current_kip, current_kip_size);
|
|
remaining_size -= current_kip_size;
|
|
current_dst_kip += current_kip_size;
|
|
}
|
|
|
|
process_list[merged->num_processes++] = current_kip->title_id;
|
|
}
|
|
}
|
|
merged->size = sizeof(ini1_header_t) + (uint32_t)(current_dst_kip - merged->kip_data);
|
|
|
|
/* Copy merged INI1 to destination. */
|
|
memcpy(dst, merged, merged->size);
|
|
|
|
return merged->size;
|
|
} |