601 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			601 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
 |