blob: f9ef34746f16a1a8598c7b2ca75d3b40d804d308 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Linus Torvalds1da177e2005-04-16 15:20:36 -07002 * Copyright (C) 1999 Eddie C. Dost (ecd@atecom.com)
3 */
4
Linus Torvalds1da177e2005-04-16 15:20:36 -07005#include <linux/types.h>
6#include <linux/sched.h>
7
8#include <asm/uaccess.h>
9#include <asm/reg.h>
10
Kumar Galad2b194e2008-06-04 02:59:29 -050011#include <asm/sfp-machine.h>
12#include <math-emu/double.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070013
14#define FLOATFUNC(x) extern int x(void *, void *, void *, void *)
15
16FLOATFUNC(fadd);
17FLOATFUNC(fadds);
18FLOATFUNC(fdiv);
19FLOATFUNC(fdivs);
20FLOATFUNC(fmul);
21FLOATFUNC(fmuls);
22FLOATFUNC(fsub);
23FLOATFUNC(fsubs);
24
25FLOATFUNC(fmadd);
26FLOATFUNC(fmadds);
27FLOATFUNC(fmsub);
28FLOATFUNC(fmsubs);
29FLOATFUNC(fnmadd);
30FLOATFUNC(fnmadds);
31FLOATFUNC(fnmsub);
32FLOATFUNC(fnmsubs);
33
34FLOATFUNC(fctiw);
35FLOATFUNC(fctiwz);
36FLOATFUNC(frsp);
37
38FLOATFUNC(fcmpo);
39FLOATFUNC(fcmpu);
40
41FLOATFUNC(mcrfs);
42FLOATFUNC(mffs);
43FLOATFUNC(mtfsb0);
44FLOATFUNC(mtfsb1);
45FLOATFUNC(mtfsf);
46FLOATFUNC(mtfsfi);
47
48FLOATFUNC(lfd);
49FLOATFUNC(lfs);
50
51FLOATFUNC(stfd);
52FLOATFUNC(stfs);
53FLOATFUNC(stfiwx);
54
55FLOATFUNC(fabs);
56FLOATFUNC(fmr);
57FLOATFUNC(fnabs);
58FLOATFUNC(fneg);
59
60/* Optional */
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +100061FLOATFUNC(fre);
Linus Torvalds1da177e2005-04-16 15:20:36 -070062FLOATFUNC(fres);
63FLOATFUNC(frsqrte);
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +100064FLOATFUNC(frsqrtes);
Linus Torvalds1da177e2005-04-16 15:20:36 -070065FLOATFUNC(fsel);
66FLOATFUNC(fsqrt);
67FLOATFUNC(fsqrts);
68
69
70#define OP31 0x1f /* 31 */
71#define LFS 0x30 /* 48 */
72#define LFSU 0x31 /* 49 */
73#define LFD 0x32 /* 50 */
74#define LFDU 0x33 /* 51 */
75#define STFS 0x34 /* 52 */
76#define STFSU 0x35 /* 53 */
77#define STFD 0x36 /* 54 */
78#define STFDU 0x37 /* 55 */
79#define OP59 0x3b /* 59 */
80#define OP63 0x3f /* 63 */
81
82/* Opcode 31: */
83/* X-Form: */
84#define LFSX 0x217 /* 535 */
85#define LFSUX 0x237 /* 567 */
86#define LFDX 0x257 /* 599 */
87#define LFDUX 0x277 /* 631 */
88#define STFSX 0x297 /* 663 */
89#define STFSUX 0x2b7 /* 695 */
90#define STFDX 0x2d7 /* 727 */
91#define STFDUX 0x2f7 /* 759 */
92#define STFIWX 0x3d7 /* 983 */
93
94/* Opcode 59: */
95/* A-Form: */
96#define FDIVS 0x012 /* 18 */
97#define FSUBS 0x014 /* 20 */
98#define FADDS 0x015 /* 21 */
99#define FSQRTS 0x016 /* 22 */
100#define FRES 0x018 /* 24 */
101#define FMULS 0x019 /* 25 */
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000102#define FRSQRTES 0x01a /* 26 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103#define FMSUBS 0x01c /* 28 */
104#define FMADDS 0x01d /* 29 */
105#define FNMSUBS 0x01e /* 30 */
106#define FNMADDS 0x01f /* 31 */
107
108/* Opcode 63: */
109/* A-Form: */
110#define FDIV 0x012 /* 18 */
111#define FSUB 0x014 /* 20 */
112#define FADD 0x015 /* 21 */
113#define FSQRT 0x016 /* 22 */
114#define FSEL 0x017 /* 23 */
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000115#define FRE 0x018 /* 24 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700116#define FMUL 0x019 /* 25 */
117#define FRSQRTE 0x01a /* 26 */
118#define FMSUB 0x01c /* 28 */
119#define FMADD 0x01d /* 29 */
120#define FNMSUB 0x01e /* 30 */
121#define FNMADD 0x01f /* 31 */
122
123/* X-Form: */
124#define FCMPU 0x000 /* 0 */
125#define FRSP 0x00c /* 12 */
126#define FCTIW 0x00e /* 14 */
127#define FCTIWZ 0x00f /* 15 */
128#define FCMPO 0x020 /* 32 */
129#define MTFSB1 0x026 /* 38 */
130#define FNEG 0x028 /* 40 */
131#define MCRFS 0x040 /* 64 */
132#define MTFSB0 0x046 /* 70 */
133#define FMR 0x048 /* 72 */
134#define MTFSFI 0x086 /* 134 */
135#define FNABS 0x088 /* 136 */
136#define FABS 0x108 /* 264 */
137#define MFFS 0x247 /* 583 */
138#define MTFSF 0x2c7 /* 711 */
139
140
141#define AB 2
142#define AC 3
143#define ABC 4
144#define D 5
145#define DU 6
146#define X 7
147#define XA 8
148#define XB 9
149#define XCR 11
150#define XCRB 12
151#define XCRI 13
152#define XCRL 16
153#define XE 14
154#define XEU 15
155#define XFLB 10
156
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157static int
158record_exception(struct pt_regs *regs, int eflag)
159{
160 u32 fpscr;
161
162 fpscr = __FPU_FPSCR;
163
164 if (eflag) {
165 fpscr |= FPSCR_FX;
166 if (eflag & EFLAG_OVERFLOW)
167 fpscr |= FPSCR_OX;
168 if (eflag & EFLAG_UNDERFLOW)
169 fpscr |= FPSCR_UX;
170 if (eflag & EFLAG_DIVZERO)
171 fpscr |= FPSCR_ZX;
172 if (eflag & EFLAG_INEXACT)
173 fpscr |= FPSCR_XX;
Kumar Galad2b194e2008-06-04 02:59:29 -0500174 if (eflag & EFLAG_INVALID)
175 fpscr |= FPSCR_VX;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700176 if (eflag & EFLAG_VXSNAN)
177 fpscr |= FPSCR_VXSNAN;
178 if (eflag & EFLAG_VXISI)
179 fpscr |= FPSCR_VXISI;
180 if (eflag & EFLAG_VXIDI)
181 fpscr |= FPSCR_VXIDI;
182 if (eflag & EFLAG_VXZDZ)
183 fpscr |= FPSCR_VXZDZ;
184 if (eflag & EFLAG_VXIMZ)
185 fpscr |= FPSCR_VXIMZ;
186 if (eflag & EFLAG_VXVC)
187 fpscr |= FPSCR_VXVC;
188 if (eflag & EFLAG_VXSOFT)
189 fpscr |= FPSCR_VXSOFT;
190 if (eflag & EFLAG_VXSQRT)
191 fpscr |= FPSCR_VXSQRT;
192 if (eflag & EFLAG_VXCVI)
193 fpscr |= FPSCR_VXCVI;
194 }
195
Kumar Galad2b194e2008-06-04 02:59:29 -0500196// fpscr &= ~(FPSCR_VX);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700197 if (fpscr & (FPSCR_VXSNAN | FPSCR_VXISI | FPSCR_VXIDI |
198 FPSCR_VXZDZ | FPSCR_VXIMZ | FPSCR_VXVC |
199 FPSCR_VXSOFT | FPSCR_VXSQRT | FPSCR_VXCVI))
200 fpscr |= FPSCR_VX;
201
202 fpscr &= ~(FPSCR_FEX);
203 if (((fpscr & FPSCR_VX) && (fpscr & FPSCR_VE)) ||
204 ((fpscr & FPSCR_OX) && (fpscr & FPSCR_OE)) ||
205 ((fpscr & FPSCR_UX) && (fpscr & FPSCR_UE)) ||
206 ((fpscr & FPSCR_ZX) && (fpscr & FPSCR_ZE)) ||
207 ((fpscr & FPSCR_XX) && (fpscr & FPSCR_XE)))
208 fpscr |= FPSCR_FEX;
209
210 __FPU_FPSCR = fpscr;
211
212 return (fpscr & FPSCR_FEX) ? 1 : 0;
213}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700214
215int
216do_mathemu(struct pt_regs *regs)
217{
218 void *op0 = 0, *op1 = 0, *op2 = 0, *op3 = 0;
219 unsigned long pc = regs->nip;
220 signed short sdisp;
221 u32 insn = 0;
222 int idx = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700223 int (*func)(void *, void *, void *, void *);
224 int type = 0;
225 int eflag, trap;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700226
227 if (get_user(insn, (u32 *)pc))
228 return -EFAULT;
229
Linus Torvalds1da177e2005-04-16 15:20:36 -0700230 switch (insn >> 26) {
231 case LFS: func = lfs; type = D; break;
232 case LFSU: func = lfs; type = DU; break;
233 case LFD: func = lfd; type = D; break;
234 case LFDU: func = lfd; type = DU; break;
235 case STFS: func = stfs; type = D; break;
236 case STFSU: func = stfs; type = DU; break;
237 case STFD: func = stfd; type = D; break;
238 case STFDU: func = stfd; type = DU; break;
239
240 case OP31:
241 switch ((insn >> 1) & 0x3ff) {
242 case LFSX: func = lfs; type = XE; break;
243 case LFSUX: func = lfs; type = XEU; break;
244 case LFDX: func = lfd; type = XE; break;
245 case LFDUX: func = lfd; type = XEU; break;
246 case STFSX: func = stfs; type = XE; break;
247 case STFSUX: func = stfs; type = XEU; break;
248 case STFDX: func = stfd; type = XE; break;
249 case STFDUX: func = stfd; type = XEU; break;
250 case STFIWX: func = stfiwx; type = XE; break;
251 default:
252 goto illegal;
253 }
254 break;
255
256 case OP59:
257 switch ((insn >> 1) & 0x1f) {
258 case FDIVS: func = fdivs; type = AB; break;
259 case FSUBS: func = fsubs; type = AB; break;
260 case FADDS: func = fadds; type = AB; break;
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000261 case FSQRTS: func = fsqrts; type = XB; break;
262 case FRES: func = fres; type = XB; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263 case FMULS: func = fmuls; type = AC; break;
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000264 case FRSQRTES: func = frsqrtes;type = XB; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700265 case FMSUBS: func = fmsubs; type = ABC; break;
266 case FMADDS: func = fmadds; type = ABC; break;
267 case FNMSUBS: func = fnmsubs; type = ABC; break;
268 case FNMADDS: func = fnmadds; type = ABC; break;
269 default:
270 goto illegal;
271 }
272 break;
273
274 case OP63:
275 if (insn & 0x20) {
276 switch ((insn >> 1) & 0x1f) {
277 case FDIV: func = fdiv; type = AB; break;
278 case FSUB: func = fsub; type = AB; break;
279 case FADD: func = fadd; type = AB; break;
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000280 case FSQRT: func = fsqrt; type = XB; break;
281 case FRE: func = fre; type = XB; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 case FSEL: func = fsel; type = ABC; break;
283 case FMUL: func = fmul; type = AC; break;
Benjamin Herrenschmidt04ae9002013-06-09 17:00:42 +1000284 case FRSQRTE: func = frsqrte; type = XB; break;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700285 case FMSUB: func = fmsub; type = ABC; break;
286 case FMADD: func = fmadd; type = ABC; break;
287 case FNMSUB: func = fnmsub; type = ABC; break;
288 case FNMADD: func = fnmadd; type = ABC; break;
289 default:
290 goto illegal;
291 }
292 break;
293 }
294
295 switch ((insn >> 1) & 0x3ff) {
296 case FCMPU: func = fcmpu; type = XCR; break;
297 case FRSP: func = frsp; type = XB; break;
298 case FCTIW: func = fctiw; type = XB; break;
299 case FCTIWZ: func = fctiwz; type = XB; break;
300 case FCMPO: func = fcmpo; type = XCR; break;
301 case MTFSB1: func = mtfsb1; type = XCRB; break;
302 case FNEG: func = fneg; type = XB; break;
303 case MCRFS: func = mcrfs; type = XCRL; break;
304 case MTFSB0: func = mtfsb0; type = XCRB; break;
305 case FMR: func = fmr; type = XB; break;
306 case MTFSFI: func = mtfsfi; type = XCRI; break;
307 case FNABS: func = fnabs; type = XB; break;
308 case FABS: func = fabs; type = XB; break;
309 case MFFS: func = mffs; type = X; break;
310 case MTFSF: func = mtfsf; type = XFLB; break;
311 default:
312 goto illegal;
313 }
314 break;
315
316 default:
317 goto illegal;
318 }
319
320 switch (type) {
321 case AB:
Michael Neuling9c75a312008-06-26 17:07:48 +1000322 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
323 op1 = (void *)&current->thread.TS_FPR((insn >> 16) & 0x1f);
324 op2 = (void *)&current->thread.TS_FPR((insn >> 11) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700325 break;
326
327 case AC:
Michael Neuling9c75a312008-06-26 17:07:48 +1000328 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
329 op1 = (void *)&current->thread.TS_FPR((insn >> 16) & 0x1f);
330 op2 = (void *)&current->thread.TS_FPR((insn >> 6) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331 break;
332
333 case ABC:
Michael Neuling9c75a312008-06-26 17:07:48 +1000334 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
335 op1 = (void *)&current->thread.TS_FPR((insn >> 16) & 0x1f);
336 op2 = (void *)&current->thread.TS_FPR((insn >> 11) & 0x1f);
337 op3 = (void *)&current->thread.TS_FPR((insn >> 6) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338 break;
339
340 case D:
341 idx = (insn >> 16) & 0x1f;
342 sdisp = (insn & 0xffff);
Michael Neuling9c75a312008-06-26 17:07:48 +1000343 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700344 op1 = (void *)((idx ? regs->gpr[idx] : 0) + sdisp);
345 break;
346
347 case DU:
348 idx = (insn >> 16) & 0x1f;
349 if (!idx)
350 goto illegal;
351
352 sdisp = (insn & 0xffff);
Michael Neuling9c75a312008-06-26 17:07:48 +1000353 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700354 op1 = (void *)(regs->gpr[idx] + sdisp);
355 break;
356
357 case X:
Michael Neuling9c75a312008-06-26 17:07:48 +1000358 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700359 break;
360
361 case XA:
Michael Neuling9c75a312008-06-26 17:07:48 +1000362 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
363 op1 = (void *)&current->thread.TS_FPR((insn >> 16) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700364 break;
365
366 case XB:
Michael Neuling9c75a312008-06-26 17:07:48 +1000367 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
368 op1 = (void *)&current->thread.TS_FPR((insn >> 11) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700369 break;
370
371 case XE:
372 idx = (insn >> 16) & 0x1f;
Michael Neuling9c75a312008-06-26 17:07:48 +1000373 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
Kumar Galaba029462007-10-11 17:07:34 -0500374 if (!idx) {
375 if (((insn >> 1) & 0x3ff) == STFIWX)
376 op1 = (void *)(regs->gpr[(insn >> 11) & 0x1f]);
377 else
378 goto illegal;
379 } else {
380 op1 = (void *)(regs->gpr[idx] + regs->gpr[(insn >> 11) & 0x1f]);
381 }
382
Linus Torvalds1da177e2005-04-16 15:20:36 -0700383 break;
384
385 case XEU:
386 idx = (insn >> 16) & 0x1f;
Michael Neuling9c75a312008-06-26 17:07:48 +1000387 op0 = (void *)&current->thread.TS_FPR((insn >> 21) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700388 op1 = (void *)((idx ? regs->gpr[idx] : 0)
389 + regs->gpr[(insn >> 11) & 0x1f]);
390 break;
391
392 case XCR:
393 op0 = (void *)&regs->ccr;
394 op1 = (void *)((insn >> 23) & 0x7);
Michael Neuling9c75a312008-06-26 17:07:48 +1000395 op2 = (void *)&current->thread.TS_FPR((insn >> 16) & 0x1f);
396 op3 = (void *)&current->thread.TS_FPR((insn >> 11) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700397 break;
398
399 case XCRL:
400 op0 = (void *)&regs->ccr;
401 op1 = (void *)((insn >> 23) & 0x7);
402 op2 = (void *)((insn >> 18) & 0x7);
403 break;
404
405 case XCRB:
406 op0 = (void *)((insn >> 21) & 0x1f);
407 break;
408
409 case XCRI:
410 op0 = (void *)((insn >> 23) & 0x7);
411 op1 = (void *)((insn >> 12) & 0xf);
412 break;
413
414 case XFLB:
415 op0 = (void *)((insn >> 17) & 0xff);
Michael Neuling9c75a312008-06-26 17:07:48 +1000416 op1 = (void *)&current->thread.TS_FPR((insn >> 11) & 0x1f);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700417 break;
418
419 default:
420 goto illegal;
421 }
422
Kevin Hao6761ee32013-07-14 16:40:06 +0800423 /*
424 * If we support a HW FPU, we need to ensure the FP state
425 * is flushed into the thread_struct before attempting
426 * emulation
427 */
428#ifdef CONFIG_PPC_FPU
429 flush_fp_to_thread(current);
430#endif
431
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432 eflag = func(op0, op1, op2, op3);
433
434 if (insn & 1) {
435 regs->ccr &= ~(0x0f000000);
436 regs->ccr |= (__FPU_FPSCR >> 4) & 0x0f000000;
437 }
438
439 trap = record_exception(regs, eflag);
440 if (trap)
441 return 1;
442
443 switch (type) {
444 case DU:
445 case XEU:
446 regs->gpr[idx] = (unsigned long)op1;
447 break;
448
449 default:
450 break;
451 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700452
453 regs->nip += 4;
454 return 0;
455
456illegal:
457 return -ENOSYS;
458}