xref: /illumos-gate/usr/src/lib/libm/common/m9x/__fex_i386.c (revision 7d0b359ca572cd04474eb1f2ceec5a8ff39e36c9)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright 2011 Nexenta Systems, Inc.  All rights reserved.
24  */
25 /*
26  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
27  * Use is subject to license terms.
28  */
29 
30 #include <stdio.h>
31 #include <unistd.h>
32 #include <stdlib.h>
33 #include <string.h>
34 #include <signal.h>
35 #include <siginfo.h>
36 #include <ucontext.h>
37 #include <thread.h>
38 #include <math.h>
39 #if defined(__SUNPRO_C)
40 #include <sunmath.h>
41 #endif
42 #include <fenv.h>
43 #include "fex_handler.h"
44 #include "fenv_inlines.h"
45 
46 #if defined(__amd64)
47 #define test_sse_hw	1
48 #else
49 /*
50  * The following variable lives in libc on Solaris 10, where it
51  * gets set to a nonzero value at startup time on systems with SSE.
52  */
53 extern int _sse_hw;
54 #define test_sse_hw	_sse_hw
55 #endif
56 
57 static int accrued = 0;
58 static thread_key_t accrued_key;
59 static mutex_t accrued_key_lock = DEFAULTMUTEX;
60 
61 int *
62 __fex_accrued()
63 {
64 	int		*p;
65 
66 	if (thr_main())
67 		return &accrued;
68 	else {
69 		p = NULL;
70 		mutex_lock(&accrued_key_lock);
71 		if (thr_getspecific(accrued_key, (void **)&p) != 0 &&
72 			thr_keycreate(&accrued_key, free) != 0) {
73 			mutex_unlock(&accrued_key_lock);
74 			return NULL;
75 		}
76 		mutex_unlock(&accrued_key_lock);
77 		if (!p) {
78 			if ((p = (int*) malloc(sizeof(int))) == NULL)
79 				return NULL;
80 			if (thr_setspecific(accrued_key, (void *)p) != 0) {
81 				(void)free(p);
82 				return NULL;
83 			}
84 			*p = 0;
85 		}
86 		return p;
87 	}
88 }
89 
90 void
91 __fenv_getfsr(unsigned long *fsr)
92 {
93 	unsigned int	cwsw, mxcsr;
94 
95 	__fenv_getcwsw(&cwsw);
96 	/* clear reserved bits for no particularly good reason */
97 	cwsw &= ~0xe0c00000u;
98 	if (test_sse_hw) {
99 		/* pick up exception flags (excluding denormal operand
100 		   flag) from mxcsr */
101 		__fenv_getmxcsr(&mxcsr);
102 		cwsw |= (mxcsr & 0x3d);
103 	}
104 	cwsw |= *__fex_accrued();
105 	*fsr = cwsw ^ 0x003f0000u;
106 }
107 
108 void
109 __fenv_setfsr(const unsigned long *fsr)
110 {
111 	unsigned int	cwsw, mxcsr;
112 	int				te;
113 
114 	/* save accrued exception flags corresponding to enabled exceptions */
115 	cwsw = (unsigned int)*fsr;
116 	te = __fenv_get_te(cwsw);
117 	*__fex_accrued() = cwsw & te;
118 	cwsw = (cwsw & ~te) ^ 0x003f0000;
119 	if (test_sse_hw) {
120 		/* propagate rounding direction, masks, and exception flags
121 		   (excluding denormal operand mask and flag) to mxcsr */
122 		__fenv_getmxcsr(&mxcsr);
123 		mxcsr = (mxcsr & ~0x7ebd) | ((cwsw >> 13) & 0x6000) |
124 			((cwsw >> 9) & 0x1e80) | (cwsw & 0x3d);
125 		__fenv_setmxcsr(&mxcsr);
126 	}
127 	__fenv_setcwsw(&cwsw);
128 }
129 
130 /* Offsets into the fp environment save area (assumes 32-bit protected mode) */
131 #define CW	0	/* control word */
132 #define SW	1	/* status word */
133 #define TW	2	/* tag word */
134 #define IP	3	/* instruction pointer */
135 #define OP	4	/* opcode */
136 #define EA	5	/* operand address */
137 
138 /* macro for accessing fp registers in the save area */
139 #if defined(__amd64)
140 #define fpreg(u,x)	*(long double *)(10*(x)+(char*)&(u)->uc_mcontext.fpregs.fp_reg_set.fpchip_state.st)
141 #else
142 #define fpreg(u,x)	*(long double *)(10*(x)+(char*)&(u)->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[7])
143 #endif
144 
145 /*
146 *  Fix sip->si_code; the Solaris x86 kernel can get it wrong
147 */
148 void
149 __fex_get_x86_exc(siginfo_t *sip, ucontext_t *uap)
150 {
151 	unsigned	sw, cw;
152 
153 	sw = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.status;
154 #if defined(__amd64)
155 	cw = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.cw;
156 #else
157 	cw = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[CW];
158 #endif
159 	if ((sw & FE_INVALID) && !(cw & (1 << fp_trap_invalid)))
160 		/* store 0 for stack fault, FPE_FLTINV for IEEE invalid op */
161 		sip->si_code = ((sw & 0x40)? 0 : FPE_FLTINV);
162 	else if ((sw & FE_DIVBYZERO) && !(cw & (1 << fp_trap_division)))
163 		sip->si_code = FPE_FLTDIV;
164 	else if ((sw & FE_OVERFLOW) && !(cw & (1 << fp_trap_overflow)))
165 		sip->si_code = FPE_FLTOVF;
166 	else if ((sw & FE_UNDERFLOW) && !(cw & (1 << fp_trap_underflow)))
167 		sip->si_code = FPE_FLTUND;
168 	else if ((sw & FE_INEXACT) && !(cw & (1 << fp_trap_inexact)))
169 		sip->si_code = FPE_FLTRES;
170 	else
171 		sip->si_code = 0;
172 }
173 
174 static enum fp_class_type
175 my_fp_classf(float *x)
176 {
177 	int		i = *(int*)x & ~0x80000000;
178 
179 	if (i < 0x7f800000) {
180 		if (i < 0x00800000)
181 			return ((i == 0)? fp_zero : fp_subnormal);
182 		return fp_normal;
183 	}
184 	else if (i == 0x7f800000)
185 		return fp_infinity;
186 	else if (i & 0x400000)
187 		return fp_quiet;
188 	else
189 		return fp_signaling;
190 }
191 
192 static enum fp_class_type
193 my_fp_class(double *x)
194 {
195 	int		i = *(1+(int*)x) & ~0x80000000;
196 
197 	if (i < 0x7ff00000) {
198 		if (i < 0x00100000)
199 			return (((i | *(int*)x) == 0)? fp_zero : fp_subnormal);
200 		return fp_normal;
201 	}
202 	else if (i == 0x7ff00000 && *(int*)x == 0)
203 		return fp_infinity;
204 	else if (i & 0x80000)
205 		return fp_quiet;
206 	else
207 		return fp_signaling;
208 }
209 
210 static enum fp_class_type
211 my_fp_classl(long double *x)
212 {
213 	int		i = *(2+(int*)x) & 0x7fff;
214 
215 	if (i < 0x7fff) {
216 		if (i < 1) {
217 			if (*(1+(int*)x) < 0) return fp_normal; /* pseudo-denormal */
218 			return (((*(1+(int*)x) | *(int*)x) == 0)?
219 				fp_zero : fp_subnormal);
220 		}
221 		return ((*(1+(int*)x) < 0)? fp_normal :
222 			(enum fp_class_type) -1); /* unsupported format */
223 	}
224 	else if (*(1+(int*)x) == 0x80000000 && *(int*)x == 0)
225 		return fp_infinity;
226 	else if (*(1+(unsigned*)x) >= 0xc0000000)
227 		return fp_quiet;
228 	else if (*(1+(int*)x) < 0)
229 		return fp_signaling;
230 	else
231 		return (enum fp_class_type) -1; /* unsupported format */
232 }
233 
234 /*
235 *  Determine which type of invalid operation exception occurred
236 */
237 enum fex_exception
238 __fex_get_invalid_type(siginfo_t *sip, ucontext_t *uap)
239 {
240 	unsigned			op;
241 	unsigned long			ea;
242 	enum fp_class_type	t1, t2;
243 
244 	/* get the opcode and data address */
245 #if defined(__amd64)
246 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.fop >> 16;
247 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.rdp;
248 #else
249 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[OP] >> 16;
250 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[EA];
251 #endif
252 
253 	/* if the instruction is fld, the source must be snan (it can't be
254 	   an unsupported format, since fldt doesn't raise any exceptions) */
255 	switch (op & 0x7f8) {
256 	case 0x100:
257 	case 0x140:
258 	case 0x180:
259 	case 0x500:
260 	case 0x540:
261 	case 0x580:
262 		return fex_inv_snan;
263 	}
264 
265 	/* otherwise st is one of the operands; see if it's snan */
266 	t1 = my_fp_classl(&fpreg(uap, 0));
267 	if (t1 == fp_signaling)
268 		return fex_inv_snan;
269 	else if (t1 == (enum fp_class_type) -1)
270 		return (enum fex_exception) -1;
271 
272 	/* determine the class of the second operand if there is one */
273 	t2 = fp_normal;
274 	switch (op & 0x7e0) {
275 	case 0x600:
276 	case 0x620:
277 	case 0x640:
278 	case 0x660:
279 	case 0x680:
280 	case 0x6a0:
281 		/* short memory operand */
282 		if (!ea)
283 			return (enum fex_exception) -1;
284 		if (*(short *)ea == 0)
285 			t2 = fp_zero;
286 		break;
287 
288 	case 0x200:
289 	case 0x220:
290 	case 0x240:
291 	case 0x260:
292 	case 0x280:
293 	case 0x2a0:
294 		/* int memory operand */
295 		if (!ea)
296 			return (enum fex_exception) -1;
297 		if (*(int *)ea == 0)
298 			t2 = fp_zero;
299 		break;
300 
301 	case 0x000:
302 	case 0x020:
303 	case 0x040:
304 	case 0x060:
305 	case 0x080:
306 	case 0x0a0:
307 		/* single precision memory operand */
308 		if (!ea)
309 			return (enum fex_exception) -1;
310 		t2 = my_fp_classf((float *)ea);
311 		break;
312 
313 	case 0x400:
314 	case 0x420:
315 	case 0x440:
316 	case 0x460:
317 	case 0x480:
318 	case 0x4a0:
319 		/* double precision memory operand */
320 		if (!ea)
321 			return (enum fex_exception) -1;
322 		t2 = my_fp_class((double *)ea);
323 		break;
324 
325 	case 0x0c0:
326 	case 0x0e0:
327 	case 0x3e0:
328 	case 0x4c0:
329 	case 0x4e0:
330 	case 0x5e0:
331 	case 0x6c0:
332 	case 0x6e0:
333 	case 0x7e0:
334 		/* register operand determined by opcode */
335 		switch (op & 0x7f8) {
336 		case 0x3e0:
337 		case 0x3f8:
338 		case 0x5f0:
339 		case 0x5f8:
340 		case 0x7e0:
341 		case 0x7f8:
342 			/* weed out nonexistent opcodes */
343 			break;
344 
345 		default:
346 			t2 = my_fp_classl(&fpreg(uap, op & 7));
347 		}
348 		break;
349 
350 	case 0x1e0:
351 	case 0x2e0:
352 		/* special forms */
353 		switch (op) {
354 		case 0x1f1: /* fyl2x */
355 		case 0x1f3: /* fpatan */
356 		case 0x1f5: /* fprem1 */
357 		case 0x1f8: /* fprem */
358 		case 0x1f9: /* fyl2xp1 */
359 		case 0x1fd: /* fscale */
360 		case 0x2e9: /* fucompp */
361 			t2 = my_fp_classl(&fpreg(uap, 1));
362 			break;
363 		}
364 		break;
365 	}
366 
367 	/* see if the second op is snan */
368 	if (t2 == fp_signaling)
369 		return fex_inv_snan;
370 	else if (t2 == (enum fp_class_type) -1)
371 		return (enum fex_exception) -1;
372 
373 	/* determine the type of operation */
374 	switch (op & 0x7f8) {
375 	case 0x000:
376 	case 0x020:
377 	case 0x028:
378 	case 0x040:
379 	case 0x060:
380 	case 0x068:
381 	case 0x080:
382 	case 0x0a0:
383 	case 0x0a8:
384 	case 0x0c0:
385 	case 0x0e0:
386 	case 0x0e8:
387 	case 0x400:
388 	case 0x420:
389 	case 0x428:
390 	case 0x440:
391 	case 0x460:
392 	case 0x468:
393 	case 0x480:
394 	case 0x4a0:
395 	case 0x4a8:
396 	case 0x4c0:
397 	case 0x4e0:
398 	case 0x4e8:
399 	case 0x6c0:
400 	case 0x6e0:
401 	case 0x6e8:
402 		/* fadd, fsub, fsubr */
403 		if (t1 == fp_infinity && t2 == fp_infinity)
404 			return fex_inv_isi;
405 		break;
406 
407 	case 0x008:
408 	case 0x048:
409 	case 0x088:
410 	case 0x0c8:
411 	case 0x208:
412 	case 0x248:
413 	case 0x288:
414 	case 0x408:
415 	case 0x448:
416 	case 0x488:
417 	case 0x4c8:
418 	case 0x608:
419 	case 0x648:
420 	case 0x688:
421 	case 0x6c8:
422 		/* fmul */
423 		if ((t1 == fp_zero && t2 == fp_infinity) || (t2 == fp_zero &&
424 		  t1 == fp_infinity))
425 			return fex_inv_zmi;
426 		break;
427 
428 	case 0x030:
429 	case 0x038:
430 	case 0x070:
431 	case 0x078:
432 	case 0x0b0:
433 	case 0x0b8:
434 	case 0x0f0:
435 	case 0x0f8:
436 	case 0x230:
437 	case 0x238:
438 	case 0x270:
439 	case 0x278:
440 	case 0x2b0:
441 	case 0x2b8:
442 	case 0x430:
443 	case 0x438:
444 	case 0x470:
445 	case 0x478:
446 	case 0x4b0:
447 	case 0x4b8:
448 	case 0x4f0:
449 	case 0x4f8:
450 	case 0x630:
451 	case 0x638:
452 	case 0x670:
453 	case 0x678:
454 	case 0x6b0:
455 	case 0x6b8:
456 	case 0x6f0:
457 	case 0x6f8:
458 		/* fdiv */
459 		if (t1 == fp_zero && t2 == fp_zero)
460 			return fex_inv_zdz;
461 		else if (t1 == fp_infinity && t2 == fp_infinity)
462 			return fex_inv_idi;
463 		break;
464 
465 	case 0x1f0:
466 	case 0x1f8:
467 		/* fsqrt, other special ops */
468 		return fex_inv_sqrt;
469 
470 	case 0x010:
471 	case 0x018:
472 	case 0x050:
473 	case 0x058:
474 	case 0x090:
475 	case 0x098:
476 	case 0x0d0:
477 	case 0x0d8:
478 	case 0x210:
479 	case 0x218:
480 	case 0x250:
481 	case 0x258:
482 	case 0x290:
483 	case 0x298:
484 	case 0x2e8:
485 	case 0x3f0:
486 	case 0x410:
487 	case 0x418:
488 	case 0x450:
489 	case 0x458:
490 	case 0x490:
491 	case 0x498:
492 	case 0x4d0:
493 	case 0x4d8:
494 	case 0x5e0:
495 	case 0x5e8:
496 	case 0x610:
497 	case 0x618:
498 	case 0x650:
499 	case 0x658:
500 	case 0x690:
501 	case 0x698:
502 	case 0x6d0:
503 	case 0x6d8:
504 	case 0x7f0:
505 		/* fcom */
506 		if (t1 == fp_quiet || t2 == fp_quiet)
507 			return fex_inv_cmp;
508 		break;
509 
510 	case 0x1e0:
511 		/* ftst */
512 		if (op == 0x1e4 && t1 == fp_quiet)
513 			return fex_inv_cmp;
514 		break;
515 
516 	case 0x310:
517 	case 0x318:
518 	case 0x350:
519 	case 0x358:
520 	case 0x390:
521 	case 0x398:
522 	case 0x710:
523 	case 0x718:
524 	case 0x730:
525 	case 0x738:
526 	case 0x750:
527 	case 0x758:
528 	case 0x770:
529 	case 0x778:
530 	case 0x790:
531 	case 0x798:
532 	case 0x7b0:
533 	case 0x7b8:
534 		/* fist, fbst */
535 		return fex_inv_int;
536 	}
537 
538 	return (enum fex_exception) -1;
539 }
540 
541 /* scale factors for exponent unwrapping */
542 static const long double
543 	two12288 = 1.139165225263043370845938579315932009e+3699l,	/* 2^12288 */
544 	twom12288 = 8.778357852076208839765066529179033145e-3700l,	/* 2^-12288 */
545 	twom12288mulp = 8.778357852076208839289190796475222545e-3700l;
546 		/* (")*(1-2^-64) */
547 
548 /* inline templates */
549 extern long double f2xm1(long double);
550 extern long double fyl2x(long double, long double);
551 extern long double fptan(long double);
552 extern long double fpatan(long double, long double);
553 extern long double fxtract(long double);
554 extern long double fprem1(long double, long double);
555 extern long double fprem(long double, long double);
556 extern long double fyl2xp1(long double, long double);
557 extern long double fsqrt(long double);
558 extern long double fsincos(long double);
559 extern long double frndint(long double);
560 extern long double fscale(long double, long double);
561 extern long double fsin(long double);
562 extern long double fcos(long double);
563 
564 /*
565 *  Get the operands, generate the default untrapped result with
566 *  exceptions, and set a code indicating the type of operation
567 */
568 void
569 __fex_get_op(siginfo_t *sip, ucontext_t *uap, fex_info_t *info)
570 {
571 	fex_numeric_t			t;
572 	long double			op2v, x;
573 	unsigned int			cwsw, ex, sw, op;
574 	unsigned long			ea;
575 	volatile int			c;
576 
577 	/* get the exception type, status word, opcode, and data address */
578 	ex = sip->si_code;
579 	sw = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.status;
580 #if defined(__amd64)
581 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.fop >> 16;
582 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.rdp;
583 #else
584 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[OP] >> 16;
585 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[EA];
586 #endif
587 
588 	/* initialize res to the default untrapped result and ex to the
589 	   corresponding flags (assume trapping is disabled and flags
590 	   are clear) */
591 
592 	/* single operand instructions */
593 	info->op = fex_cnvt;
594 	info->op2.type = fex_nodata;
595 	switch (op & 0x7f8) {
596 	/* load instructions */
597 	case 0x100:
598 	case 0x140:
599 	case 0x180:
600 		if (!ea) {
601 			info->op = fex_other;
602 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
603 			info->flags = 0;
604 			return;
605 		}
606 		info->op1.type = fex_float;
607 		info->op1.val.f = *(float *)ea;
608 		info->res.type = fex_ldouble;
609 		info->res.val.q = (long double) info->op1.val.f;
610 		goto done;
611 
612 	case 0x500:
613 	case 0x540:
614 	case 0x580:
615 		if (!ea) {
616 			info->op = fex_other;
617 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
618 			info->flags = 0;
619 			return;
620 		}
621 		info->op1.type = fex_double;
622 		info->op1.val.d = *(double *)ea;
623 		info->res.type = fex_ldouble;
624 		info->res.val.q = (long double) info->op1.val.d;
625 		goto done;
626 
627 	/* store instructions */
628 	case 0x110:
629 	case 0x118:
630 	case 0x150:
631 	case 0x158:
632 	case 0x190:
633 	case 0x198:
634 		info->res.type = fex_float;
635 		if (ex == FPE_FLTRES && (op & 8) != 0) {
636 			/* inexact, stack popped */
637 			if (!ea) {
638 				info->op = fex_other;
639 				info->op1.type = info->op2.type = info->res.type = fex_nodata;
640 				info->flags = 0;
641 				return;
642 			}
643 			info->op1.type = fex_nodata;
644 			info->res.val.f = *(float *)ea;
645 			info->flags = FE_INEXACT;
646 			return;
647 		}
648 		info->op1.type = fex_ldouble;
649 		info->op1.val.q = fpreg(uap, 0);
650 		info->res.val.f = (float) info->op1.val.q;
651 		goto done;
652 
653 	case 0x310:
654 	case 0x318:
655 	case 0x350:
656 	case 0x358:
657 	case 0x390:
658 	case 0x398:
659 		info->res.type = fex_int;
660 		if (ex == FPE_FLTRES && (op & 8) != 0) {
661 			/* inexact, stack popped */
662 			if (!ea) {
663 				info->op = fex_other;
664 				info->op1.type = info->op2.type = info->res.type = fex_nodata;
665 				info->flags = 0;
666 				return;
667 			}
668 			info->op1.type = fex_nodata;
669 			info->res.val.i = *(int *)ea;
670 			info->flags = FE_INEXACT;
671 			return;
672 		}
673 		info->op1.type = fex_ldouble;
674 		info->op1.val.q = fpreg(uap, 0);
675 		info->res.val.i = (int) info->op1.val.q;
676 		goto done;
677 
678 	case 0x510:
679 	case 0x518:
680 	case 0x550:
681 	case 0x558:
682 	case 0x590:
683 	case 0x598:
684 		info->res.type = fex_double;
685 		if (ex == FPE_FLTRES && (op & 8) != 0) {
686 			/* inexact, stack popped */
687 			if (!ea) {
688 				info->op = fex_other;
689 				info->op1.type = info->op2.type = info->res.type = fex_nodata;
690 				info->flags = 0;
691 				return;
692 			}
693 			info->op1.type = fex_nodata;
694 			info->res.val.d = *(double *)ea;
695 			info->flags = FE_INEXACT;
696 			return;
697 		}
698 		info->op1.type = fex_ldouble;
699 		info->op1.val.q = fpreg(uap, 0);
700 		info->res.val.d = (double) info->op1.val.q;
701 		goto done;
702 
703 	case 0x710:
704 	case 0x718:
705 	case 0x750:
706 	case 0x758:
707 	case 0x790:
708 	case 0x798:
709 		info->res.type = fex_int;
710 		if (ex == FPE_FLTRES && (op & 8) != 0) {
711 			/* inexact, stack popped */
712 			if (!ea) {
713 				info->op = fex_other;
714 				info->op1.type = info->op2.type = info->res.type = fex_nodata;
715 				info->flags = 0;
716 				return;
717 			}
718 			info->op1.type = fex_nodata;
719 			info->res.val.i = *(short *)ea;
720 			info->flags = FE_INEXACT;
721 			return;
722 		}
723 		info->op1.type = fex_ldouble;
724 		info->op1.val.q = fpreg(uap, 0);
725 		info->res.val.i = (short) info->op1.val.q;
726 		goto done;
727 
728 	case 0x730:
729 	case 0x770:
730 	case 0x7b0:
731 		/* fbstp; don't bother */
732 		info->op = fex_other;
733 		info->op1.type = info->res.type = fex_nodata;
734 		info->flags = 0;
735 		return;
736 
737 	case 0x738:
738 	case 0x778:
739 	case 0x7b8:
740 		info->res.type = fex_llong;
741 		if (ex == FPE_FLTRES) {
742 			/* inexact, stack popped */
743 			if (!ea) {
744 				info->op = fex_other;
745 				info->op1.type = info->op2.type = info->res.type = fex_nodata;
746 				info->flags = 0;
747 				return;
748 			}
749 			info->op1.type = fex_nodata;
750 			info->res.val.l = *(long long *)ea;
751 			info->flags = FE_INEXACT;
752 			return;
753 		}
754 		info->op1.type = fex_ldouble;
755 		info->op1.val.q = fpreg(uap, 0);
756 		info->res.val.l = (long long) info->op1.val.q;
757 		goto done;
758 	}
759 
760 	/* all other ops (except compares) have destinations on the stack
761 	   so overflow, underflow, and inexact will stomp their operands */
762 	if (ex == FPE_FLTOVF || ex == FPE_FLTUND || ex == FPE_FLTRES) {
763 		/* find the trapped result */
764 		info->op1.type = info->op2.type = fex_nodata;
765 		info->res.type = fex_ldouble;
766 		switch (op & 0x7f8) {
767 		case 0x1f0:
768 			/* fptan pushes 1.0 afterward, so result is in st(1) */
769 			info->res.val.q = ((op == 0x1f2)? fpreg(uap, 1) :
770 				fpreg(uap, 0));
771 			break;
772 
773 		case 0x4c0:
774 		case 0x4c8:
775 		case 0x4e0:
776 		case 0x4e8:
777 		case 0x4f0:
778 		case 0x4f8:
779 			info->res.val.q = fpreg(uap, op & 7);
780 			break;
781 
782 		case 0x6c0:
783 		case 0x6c8:
784 		case 0x6e0:
785 		case 0x6e8:
786 		case 0x6f0:
787 		case 0x6f8:
788 			/* stack was popped afterward */
789 			info->res.val.q = fpreg(uap, (op - 1) & 7);
790 			break;
791 
792 		default:
793 			info->res.val.q = fpreg(uap, 0);
794 		}
795 
796 		/* reconstruct default untrapped result */
797 		if (ex == FPE_FLTOVF) {
798 			/* generate an overflow with the sign of the result */
799 			x = two12288;
800 			*(4+(short*)&x) |= (*(4+(short*)&info->res.val.q) & 0x8000);
801 			info->res.val.q = x * two12288;
802 			info->flags = FE_OVERFLOW | FE_INEXACT;
803 			__fenv_getcwsw(&cwsw);
804 			cwsw &= ~FE_ALL_EXCEPT;
805 			__fenv_setcwsw(&cwsw);
806 		}
807 		else if (ex == FPE_FLTUND) {
808 			/* undo the scaling; we can't distinguish a chopped result
809 			   from an exact one without futzing around to trap all in-
810 			   exact exceptions so as to keep the flag clear, so we just
811 			   punt */
812 			if (sw & 0x200) /* result was rounded up */
813 				info->res.val.q = (info->res.val.q * twom12288) * twom12288mulp;
814 			else
815 				info->res.val.q = (info->res.val.q * twom12288) * twom12288;
816 			__fenv_getcwsw(&cwsw);
817 			info->flags = (cwsw & FE_INEXACT) | FE_UNDERFLOW;
818 			cwsw &= ~FE_ALL_EXCEPT;
819 			__fenv_setcwsw(&cwsw);
820 		}
821 		else
822 			info->flags = FE_INEXACT;
823 
824 		/* determine the operation code */
825 		switch (op) {
826 		case 0x1f0: /* f2xm1 */
827 		case 0x1f1: /* fyl2x */
828 		case 0x1f2: /* fptan */
829 		case 0x1f3: /* fpatan */
830 		case 0x1f5: /* fprem1 */
831 		case 0x1f8: /* fprem */
832 		case 0x1f9: /* fyl2xp1 */
833 		case 0x1fb: /* fsincos */
834 		case 0x1fc: /* frndint */
835 		case 0x1fd: /* fscale */
836 		case 0x1fe: /* fsin */
837 		case 0x1ff: /* fcos */
838 			info->op = fex_other;
839 			return;
840 
841 		case 0x1fa: /* fsqrt */
842 			info->op = fex_sqrt;
843 			return;
844 		}
845 
846 		info->op = fex_other;
847 		switch (op & 0x7c0) {
848 		case 0x000:
849 		case 0x040:
850 		case 0x080:
851 		case 0x0c0:
852 		case 0x200:
853 		case 0x240:
854 		case 0x280:
855 		case 0x400:
856 		case 0x440:
857 		case 0x480:
858 		case 0x4c0:
859 		case 0x600:
860 		case 0x640:
861 		case 0x680:
862 		case 0x6c0:
863 			switch (op & 0x38) {
864 			case 0x00:
865 				info->op = fex_add;
866 				break;
867 
868 			case 0x08:
869 				info->op = fex_mul;
870 				break;
871 
872 			case 0x20:
873 			case 0x28:
874 				info->op = fex_sub;
875 				break;
876 
877 			case 0x30:
878 			case 0x38:
879 				info->op = fex_div;
880 				break;
881 			}
882 		}
883 		return;
884 	}
885 
886 	/* for other exceptions, the operands are preserved, so we can
887 	   just emulate the operation with traps disabled */
888 
889 	/* one operand is always in st */
890 	info->op1.type = fex_ldouble;
891 	info->op1.val.q = fpreg(uap, 0);
892 
893 	/* oddball instructions */
894 	info->op = fex_other;
895 	switch (op) {
896 	case 0x1e4: /* ftst */
897 		info->op = fex_cmp;
898 		info->op2.type = fex_ldouble;
899 		info->op2.val.q = 0.0l;
900 		info->res.type = fex_nodata;
901 		c = (info->op1.val.q < info->op2.val.q);
902 		goto done;
903 
904 	case 0x1f0: /* f2xm1 */
905 		info->res.type = fex_ldouble;
906 		info->res.val.q = f2xm1(info->op1.val.q);
907 		goto done;
908 
909 	case 0x1f1: /* fyl2x */
910 		info->op2.type = fex_ldouble;
911 		info->op2.val.q = fpreg(uap, 1);
912 		info->res.type = fex_ldouble;
913 		info->res.val.q = fyl2x(info->op1.val.q, info->op2.val.q);
914 		goto done;
915 
916 	case 0x1f2: /* fptan */
917 		info->res.type = fex_ldouble;
918 		info->res.val.q = fptan(info->op1.val.q);
919 		goto done;
920 
921 	case 0x1f3: /* fpatan */
922 		info->op2.type = fex_ldouble;
923 		info->op2.val.q = fpreg(uap, 1);
924 		info->res.type = fex_ldouble;
925 		info->res.val.q = fpatan(info->op1.val.q, info->op2.val.q);
926 		goto done;
927 
928 	case 0x1f4: /* fxtract */
929 		info->res.type = fex_ldouble;
930 		info->res.val.q = fxtract(info->op1.val.q);
931 		goto done;
932 
933 	case 0x1f5: /* fprem1 */
934 		info->op2.type = fex_ldouble;
935 		info->op2.val.q = fpreg(uap, 1);
936 		info->res.type = fex_ldouble;
937 		info->res.val.q = fprem1(info->op1.val.q, info->op2.val.q);
938 		goto done;
939 
940 	case 0x1f8: /* fprem */
941 		info->op2.type = fex_ldouble;
942 		info->op2.val.q = fpreg(uap, 1);
943 		info->res.type = fex_ldouble;
944 		info->res.val.q = fprem(info->op1.val.q, info->op2.val.q);
945 		goto done;
946 
947 	case 0x1f9: /* fyl2xp1 */
948 		info->op2.type = fex_ldouble;
949 		info->op2.val.q = fpreg(uap, 1);
950 		info->res.type = fex_ldouble;
951 		info->res.val.q = fyl2xp1(info->op1.val.q, info->op2.val.q);
952 		goto done;
953 
954 	case 0x1fa: /* fsqrt */
955 		info->op = fex_sqrt;
956 		info->res.type = fex_ldouble;
957 		info->res.val.q = fsqrt(info->op1.val.q);
958 		goto done;
959 
960 	case 0x1fb: /* fsincos */
961 		info->res.type = fex_ldouble;
962 		info->res.val.q = fsincos(info->op1.val.q);
963 		goto done;
964 
965 	case 0x1fc: /* frndint */
966 		info->res.type = fex_ldouble;
967 		info->res.val.q = frndint(info->op1.val.q);
968 		goto done;
969 
970 	case 0x1fd: /* fscale */
971 		info->op2.type = fex_ldouble;
972 		info->op2.val.q = fpreg(uap, 1);
973 		info->res.type = fex_ldouble;
974 		info->res.val.q = fscale(info->op1.val.q, info->op2.val.q);
975 		goto done;
976 
977 	case 0x1fe: /* fsin */
978 		info->res.type = fex_ldouble;
979 		info->res.val.q = fsin(info->op1.val.q);
980 		goto done;
981 
982 	case 0x1ff: /* fcos */
983 		info->res.type = fex_ldouble;
984 		info->res.val.q = fcos(info->op1.val.q);
985 		goto done;
986 
987 	case 0x2e9: /* fucompp */
988 		info->op = fex_cmp;
989 		info->op2.type = fex_ldouble;
990 		info->op2.val.q = fpreg(uap, 1);
991 		info->res.type = fex_nodata;
992 		c = (info->op1.val.q == info->op2.val.q);
993 		goto done;
994 	}
995 
996 	/* fucom[p], fcomi[p], fucomi[p] */
997 	switch (op & 0x7f8) {
998 	case 0x3e8:
999 	case 0x5e0:
1000 	case 0x5e8:
1001 	case 0x7e8: /* unordered compares */
1002 		info->op = fex_cmp;
1003 		info->op2.type = fex_ldouble;
1004 		info->op2.val.q = fpreg(uap, op & 7);
1005 		info->res.type = fex_nodata;
1006 		c = (info->op1.val.q == info->op2.val.q);
1007 		goto done;
1008 
1009 	case 0x3f0:
1010 	case 0x7f0: /* ordered compares */
1011 		info->op = fex_cmp;
1012 		info->op2.type = fex_ldouble;
1013 		info->op2.val.q = fpreg(uap, op & 7);
1014 		info->res.type = fex_nodata;
1015 		c = (info->op1.val.q < info->op2.val.q);
1016 		goto done;
1017 	}
1018 
1019 	/* all other instructions come in groups of the form
1020 	   fadd, fmul, fcom, fcomp, fsub, fsubr, fdiv, fdivr */
1021 
1022 	/* get the second operand */
1023 	switch (op & 0x7c0) {
1024 	case 0x000:
1025 	case 0x040:
1026 	case 0x080:
1027 		if (!ea) {
1028 			info->op = fex_other;
1029 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
1030 			info->flags = 0;
1031 			return;
1032 		}
1033 		info->op2.type = fex_float;
1034 		info->op2.val.f = *(float *)ea;
1035 		op2v = (long double) info->op2.val.f;
1036 		break;
1037 
1038 	case 0x0c0:
1039 		info->op2.type = fex_ldouble;
1040 		op2v = info->op2.val.q = fpreg(uap, op & 7);
1041 		break;
1042 
1043 	case 0x200:
1044 	case 0x240:
1045 	case 0x280:
1046 		if (!ea) {
1047 			info->op = fex_other;
1048 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
1049 			info->flags = 0;
1050 			return;
1051 		}
1052 		info->op2.type = fex_int;
1053 		info->op2.val.i = *(int *)ea;
1054 		op2v = (long double) info->op2.val.i;
1055 		break;
1056 
1057 	case 0x400:
1058 	case 0x440:
1059 	case 0x480:
1060 		if (!ea) {
1061 			info->op = fex_other;
1062 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
1063 			info->flags = 0;
1064 			return;
1065 		}
1066 		info->op2.type = fex_double;
1067 		info->op2.val.d = *(double *)ea;
1068 		op2v = (long double) info->op2.val.d;
1069 		break;
1070 
1071 	case 0x4c0:
1072 	case 0x6c0:
1073 		info->op2.type = fex_ldouble;
1074 		info->op2.val.q = fpreg(uap, op & 7);
1075 		t = info->op1;
1076 		info->op1 = info->op2;
1077 		info->op2 = t;
1078 		op2v = info->op2.val.q;
1079 		break;
1080 
1081 	case 0x600:
1082 	case 0x640:
1083 	case 0x680:
1084 		if (!ea) {
1085 			info->op = fex_other;
1086 			info->op1.type = info->op2.type = info->res.type = fex_nodata;
1087 			info->flags = 0;
1088 			return;
1089 		}
1090 		info->op2.type = fex_int;
1091 		info->op2.val.i = *(short *)ea;
1092 		op2v = (long double) info->op2.val.i;
1093 		break;
1094 
1095 	default:
1096 		info->op = fex_other;
1097 		info->op1.type = info->op2.type = info->res.type = fex_nodata;
1098 		info->flags = 0;
1099 		return;
1100 	}
1101 
1102 	/* distinguish different operations in the group */
1103 	info->res.type = fex_ldouble;
1104 	switch (op & 0x38) {
1105 	case 0x00:
1106 		info->op = fex_add;
1107 		info->res.val.q = info->op1.val.q + op2v;
1108 		break;
1109 
1110 	case 0x08:
1111 		info->op = fex_mul;
1112 		info->res.val.q = info->op1.val.q * op2v;
1113 		break;
1114 
1115 	case 0x10:
1116 	case 0x18:
1117 		info->op = fex_cmp;
1118 		info->res.type = fex_nodata;
1119 		c = (info->op1.val.q < op2v);
1120 		break;
1121 
1122 	case 0x20:
1123 		info->op = fex_sub;
1124 		info->res.val.q = info->op1.val.q - op2v;
1125 		break;
1126 
1127 	case 0x28:
1128 		info->op = fex_sub;
1129 		info->res.val.q = op2v - info->op1.val.q;
1130 		t = info->op1;
1131 		info->op1 = info->op2;
1132 		info->op2 = t;
1133 		break;
1134 
1135 	case 0x30:
1136 		info->op = fex_div;
1137 		info->res.val.q = info->op1.val.q / op2v;
1138 		break;
1139 
1140 	case 0x38:
1141 		info->op = fex_div;
1142 		info->res.val.q = op2v / info->op1.val.q;
1143 		t = info->op1;
1144 		info->op1 = info->op2;
1145 		info->op2 = t;
1146 		break;
1147 
1148 	default:
1149 		info->op = fex_other;
1150 		info->op1.type = info->op2.type = info->res.type = fex_nodata;
1151 		info->flags = 0;
1152 		return;
1153 	}
1154 
1155 done:
1156 	__fenv_getcwsw(&cwsw);
1157 	info->flags = cwsw & FE_ALL_EXCEPT;
1158 	cwsw &= ~FE_ALL_EXCEPT;
1159 	__fenv_setcwsw(&cwsw);
1160 }
1161 
1162 /* pop the saved stack */
1163 static void pop(ucontext_t *uap)
1164 {
1165 	unsigned top;
1166 
1167 	fpreg(uap, 0) = fpreg(uap, 1);
1168 	fpreg(uap, 1) = fpreg(uap, 2);
1169 	fpreg(uap, 2) = fpreg(uap, 3);
1170 	fpreg(uap, 3) = fpreg(uap, 4);
1171 	fpreg(uap, 4) = fpreg(uap, 5);
1172 	fpreg(uap, 5) = fpreg(uap, 6);
1173 	fpreg(uap, 6) = fpreg(uap, 7);
1174 #if defined(__amd64)
1175 	top = (uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw >> 10)
1176 		& 0xe;
1177 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.fctw |= (3 << top);
1178 	top = (top + 2) & 0xe;
1179 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw =
1180 		(uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw & ~0x3800)
1181 		| (top << 10);
1182 #else
1183 	top = (uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] >> 10)
1184 		& 0xe;
1185 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[TW] |= (3 << top);
1186 	top = (top + 2) & 0xe;
1187 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] =
1188 		(uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] & ~0x3800)
1189 		| (top << 10);
1190 #endif
1191 }
1192 
1193 /* push x onto the saved stack */
1194 static void push(long double x, ucontext_t *uap)
1195 {
1196 	unsigned top;
1197 
1198 	fpreg(uap, 7) = fpreg(uap, 6);
1199 	fpreg(uap, 6) = fpreg(uap, 5);
1200 	fpreg(uap, 5) = fpreg(uap, 4);
1201 	fpreg(uap, 4) = fpreg(uap, 3);
1202 	fpreg(uap, 3) = fpreg(uap, 2);
1203 	fpreg(uap, 2) = fpreg(uap, 1);
1204 	fpreg(uap, 1) = fpreg(uap, 0);
1205 	fpreg(uap, 0) = x;
1206 #if defined(__amd64)
1207 	top = (uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw >> 10)
1208 		& 0xe;
1209 	top = (top - 2) & 0xe;
1210 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.fctw &= ~(3 << top);
1211 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw =
1212 		(uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw & ~0x3800)
1213 		| (top << 10);
1214 #else
1215 	top = (uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] >> 10)
1216 		& 0xe;
1217 	top = (top - 2) & 0xe;
1218 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[TW] &= ~(3 << top);
1219 	uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] =
1220 		(uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] & ~0x3800)
1221 		| (top << 10);
1222 #endif
1223 }
1224 
1225 /* scale factors for exponent wrapping */
1226 static const float
1227 	fun = 7.922816251e+28f,	/* 2^96 */
1228 	fov = 1.262177448e-29f;	/* 2^-96 */
1229 static const double
1230 	dun = 1.552518092300708935e+231,	/* 2^768 */
1231 	dov = 6.441148769597133308e-232;	/* 2^-768 */
1232 
1233 /*
1234 *  Store the specified result; if no result is given but the exception
1235 *  is underflow or overflow, use the default trapped result
1236 */
1237 void
1238 __fex_st_result(siginfo_t *sip, ucontext_t *uap, fex_info_t *info)
1239 {
1240 	fex_numeric_t	r;
1241 	unsigned long		ex, op, ea, stack;
1242 
1243 	/* get the exception type, opcode, and data address */
1244 	ex = sip->si_code;
1245 #if defined(__amd64)
1246 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.fop >> 16;
1247 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.rdp; /*???*/
1248 #else
1249 	op = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[OP] >> 16;
1250 	ea = uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[EA];
1251 #endif
1252 
1253 	/* if the instruction is a compare, set the condition codes
1254 	   to unordered and update the stack */
1255 	switch (op & 0x7f8) {
1256 	case 0x010:
1257 	case 0x050:
1258 	case 0x090:
1259 	case 0x0d0:
1260 	case 0x210:
1261 	case 0x250:
1262 	case 0x290:
1263 	case 0x410:
1264 	case 0x450:
1265 	case 0x490:
1266 	case 0x4d0:
1267 	case 0x5e0:
1268 	case 0x610:
1269 	case 0x650:
1270 	case 0x690:
1271 		/* f[u]com */
1272 #if defined(__amd64)
1273 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw |= 0x4500;
1274 #else
1275 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] |= 0x4500;
1276 #endif
1277 		return;
1278 
1279 	case 0x018:
1280 	case 0x058:
1281 	case 0x098:
1282 	case 0x0d8:
1283 	case 0x218:
1284 	case 0x258:
1285 	case 0x298:
1286 	case 0x418:
1287 	case 0x458:
1288 	case 0x498:
1289 	case 0x4d8:
1290 	case 0x5e8:
1291 	case 0x618:
1292 	case 0x658:
1293 	case 0x698:
1294 	case 0x6d0:
1295 		/* f[u]comp */
1296 #if defined(__amd64)
1297 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw |= 0x4500;
1298 #else
1299 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] |= 0x4500;
1300 #endif
1301 		pop(uap);
1302 		return;
1303 
1304 	case 0x2e8:
1305 	case 0x6d8:
1306 		/* f[u]compp */
1307 #if defined(__amd64)
1308 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw |= 0x4500;
1309 #else
1310 		uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] |= 0x4500;
1311 #endif
1312 		pop(uap);
1313 		pop(uap);
1314 		return;
1315 
1316 	case 0x1e0:
1317 		if (op == 0x1e4) { /* ftst */
1318 #if defined(__amd64)
1319 			uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.sw |= 0x4500;
1320 #else
1321 			uap->uc_mcontext.fpregs.fp_reg_set.fpchip_state.state[SW] |= 0x4500;
1322 #endif
1323 			return;
1324 		}
1325 		break;
1326 
1327 	case 0x3e8:
1328 	case 0x3f0:
1329 		/* f[u]comi */
1330 #if defined(__amd64)
1331 		uap->uc_mcontext.gregs[REG_PS] |= 0x45;
1332 #else
1333 		uap->uc_mcontext.gregs[EFL] |= 0x45;
1334 #endif
1335 		return;
1336 
1337 	case 0x7e8:
1338 	case 0x7f0:
1339 		/* f[u]comip */
1340 #if defined(__amd64)
1341 		uap->uc_mcontext.gregs[REG_PS] |= 0x45;
1342 #else
1343 		uap->uc_mcontext.gregs[EFL] |= 0x45;
1344 #endif
1345 		pop(uap);
1346 		return;
1347 	}
1348 
1349 	/* if there is no result available and the exception is overflow
1350 	   or underflow, use the wrapped result */
1351 	r = info->res;
1352 	if (r.type == fex_nodata) {
1353 		if (ex == FPE_FLTOVF || ex == FPE_FLTUND) {
1354 			/* for store instructions, do the scaling and store */
1355 			switch (op & 0x7f8) {
1356 			case 0x110:
1357 			case 0x118:
1358 			case 0x150:
1359 			case 0x158:
1360 			case 0x190:
1361 			case 0x198:
1362 				if (!ea)
1363 					return;
1364 				if (ex == FPE_FLTOVF)
1365 					*(float *)ea = (fpreg(uap, 0) * fov) * fov;
1366 				else
1367 					*(float *)ea = (fpreg(uap, 0) * fun) * fun;
1368 				if ((op & 8) != 0)
1369 					pop(uap);
1370 				break;
1371 
1372 			case 0x510:
1373 			case 0x518:
1374 			case 0x550:
1375 			case 0x558:
1376 			case 0x590:
1377 			case 0x598:
1378 				if (!ea)
1379 					return;
1380 				if (ex == FPE_FLTOVF)
1381 					*(double *)ea = (fpreg(uap, 0) * dov) * dov;
1382 				else
1383 					*(double *)ea = (fpreg(uap, 0) * dun) * dun;
1384 				if ((op & 8) != 0)
1385 					pop(uap);
1386 				break;
1387 			}
1388 		}
1389 #ifdef DEBUG
1390 		else if (ex != FPE_FLTRES)
1391 			printf("No result supplied, stack may be hosed\n");
1392 #endif
1393 		return;
1394 	}
1395 
1396 	/* otherwise convert the supplied result to the correct type,
1397 	   put it in the destination, and update the stack as need be */
1398 
1399 	/* store instructions */
1400 	switch (op & 0x7f8) {
1401 	case 0x110:
1402 	case 0x118:
1403 	case 0x150:
1404 	case 0x158:
1405 	case 0x190:
1406 	case 0x198:
1407 		if (!ea)
1408 			return;
1409 		switch (r.type) {
1410 		case fex_int:
1411 			*(float *)ea = (float) r.val.i;
1412 			break;
1413 
1414 		case fex_llong:
1415 			*(float *)ea = (float) r.val.l;
1416 			break;
1417 
1418 		case fex_float:
1419 			*(float *)ea = r.val.f;
1420 			break;
1421 
1422 		case fex_double:
1423 			*(float *)ea = (float) r.val.d;
1424 			break;
1425 
1426 		case fex_ldouble:
1427 			*(float *)ea = (float) r.val.q;
1428 			break;
1429 
1430 		default:
1431 			break;
1432 		}
1433 		if (ex != FPE_FLTRES && (op & 8) != 0)
1434 			pop(uap);
1435 		return;
1436 
1437 	case 0x310:
1438 	case 0x318:
1439 	case 0x350:
1440 	case 0x358:
1441 	case 0x390:
1442 	case 0x398:
1443 		if (!ea)
1444 			return;
1445 		switch (r.type) {
1446 		case fex_int:
1447 			*(int *)ea = r.val.i;
1448 			break;
1449 
1450 		case fex_llong:
1451 			*(int *)ea = (int) r.val.l;
1452 			break;
1453 
1454 		case fex_float:
1455 			*(int *)ea = (int) r.val.f;
1456 			break;
1457 
1458 		case fex_double:
1459 			*(int *)ea = (int) r.val.d;
1460 			break;
1461 
1462 		case fex_ldouble:
1463 			*(int *)ea = (int) r.val.q;
1464 			break;
1465 
1466 		default:
1467 			break;
1468 		}
1469 		if (ex != FPE_FLTRES && (op & 8) != 0)
1470 			pop(uap);
1471 		return;
1472 
1473 	case 0x510:
1474 	case 0x518:
1475 	case 0x550:
1476 	case 0x558:
1477 	case 0x590:
1478 	case 0x598:
1479 		if (!ea)
1480 			return;
1481 		switch (r.type) {
1482 		case fex_int:
1483 			*(double *)ea = (double) r.val.i;
1484 			break;
1485 
1486 		case fex_llong:
1487 			*(double *)ea = (double) r.val.l;
1488 			break;
1489 
1490 		case fex_float:
1491 			*(double *)ea = (double) r.val.f;
1492 			break;
1493 
1494 		case fex_double:
1495 			*(double *)ea = r.val.d;
1496 			break;
1497 
1498 		case fex_ldouble:
1499 			*(double *)ea = (double) r.val.q;
1500 			break;
1501 
1502 		default:
1503 			break;
1504 		}
1505 		if (ex != FPE_FLTRES && (op & 8) != 0)
1506 			pop(uap);
1507 		return;
1508 
1509 	case 0x710:
1510 	case 0x718:
1511 	case 0x750:
1512 	case 0x758:
1513 	case 0x790:
1514 	case 0x798:
1515 		if (!ea)
1516 			return;
1517 		switch (r.type) {
1518 		case fex_int:
1519 			*(short *)ea = (short) r.val.i;
1520 			break;
1521 
1522 		case fex_llong:
1523 			*(short *)ea = (short) r.val.l;
1524 			break;
1525 
1526 		case fex_float:
1527 			*(short *)ea = (short) r.val.f;
1528 			break;
1529 
1530 		case fex_double:
1531 			*(short *)ea = (short) r.val.d;
1532 			break;
1533 
1534 		case fex_ldouble:
1535 			*(short *)ea = (short) r.val.q;
1536 			break;
1537 
1538 		default:
1539 			break;
1540 		}
1541 		if (ex != FPE_FLTRES && (op & 8) != 0)
1542 			pop(uap);
1543 		return;
1544 
1545 	case 0x730:
1546 	case 0x770:
1547 	case 0x7b0:
1548 		/* fbstp; don't bother */
1549 		if (ea && ex != FPE_FLTRES)
1550 			pop(uap);
1551 		return;
1552 
1553 	case 0x738:
1554 	case 0x778:
1555 	case 0x7b8:
1556 		if (!ea)
1557 			return;
1558 		switch (r.type) {
1559 		case fex_int:
1560 			*(long long *)ea = (long long) r.val.i;
1561 			break;
1562 
1563 		case fex_llong:
1564 			*(long long *)ea = r.val.l;
1565 			break;
1566 
1567 		case fex_float:
1568 			*(long long *)ea = (long long) r.val.f;
1569 			break;
1570 
1571 		case fex_double:
1572 			*(long long *)ea = (long long) r.val.d;
1573 			break;
1574 
1575 		case fex_ldouble:
1576 			*(long long *)ea = (long long) r.val.q;
1577 			break;
1578 
1579 		default:
1580 			break;
1581 		}
1582 		if (ex != FPE_FLTRES)
1583 			pop(uap);
1584 		return;
1585 	}
1586 
1587 	/* for all other instructions, the result goes into a register */
1588 	switch (r.type) {
1589 	case fex_int:
1590 		r.val.q = (long double) r.val.i;
1591 		break;
1592 
1593 	case fex_llong:
1594 		r.val.q = (long double) r.val.l;
1595 		break;
1596 
1597 	case fex_float:
1598 		r.val.q = (long double) r.val.f;
1599 		break;
1600 
1601 	case fex_double:
1602 		r.val.q = (long double) r.val.d;
1603 		break;
1604 
1605 	default:
1606 		break;
1607 	}
1608 
1609 	/* for load instructions, push the result onto the stack */
1610 	switch (op & 0x7f8) {
1611 	case 0x100:
1612 	case 0x140:
1613 	case 0x180:
1614 	case 0x500:
1615 	case 0x540:
1616 	case 0x580:
1617 		if (ea)
1618 			push(r.val.q, uap);
1619 		return;
1620 	}
1621 
1622 	/* for all other instructions, if the exception is overflow,
1623 	   underflow, or inexact, the stack has already been updated */
1624 	stack = (ex == FPE_FLTOVF || ex == FPE_FLTUND || ex == FPE_FLTRES);
1625 	switch (op & 0x7f8) {
1626 	case 0x1f0: /* oddballs */
1627 		switch (op) {
1628 		case 0x1f1: /* fyl2x */
1629 		case 0x1f3: /* fpatan */
1630 		case 0x1f9: /* fyl2xp1 */
1631 			/* pop the stack, leaving the result in st */
1632 			if (!stack)
1633 				pop(uap);
1634 			fpreg(uap, 0) = r.val.q;
1635 			return;
1636 
1637 		case 0x1f2: /* fpatan */
1638 			/* fptan pushes 1.0 afterward */
1639 			if (stack)
1640 				fpreg(uap, 1) = r.val.q;
1641 			else {
1642 				fpreg(uap, 0) = r.val.q;
1643 				push(1.0L, uap);
1644 			}
1645 			return;
1646 
1647 		case 0x1f4: /* fxtract */
1648 		case 0x1fb: /* fsincos */
1649 			/* leave the supplied result in st */
1650 			if (stack)
1651 				fpreg(uap, 0) = r.val.q;
1652 			else {
1653 				fpreg(uap, 0) = 0.0; /* punt */
1654 				push(r.val.q, uap);
1655 			}
1656 			return;
1657 		}
1658 
1659 		/* all others leave the stack alone and the result in st */
1660 		fpreg(uap, 0) = r.val.q;
1661 		return;
1662 
1663 	case 0x4c0:
1664 	case 0x4c8:
1665 	case 0x4e0:
1666 	case 0x4e8:
1667 	case 0x4f0:
1668 	case 0x4f8:
1669 		fpreg(uap, op & 7) = r.val.q;
1670 		return;
1671 
1672 	case 0x6c0:
1673 	case 0x6c8:
1674 	case 0x6e0:
1675 	case 0x6e8:
1676 	case 0x6f0:
1677 	case 0x6f8:
1678 		/* stack is popped afterward */
1679 		if (stack)
1680 			fpreg(uap, (op - 1) & 7) = r.val.q;
1681 		else {
1682 			fpreg(uap, op & 7) = r.val.q;
1683 			pop(uap);
1684 		}
1685 		return;
1686 
1687 	default:
1688 		fpreg(uap, 0) = r.val.q;
1689 		return;
1690 	}
1691 }
1692