Coverage Report

Created: 2026-04-12 06:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/libpcap/nametoaddr.c
Line
Count
Source
1
/*
2
 * Copyright (c) 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998
3
 *  The Regents of the University of California.  All rights reserved.
4
 *
5
 * Redistribution and use in source and binary forms, with or without
6
 * modification, are permitted provided that: (1) source code distributions
7
 * retain the above copyright notice and this paragraph in its entirety, (2)
8
 * distributions including binary code include the above copyright notice and
9
 * this paragraph in its entirety in the documentation or other materials
10
 * provided with the distribution, and (3) all advertising materials mentioning
11
 * features or use of this software display the following acknowledgement:
12
 * ``This product includes software developed by the University of California,
13
 * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
14
 * the University nor the names of its contributors may be used to endorse
15
 * or promote products derived from this software without specific prior
16
 * written permission.
17
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
18
 * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
19
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
20
 *
21
 * Name to id translation routines used by the scanner.
22
 * These functions are not time critical.
23
 */
24
25
#include <config.h>
26
27
#ifdef _WIN32
28
  #include <winsock2.h>
29
  #include <ws2tcpip.h>
30
#else /* _WIN32 */
31
  #include <sys/param.h>
32
  #include <sys/types.h>
33
  #include <sys/socket.h>
34
  #include <sys/time.h>
35
36
  #include <netinet/in.h>
37
38
  #if defined(__linux__) && defined(HAVE_ETHER_HOSTTON)
39
    #include <features.h>
40
    #if ! defined(__GLIBC__) && ! defined(__UCLIBC__)
41
      /*
42
       * In musl libc (which does not identify itself) ether_hostton() is
43
       * present and does not work.
44
       */
45
      #undef HAVE_ETHER_HOSTTON
46
    #endif
47
  #endif // defined(__linux__) && defined(HAVE_ETHER_HOSTTON)
48
49
  #ifdef HAVE_ETHER_HOSTTON
50
    #if defined(NET_ETHERNET_H_DECLARES_ETHER_HOSTTON)
51
      /*
52
       * OK, just include <net/ethernet.h>.
53
       */
54
      #include <net/ethernet.h>
55
    #elif defined(NETINET_ETHER_H_DECLARES_ETHER_HOSTTON)
56
      /*
57
       * OK, just include <netinet/ether.h>
58
       */
59
      #include <netinet/ether.h>
60
    #elif defined(SYS_ETHERNET_H_DECLARES_ETHER_HOSTTON)
61
      /*
62
       * OK, just include <sys/ethernet.h>
63
       */
64
      #include <sys/ethernet.h>
65
    #elif defined(ARPA_INET_H_DECLARES_ETHER_HOSTTON)
66
      /*
67
       * OK, just include <arpa/inet.h>
68
       */
69
      #include <arpa/inet.h>
70
    #elif defined(NETINET_IF_ETHER_H_DECLARES_ETHER_HOSTTON)
71
      /*
72
       * OK, include <netinet/if_ether.h>, after all the other stuff we
73
       * need to include or define for its benefit.
74
       */
75
      #define NEED_NETINET_IF_ETHER_H
76
    #else
77
      /*
78
       * We'll have to declare it ourselves.
79
       * If <netinet/if_ether.h> defines struct ether_addr, include
80
       * it.  Otherwise, define it ourselves.
81
       */
82
      #ifdef HAVE_STRUCT_ETHER_ADDR
83
        #define NEED_NETINET_IF_ETHER_H
84
      #else /* HAVE_STRUCT_ETHER_ADDR */
85
  struct ether_addr {
86
    unsigned char ether_addr_octet[6];
87
  };
88
      #endif /* HAVE_STRUCT_ETHER_ADDR */
89
    #endif /* what declares ether_hostton() */
90
91
    #ifdef NEED_NETINET_IF_ETHER_H
92
      #include <net/if.h> /* Needed on some platforms */
93
      #include <netinet/in.h> /* Needed on some platforms */
94
      #include <netinet/if_ether.h>
95
    #endif /* NEED_NETINET_IF_ETHER_H */
96
97
    #ifndef HAVE_DECL_ETHER_HOSTTON
98
      /*
99
       * No header declares it, so declare it ourselves.
100
       */
101
      extern int ether_hostton(const char *, struct ether_addr *);
102
    #endif /* !defined(HAVE_DECL_ETHER_HOSTTON) */
103
  #endif /* HAVE_ETHER_HOSTTON */
104
105
  #include <arpa/inet.h>
106
  #include <netdb.h>
107
#endif /* _WIN32 */
108
109
#include <errno.h>
110
#include <stdlib.h>
111
#include <string.h>
112
#include <stdio.h>
113
114
#include "pcap-int.h"
115
116
#include "diag-control.h"
117
118
#include "gencode.h"
119
#include <pcap/namedb.h>
120
#include "nametoaddr.h"
121
122
#include "thread-local.h"
123
124
#ifdef HAVE_OS_PROTO_H
125
#include "os-proto.h"
126
#endif
127
128
#ifndef NTOHL
129
0
#define NTOHL(x) (x) = ntohl(x)
130
#endif
131
132
/*
133
 *  Convert host name to internet address.
134
 *  Return 0 upon failure.
135
 *  XXX - not thread-safe; don't use it inside libpcap.
136
 */
