diff options
| -rw-r--r-- | drivers/firewire/fw-card.c | 30 | ||||
| -rw-r--r-- | drivers/firewire/fw-device.c | 5 | ||||
| -rw-r--r-- | drivers/firewire/fw-transaction.h | 20 |
3 files changed, 40 insertions, 15 deletions
diff --git a/drivers/firewire/fw-card.c b/drivers/firewire/fw-card.c index 1332b66ae0b3..da873d795aad 100644 --- a/drivers/firewire/fw-card.c +++ b/drivers/firewire/fw-card.c | |||
| @@ -16,12 +16,15 @@ | |||
| 16 | * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. | 16 | * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. |
| 17 | */ | 17 | */ |
| 18 | 18 | ||
| 19 | #include <linux/module.h> | 19 | #include <linux/completion.h> |
| 20 | #include <linux/errno.h> | 20 | #include <linux/crc-itu-t.h> |
| 21 | #include <linux/delay.h> | 21 | #include <linux/delay.h> |
| 22 | #include <linux/device.h> | 22 | #include <linux/device.h> |
| 23 | #include <linux/errno.h> | ||
| 24 | #include <linux/kref.h> | ||
| 25 | #include <linux/module.h> | ||
| 23 | #include <linux/mutex.h> | 26 | #include <linux/mutex.h> |
| 24 | #include <linux/crc-itu-t.h> | 27 | |
| 25 | #include "fw-transaction.h" | 28 | #include "fw-transaction.h" |
| 26 | #include "fw-topology.h" | 29 | #include "fw-topology.h" |
| 27 | #include "fw-device.h" | 30 | #include "fw-device.h" |
| @@ -396,7 +399,6 @@ fw_card_initialize(struct fw_card *card, const struct fw_card_driver *driver, | |||
| 396 | { | 399 | { |
| 397 | static atomic_t index = ATOMIC_INIT(-1); | 400 | static atomic_t index = ATOMIC_INIT(-1); |
| 398 | 401 | ||
| 399 | atomic_set(&card->device_count, 0); | ||
| 400 | card->index = atomic_inc_return(&index); | 402 | card->index = atomic_inc_return(&index); |
| 401 | card->driver = driver; | 403 | card->driver = driver; |
| 402 | card->device = device; | 404 | card->device = device; |
| @@ -405,6 +407,8 @@ fw_card_initialize(struct fw_card *card, const struct fw_card_driver *driver, | |||
| 405 | card->color = 0; | 407 | card->color = 0; |
| 406 | card->broadcast_channel = BROADCAST_CHANNEL_INITIAL; | 408 | card->broadcast_channel = BROADCAST_CHANNEL_INITIAL; |
| 407 | 409 | ||
| 410 | kref_init(&card->kref); | ||
| 411 | init_completion(&card->done); | ||
| 408 | INIT_LIST_HEAD(&card->transaction_list); | 412 | INIT_LIST_HEAD(&card->transaction_list); |
| 409 | spin_lock_init(&card->lock); | 413 | spin_lock_init(&card->lock); |
| 410 | setup_timer(&card->flush_timer, | 414 | setup_timer(&card->flush_timer, |
| @@ -507,6 +511,14 @@ static struct fw_card_driver dummy_driver = { | |||
| 507 | }; | 511 | }; |
| 508 | 512 | ||
| 509 | void | 513 | void |
| 514 | fw_card_release(struct kref *kref) | ||
| 515 | { | ||
| 516 | struct fw_card *card = container_of(kref, struct fw_card, kref); | ||
| 517 | |||
| 518 | complete(&card->done); | ||
| 519 | } | ||
| 520 | |||
| 521 | void | ||
| 510 | fw_core_remove_card(struct fw_card *card) | 522 | fw_core_remove_card(struct fw_card *card) |
| 511 | { | 523 | { |
| 512 | card->driver->update_phy_reg(card, 4, | 524 | card->driver->update_phy_reg(card, 4, |
| @@ -521,12 +533,10 @@ fw_core_remove_card(struct fw_card *card) | |||
| 521 | card->driver = &dummy_driver; | 533 | card->driver = &dummy_driver; |
| 522 | 534 | ||
| 523 | fw_destroy_nodes(card); | 535 | fw_destroy_nodes(card); |
| 524 | /* | 536 | |
| 525 | * Wait for all device workqueue jobs to finish. Otherwise the | 537 | /* Wait for all users, especially device workqueue jobs, to finish. */ |
| 526 | * firewire-core module could be unloaded before the jobs ran. | 538 | fw_card_put(card); |
| 527 | */ | 539 | wait_for_completion(&card->done); |
| 528 | while (atomic_read(&card->device_count) > 0) | ||
| 529 | msleep(100); | ||
| 530 | 540 | ||
| 531 | cancel_delayed_work_sync(&card->work); | 541 | cancel_delayed_work_sync(&card->work); |
| 532 | fw_flush_transactions(card); | 542 | fw_flush_transactions(card); |
diff --git a/drivers/firewire/fw-device.c b/drivers/firewire/fw-device.c index d9c8daf7ae7d..0855fb5568e8 100644 --- a/drivers/firewire/fw-device.c +++ b/drivers/firewire/fw-device.c | |||
| @@ -168,7 +168,7 @@ static void fw_device_release(struct device *dev) | |||
| 168 | fw_node_put(device->node); | 168 | fw_node_put(device->node); |
| 169 | kfree(device->config_rom); | 169 | kfree(device->config_rom); |
| 170 | kfree(device); | 170 | kfree(device); |
| 171 | atomic_dec(&card->device_count); | 171 | fw_card_put(card); |
| 172 | } | 172 | } |
| 173 | 173 | ||
| 174 | int fw_device_enable_phys_dma(struct fw_device *device) | 174 | int fw_device_enable_phys_dma(struct fw_device *device) |
| @@ -946,8 +946,7 @@ void fw_node_event(struct fw_card *card, struct fw_node *node, int event) | |||
| 946 | */ | 946 | */ |
| 947 | device_initialize(&device->device); | 947 | device_initialize(&device->device); |
| 948 | atomic_set(&device->state, FW_DEVICE_INITIALIZING); | 948 | atomic_set(&device->state, FW_DEVICE_INITIALIZING); |
| 949 | atomic_inc(&card->device_count); | 949 | device->card = fw_card_get(card); |
| 950 | device->card = card; | ||
| 951 | device->node = fw_node_get(node); | 950 | device->node = fw_node_get(node); |
| 952 | device->node_id = node->node_id; | 951 | device->node_id = node->node_id; |
| 953 | device->generation = card->generation; | 952 | device->generation = card->generation; |
diff --git a/drivers/firewire/fw-transaction.h b/drivers/firewire/fw-transaction.h index 219094e38542..2ae1b0d6cb7b 100644 --- a/drivers/firewire/fw-transaction.h +++ b/drivers/firewire/fw-transaction.h | |||
| @@ -19,14 +19,15 @@ | |||
| 19 | #ifndef __fw_transaction_h | 19 | #ifndef __fw_transaction_h |
| 20 | #define __fw_transaction_h | 20 | #define __fw_transaction_h |
| 21 | 21 | ||
| 22 | #include <linux/completion.h> | ||
| 22 | #include <linux/device.h> | 23 | #include <linux/device.h> |
| 23 | #include <linux/dma-mapping.h> | 24 | #include <linux/dma-mapping.h> |
| 24 | #include <linux/firewire-constants.h> | 25 | #include <linux/firewire-constants.h> |
| 26 | #include <linux/kref.h> | ||
| 25 | #include <linux/list.h> | 27 | #include <linux/list.h> |
| 26 | #include <linux/spinlock_types.h> | 28 | #include <linux/spinlock_types.h> |
| 27 | #include <linux/timer.h> | 29 | #include <linux/timer.h> |
| 28 | #include <linux/workqueue.h> | 30 | #include <linux/workqueue.h> |
| 29 | #include <asm/atomic.h> | ||
| 30 | 31 | ||
| 31 | #define TCODE_IS_READ_REQUEST(tcode) (((tcode) & ~1) == 4) | 32 | #define TCODE_IS_READ_REQUEST(tcode) (((tcode) & ~1) == 4) |
| 32 | #define TCODE_IS_BLOCK_PACKET(tcode) (((tcode) & 1) != 0) | 33 | #define TCODE_IS_BLOCK_PACKET(tcode) (((tcode) & 1) != 0) |
| @@ -219,7 +220,8 @@ extern struct bus_type fw_bus_type; | |||
| 219 | struct fw_card { | 220 | struct fw_card { |
| 220 | const struct fw_card_driver *driver; | 221 | const struct fw_card_driver *driver; |
| 221 | struct device *device; | 222 | struct device *device; |
| 222 | atomic_t device_count; | 223 | struct kref kref; |
| 224 | struct completion done; | ||
| 223 | 225 | ||
| 224 | int node_id; | 226 | int node_id; |
| 225 | int generation; | 227 | int generation; |
| @@ -260,6 +262,20 @@ struct fw_card { | |||
| 260 | int bm_generation; | 262 | int bm_generation; |
| 261 | }; | 263 | }; |
| 262 | 264 | ||
| 265 | static inline struct fw_card *fw_card_get(struct fw_card *card) | ||
| 266 | { | ||
| 267 | kref_get(&card->kref); | ||
| 268 | |||
| 269 | return card; | ||
| 270 | } | ||
| 271 | |||
| 272 | void fw_card_release(struct kref *kref); | ||
| 273 | |||
| 274 | static inline void fw_card_put(struct fw_card *card) | ||
| 275 | { | ||
| 276 | kref_put(&card->kref, fw_card_release); | ||
| 277 | } | ||
| 278 | |||
| 263 | /* | 279 | /* |
| 264 | * The iso packet format allows for an immediate header/payload part | 280 | * The iso packet format allows for an immediate header/payload part |
| 265 | * stored in 'header' immediately after the packet info plus an | 281 | * stored in 'header' immediately after the packet info plus an |
