diff --git a/ntex-server/CHANGES.md b/ntex-server/CHANGES.md index fec1ffa7..a1c62378 100644 --- a/ntex-server/CHANGES.md +++ b/ntex-server/CHANGES.md @@ -1,5 +1,9 @@ # Changes +## [2.7.0] - 2025-01-31 + +* Cpu affinity support for workers + ## [2.6.2] - 2024-12-30 * Fix error log diff --git a/ntex-server/Cargo.toml b/ntex-server/Cargo.toml index 88575028..deb51ba3 100644 --- a/ntex-server/Cargo.toml +++ b/ntex-server/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "ntex-server" -version = "2.6.2" +version = "2.7.0" authors = ["ntex contributors "] description = "Server for ntex framework" keywords = ["network", "framework", "async", "futures"] @@ -24,6 +24,7 @@ ntex-util = "2.8" async-channel = "2" async-broadcast = "0.7" +core_affinity = "0.8" polling = "3.3" log = "0.4" socket2 = "0.5" diff --git a/ntex-server/src/manager.rs b/ntex-server/src/manager.rs index ae0f7974..c37aad79 100644 --- a/ntex-server/src/manager.rs +++ b/ntex-server/src/manager.rs @@ -2,6 +2,7 @@ use std::sync::atomic::{AtomicBool, Ordering}; use std::{cell::Cell, cell::RefCell, collections::VecDeque, rc::Rc, sync::Arc}; use async_channel::{unbounded, Receiver, Sender}; +use core_affinity::CoreId; use ntex_rt::System; use ntex_util::future::join_all; use ntex_util::time::{sleep, timeout, Millis}; @@ -69,9 +70,16 @@ impl ServerManager { // handle cmd let _ = ntex_rt::spawn(handle_cmd(mgr.clone(), rx)); + // Retrieve the IDs of all active CPU cores. + let mut cores = if cfg.affinity { + core_affinity::get_core_ids().unwrap_or_default() + } else { + Vec::new() + }; + // start workers for _ in 0..mgr.0.cfg.num { - start_worker(mgr.clone()); + start_worker(mgr.clone(), cores.pop()); } let srv = Server::new(tx, shared); @@ -128,9 +136,16 @@ impl ServerManager { } } -fn start_worker(mgr: ServerManager) { +fn start_worker(mgr: ServerManager, cid: Option) { let _ = ntex_rt::spawn(async move { let id = mgr.next_id(); + + if let Some(cid) = cid { + if core_affinity::set_for_current(cid) { + log::info!("Set affinity to {:?} for worker {:?}", cid, id); + } + } + let mut wrk = Worker::start(id, mgr.factory()); loop { diff --git a/ntex-server/src/net/builder.rs b/ntex-server/src/net/builder.rs index 0569f36a..9b11684a 100644 --- a/ntex-server/src/net/builder.rs +++ b/ntex-server/src/net/builder.rs @@ -110,6 +110,14 @@ impl ServerBuilder { self } + /// Enable cpu affinity + /// + /// By default affinity is disabled. + pub fn enable_affinity(mut self) -> Self { + self.pool = self.pool.enable_affinity(); + self + } + /// Timeout for graceful workers shutdown. /// /// After receiving a stop signal, workers have this much time to finish diff --git a/ntex-server/src/pool.rs b/ntex-server/src/pool.rs index 229ea8ba..20271a2f 100644 --- a/ntex-server/src/pool.rs +++ b/ntex-server/src/pool.rs @@ -11,6 +11,7 @@ pub struct WorkerPool { pub(crate) no_signals: bool, pub(crate) stop_runtime: bool, pub(crate) shutdown_timeout: Millis, + pub(crate) affinity: bool, } impl Default for WorkerPool { @@ -28,6 +29,7 @@ impl WorkerPool { no_signals: false, stop_runtime: false, shutdown_timeout: DEFAULT_SHUTDOWN_TIMEOUT, + affinity: false, } } @@ -68,6 +70,14 @@ impl WorkerPool { self } + /// Enable core affinity + /// + /// By default affinity is disabled. + pub fn enable_affinity(mut self) -> Self { + self.affinity = true; + self + } + /// Starts processing incoming items and return server controller. pub fn run(self, factory: F) -> Server { crate::manager::ServerManager::start(self, factory) diff --git a/ntex/CHANGES.md b/ntex/CHANGES.md index 24eb9de4..ace6a49a 100644 --- a/ntex/CHANGES.md +++ b/ntex/CHANGES.md @@ -1,5 +1,9 @@ # Changes +## [2.11.0] - 2025-01-31 + +* Cpu affinity support for server + ## [2.10.0] - 2024-12-04 * Use updated Service trait diff --git a/ntex/Cargo.toml b/ntex/Cargo.toml index 709f57f1..96affad8 100644 --- a/ntex/Cargo.toml +++ b/ntex/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "ntex" -version = "2.10.0" +version = "2.11.0" authors = ["ntex contributors "] description = "Framework for composable network services" readme = "README.md" @@ -68,8 +68,8 @@ ntex-service = "3.4" ntex-macros = "0.1" ntex-util = "2.8" ntex-bytes = "0.1.27" -ntex-server = "2.6" -ntex-h2 = "1.5" +ntex-server = "2.7" +ntex-h2 = "1.8.1" ntex-rt = "0.4.22" ntex-io = "2.9" ntex-net = "2.4" diff --git a/ntex/src/web/server.rs b/ntex/src/web/server.rs index 5b774b97..05bb12b6 100644 --- a/ntex/src/web/server.rs +++ b/ntex/src/web/server.rs @@ -310,6 +310,14 @@ where self } + /// Enable cpu affinity + /// + /// By default affinity is disabled. + pub fn enable_affinity(mut self) -> Self { + self.builder = self.builder.enable_affinity(); + self + } + /// Set io tag for web server pub fn tag(self, tag: &'static str) -> Self { self.config.lock().unwrap().tag = tag;