137
bpf_u_int32 **
138
pcap_nametoaddr(const char *name)
139
0
{
140
#ifndef h_addr
141
  static bpf_u_int32 *hlist[2];
142
#endif
143
0
  bpf_u_int32 **p;
144
0
  struct hostent *hp;
145
146
  /*
147
   * gethostbyname() is deprecated on Windows, perhaps because
148
   * it's not thread-safe, or because it doesn't support IPv6,
149
   * or both.
150
   *
151
   * We deprecate pcap_nametoaddr() on all platforms because
152
   * it's not thread-safe; we supply it for backwards compatibility,
153
   * so suppress the deprecation warning.  We could, I guess,
154
   * use getaddrinfo() and construct the array ourselves, but
155
   * that's probably not worth the effort, as that wouldn't make
156
   * this thread-safe - we can't change the API to require that
157
   * our caller free the address array, so we still have to reuse
158
   * a local array.
159
   */
160
0
DIAG_OFF_DEPRECATION
161
0
  if ((hp = gethostbyname(name)) != NULL) {
162
0
DIAG_ON_DEPRECATION
163
#ifndef h_addr
164
    hlist[0] = (bpf_u_int32 *)hp->h_addr;
165
    NTOHL(hp->h_addr);
166
    return hlist;
167
#else
168
0
    for (p = (bpf_u_int32 **)hp->h_addr_list; *p; ++p)
169
0
      NTOHL(**p);
170
0
    return (bpf_u_int32 **)hp->h_addr_list;
171
0
#endif
172
0
  }
173
0
  else
174
0
    return 0;
175
0
}
176
177
struct addrinfo *
178
pcap_nametoaddrinfo(const char *name)
179
0
{
180
0
  struct addrinfo hints, *res;
181
0
  int error;
182
183
0
  memset(&hints, 0, sizeof(hints));
184
0
  hints.ai_family = PF_UNSPEC;
185
0
  hints.ai_socktype = SOCK_STREAM;  /*not really*/
186
0
  hints.ai_protocol = IPPROTO_TCP;  /*not really*/
187
0
  error = getaddrinfo(name, NULL, &hints, &res);
188
0
  if (error)
189
0
    return NULL;
190
0
  else
191
0
    return res;
192
0
}
193
194
/*
195
 *  Convert net name to internet address.
196
 *  Return 0 upon failure.
197
 *  XXX - not guaranteed to be thread-safe!  See below for platforms
198
 *  on which it is thread-safe and on which it isn't.
199
 */
200
#if defined(_WIN32) || defined(__CYGWIN__)
201
bpf_u_int32
202
pcap_nametonetaddr(const char *name _U_)
203
{
204
  /*
205
   * There's no "getnetbyname()" on Windows.
206
   *
207
   * XXX - I guess we could use the BSD code to read
208
   * C:\Windows\System32\drivers\etc/networks, assuming
209
   * that's its home on all the versions of Windows
210
   * we use, but that file probably just has the loopback
211
   * network on 127/24 on 99 44/100% of Windows machines.
212
   *
213
   * (Heck, these days it probably just has that on 99 44/100%
214
   * of *UN*X* machines.)
215
   */
216
  return 0;
217
}
218
#else /* _WIN32 */
219
bpf_u_int32
220
pcap_nametonetaddr(const char *name)
221
0
{
222
  /*
223
   * UN*X.
224
   */
225
0
  struct netent *np;
226
0
  #if defined(HAVE_LINUX_GETNETBYNAME_R)
227
  /*
228
   * We have Linux's reentrant getnetbyname_r().
229
   */
230
0
  struct netent result_buf;
231
0
  char buf[1024]; /* arbitrary size */
232
0
  int h_errnoval;
233
0
  int err;
234
235
  /*
236
   * Apparently, the man page at
237
   *
238
   *    http://man7.org/linux/man-pages/man3/getnetbyname_r.3.html
239
   *
240
   * lies when it says
241
   *
242
   *    If the function call successfully obtains a network record,
243
   *    then *result is set pointing to result_buf; otherwise, *result
244
   *    is set to NULL.
245
   *
246
   * and, in fact, at least in some versions of GNU libc, it does
247
   * *not* always get set if getnetbyname_r() succeeds.
248
   */
249
0
  np = NULL;
250
0
  err = getnetbyname_r(name, &result_buf, buf, sizeof buf, &np,
251
0
      &h_errnoval);
252
0
  if (err != 0) {
253
    /*
254
     * XXX - dynamically allocate the buffer, and make it
255
     * bigger if we get ERANGE back?
256
     */
257
0
    return 0;
258
0
  }
259
  #elif defined(HAVE_SOLARIS_GETNETBYNAME_R)
260
  /*
261
   * We have Solaris's reentrant getnetbyname_r().
262
   */
263
  struct netent result_buf;
264
  char buf[1024]; /* arbitrary size */
265
266
  np = getnetbyname_r(name, &result_buf, buf, (int)sizeof buf);
267
  #elif defined(HAVE_AIX_GETNETBYNAME_R)
268
  /*
269
   * We have AIX's reentrant getnetbyname_r().
270
   */
271
  struct netent result_buf;
272
  struct netent_data net_data;
273
274
  if (getnetbyname_r(name, &result_buf, &net_data) == -1)
275
    np = NULL;
276
  else
277
    np = &result_buf;
278
  #else
279
  /*
280
   * We don't have any getnetbyname_r(); either we have a
281
   * getnetbyname() that uses thread-specific data, in which
282
   * case we're thread-safe (sufficiently recent FreeBSD,
283
   * sufficiently recent Darwin-based OS, sufficiently recent
284
   * HP-UX, or we have the
285
   * traditional getnetbyname() (everything else, including
286
   * current NetBSD and OpenBSD), in which case we're not
287
   * thread-safe.
288
   */
289
  np = getnetbyname(name);
290
  #endif
291
0
  if (np != NULL)
292
0
    return np->n_net;
293
0
  else
294
0
    return 0;
295
0
}
296
#endif /* _WIN32 */
297
298
/*
299
 * Convert a port name to its port and protocol numbers.
300
 * We assume only TCP or UDP.
301
 * Return 0 upon failure.
302
 */
