b6d35f6faf
Updated 4763 files with dual copyright: - Parity Technologies (UK) Ltd. - Dijital Kurdistan Tech Institute
101 lines
3.0 KiB
Rust
101 lines
3.0 KiB
Rust
// This file is part of Bizinikiwi.
|
|
|
|
// Copyright (C) Parity Technologies (UK) Ltd. and Dijital Kurdistan Tech Institute
|
|
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
use pezsp_core::{testing::TaskExecutor, traits::SpawnNamed};
|
|
use std::sync::{atomic::AtomicUsize, Arc};
|
|
use tokio::sync::mpsc;
|
|
|
|
/// Wrap the `TaskExecutor` to know when the broadcast future is dropped.
|
|
#[derive(Clone)]
|
|
pub struct TaskExecutorBroadcast {
|
|
executor: TaskExecutor,
|
|
sender: mpsc::UnboundedSender<()>,
|
|
num_tasks: Arc<AtomicUsize>,
|
|
}
|
|
|
|
/// The channel that receives events when the broadcast futures are dropped.
|
|
pub type TaskExecutorRecv = mpsc::UnboundedReceiver<()>;
|
|
|
|
/// The state of the `TaskExecutorBroadcast`.
|
|
pub struct TaskExecutorState {
|
|
pub recv: TaskExecutorRecv,
|
|
pub num_tasks: Arc<AtomicUsize>,
|
|
}
|
|
|
|
impl TaskExecutorState {
|
|
pub fn num_tasks(&self) -> usize {
|
|
self.num_tasks.load(std::sync::atomic::Ordering::Acquire)
|
|
}
|
|
}
|
|
|
|
impl TaskExecutorBroadcast {
|
|
/// Construct a new `TaskExecutorBroadcast` and a receiver to know when the broadcast futures
|
|
/// are dropped.
|
|
pub fn new() -> (Self, TaskExecutorState) {
|
|
let (sender, recv) = mpsc::unbounded_channel();
|
|
let num_tasks = Arc::new(AtomicUsize::new(0));
|
|
|
|
(
|
|
Self { executor: TaskExecutor::new(), sender, num_tasks: num_tasks.clone() },
|
|
TaskExecutorState { recv, num_tasks },
|
|
)
|
|
}
|
|
}
|
|
|
|
impl SpawnNamed for TaskExecutorBroadcast {
|
|
fn spawn(
|
|
&self,
|
|
name: &'static str,
|
|
group: Option<&'static str>,
|
|
future: futures::future::BoxFuture<'static, ()>,
|
|
) {
|
|
let sender = self.sender.clone();
|
|
let num_tasks = self.num_tasks.clone();
|
|
|
|
let future = Box::pin(async move {
|
|
num_tasks.fetch_add(1, std::sync::atomic::Ordering::AcqRel);
|
|
future.await;
|
|
num_tasks.fetch_sub(1, std::sync::atomic::Ordering::AcqRel);
|
|
|
|
let _ = sender.send(());
|
|
});
|
|
|
|
self.executor.spawn(name, group, future)
|
|
}
|
|
|
|
fn spawn_blocking(
|
|
&self,
|
|
name: &'static str,
|
|
group: Option<&'static str>,
|
|
future: futures::future::BoxFuture<'static, ()>,
|
|
) {
|
|
let sender = self.sender.clone();
|
|
let num_tasks = self.num_tasks.clone();
|
|
|
|
let future = Box::pin(async move {
|
|
num_tasks.fetch_add(1, std::sync::atomic::Ordering::AcqRel);
|
|
future.await;
|
|
num_tasks.fetch_sub(1, std::sync::atomic::Ordering::AcqRel);
|
|
|
|
let _ = sender.send(());
|
|
});
|
|
|
|
self.executor.spawn_blocking(name, group, future)
|
|
}
|
|
}
|