app/testpmd: fix flow director mask
[dpdk.git] / app / test-pmd / cmdline.c
index 0afdc96..a82b4cc 100644 (file)
@@ -1,7 +1,7 @@
 /*-
  *   BSD LICENSE
  *
- *   Copyright(c) 2010-2015 Intel Corporation. All rights reserved.
+ *   Copyright(c) 2010-2016 Intel Corporation. All rights reserved.
  *   Copyright(c) 2014 6WIND S.A.
  *   All rights reserved.
  *
 #ifdef RTE_LIBRTE_PMD_BOND
 #include <rte_eth_bond.h>
 #endif
-
+#ifdef RTE_LIBRTE_IXGBE_PMD
+#include <rte_pmd_ixgbe.h>
+#endif
 #include "testpmd.h"
 
-static void cmd_reconfig_device_queue(portid_t id, uint8_t dev, uint8_t queue);
+static struct cmdline *testpmd_cl;
 
-#ifdef RTE_NIC_BYPASS
-uint8_t bypass_is_supported(portid_t port_id);
-#endif
+static void cmd_reconfig_device_queue(portid_t id, uint8_t dev, uint8_t queue);
 
 /* *** Help command with introduction. *** */
 struct cmd_help_brief_result {
@@ -199,7 +199,10 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "clear port (info|stats|xstats|fdir|stat_qmap) (port_id|all)\n"
                        "    Clear information for port_id, or all.\n\n"
 
-                       "show config (rxtx|cores|fwd)\n"
+                       "show (rxq|txq) info (port_id) (queue_id)\n"
+                       "    Display information for configured RX/TX queue.\n\n"
+
+                       "show config (rxtx|cores|fwd|txpkts)\n"
                        "    Display the given configuration.\n\n"
 
                        "read rxd (port_id) (queue_id) (rxd_id)\n"
@@ -241,12 +244,17 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "    Set number of packets per burst.\n\n"
 
                        "set burst tx delay (microseconds) retry (num)\n"
-                       "    Set the transmit delay time and number of retries"
-                       " in mac_retry forwarding mode.\n\n"
+                       "    Set the transmit delay time and number of retries,"
+                       " effective when retry is enabled.\n\n"
 
                        "set txpkts (x[,y]*)\n"
                        "    Set the length of each segment of TXONLY"
-                       " packets.\n\n"
+                       " and optionally CSUM packets.\n\n"
+
+                       "set txsplit (off|on|rand)\n"
+                       "    Set the split policy for the TX packets."
+                       " Right now only applicable for CSUM and TXONLY"
+                       " modes\n\n"
 
                        "set corelist (x[,y]*)\n"
                        "    Set the list of forwarding cores.\n\n"
@@ -254,12 +262,37 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "set portlist (x[,y]*)\n"
                        "    Set the list of forwarding ports.\n\n"
 
+#ifdef RTE_LIBRTE_IXGBE_PMD
+                       "set tx loopback (port_id) (on|off)\n"
+                       "    Enable or disable tx loopback.\n\n"
+
+                       "set all queues drop (port_id) (on|off)\n"
+                       "    Set drop enable bit for all queues.\n\n"
+
+                       "set vf split drop (port_id) (vf_id) (on|off)\n"
+                       "    Set split drop enable bit for a VF from the PF.\n\n"
+
+                       "set vf mac antispoof (port_id) (vf_id) (on|off).\n"
+                       "    Set MAC antispoof for a VF from the PF.\n\n"
+#endif
+
                        "vlan set strip (on|off) (port_id)\n"
                        "    Set the VLAN strip on a port.\n\n"
 
                        "vlan set stripq (on|off) (port_id,queue_id)\n"
                        "    Set the VLAN strip for a queue on a port.\n\n"
 
+#ifdef RTE_LIBRTE_IXGBE_PMD
+                       "set vf vlan stripq (port_id) (vf_id) (on|off)\n"
+                       "    Set the VLAN strip for all queues in a pool for a VF from the PF.\n\n"
+
+                       "set vf vlan insert (port_id) (vf_id) (on|off)\n"
+                       "    Set VLAN insert for a VF from the PF.\n\n"
+
+                       "set vf vlan antispoof (port_id) (vf_id) (on|off)\n"
+                       "    Set VLAN antispoof for a VF from the PF.\n\n"
+#endif
+
                        "vlan set filter (on|off) (port_id)\n"
                        "    Set the VLAN filter on a port.\n\n"
 
@@ -267,8 +300,8 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "    Set the VLAN QinQ (extended queue in queue)"
                        " on a port.\n\n"
 
-                       "vlan set tpid (value) (port_id)\n"
-                       "    Set the outer VLAN TPID for Packet Filtering on"
+                       "vlan set (inner|outer) tpid (value) (port_id)\n"
+                       "    Set the VLAN TPID for Packet Filtering on"
                        " a port\n\n"
 
                        "rx_vlan add (vlan_id|all) (port_id)\n"
@@ -287,16 +320,14 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "    Remove a vlan_id, to the set of VLAN identifiers"
                        "filtered for VF(s) from port_id.\n\n"
 
-                       "rx_vlan set tpid (value) (port_id)\n"
-                       "    Set the outer VLAN TPID for Packet Filtering on"
-                       " a port\n\n"
-
                        "tunnel_filter add (port_id) (outer_mac) (inner_mac) (ip_addr) "
-                       "(inner_vlan) (vxlan|nvgre) (filter_type) (tenant_id) (queue_id)\n"
+                       "(inner_vlan) (vxlan|nvgre|ipingre) (imac-ivlan|imac-ivlan-tenid|"
+                       "imac-tenid|imac|omac-imac-tenid|oip|iip) (tenant_id) (queue_id)\n"
                        "   add a tunnel filter of a port.\n\n"
 
                        "tunnel_filter rm (port_id) (outer_mac) (inner_mac) (ip_addr) "
-                       "(inner_vlan) (vxlan|nvgre) (filter_type) (tenant_id) (queue_id)\n"
+                       "(inner_vlan) (vxlan|nvgre|ipingre) (imac-ivlan|imac-ivlan-tenid|"
+                       "imac-tenid|imac|omac-imac-tenid|oip|iip) (tenant_id) (queue_id)\n"
                        "   remove a tunnel filter of a port.\n\n"
 
                        "rx_vxlan_port add (udp_port) (port_id)\n"
@@ -355,6 +386,11 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "mac_addr add port (port_id) vf (vf_id) (mac_address)\n"
                        "    Add a MAC address for a VF on the port.\n\n"
 
+#ifdef RTE_LIBRTE_IXGBE_PMD
+                       "set vf mac addr (port_id) (vf_id) (XX:XX:XX:XX:XX:XX)\n"
+                       "    Set the MAC address for a VF from the PF.\n\n"
+#endif
+
                        "set port (port_id) uta (mac_address|all) (on|off)\n"
                        "    Add/Remove a or all unicast hash filter(s)"
                        "from port X.\n\n"
@@ -492,6 +528,27 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "set link-down port (port_id)\n"
                        "       Set link down for a port.\n\n"
 
+                       "E-tag set insertion on port-tag-id (value)"
+                       " port (port_id) vf (vf_id)\n"
+                       "    Enable E-tag insertion for a VF on a port\n\n"
+
+                       "E-tag set insertion off port (port_id) vf (vf_id)\n"
+                       "    Disable E-tag insertion for a VF on a port\n\n"
+
+                       "E-tag set stripping (on|off) port (port_id)\n"
+                       "    Enable/disable E-tag stripping on a port\n\n"
+
+                       "E-tag set forwarding (on|off) port (port_id)\n"
+                       "    Enable/disable E-tag based forwarding"
+                       " on a port\n\n"
+
+                       "E-tag set filter add e-tag-id (value) dst-pool"
+                       " (pool_id) port (port_id)\n"
+                       "    Add an E-tag forwarding filter on a port\n\n"
+
+                       "E-tag set filter del e-tag-id (value) port (port_id)\n"
+                       "    Delete an E-tag forwarding filter on a port\n\n"
+
                        , list_pkt_forwarding_modes()
                );
        }
@@ -520,7 +577,7 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "    Detach physical or virtual dev by port_id\n\n"
 
                        "port config (port_id|all)"
-                       " speed (10|100|1000|10000|40000|auto)"
+                       " speed (10|100|1000|10000|25000|40000|50000|100000|auto)"
                        " duplex (half|full|auto)\n"
                        "    Set speed and duplex for all ports or port_id\n\n"
 
@@ -530,13 +587,13 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "port config all max-pkt-len (value)\n"
                        "    Set the max packet length.\n\n"
 
-                       "port config all (crc-strip|rx-cksum|hw-vlan|hw-vlan-filter|"
+                       "port config all (crc-strip|scatter|rx-cksum|hw-vlan|hw-vlan-filter|"
                        "hw-vlan-strip|hw-vlan-extend|drop-en)"
                        " (on|off)\n"
-                       "    Set crc-strip/rx-checksum/hardware-vlan/drop_en"
+                       "    Set crc-strip/scatter/rx-checksum/hardware-vlan/drop_en"
                        " for ports.\n\n"
 
-                       "port config all rss (all|ip|tcp|udp|sctp|ether|none)\n"
+                       "port config all rss (all|ip|tcp|udp|sctp|ether|port|vxlan|geneve|nvgre|none)\n"
                        "    Set the RSS mode.\n\n"
 
                        "port config port-id rss reta (hash,queue)[,(hash,queue)]\n"
@@ -562,7 +619,15 @@ static void cmd_help_long_parsed(void *parsed_result,
 
                        "port (port_id) (rxq|txq) (queue_id) (start|stop)\n"
                        "    Start/stop a rx/tx queue of port X. Only take effect"
-                       " when port X is started\n"
+                       " when port X is started\n\n"
+
+                       "port config (port_id|all) l2-tunnel E-tag ether-type"
+                       " (value)\n"
+                       "    Set the value of E-tag ether-type.\n\n"
+
+                       "port config (port_id|all) l2-tunnel E-tag"
+                       " (enable|disable)\n"
+                       "    Enable/disable the E-tag support.\n\n"
                );
        }
 
@@ -629,43 +694,72 @@ static void cmd_help_long_parsed(void *parsed_result,
                        " priority (prio_value) queue (queue_id)\n"
                        "    Add/Del a flex filter.\n\n"
 
-                       "flow_director_filter (port_id) (add|del|update)"
+                       "flow_director_filter (port_id) mode IP (add|del|update)"
                        " flow (ipv4-other|ipv4-frag|ipv6-other|ipv6-frag)"
                        " src (src_ip_address) dst (dst_ip_address)"
+                       " tos (tos_value) proto (proto_value) ttl (ttl_value)"
                        " vlan (vlan_value) flexbytes (flexbytes_value)"
-                       " (drop|fwd) queue (queue_id) fd_id (fd_id_value)\n"
+                       " (drop|fwd) pf|vf(vf_id) queue (queue_id)"
+                       " fd_id (fd_id_value)\n"
                        "    Add/Del an IP type flow director filter.\n\n"
 
-                       "flow_director_filter (port_id) (add|del|update)"
+                       "flow_director_filter (port_id) mode IP (add|del|update)"
                        " flow (ipv4-tcp|ipv4-udp|ipv6-tcp|ipv6-udp)"
                        " src (src_ip_address) (src_port)"
                        " dst (dst_ip_address) (dst_port)"
+                       " tos (tos_value) ttl (ttl_value)"
                        " vlan (vlan_value) flexbytes (flexbytes_value)"
-                       " (drop|fwd) queue (queue_id) fd_id (fd_id_value)\n"
+                       " (drop|fwd) pf|vf(vf_id) queue (queue_id)"
+                       " fd_id (fd_id_value)\n"
                        "    Add/Del an UDP/TCP type flow director filter.\n\n"
 
-                       "flow_director_filter (port_id) (add|del|update)"
+                       "flow_director_filter (port_id) mode IP (add|del|update)"
                        " flow (ipv4-sctp|ipv6-sctp)"
                        " src (src_ip_address) (src_port)"
                        " dst (dst_ip_address) (dst_port)"
-                       " tag (verification_tag) vlan (vlan_value)"
+                       " tag (verification_tag) "
+                       " tos (tos_value) ttl (ttl_value)"
+                       " vlan (vlan_value)"
                        " flexbytes (flexbytes_value) (drop|fwd)"
-                       " queue (queue_id) fd_id (fd_id_value)\n"
+                       " pf|vf(vf_id) queue (queue_id) fd_id (fd_id_value)\n"
                        "    Add/Del a SCTP type flow director filter.\n\n"
 
-                       "flow_director_filter (port_id) (add|del|update)"
+                       "flow_director_filter (port_id) mode IP (add|del|update)"
                        " flow l2_payload ether (ethertype)"
                        " flexbytes (flexbytes_value) (drop|fwd)"
-                       " queue (queue_id) fd_id (fd_id_value)\n"
+                       " pf|vf(vf_id) queue (queue_id) fd_id (fd_id_value)\n"
                        "    Add/Del a l2 payload type flow director filter.\n\n"
 
+                       "flow_director_filter (port_id) mode MAC-VLAN (add|del|update)"
+                       " mac (mac_address) vlan (vlan_value)"
+                       " flexbytes (flexbytes_value) (drop|fwd)"
+                       " queue (queue_id) fd_id (fd_id_value)\n"
+                       "    Add/Del a MAC-VLAN flow director filter.\n\n"
+
+                       "flow_director_filter (port_id) mode Tunnel (add|del|update)"
+                       " mac (mac_address) vlan (vlan_value)"
+                       " tunnel (NVGRE|VxLAN) tunnel-id (tunnel_id_value)"
+                       " flexbytes (flexbytes_value) (drop|fwd)"
+                       " queue (queue_id) fd_id (fd_id_value)\n"
+                       "    Add/Del a Tunnel flow director filter.\n\n"
+
                        "flush_flow_director (port_id)\n"
                        "    Flush all flow director entries of a device.\n\n"
 
-                       "flow_director_mask (port_id) vlan (vlan_value)"
+                       "flow_director_mask (port_id) mode IP vlan (vlan_value)"
                        " src_mask (ipv4_src) (ipv6_src) (src_port)"
                        " dst_mask (ipv4_dst) (ipv6_dst) (dst_port)\n"
-                       "    Set flow director mask.\n\n"
+                       "    Set flow director IP mask.\n\n"
+
+                       "flow_director_mask (port_id) mode MAC-VLAN"
+                       " vlan (vlan_value)\n"
+                       "    Set flow director MAC-VLAN mask.\n\n"
+
+                       "flow_director_mask (port_id) mode Tunnel"
+                       " vlan (vlan_value) mac (mac_value)"
+                       " tunnel-type (tunnel_type_value)"
+                       " tunnel-id (tunnel_id_value)\n"
+                       "    Set flow director Tunnel mask.\n\n"
 
                        "flow_director_flex_mask (port_id)"
                        " flow (none|ipv4-other|ipv4-frag|ipv4-tcp|ipv4-udp|ipv4-sctp|"
@@ -692,6 +786,29 @@ static void cmd_help_long_parsed(void *parsed_result,
                        "ipv6-frag|ipv6-tcp|ipv6-udp|ipv6-sctp|ipv6-other|l2_payload)"
                        " (enable|disable)\n"
                        "    Set the global configurations of hash filters.\n\n"
+
+                       "set_hash_input_set (port_id) (ipv4|ipv4-frag|"
+                       "ipv4-tcp|ipv4-udp|ipv4-sctp|ipv4-other|ipv6|"
+                       "ipv6-frag|ipv6-tcp|ipv6-udp|ipv6-sctp|ipv6-other|"
+                       "l2_payload) (ovlan|ivlan|src-ipv4|dst-ipv4|src-ipv6|"
+                       "dst-ipv6|ipv4-tos|ipv4-proto|ipv6-tc|"
+                       "ipv6-next-header|udp-src-port|udp-dst-port|"
+                       "tcp-src-port|tcp-dst-port|sctp-src-port|"
+                       "sctp-dst-port|sctp-veri-tag|udp-key|gre-key|fld-1st|"
+                       "fld-2nd|fld-3rd|fld-4th|fld-5th|fld-6th|fld-7th|"
+                       "fld-8th|none) (select|add)\n"
+                       "    Set the input set for hash.\n\n"
+
+                       "set_fdir_input_set (port_id) "
+                       "(ipv4-frag|ipv4-tcp|ipv4-udp|ipv4-sctp|ipv4-other|"
+                       "ipv6-frag|ipv6-tcp|ipv6-udp|ipv6-sctp|ipv6-other|"
+                       "l2_payload) (ivlan|ethertype|src-ipv4|dst-ipv4|src-ipv6|"
+                       "dst-ipv6|ipv4-tos|ipv4-proto|ipv4-ttl|ipv6-tc|"
+                       "ipv6-next-header|ipv6-hop-limits|udp-src-port|"
+                       "udp-dst-port|tcp-src-port|tcp-dst-port|"
+                       "sctp-src-port|sctp-dst-port|sctp-veri-tag|none)"
+                       " (select|add)\n"
+                       "    Set the input set for FDir.\n\n"
                );
        }
 }
@@ -899,14 +1016,64 @@ struct cmd_config_speed_all {
        cmdline_fixed_string_t value2;
 };
 
