如何使用libnl路由设置netem qdisc和tbf qdisc?
我正在尝试在ubuntu 12.04上进行网络仿真,而shell命令可以工作:如何使用libnl路由设置netem qdisc和tbf qdisc?,c,linux,ubuntu,netlink,C,Linux,Ubuntu,Netlink,我正在尝试在ubuntu 12.04上进行网络仿真,而shell命令可以工作: tc qdisc add dev eth1 root handle 1:0 tbf rate 200kbit buffer 1600 limit 3000 或 现在我想在c代码中做同样的事情,我发现了libnl 我已使用文档成功添加了prio和htb qdisc 但是当我执行netemqdisc时,它返回“无效的输入数据或参数”,在tbfqdisc的情况下返回“缺少属性”。我的代码如下: 内特姆 q = rtnl
tc qdisc add dev eth1 root handle 1:0 tbf rate 200kbit buffer 1600 limit 3000
或
现在我想在c代码中做同样的事情,我发现了libnl
我已使用文档成功添加了prio和htb qdisc
但是当我执行netemqdisc时,它返回“无效的输入数据或参数”,在tbfqdisc的情况下返回“缺少属性”。我的代码如下:
q = rtnl_qdisc_alloc();
rtnl_tc_set_ifindex(TC_CAST(q), 2);
rtnl_tc_set_parent(TC_CAST(q), TC_H_ROOT);
rtnl_tc_set_handle(TC_CAST(q), TC_HANDLE(1, 0));
rtnl_tc_set_kind(TC_CAST(q), "netem");
rtnl_netem_set_delay(q, 100);
rtnl_netem_set_loss(q, 10);
int err = rtnl_qdisc_add(sock, q, NLM_F_CREATE);
if(err<0){
printf("netem error: %s\n", nl_geterror(err));
}
q=rtnl_qdisc_alloc();
rtnl_tc_set_iIndex(tc_CAST(q),2);
rtnl_tc_set_parent(tc_CAST(q),tc_H_ROOT);
rtnl_tc_set_句柄(tc_CAST(q),tc_句柄(1,0));
rtnl_tc_set_kind(tc_CAST(q),“netem”);
rtnl_netem_set_延迟(q,100);
rtnl_netem_set_损耗(q,10);
int err=rtnl_qdisc_add(sock、q、NLM_F_CREATE);
if(err我查看了您的代码,但遗漏了一些参数。例如您正在使用的库的版本和gcc标志。因此,我决定给您一个示例代码,您可以比较与您的系统/代码的区别 我系统上的所有库都有版本3.2.24-1(debian不稳定); 您需要的图书馆:
- libnl-3-dev
- libnl-cli-3-dev
- libnl-genl-3-dev
- libnl-nf-3-dev
- libnl-route-3-dev
- libnl-3-200
- libnl-cli-3-200
- libnl-genl-3-200
- libnl-nf-3-200
- libnl-route-3-200
- libnl-utils
#包括
#包括
#包括
#包括
int main(int argc,字符**argv)
{
结构nl_sock*sock;
结构rtnl_qdisc*q;
结构nl_缓存*缓存;
结构rtnl_链接*链接;
int if_索引;
sock=nl_socket_alloc();
nl_连接(sock、NETLINK_路由);
rtnl_链路_分配_缓存(sock、AF_unsec和cache);
link=rtnl_link_get_by_name(缓存,“eth0”);
如果索引=rtnl链接获取ifindex(链接);
q=rtnl_qdisc_alloc();
rtnl_tc_集合ifindex(tc_CAST(q),if_索引);
rtnl_tc_set_parent(tc_CAST(q),tc_H_ROOT);
rtnl_tc_set_句柄(tc_CAST(q),tc_句柄(1,0));
rtnl_tc_set_kind(tc_CAST(q),“tbf”);
/*
*netem好的,htb好的,请评论
*和取消注释qdiscs的特殊参数
*/
rtnl_qdisc_tbf_set_limit(q,1000);
rtnl_qdisc_tbf_设置速率(q,1000,1000,8);
/*
*rtnl_netem_set_延迟(q,100);
*rtnl_netem_set_损耗(q,10);
*/
rtnl_qdisc_add(sock、q、NLM_F_CREATE);
rtnl_qdisc_put(q);
nl_无插座(sock);
rtnl_链路_put(链路);
nl_cache_put(缓存);
返回0;
}
我省略了所有带有“如果”的检查,以尽量减少行数。如果需要,可以添加检查。如果您有进一步的问题,请与我联系
西娅,
菲利普
q = rtnl_qdisc_alloc();
rtnl_tc_set_ifindex(TC_CAST(q), 2);
rtnl_tc_set_parent(TC_CAST(q), TC_H_ROOT);
rtnl_tc_set_handle(TC_CAST(q), TC_HANDLE(1, 0));
rtnl_tc_set_kind(TC_CAST(q), "netem");
rtnl_netem_set_delay(q, 100);
rtnl_netem_set_loss(q, 10);
int err = rtnl_qdisc_add(sock, q, NLM_F_CREATE);
if(err<0){
printf("netem error: %s\n", nl_geterror(err));
}
q = rtnl_qdisc_alloc();
rtnl_tc_set_ifindex(TC_CAST(q), 2);
rtnl_tc_set_parent(TC_CAST(q), TC_H_ROOT);
rtnl_tc_set_handle(TC_CAST(q), TC_HANDLE(1, 0));
rtnl_tc_set_kind(TC_CAST(q), "tbf");
rtnl_qdisc_tbf_set_limit(q, 1000);
rtnl_qdisc_tbf_set_rate(q, 1000, 1000, 8);
int err = rtnl_qdisc_add(sock, q, NLM_F_CREATE);
if(err<0){
printf("tbf error: %s\n", nl_geterror(err));
}
#include <libnl3/netlink/route/tc.h>
#include <libnl3/netlink/route/qdisc.h>
#include <libnl3/netlink/route/qdisc/netem.h>
#include <libnl3/netlink/route/qdisc/tbf.h>
int main(int argc, char ** argv)
{
struct nl_sock *sock;
struct rtnl_qdisc *q;
struct nl_cache *cache;
struct rtnl_link *link;
int if_index;
sock = nl_socket_alloc();
nl_connect(sock, NETLINK_ROUTE);
rtnl_link_alloc_cache(sock, AF_UNSPEC, &cache);
link = rtnl_link_get_by_name(cache, "eth0");
if_index = rtnl_link_get_ifindex(link);
q = rtnl_qdisc_alloc();
rtnl_tc_set_ifindex(TC_CAST(q), if_index);
rtnl_tc_set_parent(TC_CAST(q), TC_H_ROOT);
rtnl_tc_set_handle(TC_CAST(q), TC_HANDLE(1, 0));
rtnl_tc_set_kind(TC_CAST(q), "tbf");
/*
* netem okay, htb okay, please comment
* and uncomment the special parameters for the qdiscs
*/
rtnl_qdisc_tbf_set_limit(q, 1000);
rtnl_qdisc_tbf_set_rate(q, 1000, 1000, 8);
/*
* rtnl_netem_set_delay(q, 100);
* rtnl_netem_set_loss(q, 10);
*/
rtnl_qdisc_add(sock, q, NLM_F_CREATE);
rtnl_qdisc_put(q);
nl_socket_free(sock);
rtnl_link_put(link);
nl_cache_put(cache);
return 0;
}