+ if (mode & RTLD_GLOBAL) {
+ if (p->deps) for (i=0; p->deps[i]; i++)
+ p->deps[i]->global = 1;
+ p->global = 1;
+ }
+
+ do_init_fini(tail);
+end:
+ pthread_rwlock_unlock(&lock);
+ pthread_setcancelstate(cs, 0);
+ return p;
+}
+
+static void *do_dlsym(struct dso *p, const char *s, void *ra)
+{
+ size_t i;
+ uint32_t h;
+ Sym *sym;
+ if (p == RTLD_NEXT) {
+ for (p=head; p && (unsigned char *)ra-p->map>p->map_len; p=p->next);
+ if (!p) p=head;
+ p=p->next;
+ }
+ if (p == head || p == RTLD_DEFAULT) {
+ void *res = find_sym(head, s, 0);
+ if (!res) errflag = 1;
+ return res;
+ }
+ h = hash(s);
+ sym = lookup(s, h, p->syms, p->hashtab, p->strings);
+ if (sym && sym->st_value && (1<<(sym->st_info&0xf) & OK_TYPES))
+ return p->base + sym->st_value;
+ if (p->deps) for (i=0; p->deps[i]; i++) {
+ sym = lookup(s, h, p->deps[i]->syms,
+ p->deps[i]->hashtab, p->deps[i]->strings);
+ if (sym && sym->st_value && (1<<(sym->st_info&0xf) & OK_TYPES))
+ return p->deps[i]->base + sym->st_value;
+ }
+ errflag = 1;
+ snprintf(errbuf, sizeof errbuf, "Symbol not found: %s", s);
+ return 0;
+}
+
+void *__dlsym(void *p, const char *s, void *ra)
+{
+ void *res;
+ pthread_rwlock_rdlock(&lock);
+ res = do_dlsym(p, s, ra);
+ pthread_rwlock_unlock(&lock);
+ return res;
+}
+#else
+void *dlopen(const char *file, int mode)
+{
+ return 0;
+}
+void *__dlsym(void *p, const char *s, void *ra)
+{
+ return 0;
+}
+#endif
+
+char *dlerror()
+{
+ if (!errflag) return 0;
+ errflag = 0;
+ return errbuf;
+}
+
+int dlclose(void *p)
+{
+ return 0;