diff options
author | Alexei Starovoitov <ast@plumgrid.com> | 2014-09-30 20:53:21 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2014-10-01 22:08:12 -0400 |
commit | 38b2cf2982dc73d3f07fe84fec8cc4ed9f64c1c5 (patch) | |
tree | e1eeab1caf5d330a4bb9c8eeb4543af5c38c769b /net | |
parent | 775dd692bd34f9201ed2aa775a0edcba4f973f3e (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.c | 26 |
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 | ||
3298 | static void pktgen_xmit(struct pktgen_dev *pkt_dev) | 3312 | static 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 | |
3367 | xmit_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); | ||
3377 | unlock: | 3398 | unlock: |
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); |