hc
2024-03-22 619f0f87159c5dbd2755b1b0a0eb35784be84e7a
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
/* SPDX-License-Identifier: GPL-2.0 */
/*---------------------------------------------------------------------------+
 |  reg_norm.S                                                               |
 |                                                                           |
 | Copyright (C) 1992,1993,1994,1995,1997                                    |
 |                       W. Metzenthen, 22 Parker St, Ormond, Vic 3163,      |
 |                       Australia.  E-mail billm@suburbia.net               |
 |                                                                           |
 | Normalize the value in a FPU_REG.                                         |
 |                                                                           |
 | Call from C as:                                                           |
 |    int FPU_normalize(FPU_REG *n)                                          |
 |                                                                           |
 |    int FPU_normalize_nuo(FPU_REG *n)                                      |
 |                                                                           |
 |    Return value is the tag of the answer, or-ed with FPU_Exception if     |
 |    one was raised, or -1 on internal error.                               |
 |                                                                           |
 +---------------------------------------------------------------------------*/
 
#include "fpu_emu.h"
 
 
.text
SYM_FUNC_START(FPU_normalize)
   pushl    %ebp
   movl    %esp,%ebp
   pushl    %ebx
 
   movl    PARAM1,%ebx
 
   movl    SIGH(%ebx),%edx
   movl    SIGL(%ebx),%eax
 
   orl    %edx,%edx    /* ms bits */
   js    L_done        /* Already normalized */
   jnz    L_shift_1    /* Shift left 1 - 31 bits */
 
   orl    %eax,%eax
   jz    L_zero        /* The contents are zero */
 
   movl    %eax,%edx
   xorl    %eax,%eax
   subw    $32,EXP(%ebx)    /* This can cause an underflow */
 
/* We need to shift left by 1 - 31 bits */
L_shift_1:
   bsrl    %edx,%ecx    /* get the required shift in %ecx */
   subl    $31,%ecx
   negl    %ecx
   shld    %cl,%eax,%edx
   shl    %cl,%eax
   subw    %cx,EXP(%ebx)    /* This can cause an underflow */
 
   movl    %edx,SIGH(%ebx)
   movl    %eax,SIGL(%ebx)
 
L_done:
   cmpw    EXP_OVER,EXP(%ebx)
   jge    L_overflow
 
   cmpw    EXP_UNDER,EXP(%ebx)
   jle    L_underflow
 
L_exit_valid:
   movl    TAG_Valid,%eax
 
   /* Convert the exponent to 80x87 form. */
   addw    EXTENDED_Ebias,EXP(%ebx)
   andw    $0x7fff,EXP(%ebx)
 
L_exit:
   popl    %ebx
   leave
   RET
 
 
L_zero:
   movw    $0,EXP(%ebx)
   movl    TAG_Zero,%eax
   jmp    L_exit
 
L_underflow:
   /* Convert the exponent to 80x87 form. */
   addw    EXTENDED_Ebias,EXP(%ebx)
   push    %ebx
   call    arith_underflow
   pop    %ebx
   jmp    L_exit
 
L_overflow:
   /* Convert the exponent to 80x87 form. */
   addw    EXTENDED_Ebias,EXP(%ebx)
   push    %ebx
   call    arith_overflow
   pop    %ebx
   jmp    L_exit
SYM_FUNC_END(FPU_normalize)
 
 
 
/* Normalise without reporting underflow or overflow */
SYM_FUNC_START(FPU_normalize_nuo)
   pushl    %ebp
   movl    %esp,%ebp
   pushl    %ebx
 
   movl    PARAM1,%ebx
 
   movl    SIGH(%ebx),%edx
   movl    SIGL(%ebx),%eax
 
   orl    %edx,%edx    /* ms bits */
   js    L_exit_nuo_valid    /* Already normalized */
   jnz    L_nuo_shift_1    /* Shift left 1 - 31 bits */
 
   orl    %eax,%eax
   jz    L_exit_nuo_zero        /* The contents are zero */
 
   movl    %eax,%edx
   xorl    %eax,%eax
   subw    $32,EXP(%ebx)    /* This can cause an underflow */
 
/* We need to shift left by 1 - 31 bits */
L_nuo_shift_1:
   bsrl    %edx,%ecx    /* get the required shift in %ecx */
   subl    $31,%ecx
   negl    %ecx
   shld    %cl,%eax,%edx
   shl    %cl,%eax
   subw    %cx,EXP(%ebx)    /* This can cause an underflow */
 
   movl    %edx,SIGH(%ebx)
   movl    %eax,SIGL(%ebx)
 
L_exit_nuo_valid:
   movl    TAG_Valid,%eax
 
   popl    %ebx
   leave
   RET
 
L_exit_nuo_zero:
   movl    TAG_Zero,%eax
   movw    EXP_UNDER,EXP(%ebx)
 
   popl    %ebx
   leave
   RET
SYM_FUNC_END(FPU_normalize_nuo)