2022-02-13 22:12:44 -05:00
|
|
|
#ifndef _STDC_COMMON_H
|
|
|
|
#define _STDC_COMMON_H
|
|
|
|
|
|
|
|
#define signed
|
|
|
|
#define volatile
|
|
|
|
#define register
|
|
|
|
#define const
|
2022-02-14 12:04:49 -05:00
|
|
|
#define NULL ((void*)0)
|
2022-02-13 22:12:44 -05:00
|
|
|
|
|
|
|
typedef unsigned char uint8_t;
|
|
|
|
typedef char int8_t;
|
|
|
|
typedef unsigned short uint16_t;
|
|
|
|
typedef short int16_t;
|
|
|
|
typedef unsigned int uint32_t;
|
|
|
|
typedef int int32_t;
|
|
|
|
typedef unsigned long uint64_t;
|
|
|
|
typedef long int64_t;
|
|
|
|
typedef unsigned long size_t;
|
|
|
|
typedef long ptrdiff_t;
|
|
|
|
typedef unsigned long uintptr_t;
|
|
|
|
typedef long intptr_t;
|
|
|
|
|
2022-02-14 22:43:25 -05:00
|
|
|
#define INT8_MAX 0x7f
|
|
|
|
#define INT8_MIN (-0x80)
|
|
|
|
#define INT16_MAX 0x7fff
|
|
|
|
#define INT16_MIN (-0x8000)
|
|
|
|
#define INT32_MAX 0x7fffffff
|
|
|
|
#define INT32_MIN (-0x80000000)
|
|
|
|
#define INT64_MAX 0x7fffffffffffffff
|
|
|
|
#define INT64_MIN (-0x8000000000000000)
|
|
|
|
#define UINT8_MAX 0xff
|
|
|
|
#define UINT16_MAX 0xffff
|
|
|
|
#define UINT32_MAX 0xffffffff
|
|
|
|
#define UINT64_MAX 0xffffffffffffffff
|
|
|
|
#define CHAR_BIT 8
|
|
|
|
#define MB_LEN_MAX 4
|
|
|
|
#define CHAR_MIN INT8_MIN
|
|
|
|
#define CHAR_MAX INT8_MAX
|
|
|
|
#define SCHAR_MIN INT8_MIN
|
|
|
|
#define SCHAR_MAX INT8_MAX
|
|
|
|
#define INT_MIN INT32_MIN
|
|
|
|
#define INT_MAX INT32_MAX
|
|
|
|
#define LONG_MIN INT64_MIN
|
|
|
|
#define LONG_MAX INT64_MAX
|
|
|
|
#define SHRT_MIN INT16_MIN
|
|
|
|
#define SHRT_MAX INT16_MAX
|
|
|
|
#define UCHAR_MAX UINT8_MAX
|
|
|
|
#define USHRT_MAX UINT16_MAX
|
|
|
|
#define UINT_MAX UINT32_MAX
|
|
|
|
#define ULONG_MAX UINT64_MAX
|
|
|
|
|
2022-02-13 22:12:44 -05:00
|
|
|
static unsigned char __syscall_data[] = {
|
|
|
|
// mov rax, [rsp+24]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 24, 0, 0, 0,
|
|
|
|
// mov rdi, rax
|
|
|
|
0x48, 0x89, 0xc7,
|
|
|
|
// mov rax, [rsp+32]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 32, 0, 0, 0,
|
|
|
|
// mov rsi, rax
|
|
|
|
0x48, 0x89, 0xc6,
|
|
|
|
// mov rax, [rsp+40]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 40, 0, 0, 0,
|
|
|
|
// mov rdx, rax
|
|
|
|
0x48, 0x89, 0xc2,
|
|
|
|
// mov rax, [rsp+48]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 48, 0, 0, 0,
|
|
|
|
// mov r10, rax
|
|
|
|
0x49, 0x89, 0xc2,
|
|
|
|
// mov rax, [rsp+56]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 56, 0, 0, 0,
|
|
|
|
// mov r8, rax
|
|
|
|
0x49, 0x89, 0xc0,
|
|
|
|
// mov rax, [rsp+64]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 64, 0, 0, 0,
|
|
|
|
// mov r9, rax
|
|
|
|
0x49, 0x89, 0xc1,
|
|
|
|
// mov rax, [rsp+16]
|
|
|
|
0x48, 0x8b, 0x84, 0x24, 16, 0, 0, 0,
|
|
|
|
// syscall
|
|
|
|
0x0f, 0x05,
|
|
|
|
// mov [rsp+8], rax
|
|
|
|
0x48, 0x89, 0x84, 0x24, 8, 0, 0, 0,
|
|
|
|
// ret
|
|
|
|
0xc3
|
|
|
|
};
|
|
|
|
|
|
|
|
#define __syscall(no, arg1, arg2, arg3, arg4, arg5, arg6)\
|
|
|
|
(((unsigned long (*)(unsigned long, unsigned long, unsigned long, unsigned long, unsigned long, unsigned long, unsigned long))__syscall_data)\
|
|
|
|
(no, arg1, arg2, arg3, arg4, arg5, arg6))
|
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
long read(int fd, void *buf, size_t count) {
|
|
|
|
return __syscall(0, fd, buf, count, 0, 0, 0);
|
|
|
|
}
|
2022-02-13 22:12:44 -05:00
|
|
|
|
|
|
|
long write(int fd, void *buf, size_t count) {
|
2022-02-14 16:52:33 -05:00
|
|
|
return __syscall(1, fd, buf, count, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _Exit(int status) {
|
|
|
|
return __syscall(60, status, 0, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-15 12:16:21 -05:00
|
|
|
int kill(int pid, int sig) {
|
|
|
|
return __syscall(62, pid, sig, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int getpid(void) {
|
|
|
|
return __syscall(39, 0, 0, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
int fork(void) {
|
|
|
|
return __syscall(57, 0, 0, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int execve(const char *pathname, char *const argv[], char *const envp[]) {
|
|
|
|
return __syscall(59, pathname, argv, envp, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define _WEXITSTATUS(status) (((status) & 0xff00) >> 8)
|
|
|
|
#define _WIFEXITED(status) (__WTERMSIG(status) == 0)
|
|
|
|
#define _WIFSIGNALED(status) \
|
|
|
|
(((signed char) (((status) & 0x7f) + 1) >> 1) > 0)
|
|
|
|
int wait4(int pid, int *status, int options, struct rusage *rusage) {
|
|
|
|
return __syscall(61, pid, status, options, rusage, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-15 12:16:21 -05:00
|
|
|
#define SIGABRT 6
|
|
|
|
#define SIGFPE 8
|
2022-02-16 13:33:35 -05:00
|
|
|
#define SIGKILL 9
|
2022-02-15 12:16:21 -05:00
|
|
|
#define SIGILL 4
|
|
|
|
#define SIGINT 2
|
|
|
|
#define SIGSEGV 11
|
|
|
|
#define SIGTERM 15
|
|
|
|
void abort(void) {
|
|
|
|
kill(getpid(), SIGABRT);
|
|
|
|
}
|
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
typedef long time_t;
|
|
|
|
|
|
|
|
struct timespec {
|
|
|
|
time_t tv_sec;
|
|
|
|
long tv_nsec;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define CLOCK_REALTIME 0
|
|
|
|
#define CLOCK_MONOTONIC 1
|
|
|
|
int clock_gettime(int clock, struct timespec *tp) {
|
|
|
|
return __syscall(228, clock, tp, 0, 0, 0, 0);
|
2022-02-13 22:12:44 -05:00
|
|
|
}
|
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
#define F_OK 0
|
|
|
|
#define R_OK 4
|
|
|
|
#define W_OK 2
|
|
|
|
#define X_OK 1
|
|
|
|
int access(const char *pathname, int mode) {
|
|
|
|
return __syscall(21, pathname, mode, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-15 12:16:21 -05:00
|
|
|
typedef struct {
|
|
|
|
int fd;
|
|
|
|
unsigned char eof;
|
|
|
|
unsigned char err;
|
2022-02-15 15:40:15 -05:00
|
|
|
unsigned char has_ungetc;
|
|
|
|
char ungetc; // character which was pushed by ungetc()
|
2022-02-15 12:16:21 -05:00
|
|
|
} FILE;
|
2022-02-14 16:52:33 -05:00
|
|
|
|
|
|
|
int errno;
|
2022-02-15 12:16:21 -05:00
|
|
|
int printf(char *, ...);
|
|
|
|
int fprintf(FILE *, char *, ...); // needed now for assert()
|
|
|
|
|
|
|
|
FILE _stdin = {0}, *stdin;
|
|
|
|
FILE _stdout = {1}, *stdout;
|
|
|
|
FILE _stderr = {2}, *stderr;
|
|
|
|
|
|
|
|
#ifdef NDEBUG
|
|
|
|
#define assert(x) ((void)0)
|
|
|
|
#else
|
|
|
|
int __assert_failed(const char *file, int line, const char *expr) {
|
|
|
|
fprintf(stderr, "Assertion failed at %s:%d: %s\n", file, line, expr);
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
#define assert(x) (void)((x) || __assert_failed(__FILE__, __LINE__, #x))
|
|
|
|
#endif
|
2022-02-14 16:52:33 -05:00
|
|
|
|
|
|
|
#define EIO 5
|
|
|
|
#define EDOM 33
|
|
|
|
#define ERANGE 34
|
|
|
|
|
|
|
|
#define PROT_READ 1
|
|
|
|
#define PROT_WRITE 2
|
|
|
|
#define PROT_EXEC 4
|
|
|
|
#define MAP_SHARED 0x01
|
|
|
|
#define MAP_ANONYMOUS 0x20
|
|
|
|
#define MAP_PRIVATE 0x02
|
|
|
|
void *mmap(void *addr, size_t length, int prot, int flags, int fd, long offset) {
|
|
|
|
return __syscall(9, addr, length, prot, flags, fd, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
int munmap(void *addr, size_t length) {
|
|
|
|
return __syscall(11, addr, length, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
#define MREMAP_MAYMOVE 1
|
|
|
|
void *_mremap(void *addr, size_t old_size, size_t new_size, int flags) {
|
|
|
|
return __syscall(25, addr, old_size, new_size, flags, 0, 0);
|
|
|
|
}
|
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
void *malloc(size_t n) {
|
2022-02-16 15:41:30 -05:00
|
|
|
if (!n) return NULL;
|
2022-02-14 16:52:33 -05:00
|
|
|
void *memory;
|
|
|
|
size_t bytes = n + 16;
|
|
|
|
memory = mmap(0, bytes, PROT_READ|PROT_WRITE, MAP_ANONYMOUS|MAP_PRIVATE, -1, 0);
|
|
|
|
if ((uint64_t)memory > 0xffffffffffff0000) return NULL;
|
|
|
|
*(uint64_t *)memory = bytes;
|
|
|
|
return (char *)memory + 16;
|
|
|
|
}
|
|
|
|
|
|
|
|
void free(void *ptr) {
|
2022-02-16 15:41:30 -05:00
|
|
|
if (!ptr) return;
|
2022-02-14 16:52:33 -05:00
|
|
|
uint64_t *memory = (char *)ptr - 16;
|
|
|
|
uint64_t size = *memory;
|
|
|
|
munmap(memory, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-02-14 11:39:33 -05:00
|
|
|
size_t strlen(char *s) {
|
|
|
|
char *t = s;
|
|
|
|
while (*t) ++t;
|
|
|
|
return t - s;
|
|
|
|
}
|
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
void *memcpy(void *s1, const void *s2, size_t n) {
|
|
|
|
char *p = s1, *end = p + n, *q = s2;
|
|
|
|
while (p < end)
|
|
|
|
*p++ = *q++;
|
|
|
|
return s1;
|
|
|
|
}
|
|
|
|
|
2022-02-14 22:43:25 -05:00
|
|
|
int isspace(int c) {
|
|
|
|
return c == ' ' || c == '\f' || c == '\n' || c == '\r' || c == '\t' || c == '\v';
|
|
|
|
}
|
|
|
|
|
2022-02-15 15:40:15 -05:00
|
|
|
int isdigit(int c) {
|
|
|
|
return c >= '0' && c <= '9';
|
|
|
|
}
|
|
|
|
|
|
|
|
int _isdigit_in_base(int c, int base) {
|
|
|
|
if (c >= '0' && c <= '9') {
|
|
|
|
return c - '0' < base;
|
|
|
|
} else if (c >= 'a' && c <= 'z') {
|
|
|
|
return c - 'a' + 10 < base;
|
|
|
|
} else if (c >= 'A' && c <= 'Z') {
|
|
|
|
return c - 'A' + 10 < base;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *memset(void *s, int c, size_t n) {
|
|
|
|
char *p = s, *end = p + n;
|
|
|
|
while (p < end)
|
|
|
|
*p++ = c;
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2022-02-14 22:43:25 -05:00
|
|
|
unsigned long strtoul(const char *nptr, char **endptr, int base) {
|
|
|
|
unsigned long value = 0, newvalue;
|
|
|
|
int overflow = 0;
|
|
|
|
|
|
|
|
while (isspace(*nptr)) ++nptr;
|
|
|
|
if (*nptr == '+') ++nptr;
|
|
|
|
if (base == 0) {
|
|
|
|
if (*nptr == '0') {
|
|
|
|
++nptr;
|
|
|
|
switch (*nptr) {
|
|
|
|
case 'x':
|
|
|
|
case 'X':
|
|
|
|
base = 16;
|
|
|
|
++nptr;
|
|
|
|
break;
|
|
|
|
case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7':
|
|
|
|
base = 8;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// this must just be the number 0.
|
|
|
|
if (endptr) *endptr = nptr;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
base = 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
int c = *nptr;
|
|
|
|
unsigned v;
|
|
|
|
if (c >= '0' && c <= '9')
|
|
|
|
v = c - '0';
|
|
|
|
else if (c >= 'a' && c <= 'z')
|
|
|
|
v = c - 'a' + 10;
|
|
|
|
else if (c >= 'A' && c <= 'Z')
|
|
|
|
v = c - 'A' + 10;
|
|
|
|
else break;
|
|
|
|
if (v >= base) break;
|
|
|
|
unsigned long newvalue = value * base + v;
|
|
|
|
if (newvalue < value) overflow = 1;
|
|
|
|
value = newvalue;
|
|
|
|
++nptr;
|
|
|
|
}
|
2022-02-15 12:16:21 -05:00
|
|
|
if (endptr) *endptr = nptr;
|
2022-02-14 22:43:25 -05:00
|
|
|
if (overflow) {
|
|
|
|
errno = ERANGE;
|
|
|
|
return ULONG_MAX;
|
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
long strtol(const char *nptr, char **endptr, int base) {
|
|
|
|
int sign = 1;
|
|
|
|
while (isspace(*nptr)) ++nptr;
|
|
|
|
if (*nptr == '-') {
|
|
|
|
sign = -1;
|
|
|
|
++nptr;
|
|
|
|
}
|
|
|
|
unsigned long mag = strtoul(nptr, endptr, base);
|
|
|
|
if (sign > 0) {
|
|
|
|
if (mag > LONG_MAX) {
|
|
|
|
errno = ERANGE;
|
2022-02-15 15:40:15 -05:00
|
|
|
return LONG_MAX;
|
2022-02-14 22:43:25 -05:00
|
|
|
}
|
|
|
|
return (long)mag;
|
|
|
|
} else {
|
|
|
|
if (mag > (unsigned long)LONG_MAX + 1) {
|
|
|
|
errno = ERANGE;
|
|
|
|
return LONG_MIN;
|
|
|
|
}
|
|
|
|
return -(long)mag;
|
|
|
|
}
|
|
|
|
}
|
2022-02-14 16:52:33 -05:00
|
|
|
|
2022-02-15 12:16:21 -05:00
|
|
|
long _strtol_clamped(const char *nptr, char **endptr, int base, int min, int max) {
|
|
|
|
long l = strtol(nptr, endptr, base);
|
|
|
|
if (l < min) return min;
|
|
|
|
if (l > max) return max;
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define _NPOW10 310
|
|
|
|
#define _INFINITY 1e1000
|
|
|
|
// non-negative floating-point number with more precision than a double
|
|
|
|
// its value is equal to fraction * 2^exponent
|
2022-02-14 16:52:33 -05:00
|
|
|
typedef struct {
|
2022-02-15 12:16:21 -05:00
|
|
|
unsigned long fraction;
|
|
|
|
int exponent;
|
|
|
|
} _Float;
|
|
|
|
|
|
|
|
// ensure that f->fraction >= 2^64 / 2
|
|
|
|
static void _normalize_float(_Float *f) {
|
|
|
|
if (!f->fraction) return;
|
|
|
|
while (f->fraction < 0x8000000000000000) {
|
|
|
|
f->exponent -= 1;
|
|
|
|
f->fraction <<= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static double _Float_to_double(_Float f) {
|
|
|
|
unsigned long dbl_fraction;
|
|
|
|
int dbl_exponent;
|
|
|
|
unsigned long dbl_value;
|
|
|
|
if (f.fraction == 0) return 0;
|
|
|
|
_normalize_float(&f);
|
|
|
|
f.fraction &= 0x7fffffffffffffff; // remove the "1." in 1.01101110111... to get 63-bit significand
|
|
|
|
dbl_fraction = (f.fraction + 0x3ff) >> 11;
|
|
|
|
dbl_exponent = f.exponent + 63;
|
|
|
|
if (dbl_exponent < -1022) return 0;
|
|
|
|
if (dbl_exponent > 1023) return _INFINITY;
|
|
|
|
dbl_exponent += 1023;
|
|
|
|
dbl_value = (unsigned long)dbl_exponent << 52 | dbl_fraction;
|
|
|
|
return *(double *)&dbl_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static _Float _powers_of_10_dat[2*_NPOW10+1];
|
|
|
|
static _Float *_powers_of_10;
|
|
|
|
static _Float _Float_ZERO = {0, 1};
|
|
|
|
static _Float _Float_INFINITY = {0x8000000000000000, 100000};
|
|
|
|
|
|
|
|
|
|
|
|
_Float _int_pow10(int x) {
|
|
|
|
if (x <= -_NPOW10) return _Float_ZERO;
|
|
|
|
if (x >= _NPOW10) return _Float_INFINITY;
|
|
|
|
return _powers_of_10[x];
|
|
|
|
}
|
|
|
|
|
|
|
|
double strtod(const char *nptr, char **endptr) {
|
|
|
|
const char *flt, *dot, *p, *number_end;
|
|
|
|
double sign = 1;
|
|
|
|
int exponent = 0;
|
|
|
|
while (isspace(*nptr)) ++nptr;
|
|
|
|
|
|
|
|
flt = nptr; // start of float
|
|
|
|
if (*flt == '+') ++flt;
|
|
|
|
else if (*flt == '-') sign = -1, ++flt;
|
|
|
|
|
|
|
|
if (*flt != '.' && (*flt < '0' || *flt > '9')) {
|
|
|
|
// this isn't a float
|
|
|
|
*endptr = nptr;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// find the decimal point, if any
|
|
|
|
dot = flt;
|
|
|
|
while (*dot >= '0' && *dot <= '9') ++dot;
|
|
|
|
|
|
|
|
nptr = dot + (*dot == '.');
|
|
|
|
// skip digits after the dot
|
|
|
|
while (*nptr >= '0' && *nptr <= '9') ++nptr;
|
|
|
|
number_end = nptr;
|
|
|
|
|
|
|
|
if (*nptr == 'e') {
|
|
|
|
++nptr;
|
|
|
|
exponent = 1;
|
|
|
|
if (*nptr == '+') ++nptr;
|
|
|
|
else if (*nptr == '-') ++nptr, exponent = -1;
|
|
|
|
exponent *= _strtol_clamped(nptr, &nptr, 10, -10000, 10000); // use _strtol_clamped to prevent problems with -LONG_MIN
|
|
|
|
}
|
|
|
|
|
|
|
|
// construct the value using the Kahan summation algorithm (https://en.wikipedia.org/wiki/Kahan_summation_algorithm)
|
|
|
|
double sum = 0;
|
|
|
|
double c = 0;
|
|
|
|
for (p = flt; p < number_end; ++p) {
|
|
|
|
if (*p == '.') continue;
|
|
|
|
int n = *p - '0';
|
|
|
|
assert(n >= 0 && n <= 9);
|
|
|
|
int pow10 = dot - p;
|
|
|
|
pow10 -= pow10 > 0;
|
|
|
|
pow10 += exponent;
|
|
|
|
_Float f_val = _int_pow10(pow10);
|
|
|
|
f_val.fraction >>= 4;
|
|
|
|
f_val.exponent += 4;
|
|
|
|
f_val.fraction *= n;
|
|
|
|
double value = _Float_to_double(f_val);
|
|
|
|
if (value == _INFINITY || sum == _INFINITY) {
|
|
|
|
sum = _INFINITY;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
double y = value - c;
|
|
|
|
double t = sum + y;
|
|
|
|
c = (t - sum) - y;
|
|
|
|
sum = t;
|
|
|
|
}
|
|
|
|
|
2022-02-15 22:41:18 -05:00
|
|
|
if (sum == _INFINITY) errno = ERANGE;
|
2022-02-15 12:16:21 -05:00
|
|
|
if (endptr) *endptr = nptr;
|
|
|
|
return sum * sign;
|
|
|
|
}
|
2022-02-14 16:52:33 -05:00
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
char *strerror(int errnum) {
|
|
|
|
switch (errnum) {
|
|
|
|
case ERANGE: return "Range error";
|
|
|
|
case EDOM: return "Domain error";
|
|
|
|
case EIO: return "I/O error";
|
|
|
|
}
|
|
|
|
return "Other error";
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef void (*_ExitHandler)(void);
|
|
|
|
_ExitHandler _exit_handlers[33];
|
|
|
|
int _n_exit_handlers;
|
|
|
|
|
|
|
|
void exit(int status) {
|
|
|
|
int i;
|
|
|
|
for (i = _n_exit_handlers - 1; i >= 0; --i)
|
|
|
|
_exit_handlers[i]();
|
|
|
|
_Exit(status);
|
|
|
|
}
|
2022-02-14 16:52:33 -05:00
|
|
|
|
|
|
|
int main();
|
|
|
|
|
2022-02-15 16:36:52 -05:00
|
|
|
static char **_envp;
|
2022-02-16 15:41:30 -05:00
|
|
|
static uint64_t _rand_seed;
|
2022-02-15 16:36:52 -05:00
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
int _main(int argc, char **argv) {
|
2022-02-15 12:16:21 -05:00
|
|
|
int i;
|
|
|
|
_Float p = {1, 0};
|
|
|
|
|
2022-02-15 16:36:52 -05:00
|
|
|
_envp = argv + argc + 1; // this is where the environment variables will be
|
|
|
|
|
2022-02-14 16:52:33 -05:00
|
|
|
stdin = &_stdin;
|
|
|
|
stdout = &_stdout;
|
|
|
|
stderr = &_stderr;
|
2022-02-15 12:16:21 -05:00
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
/*
|
|
|
|
"If rand is called before any calls to srand have been made,
|
|
|
|
the same sequence shall be generated as when srand is first
|
|
|
|
called with a seed value of 1." C89 § 4.10.2.2
|
|
|
|
*/
|
|
|
|
_rand_seed = 1;
|
2022-02-15 12:16:21 -05:00
|
|
|
// initialize powers of 10
|
|
|
|
_powers_of_10 = _powers_of_10_dat + _NPOW10;
|
|
|
|
for (i = 0; i < _NPOW10; ++i) {
|
|
|
|
_normalize_float(&p);
|
|
|
|
_powers_of_10[i] = p;
|
|
|
|
p.exponent += 4;
|
|
|
|
p.fraction >>= 4;
|
|
|
|
p.fraction *= 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
p.fraction = 1;
|
|
|
|
p.exponent = 0;
|
|
|
|
for (i = 0; i > -_NPOW10; --i) {
|
|
|
|
_normalize_float(&p);
|
|
|
|
_powers_of_10[i] = p;
|
|
|
|
p.fraction /= 5;
|
|
|
|
p.exponent -= 1;
|
|
|
|
}
|
|
|
|
|
2022-02-16 15:41:30 -05:00
|
|
|
exit(main(argc, argv));
|
2022-02-14 16:52:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-02-13 22:12:44 -05:00
|
|
|
#endif // _STDC_COMMON_H
|