aboutsummaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
authorAlexei Starovoitov <ast@plumgrid.com>2014-09-30 20:53:21 -0400
committerDavid S. Miller <davem@davemloft.net>2014-10-01 22:08:12 -0400
commit38b2cf2982dc73d3f07fe84fec8cc4ed9f64c1c5 (patch)
treee1eeab1caf5d330a4bb9c8eeb4543af5c38c769b /net
parent775dd692bd34f9201ed2aa775a0edcba4f973f3e (diff)
net: pktgen: packet bursting via skb->xmit_more
This patch demonstrates the effect of delaying update of HW tailptr. (based on earlier patch by Jesper) burst=1 is the default. It sends one packet with xmit_more=false burst=2 sends one packet with xmit_more=true and 2nd copy of the same packet with xmit_more=false burst=3 sends two copies of the same packet with xmit_more=true and 3rd copy with xmit_more=false Performance with ixgbe (usec 30): burst=1 tx:9.2 Mpps burst=2 tx:13.5 Mpps burst=3 tx:14.5 Mpps full 10G line rate Signed-off-by: Alexei Starovoitov <ast@plumgrid.com> Signed-off-by: Eric Dumazet <edumazet@google.com> Acked-by: Jesper Dangaard Brouer <brouer@redhat.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net')
-rw-r--r--net/core/pktgen.c26
1 files changed, 24 insertions, 2 deletions
diff --git a/net/core/pktgen.c b/net/core/pktgen.c
index 5c728aaf8d6c..443256bdcddc 100644
--- a/net/core/pktgen.c
+++ b/net/core/pktgen.c
@@ -387,6 +387,7 @@ struct pktgen_dev {
387 u16 queue_map_min; 387 u16 queue_map_min;
388 u16 queue_map_max; 388 u16 queue_map_max;
389 __u32 skb_priority; /* skb priority field */ 389 __u32 skb_priority; /* skb priority field */
390 unsigned int burst; /* number of duplicated packets to burst */
390 int node; /* Memory node */ 391 int node; /* Memory node */
391 392
392#ifdef CONFIG_XFRM 393#ifdef CONFIG_XFRM
@@ -613,6 +614,9 @@ static int pktgen_if_show(struct seq_file *seq, void *v)
613 if (pkt_dev->traffic_class) 614 if (pkt_dev->traffic_class)
614 seq_printf(seq, " traffic_class: 0x%02x\n", pkt_dev->traffic_class); 615 seq_printf(seq, " traffic_class: 0x%02x\n", pkt_dev->traffic_class);
615 616
617 if (pkt_dev->burst > 1)
618 seq_printf(seq, " burst: %d\n", pkt_dev->burst);
619
616 if (pkt_dev->node >= 0) 620 if (pkt_dev->node >= 0)
617 seq_printf(seq, " node: %d\n", pkt_dev->node); 621 seq_printf(seq, " node: %d\n", pkt_dev->node);
618 622
@@ -1124,6 +1128,16 @@ static ssize_t pktgen_if_write(struct file *file,
1124 pkt_dev->dst_mac_count); 1128 pkt_dev->dst_mac_count);
1125 return count; 1129 return count;
1126 } 1130 }
1131 if (!strcmp(name, "burst")) {
1132 len = num_arg(&user_buffer[i], 10, &value);
1133 if (len < 0)
1134 return len;
1135
1136 i += len;
1137 pkt_dev->burst = value < 1 ? 1 : value;
1138 sprintf(pg_result, "OK: burst=%d", pkt_dev->burst);
1139 return count;
1140 }
1127 if (!strcmp(name, "node")) { 1141 if (!strcmp(name, "node")) {
1128 len = num_arg(&user_buffer[i], 10, &value); 1142 len = num_arg(&user_buffer[i], 10, &value);
1129 if (len < 0) 1143 if (len < 0)
@@ -3297,6 +3311,7 @@ static void pktgen_wait_for_skb(struct pktgen_dev *pkt_dev)
3297 3311
3298static void pktgen_xmit(struct pktgen_dev *pkt_dev) 3312static void pktgen_xmit(struct pktgen_dev *pkt_dev)
3299{ 3313{
3314 unsigned int burst = ACCESS_ONCE(pkt_dev->burst);
3300 struct net_device *odev = pkt_dev->odev; 3315 struct net_device *odev = pkt_dev->odev;
3301 struct netdev_queue *txq; 3316 struct netdev_queue *txq;
3302 int ret; 3317 int ret;
@@ -3347,8 +3362,10 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
3347 pkt_dev->last_ok = 0; 3362 pkt_dev->last_ok = 0;
3348 goto unlock; 3363 goto unlock;
3349 } 3364 }
3350 atomic_inc(&(pkt_dev->skb->users)); 3365 atomic_add(burst, &pkt_dev->skb->users);
3351 ret = netdev_start_xmit(pkt_dev->skb, odev, txq, false); 3366
3367xmit_more:
3368 ret = netdev_start_xmit(pkt_dev->skb, odev, txq, --burst > 0);
3352 3369
3353 switch (ret) { 3370 switch (ret) {
3354 case NETDEV_TX_OK: 3371 case NETDEV_TX_OK:
@@ -3356,6 +3373,8 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
3356 pkt_dev->sofar++; 3373 pkt_dev->sofar++;
3357 pkt_dev->seq_num++; 3374 pkt_dev->seq_num++;
3358 pkt_dev->tx_bytes += pkt_dev->last_pkt_size; 3375 pkt_dev->tx_bytes += pkt_dev->last_pkt_size;
3376 if (burst > 0 && !netif_xmit_frozen_or_drv_stopped(txq))
3377 goto xmit_more;
3359 break; 3378 break;
3360 case NET_XMIT_DROP: 3379 case NET_XMIT_DROP:
3361 case NET_XMIT_CN: 3380 case NET_XMIT_CN:
@@ -3374,6 +3393,8 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
3374 atomic_dec(&(pkt_dev->skb->users)); 3393 atomic_dec(&(pkt_dev->skb->users));
3375 pkt_dev->last_ok = 0; 3394 pkt_dev->last_ok = 0;
3376 } 3395 }
3396 if (unlikely(burst))
3397 atomic_sub(burst, &pkt_dev->skb->users);
3377unlock: 3398unlock:
3378 HARD_TX_UNLOCK(odev, txq); 3399 HARD_TX_UNLOCK(odev, txq);
3379 3400
@@ -3572,6 +3593,7 @@ static int pktgen_add_device(struct pktgen_thread *t, const char *ifname)
3572 pkt_dev->svlan_p = 0; 3593 pkt_dev->svlan_p = 0;
3573 pkt_dev->svlan_cfi = 0; 3594 pkt_dev->svlan_cfi = 0;
3574 pkt_dev->svlan_id = 0xffff; 3595 pkt_dev->svlan_id = 0xffff;
3596 pkt_dev->burst = 1;
3575 pkt_dev->node = -1; 3597 pkt_dev->node = -1;
3576 3598
3577 err = pktgen_setup_dev(t->net, pkt_dev, ifname); 3599 err = pktgen_setup_dev(t->net, pkt_dev, ifname);