summaryrefslogtreecommitdiffstats
path: root/src/addr.c
blob: e0d8e6baadef1e73db556cf1c3aba8d45fb37dc1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#include <stdio.h>
#include <string.h>

#include "addr.h"

int addr_len(const sockaddr_any *addr)
{
	switch (addr->any.sa_family) {
	case AF_INET:
		return sizeof(struct sockaddr_in);
	default:
		return 0;
	}
}

sockaddr_any *addr_parse_prefix(blob_t b, sockaddr_any *addr, uint8_t *prefix_len)
{
	memset(addr, 0, sizeof(*addr));

	if (blob_pull_inet_addr(&b, &addr->ipv4.sin_addr)) {
		addr->ipv4.sin_family = AF_INET;
		if (blob_pull_matching(&b, BLOB_STR("/"))) {
			if (prefix_len == NULL)
				return NULL;
			*prefix_len = blob_pull_uint(&b, 10);
		}
		if (b.len != 0)
			return NULL;
		return addr;
	}
	return NULL;
}

unsigned long addr_hash(const sockaddr_any *addr)
{
	switch (addr->any.sa_family) {
	case AF_INET:
		return htonl(addr->ipv4.sin_addr.s_addr);
	default:
		return 0;
	}
}

const char *addr_print(const sockaddr_any *addr)
{
	switch (addr->any.sa_family) {
	case AF_INET:
		return inet_ntoa(addr->ipv4.sin_addr);
	default:
		return "unknown";
	}
}

blob_t addr_get_hostaddr_blob(const sockaddr_any *addr)
{
	switch (addr->any.sa_family) {
	case AF_INET:
		return BLOB_BUF(&addr->ipv4.sin_addr);
	default:
		return BLOB_NULL;
	}
}

void addr_push_hostaddr(blob_t *b, const sockaddr_any *addr)
{
	char buf[64];
	blob_t f;
	unsigned int t;

	switch (addr->any.sa_family) {
	case AF_INET:
		t = ntohl(addr->ipv4.sin_addr.s_addr);
		f.ptr = buf;
		f.len = sprintf(buf, "%d.%d.%d.%d",
				(t >> 24) & 0xff, (t >>  16) & 0xff,
				(t >>  8) & 0xff, (t       ) & 0xff);
		break;
	default:
		return;
	}
	blob_push(b, f);
}

static int bitcmp(const uint8_t *a, const uint8_t *b, int len)
{
	int bytes, bits, mask, r;

	bytes = len / 8;
	bits  = len % 8;

	if (bytes != 0) {
		r = memcmp(a, b, bytes);
		if (r != 0)
			return r;
	}
	if (bits != 0) {
		mask = (0xff << (8 - bits)) & 0xff;
		return ((int) (a[bytes] & mask)) - ((int) (b[bytes] & mask));
	}
	return 0;
}

int addr_prefix_cmp(const sockaddr_any *a, const sockaddr_any *b, int prefix)
{
	if (a->any.sa_family != b->any.sa_family)
		return a->any.sa_family - b->any.sa_family;

	return bitcmp((const uint8_t *) &a->ipv4.sin_addr.s_addr,
		      (const uint8_t *) &b->ipv4.sin_addr.s_addr,
		      prefix);
}