]> git.ipfire.org Git - people/ms/libloc.git/blame - src/libloc/private.h
network: Add function that counts the bit length of an addres
[people/ms/libloc.git] / src / libloc / private.h
CommitLineData
46aded9a
MT
1/*
2 libloc - A library to determine the location of someone on the Internet
3
4 Copyright (C) 2017 IPFire Development Team <info@ipfire.org>
5
6 This library is free software; you can redistribute it and/or
7 modify it under the terms of the GNU Lesser General Public
8 License as published by the Free Software Foundation; either
9 version 2.1 of the License, or (at your option) any later version.
10
11 This library is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 Lesser General Public License for more details.
15*/
16
17#ifndef LIBLOC_PRIVATE_H
18#define LIBLOC_PRIVATE_H
19
1a3cb1b4
MT
20#ifdef LIBLOC_PRIVATE
21
2a30e4de 22#include <netinet/in.h>
46aded9a 23#include <stdbool.h>
726f9984 24#include <stdio.h>
46aded9a
MT
25#include <syslog.h>
26
c12a1385 27#include <libloc/libloc.h>
46aded9a
MT
28
29static inline void __attribute__((always_inline, format(printf, 2, 3)))
30loc_log_null(struct loc_ctx *ctx, const char *format, ...) {}
31
32#define loc_log_cond(ctx, prio, arg...) \
33 do { \
34 if (loc_get_log_priority(ctx) >= prio) \
35 loc_log(ctx, prio, __FILE__, __LINE__, __FUNCTION__, ## arg); \
36 } while (0)
37
38#ifdef ENABLE_DEBUG
39# define DEBUG(ctx, arg...) loc_log_cond(ctx, LOG_DEBUG, ## arg)
40#else
41# define DEBUG(ctx, arg...) loc_log_null(ctx, ## arg)
42#endif
43
44#define INFO(ctx, arg...) loc_log_cond(ctx, LOG_INFO, ## arg)
45#define ERROR(ctx, arg...) loc_log_cond(ctx, LOG_ERR, ## arg)
46
47#ifndef HAVE_SECURE_GETENV
48# ifdef HAVE___SECURE_GETENV
49# define secure_getenv __secure_getenv
50# else
d0cef0e8 51# define secure_getenv getenv
46aded9a
MT
52# endif
53#endif
54
55#define LOC_EXPORT __attribute__ ((visibility("default")))
56
57void loc_log(struct loc_ctx *ctx,
58 int priority, const char *file, int line, const char *fn,
59 const char *format, ...) __attribute__((format(printf, 6, 7)));
60
2a30e4de 61static inline int in6_addr_cmp(const struct in6_addr* a1, const struct in6_addr* a2) {
e14564aa
MT
62 for (unsigned int i = 0; i < 16; i++) {
63 if (a1->s6_addr[i] > a2->s6_addr[i])
64 return 1;
65
66 else if (a1->s6_addr[i] < a2->s6_addr[i])
67 return -1;
68 }
69
70 return 0;
2a30e4de
MT
71}
72
73static inline int in6_addr_get_bit(const struct in6_addr* address, unsigned int i) {
20e3a06f 74 return ((address->s6_addr[i / 8] >> (7 - (i % 8))) & 1);
2a30e4de
MT
75}
76
77static inline void in6_addr_set_bit(struct in6_addr* address, unsigned int i, unsigned int val) {
20e3a06f 78 address->s6_addr[i / 8] ^= (-val ^ address->s6_addr[i / 8]) & (1 << (7 - (i % 8)));
2a30e4de
MT
79}
80
30f0f0ed
MT
81static inline struct in6_addr loc_prefix_to_bitmask(const unsigned int prefix) {
82 struct in6_addr bitmask;
83
84 for (unsigned int i = 0; i < 16; i++)
85 bitmask.s6_addr[i] = 0;
86
87 for (int i = prefix, j = 0; i > 0; i -= 8, j++) {
88 if (i >= 8)
89 bitmask.s6_addr[j] = 0xff;
90 else
91 bitmask.s6_addr[j] = 0xff << (8 - i);
92 }
93
94 return bitmask;
95}
96
f69e0c8b
MT
97static inline unsigned int loc_address_bit_length(const struct in6_addr* address) {
98 unsigned int length = 128;
99
100 for (int octet = 0; octet <= 15; octet++) {
101 if (address->s6_addr[octet]) {
102 length -= __builtin_clz(address->s6_addr[octet]) - 24;
103 break;
104 } else
105 length -= 8;
106 }
107
108 return length;
109}
110
30f0f0ed
MT
111static inline struct in6_addr loc_address_and(
112 const struct in6_addr* address, const struct in6_addr* bitmask) {
113 struct in6_addr a;
114
115 // Perform bitwise AND
116 for (unsigned int i = 0; i < 4; i++)
117 a.s6_addr32[i] = address->s6_addr32[i] & bitmask->s6_addr32[i];
118
119 return a;
120}
121
122static inline struct in6_addr loc_address_or(
123 const struct in6_addr* address, const struct in6_addr* bitmask) {
124 struct in6_addr a;
125
126 // Perform bitwise OR
127 for (unsigned int i = 0; i < 4; i++)
128 a.s6_addr32[i] = address->s6_addr32[i] | ~bitmask->s6_addr32[i];
129
130 return a;
131}
132
257626f5
MT
133static inline void hexdump(struct loc_ctx* ctx, const void* addr, size_t len) {
134 char buffer_hex[16 * 3 + 6];
135 char buffer_ascii[17];
136
137 unsigned int i = 0;
138 unsigned char* p = (unsigned char*)addr;
139
356cabc1
MT
140 DEBUG(ctx, "Dumping %zu byte(s)\n", len);
141
257626f5
MT
142 // Process every byte in the data
143 for (i = 0; i < len; i++) {
144 // Multiple of 16 means new line (with line offset)
145 if ((i % 16) == 0) {
146 // Just don't print ASCII for the zeroth line
147 if (i != 0)
148 DEBUG(ctx, " %s %s\n", buffer_hex, buffer_ascii);
149
150 // Output the offset.
151 sprintf(buffer_hex, "%04x ", i);
152 }
153
154 // Now the hex code for the specific character
155 sprintf(buffer_hex + 5 + ((i % 16) * 3), " %02x", p[i]);
156
157 // And store a printable ASCII character for later
158 if ((p[i] < 0x20) || (p[i] > 0x7e))
159 buffer_ascii[i % 16] = '.';
160 else
161 buffer_ascii[i % 16] = p[i];
162
163 // Terminate string
164 buffer_ascii[(i % 16) + 1] = '\0';
165 }
166
167 // Pad out last line if not exactly 16 characters
168 while ((i % 16) != 0) {
169 sprintf(buffer_hex + 5 + ((i % 16) * 3), " ");
170 i++;
171 }
172
173 // And print the final bit
174 DEBUG(ctx, " %s %s\n", buffer_hex, buffer_ascii);
175}
176
46aded9a 177#endif
1a3cb1b4 178#endif