[PATCH] IB: Add ib_modify_mad API to MAD
Add new MAD layer call to modify (ib_modify_mad) the timeout of a sent MAD, and simplify cancel code. Signed-off-by: Sean Hefty <sean.hefty@intel.com> Signed-off-by: Hal Rosenstock <halr@voltaire.com> Cc: Roland Dreier <rolandd@cisco.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
This commit is contained in:
Родитель
2c153b934d
Коммит
03b61ad2f2
|
@ -65,7 +65,6 @@ static void cancel_mads(struct ib_mad_agent_private *mad_agent_priv);
|
||||||
static void ib_mad_complete_send_wr(struct ib_mad_send_wr_private *mad_send_wr,
|
static void ib_mad_complete_send_wr(struct ib_mad_send_wr_private *mad_send_wr,
|
||||||
struct ib_mad_send_wc *mad_send_wc);
|
struct ib_mad_send_wc *mad_send_wc);
|
||||||
static void timeout_sends(void *data);
|
static void timeout_sends(void *data);
|
||||||
static void cancel_sends(void *data);
|
|
||||||
static void local_completions(void *data);
|
static void local_completions(void *data);
|
||||||
static int add_nonoui_reg_req(struct ib_mad_reg_req *mad_reg_req,
|
static int add_nonoui_reg_req(struct ib_mad_reg_req *mad_reg_req,
|
||||||
struct ib_mad_agent_private *agent_priv,
|
struct ib_mad_agent_private *agent_priv,
|
||||||
|
@ -346,8 +345,6 @@ struct ib_mad_agent *ib_register_mad_agent(struct ib_device *device,
|
||||||
INIT_LIST_HEAD(&mad_agent_priv->local_list);
|
INIT_LIST_HEAD(&mad_agent_priv->local_list);
|
||||||
INIT_WORK(&mad_agent_priv->local_work, local_completions,
|
INIT_WORK(&mad_agent_priv->local_work, local_completions,
|
||||||
mad_agent_priv);
|
mad_agent_priv);
|
||||||
INIT_LIST_HEAD(&mad_agent_priv->canceled_list);
|
|
||||||
INIT_WORK(&mad_agent_priv->canceled_work, cancel_sends, mad_agent_priv);
|
|
||||||
atomic_set(&mad_agent_priv->refcount, 1);
|
atomic_set(&mad_agent_priv->refcount, 1);
|
||||||
init_waitqueue_head(&mad_agent_priv->wait);
|
init_waitqueue_head(&mad_agent_priv->wait);
|
||||||
|
|
||||||
|
@ -1775,6 +1772,13 @@ static void wait_for_response(struct ib_mad_send_wr_private *mad_send_wr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ib_reset_mad_timeout(struct ib_mad_send_wr_private *mad_send_wr,
|
||||||
|
int timeout_ms)
|
||||||
|
{
|
||||||
|
mad_send_wr->timeout = msecs_to_jiffies(timeout_ms);
|
||||||
|
wait_for_response(mad_send_wr);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Process a send work completion
|
* Process a send work completion
|
||||||
*/
|
*/
|
||||||
|
@ -2034,41 +2038,7 @@ find_send_by_wr_id(struct ib_mad_agent_private *mad_agent_priv,
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cancel_sends(void *data)
|
int ib_modify_mad(struct ib_mad_agent *mad_agent, u64 wr_id, u32 timeout_ms)
|
||||||
{
|
|
||||||
struct ib_mad_agent_private *mad_agent_priv;
|
|
||||||
struct ib_mad_send_wr_private *mad_send_wr;
|
|
||||||
struct ib_mad_send_wc mad_send_wc;
|
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
mad_agent_priv = data;
|
|
||||||
|
|
||||||
mad_send_wc.status = IB_WC_WR_FLUSH_ERR;
|
|
||||||
mad_send_wc.vendor_err = 0;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
|
||||||
while (!list_empty(&mad_agent_priv->canceled_list)) {
|
|
||||||
mad_send_wr = list_entry(mad_agent_priv->canceled_list.next,
|
|
||||||
struct ib_mad_send_wr_private,
|
|
||||||
agent_list);
|
|
||||||
|
|
||||||
list_del(&mad_send_wr->agent_list);
|
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
|
||||||
|
|
||||||
mad_send_wc.wr_id = mad_send_wr->wr_id;
|
|
||||||
mad_agent_priv->agent.send_handler(&mad_agent_priv->agent,
|
|
||||||
&mad_send_wc);
|
|
||||||
|
|
||||||
kfree(mad_send_wr);
|
|
||||||
if (atomic_dec_and_test(&mad_agent_priv->refcount))
|
|
||||||
wake_up(&mad_agent_priv->wait);
|
|
||||||
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
|
||||||
}
|
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ib_cancel_mad(struct ib_mad_agent *mad_agent,
|
|
||||||
u64 wr_id)
|
|
||||||
{
|
{
|
||||||
struct ib_mad_agent_private *mad_agent_priv;
|
struct ib_mad_agent_private *mad_agent_priv;
|
||||||
struct ib_mad_send_wr_private *mad_send_wr;
|
struct ib_mad_send_wr_private *mad_send_wr;
|
||||||
|
@ -2078,29 +2048,30 @@ void ib_cancel_mad(struct ib_mad_agent *mad_agent,
|
||||||
agent);
|
agent);
|
||||||
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
||||||
mad_send_wr = find_send_by_wr_id(mad_agent_priv, wr_id);
|
mad_send_wr = find_send_by_wr_id(mad_agent_priv, wr_id);
|
||||||
if (!mad_send_wr) {
|
if (!mad_send_wr || mad_send_wr->status != IB_WC_SUCCESS) {
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
goto out;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mad_send_wr->status == IB_WC_SUCCESS)
|
if (!timeout_ms) {
|
||||||
mad_send_wr->refcount -= (mad_send_wr->timeout > 0);
|
|
||||||
|
|
||||||
if (mad_send_wr->refcount != 0) {
|
|
||||||
mad_send_wr->status = IB_WC_WR_FLUSH_ERR;
|
mad_send_wr->status = IB_WC_WR_FLUSH_ERR;
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
mad_send_wr->refcount -= (mad_send_wr->timeout > 0);
|
||||||
goto out;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
list_del(&mad_send_wr->agent_list);
|
mad_send_wr->send_wr.wr.ud.timeout_ms = timeout_ms;
|
||||||
list_add_tail(&mad_send_wr->agent_list, &mad_agent_priv->canceled_list);
|
if (!mad_send_wr->timeout || mad_send_wr->refcount > 1)
|
||||||
adjust_timeout(mad_agent_priv);
|
mad_send_wr->timeout = msecs_to_jiffies(timeout_ms);
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
else
|
||||||
|
ib_reset_mad_timeout(mad_send_wr, timeout_ms);
|
||||||
|
|
||||||
queue_work(mad_agent_priv->qp_info->port_priv->wq,
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
&mad_agent_priv->canceled_work);
|
return 0;
|
||||||
out:
|
}
|
||||||
return;
|
EXPORT_SYMBOL(ib_modify_mad);
|
||||||
|
|
||||||
|
void ib_cancel_mad(struct ib_mad_agent *mad_agent, u64 wr_id)
|
||||||
|
{
|
||||||
|
ib_modify_mad(mad_agent, wr_id, 0);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(ib_cancel_mad);
|
EXPORT_SYMBOL(ib_cancel_mad);
|
||||||
|
|
||||||
|
@ -2207,8 +2178,6 @@ static void timeout_sends(void *data)
|
||||||
unsigned long flags, delay;
|
unsigned long flags, delay;
|
||||||
|
|
||||||
mad_agent_priv = (struct ib_mad_agent_private *)data;
|
mad_agent_priv = (struct ib_mad_agent_private *)data;
|
||||||
|
|
||||||
mad_send_wc.status = IB_WC_RESP_TIMEOUT_ERR;
|
|
||||||
mad_send_wc.vendor_err = 0;
|
mad_send_wc.vendor_err = 0;
|
||||||
|
|
||||||
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
spin_lock_irqsave(&mad_agent_priv->lock, flags);
|
||||||
|
@ -2233,6 +2202,10 @@ static void timeout_sends(void *data)
|
||||||
|
|
||||||
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
spin_unlock_irqrestore(&mad_agent_priv->lock, flags);
|
||||||
|
|
||||||
|
if (mad_send_wr->status == IB_WC_SUCCESS)
|
||||||
|
mad_send_wc.status = IB_WC_RESP_TIMEOUT_ERR;
|
||||||
|
else
|
||||||
|
mad_send_wc.status = mad_send_wr->status;
|
||||||
mad_send_wc.wr_id = mad_send_wr->wr_id;
|
mad_send_wc.wr_id = mad_send_wr->wr_id;
|
||||||
mad_agent_priv->agent.send_handler(&mad_agent_priv->agent,
|
mad_agent_priv->agent.send_handler(&mad_agent_priv->agent,
|
||||||
&mad_send_wc);
|
&mad_send_wc);
|
||||||
|
|
|
@ -97,8 +97,6 @@ struct ib_mad_agent_private {
|
||||||
unsigned long timeout;
|
unsigned long timeout;
|
||||||
struct list_head local_list;
|
struct list_head local_list;
|
||||||
struct work_struct local_work;
|
struct work_struct local_work;
|
||||||
struct list_head canceled_list;
|
|
||||||
struct work_struct canceled_work;
|
|
||||||
|
|
||||||
atomic_t refcount;
|
atomic_t refcount;
|
||||||
wait_queue_head_t wait;
|
wait_queue_head_t wait;
|
||||||
|
|
|
@ -385,8 +385,18 @@ void ib_free_recv_mad(struct ib_mad_recv_wc *mad_recv_wc);
|
||||||
* MADs will be returned to the user through the corresponding
|
* MADs will be returned to the user through the corresponding
|
||||||
* ib_mad_send_handler.
|
* ib_mad_send_handler.
|
||||||
*/
|
*/
|
||||||
void ib_cancel_mad(struct ib_mad_agent *mad_agent,
|
void ib_cancel_mad(struct ib_mad_agent *mad_agent, u64 wr_id);
|
||||||
u64 wr_id);
|
|
||||||
|
/**
|
||||||
|
* ib_modify_mad - Modifies an outstanding send MAD operation.
|
||||||
|
* @mad_agent: Specifies the registration associated with sent MAD.
|
||||||
|
* @wr_id: Indicates the work request identifier of the MAD to modify.
|
||||||
|
* @timeout_ms: New timeout value for sent MAD.
|
||||||
|
*
|
||||||
|
* This call will reset the timeout value for a sent MAD to the specified
|
||||||
|
* value.
|
||||||
|
*/
|
||||||
|
int ib_modify_mad(struct ib_mad_agent *mad_agent, u64 wr_id, u32 timeout_ms);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ib_redirect_mad_qp - Registers a QP for MAD services.
|
* ib_redirect_mad_qp - Registers a QP for MAD services.
|
||||||
|
|
Загрузка…
Ссылка в новой задаче