2018-04-07 23:43:54 +02:00
|
|
|
#include "utils.h"
|
|
|
|
#include "hwinit.h"
|
|
|
|
#include "loader.h"
|
2018-04-08 13:06:04 +02:00
|
|
|
#include "stage2.h"
|
2018-04-09 23:34:23 +02:00
|
|
|
#include "nxboot.h"
|
2018-05-04 19:47:05 +02:00
|
|
|
#include "sd_utils.h"
|
2018-04-08 13:13:15 +02:00
|
|
|
#include "lib/printk.h"
|
|
|
|
#include "display/video_fb.h"
|
2018-05-05 17:33:49 +02:00
|
|
|
#include "fs_dev.h"
|
2018-04-08 13:13:15 +02:00
|
|
|
|
2018-04-09 23:44:59 +02:00
|
|
|
/* TODO: Add a #define for this size, somewhere. Also, primary can only actually load 0x7000. */
|
|
|
|
char g_bct0[0x8000];
|
|
|
|
|
2018-04-07 23:43:54 +02:00
|
|
|
|
2018-04-08 13:06:04 +02:00
|
|
|
/* Allow for main(int argc, void **argv) signature. */
|
|
|
|
#pragma GCC diagnostic ignored "-Wmain"
|
|
|
|
|
2018-05-04 23:56:01 +02:00
|
|
|
void __init_heap(void) {
|
|
|
|
extern char* fake_heap_start;
|
|
|
|
extern char* fake_heap_end;
|
|
|
|
|
|
|
|
fake_heap_start = (char*)0xF0000000;
|
|
|
|
fake_heap_end = (char*)0xFFF00000;
|
|
|
|
}
|
|
|
|
|
2018-04-08 13:06:04 +02:00
|
|
|
int main(int argc, void **argv) {
|
2018-05-04 19:47:05 +02:00
|
|
|
stage2_args_t args = {0};
|
2018-04-09 23:44:59 +02:00
|
|
|
loader_ctx_t *loader_ctx = get_loader_ctx();
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-09 23:44:59 +02:00
|
|
|
if (argc != STAGE2_ARGC || ((args = *((stage2_args_t *)argv[STAGE2_ARGV_ARGUMENT_STRUCT])).version != 0)) {
|
2018-04-08 13:13:15 +02:00
|
|
|
generic_panic();
|
|
|
|
}
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-05-05 17:33:49 +02:00
|
|
|
resume_sd_state((struct mmc *)args.sd_mmc);
|
|
|
|
fsdev_mount_all();
|
|
|
|
fsdev_set_default_device("sdmc");
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-09 23:44:59 +02:00
|
|
|
/* Copy the BCT0 from unsafe primary memory into our memory. */
|
|
|
|
strncpy(g_bct0, args.bct0, sizeof(g_bct0));
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-07 23:43:54 +02:00
|
|
|
/* TODO: What other hardware init should we do here? */
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-08 13:13:15 +02:00
|
|
|
/* Setup LFB. */
|
2018-04-09 23:44:59 +02:00
|
|
|
video_resume(args.lfb, args.console_row, args.console_col);
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-08 13:13:15 +02:00
|
|
|
printk("Welcome to Atmosph\xe8re Fus\xe9" "e Stage 2!\n");
|
2018-04-08 21:51:44 +02:00
|
|
|
printk("Stage 2 executing from: %s\n", (const char *)argv[STAGE2_ARGV_PROGRAM_PATH]);
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-04-07 23:43:54 +02:00
|
|
|
/* This will load all remaining binaries off of the SD. */
|
2018-04-09 23:44:59 +02:00
|
|
|
load_payload(g_bct0);
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-05-04 19:47:05 +02:00
|
|
|
printk("Loaded payloads!\n");
|
2018-05-04 23:56:01 +02:00
|
|
|
|
2018-05-05 17:33:49 +02:00
|
|
|
/* Unmount everything (this causes all open files to be flushed and closed) */
|
|
|
|
fsdev_unmount_all();
|
|
|
|
|
2018-04-09 23:34:23 +02:00
|
|
|
if (loader_ctx->chainload_entrypoint != NULL) {
|
|
|
|
/* TODO: What do we want to do in terms of argc/argv? */
|
|
|
|
loader_ctx->chainload_entrypoint(0, NULL);
|
|
|
|
} else {
|
|
|
|
/* If we don't have a chainload entrypoint set, we're booting Horizon. */
|
|
|
|
nxboot_main();
|
|
|
|
}
|
2018-04-07 23:43:54 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|