procuding a (non-working) executable for tcc
This commit is contained in:
parent
6e1158f49a
commit
e900dd8d6f
71 changed files with 42663 additions and 96 deletions
|
@ -2902,6 +2902,7 @@ function generate_function
|
||||||
local out0
|
local out0
|
||||||
local n_stack_bytes
|
local n_stack_bytes
|
||||||
|
|
||||||
|
debug_puts(.str_gen_code_for)
|
||||||
debug_putsln(function_name)
|
debug_putsln(function_name)
|
||||||
|
|
||||||
function_type = ident_list_lookup(function_types, function_name)
|
function_type = ident_list_lookup(function_types, function_name)
|
||||||
|
@ -2930,6 +2931,10 @@ function generate_function
|
||||||
generate_return()
|
generate_return()
|
||||||
|
|
||||||
return
|
return
|
||||||
|
:str_gen_code_for
|
||||||
|
string Generating code for
|
||||||
|
byte 32
|
||||||
|
byte 0
|
||||||
|
|
||||||
function generate_functions
|
function generate_functions
|
||||||
local addr
|
local addr
|
||||||
|
|
19
05/main.b
19
05/main.b
|
@ -1,4 +1,4 @@
|
||||||
#define G_DEBUG 0
|
#define G_DEBUG 1
|
||||||
|
|
||||||
; add 24 + 16 = 40 to the stack pointer to put argc, argv in the right place
|
; add 24 + 16 = 40 to the stack pointer to put argc, argv in the right place
|
||||||
byte 0x48
|
byte 0x48
|
||||||
|
@ -201,6 +201,18 @@ function compile_warning
|
||||||
byte 32
|
byte 32
|
||||||
byte 0
|
byte 0
|
||||||
|
|
||||||
|
:str_preprocessing
|
||||||
|
string Preprocessing...
|
||||||
|
byte 10
|
||||||
|
byte 0
|
||||||
|
:str_tokenizing
|
||||||
|
string Turning preprocessing tokens into tokens...
|
||||||
|
byte 10
|
||||||
|
byte 0
|
||||||
|
:str_parsing
|
||||||
|
string Parsing...
|
||||||
|
byte 10
|
||||||
|
byte 0
|
||||||
|
|
||||||
function main
|
function main
|
||||||
argument argv2
|
argument argv2
|
||||||
|
@ -292,6 +304,8 @@ function main
|
||||||
output_file_data = mmap(0, RWDATA_END, PROT_READ_WRITE, MAP_SHARED, output_fd, 0)
|
output_file_data = mmap(0, RWDATA_END, PROT_READ_WRITE, MAP_SHARED, output_fd, 0)
|
||||||
if output_file_data ] 0xffffffffffff0000 goto mmap_output_fd_failed
|
if output_file_data ] 0xffffffffffff0000 goto mmap_output_fd_failed
|
||||||
|
|
||||||
|
debug_puts(.str_preprocessing)
|
||||||
|
|
||||||
pptokens = split_into_preprocessing_tokens(input_filename)
|
pptokens = split_into_preprocessing_tokens(input_filename)
|
||||||
;print_pptokens(pptokens)
|
;print_pptokens(pptokens)
|
||||||
;print_separator()
|
;print_separator()
|
||||||
|
@ -304,12 +318,15 @@ function main
|
||||||
;print_object_macros()
|
;print_object_macros()
|
||||||
;print_function_macros()
|
;print_function_macros()
|
||||||
|
|
||||||
|
debug_puts(.str_tokenizing)
|
||||||
|
|
||||||
tokens = malloc(16000000)
|
tokens = malloc(16000000)
|
||||||
p = tokenize(pptokens, tokens, input_filename, 1)
|
p = tokenize(pptokens, tokens, input_filename, 1)
|
||||||
;print_tokens(tokens, p)
|
;print_tokens(tokens, p)
|
||||||
;print_separator()
|
;print_separator()
|
||||||
; NOTE: do NOT free pptokens; identifiers still reference them.
|
; NOTE: do NOT free pptokens; identifiers still reference them.
|
||||||
|
|
||||||
|
debug_puts(.str_parsing)
|
||||||
parse_tokens(tokens)
|
parse_tokens(tokens)
|
||||||
generate_code()
|
generate_code()
|
||||||
|
|
||||||
|
|
|
@ -6,10 +6,12 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
#include <float.h>
|
#include <float.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
srand(time(NULL));
|
jmp_buf test;
|
||||||
printf("%d\n",rand());
|
setjmp(test);
|
||||||
|
longjmp(test, 5);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
25
05/parse.b
25
05/parse.b
|
@ -344,13 +344,20 @@ function parse_toplevel_declaration
|
||||||
|
|
||||||
ident_list_add(functions_required_stack_space, f_name, curr_function_stack_space)
|
ident_list_add(functions_required_stack_space, f_name, curr_function_stack_space)
|
||||||
|
|
||||||
; ENABLE/DISABLE PARSING DEBUG OUTPUT:
|
debug_puts(.str_parsed)
|
||||||
if G_DEBUG == 0 goto skip_print_statement
|
debug_putsln(f_name)
|
||||||
print_statement(out0)
|
|
||||||
:skip_print_statement
|
;PARSING DEBUG OUTPUT:
|
||||||
|
;if G_DEBUG == 0 goto skip_print_statement
|
||||||
|
;print_statement(out0)
|
||||||
|
;:skip_print_statement
|
||||||
|
|
||||||
goto parse_tld_ret
|
goto parse_tld_ret
|
||||||
|
|
||||||
|
:str_parsed
|
||||||
|
string Parsed
|
||||||
|
byte 32
|
||||||
|
byte 0
|
||||||
:function_no_param_name
|
:function_no_param_name
|
||||||
token_error(base_type, .str_function_no_param_name)
|
token_error(base_type, .str_function_no_param_name)
|
||||||
:str_function_no_param_name
|
:str_function_no_param_name
|
||||||
|
@ -2127,7 +2134,7 @@ function parse_base_type
|
||||||
goto base_type_normal_loop
|
goto base_type_normal_loop
|
||||||
:base_type_flag_long
|
:base_type_flag_long
|
||||||
c = flags & PARSETYPE_FLAG_LONG
|
c = flags & PARSETYPE_FLAG_LONG
|
||||||
if c != 0 goto repeated_base_type
|
;if c != 0 goto repeated_base_type ; allow "long long" ...and also "long long long" i guess
|
||||||
flags |= PARSETYPE_FLAG_LONG
|
flags |= PARSETYPE_FLAG_LONG
|
||||||
goto base_type_normal_loop
|
goto base_type_normal_loop
|
||||||
:base_type_flag_unsigned
|
:base_type_flag_unsigned
|
||||||
|
@ -3299,9 +3306,13 @@ function parse_expression
|
||||||
return out
|
return out
|
||||||
:undeclared_variable
|
:undeclared_variable
|
||||||
; @NONSTANDARD: C89 allows calling functions without declaring them
|
; @NONSTANDARD: C89 allows calling functions without declaring them
|
||||||
token_error(in, .str_undeclared_variable)
|
print_token_location(in)
|
||||||
|
puts(.str_undeclared_variable)
|
||||||
|
putsln(a)
|
||||||
|
exit(1)
|
||||||
:str_undeclared_variable
|
:str_undeclared_variable
|
||||||
string Undeclared variable.
|
string : Undeclared variable:
|
||||||
|
byte 32
|
||||||
byte 0
|
byte 0
|
||||||
|
|
||||||
:found_local_variable
|
:found_local_variable
|
||||||
|
|
|
@ -850,6 +850,8 @@ function translation_phase_4
|
||||||
|
|
||||||
local included_pptokens
|
local included_pptokens
|
||||||
included_pptokens = split_into_preprocessing_tokens(inc_filename)
|
included_pptokens = split_into_preprocessing_tokens(inc_filename)
|
||||||
|
debug_puts(.str_including)
|
||||||
|
debug_putsln(inc_filename)
|
||||||
out = translation_phase_4(inc_filename, included_pptokens, out)
|
out = translation_phase_4(inc_filename, included_pptokens, out)
|
||||||
free(included_pptokens)
|
free(included_pptokens)
|
||||||
free(inc_filename)
|
free(inc_filename)
|
||||||
|
@ -865,6 +867,10 @@ function translation_phase_4
|
||||||
out += 1
|
out += 1
|
||||||
|
|
||||||
goto process_pptoken
|
goto process_pptoken
|
||||||
|
:str_including
|
||||||
|
string Including
|
||||||
|
byte 32
|
||||||
|
byte 0
|
||||||
:pp_directive_ifdef
|
:pp_directive_ifdef
|
||||||
pptoken_skip(&in)
|
pptoken_skip(&in)
|
||||||
pptoken_skip_spaces(&in)
|
pptoken_skip_spaces(&in)
|
||||||
|
@ -917,7 +923,8 @@ function translation_phase_4
|
||||||
if_expr = if_tokens + 2500
|
if_expr = if_tokens + 2500
|
||||||
|
|
||||||
p = if_pptokens
|
p = if_pptokens
|
||||||
macro_replacement_to_terminator(filename, line_number, &in, &p, 10)
|
macro_replacement_to_terminator(filename, &line_number, &in, &p, 10)
|
||||||
|
|
||||||
if_tokens_end = tokenize(if_pptokens, if_tokens, filename, line_number)
|
if_tokens_end = tokenize(if_pptokens, if_tokens, filename, line_number)
|
||||||
; replace all identifiers with 0
|
; replace all identifiers with 0
|
||||||
p = if_tokens
|
p = if_tokens
|
||||||
|
@ -927,58 +934,11 @@ function translation_phase_4
|
||||||
p += 16
|
p += 16
|
||||||
goto pp_if_idents0_loop
|
goto pp_if_idents0_loop
|
||||||
:pp_if_replace_ident
|
:pp_if_replace_ident
|
||||||
p += 8
|
|
||||||
b = str_equals(*8p, .str_defined)
|
|
||||||
p -= 8
|
|
||||||
if b != 0 goto pp_replace_defined
|
|
||||||
*1p = TOKEN_CONSTANT_INT
|
*1p = TOKEN_CONSTANT_INT
|
||||||
p += 8
|
p += 8
|
||||||
*8p = 0
|
*8p = 0
|
||||||
p += 8
|
p += 8
|
||||||
goto pp_if_idents0_loop
|
goto pp_if_idents0_loop
|
||||||
:pp_replace_defined
|
|
||||||
; handle, e.g. #if defined(SOMETHING)
|
|
||||||
p += 16
|
|
||||||
if *1p != SYMBOL_LPAREN goto pp_defined_nolparen
|
|
||||||
p += 16
|
|
||||||
if *1p != TOKEN_IDENTIFIER goto pp_bad_defined
|
|
||||||
p += 8
|
|
||||||
def_name = *8p
|
|
||||||
p += 8
|
|
||||||
if *1p != SYMBOL_RPAREN goto pp_bad_defined
|
|
||||||
q = p + 16
|
|
||||||
p -= 32
|
|
||||||
n = if_tokens_end - q
|
|
||||||
memcpy(p, q, n) ; shift everything over because 0 is less tokens than defined(X)
|
|
||||||
p -= 16
|
|
||||||
goto pp_defined_lparen_cont
|
|
||||||
:pp_defined_nolparen
|
|
||||||
if *1p != TOKEN_IDENTIFIER goto pp_bad_defined
|
|
||||||
p += 8
|
|
||||||
def_name = *8p
|
|
||||||
p -= 8
|
|
||||||
q = p + 16
|
|
||||||
n = if_tokens_end - q
|
|
||||||
memcpy(p, q, n) ; shift everything over because 0 is less tokens than defined X
|
|
||||||
p -= 16
|
|
||||||
:pp_defined_lparen_cont
|
|
||||||
|
|
||||||
*1p = TOKEN_CONSTANT_INT
|
|
||||||
p += 8
|
|
||||||
b = look_up_object_macro(def_name)
|
|
||||||
if b != 0 goto pp_defined_1
|
|
||||||
b = look_up_function_macro(def_name)
|
|
||||||
if b != 0 goto pp_defined_1
|
|
||||||
; not defined
|
|
||||||
*8p = 0
|
|
||||||
goto pp_defined_cont
|
|
||||||
:pp_defined_1
|
|
||||||
; defined
|
|
||||||
*8p = 1
|
|
||||||
:pp_defined_cont
|
|
||||||
p += 8
|
|
||||||
goto pp_if_idents0_loop
|
|
||||||
|
|
||||||
:pp_if_idents0_done
|
:pp_if_idents0_done
|
||||||
;print_tokens(if_tokens, p)
|
;print_tokens(if_tokens, p)
|
||||||
parse_expression(if_tokens, p, if_expr)
|
parse_expression(if_tokens, p, if_expr)
|
||||||
|
@ -1176,16 +1136,59 @@ function macro_replacement_to_terminator
|
||||||
argument terminator
|
argument terminator
|
||||||
local in
|
local in
|
||||||
local out
|
local out
|
||||||
|
local b
|
||||||
|
local lparen
|
||||||
in = *8p_in
|
in = *8p_in
|
||||||
out = *8p_out
|
out = *8p_out
|
||||||
:macro_replacement_to_terminator_loop
|
:macro_replacement_to_terminator_loop
|
||||||
if *1in == terminator goto macro_replacement_to_terminator_loop_end
|
if *1in == terminator goto macro_replacement_to_terminator_loop_end
|
||||||
|
b = str_equals(in, .str_defined)
|
||||||
|
if b != 0 goto replace_defined
|
||||||
macro_replacement(filename, p_line_number, &in, &out)
|
macro_replacement(filename, p_line_number, &in, &out)
|
||||||
goto macro_replacement_to_terminator_loop
|
goto macro_replacement_to_terminator_loop
|
||||||
|
:replace_defined
|
||||||
|
; @NONSTANDARD: technically this should only happen in #ifs but whatever
|
||||||
|
pptoken_skip(&in)
|
||||||
|
pptoken_skip_spaces(&in)
|
||||||
|
lparen = 0
|
||||||
|
if *1in != 40 goto defined_no_lparen
|
||||||
|
lparen = 1
|
||||||
|
pptoken_skip(&in)
|
||||||
|
pptoken_skip_spaces(&in)
|
||||||
|
:defined_no_lparen
|
||||||
|
b = isalnum_or_underscore(*1in)
|
||||||
|
if b == 0 goto bad_defined
|
||||||
|
b = look_up_object_macro(in)
|
||||||
|
if b != 0 goto defined_1
|
||||||
|
b = look_up_function_macro(in)
|
||||||
|
if b != 0 goto defined_1
|
||||||
|
; not defined
|
||||||
|
*1out = '0
|
||||||
|
out += 1
|
||||||
|
*1out = 0
|
||||||
|
out += 1
|
||||||
|
goto defined_cont
|
||||||
|
:defined_1
|
||||||
|
*1out = '1
|
||||||
|
out += 1
|
||||||
|
*1out = 0
|
||||||
|
out += 1
|
||||||
|
:defined_cont
|
||||||
|
pptoken_skip(&in)
|
||||||
|
if lparen == 0 goto macro_replacement_to_terminator_loop
|
||||||
|
pptoken_skip_spaces(&in)
|
||||||
|
if *1in != 41 goto bad_defined
|
||||||
|
pptoken_skip(&in)
|
||||||
|
goto macro_replacement_to_terminator_loop
|
||||||
:macro_replacement_to_terminator_loop_end
|
:macro_replacement_to_terminator_loop_end
|
||||||
*8p_in = in
|
*8p_in = in
|
||||||
*8p_out = out
|
*8p_out = out
|
||||||
return
|
return
|
||||||
|
:bad_defined
|
||||||
|
compile_error(filename, *8p_line_number, .str_bad_defined)
|
||||||
|
:str_bad_defined
|
||||||
|
string Bad use of defined().
|
||||||
|
byte 0
|
||||||
|
|
||||||
; @NONSTANDARD:
|
; @NONSTANDARD:
|
||||||
; Macro replacement isn't handled properly in the following ways:
|
; Macro replacement isn't handled properly in the following ways:
|
||||||
|
|
23
05/setjmp.h
23
05/setjmp.h
|
@ -1,2 +1,21 @@
|
||||||
// @NONSTANDARD
|
#ifndef _SETJMP_H
|
||||||
#error "longjmp is not supported."
|
#define _SETJMP_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
typedef long jmp_buf[3];
|
||||||
|
|
||||||
|
// @NONSTANDARD: we don't actually support setjmp
|
||||||
|
|
||||||
|
int setjmp(jmp_buf env) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void __longjmp(jmp_buf env, int val, const char *filename, int line) {
|
||||||
|
fprintf(stderr, "Error: Tried to longjmp from %s:%d with value %d\n", filename, line, val);
|
||||||
|
_Exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define longjmp(env, val) __longjmp(env, val, __FILE__, __LINE__)
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
179
05/signal.h
179
05/signal.h
|
@ -13,10 +13,11 @@ typedef long sig_atomic_t; // there are no "asynchronous interrupts"
|
||||||
typedef void (*_Sighandler)(int);
|
typedef void (*_Sighandler)(int);
|
||||||
|
|
||||||
struct sigaction {
|
struct sigaction {
|
||||||
void (*handler)(int);
|
void (*sa_handler)(int);
|
||||||
unsigned long flags;
|
#define sa_sigaction sa_handler
|
||||||
void (*restorer)(void);
|
unsigned long sa_flags;
|
||||||
unsigned long mask;
|
void (*sa_restorer)(void);
|
||||||
|
unsigned long sa_mask;
|
||||||
};
|
};
|
||||||
|
|
||||||
unsigned char _signal_restorer[] = {
|
unsigned char _signal_restorer[] = {
|
||||||
|
@ -49,11 +50,16 @@ unsigned char _signal_handler[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
#define _SA_RESTORER 0x04000000
|
#define _SA_RESTORER 0x04000000
|
||||||
|
#define SA_SIGINFO 4
|
||||||
|
#define SA_RESETHAND 0x80000000
|
||||||
|
|
||||||
int __sigaction(int signum, const struct sigaction *act, struct sigaction *oldact) {
|
int __sigaction(int signum, const struct sigaction *act, struct sigaction *oldact) {
|
||||||
return __syscall(13, signum, act, oldact, 8, 0, 0);
|
return __syscall(13, signum, act, oldact, 8, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void sigemptyset(unsigned long *set) {
|
||||||
|
*set = 0;
|
||||||
|
}
|
||||||
|
|
||||||
void _sig_ign(int signal) {
|
void _sig_ign(int signal) {
|
||||||
return;
|
return;
|
||||||
|
@ -75,10 +81,10 @@ _Sighandler signal(int sig, _Sighandler func) {
|
||||||
_sig_mask |= 1ul << (sig-1);
|
_sig_mask |= 1ul << (sig-1);
|
||||||
}
|
}
|
||||||
struct sigaction act = {0};
|
struct sigaction act = {0};
|
||||||
act.handler = func == SIG_DFL ? SIG_DFL : (void*)_signal_handler;
|
act.sa_handler = func == SIG_DFL ? SIG_DFL : (void*)_signal_handler;
|
||||||
act.mask = _sig_mask;
|
act.sa_mask = _sig_mask;
|
||||||
act.flags = _SA_RESTORER;
|
act.sa_flags = _SA_RESTORER;
|
||||||
act.restorer = _signal_restorer;
|
act.sa_restorer = _signal_restorer;
|
||||||
__sigaction(sig, &act, NULL);
|
__sigaction(sig, &act, NULL);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -87,4 +93,161 @@ int raise(int signal) {
|
||||||
return kill(getpid(), signal);
|
return kill(getpid(), signal);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define FPE_INTDIV 1
|
||||||
|
#define FPE_FLTDIV 3
|
||||||
|
|
||||||
|
#define __SI_MAX_SIZE 128
|
||||||
|
#if __WORDSIZE == 64
|
||||||
|
# define __SI_PAD_SIZE ((__SI_MAX_SIZE / sizeof (int)) - 4)
|
||||||
|
#else
|
||||||
|
# define __SI_PAD_SIZE ((__SI_MAX_SIZE / sizeof (int)) - 3)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef __SI_ALIGNMENT
|
||||||
|
# define __SI_ALIGNMENT /* nothing */
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_BAND_TYPE
|
||||||
|
# define __SI_BAND_TYPE long int
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_CLOCK_T
|
||||||
|
# define __SI_CLOCK_T __clock_t
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_ERRNO_THEN_CODE
|
||||||
|
# define __SI_ERRNO_THEN_CODE 1
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_HAVE_SIGSYS
|
||||||
|
# define __SI_HAVE_SIGSYS 1
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_SIGFAULT_ADDL
|
||||||
|
# define __SI_SIGFAULT_ADDL /* nothing */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef int __pid_t;
|
||||||
|
typedef unsigned __uid_t;
|
||||||
|
|
||||||
|
union __sigval
|
||||||
|
{
|
||||||
|
int __sival_int;
|
||||||
|
void *__sival_ptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef union __sigval __sigval_t;
|
||||||
|
typedef long __clock_t;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
int si_signo; /* Signal number. */
|
||||||
|
#if __SI_ERRNO_THEN_CODE
|
||||||
|
int si_errno; /* If non-zero, an errno value associated with
|
||||||
|
this signal, as defined in <errno.h>. */
|
||||||
|
int si_code; /* Signal code. */
|
||||||
|
#else
|
||||||
|
int si_code;
|
||||||
|
int si_errno;
|
||||||
|
#endif
|
||||||
|
#if __WORDSIZE == 64
|
||||||
|
int __pad0; /* Explicit padding. */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
union
|
||||||
|
{
|
||||||
|
int _pad[__SI_PAD_SIZE];
|
||||||
|
|
||||||
|
/* kill(). */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Sending process ID. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
} _kill;
|
||||||
|
|
||||||
|
/* POSIX.1b timers. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
int si_tid; /* Timer ID. */
|
||||||
|
int si_overrun; /* Overrun count. */
|
||||||
|
__sigval_t si_sigval; /* Signal value. */
|
||||||
|
} _timer;
|
||||||
|
|
||||||
|
/* POSIX.1b signals. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Sending process ID. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
__sigval_t si_sigval; /* Signal value. */
|
||||||
|
} _rt;
|
||||||
|
|
||||||
|
/* SIGCHLD. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Which child. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
int si_status; /* Exit value or signal. */
|
||||||
|
__SI_CLOCK_T si_utime;
|
||||||
|
__SI_CLOCK_T si_stime;
|
||||||
|
} _sigchld;
|
||||||
|
|
||||||
|
/* SIGILL, SIGFPE, SIGSEGV, SIGBUS. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *si_addr; /* Faulting insn/memory ref. */
|
||||||
|
__SI_SIGFAULT_ADDL
|
||||||
|
short int si_addr_lsb; /* Valid LSB of the reported address. */
|
||||||
|
union
|
||||||
|
{
|
||||||
|
/* used when si_code=SEGV_BNDERR */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *_lower;
|
||||||
|
void *_upper;
|
||||||
|
} _addr_bnd;
|
||||||
|
/* used when si_code=SEGV_PKUERR */
|
||||||
|
uint32_t _pkey;
|
||||||
|
} _bounds;
|
||||||
|
} _sigfault;
|
||||||
|
|
||||||
|
/* SIGPOLL. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__SI_BAND_TYPE si_band; /* Band event for SIGPOLL. */
|
||||||
|
int si_fd;
|
||||||
|
} _sigpoll;
|
||||||
|
|
||||||
|
/* SIGSYS. */
|
||||||
|
#if __SI_HAVE_SIGSYS
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *_call_addr; /* Calling user insn. */
|
||||||
|
int _syscall; /* Triggering system call number. */
|
||||||
|
unsigned int _arch; /* AUDIT_ARCH_* of syscall. */
|
||||||
|
} _sigsys;
|
||||||
|
#endif
|
||||||
|
} _sifields;
|
||||||
|
} siginfo_t __SI_ALIGNMENT;
|
||||||
|
|
||||||
|
|
||||||
|
/* X/Open requires some more fields with fixed names. */
|
||||||
|
#define si_pid _sifields._kill.si_pid
|
||||||
|
#define si_uid _sifields._kill.si_uid
|
||||||
|
#define si_timerid _sifields._timer.si_tid
|
||||||
|
#define si_overrun _sifields._timer.si_overrun
|
||||||
|
#define si_status _sifields._sigchld.si_status
|
||||||
|
#define si_utime _sifields._sigchld.si_utime
|
||||||
|
#define si_stime _sifields._sigchld.si_stime
|
||||||
|
#define si_value _sifields._rt.si_sigval
|
||||||
|
#define si_int _sifields._rt.si_sigval.sival_int
|
||||||
|
#define si_ptr _sifields._rt.si_sigval.sival_ptr
|
||||||
|
#define si_addr _sifields._sigfault.si_addr
|
||||||
|
#define si_addr_lsb _sifields._sigfault.si_addr_lsb
|
||||||
|
#define si_lower _sifields._sigfault._bounds._addr_bnd._lower
|
||||||
|
#define si_upper _sifields._sigfault._bounds._addr_bnd._upper
|
||||||
|
#define si_pkey _sifields._sigfault._bounds._pkey
|
||||||
|
#define si_band _sifields._sigpoll.si_band
|
||||||
|
#define si_fd _sifields._sigpoll.si_fd
|
||||||
|
#if __SI_HAVE_SIGSYS
|
||||||
|
# define si_call_addr _sifields._sigsys._call_addr
|
||||||
|
# define si_syscall _sifields._sigsys._syscall
|
||||||
|
# define si_arch _sifields._sigsys._arch
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
#endif // _SIGNAL_H
|
#endif // _SIGNAL_H
|
||||||
|
|
173
05/stdc_common.h
173
05/stdc_common.h
|
@ -88,6 +88,111 @@ static unsigned char __syscall_data[] = {
|
||||||
(((unsigned long (*)(unsigned long, unsigned long, unsigned long, unsigned long, unsigned long, unsigned long, unsigned long))__syscall_data)\
|
(((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))
|
(no, arg1, arg2, arg3, arg4, arg5, arg6))
|
||||||
|
|
||||||
|
// we need to define ucontext_t
|
||||||
|
# define __ctx(fld) fld
|
||||||
|
typedef long long int greg_t;
|
||||||
|
#define __NGREG 23
|
||||||
|
typedef greg_t gregset_t[__NGREG];
|
||||||
|
#define _SIGSET_NWORDS (1024 / (8 * sizeof (unsigned long int)))
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
unsigned long int __val[_SIGSET_NWORDS];
|
||||||
|
} __sigset_t, sigset_t;
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
void *ss_sp;
|
||||||
|
int ss_flags;
|
||||||
|
size_t ss_size;
|
||||||
|
} stack_t;
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
REG_R8 = 0,
|
||||||
|
# define REG_R8 REG_R8
|
||||||
|
REG_R9,
|
||||||
|
# define REG_R9 REG_R9
|
||||||
|
REG_R10,
|
||||||
|
# define REG_R10 REG_R10
|
||||||
|
REG_R11,
|
||||||
|
# define REG_R11 REG_R11
|
||||||
|
REG_R12,
|
||||||
|
# define REG_R12 REG_R12
|
||||||
|
REG_R13,
|
||||||
|
# define REG_R13 REG_R13
|
||||||
|
REG_R14,
|
||||||
|
# define REG_R14 REG_R14
|
||||||
|
REG_R15,
|
||||||
|
# define REG_R15 REG_R15
|
||||||
|
REG_RDI,
|
||||||
|
# define REG_RDI REG_RDI
|
||||||
|
REG_RSI,
|
||||||
|
# define REG_RSI REG_RSI
|
||||||
|
REG_RBP,
|
||||||
|
# define REG_RBP REG_RBP
|
||||||
|
REG_RBX,
|
||||||
|
# define REG_RBX REG_RBX
|
||||||
|
REG_RDX,
|
||||||
|
# define REG_RDX REG_RDX
|
||||||
|
REG_RAX,
|
||||||
|
# define REG_RAX REG_RAX
|
||||||
|
REG_RCX,
|
||||||
|
# define REG_RCX REG_RCX
|
||||||
|
REG_RSP,
|
||||||
|
# define REG_RSP REG_RSP
|
||||||
|
REG_RIP,
|
||||||
|
# define REG_RIP REG_RIP
|
||||||
|
REG_EFL,
|
||||||
|
# define REG_EFL REG_EFL
|
||||||
|
REG_CSGSFS, /* Actually short cs, gs, fs, __pad0. */
|
||||||
|
# define REG_CSGSFS REG_CSGSFS
|
||||||
|
REG_ERR,
|
||||||
|
# define REG_ERR REG_ERR
|
||||||
|
REG_TRAPNO,
|
||||||
|
# define REG_TRAPNO REG_TRAPNO
|
||||||
|
REG_OLDMASK,
|
||||||
|
# define REG_OLDMASK REG_OLDMASK
|
||||||
|
REG_CR2
|
||||||
|
# define REG_CR2 REG_CR2
|
||||||
|
};
|
||||||
|
struct _libc_fpxreg
|
||||||
|
{
|
||||||
|
unsigned short int __ctx(significand)[4];
|
||||||
|
unsigned short int __ctx(exponent);
|
||||||
|
unsigned short int __glibc_reserved1[3];
|
||||||
|
};
|
||||||
|
struct _libc_xmmreg
|
||||||
|
{
|
||||||
|
uint32_t __ctx(element)[4];
|
||||||
|
};
|
||||||
|
struct _libc_fpstate
|
||||||
|
{
|
||||||
|
uint16_t __ctx(cwd);
|
||||||
|
uint16_t __ctx(swd);
|
||||||
|
uint16_t __ctx(ftw);
|
||||||
|
uint16_t __ctx(fop);
|
||||||
|
uint64_t __ctx(rip);
|
||||||
|
uint64_t __ctx(rdp);
|
||||||
|
uint32_t __ctx(mxcsr);
|
||||||
|
uint32_t __ctx(mxcr_mask);
|
||||||
|
struct _libc_fpxreg _st[8];
|
||||||
|
struct _libc_xmmreg _xmm[16];
|
||||||
|
uint32_t __glibc_reserved1[24];
|
||||||
|
};
|
||||||
|
typedef struct _libc_fpstate *fpregset_t;
|
||||||
|
typedef struct {
|
||||||
|
gregset_t __ctx(gregs);
|
||||||
|
fpregset_t __ctx(fpregs);
|
||||||
|
unsigned long long __reserved1 [8];
|
||||||
|
} mcontext_t;
|
||||||
|
typedef struct ucontext_t {
|
||||||
|
unsigned long int __ctx(uc_flags);
|
||||||
|
struct ucontext_t *uc_link;
|
||||||
|
stack_t uc_stack;
|
||||||
|
mcontext_t uc_mcontext;
|
||||||
|
sigset_t uc_sigmask;
|
||||||
|
struct _libc_fpstate __fpregs_mem;
|
||||||
|
unsigned long long int __ssp[4];
|
||||||
|
} ucontext_t;
|
||||||
|
|
||||||
long read(int fd, void *buf, size_t count) {
|
long read(int fd, void *buf, size_t count) {
|
||||||
return __syscall(0, fd, buf, count, 0, 0, 0);
|
return __syscall(0, fd, buf, count, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
@ -116,6 +221,32 @@ int execve(const char *pathname, char *const argv[], char *const envp[]) {
|
||||||
return __syscall(59, pathname, argv, envp, 0, 0, 0);
|
return __syscall(59, pathname, argv, envp, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int gettimeofday(struct timeval *tv, struct timezone *tz) {
|
||||||
|
return __syscall(96, tv, tz, 0, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef long time_t;
|
||||||
|
|
||||||
|
struct timespec {
|
||||||
|
time_t tv_sec;
|
||||||
|
long tv_nsec;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct timeval {
|
||||||
|
time_t tv_sec;
|
||||||
|
long tv_usec;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct timezone {
|
||||||
|
int tz_minuteswest;
|
||||||
|
int tz_dsttime;
|
||||||
|
};
|
||||||
|
|
||||||
|
char *getcwd(char *buf, size_t size) {
|
||||||
|
long n = __syscall(79, buf, size, 0, 0, 0, 0);
|
||||||
|
if (n < 0) return NULL;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
#define _WEXITSTATUS(status) (((status) & 0xff00) >> 8)
|
#define _WEXITSTATUS(status) (((status) & 0xff00) >> 8)
|
||||||
#define _WIFEXITED(status) (__WTERMSIG(status) == 0)
|
#define _WIFEXITED(status) (__WTERMSIG(status) == 0)
|
||||||
|
@ -132,16 +263,11 @@ int wait4(int pid, int *status, int options, struct rusage *rusage) {
|
||||||
#define SIGINT 2
|
#define SIGINT 2
|
||||||
#define SIGSEGV 11
|
#define SIGSEGV 11
|
||||||
#define SIGTERM 15
|
#define SIGTERM 15
|
||||||
|
#define SIGBUS 7
|
||||||
void abort(void) {
|
void abort(void) {
|
||||||
kill(getpid(), SIGABRT);
|
kill(getpid(), SIGABRT);
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef long time_t;
|
|
||||||
|
|
||||||
struct timespec {
|
|
||||||
time_t tv_sec;
|
|
||||||
long tv_nsec;
|
|
||||||
};
|
|
||||||
|
|
||||||
#define CLOCK_REALTIME 0
|
#define CLOCK_REALTIME 0
|
||||||
#define CLOCK_MONOTONIC 1
|
#define CLOCK_MONOTONIC 1
|
||||||
|
@ -183,6 +309,29 @@ int __assert_failed(const char *file, int line, const char *expr) {
|
||||||
#define assert(x) (void)((x) || __assert_failed(__FILE__, __LINE__, #x))
|
#define assert(x) (void)((x) || __assert_failed(__FILE__, __LINE__, #x))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
int _clamp_long_to_int(long x) {
|
||||||
|
if (x < INT_MIN) return INT_MIN;
|
||||||
|
if (x > INT_MAX) return INT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
short _clamp_long_to_short(long x) {
|
||||||
|
if (x < SHRT_MIN) return SHRT_MIN;
|
||||||
|
if (x > SHRT_MAX) return SHRT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned _clamp_ulong_to_uint(unsigned long x) {
|
||||||
|
if (x > UINT_MAX) return UINT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned short _clamp_ulong_to_ushort(unsigned long x) {
|
||||||
|
if (x > USHRT_MAX) return USHRT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
#define EIO 5
|
#define EIO 5
|
||||||
#define EDOM 33
|
#define EDOM 33
|
||||||
#define ERANGE 34
|
#define ERANGE 34
|
||||||
|
@ -201,6 +350,10 @@ int munmap(void *addr, size_t length) {
|
||||||
return __syscall(11, addr, length, 0, 0, 0, 0);
|
return __syscall(11, addr, length, 0, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int mprotect(void *addr, size_t len, int prot) {
|
||||||
|
return __syscall(10, addr, len, prot, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
#define MREMAP_MAYMOVE 1
|
#define MREMAP_MAYMOVE 1
|
||||||
void *_mremap(void *addr, size_t old_size, size_t new_size, int flags) {
|
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);
|
return __syscall(25, addr, old_size, new_size, flags, 0, 0);
|
||||||
|
@ -455,6 +608,14 @@ double strtod(const char *nptr, char **endptr) {
|
||||||
return sum * sign;
|
return sum * sign;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
float strtof(const char *nptr, char **endptr) {
|
||||||
|
return strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
long double strtold(const char *nptr, char **endptr) {
|
||||||
|
return strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
|
||||||
char *strerror(int errnum) {
|
char *strerror(int errnum) {
|
||||||
switch (errnum) {
|
switch (errnum) {
|
||||||
case ERANGE: return "Range error";
|
case ERANGE: return "Range error";
|
||||||
|
|
28
05/stdio.h
28
05/stdio.h
|
@ -1699,6 +1699,12 @@ FILE *fopen(const char *filename, const char *mode) {
|
||||||
return _FILE_from_fd(fd);
|
return _FILE_from_fd(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FILE *fdopen(int fd, const char *mode) {
|
||||||
|
// mode doesn't matter, hopefully
|
||||||
|
return _FILE_from_fd(fd);
|
||||||
|
}
|
||||||
|
|
||||||
int fclose(FILE *stream) {
|
int fclose(FILE *stream) {
|
||||||
int ret = close(stream->fd);
|
int ret = close(stream->fd);
|
||||||
free(stream);
|
free(stream);
|
||||||
|
@ -1932,28 +1938,6 @@ int _file_peek_char(void *dat) {
|
||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _clamp_long_to_int(long x) {
|
|
||||||
if (x < INT_MIN) return INT_MIN;
|
|
||||||
if (x > INT_MAX) return INT_MAX;
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
short _clamp_long_to_short(long x) {
|
|
||||||
if (x < SHRT_MIN) return SHRT_MIN;
|
|
||||||
if (x > SHRT_MAX) return SHRT_MAX;
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned _clamp_ulong_to_uint(unsigned long x) {
|
|
||||||
if (x > UINT_MAX) return UINT_MAX;
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned short _clamp_ulong_to_ushort(unsigned long x) {
|
|
||||||
if (x > USHRT_MAX) return USHRT_MAX;
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
void _bad_scanf(void) {
|
void _bad_scanf(void) {
|
||||||
fprintf(stderr, "bad scanf format.\n");
|
fprintf(stderr, "bad scanf format.\n");
|
||||||
abort();
|
abort();
|
||||||
|
|
37
05/tcc-0.9.25/.cvsignore
Normal file
37
05/tcc-0.9.25/.cvsignore
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
tcc_g
|
||||||
|
tcc
|
||||||
|
tc2.c
|
||||||
|
doc
|
||||||
|
tc3s.c
|
||||||
|
p3.c
|
||||||
|
tc1.c
|
||||||
|
error.c
|
||||||
|
i386-gen1.c
|
||||||
|
test.out2
|
||||||
|
test.out3
|
||||||
|
web.sh
|
||||||
|
memdebug.c
|
||||||
|
bench
|
||||||
|
Makefile.uClibc
|
||||||
|
boundtest
|
||||||
|
prog.ref
|
||||||
|
test.ref
|
||||||
|
test.out
|
||||||
|
tcc-doc.html
|
||||||
|
ideas
|
||||||
|
tcctest.ref
|
||||||
|
linux.tcc
|
||||||
|
ldtest
|
||||||
|
libtcc_test
|
||||||
|
instr.S
|
||||||
|
p.c
|
||||||
|
p2.c
|
||||||
|
tcctest[1234]
|
||||||
|
test[1234].out
|
||||||
|
.gdb_history
|
||||||
|
tcc.1
|
||||||
|
tcc.pod
|
||||||
|
config.h
|
||||||
|
config.mak
|
||||||
|
config.texi
|
||||||
|
tests
|
4
05/tcc-0.9.25/.gitignore
vendored
Normal file
4
05/tcc-0.9.25/.gitignore
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
tcc
|
||||||
|
examples
|
||||||
|
tests
|
||||||
|
win32
|
504
05/tcc-0.9.25/COPYING
Normal file
504
05/tcc-0.9.25/COPYING
Normal file
|
@ -0,0 +1,504 @@
|
||||||
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
|
Version 2.1, February 1999
|
||||||
|
|
||||||
|
Copyright (C) 1991, 1999 Free Software Foundation, Inc.
|
||||||
|
59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
|
of this license document, but changing it is not allowed.
|
||||||
|
|
||||||
|
[This is the first released version of the Lesser GPL. It also counts
|
||||||
|
as the successor of the GNU Library Public License, version 2, hence
|
||||||
|
the version number 2.1.]
|
||||||
|
|
||||||
|
Preamble
|
||||||
|
|
||||||
|
The licenses for most software are designed to take away your
|
||||||
|
freedom to share and change it. By contrast, the GNU General Public
|
||||||
|
Licenses are intended to guarantee your freedom to share and change
|
||||||
|
free software--to make sure the software is free for all its users.
|
||||||
|
|
||||||
|
This license, the Lesser General Public License, applies to some
|
||||||
|
specially designated software packages--typically libraries--of the
|
||||||
|
Free Software Foundation and other authors who decide to use it. You
|
||||||
|
can use it too, but we suggest you first think carefully about whether
|
||||||
|
this license or the ordinary General Public License is the better
|
||||||
|
strategy to use in any particular case, based on the explanations below.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom of use,
|
||||||
|
not price. Our General Public Licenses are designed to make sure that
|
||||||
|
you have the freedom to distribute copies of free software (and charge
|
||||||
|
for this service if you wish); that you receive source code or can get
|
||||||
|
it if you want it; that you can change the software and use pieces of
|
||||||
|
it in new free programs; and that you are informed that you can do
|
||||||
|
these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to make restrictions that forbid
|
||||||
|
distributors to deny you these rights or to ask you to surrender these
|
||||||
|
rights. These restrictions translate to certain responsibilities for
|
||||||
|
you if you distribute copies of the library or if you modify it.
|
||||||
|
|
||||||
|
For example, if you distribute copies of the library, whether gratis
|
||||||
|
or for a fee, you must give the recipients all the rights that we gave
|
||||||
|
you. You must make sure that they, too, receive or can get the source
|
||||||
|
code. If you link other code with the library, you must provide
|
||||||
|
complete object files to the recipients, so that they can relink them
|
||||||
|
with the library after making changes to the library and recompiling
|
||||||
|
it. And you must show them these terms so they know their rights.
|
||||||
|
|
||||||
|
We protect your rights with a two-step method: (1) we copyright the
|
||||||
|
library, and (2) we offer you this license, which gives you legal
|
||||||
|
permission to copy, distribute and/or modify the library.
|
||||||
|
|
||||||
|
To protect each distributor, we want to make it very clear that
|
||||||
|
there is no warranty for the free library. Also, if the library is
|
||||||
|
modified by someone else and passed on, the recipients should know
|
||||||
|
that what they have is not the original version, so that the original
|
||||||
|
author's reputation will not be affected by problems that might be
|
||||||
|
introduced by others.
|
||||||
|
|
||||||
|
Finally, software patents pose a constant threat to the existence of
|
||||||
|
any free program. We wish to make sure that a company cannot
|
||||||
|
effectively restrict the users of a free program by obtaining a
|
||||||
|
restrictive license from a patent holder. Therefore, we insist that
|
||||||
|
any patent license obtained for a version of the library must be
|
||||||
|
consistent with the full freedom of use specified in this license.
|
||||||
|
|
||||||
|
Most GNU software, including some libraries, is covered by the
|
||||||
|
ordinary GNU General Public License. This license, the GNU Lesser
|
||||||
|
General Public License, applies to certain designated libraries, and
|
||||||
|
is quite different from the ordinary General Public License. We use
|
||||||
|
this license for certain libraries in order to permit linking those
|
||||||
|
libraries into non-free programs.
|
||||||
|
|
||||||
|
When a program is linked with a library, whether statically or using
|
||||||
|
a shared library, the combination of the two is legally speaking a
|
||||||
|
combined work, a derivative of the original library. The ordinary
|
||||||
|
General Public License therefore permits such linking only if the
|
||||||
|
entire combination fits its criteria of freedom. The Lesser General
|
||||||
|
Public License permits more lax criteria for linking other code with
|
||||||
|
the library.
|
||||||
|
|
||||||
|
We call this license the "Lesser" General Public License because it
|
||||||
|
does Less to protect the user's freedom than the ordinary General
|
||||||
|
Public License. It also provides other free software developers Less
|
||||||
|
of an advantage over competing non-free programs. These disadvantages
|
||||||
|
are the reason we use the ordinary General Public License for many
|
||||||
|
libraries. However, the Lesser license provides advantages in certain
|
||||||
|
special circumstances.
|
||||||
|
|
||||||
|
For example, on rare occasions, there may be a special need to
|
||||||
|
encourage the widest possible use of a certain library, so that it becomes
|
||||||
|
a de-facto standard. To achieve this, non-free programs must be
|
||||||
|
allowed to use the library. A more frequent case is that a free
|
||||||
|
library does the same job as widely used non-free libraries. In this
|
||||||
|
case, there is little to gain by limiting the free library to free
|
||||||
|
software only, so we use the Lesser General Public License.
|
||||||
|
|
||||||
|
In other cases, permission to use a particular library in non-free
|
||||||
|
programs enables a greater number of people to use a large body of
|
||||||
|
free software. For example, permission to use the GNU C Library in
|
||||||
|
non-free programs enables many more people to use the whole GNU
|
||||||
|
operating system, as well as its variant, the GNU/Linux operating
|
||||||
|
system.
|
||||||
|
|
||||||
|
Although the Lesser General Public License is Less protective of the
|
||||||
|
users' freedom, it does ensure that the user of a program that is
|
||||||
|
linked with the Library has the freedom and the wherewithal to run
|
||||||
|
that program using a modified version of the Library.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow. Pay close attention to the difference between a
|
||||||
|
"work based on the library" and a "work that uses the library". The
|
||||||
|
former contains code derived from the library, whereas the latter must
|
||||||
|
be combined with the library in order to run.
|
||||||
|
|
||||||
|
GNU LESSER GENERAL PUBLIC LICENSE
|
||||||
|
TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
|
||||||
|
|
||||||
|
0. This License Agreement applies to any software library or other
|
||||||
|
program which contains a notice placed by the copyright holder or
|
||||||
|
other authorized party saying it may be distributed under the terms of
|
||||||
|
this Lesser General Public License (also called "this License").
|
||||||
|
Each licensee is addressed as "you".
|
||||||
|
|
||||||
|
A "library" means a collection of software functions and/or data
|
||||||
|
prepared so as to be conveniently linked with application programs
|
||||||
|
(which use some of those functions and data) to form executables.
|
||||||
|
|
||||||
|
The "Library", below, refers to any such software library or work
|
||||||
|
which has been distributed under these terms. A "work based on the
|
||||||
|
Library" means either the Library or any derivative work under
|
||||||
|
copyright law: that is to say, a work containing the Library or a
|
||||||
|
portion of it, either verbatim or with modifications and/or translated
|
||||||
|
straightforwardly into another language. (Hereinafter, translation is
|
||||||
|
included without limitation in the term "modification".)
|
||||||
|
|
||||||
|
"Source code" for a work means the preferred form of the work for
|
||||||
|
making modifications to it. For a library, complete source code means
|
||||||
|
all the source code for all modules it contains, plus any associated
|
||||||
|
interface definition files, plus the scripts used to control compilation
|
||||||
|
and installation of the library.
|
||||||
|
|
||||||
|
Activities other than copying, distribution and modification are not
|
||||||
|
covered by this License; they are outside its scope. The act of
|
||||||
|
running a program using the Library is not restricted, and output from
|
||||||
|
such a program is covered only if its contents constitute a work based
|
||||||
|
on the Library (independent of the use of the Library in a tool for
|
||||||
|
writing it). Whether that is true depends on what the Library does
|
||||||
|
and what the program that uses the Library does.
|
||||||
|
|
||||||
|
1. You may copy and distribute verbatim copies of the Library's
|
||||||
|
complete source code as you receive it, in any medium, provided that
|
||||||
|
you conspicuously and appropriately publish on each copy an
|
||||||
|
appropriate copyright notice and disclaimer of warranty; keep intact
|
||||||
|
all the notices that refer to this License and to the absence of any
|
||||||
|
warranty; and distribute a copy of this License along with the
|
||||||
|
Library.
|
||||||
|
|
||||||
|
You may charge a fee for the physical act of transferring a copy,
|
||||||
|
and you may at your option offer warranty protection in exchange for a
|
||||||
|
fee.
|
||||||
|
|
||||||
|
2. You may modify your copy or copies of the Library or any portion
|
||||||
|
of it, thus forming a work based on the Library, and copy and
|
||||||
|
distribute such modifications or work under the terms of Section 1
|
||||||
|
above, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The modified work must itself be a software library.
|
||||||
|
|
||||||
|
b) You must cause the files modified to carry prominent notices
|
||||||
|
stating that you changed the files and the date of any change.
|
||||||
|
|
||||||
|
c) You must cause the whole of the work to be licensed at no
|
||||||
|
charge to all third parties under the terms of this License.
|
||||||
|
|
||||||
|
d) If a facility in the modified Library refers to a function or a
|
||||||
|
table of data to be supplied by an application program that uses
|
||||||
|
the facility, other than as an argument passed when the facility
|
||||||
|
is invoked, then you must make a good faith effort to ensure that,
|
||||||
|
in the event an application does not supply such function or
|
||||||
|
table, the facility still operates, and performs whatever part of
|
||||||
|
its purpose remains meaningful.
|
||||||
|
|
||||||
|
(For example, a function in a library to compute square roots has
|
||||||
|
a purpose that is entirely well-defined independent of the
|
||||||
|
application. Therefore, Subsection 2d requires that any
|
||||||
|
application-supplied function or table used by this function must
|
||||||
|
be optional: if the application does not supply it, the square
|
||||||
|
root function must still compute square roots.)
|
||||||
|
|
||||||
|
These requirements apply to the modified work as a whole. If
|
||||||
|
identifiable sections of that work are not derived from the Library,
|
||||||
|
and can be reasonably considered independent and separate works in
|
||||||
|
themselves, then this License, and its terms, do not apply to those
|
||||||
|
sections when you distribute them as separate works. But when you
|
||||||
|
distribute the same sections as part of a whole which is a work based
|
||||||
|
on the Library, the distribution of the whole must be on the terms of
|
||||||
|
this License, whose permissions for other licensees extend to the
|
||||||
|
entire whole, and thus to each and every part regardless of who wrote
|
||||||
|
it.
|
||||||
|
|
||||||
|
Thus, it is not the intent of this section to claim rights or contest
|
||||||
|
your rights to work written entirely by you; rather, the intent is to
|
||||||
|
exercise the right to control the distribution of derivative or
|
||||||
|
collective works based on the Library.
|
||||||
|
|
||||||
|
In addition, mere aggregation of another work not based on the Library
|
||||||
|
with the Library (or with a work based on the Library) on a volume of
|
||||||
|
a storage or distribution medium does not bring the other work under
|
||||||
|
the scope of this License.
|
||||||
|
|
||||||
|
3. You may opt to apply the terms of the ordinary GNU General Public
|
||||||
|
License instead of this License to a given copy of the Library. To do
|
||||||
|
this, you must alter all the notices that refer to this License, so
|
||||||
|
that they refer to the ordinary GNU General Public License, version 2,
|
||||||
|
instead of to this License. (If a newer version than version 2 of the
|
||||||
|
ordinary GNU General Public License has appeared, then you can specify
|
||||||
|
that version instead if you wish.) Do not make any other change in
|
||||||
|
these notices.
|
||||||
|
|
||||||
|
Once this change is made in a given copy, it is irreversible for
|
||||||
|
that copy, so the ordinary GNU General Public License applies to all
|
||||||
|
subsequent copies and derivative works made from that copy.
|
||||||
|
|
||||||
|
This option is useful when you wish to copy part of the code of
|
||||||
|
the Library into a program that is not a library.
|
||||||
|
|
||||||
|
4. You may copy and distribute the Library (or a portion or
|
||||||
|
derivative of it, under Section 2) in object code or executable form
|
||||||
|
under the terms of Sections 1 and 2 above provided that you accompany
|
||||||
|
it with the complete corresponding machine-readable source code, which
|
||||||
|
must be distributed under the terms of Sections 1 and 2 above on a
|
||||||
|
medium customarily used for software interchange.
|
||||||
|
|
||||||
|
If distribution of object code is made by offering access to copy
|
||||||
|
from a designated place, then offering equivalent access to copy the
|
||||||
|
source code from the same place satisfies the requirement to
|
||||||
|
distribute the source code, even though third parties are not
|
||||||
|
compelled to copy the source along with the object code.
|
||||||
|
|
||||||
|
5. A program that contains no derivative of any portion of the
|
||||||
|
Library, but is designed to work with the Library by being compiled or
|
||||||
|
linked with it, is called a "work that uses the Library". Such a
|
||||||
|
work, in isolation, is not a derivative work of the Library, and
|
||||||
|
therefore falls outside the scope of this License.
|
||||||
|
|
||||||
|
However, linking a "work that uses the Library" with the Library
|
||||||
|
creates an executable that is a derivative of the Library (because it
|
||||||
|
contains portions of the Library), rather than a "work that uses the
|
||||||
|
library". The executable is therefore covered by this License.
|
||||||
|
Section 6 states terms for distribution of such executables.
|
||||||
|
|
||||||
|
When a "work that uses the Library" uses material from a header file
|
||||||
|
that is part of the Library, the object code for the work may be a
|
||||||
|
derivative work of the Library even though the source code is not.
|
||||||
|
Whether this is true is especially significant if the work can be
|
||||||
|
linked without the Library, or if the work is itself a library. The
|
||||||
|
threshold for this to be true is not precisely defined by law.
|
||||||
|
|
||||||
|
If such an object file uses only numerical parameters, data
|
||||||
|
structure layouts and accessors, and small macros and small inline
|
||||||
|
functions (ten lines or less in length), then the use of the object
|
||||||
|
file is unrestricted, regardless of whether it is legally a derivative
|
||||||
|
work. (Executables containing this object code plus portions of the
|
||||||
|
Library will still fall under Section 6.)
|
||||||
|
|
||||||
|
Otherwise, if the work is a derivative of the Library, you may
|
||||||
|
distribute the object code for the work under the terms of Section 6.
|
||||||
|
Any executables containing that work also fall under Section 6,
|
||||||
|
whether or not they are linked directly with the Library itself.
|
||||||
|
|
||||||
|
6. As an exception to the Sections above, you may also combine or
|
||||||
|
link a "work that uses the Library" with the Library to produce a
|
||||||
|
work containing portions of the Library, and distribute that work
|
||||||
|
under terms of your choice, provided that the terms permit
|
||||||
|
modification of the work for the customer's own use and reverse
|
||||||
|
engineering for debugging such modifications.
|
||||||
|
|
||||||
|
You must give prominent notice with each copy of the work that the
|
||||||
|
Library is used in it and that the Library and its use are covered by
|
||||||
|
this License. You must supply a copy of this License. If the work
|
||||||
|
during execution displays copyright notices, you must include the
|
||||||
|
copyright notice for the Library among them, as well as a reference
|
||||||
|
directing the user to the copy of this License. Also, you must do one
|
||||||
|
of these things:
|
||||||
|
|
||||||
|
a) Accompany the work with the complete corresponding
|
||||||
|
machine-readable source code for the Library including whatever
|
||||||
|
changes were used in the work (which must be distributed under
|
||||||
|
Sections 1 and 2 above); and, if the work is an executable linked
|
||||||
|
with the Library, with the complete machine-readable "work that
|
||||||
|
uses the Library", as object code and/or source code, so that the
|
||||||
|
user can modify the Library and then relink to produce a modified
|
||||||
|
executable containing the modified Library. (It is understood
|
||||||
|
that the user who changes the contents of definitions files in the
|
||||||
|
Library will not necessarily be able to recompile the application
|
||||||
|
to use the modified definitions.)
|
||||||
|
|
||||||
|
b) Use a suitable shared library mechanism for linking with the
|
||||||
|
Library. A suitable mechanism is one that (1) uses at run time a
|
||||||
|
copy of the library already present on the user's computer system,
|
||||||
|
rather than copying library functions into the executable, and (2)
|
||||||
|
will operate properly with a modified version of the library, if
|
||||||
|
the user installs one, as long as the modified version is
|
||||||
|
interface-compatible with the version that the work was made with.
|
||||||
|
|
||||||
|
c) Accompany the work with a written offer, valid for at
|
||||||
|
least three years, to give the same user the materials
|
||||||
|
specified in Subsection 6a, above, for a charge no more
|
||||||
|
than the cost of performing this distribution.
|
||||||
|
|
||||||
|
d) If distribution of the work is made by offering access to copy
|
||||||
|
from a designated place, offer equivalent access to copy the above
|
||||||
|
specified materials from the same place.
|
||||||
|
|
||||||
|
e) Verify that the user has already received a copy of these
|
||||||
|
materials or that you have already sent this user a copy.
|
||||||
|
|
||||||
|
For an executable, the required form of the "work that uses the
|
||||||
|
Library" must include any data and utility programs needed for
|
||||||
|
reproducing the executable from it. However, as a special exception,
|
||||||
|
the materials to be distributed need not include anything that is
|
||||||
|
normally distributed (in either source or binary form) with the major
|
||||||
|
components (compiler, kernel, and so on) of the operating system on
|
||||||
|
which the executable runs, unless that component itself accompanies
|
||||||
|
the executable.
|
||||||
|
|
||||||
|
It may happen that this requirement contradicts the license
|
||||||
|
restrictions of other proprietary libraries that do not normally
|
||||||
|
accompany the operating system. Such a contradiction means you cannot
|
||||||
|
use both them and the Library together in an executable that you
|
||||||
|
distribute.
|
||||||
|
|
||||||
|
7. You may place library facilities that are a work based on the
|
||||||
|
Library side-by-side in a single library together with other library
|
||||||
|
facilities not covered by this License, and distribute such a combined
|
||||||
|
library, provided that the separate distribution of the work based on
|
||||||
|
the Library and of the other library facilities is otherwise
|
||||||
|
permitted, and provided that you do these two things:
|
||||||
|
|
||||||
|
a) Accompany the combined library with a copy of the same work
|
||||||
|
based on the Library, uncombined with any other library
|
||||||
|
facilities. This must be distributed under the terms of the
|
||||||
|
Sections above.
|
||||||
|
|
||||||
|
b) Give prominent notice with the combined library of the fact
|
||||||
|
that part of it is a work based on the Library, and explaining
|
||||||
|
where to find the accompanying uncombined form of the same work.
|
||||||
|
|
||||||
|
8. You may not copy, modify, sublicense, link with, or distribute
|
||||||
|
the Library except as expressly provided under this License. Any
|
||||||
|
attempt otherwise to copy, modify, sublicense, link with, or
|
||||||
|
distribute the Library is void, and will automatically terminate your
|
||||||
|
rights under this License. However, parties who have received copies,
|
||||||
|
or rights, from you under this License will not have their licenses
|
||||||
|
terminated so long as such parties remain in full compliance.
|
||||||
|
|
||||||
|
9. You are not required to accept this License, since you have not
|
||||||
|
signed it. However, nothing else grants you permission to modify or
|
||||||
|
distribute the Library or its derivative works. These actions are
|
||||||
|
prohibited by law if you do not accept this License. Therefore, by
|
||||||
|
modifying or distributing the Library (or any work based on the
|
||||||
|
Library), you indicate your acceptance of this License to do so, and
|
||||||
|
all its terms and conditions for copying, distributing or modifying
|
||||||
|
the Library or works based on it.
|
||||||
|
|
||||||
|
10. Each time you redistribute the Library (or any work based on the
|
||||||
|
Library), the recipient automatically receives a license from the
|
||||||
|
original licensor to copy, distribute, link with or modify the Library
|
||||||
|
subject to these terms and conditions. You may not impose any further
|
||||||
|
restrictions on the recipients' exercise of the rights granted herein.
|
||||||
|
You are not responsible for enforcing compliance by third parties with
|
||||||
|
this License.
|
||||||
|
|
||||||
|
11. If, as a consequence of a court judgment or allegation of patent
|
||||||
|
infringement or for any other reason (not limited to patent issues),
|
||||||
|
conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot
|
||||||
|
distribute so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you
|
||||||
|
may not distribute the Library at all. For example, if a patent
|
||||||
|
license would not permit royalty-free redistribution of the Library by
|
||||||
|
all those who receive copies directly or indirectly through you, then
|
||||||
|
the only way you could satisfy both it and this License would be to
|
||||||
|
refrain entirely from distribution of the Library.
|
||||||
|
|
||||||
|
If any portion of this section is held invalid or unenforceable under any
|
||||||
|
particular circumstance, the balance of the section is intended to apply,
|
||||||
|
and the section as a whole is intended to apply in other circumstances.
|
||||||
|
|
||||||
|
It is not the purpose of this section to induce you to infringe any
|
||||||
|
patents or other property right claims or to contest validity of any
|
||||||
|
such claims; this section has the sole purpose of protecting the
|
||||||
|
integrity of the free software distribution system which is
|
||||||
|
implemented by public license practices. Many people have made
|
||||||
|
generous contributions to the wide range of software distributed
|
||||||
|
through that system in reliance on consistent application of that
|
||||||
|
system; it is up to the author/donor to decide if he or she is willing
|
||||||
|
to distribute software through any other system and a licensee cannot
|
||||||
|
impose that choice.
|
||||||
|
|
||||||
|
This section is intended to make thoroughly clear what is believed to
|
||||||
|
be a consequence of the rest of this License.
|
||||||
|
|
||||||
|
12. If the distribution and/or use of the Library is restricted in
|
||||||
|
certain countries either by patents or by copyrighted interfaces, the
|
||||||
|
original copyright holder who places the Library under this License may add
|
||||||
|
an explicit geographical distribution limitation excluding those countries,
|
||||||
|
so that distribution is permitted only in or among countries not thus
|
||||||
|
excluded. In such case, this License incorporates the limitation as if
|
||||||
|
written in the body of this License.
|
||||||
|
|
||||||
|
13. The Free Software Foundation may publish revised and/or new
|
||||||
|
versions of the Lesser General Public License from time to time.
|
||||||
|
Such new versions will be similar in spirit to the present version,
|
||||||
|
but may differ in detail to address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the Library
|
||||||
|
specifies a version number of this License which applies to it and
|
||||||
|
"any later version", you have the option of following the terms and
|
||||||
|
conditions either of that version or of any later version published by
|
||||||
|
the Free Software Foundation. If the Library does not specify a
|
||||||
|
license version number, you may choose any version ever published by
|
||||||
|
the Free Software Foundation.
|
||||||
|
|
||||||
|
14. If you wish to incorporate parts of the Library into other free
|
||||||
|
programs whose distribution conditions are incompatible with these,
|
||||||
|
write to the author to ask for permission. For software which is
|
||||||
|
copyrighted by the Free Software Foundation, write to the Free
|
||||||
|
Software Foundation; we sometimes make exceptions for this. Our
|
||||||
|
decision will be guided by the two goals of preserving the free status
|
||||||
|
of all derivatives of our free software and of promoting the sharing
|
||||||
|
and reuse of software generally.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
|
||||||
|
15. BECAUSE THE LIBRARY IS LICENSED FREE OF CHARGE, THERE IS NO
|
||||||
|
WARRANTY FOR THE LIBRARY, TO THE EXTENT PERMITTED BY APPLICABLE LAW.
|
||||||
|
EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR
|
||||||
|
OTHER PARTIES PROVIDE THE LIBRARY "AS IS" WITHOUT WARRANTY OF ANY
|
||||||
|
KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE
|
||||||
|
LIBRARY IS WITH YOU. SHOULD THE LIBRARY PROVE DEFECTIVE, YOU ASSUME
|
||||||
|
THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN
|
||||||
|
WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY
|
||||||
|
AND/OR REDISTRIBUTE THE LIBRARY AS PERMITTED ABOVE, BE LIABLE TO YOU
|
||||||
|
FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR
|
||||||
|
CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE
|
||||||
|
LIBRARY (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING
|
||||||
|
RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A
|
||||||
|
FAILURE OF THE LIBRARY TO OPERATE WITH ANY OTHER SOFTWARE), EVEN IF
|
||||||
|
SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH
|
||||||
|
DAMAGES.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Libraries
|
||||||
|
|
||||||
|
If you develop a new library, and you want it to be of the greatest
|
||||||
|
possible use to the public, we recommend making it free software that
|
||||||
|
everyone can redistribute and change. You can do so by permitting
|
||||||
|
redistribution under these terms (or, alternatively, under the terms of the
|
||||||
|
ordinary General Public License).
|
||||||
|
|
||||||
|
To apply these terms, attach the following notices to the library. It is
|
||||||
|
safest to attach them to the start of each source file to most effectively
|
||||||
|
convey the exclusion of warranty; and each file should have at least the
|
||||||
|
"copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the library's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or your
|
||||||
|
school, if any, to sign a "copyright disclaimer" for the library, if
|
||||||
|
necessary. Here is a sample; alter the names:
|
||||||
|
|
||||||
|
Yoyodyne, Inc., hereby disclaims all copyright interest in the
|
||||||
|
library `Frob' (a library for tweaking knobs) written by James Random Hacker.
|
||||||
|
|
||||||
|
<signature of Ty Coon>, 1 April 1990
|
||||||
|
Ty Coon, President of Vice
|
||||||
|
|
||||||
|
That's all there is to it!
|
||||||
|
|
||||||
|
|
368
05/tcc-0.9.25/Changelog
Normal file
368
05/tcc-0.9.25/Changelog
Normal file
|
@ -0,0 +1,368 @@
|
||||||
|
version 0.9.25:
|
||||||
|
|
||||||
|
- first support for x86-64 target (Shinichiro Hamaji)
|
||||||
|
- support µClibc
|
||||||
|
- split tcc.c into tcc.h libtcc.c tccpp.c tccgen.c tcc.c
|
||||||
|
- improved preprocess output with linenumbers and spaces preserved
|
||||||
|
- tcc_relocate now copies code into user buffer
|
||||||
|
- fix bitfields with non-int types and in unions
|
||||||
|
- improve ARM cross-compiling (Daniel Glöckner)
|
||||||
|
- link stabstr sections from multiple objects
|
||||||
|
- better (still limited) support for multiple TCCStates
|
||||||
|
|
||||||
|
version 0.9.24:
|
||||||
|
|
||||||
|
- added verbosity levels -v, -vv, -vvv
|
||||||
|
- Accept standard input as an inputstream (Hanzac Chen)
|
||||||
|
- Support c89 compilers other than gcc (Hanzac Chen)
|
||||||
|
- -soname linker option (Marc Andre Tanner)
|
||||||
|
- Just warn about unknown directives, ignore quotes in #error/#warning
|
||||||
|
- Define __STDC_VERSION__=199901L (477)
|
||||||
|
- Switch to newer tccpe.c (includes support for resources)
|
||||||
|
- Handle backslashes within #include/#error/#warning
|
||||||
|
- Import changesets (part 4) 428,457,460,467: defines for openbsd etc.
|
||||||
|
- Use _WIN32 for a windows hosted tcc and define it for the PE target,
|
||||||
|
otherwise define __unix / __linux (Detlef Riekenberg)
|
||||||
|
- Import changesets (part 3) 409,410: ARM EABI by Daniel Glöckner
|
||||||
|
- Some in-between fixes:
|
||||||
|
TCC -E no longer hangs with macro calls involving newlines.
|
||||||
|
(next_nomacro1 now advances the read-pointer with TOK_LINEFEED)
|
||||||
|
Global cast (int g_i = 1LL;) no longer crashes tcc.
|
||||||
|
(nocode_wanted is initially 1, and only 0 for gen_function)
|
||||||
|
On win32 now tcc.exe finds 'include' & 'lib' even if itself is in 'bin'.
|
||||||
|
(new function w32_tcc_lib_path removes 'bin' if detected)
|
||||||
|
Added quick build batch file for mingw (win32/build-tcc.bat)
|
||||||
|
Last added case label optimization (455) produced wrong code. Reverted.
|
||||||
|
|
||||||
|
- Import more changesets from Rob Landley's fork (part 2):
|
||||||
|
487: Handle long long constants in gen_opic() (Rob Landley)
|
||||||
|
484: Handle parentheses within __attribute__((...)) (Rob Landley)
|
||||||
|
480: Remove a goto in decl_initializer_alloc (Rob Landley)
|
||||||
|
475: Fix dereferences in inline assembly output (Joshua Phillips)
|
||||||
|
474: Cast ptrs to ints of different sizes correctly (Joshua Phillips)
|
||||||
|
473: Fix size of structs with empty array member (Joshua Phillips)
|
||||||
|
470: No warning for && and || with mixed pointers/integers (Rob Landley)
|
||||||
|
469: Fix symbol visibility problems in the linker (Vincent Pit)
|
||||||
|
468: Allow && and || involving pointer arguments (Rob Landley)
|
||||||
|
455: Optimize case labels with no code in between (Zdenek Pavlas)
|
||||||
|
450: Implement alloca for x86 (grischka)
|
||||||
|
415: Parse unicode escape sequences (Axel Liljencrantz)
|
||||||
|
407: Add a simple va_copy() in stdarg.h (Hasso Tepper)
|
||||||
|
400: Allow typedef names as symbols (Dave Dodge)
|
||||||
|
|
||||||
|
- Import some changesets from Rob Landley's fork (part 1):
|
||||||
|
462: Use LGPL with bcheck.c and il-gen.c
|
||||||
|
458: Fix global compound literals (in unary: case '&':) (Andrew Johnson)
|
||||||
|
456: Use return code from tcc_output_file in main() (Michael Somos)
|
||||||
|
442: Fix indirections with function pointers (***fn)() (grischka)
|
||||||
|
441: Fix LL left shift in libtcc1.c:__shldi3 (grischka)
|
||||||
|
440: Pass structures and function ptrs through ?: (grischka)
|
||||||
|
439: Keep rvalue in bit assignment (bit2 = bit1 = x) (grischka)
|
||||||
|
438: Degrade nonportable pointer assignment to warning (grischka)
|
||||||
|
437: Call 'saveregs()' before jumping with logical and/or/not (grischka)
|
||||||
|
435: Put local static variables into global memory (grischka)
|
||||||
|
432/434: Cast double and ptr to bool (grischka)
|
||||||
|
420: Zero pad x87 tenbyte long doubles (Felix Nawothnig)
|
||||||
|
417: Make 'sizeof' unsigned (Rob Landley)
|
||||||
|
397: Fix save_reg for longlongs (Daniel Glöckner)
|
||||||
|
396: Fix "invalid relocation entry" problem on ubuntu - (Bernhard Fischer)
|
||||||
|
|
||||||
|
- ignore AS_NEEDED ld command
|
||||||
|
- mark executable sections as executable when running in memory
|
||||||
|
- added support for win32 wchar_t (Filip Navara)
|
||||||
|
- segment override prefix support (Filip Navara)
|
||||||
|
- normalized slashes in paths (Filip Navara)
|
||||||
|
- windows style fastcall (Filip Navara)
|
||||||
|
- support for empty input register section in asm (Filip Navara)
|
||||||
|
- anonymous union/struct support (Filip Navara)
|
||||||
|
- fixed parsing of function parameters
|
||||||
|
- workaround for function pointers in conditional expressions (Dave Dodge)
|
||||||
|
- initial '-E' option support to use the C preprocessor alone
|
||||||
|
- discard type qualifiers when comparing function parameters (Dave Dodge)
|
||||||
|
- Bug fix: A long long value used as a test expression ignores the
|
||||||
|
upper 32 bits at runtime (Dave Dodge)
|
||||||
|
- fixed multiple concatenation of PPNUM tokens (initial patch by Dave Dodge)
|
||||||
|
- fixed multiple typedef specifiers handling
|
||||||
|
- fixed sign extension in some type conversions (Dave Dodge)
|
||||||
|
|
||||||
|
version 0.9.23:
|
||||||
|
|
||||||
|
- initial PE executable format for windows version (grischka)
|
||||||
|
- '#pragma pack' support (grischka)
|
||||||
|
- '#include_next' support (Bernhard Fischer)
|
||||||
|
- ignore '-pipe' option
|
||||||
|
- added -f[no-]leading-underscore
|
||||||
|
- preprocessor function macro parsing fix (grischka)
|
||||||
|
|
||||||
|
version 0.9.22:
|
||||||
|
|
||||||
|
- simple memory optimisations: kernel compilation is 30% faster
|
||||||
|
- linker symbol definitions fixes
|
||||||
|
- gcc 3.4 fixes
|
||||||
|
- fixed value stack full error
|
||||||
|
- 'packed' attribute support for variables and structure fields
|
||||||
|
- ignore 'const' and 'volatile' in function prototypes
|
||||||
|
- allow '_Bool' in bit fields
|
||||||
|
|
||||||
|
version 0.9.21:
|
||||||
|
|
||||||
|
- ARM target support (Daniel Glöckner)
|
||||||
|
- added '-funsigned-char, '-fsigned-char' and
|
||||||
|
'-Wimplicit-function-declaration'
|
||||||
|
- fixed assignment of const struct in struct
|
||||||
|
- line comment fix (reported by Bertram Felgenhauer)
|
||||||
|
- initial TMS320C67xx target support (TK)
|
||||||
|
- win32 configure
|
||||||
|
- regparm() attribute
|
||||||
|
- many built-in assembler fixes
|
||||||
|
- added '.org', '.fill' and '.previous' assembler directives
|
||||||
|
- '-fno-common' option
|
||||||
|
- '-Ttext' linker option
|
||||||
|
- section alignment fixes
|
||||||
|
- bit fields fixes
|
||||||
|
- do not generate code for unused inline functions
|
||||||
|
- '-oformat' linker option.
|
||||||
|
- added 'binary' output format.
|
||||||
|
|
||||||
|
version 0.9.20:
|
||||||
|
|
||||||
|
- added '-w' option
|
||||||
|
- added '.gnu.linkonce' ELF sections support
|
||||||
|
- fixed libc linking when running in memory (avoid 'stat' function
|
||||||
|
errors).
|
||||||
|
- extended '-run' option to be able to give several arguments to a C
|
||||||
|
script.
|
||||||
|
|
||||||
|
version 0.9.19:
|
||||||
|
|
||||||
|
- "alacarte" linking (Dave Long)
|
||||||
|
- simpler function call
|
||||||
|
- more strict type checks
|
||||||
|
- added 'const' and 'volatile' support and associated warnings
|
||||||
|
- added -Werror, -Wunsupported, -Wwrite-strings, -Wall.
|
||||||
|
- added __builtin_types_compatible_p() and __builtin_constant_p()
|
||||||
|
- chars support in assembler (Dave Long)
|
||||||
|
- .string, .globl, .section, .text, .data and .bss asm directive
|
||||||
|
support (Dave Long)
|
||||||
|
- man page generated from tcc-doc.texi
|
||||||
|
- fixed macro argument substitution
|
||||||
|
- fixed zero argument macro parsing
|
||||||
|
- changed license to LGPL
|
||||||
|
- added -rdynamic option support
|
||||||
|
|
||||||
|
version 0.9.18:
|
||||||
|
|
||||||
|
- header fix (time.h)
|
||||||
|
- fixed inline asm without operand case
|
||||||
|
- fixed 'default:' or 'case x:' with '}' after (incorrect C construct accepted
|
||||||
|
by gcc)
|
||||||
|
- added 'A' inline asm constraint.
|
||||||
|
|
||||||
|
version 0.9.17:
|
||||||
|
|
||||||
|
- PLT generation fix
|
||||||
|
- tcc doc fixes (Peter Lund)
|
||||||
|
- struct parse fix (signaled by Pedro A. Aranda Gutierrez)
|
||||||
|
- better _Bool lvalue support (signaled by Alex Measday)
|
||||||
|
- function parameters must be converted to pointers (signaled by Neil Brown)
|
||||||
|
- sanitized string and character constant parsing
|
||||||
|
- fixed comment parse (signaled by Damian M Gryski)
|
||||||
|
- fixed macro function bug (signaled by Philippe Ribet)
|
||||||
|
- added configure (initial patch by Mitchell N Charity)
|
||||||
|
- added '-run' and '-v' options (initial patch by vlindos)
|
||||||
|
- added real date report in __DATE__ and __TIME__ macros
|
||||||
|
|
||||||
|
version 0.9.16:
|
||||||
|
|
||||||
|
- added assembler language support
|
||||||
|
- added GCC inline asm() support
|
||||||
|
- fixed multiple variable definitions : uninitialized variables are
|
||||||
|
created as COMMON symbols.
|
||||||
|
- optimized macro processing
|
||||||
|
- added GCC statement expressions support
|
||||||
|
- added GCC local labels support
|
||||||
|
- fixed array declaration in old style function parameters
|
||||||
|
- support casts in static structure initializations
|
||||||
|
- added various __xxx[__] keywords for GCC compatibility
|
||||||
|
- ignore __extension__ GCC in an expression or in a type (still not perfect)
|
||||||
|
- added '? :' GCC extension support
|
||||||
|
|
||||||
|
version 0.9.15:
|
||||||
|
|
||||||
|
- compilation fixes for glibc 2.2, gcc 2.95.3 and gcc 3.2.
|
||||||
|
- FreeBSD compile fixes. Makefile patches still missing (Carl Drougge).
|
||||||
|
- fixed file type guessing if '.' is in the path.
|
||||||
|
- fixed tcc_compile_string()
|
||||||
|
- add a dummy page in ELF files to fix RX/RW accesses (pageexec at
|
||||||
|
freemail dot hu).
|
||||||
|
|
||||||
|
version 0.9.14:
|
||||||
|
|
||||||
|
- added #warning. error message if invalid preprocessing directive.
|
||||||
|
- added CType structure to ease typing (faster parse).
|
||||||
|
- suppressed secondary hash tables (faster parse).
|
||||||
|
- rewrote parser by optimizing common cases (faster parse).
|
||||||
|
- fixed signed long long comparisons.
|
||||||
|
- fixed 'int a(), b();' declaration case.
|
||||||
|
- fixed structure init without '{}'.
|
||||||
|
- correct alignment support in structures.
|
||||||
|
- empty structures support.
|
||||||
|
- gcc testsuite now supported.
|
||||||
|
- output only warning if implicit integer/pointer conversions.
|
||||||
|
- added static bitfield init.
|
||||||
|
|
||||||
|
version 0.9.13:
|
||||||
|
|
||||||
|
- correct preprocessing token pasting (## operator) in all cases (added
|
||||||
|
preprocessing number token).
|
||||||
|
- fixed long long register spill.
|
||||||
|
- fixed signed long long '>>'.
|
||||||
|
- removed memory leaks.
|
||||||
|
- better error handling : processing can continue on link errors. A
|
||||||
|
custom callback can be added to display error messages. Most
|
||||||
|
errors do not call exit() now.
|
||||||
|
- ignore -O, -W, -m and -f options
|
||||||
|
- added old style function declarations
|
||||||
|
- added GCC __alignof__ support.
|
||||||
|
- added GCC typeof support.
|
||||||
|
- added GCC computed gotos support.
|
||||||
|
- added stack backtrace in runtime error message. Improved runtime
|
||||||
|
error position display.
|
||||||
|
|
||||||
|
version 0.9.12:
|
||||||
|
|
||||||
|
- more fixes for || and && handling.
|
||||||
|
- improved '? :' type handling.
|
||||||
|
- fixed bound checking generation with structures
|
||||||
|
- force '#endif' to be in same file as matching '#if'
|
||||||
|
- #include file optimization with '#ifndef #endif' construct detection
|
||||||
|
- macro handling optimization
|
||||||
|
- added tcc_relocate() and tcc_get_symbol() in libtcc.
|
||||||
|
|
||||||
|
version 0.9.11:
|
||||||
|
|
||||||
|
- stdarg.h fix for double type (thanks to Philippe Ribet).
|
||||||
|
- correct white space characters and added MSDOS newline support.
|
||||||
|
- fixed invalid implicit function call type declaration.
|
||||||
|
- special macros such as __LINE__ are defined if tested with defined().
|
||||||
|
- fixed '!' operator with relocated address.
|
||||||
|
- added symbol + offset relocation (fixes some static variable initializers)
|
||||||
|
- '-l' option can be specified anywhere. '-c' option yields default
|
||||||
|
output name. added '-r' option for relocatable output.
|
||||||
|
- fixed '\nnn' octal parsing.
|
||||||
|
- fixed local extern variables declarations.
|
||||||
|
|
||||||
|
version 0.9.10:
|
||||||
|
|
||||||
|
- fixed lvalue type when saved in local stack.
|
||||||
|
- fixed '#include' syntax when using macros.
|
||||||
|
- fixed '#line' bug.
|
||||||
|
- removed size limit on strings. Unified string constants handling
|
||||||
|
with variable declarations.
|
||||||
|
- added correct support for '\xX' in wchar_t strings.
|
||||||
|
- added support for bound checking in generated executables
|
||||||
|
- fixed -I include order.
|
||||||
|
- fixed incorrect function displayed in runtime error.
|
||||||
|
|
||||||
|
version 0.9.9:
|
||||||
|
|
||||||
|
- fixed preprocessor expression parsing for #if/#elif.
|
||||||
|
- relocated debug info (.stab section).
|
||||||
|
- relocated bounds info (.bounds section).
|
||||||
|
- fixed cast to char of char constants ('\377' is -1 instead of 255)
|
||||||
|
- fixed implicit cast for unary plus.
|
||||||
|
- strings and '__func__' have now 'char[]' type instead of 'char *'
|
||||||
|
(fixes sizeof() return value).
|
||||||
|
- added __start_xxx and __stop_xxx symbols in linker.
|
||||||
|
- better DLL creation support (option -shared begins to work).
|
||||||
|
- ELF sections and hash tables are resized dynamically.
|
||||||
|
- executables and DLLs are stripped by default.
|
||||||
|
|
||||||
|
version 0.9.8:
|
||||||
|
|
||||||
|
- First version of full ELF linking support (generate objects, static
|
||||||
|
executable, dynamic executable, dynamic libraries). Dynamic library
|
||||||
|
support is not finished (need PIC support in compiler and some
|
||||||
|
patches in symbol exporting).
|
||||||
|
- First version of ELF loader for object (.o) and archive (.a) files.
|
||||||
|
- Support of simple GNU ld scripts (GROUP and FILE commands)
|
||||||
|
- Separated runtime library and bound check code from TCC (smaller
|
||||||
|
compiler core).
|
||||||
|
- fixed register reload in float compare.
|
||||||
|
- fixed implicit char/short to int casting.
|
||||||
|
- allow array type for address of ('&') operator.
|
||||||
|
- fixed unused || or && result.
|
||||||
|
- added GCC style variadic macro support.
|
||||||
|
- optimized bound checking code for array access.
|
||||||
|
- tcc includes are now in $(prefix)/lib/tcc/include.
|
||||||
|
- more command line options - more consistent handling of multiple
|
||||||
|
input files.
|
||||||
|
- added tcc man page (thanks to Cyril Bouthors).
|
||||||
|
- uClibc Makefile update
|
||||||
|
- converted documentation to texinfo format.
|
||||||
|
- added developper's guide in documentation.
|
||||||
|
|
||||||
|
version 0.9.7:
|
||||||
|
|
||||||
|
- added library API for easy dynamic compilation (see libtcc.h - first
|
||||||
|
draft).
|
||||||
|
- fixed long long register spill bug.
|
||||||
|
- fixed '? :' register spill bug.
|
||||||
|
|
||||||
|
version 0.9.6:
|
||||||
|
|
||||||
|
- added floating point constant propagation (fixes negative floating
|
||||||
|
point constants bug).
|
||||||
|
|
||||||
|
version 0.9.5:
|
||||||
|
|
||||||
|
- uClibc patches (submitted by Alfonso Martone).
|
||||||
|
- error reporting fix
|
||||||
|
- added CONFIG_TCC_BCHECK to get smaller code if needed.
|
||||||
|
|
||||||
|
version 0.9.4:
|
||||||
|
|
||||||
|
- windows port (currently cannot use -g, -b and dll functions).
|
||||||
|
- faster and simpler I/O handling.
|
||||||
|
- '-D' option works in all cases.
|
||||||
|
- preprocessor fixes (#elif and empty macro args)
|
||||||
|
- floating point fixes
|
||||||
|
- first code for CIL generation (does not work yet)
|
||||||
|
|
||||||
|
version 0.9.3:
|
||||||
|
|
||||||
|
- better and smaller code generator.
|
||||||
|
- full ISOC99 64 bit 'long long' support.
|
||||||
|
- full 32 bit 'float', 64 bit 'double' and 96 bit 'long double' support.
|
||||||
|
- added '-U' option.
|
||||||
|
- added assembly sections support.
|
||||||
|
- even faster startup time by mmaping sections instead of mallocing them.
|
||||||
|
- added GNUC __attribute__ keyword support (currently supports
|
||||||
|
'section' and 'aligned' attributes).
|
||||||
|
- added ELF file output (only usable for debugging now)
|
||||||
|
- added debug symbol generation (STAB format).
|
||||||
|
- added integrated runtime error analysis ('-g' option: print clear
|
||||||
|
run time error messages instead of "Segmentation fault").
|
||||||
|
- added first version of tiny memory and bound checker ('-b' option).
|
||||||
|
|
||||||
|
version 0.9.2:
|
||||||
|
|
||||||
|
- even faster parsing.
|
||||||
|
- various syntax parsing fixes.
|
||||||
|
- fixed external relocation handling for variables or functions pointers.
|
||||||
|
- better function pointers type handling.
|
||||||
|
- can compile multiple files (-i option).
|
||||||
|
- ANSI C bit fields are supported.
|
||||||
|
- beginning of float/double/long double support.
|
||||||
|
- beginning of long long support.
|
||||||
|
|
||||||
|
version 0.9.1:
|
||||||
|
|
||||||
|
- full ISOC99 initializers handling.
|
||||||
|
- compound literals.
|
||||||
|
- structures handle in assignments and as function param or return value.
|
||||||
|
- wide chars and strings.
|
||||||
|
- macro bug fix
|
||||||
|
|
||||||
|
version 0.9:
|
||||||
|
- initial version.
|
272
05/tcc-0.9.25/Makefile
Normal file
272
05/tcc-0.9.25/Makefile
Normal file
|
@ -0,0 +1,272 @@
|
||||||
|
#
|
||||||
|
# Tiny C Compiler Makefile
|
||||||
|
#
|
||||||
|
|
||||||
|
TOP ?= .
|
||||||
|
include $(TOP)/config.mak
|
||||||
|
|
||||||
|
CFLAGS+=-g -Wall
|
||||||
|
CFLAGS_P=$(CFLAGS) -pg -static -DCONFIG_TCC_STATIC
|
||||||
|
LIBS_P=
|
||||||
|
|
||||||
|
ifneq ($(GCC_MAJOR),2)
|
||||||
|
CFLAGS+=-fno-strict-aliasing
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(ARCH),i386)
|
||||||
|
CFLAGS+=-mpreferred-stack-boundary=2
|
||||||
|
ifeq ($(GCC_MAJOR),2)
|
||||||
|
CFLAGS+=-m386 -malign-functions=0
|
||||||
|
else
|
||||||
|
CFLAGS+=-march=i386 -falign-functions=0
|
||||||
|
ifneq ($(GCC_MAJOR),3)
|
||||||
|
CFLAGS+=-Wno-pointer-sign -Wno-sign-compare -D_FORTIFY_SOURCE=0
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(ARCH),x86-64)
|
||||||
|
CFLAGS+=-Wno-pointer-sign
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifndef CONFIG_WIN32
|
||||||
|
LIBS=-lm
|
||||||
|
ifndef CONFIG_NOLDL
|
||||||
|
LIBS+=-ldl
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifdef CONFIG_WIN32
|
||||||
|
NATIVE_TARGET=-DTCC_TARGET_PE
|
||||||
|
LIBTCC1=libtcc1.a
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),i386)
|
||||||
|
NATIVE_TARGET=-DTCC_TARGET_I386
|
||||||
|
LIBTCC1=libtcc1.a
|
||||||
|
BCHECK_O=bcheck.o
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),arm)
|
||||||
|
NATIVE_TARGET=-DTCC_TARGET_ARM
|
||||||
|
NATIVE_TARGET+=$(if $(wildcard /lib/ld-linux.so.3),-DTCC_ARM_EABI)
|
||||||
|
NATIVE_TARGET+=$(if $(shell grep -l "^Features.* \(vfp\|iwmmxt\) " /proc/cpuinfo),-DTCC_ARM_VFP)
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),x86-64)
|
||||||
|
NATIVE_TARGET=-DTCC_TARGET_X86_64
|
||||||
|
LIBTCC1=libtcc1.a
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifneq ($(wildcard /lib/ld-uClibc.so.0),)
|
||||||
|
NATIVE_TARGET+=-DTCC_UCLIBC
|
||||||
|
BCHECK_O=
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifdef CONFIG_USE_LIBGCC
|
||||||
|
LIBTCC1=
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(TOP),.)
|
||||||
|
|
||||||
|
PROGS=tcc$(EXESUF)
|
||||||
|
|
||||||
|
I386_CROSS = i386-tcc$(EXESUF)
|
||||||
|
WIN32_CROSS = i386-win32-tcc$(EXESUF)
|
||||||
|
X64_CROSS = x86_64-tcc$(EXESUF)
|
||||||
|
ARM_CROSS = arm-tcc-fpa$(EXESUF) arm-tcc-fpa-ld$(EXESUF) \
|
||||||
|
arm-tcc-vfp$(EXESUF) arm-tcc-vfp-eabi$(EXESUF)
|
||||||
|
C67_CROSS = c67-tcc$(EXESUF)
|
||||||
|
|
||||||
|
CORE_FILES = tcc.c libtcc.c tccpp.c tccgen.c tccelf.c tccasm.c \
|
||||||
|
tcc.h config.h libtcc.h tcctok.h
|
||||||
|
I386_FILES = $(CORE_FILES) i386-gen.c i386-asm.c i386-asm.h
|
||||||
|
WIN32_FILES = $(CORE_FILES) i386-gen.c i386-asm.c i386-asm.h tccpe.c
|
||||||
|
X86_64_FILES = $(CORE_FILES) x86_64-gen.c
|
||||||
|
ARM_FILES = $(CORE_FILES) arm-gen.c
|
||||||
|
C67_FILES = $(CORE_FILES) c67-gen.c tcccoff.c
|
||||||
|
|
||||||
|
ifdef CONFIG_WIN32
|
||||||
|
PROGS+=tiny_impdef$(EXESUF) tiny_libmaker$(EXESUF)
|
||||||
|
NATIVE_FILES=$(WIN32_FILES)
|
||||||
|
PROGS_CROSS=$(I386_CROSS) $(X64_CROSS) $(ARM_CROSS) $(C67_CROSS)
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),i386)
|
||||||
|
NATIVE_FILES=$(I386_FILES)
|
||||||
|
PROGS_CROSS=$(X64_CROSS) $(WIN32_CROSS) $(ARM_CROSS) $(C67_CROSS)
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),x86-64)
|
||||||
|
NATIVE_FILES=$(X86_64_FILES)
|
||||||
|
PROGS_CROSS=$(I386_CROSS) $(WIN32_CROSS) $(ARM_CROSS) $(C67_CROSS)
|
||||||
|
else
|
||||||
|
ifeq ($(ARCH),arm)
|
||||||
|
NATIVE_FILES=$(ARM_FILES)
|
||||||
|
PROGS_CROSS=$(I386_CROSS) $(X64_CROSS) $(WIN32_CROSS) $(C67_CROSS)
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifdef CONFIG_CROSS
|
||||||
|
PROGS+=$(PROGS_CROSS)
|
||||||
|
endif
|
||||||
|
|
||||||
|
all: $(PROGS) $(LIBTCC1) $(BCHECK_O) libtcc.a tcc-doc.html tcc.1 libtcc_test$(EXESUF)
|
||||||
|
|
||||||
|
# Host Tiny C Compiler
|
||||||
|
tcc$(EXESUF): $(NATIVE_FILES)
|
||||||
|
$(CC) -o $@ $< $(NATIVE_TARGET) $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
# Cross Tiny C Compilers
|
||||||
|
i386-tcc$(EXESUF): $(I386_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_I386 $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
i386-win32-tcc$(EXESUF): $(WIN32_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_PE $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
x86_64-tcc$(EXESUF): $(X86_64_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_X86_64 $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
c67-tcc$(EXESUF): $(C67_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_C67 $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
arm-tcc-fpa$(EXESUF): $(ARM_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_ARM $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
arm-tcc-fpa-ld$(EXESUF): $(ARM_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_ARM -DLDOUBLE_SIZE=12 $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
arm-tcc-vfp$(EXESUF): $(ARM_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_ARM -DTCC_ARM_VFP $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
arm-tcc-vfp-eabi$(EXESUF): $(ARM_FILES)
|
||||||
|
$(CC) -o $@ $< -DTCC_TARGET_ARM -DTCC_ARM_EABI $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
# libtcc generation and test
|
||||||
|
libtcc.o: $(NATIVE_FILES)
|
||||||
|
$(CC) -o $@ -c libtcc.c $(NATIVE_TARGET) $(CFLAGS)
|
||||||
|
|
||||||
|
libtcc.a: libtcc.o
|
||||||
|
$(AR) rcs $@ $^
|
||||||
|
|
||||||
|
libtcc_test$(EXESUF): tests/libtcc_test.c libtcc.a
|
||||||
|
$(CC) -o $@ $^ -I. $(CFLAGS) $(LIBS)
|
||||||
|
|
||||||
|
libtest: libtcc_test$(EXESUF) $(LIBTCC1)
|
||||||
|
./libtcc_test$(EXESUF) lib_path=.
|
||||||
|
|
||||||
|
# profiling version
|
||||||
|
tcc_p$(EXESUF): $(NATIVE_FILES)
|
||||||
|
$(CC) -o $@ $< $(NATIVE_TARGET) $(CFLAGS_P) $(LIBS_P)
|
||||||
|
|
||||||
|
# windows utilities
|
||||||
|
tiny_impdef$(EXESUF): win32/tools/tiny_impdef.c
|
||||||
|
$(CC) -o $@ $< $(CFLAGS)
|
||||||
|
tiny_libmaker$(EXESUF): win32/tools/tiny_libmaker.c
|
||||||
|
$(CC) -o $@ $< $(CFLAGS)
|
||||||
|
|
||||||
|
# TinyCC runtime libraries
|
||||||
|
LIBTCC1_OBJS=libtcc1.o
|
||||||
|
LIBTCC1_CC=$(CC)
|
||||||
|
VPATH+=lib
|
||||||
|
ifdef CONFIG_WIN32
|
||||||
|
# for windows, we must use TCC because we generate ELF objects
|
||||||
|
LIBTCC1_OBJS+=crt1.o wincrt1.o dllcrt1.o dllmain.o chkstk.o
|
||||||
|
LIBTCC1_CC=./tcc.exe -Bwin32 -DTCC_TARGET_PE
|
||||||
|
VPATH+=win32/lib
|
||||||
|
endif
|
||||||
|
ifeq ($(ARCH),i386)
|
||||||
|
LIBTCC1_OBJS+=alloca86.o alloca86-bt.o
|
||||||
|
endif
|
||||||
|
|
||||||
|
%.o: %.c
|
||||||
|
$(LIBTCC1_CC) -o $@ -c $< -O2 -Wall
|
||||||
|
|
||||||
|
%.o: %.S
|
||||||
|
$(LIBTCC1_CC) -o $@ -c $<
|
||||||
|
|
||||||
|
libtcc1.a: $(LIBTCC1_OBJS)
|
||||||
|
$(AR) rcs $@ $^
|
||||||
|
|
||||||
|
bcheck.o: bcheck.c
|
||||||
|
$(CC) -o $@ -c $< -O2 -Wall
|
||||||
|
|
||||||
|
# install
|
||||||
|
TCC_INCLUDES = stdarg.h stddef.h stdbool.h float.h varargs.h tcclib.h
|
||||||
|
INSTALL=install
|
||||||
|
|
||||||
|
ifndef CONFIG_WIN32
|
||||||
|
install: $(PROGS) $(LIBTCC1) $(BCHECK_O) libtcc.a tcc.1 tcc-doc.html
|
||||||
|
mkdir -p "$(bindir)"
|
||||||
|
$(INSTALL) -s -m755 $(PROGS) "$(bindir)"
|
||||||
|
mkdir -p "$(mandir)/man1"
|
||||||
|
$(INSTALL) tcc.1 "$(mandir)/man1"
|
||||||
|
mkdir -p "$(tccdir)"
|
||||||
|
mkdir -p "$(tccdir)/include"
|
||||||
|
ifneq ($(LIBTCC1),)
|
||||||
|
$(INSTALL) -m644 $(LIBTCC1) "$(tccdir)"
|
||||||
|
endif
|
||||||
|
ifneq ($(BCHECK_O),)
|
||||||
|
$(INSTALL) -m644 $(BCHECK_O) "$(tccdir)"
|
||||||
|
endif
|
||||||
|
$(INSTALL) -m644 $(addprefix include/,$(TCC_INCLUDES)) "$(tccdir)/include"
|
||||||
|
mkdir -p "$(docdir)"
|
||||||
|
$(INSTALL) -m644 tcc-doc.html "$(docdir)"
|
||||||
|
mkdir -p "$(libdir)"
|
||||||
|
$(INSTALL) -m644 libtcc.a "$(libdir)"
|
||||||
|
mkdir -p "$(includedir)"
|
||||||
|
$(INSTALL) -m644 libtcc.h "$(includedir)"
|
||||||
|
|
||||||
|
uninstall:
|
||||||
|
rm -fv $(foreach P,$(PROGS),"$(bindir)/$P")
|
||||||
|
rm -fv $(foreach P,$(LIBTCC1) $(BCHECK_O),"$(tccdir)/$P")
|
||||||
|
rm -fv $(foreach P,$(TCC_INCLUDES),"$(tccdir)/include/$P")
|
||||||
|
rm -fv "$(docdir)/tcc-doc.html" "$(mandir)/man1/tcc.1"
|
||||||
|
rm -fv "$(libdir)/libtcc.a" "$(includedir)/libtcc.h"
|
||||||
|
|
||||||
|
else
|
||||||
|
install: $(PROGS) $(LIBTCC1) libtcc.a tcc-doc.html
|
||||||
|
mkdir -p "$(tccdir)"
|
||||||
|
mkdir -p "$(tccdir)/lib"
|
||||||
|
mkdir -p "$(tccdir)/include"
|
||||||
|
mkdir -p "$(tccdir)/examples"
|
||||||
|
mkdir -p "$(tccdir)/doc"
|
||||||
|
mkdir -p "$(tccdir)/libtcc"
|
||||||
|
$(INSTALL) -s -m755 $(PROGS) "$(tccdir)"
|
||||||
|
$(INSTALL) -m644 $(LIBTCC1) win32/lib/*.def "$(tccdir)/lib"
|
||||||
|
cp -r win32/include/. "$(tccdir)/include"
|
||||||
|
cp -r win32/examples/. "$(tccdir)/examples"
|
||||||
|
# $(INSTALL) -m644 $(addprefix include/,$(TCC_INCLUDES)) "$(tccdir)/include"
|
||||||
|
$(INSTALL) -m644 tcc-doc.html win32/tcc-win32.txt "$(tccdir)/doc"
|
||||||
|
$(INSTALL) -m644 libtcc.a libtcc.h "$(tccdir)/libtcc"
|
||||||
|
endif
|
||||||
|
|
||||||
|
# documentation and man page
|
||||||
|
tcc-doc.html: tcc-doc.texi
|
||||||
|
-texi2html -monolithic -number $<
|
||||||
|
|
||||||
|
tcc.1: tcc-doc.texi
|
||||||
|
-./texi2pod.pl $< tcc.pod
|
||||||
|
-pod2man --section=1 --center=" " --release=" " tcc.pod > $@
|
||||||
|
|
||||||
|
# tar release (use 'make -k tar' on a checkouted tree)
|
||||||
|
TCC-VERSION=tcc-$(shell cat VERSION)
|
||||||
|
tar:
|
||||||
|
rm -rf /tmp/$(TCC-VERSION)
|
||||||
|
cp -r . /tmp/$(TCC-VERSION)
|
||||||
|
( cd /tmp ; tar zcvf ~/$(TCC-VERSION).tar.gz $(TCC-VERSION) --exclude CVS )
|
||||||
|
rm -rf /tmp/$(TCC-VERSION)
|
||||||
|
|
||||||
|
# in tests subdir
|
||||||
|
test clean :
|
||||||
|
$(MAKE) -C tests $@
|
||||||
|
|
||||||
|
# clean
|
||||||
|
clean: local_clean
|
||||||
|
local_clean:
|
||||||
|
rm -vf $(PROGS) tcc_p$(EXESUF) tcc.pod *~ *.o *.a *.out libtcc_test$(EXESUF)
|
||||||
|
|
||||||
|
distclean: clean
|
||||||
|
rm -vf config.h config.mak config.texi tcc.1 tcc-doc.html
|
||||||
|
|
||||||
|
endif # ifeq ($(TOP),.)
|
90
05/tcc-0.9.25/README
Normal file
90
05/tcc-0.9.25/README
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
Tiny C Compiler - C Scripting Everywhere - The Smallest ANSI C compiler
|
||||||
|
-----------------------------------------------------------------------
|
||||||
|
|
||||||
|
Features:
|
||||||
|
--------
|
||||||
|
|
||||||
|
- SMALL! You can compile and execute C code everywhere, for example on
|
||||||
|
rescue disks.
|
||||||
|
|
||||||
|
- FAST! tcc generates optimized x86 code. No byte code
|
||||||
|
overhead. Compile, assemble and link about 7 times faster than 'gcc
|
||||||
|
-O0'.
|
||||||
|
|
||||||
|
- UNLIMITED! Any C dynamic library can be used directly. TCC is
|
||||||
|
heading torward full ISOC99 compliance. TCC can of course compile
|
||||||
|
itself.
|
||||||
|
|
||||||
|
- SAFE! tcc includes an optional memory and bound checker. Bound
|
||||||
|
checked code can be mixed freely with standard code.
|
||||||
|
|
||||||
|
- Compile and execute C source directly. No linking or assembly
|
||||||
|
necessary. Full C preprocessor included.
|
||||||
|
|
||||||
|
- C script supported : just add '#!/usr/local/bin/tcc -run' at the first
|
||||||
|
line of your C source, and execute it directly from the command
|
||||||
|
line.
|
||||||
|
|
||||||
|
Documentation:
|
||||||
|
-------------
|
||||||
|
|
||||||
|
1) Installation on a i386 Linux host (for Windows read tcc-win32.txt)
|
||||||
|
|
||||||
|
./configure
|
||||||
|
make
|
||||||
|
make test
|
||||||
|
make install
|
||||||
|
|
||||||
|
By default, tcc is installed in /usr/local/bin.
|
||||||
|
./configure --help shows configuration options.
|
||||||
|
|
||||||
|
|
||||||
|
2) Introduction
|
||||||
|
|
||||||
|
We assume here that you know ANSI C. Look at the example ex1.c to know
|
||||||
|
what the programs look like.
|
||||||
|
|
||||||
|
The include file <tcclib.h> can be used if you want a small basic libc
|
||||||
|
include support (especially useful for floppy disks). Of course, you
|
||||||
|
can also use standard headers, although they are slower to compile.
|
||||||
|
|
||||||
|
You can begin your C script with '#!/usr/local/bin/tcc -run' on the first
|
||||||
|
line and set its execute bits (chmod a+x your_script). Then, you can
|
||||||
|
launch the C code as a shell or perl script :-) The command line
|
||||||
|
arguments are put in 'argc' and 'argv' of the main functions, as in
|
||||||
|
ANSI C.
|
||||||
|
|
||||||
|
3) Examples
|
||||||
|
|
||||||
|
ex1.c: simplest example (hello world). Can also be launched directly
|
||||||
|
as a script: './ex1.c'.
|
||||||
|
|
||||||
|
ex2.c: more complicated example: find a number with the four
|
||||||
|
operations given a list of numbers (benchmark).
|
||||||
|
|
||||||
|
ex3.c: compute fibonacci numbers (benchmark).
|
||||||
|
|
||||||
|
ex4.c: more complicated: X11 program. Very complicated test in fact
|
||||||
|
because standard headers are being used !
|
||||||
|
|
||||||
|
ex5.c: 'hello world' with standard glibc headers.
|
||||||
|
|
||||||
|
tcc.c: TCC can of course compile itself. Used to check the code
|
||||||
|
generator.
|
||||||
|
|
||||||
|
tcctest.c: auto test for TCC which tests many subtle possible bugs. Used
|
||||||
|
when doing 'make test'.
|
||||||
|
|
||||||
|
4) Full Documentation
|
||||||
|
|
||||||
|
Please read tcc-doc.html to have all the features of TCC.
|
||||||
|
|
||||||
|
Additional information is available for the Windows port in tcc-win32.txt.
|
||||||
|
|
||||||
|
License:
|
||||||
|
-------
|
||||||
|
|
||||||
|
TCC is distributed under the GNU Lesser General Public License (see
|
||||||
|
COPYING file).
|
||||||
|
|
||||||
|
Fabrice Bellard.
|
95
05/tcc-0.9.25/TODO
Normal file
95
05/tcc-0.9.25/TODO
Normal file
|
@ -0,0 +1,95 @@
|
||||||
|
TODO list:
|
||||||
|
|
||||||
|
Bugs:
|
||||||
|
|
||||||
|
- fix macro substitution with nested definitions (ShangHongzhang)
|
||||||
|
- FPU st(0) is left unclean (kwisatz haderach). Incompatible with
|
||||||
|
optimized gcc/msc code
|
||||||
|
|
||||||
|
- constructors
|
||||||
|
- cast bug (Peter Wang)
|
||||||
|
- define incomplete type if defined several times (Peter Wang).
|
||||||
|
- configure --cc=tcc (still one bug in libtcc1.c)
|
||||||
|
- test binutils/gcc compile
|
||||||
|
- tci patch + argument.
|
||||||
|
- see -lxxx bug (Michael Charity).
|
||||||
|
- see transparent union pb in /urs/include/sys/socket.h
|
||||||
|
- precise behaviour of typeof with arrays ? (__put_user macro)
|
||||||
|
but should suffice for most cases)
|
||||||
|
- handle '? x, y : z' in unsized variable initialization (',' is
|
||||||
|
considered incorrectly as separator in preparser)
|
||||||
|
- transform functions to function pointers in function parameters
|
||||||
|
(net/ipv4/ip_output.c)
|
||||||
|
- fix function pointer type display
|
||||||
|
- check lcc test suite -> fix bitfield binary operations
|
||||||
|
- check section alignment in C
|
||||||
|
- fix invalid cast in comparison 'if (v == (int8_t)v)'
|
||||||
|
- finish varargs.h support (gcc 3.2 testsuite issue)
|
||||||
|
- fix static functions declared inside block
|
||||||
|
- fix multiple unions init
|
||||||
|
- sizeof, alignof, typeof can still generate code in some cases.
|
||||||
|
- Fix the remaining libtcc memory leaks.
|
||||||
|
- make libtcc fully reentrant (except for the compilation stage itself).
|
||||||
|
|
||||||
|
Bound checking:
|
||||||
|
|
||||||
|
- '-b' bug.
|
||||||
|
- fix bound exit on RedHat 7.3
|
||||||
|
- setjmp is not supported properly in bound checking.
|
||||||
|
- fix bound check code with '&' on local variables (currently done
|
||||||
|
only for local arrays).
|
||||||
|
- bound checking and float/long long/struct copy code. bound
|
||||||
|
checking and symbol + offset optimization
|
||||||
|
|
||||||
|
Missing features:
|
||||||
|
|
||||||
|
- disable-asm and disable-bcheck options
|
||||||
|
- __builtin_expect()
|
||||||
|
- improve '-E' option.
|
||||||
|
- add '-MD' option
|
||||||
|
- atexit (Nigel Horne)
|
||||||
|
- packed attribute
|
||||||
|
- C99: add variable size arrays (gcc 3.2 testsuite issue)
|
||||||
|
- C99: add complex types (gcc 3.2 testsuite issue)
|
||||||
|
- postfix compound literals (see 20010124-1.c)
|
||||||
|
|
||||||
|
Optimizations:
|
||||||
|
|
||||||
|
- suppress specific anonymous symbol handling
|
||||||
|
- more parse optimizations (=even faster compilation)
|
||||||
|
- memory alloc optimizations (=even faster compilation)
|
||||||
|
- optimize VT_LOCAL + const
|
||||||
|
- better local variables handling (needed for other targets)
|
||||||
|
|
||||||
|
Not critical:
|
||||||
|
|
||||||
|
- C99: fix multiple compound literals inits in blocks (ISOC99
|
||||||
|
normative example - only relevant when using gotos! -> must add
|
||||||
|
boolean variable to tell if compound literal was already
|
||||||
|
initialized).
|
||||||
|
- add PowerPC or ARM code generator and improve codegen for RISC (need
|
||||||
|
to suppress VT_LOCAL and use a base register instead).
|
||||||
|
- interactive mode / integrated debugger
|
||||||
|
- fix preprocessor symbol redefinition
|
||||||
|
- better constant opt (&&, ||, ?:)
|
||||||
|
- add portable byte code generator and interpreter for other
|
||||||
|
unsupported architectures.
|
||||||
|
- C++: variable declaration in for, minimal 'class' support.
|
||||||
|
- win32: __intxx. use resolve for bchecked malloc et al.
|
||||||
|
check exception code (exception filter func).
|
||||||
|
- handle void (__attribute__() *ptr)()
|
||||||
|
|
||||||
|
Fixed (probably):
|
||||||
|
|
||||||
|
- bug with defines:
|
||||||
|
#define spin_lock(lock) do { } while (0)
|
||||||
|
#define wq_spin_lock spin_lock
|
||||||
|
#define TEST() wq_spin_lock(a)
|
||||||
|
- typedefs can be structure fields
|
||||||
|
- see bugfixes.diff + improvement.diff from Daniel Glockner
|
||||||
|
- long long constant evaluation
|
||||||
|
- add alloca()
|
||||||
|
- gcc '-E' option.
|
||||||
|
- #include_next support for /usr/include/limits ?
|
||||||
|
- function pointers/lvalues in ? : (linux kernel net/core/dev.c)
|
||||||
|
- win32: add __stdcall, check GetModuleHandle for dlls.
|
1
05/tcc-0.9.25/VERSION
Normal file
1
05/tcc-0.9.25/VERSION
Normal file
|
@ -0,0 +1 @@
|
||||||
|
0.9.25
|
1734
05/tcc-0.9.25/arm-gen.c
Normal file
1734
05/tcc-0.9.25/arm-gen.c
Normal file
File diff suppressed because it is too large
Load diff
7
05/tcc-0.9.25/assert.h
Normal file
7
05/tcc-0.9.25/assert.h
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
#ifndef _ASSERT_H
|
||||||
|
#define _ASSERT_H
|
||||||
|
|
||||||
|
// assert is defined in stdc_common.h
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
#endif // _ASSERT_H
|
2548
05/tcc-0.9.25/c67-gen.c
Normal file
2548
05/tcc-0.9.25/c67-gen.c
Normal file
File diff suppressed because it is too large
Load diff
446
05/tcc-0.9.25/coff.h
Normal file
446
05/tcc-0.9.25/coff.h
Normal file
|
@ -0,0 +1,446 @@
|
||||||
|
/**************************************************************************/
|
||||||
|
/* COFF.H */
|
||||||
|
/* COFF data structures and related definitions used by the linker */
|
||||||
|
/**************************************************************************/
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* COFF FILE HEADER */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
struct filehdr {
|
||||||
|
unsigned short f_magic; /* magic number */
|
||||||
|
unsigned short f_nscns; /* number of sections */
|
||||||
|
long f_timdat; /* time & date stamp */
|
||||||
|
long f_symptr; /* file pointer to symtab */
|
||||||
|
long f_nsyms; /* number of symtab entries */
|
||||||
|
unsigned short f_opthdr; /* sizeof(optional hdr) */
|
||||||
|
unsigned short f_flags; /* flags */
|
||||||
|
unsigned short f_TargetID; /* for C6x = 0x0099 */
|
||||||
|
};
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* File header flags */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define F_RELFLG 0x01 /* relocation info stripped from file */
|
||||||
|
#define F_EXEC 0x02 /* file is executable (no unresolved refs) */
|
||||||
|
#define F_LNNO 0x04 /* line nunbers stripped from file */
|
||||||
|
#define F_LSYMS 0x08 /* local symbols stripped from file */
|
||||||
|
#define F_GSP10 0x10 /* 34010 version */
|
||||||
|
#define F_GSP20 0x20 /* 34020 version */
|
||||||
|
#define F_SWABD 0x40 /* bytes swabbed (in names) */
|
||||||
|
#define F_AR16WR 0x80 /* byte ordering of an AR16WR (PDP-11) */
|
||||||
|
#define F_LITTLE 0x100 /* byte ordering of an AR32WR (vax) */
|
||||||
|
#define F_BIG 0x200 /* byte ordering of an AR32W (3B, maxi) */
|
||||||
|
#define F_PATCH 0x400 /* contains "patch" list in optional header */
|
||||||
|
#define F_NODF 0x400
|
||||||
|
|
||||||
|
#define F_VERSION (F_GSP10 | F_GSP20)
|
||||||
|
#define F_BYTE_ORDER (F_LITTLE | F_BIG)
|
||||||
|
#define FILHDR struct filehdr
|
||||||
|
|
||||||
|
//#define FILHSZ sizeof(FILHDR)
|
||||||
|
#define FILHSZ 22 // above rounds to align on 4 bytes which causes problems
|
||||||
|
|
||||||
|
#define COFF_C67_MAGIC 0x00c2
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* Macros to recognize magic numbers */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define ISMAGIC(x) (((unsigned short)(x))==(unsigned short)magic)
|
||||||
|
#define ISARCHIVE(x) ((((unsigned short)(x))==(unsigned short)ARTYPE))
|
||||||
|
#define BADMAGIC(x) (((unsigned short)(x) & 0x8080) && !ISMAGIC(x))
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* OPTIONAL FILE HEADER */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
typedef struct aouthdr {
|
||||||
|
short magic; /* see magic.h */
|
||||||
|
short vstamp; /* version stamp */
|
||||||
|
long tsize; /* text size in bytes, padded to FW bdry*/
|
||||||
|
long dsize; /* initialized data " " */
|
||||||
|
long bsize; /* uninitialized data " " */
|
||||||
|
long entrypt; /* entry pt. */
|
||||||
|
long text_start; /* base of text used for this file */
|
||||||
|
long data_start; /* base of data used for this file */
|
||||||
|
} AOUTHDR;
|
||||||
|
|
||||||
|
#define AOUTSZ sizeof(AOUTHDR)
|
||||||
|
|
||||||
|
/*----------------------------------------------------------------------*/
|
||||||
|
/* When a UNIX aout header is to be built in the optional header, */
|
||||||
|
/* the following magic numbers can appear in that header: */
|
||||||
|
/* */
|
||||||
|
/* AOUT1MAGIC : default : readonly sharable text segment */
|
||||||
|
/* AOUT2MAGIC: : writable text segment */
|
||||||
|
/* PAGEMAGIC : : configured for paging */
|
||||||
|
/*----------------------------------------------------------------------*/
|
||||||
|
#define AOUT1MAGIC 0410
|
||||||
|
#define AOUT2MAGIC 0407
|
||||||
|
#define PAGEMAGIC 0413
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* COMMON ARCHIVE FILE STRUCTURES */
|
||||||
|
/* */
|
||||||
|
/* ARCHIVE File Organization: */
|
||||||
|
/* _______________________________________________ */
|
||||||
|
/* |__________ARCHIVE_MAGIC_STRING_______________| */
|
||||||
|
/* |__________ARCHIVE_FILE_MEMBER_1______________| */
|
||||||
|
/* | | */
|
||||||
|
/* | Archive File Header "ar_hdr" | */
|
||||||
|
/* |.............................................| */
|
||||||
|
/* | Member Contents | */
|
||||||
|
/* | 1. External symbol directory | */
|
||||||
|
/* | 2. Text file | */
|
||||||
|
/* |_____________________________________________| */
|
||||||
|
/* |________ARCHIVE_FILE_MEMBER_2________________| */
|
||||||
|
/* | "ar_hdr" | */
|
||||||
|
/* |.............................................| */
|
||||||
|
/* | Member Contents (.o or text file) | */
|
||||||
|
/* |_____________________________________________| */
|
||||||
|
/* | . . . | */
|
||||||
|
/* | . . . | */
|
||||||
|
/* | . . . | */
|
||||||
|
/* |_____________________________________________| */
|
||||||
|
/* |________ARCHIVE_FILE_MEMBER_n________________| */
|
||||||
|
/* | "ar_hdr" | */
|
||||||
|
/* |.............................................| */
|
||||||
|
/* | Member Contents | */
|
||||||
|
/* |_____________________________________________| */
|
||||||
|
/* */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define COFF_ARMAG "!<arch>\n"
|
||||||
|
#define SARMAG 8
|
||||||
|
#define ARFMAG "`\n"
|
||||||
|
|
||||||
|
struct ar_hdr /* archive file member header - printable ascii */
|
||||||
|
{
|
||||||
|
char ar_name[16]; /* file member name - `/' terminated */
|
||||||
|
char ar_date[12]; /* file member date - decimal */
|
||||||
|
char ar_uid[6]; /* file member user id - decimal */
|
||||||
|
char ar_gid[6]; /* file member group id - decimal */
|
||||||
|
char ar_mode[8]; /* file member mode - octal */
|
||||||
|
char ar_size[10]; /* file member size - decimal */
|
||||||
|
char ar_fmag[2]; /* ARFMAG - string to end header */
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* SECTION HEADER */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
struct scnhdr {
|
||||||
|
char s_name[8]; /* section name */
|
||||||
|
long s_paddr; /* physical address */
|
||||||
|
long s_vaddr; /* virtual address */
|
||||||
|
long s_size; /* section size */
|
||||||
|
long s_scnptr; /* file ptr to raw data for section */
|
||||||
|
long s_relptr; /* file ptr to relocation */
|
||||||
|
long s_lnnoptr; /* file ptr to line numbers */
|
||||||
|
unsigned int s_nreloc; /* number of relocation entries */
|
||||||
|
unsigned int s_nlnno; /* number of line number entries */
|
||||||
|
unsigned int s_flags; /* flags */
|
||||||
|
unsigned short s_reserved; /* reserved byte */
|
||||||
|
unsigned short s_page; /* memory page id */
|
||||||
|
};
|
||||||
|
|
||||||
|
#define SCNHDR struct scnhdr
|
||||||
|
#define SCNHSZ sizeof(SCNHDR)
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* Define constants for names of "special" sections */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
//#define _TEXT ".text"
|
||||||
|
#define _DATA ".data"
|
||||||
|
#define _BSS ".bss"
|
||||||
|
#define _CINIT ".cinit"
|
||||||
|
#define _TV ".tv"
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* The low 4 bits of s_flags is used as a section "type" */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define STYP_REG 0x00 /* "regular" : allocated, relocated, loaded */
|
||||||
|
#define STYP_DSECT 0x01 /* "dummy" : not allocated, relocated, not loaded */
|
||||||
|
#define STYP_NOLOAD 0x02 /* "noload" : allocated, relocated, not loaded */
|
||||||
|
#define STYP_GROUP 0x04 /* "grouped" : formed of input sections */
|
||||||
|
#define STYP_PAD 0x08 /* "padding" : not allocated, not relocated, loaded */
|
||||||
|
#define STYP_COPY 0x10 /* "copy" : used for C init tables -
|
||||||
|
not allocated, relocated,
|
||||||
|
loaded; reloc & lineno
|
||||||
|
entries processed normally */
|
||||||
|
#define STYP_TEXT 0x20 /* section contains text only */
|
||||||
|
#define STYP_DATA 0x40 /* section contains data only */
|
||||||
|
#define STYP_BSS 0x80 /* section contains bss only */
|
||||||
|
|
||||||
|
#define STYP_ALIGN 0x100 /* align flag passed by old version assemblers */
|
||||||
|
#define ALIGN_MASK 0x0F00 /* part of s_flags that is used for align vals */
|
||||||
|
#define ALIGNSIZE(x) (1 << ((x & ALIGN_MASK) >> 8))
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* RELOCATION ENTRIES */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
struct reloc
|
||||||
|
{
|
||||||
|
long r_vaddr; /* (virtual) address of reference */
|
||||||
|
short r_symndx; /* index into symbol table */
|
||||||
|
unsigned short r_disp; /* additional bits for address calculation */
|
||||||
|
unsigned short r_type; /* relocation type */
|
||||||
|
};
|
||||||
|
|
||||||
|
#define RELOC struct reloc
|
||||||
|
#define RELSZ 10 /* sizeof(RELOC) */
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
/* define all relocation types */
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define R_ABS 0 /* absolute address - no relocation */
|
||||||
|
#define R_DIR16 01 /* UNUSED */
|
||||||
|
#define R_REL16 02 /* UNUSED */
|
||||||
|
#define R_DIR24 04 /* UNUSED */
|
||||||
|
#define R_REL24 05 /* 24 bits, direct */
|
||||||
|
#define R_DIR32 06 /* UNUSED */
|
||||||
|
#define R_RELBYTE 017 /* 8 bits, direct */
|
||||||
|
#define R_RELWORD 020 /* 16 bits, direct */
|
||||||
|
#define R_RELLONG 021 /* 32 bits, direct */
|
||||||
|
#define R_PCRBYTE 022 /* 8 bits, PC-relative */
|
||||||
|
#define R_PCRWORD 023 /* 16 bits, PC-relative */
|
||||||
|
#define R_PCRLONG 024 /* 32 bits, PC-relative */
|
||||||
|
#define R_OCRLONG 030 /* GSP: 32 bits, one's complement direct */
|
||||||
|
#define R_GSPPCR16 031 /* GSP: 16 bits, PC relative (in words) */
|
||||||
|
#define R_GSPOPR32 032 /* GSP: 32 bits, direct big-endian */
|
||||||
|
#define R_PARTLS16 040 /* Brahma: 16 bit offset of 24 bit address*/
|
||||||
|
#define R_PARTMS8 041 /* Brahma: 8 bit page of 24 bit address */
|
||||||
|
#define R_PARTLS7 050 /* DSP: 7 bit offset of 16 bit address */
|
||||||
|
#define R_PARTMS9 051 /* DSP: 9 bit page of 16 bit address */
|
||||||
|
#define R_REL13 052 /* DSP: 13 bits, direct */
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* LINE NUMBER ENTRIES */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
struct lineno
|
||||||
|
{
|
||||||
|
union
|
||||||
|
{
|
||||||
|
long l_symndx ; /* sym. table index of function name
|
||||||
|
iff l_lnno == 0 */
|
||||||
|
long l_paddr ; /* (physical) address of line number */
|
||||||
|
} l_addr ;
|
||||||
|
unsigned short l_lnno ; /* line number */
|
||||||
|
};
|
||||||
|
|
||||||
|
#define LINENO struct lineno
|
||||||
|
#define LINESZ 6 /* sizeof(LINENO) */
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* STORAGE CLASSES */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define C_EFCN -1 /* physical end of function */
|
||||||
|
#define C_NULL 0
|
||||||
|
#define C_AUTO 1 /* automatic variable */
|
||||||
|
#define C_EXT 2 /* external symbol */
|
||||||
|
#define C_STAT 3 /* static */
|
||||||
|
#define C_REG 4 /* register variable */
|
||||||
|
#define C_EXTDEF 5 /* external definition */
|
||||||
|
#define C_LABEL 6 /* label */
|
||||||
|
#define C_ULABEL 7 /* undefined label */
|
||||||
|
#define C_MOS 8 /* member of structure */
|
||||||
|
#define C_ARG 9 /* function argument */
|
||||||
|
#define C_STRTAG 10 /* structure tag */
|
||||||
|
#define C_MOU 11 /* member of union */
|
||||||
|
#define C_UNTAG 12 /* union tag */
|
||||||
|
#define C_TPDEF 13 /* type definition */
|
||||||
|
#define C_USTATIC 14 /* undefined static */
|
||||||
|
#define C_ENTAG 15 /* enumeration tag */
|
||||||
|
#define C_MOE 16 /* member of enumeration */
|
||||||
|
#define C_REGPARM 17 /* register parameter */
|
||||||
|
#define C_FIELD 18 /* bit field */
|
||||||
|
|
||||||
|
#define C_BLOCK 100 /* ".bb" or ".eb" */
|
||||||
|
#define C_FCN 101 /* ".bf" or ".ef" */
|
||||||
|
#define C_EOS 102 /* end of structure */
|
||||||
|
#define C_FILE 103 /* file name */
|
||||||
|
#define C_LINE 104 /* dummy sclass for line number entry */
|
||||||
|
#define C_ALIAS 105 /* duplicate tag */
|
||||||
|
#define C_HIDDEN 106 /* special storage class for external */
|
||||||
|
/* symbols in dmert public libraries */
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* SYMBOL TABLE ENTRIES */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#define SYMNMLEN 8 /* Number of characters in a symbol name */
|
||||||
|
#define FILNMLEN 14 /* Number of characters in a file name */
|
||||||
|
#define DIMNUM 4 /* Number of array dimensions in auxiliary entry */
|
||||||
|
|
||||||
|
|
||||||
|
struct syment
|
||||||
|
{
|
||||||
|
union
|
||||||
|
{
|
||||||
|
char _n_name[SYMNMLEN]; /* old COFF version */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
long _n_zeroes; /* new == 0 */
|
||||||
|
long _n_offset; /* offset into string table */
|
||||||
|
} _n_n;
|
||||||
|
char *_n_nptr[2]; /* allows for overlaying */
|
||||||
|
} _n;
|
||||||
|
long n_value; /* value of symbol */
|
||||||
|
short n_scnum; /* section number */
|
||||||
|
unsigned short n_type; /* type and derived type */
|
||||||
|
char n_sclass; /* storage class */
|
||||||
|
char n_numaux; /* number of aux. entries */
|
||||||
|
};
|
||||||
|
|
||||||
|
#define n_name _n._n_name
|
||||||
|
#define n_nptr _n._n_nptr[1]
|
||||||
|
#define n_zeroes _n._n_n._n_zeroes
|
||||||
|
#define n_offset _n._n_n._n_offset
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* Relocatable symbols have a section number of the */
|
||||||
|
/* section in which they are defined. Otherwise, section */
|
||||||
|
/* numbers have the following meanings: */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define N_UNDEF 0 /* undefined symbol */
|
||||||
|
#define N_ABS -1 /* value of symbol is absolute */
|
||||||
|
#define N_DEBUG -2 /* special debugging symbol */
|
||||||
|
#define N_TV (unsigned short)-3 /* needs transfer vector (preload) */
|
||||||
|
#define P_TV (unsigned short)-4 /* needs transfer vector (postload) */
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* The fundamental type of a symbol packed into the low */
|
||||||
|
/* 4 bits of the word. */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define _EF ".ef"
|
||||||
|
|
||||||
|
#define T_NULL 0 /* no type info */
|
||||||
|
#define T_ARG 1 /* function argument (only used by compiler) */
|
||||||
|
#define T_CHAR 2 /* character */
|
||||||
|
#define T_SHORT 3 /* short integer */
|
||||||
|
#define T_INT 4 /* integer */
|
||||||
|
#define T_LONG 5 /* long integer */
|
||||||
|
#define T_FLOAT 6 /* floating point */
|
||||||
|
#define T_DOUBLE 7 /* double word */
|
||||||
|
#define T_STRUCT 8 /* structure */
|
||||||
|
#define T_UNION 9 /* union */
|
||||||
|
#define T_ENUM 10 /* enumeration */
|
||||||
|
#define T_MOE 11 /* member of enumeration */
|
||||||
|
#define T_UCHAR 12 /* unsigned character */
|
||||||
|
#define T_USHORT 13 /* unsigned short */
|
||||||
|
#define T_UINT 14 /* unsigned integer */
|
||||||
|
#define T_ULONG 15 /* unsigned long */
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* derived types are: */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define DT_NON 0 /* no derived type */
|
||||||
|
#define DT_PTR 1 /* pointer */
|
||||||
|
#define DT_FCN 2 /* function */
|
||||||
|
#define DT_ARY 3 /* array */
|
||||||
|
|
||||||
|
#define MKTYPE(basic, d1,d2,d3,d4,d5,d6) \
|
||||||
|
((basic) | ((d1) << 4) | ((d2) << 6) | ((d3) << 8) |\
|
||||||
|
((d4) << 10) | ((d5) << 12) | ((d6) << 14))
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* type packing constants and macros */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define N_BTMASK_COFF 017
|
||||||
|
#define N_TMASK_COFF 060
|
||||||
|
#define N_TMASK1_COFF 0300
|
||||||
|
#define N_TMASK2_COFF 0360
|
||||||
|
#define N_BTSHFT_COFF 4
|
||||||
|
#define N_TSHIFT_COFF 2
|
||||||
|
|
||||||
|
#define BTYPE_COFF(x) ((x) & N_BTMASK_COFF)
|
||||||
|
#define ISINT(x) (((x) >= T_CHAR && (x) <= T_LONG) || \
|
||||||
|
((x) >= T_UCHAR && (x) <= T_ULONG) || (x) == T_ENUM)
|
||||||
|
#define ISFLT_COFF(x) ((x) == T_DOUBLE || (x) == T_FLOAT)
|
||||||
|
#define ISPTR_COFF(x) (((x) & N_TMASK_COFF) == (DT_PTR << N_BTSHFT_COFF))
|
||||||
|
#define ISFCN_COFF(x) (((x) & N_TMASK_COFF) == (DT_FCN << N_BTSHFT_COFF))
|
||||||
|
#define ISARY_COFF(x) (((x) & N_TMASK_COFF) == (DT_ARY << N_BTSHFT_COFF))
|
||||||
|
#define ISTAG_COFF(x) ((x)==C_STRTAG || (x)==C_UNTAG || (x)==C_ENTAG)
|
||||||
|
|
||||||
|
#define INCREF_COFF(x) ((((x)&~N_BTMASK_COFF)<<N_TSHIFT_COFF)|(DT_PTR<<N_BTSHFT_COFF)|(x&N_BTMASK_COFF))
|
||||||
|
#define DECREF_COFF(x) ((((x)>>N_TSHIFT_COFF)&~N_BTMASK_COFF)|((x)&N_BTMASK_COFF))
|
||||||
|
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* AUXILIARY SYMBOL ENTRY */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
union auxent
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
long x_tagndx; /* str, un, or enum tag indx */
|
||||||
|
union
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
unsigned short x_lnno; /* declaration line number */
|
||||||
|
unsigned short x_size; /* str, union, array size */
|
||||||
|
} x_lnsz;
|
||||||
|
long x_fsize; /* size of function */
|
||||||
|
} x_misc;
|
||||||
|
union
|
||||||
|
{
|
||||||
|
struct /* if ISFCN, tag, or .bb */
|
||||||
|
{
|
||||||
|
long x_lnnoptr; /* ptr to fcn line # */
|
||||||
|
long x_endndx; /* entry ndx past block end */
|
||||||
|
} x_fcn;
|
||||||
|
struct /* if ISARY, up to 4 dimen. */
|
||||||
|
{
|
||||||
|
unsigned short x_dimen[DIMNUM];
|
||||||
|
} x_ary;
|
||||||
|
} x_fcnary;
|
||||||
|
unsigned short x_regcount; /* number of registers used by func */
|
||||||
|
} x_sym;
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
char x_fname[FILNMLEN];
|
||||||
|
} x_file;
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
long x_scnlen; /* section length */
|
||||||
|
unsigned short x_nreloc; /* number of relocation entries */
|
||||||
|
unsigned short x_nlinno; /* number of line numbers */
|
||||||
|
} x_scn;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define SYMENT struct syment
|
||||||
|
#define SYMESZ 18 /* sizeof(SYMENT) */
|
||||||
|
|
||||||
|
#define AUXENT union auxent
|
||||||
|
#define AUXESZ 18 /* sizeof(AUXENT) */
|
||||||
|
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
/* NAMES OF "SPECIAL" SYMBOLS */
|
||||||
|
/*------------------------------------------------------------------------*/
|
||||||
|
#define _STEXT ".text"
|
||||||
|
#define _ETEXT "etext"
|
||||||
|
#define _SDATA ".data"
|
||||||
|
#define _EDATA "edata"
|
||||||
|
#define _SBSS ".bss"
|
||||||
|
#define _END "end"
|
||||||
|
#define _CINITPTR "cinit"
|
||||||
|
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
/* ENTRY POINT SYMBOLS */
|
||||||
|
/*--------------------------------------------------------------------------*/
|
||||||
|
#define _START "_start"
|
||||||
|
#define _MAIN "_main"
|
||||||
|
/* _CSTART "_c_int00" (defined in params.h) */
|
||||||
|
|
||||||
|
|
||||||
|
#define _TVORIG "_tvorig"
|
||||||
|
#define _TORIGIN "_torigin"
|
||||||
|
#define _DORIGIN "_dorigin"
|
||||||
|
|
||||||
|
#define _SORIGIN "_sorigin"
|
10
05/tcc-0.9.25/config.h
Normal file
10
05/tcc-0.9.25/config.h
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#ifndef CONFIG_TCCDIR
|
||||||
|
# define CONFIG_TCCDIR "/usr/local/lib/tcc-bootstrap"
|
||||||
|
#endif
|
||||||
|
#define TCC_VERSION "0.9.27"
|
||||||
|
#define CONFIG_TCC_STATIC 1
|
||||||
|
#define TCC_TARGET_X86_64 1
|
||||||
|
#define CONFIG_TCC_ELFINTERP "/XXX"
|
||||||
|
#define CONFIG_TCC_CRT_PREFIX "/XXX"
|
||||||
|
#define CONFIG_SYSROOT "/XXX"
|
||||||
|
#define inline
|
382
05/tcc-0.9.25/configure
vendored
Executable file
382
05/tcc-0.9.25/configure
vendored
Executable file
|
@ -0,0 +1,382 @@
|
||||||
|
#!/bin/sh
|
||||||
|
#
|
||||||
|
# tcc configure script (c) 2003 Fabrice Bellard
|
||||||
|
#
|
||||||
|
# set temporary file name
|
||||||
|
if test ! -z "$TMPDIR" ; then
|
||||||
|
TMPDIR1="${TMPDIR}"
|
||||||
|
elif test ! -z "$TEMPDIR" ; then
|
||||||
|
TMPDIR1="${TEMPDIR}"
|
||||||
|
else
|
||||||
|
TMPDIR1="/tmp"
|
||||||
|
fi
|
||||||
|
|
||||||
|
TMPC="${TMPDIR1}/tcc-conf-${RANDOM}-$$-${RANDOM}.c"
|
||||||
|
TMPO="${TMPDIR1}/tcc-conf-${RANDOM}-$$-${RANDOM}.o"
|
||||||
|
TMPE="${TMPDIR1}/tcc-conf-${RANDOM}-$$-${RANDOM}"
|
||||||
|
TMPS="${TMPDIR1}/tcc-conf-${RANDOM}-$$-${RANDOM}.S"
|
||||||
|
TMPH="${TMPDIR1}/tcc-conf-${RANDOM}-$$-${RANDOM}.h"
|
||||||
|
|
||||||
|
# default parameters
|
||||||
|
build_cross="no"
|
||||||
|
use_libgcc="no"
|
||||||
|
prefix=""
|
||||||
|
execprefix=""
|
||||||
|
bindir=""
|
||||||
|
libdir=""
|
||||||
|
tccdir=""
|
||||||
|
includedir=""
|
||||||
|
mandir=""
|
||||||
|
sysroot=""
|
||||||
|
cross_prefix=""
|
||||||
|
cc="gcc"
|
||||||
|
host_cc="gcc"
|
||||||
|
ar="ar"
|
||||||
|
strip="strip"
|
||||||
|
cpu=`uname -m`
|
||||||
|
case "$cpu" in
|
||||||
|
i386|i486|i586|i686|i86pc|BePC)
|
||||||
|
cpu="x86"
|
||||||
|
;;
|
||||||
|
x86_64)
|
||||||
|
cpu="x86-64"
|
||||||
|
;;
|
||||||
|
armv4l)
|
||||||
|
cpu="armv4l"
|
||||||
|
;;
|
||||||
|
alpha)
|
||||||
|
cpu="alpha"
|
||||||
|
;;
|
||||||
|
"Power Macintosh"|ppc|ppc64)
|
||||||
|
cpu="powerpc"
|
||||||
|
;;
|
||||||
|
mips)
|
||||||
|
cpu="mips"
|
||||||
|
;;
|
||||||
|
s390)
|
||||||
|
cpu="s390"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
cpu="unknown"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
gprof="no"
|
||||||
|
bigendian="no"
|
||||||
|
mingw32="no"
|
||||||
|
LIBSUF=".a"
|
||||||
|
EXESUF=""
|
||||||
|
|
||||||
|
# OS specific
|
||||||
|
targetos=`uname -s`
|
||||||
|
case $targetos in
|
||||||
|
MINGW32*)
|
||||||
|
mingw32="yes"
|
||||||
|
;;
|
||||||
|
DragonFly)
|
||||||
|
noldl="yes"
|
||||||
|
;;
|
||||||
|
OpenBSD)
|
||||||
|
noldl="yes"
|
||||||
|
;;
|
||||||
|
*) ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# find source path
|
||||||
|
# XXX: we assume an absolute path is given when launching configure,
|
||||||
|
# except in './configure' case.
|
||||||
|
source_path=${0%configure}
|
||||||
|
source_path=${source_path%/}
|
||||||
|
source_path_used="yes"
|
||||||
|
if test -z "$source_path" -o "$source_path" = "." ; then
|
||||||
|
source_path=`pwd`
|
||||||
|
source_path_used="no"
|
||||||
|
fi
|
||||||
|
|
||||||
|
for opt do
|
||||||
|
case "$opt" in
|
||||||
|
--prefix=*) prefix=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--exec-prefix=*) execprefix=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--bindir=*) bindir=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--libdir=*) libdir=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--includedir=*) includedir=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--mandir=*) mandir=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--sysroot=*) sysroot=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--source-path=*) source_path=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--cross-prefix=*) cross_prefix=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--cc=*) cc=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--extra-cflags=*) CFLAGS="${opt#--extra-cflags=}"
|
||||||
|
;;
|
||||||
|
--extra-ldflags=*) LDFLAGS="${opt#--extra-ldflags=}"
|
||||||
|
;;
|
||||||
|
--extra-libs=*) extralibs=${opt#--extra-libs=}
|
||||||
|
;;
|
||||||
|
--cpu=*) cpu=`echo $opt | cut -d '=' -f 2`
|
||||||
|
;;
|
||||||
|
--enable-gprof) gprof="yes"
|
||||||
|
;;
|
||||||
|
--enable-mingw32) mingw32="yes" ; cross_prefix="i386-mingw32-"
|
||||||
|
;;
|
||||||
|
--enable-cross) build_cross="yes"
|
||||||
|
;;
|
||||||
|
--with-libgcc) use_libgcc="yes"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
# Checking for CFLAGS
|
||||||
|
if test -z "$CFLAGS"; then
|
||||||
|
CFLAGS="-O2"
|
||||||
|
fi
|
||||||
|
|
||||||
|
cc="${cross_prefix}${cc}"
|
||||||
|
ar="${cross_prefix}${ar}"
|
||||||
|
strip="${cross_prefix}${strip}"
|
||||||
|
|
||||||
|
if test "$mingw32" = "yes" ; then
|
||||||
|
LIBSUF=".lib"
|
||||||
|
EXESUF=".exe"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test -z "$cross_prefix" ; then
|
||||||
|
|
||||||
|
# ---
|
||||||
|
# big/little endian test
|
||||||
|
cat > $TMPC << EOF
|
||||||
|
#include <inttypes.h>
|
||||||
|
int main(int argc, char ** argv){
|
||||||
|
volatile uint32_t i=0x01234567;
|
||||||
|
return (*((uint8_t*)(&i))) == 0x67;
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
|
||||||
|
if $cc -o $TMPE $TMPC 2>/dev/null ; then
|
||||||
|
$TMPE && bigendian="yes"
|
||||||
|
else
|
||||||
|
echo big/little test failed
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
# if cross compiling, cannot launch a program, so make a static guess
|
||||||
|
if test "$cpu" = "powerpc" -o "$cpu" = "mips" -o "$cpu" = "s390" ; then
|
||||||
|
bigendian="yes"
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
# check gcc version
|
||||||
|
cat > $TMPC <<EOF
|
||||||
|
int main(void) {
|
||||||
|
#if __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 2)
|
||||||
|
return 0;
|
||||||
|
#else
|
||||||
|
#error gcc < 3.2
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
|
||||||
|
gcc_major="2"
|
||||||
|
if $cc -o $TMPO $TMPC 2> /dev/null ; then
|
||||||
|
gcc_major="3"
|
||||||
|
fi
|
||||||
|
cat > $TMPC <<EOF
|
||||||
|
int main(void) {
|
||||||
|
#if __GNUC__ >= 4
|
||||||
|
return 0;
|
||||||
|
#else
|
||||||
|
#error gcc < 4
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
|
||||||
|
if $cc -o $TMPO $TMPC 2> /dev/null ; then
|
||||||
|
gcc_major="4"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test x"$1" = x"-h" -o x"$1" = x"--help" ; then
|
||||||
|
cat << EOF
|
||||||
|
|
||||||
|
Usage: configure [options]
|
||||||
|
Options: [defaults in brackets after descriptions]
|
||||||
|
|
||||||
|
EOF
|
||||||
|
echo "Standard options:"
|
||||||
|
echo " --help print this message"
|
||||||
|
echo " --prefix=PREFIX install in PREFIX [$prefix]"
|
||||||
|
echo " --exec-prefix=EPREFIX install architecture-dependent files in EPREFIX"
|
||||||
|
echo " [same as prefix]"
|
||||||
|
echo " --bindir=DIR user executables in DIR [EPREFIX/bin]"
|
||||||
|
echo " --libdir=DIR object code libraries in DIR [EPREFIX/lib]"
|
||||||
|
echo " --includedir=DIR C header files in DIR [PREFIX/include]"
|
||||||
|
echo " --mandir=DIR man documentation in DIR [PREFIX/man]"
|
||||||
|
echo " --enable-cross build cross compilers"
|
||||||
|
echo ""
|
||||||
|
echo "Advanced options (experts only):"
|
||||||
|
echo " --source-path=PATH path of source code [$source_path]"
|
||||||
|
echo " --cross-prefix=PREFIX use PREFIX for compile tools [$cross_prefix]"
|
||||||
|
echo " --sysroot=PREFIX prepend PREFIX to library/include paths []"
|
||||||
|
echo " --cc=CC use C compiler CC [$cc]"
|
||||||
|
echo " --with-libgcc use /lib/libgcc_s.so.1 instead of libtcc1.a"
|
||||||
|
echo ""
|
||||||
|
#echo "NOTE: The object files are build at the place where configure is launched"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test "$mingw32" = "yes" ; then
|
||||||
|
if test -z "$prefix" ; then
|
||||||
|
prefix="C:/Program Files/tcc"
|
||||||
|
fi
|
||||||
|
execprefix="$prefix"
|
||||||
|
bindir="$prefix"
|
||||||
|
tccdir="$prefix"
|
||||||
|
docdir="$prefix/doc"
|
||||||
|
else
|
||||||
|
if test -z "$prefix" ; then
|
||||||
|
prefix="/usr/local"
|
||||||
|
fi
|
||||||
|
if test x"$execprefix" = x""; then
|
||||||
|
execprefix="${prefix}"
|
||||||
|
fi
|
||||||
|
if test x"$bindir" = x""; then
|
||||||
|
bindir="${execprefix}/bin"
|
||||||
|
fi
|
||||||
|
if test x"$docdir" = x""; then
|
||||||
|
docdir="$prefix/share/doc/tcc"
|
||||||
|
fi
|
||||||
|
fi # mingw32
|
||||||
|
|
||||||
|
if test x"$libdir" = x""; then
|
||||||
|
libdir="${execprefix}/lib"
|
||||||
|
fi
|
||||||
|
if test x"$tccdir" = x""; then
|
||||||
|
tccdir="${execprefix}/lib/tcc"
|
||||||
|
fi
|
||||||
|
if test x"$mandir" = x""; then
|
||||||
|
mandir="${prefix}/man"
|
||||||
|
fi
|
||||||
|
if test x"$includedir" = x""; then
|
||||||
|
includedir="${prefix}/include"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Binary directory $bindir"
|
||||||
|
echo "TinyCC directory $tccdir"
|
||||||
|
echo "Library directory $libdir"
|
||||||
|
echo "Include directory $includedir"
|
||||||
|
echo "Manual directory $mandir"
|
||||||
|
echo "Doc directory $docdir"
|
||||||
|
echo "Target root prefix $sysroot"
|
||||||
|
echo "Source path $source_path"
|
||||||
|
echo "C compiler $cc"
|
||||||
|
echo "CPU $cpu"
|
||||||
|
echo "Big Endian $bigendian"
|
||||||
|
echo "gprof enabled $gprof"
|
||||||
|
echo "cross compilers $build_cross"
|
||||||
|
echo "use libgcc $use_libgcc"
|
||||||
|
|
||||||
|
echo "Creating config.mak and config.h"
|
||||||
|
|
||||||
|
echo "# Automatically generated by configure - do not modify" > config.mak
|
||||||
|
echo "/* Automatically generated by configure - do not modify */" > $TMPH
|
||||||
|
|
||||||
|
echo "prefix=$prefix" >> config.mak
|
||||||
|
echo "bindir=$bindir" >> config.mak
|
||||||
|
echo "tccdir=$tccdir" >> config.mak
|
||||||
|
echo "libdir=$libdir" >> config.mak
|
||||||
|
echo "includedir=$includedir" >> config.mak
|
||||||
|
echo "mandir=$mandir" >> config.mak
|
||||||
|
echo "docdir=$docdir" >> config.mak
|
||||||
|
echo "#define CONFIG_SYSROOT \"$sysroot\"" >> $TMPH
|
||||||
|
echo "#define CONFIG_TCCDIR \"$tccdir\"" >> $TMPH
|
||||||
|
echo "CC=$cc" >> config.mak
|
||||||
|
echo "GCC_MAJOR=$gcc_major" >> config.mak
|
||||||
|
echo "#define GCC_MAJOR $gcc_major" >> $TMPH
|
||||||
|
echo "HOST_CC=$host_cc" >> config.mak
|
||||||
|
echo "AR=$ar" >> config.mak
|
||||||
|
echo "STRIP=$strip -s -R .comment -R .note" >> config.mak
|
||||||
|
echo "CFLAGS=$CFLAGS" >> config.mak
|
||||||
|
echo "LDFLAGS=$LDFLAGS" >> config.mak
|
||||||
|
echo "LIBSUF=$LIBSUF" >> config.mak
|
||||||
|
echo "EXESUF=$EXESUF" >> config.mak
|
||||||
|
if test "$cpu" = "x86" ; then
|
||||||
|
echo "ARCH=i386" >> config.mak
|
||||||
|
echo "#define HOST_I386 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "x86-64" ; then
|
||||||
|
echo "ARCH=x86-64" >> config.mak
|
||||||
|
echo "#define HOST_X86_64 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "armv4l" ; then
|
||||||
|
echo "ARCH=arm" >> config.mak
|
||||||
|
echo "#define HOST_ARM 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "powerpc" ; then
|
||||||
|
echo "ARCH=ppc" >> config.mak
|
||||||
|
echo "#define HOST_PPC 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "mips" ; then
|
||||||
|
echo "ARCH=mips" >> config.mak
|
||||||
|
echo "#define HOST_MIPS 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "s390" ; then
|
||||||
|
echo "ARCH=s390" >> config.mak
|
||||||
|
echo "#define HOST_S390 1" >> $TMPH
|
||||||
|
elif test "$cpu" = "alpha" ; then
|
||||||
|
echo "ARCH=alpha" >> config.mak
|
||||||
|
echo "#define HOST_ALPHA 1" >> $TMPH
|
||||||
|
else
|
||||||
|
echo "Unsupported CPU"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
if test "$noldl" = "yes" ; then
|
||||||
|
echo "CONFIG_NOLDL=yes" >> config.mak
|
||||||
|
fi
|
||||||
|
if test "$mingw32" = "yes" ; then
|
||||||
|
echo "CONFIG_WIN32=yes" >> config.mak
|
||||||
|
echo "#define CONFIG_WIN32 1" >> $TMPH
|
||||||
|
fi
|
||||||
|
if test "$bigendian" = "yes" ; then
|
||||||
|
echo "WORDS_BIGENDIAN=yes" >> config.mak
|
||||||
|
echo "#define WORDS_BIGENDIAN 1" >> $TMPH
|
||||||
|
fi
|
||||||
|
if test "$gprof" = "yes" ; then
|
||||||
|
echo "TARGET_GPROF=yes" >> config.mak
|
||||||
|
echo "#define HAVE_GPROF 1" >> $TMPH
|
||||||
|
fi
|
||||||
|
if test "$build_cross" = "yes" ; then
|
||||||
|
echo "CONFIG_CROSS=yes" >> config.mak
|
||||||
|
fi
|
||||||
|
if test "$use_libgcc" = "yes" ; then
|
||||||
|
echo "#define CONFIG_USE_LIBGCC" >> $TMPH
|
||||||
|
echo "CONFIG_USE_LIBGCC=yes" >> config.mak
|
||||||
|
fi
|
||||||
|
version=`head $source_path/VERSION`
|
||||||
|
echo "VERSION=$version" >>config.mak
|
||||||
|
echo "#define TCC_VERSION \"$version\"" >> $TMPH
|
||||||
|
echo "@set VERSION $version" > config.texi
|
||||||
|
|
||||||
|
# build tree in object directory if source path is different from current one
|
||||||
|
if test "$source_path_used" = "yes" ; then
|
||||||
|
DIRS="tests"
|
||||||
|
FILES="Makefile tests/Makefile"
|
||||||
|
for dir in $DIRS ; do
|
||||||
|
mkdir -p $dir
|
||||||
|
done
|
||||||
|
for f in $FILES ; do
|
||||||
|
ln -sf $source_path/$f $f
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
echo "SRC_PATH=$source_path" >> config.mak
|
||||||
|
|
||||||
|
diff $TMPH config.h >/dev/null 2>&1
|
||||||
|
if test $? -ne 0 ; then
|
||||||
|
mv -f $TMPH config.h
|
||||||
|
else
|
||||||
|
echo "config.h is unchanged"
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -f $TMPO $TMPC $TMPE $TMPS $TMPH
|
92
05/tcc-0.9.25/ctype.h
Normal file
92
05/tcc-0.9.25/ctype.h
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
#ifndef _CTYPE_H
|
||||||
|
#define _CTYPE_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
int islower(int c) {
|
||||||
|
return c >= 'a' && c <= 'z';
|
||||||
|
}
|
||||||
|
|
||||||
|
int isupper(int c) {
|
||||||
|
return c >= 'A' && c <= 'Z';
|
||||||
|
}
|
||||||
|
|
||||||
|
int isalpha(int c) {
|
||||||
|
return isupper(c) || islower(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
int isalnum(int c) {
|
||||||
|
return isalpha(c) || isdigit(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
int isprint(int c) {
|
||||||
|
if (isalnum(c)) return 1;
|
||||||
|
switch (c) {
|
||||||
|
case '!': return 1;
|
||||||
|
case '@': return 1;
|
||||||
|
case '#': return 1;
|
||||||
|
case '$': return 1;
|
||||||
|
case '%': return 1;
|
||||||
|
case '^': return 1;
|
||||||
|
case '&': return 1;
|
||||||
|
case '*': return 1;
|
||||||
|
case '(': return 1;
|
||||||
|
case ')': return 1;
|
||||||
|
case '-': return 1;
|
||||||
|
case '=': return 1;
|
||||||
|
case '_': return 1;
|
||||||
|
case '+': return 1;
|
||||||
|
case '`': return 1;
|
||||||
|
case '~': return 1;
|
||||||
|
case '[': return 1;
|
||||||
|
case '{': return 1;
|
||||||
|
case ']': return 1;
|
||||||
|
case '}': return 1;
|
||||||
|
case '\\': return 1;
|
||||||
|
case '|': return 1;
|
||||||
|
case ';': return 1;
|
||||||
|
case ':': return 1;
|
||||||
|
case '\'': return 1;
|
||||||
|
case '"': return 1;
|
||||||
|
case ',': return 1;
|
||||||
|
case '<': return 1;
|
||||||
|
case '.': return 1;
|
||||||
|
case '>': return 1;
|
||||||
|
case '/': return 1;
|
||||||
|
case '?': return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int iscntrl(int c) {
|
||||||
|
return !isprint(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
int isgraph(int c) {
|
||||||
|
return isprint(c) && c != ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
int ispunct(int c) {
|
||||||
|
return isprint(c) && c != ' ' && !isalnum(c);
|
||||||
|
}
|
||||||
|
|
||||||
|
int isxdigit(int c) {
|
||||||
|
if (isdigit(c)) return 1;
|
||||||
|
if (c >= 'a' && c <= 'f') return 1;
|
||||||
|
if (c >= 'A' && c <= 'F') return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int tolower(int c) {
|
||||||
|
if (c >= 'A' && c <= 'Z')
|
||||||
|
return c - 'A' + 'a';
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
int toupper(int c) {
|
||||||
|
if (c >= 'a' && c <= 'z')
|
||||||
|
return c - 'a' + 'A';
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // _CTYPE_H
|
1714
05/tcc-0.9.25/elf.h
Normal file
1714
05/tcc-0.9.25/elf.h
Normal file
File diff suppressed because it is too large
Load diff
6
05/tcc-0.9.25/errno.h
Normal file
6
05/tcc-0.9.25/errno.h
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
#ifndef _ERRNO_H
|
||||||
|
#define _ERRNO_H
|
||||||
|
|
||||||
|
#include <stdc_common.h> // we define all the relevant things here
|
||||||
|
|
||||||
|
#endif // _ERRNO_H
|
34
05/tcc-0.9.25/float.h
Normal file
34
05/tcc-0.9.25/float.h
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
#ifndef _FLOAT_H
|
||||||
|
#define _FLOAT_H
|
||||||
|
|
||||||
|
#define DBL_DIG 15
|
||||||
|
#define DBL_EPSILON 2.2204460492503131e-16
|
||||||
|
#define DBL_MANT_DIG 53
|
||||||
|
#define DBL_MAX 1.7976931348623157e+308
|
||||||
|
#define DBL_MAX_10_EXP 308
|
||||||
|
#define DBL_MAX_EXP 1024
|
||||||
|
#define DBL_MIN 2.2250738585072027e-308
|
||||||
|
#define DBL_MIN_10_EXP (-307)
|
||||||
|
#define DBL_MIN_EXP (-1021)
|
||||||
|
#define FLT_DIG 6
|
||||||
|
#define FLT_EPSILON 1.19209290e-07
|
||||||
|
#define FLT_MANT_DIG 24
|
||||||
|
#define FLT_MAX 3.40282347e+38
|
||||||
|
#define FLT_MAX_10_EXP +38
|
||||||
|
#define FLT_MAX_EXP 128
|
||||||
|
#define FLT_MIN 1.17549435e-38
|
||||||
|
#define FLT_MIN_10_EXP (-37)
|
||||||
|
#define FLT_MIN_EXP (-125)
|
||||||
|
#define FLT_RADIX 2
|
||||||
|
#define FLT_ROUNDS 1
|
||||||
|
#define LDBL_DIG DBL_DIG
|
||||||
|
#define LDBL_EPSILON DBL_EPSILON
|
||||||
|
#define LDBL_MANT_DIG DBL_MANT_DIG
|
||||||
|
#define LDBL_MAX DBL_MAX
|
||||||
|
#define LDBL_MAX_10_EXP DBL_MAX_10_EXP
|
||||||
|
#define LDBL_MAX_EXP DBL_MAX_EXP
|
||||||
|
#define LDBL_MIN DBL_MIN
|
||||||
|
#define LDBL_MIN_10_EXP DBL_MIN_10_EXP
|
||||||
|
#define LDBL_MIN_EXP DBL_MIN_EXP
|
||||||
|
|
||||||
|
#endif // _FLOAT_H
|
1211
05/tcc-0.9.25/i386-asm.c
Normal file
1211
05/tcc-0.9.25/i386-asm.c
Normal file
File diff suppressed because it is too large
Load diff
446
05/tcc-0.9.25/i386-asm.h
Normal file
446
05/tcc-0.9.25/i386-asm.h
Normal file
|
@ -0,0 +1,446 @@
|
||||||
|
DEF_ASM_OP0(pusha, 0x60) /* must be first OP0 */
|
||||||
|
DEF_ASM_OP0(popa, 0x61)
|
||||||
|
DEF_ASM_OP0(clc, 0xf8)
|
||||||
|
DEF_ASM_OP0(cld, 0xfc)
|
||||||
|
DEF_ASM_OP0(cli, 0xfa)
|
||||||
|
DEF_ASM_OP0(clts, 0x0f06)
|
||||||
|
DEF_ASM_OP0(cmc, 0xf5)
|
||||||
|
DEF_ASM_OP0(lahf, 0x9f)
|
||||||
|
DEF_ASM_OP0(sahf, 0x9e)
|
||||||
|
DEF_ASM_OP0(pushfl, 0x9c)
|
||||||
|
DEF_ASM_OP0(popfl, 0x9d)
|
||||||
|
DEF_ASM_OP0(pushf, 0x9c)
|
||||||
|
DEF_ASM_OP0(popf, 0x9d)
|
||||||
|
DEF_ASM_OP0(stc, 0xf9)
|
||||||
|
DEF_ASM_OP0(std, 0xfd)
|
||||||
|
DEF_ASM_OP0(sti, 0xfb)
|
||||||
|
DEF_ASM_OP0(aaa, 0x37)
|
||||||
|
DEF_ASM_OP0(aas, 0x3f)
|
||||||
|
DEF_ASM_OP0(daa, 0x27)
|
||||||
|
DEF_ASM_OP0(das, 0x2f)
|
||||||
|
DEF_ASM_OP0(aad, 0xd50a)
|
||||||
|
DEF_ASM_OP0(aam, 0xd40a)
|
||||||
|
DEF_ASM_OP0(cbw, 0x6698)
|
||||||
|
DEF_ASM_OP0(cwd, 0x6699)
|
||||||
|
DEF_ASM_OP0(cwde, 0x98)
|
||||||
|
DEF_ASM_OP0(cdq, 0x99)
|
||||||
|
DEF_ASM_OP0(cbtw, 0x6698)
|
||||||
|
DEF_ASM_OP0(cwtl, 0x98)
|
||||||
|
DEF_ASM_OP0(cwtd, 0x6699)
|
||||||
|
DEF_ASM_OP0(cltd, 0x99)
|
||||||
|
DEF_ASM_OP0(int3, 0xcc)
|
||||||
|
DEF_ASM_OP0(into, 0xce)
|
||||||
|
DEF_ASM_OP0(iret, 0xcf)
|
||||||
|
DEF_ASM_OP0(rsm, 0x0faa)
|
||||||
|
DEF_ASM_OP0(hlt, 0xf4)
|
||||||
|
DEF_ASM_OP0(wait, 0x9b)
|
||||||
|
DEF_ASM_OP0(nop, 0x90)
|
||||||
|
DEF_ASM_OP0(xlat, 0xd7)
|
||||||
|
|
||||||
|
/* strings */
|
||||||
|
ALT(DEF_ASM_OP0L(cmpsb, 0xa6, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(scmpb, 0xa6, 0, OPC_BWL))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP0L(insb, 0x6c, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(outsb, 0x6e, 0, OPC_BWL))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP0L(lodsb, 0xac, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(slodb, 0xac, 0, OPC_BWL))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP0L(movsb, 0xa4, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(smovb, 0xa4, 0, OPC_BWL))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP0L(scasb, 0xae, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(sscab, 0xae, 0, OPC_BWL))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP0L(stosb, 0xaa, 0, OPC_BWL))
|
||||||
|
ALT(DEF_ASM_OP0L(sstob, 0xaa, 0, OPC_BWL))
|
||||||
|
|
||||||
|
/* bits */
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(bsfw, 0x0fbc, 0, OPC_MODRM | OPC_WL, OPT_REGW | OPT_EA, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(bsrw, 0x0fbd, 0, OPC_MODRM | OPC_WL, OPT_REGW | OPT_EA, OPT_REGW))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(btw, 0x0fa3, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(btw, 0x0fba, 4, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(btsw, 0x0fab, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(btsw, 0x0fba, 5, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(btrw, 0x0fb3, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(btrw, 0x0fba, 6, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(btcw, 0x0fbb, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(btcw, 0x0fba, 7, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW | OPT_EA))
|
||||||
|
|
||||||
|
/* prefixes */
|
||||||
|
DEF_ASM_OP0(aword, 0x67)
|
||||||
|
DEF_ASM_OP0(addr16, 0x67)
|
||||||
|
DEF_ASM_OP0(word, 0x66)
|
||||||
|
DEF_ASM_OP0(data16, 0x66)
|
||||||
|
DEF_ASM_OP0(lock, 0xf0)
|
||||||
|
DEF_ASM_OP0(rep, 0xf3)
|
||||||
|
DEF_ASM_OP0(repe, 0xf3)
|
||||||
|
DEF_ASM_OP0(repz, 0xf3)
|
||||||
|
DEF_ASM_OP0(repne, 0xf2)
|
||||||
|
DEF_ASM_OP0(repnz, 0xf2)
|
||||||
|
|
||||||
|
DEF_ASM_OP0(invd, 0x0f08)
|
||||||
|
DEF_ASM_OP0(wbinvd, 0x0f09)
|
||||||
|
DEF_ASM_OP0(cpuid, 0x0fa2)
|
||||||
|
DEF_ASM_OP0(wrmsr, 0x0f30)
|
||||||
|
DEF_ASM_OP0(rdtsc, 0x0f31)
|
||||||
|
DEF_ASM_OP0(rdmsr, 0x0f32)
|
||||||
|
DEF_ASM_OP0(rdpmc, 0x0f33)
|
||||||
|
DEF_ASM_OP0(ud2, 0x0f0b)
|
||||||
|
|
||||||
|
/* NOTE: we took the same order as gas opcode definition order */
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0xa0, 0, OPC_BWL, OPT_ADDR, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0xa2, 0, OPC_BWL, OPT_EAX, OPT_ADDR))
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0x88, 0, OPC_MODRM | OPC_BWL, OPT_REG, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0x8a, 0, OPC_MODRM | OPC_BWL, OPT_EA | OPT_REG, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0xb0, 0, OPC_REG | OPC_BWL, OPT_IM, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(movb, 0xc6, 0, OPC_MODRM | OPC_BWL, OPT_IM, OPT_REG | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x8c, 0, OPC_MODRM | OPC_WL, OPT_SEG, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x8e, 0, OPC_MODRM | OPC_WL, OPT_EA | OPT_REG, OPT_SEG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f20, 0, OPC_MODRM | OPC_WL, OPT_CR, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f21, 0, OPC_MODRM | OPC_WL, OPT_DB, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f24, 0, OPC_MODRM | OPC_WL, OPT_TR, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f22, 0, OPC_MODRM | OPC_WL, OPT_REG32, OPT_CR))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f23, 0, OPC_MODRM | OPC_WL, OPT_REG32, OPT_DB))
|
||||||
|
ALT(DEF_ASM_OP2(movw, 0x0f26, 0, OPC_MODRM | OPC_WL, OPT_REG32, OPT_TR))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(movsbl, 0x0fbe, 0, OPC_MODRM, OPT_REG8 | OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(movsbw, 0x0fbe, 0, OPC_MODRM | OPC_D16, OPT_REG8 | OPT_EA, OPT_REG16))
|
||||||
|
ALT(DEF_ASM_OP2(movswl, 0x0fbf, 0, OPC_MODRM, OPT_REG16 | OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(movzbw, 0x0fb6, 0, OPC_MODRM | OPC_WL, OPT_REG8 | OPT_EA, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(movzwl, 0x0fb7, 0, OPC_MODRM, OPT_REG16 | OPT_EA, OPT_REG32))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(pushw, 0x50, 0, OPC_REG | OPC_WL, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP1(pushw, 0xff, 6, OPC_MODRM | OPC_WL, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(pushw, 0x6a, 0, OPC_WL, OPT_IM8S))
|
||||||
|
ALT(DEF_ASM_OP1(pushw, 0x68, 0, OPC_WL, OPT_IM32))
|
||||||
|
ALT(DEF_ASM_OP1(pushw, 0x06, 0, OPC_WL, OPT_SEG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(popw, 0x58, 0, OPC_REG | OPC_WL, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP1(popw, 0x8f, 0, OPC_MODRM | OPC_WL, OPT_REGW | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(popw, 0x07, 0, OPC_WL, OPT_SEG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(xchgw, 0x90, 0, OPC_REG | OPC_WL, OPT_REG, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP2(xchgw, 0x90, 0, OPC_REG | OPC_WL, OPT_EAX, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(xchgb, 0x86, 0, OPC_MODRM | OPC_BWL, OPT_REG, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(xchgb, 0x86, 0, OPC_MODRM | OPC_BWL, OPT_EA | OPT_REG, OPT_REG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(inb, 0xe4, 0, OPC_BWL, OPT_IM8, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP1(inb, 0xe4, 0, OPC_BWL, OPT_IM8))
|
||||||
|
ALT(DEF_ASM_OP2(inb, 0xec, 0, OPC_BWL, OPT_DX, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP1(inb, 0xec, 0, OPC_BWL, OPT_DX))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(outb, 0xe6, 0, OPC_BWL, OPT_EAX, OPT_IM8))
|
||||||
|
ALT(DEF_ASM_OP1(outb, 0xe6, 0, OPC_BWL, OPT_IM8))
|
||||||
|
ALT(DEF_ASM_OP2(outb, 0xee, 0, OPC_BWL, OPT_EAX, OPT_DX))
|
||||||
|
ALT(DEF_ASM_OP1(outb, 0xee, 0, OPC_BWL, OPT_DX))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(leaw, 0x8d, 0, OPC_MODRM | OPC_WL, OPT_EA, OPT_REG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(les, 0xc4, 0, OPC_MODRM, OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(lds, 0xc5, 0, OPC_MODRM, OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(lss, 0x0fb2, 0, OPC_MODRM, OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(lfs, 0x0fb4, 0, OPC_MODRM, OPT_EA, OPT_REG32))
|
||||||
|
ALT(DEF_ASM_OP2(lgs, 0x0fb5, 0, OPC_MODRM, OPT_EA, OPT_REG32))
|
||||||
|
|
||||||
|
/* arith */
|
||||||
|
ALT(DEF_ASM_OP2(addb, 0x00, 0, OPC_ARITH | OPC_MODRM | OPC_BWL, OPT_REG, OPT_EA | OPT_REG)) /* XXX: use D bit ? */
|
||||||
|
ALT(DEF_ASM_OP2(addb, 0x02, 0, OPC_ARITH | OPC_MODRM | OPC_BWL, OPT_EA | OPT_REG, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(addb, 0x04, 0, OPC_ARITH | OPC_BWL, OPT_IM, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP2(addb, 0x80, 0, OPC_ARITH | OPC_MODRM | OPC_BWL, OPT_IM, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(addw, 0x83, 0, OPC_ARITH | OPC_MODRM | OPC_WL, OPT_IM8S, OPT_EA | OPT_REG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(testb, 0x84, 0, OPC_MODRM | OPC_BWL, OPT_EA | OPT_REG, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(testb, 0x84, 0, OPC_MODRM | OPC_BWL, OPT_REG, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(testb, 0xa8, 0, OPC_BWL, OPT_IM, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP2(testb, 0xf6, 0, OPC_MODRM | OPC_BWL, OPT_IM, OPT_EA | OPT_REG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(incw, 0x40, 0, OPC_REG | OPC_WL, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP1(incb, 0xfe, 0, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(decw, 0x48, 0, OPC_REG | OPC_WL, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP1(decb, 0xfe, 1, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(notb, 0xf6, 2, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(negb, 0xf6, 3, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(mulb, 0xf6, 4, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(imulb, 0xf6, 5, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(imulw, 0x0faf, 0, OPC_MODRM | OPC_WL, OPT_REG | OPT_EA, OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP3(imulw, 0x6b, 0, OPC_MODRM | OPC_WL, OPT_IM8S, OPT_REGW | OPT_EA, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(imulw, 0x6b, 0, OPC_MODRM | OPC_WL, OPT_IM8S, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP3(imulw, 0x69, 0, OPC_MODRM | OPC_WL, OPT_IMW, OPT_REGW | OPT_EA, OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(imulw, 0x69, 0, OPC_MODRM | OPC_WL, OPT_IMW, OPT_REGW))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(divb, 0xf6, 6, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(divb, 0xf6, 6, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA, OPT_EAX))
|
||||||
|
ALT(DEF_ASM_OP1(idivb, 0xf6, 7, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(idivb, 0xf6, 7, OPC_MODRM | OPC_BWL, OPT_REG | OPT_EA, OPT_EAX))
|
||||||
|
|
||||||
|
/* shifts */
|
||||||
|
ALT(DEF_ASM_OP2(rolb, 0xc0, 0, OPC_MODRM | OPC_BWL | OPC_SHIFT, OPT_IM8, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP2(rolb, 0xd2, 0, OPC_MODRM | OPC_BWL | OPC_SHIFT, OPT_CL, OPT_EA | OPT_REG))
|
||||||
|
ALT(DEF_ASM_OP1(rolb, 0xd0, 0, OPC_MODRM | OPC_BWL | OPC_SHIFT, OPT_EA | OPT_REG))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP3(shldw, 0x0fa4, 0, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP3(shldw, 0x0fa5, 0, OPC_MODRM | OPC_WL, OPT_CL, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(shldw, 0x0fa5, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP3(shrdw, 0x0fac, 0, OPC_MODRM | OPC_WL, OPT_IM8, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP3(shrdw, 0x0fad, 0, OPC_MODRM | OPC_WL, OPT_CL, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
ALT(DEF_ASM_OP2(shrdw, 0x0fad, 0, OPC_MODRM | OPC_WL, OPT_REGW, OPT_EA | OPT_REGW))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(call, 0xff, 2, OPC_MODRM, OPT_INDIR))
|
||||||
|
ALT(DEF_ASM_OP1(call, 0xe8, 0, OPC_JMP, OPT_ADDR))
|
||||||
|
ALT(DEF_ASM_OP1(jmp, 0xff, 4, OPC_MODRM, OPT_INDIR))
|
||||||
|
ALT(DEF_ASM_OP1(jmp, 0xeb, 0, OPC_SHORTJMP | OPC_JMP, OPT_ADDR))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP2(lcall, 0x9a, 0, 0, OPT_IM16, OPT_IM32))
|
||||||
|
ALT(DEF_ASM_OP1(lcall, 0xff, 3, 0, OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP2(ljmp, 0xea, 0, 0, OPT_IM16, OPT_IM32))
|
||||||
|
ALT(DEF_ASM_OP1(ljmp, 0xff, 5, 0, OPT_EA))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(int, 0xcd, 0, 0, OPT_IM8))
|
||||||
|
ALT(DEF_ASM_OP1(seto, 0x0f90, 0, OPC_MODRM | OPC_TEST, OPT_REG8 | OPT_EA))
|
||||||
|
DEF_ASM_OP2(enter, 0xc8, 0, 0, OPT_IM16, OPT_IM8)
|
||||||
|
DEF_ASM_OP0(leave, 0xc9)
|
||||||
|
DEF_ASM_OP0(ret, 0xc3)
|
||||||
|
ALT(DEF_ASM_OP1(ret, 0xc2, 0, 0, OPT_IM16))
|
||||||
|
DEF_ASM_OP0(lret, 0xcb)
|
||||||
|
ALT(DEF_ASM_OP1(lret, 0xca, 0, 0, OPT_IM16))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(jo, 0x70, 0, OPC_SHORTJMP | OPC_JMP | OPC_TEST, OPT_ADDR))
|
||||||
|
DEF_ASM_OP1(loopne, 0xe0, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
DEF_ASM_OP1(loopnz, 0xe0, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
DEF_ASM_OP1(loope, 0xe1, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
DEF_ASM_OP1(loopz, 0xe1, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
DEF_ASM_OP1(loop, 0xe2, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
DEF_ASM_OP1(jecxz, 0xe3, 0, OPC_SHORTJMP, OPT_ADDR)
|
||||||
|
|
||||||
|
/* float */
|
||||||
|
/* specific fcomp handling */
|
||||||
|
ALT(DEF_ASM_OP0L(fcomp, 0xd8d9, 0, 0))
|
||||||
|
|
||||||
|
ALT(DEF_ASM_OP1(fadd, 0xd8c0, 0, OPC_FARITH | OPC_REG, OPT_ST))
|
||||||
|
ALT(DEF_ASM_OP2(fadd, 0xd8c0, 0, OPC_FARITH | OPC_REG, OPT_ST, OPT_ST0))
|
||||||
|
ALT(DEF_ASM_OP0L(fadd, 0xdec1, 0, OPC_FARITH))
|
||||||
|
ALT(DEF_ASM_OP1(faddp, 0xdec0, 0, OPC_FARITH | OPC_REG, OPT_ST))
|
||||||
|
ALT(DEF_ASM_OP2(faddp, 0xdec0, 0, OPC_FARITH | OPC_REG, OPT_ST, OPT_ST0))
|
||||||
|
ALT(DEF_ASM_OP2(faddp, 0xdec0, 0, OPC_FARITH | OPC_REG, OPT_ST0, OPT_ST))
|
||||||
|
ALT(DEF_ASM_OP0L(faddp, 0xdec1, 0, OPC_FARITH))
|
||||||
|
ALT(DEF_ASM_OP1(fadds, 0xd8, 0, OPC_FARITH | OPC_MODRM, OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(fiaddl, 0xda, 0, OPC_FARITH | OPC_MODRM, OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(faddl, 0xdc, 0, OPC_FARITH | OPC_MODRM, OPT_EA))
|
||||||
|
ALT(DEF_ASM_OP1(fiadds, 0xde, 0, OPC_FARITH | OPC_MODRM, OPT_EA))
|
||||||
|
|
||||||
|
DEF_ASM_OP0(fucompp, 0xdae9)
|
||||||
|
DEF_ASM_OP0(ftst, 0xd9e4)
|
||||||
|
DEF_ASM_OP0(fxam, 0xd9e5)
|
||||||
|
DEF_ASM_OP0(fld1, 0xd9e8)
|
||||||
|
DEF_ASM_OP0(fldl2t, 0xd9e9)
|
||||||
|
DEF_ASM_OP0(fldl2e, 0xd9ea)
|
||||||
|
DEF_ASM_OP0(fldpi, 0xd9eb)
|
||||||
|
DEF_ASM_OP0(fldlg2, 0xd9ec)
|
||||||
|
DEF_ASM_OP0(fldln2, 0xd9ed)
|
||||||
|
DEF_ASM_OP0(fldz, 0xd9ee)
|
||||||
|
|
||||||
|
DEF_ASM_OP0(f2xm1, 0xd9f0)
|
||||||
|
DEF_ASM_OP0(fyl2x, 0xd9f1)
|
||||||
|
DEF_ASM_OP0(fptan, 0xd9f2)
|
||||||
|
DEF_ASM_OP0(fpatan, 0xd9f3)
|
||||||
|
DEF_ASM_OP0(fxtract, 0xd9f4)
|
||||||
|
DEF_ASM_OP0(fprem1, 0xd9f5)
|
||||||
|
DEF_ASM_OP0(fdecstp, 0xd9f6)
|
||||||
|
DEF_ASM_OP0(fincstp, 0xd9f7)
|
||||||
|
DEF_ASM_OP0(fprem, 0xd9f8)
|
||||||
|
DEF_ASM_OP0(fyl2xp1, 0xd9f9)
|
||||||
|
DEF_ASM_OP0(fsqrt, 0xd9fa)
|
||||||
|
DEF_ASM_OP0(fsincos, 0xd9fb)
|
||||||
|
DEF_ASM_OP0(frndint, 0xd9fc)
|
||||||
|
DEF_ASM_OP0(fscale, 0xd9fd)
|
||||||
|
DEF_ASM_OP0(fsin, 0xd9fe)
|
||||||
|
DEF_ASM_OP0(fcos, 0xd9ff)
|
||||||
|
DEF_ASM_OP0(fchs, 0xd9e0)
|
||||||
|
DEF_ASM_OP0(fabs, 0xd9e1)
|
||||||
|
DEF_ASM_OP0(fninit, 0xdbe3)
|
||||||
|
DEF_ASM_OP0(fnclex, 0xdbe2)
|
||||||
|
DEF_ASM_OP0(fnop, 0xd9d0)
|
||||||
|
DEF_ASM_OP0(fwait, 0x9b)
|
||||||
|
|
||||||
|
/* fp load */
|
||||||
|
DEF_ASM_OP1(fld, 0xd9c0, 0, OPC_REG, OPT_ST)
|
||||||
|
DEF_ASM_OP1(fldl, 0xd9c0, 0, OPC_REG, OPT_ST)
|
||||||
|
DEF_ASM_OP1(flds, 0xd9, 0, OPC_MODRM, OPT_EA)
|
||||||
|
ALT(DEF_ASM_OP1(fldl, 0xdd, 0, OPC_MODRM, OPT_EA))
|
||||||
|
DEF_ASM_OP1(fildl, 0xdb, 0, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fildq, 0xdf, 5, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fildll, 0xdf, 5, OPC_MODRM,OPT_EA)
|
||||||
|
DEF_ASM_OP1(fldt, 0xdb, 5, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fbld, 0xdf, 4, OPC_MODRM, OPT_EA)
|
||||||
|
|
||||||
|
/* fp store */
|
||||||
|
DEF_ASM_OP1(fst, 0xddd0, 0, OPC_REG, OPT_ST)
|
||||||
|
DEF_ASM_OP1(fstl, 0xddd0, 0, OPC_REG, OPT_ST)
|
||||||
|
DEF_ASM_OP1(fsts, 0xd9, 2, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fstps, 0xd9, 3, OPC_MODRM, OPT_EA)
|
||||||
|
ALT(DEF_ASM_OP1(fstl, 0xdd, 2, OPC_MODRM, OPT_EA))
|
||||||
|
DEF_ASM_OP1(fstpl, 0xdd, 3, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fist, 0xdf, 2, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fistp, 0xdf, 3, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fistl, 0xdb, 2, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fistpl, 0xdb, 3, OPC_MODRM, OPT_EA)
|
||||||
|
|
||||||
|
DEF_ASM_OP1(fstp, 0xddd8, 0, OPC_REG, OPT_ST)
|
||||||
|
DEF_ASM_OP1(fistpq, 0xdf, 7, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fistpll, 0xdf, 7, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fstpt, 0xdb, 7, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(fbstp, 0xdf, 6, OPC_MODRM, OPT_EA)
|
||||||
|
|
||||||
|
/* exchange */
|
||||||
|
DEF_ASM_OP0(fxch, 0xd9c9)
|
||||||
|
ALT(DEF_ASM_OP1(fxch, 0xd9c8, 0, OPC_REG, OPT_ST))
|
||||||
|
|
||||||
|
/* misc FPU */
|
||||||
|
DEF_ASM_OP1(fucom, 0xdde0, 0, OPC_REG, OPT_ST )
|
||||||
|
DEF_ASM_OP1(fucomp, 0xdde8, 0, OPC_REG, OPT_ST )
|
||||||
|
|
||||||
|
DEF_ASM_OP0L(finit, 0xdbe3, 0, OPC_FWAIT)
|
||||||
|
DEF_ASM_OP1(fldcw, 0xd9, 5, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fnstcw, 0xd9, 7, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fstcw, 0xd9, 7, OPC_MODRM | OPC_FWAIT, OPT_EA )
|
||||||
|
DEF_ASM_OP0(fnstsw, 0xdfe0)
|
||||||
|
ALT(DEF_ASM_OP1(fnstsw, 0xdfe0, 0, 0, OPT_EAX ))
|
||||||
|
ALT(DEF_ASM_OP1(fnstsw, 0xdd, 7, OPC_MODRM, OPT_EA ))
|
||||||
|
DEF_ASM_OP1(fstsw, 0xdfe0, 0, OPC_FWAIT, OPT_EAX )
|
||||||
|
ALT(DEF_ASM_OP0L(fstsw, 0xdfe0, 0, OPC_FWAIT))
|
||||||
|
ALT(DEF_ASM_OP1(fstsw, 0xdd, 7, OPC_MODRM | OPC_FWAIT, OPT_EA ))
|
||||||
|
DEF_ASM_OP0L(fclex, 0xdbe2, 0, OPC_FWAIT)
|
||||||
|
DEF_ASM_OP1(fnstenv, 0xd9, 6, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fstenv, 0xd9, 6, OPC_MODRM | OPC_FWAIT, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fldenv, 0xd9, 4, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fnsave, 0xdd, 6, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fsave, 0xdd, 6, OPC_MODRM | OPC_FWAIT, OPT_EA )
|
||||||
|
DEF_ASM_OP1(frstor, 0xdd, 4, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(ffree, 0xddc0, 4, OPC_REG, OPT_ST )
|
||||||
|
DEF_ASM_OP1(ffreep, 0xdfc0, 4, OPC_REG, OPT_ST )
|
||||||
|
DEF_ASM_OP1(fxsave, 0x0fae, 0, OPC_MODRM, OPT_EA )
|
||||||
|
DEF_ASM_OP1(fxrstor, 0x0fae, 1, OPC_MODRM, OPT_EA )
|
||||||
|
|
||||||
|
/* segments */
|
||||||
|
DEF_ASM_OP2(arpl, 0x63, 0, OPC_MODRM, OPT_REG16, OPT_REG16 | OPT_EA)
|
||||||
|
DEF_ASM_OP2(lar, 0x0f02, 0, OPC_MODRM, OPT_REG32 | OPT_EA, OPT_REG32)
|
||||||
|
DEF_ASM_OP1(lgdt, 0x0f01, 2, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(lidt, 0x0f01, 3, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(lldt, 0x0f00, 2, OPC_MODRM, OPT_EA | OPT_REG)
|
||||||
|
DEF_ASM_OP1(lmsw, 0x0f01, 6, OPC_MODRM, OPT_EA | OPT_REG)
|
||||||
|
ALT(DEF_ASM_OP2(lslw, 0x0f03, 0, OPC_MODRM | OPC_WL, OPT_EA | OPT_REG, OPT_REG))
|
||||||
|
DEF_ASM_OP1(ltr, 0x0f00, 3, OPC_MODRM, OPT_EA | OPT_REG)
|
||||||
|
DEF_ASM_OP1(sgdt, 0x0f01, 0, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(sidt, 0x0f01, 1, OPC_MODRM, OPT_EA)
|
||||||
|
DEF_ASM_OP1(sldt, 0x0f00, 0, OPC_MODRM, OPT_REG | OPT_EA)
|
||||||
|
DEF_ASM_OP1(smsw, 0x0f01, 4, OPC_MODRM, OPT_REG | OPT_EA)
|
||||||
|
DEF_ASM_OP1(str, 0x0f00, 1, OPC_MODRM, OPT_REG16| OPT_EA)
|
||||||
|
DEF_ASM_OP1(verr, 0x0f00, 4, OPC_MODRM, OPT_REG | OPT_EA)
|
||||||
|
DEF_ASM_OP1(verw, 0x0f00, 5, OPC_MODRM, OPT_REG | OPT_EA)
|
||||||
|
|
||||||
|
/* 486 */
|
||||||
|
DEF_ASM_OP1(bswap, 0x0fc8, 0, OPC_REG, OPT_REG32 )
|
||||||
|
ALT(DEF_ASM_OP2(xaddb, 0x0fc0, 0, OPC_MODRM | OPC_BWL, OPT_REG, OPT_REG | OPT_EA ))
|
||||||
|
ALT(DEF_ASM_OP2(cmpxchgb, 0x0fb0, 0, OPC_MODRM | OPC_BWL, OPT_REG, OPT_REG | OPT_EA ))
|
||||||
|
DEF_ASM_OP1(invlpg, 0x0f01, 7, OPC_MODRM, OPT_EA )
|
||||||
|
|
||||||
|
DEF_ASM_OP2(boundl, 0x62, 0, OPC_MODRM, OPT_REG32, OPT_EA)
|
||||||
|
DEF_ASM_OP2(boundw, 0x62, 0, OPC_MODRM | OPC_D16, OPT_REG16, OPT_EA)
|
||||||
|
|
||||||
|
/* pentium */
|
||||||
|
DEF_ASM_OP1(cmpxchg8b, 0x0fc7, 1, OPC_MODRM, OPT_EA )
|
||||||
|
|
||||||
|
/* pentium pro */
|
||||||
|
ALT(DEF_ASM_OP2(cmovo, 0x0f40, 0, OPC_MODRM | OPC_TEST, OPT_REG32 | OPT_EA, OPT_REG32))
|
||||||
|
|
||||||
|
DEF_ASM_OP2(fcmovb, 0xdac0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmove, 0xdac8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovbe, 0xdad0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovu, 0xdad8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovnb, 0xdbc0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovne, 0xdbc8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovnbe, 0xdbd0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcmovnu, 0xdbd8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
|
||||||
|
DEF_ASM_OP2(fucomi, 0xdbe8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcomi, 0xdbf0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fucomip, 0xdfe8, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
DEF_ASM_OP2(fcomip, 0xdff0, 0, OPC_REG, OPT_ST, OPT_ST0 )
|
||||||
|
|
||||||
|
/* mmx */
|
||||||
|
DEF_ASM_OP0(emms, 0x0f77) /* must be last OP0 */
|
||||||
|
DEF_ASM_OP2(movd, 0x0f6e, 0, OPC_MODRM, OPT_EA | OPT_REG32, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(movd, 0x0f7e, 0, OPC_MODRM, OPT_MMX, OPT_EA | OPT_REG32 ))
|
||||||
|
DEF_ASM_OP2(movq, 0x0f6f, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(movq, 0x0f7f, 0, OPC_MODRM, OPT_MMX, OPT_EA | OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(packssdw, 0x0f6b, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(packsswb, 0x0f63, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(packuswb, 0x0f67, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddb, 0x0ffc, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddw, 0x0ffd, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddd, 0x0ffe, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddsb, 0x0fec, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddsw, 0x0fed, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddusb, 0x0fdc, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(paddusw, 0x0fdd, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pand, 0x0fdb, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pandn, 0x0fdf, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpeqb, 0x0f74, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpeqw, 0x0f75, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpeqd, 0x0f76, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpgtb, 0x0f64, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpgtw, 0x0f65, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pcmpgtd, 0x0f66, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pmaddwd, 0x0ff5, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pmulhw, 0x0fe5, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pmullw, 0x0fd5, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(por, 0x0feb, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psllw, 0x0ff1, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psllw, 0x0f71, 6, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(pslld, 0x0ff2, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(pslld, 0x0f72, 6, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psllq, 0x0ff3, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psllq, 0x0f73, 6, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psraw, 0x0fe1, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psraw, 0x0f71, 4, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psrad, 0x0fe2, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psrad, 0x0f72, 4, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psrlw, 0x0fd1, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psrlw, 0x0f71, 2, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psrld, 0x0fd2, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psrld, 0x0f72, 2, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psrlq, 0x0fd3, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
ALT(DEF_ASM_OP2(psrlq, 0x0f73, 2, OPC_MODRM, OPT_IM8, OPT_MMX ))
|
||||||
|
DEF_ASM_OP2(psubb, 0x0ff8, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubw, 0x0ff9, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubd, 0x0ffa, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubsb, 0x0fe8, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubsw, 0x0fe9, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubusb, 0x0fd8, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(psubusw, 0x0fd9, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpckhbw, 0x0f68, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpckhwd, 0x0f69, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpckhdq, 0x0f6a, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpcklbw, 0x0f60, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpcklwd, 0x0f61, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(punpckldq, 0x0f62, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
DEF_ASM_OP2(pxor, 0x0fef, 0, OPC_MODRM, OPT_EA | OPT_MMX, OPT_MMX )
|
||||||
|
|
||||||
|
#undef ALT
|
||||||
|
#undef DEF_ASM_OP0
|
||||||
|
#undef DEF_ASM_OP0L
|
||||||
|
#undef DEF_ASM_OP1
|
||||||
|
#undef DEF_ASM_OP2
|
||||||
|
#undef DEF_ASM_OP3
|
1034
05/tcc-0.9.25/i386-gen.c
Normal file
1034
05/tcc-0.9.25/i386-gen.c
Normal file
File diff suppressed because it is too large
Load diff
667
05/tcc-0.9.25/il-gen.c
Normal file
667
05/tcc-0.9.25/il-gen.c
Normal file
|
@ -0,0 +1,667 @@
|
||||||
|
/*
|
||||||
|
* CIL code generator for TCC
|
||||||
|
*
|
||||||
|
* Copyright (c) 2002 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* number of available registers */
|
||||||
|
#define NB_REGS 3
|
||||||
|
|
||||||
|
/* a register can belong to several classes. The classes must be
|
||||||
|
sorted from more general to more precise (see gv2() code which does
|
||||||
|
assumptions on it). */
|
||||||
|
#define RC_ST 0x0001 /* any stack entry */
|
||||||
|
#define RC_ST0 0x0002 /* top of stack */
|
||||||
|
#define RC_ST1 0x0004 /* top - 1 */
|
||||||
|
|
||||||
|
#define RC_INT RC_ST
|
||||||
|
#define RC_FLOAT RC_ST
|
||||||
|
#define RC_IRET RC_ST0 /* function return: integer register */
|
||||||
|
#define RC_LRET RC_ST0 /* function return: second integer register */
|
||||||
|
#define RC_FRET RC_ST0 /* function return: float register */
|
||||||
|
|
||||||
|
/* pretty names for the registers */
|
||||||
|
enum {
|
||||||
|
REG_ST0 = 0,
|
||||||
|
REG_ST1,
|
||||||
|
REG_ST2,
|
||||||
|
};
|
||||||
|
|
||||||
|
int reg_classes[NB_REGS] = {
|
||||||
|
/* ST0 */ RC_ST | RC_ST0,
|
||||||
|
/* ST1 */ RC_ST | RC_ST1,
|
||||||
|
/* ST2 */ RC_ST,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* return registers for function */
|
||||||
|
#define REG_IRET REG_ST0 /* single word int return register */
|
||||||
|
#define REG_LRET REG_ST0 /* second word return register (for long long) */
|
||||||
|
#define REG_FRET REG_ST0 /* float return register */
|
||||||
|
|
||||||
|
/* defined if function parameters must be evaluated in reverse order */
|
||||||
|
//#define INVERT_FUNC_PARAMS
|
||||||
|
|
||||||
|
/* defined if structures are passed as pointers. Otherwise structures
|
||||||
|
are directly pushed on stack. */
|
||||||
|
//#define FUNC_STRUCT_PARAM_AS_PTR
|
||||||
|
|
||||||
|
/* pointer size, in bytes */
|
||||||
|
#define PTR_SIZE 4
|
||||||
|
|
||||||
|
/* long double size and alignment, in bytes */
|
||||||
|
#define LDOUBLE_SIZE 8
|
||||||
|
#define LDOUBLE_ALIGN 8
|
||||||
|
|
||||||
|
/* function call context */
|
||||||
|
typedef struct GFuncContext {
|
||||||
|
int func_call; /* func call type (FUNC_STDCALL or FUNC_CDECL) */
|
||||||
|
} GFuncContext;
|
||||||
|
|
||||||
|
/******************************************************/
|
||||||
|
/* opcode definitions */
|
||||||
|
|
||||||
|
#define IL_OP_PREFIX 0xFE
|
||||||
|
|
||||||
|
enum ILOPCodes {
|
||||||
|
#define OP(name, str, n) IL_OP_ ## name = n,
|
||||||
|
#include "il-opcodes.h"
|
||||||
|
#undef OP
|
||||||
|
};
|
||||||
|
|
||||||
|
char *il_opcodes_str[] = {
|
||||||
|
#define OP(name, str, n) [n] = str,
|
||||||
|
#include "il-opcodes.h"
|
||||||
|
#undef OP
|
||||||
|
};
|
||||||
|
|
||||||
|
/******************************************************/
|
||||||
|
|
||||||
|
/* arguments variable numbers start from there */
|
||||||
|
#define ARG_BASE 0x70000000
|
||||||
|
|
||||||
|
static FILE *il_outfile;
|
||||||
|
|
||||||
|
static void out_byte(int c)
|
||||||
|
{
|
||||||
|
*(char *)ind++ = c;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void out_le32(int c)
|
||||||
|
{
|
||||||
|
out_byte(c);
|
||||||
|
out_byte(c >> 8);
|
||||||
|
out_byte(c >> 16);
|
||||||
|
out_byte(c >> 24);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void init_outfile(void)
|
||||||
|
{
|
||||||
|
if (!il_outfile) {
|
||||||
|
il_outfile = stdout;
|
||||||
|
fprintf(il_outfile,
|
||||||
|
".assembly extern mscorlib\n"
|
||||||
|
"{\n"
|
||||||
|
".ver 1:0:2411:0\n"
|
||||||
|
"}\n\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void out_op1(int op)
|
||||||
|
{
|
||||||
|
if (op & 0x100)
|
||||||
|
out_byte(IL_OP_PREFIX);
|
||||||
|
out_byte(op & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* output an opcode with prefix */
|
||||||
|
static void out_op(int op)
|
||||||
|
{
|
||||||
|
out_op1(op);
|
||||||
|
fprintf(il_outfile, " %s\n", il_opcodes_str[op]);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void out_opb(int op, int c)
|
||||||
|
{
|
||||||
|
out_op1(op);
|
||||||
|
out_byte(c);
|
||||||
|
fprintf(il_outfile, " %s %d\n", il_opcodes_str[op], c);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void out_opi(int op, int c)
|
||||||
|
{
|
||||||
|
out_op1(op);
|
||||||
|
out_le32(c);
|
||||||
|
fprintf(il_outfile, " %s 0x%x\n", il_opcodes_str[op], c);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: not complete */
|
||||||
|
static void il_type_to_str(char *buf, int buf_size,
|
||||||
|
int t, const char *varstr)
|
||||||
|
{
|
||||||
|
int bt;
|
||||||
|
Sym *s, *sa;
|
||||||
|
char buf1[256];
|
||||||
|
const char *tstr;
|
||||||
|
|
||||||
|
t = t & VT_TYPE;
|
||||||
|
bt = t & VT_BTYPE;
|
||||||
|
buf[0] = '\0';
|
||||||
|
if (t & VT_UNSIGNED)
|
||||||
|
pstrcat(buf, buf_size, "unsigned ");
|
||||||
|
switch(bt) {
|
||||||
|
case VT_VOID:
|
||||||
|
tstr = "void";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_BOOL:
|
||||||
|
tstr = "bool";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_BYTE:
|
||||||
|
tstr = "int8";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_SHORT:
|
||||||
|
tstr = "int16";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_ENUM:
|
||||||
|
case VT_INT:
|
||||||
|
case VT_LONG:
|
||||||
|
tstr = "int32";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_LLONG:
|
||||||
|
tstr = "int64";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_FLOAT:
|
||||||
|
tstr = "float32";
|
||||||
|
goto add_tstr;
|
||||||
|
case VT_DOUBLE:
|
||||||
|
case VT_LDOUBLE:
|
||||||
|
tstr = "float64";
|
||||||
|
add_tstr:
|
||||||
|
pstrcat(buf, buf_size, tstr);
|
||||||
|
break;
|
||||||
|
case VT_STRUCT:
|
||||||
|
error("structures not handled yet");
|
||||||
|
break;
|
||||||
|
case VT_FUNC:
|
||||||
|
s = sym_find((unsigned)t >> VT_STRUCT_SHIFT);
|
||||||
|
il_type_to_str(buf, buf_size, s->t, varstr);
|
||||||
|
pstrcat(buf, buf_size, "(");
|
||||||
|
sa = s->next;
|
||||||
|
while (sa != NULL) {
|
||||||
|
il_type_to_str(buf1, sizeof(buf1), sa->t, NULL);
|
||||||
|
pstrcat(buf, buf_size, buf1);
|
||||||
|
sa = sa->next;
|
||||||
|
if (sa)
|
||||||
|
pstrcat(buf, buf_size, ", ");
|
||||||
|
}
|
||||||
|
pstrcat(buf, buf_size, ")");
|
||||||
|
goto no_var;
|
||||||
|
case VT_PTR:
|
||||||
|
s = sym_find((unsigned)t >> VT_STRUCT_SHIFT);
|
||||||
|
pstrcpy(buf1, sizeof(buf1), "*");
|
||||||
|
if (varstr)
|
||||||
|
pstrcat(buf1, sizeof(buf1), varstr);
|
||||||
|
il_type_to_str(buf, buf_size, s->t, buf1);
|
||||||
|
goto no_var;
|
||||||
|
}
|
||||||
|
if (varstr) {
|
||||||
|
pstrcat(buf, buf_size, " ");
|
||||||
|
pstrcat(buf, buf_size, varstr);
|
||||||
|
}
|
||||||
|
no_var: ;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* patch relocation entry with value 'val' */
|
||||||
|
void greloc_patch1(Reloc *p, int val)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/* output a symbol and patch all calls to it */
|
||||||
|
void gsym_addr(t, a)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
/* output jump and return symbol */
|
||||||
|
static int out_opj(int op, int c)
|
||||||
|
{
|
||||||
|
out_op1(op);
|
||||||
|
out_le32(0);
|
||||||
|
if (c == 0) {
|
||||||
|
c = ind - (int)cur_text_section->data;
|
||||||
|
}
|
||||||
|
fprintf(il_outfile, " %s L%d\n", il_opcodes_str[op], c);
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
|
void gsym(int t)
|
||||||
|
{
|
||||||
|
fprintf(il_outfile, "L%d:\n", t);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* load 'r' from value 'sv' */
|
||||||
|
void load(int r, SValue *sv)
|
||||||
|
{
|
||||||
|
int v, fc, ft;
|
||||||
|
|
||||||
|
v = sv->r & VT_VALMASK;
|
||||||
|
fc = sv->c.i;
|
||||||
|
ft = sv->t;
|
||||||
|
|
||||||
|
if (sv->r & VT_LVAL) {
|
||||||
|
if (v == VT_LOCAL) {
|
||||||
|
if (fc >= ARG_BASE) {
|
||||||
|
fc -= ARG_BASE;
|
||||||
|
if (fc >= 0 && fc <= 4) {
|
||||||
|
out_op(IL_OP_LDARG_0 + fc);
|
||||||
|
} else if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_LDARG_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_LDARG, fc);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (fc >= 0 && fc <= 4) {
|
||||||
|
out_op(IL_OP_LDLOC_0 + fc);
|
||||||
|
} else if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_LDLOC_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_LDLOC, fc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (v == VT_CONST) {
|
||||||
|
/* XXX: handle globals */
|
||||||
|
out_opi(IL_OP_LDSFLD, 0);
|
||||||
|
} else {
|
||||||
|
if ((ft & VT_BTYPE) == VT_FLOAT) {
|
||||||
|
out_op(IL_OP_LDIND_R4);
|
||||||
|
} else if ((ft & VT_BTYPE) == VT_DOUBLE) {
|
||||||
|
out_op(IL_OP_LDIND_R8);
|
||||||
|
} else if ((ft & VT_BTYPE) == VT_LDOUBLE) {
|
||||||
|
out_op(IL_OP_LDIND_R8);
|
||||||
|
} else if ((ft & VT_TYPE) == VT_BYTE)
|
||||||
|
out_op(IL_OP_LDIND_I1);
|
||||||
|
else if ((ft & VT_TYPE) == (VT_BYTE | VT_UNSIGNED))
|
||||||
|
out_op(IL_OP_LDIND_U1);
|
||||||
|
else if ((ft & VT_TYPE) == VT_SHORT)
|
||||||
|
out_op(IL_OP_LDIND_I2);
|
||||||
|
else if ((ft & VT_TYPE) == (VT_SHORT | VT_UNSIGNED))
|
||||||
|
out_op(IL_OP_LDIND_U2);
|
||||||
|
else
|
||||||
|
out_op(IL_OP_LDIND_I4);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (v == VT_CONST) {
|
||||||
|
/* XXX: handle globals */
|
||||||
|
if (fc >= -1 && fc <= 8) {
|
||||||
|
out_op(IL_OP_LDC_I4_M1 + fc + 1);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_LDC_I4, fc);
|
||||||
|
}
|
||||||
|
} else if (v == VT_LOCAL) {
|
||||||
|
if (fc >= ARG_BASE) {
|
||||||
|
fc -= ARG_BASE;
|
||||||
|
if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_LDARGA_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_LDARGA, fc);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_LDLOCA_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_LDLOCA, fc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* XXX: do it */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* store register 'r' in lvalue 'v' */
|
||||||
|
void store(int r, SValue *sv)
|
||||||
|
{
|
||||||
|
int v, fc, ft;
|
||||||
|
|
||||||
|
v = sv->r & VT_VALMASK;
|
||||||
|
fc = sv->c.i;
|
||||||
|
ft = sv->t;
|
||||||
|
if (v == VT_LOCAL) {
|
||||||
|
if (fc >= ARG_BASE) {
|
||||||
|
fc -= ARG_BASE;
|
||||||
|
/* XXX: check IL arg store semantics */
|
||||||
|
if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_STARG_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_STARG, fc);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (fc >= 0 && fc <= 4) {
|
||||||
|
out_op(IL_OP_STLOC_0 + fc);
|
||||||
|
} else if (fc <= 0xff) {
|
||||||
|
out_opb(IL_OP_STLOC_S, fc);
|
||||||
|
} else {
|
||||||
|
out_opi(IL_OP_STLOC, fc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (v == VT_CONST) {
|
||||||
|
/* XXX: handle globals */
|
||||||
|
out_opi(IL_OP_STSFLD, 0);
|
||||||
|
} else {
|
||||||
|
if ((ft & VT_BTYPE) == VT_FLOAT)
|
||||||
|
out_op(IL_OP_STIND_R4);
|
||||||
|
else if ((ft & VT_BTYPE) == VT_DOUBLE)
|
||||||
|
out_op(IL_OP_STIND_R8);
|
||||||
|
else if ((ft & VT_BTYPE) == VT_LDOUBLE)
|
||||||
|
out_op(IL_OP_STIND_R8);
|
||||||
|
else if ((ft & VT_BTYPE) == VT_BYTE)
|
||||||
|
out_op(IL_OP_STIND_I1);
|
||||||
|
else if ((ft & VT_BTYPE) == VT_SHORT)
|
||||||
|
out_op(IL_OP_STIND_I2);
|
||||||
|
else
|
||||||
|
out_op(IL_OP_STIND_I4);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* start function call and return function call context */
|
||||||
|
void gfunc_start(GFuncContext *c, int func_call)
|
||||||
|
{
|
||||||
|
c->func_call = func_call;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* push function parameter which is in (vtop->t, vtop->c). Stack entry
|
||||||
|
is then popped. */
|
||||||
|
void gfunc_param(GFuncContext *c)
|
||||||
|
{
|
||||||
|
if ((vtop->t & VT_BTYPE) == VT_STRUCT) {
|
||||||
|
error("structures passed as value not handled yet");
|
||||||
|
} else {
|
||||||
|
/* simply push on stack */
|
||||||
|
gv(RC_ST0);
|
||||||
|
}
|
||||||
|
vtop--;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate function call with address in (vtop->t, vtop->c) and free function
|
||||||
|
context. Stack entry is popped */
|
||||||
|
void gfunc_call(GFuncContext *c)
|
||||||
|
{
|
||||||
|
char buf[1024];
|
||||||
|
|
||||||
|
if ((vtop->r & (VT_VALMASK | VT_LVAL)) == VT_CONST) {
|
||||||
|
/* XXX: more info needed from tcc */
|
||||||
|
il_type_to_str(buf, sizeof(buf), vtop->t, "xxx");
|
||||||
|
fprintf(il_outfile, " call %s\n", buf);
|
||||||
|
} else {
|
||||||
|
/* indirect call */
|
||||||
|
gv(RC_INT);
|
||||||
|
il_type_to_str(buf, sizeof(buf), vtop->t, NULL);
|
||||||
|
fprintf(il_outfile, " calli %s\n", buf);
|
||||||
|
}
|
||||||
|
vtop--;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate function prolog of type 't' */
|
||||||
|
void gfunc_prolog(int t)
|
||||||
|
{
|
||||||
|
int addr, u, func_call;
|
||||||
|
Sym *sym;
|
||||||
|
char buf[1024];
|
||||||
|
|
||||||
|
init_outfile();
|
||||||
|
|
||||||
|
/* XXX: pass function name to gfunc_prolog */
|
||||||
|
il_type_to_str(buf, sizeof(buf), t, funcname);
|
||||||
|
fprintf(il_outfile, ".method static %s il managed\n", buf);
|
||||||
|
fprintf(il_outfile, "{\n");
|
||||||
|
/* XXX: cannot do better now */
|
||||||
|
fprintf(il_outfile, " .maxstack %d\n", NB_REGS);
|
||||||
|
fprintf(il_outfile, " .locals (int32, int32, int32, int32, int32, int32, int32, int32)\n");
|
||||||
|
|
||||||
|
if (!strcmp(funcname, "main"))
|
||||||
|
fprintf(il_outfile, " .entrypoint\n");
|
||||||
|
|
||||||
|
sym = sym_find((unsigned)t >> VT_STRUCT_SHIFT);
|
||||||
|
func_call = sym->r;
|
||||||
|
|
||||||
|
addr = ARG_BASE;
|
||||||
|
/* if the function returns a structure, then add an
|
||||||
|
implicit pointer parameter */
|
||||||
|
func_vt = sym->t;
|
||||||
|
if ((func_vt & VT_BTYPE) == VT_STRUCT) {
|
||||||
|
func_vc = addr;
|
||||||
|
addr++;
|
||||||
|
}
|
||||||
|
/* define parameters */
|
||||||
|
while ((sym = sym->next) != NULL) {
|
||||||
|
u = sym->t;
|
||||||
|
sym_push(sym->v & ~SYM_FIELD, u,
|
||||||
|
VT_LOCAL | lvalue_type(sym->type.t), addr);
|
||||||
|
addr++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate function epilog */
|
||||||
|
void gfunc_epilog(void)
|
||||||
|
{
|
||||||
|
out_op(IL_OP_RET);
|
||||||
|
fprintf(il_outfile, "}\n\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate a jump to a label */
|
||||||
|
int gjmp(int t)
|
||||||
|
{
|
||||||
|
return out_opj(IL_OP_BR, t);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate a jump to a fixed address */
|
||||||
|
void gjmp_addr(int a)
|
||||||
|
{
|
||||||
|
/* XXX: handle syms */
|
||||||
|
out_opi(IL_OP_BR, a);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate a test. set 'inv' to invert test. Stack entry is popped */
|
||||||
|
int gtst(int inv, int t)
|
||||||
|
{
|
||||||
|
int v, *p, c;
|
||||||
|
|
||||||
|
v = vtop->r & VT_VALMASK;
|
||||||
|
if (v == VT_CMP) {
|
||||||
|
c = vtop->c.i ^ inv;
|
||||||
|
switch(c) {
|
||||||
|
case TOK_EQ:
|
||||||
|
c = IL_OP_BEQ;
|
||||||
|
break;
|
||||||
|
case TOK_NE:
|
||||||
|
c = IL_OP_BNE_UN;
|
||||||
|
break;
|
||||||
|
case TOK_LT:
|
||||||
|
c = IL_OP_BLT;
|
||||||
|
break;
|
||||||
|
case TOK_LE:
|
||||||
|
c = IL_OP_BLE;
|
||||||
|
break;
|
||||||
|
case TOK_GT:
|
||||||
|
c = IL_OP_BGT;
|
||||||
|
break;
|
||||||
|
case TOK_GE:
|
||||||
|
c = IL_OP_BGE;
|
||||||
|
break;
|
||||||
|
case TOK_ULT:
|
||||||
|
c = IL_OP_BLT_UN;
|
||||||
|
break;
|
||||||
|
case TOK_ULE:
|
||||||
|
c = IL_OP_BLE_UN;
|
||||||
|
break;
|
||||||
|
case TOK_UGT:
|
||||||
|
c = IL_OP_BGT_UN;
|
||||||
|
break;
|
||||||
|
case TOK_UGE:
|
||||||
|
c = IL_OP_BGE_UN;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
t = out_opj(c, t);
|
||||||
|
} else if (v == VT_JMP || v == VT_JMPI) {
|
||||||
|
/* && or || optimization */
|
||||||
|
if ((v & 1) == inv) {
|
||||||
|
/* insert vtop->c jump list in t */
|
||||||
|
p = &vtop->c.i;
|
||||||
|
while (*p != 0)
|
||||||
|
p = (int *)*p;
|
||||||
|
*p = t;
|
||||||
|
t = vtop->c.i;
|
||||||
|
} else {
|
||||||
|
t = gjmp(t);
|
||||||
|
gsym(vtop->c.i);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (is_float(vtop->t)) {
|
||||||
|
vpushi(0);
|
||||||
|
gen_op(TOK_NE);
|
||||||
|
}
|
||||||
|
if ((vtop->r & (VT_VALMASK | VT_LVAL | VT_FORWARD)) == VT_CONST) {
|
||||||
|
/* constant jmp optimization */
|
||||||
|
if ((vtop->c.i != 0) != inv)
|
||||||
|
t = gjmp(t);
|
||||||
|
} else {
|
||||||
|
v = gv(RC_INT);
|
||||||
|
t = out_opj(IL_OP_BRTRUE - inv, t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
vtop--;
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate an integer binary operation */
|
||||||
|
void gen_opi(int op)
|
||||||
|
{
|
||||||
|
gv2(RC_ST1, RC_ST0);
|
||||||
|
switch(op) {
|
||||||
|
case '+':
|
||||||
|
out_op(IL_OP_ADD);
|
||||||
|
goto std_op;
|
||||||
|
case '-':
|
||||||
|
out_op(IL_OP_SUB);
|
||||||
|
goto std_op;
|
||||||
|
case '&':
|
||||||
|
out_op(IL_OP_AND);
|
||||||
|
goto std_op;
|
||||||
|
case '^':
|
||||||
|
out_op(IL_OP_XOR);
|
||||||
|
goto std_op;
|
||||||
|
case '|':
|
||||||
|
out_op(IL_OP_OR);
|
||||||
|
goto std_op;
|
||||||
|
case '*':
|
||||||
|
out_op(IL_OP_MUL);
|
||||||
|
goto std_op;
|
||||||
|
case TOK_SHL:
|
||||||
|
out_op(IL_OP_SHL);
|
||||||
|
goto std_op;
|
||||||
|
case TOK_SHR:
|
||||||
|
out_op(IL_OP_SHR_UN);
|
||||||
|
goto std_op;
|
||||||
|
case TOK_SAR:
|
||||||
|
out_op(IL_OP_SHR);
|
||||||
|
goto std_op;
|
||||||
|
case '/':
|
||||||
|
case TOK_PDIV:
|
||||||
|
out_op(IL_OP_DIV);
|
||||||
|
goto std_op;
|
||||||
|
case TOK_UDIV:
|
||||||
|
out_op(IL_OP_DIV_UN);
|
||||||
|
goto std_op;
|
||||||
|
case '%':
|
||||||
|
out_op(IL_OP_REM);
|
||||||
|
goto std_op;
|
||||||
|
case TOK_UMOD:
|
||||||
|
out_op(IL_OP_REM_UN);
|
||||||
|
std_op:
|
||||||
|
vtop--;
|
||||||
|
vtop[0].r = REG_ST0;
|
||||||
|
break;
|
||||||
|
case TOK_EQ:
|
||||||
|
case TOK_NE:
|
||||||
|
case TOK_LT:
|
||||||
|
case TOK_LE:
|
||||||
|
case TOK_GT:
|
||||||
|
case TOK_GE:
|
||||||
|
case TOK_ULT:
|
||||||
|
case TOK_ULE:
|
||||||
|
case TOK_UGT:
|
||||||
|
case TOK_UGE:
|
||||||
|
vtop--;
|
||||||
|
vtop[0].r = VT_CMP;
|
||||||
|
vtop[0].c.i = op;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* generate a floating point operation 'v = t1 op t2' instruction. The
|
||||||
|
two operands are guaranted to have the same floating point type */
|
||||||
|
void gen_opf(int op)
|
||||||
|
{
|
||||||
|
/* same as integer */
|
||||||
|
gen_opi(op);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* convert integers to fp 't' type. Must handle 'int', 'unsigned int'
|
||||||
|
and 'long long' cases. */
|
||||||
|
void gen_cvt_itof(int t)
|
||||||
|
{
|
||||||
|
gv(RC_ST0);
|
||||||
|
if (t == VT_FLOAT)
|
||||||
|
out_op(IL_OP_CONV_R4);
|
||||||
|
else
|
||||||
|
out_op(IL_OP_CONV_R8);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* convert fp to int 't' type */
|
||||||
|
/* XXX: handle long long case */
|
||||||
|
void gen_cvt_ftoi(int t)
|
||||||
|
{
|
||||||
|
gv(RC_ST0);
|
||||||
|
switch(t) {
|
||||||
|
case VT_INT | VT_UNSIGNED:
|
||||||
|
out_op(IL_OP_CONV_U4);
|
||||||
|
break;
|
||||||
|
case VT_LLONG:
|
||||||
|
out_op(IL_OP_CONV_I8);
|
||||||
|
break;
|
||||||
|
case VT_LLONG | VT_UNSIGNED:
|
||||||
|
out_op(IL_OP_CONV_U8);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
out_op(IL_OP_CONV_I4);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* convert from one floating point type to another */
|
||||||
|
void gen_cvt_ftof(int t)
|
||||||
|
{
|
||||||
|
gv(RC_ST0);
|
||||||
|
if (t == VT_FLOAT) {
|
||||||
|
out_op(IL_OP_CONV_R4);
|
||||||
|
} else {
|
||||||
|
out_op(IL_OP_CONV_R8);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* end of CIL code generator */
|
||||||
|
/*************************************************************/
|
||||||
|
|
251
05/tcc-0.9.25/il-opcodes.h
Normal file
251
05/tcc-0.9.25/il-opcodes.h
Normal file
|
@ -0,0 +1,251 @@
|
||||||
|
/*
|
||||||
|
* CIL opcode definition
|
||||||
|
*
|
||||||
|
* Copyright (c) 2002 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||||||
|
*/
|
||||||
|
OP(NOP, "nop", 0x00)
|
||||||
|
OP(BREAK, "break", 0x01)
|
||||||
|
OP(LDARG_0, "ldarg.0", 0x02)
|
||||||
|
OP(LDARG_1, "ldarg.1", 0x03)
|
||||||
|
OP(LDARG_2, "ldarg.2", 0x04)
|
||||||
|
OP(LDARG_3, "ldarg.3", 0x05)
|
||||||
|
OP(LDLOC_0, "ldloc.0", 0x06)
|
||||||
|
OP(LDLOC_1, "ldloc.1", 0x07)
|
||||||
|
OP(LDLOC_2, "ldloc.2", 0x08)
|
||||||
|
OP(LDLOC_3, "ldloc.3", 0x09)
|
||||||
|
OP(STLOC_0, "stloc.0", 0x0a)
|
||||||
|
OP(STLOC_1, "stloc.1", 0x0b)
|
||||||
|
OP(STLOC_2, "stloc.2", 0x0c)
|
||||||
|
OP(STLOC_3, "stloc.3", 0x0d)
|
||||||
|
OP(LDARG_S, "ldarg.s", 0x0e)
|
||||||
|
OP(LDARGA_S, "ldarga.s", 0x0f)
|
||||||
|
OP(STARG_S, "starg.s", 0x10)
|
||||||
|
OP(LDLOC_S, "ldloc.s", 0x11)
|
||||||
|
OP(LDLOCA_S, "ldloca.s", 0x12)
|
||||||
|
OP(STLOC_S, "stloc.s", 0x13)
|
||||||
|
OP(LDNULL, "ldnull", 0x14)
|
||||||
|
OP(LDC_I4_M1, "ldc.i4.m1", 0x15)
|
||||||
|
OP(LDC_I4_0, "ldc.i4.0", 0x16)
|
||||||
|
OP(LDC_I4_1, "ldc.i4.1", 0x17)
|
||||||
|
OP(LDC_I4_2, "ldc.i4.2", 0x18)
|
||||||
|
OP(LDC_I4_3, "ldc.i4.3", 0x19)
|
||||||
|
OP(LDC_I4_4, "ldc.i4.4", 0x1a)
|
||||||
|
OP(LDC_I4_5, "ldc.i4.5", 0x1b)
|
||||||
|
OP(LDC_I4_6, "ldc.i4.6", 0x1c)
|
||||||
|
OP(LDC_I4_7, "ldc.i4.7", 0x1d)
|
||||||
|
OP(LDC_I4_8, "ldc.i4.8", 0x1e)
|
||||||
|
OP(LDC_I4_S, "ldc.i4.s", 0x1f)
|
||||||
|
OP(LDC_I4, "ldc.i4", 0x20)
|
||||||
|
OP(LDC_I8, "ldc.i8", 0x21)
|
||||||
|
OP(LDC_R4, "ldc.r4", 0x22)
|
||||||
|
OP(LDC_R8, "ldc.r8", 0x23)
|
||||||
|
OP(LDPTR, "ldptr", 0x24)
|
||||||
|
OP(DUP, "dup", 0x25)
|
||||||
|
OP(POP, "pop", 0x26)
|
||||||
|
OP(JMP, "jmp", 0x27)
|
||||||
|
OP(CALL, "call", 0x28)
|
||||||
|
OP(CALLI, "calli", 0x29)
|
||||||
|
OP(RET, "ret", 0x2a)
|
||||||
|
OP(BR_S, "br.s", 0x2b)
|
||||||
|
OP(BRFALSE_S, "brfalse.s", 0x2c)
|
||||||
|
OP(BRTRUE_S, "brtrue.s", 0x2d)
|
||||||
|
OP(BEQ_S, "beq.s", 0x2e)
|
||||||
|
OP(BGE_S, "bge.s", 0x2f)
|
||||||
|
OP(BGT_S, "bgt.s", 0x30)
|
||||||
|
OP(BLE_S, "ble.s", 0x31)
|
||||||
|
OP(BLT_S, "blt.s", 0x32)
|
||||||
|
OP(BNE_UN_S, "bne.un.s", 0x33)
|
||||||
|
OP(BGE_UN_S, "bge.un.s", 0x34)
|
||||||
|
OP(BGT_UN_S, "bgt.un.s", 0x35)
|
||||||
|
OP(BLE_UN_S, "ble.un.s", 0x36)
|
||||||
|
OP(BLT_UN_S, "blt.un.s", 0x37)
|
||||||
|
OP(BR, "br", 0x38)
|
||||||
|
OP(BRFALSE, "brfalse", 0x39)
|
||||||
|
OP(BRTRUE, "brtrue", 0x3a)
|
||||||
|
OP(BEQ, "beq", 0x3b)
|
||||||
|
OP(BGE, "bge", 0x3c)
|
||||||
|
OP(BGT, "bgt", 0x3d)
|
||||||
|
OP(BLE, "ble", 0x3e)
|
||||||
|
OP(BLT, "blt", 0x3f)
|
||||||
|
OP(BNE_UN, "bne.un", 0x40)
|
||||||
|
OP(BGE_UN, "bge.un", 0x41)
|
||||||
|
OP(BGT_UN, "bgt.un", 0x42)
|
||||||
|
OP(BLE_UN, "ble.un", 0x43)
|
||||||
|
OP(BLT_UN, "blt.un", 0x44)
|
||||||
|
OP(SWITCH, "switch", 0x45)
|
||||||
|
OP(LDIND_I1, "ldind.i1", 0x46)
|
||||||
|
OP(LDIND_U1, "ldind.u1", 0x47)
|
||||||
|
OP(LDIND_I2, "ldind.i2", 0x48)
|
||||||
|
OP(LDIND_U2, "ldind.u2", 0x49)
|
||||||
|
OP(LDIND_I4, "ldind.i4", 0x4a)
|
||||||
|
OP(LDIND_U4, "ldind.u4", 0x4b)
|
||||||
|
OP(LDIND_I8, "ldind.i8", 0x4c)
|
||||||
|
OP(LDIND_I, "ldind.i", 0x4d)
|
||||||
|
OP(LDIND_R4, "ldind.r4", 0x4e)
|
||||||
|
OP(LDIND_R8, "ldind.r8", 0x4f)
|
||||||
|
OP(LDIND_REF, "ldind.ref", 0x50)
|
||||||
|
OP(STIND_REF, "stind.ref", 0x51)
|
||||||
|
OP(STIND_I1, "stind.i1", 0x52)
|
||||||
|
OP(STIND_I2, "stind.i2", 0x53)
|
||||||
|
OP(STIND_I4, "stind.i4", 0x54)
|
||||||
|
OP(STIND_I8, "stind.i8", 0x55)
|
||||||
|
OP(STIND_R4, "stind.r4", 0x56)
|
||||||
|
OP(STIND_R8, "stind.r8", 0x57)
|
||||||
|
OP(ADD, "add", 0x58)
|
||||||
|
OP(SUB, "sub", 0x59)
|
||||||
|
OP(MUL, "mul", 0x5a)
|
||||||
|
OP(DIV, "div", 0x5b)
|
||||||
|
OP(DIV_UN, "div.un", 0x5c)
|
||||||
|
OP(REM, "rem", 0x5d)
|
||||||
|
OP(REM_UN, "rem.un", 0x5e)
|
||||||
|
OP(AND, "and", 0x5f)
|
||||||
|
OP(OR, "or", 0x60)
|
||||||
|
OP(XOR, "xor", 0x61)
|
||||||
|
OP(SHL, "shl", 0x62)
|
||||||
|
OP(SHR, "shr", 0x63)
|
||||||
|
OP(SHR_UN, "shr.un", 0x64)
|
||||||
|
OP(NEG, "neg", 0x65)
|
||||||
|
OP(NOT, "not", 0x66)
|
||||||
|
OP(CONV_I1, "conv.i1", 0x67)
|
||||||
|
OP(CONV_I2, "conv.i2", 0x68)
|
||||||
|
OP(CONV_I4, "conv.i4", 0x69)
|
||||||
|
OP(CONV_I8, "conv.i8", 0x6a)
|
||||||
|
OP(CONV_R4, "conv.r4", 0x6b)
|
||||||
|
OP(CONV_R8, "conv.r8", 0x6c)
|
||||||
|
OP(CONV_U4, "conv.u4", 0x6d)
|
||||||
|
OP(CONV_U8, "conv.u8", 0x6e)
|
||||||
|
OP(CALLVIRT, "callvirt", 0x6f)
|
||||||
|
OP(CPOBJ, "cpobj", 0x70)
|
||||||
|
OP(LDOBJ, "ldobj", 0x71)
|
||||||
|
OP(LDSTR, "ldstr", 0x72)
|
||||||
|
OP(NEWOBJ, "newobj", 0x73)
|
||||||
|
OP(CASTCLASS, "castclass", 0x74)
|
||||||
|
OP(ISINST, "isinst", 0x75)
|
||||||
|
OP(CONV_R_UN, "conv.r.un", 0x76)
|
||||||
|
OP(ANN_DATA_S, "ann.data.s", 0x77)
|
||||||
|
OP(UNBOX, "unbox", 0x79)
|
||||||
|
OP(THROW, "throw", 0x7a)
|
||||||
|
OP(LDFLD, "ldfld", 0x7b)
|
||||||
|
OP(LDFLDA, "ldflda", 0x7c)
|
||||||
|
OP(STFLD, "stfld", 0x7d)
|
||||||
|
OP(LDSFLD, "ldsfld", 0x7e)
|
||||||
|
OP(LDSFLDA, "ldsflda", 0x7f)
|
||||||
|
OP(STSFLD, "stsfld", 0x80)
|
||||||
|
OP(STOBJ, "stobj", 0x81)
|
||||||
|
OP(CONV_OVF_I1_UN, "conv.ovf.i1.un", 0x82)
|
||||||
|
OP(CONV_OVF_I2_UN, "conv.ovf.i2.un", 0x83)
|
||||||
|
OP(CONV_OVF_I4_UN, "conv.ovf.i4.un", 0x84)
|
||||||
|
OP(CONV_OVF_I8_UN, "conv.ovf.i8.un", 0x85)
|
||||||
|
OP(CONV_OVF_U1_UN, "conv.ovf.u1.un", 0x86)
|
||||||
|
OP(CONV_OVF_U2_UN, "conv.ovf.u2.un", 0x87)
|
||||||
|
OP(CONV_OVF_U4_UN, "conv.ovf.u4.un", 0x88)
|
||||||
|
OP(CONV_OVF_U8_UN, "conv.ovf.u8.un", 0x89)
|
||||||
|
OP(CONV_OVF_I_UN, "conv.ovf.i.un", 0x8a)
|
||||||
|
OP(CONV_OVF_U_UN, "conv.ovf.u.un", 0x8b)
|
||||||
|
OP(BOX, "box", 0x8c)
|
||||||
|
OP(NEWARR, "newarr", 0x8d)
|
||||||
|
OP(LDLEN, "ldlen", 0x8e)
|
||||||
|
OP(LDELEMA, "ldelema", 0x8f)
|
||||||
|
OP(LDELEM_I1, "ldelem.i1", 0x90)
|
||||||
|
OP(LDELEM_U1, "ldelem.u1", 0x91)
|
||||||
|
OP(LDELEM_I2, "ldelem.i2", 0x92)
|
||||||
|
OP(LDELEM_U2, "ldelem.u2", 0x93)
|
||||||
|
OP(LDELEM_I4, "ldelem.i4", 0x94)
|
||||||
|
OP(LDELEM_U4, "ldelem.u4", 0x95)
|
||||||
|
OP(LDELEM_I8, "ldelem.i8", 0x96)
|
||||||
|
OP(LDELEM_I, "ldelem.i", 0x97)
|
||||||
|
OP(LDELEM_R4, "ldelem.r4", 0x98)
|
||||||
|
OP(LDELEM_R8, "ldelem.r8", 0x99)
|
||||||
|
OP(LDELEM_REF, "ldelem.ref", 0x9a)
|
||||||
|
OP(STELEM_I, "stelem.i", 0x9b)
|
||||||
|
OP(STELEM_I1, "stelem.i1", 0x9c)
|
||||||
|
OP(STELEM_I2, "stelem.i2", 0x9d)
|
||||||
|
OP(STELEM_I4, "stelem.i4", 0x9e)
|
||||||
|
OP(STELEM_I8, "stelem.i8", 0x9f)
|
||||||
|
OP(STELEM_R4, "stelem.r4", 0xa0)
|
||||||
|
OP(STELEM_R8, "stelem.r8", 0xa1)
|
||||||
|
OP(STELEM_REF, "stelem.ref", 0xa2)
|
||||||
|
OP(CONV_OVF_I1, "conv.ovf.i1", 0xb3)
|
||||||
|
OP(CONV_OVF_U1, "conv.ovf.u1", 0xb4)
|
||||||
|
OP(CONV_OVF_I2, "conv.ovf.i2", 0xb5)
|
||||||
|
OP(CONV_OVF_U2, "conv.ovf.u2", 0xb6)
|
||||||
|
OP(CONV_OVF_I4, "conv.ovf.i4", 0xb7)
|
||||||
|
OP(CONV_OVF_U4, "conv.ovf.u4", 0xb8)
|
||||||
|
OP(CONV_OVF_I8, "conv.ovf.i8", 0xb9)
|
||||||
|
OP(CONV_OVF_U8, "conv.ovf.u8", 0xba)
|
||||||
|
OP(REFANYVAL, "refanyval", 0xc2)
|
||||||
|
OP(CKFINITE, "ckfinite", 0xc3)
|
||||||
|
OP(MKREFANY, "mkrefany", 0xc6)
|
||||||
|
OP(ANN_CALL, "ann.call", 0xc7)
|
||||||
|
OP(ANN_CATCH, "ann.catch", 0xc8)
|
||||||
|
OP(ANN_DEAD, "ann.dead", 0xc9)
|
||||||
|
OP(ANN_HOISTED, "ann.hoisted", 0xca)
|
||||||
|
OP(ANN_HOISTED_CALL, "ann.hoisted.call", 0xcb)
|
||||||
|
OP(ANN_LAB, "ann.lab", 0xcc)
|
||||||
|
OP(ANN_DEF, "ann.def", 0xcd)
|
||||||
|
OP(ANN_REF_S, "ann.ref.s", 0xce)
|
||||||
|
OP(ANN_PHI, "ann.phi", 0xcf)
|
||||||
|
OP(LDTOKEN, "ldtoken", 0xd0)
|
||||||
|
OP(CONV_U2, "conv.u2", 0xd1)
|
||||||
|
OP(CONV_U1, "conv.u1", 0xd2)
|
||||||
|
OP(CONV_I, "conv.i", 0xd3)
|
||||||
|
OP(CONV_OVF_I, "conv.ovf.i", 0xd4)
|
||||||
|
OP(CONV_OVF_U, "conv.ovf.u", 0xd5)
|
||||||
|
OP(ADD_OVF, "add.ovf", 0xd6)
|
||||||
|
OP(ADD_OVF_UN, "add.ovf.un", 0xd7)
|
||||||
|
OP(MUL_OVF, "mul.ovf", 0xd8)
|
||||||
|
OP(MUL_OVF_UN, "mul.ovf.un", 0xd9)
|
||||||
|
OP(SUB_OVF, "sub.ovf", 0xda)
|
||||||
|
OP(SUB_OVF_UN, "sub.ovf.un", 0xdb)
|
||||||
|
OP(ENDFINALLY, "endfinally", 0xdc)
|
||||||
|
OP(LEAVE, "leave", 0xdd)
|
||||||
|
OP(LEAVE_S, "leave.s", 0xde)
|
||||||
|
OP(STIND_I, "stind.i", 0xdf)
|
||||||
|
OP(CONV_U, "conv.u", 0xe0)
|
||||||
|
|
||||||
|
/* prefix instructions. we use an opcode >= 256 to ease coding */
|
||||||
|
|
||||||
|
OP(ARGLIST, "arglist", 0x100)
|
||||||
|
OP(CEQ, "ceq", 0x101)
|
||||||
|
OP(CGT, "cgt", 0x102)
|
||||||
|
OP(CGT_UN, "cgt.un", 0x103)
|
||||||
|
OP(CLT, "clt", 0x104)
|
||||||
|
OP(CLT_UN, "clt.un", 0x105)
|
||||||
|
OP(LDFTN, "ldftn", 0x106)
|
||||||
|
OP(LDVIRTFTN, "ldvirtftn", 0x107)
|
||||||
|
OP(JMPI, "jmpi", 0x108)
|
||||||
|
OP(LDARG, "ldarg", 0x109)
|
||||||
|
OP(LDARGA, "ldarga", 0x10a)
|
||||||
|
OP(STARG, "starg", 0x10b)
|
||||||
|
OP(LDLOC, "ldloc", 0x10c)
|
||||||
|
OP(LDLOCA, "ldloca", 0x10d)
|
||||||
|
OP(STLOC, "stloc", 0x10e)
|
||||||
|
OP(LOCALLOC, "localloc", 0x10f)
|
||||||
|
OP(ENDFILTER, "endfilter", 0x111)
|
||||||
|
OP(UNALIGNED, "unaligned", 0x112)
|
||||||
|
OP(VOLATILE, "volatile", 0x113)
|
||||||
|
OP(TAIL, "tail", 0x114)
|
||||||
|
OP(INITOBJ, "initobj", 0x115)
|
||||||
|
OP(ANN_LIVE, "ann.live", 0x116)
|
||||||
|
OP(CPBLK, "cpblk", 0x117)
|
||||||
|
OP(INITBLK, "initblk", 0x118)
|
||||||
|
OP(ANN_REF, "ann.ref", 0x119)
|
||||||
|
OP(RETHROW, "rethrow", 0x11a)
|
||||||
|
OP(SIZEOF, "sizeof", 0x11c)
|
||||||
|
OP(REFANYTYPE, "refanytype", 0x11d)
|
||||||
|
OP(ANN_DATA, "ann.data", 0x122)
|
||||||
|
OP(ANN_ARG, "ann.arg", 0x123)
|
57
05/tcc-0.9.25/include/float.h
Normal file
57
05/tcc-0.9.25/include/float.h
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
#ifndef _FLOAT_H_
|
||||||
|
#define _FLOAT_H_
|
||||||
|
|
||||||
|
#define FLT_RADIX 2
|
||||||
|
|
||||||
|
/* IEEE float */
|
||||||
|
#define FLT_MANT_DIG 24
|
||||||
|
#define FLT_DIG 6
|
||||||
|
#define FLT_ROUNDS 1
|
||||||
|
#define FLT_EPSILON 1.19209290e-07F
|
||||||
|
#define FLT_MIN_EXP (-125)
|
||||||
|
#define FLT_MIN 1.17549435e-38F
|
||||||
|
#define FLT_MIN_10_EXP (-37)
|
||||||
|
#define FLT_MAX_EXP 128
|
||||||
|
#define FLT_MAX 3.40282347e+38F
|
||||||
|
#define FLT_MAX_10_EXP 38
|
||||||
|
|
||||||
|
/* IEEE double */
|
||||||
|
#define DBL_MANT_DIG 53
|
||||||
|
#define DBL_DIG 15
|
||||||
|
#define DBL_EPSILON 2.2204460492503131e-16
|
||||||
|
#define DBL_MIN_EXP (-1021)
|
||||||
|
#define DBL_MIN 2.2250738585072014e-308
|
||||||
|
#define DBL_MIN_10_EXP (-307)
|
||||||
|
#define DBL_MAX_EXP 1024
|
||||||
|
#define DBL_MAX 1.7976931348623157e+308
|
||||||
|
#define DBL_MAX_10_EXP 308
|
||||||
|
|
||||||
|
/* horrible intel long double */
|
||||||
|
#ifdef __i386__
|
||||||
|
|
||||||
|
#define LDBL_MANT_DIG 64
|
||||||
|
#define LDBL_DIG 18
|
||||||
|
#define LDBL_EPSILON 1.08420217248550443401e-19L
|
||||||
|
#define LDBL_MIN_EXP (-16381)
|
||||||
|
#define LDBL_MIN 3.36210314311209350626e-4932L
|
||||||
|
#define LDBL_MIN_10_EXP (-4931)
|
||||||
|
#define LDBL_MAX_EXP 16384
|
||||||
|
#define LDBL_MAX 1.18973149535723176502e+4932L
|
||||||
|
#define LDBL_MAX_10_EXP 4932
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
/* same as IEEE double */
|
||||||
|
#define LDBL_MANT_DIG 53
|
||||||
|
#define LDBL_DIG 15
|
||||||
|
#define LDBL_EPSILON 2.2204460492503131e-16
|
||||||
|
#define LDBL_MIN_EXP (-1021)
|
||||||
|
#define LDBL_MIN 2.2250738585072014e-308
|
||||||
|
#define LDBL_MIN_10_EXP (-307)
|
||||||
|
#define LDBL_MAX_EXP 1024
|
||||||
|
#define LDBL_MAX 1.7976931348623157e+308
|
||||||
|
#define LDBL_MAX_10_EXP 308
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* _FLOAT_H_ */
|
67
05/tcc-0.9.25/include/stdarg.h
Normal file
67
05/tcc-0.9.25/include/stdarg.h
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
#ifndef _STDARG_H
|
||||||
|
#define _STDARG_H
|
||||||
|
|
||||||
|
#ifdef __x86_64__
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
/* GCC compatible definition of va_list. */
|
||||||
|
struct __va_list_struct {
|
||||||
|
unsigned int gp_offset;
|
||||||
|
unsigned int fp_offset;
|
||||||
|
union {
|
||||||
|
unsigned int overflow_offset;
|
||||||
|
char *overflow_arg_area;
|
||||||
|
};
|
||||||
|
char *reg_save_area;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct __va_list_struct *va_list;
|
||||||
|
|
||||||
|
/* we use __builtin_(malloc|free) to avoid #define malloc tcc_malloc */
|
||||||
|
/* XXX: this lacks the support of aggregated types. */
|
||||||
|
#define va_start(ap, last) \
|
||||||
|
(ap = (va_list)__builtin_malloc(sizeof(struct __va_list_struct)), \
|
||||||
|
*ap = *(struct __va_list_struct*)( \
|
||||||
|
(char*)__builtin_frame_address(0) - 16), \
|
||||||
|
ap->overflow_arg_area = ((char *)__builtin_frame_address(0) + \
|
||||||
|
ap->overflow_offset), \
|
||||||
|
ap->reg_save_area = (char *)__builtin_frame_address(0) - 176 - 16 \
|
||||||
|
)
|
||||||
|
#define va_arg(ap, type) \
|
||||||
|
(*(type*)(__builtin_types_compatible_p(type, long double) \
|
||||||
|
? (ap->overflow_arg_area += 16, \
|
||||||
|
ap->overflow_arg_area - 16) \
|
||||||
|
: __builtin_types_compatible_p(type, double) \
|
||||||
|
? (ap->fp_offset < 128 + 48 \
|
||||||
|
? (ap->fp_offset += 16, \
|
||||||
|
ap->reg_save_area + ap->fp_offset - 16) \
|
||||||
|
: (ap->overflow_arg_area += 8, \
|
||||||
|
ap->overflow_arg_area - 8)) \
|
||||||
|
: (ap->gp_offset < 48 \
|
||||||
|
? (ap->gp_offset += 8, \
|
||||||
|
ap->reg_save_area + ap->gp_offset - 8) \
|
||||||
|
: (ap->overflow_arg_area += 8, \
|
||||||
|
ap->overflow_arg_area - 8)) \
|
||||||
|
))
|
||||||
|
#define va_copy(dest, src) \
|
||||||
|
((dest) = (va_list)malloc(sizeof(struct __va_list_struct)), \
|
||||||
|
*(dest) = *(src))
|
||||||
|
#define va_end(ap) __builtin_free(ap)
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
typedef char *va_list;
|
||||||
|
|
||||||
|
/* only correct for i386 */
|
||||||
|
#define va_start(ap,last) ap = ((char *)&(last)) + ((sizeof(last)+3)&~3)
|
||||||
|
#define va_arg(ap,type) (ap += (sizeof(type)+3)&~3, *(type *)(ap - ((sizeof(type)+3)&~3)))
|
||||||
|
#define va_copy(dest, src) (dest) = (src)
|
||||||
|
#define va_end(ap)
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* fix a buggy dependency on GCC in libio.h */
|
||||||
|
typedef va_list __gnuc_va_list;
|
||||||
|
#define _VA_LIST_DEFINED
|
||||||
|
|
||||||
|
#endif /* _STDARG_H */
|
10
05/tcc-0.9.25/include/stdbool.h
Normal file
10
05/tcc-0.9.25/include/stdbool.h
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#ifndef _STDBOOL_H
|
||||||
|
#define _STDBOOL_H
|
||||||
|
|
||||||
|
/* ISOC99 boolean */
|
||||||
|
|
||||||
|
#define bool _Bool
|
||||||
|
#define true 1
|
||||||
|
#define false 0
|
||||||
|
|
||||||
|
#endif /* _STDBOOL_H */
|
20
05/tcc-0.9.25/include/stddef.h
Normal file
20
05/tcc-0.9.25/include/stddef.h
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
#ifndef _STDDEF_H
|
||||||
|
#define _STDDEF_H
|
||||||
|
|
||||||
|
#define NULL ((void *)0)
|
||||||
|
typedef __SIZE_TYPE__ size_t;
|
||||||
|
typedef __WCHAR_TYPE__ wchar_t;
|
||||||
|
typedef __PTRDIFF_TYPE__ ptrdiff_t;
|
||||||
|
#define offsetof(type, field) ((size_t) &((type *)0)->field)
|
||||||
|
|
||||||
|
#ifndef __int8_t_defined
|
||||||
|
#define __int8_t_defined
|
||||||
|
typedef char int8_t;
|
||||||
|
typedef short int int16_t;
|
||||||
|
typedef int int32_t;
|
||||||
|
typedef long long int int64_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void *alloca(size_t size);
|
||||||
|
|
||||||
|
#endif
|
78
05/tcc-0.9.25/include/tcclib.h
Normal file
78
05/tcc-0.9.25/include/tcclib.h
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
/* Simple libc header for TCC
|
||||||
|
*
|
||||||
|
* Add any function you want from the libc there. This file is here
|
||||||
|
* only for your convenience so that you do not need to put the whole
|
||||||
|
* glibc include files on your floppy disk
|
||||||
|
*/
|
||||||
|
#ifndef _TCCLIB_H
|
||||||
|
#define _TCCLIB_H
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
|
||||||
|
/* stdlib.h */
|
||||||
|
void *calloc(size_t nmemb, size_t size);
|
||||||
|
void *malloc(size_t size);
|
||||||
|
void free(void *ptr);
|
||||||
|
void *realloc(void *ptr, size_t size);
|
||||||
|
int atoi(const char *nptr);
|
||||||
|
long int strtol(const char *nptr, char **endptr, int base);
|
||||||
|
unsigned long int strtoul(const char *nptr, char **endptr, int base);
|
||||||
|
void exit(int);
|
||||||
|
|
||||||
|
/* stdio.h */
|
||||||
|
typedef struct __FILE FILE;
|
||||||
|
#define EOF (-1)
|
||||||
|
extern FILE *stdin;
|
||||||
|
extern FILE *stdout;
|
||||||
|
extern FILE *stderr;
|
||||||
|
FILE *fopen(const char *path, const char *mode);
|
||||||
|
FILE *fdopen(int fildes, const char *mode);
|
||||||
|
FILE *freopen(const char *path, const char *mode, FILE *stream);
|
||||||
|
int fclose(FILE *stream);
|
||||||
|
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
|
||||||
|
size_t fwrite(void *ptr, size_t size, size_t nmemb, FILE *stream);
|
||||||
|
int fgetc(FILE *stream);
|
||||||
|
char *fgets(char *s, int size, FILE *stream);
|
||||||
|
int getc(FILE *stream);
|
||||||
|
int getchar(void);
|
||||||
|
char *gets(char *s);
|
||||||
|
int ungetc(int c, FILE *stream);
|
||||||
|
int fflush(FILE *stream);
|
||||||
|
|
||||||
|
int printf(const char *format, ...);
|
||||||
|
int fprintf(FILE *stream, const char *format, ...);
|
||||||
|
int sprintf(char *str, const char *format, ...);
|
||||||
|
int snprintf(char *str, size_t size, const char *format, ...);
|
||||||
|
int asprintf(char **strp, const char *format, ...);
|
||||||
|
int dprintf(int fd, const char *format, ...);
|
||||||
|
int vprintf(const char *format, va_list ap);
|
||||||
|
int vfprintf(FILE *stream, const char *format, va_list ap);
|
||||||
|
int vsprintf(char *str, const char *format, va_list ap);
|
||||||
|
int vsnprintf(char *str, size_t size, const char *format, va_list ap);
|
||||||
|
int vasprintf(char **strp, const char *format, va_list ap);
|
||||||
|
int vdprintf(int fd, const char *format, va_list ap);
|
||||||
|
|
||||||
|
void perror(const char *s);
|
||||||
|
|
||||||
|
/* string.h */
|
||||||
|
char *strcat(char *dest, const char *src);
|
||||||
|
char *strchr(const char *s, int c);
|
||||||
|
char *strrchr(const char *s, int c);
|
||||||
|
char *strcpy(char *dest, const char *src);
|
||||||
|
void *memcpy(void *dest, const void *src, size_t n);
|
||||||
|
void *memmove(void *dest, const void *src, size_t n);
|
||||||
|
void *memset(void *s, int c, size_t n);
|
||||||
|
char *strdup(const char *s);
|
||||||
|
|
||||||
|
/* dlfcn.h */
|
||||||
|
#define RTLD_LAZY 0x001
|
||||||
|
#define RTLD_NOW 0x002
|
||||||
|
#define RTLD_GLOBAL 0x100
|
||||||
|
|
||||||
|
void *dlopen(const char *filename, int flag);
|
||||||
|
const char *dlerror(void);
|
||||||
|
void *dlsym(void *handle, char *symbol);
|
||||||
|
int dlclose(void *handle);
|
||||||
|
|
||||||
|
#endif /* _TCCLIB_H */
|
11
05/tcc-0.9.25/include/varargs.h
Normal file
11
05/tcc-0.9.25/include/varargs.h
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
#ifndef _VARARGS_H
|
||||||
|
#define _VARARGS_H
|
||||||
|
|
||||||
|
#include <stdarg.h>
|
||||||
|
|
||||||
|
#define va_dcl
|
||||||
|
#define va_alist __va_alist
|
||||||
|
#undef va_start
|
||||||
|
#define va_start(ap) ap = __builtin_varargs_start
|
||||||
|
|
||||||
|
#endif
|
45
05/tcc-0.9.25/lib/alloca86-bt.S
Normal file
45
05/tcc-0.9.25/lib/alloca86-bt.S
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/* ---------------------------------------------- */
|
||||||
|
/* alloca86b.S */
|
||||||
|
|
||||||
|
#include "../config.h"
|
||||||
|
|
||||||
|
.globl __bound_alloca
|
||||||
|
|
||||||
|
__bound_alloca:
|
||||||
|
pop %edx
|
||||||
|
pop %eax
|
||||||
|
mov %eax, %ecx
|
||||||
|
add $3,%eax
|
||||||
|
and $-4,%eax
|
||||||
|
jz p6
|
||||||
|
|
||||||
|
#ifdef TCC_TARGET_PE
|
||||||
|
p4:
|
||||||
|
cmp $4096,%eax
|
||||||
|
jle p5
|
||||||
|
sub $4096,%esp
|
||||||
|
sub $4096,%eax
|
||||||
|
test %eax,(%esp)
|
||||||
|
jmp p4
|
||||||
|
|
||||||
|
p5:
|
||||||
|
#endif
|
||||||
|
|
||||||
|
sub %eax,%esp
|
||||||
|
mov %esp,%eax
|
||||||
|
|
||||||
|
push %edx
|
||||||
|
push %eax
|
||||||
|
push %ecx
|
||||||
|
push %eax
|
||||||
|
call __bound_new_region
|
||||||
|
add $8, %esp
|
||||||
|
pop %eax
|
||||||
|
pop %edx
|
||||||
|
|
||||||
|
p6:
|
||||||
|
push %edx
|
||||||
|
push %edx
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ---------------------------------------------- */
|
33
05/tcc-0.9.25/lib/alloca86.S
Normal file
33
05/tcc-0.9.25/lib/alloca86.S
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
/* ---------------------------------------------- */
|
||||||
|
/* alloca86.S */
|
||||||
|
|
||||||
|
#include "../config.h"
|
||||||
|
|
||||||
|
.globl alloca
|
||||||
|
|
||||||
|
alloca:
|
||||||
|
pop %edx
|
||||||
|
pop %eax
|
||||||
|
add $3,%eax
|
||||||
|
and $-4,%eax
|
||||||
|
jz p3
|
||||||
|
|
||||||
|
#ifdef TCC_TARGET_PE
|
||||||
|
p1:
|
||||||
|
cmp $4096,%eax
|
||||||
|
jle p2
|
||||||
|
sub $4096,%esp
|
||||||
|
sub $4096,%eax
|
||||||
|
test %eax,(%esp)
|
||||||
|
jmp p1
|
||||||
|
p2:
|
||||||
|
#endif
|
||||||
|
|
||||||
|
sub %eax,%esp
|
||||||
|
mov %esp,%eax
|
||||||
|
p3:
|
||||||
|
push %edx
|
||||||
|
push %edx
|
||||||
|
ret
|
||||||
|
|
||||||
|
/* ---------------------------------------------- */
|
868
05/tcc-0.9.25/lib/bcheck.c
Normal file
868
05/tcc-0.9.25/lib/bcheck.c
Normal file
|
@ -0,0 +1,868 @@
|
||||||
|
/*
|
||||||
|
* Tiny C Memory and bounds checker
|
||||||
|
*
|
||||||
|
* Copyright (c) 2002 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <string.h>
|
||||||
|
#if !defined(__FreeBSD__) && !defined(__DragonFly__) && !defined(__OpenBSD__)
|
||||||
|
#include <malloc.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//#define BOUND_DEBUG
|
||||||
|
|
||||||
|
/* define so that bound array is static (faster, but use memory if
|
||||||
|
bound checking not used) */
|
||||||
|
//#define BOUND_STATIC
|
||||||
|
|
||||||
|
/* use malloc hooks. Currently the code cannot be reliable if no hooks */
|
||||||
|
#define CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
|
||||||
|
#define HAVE_MEMALIGN
|
||||||
|
|
||||||
|
#if defined(__FreeBSD__) || defined(__DragonFly__) || defined(__dietlibc__) \
|
||||||
|
|| defined(__UCLIBC__) || defined(__OpenBSD__)
|
||||||
|
#warning Bound checking not fully supported in this environment.
|
||||||
|
#undef CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
#undef HAVE_MEMALIGN
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define BOUND_T1_BITS 13
|
||||||
|
#define BOUND_T2_BITS 11
|
||||||
|
#define BOUND_T3_BITS (32 - BOUND_T1_BITS - BOUND_T2_BITS)
|
||||||
|
|
||||||
|
#define BOUND_T1_SIZE (1 << BOUND_T1_BITS)
|
||||||
|
#define BOUND_T2_SIZE (1 << BOUND_T2_BITS)
|
||||||
|
#define BOUND_T3_SIZE (1 << BOUND_T3_BITS)
|
||||||
|
#define BOUND_E_BITS 4
|
||||||
|
|
||||||
|
#define BOUND_T23_BITS (BOUND_T2_BITS + BOUND_T3_BITS)
|
||||||
|
#define BOUND_T23_SIZE (1 << BOUND_T23_BITS)
|
||||||
|
|
||||||
|
|
||||||
|
/* this pointer is generated when bound check is incorrect */
|
||||||
|
#define INVALID_POINTER ((void *)(-2))
|
||||||
|
/* size of an empty region */
|
||||||
|
#define EMPTY_SIZE 0xffffffff
|
||||||
|
/* size of an invalid region */
|
||||||
|
#define INVALID_SIZE 0
|
||||||
|
|
||||||
|
typedef struct BoundEntry {
|
||||||
|
unsigned long start;
|
||||||
|
unsigned long size;
|
||||||
|
struct BoundEntry *next;
|
||||||
|
unsigned long is_invalid; /* true if pointers outside region are invalid */
|
||||||
|
} BoundEntry;
|
||||||
|
|
||||||
|
/* external interface */
|
||||||
|
void __bound_init(void);
|
||||||
|
void __bound_new_region(void *p, unsigned long size);
|
||||||
|
int __bound_delete_region(void *p);
|
||||||
|
|
||||||
|
#define FASTCALL __attribute__((regparm(3)))
|
||||||
|
|
||||||
|
void *__bound_malloc(size_t size, const void *caller);
|
||||||
|
void *__bound_memalign(size_t size, size_t align, const void *caller);
|
||||||
|
void __bound_free(void *ptr, const void *caller);
|
||||||
|
void *__bound_realloc(void *ptr, size_t size, const void *caller);
|
||||||
|
static void *libc_malloc(size_t size);
|
||||||
|
static void libc_free(void *ptr);
|
||||||
|
static void install_malloc_hooks(void);
|
||||||
|
static void restore_malloc_hooks(void);
|
||||||
|
|
||||||
|
#ifdef CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
static void *saved_malloc_hook;
|
||||||
|
static void *saved_free_hook;
|
||||||
|
static void *saved_realloc_hook;
|
||||||
|
static void *saved_memalign_hook;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* linker definitions */
|
||||||
|
extern char _end;
|
||||||
|
|
||||||
|
/* TCC definitions */
|
||||||
|
extern char __bounds_start; /* start of static bounds table */
|
||||||
|
/* error message, just for TCC */
|
||||||
|
const char *__bound_error_msg;
|
||||||
|
|
||||||
|
/* runtime error output */
|
||||||
|
extern void rt_error(unsigned long pc, const char *fmt, ...);
|
||||||
|
|
||||||
|
#ifdef BOUND_STATIC
|
||||||
|
static BoundEntry *__bound_t1[BOUND_T1_SIZE]; /* page table */
|
||||||
|
#else
|
||||||
|
static BoundEntry **__bound_t1; /* page table */
|
||||||
|
#endif
|
||||||
|
static BoundEntry *__bound_empty_t2; /* empty page, for unused pages */
|
||||||
|
static BoundEntry *__bound_invalid_t2; /* invalid page, for invalid pointers */
|
||||||
|
|
||||||
|
static BoundEntry *__bound_find_region(BoundEntry *e1, void *p)
|
||||||
|
{
|
||||||
|
unsigned long addr, tmp;
|
||||||
|
BoundEntry *e;
|
||||||
|
|
||||||
|
e = e1;
|
||||||
|
while (e != NULL) {
|
||||||
|
addr = (unsigned long)p;
|
||||||
|
addr -= e->start;
|
||||||
|
if (addr <= e->size) {
|
||||||
|
/* put region at the head */
|
||||||
|
tmp = e1->start;
|
||||||
|
e1->start = e->start;
|
||||||
|
e->start = tmp;
|
||||||
|
tmp = e1->size;
|
||||||
|
e1->size = e->size;
|
||||||
|
e->size = tmp;
|
||||||
|
return e1;
|
||||||
|
}
|
||||||
|
e = e->next;
|
||||||
|
}
|
||||||
|
/* no entry found: return empty entry or invalid entry */
|
||||||
|
if (e1->is_invalid)
|
||||||
|
return __bound_invalid_t2;
|
||||||
|
else
|
||||||
|
return __bound_empty_t2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* print a bound error message */
|
||||||
|
static void bound_error(const char *fmt, ...)
|
||||||
|
{
|
||||||
|
__bound_error_msg = fmt;
|
||||||
|
*(int *)0 = 0; /* force a runtime error */
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bound_alloc_error(void)
|
||||||
|
{
|
||||||
|
bound_error("not enough memory for bound checking code");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* currently, tcc cannot compile that because we use GNUC extensions */
|
||||||
|
#if !defined(__TINYC__)
|
||||||
|
|
||||||
|
/* return '(p + offset)' for pointer arithmetic (a pointer can reach
|
||||||
|
the end of a region in this case */
|
||||||
|
void * FASTCALL __bound_ptr_add(void *p, int offset)
|
||||||
|
{
|
||||||
|
unsigned long addr = (unsigned long)p;
|
||||||
|
BoundEntry *e;
|
||||||
|
#if defined(BOUND_DEBUG)
|
||||||
|
printf("add: 0x%x %d\n", (int)p, offset);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
e = __bound_t1[addr >> (BOUND_T2_BITS + BOUND_T3_BITS)];
|
||||||
|
e = (BoundEntry *)((char *)e +
|
||||||
|
((addr >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS)));
|
||||||
|
addr -= e->start;
|
||||||
|
if (addr > e->size) {
|
||||||
|
e = __bound_find_region(e, p);
|
||||||
|
addr = (unsigned long)p - e->start;
|
||||||
|
}
|
||||||
|
addr += offset;
|
||||||
|
if (addr > e->size)
|
||||||
|
return INVALID_POINTER; /* return an invalid pointer */
|
||||||
|
return p + offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* return '(p + offset)' for pointer indirection (the resulting must
|
||||||
|
be strictly inside the region */
|
||||||
|
#define BOUND_PTR_INDIR(dsize) \
|
||||||
|
void * FASTCALL __bound_ptr_indir ## dsize (void *p, int offset) \
|
||||||
|
{ \
|
||||||
|
unsigned long addr = (unsigned long)p; \
|
||||||
|
BoundEntry *e; \
|
||||||
|
\
|
||||||
|
e = __bound_t1[addr >> (BOUND_T2_BITS + BOUND_T3_BITS)]; \
|
||||||
|
e = (BoundEntry *)((char *)e + \
|
||||||
|
((addr >> (BOUND_T3_BITS - BOUND_E_BITS)) & \
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS))); \
|
||||||
|
addr -= e->start; \
|
||||||
|
if (addr > e->size) { \
|
||||||
|
e = __bound_find_region(e, p); \
|
||||||
|
addr = (unsigned long)p - e->start; \
|
||||||
|
} \
|
||||||
|
addr += offset + dsize; \
|
||||||
|
if (addr > e->size) \
|
||||||
|
return INVALID_POINTER; /* return an invalid pointer */ \
|
||||||
|
return p + offset; \
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef __i386__
|
||||||
|
/* return the frame pointer of the caller */
|
||||||
|
#define GET_CALLER_FP(fp)\
|
||||||
|
{\
|
||||||
|
unsigned long *fp1;\
|
||||||
|
__asm__ __volatile__ ("movl %%ebp,%0" :"=g" (fp1));\
|
||||||
|
fp = fp1[0];\
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
#error put code to extract the calling frame pointer
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* called when entering a function to add all the local regions */
|
||||||
|
void FASTCALL __bound_local_new(void *p1)
|
||||||
|
{
|
||||||
|
unsigned long addr, size, fp, *p = p1;
|
||||||
|
GET_CALLER_FP(fp);
|
||||||
|
for(;;) {
|
||||||
|
addr = p[0];
|
||||||
|
if (addr == 0)
|
||||||
|
break;
|
||||||
|
addr += fp;
|
||||||
|
size = p[1];
|
||||||
|
p += 2;
|
||||||
|
__bound_new_region((void *)addr, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* called when leaving a function to delete all the local regions */
|
||||||
|
void FASTCALL __bound_local_delete(void *p1)
|
||||||
|
{
|
||||||
|
unsigned long addr, fp, *p = p1;
|
||||||
|
GET_CALLER_FP(fp);
|
||||||
|
for(;;) {
|
||||||
|
addr = p[0];
|
||||||
|
if (addr == 0)
|
||||||
|
break;
|
||||||
|
addr += fp;
|
||||||
|
p += 2;
|
||||||
|
__bound_delete_region((void *)addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
void __bound_local_new(void *p)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
void __bound_local_delete(void *p)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_ptr_add(void *p, int offset)
|
||||||
|
{
|
||||||
|
return p + offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define BOUND_PTR_INDIR(dsize) \
|
||||||
|
void *__bound_ptr_indir ## dsize (void *p, int offset) \
|
||||||
|
{ \
|
||||||
|
return p + offset; \
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
BOUND_PTR_INDIR(1)
|
||||||
|
BOUND_PTR_INDIR(2)
|
||||||
|
BOUND_PTR_INDIR(4)
|
||||||
|
BOUND_PTR_INDIR(8)
|
||||||
|
BOUND_PTR_INDIR(12)
|
||||||
|
BOUND_PTR_INDIR(16)
|
||||||
|
|
||||||
|
static BoundEntry *__bound_new_page(void)
|
||||||
|
{
|
||||||
|
BoundEntry *page;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
page = libc_malloc(sizeof(BoundEntry) * BOUND_T2_SIZE);
|
||||||
|
if (!page)
|
||||||
|
bound_alloc_error();
|
||||||
|
for(i=0;i<BOUND_T2_SIZE;i++) {
|
||||||
|
/* put empty entries */
|
||||||
|
page[i].start = 0;
|
||||||
|
page[i].size = EMPTY_SIZE;
|
||||||
|
page[i].next = NULL;
|
||||||
|
page[i].is_invalid = 0;
|
||||||
|
}
|
||||||
|
return page;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* currently we use malloc(). Should use bound_new_page() */
|
||||||
|
static BoundEntry *bound_new_entry(void)
|
||||||
|
{
|
||||||
|
BoundEntry *e;
|
||||||
|
e = libc_malloc(sizeof(BoundEntry));
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void bound_free_entry(BoundEntry *e)
|
||||||
|
{
|
||||||
|
libc_free(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline BoundEntry *get_page(int index)
|
||||||
|
{
|
||||||
|
BoundEntry *page;
|
||||||
|
page = __bound_t1[index];
|
||||||
|
if (page == __bound_empty_t2 || page == __bound_invalid_t2) {
|
||||||
|
/* create a new page if necessary */
|
||||||
|
page = __bound_new_page();
|
||||||
|
__bound_t1[index] = page;
|
||||||
|
}
|
||||||
|
return page;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* mark a region as being invalid (can only be used during init) */
|
||||||
|
static void mark_invalid(unsigned long addr, unsigned long size)
|
||||||
|
{
|
||||||
|
unsigned long start, end;
|
||||||
|
BoundEntry *page;
|
||||||
|
int t1_start, t1_end, i, j, t2_start, t2_end;
|
||||||
|
|
||||||
|
start = addr;
|
||||||
|
end = addr + size;
|
||||||
|
|
||||||
|
t2_start = (start + BOUND_T3_SIZE - 1) >> BOUND_T3_BITS;
|
||||||
|
if (end != 0)
|
||||||
|
t2_end = end >> BOUND_T3_BITS;
|
||||||
|
else
|
||||||
|
t2_end = 1 << (BOUND_T1_BITS + BOUND_T2_BITS);
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
printf("mark_invalid: start = %x %x\n", t2_start, t2_end);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* first we handle full pages */
|
||||||
|
t1_start = (t2_start + BOUND_T2_SIZE - 1) >> BOUND_T2_BITS;
|
||||||
|
t1_end = t2_end >> BOUND_T2_BITS;
|
||||||
|
|
||||||
|
i = t2_start & (BOUND_T2_SIZE - 1);
|
||||||
|
j = t2_end & (BOUND_T2_SIZE - 1);
|
||||||
|
|
||||||
|
if (t1_start == t1_end) {
|
||||||
|
page = get_page(t2_start >> BOUND_T2_BITS);
|
||||||
|
for(; i < j; i++) {
|
||||||
|
page[i].size = INVALID_SIZE;
|
||||||
|
page[i].is_invalid = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (i > 0) {
|
||||||
|
page = get_page(t2_start >> BOUND_T2_BITS);
|
||||||
|
for(; i < BOUND_T2_SIZE; i++) {
|
||||||
|
page[i].size = INVALID_SIZE;
|
||||||
|
page[i].is_invalid = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for(i = t1_start; i < t1_end; i++) {
|
||||||
|
__bound_t1[i] = __bound_invalid_t2;
|
||||||
|
}
|
||||||
|
if (j != 0) {
|
||||||
|
page = get_page(t1_end);
|
||||||
|
for(i = 0; i < j; i++) {
|
||||||
|
page[i].size = INVALID_SIZE;
|
||||||
|
page[i].is_invalid = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void __bound_init(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
BoundEntry *page;
|
||||||
|
unsigned long start, size;
|
||||||
|
int *p;
|
||||||
|
|
||||||
|
/* save malloc hooks and install bound check hooks */
|
||||||
|
install_malloc_hooks();
|
||||||
|
|
||||||
|
#ifndef BOUND_STATIC
|
||||||
|
__bound_t1 = libc_malloc(BOUND_T1_SIZE * sizeof(BoundEntry *));
|
||||||
|
if (!__bound_t1)
|
||||||
|
bound_alloc_error();
|
||||||
|
#endif
|
||||||
|
__bound_empty_t2 = __bound_new_page();
|
||||||
|
for(i=0;i<BOUND_T1_SIZE;i++) {
|
||||||
|
__bound_t1[i] = __bound_empty_t2;
|
||||||
|
}
|
||||||
|
|
||||||
|
page = __bound_new_page();
|
||||||
|
for(i=0;i<BOUND_T2_SIZE;i++) {
|
||||||
|
/* put invalid entries */
|
||||||
|
page[i].start = 0;
|
||||||
|
page[i].size = INVALID_SIZE;
|
||||||
|
page[i].next = NULL;
|
||||||
|
page[i].is_invalid = 1;
|
||||||
|
}
|
||||||
|
__bound_invalid_t2 = page;
|
||||||
|
|
||||||
|
/* invalid pointer zone */
|
||||||
|
start = (unsigned long)INVALID_POINTER & ~(BOUND_T23_SIZE - 1);
|
||||||
|
size = BOUND_T23_SIZE;
|
||||||
|
mark_invalid(start, size);
|
||||||
|
|
||||||
|
#if !defined(__TINYC__) && defined(CONFIG_TCC_MALLOC_HOOKS)
|
||||||
|
/* malloc zone is also marked invalid. can only use that with
|
||||||
|
hooks because all libs should use the same malloc. The solution
|
||||||
|
would be to build a new malloc for tcc. */
|
||||||
|
start = (unsigned long)&_end;
|
||||||
|
size = 128 * 0x100000;
|
||||||
|
mark_invalid(start, size);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* add all static bound check values */
|
||||||
|
p = (int *)&__bounds_start;
|
||||||
|
while (p[0] != 0) {
|
||||||
|
__bound_new_region((void *)p[0], p[1]);
|
||||||
|
p += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void add_region(BoundEntry *e,
|
||||||
|
unsigned long start, unsigned long size)
|
||||||
|
{
|
||||||
|
BoundEntry *e1;
|
||||||
|
if (e->start == 0) {
|
||||||
|
/* no region : add it */
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
} else {
|
||||||
|
/* already regions in the list: add it at the head */
|
||||||
|
e1 = bound_new_entry();
|
||||||
|
e1->start = e->start;
|
||||||
|
e1->size = e->size;
|
||||||
|
e1->next = e->next;
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
e->next = e1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* create a new region. It should not already exist in the region list */
|
||||||
|
void __bound_new_region(void *p, unsigned long size)
|
||||||
|
{
|
||||||
|
unsigned long start, end;
|
||||||
|
BoundEntry *page, *e, *e2;
|
||||||
|
int t1_start, t1_end, i, t2_start, t2_end;
|
||||||
|
|
||||||
|
start = (unsigned long)p;
|
||||||
|
end = start + size;
|
||||||
|
t1_start = start >> (BOUND_T2_BITS + BOUND_T3_BITS);
|
||||||
|
t1_end = end >> (BOUND_T2_BITS + BOUND_T3_BITS);
|
||||||
|
|
||||||
|
/* start */
|
||||||
|
page = get_page(t1_start);
|
||||||
|
t2_start = (start >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS);
|
||||||
|
t2_end = (end >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS);
|
||||||
|
#ifdef BOUND_DEBUG
|
||||||
|
printf("new %lx %lx %x %x %x %x\n",
|
||||||
|
start, end, t1_start, t1_end, t2_start, t2_end);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
e = (BoundEntry *)((char *)page + t2_start);
|
||||||
|
add_region(e, start, size);
|
||||||
|
|
||||||
|
if (t1_end == t1_start) {
|
||||||
|
/* same ending page */
|
||||||
|
e2 = (BoundEntry *)((char *)page + t2_end);
|
||||||
|
if (e2 > e) {
|
||||||
|
e++;
|
||||||
|
for(;e<e2;e++) {
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
}
|
||||||
|
add_region(e, start, size);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* mark until end of page */
|
||||||
|
e2 = page + BOUND_T2_SIZE;
|
||||||
|
e++;
|
||||||
|
for(;e<e2;e++) {
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
}
|
||||||
|
/* mark intermediate pages, if any */
|
||||||
|
for(i=t1_start+1;i<t1_end;i++) {
|
||||||
|
page = get_page(i);
|
||||||
|
e2 = page + BOUND_T2_SIZE;
|
||||||
|
for(e=page;e<e2;e++) {
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* last page */
|
||||||
|
page = get_page(t1_end);
|
||||||
|
e2 = (BoundEntry *)((char *)page + t2_end);
|
||||||
|
for(e=page;e<e2;e++) {
|
||||||
|
e->start = start;
|
||||||
|
e->size = size;
|
||||||
|
}
|
||||||
|
add_region(e, start, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* delete a region */
|
||||||
|
static inline void delete_region(BoundEntry *e,
|
||||||
|
void *p, unsigned long empty_size)
|
||||||
|
{
|
||||||
|
unsigned long addr;
|
||||||
|
BoundEntry *e1;
|
||||||
|
|
||||||
|
addr = (unsigned long)p;
|
||||||
|
addr -= e->start;
|
||||||
|
if (addr <= e->size) {
|
||||||
|
/* region found is first one */
|
||||||
|
e1 = e->next;
|
||||||
|
if (e1 == NULL) {
|
||||||
|
/* no more region: mark it empty */
|
||||||
|
e->start = 0;
|
||||||
|
e->size = empty_size;
|
||||||
|
} else {
|
||||||
|
/* copy next region in head */
|
||||||
|
e->start = e1->start;
|
||||||
|
e->size = e1->size;
|
||||||
|
e->next = e1->next;
|
||||||
|
bound_free_entry(e1);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* find the matching region */
|
||||||
|
for(;;) {
|
||||||
|
e1 = e;
|
||||||
|
e = e->next;
|
||||||
|
/* region not found: do nothing */
|
||||||
|
if (e == NULL)
|
||||||
|
break;
|
||||||
|
addr = (unsigned long)p - e->start;
|
||||||
|
if (addr <= e->size) {
|
||||||
|
/* found: remove entry */
|
||||||
|
e1->next = e->next;
|
||||||
|
bound_free_entry(e);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* WARNING: 'p' must be the starting point of the region. */
|
||||||
|
/* return non zero if error */
|
||||||
|
int __bound_delete_region(void *p)
|
||||||
|
{
|
||||||
|
unsigned long start, end, addr, size, empty_size;
|
||||||
|
BoundEntry *page, *e, *e2;
|
||||||
|
int t1_start, t1_end, t2_start, t2_end, i;
|
||||||
|
|
||||||
|
start = (unsigned long)p;
|
||||||
|
t1_start = start >> (BOUND_T2_BITS + BOUND_T3_BITS);
|
||||||
|
t2_start = (start >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS);
|
||||||
|
|
||||||
|
/* find region size */
|
||||||
|
page = __bound_t1[t1_start];
|
||||||
|
e = (BoundEntry *)((char *)page + t2_start);
|
||||||
|
addr = start - e->start;
|
||||||
|
if (addr > e->size)
|
||||||
|
e = __bound_find_region(e, p);
|
||||||
|
/* test if invalid region */
|
||||||
|
if (e->size == EMPTY_SIZE || (unsigned long)p != e->start)
|
||||||
|
return -1;
|
||||||
|
/* compute the size we put in invalid regions */
|
||||||
|
if (e->is_invalid)
|
||||||
|
empty_size = INVALID_SIZE;
|
||||||
|
else
|
||||||
|
empty_size = EMPTY_SIZE;
|
||||||
|
size = e->size;
|
||||||
|
end = start + size;
|
||||||
|
|
||||||
|
/* now we can free each entry */
|
||||||
|
t1_end = end >> (BOUND_T2_BITS + BOUND_T3_BITS);
|
||||||
|
t2_end = (end >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS);
|
||||||
|
|
||||||
|
delete_region(e, p, empty_size);
|
||||||
|
if (t1_end == t1_start) {
|
||||||
|
/* same ending page */
|
||||||
|
e2 = (BoundEntry *)((char *)page + t2_end);
|
||||||
|
if (e2 > e) {
|
||||||
|
e++;
|
||||||
|
for(;e<e2;e++) {
|
||||||
|
e->start = 0;
|
||||||
|
e->size = empty_size;
|
||||||
|
}
|
||||||
|
delete_region(e, p, empty_size);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* mark until end of page */
|
||||||
|
e2 = page + BOUND_T2_SIZE;
|
||||||
|
e++;
|
||||||
|
for(;e<e2;e++) {
|
||||||
|
e->start = 0;
|
||||||
|
e->size = empty_size;
|
||||||
|
}
|
||||||
|
/* mark intermediate pages, if any */
|
||||||
|
/* XXX: should free them */
|
||||||
|
for(i=t1_start+1;i<t1_end;i++) {
|
||||||
|
page = get_page(i);
|
||||||
|
e2 = page + BOUND_T2_SIZE;
|
||||||
|
for(e=page;e<e2;e++) {
|
||||||
|
e->start = 0;
|
||||||
|
e->size = empty_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* last page */
|
||||||
|
page = get_page(t2_end);
|
||||||
|
e2 = (BoundEntry *)((char *)page + t2_end);
|
||||||
|
for(e=page;e<e2;e++) {
|
||||||
|
e->start = 0;
|
||||||
|
e->size = empty_size;
|
||||||
|
}
|
||||||
|
delete_region(e, p, empty_size);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* return the size of the region starting at p, or EMPTY_SIZE if non
|
||||||
|
existant region. */
|
||||||
|
static unsigned long get_region_size(void *p)
|
||||||
|
{
|
||||||
|
unsigned long addr = (unsigned long)p;
|
||||||
|
BoundEntry *e;
|
||||||
|
|
||||||
|
e = __bound_t1[addr >> (BOUND_T2_BITS + BOUND_T3_BITS)];
|
||||||
|
e = (BoundEntry *)((char *)e +
|
||||||
|
((addr >> (BOUND_T3_BITS - BOUND_E_BITS)) &
|
||||||
|
((BOUND_T2_SIZE - 1) << BOUND_E_BITS)));
|
||||||
|
addr -= e->start;
|
||||||
|
if (addr > e->size)
|
||||||
|
e = __bound_find_region(e, p);
|
||||||
|
if (e->start != (unsigned long)p)
|
||||||
|
return EMPTY_SIZE;
|
||||||
|
return e->size;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* patched memory functions */
|
||||||
|
|
||||||
|
static void install_malloc_hooks(void)
|
||||||
|
{
|
||||||
|
#ifdef CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
saved_malloc_hook = __malloc_hook;
|
||||||
|
saved_free_hook = __free_hook;
|
||||||
|
saved_realloc_hook = __realloc_hook;
|
||||||
|
saved_memalign_hook = __memalign_hook;
|
||||||
|
__malloc_hook = __bound_malloc;
|
||||||
|
__free_hook = __bound_free;
|
||||||
|
__realloc_hook = __bound_realloc;
|
||||||
|
__memalign_hook = __bound_memalign;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static void restore_malloc_hooks(void)
|
||||||
|
{
|
||||||
|
#ifdef CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
__malloc_hook = saved_malloc_hook;
|
||||||
|
__free_hook = saved_free_hook;
|
||||||
|
__realloc_hook = saved_realloc_hook;
|
||||||
|
__memalign_hook = saved_memalign_hook;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static void *libc_malloc(size_t size)
|
||||||
|
{
|
||||||
|
void *ptr;
|
||||||
|
restore_malloc_hooks();
|
||||||
|
ptr = malloc(size);
|
||||||
|
install_malloc_hooks();
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void libc_free(void *ptr)
|
||||||
|
{
|
||||||
|
restore_malloc_hooks();
|
||||||
|
free(ptr);
|
||||||
|
install_malloc_hooks();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: we should use a malloc which ensure that it is unlikely that
|
||||||
|
two malloc'ed data have the same address if 'free' are made in
|
||||||
|
between. */
|
||||||
|
void *__bound_malloc(size_t size, const void *caller)
|
||||||
|
{
|
||||||
|
void *ptr;
|
||||||
|
|
||||||
|
/* we allocate one more byte to ensure the regions will be
|
||||||
|
separated by at least one byte. With the glibc malloc, it may
|
||||||
|
be in fact not necessary */
|
||||||
|
ptr = libc_malloc(size + 1);
|
||||||
|
|
||||||
|
if (!ptr)
|
||||||
|
return NULL;
|
||||||
|
__bound_new_region(ptr, size);
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_memalign(size_t size, size_t align, const void *caller)
|
||||||
|
{
|
||||||
|
void *ptr;
|
||||||
|
|
||||||
|
restore_malloc_hooks();
|
||||||
|
|
||||||
|
#ifndef HAVE_MEMALIGN
|
||||||
|
if (align > 4) {
|
||||||
|
/* XXX: handle it ? */
|
||||||
|
ptr = NULL;
|
||||||
|
} else {
|
||||||
|
/* we suppose that malloc aligns to at least four bytes */
|
||||||
|
ptr = malloc(size + 1);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
/* we allocate one more byte to ensure the regions will be
|
||||||
|
separated by at least one byte. With the glibc malloc, it may
|
||||||
|
be in fact not necessary */
|
||||||
|
ptr = memalign(size + 1, align);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
install_malloc_hooks();
|
||||||
|
|
||||||
|
if (!ptr)
|
||||||
|
return NULL;
|
||||||
|
__bound_new_region(ptr, size);
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void __bound_free(void *ptr, const void *caller)
|
||||||
|
{
|
||||||
|
if (ptr == NULL)
|
||||||
|
return;
|
||||||
|
if (__bound_delete_region(ptr) != 0)
|
||||||
|
bound_error("freeing invalid region");
|
||||||
|
|
||||||
|
libc_free(ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_realloc(void *ptr, size_t size, const void *caller)
|
||||||
|
{
|
||||||
|
void *ptr1;
|
||||||
|
int old_size;
|
||||||
|
|
||||||
|
if (size == 0) {
|
||||||
|
__bound_free(ptr, caller);
|
||||||
|
return NULL;
|
||||||
|
} else {
|
||||||
|
ptr1 = __bound_malloc(size, caller);
|
||||||
|
if (ptr == NULL || ptr1 == NULL)
|
||||||
|
return ptr1;
|
||||||
|
old_size = get_region_size(ptr);
|
||||||
|
if (old_size == EMPTY_SIZE)
|
||||||
|
bound_error("realloc'ing invalid pointer");
|
||||||
|
memcpy(ptr1, ptr, old_size);
|
||||||
|
__bound_free(ptr, caller);
|
||||||
|
return ptr1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef CONFIG_TCC_MALLOC_HOOKS
|
||||||
|
void *__bound_calloc(size_t nmemb, size_t size)
|
||||||
|
{
|
||||||
|
void *ptr;
|
||||||
|
size = size * nmemb;
|
||||||
|
ptr = __bound_malloc(size, NULL);
|
||||||
|
if (!ptr)
|
||||||
|
return NULL;
|
||||||
|
memset(ptr, 0, size);
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
static void bound_dump(void)
|
||||||
|
{
|
||||||
|
BoundEntry *page, *e;
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
printf("region dump:\n");
|
||||||
|
for(i=0;i<BOUND_T1_SIZE;i++) {
|
||||||
|
page = __bound_t1[i];
|
||||||
|
for(j=0;j<BOUND_T2_SIZE;j++) {
|
||||||
|
e = page + j;
|
||||||
|
/* do not print invalid or empty entries */
|
||||||
|
if (e->size != EMPTY_SIZE && e->start != 0) {
|
||||||
|
printf("%08x:",
|
||||||
|
(i << (BOUND_T2_BITS + BOUND_T3_BITS)) +
|
||||||
|
(j << BOUND_T3_BITS));
|
||||||
|
do {
|
||||||
|
printf(" %08lx:%08lx", e->start, e->start + e->size);
|
||||||
|
e = e->next;
|
||||||
|
} while (e != NULL);
|
||||||
|
printf("\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* some useful checked functions */
|
||||||
|
|
||||||
|
/* check that (p ... p + size - 1) lies inside 'p' region, if any */
|
||||||
|
static void __bound_check(const void *p, size_t size)
|
||||||
|
{
|
||||||
|
if (size == 0)
|
||||||
|
return;
|
||||||
|
p = __bound_ptr_add((void *)p, size);
|
||||||
|
if (p == INVALID_POINTER)
|
||||||
|
bound_error("invalid pointer");
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_memcpy(void *dst, const void *src, size_t size)
|
||||||
|
{
|
||||||
|
__bound_check(dst, size);
|
||||||
|
__bound_check(src, size);
|
||||||
|
/* check also region overlap */
|
||||||
|
if (src >= dst && src < dst + size)
|
||||||
|
bound_error("overlapping regions in memcpy()");
|
||||||
|
return memcpy(dst, src, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_memmove(void *dst, const void *src, size_t size)
|
||||||
|
{
|
||||||
|
__bound_check(dst, size);
|
||||||
|
__bound_check(src, size);
|
||||||
|
return memmove(dst, src, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *__bound_memset(void *dst, int c, size_t size)
|
||||||
|
{
|
||||||
|
__bound_check(dst, size);
|
||||||
|
return memset(dst, c, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: could be optimized */
|
||||||
|
int __bound_strlen(const char *s)
|
||||||
|
{
|
||||||
|
const char *p;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
len = 0;
|
||||||
|
for(;;) {
|
||||||
|
p = __bound_ptr_indir1((char *)s, len);
|
||||||
|
if (p == INVALID_POINTER)
|
||||||
|
bound_error("bad pointer in strlen()");
|
||||||
|
if (*p == '\0')
|
||||||
|
break;
|
||||||
|
len++;
|
||||||
|
}
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *__bound_strcpy(char *dst, const char *src)
|
||||||
|
{
|
||||||
|
int len;
|
||||||
|
len = __bound_strlen(src);
|
||||||
|
return __bound_memcpy(dst, src, len + 1);
|
||||||
|
}
|
||||||
|
|
607
05/tcc-0.9.25/lib/libtcc1.c
Normal file
607
05/tcc-0.9.25/lib/libtcc1.c
Normal file
|
@ -0,0 +1,607 @@
|
||||||
|
/* TCC runtime library.
|
||||||
|
Parts of this code are (c) 2002 Fabrice Bellard
|
||||||
|
|
||||||
|
Copyright (C) 1987, 1988, 1992, 1994, 1995 Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
This file is free software; you can redistribute it and/or modify it
|
||||||
|
under the terms of the GNU General Public License as published by the
|
||||||
|
Free Software Foundation; either version 2, or (at your option) any
|
||||||
|
later version.
|
||||||
|
|
||||||
|
In addition to the permissions in the GNU General Public License, the
|
||||||
|
Free Software Foundation gives you unlimited permission to link the
|
||||||
|
compiled version of this file into combinations with other programs,
|
||||||
|
and to distribute those combinations without any restriction coming
|
||||||
|
from the use of this file. (The General Public License restrictions
|
||||||
|
do apply in other respects; for example, they cover modification of
|
||||||
|
the file, and distribution when not linked into a combine
|
||||||
|
executable.)
|
||||||
|
|
||||||
|
This file is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program; see the file COPYING. If not, write to
|
||||||
|
the Free Software Foundation, 59 Temple Place - Suite 330,
|
||||||
|
Boston, MA 02111-1307, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define W_TYPE_SIZE 32
|
||||||
|
#define BITS_PER_UNIT 8
|
||||||
|
|
||||||
|
typedef int Wtype;
|
||||||
|
typedef unsigned int UWtype;
|
||||||
|
typedef unsigned int USItype;
|
||||||
|
typedef long long DWtype;
|
||||||
|
typedef unsigned long long UDWtype;
|
||||||
|
|
||||||
|
struct DWstruct {
|
||||||
|
Wtype low, high;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef union
|
||||||
|
{
|
||||||
|
struct DWstruct s;
|
||||||
|
DWtype ll;
|
||||||
|
} DWunion;
|
||||||
|
|
||||||
|
typedef long double XFtype;
|
||||||
|
#define WORD_SIZE (sizeof (Wtype) * BITS_PER_UNIT)
|
||||||
|
#define HIGH_WORD_COEFF (((UDWtype) 1) << WORD_SIZE)
|
||||||
|
|
||||||
|
/* the following deal with IEEE single-precision numbers */
|
||||||
|
#define EXCESS 126
|
||||||
|
#define SIGNBIT 0x80000000
|
||||||
|
#define HIDDEN (1 << 23)
|
||||||
|
#define SIGN(fp) ((fp) & SIGNBIT)
|
||||||
|
#define EXP(fp) (((fp) >> 23) & 0xFF)
|
||||||
|
#define MANT(fp) (((fp) & 0x7FFFFF) | HIDDEN)
|
||||||
|
#define PACK(s,e,m) ((s) | ((e) << 23) | (m))
|
||||||
|
|
||||||
|
/* the following deal with IEEE double-precision numbers */
|
||||||
|
#define EXCESSD 1022
|
||||||
|
#define HIDDEND (1 << 20)
|
||||||
|
#define EXPD(fp) (((fp.l.upper) >> 20) & 0x7FF)
|
||||||
|
#define SIGND(fp) ((fp.l.upper) & SIGNBIT)
|
||||||
|
#define MANTD(fp) (((((fp.l.upper) & 0xFFFFF) | HIDDEND) << 10) | \
|
||||||
|
(fp.l.lower >> 22))
|
||||||
|
#define HIDDEND_LL ((long long)1 << 52)
|
||||||
|
#define MANTD_LL(fp) ((fp.ll & (HIDDEND_LL-1)) | HIDDEND_LL)
|
||||||
|
#define PACKD_LL(s,e,m) (((long long)((s)+((e)<<20))<<32)|(m))
|
||||||
|
|
||||||
|
/* the following deal with x86 long double-precision numbers */
|
||||||
|
#define EXCESSLD 16382
|
||||||
|
#define EXPLD(fp) (fp.l.upper & 0x7fff)
|
||||||
|
#define SIGNLD(fp) ((fp.l.upper) & 0x8000)
|
||||||
|
|
||||||
|
/* only for x86 */
|
||||||
|
union ldouble_long {
|
||||||
|
long double ld;
|
||||||
|
struct {
|
||||||
|
unsigned long long lower;
|
||||||
|
unsigned short upper;
|
||||||
|
} l;
|
||||||
|
};
|
||||||
|
|
||||||
|
union double_long {
|
||||||
|
double d;
|
||||||
|
#if 1
|
||||||
|
struct {
|
||||||
|
unsigned int lower;
|
||||||
|
int upper;
|
||||||
|
} l;
|
||||||
|
#else
|
||||||
|
struct {
|
||||||
|
int upper;
|
||||||
|
unsigned int lower;
|
||||||
|
} l;
|
||||||
|
#endif
|
||||||
|
long long ll;
|
||||||
|
};
|
||||||
|
|
||||||
|
union float_long {
|
||||||
|
float f;
|
||||||
|
long l;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* XXX: we don't support several builtin supports for now */
|
||||||
|
#ifndef __x86_64__
|
||||||
|
|
||||||
|
/* XXX: use gcc/tcc intrinsic ? */
|
||||||
|
#if defined(__i386__)
|
||||||
|
#define sub_ddmmss(sh, sl, ah, al, bh, bl) \
|
||||||
|
__asm__ ("subl %5,%1\n\tsbbl %3,%0" \
|
||||||
|
: "=r" ((USItype) (sh)), \
|
||||||
|
"=&r" ((USItype) (sl)) \
|
||||||
|
: "0" ((USItype) (ah)), \
|
||||||
|
"g" ((USItype) (bh)), \
|
||||||
|
"1" ((USItype) (al)), \
|
||||||
|
"g" ((USItype) (bl)))
|
||||||
|
#define umul_ppmm(w1, w0, u, v) \
|
||||||
|
__asm__ ("mull %3" \
|
||||||
|
: "=a" ((USItype) (w0)), \
|
||||||
|
"=d" ((USItype) (w1)) \
|
||||||
|
: "%0" ((USItype) (u)), \
|
||||||
|
"rm" ((USItype) (v)))
|
||||||
|
#define udiv_qrnnd(q, r, n1, n0, dv) \
|
||||||
|
__asm__ ("divl %4" \
|
||||||
|
: "=a" ((USItype) (q)), \
|
||||||
|
"=d" ((USItype) (r)) \
|
||||||
|
: "0" ((USItype) (n0)), \
|
||||||
|
"1" ((USItype) (n1)), \
|
||||||
|
"rm" ((USItype) (dv)))
|
||||||
|
#define count_leading_zeros(count, x) \
|
||||||
|
do { \
|
||||||
|
USItype __cbtmp; \
|
||||||
|
__asm__ ("bsrl %1,%0" \
|
||||||
|
: "=r" (__cbtmp) : "rm" ((USItype) (x))); \
|
||||||
|
(count) = __cbtmp ^ 31; \
|
||||||
|
} while (0)
|
||||||
|
#else
|
||||||
|
#error unsupported CPU type
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* most of this code is taken from libgcc2.c from gcc */
|
||||||
|
|
||||||
|
static UDWtype __udivmoddi4 (UDWtype n, UDWtype d, UDWtype *rp)
|
||||||
|
{
|
||||||
|
DWunion ww;
|
||||||
|
DWunion nn, dd;
|
||||||
|
DWunion rr;
|
||||||
|
UWtype d0, d1, n0, n1, n2;
|
||||||
|
UWtype q0, q1;
|
||||||
|
UWtype b, bm;
|
||||||
|
|
||||||
|
nn.ll = n;
|
||||||
|
dd.ll = d;
|
||||||
|
|
||||||
|
d0 = dd.s.low;
|
||||||
|
d1 = dd.s.high;
|
||||||
|
n0 = nn.s.low;
|
||||||
|
n1 = nn.s.high;
|
||||||
|
|
||||||
|
#if !UDIV_NEEDS_NORMALIZATION
|
||||||
|
if (d1 == 0)
|
||||||
|
{
|
||||||
|
if (d0 > n1)
|
||||||
|
{
|
||||||
|
/* 0q = nn / 0D */
|
||||||
|
|
||||||
|
udiv_qrnnd (q0, n0, n1, n0, d0);
|
||||||
|
q1 = 0;
|
||||||
|
|
||||||
|
/* Remainder in n0. */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* qq = NN / 0d */
|
||||||
|
|
||||||
|
if (d0 == 0)
|
||||||
|
d0 = 1 / d0; /* Divide intentionally by zero. */
|
||||||
|
|
||||||
|
udiv_qrnnd (q1, n1, 0, n1, d0);
|
||||||
|
udiv_qrnnd (q0, n0, n1, n0, d0);
|
||||||
|
|
||||||
|
/* Remainder in n0. */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rp != 0)
|
||||||
|
{
|
||||||
|
rr.s.low = n0;
|
||||||
|
rr.s.high = 0;
|
||||||
|
*rp = rr.ll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* UDIV_NEEDS_NORMALIZATION */
|
||||||
|
|
||||||
|
if (d1 == 0)
|
||||||
|
{
|
||||||
|
if (d0 > n1)
|
||||||
|
{
|
||||||
|
/* 0q = nn / 0D */
|
||||||
|
|
||||||
|
count_leading_zeros (bm, d0);
|
||||||
|
|
||||||
|
if (bm != 0)
|
||||||
|
{
|
||||||
|
/* Normalize, i.e. make the most significant bit of the
|
||||||
|
denominator set. */
|
||||||
|
|
||||||
|
d0 = d0 << bm;
|
||||||
|
n1 = (n1 << bm) | (n0 >> (W_TYPE_SIZE - bm));
|
||||||
|
n0 = n0 << bm;
|
||||||
|
}
|
||||||
|
|
||||||
|
udiv_qrnnd (q0, n0, n1, n0, d0);
|
||||||
|
q1 = 0;
|
||||||
|
|
||||||
|
/* Remainder in n0 >> bm. */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* qq = NN / 0d */
|
||||||
|
|
||||||
|
if (d0 == 0)
|
||||||
|
d0 = 1 / d0; /* Divide intentionally by zero. */
|
||||||
|
|
||||||
|
count_leading_zeros (bm, d0);
|
||||||
|
|
||||||
|
if (bm == 0)
|
||||||
|
{
|
||||||
|
/* From (n1 >= d0) /\ (the most significant bit of d0 is set),
|
||||||
|
conclude (the most significant bit of n1 is set) /\ (the
|
||||||
|
leading quotient digit q1 = 1).
|
||||||
|
|
||||||
|
This special case is necessary, not an optimization.
|
||||||
|
(Shifts counts of W_TYPE_SIZE are undefined.) */
|
||||||
|
|
||||||
|
n1 -= d0;
|
||||||
|
q1 = 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Normalize. */
|
||||||
|
|
||||||
|
b = W_TYPE_SIZE - bm;
|
||||||
|
|
||||||
|
d0 = d0 << bm;
|
||||||
|
n2 = n1 >> b;
|
||||||
|
n1 = (n1 << bm) | (n0 >> b);
|
||||||
|
n0 = n0 << bm;
|
||||||
|
|
||||||
|
udiv_qrnnd (q1, n1, n2, n1, d0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* n1 != d0... */
|
||||||
|
|
||||||
|
udiv_qrnnd (q0, n0, n1, n0, d0);
|
||||||
|
|
||||||
|
/* Remainder in n0 >> bm. */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rp != 0)
|
||||||
|
{
|
||||||
|
rr.s.low = n0 >> bm;
|
||||||
|
rr.s.high = 0;
|
||||||
|
*rp = rr.ll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* UDIV_NEEDS_NORMALIZATION */
|
||||||
|
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (d1 > n1)
|
||||||
|
{
|
||||||
|
/* 00 = nn / DD */
|
||||||
|
|
||||||
|
q0 = 0;
|
||||||
|
q1 = 0;
|
||||||
|
|
||||||
|
/* Remainder in n1n0. */
|
||||||
|
if (rp != 0)
|
||||||
|
{
|
||||||
|
rr.s.low = n0;
|
||||||
|
rr.s.high = n1;
|
||||||
|
*rp = rr.ll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* 0q = NN / dd */
|
||||||
|
|
||||||
|
count_leading_zeros (bm, d1);
|
||||||
|
if (bm == 0)
|
||||||
|
{
|
||||||
|
/* From (n1 >= d1) /\ (the most significant bit of d1 is set),
|
||||||
|
conclude (the most significant bit of n1 is set) /\ (the
|
||||||
|
quotient digit q0 = 0 or 1).
|
||||||
|
|
||||||
|
This special case is necessary, not an optimization. */
|
||||||
|
|
||||||
|
/* The condition on the next line takes advantage of that
|
||||||
|
n1 >= d1 (true due to program flow). */
|
||||||
|
if (n1 > d1 || n0 >= d0)
|
||||||
|
{
|
||||||
|
q0 = 1;
|
||||||
|
sub_ddmmss (n1, n0, n1, n0, d1, d0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
q0 = 0;
|
||||||
|
|
||||||
|
q1 = 0;
|
||||||
|
|
||||||
|
if (rp != 0)
|
||||||
|
{
|
||||||
|
rr.s.low = n0;
|
||||||
|
rr.s.high = n1;
|
||||||
|
*rp = rr.ll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
UWtype m1, m0;
|
||||||
|
/* Normalize. */
|
||||||
|
|
||||||
|
b = W_TYPE_SIZE - bm;
|
||||||
|
|
||||||
|
d1 = (d1 << bm) | (d0 >> b);
|
||||||
|
d0 = d0 << bm;
|
||||||
|
n2 = n1 >> b;
|
||||||
|
n1 = (n1 << bm) | (n0 >> b);
|
||||||
|
n0 = n0 << bm;
|
||||||
|
|
||||||
|
udiv_qrnnd (q0, n1, n2, n1, d1);
|
||||||
|
umul_ppmm (m1, m0, q0, d0);
|
||||||
|
|
||||||
|
if (m1 > n1 || (m1 == n1 && m0 > n0))
|
||||||
|
{
|
||||||
|
q0--;
|
||||||
|
sub_ddmmss (m1, m0, m1, m0, d1, d0);
|
||||||
|
}
|
||||||
|
|
||||||
|
q1 = 0;
|
||||||
|
|
||||||
|
/* Remainder in (n1n0 - m1m0) >> bm. */
|
||||||
|
if (rp != 0)
|
||||||
|
{
|
||||||
|
sub_ddmmss (n1, n0, n1, n0, m1, m0);
|
||||||
|
rr.s.low = (n1 << b) | (n0 >> bm);
|
||||||
|
rr.s.high = n1 >> bm;
|
||||||
|
*rp = rr.ll;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ww.s.low = q0;
|
||||||
|
ww.s.high = q1;
|
||||||
|
return ww.ll;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define __negdi2(a) (-(a))
|
||||||
|
|
||||||
|
long long __divdi3(long long u, long long v)
|
||||||
|
{
|
||||||
|
int c = 0;
|
||||||
|
DWunion uu, vv;
|
||||||
|
DWtype w;
|
||||||
|
|
||||||
|
uu.ll = u;
|
||||||
|
vv.ll = v;
|
||||||
|
|
||||||
|
if (uu.s.high < 0) {
|
||||||
|
c = ~c;
|
||||||
|
uu.ll = __negdi2 (uu.ll);
|
||||||
|
}
|
||||||
|
if (vv.s.high < 0) {
|
||||||
|
c = ~c;
|
||||||
|
vv.ll = __negdi2 (vv.ll);
|
||||||
|
}
|
||||||
|
w = __udivmoddi4 (uu.ll, vv.ll, (UDWtype *) 0);
|
||||||
|
if (c)
|
||||||
|
w = __negdi2 (w);
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
long long __moddi3(long long u, long long v)
|
||||||
|
{
|
||||||
|
int c = 0;
|
||||||
|
DWunion uu, vv;
|
||||||
|
DWtype w;
|
||||||
|
|
||||||
|
uu.ll = u;
|
||||||
|
vv.ll = v;
|
||||||
|
|
||||||
|
if (uu.s.high < 0) {
|
||||||
|
c = ~c;
|
||||||
|
uu.ll = __negdi2 (uu.ll);
|
||||||
|
}
|
||||||
|
if (vv.s.high < 0)
|
||||||
|
vv.ll = __negdi2 (vv.ll);
|
||||||
|
|
||||||
|
__udivmoddi4 (uu.ll, vv.ll, (UDWtype *) &w);
|
||||||
|
if (c)
|
||||||
|
w = __negdi2 (w);
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long long __udivdi3(unsigned long long u, unsigned long long v)
|
||||||
|
{
|
||||||
|
return __udivmoddi4 (u, v, (UDWtype *) 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long long __umoddi3(unsigned long long u, unsigned long long v)
|
||||||
|
{
|
||||||
|
UDWtype w;
|
||||||
|
|
||||||
|
__udivmoddi4 (u, v, &w);
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: fix tcc's code generator to do this instead */
|
||||||
|
long long __ashrdi3(long long a, int b)
|
||||||
|
{
|
||||||
|
#ifdef __TINYC__
|
||||||
|
DWunion u;
|
||||||
|
u.ll = a;
|
||||||
|
if (b >= 32) {
|
||||||
|
u.s.low = u.s.high >> (b - 32);
|
||||||
|
u.s.high = u.s.high >> 31;
|
||||||
|
} else if (b != 0) {
|
||||||
|
u.s.low = ((unsigned)u.s.low >> b) | (u.s.high << (32 - b));
|
||||||
|
u.s.high = u.s.high >> b;
|
||||||
|
}
|
||||||
|
return u.ll;
|
||||||
|
#else
|
||||||
|
return a >> b;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: fix tcc's code generator to do this instead */
|
||||||
|
unsigned long long __lshrdi3(unsigned long long a, int b)
|
||||||
|
{
|
||||||
|
#ifdef __TINYC__
|
||||||
|
DWunion u;
|
||||||
|
u.ll = a;
|
||||||
|
if (b >= 32) {
|
||||||
|
u.s.low = (unsigned)u.s.high >> (b - 32);
|
||||||
|
u.s.high = 0;
|
||||||
|
} else if (b != 0) {
|
||||||
|
u.s.low = ((unsigned)u.s.low >> b) | (u.s.high << (32 - b));
|
||||||
|
u.s.high = (unsigned)u.s.high >> b;
|
||||||
|
}
|
||||||
|
return u.ll;
|
||||||
|
#else
|
||||||
|
return a >> b;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
/* XXX: fix tcc's code generator to do this instead */
|
||||||
|
long long __ashldi3(long long a, int b)
|
||||||
|
{
|
||||||
|
#ifdef __TINYC__
|
||||||
|
DWunion u;
|
||||||
|
u.ll = a;
|
||||||
|
if (b >= 32) {
|
||||||
|
u.s.high = (unsigned)u.s.low << (b - 32);
|
||||||
|
u.s.low = 0;
|
||||||
|
} else if (b != 0) {
|
||||||
|
u.s.high = ((unsigned)u.s.high << b) | ((unsigned)u.s.low >> (32 - b));
|
||||||
|
u.s.low = (unsigned)u.s.low << b;
|
||||||
|
}
|
||||||
|
return u.ll;
|
||||||
|
#else
|
||||||
|
return a << b;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#if defined(__i386__)
|
||||||
|
/* FPU control word for rounding to nearest mode */
|
||||||
|
unsigned short __tcc_fpu_control = 0x137f;
|
||||||
|
/* FPU control word for round to zero mode for int conversion */
|
||||||
|
unsigned short __tcc_int_fpu_control = 0x137f | 0x0c00;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* !__x86_64__ */
|
||||||
|
|
||||||
|
/* XXX: fix tcc's code generator to do this instead */
|
||||||
|
float __floatundisf(unsigned long long a)
|
||||||
|
{
|
||||||
|
DWunion uu;
|
||||||
|
XFtype r;
|
||||||
|
|
||||||
|
uu.ll = a;
|
||||||
|
if (uu.s.high >= 0) {
|
||||||
|
return (float)uu.ll;
|
||||||
|
} else {
|
||||||
|
r = (XFtype)uu.ll;
|
||||||
|
r += 18446744073709551616.0;
|
||||||
|
return (float)r;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double __floatundidf(unsigned long long a)
|
||||||
|
{
|
||||||
|
DWunion uu;
|
||||||
|
XFtype r;
|
||||||
|
|
||||||
|
uu.ll = a;
|
||||||
|
if (uu.s.high >= 0) {
|
||||||
|
return (double)uu.ll;
|
||||||
|
} else {
|
||||||
|
r = (XFtype)uu.ll;
|
||||||
|
r += 18446744073709551616.0;
|
||||||
|
return (double)r;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
long double __floatundixf(unsigned long long a)
|
||||||
|
{
|
||||||
|
DWunion uu;
|
||||||
|
XFtype r;
|
||||||
|
|
||||||
|
uu.ll = a;
|
||||||
|
if (uu.s.high >= 0) {
|
||||||
|
return (long double)uu.ll;
|
||||||
|
} else {
|
||||||
|
r = (XFtype)uu.ll;
|
||||||
|
r += 18446744073709551616.0;
|
||||||
|
return (long double)r;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long long __fixunssfdi (float a1)
|
||||||
|
{
|
||||||
|
register union float_long fl1;
|
||||||
|
register int exp;
|
||||||
|
register unsigned long l;
|
||||||
|
|
||||||
|
fl1.f = a1;
|
||||||
|
|
||||||
|
if (fl1.l == 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
exp = EXP (fl1.l) - EXCESS - 24;
|
||||||
|
|
||||||
|
l = MANT(fl1.l);
|
||||||
|
if (exp >= 41)
|
||||||
|
return (unsigned long long)-1;
|
||||||
|
else if (exp >= 0)
|
||||||
|
return (unsigned long long)l << exp;
|
||||||
|
else if (exp >= -23)
|
||||||
|
return l >> -exp;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long long __fixunsdfdi (double a1)
|
||||||
|
{
|
||||||
|
register union double_long dl1;
|
||||||
|
register int exp;
|
||||||
|
register unsigned long long l;
|
||||||
|
|
||||||
|
dl1.d = a1;
|
||||||
|
|
||||||
|
if (dl1.ll == 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
exp = EXPD (dl1) - EXCESSD - 53;
|
||||||
|
|
||||||
|
l = MANTD_LL(dl1);
|
||||||
|
|
||||||
|
if (exp >= 12)
|
||||||
|
return (unsigned long long)-1;
|
||||||
|
else if (exp >= 0)
|
||||||
|
return l << exp;
|
||||||
|
else if (exp >= -52)
|
||||||
|
return l >> -exp;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long long __fixunsxfdi (long double a1)
|
||||||
|
{
|
||||||
|
register union ldouble_long dl1;
|
||||||
|
register int exp;
|
||||||
|
register unsigned long long l;
|
||||||
|
|
||||||
|
dl1.ld = a1;
|
||||||
|
|
||||||
|
if (dl1.l.lower == 0 && dl1.l.upper == 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
exp = EXPLD (dl1) - EXCESSLD - 64;
|
||||||
|
|
||||||
|
l = dl1.l.lower;
|
||||||
|
|
||||||
|
if (exp > 0)
|
||||||
|
return (unsigned long long)-1;
|
||||||
|
else if (exp >= -63)
|
||||||
|
return l >> -exp;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
2276
05/tcc-0.9.25/libtcc.c
Normal file
2276
05/tcc-0.9.25/libtcc.c
Normal file
File diff suppressed because it is too large
Load diff
108
05/tcc-0.9.25/libtcc.h
Normal file
108
05/tcc-0.9.25/libtcc.h
Normal file
|
@ -0,0 +1,108 @@
|
||||||
|
#ifndef LIBTCC_H
|
||||||
|
#define LIBTCC_H
|
||||||
|
|
||||||
|
#ifdef LIBTCC_AS_DLL
|
||||||
|
#define LIBTCCAPI __declspec(dllexport)
|
||||||
|
#else
|
||||||
|
#define LIBTCCAPI
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct TCCState;
|
||||||
|
|
||||||
|
typedef struct TCCState TCCState;
|
||||||
|
|
||||||
|
/* create a new TCC compilation context */
|
||||||
|
LIBTCCAPI TCCState *tcc_new(void);
|
||||||
|
|
||||||
|
/* free a TCC compilation context */
|
||||||
|
LIBTCCAPI void tcc_delete(TCCState *s);
|
||||||
|
|
||||||
|
/* add debug information in the generated code */
|
||||||
|
LIBTCCAPI void tcc_enable_debug(TCCState *s);
|
||||||
|
|
||||||
|
/* set error/warning display callback */
|
||||||
|
LIBTCCAPI void tcc_set_error_func(TCCState *s, void *error_opaque,
|
||||||
|
void (*error_func)(void *opaque, const char *msg));
|
||||||
|
|
||||||
|
/* set/reset a warning */
|
||||||
|
LIBTCCAPI int tcc_set_warning(TCCState *s, const char *warning_name, int value);
|
||||||
|
|
||||||
|
/*****************************/
|
||||||
|
/* preprocessor */
|
||||||
|
|
||||||
|
/* add include path */
|
||||||
|
LIBTCCAPI int tcc_add_include_path(TCCState *s, const char *pathname);
|
||||||
|
|
||||||
|
/* add in system include path */
|
||||||
|
LIBTCCAPI int tcc_add_sysinclude_path(TCCState *s, const char *pathname);
|
||||||
|
|
||||||
|
/* define preprocessor symbol 'sym'. Can put optional value */
|
||||||
|
LIBTCCAPI void tcc_define_symbol(TCCState *s, const char *sym, const char *value);
|
||||||
|
|
||||||
|
/* undefine preprocess symbol 'sym' */
|
||||||
|
LIBTCCAPI void tcc_undefine_symbol(TCCState *s, const char *sym);
|
||||||
|
|
||||||
|
/*****************************/
|
||||||
|
/* compiling */
|
||||||
|
|
||||||
|
/* add a file (either a C file, dll, an object, a library or an ld
|
||||||
|
script). Return -1 if error. */
|
||||||
|
LIBTCCAPI int tcc_add_file(TCCState *s, const char *filename);
|
||||||
|
|
||||||
|
/* compile a string containing a C source. Return non zero if
|
||||||
|
error. */
|
||||||
|
LIBTCCAPI int tcc_compile_string(TCCState *s, const char *buf);
|
||||||
|
|
||||||
|
/*****************************/
|
||||||
|
/* linking commands */
|
||||||
|
|
||||||
|
/* set output type. MUST BE CALLED before any compilation */
|
||||||
|
#define TCC_OUTPUT_MEMORY 0 /* output will be ran in memory (no
|
||||||
|
output file) (default) */
|
||||||
|
#define TCC_OUTPUT_EXE 1 /* executable file */
|
||||||
|
#define TCC_OUTPUT_DLL 2 /* dynamic library */
|
||||||
|
#define TCC_OUTPUT_OBJ 3 /* object file */
|
||||||
|
#define TCC_OUTPUT_PREPROCESS 4 /* preprocessed file (used internally) */
|
||||||
|
LIBTCCAPI int tcc_set_output_type(TCCState *s, int output_type);
|
||||||
|
|
||||||
|
#define TCC_OUTPUT_FORMAT_ELF 0 /* default output format: ELF */
|
||||||
|
#define TCC_OUTPUT_FORMAT_BINARY 1 /* binary image output */
|
||||||
|
#define TCC_OUTPUT_FORMAT_COFF 2 /* COFF */
|
||||||
|
|
||||||
|
/* equivalent to -Lpath option */
|
||||||
|
LIBTCCAPI int tcc_add_library_path(TCCState *s, const char *pathname);
|
||||||
|
|
||||||
|
/* the library name is the same as the argument of the '-l' option */
|
||||||
|
LIBTCCAPI int tcc_add_library(TCCState *s, const char *libraryname);
|
||||||
|
|
||||||
|
/* add a symbol to the compiled program */
|
||||||
|
LIBTCCAPI int tcc_add_symbol(TCCState *s, const char *name, void *val);
|
||||||
|
|
||||||
|
/* output an executable, library or object file. DO NOT call
|
||||||
|
tcc_relocate() before. */
|
||||||
|
LIBTCCAPI int tcc_output_file(TCCState *s, const char *filename);
|
||||||
|
|
||||||
|
/* link and run main() function and return its value. DO NOT call
|
||||||
|
tcc_relocate() before. */
|
||||||
|
LIBTCCAPI int tcc_run(TCCState *s, int argc, char **argv);
|
||||||
|
|
||||||
|
/* copy code into memory passed in by the caller and do all relocations
|
||||||
|
(needed before using tcc_get_symbol()).
|
||||||
|
returns -1 on error and required size if ptr is NULL */
|
||||||
|
LIBTCCAPI int tcc_relocate(TCCState *s1, void *ptr);
|
||||||
|
|
||||||
|
/* return symbol value or NULL if not found */
|
||||||
|
LIBTCCAPI void *tcc_get_symbol(TCCState *s, const char *name);
|
||||||
|
|
||||||
|
/* set CONFIG_TCCDIR at runtime */
|
||||||
|
LIBTCCAPI void tcc_set_lib_path(TCCState *s, const char *path);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
6
05/tcc-0.9.25/limits.h
Normal file
6
05/tcc-0.9.25/limits.h
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
#ifndef _LIMITS_H
|
||||||
|
#define _LIMITS_H
|
||||||
|
|
||||||
|
#include <stdc_common.h> // we define all the relevant constants here
|
||||||
|
|
||||||
|
#endif // _LIMITS_H
|
60
05/tcc-0.9.25/locale.h
Normal file
60
05/tcc-0.9.25/locale.h
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
#ifndef _LOCALE_H
|
||||||
|
#define _LOCALE_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
struct lconv {
|
||||||
|
char *decimal_point; /* "." */
|
||||||
|
char *thousands_sep; /* "" */
|
||||||
|
char *grouping; /* "" */
|
||||||
|
char *int_curr_symbol; /* "" */
|
||||||
|
char *currency_symbol; /* "" */
|
||||||
|
char *mon_decimal_point; /* "" */
|
||||||
|
char *mon_thousands_sep; /* "" */
|
||||||
|
char *mon_grouping; /* "" */
|
||||||
|
char *positive_sign; /* "" */
|
||||||
|
char *negative_sign; /* "" */
|
||||||
|
char int_frac_digits; /* CHAR_MAX */
|
||||||
|
char frac_digits; /* CHAR_MAX */
|
||||||
|
char p_cs_precedes; /* CHAR_MAX */
|
||||||
|
char p_sep_by_space; /* CHAR_MAX */
|
||||||
|
char n_cs_precedes; /* CHAR_MAX */
|
||||||
|
char n_sep_by_space; /* CHAR_MAX */
|
||||||
|
char p_sign_posn; /* CHAR_MAX */
|
||||||
|
char n_sign_posn; /* CHAR_MAX */
|
||||||
|
};
|
||||||
|
|
||||||
|
// these are GCC's constants, but it doesn't really matter which constants we use.
|
||||||
|
#define LC_ALL 6
|
||||||
|
#define LC_COLLATE 3
|
||||||
|
#define LC_CTYPE 0
|
||||||
|
#define LC_MONETARY 4
|
||||||
|
#define LC_NUMERIC 1
|
||||||
|
#define LC_TIME 2
|
||||||
|
|
||||||
|
char *setlocale(int category, char *locale) {
|
||||||
|
if (!locale) return "C";
|
||||||
|
if (*locale == 'C' && !locale[1]) {
|
||||||
|
// yep
|
||||||
|
return "C";
|
||||||
|
}
|
||||||
|
|
||||||
|
// we only support the C locale
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct lconv *localeconv(void) {
|
||||||
|
static struct lconv conv = {
|
||||||
|
".",
|
||||||
|
"", "", "",
|
||||||
|
"", "", "",
|
||||||
|
"", "", "",
|
||||||
|
CHAR_MAX, CHAR_MAX, CHAR_MAX,
|
||||||
|
CHAR_MAX, CHAR_MAX, CHAR_MAX,
|
||||||
|
CHAR_MAX, CHAR_MAX
|
||||||
|
};
|
||||||
|
return &conv;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // _LOCALE_H
|
409
05/tcc-0.9.25/math.h
Normal file
409
05/tcc-0.9.25/math.h
Normal file
|
@ -0,0 +1,409 @@
|
||||||
|
#ifndef _MATH_H
|
||||||
|
#define _MATH_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
#define HUGE_VAL _INFINITY // glibc defines HUGE_VAL as infinity (the C standard only requires it to be positive, funnily enough)
|
||||||
|
#define _NAN (-(_INFINITY-_INFINITY))
|
||||||
|
#define _PI 3.141592653589793
|
||||||
|
#define _2PI 6.283185307179586
|
||||||
|
#define _HALF_PI 1.5707963267948966
|
||||||
|
#define _THREE_HALVES_PI 4.71238898038469
|
||||||
|
|
||||||
|
// NOTE: these functions are not IEEE 754-compliant (the C standard doesn't require them to be), but they're pretty good
|
||||||
|
|
||||||
|
double frexp(double value, int *exp) {
|
||||||
|
if (value == 0) {
|
||||||
|
*exp = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
unsigned long u = *(unsigned long *)&value, significand;
|
||||||
|
*exp = ((u >> 52) & 0x7ff) - 1022;
|
||||||
|
// replace exponent with 1022
|
||||||
|
u &= 0x800fffffffffffff;
|
||||||
|
u |= 0x3fe0000000000000;
|
||||||
|
return *(double *)&u;
|
||||||
|
}
|
||||||
|
|
||||||
|
double ldexp(double x, int exp) {
|
||||||
|
int e;
|
||||||
|
double y = frexp(x, &e);
|
||||||
|
// since x = y * 2^e, x * 2^exp = y * 2^(e+exp)
|
||||||
|
exp += e;
|
||||||
|
if (exp < -1022) return 0;
|
||||||
|
if (exp > 1023) return _INFINITY;
|
||||||
|
unsigned long pow2 = (unsigned long)(exp + 1023) << 52;
|
||||||
|
return y * *(double *)&pow2;
|
||||||
|
}
|
||||||
|
|
||||||
|
double floor(double x) {
|
||||||
|
if (x >= 0.0) {
|
||||||
|
if (x > 1073741824.0 * 1073741824.0)
|
||||||
|
return x; // floats this big must be integers
|
||||||
|
return (unsigned long)x;
|
||||||
|
} else {
|
||||||
|
if (x < -1073741824.0 * 1073741824.0)
|
||||||
|
return x; // floats this big must be integers
|
||||||
|
double i = (long)x;
|
||||||
|
if (x == i) return x;
|
||||||
|
return i - 1.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double ceil(double x) {
|
||||||
|
double f = floor(x);
|
||||||
|
if (x == f) return f;
|
||||||
|
return f + 1.;
|
||||||
|
}
|
||||||
|
|
||||||
|
double fabs(double x) {
|
||||||
|
// this is better than x >= 0 ? x : -x because it deals with -0 properly
|
||||||
|
unsigned long u = *(unsigned long *)&x;
|
||||||
|
u &= 0x7fffffffffffffff;
|
||||||
|
return *(double *)&u;
|
||||||
|
}
|
||||||
|
|
||||||
|
double fmod(double x, double y) {
|
||||||
|
if (y == 0.0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return 0.0;
|
||||||
|
}
|
||||||
|
return x - (floor(x / y) * y);
|
||||||
|
}
|
||||||
|
|
||||||
|
double _sin_taylor(double x) {
|
||||||
|
double i;
|
||||||
|
double term = x;
|
||||||
|
// taylor expansion for sin: x - x³/3! + x⁵/5! - ...
|
||||||
|
|
||||||
|
// https://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
||||||
|
double prev = -1.0;
|
||||||
|
double sum = 0.0;
|
||||||
|
double c = 0.0;
|
||||||
|
for (i = 0.0; i < 100.0 && sum != prev; ++i) {
|
||||||
|
prev = sum;
|
||||||
|
double y = term - c;
|
||||||
|
double t = sum + y;
|
||||||
|
c = (t - sum) - y;
|
||||||
|
sum = t;
|
||||||
|
term *= -(x * x) / ((2.0*i+2.0)*(2.0*i+3.0));
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
double _cos_taylor(double x) {
|
||||||
|
double i;
|
||||||
|
double term = 1.0;
|
||||||
|
// taylor expansion for cos: 1 - x²/2! + x⁴/4! - ...
|
||||||
|
|
||||||
|
// https://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
||||||
|
double prev = -1.0;
|
||||||
|
double sum = 0.0;
|
||||||
|
double c = 0.0;
|
||||||
|
for (i = 0.0; i < 100.0 && sum != prev; ++i) {
|
||||||
|
prev = sum;
|
||||||
|
double y = term - c;
|
||||||
|
double t = sum + y;
|
||||||
|
c = (t - sum) - y;
|
||||||
|
sum = t;
|
||||||
|
term *= -(x * x) / ((2.0*i+1.0)*(2.0*i+2.0));
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
double sin(double x) {
|
||||||
|
x = fmod(x, 2.0*_PI);
|
||||||
|
// the Taylor series works best for small inputs. so, provide _sin_taylor with a value in the range [0,π/2]
|
||||||
|
if (x < _HALF_PI)
|
||||||
|
return _sin_taylor(x);
|
||||||
|
if (x < _PI)
|
||||||
|
return _sin_taylor(_PI - x);
|
||||||
|
if (x < _THREE_HALVES_PI)
|
||||||
|
return -_sin_taylor(x - _PI);
|
||||||
|
return -_sin_taylor(_2PI - x);
|
||||||
|
}
|
||||||
|
|
||||||
|
double cos(double x) {
|
||||||
|
x = fmod(x, 2.0*_PI);
|
||||||
|
// the Taylor series works best for small inputs. so, provide _cos_taylor with a value in the range [0,π/2]
|
||||||
|
if (x < _HALF_PI)
|
||||||
|
return _cos_taylor(x);
|
||||||
|
if (x < _PI)
|
||||||
|
return -_cos_taylor(_PI - x);
|
||||||
|
if (x < _THREE_HALVES_PI)
|
||||||
|
return -_cos_taylor(x - _PI);
|
||||||
|
return _cos_taylor(_2PI - x);
|
||||||
|
}
|
||||||
|
|
||||||
|
double tan(double x) {
|
||||||
|
return sin(x)/cos(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
// for sqrt and the inverse trigonometric functions, we use Newton's method
|
||||||
|
// https://en.wikipedia.org/wiki/Newton%27s_method
|
||||||
|
|
||||||
|
double sqrt(double x) {
|
||||||
|
if (x < 0.0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
if (x == 0.0) return 0.0;
|
||||||
|
if (x == _INFINITY) return _INFINITY;
|
||||||
|
// we want to find the root of: f(t) = t² - x
|
||||||
|
// f'(t) = 2t
|
||||||
|
int exp;
|
||||||
|
double y = frexp(x, &exp);
|
||||||
|
if (exp & 1) {
|
||||||
|
y *= 2;
|
||||||
|
--exp;
|
||||||
|
}
|
||||||
|
// newton's method will be slow for very small or very large numbers.
|
||||||
|
// so we have ensured that
|
||||||
|
// 0.5 ≤ y < 2
|
||||||
|
// and also x = y * 2^exp; sqrt(x) = sqrt(y) * 2^(exp/2) NB: we've ensured that exp is even
|
||||||
|
|
||||||
|
// 7 iterations seems to be more than enough for any number
|
||||||
|
double t = y;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
t = (y / t + t) * 0.5;
|
||||||
|
|
||||||
|
return ldexp(t, exp>>1);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
double _acos_newton(double x) {
|
||||||
|
// we want to find the root of: f(t) = cos(t) - x
|
||||||
|
// f'(t) = -sin(t)
|
||||||
|
double t = _HALF_PI - x; // reasonably good first approximation
|
||||||
|
double prev_t = -100.0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < 100 && prev_t != t; ++i) {
|
||||||
|
prev_t = t;
|
||||||
|
t += (cos(t) - x) / sin(t);
|
||||||
|
}
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
|
||||||
|
double _asin_newton(double x) {
|
||||||
|
// we want to find the root of: f(t) = sin(t) - x
|
||||||
|
// f'(t) = cos(t)
|
||||||
|
double t = x; // reasonably good first approximation
|
||||||
|
double prev_t = -100.0;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < 100 && prev_t != t; ++i) {
|
||||||
|
prev_t = t;
|
||||||
|
t += (x - sin(t)) / cos(t);
|
||||||
|
}
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
|
||||||
|
double acos(double x) {
|
||||||
|
if (x > 1.0 || x < -1.0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
// Newton's method doesn't work well near -1 and 1, because f(x) / f'(x) is very large.
|
||||||
|
if (x > 0.8)
|
||||||
|
return _asin_newton(sqrt(1-x*x));
|
||||||
|
if (x < -0.8)
|
||||||
|
return _PI-_asin_newton(sqrt(1-x*x));
|
||||||
|
|
||||||
|
return _acos_newton(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
double asin(double x) {
|
||||||
|
if (x > 1.0 || x < -1.0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
// Newton's method doesn't work well near -1 and 1, because f(x) / f'(x) is very large.
|
||||||
|
if (x > 0.8)
|
||||||
|
return _acos_newton(sqrt(1.0-x*x));
|
||||||
|
if (x < -0.8)
|
||||||
|
return -_acos_newton(sqrt(1.0-x*x));
|
||||||
|
|
||||||
|
return _asin_newton(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
double atan(double x) {
|
||||||
|
// the formula below breaks for really large inputs; tan(10^20) as a double is indistinguishable from pi/2 anyways
|
||||||
|
if (x > 1e20) return _HALF_PI;
|
||||||
|
if (x < -1e20) return -_HALF_PI;
|
||||||
|
|
||||||
|
// we can use a nice trigonometric identity here
|
||||||
|
return asin(x / sqrt(1+x*x));
|
||||||
|
}
|
||||||
|
|
||||||
|
double atan2(double y, double x) {
|
||||||
|
if (x == 0.0) {
|
||||||
|
if (y > 0.0) return _HALF_PI;
|
||||||
|
if (y < 0.0) return -_HALF_PI;
|
||||||
|
return 0.0; // this is what IEEE 754 does
|
||||||
|
}
|
||||||
|
|
||||||
|
double a = atan(y/x);
|
||||||
|
if (x > 0.0) {
|
||||||
|
return a;
|
||||||
|
} else if (y > 0.0) {
|
||||||
|
return a + _PI;
|
||||||
|
} else {
|
||||||
|
return a - _PI;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double _exp_taylor(double x) {
|
||||||
|
double i;
|
||||||
|
double term = 1.0;
|
||||||
|
// taylor expansion for exp: 1 + x/1! + x²/2! + ...
|
||||||
|
|
||||||
|
// https://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
||||||
|
double prev = -1.0;
|
||||||
|
double sum = 0.0;
|
||||||
|
double c = 0.0;
|
||||||
|
for (i = 1.0; i < 100.0 && sum != prev; ++i) {
|
||||||
|
prev = sum;
|
||||||
|
double y = term - c;
|
||||||
|
double t = sum + y;
|
||||||
|
c = (t - sum) - y;
|
||||||
|
sum = t;
|
||||||
|
term *= x / i;
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
double exp(double x) {
|
||||||
|
if (x > 709.782712893384) {
|
||||||
|
errno = ERANGE;
|
||||||
|
return _INFINITY;
|
||||||
|
}
|
||||||
|
if (x == 0.0) return 1;
|
||||||
|
if (x < -744.4400719213812)
|
||||||
|
return 0;
|
||||||
|
int i, e;
|
||||||
|
double y = frexp(x, &e);
|
||||||
|
if (e < 1.0) return _exp_taylor(x);
|
||||||
|
// the taylor series doesn't work well for large x (positive or negative),
|
||||||
|
// so we use the fact that exp(y*2^e) = exp(y)^(2^e)
|
||||||
|
double value = _exp_taylor(y);
|
||||||
|
for (i = 0; i < e; ++i)
|
||||||
|
value *= value;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define _LOG2 0.6931471805599453
|
||||||
|
|
||||||
|
double log(double x) {
|
||||||
|
if (x < 0.0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
if (x == 0.0) return -_INFINITY;
|
||||||
|
if (x == 1.0) return 0.0;
|
||||||
|
int e;
|
||||||
|
double sum;
|
||||||
|
double a = frexp(x, &e);
|
||||||
|
// since x = a * 2^e, log(x) = log(a) + log(2^e) = log(a) + e log(2)
|
||||||
|
sum = e * _LOG2;
|
||||||
|
// now that a is in [1/2,1), the series log(a) = (a-1) - (a-1)²/2 + (a-1)³/3 - ... converges quickly
|
||||||
|
|
||||||
|
a -= 1;
|
||||||
|
// https://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
||||||
|
double prev = HUGE_VAL;
|
||||||
|
double c = 0;
|
||||||
|
double term = a;
|
||||||
|
double i;
|
||||||
|
for (i = 1.0; i < 100.0 && sum != prev; ++i) {
|
||||||
|
prev = sum;
|
||||||
|
double y = term / i - c;
|
||||||
|
double t = sum + y;
|
||||||
|
c = (t - sum) - y;
|
||||||
|
sum = t;
|
||||||
|
term *= -a;
|
||||||
|
}
|
||||||
|
return sum;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define _INVLOG10 0.43429448190325176 // = 1/log(10)
|
||||||
|
double log10(double x) {
|
||||||
|
return log(x) * _INVLOG10;
|
||||||
|
}
|
||||||
|
|
||||||
|
double modf(double value, double *iptr) {
|
||||||
|
double m = fmod(value, 1.0);
|
||||||
|
if (value >= 0.0) {
|
||||||
|
*iptr = value - m;
|
||||||
|
return m;
|
||||||
|
} else if (m == 0.0) {
|
||||||
|
*iptr = value;
|
||||||
|
return 0.0;
|
||||||
|
} else {
|
||||||
|
*iptr = value - m + 1.0;
|
||||||
|
return m - 1.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// double raised to the power of an integer
|
||||||
|
double _dpowi(double x, unsigned long y) {
|
||||||
|
double result = 1.0;
|
||||||
|
if (y & 1) {
|
||||||
|
--y;
|
||||||
|
result *= x;
|
||||||
|
}
|
||||||
|
if (y > 0) {
|
||||||
|
double p = _dpowi(x, y >> 1);
|
||||||
|
result *= p * p;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
double pow(double x, double y) {
|
||||||
|
if (x > 0.0) {
|
||||||
|
return exp(y * log(x));
|
||||||
|
} else if (x < 0.0) {
|
||||||
|
if (fmod(y, 1.0) != 0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
if (y > 1.6602069666338597e+19)
|
||||||
|
return x < -1. ? -_INFINITY : 0.;
|
||||||
|
if (y < -1.6602069666338597e+19)
|
||||||
|
return x < -1. ? 0. : -_INFINITY;
|
||||||
|
return _dpowi(x, (unsigned long)y);
|
||||||
|
} else {
|
||||||
|
if (y < 0) {
|
||||||
|
errno = EDOM;
|
||||||
|
return _NAN;
|
||||||
|
}
|
||||||
|
if (y > 0) {
|
||||||
|
// 0^x = 0 for x>0
|
||||||
|
return 0.;
|
||||||
|
}
|
||||||
|
// 0^0 = 1
|
||||||
|
return 1.;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double cosh(double x) {
|
||||||
|
double e = exp(x);
|
||||||
|
return (e + 1./e) * 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
double sinh(double x) {
|
||||||
|
double e = exp(x);
|
||||||
|
return (e - 1./e) * 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
double tanh(double x) {
|
||||||
|
if (x > 20.0) return 1.;
|
||||||
|
if (x < -20.0) return -1.;
|
||||||
|
double e = exp(x);
|
||||||
|
double f = 1./e;
|
||||||
|
return (e - f) / (e + f);
|
||||||
|
}
|
||||||
|
#endif // _MATH_H
|
21
05/tcc-0.9.25/setjmp.h
Normal file
21
05/tcc-0.9.25/setjmp.h
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
#ifndef _SETJMP_H
|
||||||
|
#define _SETJMP_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
typedef long jmp_buf[3];
|
||||||
|
|
||||||
|
// @NONSTANDARD: we don't actually support setjmp
|
||||||
|
|
||||||
|
int setjmp(jmp_buf env) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void __longjmp(jmp_buf env, int val, const char *filename, int line) {
|
||||||
|
fprintf(stderr, "Error: Tried to longjmp from %s:%d with value %d\n", filename, line, val);
|
||||||
|
_Exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define longjmp(env, val) __longjmp(env, val, __FILE__, __LINE__)
|
||||||
|
|
||||||
|
#endif
|
253
05/tcc-0.9.25/signal.h
Normal file
253
05/tcc-0.9.25/signal.h
Normal file
|
@ -0,0 +1,253 @@
|
||||||
|
#ifndef _SIGNAL_H
|
||||||
|
#define _SIGNAL_H
|
||||||
|
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
typedef long sig_atomic_t; // there are no "asynchronous interrupts"
|
||||||
|
|
||||||
|
#define SIG_DFL ((void *)0)
|
||||||
|
#define SIG_IGN _sig_ign
|
||||||
|
#define SIG_ERR ((void *)-1)
|
||||||
|
|
||||||
|
typedef void (*_Sighandler)(int);
|
||||||
|
|
||||||
|
struct sigaction {
|
||||||
|
void (*sa_handler)(int);
|
||||||
|
#define sa_sigaction sa_handler
|
||||||
|
unsigned long sa_flags;
|
||||||
|
void (*sa_restorer)(void);
|
||||||
|
unsigned long sa_mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
unsigned char _signal_restorer[] = {
|
||||||
|
0x48,0xb8,15,0,0,0,0,0,0,0, // mov rax, 15 (sigreturn)
|
||||||
|
0x0f,0x05 // syscall
|
||||||
|
};
|
||||||
|
|
||||||
|
#define _SIGNAL_HANDLERS 0xfff000
|
||||||
|
#define _LE64(x) (x)&0xff, ((x)>> 8)&0xff, ((x)>>16)&0xff, ((x)>>24)&0xff, \
|
||||||
|
((x)>>32)&0xff, ((x)>>40)&0xff, ((x)>>48)&0xff, (x)>>56
|
||||||
|
|
||||||
|
// we need to do this weird indirection because linux has a different
|
||||||
|
// calling convention from us.
|
||||||
|
|
||||||
|
unsigned char _signal_handler[] = {
|
||||||
|
// signal # passed in rdi
|
||||||
|
0x48,0x89,0xf8, // mov rax, rdi (signal #)
|
||||||
|
0x50, // push rax
|
||||||
|
0x50, // push rax (allocate space for return value)
|
||||||
|
0x48,0xb8,_LE64(_SIGNAL_HANDLERS), // mov rax, _SIGNAL_HANDLERS
|
||||||
|
0x48,0x89,0xc3, // mov rbx, rax
|
||||||
|
0x48,0x89,0xf8, // mov rax, rdi (signal #)
|
||||||
|
0x48,0xc1,0xe0,0x03, // shl rax, 3
|
||||||
|
0x48,0x01,0xd8, // add rax, rbx
|
||||||
|
0x48,0x89,0xc3, // mov rbx, rax
|
||||||
|
0x48,0x8b,0x03, // mov rax, [rbx]
|
||||||
|
0xff,0xd0, // call rax
|
||||||
|
0x48,0x81,0xc4,16,0,0,0, // add rsp, 16
|
||||||
|
0xc3 // ret
|
||||||
|
};
|
||||||
|
|
||||||
|
#define _SA_RESTORER 0x04000000
|
||||||
|
#define SA_SIGINFO 4
|
||||||
|
#define SA_RESETHAND 0x80000000
|
||||||
|
|
||||||
|
int __sigaction(int signum, const struct sigaction *act, struct sigaction *oldact) {
|
||||||
|
return __syscall(13, signum, act, oldact, 8, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void sigemptyset(unsigned long *set) {
|
||||||
|
*set = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void _sig_ign(int signal) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned long _sig_mask = 0;
|
||||||
|
|
||||||
|
_Sighandler signal(int sig, _Sighandler func) {
|
||||||
|
void **handlers = _SIGNAL_HANDLERS;
|
||||||
|
_Sighandler ret = handlers[sig];
|
||||||
|
if (func == SIG_IGN) {
|
||||||
|
func = _sig_ign;
|
||||||
|
}
|
||||||
|
handlers[sig] = func;
|
||||||
|
|
||||||
|
if (func == SIG_DFL) {
|
||||||
|
_sig_mask &= ~(1ul << (sig-1));
|
||||||
|
} else {
|
||||||
|
_sig_mask |= 1ul << (sig-1);
|
||||||
|
}
|
||||||
|
struct sigaction act = {0};
|
||||||
|
act.sa_handler = func == SIG_DFL ? SIG_DFL : (void*)_signal_handler;
|
||||||
|
act.sa_mask = _sig_mask;
|
||||||
|
act.sa_flags = _SA_RESTORER;
|
||||||
|
act.sa_restorer = _signal_restorer;
|
||||||
|
__sigaction(sig, &act, NULL);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int raise(int signal) {
|
||||||
|
return kill(getpid(), signal);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define FPE_INTDIV 1
|
||||||
|
#define FPE_FLTDIV 3
|
||||||
|
|
||||||
|
#define __SI_MAX_SIZE 128
|
||||||
|
#if __WORDSIZE == 64
|
||||||
|
# define __SI_PAD_SIZE ((__SI_MAX_SIZE / sizeof (int)) - 4)
|
||||||
|
#else
|
||||||
|
# define __SI_PAD_SIZE ((__SI_MAX_SIZE / sizeof (int)) - 3)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef __SI_ALIGNMENT
|
||||||
|
# define __SI_ALIGNMENT /* nothing */
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_BAND_TYPE
|
||||||
|
# define __SI_BAND_TYPE long int
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_CLOCK_T
|
||||||
|
# define __SI_CLOCK_T __clock_t
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_ERRNO_THEN_CODE
|
||||||
|
# define __SI_ERRNO_THEN_CODE 1
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_HAVE_SIGSYS
|
||||||
|
# define __SI_HAVE_SIGSYS 1
|
||||||
|
#endif
|
||||||
|
#ifndef __SI_SIGFAULT_ADDL
|
||||||
|
# define __SI_SIGFAULT_ADDL /* nothing */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef int __pid_t;
|
||||||
|
typedef unsigned __uid_t;
|
||||||
|
|
||||||
|
union __sigval
|
||||||
|
{
|
||||||
|
int __sival_int;
|
||||||
|
void *__sival_ptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef union __sigval __sigval_t;
|
||||||
|
typedef long __clock_t;
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
int si_signo; /* Signal number. */
|
||||||
|
#if __SI_ERRNO_THEN_CODE
|
||||||
|
int si_errno; /* If non-zero, an errno value associated with
|
||||||
|
this signal, as defined in <errno.h>. */
|
||||||
|
int si_code; /* Signal code. */
|
||||||
|
#else
|
||||||
|
int si_code;
|
||||||
|
int si_errno;
|
||||||
|
#endif
|
||||||
|
#if __WORDSIZE == 64
|
||||||
|
int __pad0; /* Explicit padding. */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
union
|
||||||
|
{
|
||||||
|
int _pad[__SI_PAD_SIZE];
|
||||||
|
|
||||||
|
/* kill(). */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Sending process ID. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
} _kill;
|
||||||
|
|
||||||
|
/* POSIX.1b timers. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
int si_tid; /* Timer ID. */
|
||||||
|
int si_overrun; /* Overrun count. */
|
||||||
|
__sigval_t si_sigval; /* Signal value. */
|
||||||
|
} _timer;
|
||||||
|
|
||||||
|
/* POSIX.1b signals. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Sending process ID. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
__sigval_t si_sigval; /* Signal value. */
|
||||||
|
} _rt;
|
||||||
|
|
||||||
|
/* SIGCHLD. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__pid_t si_pid; /* Which child. */
|
||||||
|
__uid_t si_uid; /* Real user ID of sending process. */
|
||||||
|
int si_status; /* Exit value or signal. */
|
||||||
|
__SI_CLOCK_T si_utime;
|
||||||
|
__SI_CLOCK_T si_stime;
|
||||||
|
} _sigchld;
|
||||||
|
|
||||||
|
/* SIGILL, SIGFPE, SIGSEGV, SIGBUS. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *si_addr; /* Faulting insn/memory ref. */
|
||||||
|
__SI_SIGFAULT_ADDL
|
||||||
|
short int si_addr_lsb; /* Valid LSB of the reported address. */
|
||||||
|
union
|
||||||
|
{
|
||||||
|
/* used when si_code=SEGV_BNDERR */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *_lower;
|
||||||
|
void *_upper;
|
||||||
|
} _addr_bnd;
|
||||||
|
/* used when si_code=SEGV_PKUERR */
|
||||||
|
uint32_t _pkey;
|
||||||
|
} _bounds;
|
||||||
|
} _sigfault;
|
||||||
|
|
||||||
|
/* SIGPOLL. */
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
__SI_BAND_TYPE si_band; /* Band event for SIGPOLL. */
|
||||||
|
int si_fd;
|
||||||
|
} _sigpoll;
|
||||||
|
|
||||||
|
/* SIGSYS. */
|
||||||
|
#if __SI_HAVE_SIGSYS
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
void *_call_addr; /* Calling user insn. */
|
||||||
|
int _syscall; /* Triggering system call number. */
|
||||||
|
unsigned int _arch; /* AUDIT_ARCH_* of syscall. */
|
||||||
|
} _sigsys;
|
||||||
|
#endif
|
||||||
|
} _sifields;
|
||||||
|
} siginfo_t __SI_ALIGNMENT;
|
||||||
|
|
||||||
|
|
||||||
|
/* X/Open requires some more fields with fixed names. */
|
||||||
|
#define si_pid _sifields._kill.si_pid
|
||||||
|
#define si_uid _sifields._kill.si_uid
|
||||||
|
#define si_timerid _sifields._timer.si_tid
|
||||||
|
#define si_overrun _sifields._timer.si_overrun
|
||||||
|
#define si_status _sifields._sigchld.si_status
|
||||||
|
#define si_utime _sifields._sigchld.si_utime
|
||||||
|
#define si_stime _sifields._sigchld.si_stime
|
||||||
|
#define si_value _sifields._rt.si_sigval
|
||||||
|
#define si_int _sifields._rt.si_sigval.sival_int
|
||||||
|
#define si_ptr _sifields._rt.si_sigval.sival_ptr
|
||||||
|
#define si_addr _sifields._sigfault.si_addr
|
||||||
|
#define si_addr_lsb _sifields._sigfault.si_addr_lsb
|
||||||
|
#define si_lower _sifields._sigfault._bounds._addr_bnd._lower
|
||||||
|
#define si_upper _sifields._sigfault._bounds._addr_bnd._upper
|
||||||
|
#define si_pkey _sifields._sigfault._bounds._pkey
|
||||||
|
#define si_band _sifields._sigpoll.si_band
|
||||||
|
#define si_fd _sifields._sigpoll.si_fd
|
||||||
|
#if __SI_HAVE_SIGSYS
|
||||||
|
# define si_call_addr _sifields._sigsys._call_addr
|
||||||
|
# define si_syscall _sifields._sigsys._syscall
|
||||||
|
# define si_arch _sifields._sigsys._arch
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
#endif // _SIGNAL_H
|
238
05/tcc-0.9.25/stab.def
Normal file
238
05/tcc-0.9.25/stab.def
Normal file
|
@ -0,0 +1,238 @@
|
||||||
|
/* Table of DBX symbol codes for the GNU system.
|
||||||
|
Copyright (C) 1988, 1997 Free Software Foundation, Inc.
|
||||||
|
This file is part of the GNU C Library.
|
||||||
|
|
||||||
|
The GNU C Library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Library General Public License as
|
||||||
|
published by the Free Software Foundation; either version 2 of the
|
||||||
|
License, or (at your option) any later version.
|
||||||
|
|
||||||
|
The GNU C Library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Library General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Library General Public
|
||||||
|
License along with the GNU C Library; see the file COPYING.LIB. If not,
|
||||||
|
write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
||||||
|
Boston, MA 02111-1307, USA. */
|
||||||
|
|
||||||
|
/* This contains contribution from Cygnus Support. */
|
||||||
|
#if 0
|
||||||
|
// random page break character for some reason
|
||||||
|
#endif
|
||||||
|
/* Global variable. Only the name is significant.
|
||||||
|
To find the address, look in the corresponding external symbol. */
|
||||||
|
__define_stab (N_GSYM, 0x20, "GSYM")
|
||||||
|
|
||||||
|
/* Function name for BSD Fortran. Only the name is significant.
|
||||||
|
To find the address, look in the corresponding external symbol. */
|
||||||
|
__define_stab (N_FNAME, 0x22, "FNAME")
|
||||||
|
|
||||||
|
/* Function name or text-segment variable for C. Value is its address.
|
||||||
|
Desc is supposedly starting line number, but GCC doesn't set it
|
||||||
|
and DBX seems not to miss it. */
|
||||||
|
__define_stab (N_FUN, 0x24, "FUN")
|
||||||
|
|
||||||
|
/* Data-segment variable with internal linkage. Value is its address.
|
||||||
|
"Static Sym". */
|
||||||
|
__define_stab (N_STSYM, 0x26, "STSYM")
|
||||||
|
|
||||||
|
/* BSS-segment variable with internal linkage. Value is its address. */
|
||||||
|
__define_stab (N_LCSYM, 0x28, "LCSYM")
|
||||||
|
|
||||||
|
/* Name of main routine. Only the name is significant.
|
||||||
|
This is not used in C. */
|
||||||
|
__define_stab (N_MAIN, 0x2a, "MAIN")
|
||||||
|
|
||||||
|
/* Global symbol in Pascal.
|
||||||
|
Supposedly the value is its line number; I'm skeptical. */
|
||||||
|
__define_stab (N_PC, 0x30, "PC")
|
||||||
|
|
||||||
|
/* Number of symbols: 0, files,,funcs,lines according to Ultrix V4.0. */
|
||||||
|
__define_stab (N_NSYMS, 0x32, "NSYMS")
|
||||||
|
|
||||||
|
/* "No DST map for sym: name, ,0,type,ignored" according to Ultrix V4.0. */
|
||||||
|
__define_stab (N_NOMAP, 0x34, "NOMAP")
|
||||||
|
|
||||||
|
/* New stab from Solaris. I don't know what it means, but it
|
||||||
|
don't seem to contain useful information. */
|
||||||
|
__define_stab (N_OBJ, 0x38, "OBJ")
|
||||||
|
|
||||||
|
/* New stab from Solaris. I don't know what it means, but it
|
||||||
|
don't seem to contain useful information. Possibly related to the
|
||||||
|
optimization flags used in this module. */
|
||||||
|
__define_stab (N_OPT, 0x3c, "OPT")
|
||||||
|
|
||||||
|
/* Register variable. Value is number of register. */
|
||||||
|
__define_stab (N_RSYM, 0x40, "RSYM")
|
||||||
|
|
||||||
|
/* Modula-2 compilation unit. Can someone say what info it contains? */
|
||||||
|
__define_stab (N_M2C, 0x42, "M2C")
|
||||||
|
|
||||||
|
/* Line number in text segment. Desc is the line number;
|
||||||
|
value is corresponding address. */
|
||||||
|
__define_stab (N_SLINE, 0x44, "SLINE")
|
||||||
|
|
||||||
|
/* Similar, for data segment. */
|
||||||
|
__define_stab (N_DSLINE, 0x46, "DSLINE")
|
||||||
|
|
||||||
|
/* Similar, for bss segment. */
|
||||||
|
__define_stab (N_BSLINE, 0x48, "BSLINE")
|
||||||
|
|
||||||
|
/* Sun's source-code browser stabs. ?? Don't know what the fields are.
|
||||||
|
Supposedly the field is "path to associated .cb file". THIS VALUE
|
||||||
|
OVERLAPS WITH N_BSLINE! */
|
||||||
|
__define_stab (N_BROWS, 0x48, "BROWS")
|
||||||
|
|
||||||
|
/* GNU Modula-2 definition module dependency. Value is the modification time
|
||||||
|
of the definition file. Other is non-zero if it is imported with the
|
||||||
|
GNU M2 keyword %INITIALIZE. Perhaps N_M2C can be used if there
|
||||||
|
are enough empty fields? */
|
||||||
|
__define_stab(N_DEFD, 0x4a, "DEFD")
|
||||||
|
|
||||||
|
/* THE FOLLOWING TWO STAB VALUES CONFLICT. Happily, one is for Modula-2
|
||||||
|
and one is for C++. Still,... */
|
||||||
|
/* GNU C++ exception variable. Name is variable name. */
|
||||||
|
__define_stab (N_EHDECL, 0x50, "EHDECL")
|
||||||
|
/* Modula2 info "for imc": name,,0,0,0 according to Ultrix V4.0. */
|
||||||
|
__define_stab (N_MOD2, 0x50, "MOD2")
|
||||||
|
|
||||||
|
/* GNU C++ `catch' clause. Value is its address. Desc is nonzero if
|
||||||
|
this entry is immediately followed by a CAUGHT stab saying what exception
|
||||||
|
was caught. Multiple CAUGHT stabs means that multiple exceptions
|
||||||
|
can be caught here. If Desc is 0, it means all exceptions are caught
|
||||||
|
here. */
|
||||||
|
__define_stab (N_CATCH, 0x54, "CATCH")
|
||||||
|
|
||||||
|
/* Structure or union element. Value is offset in the structure. */
|
||||||
|
__define_stab (N_SSYM, 0x60, "SSYM")
|
||||||
|
|
||||||
|
/* Name of main source file.
|
||||||
|
Value is starting text address of the compilation. */
|
||||||
|
__define_stab (N_SO, 0x64, "SO")
|
||||||
|
|
||||||
|
/* Automatic variable in the stack. Value is offset from frame pointer.
|
||||||
|
Also used for type descriptions. */
|
||||||
|
__define_stab (N_LSYM, 0x80, "LSYM")
|
||||||
|
|
||||||
|
/* Beginning of an include file. Only Sun uses this.
|
||||||
|
In an object file, only the name is significant.
|
||||||
|
The Sun linker puts data into some of the other fields. */
|
||||||
|
__define_stab (N_BINCL, 0x82, "BINCL")
|
||||||
|
|
||||||
|
/* Name of sub-source file (#include file).
|
||||||
|
Value is starting text address of the compilation. */
|
||||||
|
__define_stab (N_SOL, 0x84, "SOL")
|
||||||
|
|
||||||
|
/* Parameter variable. Value is offset from argument pointer.
|
||||||
|
(On most machines the argument pointer is the same as the frame pointer. */
|
||||||
|
__define_stab (N_PSYM, 0xa0, "PSYM")
|
||||||
|
|
||||||
|
/* End of an include file. No name.
|
||||||
|
This and N_BINCL act as brackets around the file's output.
|
||||||
|
In an object file, there is no significant data in this entry.
|
||||||
|
The Sun linker puts data into some of the fields. */
|
||||||
|
__define_stab (N_EINCL, 0xa2, "EINCL")
|
||||||
|
|
||||||
|
/* Alternate entry point. Value is its address. */
|
||||||
|
__define_stab (N_ENTRY, 0xa4, "ENTRY")
|
||||||
|
|
||||||
|
/* Beginning of lexical block.
|
||||||
|
The desc is the nesting level in lexical blocks.
|
||||||
|
The value is the address of the start of the text for the block.
|
||||||
|
The variables declared inside the block *precede* the N_LBRAC symbol. */
|
||||||
|
__define_stab (N_LBRAC, 0xc0, "LBRAC")
|
||||||
|
|
||||||
|
/* Place holder for deleted include file. Replaces a N_BINCL and everything
|
||||||
|
up to the corresponding N_EINCL. The Sun linker generates these when
|
||||||
|
it finds multiple identical copies of the symbols from an include file.
|
||||||
|
This appears only in output from the Sun linker. */
|
||||||
|
__define_stab (N_EXCL, 0xc2, "EXCL")
|
||||||
|
|
||||||
|
/* Modula-2 scope information. Can someone say what info it contains? */
|
||||||
|
__define_stab (N_SCOPE, 0xc4, "SCOPE")
|
||||||
|
|
||||||
|
/* End of a lexical block. Desc matches the N_LBRAC's desc.
|
||||||
|
The value is the address of the end of the text for the block. */
|
||||||
|
__define_stab (N_RBRAC, 0xe0, "RBRAC")
|
||||||
|
|
||||||
|
/* Begin named common block. Only the name is significant. */
|
||||||
|
__define_stab (N_BCOMM, 0xe2, "BCOMM")
|
||||||
|
|
||||||
|
/* End named common block. Only the name is significant
|
||||||
|
(and it should match the N_BCOMM). */
|
||||||
|
__define_stab (N_ECOMM, 0xe4, "ECOMM")
|
||||||
|
|
||||||
|
/* End common (local name): value is address.
|
||||||
|
I'm not sure how this is used. */
|
||||||
|
__define_stab (N_ECOML, 0xe8, "ECOML")
|
||||||
|
|
||||||
|
/* These STAB's are used on Gould systems for Non-Base register symbols
|
||||||
|
or something like that. FIXME. I have assigned the values at random
|
||||||
|
since I don't have a Gould here. Fixups from Gould folk welcome... */
|
||||||
|
__define_stab (N_NBTEXT, 0xF0, "NBTEXT")
|
||||||
|
__define_stab (N_NBDATA, 0xF2, "NBDATA")
|
||||||
|
__define_stab (N_NBBSS, 0xF4, "NBBSS")
|
||||||
|
__define_stab (N_NBSTS, 0xF6, "NBSTS")
|
||||||
|
__define_stab (N_NBLCS, 0xF8, "NBLCS")
|
||||||
|
|
||||||
|
/* Second symbol entry containing a length-value for the preceding entry.
|
||||||
|
The value is the length. */
|
||||||
|
__define_stab (N_LENG, 0xfe, "LENG")
|
||||||
|
#if 0
|
||||||
|
// random page break character for some reason
|
||||||
|
#endif
|
||||||
|
/* The above information, in matrix format.
|
||||||
|
|
||||||
|
STAB MATRIX
|
||||||
|
_________________________________________________
|
||||||
|
| 00 - 1F are not dbx stab symbols |
|
||||||
|
| In most cases, the low bit is the EXTernal bit|
|
||||||
|
|
||||||
|
| 00 UNDEF | 02 ABS | 04 TEXT | 06 DATA |
|
||||||
|
| 01 |EXT | 03 |EXT | 05 |EXT | 07 |EXT |
|
||||||
|
|
||||||
|
| 08 BSS | 0A INDR | 0C FN_SEQ | 0E |
|
||||||
|
| 09 |EXT | 0B | 0D | 0F |
|
||||||
|
|
||||||
|
| 10 | 12 COMM | 14 SETA | 16 SETT |
|
||||||
|
| 11 | 13 | 15 | 17 |
|
||||||
|
|
||||||
|
| 18 SETD | 1A SETB | 1C SETV | 1E WARNING|
|
||||||
|
| 19 | 1B | 1D | 1F FN |
|
||||||
|
|
||||||
|
|_______________________________________________|
|
||||||
|
| Debug entries with bit 01 set are unused. |
|
||||||
|
| 20 GSYM | 22 FNAME | 24 FUN | 26 STSYM |
|
||||||
|
| 28 LCSYM | 2A MAIN | 2C | 2E |
|
||||||
|
| 30 PC | 32 NSYMS | 34 NOMAP | 36 |
|
||||||
|
| 38 OBJ | 3A | 3C OPT | 3E |
|
||||||
|
| 40 RSYM | 42 M2C | 44 SLINE | 46 DSLINE |
|
||||||
|
| 48 BSLINE*| 4A DEFD | 4C | 4E |
|
||||||
|
| 50 EHDECL*| 52 | 54 CATCH | 56 |
|
||||||
|
| 58 | 5A | 5C | 5E |
|
||||||
|
| 60 SSYM | 62 | 64 SO | 66 |
|
||||||
|
| 68 | 6A | 6C | 6E |
|
||||||
|
| 70 | 72 | 74 | 76 |
|
||||||
|
| 78 | 7A | 7C | 7E |
|
||||||
|
| 80 LSYM | 82 BINCL | 84 SOL | 86 |
|
||||||
|
| 88 | 8A | 8C | 8E |
|
||||||
|
| 90 | 92 | 94 | 96 |
|
||||||
|
| 98 | 9A | 9C | 9E |
|
||||||
|
| A0 PSYM | A2 EINCL | A4 ENTRY | A6 |
|
||||||
|
| A8 | AA | AC | AE |
|
||||||
|
| B0 | B2 | B4 | B6 |
|
||||||
|
| B8 | BA | BC | BE |
|
||||||
|
| C0 LBRAC | C2 EXCL | C4 SCOPE | C6 |
|
||||||
|
| C8 | CA | CC | CE |
|
||||||
|
| D0 | D2 | D4 | D6 |
|
||||||
|
| D8 | DA | DC | DE |
|
||||||
|
| E0 RBRAC | E2 BCOMM | E4 ECOMM | E6 |
|
||||||
|
| E8 ECOML | EA | EC | EE |
|
||||||
|
| F0 | F2 | F4 | F6 |
|
||||||
|
| F8 | FA | FC | FE LENG |
|
||||||
|
+-----------------------------------------------+
|
||||||
|
* 50 EHDECL is also MOD2.
|
||||||
|
* 48 BSLINE is also BROWS.
|
||||||
|
*/
|
17
05/tcc-0.9.25/stab.h
Normal file
17
05/tcc-0.9.25/stab.h
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#ifndef __GNU_STAB__
|
||||||
|
|
||||||
|
/* Indicate the GNU stab.h is in use. */
|
||||||
|
|
||||||
|
#define __GNU_STAB__
|
||||||
|
|
||||||
|
#define __define_stab(NAME, CODE, STRING) NAME=CODE,
|
||||||
|
|
||||||
|
enum __stab_debug_code
|
||||||
|
{
|
||||||
|
#include "stab.def"
|
||||||
|
LAST_UNUSED_STAB_CODE
|
||||||
|
};
|
||||||
|
|
||||||
|
#undef __define_stab
|
||||||
|
|
||||||
|
#endif /* __GNU_STAB_ */
|
10
05/tcc-0.9.25/stdarg.h
Normal file
10
05/tcc-0.9.25/stdarg.h
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#ifndef _STDARG_H
|
||||||
|
#define _STDARG_H
|
||||||
|
|
||||||
|
typedef unsigned long va_list;
|
||||||
|
|
||||||
|
#define va_start(list, arg) ((list) = (unsigned long)&arg)
|
||||||
|
#define va_arg(list, type) (*((type *)(list += ((sizeof(type) + 7) & 0xfffffffffffffff8))))
|
||||||
|
#define va_end(list)
|
||||||
|
|
||||||
|
#endif // _STDARG_H
|
683
05/tcc-0.9.25/stdc_common.h
Normal file
683
05/tcc-0.9.25/stdc_common.h
Normal file
|
@ -0,0 +1,683 @@
|
||||||
|
#ifndef _STDC_COMMON_H
|
||||||
|
#define _STDC_COMMON_H
|
||||||
|
|
||||||
|
#define signed
|
||||||
|
#define volatile
|
||||||
|
#define register
|
||||||
|
#define const
|
||||||
|
#define NULL ((void*)0)
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
#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
|
||||||
|
|
||||||
|
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))
|
||||||
|
|
||||||
|
// we need to define ucontext_t
|
||||||
|
# define __ctx(fld) fld
|
||||||
|
typedef long long int greg_t;
|
||||||
|
#define __NGREG 23
|
||||||
|
typedef greg_t gregset_t[__NGREG];
|
||||||
|
#define _SIGSET_NWORDS (1024 / (8 * sizeof (unsigned long int)))
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
unsigned long int __val[_SIGSET_NWORDS];
|
||||||
|
} __sigset_t, sigset_t;
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
void *ss_sp;
|
||||||
|
int ss_flags;
|
||||||
|
size_t ss_size;
|
||||||
|
} stack_t;
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
REG_R8 = 0,
|
||||||
|
# define REG_R8 REG_R8
|
||||||
|
REG_R9,
|
||||||
|
# define REG_R9 REG_R9
|
||||||
|
REG_R10,
|
||||||
|
# define REG_R10 REG_R10
|
||||||
|
REG_R11,
|
||||||
|
# define REG_R11 REG_R11
|
||||||
|
REG_R12,
|
||||||
|
# define REG_R12 REG_R12
|
||||||
|
REG_R13,
|
||||||
|
# define REG_R13 REG_R13
|
||||||
|
REG_R14,
|
||||||
|
# define REG_R14 REG_R14
|
||||||
|
REG_R15,
|
||||||
|
# define REG_R15 REG_R15
|
||||||
|
REG_RDI,
|
||||||
|
# define REG_RDI REG_RDI
|
||||||
|
REG_RSI,
|
||||||
|
# define REG_RSI REG_RSI
|
||||||
|
REG_RBP,
|
||||||
|
# define REG_RBP REG_RBP
|
||||||
|
REG_RBX,
|
||||||
|
# define REG_RBX REG_RBX
|
||||||
|
REG_RDX,
|
||||||
|
# define REG_RDX REG_RDX
|
||||||
|
REG_RAX,
|
||||||
|
# define REG_RAX REG_RAX
|
||||||
|
REG_RCX,
|
||||||
|
# define REG_RCX REG_RCX
|
||||||
|
REG_RSP,
|
||||||
|
# define REG_RSP REG_RSP
|
||||||
|
REG_RIP,
|
||||||
|
# define REG_RIP REG_RIP
|
||||||
|
REG_EFL,
|
||||||
|
# define REG_EFL REG_EFL
|
||||||
|
REG_CSGSFS, /* Actually short cs, gs, fs, __pad0. */
|
||||||
|
# define REG_CSGSFS REG_CSGSFS
|
||||||
|
REG_ERR,
|
||||||
|
# define REG_ERR REG_ERR
|
||||||
|
REG_TRAPNO,
|
||||||
|
# define REG_TRAPNO REG_TRAPNO
|
||||||
|
REG_OLDMASK,
|
||||||
|
# define REG_OLDMASK REG_OLDMASK
|
||||||
|
REG_CR2
|
||||||
|
# define REG_CR2 REG_CR2
|
||||||
|
};
|
||||||
|
struct _libc_fpxreg
|
||||||
|
{
|
||||||
|
unsigned short int __ctx(significand)[4];
|
||||||
|
unsigned short int __ctx(exponent);
|
||||||
|
unsigned short int __glibc_reserved1[3];
|
||||||
|
};
|
||||||
|
struct _libc_xmmreg
|
||||||
|
{
|
||||||
|
uint32_t __ctx(element)[4];
|
||||||
|
};
|
||||||
|
struct _libc_fpstate
|
||||||
|
{
|
||||||
|
uint16_t __ctx(cwd);
|
||||||
|
uint16_t __ctx(swd);
|
||||||
|
uint16_t __ctx(ftw);
|
||||||
|
uint16_t __ctx(fop);
|
||||||
|
uint64_t __ctx(rip);
|
||||||
|
uint64_t __ctx(rdp);
|
||||||
|
uint32_t __ctx(mxcsr);
|
||||||
|
uint32_t __ctx(mxcr_mask);
|
||||||
|
struct _libc_fpxreg _st[8];
|
||||||
|
struct _libc_xmmreg _xmm[16];
|
||||||
|
uint32_t __glibc_reserved1[24];
|
||||||
|
};
|
||||||
|
typedef struct _libc_fpstate *fpregset_t;
|
||||||
|
typedef struct {
|
||||||
|
gregset_t __ctx(gregs);
|
||||||
|
fpregset_t __ctx(fpregs);
|
||||||
|
unsigned long long __reserved1 [8];
|
||||||
|
} mcontext_t;
|
||||||
|
typedef struct ucontext_t {
|
||||||
|
unsigned long int __ctx(uc_flags);
|
||||||
|
struct ucontext_t *uc_link;
|
||||||
|
stack_t uc_stack;
|
||||||
|
mcontext_t uc_mcontext;
|
||||||
|
sigset_t uc_sigmask;
|
||||||
|
struct _libc_fpstate __fpregs_mem;
|
||||||
|
unsigned long long int __ssp[4];
|
||||||
|
} ucontext_t;
|
||||||
|
|
||||||
|
long read(int fd, void *buf, size_t count) {
|
||||||
|
return __syscall(0, fd, buf, count, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
long write(int fd, void *buf, size_t count) {
|
||||||
|
return __syscall(1, fd, buf, count, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void _Exit(int status) {
|
||||||
|
return __syscall(60, status, 0, 0, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
int gettimeofday(struct timeval *tv, struct timezone *tz) {
|
||||||
|
return __syscall(96, tv, tz, 0, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef long time_t;
|
||||||
|
|
||||||
|
struct timespec {
|
||||||
|
time_t tv_sec;
|
||||||
|
long tv_nsec;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct timeval {
|
||||||
|
time_t tv_sec;
|
||||||
|
long tv_usec;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct timezone {
|
||||||
|
int tz_minuteswest;
|
||||||
|
int tz_dsttime;
|
||||||
|
};
|
||||||
|
|
||||||
|
char *getcwd(char *buf, size_t size) {
|
||||||
|
long n = __syscall(79, buf, size, 0, 0, 0, 0);
|
||||||
|
if (n < 0) return NULL;
|
||||||
|
return buf;
|
||||||
|
}
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define SIGABRT 6
|
||||||
|
#define SIGFPE 8
|
||||||
|
#define SIGKILL 9
|
||||||
|
#define SIGILL 4
|
||||||
|
#define SIGINT 2
|
||||||
|
#define SIGSEGV 11
|
||||||
|
#define SIGTERM 15
|
||||||
|
#define SIGBUS 7
|
||||||
|
void abort(void) {
|
||||||
|
kill(getpid(), SIGABRT);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int fd;
|
||||||
|
unsigned char eof;
|
||||||
|
unsigned char err;
|
||||||
|
unsigned char has_ungetc;
|
||||||
|
char ungetc; // character which was pushed by ungetc()
|
||||||
|
} FILE;
|
||||||
|
|
||||||
|
int errno;
|
||||||
|
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
|
||||||
|
|
||||||
|
|
||||||
|
int _clamp_long_to_int(long x) {
|
||||||
|
if (x < INT_MIN) return INT_MIN;
|
||||||
|
if (x > INT_MAX) return INT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
short _clamp_long_to_short(long x) {
|
||||||
|
if (x < SHRT_MIN) return SHRT_MIN;
|
||||||
|
if (x > SHRT_MAX) return SHRT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned _clamp_ulong_to_uint(unsigned long x) {
|
||||||
|
if (x > UINT_MAX) return UINT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned short _clamp_ulong_to_ushort(unsigned long x) {
|
||||||
|
if (x > USHRT_MAX) return USHRT_MAX;
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
|
||||||
|
int mprotect(void *addr, size_t len, int prot) {
|
||||||
|
return __syscall(10, addr, len, prot, 0, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
#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);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *malloc(size_t n) {
|
||||||
|
if (!n) return NULL;
|
||||||
|
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) {
|
||||||
|
if (!ptr) return;
|
||||||
|
uint64_t *memory = (char *)ptr - 16;
|
||||||
|
uint64_t size = *memory;
|
||||||
|
munmap(memory, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
size_t strlen(char *s) {
|
||||||
|
char *t = s;
|
||||||
|
while (*t) ++t;
|
||||||
|
return t - s;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
int isspace(int c) {
|
||||||
|
return c == ' ' || c == '\f' || c == '\n' || c == '\r' || c == '\t' || c == '\v';
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
if (endptr) *endptr = nptr;
|
||||||
|
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;
|
||||||
|
return LONG_MAX;
|
||||||
|
}
|
||||||
|
return (long)mag;
|
||||||
|
} else {
|
||||||
|
if (mag > (unsigned long)LONG_MAX + 1) {
|
||||||
|
errno = ERANGE;
|
||||||
|
return LONG_MIN;
|
||||||
|
}
|
||||||
|
return -(long)mag;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
typedef struct {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sum == _INFINITY) errno = ERANGE;
|
||||||
|
if (endptr) *endptr = nptr;
|
||||||
|
return sum * sign;
|
||||||
|
}
|
||||||
|
|
||||||
|
float strtof(const char *nptr, char **endptr) {
|
||||||
|
return strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
long double strtold(const char *nptr, char **endptr) {
|
||||||
|
return strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main();
|
||||||
|
|
||||||
|
static char **_envp;
|
||||||
|
static uint64_t _rand_seed;
|
||||||
|
|
||||||
|
int _main(int argc, char **argv) {
|
||||||
|
int i;
|
||||||
|
_Float p = {1, 0};
|
||||||
|
|
||||||
|
_envp = argv + argc + 1; // this is where the environment variables will be
|
||||||
|
|
||||||
|
stdin = &_stdin;
|
||||||
|
stdout = &_stdout;
|
||||||
|
stderr = &_stderr;
|
||||||
|
|
||||||
|
/*
|
||||||
|
"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;
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
|
||||||
|
exit(main(argc, argv));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#endif // _STDC_COMMON_H
|
8
05/tcc-0.9.25/stddef.h
Normal file
8
05/tcc-0.9.25/stddef.h
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
#ifndef _STDDEF_H
|
||||||
|
#define _STDDEF_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
#define offsetof(struct, member) ((size_t)(&((struct *)NULL)->member))
|
||||||
|
// @NONSTANDARD: we don't have wchar_t
|
||||||
|
|
||||||
|
#endif // _STDDEF_H
|
2270
05/tcc-0.9.25/stdio.h
Normal file
2270
05/tcc-0.9.25/stdio.h
Normal file
File diff suppressed because it is too large
Load diff
193
05/tcc-0.9.25/stdlib.h
Normal file
193
05/tcc-0.9.25/stdlib.h
Normal file
|
@ -0,0 +1,193 @@
|
||||||
|
#ifndef _STDLIB_H
|
||||||
|
#define _STDLIB_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
#define EXIT_FAILURE (-1)
|
||||||
|
#define EXIT_SUCCESS 0
|
||||||
|
#define RAND_MAX 2147483647
|
||||||
|
// @NONSTANDARD: we don't define MB_CUR_MAX or any of the mbtowc functions
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int quot;
|
||||||
|
int rem;
|
||||||
|
} div_t;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
long quot;
|
||||||
|
long rem;
|
||||||
|
} ldiv_t;
|
||||||
|
|
||||||
|
char *getenv(const char *name) {
|
||||||
|
int i, j;
|
||||||
|
for (i = 0; _envp[i]; ++i) {
|
||||||
|
char *key = _envp[i];
|
||||||
|
for (j = 0; key[j] != '=' && name[j]; ++j)
|
||||||
|
if (name[j] != key[j])
|
||||||
|
break;
|
||||||
|
if (key[j] == '=' && !name[j])
|
||||||
|
return key + (j+1);
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
double atof(const char *nptr) {
|
||||||
|
return strtod(nptr, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int atoi(const char *nptr) {
|
||||||
|
return _clamp_long_to_int(strtol(nptr, NULL, 10));
|
||||||
|
}
|
||||||
|
|
||||||
|
long atol(const char *nptr) {
|
||||||
|
return strtol(nptr, NULL, 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
int rand(void) {
|
||||||
|
// https://en.wikipedia.org/wiki/Linear_congruential_generator
|
||||||
|
// we're using musl/newlib's constants
|
||||||
|
_rand_seed = 6364136223846793005 * _rand_seed + 1;
|
||||||
|
return _rand_seed >> 33;
|
||||||
|
}
|
||||||
|
|
||||||
|
void srand(unsigned seed) {
|
||||||
|
_rand_seed = seed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *calloc(size_t nmemb, size_t size) {
|
||||||
|
if (nmemb > 0xffffffffffffffff / size)
|
||||||
|
return NULL;
|
||||||
|
// NB: our malloc implementation returns zeroed memory
|
||||||
|
return malloc(nmemb * size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void *realloc(void *ptr, size_t size) {
|
||||||
|
if (!ptr) return malloc(size);
|
||||||
|
if (!size) {
|
||||||
|
free(ptr);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
uint64_t *memory = (char *)ptr - 16;
|
||||||
|
uint64_t old_size = *memory;
|
||||||
|
uint64_t *new_memory = _mremap(memory, old_size, size, MREMAP_MAYMOVE);
|
||||||
|
if ((uint64_t)new_memory > 0xffffffffffff0000) return NULL;
|
||||||
|
*new_memory = size;
|
||||||
|
return (char *)new_memory + 16;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int atexit(void (*func)(void)) {
|
||||||
|
if (_n_exit_handlers >= 32) return -1;
|
||||||
|
_exit_handlers[_n_exit_handlers++] = func;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int system(const char *string) {
|
||||||
|
if (!string) return 1;
|
||||||
|
|
||||||
|
int pid = fork();
|
||||||
|
if (pid < 0) {
|
||||||
|
return -1;
|
||||||
|
} else if (pid == 0) {
|
||||||
|
// child
|
||||||
|
char *argv[] = {
|
||||||
|
"/bin/sh",
|
||||||
|
"-c",
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
};
|
||||||
|
argv[2] = string;
|
||||||
|
execve("/bin/sh", argv, _envp);
|
||||||
|
// on success, execve does not return.
|
||||||
|
_Exit(-1);
|
||||||
|
} else {
|
||||||
|
// parent
|
||||||
|
int status = 0;
|
||||||
|
int ret = wait4(pid, &status, 0, NULL);
|
||||||
|
if (ret != pid) return -1;
|
||||||
|
if (_WIFSIGNALED(status)) return -1;
|
||||||
|
return _WEXITSTATUS(status);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void *bsearch(const void *key, const void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *)) {
|
||||||
|
size_t lo = 0;
|
||||||
|
size_t hi = nmemb;
|
||||||
|
while (lo < hi) {
|
||||||
|
size_t mid = (lo + hi) >> 1;
|
||||||
|
void *elem = (char *)base + mid * size;
|
||||||
|
int cmp = compar(key, elem);
|
||||||
|
if (cmp < 0) {
|
||||||
|
// key < elem
|
||||||
|
hi = mid;
|
||||||
|
} else if (cmp) {
|
||||||
|
// key > elem
|
||||||
|
lo = mid + 1;
|
||||||
|
} else {
|
||||||
|
return elem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void qsort(void *base, size_t nmemb, size_t size, int (*compar)(const void *, const void *)) {
|
||||||
|
// quicksort
|
||||||
|
if (nmemb < 2) return;
|
||||||
|
|
||||||
|
void *temp = malloc(size);
|
||||||
|
void *mid = (char *)base + ((nmemb >> 1) * size); // choose middle element to speed up sorting an already-sorted array
|
||||||
|
size_t pivot_index = 0, i;
|
||||||
|
for (i = 0; i < nmemb; ++i) {
|
||||||
|
void *elem = (char *)base + i * size;
|
||||||
|
if (compar(elem, mid) < 0)
|
||||||
|
++pivot_index;
|
||||||
|
}
|
||||||
|
void *pivot = (char *)base + pivot_index * size;
|
||||||
|
memcpy(temp, pivot, size);
|
||||||
|
memcpy(pivot, mid, size);
|
||||||
|
memcpy(mid, temp, size);
|
||||||
|
|
||||||
|
char *l, *r = (char *)base + (nmemb-1) * size;
|
||||||
|
for (l = base; l < r;) {
|
||||||
|
if (compar(l, pivot) > 0) {
|
||||||
|
// swap l and r
|
||||||
|
memcpy(temp, l, size);
|
||||||
|
memcpy(l, r, size);
|
||||||
|
memcpy(r, temp, size);
|
||||||
|
r -= size;
|
||||||
|
} else {
|
||||||
|
// l is already in the right place
|
||||||
|
l += size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
qsort(base, pivot_index, size, compar);
|
||||||
|
qsort((char *)pivot + size, nmemb - 1 - pivot_index, size, compar);
|
||||||
|
|
||||||
|
free(temp);
|
||||||
|
}
|
||||||
|
|
||||||
|
int abs(int x) {
|
||||||
|
return x >= 0 ? x : -x;
|
||||||
|
}
|
||||||
|
|
||||||
|
long labs(long x) {
|
||||||
|
return x >= 0 ? x : -x;
|
||||||
|
}
|
||||||
|
|
||||||
|
div_t div(int numer, int denom) {
|
||||||
|
div_t d;
|
||||||
|
d.quot = numer / denom;
|
||||||
|
d.rem = numer % denom;
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
ldiv_t ldiv(long numer, long denom) {
|
||||||
|
ldiv_t d;
|
||||||
|
d.quot = numer / denom;
|
||||||
|
d.rem = numer % denom;
|
||||||
|
return d;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // _STDLIB_H
|
185
05/tcc-0.9.25/string.h
Normal file
185
05/tcc-0.9.25/string.h
Normal file
|
@ -0,0 +1,185 @@
|
||||||
|
#ifndef _STRING_H
|
||||||
|
#define _STRING_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
|
||||||
|
|
||||||
|
void *memmove(void *s1, const void *s2, size_t n) {
|
||||||
|
if (s1 < s2) return memcpy(s1, s2, n); // our memcpy does a forwards copy
|
||||||
|
// backwards copy
|
||||||
|
char *p = (char*)s1 + n, *q = (char*)s2 + n;
|
||||||
|
while (p > s1)
|
||||||
|
*--p = *--q;
|
||||||
|
return s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strcpy(char *s1, const char *s2) {
|
||||||
|
char *p = s1 - 1, *q = s2 - 1;
|
||||||
|
while ((*++p = *++q));
|
||||||
|
return s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strncpy(char *s1, const char *s2, size_t n) {
|
||||||
|
char *p = s1 - 1, *q = s2 - 1;
|
||||||
|
size_t i;
|
||||||
|
for (i = 0; i < n; ++i)
|
||||||
|
if (!(*++p = *++q))
|
||||||
|
break;
|
||||||
|
for (; i < n; ++i)
|
||||||
|
*++p = 0;
|
||||||
|
return s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strcat(char *s1, const char *s2) {
|
||||||
|
return strcpy(s1 + strlen(s1), s2);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strncat(char *s1, const char *s2, size_t n) {
|
||||||
|
// oddly, not equivalent to strncpy(s1 + strlen(s1), s2, n)
|
||||||
|
char *p = s1 + strlen(s1) - 1, *q = s2 - 1;
|
||||||
|
size_t i;
|
||||||
|
for (i = 0; i < n; ++i)
|
||||||
|
if (!(*++p = *++q))
|
||||||
|
break;
|
||||||
|
*++p = 0;
|
||||||
|
return s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int memcmp(const void *s1, const void *s2, size_t n) {
|
||||||
|
char *p = s1, *q = s2;
|
||||||
|
size_t i;
|
||||||
|
for (i = 0; i < n; ++i, ++p, ++q) {
|
||||||
|
if (*p > *q)
|
||||||
|
return 1;
|
||||||
|
if (*p < *q)
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int strcmp(const char *s1, const char *s2) {
|
||||||
|
char *p = s1, *q = s2;
|
||||||
|
for (; ; ++p, ++q) {
|
||||||
|
if (*p > *q)
|
||||||
|
return 1;
|
||||||
|
if (*p < *q)
|
||||||
|
return -1;
|
||||||
|
if (!*p) break;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int strcoll(const char *s1, const char *s2) {
|
||||||
|
// we only support the C locale
|
||||||
|
return strcmp(s1, s2);
|
||||||
|
}
|
||||||
|
|
||||||
|
int strncmp(const char *s1, const char *s2, size_t n) {
|
||||||
|
char *p = s1, *q = s2;
|
||||||
|
size_t i;
|
||||||
|
for (i = 0; i < n; ++i, ++p, ++q) {
|
||||||
|
if (*p > *q)
|
||||||
|
return 1;
|
||||||
|
if (*p < *q)
|
||||||
|
return -1;
|
||||||
|
if (!*p) break;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t strxfrm(char *s1, const char *s2, size_t n) {
|
||||||
|
// we only support the C locale
|
||||||
|
size_t l = strlen(s2);
|
||||||
|
if (l >= n) return l;
|
||||||
|
strcpy(s1, s2);
|
||||||
|
return l;
|
||||||
|
}
|
||||||
|
|
||||||
|
void *memchr(const void *s, int c, size_t n) {
|
||||||
|
char *p = s, *end = p + n;
|
||||||
|
while (p < end) {
|
||||||
|
if ((unsigned char)*p == c)
|
||||||
|
return p;
|
||||||
|
++p;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strchr(const char *s, int c) {
|
||||||
|
return memchr(s, c, strlen(s)+1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
size_t strcspn(const char *s1, const char *s2) {
|
||||||
|
const char *p, *q;
|
||||||
|
for (p = s1; *p; ++p) {
|
||||||
|
for (q = s2; *q; ++q) {
|
||||||
|
if (*p == *q)
|
||||||
|
goto ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ret:
|
||||||
|
return p - s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strpbrk(const char *s1, const char *s2) {
|
||||||
|
const char *p, *q;
|
||||||
|
for (p = s1; *p; ++p) {
|
||||||
|
for (q = s2; *q; ++q) {
|
||||||
|
if (*p == *q)
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strrchr(const char *s, int c) {
|
||||||
|
char *p;
|
||||||
|
for (p = s + strlen(s); p >= s; --p) {
|
||||||
|
if (*p == c)
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t strspn(const char *s1, const char *s2) {
|
||||||
|
const char *p, *q;
|
||||||
|
for (p = s1; *p; ++p) {
|
||||||
|
for (q = s2; *q; ++q) {
|
||||||
|
if (*p == *q) break;
|
||||||
|
}
|
||||||
|
if (!*q) break;
|
||||||
|
}
|
||||||
|
return p - s1;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strstr(const char *s1, const char *s2) {
|
||||||
|
char *p;
|
||||||
|
size_t l = strlen(s2);
|
||||||
|
for (p = s1; *p; ++p) {
|
||||||
|
if (memcmp(p, s2, l) == 0)
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *strtok(char *s1, const char *s2) {
|
||||||
|
static char *str;
|
||||||
|
if (s1) str = s1;
|
||||||
|
if (!str) return NULL;
|
||||||
|
char *p = str + strspn(str, s2);
|
||||||
|
if (!*p) {
|
||||||
|
str = NULL;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
char *q = strpbrk(p, s2);
|
||||||
|
if (q) {
|
||||||
|
*q = 0;
|
||||||
|
str = q + 1;
|
||||||
|
} else {
|
||||||
|
str = NULL;
|
||||||
|
}
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // _STRING_H
|
2241
05/tcc-0.9.25/tcc-doc.html
Normal file
2241
05/tcc-0.9.25/tcc-doc.html
Normal file
File diff suppressed because it is too large
Load diff
1227
05/tcc-0.9.25/tcc-doc.texi
Normal file
1227
05/tcc-0.9.25/tcc-doc.texi
Normal file
File diff suppressed because it is too large
Load diff
556
05/tcc-0.9.25/tcc.c
Normal file
556
05/tcc-0.9.25/tcc.c
Normal file
|
@ -0,0 +1,556 @@
|
||||||
|
/*
|
||||||
|
* TCC - Tiny C Compiler
|
||||||
|
*
|
||||||
|
* Copyright (c) 2001-2004 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "libtcc.c"
|
||||||
|
|
||||||
|
void help(void)
|
||||||
|
{
|
||||||
|
printf("tcc version " TCC_VERSION " - Tiny C Compiler - Copyright (C) 2001-2006 Fabrice Bellard\n"
|
||||||
|
"usage: tcc [-v] [-c] [-o outfile] [-Bdir] [-bench] [-Idir] [-Dsym[=val]] [-Usym]\n"
|
||||||
|
" [-Wwarn] [-g] [-b] [-bt N] [-Ldir] [-llib] [-shared] [-soname name]\n"
|
||||||
|
" [-static] [infile1 infile2...] [-run infile args...]\n"
|
||||||
|
"\n"
|
||||||
|
"General options:\n"
|
||||||
|
" -v display current version, increase verbosity\n"
|
||||||
|
" -c compile only - generate an object file\n"
|
||||||
|
" -o outfile set output filename\n"
|
||||||
|
" -Bdir set tcc internal library path\n"
|
||||||
|
" -bench output compilation statistics\n"
|
||||||
|
" -run run compiled source\n"
|
||||||
|
" -fflag set or reset (with 'no-' prefix) 'flag' (see man page)\n"
|
||||||
|
" -Wwarning set or reset (with 'no-' prefix) 'warning' (see man page)\n"
|
||||||
|
" -w disable all warnings\n"
|
||||||
|
"Preprocessor options:\n"
|
||||||
|
" -E preprocess only\n"
|
||||||
|
" -Idir add include path 'dir'\n"
|
||||||
|
" -Dsym[=val] define 'sym' with value 'val'\n"
|
||||||
|
" -Usym undefine 'sym'\n"
|
||||||
|
"Linker options:\n"
|
||||||
|
" -Ldir add library path 'dir'\n"
|
||||||
|
" -llib link with dynamic or static library 'lib'\n"
|
||||||
|
" -shared generate a shared library\n"
|
||||||
|
" -soname set name for shared library to be used at runtime\n"
|
||||||
|
" -static static linking\n"
|
||||||
|
" -rdynamic export all global symbols to dynamic linker\n"
|
||||||
|
" -r generate (relocatable) object file\n"
|
||||||
|
"Debugger options:\n"
|
||||||
|
" -g generate runtime debug info\n"
|
||||||
|
#ifdef CONFIG_TCC_BCHECK
|
||||||
|
" -b compile with built-in memory and bounds checker (implies -g)\n"
|
||||||
|
#endif
|
||||||
|
#ifdef CONFIG_TCC_BACKTRACE
|
||||||
|
" -bt N show N callers in stack traces\n"
|
||||||
|
#endif
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static char **files;
|
||||||
|
static int nb_files, nb_libraries;
|
||||||
|
static int multiple_files;
|
||||||
|
static int print_search_dirs;
|
||||||
|
static int output_type;
|
||||||
|
static int reloc_output;
|
||||||
|
static const char *outfile;
|
||||||
|
static int do_bench = 0;
|
||||||
|
|
||||||
|
#define TCC_OPTION_HAS_ARG 0x0001
|
||||||
|
#define TCC_OPTION_NOSEP 0x0002 /* cannot have space before option and arg */
|
||||||
|
|
||||||
|
typedef struct TCCOption {
|
||||||
|
const char *name;
|
||||||
|
uint16_t index;
|
||||||
|
uint16_t flags;
|
||||||
|
} TCCOption;
|
||||||
|
|
||||||
|
enum {
|
||||||
|
TCC_OPTION_HELP,
|
||||||
|
TCC_OPTION_I,
|
||||||
|
TCC_OPTION_D,
|
||||||
|
TCC_OPTION_U,
|
||||||
|
TCC_OPTION_L,
|
||||||
|
TCC_OPTION_B,
|
||||||
|
TCC_OPTION_l,
|
||||||
|
TCC_OPTION_bench,
|
||||||
|
TCC_OPTION_bt,
|
||||||
|
TCC_OPTION_b,
|
||||||
|
TCC_OPTION_g,
|
||||||
|
TCC_OPTION_c,
|
||||||
|
TCC_OPTION_static,
|
||||||
|
TCC_OPTION_shared,
|
||||||
|
TCC_OPTION_soname,
|
||||||
|
TCC_OPTION_o,
|
||||||
|
TCC_OPTION_r,
|
||||||
|
TCC_OPTION_Wl,
|
||||||
|
TCC_OPTION_W,
|
||||||
|
TCC_OPTION_O,
|
||||||
|
TCC_OPTION_m,
|
||||||
|
TCC_OPTION_f,
|
||||||
|
TCC_OPTION_nostdinc,
|
||||||
|
TCC_OPTION_nostdlib,
|
||||||
|
TCC_OPTION_print_search_dirs,
|
||||||
|
TCC_OPTION_rdynamic,
|
||||||
|
TCC_OPTION_run,
|
||||||
|
TCC_OPTION_v,
|
||||||
|
TCC_OPTION_w,
|
||||||
|
TCC_OPTION_pipe,
|
||||||
|
TCC_OPTION_E,
|
||||||
|
};
|
||||||
|
|
||||||
|
static const TCCOption tcc_options[] = {
|
||||||
|
{ "h", TCC_OPTION_HELP, 0 },
|
||||||
|
{ "?", TCC_OPTION_HELP, 0 },
|
||||||
|
{ "I", TCC_OPTION_I, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "D", TCC_OPTION_D, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "U", TCC_OPTION_U, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "L", TCC_OPTION_L, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "B", TCC_OPTION_B, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "l", TCC_OPTION_l, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "bench", TCC_OPTION_bench, 0 },
|
||||||
|
{ "bt", TCC_OPTION_bt, TCC_OPTION_HAS_ARG },
|
||||||
|
#ifdef CONFIG_TCC_BCHECK
|
||||||
|
{ "b", TCC_OPTION_b, 0 },
|
||||||
|
#endif
|
||||||
|
{ "g", TCC_OPTION_g, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "c", TCC_OPTION_c, 0 },
|
||||||
|
{ "static", TCC_OPTION_static, 0 },
|
||||||
|
{ "shared", TCC_OPTION_shared, 0 },
|
||||||
|
{ "soname", TCC_OPTION_soname, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "o", TCC_OPTION_o, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "run", TCC_OPTION_run, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "rdynamic", TCC_OPTION_rdynamic, 0 },
|
||||||
|
{ "r", TCC_OPTION_r, 0 },
|
||||||
|
{ "Wl,", TCC_OPTION_Wl, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "W", TCC_OPTION_W, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "O", TCC_OPTION_O, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "m", TCC_OPTION_m, TCC_OPTION_HAS_ARG },
|
||||||
|
{ "f", TCC_OPTION_f, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "nostdinc", TCC_OPTION_nostdinc, 0 },
|
||||||
|
{ "nostdlib", TCC_OPTION_nostdlib, 0 },
|
||||||
|
{ "print-search-dirs", TCC_OPTION_print_search_dirs, 0 },
|
||||||
|
{ "v", TCC_OPTION_v, TCC_OPTION_HAS_ARG | TCC_OPTION_NOSEP },
|
||||||
|
{ "w", TCC_OPTION_w, 0 },
|
||||||
|
{ "pipe", TCC_OPTION_pipe, 0},
|
||||||
|
{ "E", TCC_OPTION_E, 0},
|
||||||
|
{ NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
static int64_t getclock_us(void)
|
||||||
|
{
|
||||||
|
#ifdef _WIN32
|
||||||
|
struct _timeb tb;
|
||||||
|
_ftime(&tb);
|
||||||
|
return (tb.time * 1000LL + tb.millitm) * 1000LL;
|
||||||
|
#else
|
||||||
|
struct timeval tv;
|
||||||
|
gettimeofday(&tv, NULL);
|
||||||
|
return tv.tv_sec * 1000000LL + tv.tv_usec;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static int strstart(const char *str, const char *val, const char **ptr)
|
||||||
|
{
|
||||||
|
const char *p, *q;
|
||||||
|
p = str;
|
||||||
|
q = val;
|
||||||
|
while (*q != '\0') {
|
||||||
|
if (*p != *q)
|
||||||
|
return 0;
|
||||||
|
p++;
|
||||||
|
q++;
|
||||||
|
}
|
||||||
|
if (ptr)
|
||||||
|
*ptr = p;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* convert 'str' into an array of space separated strings */
|
||||||
|
static int expand_args(char ***pargv, const char *str)
|
||||||
|
{
|
||||||
|
const char *s1;
|
||||||
|
char **argv, *arg;
|
||||||
|
int argc, len;
|
||||||
|
|
||||||
|
argc = 0;
|
||||||
|
argv = NULL;
|
||||||
|
for(;;) {
|
||||||
|
while (is_space(*str))
|
||||||
|
str++;
|
||||||
|
if (*str == '\0')
|
||||||
|
break;
|
||||||
|
s1 = str;
|
||||||
|
while (*str != '\0' && !is_space(*str))
|
||||||
|
str++;
|
||||||
|
len = str - s1;
|
||||||
|
arg = tcc_malloc(len + 1);
|
||||||
|
memcpy(arg, s1, len);
|
||||||
|
arg[len] = '\0';
|
||||||
|
dynarray_add((void ***)&argv, &argc, arg);
|
||||||
|
}
|
||||||
|
*pargv = argv;
|
||||||
|
return argc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int parse_args(TCCState *s, int argc, char **argv)
|
||||||
|
{
|
||||||
|
int optind;
|
||||||
|
const TCCOption *popt;
|
||||||
|
const char *optarg, *p1, *r1;
|
||||||
|
char *r;
|
||||||
|
|
||||||
|
optind = 0;
|
||||||
|
while (optind < argc) {
|
||||||
|
|
||||||
|
r = argv[optind++];
|
||||||
|
if (r[0] != '-' || r[1] == '\0') {
|
||||||
|
/* add a new file */
|
||||||
|
dynarray_add((void ***)&files, &nb_files, r);
|
||||||
|
if (!multiple_files) {
|
||||||
|
optind--;
|
||||||
|
/* argv[0] will be this file */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* find option in table (match only the first chars */
|
||||||
|
popt = tcc_options;
|
||||||
|
for(;;) {
|
||||||
|
p1 = popt->name;
|
||||||
|
if (p1 == NULL)
|
||||||
|
error("invalid option -- '%s'", r);
|
||||||
|
r1 = r + 1;
|
||||||
|
for(;;) {
|
||||||
|
if (*p1 == '\0')
|
||||||
|
goto option_found;
|
||||||
|
if (*r1 != *p1)
|
||||||
|
break;
|
||||||
|
p1++;
|
||||||
|
r1++;
|
||||||
|
}
|
||||||
|
popt++;
|
||||||
|
}
|
||||||
|
option_found:
|
||||||
|
if (popt->flags & TCC_OPTION_HAS_ARG) {
|
||||||
|
if (*r1 != '\0' || (popt->flags & TCC_OPTION_NOSEP)) {
|
||||||
|
optarg = r1;
|
||||||
|
} else {
|
||||||
|
if (optind >= argc)
|
||||||
|
error("argument to '%s' is missing", r);
|
||||||
|
optarg = argv[optind++];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (*r1 != '\0')
|
||||||
|
return 0;
|
||||||
|
optarg = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch(popt->index) {
|
||||||
|
case TCC_OPTION_HELP:
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
case TCC_OPTION_I:
|
||||||
|
if (tcc_add_include_path(s, optarg) < 0)
|
||||||
|
error("too many include paths");
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_D:
|
||||||
|
{
|
||||||
|
char *sym, *value;
|
||||||
|
sym = (char *)optarg;
|
||||||
|
value = strchr(sym, '=');
|
||||||
|
if (value) {
|
||||||
|
*value = '\0';
|
||||||
|
value++;
|
||||||
|
}
|
||||||
|
tcc_define_symbol(s, sym, value);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_U:
|
||||||
|
tcc_undefine_symbol(s, optarg);
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_L:
|
||||||
|
tcc_add_library_path(s, optarg);
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_B:
|
||||||
|
/* set tcc utilities path (mainly for tcc development) */
|
||||||
|
tcc_set_lib_path(s, optarg);
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_l:
|
||||||
|
dynarray_add((void ***)&files, &nb_files, r);
|
||||||
|
nb_libraries++;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_bench:
|
||||||
|
do_bench = 1;
|
||||||
|
break;
|
||||||
|
#ifdef CONFIG_TCC_BACKTRACE
|
||||||
|
case TCC_OPTION_bt:
|
||||||
|
num_callers = atoi(optarg);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef CONFIG_TCC_BCHECK
|
||||||
|
case TCC_OPTION_b:
|
||||||
|
s->do_bounds_check = 1;
|
||||||
|
s->do_debug = 1;
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
case TCC_OPTION_g:
|
||||||
|
s->do_debug = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_c:
|
||||||
|
multiple_files = 1;
|
||||||
|
output_type = TCC_OUTPUT_OBJ;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_static:
|
||||||
|
s->static_link = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_shared:
|
||||||
|
output_type = TCC_OUTPUT_DLL;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_soname:
|
||||||
|
s->soname = optarg;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_o:
|
||||||
|
multiple_files = 1;
|
||||||
|
outfile = optarg;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_r:
|
||||||
|
/* generate a .o merging several output files */
|
||||||
|
reloc_output = 1;
|
||||||
|
output_type = TCC_OUTPUT_OBJ;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_nostdinc:
|
||||||
|
s->nostdinc = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_nostdlib:
|
||||||
|
s->nostdlib = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_print_search_dirs:
|
||||||
|
print_search_dirs = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_run:
|
||||||
|
{
|
||||||
|
int argc1;
|
||||||
|
char **argv1;
|
||||||
|
argc1 = expand_args(&argv1, optarg);
|
||||||
|
if (argc1 > 0) {
|
||||||
|
parse_args(s, argc1, argv1);
|
||||||
|
}
|
||||||
|
multiple_files = 0;
|
||||||
|
output_type = TCC_OUTPUT_MEMORY;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_v:
|
||||||
|
do {
|
||||||
|
if (0 == s->verbose++)
|
||||||
|
printf("tcc version %s\n", TCC_VERSION);
|
||||||
|
} while (*optarg++ == 'v');
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_f:
|
||||||
|
if (tcc_set_flag(s, optarg, 1) < 0 && s->warn_unsupported)
|
||||||
|
goto unsupported_option;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_W:
|
||||||
|
if (tcc_set_warning(s, optarg, 1) < 0 &&
|
||||||
|
s->warn_unsupported)
|
||||||
|
goto unsupported_option;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_w:
|
||||||
|
s->warn_none = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_rdynamic:
|
||||||
|
s->rdynamic = 1;
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_Wl:
|
||||||
|
{
|
||||||
|
const char *p;
|
||||||
|
if (strstart(optarg, "-Ttext,", &p)) {
|
||||||
|
s->text_addr = strtoul(p, NULL, 16);
|
||||||
|
s->has_text_addr = 1;
|
||||||
|
} else if (strstart(optarg, "--oformat,", &p)) {
|
||||||
|
if (strstart(p, "elf32-", NULL)) {
|
||||||
|
s->output_format = TCC_OUTPUT_FORMAT_ELF;
|
||||||
|
} else if (!strcmp(p, "binary")) {
|
||||||
|
s->output_format = TCC_OUTPUT_FORMAT_BINARY;
|
||||||
|
} else
|
||||||
|
#ifdef TCC_TARGET_COFF
|
||||||
|
if (!strcmp(p, "coff")) {
|
||||||
|
s->output_format = TCC_OUTPUT_FORMAT_COFF;
|
||||||
|
} else
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
error("target %s not found", p);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
error("unsupported linker option '%s'", optarg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case TCC_OPTION_E:
|
||||||
|
output_type = TCC_OUTPUT_PREPROCESS;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (s->warn_unsupported) {
|
||||||
|
unsupported_option:
|
||||||
|
warning("unsupported option '%s'", r);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return optind + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
TCCState *s;
|
||||||
|
int nb_objfiles, ret, optind;
|
||||||
|
char objfilename[1024];
|
||||||
|
int64_t start_time = 0;
|
||||||
|
_init_tcc_syms();
|
||||||
|
_init_warning_defs();
|
||||||
|
_init_flag_defs();
|
||||||
|
|
||||||
|
s = tcc_new();
|
||||||
|
#ifdef _WIN32
|
||||||
|
tcc_set_lib_path_w32(s);
|
||||||
|
#endif
|
||||||
|
output_type = TCC_OUTPUT_EXE;
|
||||||
|
outfile = NULL;
|
||||||
|
multiple_files = 1;
|
||||||
|
files = NULL;
|
||||||
|
nb_files = 0;
|
||||||
|
nb_libraries = 0;
|
||||||
|
reloc_output = 0;
|
||||||
|
print_search_dirs = 0;
|
||||||
|
ret = 0;
|
||||||
|
|
||||||
|
optind = parse_args(s, argc - 1, argv + 1);
|
||||||
|
if (print_search_dirs) {
|
||||||
|
/* enough for Linux kernel */
|
||||||
|
printf("install: %s/\n", s->tcc_lib_path);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (optind == 0 || nb_files == 0) {
|
||||||
|
if (optind && s->verbose)
|
||||||
|
return 0;
|
||||||
|
help();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
nb_objfiles = nb_files - nb_libraries;
|
||||||
|
|
||||||
|
/* if outfile provided without other options, we output an
|
||||||
|
executable */
|
||||||
|
if (outfile && output_type == TCC_OUTPUT_MEMORY)
|
||||||
|
output_type = TCC_OUTPUT_EXE;
|
||||||
|
|
||||||
|
/* check -c consistency : only single file handled. XXX: checks file type */
|
||||||
|
if (output_type == TCC_OUTPUT_OBJ && !reloc_output) {
|
||||||
|
/* accepts only a single input file */
|
||||||
|
if (nb_objfiles != 1)
|
||||||
|
error("cannot specify multiple files with -c");
|
||||||
|
if (nb_libraries != 0)
|
||||||
|
error("cannot specify libraries with -c");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if (output_type == TCC_OUTPUT_PREPROCESS) {
|
||||||
|
if (!outfile) {
|
||||||
|
s->outfile = stdout;
|
||||||
|
} else {
|
||||||
|
s->outfile = fopen(outfile, "w");
|
||||||
|
if (!s->outfile)
|
||||||
|
error("could not open '%s", outfile);
|
||||||
|
}
|
||||||
|
} else if (output_type != TCC_OUTPUT_MEMORY) {
|
||||||
|
if (!outfile) {
|
||||||
|
/* compute default outfile name */
|
||||||
|
char *ext;
|
||||||
|
const char *name =
|
||||||
|
strcmp(files[0], "-") == 0 ? "a" : tcc_basename(files[0]);
|
||||||
|
pstrcpy(objfilename, sizeof(objfilename), name);
|
||||||
|
ext = tcc_fileextension(objfilename);
|
||||||
|
#ifdef TCC_TARGET_PE
|
||||||
|
if (output_type == TCC_OUTPUT_DLL)
|
||||||
|
strcpy(ext, ".dll");
|
||||||
|
else
|
||||||
|
if (output_type == TCC_OUTPUT_EXE)
|
||||||
|
strcpy(ext, ".exe");
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
if (output_type == TCC_OUTPUT_OBJ && !reloc_output && *ext)
|
||||||
|
strcpy(ext, ".o");
|
||||||
|
else
|
||||||
|
pstrcpy(objfilename, sizeof(objfilename), "a.out");
|
||||||
|
outfile = objfilename;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (do_bench) {
|
||||||
|
start_time = getclock_us();
|
||||||
|
}
|
||||||
|
|
||||||
|
tcc_set_output_type(s, output_type);
|
||||||
|
|
||||||
|
/* compile or add each files or library */
|
||||||
|
for(i = 0; i < nb_files && ret == 0; i++) {
|
||||||
|
const char *filename;
|
||||||
|
|
||||||
|
filename = files[i];
|
||||||
|
if (filename[0] == '-' && filename[1]) {
|
||||||
|
if (tcc_add_library(s, filename + 2) < 0) {
|
||||||
|
error_noabort("cannot find %s", filename);
|
||||||
|
ret = 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (1 == s->verbose)
|
||||||
|
printf("-> %s\n", filename);
|
||||||
|
if (tcc_add_file(s, filename) < 0)
|
||||||
|
ret = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* free all files */
|
||||||
|
tcc_free(files);
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
goto the_end;
|
||||||
|
|
||||||
|
if (do_bench)
|
||||||
|
tcc_print_stats(s, getclock_us() - start_time);
|
||||||
|
|
||||||
|
if (s->output_type == TCC_OUTPUT_PREPROCESS) {
|
||||||
|
if (outfile)
|
||||||
|
fclose(s->outfile);
|
||||||
|
} else if (s->output_type == TCC_OUTPUT_MEMORY) {
|
||||||
|
ret = tcc_run(s, argc - optind, argv + optind);
|
||||||
|
} else
|
||||||
|
ret = tcc_output_file(s, outfile) ? 1 : 0;
|
||||||
|
the_end:
|
||||||
|
/* XXX: cannot do it with bound checking because of the malloc hooks */
|
||||||
|
if (!s->do_bounds_check)
|
||||||
|
tcc_delete(s);
|
||||||
|
|
||||||
|
#ifdef MEM_DEBUG
|
||||||
|
if (do_bench) {
|
||||||
|
printf("memory: %d bytes, max = %d bytes\n", mem_cur_size, mem_max_size);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
757
05/tcc-0.9.25/tcc.h
Normal file
757
05/tcc-0.9.25/tcc.h
Normal file
|
@ -0,0 +1,757 @@
|
||||||
|
/*
|
||||||
|
* TCC - Tiny C Compiler
|
||||||
|
*
|
||||||
|
* Copyright (c) 2001-2004 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define _GNU_SOURCE
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
#ifdef CONFIG_TCCBOOT
|
||||||
|
|
||||||
|
#include "tccboot.h"
|
||||||
|
#define CONFIG_TCC_STATIC
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <math.h>
|
||||||
|
#include <signal.h>
|
||||||
|
#include <setjmp.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
#include <windows.h>
|
||||||
|
#include <sys/timeb.h>
|
||||||
|
#include <io.h> /* open, close etc. */
|
||||||
|
#include <direct.h> /* getcwd */
|
||||||
|
#define inline __inline
|
||||||
|
#define inp next_inp
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* !CONFIG_TCCBOOT */
|
||||||
|
|
||||||
|
#ifndef PAGESIZE
|
||||||
|
#define PAGESIZE 4096
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "elf.h"
|
||||||
|
#include "stab.h"
|
||||||
|
|
||||||
|
#ifndef O_BINARY
|
||||||
|
#define O_BINARY 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "libtcc.h"
|
||||||
|
|
||||||
|
/* parser debug */
|
||||||
|
//#define PARSE_DEBUG
|
||||||
|
/* preprocessor debug */
|
||||||
|
//#define PP_DEBUG
|
||||||
|
/* include file debug */
|
||||||
|
//#define INC_DEBUG
|
||||||
|
|
||||||
|
//#define MEM_DEBUG
|
||||||
|
|
||||||
|
/* assembler debug */
|
||||||
|
//#define ASM_DEBUG
|
||||||
|
|
||||||
|
/* target selection */
|
||||||
|
//#define TCC_TARGET_I386 /* i386 code generator */
|
||||||
|
//#define TCC_TARGET_ARM /* ARMv4 code generator */
|
||||||
|
//#define TCC_TARGET_C67 /* TMS320C67xx code generator */
|
||||||
|
//#define TCC_TARGET_X86_64 /* x86-64 code generator */
|
||||||
|
|
||||||
|
/* default target is I386 */
|
||||||
|
#if !defined(TCC_TARGET_I386) && !defined(TCC_TARGET_ARM) && \
|
||||||
|
!defined(TCC_TARGET_C67) && !defined(TCC_TARGET_X86_64)
|
||||||
|
#define TCC_TARGET_I386
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(_WIN32) && !defined(TCC_UCLIBC) && !defined(TCC_TARGET_ARM) && \
|
||||||
|
!defined(TCC_TARGET_C67) && !defined(TCC_TARGET_X86_64)
|
||||||
|
#define CONFIG_TCC_BCHECK /* enable bound checking code */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(_WIN32) && !defined(TCC_TARGET_PE)
|
||||||
|
#define CONFIG_TCC_STATIC
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* define it to include assembler support */
|
||||||
|
#if !defined(TCC_TARGET_ARM) && !defined(TCC_TARGET_C67) && \
|
||||||
|
!defined(TCC_TARGET_X86_64)
|
||||||
|
#define CONFIG_TCC_ASM
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* object format selection */
|
||||||
|
#if defined(TCC_TARGET_C67)
|
||||||
|
#define TCC_TARGET_COFF
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(_WIN32) && !defined(CONFIG_TCCBOOT)
|
||||||
|
//#define CONFIG_TCC_BACKTRACE// this uses sigaction() which we don't have
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define FALSE 0
|
||||||
|
#define false 0
|
||||||
|
#define TRUE 1
|
||||||
|
#define true 1
|
||||||
|
typedef int BOOL;
|
||||||
|
|
||||||
|
/* path to find crt1.o, crti.o and crtn.o. Only needed when generating
|
||||||
|
executables or dlls */
|
||||||
|
#ifndef CONFIG_TCC_CRT_PREFIX
|
||||||
|
#define CONFIG_TCC_CRT_PREFIX CONFIG_SYSROOT "/usr/lib"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define INCLUDE_STACK_SIZE 32
|
||||||
|
#define IFDEF_STACK_SIZE 64
|
||||||
|
#define VSTACK_SIZE 256
|
||||||
|
#define STRING_MAX_SIZE 1024
|
||||||
|
#define PACK_STACK_SIZE 8
|
||||||
|
|
||||||
|
#define TOK_HASH_SIZE 8192 /* must be a power of two */
|
||||||
|
#define TOK_ALLOC_INCR 512 /* must be a power of two */
|
||||||
|
#define TOK_MAX_SIZE 4 /* token max size in int unit when stored in string */
|
||||||
|
|
||||||
|
/* token symbol management */
|
||||||
|
typedef struct TokenSym {
|
||||||
|
struct TokenSym *hash_next;
|
||||||
|
struct Sym *sym_define; /* direct pointer to define */
|
||||||
|
struct Sym *sym_label; /* direct pointer to label */
|
||||||
|
struct Sym *sym_struct; /* direct pointer to structure */
|
||||||
|
struct Sym *sym_identifier; /* direct pointer to identifier */
|
||||||
|
int tok; /* token number */
|
||||||
|
int len;
|
||||||
|
char str[1];
|
||||||
|
} TokenSym;
|
||||||
|
|
||||||
|
#ifdef TCC_TARGET_PE
|
||||||
|
typedef unsigned short nwchar_t;
|
||||||
|
#else
|
||||||
|
typedef int nwchar_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
typedef struct CString {
|
||||||
|
int size; /* size in bytes */
|
||||||
|
void *data; /* either 'char *' or 'nwchar_t *' */
|
||||||
|
int size_allocated;
|
||||||
|
void *data_allocated; /* if non NULL, data has been malloced */
|
||||||
|
} CString;
|
||||||
|
|
||||||
|
/* type definition */
|
||||||
|
typedef struct CType {
|
||||||
|
int t;
|
||||||
|
struct Sym *ref;
|
||||||
|
} CType;
|
||||||
|
|
||||||
|
/* constant value */
|
||||||
|
typedef union CValue {
|
||||||
|
long double ld;
|
||||||
|
double d;
|
||||||
|
float f;
|
||||||
|
int i;
|
||||||
|
unsigned int ui;
|
||||||
|
unsigned int ul; /* address (should be unsigned long on 64 bit cpu) */
|
||||||
|
long long ll;
|
||||||
|
unsigned long long ull;
|
||||||
|
struct CString *cstr;
|
||||||
|
void *ptr;
|
||||||
|
int tab[1];
|
||||||
|
} CValue;
|
||||||
|
|
||||||
|
/* value on stack */
|
||||||
|
typedef struct SValue {
|
||||||
|
CType type; /* type */
|
||||||
|
unsigned short r; /* register + flags */
|
||||||
|
unsigned short r2; /* second register, used for 'long long'
|
||||||
|
type. If not used, set to VT_CONST */
|
||||||
|
CValue c; /* constant, if VT_CONST */
|
||||||
|
struct Sym *sym; /* symbol, if (VT_SYM | VT_CONST) */
|
||||||
|
} SValue;
|
||||||
|
|
||||||
|
/* symbol management */
|
||||||
|
typedef struct Sym {
|
||||||
|
int v; /* symbol token */
|
||||||
|
long r; /* associated register */
|
||||||
|
long c; /* associated number */
|
||||||
|
CType type; /* associated type */
|
||||||
|
struct Sym *next; /* next related symbol */
|
||||||
|
struct Sym *prev; /* prev symbol in stack */
|
||||||
|
struct Sym *prev_tok; /* previous symbol for this token */
|
||||||
|
} Sym;
|
||||||
|
|
||||||
|
/* section definition */
|
||||||
|
/* XXX: use directly ELF structure for parameters ? */
|
||||||
|
/* special flag to indicate that the section should not be linked to
|
||||||
|
the other ones */
|
||||||
|
#define SHF_PRIVATE 0x80000000
|
||||||
|
|
||||||
|
/* special flag, too */
|
||||||
|
#define SECTION_ABS ((void *)1)
|
||||||
|
|
||||||
|
typedef struct Section {
|
||||||
|
unsigned long data_offset; /* current data offset */
|
||||||
|
unsigned char *data; /* section data */
|
||||||
|
unsigned long data_allocated; /* used for realloc() handling */
|
||||||
|
int sh_name; /* elf section name (only used during output) */
|
||||||
|
int sh_num; /* elf section number */
|
||||||
|
int sh_type; /* elf section type */
|
||||||
|
int sh_flags; /* elf section flags */
|
||||||
|
int sh_info; /* elf section info */
|
||||||
|
int sh_addralign; /* elf section alignment */
|
||||||
|
int sh_entsize; /* elf entry size */
|
||||||
|
unsigned long sh_size; /* section size (only used during output) */
|
||||||
|
unsigned long sh_addr; /* address at which the section is relocated */
|
||||||
|
unsigned long sh_offset; /* file offset */
|
||||||
|
int nb_hashed_syms; /* used to resize the hash table */
|
||||||
|
struct Section *link; /* link to another section */
|
||||||
|
struct Section *reloc; /* corresponding section for relocation, if any */
|
||||||
|
struct Section *hash; /* hash table for symbols */
|
||||||
|
struct Section *next;
|
||||||
|
char name[1]; /* section name */
|
||||||
|
} Section;
|
||||||
|
|
||||||
|
typedef struct DLLReference {
|
||||||
|
int level;
|
||||||
|
void *handle;
|
||||||
|
char name[1];
|
||||||
|
} DLLReference;
|
||||||
|
|
||||||
|
/* GNUC attribute definition */
|
||||||
|
typedef struct AttributeDef {
|
||||||
|
int aligned;
|
||||||
|
int packed;
|
||||||
|
Section *section;
|
||||||
|
int func_attr; /* calling convention, exports, ... */
|
||||||
|
} AttributeDef;
|
||||||
|
|
||||||
|
/* -------------------------------------------------- */
|
||||||
|
/* gr: wrappers for casting sym->r for other purposes */
|
||||||
|
typedef struct {
|
||||||
|
unsigned char func_call, func_args, func_export; // should be equivalent to the bitfields that were here before
|
||||||
|
} func_attr_t;
|
||||||
|
|
||||||
|
#define FUNC_CALL(r) (((func_attr_t*)&(r))->func_call)
|
||||||
|
#define FUNC_EXPORT(r) (((func_attr_t*)&(r))->func_export)
|
||||||
|
#define FUNC_ARGS(r) (((func_attr_t*)&(r))->func_args)
|
||||||
|
#define INLINE_DEF(r) (*(int **)&(r))
|
||||||
|
/* -------------------------------------------------- */
|
||||||
|
|
||||||
|
#define SYM_STRUCT 0x40000000 /* struct/union/enum symbol space */
|
||||||
|
#define SYM_FIELD 0x20000000 /* struct/union field symbol space */
|
||||||
|
#define SYM_FIRST_ANOM 0x10000000 /* first anonymous sym */
|
||||||
|
|
||||||
|
/* stored in 'Sym.c' field */
|
||||||
|
#define FUNC_NEW 1 /* ansi function prototype */
|
||||||
|
#define FUNC_OLD 2 /* old function prototype */
|
||||||
|
#define FUNC_ELLIPSIS 3 /* ansi function prototype with ... */
|
||||||
|
|
||||||
|
/* stored in 'Sym.r' field */
|
||||||
|
#define FUNC_CDECL 0 /* standard c call */
|
||||||
|
#define FUNC_STDCALL 1 /* pascal c call */
|
||||||
|
#define FUNC_FASTCALL1 2 /* first param in %eax */
|
||||||
|
#define FUNC_FASTCALL2 3 /* first parameters in %eax, %edx */
|
||||||
|
#define FUNC_FASTCALL3 4 /* first parameter in %eax, %edx, %ecx */
|
||||||
|
#define FUNC_FASTCALLW 5 /* first parameter in %ecx, %edx */
|
||||||
|
|
||||||
|
/* field 'Sym.t' for macros */
|
||||||
|
#define MACRO_OBJ 0 /* object like macro */
|
||||||
|
#define MACRO_FUNC 1 /* function like macro */
|
||||||
|
|
||||||
|
/* field 'Sym.r' for C labels */
|
||||||
|
#define LABEL_DEFINED 0 /* label is defined */
|
||||||
|
#define LABEL_FORWARD 1 /* label is forward defined */
|
||||||
|
#define LABEL_DECLARED 2 /* label is declared but never used */
|
||||||
|
|
||||||
|
/* type_decl() types */
|
||||||
|
#define TYPE_ABSTRACT 1 /* type without variable */
|
||||||
|
#define TYPE_DIRECT 2 /* type with variable */
|
||||||
|
|
||||||
|
#define IO_BUF_SIZE 8192
|
||||||
|
|
||||||
|
typedef struct BufferedFile {
|
||||||
|
uint8_t *buf_ptr;
|
||||||
|
uint8_t *buf_end;
|
||||||
|
int fd;
|
||||||
|
int line_num; /* current line number - here to simplify code */
|
||||||
|
int ifndef_macro; /* #ifndef macro / #endif search */
|
||||||
|
int ifndef_macro_saved; /* saved ifndef_macro */
|
||||||
|
int *ifdef_stack_ptr; /* ifdef_stack value at the start of the file */
|
||||||
|
char inc_type; /* type of include */
|
||||||
|
char inc_filename[512]; /* filename specified by the user */
|
||||||
|
char filename[1024]; /* current filename - here to simplify code */
|
||||||
|
unsigned char buffer[IO_BUF_SIZE + 1]; /* extra size for CH_EOB char */
|
||||||
|
} BufferedFile;
|
||||||
|
|
||||||
|
#define CH_EOB '\\' /* end of buffer or '\0' char in file */
|
||||||
|
#define CH_EOF (-1) /* end of file */
|
||||||
|
|
||||||
|
/* parsing state (used to save parser state to reparse part of the
|
||||||
|
source several times) */
|
||||||
|
typedef struct ParseState {
|
||||||
|
int *macro_ptr;
|
||||||
|
int line_num;
|
||||||
|
int tok;
|
||||||
|
CValue tokc;
|
||||||
|
} ParseState;
|
||||||
|
|
||||||
|
/* used to record tokens */
|
||||||
|
typedef struct TokenString {
|
||||||
|
int *str;
|
||||||
|
int len;
|
||||||
|
int allocated_len;
|
||||||
|
int last_line_num;
|
||||||
|
} TokenString;
|
||||||
|
|
||||||
|
/* include file cache, used to find files faster and also to eliminate
|
||||||
|
inclusion if the include file is protected by #ifndef ... #endif */
|
||||||
|
typedef struct CachedInclude {
|
||||||
|
int ifndef_macro;
|
||||||
|
int hash_next; /* -1 if none */
|
||||||
|
char type; /* '"' or '>' to give include type */
|
||||||
|
char filename[1]; /* path specified in #include */
|
||||||
|
} CachedInclude;
|
||||||
|
|
||||||
|
#define CACHED_INCLUDES_HASH_SIZE 512
|
||||||
|
|
||||||
|
#ifdef CONFIG_TCC_ASM
|
||||||
|
typedef struct ExprValue {
|
||||||
|
uint32_t v;
|
||||||
|
Sym *sym;
|
||||||
|
} ExprValue;
|
||||||
|
|
||||||
|
#define MAX_ASM_OPERANDS 30
|
||||||
|
typedef struct ASMOperand {
|
||||||
|
int id; /* GCC 3 optionnal identifier (0 if number only supported */
|
||||||
|
char *constraint;
|
||||||
|
char asm_str[16]; /* computed asm string for operand */
|
||||||
|
SValue *vt; /* C value of the expression */
|
||||||
|
int ref_index; /* if >= 0, gives reference to a output constraint */
|
||||||
|
int input_index; /* if >= 0, gives reference to an input constraint */
|
||||||
|
int priority; /* priority, used to assign registers */
|
||||||
|
int reg; /* if >= 0, register number used for this operand */
|
||||||
|
int is_llong; /* true if double register value */
|
||||||
|
int is_memory; /* true if memory operand */
|
||||||
|
int is_rw; /* for '+' modifier */
|
||||||
|
} ASMOperand;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct TCCState {
|
||||||
|
int output_type;
|
||||||
|
|
||||||
|
BufferedFile **include_stack_ptr;
|
||||||
|
int *ifdef_stack_ptr;
|
||||||
|
|
||||||
|
/* include file handling */
|
||||||
|
char **include_paths;
|
||||||
|
int nb_include_paths;
|
||||||
|
char **sysinclude_paths;
|
||||||
|
int nb_sysinclude_paths;
|
||||||
|
CachedInclude **cached_includes;
|
||||||
|
int nb_cached_includes;
|
||||||
|
|
||||||
|
char **library_paths;
|
||||||
|
int nb_library_paths;
|
||||||
|
|
||||||
|
/* array of all loaded dlls (including those referenced by loaded
|
||||||
|
dlls) */
|
||||||
|
DLLReference **loaded_dlls;
|
||||||
|
int nb_loaded_dlls;
|
||||||
|
|
||||||
|
/* sections */
|
||||||
|
Section **sections;
|
||||||
|
int nb_sections; /* number of sections, including first dummy section */
|
||||||
|
|
||||||
|
Section **priv_sections;
|
||||||
|
int nb_priv_sections; /* number of private sections */
|
||||||
|
|
||||||
|
/* got handling */
|
||||||
|
Section *got;
|
||||||
|
Section *plt;
|
||||||
|
unsigned long *got_offsets;
|
||||||
|
int nb_got_offsets;
|
||||||
|
/* give the correspondance from symtab indexes to dynsym indexes */
|
||||||
|
int *symtab_to_dynsym;
|
||||||
|
|
||||||
|
/* temporary dynamic symbol sections (for dll loading) */
|
||||||
|
Section *dynsymtab_section;
|
||||||
|
/* exported dynamic symbol section */
|
||||||
|
Section *dynsym;
|
||||||
|
|
||||||
|
int nostdinc; /* if true, no standard headers are added */
|
||||||
|
int nostdlib; /* if true, no standard libraries are added */
|
||||||
|
int nocommon; /* if true, do not use common symbols for .bss data */
|
||||||
|
|
||||||
|
/* if true, static linking is performed */
|
||||||
|
int static_link;
|
||||||
|
|
||||||
|
/* soname as specified on the command line (-soname) */
|
||||||
|
const char *soname;
|
||||||
|
|
||||||
|
/* if true, all symbols are exported */
|
||||||
|
int rdynamic;
|
||||||
|
|
||||||
|
/* if true, only link in referenced objects from archive */
|
||||||
|
int alacarte_link;
|
||||||
|
|
||||||
|
/* address of text section */
|
||||||
|
unsigned long text_addr;
|
||||||
|
int has_text_addr;
|
||||||
|
|
||||||
|
/* output format, see TCC_OUTPUT_FORMAT_xxx */
|
||||||
|
int output_format;
|
||||||
|
|
||||||
|
/* C language options */
|
||||||
|
int char_is_unsigned;
|
||||||
|
int leading_underscore;
|
||||||
|
|
||||||
|
/* warning switches */
|
||||||
|
int warn_write_strings;
|
||||||
|
int warn_unsupported;
|
||||||
|
int warn_error;
|
||||||
|
int warn_none;
|
||||||
|
int warn_implicit_function_declaration;
|
||||||
|
|
||||||
|
/* display some information during compilation */
|
||||||
|
int verbose;
|
||||||
|
/* compile with debug symbol (and use them if error during execution) */
|
||||||
|
int do_debug;
|
||||||
|
/* compile with built-in memory and bounds checker */
|
||||||
|
int do_bounds_check;
|
||||||
|
/* give the path of the tcc libraries */
|
||||||
|
const char *tcc_lib_path;
|
||||||
|
|
||||||
|
/* error handling */
|
||||||
|
void *error_opaque;
|
||||||
|
void (*error_func)(void *opaque, const char *msg);
|
||||||
|
int error_set_jmp_enabled;
|
||||||
|
jmp_buf error_jmp_buf;
|
||||||
|
int nb_errors;
|
||||||
|
|
||||||
|
/* tiny assembler state */
|
||||||
|
Sym *asm_labels;
|
||||||
|
|
||||||
|
/* see include_stack_ptr */
|
||||||
|
BufferedFile *include_stack[INCLUDE_STACK_SIZE];
|
||||||
|
|
||||||
|
/* see ifdef_stack_ptr */
|
||||||
|
int ifdef_stack[IFDEF_STACK_SIZE];
|
||||||
|
|
||||||
|
/* see cached_includes */
|
||||||
|
int cached_includes_hash[CACHED_INCLUDES_HASH_SIZE];
|
||||||
|
|
||||||
|
/* pack stack */
|
||||||
|
int pack_stack[PACK_STACK_SIZE];
|
||||||
|
int *pack_stack_ptr;
|
||||||
|
|
||||||
|
/* output file for preprocessing */
|
||||||
|
FILE *outfile;
|
||||||
|
|
||||||
|
/* for tcc_relocate */
|
||||||
|
int runtime_added;
|
||||||
|
|
||||||
|
#ifdef TCC_TARGET_X86_64
|
||||||
|
/* write PLT and GOT here */
|
||||||
|
char *runtime_plt_and_got;
|
||||||
|
unsigned int runtime_plt_and_got_offset;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
/* The current value can be: */
|
||||||
|
#define VT_VALMASK 0x00ff
|
||||||
|
#define VT_CONST 0x00f0 /* constant in vc
|
||||||
|
(must be first non register value) */
|
||||||
|
#define VT_LLOCAL 0x00f1 /* lvalue, offset on stack */
|
||||||
|
#define VT_LOCAL 0x00f2 /* offset on stack */
|
||||||
|
#define VT_CMP 0x00f3 /* the value is stored in processor flags (in vc) */
|
||||||
|
#define VT_JMP 0x00f4 /* value is the consequence of jmp true (even) */
|
||||||
|
#define VT_JMPI 0x00f5 /* value is the consequence of jmp false (odd) */
|
||||||
|
#define VT_LVAL 0x0100 /* var is an lvalue */
|
||||||
|
#define VT_SYM 0x0200 /* a symbol value is added */
|
||||||
|
#define VT_MUSTCAST 0x0400 /* value must be casted to be correct (used for
|
||||||
|
char/short stored in integer registers) */
|
||||||
|
#define VT_MUSTBOUND 0x0800 /* bound checking must be done before
|
||||||
|
dereferencing value */
|
||||||
|
#define VT_BOUNDED 0x8000 /* value is bounded. The address of the
|
||||||
|
bounding function call point is in vc */
|
||||||
|
#define VT_LVAL_BYTE 0x1000 /* lvalue is a byte */
|
||||||
|
#define VT_LVAL_SHORT 0x2000 /* lvalue is a short */
|
||||||
|
#define VT_LVAL_UNSIGNED 0x4000 /* lvalue is unsigned */
|
||||||
|
#define VT_LVAL_TYPE (VT_LVAL_BYTE | VT_LVAL_SHORT | VT_LVAL_UNSIGNED)
|
||||||
|
|
||||||
|
/* types */
|
||||||
|
#define VT_INT 0 /* integer type */
|
||||||
|
#define VT_BYTE 1 /* signed byte type */
|
||||||
|
#define VT_SHORT 2 /* short type */
|
||||||
|
#define VT_VOID 3 /* void type */
|
||||||
|
#define VT_PTR 4 /* pointer */
|
||||||
|
#define VT_ENUM 5 /* enum definition */
|
||||||
|
#define VT_FUNC 6 /* function type */
|
||||||
|
#define VT_STRUCT 7 /* struct/union definition */
|
||||||
|
#define VT_FLOAT 8 /* IEEE float */
|
||||||
|
#define VT_DOUBLE 9 /* IEEE double */
|
||||||
|
#define VT_LDOUBLE 10 /* IEEE long double */
|
||||||
|
#define VT_BOOL 11 /* ISOC99 boolean type */
|
||||||
|
#define VT_LLONG 12 /* 64 bit integer */
|
||||||
|
#define VT_LONG 13 /* long integer (NEVER USED as type, only
|
||||||
|
during parsing) */
|
||||||
|
#define VT_BTYPE 0x000f /* mask for basic type */
|
||||||
|
#define VT_UNSIGNED 0x0010 /* unsigned type */
|
||||||
|
#define VT_ARRAY 0x0020 /* array type (also has VT_PTR) */
|
||||||
|
#define VT_BITFIELD 0x0040 /* bitfield modifier */
|
||||||
|
#define VT_CONSTANT 0x0800 /* const modifier */
|
||||||
|
#define VT_VOLATILE 0x1000 /* volatile modifier */
|
||||||
|
#define VT_SIGNED 0x2000 /* signed type */
|
||||||
|
|
||||||
|
/* storage */
|
||||||
|
#define VT_EXTERN 0x00000080 /* extern definition */
|
||||||
|
#define VT_STATIC 0x00000100 /* static variable */
|
||||||
|
#define VT_TYPEDEF 0x00000200 /* typedef definition */
|
||||||
|
#define VT_INLINE 0x00000400 /* inline definition */
|
||||||
|
|
||||||
|
#define VT_STRUCT_SHIFT 16 /* shift for bitfield shift values */
|
||||||
|
|
||||||
|
/* type mask (except storage) */
|
||||||
|
#define VT_STORAGE (VT_EXTERN | VT_STATIC | VT_TYPEDEF | VT_INLINE)
|
||||||
|
#define VT_TYPE (~(VT_STORAGE))
|
||||||
|
|
||||||
|
/* token values */
|
||||||
|
|
||||||
|
/* warning: the following compare tokens depend on i386 asm code */
|
||||||
|
#define TOK_ULT 0x92
|
||||||
|
#define TOK_UGE 0x93
|
||||||
|
#define TOK_EQ 0x94
|
||||||
|
#define TOK_NE 0x95
|
||||||
|
#define TOK_ULE 0x96
|
||||||
|
#define TOK_UGT 0x97
|
||||||
|
#define TOK_Nset 0x98
|
||||||
|
#define TOK_Nclear 0x99
|
||||||
|
#define TOK_LT 0x9c
|
||||||
|
#define TOK_GE 0x9d
|
||||||
|
#define TOK_LE 0x9e
|
||||||
|
#define TOK_GT 0x9f
|
||||||
|
|
||||||
|
#define TOK_LAND 0xa0
|
||||||
|
#define TOK_LOR 0xa1
|
||||||
|
|
||||||
|
#define TOK_DEC 0xa2
|
||||||
|
#define TOK_MID 0xa3 /* inc/dec, to void constant */
|
||||||
|
#define TOK_INC 0xa4
|
||||||
|
#define TOK_UDIV 0xb0 /* unsigned division */
|
||||||
|
#define TOK_UMOD 0xb1 /* unsigned modulo */
|
||||||
|
#define TOK_PDIV 0xb2 /* fast division with undefined rounding for pointers */
|
||||||
|
#define TOK_CINT 0xb3 /* number in tokc */
|
||||||
|
#define TOK_CCHAR 0xb4 /* char constant in tokc */
|
||||||
|
#define TOK_STR 0xb5 /* pointer to string in tokc */
|
||||||
|
#define TOK_TWOSHARPS 0xb6 /* ## preprocessing token */
|
||||||
|
#define TOK_LCHAR 0xb7
|
||||||
|
#define TOK_LSTR 0xb8
|
||||||
|
#define TOK_CFLOAT 0xb9 /* float constant */
|
||||||
|
#define TOK_LINENUM 0xba /* line number info */
|
||||||
|
#define TOK_CDOUBLE 0xc0 /* double constant */
|
||||||
|
#define TOK_CLDOUBLE 0xc1 /* long double constant */
|
||||||
|
#define TOK_UMULL 0xc2 /* unsigned 32x32 -> 64 mul */
|
||||||
|
#define TOK_ADDC1 0xc3 /* add with carry generation */
|
||||||
|
#define TOK_ADDC2 0xc4 /* add with carry use */
|
||||||
|
#define TOK_SUBC1 0xc5 /* add with carry generation */
|
||||||
|
#define TOK_SUBC2 0xc6 /* add with carry use */
|
||||||
|
#define TOK_CUINT 0xc8 /* unsigned int constant */
|
||||||
|
#define TOK_CLLONG 0xc9 /* long long constant */
|
||||||
|
#define TOK_CULLONG 0xca /* unsigned long long constant */
|
||||||
|
#define TOK_ARROW 0xcb
|
||||||
|
#define TOK_DOTS 0xcc /* three dots */
|
||||||
|
#define TOK_SHR 0xcd /* unsigned shift right */
|
||||||
|
#define TOK_PPNUM 0xce /* preprocessor number */
|
||||||
|
|
||||||
|
#define TOK_SHL 0x01 /* shift left */
|
||||||
|
#define TOK_SAR 0x02 /* signed shift right */
|
||||||
|
|
||||||
|
/* assignement operators : normal operator or 0x80 */
|
||||||
|
#define TOK_A_MOD 0xa5
|
||||||
|
#define TOK_A_AND 0xa6
|
||||||
|
#define TOK_A_MUL 0xaa
|
||||||
|
#define TOK_A_ADD 0xab
|
||||||
|
#define TOK_A_SUB 0xad
|
||||||
|
#define TOK_A_DIV 0xaf
|
||||||
|
#define TOK_A_XOR 0xde
|
||||||
|
#define TOK_A_OR 0xfc
|
||||||
|
#define TOK_A_SHL 0x81
|
||||||
|
#define TOK_A_SAR 0x82
|
||||||
|
|
||||||
|
#ifndef offsetof
|
||||||
|
#define offsetof(type, field) ((size_t) &((type *)0)->field)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef countof
|
||||||
|
#define countof(tab) (sizeof(tab) / sizeof((tab)[0]))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define TOK_EOF (-1) /* end of file */
|
||||||
|
#define TOK_LINEFEED 10 /* line feed */
|
||||||
|
|
||||||
|
/* all identificators and strings have token above that */
|
||||||
|
#define TOK_IDENT 256
|
||||||
|
|
||||||
|
/* only used for i386 asm opcodes definitions */
|
||||||
|
#define DEF_ASM(x) DEF(TOK_ASM_ ## x, #x)
|
||||||
|
|
||||||
|
#define DEF_BWL(x) \
|
||||||
|
DEF(TOK_ASM_ ## x ## b, #x "b") \
|
||||||
|
DEF(TOK_ASM_ ## x ## w, #x "w") \
|
||||||
|
DEF(TOK_ASM_ ## x ## l, #x "l") \
|
||||||
|
DEF(TOK_ASM_ ## x, #x)
|
||||||
|
|
||||||
|
#define DEF_WL(x) \
|
||||||
|
DEF(TOK_ASM_ ## x ## w, #x "w") \
|
||||||
|
DEF(TOK_ASM_ ## x ## l, #x "l") \
|
||||||
|
DEF(TOK_ASM_ ## x, #x)
|
||||||
|
|
||||||
|
#define DEF_FP1(x) \
|
||||||
|
DEF(TOK_ASM_ ## f ## x ## s, "f" #x "s") \
|
||||||
|
DEF(TOK_ASM_ ## fi ## x ## l, "fi" #x "l") \
|
||||||
|
DEF(TOK_ASM_ ## f ## x ## l, "f" #x "l") \
|
||||||
|
DEF(TOK_ASM_ ## fi ## x ## s, "fi" #x "s")
|
||||||
|
|
||||||
|
#define DEF_FP(x) \
|
||||||
|
DEF(TOK_ASM_ ## f ## x, "f" #x ) \
|
||||||
|
DEF(TOK_ASM_ ## f ## x ## p, "f" #x "p") \
|
||||||
|
DEF_FP1(x)
|
||||||
|
|
||||||
|
#define DEF_ASMTEST(x) \
|
||||||
|
DEF_ASM(x ## o) \
|
||||||
|
DEF_ASM(x ## no) \
|
||||||
|
DEF_ASM(x ## b) \
|
||||||
|
DEF_ASM(x ## c) \
|
||||||
|
DEF_ASM(x ## nae) \
|
||||||
|
DEF_ASM(x ## nb) \
|
||||||
|
DEF_ASM(x ## nc) \
|
||||||
|
DEF_ASM(x ## ae) \
|
||||||
|
DEF_ASM(x ## e) \
|
||||||
|
DEF_ASM(x ## z) \
|
||||||
|
DEF_ASM(x ## ne) \
|
||||||
|
DEF_ASM(x ## nz) \
|
||||||
|
DEF_ASM(x ## be) \
|
||||||
|
DEF_ASM(x ## na) \
|
||||||
|
DEF_ASM(x ## nbe) \
|
||||||
|
DEF_ASM(x ## a) \
|
||||||
|
DEF_ASM(x ## s) \
|
||||||
|
DEF_ASM(x ## ns) \
|
||||||
|
DEF_ASM(x ## p) \
|
||||||
|
DEF_ASM(x ## pe) \
|
||||||
|
DEF_ASM(x ## np) \
|
||||||
|
DEF_ASM(x ## po) \
|
||||||
|
DEF_ASM(x ## l) \
|
||||||
|
DEF_ASM(x ## nge) \
|
||||||
|
DEF_ASM(x ## nl) \
|
||||||
|
DEF_ASM(x ## ge) \
|
||||||
|
DEF_ASM(x ## le) \
|
||||||
|
DEF_ASM(x ## ng) \
|
||||||
|
DEF_ASM(x ## nle) \
|
||||||
|
DEF_ASM(x ## g)
|
||||||
|
|
||||||
|
#define TOK_ASM_int TOK_INT
|
||||||
|
|
||||||
|
enum tcc_token {
|
||||||
|
TOK_LAST = TOK_IDENT - 1,
|
||||||
|
#define DEF(id, str) id,
|
||||||
|
#include "tcctok.h"
|
||||||
|
#undef DEF
|
||||||
|
};
|
||||||
|
|
||||||
|
#define TOK_UIDENT TOK_DEFINE
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
#define snprintf _snprintf
|
||||||
|
#define vsnprintf _vsnprintf
|
||||||
|
#ifndef __GNUC__
|
||||||
|
#define strtold (long double)strtod
|
||||||
|
#define strtof (float)strtod
|
||||||
|
#define strtoll (long long)strtol
|
||||||
|
#endif
|
||||||
|
#elif defined(TCC_UCLIBC) || defined(__FreeBSD__) || defined(__DragonFly__) \
|
||||||
|
|| defined(__OpenBSD__)
|
||||||
|
/* currently incorrect */
|
||||||
|
long double strtold(const char *nptr, char **endptr)
|
||||||
|
{
|
||||||
|
return (long double)strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
float strtof(const char *nptr, char **endptr)
|
||||||
|
{
|
||||||
|
return (float)strtod(nptr, endptr);
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
/* XXX: need to define this to use them in non ISOC99 context */
|
||||||
|
extern float strtof (const char *__nptr, char **__endptr);
|
||||||
|
extern long double strtold (const char *__nptr, char **__endptr);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
#define IS_PATHSEP(c) (c == '/' || c == '\\')
|
||||||
|
#define IS_ABSPATH(p) (IS_PATHSEP(p[0]) || (p[0] && p[1] == ':' && IS_PATHSEP(p[2])))
|
||||||
|
#define PATHCMP stricmp
|
||||||
|
#else
|
||||||
|
#define IS_PATHSEP(c) (c == '/')
|
||||||
|
#define IS_ABSPATH(p) IS_PATHSEP(p[0])
|
||||||
|
#define PATHCMP strcmp
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void error(const char *fmt, ...);
|
||||||
|
void error_noabort(const char *fmt, ...);
|
||||||
|
void warning(const char *fmt, ...);
|
||||||
|
|
||||||
|
void tcc_set_lib_path_w32(TCCState *s);
|
||||||
|
int tcc_set_flag(TCCState *s, const char *flag_name, int value);
|
||||||
|
void tcc_print_stats(TCCState *s, int64_t total_time);
|
||||||
|
|
||||||
|
void tcc_free(void *ptr);
|
||||||
|
void *tcc_malloc(unsigned long size);
|
||||||
|
void *tcc_mallocz(unsigned long size);
|
||||||
|
void *tcc_realloc(void *ptr, unsigned long size);
|
||||||
|
char *tcc_strdup(const char *str);
|
||||||
|
|
||||||
|
char *tcc_basename(const char *name);
|
||||||
|
char *tcc_fileextension (const char *name);
|
||||||
|
char *pstrcpy(char *buf, int buf_size, const char *s);
|
||||||
|
char *pstrcat(char *buf, int buf_size, const char *s);
|
||||||
|
void dynarray_add(void ***ptab, int *nb_ptr, void *data);
|
||||||
|
void dynarray_reset(void *pp, int *n);
|
||||||
|
|
||||||
|
#ifdef CONFIG_TCC_BACKTRACE
|
||||||
|
extern int num_callers;
|
||||||
|
extern const char **rt_bound_error_msg;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* true if float/double/long double type */
|
||||||
|
static inline int is_float(int t)
|
||||||
|
{
|
||||||
|
int bt;
|
||||||
|
bt = t & VT_BTYPE;
|
||||||
|
return bt == VT_LDOUBLE || bt == VT_DOUBLE || bt == VT_FLOAT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* space exlcuding newline */
|
||||||
|
static inline int is_space(int ch)
|
||||||
|
{
|
||||||
|
return ch == ' ' || ch == '\t' || ch == '\v' || ch == '\f' || ch == '\r';
|
||||||
|
}
|
||||||
|
|
1021
05/tcc-0.9.25/tccasm.c
Normal file
1021
05/tcc-0.9.25/tccasm.c
Normal file
File diff suppressed because it is too large
Load diff
957
05/tcc-0.9.25/tcccoff.c
Normal file
957
05/tcc-0.9.25/tcccoff.c
Normal file
|
@ -0,0 +1,957 @@
|
||||||
|
/*
|
||||||
|
* COFF file handling for TCC
|
||||||
|
*
|
||||||
|
* Copyright (c) 2003, 2004 TK
|
||||||
|
* Copyright (c) 2004 Fabrice Bellard
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
*/
|
||||||
|
#include "coff.h"
|
||||||
|
|
||||||
|
#define MAXNSCNS 255 /* MAXIMUM NUMBER OF SECTIONS */
|
||||||
|
#define MAX_STR_TABLE 1000000
|
||||||
|
AOUTHDR o_filehdr; /* OPTIONAL (A.OUT) FILE HEADER */
|
||||||
|
|
||||||
|
SCNHDR section_header[MAXNSCNS];
|
||||||
|
|
||||||
|
#define MAX_FUNCS 1000
|
||||||
|
#define MAX_FUNC_NAME_LENGTH 128
|
||||||
|
|
||||||
|
int nFuncs;
|
||||||
|
char Func[MAX_FUNCS][MAX_FUNC_NAME_LENGTH];
|
||||||
|
char AssociatedFile[MAX_FUNCS][MAX_FUNC_NAME_LENGTH];
|
||||||
|
int LineNoFilePtr[MAX_FUNCS];
|
||||||
|
int EndAddress[MAX_FUNCS];
|
||||||
|
int LastLineNo[MAX_FUNCS];
|
||||||
|
int FuncEntries[MAX_FUNCS];
|
||||||
|
|
||||||
|
BOOL OutputTheSection(Section * sect);
|
||||||
|
short int GetCoffFlags(const char *s);
|
||||||
|
void SortSymbolTable(void);
|
||||||
|
Section *FindSection(TCCState * s1, const char *sname);
|
||||||
|
|
||||||
|
int C67_main_entry_point;
|
||||||
|
|
||||||
|
int FindCoffSymbolIndex(const char *func_name);
|
||||||
|
int nb_syms;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
long tag;
|
||||||
|
long size;
|
||||||
|
long fileptr;
|
||||||
|
long nextsym;
|
||||||
|
short int dummy;
|
||||||
|
} AUXFUNC;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
long regmask;
|
||||||
|
unsigned short lineno;
|
||||||
|
unsigned short nentries;
|
||||||
|
int localframe;
|
||||||
|
int nextentry;
|
||||||
|
short int dummy;
|
||||||
|
} AUXBF;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
long dummy;
|
||||||
|
unsigned short lineno;
|
||||||
|
unsigned short dummy1;
|
||||||
|
int dummy2;
|
||||||
|
int dummy3;
|
||||||
|
unsigned short dummy4;
|
||||||
|
} AUXEF;
|
||||||
|
|
||||||
|
int tcc_output_coff(TCCState *s1, FILE *f)
|
||||||
|
{
|
||||||
|
Section *tcc_sect;
|
||||||
|
SCNHDR *coff_sec;
|
||||||
|
int file_pointer;
|
||||||
|
char *Coff_str_table, *pCoff_str_table;
|
||||||
|
int CoffTextSectionNo, coff_nb_syms;
|
||||||
|
FILHDR file_hdr; /* FILE HEADER STRUCTURE */
|
||||||
|
Section *stext, *sdata, *sbss;
|
||||||
|
int i, NSectionsToOutput = 0;
|
||||||
|
|
||||||
|
Coff_str_table = pCoff_str_table = NULL;
|
||||||
|
|
||||||
|
stext = FindSection(s1, ".text");
|
||||||
|
sdata = FindSection(s1, ".data");
|
||||||
|
sbss = FindSection(s1, ".bss");
|
||||||
|
|
||||||
|
nb_syms = symtab_section->data_offset / sizeof(Elf32_Sym);
|
||||||
|
coff_nb_syms = FindCoffSymbolIndex("XXXXXXXXXX1");
|
||||||
|
|
||||||
|
file_hdr.f_magic = COFF_C67_MAGIC; /* magic number */
|
||||||
|
file_hdr.f_timdat = 0; /* time & date stamp */
|
||||||
|
file_hdr.f_opthdr = sizeof(AOUTHDR); /* sizeof(optional hdr) */
|
||||||
|
file_hdr.f_flags = 0x1143; /* flags (copied from what code composer does) */
|
||||||
|
file_hdr.f_TargetID = 0x99; /* for C6x = 0x0099 */
|
||||||
|
|
||||||
|
o_filehdr.magic = 0x0108; /* see magic.h */
|
||||||
|
o_filehdr.vstamp = 0x0190; /* version stamp */
|
||||||
|
o_filehdr.tsize = stext->data_offset; /* text size in bytes, padded to FW bdry */
|
||||||
|
o_filehdr.dsize = sdata->data_offset; /* initialized data " " */
|
||||||
|
o_filehdr.bsize = sbss->data_offset; /* uninitialized data " " */
|
||||||
|
o_filehdr.entrypt = C67_main_entry_point; /* entry pt. */
|
||||||
|
o_filehdr.text_start = stext->sh_addr; /* base of text used for this file */
|
||||||
|
o_filehdr.data_start = sdata->sh_addr; /* base of data used for this file */
|
||||||
|
|
||||||
|
|
||||||
|
// create all the section headers
|
||||||
|
|
||||||
|
file_pointer = FILHSZ + sizeof(AOUTHDR);
|
||||||
|
|
||||||
|
CoffTextSectionNo = -1;
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
NSectionsToOutput++;
|
||||||
|
|
||||||
|
if (CoffTextSectionNo == -1 && tcc_sect == stext)
|
||||||
|
CoffTextSectionNo = NSectionsToOutput; // rem which coff sect number the .text sect is
|
||||||
|
|
||||||
|
strcpy(coff_sec->s_name, tcc_sect->name); /* section name */
|
||||||
|
|
||||||
|
coff_sec->s_paddr = tcc_sect->sh_addr; /* physical address */
|
||||||
|
coff_sec->s_vaddr = tcc_sect->sh_addr; /* virtual address */
|
||||||
|
coff_sec->s_size = tcc_sect->data_offset; /* section size */
|
||||||
|
coff_sec->s_scnptr = 0; /* file ptr to raw data for section */
|
||||||
|
coff_sec->s_relptr = 0; /* file ptr to relocation */
|
||||||
|
coff_sec->s_lnnoptr = 0; /* file ptr to line numbers */
|
||||||
|
coff_sec->s_nreloc = 0; /* number of relocation entries */
|
||||||
|
coff_sec->s_flags = GetCoffFlags(coff_sec->s_name); /* flags */
|
||||||
|
coff_sec->s_reserved = 0; /* reserved byte */
|
||||||
|
coff_sec->s_page = 0; /* memory page id */
|
||||||
|
|
||||||
|
file_pointer += sizeof(SCNHDR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
file_hdr.f_nscns = NSectionsToOutput; /* number of sections */
|
||||||
|
|
||||||
|
// now loop through and determine file pointer locations
|
||||||
|
// for the raw data
|
||||||
|
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
// put raw data
|
||||||
|
coff_sec->s_scnptr = file_pointer; /* file ptr to raw data for section */
|
||||||
|
file_pointer += coff_sec->s_size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// now loop through and determine file pointer locations
|
||||||
|
// for the relocation data
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
// put relocations data
|
||||||
|
if (coff_sec->s_nreloc > 0) {
|
||||||
|
coff_sec->s_relptr = file_pointer; /* file ptr to relocation */
|
||||||
|
file_pointer += coff_sec->s_nreloc * sizeof(struct reloc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// now loop through and determine file pointer locations
|
||||||
|
// for the line number data
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
coff_sec->s_nlnno = 0;
|
||||||
|
coff_sec->s_lnnoptr = 0;
|
||||||
|
|
||||||
|
if (s1->do_debug && tcc_sect == stext) {
|
||||||
|
// count how many line nos data
|
||||||
|
|
||||||
|
// also find association between source file name and function
|
||||||
|
// so we can sort the symbol table
|
||||||
|
|
||||||
|
|
||||||
|
Stab_Sym *sym, *sym_end;
|
||||||
|
char func_name[MAX_FUNC_NAME_LENGTH],
|
||||||
|
last_func_name[MAX_FUNC_NAME_LENGTH];
|
||||||
|
unsigned long func_addr, last_pc, pc;
|
||||||
|
const char *incl_files[INCLUDE_STACK_SIZE];
|
||||||
|
int incl_index, len, last_line_num;
|
||||||
|
const char *str, *p;
|
||||||
|
|
||||||
|
coff_sec->s_lnnoptr = file_pointer; /* file ptr to linno */
|
||||||
|
|
||||||
|
|
||||||
|
func_name[0] = '\0';
|
||||||
|
func_addr = 0;
|
||||||
|
incl_index = 0;
|
||||||
|
last_func_name[0] = '\0';
|
||||||
|
last_pc = 0xffffffff;
|
||||||
|
last_line_num = 1;
|
||||||
|
sym = (Stab_Sym *) stab_section->data + 1;
|
||||||
|
sym_end =
|
||||||
|
(Stab_Sym *) (stab_section->data +
|
||||||
|
stab_section->data_offset);
|
||||||
|
|
||||||
|
nFuncs = 0;
|
||||||
|
while (sym < sym_end) {
|
||||||
|
switch (sym->n_type) {
|
||||||
|
/* function start or end */
|
||||||
|
case N_FUN:
|
||||||
|
if (sym->n_strx == 0) {
|
||||||
|
// end of function
|
||||||
|
|
||||||
|
coff_sec->s_nlnno++;
|
||||||
|
file_pointer += LINESZ;
|
||||||
|
|
||||||
|
pc = sym->n_value + func_addr;
|
||||||
|
func_name[0] = '\0';
|
||||||
|
func_addr = 0;
|
||||||
|
EndAddress[nFuncs] = pc;
|
||||||
|
FuncEntries[nFuncs] =
|
||||||
|
(file_pointer -
|
||||||
|
LineNoFilePtr[nFuncs]) / LINESZ - 1;
|
||||||
|
LastLineNo[nFuncs++] = last_line_num + 1;
|
||||||
|
} else {
|
||||||
|
// beginning of function
|
||||||
|
|
||||||
|
LineNoFilePtr[nFuncs] = file_pointer;
|
||||||
|
coff_sec->s_nlnno++;
|
||||||
|
file_pointer += LINESZ;
|
||||||
|
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data +
|
||||||
|
sym->n_strx;
|
||||||
|
|
||||||
|
p = strchr(str, ':');
|
||||||
|
if (!p) {
|
||||||
|
pstrcpy(func_name, sizeof(func_name), str);
|
||||||
|
pstrcpy(Func[nFuncs], sizeof(func_name), str);
|
||||||
|
} else {
|
||||||
|
len = p - str;
|
||||||
|
if (len > sizeof(func_name) - 1)
|
||||||
|
len = sizeof(func_name) - 1;
|
||||||
|
memcpy(func_name, str, len);
|
||||||
|
memcpy(Func[nFuncs], str, len);
|
||||||
|
func_name[len] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
// save the file that it came in so we can sort later
|
||||||
|
pstrcpy(AssociatedFile[nFuncs], sizeof(func_name),
|
||||||
|
incl_files[incl_index - 1]);
|
||||||
|
|
||||||
|
func_addr = sym->n_value;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* line number info */
|
||||||
|
case N_SLINE:
|
||||||
|
pc = sym->n_value + func_addr;
|
||||||
|
|
||||||
|
last_pc = pc;
|
||||||
|
last_line_num = sym->n_desc;
|
||||||
|
|
||||||
|
/* XXX: slow! */
|
||||||
|
strcpy(last_func_name, func_name);
|
||||||
|
|
||||||
|
coff_sec->s_nlnno++;
|
||||||
|
file_pointer += LINESZ;
|
||||||
|
break;
|
||||||
|
/* include files */
|
||||||
|
case N_BINCL:
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data + sym->n_strx;
|
||||||
|
add_incl:
|
||||||
|
if (incl_index < INCLUDE_STACK_SIZE) {
|
||||||
|
incl_files[incl_index++] = str;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case N_EINCL:
|
||||||
|
if (incl_index > 1)
|
||||||
|
incl_index--;
|
||||||
|
break;
|
||||||
|
case N_SO:
|
||||||
|
if (sym->n_strx == 0) {
|
||||||
|
incl_index = 0; /* end of translation unit */
|
||||||
|
} else {
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data +
|
||||||
|
sym->n_strx;
|
||||||
|
/* do not add path */
|
||||||
|
len = strlen(str);
|
||||||
|
if (len > 0 && str[len - 1] != '/')
|
||||||
|
goto add_incl;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sym++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
file_hdr.f_symptr = file_pointer; /* file pointer to symtab */
|
||||||
|
|
||||||
|
if (s1->do_debug)
|
||||||
|
file_hdr.f_nsyms = coff_nb_syms; /* number of symtab entries */
|
||||||
|
else
|
||||||
|
file_hdr.f_nsyms = 0;
|
||||||
|
|
||||||
|
file_pointer += file_hdr.f_nsyms * SYMNMLEN;
|
||||||
|
|
||||||
|
// OK now we are all set to write the file
|
||||||
|
|
||||||
|
|
||||||
|
fwrite(&file_hdr, FILHSZ, 1, f);
|
||||||
|
fwrite(&o_filehdr, sizeof(o_filehdr), 1, f);
|
||||||
|
|
||||||
|
// write section headers
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
fwrite(coff_sec, sizeof(SCNHDR), 1, f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// write raw data
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
fwrite(tcc_sect->data, tcc_sect->data_offset, 1, f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// write relocation data
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (OutputTheSection(tcc_sect)) {
|
||||||
|
// put relocations data
|
||||||
|
if (coff_sec->s_nreloc > 0) {
|
||||||
|
fwrite(tcc_sect->reloc,
|
||||||
|
coff_sec->s_nreloc * sizeof(struct reloc), 1, f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// group the symbols in order of filename, func1, func2, etc
|
||||||
|
// finally global symbols
|
||||||
|
|
||||||
|
if (s1->do_debug)
|
||||||
|
SortSymbolTable();
|
||||||
|
|
||||||
|
// write line no data
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
coff_sec = §ion_header[i];
|
||||||
|
tcc_sect = s1->sections[i];
|
||||||
|
|
||||||
|
if (s1->do_debug && tcc_sect == stext) {
|
||||||
|
// count how many line nos data
|
||||||
|
|
||||||
|
|
||||||
|
Stab_Sym *sym, *sym_end;
|
||||||
|
char func_name[128], last_func_name[128];
|
||||||
|
unsigned long func_addr, last_pc, pc;
|
||||||
|
const char *incl_files[INCLUDE_STACK_SIZE];
|
||||||
|
int incl_index, len, last_line_num;
|
||||||
|
const char *str, *p;
|
||||||
|
|
||||||
|
LINENO CoffLineNo;
|
||||||
|
|
||||||
|
func_name[0] = '\0';
|
||||||
|
func_addr = 0;
|
||||||
|
incl_index = 0;
|
||||||
|
last_func_name[0] = '\0';
|
||||||
|
last_pc = 0;
|
||||||
|
last_line_num = 1;
|
||||||
|
sym = (Stab_Sym *) stab_section->data + 1;
|
||||||
|
sym_end =
|
||||||
|
(Stab_Sym *) (stab_section->data +
|
||||||
|
stab_section->data_offset);
|
||||||
|
|
||||||
|
while (sym < sym_end) {
|
||||||
|
switch (sym->n_type) {
|
||||||
|
/* function start or end */
|
||||||
|
case N_FUN:
|
||||||
|
if (sym->n_strx == 0) {
|
||||||
|
// end of function
|
||||||
|
|
||||||
|
CoffLineNo.l_addr.l_paddr = last_pc;
|
||||||
|
CoffLineNo.l_lnno = last_line_num + 1;
|
||||||
|
fwrite(&CoffLineNo, 6, 1, f);
|
||||||
|
|
||||||
|
pc = sym->n_value + func_addr;
|
||||||
|
func_name[0] = '\0';
|
||||||
|
func_addr = 0;
|
||||||
|
} else {
|
||||||
|
// beginning of function
|
||||||
|
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data +
|
||||||
|
sym->n_strx;
|
||||||
|
|
||||||
|
|
||||||
|
p = strchr(str, ':');
|
||||||
|
if (!p) {
|
||||||
|
pstrcpy(func_name, sizeof(func_name), str);
|
||||||
|
} else {
|
||||||
|
len = p - str;
|
||||||
|
if (len > sizeof(func_name) - 1)
|
||||||
|
len = sizeof(func_name) - 1;
|
||||||
|
memcpy(func_name, str, len);
|
||||||
|
func_name[len] = '\0';
|
||||||
|
}
|
||||||
|
func_addr = sym->n_value;
|
||||||
|
last_pc = func_addr;
|
||||||
|
last_line_num = -1;
|
||||||
|
|
||||||
|
// output a function begin
|
||||||
|
|
||||||
|
CoffLineNo.l_addr.l_symndx =
|
||||||
|
FindCoffSymbolIndex(func_name);
|
||||||
|
CoffLineNo.l_lnno = 0;
|
||||||
|
|
||||||
|
fwrite(&CoffLineNo, 6, 1, f);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* line number info */
|
||||||
|
case N_SLINE:
|
||||||
|
pc = sym->n_value + func_addr;
|
||||||
|
|
||||||
|
|
||||||
|
/* XXX: slow! */
|
||||||
|
strcpy(last_func_name, func_name);
|
||||||
|
|
||||||
|
// output a line reference
|
||||||
|
|
||||||
|
CoffLineNo.l_addr.l_paddr = last_pc;
|
||||||
|
|
||||||
|
if (last_line_num == -1) {
|
||||||
|
CoffLineNo.l_lnno = sym->n_desc;
|
||||||
|
} else {
|
||||||
|
CoffLineNo.l_lnno = last_line_num + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fwrite(&CoffLineNo, 6, 1, f);
|
||||||
|
|
||||||
|
last_pc = pc;
|
||||||
|
last_line_num = sym->n_desc;
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* include files */
|
||||||
|
case N_BINCL:
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data + sym->n_strx;
|
||||||
|
add_incl2:
|
||||||
|
if (incl_index < INCLUDE_STACK_SIZE) {
|
||||||
|
incl_files[incl_index++] = str;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case N_EINCL:
|
||||||
|
if (incl_index > 1)
|
||||||
|
incl_index--;
|
||||||
|
break;
|
||||||
|
case N_SO:
|
||||||
|
if (sym->n_strx == 0) {
|
||||||
|
incl_index = 0; /* end of translation unit */
|
||||||
|
} else {
|
||||||
|
str =
|
||||||
|
(const char *) stabstr_section->data +
|
||||||
|
sym->n_strx;
|
||||||
|
/* do not add path */
|
||||||
|
len = strlen(str);
|
||||||
|
if (len > 0 && str[len - 1] != '/')
|
||||||
|
goto add_incl2;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sym++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// write symbol table
|
||||||
|
if (s1->do_debug) {
|
||||||
|
int k;
|
||||||
|
struct syment csym;
|
||||||
|
AUXFUNC auxfunc;
|
||||||
|
AUXBF auxbf;
|
||||||
|
AUXEF auxef;
|
||||||
|
int i;
|
||||||
|
Elf32_Sym *p;
|
||||||
|
const char *name;
|
||||||
|
int nstr;
|
||||||
|
int n = 0;
|
||||||
|
|
||||||
|
Coff_str_table = (char *) tcc_malloc(MAX_STR_TABLE);
|
||||||
|
pCoff_str_table = Coff_str_table;
|
||||||
|
nstr = 0;
|
||||||
|
|
||||||
|
p = (Elf32_Sym *) symtab_section->data;
|
||||||
|
|
||||||
|
|
||||||
|
for (i = 0; i < nb_syms; i++) {
|
||||||
|
|
||||||
|
name = symtab_section->link->data + p->st_name;
|
||||||
|
|
||||||
|
for (k = 0; k < 8; k++)
|
||||||
|
csym._n._n_name[k] = 0;
|
||||||
|
|
||||||
|
if (strlen(name) <= 8) {
|
||||||
|
strcpy(csym._n._n_name, name);
|
||||||
|
} else {
|
||||||
|
if (pCoff_str_table - Coff_str_table + strlen(name) >
|
||||||
|
MAX_STR_TABLE - 1)
|
||||||
|
error("String table too large");
|
||||||
|
|
||||||
|
csym._n._n_n._n_zeroes = 0;
|
||||||
|
csym._n._n_n._n_offset =
|
||||||
|
pCoff_str_table - Coff_str_table + 4;
|
||||||
|
|
||||||
|
strcpy(pCoff_str_table, name);
|
||||||
|
pCoff_str_table += strlen(name) + 1; // skip over null
|
||||||
|
nstr++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (p->st_info == 4) {
|
||||||
|
// put a filename symbol
|
||||||
|
csym.n_value = 33; // ?????
|
||||||
|
csym.n_scnum = N_DEBUG;
|
||||||
|
csym.n_type = 0;
|
||||||
|
csym.n_sclass = C_FILE;
|
||||||
|
csym.n_numaux = 0;
|
||||||
|
fwrite(&csym, 18, 1, f);
|
||||||
|
n++;
|
||||||
|
|
||||||
|
} else if (p->st_info == 0x12) {
|
||||||
|
// find the function data
|
||||||
|
|
||||||
|
for (k = 0; k < nFuncs; k++) {
|
||||||
|
if (strcmp(name, Func[k]) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (k >= nFuncs) {
|
||||||
|
char s[256];
|
||||||
|
|
||||||
|
sprintf(s, "debug info can't find function: %s", name);
|
||||||
|
|
||||||
|
error(s);
|
||||||
|
}
|
||||||
|
// put a Function Name
|
||||||
|
|
||||||
|
csym.n_value = p->st_value; // physical address
|
||||||
|
csym.n_scnum = CoffTextSectionNo;
|
||||||
|
csym.n_type = MKTYPE(T_INT, DT_FCN, 0, 0, 0, 0, 0);
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
csym.n_numaux = 1;
|
||||||
|
fwrite(&csym, 18, 1, f);
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
auxfunc.tag = 0;
|
||||||
|
auxfunc.size = EndAddress[k] - p->st_value;
|
||||||
|
auxfunc.fileptr = LineNoFilePtr[k];
|
||||||
|
auxfunc.nextsym = n + 6; // tktk
|
||||||
|
auxfunc.dummy = 0;
|
||||||
|
fwrite(&auxfunc, 18, 1, f);
|
||||||
|
|
||||||
|
// put a .bf
|
||||||
|
|
||||||
|
strcpy(csym._n._n_name, ".bf");
|
||||||
|
csym.n_value = p->st_value; // physical address
|
||||||
|
csym.n_scnum = CoffTextSectionNo;
|
||||||
|
csym.n_type = 0;
|
||||||
|
csym.n_sclass = C_FCN;
|
||||||
|
csym.n_numaux = 1;
|
||||||
|
fwrite(&csym, 18, 1, f);
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
auxbf.regmask = 0;
|
||||||
|
auxbf.lineno = 0;
|
||||||
|
auxbf.nentries = FuncEntries[k];
|
||||||
|
auxbf.localframe = 0;
|
||||||
|
auxbf.nextentry = n + 6;
|
||||||
|
auxbf.dummy = 0;
|
||||||
|
fwrite(&auxbf, 18, 1, f);
|
||||||
|
|
||||||
|
// put a .ef
|
||||||
|
|
||||||
|
strcpy(csym._n._n_name, ".ef");
|
||||||
|
csym.n_value = EndAddress[k]; // physical address
|
||||||
|
csym.n_scnum = CoffTextSectionNo;
|
||||||
|
csym.n_type = 0;
|
||||||
|
csym.n_sclass = C_FCN;
|
||||||
|
csym.n_numaux = 1;
|
||||||
|
fwrite(&csym, 18, 1, f);
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
auxef.dummy = 0;
|
||||||
|
auxef.lineno = LastLineNo[k];
|
||||||
|
auxef.dummy1 = 0;
|
||||||
|
auxef.dummy2 = 0;
|
||||||
|
auxef.dummy3 = 0;
|
||||||
|
auxef.dummy4 = 0;
|
||||||
|
fwrite(&auxef, 18, 1, f);
|
||||||
|
|
||||||
|
n += 6;
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// try an put some type info
|
||||||
|
|
||||||
|
if ((p->st_other & VT_BTYPE) == VT_DOUBLE) {
|
||||||
|
csym.n_type = T_DOUBLE; // int
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
} else if ((p->st_other & VT_BTYPE) == VT_FLOAT) {
|
||||||
|
csym.n_type = T_FLOAT;
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
} else if ((p->st_other & VT_BTYPE) == VT_INT) {
|
||||||
|
csym.n_type = T_INT; // int
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
} else if ((p->st_other & VT_BTYPE) == VT_SHORT) {
|
||||||
|
csym.n_type = T_SHORT;
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
} else if ((p->st_other & VT_BTYPE) == VT_BYTE) {
|
||||||
|
csym.n_type = T_CHAR;
|
||||||
|
csym.n_sclass = C_EXT;
|
||||||
|
} else {
|
||||||
|
csym.n_type = T_INT; // just mark as a label
|
||||||
|
csym.n_sclass = C_LABEL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
csym.n_value = p->st_value;
|
||||||
|
csym.n_scnum = 2;
|
||||||
|
csym.n_numaux = 1;
|
||||||
|
fwrite(&csym, 18, 1, f);
|
||||||
|
|
||||||
|
auxfunc.tag = 0;
|
||||||
|
auxfunc.size = 0x20;
|
||||||
|
auxfunc.fileptr = 0;
|
||||||
|
auxfunc.nextsym = 0;
|
||||||
|
auxfunc.dummy = 0;
|
||||||
|
fwrite(&auxfunc, 18, 1, f);
|
||||||
|
n++;
|
||||||
|
n++;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s1->do_debug) {
|
||||||
|
// write string table
|
||||||
|
|
||||||
|
// first write the size
|
||||||
|
i = pCoff_str_table - Coff_str_table;
|
||||||
|
fwrite(&i, 4, 1, f);
|
||||||
|
|
||||||
|
// then write the strings
|
||||||
|
fwrite(Coff_str_table, i, 1, f);
|
||||||
|
|
||||||
|
tcc_free(Coff_str_table);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// group the symbols in order of filename, func1, func2, etc
|
||||||
|
// finally global symbols
|
||||||
|
|
||||||
|
void SortSymbolTable(void)
|
||||||
|
{
|
||||||
|
int i, j, k, n = 0;
|
||||||
|
Elf32_Sym *p, *p2, *NewTable;
|
||||||
|
char *name, *name2;
|
||||||
|
|
||||||
|
NewTable = (Elf32_Sym *) tcc_malloc(nb_syms * sizeof(Elf32_Sym));
|
||||||
|
|
||||||
|
p = (Elf32_Sym *) symtab_section->data;
|
||||||
|
|
||||||
|
|
||||||
|
// find a file symbol, copy it over
|
||||||
|
// then scan the whole symbol list and copy any function
|
||||||
|
// symbols that match the file association
|
||||||
|
|
||||||
|
for (i = 0; i < nb_syms; i++) {
|
||||||
|
if (p->st_info == 4) {
|
||||||
|
name = (char *) symtab_section->link->data + p->st_name;
|
||||||
|
|
||||||
|
// this is a file symbol, copy it over
|
||||||
|
|
||||||
|
NewTable[n++] = *p;
|
||||||
|
|
||||||
|
p2 = (Elf32_Sym *) symtab_section->data;
|
||||||
|
|
||||||
|
for (j = 0; j < nb_syms; j++) {
|
||||||
|
if (p2->st_info == 0x12) {
|
||||||
|
// this is a func symbol
|
||||||
|
|
||||||
|
name2 =
|
||||||
|
(char *) symtab_section->link->data + p2->st_name;
|
||||||
|
|
||||||
|
// find the function data index
|
||||||
|
|
||||||
|
for (k = 0; k < nFuncs; k++) {
|
||||||
|
if (strcmp(name2, Func[k]) == 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (k >= nFuncs) {
|
||||||
|
char s[256];
|
||||||
|
|
||||||
|
sprintf(s,
|
||||||
|
"debug (sort) info can't find function: %s",
|
||||||
|
name2);
|
||||||
|
|
||||||
|
error(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strcmp(AssociatedFile[k], name) == 0) {
|
||||||
|
// yes they match copy it over
|
||||||
|
|
||||||
|
NewTable[n++] = *p2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p2++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// now all the filename and func symbols should have been copied over
|
||||||
|
// copy all the rest over (all except file and funcs)
|
||||||
|
|
||||||
|
p = (Elf32_Sym *) symtab_section->data;
|
||||||
|
for (i = 0; i < nb_syms; i++) {
|
||||||
|
if (p->st_info != 4 && p->st_info != 0x12) {
|
||||||
|
NewTable[n++] = *p;
|
||||||
|
}
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (n != nb_syms)
|
||||||
|
error("Internal Compiler error, debug info");
|
||||||
|
|
||||||
|
// copy it all back
|
||||||
|
|
||||||
|
p = (Elf32_Sym *) symtab_section->data;
|
||||||
|
for (i = 0; i < nb_syms; i++) {
|
||||||
|
*p++ = NewTable[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
tcc_free(NewTable);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int FindCoffSymbolIndex(const char *func_name)
|
||||||
|
{
|
||||||
|
int i, n = 0;
|
||||||
|
Elf32_Sym *p;
|
||||||
|
char *name;
|
||||||
|
|
||||||
|
p = (Elf32_Sym *) symtab_section->data;
|
||||||
|
|
||||||
|
for (i = 0; i < nb_syms; i++) {
|
||||||
|
|
||||||
|
name = (char *) symtab_section->link->data + p->st_name;
|
||||||
|
|
||||||
|
if (p->st_info == 4) {
|
||||||
|
// put a filename symbol
|
||||||
|
n++;
|
||||||
|
} else if (p->st_info == 0x12) {
|
||||||
|
|
||||||
|
if (strcmp(func_name, name) == 0)
|
||||||
|
return n;
|
||||||
|
|
||||||
|
n += 6;
|
||||||
|
|
||||||
|
// put a Function Name
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
// put a .bf
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
// put a .ef
|
||||||
|
|
||||||
|
// now put aux info
|
||||||
|
|
||||||
|
} else {
|
||||||
|
n += 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
p++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return n; // total number of symbols
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL OutputTheSection(Section * sect)
|
||||||
|
{
|
||||||
|
const char *s = sect->name;
|
||||||
|
|
||||||
|
if (!strcmp(s, ".text"))
|
||||||
|
return true;
|
||||||
|
else if (!strcmp(s, ".data"))
|
||||||
|
return true;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
short int GetCoffFlags(const char *s)
|
||||||
|
{
|
||||||
|
if (!strcmp(s, ".text"))
|
||||||
|
return STYP_TEXT | STYP_DATA | STYP_ALIGN | 0x400;
|
||||||
|
else if (!strcmp(s, ".data"))
|
||||||
|
return STYP_DATA;
|
||||||
|
else if (!strcmp(s, ".bss"))
|
||||||
|
return STYP_BSS;
|
||||||
|
else if (!strcmp(s, ".stack"))
|
||||||
|
return STYP_BSS | STYP_ALIGN | 0x200;
|
||||||
|
else if (!strcmp(s, ".cinit"))
|
||||||
|
return STYP_COPY | STYP_DATA | STYP_ALIGN | 0x200;
|
||||||
|
else
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
Section *FindSection(TCCState * s1, const char *sname)
|
||||||
|
{
|
||||||
|
Section *s;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 1; i < s1->nb_sections; i++) {
|
||||||
|
s = s1->sections[i];
|
||||||
|
|
||||||
|
if (!strcmp(sname, s->name))
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
error("could not find section %s", sname);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int tcc_load_coff(TCCState * s1, int fd)
|
||||||
|
{
|
||||||
|
// tktk TokenSym *ts;
|
||||||
|
|
||||||
|
FILE *f;
|
||||||
|
unsigned int str_size;
|
||||||
|
char *Coff_str_table, *name;
|
||||||
|
int i, k;
|
||||||
|
struct syment csym;
|
||||||
|
char name2[9];
|
||||||
|
FILHDR file_hdr; /* FILE HEADER STRUCTURE */
|
||||||
|
|
||||||
|
f = fdopen(fd, "rb");
|
||||||
|
if (!f) {
|
||||||
|
error("Unable to open .out file for input");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fread(&file_hdr, FILHSZ, 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
if (fread(&o_filehdr, sizeof(o_filehdr), 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
// first read the string table
|
||||||
|
|
||||||
|
if (fseek(f, file_hdr.f_symptr + file_hdr.f_nsyms * SYMESZ, SEEK_SET))
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
if (fread(&str_size, sizeof(int), 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
|
||||||
|
Coff_str_table = (char *) tcc_malloc(str_size);
|
||||||
|
|
||||||
|
if (fread(Coff_str_table, str_size - 4, 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
// read/process all the symbols
|
||||||
|
|
||||||
|
// seek back to symbols
|
||||||
|
|
||||||
|
if (fseek(f, file_hdr.f_symptr, SEEK_SET))
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
for (i = 0; i < file_hdr.f_nsyms; i++) {
|
||||||
|
if (fread(&csym, SYMESZ, 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
|
||||||
|
if (csym._n._n_n._n_zeroes == 0) {
|
||||||
|
name = Coff_str_table + csym._n._n_n._n_offset - 4;
|
||||||
|
} else {
|
||||||
|
name = csym._n._n_name;
|
||||||
|
|
||||||
|
if (name[7] != 0) {
|
||||||
|
for (k = 0; k < 8; k++)
|
||||||
|
name2[k] = name[k];
|
||||||
|
|
||||||
|
name2[8] = 0;
|
||||||
|
|
||||||
|
name = name2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// if (strcmp("_DAC_Buffer",name)==0) // tktk
|
||||||
|
// name[0]=0;
|
||||||
|
|
||||||
|
if (((csym.n_type & 0x30) == 0x20 && csym.n_sclass == 0x2) || ((csym.n_type & 0x30) == 0x30 && csym.n_sclass == 0x2) || (csym.n_type == 0x4 && csym.n_sclass == 0x2) || (csym.n_type == 0x8 && csym.n_sclass == 0x2) || // structures
|
||||||
|
(csym.n_type == 0x18 && csym.n_sclass == 0x2) || // pointer to structure
|
||||||
|
(csym.n_type == 0x7 && csym.n_sclass == 0x2) || // doubles
|
||||||
|
(csym.n_type == 0x6 && csym.n_sclass == 0x2)) // floats
|
||||||
|
{
|
||||||
|
// strip off any leading underscore (except for other main routine)
|
||||||
|
|
||||||
|
if (name[0] == '_' && strcmp(name, "_main") != 0)
|
||||||
|
name++;
|
||||||
|
|
||||||
|
tcc_add_symbol(s1, name, (void*)csym.n_value);
|
||||||
|
}
|
||||||
|
// skip any aux records
|
||||||
|
|
||||||
|
if (csym.n_numaux == 1) {
|
||||||
|
if (fread(&csym, SYMESZ, 1, f) != 1)
|
||||||
|
error("error reading .out file for input");
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
2732
05/tcc-0.9.25/tccelf.c
Normal file
2732
05/tcc-0.9.25/tccelf.c
Normal file
File diff suppressed because it is too large
Load diff
5123
05/tcc-0.9.25/tccgen.c
Normal file
5123
05/tcc-0.9.25/tccgen.c
Normal file
File diff suppressed because it is too large
Load diff
1559
05/tcc-0.9.25/tccpe.c
Normal file
1559
05/tcc-0.9.25/tccpe.c
Normal file
File diff suppressed because it is too large
Load diff
2935
05/tcc-0.9.25/tccpp.c
Normal file
2935
05/tcc-0.9.25/tccpp.c
Normal file
File diff suppressed because it is too large
Load diff
469
05/tcc-0.9.25/tcctok.h
Normal file
469
05/tcc-0.9.25/tcctok.h
Normal file
|
@ -0,0 +1,469 @@
|
||||||
|
/* keywords */
|
||||||
|
DEF(TOK_INT, "int")
|
||||||
|
DEF(TOK_VOID, "void")
|
||||||
|
DEF(TOK_CHAR, "char")
|
||||||
|
DEF(TOK_IF, "if")
|
||||||
|
DEF(TOK_ELSE, "else")
|
||||||
|
DEF(TOK_WHILE, "while")
|
||||||
|
DEF(TOK_BREAK, "break")
|
||||||
|
DEF(TOK_RETURN, "return")
|
||||||
|
DEF(TOK_FOR, "for")
|
||||||
|
DEF(TOK_EXTERN, "extern")
|
||||||
|
DEF(TOK_STATIC, "static")
|
||||||
|
DEF(TOK_UNSIGNED, "unsigned")
|
||||||
|
DEF(TOK_GOTO, "goto")
|
||||||
|
DEF(TOK_DO, "do")
|
||||||
|
DEF(TOK_CONTINUE, "continue")
|
||||||
|
DEF(TOK_SWITCH, "switch")
|
||||||
|
DEF(TOK_CASE, "case")
|
||||||
|
|
||||||
|
DEF(TOK_CONST1, "const")
|
||||||
|
DEF(TOK_CONST2, "__const") /* gcc keyword */
|
||||||
|
DEF(TOK_CONST3, "__const__") /* gcc keyword */
|
||||||
|
DEF(TOK_VOLATILE1, "volatile")
|
||||||
|
DEF(TOK_VOLATILE2, "__volatile") /* gcc keyword */
|
||||||
|
DEF(TOK_VOLATILE3, "__volatile__") /* gcc keyword */
|
||||||
|
DEF(TOK_LONG, "long")
|
||||||
|
DEF(TOK_REGISTER, "register")
|
||||||
|
DEF(TOK_SIGNED1, "signed")
|
||||||
|
DEF(TOK_SIGNED2, "__signed") /* gcc keyword */
|
||||||
|
DEF(TOK_SIGNED3, "__signed__") /* gcc keyword */
|
||||||
|
DEF(TOK_AUTO, "auto")
|
||||||
|
DEF(TOK_INLINE1, "inline")
|
||||||
|
DEF(TOK_INLINE2, "__inline") /* gcc keyword */
|
||||||
|
DEF(TOK_INLINE3, "__inline__") /* gcc keyword */
|
||||||
|
DEF(TOK_RESTRICT1, "restrict")
|
||||||
|
DEF(TOK_RESTRICT2, "__restrict")
|
||||||
|
DEF(TOK_RESTRICT3, "__restrict__")
|
||||||
|
DEF(TOK_EXTENSION, "__extension__") /* gcc keyword */
|
||||||
|
|
||||||
|
DEF(TOK_FLOAT, "float")
|
||||||
|
DEF(TOK_DOUBLE, "double")
|
||||||
|
DEF(TOK_BOOL, "_Bool")
|
||||||
|
DEF(TOK_SHORT, "short")
|
||||||
|
DEF(TOK_STRUCT, "struct")
|
||||||
|
DEF(TOK_UNION, "union")
|
||||||
|
DEF(TOK_TYPEDEF, "typedef")
|
||||||
|
DEF(TOK_DEFAULT, "default")
|
||||||
|
DEF(TOK_ENUM, "enum")
|
||||||
|
DEF(TOK_SIZEOF, "sizeof")
|
||||||
|
DEF(TOK_ATTRIBUTE1, "__attribute")
|
||||||
|
DEF(TOK_ATTRIBUTE2, "__attribute__")
|
||||||
|
DEF(TOK_ALIGNOF1, "__alignof")
|
||||||
|
DEF(TOK_ALIGNOF2, "__alignof__")
|
||||||
|
DEF(TOK_TYPEOF1, "typeof")
|
||||||
|
DEF(TOK_TYPEOF2, "__typeof")
|
||||||
|
DEF(TOK_TYPEOF3, "__typeof__")
|
||||||
|
DEF(TOK_LABEL, "__label__")
|
||||||
|
DEF(TOK_ASM1, "asm")
|
||||||
|
DEF(TOK_ASM2, "__asm")
|
||||||
|
DEF(TOK_ASM3, "__asm__")
|
||||||
|
|
||||||
|
/*********************************************************************/
|
||||||
|
/* the following are not keywords. They are included to ease parsing */
|
||||||
|
/* preprocessor only */
|
||||||
|
DEF(TOK_DEFINE, "define")
|
||||||
|
DEF(TOK_INCLUDE, "include")
|
||||||
|
DEF(TOK_INCLUDE_NEXT, "include_next")
|
||||||
|
DEF(TOK_IFDEF, "ifdef")
|
||||||
|
DEF(TOK_IFNDEF, "ifndef")
|
||||||
|
DEF(TOK_ELIF, "elif")
|
||||||
|
DEF(TOK_ENDIF, "endif")
|
||||||
|
DEF(TOK_DEFINED, "defined")
|
||||||
|
DEF(TOK_UNDEF, "undef")
|
||||||
|
DEF(TOK_ERROR, "error")
|
||||||
|
DEF(TOK_WARNING, "warning")
|
||||||
|
DEF(TOK_LINE, "line")
|
||||||
|
DEF(TOK_PRAGMA, "pragma")
|
||||||
|
DEF(TOK___LINE__, "__LINE__")
|
||||||
|
DEF(TOK___FILE__, "__FILE__")
|
||||||
|
DEF(TOK___DATE__, "__DATE__")
|
||||||
|
DEF(TOK___TIME__, "__TIME__")
|
||||||
|
DEF(TOK___FUNCTION__, "__FUNCTION__")
|
||||||
|
DEF(TOK___VA_ARGS__, "__VA_ARGS__")
|
||||||
|
|
||||||
|
/* special identifiers */
|
||||||
|
DEF(TOK___FUNC__, "__func__")
|
||||||
|
|
||||||
|
/* attribute identifiers */
|
||||||
|
/* XXX: handle all tokens generically since speed is not critical */
|
||||||
|
DEF(TOK_SECTION1, "section")
|
||||||
|
DEF(TOK_SECTION2, "__section__")
|
||||||
|
DEF(TOK_ALIGNED1, "aligned")
|
||||||
|
DEF(TOK_ALIGNED2, "__aligned__")
|
||||||
|
DEF(TOK_PACKED1, "packed")
|
||||||
|
DEF(TOK_PACKED2, "__packed__")
|
||||||
|
DEF(TOK_UNUSED1, "unused")
|
||||||
|
DEF(TOK_UNUSED2, "__unused__")
|
||||||
|
DEF(TOK_CDECL1, "cdecl")
|
||||||
|
DEF(TOK_CDECL2, "__cdecl")
|
||||||
|
DEF(TOK_CDECL3, "__cdecl__")
|
||||||
|
DEF(TOK_STDCALL1, "stdcall")
|
||||||
|
DEF(TOK_STDCALL2, "__stdcall")
|
||||||
|
DEF(TOK_STDCALL3, "__stdcall__")
|
||||||
|
DEF(TOK_FASTCALL1, "fastcall")
|
||||||
|
DEF(TOK_FASTCALL2, "__fastcall")
|
||||||
|
DEF(TOK_FASTCALL3, "__fastcall__")
|
||||||
|
DEF(TOK_DLLEXPORT, "dllexport")
|
||||||
|
DEF(TOK_NORETURN1, "noreturn")
|
||||||
|
DEF(TOK_NORETURN2, "__noreturn__")
|
||||||
|
DEF(TOK_builtin_types_compatible_p, "__builtin_types_compatible_p")
|
||||||
|
DEF(TOK_builtin_constant_p, "__builtin_constant_p")
|
||||||
|
DEF(TOK_builtin_frame_address, "__builtin_frame_address")
|
||||||
|
#ifdef TCC_TARGET_X86_64
|
||||||
|
DEF(TOK_builtin_malloc, "__builtin_malloc")
|
||||||
|
DEF(TOK_builtin_free, "__builtin_free")
|
||||||
|
DEF(TOK_malloc, "malloc")
|
||||||
|
DEF(TOK_free, "free")
|
||||||
|
#endif
|
||||||
|
DEF(TOK_REGPARM1, "regparm")
|
||||||
|
DEF(TOK_REGPARM2, "__regparm__")
|
||||||
|
|
||||||
|
/* pragma */
|
||||||
|
DEF(TOK_pack, "pack")
|
||||||
|
#if !defined(TCC_TARGET_I386)
|
||||||
|
/* already defined for assembler */
|
||||||
|
DEF(TOK_ASM_push, "push")
|
||||||
|
DEF(TOK_ASM_pop, "pop")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* builtin functions or variables */
|
||||||
|
#ifdef TCC_ARM_EABI
|
||||||
|
DEF(TOK_memcpy, "__aeabi_memcpy")
|
||||||
|
DEF(TOK_memcpy4, "__aeabi_memcpy4")
|
||||||
|
DEF(TOK_memcpy8, "__aeabi_memcpy8")
|
||||||
|
DEF(TOK_memset, "__aeabi_memset")
|
||||||
|
DEF(TOK___aeabi_ldivmod, "__aeabi_ldivmod")
|
||||||
|
DEF(TOK___aeabi_uldivmod, "__aeabi_uldivmod")
|
||||||
|
#else
|
||||||
|
DEF(TOK_memcpy, "memcpy")
|
||||||
|
DEF(TOK_memset, "memset")
|
||||||
|
DEF(TOK___divdi3, "__divdi3")
|
||||||
|
DEF(TOK___moddi3, "__moddi3")
|
||||||
|
DEF(TOK___udivdi3, "__udivdi3")
|
||||||
|
DEF(TOK___umoddi3, "__umoddi3")
|
||||||
|
#endif
|
||||||
|
#if defined(TCC_TARGET_ARM)
|
||||||
|
#ifdef TCC_ARM_EABI
|
||||||
|
DEF(TOK___aeabi_idivmod, "__aeabi_idivmod")
|
||||||
|
DEF(TOK___aeabi_uidivmod, "__aeabi_uidivmod")
|
||||||
|
DEF(TOK___divsi3, "__aeabi_idiv")
|
||||||
|
DEF(TOK___udivsi3, "__aeabi_uidiv")
|
||||||
|
DEF(TOK___floatdisf, "__aeabi_l2f")
|
||||||
|
DEF(TOK___floatdidf, "__aeabi_l2d")
|
||||||
|
DEF(TOK___fixsfdi, "__aeabi_f2lz")
|
||||||
|
DEF(TOK___fixdfdi, "__aeabi_d2lz")
|
||||||
|
#else
|
||||||
|
DEF(TOK___modsi3, "__modsi3")
|
||||||
|
DEF(TOK___umodsi3, "__umodsi3")
|
||||||
|
DEF(TOK___divsi3, "__divsi3")
|
||||||
|
DEF(TOK___udivsi3, "__udivsi3")
|
||||||
|
DEF(TOK___floatdisf, "__floatdisf")
|
||||||
|
DEF(TOK___floatdidf, "__floatdidf")
|
||||||
|
#ifndef TCC_ARM_VFP
|
||||||
|
DEF(TOK___floatdixf, "__floatdixf")
|
||||||
|
DEF(TOK___fixunssfsi, "__fixunssfsi")
|
||||||
|
DEF(TOK___fixunsdfsi, "__fixunsdfsi")
|
||||||
|
DEF(TOK___fixunsxfsi, "__fixunsxfsi")
|
||||||
|
DEF(TOK___fixxfdi, "__fixxfdi")
|
||||||
|
#endif
|
||||||
|
DEF(TOK___fixsfdi, "__fixsfdi")
|
||||||
|
DEF(TOK___fixdfdi, "__fixdfdi")
|
||||||
|
#endif
|
||||||
|
#elif defined(TCC_TARGET_C67)
|
||||||
|
DEF(TOK__divi, "_divi")
|
||||||
|
DEF(TOK__divu, "_divu")
|
||||||
|
DEF(TOK__divf, "_divf")
|
||||||
|
DEF(TOK__divd, "_divd")
|
||||||
|
DEF(TOK__remi, "_remi")
|
||||||
|
DEF(TOK__remu, "_remu")
|
||||||
|
#endif
|
||||||
|
#ifdef TCC_TARGET_I386
|
||||||
|
DEF(TOK___tcc_int_fpu_control, "__tcc_int_fpu_control")
|
||||||
|
DEF(TOK___tcc_fpu_control, "__tcc_fpu_control")
|
||||||
|
#endif
|
||||||
|
#ifdef TCC_ARM_EABI
|
||||||
|
DEF(TOK___ashrdi3, "__aeabi_lasr")
|
||||||
|
DEF(TOK___lshrdi3, "__aeabi_llsr")
|
||||||
|
DEF(TOK___ashldi3, "__aeabi_llsl")
|
||||||
|
DEF(TOK___floatundisf, "__aeabi_ul2f")
|
||||||
|
DEF(TOK___floatundidf, "__aeabi_ul2d")
|
||||||
|
DEF(TOK___fixunssfdi, "__aeabi_f2ulz")
|
||||||
|
DEF(TOK___fixunsdfdi, "__aeabi_d2ulz")
|
||||||
|
#else
|
||||||
|
DEF(TOK___ashrdi3, "__ashrdi3")
|
||||||
|
DEF(TOK___lshrdi3, "__lshrdi3")
|
||||||
|
DEF(TOK___ashldi3, "__ashldi3")
|
||||||
|
DEF(TOK___floatundisf, "__floatundisf")
|
||||||
|
DEF(TOK___floatundidf, "__floatundidf")
|
||||||
|
#ifndef TCC_ARM_VFP
|
||||||
|
DEF(TOK___floatundixf, "__floatundixf")
|
||||||
|
DEF(TOK___fixunsxfdi, "__fixunsxfdi")
|
||||||
|
#endif
|
||||||
|
DEF(TOK___fixunssfdi, "__fixunssfdi")
|
||||||
|
DEF(TOK___fixunsdfdi, "__fixunsdfdi")
|
||||||
|
#endif
|
||||||
|
#ifdef TCC_TARGET_PE
|
||||||
|
DEF(TOK___chkstk, "__chkstk")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* bound checking symbols */
|
||||||
|
#ifdef CONFIG_TCC_BCHECK
|
||||||
|
DEF(TOK___bound_ptr_add, "__bound_ptr_add")
|
||||||
|
DEF(TOK___bound_ptr_indir1, "__bound_ptr_indir1")
|
||||||
|
DEF(TOK___bound_ptr_indir2, "__bound_ptr_indir2")
|
||||||
|
DEF(TOK___bound_ptr_indir4, "__bound_ptr_indir4")
|
||||||
|
DEF(TOK___bound_ptr_indir8, "__bound_ptr_indir8")
|
||||||
|
DEF(TOK___bound_ptr_indir12, "__bound_ptr_indir12")
|
||||||
|
DEF(TOK___bound_ptr_indir16, "__bound_ptr_indir16")
|
||||||
|
DEF(TOK___bound_local_new, "__bound_local_new")
|
||||||
|
DEF(TOK___bound_local_delete, "__bound_local_delete")
|
||||||
|
#if 0
|
||||||
|
DEF(TOK_malloc, "malloc")
|
||||||
|
DEF(TOK_free, "free")
|
||||||
|
DEF(TOK_realloc, "realloc")
|
||||||
|
DEF(TOK_memalign, "memalign")
|
||||||
|
DEF(TOK_calloc, "calloc")
|
||||||
|
#endif
|
||||||
|
DEF(TOK_memmove, "memmove")
|
||||||
|
DEF(TOK_strlen, "strlen")
|
||||||
|
DEF(TOK_strcpy, "strcpy")
|
||||||
|
DEF(TOK_alloca, "alloca")
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Tiny Assembler */
|
||||||
|
|
||||||
|
DEF_ASM(byte)
|
||||||
|
DEF_ASM(align)
|
||||||
|
DEF_ASM(skip)
|
||||||
|
DEF_ASM(space)
|
||||||
|
DEF_ASM(string)
|
||||||
|
DEF_ASM(asciz)
|
||||||
|
DEF_ASM(ascii)
|
||||||
|
DEF_ASM(globl)
|
||||||
|
DEF_ASM(global)
|
||||||
|
DEF_ASM(text)
|
||||||
|
DEF_ASM(data)
|
||||||
|
DEF_ASM(bss)
|
||||||
|
DEF_ASM(previous)
|
||||||
|
DEF_ASM(fill)
|
||||||
|
DEF_ASM(org)
|
||||||
|
DEF_ASM(quad)
|
||||||
|
|
||||||
|
#ifdef TCC_TARGET_I386
|
||||||
|
|
||||||
|
/* WARNING: relative order of tokens is important. */
|
||||||
|
DEF_ASM(al)
|
||||||
|
DEF_ASM(cl)
|
||||||
|
DEF_ASM(dl)
|
||||||
|
DEF_ASM(bl)
|
||||||
|
DEF_ASM(ah)
|
||||||
|
DEF_ASM(ch)
|
||||||
|
DEF_ASM(dh)
|
||||||
|
DEF_ASM(bh)
|
||||||
|
DEF_ASM(ax)
|
||||||
|
DEF_ASM(cx)
|
||||||
|
DEF_ASM(dx)
|
||||||
|
DEF_ASM(bx)
|
||||||
|
DEF_ASM(sp)
|
||||||
|
DEF_ASM(bp)
|
||||||
|
DEF_ASM(si)
|
||||||
|
DEF_ASM(di)
|
||||||
|
DEF_ASM(eax)
|
||||||
|
DEF_ASM(ecx)
|
||||||
|
DEF_ASM(edx)
|
||||||
|
DEF_ASM(ebx)
|
||||||
|
DEF_ASM(esp)
|
||||||
|
DEF_ASM(ebp)
|
||||||
|
DEF_ASM(esi)
|
||||||
|
DEF_ASM(edi)
|
||||||
|
DEF_ASM(mm0)
|
||||||
|
DEF_ASM(mm1)
|
||||||
|
DEF_ASM(mm2)
|
||||||
|
DEF_ASM(mm3)
|
||||||
|
DEF_ASM(mm4)
|
||||||
|
DEF_ASM(mm5)
|
||||||
|
DEF_ASM(mm6)
|
||||||
|
DEF_ASM(mm7)
|
||||||
|
DEF_ASM(xmm0)
|
||||||
|
DEF_ASM(xmm1)
|
||||||
|
DEF_ASM(xmm2)
|
||||||
|
DEF_ASM(xmm3)
|
||||||
|
DEF_ASM(xmm4)
|
||||||
|
DEF_ASM(xmm5)
|
||||||
|
DEF_ASM(xmm6)
|
||||||
|
DEF_ASM(xmm7)
|
||||||
|
DEF_ASM(cr0)
|
||||||
|
DEF_ASM(cr1)
|
||||||
|
DEF_ASM(cr2)
|
||||||
|
DEF_ASM(cr3)
|
||||||
|
DEF_ASM(cr4)
|
||||||
|
DEF_ASM(cr5)
|
||||||
|
DEF_ASM(cr6)
|
||||||
|
DEF_ASM(cr7)
|
||||||
|
DEF_ASM(tr0)
|
||||||
|
DEF_ASM(tr1)
|
||||||
|
DEF_ASM(tr2)
|
||||||
|
DEF_ASM(tr3)
|
||||||
|
DEF_ASM(tr4)
|
||||||
|
DEF_ASM(tr5)
|
||||||
|
DEF_ASM(tr6)
|
||||||
|
DEF_ASM(tr7)
|
||||||
|
DEF_ASM(db0)
|
||||||
|
DEF_ASM(db1)
|
||||||
|
DEF_ASM(db2)
|
||||||
|
DEF_ASM(db3)
|
||||||
|
DEF_ASM(db4)
|
||||||
|
DEF_ASM(db5)
|
||||||
|
DEF_ASM(db6)
|
||||||
|
DEF_ASM(db7)
|
||||||
|
DEF_ASM(dr0)
|
||||||
|
DEF_ASM(dr1)
|
||||||
|
DEF_ASM(dr2)
|
||||||
|
DEF_ASM(dr3)
|
||||||
|
DEF_ASM(dr4)
|
||||||
|
DEF_ASM(dr5)
|
||||||
|
DEF_ASM(dr6)
|
||||||
|
DEF_ASM(dr7)
|
||||||
|
DEF_ASM(es)
|
||||||
|
DEF_ASM(cs)
|
||||||
|
DEF_ASM(ss)
|
||||||
|
DEF_ASM(ds)
|
||||||
|
DEF_ASM(fs)
|
||||||
|
DEF_ASM(gs)
|
||||||
|
DEF_ASM(st)
|
||||||
|
|
||||||
|
DEF_BWL(mov)
|
||||||
|
|
||||||
|
/* generic two operands */
|
||||||
|
DEF_BWL(add)
|
||||||
|
DEF_BWL(or)
|
||||||
|
DEF_BWL(adc)
|
||||||
|
DEF_BWL(sbb)
|
||||||
|
DEF_BWL(and)
|
||||||
|
DEF_BWL(sub)
|
||||||
|
DEF_BWL(xor)
|
||||||
|
DEF_BWL(cmp)
|
||||||
|
|
||||||
|
/* unary ops */
|
||||||
|
DEF_BWL(inc)
|
||||||
|
DEF_BWL(dec)
|
||||||
|
DEF_BWL(not)
|
||||||
|
DEF_BWL(neg)
|
||||||
|
DEF_BWL(mul)
|
||||||
|
DEF_BWL(imul)
|
||||||
|
DEF_BWL(div)
|
||||||
|
DEF_BWL(idiv)
|
||||||
|
|
||||||
|
DEF_BWL(xchg)
|
||||||
|
DEF_BWL(test)
|
||||||
|
|
||||||
|
/* shifts */
|
||||||
|
DEF_BWL(rol)
|
||||||
|
DEF_BWL(ror)
|
||||||
|
DEF_BWL(rcl)
|
||||||
|
DEF_BWL(rcr)
|
||||||
|
DEF_BWL(shl)
|
||||||
|
DEF_BWL(shr)
|
||||||
|
DEF_BWL(sar)
|
||||||
|
|
||||||
|
DEF_ASM(shldw)
|
||||||
|
DEF_ASM(shldl)
|
||||||
|
DEF_ASM(shld)
|
||||||
|
DEF_ASM(shrdw)
|
||||||
|
DEF_ASM(shrdl)
|
||||||
|
DEF_ASM(shrd)
|
||||||
|
|
||||||
|
DEF_ASM(pushw)
|
||||||
|
DEF_ASM(pushl)
|
||||||
|
DEF_ASM(push)
|
||||||
|
DEF_ASM(popw)
|
||||||
|
DEF_ASM(popl)
|
||||||
|
DEF_ASM(pop)
|
||||||
|
DEF_BWL(in)
|
||||||
|
DEF_BWL(out)
|
||||||
|
|
||||||
|
DEF_WL(movzb)
|
||||||
|
|
||||||
|
DEF_ASM(movzwl)
|
||||||
|
DEF_ASM(movsbw)
|
||||||
|
DEF_ASM(movsbl)
|
||||||
|
DEF_ASM(movswl)
|
||||||
|
|
||||||
|
DEF_WL(lea)
|
||||||
|
|
||||||
|
DEF_ASM(les)
|
||||||
|
DEF_ASM(lds)
|
||||||
|
DEF_ASM(lss)
|
||||||
|
DEF_ASM(lfs)
|
||||||
|
DEF_ASM(lgs)
|
||||||
|
|
||||||
|
DEF_ASM(call)
|
||||||
|
DEF_ASM(jmp)
|
||||||
|
DEF_ASM(lcall)
|
||||||
|
DEF_ASM(ljmp)
|
||||||
|
|
||||||
|
DEF_ASMTEST(j)
|
||||||
|
|
||||||
|
DEF_ASMTEST(set)
|
||||||
|
DEF_ASMTEST(cmov)
|
||||||
|
|
||||||
|
DEF_WL(bsf)
|
||||||
|
DEF_WL(bsr)
|
||||||
|
DEF_WL(bt)
|
||||||
|
DEF_WL(bts)
|
||||||
|
DEF_WL(btr)
|
||||||
|
DEF_WL(btc)
|
||||||
|
|
||||||
|
DEF_WL(lsl)
|
||||||
|
|
||||||
|
/* generic FP ops */
|
||||||
|
DEF_FP(add)
|
||||||
|
DEF_FP(mul)
|
||||||
|
|
||||||
|
DEF_ASM(fcom)
|
||||||
|
DEF_ASM(fcom_1) /* non existant op, just to have a regular table */
|
||||||
|
DEF_FP1(com)
|
||||||
|
|
||||||
|
DEF_FP(comp)
|
||||||
|
DEF_FP(sub)
|
||||||
|
DEF_FP(subr)
|
||||||
|
DEF_FP(div)
|
||||||
|
DEF_FP(divr)
|
||||||
|
|
||||||
|
DEF_BWL(xadd)
|
||||||
|
DEF_BWL(cmpxchg)
|
||||||
|
|
||||||
|
/* string ops */
|
||||||
|
DEF_BWL(cmps)
|
||||||
|
DEF_BWL(scmp)
|
||||||
|
DEF_BWL(ins)
|
||||||
|
DEF_BWL(outs)
|
||||||
|
DEF_BWL(lods)
|
||||||
|
DEF_BWL(slod)
|
||||||
|
DEF_BWL(movs)
|
||||||
|
DEF_BWL(smov)
|
||||||
|
DEF_BWL(scas)
|
||||||
|
DEF_BWL(ssca)
|
||||||
|
DEF_BWL(stos)
|
||||||
|
DEF_BWL(ssto)
|
||||||
|
|
||||||
|
/* generic asm ops */
|
||||||
|
|
||||||
|
#define ALT(x)
|
||||||
|
#define DEF_ASM_OP0(name, opcode) DEF_ASM(name)
|
||||||
|
#define DEF_ASM_OP0L(name, opcode, group, instr_type)
|
||||||
|
#define DEF_ASM_OP1(name, opcode, group, instr_type, op0)
|
||||||
|
#define DEF_ASM_OP2(name, opcode, group, instr_type, op0, op1)
|
||||||
|
#define DEF_ASM_OP3(name, opcode, group, instr_type, op0, op1, op2)
|
||||||
|
#include "i386-asm.h"
|
||||||
|
|
||||||
|
#define ALT(x)
|
||||||
|
#define DEF_ASM_OP0(name, opcode)
|
||||||
|
#define DEF_ASM_OP0L(name, opcode, group, instr_type) DEF_ASM(name)
|
||||||
|
#define DEF_ASM_OP1(name, opcode, group, instr_type, op0) DEF_ASM(name)
|
||||||
|
#define DEF_ASM_OP2(name, opcode, group, instr_type, op0, op1) DEF_ASM(name)
|
||||||
|
#define DEF_ASM_OP3(name, opcode, group, instr_type, op0, op1, op2) DEF_ASM(name)
|
||||||
|
#include "i386-asm.h"
|
||||||
|
|
||||||
|
#endif
|
427
05/tcc-0.9.25/texi2pod.pl
Executable file
427
05/tcc-0.9.25/texi2pod.pl
Executable file
|
@ -0,0 +1,427 @@
|
||||||
|
#! /usr/bin/perl -w
|
||||||
|
|
||||||
|
# Copyright (C) 1999, 2000, 2001 Free Software Foundation, Inc.
|
||||||
|
|
||||||
|
# This file is part of GNU CC.
|
||||||
|
|
||||||
|
# GNU CC is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation; either version 2, or (at your option)
|
||||||
|
# any later version.
|
||||||
|
|
||||||
|
# GNU CC is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with GNU CC; see the file COPYING. If not, write to
|
||||||
|
# the Free Software Foundation, 59 Temple Place - Suite 330,
|
||||||
|
# Boston MA 02111-1307, USA.
|
||||||
|
|
||||||
|
# This does trivial (and I mean _trivial_) conversion of Texinfo
|
||||||
|
# markup to Perl POD format. It's intended to be used to extract
|
||||||
|
# something suitable for a manpage from a Texinfo document.
|
||||||
|
|
||||||
|
$output = 0;
|
||||||
|
$skipping = 0;
|
||||||
|
%sects = ();
|
||||||
|
$section = "";
|
||||||
|
@icstack = ();
|
||||||
|
@endwstack = ();
|
||||||
|
@skstack = ();
|
||||||
|
@instack = ();
|
||||||
|
$shift = "";
|
||||||
|
%defs = ();
|
||||||
|
$fnno = 1;
|
||||||
|
$inf = "";
|
||||||
|
$ibase = "";
|
||||||
|
|
||||||
|
while ($_ = shift) {
|
||||||
|
if (/^-D(.*)$/) {
|
||||||
|
if ($1 ne "") {
|
||||||
|
$flag = $1;
|
||||||
|
} else {
|
||||||
|
$flag = shift;
|
||||||
|
}
|
||||||
|
$value = "";
|
||||||
|
($flag, $value) = ($flag =~ /^([^=]+)(?:=(.+))?/);
|
||||||
|
die "no flag specified for -D\n"
|
||||||
|
unless $flag ne "";
|
||||||
|
die "flags may only contain letters, digits, hyphens, dashes and underscores\n"
|
||||||
|
unless $flag =~ /^[a-zA-Z0-9_-]+$/;
|
||||||
|
$defs{$flag} = $value;
|
||||||
|
} elsif (/^-/) {
|
||||||
|
usage();
|
||||||
|
} else {
|
||||||
|
$in = $_, next unless defined $in;
|
||||||
|
$out = $_, next unless defined $out;
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $in) {
|
||||||
|
$inf = gensym();
|
||||||
|
open($inf, "<$in") or die "opening \"$in\": $!\n";
|
||||||
|
$ibase = $1 if $in =~ m|^(.+)/[^/]+$|;
|
||||||
|
} else {
|
||||||
|
$inf = \*STDIN;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $out) {
|
||||||
|
open(STDOUT, ">$out") or die "opening \"$out\": $!\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
while(defined $inf) {
|
||||||
|
while(<$inf>) {
|
||||||
|
# Certain commands are discarded without further processing.
|
||||||
|
/^\@(?:
|
||||||
|
[a-z]+index # @*index: useful only in complete manual
|
||||||
|
|need # @need: useful only in printed manual
|
||||||
|
|(?:end\s+)?group # @group .. @end group: ditto
|
||||||
|
|page # @page: ditto
|
||||||
|
|node # @node: useful only in .info file
|
||||||
|
|(?:end\s+)?ifnottex # @ifnottex .. @end ifnottex: use contents
|
||||||
|
)\b/x and next;
|
||||||
|
|
||||||
|
chomp;
|
||||||
|
|
||||||
|
# Look for filename and title markers.
|
||||||
|
/^\@setfilename\s+([^.]+)/ and $fn = $1, next;
|
||||||
|
/^\@settitle\s+([^.]+)/ and $tl = postprocess($1), next;
|
||||||
|
|
||||||
|
# Identify a man title but keep only the one we are interested in.
|
||||||
|
/^\@c\s+man\s+title\s+([A-Za-z0-9-]+)\s+(.+)/ and do {
|
||||||
|
if (exists $defs{$1}) {
|
||||||
|
$fn = $1;
|
||||||
|
$tl = postprocess($2);
|
||||||
|
}
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
# Look for blocks surrounded by @c man begin SECTION ... @c man end.
|
||||||
|
# This really oughta be @ifman ... @end ifman and the like, but such
|
||||||
|
# would require rev'ing all other Texinfo translators.
|
||||||
|
/^\@c\s+man\s+begin\s+([A-Z]+)\s+([A-Za-z0-9-]+)/ and do {
|
||||||
|
$output = 1 if exists $defs{$2};
|
||||||
|
$sect = $1;
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
/^\@c\s+man\s+begin\s+([A-Z]+)/ and $sect = $1, $output = 1, next;
|
||||||
|
/^\@c\s+man\s+end/ and do {
|
||||||
|
$sects{$sect} = "" unless exists $sects{$sect};
|
||||||
|
$sects{$sect} .= postprocess($section);
|
||||||
|
$section = "";
|
||||||
|
$output = 0;
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
# handle variables
|
||||||
|
/^\@set\s+([a-zA-Z0-9_-]+)\s*(.*)$/ and do {
|
||||||
|
$defs{$1} = $2;
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
/^\@clear\s+([a-zA-Z0-9_-]+)/ and do {
|
||||||
|
delete $defs{$1};
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
next unless $output;
|
||||||
|
|
||||||
|
# Discard comments. (Can't do it above, because then we'd never see
|
||||||
|
# @c man lines.)
|
||||||
|
/^\@c\b/ and next;
|
||||||
|
|
||||||
|
# End-block handler goes up here because it needs to operate even
|
||||||
|
# if we are skipping.
|
||||||
|
/^\@end\s+([a-z]+)/ and do {
|
||||||
|
# Ignore @end foo, where foo is not an operation which may
|
||||||
|
# cause us to skip, if we are presently skipping.
|
||||||
|
my $ended = $1;
|
||||||
|
next if $skipping && $ended !~ /^(?:ifset|ifclear|ignore|menu|iftex)$/;
|
||||||
|
|
||||||
|
die "\@end $ended without \@$ended at line $.\n" unless defined $endw;
|
||||||
|
die "\@$endw ended by \@end $ended at line $.\n" unless $ended eq $endw;
|
||||||
|
|
||||||
|
$endw = pop @endwstack;
|
||||||
|
|
||||||
|
if ($ended =~ /^(?:ifset|ifclear|ignore|menu|iftex)$/) {
|
||||||
|
$skipping = pop @skstack;
|
||||||
|
next;
|
||||||
|
} elsif ($ended =~ /^(?:example|smallexample|display)$/) {
|
||||||
|
$shift = "";
|
||||||
|
$_ = ""; # need a paragraph break
|
||||||
|
} elsif ($ended =~ /^(?:itemize|enumerate|[fv]?table)$/) {
|
||||||
|
$_ = "\n=back\n";
|
||||||
|
$ic = pop @icstack;
|
||||||
|
} else {
|
||||||
|
die "unknown command \@end $ended at line $.\n";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
# We must handle commands which can cause skipping even while we
|
||||||
|
# are skipping, otherwise we will not process nested conditionals
|
||||||
|
# correctly.
|
||||||
|
/^\@ifset\s+([a-zA-Z0-9_-]+)/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @skstack, $skipping;
|
||||||
|
$endw = "ifset";
|
||||||
|
$skipping = 1 unless exists $defs{$1};
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@ifclear\s+([a-zA-Z0-9_-]+)/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @skstack, $skipping;
|
||||||
|
$endw = "ifclear";
|
||||||
|
$skipping = 1 if exists $defs{$1};
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@(ignore|menu|iftex)\b/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @skstack, $skipping;
|
||||||
|
$endw = $1;
|
||||||
|
$skipping = 1;
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
next if $skipping;
|
||||||
|
|
||||||
|
# Character entities. First the ones that can be replaced by raw text
|
||||||
|
# or discarded outright:
|
||||||
|
s/\@copyright\{\}/(c)/g;
|
||||||
|
s/\@dots\{\}/.../g;
|
||||||
|
s/\@enddots\{\}/..../g;
|
||||||
|
s/\@([.!? ])/$1/g;
|
||||||
|
s/\@[:-]//g;
|
||||||
|
s/\@bullet(?:\{\})?/*/g;
|
||||||
|
s/\@TeX\{\}/TeX/g;
|
||||||
|
s/\@pounds\{\}/\#/g;
|
||||||
|
s/\@minus(?:\{\})?/-/g;
|
||||||
|
s/\\,/,/g;
|
||||||
|
|
||||||
|
# Now the ones that have to be replaced by special escapes
|
||||||
|
# (which will be turned back into text by unmunge())
|
||||||
|
s/&/&/g;
|
||||||
|
s/\@\{/{/g;
|
||||||
|
s/\@\}/}/g;
|
||||||
|
s/\@\@/&at;/g;
|
||||||
|
|
||||||
|
# Inside a verbatim block, handle @var specially.
|
||||||
|
if ($shift ne "") {
|
||||||
|
s/\@var\{([^\}]*)\}/<$1>/g;
|
||||||
|
}
|
||||||
|
|
||||||
|
# POD doesn't interpret E<> inside a verbatim block.
|
||||||
|
if ($shift eq "") {
|
||||||
|
s/</</g;
|
||||||
|
s/>/>/g;
|
||||||
|
} else {
|
||||||
|
s/</</g;
|
||||||
|
s/>/>/g;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Single line command handlers.
|
||||||
|
|
||||||
|
/^\@include\s+(.+)$/ and do {
|
||||||
|
push @instack, $inf;
|
||||||
|
$inf = gensym();
|
||||||
|
|
||||||
|
# Try cwd and $ibase.
|
||||||
|
open($inf, "<" . $1)
|
||||||
|
or open($inf, "<" . $ibase . "/" . $1)
|
||||||
|
or die "cannot open $1 or $ibase/$1: $!\n";
|
||||||
|
next;
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@(?:section|unnumbered|unnumberedsec|center)\s+(.+)$/
|
||||||
|
and $_ = "\n=head2 $1\n";
|
||||||
|
/^\@subsection\s+(.+)$/
|
||||||
|
and $_ = "\n=head3 $1\n";
|
||||||
|
|
||||||
|
# Block command handlers:
|
||||||
|
/^\@itemize\s+(\@[a-z]+|\*|-)/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @icstack, $ic;
|
||||||
|
$ic = $1;
|
||||||
|
$_ = "\n=over 4\n";
|
||||||
|
$endw = "itemize";
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@enumerate(?:\s+([a-zA-Z0-9]+))?/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @icstack, $ic;
|
||||||
|
if (defined $1) {
|
||||||
|
$ic = $1 . ".";
|
||||||
|
} else {
|
||||||
|
$ic = "1.";
|
||||||
|
}
|
||||||
|
$_ = "\n=over 4\n";
|
||||||
|
$endw = "enumerate";
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@([fv]?table)\s+(\@[a-z]+)/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
push @icstack, $ic;
|
||||||
|
$endw = $1;
|
||||||
|
$ic = $2;
|
||||||
|
$ic =~ s/\@(?:samp|strong|key|gcctabopt|option|env)/B/;
|
||||||
|
$ic =~ s/\@(?:code|kbd)/C/;
|
||||||
|
$ic =~ s/\@(?:dfn|var|emph|cite|i)/I/;
|
||||||
|
$ic =~ s/\@(?:file)/F/;
|
||||||
|
$_ = "\n=over 4\n";
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@((?:small)?example|display)/ and do {
|
||||||
|
push @endwstack, $endw;
|
||||||
|
$endw = $1;
|
||||||
|
$shift = "\t";
|
||||||
|
$_ = ""; # need a paragraph break
|
||||||
|
};
|
||||||
|
|
||||||
|
/^\@itemx?\s*(.+)?$/ and do {
|
||||||
|
if (defined $1) {
|
||||||
|
# Entity escapes prevent munging by the <> processing below.
|
||||||
|
$_ = "\n=item $ic\<$1\>\n";
|
||||||
|
} else {
|
||||||
|
$_ = "\n=item $ic\n";
|
||||||
|
$ic =~ y/A-Ya-y/B-Zb-z/;
|
||||||
|
$ic =~ s/(\d+)/$1 + 1/eg;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
$section .= $shift.$_."\n";
|
||||||
|
}
|
||||||
|
# End of current file.
|
||||||
|
close($inf);
|
||||||
|
$inf = pop @instack;
|
||||||
|
}
|
||||||
|
|
||||||
|
die "No filename or title\n" unless defined $fn && defined $tl;
|
||||||
|
|
||||||
|
$sects{NAME} = "$fn \- $tl\n";
|
||||||
|
$sects{FOOTNOTES} .= "=back\n" if exists $sects{FOOTNOTES};
|
||||||
|
|
||||||
|
for $sect (qw(NAME SYNOPSIS DESCRIPTION OPTIONS ENVIRONMENT FILES
|
||||||
|
BUGS NOTES FOOTNOTES SEEALSO AUTHOR COPYRIGHT)) {
|
||||||
|
if(exists $sects{$sect}) {
|
||||||
|
$head = $sect;
|
||||||
|
$head =~ s/SEEALSO/SEE ALSO/;
|
||||||
|
print "=head1 $head\n\n";
|
||||||
|
print scalar unmunge ($sects{$sect});
|
||||||
|
print "\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub usage
|
||||||
|
{
|
||||||
|
die "usage: $0 [-D toggle...] [infile [outfile]]\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
sub postprocess
|
||||||
|
{
|
||||||
|
local $_ = $_[0];
|
||||||
|
|
||||||
|
# @value{foo} is replaced by whatever 'foo' is defined as.
|
||||||
|
while (m/(\@value\{([a-zA-Z0-9_-]+)\})/g) {
|
||||||
|
if (! exists $defs{$2}) {
|
||||||
|
print STDERR "Option $2 not defined\n";
|
||||||
|
s/\Q$1\E//;
|
||||||
|
} else {
|
||||||
|
$value = $defs{$2};
|
||||||
|
s/\Q$1\E/$value/;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# Formatting commands.
|
||||||
|
# Temporary escape for @r.
|
||||||
|
s/\@r\{([^\}]*)\}/R<$1>/g;
|
||||||
|
s/\@(?:dfn|var|emph|cite|i)\{([^\}]*)\}/I<$1>/g;
|
||||||
|
s/\@(?:code|kbd)\{([^\}]*)\}/C<$1>/g;
|
||||||
|
s/\@(?:gccoptlist|samp|strong|key|option|env|command|b)\{([^\}]*)\}/B<$1>/g;
|
||||||
|
s/\@sc\{([^\}]*)\}/\U$1/g;
|
||||||
|
s/\@file\{([^\}]*)\}/F<$1>/g;
|
||||||
|
s/\@w\{([^\}]*)\}/S<$1>/g;
|
||||||
|
s/\@(?:dmn|math)\{([^\}]*)\}/$1/g;
|
||||||
|
|
||||||
|
# Cross references are thrown away, as are @noindent and @refill.
|
||||||
|
# (@noindent is impossible in .pod, and @refill is unnecessary.)
|
||||||
|
# @* is also impossible in .pod; we discard it and any newline that
|
||||||
|
# follows it. Similarly, our macro @gol must be discarded.
|
||||||
|
|
||||||
|
s/\(?\@xref\{(?:[^\}]*)\}(?:[^.<]|(?:<[^<>]*>))*\.\)?//g;
|
||||||
|
s/\s+\(\@pxref\{(?:[^\}]*)\}\)//g;
|
||||||
|
s/;\s+\@pxref\{(?:[^\}]*)\}//g;
|
||||||
|
s/\@noindent\s*//g;
|
||||||
|
s/\@refill//g;
|
||||||
|
s/\@gol//g;
|
||||||
|
s/\@\*\s*\n?//g;
|
||||||
|
|
||||||
|
# @uref can take one, two, or three arguments, with different
|
||||||
|
# semantics each time. @url and @email are just like @uref with
|
||||||
|
# one argument, for our purposes.
|
||||||
|
s/\@(?:uref|url|email)\{([^\},]*)\}/<B<$1>>/g;
|
||||||
|
s/\@uref\{([^\},]*),([^\},]*)\}/$2 (C<$1>)/g;
|
||||||
|
s/\@uref\{([^\},]*),([^\},]*),([^\},]*)\}/$3/g;
|
||||||
|
|
||||||
|
# Turn B<blah I<blah> blah> into B<blah> I<blah> B<blah> to
|
||||||
|
# match Texinfo semantics of @emph inside @samp. Also handle @r
|
||||||
|
# inside bold.
|
||||||
|
s/</</g;
|
||||||
|
s/>/>/g;
|
||||||
|
1 while s/B<((?:[^<>]|I<[^<>]*>)*)R<([^>]*)>/B<$1>${2}B</g;
|
||||||
|
1 while (s/B<([^<>]*)I<([^>]+)>/B<$1>I<$2>B</g);
|
||||||
|
1 while (s/I<([^<>]*)B<([^>]+)>/I<$1>B<$2>I</g);
|
||||||
|
s/[BI]<>//g;
|
||||||
|
s/([BI])<(\s+)([^>]+)>/$2$1<$3>/g;
|
||||||
|
s/([BI])<([^>]+?)(\s+)>/$1<$2>$3/g;
|
||||||
|
|
||||||
|
# Extract footnotes. This has to be done after all other
|
||||||
|
# processing because otherwise the regexp will choke on formatting
|
||||||
|
# inside @footnote.
|
||||||
|
while (/\@footnote/g) {
|
||||||
|
s/\@footnote\{([^\}]+)\}/[$fnno]/;
|
||||||
|
add_footnote($1, $fnno);
|
||||||
|
$fnno++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $_;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub unmunge
|
||||||
|
{
|
||||||
|
# Replace escaped symbols with their equivalents.
|
||||||
|
local $_ = $_[0];
|
||||||
|
|
||||||
|
s/</E<lt>/g;
|
||||||
|
s/>/E<gt>/g;
|
||||||
|
s/{/\{/g;
|
||||||
|
s/}/\}/g;
|
||||||
|
s/&at;/\@/g;
|
||||||
|
s/&/&/g;
|
||||||
|
return $_;
|
||||||
|
}
|
||||||
|
|
||||||
|
sub add_footnote
|
||||||
|
{
|
||||||
|
unless (exists $sects{FOOTNOTES}) {
|
||||||
|
$sects{FOOTNOTES} = "\n=over 4\n\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$sects{FOOTNOTES} .= "=item $fnno.\n\n"; $fnno++;
|
||||||
|
$sects{FOOTNOTES} .= $_[0];
|
||||||
|
$sects{FOOTNOTES} .= "\n\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# stolen from Symbol.pm
|
||||||
|
{
|
||||||
|
my $genseq = 0;
|
||||||
|
sub gensym
|
||||||
|
{
|
||||||
|
my $name = "GEN" . $genseq++;
|
||||||
|
my $ref = \*{$name};
|
||||||
|
delete $::{$name};
|
||||||
|
return $ref;
|
||||||
|
}
|
||||||
|
}
|
293
05/tcc-0.9.25/time.h
Normal file
293
05/tcc-0.9.25/time.h
Normal file
|
@ -0,0 +1,293 @@
|
||||||
|
#ifndef _TIME_H
|
||||||
|
#define _TIME_H
|
||||||
|
|
||||||
|
#include <stdc_common.h>
|
||||||
|
#define CLK_TCK 1000000000 // doesnt matter; clock() will always fail.
|
||||||
|
|
||||||
|
typedef long clock_t;
|
||||||
|
|
||||||
|
clock_t clock(void) {
|
||||||
|
// "If the processor time used is not available or its value cannot be represented, the function returns the value (clock_t)-1." C89 § 4.12.2.1
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
double difftime(time_t time1, time_t time0) {
|
||||||
|
return (double)(time1 - time0);
|
||||||
|
}
|
||||||
|
|
||||||
|
time_t time(time_t *timer) {
|
||||||
|
struct timespec ts = {0};
|
||||||
|
if (clock_gettime(CLOCK_REALTIME, &ts) != 0) return -1;
|
||||||
|
if (timer) *timer = ts.tv_sec;
|
||||||
|
return ts.tv_sec;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// @NONSTANDARD(except in UTC+0): we don't support local time in timezones other than UTC+0.
|
||||||
|
|
||||||
|
struct tm {
|
||||||
|
int tm_sec; /* seconds after the minute --- [0, 60] */
|
||||||
|
int tm_min; /* minutes after the hour --- [0, 59] */
|
||||||
|
int tm_hour; /* hours since midnight --- [0, 23] */
|
||||||
|
int tm_mday; /* day of the month --- [1, 31] */
|
||||||
|
int tm_mon; /* months since January --- [0, 11] */
|
||||||
|
int tm_year; /* years since 1900 */
|
||||||
|
int tm_wday; /* days since Sunday --- [0, 6] */
|
||||||
|
int tm_yday; /* days since January 1 --- [0, 365] */
|
||||||
|
int tm_isdst; /* Daylight Saving Time flag */
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
void _gmtime_r(const time_t *timer, struct tm *tm) {
|
||||||
|
time_t t = *timer;
|
||||||
|
int year = 1970;
|
||||||
|
int days = t / 86400;
|
||||||
|
int leap_year;
|
||||||
|
int month;
|
||||||
|
|
||||||
|
tm->tm_isdst = 0;
|
||||||
|
tm->tm_wday = (4 + days) % 7; // jan 1 1970 was a thursday
|
||||||
|
while (1) {
|
||||||
|
leap_year = year % 4 == 0 && (year % 100 != 0 || year % 400 == 0);
|
||||||
|
int ydays = leap_year ? 366 : 365;
|
||||||
|
if (days < ydays) break;
|
||||||
|
days -= ydays;
|
||||||
|
++year;
|
||||||
|
}
|
||||||
|
tm->tm_year = year - 1900;
|
||||||
|
tm->tm_yday = days;
|
||||||
|
for (month = 0; month < 12; ++month) {
|
||||||
|
int mdays;
|
||||||
|
switch (month) {
|
||||||
|
case 0: case 2: case 4: case 6: case 7: case 9: case 11:
|
||||||
|
mdays = 31;
|
||||||
|
break;
|
||||||
|
case 3: case 5: case 8: case 10:
|
||||||
|
mdays = 30;
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
mdays = 28 + leap_year;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (days < mdays) break;
|
||||||
|
days -= mdays;
|
||||||
|
}
|
||||||
|
tm->tm_mday = days + 1;
|
||||||
|
tm->tm_mon = month;
|
||||||
|
t %= 86400;
|
||||||
|
tm->tm_hour = t / 3600;
|
||||||
|
t %= 3600;
|
||||||
|
tm->tm_min = t / 60;
|
||||||
|
tm->tm_sec = t % 60;
|
||||||
|
}
|
||||||
|
|
||||||
|
time_t mktime(struct tm *tm) {
|
||||||
|
// @NONSTANDARD-ish.
|
||||||
|
// not implementing this -- note that the implementation has to support tm_* values
|
||||||
|
// outside of their proper ranges.
|
||||||
|
return (time_t)-1;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct tm *gmtime(const time_t *timer) {
|
||||||
|
static struct tm result;
|
||||||
|
_gmtime_r(timer, &result);
|
||||||
|
return &result;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct tm *localtime(const time_t *timer) {
|
||||||
|
static struct tm result;
|
||||||
|
_gmtime_r(timer, &result);
|
||||||
|
return &result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char _wday_name[7][4] = {
|
||||||
|
"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
|
||||||
|
};
|
||||||
|
static const char _weekday_name[7][16] = {
|
||||||
|
"Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"
|
||||||
|
};
|
||||||
|
static const char _mon_name[12][4] = {
|
||||||
|
"Jan", "Feb", "Mar", "Apr", "May", "Jun",
|
||||||
|
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
|
||||||
|
};
|
||||||
|
static const char _month_name[12][16] = {
|
||||||
|
"January", "February", "March", "April", "May", "June",
|
||||||
|
"July", "August", "September", "October", "November", "December"
|
||||||
|
};
|
||||||
|
|
||||||
|
char *asctime(const struct tm *timeptr) {
|
||||||
|
// lifted from the (draft of the) C standard
|
||||||
|
static char result[32];
|
||||||
|
sprintf(result, "%.3s %.3s%3d %.2d:%.2d:%.2d %d\n",
|
||||||
|
_wday_name[timeptr->tm_wday],
|
||||||
|
_mon_name[timeptr->tm_mon],
|
||||||
|
timeptr->tm_mday, timeptr->tm_hour,
|
||||||
|
timeptr->tm_min, timeptr->tm_sec,
|
||||||
|
1900 + timeptr->tm_year);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *ctime(const time_t *timer) {
|
||||||
|
return asctime(localtime(timer));
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t strftime(char *s, size_t maxsize, const char *format, const struct tm *tm) {
|
||||||
|
size_t n = 0, l;
|
||||||
|
char *name;
|
||||||
|
|
||||||
|
while (*format) {
|
||||||
|
if (*format == '%') {
|
||||||
|
++format;
|
||||||
|
int c = *format++;
|
||||||
|
switch (c) {
|
||||||
|
case 'a':
|
||||||
|
if (n+4 >= maxsize) return 0;
|
||||||
|
strcpy(s, _wday_name[tm->tm_wday]);
|
||||||
|
s += 3;
|
||||||
|
n += 3;
|
||||||
|
break;
|
||||||
|
case 'A':
|
||||||
|
name = _weekday_name[tm->tm_wday];
|
||||||
|
l = strlen(name);
|
||||||
|
if (n+l+1 >= maxsize) return 0;
|
||||||
|
strcpy(s, name);
|
||||||
|
s += l;
|
||||||
|
n += l;
|
||||||
|
break;
|
||||||
|
case 'b':
|
||||||
|
if (n+4 >= maxsize) return 0;
|
||||||
|
strcpy(s, _mon_name[tm->tm_mon]);
|
||||||
|
s += 3;
|
||||||
|
n += 3;
|
||||||
|
break;
|
||||||
|
case 'B':
|
||||||
|
name = _month_name[tm->tm_mon];
|
||||||
|
l = strlen(name);
|
||||||
|
if (n+l+1 >= maxsize) return 0;
|
||||||
|
strcpy(s, name);
|
||||||
|
s += l;
|
||||||
|
n += l;
|
||||||
|
break;
|
||||||
|
case 'c':
|
||||||
|
if (n+32 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%s %02d %s %d %02d:%02d:%02d %s UTC",
|
||||||
|
_wday_name[tm->tm_wday], tm->tm_mday, _mon_name[tm->tm_mon],
|
||||||
|
1900+tm->tm_year, (tm->tm_hour + 11) % 12 + 1, tm->tm_min, tm->tm_sec,
|
||||||
|
tm->tm_hour >= 12 ? "PM" : "AM");
|
||||||
|
s += 31;
|
||||||
|
n += 31;
|
||||||
|
break;
|
||||||
|
case 'd':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_mday);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'H':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_hour);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'I':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", (tm->tm_hour + 11) % 12 + 1);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'j':
|
||||||
|
if (n+4 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%03d", tm->tm_yday + 1);
|
||||||
|
s += 3;
|
||||||
|
n += 3;
|
||||||
|
break;
|
||||||
|
case 'm':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_mon + 1);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'M':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_min);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'p':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%s", tm->tm_hour >= 12 ? "PM" : "AM");
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_sec);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'w':
|
||||||
|
if (n+2 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%d", tm->tm_wday);
|
||||||
|
s += 1;
|
||||||
|
n += 1;
|
||||||
|
break;
|
||||||
|
case 'x':
|
||||||
|
if (n+16 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%s %02d %s %d",
|
||||||
|
_wday_name[tm->tm_wday], tm->tm_mday, _mon_name[tm->tm_mon],
|
||||||
|
1900+tm->tm_year);
|
||||||
|
s += 15;
|
||||||
|
n += 15;
|
||||||
|
break;
|
||||||
|
case 'X':
|
||||||
|
if (n+16 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d:%02d:%02d %s UTC",
|
||||||
|
(tm->tm_hour + 11) % 12 + 1, tm->tm_min, tm->tm_sec,
|
||||||
|
tm->tm_hour >= 12 ? "PM" : "AM");
|
||||||
|
s += 15;
|
||||||
|
n += 15;
|
||||||
|
break;
|
||||||
|
case 'y':
|
||||||
|
if (n+3 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%02d", tm->tm_year % 100);
|
||||||
|
s += 2;
|
||||||
|
n += 2;
|
||||||
|
break;
|
||||||
|
case 'Y':
|
||||||
|
if (n+5 >= maxsize) return 0;
|
||||||
|
sprintf(s, "%d", tm->tm_year + 1900);
|
||||||
|
s += 4;
|
||||||
|
n += 4;
|
||||||
|
break;
|
||||||
|
case 'Z':
|
||||||
|
if (n+4 >= maxsize) return 0;
|
||||||
|
strcpy(s, "UTC");
|
||||||
|
s += 3;
|
||||||
|
n += 3;
|
||||||
|
break;
|
||||||
|
case '%':
|
||||||
|
if (n >= maxsize) return 0;
|
||||||
|
*s++ = '%';
|
||||||
|
n += 1;
|
||||||
|
break;
|
||||||
|
case 'U': // WEEK NUMBER OF THE YEAR? WHO GIVES A SHIT?
|
||||||
|
case 'W': // WEEK NUMBER OF THE YEAR MONDAY-BASED EDITION. IF YOU DIDNT ALREADY GET ENOUGH WEEK NUMBERS. FUCK YOU
|
||||||
|
default:
|
||||||
|
fprintf(stderr, "Bad strftime format.\n");
|
||||||
|
abort();
|
||||||
|
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (n >= maxsize) return 0;
|
||||||
|
*s++ = *format++;
|
||||||
|
n += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (n >= maxsize) return 0;
|
||||||
|
*s = 0;
|
||||||
|
#undef _Push_str
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // _TIME_H
|
1419
05/tcc-0.9.25/x86_64-gen.c
Normal file
1419
05/tcc-0.9.25/x86_64-gen.c
Normal file
File diff suppressed because it is too large
Load diff
Loading…
Add table
Add a link
Reference in a new issue