From 502134903df83ec48ef945efd40768019c614558 Mon Sep 17 00:00:00 2001 From: Pedro Henrique Penna Date: Tue, 2 Jul 2024 09:43:39 -0700 Subject: [PATCH] [catpowder] Enhancement: Improve XDP LibOS --- src/rust/catpowder/win/api.rs | 28 ++++++++------ src/rust/catpowder/win/mod.rs | 11 +++++- src/rust/catpowder/win/params.rs | 3 +- src/rust/catpowder/win/program.rs | 25 +++++++------ src/rust/catpowder/win/rule.rs | 5 --- src/rust/catpowder/win/runtime.rs | 62 ++++++++++++++++++------------- src/rust/catpowder/win/rx_ring.rs | 19 +++++----- src/rust/catpowder/win/socket.rs | 25 ++++++++----- src/rust/catpowder/win/tx_ring.rs | 9 ++--- 9 files changed, 108 insertions(+), 79 deletions(-) diff --git a/src/rust/catpowder/win/api.rs b/src/rust/catpowder/win/api.rs index 8cf794891e..f33c27945c 100644 --- a/src/rust/catpowder/win/api.rs +++ b/src/rust/catpowder/win/api.rs @@ -6,38 +6,42 @@ //====================================================================================================================== use crate::runtime::fail::Fail; -use ::windows::core::HRESULT; +use ::std::ptr; +use ::windows::core::{ + Error, + HRESULT, +}; use ::xdp_rs; //====================================================================================================================== // Structures //====================================================================================================================== -#[derive(Clone)] -pub struct XdpApi { - pub endpoint: *const xdp_rs::XDP_API_TABLE, -} +/// A wrapper structure for an XDP API. +#[repr(C)] +pub struct XdpApi(*const xdp_rs::XDP_API_TABLE); //====================================================================================================================== // Implementations //====================================================================================================================== impl XdpApi { + /// Opens a new XDP API endpoint. pub fn new() -> Result { - let mut api: *const xdp_rs::XDP_API_TABLE = std::ptr::null_mut(); + let mut api: *const xdp_rs::XDP_API_TABLE = ptr::null_mut(); let result: HRESULT = unsafe { xdp_rs::XdpOpenApi(xdp_rs::XDP_API_VERSION_1, &mut api) }; - let error: windows::core::Error = windows::core::Error::from_hresult(result); + let error: windows::core::Error = Error::from_hresult(result); match error.code().is_ok() { - true => Ok(Self { endpoint: api }), + true => Ok(Self(api)), false => Err(Fail::from(&error)), } } - pub fn endpoint(&self) -> xdp_rs::XDP_API_TABLE { + pub fn get(&self) -> xdp_rs::XDP_API_TABLE { unsafe { - let api: *const xdp_rs::XDP_API_TABLE = self.endpoint; + let api: *const xdp_rs::XDP_API_TABLE = self.0; *api } } @@ -46,12 +50,12 @@ impl XdpApi { impl Drop for XdpApi { fn drop(&mut self) { let api: xdp_rs::XDP_API_TABLE = unsafe { - let api: *const xdp_rs::XDP_API_TABLE = self.endpoint; + let api: *const xdp_rs::XDP_API_TABLE = self.0; *api }; if let Some(close) = api.XdpCloseApi { - unsafe { close(self.endpoint) }; + unsafe { close(self.0) }; } } } diff --git a/src/rust/catpowder/win/mod.rs b/src/rust/catpowder/win/mod.rs index bb5037f41c..a6de005d32 100644 --- a/src/rust/catpowder/win/mod.rs +++ b/src/rust/catpowder/win/mod.rs @@ -1,13 +1,22 @@ // Copyright (c) Microsoft Corporation. // Licensed under the MIT license. +//====================================================================================================================== +// Modules +//====================================================================================================================== + mod api; mod buffer; mod params; mod program; mod rule; -pub mod runtime; mod rx_ring; mod socket; mod tx_ring; mod umemreg; + +//====================================================================================================================== +// Exports +//====================================================================================================================== + +pub mod runtime; diff --git a/src/rust/catpowder/win/params.rs b/src/rust/catpowder/win/params.rs index c73ed25094..c94bac4840 100644 --- a/src/rust/catpowder/win/params.rs +++ b/src/rust/catpowder/win/params.rs @@ -12,6 +12,7 @@ use ::std::mem; // Structures //====================================================================================================================== +#[repr(C)] pub struct XdpRedirectParams { redirect: xdp_rs::XDP_REDIRECT_PARAMS, } @@ -25,7 +26,7 @@ impl XdpRedirectParams { let redirect: xdp_rs::XDP_REDIRECT_PARAMS = { let mut redirect: xdp_rs::_XDP_REDIRECT_PARAMS = unsafe { mem::zeroed() }; redirect.TargetType = xdp_rs::_XDP_REDIRECT_TARGET_TYPE_XDP_REDIRECT_TARGET_TYPE_XSK; - redirect.Target = socket.socket; + redirect.Target = socket.get_socket(); redirect }; Self { redirect } diff --git a/src/rust/catpowder/win/program.rs b/src/rust/catpowder/win/program.rs index d28d7ba4b7..99fa9e2877 100644 --- a/src/rust/catpowder/win/program.rs +++ b/src/rust/catpowder/win/program.rs @@ -26,7 +26,6 @@ use ::xdp_rs; //====================================================================================================================== #[repr(C)] -#[derive(Default)] pub struct XdpProgram(HANDLE); //====================================================================================================================== @@ -45,15 +44,24 @@ impl XdpProgram { ) -> Result { let rule: *const xdp_rs::XDP_RULE = rules.as_ptr() as *const xdp_rs::XDP_RULE; let rule_count: u32 = rules.len() as u32; - let mut program: XdpProgram = XdpProgram::default(); + let mut handle: HANDLE = HANDLE::default(); // Attempt to create the XDP program. - if let Some(create_program) = api.endpoint().XdpCreateProgram { - let result: HRESULT = - unsafe { create_program(ifindex, hookid, queueid, flags, rule, rule_count, program.as_ptr()) }; + if let Some(create_program) = api.get().XdpCreateProgram { + let result: HRESULT = unsafe { + create_program( + ifindex, + hookid, + queueid, + flags, + rule, + rule_count, + &mut handle as *mut HANDLE, + ) + }; let error: windows::core::Error = windows::core::Error::from_hresult(result); match error.code().is_ok() { - true => Ok(program), + true => Ok(Self(handle)), false => Err(Fail::from(&error)), } } else { @@ -62,11 +70,6 @@ impl XdpProgram { Err(Fail::new(libc::ENOSYS, &cause)) } } - - /// Casts the target XDP program to a raw pointer. - pub fn as_ptr(&mut self) -> *mut HANDLE { - &mut self.0 as *mut HANDLE - } } //====================================================================================================================== diff --git a/src/rust/catpowder/win/rule.rs b/src/rust/catpowder/win/rule.rs index 5240473887..2238fbce70 100644 --- a/src/rust/catpowder/win/rule.rs +++ b/src/rust/catpowder/win/rule.rs @@ -29,7 +29,6 @@ impl XdpRule { let mut rule: xdp_rs::XDP_RULE = std::mem::zeroed(); rule.Match = xdp_rs::_XDP_MATCH_TYPE_XDP_MATCH_ALL; rule.Action = xdp_rs::_XDP_RULE_ACTION_XDP_PROGRAM_ACTION_REDIRECT; - // TODO: Set pattern // Perform bitwise copy from redirect to rule. rule.__bindgen_anon_1 = mem::transmute_copy::(redirect.as_ptr()); @@ -38,8 +37,4 @@ impl XdpRule { }; Self(rule) } - - pub fn as_ptr(&self) -> *const xdp_rs::XDP_RULE { - &self.0 - } } diff --git a/src/rust/catpowder/win/runtime.rs b/src/rust/catpowder/win/runtime.rs index 9a0dde26c2..2497bd9b5d 100644 --- a/src/rust/catpowder/win/runtime.rs +++ b/src/rust/catpowder/win/runtime.rs @@ -40,6 +40,7 @@ use ::std::borrow::{ //====================================================================================================================== struct CatpowderRuntimeInner { + api: XdpApi, tx: TxRing, rx: RxRing, } @@ -47,43 +48,55 @@ struct CatpowderRuntimeInner { /// A LibOS built on top of Windows XDP. #[derive(Clone)] pub struct CatpowderRuntime { - api: XdpApi, inner: SharedObject, } //====================================================================================================================== // Associated Functions //====================================================================================================================== + +impl CatpowderRuntimeInner { + fn notify_socket(&mut self, flags: i32, timeout: u32, outflags: &mut i32) -> Result<(), Fail> { + self.tx.notify_socket(&mut self.api, flags, timeout, outflags) + } +} + impl CatpowderRuntime {} impl NetworkRuntime for CatpowderRuntime { fn new(config: &Config) -> Result { - // TODO: read the following from the config file. - let index: u32 = config.local_interface_index()?; - let queueid: u32 = 0; + let ifindex: u32 = config.local_interface_index()?; + const QUEUEID: u32 = 0; // We do no use RSS, thus `queueid` is always 0. trace!("Creating XDP runtime."); let mut api: XdpApi = XdpApi::new()?; - let rx: RxRing = RxRing::new(&mut api, index, queueid)?; - let tx: TxRing = TxRing::new(&mut api, index, queueid)?; + // Open TX and RX rings + let rx: RxRing = RxRing::new(&mut api, ifindex, QUEUEID)?; + let tx: TxRing = TxRing::new(&mut api, ifindex, QUEUEID)?; Ok(Self { - api, - inner: SharedObject::new(CatpowderRuntimeInner { rx, tx }), + inner: SharedObject::new(CatpowderRuntimeInner { api, rx, tx }), }) } fn transmit(&mut self, pkt: Box) { let header_size: usize = pkt.header_size(); let body_size: usize = pkt.body_size(); - assert!(header_size + body_size < u16::MAX as usize); - trace!("header_size={:?}, body_size={:?}", header_size, body_size); + trace!("transmit(): header_size={:?}, body_size={:?}", header_size, body_size); + + if header_size + body_size >= u16::MAX as usize { + warn!("packet is too large: {:?}", header_size + body_size); + return; + } const COUNT: u32 = 1; let mut idx: u32 = 0; - assert!(self.inner.borrow_mut().tx.producer_reserve(COUNT, &mut idx) == COUNT); + if self.inner.borrow_mut().tx.producer_reserve(COUNT, &mut idx) != COUNT { + warn!("failed to reserve producer space for packet"); + return; + } let mut buf: XdpBuffer = self.inner.borrow_mut().tx.get_element(idx); buf.set_len(header_size + body_size); @@ -96,24 +109,21 @@ impl NetworkRuntime for CatpowderRuntime { self.inner.borrow_mut().tx.producer_submit(COUNT); // Notify socket. - let mut outflags = xdp_rs::XSK_NOTIFY_RESULT_FLAGS::default(); - self.inner - .borrow() - .tx - .notify_socket( - &mut self.api, - xdp_rs::_XSK_NOTIFY_FLAGS_XSK_NOTIFY_FLAG_POKE_TX | xdp_rs::_XSK_NOTIFY_FLAGS_XSK_NOTIFY_FLAG_WAIT_TX, - u32::MAX, - &mut outflags, - ) - .unwrap(); - - if self.inner.borrow_mut().tx.consumer_reserve(COUNT, &mut idx) == COUNT { - self.inner.borrow_mut().tx.consumer_release(COUNT); + let mut outflags: i32 = xdp_rs::XSK_NOTIFY_RESULT_FLAGS::default(); + let flags: i32 = + xdp_rs::_XSK_NOTIFY_FLAGS_XSK_NOTIFY_FLAG_POKE_TX | xdp_rs::_XSK_NOTIFY_FLAGS_XSK_NOTIFY_FLAG_WAIT_TX; + + if let Err(e) = self.inner.borrow_mut().notify_socket(flags, u32::MAX, &mut outflags) { + warn!("failed to notify socket: {:?}", e); + return; + } + + if self.inner.borrow_mut().tx.consumer_reserve(COUNT, &mut idx) != COUNT { + warn!("failed to send packet"); return; } - warn!("failed to send packet"); + self.inner.borrow_mut().tx.consumer_release(COUNT); } fn receive(&mut self) -> ArrayVec { diff --git a/src/rust/catpowder/win/rx_ring.rs b/src/rust/catpowder/win/rx_ring.rs index 7ba951a8e7..514eb6a202 100644 --- a/src/rust/catpowder/win/rx_ring.rs +++ b/src/rust/catpowder/win/rx_ring.rs @@ -28,13 +28,17 @@ use crate::{ //====================================================================================================================== pub struct RxRing { - program: XdpProgram, mem: UmemReg, - socket: XdpSocket, rx_ring: XdpRing, rx_fill_ring: XdpRing, + _program: XdpProgram, + _socket: XdpSocket, } +//====================================================================================================================== +// Implementations +//====================================================================================================================== + impl RxRing { pub fn new(api: &mut XdpApi, ifindex: u32, queueid: u32) -> Result { trace!("Creating XDP socket."); @@ -90,7 +94,7 @@ impl RxRing { let mut ring_index: u32 = 0; rx_fill_ring.ring_producer_reserve(RING_SIZE, &mut ring_index); - let b = rx_fill_ring.ring_get_element(ring_index) as *mut u64; + let b: *mut u64 = rx_fill_ring.ring_get_element(ring_index) as *mut u64; unsafe { *b = 0 }; trace!("Submitting RX ring buffer."); @@ -99,22 +103,19 @@ impl RxRing { trace!("Setting RX Fill ring."); // Create XDP program. + trace!("Creating XDP program..."); const XDP_INSPECT_RX: xdp_rs::XDP_HOOK_ID = xdp_rs::XDP_HOOK_ID { Layer: xdp_rs::_XDP_HOOK_LAYER_XDP_HOOK_L2, Direction: xdp_rs::_XDP_HOOK_DATAPATH_DIRECTION_XDP_HOOK_RX, SubLayer: xdp_rs::_XDP_HOOK_SUBLAYER_XDP_HOOK_INSPECT, }; - let rules: Vec = vec![XdpRule::new(&socket)]; - - trace!("Creating XDP program."); let program: XdpProgram = XdpProgram::new(api, &rules, ifindex, &XDP_INSPECT_RX, queueid, 0)?; - trace!("XDP program created."); Ok(Self { - program, + _program: program, mem, - socket, + _socket: socket, rx_ring, rx_fill_ring, }) diff --git a/src/rust/catpowder/win/socket.rs b/src/rust/catpowder/win/socket.rs index 719797f81d..cd5b5bd877 100644 --- a/src/rust/catpowder/win/socket.rs +++ b/src/rust/catpowder/win/socket.rs @@ -5,8 +5,10 @@ // Imports //====================================================================================================================== -use super::api::XdpApi; -use crate::runtime::fail::Fail; +use crate::{ + catpowder::win::api::XdpApi, + runtime::fail::Fail, +}; use ::windows::{ core::{ Error, @@ -21,8 +23,9 @@ use ::xdp_rs; //====================================================================================================================== /// A XDP socket. +#[repr(C)] pub struct XdpSocket { - pub socket: HANDLE, + socket: HANDLE, } //====================================================================================================================== @@ -32,7 +35,7 @@ pub struct XdpSocket { /// Associated functions for XDP sockets. impl XdpSocket { pub fn create(api: &mut XdpApi) -> Result { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); let mut socket: HANDLE = HANDLE::default(); if let Some(create) = api.XskCreate { @@ -50,7 +53,7 @@ impl XdpSocket { } pub fn bind(&self, api: &mut XdpApi, ifindex: u32, queueid: u32, flags: i32) -> Result<(), Fail> { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); if let Some(bind) = api.XskBind { let result: HRESULT = unsafe { bind(self.socket, ifindex, queueid, flags) }; @@ -76,7 +79,7 @@ impl XdpSocket { val: *const std::ffi::c_void, len: u32, ) -> Result<(), Fail> { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); if let Some(setsocket) = api.XskSetSockopt { let result: HRESULT = unsafe { setsocket(self.socket, opt, val, len) }; @@ -99,7 +102,7 @@ impl XdpSocket { val: *mut std::ffi::c_void, len: *mut u32, ) -> Result<(), Fail> { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); if let Some(getsockopt) = api.XskGetSockopt { let result: HRESULT = unsafe { getsockopt(self.socket, opt, val, len) }; @@ -116,7 +119,7 @@ impl XdpSocket { } pub fn activate(&self, api: &mut XdpApi, flags: i32) -> Result<(), Fail> { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); if let Some(activate) = api.XskActivate { let result: HRESULT = unsafe { activate(self.socket, flags) }; @@ -139,7 +142,7 @@ impl XdpSocket { timeout: u32, result: *mut xdp_rs::XSK_NOTIFY_RESULT_FLAGS, ) -> Result<(), Fail> { - let api: xdp_rs::XDP_API_TABLE = api.endpoint(); + let api: xdp_rs::XDP_API_TABLE = api.get(); if let Some(notify) = api.XskNotifySocket { let result: HRESULT = unsafe { notify(self.socket, flags, timeout, result) }; @@ -154,6 +157,10 @@ impl XdpSocket { Err(Fail::new(libc::ENOSYS, &cause)) } } + + pub fn get_socket(&self) -> HANDLE { + self.socket + } } pub struct XdpRing { diff --git a/src/rust/catpowder/win/tx_ring.rs b/src/rust/catpowder/win/tx_ring.rs index 69083c8f8e..2a7dbdbf1f 100644 --- a/src/rust/catpowder/win/tx_ring.rs +++ b/src/rust/catpowder/win/tx_ring.rs @@ -5,9 +5,9 @@ // Imports //====================================================================================================================== -use super::api::XdpApi; use crate::{ catpowder::win::{ + api::XdpApi, buffer::XdpBuffer, socket::{ XdpRing, @@ -25,16 +25,15 @@ use crate::{ // Structures //====================================================================================================================== -#[allow(dead_code)] pub struct TxRing { mem: UmemReg, socket: XdpSocket, tx_ring: XdpRing, - pub tx_completion_ring: XdpRing, + tx_completion_ring: XdpRing, } impl TxRing { - pub fn new(api: &mut XdpApi, index: u32, queueid: u32) -> Result { + pub fn new(api: &mut XdpApi, ifindex: u32, queueid: u32) -> Result { trace!("Creating XDP socket."); let mut socket: XdpSocket = XdpSocket::create(api)?; @@ -66,7 +65,7 @@ impl TxRing { )?; trace!("Binding TX queue."); - socket.bind(api, index, queueid, xdp_rs::_XSK_BIND_FLAGS_XSK_BIND_FLAG_TX)?; + socket.bind(api, ifindex, queueid, xdp_rs::_XSK_BIND_FLAGS_XSK_BIND_FLAG_TX)?; trace!("Activating XDP socket."); socket.activate(api, xdp_rs::_XSK_ACTIVATE_FLAGS_XSK_ACTIVATE_FLAG_NONE)?;