fuzz_injector.rs 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. use crate::phy::{self, Device, DeviceCapabilities};
  2. use crate::time::Instant;
  3. use alloc::vec::Vec;
  4. // This could be fixed once associated consts are stable.
  5. const MTU: usize = 1536;
  6. /// Represents a fuzzer. It is expected to replace bytes in the packet with fuzzed data.
  7. pub trait Fuzzer {
  8. /// Modify a single packet with fuzzed data.
  9. fn fuzz_packet(&self, packet_data: &mut [u8]);
  10. }
  11. /// A fuzz injector device.
  12. ///
  13. /// A fuzz injector is a device that alters packets traversing through it according to the
  14. /// directions of a guided fuzzer. It is designed to support fuzzing internal state machines inside
  15. /// smoltcp, and is not for production use.
  16. #[allow(unused)]
  17. #[derive(Debug)]
  18. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  19. pub struct FuzzInjector<D: Device, FTx: Fuzzer, FRx: Fuzzer> {
  20. inner: D,
  21. fuzz_tx: FTx,
  22. fuzz_rx: FRx,
  23. }
  24. #[allow(unused)]
  25. impl<D: Device, FTx: Fuzzer, FRx: Fuzzer> FuzzInjector<D, FTx, FRx> {
  26. /// Create a fuzz injector device.
  27. pub fn new(inner: D, fuzz_tx: FTx, fuzz_rx: FRx) -> FuzzInjector<D, FTx, FRx> {
  28. FuzzInjector {
  29. inner,
  30. fuzz_tx,
  31. fuzz_rx,
  32. }
  33. }
  34. /// Return the underlying device, consuming the fuzz injector.
  35. pub fn into_inner(self) -> D {
  36. self.inner
  37. }
  38. }
  39. impl<D: Device, FTx, FRx> Device for FuzzInjector<D, FTx, FRx>
  40. where
  41. FTx: Fuzzer,
  42. FRx: Fuzzer,
  43. {
  44. type RxToken<'a> = RxToken<'a, D::RxToken<'a>, FRx>
  45. where
  46. Self: 'a;
  47. type TxToken<'a> = TxToken<'a, D::TxToken<'a>, FTx>
  48. where
  49. Self: 'a;
  50. fn capabilities(&self) -> DeviceCapabilities {
  51. let mut caps = self.inner.capabilities();
  52. if caps.max_transmission_unit > MTU {
  53. caps.max_transmission_unit = MTU;
  54. }
  55. caps
  56. }
  57. fn receive(&mut self, timestamp: Instant) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> {
  58. self.inner.receive(timestamp).map(|(rx_token, tx_token)| {
  59. let rx = RxToken {
  60. fuzzer: &mut self.fuzz_rx,
  61. token: rx_token,
  62. };
  63. let tx = TxToken {
  64. fuzzer: &mut self.fuzz_tx,
  65. token: tx_token,
  66. };
  67. (rx, tx)
  68. })
  69. }
  70. fn transmit(&mut self, timestamp: Instant) -> Option<Self::TxToken<'_>> {
  71. self.inner.transmit(timestamp).map(|token| TxToken {
  72. fuzzer: &mut self.fuzz_tx,
  73. token: token,
  74. })
  75. }
  76. }
  77. #[doc(hidden)]
  78. pub struct RxToken<'a, Rx: phy::RxToken, F: Fuzzer + 'a> {
  79. fuzzer: &'a F,
  80. token: Rx,
  81. }
  82. impl<'a, Rx: phy::RxToken, FRx: Fuzzer> phy::RxToken for RxToken<'a, Rx, FRx> {
  83. fn consume<R, F>(self, f: F) -> R
  84. where
  85. F: FnOnce(&[u8]) -> R,
  86. {
  87. self.token.consume(|buffer| {
  88. let mut new_buffer: Vec<u8> = buffer.to_vec();
  89. self.fuzzer.fuzz_packet(&mut new_buffer);
  90. f(&mut new_buffer)
  91. })
  92. }
  93. fn meta(&self) -> phy::PacketMeta {
  94. self.token.meta()
  95. }
  96. }
  97. #[doc(hidden)]
  98. pub struct TxToken<'a, Tx: phy::TxToken, F: Fuzzer + 'a> {
  99. fuzzer: &'a F,
  100. token: Tx,
  101. }
  102. impl<'a, Tx: phy::TxToken, FTx: Fuzzer> phy::TxToken for TxToken<'a, Tx, FTx> {
  103. fn consume<R, F>(self, len: usize, f: F) -> R
  104. where
  105. F: FnOnce(&mut [u8]) -> R,
  106. {
  107. self.token.consume(len, |buf| {
  108. let result = f(buf);
  109. self.fuzzer.fuzz_packet(buf);
  110. result
  111. })
  112. }
  113. fn set_meta(&mut self, meta: phy::PacketMeta) {
  114. self.token.set_meta(meta)
  115. }
  116. }