mirror of
https://github.com/frida/tinycc
synced 2025-01-20 18:29:18 +03:00
Fix symbol versions with shared libs
ELF files that refer to shared libs containing sym-versions, but don't refer to any dynamic symbols with symbol versions (should happen only with very simple shared libs) would generate an empty .gnu.version_r section. Some dynamic linker contain bugs in that they don't check the section size or DT_VERNEEDNUM (which are both zero for such files we generate) before accessing the first entry, and then bail out with a message like ./a.exe: error while loading shared libraries: ./a1.so: unsupported version 25960 of Verneed record (where the "version number" actually comes from neighboring bytes from different sections). So, there's not much choice, we simply must not generate such section.
This commit is contained in:
parent
d79e1dee8c
commit
fdeeb62e28
86
tccelf.c
86
tccelf.c
@ -558,9 +558,7 @@ version_add (TCCState *s1)
|
|||||||
return;
|
return;
|
||||||
versym_section = new_section(s1, ".gnu.version", SHT_GNU_versym, SHF_ALLOC);
|
versym_section = new_section(s1, ".gnu.version", SHT_GNU_versym, SHF_ALLOC);
|
||||||
versym_section->sh_entsize = sizeof(ElfW(Half));
|
versym_section->sh_entsize = sizeof(ElfW(Half));
|
||||||
verneed_section = new_section(s1, ".gnu.version_r", SHT_GNU_verneed, SHF_ALLOC);
|
|
||||||
versym_section->link = s1->dynsym;
|
versym_section->link = s1->dynsym;
|
||||||
verneed_section->link = s1->dynsym->link;
|
|
||||||
|
|
||||||
/* add needed symbols */
|
/* add needed symbols */
|
||||||
symtab = s1->dynsym;
|
symtab = s1->dynsym;
|
||||||
@ -580,43 +578,50 @@ version_add (TCCState *s1)
|
|||||||
} else
|
} else
|
||||||
versym[sym_index] = 0;
|
versym[sym_index] = 0;
|
||||||
}
|
}
|
||||||
/* generate verneed section */
|
/* generate verneed section, but not when it will be empty. Some
|
||||||
for (i = nb_sym_versions; i-- > 0;) {
|
dynamic linkers look at their contents even when DTVERNEEDNUM and
|
||||||
struct sym_version *sv = &sym_versions[i];
|
section size is zero. */
|
||||||
int n_same_libs = 0, prev;
|
if (nb_versions > 2) {
|
||||||
size_t vnofs;
|
verneed_section = new_section(s1, ".gnu.version_r",
|
||||||
ElfW(Vernaux) *vna = 0;
|
SHT_GNU_verneed, SHF_ALLOC);
|
||||||
if (sv->out_index < 1)
|
verneed_section->link = s1->dynsym->link;
|
||||||
continue;
|
for (i = nb_sym_versions; i-- > 0;) {
|
||||||
vnofs = section_add(verneed_section, sizeof(*vn), 1);
|
struct sym_version *sv = &sym_versions[i];
|
||||||
vn = (ElfW(Verneed)*)(verneed_section->data + vnofs);
|
int n_same_libs = 0, prev;
|
||||||
vn->vn_version = 1;
|
size_t vnofs;
|
||||||
vn->vn_file = put_elf_str(verneed_section->link, sv->lib);
|
ElfW(Vernaux) *vna = 0;
|
||||||
vn->vn_aux = sizeof (*vn);
|
if (sv->out_index < 1)
|
||||||
do {
|
continue;
|
||||||
prev = sv->prev_same_lib;
|
vnofs = section_add(verneed_section, sizeof(*vn), 1);
|
||||||
if (sv->out_index > 0) {
|
vn = (ElfW(Verneed)*)(verneed_section->data + vnofs);
|
||||||
vna = section_ptr_add(verneed_section, sizeof(*vna));
|
vn->vn_version = 1;
|
||||||
vna->vna_hash = elf_hash ((const unsigned char *)sv->version);
|
vn->vn_file = put_elf_str(verneed_section->link, sv->lib);
|
||||||
vna->vna_flags = 0;
|
vn->vn_aux = sizeof (*vn);
|
||||||
vna->vna_other = sv->out_index;
|
do {
|
||||||
sv->out_index = -2;
|
prev = sv->prev_same_lib;
|
||||||
vna->vna_name = put_elf_str(verneed_section->link, sv->version);
|
if (sv->out_index > 0) {
|
||||||
vna->vna_next = sizeof (*vna);
|
vna = section_ptr_add(verneed_section, sizeof(*vna));
|
||||||
n_same_libs++;
|
vna->vna_hash = elf_hash ((const unsigned char *)sv->version);
|
||||||
}
|
vna->vna_flags = 0;
|
||||||
if (prev >= 0)
|
vna->vna_other = sv->out_index;
|
||||||
sv = &sym_versions[prev];
|
sv->out_index = -2;
|
||||||
} while(prev >= 0);
|
vna->vna_name = put_elf_str(verneed_section->link, sv->version);
|
||||||
vna->vna_next = 0;
|
vna->vna_next = sizeof (*vna);
|
||||||
vn = (ElfW(Verneed)*)(verneed_section->data + vnofs);
|
n_same_libs++;
|
||||||
vn->vn_cnt = n_same_libs;
|
}
|
||||||
vn->vn_next = sizeof(*vn) + n_same_libs * sizeof(*vna);
|
if (prev >= 0)
|
||||||
nb_entries++;
|
sv = &sym_versions[prev];
|
||||||
|
} while(prev >= 0);
|
||||||
|
vna->vna_next = 0;
|
||||||
|
vn = (ElfW(Verneed)*)(verneed_section->data + vnofs);
|
||||||
|
vn->vn_cnt = n_same_libs;
|
||||||
|
vn->vn_next = sizeof(*vn) + n_same_libs * sizeof(*vna);
|
||||||
|
nb_entries++;
|
||||||
|
}
|
||||||
|
if (vn)
|
||||||
|
vn->vn_next = 0;
|
||||||
|
verneed_section->sh_info = nb_entries;
|
||||||
}
|
}
|
||||||
if (vn)
|
|
||||||
vn->vn_next = 0;
|
|
||||||
verneed_section->sh_info = nb_entries;
|
|
||||||
dt_verneednum = nb_entries;
|
dt_verneednum = nb_entries;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@ -2045,10 +2050,11 @@ static void fill_dynamic(TCCState *s1, struct dyn_inf *dyninf)
|
|||||||
put_dt(dynamic, DT_RELENT, sizeof(ElfW_Rel));
|
put_dt(dynamic, DT_RELENT, sizeof(ElfW_Rel));
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
if (versym_section) {
|
if (versym_section)
|
||||||
|
put_dt(dynamic, DT_VERSYM, versym_section->sh_addr);
|
||||||
|
if (verneed_section) {
|
||||||
put_dt(dynamic, DT_VERNEED, verneed_section->sh_addr);
|
put_dt(dynamic, DT_VERNEED, verneed_section->sh_addr);
|
||||||
put_dt(dynamic, DT_VERNEEDNUM, dt_verneednum);
|
put_dt(dynamic, DT_VERNEEDNUM, dt_verneednum);
|
||||||
put_dt(dynamic, DT_VERSYM, versym_section->sh_addr);
|
|
||||||
}
|
}
|
||||||
s = find_section_create (s1, ".preinit_array", 0);
|
s = find_section_create (s1, ".preinit_array", 0);
|
||||||
if (s && s->data_offset) {
|
if (s && s->data_offset) {
|
||||||
|
Loading…
Reference in New Issue
Block a user