1
0
mirror of https://gitlab.os-k.eu/os-k-team/kvisc.git synced 2023-08-25 14:05:46 +02:00
kvisc/vm/pc/main.c
2019-06-20 12:31:36 +02:00

175 lines
3.0 KiB
C

// The OS/K Team licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
#include <sys/time.h>
#include <signal.h>
#include <pc/dev.h>
#include <cn/console.h>
#define FWPROGSIZE (1024 * 1024 * 1024)
static ssize_t fwsize;
static ushort *fwprog;
ushort bget(ctx_t *ctx)
{
ulong addr = rip + cr1;
if (addr2real(addr) >= ctx->mz) {
_except(ctx, E_ACC, "Executing out of memory: "
"rip=0x%lX cr1=0x%lX",
rip, cr1);
}
ushort c = ctx->mp[addr2real(addr)];
//log("bget 0x%lX: 0x%lX\n", addr, c);
rip += 2;
return c;
}
ctx_t main_ctx;
void sigcommon(void)
{
_except(&main_ctx, 1023, "SIGNAL'ed");
}
void sigint(int _)
{
sigcommon();
}
void sigsegv(int _)
{
logerr("Segmentation fault\n");
sigcommon();
}
//
// Main loop
//
void main_loop(void)
{
SDL_Event evt;
//
// Start decoding
//
while (!dying) {
// Execute one instruction
decode(&main_ctx);
if (__builtin_expect(SDL_PollEvent(&evt) == 1, 0))
{
if (evt.type == SDL_QUIT)
die(0);
}
if (main_ctx.step)
getchar();
}
die(0);
}
int main(int argc, char **argv)
{
FILE *fwfile;
main_ctx.r = arch_r;
main_ctx.i = arch_i;
//
// srand
//
struct timeval time;
gettimeofday(&time, NULL);
srandom((time.tv_sec * 1000) + (time.tv_usec / 1000));
//
// Signal handling
//
signal(SIGINT, sigint);
signal(SIGBUS, sigsegv);
signal(SIGSEGV, sigsegv);
//
// Load program
//
if (argc < 3) {
logerr("Not enough arguments\n");
exit(-3);
}
create_symtab(argv[2]);
fwfile = fopen(argv[1], "rb");
if (!fwfile) {
logerr("Couldn't open program file\n");
exit(-2);
}
fwprog = malloc(FWPROGSIZE);
if (!fwprog) {
logerr("Couldn't allocate firmware buffer\n");
exit(-1);
}
fwsize = fread(fwprog, 1, FWPROGSIZE, fwfile);
if (fwsize < 2) {
logerr("Program file too small or empty\n");
free(fwprog);
exit(-3);
}
fclose(fwfile);
//
// Register frame allocation
//
main_ctx.rf = calloc(NREGS, sizeof(ulong));
//
// Memory allocation
//
main_ctx.mp = malloc(MEMSIZE + 16);
main_ctx.mz = MEMSIZE;
main_ctx.get = bget;
main_ctx.rf[RIP] = MEMOFF;
if (main_ctx.mp == 0) {
logerr("Couldn't allocate RAM\n");
free(main_ctx.rf);
free(fwprog);
exit(-1);
}
memcpy(&main_ctx.mp[addr2real(main_ctx.rf[RIP])], fwprog, fwsize);
//
// Devices initialization
//
main_ctx.dh = 0;
if (devinitall(&main_ctx) < 0) {
logerr("Couldn't initialize devices\n");
free(main_ctx.rf);
free(fwprog);
exit(-10);
}
// To be moved to some screen device
console_init(&main_ctx);
disable_stdin_echoing();
main_loop();
}