blob: 321437bbf87dd304a82d3452766b2102e60e7383 [file] [log] [blame]
Jason Barone9d376f2009-02-05 11:51:38 -05001/*
2 * lib/dynamic_debug.c
3 *
4 * make pr_debug()/dev_dbg() calls runtime configurable based upon their
5 * source module.
6 *
7 * Copyright (C) 2008 Jason Baron <jbaron@redhat.com>
8 * By Greg Banks <gnb@melbourne.sgi.com>
9 * Copyright (c) 2008 Silicon Graphics Inc. All Rights Reserved.
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010010 * Copyright (C) 2011 Bart Van Assche. All Rights Reserved.
Du, Changbin578b1e02014-01-23 15:54:14 -080011 * Copyright (C) 2013 Du, Changbin <changbin.du@gmail.com>
Jason Barone9d376f2009-02-05 11:51:38 -050012 */
13
Joe Perches4ad275e2011-08-11 14:36:33 -040014#define pr_fmt(fmt) KBUILD_MODNAME ":%s: " fmt, __func__
15
Jason Barone9d376f2009-02-05 11:51:38 -050016#include <linux/kernel.h>
17#include <linux/module.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070018#include <linux/moduleparam.h>
19#include <linux/kallsyms.h>
20#include <linux/types.h>
Jason Barone9d376f2009-02-05 11:51:38 -050021#include <linux/mutex.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070022#include <linux/proc_fs.h>
Jason Barone9d376f2009-02-05 11:51:38 -050023#include <linux/seq_file.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070024#include <linux/list.h>
25#include <linux/sysctl.h>
Jason Barone9d376f2009-02-05 11:51:38 -050026#include <linux/ctype.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070027#include <linux/string.h>
Du, Changbin578b1e02014-01-23 15:54:14 -080028#include <linux/parser.h>
Andy Shevchenkod338b132013-04-30 15:27:32 -070029#include <linux/string_helpers.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070030#include <linux/uaccess.h>
Jason Barone9d376f2009-02-05 11:51:38 -050031#include <linux/dynamic_debug.h>
32#include <linux/debugfs.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090033#include <linux/slab.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070034#include <linux/jump_label.h>
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010035#include <linux/hardirq.h>
Greg Kroah-Hartmane8d97922011-02-03 15:59:58 -080036#include <linux/sched.h>
Greg Kroah-Hartmanfef15d22012-05-07 16:47:32 -070037#include <linux/device.h>
Jason Baronffa10cb2011-08-11 14:36:48 -040038#include <linux/netdevice.h>
Jason Barone9d376f2009-02-05 11:51:38 -050039
Gal Pressman923abb92019-05-01 13:48:13 +030040#include <rdma/ib_verbs.h>
41
Jason Barone9d376f2009-02-05 11:51:38 -050042extern struct _ddebug __start___verbose[];
43extern struct _ddebug __stop___verbose[];
44
Jason Barone9d376f2009-02-05 11:51:38 -050045struct ddebug_table {
46 struct list_head link;
Rasmus Villemoes3e406b12015-11-06 16:30:15 -080047 const char *mod_name;
Jason Barone9d376f2009-02-05 11:51:38 -050048 unsigned int num_ddebugs;
Jason Barone9d376f2009-02-05 11:51:38 -050049 struct _ddebug *ddebugs;
50};
51
52struct ddebug_query {
53 const char *filename;
54 const char *module;
55 const char *function;
56 const char *format;
57 unsigned int first_lineno, last_lineno;
58};
59
60struct ddebug_iter {
61 struct ddebug_table *table;
62 unsigned int idx;
63};
64
65static DEFINE_MUTEX(ddebug_lock);
66static LIST_HEAD(ddebug_tables);
Joe Perchesf657fd22012-12-05 16:48:26 -050067static int verbose;
Jim Cromie74df1382011-12-19 17:12:24 -050068module_param(verbose, int, 0644);
Jason Barone9d376f2009-02-05 11:51:38 -050069
Jim Cromie2b678312011-12-19 17:13:12 -050070/* Return the path relative to source root */
71static inline const char *trim_prefix(const char *path)
72{
73 int skip = strlen(__FILE__) - strlen("lib/dynamic_debug.c");
74
75 if (strncmp(path, __FILE__, skip))
76 skip = 0; /* prefix mismatch, don't skip */
77
78 return path + skip;
79}
80
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010081static struct { unsigned flag:8; char opt_char; } opt_array[] = {
82 { _DPRINTK_FLAGS_PRINT, 'p' },
83 { _DPRINTK_FLAGS_INCL_MODNAME, 'm' },
84 { _DPRINTK_FLAGS_INCL_FUNCNAME, 'f' },
85 { _DPRINTK_FLAGS_INCL_LINENO, 'l' },
86 { _DPRINTK_FLAGS_INCL_TID, 't' },
Jim Cromie5ca7d2a2011-12-19 17:12:44 -050087 { _DPRINTK_FLAGS_NONE, '_' },
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010088};
89
Jason Barone9d376f2009-02-05 11:51:38 -050090/* format a string into buf[] which describes the _ddebug's flags */
91static char *ddebug_describe_flags(struct _ddebug *dp, char *buf,
92 size_t maxlen)
93{
94 char *p = buf;
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010095 int i;
Jason Barone9d376f2009-02-05 11:51:38 -050096
Jim Cromie5ca7d2a2011-12-19 17:12:44 -050097 BUG_ON(maxlen < 6);
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +010098 for (i = 0; i < ARRAY_SIZE(opt_array); ++i)
99 if (dp->flags & opt_array[i].flag)
100 *p++ = opt_array[i].opt_char;
Jason Barone9d376f2009-02-05 11:51:38 -0500101 if (p == buf)
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500102 *p++ = '_';
Jason Barone9d376f2009-02-05 11:51:38 -0500103 *p = '\0';
104
105 return buf;
106}
107
Joe Perchesf657fd22012-12-05 16:48:26 -0500108#define vpr_info(fmt, ...) \
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600109do { \
Joe Perchesf657fd22012-12-05 16:48:26 -0500110 if (verbose) \
111 pr_info(fmt, ##__VA_ARGS__); \
Jim Cromie574b3722011-12-19 17:13:16 -0500112} while (0)
113
Joe Perchesf657fd22012-12-05 16:48:26 -0500114static void vpr_info_dq(const struct ddebug_query *query, const char *msg)
115{
116 /* trim any trailing newlines */
117 int fmtlen = 0;
118
119 if (query->format) {
120 fmtlen = strlen(query->format);
121 while (fmtlen && query->format[fmtlen - 1] == '\n')
122 fmtlen--;
123 }
124
125 vpr_info("%s: func=\"%s\" file=\"%s\" module=\"%s\" format=\"%.*s\" lineno=%u-%u\n",
126 msg,
127 query->function ? query->function : "",
128 query->filename ? query->filename : "",
129 query->module ? query->module : "",
130 fmtlen, query->format ? query->format : "",
131 query->first_lineno, query->last_lineno);
132}
133
Jason Barone9d376f2009-02-05 11:51:38 -0500134/*
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500135 * Search the tables for _ddebug's which match the given `query' and
136 * apply the `flags' and `mask' to them. Returns number of matching
137 * callsites, normally the same as number of changes. If verbose,
138 * logs the changes. Takes ddebug_lock.
Jason Barone9d376f2009-02-05 11:51:38 -0500139 */
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500140static int ddebug_change(const struct ddebug_query *query,
141 unsigned int flags, unsigned int mask)
Jason Barone9d376f2009-02-05 11:51:38 -0500142{
143 int i;
144 struct ddebug_table *dt;
145 unsigned int newflags;
146 unsigned int nfound = 0;
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500147 char flagbuf[10];
Jason Barone9d376f2009-02-05 11:51:38 -0500148
149 /* search for matching ddebugs */
150 mutex_lock(&ddebug_lock);
151 list_for_each_entry(dt, &ddebug_tables, link) {
152
153 /* match against the module name */
Du, Changbin578b1e02014-01-23 15:54:14 -0800154 if (query->module &&
155 !match_wildcard(query->module, dt->mod_name))
Jason Barone9d376f2009-02-05 11:51:38 -0500156 continue;
157
Joe Perchesf657fd22012-12-05 16:48:26 -0500158 for (i = 0; i < dt->num_ddebugs; i++) {
Jason Barone9d376f2009-02-05 11:51:38 -0500159 struct _ddebug *dp = &dt->ddebugs[i];
160
161 /* match against the source filename */
Jim Cromied6a238d2011-12-19 17:12:39 -0500162 if (query->filename &&
Du, Changbin578b1e02014-01-23 15:54:14 -0800163 !match_wildcard(query->filename, dp->filename) &&
164 !match_wildcard(query->filename,
165 kbasename(dp->filename)) &&
166 !match_wildcard(query->filename,
167 trim_prefix(dp->filename)))
Jason Barone9d376f2009-02-05 11:51:38 -0500168 continue;
169
170 /* match against the function */
Jim Cromied6a238d2011-12-19 17:12:39 -0500171 if (query->function &&
Du, Changbin578b1e02014-01-23 15:54:14 -0800172 !match_wildcard(query->function, dp->function))
Jason Barone9d376f2009-02-05 11:51:38 -0500173 continue;
174
175 /* match against the format */
Jim Cromied6a238d2011-12-19 17:12:39 -0500176 if (query->format &&
177 !strstr(dp->format, query->format))
Jason Barone9d376f2009-02-05 11:51:38 -0500178 continue;
179
180 /* match against the line number range */
181 if (query->first_lineno &&
182 dp->lineno < query->first_lineno)
183 continue;
184 if (query->last_lineno &&
185 dp->lineno > query->last_lineno)
186 continue;
187
188 nfound++;
189
190 newflags = (dp->flags & mask) | flags;
191 if (newflags == dp->flags)
192 continue;
Masahiro Yamadae9666d12018-12-31 00:14:15 +0900193#ifdef CONFIG_JUMP_LABEL
Jason Baron9049fc72016-08-03 13:46:39 -0700194 if (dp->flags & _DPRINTK_FLAGS_PRINT) {
195 if (!(flags & _DPRINTK_FLAGS_PRINT))
196 static_branch_disable(&dp->key.dd_key_true);
197 } else if (flags & _DPRINTK_FLAGS_PRINT)
198 static_branch_enable(&dp->key.dd_key_true);
199#endif
Jason Barone9d376f2009-02-05 11:51:38 -0500200 dp->flags = newflags;
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600201 vpr_info("changed %s:%d [%s]%s =%s\n",
Joe Perchesf657fd22012-12-05 16:48:26 -0500202 trim_prefix(dp->filename), dp->lineno,
203 dt->mod_name, dp->function,
204 ddebug_describe_flags(dp, flagbuf,
205 sizeof(flagbuf)));
Jason Barone9d376f2009-02-05 11:51:38 -0500206 }
207 }
208 mutex_unlock(&ddebug_lock);
209
210 if (!nfound && verbose)
Joe Perches4ad275e2011-08-11 14:36:33 -0400211 pr_info("no matches for query\n");
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500212
213 return nfound;
Jason Barone9d376f2009-02-05 11:51:38 -0500214}
215
216/*
Jason Barone9d376f2009-02-05 11:51:38 -0500217 * Split the buffer `buf' into space-separated words.
Greg Banks9898abb2009-02-06 12:54:26 +1100218 * Handles simple " and ' quoting, i.e. without nested,
219 * embedded or escaped \". Return the number of words
220 * or <0 on error.
Jason Barone9d376f2009-02-05 11:51:38 -0500221 */
222static int ddebug_tokenize(char *buf, char *words[], int maxwords)
223{
224 int nwords = 0;
225
Greg Banks9898abb2009-02-06 12:54:26 +1100226 while (*buf) {
227 char *end;
228
229 /* Skip leading whitespace */
André Goddard Rosae7d28602009-12-14 18:01:06 -0800230 buf = skip_spaces(buf);
Greg Banks9898abb2009-02-06 12:54:26 +1100231 if (!*buf)
232 break; /* oh, it was trailing whitespace */
Jim Cromie8bd60262011-12-19 17:13:03 -0500233 if (*buf == '#')
234 break; /* token starts comment, skip rest of line */
Greg Banks9898abb2009-02-06 12:54:26 +1100235
Jim Cromie07100be2011-12-19 17:11:09 -0500236 /* find `end' of word, whitespace separated or quoted */
Greg Banks9898abb2009-02-06 12:54:26 +1100237 if (*buf == '"' || *buf == '\'') {
238 int quote = *buf++;
Joe Perchesf657fd22012-12-05 16:48:26 -0500239 for (end = buf; *end && *end != quote; end++)
Greg Banks9898abb2009-02-06 12:54:26 +1100240 ;
Jim Cromie18c216c2012-12-05 16:48:27 -0500241 if (!*end) {
242 pr_err("unclosed quote: %s\n", buf);
Greg Banks9898abb2009-02-06 12:54:26 +1100243 return -EINVAL; /* unclosed quote */
Jim Cromie18c216c2012-12-05 16:48:27 -0500244 }
Greg Banks9898abb2009-02-06 12:54:26 +1100245 } else {
Joe Perchesf657fd22012-12-05 16:48:26 -0500246 for (end = buf; *end && !isspace(*end); end++)
Greg Banks9898abb2009-02-06 12:54:26 +1100247 ;
248 BUG_ON(end == buf);
249 }
Greg Banks9898abb2009-02-06 12:54:26 +1100250
Jim Cromie07100be2011-12-19 17:11:09 -0500251 /* `buf' is start of word, `end' is one past its end */
Jim Cromie18c216c2012-12-05 16:48:27 -0500252 if (nwords == maxwords) {
253 pr_err("too many words, legal max <=%d\n", maxwords);
Greg Banks9898abb2009-02-06 12:54:26 +1100254 return -EINVAL; /* ran out of words[] before bytes */
Jim Cromie18c216c2012-12-05 16:48:27 -0500255 }
Greg Banks9898abb2009-02-06 12:54:26 +1100256 if (*end)
257 *end++ = '\0'; /* terminate the word */
258 words[nwords++] = buf;
259 buf = end;
260 }
Jason Barone9d376f2009-02-05 11:51:38 -0500261
262 if (verbose) {
263 int i;
Joe Perches4ad275e2011-08-11 14:36:33 -0400264 pr_info("split into words:");
Joe Perchesf657fd22012-12-05 16:48:26 -0500265 for (i = 0; i < nwords; i++)
Joe Perches4ad275e2011-08-11 14:36:33 -0400266 pr_cont(" \"%s\"", words[i]);
267 pr_cont("\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500268 }
269
270 return nwords;
271}
272
273/*
274 * Parse a single line number. Note that the empty string ""
275 * is treated as a special case and converted to zero, which
276 * is later treated as a "don't care" value.
277 */
278static inline int parse_lineno(const char *str, unsigned int *val)
279{
Jason Barone9d376f2009-02-05 11:51:38 -0500280 BUG_ON(str == NULL);
281 if (*str == '\0') {
282 *val = 0;
283 return 0;
284 }
Andrey Ryabinin45925992014-01-27 17:06:59 -0800285 if (kstrtouint(str, 10, val) < 0) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500286 pr_err("bad line-number: %s\n", str);
287 return -EINVAL;
288 }
289 return 0;
Jason Barone9d376f2009-02-05 11:51:38 -0500290}
291
Jim Cromie820874c2011-12-19 17:12:49 -0500292static int check_set(const char **dest, char *src, char *name)
293{
294 int rc = 0;
295
296 if (*dest) {
297 rc = -EINVAL;
Joe Perchesf657fd22012-12-05 16:48:26 -0500298 pr_err("match-spec:%s val:%s overridden by %s\n",
299 name, *dest, src);
Jim Cromie820874c2011-12-19 17:12:49 -0500300 }
301 *dest = src;
302 return rc;
303}
304
Jason Barone9d376f2009-02-05 11:51:38 -0500305/*
306 * Parse words[] as a ddebug query specification, which is a series
307 * of (keyword, value) pairs chosen from these possibilities:
308 *
309 * func <function-name>
310 * file <full-pathname>
311 * file <base-filename>
312 * module <module-name>
313 * format <escaped-string-to-find-in-format>
314 * line <lineno>
315 * line <first-lineno>-<last-lineno> // where either may be empty
Jim Cromie820874c2011-12-19 17:12:49 -0500316 *
317 * Only 1 of each type is allowed.
318 * Returns 0 on success, <0 on error.
Jason Barone9d376f2009-02-05 11:51:38 -0500319 */
320static int ddebug_parse_query(char *words[], int nwords,
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600321 struct ddebug_query *query, const char *modname)
Jason Barone9d376f2009-02-05 11:51:38 -0500322{
323 unsigned int i;
jbaron@akamai.combd8c1542013-08-27 16:50:03 +0000324 int rc = 0;
Jason Barone9d376f2009-02-05 11:51:38 -0500325
326 /* check we have an even number of words */
Jim Cromie18c216c2012-12-05 16:48:27 -0500327 if (nwords % 2 != 0) {
328 pr_err("expecting pairs of match-spec <value>\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500329 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500330 }
Jason Barone9d376f2009-02-05 11:51:38 -0500331 memset(query, 0, sizeof(*query));
332
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600333 if (modname)
334 /* support $modname.dyndbg=<multiple queries> */
335 query->module = modname;
336
Joe Perchesf657fd22012-12-05 16:48:26 -0500337 for (i = 0; i < nwords; i += 2) {
338 if (!strcmp(words[i], "func")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500339 rc = check_set(&query->function, words[i+1], "func");
Joe Perchesf657fd22012-12-05 16:48:26 -0500340 } else if (!strcmp(words[i], "file")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500341 rc = check_set(&query->filename, words[i+1], "file");
Joe Perchesf657fd22012-12-05 16:48:26 -0500342 } else if (!strcmp(words[i], "module")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500343 rc = check_set(&query->module, words[i+1], "module");
Joe Perchesf657fd22012-12-05 16:48:26 -0500344 } else if (!strcmp(words[i], "format")) {
Andy Shevchenkod338b132013-04-30 15:27:32 -0700345 string_unescape_inplace(words[i+1], UNESCAPE_SPACE |
346 UNESCAPE_OCTAL |
347 UNESCAPE_SPECIAL);
348 rc = check_set(&query->format, words[i+1], "format");
Joe Perchesf657fd22012-12-05 16:48:26 -0500349 } else if (!strcmp(words[i], "line")) {
Jason Barone9d376f2009-02-05 11:51:38 -0500350 char *first = words[i+1];
351 char *last = strchr(first, '-');
Jim Cromie820874c2011-12-19 17:12:49 -0500352 if (query->first_lineno || query->last_lineno) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500353 pr_err("match-spec: line used 2x\n");
Jim Cromie820874c2011-12-19 17:12:49 -0500354 return -EINVAL;
355 }
Jason Barone9d376f2009-02-05 11:51:38 -0500356 if (last)
357 *last++ = '\0';
Andrey Ryabinind9e133e2014-01-27 17:06:57 -0800358 if (parse_lineno(first, &query->first_lineno) < 0)
Jason Barone9d376f2009-02-05 11:51:38 -0500359 return -EINVAL;
Jim Cromie820874c2011-12-19 17:12:49 -0500360 if (last) {
Jason Barone9d376f2009-02-05 11:51:38 -0500361 /* range <first>-<last> */
Andrey Ryabinin3ace6782014-01-27 17:06:58 -0800362 if (parse_lineno(last, &query->last_lineno) < 0)
363 return -EINVAL;
364
Randy Dunlap1f3c7902017-11-17 15:27:35 -0800365 /* special case for last lineno not specified */
366 if (query->last_lineno == 0)
367 query->last_lineno = UINT_MAX;
368
Andrey Ryabinin3ace6782014-01-27 17:06:58 -0800369 if (query->last_lineno < query->first_lineno) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500370 pr_err("last-line:%d < 1st-line:%d\n",
371 query->last_lineno,
372 query->first_lineno);
Jason Barone9d376f2009-02-05 11:51:38 -0500373 return -EINVAL;
Jim Cromie820874c2011-12-19 17:12:49 -0500374 }
Jason Barone9d376f2009-02-05 11:51:38 -0500375 } else {
376 query->last_lineno = query->first_lineno;
377 }
378 } else {
Jim Cromieae27f862011-12-19 17:12:34 -0500379 pr_err("unknown keyword \"%s\"\n", words[i]);
Jason Barone9d376f2009-02-05 11:51:38 -0500380 return -EINVAL;
381 }
Jim Cromie820874c2011-12-19 17:12:49 -0500382 if (rc)
383 return rc;
Jason Barone9d376f2009-02-05 11:51:38 -0500384 }
Jim Cromie574b3722011-12-19 17:13:16 -0500385 vpr_info_dq(query, "parsed");
Jason Barone9d376f2009-02-05 11:51:38 -0500386 return 0;
387}
388
389/*
390 * Parse `str' as a flags specification, format [-+=][p]+.
391 * Sets up *maskp and *flagsp to be used when changing the
392 * flags fields of matched _ddebug's. Returns 0 on success
393 * or <0 on error.
394 */
395static int ddebug_parse_flags(const char *str, unsigned int *flagsp,
396 unsigned int *maskp)
397{
398 unsigned flags = 0;
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100399 int op = '=', i;
Jason Barone9d376f2009-02-05 11:51:38 -0500400
401 switch (*str) {
402 case '+':
403 case '-':
404 case '=':
405 op = *str++;
406 break;
407 default:
Jim Cromie18c216c2012-12-05 16:48:27 -0500408 pr_err("bad flag-op %c, at start of %s\n", *str, str);
Jason Barone9d376f2009-02-05 11:51:38 -0500409 return -EINVAL;
410 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600411 vpr_info("op='%c'\n", op);
Jason Barone9d376f2009-02-05 11:51:38 -0500412
Joe Perchesf657fd22012-12-05 16:48:26 -0500413 for (; *str ; ++str) {
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100414 for (i = ARRAY_SIZE(opt_array) - 1; i >= 0; i--) {
415 if (*str == opt_array[i].opt_char) {
416 flags |= opt_array[i].flag;
417 break;
418 }
Jason Barone9d376f2009-02-05 11:51:38 -0500419 }
Jim Cromie18c216c2012-12-05 16:48:27 -0500420 if (i < 0) {
421 pr_err("unknown flag '%c' in \"%s\"\n", *str, str);
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100422 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500423 }
Jason Barone9d376f2009-02-05 11:51:38 -0500424 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600425 vpr_info("flags=0x%x\n", flags);
Jason Barone9d376f2009-02-05 11:51:38 -0500426
427 /* calculate final *flagsp, *maskp according to mask and op */
428 switch (op) {
429 case '=':
430 *maskp = 0;
431 *flagsp = flags;
432 break;
433 case '+':
434 *maskp = ~0U;
435 *flagsp = flags;
436 break;
437 case '-':
438 *maskp = ~flags;
439 *flagsp = 0;
440 break;
441 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600442 vpr_info("*flagsp=0x%x *maskp=0x%x\n", *flagsp, *maskp);
Jason Barone9d376f2009-02-05 11:51:38 -0500443 return 0;
444}
445
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600446static int ddebug_exec_query(char *query_string, const char *modname)
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200447{
448 unsigned int flags = 0, mask = 0;
449 struct ddebug_query query;
450#define MAXWORDS 9
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500451 int nwords, nfound;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200452 char *words[MAXWORDS];
453
454 nwords = ddebug_tokenize(query_string, words, MAXWORDS);
Jim Cromie18c216c2012-12-05 16:48:27 -0500455 if (nwords <= 0) {
456 pr_err("tokenize failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200457 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500458 }
459 /* check flags 1st (last arg) so query is pairs of spec,val */
460 if (ddebug_parse_flags(words[nwords-1], &flags, &mask)) {
461 pr_err("flags parse failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200462 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500463 }
464 if (ddebug_parse_query(words, nwords-1, &query, modname)) {
465 pr_err("query parse failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200466 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500467 }
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200468 /* actually go and implement the change */
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500469 nfound = ddebug_change(&query, flags, mask);
Joe Perchesf657fd22012-12-05 16:48:26 -0500470 vpr_info_dq(&query, nfound ? "applied" : "no-match");
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500471
472 return nfound;
473}
474
475/* handle multiple queries in query string, continue on error, return
476 last error or number of matching callsites. Module name is either
477 in param (for boot arg) or perhaps in query string.
478*/
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600479static int ddebug_exec_queries(char *query, const char *modname)
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500480{
481 char *split;
482 int i, errs = 0, exitcode = 0, rc, nfound = 0;
483
484 for (i = 0; query; query = split) {
485 split = strpbrk(query, ";\n");
486 if (split)
487 *split++ = '\0';
488
489 query = skip_spaces(query);
490 if (!query || !*query || *query == '#')
491 continue;
492
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600493 vpr_info("query %d: \"%s\"\n", i, query);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500494
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600495 rc = ddebug_exec_query(query, modname);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500496 if (rc < 0) {
497 errs++;
498 exitcode = rc;
Joe Perchesf657fd22012-12-05 16:48:26 -0500499 } else {
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500500 nfound += rc;
Joe Perchesf657fd22012-12-05 16:48:26 -0500501 }
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500502 i++;
503 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600504 vpr_info("processed %d queries, with %d matches, %d errs\n",
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500505 i, nfound, errs);
506
507 if (exitcode)
508 return exitcode;
509 return nfound;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200510}
511
Jason Baron431625d2011-10-04 14:13:19 -0700512#define PREFIX_SIZE 64
513
514static int remaining(int wrote)
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100515{
Jason Baron431625d2011-10-04 14:13:19 -0700516 if (PREFIX_SIZE - wrote > 0)
517 return PREFIX_SIZE - wrote;
518 return 0;
519}
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100520
Jason Baron431625d2011-10-04 14:13:19 -0700521static char *dynamic_emit_prefix(const struct _ddebug *desc, char *buf)
522{
523 int pos_after_tid;
524 int pos = 0;
525
Joe Perches798efc62012-09-12 20:11:29 -0700526 *buf = '\0';
527
Jason Baron431625d2011-10-04 14:13:19 -0700528 if (desc->flags & _DPRINTK_FLAGS_INCL_TID) {
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100529 if (in_interrupt())
Joe Perches798efc62012-09-12 20:11:29 -0700530 pos += snprintf(buf + pos, remaining(pos), "<intr> ");
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100531 else
Jason Baron431625d2011-10-04 14:13:19 -0700532 pos += snprintf(buf + pos, remaining(pos), "[%d] ",
Joe Perches798efc62012-09-12 20:11:29 -0700533 task_pid_vnr(current));
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100534 }
Jason Baron431625d2011-10-04 14:13:19 -0700535 pos_after_tid = pos;
536 if (desc->flags & _DPRINTK_FLAGS_INCL_MODNAME)
537 pos += snprintf(buf + pos, remaining(pos), "%s:",
Joe Perches798efc62012-09-12 20:11:29 -0700538 desc->modname);
Jason Baron431625d2011-10-04 14:13:19 -0700539 if (desc->flags & _DPRINTK_FLAGS_INCL_FUNCNAME)
540 pos += snprintf(buf + pos, remaining(pos), "%s:",
Joe Perches798efc62012-09-12 20:11:29 -0700541 desc->function);
Jason Baron431625d2011-10-04 14:13:19 -0700542 if (desc->flags & _DPRINTK_FLAGS_INCL_LINENO)
Jim Cromie07100be2011-12-19 17:11:09 -0500543 pos += snprintf(buf + pos, remaining(pos), "%d:",
Joe Perches798efc62012-09-12 20:11:29 -0700544 desc->lineno);
Jason Baron431625d2011-10-04 14:13:19 -0700545 if (pos - pos_after_tid)
546 pos += snprintf(buf + pos, remaining(pos), " ");
547 if (pos >= PREFIX_SIZE)
548 buf[PREFIX_SIZE - 1] = '\0';
Joe Perches6c2140e2011-08-11 14:36:25 -0400549
Jason Baron431625d2011-10-04 14:13:19 -0700550 return buf;
Joe Perches6c2140e2011-08-11 14:36:25 -0400551}
552
Joe Perches906d2012014-09-24 11:17:56 -0700553void __dynamic_pr_debug(struct _ddebug *descriptor, const char *fmt, ...)
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100554{
555 va_list args;
Jason Baron431625d2011-10-04 14:13:19 -0700556 struct va_format vaf;
557 char buf[PREFIX_SIZE];
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100558
559 BUG_ON(!descriptor);
560 BUG_ON(!fmt);
561
562 va_start(args, fmt);
Joe Perches798efc62012-09-12 20:11:29 -0700563
Jason Baron431625d2011-10-04 14:13:19 -0700564 vaf.fmt = fmt;
565 vaf.va = &args;
Joe Perches798efc62012-09-12 20:11:29 -0700566
Joe Perches906d2012014-09-24 11:17:56 -0700567 printk(KERN_DEBUG "%s%pV", dynamic_emit_prefix(descriptor, buf), &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700568
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100569 va_end(args);
Bart Van Assche8ba6ebf52011-01-23 17:17:24 +0100570}
571EXPORT_SYMBOL(__dynamic_pr_debug);
572
Joe Perches906d2012014-09-24 11:17:56 -0700573void __dynamic_dev_dbg(struct _ddebug *descriptor,
Joe Perchescbc46632011-08-11 14:36:21 -0400574 const struct device *dev, const char *fmt, ...)
575{
576 struct va_format vaf;
577 va_list args;
Joe Perchescbc46632011-08-11 14:36:21 -0400578
579 BUG_ON(!descriptor);
580 BUG_ON(!fmt);
581
582 va_start(args, fmt);
Joe Perches798efc62012-09-12 20:11:29 -0700583
Joe Perchescbc46632011-08-11 14:36:21 -0400584 vaf.fmt = fmt;
585 vaf.va = &args;
Joe Perches798efc62012-09-12 20:11:29 -0700586
587 if (!dev) {
Joe Perches906d2012014-09-24 11:17:56 -0700588 printk(KERN_DEBUG "(NULL device *): %pV", &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700589 } else {
590 char buf[PREFIX_SIZE];
Joe Perches798efc62012-09-12 20:11:29 -0700591
Joe Perchesa39d4a82014-12-10 15:50:15 -0800592 dev_printk_emit(LOGLEVEL_DEBUG, dev, "%s%s %s: %pV",
Joe Perches906d2012014-09-24 11:17:56 -0700593 dynamic_emit_prefix(descriptor, buf),
594 dev_driver_string(dev), dev_name(dev),
595 &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700596 }
597
Joe Perchescbc46632011-08-11 14:36:21 -0400598 va_end(args);
Joe Perchescbc46632011-08-11 14:36:21 -0400599}
600EXPORT_SYMBOL(__dynamic_dev_dbg);
601
Jason Baron0feefd92011-10-04 14:13:22 -0700602#ifdef CONFIG_NET
603
Joe Perches906d2012014-09-24 11:17:56 -0700604void __dynamic_netdev_dbg(struct _ddebug *descriptor,
605 const struct net_device *dev, const char *fmt, ...)
Jason Baronffa10cb2011-08-11 14:36:48 -0400606{
607 struct va_format vaf;
608 va_list args;
Jason Baronffa10cb2011-08-11 14:36:48 -0400609
610 BUG_ON(!descriptor);
611 BUG_ON(!fmt);
612
613 va_start(args, fmt);
Joe Perchesb004ff42012-09-12 20:12:19 -0700614
Jason Baronffa10cb2011-08-11 14:36:48 -0400615 vaf.fmt = fmt;
616 vaf.va = &args;
Joe Perchesb004ff42012-09-12 20:12:19 -0700617
618 if (dev && dev->dev.parent) {
619 char buf[PREFIX_SIZE];
Joe Perchesb004ff42012-09-12 20:12:19 -0700620
Joe Perchesa39d4a82014-12-10 15:50:15 -0800621 dev_printk_emit(LOGLEVEL_DEBUG, dev->dev.parent,
Joe Perches906d2012014-09-24 11:17:56 -0700622 "%s%s %s %s%s: %pV",
623 dynamic_emit_prefix(descriptor, buf),
624 dev_driver_string(dev->dev.parent),
625 dev_name(dev->dev.parent),
626 netdev_name(dev), netdev_reg_state(dev),
627 &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700628 } else if (dev) {
Joe Perches906d2012014-09-24 11:17:56 -0700629 printk(KERN_DEBUG "%s%s: %pV", netdev_name(dev),
630 netdev_reg_state(dev), &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700631 } else {
Joe Perches906d2012014-09-24 11:17:56 -0700632 printk(KERN_DEBUG "(NULL net_device): %pV", &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700633 }
634
Jason Baronffa10cb2011-08-11 14:36:48 -0400635 va_end(args);
Jason Baronffa10cb2011-08-11 14:36:48 -0400636}
637EXPORT_SYMBOL(__dynamic_netdev_dbg);
638
Jason Baron0feefd92011-10-04 14:13:22 -0700639#endif
640
Gal Pressman923abb92019-05-01 13:48:13 +0300641#if IS_ENABLED(CONFIG_INFINIBAND)
642
643void __dynamic_ibdev_dbg(struct _ddebug *descriptor,
644 const struct ib_device *ibdev, const char *fmt, ...)
645{
646 struct va_format vaf;
647 va_list args;
648
649 va_start(args, fmt);
650
651 vaf.fmt = fmt;
652 vaf.va = &args;
653
654 if (ibdev && ibdev->dev.parent) {
655 char buf[PREFIX_SIZE];
656
657 dev_printk_emit(LOGLEVEL_DEBUG, ibdev->dev.parent,
658 "%s%s %s %s: %pV",
659 dynamic_emit_prefix(descriptor, buf),
660 dev_driver_string(ibdev->dev.parent),
661 dev_name(ibdev->dev.parent),
662 dev_name(&ibdev->dev),
663 &vaf);
664 } else if (ibdev) {
665 printk(KERN_DEBUG "%s: %pV", dev_name(&ibdev->dev), &vaf);
666 } else {
667 printk(KERN_DEBUG "(NULL ib_device): %pV", &vaf);
668 }
669
670 va_end(args);
671}
672EXPORT_SYMBOL(__dynamic_ibdev_dbg);
673
674#endif
675
Jim Cromiebc757f62011-12-19 17:12:29 -0500676#define DDEBUG_STRING_SIZE 1024
677static __initdata char ddebug_setup_string[DDEBUG_STRING_SIZE];
678
Thomas Renningera648ec02010-08-06 16:11:02 +0200679static __init int ddebug_setup_query(char *str)
680{
Jim Cromiebc757f62011-12-19 17:12:29 -0500681 if (strlen(str) >= DDEBUG_STRING_SIZE) {
Joe Perches4ad275e2011-08-11 14:36:33 -0400682 pr_warn("ddebug boot param string too large\n");
Thomas Renningera648ec02010-08-06 16:11:02 +0200683 return 0;
684 }
Jim Cromiebc757f62011-12-19 17:12:29 -0500685 strlcpy(ddebug_setup_string, str, DDEBUG_STRING_SIZE);
Thomas Renningera648ec02010-08-06 16:11:02 +0200686 return 1;
687}
688
689__setup("ddebug_query=", ddebug_setup_query);
690
Jason Barone9d376f2009-02-05 11:51:38 -0500691/*
Masatake YAMATO231821d2014-12-15 12:04:16 +0900692 * File_ops->write method for <debugfs>/dynamic_debug/control. Gathers the
Jason Barone9d376f2009-02-05 11:51:38 -0500693 * command text from userspace, parses and executes it.
694 */
Jim Cromie72814912011-12-19 17:13:07 -0500695#define USER_BUF_PAGE 4096
Jason Barone9d376f2009-02-05 11:51:38 -0500696static ssize_t ddebug_proc_write(struct file *file, const char __user *ubuf,
697 size_t len, loff_t *offp)
698{
Jim Cromie72814912011-12-19 17:13:07 -0500699 char *tmpbuf;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200700 int ret;
Jason Barone9d376f2009-02-05 11:51:38 -0500701
702 if (len == 0)
703 return 0;
Jim Cromie72814912011-12-19 17:13:07 -0500704 if (len > USER_BUF_PAGE - 1) {
705 pr_warn("expected <%d bytes into control\n", USER_BUF_PAGE);
Jason Barone9d376f2009-02-05 11:51:38 -0500706 return -E2BIG;
Jim Cromie72814912011-12-19 17:13:07 -0500707 }
Al Viro16e5c1f2015-12-24 00:06:05 -0500708 tmpbuf = memdup_user_nul(ubuf, len);
709 if (IS_ERR(tmpbuf))
710 return PTR_ERR(tmpbuf);
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600711 vpr_info("read %d bytes from userspace\n", (int)len);
Jason Barone9d376f2009-02-05 11:51:38 -0500712
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600713 ret = ddebug_exec_queries(tmpbuf, NULL);
Jim Cromie72814912011-12-19 17:13:07 -0500714 kfree(tmpbuf);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500715 if (ret < 0)
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200716 return ret;
Jason Barone9d376f2009-02-05 11:51:38 -0500717
718 *offp += len;
719 return len;
720}
721
722/*
723 * Set the iterator to point to the first _ddebug object
724 * and return a pointer to that first object. Returns
725 * NULL if there are no _ddebugs at all.
726 */
727static struct _ddebug *ddebug_iter_first(struct ddebug_iter *iter)
728{
729 if (list_empty(&ddebug_tables)) {
730 iter->table = NULL;
731 iter->idx = 0;
732 return NULL;
733 }
734 iter->table = list_entry(ddebug_tables.next,
735 struct ddebug_table, link);
736 iter->idx = 0;
737 return &iter->table->ddebugs[iter->idx];
738}
739
740/*
741 * Advance the iterator to point to the next _ddebug
742 * object from the one the iterator currently points at,
743 * and returns a pointer to the new _ddebug. Returns
744 * NULL if the iterator has seen all the _ddebugs.
745 */
746static struct _ddebug *ddebug_iter_next(struct ddebug_iter *iter)
747{
748 if (iter->table == NULL)
749 return NULL;
750 if (++iter->idx == iter->table->num_ddebugs) {
751 /* iterate to next table */
752 iter->idx = 0;
753 if (list_is_last(&iter->table->link, &ddebug_tables)) {
754 iter->table = NULL;
755 return NULL;
756 }
757 iter->table = list_entry(iter->table->link.next,
758 struct ddebug_table, link);
759 }
760 return &iter->table->ddebugs[iter->idx];
761}
762
763/*
764 * Seq_ops start method. Called at the start of every
765 * read() call from userspace. Takes the ddebug_lock and
766 * seeks the seq_file's iterator to the given position.
767 */
768static void *ddebug_proc_start(struct seq_file *m, loff_t *pos)
769{
770 struct ddebug_iter *iter = m->private;
771 struct _ddebug *dp;
772 int n = *pos;
773
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600774 vpr_info("called m=%p *pos=%lld\n", m, (unsigned long long)*pos);
Jason Barone9d376f2009-02-05 11:51:38 -0500775
776 mutex_lock(&ddebug_lock);
777
778 if (!n)
779 return SEQ_START_TOKEN;
780 if (n < 0)
781 return NULL;
782 dp = ddebug_iter_first(iter);
783 while (dp != NULL && --n > 0)
784 dp = ddebug_iter_next(iter);
785 return dp;
786}
787
788/*
789 * Seq_ops next method. Called several times within a read()
790 * call from userspace, with ddebug_lock held. Walks to the
791 * next _ddebug object with a special case for the header line.
792 */
793static void *ddebug_proc_next(struct seq_file *m, void *p, loff_t *pos)
794{
795 struct ddebug_iter *iter = m->private;
796 struct _ddebug *dp;
797
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600798 vpr_info("called m=%p p=%p *pos=%lld\n",
Joe Perchesf657fd22012-12-05 16:48:26 -0500799 m, p, (unsigned long long)*pos);
Jason Barone9d376f2009-02-05 11:51:38 -0500800
801 if (p == SEQ_START_TOKEN)
802 dp = ddebug_iter_first(iter);
803 else
804 dp = ddebug_iter_next(iter);
805 ++*pos;
806 return dp;
807}
808
809/*
810 * Seq_ops show method. Called several times within a read()
811 * call from userspace, with ddebug_lock held. Formats the
812 * current _ddebug as a single human-readable line, with a
813 * special case for the header line.
814 */
815static int ddebug_proc_show(struct seq_file *m, void *p)
816{
817 struct ddebug_iter *iter = m->private;
818 struct _ddebug *dp = p;
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500819 char flagsbuf[10];
Jason Barone9d376f2009-02-05 11:51:38 -0500820
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600821 vpr_info("called m=%p p=%p\n", m, p);
Jason Barone9d376f2009-02-05 11:51:38 -0500822
823 if (p == SEQ_START_TOKEN) {
824 seq_puts(m,
Joe Perchesf657fd22012-12-05 16:48:26 -0500825 "# filename:lineno [module]function flags format\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500826 return 0;
827 }
828
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500829 seq_printf(m, "%s:%u [%s]%s =%s \"",
Joe Perchesf657fd22012-12-05 16:48:26 -0500830 trim_prefix(dp->filename), dp->lineno,
831 iter->table->mod_name, dp->function,
832 ddebug_describe_flags(dp, flagsbuf, sizeof(flagsbuf)));
Jason Barone9d376f2009-02-05 11:51:38 -0500833 seq_escape(m, dp->format, "\t\r\n\"");
834 seq_puts(m, "\"\n");
835
836 return 0;
837}
838
839/*
840 * Seq_ops stop method. Called at the end of each read()
841 * call from userspace. Drops ddebug_lock.
842 */
843static void ddebug_proc_stop(struct seq_file *m, void *p)
844{
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600845 vpr_info("called m=%p p=%p\n", m, p);
Jason Barone9d376f2009-02-05 11:51:38 -0500846 mutex_unlock(&ddebug_lock);
847}
848
849static const struct seq_operations ddebug_proc_seqops = {
850 .start = ddebug_proc_start,
851 .next = ddebug_proc_next,
852 .show = ddebug_proc_show,
853 .stop = ddebug_proc_stop
854};
855
856/*
Jim Cromie07100be2011-12-19 17:11:09 -0500857 * File_ops->open method for <debugfs>/dynamic_debug/control. Does
858 * the seq_file setup dance, and also creates an iterator to walk the
859 * _ddebugs. Note that we create a seq_file always, even for O_WRONLY
860 * files where it's not needed, as doing so simplifies the ->release
861 * method.
Jason Barone9d376f2009-02-05 11:51:38 -0500862 */
863static int ddebug_proc_open(struct inode *inode, struct file *file)
864{
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600865 vpr_info("called\n");
Rob Jones4bad78c2014-10-13 15:51:32 -0700866 return seq_open_private(file, &ddebug_proc_seqops,
867 sizeof(struct ddebug_iter));
Jason Barone9d376f2009-02-05 11:51:38 -0500868}
869
870static const struct file_operations ddebug_proc_fops = {
871 .owner = THIS_MODULE,
872 .open = ddebug_proc_open,
873 .read = seq_read,
874 .llseek = seq_lseek,
875 .release = seq_release_private,
876 .write = ddebug_proc_write
877};
878
Greg Kroah-Hartman239a5792020-02-10 13:11:42 -0800879static const struct proc_ops proc_fops = {
880 .proc_open = ddebug_proc_open,
881 .proc_read = seq_read,
882 .proc_lseek = seq_lseek,
883 .proc_release = seq_release_private,
884 .proc_write = ddebug_proc_write
885};
886
Jason Barone9d376f2009-02-05 11:51:38 -0500887/*
888 * Allocate a new ddebug_table for the given module
889 * and add it to the global list.
890 */
891int ddebug_add_module(struct _ddebug *tab, unsigned int n,
892 const char *name)
893{
894 struct ddebug_table *dt;
Jason Barone9d376f2009-02-05 11:51:38 -0500895
896 dt = kzalloc(sizeof(*dt), GFP_KERNEL);
Rasmus Villemoes513770f2019-03-07 16:27:48 -0800897 if (dt == NULL) {
898 pr_err("error adding module: %s\n", name);
Jason Barone9d376f2009-02-05 11:51:38 -0500899 return -ENOMEM;
Rasmus Villemoes513770f2019-03-07 16:27:48 -0800900 }
Rasmus Villemoescdf6d002019-03-07 16:27:37 -0800901 /*
902 * For built-in modules, name lives in .rodata and is
903 * immortal. For loaded modules, name points at the name[]
904 * member of struct module, which lives at least as long as
905 * this struct ddebug_table.
906 */
907 dt->mod_name = name;
Jason Barone9d376f2009-02-05 11:51:38 -0500908 dt->num_ddebugs = n;
Jason Barone9d376f2009-02-05 11:51:38 -0500909 dt->ddebugs = tab;
910
911 mutex_lock(&ddebug_lock);
912 list_add_tail(&dt->link, &ddebug_tables);
913 mutex_unlock(&ddebug_lock);
914
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600915 vpr_info("%u debug prints in module %s\n", n, dt->mod_name);
Jason Barone9d376f2009-02-05 11:51:38 -0500916 return 0;
917}
Jason Barone9d376f2009-02-05 11:51:38 -0500918
Jim Cromie6ab676e2012-04-27 14:30:37 -0600919/* helper for ddebug_dyndbg_(boot|module)_param_cb */
920static int ddebug_dyndbg_param_cb(char *param, char *val,
921 const char *modname, int on_err)
Jim Cromieb48420c2012-04-27 14:30:35 -0600922{
Jim Cromieb48420c2012-04-27 14:30:35 -0600923 char *sep;
924
925 sep = strchr(param, '.');
926 if (sep) {
Jim Cromie6ab676e2012-04-27 14:30:37 -0600927 /* needed only for ddebug_dyndbg_boot_param_cb */
Jim Cromieb48420c2012-04-27 14:30:35 -0600928 *sep = '\0';
929 modname = param;
930 param = sep + 1;
931 }
932 if (strcmp(param, "dyndbg"))
Jim Cromie6ab676e2012-04-27 14:30:37 -0600933 return on_err; /* determined by caller */
Jim Cromieb48420c2012-04-27 14:30:35 -0600934
Jim Cromie8e59b5cf2012-04-27 14:30:40 -0600935 ddebug_exec_queries((val ? val : "+p"), modname);
936
Jim Cromieb48420c2012-04-27 14:30:35 -0600937 return 0; /* query failure shouldnt stop module load */
938}
939
Jim Cromie6ab676e2012-04-27 14:30:37 -0600940/* handle both dyndbg and $module.dyndbg params at boot */
941static int ddebug_dyndbg_boot_param_cb(char *param, char *val,
Luis R. Rodriguezecc86172015-03-30 16:20:03 -0700942 const char *unused, void *arg)
Jim Cromieb48420c2012-04-27 14:30:35 -0600943{
Jim Cromie6ab676e2012-04-27 14:30:37 -0600944 vpr_info("%s=\"%s\"\n", param, val);
945 return ddebug_dyndbg_param_cb(param, val, NULL, 0);
946}
Jim Cromieb48420c2012-04-27 14:30:35 -0600947
Jim Cromie6ab676e2012-04-27 14:30:37 -0600948/*
949 * modprobe foo finds foo.params in boot-args, strips "foo.", and
950 * passes them to load_module(). This callback gets unknown params,
951 * processes dyndbg params, rejects others.
952 */
953int ddebug_dyndbg_module_param_cb(char *param, char *val, const char *module)
954{
955 vpr_info("module: %s %s=\"%s\"\n", module, param, val);
956 return ddebug_dyndbg_param_cb(param, val, module, -ENOENT);
Jim Cromieb48420c2012-04-27 14:30:35 -0600957}
958
Jason Barone9d376f2009-02-05 11:51:38 -0500959static void ddebug_table_free(struct ddebug_table *dt)
960{
961 list_del_init(&dt->link);
Jason Barone9d376f2009-02-05 11:51:38 -0500962 kfree(dt);
963}
964
965/*
966 * Called in response to a module being unloaded. Removes
967 * any ddebug_table's which point at the module.
968 */
Yehuda Sadehff49d742010-07-03 13:07:35 +1000969int ddebug_remove_module(const char *mod_name)
Jason Barone9d376f2009-02-05 11:51:38 -0500970{
971 struct ddebug_table *dt, *nextdt;
972 int ret = -ENOENT;
973
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600974 vpr_info("removing module \"%s\"\n", mod_name);
Jason Barone9d376f2009-02-05 11:51:38 -0500975
976 mutex_lock(&ddebug_lock);
977 list_for_each_entry_safe(dt, nextdt, &ddebug_tables, link) {
Rasmus Villemoes4573fe12019-03-07 16:27:41 -0800978 if (dt->mod_name == mod_name) {
Jason Barone9d376f2009-02-05 11:51:38 -0500979 ddebug_table_free(dt);
980 ret = 0;
Rasmus Villemoes4573fe12019-03-07 16:27:41 -0800981 break;
Jason Barone9d376f2009-02-05 11:51:38 -0500982 }
983 }
984 mutex_unlock(&ddebug_lock);
985 return ret;
986}
Jason Barone9d376f2009-02-05 11:51:38 -0500987
988static void ddebug_remove_all_tables(void)
989{
990 mutex_lock(&ddebug_lock);
991 while (!list_empty(&ddebug_tables)) {
992 struct ddebug_table *dt = list_entry(ddebug_tables.next,
993 struct ddebug_table,
994 link);
995 ddebug_table_free(dt);
996 }
997 mutex_unlock(&ddebug_lock);
998}
999
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001000static __initdata int ddebug_init_success;
1001
Greg Kroah-Hartman239a5792020-02-10 13:11:42 -08001002static int __init dynamic_debug_init_control(void)
Jason Barone9d376f2009-02-05 11:51:38 -05001003{
Greg Kroah-Hartman239a5792020-02-10 13:11:42 -08001004 struct proc_dir_entry *procfs_dir;
1005 struct dentry *debugfs_dir;
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001006
1007 if (!ddebug_init_success)
1008 return -ENODEV;
Jason Barone9d376f2009-02-05 11:51:38 -05001009
Greg Kroah-Hartman239a5792020-02-10 13:11:42 -08001010 /* Create the control file in debugfs if it is enabled */
1011 if (debugfs_initialized()) {
1012 debugfs_dir = debugfs_create_dir("dynamic_debug", NULL);
1013 debugfs_create_file("control", 0644, debugfs_dir, NULL,
1014 &ddebug_proc_fops);
1015 }
1016
1017 /* Also create the control file in procfs */
1018 procfs_dir = proc_mkdir("dynamic_debug", NULL);
1019 if (procfs_dir)
1020 proc_create("control", 0644, procfs_dir, &proc_fops);
Greg Kroah-Hartman9fd714c2019-06-12 17:35:34 +02001021
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001022 return 0;
1023}
1024
1025static int __init dynamic_debug_init(void)
1026{
1027 struct _ddebug *iter, *iter_start;
1028 const char *modname = NULL;
Jim Cromieb48420c2012-04-27 14:30:35 -06001029 char *cmdline;
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001030 int ret = 0;
Jim Cromie41076922012-04-27 14:30:39 -06001031 int n = 0, entries = 0, modct = 0;
1032 int verbose_bytes = 0;
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001033
Nathan Chancellor8306b052020-04-06 20:10:45 -07001034 if (&__start___verbose == &__stop___verbose) {
Orson Zhaiceabef72020-06-07 21:40:14 -07001035 if (IS_ENABLED(CONFIG_DYNAMIC_DEBUG)) {
1036 pr_warn("_ddebug table is empty in a CONFIG_DYNAMIC_DEBUG build\n");
1037 return 1;
1038 }
1039 pr_info("Ignore empty _ddebug table in a CONFIG_DYNAMIC_DEBUG_CORE build\n");
1040 ddebug_init_success = 1;
1041 return 0;
Jason Barone9d376f2009-02-05 11:51:38 -05001042 }
Jim Cromieb5b78f82011-12-19 17:12:54 -05001043 iter = __start___verbose;
1044 modname = iter->modname;
1045 iter_start = iter;
1046 for (; iter < __stop___verbose; iter++) {
Jim Cromie41076922012-04-27 14:30:39 -06001047 entries++;
1048 verbose_bytes += strlen(iter->modname) + strlen(iter->function)
1049 + strlen(iter->filename) + strlen(iter->format);
1050
Jim Cromieb5b78f82011-12-19 17:12:54 -05001051 if (strcmp(modname, iter->modname)) {
Jim Cromie41076922012-04-27 14:30:39 -06001052 modct++;
Jim Cromieb5b78f82011-12-19 17:12:54 -05001053 ret = ddebug_add_module(iter_start, n, modname);
1054 if (ret)
Jim Cromieaf442392012-04-27 14:30:38 -06001055 goto out_err;
Jim Cromieb5b78f82011-12-19 17:12:54 -05001056 n = 0;
1057 modname = iter->modname;
1058 iter_start = iter;
1059 }
1060 n++;
1061 }
1062 ret = ddebug_add_module(iter_start, n, modname);
1063 if (ret)
Jim Cromieaf442392012-04-27 14:30:38 -06001064 goto out_err;
Thomas Renningera648ec02010-08-06 16:11:02 +02001065
Jim Cromieaf442392012-04-27 14:30:38 -06001066 ddebug_init_success = 1;
Joe Perchesf657fd22012-12-05 16:48:26 -05001067 vpr_info("%d modules, %d entries and %d bytes in ddebug tables, %d bytes in (readonly) verbose section\n",
1068 modct, entries, (int)(modct * sizeof(struct ddebug_table)),
1069 verbose_bytes + (int)(__stop___verbose - __start___verbose));
Jim Cromieaf442392012-04-27 14:30:38 -06001070
1071 /* apply ddebug_query boot param, dont unload tables on err */
Thomas Renningera648ec02010-08-06 16:11:02 +02001072 if (ddebug_setup_string[0] != '\0') {
Joe Perchesf657fd22012-12-05 16:48:26 -05001073 pr_warn("ddebug_query param name is deprecated, change it to dyndbg\n");
Jim Cromie8e59b5cf2012-04-27 14:30:40 -06001074 ret = ddebug_exec_queries(ddebug_setup_string, NULL);
Jim Cromie85f7f6c2011-12-19 17:13:21 -05001075 if (ret < 0)
Joe Perchesf657fd22012-12-05 16:48:26 -05001076 pr_warn("Invalid ddebug boot param %s\n",
Joe Perches4ad275e2011-08-11 14:36:33 -04001077 ddebug_setup_string);
Thomas Renningera648ec02010-08-06 16:11:02 +02001078 else
Jim Cromie85f7f6c2011-12-19 17:13:21 -05001079 pr_info("%d changes by ddebug_query\n", ret);
Jason Barone9d376f2009-02-05 11:51:38 -05001080 }
Jim Cromieb48420c2012-04-27 14:30:35 -06001081 /* now that ddebug tables are loaded, process all boot args
1082 * again to find and activate queries given in dyndbg params.
1083 * While this has already been done for known boot params, it
1084 * ignored the unknown ones (dyndbg in particular). Reusing
1085 * parse_args avoids ad-hoc parsing. This will also attempt
1086 * to activate queries for not-yet-loaded modules, which is
1087 * slightly noisy if verbose, but harmless.
1088 */
1089 cmdline = kstrdup(saved_command_line, GFP_KERNEL);
1090 parse_args("dyndbg params", cmdline, NULL,
Luis R. Rodriguezecc86172015-03-30 16:20:03 -07001091 0, 0, 0, NULL, &ddebug_dyndbg_boot_param_cb);
Jim Cromieb48420c2012-04-27 14:30:35 -06001092 kfree(cmdline);
Jim Cromieaf442392012-04-27 14:30:38 -06001093 return 0;
Thomas Renningera648ec02010-08-06 16:11:02 +02001094
Jim Cromieaf442392012-04-27 14:30:38 -06001095out_err:
1096 ddebug_remove_all_tables();
Jason Barone9d376f2009-02-05 11:51:38 -05001097 return 0;
1098}
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001099/* Allow early initialization for boot messages via boot param */
Jim Cromie3ec56522012-04-27 14:30:42 -06001100early_initcall(dynamic_debug_init);
Jim Cromieb48420c2012-04-27 14:30:35 -06001101
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001102/* Debugfs setup must be done later */
Greg Kroah-Hartman239a5792020-02-10 13:11:42 -08001103fs_initcall(dynamic_debug_init_control);