summaryrefslogtreecommitdiff
path: root/sys/ddb
diff options
context:
space:
mode:
Diffstat (limited to 'sys/ddb')
-rw-r--r--sys/ddb/db_ctf.c212
1 files changed, 61 insertions, 151 deletions
diff --git a/sys/ddb/db_ctf.c b/sys/ddb/db_ctf.c
index 80c1f75b4c5..3ed8174b5b0 100644
--- a/sys/ddb/db_ctf.c
+++ b/sys/ddb/db_ctf.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: db_ctf.c,v 1.5 2017/05/09 11:09:38 mpi Exp $ */
+/* $OpenBSD: db_ctf.c,v 1.6 2017/05/27 15:05:16 mpi Exp $ */
/*
* Copyright (c) 2016 Jasper Lievisse Adriaanse <jasper@openbsd.org>
@@ -39,31 +39,20 @@ extern db_symtab_t db_symtab;
struct ddb_ctf {
struct ctf_header *cth;
- const char *data;
- off_t dlen;
+ const char *rawctf; /* raw .SUNW_ctf section */
+ size_t rawctflen; /* raw .SUNW_ctf section size */
+ const char *data; /* decompressed CTF data */
+ size_t dlen; /* decompressed CTF data size */
+ char *strtab; /* ELF string table */
uint32_t ctf_found;
- uint32_t nsyms;
- size_t ctftab_size;
- const char *ctftab;
};
struct ddb_ctf db_ctf;
-/*
- * We need a way to get the number of symbols, so (ab)use db_elf_sym_forall()
- * to give us the count.
- */
-struct db_ctf_forall_arg {
- int cnt;
- db_sym_t sym;
-};
-
static const char *db_ctf_lookup_name(uint32_t);
static const char *db_ctf_idx2sym(size_t *, uint8_t);
-static const char *db_elf_find_ctftab(db_symtab_t *, size_t *);
+static const char *db_elf_find_ctf(db_symtab_t *, size_t *);
static char *db_ctf_decompress(const char *, size_t, off_t);
-static int db_ctf_print_functions();
-static void db_ctf_forall(db_sym_t, char *, char *, int, void *);
/*
* Entrypoint to verify CTF presence, initialize the header, decompress
@@ -72,70 +61,47 @@ static void db_ctf_forall(db_sym_t, char *, char *, int, void *);
void
db_ctf_init(void)
{
- struct db_ctf_forall_arg dfa;
db_symtab_t *stab = &db_symtab;
- const char *ctftab;
- size_t ctftab_size;
- int nsyms;
+ size_t rawctflen;
/* Assume nothing was correct found until proven otherwise. */
db_ctf.ctf_found = 0;
- ctftab = db_elf_find_ctftab(stab, &ctftab_size);
- if (ctftab == NULL) {
+ if (stab->private == NULL)
return;
- }
- db_ctf.ctftab = ctftab;
- db_ctf.ctftab_size = ctftab_size;
- db_ctf.cth = (struct ctf_header *)ctftab;
+ db_ctf.strtab = db_elf_find_strtab(stab);
+ if (db_ctf.strtab == NULL)
+ return;
+
+ db_ctf.rawctf = db_elf_find_ctf(stab, &rawctflen);
+ if (db_ctf.rawctf == NULL)
+ return;
+
+ db_ctf.rawctflen = rawctflen;
+ db_ctf.cth = (struct ctf_header *)db_ctf.rawctf;
db_ctf.dlen = db_ctf.cth->cth_stroff + db_ctf.cth->cth_strlen;
- /* Now decompress the section, take into account to skip the header */
- if (db_ctf.cth->cth_flags & CTF_F_COMPRESS) {
- if ((db_ctf.data =
- db_ctf_decompress(db_ctf.ctftab + sizeof(*db_ctf.cth),
- db_ctf.ctftab_size - sizeof(*db_ctf.cth),
- db_ctf.dlen)) == NULL)
- return;
- } else {
- db_printf("Unsupported non-compressed CTF section encountered\n");
+ if ((db_ctf.cth->cth_flags & CTF_F_COMPRESS) == 0) {
+ printf("unsupported non-compressed CTF section\n");
return;
}
- /*
- * Lookup the total number of kernel symbols. It's unlikely for the
- * kernel to have zero symbols so bail out if that's what we end
- * up finding.
- */
- dfa.cnt = 0;
- db_elf_sym_forall(db_ctf_forall, &dfa);
- nsyms = -dfa.cnt;
-
- if (nsyms == 0)
+ /* Now decompress the section, take into account to skip the header */
+ db_ctf.data = db_ctf_decompress(db_ctf.rawctf + sizeof(*db_ctf.cth),
+ db_ctf.rawctflen - sizeof(*db_ctf.cth), db_ctf.dlen);
+ if (db_ctf.data == NULL)
return;
- else
- db_ctf.nsyms = nsyms;
/* We made it this far, everything seems fine. */
db_ctf.ctf_found = 1;
}
-static void
-db_ctf_forall(db_sym_t sym, char *name, char *suff, int pre, void *varg)
-{
- struct db_ctf_forall_arg *arg = varg;
-
- if (arg->cnt-- == 0)
- arg->sym = sym;
-}
-
/*
- * Internal helper function - return a pointer to the CTF table
- * for the current symbol table (and update the size).
+ * Internal helper function - return a pointer to the CTF section
*/
static const char *
-db_elf_find_ctftab(db_symtab_t *stab, size_t *size)
+db_elf_find_ctf(db_symtab_t *stab, size_t *size)
{
Elf_Ehdr *elf = STAB_TO_EHDR(stab);
Elf_Shdr *shp = STAB_TO_SHDR(stab, elf);
@@ -161,8 +127,8 @@ db_dump_ctf_header(void)
if (!db_ctf.ctf_found)
return;
- db_printf("CTF header found at %p (%ld)\n", db_ctf.ctftab,
- db_ctf.ctftab_size);
+ db_printf("CTF header found at %p (%ld)\n", db_ctf.rawctf,
+ db_ctf.rawctflen);
db_printf("cth_magic: 0x%04x\n", db_ctf.cth->cth_magic);
db_printf("cth_verion: %d\n", db_ctf.cth->cth_version);
db_printf("cth_flags: 0x%02x", db_ctf.cth->cth_flags);
@@ -180,10 +146,6 @@ db_dump_ctf_header(void)
db_printf("cth_typeoff: %d\n", db_ctf.cth->cth_typeoff);
db_printf("cth_stroff: %d\n", db_ctf.cth->cth_stroff);
db_printf("cth_strlen: %d\n", db_ctf.cth->cth_strlen);
-
-#if 1
- db_ctf_print_functions();
-#endif
}
/*
@@ -194,33 +156,21 @@ db_dump_ctf_header(void)
static const char *
db_ctf_idx2sym(size_t *idx, uint8_t type)
{
- db_symtab_t *stab = &db_symtab;
- Elf_Sym *symp, *symtab_start;
- const Elf_Sym *st;
- char *strtab;
- size_t i;
-
- if (stab->private == NULL)
- return (NULL);
+ Elf_Sym *symp, *symtab_start, *symtab_end;
+ size_t i = *idx + 1;
- strtab = db_elf_find_strtab(stab);
- if (strtab == NULL)
- return (NULL);
+ symtab_start = STAB_TO_SYMSTART(&db_symtab);
+ symtab_end = STAB_TO_SYMEND(&db_symtab);
- symtab_start = STAB_TO_SYMSTART(stab);
- symp = symtab_start;
-
- for (i = *idx + 1; i < db_ctf.nsyms; i++) {
- st = &symp[i];
-
- if (ELF_ST_TYPE(st->st_info) != type)
+ for (symp = &symtab_start[i]; symp < symtab_end; i++, symp++) {
+ if (ELF_ST_TYPE(symp->st_info) != type)
continue;
*idx = i;
- return strtab + st->st_name;
+ return db_ctf.strtab + symp->st_name;
}
- return (NULL);
+ return NULL;
}
/*
@@ -229,79 +179,41 @@ db_ctf_idx2sym(size_t *idx, uint8_t type)
int
db_ctf_func_numargs(const char *funcname)
{
- const char *s;
+ uint16_t *fstart, *fend;
+ const char *fname;
uint16_t *fsp, kind, vlen;
- size_t idx = 0;
- int nargs;
+ size_t i, idx = 0;
if (!db_ctf.ctf_found)
- return (0);
+ return 0;
+
+ fstart = (uint16_t *)(db_ctf.data + db_ctf.cth->cth_funcoff);
+ fend = (uint16_t *)(db_ctf.data + db_ctf.cth->cth_typeoff);
+
+ fsp = fstart;
+ while (fsp < fend) {
+ fname = db_ctf_idx2sym(&idx, STT_FUNC);
+ if (fname == NULL)
+ break;
- fsp = (uint16_t *)(db_ctf.data + db_ctf.cth->cth_funcoff);
- while (fsp < (uint16_t *)(db_ctf.data + db_ctf.cth->cth_typeoff)) {
kind = CTF_INFO_KIND(*fsp);
vlen = CTF_INFO_VLEN(*fsp);
- s = db_ctf_idx2sym(&idx, STT_FUNC);
fsp++;
if (kind == CTF_K_UNKNOWN && vlen == 0)
continue;
- nargs = 0;
- if (s != NULL) {
- /*
- * We have to keep increasing fsp++ while walking the
- * table even if we discard the value at that location.
- * This is required to keep a moving index.
- *
- * First increment for the return type, then for each
- * parameter type.
- */
- fsp++;
-
- while (vlen-- > 0) {
- nargs++;
- fsp++;
- }
-
- if (strncmp(funcname, s, strlen(funcname)) == 0) {
- return (nargs);
- }
- }
- }
-
- return (0);
-}
-
-static int
-db_ctf_print_functions(void)
-{
- uint16_t *fsp, kind, vlen;
- size_t idx = 0, i = 0;
- const char *s;
- int l;
-
- if (!db_ctf.ctf_found)
- return 1;
-
- fsp = (uint16_t *)(db_ctf.data + db_ctf.cth->cth_funcoff);
- while (fsp < (uint16_t *)(db_ctf.data + db_ctf.cth->cth_typeoff)) {
- kind = CTF_INFO_KIND(*fsp);
- vlen = CTF_INFO_VLEN(*fsp);
+ /* Skip return type */
fsp++;
- if (kind == CTF_K_UNKNOWN && vlen == 0)
- continue;
+ /* Skip argument types */
+ for (i = 0; i < vlen; i++)
+ fsp++;
- l = db_printf(" [%zu] FUNC ", i++);
- if ((s = db_ctf_idx2sym(&idx, STT_FUNC)) != NULL)
- db_printf("(%s)", s);
- db_printf(" returns: %u args: (", *fsp++);
- while (vlen-- > 0)
- db_printf("%u%s", *fsp++, (vlen > 0) ? ", " : "");
- db_printf(") idx: %zu\n", idx);
+ if (strcmp(funcname, fname) == 0)
+ return vlen;
}
- db_printf("\n");
+
return 0;
}
@@ -333,11 +245,9 @@ db_ctf_decompress(const char *buf, size_t size, off_t len)
char *data;
int error;
- /* XXX: drop malloc(9) usage */
data = malloc(len, M_TEMP, M_WAITOK|M_ZERO|M_CANFAIL);
- if (data == NULL) {
- return (NULL);
- }
+ if (data == NULL)
+ return NULL;
memset(&stream, 0, sizeof(stream));
stream.next_in = (void *)buf;
@@ -367,9 +277,9 @@ db_ctf_decompress(const char *buf, size_t size, off_t len)
goto exit;
}
- return (data);
+ return data;
exit:
free(data, M_DEVBUF, sizeof(*data));
- return (NULL);
+ return NULL;
}