token_bucket.h 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. /* Copyright (c) 2018-2019, The Tor Project, Inc. */
  2. /* See LICENSE for licensing information */
  3. /**
  4. * \file token_bucket.h
  5. * \brief Headers for token_bucket.c
  6. **/
  7. #ifndef TOR_TOKEN_BUCKET_H
  8. #define TOR_TOKEN_BUCKET_H
  9. #include "lib/cc/torint.h"
  10. #include "lib/testsupport/testsupport.h"
  11. /** Largest allowable burst value for a token buffer. */
  12. #define TOKEN_BUCKET_MAX_BURST INT32_MAX
  13. /** A generic token buffer configuration: determines the number of tokens
  14. * added to the bucket in each time unit (the "rate"), and the maximum number
  15. * of tokens in the bucket (the "burst") */
  16. typedef struct token_bucket_cfg_t {
  17. uint32_t rate;
  18. int32_t burst;
  19. } token_bucket_cfg_t;
  20. /** A raw token bucket, decoupled from its configuration and timestamp. */
  21. typedef struct token_bucket_raw_t {
  22. int32_t bucket;
  23. } token_bucket_raw_t;
  24. void token_bucket_cfg_init(token_bucket_cfg_t *cfg,
  25. uint32_t rate,
  26. uint32_t burst);
  27. void token_bucket_raw_adjust(token_bucket_raw_t *bucket,
  28. const token_bucket_cfg_t *cfg);
  29. void token_bucket_raw_reset(token_bucket_raw_t *bucket,
  30. const token_bucket_cfg_t *cfg);
  31. int token_bucket_raw_dec(token_bucket_raw_t *bucket,
  32. ssize_t n);
  33. int token_bucket_raw_refill_steps(token_bucket_raw_t *bucket,
  34. const token_bucket_cfg_t *cfg,
  35. const uint32_t elapsed_steps);
  36. static inline size_t token_bucket_raw_get(const token_bucket_raw_t *bucket);
  37. /** Return the current number of bytes set in a token bucket. */
  38. static inline size_t
  39. token_bucket_raw_get(const token_bucket_raw_t *bucket)
  40. {
  41. return bucket->bucket >= 0 ? bucket->bucket : 0;
  42. }
  43. /** A convenience type containing all the pieces needed for a coupled
  44. * read-bucket and write-bucket that have the same rate limit, and which use
  45. * "timestamp units" (see compat_time.h) for their time. */
  46. typedef struct token_bucket_rw_t {
  47. token_bucket_cfg_t cfg;
  48. token_bucket_raw_t read_bucket;
  49. token_bucket_raw_t write_bucket;
  50. uint32_t last_refilled_at_timestamp;
  51. } token_bucket_rw_t;
  52. void token_bucket_rw_init(token_bucket_rw_t *bucket,
  53. uint32_t rate,
  54. uint32_t burst,
  55. uint32_t now_ts);
  56. void token_bucket_rw_adjust(token_bucket_rw_t *bucket,
  57. uint32_t rate, uint32_t burst);
  58. void token_bucket_rw_reset(token_bucket_rw_t *bucket,
  59. uint32_t now_ts);
  60. #define TB_READ 1
  61. #define TB_WRITE 2
  62. int token_bucket_rw_refill(token_bucket_rw_t *bucket,
  63. uint32_t now_ts);
  64. int token_bucket_rw_dec_read(token_bucket_rw_t *bucket,
  65. ssize_t n);
  66. int token_bucket_rw_dec_write(token_bucket_rw_t *bucket,
  67. ssize_t n);
  68. int token_bucket_rw_dec(token_bucket_rw_t *bucket,
  69. ssize_t n_read, ssize_t n_written);
  70. static inline size_t token_bucket_rw_get_read(const token_bucket_rw_t *bucket);
  71. static inline size_t
  72. token_bucket_rw_get_read(const token_bucket_rw_t *bucket)
  73. {
  74. return token_bucket_raw_get(&bucket->read_bucket);
  75. }
  76. static inline size_t token_bucket_rw_get_write(
  77. const token_bucket_rw_t *bucket);
  78. static inline size_t
  79. token_bucket_rw_get_write(const token_bucket_rw_t *bucket)
  80. {
  81. return token_bucket_raw_get(&bucket->write_bucket);
  82. }
  83. /**
  84. * A specialized bucket containing a single counter.
  85. */
  86. typedef struct token_bucket_ctr_t {
  87. token_bucket_cfg_t cfg;
  88. token_bucket_raw_t counter;
  89. uint32_t last_refilled_at_timestamp;
  90. } token_bucket_ctr_t;
  91. void token_bucket_ctr_init(token_bucket_ctr_t *bucket, uint32_t rate,
  92. uint32_t burst, uint32_t now_ts);
  93. void token_bucket_ctr_adjust(token_bucket_ctr_t *bucket, uint32_t rate,
  94. uint32_t burst);
  95. void token_bucket_ctr_reset(token_bucket_ctr_t *bucket, uint32_t now_ts);
  96. void token_bucket_ctr_refill(token_bucket_ctr_t *bucket, uint32_t now_ts);
  97. static inline bool
  98. token_bucket_ctr_dec(token_bucket_ctr_t *bucket, ssize_t n)
  99. {
  100. return token_bucket_raw_dec(&bucket->counter, n);
  101. }
  102. static inline size_t
  103. token_bucket_ctr_get(const token_bucket_ctr_t *bucket)
  104. {
  105. return token_bucket_raw_get(&bucket->counter);
  106. }
  107. #ifdef TOKEN_BUCKET_PRIVATE
  108. /* To avoid making the rates too small, we consider units of "steps",
  109. * where a "step" is defined as this many timestamp ticks. Keep this
  110. * a power of two if you can. */
  111. #define TICKS_PER_STEP 16
  112. STATIC uint32_t rate_per_sec_to_rate_per_step(uint32_t rate);
  113. #endif /* defined(TOKEN_BUCKET_PRIVATE) */
  114. #endif /* !defined(TOR_TOKEN_BUCKET_H) */