summaryrefslogtreecommitdiff
path: root/src/internal/vdso.c
diff options
context:
space:
mode:
authorRich Felker <dalias@aerifal.cx>2014-04-16 02:33:29 -0400
committerRich Felker <dalias@aerifal.cx>2014-04-16 02:33:29 -0400
commit58e75db47160bc7bcac2ae98a7a8660b8fce35c9 (patch)
treefc8059d8f3289b50dff67ccce6224ca6e3348741 /src/internal/vdso.c
parent0d0c2f40344640a2a6942dda156509593f51db5d (diff)
downloadmusl-58e75db47160bc7bcac2ae98a7a8660b8fce35c9.tar.gz
add working vdso clock_gettime support, including static linking
the vdso symbol lookup code is based on the original 2011 patch by Nicholas J. Kain, with some streamlining, pointer arithmetic fixes, and one symbol version matching fix. on the consumer side (clock_gettime), per-arch macros for the particular symbol name and version to lookup are added in syscall_arch.h, and no vdso code is pulled in on archs which do not define these macros. at this time, vdso is enabled only on x86_64. the vdso support at the dynamic linker level is no longer useful to libc, but is left in place for the sake of debuggers (which may need the vdso in the link map to find its functions) and possibly use with dlsym.
Diffstat (limited to 'src/internal/vdso.c')
-rw-r--r--src/internal/vdso.c91
1 files changed, 91 insertions, 0 deletions
diff --git a/src/internal/vdso.c b/src/internal/vdso.c
new file mode 100644
index 00000000..6ae0212e
--- /dev/null
+++ b/src/internal/vdso.c
@@ -0,0 +1,91 @@
+#include <elf.h>
+#include <limits.h>
+#include <stdint.h>
+#include <string.h>
+#include "libc.h"
+#include "syscall.h"
+
+#ifdef VDSO_USEFUL
+
+#if ULONG_MAX == 0xffffffff
+typedef Elf32_Ehdr Ehdr;
+typedef Elf32_Phdr Phdr;
+typedef Elf32_Sym Sym;
+typedef Elf32_Verdef Verdef;
+typedef Elf32_Verdaux Verdaux;
+#else
+typedef Elf64_Ehdr Ehdr;
+typedef Elf64_Phdr Phdr;
+typedef Elf64_Sym Sym;
+typedef Elf64_Verdef Verdef;
+typedef Elf64_Verdaux Verdaux;
+#endif
+
+static int checkver(Verdef *def, int vsym, const char *vername, char *strings)
+{
+ vsym &= 0x7fff;
+ for (;;) {
+ if (!(def->vd_flags & VER_FLG_BASE)
+ && (def->vd_ndx & 0x7fff) == vsym)
+ break;
+ if (def->vd_next == 0)
+ return 0;
+ def = (Verdef *)((char *)def + def->vd_next);
+ }
+ Verdaux *aux = (Verdaux *)((char *)def + def->vd_aux);
+ return !strcmp(vername, strings + aux->vda_name);
+}
+
+#define OK_TYPES (1<<STT_NOTYPE | 1<<STT_OBJECT | 1<<STT_FUNC | 1<<STT_COMMON)
+#define OK_BINDS (1<<STB_GLOBAL | 1<<STB_WEAK | 1<<STB_GNU_UNIQUE)
+
+void *__vdsosym(const char *vername, const char *name)
+{
+ size_t i;
+ for (i=0; libc.auxv[i] != AT_SYSINFO_EHDR; i+=2)
+ if (!libc.auxv[i]) return 0;
+ Ehdr *eh = (void *)libc.auxv[i+1];
+ Phdr *ph = (void *)((char *)eh + eh->e_phoff);
+ size_t *dynv=0, base=-1;
+ for (i=0; i<eh->e_phnum; i++, ph=(void *)((char *)ph+eh->e_phentsize)) {
+ if (ph->p_type == PT_LOAD)
+ base = (size_t)eh + ph->p_offset - ph->p_vaddr;
+ else if (ph->p_type == PT_DYNAMIC)
+ dynv = (void *)((char *)eh + ph->p_offset);
+ }
+ if (!dynv || base==(size_t)-1) return 0;
+
+ char *strings = 0;
+ Sym *syms = 0;
+ uint32_t *hashtab = 0;
+ uint16_t *versym = 0;
+ Verdef *verdef = 0;
+
+ for (i=0; dynv[i]; i+=2) {
+ void *p = (void *)(base + dynv[i+1]);
+ switch(dynv[i]) {
+ case DT_STRTAB: strings = p; break;
+ case DT_SYMTAB: syms = p; break;
+ case DT_HASH: hashtab = p; break;
+ case DT_VERSYM: versym = p; break;
+ case DT_VERDEF: verdef = p; break;
+ }
+ }
+
+ if (!strings || !syms || !hashtab) return 0;
+ if (!verdef) versym = 0;
+
+ for (i=0; i<hashtab[1]; i++) {
+ if (!(1<<(syms[i].st_info&0xf) & OK_TYPES)) continue;
+ if (!(1<<(syms[i].st_info>>4) & OK_BINDS)) continue;
+ if (!syms[i].st_shndx) continue;
+ if (strcmp(name, strings+syms[i].st_name)) continue;
+ if (versym && !checkver(verdef, versym[i], vername, strings))
+ continue;
+ return (void *)(base + syms[i].st_value);
+ }
+
+ return 0;
+}
+
+#endif