Incorporated the change 1619 into our tree. Cleaned up the source.
git-svn-id: file:///srv/svn/repos/haiku/trunk/current@958 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
5813c3dff5
commit
3f8b94d3cd
@ -98,11 +98,11 @@ struct image_queue_t {
|
|||||||
} image_queue_t;
|
} image_queue_t;
|
||||||
|
|
||||||
|
|
||||||
static image_queue_t loaded_images= { 0, 0 };
|
static image_queue_t loaded_images = { 0, 0 };
|
||||||
static image_queue_t loading_images= { 0, 0 };
|
static image_queue_t loading_images = { 0, 0 };
|
||||||
static image_queue_t disposable_images= { 0, 0 };
|
static image_queue_t disposable_images = { 0, 0 };
|
||||||
static unsigned loaded_image_count= 0;
|
static unsigned loaded_image_count = 0;
|
||||||
static unsigned imageid_count= 0;
|
static unsigned imageid_count = 0;
|
||||||
|
|
||||||
static sem_id rld_sem;
|
static sem_id rld_sem;
|
||||||
static struct uspace_prog_args_t const *uspa;
|
static struct uspace_prog_args_t const *uspa;
|
||||||
@ -120,31 +120,29 @@ static struct uspace_prog_args_t const *uspa;
|
|||||||
* This macro is non ISO compliant, but a gcc extension
|
* This macro is non ISO compliant, but a gcc extension
|
||||||
*/
|
*/
|
||||||
#define FATAL(x,y...) \
|
#define FATAL(x,y...) \
|
||||||
if(x) { \
|
if (x) { \
|
||||||
printf("rld.so: " y); \
|
printf("rld.so: " y); \
|
||||||
sys_exit(0); \
|
sys_exit(0); \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static
|
static void
|
||||||
void
|
|
||||||
enqueue_image(image_queue_t *queue, image_t *img)
|
enqueue_image(image_queue_t *queue, image_t *img)
|
||||||
{
|
{
|
||||||
img->next= 0;
|
img->next = 0;
|
||||||
|
|
||||||
img->prev= queue->tail;
|
img->prev = queue->tail;
|
||||||
if(queue->tail) {
|
if (queue->tail)
|
||||||
queue->tail->next= img;
|
queue->tail->next= img;
|
||||||
}
|
|
||||||
queue->tail= img;
|
queue->tail = img;
|
||||||
if(!queue->head) {
|
if (!queue->head)
|
||||||
queue->head= img;
|
queue->head= img;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
void
|
static void
|
||||||
dequeue_image(image_queue_t *queue, image_t *img)
|
dequeue_image(image_queue_t *queue, image_t *img)
|
||||||
{
|
{
|
||||||
if(img->next) {
|
if(img->next) {
|
||||||
@ -163,16 +161,16 @@ dequeue_image(image_queue_t *queue, image_t *img)
|
|||||||
img->next= 0;
|
img->next= 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
unsigned long
|
static unsigned long
|
||||||
elf_hash(const unsigned char *name)
|
elf_hash(const unsigned char *name)
|
||||||
{
|
{
|
||||||
unsigned long hash = 0;
|
unsigned long hash = 0;
|
||||||
unsigned long temp;
|
unsigned long temp;
|
||||||
|
|
||||||
while(*name) {
|
while (*name) {
|
||||||
hash = (hash << 4) + *name++;
|
hash = (hash << 4) + *name++;
|
||||||
if((temp = hash & 0xf0000000)) {
|
if ((temp = hash & 0xf0000000)) {
|
||||||
hash ^= temp >> 24;
|
hash ^= temp >> 24;
|
||||||
}
|
}
|
||||||
hash &= ~temp;
|
hash &= ~temp;
|
||||||
@ -180,78 +178,70 @@ elf_hash(const unsigned char *name)
|
|||||||
return hash;
|
return hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
image_t *
|
static image_t *
|
||||||
find_image(char const *name)
|
find_image(char const *name)
|
||||||
{
|
{
|
||||||
image_t *iter;
|
image_t *iter;
|
||||||
|
|
||||||
iter= loaded_images.head;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
if (strncmp(iter->name, name, sizeof(iter->name)) == 0)
|
||||||
if(strncmp(iter->name, name, sizeof(iter->name)) == 0) {
|
|
||||||
return iter;
|
return iter;
|
||||||
}
|
}
|
||||||
iter= iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
iter= loading_images.head;
|
for (iter = loading_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
if (strncmp(iter->name, name, sizeof(iter->name)) == 0)
|
||||||
if(strncmp(iter->name, name, sizeof(iter->name)) == 0) {
|
|
||||||
return iter;
|
return iter;
|
||||||
}
|
}
|
||||||
iter= iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
int
|
static int
|
||||||
parse_eheader(struct Elf32_Ehdr *eheader)
|
parse_eheader(struct Elf32_Ehdr *eheader)
|
||||||
{
|
{
|
||||||
if(memcmp(eheader->e_ident, ELF_MAGIC, 4) != 0)
|
if (memcmp(eheader->e_ident, ELF_MAGIC, 4) != 0)
|
||||||
return ERR_INVALID_BINARY;
|
return ERR_INVALID_BINARY;
|
||||||
|
|
||||||
if(eheader->e_ident[4] != ELFCLASS32)
|
if (eheader->e_ident[4] != ELFCLASS32)
|
||||||
return ERR_INVALID_BINARY;
|
return ERR_INVALID_BINARY;
|
||||||
|
|
||||||
if(eheader->e_phoff == 0)
|
if (eheader->e_phoff == 0)
|
||||||
return ERR_INVALID_BINARY;
|
return ERR_INVALID_BINARY;
|
||||||
|
|
||||||
if(eheader->e_phentsize < sizeof(struct Elf32_Phdr))
|
if (eheader->e_phentsize < sizeof(struct Elf32_Phdr))
|
||||||
return ERR_INVALID_BINARY;
|
return ERR_INVALID_BINARY;
|
||||||
|
|
||||||
return eheader->e_phentsize*eheader->e_phnum;
|
return eheader->e_phentsize*eheader->e_phnum;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
int
|
static int
|
||||||
count_regions(char const *buff, int phnum, int phentsize)
|
count_regions(char const *buff, int phnum, int phentsize)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int retval;
|
int retval = 0;
|
||||||
struct Elf32_Phdr *pheaders;
|
struct Elf32_Phdr *pheaders;
|
||||||
|
|
||||||
retval= 0;
|
for (i = 0; i < phnum; i++) {
|
||||||
for(i= 0; i< phnum; i++) {
|
pheaders = (struct Elf32_Phdr *)(buff + i * phentsize);
|
||||||
pheaders= (struct Elf32_Phdr *)(buff+i*phentsize);
|
|
||||||
|
|
||||||
switch(pheaders->p_type) {
|
switch (pheaders->p_type) {
|
||||||
case PT_NULL:
|
case PT_NULL:
|
||||||
/* NOP header */
|
/* NOP header */
|
||||||
break;
|
break;
|
||||||
case PT_LOAD:
|
case PT_LOAD:
|
||||||
retval+= 1;
|
retval += 1;
|
||||||
if(pheaders->p_memsz!= pheaders->p_filesz) {
|
if (pheaders->p_memsz!= pheaders->p_filesz) {
|
||||||
unsigned A= pheaders->p_vaddr+pheaders->p_memsz;
|
unsigned A = pheaders->p_vaddr + pheaders->p_memsz;
|
||||||
unsigned B= pheaders->p_vaddr+pheaders->p_filesz;
|
unsigned B = pheaders->p_vaddr + pheaders->p_filesz - 1;
|
||||||
|
|
||||||
A= PAGE_BASE(A);
|
A = PAGE_BASE(A);
|
||||||
B= PAGE_BASE(B);
|
B = PAGE_BASE(B);
|
||||||
|
|
||||||
if(A!= B) {
|
if (A != B)
|
||||||
retval+= 1;
|
retval += 1;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case PT_DYNAMIC:
|
case PT_DYNAMIC:
|
||||||
@ -279,43 +269,42 @@ count_regions(char const *buff, int phnum, int phentsize)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* create_image() & destroy_image()
|
* create_image() & destroy_image()
|
||||||
*
|
*
|
||||||
* Create and destroy image_t structures. The destroyer makes sure that the
|
* Create and destroy image_t structures. The destroyer makes sure that the
|
||||||
* memory buffers are full of garbage before freeing.
|
* memory buffers are full of garbage before freeing.
|
||||||
*/
|
*/
|
||||||
static
|
|
||||||
image_t *
|
static image_t *
|
||||||
create_image(char const *name, int num_regions)
|
create_image(char const *name, int num_regions)
|
||||||
{
|
{
|
||||||
image_t *retval;
|
image_t *retval;
|
||||||
size_t alloc_size;
|
size_t alloc_size;
|
||||||
|
|
||||||
alloc_size= sizeof(image_t)+(num_regions-1)*sizeof(elf_region_t);
|
alloc_size = sizeof(image_t) + (num_regions - 1) * sizeof(elf_region_t);
|
||||||
|
|
||||||
retval= rldalloc(alloc_size);
|
retval = rldalloc(alloc_size);
|
||||||
|
|
||||||
memset(retval, 0, alloc_size);
|
memset(retval, 0, alloc_size);
|
||||||
|
|
||||||
strlcpy(retval->name, name, sizeof(retval->name));
|
strlcpy(retval->name, name, sizeof(retval->name));
|
||||||
retval->imageid= imageid_count;
|
retval->imageid = imageid_count;
|
||||||
retval->refcount= 1;
|
retval->refcount = 1;
|
||||||
retval->num_regions= num_regions;
|
retval->num_regions = num_regions;
|
||||||
|
|
||||||
imageid_count+= 1;
|
imageid_count += 1;
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
void
|
static void
|
||||||
destroy_image(image_t *image)
|
destroy_image(image_t *image)
|
||||||
{
|
{
|
||||||
size_t alloc_size;
|
size_t alloc_size;
|
||||||
|
|
||||||
alloc_size= sizeof(image_t)+(image->num_regions-1)*sizeof(elf_region_t);
|
alloc_size = sizeof(image_t) + (image->num_regions - 1) * sizeof(elf_region_t);
|
||||||
|
|
||||||
memset(image->needed, 0xa5, sizeof(image->needed[0])*image->num_needed);
|
memset(image->needed, 0xa5, sizeof(image->needed[0])*image->num_needed);
|
||||||
rldfree(image->needed);
|
rldfree(image->needed);
|
||||||
@ -325,25 +314,23 @@ destroy_image(image_t *image)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
static
|
|
||||||
void
|
|
||||||
parse_program_headers(image_t *image, char *buff, int phnum, int phentsize)
|
parse_program_headers(image_t *image, char *buff, int phnum, int phentsize)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int regcount;
|
int regcount;
|
||||||
struct Elf32_Phdr *pheaders;
|
struct Elf32_Phdr *pheaders;
|
||||||
|
|
||||||
regcount= 0;
|
regcount = 0;
|
||||||
for(i= 0; i< phnum; i++) {
|
for (i = 0; i < phnum; i++) {
|
||||||
pheaders= (struct Elf32_Phdr *)(buff+i*phentsize);
|
pheaders = (struct Elf32_Phdr *)(buff + i * phentsize);
|
||||||
|
|
||||||
switch(pheaders->p_type) {
|
switch (pheaders->p_type) {
|
||||||
case PT_NULL:
|
case PT_NULL:
|
||||||
/* NOP header */
|
/* NOP header */
|
||||||
break;
|
break;
|
||||||
case PT_LOAD:
|
case PT_LOAD:
|
||||||
if(pheaders->p_memsz== pheaders->p_filesz) {
|
if (pheaders->p_memsz== pheaders->p_filesz) {
|
||||||
/*
|
/*
|
||||||
* everything in one area
|
* everything in one area
|
||||||
*/
|
*/
|
||||||
@ -363,45 +350,45 @@ parse_program_headers(image_t *image, char *buff, int phnum, int phentsize)
|
|||||||
/*
|
/*
|
||||||
* may require splitting
|
* may require splitting
|
||||||
*/
|
*/
|
||||||
unsigned A= pheaders->p_vaddr+pheaders->p_memsz;
|
unsigned A = pheaders->p_vaddr + pheaders->p_memsz;
|
||||||
unsigned B= pheaders->p_vaddr+pheaders->p_filesz;
|
unsigned B = pheaders->p_vaddr + pheaders->p_filesz - 1;
|
||||||
|
|
||||||
A= PAGE_BASE(A);
|
A = PAGE_BASE(A);
|
||||||
B= PAGE_BASE(B);
|
B = PAGE_BASE(B);
|
||||||
|
|
||||||
image->regions[regcount].start = pheaders->p_vaddr;
|
image->regions[regcount].start = pheaders->p_vaddr;
|
||||||
image->regions[regcount].size = pheaders->p_filesz;
|
image->regions[regcount].size = pheaders->p_filesz;
|
||||||
image->regions[regcount].vmstart= _ROUNDOWN(pheaders->p_vaddr, PAGE_SIZE);
|
image->regions[regcount].vmstart = _ROUNDOWN(pheaders->p_vaddr, PAGE_SIZE);
|
||||||
image->regions[regcount].vmsize = _ROUNDUP (pheaders->p_filesz + (pheaders->p_vaddr % PAGE_SIZE), PAGE_SIZE);
|
image->regions[regcount].vmsize = _ROUNDUP (pheaders->p_filesz + (pheaders->p_vaddr % PAGE_SIZE), PAGE_SIZE);
|
||||||
image->regions[regcount].fdstart= pheaders->p_offset;
|
image->regions[regcount].fdstart = pheaders->p_offset;
|
||||||
image->regions[regcount].fdsize = pheaders->p_filesz;
|
image->regions[regcount].fdsize = pheaders->p_filesz;
|
||||||
image->regions[regcount].delta= 0;
|
image->regions[regcount].delta = 0;
|
||||||
image->regions[regcount].flags= 0;
|
image->regions[regcount].flags = 0;
|
||||||
if(pheaders->p_flags & PF_W) {
|
if (pheaders->p_flags & PF_W) {
|
||||||
// this is a writable segment
|
// this is a writable segment
|
||||||
image->regions[regcount].flags|= RFLAG_RW;
|
image->regions[regcount].flags|= RFLAG_RW;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(A!= B) {
|
if (A!= B) {
|
||||||
/*
|
/*
|
||||||
* yeah, it requires splitting
|
* yeah, it requires splitting
|
||||||
*/
|
*/
|
||||||
regcount+= 1;
|
regcount += 1;
|
||||||
image->regions[regcount].start = pheaders->p_vaddr;
|
image->regions[regcount].start = pheaders->p_vaddr;
|
||||||
image->regions[regcount].size = pheaders->p_memsz - pheaders->p_filesz;
|
image->regions[regcount].size = pheaders->p_memsz - pheaders->p_filesz;
|
||||||
image->regions[regcount].vmstart= image->regions[regcount-1].vmstart + image->regions[regcount-1].vmsize;
|
image->regions[regcount].vmstart = image->regions[regcount-1].vmstart + image->regions[regcount-1].vmsize;
|
||||||
image->regions[regcount].vmsize = _ROUNDUP (pheaders->p_memsz + (pheaders->p_vaddr % PAGE_SIZE), PAGE_SIZE) - image->regions[regcount-1].vmsize;
|
image->regions[regcount].vmsize = _ROUNDUP (pheaders->p_memsz + (pheaders->p_vaddr % PAGE_SIZE), PAGE_SIZE) - image->regions[regcount-1].vmsize;
|
||||||
image->regions[regcount].fdstart= 0;
|
image->regions[regcount].fdstart = 0;
|
||||||
image->regions[regcount].fdsize = 0;
|
image->regions[regcount].fdsize = 0;
|
||||||
image->regions[regcount].delta= 0;
|
image->regions[regcount].delta = 0;
|
||||||
image->regions[regcount].flags= RFLAG_ANON;
|
image->regions[regcount].flags = RFLAG_ANON;
|
||||||
if(pheaders->p_flags & PF_W) {
|
if (pheaders->p_flags & PF_W) {
|
||||||
// this is a writable segment
|
// this is a writable segment
|
||||||
image->regions[regcount].flags|= RFLAG_RW;
|
image->regions[regcount].flags|= RFLAG_RW;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
regcount+= 1;
|
regcount += 1;
|
||||||
break;
|
break;
|
||||||
case PT_DYNAMIC:
|
case PT_DYNAMIC:
|
||||||
image->dynamic_ptr = pheaders->p_vaddr;
|
image->dynamic_ptr = pheaders->p_vaddr;
|
||||||
@ -425,72 +412,64 @@ parse_program_headers(image_t *image, char *buff, int phnum, int phentsize)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
bool
|
static bool
|
||||||
assert_dynamic_loadable(image_t *image)
|
assert_dynamic_loadable(image_t *image)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
|
|
||||||
if(!image->dynamic_ptr) {
|
if (!image->dynamic_ptr)
|
||||||
return true;
|
return true;
|
||||||
}
|
|
||||||
|
|
||||||
for(i= 0; i< image->num_regions; i++) {
|
for (i = 0; i < image->num_regions; i++) {
|
||||||
if(image->dynamic_ptr>= image->regions[i].start) {
|
if (image->dynamic_ptr >= image->regions[i].start
|
||||||
if(image->dynamic_ptr< image->regions[i].start+image->regions[i].size) {
|
&& image->dynamic_ptr < image->regions[i].start + image->regions[i].size)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
bool
|
static bool
|
||||||
map_image(int fd, char const *path, image_t *image, bool fixed)
|
map_image(int fd, char const *path, image_t *image, bool fixed)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
|
|
||||||
(void)(fd);
|
(void)(fd);
|
||||||
|
|
||||||
for(i= 0; i< image->num_regions; i++) {
|
for (i = 0; i < image->num_regions; i++) {
|
||||||
char region_name[256];
|
char region_name[256];
|
||||||
addr load_address;
|
addr load_address;
|
||||||
unsigned addr_specifier;
|
unsigned addr_specifier;
|
||||||
|
|
||||||
sprintf(
|
sprintf(region_name, "%s:seg_%d(%s)",
|
||||||
region_name,
|
path, i, (image->regions[i].flags & RFLAG_RW) ? "RW" : "RO");
|
||||||
"%s:seg_%d(%s)",
|
|
||||||
path,
|
|
||||||
i,
|
|
||||||
(image->regions[i].flags&RFLAG_RW)?"RW":"RO"
|
|
||||||
);
|
|
||||||
|
|
||||||
if(image->dynamic_ptr && !fixed) {
|
if (image->dynamic_ptr && !fixed) {
|
||||||
/*
|
/*
|
||||||
* relocatable image... we can afford to place wherever
|
* relocatable image... we can afford to place wherever
|
||||||
*/
|
*/
|
||||||
if(i== 0) {
|
if (i == 0) {
|
||||||
/*
|
/*
|
||||||
* but only the first segment gets a free ride
|
* but only the first segment gets a free ride
|
||||||
*/
|
*/
|
||||||
load_address= 0;
|
load_address = 0;
|
||||||
addr_specifier= REGION_ADDR_ANY_ADDRESS;
|
addr_specifier = REGION_ADDR_ANY_ADDRESS;
|
||||||
} else {
|
} else {
|
||||||
load_address= image->regions[i].vmstart + image->regions[i-1].delta;
|
load_address = image->regions[i].vmstart + image->regions[i-1].delta;
|
||||||
addr_specifier= REGION_ADDR_EXACT_ADDRESS;
|
addr_specifier = REGION_ADDR_EXACT_ADDRESS;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/*
|
/*
|
||||||
* not relocatable, put it where it asks or die trying
|
* not relocatable, put it where it asks or die trying
|
||||||
*/
|
*/
|
||||||
load_address= image->regions[i].vmstart;
|
load_address = image->regions[i].vmstart;
|
||||||
addr_specifier= REGION_ADDR_EXACT_ADDRESS;
|
addr_specifier = REGION_ADDR_EXACT_ADDRESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(image->regions[i].flags & RFLAG_ANON) {
|
if (image->regions[i].flags & RFLAG_ANON) {
|
||||||
image->regions[i].id= sys_vm_create_anonymous_region(
|
image->regions[i].id = sys_vm_create_anonymous_region(
|
||||||
region_name,
|
region_name,
|
||||||
(void **)&load_address,
|
(void **)&load_address,
|
||||||
addr_specifier,
|
addr_specifier,
|
||||||
@ -499,13 +478,13 @@ map_image(int fd, char const *path, image_t *image, bool fixed)
|
|||||||
LOCK_RW
|
LOCK_RW
|
||||||
);
|
);
|
||||||
|
|
||||||
if(image->regions[i].id < 0) {
|
if (image->regions[i].id < 0)
|
||||||
goto error;
|
goto error;
|
||||||
}
|
|
||||||
image->regions[i].delta = load_address - image->regions[i].vmstart;
|
image->regions[i].delta = load_address - image->regions[i].vmstart;
|
||||||
image->regions[i].vmstart= load_address;
|
image->regions[i].vmstart = load_address;
|
||||||
} else {
|
} else {
|
||||||
image->regions[i].id= sys_vm_map_file(
|
image->regions[i].id = sys_vm_map_file(
|
||||||
region_name,
|
region_name,
|
||||||
(void **)&load_address,
|
(void **)&load_address,
|
||||||
addr_specifier,
|
addr_specifier,
|
||||||
@ -515,25 +494,23 @@ map_image(int fd, char const *path, image_t *image, bool fixed)
|
|||||||
path,
|
path,
|
||||||
_ROUNDOWN(image->regions[i].fdstart, PAGE_SIZE)
|
_ROUNDOWN(image->regions[i].fdstart, PAGE_SIZE)
|
||||||
);
|
);
|
||||||
if(image->regions[i].id < 0) {
|
if (image->regions[i].id < 0)
|
||||||
goto error;
|
goto error;
|
||||||
}
|
|
||||||
image->regions[i].delta = load_address - image->regions[i].vmstart;
|
image->regions[i].delta = load_address - image->regions[i].vmstart;
|
||||||
image->regions[i].vmstart= load_address;
|
image->regions[i].vmstart = load_address;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* handle trailer bits in data segment
|
* handle trailer bits in data segment
|
||||||
*/
|
*/
|
||||||
if(image->regions[i].flags & RFLAG_RW) {
|
if (image->regions[i].flags & RFLAG_RW) {
|
||||||
unsigned start_clearing;
|
unsigned start_clearing;
|
||||||
unsigned to_clear;
|
unsigned to_clear;
|
||||||
|
|
||||||
start_clearing=
|
start_clearing = image->regions[i].vmstart
|
||||||
image->regions[i].vmstart
|
|
||||||
+ PAGE_OFFS(image->regions[i].start)
|
+ PAGE_OFFS(image->regions[i].start)
|
||||||
+ image->regions[i].size;
|
+ image->regions[i].size;
|
||||||
to_clear=
|
to_clear = image->regions[i].vmsize
|
||||||
image->regions[i].vmsize
|
|
||||||
- PAGE_OFFS(image->regions[i].start)
|
- PAGE_OFFS(image->regions[i].start)
|
||||||
- image->regions[i].size;
|
- image->regions[i].size;
|
||||||
memset((void*)start_clearing, 0, to_clear);
|
memset((void*)start_clearing, 0, to_clear);
|
||||||
@ -541,9 +518,8 @@ map_image(int fd, char const *path, image_t *image, bool fixed)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(image->dynamic_ptr) {
|
if (image->dynamic_ptr)
|
||||||
image->dynamic_ptr+= image->regions[0].delta;
|
image->dynamic_ptr += image->regions[0].delta;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
@ -551,21 +527,21 @@ error:
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
void
|
static void
|
||||||
unmap_image(image_t *image)
|
unmap_image(image_t *image)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
|
|
||||||
for(i= 0; i< image->num_regions; i++) {
|
for (i = 0; i < image->num_regions; i++) {
|
||||||
sys_vm_delete_region(image->regions[i].id);
|
sys_vm_delete_region(image->regions[i].id);
|
||||||
|
|
||||||
image->regions[i].id= -1;
|
image->regions[i].id = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
bool
|
static bool
|
||||||
parse_dynamic_segment(image_t *image)
|
parse_dynamic_segment(image_t *image)
|
||||||
{
|
{
|
||||||
struct Elf32_Dyn *d;
|
struct Elf32_Dyn *d;
|
||||||
@ -576,14 +552,13 @@ parse_dynamic_segment(image_t *image)
|
|||||||
image->strtab = 0;
|
image->strtab = 0;
|
||||||
|
|
||||||
d = (struct Elf32_Dyn *)image->dynamic_ptr;
|
d = (struct Elf32_Dyn *)image->dynamic_ptr;
|
||||||
if(!d) {
|
if (!d)
|
||||||
return true;
|
return true;
|
||||||
}
|
|
||||||
|
|
||||||
for(i=0; d[i].d_tag != DT_NULL; i++) {
|
for (i = 0; d[i].d_tag != DT_NULL; i++) {
|
||||||
switch(d[i].d_tag) {
|
switch (d[i].d_tag) {
|
||||||
case DT_NEEDED:
|
case DT_NEEDED:
|
||||||
image->num_needed+= 1;
|
image->num_needed += 1;
|
||||||
break;
|
break;
|
||||||
case DT_HASH:
|
case DT_HASH:
|
||||||
image->symhash = (unsigned int *)(d[i].d_un.d_ptr + image->regions[0].delta);
|
image->symhash = (unsigned int *)(d[i].d_un.d_ptr + image->regions[0].delta);
|
||||||
@ -619,108 +594,106 @@ parse_dynamic_segment(image_t *image)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// lets make sure we found all the required sections
|
// lets make sure we found all the required sections
|
||||||
if(!image->symhash || !image->syms || !image->strtab) {
|
if (!image->symhash || !image->syms || !image->strtab)
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
struct Elf32_Sym *
|
static struct Elf32_Sym *
|
||||||
find_symbol_xxx(image_t *img, const char *name)
|
find_symbol_xxx(image_t *img, const char *name)
|
||||||
{
|
{
|
||||||
unsigned int hash;
|
unsigned int hash;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
if(img->dynamic_ptr) {
|
if (img->dynamic_ptr == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
hash = elf_hash(name) % HASHTABSIZE(img);
|
hash = elf_hash(name) % HASHTABSIZE(img);
|
||||||
for(i = HASHBUCKETS(img)[hash]; i != STN_UNDEF; i = HASHCHAINS(img)[i]) {
|
|
||||||
if(img->syms[i].st_shndx!= SHN_UNDEF) {
|
for (i = HASHBUCKETS(img)[hash]; i != STN_UNDEF; i = HASHCHAINS(img)[i]) {
|
||||||
if((ELF32_ST_BIND(img->syms[i].st_info)== STB_GLOBAL) || (ELF32_ST_BIND(img->syms[i].st_info)== STB_WEAK)) {
|
if (img->syms[i].st_shndx != SHN_UNDEF
|
||||||
if(!strcmp(SYMNAME(img, &img->syms[i]), name)) {
|
&& ((ELF32_ST_BIND(img->syms[i].st_info)== STB_GLOBAL) || (ELF32_ST_BIND(img->syms[i].st_info) == STB_WEAK))
|
||||||
|
&& !strcmp(SYMNAME(img, &img->syms[i]), name))
|
||||||
return &img->syms[i];
|
return &img->syms[i];
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
struct Elf32_Sym *
|
static struct Elf32_Sym *
|
||||||
find_symbol(image_t **shimg, const char *name)
|
find_symbol(image_t **shimg, const char *name)
|
||||||
{
|
{
|
||||||
image_t *iter;
|
image_t *iter;
|
||||||
unsigned int hash;
|
unsigned int hash;
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
iter= loaded_images.head;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
if (iter->dynamic_ptr == NULL)
|
||||||
if(iter->dynamic_ptr) {
|
continue;
|
||||||
|
|
||||||
hash = elf_hash(name) % HASHTABSIZE(iter);
|
hash = elf_hash(name) % HASHTABSIZE(iter);
|
||||||
|
|
||||||
for(i = HASHBUCKETS(iter)[hash]; i != STN_UNDEF; i = HASHCHAINS(iter)[i]) {
|
for(i = HASHBUCKETS(iter)[hash]; i != STN_UNDEF; i = HASHCHAINS(iter)[i]) {
|
||||||
if(iter->syms[i].st_shndx!= SHN_UNDEF) {
|
if (iter->syms[i].st_shndx!= SHN_UNDEF
|
||||||
if((ELF32_ST_BIND(iter->syms[i].st_info)== STB_GLOBAL) || (ELF32_ST_BIND(iter->syms[i].st_info)== STB_WEAK)) {
|
&& ((ELF32_ST_BIND(iter->syms[i].st_info)== STB_GLOBAL) || (ELF32_ST_BIND(iter->syms[i].st_info)== STB_WEAK))
|
||||||
if(!strcmp(SYMNAME(iter, &iter->syms[i]), name)) {
|
&& !strcmp(SYMNAME(iter, &iter->syms[i]), name)) {
|
||||||
*shimg= iter;
|
*shimg = iter;
|
||||||
return &iter->syms[i];
|
return &iter->syms[i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
iter= iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
int
|
static int
|
||||||
resolve_symbol(image_t *image, struct Elf32_Sym *sym, addr *sym_addr)
|
resolve_symbol(image_t *image, struct Elf32_Sym *sym, addr *sym_addr)
|
||||||
{
|
{
|
||||||
struct Elf32_Sym *sym2;
|
struct Elf32_Sym *sym2;
|
||||||
char *symname;
|
char *symname;
|
||||||
image_t *shimg;
|
image_t *shimg;
|
||||||
|
|
||||||
switch(sym->st_shndx) {
|
switch (sym->st_shndx) {
|
||||||
case SHN_UNDEF:
|
case SHN_UNDEF:
|
||||||
// patch the symbol name
|
// patch the symbol name
|
||||||
symname= SYMNAME(image, sym);
|
symname= SYMNAME(image, sym);
|
||||||
|
|
||||||
// it's undefined, must be outside this image, try the other image
|
// it's undefined, must be outside this image, try the other image
|
||||||
sym2 = find_symbol(&shimg, symname);
|
sym2 = find_symbol(&shimg, symname);
|
||||||
if(!sym2) {
|
if (!sym2) {
|
||||||
printf("elf_resolve_symbol: could not resolve symbol '%s'\n", symname);
|
printf("elf_resolve_symbol: could not resolve symbol '%s'\n", symname);
|
||||||
return ERR_ELF_RESOLVING_SYMBOL;
|
return ERR_ELF_RESOLVING_SYMBOL;
|
||||||
}
|
}
|
||||||
|
|
||||||
// make sure they're the same type
|
// make sure they're the same type
|
||||||
if(ELF32_ST_TYPE(sym->st_info)!= STT_NOTYPE) {
|
if (ELF32_ST_TYPE(sym->st_info) != STT_NOTYPE
|
||||||
if(ELF32_ST_TYPE(sym->st_info) != ELF32_ST_TYPE(sym2->st_info)) {
|
&& ELF32_ST_TYPE(sym->st_info) != ELF32_ST_TYPE(sym2->st_info)) {
|
||||||
printf("elf_resolve_symbol: found symbol '%s' in shared image but wrong type\n", symname);
|
printf("elf_resolve_symbol: found symbol '%s' in shared image but wrong type\n", symname);
|
||||||
return ERR_ELF_RESOLVING_SYMBOL;
|
return ERR_ELF_RESOLVING_SYMBOL;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if(ELF32_ST_BIND(sym2->st_info) != STB_GLOBAL && ELF32_ST_BIND(sym2->st_info) != STB_WEAK) {
|
if (ELF32_ST_BIND(sym2->st_info) != STB_GLOBAL
|
||||||
|
&& ELF32_ST_BIND(sym2->st_info) != STB_WEAK) {
|
||||||
printf("elf_resolve_symbol: found symbol '%s' but not exported\n", symname);
|
printf("elf_resolve_symbol: found symbol '%s' but not exported\n", symname);
|
||||||
return ERR_ELF_RESOLVING_SYMBOL;
|
return ERR_ELF_RESOLVING_SYMBOL;
|
||||||
}
|
}
|
||||||
|
|
||||||
*sym_addr = sym2->st_value + shimg->regions[0].delta;
|
*sym_addr = sym2->st_value + shimg->regions[0].delta;
|
||||||
return B_NO_ERROR;
|
return B_NO_ERROR;
|
||||||
|
|
||||||
case SHN_ABS:
|
case SHN_ABS:
|
||||||
*sym_addr = sym->st_value + image->regions[0].delta;
|
*sym_addr = sym->st_value + image->regions[0].delta;
|
||||||
return B_NO_ERROR;
|
return B_NO_ERROR;
|
||||||
|
|
||||||
case SHN_COMMON:
|
case SHN_COMMON:
|
||||||
// XXX finish this
|
// XXX finish this
|
||||||
printf("elf_resolve_symbol: COMMON symbol, finish me!\n");
|
printf("elf_resolve_symbol: COMMON symbol, finish me!\n");
|
||||||
return ERR_NOT_IMPLEMENTED_YET;
|
return ERR_NOT_IMPLEMENTED_YET;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
// standard symbol
|
// standard symbol
|
||||||
*sym_addr = sym->st_value + image->regions[0].delta;
|
*sym_addr = sym->st_value + image->regions[0].delta;
|
||||||
@ -732,8 +705,7 @@ resolve_symbol(image_t *image, struct Elf32_Sym *sym, addr *sym_addr)
|
|||||||
#include "arch/rldreloc.inc"
|
#include "arch/rldreloc.inc"
|
||||||
|
|
||||||
|
|
||||||
static
|
static image_t *
|
||||||
image_t *
|
|
||||||
load_container(char const *path, char const *name, bool fixed)
|
load_container(char const *path, char const *name, bool fixed)
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
@ -792,8 +764,7 @@ load_container(char const *path, char const *name, bool fixed)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static
|
static void
|
||||||
void
|
|
||||||
load_dependencies(image_t *img)
|
load_dependencies(image_t *img)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
@ -811,7 +782,7 @@ load_dependencies(image_t *img)
|
|||||||
FATAL((!img->needed), "failed to allocate needed struct\n");
|
FATAL((!img->needed), "failed to allocate needed struct\n");
|
||||||
memset(img->needed, 0, img->num_needed*sizeof(image_t*));
|
memset(img->needed, 0, img->num_needed*sizeof(image_t*));
|
||||||
|
|
||||||
for(i=0, j= 0; d[i].d_tag != DT_NULL; i++) {
|
for (i = 0, j = 0; d[i].d_tag != DT_NULL; i++) {
|
||||||
switch(d[i].d_tag) {
|
switch(d[i].d_tag) {
|
||||||
case DT_NEEDED:
|
case DT_NEEDED:
|
||||||
needed_offset = d[i].d_un.d_ptr;
|
needed_offset = d[i].d_un.d_ptr;
|
||||||
@ -833,8 +804,8 @@ load_dependencies(image_t *img)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
unsigned
|
static unsigned
|
||||||
topological_sort(image_t *img, unsigned slot, image_t **init_list)
|
topological_sort(image_t *img, unsigned slot, image_t **init_list)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
@ -850,56 +821,53 @@ topological_sort(image_t *img, unsigned slot, image_t **init_list)
|
|||||||
return slot+1;
|
return slot+1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
|
||||||
void
|
static void
|
||||||
init_dependencies(image_t *img, bool init_head)
|
init_dependencies(image_t *img, bool init_head)
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
unsigned slot;
|
unsigned slot;
|
||||||
image_t **init_list;
|
image_t **init_list;
|
||||||
|
|
||||||
init_list= rldalloc(loaded_image_count*sizeof(image_t*));
|
init_list = rldalloc(loaded_image_count*sizeof(image_t*));
|
||||||
FATAL((!init_list), "memory shortage in init_dependencies()");
|
FATAL((!init_list), "memory shortage in init_dependencies()");
|
||||||
memset(init_list, 0, loaded_image_count*sizeof(image_t*));
|
memset(init_list, 0, loaded_image_count*sizeof(image_t*));
|
||||||
|
|
||||||
img->flags|= RFLAG_SORTED; /* make sure we don't visit this one */
|
img->flags |= RFLAG_SORTED; /* make sure we don't visit this one */
|
||||||
slot= 0;
|
slot = 0;
|
||||||
for(i= 0; i< img->num_needed; i++) {
|
for (i = 0; i < img->num_needed; i++) {
|
||||||
if(!(img->needed[i]->flags & RFLAG_SORTED)) {
|
if (!(img->needed[i]->flags & RFLAG_SORTED))
|
||||||
slot= topological_sort(img->needed[i], slot, init_list);
|
slot = topological_sort(img->needed[i], slot, init_list);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(init_head) {
|
if (init_head) {
|
||||||
init_list[slot]= img;
|
init_list[slot] = img;
|
||||||
slot+= 1;
|
slot += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
for(i= 0; i< slot; i++) {
|
for (i = 0; i < slot; i++) {
|
||||||
addr _initf= init_list[i]->entry_point;
|
addr _initf = init_list[i]->entry_point;
|
||||||
libinit_f *initf= (libinit_f *)(_initf);
|
libinit_f *initf = (libinit_f *)(_initf);
|
||||||
|
|
||||||
if(initf) {
|
if (initf)
|
||||||
initf(init_list[i]->imageid, uspa);
|
initf(init_list[i]->imageid, uspa);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
rldfree(init_list);
|
rldfree(init_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static
|
static void
|
||||||
void
|
|
||||||
put_image(image_t *img)
|
put_image(image_t *img)
|
||||||
{
|
{
|
||||||
img->refcount-= 1;
|
img->refcount -= 1;
|
||||||
if(img->refcount== 0) {
|
if (img->refcount == 0) {
|
||||||
size_t i;
|
size_t i;
|
||||||
|
|
||||||
dequeue_image(&loaded_images, img);
|
dequeue_image(&loaded_images, img);
|
||||||
enqueue_image(&disposable_images, img);
|
enqueue_image(&disposable_images, img);
|
||||||
|
|
||||||
for(i= 0; i< img->num_needed; i++) {
|
for(i = 0; i < img->num_needed; i++) {
|
||||||
put_image(img->needed[i]);
|
put_image(img->needed[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -917,75 +885,64 @@ put_image(image_t *img)
|
|||||||
* + unload_addon()
|
* + unload_addon()
|
||||||
* + dynamic_symbol()
|
* + dynamic_symbol()
|
||||||
*/
|
*/
|
||||||
|
|
||||||
dynmodule_id
|
dynmodule_id
|
||||||
load_program(char const *path, void **entry)
|
load_program(char const *path, void **entry)
|
||||||
{
|
{
|
||||||
image_t *image;
|
image_t *image;
|
||||||
image_t *iter;
|
image_t *iter;
|
||||||
|
|
||||||
|
|
||||||
image = load_container(path, NEWOS_MAGIC_APPNAME, true);
|
image = load_container(path, NEWOS_MAGIC_APPNAME, true);
|
||||||
|
|
||||||
iter= loaded_images.head;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
|
||||||
load_dependencies(iter);
|
load_dependencies(iter);
|
||||||
|
}
|
||||||
|
|
||||||
iter= iter->next;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
};
|
|
||||||
|
|
||||||
iter= loaded_images.head;
|
|
||||||
while(iter) {
|
|
||||||
bool relocate_success;
|
bool relocate_success;
|
||||||
|
|
||||||
relocate_success= relocate_image(iter);
|
relocate_success = relocate_image(iter);
|
||||||
FATAL(!relocate_success, "troubles relocating\n");
|
FATAL(!relocate_success, "troubles relocating\n");
|
||||||
|
}
|
||||||
iter= iter->next;
|
|
||||||
};
|
|
||||||
|
|
||||||
init_dependencies(loaded_images.head, false);
|
init_dependencies(loaded_images.head, false);
|
||||||
|
|
||||||
*entry= (void*)(image->entry_point);
|
*entry = (void*)(image->entry_point);
|
||||||
return image->imageid;
|
return image->imageid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
dynmodule_id
|
dynmodule_id
|
||||||
load_library(char const *path)
|
load_library(char const *path)
|
||||||
{
|
{
|
||||||
image_t *image;
|
image_t *image;
|
||||||
image_t *iter;
|
image_t *iter;
|
||||||
|
|
||||||
|
|
||||||
image = find_image(path);
|
image = find_image(path);
|
||||||
if(image) {
|
if (image) {
|
||||||
image->refcount+= 1;
|
image->refcount += 1;
|
||||||
return image->imageid;
|
return image->imageid;
|
||||||
}
|
}
|
||||||
|
|
||||||
image = load_container(path, path, false);
|
image = load_container(path, path, false);
|
||||||
|
|
||||||
iter= loaded_images.head;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
|
||||||
load_dependencies(iter);
|
load_dependencies(iter);
|
||||||
|
}
|
||||||
|
|
||||||
iter= iter->next;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
};
|
|
||||||
|
|
||||||
iter= loaded_images.head;
|
|
||||||
while(iter) {
|
|
||||||
bool relocate_success;
|
bool relocate_success;
|
||||||
|
|
||||||
relocate_success= relocate_image(iter);
|
relocate_success = relocate_image(iter);
|
||||||
FATAL(!relocate_success, "troubles relocating\n");
|
FATAL(!relocate_success, "troubles relocating\n");
|
||||||
|
}
|
||||||
iter= iter->next;
|
|
||||||
};
|
|
||||||
|
|
||||||
init_dependencies(image, true);
|
init_dependencies(image, true);
|
||||||
|
|
||||||
return image->imageid;
|
return image->imageid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
dynmodule_id
|
dynmodule_id
|
||||||
unload_library(dynmodule_id imid)
|
unload_library(dynmodule_id imid)
|
||||||
{
|
{
|
||||||
@ -995,9 +952,9 @@ unload_library(dynmodule_id imid)
|
|||||||
/*
|
/*
|
||||||
* we only check images that have been already initialized
|
* we only check images that have been already initialized
|
||||||
*/
|
*/
|
||||||
iter= loaded_images.head;
|
|
||||||
while(iter) {
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
if(iter->imageid== imid) {
|
if (iter->imageid == imid) {
|
||||||
/*
|
/*
|
||||||
* do the unloading
|
* do the unloading
|
||||||
*/
|
*/
|
||||||
@ -1005,31 +962,23 @@ unload_library(dynmodule_id imid)
|
|||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
iter= iter->next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(iter) {
|
retval = iter ? 0 : -1;
|
||||||
retval= 0;
|
|
||||||
} else {
|
|
||||||
retval= -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
iter= disposable_images.head;
|
while ((iter = disposable_images.head) != NULL) {
|
||||||
while(iter) {
|
|
||||||
// call image fini here...
|
// call image fini here...
|
||||||
|
|
||||||
dequeue_image(&disposable_images, iter);
|
dequeue_image(&disposable_images, iter);
|
||||||
unmap_image(iter);
|
unmap_image(iter);
|
||||||
|
|
||||||
destroy_image(iter);
|
destroy_image(iter);
|
||||||
iter= disposable_images.head;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void *
|
void *
|
||||||
dynamic_symbol(dynmodule_id imid, char const *symname)
|
dynamic_symbol(dynmodule_id imid, char const *symname)
|
||||||
{
|
{
|
||||||
@ -1038,29 +987,30 @@ dynamic_symbol(dynmodule_id imid, char const *symname)
|
|||||||
/*
|
/*
|
||||||
* we only check images that have been already initialized
|
* we only check images that have been already initialized
|
||||||
*/
|
*/
|
||||||
iter= loaded_images.head;
|
for (iter = loaded_images.head; iter; iter = iter->next) {
|
||||||
while(iter) {
|
struct Elf32_Sym *sym;
|
||||||
if(iter->imageid== imid) {
|
|
||||||
struct Elf32_Sym *sym= find_symbol_xxx(iter, symname);
|
|
||||||
|
|
||||||
if(sym) {
|
if (iter->imageid != imid)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
sym = find_symbol_xxx(iter, symname);
|
||||||
|
if (sym)
|
||||||
return (void*)(sym->st_value + iter->regions[0].delta);
|
return (void*)(sym->st_value + iter->regions[0].delta);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
iter= iter->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* init routine, just get hold of the uspa args
|
* init routine, just get hold of the uspa args
|
||||||
*/
|
*/
|
||||||
|
|
||||||
void
|
void
|
||||||
rldelf_init(struct uspace_prog_args_t const *_uspa)
|
rldelf_init(struct uspace_prog_args_t const *_uspa)
|
||||||
{
|
{
|
||||||
uspa= _uspa;
|
uspa = _uspa;
|
||||||
|
|
||||||
rld_sem= create_sem(1, "rld_lock\n");
|
rld_sem = create_sem(1, "rld_lock\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user