LCOV - code coverage report
Current view: top level - crtools - libnetlink.c (source / functions) Hit Total Coverage
Test: crtools.info Lines: 52 68 76.5 %
Date: 2012-12-28 Functions: 4 4 100.0 %
Branches: 27 46 58.7 %

           Branch data     Line data    Source code
       1                 :            : #include <linux/types.h>
       2                 :            : #include <sys/socket.h>
       3                 :            : #include <linux/netlink.h>
       4                 :            : #include <linux/rtnetlink.h>
       5                 :            : #include <string.h>
       6                 :            : #include <unistd.h>
       7                 :            : 
       8                 :            : #include "libnetlink.h"
       9                 :            : #include "util.h"
      10                 :            : 
      11                 :       7183 : int parse_rtattr(struct rtattr *tb[], int max, struct rtattr *rta, int len)
      12                 :            : {
      13                 :       7183 :         memset(tb, 0, sizeof(struct rtattr *) * (max + 1));
      14 [ +  + ][ +  - ]:      29477 :         while (RTA_OK(rta, len)) {
                 [ +  - ]
      15 [ +  - ][ +  - ]:      22294 :                 if ((rta->rta_type <= max) && (!tb[rta->rta_type]))
      16                 :      22294 :                         tb[rta->rta_type] = rta;
      17                 :      22294 :                 rta = RTA_NEXT(rta, len);
      18                 :            :         }
      19         [ -  + ]:       7183 :         if (len)
      20                 :          0 :                 pr_warn("Trimmed RTA: len %d, rta_len %d\n", len, rta->rta_len);
      21                 :       7183 :         return 0;
      22                 :            : }
      23                 :            : 
      24                 :       2214 : static int nlmsg_receive(char *buf, int len, int (*cb)(struct nlmsghdr *, void *), void *arg)
      25                 :            : {
      26                 :            :         struct nlmsghdr *hdr;
      27                 :            : 
      28 [ +  + ][ +  - ]:      10885 :         for (hdr = (struct nlmsghdr *)buf; NLMSG_OK(hdr, len); hdr = NLMSG_NEXT(hdr, len)) {
                 [ +  - ]
      29         [ -  + ]:       8671 :                 if (hdr->nlmsg_seq != CR_NLMSG_SEQ)
      30                 :          0 :                         continue;
      31         [ +  + ]:       8671 :                 if (hdr->nlmsg_type == NLMSG_DONE) {
      32                 :       1412 :                         int *len = (int *)NLMSG_DATA(hdr);
      33                 :            : 
      34         [ -  + ]:       1412 :                         if (*len < 0) {
      35                 :          0 :                                 pr_err("ERROR %d reported by netlink (%s)\n",
      36                 :            :                                         *len, strerror(-*len));
      37                 :          0 :                                 return -1;
      38                 :            :                         }
      39                 :            : 
      40                 :            :                         return 0;
      41                 :            :                 }
      42         [ +  + ]:       7259 :                 if (hdr->nlmsg_type == NLMSG_ERROR) {
      43                 :         76 :                         struct nlmsgerr *err = (struct nlmsgerr *)NLMSG_DATA(hdr);
      44                 :            : 
      45         [ -  + ]:         76 :                         if (hdr->nlmsg_len - sizeof(*hdr) < sizeof(struct nlmsgerr)) {
      46                 :          0 :                                 pr_err("ERROR truncated\n");
      47                 :          0 :                                 return -1;
      48                 :            :                         }
      49                 :            : 
      50         [ -  + ]:         76 :                         if (err->error == 0)
      51                 :            :                                 return 0;
      52                 :            : 
      53                 :          0 :                         pr_err("ERROR %d reported by netlink\n", -err->error);
      54                 :          0 :                         return -1;
      55                 :            :                 }
      56         [ +  - ]:       7183 :                 if (cb(hdr, arg))
      57                 :            :                         return -1;
      58                 :            :         }
      59                 :            : 
      60                 :            :         return 1;
      61                 :            : }
      62                 :            : 
      63                 :       1488 : int do_rtnl_req(int nl, void *req, int size,
      64                 :            :                 int (*receive_callback)(struct nlmsghdr *h, void *), void *arg)
      65                 :            : {
      66                 :            :         struct msghdr msg;
      67                 :            :         struct sockaddr_nl nladdr;
      68                 :            :         struct iovec iov;
      69                 :            :         static char buf[4096];
      70                 :            : 
      71                 :       1488 :         memset(&msg, 0, sizeof(msg));
      72                 :       1488 :         msg.msg_name    = &nladdr;
      73                 :       1488 :         msg.msg_namelen = sizeof(nladdr);
      74                 :       1488 :         msg.msg_iov     = &iov;
      75                 :       1488 :         msg.msg_iovlen  = 1;
      76                 :            : 
      77                 :       1488 :         memset(&nladdr, 0, sizeof(nladdr));
      78                 :       1488 :         nladdr.nl_family = AF_NETLINK;
      79                 :            : 
      80                 :       1488 :         iov.iov_base    = req;
      81                 :       1488 :         iov.iov_len     = size;
      82                 :            : 
      83         [ -  + ]:       1488 :         if (sendmsg(nl, &msg, 0) < 0) {
      84                 :          0 :                 pr_perror("Can't send request message");
      85                 :          0 :                 goto err;
      86                 :            :         }
      87                 :            : 
      88                 :       1488 :         iov.iov_base    = buf;
      89                 :       2214 :         iov.iov_len     = sizeof(buf);
      90                 :            : 
      91                 :            :         while (1) {
      92                 :            :                 int err;
      93                 :            : 
      94                 :       2214 :                 memset(&msg, 0, sizeof(msg));
      95                 :       2214 :                 msg.msg_name    = &nladdr;
      96                 :       2214 :                 msg.msg_namelen = sizeof(nladdr);
      97                 :       2214 :                 msg.msg_iov     = &iov;
      98                 :       2214 :                 msg.msg_iovlen  = 1;
      99                 :            : 
     100                 :       2214 :                 err = recvmsg(nl, &msg, 0);
     101         [ -  + ]:       2214 :                 if (err < 0) {
     102         [ #  # ]:          0 :                         if (errno == EINTR)
     103                 :          0 :                                 continue;
     104                 :            :                         else {
     105                 :          0 :                                 pr_perror("Error receiving nl report");
     106                 :          0 :                                 goto err;
     107                 :            :                         }
     108                 :            :                 }
     109         [ +  - ]:       2214 :                 if (err == 0)
     110                 :            :                         break;
     111                 :            : 
     112                 :       2214 :                 err = nlmsg_receive(buf, err, receive_callback, arg);
     113         [ +  - ]:       2214 :                 if (err < 0)
     114                 :            :                         goto err;
     115         [ +  + ]:       2214 :                 if (err == 0)
     116                 :            :                         break;
     117                 :            :         }
     118                 :            : 
     119                 :            :         return 0;
     120                 :            : 
     121                 :            : err:
     122                 :            :         return -1;
     123                 :            : }
     124                 :            : 
     125                 :        152 : int addattr_l(struct nlmsghdr *n, int maxlen, int type, const void *data,
     126                 :            :                 int alen)
     127                 :            : {
     128                 :        152 :         int len = RTA_LENGTH(alen);
     129                 :            :         struct rtattr *rta;
     130                 :            : 
     131         [ -  + ]:        152 :         if (NLMSG_ALIGN(n->nlmsg_len) + RTA_ALIGN(len) > maxlen) {
     132                 :          0 :                 pr_err("addattr_l ERROR: message exceeded bound of %d\n", maxlen);
     133                 :          0 :                 return -1;
     134                 :            :         }
     135                 :            : 
     136                 :        152 :         rta = NLMSG_TAIL(n);
     137                 :        152 :         rta->rta_type = type;
     138                 :        152 :         rta->rta_len = len;
     139                 :        152 :         memcpy(RTA_DATA(rta), data, alen);
     140                 :        152 :         n->nlmsg_len = NLMSG_ALIGN(n->nlmsg_len) + RTA_ALIGN(len);
     141                 :        152 :         return 0;
     142                 :            : }

Generated by: LCOV version 1.9