免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2296 | 回复: 1
打印 上一主题 下一主题

[内核模块] netlink套接字实现类似ss命令,统计套接字以及TCP信息 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2014-05-31 15:35 |只看该作者 |倒序浏览
本帖最后由 SCDXMOE 于 2014-05-31 15:41 编辑

参考了 ss的源代码
以及 netlink相关资料:http://blog.csdn.net/scdxmoe/article/details/27711205

实现结果为:
gcc netlink_dig_530_7.c -o netlink_dig_530_7
./netlink_dig_530_7

state      family     l.addr     l.port       r.addr     r.rport   
LISTEN     AF_INET   localhost      53         0.0.0.0        0         
LISTEN     AF_INET   (null)         21         0.0.0.0        0         
LISTEN     AF_INET   (null)         22         0.0.0.0        0         
LISTEN     AF_INET   (null)         22         0.0.0.0        0         
LISTEN     AF_INET   localhost      631        0.0.0.0        0         
LISTEN     AF_INET   (null)         12865      0.0.0.0        0         
ESTAB      AF_INET   ubuntu.local   59208      91.189.89.134  80        
ESTAB      AF_INET   ubuntu.local   22         192.168.0.248  9689      
ESTAB      AF_INET   ubuntu.local   22         192.168.0.248  9295      
ESTAB      AF_INET   ubuntu.local   35531      91.189.94.25   80        
ESTAB      AF_INET   ubuntu.local   22         192.168.0.248  9691  

本文的实验 并没有实现怎么样获取TCP的窗口值

cwnd和RTT值,在ss源码中我看到了他利用了/proc
文件来实现获取窗口和RTT值,怎么样用netlink套接


字实现呢?还请教各位大神

