如何使用libnl路由设置netem qdisc和tbf qdisc?

如何使用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

我正在尝试在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_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
    要编译代码,请使用此命令。您可以根据需要进行调整,但“-lnl-genl-3-lnl-3-lnl-route-3”是必需的

    “gcc-lnl-genl-3-lnl-3-lnl-route-3-F允许——学究式错误——墙——Wextra——ftabstop=4——三月——本地——fshow列——ftabstop=4——起泡数学——管道——ggdb3——O0 main.c——你的qdisc应用程序”

    #包括
    #包括
    #包括
    #包括
    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;
    }