2018-04-19 13:24:26 +02:00
|
|
|
//---
|
2019-07-04 18:11:43 +02:00
|
|
|
// gint:core:start - Kernel initialization and C runtime
|
2018-04-19 13:24:26 +02:00
|
|
|
//--
|
|
|
|
|
2019-02-21 20:58:38 +01:00
|
|
|
#include <gint/defs/attributes.h>
|
|
|
|
#include <gint/defs/types.h>
|
2020-06-14 11:01:27 +02:00
|
|
|
#include <gint/mmu.h>
|
2018-04-19 13:24:26 +02:00
|
|
|
#include <gint/drivers.h>
|
|
|
|
#include <gint/gint.h>
|
2019-07-04 18:11:43 +02:00
|
|
|
#include <gint/hardware.h>
|
2019-09-03 22:15:00 +02:00
|
|
|
#include <gint/exc.h>
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
#include "kernel.h"
|
|
|
|
|
2018-04-19 13:24:26 +02:00
|
|
|
/* Symbols provided by the linker script. For sections:
|
|
|
|
- l* represents the load address (source address in ROM)
|
|
|
|
- s* represents the size of the section
|
|
|
|
- r* represents the relocation address (destination address in RAM)
|
|
|
|
gint's BSS section is not mentioned here because it's never initialized */
|
|
|
|
extern uint32_t
|
|
|
|
brom, srom, /* Limits of ROM mappings */
|
|
|
|
lgdata, sgdata, rgdata, /* gint's data section */
|
|
|
|
ldata, sdata, rdata, /* User's data section */
|
2019-09-15 19:29:47 +02:00
|
|
|
lilram, silram, rilram, /* IL memory section */
|
|
|
|
lxram, sxram, rxram, /* X memory section */
|
|
|
|
lyram, syram, ryram, /* Y memory section */
|
2020-06-20 17:18:51 +02:00
|
|
|
sbss, rbss; /* User's BSS section */
|
|
|
|
|
|
|
|
/* Constructor and destructor arrays */
|
|
|
|
extern void (*bctors)(void), (*ectors)(void);
|
|
|
|
extern void (*bdtors)(void), (*edtors)(void);
|
2018-04-19 13:24:26 +02:00
|
|
|
|
|
|
|
/* User-provided main() function */
|
|
|
|
int main(int isappli, int optnum);
|
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* regcpy(): Copy a memory region using symbol information
|
2018-04-19 13:24:26 +02:00
|
|
|
@l Source pointer (load address)
|
|
|
|
@s Size of area (should be a multiple of 16)
|
|
|
|
@r Destination pointer (relocation address) */
|
|
|
|
static void regcpy(uint32_t * restrict l, int32_t s, uint32_t * restrict r)
|
|
|
|
{
|
|
|
|
while(s > 0)
|
|
|
|
{
|
|
|
|
*r++ = *l++;
|
|
|
|
*r++ = *l++;
|
|
|
|
*r++ = *l++;
|
|
|
|
*r++ = *l++;
|
|
|
|
s -= 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#define regcpy(l, s, r) regcpy(&l, (int32_t)&s, &r)
|
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* regclr(): Clear a memory region using symbol information
|
2018-04-19 13:24:26 +02:00
|
|
|
@r Source pointer (base address)
|
|
|
|
@s Size of area (should be a multiple of 16) */
|
|
|
|
static void regclr(uint32_t *r, int32_t s)
|
|
|
|
{
|
|
|
|
while(s > 0)
|
|
|
|
{
|
|
|
|
*r++ = 0;
|
|
|
|
*r++ = 0;
|
|
|
|
*r++ = 0;
|
|
|
|
*r++ = 0;
|
|
|
|
s -= 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#define regclr(r, s) regclr(&r, (int32_t)&s)
|
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* callarray(): Call an array of functions (constructors or destructors)
|
2018-04-19 13:24:26 +02:00
|
|
|
@f First element of array
|
|
|
|
@l First element outside of the array */
|
2020-06-20 17:18:51 +02:00
|
|
|
static void callarray(void (**f)(void), void (**l)(void))
|
2018-04-19 13:24:26 +02:00
|
|
|
{
|
|
|
|
while(f < l) (*(*f++))();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start() - this is where it all starts
|
|
|
|
Returns a status code. Invoking main menu is better than returning! */
|
2020-06-20 17:18:51 +02:00
|
|
|
GSECTION(".text.entry")
|
2018-04-19 13:24:26 +02:00
|
|
|
int start(int isappli, int optnum)
|
|
|
|
{
|
|
|
|
/* We are currently in a dynamic userspace mapping of an add-in run
|
|
|
|
from the storage memory. We are running in privileged mode with one
|
|
|
|
golden rule:
|
|
|
|
|
|
|
|
Do not disturb the operating system.
|
|
|
|
|
|
|
|
gint will silently run in an isolated part of the memory (fx9860g)
|
2020-06-20 17:18:51 +02:00
|
|
|
or at the start of the RAM (fxcg50). The kernel will redirect
|
|
|
|
interrupts and load its own drivers, so we can't rely too much on
|
|
|
|
the system. Ladies and gentlemen, let's have fun! ;D */
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-15 20:55:18 +02:00
|
|
|
/* For now, we use the system's memory mapper for ROM. We'll still do
|
2020-06-20 17:18:51 +02:00
|
|
|
it later in our TLB miss handler once we're installed. RAM is always
|
2018-04-19 13:24:26 +02:00
|
|
|
fully mapped, but we need to initialize it. We also need to do some
|
2019-03-06 14:32:51 +01:00
|
|
|
hardware detection because old fx9860g models have a different
|
2020-06-15 20:55:18 +02:00
|
|
|
processor with some incompatible features. */
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* Detect architecture; this will tell SH3 from SH4 on fx9860g */
|
2019-07-04 18:11:43 +02:00
|
|
|
hw_detect();
|
2018-04-19 13:24:26 +02:00
|
|
|
|
|
|
|
/* Load data sections and wipe the bss section. This has to be done
|
|
|
|
first for static and global variables to be initialized */
|
|
|
|
regcpy(lgdata, sgdata, rgdata);
|
2019-09-15 19:29:47 +02:00
|
|
|
regcpy(ldata, sdata, rdata);
|
|
|
|
regcpy(lilram, silram, rilram);
|
|
|
|
regcpy(lxram, sxram, rxram);
|
|
|
|
regcpy(lyram, syram, ryram);
|
2018-04-19 13:24:26 +02:00
|
|
|
regclr(rbss, sbss);
|
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* Install gint, switch VBR and initialize drivers */
|
|
|
|
kinit();
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2019-02-21 20:58:38 +01:00
|
|
|
/* We are now running on our own in kernel mode. Since we have taken
|
2019-03-06 14:32:51 +01:00
|
|
|
control of interrupts, pretty much any interaction with the system
|
2019-02-21 20:58:38 +01:00
|
|
|
will break it. We'll limit our use of syscalls and do device driving
|
2020-06-15 20:55:18 +02:00
|
|
|
ourselves. (Hopefully we can add cool features in the process!) */
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* Now that we have initialized the kernel, we are ready to start the
|
|
|
|
hosted user application, which has its own constructors and
|
|
|
|
destructors to work with. */
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
callarray(&bctors, &ectors);
|
2018-04-19 13:24:26 +02:00
|
|
|
int ret = main(isappli, optnum);
|
2020-06-20 17:18:51 +02:00
|
|
|
callarray(&bdtors, &edtors);
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* Before leaving the application, we need to clean everything we
|
|
|
|
changed to hardware settings and peripheral modules. The OS is bound
|
|
|
|
to be confused (and hang, or crash, or any other kind of giving up)
|
|
|
|
if we don't restore them. */
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2018-08-01 20:41:36 +02:00
|
|
|
/* Unload gint and give back control to the system. Driver settings
|
|
|
|
will be restored while interrupts are disabled */
|
2020-06-20 17:18:51 +02:00
|
|
|
kquit();
|
2018-04-19 13:24:26 +02:00
|
|
|
|
2020-06-20 17:18:51 +02:00
|
|
|
/* TODO: Invoke main menu so that add-in can restart? */
|
2018-04-19 13:24:26 +02:00
|
|
|
return ret;
|
|
|
|
}
|