60 in = (in & 0xF0) >> 4 | (in & 0x0F) << 4;
61 in = (in & 0xCC) >> 2 | (in & 0x33) << 2;
62 in = (in & 0xAA) >> 1 | (in & 0x55) << 1;
75 uint32_t **net_mask_array = malloc(129 *
sizeof(uint32_t *));
76 if (net_mask_array == NULL) {
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]);
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);
95 for (j = 0; j < 4; ++j) {
96 net_mask_array[i][j] = (
bit_endian_swap((net_mask_array[i][j] & 0x000000FF) >> 0) << 0) |
103 return net_mask_array;
116 for (index = 0; index < 129; index++) {
117 free(net_mask_array[index]);
119 free(net_mask_array);
138 if (ip_cmp_result == 0) {
139 return memcmp(&n1->
mask, &n2->
mask, 4);
141 return ip_cmp_result;
161 if (ip_cmp_result == 0) {
162 return memcmp(&n1->
mask, &n2->
mask, 4);
164 return ip_cmp_result;
180 for (i = 0; i < 4; i++) {
181 masked_ipv6->
ui32[i] = ip->
ui32[i] & net_mask_array[mask][i];
195 uint32_t **net_mask_array)
212 for (i = 0; i < 4; i++) {
229 if (new_node == NULL) {
230 fprintf(stderr,
"ERROR allocating memory for network interval node\n");
236 fprintf(stderr,
"ERROR allocating memory for network interval\n");
240 new_node->
next = NULL;
249 fprintf(stderr,
"ERROR allocating memory for data pointers\n");
271 if (new_interval_node == NULL) {
276 new_interval_node->
next = position->
next;
277 position->
next = new_interval_node;
279 return new_interval_node;
294 uint32_t tmp = 0xffffffff;
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) {
304 ip_dec->ui32[2] = htonl(ntohl(ip->
ui32[2]) - 1);
305 ip_dec->ui32[3] = 0xffffffff;
321 uint32_t tmp = 0xffffffff;
323 for (i = 3; i >= 0; i--) {
324 ip_inc->ui32[i] = htonl(ntohl(ip->
ui32[i]) + 1);
325 if (ip->
ui32[i] < tmp) {
331 ip_inc->ui32[2] = htonl(ntohl(ip->
ui32[2]) + 1);
332 ip_inc->ui32[3] = 0xffffffff;
345 void **data_collector;
346 uint32_t data_collector_cnt = 0;
348 if (prefix_context == NULL) {
349 fprintf(stderr,
"ERROR NULL pointer passed to ipps_destroy\n");
354 if (data_collector == NULL) {
355 fprintf(stderr,
"ERROR allocating memory for freed data collector\n");
360 for (i = 0; i < prefix_context->
v4_count; ++i) {
367 data_collector_cnt = 0;
368 for (i = 0; i < prefix_context->
v6_count; ++i) {
374 free(data_collector);
377 free(prefix_context);
388 if (prefix_context == NULL) {
389 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
398 return prefix_context;
414 if (network_list == NULL) {
415 fprintf(stderr,
"ERROR Network list is not initialized");
420 fprintf(stderr,
"ERROR Network lists are empty, nothing to do");
426 if (prefix_context == NULL) {
432 if (net_mask_array == NULL) {
433 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
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) {
457 fprintf(stderr,
"ERROR allocating sorted network structures\n");
462 for (index = 0; index < network_list->
net_count; ++index)
464 current_net = &network_list->
networks[index];
466 masked_ip = ¤t_net->
addr;
470 if (i_v6_alloc < i_v6) {
472 tmp = realloc(networks_v6, i_v6_alloc *
sizeof(
ipps_network_t *));
474 fprintf(stderr,
"ERROR allocating memory for ipv6 network collector\n");
484 networks_v6[i_v6-1] = current_net;
486 masked_ip = ¤t_net->
addr;
487 masked_ip->
ui32[2] = masked_ip->
ui32[2] & net_mask_array[current_net->
mask][0];
490 if (i_v4_alloc < i_v4) {
492 tmp = realloc(networks_v4, i_v4_alloc *
sizeof(
ipps_network_t *));
494 fprintf(stderr,
"ERROR allocating memory for ipv6 network collector\n");
503 networks_v4[i_v4 - 1] = current_net;
507 if (i_v4 > 0 && networks_v4[0] != NULL) {
512 &prefix_context->
v4_count, net_mask_array);
524 if (i_v6 > 0 && networks_v6[0] != NULL) {
529 &prefix_context->
v6_count, net_mask_array);
542 return prefix_context;
562 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
586 void *new_data = malloc(data_len);
587 if (new_data == NULL) {
588 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
592 memcpy(new_data, data, data_len);
601 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
631 uint32_t *context_counter, uint32_t **net_mask_array)
633 uint32_t interval_counter = 0;
645 if (interval_list == NULL) {
661 conductor = interval_list;
664 for (index = 1; index < network_count; ++index) {
668 while (conductor != NULL) {
672 if (ip_cmp_result >= 0) {
674 if (ip_cmp_result > 0) {
677 if (ip_cmp_result > 0) {
681 fprintf(stderr,
"ERROR Inserting to list");
684 }
else if (ip_cmp_result < 0) {
688 fprintf(stderr,
"ERROR Inserting to list");
695 fprintf(stderr,
"ERROR Inserting to list");
699 }
else if (ip_cmp_result < 0) {
702 if (ip_cmp_result > 0) {
714 ¤t_interval.
high_ip) == NULL) {
766 if (end_of_list == conductor) {
767 end_of_list = conductor->
next->next;
771 else if (ip_cmp_result < 0) {
775 fprintf(stderr,
"ERROR Inserting to list");
799 if (end_of_list == conductor) {
800 end_of_list = conductor->
next;
808 if (ip_cmp_result > 0) {
819 if (end_of_list == conductor) {
820 end_of_list = conductor->
next;
829 }
else if (ip_cmp_result < 0) {
833 fprintf(stderr,
"ERROR Inserting to list");
849 }
else if (ip_cmp_result < 0) {
852 conductor = conductor->
next;
855 fprintf(stderr,
"ERROR Inserting to list");
861 if (conductor == NULL) {
869 end_of_list = end_of_list->
next;
877 conductor = end_of_list;
883 if (prefix_context == NULL) {
884 fprintf(stderr,
"ERROR allocating memory for prefix interval_search_context\n");
891 conductor = interval_list;
894 *context_counter = interval_counter;
897 while (conductor != NULL) {
899 memcpy(array_iterator, conductor->
interval, size_of_pref_interval);
903 interval_list = conductor->
next;
906 conductor = interval_list;
909 return prefix_context;
926 int first, last, middle;
929 uint8_t *middle_interval;
931 int ip_high_cmp_result;
932 int ip_low_cmp_result;
939 size_t low_ip_offset;
940 size_t high_ip_offset;
943 if (prefix_context->
v6_count == 0) {
947 last = prefix_context->
v6_count - 1;
948 middle = (first + last)>>1;
953 high_ip_offset = ip_addr_len;
955 ip_addr_start = (uint8_t *)ip;
958 if (prefix_context->
v4_count == 0) {
962 last = prefix_context->
v4_count - 1;
963 middle = (first + last)>>1;
968 high_ip_offset = ip_addr_len + 8;
971 ip_addr_start = ((uint8_t *)ip) + 8;
974 while (first <= last ) {
975 middle_interval = (uint8_t *)(interval_array + middle);
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);
980 if (ip_low_cmp_result <= 0 && ip_high_cmp_result >= 0) {
983 }
else if (ip_high_cmp_result > 0) {
988 middle = (first + last) >> 1;
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]) {
1013 if (k == *data_coll_cnt) {
1016 tmp = realloc(*data_collector, ((*data_coll_cnt) +
COLLECTORSLOTS) *
sizeof(
void *));
1018 fprintf(stderr,
"ERROR allocating memory for network mask array\n");
1021 *data_collector = tmp;
1024 (*data_collector)[*data_coll_cnt] = interval->
data_array[j];
1043 void **data_collector;
1044 uint32_t data_collector_cnt = 0;
1048 if (data_collector == NULL) {
1049 fprintf(stderr,
"ERROR allocating memory for freed data collector\n");
1053 while (interval_list != NULL) {
1054 tmp_interval = interval_list;
1055 interval_list = interval_list->
next;
1063 free(data_collector);
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.
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 'data_array'.
INLINE int ip_cmp(const ip_addr_t *addr1, const ip_addr_t *addr2)
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)
int copy_all_data(ipps_interval_t *dest, ipps_interval_t *src)
size_t data_len
Number of bytes in 'data'.
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 v6_count
Number of intervals in IPv6 array.
uint32_t ** create_ip_v6_net_mask_array()
ipps_interval_t * interval
Pointer to interval structure.
INLINE int ip_is6(const ip_addr_t *addr)
void ip_dec(const ip_addr_t *ip, ip_addr_t *ip_dec)
size_t array_len
Allocated size of 'data_array' => 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 'networks' 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)