PR/14498: Wesley Chen: Support symbol loading on elf lkms.

This commit is contained in:
christos 2001-11-08 15:33:15 +00:00
parent e415081a60
commit 522d8e170c
3 changed files with 234 additions and 64 deletions

View File

@ -1,4 +1,4 @@
/* $NetBSD: elf.c,v 1.6 2001/06/19 00:40:57 fvdl Exp $ */
/* $NetBSD: elf.c,v 1.7 2001/11/08 15:33:15 christos Exp $ */
/*
* Copyright (c) 1998 Johan Danielsson <joda@pdc.kth.se>
@ -29,7 +29,7 @@
*/
#include <sys/cdefs.h>
__RCSID("$NetBSD: elf.c,v 1.6 2001/06/19 00:40:57 fvdl Exp $");
__RCSID("$NetBSD: elf.c,v 1.7 2001/11/08 15:33:15 christos Exp $");
#include <sys/param.h>
@ -53,6 +53,8 @@ __RCSID("$NetBSD: elf.c,v 1.6 2001/06/19 00:40:57 fvdl Exp $");
#include "modload.h"
char *strtab;
static void
read_section_header(int fd, Elf_Ehdr *ehdr, int num, Elf_Shdr *shdr)
{
@ -95,7 +97,7 @@ add_section(struct elf_section **head, struct elf_section *s)
/* make a linked list of all sections containing ALLOCatable data */
static void
read_sections(int fd, Elf_Ehdr *ehdr, char *strtab, struct elf_section **head)
read_sections(int fd, Elf_Ehdr *ehdr, char *shstrtab, struct elf_section **head)
{
int i;
Elf_Shdr shdr;
@ -104,13 +106,17 @@ read_sections(int fd, Elf_Ehdr *ehdr, char *strtab, struct elf_section **head)
for (i = 0; i < ehdr->e_shnum; i++) {
struct elf_section *s;
read_section_header(fd, ehdr, i, &shdr);
if ((shdr.sh_flags & SHF_ALLOC) == 0)
if (((shdr.sh_flags & SHF_ALLOC) == 0)
&& (shdr.sh_type != SHT_STRTAB)
&& (shdr.sh_type != SHT_SYMTAB)
&& (shdr.sh_type != SHT_DYNSYM)) {
/* skip non-ALLOC sections */
continue;
}
s = malloc(sizeof(*s));
if (s == NULL)
errx(1, "failed to allocate %lu bytes", (u_long)sizeof(*s));
s->name = strtab + shdr.sh_name;
s->name = shstrtab + shdr.sh_name;
s->type = shdr.sh_type;
s->addr = (void*)shdr.sh_addr;
s->offset = shdr.sh_offset;
@ -120,6 +126,38 @@ read_sections(int fd, Elf_Ehdr *ehdr, char *strtab, struct elf_section **head)
}
}
/* get the symbol table sections and free the rest of them */
static void
get_symtab(struct elf_section **symtab)
{
struct elf_section *head, *cur, *prev;
head = NULL;
prev = NULL;
cur = *symtab;
while (cur) {
if ((cur->type == SHT_SYMTAB) || (cur->type == SHT_DYNSYM)) {
if (head == NULL) {
head = cur;
}
if (prev != NULL) {
prev->next = cur;
}
prev = cur;
cur = cur->next;
} else {
struct elf_section *p = cur;
cur = cur->next;
p->next = NULL;
free(p);
}
}
if (prev) {
prev->next = NULL;
}
*symtab = head;
}
/* free a list of section headers */
static void
free_sections(struct elf_section *head)
@ -132,21 +170,47 @@ free_sections(struct elf_section *head)
}
}
/* read string table */
/* read section header's string table */
static char *
read_string_table(int fd, Elf_Ehdr *ehdr)
read_shstring_table(int fd, Elf_Ehdr *ehdr)
{
Elf_Shdr shdr;
char *strtab;
char *shstrtab;
read_section_header(fd, ehdr, ehdr->e_shstrndx, &shdr);
strtab = malloc(shdr.sh_size);
if (strtab == NULL)
shstrtab = malloc(shdr.sh_size);
if (shstrtab == NULL)
errx(1, "failed to allocate %lu bytes", (u_long)shdr.sh_size);
if (lseek(fd, shdr.sh_offset, SEEK_SET) < 0)
err(1, "lseek");
if (read(fd, strtab, shdr.sh_size) != shdr.sh_size)
if (read(fd, shstrtab, shdr.sh_size) != shdr.sh_size)
err(1, "read");
return strtab;
return shstrtab;
}
/* read string table */
static char *
read_string_table(int fd, struct elf_section *head, int *strtablen)
{
char *string_table;
while (head) {
if ((strcmp(head->name, ".strtab") == 0 )
&& (head->type == SHT_STRTAB)) {
string_table = malloc(head->size);
if (string_table == NULL)
errx(1, "failed to allocate %lu bytes",
(u_long)head->size);
if (lseek(fd, head->offset, SEEK_SET) < 0)
err(1, "lseek");
if (read(fd, string_table, head->size) != head->size)
err(1, "read");
*strtablen = head->size;
break;
} else {
head = head->next;
}
}
return string_table;
}
static int
@ -188,15 +252,19 @@ elf_mod_sizes(fd, modsize, strtablen, resrvp, sp)
Elf_Ehdr ehdr;
ssize_t off = 0;
size_t data_hole = 0;
char *strtab;
struct elf_section *head, *s;
char *shstrtab, *strtab;
struct elf_section *head, *s, *symtab;
if (read_elf_header(fd, &ehdr) < 0)
return -1;
strtab = read_string_table(fd, &ehdr);
read_sections(fd, &ehdr, strtab, &head);
shstrtab = read_shstring_table(fd, &ehdr);
read_sections(fd, &ehdr, shstrtab, &head);
for (s = head; s; s = s->next) {
if ((s->type == SHT_STRTAB) && (s->type == SHT_SYMTAB)
&& (s->type == SHT_DYNSYM)) {
continue;
}
if (debug)
fprintf(stderr,
"%s: addr = %p size = %#lx align = %#lx\n",
@ -218,18 +286,24 @@ elf_mod_sizes(fd, modsize, strtablen, resrvp, sp)
}
off -= data_hole;
/* cleanup */
free_sections(head);
free(strtab);
/* XXX round to pagesize? */
*modsize = ROUND(off, sysconf(_SC_PAGESIZE));
free(shstrtab);
/*
* XXX no symbol table support
*/
*strtablen = 0;
resrvp->sym_size = resrvp->sym_symsize = 0;
/* get string table length */
strtab = read_string_table(fd, head, strtablen);
free(strtab);
/* get symbol table sections */
get_symtab(&head);
symtab = head;
resrvp->sym_symsize = 0;
while (symtab) {
resrvp->sym_symsize += symtab->size;
symtab = symtab->next;
}
resrvp->sym_size = resrvp->sym_symsize + *strtablen;
free_sections(head);
return (0);
}
@ -280,7 +354,7 @@ elf_mod_load(int fd)
size_t zero_size = 0;
size_t b;
ssize_t n;
char *strtab;
char *shstrtab;
struct elf_section *head, *s;
char buf[10 * BUFSIZ];
void *addr = NULL;
@ -288,55 +362,127 @@ elf_mod_load(int fd)
if (read_elf_header(fd, &ehdr) < 0)
return NULL;
strtab = read_string_table(fd, &ehdr);
read_sections(fd, &ehdr, strtab, &head);
shstrtab = read_shstring_table(fd, &ehdr);
read_sections(fd, &ehdr, shstrtab, &head);
for (s = head; s; s = s->next) {
if (debug)
fprintf(stderr, "loading `%s': addr = %p, "
"size = %#lx\n",
s->name, s->addr, (u_long)s->size);
if (s->type == SHT_NOBITS)
/* skip some space */
zero_size += s->size;
else {
if (addr != NULL) {
/* if there is a gap in the prelinked
module, transfer some empty
space... */
zero_size += (char*)s->addr - (char*)addr;
if ((s->type != SHT_STRTAB) && (s->type != SHT_SYMTAB)
&& (s->type != SHT_DYNSYM)) {
if (debug)
fprintf(stderr, "loading `%s': addr = %p, "
"size = %#lx\n",
s->name, s->addr, (u_long)s->size);
if (s->type == SHT_NOBITS) {
/* skip some space */
zero_size += s->size;
} else {
if (addr != NULL) {
/* if there is a gap in the prelinked
module, transfer some empty
space... */
zero_size += (char*)s->addr
- (char*)addr;
}
if (zero_size) {
loadspace(zero_size);
zero_size = 0;
}
b = s->size;
if (lseek(fd, s->offset, SEEK_SET) == -1)
err(1, "lseek");
while (b) {
n = read(fd, buf, MIN(b, sizeof(buf)));
if (n == 0)
errx(1, "unexpected EOF");
if (n < 0)
err(1, "read");
loadbuf(buf, n);
b -= n;
}
addr = (char*)s->addr + s->size;
}
if (zero_size) {
loadspace(zero_size);
zero_size = 0;
}
b = s->size;
if (lseek(fd, s->offset, SEEK_SET) == -1)
err(1, "lseek");
while (b) {
n = read(fd, buf, MIN(b, sizeof(buf)));
if (n == 0)
errx(1, "unexpected EOF");
if (n < 0)
err(1, "read");
loadbuf(buf, n);
b -= n;
}
addr = (char*)s->addr + s->size;
}
}
if (zero_size)
loadspace(zero_size);
free_sections(head);
free(strtab);
free(shstrtab);
return (void*)ehdr.e_entry;
}
extern int devfd, modfd;
void
elf_mod_symload(strtablen)
int strtablen;
{
Elf_Ehdr ehdr;
char *shstrtab;
struct elf_section *head, *s;
char *symbuf, *strbuf;
warnx("ELF can not load symbols");
/*
* Seek to the text offset to start loading...
*/
if (lseek(modfd, 0, SEEK_SET) == -1)
err(12, "lseek");
if (read_elf_header(modfd, &ehdr) < 0)
return;
shstrtab = read_shstring_table(modfd, &ehdr);
read_sections(modfd, &ehdr, shstrtab, &head);
for (s = head; s; s = s->next) {
struct elf_section *p = s;
if ((p->type == SHT_SYMTAB) || (p->type == SHT_DYNSYM)) {
if (debug)
fprintf(stderr, "loading `%s': addr = %p, "
"size = %#lx\n",
s->name, s->addr, (u_long)s->size);
/*
* Seek to the file offset to start loading it...
*/
if (lseek(modfd, p->offset, SEEK_SET) == -1)
err(12, "lseek");
symbuf = malloc(p->size);
if (symbuf == 0)
err(13, "malloc");
if (read(modfd, symbuf, p->size) != p->size)
err(14, "read");
loadsym(symbuf, p->size);
free(symbuf);
}
}
for (s = head; s; s = s->next) {
struct elf_section *p = s;
if ((p->type == SHT_STRTAB)
&& (strcmp(p->name, ".strtab") == 0 )) {
if (debug)
fprintf(stderr, "loading `%s': addr = %p, "
"size = %#lx\n",
s->name, s->addr, (u_long)s->size);
/*
* Seek to the file offset to start loading it...
*/
if (lseek(modfd, p->offset, SEEK_SET) == -1)
err(12, "lseek");
strbuf = malloc(p->size);
if (strbuf == 0)
err(13, "malloc");
if (read(modfd, strbuf, p->size) != p->size)
err(14, "read");
loadsym(strbuf, p->size);
free(strbuf);
}
}
free(shstrtab);
free_sections(head);
return;
}

View File

@ -1,4 +1,4 @@
/* $NetBSD: modload.c,v 1.29 2001/09/29 21:15:11 jdolecek Exp $ */
/* $NetBSD: modload.c,v 1.30 2001/11/08 15:33:15 christos Exp $ */
/*
* Copyright (c) 1993 Terrence R. Lambert.
@ -34,7 +34,7 @@
#include <sys/cdefs.h>
#ifndef lint
__RCSID("$NetBSD: modload.c,v 1.29 2001/09/29 21:15:11 jdolecek Exp $");
__RCSID("$NetBSD: modload.c,v 1.30 2001/11/08 15:33:15 christos Exp $");
#endif /* not lint */
#include <sys/param.h>
@ -193,7 +193,7 @@ loadbuf(void *buf, size_t len)
n = MIN(len, MODIOBUF);
ldbuf.cnt = n;
ldbuf.data = p;
if(ioctl(devfd, LMLOADBUF, &ldbuf) == -1)
if (ioctl(devfd, LMLOADBUF, &ldbuf) == -1)
err(11, "error loading buffer");
len -= n;
p += n;
@ -214,6 +214,29 @@ loadspace(size_t len)
}
}
/*
* Transfer symbol table to kernel memory in chunks
* of MODIOBUF size at a time.
*/
void
loadsym(void *buf, size_t len)
{
struct lmc_loadbuf ldbuf;
size_t n;
char *p = buf;
while(len) {
n = MIN(len, MODIOBUF);
ldbuf.cnt = n;
ldbuf.data = p;
if(ioctl(devfd, LMLOADSYMS, &ldbuf) == -1)
err(11, "error loading buffer");
len -= n;
p += n;
}
}
/* Transfer some empty space. */
int
main(int argc, char **argv)
{

View File

@ -1,4 +1,4 @@
/* $NetBSD: modload.h,v 1.1 1999/06/13 12:54:40 mrg Exp $ */
/* $NetBSD: modload.h,v 1.2 2001/11/08 15:33:15 christos Exp $ */
/*
* Copyright (c) 1993 Terrence R. Lambert.
@ -63,6 +63,7 @@ void a_out_mod_symload __P((int));
void loadbuf(void*, size_t);
void loadspace(size_t);
void loadsym(void*, size_t);
extern int debug;
extern int verbose;