303
int
304
pcap_nametoport(const char *name, int *port, int *proto)
305
0
{
306
0
  struct addrinfo hints, *res, *ai;
307
0
  int error;
308
0
  struct sockaddr_in *in4;
309
0
  struct sockaddr_in6 *in6;
310
0
  int tcp_port = -1;
311
0
  int udp_port = -1;
312
313
  /*
314
   * We check for both TCP and UDP in case there are
315
   * ambiguous entries.
316
   */
317
0
  memset(&hints, 0, sizeof(hints));
318
0
  hints.ai_family = PF_UNSPEC;
319
0
  hints.ai_socktype = SOCK_STREAM;
320
0
  hints.ai_protocol = IPPROTO_TCP;
321
0
  error = getaddrinfo(NULL, name, &hints, &res);
322
0
  if (error != 0) {
323
0
    if (error != EAI_NONAME &&
324
0
        error != EAI_SERVICE) {
325
      /*
326
       * This is a real error, not just "there's
327
       * no such service name".
328
       * XXX - this doesn't return an error string.
329
       */
330
0
      return 0;
331
0
    }
332
0
  } else {
333
    /*
334
     * OK, we found it.  Did it find anything?
335
     */
336
0
    for (ai = res; ai != NULL; ai = ai->ai_next) {
337
      /*
338
       * Does it have an address?
339
       */
340
0
      if (ai->ai_addr != NULL) {
341
        /*
342
         * Yes.  Get a port number; we're done.
343
         */
344
0
        if (ai->ai_addr->sa_family == AF_INET) {
345
0
          in4 = (struct sockaddr_in *)ai->ai_addr;
346
0
          tcp_port = ntohs(in4->sin_port);
347
0
          break;
348
0
        }
349
0
        if (ai->ai_addr->sa_family == AF_INET6) {
350
0
          in6 = (struct sockaddr_in6 *)ai->ai_addr;
351
0
          tcp_port = ntohs(in6->sin6_port);
352
0
          break;
353
0
        }
354
0
      }
355
0
    }
356
0
    freeaddrinfo(res);
357
0
  }
358
359
0
  memset(&hints, 0, sizeof(hints));
360
0
  hints.ai_family = PF_UNSPEC;
361
0
  hints.ai_socktype = SOCK_DGRAM;
362
0
  hints.ai_protocol = IPPROTO_UDP;
363
0
  error = getaddrinfo(NULL, name, &hints, &res);
364
0
  if (error != 0) {
365
0
    if (error != EAI_NONAME &&
366
0
        error != EAI_SERVICE) {
367
      /*
368
       * This is a real error, not just "there's
369
       * no such service name".
370
       * XXX - this doesn't return an error string.
371
       */
372
0
      return 0;
373
0
    }
374
0
  } else {
375
    /*
376
     * OK, we found it.  Did it find anything?
377
     */
378
0
    for (ai = res; ai != NULL; ai = ai->ai_next) {
379
      /*
380
       * Does it have an address?
381
       */
382
0
      if (ai->ai_addr != NULL) {
383
        /*
384
         * Yes.  Get a port number; we're done.
385
         */
386
0
        if (ai->ai_addr->sa_family == AF_INET) {
387
0
          in4 = (struct sockaddr_in *)ai->ai_addr;
388
0
          udp_port = ntohs(in4->sin_port);
389
0
          break;
390
0
        }
391
0
        if (ai->ai_addr->sa_family == AF_INET6) {
392
0
          in6 = (struct sockaddr_in6 *)ai->ai_addr;
393
0
          udp_port = ntohs(in6->sin6_port);
394
0
          break;
395
0
        }
396
0
      }
397
0
    }
398
0
    freeaddrinfo(res);
399
0
  }
400
401
  /*
402
   * We need to check /etc/services for ambiguous entries.
403
   * If we find an ambiguous entry, and it has the
404
   * same port number, change the proto to PROTO_UNDEF
405
   * so both TCP and UDP will be checked.
406
   */
407
0
  if (tcp_port >= 0) {
408
0
    *port = tcp_port;
409
0
    *proto = IPPROTO_TCP;
410
0
    if (udp_port >= 0) {
411
0
      if (udp_port == tcp_port)
412
0
        *proto = PROTO_UNDEF;
413
#ifdef notdef
414
      else
415
        /* Can't handle ambiguous names that refer
416
           to different port numbers. */
417
        warning("ambiguous port %s in /etc/services",
418
          name);
419
#endif
420
0
    }
421
0
    return 1;
422
0
  }
423
0
  if (udp_port >= 0) {
424
0
    *port = udp_port;
425
0
    *proto = IPPROTO_UDP;
426
0
    return 1;
427
0
  }
428
0
  return 0;
429
0
}
430
431
/*
432
 * Convert a string in the form PPP-PPP, where correspond to ports, to
433
 * a starting and ending port in a port range.
434
 * Return 0 on failure.
435
 */
436
int
437
pcap_nametoportrange(const char *name, int *port1, int *port2, int *proto)
438
0
{
439
0
  char *off, *cpy;
440
0
  int save_proto;
441
442
0
  if ((cpy = strdup(name)) == NULL)
443
0
    return 0;
444
445
0
  if ((off = strchr(cpy, '-')) == NULL) {
446
0
    free(cpy);
447
0
    return 0;
448
0
  }
449
450
0
  *off = '\0';
451
452
0
  if (pcap_nametoport(cpy, port1, proto) == 0) {
453
0
    free(cpy);
454
0
    return 0;
455
0
  }
456
0
  save_proto = *proto;
457
458
0
  if (pcap_nametoport(off + 1, port2, proto) == 0) {
459
0
    free(cpy);
460
0
    return 0;
461
0
  }
462
0
  free(cpy);
463
464
0
  if (*proto != save_proto)
465
0
    *proto = PROTO_UNDEF;
466
467
0
  return 1;
468
0
}
469
470
/*
471
 * XXX - not guaranteed to be thread-safe!  See below for platforms
472
 * on which it is thread-safe and on which it isn't.
473
 */
