Coverage Report

Created: 2026-02-26 06:15

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
343
{
180
343
  struct addrinfo hints, *res;
181
343
  int error;
182
183
343
  memset(&hints, 0, sizeof(hints));
184
343
  hints.ai_family = PF_UNSPEC;
185
343
  hints.ai_socktype = SOCK_STREAM;  /*not really*/
186
343
  hints.ai_protocol = IPPROTO_TCP;  /*not really*/
187
343
  error = getaddrinfo(name, NULL, &hints, &res);
188
343
  if (error)
189
23
    return NULL;
190
320
  else
191
320
    return res;
192
343
}
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
13
{
222
  /*
223
   * UN*X.
224
   */
225
13
  struct netent *np;
226
13
  #if defined(HAVE_LINUX_GETNETBYNAME_R)
227
  /*
228
   * We have Linux's reentrant getnetbyname_r().
229
   */
230
13
  struct netent result_buf;
231
13
  char buf[1024]; /* arbitrary size */
232
13
  int h_errnoval;
233
13
  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
13
  np = NULL;
250
13
  err = getnetbyname_r(name, &result_buf, buf, sizeof buf, &np,
251
13
      &h_errnoval);
252
13
  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
13
  if (np != NULL)
292
0
    return np->n_net;
293
13
  else
294
13
    return 0;
295
13
}
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
531
{
306
531
  struct addrinfo hints, *res, *ai;
307
531
  int error;
308
531
  struct sockaddr_in *in4;
309
531
  struct sockaddr_in6 *in6;
310
531
  int tcp_port = -1;
311
531
  int udp_port = -1;
312
313
  /*
314
   * We check for both TCP and UDP in case there are
315
   * ambiguous entries.
316
   */
317
531
  memset(&hints, 0, sizeof(hints));
318
531
  hints.ai_family = PF_UNSPEC;
319
531
  hints.ai_socktype = SOCK_STREAM;
320
531
  hints.ai_protocol = IPPROTO_TCP;
321
531
  error = getaddrinfo(NULL, name, &hints, &res);
322
531
  if (error != 0) {
323
196
    if (error != EAI_NONAME &&
324
195
        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
335
  } else {
333
    /*
334
     * OK, we found it.  Did it find anything?
335
     */
336
335
    for (ai = res; ai != NULL; ai = ai->ai_next) {
337
      /*
338
       * Does it have an address?
339
       */
340
335
      if (ai->ai_addr != NULL) {
341
        /*
342
         * Yes.  Get a port number; we're done.
343
         */
344
335
        if (ai->ai_addr->sa_family == AF_INET) {
345
335
          in4 = (struct sockaddr_in *)ai->ai_addr;
346
335
          tcp_port = ntohs(in4->sin_port);
347
335
          break;
348
335
        }
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
335
    }
356
335
    freeaddrinfo(res);
357
335
  }
358
359
531
  memset(&hints, 0, sizeof(hints));
360
531
  hints.ai_family = PF_UNSPEC;
361
531
  hints.ai_socktype = SOCK_DGRAM;
362
531
  hints.ai_protocol = IPPROTO_UDP;
363
531
  error = getaddrinfo(NULL, name, &hints, &res);
364
531
  if (error != 0) {
365
196
    if (error != EAI_NONAME &&
366
195
        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
335
  } else {
375
    /*
376
     * OK, we found it.  Did it find anything?
377
     */
378
335
    for (ai = res; ai != NULL; ai = ai->ai_next) {
379
      /*
380
       * Does it have an address?
381
       */
382
335
      if (ai->ai_addr != NULL) {
383
        /*
384
         * Yes.  Get a port number; we're done.
385
         */
386
335
        if (ai->ai_addr->sa_family == AF_INET) {
387
335
          in4 = (struct sockaddr_in *)ai->ai_addr;
388
335
          udp_port = ntohs(in4->sin_port);
389
335
          break;
390
335
        }
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
335
    }
398
335
    freeaddrinfo(res);
399
335
  }
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
531
  if (tcp_port >= 0) {
408
335
    *port = tcp_port;
409
335
    *proto = IPPROTO_TCP;
410
335
    if (udp_port >= 0) {
411
335
      if (udp_port == tcp_port)
412
335
        *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
335
    }
421
335
    return 1;
422
335
  }
423
196
  if (udp_port >= 0) {
424
0
    *port = udp_port;
425
0
    *proto = IPPROTO_UDP;
426
0
    return 1;
427
0
  }
428
196
  return 0;
429
196
}
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
291
{
477
291
  struct protoent *p;
478
291
  #if defined(HAVE_LINUX_GETPROTOBYNAME_R)
479
  /*
480
   * We have Linux's reentrant getprotobyname_r().
481
   */
482
291
  struct protoent result_buf;
483
291
  char buf[1024]; // "...1024 bytes should be sufficient for most applications."
484
291
  int err;
485
486
291
  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
291
  if (err != 0) {
502
    /*
503
     * XXX - dynamically allocate the buffer, and make it
504
     * bigger if we get ERANGE back?
505
     */
506
291
    return PROTO_UNDEF;
507
291
  }
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
368
{
587
3.43k
  while (p->s) {
588
3.17k
    if (strcmp(p->s, s) == 0)
589
107
      return p->p;
590
3.06k
    p += 1;
591
3.06k
  }
592
261
  return PROTO_UNDEF;
593
368
}
594
595
int
596
pcap_nametoeproto(const char *s)
597
196
{
598
196
  return proto_or_undef(eproto_db, s);
599
196
}
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
172
{
615
172
  return proto_or_undef(llc_db, s);
616
172
}
617
618
/* Hex digit to 8-bit unsigned integer. */
619
u_char
620
pcapint_xdtoi(const u_char c)
621
21.8k
{
622
21.8k
  if (c >= '0' && c <= '9')
623
7.85k
    return (u_char)(c - '0');
624
13.9k
  else if (c >= 'a' && c <= 'f')
625
7.69k
    return (u_char)(c - 'a' + 10);
626
6.29k
  else
627
6.29k
    return (u_char)(c - 'A' + 10);
628
21.8k
}
629
630
int
631
pcapint_atoin(const char *s, bpf_u_int32 *addr)
632
428
{
633
428
  u_int n;
634
428
  int len;
635
636
428
  *addr = 0;
637
428
  len = 0;
638
884
  for (;;) {
639
884
    n = 0;
640
1.96k
    while (*s && *s != '.') {
641
1.08k
      if (n > 25) {
642
        /* The result will be > 255 */
643
2
        return -1;
644
2
      }
645
1.08k
      n = n * 10 + *s++ - '0';
646
1.08k
    }
647
882
    if (n > 255)
648
1
      return -1;
649
881
    *addr <<= 8;
650
881
    *addr |= n & 0xff;
651
881
    len += 8;
652
881
    if (*s == '\0')
653
425
      return len;
654
456
    ++s;
655
456
  }
656
  /* NOTREACHED */
657
428
}
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
258
{
670
511
#define AREASHIFT 10
671
264
#define AREAMASK 0176000
672
258
#define NODEMASK 01777
673
674
  /* Initialize to squelch a compiler warning only. */
675
258
  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
258
  enum {
686
258
    START,
687
258
    AREA,
688
258
    DOT,
689
258
    NODE,
690
258
    INVALID
691
258
  } fsm_state = START;
692
693
1.51k
  while (*s) {
694
1.26k
    switch (fsm_state) {
695
258
    case START:
696
258
      if (PCAP_ISDIGIT(*s)) {
697
258
        area = *s - '0';
698
258
        fsm_state = AREA;
699
258
        break;
700
258
      }
701
0
      fsm_state = INVALID;
702
0
      break;
703
518
    case AREA:
704
518
      if (*s == '.') {
705
253
        fsm_state = DOT;
706
253
        break;
707
253
      }
708
265
      if (PCAP_ISDIGIT(*s)) {
709
264
        area = area * 10 + *s - '0';
710
264
        if (area <= AREAMASK >> AREASHIFT)
711
260
          break;
712
264
      }
713
5
      fsm_state = INVALID;
714
5
      break;
715
253
    case DOT:
716
253
      if (PCAP_ISDIGIT(*s)) {
717
253
        node = *s - '0';
718
253
        fsm_state = NODE;
719
253
        break;
720
253
      }
721
0
      fsm_state = INVALID;
722
0
      break;
723
224
    case NODE:
724
224
      if (PCAP_ISDIGIT(*s)) {
725
221
        node = node * 10 + *s - '0';
726
221
        if (node <= NODEMASK)
727
218
          break;
728
221
      }
729
6
      fsm_state = INVALID;
730
6
      break;
731
8
    case INVALID:
732
8
      return 0;
733
1.26k
    } /* switch */
734
1.25k
    s++;
735
1.25k
  } /* while */
736
  /*
737
   * This condition is false if the string comes from the lexer, but
738
   * let's not depend on that.
739
   */
740
250
  if (fsm_state != NODE)
741
3
    return 0;
742
743
  // Both 'area' and 'node' have been verified to be in range.
744
247
  *addr = (uint16_t)(area << AREASHIFT | node);
745
247
  return 1;
746
250
}
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
607
{
764
607
  enum {
765
607
    START,
766
607
    DOLLAR,
767
607
    HEX1,
768
607
    HEX2,
769
607
  } fsm_state = START;
770
607
  uint8_t tmp = 0;
771
772
2.13k
  while (*s) {
773
1.52k
    switch (fsm_state) {
774
607
    case START:
775
607
      if (*s != '$')
776
0
        goto invalid;
777
607
      fsm_state = DOLLAR;
778
607
      break;
779
607
    case DOLLAR:
780
607
      if (! PCAP_ISXDIGIT(*s))
781
0
        goto invalid;
782
607
      tmp = pcapint_xdtoi(*s);
783
607
      fsm_state = HEX1;
784
607
      break;
785
315
    case HEX1:
786
315
      if (! PCAP_ISXDIGIT(*s))
787
0
        goto invalid;
788
315
      tmp <<= 4;
789
315
      tmp |= pcapint_xdtoi(*s);
790
315
      fsm_state = HEX2;
791
315
      break;
792
0
    case HEX2:
793
0
      goto invalid;
794
1.52k
    } // switch
795
1.52k
    s++;
796
1.52k
  } // while
797
607
  if (fsm_state == HEX1 || fsm_state == HEX2) {
798
607
    *addr = tmp;
799
607
    return 1;
800
607
  }
801
802
0
invalid:
803
0
  return 0;
804
607
}
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
2.06k
{
810
2.06k
  if (strlen(s) == 12 &&
811
1.34k
      PCAP_ISXDIGIT(s[0]) &&
812
1.34k
      PCAP_ISXDIGIT(s[1]) &&
813
1.03k
      PCAP_ISXDIGIT(s[2]) &&
814
955
      PCAP_ISXDIGIT(s[3]) &&
815
955
      PCAP_ISXDIGIT(s[4]) &&
816
955
      PCAP_ISXDIGIT(s[5]) &&
817
955
      PCAP_ISXDIGIT(s[6]) &&
818
955
      PCAP_ISXDIGIT(s[7]) &&
819
955
      PCAP_ISXDIGIT(s[8]) &&
820
955
      PCAP_ISXDIGIT(s[9]) &&
821
955
      PCAP_ISXDIGIT(s[10]) &&
822
955
      PCAP_ISXDIGIT(s[11])) {
823
955
    addr[0] = pcapint_xdtoi(s[0]) << 4 | pcapint_xdtoi(s[1]);
824
955
    addr[1] = pcapint_xdtoi(s[2]) << 4 | pcapint_xdtoi(s[3]);
825
955
    addr[2] = pcapint_xdtoi(s[4]) << 4 | pcapint_xdtoi(s[5]);
826
955
    addr[3] = pcapint_xdtoi(s[6]) << 4 | pcapint_xdtoi(s[7]);
827
955
    addr[4] = pcapint_xdtoi(s[8]) << 4 | pcapint_xdtoi(s[9]);
828
955
    addr[5] = pcapint_xdtoi(s[10]) << 4 | pcapint_xdtoi(s[11]);
829
955
    return 1;
830
955
  }
831
1.11k
  return 0;
832
2.06k
}
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
1.11k
{
838
1.11k
  const char sep = '.';
839
1.11k
  if (strlen(s) == 14 &&
840
421
      PCAP_ISXDIGIT(s[0]) &&
841
421
      PCAP_ISXDIGIT(s[1]) &&
842
342
      PCAP_ISXDIGIT(s[2]) &&
843
273
      PCAP_ISXDIGIT(s[3]) &&
844
273
      s[4] == sep &&
845
273
      PCAP_ISXDIGIT(s[5]) &&
846
273
      PCAP_ISXDIGIT(s[6]) &&
847
273
      PCAP_ISXDIGIT(s[7]) &&
848
273
      PCAP_ISXDIGIT(s[8]) &&
849
273
      s[9] == sep &&
850
273
      PCAP_ISXDIGIT(s[10]) &&
851
273
      PCAP_ISXDIGIT(s[11]) &&
852
273
      PCAP_ISXDIGIT(s[12]) &&
853
273
      PCAP_ISXDIGIT(s[13])) {
854
273
    addr[0] = pcapint_xdtoi(s[0]) << 4 | pcapint_xdtoi(s[1]);
855
273
    addr[1] = pcapint_xdtoi(s[2]) << 4 | pcapint_xdtoi(s[3]);
856
273
    addr[2] = pcapint_xdtoi(s[5]) << 4 | pcapint_xdtoi(s[6]);
857
273
    addr[3] = pcapint_xdtoi(s[7]) << 4 | pcapint_xdtoi(s[8]);
858
273
    addr[4] = pcapint_xdtoi(s[10]) << 4 | pcapint_xdtoi(s[11]);
859
273
    addr[5] = pcapint_xdtoi(s[12]) << 4 | pcapint_xdtoi(s[13]);
860
273
    return 1;
861
273
  }
862
837
  return 0;
863
1.11k
}
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
837
{
955
837
  enum {
956
837
    START,
957
837
    BYTE0_X,
958
837
    BYTE0_XX,
959
837
    BYTE0_SEP_BYTE1,
960
837
    BYTE1_X,
961
837
    BYTE1_XX,
962
837
    BYTE1_SEP_BYTE2,
963
837
    BYTE2_X,
964
837
    BYTE2_XX,
965
837
    BYTE2_SEP_BYTE3,
966
837
    BYTE3_X,
967
837
    BYTE3_XX,
968
837
    BYTE3_SEP_BYTE4,
969
837
    BYTE4_X,
970
837
    BYTE4_XX,
971
837
    BYTE4_SEP_BYTE5,
972
837
    BYTE5_X,
973
837
    BYTE5_XX,
974
837
  } fsm_state = START;
975
837
  uint8_t buf[6];
976
837
  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
837
  char sep = '\0';
985
986
11.2k
  while (*s) {
987
10.3k
    switch (fsm_state) {
988
837
    case START:
989
837
      if (PCAP_ISXDIGIT(*s)) {
990
837
        buf[0] = pcapint_xdtoi(*s);
991
837
        fsm_state = BYTE0_X;
992
837
        break;
993
837
      }
994
0
      goto reject;
995
837
    case BYTE0_X:
996
837
      if (strchr(seplist, *s)) {
997
634
        sep = *s;
998
634
        fsm_state = BYTE0_SEP_BYTE1;
999
634
        break;
1000
634
      }
1001
203
      if (PCAP_ISXDIGIT(*s)) {
1002
203
        buf[0] = buf[0] << 4 | pcapint_xdtoi(*s);
1003
203
        fsm_state = BYTE0_XX;
1004
203
        break;
1005
203
      }
1006
0
      goto reject;
1007
203
    case BYTE0_XX:
1008
203
      if (strchr(seplist, *s)) {
1009
203
        sep = *s;
1010
203
        fsm_state = BYTE0_SEP_BYTE1;
1011
203
        break;
1012
203
      }
1013
0
      goto reject;
1014
837
    case BYTE0_SEP_BYTE1:
1015
837
      if (PCAP_ISXDIGIT(*s)) {
1016
837
        buf[1] = pcapint_xdtoi(*s);
1017
837
        fsm_state = BYTE1_X;
1018
837
        break;
1019
837
      }
1020
0
      goto reject;
1021
837
    case BYTE1_X:
1022
837
      if (*s == sep) {
1023
667
        fsm_state = BYTE1_SEP_BYTE2;
1024
667
        break;
1025
667
      }
1026
170
      if (PCAP_ISXDIGIT(*s)) {
1027
170
        buf[1] = buf[1] << 4 | pcapint_xdtoi(*s);
1028
170
        fsm_state = BYTE1_XX;
1029
170
        break;
1030
170
      }
1031
0
      goto reject;
1032
170
    case BYTE1_XX:
1033
170
      if (*s == sep) {
1034
170
        fsm_state = BYTE1_SEP_BYTE2;
1035
170
        break;
1036
170
      }
1037
0
      goto reject;
1038
837
    case BYTE1_SEP_BYTE2:
1039
837
      if (PCAP_ISXDIGIT(*s)) {
1040
837
        buf[2] = pcapint_xdtoi(*s);
1041
837
        fsm_state = BYTE2_X;
1042
837
        break;
1043
837
      }
1044
0
      goto reject;
1045
837
    case BYTE2_X:
1046
837
      if (*s == sep) {
1047
656
        fsm_state = BYTE2_SEP_BYTE3;
1048
656
        break;
1049
656
      }
1050
181
      if (PCAP_ISXDIGIT(*s)) {
1051
181
        buf[2] = buf[2] << 4 | pcapint_xdtoi(*s);
1052
181
        fsm_state = BYTE2_XX;
1053
181
        break;
1054
181
      }
1055
0
      goto reject;
1056
181
    case BYTE2_XX:
1057
181
      if (*s == sep) {
1058
181
        fsm_state = BYTE2_SEP_BYTE3;
1059
181
        break;
1060
181
      }
1061
0
      goto reject;
1062
837
    case BYTE2_SEP_BYTE3:
1063
837
      if (PCAP_ISXDIGIT(*s)) {
1064
837
        buf[3] = pcapint_xdtoi(*s);
1065
837
        fsm_state = BYTE3_X;
1066
837
        break;
1067
837
      }
1068
0
      goto reject;
1069
837
    case BYTE3_X:
1070
837
      if (*s == sep) {
1071
658
        fsm_state = BYTE3_SEP_BYTE4;
1072
658
        break;
1073
658
      }
1074
179
      if (PCAP_ISXDIGIT(*s)) {
1075
179
        buf[3] = buf[3] << 4 | pcapint_xdtoi(*s);
1076
179
        fsm_state = BYTE3_XX;
1077
179
        break;
1078
179
      }
1079
0
      goto reject;
1080
179
    case BYTE3_XX:
1081
179
      if (*s == sep) {
1082
179
        fsm_state = BYTE3_SEP_BYTE4;
1083
179
        break;
1084
179
      }
1085
0
      goto reject;
1086
837
    case BYTE3_SEP_BYTE4:
1087
837
      if (PCAP_ISXDIGIT(*s)) {
1088
837
        buf[4] = pcapint_xdtoi(*s);
1089
837
        fsm_state = BYTE4_X;
1090
837
        break;
1091
837
      }
1092
0
      goto reject;
1093
837
    case BYTE4_X:
1094
837
      if (*s == sep) {
1095
605
        fsm_state = BYTE4_SEP_BYTE5;
1096
605
        break;
1097
605
      }
1098
232
      if (PCAP_ISXDIGIT(*s)) {
1099
232
        buf[4] = buf[4] << 4 | pcapint_xdtoi(*s);
1100
232
        fsm_state = BYTE4_XX;
1101
232
        break;
1102
232
      }
1103
0
      goto reject;
1104
232
    case BYTE4_XX:
1105
232
      if (*s == sep) {
1106
232
        fsm_state = BYTE4_SEP_BYTE5;
1107
232
        break;
1108
232
      }
1109
0
      goto reject;
1110
837
    case BYTE4_SEP_BYTE5:
1111
837
      if (PCAP_ISXDIGIT(*s)) {
1112
837
        buf[5] = pcapint_xdtoi(*s);
1113
837
        fsm_state = BYTE5_X;
1114
837
        break;
1115
837
      }
1116
0
      goto reject;
1117
201
    case BYTE5_X:
1118
201
      if (PCAP_ISXDIGIT(*s)) {
1119
201
        buf[5] = buf[5] << 4 | pcapint_xdtoi(*s);
1120
201
        fsm_state = BYTE5_XX;
1121
201
        break;
1122
201
      }
1123
0
      goto reject;
1124
0
    case BYTE5_XX:
1125
0
      goto reject;
1126
10.3k
    } // switch
1127
10.3k
    s++;
1128
10.3k
  } // while
1129
1130
837
  if (fsm_state == BYTE5_X || fsm_state == BYTE5_XX) {
1131
    // accept
1132
837
    memcpy(addr, buf, sizeof(buf));
1133
837
    return 1;
1134
837
  }
1135
1136
0
reject:
1137
0
  return 0;
1138
837
}
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
2.06k
{
1144
2.06k
  return s && (
1145
2.06k
      pcapint_atomac48_xxxxxxxxxxxx(s, addr) ||
1146
1.11k
      pcapint_atomac48_xxxx_3_times(s, addr) ||
1147
837
      pcapint_atomac48_x_xx_6_times(s, addr)
1148
2.06k
  );
1149
2.06k
}
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
98
{
1216
98
  u_char *ap;
1217
98
  u_char a[6];
1218
98
  char namebuf[1024];
1219
1220
  /*
1221
   * In AIX 7.1 and 7.2: int ether_hostton(char *, struct ether_addr *);
1222
   */
1223
98
  pcapint_strlcpy(namebuf, name, sizeof(namebuf));
1224
98
  ap = NULL;
1225
98
  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
98
  return (ap);
1231
98
}
1232
#endif