app/flow-perf: configure rule batches
[dpdk.git] / app / test-flow-perf / main.c
index 1b456e6..e2fc5b7 100644 (file)
@@ -40,7 +40,8 @@
 
 #define MAX_ITERATIONS             100
 #define DEFAULT_RULES_COUNT    4000000
-#define DEFAULT_ITERATION       100000
+#define DEFAULT_RULES_BATCH     100000
+#define DEFAULT_GROUP                0
 
 struct rte_flow *flow;
 static uint8_t flow_group;
@@ -53,6 +54,7 @@ static uint64_t flow_actions[MAX_ACTIONS_NUM];
 static uint64_t flow_attrs[MAX_ATTRS_NUM];
 static uint8_t items_idx, actions_idx, attrs_idx;
 
+static uint64_t ports_mask;
 static volatile bool force_quit;
 static bool dump_iterations;
 static bool delete_flag;
@@ -61,8 +63,8 @@ static bool enable_fwd;
 
 static struct rte_mempool *mbuf_mp;
 static uint32_t nb_lcores;
-static uint32_t flows_count;
-static uint32_t iterations_number;
+static uint32_t rules_count;
+static uint32_t rules_batch;
 static uint32_t hairpin_queues_num; /* total hairpin q number - default: 0 */
 static uint32_t nb_lcores;
 
@@ -97,8 +99,10 @@ usage(char *progname)
 {
        printf("\nusage: %s\n", progname);
        printf("\nControl configurations:\n");
-       printf("  --flows-count=N: to set the number of needed"
-               " flows to insert, default is 4,000,000\n");
+       printf("  --rules-count=N: to set the number of needed"
+               " rules to insert, default is %d\n", DEFAULT_RULES_COUNT);
+       printf("  --rules-batch=N: set number of batched rules,"
+               " default is %d\n", DEFAULT_RULES_BATCH);
        printf("  --dump-iterations: To print rates for each"
                " iteration\n");
        printf("  --deletion-rate: Enable deletion rate"
@@ -106,13 +110,14 @@ usage(char *progname)
        printf("  --dump-socket-mem: To dump all socket memory\n");
        printf("  --enable-fwd: To enable packets forwarding"
                " after insertion\n");
+       printf("  --portmask=N: hexadecimal bitmask of ports used\n");
 
        printf("To set flow attributes:\n");
        printf("  --ingress: set ingress attribute in flows\n");
        printf("  --egress: set egress attribute in flows\n");
        printf("  --transfer: set transfer attribute in flows\n");
        printf("  --group=N: set group for all flows,"
-               " default is 0\n");
+               " default is %d\n", DEFAULT_GROUP);
 
        printf("To set flow items:\n");
        printf("  --ether: add ether layer in flow items\n");
@@ -128,6 +133,8 @@ usage(char *progname)
        printf("  --gtp: add gtp layer in flow items\n");
        printf("  --meta: add meta layer in flow items\n");
        printf("  --tag: add tag layer in flow items\n");
+       printf("  --icmpv4: add icmpv4 layer in flow items\n");
+       printf("  --icmpv6: add icmpv6 layer in flow items\n");
 
        printf("To set flow actions:\n");
        printf("  --port-id: add port-id action in flow actions\n");
@@ -180,13 +187,19 @@ usage(char *progname)
        printf("  --raw-decap=<data>: add raw decap action to flow actions\n"
                "Data is the data needed to be decaped\n"
                "Example: raw-decap=ether,ipv4,udp,vxlan\n");
+       printf("  --vxlan-encap: add vxlan-encap action to flow actions\n"
+               "Encapped data is fixed with pattern: ether,ipv4,udp,vxlan\n"
+               "With fixed values\n");
+       printf("  --vxlan-decap: add vxlan_decap action to flow actions\n");
 }
 
 static void
 args_parse(int argc, char **argv)
 {
+       uint64_t pm;
        char **argvopt;
        char *token;
+       char *end;
        int n, opt;
        int opt_idx;
        size_t i;
@@ -276,6 +289,18 @@ args_parse(int argc, char **argv)
                        .map = &flow_items[0],
                        .map_idx = &items_idx
                },