474
int
475
pcap_nametoproto(const char *str)
476
0
{
477
0
  struct protoent *p;
478
0
  #if defined(HAVE_LINUX_GETPROTOBYNAME_R)
479
  /*
480
   * We have Linux's reentrant getprotobyname_r().
481
   */
482
0
  struct protoent result_buf;
483
0
  char buf[1024]; // "...1024 bytes should be sufficient for most applications."
484
0
  int err;
485
486
0
  err = getprotobyname_r(str, &result_buf, buf, sizeof buf, &p);
487
  /*
488
   * As far as GNU libc implementation goes, an "error" means the
489
   * protocol database could not be searched, which could mean err ==
490
   * ERANGE if the buffer is too small or ENOENT if the protocols(5)
491
   * file does not exist (the man page does not document the latter
492
   * eventuality).  If the database has been searched normally and the
493
   * requested protocol name was not found, it is not an "error" and
494
   * err == 0.
495
   *
496
   * This notwithstanding, p == NULL iff a record was not found for any
497
   * reason (whether an "error" or not), which is the same semantics as
498
   * in every other HAVE_xxxxx branch of this block.  The final check
499
   * after the block will catch that if necessary.
500
   */
501
0
  if (err != 0) {
502
    /*
503
     * XXX - dynamically allocate the buffer, and make it
504
     * bigger if we get ERANGE back?
505
     */
506
0
    return PROTO_UNDEF;
507
0
  }
508
  #elif defined(HAVE_SOLARIS_GETPROTOBYNAME_R)
509
  /*
510
   * We have Solaris's reentrant getprotobyname_r().
511
   */
512
  struct protoent result_buf;
513
  char buf[1024]; // "...must be at least 1024 bytes."
514
515
  p = getprotobyname_r(str, &result_buf, buf, (int)sizeof buf);
516
  #elif defined(HAVE_AIX_GETPROTOBYNAME_R)
517
  /*
518
   * We have AIX's reentrant getprotobyname_r().
519
   */
520
  struct protoent result_buf;
521
  // "The structure must be zero-filled before it is used..." (OpenBSD).
522
  struct protoent_data proto_data = {0};
523
524
  if (getprotobyname_r(str, &result_buf, &proto_data) == -1)
525
    p = NULL;
526
  else
527
    p = &result_buf;
528
  #else
529
  /*
530
   * We don't have any getprotobyname_r(); either we have a
531
   * getprotobyname() that uses thread-specific data, in which
532
   * case we're thread-safe (sufficiently recent FreeBSD,
533
   * sufficiently recent Darwin-based OS, sufficiently recent
534
   * HP-UX, Windows), or we have
535
   * the traditional getprotobyname() (everything else, including
536
   * current NetBSD and OpenBSD), in which case we're not
537
   * thread-safe.
538
   */
539
  p = getprotobyname(str);
540
  #endif
541
0
  if (p != 0)
542
0
    return p->p_proto;
543
0
  else
544
0
    return PROTO_UNDEF;
545
0
}
546
547
#include "ethertype.h"
548
549
struct eproto {
550
  const char *s;
551
  u_short p;
552
};
553
554
/*
555
 * Static data base of ether protocol types.
556
 * tcpdump used to import this, and it's declared as an export on
557
 * Debian, at least, so make it a public symbol, even though we
558
 * don't officially export it by declaring it in a header file.
559
 * (Programs *should* do this themselves, as tcpdump now does.)
560
 *
561
 * We declare it here, right before defining it, to squelch any
562
 * warnings we might get from compilers about the lack of a
563
 * declaration.
564
 */
565
PCAP_API struct eproto eproto_db[];
566
PCAP_API_DEF struct eproto eproto_db[] = {
567
  { "aarp", ETHERTYPE_AARP },
568
  { "arp", ETHERTYPE_ARP },
569
  { "atalk", ETHERTYPE_ATALK },
570
  { "decnet", ETHERTYPE_DN },
571
  { "ip", ETHERTYPE_IP },
572
  { "ip6", ETHERTYPE_IPV6 },
573
  { "lat", ETHERTYPE_LAT },
574
  { "lldp", ETHERTYPE_LLDP },
575
  { "loopback", ETHERTYPE_LOOPBACK },
576
  { "mopdl", ETHERTYPE_MOPDL },
577
  { "moprc", ETHERTYPE_MOPRC },
578
  { "rarp", ETHERTYPE_REVARP },
579
  { "sca", ETHERTYPE_SCA },
580
  { "slow", ETHERTYPE_SLOW },
581
  { NULL, 0 }
582
};
583
584
static int
585
proto_or_undef(const struct eproto *p, const char *s)
586
0
{
587
0
  while (p->s) {
588
0
    if (strcmp(p->s, s) == 0)
589
0
      return p->p;
590
0
    p += 1;
591
0
  }
592
0
  return PROTO_UNDEF;
593
0
}
594
595
int
596
pcap_nametoeproto(const char *s)
597
0
{
598
0
  return proto_or_undef(eproto_db, s);
599
0
}
600
601
#include "llc.h"
602
603
/* Static data base of LLC values. */
604
static struct eproto llc_db[] = {
605
  { "iso", LLCSAP_ISONS },
606
  { "stp", LLCSAP_8021D },
607
  { "ipx", LLCSAP_IPX },
608
  { "netbeui", LLCSAP_NETBEUI },
609
  { NULL, 0 }
610
};
611
612
int
613
pcap_nametollc(const char *s)
614
0
{
615
0
  return proto_or_undef(llc_db, s);
616
0
}
617
618
/* Hex digit to 8-bit unsigned integer. */
619
u_char
620
pcapint_xdtoi(const u_char c)
621
0
{
622
0
  if (c >= '0' && c <= '9')
623
0
    return (u_char)(c - '0');
624
0
  else if (c >= 'a' && c <= 'f')
625
0
    return (u_char)(c - 'a' + 10);
626
0
  else
627
0
    return (u_char)(c - 'A' + 10);
628
0
}
629
630
int
631
pcapint_atoin(const char *s, bpf_u_int32 *addr)
632
0
{
633
0
  u_int n;
634
0
  int len;
635
636
0
  *addr = 0;
637
0
  len = 0;
638
0
  for (;;) {
639
0
    n = 0;
640
0
    while (*s && *s != '.') {
641
0
      if (n > 25) {
642
        /* The result will be > 255 */
643
0
        return -1;
644
0
      }
645
0
      n = n * 10 + *s++ - '0';
646
0
    }
647
0
    if (n > 255)
648
0
      return -1;
649
0
    *addr <<= 8;
650
0
    *addr |= n & 0xff;
651
0
    len += 8;
652
0
    if (*s == '\0')
653
0
      return len;
654
0
    ++s;
655
0
  }
656
  /* NOTREACHED */
657
0
}
658
659
/*
660
 * Iff 's' is a string that is a well-formed DECnet address (aa.nnnn), parse
661
 * the address into 'addr' and return 1, otherwise return 0.
662
 *
663
 * A binary DECnet address consists of a 6-bit area number and a 10-bit node
664
 * number; neither area 0 nor node 0 are valid for normal addressing purposes,
665
 * but either can potentially appear on the wire.
666
 */
