From ae2af10cf84475da13f26722769796bfbf652502 Mon Sep 17 00:00:00 2001 From: Florian Klink Date: Sun, 29 Jun 2025 01:15:04 +0300 Subject: [PATCH] fix(castore/fs): enter the runtime context before blocking on tasks fuse-backend-rs spawns multiple threads, but these don't automatically inherit the tokio runtime context. Spawning tasks from "the wrong thread" would then cause a panic, as seen in https://git.snix.dev/snix/snix/issues/147. However, cl/30575 didn't really fix this, it only removed one place spawning tasks, without fixing the underlying issue. PathInfoService / BlobService / DirectoryService are expected to spawn tasks. We need to simply invoke `.enter()` from all worker threads, and that's what this CL does. An alternative would be to manually (re)-enter inside every function of the FileSystem trait, but that'd be very messy. A similar fix needs to end up in the virtiofs implementation, but we don't have control over the (single) thread being spawned by VhostUserDaemon there, so cannot just enter the runtime context there, so virtiofs will stay broken for now. Maybe it's time to re-architect this a bit - have our FileSystem impl be little code and call to sync endpoints to do the actual work, which is then handled by workers on another thread - but that's left for another CL. Change-Id: I58cdbd952f4ecc39bdc2f2fa69a788caa0cc78ba Reviewed-on: https://cl.snix.dev/c/snix/+/30585 Tested-by: besadii Reviewed-by: Vova Kryachko --- snix/castore/src/fs/fuse/mod.rs | 16 +++++++++++++--- 1 file changed, 13 insertions(+), 3 deletions(-) diff --git a/snix/castore/src/fs/fuse/mod.rs b/snix/castore/src/fs/fuse/mod.rs index ae3ef92ac..aab3cb0bc 100644 --- a/snix/castore/src/fs/fuse/mod.rs +++ b/snix/castore/src/fs/fuse/mod.rs @@ -25,7 +25,9 @@ impl FuseServer where FS: FileSystem + Sync + Send, { - fn start(&mut self) -> io::Result<()> { + fn start(&mut self, tokio_handle: tokio::runtime::Handle) -> io::Result<()> { + let _guard = tokio_handle.enter(); + while let Some((reader, writer)) = self .channel .get_request() @@ -88,6 +90,9 @@ impl FuseDaemon { .thread_name("fuse_server".to_string()) .build(); + // get a handle to the current tokio runtime + let runtime_handle = tokio::runtime::Handle::current(); + for _ in 0..num_threads { // for each thread requested, create and start a FuseServer accepting requests. let mut server = FuseServer { @@ -97,8 +102,13 @@ impl FuseDaemon { .map_err(|e| io::Error::new(io::ErrorKind::Other, e.to_string()))?, }; - threads.execute(move || { - let _ = server.start(); + // Start the FuseServer in each thread, and enter the tokio runtime context, + // so we can block on tasks. + threads.execute({ + let runtime_handle = runtime_handle.clone(); + move || { + let _ = server.start(runtime_handle); + } }); }