toaruos/libc/main.c
2021-05-31 10:54:11 +09:00

119 lines
2.3 KiB
C

#include <stdint.h>
#include <stddef.h>
#include <stdlib.h>
#include <syscall.h>
#include <syscall_nums.h>
DEFN_SYSCALL1(exit, SYS_EXT, int);
DEFN_SYSCALL2(sleepabs, SYS_SLEEPABS, unsigned long, unsigned long);
extern void _init();
extern void _fini();
char ** environ = NULL;
int _environ_size = 0;
char * _argv_0 = NULL;
int __libc_debug = 0;
char ** __argv = NULL;
extern char ** __get_argv(void) {
return __argv;
}
extern void __stdio_init_buffers(void);
extern void __stdio_cleanup(void);
void _exit(int val){
_fini();
__stdio_cleanup();
syscall_exit(val);
__builtin_unreachable();
}
extern void __make_tls(void);
static int __libc_init_called = 0;
__attribute__((constructor))
static void _libc_init(void) {
__libc_init_called = 1;
__make_tls();
__stdio_init_buffers();
unsigned int x = 0;
unsigned int nulls = 0;
for (x = 0; 1; ++x) {
if (!__get_argv()[x]) {
++nulls;
if (nulls == 2) {
break;
}
continue;
}
if (nulls == 1) {
environ = &__get_argv()[x];
break;
}
}
if (!environ) {
environ = malloc(sizeof(char *) * 4);
environ[0] = NULL;
environ[1] = NULL;
environ[2] = NULL;
environ[3] = NULL;
_environ_size = 4;
} else {
/* Find actual size */
int size = 0;
char ** tmp = environ;
while (*tmp) {
size++;
tmp++;
}
if (size < 4) {
_environ_size = 4;
} else {
/* Multiply by two */
_environ_size = size * 2;
}
char ** new_environ = malloc(sizeof(char*) * _environ_size);
int i = 0;
while (i < _environ_size && environ[i]) {
new_environ[i] = environ[i];
i++;
}
while (i < _environ_size) {
new_environ[i] = NULL;
i++;
}
environ = new_environ;
}
if (getenv("__LIBC_DEBUG")) __libc_debug = 1;
_argv_0 = __get_argv()[0];
}
void pre_main(int argc, char * argv[], char ** envp, int (*main)(int,char**)) {
if (!__get_argv()) {
/* Statically loaded, must set __argv so __get_argv() works */
__argv = argv;
/* Run our initializers, because I'm pretty sure the kernel didn't... */
if (!__libc_init_called) {
extern uintptr_t __init_array_start;
extern uintptr_t __init_array_end;
for (uintptr_t * constructor = &__init_array_start; constructor < &__init_array_end; ++constructor) {
void (*constr)(void) = (void*)*constructor;
constr();
}
}
}
_init();
exit(main(argc, argv));
}