667
int
668
pcapint_atodn(const char *s, uint16_t *addr)
669
0
{
670
0
#define AREASHIFT 10
671
0
#define AREAMASK 0176000
672
0
#define NODEMASK 01777
673
674
  /* Initialize to squelch a compiler warning only. */
675
0
  u_int node = 0, area = 0;
676
  /*
677
   *               +--+             +--+
678
   *               |  |             |  |
679
   *               v  |             v  |
680
   * --> START --> AREA --> DOT --> NODE -->
681
   *       |          |     |        |
682
   *       |          v     v        |
683
   *       +--------> INVALID <------+
684
   */
685
0
  enum {
686
0
    START,
687
0
    AREA,
688
0
    DOT,
689
0
    NODE,
690
0
    INVALID
691
0
  } fsm_state = START;
692
693
0
  while (*s) {
694
0
    switch (fsm_state) {
695
0
    case START:
696
0
      if (PCAP_ISDIGIT(*s)) {
697
0
        area = *s - '0';
698
0
        fsm_state = AREA;
699
0
        break;
700
0
      }
701
0
      fsm_state = INVALID;
702
0
      break;
703
0
    case AREA:
704
0
      if (*s == '.') {
705
0
        fsm_state = DOT;
706
0
        break;
707
0
      }
708
0
      if (PCAP_ISDIGIT(*s)) {
709
0
        area = area * 10 + *s - '0';
710
0
        if (area <= AREAMASK >> AREASHIFT)
711
0
          break;
712
0
      }
713
0
      fsm_state = INVALID;
714
0
      break;
715
0
    case DOT:
716
0
      if (PCAP_ISDIGIT(*s)) {
717
0
        node = *s - '0';
718
0
        fsm_state = NODE;
719
0
        break;
720
0
      }
721
0
      fsm_state = INVALID;
722
0
      break;
723
0
    case NODE:
724
0
      if (PCAP_ISDIGIT(*s)) {
725
0
        node = node * 10 + *s - '0';
726
0
        if (node <= NODEMASK)
727
0
          break;
728
0
      }
729
0
      fsm_state = INVALID;
730
0
      break;
731
0
    case INVALID:
732
0
      return 0;
733
0
    } /* switch */
734
0
    s++;
735
0
  } /* while */
736
  /*
737
   * This condition is false if the string comes from the lexer, but
738
   * let's not depend on that.
739
   */
740
0
  if (fsm_state != NODE)
741
0
    return 0;
742
743
  // Both 'area' and 'node' have been verified to be in range.
744
0
  *addr = (uint16_t)(area << AREASHIFT | node);
745
0
  return 1;
746
0
}
747
748
/*
749
 * libpcap ARCnet address format is "^\$[0-9a-fA-F]{1,2}$" in regexp syntax.
750
 * Iff the given string is a well-formed ARCnet address, parse the string,
751
 * store the 8-bit unsigned value into the provided integer and return 1.
752
 * Otherwise return 0.
753
 *
754
 *  --> START -- $ --> DOLLAR -- [0-9a-fA-F] --> HEX1 -- \0 -->-+
755
 *        |              |                        |             |
756
 *       [.]            [.]                  [0-9a-fA-F]        |
757
 *        |              |                        |             |
758
 *        v              v                        v             v
759
 *    (invalid) <--------+-<---------------[.]-- HEX2 -- \0 -->-+--> (valid)
760
 */
