Make draining a sendqueue more robust.
Add own state variable to track if a sendqueue is stopped or not. This will prevent traffic from entering the sendqueue while it is being destroyed. Update drain function to wait for traffic to be transmitted before returning when the link state is active. Add extra checks in transmit path for stopped SQ's. While at it: - Use likely() for a mbuf pointer check. - Remove redundant IFF_DRV_RUNNING check. MFC after: 1 week Sponsored by: Mellanox Technologies
This commit is contained in:
parent
d2bf00a918
commit
3dfa7645c5
@ -507,10 +507,11 @@ struct mlx5e_sq {
|
|||||||
u16 bf_offset;
|
u16 bf_offset;
|
||||||
u16 cev_counter; /* completion event counter */
|
u16 cev_counter; /* completion event counter */
|
||||||
u16 cev_factor; /* completion event factor */
|
u16 cev_factor; /* completion event factor */
|
||||||
u32 cev_next_state; /* next completion event state */
|
u16 cev_next_state; /* next completion event state */
|
||||||
#define MLX5E_CEV_STATE_INITIAL 0 /* timer not started */
|
#define MLX5E_CEV_STATE_INITIAL 0 /* timer not started */
|
||||||
#define MLX5E_CEV_STATE_SEND_NOPS 1 /* send NOPs */
|
#define MLX5E_CEV_STATE_SEND_NOPS 1 /* send NOPs */
|
||||||
#define MLX5E_CEV_STATE_HOLD_NOPS 2 /* don't send NOPs yet */
|
#define MLX5E_CEV_STATE_HOLD_NOPS 2 /* don't send NOPs yet */
|
||||||
|
u16 stopped; /* set if SQ is stopped */
|
||||||
struct callout cev_callout;
|
struct callout cev_callout;
|
||||||
union {
|
union {
|
||||||
u32 d32[2];
|
u32 d32[2];
|
||||||
|
@ -1219,8 +1219,25 @@ mlx5e_sq_cev_timeout(void *arg)
|
|||||||
void
|
void
|
||||||
mlx5e_drain_sq(struct mlx5e_sq *sq)
|
mlx5e_drain_sq(struct mlx5e_sq *sq)
|
||||||
{
|
{
|
||||||
|
int error;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check if already stopped.
|
||||||
|
*
|
||||||
|
* NOTE: The "stopped" variable is only written when both the
|
||||||
|
* priv's configuration lock and the SQ's lock is locked. It
|
||||||
|
* can therefore safely be read when only one of the two locks
|
||||||
|
* is locked. This function is always called when the priv's
|
||||||
|
* configuration lock is locked.
|
||||||
|
*/
|
||||||
|
if (sq->stopped != 0)
|
||||||
|
return;
|
||||||
|
|
||||||
mtx_lock(&sq->lock);
|
mtx_lock(&sq->lock);
|
||||||
|
|
||||||
|
/* don't put more packets into the SQ */
|
||||||
|
sq->stopped = 1;
|
||||||
|
|
||||||
/* teardown event factor timer, if any */
|
/* teardown event factor timer, if any */
|
||||||
sq->cev_next_state = MLX5E_CEV_STATE_HOLD_NOPS;
|
sq->cev_next_state = MLX5E_CEV_STATE_HOLD_NOPS;
|
||||||
callout_stop(&sq->cev_callout);
|
callout_stop(&sq->cev_callout);
|
||||||
@ -1232,14 +1249,29 @@ mlx5e_drain_sq(struct mlx5e_sq *sq)
|
|||||||
/* make sure it is safe to free the callout */
|
/* make sure it is safe to free the callout */
|
||||||
callout_drain(&sq->cev_callout);
|
callout_drain(&sq->cev_callout);
|
||||||
|
|
||||||
|
/* wait till SQ is empty or link is down */
|
||||||
|
mtx_lock(&sq->lock);
|
||||||
|
while (sq->cc != sq->pc &&
|
||||||
|
(sq->priv->media_status_last & IFM_ACTIVE) != 0) {
|
||||||
|
mtx_unlock(&sq->lock);
|
||||||
|
msleep(1);
|
||||||
|
sq->cq.mcq.comp(&sq->cq.mcq);
|
||||||
|
mtx_lock(&sq->lock);
|
||||||
|
}
|
||||||
|
mtx_unlock(&sq->lock);
|
||||||
|
|
||||||
/* error out remaining requests */
|
/* error out remaining requests */
|
||||||
mlx5e_modify_sq(sq, MLX5_SQC_STATE_RDY, MLX5_SQC_STATE_ERR);
|
error = mlx5e_modify_sq(sq, MLX5_SQC_STATE_RDY, MLX5_SQC_STATE_ERR);
|
||||||
|
if (error != 0) {
|
||||||
|
if_printf(sq->ifp,
|
||||||
|
"mlx5e_modify_sq() from RDY to ERR failed: %d\n", error);
|
||||||
|
}
|
||||||
|
|
||||||
/* wait till SQ is empty */
|
/* wait till SQ is empty */
|
||||||
mtx_lock(&sq->lock);
|
mtx_lock(&sq->lock);
|
||||||
while (sq->cc != sq->pc) {
|
while (sq->cc != sq->pc) {
|
||||||
mtx_unlock(&sq->lock);
|
mtx_unlock(&sq->lock);
|
||||||
msleep(4);
|
msleep(1);
|
||||||
sq->cq.mcq.comp(&sq->cq.mcq);
|
sq->cq.mcq.comp(&sq->cq.mcq);
|
||||||
mtx_lock(&sq->lock);
|
mtx_lock(&sq->lock);
|
||||||
}
|
}
|
||||||
|
@ -81,11 +81,15 @@ static struct mlx5e_sq *
|
|||||||
mlx5e_select_queue(struct ifnet *ifp, struct mbuf *mb)
|
mlx5e_select_queue(struct ifnet *ifp, struct mbuf *mb)
|
||||||
{
|
{
|
||||||
struct mlx5e_priv *priv = ifp->if_softc;
|
struct mlx5e_priv *priv = ifp->if_softc;
|
||||||
|
struct mlx5e_channel * volatile *ppch;
|
||||||
|
struct mlx5e_channel *pch;
|
||||||
u32 ch;
|
u32 ch;
|
||||||
u32 tc;
|
u32 tc;
|
||||||
|
|
||||||
|
ppch = priv->channel;
|
||||||
|
|
||||||
/* check if channels are successfully opened */
|
/* check if channels are successfully opened */
|
||||||
if (unlikely(priv->channel == NULL))
|
if (unlikely(ppch == NULL))
|
||||||
return (NULL);
|
return (NULL);
|
||||||
|
|
||||||
/* obtain VLAN information if present */
|
/* obtain VLAN information if present */
|
||||||
@ -123,11 +127,11 @@ mlx5e_select_queue(struct ifnet *ifp, struct mbuf *mb)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/* check if channel is allocated */
|
/* check if channel is allocated and not stopped */
|
||||||
if (unlikely(priv->channel[ch] == NULL))
|
pch = ppch[ch];
|
||||||
return (NULL);
|
if (likely(pch != NULL && pch->sq[tc].stopped == 0))
|
||||||
|
return (&pch->sq[tc]);
|
||||||
return (&priv->channel[ch]->sq[tc]);
|
return (NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u16
|
static inline u16
|
||||||
@ -445,18 +449,21 @@ mlx5e_xmit_locked(struct ifnet *ifp, struct mlx5e_sq *sq, struct mbuf *mb)
|
|||||||
struct mbuf *next;
|
struct mbuf *next;
|
||||||
int err = 0;
|
int err = 0;
|
||||||
|
|
||||||
if ((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0) {
|
if (likely(mb != NULL)) {
|
||||||
if (mb)
|
|
||||||
err = drbr_enqueue(ifp, sq->br, mb);
|
|
||||||
return (err);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mb != NULL)
|
|
||||||
/*
|
/*
|
||||||
* If we can't insert mbuf into drbr, try to xmit anyway.
|
* If we can't insert mbuf into drbr, try to xmit anyway.
|
||||||
* We keep the error we got so we could return that after xmit.
|
* We keep the error we got so we could return that after xmit.
|
||||||
*/
|
*/
|
||||||
err = drbr_enqueue(ifp, sq->br, mb);
|
err = drbr_enqueue(ifp, sq->br, mb);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check if the network interface is closed or if the SQ is
|
||||||
|
* being stopped:
|
||||||
|
*/
|
||||||
|
if (unlikely((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0 ||
|
||||||
|
sq->stopped != 0))
|
||||||
|
return (err);
|
||||||
|
|
||||||
/* Process the queue */
|
/* Process the queue */
|
||||||
while ((next = drbr_peek(ifp, sq->br)) != NULL) {
|
while ((next = drbr_peek(ifp, sq->br)) != NULL) {
|
||||||
@ -470,8 +477,6 @@ mlx5e_xmit_locked(struct ifnet *ifp, struct mlx5e_sq *sq, struct mbuf *mb)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
drbr_advance(ifp, sq->br);
|
drbr_advance(ifp, sq->br);
|
||||||
if ((ifp->if_drv_flags & IFF_DRV_RUNNING) == 0)
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
/* Check if we need to write the doorbell */
|
/* Check if we need to write the doorbell */
|
||||||
if (likely(sq->doorbell.d64 != 0)) {
|
if (likely(sq->doorbell.d64 != 0)) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user