summaryrefslogtreecommitdiff
path: root/libc/inet/ethers.c
blob: 857e5d165cb55062a73b4093424668fe65ce828e (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
112
113
114
115
116
117
118
119
120
121
122
/*
 * libc/inet/ethers.c
 *
 * Programmatic interface for the /etc/ethers file
 *
 * Copyright 2007 by Matthew Wilcox <matthew@wil.cx>
 *
 * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball.
 */

#include <ctype.h>
#include <stdio.h>
#include <string.h>
#include <netinet/ether.h>

#define ETHER_LINE_LEN	256

/*
 * Internal function which returns a pointer to the part of the line
 * with the start of the hostname, or NULL if we couldn't parse the line.
 * Note that this line may have a comment symbol on it somewhere; if so
 * it will return NULL if the # is before or within the ether_addr, and
 * succeed if the # is before or within the host.  It's up to the callers
 * to be aware of this.
 *
 * I would have preferred to write a NUL to the location of the comment
 * character, but ether_line takes a const argument.  See __ether_line_w.
 */
static const char *__ether_line(const char *line, struct ether_addr *addr)
{
	struct ether_addr *res = ether_aton_r(line, addr);
	if (!res)
		return NULL;

	while (*line && (*line != ' ') && (*line != '\t'))
		line++;
	while (*line && ((*line == ' ')	|| (*line == '\t')))
		line++;
	return (*line) ? line : NULL;
}

/*
 * Strips out the comment before calling __ether_line.  We can do this,
 * since we know the buffer is writable.
 */
static const char *__ether_line_w(char *line, struct ether_addr *addr)
{
	char *end = strchr(line, '#');
	if (!end)
		end = strchr(line, '\n');
	if (end)
		*end = '\0';
	return __ether_line(line, addr);
}

int ether_line(const char *line, struct ether_addr *addr, char *hostname)
{
	const char *name = __ether_line(line, addr);
	if (!name)
		return -1;

	while (*name) {
		if ((*name == '#') || isspace(*name))
			break;
		*hostname++ = *name++;
	}
	*hostname = '\0';

	return 0;
}

int ether_ntohost(char *hostname, const struct ether_addr *addr)
{
	int res = -1;
	FILE *fp;
	char buf[ETHER_LINE_LEN];

	fp = fopen(ETHER_FILE_NAME, "r");
	if (!fp)
		return -1;

	while (fgets(buf, sizeof(buf), fp)) {
		struct ether_addr tmp_addr;
		const char *cp = __ether_line_w(buf, &tmp_addr);
		if (!cp)
			continue;
		if (memcmp(addr, &tmp_addr, sizeof(tmp_addr)))
			continue;

		strcpy(hostname, cp);
		res = 0;
		break;
	}

	fclose(fp);
	return res;
}

int ether_hostton(const char *hostname, struct ether_addr *addr)
{
	int res = -1;
	FILE *fp;
	char buf[ETHER_LINE_LEN];

	fp = fopen(ETHER_FILE_NAME, "r");
	if (!fp)
		return -1;

	while (fgets(buf, sizeof(buf), fp)) {
		const char *cp = __ether_line_w(buf, addr);
		if (!cp)
			continue;
		if (strcasecmp(hostname, cp))
			continue;

		res = 0;
		break;
	}

	fclose(fp);
	return res;
}