Hi Alan, Thanks for your work!
I do have some comments (see below), but generally looks good to me. > -----Original Message----- > From: alangordonde...@gmail.com [mailto:alangordonde...@gmail.com] > Sent: Tuesday, October 3, 2017 10:22 AM > To: Dumitrescu, Cristian <cristian.dumitre...@intel.com> > Cc: dev@dpdk.org; Alan Dewar <alan.de...@att.com> > Subject: [PATCH v4] sched: make RED scaling configurable > > From: Alan Dewar <alan.de...@att.com> > > The RED code stores the weighted moving average in a 32-bit integer as > a pseudo fixed-point floating number with 10 fractional bits. Twelve > other bits are used to encode the filter weight, leaving just 10 bits > for the queue length. This limits the maximum queue length supported > by RED queues to 1024 packets. > > Introduce a new API to allow the RED scaling factor to be configured > based upon maximum queue length. If this API is not called, the RED > scaling factor remains at its default value. > > Added some new RED scaling unit-tests to test with RED queue-lengths > up to 8192 packets long. > > Signed-off-by: Alan Dewar <alan.de...@att.com> > --- > lib/librte_sched/rte_red.c | 53 ++++++- > lib/librte_sched/rte_red.h | 63 ++++++-- > lib/librte_sched/rte_sched_version.map | 6 + > test/test/test_red.c | 274 > ++++++++++++++++++++++++++++++++- > 4 files changed, 374 insertions(+), 22 deletions(-) > > diff --git a/lib/librte_sched/rte_red.c b/lib/librte_sched/rte_red.c > index ade57d1..0dc8d28 100644 > --- a/lib/librte_sched/rte_red.c > +++ b/lib/librte_sched/rte_red.c > @@ -43,6 +43,8 @@ > static int rte_red_init_done = 0; /**< Flag to indicate that global > initialisation is done */ > uint32_t rte_red_rand_val = 0; /**< Random value cache */ > uint32_t rte_red_rand_seed = 0; /**< Seed for random number > generation */ > +uint8_t rte_red_scaling = RTE_RED_SCALING_DEFAULT; > +uint16_t rte_red_max_threshold = RTE_RED_DEFAULT_QUEUE_LENGTH - > 1; > > /** > * table[i] = log2(1-Wq) * Scale * -1 > @@ -66,7 +68,7 @@ __rte_red_init_tables(void) > double scale = 0.0; > double table_size = 0.0; > > - scale = (double)(1 << RTE_RED_SCALING); > + scale = (double)(1 << rte_red_scaling); > table_size = (double)(RTE_DIM(rte_red_pow2_frac_inv)); > > for (i = 0; i < RTE_DIM(rte_red_pow2_frac_inv); i++) { > @@ -119,7 +121,7 @@ rte_red_config_init(struct rte_red_config *red_cfg, > if (red_cfg == NULL) { > return -1; > } > - if (max_th > RTE_RED_MAX_TH_MAX) { > + if (max_th > rte_red_max_threshold) { > return -2; > } > if (min_th >= max_th) { > @@ -148,11 +150,52 @@ rte_red_config_init(struct rte_red_config > *red_cfg, > rte_red_init_done = 1; > } > > - red_cfg->min_th = ((uint32_t) min_th) << (wq_log2 + > RTE_RED_SCALING); > - red_cfg->max_th = ((uint32_t) max_th) << (wq_log2 + > RTE_RED_SCALING); > - red_cfg->pa_const = (2 * (max_th - min_th) * maxp_inv) << > RTE_RED_SCALING; > + red_cfg->min_th = ((uint32_t) min_th) << (wq_log2 + > rte_red_scaling); > + red_cfg->max_th = ((uint32_t) max_th) << (wq_log2 + > rte_red_scaling); > + red_cfg->pa_const = (2 * (max_th - min_th) * maxp_inv) << > + rte_red_scaling; > red_cfg->maxp_inv = maxp_inv; > red_cfg->wq_log2 = wq_log2; > > return 0; > } > + > +int > +rte_red_set_scaling(uint16_t max_red_queue_length) > +{ > + int8_t count; > + > + if (rte_red_init_done) > + /** > + * Can't change the scaling once the red table has been > + * computed. > + */ > + return -1; > + > + if (max_red_queue_length < RTE_RED_MIN_QUEUE_LENGTH) > + return -2; > + > + if (max_red_queue_length > RTE_RED_MAX_QUEUE_LENGTH) > + return -3; > + > + if (!rte_is_power_of_2(max_red_queue_length)) > + return -4; > + > + count = 0; > + while (max_red_queue_length != 0) { > + max_red_queue_length >>= 1; > + count++; > + } > + > + rte_red_scaling -= count - RTE_RED_SCALING_DEFAULT; > + rte_red_max_threshold = max_red_queue_length - 1; > + return 0; > +} > + > +void > +rte_red_reset_scaling(void) > +{ > + rte_red_init_done = 0; > + rte_red_scaling = RTE_RED_SCALING_DEFAULT; > + rte_red_max_threshold = RTE_RED_DEFAULT_QUEUE_LENGTH - 1; > +} Why do we need this function? These global variables are already initialized at the top of the file. My vote is to remove it. > diff --git a/lib/librte_sched/rte_red.h b/lib/librte_sched/rte_red.h > index ca12227..be1fb0f 100644 > --- a/lib/librte_sched/rte_red.h > +++ b/lib/librte_sched/rte_red.h > @@ -52,14 +52,31 @@ extern "C" { > #include <rte_cycles.h> > #include <rte_branch_prediction.h> > > -#define RTE_RED_SCALING 10 /**< Fraction size > for fixed- > point */ > -#define RTE_RED_S (1 << 22) /**< Packet size > multiplied > by number of leaf queues */ > -#define RTE_RED_MAX_TH_MAX 1023 /**< Max threshold > limit in fixed point format */ > -#define RTE_RED_WQ_LOG2_MIN 1 /**< Min inverse > filter > weight value */ > -#define RTE_RED_WQ_LOG2_MAX 12 /**< Max inverse > filter weight value */ > -#define RTE_RED_MAXP_INV_MIN 1 /**< Min inverse mark > probability value */ > -#define RTE_RED_MAXP_INV_MAX 255 /**< Max inverse > mark probability value */ > -#define RTE_RED_2POW16 (1<<16) /**< 2 power 16 */ > +/**< Default fraction size for fixed-point */ > +#define RTE_RED_SCALING_DEFAULT 10 Let's keep the name of this macro undchanges, i.e. RTE_RED_SCALING. IMO it is ckear enough this is the default value (also stated in the comment). > + > +/**< Packet size multiplied by number of leaf queues */ > +#define RTE_RED_S (1 << 22) > + > +/**< Minimum, default and maximum RED queue length */ > +#define RTE_RED_MIN_QUEUE_LENGTH 64 > +#define RTE_RED_DEFAULT_QUEUE_LENGTH 1024 > +#define RTE_RED_MAX_QUEUE_LENGTH 8192 > + > +/**< Min inverse filter weight value */ > +#define RTE_RED_WQ_LOG2_MIN 1 > + > +/**< Max inverse filter weight value */ > +#define RTE_RED_WQ_LOG2_MAX 12 > + > +/**< Min inverse mark probability value */ > +#define RTE_RED_MAXP_INV_MIN 1 > + > +/**< Max inverse mark probability value */ > +#define RTE_RED_MAXP_INV_MAX 255 > + > +/**< 2 power 16 */ > +#define RTE_RED_2POW16 (1<<16) > #define RTE_RED_INT16_NBITS (sizeof(uint16_t) * CHAR_BIT) > #define RTE_RED_WQ_LOG2_NUM (RTE_RED_WQ_LOG2_MAX - > RTE_RED_WQ_LOG2_MIN + 1) > All the above Doxygen comments are incorrectly formatted: as you want to change their behavior to refer to the variable _after_ the comment, they have to start with /**, not with /**< (which should be used if the comment refers to the variable _before_ the comment). Please fix. > @@ -71,6 +88,8 @@ extern uint32_t rte_red_rand_val; > extern uint32_t rte_red_rand_seed; > extern uint16_t rte_red_log2_1_minus_Wq[RTE_RED_WQ_LOG2_NUM]; > extern uint16_t rte_red_pow2_frac_inv[16]; > +extern uint8_t rte_red_scaling; > +extern uint16_t rte_red_max_threshold; > > /** > * RED configuration parameters passed by user > @@ -137,6 +156,26 @@ rte_red_config_init(struct rte_red_config *red_cfg, > const uint16_t maxp_inv); > > /** > + * @brief Configures the global setting for the RED scaling factor > + * > + * @param max_red_queue_length [in] must be a power of two > + * > + * @return Operation status > + * @retval 0 success > + * @retval !0 error > + */ > +int > +rte_red_set_scaling(uint16_t max_red_queue_length); > + > +/** > + * @brief Reset the RED scaling factor - only for use by RED unit-tests > + * > + * @return Operation status > + */ > +void > +rte_red_reset_scaling(void); As stated above, this function is probably not useful and my vote is to remove it. > + > +/** > * @brief Generate random number for RED > * > * Implemenetation based on: > @@ -206,7 +245,7 @@ __rte_red_calc_qempty_factor(uint8_t wq_log2, > uint16_t m) > f = (n >> 6) & 0xf; > n >>= 10; > > - if (n < RTE_RED_SCALING) > + if (n < rte_red_scaling) > return (uint16_t) ((rte_red_pow2_frac_inv[f] + (1 << (n - 1))) > >> n); > > return 0; > @@ -258,7 +297,9 @@ rte_red_enqueue_empty(const struct rte_red_config > *red_cfg, > if (m >= RTE_RED_2POW16) { > red->avg = 0; > } else { > - red->avg = (red->avg >> RTE_RED_SCALING) * > __rte_red_calc_qempty_factor(red_cfg->wq_log2, (uint16_t) m); > + red->avg = (red->avg >> rte_red_scaling) * > + __rte_red_calc_qempty_factor(red_cfg->wq_log2, > + (uint16_t) m); > } > > return 0; > @@ -365,7 +406,7 @@ rte_red_enqueue_nonempty(const struct > rte_red_config *red_cfg, > */ > > /* avg update */ > - red->avg += (q << RTE_RED_SCALING) - (red->avg >> red_cfg- > >wq_log2); > + red->avg += (q << rte_red_scaling) - (red->avg >> red_cfg- > >wq_log2); > > /* avg < min_th: do not mark the packet */ > if (red->avg < red_cfg->min_th) { > diff --git a/lib/librte_sched/rte_sched_version.map > b/lib/librte_sched/rte_sched_version.map > index 3aa159a..92e51f8 100644 > --- a/lib/librte_sched/rte_sched_version.map > +++ b/lib/librte_sched/rte_sched_version.map > @@ -29,3 +29,9 @@ DPDK_2.1 { > rte_sched_port_pkt_read_color; > > } DPDK_2.0; > + > +DPDK_17.08 { > + global; > + > + rte_red_set_scaling; > +} DPDK_2.1; > diff --git a/test/test/test_red.c b/test/test/test_red.c > index 348075d..f2d50ef 100644 > --- a/test/test/test_red.c > +++ b/test/test/test_red.c > @@ -67,6 +67,7 @@ struct test_rte_red_config { /**< Test structure > for RTE_RED config */ > uint32_t min_th; /**< Queue minimum threshold */ > uint32_t max_th; /**< Queue maximum threshold */ > uint8_t *maxp_inv; /**< Inverse mark probability */ > + uint16_t max_queue_len; /**< Maximum queue length */ > }; > > struct test_queue { /**< Test structure for RTE_RED Queues */ > @@ -181,7 +182,7 @@ static uint32_t rte_red_get_avg_int(const struct > rte_red_config *red_cfg, > /** > * scale by 1/n and convert from fixed-point to integer > */ > - return red->avg >> (RTE_RED_SCALING + red_cfg->wq_log2); > + return red->avg >> (rte_red_scaling + red_cfg->wq_log2); > } > > static double rte_red_get_avg_float(const struct rte_red_config *red_cfg, > @@ -190,7 +191,7 @@ static double rte_red_get_avg_float(const struct > rte_red_config *red_cfg, > /** > * scale by 1/n and convert from fixed-point to floating-point > */ > - return ldexp((double)red->avg, -(RTE_RED_SCALING + red_cfg- > >wq_log2)); > + return ldexp((double)red->avg, -(rte_red_scaling + red_cfg- > >wq_log2)); > } > > static void rte_red_set_avg_int(const struct rte_red_config *red_cfg, > @@ -200,7 +201,7 @@ static void rte_red_set_avg_int(const struct > rte_red_config *red_cfg, > /** > * scale by n and convert from integer to fixed-point > */ > - red->avg = avg << (RTE_RED_SCALING + red_cfg->wq_log2); > + red->avg = avg << (rte_red_scaling + red_cfg->wq_log2); > } > > static double calc_exp_avg_on_empty(double avg, uint32_t n, uint32_t > time_diff) > @@ -280,10 +281,23 @@ static enum test_result > test_rte_red_init(struct test_config *tcfg) > { > unsigned i = 0; > + int ret; > > tcfg->tvar->clk_freq = rte_get_timer_hz(); > init_port_ts( tcfg->tvar->clk_freq ); > > + rte_red_reset_scaling(); > + ret = rte_red_set_scaling(tcfg->tconfig->max_queue_len); > + if (ret) { > + printf("rte_red_set_scaling failed: %d\n", ret); > + return FAIL; > + } > + if (rte_red_max_threshold < tcfg->tconfig->max_th) { > + printf("rte_red_max_th (%u) < tconfig->max_th (%u)\n", > + rte_red_max_threshold, tcfg->tconfig->max_th); > + return FAIL; > + } > + > for (i = 0; i < tcfg->tconfig->num_cfg; i++) { > if (rte_red_config_init(&tcfg->tconfig->rconfig[i], > (uint16_t)tcfg->tconfig->wq_log2[i], > @@ -385,6 +399,7 @@ static struct test_rte_red_config ft_tconfig = { > .min_th = 32, > .max_th = 128, > .maxp_inv = ft_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue ft_tqueue = { > @@ -554,6 +569,7 @@ static struct test_rte_red_config ft2_tconfig = { > .min_th = 32, > .max_th = 128, > .maxp_inv = ft2_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_config func_test2_config = { > @@ -576,6 +592,41 @@ static struct test_config func_test2_config = { > .tlevel = ft2_tlevel, > }; > > +/** > + * Test F2: functional test 2 - with long queues and smaller red-scaling > factor > + */ > +static uint32_t ft2_tlevel_scaling[] = {8190}; > + > +static struct test_rte_red_config ft2_tconfig_scaling = { > + .rconfig = ft2_rconfig, > + .num_cfg = RTE_DIM(ft2_rconfig), > + .wq_log2 = ft2_wq_log2, > + .min_th = 32, > + .max_th = 8191, > + .maxp_inv = ft2_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config func_test2_config_scaling = { > + .ifname = "functional test 2 interface", > + .msg = "functional test 2 : use several RED configurations and long > queues,\n" > + " increase average queue size to just below > maximum threshold,\n" > + " compare drop rate to drop probability\n\n", > + .htxt = "RED config " > + "avg queue size " > + "min threshold " > + "max threshold " > + "drop prob % " > + "drop rate % " > + "diff % " > + "tolerance % " > + "\n", > + .tconfig = &ft2_tconfig_scaling, > + .tqueue = &ft_tqueue, > + .tvar = &ft_tvar, > + .tlevel = ft2_tlevel_scaling, > +}; > + > static enum test_result func_test2(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -662,6 +713,7 @@ static struct test_rte_red_config ft3_tconfig = { > .min_th = 32, > .max_th = 1023, > .maxp_inv = ft_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_config func_test3_config = { > @@ -683,6 +735,40 @@ static struct test_config func_test3_config = { > .tlevel = ft3_tlevel, > }; > > +/** > + * Test F3: functional test 3 - with large queues and smaller red scaling > factor > + */ > +static uint32_t ft3_tlevel_scaling[] = {8190}; > + > +static struct test_rte_red_config ft3_tconfig_scaling = { > + .rconfig = ft_wrconfig, > + .num_cfg = RTE_DIM(ft_wrconfig), > + .wq_log2 = ft_wq_log2, > + .min_th = 32, > + .max_th = 8191, > + .maxp_inv = ft_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config func_test3_config_scaling = { > + .ifname = "functional test 3 interface", > + .msg = "functional test 3 : use one RED configuration and long > queues,\n" > + " increase average queue size to target level,\n" > + " dequeue all packets until queue is empty,\n" > + " confirm that average queue size is computed > correctly while queue is empty\n\n", > + .htxt = "q avg before " > + "q avg after " > + "expected " > + "difference % " > + "tolerance % " > + "result " > + "\n", > + .tconfig = &ft3_tconfig_scaling, > + .tqueue = &ft_tqueue, > + .tvar = &ft_tvar, > + .tlevel = ft3_tlevel_scaling, > +}; > + > static enum test_result func_test3(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -776,6 +862,7 @@ static struct test_rte_red_config ft4_tconfig = { > .max_th = 1023, > .wq_log2 = ft4_wq_log2, > .maxp_inv = ft_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue ft4_tqueue = { > @@ -810,6 +897,42 @@ static struct test_config func_test4_config = { > .tlevel = ft4_tlevel, > }; > > +/** > + * Test F4: functional test 4 > + */ > +static uint32_t ft4_tlevel_scaling[] = {8190}; > + > +static struct test_rte_red_config ft4_tconfig_scaling = { > + .rconfig = ft_wrconfig, > + .num_cfg = RTE_DIM(ft_wrconfig), > + .min_th = 32, > + .max_th = 8191, > + .wq_log2 = ft4_wq_log2, > + .maxp_inv = ft_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config func_test4_config_scaling = { > + .ifname = "functional test 4 interface", > + .msg = "functional test 4 : use one RED configuration on long > queue,\n" > + " increase average queue size to target level,\n" > + " dequeue all packets until queue is empty,\n" > + " confirm that average queue size is computed > correctly while\n" > + " queue is empty for more than 50 sec,\n" > + " (this test takes 52 sec to run)\n\n", > + .htxt = "q avg before " > + "q avg after " > + "expected " > + "difference % " > + "tolerance % " > + "result " > + "\n", > + .tconfig = &ft4_tconfig_scaling, > + .tqueue = &ft4_tqueue, > + .tvar = &ft_tvar, > + .tlevel = ft4_tlevel_scaling, > +}; > + > static enum test_result func_test4(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -924,6 +1047,7 @@ static struct test_rte_red_config ft5_tconfig = { > .max_th = 128, > .wq_log2 = ft5_wq_log2, > .maxp_inv = ft5_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue ft5_tqueue = { > @@ -970,6 +1094,45 @@ static struct test_config func_test5_config = { > .tlevel = ft5_tlevel, > }; > > + > +/** > + * Test F5: functional test 5 > + */ > +static uint32_t ft5_tlevel_scaling[] = {8190}; > + > +static struct test_rte_red_config ft5_tconfig_scaling = { > + .rconfig = ft5_config, > + .num_cfg = RTE_DIM(ft5_config), > + .min_th = 32, > + .max_th = 8191, > + .wq_log2 = ft5_wq_log2, > + .maxp_inv = ft5_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config func_test5_config_scaling = { > + .ifname = "functional test 5 interface", > + .msg = "functional test 5 : use several long queues (each with its > own run-time data),\n" > + " use several RED configurations (such that each > configuration is shared by multiple queues),\n" > + " increase average queue size to just below > maximum threshold,\n" > + " compare drop rate to drop probability,\n" > + " (this is a larger scale version of functional test > 2)\n\n", > + .htxt = "queue " > + "config " > + "avg queue size " > + "min threshold " > + "max threshold " > + "drop prob % " > + "drop rate % " > + "diff % " > + "tolerance % " > + "\n", > + .tconfig = &ft5_tconfig_scaling, > + .tqueue = &ft5_tqueue, > + .tvar = &ft5_tvar, > + .tlevel = ft5_tlevel_scaling, > +}; > + > static enum test_result func_test5(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -1062,6 +1225,7 @@ static struct test_rte_red_config ft6_tconfig = { > .max_th = 1023, > .wq_log2 = ft6_wq_log2, > .maxp_inv = ft6_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue ft6_tqueue = { > @@ -1078,7 +1242,7 @@ static struct test_queue ft6_tqueue = { > static struct test_config func_test6_config = { > .ifname = "functional test 6 interface", > .msg = "functional test 6 : use several queues (each with its own > run-time data),\n" > - " use several RED configurations (such that each > configuration is sharte_red by multiple queues),\n" > + " use several RED configurations (such that each > configuration is shared by multiple queues),\n" > " increase average queue size to target level,\n" > " dequeue all packets until queue is empty,\n" > " confirm that average queue size is computed > correctly while queue is empty\n" > @@ -1097,6 +1261,44 @@ static struct test_config func_test6_config = { > .tlevel = ft6_tlevel, > }; > > +/** > + * Test F6: functional test 6 > + */ > +static uint32_t ft6_tlevel_scaling[] = {8190}; > + > +static struct test_rte_red_config ft6_tconfig_scaling = { > + .rconfig = ft6_config, > + .num_cfg = RTE_DIM(ft6_config), > + .min_th = 32, > + .max_th = 8191, > + .wq_log2 = ft6_wq_log2, > + .maxp_inv = ft6_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config func_test6_config_scaling = { > + .ifname = "functional test 6 interface", > + .msg = "functional test 6 : use several long queues (each with its > own run-time data),\n" > + " use several RED configurations (such that each > configuration is shared by multiple queues),\n" > + " increase average queue size to target level,\n" > + " dequeue all packets until queue is empty,\n" > + " confirm that average queue size is computed > correctly while queue is empty\n" > + " (this is a larger scale version of functional test > 3)\n\n", > + .htxt = "queue " > + "config " > + "q avg before " > + "q avg after " > + "expected " > + "difference % " > + "tolerance % " > + "result " > + "\n", > + .tconfig = &ft6_tconfig_scaling, > + .tqueue = &ft6_tqueue, > + .tvar = &ft_tvar, > + .tlevel = ft6_tlevel_scaling, > +}; > + > static enum test_result func_test6(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -1195,6 +1397,7 @@ static struct test_rte_red_config pt_tconfig = { > .min_th = 32, > .max_th = 128, > .maxp_inv = pt_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue pt_tqueue = { > @@ -1557,6 +1760,7 @@ static struct test_rte_red_config ovfl_tconfig = { > .min_th = 32, > .max_th = 1023, > .maxp_inv = ovfl_maxp_inv, > + .max_queue_len = RTE_RED_DEFAULT_QUEUE_LENGTH, > }; > > static struct test_queue ovfl_tqueue = { > @@ -1598,7 +1802,7 @@ static struct test_config ovfl_test1_config = { > .ifname = "queue avergage overflow test interface", > .msg = "overflow test 1 : use one RED configuration,\n" > " increase average queue size to target level,\n" > - " check maximum number of bits requirte_red to > represent avg_s\n\n", > + " check maximum number of bits required to > represent avg_s\n\n", > .htxt = "avg queue size " > "wq_log2 " > "fraction bits " > @@ -1615,6 +1819,39 @@ static struct test_config ovfl_test1_config = { > .tlevel = ovfl_tlevel, > }; > > +static uint32_t ovfl_tlevel_scaling[] = {8191}; > + > +static struct test_rte_red_config ovfl_tconfig_scaling = { > + .rconfig = ovfl_wrconfig, > + .num_cfg = RTE_DIM(ovfl_wrconfig), > + .wq_log2 = ovfl_wq_log2, > + .min_th = 32, > + .max_th = 8191, > + .maxp_inv = ovfl_maxp_inv, > + .max_queue_len = 8192, > +}; > + > +static struct test_config ovfl_test1_config_scaling = { > + .ifname = "queue average overflow test interface", > + .msg = "overflow test 1 on long queue: use one RED > configuration,\n" > + " increase average queue size to target level,\n" > + " check maximum number of bits required to > represent avg_s\n\n", > + .htxt = "avg queue size " > + "wq_log2 " > + "fraction bits " > + "max queue avg " > + "num bits " > + "enqueued " > + "dropped " > + "drop prob % " > + "drop rate % " > + "\n", > + .tconfig = &ovfl_tconfig_scaling, > + .tqueue = &ovfl_tqueue, > + .tvar = &ovfl_tvar, > + .tlevel = ovfl_tlevel_scaling, > +}; > + > static enum test_result ovfl_test1(struct test_config *tcfg) > { > enum test_result result = PASS; > @@ -1690,7 +1927,7 @@ static enum test_result ovfl_test1(struct > test_config *tcfg) > printf("%s", tcfg->htxt); > > printf("%-16u%-9u%-15u0x%08x %-10u%-10u%-10u%-13.2lf%- > 13.2lf\n", > - avg, *tcfg->tconfig->wq_log2, RTE_RED_SCALING, > + avg, *tcfg->tconfig->wq_log2, rte_red_scaling, > avg_max, avg_max_bits, > *tcfg->tvar->enqueued, *tcfg->tvar->dropped, > drop_prob * 100.0, drop_rate * 100.0); > @@ -1730,6 +1967,15 @@ struct tests perf_tests[] = { > { &perf2_test6_config, perf2_test }, > }; > > +struct tests scale_tests[] = { > + { &func_test2_config_scaling, func_test2 }, > + { &func_test3_config_scaling, func_test3 }, > + { &func_test4_config_scaling, func_test4 }, > + { &func_test5_config_scaling, func_test5 }, > + { &func_test6_config_scaling, func_test6 }, > + { &ovfl_test1_config_scaling, ovfl_test1 }, > +}; > + > /** > * function to execute the required_red tests > */ > @@ -1866,6 +2112,20 @@ test_red_perf(void) > } > > static int > +test_red_scaling(void) > +{ > + uint32_t num_tests = 0; > + uint32_t num_pass = 0; > + > + if (test_invalid_parameters() < 0) > + return -1; > + > + run_tests(scale_tests, RTE_DIM(scale_tests), &num_tests, > &num_pass); > + show_stats(num_tests, num_pass); > + return tell_the_result(num_tests, num_pass); > +} > + > +static int > test_red_all(void) > { > uint32_t num_tests = 0; > @@ -1876,10 +2136,12 @@ test_red_all(void) > > run_tests(func_tests, RTE_DIM(func_tests), &num_tests, > &num_pass); > run_tests(perf_tests, RTE_DIM(perf_tests), &num_tests, > &num_pass); > + run_tests(scale_tests, RTE_DIM(scale_tests), &num_tests, > &num_pass); > show_stats(num_tests, num_pass); > return tell_the_result(num_tests, num_pass); > } > > REGISTER_TEST_COMMAND(red_autotest, test_red); > REGISTER_TEST_COMMAND(red_perf, test_red_perf); > +REGISTER_TEST_COMMAND(red_scaling, test_red_scaling); > REGISTER_TEST_COMMAND(red_all, test_red_all); > -- > 2.1.4 Regards, Cristian