FrameDecider.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831
  1. /**
  2. * @file FrameDecider.c
  3. * @author Ambroz Bizjak <ambrop7@gmail.com>
  4. *
  5. * @section LICENSE
  6. *
  7. * This file is part of BadVPN.
  8. *
  9. * BadVPN is free software: you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License version 2
  11. * as published by the Free Software Foundation.
  12. *
  13. * BadVPN is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License along
  19. * with this program; if not, write to the Free Software Foundation, Inc.,
  20. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  21. */
  22. #include <string.h>
  23. #include <stddef.h>
  24. #include <inttypes.h>
  25. #include <misc/debug.h>
  26. #include <misc/offset.h>
  27. #include <misc/balloc.h>
  28. #include <misc/ethernet_proto.h>
  29. #include <misc/ipv4_proto.h>
  30. #include <misc/igmp_proto.h>
  31. #include <misc/byteorder.h>
  32. #include <system/BLog.h>
  33. #include <client/FrameDecider.h>
  34. #include <generated/blog_channel_FrameDecider.h>
  35. #define DECIDE_STATE_NONE 1
  36. #define DECIDE_STATE_UNICAST 2
  37. #define DECIDE_STATE_FLOOD 3
  38. #define DECIDE_STATE_MULTICAST 4
  39. static int mac_comparator (void *user, uint8_t *mac1, uint8_t *mac2)
  40. {
  41. int c = memcmp(mac1, mac2, 6);
  42. if (c < 0) {
  43. return -1;
  44. }
  45. if (c > 0) {
  46. return 1;
  47. }
  48. return 0;
  49. }
  50. static int uint32_comparator (void *user, uint32_t *v1, uint32_t *v2)
  51. {
  52. if (*v1 < *v2) {
  53. return -1;
  54. }
  55. if (*v1 > *v2) {
  56. return 1;
  57. }
  58. return 0;
  59. }
  60. static void add_mac_to_peer (FrameDeciderPeer *o, uint8_t *mac)
  61. {
  62. FrameDecider *d = o->d;
  63. // locate entry in tree
  64. BAVLNode *tree_node = BAVL_LookupExact(&d->macs_tree, mac);
  65. if (tree_node) {
  66. struct _FrameDecider_mac_entry *entry = UPPER_OBJECT(tree_node, struct _FrameDecider_mac_entry, tree_node);
  67. if (entry->peer == o) {
  68. // this is our MAC; only move it to the end of the used list
  69. LinkedList2_Remove(&o->mac_entries_used, &entry->list_node);
  70. LinkedList2_Append(&o->mac_entries_used, &entry->list_node);
  71. return;
  72. }
  73. // some other peer has that MAC; disassociate it
  74. BAVL_Remove(&d->macs_tree, &entry->tree_node);
  75. LinkedList2_Remove(&entry->peer->mac_entries_used, &entry->list_node);
  76. LinkedList2_Append(&entry->peer->mac_entries_free, &entry->list_node);
  77. }
  78. // aquire MAC address entry, if there are no free ones reuse the oldest used one
  79. LinkedList2Node *list_node;
  80. struct _FrameDecider_mac_entry *entry;
  81. if (list_node = LinkedList2_GetFirst(&o->mac_entries_free)) {
  82. entry = UPPER_OBJECT(list_node, struct _FrameDecider_mac_entry, list_node);
  83. ASSERT(entry->peer == o)
  84. // remove from free
  85. LinkedList2_Remove(&o->mac_entries_free, &entry->list_node);
  86. } else {
  87. list_node = LinkedList2_GetFirst(&o->mac_entries_used);
  88. ASSERT(list_node)
  89. entry = UPPER_OBJECT(list_node, struct _FrameDecider_mac_entry, list_node);
  90. ASSERT(entry->peer == o)
  91. // remove from used
  92. BAVL_Remove(&d->macs_tree, &entry->tree_node);
  93. LinkedList2_Remove(&o->mac_entries_used, &entry->list_node);
  94. }
  95. BLog(BLOG_INFO, "adding MAC %02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8"", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
  96. // set MAC in entry
  97. memcpy(entry->mac, mac, sizeof(entry->mac));
  98. // add to used
  99. LinkedList2_Append(&o->mac_entries_used, &entry->list_node);
  100. ASSERT_EXECUTE(BAVL_Insert(&d->macs_tree, &entry->tree_node, NULL))
  101. }
  102. static uint32_t compute_sig_for_group (uint32_t group)
  103. {
  104. return hton32(ntoh32(group)&0x7FFFFF);
  105. }
  106. static uint32_t compute_sig_for_mac (uint8_t *mac)
  107. {
  108. uint32_t sig;
  109. memcpy(&sig, mac + 2, 4);
  110. sig = hton32(ntoh32(sig)&0x7FFFFF);
  111. return sig;
  112. }
  113. static void add_to_multicast (FrameDecider *d, struct _FrameDecider_group_entry *group_entry)
  114. {
  115. // compute sig
  116. uint32_t sig = compute_sig_for_group(group_entry->group);
  117. BAVLNode *node;
  118. if (node = BAVL_LookupExact(&d->multicast_tree, &sig)) {
  119. // use existing master
  120. struct _FrameDecider_group_entry *master = UPPER_OBJECT(node, struct _FrameDecider_group_entry, master.tree_node);
  121. ASSERT(master->is_master)
  122. // set not master
  123. group_entry->is_master = 0;
  124. // insert to list
  125. LinkedList3Node_InitAfter(&group_entry->sig_list_node, &master->sig_list_node);
  126. } else {
  127. // make this entry master
  128. // set master
  129. group_entry->is_master = 1;
  130. // set sig
  131. group_entry->master.sig = sig;
  132. // insert to multicast tree
  133. ASSERT_EXECUTE(BAVL_Insert(&d->multicast_tree, &group_entry->master.tree_node, NULL))
  134. // init list node
  135. LinkedList3Node_InitLonely(&group_entry->sig_list_node);
  136. }
  137. }
  138. static void remove_from_multicast (FrameDecider *d, struct _FrameDecider_group_entry *group_entry)
  139. {
  140. // compute sig
  141. uint32_t sig = compute_sig_for_group(group_entry->group);
  142. if (group_entry->is_master) {
  143. // remove master from multicast tree
  144. BAVL_Remove(&d->multicast_tree, &group_entry->master.tree_node);
  145. if (!LinkedList3Node_IsLonely(&group_entry->sig_list_node)) {
  146. // at least one more group entry for this sig; make another entry the master
  147. // get an entry
  148. LinkedList3Node *list_node = LinkedList3Node_NextOrPrev(&group_entry->sig_list_node);
  149. struct _FrameDecider_group_entry *newmaster = UPPER_OBJECT(list_node, struct _FrameDecider_group_entry, sig_list_node);
  150. ASSERT(!newmaster->is_master)
  151. // set master
  152. newmaster->is_master = 1;
  153. // set sig
  154. newmaster->master.sig = sig;
  155. // insert to multicast tree
  156. ASSERT_EXECUTE(BAVL_Insert(&d->multicast_tree, &newmaster->master.tree_node, NULL))
  157. }
  158. }
  159. // free linked list node
  160. LinkedList3Node_Free(&group_entry->sig_list_node);
  161. }
  162. static void add_group_to_peer (FrameDeciderPeer *o, uint32_t group)
  163. {
  164. FrameDecider *d = o->d;
  165. struct _FrameDecider_group_entry *group_entry;
  166. // lookup if the peer already has that group
  167. BAVLNode *old_tree_node;
  168. if ((old_tree_node = BAVL_LookupExact(&o->groups_tree, &group))) {
  169. group_entry = UPPER_OBJECT(old_tree_node, struct _FrameDecider_group_entry, tree_node);
  170. // move to end of used list
  171. LinkedList2_Remove(&o->group_entries_used, &group_entry->list_node);
  172. LinkedList2_Append(&o->group_entries_used, &group_entry->list_node);
  173. } else {
  174. BLog(BLOG_INFO, "joined group %"PRIu8".%"PRIu8".%"PRIu8".%"PRIu8"",
  175. ((uint8_t *)&group)[0], ((uint8_t *)&group)[1], ((uint8_t *)&group)[2], ((uint8_t *)&group)[3]
  176. );
  177. // aquire group entry, if there are no free ones reuse the earliest used one
  178. LinkedList2Node *node;
  179. if (node = LinkedList2_GetFirst(&o->group_entries_free)) {
  180. group_entry = UPPER_OBJECT(node, struct _FrameDecider_group_entry, list_node);
  181. // remove from free list
  182. LinkedList2_Remove(&o->group_entries_free, &group_entry->list_node);
  183. } else {
  184. node = LinkedList2_GetFirst(&o->group_entries_used);
  185. ASSERT(node)
  186. group_entry = UPPER_OBJECT(node, struct _FrameDecider_group_entry, list_node);
  187. // remove from multicast
  188. remove_from_multicast(d, group_entry);
  189. // remove from peer's groups tree
  190. BAVL_Remove(&o->groups_tree, &group_entry->tree_node);
  191. // remove from used list
  192. LinkedList2_Remove(&o->group_entries_used, &group_entry->list_node);
  193. }
  194. // add entry to used list
  195. LinkedList2_Append(&o->group_entries_used, &group_entry->list_node);
  196. // set group address
  197. group_entry->group = group;
  198. // insert to peer's groups tree
  199. ASSERT_EXECUTE(BAVL_Insert(&o->groups_tree, &group_entry->tree_node, NULL))
  200. // add to multicast
  201. add_to_multicast(d, group_entry);
  202. }
  203. // set timer
  204. group_entry->timer_endtime = btime_gettime() + d->igmp_group_membership_interval;
  205. BReactor_SetTimerAbsolute(d->reactor, &group_entry->timer, group_entry->timer_endtime);
  206. }
  207. static void remove_group_entry (struct _FrameDecider_group_entry *group_entry)
  208. {
  209. FrameDeciderPeer *peer = group_entry->peer;
  210. FrameDecider *d = peer->d;
  211. uint32_t group = group_entry->group;
  212. BLog(BLOG_INFO, "left group %"PRIu8".%"PRIu8".%"PRIu8".%"PRIu8"",
  213. ((uint8_t *)&group)[0], ((uint8_t *)&group)[1], ((uint8_t *)&group)[2], ((uint8_t *)&group)[3]
  214. );
  215. // remove from multicast
  216. remove_from_multicast(d, group_entry);
  217. // remove from peer's groups tree
  218. BAVL_Remove(&peer->groups_tree, &group_entry->tree_node);
  219. // remove from used list
  220. LinkedList2_Remove(&peer->group_entries_used, &group_entry->list_node);
  221. // add to free list
  222. LinkedList2_Append(&peer->group_entries_free, &group_entry->list_node);
  223. // stop timer
  224. BReactor_RemoveTimer(d->reactor, &group_entry->timer);
  225. }
  226. static void lower_group_timers_to_lmqt (FrameDecider *d, uint32_t group)
  227. {
  228. // have to lower all the group timers of this group down to LMQT
  229. // compute sig
  230. uint32_t sig = compute_sig_for_group(group);
  231. // look up the sig in multicast tree
  232. BAVLNode *tree_node;
  233. if (!(tree_node = BAVL_LookupExact(&d->multicast_tree, &sig))) {
  234. return;
  235. }
  236. struct _FrameDecider_group_entry *master = UPPER_OBJECT(tree_node, struct _FrameDecider_group_entry, master.tree_node);
  237. ASSERT(master->is_master)
  238. // iterate all group entries with this sig
  239. LinkedList3Iterator it;
  240. LinkedList3Iterator_Init(&it, LinkedList3Node_First(&master->sig_list_node), 1);
  241. LinkedList3Node *sig_list_node;
  242. while (sig_list_node = LinkedList3Iterator_Next(&it)) {
  243. struct _FrameDecider_group_entry *group_entry = UPPER_OBJECT(sig_list_node, struct _FrameDecider_group_entry, sig_list_node);
  244. // skip wrong groups
  245. if (group_entry->group != group) {
  246. continue;
  247. }
  248. // lower timer down to LMQT
  249. btime_t now = btime_gettime();
  250. if (group_entry->timer_endtime > now + d->igmp_last_member_query_time) {
  251. group_entry->timer_endtime = now + d->igmp_last_member_query_time;
  252. BReactor_SetTimerAbsolute(d->reactor, &group_entry->timer, group_entry->timer_endtime);
  253. }
  254. }
  255. }
  256. static void group_entry_timer_handler (struct _FrameDecider_group_entry *group_entry)
  257. {
  258. DebugObject_Access(&group_entry->peer->d_obj);
  259. remove_group_entry(group_entry);
  260. }
  261. static int check_ipv4_packet (uint8_t *data, int data_len, struct ipv4_header **out_header, uint8_t **out_payload, int *out_payload_len)
  262. {
  263. ASSERT(data_len >= 0)
  264. // check base header
  265. if (data_len < sizeof(struct ipv4_header)) {
  266. BLog(BLOG_DEBUG, "check ipv4: packet too short (base header)");
  267. return 0;
  268. }
  269. struct ipv4_header *header = (struct ipv4_header *)data;
  270. // check version
  271. if (IPV4_GET_VERSION(*header) != 4) {
  272. BLog(BLOG_DEBUG, "check ipv4: version not 4");
  273. return 0;
  274. }
  275. // check options
  276. int header_len = IPV4_GET_IHL(*header) * 4;
  277. if (header_len < sizeof(struct ipv4_header)) {
  278. BLog(BLOG_DEBUG, "check ipv4: ihl too small");
  279. return 0;
  280. }
  281. if (header_len > data_len) {
  282. BLog(BLOG_DEBUG, "check ipv4: packet too short for ihl");
  283. return 0;
  284. }
  285. // check total length
  286. uint16_t total_length = ntoh16(header->total_length);
  287. if (total_length < header_len) {
  288. BLog(BLOG_DEBUG, "check ipv4: total length too small");
  289. return 0;
  290. }
  291. if (total_length > data_len) {
  292. BLog(BLOG_DEBUG, "check ipv4: total length too large");
  293. return 0;
  294. }
  295. *out_header = header;
  296. *out_payload = data + header_len;
  297. *out_payload_len = total_length - header_len;
  298. return 1;
  299. }
  300. void FrameDecider_Init (FrameDecider *o, int max_peer_macs, int max_peer_groups, btime_t igmp_group_membership_interval, btime_t igmp_last_member_query_time, BReactor *reactor)
  301. {
  302. ASSERT(max_peer_macs > 0)
  303. ASSERT(max_peer_groups > 0)
  304. // init arguments
  305. o->max_peer_macs = max_peer_macs;
  306. o->max_peer_groups = max_peer_groups;
  307. o->igmp_group_membership_interval = igmp_group_membership_interval;
  308. o->igmp_last_member_query_time = igmp_last_member_query_time;
  309. o->reactor = reactor;
  310. // init peers list
  311. LinkedList2_Init(&o->peers_list);
  312. // init MAC tree
  313. BAVL_Init(&o->macs_tree, OFFSET_DIFF(struct _FrameDecider_mac_entry, mac, tree_node), (BAVL_comparator)mac_comparator, NULL);
  314. // init multicast tree
  315. BAVL_Init(&o->multicast_tree, OFFSET_DIFF(struct _FrameDecider_group_entry, master.sig, master.tree_node), (BAVL_comparator)uint32_comparator, NULL);
  316. // init decide state
  317. o->decide_state = DECIDE_STATE_NONE;
  318. DebugObject_Init(&o->d_obj);
  319. }
  320. void FrameDecider_Free (FrameDecider *o)
  321. {
  322. ASSERT(BAVL_IsEmpty(&o->multicast_tree))
  323. ASSERT(BAVL_IsEmpty(&o->macs_tree))
  324. ASSERT(LinkedList2_IsEmpty(&o->peers_list))
  325. DebugObject_Free(&o->d_obj);
  326. }
  327. void FrameDecider_AnalyzeAndDecide (FrameDecider *o, uint8_t *frame, int frame_len)
  328. {
  329. ASSERT(frame_len >= 0)
  330. DebugObject_Access(&o->d_obj);
  331. // reset decide state
  332. switch (o->decide_state) {
  333. case DECIDE_STATE_NONE:
  334. break;
  335. case DECIDE_STATE_UNICAST:
  336. break;
  337. case DECIDE_STATE_FLOOD:
  338. LinkedList2Iterator_Free(&o->decide_flood_it);
  339. break;
  340. case DECIDE_STATE_MULTICAST:
  341. LinkedList3Iterator_Free(&o->decide_multicast_it);
  342. return;
  343. default:
  344. ASSERT(0);
  345. }
  346. o->decide_state = DECIDE_STATE_NONE;
  347. // analyze frame
  348. uint8_t *pos = frame;
  349. int len = frame_len;
  350. if (len < sizeof(struct ethernet_header)) {
  351. return;
  352. }
  353. struct ethernet_header *eh = (struct ethernet_header *)pos;
  354. pos += sizeof(struct ethernet_header);
  355. len -= sizeof(struct ethernet_header);
  356. int is_igmp = 0;
  357. switch (ntoh16(eh->type)) {
  358. case ETHERTYPE_IPV4: {
  359. // check IPv4 header
  360. struct ipv4_header *ipv4_header;
  361. if (!check_ipv4_packet(pos, len, &ipv4_header, &pos, &len)) {
  362. BLog(BLOG_INFO, "decide: wrong IP packet");
  363. goto out;
  364. }
  365. // check if it's IGMP
  366. if (ntoh8(ipv4_header->protocol) != IPV4_PROTOCOL_IGMP) {
  367. goto out;
  368. }
  369. // remember that it's IGMP; we have to flood IGMP frames
  370. is_igmp = 1;
  371. // check IGMP header
  372. if (len < sizeof(struct igmp_base)) {
  373. BLog(BLOG_INFO, "decide: IGMP: short packet");
  374. goto out;
  375. }
  376. struct igmp_base *igmp_base = (struct igmp_base *)pos;
  377. pos += sizeof(struct igmp_base);
  378. len -= sizeof(struct igmp_base);
  379. switch (ntoh8(igmp_base->type)) {
  380. case IGMP_TYPE_MEMBERSHIP_QUERY: {
  381. if (len == sizeof(struct igmp_v2_extra) && ntoh8(igmp_base->max_resp_code) != 0) {
  382. // V2 query
  383. struct igmp_v2_extra *query = (struct igmp_v2_extra *)pos;
  384. pos += sizeof(struct igmp_v2_extra);
  385. len -= sizeof(struct igmp_v2_extra);
  386. if (ntoh32(query->group) != 0) {
  387. // got a Group-Specific Query, lower group timers to LMQT
  388. lower_group_timers_to_lmqt(o, query->group);
  389. }
  390. }
  391. else if (len >= sizeof(struct igmp_v3_query_extra)) {
  392. // V3 query
  393. struct igmp_v3_query_extra *query = (struct igmp_v3_query_extra *)pos;
  394. pos += sizeof(struct igmp_v3_query_extra);
  395. len -= sizeof(struct igmp_v3_query_extra);
  396. // iterate sources
  397. uint16_t num_sources = ntoh16(query->number_of_sources);
  398. int i;
  399. for (i = 0; i < num_sources; i++) {
  400. // check source
  401. if (len < sizeof(struct igmp_source)) {
  402. BLog(BLOG_NOTICE, "decide: IGMP: short source");
  403. goto out;
  404. }
  405. pos += sizeof(struct igmp_source);
  406. len -= sizeof(struct igmp_source);
  407. }
  408. if (ntoh32(query->group) != 0 && num_sources == 0) {
  409. // got a Group-Specific Query, lower group timers to LMQT
  410. lower_group_timers_to_lmqt(o, query->group);
  411. }
  412. }
  413. } break;
  414. }
  415. } break;
  416. }
  417. out:;
  418. const uint8_t broadcast_mac[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
  419. const uint8_t multicast_mac_header[] = {0x01, 0x00, 0x5e};
  420. // if it's broadcast or IGMP, flood it
  421. if (is_igmp || !memcmp(eh->dest, broadcast_mac, sizeof(broadcast_mac))) {
  422. o->decide_state = DECIDE_STATE_FLOOD;
  423. LinkedList2Iterator_InitForward(&o->decide_flood_it, &o->peers_list);
  424. return;
  425. }
  426. // if it's multicast, forward to all peers with the given sig
  427. if (!memcmp(eh->dest, multicast_mac_header, sizeof(multicast_mac_header))) {
  428. // extract group's sig from destination MAC
  429. uint32_t sig = compute_sig_for_mac(eh->dest);
  430. // look up the sig in multicast tree
  431. BAVLNode *node;
  432. if (node = BAVL_LookupExact(&o->multicast_tree, &sig)) {
  433. struct _FrameDecider_group_entry *master = UPPER_OBJECT(node, struct _FrameDecider_group_entry, master.tree_node);
  434. ASSERT(master->is_master)
  435. o->decide_state = DECIDE_STATE_MULTICAST;
  436. LinkedList3Iterator_Init(&o->decide_multicast_it, LinkedList3Node_First(&master->sig_list_node), 1);
  437. }
  438. return;
  439. }
  440. // look for MAC entry
  441. BAVLNode *tree_node;
  442. if (tree_node = BAVL_LookupExact(&o->macs_tree, eh->dest)) {
  443. struct _FrameDecider_mac_entry *entry = UPPER_OBJECT(tree_node, struct _FrameDecider_mac_entry, tree_node);
  444. o->decide_state = DECIDE_STATE_UNICAST;
  445. o->decide_unicast_peer = entry->peer;
  446. return;
  447. }
  448. // unknown destination MAC, flood
  449. o->decide_state = DECIDE_STATE_FLOOD;
  450. LinkedList2Iterator_InitForward(&o->decide_flood_it, &o->peers_list);
  451. return;
  452. }
  453. FrameDeciderPeer * FrameDecider_NextDestination (FrameDecider *o)
  454. {
  455. DebugObject_Access(&o->d_obj);
  456. switch (o->decide_state) {
  457. case DECIDE_STATE_NONE: {
  458. return NULL;
  459. } break;
  460. case DECIDE_STATE_UNICAST: {
  461. o->decide_state = DECIDE_STATE_NONE;
  462. return o->decide_unicast_peer;
  463. } break;
  464. case DECIDE_STATE_FLOOD: {
  465. LinkedList2Node *list_node = LinkedList2Iterator_Next(&o->decide_flood_it);
  466. if (!list_node) {
  467. o->decide_state = DECIDE_STATE_NONE;
  468. return NULL;
  469. }
  470. FrameDeciderPeer *peer = UPPER_OBJECT(list_node, FrameDeciderPeer, list_node);
  471. return peer;
  472. } break;
  473. case DECIDE_STATE_MULTICAST: {
  474. LinkedList3Node *list_node = LinkedList3Iterator_Next(&o->decide_multicast_it);
  475. if (!list_node) {
  476. o->decide_state = DECIDE_STATE_NONE;
  477. return NULL;
  478. }
  479. struct _FrameDecider_group_entry *group_entry = UPPER_OBJECT(list_node, struct _FrameDecider_group_entry, sig_list_node);
  480. return group_entry->peer;
  481. } break;
  482. default:
  483. ASSERT(0);
  484. return NULL;
  485. }
  486. }
  487. int FrameDeciderPeer_Init (FrameDeciderPeer *o, FrameDecider *d)
  488. {
  489. // init arguments
  490. o->d = d;
  491. // allocate MAC entries
  492. if (!(o->mac_entries = BAllocArray(d->max_peer_macs, sizeof(struct _FrameDecider_mac_entry)))) {
  493. DEBUG("failed to allocate MAC entries");
  494. goto fail0;
  495. }
  496. // allocate group entries
  497. if (!(o->group_entries = BAllocArray(d->max_peer_groups, sizeof(struct _FrameDecider_group_entry)))) {
  498. DEBUG("failed to allocate group entries");
  499. goto fail1;
  500. }
  501. // insert to peers list
  502. LinkedList2_Append(&d->peers_list, &o->list_node);
  503. // init MAC entry lists
  504. LinkedList2_Init(&o->mac_entries_free);
  505. LinkedList2_Init(&o->mac_entries_used);
  506. // initialize MAC entries
  507. for (int i = 0; i < d->max_peer_macs; i++) {
  508. struct _FrameDecider_mac_entry *entry = &o->mac_entries[i];
  509. // set peer
  510. entry->peer = o;
  511. // insert to free list
  512. LinkedList2_Append(&o->mac_entries_free, &entry->list_node);
  513. }
  514. // init group entry lists
  515. LinkedList2_Init(&o->group_entries_free);
  516. LinkedList2_Init(&o->group_entries_used);
  517. // initialize group entries
  518. for (int i = 0; i < d->max_peer_groups; i++) {
  519. struct _FrameDecider_group_entry *entry = &o->group_entries[i];
  520. // set peer
  521. entry->peer = o;
  522. // insert to free list
  523. LinkedList2_Append(&o->group_entries_free, &entry->list_node);
  524. // init timer
  525. BTimer_Init(&entry->timer, 0, (BTimer_handler)group_entry_timer_handler, entry);
  526. }
  527. // initialize groups tree
  528. BAVL_Init(&o->groups_tree, OFFSET_DIFF(struct _FrameDecider_group_entry, group, tree_node), (BAVL_comparator)uint32_comparator, NULL);
  529. DebugObject_Init(&o->d_obj);
  530. return 1;
  531. fail1:
  532. BFree(o->mac_entries);
  533. fail0:
  534. return 0;
  535. }
  536. void FrameDeciderPeer_Free (FrameDeciderPeer *o)
  537. {
  538. DebugObject_Free(&o->d_obj);
  539. FrameDecider *d = o->d;
  540. // remove decide unicast reference
  541. if (d->decide_state == DECIDE_STATE_UNICAST && d->decide_unicast_peer == o) {
  542. d->decide_state = DECIDE_STATE_NONE;
  543. }
  544. LinkedList2Iterator it;
  545. LinkedList2Node *node;
  546. // free group entries
  547. LinkedList2Iterator_InitForward(&it, &o->group_entries_used);
  548. while (node = LinkedList2Iterator_Next(&it)) {
  549. struct _FrameDecider_group_entry *entry = UPPER_OBJECT(node, struct _FrameDecider_group_entry, list_node);
  550. // remove from multicast
  551. remove_from_multicast(d, entry);
  552. // stop timer
  553. BReactor_RemoveTimer(d->reactor, &entry->timer);
  554. }
  555. // remove used MAC entries from tree
  556. LinkedList2Iterator_InitForward(&it, &o->mac_entries_used);
  557. while (node = LinkedList2Iterator_Next(&it)) {
  558. struct _FrameDecider_mac_entry *entry = UPPER_OBJECT(node, struct _FrameDecider_mac_entry, list_node);
  559. // remove from tree
  560. BAVL_Remove(&d->macs_tree, &entry->tree_node);
  561. }
  562. // remove from peers list
  563. LinkedList2_Remove(&d->peers_list, &o->list_node);
  564. // free group entries
  565. BFree(o->group_entries);
  566. // free MAC entries
  567. BFree(o->mac_entries);
  568. }
  569. void FrameDeciderPeer_Analyze (FrameDeciderPeer *o, uint8_t *frame, int frame_len)
  570. {
  571. ASSERT(frame_len >= 0)
  572. DebugObject_Access(&o->d_obj);
  573. uint8_t *pos = frame;
  574. int len = frame_len;
  575. if (len < sizeof(struct ethernet_header)) {
  576. goto out;
  577. }
  578. struct ethernet_header *eh = (struct ethernet_header *)pos;
  579. pos += sizeof(struct ethernet_header);
  580. len -= sizeof(struct ethernet_header);
  581. // register source MAC address with this peer
  582. add_mac_to_peer(o, eh->source);
  583. switch (ntoh16(eh->type)) {
  584. case ETHERTYPE_IPV4: {
  585. // check IPv4 header
  586. struct ipv4_header *ipv4_header;
  587. if (!check_ipv4_packet(pos, len, &ipv4_header, &pos, &len)) {
  588. BLog(BLOG_INFO, "analyze: wrong IP packet");
  589. goto out;
  590. }
  591. // check if it's IGMP
  592. if (ntoh8(ipv4_header->protocol) != IPV4_PROTOCOL_IGMP) {
  593. goto out;
  594. }
  595. // check IGMP header
  596. if (len < sizeof(struct igmp_base)) {
  597. BLog(BLOG_INFO, "analyze: IGMP: short packet");
  598. goto out;
  599. }
  600. struct igmp_base *igmp_base = (struct igmp_base *)pos;
  601. pos += sizeof(struct igmp_base);
  602. len -= sizeof(struct igmp_base);
  603. switch (ntoh8(igmp_base->type)) {
  604. case IGMP_TYPE_V2_MEMBERSHIP_REPORT: {
  605. // check extra
  606. if (len < sizeof(struct igmp_v2_extra)) {
  607. BLog(BLOG_INFO, "analyze: IGMP: short v2 report");
  608. goto out;
  609. }
  610. struct igmp_v2_extra *report = (struct igmp_v2_extra *)pos;
  611. pos += sizeof(struct igmp_v2_extra);
  612. len -= sizeof(struct igmp_v2_extra);
  613. // add to group
  614. add_group_to_peer(o, report->group);
  615. } break;
  616. case IGMP_TYPE_V3_MEMBERSHIP_REPORT: {
  617. // check extra
  618. if (len < sizeof(struct igmp_v3_report_extra)) {
  619. BLog(BLOG_INFO, "analyze: IGMP: short v3 report");
  620. goto out;
  621. }
  622. struct igmp_v3_report_extra *report = (struct igmp_v3_report_extra *)pos;
  623. pos += sizeof(struct igmp_v3_report_extra);
  624. len -= sizeof(struct igmp_v3_report_extra);
  625. // iterate records
  626. uint16_t num_records = ntoh16(report->number_of_group_records);
  627. for (int i = 0; i < num_records; i++) {
  628. // check record
  629. if (len < sizeof(struct igmp_v3_report_record)) {
  630. BLog(BLOG_INFO, "analyze: IGMP: short record header");
  631. goto out;
  632. }
  633. struct igmp_v3_report_record *record = (struct igmp_v3_report_record *)pos;
  634. pos += sizeof(struct igmp_v3_report_record);
  635. len -= sizeof(struct igmp_v3_report_record);
  636. // iterate sources
  637. uint16_t num_sources = ntoh16(record->number_of_sources);
  638. int j;
  639. for (j = 0; j < num_sources; j++) {
  640. // check source
  641. if (len < sizeof(struct igmp_source)) {
  642. BLog(BLOG_INFO, "analyze: IGMP: short source");
  643. goto out;
  644. }
  645. struct igmp_source *source = (struct igmp_source *)pos;
  646. pos += sizeof(struct igmp_source);
  647. len -= sizeof(struct igmp_source);
  648. }
  649. // check aux data
  650. uint16_t aux_len = ntoh16(record->aux_data_len);
  651. if (len < aux_len) {
  652. BLog(BLOG_INFO, "analyze: IGMP: short record aux data");
  653. goto out;
  654. }
  655. pos += aux_len;
  656. len -= aux_len;
  657. switch (record->type) {
  658. case IGMP_RECORD_TYPE_MODE_IS_INCLUDE:
  659. case IGMP_RECORD_TYPE_CHANGE_TO_INCLUDE_MODE:
  660. if (num_sources != 0) {
  661. add_group_to_peer(o, record->group);
  662. }
  663. break;
  664. case IGMP_RECORD_TYPE_MODE_IS_EXCLUDE:
  665. case IGMP_RECORD_TYPE_CHANGE_TO_EXCLUDE_MODE:
  666. add_group_to_peer(o, record->group);
  667. break;
  668. }
  669. }
  670. } break;
  671. }
  672. } break;
  673. }
  674. out:;
  675. }