1*bf198cc6Smillert /* $OpenBSD: ethers.c,v 1.27 2019/01/25 00:19:25 millert Exp $ */
225955e21Smillert
3df930be7Sderaadt /*
4*bf198cc6Smillert * Copyright (c) 1998 Todd C. Miller <millert@openbsd.org>
5df930be7Sderaadt *
606f01696Smillert * Permission to use, copy, modify, and distribute this software for any
706f01696Smillert * purpose with or without fee is hereby granted, provided that the above
806f01696Smillert * copyright notice and this permission notice appear in all copies.
925955e21Smillert *
10328f1f07Smillert * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11328f1f07Smillert * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12328f1f07Smillert * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13328f1f07Smillert * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14328f1f07Smillert * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15328f1f07Smillert * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16328f1f07Smillert * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1725955e21Smillert */
1825955e21Smillert
1925955e21Smillert /*
2025955e21Smillert * ethers(3) a la Sun.
2125955e21Smillert * Originally Written by Roland McGrath <roland@frob.com> 10/14/93.
22*bf198cc6Smillert * Substantially modified by Todd C. Miller <millert@openbsd.org>
23df930be7Sderaadt */
24df930be7Sderaadt
25df930be7Sderaadt #include <sys/types.h>
26df930be7Sderaadt #include <sys/socket.h>
27df930be7Sderaadt #include <net/if.h>
28df930be7Sderaadt #include <netinet/in.h>
29df930be7Sderaadt #include <netinet/if_ether.h>
30df930be7Sderaadt #include <paths.h>
31df930be7Sderaadt #include <errno.h>
32df930be7Sderaadt #include <stdio.h>
33df930be7Sderaadt #include <stdlib.h>
34df930be7Sderaadt #include <string.h>
3525955e21Smillert #include <ctype.h>
36aea60beeSderaadt #include <limits.h>
37df930be7Sderaadt
38df930be7Sderaadt #ifndef _PATH_ETHERS
39df930be7Sderaadt #define _PATH_ETHERS "/etc/ethers"
40df930be7Sderaadt #endif
41df930be7Sderaadt
429f33d8a6Stedu static char * _ether_aton(const char *, struct ether_addr *);
4325955e21Smillert
44df930be7Sderaadt char *
ether_ntoa(struct ether_addr * e)45db5b349cSotto ether_ntoa(struct ether_addr *e)
46df930be7Sderaadt {
47df930be7Sderaadt static char a[] = "xx:xx:xx:xx:xx:xx";
48df930be7Sderaadt
495af14704Sderaadt (void)snprintf(a, sizeof a, "%02x:%02x:%02x:%02x:%02x:%02x",
50df930be7Sderaadt e->ether_addr_octet[0], e->ether_addr_octet[1],
51df930be7Sderaadt e->ether_addr_octet[2], e->ether_addr_octet[3],
52df930be7Sderaadt e->ether_addr_octet[4], e->ether_addr_octet[5]);
5325955e21Smillert
5425955e21Smillert return (a);
5525955e21Smillert }
5625955e21Smillert
5725955e21Smillert static char *
_ether_aton(const char * s,struct ether_addr * e)589f33d8a6Stedu _ether_aton(const char *s, struct ether_addr *e)
5925955e21Smillert {
6025955e21Smillert int i;
6125955e21Smillert long l;
6225955e21Smillert char *pp;
6325955e21Smillert
64dfe5467eSderaadt while (isspace((unsigned char)*s))
6525955e21Smillert s++;
6625955e21Smillert
6725955e21Smillert /* expect 6 hex octets separated by ':' or space/NUL if last octet */
6825955e21Smillert for (i = 0; i < 6; i++) {
6925955e21Smillert l = strtol(s, &pp, 16);
70ec510e63Sderaadt if (pp == s || l > 0xFF || l < 0)
7125955e21Smillert return (NULL);
72dfe5467eSderaadt if (!(*pp == ':' ||
73dfe5467eSderaadt (i == 5 && (isspace((unsigned char)*pp) ||
74dfe5467eSderaadt *pp == '\0'))))
7525955e21Smillert return (NULL);
7625955e21Smillert e->ether_addr_octet[i] = (u_char)l;
7725955e21Smillert s = pp + 1;
7825955e21Smillert }
7925955e21Smillert
8025955e21Smillert /* return character after the octets ala strtol(3) */
8125955e21Smillert return (pp);
82df930be7Sderaadt }
83df930be7Sderaadt
84df930be7Sderaadt struct ether_addr *
ether_aton(const char * s)859f33d8a6Stedu ether_aton(const char *s)
86df930be7Sderaadt {
87df930be7Sderaadt static struct ether_addr n;
88df930be7Sderaadt
8925955e21Smillert return (_ether_aton(s, &n) ? &n : NULL);
90df930be7Sderaadt }
91df930be7Sderaadt
92ceeee157Sderaadt int
ether_ntohost(char * hostname,struct ether_addr * e)93db5b349cSotto ether_ntohost(char *hostname, struct ether_addr *e)
94df930be7Sderaadt {
95df930be7Sderaadt FILE *f;
96aff2b1e4Smillert char buf[BUFSIZ+1], *p;
97aff2b1e4Smillert size_t len;
98df930be7Sderaadt struct ether_addr try;
99df930be7Sderaadt
100241db059Sguenther f = fopen(_PATH_ETHERS, "re");
101df930be7Sderaadt if (f == NULL)
10225955e21Smillert return (-1);
103aff2b1e4Smillert while ((p = fgetln(f, &len)) != NULL) {
104aff2b1e4Smillert if (p[len-1] == '\n')
105aff2b1e4Smillert len--;
106aff2b1e4Smillert if (len > sizeof(buf) - 2)
107aff2b1e4Smillert continue;
10825955e21Smillert (void)memcpy(buf, p, len);
109aff2b1e4Smillert buf[len] = '\n'; /* code assumes newlines later on */
110aff2b1e4Smillert buf[len+1] = '\0';
111a33788ceSbrynet /* A + in the file meant try YP, ignore it. */
112a33788ceSbrynet if (!strncmp(buf, "+\n", sizeof(buf)))
113df930be7Sderaadt continue;
114df930be7Sderaadt if (ether_line(buf, &try, hostname) == 0 &&
1153240e6a8Sguenther memcmp(&try, e, sizeof(try)) == 0) {
116df930be7Sderaadt (void)fclose(f);
11725955e21Smillert return (0);
118df930be7Sderaadt }
119df930be7Sderaadt }
120df930be7Sderaadt (void)fclose(f);
121df930be7Sderaadt errno = ENOENT;
12225955e21Smillert return (-1);
123df930be7Sderaadt }
124df930be7Sderaadt
125ceeee157Sderaadt int
ether_hostton(const char * hostname,struct ether_addr * e)1269f33d8a6Stedu ether_hostton(const char *hostname, struct ether_addr *e)
127df930be7Sderaadt {
128df930be7Sderaadt FILE *f;
129aff2b1e4Smillert char buf[BUFSIZ+1], *p;
130aea60beeSderaadt char try[HOST_NAME_MAX+1];
131aff2b1e4Smillert size_t len;
132df930be7Sderaadt
133241db059Sguenther f = fopen(_PATH_ETHERS, "re");
134df930be7Sderaadt if (f==NULL)
13525955e21Smillert return (-1);
136df930be7Sderaadt
137aff2b1e4Smillert while ((p = fgetln(f, &len)) != NULL) {
138aff2b1e4Smillert if (p[len-1] == '\n')
139aff2b1e4Smillert len--;
140aff2b1e4Smillert if (len > sizeof(buf) - 2)
141aff2b1e4Smillert continue;
142aff2b1e4Smillert memcpy(buf, p, len);
143aff2b1e4Smillert buf[len] = '\n'; /* code assumes newlines later on */
144aff2b1e4Smillert buf[len+1] = '\0';
145a33788ceSbrynet /* A + in the file meant try YP, ignore it. */
146a33788ceSbrynet if (!strncmp(buf, "+\n", sizeof(buf)))
147df930be7Sderaadt continue;
148df930be7Sderaadt if (ether_line(buf, e, try) == 0 && strcmp(hostname, try) == 0) {
149df930be7Sderaadt (void)fclose(f);
15025955e21Smillert return (0);
151df930be7Sderaadt }
152df930be7Sderaadt }
153df930be7Sderaadt (void)fclose(f);
154df930be7Sderaadt errno = ENOENT;
15525955e21Smillert return (-1);
156df930be7Sderaadt }
157df930be7Sderaadt
158ceeee157Sderaadt int
ether_line(const char * line,struct ether_addr * e,char * hostname)1599f33d8a6Stedu ether_line(const char *line, struct ether_addr *e, char *hostname)
160df930be7Sderaadt {
16125955e21Smillert char *p;
16225955e21Smillert size_t n;
163df930be7Sderaadt
16425955e21Smillert /* Parse "xx:xx:xx:xx:xx:xx" */
16525955e21Smillert if ((p = _ether_aton(line, e)) == NULL || (*p != ' ' && *p != '\t'))
16625955e21Smillert goto bad;
16725955e21Smillert
16825955e21Smillert /* Now get the hostname */
169dfe5467eSderaadt while (isspace((unsigned char)*p))
17025955e21Smillert p++;
17125955e21Smillert if (*p == '\0')
17225955e21Smillert goto bad;
17325955e21Smillert n = strcspn(p, " \t\n");
174aea60beeSderaadt if (n >= HOST_NAME_MAX+1)
17525955e21Smillert goto bad;
176f1a075daSlebel strlcpy(hostname, p, n + 1);
17725955e21Smillert return (0);
17825955e21Smillert
17925955e21Smillert bad:
180df930be7Sderaadt errno = EINVAL;
18125955e21Smillert return (-1);
182df930be7Sderaadt }
18392a24078Sguenther DEF_WEAK(ether_line);
184