diff options
author | Faisal Latif <faisal.latif@intel.com> | 2009-12-09 18:54:14 -0500 |
---|---|---|
committer | Roland Dreier <rolandd@cisco.com> | 2009-12-09 18:54:14 -0500 |
commit | f9f3f1e08b4d66bfda2a0c2d49a26c80489a0725 (patch) | |
tree | 22b2c03874a49781f9c9681aad807d75b572c316 /drivers/infiniband/hw/nes | |
parent | c5a7d4897156667a58fd8479f6227143573fe82d (diff) |
RDMA/nes: Abnormal listener exit causes loopback node crash
When the listener is destroyed for a loopback connection, the listener
node gets a reset event. This causes a crash as the listener is not
expecting a reset event. Code review of cm_event_reset() during
debugging showed the cm_id ref count is incremented after calling its
event handler and not before.
Signed-off-by: Faisal Latif <faisal.latif@intel.com>
Signed-off-by: Roland Dreier <rolandd@cisco.com>
Diffstat (limited to 'drivers/infiniband/hw/nes')
-rw-r--r-- | drivers/infiniband/hw/nes/nes_cm.c | 16 |
1 files changed, 3 insertions, 13 deletions
diff --git a/drivers/infiniband/hw/nes/nes_cm.c b/drivers/infiniband/hw/nes/nes_cm.c index ec04786b6069..20e21f1a18b9 100644 --- a/drivers/infiniband/hw/nes/nes_cm.c +++ b/drivers/infiniband/hw/nes/nes_cm.c | |||
@@ -1014,18 +1014,6 @@ static int mini_cm_dec_refcnt_listen(struct nes_cm_core *cm_core, | |||
1014 | cm_node->state = NES_CM_STATE_LISTENER_DESTROYED; | 1014 | cm_node->state = NES_CM_STATE_LISTENER_DESTROYED; |
1015 | loopback->state = NES_CM_STATE_CLOSED; | 1015 | loopback->state = NES_CM_STATE_CLOSED; |
1016 | 1016 | ||
1017 | event.cm_node = cm_node; | ||
1018 | event.cm_info.rem_addr = | ||
1019 | cm_node->rem_addr; | ||
1020 | event.cm_info.loc_addr = | ||
1021 | cm_node->loc_addr; | ||
1022 | event.cm_info.rem_port = | ||
1023 | cm_node->rem_port; | ||
1024 | event.cm_info.loc_port = | ||
1025 | cm_node->loc_port; | ||
1026 | event.cm_info.cm_id = cm_node->cm_id; | ||
1027 | cm_event_reset(&event); | ||
1028 | |||
1029 | rem_ref_cm_node(cm_node->cm_core, | 1017 | rem_ref_cm_node(cm_node->cm_core, |
1030 | cm_node); | 1018 | cm_node); |
1031 | 1019 | ||
@@ -3440,6 +3428,8 @@ static void cm_event_reset(struct nes_cm_event *event) | |||
3440 | 3428 | ||
3441 | nes_debug(NES_DBG_CM, "%p - cm_id = %p\n", event->cm_node, cm_id); | 3429 | nes_debug(NES_DBG_CM, "%p - cm_id = %p\n", event->cm_node, cm_id); |
3442 | nesqp = cm_id->provider_data; | 3430 | nesqp = cm_id->provider_data; |
3431 | if (!nesqp) | ||
3432 | return; | ||
3443 | 3433 | ||
3444 | nesqp->cm_id = NULL; | 3434 | nesqp->cm_id = NULL; |
3445 | /* cm_id->provider_data = NULL; */ | 3435 | /* cm_id->provider_data = NULL; */ |
@@ -3451,8 +3441,8 @@ static void cm_event_reset(struct nes_cm_event *event) | |||
3451 | cm_event.private_data = NULL; | 3441 | cm_event.private_data = NULL; |
3452 | cm_event.private_data_len = 0; | 3442 | cm_event.private_data_len = 0; |
3453 | 3443 | ||
3454 | ret = cm_id->event_handler(cm_id, &cm_event); | ||
3455 | cm_id->add_ref(cm_id); | 3444 | cm_id->add_ref(cm_id); |
3445 | ret = cm_id->event_handler(cm_id, &cm_event); | ||
3456 | atomic_inc(&cm_closes); | 3446 | atomic_inc(&cm_closes); |
3457 | cm_event.event = IW_CM_EVENT_CLOSE; | 3447 | cm_event.event = IW_CM_EVENT_CLOSE; |
3458 | cm_event.status = IW_CM_EVENT_STATUS_OK; | 3448 | cm_event.status = IW_CM_EVENT_STATUS_OK; |