IB/cm: Call the correct message free functions in cm_send_handler()
There are now three destroy functions for the cm_msg, and all places except the general send completion handler use the correct function. Fix cm_send_handler() to detect which kind of message is being completed and destroy it using the correct function with the correct locking. Link: https://lore.kernel.org/r/62a507195b8db85bb11228d0c6e7fa944204bf12.1622629024.git.leonro@nvidia.com Signed-off-by: Leon Romanovsky <leonro@nvidia.com> Signed-off-by: Jason Gunthorpe <jgg@nvidia.com>
This commit is contained in:
parent
4b4e586ebe
commit
c1cf6d9f74
@ -3795,22 +3795,26 @@ out:
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
static void cm_process_send_error(struct ib_mad_send_buf *msg,
|
||||
static void cm_process_send_error(struct cm_id_private *cm_id_priv,
|
||||
struct ib_mad_send_buf *msg,
|
||||
enum ib_cm_state state,
|
||||
enum ib_wc_status wc_status)
|
||||
{
|
||||
struct cm_id_private *cm_id_priv;
|
||||
struct ib_cm_event cm_event;
|
||||
enum ib_cm_state state;
|
||||
struct ib_cm_event cm_event = {};
|
||||
int ret;
|
||||
|
||||
memset(&cm_event, 0, sizeof cm_event);
|
||||
cm_id_priv = msg->context[0];
|
||||
|
||||
/* Discard old sends or ones without a response. */
|
||||
spin_lock_irq(&cm_id_priv->lock);
|
||||
state = (enum ib_cm_state) (unsigned long) msg->context[1];
|
||||
if (msg != cm_id_priv->msg || state != cm_id_priv->id.state)
|
||||
goto discard;
|
||||
if (msg != cm_id_priv->msg) {
|
||||
spin_unlock_irq(&cm_id_priv->lock);
|
||||
cm_free_msg(msg);
|
||||
return;
|
||||
}
|
||||
cm_free_priv_msg(msg);
|
||||
|
||||
if (state != cm_id_priv->id.state || wc_status == IB_WC_SUCCESS ||
|
||||
wc_status == IB_WC_WR_FLUSH_ERR)
|
||||
goto out_unlock;
|
||||
|
||||
trace_icm_mad_send_err(state, wc_status);
|
||||
switch (state) {
|
||||
@ -3833,26 +3837,27 @@ static void cm_process_send_error(struct ib_mad_send_buf *msg,
|
||||
cm_event.event = IB_CM_SIDR_REQ_ERROR;
|
||||
break;
|
||||
default:
|
||||
goto discard;
|
||||
goto out_unlock;
|
||||
}
|
||||
spin_unlock_irq(&cm_id_priv->lock);
|
||||
cm_event.param.send_status = wc_status;
|
||||
|
||||
/* No other events can occur on the cm_id at this point. */
|
||||
ret = cm_id_priv->id.cm_handler(&cm_id_priv->id, &cm_event);
|
||||
cm_free_msg(msg);
|
||||
if (ret)
|
||||
ib_destroy_cm_id(&cm_id_priv->id);
|
||||
return;
|
||||
discard:
|
||||
out_unlock:
|
||||
spin_unlock_irq(&cm_id_priv->lock);
|
||||
cm_free_msg(msg);
|
||||
}
|
||||
|
||||
static void cm_send_handler(struct ib_mad_agent *mad_agent,
|
||||
struct ib_mad_send_wc *mad_send_wc)
|
||||
{
|
||||
struct ib_mad_send_buf *msg = mad_send_wc->send_buf;
|
||||
struct cm_id_private *cm_id_priv = msg->context[0];
|
||||
enum ib_cm_state state =
|
||||
(enum ib_cm_state)(unsigned long)msg->context[1];
|
||||
struct cm_port *port;
|
||||
u16 attr_index;
|
||||
|
||||
@ -3865,7 +3870,7 @@ static void cm_send_handler(struct ib_mad_agent *mad_agent,
|
||||
* set to a cm_id), and is not a REJ, then it is a send that was
|
||||
* manually retried.
|
||||
*/
|
||||
if (!msg->context[0] && (attr_index != CM_REJ_COUNTER))
|
||||
if (!cm_id_priv && (attr_index != CM_REJ_COUNTER))
|
||||
msg->retries = 1;
|
||||
|
||||
atomic_long_add(1 + msg->retries,
|
||||
@ -3875,18 +3880,11 @@ static void cm_send_handler(struct ib_mad_agent *mad_agent,
|
||||
&port->counter_group[CM_XMIT_RETRIES].
|
||||
counter[attr_index]);
|
||||
|
||||
switch (mad_send_wc->status) {
|
||||
case IB_WC_SUCCESS:
|
||||
case IB_WC_WR_FLUSH_ERR:
|
||||
cm_free_msg(msg);
|
||||
break;
|
||||
default:
|
||||
if (msg->context[0] && msg->context[1])
|
||||
cm_process_send_error(msg, mad_send_wc->status);
|
||||
else
|
||||
cm_free_msg(msg);
|
||||
break;
|
||||
}
|
||||
if (cm_id_priv)
|
||||
cm_process_send_error(cm_id_priv, msg, state,
|
||||
mad_send_wc->status);
|
||||
else
|
||||
cm_free_response_msg(msg);
|
||||
}
|
||||
|
||||
static void cm_work_handler(struct work_struct *_work)
|
||||
|
Loading…
x
Reference in New Issue
Block a user