-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcrash.c
97 lines (79 loc) · 2.37 KB
/
crash.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
* Copyright(c) 2021 John Sanpe <[email protected]>
*/
#define MODULE_NAME "crash"
#define pr_fmt(fmt) MODULE_NAME ": " fmt
#include <crash.h>
#include <panic.h>
#include <sort.h>
#include <bsearch.h>
#include <sections.h>
#include <printk.h>
#include <export.h>
int __cold crash_printk(const char *fmt, ...)
{
va_list args;
int retval;
va_start(args, fmt);
retval = vprintk(fmt, args);
va_end(args);
return retval;
}
EXPORT_SYMBOL(crash_printk);
void __cold __noreturn crash_panic(const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
vpanic(fmt, args);
}
EXPORT_SYMBOL(crash_panic);
static long crash_table_search(const void *key, void *pdata)
{
const struct crash_entry *crash = key;
uintptr_t addr = (uintptr_t)pdata;
if (crash->addr == addr) return 0;
return crash->addr > addr ? 1 : -1;
}
static long crash_table_sort(const void *nodea, const void *nodeb, void *pdata)
{
const struct crash_entry *crasha = nodea;
const struct crash_entry *crashb = nodeb;
if (crasha->addr == crashb->addr) return 0;
return crasha->addr > crashb->addr ? 1 : -1;
}
struct crash_entry *crash_find(uintptr_t addr)
{
struct crash_entry *walk;
walk = bsearch(_ld_data_bug_table_start, ((uintptr_t)_ld_data_bug_table_end -
(uintptr_t)_ld_data_bug_table_start) / sizeof(struct crash_entry),
sizeof(struct crash_entry), crash_table_search, (void *)addr);
return walk;
}
void __init crash_init(void)
{
qsort(_ld_data_bug_table_start, ((uintptr_t)_ld_data_bug_table_end -
(uintptr_t)_ld_data_bug_table_start) / sizeof(struct crash_entry),
sizeof(struct crash_entry), crash_table_sort, NULL);
}
enum crash_type crash_report(uintptr_t addr, struct regs *regs)
{
struct crash_entry *crash;
enum crash_type type;
bool warn;
crash = crash_find(addr);
if (unlikely(!crash))
return CRASH_TYPE_NONE;
warn = crash_test_warning(crash);
type = warn ? CRASH_TYPE_WARN : CRASH_TYPE_BUG;
if (crash_test_once(crash)) {
if (crash_test_set_done(crash))
return type;
}
if (!crash_test_ncut_here(crash))
printk(CRASH_CUT_HERE);
if (crash->info)
pr_crit("%s\n", crash->info);
pr_crit("Kernel BUG at %s:%u!\n", crash->file, crash->line);
return type;
}