examples/tep_term: add TSO offload configuration
[dpdk.git] / examples / tep_termination / main.c
index 618093a..1673c0f 100644 (file)
 #define MAC_ADDR_CMP 0xFFFFFFFFFFFFULL
 
 #define CMD_LINE_OPT_NB_DEVICES "nb-devices"
+#define CMD_LINE_OPT_UDP_PORT "udp-port"
+#define CMD_LINE_OPT_TX_CHECKSUM "tx-checksum"
+#define CMD_LINE_OPT_TSO_SEGSZ "tso-segsz"
+#define CMD_LINE_OPT_FILTER_TYPE "filter-type"
 #define CMD_LINE_OPT_RX_RETRY "rx-retry"
 #define CMD_LINE_OPT_RX_RETRY_DELAY "rx-retry-delay"
 #define CMD_LINE_OPT_RX_RETRY_NUM "rx-retry-num"
@@ -125,7 +129,7 @@ static uint32_t enabled_port_mask;
 static uint32_t nb_switching_cores;
 
 /* number of devices/queues to support*/
-uint32_t nb_devices;
+uint16_t nb_devices = 2;
 
 /* max ring descriptor, ixgbe, i40e, e1000 all are 4096. */
 #define MAX_RING_DESC 4096
@@ -136,6 +140,28 @@ struct vpool {
        uint32_t buf_size;
 } vpool_array[MAX_QUEUES+MAX_QUEUES];
 
+/* UDP tunneling port */
+uint16_t udp_port = 4789;
+
+/* enable/disable inner TX checksum */
+uint8_t tx_checksum = 0;
+
+/* TCP segment size */
+uint16_t tso_segsz = 0;
+
+/* RX filter type for tunneling packet */
+uint8_t filter_idx = 1;
+
+/* overlay packet operation */
+struct ol_switch_ops overlay_options = {
+       .port_configure = vxlan_port_init,
+       .tunnel_setup = vxlan_link,
+       .tunnel_destroy = vxlan_unlink,
+       .tx_handle = vxlan_tx_pkts,
+       .rx_handle = vxlan_rx_pkts,
+       .param_handle = NULL,
+};
+
 /* Enable stats. */
 uint32_t enable_stats = 0;
 /* Enable retries on RX. */
@@ -240,7 +266,14 @@ static void
 tep_termination_usage(const char *prgname)
 {
        RTE_LOG(INFO, VHOST_CONFIG, "%s [EAL options] -- -p PORTMASK\n"
+       "               --udp-port: UDP destination port for VXLAN packet\n"
        "               --nb-devices[1-64]: The number of virtIO device\n"
+       "               --tx-checksum [0|1]: inner Tx checksum offload\n"
+       "               --tso-segsz [0-N]: TCP segment size\n"
+       "               --filter-type[1-3]: filter type for tunneling packet\n"
+       "                   1: Inner MAC and tenent ID\n"
+       "                   2: Inner MAC and VLAN, and tenent ID\n"
+       "                   3: Outer MAC, Inner MAC and tenent ID\n"
        "               -p PORTMASK: Set mask for ports to be used by application\n"
        "               --rx-retry [0|1]: disable/enable(default) retries on rx."
        "                Enable retry if destintation queue is full\n"
@@ -265,6 +298,10 @@ tep_termination_parse_args(int argc, char **argv)
        const char *prgname = argv[0];
        static struct option long_option[] = {
                {CMD_LINE_OPT_NB_DEVICES, required_argument, NULL, 0},
+               {CMD_LINE_OPT_UDP_PORT, required_argument, NULL, 0},
+               {CMD_LINE_OPT_TX_CHECKSUM, required_argument, NULL, 0},
+               {CMD_LINE_OPT_TSO_SEGSZ, required_argument, NULL, 0},
+               {CMD_LINE_OPT_FILTER_TYPE, required_argument, NULL, 0},
                {CMD_LINE_OPT_RX_RETRY, required_argument, NULL, 0},
                {CMD_LINE_OPT_RX_RETRY_DELAY, required_argument, NULL, 0},
                {CMD_LINE_OPT_RX_RETRY_NUM, required_argument, NULL, 0},
@@ -312,9 +349,34 @@ tep_termination_parse_args(int argc, char **argv)
                                                "Invalid argument for rx-retry [0|1]\n");
                                        tep_termination_usage(prgname);
                                        return -1;
-                               } else {
+                               } else
                                        enable_retry = ret;
-                               }
+                       }
+
+                       if (!strncmp(long_option[option_index].name,
+                               CMD_LINE_OPT_TSO_SEGSZ,
+                               sizeof(CMD_LINE_OPT_TSO_SEGSZ))) {
+                               ret = parse_num_opt(optarg, INT16_MAX);
+                               if (ret == -1) {
+                                       RTE_LOG(INFO, VHOST_CONFIG,
+                                               "Invalid argument for TCP segment size [0-N]\n");
+                                       tep_termination_usage(prgname);
+                                       return -1;
+                               } else
+                                       tso_segsz = ret;
+                       }
+
+                       if (!strncmp(long_option[option_index].name,
+                                       CMD_LINE_OPT_UDP_PORT,
+                                       sizeof(CMD_LINE_OPT_UDP_PORT))) {
+                               ret = parse_num_opt(optarg, INT16_MAX);
+                               if (ret == -1) {
+                                       RTE_LOG(INFO, VHOST_CONFIG,
+                                               "Invalid argument for UDP port [0-N]\n");
+                                       tep_termination_usage(prgname);
+                                       return -1;
+                               } else
+                                       udp_port = ret;
                        }
 
                        /* Specify the retries delay time (in useconds) on RX.*/
