Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

crash: move crashkernel parsing and vmcore related code under CONFIG_CRASH_CORE

Patch series "kexec/fadump: remove dependency with CONFIG_KEXEC and
reuse crashkernel parameter for fadump", v4.

Traditionally, kdump is used to save vmcore in case of a crash. Some
architectures like powerpc can save vmcore using architecture specific
support instead of kexec/kdump mechanism. Such architecture specific
support also needs to reserve memory, to be used by dump capture kernel.
crashkernel parameter can be a reused, for memory reservation, by such
architecture specific infrastructure.

This patchset removes dependency with CONFIG_KEXEC for crashkernel
parameter and vmcoreinfo related code as it can be reused without kexec
support. Also, crashkernel parameter is reused instead of
fadump_reserve_mem to reserve memory for fadump.

The first patch moves crashkernel parameter parsing and vmcoreinfo
related code under CONFIG_CRASH_CORE instead of CONFIG_KEXEC_CORE. The
second patch reuses the definitions of append_elf_note() & final_note()
functions under CONFIG_CRASH_CORE in IA64 arch code. The third patch
removes dependency on CONFIG_KEXEC for firmware-assisted dump (fadump)
in powerpc. The next patch reuses crashkernel parameter for reserving
memory for fadump, instead of the fadump_reserve_mem parameter. This
has the advantage of using all syntaxes crashkernel parameter supports,
for fadump as well. The last patch updates fadump kernel documentation
about use of crashkernel parameter.

This patch (of 5):

Traditionally, kdump is used to save vmcore in case of a crash. Some
architectures like powerpc can save vmcore using architecture specific
support instead of kexec/kdump mechanism. Such architecture specific
support also needs to reserve memory, to be used by dump capture kernel.
crashkernel parameter can be a reused, for memory reservation, by such
architecture specific infrastructure.

But currently, code related to vmcoreinfo and parsing of crashkernel
parameter is built under CONFIG_KEXEC_CORE. This patch introduces
CONFIG_CRASH_CORE and moves the above mentioned code under this config,
allowing code reuse without dependency on CONFIG_KEXEC. There is no
functional change with this patch.

Link: http://lkml.kernel.org/r/149035338104.6881.4550894432615189948.stgit@hbathini.in.ibm.com
Signed-off-by: Hari Bathini <hbathini@linux.vnet.ibm.com>
Acked-by: Dave Young <dyoung@redhat.com>
Cc: Fenghua Yu <fenghua.yu@intel.com>
Cc: Tony Luck <tony.luck@intel.com>
Cc: Eric Biederman <ebiederm@xmission.com>
Cc: Mahesh Salgaonkar <mahesh@linux.vnet.ibm.com>
Cc: Vivek Goyal <vgoyal@redhat.com>
Cc: Michael Ellerman <mpe@ellerman.id.au>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>

authored by

Hari Bathini and committed by
Linus Torvalds
692f66f2 c311c797

