Commit 3f724c40 authored by Tony Camuso's avatar Tony Camuso Committed by Corey Minyard

ipmi_si: use smi_num for init_name

Commit 1abf71ee moved the creation of new_smi->dev to earlier in the init
sequence in order to provide infrastructure for log printing.

However, the init_name was created with a hard-coded value of zero. This
presents a problem in systems with more than one interface, producing a
call trace in dmesg.

To correct the problem, simply use smi_num instead of the hard-coded
value of zero.

Tested on a lenovo x3950.
Signed-off-by: default avatarTony Camuso <tcamuso@redhat.com>

There was actually a more general problem, the platform device wasn't
being set correctly, either, and there was a possible (though extremely
unlikely) race on smi_num.  Add locks to clean up the race and use the
proper value for the platform device, too.

Tested on qemu in various configurations.
Signed-off-by: default avatarCorey Minyard <cminyard@mvista.com>
parent 7ecaff77
...@@ -1954,7 +1954,9 @@ static int hotmod_handler(const char *val, struct kernel_param *kp) ...@@ -1954,7 +1954,9 @@ static int hotmod_handler(const char *val, struct kernel_param *kp)
kfree(info); kfree(info);
goto out; goto out;
} }
mutex_lock(&smi_infos_lock);
rv = try_smi_init(info); rv = try_smi_init(info);
mutex_unlock(&smi_infos_lock);
if (rv) { if (rv) {
cleanup_one_si(info); cleanup_one_si(info);
goto out; goto out;
...@@ -2042,8 +2044,10 @@ static int hardcode_find_bmc(void) ...@@ -2042,8 +2044,10 @@ static int hardcode_find_bmc(void)
info->slave_addr = slave_addrs[i]; info->slave_addr = slave_addrs[i];
if (!add_smi(info)) { if (!add_smi(info)) {
mutex_lock(&smi_infos_lock);
if (try_smi_init(info)) if (try_smi_init(info))
cleanup_one_si(info); cleanup_one_si(info);
mutex_unlock(&smi_infos_lock);
ret = 0; ret = 0;
} else { } else {
kfree(info); kfree(info);
...@@ -3492,6 +3496,11 @@ static int add_smi(struct smi_info *new_smi) ...@@ -3492,6 +3496,11 @@ static int add_smi(struct smi_info *new_smi)
return rv; return rv;
} }
/*
* Try to start up an interface. Must be called with smi_infos_lock
* held, primarily to keep smi_num consistent, we only one to do these
* one at a time.
*/
static int try_smi_init(struct smi_info *new_smi) static int try_smi_init(struct smi_info *new_smi)
{ {
int rv = 0; int rv = 0;
...@@ -3524,9 +3533,12 @@ static int try_smi_init(struct smi_info *new_smi) ...@@ -3524,9 +3533,12 @@ static int try_smi_init(struct smi_info *new_smi)
goto out_err; goto out_err;
} }
new_smi->intf_num = smi_num;
/* Do this early so it's available for logs. */ /* Do this early so it's available for logs. */
if (!new_smi->dev) { if (!new_smi->dev) {
init_name = kasprintf(GFP_KERNEL, "ipmi_si.%d", 0); init_name = kasprintf(GFP_KERNEL, "ipmi_si.%d",
new_smi->intf_num);
/* /*
* If we don't already have a device from something * If we don't already have a device from something
...@@ -3593,8 +3605,6 @@ static int try_smi_init(struct smi_info *new_smi) ...@@ -3593,8 +3605,6 @@ static int try_smi_init(struct smi_info *new_smi)
new_smi->interrupt_disabled = true; new_smi->interrupt_disabled = true;
atomic_set(&new_smi->need_watch, 0); atomic_set(&new_smi->need_watch, 0);
new_smi->intf_num = smi_num;
smi_num++;
rv = try_enable_event_buffer(new_smi); rv = try_enable_event_buffer(new_smi);
if (rv == 0) if (rv == 0)
...@@ -3661,6 +3671,9 @@ static int try_smi_init(struct smi_info *new_smi) ...@@ -3661,6 +3671,9 @@ static int try_smi_init(struct smi_info *new_smi)
goto out_err_stop_timer; goto out_err_stop_timer;
} }
/* Don't increment till we know we have succeeded. */
smi_num++;
dev_info(new_smi->dev, "IPMI %s interface initialized\n", dev_info(new_smi->dev, "IPMI %s interface initialized\n",
si_to_str[new_smi->si_type]); si_to_str[new_smi->si_type]);
......
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