| .. | .. |
|---|
| 1 | 1 | /* SPDX-License-Identifier: GPL-2.0 */ |
|---|
| 2 | 2 | #ifndef __S390_EXTABLE_H |
|---|
| 3 | 3 | #define __S390_EXTABLE_H |
|---|
| 4 | + |
|---|
| 5 | +#include <asm/ptrace.h> |
|---|
| 6 | +#include <linux/compiler.h> |
|---|
| 7 | + |
|---|
| 4 | 8 | /* |
|---|
| 5 | | - * The exception table consists of pairs of addresses: the first is the |
|---|
| 6 | | - * address of an instruction that is allowed to fault, and the second is |
|---|
| 7 | | - * the address at which the program should continue. No registers are |
|---|
| 8 | | - * modified, so it is entirely up to the continuation code to figure out |
|---|
| 9 | | - * what to do. |
|---|
| 9 | + * The exception table consists of three addresses: |
|---|
| 10 | + * |
|---|
| 11 | + * - Address of an instruction that is allowed to fault. |
|---|
| 12 | + * - Address at which the program should continue. |
|---|
| 13 | + * - Optional address of handler that takes pt_regs * argument and runs in |
|---|
| 14 | + * interrupt context. |
|---|
| 15 | + * |
|---|
| 16 | + * No registers are modified, so it is entirely up to the continuation code |
|---|
| 17 | + * to figure out what to do. |
|---|
| 10 | 18 | * |
|---|
| 11 | 19 | * All the routines below use bits of fixup code that are out of line |
|---|
| 12 | 20 | * with the main instruction path. This means when everything is well, |
|---|
| .. | .. |
|---|
| 17 | 25 | struct exception_table_entry |
|---|
| 18 | 26 | { |
|---|
| 19 | 27 | int insn, fixup; |
|---|
| 28 | + long handler; |
|---|
| 20 | 29 | }; |
|---|
| 30 | + |
|---|
| 31 | +extern struct exception_table_entry *__start_dma_ex_table; |
|---|
| 32 | +extern struct exception_table_entry *__stop_dma_ex_table; |
|---|
| 33 | + |
|---|
| 34 | +const struct exception_table_entry *s390_search_extables(unsigned long addr); |
|---|
| 21 | 35 | |
|---|
| 22 | 36 | static inline unsigned long extable_fixup(const struct exception_table_entry *x) |
|---|
| 23 | 37 | { |
|---|
| 24 | 38 | return (unsigned long)&x->fixup + x->fixup; |
|---|
| 25 | 39 | } |
|---|
| 26 | 40 | |
|---|
| 41 | +typedef bool (*ex_handler_t)(const struct exception_table_entry *, |
|---|
| 42 | + struct pt_regs *); |
|---|
| 43 | + |
|---|
| 44 | +static inline ex_handler_t |
|---|
| 45 | +ex_fixup_handler(const struct exception_table_entry *x) |
|---|
| 46 | +{ |
|---|
| 47 | + if (likely(!x->handler)) |
|---|
| 48 | + return NULL; |
|---|
| 49 | + return (ex_handler_t)((unsigned long)&x->handler + x->handler); |
|---|
| 50 | +} |
|---|
| 51 | + |
|---|
| 52 | +static inline bool ex_handle(const struct exception_table_entry *x, |
|---|
| 53 | + struct pt_regs *regs) |
|---|
| 54 | +{ |
|---|
| 55 | + ex_handler_t handler = ex_fixup_handler(x); |
|---|
| 56 | + |
|---|
| 57 | + if (unlikely(handler)) |
|---|
| 58 | + return handler(x, regs); |
|---|
| 59 | + regs->psw.addr = extable_fixup(x); |
|---|
| 60 | + return true; |
|---|
| 61 | +} |
|---|
| 62 | + |
|---|
| 27 | 63 | #define ARCH_HAS_RELATIVE_EXTABLE |
|---|
| 28 | 64 | |
|---|
| 65 | +static inline void swap_ex_entry_fixup(struct exception_table_entry *a, |
|---|
| 66 | + struct exception_table_entry *b, |
|---|
| 67 | + struct exception_table_entry tmp, |
|---|
| 68 | + int delta) |
|---|
| 69 | +{ |
|---|
| 70 | + a->fixup = b->fixup + delta; |
|---|
| 71 | + b->fixup = tmp.fixup - delta; |
|---|
| 72 | + a->handler = b->handler; |
|---|
| 73 | + if (a->handler) |
|---|
| 74 | + a->handler += delta; |
|---|
| 75 | + b->handler = tmp.handler; |
|---|
| 76 | + if (b->handler) |
|---|
| 77 | + b->handler -= delta; |
|---|
| 78 | +} |
|---|
| 79 | +#define swap_ex_entry_fixup swap_ex_entry_fixup |
|---|
| 80 | + |
|---|
| 29 | 81 | #endif |
|---|