+               {
+                       .str = "icmpv4",
+                       .mask = FLOW_ITEM_MASK(RTE_FLOW_ITEM_TYPE_ICMP),
+                       .map = &flow_items[0],
+                       .map_idx = &items_idx
+               },
+               {
+                       .str = "icmpv6",
+                       .mask = FLOW_ITEM_MASK(RTE_FLOW_ITEM_TYPE_ICMP6),
+                       .map = &flow_items[0],
+                       .map_idx = &items_idx
+               },
                {
                        .str = "ingress",
                        .mask = INGRESS,
@@ -484,16 +509,34 @@ args_parse(int argc, char **argv)
                        .map = &flow_actions[0],
                        .map_idx = &actions_idx
                },
+               {
+                       .str = "vxlan-encap",
+                       .mask = FLOW_ACTION_MASK(
+                               RTE_FLOW_ACTION_TYPE_VXLAN_ENCAP
+                       ),
+                       .map = &flow_actions[0],
+                       .map_idx = &actions_idx
+               },
+               {
+                       .str = "vxlan-decap",
+                       .mask = FLOW_ACTION_MASK(
+                               RTE_FLOW_ACTION_TYPE_VXLAN_DECAP
+                       ),
+                       .map = &flow_actions[0],
+                       .map_idx = &actions_idx
+               },
        };
 
        static const struct option lgopts[] = {
                /* Control */
                { "help",                       0, 0, 0 },
-               { "flows-count",                1, 0, 0 },
+               { "rules-count",                1, 0, 0 },
+               { "rules-batch",                1, 0, 0 },
                { "dump-iterations",            0, 0, 0 },
                { "deletion-rate",              0, 0, 0 },
                { "dump-socket-mem",            0, 0, 0 },
                { "enable-fwd",                 0, 0, 0 },
+               { "portmask",                   1, 0, 0 },
                /* Attributes */
                { "ingress",                    0, 0, 0 },
                { "egress",                     0, 0, 0 },
@@ -513,6 +556,8 @@ args_parse(int argc, char **argv)
                { "gtp",                        0, 0, 0 },
                { "meta",                       0, 0, 0 },
                { "tag",                        0, 0, 0 },
+               { "icmpv4",                     0, 0, 0 },
+               { "icmpv6",                     0, 0, 0 },
                /* Actions */
                { "port-id",                    0, 0, 0 },
                { "rss",                        0, 0, 0 },
@@ -544,8 +589,13 @@ args_parse(int argc, char **argv)
                { "flag",                       0, 0, 0 },
                { "raw-encap",                  1, 0, 0 },
                { "raw-decap",                  1, 0, 0 },
+               { "vxlan-encap",                0, 0, 0 },
+               { "vxlan-decap",                0, 0, 0 },
        };
 
+       RTE_ETH_FOREACH_DEV(i)
+               ports_mask |= 1 << i;
+
        hairpin_queues_num = 0;
        argvopt = argv;
 
@@ -659,16 +709,26 @@ args_parse(int argc, char **argv)
                        }
                        /* Control */
                        if (strcmp(lgopts[opt_idx].name,
-                                       "flows-count") == 0) {
+                                       "rules-batch") == 0) {
                                n = atoi(optarg);
-                               if (n > (int) iterations_number)
-                                       flows_count = n;
+                               if (n >= DEFAULT_RULES_BATCH)
+                                       rules_batch = n;
                                else {
-                                       printf("\n\nflows_count should be > %d\n",
-                                               iterations_number);
+                                       printf("\n\nrules_batch should be >= %d\n",
+                                               DEFAULT_RULES_BATCH);
                                        rte_exit(EXIT_SUCCESS, " ");
                                }
                        }
+                       if (strcmp(lgopts[opt_idx].name,
+                                       "rules-count") == 0) {
+                               n = atoi(optarg);
+                               if (n >= (int) rules_batch)
+                                       rules_count = n;
+                               else {
+                                       printf("\n\nrules_count should be >= %d\n",
+                                               rules_batch);
+                               }
+                       }
                        if (strcmp(lgopts[opt_idx].name,
                                        "dump-iterations") == 0)
                                dump_iterations = true;
