Commit fcfa4724 authored by Corey Minyard's avatar Corey Minyard Committed by Linus Torvalds

IPMI: add polled interface

Currently the IPMI watchdog timer sets the watchdog timeout on a panic, but it
doesn't actually poll the interface to make sure the message goes out.

Add an interface for polling the IPMI driver, and add code to the IPMI
watchdog timer to poll the interface when the timer is set from a panic.
Signed-off-by: default avatarCorey Minyard <cminyard@mvista.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 650dd0c7
...@@ -2614,6 +2614,14 @@ channel_handler(ipmi_smi_t intf, struct ipmi_recv_msg *msg) ...@@ -2614,6 +2614,14 @@ channel_handler(ipmi_smi_t intf, struct ipmi_recv_msg *msg)
return; return;
} }
void ipmi_poll_interface(ipmi_user_t user)
{
ipmi_smi_t intf = user->intf;
if (intf->handlers->poll)
intf->handlers->poll(intf->send_info);
}
int ipmi_register_smi(struct ipmi_smi_handlers *handlers, int ipmi_register_smi(struct ipmi_smi_handlers *handlers,
void *send_info, void *send_info,
struct ipmi_device_id *device_id, struct ipmi_device_id *device_id,
...@@ -4166,6 +4174,7 @@ EXPORT_SYMBOL(ipmi_destroy_user); ...@@ -4166,6 +4174,7 @@ EXPORT_SYMBOL(ipmi_destroy_user);
EXPORT_SYMBOL(ipmi_get_version); EXPORT_SYMBOL(ipmi_get_version);
EXPORT_SYMBOL(ipmi_request_settime); EXPORT_SYMBOL(ipmi_request_settime);
EXPORT_SYMBOL(ipmi_request_supply_msgs); EXPORT_SYMBOL(ipmi_request_supply_msgs);
EXPORT_SYMBOL(ipmi_poll_interface);
EXPORT_SYMBOL(ipmi_register_smi); EXPORT_SYMBOL(ipmi_register_smi);
EXPORT_SYMBOL(ipmi_unregister_smi); EXPORT_SYMBOL(ipmi_unregister_smi);
EXPORT_SYMBOL(ipmi_register_for_cmd); EXPORT_SYMBOL(ipmi_register_for_cmd);
......
...@@ -675,7 +675,8 @@ static void handle_transaction_done(struct smi_info *smi_info) ...@@ -675,7 +675,8 @@ static void handle_transaction_done(struct smi_info *smi_info)
} }
/* Called on timeouts and events. Timeouts should pass the elapsed /* Called on timeouts and events. Timeouts should pass the elapsed
time, interrupts should pass in zero. */ time, interrupts should pass in zero. Must be called with
si_lock held and interrupts disabled. */
static enum si_sm_result smi_event_handler(struct smi_info *smi_info, static enum si_sm_result smi_event_handler(struct smi_info *smi_info,
int time) int time)
{ {
...@@ -892,13 +893,16 @@ static int ipmi_thread(void *data) ...@@ -892,13 +893,16 @@ static int ipmi_thread(void *data)
static void poll(void *send_info) static void poll(void *send_info)
{ {
struct smi_info *smi_info = send_info; struct smi_info *smi_info = send_info;
unsigned long flags;
/* /*
* Make sure there is some delay in the poll loop so we can * Make sure there is some delay in the poll loop so we can
* drive time forward and timeout things. * drive time forward and timeout things.
*/ */
udelay(10); udelay(10);
spin_lock_irqsave(&smi_info->si_lock, flags);
smi_event_handler(smi_info, 10); smi_event_handler(smi_info, 10);
spin_unlock_irqrestore(&smi_info->si_lock, flags);
} }
static void request_events(void *send_info) static void request_events(void *send_info)
......
...@@ -314,8 +314,6 @@ static unsigned char ipmi_version_minor; ...@@ -314,8 +314,6 @@ static unsigned char ipmi_version_minor;
static atomic_t preop_panic_excl = ATOMIC_INIT(-1); static atomic_t preop_panic_excl = ATOMIC_INIT(-1);
static int ipmi_heartbeat(void); static int ipmi_heartbeat(void);
static void panic_halt_ipmi_heartbeat(void);
/* We use a mutex to make sure that only one thing can send a set /* We use a mutex to make sure that only one thing can send a set
timeout at one time, because we only have one copy of the data. timeout at one time, because we only have one copy of the data.
...@@ -440,19 +438,64 @@ static int ipmi_set_timeout(int do_heartbeat) ...@@ -440,19 +438,64 @@ static int ipmi_set_timeout(int do_heartbeat)
return rv; return rv;
} }
static void dummy_smi_free(struct ipmi_smi_msg *msg) static atomic_t panic_done_count = ATOMIC_INIT(0);
static void panic_smi_free(struct ipmi_smi_msg *msg)
{ {
atomic_dec(&panic_done_count);
} }
static void dummy_recv_free(struct ipmi_recv_msg *msg) static void panic_recv_free(struct ipmi_recv_msg *msg)
{
atomic_dec(&panic_done_count);
}
static struct ipmi_smi_msg panic_halt_heartbeat_smi_msg =
{
.done = panic_smi_free
};
static struct ipmi_recv_msg panic_halt_heartbeat_recv_msg =
{ {
.done = panic_recv_free
};
static void panic_halt_ipmi_heartbeat(void)
{
struct kernel_ipmi_msg msg;
struct ipmi_system_interface_addr addr;
int rv;
/* Don't reset the timer if we have the timer turned off, that
re-enables the watchdog. */
if (ipmi_watchdog_state == WDOG_TIMEOUT_NONE)
return;
addr.addr_type = IPMI_SYSTEM_INTERFACE_ADDR_TYPE;
addr.channel = IPMI_BMC_CHANNEL;
addr.lun = 0;
msg.netfn = 0x06;
msg.cmd = IPMI_WDOG_RESET_TIMER;
msg.data = NULL;
msg.data_len = 0;
rv = ipmi_request_supply_msgs(watchdog_user,
(struct ipmi_addr *) &addr,
0,
&msg,
NULL,
&panic_halt_heartbeat_smi_msg,
&panic_halt_heartbeat_recv_msg,
1);
if (!rv)
atomic_add(2, &panic_done_count);
} }
static struct ipmi_smi_msg panic_halt_smi_msg = static struct ipmi_smi_msg panic_halt_smi_msg =
{ {
.done = dummy_smi_free .done = panic_smi_free
}; };
static struct ipmi_recv_msg panic_halt_recv_msg = static struct ipmi_recv_msg panic_halt_recv_msg =
{ {
.done = dummy_recv_free .done = panic_recv_free
}; };
/* Special call, doesn't claim any locks. This is only to be called /* Special call, doesn't claim any locks. This is only to be called
...@@ -464,13 +507,21 @@ static void panic_halt_ipmi_set_timeout(void) ...@@ -464,13 +507,21 @@ static void panic_halt_ipmi_set_timeout(void)
int send_heartbeat_now; int send_heartbeat_now;
int rv; int rv;
/* Wait for the messages to be free. */
while (atomic_read(&panic_done_count) != 0)
ipmi_poll_interface(watchdog_user);
rv = i_ipmi_set_timeout(&panic_halt_smi_msg, rv = i_ipmi_set_timeout(&panic_halt_smi_msg,
&panic_halt_recv_msg, &panic_halt_recv_msg,
&send_heartbeat_now); &send_heartbeat_now);
if (!rv) { if (!rv) {
atomic_add(2, &panic_done_count);
if (send_heartbeat_now) if (send_heartbeat_now)
panic_halt_ipmi_heartbeat(); panic_halt_ipmi_heartbeat();
} } else
printk(KERN_WARNING PFX
"Unable to extend the watchdog timeout.");
while (atomic_read(&panic_done_count) != 0)
ipmi_poll_interface(watchdog_user);
} }
/* We use a semaphore to make sure that only one thing can send a /* We use a semaphore to make sure that only one thing can send a
...@@ -499,15 +550,6 @@ static struct ipmi_recv_msg heartbeat_recv_msg = ...@@ -499,15 +550,6 @@ static struct ipmi_recv_msg heartbeat_recv_msg =
.done = heartbeat_free_recv .done = heartbeat_free_recv
}; };
static struct ipmi_smi_msg panic_halt_heartbeat_smi_msg =
{
.done = dummy_smi_free
};
static struct ipmi_recv_msg panic_halt_heartbeat_recv_msg =
{
.done = dummy_recv_free
};
static int ipmi_heartbeat(void) static int ipmi_heartbeat(void)
{ {
struct kernel_ipmi_msg msg; struct kernel_ipmi_msg msg;
...@@ -580,35 +622,6 @@ static int ipmi_heartbeat(void) ...@@ -580,35 +622,6 @@ static int ipmi_heartbeat(void)
return rv; return rv;
} }
static void panic_halt_ipmi_heartbeat(void)
{
struct kernel_ipmi_msg msg;
struct ipmi_system_interface_addr addr;
/* Don't reset the timer if we have the timer turned off, that
re-enables the watchdog. */
if (ipmi_watchdog_state == WDOG_TIMEOUT_NONE)
return;
addr.addr_type = IPMI_SYSTEM_INTERFACE_ADDR_TYPE;
addr.channel = IPMI_BMC_CHANNEL;
addr.lun = 0;
msg.netfn = 0x06;
msg.cmd = IPMI_WDOG_RESET_TIMER;
msg.data = NULL;
msg.data_len = 0;
ipmi_request_supply_msgs(watchdog_user,
(struct ipmi_addr *) &addr,
0,
&msg,
NULL,
&panic_halt_heartbeat_smi_msg,
&panic_halt_heartbeat_recv_msg,
1);
}
static struct watchdog_info ident = static struct watchdog_info ident =
{ {
.options = 0, /* WDIOF_SETTIMEOUT, */ .options = 0, /* WDIOF_SETTIMEOUT, */
...@@ -998,7 +1011,7 @@ static int wdog_reboot_handler(struct notifier_block *this, ...@@ -998,7 +1011,7 @@ static int wdog_reboot_handler(struct notifier_block *this,
/* Make sure we only do this once. */ /* Make sure we only do this once. */
reboot_event_handled = 1; reboot_event_handled = 1;
if (code == SYS_DOWN || code == SYS_HALT) { if (code == SYS_POWER_OFF || code == SYS_HALT) {
/* Disable the WDT if we are shutting down. */ /* Disable the WDT if we are shutting down. */
ipmi_watchdog_state = WDOG_TIMEOUT_NONE; ipmi_watchdog_state = WDOG_TIMEOUT_NONE;
panic_halt_ipmi_set_timeout(); panic_halt_ipmi_set_timeout();
......
...@@ -364,6 +364,16 @@ int ipmi_request_supply_msgs(ipmi_user_t user, ...@@ -364,6 +364,16 @@ int ipmi_request_supply_msgs(ipmi_user_t user,
struct ipmi_recv_msg *supplied_recv, struct ipmi_recv_msg *supplied_recv,
int priority); int priority);
/*
* Poll the IPMI interface for the user. This causes the IPMI code to
* do an immediate check for information from the driver and handle
* anything that is immediately pending. This will not block in any
* way. This is useful if you need to implement polling from the user
* for things like modifying the watchdog timeout when a panic occurs
* or disabling the watchdog timer on a reboot.
*/
void ipmi_poll_interface(ipmi_user_t user);
/* /*
* When commands come in to the SMS, the user can register to receive * When commands come in to the SMS, the user can register to receive
* them. Only one user can be listening on a specific netfn/cmd/chan tuple * them. Only one user can be listening on a specific netfn/cmd/chan tuple
......
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