2009-02-06 00:51:38 +08:00
|
|
|
/*
|
|
|
|
* lib/dynamic_debug.c
|
|
|
|
*
|
|
|
|
* make pr_debug()/dev_dbg() calls runtime configurable based upon their
|
|
|
|
* source module.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Jason Baron <jbaron@redhat.com>
|
|
|
|
* By Greg Banks <gnb@melbourne.sgi.com>
|
|
|
|
* Copyright (c) 2008 Silicon Graphics Inc. All Rights Reserved.
|
2011-01-24 00:17:24 +08:00
|
|
|
* Copyright (C) 2011 Bart Van Assche. All Rights Reserved.
|
2014-01-24 07:54:14 +08:00
|
|
|
* Copyright (C) 2013 Du, Changbin <changbin.du@gmail.com>
|
2009-02-06 00:51:38 +08:00
|
|
|
*/
|
|
|
|
|
2020-07-20 07:10:57 +08:00
|
|
|
#define pr_fmt(fmt) "dyndbg: " fmt
|
2011-08-12 02:36:33 +08:00
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
#include <linux/kallsyms.h>
|
|
|
|
#include <linux/types.h>
|
2009-02-06 00:51:38 +08:00
|
|
|
#include <linux/mutex.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/proc_fs.h>
|
2009-02-06 00:51:38 +08:00
|
|
|
#include <linux/seq_file.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/list.h>
|
|
|
|
#include <linux/sysctl.h>
|
2009-02-06 00:51:38 +08:00
|
|
|
#include <linux/ctype.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/string.h>
|
2014-01-24 07:54:14 +08:00
|
|
|
#include <linux/parser.h>
|
2013-05-01 06:27:32 +08:00
|
|
|
#include <linux/string_helpers.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/uaccess.h>
|
2009-02-06 00:51:38 +08:00
|
|
|
#include <linux/dynamic_debug.h>
|
|
|
|
#include <linux/debugfs.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 16:04:11 +08:00
|
|
|
#include <linux/slab.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/jump_label.h>
|
2011-01-24 00:17:24 +08:00
|
|
|
#include <linux/hardirq.h>
|
2011-02-04 07:59:58 +08:00
|
|
|
#include <linux/sched.h>
|
2012-05-08 07:47:32 +08:00
|
|
|
#include <linux/device.h>
|
2011-08-12 02:36:48 +08:00
|
|
|
#include <linux/netdevice.h>
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2019-05-01 18:48:13 +08:00
|
|
|
#include <rdma/ib_verbs.h>
|
|
|
|
|
2020-07-20 07:10:45 +08:00
|
|
|
extern struct _ddebug __start___dyndbg[];
|
|
|
|
extern struct _ddebug __stop___dyndbg[];
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
struct ddebug_table {
|
|
|
|
struct list_head link;
|
2015-11-07 08:30:15 +08:00
|
|
|
const char *mod_name;
|
2009-02-06 00:51:38 +08:00
|
|
|
unsigned int num_ddebugs;
|
|
|
|
struct _ddebug *ddebugs;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ddebug_query {
|
|
|
|
const char *filename;
|
|
|
|
const char *module;
|
|
|
|
const char *function;
|
|
|
|
const char *format;
|
|
|
|
unsigned int first_lineno, last_lineno;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ddebug_iter {
|
|
|
|
struct ddebug_table *table;
|
|
|
|
unsigned int idx;
|
|
|
|
};
|
|
|
|
|
2020-07-20 07:10:55 +08:00
|
|
|
struct flag_settings {
|
|
|
|
unsigned int flags;
|
|
|
|
unsigned int mask;
|
|
|
|
};
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
static DEFINE_MUTEX(ddebug_lock);
|
|
|
|
static LIST_HEAD(ddebug_tables);
|
2012-12-06 05:48:26 +08:00
|
|
|
static int verbose;
|
2011-12-20 06:12:24 +08:00
|
|
|
module_param(verbose, int, 0644);
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2011-12-20 06:13:12 +08:00
|
|
|
/* Return the path relative to source root */
|
|
|
|
static inline const char *trim_prefix(const char *path)
|
|
|
|
{
|
|
|
|
int skip = strlen(__FILE__) - strlen("lib/dynamic_debug.c");
|
|
|
|
|
|
|
|
if (strncmp(path, __FILE__, skip))
|
|
|
|
skip = 0; /* prefix mismatch, don't skip */
|
|
|
|
|
|
|
|
return path + skip;
|
|
|
|
}
|
|
|
|
|
2011-01-24 00:17:24 +08:00
|
|
|
static struct { unsigned flag:8; char opt_char; } opt_array[] = {
|
|
|
|
{ _DPRINTK_FLAGS_PRINT, 'p' },
|
|
|
|
{ _DPRINTK_FLAGS_INCL_MODNAME, 'm' },
|
|
|
|
{ _DPRINTK_FLAGS_INCL_FUNCNAME, 'f' },
|
|
|
|
{ _DPRINTK_FLAGS_INCL_LINENO, 'l' },
|
|
|
|
{ _DPRINTK_FLAGS_INCL_TID, 't' },
|
2011-12-20 06:12:44 +08:00
|
|
|
{ _DPRINTK_FLAGS_NONE, '_' },
|
2011-01-24 00:17:24 +08:00
|
|
|
};
|
|
|
|
|
2020-07-20 07:10:47 +08:00
|
|
|
struct flagsbuf { char buf[ARRAY_SIZE(opt_array)+1]; };
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
/* format a string into buf[] which describes the _ddebug's flags */
|
2020-07-20 07:10:47 +08:00
|
|
|
static char *ddebug_describe_flags(unsigned int flags, struct flagsbuf *fb)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
2020-07-20 07:10:47 +08:00
|
|
|
char *p = fb->buf;
|
2011-01-24 00:17:24 +08:00
|
|
|
int i;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2011-01-24 00:17:24 +08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(opt_array); ++i)
|
2020-07-20 07:10:47 +08:00
|
|
|
if (flags & opt_array[i].flag)
|
2011-01-24 00:17:24 +08:00
|
|
|
*p++ = opt_array[i].opt_char;
|
2020-07-20 07:10:47 +08:00
|
|
|
if (p == fb->buf)
|
2011-12-20 06:12:44 +08:00
|
|
|
*p++ = '_';
|
2009-02-06 00:51:38 +08:00
|
|
|
*p = '\0';
|
|
|
|
|
2020-07-20 07:10:47 +08:00
|
|
|
return fb->buf;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
|
2020-07-20 07:10:44 +08:00
|
|
|
#define vnpr_info(lvl, fmt, ...) \
|
2012-04-28 04:30:32 +08:00
|
|
|
do { \
|
2020-07-20 07:10:44 +08:00
|
|
|
if (verbose >= lvl) \
|
2012-12-06 05:48:26 +08:00
|
|
|
pr_info(fmt, ##__VA_ARGS__); \
|
2011-12-20 06:13:16 +08:00
|
|
|
} while (0)
|
|
|
|
|
2020-07-20 07:10:44 +08:00
|
|
|
#define vpr_info(fmt, ...) vnpr_info(1, fmt, ##__VA_ARGS__)
|
|
|
|
#define v2pr_info(fmt, ...) vnpr_info(2, fmt, ##__VA_ARGS__)
|
|
|
|
|
2012-12-06 05:48:26 +08:00
|
|
|
static void vpr_info_dq(const struct ddebug_query *query, const char *msg)
|
|
|
|
{
|
|
|
|
/* trim any trailing newlines */
|
|
|
|
int fmtlen = 0;
|
|
|
|
|
|
|
|
if (query->format) {
|
|
|
|
fmtlen = strlen(query->format);
|
|
|
|
while (fmtlen && query->format[fmtlen - 1] == '\n')
|
|
|
|
fmtlen--;
|
|
|
|
}
|
|
|
|
|
|
|
|
vpr_info("%s: func=\"%s\" file=\"%s\" module=\"%s\" format=\"%.*s\" lineno=%u-%u\n",
|
|
|
|
msg,
|
2020-07-20 07:10:51 +08:00
|
|
|
query->function ?: "",
|
|
|
|
query->filename ?: "",
|
|
|
|
query->module ?: "",
|
|
|
|
fmtlen, query->format ?: "",
|
2012-12-06 05:48:26 +08:00
|
|
|
query->first_lineno, query->last_lineno);
|
|
|
|
}
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
/*
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
* Search the tables for _ddebug's which match the given `query' and
|
|
|
|
* apply the `flags' and `mask' to them. Returns number of matching
|
|
|
|
* callsites, normally the same as number of changes. If verbose,
|
|
|
|
* logs the changes. Takes ddebug_lock.
|
2009-02-06 00:51:38 +08:00
|
|
|
*/
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
static int ddebug_change(const struct ddebug_query *query,
|
2020-07-20 07:10:55 +08:00
|
|
|
struct flag_settings *modifiers)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
struct ddebug_table *dt;
|
|
|
|
unsigned int newflags;
|
|
|
|
unsigned int nfound = 0;
|
2020-07-20 07:10:47 +08:00
|
|
|
struct flagsbuf fbuf;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
/* search for matching ddebugs */
|
|
|
|
mutex_lock(&ddebug_lock);
|
|
|
|
list_for_each_entry(dt, &ddebug_tables, link) {
|
|
|
|
|
|
|
|
/* match against the module name */
|
2014-01-24 07:54:14 +08:00
|
|
|
if (query->module &&
|
|
|
|
!match_wildcard(query->module, dt->mod_name))
|
2009-02-06 00:51:38 +08:00
|
|
|
continue;
|
|
|
|
|
2012-12-06 05:48:26 +08:00
|
|
|
for (i = 0; i < dt->num_ddebugs; i++) {
|
2009-02-06 00:51:38 +08:00
|
|
|
struct _ddebug *dp = &dt->ddebugs[i];
|
|
|
|
|
|
|
|
/* match against the source filename */
|
2011-12-20 06:12:39 +08:00
|
|
|
if (query->filename &&
|
2014-01-24 07:54:14 +08:00
|
|
|
!match_wildcard(query->filename, dp->filename) &&
|
|
|
|
!match_wildcard(query->filename,
|
|
|
|
kbasename(dp->filename)) &&
|
|
|
|
!match_wildcard(query->filename,
|
|
|
|
trim_prefix(dp->filename)))
|
2009-02-06 00:51:38 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* match against the function */
|
2011-12-20 06:12:39 +08:00
|
|
|
if (query->function &&
|
2014-01-24 07:54:14 +08:00
|
|
|
!match_wildcard(query->function, dp->function))
|
2009-02-06 00:51:38 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* match against the format */
|
2020-07-20 07:10:56 +08:00
|
|
|
if (query->format) {
|
|
|
|
if (*query->format == '^') {
|
|
|
|
char *p;
|
|
|
|
/* anchored search. match must be at beginning */
|
|
|
|
p = strstr(dp->format, query->format+1);
|
|
|
|
if (p != dp->format)
|
|
|
|
continue;
|
|
|
|
} else if (!strstr(dp->format, query->format))
|
|
|
|
continue;
|
|
|
|
}
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
/* match against the line number range */
|
|
|
|
if (query->first_lineno &&
|
|
|
|
dp->lineno < query->first_lineno)
|
|
|
|
continue;
|
|
|
|
if (query->last_lineno &&
|
|
|
|
dp->lineno > query->last_lineno)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
nfound++;
|
|
|
|
|
2020-07-20 07:10:55 +08:00
|
|
|
newflags = (dp->flags & modifiers->mask) | modifiers->flags;
|
2009-02-06 00:51:38 +08:00
|
|
|
if (newflags == dp->flags)
|
|
|
|
continue;
|
2018-12-30 23:14:15 +08:00
|
|
|
#ifdef CONFIG_JUMP_LABEL
|
2016-08-04 04:46:39 +08:00
|
|
|
if (dp->flags & _DPRINTK_FLAGS_PRINT) {
|
2020-07-20 07:10:55 +08:00
|
|
|
if (!(modifiers->flags & _DPRINTK_FLAGS_PRINT))
|
2016-08-04 04:46:39 +08:00
|
|
|
static_branch_disable(&dp->key.dd_key_true);
|
2020-07-20 07:10:55 +08:00
|
|
|
} else if (modifiers->flags & _DPRINTK_FLAGS_PRINT)
|
2016-08-04 04:46:39 +08:00
|
|
|
static_branch_enable(&dp->key.dd_key_true);
|
|
|
|
#endif
|
2009-02-06 00:51:38 +08:00
|
|
|
dp->flags = newflags;
|
2020-07-20 07:10:44 +08:00
|
|
|
v2pr_info("changed %s:%d [%s]%s =%s\n",
|
2012-12-06 05:48:26 +08:00
|
|
|
trim_prefix(dp->filename), dp->lineno,
|
|
|
|
dt->mod_name, dp->function,
|
2020-07-20 07:10:47 +08:00
|
|
|
ddebug_describe_flags(dp->flags, &fbuf));
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&ddebug_lock);
|
|
|
|
|
|
|
|
if (!nfound && verbose)
|
2011-08-12 02:36:33 +08:00
|
|
|
pr_info("no matches for query\n");
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
|
|
|
|
return nfound;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Split the buffer `buf' into space-separated words.
|
2009-02-06 09:54:26 +08:00
|
|
|
* Handles simple " and ' quoting, i.e. without nested,
|
|
|
|
* embedded or escaped \". Return the number of words
|
|
|
|
* or <0 on error.
|
2009-02-06 00:51:38 +08:00
|
|
|
*/
|
|
|
|
static int ddebug_tokenize(char *buf, char *words[], int maxwords)
|
|
|
|
{
|
|
|
|
int nwords = 0;
|
|
|
|
|
2009-02-06 09:54:26 +08:00
|
|
|
while (*buf) {
|
|
|
|
char *end;
|
|
|
|
|
|
|
|
/* Skip leading whitespace */
|
2009-12-15 10:01:06 +08:00
|
|
|
buf = skip_spaces(buf);
|
2009-02-06 09:54:26 +08:00
|
|
|
if (!*buf)
|
|
|
|
break; /* oh, it was trailing whitespace */
|
2011-12-20 06:13:03 +08:00
|
|
|
if (*buf == '#')
|
|
|
|
break; /* token starts comment, skip rest of line */
|
2009-02-06 09:54:26 +08:00
|
|
|
|
2011-12-20 06:11:09 +08:00
|
|
|
/* find `end' of word, whitespace separated or quoted */
|
2009-02-06 09:54:26 +08:00
|
|
|
if (*buf == '"' || *buf == '\'') {
|
|
|
|
int quote = *buf++;
|
2012-12-06 05:48:26 +08:00
|
|
|
for (end = buf; *end && *end != quote; end++)
|
2009-02-06 09:54:26 +08:00
|
|
|
;
|
2012-12-06 05:48:27 +08:00
|
|
|
if (!*end) {
|
|
|
|
pr_err("unclosed quote: %s\n", buf);
|
2009-02-06 09:54:26 +08:00
|
|
|
return -EINVAL; /* unclosed quote */
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
2009-02-06 09:54:26 +08:00
|
|
|
} else {
|
2012-12-06 05:48:26 +08:00
|
|
|
for (end = buf; *end && !isspace(*end); end++)
|
2009-02-06 09:54:26 +08:00
|
|
|
;
|
|
|
|
BUG_ON(end == buf);
|
|
|
|
}
|
|
|
|
|
2011-12-20 06:11:09 +08:00
|
|
|
/* `buf' is start of word, `end' is one past its end */
|
2012-12-06 05:48:27 +08:00
|
|
|
if (nwords == maxwords) {
|
|
|
|
pr_err("too many words, legal max <=%d\n", maxwords);
|
2009-02-06 09:54:26 +08:00
|
|
|
return -EINVAL; /* ran out of words[] before bytes */
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
2009-02-06 09:54:26 +08:00
|
|
|
if (*end)
|
|
|
|
*end++ = '\0'; /* terminate the word */
|
|
|
|
words[nwords++] = buf;
|
|
|
|
buf = end;
|
|
|
|
}
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
if (verbose) {
|
|
|
|
int i;
|
2011-08-12 02:36:33 +08:00
|
|
|
pr_info("split into words:");
|
2012-12-06 05:48:26 +08:00
|
|
|
for (i = 0; i < nwords; i++)
|
2011-08-12 02:36:33 +08:00
|
|
|
pr_cont(" \"%s\"", words[i]);
|
|
|
|
pr_cont("\n");
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return nwords;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parse a single line number. Note that the empty string ""
|
|
|
|
* is treated as a special case and converted to zero, which
|
|
|
|
* is later treated as a "don't care" value.
|
|
|
|
*/
|
|
|
|
static inline int parse_lineno(const char *str, unsigned int *val)
|
|
|
|
{
|
|
|
|
BUG_ON(str == NULL);
|
|
|
|
if (*str == '\0') {
|
|
|
|
*val = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
2014-01-28 09:06:59 +08:00
|
|
|
if (kstrtouint(str, 10, val) < 0) {
|
2012-12-06 05:48:27 +08:00
|
|
|
pr_err("bad line-number: %s\n", str);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
return 0;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
|
2020-07-20 07:10:52 +08:00
|
|
|
static int parse_linerange(struct ddebug_query *query, const char *first)
|
|
|
|
{
|
|
|
|
char *last = strchr(first, '-');
|
|
|
|
|
|
|
|
if (query->first_lineno || query->last_lineno) {
|
|
|
|
pr_err("match-spec: line used 2x\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (last)
|
|
|
|
*last++ = '\0';
|
|
|
|
if (parse_lineno(first, &query->first_lineno) < 0)
|
|
|
|
return -EINVAL;
|
|
|
|
if (last) {
|
|
|
|
/* range <first>-<last> */
|
|
|
|
if (parse_lineno(last, &query->last_lineno) < 0)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* special case for last lineno not specified */
|
|
|
|
if (query->last_lineno == 0)
|
|
|
|
query->last_lineno = UINT_MAX;
|
|
|
|
|
|
|
|
if (query->last_lineno < query->first_lineno) {
|
|
|
|
pr_err("last-line:%d < 1st-line:%d\n",
|
|
|
|
query->last_lineno,
|
|
|
|
query->first_lineno);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
query->last_lineno = query->first_lineno;
|
|
|
|
}
|
|
|
|
vpr_info("parsed line %d-%d\n", query->first_lineno,
|
|
|
|
query->last_lineno);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-20 06:12:49 +08:00
|
|
|
static int check_set(const char **dest, char *src, char *name)
|
|
|
|
{
|
|
|
|
int rc = 0;
|
|
|
|
|
|
|
|
if (*dest) {
|
|
|
|
rc = -EINVAL;
|
2012-12-06 05:48:26 +08:00
|
|
|
pr_err("match-spec:%s val:%s overridden by %s\n",
|
|
|
|
name, *dest, src);
|
2011-12-20 06:12:49 +08:00
|
|
|
}
|
|
|
|
*dest = src;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
/*
|
|
|
|
* Parse words[] as a ddebug query specification, which is a series
|
2020-07-20 07:10:54 +08:00
|
|
|
* of (keyword, value) pairs or combined keyword=value terms,
|
|
|
|
* chosen from these possibilities:
|
2009-02-06 00:51:38 +08:00
|
|
|
*
|
|
|
|
* func <function-name>
|
|
|
|
* file <full-pathname>
|
|
|
|
* file <base-filename>
|
|
|
|
* module <module-name>
|
|
|
|
* format <escaped-string-to-find-in-format>
|
|
|
|
* line <lineno>
|
|
|
|
* line <first-lineno>-<last-lineno> // where either may be empty
|
2011-12-20 06:12:49 +08:00
|
|
|
*
|
|
|
|
* Only 1 of each type is allowed.
|
|
|
|
* Returns 0 on success, <0 on error.
|
2009-02-06 00:51:38 +08:00
|
|
|
*/
|
|
|
|
static int ddebug_parse_query(char *words[], int nwords,
|
2012-04-28 04:30:40 +08:00
|
|
|
struct ddebug_query *query, const char *modname)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
|
|
|
unsigned int i;
|
2013-08-28 00:50:03 +08:00
|
|
|
int rc = 0;
|
2020-07-20 07:10:53 +08:00
|
|
|
char *fline;
|
2020-07-20 07:10:54 +08:00
|
|
|
char *keyword, *arg;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2012-04-28 04:30:40 +08:00
|
|
|
if (modname)
|
|
|
|
/* support $modname.dyndbg=<multiple queries> */
|
|
|
|
query->module = modname;
|
|
|
|
|
2020-07-20 07:10:54 +08:00
|
|
|
for (i = 0; i < nwords; i++) {
|
|
|
|
/* accept keyword=arg */
|
|
|
|
vpr_info("%d w:%s\n", i, words[i]);
|
|
|
|
|
|
|
|
keyword = words[i];
|
|
|
|
arg = strchr(keyword, '=');
|
|
|
|
if (arg) {
|
|
|
|
*arg++ = '\0';
|
|
|
|
} else {
|
|
|
|
i++; /* next word is arg */
|
|
|
|
if (!(i < nwords)) {
|
|
|
|
pr_err("missing arg to keyword: %s\n", keyword);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
arg = words[i];
|
|
|
|
}
|
|
|
|
vpr_info("%d key:%s arg:%s\n", i, keyword, arg);
|
|
|
|
|
|
|
|
if (!strcmp(keyword, "func")) {
|
|
|
|
rc = check_set(&query->function, arg, "func");
|
|
|
|
} else if (!strcmp(keyword, "file")) {
|
|
|
|
if (check_set(&query->filename, arg, "file"))
|
2020-07-20 07:10:53 +08:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* tail :$info is function or line-range */
|
|
|
|
fline = strchr(query->filename, ':');
|
|
|
|
if (!fline)
|
|
|
|
break;
|
|
|
|
*fline++ = '\0';
|
|
|
|
if (isalpha(*fline) || *fline == '*' || *fline == '?') {
|
|
|
|
/* take as function name */
|
|
|
|
if (check_set(&query->function, fline, "func"))
|
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
|
|
|
if (parse_linerange(query, fline))
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2020-07-20 07:10:54 +08:00
|
|
|
} else if (!strcmp(keyword, "module")) {
|
|
|
|
rc = check_set(&query->module, arg, "module");
|
|
|
|
} else if (!strcmp(keyword, "format")) {
|
|
|
|
string_unescape_inplace(arg, UNESCAPE_SPACE |
|
2013-05-01 06:27:32 +08:00
|
|
|
UNESCAPE_OCTAL |
|
|
|
|
UNESCAPE_SPECIAL);
|
2020-07-20 07:10:54 +08:00
|
|
|
rc = check_set(&query->format, arg, "format");
|
|
|
|
} else if (!strcmp(keyword, "line")) {
|
|
|
|
if (parse_linerange(query, arg))
|
2009-02-06 00:51:38 +08:00
|
|
|
return -EINVAL;
|
|
|
|
} else {
|
2020-07-20 07:10:54 +08:00
|
|
|
pr_err("unknown keyword \"%s\"\n", keyword);
|
2009-02-06 00:51:38 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2011-12-20 06:12:49 +08:00
|
|
|
if (rc)
|
|
|
|
return rc;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
2011-12-20 06:13:16 +08:00
|
|
|
vpr_info_dq(query, "parsed");
|
2009-02-06 00:51:38 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parse `str' as a flags specification, format [-+=][p]+.
|
|
|
|
* Sets up *maskp and *flagsp to be used when changing the
|
|
|
|
* flags fields of matched _ddebug's. Returns 0 on success
|
|
|
|
* or <0 on error.
|
|
|
|
*/
|
2020-07-20 07:10:55 +08:00
|
|
|
static int ddebug_parse_flags(const char *str, struct flag_settings *modifiers)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
2020-07-20 07:10:55 +08:00
|
|
|
int op, i;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
switch (*str) {
|
|
|
|
case '+':
|
|
|
|
case '-':
|
|
|
|
case '=':
|
|
|
|
op = *str++;
|
|
|
|
break;
|
|
|
|
default:
|
2012-12-06 05:48:27 +08:00
|
|
|
pr_err("bad flag-op %c, at start of %s\n", *str, str);
|
2009-02-06 00:51:38 +08:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
2012-04-28 04:30:32 +08:00
|
|
|
vpr_info("op='%c'\n", op);
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2012-12-06 05:48:26 +08:00
|
|
|
for (; *str ; ++str) {
|
2011-01-24 00:17:24 +08:00
|
|
|
for (i = ARRAY_SIZE(opt_array) - 1; i >= 0; i--) {
|
|
|
|
if (*str == opt_array[i].opt_char) {
|
2020-07-20 07:10:55 +08:00
|
|
|
modifiers->flags |= opt_array[i].flag;
|
2011-01-24 00:17:24 +08:00
|
|
|
break;
|
|
|
|
}
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
2012-12-06 05:48:27 +08:00
|
|
|
if (i < 0) {
|
2020-07-20 07:10:48 +08:00
|
|
|
pr_err("unknown flag '%c'\n", *str);
|
2011-01-24 00:17:24 +08:00
|
|
|
return -EINVAL;
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
2020-07-20 07:10:55 +08:00
|
|
|
vpr_info("flags=0x%x\n", modifiers->flags);
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2020-07-20 07:10:55 +08:00
|
|
|
/* calculate final flags, mask based upon op */
|
2009-02-06 00:51:38 +08:00
|
|
|
switch (op) {
|
|
|
|
case '=':
|
2020-07-20 07:10:55 +08:00
|
|
|
/* modifiers->flags already set */
|
|
|
|
modifiers->mask = 0;
|
2009-02-06 00:51:38 +08:00
|
|
|
break;
|
|
|
|
case '+':
|
2020-07-20 07:10:55 +08:00
|
|
|
modifiers->mask = ~0U;
|
2009-02-06 00:51:38 +08:00
|
|
|
break;
|
|
|
|
case '-':
|
2020-07-20 07:10:55 +08:00
|
|
|
modifiers->mask = ~modifiers->flags;
|
|
|
|
modifiers->flags = 0;
|
2009-02-06 00:51:38 +08:00
|
|
|
break;
|
|
|
|
}
|
2020-07-20 07:10:55 +08:00
|
|
|
vpr_info("*flagsp=0x%x *maskp=0x%x\n", modifiers->flags, modifiers->mask);
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-28 04:30:40 +08:00
|
|
|
static int ddebug_exec_query(char *query_string, const char *modname)
|
2010-08-06 22:11:01 +08:00
|
|
|
{
|
2020-07-20 07:10:55 +08:00
|
|
|
struct flag_settings modifiers = {};
|
2020-07-20 07:10:49 +08:00
|
|
|
struct ddebug_query query = {};
|
2010-08-06 22:11:01 +08:00
|
|
|
#define MAXWORDS 9
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
int nwords, nfound;
|
2010-08-06 22:11:01 +08:00
|
|
|
char *words[MAXWORDS];
|
|
|
|
|
|
|
|
nwords = ddebug_tokenize(query_string, words, MAXWORDS);
|
2012-12-06 05:48:27 +08:00
|
|
|
if (nwords <= 0) {
|
|
|
|
pr_err("tokenize failed\n");
|
2010-08-06 22:11:01 +08:00
|
|
|
return -EINVAL;
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
|
|
|
/* check flags 1st (last arg) so query is pairs of spec,val */
|
2020-07-20 07:10:55 +08:00
|
|
|
if (ddebug_parse_flags(words[nwords-1], &modifiers)) {
|
2012-12-06 05:48:27 +08:00
|
|
|
pr_err("flags parse failed\n");
|
2010-08-06 22:11:01 +08:00
|
|
|
return -EINVAL;
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
|
|
|
if (ddebug_parse_query(words, nwords-1, &query, modname)) {
|
|
|
|
pr_err("query parse failed\n");
|
2010-08-06 22:11:01 +08:00
|
|
|
return -EINVAL;
|
2012-12-06 05:48:27 +08:00
|
|
|
}
|
2010-08-06 22:11:01 +08:00
|
|
|
/* actually go and implement the change */
|
2020-07-20 07:10:55 +08:00
|
|
|
nfound = ddebug_change(&query, &modifiers);
|
2012-12-06 05:48:26 +08:00
|
|
|
vpr_info_dq(&query, nfound ? "applied" : "no-match");
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
|
|
|
|
return nfound;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle multiple queries in query string, continue on error, return
|
|
|
|
last error or number of matching callsites. Module name is either
|
|
|
|
in param (for boot arg) or perhaps in query string.
|
|
|
|
*/
|
dyndbg: export ddebug_exec_queries
Export ddebug_exec_queries() for use by modules.
This will allow module authors to control all their *pr_debug*s
dynamically. And since ddebug_exec_queries() is what implements
"echo $query >control", it gives the same per-callsite control.
Virtues of this:
- simplicity. just an export.
- full control over any/all subsets of callsites.
- same "query/command-string" in code and console
- full callsite selectivity with module file line format
Format in particular deserves special attention; it is where
low-hanging fruit will be found.
Consider: drivers/gpu/drm/amd/display/include/logger_types.h:
#define DC_LOG_SURFACE(...) pr_debug("[SURFACE]:"__VA_ARGS__)
#define DC_LOG_HW_LINK_TRAINING(...) pr_debug("[HW_LINK_TRAINING]:"__VA_ARGS__)
.. 9 more ..
Thats 11 string prefixes, used in 804 places in drivers/gpu/**
Clearly this is a systematized classification of those callsites.
And one I'd expect to see repeated often.
Using ddebug_exec_queries(), authors can select on those prefixes
as a unitary set, equivalent to:
echo "module=MODULE_NAME format=^[SURFACE]: +p" >control
Trivially, those sets can be subsected with the other query terms too,
say file=foo, should the author see fit.
Perhaps as important, users can modify the set of enabled callsites,
presumably to aid debugging by enabling helpful debug callsites, and
disabling those that just clutter the info.
Authors could even alter [fmlt] flags, though I dont see a good reason
why they would. Perhaps harnessed by bug-logging automation to get
fuller, or more minimal bug-reports.
DRM
drm has both drm.debug, which defines 32 categories of drm_printk
logging, and entirely separate uses of pr_debug, which are dynamic on
this i915 laptop, running mainline. So I can observe and report on
both.
The i915 driver has 118 dyndbg callsites, with following
"classifications" defined in drivers/gpu/drm/i915/gvt/**
$ grep 915 /proc/dynamic_debug/control | cut -d= -f2 | cut -d: -f1,2 | sort -u
_ "gvt: cmd
_ "gvt: core
_ "gvt: dpy
_ "gvt: el
_ "gvt: irq
_ "gvt: mm
_ "gvt: mmio
_ "gvt: render
_ "gvt: sched
_ "%s for root hub!\012"
_ "Vendor defined info completion code %u\012"
This classification is entirely out-of-band for control by drm.debug,
and is only available to root user at the console. But module authors
can activate them with ddebug_exec_queries(sprintf("format=^%s +p")),
and then decide how to expose the groups to the user for max utility.
drm.debug
drm.debug has 32 bit-flags, and matching enum drm_debug_category
values to classify the ~2943 DRM_DEBUG*() callsites in drivers/gpu
The drm.debug callback could invoke ddebug_exec_queries() with 32
different hardcoded query strings, needing only (bit) ? " +p" : " -p"
added.
I briefly enabled drm.debug=0xff on my i915 laptop, which yielded
these unique prefixes: (dmesg | cut -c17- | cut -d\] -f1 | sort -u)
[drm:drm_atomic_check_only [drm
[drm:drm_atomic_get_crtc_state [drm
[drm:drm_atomic_get_plane_state [drm
[drm:drm_atomic_nonblocking_commit [drm
[drm:drm_atomic_set_fb_for_plane [drm
[drm:drm_atomic_state_default_clear [drm
[drm:__drm_atomic_state_free [drm
[drm:drm_atomic_state_init [drm
[drm:drm_crtc_vblank_helper_get_vblank_timestamp_internal [drm
[drm:drm_handle_vblank [drm
[drm:drm_ioctl [drm
[drm:drm_mode_addfb2 [drm
[drm:drm_mode_object_get [drm
[drm:drm_mode_object_put.part.0 [drm
[drm:drm_update_vblank_count [drm
[drm:drm_vblank_enable [drm
[drm:drm_vblank_restore [drm
[drm:vblank_disable_fn [drm
i915 0000:00:02.0: [drm:gen9_set_dc_state [i915
i915 0000:00:02.0: [drm:intel_atomic_get_global_obj_state [i915
i915 0000:00:02.0: [drm:__intel_display_power_get_domain.part.0 [i915
i915 0000:00:02.0: [drm:__intel_display_power_put_domain [i915
i915 0000:00:02.0: [drm:intel_plane_atomic_calc_changes [i915
i915 0000:00:02.0: [drm:skl_enable_dc6 [i915
Several good format=^prefixes are apparent there, and some misses.
^[drm:drm_atomic_ # misses: [drm:__drm_atomic_state_free [drm
^[drm:drm_ioctl
^[drm:drm_mode
^[drm:drm_vblank_ # misses: [drm:drm_update_vblank_count & [drm:vblank_disable_fn
Its not a perfect 1:1 single format-match per class, but the misses
above can be covered with 1 & 2 additional queries, which can be
concatenated together with ";" separators and submitted with 1 call.
Benefits:
For drm, adapting DRM_DEBUG to use dynamic-debug inside could
replicate (and thereby obsolete) lots of bit-checking in current
DRM_DEBUG callsites, at least with JUMP_LABEL optimized code.
ddebug_exec_queries() and a handful of fixed query-strings can select
and thereby control the already classified callsites.
With the classes mapped to queries, the enum type and parameter can be
eliminated (folded away with macro magic), at least for DYNAMIC_DEBUG
& JUMP_LABEL builds.
Is it safe ?
ddebug_exec_queries() is currently exposed to user space in
several limited ways;
1 it is called from module-load callback, where it implements the
$modname.dyndbg=+p "fake" parameter provided to all modules.
2 it handles query input via >control directly
IOW, it is "fully" exposed to local root user; exposing the same
functionality to other kernel modules is no additional risk.
The other standard issue to check is locking:
dyndbg has a single mutex, taken by ddebug_change to handle >control,
and by ddebug_proc_(start|stop) to span `cat control`. Queries
submitted via export will typically have module specified, which
dramatically cuts the scan by ddebug_change vs "module=* +p".
ISTM this proposed export presents no locking problems.
TLDR;
It would be interesting to see how drm.dyndbg=$QUERY and
drm.debug=$HEXY would interact; it might be order dependent, as
if given as modprobe args or in /etc/modprobe.d/
Acked-by: <jbaron@akamai.com>
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Link: https://lore.kernel.org/r/20200719231058.1586423-19-jim.cromie@gmail.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-07-20 07:10:58 +08:00
|
|
|
int ddebug_exec_queries(char *query, const char *modname)
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
{
|
|
|
|
char *split;
|
|
|
|
int i, errs = 0, exitcode = 0, rc, nfound = 0;
|
|
|
|
|
|
|
|
for (i = 0; query; query = split) {
|
|
|
|
split = strpbrk(query, ";\n");
|
|
|
|
if (split)
|
|
|
|
*split++ = '\0';
|
|
|
|
|
|
|
|
query = skip_spaces(query);
|
|
|
|
if (!query || !*query || *query == '#')
|
|
|
|
continue;
|
|
|
|
|
2012-04-28 04:30:32 +08:00
|
|
|
vpr_info("query %d: \"%s\"\n", i, query);
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
|
2012-04-28 04:30:40 +08:00
|
|
|
rc = ddebug_exec_query(query, modname);
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
if (rc < 0) {
|
|
|
|
errs++;
|
|
|
|
exitcode = rc;
|
2012-12-06 05:48:26 +08:00
|
|
|
} else {
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
nfound += rc;
|
2012-12-06 05:48:26 +08:00
|
|
|
}
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
i++;
|
|
|
|
}
|
2012-04-28 04:30:32 +08:00
|
|
|
vpr_info("processed %d queries, with %d matches, %d errs\n",
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
i, nfound, errs);
|
|
|
|
|
|
|
|
if (exitcode)
|
|
|
|
return exitcode;
|
|
|
|
return nfound;
|
2010-08-06 22:11:01 +08:00
|
|
|
}
|
dyndbg: export ddebug_exec_queries
Export ddebug_exec_queries() for use by modules.
This will allow module authors to control all their *pr_debug*s
dynamically. And since ddebug_exec_queries() is what implements
"echo $query >control", it gives the same per-callsite control.
Virtues of this:
- simplicity. just an export.
- full control over any/all subsets of callsites.
- same "query/command-string" in code and console
- full callsite selectivity with module file line format
Format in particular deserves special attention; it is where
low-hanging fruit will be found.
Consider: drivers/gpu/drm/amd/display/include/logger_types.h:
#define DC_LOG_SURFACE(...) pr_debug("[SURFACE]:"__VA_ARGS__)
#define DC_LOG_HW_LINK_TRAINING(...) pr_debug("[HW_LINK_TRAINING]:"__VA_ARGS__)
.. 9 more ..
Thats 11 string prefixes, used in 804 places in drivers/gpu/**
Clearly this is a systematized classification of those callsites.
And one I'd expect to see repeated often.
Using ddebug_exec_queries(), authors can select on those prefixes
as a unitary set, equivalent to:
echo "module=MODULE_NAME format=^[SURFACE]: +p" >control
Trivially, those sets can be subsected with the other query terms too,
say file=foo, should the author see fit.
Perhaps as important, users can modify the set of enabled callsites,
presumably to aid debugging by enabling helpful debug callsites, and
disabling those that just clutter the info.
Authors could even alter [fmlt] flags, though I dont see a good reason
why they would. Perhaps harnessed by bug-logging automation to get
fuller, or more minimal bug-reports.
DRM
drm has both drm.debug, which defines 32 categories of drm_printk
logging, and entirely separate uses of pr_debug, which are dynamic on
this i915 laptop, running mainline. So I can observe and report on
both.
The i915 driver has 118 dyndbg callsites, with following
"classifications" defined in drivers/gpu/drm/i915/gvt/**
$ grep 915 /proc/dynamic_debug/control | cut -d= -f2 | cut -d: -f1,2 | sort -u
_ "gvt: cmd
_ "gvt: core
_ "gvt: dpy
_ "gvt: el
_ "gvt: irq
_ "gvt: mm
_ "gvt: mmio
_ "gvt: render
_ "gvt: sched
_ "%s for root hub!\012"
_ "Vendor defined info completion code %u\012"
This classification is entirely out-of-band for control by drm.debug,
and is only available to root user at the console. But module authors
can activate them with ddebug_exec_queries(sprintf("format=^%s +p")),
and then decide how to expose the groups to the user for max utility.
drm.debug
drm.debug has 32 bit-flags, and matching enum drm_debug_category
values to classify the ~2943 DRM_DEBUG*() callsites in drivers/gpu
The drm.debug callback could invoke ddebug_exec_queries() with 32
different hardcoded query strings, needing only (bit) ? " +p" : " -p"
added.
I briefly enabled drm.debug=0xff on my i915 laptop, which yielded
these unique prefixes: (dmesg | cut -c17- | cut -d\] -f1 | sort -u)
[drm:drm_atomic_check_only [drm
[drm:drm_atomic_get_crtc_state [drm
[drm:drm_atomic_get_plane_state [drm
[drm:drm_atomic_nonblocking_commit [drm
[drm:drm_atomic_set_fb_for_plane [drm
[drm:drm_atomic_state_default_clear [drm
[drm:__drm_atomic_state_free [drm
[drm:drm_atomic_state_init [drm
[drm:drm_crtc_vblank_helper_get_vblank_timestamp_internal [drm
[drm:drm_handle_vblank [drm
[drm:drm_ioctl [drm
[drm:drm_mode_addfb2 [drm
[drm:drm_mode_object_get [drm
[drm:drm_mode_object_put.part.0 [drm
[drm:drm_update_vblank_count [drm
[drm:drm_vblank_enable [drm
[drm:drm_vblank_restore [drm
[drm:vblank_disable_fn [drm
i915 0000:00:02.0: [drm:gen9_set_dc_state [i915
i915 0000:00:02.0: [drm:intel_atomic_get_global_obj_state [i915
i915 0000:00:02.0: [drm:__intel_display_power_get_domain.part.0 [i915
i915 0000:00:02.0: [drm:__intel_display_power_put_domain [i915
i915 0000:00:02.0: [drm:intel_plane_atomic_calc_changes [i915
i915 0000:00:02.0: [drm:skl_enable_dc6 [i915
Several good format=^prefixes are apparent there, and some misses.
^[drm:drm_atomic_ # misses: [drm:__drm_atomic_state_free [drm
^[drm:drm_ioctl
^[drm:drm_mode
^[drm:drm_vblank_ # misses: [drm:drm_update_vblank_count & [drm:vblank_disable_fn
Its not a perfect 1:1 single format-match per class, but the misses
above can be covered with 1 & 2 additional queries, which can be
concatenated together with ";" separators and submitted with 1 call.
Benefits:
For drm, adapting DRM_DEBUG to use dynamic-debug inside could
replicate (and thereby obsolete) lots of bit-checking in current
DRM_DEBUG callsites, at least with JUMP_LABEL optimized code.
ddebug_exec_queries() and a handful of fixed query-strings can select
and thereby control the already classified callsites.
With the classes mapped to queries, the enum type and parameter can be
eliminated (folded away with macro magic), at least for DYNAMIC_DEBUG
& JUMP_LABEL builds.
Is it safe ?
ddebug_exec_queries() is currently exposed to user space in
several limited ways;
1 it is called from module-load callback, where it implements the
$modname.dyndbg=+p "fake" parameter provided to all modules.
2 it handles query input via >control directly
IOW, it is "fully" exposed to local root user; exposing the same
functionality to other kernel modules is no additional risk.
The other standard issue to check is locking:
dyndbg has a single mutex, taken by ddebug_change to handle >control,
and by ddebug_proc_(start|stop) to span `cat control`. Queries
submitted via export will typically have module specified, which
dramatically cuts the scan by ddebug_change vs "module=* +p".
ISTM this proposed export presents no locking problems.
TLDR;
It would be interesting to see how drm.dyndbg=$QUERY and
drm.debug=$HEXY would interact; it might be order dependent, as
if given as modprobe args or in /etc/modprobe.d/
Acked-by: <jbaron@akamai.com>
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Link: https://lore.kernel.org/r/20200719231058.1586423-19-jim.cromie@gmail.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-07-20 07:10:58 +08:00
|
|
|
EXPORT_SYMBOL_GPL(ddebug_exec_queries);
|
2010-08-06 22:11:01 +08:00
|
|
|
|
2011-10-05 05:13:19 +08:00
|
|
|
#define PREFIX_SIZE 64
|
|
|
|
|
|
|
|
static int remaining(int wrote)
|
|
|
|
{
|
|
|
|
if (PREFIX_SIZE - wrote > 0)
|
|
|
|
return PREFIX_SIZE - wrote;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *dynamic_emit_prefix(const struct _ddebug *desc, char *buf)
|
2011-01-24 00:17:24 +08:00
|
|
|
{
|
2011-10-05 05:13:19 +08:00
|
|
|
int pos_after_tid;
|
|
|
|
int pos = 0;
|
2011-01-24 00:17:24 +08:00
|
|
|
|
2012-09-13 11:11:29 +08:00
|
|
|
*buf = '\0';
|
|
|
|
|
2011-10-05 05:13:19 +08:00
|
|
|
if (desc->flags & _DPRINTK_FLAGS_INCL_TID) {
|
2011-01-24 00:17:24 +08:00
|
|
|
if (in_interrupt())
|
2012-09-13 11:11:29 +08:00
|
|
|
pos += snprintf(buf + pos, remaining(pos), "<intr> ");
|
2011-01-24 00:17:24 +08:00
|
|
|
else
|
2011-10-05 05:13:19 +08:00
|
|
|
pos += snprintf(buf + pos, remaining(pos), "[%d] ",
|
2012-09-13 11:11:29 +08:00
|
|
|
task_pid_vnr(current));
|
2011-01-24 00:17:24 +08:00
|
|
|
}
|
2011-10-05 05:13:19 +08:00
|
|
|
pos_after_tid = pos;
|
|
|
|
if (desc->flags & _DPRINTK_FLAGS_INCL_MODNAME)
|
|
|
|
pos += snprintf(buf + pos, remaining(pos), "%s:",
|
2012-09-13 11:11:29 +08:00
|
|
|
desc->modname);
|
2011-10-05 05:13:19 +08:00
|
|
|
if (desc->flags & _DPRINTK_FLAGS_INCL_FUNCNAME)
|
|
|
|
pos += snprintf(buf + pos, remaining(pos), "%s:",
|
2012-09-13 11:11:29 +08:00
|
|
|
desc->function);
|
2011-10-05 05:13:19 +08:00
|
|
|
if (desc->flags & _DPRINTK_FLAGS_INCL_LINENO)
|
2011-12-20 06:11:09 +08:00
|
|
|
pos += snprintf(buf + pos, remaining(pos), "%d:",
|
2012-09-13 11:11:29 +08:00
|
|
|
desc->lineno);
|
2011-10-05 05:13:19 +08:00
|
|
|
if (pos - pos_after_tid)
|
|
|
|
pos += snprintf(buf + pos, remaining(pos), " ");
|
|
|
|
if (pos >= PREFIX_SIZE)
|
|
|
|
buf[PREFIX_SIZE - 1] = '\0';
|
2011-08-12 02:36:25 +08:00
|
|
|
|
2011-10-05 05:13:19 +08:00
|
|
|
return buf;
|
2011-08-12 02:36:25 +08:00
|
|
|
}
|
|
|
|
|
2014-09-25 02:17:56 +08:00
|
|
|
void __dynamic_pr_debug(struct _ddebug *descriptor, const char *fmt, ...)
|
2011-01-24 00:17:24 +08:00
|
|
|
{
|
|
|
|
va_list args;
|
2011-10-05 05:13:19 +08:00
|
|
|
struct va_format vaf;
|
|
|
|
char buf[PREFIX_SIZE];
|
2011-01-24 00:17:24 +08:00
|
|
|
|
|
|
|
BUG_ON(!descriptor);
|
|
|
|
BUG_ON(!fmt);
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
2012-09-13 11:11:29 +08:00
|
|
|
|
2011-10-05 05:13:19 +08:00
|
|
|
vaf.fmt = fmt;
|
|
|
|
vaf.va = &args;
|
2012-09-13 11:11:29 +08:00
|
|
|
|
2014-09-25 02:17:56 +08:00
|
|
|
printk(KERN_DEBUG "%s%pV", dynamic_emit_prefix(descriptor, buf), &vaf);
|
2012-09-13 11:11:29 +08:00
|
|
|
|
2011-01-24 00:17:24 +08:00
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(__dynamic_pr_debug);
|
|
|
|
|
2014-09-25 02:17:56 +08:00
|
|
|
void __dynamic_dev_dbg(struct _ddebug *descriptor,
|
2011-08-12 02:36:21 +08:00
|
|
|
const struct device *dev, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
struct va_format vaf;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
BUG_ON(!descriptor);
|
|
|
|
BUG_ON(!fmt);
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
2012-09-13 11:11:29 +08:00
|
|
|
|
2011-08-12 02:36:21 +08:00
|
|
|
vaf.fmt = fmt;
|
|
|
|
vaf.va = &args;
|
2012-09-13 11:11:29 +08:00
|
|
|
|
|
|
|
if (!dev) {
|
2014-09-25 02:17:56 +08:00
|
|
|
printk(KERN_DEBUG "(NULL device *): %pV", &vaf);
|
2012-09-13 11:11:29 +08:00
|
|
|
} else {
|
|
|
|
char buf[PREFIX_SIZE];
|
|
|
|
|
2014-12-11 07:50:15 +08:00
|
|
|
dev_printk_emit(LOGLEVEL_DEBUG, dev, "%s%s %s: %pV",
|
2014-09-25 02:17:56 +08:00
|
|
|
dynamic_emit_prefix(descriptor, buf),
|
|
|
|
dev_driver_string(dev), dev_name(dev),
|
|
|
|
&vaf);
|
2012-09-13 11:11:29 +08:00
|
|
|
}
|
|
|
|
|
2011-08-12 02:36:21 +08:00
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(__dynamic_dev_dbg);
|
|
|
|
|
2011-10-05 05:13:22 +08:00
|
|
|
#ifdef CONFIG_NET
|
|
|
|
|
2014-09-25 02:17:56 +08:00
|
|
|
void __dynamic_netdev_dbg(struct _ddebug *descriptor,
|
|
|
|
const struct net_device *dev, const char *fmt, ...)
|
2011-08-12 02:36:48 +08:00
|
|
|
{
|
|
|
|
struct va_format vaf;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
BUG_ON(!descriptor);
|
|
|
|
BUG_ON(!fmt);
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
2012-09-13 11:12:19 +08:00
|
|
|
|
2011-08-12 02:36:48 +08:00
|
|
|
vaf.fmt = fmt;
|
|
|
|
vaf.va = &args;
|
2012-09-13 11:12:19 +08:00
|
|
|
|
|
|
|
if (dev && dev->dev.parent) {
|
|
|
|
char buf[PREFIX_SIZE];
|
2012-09-13 11:14:11 +08:00
|
|
|
|
2014-12-11 07:50:15 +08:00
|
|
|
dev_printk_emit(LOGLEVEL_DEBUG, dev->dev.parent,
|
2014-09-25 02:17:56 +08:00
|
|
|
"%s%s %s %s%s: %pV",
|
|
|
|
dynamic_emit_prefix(descriptor, buf),
|
|
|
|
dev_driver_string(dev->dev.parent),
|
|
|
|
dev_name(dev->dev.parent),
|
|
|
|
netdev_name(dev), netdev_reg_state(dev),
|
|
|
|
&vaf);
|
2012-09-13 11:12:19 +08:00
|
|
|
} else if (dev) {
|
2014-09-25 02:17:56 +08:00
|
|
|
printk(KERN_DEBUG "%s%s: %pV", netdev_name(dev),
|
|
|
|
netdev_reg_state(dev), &vaf);
|
2012-09-13 11:12:19 +08:00
|
|
|
} else {
|
2014-09-25 02:17:56 +08:00
|
|
|
printk(KERN_DEBUG "(NULL net_device): %pV", &vaf);
|
2012-09-13 11:12:19 +08:00
|
|
|
}
|
|
|
|
|
2011-08-12 02:36:48 +08:00
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(__dynamic_netdev_dbg);
|
|
|
|
|
2011-10-05 05:13:22 +08:00
|
|
|
#endif
|
|
|
|
|
2019-05-01 18:48:13 +08:00
|
|
|
#if IS_ENABLED(CONFIG_INFINIBAND)
|
|
|
|
|
|
|
|
void __dynamic_ibdev_dbg(struct _ddebug *descriptor,
|
|
|
|
const struct ib_device *ibdev, const char *fmt, ...)
|
|
|
|
{
|
|
|
|
struct va_format vaf;
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
va_start(args, fmt);
|
|
|
|
|
|
|
|
vaf.fmt = fmt;
|
|
|
|
vaf.va = &args;
|
|
|
|
|
|
|
|
if (ibdev && ibdev->dev.parent) {
|
|
|
|
char buf[PREFIX_SIZE];
|
|
|
|
|
|
|
|
dev_printk_emit(LOGLEVEL_DEBUG, ibdev->dev.parent,
|
|
|
|
"%s%s %s %s: %pV",
|
|
|
|
dynamic_emit_prefix(descriptor, buf),
|
|
|
|
dev_driver_string(ibdev->dev.parent),
|
|
|
|
dev_name(ibdev->dev.parent),
|
|
|
|
dev_name(&ibdev->dev),
|
|
|
|
&vaf);
|
|
|
|
} else if (ibdev) {
|
|
|
|
printk(KERN_DEBUG "%s: %pV", dev_name(&ibdev->dev), &vaf);
|
|
|
|
} else {
|
|
|
|
printk(KERN_DEBUG "(NULL ib_device): %pV", &vaf);
|
|
|
|
}
|
|
|
|
|
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(__dynamic_ibdev_dbg);
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2011-12-20 06:12:29 +08:00
|
|
|
#define DDEBUG_STRING_SIZE 1024
|
|
|
|
static __initdata char ddebug_setup_string[DDEBUG_STRING_SIZE];
|
|
|
|
|
2010-08-06 22:11:02 +08:00
|
|
|
static __init int ddebug_setup_query(char *str)
|
|
|
|
{
|
2011-12-20 06:12:29 +08:00
|
|
|
if (strlen(str) >= DDEBUG_STRING_SIZE) {
|
2011-08-12 02:36:33 +08:00
|
|
|
pr_warn("ddebug boot param string too large\n");
|
2010-08-06 22:11:02 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2011-12-20 06:12:29 +08:00
|
|
|
strlcpy(ddebug_setup_string, str, DDEBUG_STRING_SIZE);
|
2010-08-06 22:11:02 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
__setup("ddebug_query=", ddebug_setup_query);
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
/*
|
2014-12-15 11:04:16 +08:00
|
|
|
* File_ops->write method for <debugfs>/dynamic_debug/control. Gathers the
|
2009-02-06 00:51:38 +08:00
|
|
|
* command text from userspace, parses and executes it.
|
|
|
|
*/
|
2011-12-20 06:13:07 +08:00
|
|
|
#define USER_BUF_PAGE 4096
|
2009-02-06 00:51:38 +08:00
|
|
|
static ssize_t ddebug_proc_write(struct file *file, const char __user *ubuf,
|
|
|
|
size_t len, loff_t *offp)
|
|
|
|
{
|
2011-12-20 06:13:07 +08:00
|
|
|
char *tmpbuf;
|
2010-08-06 22:11:01 +08:00
|
|
|
int ret;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
if (len == 0)
|
|
|
|
return 0;
|
2011-12-20 06:13:07 +08:00
|
|
|
if (len > USER_BUF_PAGE - 1) {
|
|
|
|
pr_warn("expected <%d bytes into control\n", USER_BUF_PAGE);
|
2009-02-06 00:51:38 +08:00
|
|
|
return -E2BIG;
|
2011-12-20 06:13:07 +08:00
|
|
|
}
|
2015-12-24 13:06:05 +08:00
|
|
|
tmpbuf = memdup_user_nul(ubuf, len);
|
|
|
|
if (IS_ERR(tmpbuf))
|
|
|
|
return PTR_ERR(tmpbuf);
|
2012-04-28 04:30:32 +08:00
|
|
|
vpr_info("read %d bytes from userspace\n", (int)len);
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2012-04-28 04:30:40 +08:00
|
|
|
ret = ddebug_exec_queries(tmpbuf, NULL);
|
2011-12-20 06:13:07 +08:00
|
|
|
kfree(tmpbuf);
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
if (ret < 0)
|
2010-08-06 22:11:01 +08:00
|
|
|
return ret;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
*offp += len;
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the iterator to point to the first _ddebug object
|
|
|
|
* and return a pointer to that first object. Returns
|
|
|
|
* NULL if there are no _ddebugs at all.
|
|
|
|
*/
|
|
|
|
static struct _ddebug *ddebug_iter_first(struct ddebug_iter *iter)
|
|
|
|
{
|
|
|
|
if (list_empty(&ddebug_tables)) {
|
|
|
|
iter->table = NULL;
|
|
|
|
iter->idx = 0;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
iter->table = list_entry(ddebug_tables.next,
|
|
|
|
struct ddebug_table, link);
|
|
|
|
iter->idx = 0;
|
|
|
|
return &iter->table->ddebugs[iter->idx];
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Advance the iterator to point to the next _ddebug
|
|
|
|
* object from the one the iterator currently points at,
|
|
|
|
* and returns a pointer to the new _ddebug. Returns
|
|
|
|
* NULL if the iterator has seen all the _ddebugs.
|
|
|
|
*/
|
|
|
|
static struct _ddebug *ddebug_iter_next(struct ddebug_iter *iter)
|
|
|
|
{
|
|
|
|
if (iter->table == NULL)
|
|
|
|
return NULL;
|
|
|
|
if (++iter->idx == iter->table->num_ddebugs) {
|
|
|
|
/* iterate to next table */
|
|
|
|
iter->idx = 0;
|
|
|
|
if (list_is_last(&iter->table->link, &ddebug_tables)) {
|
|
|
|
iter->table = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
iter->table = list_entry(iter->table->link.next,
|
|
|
|
struct ddebug_table, link);
|
|
|
|
}
|
|
|
|
return &iter->table->ddebugs[iter->idx];
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Seq_ops start method. Called at the start of every
|
|
|
|
* read() call from userspace. Takes the ddebug_lock and
|
|
|
|
* seeks the seq_file's iterator to the given position.
|
|
|
|
*/
|
|
|
|
static void *ddebug_proc_start(struct seq_file *m, loff_t *pos)
|
|
|
|
{
|
|
|
|
struct ddebug_iter *iter = m->private;
|
|
|
|
struct _ddebug *dp;
|
|
|
|
int n = *pos;
|
|
|
|
|
|
|
|
mutex_lock(&ddebug_lock);
|
|
|
|
|
|
|
|
if (!n)
|
|
|
|
return SEQ_START_TOKEN;
|
|
|
|
if (n < 0)
|
|
|
|
return NULL;
|
|
|
|
dp = ddebug_iter_first(iter);
|
|
|
|
while (dp != NULL && --n > 0)
|
|
|
|
dp = ddebug_iter_next(iter);
|
|
|
|
return dp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Seq_ops next method. Called several times within a read()
|
|
|
|
* call from userspace, with ddebug_lock held. Walks to the
|
|
|
|
* next _ddebug object with a special case for the header line.
|
|
|
|
*/
|
|
|
|
static void *ddebug_proc_next(struct seq_file *m, void *p, loff_t *pos)
|
|
|
|
{
|
|
|
|
struct ddebug_iter *iter = m->private;
|
|
|
|
struct _ddebug *dp;
|
|
|
|
|
|
|
|
if (p == SEQ_START_TOKEN)
|
|
|
|
dp = ddebug_iter_first(iter);
|
|
|
|
else
|
|
|
|
dp = ddebug_iter_next(iter);
|
|
|
|
++*pos;
|
|
|
|
return dp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Seq_ops show method. Called several times within a read()
|
|
|
|
* call from userspace, with ddebug_lock held. Formats the
|
|
|
|
* current _ddebug as a single human-readable line, with a
|
|
|
|
* special case for the header line.
|
|
|
|
*/
|
|
|
|
static int ddebug_proc_show(struct seq_file *m, void *p)
|
|
|
|
{
|
|
|
|
struct ddebug_iter *iter = m->private;
|
|
|
|
struct _ddebug *dp = p;
|
2020-07-20 07:10:47 +08:00
|
|
|
struct flagsbuf flags;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
if (p == SEQ_START_TOKEN) {
|
|
|
|
seq_puts(m,
|
2012-12-06 05:48:26 +08:00
|
|
|
"# filename:lineno [module]function flags format\n");
|
2009-02-06 00:51:38 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-12-20 06:12:44 +08:00
|
|
|
seq_printf(m, "%s:%u [%s]%s =%s \"",
|
2012-12-06 05:48:26 +08:00
|
|
|
trim_prefix(dp->filename), dp->lineno,
|
|
|
|
iter->table->mod_name, dp->function,
|
2020-07-20 07:10:47 +08:00
|
|
|
ddebug_describe_flags(dp->flags, &flags));
|
2009-02-06 00:51:38 +08:00
|
|
|
seq_escape(m, dp->format, "\t\r\n\"");
|
|
|
|
seq_puts(m, "\"\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Seq_ops stop method. Called at the end of each read()
|
|
|
|
* call from userspace. Drops ddebug_lock.
|
|
|
|
*/
|
|
|
|
static void ddebug_proc_stop(struct seq_file *m, void *p)
|
|
|
|
{
|
|
|
|
mutex_unlock(&ddebug_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct seq_operations ddebug_proc_seqops = {
|
|
|
|
.start = ddebug_proc_start,
|
|
|
|
.next = ddebug_proc_next,
|
|
|
|
.show = ddebug_proc_show,
|
|
|
|
.stop = ddebug_proc_stop
|
|
|
|
};
|
|
|
|
|
|
|
|
static int ddebug_proc_open(struct inode *inode, struct file *file)
|
|
|
|
{
|
2012-04-28 04:30:32 +08:00
|
|
|
vpr_info("called\n");
|
2014-10-14 06:51:32 +08:00
|
|
|
return seq_open_private(file, &ddebug_proc_seqops,
|
|
|
|
sizeof(struct ddebug_iter));
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct file_operations ddebug_proc_fops = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = ddebug_proc_open,
|
|
|
|
.read = seq_read,
|
|
|
|
.llseek = seq_lseek,
|
|
|
|
.release = seq_release_private,
|
|
|
|
.write = ddebug_proc_write
|
|
|
|
};
|
|
|
|
|
2020-02-11 05:11:42 +08:00
|
|
|
static const struct proc_ops proc_fops = {
|
|
|
|
.proc_open = ddebug_proc_open,
|
|
|
|
.proc_read = seq_read,
|
|
|
|
.proc_lseek = seq_lseek,
|
|
|
|
.proc_release = seq_release_private,
|
|
|
|
.proc_write = ddebug_proc_write
|
|
|
|
};
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
/*
|
|
|
|
* Allocate a new ddebug_table for the given module
|
|
|
|
* and add it to the global list.
|
|
|
|
*/
|
|
|
|
int ddebug_add_module(struct _ddebug *tab, unsigned int n,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
struct ddebug_table *dt;
|
|
|
|
|
|
|
|
dt = kzalloc(sizeof(*dt), GFP_KERNEL);
|
2019-03-08 08:27:48 +08:00
|
|
|
if (dt == NULL) {
|
|
|
|
pr_err("error adding module: %s\n", name);
|
2009-02-06 00:51:38 +08:00
|
|
|
return -ENOMEM;
|
2019-03-08 08:27:48 +08:00
|
|
|
}
|
2019-03-08 08:27:37 +08:00
|
|
|
/*
|
|
|
|
* For built-in modules, name lives in .rodata and is
|
|
|
|
* immortal. For loaded modules, name points at the name[]
|
|
|
|
* member of struct module, which lives at least as long as
|
|
|
|
* this struct ddebug_table.
|
|
|
|
*/
|
|
|
|
dt->mod_name = name;
|
2009-02-06 00:51:38 +08:00
|
|
|
dt->num_ddebugs = n;
|
|
|
|
dt->ddebugs = tab;
|
|
|
|
|
|
|
|
mutex_lock(&ddebug_lock);
|
2020-07-20 07:10:50 +08:00
|
|
|
list_add(&dt->link, &ddebug_tables);
|
2009-02-06 00:51:38 +08:00
|
|
|
mutex_unlock(&ddebug_lock);
|
|
|
|
|
2020-09-01 02:22:08 +08:00
|
|
|
v2pr_info("%3u debug prints in module %s\n", n, dt->mod_name);
|
2009-02-06 00:51:38 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-28 04:30:37 +08:00
|
|
|
/* helper for ddebug_dyndbg_(boot|module)_param_cb */
|
|
|
|
static int ddebug_dyndbg_param_cb(char *param, char *val,
|
|
|
|
const char *modname, int on_err)
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
{
|
|
|
|
char *sep;
|
|
|
|
|
|
|
|
sep = strchr(param, '.');
|
|
|
|
if (sep) {
|
2012-04-28 04:30:37 +08:00
|
|
|
/* needed only for ddebug_dyndbg_boot_param_cb */
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
*sep = '\0';
|
|
|
|
modname = param;
|
|
|
|
param = sep + 1;
|
|
|
|
}
|
|
|
|
if (strcmp(param, "dyndbg"))
|
2012-04-28 04:30:37 +08:00
|
|
|
return on_err; /* determined by caller */
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
|
2012-04-28 04:30:40 +08:00
|
|
|
ddebug_exec_queries((val ? val : "+p"), modname);
|
|
|
|
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
return 0; /* query failure shouldnt stop module load */
|
|
|
|
}
|
|
|
|
|
2012-04-28 04:30:37 +08:00
|
|
|
/* handle both dyndbg and $module.dyndbg params at boot */
|
|
|
|
static int ddebug_dyndbg_boot_param_cb(char *param, char *val,
|
module: add extra argument for parse_params() callback
This adds an extra argument onto parse_params() to be used
as a way to make the unused callback a bit more useful and
generic by allowing the caller to pass on a data structure
of its choice. An example use case is to allow us to easily
make module parameters for every module which we will do
next.
@ parse @
identifier name, args, params, num, level_min, level_max;
identifier unknown, param, val, doing;
type s16;
@@
extern char *parse_args(const char *name,
char *args,
const struct kernel_param *params,
unsigned num,
s16 level_min,
s16 level_max,
+ void *arg,
int (*unknown)(char *param, char *val,
const char *doing
+ , void *arg
));
@ parse_mod @
identifier name, args, params, num, level_min, level_max;
identifier unknown, param, val, doing;
type s16;
@@
char *parse_args(const char *name,
char *args,
const struct kernel_param *params,
unsigned num,
s16 level_min,
s16 level_max,
+ void *arg,
int (*unknown)(char *param, char *val,
const char *doing
+ , void *arg
))
{
...
}
@ parse_args_found @
expression R, E1, E2, E3, E4, E5, E6;
identifier func;
@@
(
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
func);
|
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
&func);
|
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
NULL);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
func);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
&func);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
NULL);
)
@ parse_args_unused depends on parse_args_found @
identifier parse_args_found.func;
@@
int func(char *param, char *val, const char *unused
+ , void *arg
)
{
...
}
@ mod_unused depends on parse_args_found @
identifier parse_args_found.func;
expression A1, A2, A3;
@@
- func(A1, A2, A3);
+ func(A1, A2, A3, NULL);
Generated-by: Coccinelle SmPL
Cc: cocci@systeme.lip6.fr
Cc: Tejun Heo <tj@kernel.org>
Cc: Arjan van de Ven <arjan@linux.intel.com>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: Rusty Russell <rusty@rustcorp.com.au>
Cc: Christoph Hellwig <hch@infradead.org>
Cc: Felipe Contreras <felipe.contreras@gmail.com>
Cc: Ewan Milne <emilne@redhat.com>
Cc: Jean Delvare <jdelvare@suse.de>
Cc: Hannes Reinecke <hare@suse.de>
Cc: Jani Nikula <jani.nikula@intel.com>
Cc: linux-kernel@vger.kernel.org
Reviewed-by: Tejun Heo <tj@kernel.org>
Acked-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Luis R. Rodriguez <mcgrof@suse.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-31 07:20:03 +08:00
|
|
|
const char *unused, void *arg)
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
{
|
2012-04-28 04:30:37 +08:00
|
|
|
vpr_info("%s=\"%s\"\n", param, val);
|
|
|
|
return ddebug_dyndbg_param_cb(param, val, NULL, 0);
|
|
|
|
}
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
|
2012-04-28 04:30:37 +08:00
|
|
|
/*
|
|
|
|
* modprobe foo finds foo.params in boot-args, strips "foo.", and
|
|
|
|
* passes them to load_module(). This callback gets unknown params,
|
|
|
|
* processes dyndbg params, rejects others.
|
|
|
|
*/
|
|
|
|
int ddebug_dyndbg_module_param_cb(char *param, char *val, const char *module)
|
|
|
|
{
|
|
|
|
vpr_info("module: %s %s=\"%s\"\n", module, param, val);
|
|
|
|
return ddebug_dyndbg_param_cb(param, val, module, -ENOENT);
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
}
|
|
|
|
|
2009-02-06 00:51:38 +08:00
|
|
|
static void ddebug_table_free(struct ddebug_table *dt)
|
|
|
|
{
|
|
|
|
list_del_init(&dt->link);
|
|
|
|
kfree(dt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Called in response to a module being unloaded. Removes
|
|
|
|
* any ddebug_table's which point at the module.
|
|
|
|
*/
|
2010-07-03 11:07:35 +08:00
|
|
|
int ddebug_remove_module(const char *mod_name)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
|
|
|
struct ddebug_table *dt, *nextdt;
|
|
|
|
int ret = -ENOENT;
|
|
|
|
|
2020-07-20 07:10:44 +08:00
|
|
|
v2pr_info("removing module \"%s\"\n", mod_name);
|
2009-02-06 00:51:38 +08:00
|
|
|
|
|
|
|
mutex_lock(&ddebug_lock);
|
|
|
|
list_for_each_entry_safe(dt, nextdt, &ddebug_tables, link) {
|
2019-03-08 08:27:41 +08:00
|
|
|
if (dt->mod_name == mod_name) {
|
2009-02-06 00:51:38 +08:00
|
|
|
ddebug_table_free(dt);
|
|
|
|
ret = 0;
|
2019-03-08 08:27:41 +08:00
|
|
|
break;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
mutex_unlock(&ddebug_lock);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ddebug_remove_all_tables(void)
|
|
|
|
{
|
|
|
|
mutex_lock(&ddebug_lock);
|
|
|
|
while (!list_empty(&ddebug_tables)) {
|
|
|
|
struct ddebug_table *dt = list_entry(ddebug_tables.next,
|
|
|
|
struct ddebug_table,
|
|
|
|
link);
|
|
|
|
ddebug_table_free(dt);
|
|
|
|
}
|
|
|
|
mutex_unlock(&ddebug_lock);
|
|
|
|
}
|
|
|
|
|
2010-08-06 22:11:03 +08:00
|
|
|
static __initdata int ddebug_init_success;
|
|
|
|
|
2020-02-11 05:11:42 +08:00
|
|
|
static int __init dynamic_debug_init_control(void)
|
2009-02-06 00:51:38 +08:00
|
|
|
{
|
2020-02-11 05:11:42 +08:00
|
|
|
struct proc_dir_entry *procfs_dir;
|
|
|
|
struct dentry *debugfs_dir;
|
2010-08-06 22:11:03 +08:00
|
|
|
|
|
|
|
if (!ddebug_init_success)
|
|
|
|
return -ENODEV;
|
2009-02-06 00:51:38 +08:00
|
|
|
|
2020-02-11 05:11:42 +08:00
|
|
|
/* Create the control file in debugfs if it is enabled */
|
|
|
|
if (debugfs_initialized()) {
|
|
|
|
debugfs_dir = debugfs_create_dir("dynamic_debug", NULL);
|
|
|
|
debugfs_create_file("control", 0644, debugfs_dir, NULL,
|
|
|
|
&ddebug_proc_fops);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Also create the control file in procfs */
|
|
|
|
procfs_dir = proc_mkdir("dynamic_debug", NULL);
|
|
|
|
if (procfs_dir)
|
|
|
|
proc_create("control", 0644, procfs_dir, &proc_fops);
|
2019-06-12 23:35:34 +08:00
|
|
|
|
2010-08-06 22:11:03 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __init dynamic_debug_init(void)
|
|
|
|
{
|
|
|
|
struct _ddebug *iter, *iter_start;
|
|
|
|
const char *modname = NULL;
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
char *cmdline;
|
2010-08-06 22:11:03 +08:00
|
|
|
int ret = 0;
|
2012-04-28 04:30:39 +08:00
|
|
|
int n = 0, entries = 0, modct = 0;
|
2010-08-06 22:11:03 +08:00
|
|
|
|
2020-07-20 07:10:45 +08:00
|
|
|
if (&__start___dyndbg == &__stop___dyndbg) {
|
2020-06-08 12:40:14 +08:00
|
|
|
if (IS_ENABLED(CONFIG_DYNAMIC_DEBUG)) {
|
|
|
|
pr_warn("_ddebug table is empty in a CONFIG_DYNAMIC_DEBUG build\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
pr_info("Ignore empty _ddebug table in a CONFIG_DYNAMIC_DEBUG_CORE build\n");
|
|
|
|
ddebug_init_success = 1;
|
|
|
|
return 0;
|
2011-12-20 06:12:54 +08:00
|
|
|
}
|
2020-07-20 07:10:45 +08:00
|
|
|
iter = __start___dyndbg;
|
2011-12-20 06:12:54 +08:00
|
|
|
modname = iter->modname;
|
|
|
|
iter_start = iter;
|
2020-07-20 07:10:45 +08:00
|
|
|
for (; iter < __stop___dyndbg; iter++) {
|
2012-04-28 04:30:39 +08:00
|
|
|
entries++;
|
2011-12-20 06:12:54 +08:00
|
|
|
if (strcmp(modname, iter->modname)) {
|
2012-04-28 04:30:39 +08:00
|
|
|
modct++;
|
2011-12-20 06:12:54 +08:00
|
|
|
ret = ddebug_add_module(iter_start, n, modname);
|
|
|
|
if (ret)
|
2012-04-28 04:30:38 +08:00
|
|
|
goto out_err;
|
2011-12-20 06:12:54 +08:00
|
|
|
n = 0;
|
|
|
|
modname = iter->modname;
|
|
|
|
iter_start = iter;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
2011-12-20 06:12:54 +08:00
|
|
|
n++;
|
2009-02-06 00:51:38 +08:00
|
|
|
}
|
2011-12-20 06:12:54 +08:00
|
|
|
ret = ddebug_add_module(iter_start, n, modname);
|
|
|
|
if (ret)
|
2012-04-28 04:30:38 +08:00
|
|
|
goto out_err;
|
2010-08-06 22:11:02 +08:00
|
|
|
|
2012-04-28 04:30:38 +08:00
|
|
|
ddebug_init_success = 1;
|
dyndbg: fix overcounting of ram used by dyndbg
during dyndbg init, verbose logging prints its ram overhead. It
counted strlens of struct _ddebug's 4 string members, in all callsite
entries, which would be approximately correct if each had been
mallocd. But they are pointers into shared .rodata; for example, all
10 kobject callsites have identical filename, module values.
Its best not to count that memory at all, since we cannot know they
were linked in because of CONFIG_DYNAMIC_DEBUG=y, and we want to
report a number that reflects what ram is saved by deconfiguring it.
Also fix wording and size under-reporting of the __dyndbg section.
Heres my overhead, on a virtme-run VM on a fedora-31 laptop:
dynamic_debug:dynamic_debug_init: 260 modules, 2479 entries \
and 10400 bytes in ddebug tables, 138824 bytes in __dyndbg section
Acked-by: <jbaron@akamai.com>
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Link: https://lore.kernel.org/r/20200719231058.1586423-7-jim.cromie@gmail.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-07-20 07:10:46 +08:00
|
|
|
vpr_info("%d modules, %d entries and %d bytes in ddebug tables, %d bytes in __dyndbg section\n",
|
2012-12-06 05:48:26 +08:00
|
|
|
modct, entries, (int)(modct * sizeof(struct ddebug_table)),
|
dyndbg: fix overcounting of ram used by dyndbg
during dyndbg init, verbose logging prints its ram overhead. It
counted strlens of struct _ddebug's 4 string members, in all callsite
entries, which would be approximately correct if each had been
mallocd. But they are pointers into shared .rodata; for example, all
10 kobject callsites have identical filename, module values.
Its best not to count that memory at all, since we cannot know they
were linked in because of CONFIG_DYNAMIC_DEBUG=y, and we want to
report a number that reflects what ram is saved by deconfiguring it.
Also fix wording and size under-reporting of the __dyndbg section.
Heres my overhead, on a virtme-run VM on a fedora-31 laptop:
dynamic_debug:dynamic_debug_init: 260 modules, 2479 entries \
and 10400 bytes in ddebug tables, 138824 bytes in __dyndbg section
Acked-by: <jbaron@akamai.com>
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Link: https://lore.kernel.org/r/20200719231058.1586423-7-jim.cromie@gmail.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2020-07-20 07:10:46 +08:00
|
|
|
(int)(entries * sizeof(struct _ddebug)));
|
2012-04-28 04:30:38 +08:00
|
|
|
|
|
|
|
/* apply ddebug_query boot param, dont unload tables on err */
|
2010-08-06 22:11:02 +08:00
|
|
|
if (ddebug_setup_string[0] != '\0') {
|
2012-12-06 05:48:26 +08:00
|
|
|
pr_warn("ddebug_query param name is deprecated, change it to dyndbg\n");
|
2012-04-28 04:30:40 +08:00
|
|
|
ret = ddebug_exec_queries(ddebug_setup_string, NULL);
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
if (ret < 0)
|
2012-12-06 05:48:26 +08:00
|
|
|
pr_warn("Invalid ddebug boot param %s\n",
|
2011-08-12 02:36:33 +08:00
|
|
|
ddebug_setup_string);
|
2010-08-06 22:11:02 +08:00
|
|
|
else
|
dynamic_debug: process multiple debug-queries on a line
Insert ddebug_exec_queries() in place of ddebug_exec_query(). It
splits the query string on [;\n], and calls ddebug_exec_query() on
each. All queries are processed independent of errors, allowing a
query to fail, for example when a module is not installed. Empty
lines and comments are skipped. Errors are counted, and the last
error seen (negative) or the number of callsites found (0 or positive)
is returned. Return code checks are altered accordingly.
With this, multiple queries can be given in ddebug_query, allowing
more selective enabling of callsites. As a side effect, a set of
commands can be batched in:
cat cmd-file > $DBGMT/dynamic_debug/control
We dont want a ddebug_query syntax error to kill the dynamic debug
facility, so dynamic_debug_init() zeros ddebug_exec_queries()'s return
code after logging the appropriate message, so that ddebug tables are
preserved and $DBGMT/dynamic_debug/control file is created. This
would be appropriate even without accepting multiple queries.
This patch also alters ddebug_change() to return number of callsites
matched (which typically is the same as number of callsites changed).
ddebug_exec_query() also returns the number found, or a negative value
if theres a parse error on the query.
Splitting on [;\n] prevents their use in format-specs, but selecting
callsites on punctuation is brittle anyway, meaningful and selective
substrings are more typical.
Note: splitting queries on ';' before handling trailing #comments
means that a ';' also terminates a comment, and text after the ';' is
treated as another query. This trailing query will almost certainly
result in a parse error and thus have no effect other than the error
message. The double corner case with unexpected results is:
ddebug_query="func foo +p # enable foo ; +p"
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
Signed-off-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
2011-12-20 06:13:21 +08:00
|
|
|
pr_info("%d changes by ddebug_query\n", ret);
|
2010-08-06 22:11:02 +08:00
|
|
|
}
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
/* now that ddebug tables are loaded, process all boot args
|
|
|
|
* again to find and activate queries given in dyndbg params.
|
|
|
|
* While this has already been done for known boot params, it
|
|
|
|
* ignored the unknown ones (dyndbg in particular). Reusing
|
|
|
|
* parse_args avoids ad-hoc parsing. This will also attempt
|
|
|
|
* to activate queries for not-yet-loaded modules, which is
|
|
|
|
* slightly noisy if verbose, but harmless.
|
|
|
|
*/
|
|
|
|
cmdline = kstrdup(saved_command_line, GFP_KERNEL);
|
|
|
|
parse_args("dyndbg params", cmdline, NULL,
|
module: add extra argument for parse_params() callback
This adds an extra argument onto parse_params() to be used
as a way to make the unused callback a bit more useful and
generic by allowing the caller to pass on a data structure
of its choice. An example use case is to allow us to easily
make module parameters for every module which we will do
next.
@ parse @
identifier name, args, params, num, level_min, level_max;
identifier unknown, param, val, doing;
type s16;
@@
extern char *parse_args(const char *name,
char *args,
const struct kernel_param *params,
unsigned num,
s16 level_min,
s16 level_max,
+ void *arg,
int (*unknown)(char *param, char *val,
const char *doing
+ , void *arg
));
@ parse_mod @
identifier name, args, params, num, level_min, level_max;
identifier unknown, param, val, doing;
type s16;
@@
char *parse_args(const char *name,
char *args,
const struct kernel_param *params,
unsigned num,
s16 level_min,
s16 level_max,
+ void *arg,
int (*unknown)(char *param, char *val,
const char *doing
+ , void *arg
))
{
...
}
@ parse_args_found @
expression R, E1, E2, E3, E4, E5, E6;
identifier func;
@@
(
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
func);
|
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
&func);
|
R =
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
NULL);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
func);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
&func);
|
parse_args(E1, E2, E3, E4, E5, E6,
+ NULL,
NULL);
)
@ parse_args_unused depends on parse_args_found @
identifier parse_args_found.func;
@@
int func(char *param, char *val, const char *unused
+ , void *arg
)
{
...
}
@ mod_unused depends on parse_args_found @
identifier parse_args_found.func;
expression A1, A2, A3;
@@
- func(A1, A2, A3);
+ func(A1, A2, A3, NULL);
Generated-by: Coccinelle SmPL
Cc: cocci@systeme.lip6.fr
Cc: Tejun Heo <tj@kernel.org>
Cc: Arjan van de Ven <arjan@linux.intel.com>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: Rusty Russell <rusty@rustcorp.com.au>
Cc: Christoph Hellwig <hch@infradead.org>
Cc: Felipe Contreras <felipe.contreras@gmail.com>
Cc: Ewan Milne <emilne@redhat.com>
Cc: Jean Delvare <jdelvare@suse.de>
Cc: Hannes Reinecke <hare@suse.de>
Cc: Jani Nikula <jani.nikula@intel.com>
Cc: linux-kernel@vger.kernel.org
Reviewed-by: Tejun Heo <tj@kernel.org>
Acked-by: Rusty Russell <rusty@rustcorp.com.au>
Signed-off-by: Luis R. Rodriguez <mcgrof@suse.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2015-03-31 07:20:03 +08:00
|
|
|
0, 0, 0, NULL, &ddebug_dyndbg_boot_param_cb);
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
kfree(cmdline);
|
2012-04-28 04:30:38 +08:00
|
|
|
return 0;
|
2010-08-06 22:11:02 +08:00
|
|
|
|
2012-04-28 04:30:38 +08:00
|
|
|
out_err:
|
|
|
|
ddebug_remove_all_tables();
|
2009-02-06 00:51:38 +08:00
|
|
|
return 0;
|
|
|
|
}
|
2010-08-06 22:11:03 +08:00
|
|
|
/* Allow early initialization for boot messages via boot param */
|
2012-04-28 04:30:42 +08:00
|
|
|
early_initcall(dynamic_debug_init);
|
dynamic_debug: make dynamic-debug work for module initialization
This introduces a fake module param $module.dyndbg. Its based upon
Thomas Renninger's $module.ddebug boot-time debugging patch from
https://lkml.org/lkml/2010/9/15/397
The 'fake' module parameter is provided for all modules, whether or
not they need it. It is not explicitly added to each module, but is
implemented in callbacks invoked from parse_args.
For builtin modules, dynamic_debug_init() now directly calls
parse_args(..., &ddebug_dyndbg_boot_params_cb), to process the params
undeclared in the modules, just after the ddebug tables are processed.
While its slightly weird to reprocess the boot params, parse_args() is
already called repeatedly by do_initcall_levels(). More importantly,
the dyndbg queries (given in ddebug_query or dyndbg params) cannot be
activated until after the ddebug tables are ready, and reusing
parse_args is cleaner than doing an ad-hoc parse. This reparse would
break options like inc_verbosity, but they probably should be params,
like verbosity=3.
ddebug_dyndbg_boot_params_cb() handles both bare dyndbg (aka:
ddebug_query) and module-prefixed dyndbg params, and ignores all other
parameters. For example, the following will enable pr_debug()s in 4
builtin modules, in the order given:
dyndbg="module params +p; module aio +p" module.dyndbg=+p pci.dyndbg
For loadable modules, parse_args() in load_module() calls
ddebug_dyndbg_module_params_cb(). This handles bare dyndbg params as
passed from modprobe, and errors on other unknown params.
Note that modprobe reads /proc/cmdline, so "modprobe foo" grabs all
foo.params, strips the "foo.", and passes these to the kernel.
ddebug_dyndbg_module_params_cb() is again called for the unknown
params; it handles dyndbg, and errors on others. The "doing" arg
added previously contains the module name.
For non CONFIG_DYNAMIC_DEBUG builds, the stub function accepts
and ignores $module.dyndbg params, other unknowns get -ENOENT.
If no param value is given (as in pci.dyndbg example above), "+p" is
assumed, which enables all pr_debug callsites in the module.
The dyndbg fake parameter is not shown in /sys/module/*/parameters,
thus it does not use any resources. Changes to it are made via the
control file.
Also change pr_info in ddebug_exec_queries to vpr_info,
no need to see it all the time.
Signed-off-by: Jim Cromie <jim.cromie@gmail.com>
CC: Thomas Renninger <trenn@suse.de>
CC: Rusty Russell <rusty@rustcorp.com.au>
Acked-by: Jason Baron <jbaron@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2012-04-28 04:30:35 +08:00
|
|
|
|
2010-08-06 22:11:03 +08:00
|
|
|
/* Debugfs setup must be done later */
|
2020-02-11 05:11:42 +08:00
|
|
|
fs_initcall(dynamic_debug_init_control);
|