У меня есть FQ Qdisc , который я хочу обновить порядка миллисекунд. Мой текущий подход заключается в прямом использовании команды tc, но это слишком медленно:
char tc_cmd[200];
snprintf(tc_cmd, 200,"tc qdisc change dev %s root fq maxrate %.3fmbit &", net_dev, tx_rate / 10e5);
err = system(tc_cmd);
if (err) {
perror(stderr,"Problem with tc qdisc change");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
Я хотел бы реализовать этот фрагмент кода, используя libnl или netlink. К сожалению, хотя libnl-route поддерживает разновидность qdiscs, она не поддерживает честную очередь. Единственное, что я действительно хочу сделать, это отправить сообщение об обновлении, которое изменяет ограничение на активный FQ qdisc. Тем не менее, я не знаю много о netlink и реализации libnl, и я не уверен, как я мог бы создать такое сообщение. Как я могу отправить сообщение модулю ядра, сообщив ему об обновлении максимальной скорости справедливой очереди qdisc?
Для справки, вот как я настраиваю и изменяю htb qdisc с помощью libnl. Я хотел бы добиться чего-то похожего для fq qdisc:
struct rtnl_qdisc *tbf_qdisc;
int if_index;
int err = 0;
char *netdev = "eth0"
uint64_t tx_rate = 10e6;
uint64_t burst = 12000;
uint64_t limit = 5000;
fprintf(stderr, "Kernel Clock Rate: %d\n", nl_get_user_hz());
qdisc_sock = nl_socket_alloc();
nl_connect(qdisc_sock, NETLINK_ROUTE);
if_index = if_nametoindex(netdev);
// Allocate and configure the qdisc
fq_qdisc = rtnl_qdisc_alloc();
rtnl_tc_set_ifindex(TC_CAST(tbf_qdisc), if_index);
rtnl_tc_set_parent(TC_CAST(tbf_qdisc), TC_H_ROOT);
rtnl_tc_set_handle(TC_CAST(tbf_qdisc), TC_HANDLE(1,0));
err = rtnl_tc_set_kind(TC_CAST(tbf_qdisc), "tbf");
if (err) {
fprintf(stderr,"Can not allocate TBF: %s\n", nl_geterror(err));
return NULL;
}
rtnl_qdisc_tbf_set_rate(tbf_qdisc, (uint64_t) tx_rate, burst , 0);
rtnl_qdisc_tbf_set_limit(tbf_qdisc, limit);
rtnl_qdisc_tbf_set_peakrate(tbf_qdisc, (uint64_t) tx_rate, burst, 0);
// Add it to the interface
err = rtnl_qdisc_add(qdisc_sock, tbf_qdisc, NLM_F_CREATE);
if (err) {
fprintf(stderr,"Can not set TBF: %s\n", nl_geterror(err));
return NULL;
// Update the rate of the qdisc
rate = 1e6;
rtnl_qdisc_tbf_set_rate(tbf_qdisc, (uint64_t) tx_rate, burst , 0);
rtnl_qdisc_tbf_set_peakrate(tbf_qdisc, (uint64_t) tx_rate, burst, 0);
rtnl_qdisc_tbf_set_limit(tbf_qdisc, limit);
err = rtnl_qdisc_add(qdisc_sock, tbf_qdisc, NLM_F_REPLACE);
if(err) {
fprintf(stderr,"Rate %lu qdisc_add: %s\n", tx_rate, nl_geterror(err));
return NULL;
}