Reinstate attribute alias handling
commit 2a0167a
merged alias and asm symbol renaming, but broke
semantics of aliases, see testcase. Basically the difference between
the two is that an asm rename doesn't generate a new symbol, i.e. with
int foo __asm__("bar");
all source reference to 'foo' will be to 'bar', nothing of the name
'foo' will remain in the object file, and for instance reference to
'foo' from other compilation units won't be resolved to this one.
Aliases OTOH create an additional symbol. With:
void target (void) { return; }
void afunc (void) __attribute__((alias("target")));
reference to 'afunc' will remain 'afunc' in the object file. It will
generate two symbols, 'afunc' and 'target' referring to the same entity.
This difference matters if other compilation units make references to
'afunc'.
A side requirement of this is that for alias to work that the target
symbol needs to be defined in the same unit. For TCC we even require a
stricter variant: it must be defined before the alias is created.
Now, with this I merely re-instated the old flow of events before above
commit. It didn't seem useful anymore to place both names in the
asm_label member of attributes, and the asm_label member of Sym now
again only needs the hold the __asm__ rename.
It also follows that tcc_predefs.h can't make use of attribute alias to
e.g. map __builtin_memcpy to __bound_memcpy (simply because the latter
isn't defined in all units), but rather must use __asm__ renaming, which
in turn means that the underscore handling needs to be done by hand.
This commit is contained in:
parent
727e24cb0a
commit
352e1d0fc4
7 changed files with 80 additions and 10 deletions
1
tcc.h
1
tcc.h
|
@ -643,6 +643,7 @@ typedef struct AttributeDef {
|
|||
struct FuncAttr f;
|
||||
struct Section *section;
|
||||
Sym *cleanup_func;
|
||||
int alias_target; /* token */
|
||||
int asm_label; /* associated asm label */
|
||||
char attr_mode; /* __attribute__((__mode__(...))) */
|
||||
} AttributeDef;
|
||||
|
|
26
tccgen.c
26
tccgen.c
|
@ -939,10 +939,8 @@ ST_FUNC void put_extern_sym2(Sym *sym, int sh_num,
|
|||
#endif
|
||||
|
||||
if (sym->asm_label) {
|
||||
name = get_tok_str(sym->asm_label & ~SYM_FIELD, NULL);
|
||||
/* with SYM_FIELD it was __attribute__((alias("..."))) actually */
|
||||
if (!(sym->asm_label & SYM_FIELD))
|
||||
can_add_underscore = 0;
|
||||
name = get_tok_str(sym->asm_label, NULL);
|
||||
can_add_underscore = 0;
|
||||
}
|
||||
|
||||
if (tcc_state->leading_underscore && can_add_underscore) {
|
||||
|
@ -1524,6 +1522,8 @@ static void merge_attr(AttributeDef *ad, AttributeDef *ad1)
|
|||
|
||||
if (ad1->section)
|
||||
ad->section = ad1->section;
|
||||
if (ad1->alias_target)
|
||||
ad->alias_target = ad1->alias_target;
|
||||
if (ad1->asm_label)
|
||||
ad->asm_label = ad1->asm_label;
|
||||
if (ad1->attr_mode)
|
||||
|
@ -4147,8 +4147,8 @@ redo:
|
|||
case TOK_ALIAS2:
|
||||
skip('(');
|
||||
parse_mult_str(&astr, "alias(\"target\")");
|
||||
ad->asm_label = /* save string as token, for later */
|
||||
tok_alloc((char*)astr.data, astr.size-1)->tok | SYM_FIELD;
|
||||
ad->alias_target = /* save string as token, for later */
|
||||
tok_alloc((char*)astr.data, astr.size-1)->tok;
|
||||
skip(')');
|
||||
cstr_free(&astr);
|
||||
break;
|
||||
|
@ -8459,6 +8459,20 @@ found:
|
|||
/* external variable or function */
|
||||
type.t |= VT_EXTERN;
|
||||
sym = external_sym(v, &type, r, &ad);
|
||||
if (ad.alias_target) {
|
||||
/* Aliases need to be emitted when their target
|
||||
symbol is emitted, even if perhaps unreferenced.
|
||||
We only support the case where the base is
|
||||
already defined, otherwise we would need
|
||||
deferring to emit the aliases until the end of
|
||||
the compile unit. */
|
||||
Sym *alias_target = sym_find(ad.alias_target);
|
||||
ElfSym *esym = elfsym(alias_target);
|
||||
if (!esym)
|
||||
tcc_error("unsupported forward __alias__ attribute");
|
||||
put_extern_sym2(sym, esym->st_shndx,
|
||||
esym->st_value, esym->st_size, 0);
|
||||
}
|
||||
} else {
|
||||
if (type.t & VT_STATIC)
|
||||
r |= VT_CONST;
|
||||
|
|
13
tccpp.c
13
tccpp.c
|
@ -3673,16 +3673,21 @@ static void tcc_predefs(CString *cstr)
|
|||
"#ifndef __builtin_va_copy\n"
|
||||
"#define __builtin_va_copy(dest,src) (dest)=(src)\n"
|
||||
"#endif\n"
|
||||
"#ifdef __leading_underscore\n"
|
||||
"#define RENAME(X) __asm__(\"_\"X)\n"
|
||||
"#else\n"
|
||||
"#define RENAME(X) __asm__(X)\n"
|
||||
"#endif\n"
|
||||
/* TCC BBUILTIN AND BOUNDS ALIASES */
|
||||
"#ifdef __BOUNDS_CHECKING_ON\n"
|
||||
"#define __BUILTINBC(ret,name,params) ret __builtin_##name params __attribute__((alias(\"__bound_\"#name)));\n"
|
||||
"#define __BOUND(ret,name,params) ret name params __attribute__((alias(\"__bound_\"#name)));\n"
|
||||
"#define __BUILTINBC(ret,name,params) ret __builtin_##name params RENAME(\"__bound_\"#name);\n"
|
||||
"#define __BOUND(ret,name,params) ret name params RENAME(\"__bound_\"#name);\n"
|
||||
"#else\n"
|
||||
"#define __BUILTINBC(ret,name,params) ret __builtin_##name params __attribute__((alias(#name)));\n"
|
||||
"#define __BUILTINBC(ret,name,params) ret __builtin_##name params RENAME(#name);\n"
|
||||
"#define __BOUND(ret,name,params)\n"
|
||||
"#endif\n"
|
||||
"#define __BOTH(ret,name,params) __BUILTINBC(ret,name,params)__BOUND(ret,name,params)\n"
|
||||
"#define __BUILTIN(ret,name,params) ret __builtin_##name params __attribute__((alias(\"\"#name)));\n"
|
||||
"#define __BUILTIN(ret,name,params) ret __builtin_##name params RENAME(#name);\n"
|
||||
"__BOTH(void*,memcpy,(void*,const void*,__SIZE_TYPE__))\n"
|
||||
"__BOTH(void*,memmove,(void*,const void*,__SIZE_TYPE__))\n"
|
||||
"__BOTH(void*,memset,(void*,int,__SIZE_TYPE__))\n"
|
||||
|
|
15
tests/tests2/120+_alias.c
Normal file
15
tests/tests2/120+_alias.c
Normal file
|
@ -0,0 +1,15 @@
|
|||
extern int printf (const char *, ...);
|
||||
extern void target(void);
|
||||
extern void alias_for_target(void);
|
||||
extern void asm_for_target(void);
|
||||
|
||||
void inunit2(void);
|
||||
|
||||
void inunit2(void)
|
||||
{
|
||||
target();
|
||||
alias_for_target();
|
||||
/* This symbol is not supposed to be available in this unit:
|
||||
asm_for_target();
|
||||
*/
|
||||
}
|
25
tests/tests2/120_alias.c
Normal file
25
tests/tests2/120_alias.c
Normal file
|
@ -0,0 +1,25 @@
|
|||
/* Check semantics of various constructs to generate renamed symbols. */
|
||||
extern int printf (const char *, ...);
|
||||
void target(void);
|
||||
void target(void) {
|
||||
printf("in target function\n");
|
||||
}
|
||||
|
||||
void alias_for_target(void) __attribute__((alias("target")));
|
||||
void asm_for_target(void) __asm__("target");
|
||||
|
||||
/* This is not supposed to compile, alias targets must be defined in the
|
||||
same unit. In TCC they even must be defined before the reference
|
||||
void alias_for_undef(void) __attribute__((alias("undefined")));
|
||||
*/
|
||||
|
||||
extern void inunit2(void);
|
||||
|
||||
int main(void)
|
||||
{
|
||||
target();
|
||||
alias_for_target();
|
||||
asm_for_target();
|
||||
inunit2();
|
||||
return 0;
|
||||
}
|
5
tests/tests2/120_alias.expect
Normal file
5
tests/tests2/120_alias.expect
Normal file
|
@ -0,0 +1,5 @@
|
|||
in target function
|
||||
in target function
|
||||
in target function
|
||||
in target function
|
||||
in target function
|
|
@ -74,6 +74,11 @@ GEN-ALWAYS =
|
|||
104_inline.test : FLAGS += $(subst 104,104+,$1)
|
||||
104_inline.test : GEN = $(GEN-TCC)
|
||||
|
||||
# this test needs two files, and we want to invoke the linker
|
||||
120_alias.test : FLAGS += $(subst 120,120+,$1)
|
||||
120_alias.test : GEN = $(GEN-TCC)
|
||||
120_alias.test : NORUN = true
|
||||
|
||||
# this test needs pthread
|
||||
106_pthread.test: FLAGS += -pthread
|
||||
106_pthread.test: NORUN = true
|
||||
|
|
Loading…
Reference in a new issue