600 lines
14 KiB
C
600 lines
14 KiB
C
/*
|
|
* Mach Operating System
|
|
* Copyright (c) 1991,1990 Carnegie Mellon University
|
|
* All Rights Reserved.
|
|
*
|
|
* Permission to use, copy, modify and distribute this software and its
|
|
* documentation is hereby granted, provided that both the copyright
|
|
* notice and this permission notice appear in all copies of the
|
|
* software, derivative works or modified versions, and any portions
|
|
* thereof, and that both notices appear in supporting documentation.
|
|
*
|
|
* CARNEGIE MELLON ALLOWS FREE USE OF THIS SOFTWARE IN ITS
|
|
* CONDITION. CARNEGIE MELLON DISCLAIMS ANY LIABILITY OF ANY KIND FOR
|
|
* ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF THIS SOFTWARE.
|
|
*
|
|
* Carnegie Mellon requests users of this software to return to
|
|
*
|
|
* Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
|
|
* School of Computer Science
|
|
* Carnegie Mellon University
|
|
* Pittsburgh PA 15213-3890
|
|
*
|
|
* any improvements or extensions that they make and grant Carnegie the
|
|
* rights to redistribute these changes.
|
|
*
|
|
* $Id: db_sym.c,v 1.2 2003/01/16 01:06:09 mikesw Exp $
|
|
*/
|
|
|
|
/*
|
|
* Author: David B. Golub, Carnegie Mellon University
|
|
* Date: 7/90
|
|
*/
|
|
#if 0
|
|
//#include <sys/param.h>
|
|
//#include <sys/systm.h>
|
|
#endif
|
|
#if 0
|
|
#include <linux/kernel.h>
|
|
#include <linux/string.h>
|
|
#include <linux/kallsyms.h>
|
|
#endif
|
|
#include "ddb.h"
|
|
#include "db_sym.h"
|
|
#include "swifi.h"
|
|
|
|
#include "extra.h"
|
|
|
|
/*
|
|
* Multiple symbol tables
|
|
*/
|
|
#ifndef MAXNOSYMTABS
|
|
#define MAXNOSYMTABS 3 /* mach, ux, emulator */
|
|
#endif
|
|
#if 0
|
|
|
|
static db_symtab_t db_symtabs[MAXNOSYMTABS] = {{0,},};
|
|
static int db_nsymtab = 0;
|
|
|
|
static db_symtab_t *db_last_symtab;
|
|
|
|
static db_sym_t db_lookup __P(( char *symstr));
|
|
static char *db_qualify __P((db_sym_t sym, char *symtabname));
|
|
static boolean_t db_symbol_is_ambiguous __P((db_sym_t sym));
|
|
static boolean_t db_line_at_pc __P((db_sym_t, char **, int *,
|
|
db_expr_t));
|
|
|
|
/*
|
|
* Add symbol table, with given name, to list of symbol tables.
|
|
*/
|
|
void
|
|
db_add_symbol_table(start, end, name, ref)
|
|
char *start;
|
|
char *end;
|
|
char *name;
|
|
char *ref;
|
|
{
|
|
if (db_nsymtab >= MAXNOSYMTABS) {
|
|
printk ("No slots left for %s symbol table", name);
|
|
panic ("db_sym.c: db_add_symbol_table");
|
|
}
|
|
|
|
db_symtabs[db_nsymtab].start = start;
|
|
db_symtabs[db_nsymtab].end = end;
|
|
db_symtabs[db_nsymtab].name = name;
|
|
db_symtabs[db_nsymtab].private = ref;
|
|
db_nsymtab++;
|
|
}
|
|
|
|
/*
|
|
* db_qualify("vm_map", "ux") returns "unix:vm_map".
|
|
*
|
|
* Note: return value points to static data whose content is
|
|
* overwritten by each call... but in practice this seems okay.
|
|
*/
|
|
static char *
|
|
db_qualify(sym, symtabname)
|
|
db_sym_t sym;
|
|
register char *symtabname;
|
|
{
|
|
char *symname;
|
|
static char tmp[256];
|
|
|
|
db_symbol_values(sym, &symname, 0);
|
|
strcpy(tmp,symtabname);
|
|
strcat(tmp,":");
|
|
strcat(tmp,symname);
|
|
return tmp;
|
|
}
|
|
|
|
|
|
boolean_t
|
|
db_eqname(src, dst, c)
|
|
char *src;
|
|
char *dst;
|
|
char c;
|
|
{
|
|
if (!strcmp(src, dst))
|
|
return (TRUE);
|
|
if (src[0] == c)
|
|
return (!strcmp(src+1,dst));
|
|
return (FALSE);
|
|
}
|
|
|
|
boolean_t
|
|
db_value_of_name(name, valuep)
|
|
char *name;
|
|
db_expr_t *valuep;
|
|
{
|
|
db_sym_t sym;
|
|
|
|
sym = db_lookup(name);
|
|
if (sym == DB_SYM_NULL)
|
|
return (FALSE);
|
|
db_symbol_values(sym, &name, valuep);
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
/*
|
|
* Lookup a symbol.
|
|
* If the symbol has a qualifier (e.g., ux:vm_map),
|
|
* then only the specified symbol table will be searched;
|
|
* otherwise, all symbol tables will be searched.
|
|
*/
|
|
static db_sym_t
|
|
db_lookup(symstr)
|
|
char *symstr;
|
|
{
|
|
db_sym_t sp;
|
|
register int i;
|
|
int symtab_start = 0;
|
|
int symtab_end = db_nsymtab;
|
|
register char *cp;
|
|
|
|
/*
|
|
* Look for, remove, and remember any symbol table specifier.
|
|
*/
|
|
for (cp = symstr; *cp; cp++) {
|
|
if (*cp == ':') {
|
|
*cp = '\0';
|
|
for (i = 0; i < db_nsymtab; i++) {
|
|
if (! strcmp(symstr, db_symtabs[i].name)) {
|
|
symtab_start = i;
|
|
symtab_end = i + 1;
|
|
break;
|
|
}
|
|
}
|
|
*cp = ':';
|
|
if (i == db_nsymtab) {
|
|
db_error("invalid symbol table name");
|
|
}
|
|
symstr = cp+1;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Look in the specified set of symbol tables.
|
|
* Return on first match.
|
|
*/
|
|
for (i = symtab_start; i < symtab_end; i++) {
|
|
sp = X_db_lookup(&db_symtabs[i], symstr);
|
|
if (sp) {
|
|
db_last_symtab = &db_symtabs[i];
|
|
return sp;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Does this symbol name appear in more than one symbol table?
|
|
* Used by db_symbol_values to decide whether to qualify a symbol.
|
|
*/
|
|
static boolean_t db_qualify_ambiguous_names = FALSE;
|
|
|
|
static boolean_t
|
|
db_symbol_is_ambiguous(sym)
|
|
db_sym_t sym;
|
|
{
|
|
char *sym_name;
|
|
register int i;
|
|
register
|
|
boolean_t found_once = FALSE;
|
|
|
|
if (!db_qualify_ambiguous_names)
|
|
return FALSE;
|
|
|
|
db_symbol_values(sym, &sym_name, 0);
|
|
for (i = 0; i < db_nsymtab; i++) {
|
|
if (X_db_lookup(&db_symtabs[i], sym_name)) {
|
|
if (found_once)
|
|
return TRUE;
|
|
found_once = TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* Find the closest symbol to val, and return its name
|
|
* and the difference between val and the symbol found.
|
|
*/
|
|
db_sym_t
|
|
db_search_symbol( val, strategy, offp)
|
|
register db_addr_t val;
|
|
db_strategy_t strategy;
|
|
db_expr_t *offp;
|
|
{
|
|
register
|
|
unsigned int diff;
|
|
unsigned int newdiff;
|
|
register int i;
|
|
db_sym_t ret = DB_SYM_NULL, sym;
|
|
|
|
newdiff = diff = ~0;
|
|
db_last_symtab = 0;
|
|
for (i = 0; i < db_nsymtab; i++) {
|
|
sym = X_db_search_symbol(&db_symtabs[i], val, strategy, &newdiff);
|
|
if (newdiff < diff) {
|
|
db_last_symtab = &db_symtabs[i];
|
|
diff = newdiff;
|
|
ret = sym;
|
|
}
|
|
}
|
|
*offp = diff;
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Return name and value of a symbol
|
|
*/
|
|
void
|
|
db_symbol_values(sym, namep, valuep)
|
|
db_sym_t sym;
|
|
char **namep;
|
|
db_expr_t *valuep;
|
|
{
|
|
db_expr_t value;
|
|
|
|
if (sym == DB_SYM_NULL) {
|
|
*namep = 0;
|
|
return;
|
|
}
|
|
|
|
X_db_symbol_values(sym, namep, &value);
|
|
if (db_symbol_is_ambiguous(sym))
|
|
*namep = db_qualify(sym, db_last_symtab->name);
|
|
if (valuep)
|
|
*valuep = value;
|
|
}
|
|
|
|
|
|
/*
|
|
* Print a the closest symbol to value
|
|
*
|
|
* After matching the symbol according to the given strategy
|
|
* we print it in the name+offset format, provided the symbol's
|
|
* value is close enough (eg smaller than db_maxoff).
|
|
* We also attempt to print [filename:linenum] when applicable
|
|
* (eg for procedure names).
|
|
*
|
|
* If we could not find a reasonable name+offset representation,
|
|
* then we just print the value in hex. Small values might get
|
|
* bogus symbol associations, e.g. 3 might get some absolute
|
|
* value like _INCLUDE_VERSION or something, therefore we do
|
|
* not accept symbols whose value is "small" (and use plain hex).
|
|
*/
|
|
|
|
|
|
void
|
|
db_printsym(off, strategy)
|
|
db_expr_t off;
|
|
db_strategy_t strategy;
|
|
{
|
|
db_expr_t d;
|
|
char *filename;
|
|
char *name;
|
|
db_expr_t value;
|
|
int linenum;
|
|
db_sym_t cursym;
|
|
|
|
cursym = db_search_symbol(off, strategy, &d);
|
|
db_symbol_values(cursym, &name, &value);
|
|
if (name == 0)
|
|
value = off;
|
|
if (value >= DB_SMALL_VALUE_MIN && value <= DB_SMALL_VALUE_MAX) {
|
|
printk("0x%x", off);
|
|
return;
|
|
}
|
|
if (name == 0 || d >= db_maxoff) {
|
|
printk("0x%x", off);
|
|
return;
|
|
}
|
|
printk("%s", name);
|
|
if (d)
|
|
printk("+0x%x", d);
|
|
if (strategy == DB_STGY_PROC) {
|
|
// if (db_line_at_pc(cursym, &filename, &linenum, off))
|
|
// printk(" [%s:%d]", filename, linenum);
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
unsigned int db_maxoff = 0x10000;
|
|
unsigned long modAddr = 0;
|
|
|
|
/* NWT: fault injection routine only.
|
|
* figure out start of function address given an address (off) in kernel text.
|
|
* name = function name
|
|
* value = function address
|
|
* d = difference between off and function address
|
|
* input is the desired address off and fault type
|
|
* returns closest instruction address (if found), NULL otherwise
|
|
*/
|
|
unsigned long
|
|
find_faulty_instr(db_expr_t off, int type, int *instr_len)
|
|
{
|
|
db_expr_t d;
|
|
char *name;
|
|
db_expr_t value, cur_value, prev_value = 0;
|
|
int verbose=0, found=0;
|
|
const char * mod_name = NULL;
|
|
unsigned long mod_start;
|
|
unsigned long mod_end;
|
|
const char * sec_name = NULL;
|
|
unsigned long sec_start;
|
|
unsigned long sec_end;
|
|
const char * sym_name = NULL;
|
|
unsigned long sym_start;
|
|
unsigned long sym_end;
|
|
|
|
|
|
*instr_len = 0;
|
|
if (kallsyms_address_to_symbol(off,
|
|
&mod_name, &mod_start, &mod_end,
|
|
&sec_name, &sec_start, &sec_end,
|
|
&sym_name, &sym_start, &sym_end) == 0) {
|
|
return(0);
|
|
}
|
|
|
|
value = (db_expr_t) sym_start;
|
|
d = off - sym_start;
|
|
name = (char *) sym_name;
|
|
|
|
if (name == 0) {
|
|
value = off;
|
|
}
|
|
|
|
if (value >= DB_SMALL_VALUE_MIN && value <= DB_SMALL_VALUE_MAX) {
|
|
printk("0x%x", off);
|
|
return 0;
|
|
}
|
|
|
|
if (name == 0 || d >= db_maxoff) {
|
|
printk("0x%x", off);
|
|
return 0 ;
|
|
}
|
|
/* 2) backup to start of function (SOF)
|
|
* 3) delineate instruction boundaries, find instruction length too.
|
|
*/
|
|
|
|
if(verbose) {
|
|
printk("function %s", sym_name);
|
|
}
|
|
|
|
/* 4) skip instructions until we get to our faulty address */
|
|
cur_value = value;
|
|
while(cur_value < sec_end) {
|
|
if(verbose) {
|
|
#if 0
|
|
// db_printsym(cur_value, DB_STGY_PROC);
|
|
// printk(":\t");
|
|
#endif
|
|
}
|
|
prev_value=cur_value;
|
|
modAddr=0;
|
|
if(verbose) {
|
|
#if 0
|
|
//cur_value=db_disasm(prev_value, FALSE);
|
|
#endif
|
|
} else {
|
|
cur_value=my_disasm(prev_value, FALSE);
|
|
}
|
|
|
|
/* 4a) bail out if instruction is leave (0xc9) */
|
|
if(cur_value-prev_value == 1) {
|
|
unsigned char *c;
|
|
c=(unsigned char *) prev_value;
|
|
if(text_read_ub(c)==0xc9) {
|
|
if(verbose) printk("bailing out as we hit a leave\n");
|
|
found=0;
|
|
break;
|
|
}
|
|
}
|
|
/* 5a) init fault: from SOF, look for movl $X, -Y(%ebp),
|
|
* (C645Fxxx or C745Fxxx) and replace with nop.
|
|
*/
|
|
if(type==INIT_FAULT) {
|
|
unsigned char *c;
|
|
c=(unsigned char *) prev_value;
|
|
|
|
if(*c==0x66 || *c==0x67)
|
|
c++; /* override prefix */
|
|
|
|
if(*c==0xC6 || *c==0xC7)
|
|
c++; /* movb or movl imm */
|
|
else
|
|
continue;
|
|
|
|
if(*c==0x45)
|
|
c++; /* [ebp] */
|
|
else
|
|
continue;
|
|
|
|
if(*c & 0x80)
|
|
found=1; /* negative displacement */
|
|
else
|
|
continue;
|
|
|
|
found=1;
|
|
break;
|
|
} else if(type==NOP_FAULT) {
|
|
/* 5b) nop*: replace instruction with nop */
|
|
if(cur_value> off) {
|
|
found=1;
|
|
break;
|
|
}
|
|
} else if(type==DST_FAULT || type==SRC_FAULT) {
|
|
/* 5c) dst/src: flip bits in mod/rm, sib, disp or imm fields */
|
|
if(cur_value>off && (cur_value-prev_value) > 1) {
|
|
found=1;
|
|
break;
|
|
}
|
|
} else if(type==BRANCH_FAULT || type==LOOP_FAULT) {
|
|
/* 5e) brc*: search forward utnil we hit a Jxx or rep (F3 or F2).
|
|
* replace instr with nop.
|
|
*/
|
|
unsigned char *c;
|
|
|
|
c=(unsigned char *) prev_value;
|
|
|
|
/* look for repX prefix */
|
|
|
|
if(text_read_ub(c)==0xf3 || text_read_ub(c)==0xf2) {
|
|
if(verbose)
|
|
printk("found repX prefix\n");
|
|
/* take out repX prefix only */
|
|
found=1;
|
|
cur_value=prev_value+1;
|
|
break;
|
|
} else if( (text_read_ub(c)&0xf0)==0x70 ||
|
|
(text_read_ub(c)>=0xe0 && text_read_ub(c)<=0xe2) ) {
|
|
/* look for jXX 8 (7X), loop,jcx (e0-3), jXX 16/32 (0f 8X) */
|
|
found=1;
|
|
if(verbose)
|
|
printk("found jXX rel8, loop or jcx\n");
|
|
break;
|
|
} else if(text_read_ub(c)==0x66 ||
|
|
text_read_ub(c)==0x67) { /* override prefix */
|
|
c++;
|
|
} else if(text_read_ub(c++)==0xf && (text_read_ub(c)&0xf0)==0x80 ) {
|
|
found=1; /* 0x0f 0x8X */
|
|
if(verbose) printk("found branch!\n");
|
|
break;
|
|
}
|
|
} else if(type==PTR_FAULT) {
|
|
/* 5f) ptr: if instruction has regmodrm byte (i_has_modrm),
|
|
* and mod field has address ([eyy]dispxx), eyy!=ebp
|
|
* flip 1 bit in lower byte (0x0f) or any bit in following
|
|
* bytes (sib, imm or disp).
|
|
*/
|
|
if(cur_value>off && modAddr) {
|
|
unsigned char *c;
|
|
c=(unsigned char *) modAddr;
|
|
if( text_read_ub(c)>0x3f && text_read_ub(c)<0xc0 &&
|
|
(text_read_ub(c)&7)!=5 ) {
|
|
found=1;
|
|
break;
|
|
}
|
|
}
|
|
} else if(type==INTERFACE_FAULT) {
|
|
/* 5f) i/f: look for movl XX(ebp), reg or movb XX(ebp), reg,
|
|
* where XX is positive. replace instr with nop.
|
|
* movl=0x8a, movb=0x8b, mod=01XXX101 (disp8[ebp]), disp>0
|
|
*/
|
|
unsigned char *c;
|
|
c=(unsigned char *) prev_value;
|
|
if( text_read_ub(c)==0x8a || text_read_ub(c)==0x8b) {
|
|
c++;
|
|
if( ((text_read_ub(c++))&0xc7)==0x45 && (text_read_ub(c)&0x80)==0 ) {
|
|
/* 75% chance that we'll choose the next arg */
|
|
if(random()&0x3) {
|
|
found=1;
|
|
break;
|
|
} else {
|
|
if(verbose) printk("skipped...\n");
|
|
}
|
|
}
|
|
}
|
|
}else if(type==IRQ_FAULT) {
|
|
/* 5g) i/f: look for push reg or offset(reg) / popf,
|
|
* where XX is positive. replace instr with nop.
|
|
* movl=0x8a, movb=0x8b, mod=01XXX101 (disp8[ebp]), disp>0
|
|
*/
|
|
unsigned char *c;
|
|
c=(unsigned char *) prev_value;
|
|
if (((text_read_ub(c) & 0xf8) == 0x50) ||
|
|
(text_read_ub(c) == 0xff)) {
|
|
if (text_read_ub(c) == 0xff) {
|
|
c++;
|
|
#if 0
|
|
//
|
|
// Look for push x(ebp)
|
|
#endif
|
|
if ((text_read_ub(c) & 0x78) != 0x70) {
|
|
continue;
|
|
}
|
|
/*
|
|
// Skip the offset
|
|
*/
|
|
c++;
|
|
}
|
|
c++;
|
|
if (text_read_ub(c) == 0x9d) {
|
|
/*
|
|
// Increment cur_value to include the
|
|
// popf instruction
|
|
*/
|
|
cur_value++;
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
/* if we're doing nop fault, then we're done.
|
|
*/
|
|
if(found) {
|
|
*instr_len=cur_value-prev_value;
|
|
off=prev_value;
|
|
if(verbose) {
|
|
printk("%s", name);
|
|
if (d) printk("+0x%x", d);
|
|
printk(" @ %x, ", value);
|
|
printk("instr @ %x, len=%d, ", off, *instr_len);
|
|
#if 0
|
|
// db_disasm(prev_value, FALSE);
|
|
#endif
|
|
}
|
|
return off;
|
|
} else {
|
|
if(verbose) printk("cannot locate instruction in function\n");
|
|
*instr_len=0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
static boolean_t
|
|
db_line_at_pc( sym, filename, linenum, pc)
|
|
db_sym_t sym;
|
|
char **filename;
|
|
int *linenum;
|
|
db_expr_t pc;
|
|
{
|
|
return X_db_line_at_pc( db_last_symtab, sym, filename, linenum, pc);
|
|
}
|
|
|
|
int
|
|
db_sym_numargs(sym, nargp, argnames)
|
|
db_sym_t sym;
|
|
int *nargp;
|
|
char **argnames;
|
|
{
|
|
return X_db_sym_numargs(db_last_symtab, sym, nargp, argnames);
|
|
}
|
|
|
|
#endif
|