diff --git a/tcc.c b/tcc.c index 5a8e832..145b6f4 100644 --- a/tcc.c +++ b/tcc.c @@ -289,10 +289,13 @@ typedef struct TokenString { 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 + /* parser */ static struct BufferedFile *file; static int ch, tok; @@ -319,7 +322,9 @@ static int parse_flags; static Section *text_section, *data_section, *bss_section; /* predefined sections */ static Section *cur_text_section; /* current section where function code is generated */ +#ifdef CONFIG_TCC_ASM static Section *last_text_section; /* to handle .previous asm directive */ +#endif /* bound check related sections */ static Section *bounds_section; /* contains global data bound description */ static Section *lbounds_section; /* contains local data bound description */ @@ -352,6 +357,9 @@ static char *funcname; static Sym *global_stack, *local_stack; static Sym *define_stack; static Sym *global_label_stack, *local_label_stack; +/* symbol allocator */ +#define SYM_POOL_NB (8192 / sizeof(Sym)) +static Sym *sym_free_first; static SValue vstack[VSTACK_SIZE], *vtop; /* some predefined types */ @@ -473,6 +481,9 @@ struct TCCState { /* see ifdef_stack_ptr */ int ifdef_stack[IFDEF_STACK_SIZE]; + + /* see cached_includes */ + int cached_includes_hash[CACHED_INCLUDES_HASH_SIZE]; }; /* The current value can be: */ @@ -1048,6 +1059,41 @@ static void dynarray_add(void ***ptab, int *nb_ptr, void *data) *nb_ptr = nb; } +/* symbol allocator */ +static Sym *__sym_malloc(void) +{ + Sym *sym_pool, *sym, *last_sym; + int i; + + sym_pool = tcc_malloc(SYM_POOL_NB * sizeof(Sym)); + + last_sym = sym_free_first; + sym = sym_pool; + for(i = 0; i < SYM_POOL_NB; i++) { + sym->next = last_sym; + last_sym = sym; + sym++; + } + sym_free_first = last_sym; + return last_sym; +} + +static inline Sym *sym_malloc(void) +{ + Sym *sym; + sym = sym_free_first; + if (!sym) + sym = __sym_malloc(); + sym_free_first = sym->next; + return sym; +} + +static inline void sym_free(Sym *sym) +{ + sym->next = sym_free_first; + sym_free_first = sym; +} + Section *new_section(TCCState *s1, const char *name, int sh_type, int sh_flags) { Section *sec; @@ -1433,7 +1479,7 @@ static void cstr_realloc(CString *cstr, int new_size) } /* add a byte */ -static void cstr_ccat(CString *cstr, int ch) +static inline void cstr_ccat(CString *cstr, int ch) { int size; size = cstr->size + 1; @@ -1480,21 +1526,6 @@ static void cstr_free(CString *cstr) #define cstr_reset(cstr) cstr_free(cstr) -static CString *cstr_dup(CString *cstr1) -{ - CString *cstr; - int size; - - cstr = tcc_malloc(sizeof(CString)); - size = cstr1->size; - cstr->size = size; - cstr->size_allocated = size; - cstr->data_allocated = tcc_malloc(size); - cstr->data = cstr->data_allocated; - memcpy(cstr->data_allocated, cstr1->data_allocated, size); - return cstr; -} - /* XXX: unicode ? */ static void add_char(CString *cstr, int c) { @@ -1618,7 +1649,7 @@ char *get_tok_str(int v, CValue *cv) static Sym *sym_push2(Sym **ps, int v, int t, int c) { Sym *s; - s = tcc_malloc(sizeof(Sym)); + s = sym_malloc(); s->v = v; s->type.t = t; s->c = c; @@ -1726,7 +1757,7 @@ static void sym_pop(Sym **ptop, Sym *b) ps = &ts->sym_identifier; *ps = s->prev_tok; } - tcc_free(s); + sym_free(s); s = ss; } *ptop = b; @@ -2194,11 +2225,13 @@ static inline int tok_ext_size(int t) case TOK_CUINT: case TOK_CCHAR: case TOK_LCHAR: - case TOK_STR: - case TOK_LSTR: case TOK_CFLOAT: case TOK_LINENUM: + return 1; + case TOK_STR: + case TOK_LSTR: case TOK_PPNUM: + error("unsupported token"); return 1; case TOK_CDOUBLE: case TOK_CLLONG: @@ -2223,48 +2256,6 @@ static inline void tok_str_new(TokenString *s) static void tok_str_free(int *str) { - const int *p; - CString *cstr; - int t; - - p = str; - for(;;) { - t = *p; - /* NOTE: we test zero separately so that GCC can generate a - table for the following switch */ - if (t == 0) - break; - switch(t) { - case TOK_CINT: - case TOK_CUINT: - case TOK_CCHAR: - case TOK_LCHAR: - case TOK_CFLOAT: - case TOK_LINENUM: - p += 2; - break; - case TOK_PPNUM: - case TOK_STR: - case TOK_LSTR: - /* XXX: use a macro to be portable on 64 bit ? */ - cstr = (CString *)p[1]; - cstr_free(cstr); - tcc_free(cstr); - p += 2; - break; - case TOK_CDOUBLE: - case TOK_CLLONG: - case TOK_CULLONG: - p += 3; - break; - case TOK_CLDOUBLE: - p += 1 + (LDOUBLE_SIZE / 4); - break; - default: - p++; - break; - } - } tcc_free(str); } @@ -2320,7 +2311,22 @@ static void tok_str_add2(TokenString *s, int t, CValue *cv) case TOK_PPNUM: case TOK_STR: case TOK_LSTR: - str[len++] = (int)cstr_dup(cv->cstr); + { + int nb_words; + CString *cstr; + + nb_words = (sizeof(CString) + cv->cstr->size + 3) >> 2; + while ((len + nb_words) > s->allocated_len) + str = tok_str_realloc(s); + cstr = (CString *)(str + len); + cstr->data = NULL; + cstr->size = cv->cstr->size; + cstr->data_allocated = NULL; + cstr->size_allocated = cstr->size; + memcpy((char *)cstr + sizeof(CString), + cv->cstr->data, cstr->size); + len += nb_words; + } break; case TOK_CDOUBLE: case TOK_CLLONG: @@ -2386,10 +2392,14 @@ static void tok_str_add_tok(TokenString *s) case TOK_LCHAR: \ case TOK_CFLOAT: \ case TOK_LINENUM: \ + cv.tab[0] = *p++; \ + break; \ case TOK_STR: \ case TOK_LSTR: \ case TOK_PPNUM: \ - cv.tab[0] = *p++; \ + cv.cstr = (CString *)p; \ + cv.cstr->data = (char *)p + sizeof(CString);\ + p += (sizeof(CString) + cv.cstr->size + 3) >> 2;\ break; \ case TOK_CDOUBLE: \ case TOK_CLLONG: \ @@ -2450,7 +2460,7 @@ static void free_defines(Sym *b) v = top->v; if (v >= TOK_IDENT && v < tok_ident) table_ident[v - TOK_IDENT]->sym_define = NULL; - tcc_free(top); + sym_free(top); top = top1; } define_stack = b; @@ -2503,7 +2513,7 @@ static void label_pop(Sym **ptop, Sym *slast) } /* remove label */ table_ident[s->v - TOK_IDENT]->sym_label = s->prev_tok; - tcc_free(s); + sym_free(s); } *ptop = slast; } @@ -2619,17 +2629,37 @@ static void parse_define(void) define_push(v, t, str.str, first); } +static inline int hash_cached_include(int type, const char *filename) +{ + const unsigned char *s; + unsigned int h; + + h = TOK_HASH_INIT; + h = TOK_HASH_FUNC(h, type); + s = filename; + while (*s) { + h = TOK_HASH_FUNC(h, *s); + s++; + } + h &= (CACHED_INCLUDES_HASH_SIZE - 1); + return h; +} + /* XXX: use a token or a hash table to accelerate matching ? */ static CachedInclude *search_cached_include(TCCState *s1, int type, const char *filename) { CachedInclude *e; - int i; - - for(i = 0;i < s1->nb_cached_includes; i++) { - e = s1->cached_includes[i]; + int i, h; + h = hash_cached_include(type, filename); + i = s1->cached_includes_hash[h]; + for(;;) { + if (i == 0) + break; + e = s1->cached_includes[i - 1]; if (e->type == type && !strcmp(e->filename, filename)) return e; + i = e->hash_next; } return NULL; } @@ -2638,6 +2668,7 @@ static inline void add_cached_include(TCCState *s1, int type, const char *filename, int ifndef_macro) { CachedInclude *e; + int h; if (search_cached_include(s1, type, filename)) return; @@ -2651,6 +2682,10 @@ static inline void add_cached_include(TCCState *s1, int type, strcpy(e->filename, filename); e->ifndef_macro = ifndef_macro; dynarray_add((void ***)&s1->cached_includes, &s1->nb_cached_includes, e); + /* add in hash table */ + h = hash_cached_include(type, filename); + e->hash_next = s1->cached_includes_hash[h]; + s1->cached_includes_hash[h] = s1->nb_cached_includes; } /* is_bof is true if first non space token at beginning of file */ @@ -3989,7 +4024,7 @@ static int macro_subst_tok(TokenString *tok_str, while (sa) { sa1 = sa->prev; tok_str_free((int *)sa->c); - tcc_free(sa); + sym_free(sa); sa = sa1; } mstr_allocated = 1; @@ -3999,7 +4034,7 @@ static int macro_subst_tok(TokenString *tok_str, /* pop nested defined symbol */ sa1 = *nested_list; *nested_list = sa1->prev; - tcc_free(sa1); + sym_free(sa1); if (mstr_allocated) tok_str_free(mstr); } @@ -9409,10 +9444,12 @@ int tcc_relocate(TCCState *s1) tcc_add_runtime(s1); - build_got_entries(s1); - relocate_common_syms(); + tcc_add_linker_symbols(s1); + + build_got_entries(s1); + /* compute relocation address : section are relocated in place. We also alloc the bss space */ for(i = 1; i < s1->nb_sections; i++) {