Remove lock around tokio-rt handle (#36762)

It seems sub-optimal to to sequentialise execution by grabbing a lock,
each time we want to spawn a task onto the tokio runtime. We don't need
the lock either, so it makes sense to just remove it, which also
simplifies a bunch of the using code.

Testing: Covered by existing tests

Signed-off-by: Jonathan Schwender <schwenderjonathan@gmail.com>
This commit is contained in:
Jonathan Schwender 2025-04-30 09:09:38 +02:00 committed by GitHub
parent 955149b194
commit c46402e222
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
5 changed files with 37 additions and 44 deletions

View file

@ -2,31 +2,27 @@
* License, v. 2.0. If a copy of the MPL was not distributed with this * License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */ * file, You can obtain one at https://mozilla.org/MPL/2.0/. */
use std::cmp::Ord; use std::cmp::Ord;
use std::sync::LazyLock;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::{LazyLock, Mutex};
use std::thread; use std::thread;
use tokio::runtime::{Builder, Runtime}; use tokio::runtime::{Builder, Runtime};
pub static HANDLE: LazyLock<Mutex<Option<Runtime>>> = LazyLock::new(|| { pub static HANDLE: LazyLock<Runtime> = LazyLock::new(|| {
Mutex::new(Some( Builder::new_multi_thread()
Builder::new_multi_thread() .thread_name_fn(|| {
.thread_name_fn(|| { static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0);
static ATOMIC_ID: AtomicUsize = AtomicUsize::new(0); let id = ATOMIC_ID.fetch_add(1, Ordering::Relaxed);
let id = ATOMIC_ID.fetch_add(1, Ordering::Relaxed); format!("tokio-runtime-{}", id)
format!("tokio-runtime-{}", id) })
}) .worker_threads(
.worker_threads( thread::available_parallelism()
thread::available_parallelism() .map(|i| i.get())
.map(|i| i.get()) .unwrap_or(servo_config::pref!(threadpools_fallback_worker_num) as usize)
.unwrap_or(servo_config::pref!(threadpools_fallback_worker_num) as usize) .min(servo_config::pref!(threadpools_async_runtime_workers_max).max(1) as usize),
.min( )
servo_config::pref!(threadpools_async_runtime_workers_max).max(1) as usize, .enable_io()
), .enable_time()
) .build()
.enable_io() .expect("Unable to build tokio-runtime runtime")
.enable_time()
.build()
.unwrap(),
))
}); });

View file

@ -165,7 +165,7 @@ where
F: Future<Output = ()> + 'static + std::marker::Send, F: Future<Output = ()> + 'static + std::marker::Send,
{ {
fn execute(&self, fut: F) { fn execute(&self, fut: F) {
HANDLE.lock().unwrap().as_ref().unwrap().spawn(fut); HANDLE.spawn(fut);
} }
} }

View file

@ -493,7 +493,7 @@ impl BodySink {
match self { match self {
BodySink::Chunked(sender) => { BodySink::Chunked(sender) => {
let sender = sender.clone(); let sender = sender.clone();
HANDLE.lock().unwrap().as_mut().unwrap().spawn(async move { HANDLE.spawn(async move {
let _ = sender.send(Ok(Frame::data(bytes.into()))).await; let _ = sender.send(Ok(Frame::data(bytes.into()))).await;
}); });
}, },
@ -2016,7 +2016,7 @@ async fn http_network_fetch(
let url1 = request.url(); let url1 = request.url();
let url2 = url1.clone(); let url2 = url1.clone();
HANDLE.lock().unwrap().as_ref().unwrap().spawn( HANDLE.spawn(
res.into_body() res.into_body()
.map_err(|e| { .map_err(|e| {
warn!("Error streaming response body: {:?}", e); warn!("Error streaming response body: {:?}", e);

View file

@ -771,7 +771,7 @@ impl CoreResourceManager {
_ => (FileTokenCheck::NotRequired, None), _ => (FileTokenCheck::NotRequired, None),
}; };
HANDLE.lock().unwrap().as_ref().unwrap().spawn(async move { HANDLE.spawn(async move {
// XXXManishearth: Check origin against pipeline id (also ensure that the mode is allowed) // XXXManishearth: Check origin against pipeline id (also ensure that the mode is allowed)
// todo load context / mimesniff in fetch // todo load context / mimesniff in fetch
// todo referrer policy? // todo referrer policy?

View file

@ -418,24 +418,21 @@ fn connect(
tls_config.alpn_protocols = vec!["http/1.1".to_string().into()]; tls_config.alpn_protocols = vec!["http/1.1".to_string().into()];
let resource_event_sender2 = resource_event_sender.clone(); let resource_event_sender2 = resource_event_sender.clone();
match HANDLE.lock().unwrap().as_mut() { HANDLE.spawn(
Some(handle) => handle.spawn( start_websocket(
start_websocket( http_state,
http_state, req_url.clone(),
req_url.clone(), resource_event_sender,
resource_event_sender, protocols,
protocols, client,
client, tls_config,
tls_config, dom_action_receiver,
dom_action_receiver, )
) .map_err(move |e| {
.map_err(move |e| { warn!("Failed to establish a WebSocket connection: {:?}", e);
warn!("Failed to establish a WebSocket connection: {:?}", e); let _ = resource_event_sender2.send(WebSocketNetworkEvent::Fail);
let _ = resource_event_sender2.send(WebSocketNetworkEvent::Fail); }),
}), );
),
None => return Err("No runtime available".to_string()),
};
Ok(()) Ok(())
} }