DPDK 26.03.0-rc1
rte_red.h
Go to the documentation of this file.
1/* SPDX-License-Identifier: BSD-3-Clause
2 * Copyright(c) 2010-2014 Intel Corporation
3 */
4
5#ifndef __RTE_RED_H_INCLUDED__
6#define __RTE_RED_H_INCLUDED__
7
13#include <stdint.h>
14#include <limits.h>
15#include <rte_debug.h>
16#include <rte_cycles.h>
18
19#ifdef __cplusplus
20extern "C" {
21#endif
22
23#define RTE_RED_SCALING 10
24#define RTE_RED_S (1 << 22)
25#define RTE_RED_MAX_TH_MAX 1023
26#define RTE_RED_WQ_LOG2_MIN 1
27#define RTE_RED_WQ_LOG2_MAX 12
28#define RTE_RED_MAXP_INV_MIN 1
29#define RTE_RED_MAXP_INV_MAX 255
30#define RTE_RED_2POW16 (1<<16)
31#define RTE_RED_INT16_NBITS (sizeof(uint16_t) * CHAR_BIT)
32#define RTE_RED_WQ_LOG2_NUM (RTE_RED_WQ_LOG2_MAX - RTE_RED_WQ_LOG2_MIN + 1)
33
37extern uint32_t rte_red_rand_val;
38extern uint32_t rte_red_rand_seed;
39extern uint16_t rte_red_log2_1_minus_Wq[RTE_RED_WQ_LOG2_NUM];
40extern uint16_t rte_red_pow2_frac_inv[16];
41
46 uint16_t min_th;
47 uint16_t max_th;
48 uint16_t maxp_inv;
49 uint16_t wq_log2;
50};
51
56 uint32_t min_th;
57 uint32_t max_th;
58 uint32_t pa_const;
59 uint8_t maxp_inv;
60 uint8_t wq_log2;
61};
62
66struct rte_red {
67 uint32_t avg;
68 uint32_t count;
69 uint64_t q_time;
70};
71
81int
83
98int
100 const uint16_t wq_log2,
101 const uint16_t min_th,
102 const uint16_t max_th,
103 const uint16_t maxp_inv);
104
115static inline uint32_t
117{
118 rte_red_rand_seed = (214013 * rte_red_rand_seed) + 2531011;
119 return rte_red_rand_seed >> 10;
120}
121
132static inline uint16_t
133__rte_red_calc_qempty_factor(uint8_t wq_log2, uint16_t m)
134{
135 uint32_t n = 0;
136 uint32_t f = 0;
137
158 n = m * rte_red_log2_1_minus_Wq[wq_log2 - RTE_RED_WQ_LOG2_MIN];
159
172 f = (n >> 6) & 0xf;
173 n >>= 10;
174
175 if (n < RTE_RED_SCALING) {
176 /* When n == 0, no rounding or shifting needed.
177 * For n > 0, add 2^(n-1) for rounding before right shift.
178 * This avoids UB from (1 << -1) when n == 0.
179 */
180 if (n == 0)
181 return (uint16_t) rte_red_pow2_frac_inv[f];
182 return (uint16_t) ((rte_red_pow2_frac_inv[f] + (1 << (n - 1))) >> n);
183 }
184
185 return 0;
186}
187
202static inline int
204 struct rte_red *red,
205 const uint64_t time)
206{
207 uint64_t time_diff = 0, m = 0;
208
209 RTE_ASSERT(red_cfg != NULL);
210 RTE_ASSERT(red != NULL);
211
212 red->count ++;
213
218 time_diff = time - red->q_time;
219
226 m = time_diff / RTE_RED_S;
227
231 if (m >= RTE_RED_2POW16) {
232 red->avg = 0;
233 } else {
234 red->avg = (red->avg >> RTE_RED_SCALING) * __rte_red_calc_qempty_factor(red_cfg->wq_log2, (uint16_t) m);
235 }
236
237 return 0;
238}
239
280static inline int
281__rte_red_drop(const struct rte_red_config *red_cfg, struct rte_red *red)
282{
283 uint32_t pa_num = 0; /* numerator of drop-probability */
284 uint32_t pa_den = 0; /* denominator of drop-probability */
285 uint32_t pa_num_count = 0;
286
287 pa_num = (red->avg - red_cfg->min_th) >> (red_cfg->wq_log2);
288
289 pa_num_count = red->count * pa_num;
290
291 if (red_cfg->pa_const <= pa_num_count)
292 return 1;
293
294 pa_den = red_cfg->pa_const - pa_num_count;
295
296 /* If drop, generate and save random number to be used next time */
297 if (unlikely((rte_red_rand_val % pa_den) < pa_num)) {
299
300 return 1;
301 }
302
303 /* No drop */
304 return 0;
305}
306
319static inline int
321 struct rte_red *red,
322 const unsigned q)
323{
324 RTE_ASSERT(red_cfg != NULL);
325 RTE_ASSERT(red != NULL);
326
340 /* avg update */
341 red->avg += (q << RTE_RED_SCALING) - (red->avg >> red_cfg->wq_log2);
342
343 /* avg < min_th: do not mark the packet */
344 if (red->avg < red_cfg->min_th) {
345 red->count ++;
346 return 0;
347 }
348
349 /* min_th <= avg < max_th: mark the packet with pa probability */
350 if (red->avg < red_cfg->max_th) {
351 if (!__rte_red_drop(red_cfg, red)) {
352 red->count ++;
353 return 0;
354 }
355
356 red->count = 0;
357 return 2;
358 }
359
360 /* max_th <= avg: always mark the packet */
361 red->count = 0;
362 return 1;
363}
364
381static inline int
382rte_red_enqueue(const struct rte_red_config *red_cfg,
383 struct rte_red *red,
384 const unsigned q,
385 const uint64_t time)
386{
387 RTE_ASSERT(red_cfg != NULL);
388 RTE_ASSERT(red != NULL);
389
390 if (q != 0) {
391 return rte_red_enqueue_nonempty(red_cfg, red, q);
392 } else {
393 return rte_red_enqueue_empty(red_cfg, red, time);
394 }
395}
396
403static inline void
404rte_red_mark_queue_empty(struct rte_red *red, const uint64_t time)
405{
406 red->q_time = time;
407}
408
409#ifdef __cplusplus
410}
411#endif
412
413#endif /* __RTE_RED_H_INCLUDED__ */
#define unlikely(x)
static uint16_t __rte_red_calc_qempty_factor(uint8_t wq_log2, uint16_t m)
calculate factor to scale average queue size when queue becomes empty
Definition: rte_red.h:133
#define RTE_RED_WQ_LOG2_MIN
Definition: rte_red.h:26
int rte_red_config_init(struct rte_red_config *red_cfg, const uint16_t wq_log2, const uint16_t min_th, const uint16_t max_th, const uint16_t maxp_inv)
Configures a single RED configuration parameter structure.
static int rte_red_enqueue_nonempty(const struct rte_red_config *red_cfg, struct rte_red *red, const unsigned q)
Decides if new packet should be enqueued or dropped in queue non-empty case.
Definition: rte_red.h:320
static int rte_red_enqueue_empty(const struct rte_red_config *red_cfg, struct rte_red *red, const uint64_t time)
Updates queue average in condition when queue is empty.
Definition: rte_red.h:203
static uint32_t rte_fast_rand(void)
Generate random number for RED.
Definition: rte_red.h:116
#define RTE_RED_2POW16
Definition: rte_red.h:30
#define RTE_RED_S
Definition: rte_red.h:24
static int rte_red_enqueue(const struct rte_red_config *red_cfg, struct rte_red *red, const unsigned q, const uint64_t time)
Decides if new packet should be enqueued or dropped Updates run time data based on new queue size val...
Definition: rte_red.h:382
#define RTE_RED_SCALING
Definition: rte_red.h:23
uint32_t rte_red_rand_val
static int __rte_red_drop(const struct rte_red_config *red_cfg, struct rte_red *red)
make a decision to drop or enqueue a packet based on mark probability criteria
Definition: rte_red.h:281
static void rte_red_mark_queue_empty(struct rte_red *red, const uint64_t time)
Callback to records time that queue became empty.
Definition: rte_red.h:404
int rte_red_rt_data_init(struct rte_red *red)
Initialises run-time data.
uint32_t pa_const
Definition: rte_red.h:58
uint8_t wq_log2
Definition: rte_red.h:60
uint8_t maxp_inv
Definition: rte_red.h:59
uint32_t min_th
Definition: rte_red.h:56
uint32_t max_th
Definition: rte_red.h:57
uint16_t min_th
Definition: rte_red.h:46
uint16_t wq_log2
Definition: rte_red.h:49
uint16_t maxp_inv
Definition: rte_red.h:48
uint16_t max_th
Definition: rte_red.h:47
uint64_t q_time
Definition: rte_red.h:69
uint32_t count
Definition: rte_red.h:68
uint32_t avg
Definition: rte_red.h:67