summaryrefslogtreecommitdiffstats
path: root/rust/src/frames.rs
diff options
context:
space:
mode:
Diffstat (limited to 'rust/src/frames.rs')
-rw-r--r--rust/src/frames.rs128
1 files changed, 128 insertions, 0 deletions
diff --git a/rust/src/frames.rs b/rust/src/frames.rs
new file mode 100644
index 0000000..3a45d01
--- /dev/null
+++ b/rust/src/frames.rs
@@ -0,0 +1,128 @@
+/* Copyright (C) 2017-2021 Open Information Security Foundation
+ *
+ * You can copy, redistribute or modify this Program under the terms of
+ * the GNU General Public License version 2 as published by the Free
+ * Software Foundation.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * version 2 along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301, USA.
+ */
+
+//! Module for bindings to the Suricata C frame API.
+
+use crate::applayer::StreamSlice;
+use crate::core::Flow;
+#[cfg(not(test))]
+use crate::core::STREAM_TOSERVER;
+use crate::core::Direction;
+
+#[cfg(not(test))]
+#[repr(C)]
+struct CFrame {
+ _private: [u8; 0],
+}
+
+// Defined in app-layer-register.h
+extern {
+ #[cfg(not(test))]
+ fn AppLayerFrameNewByRelativeOffset(
+ flow: *const Flow, stream_slice: *const StreamSlice, frame_start_rel: u32, len: i64,
+ dir: i32, frame_type: u8,
+ ) -> *const CFrame;
+ fn AppLayerFrameAddEventById(flow: *const Flow, dir: i32, id: i64, event: u8);
+ fn AppLayerFrameSetLengthById(flow: *const Flow, dir: i32, id: i64, len: i64);
+ fn AppLayerFrameSetTxIdById(flow: *const Flow, dir: i32, id: i64, tx_id: u64);
+ #[cfg(not(test))]
+ fn AppLayerFrameGetId(frame: *const CFrame) -> i64;
+}
+
+pub struct Frame {
+ pub id: i64,
+ direction: Direction,
+}
+
+impl std::fmt::Debug for Frame {
+ fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
+ write!(f, "frame: {}, direction: {}", self.id, self.direction)
+ }
+}
+
+impl Frame {
+ #[cfg(not(test))]
+ #[allow(clippy::not_unsafe_ptr_arg_deref)]
+ pub fn new(
+ flow: *const Flow, stream_slice: &StreamSlice, frame_start: &[u8], frame_len: i64,
+ frame_type: u8,
+ ) -> Option<Self> {
+ let offset = frame_start.as_ptr() as usize - stream_slice.as_slice().as_ptr() as usize;
+ SCLogDebug!("offset {} stream_slice.len() {} frame_start.len() {}", offset, stream_slice.len(), frame_start.len());
+ let frame = unsafe {
+ AppLayerFrameNewByRelativeOffset(
+ flow,
+ stream_slice,
+ offset as u32,
+ frame_len,
+ (stream_slice.flags() & STREAM_TOSERVER == 0).into(),
+ frame_type,
+ )
+ };
+ let id = unsafe { AppLayerFrameGetId(frame) };
+ if id > 0 {
+ Some(Self {
+ id,
+ direction: Direction::from(stream_slice.flags()),
+ })
+ } else {
+ None
+ }
+ }
+
+ /// A variation of `new` for use when running Rust unit tests as
+ /// the C functions for building a frame are not available for
+ /// linkage.
+ #[cfg(test)]
+ pub fn new(
+ _flow: *const Flow, _stream_slice: &StreamSlice, _frame_start: &[u8], _frame_len: i64,
+ _frame_type: u8,
+ ) -> Option<Self> {
+ None
+ }
+
+ /// Conversion function to get the direction in the correct form for the
+ /// C frame methods which takes direction as a u32 value of 0 or 1 rather
+ /// than the flag value used internally by Frame.
+ fn direction(&self) -> i32 {
+ match self.direction {
+ Direction::ToServer => 0,
+ Direction::ToClient => 1,
+ }
+ }
+
+ #[allow(clippy::not_unsafe_ptr_arg_deref)]
+ pub fn set_len(&self, flow: *const Flow, len: i64) {
+ unsafe {
+ AppLayerFrameSetLengthById(flow, self.direction(), self.id, len);
+ };
+ }
+
+ #[allow(clippy::not_unsafe_ptr_arg_deref)]
+ pub fn set_tx(&self, flow: *const Flow, tx_id: u64) {
+ unsafe {
+ AppLayerFrameSetTxIdById(flow, self.direction(), self.id, tx_id);
+ };
+ }
+
+ #[allow(clippy::not_unsafe_ptr_arg_deref)]
+ pub fn add_event(&self, flow: *const Flow, event: u8) {
+ unsafe {
+ AppLayerFrameAddEventById(flow, self.direction(), self.id, event);
+ };
+ }
+}