123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- #include <stdio.h>
- #include <string.h>
- #include <stdlib.h>
- #include <math.h>
- #include <xtables.h>
- #include <linux/netfilter/x_tables.h>
- #include <linux/netfilter/xt_RATEEST.h>
- struct rateest_tg_udata {
- unsigned int interval;
- unsigned int ewma_log;
- };
- static void
- RATEEST_help(void)
- {
- printf(
- "RATEEST target options:\n"
- " --rateest-name name Rate estimator name\n"
- " --rateest-interval sec Rate measurement interval in seconds\n"
- " --rateest-ewmalog value Rate measurement averaging time constant\n");
- }
- enum {
- O_NAME = 0,
- O_INTERVAL,
- O_EWMALOG,
- };
- #define s struct xt_rateest_target_info
- static const struct xt_option_entry RATEEST_opts[] = {
- {.name = "rateest-name", .id = O_NAME, .type = XTTYPE_STRING,
- .flags = XTOPT_MAND | XTOPT_PUT, XTOPT_POINTER(s, name)},
- {.name = "rateest-interval", .id = O_INTERVAL, .type = XTTYPE_STRING,
- .flags = XTOPT_MAND},
- {.name = "rateest-ewmalog", .id = O_EWMALOG, .type = XTTYPE_STRING,
- .flags = XTOPT_MAND},
- XTOPT_TABLEEND,
- };
- #undef s
- /* Copied from iproute */
- #define TIME_UNITS_PER_SEC 1000000
- static int
- RATEEST_get_time(unsigned int *time, const char *str)
- {
- double t;
- char *p;
- t = strtod(str, &p);
- if (p == str)
- return -1;
- if (*p) {
- if (strcasecmp(p, "s") == 0 || strcasecmp(p, "sec")==0 ||
- strcasecmp(p, "secs")==0)
- t *= TIME_UNITS_PER_SEC;
- else if (strcasecmp(p, "ms") == 0 || strcasecmp(p, "msec")==0 ||
- strcasecmp(p, "msecs") == 0)
- t *= TIME_UNITS_PER_SEC/1000;
- else if (strcasecmp(p, "us") == 0 || strcasecmp(p, "usec")==0 ||
- strcasecmp(p, "usecs") == 0)
- t *= TIME_UNITS_PER_SEC/1000000;
- else
- return -1;
- }
- *time = t;
- return 0;
- }
- static void
- RATEEST_print_time(unsigned int time)
- {
- double tmp = time;
- if (tmp >= TIME_UNITS_PER_SEC)
- printf(" %.1fs", tmp / TIME_UNITS_PER_SEC);
- else if (tmp >= TIME_UNITS_PER_SEC/1000)
- printf(" %.1fms", tmp / (TIME_UNITS_PER_SEC / 1000));
- else
- printf(" %uus", time);
- }
- static void RATEEST_parse(struct xt_option_call *cb)
- {
- struct rateest_tg_udata *udata = cb->udata;
- xtables_option_parse(cb);
- switch (cb->entry->id) {
- case O_INTERVAL:
- if (RATEEST_get_time(&udata->interval, cb->arg) < 0)
- xtables_error(PARAMETER_PROBLEM,
- "RATEEST: bad interval value \"%s\"",
- cb->arg);
- break;
- case O_EWMALOG:
- if (RATEEST_get_time(&udata->ewma_log, cb->arg) < 0)
- xtables_error(PARAMETER_PROBLEM,
- "RATEEST: bad ewmalog value \"%s\"",
- cb->arg);
- break;
- }
- }
- static void RATEEST_final_check(struct xt_fcheck_call *cb)
- {
- struct xt_rateest_target_info *info = cb->data;
- struct rateest_tg_udata *udata = cb->udata;
- for (info->interval = 0; info->interval <= 5; info->interval++) {
- if (udata->interval <= (1 << info->interval) * (TIME_UNITS_PER_SEC / 4))
- break;
- }
- if (info->interval > 5)
- xtables_error(PARAMETER_PROBLEM,
- "RATEEST: interval value is too large");
- info->interval -= 2;
- for (info->ewma_log = 1; info->ewma_log < 32; info->ewma_log++) {
- double w = 1.0 - 1.0 / (1 << info->ewma_log);
- if (udata->interval / (-log(w)) > udata->ewma_log)
- break;
- }
- info->ewma_log--;
- if (info->ewma_log == 0 || info->ewma_log >= 31)
- xtables_error(PARAMETER_PROBLEM,
- "RATEEST: ewmalog value is out of range");
- }
- static void
- __RATEEST_print(const struct xt_entry_target *target, const char *prefix)
- {
- const struct xt_rateest_target_info *info = (const void *)target->data;
- unsigned int local_interval;
- unsigned int local_ewma_log;
- local_interval = (TIME_UNITS_PER_SEC << (info->interval + 2)) / 4;
- local_ewma_log = local_interval * (1 << (info->ewma_log));
- printf(" %sname %s", prefix, info->name);
- printf(" %sinterval", prefix);
- RATEEST_print_time(local_interval);
- printf(" %sewmalog", prefix);
- RATEEST_print_time(local_ewma_log);
- }
- static void
- RATEEST_print(const void *ip, const struct xt_entry_target *target,
- int numeric)
- {
- __RATEEST_print(target, "");
- }
- static void
- RATEEST_save(const void *ip, const struct xt_entry_target *target)
- {
- __RATEEST_print(target, "--rateest-");
- }
- static struct xtables_target rateest_tg_reg = {
- .family = NFPROTO_UNSPEC,
- .name = "RATEEST",
- .version = XTABLES_VERSION,
- .size = XT_ALIGN(sizeof(struct xt_rateest_target_info)),
- .userspacesize = offsetof(struct xt_rateest_target_info, est),
- .help = RATEEST_help,
- .x6_parse = RATEEST_parse,
- .x6_fcheck = RATEEST_final_check,
- .print = RATEEST_print,
- .save = RATEEST_save,
- .x6_options = RATEEST_opts,
- .udata_size = sizeof(struct rateest_tg_udata),
- };
- void _init(void)
- {
- xtables_register_target(&rateest_tg_reg);
- }
|