You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

253 lines
5.0 KiB

/*
* lib/cli/qdisc/hfsc.c HFSC module for CLI lib
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation version 2.1
* of the License.
*
* Copyright (c) 2014 Cong Wang <xiyou.wangcong@gmail.com>
*/
#include <netlink/cli/utils.h>
#include <netlink/cli/tc.h>
#include <netlink/route/qdisc/hfsc.h>
#include <linux/pkt_sched.h>
static void print_qdisc_usage(void)
{
printf(
"Usage: nl-qdisc-add [...] hfsc [OPTIONS]...\n"
"\n"
"OPTIONS\n"
" --help Show this help text.\n"
" --default=ID Default class for unclassified traffic.\n"
"\n"
"EXAMPLE"
" # Create hfsc root qdisc 1: and direct unclassified traffic to class 1:10\n"
" nl-qdisc-add --dev=eth1 --parent=root --handle=1: hfsc --default=10\n");
}
static void hfsc_parse_qdisc_argv(struct rtnl_tc *tc, int argc, char **argv)
{
struct rtnl_qdisc *qdisc = (struct rtnl_qdisc *) tc;
for (;;) {
int c, optidx = 0;
enum {
ARG_DEFAULT = 257,
};
static struct option long_opts[] = {
{ "help", 0, 0, 'h' },
{ "default", 1, 0, ARG_DEFAULT },
{ 0, 0, 0, 0 }
};
c = getopt_long(argc, argv, "hv", long_opts, &optidx);
if (c == -1)
break;
switch (c) {
case 'h':
print_qdisc_usage();
return;
case ARG_DEFAULT:
rtnl_qdisc_hfsc_set_defcls(qdisc, nl_cli_parse_u32(optarg));
break;
}
}
}
static void print_class_usage(void)
{
printf(
"Usage: nl-class-add [...] hfsc [OPTIONS]...\n"
"\n"
"OPTIONS\n"
" --help Show this help text.\n"
" --ls=SC Link-sharing service curve\n"
" --rt=SC Real-time service curve\n"
" --sc=SC Specifiy both of the above\n"
" --ul=SC Upper limit\n"
" where SC := [ [ m1 bits ] d usec ] m2 bits\n"
"\n"
"EXAMPLE"
" # Attach class 1:1 to hfsc qdisc 1: and use rt and ls curve\n"
" nl-class-add --dev=eth1 --parent=1: --classid=1:1 hfsc --sc=m1:250,d:8,m2:100\n");
}
static int
hfsc_get_sc(char *optarg, struct tc_service_curve *sc)
{
unsigned int m1 = 0, d = 0, m2 = 0;
char *tmp = strdup(optarg);
char *p, *endptr;
char *pp = tmp;
if (!tmp)
return -ENOMEM;
p = strstr(pp, "m1:");
if (p) {
char *q;
p += 3;
if (*p == 0)
goto err;
q = strchr(p, ',');
if (!q)
goto err;
*q = 0;
m1 = strtoul(p, &endptr, 10);
if (endptr == p)
goto err;
pp = q + 1;
}
p = strstr(pp, "d:");
if (p) {
char *q;
p += 2;
if (*p == 0)
goto err;
q = strchr(p, ',');
if (!q)
goto err;
*q = 0;
d = strtoul(p, &endptr, 10);
if (endptr == p)
goto err;
pp = q + 1;
}
p = strstr(pp, "m2:");
if (p) {
p += 3;
if (*p == 0)
goto err;
m2 = strtoul(p, &endptr, 10);
if (endptr == p)
goto err;
} else
goto err;
free(tmp);
sc->m1 = m1;
sc->d = d;
sc->m2 = m2;
return 0;
err:
free(tmp);
return -EINVAL;
}
static void hfsc_parse_class_argv(struct rtnl_tc *tc, int argc, char **argv)
{
struct rtnl_class *class = (struct rtnl_class *) tc;
int arg_ok = 0, ret = -EINVAL;
for (;;) {
int c, optidx = 0;
enum {
ARG_RT = 257,
ARG_LS = 258,
ARG_SC,
ARG_UL,
};
static struct option long_opts[] = {
{ "help", 0, 0, 'h' },
{ "rt", 1, 0, ARG_RT },
{ "ls", 1, 0, ARG_LS },
{ "sc", 1, 0, ARG_SC },
{ "ul", 1, 0, ARG_UL },
{ 0, 0, 0, 0 }
};
struct tc_service_curve tsc;
c = getopt_long(argc, argv, "h", long_opts, &optidx);
if (c == -1)
break;
switch (c) {
case 'h':
print_class_usage();
return;
case ARG_RT:
ret = hfsc_get_sc(optarg, &tsc);
if (ret < 0) {
nl_cli_fatal(ret, "Unable to parse sc "
"\"%s\": Invalid format.", optarg);
}
rtnl_class_hfsc_set_rsc(class, &tsc);
arg_ok++;
break;
case ARG_LS:
ret = hfsc_get_sc(optarg, &tsc);
if (ret < 0) {
nl_cli_fatal(ret, "Unable to parse sc "
"\"%s\": Invalid format.", optarg);
}
rtnl_class_hfsc_set_fsc(class, &tsc);
arg_ok++;
break;
case ARG_SC:
ret = hfsc_get_sc(optarg, &tsc);
if (ret < 0) {
nl_cli_fatal(ret, "Unable to parse sc "
"\"%s\": Invalid format.", optarg);
}
rtnl_class_hfsc_set_rsc(class, &tsc);
rtnl_class_hfsc_set_fsc(class, &tsc);
arg_ok++;
break;
case ARG_UL:
ret = hfsc_get_sc(optarg, &tsc);
if (ret < 0) {
nl_cli_fatal(ret, "Unable to parse sc "
"\"%s\": Invalid format.", optarg);
}
rtnl_class_hfsc_set_usc(class, &tsc);
arg_ok++;
break;
}
}
if (!arg_ok)
nl_cli_fatal(ret, "Invalid arguments");
}
static struct nl_cli_tc_module hfsc_qdisc_module =
{
.tm_name = "hfsc",
.tm_type = RTNL_TC_TYPE_QDISC,
.tm_parse_argv = hfsc_parse_qdisc_argv,
};
static struct nl_cli_tc_module hfsc_class_module =
{
.tm_name = "hfsc",
.tm_type = RTNL_TC_TYPE_CLASS,
.tm_parse_argv = hfsc_parse_class_argv,
};
static void __init hfsc_init(void)
{
nl_cli_tc_register(&hfsc_qdisc_module);
nl_cli_tc_register(&hfsc_class_module);
}
static void __exit hfsc_exit(void)
{
nl_cli_tc_unregister(&hfsc_class_module);
nl_cli_tc_unregister(&hfsc_qdisc_module);
}