shithub: femtolisp

ref: f51ee8f9575d4eb6e8a58a61ff0e2e92092cbb1c
dir: /flmain.c/

View raw version
#include "flisp.h"
#include "cvalues.h"
#include "print.h"
#include "iostream.h"
#include "random.h"
#include "brieflz.h"
#include "nan.h"

static value_t
argv_list(int argc, char *argv[])
{
	int i;
	value_t lst = FL_nil, temp;
	fl_gc_handle(&lst);
	fl_gc_handle(&temp);
	for(i = argc-1; i >= 0; i--){
		temp = cvalue_static_cstring(argv[i]);
		lst = fl_cons(temp, lst);
	}
	fl_free_gc_handles(2);
	return lst;
}

_Noreturn void
flmain(const uint8_t *boot, int bootsz, int argc, char **argv)
{
	nan_init();
	randomize();
	ios_init_stdstreams();
	mpsetminbits(sizeof(fixnum_t)*8);

	if(fl_init(INITIAL_HEAP_SIZE) != 0)
		exit(1);

	value_t f = cvalue(FL(iostreamtype), (int)sizeof(ios_t));
	ios_t *s = value2c(ios_t*, f);
	uint8_t *unpacked = nil;
	if(boot[0] == 0){
		uint32_t unpackedsz =
			boot[1]<<0 |
			boot[2]<<8 |
			boot[3]<<16|
			boot[4]<<24;
		unpacked = MEM_ALLOC(unpackedsz);
		unsigned long n = blz_depack_safe(boot+5, bootsz-5, unpacked, unpackedsz);
		if(n == BLZ_ERROR){
			ios_puts(ios_stderr, "failed to unpack boot image\n");
			fl_exit(1);
		}
		boot = unpacked;
		bootsz = n;
	}
	ios_static_buffer(s, boot, bootsz);

	int r = 1;
	FL_TRY_EXTERN{
		if(fl_load_system_image(f) == 0){
			MEM_FREE(unpacked);
			ios_close(s);
			fl_applyn(1, symbol_value(symbol("__start", false)), argv_list(argc, argv));
			r = 0;
		}
	}
	FL_CATCH_EXTERN_NO_RESTORE{
		ios_puts(ios_stderr, "fatal error:\n");
		fl_print(ios_stderr, FL(lasterror));
		ios_putc(ios_stderr, '\n');
		break;
	}
	fl_exit(r);
}