forked from mirrors/linux
		
	pktgen: introduce xmit_mode '<start_xmit|netif_receive>'
Introduce xmit_mode 'netif_receive' for pktgen which generates the
packets using familiar pktgen commands, but feeds them into
netif_receive_skb() instead of ndo_start_xmit().
Default mode is called 'start_xmit'.
It is designed to test netif_receive_skb and ingress qdisc
performace only. Make sure to understand how it works before
using it for other rx benchmarking.
Sample script 'pktgen.sh':
\#!/bin/bash
function pgset() {
  local result
  echo $1 > $PGDEV
  result=`cat $PGDEV | fgrep "Result: OK:"`
  if [ "$result" = "" ]; then
    cat $PGDEV | fgrep Result:
  fi
}
[ -z "$1" ] && echo "Usage: $0 DEV" && exit 1
ETH=$1
PGDEV=/proc/net/pktgen/kpktgend_0
pgset "rem_device_all"
pgset "add_device $ETH"
PGDEV=/proc/net/pktgen/$ETH
pgset "xmit_mode netif_receive"
pgset "pkt_size 60"
pgset "dst 198.18.0.1"
pgset "dst_mac 90:e2:ba:ff:ff:ff"
pgset "count 10000000"
pgset "burst 32"
PGDEV=/proc/net/pktgen/pgctrl
echo "Running... ctrl^C to stop"
pgset "start"
echo "Done"
cat /proc/net/pktgen/$ETH
Usage:
$ sudo ./pktgen.sh eth2
...
Result: OK: 232376(c232372+d3) usec, 10000000 (60byte,0frags)
  43033682pps 20656Mb/sec (20656167360bps) errors: 10000000
Raw netif_receive_skb speed should be ~43 million packet
per second on 3.7Ghz x86 and 'perf report' should look like:
  37.69%  kpktgend_0   [kernel.vmlinux]  [k] __netif_receive_skb_core
  25.81%  kpktgend_0   [kernel.vmlinux]  [k] kfree_skb
   7.22%  kpktgend_0   [kernel.vmlinux]  [k] ip_rcv
   5.68%  kpktgend_0   [pktgen]          [k] pktgen_thread_worker
If fib_table_lookup is seen on top, it means skb was processed
by the stack. To benchmark netif_receive_skb only make sure
that 'dst_mac' of your pktgen script is different from
receiving device mac and it will be dropped by ip_rcv
Signed-off-by: Alexei Starovoitov <ast@plumgrid.com>
Signed-off-by: Jesper Dangaard Brouer <brouer@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
			
			
This commit is contained in:
		
							parent
							
								
									f1f00d8ff6
								
							
						
					
					
						commit
						62f64aed62
					
				
					 2 changed files with 84 additions and 5 deletions
				
			
		| 
						 | 
					@ -193,6 +193,10 @@ Examples:
 | 
				
			||||||
 pgset "rate 300M"        set rate to 300 Mb/s
 | 
					 pgset "rate 300M"        set rate to 300 Mb/s
 | 
				
			||||||
 pgset "ratep 1000000"    set rate to 1Mpps
 | 
					 pgset "ratep 1000000"    set rate to 1Mpps
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 pgset "xmit_mode netif_receive"  RX inject into stack netif_receive_skb()
 | 
				
			||||||
 | 
									  Works with "burst" but not with "clone_skb".
 | 
				
			||||||
 | 
									  Default xmit_mode is "start_xmit".
 | 
				
			||||||
 | 
					
 | 
				
			||||||
Sample scripts
 | 
					Sample scripts
 | 
				
			||||||
==============
 | 
					==============
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -310,6 +314,9 @@ flowlen
 | 
				
			||||||
rate
 | 
					rate
 | 
				
			||||||
ratep
 | 
					ratep
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					xmit_mode <start_xmit|netif_receive>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
References:
 | 
					References:
 | 
				
			||||||
