@@ -15,10 +15,8 @@ use crate::{DataLinkReceiver, DataLinkSender, MacAddr, NetworkInterface};
15
15
16
16
use pnet_sys;
17
17
18
- use std:: cmp;
19
18
use std:: io;
20
19
use std:: mem;
21
- use std:: ptr;
22
20
use std:: sync:: Arc ;
23
21
use std:: time:: Duration ;
24
22
@@ -210,7 +208,6 @@ pub fn channel(
210
208
let fd = Arc :: new ( pnet_sys:: FileDesc { fd : socket } ) ;
211
209
let sender = Box :: new ( DataLinkSenderImpl {
212
210
socket : fd. clone ( ) ,
213
- fd_set : unsafe { mem:: zeroed ( ) } ,
214
211
write_buffer : vec ! [ 0 ; config. write_buffer_size] ,
215
212
_channel_type : config. channel_type ,
216
213
send_addr : unsafe { * ( send_addr as * const libc:: sockaddr_ll ) } ,
@@ -221,7 +218,6 @@ pub fn channel(
221
218
} ) ;
222
219
let receiver = Box :: new ( DataLinkReceiverImpl {
223
220
socket : fd. clone ( ) ,
224
- fd_set : unsafe { mem:: zeroed ( ) } ,
225
221
read_buffer : vec ! [ 0 ; config. read_buffer_size] ,
226
222
_channel_type : config. channel_type ,
227
223
timeout : config
@@ -234,7 +230,6 @@ pub fn channel(
234
230
235
231
struct DataLinkSenderImpl {
236
232
socket : Arc < pnet_sys:: FileDesc > ,
237
- fd_set : libc:: fd_set ,
238
233
write_buffer : Vec < u8 > ,
239
234
_channel_type : super :: ChannelType ,
240
235
send_addr : libc:: sockaddr_ll ,
@@ -253,35 +248,40 @@ impl DataLinkSender for DataLinkSenderImpl {
253
248
) -> Option < io:: Result < ( ) > > {
254
249
let len = num_packets * packet_size;
255
250
if len <= self . write_buffer . len ( ) {
256
- let min = cmp:: min ( self . write_buffer [ .. ] . len ( ) , len) ;
251
+ let min = std :: cmp:: min ( self . write_buffer . len ( ) , len) ;
257
252
let mut_slice = & mut self . write_buffer ;
253
+
254
+ let mut pollfd = libc:: pollfd {
255
+ fd : self . socket . fd ,
256
+ events : libc:: POLLOUT , // Monitoring for write ability
257
+ revents : 0 , // Will be filled by poll to indicate the events that occurred
258
+ } ;
259
+
260
+ // Convert timeout to milliseconds as required by poll
261
+ let timeout_ms = self
262
+ . timeout
263
+ . as_ref ( )
264
+ . map ( |to| ( to. tv_sec as i64 * 1000 ) + ( to. tv_nsec as i64 / 1_000_000 ) )
265
+ . unwrap_or ( -1 ) ; // -1 means wait indefinitely
266
+
258
267
for chunk in mut_slice[ ..min] . chunks_mut ( packet_size) {
259
268
func ( chunk) ;
260
269
let send_addr =
261
270
( & self . send_addr as * const libc:: sockaddr_ll ) as * const libc:: sockaddr ;
262
271
263
- unsafe {
264
- libc:: FD_ZERO ( & mut self . fd_set as * mut libc:: fd_set ) ;
265
- libc:: FD_SET ( self . socket . fd , & mut self . fd_set as * mut libc:: fd_set ) ;
266
- }
267
272
let ret = unsafe {
268
- libc:: pselect (
269
- self . socket . fd + 1 ,
270
- ptr:: null_mut ( ) ,
271
- & mut self . fd_set as * mut libc:: fd_set ,
272
- ptr:: null_mut ( ) ,
273
- self . timeout
274
- . as_ref ( )
275
- . map ( |to| to as * const libc:: timespec )
276
- . unwrap_or ( ptr:: null ( ) ) ,
277
- ptr:: null ( ) ,
273
+ libc:: poll (
274
+ & mut pollfd as * mut libc:: pollfd ,
275
+ 1 ,
276
+ timeout_ms as libc:: c_int ,
278
277
)
279
278
} ;
279
+
280
280
if ret == -1 {
281
281
return Some ( Err ( io:: Error :: last_os_error ( ) ) ) ;
282
282
} else if ret == 0 {
283
283
return Some ( Err ( io:: Error :: new ( io:: ErrorKind :: TimedOut , "Timed out" ) ) ) ;
284
- } else {
284
+ } else if pollfd . revents & libc :: POLLOUT != 0 {
285
285
if let Err ( e) = pnet_sys:: send_to (
286
286
self . socket . fd ,
287
287
chunk,
@@ -290,6 +290,11 @@ impl DataLinkSender for DataLinkSenderImpl {
290
290
) {
291
291
return Some ( Err ( e) ) ;
292
292
}
293
+ } else {
294
+ return Some ( Err ( io:: Error :: new (
295
+ io:: ErrorKind :: Other ,
296
+ "Unexpected poll event" ,
297
+ ) ) ) ;
293
298
}
294
299
}
295
300
@@ -301,28 +306,33 @@ impl DataLinkSender for DataLinkSenderImpl {
301
306
302
307
#[ inline]
303
308
fn send_to ( & mut self , packet : & [ u8 ] , _dst : Option < NetworkInterface > ) -> Option < io:: Result < ( ) > > {
304
- unsafe {
305
- libc:: FD_ZERO ( & mut self . fd_set as * mut libc:: fd_set ) ;
306
- libc:: FD_SET ( self . socket . fd , & mut self . fd_set as * mut libc:: fd_set ) ;
307
- }
309
+ let mut pollfd = libc:: pollfd {
310
+ fd : self . socket . fd ,
311
+ events : libc:: POLLOUT , // Monitoring for write ability
312
+ revents : 0 , // Will be filled by poll to indicate the events that occurred
313
+ } ;
314
+
315
+ // Convert timeout to milliseconds as required by poll
316
+ let timeout_ms = self
317
+ . timeout
318
+ . as_ref ( )
319
+ . map ( |to| ( to. tv_sec as i64 * 1000 ) + ( to. tv_nsec as i64 / 1_000_000 ) )
320
+ . unwrap_or ( -1 ) ; // -1 means wait indefinitely
321
+
308
322
let ret = unsafe {
309
- libc:: pselect (
310
- self . socket . fd + 1 ,
311
- ptr:: null_mut ( ) ,
312
- & mut self . fd_set as * mut libc:: fd_set ,
313
- ptr:: null_mut ( ) ,
314
- self . timeout
315
- . as_ref ( )
316
- . map ( |to| to as * const libc:: timespec )
317
- . unwrap_or ( ptr:: null ( ) ) ,
318
- ptr:: null ( ) ,
323
+ libc:: poll (
324
+ & mut pollfd as * mut libc:: pollfd ,
325
+ 1 ,
326
+ timeout_ms as libc:: c_int ,
319
327
)
320
328
} ;
329
+
321
330
if ret == -1 {
322
331
Some ( Err ( io:: Error :: last_os_error ( ) ) )
323
332
} else if ret == 0 {
324
333
Some ( Err ( io:: Error :: new ( io:: ErrorKind :: TimedOut , "Timed out" ) ) )
325
- } else {
334
+ } else if pollfd. revents & libc:: POLLOUT != 0 {
335
+ // POLLOUT is set, meaning the socket is ready for writing
326
336
match pnet_sys:: send_to (
327
337
self . socket . fd ,
328
338
packet,
@@ -332,13 +342,17 @@ impl DataLinkSender for DataLinkSenderImpl {
332
342
Err ( e) => Some ( Err ( e) ) ,
333
343
Ok ( _) => Some ( Ok ( ( ) ) ) ,
334
344
}
345
+ } else {
346
+ Some ( Err ( io:: Error :: new (
347
+ io:: ErrorKind :: Other ,
348
+ "Unexpected poll event" ,
349
+ ) ) )
335
350
}
336
351
}
337
352
}
338
353
339
354
struct DataLinkReceiverImpl {
340
355
socket : Arc < pnet_sys:: FileDesc > ,
341
- fd_set : libc:: fd_set ,
342
356
read_buffer : Vec < u8 > ,
343
357
_channel_type : super :: ChannelType ,
344
358
timeout : Option < libc:: timespec > ,
@@ -347,33 +361,43 @@ struct DataLinkReceiverImpl {
347
361
impl DataLinkReceiver for DataLinkReceiverImpl {
348
362
fn next ( & mut self ) -> io:: Result < & [ u8 ] > {
349
363
let mut caddr: libc:: sockaddr_storage = unsafe { mem:: zeroed ( ) } ;
350
- unsafe {
351
- libc:: FD_ZERO ( & mut self . fd_set as * mut libc:: fd_set ) ;
352
- libc:: FD_SET ( self . socket . fd , & mut self . fd_set as * mut libc:: fd_set ) ;
353
- }
364
+ let mut pollfd = libc:: pollfd {
365
+ fd : self . socket . fd ,
366
+ events : libc:: POLLIN , // Monitoring for read availability
367
+ revents : 0 ,
368
+ } ;
369
+
370
+ // Convert timeout to milliseconds as required by poll
371
+ let timeout_ms = self
372
+ . timeout
373
+ . as_ref ( )
374
+ . map ( |to| ( to. tv_sec as i64 * 1000 ) + ( to. tv_nsec as i64 / 1_000_000 ) )
375
+ . unwrap_or ( -1 ) ; // -1 means wait indefinitely
376
+
354
377
let ret = unsafe {
355
- libc:: pselect (
356
- self . socket . fd + 1 ,
357
- & mut self . fd_set as * mut libc:: fd_set ,
358
- ptr:: null_mut ( ) ,
359
- ptr:: null_mut ( ) ,
360
- self . timeout
361
- . as_ref ( )
362
- . map ( |to| to as * const libc:: timespec )
363
- . unwrap_or ( ptr:: null ( ) ) ,
364
- ptr:: null ( ) ,
378
+ libc:: poll (
379
+ & mut pollfd as * mut libc:: pollfd ,
380
+ 1 ,
381
+ timeout_ms as libc:: c_int ,
365
382
)
366
383
} ;
384
+
367
385
if ret == -1 {
368
386
Err ( io:: Error :: last_os_error ( ) )
369
387
} else if ret == 0 {
370
388
Err ( io:: Error :: new ( io:: ErrorKind :: TimedOut , "Timed out" ) )
371
- } else {
389
+ } else if pollfd. revents & libc:: POLLIN != 0 {
390
+ // POLLIN is set, meaning the socket has data to be read
372
391
let res = pnet_sys:: recv_from ( self . socket . fd , & mut self . read_buffer , & mut caddr) ;
373
392
match res {
374
393
Ok ( len) => Ok ( & self . read_buffer [ 0 ..len] ) ,
375
394
Err ( e) => Err ( e) ,
376
395
}
396
+ } else {
397
+ Err ( io:: Error :: new (
398
+ io:: ErrorKind :: Other ,
399
+ "Unexpected poll event" ,
400
+ ) )
377
401
}
378
402
}
379
403
}
0 commit comments