761
int
762
pcapint_atoan(const char *s, uint8_t *addr)
763
0
{
764
0
  enum {
765
0
    START,
766
0
    DOLLAR,
767
0
    HEX1,
768
0
    HEX2,
769
0
  } fsm_state = START;
770
0
  uint8_t tmp = 0;
771
772
0
  while (*s) {
773
0
    switch (fsm_state) {
774
0
    case START:
775
0
      if (*s != '$')
776
0
        goto invalid;
777
0
      fsm_state = DOLLAR;
778
0
      break;
779
0
    case DOLLAR:
780
0
      if (! PCAP_ISXDIGIT(*s))
781
0
        goto invalid;
782
0
      tmp = pcapint_xdtoi(*s);
783
0
      fsm_state = HEX1;
784
0
      break;
785
0
    case HEX1:
786
0
      if (! PCAP_ISXDIGIT(*s))
787
0
        goto invalid;
788
0
      tmp <<= 4;
789
0
      tmp |= pcapint_xdtoi(*s);
790
0
      fsm_state = HEX2;
791
0
      break;
792
0
    case HEX2:
793
0
      goto invalid;
794
0
    } // switch
795
0
    s++;
796
0
  } // while
797
0
  if (fsm_state == HEX1 || fsm_state == HEX2) {
798
0
    *addr = tmp;
799
0
    return 1;
800
0
  }
801
802
0
invalid:
803
0
  return 0;
804
0
}
805
806
// Man page: "xxxxxxxxxxxx", regexp: "^[0-9a-fA-F]{12}$".
807
static u_char
808
pcapint_atomac48_xxxxxxxxxxxx(const char *s, uint8_t *addr)
809
0
{
810
0
  if (strlen(s) == 12 &&
811
0
      PCAP_ISXDIGIT(s[0]) &&
812
0
      PCAP_ISXDIGIT(s[1]) &&
813
0
      PCAP_ISXDIGIT(s[2]) &&
814
0
      PCAP_ISXDIGIT(s[3]) &&
815
0
      PCAP_ISXDIGIT(s[4]) &&
816
0
      PCAP_ISXDIGIT(s[5]) &&
817
0
      PCAP_ISXDIGIT(s[6]) &&
818
0
      PCAP_ISXDIGIT(s[7]) &&
819
0
      PCAP_ISXDIGIT(s[8]) &&
820
0
      PCAP_ISXDIGIT(s[9]) &&
821
0
      PCAP_ISXDIGIT(s[10]) &&
822
0
      PCAP_ISXDIGIT(s[11])) {
823
0
    addr[0] = pcapint_xdtoi(s[0]) << 4 | pcapint_xdtoi(s[1]);
824
0
    addr[1] = pcapint_xdtoi(s[2]) << 4 | pcapint_xdtoi(s[3]);
825
0
    addr[2] = pcapint_xdtoi(s[4]) << 4 | pcapint_xdtoi(s[5]);
826
0
    addr[3] = pcapint_xdtoi(s[6]) << 4 | pcapint_xdtoi(s[7]);
827
0
    addr[4] = pcapint_xdtoi(s[8]) << 4 | pcapint_xdtoi(s[9]);
828
0
    addr[5] = pcapint_xdtoi(s[10]) << 4 | pcapint_xdtoi(s[11]);
829
0
    return 1;
830
0
  }
831
0
  return 0;
832
0
}
833
834
// Man page: "xxxx.xxxx.xxxx", regexp: "^[0-9a-fA-F]{4}(\.[0-9a-fA-F]{4}){2}$".
835
static u_char
836
pcapint_atomac48_xxxx_3_times(const char *s, uint8_t *addr)
837
0
{
838
0
  const char sep = '.';
839
0
  if (strlen(s) == 14 &&
840
0
      PCAP_ISXDIGIT(s[0]) &&
841
0
      PCAP_ISXDIGIT(s[1]) &&
842
0
      PCAP_ISXDIGIT(s[2]) &&
843
0
      PCAP_ISXDIGIT(s[3]) &&
844
0
      s[4] == sep &&
845
0
      PCAP_ISXDIGIT(s[5]) &&
846
0
      PCAP_ISXDIGIT(s[6]) &&
847
0
      PCAP_ISXDIGIT(s[7]) &&
848
0
      PCAP_ISXDIGIT(s[8]) &&
849
0
      s[9] == sep &&
850
0
      PCAP_ISXDIGIT(s[10]) &&
851
0
      PCAP_ISXDIGIT(s[11]) &&
852
0
      PCAP_ISXDIGIT(s[12]) &&
853
0
      PCAP_ISXDIGIT(s[13])) {
854
0
    addr[0] = pcapint_xdtoi(s[0]) << 4 | pcapint_xdtoi(s[1]);
855
0
    addr[1] = pcapint_xdtoi(s[2]) << 4 | pcapint_xdtoi(s[3]);
856
0
    addr[2] = pcapint_xdtoi(s[5]) << 4 | pcapint_xdtoi(s[6]);
857
0
    addr[3] = pcapint_xdtoi(s[7]) << 4 | pcapint_xdtoi(s[8]);
858
0
    addr[4] = pcapint_xdtoi(s[10]) << 4 | pcapint_xdtoi(s[11]);
859
0
    addr[5] = pcapint_xdtoi(s[12]) << 4 | pcapint_xdtoi(s[13]);
860
0
    return 1;
861
0
  }
862
0
  return 0;
863
0
}
864
865
/*
866
 * Man page: "xx:xx:xx:xx:xx:xx", regexp: "^[0-9a-fA-F]{1,2}(:[0-9a-fA-F]{1,2}){5}$".
867
 * Man page: "xx-xx-xx-xx-xx-xx", regexp: "^[0-9a-fA-F]{1,2}(-[0-9a-fA-F]{1,2}){5}$".
868
 * Man page: "xx.xx.xx.xx.xx.xx", regexp: "^[0-9a-fA-F]{1,2}(\.[0-9a-fA-F]{1,2}){5}$".
869
 * (Any "xx" above can be "x", which is equivalent to "0x".)
870
 *
871
 * An equivalent (and parametrisable for EUI-64) FSM could be implemented using
872
 * a smaller graph, but that graph would be neither acyclic nor planar nor
873
 * trivial to verify.
874
 *
875
 *                |
876
 *    [.]         v
877
 * +<---------- START
878
 * |              |
879
 * |              | [0-9a-fA-F]
880
 * |  [.]         v
881
 * +<--------- BYTE0_X ----------+
882
 * |              |              |
883
 * |              | [0-9a-fA-F]  |
884
 * |  [.]         v              |
885
 * +<--------- BYTE0_XX          | [:\.-]
886
 * |              |              |
887
 * |              | [:\.-]       |
888
 * |  [.]         v              |
889
 * +<----- BYTE0_SEP_BYTE1 <-----+
890
 * |              |
891
 * |              | [0-9a-fA-F]
892
 * |  [.]         v
893
 * +<--------- BYTE1_X ----------+
894
 * |              |              |
895
 * |              | [0-9a-fA-F]  |
896
 * |  [.]         v              |
897
 * +<--------- BYTE1_XX          | <sep>
898
 * |              |              |
899
 * |              | <sep>        |
900
 * |  [.]         v              |
901
 * +<----- BYTE1_SEP_BYTE2 <-----+
902
 * |              |
903
 * |              | [0-9a-fA-F]
904
 * |  [.]         v
905
 * +<--------- BYTE2_X ----------+
906
 * |              |              |
907
 * |              | [0-9a-fA-F]  |
908
 * |  [.]         v              |
909
 * +<--------- BYTE2_XX          | <sep>
910
 * |              |              |
911
 * |              | <sep>        |
912
 * |  [.]         v              |
913
 * +<----- BYTE2_SEP_BYTE3 <-----+
914
 * |              |
915
 * |              | [0-9a-fA-F]
916
 * |  [.]         v
917
 * +<--------- BYTE3_X ----------+
918
 * |              |              |
919
 * |              | [0-9a-fA-F]  |
920
 * |  [.]         v              |
921
 * +<--------- BYTE3_XX          | <sep>
922
 * |              |              |
923
 * |              | <sep>        |
924
 * |  [.]         v              |
925
 * +<----- BYTE3_SEP_BYTE4 <-----+
926
 * |              |
927
 * |              | [0-9a-fA-F]
928
 * |  [.]         v
929
 * +<--------- BYTE4_X ----------+
930
 * |              |              |
931
 * |              | [0-9a-fA-F]  |
932
 * |  [.]         v              |
933
 * +<--------- BYTE4_XX          | <sep>
934
 * |              |              |
935
 * |              | <sep>        |
936
 * |  [.]         v              |
937
 * +<----- BYTE4_SEP_BYTE5 <-----+
938
 * |              |
939
 * |              | [0-9a-fA-F]
940
 * |  [.]         v
941
 * +<--------- BYTE5_X ----------+
942
 * |              |              |
943
 * |              | [0-9a-fA-F]  |
944
 * |  [.]         v              |
945
 * +<--------- BYTE5_XX          | \0
946
 * |              |              |
947
 * |              | \0           |
948
 * |              |              v
949
 * +--> (reject)  +---------> (accept)
950
 *
951
 */
