|
|
296178 |
#define _BSD_SOURCE
|
|
|
296178 |
|
|
|
6f8ae4 |
#include <unistd.h>
|
|
|
6f8ae4 |
#include <stdint.h>
|
|
|
ca0f58 |
#include <stdlib.h>
|
|
|
eea0a8 |
#include <string.h>
|
|
|
296178 |
#include <dlfcn.h>
|
|
|
296178 |
#include "psxglue.h"
|
|
|
296178 |
#include "pthread_impl.h"
|
|
|
296178 |
|
|
|
eea0a8 |
extern const struct __ldso_vtbl * __ldso_vtbl;
|
|
|
eea0a8 |
extern const struct __psx_vtbl * __psx_vtbl;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
static pthread_rwlock_t __ldso_lock;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
void * dlopen(const char * file, int mode)
|
|
|
eea0a8 |
{
|
|
|
eea0a8 |
int status;
|
|
|
eea0a8 |
void * base;
|
|
|
eea0a8 |
int cs;
|
|
|
eea0a8 |
char * ch;
|
|
|
eea0a8 |
char * next;
|
|
|
eea0a8 |
char * epath;
|
|
|
eea0a8 |
char * lpath;
|
|
|
eea0a8 |
const char ** lpathv;
|
|
|
eea0a8 |
const char ** epathv;
|
|
|
eea0a8 |
char lpathbuf[2048];
|
|
|
eea0a8 |
const char * lpathvbuf[64];
|
|
|
eea0a8 |
int i;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
/* prolog */
|
|
|
eea0a8 |
if (!file)
|
|
|
eea0a8 |
return __ldso_vtbl->dlopen(0,mode,0,&status);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
|
|
|
eea0a8 |
pthread_rwlock_wrlock(&__ldso_lock);
|
|
|
eea0a8 |
__inhibit_ptc();
|
|
|
eea0a8 |
|
|
|
eea0a8 |
/* loader path environment variable to loader path vector */
|
|
|
eea0a8 |
if ((epath = getenv("LD_LIBRARY_PATH"))) {
|
|
|
eea0a8 |
lpath = (strncpy(lpathbuf,epath,2048) < &lpathbuf[2048])
|
|
|
eea0a8 |
? lpathbuf
|
|
|
eea0a8 |
: strdup(epath);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
if ((i = !!lpath))
|
|
|
eea0a8 |
for (ch=lpath; *ch; ch++)
|
|
|
eea0a8 |
if (*ch == ':')
|
|
|
eea0a8 |
i++;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
lpathv = (++i < 64)
|
|
|
eea0a8 |
? lpathvbuf
|
|
|
eea0a8 |
: calloc(++i,sizeof(char *));
|
|
|
eea0a8 |
} else {
|
|
|
eea0a8 |
lpath = lpathbuf;
|
|
|
eea0a8 |
lpathv = lpathvbuf;
|
|
|
eea0a8 |
lpath[0] = 0;
|
|
|
eea0a8 |
}
|
|
|
eea0a8 |
|
|
|
eea0a8 |
if (lpath && lpathv) {
|
|
|
eea0a8 |
ch = lpath;
|
|
|
eea0a8 |
next = *ch ? ch : 0;
|
|
|
eea0a8 |
epathv = lpathv;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
for (; next; ) {
|
|
|
eea0a8 |
*epathv++ = (*next == ':')
|
|
|
eea0a8 |
? "."
|
|
|
eea0a8 |
: next;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
ch = &next[1];
|
|
|
eea0a8 |
|
|
|
eea0a8 |
for (; *ch; ) {
|
|
|
eea0a8 |
if (*ch == ':') {
|
|
|
eea0a8 |
*ch = 0;
|
|
|
eea0a8 |
ch = 0;
|
|
|
eea0a8 |
} else {
|
|
|
eea0a8 |
ch++;
|
|
|
eea0a8 |
}
|
|
|
eea0a8 |
}
|
|
|
eea0a8 |
|
|
|
eea0a8 |
next = *ch ? ch : 0;
|
|
|
eea0a8 |
}
|
|
|
eea0a8 |
|
|
|
eea0a8 |
*epathv = 0;
|
|
|
eea0a8 |
}
|
|
|
eea0a8 |
|
|
|
eea0a8 |
/* dlopen */
|
|
|
eea0a8 |
base = (lpath && lpathv)
|
|
|
eea0a8 |
? __ldso_vtbl->dlopen(file,mode,lpathv,&status)
|
|
|
eea0a8 |
: 0;
|
|
|
eea0a8 |
|
|
|
eea0a8 |
/* epilog */
|
|
|
eea0a8 |
if (lpath && (lpath != lpathbuf))
|
|
|
eea0a8 |
free(lpath);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
if (lpathv && (lpathv != lpathvbuf))
|
|
|
eea0a8 |
free(lpathv);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
__release_ptc();
|
|
|
eea0a8 |
pthread_rwlock_unlock(&__ldso_lock);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
if (base)
|
|
|
eea0a8 |
__psx_vtbl->do_global_ctors_fn();
|
|
|
eea0a8 |
|
|
|
eea0a8 |
pthread_setcancelstate(cs, 0);
|
|
|
eea0a8 |
|
|
|
eea0a8 |
return base;
|
|
|
eea0a8 |
}
|
|
|
296178 |
|
|
|
296178 |
int __dladdr(const void * addr, Dl_info * info)
|
|
|
296178 |
{
|
|
|
296178 |
return __ldso_vtbl->dladdr(addr,info);
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
296178 |
int __dlinfo(void * dso, int req, void * res)
|
|
|
296178 |
{
|
|
|
e6ac10 |
return (__ldso_vtbl->dlinfo(dso,req,res)) ? -1 : 0;
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
b82ad8 |
void *__dlsym(void * restrict p, const char * restrict s)
|
|
|
296178 |
{
|
|
|
b82ad8 |
return __ldso_vtbl->dlsym(p,s,0);
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
296178 |
int dlclose(void *p)
|
|
|
296178 |
{
|
|
|
296178 |
return __ldso_vtbl->dlclose(p);
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
296178 |
char * dlerror(void)
|
|
|
296178 |
{
|
|
|
296178 |
return __ldso_vtbl->dlerror();
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
296178 |
void __reset_tls(void)
|
|
|
296178 |
{
|
|
|
296178 |
__ldso_vtbl->reset_tls();
|
|
|
296178 |
}
|
|
|
296178 |
|
|
|
296178 |
void *__copy_tls(unsigned char * mem)
|
|
|
296178 |
{
|
|
|
296178 |
/**
|
|
|
296178 |
* this is always the simple case, since:
|
|
|
296178 |
* emutls is based on PE named sections; and
|
|
|
296178 |
* tls allocation and initialization are handled by clone(2)
|
|
|
296178 |
**/
|
|
|
296178 |
|
|
|
296178 |
pthread_t td;
|
|
|
5085e9 |
uintptr_t addr;
|
|
|
296178 |
|
|
|
5085e9 |
addr = (uintptr_t)mem;
|
|
|
5085e9 |
addr >>= 4;
|
|
|
5085e9 |
addr <<= 4;
|
|
|
5085e9 |
addr += 16;
|
|
|
296178 |
|
|
|
5085e9 |
td = (struct __pthread *)addr;
|
|
|
5085e9 |
td->dtv = 0;
|
|
|
296178 |
|
|
|
296178 |
return td;
|
|
|
296178 |
}
|
|
|
d88860 |
|
|
|
d88860 |
weak_alias(__dlsym,dlsym);
|
|
|
d88860 |
weak_alias(__dlinfo,dlinfo);
|