]> git.saurik.com Git - apple/xnu.git/blame - san/ubsan.c
xnu-7195.101.1.tar.gz
[apple/xnu.git] / san / ubsan.c
CommitLineData
d9a64523
A
1/*
2 * Copyright (c) 2018 Apple Inc. All rights reserved.
3 *
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5 *
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
14 *
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
17 *
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
25 *
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27 */
28
29#include <stdatomic.h>
30#include <kern/debug.h>
c3c9b80d 31#include <kern/assert.h>
d9a64523
A
32#include <libkern/libkern.h>
33#include "ubsan.h"
34
35static const bool ubsan_print = false;
36static const uint32_t line_acquired = 0x80000000UL;
cb323159 37static const char *get_type_check_kind(uint8_t kind);
f427ee49 38
c3c9b80d
A
39static void
40ubsan_buf_log(struct ubsan_buf *ub, const char *fmt, ...)
d9a64523 41{
c3c9b80d
A
42 va_list ap;
43
44 va_start(ap, fmt);
45 int n = vscnprintf(ub->ub_buf + ub->ub_logged, ub->ub_buf_size - ub->ub_logged, fmt, ap);
46 va_end(ap);
47
48 ub->ub_logged += n;
49 assert(ub->ub_logged <= ub->ub_buf_size);
50}
51
52static void
53ubsan_buf_log_loc(struct ubsan_buf *ub, const char *desc, struct san_src_loc *loc)
54{
55 ubsan_buf_log(ub, "%s:{ file:\"%s\", line:%d, column:%d }",
56 desc,
57 loc->filename,
58 loc->line & ~line_acquired,
59 loc->col);
d9a64523
A
60}
61
62/*
63 * return true for the first visit to this loc, false every subsequent time
64 */
65static bool
66ubsan_loc_acquire(struct san_src_loc *loc)
67{
68 uint32_t line = loc->line;
69 if (line & line_acquired) {
70 return false;
71 }
72 uint32_t acq = line | line_acquired;
73 return atomic_compare_exchange_strong((_Atomic uint32_t *)&loc->line, &line, acq);
74}
75
76static const char *const
77overflow_str[] = {
78 NULL,
79 "add",
80 "sub",
81 "mul",
82 "divrem",
83 "negate",
84 NULL
85};
86
c3c9b80d
A
87static void
88format_overflow(struct ubsan_violation *v, struct ubsan_buf *ub)
d9a64523
A
89{
90 struct san_type_desc *ty = v->overflow->ty;
c3c9b80d
A
91 ubsan_buf_log(ub,
92 "problem:\"%s overflow\", op:\"%s\", ty:\"%s\", width:%d, lhs:0x%llx, rhs:0x%llx",
93 ty->issigned ? "signed" : "unsigned",
94 overflow_str[v->ubsan_type],
95 ty->name,
96 1 << ty->width,
97 v->lhs,
98 v->rhs
99 );
d9a64523
A
100}
101
c3c9b80d
A
102static void
103format_shift(struct ubsan_violation *v, struct ubsan_buf *ub)
d9a64523 104{
d9a64523
A
105 struct san_type_desc *l = v->shift->lhs_t;
106 struct san_type_desc *r = v->shift->rhs_t;
107
c3c9b80d
A
108 ubsan_buf_log(ub, "problem:\"bad shift\", ");
109 ubsan_buf_log(ub, "lhs:0x%llx, lty:\"%s\", lsigned:%d, lwidth:%d, ", v->lhs, l->name, l->issigned, 1 << l->width);
110 ubsan_buf_log(ub, "rhs:0x%llx, rty:\"%s\", rsigned:%d, rwidth:%d", v->rhs, r->name, r->issigned, 1 << r->width);
d9a64523
A
111}
112
cb323159
A
113static const char * const
114type_check_kinds[] = {
115 "load of", "store to", "reference binding to", "member access within",
116 "member call on", "constructor call on", "downcast of", "downcast of",
117 "upcast of", "cast to virtual base of", "_Nonnull binding to"
d9a64523
A
118};
119
cb323159
A
120static const char *
121get_type_check_kind(uint8_t kind)
122{
123 return (kind < (sizeof(type_check_kinds) / sizeof(type_check_kinds[0])))
124 ? type_check_kinds[kind]
125 : "some";
126}
127
c3c9b80d
A
128static void
129format_type_mismatch(struct ubsan_violation *v, struct ubsan_buf *ub)
d9a64523 130{
cb323159
A
131 size_t alignment = 1 << v->align->align;
132 void *ptr = (void*)v->lhs;
c3c9b80d
A
133 const char *kind = get_type_check_kind(v->align->kind);
134
cb323159
A
135 if (NULL == ptr) {
136 //null pointer use
c3c9b80d 137 ubsan_buf_log(ub, "problem:\"%s NULL pointer\", ty:\"%s\"", kind, v->align->ty->name);
cb323159
A
138 } else if (alignment && ((uintptr_t)ptr & (alignment - 1))) {
139 //misaligned pointer use
c3c9b80d
A
140 ubsan_buf_log(ub, "problem:\"%s mis-aligned\", address:%p, ty:\"%s\", ",
141 kind, (void*)v->lhs, v->align->ty->name);
142 ubsan_buf_log(ub, "required_alignment:%d", 1 << v->align->align);
cb323159
A
143 } else {
144 //insufficient object size
c3c9b80d 145 ubsan_buf_log(ub, "problem:\"%s insufficient object size\", ty:\"%s\", address:%p",
f427ee49 146 kind, v->align->ty->name, ptr);
cb323159 147 }
d9a64523
A
148}
149
c3c9b80d
A
150static void
151format_oob(struct ubsan_violation *v, struct ubsan_buf *ub)
d9a64523 152{
d9a64523
A
153 struct san_type_desc *aty = v->oob->array_ty;
154 struct san_type_desc *ity = v->oob->index_ty;
155 uintptr_t idx = v->lhs;
156
c3c9b80d
A
157 ubsan_buf_log(ub, "problem:\"OOB array access\", ");
158 ubsan_buf_log(ub, "idx:%ld, ", idx);
159 ubsan_buf_log(ub, "aty:\"%s\", asigned:%d, awidth:%d, ", aty->name, aty->issigned, 1 << aty->width);
160 ubsan_buf_log(ub, "ity:\"%s\", isigned:%d, iwidth:%d", ity->name, ity->issigned, 1 << ity->width);
161}
d9a64523 162
c3c9b80d
A
163static void
164format_nullability_arg(struct ubsan_violation *v, struct ubsan_buf *ub)
165{
166 struct ubsan_nullability_arg_desc *data = v->nonnull_arg;
167
168 const int arg_index = data->arg_index;
169 const char *attr_type = v->lhs ? "nonnull attribute" : "_Nonnull annotation";
170
171 ubsan_buf_log(ub, "problem:\"null in argument %d declared with %s\", ", arg_index, attr_type);
172 ubsan_buf_log_loc(ub, "declared", &data->attr_loc);
d9a64523
A
173}
174
c3c9b80d
A
175static void
176format_nonnull_return(struct ubsan_violation *v, struct ubsan_buf *ub)
f427ee49 177{
c3c9b80d
A
178 struct san_src_loc *declaration = (struct san_src_loc *)v->rhs;
179 const char *return_type = v->lhs ? "returns_nonnull attribute" : "_Nonnull return type annotation";
180
181 ubsan_buf_log(ub, "problem:\"null returned from function declared with %s\", ", return_type);
182 ubsan_buf_log_loc(ub, "declared", declaration);
f427ee49
A
183}
184
c3c9b80d
A
185static void
186format_load_invalid_value(struct ubsan_violation *v, struct ubsan_buf *ub)
d9a64523 187{
c3c9b80d
A
188 ubsan_buf_log(ub, "problem:\"invalid value load\", type:\"%s\", value:0x%llx",
189 v->invalid->type->name, v->lhs);
190}
191
192static void
193format_missing_return(struct ubsan_violation *v __unused, struct ubsan_buf *ub)
194{
195 ubsan_buf_log(ub, "problem:\"no value returned from value-returning function\"");
196}
197
198static void
199format_float_cast_overflow(struct ubsan_violation *v, struct ubsan_buf *ub)
200{
201 struct ubsan_float_desc *data = v->flt;
202 /*
203 * Cannot print out offending value (e.g. using %A, %f and so on) as kernel logging
204 * does not support float types (yet).
205 */
206 ubsan_buf_log(ub, "problem:\"%s type value outside the range of %s\"",
207 data->type_from->name, data->type_to->name);
208}
209
210static const char *
211get_implicit_conv_type(unsigned char kind)
212{
213 static const char * const conv_types[] = {
214 "integer truncation",
215 "unsigned integer truncation",
216 "signed integer truncation",
217 "integer sign change",
218 "signed integer truncation or sign change"
219 };
220 static const size_t conv_types_cnt = sizeof(conv_types) / sizeof(conv_types[0]);
221
222 return kind < conv_types_cnt ? conv_types[kind] : "unknown implicit integer conversion";
223}
224
225static void
226format_implicit_conversion(struct ubsan_violation *v, struct ubsan_buf *ub)
227{
228 struct ubsan_implicit_conv_desc *data = v->implicit;
229 struct san_type_desc *from = data->type_from;
230 struct san_type_desc *to = data->type_to;
231
232 ubsan_buf_log(ub, "problem:\"%s\", ", get_implicit_conv_type(data->kind));
233 ubsan_buf_log(ub, "src value:%#llx type:\"%s\", signed:%d, width:%d, ",
234 v->lhs, from->name, from->issigned, 1 << from->width);
235 ubsan_buf_log(ub, "dst value:%#llx type:\"%s\", signed:%d, width:%d",
236 v->rhs, to->name, to->issigned, 1 << to->width);
237}
238
239static void
240format_function_type_mismatch(struct ubsan_violation *v, struct ubsan_buf *ub)
241{
242 struct ubsan_func_type_mismatch_desc *data = v->func_mismatch;
243 ubsan_buf_log(ub, "problem:\"indirect function call through %p of a wrong type %s\"",
244 (void *)v->lhs, data->type->name);
245}
246
247static void
248format_vla_bound_not_positive(struct ubsan_violation *v, struct ubsan_buf *ub)
249{
250 struct ubsan_vla_bound_desc *data = v->vla_bound;
251 ubsan_buf_log(ub, "problem:\"VLA %s bound %#llx not positive\"", data->type->name, v->lhs);
252}
253
254static void
255format_invalid_builtin(struct ubsan_violation *v, struct ubsan_buf *ub)
256{
257 ubsan_buf_log(ub, "problem:\"passing invalid zero argument to %s\"",
258 v->invalid_builtin->kind == 0 ? "ctz()" : "clz()");
259}
260
261void
262ubsan_format(struct ubsan_violation *v, struct ubsan_buf *ub)
263{
264 ubsan_buf_log(ub, "{ ");
d9a64523
A
265
266 switch (v->ubsan_type) {
267 case UBSAN_OVERFLOW_add ... UBSAN_OVERFLOW_negate:
c3c9b80d 268 format_overflow(v, ub);
d9a64523
A
269 break;
270 case UBSAN_UNREACHABLE:
c3c9b80d 271 ubsan_buf_log(ub, "problem:\"unreachable\", ");
d9a64523
A
272 break;
273 case UBSAN_SHIFT:
c3c9b80d 274 format_shift(v, ub);
d9a64523 275 break;
cb323159 276 case UBSAN_TYPE_MISMATCH:
c3c9b80d 277 format_type_mismatch(v, ub);
d9a64523
A
278 break;
279 case UBSAN_POINTER_OVERFLOW:
c3c9b80d 280 ubsan_buf_log(ub, "problem:\"pointer overflow\", before:0x%llx, after:0x%llx", v->lhs, v->rhs);
d9a64523
A
281 break;
282 case UBSAN_OOB:
c3c9b80d 283 format_oob(v, ub);
d9a64523 284 break;
c3c9b80d
A
285 case UBSAN_NULLABILITY_ARG:
286 format_nullability_arg(v, ub);
287 break;
288 case UBSAN_NULLABILITY_RETURN:
289 format_nonnull_return(v, ub);
290 break;
291 case UBSAN_MISSING_RETURN:
292 format_missing_return(v, ub);
293 break;
294 case UBSAN_FLOAT_CAST_OVERFLOW:
295 format_float_cast_overflow(v, ub);
296 break;
297 case UBSAN_IMPLICIT_CONVERSION:
298 format_implicit_conversion(v, ub);
f427ee49 299 break;
c3c9b80d
A
300 case UBSAN_FUNCTION_TYPE_MISMATCH:
301 format_function_type_mismatch(v, ub);
302 break;
303 case UBSAN_VLA_BOUND_NOT_POSITIVE:
304 format_vla_bound_not_positive(v, ub);
305 break;
306 case UBSAN_INVALID_BUILTIN:
307 format_invalid_builtin(v, ub);
308 break;
309 case UBSAN_LOAD_INVALID_VALUE:
310 format_load_invalid_value(v, ub);
cb323159 311 break;
d9a64523
A
312 default:
313 panic("unknown violation");
314 }
315
c3c9b80d
A
316 ubsan_buf_log_loc(ub, ", found", v->loc);
317 ubsan_buf_log(ub, " },\n");
d9a64523
A
318}
319
f427ee49
A
320enum UBFatality { Fatal, FleshWound };
321
d9a64523 322static void
f427ee49 323ubsan_handle(struct ubsan_violation *v, enum UBFatality fatality)
d9a64523 324{
d9a64523
A
325 if (!ubsan_loc_acquire(v->loc)) {
326 /* violation site already reported */
327 return;
328 }
329
330 ubsan_log_append(v);
331
f427ee49 332 if (ubsan_print || (fatality == Fatal)) {
c3c9b80d
A
333 static char buf[256] = { 0 };
334 struct ubsan_buf ubsan_buf = {
335 .ub_logged = 0,
336 .ub_buf_size = sizeof(buf),
337 .ub_buf = buf
338 };
339 ubsan_format(v, &ubsan_buf);
d9a64523
A
340 printf("UBSan: %s", buf);
341 }
d9a64523
A
342}
343
344void
345__ubsan_handle_builtin_unreachable(struct ubsan_unreachable_desc *desc)
346{
347 struct ubsan_violation v = { UBSAN_UNREACHABLE, 0, 0, .unreachable = desc, &desc->loc };
f427ee49 348 ubsan_handle(&v, Fatal);
d9a64523
A
349}
350
351void
352__ubsan_handle_shift_out_of_bounds(struct ubsan_shift_desc *desc, uint64_t lhs, uint64_t rhs)
353{
354 struct ubsan_violation v = { UBSAN_SHIFT, lhs, rhs, .shift = desc, &desc->loc };
f427ee49 355 ubsan_handle(&v, FleshWound);
d9a64523
A
356}
357
358void
359__ubsan_handle_shift_out_of_bounds_abort(struct ubsan_shift_desc *desc, uint64_t lhs, uint64_t rhs)
360{
361 struct ubsan_violation v = { UBSAN_SHIFT, lhs, rhs, .shift = desc, &desc->loc };
f427ee49 362 ubsan_handle(&v, Fatal);
d9a64523
A
363}
364
365#define DEFINE_OVERFLOW(op) \
366 void __ubsan_handle_##op##_overflow(struct ubsan_overflow_desc *desc, uint64_t lhs, uint64_t rhs) { \
0a7de745 367 struct ubsan_violation v = { UBSAN_OVERFLOW_##op, lhs, rhs, .overflow = desc, &desc->loc }; \
f427ee49 368 ubsan_handle(&v, FleshWound); \
d9a64523
A
369 } \
370 void __ubsan_handle_##op##_overflow_abort(struct ubsan_overflow_desc *desc, uint64_t lhs, uint64_t rhs) { \
0a7de745 371 struct ubsan_violation v = { UBSAN_OVERFLOW_##op, lhs, rhs, .overflow = desc, &desc->loc }; \
f427ee49 372 ubsan_handle(&v, Fatal); \
d9a64523
A
373 }
374
375DEFINE_OVERFLOW(add)
376DEFINE_OVERFLOW(sub)
377DEFINE_OVERFLOW(mul)
378DEFINE_OVERFLOW(divrem)
379DEFINE_OVERFLOW(negate)
380
0a7de745
A
381void
382__ubsan_handle_type_mismatch_v1(struct ubsan_align_desc *desc, uint64_t val)
d9a64523 383{
cb323159 384 struct ubsan_violation v = { UBSAN_TYPE_MISMATCH, val, 0, .align = desc, &desc->loc };
f427ee49 385 ubsan_handle(&v, FleshWound);
d9a64523
A
386}
387
0a7de745
A
388void
389__ubsan_handle_type_mismatch_v1_abort(struct ubsan_align_desc *desc, uint64_t val)
d9a64523 390{
cb323159 391 struct ubsan_violation v = { UBSAN_TYPE_MISMATCH, val, 0, .align = desc, &desc->loc };
f427ee49 392 ubsan_handle(&v, Fatal);
d9a64523
A
393}
394
0a7de745
A
395void
396__ubsan_handle_pointer_overflow(struct ubsan_ptroverflow_desc *desc, uint64_t before, uint64_t after)
d9a64523
A
397{
398 struct ubsan_violation v = { UBSAN_POINTER_OVERFLOW, before, after, .ptroverflow = desc, &desc->loc };
f427ee49 399 ubsan_handle(&v, FleshWound);
d9a64523
A
400}
401
0a7de745
A
402void
403__ubsan_handle_pointer_overflow_abort(struct ubsan_ptroverflow_desc *desc, uint64_t before, uint64_t after)
d9a64523
A
404{
405 struct ubsan_violation v = { UBSAN_POINTER_OVERFLOW, before, after, .ptroverflow = desc, &desc->loc };
f427ee49 406 ubsan_handle(&v, Fatal);
d9a64523
A
407}
408
0a7de745
A
409void
410__ubsan_handle_out_of_bounds(struct ubsan_oob_desc *desc, uint64_t idx)
d9a64523
A
411{
412 struct ubsan_violation v = { UBSAN_OOB, idx, 0, .oob = desc, &desc->loc };
f427ee49 413 ubsan_handle(&v, FleshWound);
d9a64523
A
414}
415
0a7de745
A
416void
417__ubsan_handle_out_of_bounds_abort(struct ubsan_oob_desc *desc, uint64_t idx)
d9a64523
A
418{
419 struct ubsan_violation v = { UBSAN_OOB, idx, 0, .oob = desc, &desc->loc };
f427ee49
A
420 ubsan_handle(&v, Fatal);
421}
422
c3c9b80d
A
423void
424__ubsan_handle_nullability_arg(struct ubsan_nullability_arg_desc *desc)
425{
426 struct ubsan_violation v = { UBSAN_NULLABILITY_ARG, 0, 0, .nonnull_arg = desc, &desc->loc };
427 ubsan_handle(&v, FleshWound);
428}
429
430void
431__ubsan_handle_nullability_arg_abort(struct ubsan_nullability_arg_desc *desc)
432{
433 struct ubsan_violation v = { UBSAN_NULLABILITY_ARG, 0, 0, .nonnull_arg = desc, &desc->loc };
434 ubsan_handle(&v, Fatal);
435}
436
437void
438__ubsan_handle_nonnull_arg(struct ubsan_nullability_arg_desc *desc)
439{
440 struct ubsan_violation v = { UBSAN_NULLABILITY_ARG, 1, 0, .nonnull_arg = desc, &desc->loc };
441 ubsan_handle(&v, FleshWound);
442}
443
444void
445__ubsan_handle_nonnull_arg_abort(struct ubsan_nullability_arg_desc *desc)
446{
447 struct ubsan_violation v = { UBSAN_NULLABILITY_ARG, 1, 0, .nonnull_arg = desc, &desc->loc };
448 ubsan_handle(&v, Fatal);
449}
450
451void
452__ubsan_handle_nullability_return_v1(struct ubsan_nullability_ret_desc *desc, uint64_t declaration)
453{
454 struct ubsan_violation v = { UBSAN_NULLABILITY_RETURN, 0, (uint64_t)&desc->loc, .nonnull_ret = desc, (struct san_src_loc *)declaration };
455 ubsan_handle(&v, FleshWound);
456}
457
458void
459__ubsan_handle_nullability_return_v1_abort(struct ubsan_nullability_ret_desc *desc, uint64_t declaration)
460{
461 struct ubsan_violation v = { UBSAN_NULLABILITY_RETURN, 0, (uint64_t)&desc->loc, .nonnull_ret = desc, (struct san_src_loc *)declaration };
462 ubsan_handle(&v, Fatal);
463}
464
465void
466__ubsan_handle_nonnull_return_v1(struct ubsan_nullability_ret_desc *desc, uint64_t declaration)
467{
468 struct ubsan_violation v = { UBSAN_NULLABILITY_RETURN, 1, (uint64_t)&desc->loc, .nonnull_ret = desc, (struct san_src_loc *)declaration };
469 ubsan_handle(&v, FleshWound);
470}
471
472void
473__ubsan_handle_nonnull_return_v1_abort(struct ubsan_nullability_ret_desc *desc, uint64_t declaration)
474{
475 struct ubsan_violation v = { UBSAN_NULLABILITY_RETURN, 1, (uint64_t)&desc->loc, .nonnull_ret = desc, (struct san_src_loc *)declaration };
476 ubsan_handle(&v, Fatal);
477}
478
479void
480__ubsan_handle_missing_return(struct ubsan_missing_ret_desc *desc)
481{
482 struct ubsan_violation v = { UBSAN_MISSING_RETURN, 0, 0, .missing_ret = desc, &desc->loc };
483 ubsan_handle(&v, Fatal);
484}
485
486void
487__ubsan_handle_missing_return_abort(struct ubsan_missing_ret_desc *desc)
488{
489 struct ubsan_violation v = { UBSAN_MISSING_RETURN, 0, 0, .missing_ret = desc, &desc->loc };
490 ubsan_handle(&v, Fatal);
491}
492
493void
494__ubsan_handle_float_cast_overflow(struct ubsan_float_desc *desc, uint64_t value)
495{
496 struct ubsan_violation v = { UBSAN_FLOAT_CAST_OVERFLOW, value, 0, .flt = desc, &desc->loc };
497 ubsan_handle(&v, Fatal);
498}
499
500void
501__ubsan_handle_float_cast_overflow_abort(struct ubsan_float_desc *desc, uint64_t value)
502{
503 struct ubsan_violation v = { UBSAN_FLOAT_CAST_OVERFLOW, value, 0, .flt = desc, &desc->loc };
504 ubsan_handle(&v, Fatal);
505}
506
507void
508__ubsan_handle_implicit_conversion(struct ubsan_implicit_conv_desc *desc, uint64_t from, uint64_t to)
509{
510 struct ubsan_violation v = { UBSAN_IMPLICIT_CONVERSION, from, to, .implicit = desc, &desc->loc };
511 ubsan_handle(&v, Fatal);
512}
513
514void
515__ubsan_handle_implicit_conversion_abort(struct ubsan_implicit_conv_desc *desc, uint64_t from, uint64_t to)
516{
517 struct ubsan_violation v = { UBSAN_IMPLICIT_CONVERSION, from, to, .implicit = desc, &desc->loc };
518 ubsan_handle(&v, Fatal);
519}
520
521void
522__ubsan_handle_function_type_mismatch(struct ubsan_func_type_mismatch_desc *desc, uint64_t func)
523{
524 struct ubsan_violation v = { UBSAN_FUNCTION_TYPE_MISMATCH, func, 0, .func_mismatch = desc, &desc->loc };
525 ubsan_handle(&v, Fatal);
526}
527
528void
529__ubsan_handle_function_type_mismatch_abort(struct ubsan_func_type_mismatch_desc *desc, uint64_t func)
530{
531 struct ubsan_violation v = { UBSAN_FUNCTION_TYPE_MISMATCH, func, 0, .func_mismatch = desc, &desc->loc };
532 ubsan_handle(&v, Fatal);
533}
534
535void
536__ubsan_handle_vla_bound_not_positive(struct ubsan_vla_bound_desc *desc, uint64_t length)
537{
538 struct ubsan_violation v = { UBSAN_VLA_BOUND_NOT_POSITIVE, length, 0, .vla_bound = desc, &desc->loc };
539 ubsan_handle(&v, Fatal);
540}
541
542void
543__ubsan_handle_vla_bound_not_positive_abort(struct ubsan_vla_bound_desc *desc, uint64_t length)
544{
545 struct ubsan_violation v = { UBSAN_VLA_BOUND_NOT_POSITIVE, length, 0, .vla_bound = desc, &desc->loc };
546 ubsan_handle(&v, Fatal);
547}
548
549void
550__ubsan_handle_invalid_builtin(struct ubsan_invalid_builtin *desc)
551{
552 struct ubsan_violation v = { UBSAN_INVALID_BUILTIN, 0, 0, .invalid_builtin = desc, &desc->loc };
553 ubsan_handle(&v, Fatal);
554}
555
556void
557__ubsan_handle_invalid_builtin_abort(struct ubsan_invalid_builtin *desc)
558{
559 struct ubsan_violation v = { UBSAN_INVALID_BUILTIN, 0, 0, .invalid_builtin = desc, &desc->loc };
560 ubsan_handle(&v, Fatal);
561}
562
f427ee49
A
563void
564__ubsan_handle_load_invalid_value(struct ubsan_load_invalid_desc *desc, uint64_t invalid_value)
565{
566 struct ubsan_violation v = { UBSAN_LOAD_INVALID_VALUE, invalid_value, 0, .invalid = desc, &desc->loc };
567 ubsan_handle(&v, Fatal);
568}
569
570void
571__ubsan_handle_load_invalid_value_abort(struct ubsan_load_invalid_desc *desc, uint64_t invalid_value)
572{
573 struct ubsan_violation v = { UBSAN_LOAD_INVALID_VALUE, invalid_value, 0, .invalid = desc, &desc->loc };
574 ubsan_handle(&v, Fatal);
d9a64523 575}