ftp://robur.slu.se/pub/Linux/net-development/pktgen-testing/
 | 
					ftp://robur.slu.se/pub/Linux/net-development/pktgen-testing/
 | 
				
			||||||
ftp://robur.slu.se/pub/Linux/net-development/pktgen-testing/examples/
 | 
					ftp://robur.slu.se/pub/Linux/net-development/pktgen-testing/examples/
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -210,6 +210,10 @@
 | 
				
			||||||
#define T_REMDEVALL   (1<<2)	/* Remove all devs */
 | 
					#define T_REMDEVALL   (1<<2)	/* Remove all devs */
 | 
				
			||||||
#define T_REMDEV      (1<<3)	/* Remove one dev */
 | 
					#define T_REMDEV      (1<<3)	/* Remove one dev */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Xmit modes */
 | 
				
			||||||
 | 
					#define M_START_XMIT		0	/* Default normal TX */
 | 
				
			||||||
 | 
					#define M_NETIF_RECEIVE 	1	/* Inject packets into stack */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* If lock -- protects updating of if_list */
 | 
					/* If lock -- protects updating of if_list */
 | 
				
			||||||
#define   if_lock(t)           spin_lock(&(t->if_lock));
 | 
					#define   if_lock(t)           spin_lock(&(t->if_lock));
 | 
				
			||||||
#define   if_unlock(t)           spin_unlock(&(t->if_lock));
 | 
					#define   if_unlock(t)           spin_unlock(&(t->if_lock));
 | 
				
			||||||
