2020-12-03 00:58:52 -08:00
|
|
|
/* $OpenBSD: util.c,v 1.31 2020/12/03 08:58:52 mvs Exp $ */
|
2000-10-03 07:31:54 -07:00
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
1999-09-16 13:58:44 -07:00
|
|
|
* Copyright (c) 1990, 1991, 1993, 1994, 1995, 1996, 1997
|
1995-10-18 01:37:01 -07:00
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that: (1) source code distributions
|
|
|
|
* retain the above copyright notice and this paragraph in its entirety, (2)
|
|
|
|
* distributions including binary code include the above copyright notice and
|
|
|
|
* this paragraph in its entirety in the documentation or other materials
|
|
|
|
* provided with the distribution, and (3) all advertising materials mentioning
|
|
|
|
* features or use of this software display the following acknowledgement:
|
|
|
|
* ``This product includes software developed by the University of California,
|
|
|
|
* Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
|
|
|
|
* the University nor the names of its contributors may be used to endorse
|
|
|
|
* or promote products derived from this software without specific prior
|
|
|
|
* written permission.
|
|
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/file.h>
|
2004-01-28 11:44:55 -08:00
|
|
|
#include <sys/limits.h>
|
1995-10-18 01:37:01 -07:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#include <ctype.h>
|
2004-01-28 11:44:55 -08:00
|
|
|
#include <err.h>
|
1996-12-12 08:22:20 -08:00
|
|
|
#include <errno.h>
|
1996-07-13 04:01:05 -07:00
|
|
|
#ifdef HAVE_FCNTL_H
|
1995-10-18 01:37:01 -07:00
|
|
|
#include <fcntl.h>
|
|
|
|
#endif
|
1996-12-12 08:22:20 -08:00
|
|
|
#include <pcap.h>
|
1995-10-18 01:37:01 -07:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdarg.h>
|
1996-07-13 04:01:05 -07:00
|
|
|
#include <stdlib.h>
|
1995-10-18 01:37:01 -07:00
|
|
|
#include <string.h>
|
1996-07-13 04:01:05 -07:00
|
|
|
#ifdef TIME_WITH_SYS_TIME
|
|
|
|
#include <time.h>
|
|
|
|
#endif
|
1995-10-18 01:37:01 -07:00
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "interface.h"
|
2004-01-28 11:44:55 -08:00
|
|
|
#include "privsep.h"
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Print out a filename (or other ascii string).
|
|
|
|
* If ep is NULL, assume no truncation check is needed.
|
|
|
|
* Return true if truncated.
|
|
|
|
*/
|
|
|
|
int
|
2015-11-15 16:16:39 -08:00
|
|
|
fn_print(const u_char *s, const u_char *ep)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2015-11-15 16:16:39 -08:00
|
|
|
int ret;
|
|
|
|
u_char c;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
ret = 1; /* assume truncated */
|
|
|
|
while (ep == NULL || s < ep) {
|
|
|
|
c = *s++;
|
|
|
|
if (c == '\0') {
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!isascii(c)) {
|
|
|
|
c = toascii(c);
|
|
|
|
putchar('M');
|
|
|
|
putchar('-');
|
|
|
|
}
|
|
|
|
if (!isprint(c)) {
|
|
|
|
c ^= 0x40; /* DEL to ?, others to alpha */
|
|
|
|
putchar('^');
|
|
|
|
}
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
return(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Print out a counted filename (or other ascii string).
|
|
|
|
* If ep is NULL, assume no truncation check is needed.
|
|
|
|
* Return true if truncated.
|
|
|
|
*/
|
|
|
|
int
|
2015-11-15 16:16:39 -08:00
|
|
|
fn_printn(const u_char *s, u_int n, const u_char *ep)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2015-11-15 16:16:39 -08:00
|
|
|
int ret;
|
|
|
|
u_char c;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
ret = 1; /* assume truncated */
|
|
|
|
while (ep == NULL || s < ep) {
|
|
|
|
if (n-- <= 0) {
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
c = *s++;
|
|
|
|
if (!isascii(c)) {
|
|
|
|
c = toascii(c);
|
|
|
|
putchar('M');
|
|
|
|
putchar('-');
|
|
|
|
}
|
|
|
|
if (!isprint(c)) {
|
|
|
|
c ^= 0x40; /* DEL to ?, others to alpha */
|
|
|
|
putchar('^');
|
|
|
|
}
|
|
|
|
putchar(c);
|
|
|
|
}
|
|
|
|
return(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Print the timestamp
|
|
|
|
*/
|
|
|
|
void
|
2015-11-15 16:16:39 -08:00
|
|
|
ts_print(const struct bpf_timeval *tvp)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2015-11-15 16:16:39 -08:00
|
|
|
int s;
|
2001-09-03 06:25:53 -07:00
|
|
|
#define TSBUFLEN 32
|
|
|
|
static char buf[TSBUFLEN];
|
2019-01-25 16:53:57 -08:00
|
|
|
static struct timeval last;
|
|
|
|
struct timeval diff, cur;
|
2001-09-03 06:25:53 -07:00
|
|
|
time_t t;
|
|
|
|
|
2007-08-28 05:52:06 -07:00
|
|
|
if (Iflag && device)
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%s ", device);
|
2001-09-03 06:25:53 -07:00
|
|
|
switch(tflag){
|
|
|
|
case 0:
|
|
|
|
break;
|
|
|
|
case -1:
|
|
|
|
/* Unix timeval style */
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%u.%06u ", tvp->tv_sec, tvp->tv_usec);
|
2001-09-03 06:25:53 -07:00
|
|
|
break;
|
|
|
|
case -2:
|
2019-01-25 16:53:57 -08:00
|
|
|
t = tvp->tv_sec;
|
2004-01-28 11:44:55 -08:00
|
|
|
strftime(buf, TSBUFLEN, "%b %d %T", priv_localtime(&t));
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%s.%06u ", buf, tvp->tv_usec);
|
2001-09-03 06:25:53 -07:00
|
|
|
break;
|
2019-01-25 16:53:57 -08:00
|
|
|
case -3: /* last frame time delta */
|
|
|
|
case -4: /* first frame time delta */
|
|
|
|
cur.tv_sec = tvp->tv_sec;
|
|
|
|
cur.tv_usec = tvp->tv_usec;
|
|
|
|
timersub(&cur, &last, &diff);
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%lld.%06ld ", diff.tv_sec, diff.tv_usec);
|
2019-01-25 16:53:57 -08:00
|
|
|
if (!timerisset(&last) || tflag == -3)
|
|
|
|
last = cur;
|
2004-06-22 23:12:07 -07:00
|
|
|
break;
|
2001-09-03 06:25:53 -07:00
|
|
|
default:
|
1995-10-18 01:37:01 -07:00
|
|
|
s = (tvp->tv_sec + thiszone) % 86400;
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%02d:%02d:%02d.%06u ",
|
2019-01-25 16:53:57 -08:00
|
|
|
s / 3600, (s % 3600) / 60, s % 60, tvp->tv_usec);
|
2001-09-03 06:25:53 -07:00
|
|
|
break;
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-03-05 14:34:00 -08:00
|
|
|
/*
|
|
|
|
* Print a relative number of seconds (e.g. hold time, prune timer)
|
|
|
|
* in the form 5m1s. This does no truncation, so 32230861 seconds
|
|
|
|
* is represented as 1y1w1d1h1m1s.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
relts_print(int secs)
|
|
|
|
{
|
|
|
|
static char *lengths[] = {"y", "w", "d", "h", "m", "s"};
|
|
|
|
static int seconds[] = {31536000, 604800, 86400, 3600, 60, 1};
|
|
|
|
char **l = lengths;
|
|
|
|
int *s = seconds;
|
|
|
|
|
|
|
|
if (secs <= 0) {
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("0s");
|
2001-03-05 14:34:00 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
while (secs > 0) {
|
|
|
|
if (secs >= *s) {
|
2020-01-24 14:46:36 -08:00
|
|
|
printf("%d%s", secs / *s, *l);
|
2001-03-05 14:34:00 -08:00
|
|
|
secs -= (secs / *s) * *s;
|
|
|
|
}
|
|
|
|
s++;
|
|
|
|
l++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Convert a token value to a string; use "fmt" if not found.
|
|
|
|
*/
|
|
|
|
const char *
|
2015-11-15 16:16:39 -08:00
|
|
|
tok2str(const struct tok *lp, const char *fmt, int v)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
static char buf[128];
|
|
|
|
|
|
|
|
while (lp->s != NULL) {
|
|
|
|
if (lp->v == v)
|
|
|
|
return (lp->s);
|
|
|
|
++lp;
|
|
|
|
}
|
|
|
|
if (fmt == NULL)
|
|
|
|
fmt = "#%d";
|
2000-10-31 08:06:46 -08:00
|
|
|
(void)snprintf(buf, sizeof(buf), fmt, v);
|
1995-10-18 01:37:01 -07:00
|
|
|
return (buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
__dead void
|
1996-07-13 04:01:05 -07:00
|
|
|
error(const char *fmt, ...)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
(void)fprintf(stderr, "%s: ", program_name);
|
|
|
|
va_start(ap, fmt);
|
|
|
|
(void)vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
2006-03-13 11:05:56 -08:00
|
|
|
(void)fputc('\n', stderr);
|
1995-10-18 01:37:01 -07:00
|
|
|
exit(1);
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
1996-07-13 04:01:05 -07:00
|
|
|
warning(const char *fmt, ...)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
1996-12-12 08:22:20 -08:00
|
|
|
(void)fprintf(stderr, "%s: WARNING: ", program_name);
|
1995-10-18 01:37:01 -07:00
|
|
|
va_start(ap, fmt);
|
|
|
|
(void)vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
2006-03-13 11:05:56 -08:00
|
|
|
(void)fputc('\n', stderr);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
|
1995-10-18 01:37:01 -07:00
|
|
|
/*
|
|
|
|
* Copy arg vector into a new buffer, concatenating arguments with spaces.
|
|
|
|
*/
|
|
|
|
char *
|
2004-01-28 11:44:55 -08:00
|
|
|
copy_argv(char * const *argv)
|
1995-10-18 01:37:01 -07:00
|
|
|
{
|
2004-01-28 11:44:55 -08:00
|
|
|
size_t len = 0, n;
|
1995-10-18 01:37:01 -07:00
|
|
|
char *buf;
|
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
if (argv == NULL)
|
|
|
|
return (NULL);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
for (n = 0; argv[n]; n++)
|
|
|
|
len += strlen(argv[n])+1;
|
|
|
|
if (len == 0)
|
|
|
|
return (NULL);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
buf = malloc(len);
|
1996-07-13 04:01:05 -07:00
|
|
|
if (buf == NULL)
|
2004-01-28 11:44:55 -08:00
|
|
|
return (NULL);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
strlcpy(buf, argv[0], len);
|
|
|
|
for (n = 1; argv[n]; n++) {
|
|
|
|
strlcat(buf, " ", len);
|
|
|
|
strlcat(buf, argv[n], len);
|
|
|
|
}
|
|
|
|
return (buf);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
char *
|
|
|
|
read_infile(char *fname)
|
|
|
|
{
|
2004-01-28 11:44:55 -08:00
|
|
|
struct stat buf;
|
|
|
|
int fd;
|
|
|
|
ssize_t cc;
|
|
|
|
size_t bs;
|
|
|
|
char *cp;
|
1995-10-18 01:37:01 -07:00
|
|
|
|
|
|
|
fd = open(fname, O_RDONLY);
|
2019-06-28 06:32:41 -07:00
|
|
|
if (fd == -1)
|
1996-12-12 08:22:20 -08:00
|
|
|
error("can't open %s: %s", fname, pcap_strerror(errno));
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2019-06-28 06:32:41 -07:00
|
|
|
if (fstat(fd, &buf) == -1)
|
1996-12-12 08:22:20 -08:00
|
|
|
error("can't stat %s: %s", fname, pcap_strerror(errno));
|
1995-10-18 01:37:01 -07:00
|
|
|
|
2004-01-28 11:44:55 -08:00
|
|
|
if (buf.st_size >= SSIZE_MAX)
|
|
|
|
error("file too long");
|
|
|
|
|
|
|
|
bs = buf.st_size;
|
|
|
|
cp = malloc(bs + 1);
|
|
|
|
if (cp == NULL)
|
|
|
|
err(1, NULL);
|
|
|
|
cc = read(fd, cp, bs);
|
|
|
|
if (cc == -1)
|
1996-12-12 08:22:20 -08:00
|
|
|
error("read %s: %s", fname, pcap_strerror(errno));
|
2004-01-28 11:44:55 -08:00
|
|
|
if (cc != bs)
|
|
|
|
error("short read %s (%ld != %lu)", fname, (long)cc,
|
|
|
|
(unsigned long)bs);
|
|
|
|
cp[bs] = '\0';
|
2006-04-22 10:24:33 -07:00
|
|
|
close(fd);
|
1995-10-18 01:37:01 -07:00
|
|
|
|
1996-12-12 08:22:20 -08:00
|
|
|
return (cp);
|
1995-10-18 01:37:01 -07:00
|
|
|
}
|
2001-03-05 14:34:00 -08:00
|
|
|
|
|
|
|
void
|
|
|
|
safeputs(const char *s)
|
|
|
|
{
|
|
|
|
while (*s) {
|
|
|
|
safeputchar(*s);
|
|
|
|
s++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
safeputchar(int c)
|
|
|
|
{
|
2020-12-03 00:58:52 -08:00
|
|
|
c &= 0xff;
|
2001-03-05 14:34:00 -08:00
|
|
|
if (c < 0x80 && isprint(c))
|
2020-12-03 00:58:52 -08:00
|
|
|
putchar(c);
|
2001-03-05 14:34:00 -08:00
|
|
|
else
|
2020-12-03 00:58:52 -08:00
|
|
|
printf("\\%03o", c);
|
2001-03-05 14:34:00 -08:00
|
|
|
}
|
2005-03-07 08:13:38 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Print a value a la the %b format of the kernel's printf
|
|
|
|
* (from sbin/ifconfig/ifconfig.c)
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
printb(char *s, unsigned short v, char *bits)
|
|
|
|
{
|
|
|
|
int i, any = 0;
|
|
|
|
char c;
|
|
|
|
|
|
|
|
if (bits && *bits == 8)
|
|
|
|
printf("%s=%o", s, v);
|
|
|
|
else
|
|
|
|
printf("%s=%x", s, v);
|
2008-09-19 01:52:16 -07:00
|
|
|
|
2005-03-07 08:13:38 -08:00
|
|
|
if (bits) {
|
2008-09-19 01:52:16 -07:00
|
|
|
bits++;
|
2005-03-07 08:13:38 -08:00
|
|
|
putchar('<');
|
|
|
|
while ((i = *bits++)) {
|
|
|
|
if (v & (1 << (i-1))) {
|
|
|
|
if (any)
|
|
|
|
putchar(',');
|
|
|
|
any = 1;
|
|
|
|
for (; (c = *bits) > 32; bits++)
|
|
|
|
putchar(c);
|
|
|
|
} else
|
|
|
|
for (; *bits > 32; bits++)
|
|
|
|
;
|
|
|
|
}
|
|
|
|
putchar('>');
|
|
|
|
}
|
|
|
|
}
|