mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-08 14:54:23 +08:00
76fbfc318d
mm_isBranchInstr() did reside in the math emu code even though it logically is separate and also is used outside the math emu code. In addition GCC 4.9.0 leaves the following unnnecessarily bloated function body for a non-microMIPS configuration: <mm_isBranchInstr>: 105c: afa50004 sw a1,4(sp) 1060: afa60008 sw a2,8(sp) 1064: afa7000c sw a3,12(sp) 1068: 03e00008 jr ra 106c: 00001021 move v0,zero which stores arguments that are never going to be used on the stack frame. Move mm_isBranchInstr() from cp1emu.c to branch.c, then split mm_isBranchInstr() into a __mm_isBranchInstr() core and a mm_isBranchInstr() wrapper inline function which only invokes __mm_isBranchInstr() on microMIPS configurations. This shaves off 112 bytes off the kernel and improves code flow a bit. Signed-off-by: Ralf Baechle <ralf@linux-mips.org>
104 lines
2.4 KiB
C
104 lines
2.4 KiB
C
/*
|
|
* This file is subject to the terms and conditions of the GNU General Public
|
|
* License. See the file "COPYING" in the main directory of this archive
|
|
* for more details.
|
|
*
|
|
* Copyright (C) 1996, 1997, 1998, 2001 by Ralf Baechle
|
|
*/
|
|
#ifndef _ASM_BRANCH_H
|
|
#define _ASM_BRANCH_H
|
|
|
|
#include <asm/cpu-features.h>
|
|
#include <asm/mipsregs.h>
|
|
#include <asm/ptrace.h>
|
|
#include <asm/inst.h>
|
|
|
|
extern int __isa_exception_epc(struct pt_regs *regs);
|
|
extern int __compute_return_epc(struct pt_regs *regs);
|
|
extern int __compute_return_epc_for_insn(struct pt_regs *regs,
|
|
union mips_instruction insn);
|
|
extern int __microMIPS_compute_return_epc(struct pt_regs *regs);
|
|
extern int __MIPS16e_compute_return_epc(struct pt_regs *regs);
|
|
|
|
/*
|
|
* microMIPS bitfields
|
|
*/
|
|
#define MM_POOL32A_MINOR_MASK 0x3f
|
|
#define MM_POOL32A_MINOR_SHIFT 0x6
|
|
#define MM_MIPS32_COND_FC 0x30
|
|
|
|
extern int __mm_isBranchInstr(struct pt_regs *regs,
|
|
struct mm_decoded_insn dec_insn, unsigned long *contpc);
|
|
|
|
static inline int mm_isBranchInstr(struct pt_regs *regs,
|
|
struct mm_decoded_insn dec_insn, unsigned long *contpc)
|
|
{
|
|
if (!cpu_has_mmips)
|
|
return 0;
|
|
|
|
return __mm_isBranchInstr(regs, dec_insn, contpc);
|
|
}
|
|
|
|
static inline int delay_slot(struct pt_regs *regs)
|
|
{
|
|
return regs->cp0_cause & CAUSEF_BD;
|
|
}
|
|
|
|
static inline void clear_delay_slot(struct pt_regs *regs)
|
|
{
|
|
regs->cp0_cause &= ~CAUSEF_BD;
|
|
}
|
|
|
|
static inline void set_delay_slot(struct pt_regs *regs)
|
|
{
|
|
regs->cp0_cause |= CAUSEF_BD;
|
|
}
|
|
|
|
static inline unsigned long exception_epc(struct pt_regs *regs)
|
|
{
|
|
if (likely(!delay_slot(regs)))
|
|
return regs->cp0_epc;
|
|
|
|
if (get_isa16_mode(regs->cp0_epc))
|
|
return __isa_exception_epc(regs);
|
|
|
|
return regs->cp0_epc + 4;
|
|
}
|
|
|
|
#define BRANCH_LIKELY_TAKEN 0x0001
|
|
|
|
static inline int compute_return_epc(struct pt_regs *regs)
|
|
{
|
|
if (get_isa16_mode(regs->cp0_epc)) {
|
|
if (cpu_has_mmips)
|
|
return __microMIPS_compute_return_epc(regs);
|
|
if (cpu_has_mips16)
|
|
return __MIPS16e_compute_return_epc(regs);
|
|
return regs->cp0_epc;
|
|
}
|
|
|
|
if (!delay_slot(regs)) {
|
|
regs->cp0_epc += 4;
|
|
return 0;
|
|
}
|
|
|
|
return __compute_return_epc(regs);
|
|
}
|
|
|
|
static inline int MIPS16e_compute_return_epc(struct pt_regs *regs,
|
|
union mips16e_instruction *inst)
|
|
{
|
|
if (likely(!delay_slot(regs))) {
|
|
if (inst->ri.opcode == MIPS16e_extend_op) {
|
|
regs->cp0_epc += 4;
|
|
return 0;
|
|
}
|
|
regs->cp0_epc += 2;
|
|
return 0;
|
|
}
|
|
|
|
return __MIPS16e_compute_return_epc(regs);
|
|
}
|
|
|
|
#endif /* _ASM_BRANCH_H */
|