blob: 91c451e0f4741aaf97d272b96891a21e28047353 [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 Assche8ba6ebf2011-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 Assche8ba6ebf2011-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
40extern struct _ddebug __start___verbose[];
41extern struct _ddebug __stop___verbose[];
42
Jason Barone9d376f2009-02-05 11:51:38 -050043struct ddebug_table {
44 struct list_head link;
Rasmus Villemoes3e406b12015-11-06 16:30:15 -080045 const char *mod_name;
Jason Barone9d376f2009-02-05 11:51:38 -050046 unsigned int num_ddebugs;
Jason Barone9d376f2009-02-05 11:51:38 -050047 struct _ddebug *ddebugs;
48};
49
50struct ddebug_query {
51 const char *filename;
52 const char *module;
53 const char *function;
54 const char *format;
55 unsigned int first_lineno, last_lineno;
56};
57
58struct ddebug_iter {
59 struct ddebug_table *table;
60 unsigned int idx;
61};
62
63static DEFINE_MUTEX(ddebug_lock);
64static LIST_HEAD(ddebug_tables);
Joe Perchesf657fd22012-12-05 16:48:26 -050065static int verbose;
Jim Cromie74df1382011-12-19 17:12:24 -050066module_param(verbose, int, 0644);
Jason Barone9d376f2009-02-05 11:51:38 -050067
Jim Cromie2b678312011-12-19 17:13:12 -050068/* Return the path relative to source root */
69static inline const char *trim_prefix(const char *path)
70{
71 int skip = strlen(__FILE__) - strlen("lib/dynamic_debug.c");
72
73 if (strncmp(path, __FILE__, skip))
74 skip = 0; /* prefix mismatch, don't skip */
75
76 return path + skip;
77}
78
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +010079static struct { unsigned flag:8; char opt_char; } opt_array[] = {
80 { _DPRINTK_FLAGS_PRINT, 'p' },
81 { _DPRINTK_FLAGS_INCL_MODNAME, 'm' },
82 { _DPRINTK_FLAGS_INCL_FUNCNAME, 'f' },
83 { _DPRINTK_FLAGS_INCL_LINENO, 'l' },
84 { _DPRINTK_FLAGS_INCL_TID, 't' },
Jim Cromie5ca7d2a2011-12-19 17:12:44 -050085 { _DPRINTK_FLAGS_NONE, '_' },
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +010086};
87
Jim Cromieeb8cbd82020-07-19 17:10:47 -060088struct flagsbuf { char buf[ARRAY_SIZE(opt_array)+1]; };
89
Jason Barone9d376f2009-02-05 11:51:38 -050090/* format a string into buf[] which describes the _ddebug's flags */
Jim Cromieeb8cbd82020-07-19 17:10:47 -060091static char *ddebug_describe_flags(unsigned int flags, struct flagsbuf *fb)
Jason Barone9d376f2009-02-05 11:51:38 -050092{
Jim Cromieeb8cbd82020-07-19 17:10:47 -060093 char *p = fb->buf;
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +010094 int i;
Jason Barone9d376f2009-02-05 11:51:38 -050095
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +010096 for (i = 0; i < ARRAY_SIZE(opt_array); ++i)
Jim Cromieeb8cbd82020-07-19 17:10:47 -060097 if (flags & opt_array[i].flag)
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +010098 *p++ = opt_array[i].opt_char;
Jim Cromieeb8cbd82020-07-19 17:10:47 -060099 if (p == fb->buf)
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500100 *p++ = '_';
Jason Barone9d376f2009-02-05 11:51:38 -0500101 *p = '\0';
102
Jim Cromieeb8cbd82020-07-19 17:10:47 -0600103 return fb->buf;
Jason Barone9d376f2009-02-05 11:51:38 -0500104}
105
Joe Perchesf657fd22012-12-05 16:48:26 -0500106#define vpr_info(fmt, ...) \
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600107do { \
Joe Perchesf657fd22012-12-05 16:48:26 -0500108 if (verbose) \
109 pr_info(fmt, ##__VA_ARGS__); \
Jim Cromie574b3722011-12-19 17:13:16 -0500110} while (0)
111
Joe Perchesf657fd22012-12-05 16:48:26 -0500112static void vpr_info_dq(const struct ddebug_query *query, const char *msg)
113{
114 /* trim any trailing newlines */
115 int fmtlen = 0;
116
117 if (query->format) {
118 fmtlen = strlen(query->format);
119 while (fmtlen && query->format[fmtlen - 1] == '\n')
120 fmtlen--;
121 }
122
123 vpr_info("%s: func=\"%s\" file=\"%s\" module=\"%s\" format=\"%.*s\" lineno=%u-%u\n",
124 msg,
125 query->function ? query->function : "",
126 query->filename ? query->filename : "",
127 query->module ? query->module : "",
128 fmtlen, query->format ? query->format : "",
129 query->first_lineno, query->last_lineno);
130}
131
Jason Barone9d376f2009-02-05 11:51:38 -0500132/*
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500133 * Search the tables for _ddebug's which match the given `query' and
134 * apply the `flags' and `mask' to them. Returns number of matching
135 * callsites, normally the same as number of changes. If verbose,
136 * logs the changes. Takes ddebug_lock.
Jason Barone9d376f2009-02-05 11:51:38 -0500137 */
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500138static int ddebug_change(const struct ddebug_query *query,
139 unsigned int flags, unsigned int mask)
Jason Barone9d376f2009-02-05 11:51:38 -0500140{
141 int i;
142 struct ddebug_table *dt;
143 unsigned int newflags;
144 unsigned int nfound = 0;
Jim Cromieeb8cbd82020-07-19 17:10:47 -0600145 struct flagsbuf fbuf;
Jason Barone9d376f2009-02-05 11:51:38 -0500146
147 /* search for matching ddebugs */
148 mutex_lock(&ddebug_lock);
149 list_for_each_entry(dt, &ddebug_tables, link) {
150
151 /* match against the module name */
Du, Changbin578b1e02014-01-23 15:54:14 -0800152 if (query->module &&
153 !match_wildcard(query->module, dt->mod_name))
Jason Barone9d376f2009-02-05 11:51:38 -0500154 continue;
155
Joe Perchesf657fd22012-12-05 16:48:26 -0500156 for (i = 0; i < dt->num_ddebugs; i++) {
Jason Barone9d376f2009-02-05 11:51:38 -0500157 struct _ddebug *dp = &dt->ddebugs[i];
158
159 /* match against the source filename */
Jim Cromied6a238d2011-12-19 17:12:39 -0500160 if (query->filename &&
Du, Changbin578b1e02014-01-23 15:54:14 -0800161 !match_wildcard(query->filename, dp->filename) &&
162 !match_wildcard(query->filename,
163 kbasename(dp->filename)) &&
164 !match_wildcard(query->filename,
165 trim_prefix(dp->filename)))
Jason Barone9d376f2009-02-05 11:51:38 -0500166 continue;
167
168 /* match against the function */
Jim Cromied6a238d2011-12-19 17:12:39 -0500169 if (query->function &&
Du, Changbin578b1e02014-01-23 15:54:14 -0800170 !match_wildcard(query->function, dp->function))
Jason Barone9d376f2009-02-05 11:51:38 -0500171 continue;
172
173 /* match against the format */
Jim Cromied6a238d2011-12-19 17:12:39 -0500174 if (query->format &&
175 !strstr(dp->format, query->format))
Jason Barone9d376f2009-02-05 11:51:38 -0500176 continue;
177
178 /* match against the line number range */
179 if (query->first_lineno &&
180 dp->lineno < query->first_lineno)
181 continue;
182 if (query->last_lineno &&
183 dp->lineno > query->last_lineno)
184 continue;
185
186 nfound++;
187
188 newflags = (dp->flags & mask) | flags;
189 if (newflags == dp->flags)
190 continue;
Jason Baron9049fc72016-08-03 13:46:39 -0700191#ifdef HAVE_JUMP_LABEL
192 if (dp->flags & _DPRINTK_FLAGS_PRINT) {
193 if (!(flags & _DPRINTK_FLAGS_PRINT))
194 static_branch_disable(&dp->key.dd_key_true);
195 } else if (flags & _DPRINTK_FLAGS_PRINT)
196 static_branch_enable(&dp->key.dd_key_true);
197#endif
Jason Barone9d376f2009-02-05 11:51:38 -0500198 dp->flags = newflags;
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600199 vpr_info("changed %s:%d [%s]%s =%s\n",
Joe Perchesf657fd22012-12-05 16:48:26 -0500200 trim_prefix(dp->filename), dp->lineno,
201 dt->mod_name, dp->function,
Jim Cromieeb8cbd82020-07-19 17:10:47 -0600202 ddebug_describe_flags(dp->flags, &fbuf));
Jason Barone9d376f2009-02-05 11:51:38 -0500203 }
204 }
205 mutex_unlock(&ddebug_lock);
206
207 if (!nfound && verbose)
Joe Perches4ad275e2011-08-11 14:36:33 -0400208 pr_info("no matches for query\n");
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500209
210 return nfound;
Jason Barone9d376f2009-02-05 11:51:38 -0500211}
212
213/*
Jason Barone9d376f2009-02-05 11:51:38 -0500214 * Split the buffer `buf' into space-separated words.
Greg Banks9898abb2009-02-06 12:54:26 +1100215 * Handles simple " and ' quoting, i.e. without nested,
216 * embedded or escaped \". Return the number of words
217 * or <0 on error.
Jason Barone9d376f2009-02-05 11:51:38 -0500218 */
219static int ddebug_tokenize(char *buf, char *words[], int maxwords)
220{
221 int nwords = 0;
222
Greg Banks9898abb2009-02-06 12:54:26 +1100223 while (*buf) {
224 char *end;
225
226 /* Skip leading whitespace */
André Goddard Rosae7d28602009-12-14 18:01:06 -0800227 buf = skip_spaces(buf);
Greg Banks9898abb2009-02-06 12:54:26 +1100228 if (!*buf)
229 break; /* oh, it was trailing whitespace */
Jim Cromie8bd60262011-12-19 17:13:03 -0500230 if (*buf == '#')
231 break; /* token starts comment, skip rest of line */
Greg Banks9898abb2009-02-06 12:54:26 +1100232
Jim Cromie07100be2011-12-19 17:11:09 -0500233 /* find `end' of word, whitespace separated or quoted */
Greg Banks9898abb2009-02-06 12:54:26 +1100234 if (*buf == '"' || *buf == '\'') {
235 int quote = *buf++;
Joe Perchesf657fd22012-12-05 16:48:26 -0500236 for (end = buf; *end && *end != quote; end++)
Greg Banks9898abb2009-02-06 12:54:26 +1100237 ;
Jim Cromie18c216c2012-12-05 16:48:27 -0500238 if (!*end) {
239 pr_err("unclosed quote: %s\n", buf);
Greg Banks9898abb2009-02-06 12:54:26 +1100240 return -EINVAL; /* unclosed quote */
Jim Cromie18c216c2012-12-05 16:48:27 -0500241 }
Greg Banks9898abb2009-02-06 12:54:26 +1100242 } else {
Joe Perchesf657fd22012-12-05 16:48:26 -0500243 for (end = buf; *end && !isspace(*end); end++)
Greg Banks9898abb2009-02-06 12:54:26 +1100244 ;
245 BUG_ON(end == buf);
246 }
Greg Banks9898abb2009-02-06 12:54:26 +1100247
Jim Cromie07100be2011-12-19 17:11:09 -0500248 /* `buf' is start of word, `end' is one past its end */
Jim Cromie18c216c2012-12-05 16:48:27 -0500249 if (nwords == maxwords) {
250 pr_err("too many words, legal max <=%d\n", maxwords);
Greg Banks9898abb2009-02-06 12:54:26 +1100251 return -EINVAL; /* ran out of words[] before bytes */
Jim Cromie18c216c2012-12-05 16:48:27 -0500252 }
Greg Banks9898abb2009-02-06 12:54:26 +1100253 if (*end)
254 *end++ = '\0'; /* terminate the word */
255 words[nwords++] = buf;
256 buf = end;
257 }
Jason Barone9d376f2009-02-05 11:51:38 -0500258
259 if (verbose) {
260 int i;
Joe Perches4ad275e2011-08-11 14:36:33 -0400261 pr_info("split into words:");
Joe Perchesf657fd22012-12-05 16:48:26 -0500262 for (i = 0; i < nwords; i++)
Joe Perches4ad275e2011-08-11 14:36:33 -0400263 pr_cont(" \"%s\"", words[i]);
264 pr_cont("\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500265 }
266
267 return nwords;
268}
269
270/*
271 * Parse a single line number. Note that the empty string ""
272 * is treated as a special case and converted to zero, which
273 * is later treated as a "don't care" value.
274 */
275static inline int parse_lineno(const char *str, unsigned int *val)
276{
Jason Barone9d376f2009-02-05 11:51:38 -0500277 BUG_ON(str == NULL);
278 if (*str == '\0') {
279 *val = 0;
280 return 0;
281 }
Andrey Ryabinin45925992014-01-27 17:06:59 -0800282 if (kstrtouint(str, 10, val) < 0) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500283 pr_err("bad line-number: %s\n", str);
284 return -EINVAL;
285 }
286 return 0;
Jason Barone9d376f2009-02-05 11:51:38 -0500287}
288
Jim Cromie820874c2011-12-19 17:12:49 -0500289static int check_set(const char **dest, char *src, char *name)
290{
291 int rc = 0;
292
293 if (*dest) {
294 rc = -EINVAL;
Joe Perchesf657fd22012-12-05 16:48:26 -0500295 pr_err("match-spec:%s val:%s overridden by %s\n",
296 name, *dest, src);
Jim Cromie820874c2011-12-19 17:12:49 -0500297 }
298 *dest = src;
299 return rc;
300}
301
Jason Barone9d376f2009-02-05 11:51:38 -0500302/*
303 * Parse words[] as a ddebug query specification, which is a series
304 * of (keyword, value) pairs chosen from these possibilities:
305 *
306 * func <function-name>
307 * file <full-pathname>
308 * file <base-filename>
309 * module <module-name>
310 * format <escaped-string-to-find-in-format>
311 * line <lineno>
312 * line <first-lineno>-<last-lineno> // where either may be empty
Jim Cromie820874c2011-12-19 17:12:49 -0500313 *
314 * Only 1 of each type is allowed.
315 * Returns 0 on success, <0 on error.
Jason Barone9d376f2009-02-05 11:51:38 -0500316 */
317static int ddebug_parse_query(char *words[], int nwords,
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600318 struct ddebug_query *query, const char *modname)
Jason Barone9d376f2009-02-05 11:51:38 -0500319{
320 unsigned int i;
jbaron@akamai.combd8c154a2013-08-27 16:50:03 +0000321 int rc = 0;
Jason Barone9d376f2009-02-05 11:51:38 -0500322
323 /* check we have an even number of words */
Jim Cromie18c216c2012-12-05 16:48:27 -0500324 if (nwords % 2 != 0) {
325 pr_err("expecting pairs of match-spec <value>\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500326 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500327 }
Jason Barone9d376f2009-02-05 11:51:38 -0500328 memset(query, 0, sizeof(*query));
329
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600330 if (modname)
331 /* support $modname.dyndbg=<multiple queries> */
332 query->module = modname;
333
Joe Perchesf657fd22012-12-05 16:48:26 -0500334 for (i = 0; i < nwords; i += 2) {
335 if (!strcmp(words[i], "func")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500336 rc = check_set(&query->function, words[i+1], "func");
Joe Perchesf657fd22012-12-05 16:48:26 -0500337 } else if (!strcmp(words[i], "file")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500338 rc = check_set(&query->filename, words[i+1], "file");
Joe Perchesf657fd22012-12-05 16:48:26 -0500339 } else if (!strcmp(words[i], "module")) {
Jim Cromie820874c2011-12-19 17:12:49 -0500340 rc = check_set(&query->module, words[i+1], "module");
Joe Perchesf657fd22012-12-05 16:48:26 -0500341 } else if (!strcmp(words[i], "format")) {
Andy Shevchenkod338b132013-04-30 15:27:32 -0700342 string_unescape_inplace(words[i+1], UNESCAPE_SPACE |
343 UNESCAPE_OCTAL |
344 UNESCAPE_SPECIAL);
345 rc = check_set(&query->format, words[i+1], "format");
Joe Perchesf657fd22012-12-05 16:48:26 -0500346 } else if (!strcmp(words[i], "line")) {
Jason Barone9d376f2009-02-05 11:51:38 -0500347 char *first = words[i+1];
348 char *last = strchr(first, '-');
Jim Cromie820874c2011-12-19 17:12:49 -0500349 if (query->first_lineno || query->last_lineno) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500350 pr_err("match-spec: line used 2x\n");
Jim Cromie820874c2011-12-19 17:12:49 -0500351 return -EINVAL;
352 }
Jason Barone9d376f2009-02-05 11:51:38 -0500353 if (last)
354 *last++ = '\0';
Andrey Ryabinind9e133e2014-01-27 17:06:57 -0800355 if (parse_lineno(first, &query->first_lineno) < 0)
Jason Barone9d376f2009-02-05 11:51:38 -0500356 return -EINVAL;
Jim Cromie820874c2011-12-19 17:12:49 -0500357 if (last) {
Jason Barone9d376f2009-02-05 11:51:38 -0500358 /* range <first>-<last> */
Andrey Ryabinin3ace6782014-01-27 17:06:58 -0800359 if (parse_lineno(last, &query->last_lineno) < 0)
360 return -EINVAL;
361
Randy Dunlapa780a722017-11-17 15:27:35 -0800362 /* special case for last lineno not specified */
363 if (query->last_lineno == 0)
364 query->last_lineno = UINT_MAX;
365
Andrey Ryabinin3ace6782014-01-27 17:06:58 -0800366 if (query->last_lineno < query->first_lineno) {
Jim Cromie18c216c2012-12-05 16:48:27 -0500367 pr_err("last-line:%d < 1st-line:%d\n",
368 query->last_lineno,
369 query->first_lineno);
Jason Barone9d376f2009-02-05 11:51:38 -0500370 return -EINVAL;
Jim Cromie820874c2011-12-19 17:12:49 -0500371 }
Jason Barone9d376f2009-02-05 11:51:38 -0500372 } else {
373 query->last_lineno = query->first_lineno;
374 }
375 } else {
Jim Cromieae27f862011-12-19 17:12:34 -0500376 pr_err("unknown keyword \"%s\"\n", words[i]);
Jason Barone9d376f2009-02-05 11:51:38 -0500377 return -EINVAL;
378 }
Jim Cromie820874c2011-12-19 17:12:49 -0500379 if (rc)
380 return rc;
Jason Barone9d376f2009-02-05 11:51:38 -0500381 }
Jim Cromie574b3722011-12-19 17:13:16 -0500382 vpr_info_dq(query, "parsed");
Jason Barone9d376f2009-02-05 11:51:38 -0500383 return 0;
384}
385
386/*
387 * Parse `str' as a flags specification, format [-+=][p]+.
388 * Sets up *maskp and *flagsp to be used when changing the
389 * flags fields of matched _ddebug's. Returns 0 on success
390 * or <0 on error.
391 */
392static int ddebug_parse_flags(const char *str, unsigned int *flagsp,
393 unsigned int *maskp)
394{
395 unsigned flags = 0;
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100396 int op = '=', i;
Jason Barone9d376f2009-02-05 11:51:38 -0500397
398 switch (*str) {
399 case '+':
400 case '-':
401 case '=':
402 op = *str++;
403 break;
404 default:
Jim Cromie18c216c2012-12-05 16:48:27 -0500405 pr_err("bad flag-op %c, at start of %s\n", *str, str);
Jason Barone9d376f2009-02-05 11:51:38 -0500406 return -EINVAL;
407 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600408 vpr_info("op='%c'\n", op);
Jason Barone9d376f2009-02-05 11:51:38 -0500409
Joe Perchesf657fd22012-12-05 16:48:26 -0500410 for (; *str ; ++str) {
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100411 for (i = ARRAY_SIZE(opt_array) - 1; i >= 0; i--) {
412 if (*str == opt_array[i].opt_char) {
413 flags |= opt_array[i].flag;
414 break;
415 }
Jason Barone9d376f2009-02-05 11:51:38 -0500416 }
Jim Cromie18c216c2012-12-05 16:48:27 -0500417 if (i < 0) {
418 pr_err("unknown flag '%c' in \"%s\"\n", *str, str);
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100419 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500420 }
Jason Barone9d376f2009-02-05 11:51:38 -0500421 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600422 vpr_info("flags=0x%x\n", flags);
Jason Barone9d376f2009-02-05 11:51:38 -0500423
424 /* calculate final *flagsp, *maskp according to mask and op */
425 switch (op) {
426 case '=':
427 *maskp = 0;
428 *flagsp = flags;
429 break;
430 case '+':
431 *maskp = ~0U;
432 *flagsp = flags;
433 break;
434 case '-':
435 *maskp = ~flags;
436 *flagsp = 0;
437 break;
438 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600439 vpr_info("*flagsp=0x%x *maskp=0x%x\n", *flagsp, *maskp);
Jason Barone9d376f2009-02-05 11:51:38 -0500440 return 0;
441}
442
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600443static int ddebug_exec_query(char *query_string, const char *modname)
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200444{
445 unsigned int flags = 0, mask = 0;
446 struct ddebug_query query;
447#define MAXWORDS 9
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500448 int nwords, nfound;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200449 char *words[MAXWORDS];
450
451 nwords = ddebug_tokenize(query_string, words, MAXWORDS);
Jim Cromie18c216c2012-12-05 16:48:27 -0500452 if (nwords <= 0) {
453 pr_err("tokenize failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200454 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500455 }
456 /* check flags 1st (last arg) so query is pairs of spec,val */
457 if (ddebug_parse_flags(words[nwords-1], &flags, &mask)) {
458 pr_err("flags parse failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200459 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500460 }
461 if (ddebug_parse_query(words, nwords-1, &query, modname)) {
462 pr_err("query parse failed\n");
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200463 return -EINVAL;
Jim Cromie18c216c2012-12-05 16:48:27 -0500464 }
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200465 /* actually go and implement the change */
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500466 nfound = ddebug_change(&query, flags, mask);
Joe Perchesf657fd22012-12-05 16:48:26 -0500467 vpr_info_dq(&query, nfound ? "applied" : "no-match");
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500468
469 return nfound;
470}
471
472/* handle multiple queries in query string, continue on error, return
473 last error or number of matching callsites. Module name is either
474 in param (for boot arg) or perhaps in query string.
475*/
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600476static int ddebug_exec_queries(char *query, const char *modname)
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500477{
478 char *split;
479 int i, errs = 0, exitcode = 0, rc, nfound = 0;
480
481 for (i = 0; query; query = split) {
482 split = strpbrk(query, ";\n");
483 if (split)
484 *split++ = '\0';
485
486 query = skip_spaces(query);
487 if (!query || !*query || *query == '#')
488 continue;
489
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600490 vpr_info("query %d: \"%s\"\n", i, query);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500491
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600492 rc = ddebug_exec_query(query, modname);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500493 if (rc < 0) {
494 errs++;
495 exitcode = rc;
Joe Perchesf657fd22012-12-05 16:48:26 -0500496 } else {
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500497 nfound += rc;
Joe Perchesf657fd22012-12-05 16:48:26 -0500498 }
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500499 i++;
500 }
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600501 vpr_info("processed %d queries, with %d matches, %d errs\n",
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500502 i, nfound, errs);
503
504 if (exitcode)
505 return exitcode;
506 return nfound;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200507}
508
Jason Baron431625d2011-10-04 14:13:19 -0700509#define PREFIX_SIZE 64
510
511static int remaining(int wrote)
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100512{
Jason Baron431625d2011-10-04 14:13:19 -0700513 if (PREFIX_SIZE - wrote > 0)
514 return PREFIX_SIZE - wrote;
515 return 0;
516}
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100517
Jason Baron431625d2011-10-04 14:13:19 -0700518static char *dynamic_emit_prefix(const struct _ddebug *desc, char *buf)
519{
520 int pos_after_tid;
521 int pos = 0;
522
Joe Perches798efc62012-09-12 20:11:29 -0700523 *buf = '\0';
524
Jason Baron431625d2011-10-04 14:13:19 -0700525 if (desc->flags & _DPRINTK_FLAGS_INCL_TID) {
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100526 if (in_interrupt())
Joe Perches798efc62012-09-12 20:11:29 -0700527 pos += snprintf(buf + pos, remaining(pos), "<intr> ");
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100528 else
Jason Baron431625d2011-10-04 14:13:19 -0700529 pos += snprintf(buf + pos, remaining(pos), "[%d] ",
Joe Perches798efc62012-09-12 20:11:29 -0700530 task_pid_vnr(current));
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100531 }
Jason Baron431625d2011-10-04 14:13:19 -0700532 pos_after_tid = pos;
533 if (desc->flags & _DPRINTK_FLAGS_INCL_MODNAME)
534 pos += snprintf(buf + pos, remaining(pos), "%s:",
Joe Perches798efc62012-09-12 20:11:29 -0700535 desc->modname);
Jason Baron431625d2011-10-04 14:13:19 -0700536 if (desc->flags & _DPRINTK_FLAGS_INCL_FUNCNAME)
537 pos += snprintf(buf + pos, remaining(pos), "%s:",
Joe Perches798efc62012-09-12 20:11:29 -0700538 desc->function);
Jason Baron431625d2011-10-04 14:13:19 -0700539 if (desc->flags & _DPRINTK_FLAGS_INCL_LINENO)
Jim Cromie07100be2011-12-19 17:11:09 -0500540 pos += snprintf(buf + pos, remaining(pos), "%d:",
Joe Perches798efc62012-09-12 20:11:29 -0700541 desc->lineno);
Jason Baron431625d2011-10-04 14:13:19 -0700542 if (pos - pos_after_tid)
543 pos += snprintf(buf + pos, remaining(pos), " ");
544 if (pos >= PREFIX_SIZE)
545 buf[PREFIX_SIZE - 1] = '\0';
Joe Perches6c2140e2011-08-11 14:36:25 -0400546
Jason Baron431625d2011-10-04 14:13:19 -0700547 return buf;
Joe Perches6c2140e2011-08-11 14:36:25 -0400548}
549
Joe Perches906d2012014-09-24 11:17:56 -0700550void __dynamic_pr_debug(struct _ddebug *descriptor, const char *fmt, ...)
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100551{
552 va_list args;
Jason Baron431625d2011-10-04 14:13:19 -0700553 struct va_format vaf;
554 char buf[PREFIX_SIZE];
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100555
556 BUG_ON(!descriptor);
557 BUG_ON(!fmt);
558
559 va_start(args, fmt);
Joe Perches798efc62012-09-12 20:11:29 -0700560
Jason Baron431625d2011-10-04 14:13:19 -0700561 vaf.fmt = fmt;
562 vaf.va = &args;
Joe Perches798efc62012-09-12 20:11:29 -0700563
Joe Perches906d2012014-09-24 11:17:56 -0700564 printk(KERN_DEBUG "%s%pV", dynamic_emit_prefix(descriptor, buf), &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700565
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100566 va_end(args);
Bart Van Assche8ba6ebf2011-01-23 17:17:24 +0100567}
568EXPORT_SYMBOL(__dynamic_pr_debug);
569
Joe Perches906d2012014-09-24 11:17:56 -0700570void __dynamic_dev_dbg(struct _ddebug *descriptor,
Joe Perchescbc46632011-08-11 14:36:21 -0400571 const struct device *dev, const char *fmt, ...)
572{
573 struct va_format vaf;
574 va_list args;
Joe Perchescbc46632011-08-11 14:36:21 -0400575
576 BUG_ON(!descriptor);
577 BUG_ON(!fmt);
578
579 va_start(args, fmt);
Joe Perches798efc62012-09-12 20:11:29 -0700580
Joe Perchescbc46632011-08-11 14:36:21 -0400581 vaf.fmt = fmt;
582 vaf.va = &args;
Joe Perches798efc62012-09-12 20:11:29 -0700583
584 if (!dev) {
Joe Perches906d2012014-09-24 11:17:56 -0700585 printk(KERN_DEBUG "(NULL device *): %pV", &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700586 } else {
587 char buf[PREFIX_SIZE];
Joe Perches798efc62012-09-12 20:11:29 -0700588
Joe Perchesa39d4a82014-12-10 15:50:15 -0800589 dev_printk_emit(LOGLEVEL_DEBUG, dev, "%s%s %s: %pV",
Joe Perches906d2012014-09-24 11:17:56 -0700590 dynamic_emit_prefix(descriptor, buf),
591 dev_driver_string(dev), dev_name(dev),
592 &vaf);
Joe Perches798efc62012-09-12 20:11:29 -0700593 }
594
Joe Perchescbc46632011-08-11 14:36:21 -0400595 va_end(args);
Joe Perchescbc46632011-08-11 14:36:21 -0400596}
597EXPORT_SYMBOL(__dynamic_dev_dbg);
598
Jason Baron0feefd92011-10-04 14:13:22 -0700599#ifdef CONFIG_NET
600
Joe Perches906d2012014-09-24 11:17:56 -0700601void __dynamic_netdev_dbg(struct _ddebug *descriptor,
602 const struct net_device *dev, const char *fmt, ...)
Jason Baronffa10cb2011-08-11 14:36:48 -0400603{
604 struct va_format vaf;
605 va_list args;
Jason Baronffa10cb2011-08-11 14:36:48 -0400606
607 BUG_ON(!descriptor);
608 BUG_ON(!fmt);
609
610 va_start(args, fmt);
Joe Perchesb004ff42012-09-12 20:12:19 -0700611
Jason Baronffa10cb2011-08-11 14:36:48 -0400612 vaf.fmt = fmt;
613 vaf.va = &args;
Joe Perchesb004ff42012-09-12 20:12:19 -0700614
615 if (dev && dev->dev.parent) {
616 char buf[PREFIX_SIZE];
Joe Perchesb004ff42012-09-12 20:12:19 -0700617
Joe Perchesa39d4a82014-12-10 15:50:15 -0800618 dev_printk_emit(LOGLEVEL_DEBUG, dev->dev.parent,
Joe Perches906d2012014-09-24 11:17:56 -0700619 "%s%s %s %s%s: %pV",
620 dynamic_emit_prefix(descriptor, buf),
621 dev_driver_string(dev->dev.parent),
622 dev_name(dev->dev.parent),
623 netdev_name(dev), netdev_reg_state(dev),
624 &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700625 } else if (dev) {
Joe Perches906d2012014-09-24 11:17:56 -0700626 printk(KERN_DEBUG "%s%s: %pV", netdev_name(dev),
627 netdev_reg_state(dev), &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700628 } else {
Joe Perches906d2012014-09-24 11:17:56 -0700629 printk(KERN_DEBUG "(NULL net_device): %pV", &vaf);
Joe Perchesb004ff42012-09-12 20:12:19 -0700630 }
631
Jason Baronffa10cb2011-08-11 14:36:48 -0400632 va_end(args);
Jason Baronffa10cb2011-08-11 14:36:48 -0400633}
634EXPORT_SYMBOL(__dynamic_netdev_dbg);
635
Jason Baron0feefd92011-10-04 14:13:22 -0700636#endif
637
Jim Cromiebc757f62011-12-19 17:12:29 -0500638#define DDEBUG_STRING_SIZE 1024
639static __initdata char ddebug_setup_string[DDEBUG_STRING_SIZE];
640
Thomas Renningera648ec02010-08-06 16:11:02 +0200641static __init int ddebug_setup_query(char *str)
642{
Jim Cromiebc757f62011-12-19 17:12:29 -0500643 if (strlen(str) >= DDEBUG_STRING_SIZE) {
Joe Perches4ad275e2011-08-11 14:36:33 -0400644 pr_warn("ddebug boot param string too large\n");
Thomas Renningera648ec02010-08-06 16:11:02 +0200645 return 0;
646 }
Jim Cromiebc757f62011-12-19 17:12:29 -0500647 strlcpy(ddebug_setup_string, str, DDEBUG_STRING_SIZE);
Thomas Renningera648ec02010-08-06 16:11:02 +0200648 return 1;
649}
650
651__setup("ddebug_query=", ddebug_setup_query);
652
Jason Barone9d376f2009-02-05 11:51:38 -0500653/*
Masatake YAMATO231821d2014-12-15 12:04:16 +0900654 * File_ops->write method for <debugfs>/dynamic_debug/control. Gathers the
Jason Barone9d376f2009-02-05 11:51:38 -0500655 * command text from userspace, parses and executes it.
656 */
Jim Cromie72814912011-12-19 17:13:07 -0500657#define USER_BUF_PAGE 4096
Jason Barone9d376f2009-02-05 11:51:38 -0500658static ssize_t ddebug_proc_write(struct file *file, const char __user *ubuf,
659 size_t len, loff_t *offp)
660{
Jim Cromie72814912011-12-19 17:13:07 -0500661 char *tmpbuf;
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200662 int ret;
Jason Barone9d376f2009-02-05 11:51:38 -0500663
664 if (len == 0)
665 return 0;
Jim Cromie72814912011-12-19 17:13:07 -0500666 if (len > USER_BUF_PAGE - 1) {
667 pr_warn("expected <%d bytes into control\n", USER_BUF_PAGE);
Jason Barone9d376f2009-02-05 11:51:38 -0500668 return -E2BIG;
Jim Cromie72814912011-12-19 17:13:07 -0500669 }
Al Viro16e5c1f2015-12-24 00:06:05 -0500670 tmpbuf = memdup_user_nul(ubuf, len);
671 if (IS_ERR(tmpbuf))
672 return PTR_ERR(tmpbuf);
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600673 vpr_info("read %d bytes from userspace\n", (int)len);
Jason Barone9d376f2009-02-05 11:51:38 -0500674
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600675 ret = ddebug_exec_queries(tmpbuf, NULL);
Jim Cromie72814912011-12-19 17:13:07 -0500676 kfree(tmpbuf);
Jim Cromie85f7f6c2011-12-19 17:13:21 -0500677 if (ret < 0)
Thomas Renningerfd89cfb2010-08-06 16:11:01 +0200678 return ret;
Jason Barone9d376f2009-02-05 11:51:38 -0500679
680 *offp += len;
681 return len;
682}
683
684/*
685 * Set the iterator to point to the first _ddebug object
686 * and return a pointer to that first object. Returns
687 * NULL if there are no _ddebugs at all.
688 */
689static struct _ddebug *ddebug_iter_first(struct ddebug_iter *iter)
690{
691 if (list_empty(&ddebug_tables)) {
692 iter->table = NULL;
693 iter->idx = 0;
694 return NULL;
695 }
696 iter->table = list_entry(ddebug_tables.next,
697 struct ddebug_table, link);
698 iter->idx = 0;
699 return &iter->table->ddebugs[iter->idx];
700}
701
702/*
703 * Advance the iterator to point to the next _ddebug
704 * object from the one the iterator currently points at,
705 * and returns a pointer to the new _ddebug. Returns
706 * NULL if the iterator has seen all the _ddebugs.
707 */
708static struct _ddebug *ddebug_iter_next(struct ddebug_iter *iter)
709{
710 if (iter->table == NULL)
711 return NULL;
712 if (++iter->idx == iter->table->num_ddebugs) {
713 /* iterate to next table */
714 iter->idx = 0;
715 if (list_is_last(&iter->table->link, &ddebug_tables)) {
716 iter->table = NULL;
717 return NULL;
718 }
719 iter->table = list_entry(iter->table->link.next,
720 struct ddebug_table, link);
721 }
722 return &iter->table->ddebugs[iter->idx];
723}
724
725/*
726 * Seq_ops start method. Called at the start of every
727 * read() call from userspace. Takes the ddebug_lock and
728 * seeks the seq_file's iterator to the given position.
729 */
730static void *ddebug_proc_start(struct seq_file *m, loff_t *pos)
731{
732 struct ddebug_iter *iter = m->private;
733 struct _ddebug *dp;
734 int n = *pos;
735
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600736 vpr_info("called m=%p *pos=%lld\n", m, (unsigned long long)*pos);
Jason Barone9d376f2009-02-05 11:51:38 -0500737
738 mutex_lock(&ddebug_lock);
739
740 if (!n)
741 return SEQ_START_TOKEN;
742 if (n < 0)
743 return NULL;
744 dp = ddebug_iter_first(iter);
745 while (dp != NULL && --n > 0)
746 dp = ddebug_iter_next(iter);
747 return dp;
748}
749
750/*
751 * Seq_ops next method. Called several times within a read()
752 * call from userspace, with ddebug_lock held. Walks to the
753 * next _ddebug object with a special case for the header line.
754 */
755static void *ddebug_proc_next(struct seq_file *m, void *p, loff_t *pos)
756{
757 struct ddebug_iter *iter = m->private;
758 struct _ddebug *dp;
759
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600760 vpr_info("called m=%p p=%p *pos=%lld\n",
Joe Perchesf657fd22012-12-05 16:48:26 -0500761 m, p, (unsigned long long)*pos);
Jason Barone9d376f2009-02-05 11:51:38 -0500762
763 if (p == SEQ_START_TOKEN)
764 dp = ddebug_iter_first(iter);
765 else
766 dp = ddebug_iter_next(iter);
767 ++*pos;
768 return dp;
769}
770
771/*
772 * Seq_ops show method. Called several times within a read()
773 * call from userspace, with ddebug_lock held. Formats the
774 * current _ddebug as a single human-readable line, with a
775 * special case for the header line.
776 */
777static int ddebug_proc_show(struct seq_file *m, void *p)
778{
779 struct ddebug_iter *iter = m->private;
780 struct _ddebug *dp = p;
Jim Cromieeb8cbd82020-07-19 17:10:47 -0600781 struct flagsbuf flags;
Jason Barone9d376f2009-02-05 11:51:38 -0500782
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600783 vpr_info("called m=%p p=%p\n", m, p);
Jason Barone9d376f2009-02-05 11:51:38 -0500784
785 if (p == SEQ_START_TOKEN) {
786 seq_puts(m,
Joe Perchesf657fd22012-12-05 16:48:26 -0500787 "# filename:lineno [module]function flags format\n");
Jason Barone9d376f2009-02-05 11:51:38 -0500788 return 0;
789 }
790
Jim Cromie5ca7d2a2011-12-19 17:12:44 -0500791 seq_printf(m, "%s:%u [%s]%s =%s \"",
Joe Perchesf657fd22012-12-05 16:48:26 -0500792 trim_prefix(dp->filename), dp->lineno,
793 iter->table->mod_name, dp->function,
Jim Cromieeb8cbd82020-07-19 17:10:47 -0600794 ddebug_describe_flags(dp->flags, &flags));
Jason Barone9d376f2009-02-05 11:51:38 -0500795 seq_escape(m, dp->format, "\t\r\n\"");
796 seq_puts(m, "\"\n");
797
798 return 0;
799}
800
801/*
802 * Seq_ops stop method. Called at the end of each read()
803 * call from userspace. Drops ddebug_lock.
804 */
805static void ddebug_proc_stop(struct seq_file *m, void *p)
806{
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600807 vpr_info("called m=%p p=%p\n", m, p);
Jason Barone9d376f2009-02-05 11:51:38 -0500808 mutex_unlock(&ddebug_lock);
809}
810
811static const struct seq_operations ddebug_proc_seqops = {
812 .start = ddebug_proc_start,
813 .next = ddebug_proc_next,
814 .show = ddebug_proc_show,
815 .stop = ddebug_proc_stop
816};
817
818/*
Jim Cromie07100be2011-12-19 17:11:09 -0500819 * File_ops->open method for <debugfs>/dynamic_debug/control. Does
820 * the seq_file setup dance, and also creates an iterator to walk the
821 * _ddebugs. Note that we create a seq_file always, even for O_WRONLY
822 * files where it's not needed, as doing so simplifies the ->release
823 * method.
Jason Barone9d376f2009-02-05 11:51:38 -0500824 */
825static int ddebug_proc_open(struct inode *inode, struct file *file)
826{
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600827 vpr_info("called\n");
Rob Jones4bad78c2014-10-13 15:51:32 -0700828 return seq_open_private(file, &ddebug_proc_seqops,
829 sizeof(struct ddebug_iter));
Jason Barone9d376f2009-02-05 11:51:38 -0500830}
831
832static const struct file_operations ddebug_proc_fops = {
833 .owner = THIS_MODULE,
834 .open = ddebug_proc_open,
835 .read = seq_read,
836 .llseek = seq_lseek,
837 .release = seq_release_private,
838 .write = ddebug_proc_write
839};
840
841/*
842 * Allocate a new ddebug_table for the given module
843 * and add it to the global list.
844 */
845int ddebug_add_module(struct _ddebug *tab, unsigned int n,
846 const char *name)
847{
848 struct ddebug_table *dt;
Rasmus Villemoes3e406b12015-11-06 16:30:15 -0800849 const char *new_name;
Jason Barone9d376f2009-02-05 11:51:38 -0500850
851 dt = kzalloc(sizeof(*dt), GFP_KERNEL);
852 if (dt == NULL)
853 return -ENOMEM;
Rasmus Villemoes3e406b12015-11-06 16:30:15 -0800854 new_name = kstrdup_const(name, GFP_KERNEL);
Jason Barone9d376f2009-02-05 11:51:38 -0500855 if (new_name == NULL) {
856 kfree(dt);
857 return -ENOMEM;
858 }
859 dt->mod_name = new_name;
860 dt->num_ddebugs = n;
Jason Barone9d376f2009-02-05 11:51:38 -0500861 dt->ddebugs = tab;
862
863 mutex_lock(&ddebug_lock);
864 list_add_tail(&dt->link, &ddebug_tables);
865 mutex_unlock(&ddebug_lock);
866
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600867 vpr_info("%u debug prints in module %s\n", n, dt->mod_name);
Jason Barone9d376f2009-02-05 11:51:38 -0500868 return 0;
869}
870EXPORT_SYMBOL_GPL(ddebug_add_module);
871
Jim Cromie6ab676e2012-04-27 14:30:37 -0600872/* helper for ddebug_dyndbg_(boot|module)_param_cb */
873static int ddebug_dyndbg_param_cb(char *param, char *val,
874 const char *modname, int on_err)
Jim Cromieb48420c2012-04-27 14:30:35 -0600875{
Jim Cromieb48420c2012-04-27 14:30:35 -0600876 char *sep;
877
878 sep = strchr(param, '.');
879 if (sep) {
Jim Cromie6ab676e2012-04-27 14:30:37 -0600880 /* needed only for ddebug_dyndbg_boot_param_cb */
Jim Cromieb48420c2012-04-27 14:30:35 -0600881 *sep = '\0';
882 modname = param;
883 param = sep + 1;
884 }
885 if (strcmp(param, "dyndbg"))
Jim Cromie6ab676e2012-04-27 14:30:37 -0600886 return on_err; /* determined by caller */
Jim Cromieb48420c2012-04-27 14:30:35 -0600887
Jim Cromie8e59b5c2012-04-27 14:30:40 -0600888 ddebug_exec_queries((val ? val : "+p"), modname);
889
Jim Cromieb48420c2012-04-27 14:30:35 -0600890 return 0; /* query failure shouldnt stop module load */
891}
892
Jim Cromie6ab676e2012-04-27 14:30:37 -0600893/* handle both dyndbg and $module.dyndbg params at boot */
894static int ddebug_dyndbg_boot_param_cb(char *param, char *val,
Luis R. Rodriguezecc86172015-03-30 16:20:03 -0700895 const char *unused, void *arg)
Jim Cromieb48420c2012-04-27 14:30:35 -0600896{
Jim Cromie6ab676e2012-04-27 14:30:37 -0600897 vpr_info("%s=\"%s\"\n", param, val);
898 return ddebug_dyndbg_param_cb(param, val, NULL, 0);
899}
Jim Cromieb48420c2012-04-27 14:30:35 -0600900
Jim Cromie6ab676e2012-04-27 14:30:37 -0600901/*
902 * modprobe foo finds foo.params in boot-args, strips "foo.", and
903 * passes them to load_module(). This callback gets unknown params,
904 * processes dyndbg params, rejects others.
905 */
906int ddebug_dyndbg_module_param_cb(char *param, char *val, const char *module)
907{
908 vpr_info("module: %s %s=\"%s\"\n", module, param, val);
909 return ddebug_dyndbg_param_cb(param, val, module, -ENOENT);
Jim Cromieb48420c2012-04-27 14:30:35 -0600910}
911
Jason Barone9d376f2009-02-05 11:51:38 -0500912static void ddebug_table_free(struct ddebug_table *dt)
913{
914 list_del_init(&dt->link);
Rasmus Villemoes3e406b12015-11-06 16:30:15 -0800915 kfree_const(dt->mod_name);
Jason Barone9d376f2009-02-05 11:51:38 -0500916 kfree(dt);
917}
918
919/*
920 * Called in response to a module being unloaded. Removes
921 * any ddebug_table's which point at the module.
922 */
Yehuda Sadehff49d742010-07-03 13:07:35 +1000923int ddebug_remove_module(const char *mod_name)
Jason Barone9d376f2009-02-05 11:51:38 -0500924{
925 struct ddebug_table *dt, *nextdt;
926 int ret = -ENOENT;
927
Jim Cromieb8ccd5d2012-04-27 14:30:32 -0600928 vpr_info("removing module \"%s\"\n", mod_name);
Jason Barone9d376f2009-02-05 11:51:38 -0500929
930 mutex_lock(&ddebug_lock);
931 list_for_each_entry_safe(dt, nextdt, &ddebug_tables, link) {
932 if (!strcmp(dt->mod_name, mod_name)) {
933 ddebug_table_free(dt);
934 ret = 0;
935 }
936 }
937 mutex_unlock(&ddebug_lock);
938 return ret;
939}
940EXPORT_SYMBOL_GPL(ddebug_remove_module);
941
942static void ddebug_remove_all_tables(void)
943{
944 mutex_lock(&ddebug_lock);
945 while (!list_empty(&ddebug_tables)) {
946 struct ddebug_table *dt = list_entry(ddebug_tables.next,
947 struct ddebug_table,
948 link);
949 ddebug_table_free(dt);
950 }
951 mutex_unlock(&ddebug_lock);
952}
953
Thomas Renninger6a5c0832010-08-06 16:11:03 +0200954static __initdata int ddebug_init_success;
955
956static int __init dynamic_debug_init_debugfs(void)
Jason Barone9d376f2009-02-05 11:51:38 -0500957{
958 struct dentry *dir, *file;
Thomas Renninger6a5c0832010-08-06 16:11:03 +0200959
960 if (!ddebug_init_success)
961 return -ENODEV;
Jason Barone9d376f2009-02-05 11:51:38 -0500962
963 dir = debugfs_create_dir("dynamic_debug", NULL);
964 if (!dir)
965 return -ENOMEM;
966 file = debugfs_create_file("control", 0644, dir, NULL,
967 &ddebug_proc_fops);
968 if (!file) {
969 debugfs_remove(dir);
970 return -ENOMEM;
971 }
Thomas Renninger6a5c0832010-08-06 16:11:03 +0200972 return 0;
973}
974
975static int __init dynamic_debug_init(void)
976{
977 struct _ddebug *iter, *iter_start;
978 const char *modname = NULL;
Jim Cromieb48420c2012-04-27 14:30:35 -0600979 char *cmdline;
Thomas Renninger6a5c0832010-08-06 16:11:03 +0200980 int ret = 0;
Jim Cromie41076922012-04-27 14:30:39 -0600981 int n = 0, entries = 0, modct = 0;
982 int verbose_bytes = 0;
Thomas Renninger6a5c0832010-08-06 16:11:03 +0200983
Jim Cromieb5b78f82011-12-19 17:12:54 -0500984 if (__start___verbose == __stop___verbose) {
Joe Perchesf657fd22012-12-05 16:48:26 -0500985 pr_warn("_ddebug table is empty in a CONFIG_DYNAMIC_DEBUG build\n");
Jim Cromieb5b78f82011-12-19 17:12:54 -0500986 return 1;
Jason Barone9d376f2009-02-05 11:51:38 -0500987 }
Jim Cromieb5b78f82011-12-19 17:12:54 -0500988 iter = __start___verbose;
989 modname = iter->modname;
990 iter_start = iter;
991 for (; iter < __stop___verbose; iter++) {
Jim Cromie41076922012-04-27 14:30:39 -0600992 entries++;
993 verbose_bytes += strlen(iter->modname) + strlen(iter->function)
994 + strlen(iter->filename) + strlen(iter->format);
995
Jim Cromieb5b78f82011-12-19 17:12:54 -0500996 if (strcmp(modname, iter->modname)) {
Jim Cromie41076922012-04-27 14:30:39 -0600997 modct++;
Jim Cromieb5b78f82011-12-19 17:12:54 -0500998 ret = ddebug_add_module(iter_start, n, modname);
999 if (ret)
Jim Cromieaf442392012-04-27 14:30:38 -06001000 goto out_err;
Jim Cromieb5b78f82011-12-19 17:12:54 -05001001 n = 0;
1002 modname = iter->modname;
1003 iter_start = iter;
1004 }
1005 n++;
1006 }
1007 ret = ddebug_add_module(iter_start, n, modname);
1008 if (ret)
Jim Cromieaf442392012-04-27 14:30:38 -06001009 goto out_err;
Thomas Renningera648ec02010-08-06 16:11:02 +02001010
Jim Cromieaf442392012-04-27 14:30:38 -06001011 ddebug_init_success = 1;
Joe Perchesf657fd22012-12-05 16:48:26 -05001012 vpr_info("%d modules, %d entries and %d bytes in ddebug tables, %d bytes in (readonly) verbose section\n",
1013 modct, entries, (int)(modct * sizeof(struct ddebug_table)),
1014 verbose_bytes + (int)(__stop___verbose - __start___verbose));
Jim Cromieaf442392012-04-27 14:30:38 -06001015
1016 /* apply ddebug_query boot param, dont unload tables on err */
Thomas Renningera648ec02010-08-06 16:11:02 +02001017 if (ddebug_setup_string[0] != '\0') {
Joe Perchesf657fd22012-12-05 16:48:26 -05001018 pr_warn("ddebug_query param name is deprecated, change it to dyndbg\n");
Jim Cromie8e59b5c2012-04-27 14:30:40 -06001019 ret = ddebug_exec_queries(ddebug_setup_string, NULL);
Jim Cromie85f7f6c2011-12-19 17:13:21 -05001020 if (ret < 0)
Joe Perchesf657fd22012-12-05 16:48:26 -05001021 pr_warn("Invalid ddebug boot param %s\n",
Joe Perches4ad275e2011-08-11 14:36:33 -04001022 ddebug_setup_string);
Thomas Renningera648ec02010-08-06 16:11:02 +02001023 else
Jim Cromie85f7f6c2011-12-19 17:13:21 -05001024 pr_info("%d changes by ddebug_query\n", ret);
Jason Barone9d376f2009-02-05 11:51:38 -05001025 }
Jim Cromieb48420c2012-04-27 14:30:35 -06001026 /* now that ddebug tables are loaded, process all boot args
1027 * again to find and activate queries given in dyndbg params.
1028 * While this has already been done for known boot params, it
1029 * ignored the unknown ones (dyndbg in particular). Reusing
1030 * parse_args avoids ad-hoc parsing. This will also attempt
1031 * to activate queries for not-yet-loaded modules, which is
1032 * slightly noisy if verbose, but harmless.
1033 */
1034 cmdline = kstrdup(saved_command_line, GFP_KERNEL);
1035 parse_args("dyndbg params", cmdline, NULL,
Luis R. Rodriguezecc86172015-03-30 16:20:03 -07001036 0, 0, 0, NULL, &ddebug_dyndbg_boot_param_cb);
Jim Cromieb48420c2012-04-27 14:30:35 -06001037 kfree(cmdline);
Jim Cromieaf442392012-04-27 14:30:38 -06001038 return 0;
Thomas Renningera648ec02010-08-06 16:11:02 +02001039
Jim Cromieaf442392012-04-27 14:30:38 -06001040out_err:
1041 ddebug_remove_all_tables();
Jason Barone9d376f2009-02-05 11:51:38 -05001042 return 0;
1043}
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001044/* Allow early initialization for boot messages via boot param */
Jim Cromie3ec56522012-04-27 14:30:42 -06001045early_initcall(dynamic_debug_init);
Jim Cromieb48420c2012-04-27 14:30:35 -06001046
Thomas Renninger6a5c0832010-08-06 16:11:03 +02001047/* Debugfs setup must be done later */
Jim Cromie3ec56522012-04-27 14:30:42 -06001048fs_initcall(dynamic_debug_init_debugfs);