net: ethernet: davinci_cpdma: Add boundary for rx and tx descriptors
When there is heavy transmission traffic in the CPDMA, then Rx descriptors
memory is also utilized as tx desc memory looses all rx descriptors and the
driver stops working then.
This patch adds boundary for tx and rx descriptors in bd ram dividing the
descriptor memory to ensure that during heavy transmission tx doesn't use
rx descriptors.
This patch is already applied to davinci_emac driver, since CPSW and
davici_dmac shares the same CPDMA, moving the boundry seperation from
Davinci EMAC driver to CPDMA driver which was done in the following
commit
commit 86d8c07ff2
Author: Sascha Hauer <s.hauer@pengutronix.de>
Date: Tue Jan 3 05:27:47 2012 +0000
net/davinci: do not use all descriptors for tx packets
The driver uses a shared pool for both rx and tx descriptors.
During open it queues fixed number of 128 descriptors for receive
packets. For each received packet it tries to queue another
descriptor. If this fails the descriptor is lost for rx.
The driver has no limitation on tx descriptors to use, so it
can happen during a nmap / ping -f attack that the driver
allocates all descriptors for tx and looses all rx descriptors.
The driver stops working then.
To fix this limit the number of tx descriptors used to half of
the descriptors available, the rx path uses the other half.
Tested on a custom board using nmap / ping -f to the board from
two different hosts.
Signed-off-by: Mugunthan V N <mugunthanvnm@ti.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
ee21c7e0d1
commit
fae50823d0
@ -374,6 +374,9 @@ void cpsw_tx_handler(void *token, int len, int status)
|
|||||||
struct net_device *ndev = skb->dev;
|
struct net_device *ndev = skb->dev;
|
||||||
struct cpsw_priv *priv = netdev_priv(ndev);
|
struct cpsw_priv *priv = netdev_priv(ndev);
|
||||||
|
|
||||||
|
/* Check whether the queue is stopped due to stalled tx dma, if the
|
||||||
|
* queue is stopped then start the queue as we have free desc for tx
|
||||||
|
*/
|
||||||
if (unlikely(netif_queue_stopped(ndev)))
|
if (unlikely(netif_queue_stopped(ndev)))
|
||||||
netif_start_queue(ndev);
|
netif_start_queue(ndev);
|
||||||
cpts_tx_timestamp(&priv->cpts, skb);
|
cpts_tx_timestamp(&priv->cpts, skb);
|
||||||
@ -736,6 +739,12 @@ static netdev_tx_t cpsw_ndo_start_xmit(struct sk_buff *skb,
|
|||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* If there is no more tx desc left free then we need to
|
||||||
|
* tell the kernel to stop sending us tx frames.
|
||||||
|
*/
|
||||||
|
if (unlikely(cpdma_check_free_tx_desc(priv->txch)))
|
||||||
|
netif_stop_queue(ndev);
|
||||||
|
|
||||||
return NETDEV_TX_OK;
|
return NETDEV_TX_OK;
|
||||||
fail:
|
fail:
|
||||||
priv->stats.tx_dropped++;
|
priv->stats.tx_dropped++;
|
||||||
|
@ -105,13 +105,13 @@ struct cpdma_ctlr {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct cpdma_chan {
|
struct cpdma_chan {
|
||||||
|
struct cpdma_desc __iomem *head, *tail;
|
||||||
|
void __iomem *hdp, *cp, *rxfree;
|
||||||
enum cpdma_state state;
|
enum cpdma_state state;
|
||||||
struct cpdma_ctlr *ctlr;
|
struct cpdma_ctlr *ctlr;
|
||||||
int chan_num;
|
int chan_num;
|
||||||
spinlock_t lock;
|
spinlock_t lock;
|
||||||
struct cpdma_desc __iomem *head, *tail;
|
|
||||||
int count;
|
int count;
|
||||||
void __iomem *hdp, *cp, *rxfree;
|
|
||||||
u32 mask;
|
u32 mask;
|
||||||
cpdma_handler_fn handler;
|
cpdma_handler_fn handler;
|
||||||
enum dma_data_direction dir;
|
enum dma_data_direction dir;
|
||||||
@ -217,17 +217,27 @@ desc_from_phys(struct cpdma_desc_pool *pool, dma_addr_t dma)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static struct cpdma_desc __iomem *
|
static struct cpdma_desc __iomem *
|
||||||
cpdma_desc_alloc(struct cpdma_desc_pool *pool, int num_desc)
|
cpdma_desc_alloc(struct cpdma_desc_pool *pool, int num_desc, bool is_rx)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int index;
|
int index;
|
||||||
|
int desc_start;
|
||||||
|
int desc_end;
|
||||||
struct cpdma_desc __iomem *desc = NULL;
|
struct cpdma_desc __iomem *desc = NULL;
|
||||||
|
|
||||||
spin_lock_irqsave(&pool->lock, flags);
|
spin_lock_irqsave(&pool->lock, flags);
|
||||||
|
|
||||||
index = bitmap_find_next_zero_area(pool->bitmap, pool->num_desc, 0,
|
if (is_rx) {
|
||||||
num_desc, 0);
|
desc_start = 0;
|
||||||
if (index < pool->num_desc) {
|
desc_end = pool->num_desc/2;
|
||||||
|
} else {
|
||||||
|
desc_start = pool->num_desc/2;
|
||||||
|
desc_end = pool->num_desc;
|
||||||
|
}
|
||||||
|
|
||||||
|
index = bitmap_find_next_zero_area(pool->bitmap,
|
||||||
|
desc_end, desc_start, num_desc, 0);
|
||||||
|
if (index < desc_end) {
|
||||||
bitmap_set(pool->bitmap, index, num_desc);
|
bitmap_set(pool->bitmap, index, num_desc);
|
||||||
desc = pool->iomap + pool->desc_size * index;
|
desc = pool->iomap + pool->desc_size * index;
|
||||||
pool->used_desc++;
|
pool->used_desc++;
|
||||||
@ -668,7 +678,7 @@ int cpdma_chan_submit(struct cpdma_chan *chan, void *token, void *data,
|
|||||||
goto unlock_ret;
|
goto unlock_ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
desc = cpdma_desc_alloc(ctlr->pool, 1);
|
desc = cpdma_desc_alloc(ctlr->pool, 1, is_rx_chan(chan));
|
||||||
if (!desc) {
|
if (!desc) {
|
||||||
chan->stats.desc_alloc_fail++;
|
chan->stats.desc_alloc_fail++;
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
@ -704,6 +714,29 @@ unlock_ret:
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(cpdma_chan_submit);
|
EXPORT_SYMBOL_GPL(cpdma_chan_submit);
|
||||||
|
|
||||||
|
bool cpdma_check_free_tx_desc(struct cpdma_chan *chan)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
int index;
|
||||||
|
bool ret;
|
||||||
|
struct cpdma_ctlr *ctlr = chan->ctlr;
|
||||||
|
struct cpdma_desc_pool *pool = ctlr->pool;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&pool->lock, flags);
|
||||||
|
|
||||||
|
index = bitmap_find_next_zero_area(pool->bitmap,
|
||||||
|
pool->num_desc, pool->num_desc/2, 1, 0);
|
||||||
|
|
||||||
|
if (index < pool->num_desc)
|
||||||
|
ret = true;
|
||||||
|
else
|
||||||
|
ret = false;
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&pool->lock, flags);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(cpdma_check_free_tx_desc);
|
||||||
|
|
||||||
static void __cpdma_chan_free(struct cpdma_chan *chan,
|
static void __cpdma_chan_free(struct cpdma_chan *chan,
|
||||||
struct cpdma_desc __iomem *desc,
|
struct cpdma_desc __iomem *desc,
|
||||||
int outlen, int status)
|
int outlen, int status)
|
||||||
|
@ -88,6 +88,7 @@ int cpdma_chan_process(struct cpdma_chan *chan, int quota);
|
|||||||
int cpdma_ctlr_int_ctrl(struct cpdma_ctlr *ctlr, bool enable);
|
int cpdma_ctlr_int_ctrl(struct cpdma_ctlr *ctlr, bool enable);
|
||||||
void cpdma_ctlr_eoi(struct cpdma_ctlr *ctlr);
|
void cpdma_ctlr_eoi(struct cpdma_ctlr *ctlr);
|
||||||
int cpdma_chan_int_ctrl(struct cpdma_chan *chan, bool enable);
|
int cpdma_chan_int_ctrl(struct cpdma_chan *chan, bool enable);
|
||||||
|
bool cpdma_check_free_tx_desc(struct cpdma_chan *chan);
|
||||||
|
|
||||||
enum cpdma_control {
|
enum cpdma_control {
|
||||||
CPDMA_CMD_IDLE, /* write-only */
|
CPDMA_CMD_IDLE, /* write-only */
|
||||||
|
@ -120,7 +120,6 @@ static const char emac_version_string[] = "TI DaVinci EMAC Linux v6.1";
|
|||||||
#define EMAC_DEF_TX_CH (0) /* Default 0th channel */
|
#define EMAC_DEF_TX_CH (0) /* Default 0th channel */
|
||||||
#define EMAC_DEF_RX_CH (0) /* Default 0th channel */
|
#define EMAC_DEF_RX_CH (0) /* Default 0th channel */
|
||||||
#define EMAC_DEF_RX_NUM_DESC (128)
|
#define EMAC_DEF_RX_NUM_DESC (128)
|
||||||
#define EMAC_DEF_TX_NUM_DESC (128)
|
|
||||||
#define EMAC_DEF_MAX_TX_CH (1) /* Max TX channels configured */
|
#define EMAC_DEF_MAX_TX_CH (1) /* Max TX channels configured */
|
||||||
#define EMAC_DEF_MAX_RX_CH (1) /* Max RX channels configured */
|
#define EMAC_DEF_MAX_RX_CH (1) /* Max RX channels configured */
|
||||||
#define EMAC_POLL_WEIGHT (64) /* Default NAPI poll weight */
|
#define EMAC_POLL_WEIGHT (64) /* Default NAPI poll weight */
|
||||||
@ -342,7 +341,6 @@ struct emac_priv {
|
|||||||
u32 mac_hash2;
|
u32 mac_hash2;
|
||||||
u32 multicast_hash_cnt[EMAC_NUM_MULTICAST_BITS];
|
u32 multicast_hash_cnt[EMAC_NUM_MULTICAST_BITS];
|
||||||
u32 rx_addr_type;
|
u32 rx_addr_type;
|
||||||
atomic_t cur_tx;
|
|
||||||
const char *phy_id;
|
const char *phy_id;
|
||||||
#ifdef CONFIG_OF
|
#ifdef CONFIG_OF
|
||||||
struct device_node *phy_node;
|
struct device_node *phy_node;
|
||||||
@ -1050,10 +1048,10 @@ static void emac_tx_handler(void *token, int len, int status)
|
|||||||
{
|
{
|
||||||
struct sk_buff *skb = token;
|
struct sk_buff *skb = token;
|
||||||
struct net_device *ndev = skb->dev;
|
struct net_device *ndev = skb->dev;
|
||||||
struct emac_priv *priv = netdev_priv(ndev);
|
|
||||||
|
|
||||||
atomic_dec(&priv->cur_tx);
|
|
||||||
|
|
||||||
|
/* Check whether the queue is stopped due to stalled tx dma, if the
|
||||||
|
* queue is stopped then start the queue as we have free desc for tx
|
||||||
|
*/
|
||||||
if (unlikely(netif_queue_stopped(ndev)))
|
if (unlikely(netif_queue_stopped(ndev)))
|
||||||
netif_start_queue(ndev);
|
netif_start_queue(ndev);
|
||||||
ndev->stats.tx_packets++;
|
ndev->stats.tx_packets++;
|
||||||
@ -1101,7 +1099,10 @@ static int emac_dev_xmit(struct sk_buff *skb, struct net_device *ndev)
|
|||||||
goto fail_tx;
|
goto fail_tx;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (atomic_inc_return(&priv->cur_tx) >= EMAC_DEF_TX_NUM_DESC)
|
/* If there is no more tx desc left free then we need to
|
||||||
|
* tell the kernel to stop sending us tx frames.
|
||||||
|
*/
|
||||||
|
if (unlikely(cpdma_check_free_tx_desc(priv->txch)))
|
||||||
netif_stop_queue(ndev);
|
netif_stop_queue(ndev);
|
||||||
|
|
||||||
return NETDEV_TX_OK;
|
return NETDEV_TX_OK;
|
||||||
|
Loading…
Reference in New Issue
Block a user