FrameDecider.c 28 KB

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