| 
						 | 
					@ -251,13 +255,14 @@ struct pktgen_dev {
 | 
				
			||||||
	 * we will do a random selection from within the range.
 | 
						 * we will do a random selection from within the range.
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	__u32 flags;
 | 
						__u32 flags;
 | 
				
			||||||
	int removal_mark;	/* non-zero => the device is marked for
 | 
						int xmit_mode;
 | 
				
			||||||
				 * removal by worker thread */
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	int min_pkt_size;
 | 
						int min_pkt_size;
 | 
				
			||||||
	int max_pkt_size;
 | 
						int max_pkt_size;
 | 
				
			||||||
	int pkt_overhead;	/* overhead for MPLS, VLANs, IPSEC etc */
 | 
						int pkt_overhead;	/* overhead for MPLS, VLANs, IPSEC etc */
 | 
				
			||||||
	int nfrags;
 | 
						int nfrags;
 | 
				
			||||||
 | 
						int removal_mark;	/* non-zero => the device is marked for
 | 
				
			||||||
 | 
									 * removal by worker thread */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	struct page *page;
 | 
						struct page *page;
 | 
				
			||||||
	u64 delay;		/* nano-seconds */
 | 
						u64 delay;		/* nano-seconds */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -620,6 +625,9 @@ static int pktgen_if_show(struct seq_file *seq, void *v)
 | 
				
			||||||
	if (pkt_dev->node >= 0)
 | 
						if (pkt_dev->node >= 0)
 | 
				
			||||||
		seq_printf(seq, "     node: %d\n", pkt_dev->node);
 | 
							seq_printf(seq, "     node: %d\n", pkt_dev->node);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (pkt_dev->xmit_mode == M_NETIF_RECEIVE)
 | 
				
			||||||
 | 
							seq_puts(seq, "     xmit_mode: netif_receive\n");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	seq_puts(seq, "     Flags: ");
 | 
						seq_puts(seq, "     Flags: ");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if (pkt_dev->flags & F_IPV6)
 | 
						if (pkt_dev->flags & F_IPV6)
 | 
				
			||||||
| 
						 | 
					@ -1081,7 +1089,8 @@ static ssize_t pktgen_if_write(struct file *file,
 | 
				
			||||||
		if (len < 0)
 | 
							if (len < 0)
 | 
				
			||||||
			return len;
 | 
								return len;
 | 
				
			||||||
		if ((value > 0) &&
 | 
							if ((value > 0) &&
 | 
				
			||||||
		    (!(pkt_dev->odev->priv_flags & IFF_TX_SKB_SHARING)))
 | 
							    ((pkt_dev->xmit_mode == M_NETIF_RECEIVE) ||
 | 
				
			||||||
 | 
							     !(pkt_dev->odev->priv_flags & IFF_TX_SKB_SHARING)))
 | 
				
			||||||
			return -ENOTSUPP;
 | 
								return -ENOTSUPP;
 | 
				
			||||||
		i += len;
 | 
							i += len;
 | 
				
			||||||
		pkt_dev->clone_skb = value;
 | 
							pkt_dev->clone_skb = value;
 | 
				
			||||||
| 
						 | 
					@ -1134,7 +1143,7 @@ static ssize_t pktgen_if_write(struct file *file,
 | 
				
			||||||
			return len;
 | 
								return len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		i += len;
 | 
							i += len;
 | 
				
			||||||
		if ((value > 1) &&
 | 
							if ((value > 1) && (pkt_dev->xmit_mode == M_START_XMIT) &&
 | 
				
			||||||
		    (!(pkt_dev->odev->priv_flags & IFF_TX_SKB_SHARING)))
 | 
							    (!(pkt_dev->odev->priv_flags & IFF_TX_SKB_SHARING)))
 | 
				
			||||||
			return -ENOTSUPP;
 | 
								return -ENOTSUPP;
 | 
				
			||||||
		pkt_dev->burst = value < 1 ? 1 : value;
 | 
							pkt_dev->burst = value < 1 ? 1 : value;
 | 
				
			||||||
| 
						 | 
					@ -1160,6 +1169,35 @@ static ssize_t pktgen_if_write(struct file *file,
 | 
				
			||||||
			sprintf(pg_result, "ERROR: node not possible");
 | 
								sprintf(pg_result, "ERROR: node not possible");
 | 
				
			||||||
		return count;
 | 
							return count;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
						if (!strcmp(name, "xmit_mode")) {
 | 
				
			||||||
 | 
							char f[32];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							memset(f, 0, 32);
 | 
				
			||||||
 | 
							len = strn_len(&user_buffer[i], sizeof(f) - 1);
 | 
				
			||||||
 | 
							if (len < 0)
 | 
				
			||||||
 | 
								return len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (copy_from_user(f, &user_buffer[i], len))
 | 
				
			||||||
 | 
								return -EFAULT;
 | 
				
			||||||
 | 
							i += len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (strcmp(f, "start_xmit") == 0) {
 | 
				
			||||||
 | 
								pkt_dev->xmit_mode = M_START_XMIT;
 | 
				
			||||||
 | 
							} else if (strcmp(f, "netif_receive") == 0) {
 | 
				
			||||||
 | 
								/* clone_skb set earlier, not supported in this mode */
 | 
				
			||||||
 | 
								if (pkt_dev->clone_skb > 0)
 | 
				
			||||||
 | 
									return -ENOTSUPP;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								pkt_dev->xmit_mode = M_NETIF_RECEIVE;
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								sprintf(pg_result,
 | 
				
			||||||
 | 
									"xmit_mode -:%s:- unknown\nAvailable modes: %s",
 | 
				
			||||||
 | 
									f, "start_xmit, netif_receive\n");
 | 
				
			||||||
 | 
								return count;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							sprintf(pg_result, "OK: xmit_mode=%s", f);
 | 
				
			||||||
 | 
							return count;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	if (!strcmp(name, "flag")) {
 | 
						if (!strcmp(name, "flag")) {
 | 
				
			||||||
		char f[32];
 | 
							char f[32];
 | 
				
			||||||
		memset(f, 0, 32);
 | 
							memset(f, 0, 32);
 | 
				
			||||||
| 
						 | 
					@ -3320,6 +3358,7 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
 | 
				
			||||||
	unsigned int burst = ACCESS_ONCE(pkt_dev->burst);
 | 
						unsigned int burst = ACCESS_ONCE(pkt_dev->burst);
 | 
				
			||||||
	struct net_device *odev = pkt_dev->odev;
 | 
						struct net_device *odev = pkt_dev->odev;
 | 
				
			||||||
	struct netdev_queue *txq;
 | 
						struct netdev_queue *txq;
 | 
				
			||||||
 | 
						struct sk_buff *skb;
 | 
				
			||||||
	int ret;
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* If device is offline, then don't send */
 | 
						/* If device is offline, then don't send */
 | 
				
			||||||
| 
						 | 
					@ -3357,6 +3396,38 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
 | 
				
			||||||
	if (pkt_dev->delay && pkt_dev->last_ok)
 | 
						if (pkt_dev->delay && pkt_dev->last_ok)
 | 
				
			||||||
		spin(pkt_dev, pkt_dev->next_tx);
 | 
							spin(pkt_dev, pkt_dev->next_tx);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (pkt_dev->xmit_mode == M_NETIF_RECEIVE) {
 | 
				
			||||||
 | 
							skb = pkt_dev->skb;
 | 
				
			||||||
 | 
							skb->protocol = eth_type_trans(skb, skb->dev);
 | 
				
			||||||
 | 
							atomic_add(burst, &skb->users);
 | 
				
			||||||
 | 
							local_bh_disable();
 | 
				
			||||||
 | 
							do {
 | 
				
			||||||
 | 
								ret = netif_receive_skb(skb);
 | 
				
			||||||
 | 
								if (ret == NET_RX_DROP)
 | 
				
			||||||
 | 
									pkt_dev->errors++;
 | 
				
			||||||
 | 
								pkt_dev->sofar++;
 | 
				
			||||||
 | 
								pkt_dev->seq_num++;
 | 
				
			||||||
 | 
								if (atomic_read(&skb->users) != burst) {
 | 
				
			||||||
 | 
									/* skb was queued by rps/rfs or taps,
 | 
				
			||||||
 | 
									 * so cannot reuse this skb
 | 
				
			||||||
 | 
									 */
 | 
				
			||||||
 | 
									atomic_sub(burst - 1, &skb->users);
 | 
				
			||||||
 | 
									/* get out of the loop and wait
 | 
				
			||||||
 | 
									 * until skb is consumed
 | 
				
			||||||
 | 
									 */
 | 
				
			||||||
 | 
									pkt_dev->last_ok = 1;
 | 
				
			||||||
 | 
									break;
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								/* skb was 'freed' by stack, so clean few
 | 
				
			||||||
 | 
								 * bits and reuse it
 | 
				
			||||||
 | 
								 */
 | 
				
			||||||
 | 
					#ifdef CONFIG_NET_CLS_ACT
 | 
				
			||||||
 | 
								skb->tc_verd = 0; /* reset reclass/redir ttl */
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
							} while (--burst > 0);
 | 
				
			||||||
 | 
							goto out; /* Skips xmit_mode M_START_XMIT */
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	txq = skb_get_tx_queue(odev, pkt_dev->skb);
 | 
						txq = skb_get_tx_queue(odev, pkt_dev->skb);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	local_bh_disable();
 | 
						local_bh_disable();
 | 
				
			||||||
| 
						 | 
					@ -3404,6 +3475,7 @@ static void pktgen_xmit(struct pktgen_dev *pkt_dev)
 | 
				
			||||||
unlock:
 | 
					unlock:
 | 
				
			||||||
	HARD_TX_UNLOCK(odev, txq);
 | 
						HARD_TX_UNLOCK(odev, txq);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					out:
 | 
				
			||||||
	local_bh_enable();
 | 
						local_bh_enable();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* If pkt_dev->count is zero, then run forever */
 | 
						/* If pkt_dev->count is zero, then run forever */
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue