netlink.c 1.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. #include <errno.h>
  2. #include <string.h>
  3. #include <syscall.h>
  4. #include <sys/socket.h>
  5. #include "netlink.h"
  6. static int __netlink_enumerate(int fd, unsigned int seq, int type, int af,
  7. int (*cb)(void *ctx, struct nlmsghdr *h), void *ctx)
  8. {
  9. struct nlmsghdr *h;
  10. union {
  11. uint8_t buf[8192];
  12. struct {
  13. struct nlmsghdr nlh;
  14. struct rtgenmsg g;
  15. } req;
  16. struct nlmsghdr reply;
  17. } u;
  18. int r, ret;
  19. memset(&u.req, 0, sizeof(u.req));
  20. u.req.nlh.nlmsg_len = sizeof(u.req);
  21. u.req.nlh.nlmsg_type = type;
  22. u.req.nlh.nlmsg_flags = NLM_F_DUMP | NLM_F_REQUEST;
  23. u.req.nlh.nlmsg_seq = seq;
  24. u.req.g.rtgen_family = af;
  25. r = send(fd, &u.req, sizeof(u.req), 0);
  26. if (r < 0) return r;
  27. while (1) {
  28. r = recv(fd, u.buf, sizeof(u.buf), MSG_DONTWAIT);
  29. if (r <= 0) return -1;
  30. for (h = &u.reply; NLMSG_OK(h, (void*)&u.buf[r]); h = NLMSG_NEXT(h)) {
  31. if (h->nlmsg_type == NLMSG_DONE) return 0;
  32. if (h->nlmsg_type == NLMSG_ERROR) return -1;
  33. ret = cb(ctx, h);
  34. if (ret) return ret;
  35. }
  36. }
  37. }
  38. int __rtnetlink_enumerate(int link_af, int addr_af, int (*cb)(void *ctx, struct nlmsghdr *h), void *ctx)
  39. {
  40. int fd, r;
  41. fd = socket(PF_NETLINK, SOCK_RAW|SOCK_CLOEXEC, NETLINK_ROUTE);
  42. if (fd < 0) return -1;
  43. r = __netlink_enumerate(fd, 1, RTM_GETLINK, link_af, cb, ctx);
  44. if (!r) r = __netlink_enumerate(fd, 2, RTM_GETADDR, addr_af, cb, ctx);
  45. __syscall(SYS_close,fd);
  46. return r;
  47. }