1*fa69a806SHelge Deller /* SPDX-License-Identifier: GPL-2.0 */
2*fa69a806SHelge Deller #ifndef __PARISC_EXTABLE_H
3*fa69a806SHelge Deller #define __PARISC_EXTABLE_H
4*fa69a806SHelge Deller
5*fa69a806SHelge Deller #include <asm/ptrace.h>
6*fa69a806SHelge Deller #include <linux/compiler.h>
7*fa69a806SHelge Deller
8*fa69a806SHelge Deller /*
9*fa69a806SHelge Deller * The exception table consists of three addresses:
10*fa69a806SHelge Deller *
11*fa69a806SHelge Deller * - A relative address to the instruction that is allowed to fault.
12*fa69a806SHelge Deller * - A relative address at which the program should continue (fixup routine)
13*fa69a806SHelge Deller * - An asm statement which specifies which CPU register will
14*fa69a806SHelge Deller * receive -EFAULT when an exception happens if the lowest bit in
15*fa69a806SHelge Deller * the fixup address is set.
16*fa69a806SHelge Deller *
17*fa69a806SHelge Deller * Note: The register specified in the err_opcode instruction will be
18*fa69a806SHelge Deller * modified at runtime if a fault happens. Register %r0 will be ignored.
19*fa69a806SHelge Deller *
20*fa69a806SHelge Deller * Since relative addresses are used, 32bit values are sufficient even on
21*fa69a806SHelge Deller * 64bit kernel.
22*fa69a806SHelge Deller */
23*fa69a806SHelge Deller
24*fa69a806SHelge Deller struct pt_regs;
25*fa69a806SHelge Deller int fixup_exception(struct pt_regs *regs);
26*fa69a806SHelge Deller
27*fa69a806SHelge Deller #define ARCH_HAS_RELATIVE_EXTABLE
28*fa69a806SHelge Deller struct exception_table_entry {
29*fa69a806SHelge Deller int insn; /* relative address of insn that is allowed to fault. */
30*fa69a806SHelge Deller int fixup; /* relative address of fixup routine */
31*fa69a806SHelge Deller int err_opcode; /* sample opcode with register which holds error code */
32*fa69a806SHelge Deller };
33*fa69a806SHelge Deller
34*fa69a806SHelge Deller #define ASM_EXCEPTIONTABLE_ENTRY( fault_addr, except_addr, opcode )\
35*fa69a806SHelge Deller ".section __ex_table,\"aw\"\n" \
36*fa69a806SHelge Deller ".align 4\n" \
37*fa69a806SHelge Deller ".word (" #fault_addr " - .), (" #except_addr " - .)\n" \
38*fa69a806SHelge Deller opcode "\n" \
39*fa69a806SHelge Deller ".previous\n"
40*fa69a806SHelge Deller
41*fa69a806SHelge Deller /*
42*fa69a806SHelge Deller * ASM_EXCEPTIONTABLE_ENTRY_EFAULT() creates a special exception table entry
43*fa69a806SHelge Deller * (with lowest bit set) for which the fault handler in fixup_exception() will
44*fa69a806SHelge Deller * load -EFAULT on fault into the register specified by the err_opcode instruction,
45*fa69a806SHelge Deller * and zeroes the target register in case of a read fault in get_user().
46*fa69a806SHelge Deller */
47*fa69a806SHelge Deller #define ASM_EXCEPTIONTABLE_VAR(__err_var) \
48*fa69a806SHelge Deller int __err_var = 0
49*fa69a806SHelge Deller #define ASM_EXCEPTIONTABLE_ENTRY_EFAULT( fault_addr, except_addr, register )\
50*fa69a806SHelge Deller ASM_EXCEPTIONTABLE_ENTRY( fault_addr, except_addr + 1, "or %%r0,%%r0," register)
51*fa69a806SHelge Deller
swap_ex_entry_fixup(struct exception_table_entry * a,struct exception_table_entry * b,struct exception_table_entry tmp,int delta)52*fa69a806SHelge Deller static inline void swap_ex_entry_fixup(struct exception_table_entry *a,
53*fa69a806SHelge Deller struct exception_table_entry *b,
54*fa69a806SHelge Deller struct exception_table_entry tmp,
55*fa69a806SHelge Deller int delta)
56*fa69a806SHelge Deller {
57*fa69a806SHelge Deller a->fixup = b->fixup + delta;
58*fa69a806SHelge Deller b->fixup = tmp.fixup - delta;
59*fa69a806SHelge Deller a->err_opcode = b->err_opcode;
60*fa69a806SHelge Deller b->err_opcode = tmp.err_opcode;
61*fa69a806SHelge Deller }
62*fa69a806SHelge Deller #define swap_ex_entry_fixup swap_ex_entry_fixup
63*fa69a806SHelge Deller
64*fa69a806SHelge Deller #endif
65