+static int
+parse_and_check_speed_duplex(char *speedstr, char *duplexstr, uint32_t *speed)
+{
+
+       int duplex;
+
+       if (!strcmp(duplexstr, "half")) {
+               duplex = ETH_LINK_HALF_DUPLEX;
+       } else if (!strcmp(duplexstr, "full")) {
+               duplex = ETH_LINK_FULL_DUPLEX;
+       } else if (!strcmp(duplexstr, "auto")) {
+               duplex = ETH_LINK_FULL_DUPLEX;
+       } else {
+               printf("Unknown duplex parameter\n");
+               return -1;
+       }
+
+       if (!strcmp(speedstr, "10")) {
+               *speed = (duplex == ETH_LINK_HALF_DUPLEX) ?
+                               ETH_LINK_SPEED_10M_HD : ETH_LINK_SPEED_10M;
+       } else if (!strcmp(speedstr, "100")) {
+               *speed = (duplex == ETH_LINK_HALF_DUPLEX) ?
+                               ETH_LINK_SPEED_100M_HD : ETH_LINK_SPEED_100M;
+       } else {
+               if (duplex != ETH_LINK_FULL_DUPLEX) {
+                       printf("Invalid speed/duplex parameters\n");
+                       return -1;
+               }
+               if (!strcmp(speedstr, "1000")) {
+                       *speed = ETH_LINK_SPEED_1G;
+               } else if (!strcmp(speedstr, "10000")) {
+                       *speed = ETH_LINK_SPEED_10G;
+               } else if (!strcmp(speedstr, "25000")) {
+                       *speed = ETH_LINK_SPEED_25G;
+               } else if (!strcmp(speedstr, "40000")) {
+                       *speed = ETH_LINK_SPEED_40G;
+               } else if (!strcmp(speedstr, "50000")) {
+                       *speed = ETH_LINK_SPEED_50G;
+               } else if (!strcmp(speedstr, "100000")) {
+                       *speed = ETH_LINK_SPEED_100G;
+               } else if (!strcmp(speedstr, "auto")) {
+                       *speed = ETH_LINK_SPEED_AUTONEG;
+               } else {
+                       printf("Unknown speed parameter\n");
+                       return -1;
+               }
+       }
+
+       return 0;
+}
+
 static void
 cmd_config_speed_all_parsed(void *parsed_result,
                        __attribute__((unused)) struct cmdline *cl,
                        __attribute__((unused)) void *data)
 {
        struct cmd_config_speed_all *res = parsed_result;
-       uint16_t link_speed = ETH_LINK_SPEED_AUTONEG;
-       uint16_t link_duplex = 0;
+       uint32_t link_speed;
        portid_t pid;
 
        if (!all_ports_stopped()) {
@@ -914,37 +1081,12 @@ cmd_config_speed_all_parsed(void *parsed_result,
                return;
        }
 
-       if (!strcmp(res->value1, "10"))
-               link_speed = ETH_LINK_SPEED_10;
-       else if (!strcmp(res->value1, "100"))
-               link_speed = ETH_LINK_SPEED_100;
-       else if (!strcmp(res->value1, "1000"))
-               link_speed = ETH_LINK_SPEED_1000;
-       else if (!strcmp(res->value1, "10000"))
-               link_speed = ETH_LINK_SPEED_10G;
-       else if (!strcmp(res->value1, "40000"))
-               link_speed = ETH_LINK_SPEED_40G;
-       else if (!strcmp(res->value1, "auto"))
-               link_speed = ETH_LINK_SPEED_AUTONEG;
-       else {
-               printf("Unknown parameter\n");
-               return;
-       }
-
-       if (!strcmp(res->value2, "half"))
-               link_duplex = ETH_LINK_HALF_DUPLEX;
-       else if (!strcmp(res->value2, "full"))
-               link_duplex = ETH_LINK_FULL_DUPLEX;
-       else if (!strcmp(res->value2, "auto"))
-               link_duplex = ETH_LINK_AUTONEG_DUPLEX;
-       else {
-               printf("Unknown parameter\n");
+       if (parse_and_check_speed_duplex(res->value1, res->value2,
+                       &link_speed) < 0)
                return;
-       }
 
        FOREACH_PORT(pid, ports) {
-               ports[pid].dev_conf.link_speed = link_speed;
-               ports[pid].dev_conf.link_duplex = link_duplex;
+               ports[pid].dev_conf.link_speeds = link_speed;
        }
 
        cmd_reconfig_device_queue(RTE_PORT_ALL, 1, 1);
@@ -961,7 +1103,7 @@ cmdline_parse_token_string_t cmd_config_speed_all_item1 =
        TOKEN_STRING_INITIALIZER(struct cmd_config_speed_all, item1, "speed");
 cmdline_parse_token_string_t cmd_config_speed_all_value1 =
        TOKEN_STRING_INITIALIZER(struct cmd_config_speed_all, value1,
-                                               "10#100#1000#10000#40000#auto");
+                               "10#100#1000#10000#25000#40000#50000#100000#auto");
 cmdline_parse_token_string_t cmd_config_speed_all_item2 =
        TOKEN_STRING_INITIALIZER(struct cmd_config_speed_all, item2, "duplex");
 cmdline_parse_token_string_t cmd_config_speed_all_value2 =
@@ -971,7 +1113,8 @@ cmdline_parse_token_string_t cmd_config_speed_all_value2 =
 cmdline_parse_inst_t cmd_config_speed_all = {
        .f = cmd_config_speed_all_parsed,
        .data = NULL,
-       .help_str = "port config all speed 10|100|1000|10000|40000|auto duplex "
+       .help_str = "port config all speed "
+               "10|100|1000|10000|25000|40000|50000|100000|auto duplex "
                                                        "half|full|auto",
        .tokens = {
                (void *)&cmd_config_speed_all_port,
@@ -1002,8 +1145,7 @@ cmd_config_speed_specific_parsed(void *parsed_result,
                                __attribute__((unused)) void *data)
 {
        struct cmd_config_speed_specific *res = parsed_result;
-       uint16_t link_speed = ETH_LINK_SPEED_AUTONEG;
-       uint16_t link_duplex = 0;
+       uint32_t link_speed;
 
        if (!all_ports_stopped()) {
                printf("Please stop all ports first\n");
@@ -1013,36 +1155,11 @@ cmd_config_speed_specific_parsed(void *parsed_result,
        if (port_id_is_invalid(res->id, ENABLED_WARN))
                return;
 
-       if (!strcmp(res->value1, "10"))
-               link_speed = ETH_LINK_SPEED_10;
-       else if (!strcmp(res->value1, "100"))
-               link_speed = ETH_LINK_SPEED_100;
-       else if (!strcmp(res->value1, "1000"))
-               link_speed = ETH_LINK_SPEED_1000;
-       else if (!strcmp(res->value1, "10000"))
-               link_speed = ETH_LINK_SPEED_10000;
-       else if (!strcmp(res->value1, "40000"))
-               link_speed = ETH_LINK_SPEED_40G;
-       else if (!strcmp(res->value1, "auto"))
-               link_speed = ETH_LINK_SPEED_AUTONEG;
-       else {
-               printf("Unknown parameter\n");
-               return;
-       }
-
-       if (!strcmp(res->value2, "half"))
-               link_duplex = ETH_LINK_HALF_DUPLEX;
-       else if (!strcmp(res->value2, "full"))
-               link_duplex = ETH_LINK_FULL_DUPLEX;
-       else if (!strcmp(res->value2, "auto"))
-               link_duplex = ETH_LINK_AUTONEG_DUPLEX;
-       else {
-               printf("Unknown parameter\n");
+       if (parse_and_check_speed_duplex(res->value1, res->value2,
+                       &link_speed) < 0)
                return;
-       }
 
-       ports[res->id].dev_conf.link_speed = link_speed;
-       ports[res->id].dev_conf.link_duplex = link_duplex;
+       ports[res->id].dev_conf.link_speeds = link_speed;
 
        cmd_reconfig_device_queue(RTE_PORT_ALL, 1, 1);
 }
@@ -1061,7 +1178,7 @@ cmdline_parse_token_string_t cmd_config_speed_specific_item1 =
                                                                "speed");
 cmdline_parse_token_string_t cmd_config_speed_specific_value1 =
        TOKEN_STRING_INITIALIZER(struct cmd_config_speed_specific, value1,
-                                               "10#100#1000#10000#40000#auto");
+                               "10#100#1000#10000#25000#40000#50000#100000#auto");
 cmdline_parse_token_string_t cmd_config_speed_specific_item2 =
        TOKEN_STRING_INITIALIZER(struct cmd_config_speed_specific, item2,
                                                                "duplex");
@@ -1072,7 +1189,8 @@ cmdline_parse_token_string_t cmd_config_speed_specific_value2 =
 cmdline_parse_inst_t cmd_config_speed_specific = {
        .f = cmd_config_speed_specific_parsed,
        .data = NULL,
-       .help_str = "port config X speed 10|100|1000|10000|40000|auto duplex "
+       .help_str = "port config X speed "
+               "10|100|1000|10000|25000|40000|50000|100000|auto duplex "
                                                        "half|full|auto",
        .tokens = {
                (void *)&cmd_config_speed_specific_port,
@@ -1106,17 +1224,16 @@ cmd_config_rx_tx_parsed(void *parsed_result,
                printf("Please stop all ports first\n");
                return;
        }
-
        if (!strcmp(res->name, "rxq")) {
-               if (res->value <= 0) {
-                       printf("rxq %d invalid - must be > 0\n", res->value);
+               if (!res->value && !nb_txq) {
+                       printf("Warning: Either rx or tx queues should be non zero\n");
                        return;
                }
                nb_rxq = res->value;
        }
        else if (!strcmp(res->name, "txq")) {
-               if (res->value <= 0) {
-                       printf("txq %d invalid - must be > 0\n", res->value);
+               if (!res->value && !nb_rxq) {
+                       printf("Warning: Either rx or tx queues should be non zero\n");
                        return;
                }
                nb_txq = res->value;
@@ -1140,6 +1257,8 @@ cmd_config_rx_tx_parsed(void *parsed_result,
                return;
        }
 
+       fwd_config_setup();
+
        init_port_config();
 
        cmd_reconfig_device_queue(RTE_PORT_ALL, 1, 1);
@@ -1286,7 +1405,7 @@ cmdline_parse_token_num_t cmd_config_mtu_value =
 cmdline_parse_inst_t cmd_config_mtu = {
        .f = cmd_config_mtu_parsed,
        .data = NULL,
-       .help_str = "port config mtu value",
+       .help_str = "port config mtu port_id value",
        .tokens = {
                (void *)&cmd_config_mtu_port,
                (void *)&cmd_config_mtu_keyword,
@@ -1327,6 +1446,15 @@ cmd_config_rx_mode_flag_parsed(void *parsed_result,
                        printf("Unknown parameter\n");
                        return;
                }
+       } else if (!strcmp(res->name, "scatter")) {
+               if (!strcmp(res->value, "on"))
+                       rx_mode.enable_scatter = 1;
+               else if (!strcmp(res->value, "off"))
+                       rx_mode.enable_scatter = 0;
+               else {
+                       printf("Unknown parameter\n");
+                       return;
+               }
        } else if (!strcmp(res->name, "rx-cksum")) {
                if (!strcmp(res->value, "on"))
                        rx_mode.hw_ip_checksum = 1;
@@ -1404,7 +1532,7 @@ cmdline_parse_token_string_t cmd_config_rx_mode_flag_all =
        TOKEN_STRING_INITIALIZER(struct cmd_config_rx_mode_flag, all, "all");
 cmdline_parse_token_string_t cmd_config_rx_mode_flag_name =
        TOKEN_STRING_INITIALIZER(struct cmd_config_rx_mode_flag, name,
-                                       "crc-strip#rx-cksum#hw-vlan#"
+                                       "crc-strip#scatter#rx-cksum#hw-vlan#"
                                        "hw-vlan-filter#hw-vlan-strip#hw-vlan-extend");
 cmdline_parse_token_string_t cmd_config_rx_mode_flag_value =
        TOKEN_STRING_INITIALIZER(struct cmd_config_rx_mode_flag, value,
@@ -1413,7 +1541,7 @@ cmdline_parse_token_string_t cmd_config_rx_mode_flag_value =
 cmdline_parse_inst_t cmd_config_rx_mode_flag = {
        .f = cmd_config_rx_mode_flag_parsed,
        .data = NULL,
-       .help_str = "port config all crc-strip|rx-cksum|hw-vlan|"
+       .help_str = "port config all crc-strip|scatter|rx-cksum|hw-vlan|"
                "hw-vlan-filter|hw-vlan-strip|hw-vlan-extend on|off",
        .tokens = {
                (void *)&cmd_config_rx_mode_flag_port,
@@ -1441,6 +1569,7 @@ cmd_config_rss_parsed(void *parsed_result,
 {
        struct cmd_config_rss *res = parsed_result;
        struct rte_eth_rss_conf rss_conf;
+       int diag;
        uint8_t i;
 
        if (!strcmp(res->value, "all"))
@@ -1457,6 +1586,14 @@ cmd_config_rss_parsed(void *parsed_result,
                rss_conf.rss_hf = ETH_RSS_SCTP;
        else if (!strcmp(res->value, "ether"))
                rss_conf.rss_hf = ETH_RSS_L2_PAYLOAD;
+       else if (!strcmp(res->value, "port"))
+               rss_conf.rss_hf = ETH_RSS_PORT;
+       else if (!strcmp(res->value, "vxlan"))
+               rss_conf.rss_hf = ETH_RSS_VXLAN;
+       else if (!strcmp(res->value, "geneve"))
+               rss_conf.rss_hf = ETH_RSS_GENEVE;
+       else if (!strcmp(res->value, "nvgre"))
+               rss_conf.rss_hf = ETH_RSS_NVGRE;
        else if (!strcmp(res->value, "none"))
                rss_conf.rss_hf = 0;
        else {
@@ -1464,8 +1601,13 @@ cmd_config_rss_parsed(void *parsed_result,
                return;
        }
        rss_conf.rss_key = NULL;
-       for (i = 0; i < rte_eth_dev_count(); i++)
-               rte_eth_dev_rss_hash_update(i, &rss_conf);
+       for (i = 0; i < rte_eth_dev_count(); i++) {
+               diag = rte_eth_dev_rss_hash_update(i, &rss_conf);
+               if (diag < 0)
+                       printf("Configuration of RSS hash at ethernet port %d "
+                               "failed with error (%d): %s.\n",
+                               i, -diag, strerror(-diag));
+       }
 }
 
 cmdline_parse_token_string_t cmd_config_rss_port =
@@ -1478,12 +1620,12 @@ cmdline_parse_token_string_t cmd_config_rss_name =
        TOKEN_STRING_INITIALIZER(struct cmd_config_rss, name, "rss");
 cmdline_parse_token_string_t cmd_config_rss_value =
        TOKEN_STRING_INITIALIZER(struct cmd_config_rss, value,
-               "all#ip#tcp#udp#sctp#ether#none");
+               "all#ip#tcp#udp#sctp#ether#port#vxlan#geneve#nvgre#none");
 
 cmdline_parse_inst_t cmd_config_rss = {
        .f = cmd_config_rss_parsed,
        .data = NULL,
-       .help_str = "port config all rss all|ip|tcp|udp|sctp|ether|none",
+       .help_str = "port config all rss all|ip|tcp|udp|sctp|ether|port|vxlan|geneve|nvgre|none",
        .tokens = {
                (void *)&cmd_config_rss_port,
                (void *)&cmd_config_rss_keyword,
@@ -1504,7 +1646,6 @@ struct cmd_config_rss_hash_key {
        cmdline_fixed_string_t key;
 };
 
-#define RSS_HASH_KEY_LENGTH 40
 static uint8_t
 hexa_digit_to_value(char hexa_digit)
 {
@@ -1540,16 +1681,29 @@ cmd_config_rss_hash_key_parsed(void *parsed_result,
        uint8_t xdgt0;
        uint8_t xdgt1;
        int i;
+       struct rte_eth_dev_info dev_info;
+       uint8_t hash_key_size;
+       uint32_t key_len;
 
+       memset(&dev_info, 0, sizeof(dev_info));
+       rte_eth_dev_info_get(res->port_id, &dev_info);
+       if (dev_info.hash_key_size > 0 &&
+                       dev_info.hash_key_size <= sizeof(hash_key))
+               hash_key_size = dev_info.hash_key_size;
+       else {
+               printf("dev_info did not provide a valid hash key size\n");
+               return;
+       }
        /* Check the length of the RSS hash key */
-       if (strlen(res->key) != (RSS_HASH_KEY_LENGTH * 2)) {
+       key_len = strlen(res->key);
+       if (key_len != (hash_key_size * 2)) {
                printf("key length: %d invalid - key must be a string of %d"
-                      "hexa-decimal numbers\n", (int) strlen(res->key),
-                      RSS_HASH_KEY_LENGTH * 2);
+                          " hexa-decimal numbers\n",
+                          (int) key_len, hash_key_size * 2);
                return;
        }
        /* Translate RSS hash key into binary representation */
-       for (i = 0; i < RSS_HASH_KEY_LENGTH; i++) {
+       for (i = 0; i < hash_key_size; i++) {
                xdgt0 = parse_and_check_key_hexa_digit(res->key, (i * 2));
                if (xdgt0 == 0xFF)
                        return;
@@ -1559,7 +1713,7 @@ cmd_config_rss_hash_key_parsed(void *parsed_result,
                hash_key[i] = (uint8_t) ((xdgt0 * 16) + xdgt1);
        }
        port_rss_hash_key_update(res->port_id, res->rss_type, hash_key,
-                                RSS_HASH_KEY_LENGTH);
+                       hash_key_size);
 }
 
 cmdline_parse_token_string_t cmd_config_rss_hash_key_port =
@@ -1588,7 +1742,8 @@ cmdline_parse_inst_t cmd_config_rss_hash_key = {
                "port config X rss-hash-key ipv4|ipv4-frag|ipv4-tcp|ipv4-udp|"
                "ipv4-sctp|ipv4-other|ipv6|ipv6-frag|ipv6-tcp|ipv6-udp|"
                "ipv6-sctp|ipv6-other|l2-payload|"
-               "ipv6-ex|ipv6-tcp-ex|ipv6-udp-ex 80 hexa digits\n",
+               "ipv6-ex|ipv6-tcp-ex|ipv6-udp-ex "
+               "<string of hexa digits (variable length, NIC dependent)>\n",
        .tokens = {
                (void *)&cmd_config_rss_hash_key_port,
                (void *)&cmd_config_rss_hash_key_config,
@@ -1712,7 +1867,7 @@ parse_reta_config(const char *str,
        int i;
        unsigned size;
        uint16_t hash_index, idx, shift;
-       uint8_t nb_queue;
+       uint16_t nb_queue;
        char s[256];
        const char *p, *p0 = str;
        char *end;
@@ -1745,7 +1900,7 @@ parse_reta_config(const char *str,
                }
 
                hash_index = (uint16_t)int_fld[FLD_HASH_INDEX];
-               nb_queue = (uint8_t)int_fld[FLD_QUEUE];
+               nb_queue = (uint16_t)int_fld[FLD_QUEUE];
 
                if (hash_index >= nb_entries) {
                        printf("Invalid RETA hash index=%d\n", hash_index);
@@ -2363,11 +2518,11 @@ parse_item_list(char* str, const char* item_name, unsigned int max_items,
                }
                if (c != ',') {
                        printf("character %c is not a decimal digit\n", c);
-                       return (0);
+                       return 0;
                }
                if (! value_ok) {
                        printf("No valid value before comma\n");
-                       return (0);
+                       return 0;
                }
                if (nb_item < max_items) {
                        parsed_items[nb_item] = value;
@@ -2379,11 +2534,11 @@ parse_item_list(char* str, const char* item_name, unsigned int max_items,
        if (nb_item >= max_items) {
                printf("Number of %s = %u > %u (maximum items)\n",
                       item_name, nb_item + 1, max_items);
-               return (0);
+               return 0;
        }
        parsed_items[nb_item++] = value;
        if (! check_unique_values)
-               return (nb_item);
+               return nb_item;
 
        /*
         * Then, check that all values in the list are differents.
@@ -2394,11 +2549,11 @@ parse_item_list(char* str, const char* item_name, unsigned int max_items,
                        if (parsed_items[j] == parsed_items[i]) {
                                printf("duplicated %s %u at index %u and %u\n",
                                       item_name, parsed_items[i], i, j);
-                               return (0);
+                               return 0;
                        }
                }
        }
-       return (nb_item);
+       return nb_item;
 }
 
 struct cmd_set_list_result {
@@ -2428,16 +2583,20 @@ static void cmd_set_list_parsed(void *parsed_result,
                nb_item = parse_item_list(res->list_of_items, "core",
                                          RTE_MAX_LCORE,
                                          parsed_items.lcorelist, 1);
-               if (nb_item > 0)
+               if (nb_item > 0) {
                        set_fwd_lcores_list(parsed_items.lcorelist, nb_item);
+                       fwd_config_setup();
+               }
                return;
        }
        if (!strcmp(res->list_name, "portlist")) {
                nb_item = parse_item_list(res->list_of_items, "port",
                                          RTE_MAX_ETHPORTS,
                                          parsed_items.portlist, 1);
-               if (nb_item > 0)
+               if (nb_item > 0) {
                        set_fwd_ports_list(parsed_items.portlist, nb_item);
+                       fwd_config_setup();
+               }
        }
 }
 
@@ -2481,10 +2640,13 @@ static void cmd_set_mask_parsed(void *parsed_result,
                printf("Please stop forwarding first\n");
                return;
        }
-       if (!strcmp(res->mask, "coremask"))
+       if (!strcmp(res->mask, "coremask")) {
                set_fwd_lcores_mask(res->hexavalue);
-       else if (!strcmp(res->mask, "portmask"))
+               fwd_config_setup();
+       } else if (!strcmp(res->mask, "portmask")) {
                set_fwd_ports_mask(res->hexavalue);
+               fwd_config_setup();
+       }
 }
 
 cmdline_parse_token_string_t cmd_setmask_set =
@@ -2521,11 +2683,13 @@ static void cmd_set_parsed(void *parsed_result,
                           __attribute__((unused)) void *data)
 {
        struct cmd_set_result *res = parsed_result;
-       if (!strcmp(res->what, "nbport"))
+       if (!strcmp(res->what, "nbport")) {
                set_fwd_ports_number(res->value);
-       else if (!strcmp(res->what, "nbcore"))
+               fwd_config_setup();
+       } else if (!strcmp(res->what, "nbcore")) {
                set_fwd_lcores_number(res->value);
-       else if (!strcmp(res->what, "burst"))
+               fwd_config_setup();
+       } else if (!strcmp(res->what, "burst"))
                set_nb_pkt_per_burst(res->value);
        else if (!strcmp(res->what, "verbose"))
                set_verbose_level(res->value);
@@ -2597,6 +2761,115 @@ cmdline_parse_inst_t cmd_set_txpkts = {
        },
 };
 
+/* *** SET COPY AND SPLIT POLICY ON TX PACKETS *** */
+
+struct cmd_set_txsplit_result {
+       cmdline_fixed_string_t cmd_keyword;
+       cmdline_fixed_string_t txsplit;
+       cmdline_fixed_string_t mode;
+};
+
+static void
+cmd_set_txsplit_parsed(void *parsed_result,
+                     __attribute__((unused)) struct cmdline *cl,
+                     __attribute__((unused)) void *data)
+{
+       struct cmd_set_txsplit_result *res;
+
+       res = parsed_result;
+       set_tx_pkt_split(res->mode);
+}
+
+cmdline_parse_token_string_t cmd_set_txsplit_keyword =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_txsplit_result,
+                                cmd_keyword, "set");
+cmdline_parse_token_string_t cmd_set_txsplit_name =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_txsplit_result,
+                                txsplit, "txsplit");
+cmdline_parse_token_string_t cmd_set_txsplit_mode =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_txsplit_result,
+                                mode, NULL);
+
+cmdline_parse_inst_t cmd_set_txsplit = {
+       .f = cmd_set_txsplit_parsed,
+       .data = NULL,
+       .help_str = "set txsplit on|off|rand",
+       .tokens = {
+               (void *)&cmd_set_txsplit_keyword,
+               (void *)&cmd_set_txsplit_name,
+               (void *)&cmd_set_txsplit_mode,
+               NULL,
+       },
+};
+
+/* *** CONFIG TX QUEUE FLAGS *** */
+
+struct cmd_config_txqflags_result {
+       cmdline_fixed_string_t port;
+       cmdline_fixed_string_t config;
+       cmdline_fixed_string_t all;
+       cmdline_fixed_string_t what;
+       int32_t hexvalue;
+};
+
+static void cmd_config_txqflags_parsed(void *parsed_result,
+                               __attribute__((unused)) struct cmdline *cl,
+                               __attribute__((unused)) void *data)
+{
+       struct cmd_config_txqflags_result *res = parsed_result;
+
+       if (!all_ports_stopped()) {
+               printf("Please stop all ports first\n");
+               return;
+       }
+
+       if (strcmp(res->what, "txqflags")) {
+               printf("Unknown parameter\n");
+               return;
+       }
+
+       if (res->hexvalue >= 0) {
+               txq_flags = res->hexvalue;
+       } else {
+               printf("txqflags must be >= 0\n");
+               return;
+       }
+
+       init_port_config();
+
+       cmd_reconfig_device_queue(RTE_PORT_ALL, 1, 1);
+}
+
+cmdline_parse_token_string_t cmd_config_txqflags_port =
+       TOKEN_STRING_INITIALIZER(struct cmd_config_txqflags_result, port,
+                                "port");
+cmdline_parse_token_string_t cmd_config_txqflags_config =
+       TOKEN_STRING_INITIALIZER(struct cmd_config_txqflags_result, config,
+                                "config");
+cmdline_parse_token_string_t cmd_config_txqflags_all =
+       TOKEN_STRING_INITIALIZER(struct cmd_config_txqflags_result, all,
+                                "all");
+cmdline_parse_token_string_t cmd_config_txqflags_what =
+       TOKEN_STRING_INITIALIZER(struct cmd_config_txqflags_result, what,
+                                "txqflags");
+cmdline_parse_token_num_t cmd_config_txqflags_value =
+       TOKEN_NUM_INITIALIZER(struct cmd_config_txqflags_result,
+                               hexvalue, INT32);
+
+cmdline_parse_inst_t cmd_config_txqflags = {
+       .f = cmd_config_txqflags_parsed,
+       .data = NULL,
+       .help_str = "port config all txqflags value",
+       .tokens = {
+               (void *)&cmd_config_txqflags_port,
+               (void *)&cmd_config_txqflags_config,
+               (void *)&cmd_config_txqflags_all,
+               (void *)&cmd_config_txqflags_what,
+               (void *)&cmd_config_txqflags_value,
+               NULL,
+       },
+};
+
 /* *** ADD/REMOVE ALL VLAN IDENTIFIERS TO/FROM A PORT VLAN RX FILTER *** */
 struct cmd_rx_vlan_filter_all_result {
        cmdline_fixed_string_t rx_vlan;
@@ -2649,6 +2922,7 @@ cmdline_parse_inst_t cmd_rx_vlan_filter_all = {
 struct cmd_vlan_offload_result {
        cmdline_fixed_string_t vlan;
        cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vlan_type;
        cmdline_fixed_string_t what;
        cmdline_fixed_string_t on;
        cmdline_fixed_string_t port_id;
@@ -2749,6 +3023,7 @@ cmdline_parse_inst_t cmd_vlan_offload = {
 struct cmd_vlan_tpid_result {
        cmdline_fixed_string_t vlan;
        cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vlan_type;
        cmdline_fixed_string_t what;
        uint16_t tp_id;
        uint8_t port_id;
@@ -2760,8 +3035,17 @@ cmd_vlan_tpid_parsed(void *parsed_result,
                          __attribute__((unused)) void *data)
 {
        struct cmd_vlan_tpid_result *res = parsed_result;
-       vlan_tpid_set(res->port_id, res->tp_id);
-       return;
+       enum rte_vlan_type vlan_type;
+
+       if (!strcmp(res->vlan_type, "inner"))
+               vlan_type = ETH_VLAN_TYPE_INNER;
+       else if (!strcmp(res->vlan_type, "outer"))
+               vlan_type = ETH_VLAN_TYPE_OUTER;
+       else {
+               printf("Unknown vlan type\n");
+               return;
+       }
+       vlan_tpid_set(res->port_id, vlan_type, res->tp_id);
 }
 
 cmdline_parse_token_string_t cmd_vlan_tpid_vlan =
@@ -2770,6 +3054,9 @@ cmdline_parse_token_string_t cmd_vlan_tpid_vlan =
 cmdline_parse_token_string_t cmd_vlan_tpid_set =
        TOKEN_STRING_INITIALIZER(struct cmd_vlan_tpid_result,
                                 set, "set");
+cmdline_parse_token_string_t cmd_vlan_type =
+       TOKEN_STRING_INITIALIZER(struct cmd_vlan_tpid_result,
+                                vlan_type, "inner#outer");
 cmdline_parse_token_string_t cmd_vlan_tpid_what =
        TOKEN_STRING_INITIALIZER(struct cmd_vlan_tpid_result,
                                 what, "tpid");
@@ -2783,10 +3070,12 @@ cmdline_parse_token_num_t cmd_vlan_tpid_portid =
 cmdline_parse_inst_t cmd_vlan_tpid = {
        .f = cmd_vlan_tpid_parsed,
        .data = NULL,
-       .help_str = "set tpid tp_id port_id, set the Outer VLAN Ether type",
+       .help_str = "set inner|outer tpid tp_id port_id, set the VLAN "
+                   "Ether type",
        .tokens = {
                (void *)&cmd_vlan_tpid_vlan,
                (void *)&cmd_vlan_tpid_set,
+               (void *)&cmd_vlan_type,
                (void *)&cmd_vlan_tpid_what,
                (void *)&cmd_vlan_tpid_tpid,
                (void *)&cmd_vlan_tpid_portid,
@@ -2856,12 +3145,6 @@ cmd_tx_vlan_set_parsed(void *parsed_result,
                       __attribute__((unused)) void *data)
 {
        struct cmd_tx_vlan_set_result *res = parsed_result;
-       int vlan_offload = rte_eth_dev_get_vlan_offload(res->port_id);
-
-       if (vlan_offload & ETH_VLAN_EXTEND_OFFLOAD) {
-               printf("Error, as QinQ has been enabled.\n");
-               return;
-       }
 
        tx_vlan_set(res->port_id, res->vlan_id);
 }
@@ -2872,12 +3155,12 @@ cmdline_parse_token_string_t cmd_tx_vlan_set_tx_vlan =
 cmdline_parse_token_string_t cmd_tx_vlan_set_set =
        TOKEN_STRING_INITIALIZER(struct cmd_tx_vlan_set_result,
                                 set, "set");
-cmdline_parse_token_num_t cmd_tx_vlan_set_vlanid =
-       TOKEN_NUM_INITIALIZER(struct cmd_tx_vlan_set_result,
-                             vlan_id, UINT16);
 cmdline_parse_token_num_t cmd_tx_vlan_set_portid =
        TOKEN_NUM_INITIALIZER(struct cmd_tx_vlan_set_result,
                              port_id, UINT8);
+cmdline_parse_token_num_t cmd_tx_vlan_set_vlanid =
+       TOKEN_NUM_INITIALIZER(struct cmd_tx_vlan_set_result,
+                             vlan_id, UINT16);
 
 cmdline_parse_inst_t cmd_tx_vlan_set = {
        .f = cmd_tx_vlan_set_parsed,
@@ -2908,12 +3191,6 @@ cmd_tx_vlan_set_qinq_parsed(void *parsed_result,
                            __attribute__((unused)) void *data)
 {
        struct cmd_tx_vlan_set_qinq_result *res = parsed_result;
-       int vlan_offload = rte_eth_dev_get_vlan_offload(res->port_id);
-
-       if (!(vlan_offload & ETH_VLAN_EXTEND_OFFLOAD)) {
-               printf("Error, as QinQ hasn't been enabled.\n");
-               return;
-       }
 
        tx_qinq_set(res->port_id, res->vlan_id, res->vlan_id_outer);
 }
@@ -3246,7 +3523,7 @@ cmdline_parse_inst_t cmd_csum_tunnel = {
        },
 };
 
-/* *** ENABLE HARDWARE SEGMENTATION IN TX PACKETS *** */
+/* *** ENABLE HARDWARE SEGMENTATION IN TX NON-TUNNELED PACKETS *** */
 struct cmd_tso_set_result {
        cmdline_fixed_string_t tso;
        cmdline_fixed_string_t mode;
@@ -3269,9 +3546,9 @@ cmd_tso_set_parsed(void *parsed_result,
                ports[res->port_id].tso_segsz = res->tso_segsz;
 
        if (ports[res->port_id].tso_segsz == 0)
-               printf("TSO is disabled\n");
+               printf("TSO for non-tunneled packets is disabled\n");
        else
-               printf("TSO segment size is %d\n",
+               printf("TSO segment size for non-tunneled packets is %d\n",
                        ports[res->port_id].tso_segsz);
 
        /* display warnings if configuration is not supported by the NIC */
@@ -3299,8 +3576,8 @@ cmdline_parse_token_num_t cmd_tso_set_portid =
 cmdline_parse_inst_t cmd_tso_set = {
        .f = cmd_tso_set_parsed,
        .data = NULL,
-       .help_str = "Set TSO segment size for csum engine (0 to disable): "
-       "tso set <tso_segsz> <port>",
+       .help_str = "Set TSO segment size of non-tunneled packets "
+       "for csum engine (0 to disable): tso set <tso_segsz> <port>",
        .tokens = {
                (void *)&cmd_tso_set_tso,
                (void *)&cmd_tso_set_mode,
@@ -3318,8 +3595,8 @@ cmdline_parse_token_string_t cmd_tso_show_mode =
 cmdline_parse_inst_t cmd_tso_show = {
        .f = cmd_tso_set_parsed,
        .data = NULL,
-       .help_str = "Show TSO segment size for csum engine: "
-       "tso show <port>",
+       .help_str = "Show TSO segment size of non-tunneled packets "
+       "for csum engine: tso show <port>",
        .tokens = {
                (void *)&cmd_tso_set_tso,
                (void *)&cmd_tso_show_mode,
@@ -3328,6 +3605,122 @@ cmdline_parse_inst_t cmd_tso_show = {
        },
 };
 
+/* *** ENABLE HARDWARE SEGMENTATION IN TX TUNNELED PACKETS *** */
+struct cmd_tunnel_tso_set_result {
+       cmdline_fixed_string_t tso;
+       cmdline_fixed_string_t mode;
+       uint16_t tso_segsz;
+       uint8_t port_id;
+};
+
+static void
+check_tunnel_tso_nic_support(uint8_t port_id)
+{
+       struct rte_eth_dev_info dev_info;
+
+       rte_eth_dev_info_get(port_id, &dev_info);
+       if (!(dev_info.tx_offload_capa & DEV_TX_OFFLOAD_VXLAN_TNL_TSO))
+               printf("Warning: TSO enabled but VXLAN TUNNEL TSO not "
+                      "supported by port %d\n", port_id);
+       if (!(dev_info.tx_offload_capa & DEV_TX_OFFLOAD_GRE_TNL_TSO))
+               printf("Warning: TSO enabled but GRE TUNNEL TSO not "
+                       "supported by port %d\n", port_id);
+       if (!(dev_info.tx_offload_capa & DEV_TX_OFFLOAD_IPIP_TNL_TSO))
+               printf("Warning: TSO enabled but IPIP TUNNEL TSO not "
+                      "supported by port %d\n", port_id);
+       if (!(dev_info.tx_offload_capa & DEV_TX_OFFLOAD_GENEVE_TNL_TSO))
+               printf("Warning: TSO enabled but GENEVE TUNNEL TSO not "
+                      "supported by port %d\n", port_id);
+}
+
+static void
+cmd_tunnel_tso_set_parsed(void *parsed_result,
+                         __attribute__((unused)) struct cmdline *cl,
+                         __attribute__((unused)) void *data)
+{
+       struct cmd_tunnel_tso_set_result *res = parsed_result;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       if (!strcmp(res->mode, "set"))
+               ports[res->port_id].tunnel_tso_segsz = res->tso_segsz;
+
+       if (ports[res->port_id].tunnel_tso_segsz == 0)
+               printf("TSO for tunneled packets is disabled\n");
+       else {
+               printf("TSO segment size for tunneled packets is %d\n",
+                       ports[res->port_id].tunnel_tso_segsz);
+
+               /* Below conditions are needed to make it work:
+                * (1) tunnel TSO is supported by the NIC;
+                * (2) "csum parse_tunnel" must be set so that tunneled pkts
+                * are recognized;
+                * (3) for tunneled pkts with outer L3 of IPv4,
+                * "csum set outer-ip" must be set to hw, because after tso,
+                * total_len of outer IP header is changed, and the checksum
+                * of outer IP header calculated by sw should be wrong; that
+                * is not necessary for IPv6 tunneled pkts because there's no
+                * checksum in IP header anymore.
+                */
+               check_tunnel_tso_nic_support(res->port_id);
+
+               if (!(ports[res->port_id].tx_ol_flags &
+                     TESTPMD_TX_OFFLOAD_PARSE_TUNNEL))
+                       printf("Warning: csum parse_tunnel must be set "
+                               "so that tunneled packets are recognized\n");
+               if (!(ports[res->port_id].tx_ol_flags &
+                     TESTPMD_TX_OFFLOAD_OUTER_IP_CKSUM))
+                       printf("Warning: csum set outer-ip must be set to hw "
+                               "if outer L3 is IPv4; not necessary for IPv6\n");
+       }
+}
+
+cmdline_parse_token_string_t cmd_tunnel_tso_set_tso =
+       TOKEN_STRING_INITIALIZER(struct cmd_tunnel_tso_set_result,
+                               tso, "tunnel_tso");
+cmdline_parse_token_string_t cmd_tunnel_tso_set_mode =
+       TOKEN_STRING_INITIALIZER(struct cmd_tunnel_tso_set_result,
+                               mode, "set");
+cmdline_parse_token_num_t cmd_tunnel_tso_set_tso_segsz =
+       TOKEN_NUM_INITIALIZER(struct cmd_tunnel_tso_set_result,
+                               tso_segsz, UINT16);
+cmdline_parse_token_num_t cmd_tunnel_tso_set_portid =
+       TOKEN_NUM_INITIALIZER(struct cmd_tunnel_tso_set_result,
+                               port_id, UINT8);
+
+cmdline_parse_inst_t cmd_tunnel_tso_set = {
+       .f = cmd_tunnel_tso_set_parsed,
+       .data = NULL,
+       .help_str = "Set TSO segment size of tunneled packets for csum engine "
+       "(0 to disable): tunnel_tso set <tso_segsz> <port>",
+       .tokens = {
+               (void *)&cmd_tunnel_tso_set_tso,
+               (void *)&cmd_tunnel_tso_set_mode,
+               (void *)&cmd_tunnel_tso_set_tso_segsz,
+               (void *)&cmd_tunnel_tso_set_portid,
+               NULL,
+       },
+};
+
+cmdline_parse_token_string_t cmd_tunnel_tso_show_mode =
+       TOKEN_STRING_INITIALIZER(struct cmd_tunnel_tso_set_result,
+                               mode, "show");
+
+
+cmdline_parse_inst_t cmd_tunnel_tso_show = {
+       .f = cmd_tunnel_tso_set_parsed,
+       .data = NULL,
+       .help_str = "Show TSO segment size of tunneled packets "
+       "for csum engine: tunnel_tso show <port>",
+       .tokens = {
+               (void *)&cmd_tunnel_tso_set_tso,
+               (void *)&cmd_tunnel_tso_show_mode,
+               (void *)&cmd_tunnel_tso_set_portid,
+               NULL,
+       },
+};
+
 /* *** ENABLE/DISABLE FLUSH ON RX STREAMS *** */
 struct cmd_set_flush_rx {
        cmdline_fixed_string_t set;
@@ -3426,9 +3819,6 @@ cmd_set_bypass_mode_parsed(void *parsed_result,
        portid_t port_id = res->port_id;
        uint32_t bypass_mode = RTE_BYPASS_MODE_NORMAL;
 
-       if (!bypass_is_supported(port_id))
-               return;
-
        if (!strcmp(res->value, "bypass"))
                bypass_mode = RTE_BYPASS_MODE_BYPASS;
        else if (!strcmp(res->value, "isolate"))
@@ -3495,9 +3885,6 @@ cmd_set_bypass_event_parsed(void *parsed_result,
        uint32_t bypass_event = RTE_BYPASS_EVENT_NONE;
        uint32_t bypass_mode = RTE_BYPASS_MODE_NORMAL;
 
-       if (!bypass_is_supported(port_id))
-               return;
-
        if (!strcmp(res->event_value, "timeout"))
                bypass_event = RTE_BYPASS_EVENT_TIMEOUT;
        else if (!strcmp(res->event_value, "os_on"))
@@ -3673,9 +4060,6 @@ cmd_show_bypass_config_parsed(void *parsed_result,
                "timeout"};
        int num_events = (sizeof events) / (sizeof events[0]);
 
-       if (!bypass_is_supported(port_id))
-               return;
-
        /* Display the bypass mode.*/
        if (0 != rte_eth_dev_bypass_state_show(port_id, &bypass_mode)) {
                printf("\tFailed to get bypass mode for port = %d\n", port_id);
@@ -4174,7 +4558,7 @@ static void cmd_create_bonded_device_parsed(void *parsed_result,
                return;
        }
 
-       snprintf(ethdev_name, RTE_ETH_NAME_MAX_LEN, "eth_bond_testpmd_%d",
+       snprintf(ethdev_name, RTE_ETH_NAME_MAX_LEN, "net_bond_testpmd_%d",
                        bond_dev_num++);
 
        /* Create a new bonded device. */
@@ -4352,6 +4736,7 @@ static void cmd_set_fwd_mode_parsed(void *parsed_result,
 {
        struct cmd_set_fwd_mode_result *res = parsed_result;
 
+       retry_enabled = 0;
        set_pkt_forwarding_mode(res->mode);
 }
 
@@ -4397,40 +4782,108 @@ static void cmd_set_fwd_mode_init(void)
        token_struct->string_data.str = token;
 }
 
-/* *** SET BURST TX DELAY TIME RETRY NUMBER *** */
-struct cmd_set_burst_tx_retry_result {
+/* *** SET RETRY FORWARDING MODE *** */
+struct cmd_set_fwd_retry_mode_result {
        cmdline_fixed_string_t set;
-       cmdline_fixed_string_t burst;
-       cmdline_fixed_string_t tx;
-       cmdline_fixed_string_t delay;
-       uint32_t time;
+       cmdline_fixed_string_t fwd;
+       cmdline_fixed_string_t mode;
        cmdline_fixed_string_t retry;
-       uint32_t retry_num;
 };
 
-static void cmd_set_burst_tx_retry_parsed(void *parsed_result,
-                                       __attribute__((unused)) struct cmdline *cl,
-                                       __attribute__((unused)) void *data)
+static void cmd_set_fwd_retry_mode_parsed(void *parsed_result,
+                           __attribute__((unused)) struct cmdline *cl,
+                           __attribute__((unused)) void *data)
 {
-       struct cmd_set_burst_tx_retry_result *res = parsed_result;
-
-       if (!strcmp(res->set, "set") && !strcmp(res->burst, "burst")
-               && !strcmp(res->tx, "tx")) {
-               if (!strcmp(res->delay, "delay"))
-                       burst_tx_delay_time = res->time;
-               if (!strcmp(res->retry, "retry"))
-                       burst_tx_retry_num = res->retry_num;
-       }
+       struct cmd_set_fwd_retry_mode_result *res = parsed_result;
 
+       retry_enabled = 1;
+       set_pkt_forwarding_mode(res->mode);
 }
 
-cmdline_parse_token_string_t cmd_set_burst_tx_retry_set =
-       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, set, "set");
-cmdline_parse_token_string_t cmd_set_burst_tx_retry_burst =
-       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, burst,
-                                "burst");
-cmdline_parse_token_string_t cmd_set_burst_tx_retry_tx =
-       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, tx, "tx");
+cmdline_parse_token_string_t cmd_setfwd_retry_set =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fwd_retry_mode_result,
+                       set, "set");
+cmdline_parse_token_string_t cmd_setfwd_retry_fwd =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fwd_retry_mode_result,
+                       fwd, "fwd");
+cmdline_parse_token_string_t cmd_setfwd_retry_mode =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fwd_retry_mode_result,
+                       mode,
+               "" /* defined at init */);
+cmdline_parse_token_string_t cmd_setfwd_retry_retry =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fwd_retry_mode_result,
+                       retry, "retry");
+
+cmdline_parse_inst_t cmd_set_fwd_retry_mode = {
+       .f = cmd_set_fwd_retry_mode_parsed,
+       .data = NULL,
+       .help_str = NULL, /* defined at init */
+       .tokens = {
+               (void *)&cmd_setfwd_retry_set,
+               (void *)&cmd_setfwd_retry_fwd,
+               (void *)&cmd_setfwd_retry_mode,
+               (void *)&cmd_setfwd_retry_retry,
+               NULL,
+       },
+};
+
+static void cmd_set_fwd_retry_mode_init(void)
+{
+       char *modes, *c;
+       static char token[128];
+       static char help[256];
+       cmdline_parse_token_string_t *token_struct;
+
+       modes = list_pkt_forwarding_retry_modes();
+       snprintf(help, sizeof(help), "set fwd %s retry - "
+               "set packet forwarding mode with retry", modes);
+       cmd_set_fwd_retry_mode.help_str = help;
+
+       /* string token separator is # */
+       for (c = token; *modes != '\0'; modes++)
+               if (*modes == '|')
+                       *c++ = '#';
+               else
+                       *c++ = *modes;
+       token_struct = (cmdline_parse_token_string_t *)
+               cmd_set_fwd_retry_mode.tokens[2];
+       token_struct->string_data.str = token;
+}
+
+/* *** SET BURST TX DELAY TIME RETRY NUMBER *** */
+struct cmd_set_burst_tx_retry_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t burst;
+       cmdline_fixed_string_t tx;
+       cmdline_fixed_string_t delay;
+       uint32_t time;
+       cmdline_fixed_string_t retry;
+       uint32_t retry_num;
+};
+
+static void cmd_set_burst_tx_retry_parsed(void *parsed_result,
+                                       __attribute__((unused)) struct cmdline *cl,
+                                       __attribute__((unused)) void *data)
+{
+       struct cmd_set_burst_tx_retry_result *res = parsed_result;
+
+       if (!strcmp(res->set, "set") && !strcmp(res->burst, "burst")
+               && !strcmp(res->tx, "tx")) {
+               if (!strcmp(res->delay, "delay"))
+                       burst_tx_delay_time = res->time;
+               if (!strcmp(res->retry, "retry"))
+                       burst_tx_retry_num = res->retry_num;
+       }
+
+}
+
+cmdline_parse_token_string_t cmd_set_burst_tx_retry_set =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, set, "set");
+cmdline_parse_token_string_t cmd_set_burst_tx_retry_burst =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, burst,
+                                "burst");
+cmdline_parse_token_string_t cmd_set_burst_tx_retry_tx =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, tx, "tx");
 cmdline_parse_token_string_t cmd_set_burst_tx_retry_delay =
        TOKEN_STRING_INITIALIZER(struct cmd_set_burst_tx_retry_result, delay, "delay");
 cmdline_parse_token_num_t cmd_set_burst_tx_retry_time =
@@ -5112,6 +5565,46 @@ cmdline_parse_inst_t cmd_start_tx_first = {
        },
 };
 
+/* *** START FORWARDING WITH N TX BURST FIRST *** */
+struct cmd_start_tx_first_n_result {
+       cmdline_fixed_string_t start;
+       cmdline_fixed_string_t tx_first;
+       uint32_t tx_num;
+};
+
+static void
+cmd_start_tx_first_n_parsed(void *parsed_result,
+                         __attribute__((unused)) struct cmdline *cl,
+                         __attribute__((unused)) void *data)
+{
+       struct cmd_start_tx_first_n_result *res = parsed_result;
+
+       start_packet_forwarding(res->tx_num);
+}
+
+cmdline_parse_token_string_t cmd_start_tx_first_n_start =
+       TOKEN_STRING_INITIALIZER(struct cmd_start_tx_first_n_result,
+                       start, "start");
+cmdline_parse_token_string_t cmd_start_tx_first_n_tx_first =
+       TOKEN_STRING_INITIALIZER(struct cmd_start_tx_first_n_result,
+                       tx_first, "tx_first");
+cmdline_parse_token_num_t cmd_start_tx_first_n_tx_num =
+       TOKEN_NUM_INITIALIZER(struct cmd_start_tx_first_n_result,
+                       tx_num, UINT32);
+
+cmdline_parse_inst_t cmd_start_tx_first_n = {
+       .f = cmd_start_tx_first_n_parsed,
+       .data = NULL,
+       .help_str = "start packet forwarding, after sending <num> "
+               "bursts of packets",
+       .tokens = {
+               (void *)&cmd_start_tx_first_n_start,
+               (void *)&cmd_start_tx_first_n_tx_first,
+               (void *)&cmd_start_tx_first_n_tx_num,
+               NULL,
+       },
+};
+
 /* *** SET LINK UP *** */
 struct cmd_set_link_up_result {
        cmdline_fixed_string_t set;
@@ -5208,7 +5701,9 @@ static void cmd_showcfg_parsed(void *parsed_result,
        else if (!strcmp(res->what, "cores"))
                fwd_lcores_config_display();
        else if (!strcmp(res->what, "fwd"))
-               fwd_config_display();
+               pkt_fwd_config_display(&cur_fwd_config);
+       else if (!strcmp(res->what, "txpkts"))
+               show_tx_pkt_segments();
 }
 
 cmdline_parse_token_string_t cmd_showcfg_show =
@@ -5217,12 +5712,12 @@ cmdline_parse_token_string_t cmd_showcfg_port =
        TOKEN_STRING_INITIALIZER(struct cmd_showcfg_result, cfg, "config");
 cmdline_parse_token_string_t cmd_showcfg_what =
        TOKEN_STRING_INITIALIZER(struct cmd_showcfg_result, what,
-                                "rxtx#cores#fwd");
+                                "rxtx#cores#fwd#txpkts");
 
 cmdline_parse_inst_t cmd_showcfg = {
        .f = cmd_showcfg_parsed,
        .data = NULL,
-       .help_str = "show config rxtx|cores|fwd",
+       .help_str = "show config rxtx|cores|fwd|txpkts",
        .tokens = {
                (void *)&cmd_showcfg_show,
                (void *)&cmd_showcfg_port,
@@ -6540,8 +7035,10 @@ cmd_tunnel_filter_parsed(void *parsed_result,
        struct rte_eth_tunnel_filter_conf tunnel_filter_conf;
        int ret = 0;
 
-       tunnel_filter_conf.outer_mac = &res->outer_mac;
-       tunnel_filter_conf.inner_mac = &res->inner_mac;
+       memset(&tunnel_filter_conf, 0, sizeof(tunnel_filter_conf));
+
+       ether_addr_copy(&res->outer_mac, &tunnel_filter_conf.outer_mac);
+       ether_addr_copy(&res->inner_mac, &tunnel_filter_conf.inner_mac);
        tunnel_filter_conf.inner_vlan = res->inner_vlan;
 
        if (res->ip_value.family == AF_INET) {
@@ -6567,6 +7064,10 @@ cmd_tunnel_filter_parsed(void *parsed_result,
        else if (!strcmp(res->filter_type, "omac-imac-tenid"))
                tunnel_filter_conf.filter_type =
                        RTE_TUNNEL_FILTER_OMAC_TENID_IMAC;
+       else if (!strcmp(res->filter_type, "oip"))
+               tunnel_filter_conf.filter_type = ETH_TUNNEL_FILTER_OIP;
+       else if (!strcmp(res->filter_type, "iip"))
+               tunnel_filter_conf.filter_type = ETH_TUNNEL_FILTER_IIP;
        else {
                printf("The filter type is not supported");
                return;
@@ -6576,6 +7077,8 @@ cmd_tunnel_filter_parsed(void *parsed_result,
                tunnel_filter_conf.tunnel_type = RTE_TUNNEL_TYPE_VXLAN;
        else if (!strcmp(res->tunnel_type, "nvgre"))
                tunnel_filter_conf.tunnel_type = RTE_TUNNEL_TYPE_NVGRE;
+       else if (!strcmp(res->tunnel_type, "ipingre"))
+               tunnel_filter_conf.tunnel_type = RTE_TUNNEL_TYPE_IP_IN_GRE;
        else {
                printf("The tunnel type %s not supported.\n", res->tunnel_type);
                return;
@@ -6621,11 +7124,11 @@ cmdline_parse_token_ipaddr_t cmd_tunnel_filter_ip_value =
        ip_value);
 cmdline_parse_token_string_t cmd_tunnel_filter_tunnel_type =
        TOKEN_STRING_INITIALIZER(struct cmd_tunnel_filter_result,
-       tunnel_type, "vxlan#nvgre");
+       tunnel_type, "vxlan#nvgre#ipingre");
 
 cmdline_parse_token_string_t cmd_tunnel_filter_filter_type =
        TOKEN_STRING_INITIALIZER(struct cmd_tunnel_filter_result,
-       filter_type, "imac-ivlan#imac-ivlan-tenid#imac-tenid#"
+       filter_type, "oip#iip#imac-ivlan#imac-ivlan-tenid#imac-tenid#"
                "imac#omac-imac-tenid");
 cmdline_parse_token_num_t cmd_tunnel_filter_tenant_id =
        TOKEN_NUM_INITIALIZER(struct cmd_tunnel_filter_result,
@@ -6639,8 +7142,8 @@ cmdline_parse_inst_t cmd_tunnel_filter = {
        .data = (void *)0,
        .help_str = "add/rm tunnel filter of a port: "
                        "tunnel_filter add port_id outer_mac inner_mac ip "
-                       "inner_vlan tunnel_type(vxlan|nvgre) filter_type "
-                       "(imac-ivlan|imac-ivlan-tenid|imac-tenid|"
+                       "inner_vlan tunnel_type(vxlan|nvgre|ipingre) filter_type "
+                       "(oip|iip|imac-ivlan|imac-ivlan-tenid|imac-tenid|"
                        "imac|omac-imac-tenid) "
                        "tenant_id queue_num",
        .tokens = {
@@ -6682,9 +7185,11 @@ cmd_tunnel_udp_config_parsed(void *parsed_result,
                tunnel_udp.prot_type = RTE_TUNNEL_TYPE_VXLAN;
 
        if (!strcmp(res->what, "add"))
-               ret = rte_eth_dev_udp_tunnel_add(res->port_id, &tunnel_udp);
+               ret = rte_eth_dev_udp_tunnel_port_add(res->port_id,
+                                                     &tunnel_udp);
        else
-               ret = rte_eth_dev_udp_tunnel_delete(res->port_id, &tunnel_udp);
+               ret = rte_eth_dev_udp_tunnel_port_delete(res->port_id,
+                                                        &tunnel_udp);
 
        if (ret < 0)
                printf("udp tunneling add error: (%s)\n", strerror(-ret));
@@ -6717,6 +7222,57 @@ cmdline_parse_inst_t cmd_tunnel_udp_config = {
        },
 };
 
+/* *** GLOBAL CONFIG *** */
+struct cmd_global_config_result {
+       cmdline_fixed_string_t cmd;
+       uint8_t port_id;
+       cmdline_fixed_string_t cfg_type;
+       uint8_t len;
+};
+
+static void
+cmd_global_config_parsed(void *parsed_result,
+                        __attribute__((unused)) struct cmdline *cl,
+                        __attribute__((unused)) void *data)
+{
+       struct cmd_global_config_result *res = parsed_result;
+       struct rte_eth_global_cfg conf;
+       int ret;
+
+       memset(&conf, 0, sizeof(conf));
+       conf.cfg_type = RTE_ETH_GLOBAL_CFG_TYPE_GRE_KEY_LEN;
+       conf.cfg.gre_key_len = res->len;
+       ret = rte_eth_dev_filter_ctrl(res->port_id, RTE_ETH_FILTER_NONE,
+                                     RTE_ETH_FILTER_SET, &conf);
+       if (ret != 0)
+               printf("Global config error\n");
+}
+
+cmdline_parse_token_string_t cmd_global_config_cmd =
+       TOKEN_STRING_INITIALIZER(struct cmd_global_config_result, cmd,
+               "global_config");
+cmdline_parse_token_num_t cmd_global_config_port_id =
+       TOKEN_NUM_INITIALIZER(struct cmd_global_config_result, port_id, UINT8);
+cmdline_parse_token_string_t cmd_global_config_type =
+       TOKEN_STRING_INITIALIZER(struct cmd_global_config_result,
+               cfg_type, "gre-key-len");
+cmdline_parse_token_num_t cmd_global_config_gre_key_len =
+       TOKEN_NUM_INITIALIZER(struct cmd_global_config_result,
+               len, UINT8);
+
+cmdline_parse_inst_t cmd_global_config = {
+       .f = cmd_global_config_parsed,
+       .data = (void *)NULL,
+       .help_str = "global_config <port_id> gre-key-len <number>",
+       .tokens = {
+               (void *)&cmd_global_config_cmd,
+               (void *)&cmd_global_config_port_id,
+               (void *)&cmd_global_config_type,
+               (void *)&cmd_global_config_gre_key_len,
+               NULL,
+       },
+};
+
 /* *** CONFIGURE VM MIRROR VLAN/POOL RULE *** */
 struct cmd_set_mirror_mask_result {
        cmdline_fixed_string_t set;
@@ -7000,8 +7556,6 @@ static void cmd_dump_parsed(void *parsed_result,
                rte_dump_physmem_layout(stdout);
        else if (!strcmp(res->dump, "dump_memzone"))
                rte_memzone_dump(stdout);
-       else if (!strcmp(res->dump, "dump_log_history"))
-               rte_log_dump_history(stdout);
        else if (!strcmp(res->dump, "dump_struct_sizes"))
                dump_struct_sizes();
        else if (!strcmp(res->dump, "dump_ring"))
@@ -7016,7 +7570,6 @@ cmdline_parse_token_string_t cmd_dump_dump =
        TOKEN_STRING_INITIALIZER(struct cmd_dump_result, dump,
                "dump_physmem#"
                "dump_memzone#"
-               "dump_log_history#"
                "dump_struct_sizes#"
                "dump_ring#"
                "dump_mempool#"
@@ -7834,10 +8387,17 @@ struct cmd_flow_director_result {
        uint16_t port_dst;
        cmdline_fixed_string_t verify_tag;
        uint32_t verify_tag_value;
+       cmdline_ipaddr_t tos;
+       uint8_t tos_value;
+       cmdline_ipaddr_t proto;
+       uint8_t proto_value;
+       cmdline_ipaddr_t ttl;
+       uint8_t ttl_value;
        cmdline_fixed_string_t vlan;
        uint16_t vlan_value;
        cmdline_fixed_string_t flexbytes;
        cmdline_fixed_string_t flexbytes_value;
+       cmdline_fixed_string_t pf_vf;
        cmdline_fixed_string_t drop;
        cmdline_fixed_string_t queue;
        uint16_t  queue_id;
@@ -7920,14 +8480,14 @@ str2flowtype(char *string)
        return RTE_ETH_FLOW_UNKNOWN;
 }
 
-static uint8_t
+static enum rte_eth_fdir_tunnel_type
 str2fdir_tunneltype(char *string)
 {
        uint8_t i = 0;
 
        static const struct {
                char str[32];
-               uint8_t type;
+               enum rte_eth_fdir_tunnel_type type;
        } tunneltype_str[] = {
                {"NVGRE", RTE_FDIR_TUNNEL_TYPE_NVGRE},
                {"VxLAN", RTE_FDIR_TUNNEL_TYPE_VXLAN},
@@ -7970,6 +8530,8 @@ cmd_flow_director_filter_parsed(void *parsed_result,
        struct cmd_flow_director_result *res = parsed_result;
        struct rte_eth_fdir_filter entry;
        uint8_t flexbytes[RTE_ETH_FDIR_MAX_FLEXLEN];
+       char *end;
+       unsigned long vf_id;
        int ret = 0;
 
        ret = rte_eth_dev_filter_supported(res->port_id, RTE_ETH_FILTER_FDIR);
@@ -8010,12 +8572,15 @@ cmd_flow_director_filter_parsed(void *parsed_result,
        switch (entry.input.flow_type) {
        case RTE_ETH_FLOW_FRAG_IPV4:
        case RTE_ETH_FLOW_NONFRAG_IPV4_OTHER:
+               entry.input.flow.ip4_flow.proto = res->proto_value;
        case RTE_ETH_FLOW_NONFRAG_IPV4_UDP:
        case RTE_ETH_FLOW_NONFRAG_IPV4_TCP:
                IPV4_ADDR_TO_UINT(res->ip_dst,
                        entry.input.flow.ip4_flow.dst_ip);
                IPV4_ADDR_TO_UINT(res->ip_src,
                        entry.input.flow.ip4_flow.src_ip);
+               entry.input.flow.ip4_flow.tos = res->tos_value;
+               entry.input.flow.ip4_flow.ttl = res->ttl_value;
                /* need convert to big endian. */
                entry.input.flow.udp4_flow.dst_port =
                                rte_cpu_to_be_16(res->port_dst);
@@ -8027,6 +8592,8 @@ cmd_flow_director_filter_parsed(void *parsed_result,
                        entry.input.flow.sctp4_flow.ip.dst_ip);
                IPV4_ADDR_TO_UINT(res->ip_src,
                        entry.input.flow.sctp4_flow.ip.src_ip);
+               entry.input.flow.ip4_flow.tos = res->tos_value;
+               entry.input.flow.ip4_flow.ttl = res->ttl_value;
                /* need convert to big endian. */
                entry.input.flow.sctp4_flow.dst_port =
                                rte_cpu_to_be_16(res->port_dst);
@@ -8037,12 +8604,15 @@ cmd_flow_director_filter_parsed(void *parsed_result,
                break;
        case RTE_ETH_FLOW_FRAG_IPV6:
        case RTE_ETH_FLOW_NONFRAG_IPV6_OTHER:
+               entry.input.flow.ipv6_flow.proto = res->proto_value;
        case RTE_ETH_FLOW_NONFRAG_IPV6_UDP:
        case RTE_ETH_FLOW_NONFRAG_IPV6_TCP:
                IPV6_ADDR_TO_ARRAY(res->ip_dst,
                        entry.input.flow.ipv6_flow.dst_ip);
                IPV6_ADDR_TO_ARRAY(res->ip_src,
                        entry.input.flow.ipv6_flow.src_ip);
+               entry.input.flow.ipv6_flow.tc = res->tos_value;
+               entry.input.flow.ipv6_flow.hop_limits = res->ttl_value;
                /* need convert to big endian. */
                entry.input.flow.udp6_flow.dst_port =
                                rte_cpu_to_be_16(res->port_dst);
@@ -8054,6 +8624,8 @@ cmd_flow_director_filter_parsed(void *parsed_result,
                        entry.input.flow.sctp6_flow.ip.dst_ip);
                IPV6_ADDR_TO_ARRAY(res->ip_src,
                        entry.input.flow.sctp6_flow.ip.src_ip);
+               entry.input.flow.ipv6_flow.tc = res->tos_value;
+               entry.input.flow.ipv6_flow.hop_limits = res->ttl_value;
                /* need convert to big endian. */
                entry.input.flow.sctp6_flow.dst_port =
                                rte_cpu_to_be_16(res->port_dst);
@@ -8096,6 +8668,31 @@ cmd_flow_director_filter_parsed(void *parsed_result,
                entry.action.behavior = RTE_ETH_FDIR_REJECT;
        else
                entry.action.behavior = RTE_ETH_FDIR_ACCEPT;
+
+       if (fdir_conf.mode !=  RTE_FDIR_MODE_PERFECT_MAC_VLAN &&
+           fdir_conf.mode !=  RTE_FDIR_MODE_PERFECT_TUNNEL) {
+               if (!strcmp(res->pf_vf, "pf"))
+                       entry.input.flow_ext.is_vf = 0;
+               else if (!strncmp(res->pf_vf, "vf", 2)) {
+                       struct rte_eth_dev_info dev_info;
+
+                       memset(&dev_info, 0, sizeof(dev_info));
+                       rte_eth_dev_info_get(res->port_id, &dev_info);
+                       errno = 0;
+                       vf_id = strtoul(res->pf_vf + 2, &end, 10);
+                       if (errno != 0 || *end != '\0' ||
+                           vf_id >= dev_info.max_vfs) {
+                               printf("invalid parameter %s.\n", res->pf_vf);
+                               return;
+                       }
+                       entry.input.flow_ext.is_vf = 1;
+                       entry.input.flow_ext.dst_id = (uint16_t)vf_id;
+               } else {
+                       printf("invalid parameter %s.\n", res->pf_vf);
+                       return;
+               }
+       }
+
        /* set to report FD ID by default */
        entry.action.report_status = RTE_ETH_FDIR_REPORT_ID;
        entry.action.rx_queue = res->queue_id;
@@ -8160,6 +8757,24 @@ cmdline_parse_token_string_t cmd_flow_director_verify_tag =
 cmdline_parse_token_num_t cmd_flow_director_verify_tag_value =
        TOKEN_NUM_INITIALIZER(struct cmd_flow_director_result,
                              verify_tag_value, UINT32);
+cmdline_parse_token_string_t cmd_flow_director_tos =
+       TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
+                                tos, "tos");
+cmdline_parse_token_num_t cmd_flow_director_tos_value =
+       TOKEN_NUM_INITIALIZER(struct cmd_flow_director_result,
+                             tos_value, UINT8);
+cmdline_parse_token_string_t cmd_flow_director_proto =
+       TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
+                                proto, "proto");
+cmdline_parse_token_num_t cmd_flow_director_proto_value =
+       TOKEN_NUM_INITIALIZER(struct cmd_flow_director_result,
+                             proto_value, UINT8);
+cmdline_parse_token_string_t cmd_flow_director_ttl =
+       TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
+                                ttl, "ttl");
+cmdline_parse_token_num_t cmd_flow_director_ttl_value =
+       TOKEN_NUM_INITIALIZER(struct cmd_flow_director_result,
+                             ttl_value, UINT8);
 cmdline_parse_token_string_t cmd_flow_director_vlan =
        TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
                                 vlan, "vlan");
@@ -8175,6 +8790,9 @@ cmdline_parse_token_string_t cmd_flow_director_flexbytes_value =
 cmdline_parse_token_string_t cmd_flow_director_drop =
        TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
                                 drop, "drop#fwd");
+cmdline_parse_token_string_t cmd_flow_director_pf_vf =
+       TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
+                             pf_vf, NULL);
 cmdline_parse_token_string_t cmd_flow_director_queue =
        TOKEN_STRING_INITIALIZER(struct cmd_flow_director_result,
                                 queue, "queue");
@@ -8235,11 +8853,18 @@ cmdline_parse_inst_t cmd_add_del_ip_flow_director = {
                (void *)&cmd_flow_director_ip_src,
                (void *)&cmd_flow_director_dst,
                (void *)&cmd_flow_director_ip_dst,
+               (void *)&cmd_flow_director_tos,
+               (void *)&cmd_flow_director_tos_value,
+               (void *)&cmd_flow_director_proto,
+               (void *)&cmd_flow_director_proto_value,
+               (void *)&cmd_flow_director_ttl,
+               (void *)&cmd_flow_director_ttl_value,
                (void *)&cmd_flow_director_vlan,
                (void *)&cmd_flow_director_vlan_value,
                (void *)&cmd_flow_director_flexbytes,
                (void *)&cmd_flow_director_flexbytes_value,
                (void *)&cmd_flow_director_drop,
+               (void *)&cmd_flow_director_pf_vf,
                (void *)&cmd_flow_director_queue,
                (void *)&cmd_flow_director_queue_id,
                (void *)&cmd_flow_director_fd_id,
@@ -8266,11 +8891,16 @@ cmdline_parse_inst_t cmd_add_del_udp_flow_director = {
                (void *)&cmd_flow_director_dst,
                (void *)&cmd_flow_director_ip_dst,
                (void *)&cmd_flow_director_port_dst,
+               (void *)&cmd_flow_director_tos,
+               (void *)&cmd_flow_director_tos_value,
+               (void *)&cmd_flow_director_ttl,
+               (void *)&cmd_flow_director_ttl_value,
                (void *)&cmd_flow_director_vlan,
                (void *)&cmd_flow_director_vlan_value,
                (void *)&cmd_flow_director_flexbytes,
                (void *)&cmd_flow_director_flexbytes_value,
                (void *)&cmd_flow_director_drop,
+               (void *)&cmd_flow_director_pf_vf,
                (void *)&cmd_flow_director_queue,
                (void *)&cmd_flow_director_queue_id,
                (void *)&cmd_flow_director_fd_id,
@@ -8299,11 +8929,16 @@ cmdline_parse_inst_t cmd_add_del_sctp_flow_director = {
                (void *)&cmd_flow_director_port_dst,
                (void *)&cmd_flow_director_verify_tag,
                (void *)&cmd_flow_director_verify_tag_value,
+               (void *)&cmd_flow_director_tos,
+               (void *)&cmd_flow_director_tos_value,
+               (void *)&cmd_flow_director_ttl,
+               (void *)&cmd_flow_director_ttl_value,
                (void *)&cmd_flow_director_vlan,
                (void *)&cmd_flow_director_vlan_value,
                (void *)&cmd_flow_director_flexbytes,
                (void *)&cmd_flow_director_flexbytes_value,
                (void *)&cmd_flow_director_drop,
+               (void *)&cmd_flow_director_pf_vf,
                (void *)&cmd_flow_director_queue,
                (void *)&cmd_flow_director_queue_id,
                (void *)&cmd_flow_director_fd_id,
@@ -8329,6 +8964,7 @@ cmdline_parse_inst_t cmd_add_del_l2_flow_director = {
                (void *)&cmd_flow_director_flexbytes,
                (void *)&cmd_flow_director_flexbytes_value,
                (void *)&cmd_flow_director_drop,
+               (void *)&cmd_flow_director_pf_vf,
                (void *)&cmd_flow_director_queue,
                (void *)&cmd_flow_director_queue_id,
                (void *)&cmd_flow_director_fd_id,
@@ -8490,7 +9126,6 @@ cmd_flow_director_mask_parsed(void *parsed_result,
                }
 
                mask->vlan_tci_mask = res->vlan_mask;
-               mask->mac_addr_byte_mask = res->mac_addr_byte_mask;
        } else if (fdir_conf.mode ==  RTE_FDIR_MODE_PERFECT_TUNNEL) {
                if (strcmp(res->mode_value, "Tunnel")) {
                        printf("Please set mode to Tunnel.\n");
@@ -8507,13 +9142,13 @@ cmd_flow_director_mask_parsed(void *parsed_result,
                        return;
                }
 
-               mask->vlan_tci_mask = res->vlan_mask;
+               mask->vlan_tci_mask = rte_cpu_to_be_16(res->vlan_mask);
                IPV4_ADDR_TO_UINT(res->ipv4_src, mask->ipv4_mask.src_ip);
                IPV4_ADDR_TO_UINT(res->ipv4_dst, mask->ipv4_mask.dst_ip);
                IPV6_ADDR_TO_ARRAY(res->ipv6_src, mask->ipv6_mask.src_ip);
                IPV6_ADDR_TO_ARRAY(res->ipv6_dst, mask->ipv6_mask.dst_ip);
-               mask->src_port_mask = res->port_src;
-               mask->dst_port_mask = res->port_dst;
+               mask->src_port_mask = rte_cpu_to_be_16(res->port_src);
+               mask->dst_port_mask = rte_cpu_to_be_16(res->port_dst);
        }
 
        cmd_reconfig_device_queue(res->port_id, 1, 1);
@@ -8621,8 +9256,6 @@ cmdline_parse_inst_t cmd_set_flow_director_mac_vlan_mask = {
                (void *)&cmd_flow_director_mask_mode_mac_vlan,
                (void *)&cmd_flow_director_mask_vlan,
                (void *)&cmd_flow_director_mask_vlan_value,
-               (void *)&cmd_flow_director_mask_mac,
-               (void *)&cmd_flow_director_mask_mac_value,
                NULL,
        },
 };
@@ -9029,6 +9662,10 @@ flowtype_to_str(uint16_t ftype)
                {"ipv6-sctp", RTE_ETH_FLOW_NONFRAG_IPV6_SCTP},
                {"ipv6-other", RTE_ETH_FLOW_NONFRAG_IPV6_OTHER},
                {"l2_payload", RTE_ETH_FLOW_L2_PAYLOAD},
+               {"port", RTE_ETH_FLOW_PORT},
+               {"vxlan", RTE_ETH_FLOW_VXLAN},
+               {"geneve", RTE_ETH_FLOW_GENEVE},
+               {"nvgre", RTE_ETH_FLOW_NVGRE},
        };
 
        for (i = 0; i < RTE_DIM(ftype_table); i++) {
@@ -9205,6 +9842,205 @@ cmdline_parse_inst_t cmd_set_hash_global_config = {
        },
 };
 
+/* Set hash input set */
+struct cmd_set_hash_input_set_result {
+       cmdline_fixed_string_t set_hash_input_set;
+       uint8_t port_id;
+       cmdline_fixed_string_t flow_type;
+       cmdline_fixed_string_t inset_field;
+       cmdline_fixed_string_t select;
+};
+
+static enum rte_eth_input_set_field
+str2inset(char *string)
+{
+       uint16_t i;
+
+       static const struct {
+               char str[32];
+               enum rte_eth_input_set_field inset;
+       } inset_table[] = {
+               {"ethertype", RTE_ETH_INPUT_SET_L2_ETHERTYPE},
+               {"ovlan", RTE_ETH_INPUT_SET_L2_OUTER_VLAN},
+               {"ivlan", RTE_ETH_INPUT_SET_L2_INNER_VLAN},
+               {"src-ipv4", RTE_ETH_INPUT_SET_L3_SRC_IP4},
+               {"dst-ipv4", RTE_ETH_INPUT_SET_L3_DST_IP4},
+               {"ipv4-tos", RTE_ETH_INPUT_SET_L3_IP4_TOS},
+               {"ipv4-proto", RTE_ETH_INPUT_SET_L3_IP4_PROTO},
+               {"ipv4-ttl", RTE_ETH_INPUT_SET_L3_IP4_TTL},
+               {"src-ipv6", RTE_ETH_INPUT_SET_L3_SRC_IP6},
+               {"dst-ipv6", RTE_ETH_INPUT_SET_L3_DST_IP6},
+               {"ipv6-tc", RTE_ETH_INPUT_SET_L3_IP6_TC},
+               {"ipv6-next-header", RTE_ETH_INPUT_SET_L3_IP6_NEXT_HEADER},
+               {"ipv6-hop-limits", RTE_ETH_INPUT_SET_L3_IP6_HOP_LIMITS},
+               {"udp-src-port", RTE_ETH_INPUT_SET_L4_UDP_SRC_PORT},
+               {"udp-dst-port", RTE_ETH_INPUT_SET_L4_UDP_DST_PORT},
+               {"tcp-src-port", RTE_ETH_INPUT_SET_L4_TCP_SRC_PORT},
+               {"tcp-dst-port", RTE_ETH_INPUT_SET_L4_TCP_DST_PORT},
+               {"sctp-src-port", RTE_ETH_INPUT_SET_L4_SCTP_SRC_PORT},
+               {"sctp-dst-port", RTE_ETH_INPUT_SET_L4_SCTP_DST_PORT},
+               {"sctp-veri-tag", RTE_ETH_INPUT_SET_L4_SCTP_VERIFICATION_TAG},
+               {"udp-key", RTE_ETH_INPUT_SET_TUNNEL_L4_UDP_KEY},
+               {"gre-key", RTE_ETH_INPUT_SET_TUNNEL_GRE_KEY},
+               {"fld-1st", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_1ST_WORD},
+               {"fld-2nd", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_2ND_WORD},
+               {"fld-3rd", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_3RD_WORD},
+               {"fld-4th", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_4TH_WORD},
+               {"fld-5th", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_5TH_WORD},
+               {"fld-6th", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_6TH_WORD},
+               {"fld-7th", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_7TH_WORD},
+               {"fld-8th", RTE_ETH_INPUT_SET_FLEX_PAYLOAD_8TH_WORD},
+               {"none", RTE_ETH_INPUT_SET_NONE},
+       };
+
+       for (i = 0; i < RTE_DIM(inset_table); i++) {
+               if (!strcmp(string, inset_table[i].str))
+                       return inset_table[i].inset;
+       }
+
+       return RTE_ETH_INPUT_SET_UNKNOWN;
+}
+
+static void
+cmd_set_hash_input_set_parsed(void *parsed_result,
+                             __rte_unused struct cmdline *cl,
+                             __rte_unused void *data)
+{
+       struct cmd_set_hash_input_set_result *res = parsed_result;
+       struct rte_eth_hash_filter_info info;
+
+       memset(&info, 0, sizeof(info));
+       info.info_type = RTE_ETH_HASH_FILTER_INPUT_SET_SELECT;
+       info.info.input_set_conf.flow_type = str2flowtype(res->flow_type);
+       info.info.input_set_conf.field[0] = str2inset(res->inset_field);
+       info.info.input_set_conf.inset_size = 1;
+       if (!strcmp(res->select, "select"))
+               info.info.input_set_conf.op = RTE_ETH_INPUT_SET_SELECT;
+       else if (!strcmp(res->select, "add"))
+               info.info.input_set_conf.op = RTE_ETH_INPUT_SET_ADD;
+       rte_eth_dev_filter_ctrl(res->port_id, RTE_ETH_FILTER_HASH,
+                               RTE_ETH_FILTER_SET, &info);
+}
+
+cmdline_parse_token_string_t cmd_set_hash_input_set_cmd =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_hash_input_set_result,
+               set_hash_input_set, "set_hash_input_set");
+cmdline_parse_token_num_t cmd_set_hash_input_set_port_id =
+       TOKEN_NUM_INITIALIZER(struct cmd_set_hash_input_set_result,
+               port_id, UINT8);
+cmdline_parse_token_string_t cmd_set_hash_input_set_flow_type =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_hash_input_set_result,
+               flow_type,
+               "ipv4-frag#ipv4-tcp#ipv4-udp#ipv4-sctp#ipv4-other#"
+               "ipv6-frag#ipv6-tcp#ipv6-udp#ipv6-sctp#ipv6-other#l2_payload");
+cmdline_parse_token_string_t cmd_set_hash_input_set_field =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_hash_input_set_result,
+               inset_field,
+               "ovlan#ivlan#src-ipv4#dst-ipv4#src-ipv6#dst-ipv6#"
+               "ipv4-tos#ipv4-proto#ipv6-tc#ipv6-next-header#udp-src-port#"
+               "udp-dst-port#tcp-src-port#tcp-dst-port#sctp-src-port#"
+               "sctp-dst-port#sctp-veri-tag#udp-key#gre-key#fld-1st#"
+               "fld-2nd#fld-3rd#fld-4th#fld-5th#fld-6th#fld-7th#"
+               "fld-8th#none");
+cmdline_parse_token_string_t cmd_set_hash_input_set_select =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_hash_input_set_result,
+               select, "select#add");
+
+cmdline_parse_inst_t cmd_set_hash_input_set = {
+       .f = cmd_set_hash_input_set_parsed,
+       .data = NULL,
+       .help_str = "set_hash_input_set <port_id> "
+       "ipv4-frag|ipv4-tcp|ipv4-udp|ipv4-sctp|ipv4-other|"
+       "ipv6-frag|ipv6-tcp|ipv6-udp|ipv6-sctp|ipv6-other|l2_payload "
+       "ovlan|ivlan|src-ipv4|dst-ipv4|src-ipv6|dst-ipv6|ipv4-tos|ipv4-proto|"
+       "ipv6-tc|ipv6-next-header|udp-src-port|udp-dst-port|tcp-src-port|"
+       "tcp-dst-port|sctp-src-port|sctp-dst-port|sctp-veri-tag|udp-key|"
+       "gre-key|fld-1st|fld-2nd|fld-3rd|fld-4th|fld-5th|fld-6th|"
+       "fld-7th|fld-8th|none select|add",
+       .tokens = {
+               (void *)&cmd_set_hash_input_set_cmd,
+               (void *)&cmd_set_hash_input_set_port_id,
+               (void *)&cmd_set_hash_input_set_flow_type,
+               (void *)&cmd_set_hash_input_set_field,
+               (void *)&cmd_set_hash_input_set_select,
+               NULL,
+       },
+};
+
+/* Set flow director input set */
+struct cmd_set_fdir_input_set_result {
+       cmdline_fixed_string_t set_fdir_input_set;
+       uint8_t port_id;
+       cmdline_fixed_string_t flow_type;
+       cmdline_fixed_string_t inset_field;
+       cmdline_fixed_string_t select;
+};
+
+static void
+cmd_set_fdir_input_set_parsed(void *parsed_result,
+       __rte_unused struct cmdline *cl,
+       __rte_unused void *data)
+{
+       struct cmd_set_fdir_input_set_result *res = parsed_result;
+       struct rte_eth_fdir_filter_info info;
+
+       memset(&info, 0, sizeof(info));
+       info.info_type = RTE_ETH_FDIR_FILTER_INPUT_SET_SELECT;
+       info.info.input_set_conf.flow_type = str2flowtype(res->flow_type);
+       info.info.input_set_conf.field[0] = str2inset(res->inset_field);
+       info.info.input_set_conf.inset_size = 1;
+       if (!strcmp(res->select, "select"))
+               info.info.input_set_conf.op = RTE_ETH_INPUT_SET_SELECT;
+       else if (!strcmp(res->select, "add"))
+               info.info.input_set_conf.op = RTE_ETH_INPUT_SET_ADD;
+       rte_eth_dev_filter_ctrl(res->port_id, RTE_ETH_FILTER_FDIR,
+               RTE_ETH_FILTER_SET, &info);
+}
+
+cmdline_parse_token_string_t cmd_set_fdir_input_set_cmd =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fdir_input_set_result,
+       set_fdir_input_set, "set_fdir_input_set");
+cmdline_parse_token_num_t cmd_set_fdir_input_set_port_id =
+       TOKEN_NUM_INITIALIZER(struct cmd_set_fdir_input_set_result,
+       port_id, UINT8);
+cmdline_parse_token_string_t cmd_set_fdir_input_set_flow_type =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fdir_input_set_result,
+       flow_type,
+       "ipv4-frag#ipv4-tcp#ipv4-udp#ipv4-sctp#ipv4-other#"
+       "ipv6-frag#ipv6-tcp#ipv6-udp#ipv6-sctp#ipv6-other#l2_payload");
+cmdline_parse_token_string_t cmd_set_fdir_input_set_field =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fdir_input_set_result,
+       inset_field,
+       "ivlan#ethertype#src-ipv4#dst-ipv4#src-ipv6#dst-ipv6#"
+       "ipv4-tos#ipv4-proto#ipv4-ttl#ipv6-tc#ipv6-next-header#"
+       "ipv6-hop-limits#udp-src-port#udp-dst-port#"
+       "tcp-src-port#tcp-dst-port#sctp-src-port#sctp-dst-port#"
+       "sctp-veri-tag#none");
+cmdline_parse_token_string_t cmd_set_fdir_input_set_select =
+       TOKEN_STRING_INITIALIZER(struct cmd_set_fdir_input_set_result,
+       select, "select#add");
+
+cmdline_parse_inst_t cmd_set_fdir_input_set = {
+       .f = cmd_set_fdir_input_set_parsed,
+       .data = NULL,
+       .help_str = "set_fdir_input_set <port_id> "
+       "ipv4-frag|ipv4-tcp|ipv4-udp|ipv4-sctp|ipv4-other|"
+       "ipv6-frag|ipv6-tcp|ipv6-udp|ipv6-sctp|ipv6-other|l2_payload "
+       "ivlan|ethertype|src-ipv4|dst-ipv4|src-ipv6|dst-ipv6|"
+       "ipv4-tos|ipv4-proto|ipv4-ttl|ipv6-tc|ipv6-next-header|"
+       "ipv6-hop-limits|udp-src-port|udp-dst-port|"
+       "tcp-src-port|tcp-dst-port|sctp-src-port|sctp-dst-port|"
+       "sctp-veri-tag|none select|add",
+       .tokens = {
+               (void *)&cmd_set_fdir_input_set_cmd,
+               (void *)&cmd_set_fdir_input_set_port_id,
+               (void *)&cmd_set_fdir_input_set_flow_type,
+               (void *)&cmd_set_fdir_input_set_field,
+               (void *)&cmd_set_fdir_input_set_select,
+               NULL,
+       },
+};
+
 /* *** ADD/REMOVE A MULTICAST MAC ADDRESS TO/FROM A PORT *** */
 struct cmd_mcast_addr_result {
        cmdline_fixed_string_t mcast_addr_cmd;
@@ -9256,97 +10092,1423 @@ cmdline_parse_inst_t cmd_mcast_addr = {
        },
 };
 
-/* ******************************************************************************** */
+/* l2 tunnel config
+ * only support E-tag now.
+ */
 
-/* list of instructions */
-cmdline_parse_ctx_t main_ctx[] = {
-       (cmdline_parse_inst_t *)&cmd_help_brief,
-       (cmdline_parse_inst_t *)&cmd_help_long,
-       (cmdline_parse_inst_t *)&cmd_quit,
-       (cmdline_parse_inst_t *)&cmd_showport,
-       (cmdline_parse_inst_t *)&cmd_showqueue,
-       (cmdline_parse_inst_t *)&cmd_showportall,
-       (cmdline_parse_inst_t *)&cmd_showcfg,
-       (cmdline_parse_inst_t *)&cmd_start,
-       (cmdline_parse_inst_t *)&cmd_start_tx_first,
-       (cmdline_parse_inst_t *)&cmd_set_link_up,
-       (cmdline_parse_inst_t *)&cmd_set_link_down,
-       (cmdline_parse_inst_t *)&cmd_reset,
-       (cmdline_parse_inst_t *)&cmd_set_numbers,
-       (cmdline_parse_inst_t *)&cmd_set_txpkts,
-       (cmdline_parse_inst_t *)&cmd_set_fwd_list,
-       (cmdline_parse_inst_t *)&cmd_set_fwd_mask,
-       (cmdline_parse_inst_t *)&cmd_set_fwd_mode,
-       (cmdline_parse_inst_t *)&cmd_set_burst_tx_retry,
-       (cmdline_parse_inst_t *)&cmd_set_promisc_mode_one,
-       (cmdline_parse_inst_t *)&cmd_set_promisc_mode_all,
-       (cmdline_parse_inst_t *)&cmd_set_allmulti_mode_one,
-       (cmdline_parse_inst_t *)&cmd_set_allmulti_mode_all,
-       (cmdline_parse_inst_t *)&cmd_set_flush_rx,
-       (cmdline_parse_inst_t *)&cmd_set_link_check,
-#ifdef RTE_NIC_BYPASS
-       (cmdline_parse_inst_t *)&cmd_set_bypass_mode,
-       (cmdline_parse_inst_t *)&cmd_set_bypass_event,
-       (cmdline_parse_inst_t *)&cmd_set_bypass_timeout,
-       (cmdline_parse_inst_t *)&cmd_show_bypass_config,
-#endif
-#ifdef RTE_LIBRTE_PMD_BOND
-       (cmdline_parse_inst_t *) &cmd_set_bonding_mode,
-       (cmdline_parse_inst_t *) &cmd_show_bonding_config,
-       (cmdline_parse_inst_t *) &cmd_set_bonding_primary,
-       (cmdline_parse_inst_t *) &cmd_add_bonding_slave,
-       (cmdline_parse_inst_t *) &cmd_remove_bonding_slave,
-       (cmdline_parse_inst_t *) &cmd_create_bonded_device,
-       (cmdline_parse_inst_t *) &cmd_set_bond_mac_addr,
-       (cmdline_parse_inst_t *) &cmd_set_balance_xmit_policy,
-       (cmdline_parse_inst_t *) &cmd_set_bond_mon_period,
-#endif
-       (cmdline_parse_inst_t *)&cmd_vlan_offload,
-       (cmdline_parse_inst_t *)&cmd_vlan_tpid,
-       (cmdline_parse_inst_t *)&cmd_rx_vlan_filter_all,
-       (cmdline_parse_inst_t *)&cmd_rx_vlan_filter,
-       (cmdline_parse_inst_t *)&cmd_tx_vlan_set,
-       (cmdline_parse_inst_t *)&cmd_tx_vlan_set_qinq,
-       (cmdline_parse_inst_t *)&cmd_tx_vlan_reset,
-       (cmdline_parse_inst_t *)&cmd_tx_vlan_set_pvid,
-       (cmdline_parse_inst_t *)&cmd_csum_set,
-       (cmdline_parse_inst_t *)&cmd_csum_show,
-       (cmdline_parse_inst_t *)&cmd_csum_tunnel,
-       (cmdline_parse_inst_t *)&cmd_tso_set,
-       (cmdline_parse_inst_t *)&cmd_tso_show,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_rx,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_tx,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_hw,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_lw,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_pt,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_xon,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_macfwd,
-       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_autoneg,
-       (cmdline_parse_inst_t *)&cmd_priority_flow_control_set,
-       (cmdline_parse_inst_t *)&cmd_config_dcb,
-       (cmdline_parse_inst_t *)&cmd_read_reg,
-       (cmdline_parse_inst_t *)&cmd_read_reg_bit_field,
-       (cmdline_parse_inst_t *)&cmd_read_reg_bit,
-       (cmdline_parse_inst_t *)&cmd_write_reg,
-       (cmdline_parse_inst_t *)&cmd_write_reg_bit_field,
-       (cmdline_parse_inst_t *)&cmd_write_reg_bit,
-       (cmdline_parse_inst_t *)&cmd_read_rxd_txd,
-       (cmdline_parse_inst_t *)&cmd_stop,
-       (cmdline_parse_inst_t *)&cmd_mac_addr,
-       (cmdline_parse_inst_t *)&cmd_set_qmap,
-       (cmdline_parse_inst_t *)&cmd_operate_port,
-       (cmdline_parse_inst_t *)&cmd_operate_specific_port,
-       (cmdline_parse_inst_t *)&cmd_operate_attach_port,
-       (cmdline_parse_inst_t *)&cmd_operate_detach_port,
-       (cmdline_parse_inst_t *)&cmd_config_speed_all,
-       (cmdline_parse_inst_t *)&cmd_config_speed_specific,
-       (cmdline_parse_inst_t *)&cmd_config_rx_tx,
-       (cmdline_parse_inst_t *)&cmd_config_mtu,
-       (cmdline_parse_inst_t *)&cmd_config_max_pkt_len,
-       (cmdline_parse_inst_t *)&cmd_config_rx_mode_flag,
+/* Ether type config */
+struct cmd_config_l2_tunnel_eth_type_result {
+       cmdline_fixed_string_t port;
+       cmdline_fixed_string_t config;
+       cmdline_fixed_string_t all;
+       uint8_t id;
+       cmdline_fixed_string_t l2_tunnel;
+       cmdline_fixed_string_t l2_tunnel_type;
+       cmdline_fixed_string_t eth_type;
+       uint16_t eth_type_val;
+};
+
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_port =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                port, "port");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_config =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                config, "config");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_all_str =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                all, "all");
+cmdline_parse_token_num_t cmd_config_l2_tunnel_eth_type_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                id, UINT8);
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_l2_tunnel =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                l2_tunnel, "l2-tunnel");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_l2_tunnel_type =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                l2_tunnel_type, "E-tag");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_eth_type_eth_type =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                eth_type, "ether-type");
+cmdline_parse_token_num_t cmd_config_l2_tunnel_eth_type_eth_type_val =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_l2_tunnel_eth_type_result,
+                eth_type_val, UINT16);
+
+static enum rte_eth_tunnel_type
+str2fdir_l2_tunnel_type(char *string)
+{
+       uint32_t i = 0;
+
+       static const struct {
+               char str[32];
+               enum rte_eth_tunnel_type type;
+       } l2_tunnel_type_str[] = {
+               {"E-tag", RTE_L2_TUNNEL_TYPE_E_TAG},
+       };
+
+       for (i = 0; i < RTE_DIM(l2_tunnel_type_str); i++) {
+               if (!strcmp(l2_tunnel_type_str[i].str, string))
+                       return l2_tunnel_type_str[i].type;
+       }
+       return RTE_TUNNEL_TYPE_NONE;
+}
+
+/* ether type config for all ports */
+static void
+cmd_config_l2_tunnel_eth_type_all_parsed
+       (void *parsed_result,
+        __attribute__((unused)) struct cmdline *cl,
+        __attribute__((unused)) void *data)
+{
+       struct cmd_config_l2_tunnel_eth_type_result *res = parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+       portid_t pid;
+
+       entry.l2_tunnel_type = str2fdir_l2_tunnel_type(res->l2_tunnel_type);
+       entry.ether_type = res->eth_type_val;
+
+       FOREACH_PORT(pid, ports) {
+               rte_eth_dev_l2_tunnel_eth_type_conf(pid, &entry);
+       }
+}
+
+cmdline_parse_inst_t cmd_config_l2_tunnel_eth_type_all = {
+       .f = cmd_config_l2_tunnel_eth_type_all_parsed,
+       .data = NULL,
+       .help_str = "port config all l2-tunnel ether-type",
+       .tokens = {
+               (void *)&cmd_config_l2_tunnel_eth_type_port,
+               (void *)&cmd_config_l2_tunnel_eth_type_config,
+               (void *)&cmd_config_l2_tunnel_eth_type_all_str,
+               (void *)&cmd_config_l2_tunnel_eth_type_l2_tunnel,
+               (void *)&cmd_config_l2_tunnel_eth_type_l2_tunnel_type,
+               (void *)&cmd_config_l2_tunnel_eth_type_eth_type,
+               (void *)&cmd_config_l2_tunnel_eth_type_eth_type_val,
+               NULL,
+       },
+};
+
+/* ether type config for a specific port */
+static void
+cmd_config_l2_tunnel_eth_type_specific_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_l2_tunnel_eth_type_result *res =
+                parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = str2fdir_l2_tunnel_type(res->l2_tunnel_type);
+       entry.ether_type = res->eth_type_val;
+
+       rte_eth_dev_l2_tunnel_eth_type_conf(res->id, &entry);
+}
+
+cmdline_parse_inst_t cmd_config_l2_tunnel_eth_type_specific = {
+       .f = cmd_config_l2_tunnel_eth_type_specific_parsed,
+       .data = NULL,
+       .help_str = "port config l2-tunnel ether-type",
+       .tokens = {
+               (void *)&cmd_config_l2_tunnel_eth_type_port,
+               (void *)&cmd_config_l2_tunnel_eth_type_config,
+               (void *)&cmd_config_l2_tunnel_eth_type_id,
+               (void *)&cmd_config_l2_tunnel_eth_type_l2_tunnel,
+               (void *)&cmd_config_l2_tunnel_eth_type_l2_tunnel_type,
+               (void *)&cmd_config_l2_tunnel_eth_type_eth_type,
+               (void *)&cmd_config_l2_tunnel_eth_type_eth_type_val,
+               NULL,
+       },
+};
+
+/* Enable/disable l2 tunnel */
+struct cmd_config_l2_tunnel_en_dis_result {
+       cmdline_fixed_string_t port;
+       cmdline_fixed_string_t config;
+       cmdline_fixed_string_t all;
+       uint8_t id;
+       cmdline_fixed_string_t l2_tunnel;
+       cmdline_fixed_string_t l2_tunnel_type;
+       cmdline_fixed_string_t en_dis;
+};
+
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_port =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                port, "port");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_config =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                config, "config");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_all_str =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                all, "all");
+cmdline_parse_token_num_t cmd_config_l2_tunnel_en_dis_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                id, UINT8);
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_l2_tunnel =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                l2_tunnel, "l2-tunnel");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_l2_tunnel_type =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                l2_tunnel_type, "E-tag");
+cmdline_parse_token_string_t cmd_config_l2_tunnel_en_dis_en_dis =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_l2_tunnel_en_dis_result,
+                en_dis, "enable#disable");
+
+/* enable/disable l2 tunnel for all ports */
+static void
+cmd_config_l2_tunnel_en_dis_all_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_l2_tunnel_en_dis_result *res = parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+       portid_t pid;
+       uint8_t en;
+
+       entry.l2_tunnel_type = str2fdir_l2_tunnel_type(res->l2_tunnel_type);
+
+       if (!strcmp("enable", res->en_dis))
+               en = 1;
+       else
+               en = 0;
+
+       FOREACH_PORT(pid, ports) {
+               rte_eth_dev_l2_tunnel_offload_set(pid,
+                                                 &entry,
+                                                 ETH_L2_TUNNEL_ENABLE_MASK,
+                                                 en);
+       }
+}
+
+cmdline_parse_inst_t cmd_config_l2_tunnel_en_dis_all = {
+       .f = cmd_config_l2_tunnel_en_dis_all_parsed,
+       .data = NULL,
+       .help_str = "port config all l2-tunnel enable/disable",
+       .tokens = {
+               (void *)&cmd_config_l2_tunnel_en_dis_port,
+               (void *)&cmd_config_l2_tunnel_en_dis_config,
+               (void *)&cmd_config_l2_tunnel_en_dis_all_str,
+               (void *)&cmd_config_l2_tunnel_en_dis_l2_tunnel,
+               (void *)&cmd_config_l2_tunnel_en_dis_l2_tunnel_type,
+               (void *)&cmd_config_l2_tunnel_en_dis_en_dis,
+               NULL,
+       },
+};
+
+/* enable/disable l2 tunnel for a port */
+static void
+cmd_config_l2_tunnel_en_dis_specific_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_l2_tunnel_en_dis_result *res =
+               parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = str2fdir_l2_tunnel_type(res->l2_tunnel_type);
+
+       if (!strcmp("enable", res->en_dis))
+               rte_eth_dev_l2_tunnel_offload_set(res->id,
+                                                 &entry,
+                                                 ETH_L2_TUNNEL_ENABLE_MASK,
+                                                 1);
+       else
+               rte_eth_dev_l2_tunnel_offload_set(res->id,
+                                                 &entry,
+                                                 ETH_L2_TUNNEL_ENABLE_MASK,
+                                                 0);
+}
+
+cmdline_parse_inst_t cmd_config_l2_tunnel_en_dis_specific = {
+       .f = cmd_config_l2_tunnel_en_dis_specific_parsed,
+       .data = NULL,
+       .help_str = "port config l2-tunnel enable/disable",
+       .tokens = {
+               (void *)&cmd_config_l2_tunnel_en_dis_port,
+               (void *)&cmd_config_l2_tunnel_en_dis_config,
+               (void *)&cmd_config_l2_tunnel_en_dis_id,
+               (void *)&cmd_config_l2_tunnel_en_dis_l2_tunnel,
+               (void *)&cmd_config_l2_tunnel_en_dis_l2_tunnel_type,
+               (void *)&cmd_config_l2_tunnel_en_dis_en_dis,
+               NULL,
+       },
+};
+
+/* E-tag configuration */
+
+/* Common result structure for all E-tag configuration */
+struct cmd_config_e_tag_result {
+       cmdline_fixed_string_t e_tag;
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t insertion;
+       cmdline_fixed_string_t stripping;
+       cmdline_fixed_string_t forwarding;
+       cmdline_fixed_string_t filter;
+       cmdline_fixed_string_t add;
+       cmdline_fixed_string_t del;
+       cmdline_fixed_string_t on;
+       cmdline_fixed_string_t off;
+       cmdline_fixed_string_t on_off;
+       cmdline_fixed_string_t port_tag_id;
+       uint32_t port_tag_id_val;
+       cmdline_fixed_string_t e_tag_id;
+       uint16_t e_tag_id_val;
+       cmdline_fixed_string_t dst_pool;
+       uint8_t dst_pool_val;
+       cmdline_fixed_string_t port;
+       uint8_t port_id;
+       cmdline_fixed_string_t vf;
+       uint8_t vf_id;
+};
+
+/* Common CLI fields for all E-tag configuration */
+cmdline_parse_token_string_t cmd_config_e_tag_e_tag =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                e_tag, "E-tag");
+cmdline_parse_token_string_t cmd_config_e_tag_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_config_e_tag_insertion =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                insertion, "insertion");
+cmdline_parse_token_string_t cmd_config_e_tag_stripping =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                stripping, "stripping");
+cmdline_parse_token_string_t cmd_config_e_tag_forwarding =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                forwarding, "forwarding");
+cmdline_parse_token_string_t cmd_config_e_tag_filter =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                filter, "filter");
+cmdline_parse_token_string_t cmd_config_e_tag_add =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                add, "add");
+cmdline_parse_token_string_t cmd_config_e_tag_del =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                del, "del");
+cmdline_parse_token_string_t cmd_config_e_tag_on =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                on, "on");
+cmdline_parse_token_string_t cmd_config_e_tag_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                off, "off");
+cmdline_parse_token_string_t cmd_config_e_tag_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                on_off, "on#off");
+cmdline_parse_token_string_t cmd_config_e_tag_port_tag_id =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                port_tag_id, "port-tag-id");
+cmdline_parse_token_num_t cmd_config_e_tag_port_tag_id_val =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                port_tag_id_val, UINT32);
+cmdline_parse_token_string_t cmd_config_e_tag_e_tag_id =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                e_tag_id, "e-tag-id");
+cmdline_parse_token_num_t cmd_config_e_tag_e_tag_id_val =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                e_tag_id_val, UINT16);
+cmdline_parse_token_string_t cmd_config_e_tag_dst_pool =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                dst_pool, "dst-pool");
+cmdline_parse_token_num_t cmd_config_e_tag_dst_pool_val =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                dst_pool_val, UINT8);
+cmdline_parse_token_string_t cmd_config_e_tag_port =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                port, "port");
+cmdline_parse_token_num_t cmd_config_e_tag_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                port_id, UINT8);
+cmdline_parse_token_string_t cmd_config_e_tag_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                vf, "vf");
+cmdline_parse_token_num_t cmd_config_e_tag_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_config_e_tag_result,
+                vf_id, UINT8);
+
+/* E-tag insertion configuration */
+static void
+cmd_config_e_tag_insertion_en_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res =
+               parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+       entry.tunnel_id = res->port_tag_id_val;
+       entry.vf_id = res->vf_id;
+       rte_eth_dev_l2_tunnel_offload_set(res->port_id,
+                                         &entry,
+                                         ETH_L2_TUNNEL_INSERTION_MASK,
+                                         1);
+}
+
+static void
+cmd_config_e_tag_insertion_dis_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res =
+               parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+       entry.vf_id = res->vf_id;
+
+       rte_eth_dev_l2_tunnel_offload_set(res->port_id,
+                                         &entry,
+                                         ETH_L2_TUNNEL_INSERTION_MASK,
+                                         0);
+}
+
+cmdline_parse_inst_t cmd_config_e_tag_insertion_en = {
+       .f = cmd_config_e_tag_insertion_en_parsed,
+       .data = NULL,
+       .help_str = "E-tag insertion enable",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_insertion,
+               (void *)&cmd_config_e_tag_on,
+               (void *)&cmd_config_e_tag_port_tag_id,
+               (void *)&cmd_config_e_tag_port_tag_id_val,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               (void *)&cmd_config_e_tag_vf,
+               (void *)&cmd_config_e_tag_vf_id,
+               NULL,
+       },
+};
+
+cmdline_parse_inst_t cmd_config_e_tag_insertion_dis = {
+       .f = cmd_config_e_tag_insertion_dis_parsed,
+       .data = NULL,
+       .help_str = "E-tag insertion disable",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_insertion,
+               (void *)&cmd_config_e_tag_off,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               (void *)&cmd_config_e_tag_vf,
+               (void *)&cmd_config_e_tag_vf_id,
+               NULL,
+       },
+};
+
+/* E-tag stripping configuration */
+static void
+cmd_config_e_tag_stripping_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res =
+               parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+
+       if (!strcmp(res->on_off, "on"))
+               rte_eth_dev_l2_tunnel_offload_set
+                       (res->port_id,
+                        &entry,
+                        ETH_L2_TUNNEL_STRIPPING_MASK,
+                        1);
+       else
+               rte_eth_dev_l2_tunnel_offload_set
+                       (res->port_id,
+                        &entry,
+                        ETH_L2_TUNNEL_STRIPPING_MASK,
+                        0);
+}
+
+cmdline_parse_inst_t cmd_config_e_tag_stripping_en_dis = {
+       .f = cmd_config_e_tag_stripping_parsed,
+       .data = NULL,
+       .help_str = "E-tag stripping enable/disable",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_stripping,
+               (void *)&cmd_config_e_tag_on_off,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               NULL,
+       },
+};
+
+/* E-tag forwarding configuration */
+static void
+cmd_config_e_tag_forwarding_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res = parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+
+       if (!strcmp(res->on_off, "on"))
+               rte_eth_dev_l2_tunnel_offload_set
+                       (res->port_id,
+                        &entry,
+                        ETH_L2_TUNNEL_FORWARDING_MASK,
+                        1);
+       else
+               rte_eth_dev_l2_tunnel_offload_set
+                       (res->port_id,
+                        &entry,
+                        ETH_L2_TUNNEL_FORWARDING_MASK,
+                        0);
+}
+
+cmdline_parse_inst_t cmd_config_e_tag_forwarding_en_dis = {
+       .f = cmd_config_e_tag_forwarding_parsed,
+       .data = NULL,
+       .help_str = "E-tag forwarding enable/disable",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_forwarding,
+               (void *)&cmd_config_e_tag_on_off,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               NULL,
+       },
+};
+
+/* E-tag filter configuration */
+static void
+cmd_config_e_tag_filter_add_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res = parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+       int ret = 0;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       if (res->e_tag_id_val > 0x3fff) {
+               printf("e-tag-id must be equal or less than 0x3fff.\n");
+               return;
+       }
+
+       ret = rte_eth_dev_filter_supported(res->port_id,
+                                          RTE_ETH_FILTER_L2_TUNNEL);
+       if (ret < 0) {
+               printf("E-tag filter is not supported on port %u.\n",
+                      res->port_id);
+               return;
+       }
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+       entry.tunnel_id = res->e_tag_id_val;
+       entry.pool = res->dst_pool_val;
+
+       ret = rte_eth_dev_filter_ctrl(res->port_id,
+                                     RTE_ETH_FILTER_L2_TUNNEL,
+                                     RTE_ETH_FILTER_ADD,
+                                     &entry);
+       if (ret < 0)
+               printf("E-tag filter programming error: (%s)\n",
+                      strerror(-ret));
+}
+
+cmdline_parse_inst_t cmd_config_e_tag_filter_add = {
+       .f = cmd_config_e_tag_filter_add_parsed,
+       .data = NULL,
+       .help_str = "E-tag filter add",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_filter,
+               (void *)&cmd_config_e_tag_add,
+               (void *)&cmd_config_e_tag_e_tag_id,
+               (void *)&cmd_config_e_tag_e_tag_id_val,
+               (void *)&cmd_config_e_tag_dst_pool,
+               (void *)&cmd_config_e_tag_dst_pool_val,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               NULL,
+       },
+};
+
+static void
+cmd_config_e_tag_filter_del_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_config_e_tag_result *res = parsed_result;
+       struct rte_eth_l2_tunnel_conf entry;
+       int ret = 0;
+
+       if (port_id_is_invalid(res->port_id, ENABLED_WARN))
+               return;
+
+       if (res->e_tag_id_val > 0x3fff) {
+               printf("e-tag-id must be less than 0x3fff.\n");
+               return;
+       }
+
+       ret = rte_eth_dev_filter_supported(res->port_id,
+                                          RTE_ETH_FILTER_L2_TUNNEL);
+       if (ret < 0) {
+               printf("E-tag filter is not supported on port %u.\n",
+                      res->port_id);
+               return;
+       }
+
+       entry.l2_tunnel_type = RTE_L2_TUNNEL_TYPE_E_TAG;
+       entry.tunnel_id = res->e_tag_id_val;
+
+       ret = rte_eth_dev_filter_ctrl(res->port_id,
+                                     RTE_ETH_FILTER_L2_TUNNEL,
+                                     RTE_ETH_FILTER_DELETE,
+                                     &entry);
+       if (ret < 0)
+               printf("E-tag filter programming error: (%s)\n",
+                      strerror(-ret));
+}
+
+cmdline_parse_inst_t cmd_config_e_tag_filter_del = {
+       .f = cmd_config_e_tag_filter_del_parsed,
+       .data = NULL,
+       .help_str = "E-tag filter delete",
+       .tokens = {
+               (void *)&cmd_config_e_tag_e_tag,
+               (void *)&cmd_config_e_tag_set,
+               (void *)&cmd_config_e_tag_filter,
+               (void *)&cmd_config_e_tag_del,
+               (void *)&cmd_config_e_tag_e_tag_id,
+               (void *)&cmd_config_e_tag_e_tag_id_val,
+               (void *)&cmd_config_e_tag_port,
+               (void *)&cmd_config_e_tag_port_id,
+               NULL,
+       },
+};
+#ifdef RTE_LIBRTE_IXGBE_PMD
+
+/* vf vlan anti spoof configuration */
+
+/* Common result structure for vf vlan anti spoof */
+struct cmd_vf_vlan_anti_spoof_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t vlan;
+       cmdline_fixed_string_t antispoof;
+       uint8_t port_id;
+       uint32_t vf_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for vf vlan anti spoof enable disable */
+cmdline_parse_token_string_t cmd_vf_vlan_anti_spoof_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_vf_vlan_anti_spoof_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_vf_vlan_anti_spoof_vlan =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                vlan, "vlan");
+cmdline_parse_token_string_t cmd_vf_vlan_anti_spoof_antispoof =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                antispoof, "antispoof");
+cmdline_parse_token_num_t cmd_vf_vlan_anti_spoof_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_vf_vlan_anti_spoof_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                vf_id, UINT32);
+cmdline_parse_token_string_t cmd_vf_vlan_anti_spoof_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_anti_spoof_result,
+                on_off, "on#off");
+
+static void
+cmd_set_vf_vlan_anti_spoof_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_vf_vlan_anti_spoof_result *res = parsed_result;
+       int ret = 0;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_vf_vlan_anti_spoof(res->port_id, res->vf_id,
+                       is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d\n", res->vf_id);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_vlan_anti_spoof = {
+       .f = cmd_set_vf_vlan_anti_spoof_parsed,
+       .data = NULL,
+       .help_str = "set vf vlan antispoof port_id vf_id on|off",
+       .tokens = {
+               (void *)&cmd_vf_vlan_anti_spoof_set,
+               (void *)&cmd_vf_vlan_anti_spoof_vf,
+               (void *)&cmd_vf_vlan_anti_spoof_vlan,
+               (void *)&cmd_vf_vlan_anti_spoof_antispoof,
+               (void *)&cmd_vf_vlan_anti_spoof_port_id,
+               (void *)&cmd_vf_vlan_anti_spoof_vf_id,
+               (void *)&cmd_vf_vlan_anti_spoof_on_off,
+               NULL,
+       },
+};
+
+/* vf mac anti spoof configuration */
+
+/* Common result structure for vf mac anti spoof */
+struct cmd_vf_mac_anti_spoof_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t mac;
+       cmdline_fixed_string_t antispoof;
+       uint8_t port_id;
+       uint32_t vf_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for vf mac anti spoof enable disable */
+cmdline_parse_token_string_t cmd_vf_mac_anti_spoof_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_vf_mac_anti_spoof_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_vf_mac_anti_spoof_mac =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                mac, "mac");
+cmdline_parse_token_string_t cmd_vf_mac_anti_spoof_antispoof =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                antispoof, "antispoof");
+cmdline_parse_token_num_t cmd_vf_mac_anti_spoof_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_vf_mac_anti_spoof_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                vf_id, UINT32);
+cmdline_parse_token_string_t cmd_vf_mac_anti_spoof_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_mac_anti_spoof_result,
+                on_off, "on#off");
+
+static void
+cmd_set_vf_mac_anti_spoof_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_vf_mac_anti_spoof_result *res = parsed_result;
+       int ret;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_vf_mac_anti_spoof(res->port_id, res->vf_id,
+                       is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d or is_on %d\n", res->vf_id, is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_mac_anti_spoof = {
+       .f = cmd_set_vf_mac_anti_spoof_parsed,
+       .data = NULL,
+       .help_str = "set vf mac antispoof port_id vf_id on|off",
+       .tokens = {
+               (void *)&cmd_vf_mac_anti_spoof_set,
+               (void *)&cmd_vf_mac_anti_spoof_vf,
+               (void *)&cmd_vf_mac_anti_spoof_mac,
+               (void *)&cmd_vf_mac_anti_spoof_antispoof,
+               (void *)&cmd_vf_mac_anti_spoof_port_id,
+               (void *)&cmd_vf_mac_anti_spoof_vf_id,
+               (void *)&cmd_vf_mac_anti_spoof_on_off,
+               NULL,
+       },
+};
+
+/* vf vlan strip queue configuration */
+
+/* Common result structure for vf mac anti spoof */
+struct cmd_vf_vlan_stripq_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t vlan;
+       cmdline_fixed_string_t stripq;
+       uint8_t port_id;
+       uint16_t vf_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for vf vlan strip enable disable */
+cmdline_parse_token_string_t cmd_vf_vlan_stripq_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_vf_vlan_stripq_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_vf_vlan_stripq_vlan =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                vlan, "vlan");
+cmdline_parse_token_string_t cmd_vf_vlan_stripq_stripq =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                stripq, "stripq");
+cmdline_parse_token_num_t cmd_vf_vlan_stripq_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_vf_vlan_stripq_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                vf_id, UINT16);
+cmdline_parse_token_string_t cmd_vf_vlan_stripq_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_stripq_result,
+                on_off, "on#off");
+
+static void
+cmd_set_vf_vlan_stripq_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_vf_vlan_stripq_result *res = parsed_result;
+       int ret = 0;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_vf_vlan_stripq(res->port_id, res->vf_id, is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d or is_on %d\n", res->vf_id, is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_vlan_stripq = {
+       .f = cmd_set_vf_vlan_stripq_parsed,
+       .data = NULL,
+       .help_str = "set vf vlan stripq port_id vf_id on|off",
+       .tokens = {
+               (void *)&cmd_vf_vlan_stripq_set,
+               (void *)&cmd_vf_vlan_stripq_vf,
+               (void *)&cmd_vf_vlan_stripq_vlan,
+               (void *)&cmd_vf_vlan_stripq_stripq,
+               (void *)&cmd_vf_vlan_stripq_port_id,
+               (void *)&cmd_vf_vlan_stripq_vf_id,
+               (void *)&cmd_vf_vlan_stripq_on_off,
+               NULL,
+       },
+};
+
+/* vf vlan insert configuration */
+
+/* Common result structure for vf vlan insert */
+struct cmd_vf_vlan_insert_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t vlan;
+       cmdline_fixed_string_t insert;
+       uint8_t port_id;
+       uint16_t vf_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for vf vlan insert enable disable */
+cmdline_parse_token_string_t cmd_vf_vlan_insert_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_vf_vlan_insert_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_vf_vlan_insert_vlan =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                vlan, "vlan");
+cmdline_parse_token_string_t cmd_vf_vlan_insert_insert =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                insert, "insert");
+cmdline_parse_token_num_t cmd_vf_vlan_insert_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_vf_vlan_insert_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                vf_id, UINT16);
+cmdline_parse_token_string_t cmd_vf_vlan_insert_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_vlan_insert_result,
+                on_off, "on#off");
+
+static void
+cmd_set_vf_vlan_insert_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_vf_vlan_insert_result *res = parsed_result;
+       int ret;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_vf_vlan_insert(res->port_id, res->vf_id, is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d or is_on %d\n", res->vf_id, is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_vlan_insert = {
+       .f = cmd_set_vf_vlan_insert_parsed,
+       .data = NULL,
+       .help_str = "set vf vlan insert port_id vf_id on|off",
+       .tokens = {
+               (void *)&cmd_vf_vlan_insert_set,
+               (void *)&cmd_vf_vlan_insert_vf,
+               (void *)&cmd_vf_vlan_insert_vlan,
+               (void *)&cmd_vf_vlan_insert_insert,
+               (void *)&cmd_vf_vlan_insert_port_id,
+               (void *)&cmd_vf_vlan_insert_vf_id,
+               (void *)&cmd_vf_vlan_insert_on_off,
+               NULL,
+       },
+};
+
+/* tx loopback configuration */
+
+/* Common result structure for tx loopback */
+struct cmd_tx_loopback_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t tx;
+       cmdline_fixed_string_t loopback;
+       uint8_t port_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for tx loopback enable disable */
+cmdline_parse_token_string_t cmd_tx_loopback_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_tx_loopback_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_tx_loopback_tx =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_tx_loopback_result,
+                tx, "tx");
+cmdline_parse_token_string_t cmd_tx_loopback_loopback =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_tx_loopback_result,
+                loopback, "loopback");
+cmdline_parse_token_num_t cmd_tx_loopback_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_tx_loopback_result,
+                port_id, UINT8);
+cmdline_parse_token_string_t cmd_tx_loopback_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_tx_loopback_result,
+                on_off, "on#off");
+
+static void
+cmd_set_tx_loopback_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_tx_loopback_result *res = parsed_result;
+       int ret;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_tx_loopback(res->port_id, is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid is_on %d\n", is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_tx_loopback = {
+       .f = cmd_set_tx_loopback_parsed,
+       .data = NULL,
+       .help_str = "set tx loopback port_id on|off",
+       .tokens = {
+               (void *)&cmd_tx_loopback_set,
+               (void *)&cmd_tx_loopback_tx,
+               (void *)&cmd_tx_loopback_loopback,
+               (void *)&cmd_tx_loopback_port_id,
+               (void *)&cmd_tx_loopback_on_off,
+               NULL,
+       },
+};
+
+/* all queues drop enable configuration */
+
+/* Common result structure for all queues drop enable */
+struct cmd_all_queues_drop_en_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t all;
+       cmdline_fixed_string_t queues;
+       cmdline_fixed_string_t drop;
+       uint8_t port_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for tx loopback enable disable */
+cmdline_parse_token_string_t cmd_all_queues_drop_en_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_all_queues_drop_en_all =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                all, "all");
+cmdline_parse_token_string_t cmd_all_queues_drop_en_queues =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                queues, "queues");
+cmdline_parse_token_string_t cmd_all_queues_drop_en_drop =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                drop, "drop");
+cmdline_parse_token_num_t cmd_all_queues_drop_en_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                port_id, UINT8);
+cmdline_parse_token_string_t cmd_all_queues_drop_en_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_all_queues_drop_en_result,
+                on_off, "on#off");
+
+static void
+cmd_set_all_queues_drop_en_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_all_queues_drop_en_result *res = parsed_result;
+       int ret = 0;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_all_queues_drop_en(res->port_id, is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid is_on %d\n", is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_all_queues_drop_en = {
+       .f = cmd_set_all_queues_drop_en_parsed,
+       .data = NULL,
+       .help_str = "set all queues drop port_id on|off",
+       .tokens = {
+               (void *)&cmd_all_queues_drop_en_set,
+               (void *)&cmd_all_queues_drop_en_all,
+               (void *)&cmd_all_queues_drop_en_queues,
+               (void *)&cmd_all_queues_drop_en_drop,
+               (void *)&cmd_all_queues_drop_en_port_id,
+               (void *)&cmd_all_queues_drop_en_on_off,
+               NULL,
+       },
+};
+
+/* vf split drop enable configuration */
+
+/* Common result structure for vf split drop enable */
+struct cmd_vf_split_drop_en_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t split;
+       cmdline_fixed_string_t drop;
+       uint8_t port_id;
+       uint16_t vf_id;
+       cmdline_fixed_string_t on_off;
+};
+
+/* Common CLI fields for vf split drop enable disable */
+cmdline_parse_token_string_t cmd_vf_split_drop_en_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_vf_split_drop_en_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_vf_split_drop_en_split =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                split, "split");
+cmdline_parse_token_string_t cmd_vf_split_drop_en_drop =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                drop, "drop");
+cmdline_parse_token_num_t cmd_vf_split_drop_en_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_vf_split_drop_en_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                vf_id, UINT16);
+cmdline_parse_token_string_t cmd_vf_split_drop_en_on_off =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_vf_split_drop_en_result,
+                on_off, "on#off");
+
+static void
+cmd_set_vf_split_drop_en_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_vf_split_drop_en_result *res = parsed_result;
+       int ret;
+       int is_on = (strcmp(res->on_off, "on") == 0) ? 1 : 0;
+
+       ret = rte_pmd_ixgbe_set_vf_split_drop_en(res->port_id, res->vf_id,
+                       is_on);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d or is_on %d\n", res->vf_id, is_on);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_split_drop_en = {
+       .f = cmd_set_vf_split_drop_en_parsed,
+       .data = NULL,
+       .help_str = "set vf split drop port_id vf_id on|off",
+       .tokens = {
+               (void *)&cmd_vf_split_drop_en_set,
+               (void *)&cmd_vf_split_drop_en_vf,
+               (void *)&cmd_vf_split_drop_en_split,
+               (void *)&cmd_vf_split_drop_en_drop,
+               (void *)&cmd_vf_split_drop_en_port_id,
+               (void *)&cmd_vf_split_drop_en_vf_id,
+               (void *)&cmd_vf_split_drop_en_on_off,
+               NULL,
+       },
+};
+
+/* vf mac address configuration */
+
+/* Common result structure for vf mac address */
+struct cmd_set_vf_mac_addr_result {
+       cmdline_fixed_string_t set;
+       cmdline_fixed_string_t vf;
+       cmdline_fixed_string_t mac;
+       cmdline_fixed_string_t addr;
+       uint8_t port_id;
+       uint16_t vf_id;
+       struct ether_addr mac_addr;
+
+};
+
+/* Common CLI fields for vf split drop enable disable */
+cmdline_parse_token_string_t cmd_set_vf_mac_addr_set =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                set, "set");
+cmdline_parse_token_string_t cmd_set_vf_mac_addr_vf =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                vf, "vf");
+cmdline_parse_token_string_t cmd_set_vf_mac_addr_mac =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                mac, "mac");
+cmdline_parse_token_string_t cmd_set_vf_mac_addr_addr =
+       TOKEN_STRING_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                addr, "addr");
+cmdline_parse_token_num_t cmd_set_vf_mac_addr_port_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                port_id, UINT8);
+cmdline_parse_token_num_t cmd_set_vf_mac_addr_vf_id =
+       TOKEN_NUM_INITIALIZER
+               (struct cmd_set_vf_mac_addr_result,
+                vf_id, UINT16);
+cmdline_parse_token_etheraddr_t cmd_set_vf_mac_addr_mac_addr =
+       TOKEN_ETHERADDR_INITIALIZER(struct cmd_set_vf_mac_addr_result,
+                mac_addr);
+
+static void
+cmd_set_vf_mac_addr_parsed(
+       void *parsed_result,
+       __attribute__((unused)) struct cmdline *cl,
+       __attribute__((unused)) void *data)
+{
+       struct cmd_set_vf_mac_addr_result *res = parsed_result;
+       int ret;
+
+       ret = rte_pmd_ixgbe_set_vf_mac_addr(res->port_id, res->vf_id,
+                       &res->mac_addr);
+       switch (ret) {
+       case 0:
+               break;
+       case -EINVAL:
+               printf("invalid vf_id %d or mac_addr\n", res->vf_id);
+               break;
+       case -ENODEV:
+               printf("invalid port_id %d\n", res->port_id);
+               break;
+       default:
+               printf("programming error: (%s)\n", strerror(-ret));
+       }
+}
+
+cmdline_parse_inst_t cmd_set_vf_mac_addr = {
+       .f = cmd_set_vf_mac_addr_parsed,
+       .data = NULL,
+       .help_str = "set vf mac addr port_id vf_id xx:xx:xx:xx:xx:xx",
+       .tokens = {
+               (void *)&cmd_set_vf_mac_addr_set,
+               (void *)&cmd_set_vf_mac_addr_vf,
+               (void *)&cmd_set_vf_mac_addr_mac,
+               (void *)&cmd_set_vf_mac_addr_addr,
+               (void *)&cmd_set_vf_mac_addr_port_id,
+               (void *)&cmd_set_vf_mac_addr_vf_id,
+               (void *)&cmd_set_vf_mac_addr_mac_addr,
+               NULL,
+       },
+};
+#endif
+
+/* ******************************************************************************** */
+
+/* list of instructions */
+cmdline_parse_ctx_t main_ctx[] = {
+       (cmdline_parse_inst_t *)&cmd_help_brief,
+       (cmdline_parse_inst_t *)&cmd_help_long,
+       (cmdline_parse_inst_t *)&cmd_quit,
+       (cmdline_parse_inst_t *)&cmd_showport,
+       (cmdline_parse_inst_t *)&cmd_showqueue,
+       (cmdline_parse_inst_t *)&cmd_showportall,
+       (cmdline_parse_inst_t *)&cmd_showcfg,
+       (cmdline_parse_inst_t *)&cmd_start,
+       (cmdline_parse_inst_t *)&cmd_start_tx_first,
+       (cmdline_parse_inst_t *)&cmd_start_tx_first_n,
+       (cmdline_parse_inst_t *)&cmd_set_link_up,
+       (cmdline_parse_inst_t *)&cmd_set_link_down,
+       (cmdline_parse_inst_t *)&cmd_reset,
+       (cmdline_parse_inst_t *)&cmd_set_numbers,
+       (cmdline_parse_inst_t *)&cmd_set_txpkts,
+       (cmdline_parse_inst_t *)&cmd_set_txsplit,
+       (cmdline_parse_inst_t *)&cmd_set_fwd_list,
+       (cmdline_parse_inst_t *)&cmd_set_fwd_mask,
+       (cmdline_parse_inst_t *)&cmd_set_fwd_mode,
+       (cmdline_parse_inst_t *)&cmd_set_fwd_retry_mode,
+       (cmdline_parse_inst_t *)&cmd_set_burst_tx_retry,
+       (cmdline_parse_inst_t *)&cmd_set_promisc_mode_one,
+       (cmdline_parse_inst_t *)&cmd_set_promisc_mode_all,
+       (cmdline_parse_inst_t *)&cmd_set_allmulti_mode_one,
+       (cmdline_parse_inst_t *)&cmd_set_allmulti_mode_all,
+       (cmdline_parse_inst_t *)&cmd_set_flush_rx,
+       (cmdline_parse_inst_t *)&cmd_set_link_check,
+#ifdef RTE_NIC_BYPASS
+       (cmdline_parse_inst_t *)&cmd_set_bypass_mode,
+       (cmdline_parse_inst_t *)&cmd_set_bypass_event,
+       (cmdline_parse_inst_t *)&cmd_set_bypass_timeout,
+       (cmdline_parse_inst_t *)&cmd_show_bypass_config,
+#endif
+#ifdef RTE_LIBRTE_PMD_BOND
+       (cmdline_parse_inst_t *) &cmd_set_bonding_mode,
+       (cmdline_parse_inst_t *) &cmd_show_bonding_config,
+       (cmdline_parse_inst_t *) &cmd_set_bonding_primary,
+       (cmdline_parse_inst_t *) &cmd_add_bonding_slave,
+       (cmdline_parse_inst_t *) &cmd_remove_bonding_slave,
+       (cmdline_parse_inst_t *) &cmd_create_bonded_device,
+       (cmdline_parse_inst_t *) &cmd_set_bond_mac_addr,
+       (cmdline_parse_inst_t *) &cmd_set_balance_xmit_policy,
+       (cmdline_parse_inst_t *) &cmd_set_bond_mon_period,
+#endif
+       (cmdline_parse_inst_t *)&cmd_vlan_offload,
+       (cmdline_parse_inst_t *)&cmd_vlan_tpid,
+       (cmdline_parse_inst_t *)&cmd_rx_vlan_filter_all,
+       (cmdline_parse_inst_t *)&cmd_rx_vlan_filter,
+       (cmdline_parse_inst_t *)&cmd_tx_vlan_set,
+       (cmdline_parse_inst_t *)&cmd_tx_vlan_set_qinq,
+       (cmdline_parse_inst_t *)&cmd_tx_vlan_reset,
+       (cmdline_parse_inst_t *)&cmd_tx_vlan_set_pvid,
+       (cmdline_parse_inst_t *)&cmd_csum_set,
+       (cmdline_parse_inst_t *)&cmd_csum_show,
+       (cmdline_parse_inst_t *)&cmd_csum_tunnel,
+       (cmdline_parse_inst_t *)&cmd_tso_set,
+       (cmdline_parse_inst_t *)&cmd_tso_show,
+       (cmdline_parse_inst_t *)&cmd_tunnel_tso_set,
+       (cmdline_parse_inst_t *)&cmd_tunnel_tso_show,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_rx,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_tx,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_hw,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_lw,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_pt,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_xon,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_macfwd,
+       (cmdline_parse_inst_t *)&cmd_link_flow_control_set_autoneg,
+       (cmdline_parse_inst_t *)&cmd_priority_flow_control_set,
+       (cmdline_parse_inst_t *)&cmd_config_dcb,
+       (cmdline_parse_inst_t *)&cmd_read_reg,
+       (cmdline_parse_inst_t *)&cmd_read_reg_bit_field,
+       (cmdline_parse_inst_t *)&cmd_read_reg_bit,
+       (cmdline_parse_inst_t *)&cmd_write_reg,
+       (cmdline_parse_inst_t *)&cmd_write_reg_bit_field,
+       (cmdline_parse_inst_t *)&cmd_write_reg_bit,
+       (cmdline_parse_inst_t *)&cmd_read_rxd_txd,
+       (cmdline_parse_inst_t *)&cmd_stop,
+       (cmdline_parse_inst_t *)&cmd_mac_addr,
+       (cmdline_parse_inst_t *)&cmd_set_qmap,
+       (cmdline_parse_inst_t *)&cmd_operate_port,
+       (cmdline_parse_inst_t *)&cmd_operate_specific_port,
+       (cmdline_parse_inst_t *)&cmd_operate_attach_port,
+       (cmdline_parse_inst_t *)&cmd_operate_detach_port,
+       (cmdline_parse_inst_t *)&cmd_config_speed_all,
+       (cmdline_parse_inst_t *)&cmd_config_speed_specific,
+       (cmdline_parse_inst_t *)&cmd_config_rx_tx,
+       (cmdline_parse_inst_t *)&cmd_config_mtu,
+       (cmdline_parse_inst_t *)&cmd_config_max_pkt_len,
+       (cmdline_parse_inst_t *)&cmd_config_rx_mode_flag,
        (cmdline_parse_inst_t *)&cmd_config_rss,
        (cmdline_parse_inst_t *)&cmd_config_rxtx_queue,
+       (cmdline_parse_inst_t *)&cmd_config_txqflags,
        (cmdline_parse_inst_t *)&cmd_config_rss_reta,
        (cmdline_parse_inst_t *)&cmd_showport_reta,
        (cmdline_parse_inst_t *)&cmd_config_burst,
@@ -9363,6 +11525,7 @@ cmdline_parse_ctx_t main_ctx[] = {
        (cmdline_parse_inst_t *)&cmd_vf_rate_limit,
        (cmdline_parse_inst_t *)&cmd_tunnel_filter,
        (cmdline_parse_inst_t *)&cmd_tunnel_udp_config,
+       (cmdline_parse_inst_t *)&cmd_global_config,
        (cmdline_parse_inst_t *)&cmd_set_mirror_mask,
        (cmdline_parse_inst_t *)&cmd_set_mirror_link,
        (cmdline_parse_inst_t *)&cmd_reset_mirror_rule,
@@ -9392,7 +11555,29 @@ cmdline_parse_ctx_t main_ctx[] = {
        (cmdline_parse_inst_t *)&cmd_set_sym_hash_ena_per_port,
        (cmdline_parse_inst_t *)&cmd_get_hash_global_config,
        (cmdline_parse_inst_t *)&cmd_set_hash_global_config,
+       (cmdline_parse_inst_t *)&cmd_set_hash_input_set,
+       (cmdline_parse_inst_t *)&cmd_set_fdir_input_set,
        (cmdline_parse_inst_t *)&cmd_mcast_addr,
+       (cmdline_parse_inst_t *)&cmd_config_l2_tunnel_eth_type_all,
+       (cmdline_parse_inst_t *)&cmd_config_l2_tunnel_eth_type_specific,
+       (cmdline_parse_inst_t *)&cmd_config_l2_tunnel_en_dis_all,
+       (cmdline_parse_inst_t *)&cmd_config_l2_tunnel_en_dis_specific,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_insertion_en,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_insertion_dis,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_stripping_en_dis,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_forwarding_en_dis,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_filter_add,
+       (cmdline_parse_inst_t *)&cmd_config_e_tag_filter_del,
+#ifdef RTE_LIBRTE_IXGBE_PMD
+       (cmdline_parse_inst_t *)&cmd_set_vf_vlan_anti_spoof,
+       (cmdline_parse_inst_t *)&cmd_set_vf_mac_anti_spoof,
+       (cmdline_parse_inst_t *)&cmd_set_vf_vlan_stripq,
+       (cmdline_parse_inst_t *)&cmd_set_vf_vlan_insert,
+       (cmdline_parse_inst_t *)&cmd_set_tx_loopback,
+       (cmdline_parse_inst_t *)&cmd_set_all_queues_drop_en,
+       (cmdline_parse_inst_t *)&cmd_set_vf_split_drop_en,
+       (cmdline_parse_inst_t *)&cmd_set_vf_mac_addr,
+#endif
        NULL,
 };
 
@@ -9400,17 +11585,22 @@ cmdline_parse_ctx_t main_ctx[] = {
 void
 prompt(void)
 {
-       struct cmdline *cl;
-
        /* initialize non-constant commands */
        cmd_set_fwd_mode_init();
+       cmd_set_fwd_retry_mode_init();
 
-       cl = cmdline_stdin_new(main_ctx, "testpmd> ");
-       if (cl == NULL) {
+       testpmd_cl = cmdline_stdin_new(main_ctx, "testpmd> ");
+       if (testpmd_cl == NULL)
                return;
-       }
-       cmdline_interact(cl);
-       cmdline_stdin_exit(cl);
+       cmdline_interact(testpmd_cl);
+       cmdline_stdin_exit(testpmd_cl);
+}
+
+void
+prompt_exit(void)
+{
+       if (testpmd_cl != NULL)
+               cmdline_quit(testpmd_cl);
 }
 
 static void
@@ -9436,29 +11626,3 @@ cmd_reconfig_device_queue(portid_t id, uint8_t dev, uint8_t queue)
                        ports[id].need_reconfig_queues = queue;
        }
 }
-
-#ifdef RTE_NIC_BYPASS
-#include <rte_pci_dev_ids.h>
-uint8_t
-bypass_is_supported(portid_t port_id)
-{
-       struct rte_port   *port;
-       struct rte_pci_id *pci_id;
-
-       if (port_id_is_invalid(port_id, ENABLED_WARN))
-               return 0;
-
-       /* Get the device id. */
-       port    = &ports[port_id];
-       pci_id = &port->dev_info.pci_dev->id;
-
-       /* Check if NIC supports bypass. */
-       if (pci_id->device_id == IXGBE_DEV_ID_82599_BYPASS) {
-               return 1;
-       }
-       else {
-               printf("\tBypass not supported for port_id = %d.\n", port_id);
-               return 0;
-       }
-}
-#endif