diff options
-rw-r--r-- | include/mach.h | 504 |
1 files changed, 504 insertions, 0 deletions
diff --git a/include/mach.h b/include/mach.h new file mode 100644 index 00000000..2a1b168a --- /dev/null +++ b/include/mach.h @@ -0,0 +1,504 @@ +/* + * Architecture-dependent application data. + * + * The code assumes that ulong is big enough to hold + * an address on any system of interest as well as any + * register. Debugging 64-bit code on 32-bit machines + * will be interesting. + * + * Supported architectures: + * + * MIPS R3000 + * Motorola 68020 + * Intel 386 + * SPARC + * PowerPC (limited) + * ARM (limited) + * Intel 960 (limited) + * AT&T 3210 DSP (limited) + * MIPS2 (R4000) + */ + +typedef struct Fhdr Fhdr; +typedef struct Loc Loc; +typedef struct Mach Mach; +typedef struct Map Map; +typedef struct Regdesc Regdesc; +typedef struct Regs Regs; +typedef struct Seg Seg; +typedef struct Symbol Symbol; +typedef struct Symtype Symtype; + +typedef int (*Tracer)(Map*, Regs*, ulong, ulong, Symbol*, int); + +extern Mach *mach; +extern Mach *machcpu; + +/* + * Byte-order data layout manipulation. + * swap.c ieee.c + */ +u16int beswap2(u16int u); +u32int beswap4(u32int u); +u64int beswap8(u64int u); +int beieeeftoa32(char*, uint, void*); +int beieeeftoa64(char*, uint, void*); +int beieeeftoa80(char*, uint, void*); + +u16int leswap2(u16int u); +u32int leswap4(u32int u); +u64int leswap8(u64int u); +int leieeeftoa32(char *a, uint n, void *v); +int leieeeftoa64(char *a, uint n, void *v); +int leieeeftoa80(char *a, uint n, void *v); + +u16int beload2(uchar*); +u32int beload4(uchar*); +u64int beload8(uchar*); + +u16int leload2(uchar*); +u32int leload4(uchar*); +u64int leload8(uchar*); + +int ieeeftoa32(char *a, uint n, u32int u); +int ieeeftoa64(char *a, uint n, u32int h, u32int u); + +/* + * Machine-independent access to an executable image. + * map.c + */ +struct Seg +{ + char *name; + char *file; + uchar *p; + int fd; + int pid; + ulong base; + ulong size; + ulong offset; + int (*rw)(Map*, Seg*, ulong, void*, uint, int); +}; + +struct Map +{ + int nseg; + Seg *seg; +}; + +struct Regs +{ + int (*rw)(Regs*, char*, ulong*, int); +}; + +typedef struct UregRegs UregRegs; +struct UregRegs +{ + Regs r; + uchar *ureg; +}; +int _uregrw(Regs*, char*, ulong*, int); + +typedef struct PidRegs PidRegs; +struct PidRegs +{ + Regs r; + int pid; +}; + +Map* allocmap(void); +int addseg(Map *map, Seg seg); +int findseg(Map *map, char *name, char *file); +int addrtoseg(Map *map, ulong addr, Seg *seg); +int addrtosegafter(Map *map, ulong addr, Seg *seg); +void removeseg(Map *map, int i); +void freemap(Map*); + +int get1(Map *map, ulong addr, uchar *a, uint n); +int get2(Map *map, ulong addr, u16int *u); +int get4(Map *map, ulong addr, u32int *u); +int get8(Map *map, ulong addr, u64int *u); + +int put1(Map *map, ulong addr, uchar *a, uint n); +int put2(Map *map, ulong addr, u16int u); +int put4(Map *map, ulong addr, u32int u); +int put8(Map *map, ulong addr, u64int u); + +int rget(Regs*, char*, ulong*); +int rput(Regs*, char*, ulong); + +/* + * A location is either a memory address or a register. + * It is useful to be able to specify constant values that + * originate from outside the register set and memory, + * hence LCONST. If the register values are known, then + * we can dispense with LOFFSET, but it's useful to be able + * to look up local symbols (via findlsym) with locations + * like 8(BP). + * + * loc.c + */ + +enum +{ + /* location type */ + LNONE, + LREG, /* register */ + LADDR, /* absolute address */ + LCONST, /* constant (an anonymous readonly location) */ + LOFFSET, /* dereference offset + register ptr */ +}; + +struct Loc +{ + uint type; /* LNONE, ... */ + char *reg; /* LREG */ + ulong addr; /* LADDR, CONST */ + long offset; /* LOFFSET */ +}; + +int lget1(Map *map, Regs *regs, Loc loc, uchar *a, uint n); +int lget2(Map *map, Regs *regs, Loc loc, u16int *v); +int lget4(Map *map, Regs *regs, Loc loc, u32int *v); +int lget8(Map *map, Regs *regs, Loc loc, u64int *v); + +int lput1(Map *map, Regs *regs, Loc loc, uchar *a, uint n); +int lput2(Map *map, Regs *regs, Loc loc, u16int v); +int lput4(Map *map, Regs *regs, Loc loc, u32int v); +int lput8(Map *map, Regs *regs, Loc loc, u64int v); + +Loc locnone(void); +Loc locaddr(ulong addr); +Loc locconst(ulong con); +Loc locreg(char*); +Loc locindir(char*, long); + +/* + * Executable file parsing. + * + * An Fhdr represents an open file image. + * The contents are a grab bag of constants used for the + * various file types. Not all elements are used by all + * file types. + * + * crackadotplan9.c crackadotunix.c + * crackelf.c crackdwarf.c + */ +enum +{ + /* file types */ + FNONE, + FEXEC, /* executable image */ + FLIB, /* library */ + FOBJ, /* object file */ + FRELOC, /* relocatable executable */ + FSHLIB, /* shared library */ + FSHOBJ, /* shared object */ + FCORE, /* core dump */ + FBOOT, /* bootable image */ + FKERNEL, /* kernel image */ + NFTYPE, + + /* abi types */ + ANONE = 0, + APLAN9, + ALINUX, + AFREEBSD, + AMACH, + NATYPE +}; + +/* I wish this could be kept in stabs.h */ +struct Stab +{ + uchar *stabbase; + uint stabsize; + char *strbase; + uint strsize; + u16int (*e2)(uchar*); + u32int (*e4)(uchar*); +}; + +struct Fhdr +{ + int fd; /* file descriptor */ + char *filename; /* file name */ + Mach *mach; /* machine */ + char *mname; /* 386, power, ... */ + uint mtype; /* machine type M386, ... */ + char *fname; /* core, executable, boot image, ... */ + uint ftype; /* file type FCORE, ... */ + char *aname; /* abi name */ + uint atype; /* abi type ALINUX, ... */ + + ulong magic; /* magic number */ + ulong txtaddr; /* text address */ + ulong entry; /* entry point */ + ulong txtsz; /* text size */ + ulong txtoff; /* text offset in file */ + ulong dataddr; /* data address */ + ulong datsz; /* data size */ + ulong datoff; /* data offset in file */ + ulong bsssz; /* bss size */ + ulong symsz; /* symbol table size */ + ulong symoff; /* symbol table offset in file */ + ulong sppcsz; /* size of sp-pc table */ + ulong sppcoff; /* offset of sp-pc table in file */ + ulong lnpcsz; /* size of line number-pc table */ + ulong lnpcoff; /* size of line number-pc table */ + void *elf; /* handle to elf image */ + void *dwarf; /* handle to dwarf image */ + void *macho; /* handle to mach-o image */ + struct Stab stabs; + + /* private */ + Symbol *sym; /* cached list of symbols */ + Symbol **byname; + uint nsym; + Symbol *esym; /* elf symbols */ + Symbol **ebyname; + uint nesym; + ulong base; /* base address for relocatables */ + Fhdr *next; /* link to next fhdr (internal) */ + + /* file mapping */ + int (*map)(Fhdr*, ulong, Map*, Regs**); + + /* debugging symbol access; see below */ + int (*syminit)(Fhdr*); + void (*symclose)(Fhdr*); + + int (*pc2file)(Fhdr*, ulong, char*, uint, ulong*); + int (*file2pc)(Fhdr*, char*, ulong, ulong*); + int (*line2pc)(Fhdr*, ulong, ulong, ulong*); + + int (*lookuplsym)(Fhdr*, Symbol*, char*, Symbol*); + int (*indexlsym)(Fhdr*, Symbol*, uint, Symbol*); + int (*findlsym)(Fhdr*, Symbol*, Loc, Symbol*); + + int (*unwind)(Fhdr*, Map*, Regs*, ulong*); +}; + +Fhdr* crackhdr(char *file, int mode); +void uncrackhdr(Fhdr *hdr); +int crackelf(int fd, Fhdr *hdr); +int crackmacho(int fd, Fhdr *hdr); + +int syminit(Fhdr*); +int symdwarf(Fhdr*); +int symelf(Fhdr*); +int symstabs(Fhdr*); +int symmacho(Fhdr*); + +int mapfile(Fhdr *fp, ulong base, Map *map, Regs **regs); +void unmapfile(Fhdr *fp, Map *map); + +/* + * Process manipulation. + */ +int mapproc(int pid, Map *map, Regs **regs); +void unmapproc(Map *map); +int detachproc(int pid); +int ctlproc(int pid, char *msg); +int procnotes(int pid, char ***notes); +char* proctextfile(int pid); + +/* + * Machine descriptions. + * + * mach.c + * mach386.c dis386.c + * machsparc.c dissparc.c + * ... + */ + +/* + * Register sets. The Regs are opaque, accessed by using + * the reglist (and really the accessor functions). + */ +enum +{ + /* must be big enough for all machine register sets */ + REGSIZE = 256, + + RINT = 0<<0, + RFLT = 1<<0, + RRDONLY = 1<<1, +}; + +struct Regdesc +{ + char *name; /* register name */ + uint offset; /* offset in b */ + uint flags; /* RINT/RFLT/RRDONLY */ + uint format; /* print format: 'x', 'X', 'f', 'z', 'Z' */ +}; + +Regdesc* regdesc(char*); + +enum +{ + /* machine types */ + MNONE, + MMIPS, /* MIPS R3000 */ + MSPARC, /* SUN SPARC */ + M68000, /* Motorola 68000 */ + M386, /* Intel 32-bit x86*/ + M960, /* Intel 960 */ + M3210, /* AT&T 3210 DSP */ + MMIPS2, /* MIPS R4000 */ + M29000, /* AMD 29000 */ + MARM, /* ARM */ + MPOWER, /* PowerPC */ + MALPHA, /* DEC/Compaq Alpha */ + NMTYPE +}; + +struct Mach +{ + char *name; /* "386", ... */ + uint type; /* M386, ... */ + Regdesc *reglist; /* register set */ + uint regsize; /* size of register set in bytes */ + uint fpregsize; /* size of fp register set in bytes */ + char *pc; /* name of program counter */ + char *sp; /* name of stack pointer */ + char *fp; /* name of frame pointer */ + char *link; /* name of link register */ + char *sbreg; /* name of static base */ + ulong sb; /* value of static base */ + uint pgsize; /* page size */ + ulong kbase; /* kernel base address for Plan 9 */ + ulong ktmask; /* ktzero = kbase & ~ktmask */ + uint pcquant; /* pc quantum */ + uint szaddr; /* size of pointer in bytes */ + uint szreg; /* size of integer register */ + uint szfloat; /* size of float */ + uint szdouble; /* size of double */ + char** windreg; /* unwinding registers */ + uint nwindreg; + + uchar bpinst[4]; /* break point instruction */ + uint bpsize; /* size of bp instruction */ + + int (*foll)(Map*, Regs*, ulong, ulong*); /* follow set */ + char* (*exc)(Map*, Regs*); /* last exception */ + int (*unwind)(Map*, Regs*, ulong*); + + /* cvt to local byte order */ + u16int (*swap2)(u16int); + u32int (*swap4)(u32int); + u64int (*swap8)(u64int); + int (*ftoa32)(char*, uint, void*); + int (*ftoa64)(char*, uint, void*); + int (*ftoa80)(char*, uint, void*); + + /* disassembly */ + int (*das)(Map*, ulong, char, char*, int); /* symbolic */ + int (*kendas)(Map*, ulong, char, char*, int); /* symbolic */ + int (*codas)(Map*, ulong, char, char*, int); + int (*hexinst)(Map*, ulong, char*, int); /* hex */ + int (*instsize)(Map*, ulong); /* instruction size */ +}; + +Mach *machbyname(char*); +Mach *machbytype(uint); + +extern Mach mach386; +extern Mach machsparc; +extern Mach machmips; +extern Mach machpower; + +/* + * Debugging symbols and type information. + * (Not all objects include type information.) + * + * sym.c + */ + +enum +{ + /* symbol table classes */ + CNONE, + CAUTO, /* stack variable */ + CPARAM, /* function parameter */ + CTEXT, /* text segment */ + CDATA, /* data segment */ + CANY, +}; + +struct Symbol +{ + char *name; /* name of symbol */ + /* Symtype *typedesc; /* type info, if any */ + Loc loc; /* location of symbol */ + Loc hiloc; /* location of end of symbol */ + char class; /* CAUTO, ... */ + char type; /* type letter from a.out.h */ + Fhdr *fhdr; /* where did this come from? */ + uint index; /* in by-address list */ + + /* private use by various symbol implementations */ + union { + struct { + uint unit; + uint uoff; + } dwarf; + struct { + uint i; + uint locals; + char *dir; + char *file; + char frameptr; + uint framesize; + } stabs; + } u; +}; + +/* look through all currently cracked Fhdrs calling their fns */ +int pc2file(ulong pc, char *file, uint nfile, ulong *line); +int file2pc(char *file, ulong line, ulong *addr); +int line2pc(ulong basepc, ulong line, ulong *pc); +int fnbound(ulong pc, ulong *bounds); +int fileline(ulong pc, char *a, uint n); +int pc2line(ulong pc, ulong *line); + +int lookupsym(char *fn, char *var, Symbol *s); +int indexsym(uint ndx, Symbol *s); +int findsym(Loc loc, uint class, Symbol *s); +int findexsym(Fhdr*, uint, Symbol*); + +int lookuplsym(Symbol *s1, char *name, Symbol *s2); +int indexlsym(Symbol *s1, uint ndx, Symbol *s2); +int findlsym(Symbol *s1, Loc loc, Symbol *s); +int symoff(char *a, uint n, ulong addr, uint class); +int unwindframe(Map *map, Regs *regs, ulong *next); + +void _addhdr(Fhdr*); +void _delhdr(Fhdr*); +Fhdr* fhdrlist; + +Symbol* flookupsym(Fhdr*, char*); +Symbol* ffindsym(Fhdr*, Loc, uint); +Symbol* addsym(Fhdr*, Symbol*); + +/* + * Stack frame walking. + * + * frame.c + */ +int stacktrace(Map*, Regs*, Tracer); +int windindex(char*); +Loc* windreglocs(void); + +/* + * Debugger help. + */ +int localaddr(Map *map, Regs *regs, char *fn, char *var, ulong *val); +int fpformat(Map *map, Regdesc *reg, char *a, uint n, uint code); +char* _hexify(char*, ulong, int); +int locfmt(Fmt*); +int loccmp(Loc*, Loc*); +int locsimplify(Map *map, Regs *regs, Loc loc, Loc *newloc); + +extern int machdebug; |