| 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 |