main.cpp 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. #include "address.h"
  2. #include "hardware_address.h"
  3. #include "link.h"
  4. #include "vector.h"
  5. #include <cassert>
  6. #include <cstdlib>
  7. #include <iostream>
  8. #include <libmnl/libmnl.h>
  9. #include <linux/if_addr.h>
  10. #include <linux/netlink.h>
  11. #include <linux/rtnetlink.h> // struct ifinfomsg
  12. // https://netfilter.org/projects/libmnl/doxygen/html/modules.html
  13. void mnl_recv_run_cb_all(const mnl_socket *nl, void *buf, size_t bufsiz,
  14. mnl_cb_t cb_data, void *data) {
  15. unsigned int nlpid = mnl_socket_get_portid(nl);
  16. ssize_t numbytes = mnl_socket_recvfrom(nl, buf, bufsiz);
  17. while (numbytes > 0) {
  18. if (mnl_cb_run(buf, numbytes, 0, nlpid, cb_data, data) <= MNL_CB_STOP) {
  19. break;
  20. }
  21. numbytes = mnl_socket_recvfrom(nl, buf, bufsiz);
  22. }
  23. assert(numbytes != -1);
  24. }
  25. int link_cb(const nlmsghdr *nlh, void *data) {
  26. vector<Link> *links = (vector<Link> *)data;
  27. links->emplace_back((const ifinfomsg *)mnl_nlmsg_get_payload(nlh));
  28. assert(links->back().index == links->size());
  29. mnl_attr_parse(nlh, sizeof(ifinfomsg), Link::mnl_attr_cb, &links->back());
  30. return MNL_CB_OK;
  31. }
  32. void mnl_read_links(const mnl_socket *nl, vector<Link> *links) {
  33. uint8_t msgbuf[MNL_SOCKET_BUFFER_SIZE];
  34. nlmsghdr *nlh = mnl_nlmsg_put_header(msgbuf);
  35. nlh->nlmsg_type = RTM_GETLINK;
  36. nlh->nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
  37. mnl_nlmsg_put_extra_header(nlh, sizeof(rtgenmsg));
  38. assert(mnl_socket_sendto(nl, nlh, nlh->nlmsg_len) > 0);
  39. mnl_recv_run_cb_all(nl, msgbuf, sizeof(msgbuf), link_cb, links);
  40. }
  41. static int address_cb(const struct nlmsghdr *nlh, void *data) {
  42. vector<Address> *addrs = (vector<Address> *)data;
  43. addrs->emplace_back();
  44. addrs->back() = (const ifaddrmsg *)mnl_nlmsg_get_payload(nlh);
  45. mnl_attr_parse(nlh, sizeof(ifaddrmsg), Address::mnl_attr_cb, &addrs->back());
  46. return MNL_CB_OK;
  47. }
  48. void mnl_read_addresses(const mnl_socket *nl, vector<Address> *addrs) {
  49. char msgbuf[MNL_SOCKET_BUFFER_SIZE];
  50. nlmsghdr *nlh = mnl_nlmsg_put_header(msgbuf);
  51. nlh->nlmsg_type = RTM_GETADDR;
  52. nlh->nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
  53. rtgenmsg *rt = (rtgenmsg *)mnl_nlmsg_put_extra_header(nlh, sizeof(rtgenmsg));
  54. rt->rtgen_family = AF_UNSPEC;
  55. assert(mnl_socket_sendto(nl, nlh, nlh->nlmsg_len) > 0);
  56. mnl_recv_run_cb_all(nl, msgbuf, sizeof(msgbuf), address_cb, addrs);
  57. }
  58. int main(int argc, char *argv[]) {
  59. mnl_socket *nl = mnl_socket_open(NETLINK_ROUTE);
  60. assert(nl);
  61. assert(mnl_socket_bind(nl, 0, MNL_SOCKET_AUTOPID) == 0);
  62. vector<Link> links;
  63. mnl_read_links(nl, &links);
  64. std::cout << "links: ";
  65. links.write_yaml(std::cout);
  66. vector<Address> addrs;
  67. mnl_read_addresses(nl, &addrs);
  68. std::cout << "addresses: ";
  69. addrs.write_yaml(std::cout);
  70. std::cout << std::endl;
  71. mnl_socket_close(nl);
  72. return EXIT_SUCCESS;
  73. }