65 lines
2.2 KiB
C
65 lines
2.2 KiB
C
|
/* SPDX-License-Identifier: GPL-2.0 */
|
||
|
#ifndef __PARISC_EXTABLE_H
|
||
|
#define __PARISC_EXTABLE_H
|
||
|
|
||
|
#include <asm/ptrace.h>
|
||
|
#include <linux/compiler.h>
|
||
|
|
||
|
/*
|
||
|
* The exception table consists of three addresses:
|
||
|
*
|
||
|
* - A relative address to the instruction that is allowed to fault.
|
||
|
* - A relative address at which the program should continue (fixup routine)
|
||
|
* - An asm statement which specifies which CPU register will
|
||
|
* receive -EFAULT when an exception happens if the lowest bit in
|
||
|
* the fixup address is set.
|
||
|
*
|
||
|
* Note: The register specified in the err_opcode instruction will be
|
||
|
* modified at runtime if a fault happens. Register %r0 will be ignored.
|
||
|
*
|
||
|
* Since relative addresses are used, 32bit values are sufficient even on
|
||
|
* 64bit kernel.
|
||
|
*/
|
||
|
|
||
|
struct pt_regs;
|
||
|
int fixup_exception(struct pt_regs *regs);
|
||
|
|
||
|
#define ARCH_HAS_RELATIVE_EXTABLE
|
||
|
struct exception_table_entry {
|
||
|
int insn; /* relative address of insn that is allowed to fault. */
|
||
|
int fixup; /* relative address of fixup routine */
|
||
|
int err_opcode; /* sample opcode with register which holds error code */
|
||
|
};
|
||
|
|
||
|
#define ASM_EXCEPTIONTABLE_ENTRY( fault_addr, except_addr, opcode )\
|
||
|
".section __ex_table,\"aw\"\n" \
|
||
|
".align 4\n" \
|
||
|
".word (" #fault_addr " - .), (" #except_addr " - .)\n" \
|
||
|
opcode "\n" \
|
||
|
".previous\n"
|
||
|
|
||
|
/*
|
||
|
* ASM_EXCEPTIONTABLE_ENTRY_EFAULT() creates a special exception table entry
|
||
|
* (with lowest bit set) for which the fault handler in fixup_exception() will
|
||
|
* load -EFAULT on fault into the register specified by the err_opcode instruction,
|
||
|
* and zeroes the target register in case of a read fault in get_user().
|
||
|
*/
|
||
|
#define ASM_EXCEPTIONTABLE_VAR(__err_var) \
|
||
|
int __err_var = 0
|
||
|
#define ASM_EXCEPTIONTABLE_ENTRY_EFAULT( fault_addr, except_addr, register )\
|
||
|
ASM_EXCEPTIONTABLE_ENTRY( fault_addr, except_addr + 1, "or %%r0,%%r0," register)
|
||
|
|
||
|
static inline void swap_ex_entry_fixup(struct exception_table_entry *a,
|
||
|
struct exception_table_entry *b,
|
||
|
struct exception_table_entry tmp,
|
||
|
int delta)
|
||
|
{
|
||
|
a->fixup = b->fixup + delta;
|
||
|
b->fixup = tmp.fixup - delta;
|
||
|
a->err_opcode = b->err_opcode;
|
||
|
b->err_opcode = tmp.err_opcode;
|
||
|
}
|
||
|
#define swap_ex_entry_fixup swap_ex_entry_fixup
|
||
|
|
||
|
#endif
|