Get rid of some useless lines from the last patch
[obnox/wireshark/wip.git] / capinfos.c
1 /* capinfos.c
2  * Reports capture file information including # of packets, duration, others
3  *
4  * Copyright 2004 Ian Schorr
5  *
6  * $Id$
7  *
8  * Ethereal - Network traffic analyzer
9  * By Gerald Combs <gerald@ethereal.com>
10  * Copyright 1998 Gerald Combs
11  *
12  * This program is free software; you can redistribute it and/or
13  * modify it under the terms of the GNU General Public License
14  * as published by the Free Software Foundation; either version 2
15  * of the License, or (at your option) any later version.
16  *
17  * This program is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20  * GNU General Public License for more details.
21  *
22  * You should have received a copy of the GNU General Public License
23  * along with this program; if not, write to the Free Software
24  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
25  */
26
27 #ifdef HAVE_CONFIG_H
28 #include "config.h"
29 #endif
30
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34 #include <errno.h>
35
36 #ifdef HAVE_UNISTD_H
37 #include <unistd.h>
38 #endif
39
40 #ifdef HAVE_SYS_TIME_H
41 #include <sys/time.h>
42 #endif
43
44 #ifdef HAVE_SYS_STAT_H
45 #include <sys/stat.h>
46 #endif
47
48 #include <glib.h>
49
50 #include <epan/packet.h>
51 #include "wtap.h"
52
53 #ifdef NEED_GETOPT_H
54 #include "getopt.h"
55 #endif
56
57 static gboolean cap_file_type = FALSE;      /* Do not report capture type     */
58 static gboolean cap_packet_count = FALSE;   /* Do not produce packet count    */
59 static gboolean cap_file_size = FALSE;      /* Do not report file size        */
60 static gboolean cap_data_size = FALSE;      /* Do not report packet byte size */
61 static gboolean cap_duration = FALSE;       /* Do not report capture duration */
62 static gboolean cap_start_time = FALSE;
63 static gboolean cap_end_time = FALSE;
64
65 static gboolean cap_data_rate_byte = FALSE;
66 static gboolean cap_data_rate_bit = FALSE;
67 static gboolean cap_packet_size = FALSE;
68
69
70 typedef struct _capture_info {
71         const char              *filename;
72         guint16                 file_type;
73         guint64                 filesize;
74         guint64                 packet_bytes;
75         double                  start_time;
76         double                  stop_time;
77         guint32                 packet_count;
78         gboolean                snap_set;
79         guint32                 snaplen;
80         gboolean                drops_known;
81         guint32                 drop_count;
82         
83         double                  duration;
84         double                  packet_rate;
85         double                  packet_size;
86         double                  data_rate;              /* in bytes */
87 } capture_info;
88
89 static double
90 secs_usecs(guint32 s, guint32 us)
91 {
92   return (us / 1000000.0) + (double)s;
93 }
94
95 static void
96 print_stats(capture_info *cf_info)
97 {
98   const gchar           *file_type_string;
99   time_t                start_time_t;
100   time_t                stop_time_t;
101
102   /* Build printable strings for various stats */
103   file_type_string = wtap_file_type_string(cf_info->file_type);
104   start_time_t = (time_t)cf_info->start_time;
105   stop_time_t = (time_t)cf_info->stop_time;
106
107   if (cap_file_type) printf("File type: %s\n", file_type_string);
108   if (cap_packet_count) printf("Number of packets: %u \n", cf_info->packet_count);
109   if (cap_file_size) printf("File size: %" PRIu64 " bytes\n", cf_info->filesize);
110   if (cap_data_size) printf("Data size: %" PRIu64 " bytes\n", cf_info->packet_bytes);
111   if (cap_duration) printf("Capture duration: %f seconds\n", cf_info->duration);
112   if (cap_start_time) printf("Start time: %s", ctime (&start_time_t));
113   if (cap_end_time) printf("End time: %s", ctime (&stop_time_t));
114   if (cap_data_rate_byte) printf("Data rate: %.2f bytes/s\n", cf_info->data_rate);
115   if (cap_data_rate_bit) printf("Data rate: %.2f bits/s\n", cf_info->data_rate*8);
116   if (cap_packet_size) printf("Average packet size: %.2f bytes\n", cf_info->packet_size);
117 }
118
119 static int 
120 process_cap_file(wtap *wth, const char *filename)
121 {
122   int                   err;
123   gchar                 *err_info;
124   struct stat   cf_stat;
125   long                  data_offset;
126   
127   guint32               packet = 0;
128   gint64                bytes = 0;
129   const struct wtap_pkthdr *phdr;
130   capture_info  cf_info;
131   double                start_time = 0;
132   double                stop_time = 0;
133   double                cur_time = 0;
134   
135   /* Tally up data that we need to parse through the file to find */
136   while (wtap_read(wth, &err, &err_info, &data_offset))  {
137     phdr = wtap_phdr(wth);
138     cur_time = secs_usecs(phdr->ts.tv_sec, phdr->ts.tv_usec);
139     if(packet==0) {
140       start_time = cur_time;
141       stop_time = cur_time;
142     }
143     if (cur_time < start_time) {
144       start_time = cur_time;
145     }
146     if (cur_time > stop_time) {
147       stop_time = cur_time;
148     }
149     bytes+=phdr->len;
150     packet++;
151   }
152   
153   if (err != 0) {
154     fprintf(stderr,
155             "capinfos: An error occurred after reading %u packets from \"%s\": %s.\n",
156             packet, filename, wtap_strerror(err));
157     switch (err) {
158
159     case WTAP_ERR_UNSUPPORTED:
160     case WTAP_ERR_UNSUPPORTED_ENCAP:
161     case WTAP_ERR_BAD_RECORD:
162       fprintf(stderr, "(%s)\n", err_info);
163       break;
164     }
165     return 1;
166   }
167
168   /* File size */
169   if (fstat(wtap_fd(wth), &cf_stat) < 0) {
170     fprintf(stderr,
171             "capinfos: Can't fstat \"%s\": %s.\n",
172             filename, strerror(errno));
173     return 1;
174   }
175   
176   cf_info.filesize = cf_stat.st_size;
177   
178   /* File Type */
179   cf_info.file_type = wtap_file_type(wth);
180   
181   /* # of packets */
182   cf_info.packet_count = packet;
183   
184   /* File Times */
185   cf_info.start_time = start_time;
186   cf_info.stop_time = stop_time;
187   cf_info.duration = stop_time-start_time;
188
189   /* Number of packet bytes */
190   cf_info.packet_bytes = bytes;
191   
192   /* Data rate per second */
193   cf_info.data_rate = (double)bytes / (stop_time-start_time);
194   
195   /* Avg packet size */
196   cf_info.packet_size = (double)bytes/packet;
197   
198   printf("File name: %s\n", filename);
199   print_stats(&cf_info);
200
201   return 0;
202 }
203
204 static void usage(gboolean is_error)
205 {
206   FILE *output;
207   
208   if (!is_error) {
209     output = stdout;
210     /* XXX - add capinfos header info here */
211   }
212   else {
213     output = stderr;
214   }
215
216
217   fprintf(output, "Usage: capinfos [-t] [-c] [-s] [-d] [-u] [-a] [-e] [-y]\n");
218   fprintf(output, "                [-i] [-z] [-h] <capfile>\n");
219   fprintf(output, "  where\t-t display the capture type of <capfile>\n");
220   fprintf(output, "       \t-c count the number of packets\n");
221   fprintf(output, "       \t-s display the size of the file \n");
222   fprintf(output, "       \t-d display the total length of all packets in the file\n");
223   fprintf(output, "       \t   (in bytes)\n");
224   fprintf(output, "       \t-u display the capture duration (in seconds) \n");
225   fprintf(output, "       \t-a display the capture start time\n");
226   fprintf(output, "       \t-e display the capture end time\n");
227   fprintf(output, "       \t-y display average data rate (in bytes)\n");
228   fprintf(output, "       \t-i display average data rate (in bits)\n");
229   fprintf(output, "       \t-z display average packet size (in bytes)\n");
230   fprintf(output, "       \t-h produces this help listing.\n");
231   fprintf(output, "\n      \t    If no data flags are given, default is to display all statistics\n");
232 }
233
234 int main(int argc, char *argv[])
235 {
236   wtap *wth;
237   int err;
238   gchar *err_info;
239   extern char *optarg;
240   extern int optind;
241   int opt;
242   int status = 0;
243
244   /* Process the options first */
245
246   while ((opt = getopt(argc, argv, "tcsduaeyizvh")) !=-1) {
247
248     switch (opt) {
249
250     case 't':
251       cap_file_type = TRUE;
252       break;
253
254     case 'c':
255       cap_packet_count = TRUE;
256       break;
257
258     case 's':
259       cap_file_size = TRUE;
260       break;
261
262     case 'd':
263       cap_data_size = TRUE;
264       break;
265
266     case 'u':
267       cap_duration = TRUE;
268       break;
269
270     case 'a':
271       cap_start_time = TRUE;
272       break;
273
274     case 'e':
275       cap_end_time = TRUE;
276       break;
277
278     case 'y':
279       cap_data_rate_byte = TRUE;
280       break;
281
282     case 'i':
283       cap_data_rate_bit = TRUE;
284       break;
285
286     case 'z':
287       cap_packet_size = TRUE;
288       break;
289
290     case 'h':
291       usage(FALSE);
292       exit(1);
293       break;
294
295     case '?':              /* Bad flag - print usage message */
296       usage(TRUE);
297       exit(1);
298       break;
299     }
300   }
301
302   if (optind < 2) {
303
304     /* If no arguments were given, by default display all statistics */
305     cap_file_type = TRUE;      
306     cap_packet_count = TRUE;   
307     cap_file_size = TRUE;      
308     cap_data_size = TRUE;      
309     cap_duration = TRUE;       
310     cap_start_time = TRUE;
311     cap_end_time = TRUE;
312
313     cap_data_rate_byte = TRUE;
314     cap_data_rate_bit = TRUE;
315     cap_packet_size = TRUE;
316   }
317   
318   if ((argc - optind) < 1) {
319     usage(TRUE);
320     exit(1);
321   }
322
323   for (opt = optind; opt < argc; opt++) {
324   
325     wth = wtap_open_offline(argv[opt], &err, &err_info, FALSE);
326
327     if (!wth) {
328       fprintf(stderr, "capinfos: Can't open %s: %s\n", argv[opt],
329         wtap_strerror(err));
330       switch (err) {
331
332       case WTAP_ERR_UNSUPPORTED:
333       case WTAP_ERR_UNSUPPORTED_ENCAP:
334       case WTAP_ERR_BAD_RECORD:
335         fprintf(stderr, "(%s)\n", err_info);
336         g_free(err_info);
337         break;
338       }
339       exit(1);
340     }
341
342     if (opt > optind)
343       printf("\n");
344     status = process_cap_file(wth, argv[opt]);
345   
346     wtap_close(wth);
347     if (status)
348       exit(status);
349   }
350   return 0;
351 }
352