[444] | 1 | /* |
---|
| 2 | * mvme135-asm.S -- assembler routines for the MVME stub. |
---|
| 3 | * |
---|
| 4 | * This code was pulled out of mvme135-stub.c by Ian Taylor so that I |
---|
| 5 | * could handle different register and label prefixes in a sensible |
---|
| 6 | * way. |
---|
| 7 | */ |
---|
| 8 | |
---|
| 9 | /**************************************************************************** |
---|
| 10 | |
---|
| 11 | THIS SOFTWARE IS NOT COPYRIGHTED |
---|
| 12 | |
---|
| 13 | HP offers the following for use in the public domain. HP makes no |
---|
| 14 | warranty with regard to the software or it's performance and the |
---|
| 15 | user accepts the software "AS IS" with all faults. |
---|
| 16 | |
---|
| 17 | HP DISCLAIMS ANY WARRANTIES, EXPRESS OR IMPLIED, WITH REGARD |
---|
| 18 | TO THIS SOFTWARE INCLUDING BUT NOT LIMITED TO THE WARRANTIES |
---|
| 19 | OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. |
---|
| 20 | |
---|
| 21 | ****************************************************************************/ |
---|
| 22 | |
---|
| 23 | #include "asm.h" |
---|
| 24 | |
---|
| 25 | .title "mvme135-asm.S for m68k" |
---|
| 26 | |
---|
| 27 | .globl SYM (registers) |
---|
| 28 | .globl SYM (lastFrame) |
---|
| 29 | .globl SYM (superStack) |
---|
| 30 | .globl SYM (exceptionHook) |
---|
| 31 | .globl SYM (_returnFromException) |
---|
| 32 | .globl SYM (stackPtr) |
---|
| 33 | .globl SYM (handle_exception) |
---|
| 34 | .globl SYM (exceptionSize) |
---|
| 35 | .globl SYM (exceptionHandler) |
---|
| 36 | |
---|
| 37 | .text |
---|
| 38 | |
---|
| 39 | |
---|
| 40 | /* |
---|
| 41 | * Create a new exception vector table and populates it. Vectors from the |
---|
| 42 | * boot monitor are spliced in so I/O and the abort button will continue |
---|
| 43 | * to work. We also use the monitor's generalized vector for anything the |
---|
| 44 | * debugger doesn't want. |
---|
| 45 | */ |
---|
| 46 | .global SYM (setup_vectors) |
---|
| 47 | SYM (setup_vectors): |
---|
| 48 | link fp, IMM (-8) |
---|
| 49 | /* copy monitor vector table */ |
---|
| 50 | |
---|
| 51 | movecl vbr, a0 |
---|
| 52 | lea SYM (vbr_table), a1 |
---|
| 53 | movel 0x8(a0), d0 /* get generalized vector */ |
---|
| 54 | movew IMM (0x3fc), d1 /* load vector count */ |
---|
| 55 | |
---|
| 56 | loop: /* fill table to gen. vector */ |
---|
| 57 | movel d0, (a1,d1) |
---|
| 58 | subqw IMM (4), d1 |
---|
| 59 | bne loop |
---|
| 60 | |
---|
| 61 | movel 0x10(a0), 0x10(a1) /* breakpoint */ |
---|
| 62 | movel 0x24(a0), 0x24(a1) /* trace */ |
---|
| 63 | movel 0xbc(a0), 0xbc(a1) /* system call */ |
---|
| 64 | |
---|
| 65 | /* add stub vectors to table */ |
---|
| 66 | movel SYM (_catchException), 0x8(a1) /* vector = 2, Access Fault */ |
---|
| 67 | movel SYM (_catchException), 0xc(a1) /* vector = 3, Address Error */ |
---|
| 68 | movel SYM (_catchException), 0x10(a1) /* vector = 4, Illegal instruction */ |
---|
| 69 | movel SYM (_catchException), 0x14(a1) /* vector = 5, divide by 0 */ |
---|
| 70 | movel SYM (_catchException), 0x18(a1) /* vector = 6, chk, chk2 instruction */ |
---|
| 71 | movel SYM (_catchException), 0x1c(a1) /* vector = 7, ftrap, trap, trapv ins */ |
---|
| 72 | movel SYM (_catchException), 0x20(a1) /* vector = 8, priviledge violation */ |
---|
| 73 | movel SYM (_catchException), 0x24(a1) /* vector = 9, trace */ |
---|
| 74 | movel SYM (_catchException), 0x28(a1) /* vector = 10, Aline opcode */ |
---|
| 75 | movel SYM (_catchException), 0x2c(a1) /* vector = 11, fline opcode */ |
---|
| 76 | movel SYM (_catchException), 0x30(a1) /* vector = 12, reserved */ |
---|
| 77 | movel SYM (_catchException), 0x34(a1) /* vector = 13, coprocessor protocol violation */ |
---|
| 78 | movel SYM (_catchException), 0x38(a1) /* vector = 14, format error */ |
---|
| 79 | movel SYM (_catchException), 0x3c(a1) /* vector = 15, unitialized interupt */ |
---|
| 80 | |
---|
| 81 | /* unassigned, reserved */ |
---|
| 82 | movel SYM (_catchException), 0x40(a1) /* vector = 16 */ |
---|
| 83 | movel SYM (_catchException), 0x44(a1) /* vector = 17 */ |
---|
| 84 | movel SYM (_catchException), 0x48(a1) /* vector = 18 */ |
---|
| 85 | movel SYM (_catchException), 0x4c(a1) /* vector = 19 */ |
---|
| 86 | movel SYM (_catchException), 0x50(a1) /* vector = 20 */ |
---|
| 87 | movel SYM (_catchException), 0x54(a1) /* vector = 21 */ |
---|
| 88 | movel SYM (_catchException), 0x58(a1) /* vector = 22 */ |
---|
| 89 | movel SYM (_catchException), 0x5c(a1) /* vector = 23 */ |
---|
| 90 | |
---|
| 91 | movel SYM (_catchException), 0x84(a1) /* vector = 33, breakpoint, trap #1 */ |
---|
| 92 | movel SYM (_catchException), 0xa0(a1) /* vector = 40 , trap #8*/ |
---|
| 93 | |
---|
| 94 | /* floating point traps */ |
---|
| 95 | movel SYM (_catchException), 0xc0(a1) /* vector = 48 */ |
---|
| 96 | movel SYM (_catchException), 0xc4(a1) /* vector = 49 */ |
---|
| 97 | movel SYM (_catchException), 0xc8(a1) /* vector = 50 */ |
---|
| 98 | movel SYM (_catchException), 0xcc(a1) /* vector = 51 */ |
---|
| 99 | movel SYM (_catchException), 0xd0(a1) /* vector = 52 */ |
---|
| 100 | movel SYM (_catchException), 0xd4(a1) /* vector = 53 */ |
---|
| 101 | movel SYM (_catchException), 0xd8(a1) /* vector = 54 */ |
---|
| 102 | movel SYM (_catchException), 0xdc(a1) /* vector = 55 */ |
---|
| 103 | movel SYM (_catchException), 0xe0(a1) /* vector = 56 */ |
---|
| 104 | movel SYM (_catchException), 0xe4(a1) /* vector = 57 */ |
---|
| 105 | movel SYM (_catchException), 0xe8(a1) /* vector = 58 */ |
---|
| 106 | |
---|
| 107 | /*** movel &__debug_level7, 0x7c(a1) /* level7 interupt vector */ |
---|
| 108 | |
---|
| 109 | movecl a1, vbr /* change VBR to new table */ |
---|
| 110 | unlk fp |
---|
| 111 | rts |
---|
| 112 | /* |
---|
| 113 | * exceptionHandler -- sets up exception vector table. |
---|
| 114 | * First arg is an integer vector number |
---|
| 115 | * Second arg is the function pointer for the vector |
---|
| 116 | */ |
---|
| 117 | SYM (exceptionHandler): |
---|
| 118 | # link a6, IMM (-8) |
---|
| 119 | #str1: .ascii "Exception Handler Called\n" |
---|
| 120 | # moveal IMM (str1), a0 |
---|
| 121 | # moveal IMM (str1+25), a1 |
---|
| 122 | # jsr SYM (outln) |
---|
| 123 | |
---|
| 124 | # unlk a6 |
---|
| 125 | rts |
---|
| 126 | |
---|
| 127 | /* this never gets called */ |
---|
| 128 | movel fp@(8), d0 /* get vector number */ |
---|
| 129 | movel fp@(12), a0 /* get function address */ |
---|
| 130 | moveal &SYM (vbr_table), a1 /* FIXME */ |
---|
| 131 | |
---|
| 132 | addl d0, d0 |
---|
| 133 | addl d0, d0 |
---|
| 134 | |
---|
| 135 | addal d0, a1 |
---|
| 136 | movel a0, (a1) |
---|
| 137 | |
---|
| 138 | movecl a1, vbr |
---|
| 139 | unlk a6 |
---|
| 140 | rts |
---|
| 141 | |
---|
| 142 | .globl SYM (return_to_super) |
---|
| 143 | SYM (return_to_super): |
---|
| 144 | movel SYM (registers)+60,sp /* get new stack pointer */ |
---|
| 145 | movel SYM (lastFrame),a0 /* get last frame info */ |
---|
| 146 | bra return_to_any |
---|
| 147 | |
---|
| 148 | .globl SYM (return_to_user) |
---|
| 149 | SYM (return_to_user): |
---|
| 150 | movel SYM (registers)+60,a0 /* get usp */ |
---|
| 151 | movel a0,usp /* set usp */ |
---|
| 152 | movel SYM (superStack),sp /* get original stack pointer */ |
---|
| 153 | |
---|
| 154 | return_to_any: |
---|
| 155 | movel SYM (lastFrame),a0 /* get last frame info */ |
---|
| 156 | movel a0@+,SYM (lastFrame) /* link in previous frame */ |
---|
| 157 | addql IMM (8),a0 /* skip over pc, vector#*/ |
---|
| 158 | movew a0@+,d0 /* get # of words in cpu frame */ |
---|
| 159 | addw d0,a0 /* point to end of data */ |
---|
| 160 | addw d0,a0 /* point to end of data */ |
---|
| 161 | movel a0,a1 |
---|
| 162 | /* copy the stack frame */ |
---|
| 163 | subql IMM (1),d0 |
---|
| 164 | copyUserLoop: |
---|
| 165 | movew a1@-,sp@- |
---|
| 166 | dbf d0,copyUserLoop |
---|
| 167 | |
---|
| 168 | #ifdef __HAVE_68881__ |
---|
| 169 | fmoveml SYM (registers)+168,fpcr/fpsr/fpi |
---|
| 170 | fmovemx SYM (registers)+72,fp0-fp7 |
---|
| 171 | cmpl IMM (-1),a0@ /* skip frestore flag set ? */ |
---|
| 172 | beq skip_frestore |
---|
| 173 | frestore a0@+ |
---|
| 174 | skip_frestore: |
---|
| 175 | #endif |
---|
| 176 | |
---|
| 177 | moveml SYM (registers),d0-d7/a0-a6 |
---|
| 178 | rte /* pop and go! */ |
---|
| 179 | |
---|
| 180 | |
---|
| 181 | /* this function is called immediately when a level 7 interrupt occurs */ |
---|
| 182 | /* if the previous interrupt level was 7 then we're already servicing */ |
---|
| 183 | /* this interrupt and an rte is in order to return to the debugger. */ |
---|
| 184 | /* For the 68000, the offset for sr is 6 due to the jsr return address */ |
---|
| 185 | .text |
---|
| 186 | .globl SYM (_debug_level7) |
---|
| 187 | SYM (_debug_level7): |
---|
| 188 | movew d0,sp@- |
---|
| 189 | #ifdef mc68020 |
---|
| 190 | movew sp@(2),d0 |
---|
| 191 | #else |
---|
| 192 | movew sp@(6),d0 |
---|
| 193 | #endif |
---|
| 194 | andiw IMM (0x700),d0 |
---|
| 195 | cmpiw IMM (0x700),d0 |
---|
| 196 | beq _already7 |
---|
| 197 | movew sp@+,d0 |
---|
| 198 | bra SYM (_catchException) |
---|
| 199 | _already7: |
---|
| 200 | movew sp@+,d0 |
---|
| 201 | #ifndef mc68020 |
---|
| 202 | lea sp@(4),sp /* pull off 68000 return address */ |
---|
| 203 | #endif |
---|
| 204 | rte |
---|
| 205 | |
---|
| 206 | #ifdef mc68020 |
---|
| 207 | /* This function is called when a 68020 exception occurs. It saves |
---|
| 208 | * all the cpu and fpcp regs in the _registers array, creates a frame on a |
---|
| 209 | * linked list of frames which has the cpu and fpcp stack frames needed |
---|
| 210 | * to properly restore the context of these processors, and invokes |
---|
| 211 | * an exception handler (remcom_handler). |
---|
| 212 | * |
---|
| 213 | * stack on entry: stack on exit: |
---|
| 214 | * N bytes of junk exception # MSWord |
---|
| 215 | * Exception Format Word exception # MSWord |
---|
| 216 | * Program counter LSWord |
---|
| 217 | * Program counter MSWord |
---|
| 218 | * Status Register |
---|
| 219 | * |
---|
| 220 | * |
---|
| 221 | */ |
---|
| 222 | |
---|
| 223 | .text |
---|
| 224 | .globl SYM (_catchException) |
---|
| 225 | SYM (_catchException): |
---|
| 226 | |
---|
| 227 | oriw IMM (0x0700),sr /* Disable interrupts */ |
---|
| 228 | |
---|
| 229 | moveml d0-d7/a0-a6,SYM (registers) /* save registers */ |
---|
| 230 | movel SYM (lastFrame),a0 /* last frame pointer */ |
---|
| 231 | |
---|
| 232 | #ifdef __HAVE_68881__ |
---|
| 233 | /* do an fsave, then remember the address to begin a restore from */ |
---|
| 234 | fsave a0@- |
---|
| 235 | fmovemx fp0-fp7, SYM (registers)+72 |
---|
| 236 | fmoveml fpcr/fpsr/fpi, SYM (registers)+168 |
---|
| 237 | #endif |
---|
| 238 | |
---|
| 239 | lea SYM (registers),a5 /* get address of registers */ |
---|
| 240 | movew sp@,d1 /* get status register */ |
---|
| 241 | movew d1,a5@(66) /* save sr */ |
---|
| 242 | movel sp@(2),a4 /* save pc in a4 for later use */ |
---|
| 243 | movel a4,a5@(68) /* save pc in _regisers[] */ |
---|
| 244 | |
---|
| 245 | /* figure out how many bytes in the stack frame */ |
---|
| 246 | movew sp@(6),d0 /* get '020 exception format */ |
---|
| 247 | movew d0,d2 /* make a copy of format word */ |
---|
| 248 | andiw IMM (0xf000),d0 /* mask off format type */ |
---|
| 249 | rolw IMM (5),d0 /* rotate into the low byte *2 */ |
---|
| 250 | lea SYM (exceptionSize),a1 |
---|
| 251 | addw d0,a1 /* index into the table */ |
---|
| 252 | movew a1@,d0 /* get number of words in frame */ |
---|
| 253 | movew d0,d3 /* save it */ |
---|
| 254 | subw d0,a0 /* adjust save pointer */ |
---|
| 255 | subw d0,a0 /* adjust save pointer(bytes) */ |
---|
| 256 | movel a0,a1 /* copy save pointer */ |
---|
| 257 | subql IMM (1),d0 /* predecrement loop counter */ |
---|
| 258 | |
---|
| 259 | /* copy the frame */ |
---|
| 260 | |
---|
| 261 | saveFrameLoop: |
---|
| 262 | movew sp@+,a1@+ |
---|
| 263 | dbf d0,saveFrameLoop |
---|
| 264 | |
---|
| 265 | /* now that the stack has been clenaed, |
---|
| 266 | * save the a7 in use at time of exception |
---|
| 267 | */ |
---|
| 268 | movel sp,SYM (superStack) /* save supervisor sp */ |
---|
| 269 | andiw IMM (0x2000),d1 /* were we in supervisor mode ? */ |
---|
| 270 | beq userMode |
---|
| 271 | movel a7,a5@(60) /* save a7 */ |
---|
| 272 | bra a7saveDone |
---|
| 273 | userMode: |
---|
| 274 | movel usp,a1 |
---|
| 275 | movel a1,a5@(60) /* save user stack pointer */ |
---|
| 276 | a7saveDone: |
---|
| 277 | |
---|
| 278 | |
---|
| 279 | /* save size of frame */ |
---|
| 280 | movew d3,a0@- |
---|
| 281 | |
---|
| 282 | /* compute exception number */ |
---|
| 283 | andl IMM (0xfff),d2 /* mask off vector offset */ |
---|
| 284 | lsrw IMM (2),d2 /* divide by 4 to get vect num */ |
---|
| 285 | movel d2,a0@- /* save it */ |
---|
| 286 | |
---|
| 287 | /* save pc causing exception */ |
---|
| 288 | movel a4,a0@- |
---|
| 289 | |
---|
| 290 | /* save old frame link and set the new value*/ |
---|
| 291 | movel SYM (lastFrame),a1 /* last frame pointer */ |
---|
| 292 | movel a1,a0@- /* save pointer to prev frame */ |
---|
| 293 | movel a0,SYM (lastFrame) |
---|
| 294 | |
---|
| 295 | movel d2,sp@- /* push exception num */ |
---|
| 296 | #ifdef TMP_HACK |
---|
| 297 | movel SYM (exceptionHook),a0 /* get address of handler */ |
---|
| 298 | jbsr a0@ /* and call it */ |
---|
| 299 | #else |
---|
| 300 | jbsr SYM (remcomHandler) |
---|
| 301 | #endif |
---|
| 302 | clrl sp@ /* replace exception num parm with frame ptr */ |
---|
| 303 | jbsr SYM (_returnFromException) /* jbsr, but never returns */ |
---|
| 304 | |
---|
| 305 | #else /* mc68000 */ |
---|
| 306 | |
---|
| 307 | /* This function is called when an exception occurs. It translates the |
---|
| 308 | * return address found on the stack into an exception vector # which |
---|
| 309 | * is then handled by either handle_exception or a system handler. |
---|
| 310 | * _catchException provides a front end for both. |
---|
| 311 | * |
---|
| 312 | * stack on entry: stack on exit: |
---|
| 313 | * Program counter MSWord exception # MSWord |
---|
| 314 | * Program counter LSWord exception # MSWord |
---|
| 315 | * Status Register |
---|
| 316 | * Return Address MSWord |
---|
| 317 | * Return Address LSWord |
---|
| 318 | */ |
---|
| 319 | .text |
---|
| 320 | .globl SYM (_catchException) |
---|
| 321 | SYM (_catchException): |
---|
| 322 | |
---|
| 323 | oriw IMM (0x0700),sr /* Disable interrupts */ |
---|
| 324 | |
---|
| 325 | moveml d0-d7/a0-a6,SYM (registers) /* save registers */ |
---|
| 326 | movel SYM (lastFrame),a0 /* last frame pointer */ |
---|
| 327 | |
---|
| 328 | #ifdef __HAVE_68881__ |
---|
| 329 | /* do an fsave, then remember the address to begin a restore from */ |
---|
| 330 | fsave a0@- |
---|
| 331 | fmovemx fp0-fp7, SYM (registers)+72 |
---|
| 332 | fmoveml fpcr/fpsr/fpi, SYM (registers)+168 |
---|
| 333 | #endif |
---|
| 334 | |
---|
| 335 | lea SYM (registers),a5 /* get address of registers */ |
---|
| 336 | movel sp@+,d2 /* pop return address */ |
---|
| 337 | addl IMM (1530),d2 /* convert return addr to */ |
---|
| 338 | divs IMM (6),d2 /* exception number */ |
---|
| 339 | extl d2 |
---|
| 340 | |
---|
| 341 | moveql IMM (3),d3 /* assume a three word frame */ |
---|
| 342 | |
---|
| 343 | cmpiw IMM (3),d2 /* bus error or address error ? */ |
---|
| 344 | bgt normal /* if >3 then normal error */ |
---|
| 345 | movel sp@+,a0@- /* copy error info to frame buff*/ |
---|
| 346 | movel sp@+,a0@- /* these are never used */ |
---|
| 347 | moveql IMM (7),d3 /* this is a 7 word frame */ |
---|
| 348 | |
---|
| 349 | normal: |
---|
| 350 | movew sp@+,d1 /* pop status register */ |
---|
| 351 | movel sp@+,a4 /* pop program counter */ |
---|
| 352 | movew d1,a5@(66) /* save sr */ |
---|
| 353 | movel a4,a5@(68) /* save pc in _regisers[] */ |
---|
| 354 | movel a4,a0@- /* copy pc to frame buffer */ |
---|
| 355 | movew d1,a0@- /* copy sr to frame buffer */ |
---|
| 356 | |
---|
| 357 | movel sp,SYM (superStack) /* save supervisor sp */ |
---|
| 358 | |
---|
| 359 | andiw IMM (0x2000),d1 /* were we in supervisor mode ? */ |
---|
| 360 | beq userMode |
---|
| 361 | movel a7,a5@(60) /* save a7 */ |
---|
| 362 | bra saveDone |
---|
| 363 | userMode: |
---|
| 364 | movel usp,a1 /* save user stack pointer */ |
---|
| 365 | movel a1,a5@(60) /* save user stack pointer */ |
---|
| 366 | saveDone: |
---|
| 367 | |
---|
| 368 | movew d3,a0@- /* push frame size in words */ |
---|
| 369 | movel d2,a0@- /* push vector number */ |
---|
| 370 | movel a4,a0@- /* push exception pc */ |
---|
| 371 | |
---|
| 372 | /* save old frame link and set the new value */ |
---|
| 373 | movel SYM (lastFrame),a1 /* last frame pointer */ |
---|
| 374 | movel a1,a0@- /* save pointer to prev frame */ |
---|
| 375 | movel a0,SYM (lastFrame) |
---|
| 376 | |
---|
| 377 | movel d2,sp@- /* push exception num */ |
---|
| 378 | movel SYM (exceptionHook),a0 /* get address of handler */ |
---|
| 379 | jbsr a0@ /* and call it */ |
---|
| 380 | clrl sp@ /* replace exception num parm with frame ptr */ |
---|
| 381 | jbsr SYM (_returnFromException) /* jbsr, but never returns */ |
---|
| 382 | |
---|
| 383 | #endif /* m68000 */ |
---|
| 384 | |
---|
| 385 | /* |
---|
| 386 | * remcomHandler is a front end for handle_exception. It moves the |
---|
| 387 | * stack pointer into an area reserved for debugger use in case the |
---|
| 388 | * breakpoint happened in supervisor mode. |
---|
| 389 | */ |
---|
| 390 | .globl SYM (remcomHandler) |
---|
| 391 | SYM (remcomHandler): |
---|
| 392 | addl IMM (4),sp /* pop off return address */ |
---|
| 393 | movel sp@+,d0 /* get the exception number */ |
---|
| 394 | movel SYM (stackPtr),sp /* move to remcom stack area */ |
---|
| 395 | movel d0,sp@- /* push exception onto stack */ |
---|
| 396 | jbsr SYM (handle_exception) /* this never returns */ |
---|
| 397 | rts /* return */ |
---|