+531 -462
+4
arch/Kconfig
··· 2 2 # General architecture dependent options 3 3 # 4 4 5 + config CRASH_CORE 6 + bool 7 + 5 8 config KEXEC_CORE 9 + select CRASH_CORE 6 10 bool 7 11 8 12 config HAVE_IMA_KEXEC
+65
include/linux/crash_core.h
··· 1 + #ifndef LINUX_CRASH_CORE_H 2 + #define LINUX_CRASH_CORE_H 3 + 4 + #include <linux/linkage.h> 5 + #include <linux/elfcore.h> 6 + #include <linux/elf.h> 7 + 8 + #define CRASH_CORE_NOTE_NAME "CORE" 9 + #define CRASH_CORE_NOTE_HEAD_BYTES ALIGN(sizeof(struct elf_note), 4) 10 + #define CRASH_CORE_NOTE_NAME_BYTES ALIGN(sizeof(CRASH_CORE_NOTE_NAME), 4) 11 + #define CRASH_CORE_NOTE_DESC_BYTES ALIGN(sizeof(struct elf_prstatus), 4) 12 + 13 + #define CRASH_CORE_NOTE_BYTES ((CRASH_CORE_NOTE_HEAD_BYTES * 2) + \ 14 + CRASH_CORE_NOTE_NAME_BYTES + \ 15 + CRASH_CORE_NOTE_DESC_BYTES) 16 + 17 + #define VMCOREINFO_BYTES (4096) 18 + #define VMCOREINFO_NOTE_NAME "VMCOREINFO" 19 + #define VMCOREINFO_NOTE_NAME_BYTES ALIGN(sizeof(VMCOREINFO_NOTE_NAME), 4) 20 + #define VMCOREINFO_NOTE_SIZE ((CRASH_CORE_NOTE_HEAD_BYTES * 2) + \ 21 + VMCOREINFO_NOTE_NAME_BYTES + \ 22 + VMCOREINFO_BYTES) 23 + 24 + typedef u32 note_buf_t[CRASH_CORE_NOTE_BYTES/4]; 25 + 26 + void crash_save_vmcoreinfo(void); 27 + void arch_crash_save_vmcoreinfo(void); 28 + __printf(1, 2) 29 + void vmcoreinfo_append_str(const char *fmt, ...); 30 + phys_addr_t paddr_vmcoreinfo_note(void); 31 + 32 + #define VMCOREINFO_OSRELEASE(value) \ 33 + vmcoreinfo_append_str("OSRELEASE=%s\n", value) 34 + #define VMCOREINFO_PAGESIZE(value) \ 35 + vmcoreinfo_append_str("PAGESIZE=%ld\n", value) 36 + #define VMCOREINFO_SYMBOL(name) \ 37 + vmcoreinfo_append_str("SYMBOL(%s)=%lx\n", #name, (unsigned long)&name) 38 + #define VMCOREINFO_SIZE(name) \ 39 + vmcoreinfo_append_str("SIZE(%s)=%lu\n", #name, \ 40 + (unsigned long)sizeof(name)) 41 + #define VMCOREINFO_STRUCT_SIZE(name) \ 42 + vmcoreinfo_append_str("SIZE(%s)=%lu\n", #name, \ 43 + (unsigned long)sizeof(struct name)) 44 + #define VMCOREINFO_OFFSET(name, field) \ 45 + vmcoreinfo_append_str("OFFSET(%s.%s)=%lu\n", #name, #field, \ 46 + (unsigned long)offsetof(struct name, field)) 47 + #define VMCOREINFO_LENGTH(name, value) \ 48 + vmcoreinfo_append_str("LENGTH(%s)=%lu\n", #name, (unsigned long)value) 49 + #define VMCOREINFO_NUMBER(name) \ 50 + vmcoreinfo_append_str("NUMBER(%s)=%ld\n", #name, (long)name) 51 + #define VMCOREINFO_CONFIG(name) \ 52 + vmcoreinfo_append_str("CONFIG_%s=y\n", #name) 53 + 54 + extern u32 vmcoreinfo_note[VMCOREINFO_NOTE_SIZE/4]; 55 + extern size_t vmcoreinfo_size; 56 + extern size_t vmcoreinfo_max_size; 57 + 58 + int __init parse_crashkernel(char *cmdline, unsigned long long system_ram, 59 + unsigned long long *crash_size, unsigned long long *crash_base); 60 + int parse_crashkernel_high(char *cmdline, unsigned long long system_ram, 61 + unsigned long long *crash_size, unsigned long long *crash_base); 62 + int parse_crashkernel_low(char *cmdline, unsigned long long system_ram, 63 + unsigned long long *crash_size, unsigned long long *crash_base); 64 + 65 + #endif /* LINUX_CRASH_CORE_H */
+4 -53
include/linux/kexec.h
··· 14 14 15 15 #if !defined(__ASSEMBLY__) 16 16 17 + #include <linux/crash_core.h> 17 18 #include <asm/io.h> 18 19 19 20 #include <uapi/linux/kexec.h> 20 21 21 22 #ifdef CONFIG_KEXEC_CORE 22 23 #include <linux/list.h> 23 - #include <linux/linkage.h> 24 24 #include <linux/compat.h> 25 25 #include <linux/ioport.h> 26 - #include <linux/elfcore.h> 27 - #include <linux/elf.h> 28 26 #include <linux/module.h> 29 27 #include <asm/kexec.h> 30 28 ··· 60 62 #define KEXEC_CRASH_MEM_ALIGN PAGE_SIZE 61 63 #endif 62 64 63 - #define KEXEC_NOTE_HEAD_BYTES ALIGN(sizeof(struct elf_note), 4) 64 - #define KEXEC_CORE_NOTE_NAME "CORE" 65 - #define KEXEC_CORE_NOTE_NAME_BYTES ALIGN(sizeof(KEXEC_CORE_NOTE_NAME), 4) 66 - #define KEXEC_CORE_NOTE_DESC_BYTES ALIGN(sizeof(struct elf_prstatus), 4) 65 + #define KEXEC_CORE_NOTE_NAME CRASH_CORE_NOTE_NAME 66 + 67 67 /* 68 68 * The per-cpu notes area is a list of notes terminated by a "NULL" 69 69 * note header. For kdump, the code in vmcore.c runs in the context 70 70 * of the second kernel to combine them into one note. 71 71 */ 72 72 #ifndef KEXEC_NOTE_BYTES 73 - #define KEXEC_NOTE_BYTES ( (KEXEC_NOTE_HEAD_BYTES * 2) + \ 74 - KEXEC_CORE_NOTE_NAME_BYTES + \ 75 - KEXEC_CORE_NOTE_DESC_BYTES ) 73 + #define KEXEC_NOTE_BYTES CRASH_CORE_NOTE_BYTES 76 74 #endif 77 75 78 76 /* ··· 250 256 int kexec_should_crash(struct task_struct *); 251 257 int kexec_crash_loaded(void); 252 258 void crash_save_cpu(struct pt_regs *regs, int cpu); 253 - void crash_save_vmcoreinfo(void); 254 - void arch_crash_save_vmcoreinfo(void); 255 - __printf(1, 2) 256 - void vmcoreinfo_append_str(const char *fmt, ...); 257 - phys_addr_t paddr_vmcoreinfo_note(void); 258 - 259 - #define VMCOREINFO_OSRELEASE(value) \ 260 - vmcoreinfo_append_str("OSRELEASE=%s\n", value) 261 - #define VMCOREINFO_PAGESIZE(value) \ 262 - vmcoreinfo_append_str("PAGESIZE=%ld\n", value) 263 - #define VMCOREINFO_SYMBOL(name) \ 264 - vmcoreinfo_append_str("SYMBOL(%s)=%lx\n", #name, (unsigned long)&name) 265 - #define VMCOREINFO_SIZE(name) \ 266 - vmcoreinfo_append_str("SIZE(%s)=%lu\n", #name, \ 267 - (unsigned long)sizeof(name)) 268 - #define VMCOREINFO_STRUCT_SIZE(name) \ 269 - vmcoreinfo_append_str("SIZE(%s)=%lu\n", #name, \ 270 - (unsigned long)sizeof(struct name)) 271 - #define VMCOREINFO_OFFSET(name, field) \ 272 - vmcoreinfo_append_str("OFFSET(%s.%s)=%lu\n", #name, #field, \ 273 - (unsigned long)offsetof(struct name, field)) 274 - #define VMCOREINFO_LENGTH(name, value) \ 275 - vmcoreinfo_append_str("LENGTH(%s)=%lu\n", #name, (unsigned long)value) 276 - #define VMCOREINFO_NUMBER(name) \ 277 - vmcoreinfo_append_str("NUMBER(%s)=%ld\n", #name, (long)name) 278 - #define VMCOREINFO_CONFIG(name) \ 279 - vmcoreinfo_append_str("CONFIG_%s=y\n", #name) 280 259 281 260 extern struct kimage *kexec_image; 282 261 extern struct kimage *kexec_crash_image; ··· 270 303 #define KEXEC_FILE_FLAGS (KEXEC_FILE_UNLOAD | KEXEC_FILE_ON_CRASH | \ 271 304 KEXEC_FILE_NO_INITRAMFS) 272 305 273 - #define VMCOREINFO_BYTES (4096) 274 - #define VMCOREINFO_NOTE_NAME "VMCOREINFO" 275 - #define VMCOREINFO_NOTE_NAME_BYTES ALIGN(sizeof(VMCOREINFO_NOTE_NAME), 4) 276 - #define VMCOREINFO_NOTE_SIZE (KEXEC_NOTE_HEAD_BYTES*2 + VMCOREINFO_BYTES \ 277 - + VMCOREINFO_NOTE_NAME_BYTES) 278 - 279 306 /* Location of a reserved region to hold the crash kernel. 280 307 */ 281 308 extern struct resource crashk_res; 282 309 extern struct resource crashk_low_res; 283 - typedef u32 note_buf_t[KEXEC_NOTE_BYTES/4]; 284 310 extern note_buf_t __percpu *crash_notes; 285 - extern u32 vmcoreinfo_note[VMCOREINFO_NOTE_SIZE/4]; 286 - extern size_t vmcoreinfo_size; 287 - extern size_t vmcoreinfo_max_size; 288 311 289 312 /* flag to track if kexec reboot is in progress */ 290 313 extern bool kexec_in_progress; 291 314 292 - int __init parse_crashkernel(char *cmdline, unsigned long long system_ram, 293 - unsigned long long *crash_size, unsigned long long *crash_base); 294 - int parse_crashkernel_high(char *cmdline, unsigned long long system_ram, 295 - unsigned long long *crash_size, unsigned long long *crash_base); 296 - int parse_crashkernel_low(char *cmdline, unsigned long long system_ram, 297 - unsigned long long *crash_size, unsigned long long *crash_base); 298 315 int crash_shrink_memory(unsigned long new_size); 299 316 size_t crash_get_memory_size(void); 300 317 void crash_free_reserved_phys_range(unsigned long begin, unsigned long end);
+2 -2
include/linux/printk.h
··· 198 198 199 199 char *log_buf_addr_get(void); 200 200 u32 log_buf_len_get(void); 201 - void log_buf_kexec_setup(void); 201 + void log_buf_vmcoreinfo_setup(void); 202 202 void __init setup_log_buf(int early); 203 203 __printf(1, 2) void dump_stack_set_arch_desc(const char *fmt, ...); 204 204 void dump_stack_print_info(const char *log_lvl); ··· 246 246 return 0; 247 247 } 248 248 249 - static inline void log_buf_kexec_setup(void) 249 + static inline void log_buf_vmcoreinfo_setup(void) 250 250 { 251 251 } 252 252
+1
kernel/Makefile
··· 59 59 obj-$(CONFIG_MODULE_SIG) += module_signing.o 60 60 obj-$(CONFIG_KALLSYMS) += kallsyms.o 61 61 obj-$(CONFIG_BSD_PROCESS_ACCT) += acct.o 62 + obj-$(CONFIG_CRASH_CORE) += crash_core.o 62 63 obj-$(CONFIG_KEXEC_CORE) += kexec_core.o 63 64 obj-$(CONFIG_KEXEC) += kexec.o 64 65 obj-$(CONFIG_KEXEC_FILE) += kexec_file.o
+445
kernel/crash_core.c
··· 1 + /* 2 + * crash.c - kernel crash support code. 3 + * Copyright (C) 2002-2004 Eric Biederman <ebiederm@xmission.com> 4 + * 5 + * This source code is licensed under the GNU General Public License, 6 + * Version 2. See the file COPYING for more details. 7 + */ 8 + 9 + #include <linux/crash_core.h> 10 + #include <linux/utsname.h> 11 + #include <linux/vmalloc.h> 12 + 13 + #include <asm/page.h> 14 + #include <asm/sections.h> 15 + 16 + /* vmcoreinfo stuff */ 17 + static unsigned char vmcoreinfo_data[VMCOREINFO_BYTES]; 18 + u32 vmcoreinfo_note[VMCOREINFO_NOTE_SIZE/4]; 19 + size_t vmcoreinfo_size; 20 + size_t vmcoreinfo_max_size = sizeof(vmcoreinfo_data); 21 + 22 + /* 23 + * parsing the "crashkernel" commandline 24 + * 25 + * this code is intended to be called from architecture specific code 26 + */ 27 + 28 + 29 + /* 30 + * This function parses command lines in the format 31 + * 32 + * crashkernel=ramsize-range:size[,...][@offset] 33 + * 34 + * The function returns 0 on success and -EINVAL on failure. 35 + */ 36 + static int __init parse_crashkernel_mem(char *cmdline, 37 + unsigned long long system_ram, 38 + unsigned long long *crash_size, 39 + unsigned long long *crash_base) 40 + { 41 + char *cur = cmdline, *tmp; 42 + 43 + /* for each entry of the comma-separated list */ 44 + do { 45 + unsigned long long start, end = ULLONG_MAX, size; 46 + 47 + /* get the start of the range */ 48 + start = memparse(cur, &tmp); 49 + if (cur == tmp) { 50 + pr_warn("crashkernel: Memory value expected\n"); 51 + return -EINVAL; 52 + } 53 + cur = tmp; 54 + if (*cur != '-') { 55 + pr_warn("crashkernel: '-' expected\n"); 56 + return -EINVAL; 57 + } 58 + cur++; 59 + 60 + /* if no ':' is here, than we read the end */ 61 + if (*cur != ':') { 62 + end = memparse(cur, &tmp); 63 + if (cur == tmp) { 64 + pr_warn("crashkernel: Memory value expected\n"); 65 + return -EINVAL; 66 + } 67 + cur = tmp; 68 + if (end <= start) { 69 + pr_warn("crashkernel: end <= start\n"); 70 + return -EINVAL; 71 + } 72 + } 73 + 74 + if (*cur != ':') { 75 + pr_warn("crashkernel: ':' expected\n"); 76 + return -EINVAL; 77 + } 78 + cur++; 79 + 80 + size = memparse(cur, &tmp); 81 + if (cur == tmp) { 82 + pr_warn("Memory value expected\n"); 83 + return -EINVAL; 84 + } 85 + cur = tmp; 86 + if (size >= system_ram) { 87 + pr_warn("crashkernel: invalid size\n"); 88 + return -EINVAL; 89 + } 90 + 91 + /* match ? */ 92 + if (system_ram >= start && system_ram < end) { 93 + *crash_size = size; 94 + break; 95 + } 96 + } while (*cur++ == ','); 97 + 98 + if (*crash_size > 0) { 99 + while (*cur && *cur != ' ' && *cur != '@') 100 + cur++; 101 + if (*cur == '@') { 102 + cur++; 103 + *crash_base = memparse(cur, &tmp); 104 + if (cur == tmp) { 105 + pr_warn("Memory value expected after '@'\n"); 106 + return -EINVAL; 107 + } 108 + } 109 + } 110 + 111 + return 0; 112 + } 113 + 114 + /* 115 + * That function parses "simple" (old) crashkernel command lines like 116 + * 117 + * crashkernel=size[@offset] 118 + * 119 + * It returns 0 on success and -EINVAL on failure. 120 + */ 121 + static int __init parse_crashkernel_simple(char *cmdline, 122 + unsigned long long *crash_size, 123 + unsigned long long *crash_base) 124 + { 125 + char *cur = cmdline; 126 + 127 + *crash_size = memparse(cmdline, &cur); 128 + if (cmdline == cur) { 129 + pr_warn("crashkernel: memory value expected\n"); 130 + return -EINVAL; 131 + } 132 + 133 + if (*cur == '@') 134 + *crash_base = memparse(cur+1, &cur); 135 + else if (*cur != ' ' && *cur != '\0') { 136 + pr_warn("crashkernel: unrecognized char: %c\n", *cur); 137 + return -EINVAL; 138 + } 139 + 140 + return 0; 141 + } 142 + 143 + #define SUFFIX_HIGH 0 144 + #define SUFFIX_LOW 1 145 + #define SUFFIX_NULL 2 146 + static __initdata char *suffix_tbl[] = { 147 + [SUFFIX_HIGH] = ",high", 148 + [SUFFIX_LOW] = ",low", 149 + [SUFFIX_NULL] = NULL, 150 + }; 151 + 152 + /* 153 + * That function parses "suffix" crashkernel command lines like 154 + * 155 + * crashkernel=size,[high|low] 156 + * 157 + * It returns 0 on success and -EINVAL on failure. 158 + */ 159 + static int __init parse_crashkernel_suffix(char *cmdline, 160 + unsigned long long *crash_size, 161 + const char *suffix) 162 + { 163 + char *cur = cmdline; 164 + 165 + *crash_size = memparse(cmdline, &cur); 166 + if (cmdline == cur) { 167 + pr_warn("crashkernel: memory value expected\n"); 168 + return -EINVAL; 169 + } 170 + 171 + /* check with suffix */ 172 + if (strncmp(cur, suffix, strlen(suffix))) { 173 + pr_warn("crashkernel: unrecognized char: %c\n", *cur); 174 + return -EINVAL; 175 + } 176 + cur += strlen(suffix); 177 + if (*cur != ' ' && *cur != '\0') { 178 + pr_warn("crashkernel: unrecognized char: %c\n", *cur); 179 + return -EINVAL; 180 + } 181 + 182 + return 0; 183 + } 184 + 185 + static __init char *get_last_crashkernel(char *cmdline, 186 + const char *name, 187 + const char *suffix) 188 + { 189 + char *p = cmdline, *ck_cmdline = NULL; 190 + 191 + /* find crashkernel and use the last one if there are more */ 192 + p = strstr(p, name); 193 + while (p) { 194 + char *end_p = strchr(p, ' '); 195 + char *q; 196 + 197 + if (!end_p) 198 + end_p = p + strlen(p); 199 + 200 + if (!suffix) { 201 + int i; 202 + 203 + /* skip the one with any known suffix */ 204 + for (i = 0; suffix_tbl[i]; i++) { 205 + q = end_p - strlen(suffix_tbl[i]); 206 + if (!strncmp(q, suffix_tbl[i], 207 + strlen(suffix_tbl[i]))) 208 + goto next; 209 + } 210 + ck_cmdline = p; 211 + } else { 212 + q = end_p - strlen(suffix); 213 + if (!strncmp(q, suffix, strlen(suffix))) 214 + ck_cmdline = p; 215 + } 216 + next: 217 + p = strstr(p+1, name); 218 + } 219 + 220 + if (!ck_cmdline) 221 + return NULL; 222 + 223 + return ck_cmdline; 224 + } 225 + 226 + static int __init __parse_crashkernel(char *cmdline, 227 + unsigned long long system_ram, 228 + unsigned long long *crash_size, 229 + unsigned long long *crash_base, 230 + const char *name, 231 + const char *suffix) 232 + { 233 + char *first_colon, *first_space; 234 + char *ck_cmdline; 235 + 236 + BUG_ON(!crash_size || !crash_base); 237 + *crash_size = 0; 238 + *crash_base = 0; 239 + 240 + ck_cmdline = get_last_crashkernel(cmdline, name, suffix); 241 + 242 + if (!ck_cmdline) 243 + return -EINVAL; 244 + 245 + ck_cmdline += strlen(name); 246 + 247 + if (suffix) 248 + return parse_crashkernel_suffix(ck_cmdline, crash_size, 249 + suffix); 250 + /* 251 + * if the commandline contains a ':', then that's the extended 252 + * syntax -- if not, it must be the classic syntax 253 + */ 254 + first_colon = strchr(ck_cmdline, ':'); 255 + first_space = strchr(ck_cmdline, ' '); 256 + if (first_colon && (!first_space || first_colon < first_space)) 257 + return parse_crashkernel_mem(ck_cmdline, system_ram, 258 + crash_size, crash_base); 259 + 260 + return parse_crashkernel_simple(ck_cmdline, crash_size, crash_base); 261 + } 262 + 263 + /* 264 + * That function is the entry point for command line parsing and should be 265 + * called from the arch-specific code. 266 + */ 267 + int __init parse_crashkernel(char *cmdline, 268 + unsigned long long system_ram, 269 + unsigned long long *crash_size, 270 + unsigned long long *crash_base) 271 + { 272 + return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 273 + "crashkernel=", NULL); 274 + } 275 + 276 + int __init parse_crashkernel_high(char *cmdline, 277 + unsigned long long system_ram, 278 + unsigned long long *crash_size, 279 + unsigned long long *crash_base) 280 + { 281 + return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 282 + "crashkernel=", suffix_tbl[SUFFIX_HIGH]); 283 + } 284 + 285 + int __init parse_crashkernel_low(char *cmdline, 286 + unsigned long long system_ram, 287 + unsigned long long *crash_size, 288 + unsigned long long *crash_base) 289 + { 290 + return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 291 + "crashkernel=", suffix_tbl[SUFFIX_LOW]); 292 + } 293 + 294 + static u32 *append_elf_note(u32 *buf, char *name, unsigned int type, 295 + void *data, size_t data_len) 296 + { 297 + struct elf_note note; 298 + 299 + note.n_namesz = strlen(name) + 1; 300 + note.n_descsz = data_len; 301 + note.n_type = type; 302 + memcpy(buf, &note, sizeof(note)); 303 + buf += (sizeof(note) + 3)/4; 304 + memcpy(buf, name, note.n_namesz); 305 + buf += (note.n_namesz + 3)/4; 306 + memcpy(buf, data, note.n_descsz); 307 + buf += (note.n_descsz + 3)/4; 308 + 309 + return buf; 310 + } 311 + 312 + static void final_note(u32 *buf) 313 + { 314 + struct elf_note note; 315 + 316 + note.n_namesz = 0; 317 + note.n_descsz = 0; 318 + note.n_type = 0; 319 + memcpy(buf, &note, sizeof(note)); 320 + } 321 + 322 + static void update_vmcoreinfo_note(void) 323 + { 324 + u32 *buf = vmcoreinfo_note; 325 + 326 + if (!vmcoreinfo_size) 327 + return; 328 + buf = append_elf_note(buf, VMCOREINFO_NOTE_NAME, 0, vmcoreinfo_data, 329 + vmcoreinfo_size); 330 + final_note(buf); 331 + } 332 + 333 + void crash_save_vmcoreinfo(void) 334 + { 335 + vmcoreinfo_append_str("CRASHTIME=%ld\n", get_seconds()); 336 + update_vmcoreinfo_note(); 337 + } 338 + 339 + void vmcoreinfo_append_str(const char *fmt, ...) 340 + { 341 + va_list args; 342 + char buf[0x50]; 343 + size_t r; 344 + 345 + va_start(args, fmt); 346 + r = vscnprintf(buf, sizeof(buf), fmt, args); 347 + va_end(args); 348 + 349 + r = min(r, vmcoreinfo_max_size - vmcoreinfo_size); 350 + 351 + memcpy(&vmcoreinfo_data[vmcoreinfo_size], buf, r); 352 + 353 + vmcoreinfo_size += r; 354 + } 355 + 356 + /* 357 + * provide an empty default implementation here -- architecture 358 + * code may override this 359 + */ 360 + void __weak arch_crash_save_vmcoreinfo(void) 361 + {} 362 + 363 + phys_addr_t __weak paddr_vmcoreinfo_note(void) 364 + { 365 + return __pa_symbol((unsigned long)(char *)&vmcoreinfo_note); 366 + } 367 + 368 + static int __init crash_save_vmcoreinfo_init(void) 369 + { 370 + VMCOREINFO_OSRELEASE(init_uts_ns.name.release); 371 + VMCOREINFO_PAGESIZE(PAGE_SIZE); 372 + 373 + VMCOREINFO_SYMBOL(init_uts_ns); 374 + VMCOREINFO_SYMBOL(node_online_map); 375 + #ifdef CONFIG_MMU 376 + VMCOREINFO_SYMBOL(swapper_pg_dir); 377 + #endif 378 + VMCOREINFO_SYMBOL(_stext); 379 + VMCOREINFO_SYMBOL(vmap_area_list); 380 + 381 + #ifndef CONFIG_NEED_MULTIPLE_NODES 382 + VMCOREINFO_SYMBOL(mem_map); 383 + VMCOREINFO_SYMBOL(contig_page_data); 384 + #endif 385 + #ifdef CONFIG_SPARSEMEM 386 + VMCOREINFO_SYMBOL(mem_section); 387 + VMCOREINFO_LENGTH(mem_section, NR_SECTION_ROOTS); 388 + VMCOREINFO_STRUCT_SIZE(mem_section); 389 + VMCOREINFO_OFFSET(mem_section, section_mem_map); 390 + #endif 391 + VMCOREINFO_STRUCT_SIZE(page); 392 + VMCOREINFO_STRUCT_SIZE(pglist_data); 393 + VMCOREINFO_STRUCT_SIZE(zone); 394 + VMCOREINFO_STRUCT_SIZE(free_area); 395 + VMCOREINFO_STRUCT_SIZE(list_head); 396 + VMCOREINFO_SIZE(nodemask_t); 397 + VMCOREINFO_OFFSET(page, flags); 398 + VMCOREINFO_OFFSET(page, _refcount); 399 + VMCOREINFO_OFFSET(page, mapping); 400 + VMCOREINFO_OFFSET(page, lru); 401 + VMCOREINFO_OFFSET(page, _mapcount); 402 + VMCOREINFO_OFFSET(page, private); 403 + VMCOREINFO_OFFSET(page, compound_dtor); 404 + VMCOREINFO_OFFSET(page, compound_order); 405 + VMCOREINFO_OFFSET(page, compound_head); 406 + VMCOREINFO_OFFSET(pglist_data, node_zones); 407 + VMCOREINFO_OFFSET(pglist_data, nr_zones); 408 + #ifdef CONFIG_FLAT_NODE_MEM_MAP 409 + VMCOREINFO_OFFSET(pglist_data, node_mem_map); 410 + #endif 411 + VMCOREINFO_OFFSET(pglist_data, node_start_pfn); 412 + VMCOREINFO_OFFSET(pglist_data, node_spanned_pages); 413 + VMCOREINFO_OFFSET(pglist_data, node_id); 414 + VMCOREINFO_OFFSET(zone, free_area); 415 + VMCOREINFO_OFFSET(zone, vm_stat); 416 + VMCOREINFO_OFFSET(zone, spanned_pages); 417 + VMCOREINFO_OFFSET(free_area, free_list); 418 + VMCOREINFO_OFFSET(list_head, next); 419 + VMCOREINFO_OFFSET(list_head, prev); 420 + VMCOREINFO_OFFSET(vmap_area, va_start); 421 + VMCOREINFO_OFFSET(vmap_area, list); 422 + VMCOREINFO_LENGTH(zone.free_area, MAX_ORDER); 423 + log_buf_vmcoreinfo_setup(); 424 + VMCOREINFO_LENGTH(free_area.free_list, MIGRATE_TYPES); 425 + VMCOREINFO_NUMBER(NR_FREE_PAGES); 426 + VMCOREINFO_NUMBER(PG_lru); 427 + VMCOREINFO_NUMBER(PG_private); 428 + VMCOREINFO_NUMBER(PG_swapcache); 429 + VMCOREINFO_NUMBER(PG_slab); 430 + #ifdef CONFIG_MEMORY_FAILURE 431 + VMCOREINFO_NUMBER(PG_hwpoison); 432 + #endif 433 + VMCOREINFO_NUMBER(PG_head_mask); 434 + VMCOREINFO_NUMBER(PAGE_BUDDY_MAPCOUNT_VALUE); 435 + #ifdef CONFIG_HUGETLB_PAGE 436 + VMCOREINFO_NUMBER(HUGETLB_PAGE_DTOR); 437 + #endif 438 + 439 + arch_crash_save_vmcoreinfo(); 440 + update_vmcoreinfo_note(); 441 + 442 + return 0; 443 + } 444 + 445 + subsys_initcall(crash_save_vmcoreinfo_init);
-403
kernel/kexec_core.c
··· 51 51 /* Per cpu memory for storing cpu states in case of system crash. */ 52 52 note_buf_t __percpu *crash_notes; 53 53 54 - /* vmcoreinfo stuff */ 55 - static unsigned char vmcoreinfo_data[VMCOREINFO_BYTES]; 56 - u32 vmcoreinfo_note[VMCOREINFO_NOTE_SIZE/4]; 57 - size_t vmcoreinfo_size; 58 - size_t vmcoreinfo_max_size = sizeof(vmcoreinfo_data); 59 - 60 54 /* Flag to indicate we are going to kexec a new kernel */ 61 55 bool kexec_in_progress = false; 62 56 ··· 1077 1083 } 1078 1084 subsys_initcall(crash_notes_memory_init); 1079 1085 1080 - 1081 - /* 1082 - * parsing the "crashkernel" commandline 1083 - * 1084 - * this code is intended to be called from architecture specific code 1085 - */ 1086 - 1087 - 1088 - /* 1089 - * This function parses command lines in the format 1090 - * 1091 - * crashkernel=ramsize-range:size[,...][@offset] 1092 - * 1093 - * The function returns 0 on success and -EINVAL on failure. 1094 - */ 1095 - static int __init parse_crashkernel_mem(char *cmdline, 1096 - unsigned long long system_ram, 1097 - unsigned long long *crash_size, 1098 - unsigned long long *crash_base) 1099 - { 1100 - char *cur = cmdline, *tmp; 1101 - 1102 - /* for each entry of the comma-separated list */ 1103 - do { 1104 - unsigned long long start, end = ULLONG_MAX, size; 1105 - 1106 - /* get the start of the range */ 1107 - start = memparse(cur, &tmp); 1108 - if (cur == tmp) { 1109 - pr_warn("crashkernel: Memory value expected\n"); 1110 - return -EINVAL; 1111 - } 1112 - cur = tmp; 1113 - if (*cur != '-') { 1114 - pr_warn("crashkernel: '-' expected\n"); 1115 - return -EINVAL; 1116 - } 1117 - cur++; 1118 - 1119 - /* if no ':' is here, than we read the end */ 1120 - if (*cur != ':') { 1121 - end = memparse(cur, &tmp); 1122 - if (cur == tmp) { 1123 - pr_warn("crashkernel: Memory value expected\n"); 1124 - return -EINVAL; 1125 - } 1126 - cur = tmp; 1127 - if (end <= start) { 1128 - pr_warn("crashkernel: end <= start\n"); 1129 - return -EINVAL; 1130 - } 1131 - } 1132 - 1133 - if (*cur != ':') { 1134 - pr_warn("crashkernel: ':' expected\n"); 1135 - return -EINVAL; 1136 - } 1137 - cur++; 1138 - 1139 - size = memparse(cur, &tmp); 1140 - if (cur == tmp) { 1141 - pr_warn("Memory value expected\n"); 1142 - return -EINVAL; 1143 - } 1144 - cur = tmp; 1145 - if (size >= system_ram) { 1146 - pr_warn("crashkernel: invalid size\n"); 1147 - return -EINVAL; 1148 - } 1149 - 1150 - /* match ? */ 1151 - if (system_ram >= start && system_ram < end) { 1152 - *crash_size = size; 1153 - break; 1154 - } 1155 - } while (*cur++ == ','); 1156 - 1157 - if (*crash_size > 0) { 1158 - while (*cur && *cur != ' ' && *cur != '@') 1159 - cur++; 1160 - if (*cur == '@') { 1161 - cur++; 1162 - *crash_base = memparse(cur, &tmp); 1163 - if (cur == tmp) { 1164 - pr_warn("Memory value expected after '@'\n"); 1165 - return -EINVAL; 1166 - } 1167 - } 1168 - } 1169 - 1170 - return 0; 1171 - } 1172 - 1173 - /* 1174 - * That function parses "simple" (old) crashkernel command lines like 1175 - * 1176 - * crashkernel=size[@offset] 1177 - * 1178 - * It returns 0 on success and -EINVAL on failure. 1179 - */ 1180 - static int __init parse_crashkernel_simple(char *cmdline, 1181 - unsigned long long *crash_size, 1182 - unsigned long long *crash_base) 1183 - { 1184 - char *cur = cmdline; 1185 - 1186 - *crash_size = memparse(cmdline, &cur); 1187 - if (cmdline == cur) { 1188 - pr_warn("crashkernel: memory value expected\n"); 1189 - return -EINVAL; 1190 - } 1191 - 1192 - if (*cur == '@') 1193 - *crash_base = memparse(cur+1, &cur); 1194 - else if (*cur != ' ' && *cur != '\0') { 1195 - pr_warn("crashkernel: unrecognized char: %c\n", *cur); 1196 - return -EINVAL; 1197 - } 1198 - 1199 - return 0; 1200 - } 1201 - 1202 - #define SUFFIX_HIGH 0 1203 - #define SUFFIX_LOW 1 1204 - #define SUFFIX_NULL 2 1205 - static __initdata char *suffix_tbl[] = { 1206 - [SUFFIX_HIGH] = ",high", 1207 - [SUFFIX_LOW] = ",low", 1208 - [SUFFIX_NULL] = NULL, 1209 - }; 1210 - 1211 - /* 1212 - * That function parses "suffix" crashkernel command lines like 1213 - * 1214 - * crashkernel=size,[high|low] 1215 - * 1216 - * It returns 0 on success and -EINVAL on failure. 1217 - */ 1218 - static int __init parse_crashkernel_suffix(char *cmdline, 1219 - unsigned long long *crash_size, 1220 - const char *suffix) 1221 - { 1222 - char *cur = cmdline; 1223 - 1224 - *crash_size = memparse(cmdline, &cur); 1225 - if (cmdline == cur) { 1226 - pr_warn("crashkernel: memory value expected\n"); 1227 - return -EINVAL; 1228 - } 1229 - 1230 - /* check with suffix */ 1231 - if (strncmp(cur, suffix, strlen(suffix))) { 1232 - pr_warn("crashkernel: unrecognized char: %c\n", *cur); 1233 - return -EINVAL; 1234 - } 1235 - cur += strlen(suffix); 1236 - if (*cur != ' ' && *cur != '\0') { 1237 - pr_warn("crashkernel: unrecognized char: %c\n", *cur); 1238 - return -EINVAL; 1239 - } 1240 - 1241 - return 0; 1242 - } 1243 - 1244 - static __init char *get_last_crashkernel(char *cmdline, 1245 - const char *name, 1246 - const char *suffix) 1247 - { 1248 - char *p = cmdline, *ck_cmdline = NULL; 1249 - 1250 - /* find crashkernel and use the last one if there are more */ 1251 - p = strstr(p, name); 1252 - while (p) { 1253 - char *end_p = strchr(p, ' '); 1254 - char *q; 1255 - 1256 - if (!end_p) 1257 - end_p = p + strlen(p); 1258 - 1259 - if (!suffix) { 1260 - int i; 1261 - 1262 - /* skip the one with any known suffix */ 1263 - for (i = 0; suffix_tbl[i]; i++) { 1264 - q = end_p - strlen(suffix_tbl[i]); 1265 - if (!strncmp(q, suffix_tbl[i], 1266 - strlen(suffix_tbl[i]))) 1267 - goto next; 1268 - } 1269 - ck_cmdline = p; 1270 - } else { 1271 - q = end_p - strlen(suffix); 1272 - if (!strncmp(q, suffix, strlen(suffix))) 1273 - ck_cmdline = p; 1274 - } 1275 - next: 1276 - p = strstr(p+1, name); 1277 - } 1278 - 1279 - if (!ck_cmdline) 1280 - return NULL; 1281 - 1282 - return ck_cmdline; 1283 - } 1284 - 1285 - static int __init __parse_crashkernel(char *cmdline, 1286 - unsigned long long system_ram, 1287 - unsigned long long *crash_size, 1288 - unsigned long long *crash_base, 1289 - const char *name, 1290 - const char *suffix) 1291 - { 1292 - char *first_colon, *first_space; 1293 - char *ck_cmdline; 1294 - 1295 - BUG_ON(!crash_size || !crash_base); 1296 - *crash_size = 0; 1297 - *crash_base = 0; 1298 - 1299 - ck_cmdline = get_last_crashkernel(cmdline, name, suffix); 1300 - 1301 - if (!ck_cmdline) 1302 - return -EINVAL; 1303 - 1304 - ck_cmdline += strlen(name); 1305 - 1306 - if (suffix) 1307 - return parse_crashkernel_suffix(ck_cmdline, crash_size, 1308 - suffix); 1309 - /* 1310 - * if the commandline contains a ':', then that's the extended 1311 - * syntax -- if not, it must be the classic syntax 1312 - */ 1313 - first_colon = strchr(ck_cmdline, ':'); 1314 - first_space = strchr(ck_cmdline, ' '); 1315 - if (first_colon && (!first_space || first_colon < first_space)) 1316 - return parse_crashkernel_mem(ck_cmdline, system_ram, 1317 - crash_size, crash_base); 1318 - 1319 - return parse_crashkernel_simple(ck_cmdline, crash_size, crash_base); 1320 - } 1321 - 1322 - /* 1323 - * That function is the entry point for command line parsing and should be 1324 - * called from the arch-specific code. 1325 - */ 1326 - int __init parse_crashkernel(char *cmdline, 1327 - unsigned long long system_ram, 1328 - unsigned long long *crash_size, 1329 - unsigned long long *crash_base) 1330 - { 1331 - return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 1332 - "crashkernel=", NULL); 1333 - } 1334 - 1335 - int __init parse_crashkernel_high(char *cmdline, 1336 - unsigned long long system_ram, 1337 - unsigned long long *crash_size, 1338 - unsigned long long *crash_base) 1339 - { 1340 - return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 1341 - "crashkernel=", suffix_tbl[SUFFIX_HIGH]); 1342 - } 1343 - 1344 - int __init parse_crashkernel_low(char *cmdline, 1345 - unsigned long long system_ram, 1346 - unsigned long long *crash_size, 1347 - unsigned long long *crash_base) 1348 - { 1349 - return __parse_crashkernel(cmdline, system_ram, crash_size, crash_base, 1350 - "crashkernel=", suffix_tbl[SUFFIX_LOW]); 1351 - } 1352 - 1353 - static void update_vmcoreinfo_note(void) 1354 - { 1355 - u32 *buf = vmcoreinfo_note; 1356 - 1357 - if (!vmcoreinfo_size) 1358 - return; 1359 - buf = append_elf_note(buf, VMCOREINFO_NOTE_NAME, 0, vmcoreinfo_data, 1360 - vmcoreinfo_size); 1361 - final_note(buf); 1362 - } 1363 - 1364 - void crash_save_vmcoreinfo(void) 1365 - { 1366 - vmcoreinfo_append_str("CRASHTIME=%ld\n", get_seconds()); 1367 - update_vmcoreinfo_note(); 1368 - } 1369 - 1370 - void vmcoreinfo_append_str(const char *fmt, ...) 1371 - { 1372 - va_list args; 1373 - char buf[0x50]; 1374 - size_t r; 1375 - 1376 - va_start(args, fmt); 1377 - r = vscnprintf(buf, sizeof(buf), fmt, args); 1378 - va_end(args); 1379 - 1380 - r = min(r, vmcoreinfo_max_size - vmcoreinfo_size); 1381 - 1382 - memcpy(&vmcoreinfo_data[vmcoreinfo_size], buf, r); 1383 - 1384 - vmcoreinfo_size += r; 1385 - } 1386 - 1387 - /* 1388 - * provide an empty default implementation here -- architecture 1389 - * code may override this 1390 - */ 1391 - void __weak arch_crash_save_vmcoreinfo(void) 1392 - {} 1393 - 1394 - phys_addr_t __weak paddr_vmcoreinfo_note(void) 1395 - { 1396 - return __pa_symbol((unsigned long)(char *)&vmcoreinfo_note); 1397 - } 1398 - 1399 - static int __init crash_save_vmcoreinfo_init(void) 1400 - { 1401 - VMCOREINFO_OSRELEASE(init_uts_ns.name.release); 1402 - VMCOREINFO_PAGESIZE(PAGE_SIZE); 1403 - 1404 - VMCOREINFO_SYMBOL(init_uts_ns); 1405 - VMCOREINFO_SYMBOL(node_online_map); 1406 - #ifdef CONFIG_MMU 1407 - VMCOREINFO_SYMBOL(swapper_pg_dir); 1408 - #endif 1409 - VMCOREINFO_SYMBOL(_stext); 1410 - VMCOREINFO_SYMBOL(vmap_area_list); 1411 - 1412 - #ifndef CONFIG_NEED_MULTIPLE_NODES 1413 - VMCOREINFO_SYMBOL(mem_map); 1414 - VMCOREINFO_SYMBOL(contig_page_data); 1415 - #endif 1416 - #ifdef CONFIG_SPARSEMEM 1417 - VMCOREINFO_SYMBOL(mem_section); 1418 - VMCOREINFO_LENGTH(mem_section, NR_SECTION_ROOTS); 1419 - VMCOREINFO_STRUCT_SIZE(mem_section); 1420 - VMCOREINFO_OFFSET(mem_section, section_mem_map); 1421 - #endif 1422 - VMCOREINFO_STRUCT_SIZE(page); 1423 - VMCOREINFO_STRUCT_SIZE(pglist_data); 1424 - VMCOREINFO_STRUCT_SIZE(zone); 1425 - VMCOREINFO_STRUCT_SIZE(free_area); 1426 - VMCOREINFO_STRUCT_SIZE(list_head); 1427 - VMCOREINFO_SIZE(nodemask_t); 1428 - VMCOREINFO_OFFSET(page, flags); 1429 - VMCOREINFO_OFFSET(page, _refcount); 1430 - VMCOREINFO_OFFSET(page, mapping); 1431 - VMCOREINFO_OFFSET(page, lru); 1432 - VMCOREINFO_OFFSET(page, _mapcount); 1433 - VMCOREINFO_OFFSET(page, private); 1434 - VMCOREINFO_OFFSET(page, compound_dtor); 1435 - VMCOREINFO_OFFSET(page, compound_order); 1436 - VMCOREINFO_OFFSET(page, compound_head); 1437 - VMCOREINFO_OFFSET(pglist_data, node_zones); 1438 - VMCOREINFO_OFFSET(pglist_data, nr_zones); 1439 - #ifdef CONFIG_FLAT_NODE_MEM_MAP 1440 - VMCOREINFO_OFFSET(pglist_data, node_mem_map); 1441 - #endif 1442 - VMCOREINFO_OFFSET(pglist_data, node_start_pfn); 1443 - VMCOREINFO_OFFSET(pglist_data, node_spanned_pages); 1444 - VMCOREINFO_OFFSET(pglist_data, node_id); 1445 - VMCOREINFO_OFFSET(zone, free_area); 1446 - VMCOREINFO_OFFSET(zone, vm_stat); 1447 - VMCOREINFO_OFFSET(zone, spanned_pages); 1448 - VMCOREINFO_OFFSET(free_area, free_list); 1449 - VMCOREINFO_OFFSET(list_head, next); 1450 - VMCOREINFO_OFFSET(list_head, prev); 1451 - VMCOREINFO_OFFSET(vmap_area, va_start); 1452 - VMCOREINFO_OFFSET(vmap_area, list); 1453 - VMCOREINFO_LENGTH(zone.free_area, MAX_ORDER); 1454 - log_buf_kexec_setup(); 1455 - VMCOREINFO_LENGTH(free_area.free_list, MIGRATE_TYPES); 1456 - VMCOREINFO_NUMBER(NR_FREE_PAGES); 1457 - VMCOREINFO_NUMBER(PG_lru); 1458 - VMCOREINFO_NUMBER(PG_private); 1459 - VMCOREINFO_NUMBER(PG_swapcache); 1460 - VMCOREINFO_NUMBER(PG_slab); 1461 - #ifdef CONFIG_MEMORY_FAILURE 1462 - VMCOREINFO_NUMBER(PG_hwpoison); 1463 - #endif 1464 - VMCOREINFO_NUMBER(PG_head_mask); 1465 - VMCOREINFO_NUMBER(PAGE_BUDDY_MAPCOUNT_VALUE); 1466 - #ifdef CONFIG_HUGETLB_PAGE 1467 - VMCOREINFO_NUMBER(HUGETLB_PAGE_DTOR); 1468 - #endif 1469 - 1470 - arch_crash_save_vmcoreinfo(); 1471 - update_vmcoreinfo_note(); 1472 - 1473 - return 0; 1474 - } 1475 - 1476 - subsys_initcall(crash_save_vmcoreinfo_init); 1477 1086 1478 1087 /* 1479 1088 * Move into place and start executing a preloaded standalone
+7 -1
kernel/ksysfs.c
··· 125 125 } 126 126 KERNEL_ATTR_RW(kexec_crash_size); 127 127 128 + #endif /* CONFIG_KEXEC_CORE */ 129 + 130 + #ifdef CONFIG_CRASH_CORE 131 + 128 132 static ssize_t vmcoreinfo_show(struct kobject *kobj, 129 133 struct kobj_attribute *attr, char *buf) 130 134 { ··· 138 134 } 139 135 KERNEL_ATTR_RO(vmcoreinfo); 140 136 141 - #endif /* CONFIG_KEXEC_CORE */ 137 + #endif /* CONFIG_CRASH_CORE */ 142 138 143 139 /* whether file capabilities are enabled */ 144 140 static ssize_t fscaps_show(struct kobject *kobj, ··· 223 219 &kexec_loaded_attr.attr, 224 220 &kexec_crash_loaded_attr.attr, 225 221 &kexec_crash_size_attr.attr, 222 + #endif 223 + #ifdef CONFIG_CRASH_CORE 226 224 &vmcoreinfo_attr.attr, 227 225 #endif 228 226 #ifndef CONFIG_TINY_RCU
+3 -3
kernel/printk/printk.c
··· 32 32 #include <linux/bootmem.h> 33 33 #include <linux/memblock.h> 34 34 #include <linux/syscalls.h> 35 - #include <linux/kexec.h> 35 + #include <linux/crash_core.h> 36 36 #include <linux/kdb.h> 37 37 #include <linux/ratelimit.h> 38 38 #include <linux/kmsg_dump.h> ··· 1002 1002 .release = devkmsg_release, 1003 1003 }; 1004 1004 1005 - #ifdef CONFIG_KEXEC_CORE 1005 + #ifdef CONFIG_CRASH_CORE 1006 1006 /* 1007 1007 * This appends the listed symbols to /proc/vmcore 1008 1008 * ··· 1011 1011 * symbols are specifically used so that utilities can access and extract the 1012 1012 * dmesg log from a vmcore file after a crash. 1013 1013 */ 1014 - void log_buf_kexec_setup(void) 1014 + void log_buf_vmcoreinfo_setup(void) 1015 1015 { 1016 1016 VMCOREINFO_SYMBOL(log_buf); 1017 1017 VMCOREINFO_SYMBOL(log_buf_len);