xref: /xnu-10002.1.13/san/memory/ubsan.h (revision 1031c584a5e37aff177559b9f69dbd3c8c3fd30a)
1*1031c584SApple OSS Distributions /*
2*1031c584SApple OSS Distributions  * Copyright (c) 2018-2021 Apple Inc. All rights reserved.
3*1031c584SApple OSS Distributions  *
4*1031c584SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*1031c584SApple OSS Distributions  *
6*1031c584SApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*1031c584SApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*1031c584SApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*1031c584SApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*1031c584SApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*1031c584SApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*1031c584SApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*1031c584SApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*1031c584SApple OSS Distributions  *
15*1031c584SApple OSS Distributions  * Please obtain a copy of the License at
16*1031c584SApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*1031c584SApple OSS Distributions  *
18*1031c584SApple OSS Distributions  * The Original Code and all software distributed under the License are
19*1031c584SApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*1031c584SApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*1031c584SApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*1031c584SApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*1031c584SApple OSS Distributions  * Please see the License for the specific language governing rights and
24*1031c584SApple OSS Distributions  * limitations under the License.
25*1031c584SApple OSS Distributions  *
26*1031c584SApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*1031c584SApple OSS Distributions  */
28*1031c584SApple OSS Distributions 
29*1031c584SApple OSS Distributions #ifndef _UBSAN_H_
30*1031c584SApple OSS Distributions #define _UBSAN_H_
31*1031c584SApple OSS Distributions 
32*1031c584SApple OSS Distributions #include <stdint.h>
33*1031c584SApple OSS Distributions #include <stdbool.h>
34*1031c584SApple OSS Distributions 
35*1031c584SApple OSS Distributions struct san_type_desc {
36*1031c584SApple OSS Distributions 	uint16_t type; // 0: integer, 1: float
37*1031c584SApple OSS Distributions 	union {
38*1031c584SApple OSS Distributions 		struct {
39*1031c584SApple OSS Distributions 			uint16_t issigned : 1;
40*1031c584SApple OSS Distributions 			uint16_t width    : 15;
41*1031c584SApple OSS Distributions 		}; /* int descriptor */
42*1031c584SApple OSS Distributions 		struct {
43*1031c584SApple OSS Distributions 			uint16_t float_desc;
44*1031c584SApple OSS Distributions 		}; /* float descriptor */
45*1031c584SApple OSS Distributions 	};
46*1031c584SApple OSS Distributions 	const char name[];
47*1031c584SApple OSS Distributions };
48*1031c584SApple OSS Distributions 
49*1031c584SApple OSS Distributions struct san_src_loc {
50*1031c584SApple OSS Distributions 	const char *filename;
51*1031c584SApple OSS Distributions 	uint32_t line;
52*1031c584SApple OSS Distributions 	uint32_t col;
53*1031c584SApple OSS Distributions };
54*1031c584SApple OSS Distributions 
55*1031c584SApple OSS Distributions struct ubsan_overflow_desc {
56*1031c584SApple OSS Distributions 	struct san_src_loc loc;
57*1031c584SApple OSS Distributions 	struct san_type_desc *ty;
58*1031c584SApple OSS Distributions };
59*1031c584SApple OSS Distributions 
60*1031c584SApple OSS Distributions struct ubsan_unreachable_desc {
61*1031c584SApple OSS Distributions 	struct san_src_loc loc;
62*1031c584SApple OSS Distributions };
63*1031c584SApple OSS Distributions 
64*1031c584SApple OSS Distributions struct ubsan_shift_desc {
65*1031c584SApple OSS Distributions 	struct san_src_loc loc;
66*1031c584SApple OSS Distributions 	struct san_type_desc *lhs_t;
67*1031c584SApple OSS Distributions 	struct san_type_desc *rhs_t;
68*1031c584SApple OSS Distributions };
69*1031c584SApple OSS Distributions 
70*1031c584SApple OSS Distributions struct ubsan_align_desc {
71*1031c584SApple OSS Distributions 	struct san_src_loc loc;
72*1031c584SApple OSS Distributions 	struct san_type_desc *ty;
73*1031c584SApple OSS Distributions 	uint8_t align;
74*1031c584SApple OSS Distributions 	uint8_t kind;
75*1031c584SApple OSS Distributions };
76*1031c584SApple OSS Distributions 
77*1031c584SApple OSS Distributions struct ubsan_ptroverflow_desc {
78*1031c584SApple OSS Distributions 	struct san_src_loc loc;
79*1031c584SApple OSS Distributions };
80*1031c584SApple OSS Distributions 
81*1031c584SApple OSS Distributions struct ubsan_oob_desc {
82*1031c584SApple OSS Distributions 	struct san_src_loc loc;
83*1031c584SApple OSS Distributions 	struct san_type_desc *array_ty;
84*1031c584SApple OSS Distributions 	struct san_type_desc *index_ty;
85*1031c584SApple OSS Distributions };
86*1031c584SApple OSS Distributions 
87*1031c584SApple OSS Distributions struct ubsan_load_invalid_desc {
88*1031c584SApple OSS Distributions 	struct san_src_loc loc;
89*1031c584SApple OSS Distributions 	struct san_type_desc *type;
90*1031c584SApple OSS Distributions };
91*1031c584SApple OSS Distributions 
92*1031c584SApple OSS Distributions struct ubsan_nullability_arg_desc {
93*1031c584SApple OSS Distributions 	struct san_src_loc loc;
94*1031c584SApple OSS Distributions 	struct san_src_loc attr_loc;
95*1031c584SApple OSS Distributions 	int arg_index;
96*1031c584SApple OSS Distributions };
97*1031c584SApple OSS Distributions 
98*1031c584SApple OSS Distributions struct ubsan_nullability_ret_desc {
99*1031c584SApple OSS Distributions 	struct san_src_loc loc;
100*1031c584SApple OSS Distributions };
101*1031c584SApple OSS Distributions 
102*1031c584SApple OSS Distributions struct ubsan_missing_ret_desc {
103*1031c584SApple OSS Distributions 	struct san_src_loc loc;
104*1031c584SApple OSS Distributions };
105*1031c584SApple OSS Distributions 
106*1031c584SApple OSS Distributions struct ubsan_float_desc {
107*1031c584SApple OSS Distributions 	struct san_src_loc loc;
108*1031c584SApple OSS Distributions 	struct san_type_desc *type_from;
109*1031c584SApple OSS Distributions 	struct san_type_desc *type_to;
110*1031c584SApple OSS Distributions };
111*1031c584SApple OSS Distributions 
112*1031c584SApple OSS Distributions struct ubsan_implicit_conv_desc {
113*1031c584SApple OSS Distributions 	struct san_src_loc loc;
114*1031c584SApple OSS Distributions 	struct san_type_desc *type_from;
115*1031c584SApple OSS Distributions 	struct san_type_desc *type_to;
116*1031c584SApple OSS Distributions 	unsigned char kind;
117*1031c584SApple OSS Distributions };
118*1031c584SApple OSS Distributions 
119*1031c584SApple OSS Distributions struct ubsan_func_type_mismatch_desc {
120*1031c584SApple OSS Distributions 	struct san_src_loc loc;
121*1031c584SApple OSS Distributions 	struct san_type_desc *type;
122*1031c584SApple OSS Distributions };
123*1031c584SApple OSS Distributions 
124*1031c584SApple OSS Distributions struct ubsan_vla_bound_desc {
125*1031c584SApple OSS Distributions 	struct san_src_loc loc;
126*1031c584SApple OSS Distributions 	struct san_type_desc *type;
127*1031c584SApple OSS Distributions };
128*1031c584SApple OSS Distributions 
129*1031c584SApple OSS Distributions struct ubsan_invalid_builtin {
130*1031c584SApple OSS Distributions 	struct san_src_loc loc;
131*1031c584SApple OSS Distributions 	unsigned char kind;
132*1031c584SApple OSS Distributions };
133*1031c584SApple OSS Distributions 
134*1031c584SApple OSS Distributions OS_ENUM(ubsan_violation_type, uint8_t,
135*1031c584SApple OSS Distributions     UBSAN_OVERFLOW_add = 1,
136*1031c584SApple OSS Distributions     UBSAN_OVERFLOW_sub,
137*1031c584SApple OSS Distributions     UBSAN_OVERFLOW_mul,
138*1031c584SApple OSS Distributions     UBSAN_OVERFLOW_divrem,
139*1031c584SApple OSS Distributions     UBSAN_OVERFLOW_negate,
140*1031c584SApple OSS Distributions     UBSAN_UNREACHABLE,
141*1031c584SApple OSS Distributions     UBSAN_SHIFT,
142*1031c584SApple OSS Distributions     UBSAN_ALIGN,
143*1031c584SApple OSS Distributions     UBSAN_POINTER_OVERFLOW,
144*1031c584SApple OSS Distributions     UBSAN_OOB,
145*1031c584SApple OSS Distributions     UBSAN_TYPE_MISMATCH,
146*1031c584SApple OSS Distributions     UBSAN_LOAD_INVALID_VALUE,
147*1031c584SApple OSS Distributions     UBSAN_NULLABILITY_ARG,
148*1031c584SApple OSS Distributions     UBSAN_NULLABILITY_RETURN,
149*1031c584SApple OSS Distributions     UBSAN_MISSING_RETURN,
150*1031c584SApple OSS Distributions     UBSAN_FLOAT_CAST_OVERFLOW,
151*1031c584SApple OSS Distributions     UBSAN_IMPLICIT_CONVERSION,
152*1031c584SApple OSS Distributions     UBSAN_FUNCTION_TYPE_MISMATCH,
153*1031c584SApple OSS Distributions     UBSAN_VLA_BOUND_NOT_POSITIVE,
154*1031c584SApple OSS Distributions     UBSAN_INVALID_BUILTIN,
155*1031c584SApple OSS Distributions     UBSAN_VIOLATION_MAX
156*1031c584SApple OSS Distributions     );
157*1031c584SApple OSS Distributions 
158*1031c584SApple OSS Distributions typedef struct ubsan_violation {
159*1031c584SApple OSS Distributions 	ubsan_violation_type_t ubsan_type;
160*1031c584SApple OSS Distributions 	uint64_t lhs;
161*1031c584SApple OSS Distributions 	uint64_t rhs;
162*1031c584SApple OSS Distributions 	union {
163*1031c584SApple OSS Distributions 		struct ubsan_overflow_desc *overflow;
164*1031c584SApple OSS Distributions 		struct ubsan_unreachable_desc *unreachable;
165*1031c584SApple OSS Distributions 		struct ubsan_shift_desc *shift;
166*1031c584SApple OSS Distributions 		struct ubsan_align_desc *align;
167*1031c584SApple OSS Distributions 		struct ubsan_ptroverflow_desc *ptroverflow;
168*1031c584SApple OSS Distributions 		struct ubsan_oob_desc *oob;
169*1031c584SApple OSS Distributions 		struct ubsan_load_invalid_desc *invalid;
170*1031c584SApple OSS Distributions 		struct ubsan_nullability_arg_desc *nonnull_arg;
171*1031c584SApple OSS Distributions 		struct ubsan_nullability_ret_desc *nonnull_ret;
172*1031c584SApple OSS Distributions 		struct ubsan_missing_ret_desc *missing_ret;
173*1031c584SApple OSS Distributions 		struct ubsan_float_desc *flt;
174*1031c584SApple OSS Distributions 		struct ubsan_implicit_conv_desc *implicit;
175*1031c584SApple OSS Distributions 		struct ubsan_func_type_mismatch_desc *func_mismatch;
176*1031c584SApple OSS Distributions 		struct ubsan_vla_bound_desc *vla_bound;
177*1031c584SApple OSS Distributions 		struct ubsan_invalid_builtin *invalid_builtin;
178*1031c584SApple OSS Distributions 		const char *func;
179*1031c584SApple OSS Distributions 	};
180*1031c584SApple OSS Distributions 	struct san_src_loc *loc;
181*1031c584SApple OSS Distributions } ubsan_violation_t;
182*1031c584SApple OSS Distributions 
183*1031c584SApple OSS Distributions typedef struct ubsan_buf {
184*1031c584SApple OSS Distributions 	char    *ub_buf;
185*1031c584SApple OSS Distributions 	size_t  ub_buf_size;
186*1031c584SApple OSS Distributions 	size_t  ub_written;
187*1031c584SApple OSS Distributions 	bool    ub_err;
188*1031c584SApple OSS Distributions } ubsan_buf_t;
189*1031c584SApple OSS Distributions 
190*1031c584SApple OSS Distributions void    ubsan_log_append(ubsan_violation_t *);
191*1031c584SApple OSS Distributions 
192*1031c584SApple OSS Distributions void    ubsan_json_init(ubsan_buf_t *, char *, size_t);
193*1031c584SApple OSS Distributions void    ubsan_json_begin(ubsan_buf_t *, size_t);
194*1031c584SApple OSS Distributions size_t  ubsan_json_finish(ubsan_buf_t *);
195*1031c584SApple OSS Distributions bool    ubsan_json_format(ubsan_violation_t *, ubsan_buf_t *);
196*1031c584SApple OSS Distributions 
197*1031c584SApple OSS Distributions /*
198*1031c584SApple OSS Distributions  * UBSan ABI
199*1031c584SApple OSS Distributions  */
200*1031c584SApple OSS Distributions void __ubsan_handle_add_overflow(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
201*1031c584SApple OSS Distributions void __ubsan_handle_add_overflow_abort(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
202*1031c584SApple OSS Distributions void __ubsan_handle_builtin_unreachable(struct ubsan_unreachable_desc *);
203*1031c584SApple OSS Distributions void __ubsan_handle_divrem_overflow(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
204*1031c584SApple OSS Distributions void __ubsan_handle_divrem_overflow_abort(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
205*1031c584SApple OSS Distributions void __ubsan_handle_float_cast_overflow(struct ubsan_float_desc *, uint64_t);
206*1031c584SApple OSS Distributions void __ubsan_handle_float_cast_overflow_abort(struct ubsan_float_desc *, uint64_t);
207*1031c584SApple OSS Distributions void __ubsan_handle_function_type_mismatch(struct ubsan_func_type_mismatch_desc*, uint64_t);
208*1031c584SApple OSS Distributions void __ubsan_handle_function_type_mismatch_abort(struct ubsan_func_type_mismatch_desc *, uint64_t);
209*1031c584SApple OSS Distributions void __ubsan_handle_implicit_conversion(struct ubsan_implicit_conv_desc *, uint64_t, uint64_t);
210*1031c584SApple OSS Distributions void __ubsan_handle_implicit_conversion_abort(struct ubsan_implicit_conv_desc *, uint64_t, uint64_t);
211*1031c584SApple OSS Distributions void __ubsan_handle_invalid_builtin(struct ubsan_invalid_builtin *);
212*1031c584SApple OSS Distributions void __ubsan_handle_invalid_builtin_abort(struct ubsan_invalid_builtin *);
213*1031c584SApple OSS Distributions void __ubsan_handle_load_invalid_value(struct ubsan_load_invalid_desc *, uint64_t);
214*1031c584SApple OSS Distributions void __ubsan_handle_load_invalid_value_abort(struct ubsan_load_invalid_desc *, uint64_t);
215*1031c584SApple OSS Distributions void __ubsan_handle_missing_return(struct ubsan_missing_ret_desc *);
216*1031c584SApple OSS Distributions void __ubsan_handle_missing_return_abort(struct ubsan_missing_ret_desc *);
217*1031c584SApple OSS Distributions void __ubsan_handle_mul_overflow(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
218*1031c584SApple OSS Distributions void __ubsan_handle_mul_overflow_abort(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
219*1031c584SApple OSS Distributions void __ubsan_handle_negate_overflow(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
220*1031c584SApple OSS Distributions void __ubsan_handle_negate_overflow_abort(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
221*1031c584SApple OSS Distributions void __ubsan_handle_nonnull_arg(struct ubsan_nullability_arg_desc *);
222*1031c584SApple OSS Distributions void __ubsan_handle_nonnull_arg_abort(struct ubsan_nullability_arg_desc *);
223*1031c584SApple OSS Distributions void __ubsan_handle_nonnull_return_v1(struct ubsan_nullability_ret_desc *, uint64_t);
224*1031c584SApple OSS Distributions void __ubsan_handle_nonnull_return_v1_abort(struct ubsan_nullability_ret_desc *, uint64_t);
225*1031c584SApple OSS Distributions void __ubsan_handle_nullability_arg(struct ubsan_nullability_arg_desc *);
226*1031c584SApple OSS Distributions void __ubsan_handle_nullability_arg_abort(struct ubsan_nullability_arg_desc *);
227*1031c584SApple OSS Distributions void __ubsan_handle_nullability_return_v1(struct ubsan_nullability_ret_desc *, uint64_t);
228*1031c584SApple OSS Distributions void __ubsan_handle_nullability_return_v1_abort(struct ubsan_nullability_ret_desc *, uint64_t);
229*1031c584SApple OSS Distributions void __ubsan_handle_out_of_bounds(struct ubsan_oob_desc *, uint64_t idx);
230*1031c584SApple OSS Distributions void __ubsan_handle_out_of_bounds_abort(struct ubsan_oob_desc *, uint64_t idx);
231*1031c584SApple OSS Distributions void __ubsan_handle_pointer_overflow(struct ubsan_ptroverflow_desc *, uint64_t lhs, uint64_t rhs);
232*1031c584SApple OSS Distributions void __ubsan_handle_pointer_overflow_abort(struct ubsan_ptroverflow_desc *, uint64_t lhs, uint64_t rhs);
233*1031c584SApple OSS Distributions void __ubsan_handle_shift_out_of_bounds(struct ubsan_shift_desc *, uint64_t lhs, uint64_t rhs);
234*1031c584SApple OSS Distributions void __ubsan_handle_shift_out_of_bounds_abort(struct ubsan_shift_desc *, uint64_t lhs, uint64_t rhs);
235*1031c584SApple OSS Distributions void __ubsan_handle_sub_overflow(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
236*1031c584SApple OSS Distributions void __ubsan_handle_sub_overflow_abort(struct ubsan_overflow_desc *, uint64_t lhs, uint64_t rhs);
237*1031c584SApple OSS Distributions void __ubsan_handle_type_mismatch_v1(struct ubsan_align_desc *, uint64_t val);
238*1031c584SApple OSS Distributions void __ubsan_handle_type_mismatch_v1_abort(struct ubsan_align_desc *, uint64_t val);
239*1031c584SApple OSS Distributions void __ubsan_handle_vla_bound_not_positive(struct ubsan_vla_bound_desc *, uint64_t);
240*1031c584SApple OSS Distributions void __ubsan_handle_vla_bound_not_positive_abort(struct ubsan_vla_bound_desc *, uint64_t);
241*1031c584SApple OSS Distributions 
242*1031c584SApple OSS Distributions #endif /* _UBSAN_H_ */
243