retina_core/subscription/zc_frame.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
//! Zero-copy Ethernet frames.
//!
//! This is a packet-level subscription that delivers raw Ethernet frames in the order of arrival.
//! It has identical behavior to the [Frame](crate::subscription::frame::Frame) type, except is
//! zero-copy, meaning that callbacks are invoked on raw DPDK memory buffers instead of a
//! heap-allocated buffer. This is useful for performance sensitive applications that do not need to
//! store packet data. If ownership of the packet data is required, it is recommended to use
//! [Frame](crate::subscription::frame::Frame) instead.
//!
//! ## Warning
//! All `ZcFrame`s must be dropped (freed and returned to the memory pool) before the Retina runtime
//! is dropped.
//!
//! ## Example
//! Prints IPv4 packets with a TTL greater than 64:
//! ```
//! #[filter("ipv4.time_to_live > 64")]
//! fn main() {
//! let config = default_config();
//! let cb = |pkt: ZcFrame| {
//! println!("{:?}", pkt.data());
//! // implicit drop at end of scope
//! };
//! let mut runtime = Runtime::new(config, filter, cb).unwrap();
//! runtime.run();
//! // runtime dropped at end of scope
//! }
//! ```
use crate::conntrack::conn_id::FiveTuple;
use crate::conntrack::pdu::{L4Context, L4Pdu};
use crate::conntrack::ConnTracker;
use crate::filter::FilterResult;
use crate::memory::mbuf::Mbuf;
use crate::protocols::stream::{ConnParser, Session};
use crate::subscription::{Level, Subscribable, Subscription, Trackable};
use std::collections::HashMap;
/// A zero-copy Ethernet frame.
///
/// ## Remarks
/// This is a type alias of a DPDK message buffer. Retina allows subscriptions on raw DPDK memory
/// buffers with zero-copy (i.e., without copying into a heap-allocated buffer). This is useful for
/// performance sensitive applications that do not need to store packet data.
///
/// However, the callback does not obtain ownership of the packet. Therefore, all `ZcFrame`s must be
/// dropped before the runtime is dropped, or a segmentation fault may occur when the memory pools
/// are de-allocated. Storing `ZcFrame`s also reduces the number of available packet buffers for
/// incoming packets and can cause memory pool exhaustion.
///
/// It is recommended that `ZcFrame` be used for stateless packet analysis, and to use
/// [Frame](crate::subscription::Frame) instead if ownership of the packet is needed.
pub type ZcFrame = Mbuf;
impl Subscribable for ZcFrame {
type Tracked = TrackedZcFrame;
fn level() -> Level {
Level::Packet
}
fn parsers() -> Vec<ConnParser> {
vec![]
}
fn process_packet(
mbuf: Mbuf,
subscription: &Subscription<Self>,
conn_tracker: &mut ConnTracker<Self::Tracked>,
) {
match subscription.filter_packet(&mbuf) {
FilterResult::MatchTerminal(_idx) => {
subscription.invoke(mbuf);
}
FilterResult::MatchNonTerminal(idx) => {
if let Ok(ctxt) = L4Context::new(&mbuf, idx) {
conn_tracker.process(mbuf, ctxt, subscription);
}
}
FilterResult::NoMatch => drop(mbuf),
}
}
}
/// Buffers packets associated with parsed sessions throughout the duration of the connection.
/// ## Note
/// Internal connection state is an associated type of a `pub` trait, and therefore must also be
/// public. Documentation is hidden by default to avoid confusing users.
#[doc(hidden)]
pub struct TrackedZcFrame {
session_buf: HashMap<usize, Vec<Mbuf>>,
}
impl Trackable for TrackedZcFrame {
type Subscribed = ZcFrame;
fn new(_five_tuple: FiveTuple) -> Self {
TrackedZcFrame {
session_buf: HashMap::new(),
// misc_buf: vec![],
}
}
fn pre_match(&mut self, pdu: L4Pdu, session_id: Option<usize>) {
if let Some(session_id) = session_id {
self.session_buf
.entry(session_id)
.or_default()
.push(pdu.mbuf_own());
} else {
drop(pdu);
// self.misc_buf.push(pdu.mbuf_own());
}
}
fn on_match(&mut self, session: Session, subscription: &Subscription<Self::Subscribed>) {
if let Some(session) = self.session_buf.remove(&session.id) {
session.into_iter().for_each(|mbuf| {
subscription.invoke(mbuf);
});
}
}
fn post_match(&mut self, pdu: L4Pdu, subscription: &Subscription<Self::Subscribed>) {
subscription.invoke(pdu.mbuf_own());
}
fn on_terminate(&mut self, _subscription: &Subscription<Self::Subscribed>) {}
}