As pointed out by Joe the following commit breaks the Mac OS X build (but not the FreeBSD one), because pthread_barrier is not available there.
The following incremental appears to fix it: diff --git a/tests/test-conntrack.c b/tests/test-conntrack.c index e2224e8..37c7277 100644 --- a/tests/test-conntrack.c +++ b/tests/test-conntrack.c @@ -75,7 +75,7 @@ struct thread_aux { static struct conntrack ct; static unsigned long n_threads, n_pkts, batch_size; static bool change_conn = false; -static pthread_barrier_t barrier; +static struct ovs_barrier barrier; static void * ct_thread_main(void *aux_) @@ -85,11 +85,11 @@ ct_thread_main(void *aux_) size_t i; pkt_batch = prepare_packets(batch_size, change_conn, aux->tid); - pthread_barrier_wait(&barrier); + ovs_barrier_block(&barrier); for (i = 0; i < n_pkts; i += batch_size) { conntrack_execute(&ct, pkt_batch, true, 0, NULL, NULL, NULL); } - pthread_barrier_wait(&barrier); + ovs_barrier_block(&barrier); destroy_packets(pkt_batch); return NULL; @@ -120,7 +120,7 @@ test_benchmark(struct ovs_cmdl_context *ctx) } threads = xcalloc(n_threads, sizeof *threads); - pthread_barrier_init(&barrier, NULL, n_threads + 1); + ovs_barrier_init(&barrier, n_threads + 1); conntrack_init(&ct); /* Create threads */ @@ -130,11 +130,11 @@ test_benchmark(struct ovs_cmdl_context *ctx) &threads[i]); } /* Starts the work inside the threads */ - pthread_barrier_wait(&barrier); + ovs_barrier_block(&barrier); start = time_msec(); /* Wait for the threads to finish the work */ - pthread_barrier_wait(&barrier); + ovs_barrier_block(&barrier); printf("conntrack: %5lld ms\n", time_msec() - start); for (i = 0; i < n_threads; i++) { @@ -142,7 +142,7 @@ test_benchmark(struct ovs_cmdl_context *ctx) } conntrack_destroy(&ct); - pthread_barrier_destroy(&barrier); + ovs_barrier_destroy(&barrier); free(threads); } On 10/06/2016 15:47, "Daniele Di Proietto" <diproiet...@vmware.com> wrote: >This introduces a very limited but simple benchmark for >conntrack_execute(). It just sends repeatedly the same batch of packets >through the connection tracker and returns the time spent to process >them. > >While this is not a realistic benchmark, it has proven useful during >development to evaluate different batching and locking strategies. > >E.g. the line: > >`./test/ovstest test-conntrack benchmark 1 14880000 32` > >starts 1 thread that will send 14880000 packets to the connection >tracker, 32 at a time. It will print the time taken to process them. > >Signed-off-by: Daniele Di Proietto <diproiet...@vmware.com> >Acked-by: Flavio Leitner <f...@sysclose.org> >--- > tests/automake.mk | 1 + > tests/test-conntrack.c | 172 +++++++++++++++++++++++++++++++++++++++++++++++++ > 2 files changed, 173 insertions(+) > create mode 100644 tests/test-conntrack.c > >diff --git a/tests/automake.mk b/tests/automake.mk >index 777f6db..c3dcb6b 100644 >--- a/tests/automake.mk >+++ b/tests/automake.mk >@@ -323,6 +323,7 @@ tests_ovstest_SOURCES = \ > tests/test-classifier.c \ > tests/test-ccmap.c \ > tests/test-cmap.c \ >+ tests/test-conntrack.c \ > tests/test-csum.c \ > tests/test-flows.c \ > tests/test-hash.c \ >diff --git a/tests/test-conntrack.c b/tests/test-conntrack.c >new file mode 100644 >index 0000000..e2224e8 >--- /dev/null >+++ b/tests/test-conntrack.c >@@ -0,0 +1,172 @@ >+/* >+ * Copyright (c) 2015 Nicira, Inc. >+ * >+ * Licensed under the Apache License, Version 2.0 (the "License"); >+ * you may not use this file except in compliance with the License. >+ * You may obtain a copy of the License at: >+ * >+ * http://www.apache.org/licenses/LICENSE-2.0 >+ * >+ * Unless required by applicable law or agreed to in writing, software >+ * distributed under the License is distributed on an "AS IS" BASIS, >+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. >+ * See the License for the specific language governing permissions and >+ * limitations under the License. >+ */ >+ >+#include <config.h> >+#include "conntrack.h" >+ >+#include "dp-packet.h" >+#include "fatal-signal.h" >+#include "flow.h" >+#include "netdev.h" >+#include "ovs-thread.h" >+#include "ovstest.h" >+#include "timeval.h" >+ >+static const char payload[] = "50540000000a50540000000908004500001c0000000000" >+ "11a4cd0a0101010a0101020001000200080000"; >+ >+static struct dp_packet_batch * >+prepare_packets(size_t n, bool change, unsigned tid) >+{ >+ struct dp_packet_batch *pkt_batch = xzalloc(sizeof *pkt_batch); >+ struct flow flow; >+ size_t i; >+ >+ ovs_assert(n <= ARRAY_SIZE(pkt_batch->packets)); >+ >+ dp_packet_batch_init(pkt_batch); >+ pkt_batch->count = n; >+ >+ for (i = 0; i < n; i++) { >+ struct udp_header *udp; >+ struct dp_packet *pkt = dp_packet_new(sizeof payload/2); >+ >+ dp_packet_put_hex(pkt, payload, NULL); >+ flow_extract(pkt, &flow); >+ >+ udp = dp_packet_l4(pkt); >+ udp->udp_src = htons(ntohs(udp->udp_src) + tid); >+ >+ if (change) { >+ udp->udp_dst = htons(ntohs(udp->udp_dst) + i); >+ } >+ >+ pkt_batch->packets[i] = pkt; >+ } >+ >+ return pkt_batch; >+} >+ >+static void >+destroy_packets(struct dp_packet_batch *pkt_batch) >+{ >+ dp_packet_delete_batch(pkt_batch, true); >+ free(pkt_batch); >+} >+ >+struct thread_aux { >+ pthread_t thread; >+ unsigned tid; >+}; >+ >+static struct conntrack ct; >+static unsigned long n_threads, n_pkts, batch_size; >+static bool change_conn = false; >+static pthread_barrier_t barrier; >+ >+static void * >+ct_thread_main(void *aux_) >+{ >+ struct thread_aux *aux = aux_; >+ struct dp_packet_batch *pkt_batch; >+ size_t i; >+ >+ pkt_batch = prepare_packets(batch_size, change_conn, aux->tid); >+ pthread_barrier_wait(&barrier); >+ for (i = 0; i < n_pkts; i += batch_size) { >+ conntrack_execute(&ct, pkt_batch, true, 0, NULL, NULL, NULL); >+ } >+ pthread_barrier_wait(&barrier); >+ destroy_packets(pkt_batch); >+ >+ return NULL; >+} >+ >+static void >+test_benchmark(struct ovs_cmdl_context *ctx) >+{ >+ struct thread_aux *threads; >+ long long start; >+ unsigned i; >+ >+ fatal_signal_init(); >+ >+ /* Parse arguments */ >+ n_threads = strtoul(ctx->argv[1], NULL, 0); >+ if (!n_threads) { >+ ovs_fatal(0, "n_threads must be at least one"); >+ } >+ n_pkts = strtoul(ctx->argv[2], NULL, 0); >+ batch_size = strtoul(ctx->argv[3], NULL, 0); >+ if (batch_size == 0 || batch_size > NETDEV_MAX_BURST) { >+ ovs_fatal(0, "batch_size must be between 1 and NETDEV_MAX_BURST(%u)", >+ NETDEV_MAX_BURST); >+ } >+ if (ctx->argc > 4) { >+ change_conn = strtoul(ctx->argv[4], NULL, 0); >+ } >+ >+ threads = xcalloc(n_threads, sizeof *threads); >+ pthread_barrier_init(&barrier, NULL, n_threads + 1); >+ conntrack_init(&ct); >+ >+ /* Create threads */ >+ for (i = 0; i < n_threads; i++) { >+ threads[i].tid = i; >+ threads[i].thread = ovs_thread_create("ct_thread", ct_thread_main, >+ &threads[i]); >+ } >+ /* Starts the work inside the threads */ >+ pthread_barrier_wait(&barrier); >+ start = time_msec(); >+ >+ /* Wait for the threads to finish the work */ >+ pthread_barrier_wait(&barrier); >+ printf("conntrack: %5lld ms\n", time_msec() - start); >+ >+ for (i = 0; i < n_threads; i++) { >+ xpthread_join(threads[i].thread, NULL); >+ } >+ >+ conntrack_destroy(&ct); >+ pthread_barrier_destroy(&barrier); >+ free(threads); >+} >+? >+static const struct ovs_cmdl_command commands[] = { >+ /* Connection tracker tests. */ >+ /* Starts 'n_threads' threads. Each thread will send 'n_pkts' packets to >+ * the connection tracker, 'batch_size' per call. If 'change_connection' >+ * is '1', each packet in a batch will have a different source and >+ * destination port */ >+ {"benchmark", "n_threads n_pkts batch_size [change_connection]", 3, 4, >+ test_benchmark}, >+ >+ {NULL, NULL, 0, 0, NULL}, >+}; >+ >+static void >+test_conntrack_main(int argc, char *argv[]) >+{ >+ struct ovs_cmdl_context ctx = { >+ .argc = argc - 1, >+ .argv = argv + 1, >+ }; >+ set_program_name(argv[0]); >+ ovs_cmdl_run_command(&ctx, commands); >+} >+ >+OVSTEST_REGISTER("test-conntrack", test_conntrack_main); >-- >2.8.1 > _______________________________________________ dev mailing list dev@openvswitch.org http://openvswitch.org/mailman/listinfo/dev