| // 386 support |
| |
| defn acidinit() // Called after all the init modules are loaded |
| { |
| bplist = {}; |
| bpfmt = 'b'; |
| |
| srcpath = { |
| "./", |
| "/sys/src/libc/port/", |
| "/sys/src/libc/9sys/", |
| "/sys/src/libc/386/" |
| }; |
| |
| srcfiles = {}; // list of loaded files |
| srctext = {}; // the text of the files |
| } |
| |
| defn linkreg(addr) |
| { |
| return {}; |
| } |
| |
| defn stk() // trace |
| { |
| _stk({"PC", *PC, "SP", *SP}, 0); |
| } |
| |
| defn lstk() // trace with locals |
| { |
| _stk({"PC", *PC, "SP", *SP}, 1); |
| } |
| |
| defn gpr() // print general(hah hah!) purpose registers |
| { |
| print("AX\t", *AX, " BX\t", *BX, " CX\t", *CX, " DX\t", *DX, "\n"); |
| print("DI\t", *DI, " SI\t", *SI, " BP\t", *BP, "\n"); |
| } |
| |
| defn spr() // print special processor registers |
| { |
| local pc; |
| local cause; |
| |
| pc = *PC; |
| print("PC\t", pc, " ", fmt(pc, 'a'), " "); |
| pfl(pc); |
| print("SP\t", *SP, " ECODE ", *ECODE, " EFLAG ", *EFLAGS, "\n"); |
| print("CS\t", *CS, " DS\t ", *DS, " SS\t", *SS, "\n"); |
| print("GS\t", *GS, " FS\t ", *FS, " ES\t", *ES, "\n"); |
| |
| cause = *TRAP; |
| print("TRAP\t", cause, " ", reason(cause), "\n"); |
| } |
| |
| defn regs() // print all registers |
| { |
| spr(); |
| gpr(); |
| } |
| |
| defn mmregs() |
| { |
| print("MM0\t", *MM0, " MM1\t", *MM1, "\n"); |
| print("MM2\t", *MM2, " MM3\t", *MM3, "\n"); |
| print("MM4\t", *MM4, " MM5\t", *MM5, "\n"); |
| print("MM6\t", *MM6, " MM7\t", *MM7, "\n"); |
| } |
| |
| defn pfixstop(pid) |
| { |
| if *fmt(*PC-1, 'b') == 0xCC then { |
| // Linux stops us after the breakpoint, not at it |
| *PC = *PC-1; |
| } |
| } |
| |
| |
| defn pstop(pid) |
| { |
| local l; |
| local pc; |
| local why; |
| |
| pc = *PC; |
| |
| // FIgure out why we stopped. |
| if *fmt(pc, 'b') == 0xCC then { |
| why = "breakpoint"; |
| |
| // fix up instruction for print; will put back later |
| *pc = @pc; |
| } else if *(pc-2\x) == 0x80CD then { |
| pc = pc-2; |
| why = "system call"; |
| } else |
| why = "stopped"; |
| |
| if printstopped then { |
| print(pid,": ", why, "\t"); |
| print(fmt(pc, 'a'), "\t", *fmt(pc, 'i'), "\n"); |
| } |
| |
| if why == "breakpoint" then |
| *fmt(pc, bpfmt) = bpinst; |
| |
| if printstopped && notes then { |
| if notes[0] != "sys: breakpoint" then { |
| print("Notes pending:\n"); |
| l = notes; |
| while l do { |
| print("\t", head l, "\n"); |
| l = tail l; |
| } |
| } |
| } |
| } |
| |
| aggr Ureg |
| { |
| 'U' 0 di; |
| 'U' 4 si; |
| 'U' 8 bp; |
| 'U' 12 nsp; |
| 'U' 16 bx; |
| 'U' 20 dx; |
| 'U' 24 cx; |
| 'U' 28 ax; |
| 'U' 32 gs; |
| 'U' 36 fs; |
| 'U' 40 es; |
| 'U' 44 ds; |
| 'U' 48 trap; |
| 'U' 52 ecode; |
| 'U' 56 pc; |
| 'U' 60 cs; |
| 'U' 64 flags; |
| { |
| 'U' 68 usp; |
| 'U' 68 sp; |
| }; |
| 'U' 72 ss; |
| }; |
| |
| defn |
| Ureg(addr) { |
| complex Ureg addr; |
| print(" di ", addr.di, "\n"); |
| print(" si ", addr.si, "\n"); |
| print(" bp ", addr.bp, "\n"); |
| print(" nsp ", addr.nsp, "\n"); |
| print(" bx ", addr.bx, "\n"); |
| print(" dx ", addr.dx, "\n"); |
| print(" cx ", addr.cx, "\n"); |
| print(" ax ", addr.ax, "\n"); |
| print(" gs ", addr.gs, "\n"); |
| print(" fs ", addr.fs, "\n"); |
| print(" es ", addr.es, "\n"); |
| print(" ds ", addr.ds, "\n"); |
| print(" trap ", addr.trap, "\n"); |
| print(" ecode ", addr.ecode, "\n"); |
| print(" pc ", addr.pc, "\n"); |
| print(" cs ", addr.cs, "\n"); |
| print(" flags ", addr.flags, "\n"); |
| print(" sp ", addr.sp, "\n"); |
| print(" ss ", addr.ss, "\n"); |
| }; |
| sizeofUreg = 76; |
| |
| aggr Linkdebug |
| { |
| 'X' 0 version; |
| 'X' 4 map; |
| }; |
| |
| aggr Linkmap |
| { |
| 'X' 0 addr; |
| 'X' 4 name; |
| 'X' 8 dynsect; |
| 'X' 12 next; |
| 'X' 16 prev; |
| }; |
| |
| defn |
| linkdebug() |
| { |
| local a; |
| |
| if !havesymbol("_DYNAMIC") then |
| return 0; |
| |
| a = _DYNAMIC; |
| while *a != 0 do { |
| if *a == 21 then // 21 == DT_DEBUG |
| return *(a+4); |
| a = a+8; |
| } |
| return 0; |
| } |
| |
| defn |
| dynamicmap() |
| { |
| if systype == "linux" || systype == "freebsd" then { |
| local r, m, n; |
| |
| r = linkdebug(); |
| if r then { |
| complex Linkdebug r; |
| m = r.map; |
| n = 0; |
| while m != 0 && n < 100 do { |
| complex Linkmap m; |
| if m.name && *(m.name\b) && access(*(m.name\s)) then |
| print("textfile({\"", *(m.name\s), "\", ", m.addr\X, "});\n"); |
| m = m.next; |
| n = n+1; |
| } |
| } |
| } |
| } |
| |
| defn |
| acidmap() |
| { |
| // dynamicmap(); |
| acidtypes(); |
| } |
| |
| print(acidfile); |