952
static u_char
953
pcapint_atomac48_x_xx_6_times(const char *s, uint8_t *addr)
954
0
{
955
0
  enum {
956
0
    START,
957
0
    BYTE0_X,
958
0
    BYTE0_XX,
959
0
    BYTE0_SEP_BYTE1,
960
0
    BYTE1_X,
961
0
    BYTE1_XX,
962
0
    BYTE1_SEP_BYTE2,
963
0
    BYTE2_X,
964
0
    BYTE2_XX,
965
0
    BYTE2_SEP_BYTE3,
966
0
    BYTE3_X,
967
0
    BYTE3_XX,
968
0
    BYTE3_SEP_BYTE4,
969
0
    BYTE4_X,
970
0
    BYTE4_XX,
971
0
    BYTE4_SEP_BYTE5,
972
0
    BYTE5_X,
973
0
    BYTE5_XX,
974
0
  } fsm_state = START;
975
0
  uint8_t buf[6];
976
0
  const char *seplist = ":.-";
977
  /*
978
   * XXX - the state diagram indicates that we cannot get to states
979
   * BYTEn_X or BYTEn_XX, for n > 0, without first going through
980
   * states BYTE0_X or BYTE0_XX, but at least some versions of
981
   * MSVC complain that sep may be used uninitialized, so we
982
   * initialize it here.
983
   */
984
0
  char sep = '\0';
985
986
0
  while (*s) {
987
0
    switch (fsm_state) {
988
0
    case START:
989
0
      if (PCAP_ISXDIGIT(*s)) {
990
0
        buf[0] = pcapint_xdtoi(*s);
991
0
        fsm_state = BYTE0_X;
992
0
        break;
993
0
      }
994
0
      goto reject;
995
0
    case BYTE0_X:
996
0
      if (strchr(seplist, *s)) {
997
0
        sep = *s;
998
0
        fsm_state = BYTE0_SEP_BYTE1;
999
0
        break;
1000
0
      }
1001
0
      if (PCAP_ISXDIGIT(*s)) {
1002
0
        buf[0] = buf[0] << 4 | pcapint_xdtoi(*s);
1003
0
        fsm_state = BYTE0_XX;
1004
0
        break;
1005
0
      }
1006
0
      goto reject;
1007
0
    case BYTE0_XX:
1008
0
      if (strchr(seplist, *s)) {
1009
0
        sep = *s;
1010
0
        fsm_state = BYTE0_SEP_BYTE1;
1011
0
        break;
1012
0
      }
1013
0
      goto reject;
1014
0
    case BYTE0_SEP_BYTE1:
1015
0
      if (PCAP_ISXDIGIT(*s)) {
1016
0
        buf[1] = pcapint_xdtoi(*s);
1017
0
        fsm_state = BYTE1_X;
1018
0
        break;
1019
0
      }
1020
0
      goto reject;
1021
0
    case BYTE1_X:
1022
0
      if (*s == sep) {
1023
0
        fsm_state = BYTE1_SEP_BYTE2;
1024
0
        break;
1025
0
      }
1026
0
      if (PCAP_ISXDIGIT(*s)) {
1027
0
        buf[1] = buf[1] << 4 | pcapint_xdtoi(*s);
1028
0
        fsm_state = BYTE1_XX;
1029
0
        break;
1030
0
      }
1031
0
      goto reject;
1032
0
    case BYTE1_XX:
1033
0
      if (*s == sep) {
1034
0
        fsm_state = BYTE1_SEP_BYTE2;
1035
0
        break;
1036
0
      }
1037
0
      goto reject;
1038
0
    case BYTE1_SEP_BYTE2:
1039
0
      if (PCAP_ISXDIGIT(*s)) {
1040
0
        buf[2] = pcapint_xdtoi(*s);
1041
0
        fsm_state = BYTE2_X;
1042
0
        break;
1043
0
      }
1044
0
      goto reject;
1045
0
    case BYTE2_X:
1046
0
      if (*s == sep) {
1047
0
        fsm_state = BYTE2_SEP_BYTE3;
1048
0
        break;
1049
0
      }
1050
0
      if (PCAP_ISXDIGIT(*s)) {
1051
0
        buf[2] = buf[2] << 4 | pcapint_xdtoi(*s);
1052
0
        fsm_state = BYTE2_XX;
1053
0
        break;
1054
0
      }
1055
0
      goto reject;
1056
0
    case BYTE2_XX:
1057
0
      if (*s == sep) {
1058
0
        fsm_state = BYTE2_SEP_BYTE3;
1059
0
        break;
1060
0
      }
1061
0
      goto reject;
1062
0
    case BYTE2_SEP_BYTE3:
1063
0
      if (PCAP_ISXDIGIT(*s)) {
1064
0
        buf[3] = pcapint_xdtoi(*s);
1065
0
        fsm_state = BYTE3_X;
1066
0
        break;
1067
0
      }
1068
0
      goto reject;
1069
0
    case BYTE3_X:
1070
0
      if (*s == sep) {
1071
0
        fsm_state = BYTE3_SEP_BYTE4;
1072
0
        break;
1073
0
      }
1074
0
      if (PCAP_ISXDIGIT(*s)) {
1075
0
        buf[3] = buf[3] << 4 | pcapint_xdtoi(*s);
1076
0
        fsm_state = BYTE3_XX;
1077
0
        break;
1078
0
      }
1079
0
      goto reject;
1080
0
    case BYTE3_XX:
1081
0
      if (*s == sep) {
1082
0
        fsm_state = BYTE3_SEP_BYTE4;
1083
0
        break;
1084
0
      }
1085
0
      goto reject;
1086
0
    case BYTE3_SEP_BYTE4:
1087
0
      if (PCAP_ISXDIGIT(*s)) {
1088
0
        buf[4] = pcapint_xdtoi(*s);
1089
0
        fsm_state = BYTE4_X;
1090
0
        break;
1091
0
      }
1092
0
      goto reject;
1093
0
    case BYTE4_X:
1094
0
      if (*s == sep) {
1095
0
        fsm_state = BYTE4_SEP_BYTE5;
1096
0
        break;
1097
0
      }
1098
0
      if (PCAP_ISXDIGIT(*s)) {
1099
0
        buf[4] = buf[4] << 4 | pcapint_xdtoi(*s);
1100
0
        fsm_state = BYTE4_XX;
1101
0
        break;
1102
0
      }
1103
0
      goto reject;
1104
0
    case BYTE4_XX:
1105
0
      if (*s == sep) {
1106
0
        fsm_state = BYTE4_SEP_BYTE5;
1107
0
        break;
1108
0
      }
1109
0
      goto reject;
1110
0
    case BYTE4_SEP_BYTE5:
1111
0
      if (PCAP_ISXDIGIT(*s)) {
1112
0
        buf[5] = pcapint_xdtoi(*s);
1113
0
        fsm_state = BYTE5_X;
1114
0
        break;
1115
0
      }
1116
0
      goto reject;
1117
0
    case BYTE5_X:
1118
0
      if (PCAP_ISXDIGIT(*s)) {
1119
0
        buf[5] = buf[5] << 4 | pcapint_xdtoi(*s);
1120
0
        fsm_state = BYTE5_XX;
1121
0
        break;
1122
0
      }
1123
0
      goto reject;
1124
0
    case BYTE5_XX:
1125
0
      goto reject;
1126
0
    } // switch
1127
0
    s++;
1128
0
  } // while
1129
1130
0
  if (fsm_state == BYTE5_X || fsm_state == BYTE5_XX) {
1131
    // accept
1132
0
    memcpy(addr, buf, sizeof(buf));
1133
0
    return 1;
1134
0
  }
1135
1136
0
reject:
1137
0
  return 0;
1138
0
}
1139
1140
// The 'addr' argument must point to an array of at least 6 elements.
1141
int
1142
pcapint_atomac48(const char *s, uint8_t *addr)
1143
0
{
1144
0
  return s && (
1145
0
      pcapint_atomac48_xxxxxxxxxxxx(s, addr) ||
1146
0
      pcapint_atomac48_xxxx_3_times(s, addr) ||
1147
0
      pcapint_atomac48_x_xx_6_times(s, addr)
1148
0
  );
1149
0
}
1150
1151
/*
1152
 * If 's' is a MAC-48 address in one of the forms documented in pcap-filter(7)
1153
 * for "ether host", return a pointer to an allocated buffer with the binary
1154
 * value of the address.  Return NULL on any error.
1155
 */
