From: "Carrillo, Erik G" <erik.g.carrillo@intel.com>
To: "Sanford, Robert" <rsanford@akamai.com>
Cc: "thomas@monjalon.net" <thomas@monjalon.net>,
"dev@dpdk.org" <dev@dpdk.org>,
"nhorman@tuxdriver.com" <nhorman@tuxdriver.com>
Subject: Re: [dpdk-dev] [PATCH v4 1/2] timer: allow timer management in shared memory
Date: Thu, 21 Mar 2019 01:01:59 +0000 [thread overview]
Message-ID: <D926C980-58E3-4476-9FA9-D9180C6863A2@intel.com> (raw)
In-Reply-To: <40792313-7864-470F-B188-9AA1A0F072B8@akamai.com>
Hi Robert,
Thanks for the review and suggestions. I’m out of the office on bonding leave for the next few weeks, but I’ll update the patch to address your points below when I return.
Thanks,
Erik
> On Mar 20, 2019, at 8:53 AM, Sanford, Robert <rsanford@akamai.com> wrote:
>
> Hi Erik,
>
> I have a few questions and comments on this patch series.
>
> 1. Don't you think we need new tests (in test/test/) to verify the secondary-process APIs?
> 2. I suggest we define default_data_id as const, and explicitly set it to 0.
> 3. The outer for-loop in rte_timer_alt_manage() touches beyond the end of poll_lcores[]. I suggest a change like this:
>
> - for (i = 0, poll_lcore = poll_lcores[i]; i < nb_poll_lcores;
> - poll_lcore = poll_lcores[++i]) {
> + for (i = 0; I < nb_poll_lcores; i++) {
> + poll_lcore = poll_lcores[i];
>
> 4. Same problem (as #3) in the for-loop in rte_timer_stop_all(), in patch v4 2/2.
> 5. There seems to be no difference between "typedef void (*rte_timer_cb_t)(struct rte_timer *, void *)" and "typedef void (*rte_timer_stop_all_cb_t)(struct rte_timer *tim, void *arg)", why add rte_timer_stop_all_cb_t?
> 6. Can you provide a use case or code snippet that shows how we will use rte_timer_alt_manage()?
> 7. Why not make the argument to rte_timer_alt_manage_cb_t a "struct rte_timer *", instead of a "void *", since we pass a pointer-to-timer when we invoke the function?
>
> --
> Regards,
> Robert Sanford
>
>
> On 3/6/19, 12:20 PM, "Erik Gabriel Carrillo" <erik.g.carrillo@intel.com> wrote:
>
> Currently, the timer library uses a per-process table of structures to
> manage skiplists of timers presumably because timers contain arbitrary
> function pointers whose value may not resolve properly in other
> processes.
>
> However, if the same callback is used handle all timers, and that
> callback is only invoked in one process, then it woud be safe to allow
> the data structures to be allocated in shared memory, and to allow
> secondary processes to modify the timer lists. This would let timers be
> used in more multi-process scenarios.
>
> The library's global variables are wrapped with a struct, and an array
> of these structures is created in shared memory. The original APIs
> are updated to reference the zeroth entry in the array. This maintains
> the original behavior for both primary and secondary processes since
> the set intersection of their coremasks should be empty [1]. New APIs
> are introduced to enable the allocation/deallocation of other entries
> in the array.
>
> New variants of the APIs used to start and stop timers are introduced;
> they allow a caller to specify which array entry should be used to
> locate the timer list to insert into or delete from.
>
> Finally, a new variant of rte_timer_manage() is introduced, which
> allows a caller to specify which array entry should be used to locate
> the timer lists to process; it can also process multiple timer lists per
> invocation.
>
> [1] https://doc.dpdk.org/guides/prog_guide/multi_proc_support.html#multi-process-limitations
>
> Signed-off-by: Erik Gabriel Carrillo <erik.g.carrillo@intel.com>
> ---
> lib/librte_timer/Makefile | 1 +
> lib/librte_timer/rte_timer.c | 519 ++++++++++++++++++++++++++++++---
> lib/librte_timer/rte_timer.h | 226 +++++++++++++-
> lib/librte_timer/rte_timer_version.map | 22 ++
> 4 files changed, 723 insertions(+), 45 deletions(-)
>
> diff --git a/lib/librte_timer/Makefile b/lib/librte_timer/Makefile
> index 4ebd528..8ec63f4 100644
> --- a/lib/librte_timer/Makefile
> +++ b/lib/librte_timer/Makefile
> @@ -6,6 +6,7 @@ include $(RTE_SDK)/mk/rte.vars.mk
> # library name
> LIB = librte_timer.a
>
> +CFLAGS += -DALLOW_EXPERIMENTAL_API
> CFLAGS += $(WERROR_FLAGS) -I$(SRCDIR) -O3
> LDLIBS += -lrte_eal
>
> diff --git a/lib/librte_timer/rte_timer.c b/lib/librte_timer/rte_timer.c
> index 30c7b0a..2bd49d0 100644
> --- a/lib/librte_timer/rte_timer.c
> +++ b/lib/librte_timer/rte_timer.c
> @@ -5,6 +5,7 @@
> #include <string.h>
> #include <stdio.h>
> #include <stdint.h>
> +#include <stdbool.h>
> #include <inttypes.h>
> #include <assert.h>
> #include <sys/queue.h>
> @@ -21,11 +22,15 @@
> #include <rte_spinlock.h>
> #include <rte_random.h>
> #include <rte_pause.h>
> +#include <rte_memzone.h>
> +#include <rte_malloc.h>
> +#include <rte_compat.h>
>
> #include "rte_timer.h"
>
> -LIST_HEAD(rte_timer_list, rte_timer);
> -
> +/**
> + * Per-lcore info for timers.
> + */
> struct priv_timer {
> struct rte_timer pending_head; /**< dummy timer instance to head up list */
> rte_spinlock_t list_lock; /**< lock to protect list access */
> @@ -48,25 +53,84 @@ struct priv_timer {
> #endif
> } __rte_cache_aligned;
>
> -/** per-lcore private info for timers */
> -static struct priv_timer priv_timer[RTE_MAX_LCORE];
> +#define FL_ALLOCATED (1 << 0)
> +struct rte_timer_data {
> + struct priv_timer priv_timer[RTE_MAX_LCORE];
> + uint8_t internal_flags;
> +};
> +
> +#define RTE_MAX_DATA_ELS 64
> +static struct rte_timer_data *rte_timer_data_arr;
> +static uint32_t default_data_id;
> +static uint32_t rte_timer_subsystem_initialized;
> +
> +/* For maintaining older interfaces for a period */
> +static struct rte_timer_data default_timer_data;
>
> /* when debug is enabled, store some statistics */
> #ifdef RTE_LIBRTE_TIMER_DEBUG
> -#define __TIMER_STAT_ADD(name, n) do { \
> +#define __TIMER_STAT_ADD(priv_timer, name, n) do { \
> unsigned __lcore_id = rte_lcore_id(); \
> if (__lcore_id < RTE_MAX_LCORE) \
> priv_timer[__lcore_id].stats.name += (n); \
> } while(0)
> #else
> -#define __TIMER_STAT_ADD(name, n) do {} while(0)
> +#define __TIMER_STAT_ADD(priv_timer, name, n) do {} while (0)
> #endif
>
> -/* Init the timer library. */
> +static inline int
> +timer_data_valid(uint32_t id)
> +{
> + return !!(rte_timer_data_arr[id].internal_flags & FL_ALLOCATED);
> +}
> +
> +/* validate ID and retrieve timer data pointer, or return error value */
> +#define TIMER_DATA_VALID_GET_OR_ERR_RET(id, timer_data, retval) do { \
> + if (id >= RTE_MAX_DATA_ELS || !timer_data_valid(id)) \
> + return retval; \
> + timer_data = &rte_timer_data_arr[id]; \
> +} while (0)
> +
> +int __rte_experimental
> +rte_timer_data_alloc(uint32_t *id_ptr)
> +{
> + int i;
> + struct rte_timer_data *data;
> +
> + if (!rte_timer_subsystem_initialized)
> + return -ENOMEM;
> +
> + for (i = 0; i < RTE_MAX_DATA_ELS; i++) {
> + data = &rte_timer_data_arr[i];
> + if (!(data->internal_flags & FL_ALLOCATED)) {
> + data->internal_flags |= FL_ALLOCATED;
> +
> + if (id_ptr)
> + *id_ptr = i;
> +
> + return 0;
> + }
> + }
> +
> + return -ENOSPC;
> +}
> +
> +int __rte_experimental
> +rte_timer_data_dealloc(uint32_t id)
> +{
> + struct rte_timer_data *timer_data;
> + TIMER_DATA_VALID_GET_OR_ERR_RET(id, timer_data, -EINVAL);
> +
> + timer_data->internal_flags &= ~(FL_ALLOCATED);
> +
> + return 0;
> +}
> +
> void
> -rte_timer_subsystem_init(void)
> +rte_timer_subsystem_init_v20(void)
> {
> unsigned lcore_id;
> + struct priv_timer *priv_timer = default_timer_data.priv_timer;
>
> /* since priv_timer is static, it's zeroed by default, so only init some
> * fields.
> @@ -76,6 +140,76 @@ rte_timer_subsystem_init(void)
> priv_timer[lcore_id].prev_lcore = lcore_id;
> }
> }
> +VERSION_SYMBOL(rte_timer_subsystem_init, _v20, 2.0);
> +
> +/* Init the timer library. Allocate an array of timer data structs in shared
> + * memory, and allocate the zeroth entry for use with original timer
> + * APIs. Since the intersection of the sets of lcore ids in primary and
> + * secondary processes should be empty, the zeroth entry can be shared by
> + * multiple processes.
> + */
> +int
> +rte_timer_subsystem_init_v1905(void)
> +{
> + const struct rte_memzone *mz;
> + struct rte_timer_data *data;
> + int i, lcore_id;
> + static const char *mz_name = "rte_timer_mz";
> +
> + if (rte_timer_subsystem_initialized)
> + return -EALREADY;
> +
> + if (rte_eal_process_type() != RTE_PROC_PRIMARY) {
> + mz = rte_memzone_lookup(mz_name);
> + if (mz == NULL)
> + return -EEXIST;
> +
> + rte_timer_data_arr = mz->addr;
> +
> + rte_timer_data_arr[default_data_id].internal_flags |=
> + FL_ALLOCATED;
> +
> + rte_timer_subsystem_initialized = 1;
> +
> + return 0;
> + }
> +
> + mz = rte_memzone_reserve_aligned(mz_name,
> + RTE_MAX_DATA_ELS * sizeof(*rte_timer_data_arr),
> + SOCKET_ID_ANY, 0, RTE_CACHE_LINE_SIZE);
> + if (mz == NULL)
> + return -ENOMEM;
> +
> + rte_timer_data_arr = mz->addr;
> +
> + for (i = 0; i < RTE_MAX_DATA_ELS; i++) {
> + data = &rte_timer_data_arr[i];
> +
> + for (lcore_id = 0; lcore_id < RTE_MAX_LCORE; lcore_id++) {
> + rte_spinlock_init(
> + &data->priv_timer[lcore_id].list_lock);
> + data->priv_timer[lcore_id].prev_lcore = lcore_id;
> + }
> + }
> +
> + rte_timer_data_arr[default_data_id].internal_flags |= FL_ALLOCATED;
> +
> + rte_timer_subsystem_initialized = 1;
> +
> + return 0;
> +}
> +MAP_STATIC_SYMBOL(int rte_timer_subsystem_init(void),
> + rte_timer_subsystem_init_v1905);
> +BIND_DEFAULT_SYMBOL(rte_timer_subsystem_init, _v1905, 19.05);
> +
> +void __rte_experimental
> +rte_timer_subsystem_finalize(void)
> +{
> + if (rte_timer_data_arr)
> + rte_free(rte_timer_data_arr);
> +
> + rte_timer_subsystem_initialized = 0;
> +}
>
> /* Initialize the timer handle tim for use */
> void
> @@ -95,7 +229,8 @@ rte_timer_init(struct rte_timer *tim)
> */
> static int
> timer_set_config_state(struct rte_timer *tim,
> - union rte_timer_status *ret_prev_status)
> + union rte_timer_status *ret_prev_status,
> + struct priv_timer *priv_timer)
> {
> union rte_timer_status prev_status, status;
> int success = 0;
> @@ -207,7 +342,7 @@ timer_get_skiplist_level(unsigned curr_depth)
> */
> static void
> timer_get_prev_entries(uint64_t time_val, unsigned tim_lcore,
> - struct rte_timer **prev)
> + struct rte_timer **prev, struct priv_timer *priv_timer)
> {
> unsigned lvl = priv_timer[tim_lcore].curr_skiplist_depth;
> prev[lvl] = &priv_timer[tim_lcore].pending_head;
> @@ -226,13 +361,15 @@ timer_get_prev_entries(uint64_t time_val, unsigned tim_lcore,
> */
> static void
> timer_get_prev_entries_for_node(struct rte_timer *tim, unsigned tim_lcore,
> - struct rte_timer **prev)
> + struct rte_timer **prev,
> + struct priv_timer *priv_timer)
> {
> int i;
> +
> /* to get a specific entry in the list, look for just lower than the time
> * values, and then increment on each level individually if necessary
> */
> - timer_get_prev_entries(tim->expire - 1, tim_lcore, prev);
> + timer_get_prev_entries(tim->expire - 1, tim_lcore, prev, priv_timer);
> for (i = priv_timer[tim_lcore].curr_skiplist_depth - 1; i >= 0; i--) {
> while (prev[i]->sl_next[i] != NULL &&
> prev[i]->sl_next[i] != tim &&
> @@ -247,14 +384,15 @@ timer_get_prev_entries_for_node(struct rte_timer *tim, unsigned tim_lcore,
> * timer must not be in a list
> */
> static void
> -timer_add(struct rte_timer *tim, unsigned int tim_lcore)
> +timer_add(struct rte_timer *tim, unsigned int tim_lcore,
> + struct priv_timer *priv_timer)
> {
> unsigned lvl;
> struct rte_timer *prev[MAX_SKIPLIST_DEPTH+1];
>
> /* find where exactly this element goes in the list of elements
> * for each depth. */
> - timer_get_prev_entries(tim->expire, tim_lcore, prev);
> + timer_get_prev_entries(tim->expire, tim_lcore, prev, priv_timer);
>
> /* now assign it a new level and add at that level */
> const unsigned tim_level = timer_get_skiplist_level(
> @@ -284,7 +422,7 @@ timer_add(struct rte_timer *tim, unsigned int tim_lcore)
> */
> static void
> timer_del(struct rte_timer *tim, union rte_timer_status prev_status,
> - int local_is_locked)
> + int local_is_locked, struct priv_timer *priv_timer)
> {
> unsigned lcore_id = rte_lcore_id();
> unsigned prev_owner = prev_status.owner;
> @@ -304,7 +442,7 @@ timer_del(struct rte_timer *tim, union rte_timer_status prev_status,
> ((tim->sl_next[0] == NULL) ? 0 : tim->sl_next[0]->expire);
>
> /* adjust pointers from previous entries to point past this */
> - timer_get_prev_entries_for_node(tim, prev_owner, prev);
> + timer_get_prev_entries_for_node(tim, prev_owner, prev, priv_timer);
> for (i = priv_timer[prev_owner].curr_skiplist_depth - 1; i >= 0; i--) {
> if (prev[i]->sl_next[i] == tim)
> prev[i]->sl_next[i] = tim->sl_next[i];
> @@ -326,11 +464,13 @@ static int
> __rte_timer_reset(struct rte_timer *tim, uint64_t expire,
> uint64_t period, unsigned tim_lcore,
> rte_timer_cb_t fct, void *arg,
> - int local_is_locked)
> + int local_is_locked,
> + struct rte_timer_data *timer_data)
> {
> union rte_timer_status prev_status, status;
> int ret;
> unsigned lcore_id = rte_lcore_id();
> + struct priv_timer *priv_timer = timer_data->priv_timer;
>
> /* round robin for tim_lcore */
> if (tim_lcore == (unsigned)LCORE_ID_ANY) {
> @@ -348,11 +488,11 @@ __rte_timer_reset(struct rte_timer *tim, uint64_t expire,
>
> /* wait that the timer is in correct status before update,
> * and mark it as being configured */
> - ret = timer_set_config_state(tim, &prev_status);
> + ret = timer_set_config_state(tim, &prev_status, priv_timer);
> if (ret < 0)
> return -1;
>
> - __TIMER_STAT_ADD(reset, 1);
> + __TIMER_STAT_ADD(priv_timer, reset, 1);
> if (prev_status.state == RTE_TIMER_RUNNING &&
> lcore_id < RTE_MAX_LCORE) {
> priv_timer[lcore_id].updated = 1;
> @@ -360,8 +500,8 @@ __rte_timer_reset(struct rte_timer *tim, uint64_t expire,
>
> /* remove it from list */
> if (prev_status.state == RTE_TIMER_PENDING) {
> - timer_del(tim, prev_status, local_is_locked);
> - __TIMER_STAT_ADD(pending, -1);
> + timer_del(tim, prev_status, local_is_locked, priv_timer);
> + __TIMER_STAT_ADD(priv_timer, pending, -1);
> }
>
> tim->period = period;
> @@ -376,8 +516,8 @@ __rte_timer_reset(struct rte_timer *tim, uint64_t expire,
> if (tim_lcore != lcore_id || !local_is_locked)
> rte_spinlock_lock(&priv_timer[tim_lcore].list_lock);
>
> - __TIMER_STAT_ADD(pending, 1);
> - timer_add(tim, tim_lcore);
> + __TIMER_STAT_ADD(priv_timer, pending, 1);
> + timer_add(tim, tim_lcore, priv_timer);
>
> /* update state: as we are in CONFIG state, only us can modify
> * the state so we don't need to use cmpset() here */
> @@ -394,9 +534,9 @@ __rte_timer_reset(struct rte_timer *tim, uint64_t expire,
>
> /* Reset and start the timer associated with the timer handle tim */
> int
> -rte_timer_reset(struct rte_timer *tim, uint64_t ticks,
> - enum rte_timer_type type, unsigned tim_lcore,
> - rte_timer_cb_t fct, void *arg)
> +rte_timer_reset_v20(struct rte_timer *tim, uint64_t ticks,
> + enum rte_timer_type type, unsigned int tim_lcore,
> + rte_timer_cb_t fct, void *arg)
> {
> uint64_t cur_time = rte_get_timer_cycles();
> uint64_t period;
> @@ -412,7 +552,48 @@ rte_timer_reset(struct rte_timer *tim, uint64_t ticks,
> period = 0;
>
> return __rte_timer_reset(tim, cur_time + ticks, period, tim_lcore,
> - fct, arg, 0);
> + fct, arg, 0, &default_timer_data);
> +}
> +VERSION_SYMBOL(rte_timer_reset, _v20, 2.0);
> +
> +int
> +rte_timer_reset_v1905(struct rte_timer *tim, uint64_t ticks,
> + enum rte_timer_type type, unsigned int tim_lcore,
> + rte_timer_cb_t fct, void *arg)
> +{
> + return rte_timer_alt_reset(default_data_id, tim, ticks, type,
> + tim_lcore, fct, arg);
> +}
> +MAP_STATIC_SYMBOL(int rte_timer_reset(struct rte_timer *tim, uint64_t ticks,
> + enum rte_timer_type type,
> + unsigned int tim_lcore,
> + rte_timer_cb_t fct, void *arg),
> + rte_timer_reset_v1905);
> +BIND_DEFAULT_SYMBOL(rte_timer_reset, _v1905, 19.05);
> +
> +int __rte_experimental
> +rte_timer_alt_reset(uint32_t timer_data_id, struct rte_timer *tim,
> + uint64_t ticks, enum rte_timer_type type,
> + unsigned int tim_lcore, rte_timer_cb_t fct, void *arg)
> +{
> + uint64_t cur_time = rte_get_timer_cycles();
> + uint64_t period;
> + struct rte_timer_data *timer_data;
> +
> + TIMER_DATA_VALID_GET_OR_ERR_RET(timer_data_id, timer_data, -EINVAL);
> +
> + if (unlikely((tim_lcore != (unsigned int)LCORE_ID_ANY) &&
> + !(rte_lcore_is_enabled(tim_lcore) ||
> + rte_lcore_has_role(tim_lcore, ROLE_SERVICE))))
> + return -1;
> +
> + if (type == PERIODICAL)
> + period = ticks;
> + else
> + period = 0;
> +
> + return __rte_timer_reset(tim, cur_time + ticks, period, tim_lcore,
> + fct, arg, 0, timer_data);
> }
>
> /* loop until rte_timer_reset() succeed */
> @@ -426,21 +607,22 @@ rte_timer_reset_sync(struct rte_timer *tim, uint64_t ticks,
> rte_pause();
> }
>
> -/* Stop the timer associated with the timer handle tim */
> -int
> -rte_timer_stop(struct rte_timer *tim)
> +static int
> +__rte_timer_stop(struct rte_timer *tim, int local_is_locked,
> + struct rte_timer_data *timer_data)
> {
> union rte_timer_status prev_status, status;
> unsigned lcore_id = rte_lcore_id();
> int ret;
> + struct priv_timer *priv_timer = timer_data->priv_timer;
>
> /* wait that the timer is in correct status before update,
> * and mark it as being configured */
> - ret = timer_set_config_state(tim, &prev_status);
> + ret = timer_set_config_state(tim, &prev_status, priv_timer);
> if (ret < 0)
> return -1;
>
> - __TIMER_STAT_ADD(stop, 1);
> + __TIMER_STAT_ADD(priv_timer, stop, 1);
> if (prev_status.state == RTE_TIMER_RUNNING &&
> lcore_id < RTE_MAX_LCORE) {
> priv_timer[lcore_id].updated = 1;
> @@ -448,8 +630,8 @@ rte_timer_stop(struct rte_timer *tim)
>
> /* remove it from list */
> if (prev_status.state == RTE_TIMER_PENDING) {
> - timer_del(tim, prev_status, 0);
> - __TIMER_STAT_ADD(pending, -1);
> + timer_del(tim, prev_status, local_is_locked, priv_timer);
> + __TIMER_STAT_ADD(priv_timer, pending, -1);
> }
>
> /* mark timer as stopped */
> @@ -461,6 +643,33 @@ rte_timer_stop(struct rte_timer *tim)
> return 0;
> }
>
> +/* Stop the timer associated with the timer handle tim */
> +int
> +rte_timer_stop_v20(struct rte_timer *tim)
> +{
> + return __rte_timer_stop(tim, 0, &default_timer_data);
> +}
> +VERSION_SYMBOL(rte_timer_stop, _v20, 2.0);
> +
> +int
> +rte_timer_stop_v1905(struct rte_timer *tim)
> +{
> + return rte_timer_alt_stop(default_data_id, tim);
> +}
> +MAP_STATIC_SYMBOL(int rte_timer_stop(struct rte_timer *tim),
> + rte_timer_stop_v1905);
> +BIND_DEFAULT_SYMBOL(rte_timer_stop, _v1905, 19.05);
> +
> +int __rte_experimental
> +rte_timer_alt_stop(uint32_t timer_data_id, struct rte_timer *tim)
> +{
> + struct rte_timer_data *timer_data;
> +
> + TIMER_DATA_VALID_GET_OR_ERR_RET(timer_data_id, timer_data, -EINVAL);
> +
> + return __rte_timer_stop(tim, 0, timer_data);
> +}
> +
> /* loop until rte_timer_stop() succeed */
> void
> rte_timer_stop_sync(struct rte_timer *tim)
> @@ -477,7 +686,8 @@ rte_timer_pending(struct rte_timer *tim)
> }
>
> /* must be called periodically, run all timer that expired */
> -void rte_timer_manage(void)
> +static void
> +__rte_timer_manage(struct rte_timer_data *timer_data)
> {
> union rte_timer_status status;
> struct rte_timer *tim, *next_tim;
> @@ -486,11 +696,12 @@ void rte_timer_manage(void)
> struct rte_timer *prev[MAX_SKIPLIST_DEPTH + 1];
> uint64_t cur_time;
> int i, ret;
> + struct priv_timer *priv_timer = timer_data->priv_timer;
>
> /* timer manager only runs on EAL thread with valid lcore_id */
> assert(lcore_id < RTE_MAX_LCORE);
>
> - __TIMER_STAT_ADD(manage, 1);
> + __TIMER_STAT_ADD(priv_timer, manage, 1);
> /* optimize for the case where per-cpu list is empty */
> if (priv_timer[lcore_id].pending_head.sl_next[0] == NULL)
> return;
> @@ -518,7 +729,7 @@ void rte_timer_manage(void)
> tim = priv_timer[lcore_id].pending_head.sl_next[0];
>
> /* break the existing list at current time point */
> - timer_get_prev_entries(cur_time, lcore_id, prev);
> + timer_get_prev_entries(cur_time, lcore_id, prev, priv_timer);
> for (i = priv_timer[lcore_id].curr_skiplist_depth -1; i >= 0; i--) {
> if (prev[i] == &priv_timer[lcore_id].pending_head)
> continue;
> @@ -563,7 +774,7 @@ void rte_timer_manage(void)
> /* execute callback function with list unlocked */
> tim->f(tim, tim->arg);
>
> - __TIMER_STAT_ADD(pending, -1);
> + __TIMER_STAT_ADD(priv_timer, pending, -1);
> /* the timer was stopped or reloaded by the callback
> * function, we have nothing to do here */
> if (priv_timer[lcore_id].updated == 1)
> @@ -580,24 +791,222 @@ void rte_timer_manage(void)
> /* keep it in list and mark timer as pending */
> rte_spinlock_lock(&priv_timer[lcore_id].list_lock);
> status.state = RTE_TIMER_PENDING;
> - __TIMER_STAT_ADD(pending, 1);
> + __TIMER_STAT_ADD(priv_timer, pending, 1);
> status.owner = (int16_t)lcore_id;
> rte_wmb();
> tim->status.u32 = status.u32;
> __rte_timer_reset(tim, tim->expire + tim->period,
> - tim->period, lcore_id, tim->f, tim->arg, 1);
> + tim->period, lcore_id, tim->f, tim->arg, 1,
> + timer_data);
> rte_spinlock_unlock(&priv_timer[lcore_id].list_lock);
> }
> }
> priv_timer[lcore_id].running_tim = NULL;
> }
>
> +void
> +rte_timer_manage_v20(void)
> +{
> + __rte_timer_manage(&default_timer_data);
> +}
> +VERSION_SYMBOL(rte_timer_manage, _v20, 2.0);
> +
> +int
> +rte_timer_manage_v1905(void)
> +{
> + struct rte_timer_data *timer_data;
> +
> + TIMER_DATA_VALID_GET_OR_ERR_RET(default_data_id, timer_data, -EINVAL);
> +
> + __rte_timer_manage(timer_data);
> +
> + return 0;
> +}
> +MAP_STATIC_SYMBOL(int rte_timer_manage(void), rte_timer_manage_v1905);
> +BIND_DEFAULT_SYMBOL(rte_timer_manage, _v1905, 19.05);
> +
> +int __rte_experimental
> +rte_timer_alt_manage(uint32_t timer_data_id,
> + unsigned int *poll_lcores,
> + int nb_poll_lcores,
> + rte_timer_alt_manage_cb_t f)
> +{
> + union rte_timer_status status;
> + struct rte_timer *tim, *next_tim, **pprev;
> + struct rte_timer *run_first_tims[RTE_MAX_LCORE];
> + unsigned int runlist_lcore_ids[RTE_MAX_LCORE];
> + unsigned int this_lcore = rte_lcore_id();
> + struct rte_timer *prev[MAX_SKIPLIST_DEPTH + 1];
> + uint64_t cur_time;
> + int i, j, ret;
> + int nb_runlists = 0;
> + struct rte_timer_data *data;
> + struct priv_timer *privp;
> + uint32_t poll_lcore;
> +
> + TIMER_DATA_VALID_GET_OR_ERR_RET(timer_data_id, data, -EINVAL);
> +
> + /* timer manager only runs on EAL thread with valid lcore_id */
> + assert(this_lcore < RTE_MAX_LCORE);
> +
> + __TIMER_STAT_ADD(data->priv_timer, manage, 1);
> +
> + if (poll_lcores == NULL) {
> + poll_lcores = (unsigned int []){rte_lcore_id()};
> + nb_poll_lcores = 1;
> + }
> +
> + for (i = 0, poll_lcore = poll_lcores[i]; i < nb_poll_lcores;
> + poll_lcore = poll_lcores[++i]) {
> + privp = &data->priv_timer[poll_lcore];
> +
> + /* optimize for the case where per-cpu list is empty */
> + if (privp->pending_head.sl_next[0] == NULL)
> + continue;
> + cur_time = rte_get_timer_cycles();
> +
> +#ifdef RTE_ARCH_64
> + /* on 64-bit the value cached in the pending_head.expired will
> + * be updated atomically, so we can consult that for a quick
> + * check here outside the lock
> + */
> + if (likely(privp->pending_head.expire > cur_time))
> + continue;
> +#endif
> +
> + /* browse ordered list, add expired timers in 'expired' list */
> + rte_spinlock_lock(&privp->list_lock);
> +
> + /* if nothing to do just unlock and return */
> + if (privp->pending_head.sl_next[0] == NULL ||
> + privp->pending_head.sl_next[0]->expire > cur_time) {
> + rte_spinlock_unlock(&privp->list_lock);
> + continue;
> + }
> +
> + /* save start of list of expired timers */
> + tim = privp->pending_head.sl_next[0];
> +
> + /* break the existing list at current time point */
> + timer_get_prev_entries(cur_time, poll_lcore, prev,
> + data->priv_timer);
> + for (j = privp->curr_skiplist_depth - 1; j >= 0; j--) {
> + if (prev[j] == &privp->pending_head)
> + continue;
> + privp->pending_head.sl_next[j] =
> + prev[j]->sl_next[j];
> + if (prev[j]->sl_next[j] == NULL)
> + privp->curr_skiplist_depth--;
> +
> + prev[j]->sl_next[j] = NULL;
> + }
> +
> + /* transition run-list from PENDING to RUNNING */
> + run_first_tims[nb_runlists] = tim;
> + runlist_lcore_ids[nb_runlists] = poll_lcore;
> + pprev = &run_first_tims[nb_runlists];
> + nb_runlists++;
> +
> + for ( ; tim != NULL; tim = next_tim) {
> + next_tim = tim->sl_next[0];
> +
> + ret = timer_set_running_state(tim);
> + if (likely(ret == 0)) {
> + pprev = &tim->sl_next[0];
> + } else {
> + /* another core is trying to re-config this one,
> + * remove it from local expired list
> + */
> + *pprev = next_tim;
> + }
> + }
> +
> + /* update the next to expire timer value */
> + privp->pending_head.expire =
> + (privp->pending_head.sl_next[0] == NULL) ? 0 :
> + privp->pending_head.sl_next[0]->expire;
> +
> + rte_spinlock_unlock(&privp->list_lock);
> + }
> +
> + /* Now process the run lists */
> + while (1) {
> + bool done = true;
> + uint64_t min_expire = UINT64_MAX;
> + int min_idx = 0;
> +
> + /* Find the next oldest timer to process */
> + for (i = 0; i < nb_runlists; i++) {
> + tim = run_first_tims[i];
> +
> + if (tim != NULL && tim->expire < min_expire) {
> + min_expire = tim->expire;
> + min_idx = i;
> + done = false;
> + }
> + }
> +
> + if (done)
> + break;
> +
> + tim = run_first_tims[min_idx];
> + privp = &data->priv_timer[runlist_lcore_ids[min_idx]];
> +
> + /* Move down the runlist from which we picked a timer to
> + * execute
> + */
> + run_first_tims[min_idx] = run_first_tims[min_idx]->sl_next[0];
> +
> + privp->updated = 0;
> + privp->running_tim = tim;
> +
> + /* Call the provided callback function */
> + f(tim);
> +
> + __TIMER_STAT_ADD(privp, pending, -1);
> +
> + /* the timer was stopped or reloaded by the callback
> + * function, we have nothing to do here
> + */
> + if (privp->updated == 1)
> + continue;
> +
> + if (tim->period == 0) {
> + /* remove from done list and mark timer as stopped */
> + status.state = RTE_TIMER_STOP;
> + status.owner = RTE_TIMER_NO_OWNER;
> + rte_wmb();
> + tim->status.u32 = status.u32;
> + } else {
> + /* keep it in list and mark timer as pending */
> + rte_spinlock_lock(
> + &data->priv_timer[this_lcore].list_lock);
> + status.state = RTE_TIMER_PENDING;
> + __TIMER_STAT_ADD(data->priv_timer, pending, 1);
> + status.owner = (int16_t)this_lcore;
> + rte_wmb();
> + tim->status.u32 = status.u32;
> + __rte_timer_reset(tim, tim->expire + tim->period,
> + tim->period, this_lcore, tim->f, tim->arg, 1,
> + data);
> + rte_spinlock_unlock(
> + &data->priv_timer[this_lcore].list_lock);
> + }
> +
> + privp->running_tim = NULL;
> + }
> +
> + return 0;
> +}
> +
> /* dump statistics about timers */
> -void rte_timer_dump_stats(FILE *f)
> +static void
> +__rte_timer_dump_stats(struct rte_timer_data *timer_data __rte_unused, FILE *f)
> {
> #ifdef RTE_LIBRTE_TIMER_DEBUG
> struct rte_timer_debug_stats sum;
> unsigned lcore_id;
> + struct priv_timer *priv_timer = timer_data->priv_timer;
>
> memset(&sum, 0, sizeof(sum));
> for (lcore_id = 0; lcore_id < RTE_MAX_LCORE; lcore_id++) {
> @@ -615,3 +1024,31 @@ void rte_timer_dump_stats(FILE *f)
> fprintf(f, "No timer statistics, RTE_LIBRTE_TIMER_DEBUG is disabled\n");
> #endif
> }
> +
> +void
> +rte_timer_dump_stats_v20(FILE *f)
> +{
> + __rte_timer_dump_stats(&default_timer_data, f);
> +}
> +VERSION_SYMBOL(rte_timer_dump_stats, _v20, 2.0);
> +
> +int
> +rte_timer_dump_stats_v1905(FILE *f)
> +{
> + return rte_timer_alt_dump_stats(default_data_id, f);
> +}
> +MAP_STATIC_SYMBOL(int rte_timer_dump_stats(FILE *f),
> + rte_timer_dump_stats_v1905);
> +BIND_DEFAULT_SYMBOL(rte_timer_dump_stats, _v1905, 19.05);
> +
> +int __rte_experimental
> +rte_timer_alt_dump_stats(uint32_t timer_data_id __rte_unused, FILE *f)
> +{
> + struct rte_timer_data *timer_data;
> +
> + TIMER_DATA_VALID_GET_OR_ERR_RET(timer_data_id, timer_data, -EINVAL);
> +
> + __rte_timer_dump_stats(timer_data, f);
> +
> + return 0;
> +}
> diff --git a/lib/librte_timer/rte_timer.h b/lib/librte_timer/rte_timer.h
> index 9b95cd2..bee1676 100644
> --- a/lib/librte_timer/rte_timer.h
> +++ b/lib/librte_timer/rte_timer.h
> @@ -39,6 +39,7 @@
> #include <stddef.h>
> #include <rte_common.h>
> #include <rte_config.h>
> +#include <rte_spinlock.h>
>
> #ifdef __cplusplus
> extern "C" {
> @@ -132,12 +133,68 @@ struct rte_timer
> #endif
>
> /**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * Allocate a timer data instance in shared memory to track a set of pending
> + * timer lists.
> + *
> + * @param id_ptr
> + * Pointer to variable into which to write the identifier of the allocated
> + * timer data instance.
> + *
> + * @return
> + * - 0: Success
> + * - -ENOSPC: maximum number of timer data instances already allocated
> + */
> +int __rte_experimental rte_timer_data_alloc(uint32_t *id_ptr);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * Deallocate a timer data instance.
> + *
> + * @param id
> + * Identifier of the timer data instance to deallocate.
> + *
> + * @return
> + * - 0: Success
> + * - -EINVAL: invalid timer data instance identifier
> + */
> +int __rte_experimental rte_timer_data_dealloc(uint32_t id);
> +
> +/**
> * Initialize the timer library.
> *
> * Initializes internal variables (list, locks and so on) for the RTE
> * timer library.
> */
> -void rte_timer_subsystem_init(void);
> +void rte_timer_subsystem_init_v20(void);
> +
> +/**
> + * Initialize the timer library.
> + *
> + * Initializes internal variables (list, locks and so on) for the RTE
> + * timer library.
> + *
> + * @return
> + * - 0: Success
> + * - -EEXIST: Returned in secondary process when primary process has not
> + * yet initialized the timer subsystem
> + * - -ENOMEM: Unable to allocate memory needed to initialize timer
> + * subsystem
> + */
> +int rte_timer_subsystem_init_v1905(void);
> +int rte_timer_subsystem_init(void);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * Free timer subsystem resources.
> + */
> +void __rte_experimental rte_timer_subsystem_finalize(void);
>
> /**
> * Initialize a timer handle.
> @@ -193,6 +250,12 @@ void rte_timer_init(struct rte_timer *tim);
> * - 0: Success; the timer is scheduled.
> * - (-1): Timer is in the RUNNING or CONFIG state.
> */
> +int rte_timer_reset_v20(struct rte_timer *tim, uint64_t ticks,
> + enum rte_timer_type type, unsigned int tim_lcore,
> + rte_timer_cb_t fct, void *arg);
> +int rte_timer_reset_v1905(struct rte_timer *tim, uint64_t ticks,
> + enum rte_timer_type type, unsigned int tim_lcore,
> + rte_timer_cb_t fct, void *arg);
> int rte_timer_reset(struct rte_timer *tim, uint64_t ticks,
> enum rte_timer_type type, unsigned tim_lcore,
> rte_timer_cb_t fct, void *arg);
> @@ -252,9 +315,10 @@ rte_timer_reset_sync(struct rte_timer *tim, uint64_t ticks,
> * - 0: Success; the timer is stopped.
> * - (-1): The timer is in the RUNNING or CONFIG state.
> */
> +int rte_timer_stop_v20(struct rte_timer *tim);
> +int rte_timer_stop_v1905(struct rte_timer *tim);
> int rte_timer_stop(struct rte_timer *tim);
>
> -
> /**
> * Loop until rte_timer_stop() succeeds.
> *
> @@ -292,7 +356,25 @@ int rte_timer_pending(struct rte_timer *tim);
> * function. However, the more often the function is called, the more
> * CPU resources it will use.
> */
> -void rte_timer_manage(void);
> +void rte_timer_manage_v20(void);
> +
> +/**
> + * Manage the timer list and execute callback functions.
> + *
> + * This function must be called periodically from EAL lcores
> + * main_loop(). It browses the list of pending timers and runs all
> + * timers that are expired.
> + *
> + * The precision of the timer depends on the call frequency of this
> + * function. However, the more often the function is called, the more
> + * CPU resources it will use.
> + *
> + * @return
> + * - 0: Success
> + * - -EINVAL: timer subsystem not yet initialized
> + */
> +int rte_timer_manage_v1905(void);
> +int rte_timer_manage(void);
>
> /**
> * Dump statistics about timers.
> @@ -300,7 +382,143 @@ void rte_timer_manage(void);
> * @param f
> * A pointer to a file for output
> */
> -void rte_timer_dump_stats(FILE *f);
> +void rte_timer_dump_stats_v20(FILE *f);
> +
> +/**
> + * Dump statistics about timers.
> + *
> + * @param f
> + * A pointer to a file for output
> + * @return
> + * - 0: Success
> + * - -EINVAL: timer subsystem not yet initialized
> + */
> +int rte_timer_dump_stats_v1905(FILE *f);
> +int rte_timer_dump_stats(FILE *f);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * This function is the same as rte_timer_reset(), except that it allows a
> + * caller to specify the rte_timer_data instance containing the list to which
> + * the timer should be added.
> + *
> + * @see rte_timer_reset()
> + *
> + * @param timer_data_id
> + * An identifier indicating which instance of timer data should be used for
> + * this operation.
> + * @param tim
> + * The timer handle.
> + * @param ticks
> + * The number of cycles (see rte_get_hpet_hz()) before the callback
> + * function is called.
> + * @param type
> + * The type can be either:
> + * - PERIODICAL: The timer is automatically reloaded after execution
> + * (returns to the PENDING state)
> + * - SINGLE: The timer is one-shot, that is, the timer goes to a
> + * STOPPED state after execution.
> + * @param tim_lcore
> + * The ID of the lcore where the timer callback function has to be
> + * executed. If tim_lcore is LCORE_ID_ANY, the timer library will
> + * launch it on a different core for each call (round-robin).
> + * @param fct
> + * The callback function of the timer. This parameter can be NULL if (and
> + * only if) rte_timer_alt_manage() will be used to manage this timer.
> + * @param arg
> + * The user argument of the callback function.
> + * @return
> + * - 0: Success; the timer is scheduled.
> + * - (-1): Timer is in the RUNNING or CONFIG state.
> + * - -EINVAL: invalid timer_data_id
> + */
> +int __rte_experimental
> +rte_timer_alt_reset(uint32_t timer_data_id, struct rte_timer *tim,
> + uint64_t ticks, enum rte_timer_type type,
> + unsigned int tim_lcore, rte_timer_cb_t fct, void *arg);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * This function is the same as rte_timer_stop(), except that it allows a
> + * caller to specify the rte_timer_data instance containing the list from which
> + * this timer should be removed.
> + *
> + * @see rte_timer_stop()
> + *
> + * @param timer_data_id
> + * An identifier indicating which instance of timer data should be used for
> + * this operation.
> + * @param tim
> + * The timer handle.
> + * @return
> + * - 0: Success; the timer is stopped.
> + * - (-1): The timer is in the RUNNING or CONFIG state.
> + * - -EINVAL: invalid timer_data_id
> + */
> +int __rte_experimental
> +rte_timer_alt_stop(uint32_t timer_data_id, struct rte_timer *tim);
> +
> +/**
> + * Callback function type for rte_timer_alt_manage().
> + */
> +typedef void (*rte_timer_alt_manage_cb_t)(void *);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * Manage a set of timer lists and execute the specified callback function for
> + * all expired timers. This function is similar to rte_timer_manage(), except
> + * that it allows a caller to specify the timer_data instance that should
> + * be operated on, as well as a set of lcore IDs identifying which timer lists
> + * should be processed. Callback functions of individual timers are ignored.
> + *
> + * @see rte_timer_manage()
> + *
> + * @param timer_data_id
> + * An identifier indicating which instance of timer data should be used for
> + * this operation.
> + * @param poll_lcores
> + * An array of lcore ids identifying the timer lists that should be processed.
> + * NULL is allowed - if NULL, the timer list corresponding to the lcore
> + * calling this routine is processed (same as rte_timer_manage()).
> + * @param n_poll_lcores
> + * The size of the poll_lcores array. If 'poll_lcores' is NULL, this parameter
> + * is ignored.
> + * @param f
> + * The callback function which should be called for all expired timers.
> + * @return
> + * - 0: success
> + * - -EINVAL: invalid timer_data_id
> + */
> +int __rte_experimental
> +rte_timer_alt_manage(uint32_t timer_data_id, unsigned int *poll_lcores,
> + int n_poll_lcores, rte_timer_alt_manage_cb_t f);
> +
> +/**
> + * @warning
> + * @b EXPERIMENTAL: this API may change without prior notice
> + *
> + * This function is the same as rte_timer_dump_stats(), except that it allows
> + * the caller to specify the rte_timer_data instance that should be used.
> + *
> + * @see rte_timer_dump_stats()
> + *
> + * @param timer_data_id
> + * An identifier indicating which instance of timer data should be used for
> + * this operation.
> + * @param f
> + * A pointer to a file for output
> + * @return
> + * - 0: success
> + * - -EINVAL: invalid timer_data_id
> + */
> +int __rte_experimental
> +rte_timer_alt_dump_stats(uint32_t timer_data_id, FILE *f);
>
> #ifdef __cplusplus
> }
> diff --git a/lib/librte_timer/rte_timer_version.map b/lib/librte_timer/rte_timer_version.map
> index 9b2e4b8..c2e5836 100644
> --- a/lib/librte_timer/rte_timer_version.map
> +++ b/lib/librte_timer/rte_timer_version.map
> @@ -13,3 +13,25 @@ DPDK_2.0 {
>
> local: *;
> };
> +
> +DPDK_19.05 {
> + global:
> +
> + rte_timer_dump_stats;
> + rte_timer_manage;
> + rte_timer_reset;
> + rte_timer_stop;
> + rte_timer_subsystem_init;
> +} DPDK_2.0;
> +
> +EXPERIMENTAL {
> + global:
> +
> + rte_timer_alt_dump_stats;
> + rte_timer_alt_manage;
> + rte_timer_alt_reset;
> + rte_timer_alt_stop;
> + rte_timer_data_alloc;
> + rte_timer_data_dealloc;
> + rte_timer_subsystem_finalize;
> +};
> --
> 2.6.4
>
>
next prev parent reply other threads:[~2019-03-21 1:02 UTC|newest]
Thread overview: 77+ messages / expand[flat|nested] mbox.gz Atom feed top
2018-11-29 23:35 [dpdk-dev] [PATCH 0/3] new software event timer adapter Erik Gabriel Carrillo
2018-11-29 23:35 ` [dpdk-dev] [PATCH 1/3] timer: allow timer management in shared memory Erik Gabriel Carrillo
2018-11-29 23:35 ` [dpdk-dev] [PATCH 2/3] timer: add function to stop all timers in a list Erik Gabriel Carrillo
2018-11-29 23:35 ` [dpdk-dev] [PATCH 3/3] eventdev: add new software event timer adapter Erik Gabriel Carrillo
2018-11-30 7:26 ` [dpdk-dev] [PATCH 0/3] " Pavan Nikhilesh
2018-11-30 19:07 ` Carrillo, Erik G
2018-12-07 17:52 ` [dpdk-dev] [PATCH v2 0/2] Timer library changes Erik Gabriel Carrillo
2018-12-07 17:52 ` [dpdk-dev] [PATCH v2 1/2] timer: allow timer management in shared memory Erik Gabriel Carrillo
2018-12-07 18:10 ` Stephen Hemminger
2018-12-07 19:21 ` Carrillo, Erik G
2018-12-07 17:53 ` [dpdk-dev] [PATCH v2 2/2] timer: add function to stop all timers in a list Erik Gabriel Carrillo
2018-12-13 22:26 ` [dpdk-dev] [PATCH v3 0/2] Timer library changes Erik Gabriel Carrillo
2018-12-13 22:26 ` [dpdk-dev] [PATCH v3 1/2] timer: allow timer management in shared memory Erik Gabriel Carrillo
2018-12-13 22:26 ` [dpdk-dev] [PATCH v3 2/2] timer: add function to stop all timers in a list Erik Gabriel Carrillo
2018-12-19 3:35 ` [dpdk-dev] [PATCH v3 0/2] Timer library changes Thomas Monjalon
2018-12-19 7:33 ` Mattias Rönnblom
2019-03-05 22:41 ` Carrillo, Erik G
2019-03-05 22:58 ` [dpdk-dev] [dpdk-techboard] " Thomas Monjalon
2019-03-06 18:54 ` Carrillo, Erik G
2019-03-06 20:17 ` Thomas Monjalon
2019-03-06 2:39 ` [dpdk-dev] " Varghese, Vipin
2019-03-06 15:15 ` Carrillo, Erik G
2019-03-07 2:33 ` Varghese, Vipin
2019-03-06 17:20 ` [dpdk-dev] [PATCH v4 " Erik Gabriel Carrillo
2019-03-06 17:20 ` [dpdk-dev] [PATCH v4 1/2] timer: allow timer management in shared memory Erik Gabriel Carrillo
2019-03-20 13:52 ` Sanford, Robert
2019-03-20 13:52 ` Sanford, Robert
2019-03-21 1:01 ` Carrillo, Erik G [this message]
2019-03-21 1:01 ` Carrillo, Erik G
2019-03-27 14:03 ` Thomas Monjalon
2019-03-27 14:03 ` Thomas Monjalon
2019-03-28 12:42 ` Carrillo, Erik G
2019-03-28 12:42 ` Carrillo, Erik G
2019-04-15 21:49 ` Carrillo, Erik G
2019-04-15 21:49 ` Carrillo, Erik G
2019-03-06 17:20 ` [dpdk-dev] [PATCH v4 2/2] timer: add function to stop all timers in a list Erik Gabriel Carrillo
2019-04-15 21:41 ` [dpdk-dev] [PATCH v5 0/2] Timer library changes Erik Gabriel Carrillo
2019-04-15 21:41 ` Erik Gabriel Carrillo
2019-04-15 21:41 ` [dpdk-dev] [PATCH v5 1/2] timer: allow timer management in shared memory Erik Gabriel Carrillo
2019-04-15 21:41 ` Erik Gabriel Carrillo
2019-04-17 17:09 ` Thomas Monjalon
2019-04-17 17:09 ` Thomas Monjalon
2019-04-15 21:41 ` [dpdk-dev] [PATCH v5 2/2] timer: add function to stop all timers in a list Erik Gabriel Carrillo
2019-04-15 21:41 ` Erik Gabriel Carrillo
2019-04-17 19:54 ` [dpdk-dev] [PATCH v5 0/2] Timer library changes Thomas Monjalon
2019-04-17 19:54 ` Thomas Monjalon
2018-12-07 20:34 ` [dpdk-dev] [PATCH v2 0/1] New software event timer adapter Erik Gabriel Carrillo
2018-12-07 20:34 ` [dpdk-dev] [PATCH v2 1/1] eventdev: add new " Erik Gabriel Carrillo
2018-12-09 19:17 ` Mattias Rönnblom
2018-12-10 17:17 ` Carrillo, Erik G
2018-12-14 15:45 ` [dpdk-dev] [PATCH v3 0/1] New " Erik Gabriel Carrillo
2018-12-14 15:45 ` [dpdk-dev] [PATCH v3 1/1] eventdev: add new " Erik Gabriel Carrillo
2018-12-14 21:15 ` Mattias Rönnblom
2018-12-14 23:04 ` Carrillo, Erik G
2018-12-14 23:15 ` [dpdk-dev] [PATCH v4 0/1] New " Erik Gabriel Carrillo
2018-12-14 23:15 ` [dpdk-dev] [PATCH v4 1/1] eventdev: add new " Erik Gabriel Carrillo
2018-12-18 20:11 ` [dpdk-dev] [EXT] [PATCH v4 0/1] New " Jerin Jacob Kollanukkaran
2018-12-18 20:14 ` Carrillo, Erik G
2019-04-22 14:57 ` [dpdk-dev] [PATCH v5 " Erik Gabriel Carrillo
2019-04-22 14:57 ` Erik Gabriel Carrillo
2019-04-22 14:57 ` [dpdk-dev] [PATCH v5 1/1] eventdev: add new " Erik Gabriel Carrillo
2019-04-22 14:57 ` Erik Gabriel Carrillo
2019-04-26 15:14 ` [dpdk-dev] [PATCH v6 0/1] New " Erik Gabriel Carrillo
2019-04-26 15:14 ` Erik Gabriel Carrillo
2019-04-26 15:14 ` [dpdk-dev] [PATCH v6 1/1] eventdev: add new " Erik Gabriel Carrillo
2019-04-26 15:14 ` Erik Gabriel Carrillo
2019-04-26 18:51 ` Honnappa Nagarahalli
2019-04-26 18:51 ` Honnappa Nagarahalli
2019-04-26 18:58 ` Carrillo, Erik G
2019-04-26 18:58 ` Carrillo, Erik G
2019-06-05 13:34 ` Jerin Jacob Kollanukkaran
2019-06-19 15:14 ` [dpdk-dev] [PATCH v7 0/1] New " Erik Gabriel Carrillo
2019-06-19 15:14 ` [dpdk-dev] [PATCH v7 1/1] eventdev: add new " Erik Gabriel Carrillo
2019-06-19 16:25 ` [dpdk-dev] [PATCH v8 0/1] New " Erik Gabriel Carrillo
2019-06-19 16:25 ` [dpdk-dev] [PATCH v8 1/1] eventdev: add new " Erik Gabriel Carrillo
2019-06-24 6:12 ` Jerin Jacob Kollanukkaran
2019-06-25 6:06 ` Jerin Jacob Kollanukkaran
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=D926C980-58E3-4476-9FA9-D9180C6863A2@intel.com \
--to=erik.g.carrillo@intel.com \
--cc=dev@dpdk.org \
--cc=nhorman@tuxdriver.com \
--cc=rsanford@akamai.com \
--cc=thomas@monjalon.net \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).