mirror of
https://github.com/FEX-Emu/linux.git
synced 2024-12-17 14:30:00 +00:00
6e274d1443
Makes kexec_crashdump() take a pt_regs * as an argument. This allows to get exact register state at the point of the crash. If we come from direct panic assertion NULL will be passed and the current registers saved before crashdump. This hooks into two places: die(): check the conditions under which we will panic when calling do_exit and go there directly with the pt_regs that caused the fatal fault. die_nmi(): If we receive an NMI lockup while in the kernel use the pt_regs and go directly to crash_kexec(). We're probably nested up badly at this point so this might be the only chance to escape with proper information. Signed-off-by: Alexander Nyberg <alexn@telia.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
453 lines
11 KiB
C
453 lines
11 KiB
C
/* -*- linux-c -*-
|
|
*
|
|
* $Id: sysrq.c,v 1.15 1998/08/23 14:56:41 mj Exp $
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
#include <linux/config.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/mm.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/tty.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>
|
|
#include <linux/quotaops.h>
|
|
#include <linux/smp_lock.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/suspend.h>
|
|
#include <linux/writeback.h>
|
|
#include <linux/buffer_head.h> /* for fsync_bdev() */
|
|
#include <linux/swap.h>
|
|
#include <linux/spinlock.h>
|
|
#include <linux/vt_kern.h>
|
|
#include <linux/workqueue.h>
|
|
#include <linux/kexec.h>
|
|
|
|
#include <asm/ptrace.h>
|
|
|
|
/* Whether we react on sysrq keys or just ignore them */
|
|
int sysrq_enabled = 1;
|
|
|
|
/* Loglevel sysrq handler */
|
|
static void sysrq_handle_loglevel(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
int i;
|
|
i = key - '0';
|
|
console_loglevel = 7;
|
|
printk("Loglevel set to %d\n", i);
|
|
console_loglevel = i;
|
|
}
|
|
static struct sysrq_key_op sysrq_loglevel_op = {
|
|
.handler = sysrq_handle_loglevel,
|
|
.help_msg = "loglevel0-8",
|
|
.action_msg = "Changing Loglevel",
|
|
.enable_mask = SYSRQ_ENABLE_LOG,
|
|
};
|
|
|
|
|
|
/* SAK sysrq handler */
|
|
#ifdef CONFIG_VT
|
|
static void sysrq_handle_SAK(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
if (tty)
|
|
do_SAK(tty);
|
|
reset_vc(vc_cons[fg_console].d);
|
|
}
|
|
static struct sysrq_key_op sysrq_SAK_op = {
|
|
.handler = sysrq_handle_SAK,
|
|
.help_msg = "saK",
|
|
.action_msg = "SAK",
|
|
.enable_mask = SYSRQ_ENABLE_KEYBOARD,
|
|
};
|
|
#endif
|
|
|
|
#ifdef CONFIG_VT
|
|
/* unraw sysrq handler */
|
|
static void sysrq_handle_unraw(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
struct kbd_struct *kbd = &kbd_table[fg_console];
|
|
|
|
if (kbd)
|
|
kbd->kbdmode = VC_XLATE;
|
|
}
|
|
static struct sysrq_key_op sysrq_unraw_op = {
|
|
.handler = sysrq_handle_unraw,
|
|
.help_msg = "unRaw",
|
|
.action_msg = "Keyboard mode set to XLATE",
|
|
.enable_mask = SYSRQ_ENABLE_KEYBOARD,
|
|
};
|
|
#endif /* CONFIG_VT */
|
|
|
|
#ifdef CONFIG_KEXEC
|
|
/* crashdump sysrq handler */
|
|
static void sysrq_handle_crashdump(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
crash_kexec(pt_regs);
|
|
}
|
|
static struct sysrq_key_op sysrq_crashdump_op = {
|
|
.handler = sysrq_handle_crashdump,
|
|
.help_msg = "Crashdump",
|
|
.action_msg = "Trigger a crashdump",
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
};
|
|
#endif
|
|
|
|
/* reboot sysrq handler */
|
|
static void sysrq_handle_reboot(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
local_irq_enable();
|
|
machine_restart(NULL);
|
|
}
|
|
|
|
static struct sysrq_key_op sysrq_reboot_op = {
|
|
.handler = sysrq_handle_reboot,
|
|
.help_msg = "reBoot",
|
|
.action_msg = "Resetting",
|
|
.enable_mask = SYSRQ_ENABLE_BOOT,
|
|
};
|
|
|
|
static void sysrq_handle_sync(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
emergency_sync();
|
|
}
|
|
|
|
static struct sysrq_key_op sysrq_sync_op = {
|
|
.handler = sysrq_handle_sync,
|
|
.help_msg = "Sync",
|
|
.action_msg = "Emergency Sync",
|
|
.enable_mask = SYSRQ_ENABLE_SYNC,
|
|
};
|
|
|
|
static void sysrq_handle_mountro(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
emergency_remount();
|
|
}
|
|
|
|
static struct sysrq_key_op sysrq_mountro_op = {
|
|
.handler = sysrq_handle_mountro,
|
|
.help_msg = "Unmount",
|
|
.action_msg = "Emergency Remount R/O",
|
|
.enable_mask = SYSRQ_ENABLE_REMOUNT,
|
|
};
|
|
|
|
/* END SYNC SYSRQ HANDLERS BLOCK */
|
|
|
|
|
|
/* SHOW SYSRQ HANDLERS BLOCK */
|
|
|
|
static void sysrq_handle_showregs(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
if (pt_regs)
|
|
show_regs(pt_regs);
|
|
}
|
|
static struct sysrq_key_op sysrq_showregs_op = {
|
|
.handler = sysrq_handle_showregs,
|
|
.help_msg = "showPc",
|
|
.action_msg = "Show Regs",
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
};
|
|
|
|
|
|
static void sysrq_handle_showstate(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
show_state();
|
|
}
|
|
static struct sysrq_key_op sysrq_showstate_op = {
|
|
.handler = sysrq_handle_showstate,
|
|
.help_msg = "showTasks",
|
|
.action_msg = "Show State",
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
};
|
|
|
|
|
|
static void sysrq_handle_showmem(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
show_mem();
|
|
}
|
|
static struct sysrq_key_op sysrq_showmem_op = {
|
|
.handler = sysrq_handle_showmem,
|
|
.help_msg = "showMem",
|
|
.action_msg = "Show Memory",
|
|
.enable_mask = SYSRQ_ENABLE_DUMP,
|
|
};
|
|
|
|
/* SHOW SYSRQ HANDLERS BLOCK */
|
|
|
|
|
|
/* SIGNAL SYSRQ HANDLERS BLOCK */
|
|
|
|
/* signal sysrq helper function
|
|
* Sends a signal to all user processes */
|
|
static void send_sig_all(int sig)
|
|
{
|
|
struct task_struct *p;
|
|
|
|
for_each_process(p) {
|
|
if (p->mm && p->pid != 1)
|
|
/* Not swapper, init nor kernel thread */
|
|
force_sig(sig, p);
|
|
}
|
|
}
|
|
|
|
static void sysrq_handle_term(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
send_sig_all(SIGTERM);
|
|
console_loglevel = 8;
|
|
}
|
|
static struct sysrq_key_op sysrq_term_op = {
|
|
.handler = sysrq_handle_term,
|
|
.help_msg = "tErm",
|
|
.action_msg = "Terminate All Tasks",
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
|
};
|
|
|
|
static void moom_callback(void *ignored)
|
|
{
|
|
out_of_memory(GFP_KERNEL);
|
|
}
|
|
|
|
static DECLARE_WORK(moom_work, moom_callback, NULL);
|
|
|
|
static void sysrq_handle_moom(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
schedule_work(&moom_work);
|
|
}
|
|
static struct sysrq_key_op sysrq_moom_op = {
|
|
.handler = sysrq_handle_moom,
|
|
.help_msg = "Full",
|
|
.action_msg = "Manual OOM execution",
|
|
};
|
|
|
|
static void sysrq_handle_kill(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
send_sig_all(SIGKILL);
|
|
console_loglevel = 8;
|
|
}
|
|
static struct sysrq_key_op sysrq_kill_op = {
|
|
.handler = sysrq_handle_kill,
|
|
.help_msg = "kIll",
|
|
.action_msg = "Kill All Tasks",
|
|
.enable_mask = SYSRQ_ENABLE_SIGNAL,
|
|
};
|
|
|
|
/* END SIGNAL SYSRQ HANDLERS BLOCK */
|
|
|
|
static void sysrq_handle_unrt(int key, struct pt_regs *pt_regs,
|
|
struct tty_struct *tty)
|
|
{
|
|
normalize_rt_tasks();
|
|
}
|
|
static struct sysrq_key_op sysrq_unrt_op = {
|
|
.handler = sysrq_handle_unrt,
|
|
.help_msg = "Nice",
|
|
.action_msg = "Nice All RT Tasks",
|
|
.enable_mask = SYSRQ_ENABLE_RTNICE,
|
|
};
|
|
|
|
/* Key Operations table and lock */
|
|
static DEFINE_SPINLOCK(sysrq_key_table_lock);
|
|
#define SYSRQ_KEY_TABLE_LENGTH 36
|
|
static struct sysrq_key_op *sysrq_key_table[SYSRQ_KEY_TABLE_LENGTH] = {
|
|
/* 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 */ &sysrq_loglevel_op,
|
|
/* a */ NULL, /* Don't use for system provided sysrqs,
|
|
it is handled specially on the sparc
|
|
and will never arrive */
|
|
/* b */ &sysrq_reboot_op,
|
|
#ifdef CONFIG_KEXEC
|
|
/* c */ &sysrq_crashdump_op,
|
|
#else
|
|
/* c */ NULL,
|
|
#endif
|
|
/* d */ NULL,
|
|
/* e */ &sysrq_term_op,
|
|
/* f */ &sysrq_moom_op,
|
|
/* g */ NULL,
|
|
/* h */ NULL,
|
|
/* i */ &sysrq_kill_op,
|
|
/* j */ NULL,
|
|
#ifdef CONFIG_VT
|
|
/* k */ &sysrq_SAK_op,
|
|
#else
|
|
/* k */ NULL,
|
|
#endif
|
|
/* l */ NULL,
|
|
/* m */ &sysrq_showmem_op,
|
|
/* n */ &sysrq_unrt_op,
|
|
/* o */ NULL, /* This will often be registered
|
|
as 'Off' at init time */
|
|
/* p */ &sysrq_showregs_op,
|
|
/* q */ NULL,
|
|
#ifdef CONFIG_VT
|
|
/* r */ &sysrq_unraw_op,
|
|
#else
|
|
/* r */ NULL,
|
|
#endif
|
|
/* s */ &sysrq_sync_op,
|
|
/* t */ &sysrq_showstate_op,
|
|
/* u */ &sysrq_mountro_op,
|
|
/* v */ NULL, /* May be assigned at init time by SMP VOYAGER */
|
|
/* w */ NULL,
|
|
/* x */ NULL,
|
|
/* y */ NULL,
|
|
/* z */ NULL
|
|
};
|
|
|
|
/* key2index calculation, -1 on invalid index */
|
|
static int sysrq_key_table_key2index(int key) {
|
|
int retval;
|
|
if ((key >= '0') && (key <= '9')) {
|
|
retval = key - '0';
|
|
} else if ((key >= 'a') && (key <= 'z')) {
|
|
retval = key + 10 - 'a';
|
|
} else {
|
|
retval = -1;
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* get and put functions for the table, exposed to modules.
|
|
*/
|
|
|
|
struct sysrq_key_op *__sysrq_get_key_op (int key) {
|
|
struct sysrq_key_op *op_p;
|
|
int i;
|
|
|
|
i = sysrq_key_table_key2index(key);
|
|
op_p = (i == -1) ? NULL : sysrq_key_table[i];
|
|
return op_p;
|
|
}
|
|
|
|
void __sysrq_put_key_op (int key, struct sysrq_key_op *op_p) {
|
|
int i;
|
|
|
|
i = sysrq_key_table_key2index(key);
|
|
if (i != -1)
|
|
sysrq_key_table[i] = op_p;
|
|
}
|
|
|
|
/*
|
|
* This is the non-locking version of handle_sysrq
|
|
* It must/can only be called by sysrq key handlers,
|
|
* as they are inside of the lock
|
|
*/
|
|
|
|
void __handle_sysrq(int key, struct pt_regs *pt_regs, struct tty_struct *tty, int check_mask)
|
|
{
|
|
struct sysrq_key_op *op_p;
|
|
int orig_log_level;
|
|
int i, j;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&sysrq_key_table_lock, flags);
|
|
orig_log_level = console_loglevel;
|
|
console_loglevel = 7;
|
|
printk(KERN_INFO "SysRq : ");
|
|
|
|
op_p = __sysrq_get_key_op(key);
|
|
if (op_p) {
|
|
/* Should we check for enabled operations (/proc/sysrq-trigger should not)
|
|
* and is the invoked operation enabled? */
|
|
if (!check_mask || sysrq_enabled == 1 ||
|
|
(sysrq_enabled & op_p->enable_mask)) {
|
|
printk ("%s\n", op_p->action_msg);
|
|
console_loglevel = orig_log_level;
|
|
op_p->handler(key, pt_regs, tty);
|
|
}
|
|
else
|
|
printk("This sysrq operation is disabled.\n");
|
|
} else {
|
|
printk("HELP : ");
|
|
/* Only print the help msg once per handler */
|
|
for (i=0; i<SYSRQ_KEY_TABLE_LENGTH; i++)
|
|
if (sysrq_key_table[i]) {
|
|
for (j=0; sysrq_key_table[i] != sysrq_key_table[j]; j++);
|
|
if (j == i)
|
|
printk ("%s ", sysrq_key_table[i]->help_msg);
|
|
}
|
|
printk ("\n");
|
|
console_loglevel = orig_log_level;
|
|
}
|
|
spin_unlock_irqrestore(&sysrq_key_table_lock, flags);
|
|
}
|
|
|
|
/*
|
|
* This function is called by the keyboard handler when SysRq is pressed
|
|
* and any other keycode arrives.
|
|
*/
|
|
|
|
void handle_sysrq(int key, struct pt_regs *pt_regs, struct tty_struct *tty)
|
|
{
|
|
if (!sysrq_enabled)
|
|
return;
|
|
__handle_sysrq(key, pt_regs, tty, 1);
|
|
}
|
|
|
|
int __sysrq_swap_key_ops(int key, struct sysrq_key_op *insert_op_p,
|
|
struct sysrq_key_op *remove_op_p) {
|
|
|
|
int retval;
|
|
unsigned long flags;
|
|
|
|
spin_lock_irqsave(&sysrq_key_table_lock, flags);
|
|
if (__sysrq_get_key_op(key) == remove_op_p) {
|
|
__sysrq_put_key_op(key, insert_op_p);
|
|
retval = 0;
|
|
} else {
|
|
retval = -1;
|
|
}
|
|
spin_unlock_irqrestore(&sysrq_key_table_lock, flags);
|
|
|
|
return retval;
|
|
}
|
|
|
|
int register_sysrq_key(int key, struct sysrq_key_op *op_p)
|
|
{
|
|
return __sysrq_swap_key_ops(key, op_p, NULL);
|
|
}
|
|
|
|
int unregister_sysrq_key(int key, struct sysrq_key_op *op_p)
|
|
{
|
|
return __sysrq_swap_key_ops(key, NULL, op_p);
|
|
}
|
|
|
|
EXPORT_SYMBOL(handle_sysrq);
|
|
EXPORT_SYMBOL(register_sysrq_key);
|
|
EXPORT_SYMBOL(unregister_sysrq_key);
|