License cleanup: add SPDX GPL-2.0 license identifier to files with no license
Many source files in the tree are missing licensing information, which
makes it harder for compliance tools to determine the correct license.
By default all files without license information are under the default
license of the kernel, which is GPL version 2.
Update the files which contain no license information with the 'GPL-2.0'
SPDX license identifier. The SPDX identifier is a legally binding
shorthand, which can be used instead of the full boiler plate text.
This patch is based on work done by Thomas Gleixner and Kate Stewart and
Philippe Ombredanne.
How this work was done:
Patches were generated and checked against linux-4.14-rc6 for a subset of
the use cases:
- file had no licensing information it it.
- file was a */uapi/* one with no licensing information in it,
- file was a */uapi/* one with existing licensing information,
Further patches will be generated in subsequent months to fix up cases
where non-standard license headers were used, and references to license
had to be inferred by heuristics based on keywords.
The analysis to determine which SPDX License Identifier to be applied to
a file was done in a spreadsheet of side by side results from of the
output of two independent scanners (ScanCode & Windriver) producing SPDX
tag:value files created by Philippe Ombredanne. Philippe prepared the
base worksheet, and did an initial spot review of a few 1000 files.
The 4.13 kernel was the starting point of the analysis with 60,537 files
assessed. Kate Stewart did a file by file comparison of the scanner
results in the spreadsheet to determine which SPDX license identifier(s)
to be applied to the file. She confirmed any determination that was not
immediately clear with lawyers working with the Linux Foundation.
Criteria used to select files for SPDX license identifier tagging was:
- Files considered eligible had to be source code files.
- Make and config files were included as candidates if they contained >5
lines of source
- File already had some variant of a license header in it (even if <5
lines).
All documentation files were explicitly excluded.
The following heuristics were used to determine which SPDX license
identifiers to apply.
- when both scanners couldn't find any license traces, file was
considered to have no license information in it, and the top level
COPYING file license applied.
For non */uapi/* files that summary was:
SPDX license identifier # files
---------------------------------------------------|-------
GPL-2.0 11139
and resulted in the first patch in this series.
If that file was a */uapi/* path one, it was "GPL-2.0 WITH
Linux-syscall-note" otherwise it was "GPL-2.0". Results of that was:
SPDX license identifier # files
---------------------------------------------------|-------
GPL-2.0 WITH Linux-syscall-note 930
and resulted in the second patch in this series.
- if a file had some form of licensing information in it, and was one
of the */uapi/* ones, it was denoted with the Linux-syscall-note if
any GPL family license was found in the file or had no licensing in
it (per prior point). Results summary:
SPDX license identifier # files
---------------------------------------------------|------
GPL-2.0 WITH Linux-syscall-note 270
GPL-2.0+ WITH Linux-syscall-note 169
((GPL-2.0 WITH Linux-syscall-note) OR BSD-2-Clause) 21
((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) 17
LGPL-2.1+ WITH Linux-syscall-note 15
GPL-1.0+ WITH Linux-syscall-note 14
((GPL-2.0+ WITH Linux-syscall-note) OR BSD-3-Clause) 5
LGPL-2.0+ WITH Linux-syscall-note 4
LGPL-2.1 WITH Linux-syscall-note 3
((GPL-2.0 WITH Linux-syscall-note) OR MIT) 3
((GPL-2.0 WITH Linux-syscall-note) AND MIT) 1
and that resulted in the third patch in this series.
- when the two scanners agreed on the detected license(s), that became
the concluded license(s).
- when there was disagreement between the two scanners (one detected a
license but the other didn't, or they both detected different
licenses) a manual inspection of the file occurred.
- In most cases a manual inspection of the information in the file
resulted in a clear resolution of the license that should apply (and
which scanner probably needed to revisit its heuristics).
- When it was not immediately clear, the license identifier was
confirmed with lawyers working with the Linux Foundation.
- If there was any question as to the appropriate license identifier,
the file was flagged for further research and to be revisited later
in time.
In total, over 70 hours of logged manual review was done on the
spreadsheet to determine the SPDX license identifiers to apply to the
source files by Kate, Philippe, Thomas and, in some cases, confirmation
by lawyers working with the Linux Foundation.
Kate also obtained a third independent scan of the 4.13 code base from
FOSSology, and compared selected files where the other two scanners
disagreed against that SPDX file, to see if there was new insights. The
Windriver scanner is based on an older version of FOSSology in part, so
they are related.
Thomas did random spot checks in about 500 files from the spreadsheets
for the uapi headers and agreed with SPDX license identifier in the
files he inspected. For the non-uapi files Thomas did random spot checks
in about 15000 files.
In initial set of patches against 4.14-rc6, 3 files were found to have
copy/paste license identifier errors, and have been fixed to reflect the
correct identifier.
Additionally Philippe spent 10 hours this week doing a detailed manual
inspection and review of the 12,461 patched files from the initial patch
version early this week with:
- a full scancode scan run, collecting the matched texts, detected
license ids and scores
- reviewing anything where there was a license detected (about 500+
files) to ensure that the applied SPDX license was correct
- reviewing anything where there was no detection but the patch license
was not GPL-2.0 WITH Linux-syscall-note to ensure that the applied
SPDX license was correct
This produced a worksheet with 20 files needing minor correction. This
worksheet was then exported into 3 different .csv files for the
different types of files to be modified.
These .csv files were then reviewed by Greg. Thomas wrote a script to
parse the csv files and add the proper SPDX tag to the file, in the
format that the file expected. This script was further refined by Greg
based on the output to detect more types of files automatically and to
distinguish between header and source .c files (which need different
comment types.) Finally Greg ran the script using the .csv files to
generate the patches.
Reviewed-by: Kate Stewart <kstewart@linuxfoundation.org>
Reviewed-by: Philippe Ombredanne <pombredanne@nexb.com>
Reviewed-by: Thomas Gleixner <tglx@linutronix.de>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2017-11-01 15:07:57 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
2010-03-21 22:31:26 -07:00
|
|
|
/*
|
2005-04-16 15:20:36 -07:00
|
|
|
* Linux Magic System Request Key Hacks
|
|
|
|
*
|
|
|
|
* (c) 1997 Martin Mares <mj@atrey.karlin.mff.cuni.cz>
|
|
|
|
* based on ideas by Pavel Machek <pavel@atrey.karlin.mff.cuni.cz>
|
|
|
|
*
|
|
|
|
* (c) 2000 Crutcher Dunnavant <crutcher+kernel@datastacks.com>
|
|
|
|
* overhauled to use key registration
|
|
|
|
* based upon discusions in irc://irc.openprojects.net/#kernelnewbies
|
2010-03-21 22:31:26 -07:00
|
|
|
*
|
|
|
|
* Copyright (c) 2010 Dmitry Torokhov
|
|
|
|
* Input handler conversion
|
2005-04-16 15:20:36 -07:00
|
|
|
*/
|
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
2017-02-08 18:51:30 +01:00
|
|
|
#include <linux/sched/signal.h>
|
2013-02-07 09:47:07 -06:00
|
|
|
#include <linux/sched/rt.h>
|
2017-02-08 18:51:35 +01:00
|
|
|
#include <linux/sched/debug.h>
|
2017-02-08 18:51:36 +01:00
|
|
|
#include <linux/sched/task.h>
|
2020-08-18 13:28:24 +02:00
|
|
|
#include <linux/ctype.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
#include <linux/interrupt.h>
|
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/mount.h>
|
|
|
|
#include <linux/kdev_t.h>
|
|
|
|
#include <linux/major.h>
|
|
|
|
#include <linux/reboot.h>
|
|
|
|
#include <linux/sysrq.h>
|
|
|
|
#include <linux/kbd_kern.h>
|
2008-10-15 22:04:23 -07:00
|
|
|
#include <linux/proc_fs.h>
|
2009-08-02 11:28:21 +02:00
|
|
|
#include <linux/nmi.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
#include <linux/quotaops.h>
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 12:02:48 +02:00
|
|
|
#include <linux/perf_event.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/suspend.h>
|
|
|
|
#include <linux/writeback.h>
|
|
|
|
#include <linux/swap.h>
|
|
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/vt_kern.h>
|
|
|
|
#include <linux/workqueue.h>
|
2007-02-16 01:28:16 -08:00
|
|
|
#include <linux/hrtimer.h>
|
2007-10-16 23:25:53 -07:00
|
|
|
#include <linux/oom.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 17:04:11 +09:00
|
|
|
#include <linux/slab.h>
|
2010-03-21 22:31:26 -07:00
|
|
|
#include <linux/input.h>
|
2011-09-16 02:31:11 -04:00
|
|
|
#include <linux/uaccess.h>
|
2013-01-06 23:23:33 -08:00
|
|
|
#include <linux/moduleparam.h>
|
2013-04-01 22:14:19 -07:00
|
|
|
#include <linux/jiffies.h>
|
2013-06-05 22:51:46 -07:00
|
|
|
#include <linux/syscalls.h>
|
2013-08-03 17:22:08 -07:00
|
|
|
#include <linux/of.h>
|
2014-06-06 14:38:14 -07:00
|
|
|
#include <linux/rcupdate.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
#include <asm/ptrace.h>
|
2006-10-06 16:38:42 +02:00
|
|
|
#include <asm/irq_regs.h>
|
2005-04-16 15:20:36 -07:00
|
|
|
|
|
|
|
/* Whether we react on sysrq keys or just ignore them */
|
2013-10-07 01:05:46 +01:00
|
|
|
static int __read_mostly sysrq_enabled = CONFIG_MAGIC_SYSRQ_DEFAULT_ENABLE;
|
2010-03-21 22:31:26 -07:00
|
|
|
static bool __read_mostly sysrq_always_enabled;
|
2006-12-13 00:34:36 -08:00
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
static bool sysrq_on(void)
|
2006-12-13 00:34:36 -08:00
|
|
|
{
|
2010-03-21 22:31:26 -07:00
|
|
|
return sysrq_enabled || sysrq_always_enabled;
|
2006-12-13 00:34:36 -08:00
|
|
|
}
|
|
|
|
|
2020-03-02 17:51:34 +00:00
|
|
|
/**
|
|
|
|
* sysrq_mask - Getter for sysrq_enabled mask.
|
|
|
|
*
|
|
|
|
* Return: 1 if sysrq is always enabled, enabled sysrq_key_op mask otherwise.
|
|
|
|
*/
|
|
|
|
int sysrq_mask(void)
|
|
|
|
{
|
|
|
|
if (sysrq_always_enabled)
|
|
|
|
return 1;
|
|
|
|
return sysrq_enabled;
|
|
|
|
}
|
2020-04-20 18:23:17 +01:00
|
|
|
EXPORT_SYMBOL_GPL(sysrq_mask);
|
2020-03-02 17:51:34 +00:00
|
|
|
|
2006-12-13 00:34:36 -08:00
|
|
|
/*
|
|
|
|
* A value of 1 means 'all', other nonzero values are an op mask:
|
|
|
|
*/
|
2010-03-21 22:31:26 -07:00
|
|
|
static bool sysrq_on_mask(int mask)
|
2006-12-13 00:34:36 -08:00
|
|
|
{
|
2010-03-21 22:31:26 -07:00
|
|
|
return sysrq_always_enabled ||
|
|
|
|
sysrq_enabled == 1 ||
|
|
|
|
(sysrq_enabled & mask);
|
2006-12-13 00:34:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int __init sysrq_always_enabled_setup(char *str)
|
|
|
|
{
|
2010-03-21 22:31:26 -07:00
|
|
|
sysrq_always_enabled = true;
|
|
|
|
pr_info("sysrq always enabled.\n");
|
2006-12-13 00:34:36 -08:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
__setup("sysrq_always_enabled", sysrq_always_enabled_setup);
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_loglevel(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
int i;
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
i = key - '0';
|
2014-06-04 16:11:46 -07:00
|
|
|
console_loglevel = CONSOLE_LOGLEVEL_DEFAULT;
|
2015-02-11 15:26:21 -08:00
|
|
|
pr_info("Loglevel set to %d\n", i);
|
2005-04-16 15:20:36 -07:00
|
|
|
console_loglevel = i;
|
2006-03-25 03:07:08 -08:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_loglevel_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_loglevel,
|
2009-01-06 14:41:13 -08:00
|
|
|
.help_msg = "loglevel(0-9)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Changing Loglevel",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_LOG,
|
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_VT
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_SAK(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2007-02-10 01:44:34 -08:00
|
|
|
struct work_struct *SAK_work = &vc_cons[fg_console].SAK_work;
|
2021-04-07 15:06:41 +08:00
|
|
|
|
2007-02-10 01:44:34 -08:00
|
|
|
schedule_work(SAK_work);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_SAK_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_SAK,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "sak(k)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "SAK",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_KEYBOARD,
|
|
|
|
};
|
2006-03-25 03:07:08 -08:00
|
|
|
#else
|
2020-05-13 22:43:43 +01:00
|
|
|
#define sysrq_SAK_op (*(const struct sysrq_key_op *)NULL)
|
2005-04-16 15:20:36 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CONFIG_VT
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_unraw(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2012-02-28 14:49:23 +00:00
|
|
|
vt_reset_unicode(fg_console);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2012-02-28 14:49:23 +00:00
|
|
|
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_unraw_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_unraw,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "unraw(r)",
|
2007-10-16 23:29:38 -07:00
|
|
|
.action_msg = "Keyboard mode set to system default",
|
2005-04-16 15:20:36 -07:00
|
|
|
.enable_mask = SYSRQ_ENABLE_KEYBOARD,
|
|
|
|
};
|
2006-03-25 03:07:08 -08:00
|
|
|
#else
|
2020-05-13 22:43:43 +01:00
|
|
|
#define sysrq_unraw_op (*(const struct sysrq_key_op *)NULL)
|
2005-04-16 15:20:36 -07:00
|
|
|
#endif /* CONFIG_VT */
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_crash(int key)
|
2005-06-25 14:58:25 -07:00
|
|
|
{
|
2018-09-20 10:12:53 -07:00
|
|
|
/* release the RCU read lock before crashing */
|
2015-12-17 17:15:10 -08:00
|
|
|
rcu_read_unlock();
|
2018-09-20 10:12:53 -07:00
|
|
|
|
|
|
|
panic("sysrq triggered crash\n");
|
2005-06-25 14:58:25 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_crash_op = {
|
2009-06-17 16:28:17 -07:00
|
|
|
.handler = sysrq_handle_crash,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "crash(c)",
|
2009-06-17 16:28:17 -07:00
|
|
|
.action_msg = "Trigger a crash",
|
2005-06-25 14:58:25 -07:00
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_reboot(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2006-09-30 23:28:02 -07:00
|
|
|
lockdep_off();
|
2005-04-16 15:20:36 -07:00
|
|
|
local_irq_enable();
|
2005-07-26 11:51:06 -06:00
|
|
|
emergency_restart();
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_reboot_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_reboot,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "reboot(b)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Resetting",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_BOOT,
|
|
|
|
};
|
|
|
|
|
2020-05-13 22:43:43 +01:00
|
|
|
const struct sysrq_key_op *__sysrq_reboot_op = &sysrq_reboot_op;
|
2020-05-13 22:43:41 +01:00
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_sync(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
emergency_sync();
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_sync_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_sync,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "sync(s)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Emergency Sync",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_SYNC,
|
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_show_timers(int key)
|
2007-02-16 01:28:16 -08:00
|
|
|
{
|
|
|
|
sysrq_timer_list_show();
|
|
|
|
}
|
|
|
|
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_show_timers_op = {
|
2007-02-16 01:28:16 -08:00
|
|
|
.handler = sysrq_handle_show_timers,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-all-timers(q)",
|
2008-10-20 12:33:14 +02:00
|
|
|
.action_msg = "Show clockevent devices & pending hrtimers (no others)",
|
2007-02-16 01:28:16 -08:00
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_mountro(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
emergency_remount();
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_mountro_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_mountro,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "unmount(u)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Emergency Remount R/O",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_REMOUNT,
|
|
|
|
};
|
|
|
|
|
2006-07-03 00:24:56 -07:00
|
|
|
#ifdef CONFIG_LOCKDEP
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showlocks(int key)
|
2006-01-09 15:59:21 -08:00
|
|
|
{
|
2006-07-03 00:24:33 -07:00
|
|
|
debug_show_all_locks();
|
2006-01-09 15:59:21 -08:00
|
|
|
}
|
2006-07-03 00:24:56 -07:00
|
|
|
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showlocks_op = {
|
2006-01-09 15:59:21 -08:00
|
|
|
.handler = sysrq_handle_showlocks,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-all-locks(d)",
|
2006-01-09 15:59:21 -08:00
|
|
|
.action_msg = "Show Locks Held",
|
|
|
|
};
|
2006-03-25 03:07:08 -08:00
|
|
|
#else
|
2020-05-13 22:43:43 +01:00
|
|
|
#define sysrq_showlocks_op (*(const struct sysrq_key_op *)NULL)
|
2006-01-09 15:59:21 -08:00
|
|
|
#endif
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2008-04-29 00:59:21 -07:00
|
|
|
#ifdef CONFIG_SMP
|
2019-03-13 11:40:34 +00:00
|
|
|
static DEFINE_RAW_SPINLOCK(show_lock);
|
2008-04-29 00:59:21 -07:00
|
|
|
|
|
|
|
static void showacpu(void *dummy)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
/* Idle CPUs have no interesting backtrace. */
|
2022-01-17 23:43:00 +08:00
|
|
|
if (idle_cpu(smp_processor_id())) {
|
|
|
|
pr_info("CPU%d: backtrace skipped as idling\n", smp_processor_id());
|
2008-04-29 00:59:21 -07:00
|
|
|
return;
|
2022-01-17 23:43:00 +08:00
|
|
|
}
|
2008-04-29 00:59:21 -07:00
|
|
|
|
2019-03-13 11:40:34 +00:00
|
|
|
raw_spin_lock_irqsave(&show_lock, flags);
|
2015-02-11 15:26:21 -08:00
|
|
|
pr_info("CPU%d:\n", smp_processor_id());
|
2020-06-08 21:32:29 -07:00
|
|
|
show_stack(NULL, NULL, KERN_INFO);
|
2019-03-13 11:40:34 +00:00
|
|
|
raw_spin_unlock_irqrestore(&show_lock, flags);
|
2008-04-29 00:59:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sysrq_showregs_othercpus(struct work_struct *dummy)
|
|
|
|
{
|
2008-06-27 11:52:45 +02:00
|
|
|
smp_call_function(showacpu, NULL, 0);
|
2008-04-29 00:59:21 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static DECLARE_WORK(sysrq_showallcpus, sysrq_showregs_othercpus);
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showallcpus(int key)
|
2008-04-29 00:59:21 -07:00
|
|
|
{
|
2009-08-03 09:31:54 +02:00
|
|
|
/*
|
|
|
|
* Fall back to the workqueue based printing if the
|
|
|
|
* backtrace printing did not succeed or the
|
|
|
|
* architecture has no support for it:
|
|
|
|
*/
|
|
|
|
if (!trigger_all_cpu_backtrace()) {
|
2017-09-10 20:11:42 -07:00
|
|
|
struct pt_regs *regs = NULL;
|
2009-08-03 09:31:54 +02:00
|
|
|
|
2021-08-14 08:50:33 +08:00
|
|
|
if (in_hardirq())
|
2017-09-10 20:11:42 -07:00
|
|
|
regs = get_irq_regs();
|
2022-01-17 23:43:00 +08:00
|
|
|
|
|
|
|
pr_info("CPU%d:\n", smp_processor_id());
|
|
|
|
if (regs)
|
2009-08-03 09:31:54 +02:00
|
|
|
show_regs(regs);
|
2022-01-17 23:43:00 +08:00
|
|
|
else
|
|
|
|
show_stack(NULL, NULL, KERN_INFO);
|
|
|
|
|
2009-08-03 09:31:54 +02:00
|
|
|
schedule_work(&sysrq_showallcpus);
|
|
|
|
}
|
2008-04-29 00:59:21 -07:00
|
|
|
}
|
|
|
|
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showallcpus_op = {
|
2008-04-29 00:59:21 -07:00
|
|
|
.handler = sysrq_handle_showallcpus,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-backtrace-all-active-cpus(l)",
|
2008-04-29 00:59:21 -07:00
|
|
|
.action_msg = "Show backtrace of all active CPUs",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
2022-04-18 15:37:03 +00:00
|
|
|
#else
|
|
|
|
#define sysrq_showallcpus_op (*(const struct sysrq_key_op *)NULL)
|
2008-04-29 00:59:21 -07:00
|
|
|
#endif
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showregs(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2017-09-10 20:11:42 -07:00
|
|
|
struct pt_regs *regs = NULL;
|
|
|
|
|
2021-08-14 08:50:33 +08:00
|
|
|
if (in_hardirq())
|
2017-09-10 20:11:42 -07:00
|
|
|
regs = get_irq_regs();
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 14:55:46 +01:00
|
|
|
if (regs)
|
|
|
|
show_regs(regs);
|
perf: Do the big rename: Performance Counters -> Performance Events
Bye-bye Performance Counters, welcome Performance Events!
In the past few months the perfcounters subsystem has grown out its
initial role of counting hardware events, and has become (and is
becoming) a much broader generic event enumeration, reporting, logging,
monitoring, analysis facility.
Naming its core object 'perf_counter' and naming the subsystem
'perfcounters' has become more and more of a misnomer. With pending
code like hw-breakpoints support the 'counter' name is less and
less appropriate.
All in one, we've decided to rename the subsystem to 'performance
events' and to propagate this rename through all fields, variables
and API names. (in an ABI compatible fashion)
The word 'event' is also a bit shorter than 'counter' - which makes
it slightly more convenient to write/handle as well.
Thanks goes to Stephane Eranian who first observed this misnomer and
suggested a rename.
User-space tooling and ABI compatibility is not affected - this patch
should be function-invariant. (Also, defconfigs were not touched to
keep the size down.)
This patch has been generated via the following script:
FILES=$(find * -type f | grep -vE 'oprofile|[^K]config')
sed -i \
-e 's/PERF_EVENT_/PERF_RECORD_/g' \
-e 's/PERF_COUNTER/PERF_EVENT/g' \
-e 's/perf_counter/perf_event/g' \
-e 's/nb_counters/nb_events/g' \
-e 's/swcounter/swevent/g' \
-e 's/tpcounter_event/tp_event/g' \
$FILES
for N in $(find . -name perf_counter.[ch]); do
M=$(echo $N | sed 's/perf_counter/perf_event/g')
mv $N $M
done
FILES=$(find . -name perf_event.*)
sed -i \
-e 's/COUNTER_MASK/REG_MASK/g' \
-e 's/COUNTER/EVENT/g' \
-e 's/\<event\>/event_id/g' \
-e 's/counter/event/g' \
-e 's/Counter/Event/g' \
$FILES
... to keep it as correct as possible. This script can also be
used by anyone who has pending perfcounters patches - it converts
a Linux kernel tree over to the new naming. We tried to time this
change to the point in time where the amount of pending patches
is the smallest: the end of the merge window.
Namespace clashes were fixed up in a preparatory patch - and some
stylistic fallout will be fixed up in a subsequent patch.
( NOTE: 'counters' are still the proper terminology when we deal
with hardware registers - and these sed scripts are a bit
over-eager in renaming them. I've undone some of that, but
in case there's something left where 'counter' would be
better than 'event' we can undo that on an individual basis
instead of touching an otherwise nicely automated patch. )
Suggested-by: Stephane Eranian <eranian@google.com>
Acked-by: Peter Zijlstra <a.p.zijlstra@chello.nl>
Acked-by: Paul Mackerras <paulus@samba.org>
Reviewed-by: Arjan van de Ven <arjan@linux.intel.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Steven Rostedt <rostedt@goodmis.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: David Howells <dhowells@redhat.com>
Cc: Kyle McMartin <kyle@mcmartin.ca>
Cc: Martin Schwidefsky <schwidefsky@de.ibm.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: "H. Peter Anvin" <hpa@zytor.com>
Cc: <linux-arch@vger.kernel.org>
LKML-Reference: <new-submission>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
2009-09-21 12:02:48 +02:00
|
|
|
perf_event_print_debug();
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showregs_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_showregs,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-registers(p)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Show Regs",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showstate(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
show_state();
|
2021-10-20 14:09:00 +11:00
|
|
|
show_all_workqueues();
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showstate_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_showstate,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-task-states(t)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Show State",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showstate_blocked(int key)
|
2006-12-06 20:35:59 -08:00
|
|
|
{
|
|
|
|
show_state_filter(TASK_UNINTERRUPTIBLE);
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showstate_blocked_op = {
|
2006-12-06 20:35:59 -08:00
|
|
|
.handler = sysrq_handle_showstate_blocked,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-blocked-tasks(w)",
|
2006-12-06 20:35:59 -08:00
|
|
|
.action_msg = "Show Blocked State",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
|
2008-11-01 19:53:34 +01:00
|
|
|
#ifdef CONFIG_TRACING
|
|
|
|
#include <linux/ftrace.h>
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_ftrace_dump(int key)
|
2008-11-01 19:53:34 +01:00
|
|
|
{
|
2010-04-18 19:08:41 +02:00
|
|
|
ftrace_dump(DUMP_ALL);
|
2008-11-01 19:53:34 +01:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_ftrace_dump_op = {
|
2008-11-01 19:53:34 +01:00
|
|
|
.handler = sysrq_ftrace_dump,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "dump-ftrace-buffer(z)",
|
2008-11-01 19:53:34 +01:00
|
|
|
.action_msg = "Dump ftrace buffer",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
#else
|
2020-05-13 22:43:43 +01:00
|
|
|
#define sysrq_ftrace_dump_op (*(const struct sysrq_key_op *)NULL)
|
2008-11-01 19:53:34 +01:00
|
|
|
#endif
|
2006-12-06 20:35:59 -08:00
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_showmem(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2017-02-22 15:46:16 -08:00
|
|
|
show_mem(0, NULL);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_showmem_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_showmem,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "show-memory-usage(m)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Show Memory",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
|
|
};
|
|
|
|
|
2006-03-25 03:07:08 -08:00
|
|
|
/*
|
|
|
|
* Signal sysrq helper function. Sends a signal to all user processes.
|
|
|
|
*/
|
2005-04-16 15:20:36 -07:00
|
|
|
static void send_sig_all(int sig)
|
|
|
|
{
|
|
|
|
struct task_struct *p;
|
|
|
|
|
2012-02-07 10:49:39 +04:00
|
|
|
read_lock(&tasklist_lock);
|
2005-04-16 15:20:36 -07:00
|
|
|
for_each_process(p) {
|
2012-02-07 10:49:51 +04:00
|
|
|
if (p->flags & PF_KTHREAD)
|
|
|
|
continue;
|
|
|
|
if (is_global_init(p))
|
|
|
|
continue;
|
|
|
|
|
2018-07-21 11:35:55 -05:00
|
|
|
do_send_sig_info(sig, SEND_SIG_PRIV, p, PIDTYPE_MAX);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2012-02-07 10:49:39 +04:00
|
|
|
read_unlock(&tasklist_lock);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_term(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
send_sig_all(SIGTERM);
|
2014-06-04 16:11:46 -07:00
|
|
|
console_loglevel = CONSOLE_LOGLEVEL_DEBUG;
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_term_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_term,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "terminate-all-tasks(e)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Terminate All Tasks",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
|
|
|
};
|
|
|
|
|
2006-11-22 14:55:48 +00:00
|
|
|
static void moom_callback(struct work_struct *ignored)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2015-09-08 15:00:36 -07:00
|
|
|
const gfp_t gfp_mask = GFP_KERNEL;
|
|
|
|
struct oom_control oc = {
|
|
|
|
.zonelist = node_zonelist(first_memory_node, gfp_mask),
|
|
|
|
.nodemask = NULL,
|
2016-07-26 15:22:33 -07:00
|
|
|
.memcg = NULL,
|
2015-09-08 15:00:36 -07:00
|
|
|
.gfp_mask = gfp_mask,
|
2015-09-08 15:00:39 -07:00
|
|
|
.order = -1,
|
2015-09-08 15:00:36 -07:00
|
|
|
};
|
|
|
|
|
2015-06-24 16:57:19 -07:00
|
|
|
mutex_lock(&oom_lock);
|
2015-09-08 15:00:36 -07:00
|
|
|
if (!out_of_memory(&oc))
|
2017-05-03 14:54:57 -07:00
|
|
|
pr_info("OOM request ignored. No task eligible\n");
|
2015-06-24 16:57:19 -07:00
|
|
|
mutex_unlock(&oom_lock);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
|
2006-11-22 14:55:48 +00:00
|
|
|
static DECLARE_WORK(moom_work, moom_callback);
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_moom(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
schedule_work(&moom_work);
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_moom_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_moom,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "memory-full-oom-kill(f)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Manual OOM execution",
|
2008-10-15 22:01:43 -07:00
|
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
2005-04-16 15:20:36 -07:00
|
|
|
};
|
|
|
|
|
2022-04-18 15:37:03 +00:00
|
|
|
#ifdef CONFIG_BLOCK
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_thaw(int key)
|
2009-03-31 15:23:46 -07:00
|
|
|
{
|
|
|
|
emergency_thaw_all();
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_thaw_op = {
|
2009-03-31 15:23:46 -07:00
|
|
|
.handler = sysrq_handle_thaw,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "thaw-filesystems(j)",
|
2009-03-31 15:23:46 -07:00
|
|
|
.action_msg = "Emergency Thaw of all frozen filesystems",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
|
|
|
};
|
2022-04-18 15:37:03 +00:00
|
|
|
#else
|
|
|
|
#define sysrq_thaw_op (*(const struct sysrq_key_op *)NULL)
|
|
|
|
#endif
|
2009-03-31 15:23:46 -07:00
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_kill(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
send_sig_all(SIGKILL);
|
2014-06-04 16:11:46 -07:00
|
|
|
console_loglevel = CONSOLE_LOGLEVEL_DEBUG;
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_kill_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_kill,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "kill-all-tasks(i)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Kill All Tasks",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
|
|
|
};
|
|
|
|
|
2010-08-17 21:15:46 -07:00
|
|
|
static void sysrq_handle_unrt(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
normalize_rt_tasks();
|
|
|
|
}
|
2020-05-13 22:43:43 +01:00
|
|
|
static const struct sysrq_key_op sysrq_unrt_op = {
|
2005-04-16 15:20:36 -07:00
|
|
|
.handler = sysrq_handle_unrt,
|
2013-03-07 17:00:02 +08:00
|
|
|
.help_msg = "nice-all-RT-tasks(n)",
|
2005-04-16 15:20:36 -07:00
|
|
|
.action_msg = "Nice All RT Tasks",
|
|
|
|
.enable_mask = SYSRQ_ENABLE_RTNICE,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Key Operations table and lock */
|
|
|
|
static DEFINE_SPINLOCK(sysrq_key_table_lock);
|
2006-03-25 03:07:08 -08:00
|
|
|
|
2020-08-18 13:28:24 +02:00
|
|
|
static const struct sysrq_key_op *sysrq_key_table[62] = {
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_loglevel_op, /* 0 */
|
|
|
|
&sysrq_loglevel_op, /* 1 */
|
|
|
|
&sysrq_loglevel_op, /* 2 */
|
|
|
|
&sysrq_loglevel_op, /* 3 */
|
|
|
|
&sysrq_loglevel_op, /* 4 */
|
|
|
|
&sysrq_loglevel_op, /* 5 */
|
|
|
|
&sysrq_loglevel_op, /* 6 */
|
|
|
|
&sysrq_loglevel_op, /* 7 */
|
|
|
|
&sysrq_loglevel_op, /* 8 */
|
|
|
|
&sysrq_loglevel_op, /* 9 */
|
|
|
|
|
|
|
|
/*
|
2007-01-31 23:48:17 -08:00
|
|
|
* a: Don't use for system provided sysrqs, it is handled specially on
|
|
|
|
* sparc and will never arrive.
|
2006-03-25 03:07:08 -08:00
|
|
|
*/
|
|
|
|
NULL, /* a */
|
|
|
|
&sysrq_reboot_op, /* b */
|
2017-04-03 22:50:20 -07:00
|
|
|
&sysrq_crash_op, /* c */
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_showlocks_op, /* d */
|
|
|
|
&sysrq_term_op, /* e */
|
|
|
|
&sysrq_moom_op, /* f */
|
2009-05-13 21:56:59 -05:00
|
|
|
/* g: May be registered for the kernel debugger */
|
2006-03-25 03:07:08 -08:00
|
|
|
NULL, /* g */
|
2009-03-31 15:23:46 -07:00
|
|
|
NULL, /* h - reserved for help */
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_kill_op, /* i */
|
2009-03-31 15:23:46 -07:00
|
|
|
&sysrq_thaw_op, /* j */
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_SAK_op, /* k */
|
2008-04-29 00:59:21 -07:00
|
|
|
&sysrq_showallcpus_op, /* l */
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_showmem_op, /* m */
|
|
|
|
&sysrq_unrt_op, /* n */
|
2007-01-31 23:48:17 -08:00
|
|
|
/* o: This will often be registered as 'Off' at init time */
|
2006-03-25 03:07:08 -08:00
|
|
|
NULL, /* o */
|
|
|
|
&sysrq_showregs_op, /* p */
|
2007-02-16 01:28:16 -08:00
|
|
|
&sysrq_show_timers_op, /* q */
|
2007-01-31 23:48:17 -08:00
|
|
|
&sysrq_unraw_op, /* r */
|
2006-03-25 03:07:08 -08:00
|
|
|
&sysrq_sync_op, /* s */
|
|
|
|
&sysrq_showstate_op, /* t */
|
|
|
|
&sysrq_mountro_op, /* u */
|
2009-05-13 21:56:59 -05:00
|
|
|
/* v: May be registered for frame buffer console restore */
|
2006-03-25 03:07:08 -08:00
|
|
|
NULL, /* v */
|
2007-01-31 23:48:17 -08:00
|
|
|
&sysrq_showstate_blocked_op, /* w */
|
2015-05-19 09:50:29 +01:00
|
|
|
/* x: May be registered on mips for TLB dump */
|
2007-01-31 23:48:17 -08:00
|
|
|
/* x: May be registered on ppc/powerpc for xmon */
|
2012-10-16 09:34:01 -07:00
|
|
|
/* x: May be registered on sparc64 for global PMU dump */
|
2007-01-31 23:48:17 -08:00
|
|
|
NULL, /* x */
|
2008-05-19 23:46:00 -07:00
|
|
|
/* y: May be registered on sparc64 for global register dump */
|
2006-03-25 03:07:08 -08:00
|
|
|
NULL, /* y */
|
2008-11-01 19:53:34 +01:00
|
|
|
&sysrq_ftrace_dump_op, /* z */
|
2020-08-18 13:28:24 +02:00
|
|
|
NULL, /* A */
|
|
|
|
NULL, /* B */
|
|
|
|
NULL, /* C */
|
|
|
|
NULL, /* D */
|
|
|
|
NULL, /* E */
|
|
|
|
NULL, /* F */
|
|
|
|
NULL, /* G */
|
|
|
|
NULL, /* H */
|
|
|
|
NULL, /* I */
|
|
|
|
NULL, /* J */
|
|
|
|
NULL, /* K */
|
|
|
|
NULL, /* L */
|
|
|
|
NULL, /* M */
|
|
|
|
NULL, /* N */
|
|
|
|
NULL, /* O */
|
|
|
|
NULL, /* P */
|
|
|
|
NULL, /* Q */
|
|
|
|
NULL, /* R */
|
|
|
|
NULL, /* S */
|
|
|
|
NULL, /* T */
|
|
|
|
NULL, /* U */
|
|
|
|
NULL, /* V */
|
|
|
|
NULL, /* W */
|
|
|
|
NULL, /* X */
|
|
|
|
NULL, /* Y */
|
|
|
|
NULL, /* Z */
|
2005-04-16 15:20:36 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/* key2index calculation, -1 on invalid index */
|
2006-03-25 03:07:08 -08:00
|
|
|
static int sysrq_key_table_key2index(int key)
|
|
|
|
{
|
2005-04-16 15:20:36 -07:00
|
|
|
int retval;
|
2006-03-25 03:07:08 -08:00
|
|
|
|
|
|
|
if ((key >= '0') && (key <= '9'))
|
2005-04-16 15:20:36 -07:00
|
|
|
retval = key - '0';
|
2006-03-25 03:07:08 -08:00
|
|
|
else if ((key >= 'a') && (key <= 'z'))
|
2005-04-16 15:20:36 -07:00
|
|
|
retval = key + 10 - 'a';
|
2020-08-18 13:28:24 +02:00
|
|
|
else if ((key >= 'A') && (key <= 'Z'))
|
|
|
|
retval = key + 36 - 'A';
|
2006-03-25 03:07:08 -08:00
|
|
|
else
|
2005-04-16 15:20:36 -07:00
|
|
|
retval = -1;
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* get and put functions for the table, exposed to modules.
|
|
|
|
*/
|
2020-05-13 22:43:42 +01:00
|
|
|
static const struct sysrq_key_op *__sysrq_get_key_op(int key)
|
2006-03-25 03:07:08 -08:00
|
|
|
{
|
2021-04-07 15:06:42 +08:00
|
|
|
const struct sysrq_key_op *op_p = NULL;
|
|
|
|
int i;
|
2006-03-25 03:07:08 -08:00
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
i = sysrq_key_table_key2index(key);
|
2006-03-25 03:07:08 -08:00
|
|
|
if (i != -1)
|
2021-04-07 15:06:42 +08:00
|
|
|
op_p = sysrq_key_table[i];
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2021-04-07 15:06:42 +08:00
|
|
|
return op_p;
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
|
2020-05-13 22:43:42 +01:00
|
|
|
static void __sysrq_put_key_op(int key, const struct sysrq_key_op *op_p)
|
2006-03-25 03:07:08 -08:00
|
|
|
{
|
2021-04-07 15:06:42 +08:00
|
|
|
int i = sysrq_key_table_key2index(key);
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2021-04-07 15:06:42 +08:00
|
|
|
if (i != -1)
|
|
|
|
sysrq_key_table[i] = op_p;
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
|
2010-08-17 21:15:47 -07:00
|
|
|
void __handle_sysrq(int key, bool check_mask)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2020-05-13 22:43:42 +01:00
|
|
|
const struct sysrq_key_op *op_p;
|
2005-04-16 15:20:36 -07:00
|
|
|
int orig_log_level;
|
panic: avoid the extra noise dmesg
When kernel panic happens, it will first print the panic call stack,
then the ending msg like:
[ 35.743249] ---[ end Kernel panic - not syncing: Fatal exception
[ 35.749975] ------------[ cut here ]------------
The above message are very useful for debugging.
But if system is configured to not reboot on panic, say the
"panic_timeout" parameter equals 0, it will likely print out many noisy
message like WARN() call stack for each and every CPU except the panic
one, messages like below:
WARNING: CPU: 1 PID: 280 at kernel/sched/core.c:1198 set_task_cpu+0x183/0x190
Call Trace:
<IRQ>
try_to_wake_up
default_wake_function
autoremove_wake_function
__wake_up_common
__wake_up_common_lock
__wake_up
wake_up_klogd_work_func
irq_work_run_list
irq_work_tick
update_process_times
tick_sched_timer
__hrtimer_run_queues
hrtimer_interrupt
smp_apic_timer_interrupt
apic_timer_interrupt
For people working in console mode, the screen will first show the panic
call stack, but immediately overridden by these noisy extra messages,
which makes debugging much more difficult, as the original context gets
lost on screen.
Also these noisy messages will confuse some users, as I have seen many bug
reporters posted the noisy message into bugzilla, instead of the real
panic call stack and context.
Adding a flag "suppress_printk" which gets set in panic() to avoid those
noisy messages, without changing current kernel behavior that both panic
blinking and sysrq magic key can work as is, suggested by Petr Mladek.
To verify this, make sure kernel is not configured to reboot on panic and
in console
# echo c > /proc/sysrq-trigger
to see if console only prints out the panic call stack.
Link: http://lkml.kernel.org/r/1551430186-24169-1-git-send-email-feng.tang@intel.com
Signed-off-by: Feng Tang <feng.tang@intel.com>
Suggested-by: Petr Mladek <pmladek@suse.com>
Reviewed-by: Petr Mladek <pmladek@suse.com>
Acked-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
Acked-by: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Kees Cook <keescook@chromium.org>
Cc: Borislav Petkov <bp@suse.de>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: Jiri Slaby <jslaby@suse.com>
Cc: Sasha Levin <sashal@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2019-05-14 15:45:34 -07:00
|
|
|
int orig_suppress_printk;
|
2006-03-25 03:07:08 -08:00
|
|
|
int i;
|
2005-04-16 15:20:36 -07:00
|
|
|
|
panic: avoid the extra noise dmesg
When kernel panic happens, it will first print the panic call stack,
then the ending msg like:
[ 35.743249] ---[ end Kernel panic - not syncing: Fatal exception
[ 35.749975] ------------[ cut here ]------------
The above message are very useful for debugging.
But if system is configured to not reboot on panic, say the
"panic_timeout" parameter equals 0, it will likely print out many noisy
message like WARN() call stack for each and every CPU except the panic
one, messages like below:
WARNING: CPU: 1 PID: 280 at kernel/sched/core.c:1198 set_task_cpu+0x183/0x190
Call Trace:
<IRQ>
try_to_wake_up
default_wake_function
autoremove_wake_function
__wake_up_common
__wake_up_common_lock
__wake_up
wake_up_klogd_work_func
irq_work_run_list
irq_work_tick
update_process_times
tick_sched_timer
__hrtimer_run_queues
hrtimer_interrupt
smp_apic_timer_interrupt
apic_timer_interrupt
For people working in console mode, the screen will first show the panic
call stack, but immediately overridden by these noisy extra messages,
which makes debugging much more difficult, as the original context gets
lost on screen.
Also these noisy messages will confuse some users, as I have seen many bug
reporters posted the noisy message into bugzilla, instead of the real
panic call stack and context.
Adding a flag "suppress_printk" which gets set in panic() to avoid those
noisy messages, without changing current kernel behavior that both panic
blinking and sysrq magic key can work as is, suggested by Petr Mladek.
To verify this, make sure kernel is not configured to reboot on panic and
in console
# echo c > /proc/sysrq-trigger
to see if console only prints out the panic call stack.
Link: http://lkml.kernel.org/r/1551430186-24169-1-git-send-email-feng.tang@intel.com
Signed-off-by: Feng Tang <feng.tang@intel.com>
Suggested-by: Petr Mladek <pmladek@suse.com>
Reviewed-by: Petr Mladek <pmladek@suse.com>
Acked-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
Acked-by: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Kees Cook <keescook@chromium.org>
Cc: Borislav Petkov <bp@suse.de>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: Jiri Slaby <jslaby@suse.com>
Cc: Sasha Levin <sashal@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2019-05-14 15:45:34 -07:00
|
|
|
orig_suppress_printk = suppress_printk;
|
|
|
|
suppress_printk = 0;
|
|
|
|
|
2014-06-06 14:38:14 -07:00
|
|
|
rcu_sysrq_start();
|
2014-06-06 14:38:13 -07:00
|
|
|
rcu_read_lock();
|
2009-01-15 13:50:52 -08:00
|
|
|
/*
|
|
|
|
* Raise the apparent loglevel to maximum so that the sysrq header
|
|
|
|
* is shown to provide the user with positive feedback. We do not
|
|
|
|
* simply emit this at KERN_EMERG as that would change message
|
|
|
|
* routing in the consumers of /proc/kmsg.
|
|
|
|
*/
|
2005-04-16 15:20:36 -07:00
|
|
|
orig_log_level = console_loglevel;
|
2014-06-04 16:11:46 -07:00
|
|
|
console_loglevel = CONSOLE_LOGLEVEL_DEFAULT;
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2021-04-07 15:06:42 +08:00
|
|
|
op_p = __sysrq_get_key_op(key);
|
|
|
|
if (op_p) {
|
2006-03-25 03:07:08 -08:00
|
|
|
/*
|
|
|
|
* Should we check for enabled operations (/proc/sysrq-trigger
|
|
|
|
* should not) and is the invoked operation enabled?
|
|
|
|
*/
|
2006-12-13 00:34:36 -08:00
|
|
|
if (!check_mask || sysrq_on_mask(op_p->enable_mask)) {
|
2019-01-11 17:20:37 +01:00
|
|
|
pr_info("%s\n", op_p->action_msg);
|
2005-04-16 15:20:36 -07:00
|
|
|
console_loglevel = orig_log_level;
|
2010-08-17 21:15:46 -07:00
|
|
|
op_p->handler(key);
|
2006-03-25 03:07:08 -08:00
|
|
|
} else {
|
2019-01-11 17:20:37 +01:00
|
|
|
pr_info("This sysrq operation is disabled.\n");
|
2019-01-11 13:45:15 +01:00
|
|
|
console_loglevel = orig_log_level;
|
2006-03-25 03:07:08 -08:00
|
|
|
}
|
2005-04-16 15:20:36 -07:00
|
|
|
} else {
|
2019-01-11 17:20:37 +01:00
|
|
|
pr_info("HELP : ");
|
2005-04-16 15:20:36 -07:00
|
|
|
/* Only print the help msg once per handler */
|
2006-03-25 03:07:08 -08:00
|
|
|
for (i = 0; i < ARRAY_SIZE(sysrq_key_table); i++) {
|
|
|
|
if (sysrq_key_table[i]) {
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; sysrq_key_table[i] !=
|
|
|
|
sysrq_key_table[j]; j++)
|
|
|
|
;
|
|
|
|
if (j != i)
|
|
|
|
continue;
|
2015-02-11 15:26:21 -08:00
|
|
|
pr_cont("%s ", sysrq_key_table[i]->help_msg);
|
2006-03-25 03:07:08 -08:00
|
|
|
}
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2015-02-11 15:26:21 -08:00
|
|
|
pr_cont("\n");
|
2005-04-16 15:20:36 -07:00
|
|
|
console_loglevel = orig_log_level;
|
|
|
|
}
|
2014-06-06 14:38:13 -07:00
|
|
|
rcu_read_unlock();
|
2014-06-06 14:38:14 -07:00
|
|
|
rcu_sysrq_end();
|
panic: avoid the extra noise dmesg
When kernel panic happens, it will first print the panic call stack,
then the ending msg like:
[ 35.743249] ---[ end Kernel panic - not syncing: Fatal exception
[ 35.749975] ------------[ cut here ]------------
The above message are very useful for debugging.
But if system is configured to not reboot on panic, say the
"panic_timeout" parameter equals 0, it will likely print out many noisy
message like WARN() call stack for each and every CPU except the panic
one, messages like below:
WARNING: CPU: 1 PID: 280 at kernel/sched/core.c:1198 set_task_cpu+0x183/0x190
Call Trace:
<IRQ>
try_to_wake_up
default_wake_function
autoremove_wake_function
__wake_up_common
__wake_up_common_lock
__wake_up
wake_up_klogd_work_func
irq_work_run_list
irq_work_tick
update_process_times
tick_sched_timer
__hrtimer_run_queues
hrtimer_interrupt
smp_apic_timer_interrupt
apic_timer_interrupt
For people working in console mode, the screen will first show the panic
call stack, but immediately overridden by these noisy extra messages,
which makes debugging much more difficult, as the original context gets
lost on screen.
Also these noisy messages will confuse some users, as I have seen many bug
reporters posted the noisy message into bugzilla, instead of the real
panic call stack and context.
Adding a flag "suppress_printk" which gets set in panic() to avoid those
noisy messages, without changing current kernel behavior that both panic
blinking and sysrq magic key can work as is, suggested by Petr Mladek.
To verify this, make sure kernel is not configured to reboot on panic and
in console
# echo c > /proc/sysrq-trigger
to see if console only prints out the panic call stack.
Link: http://lkml.kernel.org/r/1551430186-24169-1-git-send-email-feng.tang@intel.com
Signed-off-by: Feng Tang <feng.tang@intel.com>
Suggested-by: Petr Mladek <pmladek@suse.com>
Reviewed-by: Petr Mladek <pmladek@suse.com>
Acked-by: Steven Rostedt (VMware) <rostedt@goodmis.org>
Acked-by: Sergey Senozhatsky <sergey.senozhatsky@gmail.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Kees Cook <keescook@chromium.org>
Cc: Borislav Petkov <bp@suse.de>
Cc: Andi Kleen <ak@linux.intel.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
Cc: Jiri Slaby <jslaby@suse.com>
Cc: Sasha Levin <sashal@kernel.org>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2019-05-14 15:45:34 -07:00
|
|
|
|
|
|
|
suppress_printk = orig_suppress_printk;
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
|
|
|
|
2010-08-17 21:15:47 -07:00
|
|
|
void handle_sysrq(int key)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
2006-12-13 00:34:36 -08:00
|
|
|
if (sysrq_on())
|
2010-08-17 21:15:47 -07:00
|
|
|
__handle_sysrq(key, true);
|
2005-04-16 15:20:36 -07:00
|
|
|
}
|
2006-03-25 03:07:08 -08:00
|
|
|
EXPORT_SYMBOL(handle_sysrq);
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
#ifdef CONFIG_INPUT
|
2015-05-26 14:45:29 -07:00
|
|
|
static int sysrq_reset_downtime_ms;
|
2010-03-21 22:31:26 -07:00
|
|
|
|
|
|
|
/* Simple translation table for the SysRq keys */
|
2010-11-15 00:23:42 -08:00
|
|
|
static const unsigned char sysrq_xlate[KEY_CNT] =
|
2021-04-07 15:06:42 +08:00
|
|
|
"\000\0331234567890-=\177\t" /* 0x00 - 0x0f */
|
|
|
|
"qwertyuiop[]\r\000as" /* 0x10 - 0x1f */
|
|
|
|
"dfghjkl;'`\000\\zxcv" /* 0x20 - 0x2f */
|
|
|
|
"bnm,./\000*\000 \000\201\202\203\204\205" /* 0x30 - 0x3f */
|
|
|
|
"\206\207\210\211\212\000\000789-456+1" /* 0x40 - 0x4f */
|
|
|
|
"230\177\000\000\213\214\000\000\000\000\000\000\000\000\000\000" /* 0x50 - 0x5f */
|
|
|
|
"\r\000/"; /* 0x60 - 0x6f */
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
struct sysrq_state {
|
|
|
|
struct input_handle handle;
|
|
|
|
struct work_struct reinject_work;
|
|
|
|
unsigned long key_down[BITS_TO_LONGS(KEY_CNT)];
|
|
|
|
unsigned int alt;
|
|
|
|
unsigned int alt_use;
|
2020-08-18 13:28:24 +02:00
|
|
|
unsigned int shift;
|
|
|
|
unsigned int shift_use;
|
2010-11-15 00:23:42 -08:00
|
|
|
bool active;
|
|
|
|
bool need_reinject;
|
2011-02-02 22:59:54 -08:00
|
|
|
bool reinjecting;
|
2013-01-06 23:23:33 -08:00
|
|
|
|
|
|
|
/* reset sequence handling */
|
|
|
|
bool reset_canceled;
|
2013-06-05 22:51:46 -07:00
|
|
|
bool reset_requested;
|
2013-01-06 23:23:33 -08:00
|
|
|
unsigned long reset_keybit[BITS_TO_LONGS(KEY_CNT)];
|
|
|
|
int reset_seq_len;
|
|
|
|
int reset_seq_cnt;
|
|
|
|
int reset_seq_version;
|
2013-04-01 22:14:19 -07:00
|
|
|
struct timer_list keyreset_timer;
|
2010-11-15 00:23:42 -08:00
|
|
|
};
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
#define SYSRQ_KEY_RESET_MAX 20 /* Should be plenty */
|
|
|
|
static unsigned short sysrq_reset_seq[SYSRQ_KEY_RESET_MAX];
|
|
|
|
static unsigned int sysrq_reset_seq_len;
|
|
|
|
static unsigned int sysrq_reset_seq_version = 1;
|
|
|
|
|
|
|
|
static void sysrq_parse_reset_sequence(struct sysrq_state *state)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
unsigned short key;
|
|
|
|
|
|
|
|
state->reset_seq_cnt = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < sysrq_reset_seq_len; i++) {
|
|
|
|
key = sysrq_reset_seq[i];
|
|
|
|
|
|
|
|
if (key == KEY_RESERVED || key > KEY_MAX)
|
|
|
|
break;
|
|
|
|
|
|
|
|
__set_bit(key, state->reset_keybit);
|
|
|
|
state->reset_seq_len++;
|
|
|
|
|
|
|
|
if (test_bit(key, state->key_down))
|
|
|
|
state->reset_seq_cnt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Disable reset until old keys are not released */
|
|
|
|
state->reset_canceled = state->reset_seq_cnt != 0;
|
|
|
|
|
|
|
|
state->reset_seq_version = sysrq_reset_seq_version;
|
|
|
|
}
|
|
|
|
|
2017-10-16 16:28:51 -07:00
|
|
|
static void sysrq_do_reset(struct timer_list *t)
|
2013-04-01 22:14:19 -07:00
|
|
|
{
|
2017-10-16 16:28:51 -07:00
|
|
|
struct sysrq_state *state = from_timer(state, t, keyreset_timer);
|
2013-06-05 22:51:46 -07:00
|
|
|
|
|
|
|
state->reset_requested = true;
|
|
|
|
|
2018-11-27 09:57:23 +13:00
|
|
|
orderly_reboot();
|
2013-04-01 22:14:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sysrq_handle_reset_request(struct sysrq_state *state)
|
|
|
|
{
|
2013-06-05 22:51:46 -07:00
|
|
|
if (state->reset_requested)
|
|
|
|
__handle_sysrq(sysrq_xlate[KEY_B], false);
|
|
|
|
|
2013-04-01 22:14:19 -07:00
|
|
|
if (sysrq_reset_downtime_ms)
|
|
|
|
mod_timer(&state->keyreset_timer,
|
|
|
|
jiffies + msecs_to_jiffies(sysrq_reset_downtime_ms));
|
|
|
|
else
|
2017-10-16 16:28:51 -07:00
|
|
|
sysrq_do_reset(&state->keyreset_timer);
|
2013-04-01 22:14:19 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void sysrq_detect_reset_sequence(struct sysrq_state *state,
|
2013-01-06 23:23:33 -08:00
|
|
|
unsigned int code, int value)
|
|
|
|
{
|
|
|
|
if (!test_bit(code, state->reset_keybit)) {
|
|
|
|
/*
|
|
|
|
* Pressing any key _not_ in reset sequence cancels
|
2013-04-01 22:14:19 -07:00
|
|
|
* the reset sequence. Also cancelling the timer in
|
|
|
|
* case additional keys were pressed after a reset
|
|
|
|
* has been requested.
|
2013-01-06 23:23:33 -08:00
|
|
|
*/
|
2013-04-01 22:14:19 -07:00
|
|
|
if (value && state->reset_seq_cnt) {
|
2013-01-06 23:23:33 -08:00
|
|
|
state->reset_canceled = true;
|
2013-04-01 22:14:19 -07:00
|
|
|
del_timer(&state->keyreset_timer);
|
|
|
|
}
|
2013-01-06 23:23:33 -08:00
|
|
|
} else if (value == 0) {
|
2013-04-01 22:14:19 -07:00
|
|
|
/*
|
|
|
|
* Key release - all keys in the reset sequence need
|
|
|
|
* to be pressed and held for the reset timeout
|
|
|
|
* to hold.
|
|
|
|
*/
|
|
|
|
del_timer(&state->keyreset_timer);
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
if (--state->reset_seq_cnt == 0)
|
|
|
|
state->reset_canceled = false;
|
|
|
|
} else if (value == 1) {
|
|
|
|
/* key press, not autorepeat */
|
|
|
|
if (++state->reset_seq_cnt == state->reset_seq_len &&
|
|
|
|
!state->reset_canceled) {
|
2013-04-01 22:14:19 -07:00
|
|
|
sysrq_handle_reset_request(state);
|
2013-01-06 23:23:33 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-03 17:22:08 -07:00
|
|
|
#ifdef CONFIG_OF
|
|
|
|
static void sysrq_of_get_keyreset_config(void)
|
|
|
|
{
|
|
|
|
u32 key;
|
|
|
|
struct device_node *np;
|
|
|
|
struct property *prop;
|
|
|
|
const __be32 *p;
|
|
|
|
|
|
|
|
np = of_find_node_by_path("/chosen/linux,sysrq-reset-seq");
|
|
|
|
if (!np) {
|
|
|
|
pr_debug("No sysrq node found");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reset in case a __weak definition was present */
|
|
|
|
sysrq_reset_seq_len = 0;
|
|
|
|
|
|
|
|
of_property_for_each_u32(np, "keyset", prop, p, key) {
|
|
|
|
if (key == KEY_RESERVED || key > KEY_MAX ||
|
|
|
|
sysrq_reset_seq_len == SYSRQ_KEY_RESET_MAX)
|
|
|
|
break;
|
|
|
|
|
|
|
|
sysrq_reset_seq[sysrq_reset_seq_len++] = (unsigned short)key;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get reset timeout if any. */
|
|
|
|
of_property_read_u32(np, "timeout-ms", &sysrq_reset_downtime_ms);
|
2018-11-21 08:12:14 -05:00
|
|
|
|
|
|
|
of_node_put(np);
|
2013-08-03 17:22:08 -07:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
static void sysrq_of_get_keyreset_config(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
static void sysrq_reinject_alt_sysrq(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct sysrq_state *sysrq =
|
|
|
|
container_of(work, struct sysrq_state, reinject_work);
|
|
|
|
struct input_handle *handle = &sysrq->handle;
|
|
|
|
unsigned int alt_code = sysrq->alt_use;
|
|
|
|
|
|
|
|
if (sysrq->need_reinject) {
|
2011-02-02 22:59:54 -08:00
|
|
|
/* we do not want the assignment to be reordered */
|
|
|
|
sysrq->reinjecting = true;
|
|
|
|
mb();
|
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
/* Simulate press and release of Alt + SysRq */
|
|
|
|
input_inject_event(handle, EV_KEY, alt_code, 1);
|
|
|
|
input_inject_event(handle, EV_KEY, KEY_SYSRQ, 1);
|
|
|
|
input_inject_event(handle, EV_SYN, SYN_REPORT, 1);
|
|
|
|
|
|
|
|
input_inject_event(handle, EV_KEY, KEY_SYSRQ, 0);
|
|
|
|
input_inject_event(handle, EV_KEY, alt_code, 0);
|
|
|
|
input_inject_event(handle, EV_SYN, SYN_REPORT, 1);
|
2011-02-02 22:59:54 -08:00
|
|
|
|
|
|
|
mb();
|
|
|
|
sysrq->reinjecting = false;
|
2010-11-15 00:23:42 -08:00
|
|
|
}
|
|
|
|
}
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
static bool sysrq_handle_keypress(struct sysrq_state *sysrq,
|
|
|
|
unsigned int code, int value)
|
2010-03-21 22:31:26 -07:00
|
|
|
{
|
2010-11-15 00:23:42 -08:00
|
|
|
bool was_active = sysrq->active;
|
2010-09-29 18:04:21 -07:00
|
|
|
bool suppress;
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
switch (code) {
|
2011-02-02 22:59:54 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
case KEY_LEFTALT:
|
|
|
|
case KEY_RIGHTALT:
|
|
|
|
if (!value) {
|
|
|
|
/* One of ALTs is being released */
|
|
|
|
if (sysrq->active && code == sysrq->alt_use)
|
|
|
|
sysrq->active = false;
|
2010-09-29 18:04:21 -07:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
sysrq->alt = KEY_RESERVED;
|
|
|
|
|
|
|
|
} else if (value != 2) {
|
|
|
|
sysrq->alt = code;
|
|
|
|
sysrq->need_reinject = false;
|
|
|
|
}
|
2010-11-15 00:23:42 -08:00
|
|
|
break;
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2020-08-18 13:28:24 +02:00
|
|
|
case KEY_LEFTSHIFT:
|
|
|
|
case KEY_RIGHTSHIFT:
|
|
|
|
if (!value)
|
|
|
|
sysrq->shift = KEY_RESERVED;
|
|
|
|
else if (value != 2)
|
|
|
|
sysrq->shift = code;
|
2021-09-25 23:42:48 -05:00
|
|
|
if (sysrq->active)
|
|
|
|
sysrq->shift_use = sysrq->shift;
|
2020-08-18 13:28:24 +02:00
|
|
|
break;
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
case KEY_SYSRQ:
|
|
|
|
if (value == 1 && sysrq->alt != KEY_RESERVED) {
|
|
|
|
sysrq->active = true;
|
|
|
|
sysrq->alt_use = sysrq->alt;
|
2020-08-18 13:28:24 +02:00
|
|
|
/* either RESERVED (for released) or actual code */
|
|
|
|
sysrq->shift_use = sysrq->shift;
|
2013-01-06 23:23:33 -08:00
|
|
|
/*
|
|
|
|
* If nothing else will be pressed we'll need
|
|
|
|
* to re-inject Alt-SysRq keysroke.
|
|
|
|
*/
|
|
|
|
sysrq->need_reinject = true;
|
|
|
|
}
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
/*
|
|
|
|
* Pretend that sysrq was never pressed at all. This
|
|
|
|
* is needed to properly handle KGDB which will try
|
|
|
|
* to release all keys after exiting debugger. If we
|
|
|
|
* do not clear key bit it KGDB will end up sending
|
|
|
|
* release events for Alt and SysRq, potentially
|
|
|
|
* triggering print screen function.
|
|
|
|
*/
|
|
|
|
if (sysrq->active)
|
|
|
|
clear_bit(KEY_SYSRQ, sysrq->handle.dev->key);
|
2010-06-09 08:13:06 -07:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
break;
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
default:
|
|
|
|
if (sysrq->active && value && value != 2) {
|
2020-08-18 13:28:24 +02:00
|
|
|
unsigned char c = sysrq_xlate[code];
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
sysrq->need_reinject = false;
|
2020-08-18 13:28:24 +02:00
|
|
|
if (sysrq->shift_use != KEY_RESERVED)
|
|
|
|
c = toupper(c);
|
|
|
|
__handle_sysrq(c, true);
|
2013-01-06 23:23:33 -08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
suppress = sysrq->active;
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
if (!sysrq->active) {
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
/*
|
|
|
|
* See if reset sequence has changed since the last time.
|
|
|
|
*/
|
|
|
|
if (sysrq->reset_seq_version != sysrq_reset_seq_version)
|
|
|
|
sysrq_parse_reset_sequence(sysrq);
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
/*
|
|
|
|
* If we are not suppressing key presses keep track of
|
|
|
|
* keyboard state so we can release keys that have been
|
|
|
|
* pressed before entering SysRq mode.
|
|
|
|
*/
|
|
|
|
if (value)
|
|
|
|
set_bit(code, sysrq->key_down);
|
|
|
|
else
|
|
|
|
clear_bit(code, sysrq->key_down);
|
|
|
|
|
|
|
|
if (was_active)
|
|
|
|
schedule_work(&sysrq->reinject_work);
|
|
|
|
|
2013-04-01 22:14:19 -07:00
|
|
|
/* Check for reset sequence */
|
|
|
|
sysrq_detect_reset_sequence(sysrq, code, value);
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
} else if (value == 0 && test_and_clear_bit(code, sysrq->key_down)) {
|
|
|
|
/*
|
|
|
|
* Pass on release events for keys that was pressed before
|
|
|
|
* entering SysRq mode.
|
|
|
|
*/
|
|
|
|
suppress = false;
|
|
|
|
}
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
return suppress;
|
|
|
|
}
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
static bool sysrq_filter(struct input_handle *handle,
|
|
|
|
unsigned int type, unsigned int code, int value)
|
|
|
|
{
|
|
|
|
struct sysrq_state *sysrq = handle->private;
|
|
|
|
bool suppress;
|
2010-11-15 00:23:42 -08:00
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
/*
|
|
|
|
* Do not filter anything if we are in the process of re-injecting
|
|
|
|
* Alt+SysRq combination.
|
|
|
|
*/
|
|
|
|
if (sysrq->reinjecting)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
|
|
|
|
case EV_SYN:
|
|
|
|
suppress = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EV_KEY:
|
|
|
|
suppress = sysrq_handle_keypress(sysrq, code, value);
|
2010-03-21 22:31:26 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2010-11-15 00:23:42 -08:00
|
|
|
suppress = sysrq->active;
|
2010-03-21 22:31:26 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-09-29 18:04:21 -07:00
|
|
|
return suppress;
|
2010-03-21 22:31:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int sysrq_connect(struct input_handler *handler,
|
|
|
|
struct input_dev *dev,
|
|
|
|
const struct input_device_id *id)
|
|
|
|
{
|
2010-11-15 00:23:42 -08:00
|
|
|
struct sysrq_state *sysrq;
|
2010-03-21 22:31:26 -07:00
|
|
|
int error;
|
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
sysrq = kzalloc(sizeof(struct sysrq_state), GFP_KERNEL);
|
|
|
|
if (!sysrq)
|
2010-03-21 22:31:26 -07:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
INIT_WORK(&sysrq->reinject_work, sysrq_reinject_alt_sysrq);
|
|
|
|
|
|
|
|
sysrq->handle.dev = dev;
|
|
|
|
sysrq->handle.handler = handler;
|
|
|
|
sysrq->handle.name = "sysrq";
|
|
|
|
sysrq->handle.private = sysrq;
|
2017-10-16 16:28:51 -07:00
|
|
|
timer_setup(&sysrq->keyreset_timer, sysrq_do_reset, 0);
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
error = input_register_handle(&sysrq->handle);
|
2010-03-21 22:31:26 -07:00
|
|
|
if (error) {
|
|
|
|
pr_err("Failed to register input sysrq handler, error %d\n",
|
|
|
|
error);
|
|
|
|
goto err_free;
|
|
|
|
}
|
|
|
|
|
2010-11-15 00:23:42 -08:00
|
|
|
error = input_open_device(&sysrq->handle);
|
2010-03-21 22:31:26 -07:00
|
|
|
if (error) {
|
|
|
|
pr_err("Failed to open input device, error %d\n", error);
|
|
|
|
goto err_unregister;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
err_unregister:
|
2010-11-15 00:23:42 -08:00
|
|
|
input_unregister_handle(&sysrq->handle);
|
2010-03-21 22:31:26 -07:00
|
|
|
err_free:
|
2010-11-15 00:23:42 -08:00
|
|
|
kfree(sysrq);
|
2010-03-21 22:31:26 -07:00
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sysrq_disconnect(struct input_handle *handle)
|
|
|
|
{
|
2010-11-15 00:23:42 -08:00
|
|
|
struct sysrq_state *sysrq = handle->private;
|
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
input_close_device(handle);
|
2010-11-15 00:23:42 -08:00
|
|
|
cancel_work_sync(&sysrq->reinject_work);
|
2013-04-01 22:14:19 -07:00
|
|
|
del_timer_sync(&sysrq->keyreset_timer);
|
2010-03-21 22:31:26 -07:00
|
|
|
input_unregister_handle(handle);
|
2010-11-15 00:23:42 -08:00
|
|
|
kfree(sysrq);
|
2010-03-21 22:31:26 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-09-29 18:04:21 -07:00
|
|
|
* We are matching on KEY_LEFTALT instead of KEY_SYSRQ because not all
|
|
|
|
* keyboards have SysRq key predefined and so user may add it to keymap
|
2010-03-21 22:31:26 -07:00
|
|
|
* later, but we expect all such keyboards to have left alt.
|
|
|
|
*/
|
|
|
|
static const struct input_device_id sysrq_ids[] = {
|
|
|
|
{
|
|
|
|
.flags = INPUT_DEVICE_ID_MATCH_EVBIT |
|
|
|
|
INPUT_DEVICE_ID_MATCH_KEYBIT,
|
2017-01-06 02:14:16 +09:00
|
|
|
.evbit = { [BIT_WORD(EV_KEY)] = BIT_MASK(EV_KEY) },
|
|
|
|
.keybit = { [BIT_WORD(KEY_LEFTALT)] = BIT_MASK(KEY_LEFTALT) },
|
2010-03-21 22:31:26 -07:00
|
|
|
},
|
|
|
|
{ },
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct input_handler sysrq_handler = {
|
|
|
|
.filter = sysrq_filter,
|
|
|
|
.connect = sysrq_connect,
|
|
|
|
.disconnect = sysrq_disconnect,
|
|
|
|
.name = "sysrq",
|
|
|
|
.id_table = sysrq_ids,
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline void sysrq_register_handler(void)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
2013-08-03 17:22:08 -07:00
|
|
|
sysrq_of_get_keyreset_config();
|
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
error = input_register_handler(&sysrq_handler);
|
|
|
|
if (error)
|
|
|
|
pr_err("Failed to register input handler, error %d", error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sysrq_unregister_handler(void)
|
|
|
|
{
|
2019-12-13 00:06:00 +00:00
|
|
|
input_unregister_handler(&sysrq_handler);
|
2010-03-21 22:31:26 -07:00
|
|
|
}
|
|
|
|
|
2013-01-06 23:23:33 -08:00
|
|
|
static int sysrq_reset_seq_param_set(const char *buffer,
|
|
|
|
const struct kernel_param *kp)
|
|
|
|
{
|
|
|
|
unsigned long val;
|
|
|
|
int error;
|
|
|
|
|
2013-06-01 16:30:06 +09:00
|
|
|
error = kstrtoul(buffer, 0, &val);
|
2013-01-06 23:23:33 -08:00
|
|
|
if (error < 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
if (val > KEY_MAX)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*((unsigned short *)kp->arg) = val;
|
|
|
|
sysrq_reset_seq_version++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-05-27 11:09:38 +09:30
|
|
|
static const struct kernel_param_ops param_ops_sysrq_reset_seq = {
|
2013-01-06 23:23:33 -08:00
|
|
|
.get = param_get_ushort,
|
|
|
|
.set = sysrq_reset_seq_param_set,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define param_check_sysrq_reset_seq(name, p) \
|
|
|
|
__param_check(name, p, unsigned short)
|
|
|
|
|
2015-08-19 17:48:06 -04:00
|
|
|
/*
|
|
|
|
* not really modular, but the easiest way to keep compat with existing
|
|
|
|
* bootargs behaviour is to continue using module_param here.
|
|
|
|
*/
|
2013-01-06 23:23:33 -08:00
|
|
|
module_param_array_named(reset_seq, sysrq_reset_seq, sysrq_reset_seq,
|
|
|
|
&sysrq_reset_seq_len, 0644);
|
|
|
|
|
2013-04-01 22:14:19 -07:00
|
|
|
module_param_named(sysrq_downtime_ms, sysrq_reset_downtime_ms, int, 0644);
|
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
#else
|
|
|
|
|
|
|
|
static inline void sysrq_register_handler(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sysrq_unregister_handler(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_INPUT */
|
|
|
|
|
|
|
|
int sysrq_toggle_support(int enable_mask)
|
|
|
|
{
|
|
|
|
bool was_enabled = sysrq_on();
|
|
|
|
|
|
|
|
sysrq_enabled = enable_mask;
|
|
|
|
|
|
|
|
if (was_enabled != sysrq_on()) {
|
|
|
|
if (sysrq_on())
|
|
|
|
sysrq_register_handler();
|
|
|
|
else
|
|
|
|
sysrq_unregister_handler();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2020-04-20 18:23:17 +01:00
|
|
|
EXPORT_SYMBOL_GPL(sysrq_toggle_support);
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2020-05-13 22:43:42 +01:00
|
|
|
static int __sysrq_swap_key_ops(int key, const struct sysrq_key_op *insert_op_p,
|
2021-04-07 15:06:42 +08:00
|
|
|
const struct sysrq_key_op *remove_op_p)
|
2006-03-25 03:07:08 -08:00
|
|
|
{
|
2005-04-16 15:20:36 -07:00
|
|
|
int retval;
|
|
|
|
|
2014-06-06 14:38:13 -07:00
|
|
|
spin_lock(&sysrq_key_table_lock);
|
2005-04-16 15:20:36 -07:00
|
|
|
if (__sysrq_get_key_op(key) == remove_op_p) {
|
|
|
|
__sysrq_put_key_op(key, insert_op_p);
|
|
|
|
retval = 0;
|
|
|
|
} else {
|
|
|
|
retval = -1;
|
|
|
|
}
|
2014-06-06 14:38:13 -07:00
|
|
|
spin_unlock(&sysrq_key_table_lock);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* A concurrent __handle_sysrq either got the old op or the new op.
|
|
|
|
* Wait for it to go away before returning, so the code for an old
|
|
|
|
* op is not freed (eg. on module unload) while it is in use.
|
|
|
|
*/
|
|
|
|
synchronize_rcu();
|
|
|
|
|
2005-04-16 15:20:36 -07:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2020-05-13 22:43:42 +01:00
|
|
|
int register_sysrq_key(int key, const struct sysrq_key_op *op_p)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
return __sysrq_swap_key_ops(key, op_p, NULL);
|
|
|
|
}
|
2006-03-25 03:07:08 -08:00
|
|
|
EXPORT_SYMBOL(register_sysrq_key);
|
2005-04-16 15:20:36 -07:00
|
|
|
|
2020-05-13 22:43:42 +01:00
|
|
|
int unregister_sysrq_key(int key, const struct sysrq_key_op *op_p)
|
2005-04-16 15:20:36 -07:00
|
|
|
{
|
|
|
|
return __sysrq_swap_key_ops(key, NULL, op_p);
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL(unregister_sysrq_key);
|
2008-10-15 22:04:23 -07:00
|
|
|
|
|
|
|
#ifdef CONFIG_PROC_FS
|
|
|
|
/*
|
|
|
|
* writing 'C' to /proc/sysrq-trigger is like sysrq-C
|
|
|
|
*/
|
|
|
|
static ssize_t write_sysrq_trigger(struct file *file, const char __user *buf,
|
|
|
|
size_t count, loff_t *ppos)
|
|
|
|
{
|
|
|
|
if (count) {
|
|
|
|
char c;
|
|
|
|
|
|
|
|
if (get_user(c, buf))
|
|
|
|
return -EFAULT;
|
2010-08-17 21:15:47 -07:00
|
|
|
__handle_sysrq(c, false);
|
2008-10-15 22:04:23 -07:00
|
|
|
}
|
2010-03-21 22:31:26 -07:00
|
|
|
|
2008-10-15 22:04:23 -07:00
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2020-02-03 17:37:17 -08:00
|
|
|
static const struct proc_ops sysrq_trigger_proc_ops = {
|
|
|
|
.proc_write = write_sysrq_trigger,
|
|
|
|
.proc_lseek = noop_llseek,
|
2008-10-15 22:04:23 -07:00
|
|
|
};
|
|
|
|
|
2010-03-21 22:31:26 -07:00
|
|
|
static void sysrq_init_procfs(void)
|
|
|
|
{
|
|
|
|
if (!proc_create("sysrq-trigger", S_IWUSR, NULL,
|
2020-02-03 17:37:17 -08:00
|
|
|
&sysrq_trigger_proc_ops))
|
2010-03-21 22:31:26 -07:00
|
|
|
pr_err("Failed to register proc interface\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
static inline void sysrq_init_procfs(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_PROC_FS */
|
|
|
|
|
2008-10-15 22:04:23 -07:00
|
|
|
static int __init sysrq_init(void)
|
|
|
|
{
|
2010-03-21 22:31:26 -07:00
|
|
|
sysrq_init_procfs();
|
|
|
|
|
|
|
|
if (sysrq_on())
|
|
|
|
sysrq_register_handler();
|
|
|
|
|
2008-10-15 22:04:23 -07:00
|
|
|
return 0;
|
|
|
|
}
|
2015-08-19 17:48:06 -04:00
|
|
|
device_initcall(sysrq_init);
|