谢谢!
源代码如下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <string.h>
  5. #include <fcntl.h>
  6. #include <errno.h>
  7. #include <string.h>
  8. #include <asm/types.h>
  9. #include <sys/socket.h>
  10. #include <linux/netlink.h>
  11. #include <linux/inet_diag.h>
  12. #include <netinet/tcp.h>
  13. #include <netdb.h>
  14. #include<arpa/inet.h>
  15. struct sk_req {
  16.     struct nlmsghdr nlh;
  17.     struct inet_diag_req r;
  18. };


  19. typedef struct{
  20. __u8 family;
  21. __u8 bytelen;
  22. __s16 bitlen;
  23. __u32 flags;
  24. __u32 data[8];
  25. } inet_prefix;

  26. /*struct namerec
  27. {
  28.         struct namerec *next;
  29.         const char *name;
  30.         inet_prefix addr;
  31. };*/

  32. struct tcpstat
  33. {
  34.         inet_prefix     local;
  35.         inet_prefix     remote;
  36.         int             lport;
  37.         int             rport;
  38.         int             state;
  39.         int             rq, wq;
  40.         int             timer;
  41.                  int             rq, wq;
  42.         int             timer;
  43.         int             timeout;
  44.         int             retrs;
  45.         unsigned        ino;
  46.         int             probes;
  47.         unsigned        uid;
  48.         int             refcnt;
  49.         unsigned long long sk;
  50.         int             rto, ato, qack, cwnd, ssthresh;
  51. };
  52. enum {
  53.         SS_UNKNOWN,
  54.         SS_ESTABLISHED,
  55.         SS_SYN_SENT,
  56.         SS_SYN_RECV,
  57.         SS_FIN_WAIT1,
  58.         SS_FIN_WAIT2,
  59.         SS_TIME_WAIT,
  60.         SS_CLOSE,
  61.         SS_CLOSE_WAIT,
  62.         SS_LAST_ACK,
  63.         SS_LISTEN,
  64.         SS_CLOSING,
  65.         SS_MAX
  66. };

  67. static const char *sstate_name[] = {
  68.         "UNKNOWN",
  69.         [SS_ESTABLISHED] = "ESTAB",
  70.         [SS_SYN_SENT] = "SYN-SENT",
  71.         [SS_SYN_RECV] = "SYN-RECV",
  72.         [SS_FIN_WAIT1] = "FIN-WAIT-1",
  73.         [SS_FIN_WAIT2] = "FIN-WAIT-2",
  74.         [SS_TIME_WAIT] = "TIME-WAIT",
  75.         [SS_CLOSE] = "UNCONN",
  76.         [SS_CLOSE_WAIT] = "CLOSE-WAIT",
  77.         [SS_LAST_ACK] = "LAST-ACK",
  78.         [SS_LISTEN] =   "LISTEN",
  79.         [SS_CLOSING] = "CLOSING",
  80. };

  81. /*  Base info structure. It contains socket identity (addrs/ports/cookie)
  82. * and, alas, the information shown by netstat.
  83. /*  Base info structure. It contains socket identity (addrs/ports/cookie)
  84. * and, alas, the information shown by netstat.

  85. struct nlmsghdr {
  86.         __u32           nlmsg_len;      // Length of message including header
  87.         __u16           nlmsg_type;     // Message content
  88.         __u16           nlmsg_flags;    //Additional flags
  89.         __u32           nlmsg_seq;      // Sequence number
  90.         __u32           nlmsg_pid;      // Sending process port ID
  91. };*/

  92. //#ifdef RESOLVE_HOSTNAMES
  93. struct namerec
  94. {
  95.         struct namerec *next;
  96.         const char *name;
  97.         inet_prefix addr;
  98. };

  99. #define NHASH 257
  100. static struct namerec *nht[NHASH];

  101. static const char *resolve_address(const void *addr, int len, int af)
  102. {
  103.         struct namerec *n;
  104.         struct hostent *h_ent;
  105.         unsigned hash;
  106.         static int notfirst;


  107.         if (af == AF_INET6 && ((__u32*)addr)[0] == 0 &&
  108.             ((__u32*)addr)[1] == 0 && ((__u32*)addr)[2] == htonl(0xffff)) {
  109.                 af = AF_INET;
  110.                 addr += 12;
  111.                 len = 4;
  112.         }

  113.         hash = *(__u32 *)(addr + len - 4) % NHASH;

  114.         for (n = nht[hash]; n; n = n->next) {
  115.                 if (n->addr.family == af &&
  116.                     n->addr.bytelen == len &&
  117.                     memcmp(n->addr.data, addr, len) == 0)
  118.                         return n->name;
  119.   memcmp(n->addr.data, addr, len) == 0)
  120.                         return n->name;
  121.         }
  122.         if ((n = malloc(sizeof(*n))) == NULL)
  123.                 return NULL;
  124.         n->addr.family = af;
  125.         n->addr.bytelen = len;
  126.         n->name = NULL;
  127.         memcpy(n->addr.data, addr, len);
  128.         n->next = nht[hash];
  129.         nht[hash] = n;
  130.         if (++notfirst == 1)
  131.                 sethostent(1);
  132.         fflush(stdout);

  133.         if ((h_ent = gethostbyaddr(addr, len, af)) != NULL)
  134.                 n->name = strdup(h_ent->h_name);

  135.         /* Even if we fail, "negative" entry is remembered. */
  136.         return n->name;
  137. }
  138. //#endif


  139. const char *rt_addr_n2a(int af, int len, const void *addr, char *buf, int buflen)
  140. {
  141.         switch (af) {
  142.         case AF_INET:
  143.         case AF_INET6:
  144.                 return inet_ntop(af, addr, buf, buflen);
  145.         /*case AF_IPX:
  146.                 return ipx_ntop(af, addr, buf, buflen);
  147.         case AF_DECnet:
  148.         {
  149.                 struct dn_naddr dna = { 2, { 0, 0, }};
  150.                 memcpy(dna.a_addr, addr, 2);
  151.                 return dnet_ntop(af, &dna, buf, buflen);
  152.         }*/
  153.         default:
  154.                 return "???";
  155.         }
  156. }



  157. void print_info(struct inet_diag_msg *pkg,struct nlmsghdr *h)
  158. {
  159.                 struct tcpstat s;
  160.                 char buf[1024];
  161.                 const char *ap = buf;//存放ip地址
  162.                 char buf2[1024];
  163.                 const char *ap2 = buf2;//存放ip地址

  164.                 struct inet_diag_msg *r = NLMSG_DATA(h);
  165.                 s.state = r->idiag_state;
  166.                 s.local.family = s.remote.family = r->idiag_family;
  167.                 s.lport = ntohs(r->id.idiag_sport);
  168.                 s.rport = ntohs(r->id.idiag_dport);
  169.                 s.local.family = s.remote.family = r->idiag_family;
  170.                 if (s.local.family == AF_INET) {
  171.                 //这里 打印
  172.                 s.local.bytelen = s.remote.bytelen = 4;
  173.                 } else {
  174.                 s.local.bytelen = s.remote.bytelen = 16;
  175.                 }
  176.                 memcpy(s.local.data, r->id.idiag_src, s.local.bytelen);
  177.                 memcpy(s.remote.data, r->id.idiag_dst, s.local.bytelen);
  178.                 //printf("\n%-*s %-*s %-*s %-*s %-*s %-*s\n", 10,"state",10,"family",10,"l.addr",10,"l.port",10,"r.addr",10,"r.rport");
  179.                 printf("%-*s ", 10, sstate_name[s.state]);
  180.                 printf("%-*s",10,"AF_INET" );

  181.                 //printf("\n--------------\n");
  182.                 const inet_prefix *a=&s.remote;
  183.                 const void *addr =  a->data;

  184.                 ap=resolve_address(&s.local.data,4,AF_INET);
  185.                 printf("%-*s",15, ap);
  186.                 printf("%-*d ", 10,s.lport );
  187.                 //ap2=resolve_address(&s.remote.data,4,AF_INET);
  188.                 ap2=rt_addr_n2a(AF_INET,4,addr,buf2,sizeof(buf2));
  189.                 printf("%-*s", 15, ap2);
  190.                 printf("%-*d\n",10,s.rport);

  191.                 //printf("L.port:%-*d R.prot:%-*d\n", 10,s.lport ,10,s.rport);

  192.                 //printf("idiag_state:%d\n", pkg->idiag_state);

  193.                 //printf("idiag_state:%d\n", pkg->idiag_state);
  194.                 //printf("Family:%s\n", pkg->idiag_family == AF_INET ? "AF_INET" : "AF_INET6");
  195.        // printf("dport:%d, sprot:%d\n", ntohs(pkg->id.idiag_sport), ntohs(pkg->id.idiag_sport));
  196.                 //printf("idiag_state:%d\n", pkg->idiag_state);

  197. }


  198. int main(int argc, char **argv)
  199. {
  200.         int fd;
  201.     struct sk_req req;
  202.     struct sockaddr_nl dest_addr;
  203.     struct msghdr msg;
  204.     char buf[8192];
  205.     char src_ip[20];
  206.     char dest_ip[20];
  207.     struct iovec iov;
  208.     if ((fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_INET_DIAG)) < 0) {
  209.         //eprint(__LINE__, errno, "socket");
  210.                 printf("socket error\n");
  211.         return -1;
  212.     }

  213. req.nlh.nlmsg_len = sizeof(req);
  214. req.nlh.nlmsg_type = TCPDIAG_GETSOCK;
  215. req.nlh.nlmsg_flags = NLM_F_REQUEST | NLM_F_ROOT;
  216. req.nlh.nlmsg_pid = 0;

  217. memset(&req.r, 0, sizeof(req.r));
  218. req.r.idiag_family = AF_INET;
  219. req.r.idiag_states = ((1 << TCP_CLOSING + 1) - 1);
  220. iov.iov_base = &req;
  221. iov.iov_len = sizeof(req);

  222. memset(&dest_addr, 0, sizeof(dest_addr));
  223. dest_addr.nl_family = AF_NETLINK;
  224. memset(&dest_addr, 0, sizeof(dest_addr));
  225. dest_addr.nl_family = AF_NETLINK;
  226. dest_addr.nl_pid = 0;
  227. dest_addr.nl_groups = 0;


  228. memset(&msg, 0, sizeof(msg));
  229. msg.msg_name = (void *)&dest_addr;
  230. msg.msg_namelen = sizeof(dest_addr);
  231. msg.msg_iov = &iov;
  232. msg.msg_iovlen = 1;

  233. if (sendmsg(fd, &msg, 0) < 0) {
  234.     //eprint(__LINE__, errno, "sendmsg");
  235.     printf("socket error\n");
  236.         return -1;
  237. }
  238. printf("\n%-*s %-*s %-*s %-*s   %-*s %-*s\n", 10,"state",10,"family",10,"l.addr",10,"l.port",10,"r.addr",10,"r.rport");
  239. memset(buf, 0 ,sizeof(buf));
  240. iov.iov_base = buf;
  241. iov.iov_len = sizeof(buf);
  242. while (1) {
  243.     int status;
  244.     struct nlmsghdr *h;
  245.     msg = (struct msghdr) {
  246.         (void *)&dest_addr, sizeof(dest_addr),
  247.             &iov, 1, NULL, 0, 0
  248.     };
  249.     status = recvmsg(fd, &msg, 0);//recvmsg函数的返回值是读取的字节数,
  250.     if (status < 0) {
  251.         if (errno == EINTR)
  252.             continue;
  253.         //eprint(__LINE__, errno, "recvmsg");
  254.         printf("socket error\n");
  255.                 continue;
  256.     }
  257.     if (status == 0) {
  258.         printf("EOF on netlink\n");
  259.         close(fd);
  260.         return 0;
  261.     }
  262.     h = (struct nlmsghdr *)buf;
  263. //      #define NLMSG_OK(nlh,len) ((len) >= (int)sizeof(struct nlmsghdr) && \
  264.                            (nlh)->nlmsg_len >= sizeof(struct nlmsghdr) && \
  265.                            (nlh)->nlmsg_len <= (len))
  266.     while (NLMSG_OK(h, status)) {
  267.         struct inet_diag_msg *pkg = NULL;
  268.                 /*
  269.   struct inet_diag_msg *pkg = NULL;
  270.                 /*
  271.                 struct inet_diag_msg {
  272.         __u8    idiag_family;
  273.         __u8    idiag_state;
  274.         __u8    idiag_timer;
  275.         __u8    idiag_retrans;

  276.         struct inet_diag_sockid id;

  277.         __u32   idiag_expires;
  278.         __u32   idiag_rqueue;
  279.         __u32   idiag_wqueue;
  280.         __u32   idiag_uid;
  281.         __u32   idiag_inode;
  282. };
  283.                 */
  284.         if (h->nlmsg_type == NLMSG_DONE) {
  285.             close(fd);
  286.             printf("NLMSG_DONE\n");
  287.             return 0;
  288.         }
  289.         if (h->nlmsg_type == NLMSG_ERROR) {
  290.             struct nlmsgerr *err;
  291.             err = (struct nlmsgerr*)NLMSG_DATA(h);
  292.             fprintf(stderr, "%d Error %d:%s\n", __LINE__, -(err->error), strerror(-(err->error)));
  293.             close(fd);
  294.             printf("NLMSG_ERROR\n");
  295.             return 0;
  296.         }
  297.         pkg = (struct inet_diag_msg *)NLMSG_DATA(h);
  298.         //print_skinfo(pkg);
  299.         //printf("\n%-*s %-*s %-*s %-*s %-*s %-*s\n", 10,"state",10,"family",10,"l.addr",10,"l.port",10,"r.addr",10,"r.rport");
  300.                 print_info(pkg,h);

  301.         //get_tcp_state(pkg->idiag_state);
  302.         h = NLMSG_NEXT(h, status);
  303.                 //#define NLMSG_NEXT(nlh,len)    ((len) -= NLMSG_ALIGN((nlh)->nlmsg_len), \
  304.                                   (struct nlmsghdr*)(((char*)(nlh)) + NLMSG_ALIGN((nlh)->nlmsg_len)))
  305.                 //#define NLMSG_ALIGN(len) ( ((len)+NLMSG_ALIGNTO-1) & ~(NLMSG_ALIGNTO-1) )
  306.     }//while
  307. }//while
  308. close(fd);
  309. return 0;
  310. }
  311.                                                   
  312.                                 


  313.                                                                                                    
  314.                                                                                                                                          


  315.                                                                              
  316.                                                                  
  317.                                        
复制代码






论坛徽章:
9
辰龙
日期:2014-08-18 20:38:42未羊
日期:2014-09-04 08:50:45丑牛
日期:2014-09-06 00:12:55寅虎
日期:2014-12-22 20:50:56摩羯座
日期:2015-01-14 22:28:15巳蛇
日期:2015-01-23 20:39:272015年辞旧岁徽章
日期:2015-03-03 16:54:1515-16赛季CBA联赛之青岛
日期:2016-03-13 23:37:1915-16赛季CBA联赛之深圳
日期:2016-03-29 18:52:38
2 [报告]
发表于 2014-05-31 19:58 |只看该作者
有这个必要吗?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP