Add two new per-service counters.

RTE_SERVICE_ATTR_IDLE_CALL_COUNT tracks the number of service function
invocations where no work was performed.

RTE_SERVICE_ATTR_ERROR_CALL_COUNT tracks the number invocations
resulting in an error.

The semantics of RTE_SERVICE_ATTR_CALL_COUNT remains the same (i.e.,
counting all invocations, regardless of return value).

The new statistics may be useful for both debugging and profiling
(e.g., calculate the average per-call processing latency for non-idle
service calls).

Service core tests are extended to cover the new counters, and
coverage for RTE_SERVICE_ATTR_CALL_COUNT is improved.

The documentation for the CYCLES attributes are updated to reflect
their actual semantics.

Signed-off-by: Mattias Rönnblom <mattias.ronnb...@ericsson.com>
---
 app/test/test_service_cores.c | 70 ++++++++++++++++++++++++++---------
 lib/eal/common/rte_service.c  | 70 ++++++++++++++++++++++++++++-------
 lib/eal/include/rte_service.h | 24 ++++++++++--
 3 files changed, 131 insertions(+), 33 deletions(-)

diff --git a/app/test/test_service_cores.c b/app/test/test_service_cores.c
index c12d52d8f1..13a01b195f 100644
--- a/app/test/test_service_cores.c
+++ b/app/test/test_service_cores.c
@@ -3,11 +3,12 @@
  */
 
 #include <rte_common.h>
+#include <rte_cycles.h>
 #include <rte_hexdump.h>
-#include <rte_mbuf.h>
 #include <rte_malloc.h>
+#include <rte_mbuf.h>
 #include <rte_memcpy.h>
-#include <rte_cycles.h>
+#include <rte_random.h>
 
 #include <rte_service.h>
 #include <rte_service_component.h>
@@ -16,8 +17,10 @@
 
 /* used as the service core ID */
 static uint32_t slcore_id;
-/* used as timestamp to detect if a service core is running */
-static uint64_t service_tick;
+/* track service call count */
+static uint64_t service_calls;
+static uint64_t service_idle_calls;
+static uint64_t service_error_calls;
 /* used as a flag to check if a function was run */
 static uint32_t service_remote_launch_flag;
 
@@ -46,9 +49,21 @@ testsuite_teardown(void)
 static int32_t dummy_cb(void *args)
 {
        RTE_SET_USED(args);
-       service_tick++;
+
+       service_calls++;
+
+       switch (rte_rand_max(3)) {
+       case 0:
+               return 0;
+       case 1:
+               service_idle_calls++;
+               return -EAGAIN;
+       default:
+               service_error_calls++;
+               return -ENOENT;
+       }
+
        rte_delay_ms(SERVICE_DELAY);
-       return 0;
 }
 
 static int32_t dummy_mt_unsafe_cb(void *args)
@@ -121,6 +136,10 @@ unregister_all(void)
        rte_service_lcore_reset_all();
        rte_eal_mp_wait_lcore();
 
+       service_calls = 0;
+       service_idle_calls = 0;
+       service_error_calls = 0;
+
        return TEST_SUCCESS;
 }
 
@@ -295,12 +314,19 @@ service_attr_get(void)
                        "Valid attr_get() call didn't return success");
        TEST_ASSERT_EQUAL(0, attr_value,
                        "attr_get() call didn't set correct cycles (zero)");
-       /* check correct call count */
+       /* check correct call counts */
        const int attr_calls = RTE_SERVICE_ATTR_CALL_COUNT;
        TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_calls, &attr_value),
                        "Valid attr_get() call didn't return success");
-       TEST_ASSERT_EQUAL(0, attr_value,
-                       "attr_get() call didn't get call count (zero)");
+       TEST_ASSERT_EQUAL(0, attr_value, "Call count was not zero");
+       const int attr_idle_calls = RTE_SERVICE_ATTR_IDLE_CALL_COUNT;
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_idle_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(0, attr_value, "Idle call count was not zero");
+       const int attr_error_calls = RTE_SERVICE_ATTR_ERROR_CALL_COUNT;
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_error_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(0, attr_value, "Error call count was not zero");
 
        /* Call service to increment cycle count */
        TEST_ASSERT_EQUAL(0, rte_service_lcore_add(slcore_id),
@@ -331,8 +357,13 @@ service_attr_get(void)
 
        TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_calls, &attr_value),
                        "Valid attr_get() call didn't return success");
-       TEST_ASSERT_EQUAL(1, (attr_value > 0),
-                       "attr_get() call didn't get call count (zero)");
+       TEST_ASSERT_EQUAL(service_calls, attr_value, "Unexpected call count");
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_idle_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(service_idle_calls, attr_value, "Unexpected idle call 
count");
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_error_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(service_error_calls, attr_value, "Unexpected error 
call count");
 
        TEST_ASSERT_EQUAL(0, rte_service_attr_reset_all(id),
                        "Valid attr_reset_all() return success");
@@ -341,11 +372,16 @@ service_attr_get(void)
                        "Valid attr_get() call didn't return success");
        TEST_ASSERT_EQUAL(0, attr_value,
                        "attr_get() call didn't set correct cycles (zero)");
-       /* ensure call count > zero */
+       /* ensure call counts are zero */
        TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_calls, &attr_value),
                        "Valid attr_get() call didn't return success");
-       TEST_ASSERT_EQUAL(0, (attr_value > 0),
-                       "attr_get() call didn't get call count (zero)");
+       TEST_ASSERT_EQUAL(0, attr_value, "Call count was not reset");
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_idle_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(0, attr_value, "Idle call count was not reset");
+       TEST_ASSERT_EQUAL(0, rte_service_attr_get(id, attr_error_calls, 
&attr_value),
+                       "Valid attr_get() call didn't return success");
+       TEST_ASSERT_EQUAL(0, attr_value, "Error call count was not reset");
 
        return unregister_all();
 }
@@ -533,10 +569,10 @@ service_lcore_en_dis_able(void)
 static int
 service_lcore_running_check(void)
 {
-       uint64_t tick = service_tick;
+       uint64_t calls = service_calls;
        rte_delay_ms(SERVICE_DELAY * 100);
-       /* if (tick != service_tick) we know the lcore as polled the service */
-       return tick != service_tick;
+       bool service_polled = calls != service_calls;
+       return service_polled;
 }
 
 static int
diff --git a/lib/eal/common/rte_service.c b/lib/eal/common/rte_service.c
index d959c91459..9438d35c50 100644
--- a/lib/eal/common/rte_service.c
+++ b/lib/eal/common/rte_service.c
@@ -57,6 +57,8 @@ struct rte_service_spec_impl {
 
 struct service_stats {
        RTE_ATOMIC(uint64_t) calls;
+       RTE_ATOMIC(uint64_t) idle_calls;
+       RTE_ATOMIC(uint64_t) error_calls;
        RTE_ATOMIC(uint64_t) cycles;
 };
 
@@ -369,6 +371,16 @@ rte_service_runstate_get(uint32_t id)
 
 }
 
+static void
+service_counter_add(uint64_t *counter, uint64_t value)
+{
+       /* The lcore service worker thread is the only writer, and
+        * thus only a non-atomic load and an atomic store is needed,
+        * and not the more expensive atomic add.
+        */
+       rte_atomic_store_explicit(counter, *counter + value, 
rte_memory_order_relaxed);
+}
+
 static inline void
 service_runner_do_callback(struct rte_service_spec_impl *s,
                           struct core_state *cs, uint32_t service_idx)
@@ -380,27 +392,23 @@ service_runner_do_callback(struct rte_service_spec_impl 
*s,
                uint64_t start = rte_rdtsc();
                int rc = s->spec.callback(userdata);
 
-               /* The lcore service worker thread is the only writer,
-                * and thus only a non-atomic load and an atomic store
-                * is needed, and not the more expensive atomic
-                * add.
-                */
                struct service_stats *service_stats =
                        &cs->service_stats[service_idx];
 
+               service_counter_add(&service_stats->calls, 1);
+
+               if (rc == -EAGAIN)
+                       service_counter_add(&service_stats->idle_calls, 1);
+               else if (rc != 0)
+                       service_counter_add(&service_stats->error_calls, 1);
+
                if (likely(rc != -EAGAIN)) {
                        uint64_t end = rte_rdtsc();
                        uint64_t cycles = end - start;
 
-                       rte_atomic_store_explicit(&cs->cycles, cs->cycles + 
cycles,
-                               rte_memory_order_relaxed);
-                       rte_atomic_store_explicit(&service_stats->cycles,
-                               service_stats->cycles + cycles,
-                               rte_memory_order_relaxed);
+                       service_counter_add(&cs->cycles, cycles);
+                       service_counter_add(&service_stats->cycles, cycles);
                }
-
-               rte_atomic_store_explicit(&service_stats->calls,
-                       service_stats->calls + 1, rte_memory_order_relaxed);
        } else {
                s->spec.callback(userdata);
        }
