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
151
152
153
154
155
|
# $OpenBSD: README,v 1.4 2007/11/25 16:40:04 jmc Exp $
* $NetBSD: README,v 1.4 1995/11/05 04:23:00 briggs Exp $
* NetBSD/m68k FPE (floating point emulation) README file
* Created Oct/??/95 by kenn@remus.rutgers.edu (Ken Nakata)
* Last updated Nov/04/95 by kenn
1. INSTALLATION AND COMPILATION
To compile a kernel with FPE built-in, do the following:
1) Add a line "options FPU_EMULATE" to your config file. If you are
going to use the resulted kernel on a machine with an FPU for
debugging purpose, add "options DEBUG_WITH_FPU" as well.
2) Follow the usual procedure to build a new kernel.
NOTE: If you add "options DEBUG_WITH_FPU", FPE will accept cpID=6 as
emulated FPU. You will need a modified gas that generates cpID=6 for
floating point instructions, instead of normal cpID=1. Apply the
following patch to the gas source:
*** /usr/src/gnu/usr.bin/gas/config/tc-m68k.c Mon Nov 21 16:30:41 1994
--- gas/config/tc-m68k.c Fri Sep 29 07:59:06 1995
***************
*** 1275,1281 ****
/* memcpy((char *)(&the_ins.operands[1]), (char *)(&the_ins.operands[0]), opsfound*sizeof(the_ins.operands[0])); */
memset((char *)(&the_ins.operands[0]), '\0', sizeof(the_ins.operands[0]));
the_ins.operands[0].mode=MSCR;
! the_ins.operands[0].reg=COPNUM; /* COP #1 */
opsfound++;
}
--- 1275,1281 ----
/* memcpy((char *)(&the_ins.operands[1]), (char *)(&the_ins.operands[0]), opsfound*sizeof(the_ins.operands[0])); */
memset((char *)(&the_ins.operands[0]), '\0', sizeof(the_ins.operands[0]));
the_ins.operands[0].mode=MSCR;
! the_ins.operands[0].reg=COP5; /* COP #6 */
opsfound++;
}
Also, with the DEBUG_WITH_FPU option, you will be able to run only ONE
process that uses FPE at once to get correct results.
2. MISSING PARTS
For missing instructions, refer to the Section 3. Other than that,
there is one thing that is missing from this version of FPE: packed
BCD support.
I have no plan to support it since it's rarely used. However, all we
need to support it is explosion/implosion functions between the
internal FP representation and the m68k PBCD format, so you are more
than welcome to write such functions if you wish to.
3. IMPLEMENTED INSTRUCTIONS
This is the list of implemented and unimplemented FPU instructions.
All 040's directly supported type 0 instructions are already
implemented except FSGLDIV and FSGLMUL.
Type field = bit 8-6 of opcode word
* Implemented Instructions
Type=0: FMOVE (mem->FPr), FINT, FINTRZ, FSQRT, FABS, FNEG, FGETEXP,
FGETMAN, FDIV, FADD, FMUL, FSGLDIV(*), FSCALE, FSGLMUL(*), FSUB,
FCMP, FTST, FMOVE (FPr->mem), FMOVEM (FPr), FMOVEM (FPcr),
FMOVECR, FLOGNP1, FLOGN, FLOG10, FLOG2, FMOD, FREM
Type=1: FDBcc, FScc, FTRAPcc,
Type=2: FBcc (word, incl. FNOP)
Type=3: FBcc (long)
Type=4: none
Type=5: none
*: currently FSGLMUL and FSGLDIV are just aliases of
FMUL and FDIV, respectively
* Unimplemented Instructions
Type=0: FSINH, FETOXM1, FTANH, FATAN, FASIN, FATANH, FSIN, FTAN,
FETOX, FTWOTOX, FTENTOX, FCOSH, FACOS, FCOS, FSINCOS
Type=1: none
Type=2: none
Type=3: none
Type=4: FSAVE
Type=5: FRESTORE
4. HOW TO ADD A NEW INSTRUCTION SUPPORT
Since we need not support FSAVE and FRESTORE operations, all
instructions we have to implement are type 0, all of which are
arithmetic operations. It is particularly easy to add a new
arithmetic instruction to the existing ones (not that it is easy to
write a "stable" function to perform floating point operation. That's
entirely another matter). In "fpu_emulate.c", there's a function
fpu_emul_arith() which calls emulation functions for all arithmetic
operations. In it, there's a large switch() { case ... } which
dispatches each instruction emulator. An emulation function of any
type 0 arithmetic instruction follows this prototype:
struct fpn *fpu_op(struct fpemu *fe);
Where fe is a pointer to a struct fpemu in which frame, fpframe, and
fetched operands are accessible. That's right, you don't have to
fetch the operands by yourself in your emulation function. For
instance, the parts calling FSQRT, FSUB, FADD and FTST look like:
switch(word1 & 0x3F) {
[...]
case 0x04: /* fsqrt */
res = fpu_sqrt(fe);
break;
[...]
case 0x28: /* fsub */
fe->fe_f2.fp_sign = !fe->fe_f2.fp_sign; /* f2 = -f2 */
case 0x22: /* fadd */
res = fpu_add(fe);
break;
[...]
case 0x3A: /* ftst */
res = &fe->fe_f2;
no_store = 1;
break;
[...]
default:
sig = SIGILL;
} /* switch */
Here, fe->fe_f1 and fe->fe_f2 are fetched operands. You can use
fe->fe_f3 for storing the result, or you can return a pointer to
either operand if you want to. At any rate, you have to follow
the following rules:
1) A dyadic instruction takes two operands fe->fe_f1 and fe->fe_f2.
2) A monadic instruction takes one operand fe->fe_f2 (NOT fe_f1).
3) Must return a pointer to struct fpn where the result is stored,
and assign the pointer to the variable "res".
4) If exceptions are detected, set corresponding bits in fe->fe_fpsr.
The rest is taken care of in fpu_emul_arith().
5) Condition code does not need to be calculated. It's taken care of in
fpu_emul_arith().
|