aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/firewire/core-card.c
diff options
context:
space:
mode:
authorStefan Richter <stefanr@s5r6.in-berlin.de>2009-09-06 12:50:29 -0400
committerStefan Richter <stefanr@s5r6.in-berlin.de>2009-09-12 08:48:40 -0400
commitb171e204b32b69e241af994d6e9be559e33535c1 (patch)
tree0ba90b20dfed4e721a573e2dfe9e4584e8a80737 /drivers/firewire/core-card.c
parent18668ff9a3232d5f942a2f7abc1ad67d2760dcdf (diff)
firewire: core: fix race with parallel PCI device probe
The config ROM buffer received from generate_config_rom is a globally shared static buffer. Extend the card_mutex protection in fw_add_card until after the config ROM was copied into the card driver's buffer. Otherwise, parallelized card driver probes may end up with ROM contents that were meant for a different card. firewire-ohci's card->driver->enable hook is safe to be called within the card_mutex. Furthermore, it is safe to reorder card_list update versus card enable, which simplifies the code a little. Signed-off-by: Stefan Richter <stefanr@s5r6.in-berlin.de>
Diffstat (limited to 'drivers/firewire/core-card.c')
-rw-r--r--drivers/firewire/core-card.c13
1 files changed, 5 insertions, 8 deletions
diff --git a/drivers/firewire/core-card.c b/drivers/firewire/core-card.c
index f74edae5cb4c..e4864e894e4f 100644
--- a/drivers/firewire/core-card.c
+++ b/drivers/firewire/core-card.c
@@ -444,16 +444,13 @@ int fw_card_add(struct fw_card *card,
444 card->guid = guid; 444 card->guid = guid;
445 445
446 mutex_lock(&card_mutex); 446 mutex_lock(&card_mutex);
447 config_rom = generate_config_rom(card, &length);
448 list_add_tail(&card->link, &card_list);
449 mutex_unlock(&card_mutex);
450 447
448 config_rom = generate_config_rom(card, &length);
451 ret = card->driver->enable(card, config_rom, length); 449 ret = card->driver->enable(card, config_rom, length);
452 if (ret < 0) { 450 if (ret == 0)
453 mutex_lock(&card_mutex); 451 list_add_tail(&card->link, &card_list);
454 list_del(&card->link); 452
455 mutex_unlock(&card_mutex); 453 mutex_unlock(&card_mutex);
456 }
457 454
458 return ret; 455 return ret;
459} 456}