@@ -867,6 +875,24 @@ lcore_attr_get_service_calls(uint32_t service_id, unsigned 
int lcore)
                rte_memory_order_relaxed);
 }
 
+static uint64_t
+lcore_attr_get_service_idle_calls(uint32_t service_id, unsigned int lcore)
+{
+       struct core_state *cs = &lcore_states[lcore];
+
+       return 
rte_atomic_load_explicit(&cs->service_stats[service_id].idle_calls,
+               rte_memory_order_relaxed);
+}
+
+static uint64_t
+lcore_attr_get_service_error_calls(uint32_t service_id, unsigned int lcore)
+{
+       struct core_state *cs = &lcore_states[lcore];
+
+       return 
rte_atomic_load_explicit(&cs->service_stats[service_id].error_calls,
+               rte_memory_order_relaxed);
+}
+
 static uint64_t
 lcore_attr_get_service_cycles(uint32_t service_id, unsigned int lcore)
 {
@@ -899,6 +925,18 @@ attr_get_service_calls(uint32_t service_id)
        return attr_get(service_id, lcore_attr_get_service_calls);
 }
 
+static uint64_t
+attr_get_service_idle_calls(uint32_t service_id)
+{
+       return attr_get(service_id, lcore_attr_get_service_idle_calls);
+}
+
+static uint64_t
+attr_get_service_error_calls(uint32_t service_id)
+{
+       return attr_get(service_id, lcore_attr_get_service_error_calls);
+}
+
 static uint64_t
 attr_get_service_cycles(uint32_t service_id)
 {
@@ -918,6 +956,12 @@ rte_service_attr_get(uint32_t id, uint32_t attr_id, 
uint64_t *attr_value)
        case RTE_SERVICE_ATTR_CALL_COUNT:
                *attr_value = attr_get_service_calls(id);
                return 0;
+       case RTE_SERVICE_ATTR_IDLE_CALL_COUNT:
+               *attr_value = attr_get_service_idle_calls(id);
+               return 0;
+       case RTE_SERVICE_ATTR_ERROR_CALL_COUNT:
+               *attr_value = attr_get_service_error_calls(id);
+               return 0;
        case RTE_SERVICE_ATTR_CYCLES:
                *attr_value = attr_get_service_cycles(id);
                return 0;
diff --git a/lib/eal/include/rte_service.h b/lib/eal/include/rte_service.h
index e49a7a877e..89057df585 100644
--- a/lib/eal/include/rte_service.h
+++ b/lib/eal/include/rte_service.h
@@ -374,15 +374,32 @@ int32_t rte_service_lcore_count_services(uint32_t lcore);
 int32_t rte_service_dump(FILE *f, uint32_t id);
 
 /**
- * Returns the number of cycles that this service has consumed
+ * Returns the number of cycles that this service has consumed. Only
+ * cycles spent in non-idle calls (i.e., calls not returning -EAGAIN)
+ * count.
  */
 #define RTE_SERVICE_ATTR_CYCLES 0
 
 /**
- * Returns the count of invocations of this service function
+ * Returns the total number of invocations of this service function
+ * (regardless of return value).
  */
 #define RTE_SERVICE_ATTR_CALL_COUNT 1
 
+/**
+ * Returns the number of invocations of this service function where the
+ * service reported having not performed any useful work (i.e.,
+ * returned -EAGAIN).
+ */
+#define RTE_SERVICE_ATTR_IDLE_CALL_COUNT 2
+
+/**
+ * Returns the number of invocations of this service function where the
+ * service reported an error (i.e., the return value was neither 0 nor
+ * -EAGAIN).
+ */
+#define RTE_SERVICE_ATTR_ERROR_CALL_COUNT 3
+
 /**
  * Get an attribute from a service.
  *
@@ -408,7 +425,8 @@ int32_t rte_service_attr_reset_all(uint32_t id);
 
 /**
  * Returns the total number of cycles that the lcore has spent on
- * running services.
+ * running services. Only non-idle calls (i.e., calls not returning
+ * -EAGAIN) count toward this total.
  */
 #define RTE_SERVICE_LCORE_ATTR_CYCLES 1
 
-- 
2.34.1

Reply via email to