diff options
author | Paul Burton <paul.burton@imgtec.com> | 2013-11-22 08:12:07 -0500 |
---|---|---|
committer | Ralf Baechle <ralf@linux-mips.org> | 2014-01-13 17:40:56 -0500 |
commit | 597ce1723e0fa0bdbe2ae4c94f18da6e29b92635 (patch) | |
tree | 21f67268915b8457dd305c6bcf7ac905772fd0ee /arch/mips/math-emu/kernel_linkage.c | |
parent | 56a22d21bf9744315f56b2bbd6416170f27b7765 (diff) |
MIPS: Support for 64-bit FP with O32 binaries
CPUs implementing MIPS32 R2 may include a 64-bit FPU, just as MIPS64 CPUs
do. In order to preserve backwards compatibility a 64-bit FPU will act
like a 32-bit FPU (by accessing doubles from the least significant 32
bits of an even-odd pair of FP registers) when the Status.FR bit is
zero, again just like a mips64 CPU. The standard O32 ABI is defined
expecting a 32-bit FPU, however recent toolchains support use of a
64-bit FPU from an O32 MIPS32 executable. When an ELF executable is
built to use a 64-bit FPU a new flag (EF_MIPS_FP64) is set in the ELF
header.
With this patch the kernel will check the EF_MIPS_FP64 flag when
executing an O32 binary, and set Status.FR accordingly. The addition
of O32 64-bit FP support lessens the opportunity for optimisation in
the FPU emulator, so a CONFIG_MIPS_O32_FP64_SUPPORT Kconfig option is
introduced to allow this support to be disabled for those that don't
require it.
Inspired by an earlier patch by Leonid Yegoshin, but implemented more
cleanly & correctly.
Signed-off-by: Paul Burton <paul.burton@imgtec.com>
Cc: linux-mips@linux-mips.org
Cc: Paul Burton <paul.burton@imgtec.com>
Patchwork: https://patchwork.linux-mips.org/patch/6154/
Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
Diffstat (limited to 'arch/mips/math-emu/kernel_linkage.c')
-rw-r--r-- | arch/mips/math-emu/kernel_linkage.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/arch/mips/math-emu/kernel_linkage.c b/arch/mips/math-emu/kernel_linkage.c index 1c586575fe17..3aeae07ed5b8 100644 --- a/arch/mips/math-emu/kernel_linkage.c +++ b/arch/mips/math-emu/kernel_linkage.c | |||
@@ -89,8 +89,9 @@ int fpu_emulator_save_context32(struct sigcontext32 __user *sc) | |||
89 | { | 89 | { |
90 | int i; | 90 | int i; |
91 | int err = 0; | 91 | int err = 0; |
92 | int inc = test_thread_flag(TIF_32BIT_FPREGS) ? 2 : 1; | ||
92 | 93 | ||
93 | for (i = 0; i < 32; i+=2) { | 94 | for (i = 0; i < 32; i += inc) { |
94 | err |= | 95 | err |= |
95 | __put_user(current->thread.fpu.fpr[i], &sc->sc_fpregs[i]); | 96 | __put_user(current->thread.fpu.fpr[i], &sc->sc_fpregs[i]); |
96 | } | 97 | } |
@@ -103,8 +104,9 @@ int fpu_emulator_restore_context32(struct sigcontext32 __user *sc) | |||
103 | { | 104 | { |
104 | int i; | 105 | int i; |
105 | int err = 0; | 106 | int err = 0; |
107 | int inc = test_thread_flag(TIF_32BIT_FPREGS) ? 2 : 1; | ||
106 | 108 | ||
107 | for (i = 0; i < 32; i+=2) { | 109 | for (i = 0; i < 32; i += inc) { |
108 | err |= | 110 | err |= |
109 | __get_user(current->thread.fpu.fpr[i], &sc->sc_fpregs[i]); | 111 | __get_user(current->thread.fpu.fpr[i], &sc->sc_fpregs[i]); |
110 | } | 112 | } |