1 | /* |
---|
2 | * mvme162lx-asm.S -- assembler routines for the MVME stub. |
---|
3 | * |
---|
4 | * This code was pulled out of mvme162lx-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 "mvme162lx-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 | |
---|
36 | .text |
---|
37 | .globl SYM (return_to_super) |
---|
38 | SYM (return_to_super): |
---|
39 | movel SYM (registers)+60,sp /* get new stack pointer */ |
---|
40 | movel SYM (lastFrame),a0 /* get last frame info */ |
---|
41 | bra return_to_any |
---|
42 | |
---|
43 | .globl SYM (return_to_user) |
---|
44 | SYM (return_to_user): |
---|
45 | movel SYM (registers)+60,a0 /* get usp */ |
---|
46 | movel a0,usp /* set usp */ |
---|
47 | movel SYM (superStack),sp /* get original stack pointer */ |
---|
48 | |
---|
49 | return_to_any: |
---|
50 | movel SYM (lastFrame),a0 /* get last frame info */ |
---|
51 | movel a0@+,SYM (lastFrame) /* link in previous frame */ |
---|
52 | addql IMM (8),a0 /* skip over pc, vector#*/ |
---|
53 | movew a0@+,d0 /* get # of words in cpu frame */ |
---|
54 | addw d0,a0 /* point to end of data */ |
---|
55 | addw d0,a0 /* point to end of data */ |
---|
56 | movel a0,a1 |
---|
57 | /* copy the stack frame */ |
---|
58 | subql IMM (1),d0 |
---|
59 | copyUserLoop: |
---|
60 | movew a1@-,sp@- |
---|
61 | dbf d0,copyUserLoop |
---|
62 | |
---|
63 | #ifdef __HAVE_68881__ |
---|
64 | fmoveml SYM (registers)+168,fpcr/fpsr/fpi |
---|
65 | fmovemx SYM (registers)+72,fp0-fp7 |
---|
66 | cmpl IMM (-1),a0@ /* skip frestore flag set ? */ |
---|
67 | beq skip_frestore |
---|
68 | frestore a0@+ |
---|
69 | skip_frestore: |
---|
70 | #endif |
---|
71 | |
---|
72 | moveml SYM (registers),d0-d7/a0-a6 |
---|
73 | rte /* pop and go! */ |
---|
74 | |
---|
75 | |
---|
76 | /* this function is called immediately when a level 7 interrupt occurs */ |
---|
77 | /* if the previous interrupt level was 7 then we're already servicing */ |
---|
78 | /* this interrupt and an rte is in order to return to the debugger. */ |
---|
79 | /* For the 68000, the offset for sr is 6 due to the jsr return address */ |
---|
80 | .text |
---|
81 | .globl SYM (_debug_level7) |
---|
82 | SYM (_debug_level7): |
---|
83 | movew d0,sp@- |
---|
84 | #ifdef mc68020 |
---|
85 | movew sp@(2),d0 |
---|
86 | #else |
---|
87 | movew sp@(6),d0 |
---|
88 | #endif |
---|
89 | andiw IMM (0x700),d0 |
---|
90 | cmpiw IMM (0x700),d0 |
---|
91 | beq _already7 |
---|
92 | movew sp@+,d0 |
---|
93 | bra SYM (_catchException) |
---|
94 | _already7: |
---|
95 | movew sp@+,d0 |
---|
96 | #ifndef mc68020 |
---|
97 | lea sp@(4),sp /* pull off 68000 return address */ |
---|
98 | #endif |
---|
99 | rte |
---|
100 | |
---|
101 | #ifdef mc68020 |
---|
102 | /* This function is called when a 68020 exception occurs. It saves |
---|
103 | * all the cpu and fpcp regs in the _registers array, creates a frame on a |
---|
104 | * linked list of frames which has the cpu and fpcp stack frames needed |
---|
105 | * to properly restore the context of these processors, and invokes |
---|
106 | * an exception handler (remcom_handler). |
---|
107 | * |
---|
108 | * stack on entry: stack on exit: |
---|
109 | * N bytes of junk exception # MSWord |
---|
110 | * Exception Format Word exception # MSWord |
---|
111 | * Program counter LSWord |
---|
112 | * Program counter MSWord |
---|
113 | * Status Register |
---|
114 | * |
---|
115 | * |
---|
116 | */ |
---|
117 | |
---|
118 | .text |
---|
119 | .globl SYM (_catchException) |
---|
120 | SYM (_catchException): |
---|
121 | |
---|
122 | oriw IMM (0x0700),sr /* Disable interrupts */ |
---|
123 | |
---|
124 | moveml d0-d7/a0-a6,SYM (registers) /* save registers */ |
---|
125 | movel SYM (lastFrame),a0 /* last frame pointer */ |
---|
126 | |
---|
127 | #ifdef __HAVE_68881__ |
---|
128 | /* do an fsave, then remember the address to begin a restore from */ |
---|
129 | fsave a0@- |
---|
130 | fmovemx fp0-fp7, SYM (registers)+72 |
---|
131 | fmoveml fpcr/fpsr/fpi, SYM (registers)+168 |
---|
132 | #endif |
---|
133 | |
---|
134 | lea SYM (registers),a5 /* get address of registers */ |
---|
135 | movew sp@,d1 /* get status register */ |
---|
136 | movew d1,a5@(66) /* save sr */ |
---|
137 | movel sp@(2),a4 /* save pc in a4 for later use */ |
---|
138 | movel a4,a5@(68) /* save pc in _regisers[] */ |
---|
139 | |
---|
140 | /* figure out how many bytes in the stack frame */ |
---|
141 | movew sp@(6),d0 /* get '020 exception format */ |
---|
142 | movew d0,d2 /* make a copy of format word */ |
---|
143 | andiw IMM (0xf000),d0 /* mask off format type */ |
---|
144 | rolw IMM (5),d0 /* rotate into the low byte *2 */ |
---|
145 | lea SYM (exceptionSize),a1 |
---|
146 | addw d0,a1 /* index into the table */ |
---|
147 | movew a1@,d0 /* get number of words in frame */ |
---|
148 | movew d0,d3 /* save it */ |
---|
149 | subw d0,a0 /* adjust save pointer */ |
---|
150 | subw d0,a0 /* adjust save pointer(bytes) */ |
---|
151 | movel a0,a1 /* copy save pointer */ |
---|
152 | subql IMM (1),d0 /* predecrement loop counter */ |
---|
153 | |
---|
154 | /* copy the frame */ |
---|
155 | |
---|
156 | saveFrameLoop: |
---|
157 | movew sp@+,a1@+ |
---|
158 | dbf d0,saveFrameLoop |
---|
159 | |
---|
160 | /* now that the stack has been clenaed, |
---|
161 | * save the a7 in use at time of exception |
---|
162 | */ |
---|
163 | movel sp,SYM (superStack) /* save supervisor sp */ |
---|
164 | andiw IMM (0x2000),d1 /* were we in supervisor mode ? */ |
---|
165 | beq userMode |
---|
166 | movel a7,a5@(60) /* save a7 */ |
---|
167 | bra a7saveDone |
---|
168 | userMode: |
---|
169 | movel usp,a1 |
---|
170 | movel a1,a5@(60) /* save user stack pointer */ |
---|
171 | a7saveDone: |
---|
172 | |
---|
173 | |
---|
174 | /* save size of frame */ |
---|
175 | movew d3,a0@- |
---|
176 | |
---|
177 | /* compute exception number */ |
---|
178 | andl IMM (0xfff),d2 /* mask off vector offset */ |
---|
179 | lsrw IMM (2),d2 /* divide by 4 to get vect num */ |
---|
180 | movel d2,a0@- /* save it */ |
---|
181 | |
---|
182 | /* save pc causing exception */ |
---|
183 | movel a4,a0@- |
---|
184 | |
---|
185 | /* save old frame link and set the new value*/ |
---|
186 | movel SYM (lastFrame),a1 /* last frame pointer */ |
---|
187 | movel a1,a0@- /* save pointer to prev frame */ |
---|
188 | movel a0,SYM (lastFrame) |
---|
189 | |
---|
190 | movel d2,sp@- /* push exception num */ |
---|
191 | #ifdef TMP_HACK |
---|
192 | movel SYM (exceptionHook),a0 /* get address of handler */ |
---|
193 | jbsr a0@ /* and call it */ |
---|
194 | #else |
---|
195 | jbsr SYM (remcomHandler) |
---|
196 | #endif |
---|
197 | clrl sp@ /* replace exception num parm with frame ptr */ |
---|
198 | jbsr SYM (_returnFromException) /* jbsr, but never returns */ |
---|
199 | |
---|
200 | #else /* mc68000 */ |
---|
201 | |
---|
202 | /* This function is called when an exception occurs. It translates the |
---|
203 | * return address found on the stack into an exception vector # which |
---|
204 | * is then handled by either handle_exception or a system handler. |
---|
205 | * _catchException provides a front end for both. |
---|
206 | * |
---|
207 | * stack on entry: stack on exit: |
---|
208 | * Program counter MSWord exception # MSWord |
---|
209 | * Program counter LSWord exception # MSWord |
---|
210 | * Status Register |
---|
211 | * Return Address MSWord |
---|
212 | * Return Address LSWord |
---|
213 | */ |
---|
214 | .text |
---|
215 | .globl SYM (_catchException) |
---|
216 | SYM (_catchException): |
---|
217 | |
---|
218 | oriw IMM (0x0700),sr /* Disable interrupts */ |
---|
219 | |
---|
220 | moveml d0-d7/a0-a6,SYM (registers) /* save registers */ |
---|
221 | movel SYM (lastFrame),a0 /* last frame pointer */ |
---|
222 | |
---|
223 | #ifdef __HAVE_68881__ |
---|
224 | /* do an fsave, then remember the address to begin a restore from */ |
---|
225 | fsave a0@- |
---|
226 | fmovemx fp0-fp7, SYM (registers)+72 |
---|
227 | fmoveml fpcr/fpsr/fpi, SYM (registers)+168 |
---|
228 | #endif |
---|
229 | |
---|
230 | lea SYM (registers),a5 /* get address of registers */ |
---|
231 | movel sp@+,d2 /* pop return address */ |
---|
232 | addl IMM (1530),d2 /* convert return addr to */ |
---|
233 | divs IMM (6),d2 /* exception number */ |
---|
234 | extl d2 |
---|
235 | |
---|
236 | moveql IMM (3),d3 /* assume a three word frame */ |
---|
237 | |
---|
238 | cmpiw IMM (3),d2 /* bus error or address error ? */ |
---|
239 | bgt normal /* if >3 then normal error */ |
---|
240 | movel sp@+,a0@- /* copy error info to frame buff*/ |
---|
241 | movel sp@+,a0@- /* these are never used */ |
---|
242 | moveql IMM (7),d3 /* this is a 7 word frame */ |
---|
243 | |
---|
244 | normal: |
---|
245 | movew sp@+,d1 /* pop status register */ |
---|
246 | movel sp@+,a4 /* pop program counter */ |
---|
247 | movew d1,a5@(66) /* save sr */ |
---|
248 | movel a4,a5@(68) /* save pc in _regisers[] */ |
---|
249 | movel a4,a0@- /* copy pc to frame buffer */ |
---|
250 | movew d1,a0@- /* copy sr to frame buffer */ |
---|
251 | |
---|
252 | movel sp,SYM (superStack) /* save supervisor sp */ |
---|
253 | |
---|
254 | andiw IMM (0x2000),d1 /* were we in supervisor mode ? */ |
---|
255 | beq userMode |
---|
256 | movel a7,a5@(60) /* save a7 */ |
---|
257 | bra saveDone |
---|
258 | userMode: |
---|
259 | movel usp,a1 /* save user stack pointer */ |
---|
260 | movel a1,a5@(60) /* save user stack pointer */ |
---|
261 | saveDone: |
---|
262 | |
---|
263 | movew d3,a0@- /* push frame size in words */ |
---|
264 | movel d2,a0@- /* push vector number */ |
---|
265 | movel a4,a0@- /* push exception pc */ |
---|
266 | |
---|
267 | /* save old frame link and set the new value */ |
---|
268 | movel SYM (lastFrame),a1 /* last frame pointer */ |
---|
269 | movel a1,a0@- /* save pointer to prev frame */ |
---|
270 | movel a0,SYM (lastFrame) |
---|
271 | |
---|
272 | movel d2,sp@- /* push exception num */ |
---|
273 | movel SYM (exceptionHook),a0 /* get address of handler */ |
---|
274 | jbsr a0@ /* and call it */ |
---|
275 | clrl sp@ /* replace exception num parm with frame ptr */ |
---|
276 | jbsr SYM (_returnFromException) /* jbsr, but never returns */ |
---|
277 | |
---|
278 | #endif /* m68000 */ |
---|
279 | |
---|
280 | /* |
---|
281 | * remcomHandler is a front end for handle_exception. It moves the |
---|
282 | * stack pointer into an area reserved for debugger use in case the |
---|
283 | * breakpoint happened in supervisor mode. |
---|
284 | */ |
---|
285 | .globl SYM (remcomHandler) |
---|
286 | SYM (remcomHandler): |
---|
287 | addl IMM (4),sp /* pop off return address */ |
---|
288 | movel sp@+,d0 /* get the exception number */ |
---|
289 | movel SYM (stackPtr),sp /* move to remcom stack area */ |
---|
290 | movel d0,sp@- /* push exception onto stack */ |
---|
291 | jbsr SYM (handle_exception) /* this never returns */ |
---|
292 | rts /* return */ |
---|