|
|
1.1 ! root 1: #ifndef lint ! 2: static char sccsid[] = "@(#)access.c 4.8 10/13/84"; ! 3: #endif ! 4: /* ! 5: * Adb: access data in file/process address space. ! 6: * ! 7: * The routines in this file access referenced data using ! 8: * the maps to access files, ptrace to access subprocesses, ! 9: * or the system page tables when debugging the kernel, ! 10: * to translate virtual to physical addresses. ! 11: */ ! 12: ! 13: #include "defs.h" ! 14: ! 15: ! 16: MAP txtmap; ! 17: MAP datmap; ! 18: INT wtflag; ! 19: STRING errflg; ! 20: INT errno; ! 21: ! 22: INT pid; ! 23: ! 24: /* ! 25: * Primitives: put a value in a space, get a value from a space ! 26: * and get a word or byte not returning if an error occurred. ! 27: */ ! 28: put(addr, space, value) ! 29: off_t addr; { (void) access(WT, addr, space, value); } ! 30: ! 31: u_int ! 32: get(addr, space) ! 33: off_t addr; { return (access(RD, addr, space, 0)); }; ! 34: ! 35: u_int ! 36: chkget(addr, space) ! 37: off_t addr; { u_int w = get(addr, space); chkerr(); return(w); } ! 38: ! 39: u_int ! 40: bchkget(addr, space) ! 41: off_t addr; { return(chkget(addr, space) & LOBYTE); } ! 42: ! 43: /* ! 44: * Read/write according to mode at address addr in i/d space. ! 45: * Value is quantity to be written, if write. ! 46: * ! 47: * This routine decides whether to get the data from the subprocess ! 48: * address space with ptrace, or to get it from the files being ! 49: * debugged. ! 50: * ! 51: * When the kernel is being debugged with the -k flag we interpret ! 52: * the system page tables for data space, mapping p0 and p1 addresses ! 53: * relative to the ``current'' process (as specified by its p_addr in ! 54: * <p) and mapping system space addresses through the system page tables. ! 55: */ ! 56: access(mode, addr, space, value) ! 57: int mode, space, value; ! 58: off_t addr; ! 59: { ! 60: int rd = mode == RD; ! 61: int file, w; ! 62: ! 63: if (space == NSP) ! 64: return(0); ! 65: if (pid) { ! 66: int pmode = (space&DSP ? ! 67: (rd ? PT_READ_D : PT_WRITE_D) : ! 68: (rd ? PT_READ_I : PT_WRITE_I)); ! 69: ! 70: w = ptrace(pmode, pid, addr, value); ! 71: if (errno) ! 72: rwerr(space); ! 73: return (w); ! 74: } ! 75: w = 0; ! 76: if (mode==WT && wtflag==0) ! 77: error("not in write mode"); ! 78: if (!chkmap(&addr, space)) ! 79: return (0); ! 80: file = (space&DSP) ? datmap.ufd : txtmap.ufd; ! 81: if (kernel && space == DSP) { ! 82: addr = vtophys(addr); ! 83: if (addr < 0) ! 84: return (0); ! 85: } ! 86: if (physrw(file, addr, rd ? &w : &value, rd) < 0) ! 87: rwerr(space); ! 88: return (w); ! 89: } ! 90: ! 91: /* ! 92: * When looking at kernel data space through /dev/mem or ! 93: * with a core file, do virtual memory mapping. ! 94: */ ! 95: vtophys(addr) ! 96: off_t addr; ! 97: { ! 98: int oldaddr = addr; ! 99: int v; ! 100: struct pte pte; ! 101: ! 102: addr &= ~0xc0000000; ! 103: v = btop(addr); ! 104: switch (oldaddr&0xc0000000) { ! 105: ! 106: case 0xc0000000: ! 107: case 0x80000000: ! 108: /* ! 109: * In system space get system pte. If ! 110: * valid or reclaimable then physical address ! 111: * is combination of its page number and the page ! 112: * offset of the original address. ! 113: */ ! 114: if (v >= slr) ! 115: goto oor; ! 116: addr = ((long)(sbr+v)) &~ 0x80000000; ! 117: goto simple; ! 118: ! 119: case 0x40000000: ! 120: /* ! 121: * In p1 space must not be in shadow region. ! 122: */ ! 123: if (v < pcb.pcb_p1lr) ! 124: goto oor; ! 125: addr = pcb.pcb_p1br+v; ! 126: break; ! 127: ! 128: case 0x00000000: ! 129: /* ! 130: * In p0 space must not be off end of region. ! 131: */ ! 132: if (v >= pcb.pcb_p0lr) ! 133: goto oor; ! 134: addr = pcb.pcb_p0br+v; ! 135: break; ! 136: oor: ! 137: errflg = "address out of segment"; ! 138: return (-1); ! 139: } ! 140: /* ! 141: * For p0/p1 address, user-level page table should ! 142: * be in kernel vm. Do second-level indirect by recursing. ! 143: */ ! 144: if ((addr & 0x80000000) == 0) { ! 145: errflg = "bad p0br or p1br in pcb"; ! 146: return (-1); ! 147: } ! 148: addr = vtophys(addr); ! 149: simple: ! 150: /* ! 151: * Addr is now address of the pte of the page we ! 152: * are interested in; get the pte and paste up the ! 153: * physical address. ! 154: */ ! 155: if (physrw(fcor, addr, (int *)&pte, 1) < 0) { ! 156: errflg = "page table botch"; ! 157: return (-1); ! 158: } ! 159: /* SHOULD CHECK NOT I/O ADDRESS; NEED CPU TYPE! */ ! 160: if (pte.pg_v == 0 && (pte.pg_fod || pte.pg_pfnum == 0)) { ! 161: errflg = "page not valid/reclaimable"; ! 162: return (-1); ! 163: } ! 164: return (ptob(pte.pg_pfnum) + (oldaddr & PGOFSET)); ! 165: } ! 166: ! 167: rwerr(space) ! 168: int space; ! 169: { ! 170: ! 171: if (space & DSP) ! 172: errflg = "data address not found"; ! 173: else ! 174: errflg = "text address not found"; ! 175: } ! 176: ! 177: physrw(file, addr, aw, rd) ! 178: off_t addr; ! 179: int *aw, rd; ! 180: { ! 181: ! 182: if (longseek(file,addr)==0 || ! 183: (rd ? read(file,aw,sizeof(int)) : write(file,aw,sizeof(int))) < 1) ! 184: return (-1); ! 185: return (0); ! 186: } ! 187: ! 188: chkmap(addr,space) ! 189: REG L_INT *addr; ! 190: REG INT space; ! 191: { ! 192: REG MAPPTR amap; ! 193: amap=((space&DSP?&datmap:&txtmap)); ! 194: IF space&STAR ORF !within(*addr,amap->b1,amap->e1) ! 195: THEN IF within(*addr,amap->b2,amap->e2) ! 196: THEN *addr += (amap->f2)-(amap->b2); ! 197: ELSE rwerr(space); return(0); ! 198: FI ! 199: ELSE *addr += (amap->f1)-(amap->b1); ! 200: FI ! 201: return(1); ! 202: } ! 203: ! 204: within(addr,lbd,ubd) ! 205: u_int addr, lbd, ubd; { return(addr>=lbd && addr<ubd); } ! 206: ! 207: longseek(f, a) ! 208: off_t a; { return(lseek(f, a, 0) != -1); }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.