5

我正在尝试在 shell 命令工作时在 ubuntu 12.04 上进行网络仿真:

tc qdisc add dev eth1 root handle 1:0 tbf rate 200kbit buffer 1600 limit 3000

或者

tc qdisc change dev eth0 root netem loss 0.1%

现在我想在 c 代码中做同样的事情,我找到了 libnl。

我已经使用文档成功添加了 prio 和 htb qdisc

http://www.carisma.slowglass.com/~tgr/libnl/doc/route.html#tc_qdisc

但是当我执行 netem qdisc 时,它在 tbf qdisc 情况下返回“无效的输入数据或参数”和“缺少属性”。我的代码如下:

  1. 网络

    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));
    }
    
  2. 待定

    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));
    }
    

我已经尝试了 api ref 引用的所有相应功能,但总是有相同的错误“输入数据或参数无效”和“缺少属性”,并且在没有帮助的情况下进行了谷歌搜索。

我被这个问题困扰了一个多星期,我需要你的帮助。

4

1 回答 1

3

我查看了您的代码,但我错过了一些参数。就像您正在使用的库版本和 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 -fpermissive --pedantic-errors -Wall -Wextra -ftabstop=4 -march=native -fshow-column -ftabstop=4 -frounding-math -管道 -ggdb3 -O0 main.c -o your_qdisc_app"

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

我用“if”省略了所有检查以最小化线条。如有必要,可以添加检查。如果您还有其他问题,请与我联系。

赛亚,菲利普

于 2014-04-15T09:49:49.817 回答