| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | | | 
|  | 2 | |	gen_except.sa 3.7 1/16/92 | 
|  | 3 | | | 
|  | 4 | |	gen_except --- FPSP routine to detect reportable exceptions | 
|  | 5 | | | 
|  | 6 | |	This routine compares the exception enable byte of the | 
|  | 7 | |	user_fpcr on the stack with the exception status byte | 
|  | 8 | |	of the user_fpsr. | 
|  | 9 | | | 
|  | 10 | |	Any routine which may report an exceptions must load | 
|  | 11 | |	the stack frame in memory with the exceptional operand(s). | 
|  | 12 | | | 
|  | 13 | |	Priority for exceptions is: | 
|  | 14 | | | 
|  | 15 | |	Highest:	bsun | 
|  | 16 | |			snan | 
|  | 17 | |			operr | 
|  | 18 | |			ovfl | 
|  | 19 | |			unfl | 
|  | 20 | |			dz | 
|  | 21 | |			inex2 | 
|  | 22 | |	Lowest:		inex1 | 
|  | 23 | | | 
|  | 24 | |	Note: The IEEE standard specifies that inex2 is to be | 
|  | 25 | |	reported if ovfl occurs and the ovfl enable bit is not | 
|  | 26 | |	set but the inex2 enable bit is. | 
|  | 27 | | | 
|  | 28 | | | 
|  | 29 | |		Copyright (C) Motorola, Inc. 1990 | 
|  | 30 | |			All Rights Reserved | 
|  | 31 | | | 
| Matt Waddel | e00d82d | 2006-02-11 17:55:48 -0800 | [diff] [blame] | 32 | |       For details on the license for this file, please see the | 
|  | 33 | |       file, README, in this same directory. | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 34 |  | 
|  | 35 | GEN_EXCEPT:    |idnt    2,1 | Motorola 040 Floating Point Software Package | 
|  | 36 |  | 
|  | 37 | |section 8 | 
|  | 38 |  | 
|  | 39 | #include "fpsp.h" | 
|  | 40 |  | 
|  | 41 | |xref	real_trace | 
|  | 42 | |xref	fpsp_done | 
|  | 43 | |xref	fpsp_fmt_error | 
|  | 44 |  | 
|  | 45 | exc_tbl: | 
|  | 46 | .long	bsun_exc | 
|  | 47 | .long	commonE1 | 
|  | 48 | .long	commonE1 | 
|  | 49 | .long	ovfl_unfl | 
|  | 50 | .long	ovfl_unfl | 
|  | 51 | .long	commonE1 | 
|  | 52 | .long	commonE3 | 
|  | 53 | .long	commonE3 | 
|  | 54 | .long	no_match | 
|  | 55 |  | 
|  | 56 | .global	gen_except | 
|  | 57 | gen_except: | 
|  | 58 | cmpib	#IDLE_SIZE-4,1(%a7)	|test for idle frame | 
|  | 59 | beq	do_check		|go handle idle frame | 
|  | 60 | cmpib	#UNIMP_40_SIZE-4,1(%a7)	|test for orig unimp frame | 
|  | 61 | beqs	unimp_x			|go handle unimp frame | 
|  | 62 | cmpib	#UNIMP_41_SIZE-4,1(%a7)	|test for rev unimp frame | 
|  | 63 | beqs	unimp_x			|go handle unimp frame | 
|  | 64 | cmpib	#BUSY_SIZE-4,1(%a7)	|if size <> $60, fmt error | 
|  | 65 | bnel	fpsp_fmt_error | 
|  | 66 | leal	BUSY_SIZE+LOCAL_SIZE(%a7),%a1 |init a1 so fpsp.h | 
|  | 67 | |					;equates will work | 
|  | 68 | | Fix up the new busy frame with entries from the unimp frame | 
|  | 69 | | | 
|  | 70 | movel	ETEMP_EX(%a6),ETEMP_EX(%a1) |copy etemp from unimp | 
|  | 71 | movel	ETEMP_HI(%a6),ETEMP_HI(%a1) |frame to busy frame | 
|  | 72 | movel	ETEMP_LO(%a6),ETEMP_LO(%a1) | 
|  | 73 | movel	CMDREG1B(%a6),CMDREG1B(%a1) |set inst in frame to unimp | 
|  | 74 | movel	CMDREG1B(%a6),%d0		|fix cmd1b to make it | 
|  | 75 | andl	#0x03c30000,%d0		|work for cmd3b | 
|  | 76 | bfextu	CMDREG1B(%a6){#13:#1},%d1	|extract bit 2 | 
|  | 77 | lsll	#5,%d1 | 
|  | 78 | swap	%d1 | 
|  | 79 | orl	%d1,%d0			|put it in the right place | 
|  | 80 | bfextu	CMDREG1B(%a6){#10:#3},%d1	|extract bit 3,4,5 | 
|  | 81 | lsll	#2,%d1 | 
|  | 82 | swap	%d1 | 
|  | 83 | orl	%d1,%d0			|put them in the right place | 
|  | 84 | movel	%d0,CMDREG3B(%a1)		|in the busy frame | 
|  | 85 | | | 
|  | 86 | | Or in the FPSR from the emulation with the USER_FPSR on the stack. | 
|  | 87 | | | 
|  | 88 | fmovel	%FPSR,%d0 | 
|  | 89 | orl	%d0,USER_FPSR(%a6) | 
|  | 90 | movel	USER_FPSR(%a6),FPSR_SHADOW(%a1) |set exc bits | 
|  | 91 | orl	#sx_mask,E_BYTE(%a1) | 
|  | 92 | bra	do_clean | 
|  | 93 |  | 
|  | 94 | | | 
|  | 95 | | Frame is an unimp frame possible resulting from an fmove <ea>,fp0 | 
|  | 96 | | that caused an exception | 
|  | 97 | | | 
|  | 98 | | a1 is modified to point into the new frame allowing fpsp equates | 
|  | 99 | | to be valid. | 
|  | 100 | | | 
|  | 101 | unimp_x: | 
|  | 102 | cmpib	#UNIMP_40_SIZE-4,1(%a7)	|test for orig unimp frame | 
|  | 103 | bnes	test_rev | 
|  | 104 | leal	UNIMP_40_SIZE+LOCAL_SIZE(%a7),%a1 | 
|  | 105 | bras	unimp_con | 
|  | 106 | test_rev: | 
|  | 107 | cmpib	#UNIMP_41_SIZE-4,1(%a7)	|test for rev unimp frame | 
|  | 108 | bnel	fpsp_fmt_error		|if not $28 or $30 | 
|  | 109 | leal	UNIMP_41_SIZE+LOCAL_SIZE(%a7),%a1 | 
|  | 110 |  | 
|  | 111 | unimp_con: | 
|  | 112 | | | 
|  | 113 | | Fix up the new unimp frame with entries from the old unimp frame | 
|  | 114 | | | 
|  | 115 | movel	CMDREG1B(%a6),CMDREG1B(%a1) |set inst in frame to unimp | 
|  | 116 | | | 
|  | 117 | | Or in the FPSR from the emulation with the USER_FPSR on the stack. | 
|  | 118 | | | 
|  | 119 | fmovel	%FPSR,%d0 | 
|  | 120 | orl	%d0,USER_FPSR(%a6) | 
|  | 121 | bra	do_clean | 
|  | 122 |  | 
|  | 123 | | | 
|  | 124 | | Frame is idle, so check for exceptions reported through | 
|  | 125 | | USER_FPSR and set the unimp frame accordingly. | 
|  | 126 | | A7 must be incremented to the point before the | 
|  | 127 | | idle fsave vector to the unimp vector. | 
|  | 128 | | | 
|  | 129 |  | 
|  | 130 | do_check: | 
|  | 131 | addl	#4,%a7			|point A7 back to unimp frame | 
|  | 132 | | | 
|  | 133 | | Or in the FPSR from the emulation with the USER_FPSR on the stack. | 
|  | 134 | | | 
|  | 135 | fmovel	%FPSR,%d0 | 
|  | 136 | orl	%d0,USER_FPSR(%a6) | 
|  | 137 | | | 
|  | 138 | | On a busy frame, we must clear the nmnexc bits. | 
|  | 139 | | | 
|  | 140 | cmpib	#BUSY_SIZE-4,1(%a7)	|check frame type | 
|  | 141 | bnes	check_fr		|if busy, clr nmnexc | 
|  | 142 | clrw	NMNEXC(%a6)		|clr nmnexc & nmcexc | 
|  | 143 | btstb	#5,CMDREG1B(%a6)		|test for fmove out | 
|  | 144 | bnes	frame_com | 
|  | 145 | movel	USER_FPSR(%a6),FPSR_SHADOW(%a6) |set exc bits | 
|  | 146 | orl	#sx_mask,E_BYTE(%a6) | 
|  | 147 | bras	frame_com | 
|  | 148 | check_fr: | 
|  | 149 | cmpb	#UNIMP_40_SIZE-4,1(%a7) | 
|  | 150 | beqs	frame_com | 
|  | 151 | clrw	NMNEXC(%a6) | 
|  | 152 | frame_com: | 
|  | 153 | moveb	FPCR_ENABLE(%a6),%d0	|get fpcr enable byte | 
|  | 154 | andb	FPSR_EXCEPT(%a6),%d0	|and in the fpsr exc byte | 
|  | 155 | bfffo	%d0{#24:#8},%d1		|test for first set bit | 
|  | 156 | leal	exc_tbl,%a0		|load jmp table address | 
|  | 157 | subib	#24,%d1			|normalize bit offset to 0-8 | 
|  | 158 | movel	(%a0,%d1.w*4),%a0		|load routine address based | 
|  | 159 | |					;based on first enabled exc | 
|  | 160 | jmp	(%a0)			|jump to routine | 
|  | 161 | | | 
|  | 162 | | Bsun is not possible in unimp or unsupp | 
|  | 163 | | | 
|  | 164 | bsun_exc: | 
|  | 165 | bra	do_clean | 
|  | 166 | | | 
|  | 167 | | The typical work to be done to the unimp frame to report an | 
|  | 168 | | exception is to set the E1/E3 byte and clr the U flag. | 
|  | 169 | | commonE1 does this for E1 exceptions, which are snan, | 
|  | 170 | | operr, and dz.  commonE3 does this for E3 exceptions, which | 
|  | 171 | | are inex2 and inex1, and also clears the E1 exception bit | 
|  | 172 | | left over from the unimp exception. | 
|  | 173 | | | 
|  | 174 | commonE1: | 
|  | 175 | bsetb	#E1,E_BYTE(%a6)		|set E1 flag | 
|  | 176 | bra	commonE			|go clean and exit | 
|  | 177 |  | 
|  | 178 | commonE3: | 
|  | 179 | tstb	UFLG_TMP(%a6)		|test flag for unsup/unimp state | 
|  | 180 | bnes	unsE3 | 
|  | 181 | uniE3: | 
|  | 182 | bsetb	#E3,E_BYTE(%a6)		|set E3 flag | 
|  | 183 | bclrb	#E1,E_BYTE(%a6)		|clr E1 from unimp | 
|  | 184 | bra	commonE | 
|  | 185 |  | 
|  | 186 | unsE3: | 
|  | 187 | tstb	RES_FLG(%a6) | 
|  | 188 | bnes	unsE3_0 | 
|  | 189 | unsE3_1: | 
|  | 190 | bsetb	#E3,E_BYTE(%a6)		|set E3 flag | 
|  | 191 | unsE3_0: | 
|  | 192 | bclrb	#E1,E_BYTE(%a6)		|clr E1 flag | 
|  | 193 | movel	CMDREG1B(%a6),%d0 | 
|  | 194 | andl	#0x03c30000,%d0		|work for cmd3b | 
|  | 195 | bfextu	CMDREG1B(%a6){#13:#1},%d1	|extract bit 2 | 
|  | 196 | lsll	#5,%d1 | 
|  | 197 | swap	%d1 | 
|  | 198 | orl	%d1,%d0			|put it in the right place | 
|  | 199 | bfextu	CMDREG1B(%a6){#10:#3},%d1	|extract bit 3,4,5 | 
|  | 200 | lsll	#2,%d1 | 
|  | 201 | swap	%d1 | 
|  | 202 | orl	%d1,%d0			|put them in the right place | 
|  | 203 | movel	%d0,CMDREG3B(%a6)		|in the busy frame | 
|  | 204 |  | 
|  | 205 | commonE: | 
|  | 206 | bclrb	#UFLAG,T_BYTE(%a6)	|clr U flag from unimp | 
|  | 207 | bra	do_clean		|go clean and exit | 
|  | 208 | | | 
|  | 209 | | No bits in the enable byte match existing exceptions.  Check for | 
|  | 210 | | the case of the ovfl exc without the ovfl enabled, but with | 
|  | 211 | | inex2 enabled. | 
|  | 212 | | | 
|  | 213 | no_match: | 
|  | 214 | btstb	#inex2_bit,FPCR_ENABLE(%a6) |check for ovfl/inex2 case | 
|  | 215 | beqs	no_exc			|if clear, exit | 
|  | 216 | btstb	#ovfl_bit,FPSR_EXCEPT(%a6) |now check ovfl | 
|  | 217 | beqs	no_exc			|if clear, exit | 
|  | 218 | bras	ovfl_unfl		|go to unfl_ovfl to determine if | 
|  | 219 | |					;it is an unsupp or unimp exc | 
|  | 220 |  | 
|  | 221 | | No exceptions are to be reported.  If the instruction was | 
|  | 222 | | unimplemented, no FPU restore is necessary.  If it was | 
|  | 223 | | unsupported, we must perform the restore. | 
|  | 224 | no_exc: | 
|  | 225 | tstb	UFLG_TMP(%a6)	|test flag for unsupp/unimp state | 
|  | 226 | beqs	uni_no_exc | 
|  | 227 | uns_no_exc: | 
|  | 228 | tstb	RES_FLG(%a6)	|check if frestore is needed | 
|  | 229 | bne	do_clean	|if clear, no frestore needed | 
|  | 230 | uni_no_exc: | 
|  | 231 | moveml	USER_DA(%a6),%d0-%d1/%a0-%a1 | 
|  | 232 | fmovemx USER_FP0(%a6),%fp0-%fp3 | 
|  | 233 | fmoveml USER_FPCR(%a6),%fpcr/%fpsr/%fpiar | 
|  | 234 | unlk	%a6 | 
|  | 235 | bra	finish_up | 
|  | 236 | | | 
|  | 237 | | Unsupported Data Type Handler: | 
|  | 238 | | Ovfl: | 
|  | 239 | |   An fmoveout that results in an overflow is reported this way. | 
|  | 240 | | Unfl: | 
|  | 241 | |   An fmoveout that results in an underflow is reported this way. | 
|  | 242 | | | 
|  | 243 | | Unimplemented Instruction Handler: | 
|  | 244 | | Ovfl: | 
|  | 245 | |   Only scosh, setox, ssinh, stwotox, and scale can set overflow in | 
|  | 246 | |   this manner. | 
|  | 247 | | Unfl: | 
|  | 248 | |   Stwotox, setox, and scale can set underflow in this manner. | 
|  | 249 | |   Any of the other Library Routines such that f(x)=x in which | 
|  | 250 | |   x is an extended denorm can report an underflow exception. | 
|  | 251 | |   It is the responsibility of the exception-causing exception | 
|  | 252 | |   to make sure that WBTEMP is correct. | 
|  | 253 | | | 
|  | 254 | |   The exceptional operand is in FP_SCR1. | 
|  | 255 | | | 
|  | 256 | ovfl_unfl: | 
|  | 257 | tstb	UFLG_TMP(%a6)	|test flag for unsupp/unimp state | 
|  | 258 | beqs	ofuf_con | 
|  | 259 | | | 
|  | 260 | | The caller was from an unsupported data type trap.  Test if the | 
|  | 261 | | caller set CU_ONLY.  If so, the exceptional operand is expected in | 
|  | 262 | | FPTEMP, rather than WBTEMP. | 
|  | 263 | | | 
|  | 264 | tstb	CU_ONLY(%a6)		|test if inst is cu-only | 
|  | 265 | beq	unsE3 | 
|  | 266 | |	move.w	#$fe,CU_SAVEPC(%a6) | 
|  | 267 | clrb	CU_SAVEPC(%a6) | 
|  | 268 | bsetb	#E1,E_BYTE(%a6)		|set E1 exception flag | 
|  | 269 | movew	ETEMP_EX(%a6),FPTEMP_EX(%a6) | 
|  | 270 | movel	ETEMP_HI(%a6),FPTEMP_HI(%a6) | 
|  | 271 | movel	ETEMP_LO(%a6),FPTEMP_LO(%a6) | 
|  | 272 | bsetb	#fptemp15_bit,DTAG(%a6)	|set fpte15 | 
|  | 273 | bclrb	#UFLAG,T_BYTE(%a6)	|clr U flag from unimp | 
|  | 274 | bra	do_clean		|go clean and exit | 
|  | 275 |  | 
|  | 276 | ofuf_con: | 
|  | 277 | moveb	(%a7),VER_TMP(%a6)	|save version number | 
|  | 278 | cmpib	#BUSY_SIZE-4,1(%a7)	|check for busy frame | 
|  | 279 | beqs	busy_fr			|if unimp, grow to busy | 
|  | 280 | cmpib	#VER_40,(%a7)		|test for orig unimp frame | 
|  | 281 | bnes	try_41			|if not, test for rev frame | 
|  | 282 | moveql	#13,%d0			|need to zero 14 lwords | 
|  | 283 | bras	ofuf_fin | 
|  | 284 | try_41: | 
|  | 285 | cmpib	#VER_41,(%a7)		|test for rev unimp frame | 
|  | 286 | bnel	fpsp_fmt_error		|if neither, exit with error | 
|  | 287 | moveql	#11,%d0			|need to zero 12 lwords | 
|  | 288 |  | 
|  | 289 | ofuf_fin: | 
|  | 290 | clrl	(%a7) | 
|  | 291 | loop1: | 
|  | 292 | clrl	-(%a7)			|clear and dec a7 | 
|  | 293 | dbra	%d0,loop1 | 
|  | 294 | moveb	VER_TMP(%a6),(%a7) | 
|  | 295 | moveb	#BUSY_SIZE-4,1(%a7)		|write busy fmt word. | 
|  | 296 | busy_fr: | 
|  | 297 | movel	FP_SCR1(%a6),WBTEMP_EX(%a6)	|write | 
|  | 298 | movel	FP_SCR1+4(%a6),WBTEMP_HI(%a6)	|exceptional op to | 
|  | 299 | movel	FP_SCR1+8(%a6),WBTEMP_LO(%a6)	|wbtemp | 
|  | 300 | bsetb	#E3,E_BYTE(%a6)			|set E3 flag | 
|  | 301 | bclrb	#E1,E_BYTE(%a6)			|make sure E1 is clear | 
|  | 302 | bclrb	#UFLAG,T_BYTE(%a6)		|clr U flag | 
|  | 303 | movel	USER_FPSR(%a6),FPSR_SHADOW(%a6) | 
|  | 304 | orl	#sx_mask,E_BYTE(%a6) | 
|  | 305 | movel	CMDREG1B(%a6),%d0		|fix cmd1b to make it | 
|  | 306 | andl	#0x03c30000,%d0		|work for cmd3b | 
|  | 307 | bfextu	CMDREG1B(%a6){#13:#1},%d1	|extract bit 2 | 
|  | 308 | lsll	#5,%d1 | 
|  | 309 | swap	%d1 | 
|  | 310 | orl	%d1,%d0			|put it in the right place | 
|  | 311 | bfextu	CMDREG1B(%a6){#10:#3},%d1	|extract bit 3,4,5 | 
|  | 312 | lsll	#2,%d1 | 
|  | 313 | swap	%d1 | 
|  | 314 | orl	%d1,%d0			|put them in the right place | 
|  | 315 | movel	%d0,CMDREG3B(%a6)		|in the busy frame | 
|  | 316 |  | 
|  | 317 | | | 
|  | 318 | | Check if the frame to be restored is busy or unimp. | 
|  | 319 | |** NOTE *** Bug fix for errata (0d43b #3) | 
|  | 320 | | If the frame is unimp, we must create a busy frame to | 
|  | 321 | | fix the bug with the nmnexc bits in cases in which they | 
|  | 322 | | are set by a previous instruction and not cleared by | 
|  | 323 | | the save. The frame will be unimp only if the final | 
|  | 324 | | instruction in an emulation routine caused the exception | 
|  | 325 | | by doing an fmove <ea>,fp0.  The exception operand, in | 
|  | 326 | | internal format, is in fptemp. | 
|  | 327 | | | 
|  | 328 | do_clean: | 
|  | 329 | cmpib	#UNIMP_40_SIZE-4,1(%a7) | 
|  | 330 | bnes	do_con | 
|  | 331 | moveql	#13,%d0			|in orig, need to zero 14 lwords | 
|  | 332 | bras	do_build | 
|  | 333 | do_con: | 
|  | 334 | cmpib	#UNIMP_41_SIZE-4,1(%a7) | 
|  | 335 | bnes	do_restore		|frame must be busy | 
|  | 336 | moveql	#11,%d0			|in rev, need to zero 12 lwords | 
|  | 337 |  | 
|  | 338 | do_build: | 
|  | 339 | moveb	(%a7),VER_TMP(%a6) | 
|  | 340 | clrl	(%a7) | 
|  | 341 | loop2: | 
|  | 342 | clrl	-(%a7)			|clear and dec a7 | 
|  | 343 | dbra	%d0,loop2 | 
|  | 344 | | | 
|  | 345 | | Use a1 as pointer into new frame.  a6 is not correct if an unimp or | 
|  | 346 | | busy frame was created as the result of an exception on the final | 
|  | 347 | | instruction of an emulation routine. | 
|  | 348 | | | 
|  | 349 | | We need to set the nmcexc bits if the exception is E1. Otherwise, | 
|  | 350 | | the exc taken will be inex2. | 
|  | 351 | | | 
|  | 352 | leal	BUSY_SIZE+LOCAL_SIZE(%a7),%a1	|init a1 for new frame | 
|  | 353 | moveb	VER_TMP(%a6),(%a7)	|write busy fmt word | 
|  | 354 | moveb	#BUSY_SIZE-4,1(%a7) | 
|  | 355 | movel	FP_SCR1(%a6),WBTEMP_EX(%a1)	|write | 
|  | 356 | movel	FP_SCR1+4(%a6),WBTEMP_HI(%a1)	|exceptional op to | 
|  | 357 | movel	FP_SCR1+8(%a6),WBTEMP_LO(%a1)	|wbtemp | 
|  | 358 | |	btst.b	#E1,E_BYTE(%a1) | 
|  | 359 | |	beq.b	do_restore | 
|  | 360 | bfextu	USER_FPSR(%a6){#17:#4},%d0	|get snan/operr/ovfl/unfl bits | 
|  | 361 | bfins	%d0,NMCEXC(%a1){#4:#4}	|and insert them in nmcexc | 
|  | 362 | movel	USER_FPSR(%a6),FPSR_SHADOW(%a1) |set exc bits | 
|  | 363 | orl	#sx_mask,E_BYTE(%a1) | 
|  | 364 |  | 
|  | 365 | do_restore: | 
|  | 366 | moveml	USER_DA(%a6),%d0-%d1/%a0-%a1 | 
|  | 367 | fmovemx USER_FP0(%a6),%fp0-%fp3 | 
|  | 368 | fmoveml USER_FPCR(%a6),%fpcr/%fpsr/%fpiar | 
|  | 369 | frestore (%a7)+ | 
|  | 370 | tstb	RES_FLG(%a6)	|RES_FLG indicates a "continuation" frame | 
|  | 371 | beq	cont | 
|  | 372 | bsr	bug1384 | 
|  | 373 | cont: | 
|  | 374 | unlk	%a6 | 
|  | 375 | | | 
|  | 376 | | If trace mode enabled, then go to trace handler.  This handler | 
|  | 377 | | cannot have any fp instructions.  If there are fp inst's and an | 
|  | 378 | | exception has been restored into the machine then the exception | 
|  | 379 | | will occur upon execution of the fp inst.  This is not desirable | 
|  | 380 | | in the kernel (supervisor mode).  See MC68040 manual Section 9.3.8. | 
|  | 381 | | | 
|  | 382 | finish_up: | 
|  | 383 | btstb	#7,(%a7)		|test T1 in SR | 
|  | 384 | bnes	g_trace | 
|  | 385 | btstb	#6,(%a7)		|test T0 in SR | 
|  | 386 | bnes	g_trace | 
|  | 387 | bral	fpsp_done | 
|  | 388 | | | 
|  | 389 | | Change integer stack to look like trace stack | 
|  | 390 | | The address of the instruction that caused the | 
|  | 391 | | exception is already in the integer stack (is | 
|  | 392 | | the same as the saved friar) | 
|  | 393 | | | 
|  | 394 | | If the current frame is already a 6-word stack then all | 
|  | 395 | | that needs to be done is to change the vector# to TRACE. | 
|  | 396 | | If the frame is only a 4-word stack (meaning we got here | 
|  | 397 | | on an Unsupported data type exception), then we need to grow | 
|  | 398 | | the stack an extra 2 words and get the FPIAR from the FPU. | 
|  | 399 | | | 
|  | 400 | g_trace: | 
|  | 401 | bftst	EXC_VEC-4(%sp){#0:#4} | 
|  | 402 | bne	g_easy | 
|  | 403 |  | 
|  | 404 | subw	#4,%sp		| make room | 
|  | 405 | movel	4(%sp),(%sp) | 
|  | 406 | movel	8(%sp),4(%sp) | 
|  | 407 | subw	#BUSY_SIZE,%sp | 
|  | 408 | fsave	(%sp) | 
|  | 409 | fmovel	%fpiar,BUSY_SIZE+EXC_EA-4(%sp) | 
|  | 410 | frestore (%sp) | 
|  | 411 | addw	#BUSY_SIZE,%sp | 
|  | 412 |  | 
|  | 413 | g_easy: | 
|  | 414 | movew	#TRACE_VEC,EXC_VEC-4(%a7) | 
|  | 415 | bral	real_trace | 
|  | 416 | | | 
|  | 417 | |  This is a work-around for hardware bug 1384. | 
|  | 418 | | | 
|  | 419 | bug1384: | 
|  | 420 | link	%a5,#0 | 
|  | 421 | fsave	-(%sp) | 
|  | 422 | cmpib	#0x41,(%sp)	| check for correct frame | 
|  | 423 | beq	frame_41 | 
|  | 424 | bgt	nofix		| if more advanced mask, do nada | 
|  | 425 |  | 
|  | 426 | frame_40: | 
|  | 427 | tstb	1(%sp)		| check to see if idle | 
|  | 428 | bne	notidle | 
|  | 429 | idle40: | 
|  | 430 | clrl	(%sp)		| get rid of old fsave frame | 
|  | 431 | movel  %d1,USER_D1(%a6)  | save d1 | 
|  | 432 | movew	#8,%d1		| place unimp frame instead | 
|  | 433 | loop40:	clrl	-(%sp) | 
|  | 434 | dbra	%d1,loop40 | 
|  | 435 | movel  USER_D1(%a6),%d1  | restore d1 | 
|  | 436 | movel	#0x40280000,-(%sp) | 
|  | 437 | frestore (%sp)+ | 
|  | 438 | unlk	%a5 | 
|  | 439 | rts | 
|  | 440 |  | 
|  | 441 | frame_41: | 
|  | 442 | tstb	1(%sp)		| check to see if idle | 
|  | 443 | bne	notidle | 
|  | 444 | idle41: | 
|  | 445 | clrl	(%sp)		| get rid of old fsave frame | 
|  | 446 | movel  %d1,USER_D1(%a6)  | save d1 | 
|  | 447 | movew	#10,%d1		| place unimp frame instead | 
|  | 448 | loop41:	clrl	-(%sp) | 
|  | 449 | dbra	%d1,loop41 | 
|  | 450 | movel  USER_D1(%a6),%d1  | restore d1 | 
|  | 451 | movel	#0x41300000,-(%sp) | 
|  | 452 | frestore (%sp)+ | 
|  | 453 | unlk	%a5 | 
|  | 454 | rts | 
|  | 455 |  | 
|  | 456 | notidle: | 
|  | 457 | bclrb	#etemp15_bit,-40(%a5) | 
|  | 458 | frestore (%sp)+ | 
|  | 459 | unlk	%a5 | 
|  | 460 | rts | 
|  | 461 |  | 
|  | 462 | nofix: | 
|  | 463 | frestore (%sp)+ | 
|  | 464 | unlk	%a5 | 
|  | 465 | rts | 
|  | 466 |  | 
|  | 467 | |end |