31
31
#include <sys/time.h>
32
32
#include <linux/if_ether.h>
33
33
#include <glib.h>
34
+ #include <pcap/pcap.h>
34
35
35
36
#define PIPE_IN ".in"
36
37
#define PIPE_OUT ".out"
@@ -46,35 +47,23 @@ static GMainLoop *main_loop = NULL;
46
47
static char * path = NULL ;
47
48
static char * pipe_1_in = NULL , * pipe_1_out = NULL ;
48
49
static char * pipe_2_in = NULL , * pipe_2_out = NULL ;
49
- static struct pcap * pcap = NULL ;
50
+ static struct pcap_fd * pcap ;
50
51
static int fd1_in , fd2_in ;
51
52
52
- struct pcap_hdr {
53
- uint32_t magic_number ; /* magic number */
54
- uint16_t version_major ; /* major version number */
55
- uint16_t version_minor ; /* minor version number */
56
- int32_t thiszone ; /* GMT to local correction */
57
- uint32_t sigfigs ; /* accuracy of timestamps */
58
- uint32_t snaplen ; /* max length of captured packets, in octets */
59
- uint32_t network ; /* data link type */
60
- } __attribute__ ((packed ));
61
- #define PCAP_HDR_SIZE (sizeof(struct pcap_hdr))
62
-
63
- struct pcap_pkt {
64
- uint32_t ts_sec ; /* timestamp seconds */
65
- uint32_t ts_usec ; /* timestamp microseconds */
66
- uint32_t incl_len ; /* number of octets of packet saved in file */
67
- uint32_t orig_len ; /* actual length of packet */
68
- } __attribute__ ((packed ));
69
- #define PCAP_PKT_SIZE (sizeof(struct pcap_pkt))
70
-
71
- struct pcap {
53
+ struct pcap_fd {
72
54
int fd ;
73
- bool closed ;
74
- uint32_t type ;
75
- uint32_t snaplen ;
76
55
};
77
56
57
+ #define PCAP_FILE_HDR_SIZE (sizeof(struct pcap_file_header))
58
+
59
+ struct pcap_frame {
60
+ uint32_t ts_sec ;
61
+ uint32_t ts_usec ;
62
+ uint32_t caplen ;
63
+ uint32_t len ;
64
+ } __attribute__ ((packed ));
65
+ #define PCAP_FRAME_SIZE (sizeof(struct pcap_frame))
66
+
78
67
struct debug_desc {
79
68
const char * name ;
80
69
const char * file ;
@@ -120,56 +109,7 @@ void log_cleanup(void)
120
109
closelog ();
121
110
}
122
111
123
- struct pcap * pcap_open (const char * pathname )
124
- {
125
- struct pcap * pcap ;
126
- struct pcap_hdr hdr ;
127
- ssize_t len ;
128
-
129
- pcap = g_new0 (struct pcap , 1 );
130
-
131
- pcap -> fd = open (pathname , O_RDONLY | O_CLOEXEC );
132
- if (pcap -> fd < 0 ) {
133
- DBG ("Failed to open PCAP file" );
134
- g_free (pcap );
135
- return NULL ;
136
- }
137
-
138
- len = read (pcap -> fd , & hdr , PCAP_HDR_SIZE );
139
- if (len < 0 ) {
140
- DBG ("Failed to read PCAP header" );
141
- goto failed ;
142
- }
143
-
144
- if (len != PCAP_HDR_SIZE ) {
145
- DBG ("Wrong PCAP header size\n" );
146
- goto failed ;
147
- }
148
-
149
- if (hdr .magic_number != 0xa1b2c3d4 ) {
150
- DBG ("Wrong PCAP header magic\n" );
151
- goto failed ;
152
- }
153
-
154
- if (hdr .version_major != 2 || hdr .version_minor != 4 ) {
155
- DBG ("Wrong PCAP version number\n" );
156
- goto failed ;
157
- }
158
-
159
- pcap -> closed = false;
160
- pcap -> snaplen = hdr .snaplen ;
161
- pcap -> type = hdr .network ;
162
-
163
- return pcap ;
164
-
165
- failed :
166
- close (pcap -> fd );
167
- g_free (pcap );
168
-
169
- return NULL ;
170
- }
171
-
172
- void pcap_close (struct pcap * pcap )
112
+ void monitor_pcap_free (void )
173
113
{
174
114
if (!pcap )
175
115
return ;
@@ -180,90 +120,80 @@ void pcap_close(struct pcap *pcap)
180
120
g_free (pcap );
181
121
}
182
122
183
- struct pcap * pcap_create (const char * pathname )
123
+ bool monitor_pcap_create (const char * pathname )
184
124
{
185
- struct pcap * pcap ;
186
- struct pcap_hdr hdr ;
125
+ struct pcap_file_header hdr ;
187
126
ssize_t len ;
188
127
189
- pcap = g_new0 (struct pcap , 1 );
128
+ pcap = g_new0 (struct pcap_fd , 1 );
190
129
pcap -> fd = open (pathname , O_WRONLY | O_CREAT | O_TRUNC | O_CLOEXEC ,
191
130
S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH );
192
131
if (pcap -> fd < 0 ) {
193
132
DBG ("Failed to create PCAP file" );
194
133
g_free (pcap );
195
- return NULL ;
134
+ return false ;
196
135
}
197
136
198
- pcap -> closed = false;
199
- pcap -> snaplen = 0x0000ffff ;
200
- /* http://www.tcpdump.org/linktypes.html */
201
- pcap -> type = 0x000000E6 ; /* LINKTYPE_IEEE802_15_4_NOFCS : 230 */
202
-
203
- memset (& hdr , 0 , sizeof (hdr ));
204
- hdr .magic_number = 0xa1b2c3d4 ;
137
+ memset (& hdr , 0 , PCAP_FILE_HDR_SIZE );
138
+ hdr .magic = 0xa1b2c3d4 ;
205
139
hdr .version_major = 0x0002 ;
206
140
hdr .version_minor = 0x0004 ;
207
141
hdr .thiszone = 0 ;
208
142
hdr .sigfigs = 0 ;
209
- hdr .snaplen = pcap -> snaplen ;
210
- hdr .network = pcap -> type ;
211
-
212
- len = write (pcap -> fd , & hdr , PCAP_HDR_SIZE );
143
+ hdr .snaplen = 0x0000ffff ;
144
+ /*
145
+ * http://www.tcpdump.org/linktypes.html
146
+ * LINKTYPE_IEEE802_15_4_NOFCS : 230
147
+ */
148
+ hdr .linktype = 0x000000E6 ;
149
+
150
+ len = write (pcap -> fd , & hdr , PCAP_FILE_HDR_SIZE );
213
151
if (len < 0 ) {
214
152
DBG ("Failed to write PCAP header" );
215
153
goto failed ;
216
154
}
217
155
218
- if (len != PCAP_HDR_SIZE ) {
156
+ if (len != PCAP_FILE_HDR_SIZE ) {
219
157
DBG ("Written PCAP header size mimatch\n" );
220
158
goto failed ;
221
159
}
222
160
223
- return pcap ;
161
+ return true ;
224
162
225
163
failed :
226
- close (pcap -> fd );
227
- g_free (pcap );
164
+ monitor_pcap_free ();
228
165
229
- return NULL ;
166
+ return false ;
230
167
}
231
168
232
- bool pcap_write (struct pcap * pcap , const struct timeval * tv ,
233
- const void * data , uint32_t size )
169
+ bool monitor_pcap_write (const void * data , uint32_t size )
234
170
{
235
- struct iovec iov [ 3 ] ;
236
- struct pcap_pkt pkt ;
237
- ssize_t written ;
171
+ struct pcap_frame frame ;
172
+ struct timeval tv ;
173
+ ssize_t len ;
238
174
239
175
if (!pcap )
240
176
return false;
241
177
242
- if (pcap -> closed )
243
- return false;
178
+ memset (& frame , 0 , PCAP_FRAME_SIZE );
244
179
245
- memset ( & pkt , 0 , sizeof ( pkt ) );
246
- if ( tv ) {
247
- pkt . ts_sec = tv -> tv_sec ;
248
- pkt . ts_usec = tv -> tv_usec ;
249
- }
180
+ gettimeofday ( & tv , NULL );
181
+ frame . ts_sec = tv . tv_sec ;
182
+ frame . ts_usec = tv . tv_usec ;
183
+ frame . caplen = size ;
184
+ frame . len = size ;
250
185
251
- pkt .incl_len = size ;
252
- pkt .orig_len = size ;
253
-
254
- iov [0 ].iov_base = & pkt ;
255
- iov [0 ].iov_len = PCAP_PKT_SIZE ;
256
- iov [1 ].iov_base = (void * ) data ;
257
- iov [1 ].iov_len = size ;
258
-
259
- written = writev (pcap -> fd , iov , 2 );
260
- if (written < 0 ) {
261
- pcap -> closed = true;
186
+ len = write (pcap -> fd , & frame , PCAP_FRAME_SIZE );
187
+ if (len < 0 || len != PCAP_FRAME_SIZE ) {
188
+ DBG ("Failed to write PCAP frame or size mismatch %d\n" , len );
262
189
return false;
263
190
}
264
191
265
- if (written < (ssize_t ) (PCAP_PKT_SIZE + size )) {
266
- pcap -> closed = true;
192
+ fsync (pcap -> fd );
193
+
194
+ len = write (pcap -> fd , data , size );
195
+ if (len < 0 || len != size ) {
196
+ DBG ("Failed to write PCAP data or size mismatch %d\n" , len );
267
197
return false;
268
198
}
269
199
@@ -278,7 +208,6 @@ static gboolean fifo_handler1(GIOChannel *channel, GIOCondition cond,
278
208
unsigned char buf [1 ];
279
209
ssize_t result ;
280
210
int fd ;
281
- struct timeval tv ;
282
211
283
212
if (cond & (G_IO_NVAL | G_IO_ERR | G_IO_HUP )) {
284
213
DBG ("First pipe closed" );
@@ -340,9 +269,7 @@ static gboolean fifo_handler1(GIOChannel *channel, GIOCondition cond,
340
269
if (input1_len && input1_len == input1_offset ) {
341
270
DBG ("Received %d bytes in pipe 1" , input1_len );
342
271
343
- /* write it to pcap file */
344
- gettimeofday (& tv , NULL );
345
- pcap_write (pcap , & tv , input1 , input1_len );
272
+ monitor_pcap_write (input1 , input1_len );
346
273
input1_len = input1_offset = 0 ;
347
274
memset (input1 , 0 , sizeof (input1 ));
348
275
@@ -360,7 +287,6 @@ static gboolean fifo_handler2(GIOChannel *channel, GIOCondition cond,
360
287
unsigned char buf [1 ];
361
288
ssize_t result ;
362
289
int fd ;
363
- struct timeval tv ;
364
290
365
291
if (cond & (G_IO_NVAL | G_IO_ERR | G_IO_HUP )) {
366
292
DBG ("Second pipe closed" );
@@ -421,9 +347,7 @@ static gboolean fifo_handler2(GIOChannel *channel, GIOCondition cond,
421
347
if (input2_len && input2_len == input2_offset ) {
422
348
DBG ("Received %d bytes in pipe 2" , input2_len );
423
349
424
- /* write it to pcap file */
425
- gettimeofday (& tv , NULL );
426
- pcap_write (pcap , & tv , input2 , input2_len );
350
+ monitor_pcap_write (input2 , input2_len );
427
351
input2_len = input2_offset = 0 ;
428
352
memset (input2 , 0 , sizeof (input2 ));
429
353
@@ -516,17 +440,16 @@ int main(int argc, char *argv[])
516
440
pipe_2_out = g_strconcat (pipe2 , PIPE_OUT , NULL );
517
441
path = g_strdup (argv [1 ]);
518
442
519
- pcap = pcap_create (path );
520
- if (!pcap ) {
521
- g_free (path );
522
- exit (- EINVAL );
523
- }
524
-
525
443
log_init ("log" , FALSE, argc > 4 ? TRUE : FALSE);
526
444
527
445
DBG ("Pipe 1 IN %s OUT %s" , pipe_1_in , pipe_1_out );
528
446
DBG ("Pipe 2 IN %s OUT %s" , pipe_2_in , pipe_2_out );
529
447
448
+ if (!monitor_pcap_create (path )) {
449
+ g_free (path );
450
+ exit (- EINVAL );
451
+ }
452
+
530
453
fifo1 = setup_fifofd1 ();
531
454
if (fifo1 < 0 ) {
532
455
ret = - EINVAL ;
@@ -567,6 +490,7 @@ int main(int argc, char *argv[])
567
490
g_free (pipe_1_out );
568
491
g_free (pipe_2_in );
569
492
g_free (pipe_2_out );
493
+ monitor_pcap_free ();
570
494
log_cleanup ();
571
495
g_main_loop_unref (main_loop );
572
496
0 commit comments