@@ -327,9 +389,8 @@ tep_termination_parse_args(int argc, char **argv)
                                                "Invalid argument for rx-retry-delay [0-N]\n");
                                        tep_termination_usage(prgname);
                                        return -1;
-                               } else {
+                               } else
                                        burst_rx_delay_time = ret;
-                               }
                        }
 
                        /* Specify the retries number on RX. */
@@ -342,9 +403,34 @@ tep_termination_parse_args(int argc, char **argv)
                                                "Invalid argument for rx-retry-num [0-N]\n");
                                        tep_termination_usage(prgname);
                                        return -1;
-                               } else {
+                               } else
                                        burst_rx_retry_num = ret;
-                               }
+                       }
+
+                       if (!strncmp(long_option[option_index].name,
+                               CMD_LINE_OPT_TX_CHECKSUM,
+                               sizeof(CMD_LINE_OPT_TX_CHECKSUM))) {
+                               ret = parse_num_opt(optarg, 1);
+                               if (ret == -1) {
+                                       RTE_LOG(INFO, VHOST_CONFIG,
+                                               "Invalid argument for tx-checksum [0|1]\n");
+                                       tep_termination_usage(prgname);
+                                       return -1;
+                               } else
+                                       tx_checksum = ret;
+                       }
+
+                       if (!strncmp(long_option[option_index].name,
+                                       CMD_LINE_OPT_FILTER_TYPE,
+                                       sizeof(CMD_LINE_OPT_FILTER_TYPE))) {
+                               ret = parse_num_opt(optarg, 3);
+                               if ((ret == -1) || (ret == 0)) {
+                                       RTE_LOG(INFO, VHOST_CONFIG,
+                                               "Invalid argument for filter type [1-3]\n");
+                                       tep_termination_usage(prgname);
+                                       return -1;
+                               } else
+                                       filter_idx = ret - 1;
                        }
 
                        /* Enable/disable stats. */
@@ -357,9 +443,8 @@ tep_termination_parse_args(int argc, char **argv)
                                                        "Invalid argument for stats [0..N]\n");
                                        tep_termination_usage(prgname);
                                        return -1;
-                               } else {
+                               } else
                                        enable_stats = ret;
-                               }
                        }
 
                        /* Set character device basename. */
@@ -459,6 +544,10 @@ virtio_tx_route(struct vhost_dev *vdev, struct rte_mbuf *m)
 
        if (unlikely(len == MAX_PKT_BURST)) {
                m_table = (struct rte_mbuf **)tx_q->m_table;
+               ret = overlay_options.tx_handle(ports[0],
+                       (uint16_t)tx_q->txq_id, m_table,
+                       (uint16_t)tx_q->len);
+
                /* Free any buffers not handled by TX and update
                 * the port stats.
                 */
@@ -525,6 +614,10 @@ switch_worker(__rte_unused void *arg)
                                LOG_DEBUG(VHOST_DATA, "TX queue drained after "
                                        "timeout with burst size %u\n",
                                        tx_q->len);
+                               ret = overlay_options.tx_handle(ports[0],
+                                       (uint16_t)tx_q->txq_id,
+                                       (struct rte_mbuf **)tx_q->m_table,
+                                       (uint16_t)tx_q->len);
                                if (unlikely(ret < tx_q->len)) {
                                        do {
                                                rte_pktmbuf_free(tx_q->m_table[ret]);
@@ -559,6 +652,7 @@ switch_worker(__rte_unused void *arg)
 
                        if (unlikely(vdev->remove)) {
                                dev_ll = dev_ll->next;
+                               overlay_options.tunnel_destroy(vdev);
                                vdev->ready = DEVICE_SAFE_REMOVE;
                                continue;
                        }
@@ -584,6 +678,7 @@ switch_worker(__rte_unused void *arg)
                                                }
                                        }
 
+                                       ret_count = overlay_options.rx_handle(dev, pkts_burst, rx_count);
                                        if (enable_stats) {
                                                rte_atomic64_add(
                                                &dev_statistics[dev->device_fh].rx_total_atomic,
@@ -606,7 +701,8 @@ switch_worker(__rte_unused void *arg)
                                                pkts_burst, MAX_PKT_BURST);
                                /* If this is the first received packet we need to learn the MAC */
                                if (unlikely(vdev->ready == DEVICE_MAC_LEARNING) && tx_count) {
-                                       if (vdev->remove) {
+                                       if (vdev->remove ||
+                                               (overlay_options.tunnel_setup(vdev, pkts_burst[0]) == -1)) {
                                                while (tx_count)
                                                        rte_pktmbuf_free(pkts_burst[--tx_count]);
                                        }
@@ -1059,7 +1155,6 @@ main(int argc, char *argv[])
                        "but only %u port can be enabled\n", nb_ports,
                        MAX_SUP_PORTS);
        }
-
        /* Create the mbuf pool. */
        mbuf_pool = rte_mempool_create(
                        "MBUF_POOL",
@@ -1087,6 +1182,9 @@ main(int argc, char *argv[])
                                "Skipping disabled port %d\n", portid);
                        continue;
                }
+               if (overlay_options.port_configure(portid, mbuf_pool) != 0)
+                       rte_exit(EXIT_FAILURE,
+                               "Cannot initialize network ports\n");
        }
 
        /* Initialise all linked lists. */
@@ -1105,7 +1203,6 @@ main(int argc, char *argv[])
                rte_eal_remote_launch(switch_worker,
                        mbuf_pool, lcore_id);
        }
-
        rte_vhost_feature_disable(1ULL << VIRTIO_NET_F_MRG_RXBUF);
 
        /* Register CUSE device to handle IOCTLs. */