/examples/blackadder.cpp

https://github.com/tsilochr/blackadder · C++ · 278 lines · 234 code · 23 blank · 21 comment · 26 complexity · 70ede64a728d86d265aba68d2b514ee0 MD5 · raw file

  1. /*
  2. * Copyright (C) 2010-2011 George Parisis and Dirk Trossen
  3. * All rights reserved.
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License version
  7. * 2 as published by the Free Software Foundation.
  8. *
  9. * Alternatively, this software may be distributed under the terms of
  10. * the BSD license.
  11. *
  12. * See LICENSE and COPYING for more details.
  13. */
  14. #include "blackadder.hpp"
  15. using namespace std;
  16. Blackadder::Blackadder(bool user_space) {
  17. int ret;
  18. if (user_space) {
  19. cout << "Initializing blackadder client for user space" << endl;
  20. sock_fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC);
  21. } else {
  22. cout << "Initializing blackadder client for kernel space" << endl;
  23. sock_fd = socket(AF_NETLINK, SOCK_RAW, NETLINK_BADDER);
  24. }
  25. if (sock_fd < 0) {
  26. perror("socket");
  27. } else {
  28. cout << "Created and opened netlink socket" << endl;
  29. }
  30. /* source address */
  31. memset(&s_nladdr, 0, sizeof (s_nladdr));
  32. s_nladdr.nl_family = AF_NETLINK;
  33. s_nladdr.nl_pad = 0;
  34. s_nladdr.nl_pid = getpid();
  35. ret = bind(sock_fd, (struct sockaddr*) &s_nladdr, sizeof (s_nladdr));
  36. if (ret < 0) {
  37. perror("bind");
  38. }
  39. /* destination address */
  40. memset(&d_nladdr, 0, sizeof (d_nladdr));
  41. d_nladdr.nl_family = AF_NETLINK;
  42. d_nladdr.nl_pad = 0;
  43. if (user_space) {
  44. d_nladdr.nl_pid = 9999; /* destined to user space blackadder */
  45. } else {
  46. d_nladdr.nl_pid = 0; /* destined to kernel */
  47. }
  48. fake_buf = (char *) malloc(1);
  49. }
  50. Blackadder::~Blackadder() {
  51. free(fake_buf);
  52. if (sock_fd != -1) {
  53. cout << "Closed netlink socket" << endl;
  54. close(sock_fd);
  55. }
  56. }
  57. void Blackadder::create_and_send_buffers(unsigned char type, string &id, string &prefix_id, char strategy, char *LID) {
  58. struct msghdr msg;
  59. struct iovec *iov;
  60. unsigned char id_len = id.length() / PURSUIT_ID_LEN;
  61. unsigned char prefix_id_len = prefix_id.length() / PURSUIT_ID_LEN;
  62. struct nlmsghdr *nlh = (struct nlmsghdr *) malloc(sizeof (struct nlmsghdr));
  63. if (LID == NULL) {
  64. iov = (struct iovec *) calloc(sizeof (struct iovec), 7);
  65. /* Fill the netlink message header */
  66. nlh->nlmsg_len = sizeof (struct nlmsghdr) + 1 /*type*/ + 1 /*for id length*/ + id.length() + 1 /*for prefix_id length*/ + prefix_id.length() + 1 /*strategy*/;
  67. nlh->nlmsg_pid = getpid();
  68. nlh->nlmsg_flags = 1;
  69. nlh->nlmsg_type = 0;
  70. } else {
  71. iov = (struct iovec *) calloc(sizeof (struct iovec), 8);
  72. /* Fill the netlink message header */
  73. nlh->nlmsg_len = sizeof (struct nlmsghdr) + 1 /*type*/ + 1 /*for id length*/ + id.length() + 1 /*for prefix_id length*/ + prefix_id.length() + 1 /*strategy*/ + FID_LEN /*optional LID*/;
  74. nlh->nlmsg_pid = getpid();
  75. nlh->nlmsg_flags = 1;
  76. nlh->nlmsg_type = 0;
  77. }
  78. iov[0].iov_base = nlh;
  79. iov[0].iov_len = sizeof (struct nlmsghdr);
  80. iov[1].iov_base = &type;
  81. iov[1].iov_len = sizeof (type);
  82. iov[2].iov_base = &id_len;
  83. iov[2].iov_len = sizeof (id_len);
  84. iov[3].iov_base = (void *) id.c_str();
  85. iov[3].iov_len = id.length();
  86. iov[4].iov_base = &prefix_id_len;
  87. iov[4].iov_len = sizeof (prefix_id_len);
  88. iov[5].iov_base = (void *) prefix_id.c_str();
  89. iov[5].iov_len = prefix_id.length();
  90. iov[6].iov_base = &strategy;
  91. iov[6].iov_len = sizeof (strategy);
  92. if (LID == NULL) {
  93. memset(&msg, 0, sizeof (msg));
  94. msg.msg_name = (void *) &d_nladdr;
  95. msg.msg_namelen = sizeof (d_nladdr);
  96. msg.msg_iov = iov;
  97. msg.msg_iovlen = 7;
  98. sendmsg(sock_fd, &msg, 0);
  99. } else {
  100. iov[7].iov_base = (void *) LID;
  101. iov[7].iov_len = FID_LEN;
  102. memset(&msg, 0, sizeof (msg));
  103. msg.msg_name = (void *) &d_nladdr;
  104. msg.msg_namelen = sizeof (d_nladdr);
  105. msg.msg_iov = iov;
  106. msg.msg_iovlen = 8;
  107. sendmsg(sock_fd, &msg, 0);
  108. }
  109. free(iov);
  110. free(nlh);
  111. }
  112. void Blackadder::disconnect() {
  113. struct msghdr msg;
  114. struct iovec *iov;
  115. struct nlmsghdr *nlh = (struct nlmsghdr *) malloc(sizeof (struct nlmsghdr));
  116. unsigned char type = DISCONNECT;
  117. /* Fill the netlink message header */
  118. nlh->nlmsg_len = sizeof (struct nlmsghdr) + 1 /*type*/;
  119. nlh->nlmsg_pid = getpid();
  120. nlh->nlmsg_flags = 1;
  121. nlh->nlmsg_type = 0;
  122. iov = new iovec[2];
  123. iov[0].iov_base = nlh;
  124. iov[0].iov_len = sizeof (struct nlmsghdr);
  125. iov[1].iov_base = &type;
  126. iov[1].iov_len = sizeof (type);
  127. memset(&msg, 0, sizeof (msg));
  128. msg.msg_name = (void *) &d_nladdr;
  129. msg.msg_namelen = sizeof (d_nladdr);
  130. msg.msg_iov = iov;
  131. msg.msg_iovlen = 2;
  132. sendmsg(sock_fd, &msg, 0);
  133. free(nlh);
  134. delete [] iov;
  135. close(sock_fd);
  136. cout << "Closed netlink socket" << endl;
  137. sock_fd = -1;
  138. }
  139. void Blackadder::publish_scope(string &id, string &prefix_id, char strategy, char *LID) {
  140. create_and_send_buffers(PUBLISH_SCOPE, id, prefix_id, strategy, LID);
  141. }
  142. void Blackadder::publish_info(string &id, string &prefix_id, char strategy, char *LID) {
  143. create_and_send_buffers(PUBLISH_INFO, id, prefix_id, strategy, LID);
  144. }
  145. void Blackadder::unpublish_scope(string &id, string &prefix_id, char strategy, char *LID) {
  146. create_and_send_buffers(UNPUBLISH_SCOPE, id, prefix_id, strategy, LID);
  147. }
  148. void Blackadder::unpublish_info(string &id, string &prefix_id, char strategy, char *LID) {
  149. create_and_send_buffers(UNPUBLISH_INFO, id, prefix_id, strategy, LID);
  150. }
  151. void Blackadder::subscribe_scope(string &id, string &prefix_id, char strategy, char *LID) {
  152. create_and_send_buffers(SUBSCRIBE_SCOPE, id, prefix_id, strategy, LID);
  153. }
  154. void Blackadder::subscribe_info(string &id, string &prefix_id, char strategy, char *LID) {
  155. create_and_send_buffers(SUBSCRIBE_INFO, id, prefix_id, strategy, LID);
  156. }
  157. void Blackadder::unsubscribe_scope(string &id, string &prefix_id, char strategy, char *LID) {
  158. create_and_send_buffers(UNSUBSCRIBE_SCOPE, id, prefix_id, strategy, LID);
  159. }
  160. void Blackadder::unsubscribe_info(string &id, string &prefix_id, char strategy, char *LID) {
  161. create_and_send_buffers(UNSUBSCRIBE_INFO, id, prefix_id, strategy, LID);
  162. }
  163. void Blackadder::publish_data(string &id, char strategy, char * LID, char *data, int data_len) {
  164. struct msghdr msg;
  165. struct iovec *iov;
  166. unsigned char type = PUBLISH_DATA;
  167. unsigned char id_len = id.length() / PURSUIT_ID_LEN;
  168. struct nlmsghdr *nlh = (struct nlmsghdr *) malloc(sizeof (struct nlmsghdr));
  169. if (LID == NULL) {
  170. /* Fill the netlink message header */
  171. nlh->nlmsg_len = sizeof (struct nlmsghdr) + 1 /*type*/ + 1 /*for id length*/ + id.length() + sizeof (strategy) + data_len;
  172. nlh->nlmsg_pid = getpid();
  173. nlh->nlmsg_flags = 1;
  174. nlh->nlmsg_type = 0;
  175. iov = (struct iovec *) calloc(sizeof (struct iovec), 6);
  176. } else {
  177. /* Fill the netlink message header */
  178. nlh->nlmsg_len = sizeof (struct nlmsghdr) + 1 /*type*/ + 1 /*for id length*/ + id.length() + sizeof (strategy) + FID_LEN + data_len;
  179. nlh->nlmsg_pid = getpid();
  180. nlh->nlmsg_flags = 1;
  181. nlh->nlmsg_type = 0;
  182. iov = (struct iovec *) calloc(sizeof (struct iovec), 7);
  183. }
  184. iov[0].iov_base = nlh;
  185. iov[0].iov_len = sizeof (struct nlmsghdr);
  186. iov[1].iov_base = &type;
  187. iov[1].iov_len = sizeof (type);
  188. iov[2].iov_base = &id_len;
  189. iov[2].iov_len = sizeof (id_len);
  190. iov[3].iov_base = (void *) id.c_str();
  191. iov[3].iov_len = id.length();
  192. iov[4].iov_base = (void *) &strategy;
  193. iov[4].iov_len = sizeof (unsigned char);
  194. if (LID == NULL) {
  195. iov[5].iov_base = (void *) data;
  196. iov[5].iov_len = data_len;
  197. memset(&msg, 0, sizeof (msg));
  198. msg.msg_name = (void *) &d_nladdr;
  199. msg.msg_namelen = sizeof (d_nladdr);
  200. msg.msg_iov = iov;
  201. msg.msg_iovlen = 6;
  202. sendmsg(sock_fd, &msg, MSG_WAITALL);
  203. } else {
  204. iov[5].iov_base = (void *) LID;
  205. iov[5].iov_len = FID_LEN;
  206. iov[6].iov_base = (void *) data;
  207. iov[6].iov_len = data_len;
  208. memset(&msg, 0, sizeof (msg));
  209. msg.msg_name = (void *) &d_nladdr;
  210. msg.msg_namelen = sizeof (d_nladdr);
  211. msg.msg_iov = iov;
  212. msg.msg_iovlen = 7;
  213. sendmsg(sock_fd, &msg, MSG_WAITALL);
  214. }
  215. free(nlh);
  216. free(iov);
  217. }
  218. Event Blackadder::getEvent() {
  219. int total_buf_size = 0;
  220. int bytes_read;
  221. int data_len;
  222. unsigned char type;
  223. unsigned char id_len;
  224. char *data;
  225. char *buf;
  226. total_buf_size = recv(sock_fd, fake_buf, 1, MSG_PEEK | MSG_TRUNC | MSG_WAITALL);
  227. if (total_buf_size > 0) {
  228. buf = (char *) malloc(total_buf_size);
  229. bytes_read = recv(sock_fd, buf, total_buf_size, MSG_WAITALL);
  230. //cout << "bytes_read: " << bytes_read << endl;
  231. type = *(buf + sizeof (struct nlmsghdr));
  232. id_len = *(buf + sizeof (struct nlmsghdr) + sizeof (unsigned char));
  233. string str_id(buf + sizeof (struct nlmsghdr) + sizeof (unsigned char) + sizeof (unsigned char), ((int) id_len) * PURSUIT_ID_LEN);
  234. if (type == PUBLISHED_DATA) {
  235. data = buf + sizeof (struct nlmsghdr) + sizeof (unsigned char) + sizeof (unsigned char) + ((int) id_len) * PURSUIT_ID_LEN;
  236. data_len = total_buf_size - (sizeof (struct nlmsghdr) + sizeof (unsigned char) + sizeof (unsigned char) + ((int) id_len) * PURSUIT_ID_LEN);
  237. } else {
  238. data = NULL;
  239. data_len = 0;
  240. }
  241. return Event(type, str_id, data, data_len, buf);
  242. } else {
  243. return Event(0, "", NULL, 0, NULL);
  244. }
  245. }
  246. Event::Event(unsigned char _type, string _id, char *_data, int _data_len, char *_buffer) {
  247. type = _type;
  248. id = _id;
  249. data_len = _data_len;
  250. data = _data;
  251. buffer = _buffer;
  252. }
  253. Event::~Event() {
  254. free(buffer);
  255. }