1156
u_char *
1157
pcap_ether_aton(const char *s)
1158
0
{
1159
0
  uint8_t tmp[6];
1160
0
  if (! pcapint_atomac48(s, tmp))
1161
0
    return (NULL);
1162
1163
0
  u_char *e = malloc(6);
1164
0
  if (e == NULL)
1165
0
    return (NULL);
1166
0
  memcpy(e, tmp, sizeof(tmp));
1167
0
  return (e);
1168
0
}
1169
1170
#ifndef HAVE_ETHER_HOSTTON
1171
/*
1172
 * Roll our own.
1173
 *
1174
 * This should be thread-safe, as we define the static variables
1175
 * we use to be thread-local, and as pcap_next_etherent() does so
1176
 * as well.
1177
 */
1178
u_char *
1179
pcap_ether_hostton(const char *name)
1180
{
1181
  struct pcap_etherent *ep;
1182
  u_char *ap;
1183
  static thread_local FILE *fp = NULL;
1184
  static thread_local int init = 0;
1185
1186
  if (!init) {
1187
    fp = fopen(PCAP_ETHERS_FILE, "r");
1188
    ++init;
1189
    if (fp == NULL)
1190
      return (NULL);
1191
  } else if (fp == NULL)
1192
    return (NULL);
1193
  else
1194
    rewind(fp);
1195
1196
  while ((ep = pcap_next_etherent(fp)) != NULL) {
1197
    if (strcmp(ep->name, name) == 0) {
1198
      ap = (u_char *)malloc(6);
1199
      if (ap != NULL) {
1200
        memcpy(ap, ep->addr, 6);
1201
        return (ap);
1202
      }
1203
      break;
1204
    }
1205
  }
1206
  return (NULL);
1207
}
1208
#else
1209
/*
1210
 * Use the OS-supplied routine.
1211
 * This *should* be thread-safe; the API doesn't have a static buffer.
1212
 */
1213
u_char *
1214
pcap_ether_hostton(const char *name)
1215
0
{
1216
0
  u_char *ap;
1217
0
  u_char a[6];
1218
0
  char namebuf[1024];
1219
1220
  /*
1221
   * In AIX 7.1 and 7.2: int ether_hostton(char *, struct ether_addr *);
1222
   */
1223
0
  pcapint_strlcpy(namebuf, name, sizeof(namebuf));
1224
0
  ap = NULL;
1225
0
  if (ether_hostton(namebuf, (struct ether_addr *)a) == 0) {
1226
0
    ap = (u_char *)malloc(6);
1227
0
    if (ap != NULL)
1228
0
      memcpy((char *)ap, (char *)a, 6);
1229
0
  }
1230
0
  return (ap);
1231
0
}
1232
#endif