1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * RAM Oops/Panic logger
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (C) 2010 Marco Stornelli <marco.stornelli@gmail.com>
6*4882a593Smuzhiyun * Copyright (C) 2011 Kees Cook <keescook@chromium.org>
7*4882a593Smuzhiyun */
8*4882a593Smuzhiyun
9*4882a593Smuzhiyun #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/kernel.h>
12*4882a593Smuzhiyun #include <linux/err.h>
13*4882a593Smuzhiyun #include <linux/module.h>
14*4882a593Smuzhiyun #include <linux/version.h>
15*4882a593Smuzhiyun #include <linux/pstore.h>
16*4882a593Smuzhiyun #include <linux/io.h>
17*4882a593Smuzhiyun #include <linux/ioport.h>
18*4882a593Smuzhiyun #include <linux/platform_device.h>
19*4882a593Smuzhiyun #include <linux/slab.h>
20*4882a593Smuzhiyun #include <linux/compiler.h>
21*4882a593Smuzhiyun #include <linux/pstore_ram.h>
22*4882a593Smuzhiyun #include <linux/of.h>
23*4882a593Smuzhiyun #include <linux/of_address.h>
24*4882a593Smuzhiyun #include <linux/of_reserved_mem.h>
25*4882a593Smuzhiyun #include "internal.h"
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #if IS_REACHABLE(CONFIG_ROCKCHIP_MINIDUMP)
28*4882a593Smuzhiyun #include <soc/rockchip/rk_minidump.h>
29*4882a593Smuzhiyun #endif
30*4882a593Smuzhiyun
31*4882a593Smuzhiyun #define RAMOOPS_KERNMSG_HDR "===="
32*4882a593Smuzhiyun #define MIN_MEM_SIZE 4096UL
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun static ulong record_size = MIN_MEM_SIZE;
35*4882a593Smuzhiyun module_param(record_size, ulong, 0400);
36*4882a593Smuzhiyun MODULE_PARM_DESC(record_size,
37*4882a593Smuzhiyun "size of each dump done on oops/panic");
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun static ulong ramoops_console_size = MIN_MEM_SIZE;
40*4882a593Smuzhiyun module_param_named(console_size, ramoops_console_size, ulong, 0400);
41*4882a593Smuzhiyun MODULE_PARM_DESC(console_size, "size of kernel console log");
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun static ulong ramoops_ftrace_size = MIN_MEM_SIZE;
44*4882a593Smuzhiyun module_param_named(ftrace_size, ramoops_ftrace_size, ulong, 0400);
45*4882a593Smuzhiyun MODULE_PARM_DESC(ftrace_size, "size of ftrace log");
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun static ulong ramoops_pmsg_size = MIN_MEM_SIZE;
48*4882a593Smuzhiyun module_param_named(pmsg_size, ramoops_pmsg_size, ulong, 0400);
49*4882a593Smuzhiyun MODULE_PARM_DESC(pmsg_size, "size of user space message log");
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun static unsigned long long mem_address;
52*4882a593Smuzhiyun module_param_hw(mem_address, ullong, other, 0400);
53*4882a593Smuzhiyun MODULE_PARM_DESC(mem_address,
54*4882a593Smuzhiyun "start of reserved RAM used to store oops/panic logs");
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun static ulong mem_size;
57*4882a593Smuzhiyun module_param(mem_size, ulong, 0400);
58*4882a593Smuzhiyun MODULE_PARM_DESC(mem_size,
59*4882a593Smuzhiyun "size of reserved RAM used to store oops/panic logs");
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun static unsigned int mem_type;
62*4882a593Smuzhiyun module_param(mem_type, uint, 0400);
63*4882a593Smuzhiyun MODULE_PARM_DESC(mem_type,
64*4882a593Smuzhiyun "memory type: 0=write-combined (default), 1=unbuffered, 2=cached");
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun static int ramoops_max_reason = -1;
67*4882a593Smuzhiyun module_param_named(max_reason, ramoops_max_reason, int, 0400);
68*4882a593Smuzhiyun MODULE_PARM_DESC(max_reason,
69*4882a593Smuzhiyun "maximum reason for kmsg dump (default 2: Oops and Panic) ");
70*4882a593Smuzhiyun
71*4882a593Smuzhiyun static int ramoops_ecc;
72*4882a593Smuzhiyun module_param_named(ecc, ramoops_ecc, int, 0400);
73*4882a593Smuzhiyun MODULE_PARM_DESC(ramoops_ecc,
74*4882a593Smuzhiyun "if non-zero, the option enables ECC support and specifies "
75*4882a593Smuzhiyun "ECC buffer size in bytes (1 is a special value, means 16 "
76*4882a593Smuzhiyun "bytes ECC)");
77*4882a593Smuzhiyun
78*4882a593Smuzhiyun static int ramoops_dump_oops = -1;
79*4882a593Smuzhiyun module_param_named(dump_oops, ramoops_dump_oops, int, 0400);
80*4882a593Smuzhiyun MODULE_PARM_DESC(dump_oops,
81*4882a593Smuzhiyun "(deprecated: use max_reason instead) set to 1 to dump oopses & panics, 0 to only dump panics");
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun struct ramoops_context {
84*4882a593Smuzhiyun struct persistent_ram_zone **dprzs; /* Oops dump zones */
85*4882a593Smuzhiyun struct persistent_ram_zone *cprz; /* Console zone */
86*4882a593Smuzhiyun struct persistent_ram_zone **fprzs; /* Ftrace zones */
87*4882a593Smuzhiyun struct persistent_ram_zone *mprz; /* PMSG zone */
88*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
89*4882a593Smuzhiyun struct persistent_ram_zone **boot_przs; /* BOOT log zones */
90*4882a593Smuzhiyun #endif
91*4882a593Smuzhiyun phys_addr_t phys_addr;
92*4882a593Smuzhiyun unsigned long size;
93*4882a593Smuzhiyun unsigned int memtype;
94*4882a593Smuzhiyun size_t record_size;
95*4882a593Smuzhiyun size_t console_size;
96*4882a593Smuzhiyun size_t ftrace_size;
97*4882a593Smuzhiyun size_t pmsg_size;
98*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
99*4882a593Smuzhiyun size_t boot_log_size;
100*4882a593Smuzhiyun #endif
101*4882a593Smuzhiyun u32 flags;
102*4882a593Smuzhiyun struct persistent_ram_ecc_info ecc_info;
103*4882a593Smuzhiyun unsigned int max_dump_cnt;
104*4882a593Smuzhiyun unsigned int dump_write_cnt;
105*4882a593Smuzhiyun /* _read_cnt need clear on ramoops_pstore_open */
106*4882a593Smuzhiyun unsigned int dump_read_cnt;
107*4882a593Smuzhiyun unsigned int console_read_cnt;
108*4882a593Smuzhiyun unsigned int max_ftrace_cnt;
109*4882a593Smuzhiyun unsigned int ftrace_read_cnt;
110*4882a593Smuzhiyun unsigned int pmsg_read_cnt;
111*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
112*4882a593Smuzhiyun unsigned int boot_log_read_cnt;
113*4882a593Smuzhiyun unsigned int max_boot_log_cnt;
114*4882a593Smuzhiyun #endif
115*4882a593Smuzhiyun struct pstore_info pstore;
116*4882a593Smuzhiyun };
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun static struct platform_device *dummy;
119*4882a593Smuzhiyun
ramoops_pstore_open(struct pstore_info * psi)120*4882a593Smuzhiyun static int ramoops_pstore_open(struct pstore_info *psi)
121*4882a593Smuzhiyun {
122*4882a593Smuzhiyun struct ramoops_context *cxt = psi->data;
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun cxt->dump_read_cnt = 0;
125*4882a593Smuzhiyun cxt->console_read_cnt = 0;
126*4882a593Smuzhiyun cxt->ftrace_read_cnt = 0;
127*4882a593Smuzhiyun cxt->pmsg_read_cnt = 0;
128*4882a593Smuzhiyun return 0;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun static struct persistent_ram_zone *
ramoops_get_next_prz(struct persistent_ram_zone * przs[],int id,struct pstore_record * record)132*4882a593Smuzhiyun ramoops_get_next_prz(struct persistent_ram_zone *przs[], int id,
133*4882a593Smuzhiyun struct pstore_record *record)
134*4882a593Smuzhiyun {
135*4882a593Smuzhiyun struct persistent_ram_zone *prz;
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun /* Give up if we never existed or have hit the end. */
138*4882a593Smuzhiyun if (!przs)
139*4882a593Smuzhiyun return NULL;
140*4882a593Smuzhiyun
141*4882a593Smuzhiyun prz = przs[id];
142*4882a593Smuzhiyun if (!prz)
143*4882a593Smuzhiyun return NULL;
144*4882a593Smuzhiyun
145*4882a593Smuzhiyun /* Update old/shadowed buffer. */
146*4882a593Smuzhiyun if (prz->type == PSTORE_TYPE_DMESG)
147*4882a593Smuzhiyun persistent_ram_save_old(prz);
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun if (!persistent_ram_old_size(prz))
150*4882a593Smuzhiyun return NULL;
151*4882a593Smuzhiyun
152*4882a593Smuzhiyun record->type = prz->type;
153*4882a593Smuzhiyun record->id = id;
154*4882a593Smuzhiyun
155*4882a593Smuzhiyun return prz;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun
ramoops_read_kmsg_hdr(char * buffer,struct timespec64 * time,bool * compressed)158*4882a593Smuzhiyun static int ramoops_read_kmsg_hdr(char *buffer, struct timespec64 *time,
159*4882a593Smuzhiyun bool *compressed)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun char data_type;
162*4882a593Smuzhiyun int header_length = 0;
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun if (sscanf(buffer, RAMOOPS_KERNMSG_HDR "%lld.%lu-%c\n%n",
165*4882a593Smuzhiyun (time64_t *)&time->tv_sec, &time->tv_nsec, &data_type,
166*4882a593Smuzhiyun &header_length) == 3) {
167*4882a593Smuzhiyun time->tv_nsec *= 1000;
168*4882a593Smuzhiyun if (data_type == 'C')
169*4882a593Smuzhiyun *compressed = true;
170*4882a593Smuzhiyun else
171*4882a593Smuzhiyun *compressed = false;
172*4882a593Smuzhiyun } else if (sscanf(buffer, RAMOOPS_KERNMSG_HDR "%lld.%lu\n%n",
173*4882a593Smuzhiyun (time64_t *)&time->tv_sec, &time->tv_nsec,
174*4882a593Smuzhiyun &header_length) == 2) {
175*4882a593Smuzhiyun time->tv_nsec *= 1000;
176*4882a593Smuzhiyun *compressed = false;
177*4882a593Smuzhiyun } else {
178*4882a593Smuzhiyun time->tv_sec = 0;
179*4882a593Smuzhiyun time->tv_nsec = 0;
180*4882a593Smuzhiyun *compressed = false;
181*4882a593Smuzhiyun }
182*4882a593Smuzhiyun return header_length;
183*4882a593Smuzhiyun }
184*4882a593Smuzhiyun
prz_ok(struct persistent_ram_zone * prz)185*4882a593Smuzhiyun static bool prz_ok(struct persistent_ram_zone *prz)
186*4882a593Smuzhiyun {
187*4882a593Smuzhiyun return !!prz && !!(persistent_ram_old_size(prz) +
188*4882a593Smuzhiyun persistent_ram_ecc_string(prz, NULL, 0));
189*4882a593Smuzhiyun }
190*4882a593Smuzhiyun
191*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
ramoops_pstore_read_for_boot_log(struct pstore_record * record)192*4882a593Smuzhiyun ssize_t ramoops_pstore_read_for_boot_log(struct pstore_record *record)
193*4882a593Smuzhiyun {
194*4882a593Smuzhiyun struct ramoops_context *cxt = record->psi->data;
195*4882a593Smuzhiyun struct persistent_ram_zone *prz;
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun if (!cxt)
198*4882a593Smuzhiyun return 0;
199*4882a593Smuzhiyun
200*4882a593Smuzhiyun prz = cxt->boot_przs[record->id];
201*4882a593Smuzhiyun
202*4882a593Smuzhiyun if (!prz)
203*4882a593Smuzhiyun return 0;
204*4882a593Smuzhiyun
205*4882a593Smuzhiyun persistent_ram_free_old(prz);
206*4882a593Smuzhiyun persistent_ram_save_old(prz);
207*4882a593Smuzhiyun record->buf = prz->old_log;
208*4882a593Smuzhiyun record->size = prz->old_log_size;
209*4882a593Smuzhiyun return record->size;
210*4882a593Smuzhiyun }
211*4882a593Smuzhiyun #endif
212*4882a593Smuzhiyun
ramoops_pstore_read(struct pstore_record * record)213*4882a593Smuzhiyun static ssize_t ramoops_pstore_read(struct pstore_record *record)
214*4882a593Smuzhiyun {
215*4882a593Smuzhiyun ssize_t size = 0;
216*4882a593Smuzhiyun struct ramoops_context *cxt = record->psi->data;
217*4882a593Smuzhiyun struct persistent_ram_zone *prz = NULL;
218*4882a593Smuzhiyun int header_length = 0;
219*4882a593Smuzhiyun bool free_prz = false;
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun /*
222*4882a593Smuzhiyun * Ramoops headers provide time stamps for PSTORE_TYPE_DMESG, but
223*4882a593Smuzhiyun * PSTORE_TYPE_CONSOLE and PSTORE_TYPE_FTRACE don't currently have
224*4882a593Smuzhiyun * valid time stamps, so it is initialized to zero.
225*4882a593Smuzhiyun */
226*4882a593Smuzhiyun record->time.tv_sec = 0;
227*4882a593Smuzhiyun record->time.tv_nsec = 0;
228*4882a593Smuzhiyun record->compressed = false;
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun /* Find the next valid persistent_ram_zone for DMESG */
231*4882a593Smuzhiyun while (cxt->dump_read_cnt < cxt->max_dump_cnt && !prz) {
232*4882a593Smuzhiyun prz = ramoops_get_next_prz(cxt->dprzs, cxt->dump_read_cnt++,
233*4882a593Smuzhiyun record);
234*4882a593Smuzhiyun if (!prz_ok(prz))
235*4882a593Smuzhiyun continue;
236*4882a593Smuzhiyun header_length = ramoops_read_kmsg_hdr(persistent_ram_old(prz),
237*4882a593Smuzhiyun &record->time,
238*4882a593Smuzhiyun &record->compressed);
239*4882a593Smuzhiyun /* Clear and skip this DMESG record if it has no valid header */
240*4882a593Smuzhiyun if (!header_length) {
241*4882a593Smuzhiyun persistent_ram_free_old(prz);
242*4882a593Smuzhiyun persistent_ram_zap(prz);
243*4882a593Smuzhiyun prz = NULL;
244*4882a593Smuzhiyun }
245*4882a593Smuzhiyun }
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun if (!prz_ok(prz) && !cxt->console_read_cnt++)
248*4882a593Smuzhiyun prz = ramoops_get_next_prz(&cxt->cprz, 0 /* single */, record);
249*4882a593Smuzhiyun
250*4882a593Smuzhiyun if (!prz_ok(prz) && !cxt->pmsg_read_cnt++)
251*4882a593Smuzhiyun prz = ramoops_get_next_prz(&cxt->mprz, 0 /* single */, record);
252*4882a593Smuzhiyun
253*4882a593Smuzhiyun /* ftrace is last since it may want to dynamically allocate memory. */
254*4882a593Smuzhiyun if (!prz_ok(prz)) {
255*4882a593Smuzhiyun if (!(cxt->flags & RAMOOPS_FLAG_FTRACE_PER_CPU) &&
256*4882a593Smuzhiyun !cxt->ftrace_read_cnt++) {
257*4882a593Smuzhiyun prz = ramoops_get_next_prz(cxt->fprzs, 0 /* single */,
258*4882a593Smuzhiyun record);
259*4882a593Smuzhiyun } else {
260*4882a593Smuzhiyun /*
261*4882a593Smuzhiyun * Build a new dummy record which combines all the
262*4882a593Smuzhiyun * per-cpu records including metadata and ecc info.
263*4882a593Smuzhiyun */
264*4882a593Smuzhiyun struct persistent_ram_zone *tmp_prz, *prz_next;
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun tmp_prz = kzalloc(sizeof(struct persistent_ram_zone),
267*4882a593Smuzhiyun GFP_KERNEL);
268*4882a593Smuzhiyun if (!tmp_prz)
269*4882a593Smuzhiyun return -ENOMEM;
270*4882a593Smuzhiyun prz = tmp_prz;
271*4882a593Smuzhiyun free_prz = true;
272*4882a593Smuzhiyun
273*4882a593Smuzhiyun while (cxt->ftrace_read_cnt < cxt->max_ftrace_cnt) {
274*4882a593Smuzhiyun prz_next = ramoops_get_next_prz(cxt->fprzs,
275*4882a593Smuzhiyun cxt->ftrace_read_cnt++, record);
276*4882a593Smuzhiyun
277*4882a593Smuzhiyun if (!prz_ok(prz_next))
278*4882a593Smuzhiyun continue;
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun tmp_prz->ecc_info = prz_next->ecc_info;
281*4882a593Smuzhiyun tmp_prz->corrected_bytes +=
282*4882a593Smuzhiyun prz_next->corrected_bytes;
283*4882a593Smuzhiyun tmp_prz->bad_blocks += prz_next->bad_blocks;
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun size = pstore_ftrace_combine_log(
286*4882a593Smuzhiyun &tmp_prz->old_log,
287*4882a593Smuzhiyun &tmp_prz->old_log_size,
288*4882a593Smuzhiyun prz_next->old_log,
289*4882a593Smuzhiyun prz_next->old_log_size);
290*4882a593Smuzhiyun if (size)
291*4882a593Smuzhiyun goto out;
292*4882a593Smuzhiyun }
293*4882a593Smuzhiyun record->id = 0;
294*4882a593Smuzhiyun }
295*4882a593Smuzhiyun }
296*4882a593Smuzhiyun
297*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
298*4882a593Smuzhiyun if (!prz_ok(prz)) {
299*4882a593Smuzhiyun while (cxt->boot_log_read_cnt < cxt->max_boot_log_cnt && !prz) {
300*4882a593Smuzhiyun prz = ramoops_get_next_prz(cxt->boot_przs, cxt->boot_log_read_cnt++, record);
301*4882a593Smuzhiyun if (!prz_ok(prz))
302*4882a593Smuzhiyun continue;
303*4882a593Smuzhiyun }
304*4882a593Smuzhiyun }
305*4882a593Smuzhiyun #endif
306*4882a593Smuzhiyun
307*4882a593Smuzhiyun if (!prz_ok(prz)) {
308*4882a593Smuzhiyun size = 0;
309*4882a593Smuzhiyun goto out;
310*4882a593Smuzhiyun }
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
313*4882a593Smuzhiyun if (record->type == PSTORE_TYPE_BOOT_LOG) {
314*4882a593Smuzhiyun persistent_ram_free_old(prz);
315*4882a593Smuzhiyun persistent_ram_save_old(prz);
316*4882a593Smuzhiyun }
317*4882a593Smuzhiyun #endif
318*4882a593Smuzhiyun
319*4882a593Smuzhiyun size = persistent_ram_old_size(prz) - header_length;
320*4882a593Smuzhiyun
321*4882a593Smuzhiyun /* ECC correction notice */
322*4882a593Smuzhiyun record->ecc_notice_size = persistent_ram_ecc_string(prz, NULL, 0);
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun record->buf = kmalloc(size + record->ecc_notice_size + 1, GFP_KERNEL);
325*4882a593Smuzhiyun if (record->buf == NULL) {
326*4882a593Smuzhiyun size = -ENOMEM;
327*4882a593Smuzhiyun goto out;
328*4882a593Smuzhiyun }
329*4882a593Smuzhiyun
330*4882a593Smuzhiyun memcpy(record->buf, (char *)persistent_ram_old(prz) + header_length,
331*4882a593Smuzhiyun size);
332*4882a593Smuzhiyun
333*4882a593Smuzhiyun persistent_ram_ecc_string(prz, record->buf + size,
334*4882a593Smuzhiyun record->ecc_notice_size + 1);
335*4882a593Smuzhiyun
336*4882a593Smuzhiyun out:
337*4882a593Smuzhiyun if (free_prz) {
338*4882a593Smuzhiyun kfree(prz->old_log);
339*4882a593Smuzhiyun kfree(prz);
340*4882a593Smuzhiyun }
341*4882a593Smuzhiyun
342*4882a593Smuzhiyun return size;
343*4882a593Smuzhiyun }
344*4882a593Smuzhiyun
ramoops_write_kmsg_hdr(struct persistent_ram_zone * prz,struct pstore_record * record)345*4882a593Smuzhiyun static size_t ramoops_write_kmsg_hdr(struct persistent_ram_zone *prz,
346*4882a593Smuzhiyun struct pstore_record *record)
347*4882a593Smuzhiyun {
348*4882a593Smuzhiyun char hdr[36]; /* "===="(4), %lld(20), "."(1), %06lu(6), "-%c\n"(3) */
349*4882a593Smuzhiyun size_t len;
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun len = scnprintf(hdr, sizeof(hdr),
352*4882a593Smuzhiyun RAMOOPS_KERNMSG_HDR "%lld.%06lu-%c\n",
353*4882a593Smuzhiyun (time64_t)record->time.tv_sec,
354*4882a593Smuzhiyun record->time.tv_nsec / 1000,
355*4882a593Smuzhiyun record->compressed ? 'C' : 'D');
356*4882a593Smuzhiyun persistent_ram_write(prz, hdr, len);
357*4882a593Smuzhiyun
358*4882a593Smuzhiyun return len;
359*4882a593Smuzhiyun }
360*4882a593Smuzhiyun
ramoops_pstore_write(struct pstore_record * record)361*4882a593Smuzhiyun static int notrace ramoops_pstore_write(struct pstore_record *record)
362*4882a593Smuzhiyun {
363*4882a593Smuzhiyun struct ramoops_context *cxt = record->psi->data;
364*4882a593Smuzhiyun struct persistent_ram_zone *prz;
365*4882a593Smuzhiyun size_t size, hlen;
366*4882a593Smuzhiyun
367*4882a593Smuzhiyun if (record->type == PSTORE_TYPE_CONSOLE) {
368*4882a593Smuzhiyun if (!cxt->cprz)
369*4882a593Smuzhiyun return -ENOMEM;
370*4882a593Smuzhiyun persistent_ram_write(cxt->cprz, record->buf, record->size);
371*4882a593Smuzhiyun return 0;
372*4882a593Smuzhiyun } else if (record->type == PSTORE_TYPE_FTRACE) {
373*4882a593Smuzhiyun int zonenum;
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun if (!cxt->fprzs)
376*4882a593Smuzhiyun return -ENOMEM;
377*4882a593Smuzhiyun /*
378*4882a593Smuzhiyun * Choose zone by if we're using per-cpu buffers.
379*4882a593Smuzhiyun */
380*4882a593Smuzhiyun if (cxt->flags & RAMOOPS_FLAG_FTRACE_PER_CPU)
381*4882a593Smuzhiyun zonenum = smp_processor_id();
382*4882a593Smuzhiyun else
383*4882a593Smuzhiyun zonenum = 0;
384*4882a593Smuzhiyun
385*4882a593Smuzhiyun persistent_ram_write(cxt->fprzs[zonenum], record->buf,
386*4882a593Smuzhiyun record->size);
387*4882a593Smuzhiyun return 0;
388*4882a593Smuzhiyun } else if (record->type == PSTORE_TYPE_PMSG) {
389*4882a593Smuzhiyun pr_warn_ratelimited("PMSG shouldn't call %s\n", __func__);
390*4882a593Smuzhiyun return -EINVAL;
391*4882a593Smuzhiyun }
392*4882a593Smuzhiyun
393*4882a593Smuzhiyun if (record->type != PSTORE_TYPE_DMESG)
394*4882a593Smuzhiyun return -EINVAL;
395*4882a593Smuzhiyun
396*4882a593Smuzhiyun /*
397*4882a593Smuzhiyun * We could filter on record->reason here if we wanted to (which
398*4882a593Smuzhiyun * would duplicate what happened before the "max_reason" setting
399*4882a593Smuzhiyun * was added), but that would defeat the purpose of a system
400*4882a593Smuzhiyun * changing printk.always_kmsg_dump, so instead log everything that
401*4882a593Smuzhiyun * the kmsg dumper sends us, since it should be doing the filtering
402*4882a593Smuzhiyun * based on the combination of printk.always_kmsg_dump and our
403*4882a593Smuzhiyun * requested "max_reason".
404*4882a593Smuzhiyun */
405*4882a593Smuzhiyun
406*4882a593Smuzhiyun /*
407*4882a593Smuzhiyun * Explicitly only take the first part of any new crash.
408*4882a593Smuzhiyun * If our buffer is larger than kmsg_bytes, this can never happen,
409*4882a593Smuzhiyun * and if our buffer is smaller than kmsg_bytes, we don't want the
410*4882a593Smuzhiyun * report split across multiple records.
411*4882a593Smuzhiyun */
412*4882a593Smuzhiyun if (record->part != 1)
413*4882a593Smuzhiyun return -ENOSPC;
414*4882a593Smuzhiyun
415*4882a593Smuzhiyun if (!cxt->dprzs)
416*4882a593Smuzhiyun return -ENOSPC;
417*4882a593Smuzhiyun
418*4882a593Smuzhiyun prz = cxt->dprzs[cxt->dump_write_cnt];
419*4882a593Smuzhiyun
420*4882a593Smuzhiyun /*
421*4882a593Smuzhiyun * Since this is a new crash dump, we need to reset the buffer in
422*4882a593Smuzhiyun * case it still has an old dump present. Without this, the new dump
423*4882a593Smuzhiyun * will get appended, which would seriously confuse anything trying
424*4882a593Smuzhiyun * to check dump file contents. Specifically, ramoops_read_kmsg_hdr()
425*4882a593Smuzhiyun * expects to find a dump header in the beginning of buffer data, so
426*4882a593Smuzhiyun * we must to reset the buffer values, in order to ensure that the
427*4882a593Smuzhiyun * header will be written to the beginning of the buffer.
428*4882a593Smuzhiyun */
429*4882a593Smuzhiyun persistent_ram_zap(prz);
430*4882a593Smuzhiyun
431*4882a593Smuzhiyun /* Build header and append record contents. */
432*4882a593Smuzhiyun hlen = ramoops_write_kmsg_hdr(prz, record);
433*4882a593Smuzhiyun if (!hlen)
434*4882a593Smuzhiyun return -ENOMEM;
435*4882a593Smuzhiyun
436*4882a593Smuzhiyun size = record->size;
437*4882a593Smuzhiyun if (size + hlen > prz->buffer_size)
438*4882a593Smuzhiyun size = prz->buffer_size - hlen;
439*4882a593Smuzhiyun persistent_ram_write(prz, record->buf, size);
440*4882a593Smuzhiyun
441*4882a593Smuzhiyun cxt->dump_write_cnt = (cxt->dump_write_cnt + 1) % cxt->max_dump_cnt;
442*4882a593Smuzhiyun
443*4882a593Smuzhiyun return 0;
444*4882a593Smuzhiyun }
445*4882a593Smuzhiyun
ramoops_pstore_write_user(struct pstore_record * record,const char __user * buf)446*4882a593Smuzhiyun static int notrace ramoops_pstore_write_user(struct pstore_record *record,
447*4882a593Smuzhiyun const char __user *buf)
448*4882a593Smuzhiyun {
449*4882a593Smuzhiyun if (record->type == PSTORE_TYPE_PMSG) {
450*4882a593Smuzhiyun struct ramoops_context *cxt = record->psi->data;
451*4882a593Smuzhiyun
452*4882a593Smuzhiyun if (!cxt->mprz)
453*4882a593Smuzhiyun return -ENOMEM;
454*4882a593Smuzhiyun return persistent_ram_write_user(cxt->mprz, buf, record->size);
455*4882a593Smuzhiyun }
456*4882a593Smuzhiyun
457*4882a593Smuzhiyun return -EINVAL;
458*4882a593Smuzhiyun }
459*4882a593Smuzhiyun
ramoops_pstore_erase(struct pstore_record * record)460*4882a593Smuzhiyun static int ramoops_pstore_erase(struct pstore_record *record)
461*4882a593Smuzhiyun {
462*4882a593Smuzhiyun struct ramoops_context *cxt = record->psi->data;
463*4882a593Smuzhiyun struct persistent_ram_zone *prz;
464*4882a593Smuzhiyun
465*4882a593Smuzhiyun switch (record->type) {
466*4882a593Smuzhiyun case PSTORE_TYPE_DMESG:
467*4882a593Smuzhiyun if (record->id >= cxt->max_dump_cnt)
468*4882a593Smuzhiyun return -EINVAL;
469*4882a593Smuzhiyun prz = cxt->dprzs[record->id];
470*4882a593Smuzhiyun break;
471*4882a593Smuzhiyun case PSTORE_TYPE_CONSOLE:
472*4882a593Smuzhiyun prz = cxt->cprz;
473*4882a593Smuzhiyun break;
474*4882a593Smuzhiyun case PSTORE_TYPE_FTRACE:
475*4882a593Smuzhiyun if (record->id >= cxt->max_ftrace_cnt)
476*4882a593Smuzhiyun return -EINVAL;
477*4882a593Smuzhiyun prz = cxt->fprzs[record->id];
478*4882a593Smuzhiyun break;
479*4882a593Smuzhiyun case PSTORE_TYPE_PMSG:
480*4882a593Smuzhiyun prz = cxt->mprz;
481*4882a593Smuzhiyun break;
482*4882a593Smuzhiyun default:
483*4882a593Smuzhiyun return -EINVAL;
484*4882a593Smuzhiyun }
485*4882a593Smuzhiyun
486*4882a593Smuzhiyun persistent_ram_free_old(prz);
487*4882a593Smuzhiyun persistent_ram_zap(prz);
488*4882a593Smuzhiyun
489*4882a593Smuzhiyun return 0;
490*4882a593Smuzhiyun }
491*4882a593Smuzhiyun
492*4882a593Smuzhiyun static struct ramoops_context oops_cxt = {
493*4882a593Smuzhiyun .pstore = {
494*4882a593Smuzhiyun .owner = THIS_MODULE,
495*4882a593Smuzhiyun .name = "ramoops",
496*4882a593Smuzhiyun .open = ramoops_pstore_open,
497*4882a593Smuzhiyun .read = ramoops_pstore_read,
498*4882a593Smuzhiyun .write = ramoops_pstore_write,
499*4882a593Smuzhiyun .write_user = ramoops_pstore_write_user,
500*4882a593Smuzhiyun .erase = ramoops_pstore_erase,
501*4882a593Smuzhiyun },
502*4882a593Smuzhiyun };
503*4882a593Smuzhiyun
ramoops_free_przs(struct ramoops_context * cxt)504*4882a593Smuzhiyun static void ramoops_free_przs(struct ramoops_context *cxt)
505*4882a593Smuzhiyun {
506*4882a593Smuzhiyun int i;
507*4882a593Smuzhiyun
508*4882a593Smuzhiyun /* Free dump PRZs */
509*4882a593Smuzhiyun if (cxt->dprzs) {
510*4882a593Smuzhiyun for (i = 0; i < cxt->max_dump_cnt; i++)
511*4882a593Smuzhiyun persistent_ram_free(cxt->dprzs[i]);
512*4882a593Smuzhiyun
513*4882a593Smuzhiyun kfree(cxt->dprzs);
514*4882a593Smuzhiyun cxt->max_dump_cnt = 0;
515*4882a593Smuzhiyun }
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun /* Free ftrace PRZs */
518*4882a593Smuzhiyun if (cxt->fprzs) {
519*4882a593Smuzhiyun for (i = 0; i < cxt->max_ftrace_cnt; i++)
520*4882a593Smuzhiyun persistent_ram_free(cxt->fprzs[i]);
521*4882a593Smuzhiyun kfree(cxt->fprzs);
522*4882a593Smuzhiyun cxt->max_ftrace_cnt = 0;
523*4882a593Smuzhiyun }
524*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
525*4882a593Smuzhiyun /* Free boot log PRZs */
526*4882a593Smuzhiyun if (cxt->boot_przs) {
527*4882a593Smuzhiyun for (i = 0; i < cxt->max_boot_log_cnt; i++)
528*4882a593Smuzhiyun persistent_ram_free(cxt->boot_przs[i]);
529*4882a593Smuzhiyun kfree(cxt->boot_przs);
530*4882a593Smuzhiyun cxt->max_boot_log_cnt = 0;
531*4882a593Smuzhiyun }
532*4882a593Smuzhiyun #endif
533*4882a593Smuzhiyun }
534*4882a593Smuzhiyun
ramoops_init_przs(const char * name,struct device * dev,struct ramoops_context * cxt,struct persistent_ram_zone *** przs,phys_addr_t * paddr,size_t mem_sz,ssize_t record_size,unsigned int * cnt,u32 sig,u32 flags)535*4882a593Smuzhiyun static int ramoops_init_przs(const char *name,
536*4882a593Smuzhiyun struct device *dev, struct ramoops_context *cxt,
537*4882a593Smuzhiyun struct persistent_ram_zone ***przs,
538*4882a593Smuzhiyun phys_addr_t *paddr, size_t mem_sz,
539*4882a593Smuzhiyun ssize_t record_size,
540*4882a593Smuzhiyun unsigned int *cnt, u32 sig, u32 flags)
541*4882a593Smuzhiyun {
542*4882a593Smuzhiyun int err = -ENOMEM;
543*4882a593Smuzhiyun int i;
544*4882a593Smuzhiyun size_t zone_sz;
545*4882a593Smuzhiyun struct persistent_ram_zone **prz_ar;
546*4882a593Smuzhiyun
547*4882a593Smuzhiyun /* Allocate nothing for 0 mem_sz or 0 record_size. */
548*4882a593Smuzhiyun if (mem_sz == 0 || record_size == 0) {
549*4882a593Smuzhiyun *cnt = 0;
550*4882a593Smuzhiyun return 0;
551*4882a593Smuzhiyun }
552*4882a593Smuzhiyun
553*4882a593Smuzhiyun /*
554*4882a593Smuzhiyun * If we have a negative record size, calculate it based on
555*4882a593Smuzhiyun * mem_sz / *cnt. If we have a positive record size, calculate
556*4882a593Smuzhiyun * cnt from mem_sz / record_size.
557*4882a593Smuzhiyun */
558*4882a593Smuzhiyun if (record_size < 0) {
559*4882a593Smuzhiyun if (*cnt == 0)
560*4882a593Smuzhiyun return 0;
561*4882a593Smuzhiyun record_size = mem_sz / *cnt;
562*4882a593Smuzhiyun if (record_size == 0) {
563*4882a593Smuzhiyun dev_err(dev, "%s record size == 0 (%zu / %u)\n",
564*4882a593Smuzhiyun name, mem_sz, *cnt);
565*4882a593Smuzhiyun goto fail;
566*4882a593Smuzhiyun }
567*4882a593Smuzhiyun } else {
568*4882a593Smuzhiyun *cnt = mem_sz / record_size;
569*4882a593Smuzhiyun if (*cnt == 0) {
570*4882a593Smuzhiyun dev_err(dev, "%s record count == 0 (%zu / %zu)\n",
571*4882a593Smuzhiyun name, mem_sz, record_size);
572*4882a593Smuzhiyun goto fail;
573*4882a593Smuzhiyun }
574*4882a593Smuzhiyun }
575*4882a593Smuzhiyun
576*4882a593Smuzhiyun if (*paddr + mem_sz - cxt->phys_addr > cxt->size) {
577*4882a593Smuzhiyun dev_err(dev, "no room for %s mem region (0x%zx@0x%llx) in (0x%lx@0x%llx)\n",
578*4882a593Smuzhiyun name,
579*4882a593Smuzhiyun mem_sz, (unsigned long long)*paddr,
580*4882a593Smuzhiyun cxt->size, (unsigned long long)cxt->phys_addr);
581*4882a593Smuzhiyun goto fail;
582*4882a593Smuzhiyun }
583*4882a593Smuzhiyun
584*4882a593Smuzhiyun zone_sz = mem_sz / *cnt;
585*4882a593Smuzhiyun if (!zone_sz) {
586*4882a593Smuzhiyun dev_err(dev, "%s zone size == 0\n", name);
587*4882a593Smuzhiyun goto fail;
588*4882a593Smuzhiyun }
589*4882a593Smuzhiyun
590*4882a593Smuzhiyun prz_ar = kcalloc(*cnt, sizeof(**przs), GFP_KERNEL);
591*4882a593Smuzhiyun if (!prz_ar)
592*4882a593Smuzhiyun goto fail;
593*4882a593Smuzhiyun
594*4882a593Smuzhiyun for (i = 0; i < *cnt; i++) {
595*4882a593Smuzhiyun char *label;
596*4882a593Smuzhiyun
597*4882a593Smuzhiyun if (*cnt == 1)
598*4882a593Smuzhiyun label = kasprintf(GFP_KERNEL, "ramoops:%s", name);
599*4882a593Smuzhiyun else
600*4882a593Smuzhiyun label = kasprintf(GFP_KERNEL, "ramoops:%s(%d/%d)",
601*4882a593Smuzhiyun name, i, *cnt - 1);
602*4882a593Smuzhiyun prz_ar[i] = persistent_ram_new(*paddr, zone_sz, sig,
603*4882a593Smuzhiyun &cxt->ecc_info,
604*4882a593Smuzhiyun cxt->memtype, flags, label);
605*4882a593Smuzhiyun kfree(label);
606*4882a593Smuzhiyun if (IS_ERR(prz_ar[i])) {
607*4882a593Smuzhiyun err = PTR_ERR(prz_ar[i]);
608*4882a593Smuzhiyun dev_err(dev, "failed to request %s mem region (0x%zx@0x%llx): %d\n",
609*4882a593Smuzhiyun name, record_size,
610*4882a593Smuzhiyun (unsigned long long)*paddr, err);
611*4882a593Smuzhiyun
612*4882a593Smuzhiyun while (i > 0) {
613*4882a593Smuzhiyun i--;
614*4882a593Smuzhiyun persistent_ram_free(prz_ar[i]);
615*4882a593Smuzhiyun }
616*4882a593Smuzhiyun kfree(prz_ar);
617*4882a593Smuzhiyun goto fail;
618*4882a593Smuzhiyun }
619*4882a593Smuzhiyun *paddr += zone_sz;
620*4882a593Smuzhiyun prz_ar[i]->type = pstore_name_to_type(name);
621*4882a593Smuzhiyun }
622*4882a593Smuzhiyun
623*4882a593Smuzhiyun *przs = prz_ar;
624*4882a593Smuzhiyun return 0;
625*4882a593Smuzhiyun
626*4882a593Smuzhiyun fail:
627*4882a593Smuzhiyun *cnt = 0;
628*4882a593Smuzhiyun return err;
629*4882a593Smuzhiyun }
630*4882a593Smuzhiyun
ramoops_init_prz(const char * name,struct device * dev,struct ramoops_context * cxt,struct persistent_ram_zone ** prz,phys_addr_t * paddr,size_t sz,u32 sig)631*4882a593Smuzhiyun static int ramoops_init_prz(const char *name,
632*4882a593Smuzhiyun struct device *dev, struct ramoops_context *cxt,
633*4882a593Smuzhiyun struct persistent_ram_zone **prz,
634*4882a593Smuzhiyun phys_addr_t *paddr, size_t sz, u32 sig)
635*4882a593Smuzhiyun {
636*4882a593Smuzhiyun char *label;
637*4882a593Smuzhiyun
638*4882a593Smuzhiyun if (!sz)
639*4882a593Smuzhiyun return 0;
640*4882a593Smuzhiyun
641*4882a593Smuzhiyun if (*paddr + sz - cxt->phys_addr > cxt->size) {
642*4882a593Smuzhiyun dev_err(dev, "no room for %s mem region (0x%zx@0x%llx) in (0x%lx@0x%llx)\n",
643*4882a593Smuzhiyun name, sz, (unsigned long long)*paddr,
644*4882a593Smuzhiyun cxt->size, (unsigned long long)cxt->phys_addr);
645*4882a593Smuzhiyun return -ENOMEM;
646*4882a593Smuzhiyun }
647*4882a593Smuzhiyun
648*4882a593Smuzhiyun label = kasprintf(GFP_KERNEL, "ramoops:%s", name);
649*4882a593Smuzhiyun *prz = persistent_ram_new(*paddr, sz, sig, &cxt->ecc_info,
650*4882a593Smuzhiyun cxt->memtype, PRZ_FLAG_ZAP_OLD, label);
651*4882a593Smuzhiyun kfree(label);
652*4882a593Smuzhiyun if (IS_ERR(*prz)) {
653*4882a593Smuzhiyun int err = PTR_ERR(*prz);
654*4882a593Smuzhiyun
655*4882a593Smuzhiyun dev_err(dev, "failed to request %s mem region (0x%zx@0x%llx): %d\n",
656*4882a593Smuzhiyun name, sz, (unsigned long long)*paddr, err);
657*4882a593Smuzhiyun return err;
658*4882a593Smuzhiyun }
659*4882a593Smuzhiyun
660*4882a593Smuzhiyun *paddr += sz;
661*4882a593Smuzhiyun (*prz)->type = pstore_name_to_type(name);
662*4882a593Smuzhiyun
663*4882a593Smuzhiyun return 0;
664*4882a593Smuzhiyun }
665*4882a593Smuzhiyun
666*4882a593Smuzhiyun /* Read a u32 from a dt property and make sure it's safe for an int. */
ramoops_parse_dt_u32(struct platform_device * pdev,const char * propname,u32 default_value,u32 * value)667*4882a593Smuzhiyun static int ramoops_parse_dt_u32(struct platform_device *pdev,
668*4882a593Smuzhiyun const char *propname,
669*4882a593Smuzhiyun u32 default_value, u32 *value)
670*4882a593Smuzhiyun {
671*4882a593Smuzhiyun u32 val32 = 0;
672*4882a593Smuzhiyun int ret;
673*4882a593Smuzhiyun
674*4882a593Smuzhiyun ret = of_property_read_u32(pdev->dev.of_node, propname, &val32);
675*4882a593Smuzhiyun if (ret == -EINVAL) {
676*4882a593Smuzhiyun /* field is missing, use default value. */
677*4882a593Smuzhiyun val32 = default_value;
678*4882a593Smuzhiyun } else if (ret < 0) {
679*4882a593Smuzhiyun dev_err(&pdev->dev, "failed to parse property %s: %d\n",
680*4882a593Smuzhiyun propname, ret);
681*4882a593Smuzhiyun return ret;
682*4882a593Smuzhiyun }
683*4882a593Smuzhiyun
684*4882a593Smuzhiyun /* Sanity check our results. */
685*4882a593Smuzhiyun if (val32 > INT_MAX) {
686*4882a593Smuzhiyun dev_err(&pdev->dev, "%s %u > INT_MAX\n", propname, val32);
687*4882a593Smuzhiyun return -EOVERFLOW;
688*4882a593Smuzhiyun }
689*4882a593Smuzhiyun
690*4882a593Smuzhiyun *value = val32;
691*4882a593Smuzhiyun return 0;
692*4882a593Smuzhiyun }
693*4882a593Smuzhiyun
ramoops_parse_dt(struct platform_device * pdev,struct ramoops_platform_data * pdata)694*4882a593Smuzhiyun static int ramoops_parse_dt(struct platform_device *pdev,
695*4882a593Smuzhiyun struct ramoops_platform_data *pdata)
696*4882a593Smuzhiyun {
697*4882a593Smuzhiyun struct device_node *of_node = pdev->dev.of_node;
698*4882a593Smuzhiyun struct device_node *parent_node;
699*4882a593Smuzhiyun struct reserved_mem *rmem;
700*4882a593Smuzhiyun struct resource *res;
701*4882a593Smuzhiyun u32 value;
702*4882a593Smuzhiyun int ret;
703*4882a593Smuzhiyun
704*4882a593Smuzhiyun dev_dbg(&pdev->dev, "using Device Tree\n");
705*4882a593Smuzhiyun
706*4882a593Smuzhiyun res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
707*4882a593Smuzhiyun if (!res) {
708*4882a593Smuzhiyun rmem = of_reserved_mem_lookup(of_node);
709*4882a593Smuzhiyun if (rmem) {
710*4882a593Smuzhiyun pdata->mem_size = rmem->size;
711*4882a593Smuzhiyun pdata->mem_address = rmem->base;
712*4882a593Smuzhiyun } else {
713*4882a593Smuzhiyun dev_err(&pdev->dev,
714*4882a593Smuzhiyun "failed to locate DT /reserved-memory resource\n");
715*4882a593Smuzhiyun return -EINVAL;
716*4882a593Smuzhiyun }
717*4882a593Smuzhiyun } else {
718*4882a593Smuzhiyun pdata->mem_size = resource_size(res);
719*4882a593Smuzhiyun pdata->mem_address = res->start;
720*4882a593Smuzhiyun }
721*4882a593Smuzhiyun
722*4882a593Smuzhiyun /*
723*4882a593Smuzhiyun * Setting "unbuffered" is deprecated and will be ignored if
724*4882a593Smuzhiyun * "mem_type" is also specified.
725*4882a593Smuzhiyun */
726*4882a593Smuzhiyun pdata->mem_type = of_property_read_bool(of_node, "unbuffered");
727*4882a593Smuzhiyun /*
728*4882a593Smuzhiyun * Setting "no-dump-oops" is deprecated and will be ignored if
729*4882a593Smuzhiyun * "max_reason" is also specified.
730*4882a593Smuzhiyun */
731*4882a593Smuzhiyun if (of_property_read_bool(of_node, "no-dump-oops"))
732*4882a593Smuzhiyun pdata->max_reason = KMSG_DUMP_PANIC;
733*4882a593Smuzhiyun else
734*4882a593Smuzhiyun pdata->max_reason = KMSG_DUMP_OOPS;
735*4882a593Smuzhiyun
736*4882a593Smuzhiyun #define parse_u32(name, field, default_value) { \
737*4882a593Smuzhiyun ret = ramoops_parse_dt_u32(pdev, name, default_value, \
738*4882a593Smuzhiyun &value); \
739*4882a593Smuzhiyun if (ret < 0) \
740*4882a593Smuzhiyun return ret; \
741*4882a593Smuzhiyun field = value; \
742*4882a593Smuzhiyun }
743*4882a593Smuzhiyun
744*4882a593Smuzhiyun parse_u32("mem-type", pdata->mem_type, pdata->mem_type);
745*4882a593Smuzhiyun parse_u32("record-size", pdata->record_size, 0);
746*4882a593Smuzhiyun parse_u32("console-size", pdata->console_size, 0);
747*4882a593Smuzhiyun parse_u32("ftrace-size", pdata->ftrace_size, 0);
748*4882a593Smuzhiyun parse_u32("pmsg-size", pdata->pmsg_size, 0);
749*4882a593Smuzhiyun parse_u32("ecc-size", pdata->ecc_info.ecc_size, 0);
750*4882a593Smuzhiyun parse_u32("flags", pdata->flags, 0);
751*4882a593Smuzhiyun parse_u32("max-reason", pdata->max_reason, pdata->max_reason);
752*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
753*4882a593Smuzhiyun parse_u32("boot-log-size", pdata->boot_log_size, 0);
754*4882a593Smuzhiyun parse_u32("boot-log-count", pdata->max_boot_log_cnt, 0);
755*4882a593Smuzhiyun #endif
756*4882a593Smuzhiyun
757*4882a593Smuzhiyun #undef parse_u32
758*4882a593Smuzhiyun
759*4882a593Smuzhiyun /*
760*4882a593Smuzhiyun * Some old Chromebooks relied on the kernel setting the
761*4882a593Smuzhiyun * console_size and pmsg_size to the record size since that's
762*4882a593Smuzhiyun * what the downstream kernel did. These same Chromebooks had
763*4882a593Smuzhiyun * "ramoops" straight under the root node which isn't
764*4882a593Smuzhiyun * according to the current upstream bindings (though it was
765*4882a593Smuzhiyun * arguably acceptable under a prior version of the bindings).
766*4882a593Smuzhiyun * Let's make those old Chromebooks work by detecting that
767*4882a593Smuzhiyun * we're not a child of "reserved-memory" and mimicking the
768*4882a593Smuzhiyun * expected behavior.
769*4882a593Smuzhiyun */
770*4882a593Smuzhiyun parent_node = of_get_parent(of_node);
771*4882a593Smuzhiyun if (!of_node_name_eq(parent_node, "reserved-memory") &&
772*4882a593Smuzhiyun !pdata->console_size && !pdata->ftrace_size &&
773*4882a593Smuzhiyun !pdata->pmsg_size && !pdata->ecc_info.ecc_size) {
774*4882a593Smuzhiyun pdata->console_size = pdata->record_size;
775*4882a593Smuzhiyun pdata->pmsg_size = pdata->record_size;
776*4882a593Smuzhiyun }
777*4882a593Smuzhiyun of_node_put(parent_node);
778*4882a593Smuzhiyun
779*4882a593Smuzhiyun return 0;
780*4882a593Smuzhiyun }
781*4882a593Smuzhiyun
782*4882a593Smuzhiyun #if IS_REACHABLE(CONFIG_ROCKCHIP_MINIDUMP)
_ramoops_register_ram_zone_info_to_minidump(struct persistent_ram_zone * prz)783*4882a593Smuzhiyun static void _ramoops_register_ram_zone_info_to_minidump(struct persistent_ram_zone *prz)
784*4882a593Smuzhiyun {
785*4882a593Smuzhiyun struct md_region md_entry = {};
786*4882a593Smuzhiyun
787*4882a593Smuzhiyun strscpy(md_entry.name, prz->label, sizeof(md_entry.name));
788*4882a593Smuzhiyun
789*4882a593Smuzhiyun md_entry.virt_addr = (u64)prz->vaddr;
790*4882a593Smuzhiyun md_entry.phys_addr = prz->paddr;
791*4882a593Smuzhiyun md_entry.size = prz->size;
792*4882a593Smuzhiyun
793*4882a593Smuzhiyun if (rk_minidump_add_region(&md_entry) < 0)
794*4882a593Smuzhiyun pr_err("Failed to add %s in Minidump\n", prz->label);
795*4882a593Smuzhiyun }
796*4882a593Smuzhiyun
ramoops_register_ram_zone_info_to_minidump(struct ramoops_context * cxt)797*4882a593Smuzhiyun static void ramoops_register_ram_zone_info_to_minidump(struct ramoops_context *cxt)
798*4882a593Smuzhiyun {
799*4882a593Smuzhiyun int i = 0;
800*4882a593Smuzhiyun struct persistent_ram_zone *prz = NULL;
801*4882a593Smuzhiyun
802*4882a593Smuzhiyun for (i = 0; i < cxt->max_boot_log_cnt; i++) {
803*4882a593Smuzhiyun prz = cxt->boot_przs[i];
804*4882a593Smuzhiyun _ramoops_register_ram_zone_info_to_minidump(prz);
805*4882a593Smuzhiyun }
806*4882a593Smuzhiyun
807*4882a593Smuzhiyun for (i = 0; i < cxt->max_dump_cnt; i++) {
808*4882a593Smuzhiyun prz = cxt->dprzs[i];
809*4882a593Smuzhiyun _ramoops_register_ram_zone_info_to_minidump(prz);
810*4882a593Smuzhiyun }
811*4882a593Smuzhiyun
812*4882a593Smuzhiyun for (i = 0; i < cxt->max_ftrace_cnt; i++) {
813*4882a593Smuzhiyun prz = cxt->fprzs[i];
814*4882a593Smuzhiyun _ramoops_register_ram_zone_info_to_minidump(prz);
815*4882a593Smuzhiyun }
816*4882a593Smuzhiyun
817*4882a593Smuzhiyun prz = cxt->cprz;
818*4882a593Smuzhiyun _ramoops_register_ram_zone_info_to_minidump(prz);
819*4882a593Smuzhiyun
820*4882a593Smuzhiyun prz = cxt->mprz;
821*4882a593Smuzhiyun _ramoops_register_ram_zone_info_to_minidump(prz);
822*4882a593Smuzhiyun }
823*4882a593Smuzhiyun #endif
824*4882a593Smuzhiyun
ramoops_probe(struct platform_device * pdev)825*4882a593Smuzhiyun static int ramoops_probe(struct platform_device *pdev)
826*4882a593Smuzhiyun {
827*4882a593Smuzhiyun struct device *dev = &pdev->dev;
828*4882a593Smuzhiyun struct ramoops_platform_data *pdata = dev->platform_data;
829*4882a593Smuzhiyun struct ramoops_platform_data pdata_local;
830*4882a593Smuzhiyun struct ramoops_context *cxt = &oops_cxt;
831*4882a593Smuzhiyun size_t dump_mem_sz;
832*4882a593Smuzhiyun phys_addr_t paddr;
833*4882a593Smuzhiyun int err = -EINVAL;
834*4882a593Smuzhiyun int i = 0;
835*4882a593Smuzhiyun
836*4882a593Smuzhiyun /*
837*4882a593Smuzhiyun * Only a single ramoops area allowed at a time, so fail extra
838*4882a593Smuzhiyun * probes.
839*4882a593Smuzhiyun */
840*4882a593Smuzhiyun if (cxt->max_dump_cnt) {
841*4882a593Smuzhiyun pr_err("already initialized\n");
842*4882a593Smuzhiyun goto fail_out;
843*4882a593Smuzhiyun }
844*4882a593Smuzhiyun
845*4882a593Smuzhiyun if (dev_of_node(dev) && !pdata) {
846*4882a593Smuzhiyun pdata = &pdata_local;
847*4882a593Smuzhiyun memset(pdata, 0, sizeof(*pdata));
848*4882a593Smuzhiyun
849*4882a593Smuzhiyun err = ramoops_parse_dt(pdev, pdata);
850*4882a593Smuzhiyun if (err < 0)
851*4882a593Smuzhiyun goto fail_out;
852*4882a593Smuzhiyun }
853*4882a593Smuzhiyun
854*4882a593Smuzhiyun /* Make sure we didn't get bogus platform data pointer. */
855*4882a593Smuzhiyun if (!pdata) {
856*4882a593Smuzhiyun pr_err("NULL platform data\n");
857*4882a593Smuzhiyun goto fail_out;
858*4882a593Smuzhiyun }
859*4882a593Smuzhiyun
860*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
861*4882a593Smuzhiyun if (!pdata->mem_size || (!pdata->record_size && !pdata->console_size &&
862*4882a593Smuzhiyun !pdata->ftrace_size && !pdata->pmsg_size && !pdata->boot_log_size)) {
863*4882a593Smuzhiyun pr_err("The memory size and the record/console size must be "
864*4882a593Smuzhiyun "non-zero\n");
865*4882a593Smuzhiyun goto fail_out;
866*4882a593Smuzhiyun }
867*4882a593Smuzhiyun #else
868*4882a593Smuzhiyun if (!pdata->mem_size || (!pdata->record_size && !pdata->console_size &&
869*4882a593Smuzhiyun !pdata->ftrace_size && !pdata->pmsg_size)) {
870*4882a593Smuzhiyun pr_err("The memory size and the record/console size must be "
871*4882a593Smuzhiyun "non-zero\n");
872*4882a593Smuzhiyun goto fail_out;
873*4882a593Smuzhiyun }
874*4882a593Smuzhiyun #endif
875*4882a593Smuzhiyun
876*4882a593Smuzhiyun #ifndef CONFIG_ARCH_ROCKCHIP
877*4882a593Smuzhiyun if (pdata->record_size && !is_power_of_2(pdata->record_size))
878*4882a593Smuzhiyun pdata->record_size = rounddown_pow_of_two(pdata->record_size);
879*4882a593Smuzhiyun if (pdata->console_size && !is_power_of_2(pdata->console_size))
880*4882a593Smuzhiyun pdata->console_size = rounddown_pow_of_two(pdata->console_size);
881*4882a593Smuzhiyun if (pdata->ftrace_size && !is_power_of_2(pdata->ftrace_size))
882*4882a593Smuzhiyun pdata->ftrace_size = rounddown_pow_of_two(pdata->ftrace_size);
883*4882a593Smuzhiyun if (pdata->pmsg_size && !is_power_of_2(pdata->pmsg_size))
884*4882a593Smuzhiyun pdata->pmsg_size = rounddown_pow_of_two(pdata->pmsg_size);
885*4882a593Smuzhiyun #endif
886*4882a593Smuzhiyun
887*4882a593Smuzhiyun cxt->size = pdata->mem_size;
888*4882a593Smuzhiyun cxt->phys_addr = pdata->mem_address;
889*4882a593Smuzhiyun cxt->memtype = pdata->mem_type;
890*4882a593Smuzhiyun cxt->record_size = pdata->record_size;
891*4882a593Smuzhiyun cxt->console_size = pdata->console_size;
892*4882a593Smuzhiyun cxt->ftrace_size = pdata->ftrace_size;
893*4882a593Smuzhiyun cxt->pmsg_size = pdata->pmsg_size;
894*4882a593Smuzhiyun cxt->flags = pdata->flags;
895*4882a593Smuzhiyun cxt->ecc_info = pdata->ecc_info;
896*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
897*4882a593Smuzhiyun cxt->boot_log_size = pdata->boot_log_size;
898*4882a593Smuzhiyun cxt->max_boot_log_cnt = pdata->max_boot_log_cnt;
899*4882a593Smuzhiyun #endif
900*4882a593Smuzhiyun
901*4882a593Smuzhiyun paddr = cxt->phys_addr;
902*4882a593Smuzhiyun
903*4882a593Smuzhiyun dump_mem_sz = cxt->size - cxt->console_size - cxt->ftrace_size
904*4882a593Smuzhiyun - cxt->pmsg_size;
905*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
906*4882a593Smuzhiyun dump_mem_sz -= cxt->boot_log_size;
907*4882a593Smuzhiyun #endif
908*4882a593Smuzhiyun
909*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
910*4882a593Smuzhiyun err = ramoops_init_przs("boot-log", dev, cxt, &cxt->boot_przs, &paddr,
911*4882a593Smuzhiyun cxt->boot_log_size, -1,
912*4882a593Smuzhiyun &cxt->max_boot_log_cnt, 0, 0);
913*4882a593Smuzhiyun if (err)
914*4882a593Smuzhiyun goto fail_clear;
915*4882a593Smuzhiyun if (cxt->boot_log_size > 0)
916*4882a593Smuzhiyun for (i = 0; i < cxt->max_boot_log_cnt; i++)
917*4882a593Smuzhiyun pr_info("boot-log-%d\t0x%zx@%pa\n", i, cxt->boot_przs[i]->size, &cxt->boot_przs[i]->paddr);
918*4882a593Smuzhiyun #endif
919*4882a593Smuzhiyun
920*4882a593Smuzhiyun err = ramoops_init_przs("dmesg", dev, cxt, &cxt->dprzs, &paddr,
921*4882a593Smuzhiyun dump_mem_sz, cxt->record_size,
922*4882a593Smuzhiyun &cxt->max_dump_cnt, 0, 0);
923*4882a593Smuzhiyun if (err)
924*4882a593Smuzhiyun goto fail_out;
925*4882a593Smuzhiyun if (cxt->record_size > 0)
926*4882a593Smuzhiyun for (i = 0; i < cxt->max_dump_cnt; i++)
927*4882a593Smuzhiyun pr_info("dmesg-%d\t0x%zx@%pa\n", i, cxt->dprzs[i]->size, &cxt->dprzs[i]->paddr);
928*4882a593Smuzhiyun
929*4882a593Smuzhiyun err = ramoops_init_prz("console", dev, cxt, &cxt->cprz, &paddr,
930*4882a593Smuzhiyun cxt->console_size, 0);
931*4882a593Smuzhiyun if (err)
932*4882a593Smuzhiyun goto fail_init_cprz;
933*4882a593Smuzhiyun if (cxt->console_size > 0)
934*4882a593Smuzhiyun pr_info("console\t0x%zx@%pa\n", cxt->cprz->size, &cxt->cprz->paddr);
935*4882a593Smuzhiyun
936*4882a593Smuzhiyun cxt->max_ftrace_cnt = (cxt->flags & RAMOOPS_FLAG_FTRACE_PER_CPU)
937*4882a593Smuzhiyun ? nr_cpu_ids
938*4882a593Smuzhiyun : 1;
939*4882a593Smuzhiyun err = ramoops_init_przs("ftrace", dev, cxt, &cxt->fprzs, &paddr,
940*4882a593Smuzhiyun cxt->ftrace_size, -1,
941*4882a593Smuzhiyun &cxt->max_ftrace_cnt, LINUX_VERSION_CODE,
942*4882a593Smuzhiyun (cxt->flags & RAMOOPS_FLAG_FTRACE_PER_CPU)
943*4882a593Smuzhiyun ? PRZ_FLAG_NO_LOCK : 0);
944*4882a593Smuzhiyun if (err)
945*4882a593Smuzhiyun goto fail_init_fprz;
946*4882a593Smuzhiyun if (cxt->ftrace_size > 0)
947*4882a593Smuzhiyun for (i = 0; i < cxt->max_ftrace_cnt; i++)
948*4882a593Smuzhiyun pr_info("ftrace-%d\t0x%zx@%pa\n", i, cxt->fprzs[i]->size, &cxt->fprzs[i]->paddr);
949*4882a593Smuzhiyun
950*4882a593Smuzhiyun err = ramoops_init_prz("pmsg", dev, cxt, &cxt->mprz, &paddr,
951*4882a593Smuzhiyun cxt->pmsg_size, 0);
952*4882a593Smuzhiyun if (err)
953*4882a593Smuzhiyun goto fail_init_mprz;
954*4882a593Smuzhiyun if (cxt->pmsg_size > 0)
955*4882a593Smuzhiyun pr_info("pmsg\t0x%zx@%pa\n", cxt->mprz->size, &cxt->mprz->paddr);
956*4882a593Smuzhiyun
957*4882a593Smuzhiyun cxt->pstore.data = cxt;
958*4882a593Smuzhiyun /*
959*4882a593Smuzhiyun * Prepare frontend flags based on which areas are initialized.
960*4882a593Smuzhiyun * For ramoops_init_przs() cases, the "max count" variable tells
961*4882a593Smuzhiyun * if there are regions present. For ramoops_init_prz() cases,
962*4882a593Smuzhiyun * the single region size is how to check.
963*4882a593Smuzhiyun */
964*4882a593Smuzhiyun cxt->pstore.flags = 0;
965*4882a593Smuzhiyun if (cxt->max_dump_cnt) {
966*4882a593Smuzhiyun cxt->pstore.flags |= PSTORE_FLAGS_DMESG;
967*4882a593Smuzhiyun cxt->pstore.max_reason = pdata->max_reason;
968*4882a593Smuzhiyun }
969*4882a593Smuzhiyun if (cxt->console_size)
970*4882a593Smuzhiyun cxt->pstore.flags |= PSTORE_FLAGS_CONSOLE;
971*4882a593Smuzhiyun if (cxt->max_ftrace_cnt)
972*4882a593Smuzhiyun cxt->pstore.flags |= PSTORE_FLAGS_FTRACE;
973*4882a593Smuzhiyun if (cxt->pmsg_size)
974*4882a593Smuzhiyun cxt->pstore.flags |= PSTORE_FLAGS_PMSG;
975*4882a593Smuzhiyun #ifdef CONFIG_PSTORE_BOOT_LOG
976*4882a593Smuzhiyun if (cxt->boot_log_size)
977*4882a593Smuzhiyun cxt->pstore.flags |= PSTORE_FLAGS_BOOT_LOG;
978*4882a593Smuzhiyun #endif
979*4882a593Smuzhiyun
980*4882a593Smuzhiyun /*
981*4882a593Smuzhiyun * Since bufsize is only used for dmesg crash dumps, it
982*4882a593Smuzhiyun * must match the size of the dprz record (after PRZ header
983*4882a593Smuzhiyun * and ECC bytes have been accounted for).
984*4882a593Smuzhiyun */
985*4882a593Smuzhiyun if (cxt->pstore.flags & PSTORE_FLAGS_DMESG) {
986*4882a593Smuzhiyun cxt->pstore.bufsize = cxt->dprzs[0]->buffer_size;
987*4882a593Smuzhiyun cxt->pstore.buf = kzalloc(cxt->pstore.bufsize, GFP_KERNEL);
988*4882a593Smuzhiyun if (!cxt->pstore.buf) {
989*4882a593Smuzhiyun pr_err("cannot allocate pstore crash dump buffer\n");
990*4882a593Smuzhiyun err = -ENOMEM;
991*4882a593Smuzhiyun goto fail_clear;
992*4882a593Smuzhiyun }
993*4882a593Smuzhiyun }
994*4882a593Smuzhiyun
995*4882a593Smuzhiyun err = pstore_register(&cxt->pstore);
996*4882a593Smuzhiyun if (err) {
997*4882a593Smuzhiyun pr_err("registering with pstore failed\n");
998*4882a593Smuzhiyun goto fail_buf;
999*4882a593Smuzhiyun }
1000*4882a593Smuzhiyun
1001*4882a593Smuzhiyun /*
1002*4882a593Smuzhiyun * Update the module parameter variables as well so they are visible
1003*4882a593Smuzhiyun * through /sys/module/ramoops/parameters/
1004*4882a593Smuzhiyun */
1005*4882a593Smuzhiyun mem_size = pdata->mem_size;
1006*4882a593Smuzhiyun mem_address = pdata->mem_address;
1007*4882a593Smuzhiyun record_size = pdata->record_size;
1008*4882a593Smuzhiyun ramoops_max_reason = pdata->max_reason;
1009*4882a593Smuzhiyun ramoops_console_size = pdata->console_size;
1010*4882a593Smuzhiyun ramoops_pmsg_size = pdata->pmsg_size;
1011*4882a593Smuzhiyun ramoops_ftrace_size = pdata->ftrace_size;
1012*4882a593Smuzhiyun #if IS_REACHABLE(CONFIG_ROCKCHIP_MINIDUMP)
1013*4882a593Smuzhiyun ramoops_register_ram_zone_info_to_minidump(cxt);
1014*4882a593Smuzhiyun #endif
1015*4882a593Smuzhiyun pr_info("using 0x%lx@0x%llx, ecc: %d\n",
1016*4882a593Smuzhiyun cxt->size, (unsigned long long)cxt->phys_addr,
1017*4882a593Smuzhiyun cxt->ecc_info.ecc_size);
1018*4882a593Smuzhiyun
1019*4882a593Smuzhiyun return 0;
1020*4882a593Smuzhiyun
1021*4882a593Smuzhiyun fail_buf:
1022*4882a593Smuzhiyun kfree(cxt->pstore.buf);
1023*4882a593Smuzhiyun fail_clear:
1024*4882a593Smuzhiyun cxt->pstore.bufsize = 0;
1025*4882a593Smuzhiyun persistent_ram_free(cxt->mprz);
1026*4882a593Smuzhiyun fail_init_mprz:
1027*4882a593Smuzhiyun fail_init_fprz:
1028*4882a593Smuzhiyun persistent_ram_free(cxt->cprz);
1029*4882a593Smuzhiyun fail_init_cprz:
1030*4882a593Smuzhiyun ramoops_free_przs(cxt);
1031*4882a593Smuzhiyun fail_out:
1032*4882a593Smuzhiyun return err;
1033*4882a593Smuzhiyun }
1034*4882a593Smuzhiyun
ramoops_remove(struct platform_device * pdev)1035*4882a593Smuzhiyun static int ramoops_remove(struct platform_device *pdev)
1036*4882a593Smuzhiyun {
1037*4882a593Smuzhiyun struct ramoops_context *cxt = &oops_cxt;
1038*4882a593Smuzhiyun
1039*4882a593Smuzhiyun pstore_unregister(&cxt->pstore);
1040*4882a593Smuzhiyun
1041*4882a593Smuzhiyun kfree(cxt->pstore.buf);
1042*4882a593Smuzhiyun cxt->pstore.bufsize = 0;
1043*4882a593Smuzhiyun
1044*4882a593Smuzhiyun persistent_ram_free(cxt->mprz);
1045*4882a593Smuzhiyun persistent_ram_free(cxt->cprz);
1046*4882a593Smuzhiyun ramoops_free_przs(cxt);
1047*4882a593Smuzhiyun
1048*4882a593Smuzhiyun return 0;
1049*4882a593Smuzhiyun }
1050*4882a593Smuzhiyun
1051*4882a593Smuzhiyun static const struct of_device_id dt_match[] = {
1052*4882a593Smuzhiyun { .compatible = "ramoops" },
1053*4882a593Smuzhiyun {}
1054*4882a593Smuzhiyun };
1055*4882a593Smuzhiyun
1056*4882a593Smuzhiyun static struct platform_driver ramoops_driver = {
1057*4882a593Smuzhiyun .probe = ramoops_probe,
1058*4882a593Smuzhiyun .remove = ramoops_remove,
1059*4882a593Smuzhiyun .driver = {
1060*4882a593Smuzhiyun .name = "ramoops",
1061*4882a593Smuzhiyun .of_match_table = dt_match,
1062*4882a593Smuzhiyun },
1063*4882a593Smuzhiyun };
1064*4882a593Smuzhiyun
ramoops_unregister_dummy(void)1065*4882a593Smuzhiyun static inline void ramoops_unregister_dummy(void)
1066*4882a593Smuzhiyun {
1067*4882a593Smuzhiyun platform_device_unregister(dummy);
1068*4882a593Smuzhiyun dummy = NULL;
1069*4882a593Smuzhiyun }
1070*4882a593Smuzhiyun
ramoops_register_dummy(void)1071*4882a593Smuzhiyun static void __init ramoops_register_dummy(void)
1072*4882a593Smuzhiyun {
1073*4882a593Smuzhiyun struct ramoops_platform_data pdata;
1074*4882a593Smuzhiyun
1075*4882a593Smuzhiyun /*
1076*4882a593Smuzhiyun * Prepare a dummy platform data structure to carry the module
1077*4882a593Smuzhiyun * parameters. If mem_size isn't set, then there are no module
1078*4882a593Smuzhiyun * parameters, and we can skip this.
1079*4882a593Smuzhiyun */
1080*4882a593Smuzhiyun if (!mem_size)
1081*4882a593Smuzhiyun return;
1082*4882a593Smuzhiyun
1083*4882a593Smuzhiyun pr_info("using module parameters\n");
1084*4882a593Smuzhiyun
1085*4882a593Smuzhiyun memset(&pdata, 0, sizeof(pdata));
1086*4882a593Smuzhiyun pdata.mem_size = mem_size;
1087*4882a593Smuzhiyun pdata.mem_address = mem_address;
1088*4882a593Smuzhiyun pdata.mem_type = mem_type;
1089*4882a593Smuzhiyun pdata.record_size = record_size;
1090*4882a593Smuzhiyun pdata.console_size = ramoops_console_size;
1091*4882a593Smuzhiyun pdata.ftrace_size = ramoops_ftrace_size;
1092*4882a593Smuzhiyun pdata.pmsg_size = ramoops_pmsg_size;
1093*4882a593Smuzhiyun /* If "max_reason" is set, its value has priority over "dump_oops". */
1094*4882a593Smuzhiyun if (ramoops_max_reason >= 0)
1095*4882a593Smuzhiyun pdata.max_reason = ramoops_max_reason;
1096*4882a593Smuzhiyun /* Otherwise, if "dump_oops" is set, parse it into "max_reason". */
1097*4882a593Smuzhiyun else if (ramoops_dump_oops != -1)
1098*4882a593Smuzhiyun pdata.max_reason = ramoops_dump_oops ? KMSG_DUMP_OOPS
1099*4882a593Smuzhiyun : KMSG_DUMP_PANIC;
1100*4882a593Smuzhiyun /* And if neither are explicitly set, use the default. */
1101*4882a593Smuzhiyun else
1102*4882a593Smuzhiyun pdata.max_reason = KMSG_DUMP_OOPS;
1103*4882a593Smuzhiyun pdata.flags = RAMOOPS_FLAG_FTRACE_PER_CPU;
1104*4882a593Smuzhiyun
1105*4882a593Smuzhiyun /*
1106*4882a593Smuzhiyun * For backwards compatibility ramoops.ecc=1 means 16 bytes ECC
1107*4882a593Smuzhiyun * (using 1 byte for ECC isn't much of use anyway).
1108*4882a593Smuzhiyun */
1109*4882a593Smuzhiyun pdata.ecc_info.ecc_size = ramoops_ecc == 1 ? 16 : ramoops_ecc;
1110*4882a593Smuzhiyun
1111*4882a593Smuzhiyun dummy = platform_device_register_data(NULL, "ramoops", -1,
1112*4882a593Smuzhiyun &pdata, sizeof(pdata));
1113*4882a593Smuzhiyun if (IS_ERR(dummy)) {
1114*4882a593Smuzhiyun pr_info("could not create platform device: %ld\n",
1115*4882a593Smuzhiyun PTR_ERR(dummy));
1116*4882a593Smuzhiyun dummy = NULL;
1117*4882a593Smuzhiyun }
1118*4882a593Smuzhiyun }
1119*4882a593Smuzhiyun
ramoops_init(void)1120*4882a593Smuzhiyun static int __init ramoops_init(void)
1121*4882a593Smuzhiyun {
1122*4882a593Smuzhiyun int ret;
1123*4882a593Smuzhiyun
1124*4882a593Smuzhiyun ramoops_register_dummy();
1125*4882a593Smuzhiyun ret = platform_driver_register(&ramoops_driver);
1126*4882a593Smuzhiyun if (ret != 0)
1127*4882a593Smuzhiyun ramoops_unregister_dummy();
1128*4882a593Smuzhiyun
1129*4882a593Smuzhiyun return ret;
1130*4882a593Smuzhiyun }
1131*4882a593Smuzhiyun postcore_initcall(ramoops_init);
1132*4882a593Smuzhiyun
ramoops_exit(void)1133*4882a593Smuzhiyun static void __exit ramoops_exit(void)
1134*4882a593Smuzhiyun {
1135*4882a593Smuzhiyun platform_driver_unregister(&ramoops_driver);
1136*4882a593Smuzhiyun ramoops_unregister_dummy();
1137*4882a593Smuzhiyun }
1138*4882a593Smuzhiyun module_exit(ramoops_exit);
1139*4882a593Smuzhiyun
1140*4882a593Smuzhiyun MODULE_LICENSE("GPL");
1141*4882a593Smuzhiyun MODULE_AUTHOR("Marco Stornelli <marco.stornelli@gmail.com>");
1142*4882a593Smuzhiyun MODULE_DESCRIPTION("RAM Oops/Panic logger/driver");
1143