Commit 64389998 authored by Mike Travis's avatar Mike Travis Committed by Ingo Molnar

x86/uv/nmi, kgdb/kdb: Fix UV NMI handler when KDB not configured

Fix UV call into kgdb to depend only on whether KGDB is defined
and not both KGDB and KDB.  This allows the power nmi command to
use the gdb remote connection if enabled.  Note new action of
'kgdb' needs to be set as well to indicate user wants to wait
for gdb to be connected.  If it's set to 'kdb' then an error
message is displayed if KDB is not configured.

Also note that if both KGDB and KDB are enabled, then the action
of 'kgdb' or 'kdb' has no affect on which is used.  See the KGDB
documentation for further information.
Signed-off-by: default avatarMike Travis <travis@sgi.com>
Reviewed-by: default avatarHedi Berriche <hedi@sgi.com>
Cc: Russ Anderson <rja@sgi.com>
Cc: Jason Wessel <jason.wessel@windriver.com>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Arnaldo Carvalho de Melo <acme@ghostprotocols.net>
Link: http://lkml.kernel.org/r/20140114162551.635540667@asylum.americas.sgi.comSigned-off-by: default avatarIngo Molnar <mingo@kernel.org>
parent 74c93f9d
...@@ -148,7 +148,8 @@ module_param_named(retry_count, uv_nmi_retry_count, int, 0644); ...@@ -148,7 +148,8 @@ module_param_named(retry_count, uv_nmi_retry_count, int, 0644);
* "dump" - dump process stack for each cpu * "dump" - dump process stack for each cpu
* "ips" - dump IP info for each cpu * "ips" - dump IP info for each cpu
* "kdump" - do crash dump * "kdump" - do crash dump
* "kdb" - enter KDB/KGDB (default) * "kdb" - enter KDB (default)
* "kgdb" - enter KGDB
*/ */
static char uv_nmi_action[8] = "kdb"; static char uv_nmi_action[8] = "kdb";
module_param_string(action, uv_nmi_action, sizeof(uv_nmi_action), 0644); module_param_string(action, uv_nmi_action, sizeof(uv_nmi_action), 0644);
...@@ -537,18 +538,45 @@ static inline void uv_nmi_kdump(int cpu, int master, struct pt_regs *regs) ...@@ -537,18 +538,45 @@ static inline void uv_nmi_kdump(int cpu, int master, struct pt_regs *regs)
} }
#endif /* !CONFIG_KEXEC */ #endif /* !CONFIG_KEXEC */
#ifdef CONFIG_KGDB
#ifdef CONFIG_KGDB_KDB #ifdef CONFIG_KGDB_KDB
/* Call KDB from NMI handler */ static inline int uv_nmi_kdb_reason(void)
static void uv_call_kdb(int cpu, struct pt_regs *regs, int master)
{ {
int ret; return KDB_REASON_SYSTEM_NMI;
}
#else /* !CONFIG_KGDB_KDB */
static inline int uv_nmi_kdb_reason(void)
{
/* Insure user is expecting to attach gdb remote */
if (uv_nmi_action_is("kgdb"))
return 0;
pr_err("UV: NMI error: KDB is not enabled in this kernel\n");
return -1;
}
#endif /* CONFIG_KGDB_KDB */
/*
* Call KGDB/KDB from NMI handler
*
* Note that if both KGDB and KDB are configured, then the action of 'kgdb' or
* 'kdb' has no affect on which is used. See the KGDB documention for further
* information.
*/
static void uv_call_kgdb_kdb(int cpu, struct pt_regs *regs, int master)
{
if (master) { if (master) {
int reason = uv_nmi_kdb_reason();
int ret;
if (reason < 0)
return;
/* call KGDB NMI handler as MASTER */ /* call KGDB NMI handler as MASTER */
ret = kgdb_nmicallin(cpu, X86_TRAP_NMI, regs, ret = kgdb_nmicallin(cpu, X86_TRAP_NMI, regs, reason,
KDB_REASON_SYSTEM_NMI, &uv_nmi_slave_continue); &uv_nmi_slave_continue);
if (ret) { if (ret) {
pr_alert("KDB returned error, is kgdboc set?\n"); pr_alert("KGDB returned error, is kgdboc set?\n");
atomic_set(&uv_nmi_slave_continue, SLAVE_EXIT); atomic_set(&uv_nmi_slave_continue, SLAVE_EXIT);
} }
} else { } else {
...@@ -567,12 +595,12 @@ static void uv_call_kdb(int cpu, struct pt_regs *regs, int master) ...@@ -567,12 +595,12 @@ static void uv_call_kdb(int cpu, struct pt_regs *regs, int master)
uv_nmi_sync_exit(master); uv_nmi_sync_exit(master);
} }
#else /* !CONFIG_KGDB_KDB */ #else /* !CONFIG_KGDB */
static inline void uv_call_kdb(int cpu, struct pt_regs *regs, int master) static inline void uv_call_kgdb_kdb(int cpu, struct pt_regs *regs, int master)
{ {
pr_err("UV: NMI error: KGDB/KDB is not enabled in this kernel\n"); pr_err("UV: NMI error: KGDB is not enabled in this kernel\n");
} }
#endif /* !CONFIG_KGDB_KDB */ #endif /* !CONFIG_KGDB */
/* /*
* UV NMI handler * UV NMI handler
...@@ -606,9 +634,9 @@ int uv_handle_nmi(unsigned int reason, struct pt_regs *regs) ...@@ -606,9 +634,9 @@ int uv_handle_nmi(unsigned int reason, struct pt_regs *regs)
if (uv_nmi_action_is("ips") || uv_nmi_action_is("dump")) if (uv_nmi_action_is("ips") || uv_nmi_action_is("dump"))
uv_nmi_dump_state(cpu, regs, master); uv_nmi_dump_state(cpu, regs, master);
/* Call KDB if enabled */ /* Call KGDB/KDB if enabled */
else if (uv_nmi_action_is("kdb")) else if (uv_nmi_action_is("kdb") || uv_nmi_action_is("kgdb"))
uv_call_kdb(cpu, regs, master); uv_call_kgdb_kdb(cpu, regs, master);
/* Clear per_cpu "in nmi" flag */ /* Clear per_cpu "in nmi" flag */
atomic_set(&uv_cpu_nmi.state, UV_NMI_STATE_OUT); atomic_set(&uv_cpu_nmi.state, UV_NMI_STATE_OUT);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment