UniRec  2.9.3
ip_prefix_search.c
Go to the documentation of this file.
1 
10 /*
11  * Copyright (C) 2013,2014 CESNET
12  *
13  * LICENSE TERMS
14  *
15  * Redistribution and use in source and binary forms, with or without
16  * modification, are permitted provided that the following conditions
17  * are met:
18  * 1. Redistributions of source code must retain the above copyright
19  * notice, this list of conditions and the following disclaimer.
20  * 2. Redistributions in binary form must reproduce the above copyright
21  * notice, this list of conditions and the following disclaimer in
22  * the documentation and/or other materials provided with the
23  * distribution.
24  * 3. Neither the name of the Company nor the names of its contributors
25  * may be used to endorse or promote products derived from this
26  * software without specific prior written permission.
27  *
28  * ALTERNATIVELY, provided that this notice is retained in full, this
29  * product may be distributed under the terms of the GNU General Public
30  * License (GPL) version 2 or later, in which case the provisions
31  * of the GPL apply INSTEAD OF those given above.
32  *
33  * This software is provided ``as is'', and any express or implied
34  * warranties, including, but not limited to, the implied warranties of
35  * merchantability and fitness for a particular purpose are disclaimed.
36  * In no event shall the company or contributors be liable for any
37  * direct, indirect, incidental, special, exemplary, or consequential
38  * damages (including, but not limited to, procurement of substitute
39  * goods or services; loss of use, data, or profits; or business
40  * interruption) however caused and on any theory of liability, whether
41  * in contract, strict liability, or tort (including negligence or
42  * otherwise) arising in any way out of the use of this software, even
43  * if advised of the possibility of such damage.
44  *
45  */
46 
47 #include <stdio.h>
48 #include <stdint.h>
49 #include <stdlib.h>
50 #include "ip_prefix_search.h"
51 #include "ipps_internal.h"
52 
58 uint8_t bit_endian_swap(uint8_t in)
59 {
60  in = (in & 0xF0) >> 4 | (in & 0x0F) << 4;
61  in = (in & 0xCC) >> 2 | (in & 0x33) << 2;
62  in = (in & 0xAA) >> 1 | (in & 0x55) << 1;
63  return in;
64 }
65 
73 {
74  int i, j;
75  uint32_t **net_mask_array = malloc(129 * sizeof(uint32_t *));
76  if (net_mask_array == NULL) {
77  return NULL;
78  }
79 
80  for (i = 0; i < 129; i++) {
81  net_mask_array[i] = malloc(4 * sizeof(uint32_t));
82  if (net_mask_array[i] == NULL) {
83  for (int k = 0; k < i; k++) {
84  free(net_mask_array[i]);
85  }
86  return NULL;
87  }
88  // Fill every word of IPv6 address
89  net_mask_array[i][0] = 0xFFFFFFFF >> (i == 0 || i >= 32 ? 0 : 32 - i);
90  net_mask_array[i][1] = i <= 32 ? 0 : 0xFFFFFFFF >> (i >= 64 ? 0 : 64 - i);
91  net_mask_array[i][2] = i <= 64 ? 0 : 0xFFFFFFFF >> (i >= 96 ? 0 : 96 - i);
92  net_mask_array[i][3] = i <= 96 ? 0 : 0xFFFFFFFF >> (128 - i);
93 
94  // Swap bits in every byte for compatibility with ip_addr_t stucture
95  for (j = 0; j < 4; ++j) {
96  net_mask_array[i][j] = (bit_endian_swap((net_mask_array[i][j] & 0x000000FF) >> 0) << 0) |
97  (bit_endian_swap((net_mask_array[i][j] & 0x0000FF00) >> 8) << 8) |
98  (bit_endian_swap((net_mask_array[i][j] & 0x00FF0000) >> 16) << 16) |
99  (bit_endian_swap((net_mask_array[i][j] & 0xFF000000) >> 24) << 24);
100  }
101  }
102 
103  return net_mask_array;
104 }
105 
113 void destroy_ip_v6_net_mask_array(uint32_t **net_mask_array)
114 {
115  int index;
116  for (index = 0; index < 129; index++) {
117  free(net_mask_array[index]);
118  }
119  free(net_mask_array);
120 }
121 
129 int cmp_net_v4(const void *v1, const void *v2)
130 {
131  const ipps_network_t *n1 = *(ipps_network_t **)v1;
132  const ipps_network_t *n2 = *(ipps_network_t **)v2;
133 
134  int ip_cmp_result;
135 
136  ip_cmp_result = memcmp(&n1->addr.ui32[2], &n2->addr.ui32[2], 4);
137  /* if they are equal - lower mask (greater network) first */
138  if (ip_cmp_result == 0) {
139  return memcmp(&n1->mask, &n2->mask, 4);
140  } else {
141  return ip_cmp_result;
142  }
143 }
144 
152 int cmp_net_v6(const void *v1, const void *v2)
153 {
154  const ipps_network_t *n1 = *(ipps_network_t **)v1;
155  const ipps_network_t *n2 = *(ipps_network_t **)v2;
156 
157  int ip_cmp_result;
158 
159  ip_cmp_result = memcmp(&n1->addr.ui8, &n2->addr.ui8, 16);
160  /* if they are equal - lower mask (greater network) first */
161  if (ip_cmp_result == 0) {
162  return memcmp(&n1->mask, &n2->mask, 4);
163  } else {
164  return ip_cmp_result;
165  }
166 }
167 
177 void mask_ipv6(ip_addr_t *ip, uint32_t mask, ip_addr_t *masked_ipv6, uint32_t **net_mask_array)
178 {
179  int i;
180  for (i = 0; i < 4; i++) {
181  masked_ipv6->ui32[i] = ip->ui32[i] & net_mask_array[mask][i];
182  }
183 }
184 
195  uint32_t **net_mask_array)
196 {
197  inter->data_cnt = 0;
198  inter->data_array = NULL;
199 
200  /* Fill network address */
201  memcpy(&inter->low_ip, &net->addr, 16);
202 
203  /* Fill broadcast address */
204  if (!ip_is6(&net->addr)) {
205  // IPv4
206  inter->high_ip.ui64[0] = 0;
207  inter->high_ip.ui32[2] = net->addr.ui32[2] | ( ~ net_mask_array[net->mask][0]);
208  inter->high_ip.ui32[3] = 0xffffffff;
209  } else {
210  // IPv6
211  int i;
212  for (i = 0; i < 4; i++) {
213  inter->high_ip.ui32[i] = net->addr.ui32[i] | ( ~ net_mask_array[net->mask][i]);
214  }
215  }
216 }
217 
225  ipps_interval_node_t *new_interval(const ip_addr_t *low_ip, const ip_addr_t *high_ip)
226 {
227 
228  ipps_interval_node_t * new_node = malloc(sizeof(ipps_interval_node_t));
229  if (new_node == NULL) {
230  fprintf(stderr, "ERROR allocating memory for network interval node\n");
231  return NULL;
232  }
233 
234  new_node->interval = malloc(sizeof(ipps_interval_t));
235  if (new_node->interval == NULL) {
236  fprintf(stderr, "ERROR allocating memory for network interval\n");
237  free(new_node);
238  return NULL;
239  }
240  new_node->next = NULL;
241 
242  /* Initialize struct members */
243  memcpy(&new_node->interval->low_ip, low_ip, 16);
244  memcpy(&new_node->interval->high_ip, high_ip, 16);
245  new_node->interval->data_cnt = 0;
246  new_node->interval->array_len = DATASLOTS;
247  new_node->interval->data_array = malloc(DATASLOTS * sizeof(void *));
248  if (new_node->interval->data_array == NULL) {
249  fprintf(stderr, "ERROR allocating memory for data pointers\n");
250  free(new_node->interval);
251  free(new_node);
252  return NULL;
253  }
254 
255  return new_node;
256 }
257 
268  const ip_addr_t *low_ip, const ip_addr_t *high_ip)
269 {
270  ipps_interval_node_t *new_interval_node = new_interval(low_ip, high_ip);
271  if (new_interval_node == NULL) {
272  return NULL;
273  }
274 
275  /* Post Insert */
276  new_interval_node->next = position->next;
277  position->next = new_interval_node;
278 
279  return new_interval_node;
280 }
281 
289 void ip_dec(const ip_addr_t *ip, ip_addr_t *ip_dec)
290 {
291  if (ip_is6(ip)) {
292  memcpy(ip_dec, ip, 16);
293 
294  uint32_t tmp = 0xffffffff;
295  int i;
296  for (i = 3; i >=0; i--) {
297  ip_dec->ui32[i] = htonl(ntohl(ip->ui32[i]) - 1);
298  if (ip_dec->ui32[i] != tmp) {
299  break;
300  }
301  }
302  } else {
303  ip_dec->ui64[0] = 0;
304  ip_dec->ui32[2] = htonl(ntohl(ip->ui32[2]) - 1);
305  ip_dec->ui32[3] = 0xffffffff;
306  }
307 }
308 
316 void ip_inc(const ip_addr_t *ip, ip_addr_t *ip_inc)
317 {
318  if (ip_is6(ip)) {
319  memcpy(ip_inc, ip, 16);
320 
321  uint32_t tmp = 0xffffffff;
322  int i;
323  for (i = 3; i >= 0; i--) {
324  ip_inc->ui32[i] = htonl(ntohl(ip->ui32[i]) + 1);
325  if (ip->ui32[i] < tmp) {
326  break;
327  }
328  }
329  } else {
330  ip_inc->ui64[0] = 0;
331  ip_inc->ui32[2] = htonl(ntohl(ip->ui32[2]) + 1);
332  ip_inc->ui32[3] = 0xffffffff;
333  }
334 }
335 
342 int ipps_destroy(ipps_context_t *prefix_context)
343 {
344  int i;
345  void **data_collector; // Array with freed memory pointers
346  uint32_t data_collector_cnt = 0; // Number of pointers in 'data_collector'
347 
348  if (prefix_context == NULL) {
349  fprintf(stderr, "ERROR NULL pointer passed to ipps_destroy\n");
350  return 1;
351  }
352 
353  data_collector = malloc(COLLECTORSLOTS * sizeof(void *));
354  if (data_collector == NULL) {
355  fprintf(stderr, "ERROR allocating memory for freed data collector\n");
356  return 1;
357  }
358 
359  // Dealloc all IPv4 data and intervals
360  for (i = 0; i < prefix_context->v4_count; ++i) {
361  if (free_data(&prefix_context->v4_prefix_intervals[i], &data_collector, &data_collector_cnt)) {
362  return 1;
363  }
364  }
365 
366  // Dealloc all IPv6 data and intervals
367  data_collector_cnt = 0;
368  for (i = 0; i < prefix_context->v6_count; ++i) {
369  if (free_data(&prefix_context->v6_prefix_intervals[i], &data_collector, &data_collector_cnt)) {
370  return 1;
371  }
372  }
373 
374  free(data_collector);
375  free(prefix_context->v4_prefix_intervals);
376  free(prefix_context->v6_prefix_intervals);
377  free(prefix_context);
378  return 0;
379 }
380 
386 {
387  ipps_context_t *prefix_context = malloc(sizeof(ipps_context_t));
388  if (prefix_context == NULL) {
389  fprintf(stderr, "ERROR allocating memory for network mask array\n");
390  return NULL;
391  }
392 
393  prefix_context->v4_count = 0;
394  prefix_context->v6_count = 0;
395  prefix_context->v4_prefix_intervals = NULL;
396  prefix_context->v6_prefix_intervals = NULL;
397 
398  return prefix_context;
399 }
400 
410 {
411  int index;
412  ip_addr_t *masked_ip;
413 
414  if (network_list == NULL) {
415  fprintf(stderr, "ERROR Network list is not initialized");
416  return NULL;
417  }
418 
419  if (network_list->net_count <= 0) {
420  fprintf(stderr, "ERROR Network lists are empty, nothing to do");
421  return NULL;
422  }
423 
424  // Create new interval_search_context
425  ipps_context_t *prefix_context = new_context();
426  if (prefix_context == NULL) {
427  return NULL;
428  }
429 
430  // Create and fill net mask array - for masking IP
431  uint32_t **net_mask_array = create_ip_v6_net_mask_array();
432  if (net_mask_array == NULL) {
433  fprintf(stderr, "ERROR allocating memory for network mask array\n");
434  return NULL;
435  }
436 
437  ipps_network_t *current_net; // Currently precessed network
438  void *tmp;
439 
440  ipps_network_t **networks_v6 = NULL; // Pointers to ipv6 networks
441  ipps_network_t **networks_v4 = NULL; // Pointers to ipv4 networks
442 
443 
444  uint32_t i_v6 = 0; // Number of ipv6 networks
445  uint32_t i_v4 = 0; // Number of ipv4 networks
446 
447  size_t i_v6_alloc = NETWORKSLOTS; // Number of available network pointers
448  size_t i_v4_alloc = NETWORKSLOTS; // Number of available network pointers
449 
450  // Allocate memory for network array
451  if ((networks_v4 = malloc(i_v4_alloc * sizeof(ipps_network_t *))) == NULL ||
452  (networks_v6 = malloc(i_v6_alloc * sizeof(ipps_network_t *))) == NULL) {
453  free(networks_v4);
454  free(networks_v6);
455  ipps_destroy(prefix_context);
456  destroy_ip_v6_net_mask_array(net_mask_array);
457  fprintf(stderr, "ERROR allocating sorted network structures\n");
458  return NULL;
459  }
460 
461  // For each network in array - mask ip address and split to ipv4 or ipv6 network
462  for (index = 0; index < network_list->net_count; ++index)
463  {
464  current_net = &network_list->networks[index];
465  if (ip_is6(&current_net->addr)) {
466  masked_ip = &current_net->addr;
467  mask_ipv6(&current_net->addr, current_net->mask, masked_ip, net_mask_array);
468 
469  i_v6++;
470  if (i_v6_alloc < i_v6) {
471  i_v6_alloc *= 2;
472  tmp = realloc(networks_v6, i_v6_alloc * sizeof(ipps_network_t *));
473  if (tmp == NULL) {
474  fprintf(stderr, "ERROR allocating memory for ipv6 network collector\n");
475  ipps_destroy(prefix_context);
476  destroy_ip_v6_net_mask_array(net_mask_array);
477  free(networks_v4);
478  free(networks_v6);
479  return NULL;
480  }
481  networks_v6 = tmp;
482 
483  }
484  networks_v6[i_v6-1] = current_net;
485  } else {
486  masked_ip = &current_net->addr;
487  masked_ip->ui32[2] = masked_ip->ui32[2] & net_mask_array[current_net->mask][0];
488 
489  i_v4++;
490  if (i_v4_alloc < i_v4) {
491  i_v4_alloc *= 2;
492  tmp = realloc(networks_v4, i_v4_alloc * sizeof(ipps_network_t *));
493  if (tmp == NULL) {
494  fprintf(stderr, "ERROR allocating memory for ipv6 network collector\n");
495  ipps_destroy(prefix_context);
496  destroy_ip_v6_net_mask_array(net_mask_array);
497  free(networks_v4);
498  free(networks_v6);
499  return NULL;
500  }
501  networks_v4 = tmp;
502  }
503  networks_v4[i_v4 - 1] = current_net;
504  }
505  }
506 
507  if (i_v4 > 0 && networks_v4[0] != NULL) {
508  qsort(networks_v4, i_v4, sizeof(ipps_network_t *), cmp_net_v4);
509 
510  // Fill intervals for IPv4 to interval_search_context array, if fail, dealloc and rtrn NULL
511  prefix_context->v4_prefix_intervals = init_context(networks_v4, i_v4,
512  &prefix_context->v4_count, net_mask_array);
513  if (prefix_context->v4_prefix_intervals == NULL) {
514  destroy_ip_v6_net_mask_array(net_mask_array);
515  ipps_destroy(prefix_context);
516  free(networks_v4);
517  free(networks_v6);
518  return NULL;
519  }
520  /************************/
521  }
522  free(networks_v4);
523 
524  if (i_v6 > 0 && networks_v6[0] != NULL) {
525  qsort(networks_v6, i_v6, sizeof(ipps_network_t *), cmp_net_v6);
526 
527  // Fill intervals for IPv6 to interval_search_context array, if fail, dealloc and return NULL
528  prefix_context->v6_prefix_intervals = init_context(networks_v6, i_v6,
529  &prefix_context->v6_count, net_mask_array);
530  if (prefix_context->v6_prefix_intervals == NULL) {
531  destroy_ip_v6_net_mask_array(net_mask_array);
532  ipps_destroy(prefix_context);
533  free(networks_v6);
534  return NULL;
535  }
536 
537  /************************/
538  }
539  free(networks_v6);
540 
541  destroy_ip_v6_net_mask_array(net_mask_array);
542  return prefix_context;
543 }
544 
554 {
555 
556  if (dest->data_cnt + src->data_cnt > dest->array_len) {
557  // no free pointer slots for src data in dest data_array
558  void **tmp;
559  dest->array_len += src->array_len;
560  tmp = realloc (dest->data_array, dest->array_len * sizeof(void *));
561  if (tmp == NULL) {
562  fprintf(stderr, "ERROR allocating memory for network mask array\n");
563  return 1;
564  }
565  dest->data_array = tmp;
566  }
567 
568  memcpy(dest->data_array + dest->data_cnt, src->data_array, src->data_cnt * sizeof(void *));
569  dest->data_cnt += src->data_cnt;
570  return 0;
571 }
572 
583 int add_data(ipps_interval_t *interval, void *data, size_t data_len)
584 {
585  // Alloc new data and copy
586  void *new_data = malloc(data_len);
587  if (new_data == NULL) {
588  fprintf(stderr, "ERROR allocating memory for network mask array\n");
589  return 1;
590  }
591 
592  memcpy(new_data, data, data_len);
593 
594  // Insert new data to interval's data array - first do some space ...
595  interval->data_cnt++;
596  if (interval->data_cnt > interval->array_len) {
597  void **tmp;
598  interval->array_len *= 2;
599  tmp = realloc (interval->data_array, interval->array_len * sizeof(void *));
600  if (tmp == NULL) {
601  fprintf(stderr, "ERROR allocating memory for network mask array\n");
602  free(new_data);
603  return 1;
604  }
605  interval->data_array = tmp;
606  }
607 
608  // ... then push new data to array
609  interval->data_array[interval->data_cnt - 1] = new_data;
610 
611  return 0;
612 }
613 
630 ipps_interval_t *init_context( ipps_network_t **networks, uint32_t network_count,
631  uint32_t *context_counter, uint32_t **net_mask_array)
632 {
633  uint32_t interval_counter = 0;
634 
635  ipps_interval_t current_interval;
636  ipps_interval_node_t *interval_list = NULL;
637 
638  int index = 0;
639 
640  // push first interval from network tree to interval_search_context interval list
641  // compute ip interval - low and high ip addresses
642  fill_interval_by_network(networks[0], &current_interval, net_mask_array);
643  // insert root interval node to list
644  interval_list = new_interval(&current_interval.low_ip, &current_interval.high_ip);
645  if (interval_list == NULL) {
646  return NULL;
647  }
648 
649  if (add_data(interval_list->interval, networks[0]->data, networks[0]->data_len)) {
650  // add data to root node
651  return NULL;
652  }
653  interval_counter++; // number of intervals
654 
655  ipps_interval_node_t *conductor = NULL; // list iterator
656  ipps_interval_node_t *end_of_list = interval_list; // last element in the list
657 
658  int ip_cmp_result; // result of ip comparison
659  ip_addr_t tmp_ip ; // temporary ip addr union for compute first or last IP address of interval
660 
661  conductor = interval_list;
662 
663  // traverse rest of networks tree
664  for (index = 1; index < network_count; ++index) {
665  // fill temporary interval variable by current interval
666  fill_interval_by_network(networks[index], &current_interval, net_mask_array);
667 
668  while (conductor != NULL) {
669  // compare low IP of actual processed interval with high IP all intervals in list
670  ip_cmp_result = ip_cmp( &conductor->interval->high_ip, &current_interval.low_ip);
671 
672  if (ip_cmp_result >= 0) {
673  ip_cmp_result = ip_cmp( &conductor->interval->low_ip, &current_interval.low_ip);
674  if (ip_cmp_result > 0) {
675  // LowI > LowT
676  ip_cmp_result = ip_cmp( &conductor->interval->high_ip, &current_interval.high_ip);
677  if (ip_cmp_result > 0) {
678  // Con <---------->
679  // Cur <---------->
680 
681  fprintf(stderr, "ERROR Inserting to list");
682  destroy_list(interval_list);
683  return NULL;
684  } else if (ip_cmp_result < 0) {
685  // Con <-------->
686  // Cur <------------>
687 
688  fprintf(stderr, "ERROR Inserting to list");
689  destroy_list(interval_list);
690  return NULL;
691  } else {
692  // Con <-------->
693  // Cur <---------->
694 
695  fprintf(stderr, "ERROR Inserting to list");
696  destroy_list(interval_list);
697  return NULL;
698  }
699  } else if (ip_cmp_result < 0) {
700  // LowI < LowT
701  ip_cmp_result = ip_cmp( &conductor->interval->high_ip, &current_interval.high_ip);
702  if (ip_cmp_result > 0) {
703  // Con <---------->
704  // Cur <----->
705 
706  /***********************/
707  /* | | ↓ | |
708  * <----------->
709  * <----->
710  */
711 
712  // Insert new interval to interval tree, conductor post insert
713  if (insert_new_interval(conductor, &current_interval.low_ip,
714  &current_interval.high_ip) == NULL) {
715  destroy_list(interval_list);
716  return NULL;
717  }
718  interval_counter++;
719 
720  // Fill data array in new interval node
721  // Copy original data and add new one from current network
722  if (copy_all_data(conductor->next->interval, conductor->interval)) {
723  destroy_list(interval_list);
724  return NULL;
725  }
726  if (add_data(conductor->next->interval, networks[index]->data, networks[index]->data_len)) {
727  destroy_list(interval_list);
728  return NULL;
729  }
730  /***********************/
731 
732  /***********************/
733  /*
734  * | | |↓ |
735  * Con <----------->
736  * Cur <----->
737  */
738 
739  // Insert new interval to interval list
740  ip_inc(&current_interval.high_ip, &tmp_ip);
741  if (insert_new_interval(conductor->next, &tmp_ip, &conductor->interval->high_ip) == NULL) {
742  destroy_list(interval_list);
743  return NULL;
744  }
745  interval_counter++;
746 
747  if (copy_all_data(conductor->next->next->interval, conductor->interval)) {
748  destroy_list(interval_list);
749  return NULL;
750  }
751 
752  /***********************/
753 
754  /***********************/
755  /* |↓| | |
756  * Con <----------->
757  * Cur <----->
758  */
759 
760  // Modify original interval
761  ip_dec(&current_interval.low_ip, &tmp_ip);
762  memcpy(&conductor->interval->high_ip, &tmp_ip, 16);
763  /***********************/
764 
765  // Modify end of interval list, 2 follower inserted
766  if (end_of_list == conductor) {
767  end_of_list = conductor->next->next;
768  }
769  break;
770  }
771  else if (ip_cmp_result < 0) {
772  // Con <---------->
773  // Cur <----------->
774 
775  fprintf(stderr, "ERROR Inserting to list");
776  destroy_list(interval_list);
777  return NULL;
778  } else {
779  // Con <---------->
780  // Cur <-------->
781 
782  if (insert_new_interval(conductor, &current_interval.low_ip,
783  &conductor->interval->high_ip) == NULL) {
784  destroy_list(interval_list);
785  return NULL;
786  }
787  interval_counter++;
788 
789  if (copy_all_data(conductor->next->interval, conductor->interval)) {
790  destroy_list(interval_list);
791  return NULL;
792  }
793  if (add_data(conductor->next->interval, networks[index]->data,
794  networks[index]->data_len)) {
795  destroy_list(interval_list);
796  return NULL;
797  }
798 
799  if (end_of_list == conductor) {
800  end_of_list = conductor->next;
801  }
802 
803  ip_dec(&current_interval.low_ip, &tmp_ip);
804  memcpy(&conductor->interval->high_ip, &tmp_ip, 16);
805  }
806  } else {
807  ip_cmp_result = ip_cmp( &conductor->interval->high_ip, &current_interval.high_ip);
808  if (ip_cmp_result > 0) {
809  // Con <---------->
810  // Cur <-------->
811 
812  ip_inc(&current_interval.high_ip, &tmp_ip);
813 
814  insert_new_interval(conductor, &tmp_ip, &conductor->interval->high_ip);
815  interval_counter++;
816 
817  copy_all_data(conductor->next->interval, conductor->interval);
818 
819  if (end_of_list == conductor) {
820  end_of_list = conductor->next;
821  }
822  memcpy(&conductor->interval->high_ip, &current_interval.high_ip, 16);
823  if (add_data(conductor->interval, networks[index]->data, networks[index]->data_len)) {
824  destroy_list(interval_list);
825  return NULL;
826  }
827 
828  break;
829  } else if (ip_cmp_result < 0) {
830  // Con <-------->
831  // Cur <---------->
832 
833  fprintf(stderr, "ERROR Inserting to list");
834  destroy_list(interval_list);
835  return NULL;
836  } else {
837  // Con <-------->
838  // Cur <-------->
839 
840  if (add_data(conductor->interval, networks[index]->data,
841  networks[index]->data_len)) {
842  destroy_list(interval_list);
843  return NULL;
844  }
845  }
846  }
847 
848  break;
849  } else if (ip_cmp_result < 0) {
850  // Low ip address of current is higher then high ip of interval list member,
851  // take next interval in list
852  conductor = conductor->next;
853  } else {
854  // Useless branch, memcmp error
855  fprintf(stderr, "ERROR Inserting to list");
856  destroy_list(interval_list);
857  return NULL;
858  }
859  }
860 
861  if (conductor == NULL) {
862  // New interval at the end of list
863  if (insert_new_interval(end_of_list, &current_interval.low_ip, &current_interval.high_ip) == NULL) {
864  destroy_list(interval_list);
865  return NULL;
866  }
867 
868  // Add new data to new end of list
869  end_of_list = end_of_list->next;
870 
871  if (add_data(end_of_list->interval, networks[index]->data, networks[index]->data_len)) {
872  destroy_list(interval_list);
873  return NULL;
874  }
875  interval_counter++;
876 
877  conductor = end_of_list;
878  }
879  }
880 
881  // Alloc new interval array
882  ipps_interval_t *prefix_context = (ipps_interval_t *)malloc(interval_counter * sizeof(ipps_interval_t));
883  if (prefix_context == NULL) {
884  fprintf(stderr, "ERROR allocating memory for prefix interval_search_context\n");
885  destroy_list(interval_list);
886  return NULL;
887  }
888 
889  // Fill interval array
890  // Hard copy intervals from list to array, duplicate data array pointer, free all list node
891  conductor = interval_list;
892  ipps_interval_t *array_iterator = prefix_context;
893  size_t size_of_pref_interval = sizeof(ipps_interval_t);
894  *context_counter = interval_counter;
895 
896  // Iterate whole interval list
897  while (conductor != NULL) {
898  // Copy from list to array
899  memcpy(array_iterator, conductor->interval, size_of_pref_interval);
900  array_iterator++; // next interval in array
901 
902  // Destroy list node, Don't destroy data!
903  interval_list = conductor->next;
904  free(conductor->interval);
905  free(conductor);
906  conductor = interval_list;
907  }
908 
909  return prefix_context;
910 }
911 
924 int ipps_search(ip_addr_t *ip, ipps_context_t *prefix_context, void ***data)
925 {
926  int first, last, middle; // Array indexes
927 
928  ipps_interval_t *interval_array; // Pointer to IPv4 or IPv6 interval array
929  uint8_t *middle_interval; // Pointer to first byte of current middle interval
930 
931  int ip_high_cmp_result; // Result of comparing high IP of 2 intervals
932  int ip_low_cmp_result; // Result of comparing low IP of 2 intervals
933 
934  void *ip_addr_start; // Ptr to first useful byte in ip_addr_t union, etc offset ui32[2] in IPv4
935 
936  size_t ip_addr_len = sizeof(ip_addr_t);
937  size_t addr_cmp_len; // Number of compare bytes, 4 for IPv4, 16 for IPv6 (IP addr size)
938 
939  size_t low_ip_offset; // Offset of 'low_ip' in 'ipps_interval_t' structure
940  size_t high_ip_offset; // Offset of 'high' in 'ipps_interval_t' structure
941 
942  if (ip_is6(ip)) {
943  if (prefix_context->v6_count == 0) {
944  return 0;
945  }
946  first = 0;
947  last = prefix_context->v6_count - 1;
948  middle = (first + last)>>1;
949 
950  interval_array = prefix_context->v6_prefix_intervals;
951 
952  low_ip_offset = 0; // interval->low_ip is first member of struct
953  high_ip_offset = ip_addr_len; // interval->high_ip is second member of struct
954  addr_cmp_len = 16; // Compare length, 16 bytes for IPv6
955  ip_addr_start = (uint8_t *)ip;
956 
957  } else {
958  if (prefix_context->v4_count == 0) {
959  return 0;
960  }
961  first = 0;
962  last = prefix_context->v4_count - 1;
963  middle = (first + last)>>1;
964 
965  interval_array = prefix_context->v4_prefix_intervals;
966 
967  low_ip_offset = 8; // low_ip.ui32[2]
968  high_ip_offset = ip_addr_len + 8; // high.ui32[2]
969  addr_cmp_len = 4; // Compare length, 4 bytes for IPv4
970 
971  ip_addr_start = ((uint8_t *)ip) + 8; // ip.ui32[2]
972  }
973 
974  while (first <= last ) {
975  middle_interval = (uint8_t *)(interval_array + middle);
976 
977  ip_low_cmp_result = memcmp(middle_interval + low_ip_offset, ip_addr_start, addr_cmp_len);
978  ip_high_cmp_result = memcmp(middle_interval + high_ip_offset, ip_addr_start, addr_cmp_len);
979 
980  if (ip_low_cmp_result <= 0 && ip_high_cmp_result >= 0) {
981  *data = ((ipps_interval_t *)middle_interval)->data_array;
982  return ((ipps_interval_t *)middle_interval)->data_cnt;
983  } else if (ip_high_cmp_result > 0) {
984  last = middle - 1;
985  } else {
986  first = middle + 1;
987  }
988  middle = (first + last) >> 1;
989  }
990  return 0;
991 }
992 
1000 int free_data(ipps_interval_t *interval, void ***data_collector, uint32_t *data_coll_cnt )
1001 {
1002  int j,k;
1003  void **tmp;
1004 
1005  for (j = 0; j < interval->data_cnt; ++j) {
1006  for (k = 0; k < *data_coll_cnt; ++k) {
1007  if (interval->data_array[j] == (*data_collector)[k]) {
1008  interval->data_array[j] = NULL; // pointer has been freed
1009  break;
1010  }
1011  }
1012 
1013  if (k == *data_coll_cnt) {
1014  // Data not match in data_collector, add data pointer to collector and free memory
1015  if (*data_coll_cnt >= COLLECTORSLOTS && *data_coll_cnt % COLLECTORSLOTS == 0) {
1016  tmp = realloc(*data_collector, ((*data_coll_cnt) + COLLECTORSLOTS) * sizeof(void *));
1017  if (tmp == NULL) {
1018  fprintf(stderr, "ERROR allocating memory for network mask array\n");
1019  return 1;
1020  }
1021  *data_collector = tmp;
1022  }
1023 
1024  (*data_collector)[*data_coll_cnt] = interval->data_array[j]; // Add pointer to collector
1025  (*data_coll_cnt)++;
1026 
1027  free(interval->data_array[j]); // free data
1028  }
1029  }
1030  free(interval->data_array); // free pointers to data
1031  return 0;
1032 }
1033 
1042 {
1043  void **data_collector; // pointers to freed memory
1044  uint32_t data_collector_cnt = 0;
1045  ipps_interval_node_t *tmp_interval;
1046 
1047  data_collector = malloc(COLLECTORSLOTS * sizeof(void *));
1048  if (data_collector == NULL) {
1049  fprintf(stderr, "ERROR allocating memory for freed data collector\n");
1050  return 1;
1051  }
1052 
1053  while (interval_list != NULL) {
1054  tmp_interval = interval_list;
1055  interval_list = interval_list->next;
1056 
1057  if (free_data(tmp_interval->interval, &data_collector, &data_collector_cnt)) {
1058  return 1;
1059  }
1060  free(tmp_interval->interval);
1061  free(tmp_interval);
1062  }
1063  free(data_collector);
1064 
1065  return 0;
1066 }
#define NETWORKSLOTS
int cmp_net_v4(const void *v1, const void *v2)
void mask_ipv6(ip_addr_t *ip, uint32_t mask, ip_addr_t *masked_ipv6, uint32_t **net_mask_array)
ip_addr_t low_ip
Low IP of interval.
Init context and prefix search.
struct ipps_interval_node * next
Next node in list, NULL if last node in list.
Definition: ipps_internal.h:54
void ** data_array
Array of pointers to data.
uint32_t v4_count
Number of intervals in IPv4 array.
void destroy_ip_v6_net_mask_array(uint32_t **net_mask_array)
uint32_t data_cnt
Number of currently used data pointers in &#39;data_array&#39;.
uint8_t ui8[16]
Definition: ipaddr.h:109
free(rec)
INLINE int ip_cmp(const ip_addr_t *addr1, const ip_addr_t *addr2)
Definition: ipaddr.h:266
ipps_interval_t * v6_prefix_intervals
Pointer to IPv6 intervals array.
ipps_interval_t * v4_prefix_intervals
Pointer to IPv4 intervals array.
ipps_interval_node_t * new_interval(const ip_addr_t *low_ip, const ip_addr_t *high_ip)
ipps_context_t * ipps_init(ipps_network_list_t *network_list)
ipps_interval_t * init_context(ipps_network_t **networks, uint32_t network_count, uint32_t *context_counter, uint32_t **net_mask_array)
#define COLLECTORSLOTS
int copy_all_data(ipps_interval_t *dest, ipps_interval_t *src)
size_t data_len
Number of bytes in &#39;data&#39;.
ipps_network_t * networks
Pointer to networks array.
int add_data(ipps_interval_t *interval, void *data, size_t data_len)
Init context and prefix search - Internal functions and structures.
uint8_t bit_endian_swap(uint8_t in)
int ipps_search(ip_addr_t *ip, ipps_context_t *prefix_context, void ***data)
int ipps_destroy(ipps_context_t *prefix_context)
uint32_t ui32[4]
Definition: ipaddr.h:117
uint32_t v6_count
Number of intervals in IPv6 array.
uint32_t ** create_ip_v6_net_mask_array()
#define DATASLOTS
ipps_interval_t * interval
Pointer to interval structure.
Definition: ipps_internal.h:53
INLINE int ip_is6(const ip_addr_t *addr)
Definition: ipaddr.h:143
void ip_dec(const ip_addr_t *ip, ip_addr_t *ip_dec)
size_t array_len
Allocated size of &#39;data_array&#39; => total available slots.
ipps_context_t * new_context()
ip_addr_t addr
Network IP address.
memcpy(buffer, rec, ur_rec_size(tmplt, rec))
int cmp_net_v6(const void *v1, const void *v2)
uint32_t mask
Network mask, CIDR notation, use for indexing.
uint32_t net_count
Number of networks in &#39;networks&#39; array.
int destroy_list(ipps_interval_node_t *interval_list)
void fill_interval_by_network(const ipps_network_t *net, ipps_interval_t *inter, uint32_t **net_mask_array)
ip_addr_t high_ip
High IP of interval.
ipps_interval_node_t * insert_new_interval(ipps_interval_node_t *position, const ip_addr_t *low_ip, const ip_addr_t *high_ip)
void * data
Pointer to same data.
int free_data(ipps_interval_t *interval, void ***data_collector, uint32_t *data_coll_cnt)
void ip_inc(const ip_addr_t *ip, ip_addr_t *ip_inc)
uint64_t ui64[2]
Definition: ipaddr.h:121