From 1c9ec669e2ac627eecae2fd7afefbebb726e2a0f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Imobach=20Gonz=C3=A1lez=20Sosa?= Date: Fri, 29 Dec 2023 08:34:09 +0000 Subject: [PATCH 1/2] Avoid a deadlock updating the network D-Bus tree Updating the tree must be done in a separate task. Otherwise, it could fight with an incoming request for the ObjectServer mutex, blocking the actions dispatching. --- rust/agama-dbus-server/src/network/system.rs | 50 +++++++++++++------- 1 file changed, 33 insertions(+), 17 deletions(-) diff --git a/rust/agama-dbus-server/src/network/system.rs b/rust/agama-dbus-server/src/network/system.rs index 416c9e261f..097c0fd5e0 100644 --- a/rust/agama-dbus-server/src/network/system.rs +++ b/rust/agama-dbus-server/src/network/system.rs @@ -1,8 +1,11 @@ use super::error::NetworkStateError; use crate::network::{dbus::Tree, model::Connection, Action, Adapter, NetworkState}; use agama_lib::network::types::DeviceType; -use std::error::Error; -use tokio::sync::mpsc::{self, UnboundedReceiver, UnboundedSender}; +use std::{error::Error, sync::Arc}; +use tokio::sync::{ + mpsc::{self, UnboundedReceiver, UnboundedSender}, + Mutex, +}; use uuid::Uuid; use zbus::zvariant::OwnedObjectPath; @@ -13,7 +16,7 @@ pub struct NetworkSystem { /// Side of the channel to send actions. actions_tx: UnboundedSender, actions_rx: UnboundedReceiver, - tree: Tree, + tree: Arc>, /// Adapter to read/write the network state. adapter: T, } @@ -26,7 +29,7 @@ impl NetworkSystem { state: NetworkState::default(), actions_tx, actions_rx, - tree, + tree: Arc::new(Mutex::new(tree)), adapter, } } @@ -48,10 +51,9 @@ impl NetworkSystem { /// Populates the D-Bus tree with the known devices and connections. pub async fn setup(&mut self) -> Result<(), Box> { self.state = self.adapter.read().await?; - self.tree - .set_connections(&mut self.state.connections) - .await?; - self.tree.set_devices(&self.state.devices).await?; + let mut tree = self.tree.lock().await; + tree.set_connections(&mut self.state.connections).await?; + tree.set_devices(&self.state.devices).await?; Ok(()) } @@ -78,15 +80,22 @@ impl NetworkSystem { tx.send(conn.cloned()).unwrap(); } Action::GetConnectionPath(id, tx) => { - let path = self.tree.connection_path(&id); + let tree = self.tree.lock().await; + let path = tree.connection_path(&id); tx.send(path).unwrap(); } Action::GetController(uuid, tx) => { let result = self.get_controller_action(uuid); tx.send(result).unwrap() } - Action::GetDevicesPaths(tx) => tx.send(self.tree.devices_paths()).unwrap(), - Action::GetConnectionsPaths(tx) => tx.send(self.tree.connections_paths()).unwrap(), + Action::GetDevicesPaths(tx) => { + let tree = self.tree.lock().await; + tx.send(tree.devices_paths()).unwrap(); + } + Action::GetConnectionsPaths(tx) => { + let tree = self.tree.lock().await; + tx.send(tree.connections_paths()).unwrap(); + } Action::SetPorts(uuid, ports, rx) => { let result = self.set_ports_action(uuid, *ports); rx.send(result).unwrap(); @@ -95,16 +104,23 @@ impl NetworkSystem { self.state.update_connection(*conn)?; } Action::RemoveConnection(id) => { - self.tree.remove_connection(&id).await?; + let mut tree = self.tree.lock().await; + tree.remove_connection(&id).await?; self.state.remove_connection(&id)?; } Action::Apply => { self.write().await?; // TODO: re-creating the tree is kind of brute-force and it sends signals about // adding/removing interfaces. We should add/update/delete objects as needed. - self.tree - .set_connections(&mut self.state.connections) - .await?; + // NOTE updating the tree at the same time than dispatching actions can cause a + // deadlock. We might consider using message passing too but at this point + // is enough to use a separate task. + let mut connections = self.state.connections.clone(); + let tree = Arc::clone(&self.tree); + tokio::spawn(async move { + let mut tree = tree.lock().await; + _ = tree.set_connections(&mut connections).await; + }); } } @@ -118,8 +134,8 @@ impl NetworkSystem { ) -> Result { let mut conn = Connection::new(name, ty); // TODO: handle tree handling problems - let path = self - .tree + let mut tree = self.tree.lock().await; + let path = tree .add_connection(&mut conn) .await .expect("Could not update the D-Bus tree"); From c40e25168b8b8b2fe318a26abc384d9c4bf9657f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Imobach=20Gonz=C3=A1lez=20Sosa?= Date: Fri, 29 Dec 2023 10:35:55 +0000 Subject: [PATCH 2/2] Log an error when the D-Bus tree cannot be updated --- rust/agama-dbus-server/src/network/system.rs | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/rust/agama-dbus-server/src/network/system.rs b/rust/agama-dbus-server/src/network/system.rs index 097c0fd5e0..5272d58596 100644 --- a/rust/agama-dbus-server/src/network/system.rs +++ b/rust/agama-dbus-server/src/network/system.rs @@ -119,7 +119,9 @@ impl NetworkSystem { let tree = Arc::clone(&self.tree); tokio::spawn(async move { let mut tree = tree.lock().await; - _ = tree.set_connections(&mut connections).await; + if let Err(e) = tree.set_connections(&mut connections).await { + log::error!("Could not update the D-Bus tree: {}", e); + } }); } }