@@ -681,6 +741,15 @@ args_parse(int argc, char **argv)
                        if (strcmp(lgopts[opt_idx].name,
                                        "enable-fwd") == 0)
                                enable_fwd = true;
+                       if (strcmp(lgopts[opt_idx].name,
+                                       "portmask") == 0) {
+                               /* parse hexadecimal string */
+                               end = NULL;
+                               pm = strtoull(optarg, &end, 16);
+                               if ((optarg[0] == '\0') || (end == NULL) || (*end != '\0'))
+                                       rte_exit(EXIT_FAILURE, "Invalid fwd port mask\n");
+                               ports_mask = pm;
+                       }
                        break;
                default:
                        fprintf(stderr, "Invalid option: %s\n", argv[optind]);
@@ -771,13 +840,13 @@ destroy_flows(int port_id, struct rte_flow **flow_list)
        for (i = 0; i < MAX_ITERATIONS; i++)
                cpu_time_per_iter[i] = -1;
 
-       if (iterations_number > flows_count)
-               iterations_number = flows_count;
+       if (rules_batch > rules_count)
+               rules_batch = rules_count;
 
        /* Deletion Rate */
        printf("Flows Deletion on port = %d\n", port_id);
        start_iter = clock();
-       for (i = 0; i < flows_count; i++) {
+       for (i = 0; i < rules_count; i++) {
                if (flow_list[i] == 0)
                        break;
 
@@ -787,11 +856,11 @@ destroy_flows(int port_id, struct rte_flow **flow_list)
                        rte_exit(EXIT_FAILURE, "Error in deleting flow");
                }
 
-               if (i && !((i + 1) % iterations_number)) {
+               if (i && !((i + 1) % rules_batch)) {
                        /* Save the deletion rate of each iter */
                        end_iter = clock();
                        delta = (double) (end_iter - start_iter);
-                       iter_id = ((i + 1) / iterations_number) - 1;
+                       iter_id = ((i + 1) / rules_batch) - 1;
                        cpu_time_per_iter[iter_id] =
                                delta / CLOCKS_PER_SEC;
                        cpu_time_used += cpu_time_per_iter[iter_id];
@@ -804,21 +873,21 @@ destroy_flows(int port_id, struct rte_flow **flow_list)
                for (i = 0; i < MAX_ITERATIONS; i++) {
                        if (cpu_time_per_iter[i] == -1)
                                continue;
-                       delta = (double)(iterations_number /
+                       delta = (double)(rules_batch /
                                cpu_time_per_iter[i]);
                        flows_rate = delta / 1000;
                        printf(":: Iteration #%d: %d flows "
                                "in %f sec[ Rate = %f K/Sec ]\n",
-                               i, iterations_number,
+                               i, rules_batch,
                                cpu_time_per_iter[i], flows_rate);
                }
 
        /* Deletion rate for all flows */
-       flows_rate = ((double) (flows_count / cpu_time_used) / 1000);
+       flows_rate = ((double) (rules_count / cpu_time_used) / 1000);
        printf("\n:: Total flow deletion rate -> %f K/Sec\n",
                flows_rate);
        printf(":: The time for deleting %d in flows %f seconds\n",
-               flows_count, cpu_time_used);
+               rules_count, cpu_time_used);
 }
 
 static inline void
@@ -847,17 +916,20 @@ flows_handler(void)
        for (i = 0; i < MAX_ITERATIONS; i++)
                cpu_time_per_iter[i] = -1;
 
-       if (iterations_number > flows_count)
-               iterations_number = flows_count;
+       if (rules_batch > rules_count)
+               rules_batch = rules_count;
 
-       printf(":: Flows Count per port: %d\n", flows_count);
+       printf(":: Flows Count per port: %d\n", rules_count);
 
        flow_list = rte_zmalloc("flow_list",
-               (sizeof(struct rte_flow *) * flows_count) + 1, 0);
+               (sizeof(struct rte_flow *) * rules_count) + 1, 0);
        if (flow_list == NULL)
                rte_exit(EXIT_FAILURE, "No Memory available!");
 
        for (port_id = 0; port_id < nr_ports; port_id++) {
+               /* If port outside portmask */
+               if (!((ports_mask >> port_id) & 0x1))
+                       continue;
                cpu_time_used = 0;
                flow_index = 0;
                if (flow_group > 0) {
@@ -883,7 +955,7 @@ flows_handler(void)
                /* Insertion Rate */
                printf("Flows insertion on port = %d\n", port_id);
                start_iter = clock();
-               for (i = 0; i < flows_count; i++) {
+               for (i = 0; i < rules_count; i++) {
                        flow = generate_flow(port_id, flow_group,
                                flow_attrs, flow_items, flow_actions,
                                JUMP_ACTION_TABLE, i,
@@ -892,7 +964,7 @@ flows_handler(void)
                                &error);
 
                        if (force_quit)
-                               i = flows_count;
+                               i = rules_count;
 
                        if (!flow) {
                                print_flow_error(error);
@@ -901,11 +973,11 @@ flows_handler(void)
 
                        flow_list[flow_index++] = flow;
 
-                       if (i && !((i + 1) % iterations_number)) {
+                       if (i && !((i + 1) % rules_batch)) {
                                /* Save the insertion rate of each iter */
                                end_iter = clock();
                                delta = (double) (end_iter - start_iter);
-                               iter_id = ((i + 1) / iterations_number) - 1;
+                               iter_id = ((i + 1) / rules_batch) - 1;
                                cpu_time_per_iter[iter_id] =
                                        delta / CLOCKS_PER_SEC;
                                cpu_time_used += cpu_time_per_iter[iter_id];
@@ -918,21 +990,21 @@ flows_handler(void)
                        for (i = 0; i < MAX_ITERATIONS; i++) {
                                if (cpu_time_per_iter[i] == -1)
                                        continue;
-                               delta = (double)(iterations_number /
+                               delta = (double)(rules_batch /
                                        cpu_time_per_iter[i]);
                                flows_rate = delta / 1000;
                                printf(":: Iteration #%d: %d flows "
                                        "in %f sec[ Rate = %f K/Sec ]\n",
-                                       i, iterations_number,
+                                       i, rules_batch,
                                        cpu_time_per_iter[i], flows_rate);
                        }
 
                /* Insertion rate for all flows */
-               flows_rate = ((double) (flows_count / cpu_time_used) / 1000);
+               flows_rate = ((double) (rules_count / cpu_time_used) / 1000);
                printf("\n:: Total flow insertion rate -> %f K/Sec\n",
                                                flows_rate);
                printf(":: The time for creating %d in flows %f seconds\n",
-                                               flows_count, cpu_time_used);
+                                               rules_count, cpu_time_used);
 
                if (delete_flag)
                        destroy_flows(port_id, flow_list);
@@ -1357,11 +1429,11 @@ main(int argc, char **argv)
 
        force_quit = false;
        dump_iterations = false;
-       flows_count = DEFAULT_RULES_COUNT;
-       iterations_number = DEFAULT_ITERATION;
+       rules_count = DEFAULT_RULES_COUNT;
+       rules_batch = DEFAULT_RULES_BATCH;
        delete_flag = false;
        dump_socket_mem_flag = false;
-       flow_group = 0;
+       flow_group = DEFAULT_GROUP;
 
        signal(SIGINT, signal_handler);
        signal(SIGTERM, signal_handler);
@@ -1387,12 +1459,13 @@ main(int argc, char **argv)
 
        if (enable_fwd) {
                init_lcore_info();
-               rte_eal_mp_remote_launch(start_forwarding, NULL, CALL_MASTER);
+               rte_eal_mp_remote_launch(start_forwarding, NULL, CALL_MAIN);
        }
 
        RTE_ETH_FOREACH_DEV(port) {
                rte_flow_flush(port, &error);
-               rte_eth_dev_stop(port);
+               if (rte_eth_dev_stop(port) != 0)
+                       printf("Failed to stop device on port %u\n", port);
                rte_eth_dev_close(port);
        }
        return 0;