1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef __ASM_EXTABLE_H
3 #define __ASM_EXTABLE_H
4 
5 #include <linux/stringify.h>
6 #include <linux/bits.h>
7 #include <asm/asm-const.h>
8 
9 #define EX_TYPE_NONE		0
10 #define EX_TYPE_FIXUP		1
11 #define EX_TYPE_BPF		2
12 #define EX_TYPE_UA_FAULT	3
13 #define EX_TYPE_UA_LOAD_REG	5
14 #define EX_TYPE_UA_LOAD_REGPAIR	6
15 #define EX_TYPE_ZEROPAD		7
16 #define EX_TYPE_FPC		8
17 #define EX_TYPE_UA_MVCOS_TO	9
18 #define EX_TYPE_UA_MVCOS_FROM	10
19 
20 #define EX_DATA_REG_ERR_SHIFT	0
21 #define EX_DATA_REG_ERR		GENMASK(3, 0)
22 
23 #define EX_DATA_REG_ADDR_SHIFT	4
24 #define EX_DATA_REG_ADDR	GENMASK(7, 4)
25 
26 #define EX_DATA_LEN_SHIFT	8
27 #define EX_DATA_LEN		GENMASK(11, 8)
28 
29 #define __EX_TABLE(_section, _fault, _target, _type, _regerr, _regaddr, _len)	\
30 	stringify_in_c(.section _section,"a";)					\
31 	stringify_in_c(.balign	4;)						\
32 	stringify_in_c(.long	(_fault) - .;)					\
33 	stringify_in_c(.long	(_target) - .;)					\
34 	stringify_in_c(.short	(_type);)					\
35 	stringify_in_c(.macro	extable_reg regerr, regaddr;)			\
36 	stringify_in_c(.set	.Lfound, 0;)					\
37 	stringify_in_c(.set	.Lcurr, 0;)					\
38 	stringify_in_c(.irp	rs,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15;)	\
39 	stringify_in_c(		.ifc	"\regerr", "%%r\rs";)			\
40 	stringify_in_c(			.set	.Lfound, 1;)			\
41 	stringify_in_c(			.set	.Lregerr, .Lcurr;)		\
42 	stringify_in_c(		.endif;)					\
43 	stringify_in_c(		.set	.Lcurr, .Lcurr+1;)			\
44 	stringify_in_c(.endr;)							\
45 	stringify_in_c(.ifne	(.Lfound != 1);)				\
46 	stringify_in_c(		.error	"extable_reg: bad register argument1";)	\
47 	stringify_in_c(.endif;)							\
48 	stringify_in_c(.set	.Lfound, 0;)					\
49 	stringify_in_c(.set	.Lcurr, 0;)					\
50 	stringify_in_c(.irp	rs,0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15;)	\
51 	stringify_in_c(		.ifc	"\regaddr", "%%r\rs";)			\
52 	stringify_in_c(			.set	.Lfound, 1;)			\
53 	stringify_in_c(			.set	.Lregaddr, .Lcurr;)		\
54 	stringify_in_c(		.endif;)					\
55 	stringify_in_c(		.set	.Lcurr, .Lcurr+1;)			\
56 	stringify_in_c(.endr;)							\
57 	stringify_in_c(.ifne	(.Lfound != 1);)				\
58 	stringify_in_c(		.error	"extable_reg: bad register argument2";)	\
59 	stringify_in_c(.endif;)							\
60 	stringify_in_c(.short	.Lregerr << EX_DATA_REG_ERR_SHIFT |		\
61 				.Lregaddr << EX_DATA_REG_ADDR_SHIFT |		\
62 				_len << EX_DATA_LEN_SHIFT;)			\
63 	stringify_in_c(.endm;)							\
64 	stringify_in_c(extable_reg _regerr,_regaddr;)				\
65 	stringify_in_c(.purgem	extable_reg;)					\
66 	stringify_in_c(.previous)
67 
68 #define EX_TABLE(_fault, _target)					\
69 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_FIXUP, __stringify(%%r0), __stringify(%%r0), 0)
70 
71 #define EX_TABLE_AMODE31(_fault, _target)				\
72 	__EX_TABLE(.amode31.ex_table, _fault, _target, EX_TYPE_FIXUP, __stringify(%%r0), __stringify(%%r0), 0)
73 
74 #define EX_TABLE_UA_FAULT(_fault, _target, _regerr)			\
75 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_UA_FAULT, _regerr, _regerr, 0)
76 
77 #define EX_TABLE_UA_LOAD_REG(_fault, _target, _regerr, _regzero)	\
78 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_UA_LOAD_REG, _regerr, _regzero, 0)
79 
80 #define EX_TABLE_UA_LOAD_REGPAIR(_fault, _target, _regerr, _regzero)	\
81 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_UA_LOAD_REGPAIR, _regerr, _regzero, 0)
82 
83 #define EX_TABLE_ZEROPAD(_fault, _target, _regdata, _regaddr)		\
84 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_ZEROPAD, _regdata, _regaddr, 0)
85 
86 #define EX_TABLE_FPC(_fault, _target)					\
87 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_FPC, __stringify(%%r0), __stringify(%%r0), 0)
88 
89 #define EX_TABLE_UA_MVCOS_TO(_fault, _target)				\
90 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_UA_MVCOS_TO, __stringify(%%r0), __stringify(%%r0), 0)
91 
92 #define EX_TABLE_UA_MVCOS_FROM(_fault, _target)				\
93 	__EX_TABLE(__ex_table, _fault, _target, EX_TYPE_UA_MVCOS_FROM, __stringify(%%r0), __stringify(%%r0), 0)
94 
95 #endif /* __ASM_EXTABLE_H */
96