MIPS: kernel: elf: Improve the overall ABI and FPU mode checks
The previous implementation did not cover all possible FPU combinations and it silently allowed ABI incompatible objects to be loaded with the wrong ABI. For example, the previous logic would set the FP_64 ABI as the matching ABI for an FP_XX object combined with an FP_64A object. This was wrong, and the matching ABI should have been FP_64A. The previous logic is now replaced with a new one which determines the appropriate FPU mode to be used rather than the FP ABI. This has the advantage that the entire logic is much simpler since it is the FPU mode we are interested in rather than the FP ABI resulting to code simplifications. This also removes the now obsolete FP32XX_HYBRID_FPRS option. Cc: Matthew Fortune <Matthew.Fortune@imgtec.com> Cc: Paul Burton <paul.burton@imgtec.com> Signed-off-by: Markos Chandras <markos.chandras@imgtec.com>
This commit is contained in:
Родитель
6134d94923
Коммит
46490b5725
|
@ -122,17 +122,4 @@ config SPINLOCK_TEST
|
||||||
help
|
help
|
||||||
Add several files to the debugfs to test spinlock speed.
|
Add several files to the debugfs to test spinlock speed.
|
||||||
|
|
||||||
config FP32XX_HYBRID_FPRS
|
|
||||||
bool "Run FP32 & FPXX code with hybrid FPRs"
|
|
||||||
depends on MIPS_O32_FP64_SUPPORT
|
|
||||||
help
|
|
||||||
The hybrid FPR scheme is normally used only when a program needs to
|
|
||||||
execute a mix of FP32 & FP64A code, since the trapping & emulation
|
|
||||||
that it entails is expensive. When enabled, this option will lead
|
|
||||||
to the kernel running programs which use the FP32 & FPXX FP ABIs
|
|
||||||
using the hybrid FPR scheme, which can be useful for debugging
|
|
||||||
purposes.
|
|
||||||
|
|
||||||
If unsure, say N.
|
|
||||||
|
|
||||||
endmenu
|
endmenu
|
||||||
|
|
|
@ -417,13 +417,15 @@ extern unsigned long arch_randomize_brk(struct mm_struct *mm);
|
||||||
struct arch_elf_state {
|
struct arch_elf_state {
|
||||||
int fp_abi;
|
int fp_abi;
|
||||||
int interp_fp_abi;
|
int interp_fp_abi;
|
||||||
int overall_abi;
|
int overall_fp_mode;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#define MIPS_ABI_FP_UNKNOWN (-1) /* Unknown FP ABI (kernel internal) */
|
||||||
|
|
||||||
#define INIT_ARCH_ELF_STATE { \
|
#define INIT_ARCH_ELF_STATE { \
|
||||||
.fp_abi = -1, \
|
.fp_abi = MIPS_ABI_FP_UNKNOWN, \
|
||||||
.interp_fp_abi = -1, \
|
.interp_fp_abi = MIPS_ABI_FP_UNKNOWN, \
|
||||||
.overall_abi = -1, \
|
.overall_fp_mode = -1, \
|
||||||
}
|
}
|
||||||
|
|
||||||
extern int arch_elf_pt_proc(void *ehdr, void *phdr, struct file *elf,
|
extern int arch_elf_pt_proc(void *ehdr, void *phdr, struct file *elf,
|
||||||
|
|
|
@ -11,29 +11,112 @@
|
||||||
#include <linux/elf.h>
|
#include <linux/elf.h>
|
||||||
#include <linux/sched.h>
|
#include <linux/sched.h>
|
||||||
|
|
||||||
|
/* FPU modes */
|
||||||
enum {
|
enum {
|
||||||
FP_ERROR = -1,
|
FP_FRE,
|
||||||
FP_DOUBLE_64A = -2,
|
FP_FR0,
|
||||||
|
FP_FR1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct mode_req - ABI FPU mode requirements
|
||||||
|
* @single: The program being loaded needs an FPU but it will only issue
|
||||||
|
* single precision instructions meaning that it can execute in
|
||||||
|
* either FR0 or FR1.
|
||||||
|
* @soft: The soft(-float) requirement means that the program being
|
||||||
|
* loaded needs has no FPU dependency at all (i.e. it has no
|
||||||
|
* FPU instructions).
|
||||||
|
* @fr1: The program being loaded depends on FPU being in FR=1 mode.
|
||||||
|
* @frdefault: The program being loaded depends on the default FPU mode.
|
||||||
|
* That is FR0 for O32 and FR1 for N32/N64.
|
||||||
|
* @fre: The program being loaded depends on FPU with FRE=1. This mode is
|
||||||
|
* a bridge which uses FR=1 whilst still being able to maintain
|
||||||
|
* full compatibility with pre-existing code using the O32 FP32
|
||||||
|
* ABI.
|
||||||
|
*
|
||||||
|
* More information about the FP ABIs can be found here:
|
||||||
|
*
|
||||||
|
* https://dmz-portal.mips.com/wiki/MIPS_O32_ABI_-_FR0_and_FR1_Interlinking#10.4.1._Basic_mode_set-up
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
struct mode_req {
|
||||||
|
bool single;
|
||||||
|
bool soft;
|
||||||
|
bool fr1;
|
||||||
|
bool frdefault;
|
||||||
|
bool fre;
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct mode_req fpu_reqs[] = {
|
||||||
|
[MIPS_ABI_FP_ANY] = { true, true, true, true, true },
|
||||||
|
[MIPS_ABI_FP_DOUBLE] = { false, false, false, true, true },
|
||||||
|
[MIPS_ABI_FP_SINGLE] = { true, false, false, false, false },
|
||||||
|
[MIPS_ABI_FP_SOFT] = { false, true, false, false, false },
|
||||||
|
[MIPS_ABI_FP_OLD_64] = { false, false, false, false, false },
|
||||||
|
[MIPS_ABI_FP_XX] = { false, false, true, true, true },
|
||||||
|
[MIPS_ABI_FP_64] = { false, false, true, false, false },
|
||||||
|
[MIPS_ABI_FP_64A] = { false, false, true, false, true }
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Mode requirements when .MIPS.abiflags is not present in the ELF.
|
||||||
|
* Not present means that everything is acceptable except FR1.
|
||||||
|
*/
|
||||||
|
static struct mode_req none_req = { true, true, false, true, true };
|
||||||
|
|
||||||
int arch_elf_pt_proc(void *_ehdr, void *_phdr, struct file *elf,
|
int arch_elf_pt_proc(void *_ehdr, void *_phdr, struct file *elf,
|
||||||
bool is_interp, struct arch_elf_state *state)
|
bool is_interp, struct arch_elf_state *state)
|
||||||
{
|
{
|
||||||
struct elfhdr *ehdr = _ehdr;
|
struct elf32_hdr *ehdr32 = _ehdr;
|
||||||
struct elf_phdr *phdr = _phdr;
|
struct elf32_phdr *phdr32 = _phdr;
|
||||||
|
struct elf64_phdr *phdr64 = _phdr;
|
||||||
struct mips_elf_abiflags_v0 abiflags;
|
struct mips_elf_abiflags_v0 abiflags;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (config_enabled(CONFIG_64BIT) &&
|
/* Lets see if this is an O32 ELF */
|
||||||
(ehdr->e_ident[EI_CLASS] != ELFCLASS32))
|
if (ehdr32->e_ident[EI_CLASS] == ELFCLASS32) {
|
||||||
return 0;
|
/* FR = 1 for N32 */
|
||||||
if (phdr->p_type != PT_MIPS_ABIFLAGS)
|
if (ehdr32->e_flags & EF_MIPS_ABI2)
|
||||||
return 0;
|
state->overall_fp_mode = FP_FR1;
|
||||||
if (phdr->p_filesz < sizeof(abiflags))
|
else
|
||||||
return -EINVAL;
|
/* Set a good default FPU mode for O32 */
|
||||||
|
state->overall_fp_mode = cpu_has_mips_r6 ?
|
||||||
|
FP_FRE : FP_FR0;
|
||||||
|
|
||||||
|
if (ehdr32->e_flags & EF_MIPS_FP64) {
|
||||||
|
/*
|
||||||
|
* Set MIPS_ABI_FP_OLD_64 for EF_MIPS_FP64. We will override it
|
||||||
|
* later if needed
|
||||||
|
*/
|
||||||
|
if (is_interp)
|
||||||
|
state->interp_fp_abi = MIPS_ABI_FP_OLD_64;
|
||||||
|
else
|
||||||
|
state->fp_abi = MIPS_ABI_FP_OLD_64;
|
||||||
|
}
|
||||||
|
if (phdr32->p_type != PT_MIPS_ABIFLAGS)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (phdr32->p_filesz < sizeof(abiflags))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
ret = kernel_read(elf, phdr32->p_offset,
|
||||||
|
(char *)&abiflags,
|
||||||
|
sizeof(abiflags));
|
||||||
|
} else {
|
||||||
|
/* FR=1 is really the only option for 64-bit */
|
||||||
|
state->overall_fp_mode = FP_FR1;
|
||||||
|
|
||||||
|
if (phdr64->p_type != PT_MIPS_ABIFLAGS)
|
||||||
|
return 0;
|
||||||
|
if (phdr64->p_filesz < sizeof(abiflags))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
ret = kernel_read(elf, phdr64->p_offset,
|
||||||
|
(char *)&abiflags,
|
||||||
|
sizeof(abiflags));
|
||||||
|
}
|
||||||
|
|
||||||
ret = kernel_read(elf, phdr->p_offset, (char *)&abiflags,
|
|
||||||
sizeof(abiflags));
|
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
return ret;
|
return ret;
|
||||||
if (ret != sizeof(abiflags))
|
if (ret != sizeof(abiflags))
|
||||||
|
@ -48,35 +131,30 @@ int arch_elf_pt_proc(void *_ehdr, void *_phdr, struct file *elf,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline unsigned get_fp_abi(struct elfhdr *ehdr, int in_abi)
|
static inline unsigned get_fp_abi(int in_abi)
|
||||||
{
|
{
|
||||||
/* If the ABI requirement is provided, simply return that */
|
/* If the ABI requirement is provided, simply return that */
|
||||||
if (in_abi != -1)
|
if (in_abi != MIPS_ABI_FP_UNKNOWN)
|
||||||
return in_abi;
|
return in_abi;
|
||||||
|
|
||||||
/* If the EF_MIPS_FP64 flag was set, return MIPS_ABI_FP_64 */
|
/* Unknown ABI */
|
||||||
if (ehdr->e_flags & EF_MIPS_FP64)
|
return MIPS_ABI_FP_UNKNOWN;
|
||||||
return MIPS_ABI_FP_64;
|
|
||||||
|
|
||||||
/* Default to MIPS_ABI_FP_DOUBLE */
|
|
||||||
return MIPS_ABI_FP_DOUBLE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int arch_check_elf(void *_ehdr, bool has_interpreter,
|
int arch_check_elf(void *_ehdr, bool has_interpreter,
|
||||||
struct arch_elf_state *state)
|
struct arch_elf_state *state)
|
||||||
{
|
{
|
||||||
struct elfhdr *ehdr = _ehdr;
|
struct elf32_hdr *ehdr = _ehdr;
|
||||||
unsigned fp_abi, interp_fp_abi, abi0, abi1;
|
struct mode_req prog_req, interp_req;
|
||||||
|
int fp_abi, interp_fp_abi, abi0, abi1, max_abi;
|
||||||
|
|
||||||
/* Ignore non-O32 binaries */
|
if (!config_enabled(CONFIG_MIPS_O32_FP64_SUPPORT))
|
||||||
if (config_enabled(CONFIG_64BIT) &&
|
|
||||||
(ehdr->e_ident[EI_CLASS] != ELFCLASS32))
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
fp_abi = get_fp_abi(ehdr, state->fp_abi);
|
fp_abi = get_fp_abi(state->fp_abi);
|
||||||
|
|
||||||
if (has_interpreter) {
|
if (has_interpreter) {
|
||||||
interp_fp_abi = get_fp_abi(ehdr, state->interp_fp_abi);
|
interp_fp_abi = get_fp_abi(state->interp_fp_abi);
|
||||||
|
|
||||||
abi0 = min(fp_abi, interp_fp_abi);
|
abi0 = min(fp_abi, interp_fp_abi);
|
||||||
abi1 = max(fp_abi, interp_fp_abi);
|
abi1 = max(fp_abi, interp_fp_abi);
|
||||||
|
@ -84,108 +162,103 @@ int arch_check_elf(void *_ehdr, bool has_interpreter,
|
||||||
abi0 = abi1 = fp_abi;
|
abi0 = abi1 = fp_abi;
|
||||||
}
|
}
|
||||||
|
|
||||||
state->overall_abi = FP_ERROR;
|
/* ABI limits. O32 = FP_64A, N32/N64 = FP_SOFT */
|
||||||
|
max_abi = ((ehdr->e_ident[EI_CLASS] == ELFCLASS32) &&
|
||||||
|
(!(ehdr->e_flags & EF_MIPS_ABI2))) ?
|
||||||
|
MIPS_ABI_FP_64A : MIPS_ABI_FP_SOFT;
|
||||||
|
|
||||||
if (abi0 == abi1) {
|
if ((abi0 > max_abi && abi0 != MIPS_ABI_FP_UNKNOWN) ||
|
||||||
state->overall_abi = abi0;
|
(abi1 > max_abi && abi1 != MIPS_ABI_FP_UNKNOWN))
|
||||||
} else if (abi0 == MIPS_ABI_FP_ANY) {
|
return -ELIBBAD;
|
||||||
state->overall_abi = abi1;
|
|
||||||
} else if (abi0 == MIPS_ABI_FP_DOUBLE) {
|
/* It's time to determine the FPU mode requirements */
|
||||||
switch (abi1) {
|
prog_req = (abi0 == MIPS_ABI_FP_UNKNOWN) ? none_req : fpu_reqs[abi0];
|
||||||
case MIPS_ABI_FP_XX:
|
interp_req = (abi1 == MIPS_ABI_FP_UNKNOWN) ? none_req : fpu_reqs[abi1];
|
||||||
state->overall_abi = MIPS_ABI_FP_DOUBLE;
|
|
||||||
break;
|
/*
|
||||||
|
* Check whether the program's and interp's ABIs have a matching FPU
|
||||||
case MIPS_ABI_FP_64A:
|
* mode requirement.
|
||||||
state->overall_abi = FP_DOUBLE_64A;
|
*/
|
||||||
break;
|
prog_req.single = interp_req.single && prog_req.single;
|
||||||
}
|
prog_req.soft = interp_req.soft && prog_req.soft;
|
||||||
} else if (abi0 == MIPS_ABI_FP_SINGLE ||
|
prog_req.fr1 = interp_req.fr1 && prog_req.fr1;
|
||||||
abi0 == MIPS_ABI_FP_SOFT) {
|
prog_req.frdefault = interp_req.frdefault && prog_req.frdefault;
|
||||||
/* Cannot link with other ABIs */
|
prog_req.fre = interp_req.fre && prog_req.fre;
|
||||||
} else if (abi0 == MIPS_ABI_FP_OLD_64) {
|
|
||||||
switch (abi1) {
|
/*
|
||||||
case MIPS_ABI_FP_XX:
|
* Determine the desired FPU mode
|
||||||
case MIPS_ABI_FP_64:
|
*
|
||||||
case MIPS_ABI_FP_64A:
|
* Decision making:
|
||||||
state->overall_abi = MIPS_ABI_FP_64;
|
*
|
||||||
break;
|
* - We want FR_FRE if FRE=1 and both FR=1 and FR=0 are false. This
|
||||||
}
|
* means that we have a combination of program and interpreter
|
||||||
} else if (abi0 == MIPS_ABI_FP_XX ||
|
* that inherently require the hybrid FP mode.
|
||||||
abi0 == MIPS_ABI_FP_64 ||
|
* - If FR1 and FRDEFAULT is true, that means we hit the any-abi or
|
||||||
abi0 == MIPS_ABI_FP_64A) {
|
* fpxx case. This is because, in any-ABI (or no-ABI) we have no FPU
|
||||||
state->overall_abi = MIPS_ABI_FP_64;
|
* instructions so we don't care about the mode. We will simply use
|
||||||
}
|
* the one preferred by the hardware. In fpxx case, that ABI can
|
||||||
|
* handle both FR=1 and FR=0, so, again, we simply choose the one
|
||||||
switch (state->overall_abi) {
|
* preferred by the hardware. Next, if we only use single-precision
|
||||||
case MIPS_ABI_FP_64:
|
* FPU instructions, and the default ABI FPU mode is not good
|
||||||
case MIPS_ABI_FP_64A:
|
* (ie single + any ABI combination), we set again the FPU mode to the
|
||||||
case FP_DOUBLE_64A:
|
* one is preferred by the hardware. Next, if we know that the code
|
||||||
if (!config_enabled(CONFIG_MIPS_O32_FP64_SUPPORT))
|
* will only use single-precision instructions, shown by single being
|
||||||
return -ELIBBAD;
|
* true but frdefault being false, then we again set the FPU mode to
|
||||||
break;
|
* the one that is preferred by the hardware.
|
||||||
|
* - We want FP_FR1 if that's the only matching mode and the default one
|
||||||
case FP_ERROR:
|
* is not good.
|
||||||
|
* - Return with -ELIBADD if we can't find a matching FPU mode.
|
||||||
|
*/
|
||||||
|
if (prog_req.fre && !prog_req.frdefault && !prog_req.fr1)
|
||||||
|
state->overall_fp_mode = FP_FRE;
|
||||||
|
else if ((prog_req.fr1 && prog_req.frdefault) ||
|
||||||
|
(prog_req.single && !prog_req.frdefault))
|
||||||
|
/* Make sure 64-bit MIPS III/IV/64R1 will not pick FR1 */
|
||||||
|
state->overall_fp_mode = ((current_cpu_data.fpu_id & MIPS_FPIR_F64) &&
|
||||||
|
cpu_has_mips_r2_r6) ?
|
||||||
|
FP_FR1 : FP_FR0;
|
||||||
|
else if (prog_req.fr1)
|
||||||
|
state->overall_fp_mode = FP_FR1;
|
||||||
|
else if (!prog_req.fre && !prog_req.frdefault &&
|
||||||
|
!prog_req.fr1 && !prog_req.single && !prog_req.soft)
|
||||||
return -ELIBBAD;
|
return -ELIBBAD;
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void set_thread_fp_mode(int hybrid, int regs32)
|
||||||
|
{
|
||||||
|
if (hybrid)
|
||||||
|
set_thread_flag(TIF_HYBRID_FPREGS);
|
||||||
|
else
|
||||||
|
clear_thread_flag(TIF_HYBRID_FPREGS);
|
||||||
|
if (regs32)
|
||||||
|
set_thread_flag(TIF_32BIT_FPREGS);
|
||||||
|
else
|
||||||
|
clear_thread_flag(TIF_32BIT_FPREGS);
|
||||||
|
}
|
||||||
|
|
||||||
void mips_set_personality_fp(struct arch_elf_state *state)
|
void mips_set_personality_fp(struct arch_elf_state *state)
|
||||||
{
|
{
|
||||||
if (config_enabled(CONFIG_FP32XX_HYBRID_FPRS)) {
|
/*
|
||||||
/*
|
* This function is only ever called for O32 ELFs so we should
|
||||||
* Use hybrid FPRs for all code which can correctly execute
|
* not be worried about N32/N64 binaries.
|
||||||
* with that mode.
|
*/
|
||||||
*/
|
|
||||||
switch (state->overall_abi) {
|
|
||||||
case MIPS_ABI_FP_DOUBLE:
|
|
||||||
case MIPS_ABI_FP_SINGLE:
|
|
||||||
case MIPS_ABI_FP_SOFT:
|
|
||||||
case MIPS_ABI_FP_XX:
|
|
||||||
case MIPS_ABI_FP_ANY:
|
|
||||||
/* FR=1, FRE=1 */
|
|
||||||
clear_thread_flag(TIF_32BIT_FPREGS);
|
|
||||||
set_thread_flag(TIF_HYBRID_FPREGS);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (state->overall_abi) {
|
if (!config_enabled(CONFIG_MIPS_O32_FP64_SUPPORT))
|
||||||
case MIPS_ABI_FP_DOUBLE:
|
return;
|
||||||
case MIPS_ABI_FP_SINGLE:
|
|
||||||
case MIPS_ABI_FP_SOFT:
|
switch (state->overall_fp_mode) {
|
||||||
/* FR=0 */
|
case FP_FRE:
|
||||||
set_thread_flag(TIF_32BIT_FPREGS);
|
set_thread_fp_mode(1, 0);
|
||||||
clear_thread_flag(TIF_HYBRID_FPREGS);
|
|
||||||
break;
|
break;
|
||||||
|
case FP_FR0:
|
||||||
case FP_DOUBLE_64A:
|
set_thread_fp_mode(0, 1);
|
||||||
/* FR=1, FRE=1 */
|
|
||||||
clear_thread_flag(TIF_32BIT_FPREGS);
|
|
||||||
set_thread_flag(TIF_HYBRID_FPREGS);
|
|
||||||
break;
|
break;
|
||||||
|
case FP_FR1:
|
||||||
case MIPS_ABI_FP_64:
|
set_thread_fp_mode(0, 0);
|
||||||
case MIPS_ABI_FP_64A:
|
|
||||||
/* FR=1, FRE=0 */
|
|
||||||
clear_thread_flag(TIF_32BIT_FPREGS);
|
|
||||||
clear_thread_flag(TIF_HYBRID_FPREGS);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MIPS_ABI_FP_XX:
|
|
||||||
case MIPS_ABI_FP_ANY:
|
|
||||||
if (!config_enabled(CONFIG_MIPS_O32_FP64_SUPPORT))
|
|
||||||
set_thread_flag(TIF_32BIT_FPREGS);
|
|
||||||
else
|
|
||||||
clear_thread_flag(TIF_32BIT_FPREGS);
|
|
||||||
|
|
||||||
clear_thread_flag(TIF_HYBRID_FPREGS);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
case FP_ERROR:
|
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Загрузка…
Ссылка в новой задаче