NetBSD/usr.sbin/tcpdump/savefile.c

307 lines
7.4 KiB
C

/*
* Copyright (c) 1990,1991 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.
*
* $Id: savefile.c,v 1.2 1994/12/23 17:06:41 cgd Exp $
*/
#ifndef lint
static char rcsid[] =
"@(#) Header: savefile.c,v 1.27 92/01/26 21:29:26 mccanne Exp (LBL)";
#endif
/*
* savefile.c - supports offline use of tcpdump
* Extraction/creation by Jeffrey Mogul, DECWRL
* Modified by Steve McCanne, LBL.
*
* Used to save the received packet headers, after filtering, to
* a file, and then read them later.
* The first record in the file contains saved values for the machine
* dependent values so we can print the dump file on any architecture.
*/
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/time.h>
#include <net/bpf.h>
#include "version.h"
#include "savefile.h"
#define TCPDUMP_MAGIC 0xa1b2c3d4
/*
* The first record in the file contains saved values for some
* of the flags used in the printout phases of tcpdump.
* Many fields here are longs so compilers won't insert unwanted
* padding; these files need to be interchangeable across architectures.
*/
struct file_header {
u_long magic;
u_short version_major;
u_short version_minor;
long thiszone; /* gmt to local correction */
u_long sigfigs; /* accuracy of timestamps */
u_long snaplen; /* max length saved portion of each pkt */
u_long linktype;
};
int sf_swapped;
FILE *sf_readfile;
FILE *sf_writefile;
static int
sf_write_header(fp, linktype, thiszone, snaplen, precision)
FILE *fp;
int linktype;
int thiszone;
int snaplen;
int precision;
{
struct file_header hdr;
hdr.magic = TCPDUMP_MAGIC;
hdr.version_major = VERSION_MAJOR;
hdr.version_minor = VERSION_MINOR;
hdr.thiszone = thiszone;
hdr.snaplen = snaplen;
hdr.sigfigs = precision;
hdr.linktype = linktype;
if (fwrite((char *)&hdr, sizeof(hdr), 1, fp) != 1)
return -1;
return 0;
}
static void
swap_hdr(hp)
struct file_header *hp;
{
hp->version_major = SWAPSHORT(hp->version_major);
hp->version_minor = SWAPSHORT(hp->version_minor);
hp->thiszone = SWAPLONG(hp->thiszone);
hp->sigfigs = SWAPLONG(hp->sigfigs);
hp->snaplen = SWAPLONG(hp->snaplen);
hp->linktype = SWAPLONG(hp->linktype);
}
int
sf_read_init(fname, linktypep, thiszonep, snaplenp, precision)
char *fname;
int *linktypep, *thiszonep, *snaplenp, *precision;
{
register FILE *fp;
struct file_header hdr;
if (fname[0] == '-' && fname[1] == '\0')
fp = stdin;
else {
fp = fopen(fname, "r");
if (fp == 0) {
(void) fprintf(stderr, "tcpdump: fopen: ");
perror(fname);
exit(1);
}
}
if (fread((char *)&hdr, sizeof(hdr), 1, fp) != 1) {
(void) fprintf(stderr, "tcpdump: fread: ");
perror(fname);
exit(1);
}
if (hdr.magic != TCPDUMP_MAGIC) {
if (SWAPLONG(hdr.magic) != TCPDUMP_MAGIC)
return SFERR_BADF;
sf_swapped = 1;
swap_hdr(&hdr);
}
if (hdr.version_major < VERSION_MAJOR)
return SFERR_BADVERSION;
*thiszonep = hdr.thiszone;
*snaplenp = hdr.snaplen;
*linktypep = hdr.linktype;
*precision = hdr.sigfigs;
sf_readfile = fp;
return 0;
}
/*
* Print out packets stored in the file initilized by sf_read_init().
* If cnt >= 0, return after 'cnt' packets, otherwise continue until eof.
*/
int
sf_read(filtp, cnt, snaplen, printit)
struct bpf_program *filtp;
int cnt, snaplen;
void (*printit)();
{
struct packet_header h;
u_char *buf;
struct bpf_insn *fcode = filtp->bf_insns;
int status = 0;
buf = (u_char *)malloc(snaplen);
while (status == 0) {
status = sf_next_packet(&h, buf, snaplen);
if (status)
break;
/*
* XXX It's possible (and likely) for us to screw up the
* network layer alignment when we pass down packets from
* this point (ip_print deals by copying the ip header
* to an aligned buffer). There doesn't seem to be a
* clean way to fix this. We could compute an offset
* from the link type (which would have to be passed in),
* but that only works for fixed size headers.
*/
if (bpf_filter(fcode, buf, h.len, h.caplen)) {
if (cnt >= 0 && --cnt < 0)
break;
(*printit)(buf, &h.ts, h.len, h.caplen);
}
}
if (status == SFERR_EOF)
/* treat EOF's as okay status */
status = 0;
free((char *)buf);
return status;
}
/*
* Read sf_readfile and return the next packet. Return the header in hdr
* and the contents in buf. Return 0 on success, SFERR_EOF if there were
* no more packets, and SFERR_TRUNC if a partial packet was encountered.
*/
int
sf_next_packet(hdr, buf, buflen)
struct packet_header *hdr;
u_char *buf;
int buflen;
{
FILE *fp = sf_readfile;
/* read the stamp */
if (fread((char *)hdr, sizeof(struct packet_header), 1, fp) != 1) {
/* probably an EOF, though could be a truncated packet */
return SFERR_EOF;
}
if (sf_swapped) {
/* these were written in opposite byte order */
hdr->caplen = SWAPLONG(hdr->caplen);
hdr->len = SWAPLONG(hdr->len);
hdr->ts.tv_sec = SWAPLONG(hdr->ts.tv_sec);
hdr->ts.tv_usec = SWAPLONG(hdr->ts.tv_usec);
}
if (hdr->caplen > buflen)
return SFERR_BADF;
/* read the packet itself */
if (fread((char *)buf, hdr->caplen, 1, fp) != 1)
return SFERR_TRUNC;
return 0;
}
/*
* Initialize so that sf_write() will output to the file named 'fname'.
*/
void
sf_write_init(fname, linktype, thiszone, snaplen, precision)
char *fname;
int linktype;
int thiszone;
int snaplen;
int precision;
{
if (fname[0] == '-' && fname[1] == '\0')
sf_writefile = stdout;
else {
sf_writefile = fopen(fname, "w");
if (sf_writefile == 0) {
(void) fprintf(stderr, "tcpdump: fopen: ");
perror(fname);
exit(1);
}
}
(void)sf_write_header(sf_writefile,
linktype, thiszone, snaplen, precision);
}
/*
* Output a packet to the intialized dump file.
*/
void
sf_write(sp, tvp, length, caplen)
u_char *sp;
struct timeval *tvp;
int length;
int caplen;
{
struct packet_header h;
h.ts.tv_sec = tvp->tv_sec;
h.ts.tv_usec = tvp->tv_usec;
h.len = length;
h.caplen = caplen;
(void)fwrite((char *)&h, sizeof h, 1, sf_writefile);
(void)fwrite((char *)sp, caplen, 1, sf_writefile);
}
void
sf_err(code)
int code;
{
switch (code) {
case SFERR_BADVERSION:
error("archaic file format");
/* NOTREACHED */
case SFERR_BADF:
error("bad dump file format");
/* NOTREACHED */
case SFERR_TRUNC:
error("truncated dump file");
/* NOTREACHED */
case SFERR_EOF:
error("EOF reading dump file");
/* NOTREACHED */
default:
error("unknown dump file error code in sf_err()");
/* NOTREACHED */
}
abort();
}