|
@@ -4,8 +4,6 @@ use std::{
|
|
|
rc::{Rc, Weak},
|
|
|
};
|
|
|
|
|
|
-use super::priority::AbstractServicePriority;
|
|
|
-
|
|
|
pub struct Event<Data: 'static> {
|
|
|
pub data: Data,
|
|
|
pub emptied: bool,
|
|
@@ -34,92 +32,29 @@ impl<Host: 'static, Context: 'static + ?Sized, Data: 'static> EventSub<Context,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-pub struct EventChannel<Tag: 'static, Context: 'static + ?Sized, Data: 'static> {
|
|
|
- subs: RefCell<Vec<Box<dyn EventSub<Context, Data>>>>,
|
|
|
+pub struct Channel<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static> {
|
|
|
+ /*subs: RefCell<super::priority::TotalOrder<Rc<dyn EventSub<Context, Data>>>>,
|
|
|
+ order_cache: RefCell<Option<Vec<Rc<dyn EventSub<Context, Data>>>>>,*/
|
|
|
+ subs: RefCell<Vec<(Priority, Rc<dyn EventSub<Context, Data>>)>>,
|
|
|
|
|
|
queue: RefCell<VecDeque<Data>>,
|
|
|
|
|
|
- _ghost: std::marker::PhantomData<(Tag,)>,
|
|
|
+ _ghost: std::marker::PhantomData<(Tag,Priority)>,
|
|
|
}
|
|
|
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Default
|
|
|
- for EventChannel<Tag, Context, Data>
|
|
|
-{
|
|
|
- fn default() -> Self {
|
|
|
- Self {
|
|
|
- subs: RefCell::new(Vec::new()),
|
|
|
- queue: Default::default(),
|
|
|
- _ghost: std::marker::PhantomData,
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> EventChannel<Tag, Context, Data> {
|
|
|
- pub fn subscribe<
|
|
|
- Host: 'static,
|
|
|
- HC: crate::helper::IntoWeak<Host>,
|
|
|
- CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
|
- >(
|
|
|
- &self,
|
|
|
- who: HC,
|
|
|
- cb: CB,
|
|
|
- ) {
|
|
|
- self.subs.borrow_mut().push(Box::new(ConcreteEventSub {
|
|
|
- host: who.as_weak(),
|
|
|
- callback: Box::new(cb),
|
|
|
- }));
|
|
|
- }
|
|
|
-
|
|
|
- pub fn queue(&self, data: Data) {
|
|
|
- self.queue.borrow_mut().push_back(data);
|
|
|
- }
|
|
|
-
|
|
|
- fn do_fire_all(&self, context: &Context) -> usize {
|
|
|
- let mut count = 0;
|
|
|
-
|
|
|
- while !self.queue.borrow().is_empty() {
|
|
|
- let mut event = Event {
|
|
|
- data: self.queue.borrow_mut().pop_front().unwrap(),
|
|
|
- emptied: false,
|
|
|
- };
|
|
|
-
|
|
|
- for sub in self.subs.borrow().iter() {
|
|
|
- sub.invoke(context, &mut event);
|
|
|
- if event.emptied {
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- count += 1;
|
|
|
- }
|
|
|
-
|
|
|
- count
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-pub struct PriorityChannel<Tag: 'static, Context: 'static + ?Sized, Data: 'static> {
|
|
|
- subs: RefCell<super::priority::TotalOrder<Rc<dyn EventSub<Context, Data>>>>,
|
|
|
- order_cache: RefCell<Option<Vec<Rc<dyn EventSub<Context, Data>>>>>,
|
|
|
-
|
|
|
- queue: RefCell<VecDeque<Data>>,
|
|
|
-
|
|
|
- _ghost: std::marker::PhantomData<(Tag,)>,
|
|
|
-}
|
|
|
-
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Default
|
|
|
- for PriorityChannel<Tag, Context, Data>
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static> Default
|
|
|
+ for Channel<Tag, Context, Data, Priority>
|
|
|
{
|
|
|
fn default() -> Self {
|
|
|
Self {
|
|
|
subs: RefCell::new(Default::default()),
|
|
|
- order_cache: Default::default(),
|
|
|
queue: RefCell::new(Default::default()),
|
|
|
_ghost: std::marker::PhantomData,
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-pub struct PrioritySubscriptionBuilder<
|
|
|
+/*pub struct PrioritySubscriptionBuilder<
|
|
|
'l,
|
|
|
Tag: 'static,
|
|
|
Context: 'static + ?Sized,
|
|
@@ -150,10 +85,11 @@ impl<
|
|
|
self.parent.subscribe::<Priority, Host, HC, CB>(who, cb);
|
|
|
}
|
|
|
}
|
|
|
+*/
|
|
|
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> PriorityChannel<Tag, Context, Data> {
|
|
|
+
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Channel<Tag, Context, Data, NoPriorityTag> {
|
|
|
pub fn subscribe<
|
|
|
- P: super::AbstractServicePriority,
|
|
|
Host: 'static,
|
|
|
HC: crate::helper::IntoWeak<Host>,
|
|
|
CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
@@ -167,27 +103,39 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> PriorityChannel<Tag
|
|
|
callback: Box::new(cb),
|
|
|
});
|
|
|
|
|
|
- self.subs.borrow_mut().add_priority::<P>(sub);
|
|
|
- self.order_cache.borrow_mut().take();
|
|
|
+ self.subs.borrow_mut().push((NoPriorityTag {}, sub));
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
- pub fn with<P: AbstractServicePriority>(
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: PartialOrd> Channel<Tag, Context, Data, Priority> {
|
|
|
+ pub fn subscribe<
|
|
|
+ Host: 'static,
|
|
|
+ HC: crate::helper::IntoWeak<Host>,
|
|
|
+ CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
|
+ >(
|
|
|
&self,
|
|
|
- ) -> PrioritySubscriptionBuilder<Tag, Context, Data, P> {
|
|
|
- PrioritySubscriptionBuilder {
|
|
|
- parent: self,
|
|
|
- _ghost: std::marker::PhantomData,
|
|
|
- }
|
|
|
+ p: Priority,
|
|
|
+ who: HC,
|
|
|
+ cb: CB,
|
|
|
+ ) {
|
|
|
+ let sub = Rc::new(ConcreteEventSub {
|
|
|
+ host: who.as_weak(),
|
|
|
+ callback: Box::new(cb),
|
|
|
+ });
|
|
|
+
|
|
|
+ let mut subs = self.subs.borrow_mut();
|
|
|
+ subs.push((p.into(), sub));
|
|
|
+ // XXX: what happens if we actually get an undefined order?...
|
|
|
+ subs.sort_by(|a,b| a.0.partial_cmp(&b.0).unwrap());
|
|
|
}
|
|
|
+}
|
|
|
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static> Channel<Tag, Context, Data, Priority> {
|
|
|
pub fn queue(&self, data: Data) {
|
|
|
self.queue.borrow_mut().push_back(data);
|
|
|
}
|
|
|
|
|
|
fn do_fire_all(&self, context: &Context) -> usize {
|
|
|
- let mut cache = self.order_cache.borrow_mut();
|
|
|
- let subs = cache.get_or_insert_with(|| self.subs.borrow().clone().order());
|
|
|
-
|
|
|
let mut count = 0;
|
|
|
|
|
|
while !self.queue.borrow().is_empty() {
|
|
@@ -196,8 +144,8 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> PriorityChannel<Tag
|
|
|
emptied: false,
|
|
|
};
|
|
|
|
|
|
- for sub in subs.iter() {
|
|
|
- sub.invoke(context, &mut event);
|
|
|
+ for sub in self.subs.borrow().iter() {
|
|
|
+ sub.1.invoke(context, &mut event);
|
|
|
if event.emptied {
|
|
|
break;
|
|
|
}
|
|
@@ -215,20 +163,8 @@ trait ChannelMetadata<Context: 'static + ?Sized> {
|
|
|
fn fire_all(&self, context: &Context) -> usize;
|
|
|
}
|
|
|
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> ChannelMetadata<Context>
|
|
|
- for EventChannel<Tag, Context, Data>
|
|
|
-{
|
|
|
- fn queue_len(&self) -> usize {
|
|
|
- self.queue.borrow().len()
|
|
|
- }
|
|
|
-
|
|
|
- fn fire_all(&self, context: &Context) -> usize {
|
|
|
- self.do_fire_all(context)
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> ChannelMetadata<Context>
|
|
|
- for PriorityChannel<Tag, Context, Data>
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static> ChannelMetadata<Context>
|
|
|
+ for Channel<Tag, Context, Data, Priority>
|
|
|
{
|
|
|
fn queue_len(&self) -> usize {
|
|
|
self.queue.borrow().len()
|
|
@@ -242,11 +178,21 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> ChannelMetadata<Con
|
|
|
pub trait ChannelSpec {
|
|
|
type Tag: 'static;
|
|
|
type Data: 'static;
|
|
|
+ type Priority: 'static;
|
|
|
+}
|
|
|
+
|
|
|
+pub struct NoPriorityTag {}
|
|
|
+
|
|
|
+impl<Tag: 'static, Data: 'static, Priority: 'static + PartialOrd> ChannelSpec for (Tag, Data, Priority) {
|
|
|
+ type Tag = Tag;
|
|
|
+ type Data = Data;
|
|
|
+ type Priority = Priority;
|
|
|
}
|
|
|
|
|
|
impl<Tag: 'static, Data: 'static> ChannelSpec for (Tag, Data) {
|
|
|
type Tag = Tag;
|
|
|
type Data = Data;
|
|
|
+ type Priority = NoPriorityTag;
|
|
|
}
|
|
|
|
|
|
pub struct EventRoot<Context: 'static + ?Sized> {
|
|
@@ -267,24 +213,24 @@ impl<Context: 'static + ?Sized> Default for EventRoot<Context> {
|
|
|
|
|
|
impl<Context: 'static + ?Sized> EventRoot<Context> {
|
|
|
pub fn create_channel<CS: ChannelSpec>(&mut self) {
|
|
|
- let tid = std::any::TypeId::of::<EventChannel<CS::Tag, Context, CS::Data>>();
|
|
|
+ let tid = std::any::TypeId::of::<Channel<CS::Tag, Context, CS::Data, CS::Priority>>();
|
|
|
|
|
|
- let ch = Rc::new(EventChannel::<CS::Tag, Context, CS::Data>::default());
|
|
|
+ let ch = Rc::new(Channel::<CS::Tag, Context, CS::Data, CS::Priority>::default());
|
|
|
self.metadata
|
|
|
.insert(tid, (std::any::type_name::<CS>(), ch.clone()));
|
|
|
self.channels.insert(tid, ch);
|
|
|
}
|
|
|
|
|
|
- pub fn create_priority_channel<CS: ChannelSpec>(&mut self) {
|
|
|
+ /*pub fn create_priority_channel<CS: ChannelSpec>(&mut self) {
|
|
|
let tid = std::any::TypeId::of::<PriorityChannel<CS::Tag, Context, CS::Data>>();
|
|
|
|
|
|
let ch = Rc::new(PriorityChannel::<CS::Tag, Context, CS::Data>::default());
|
|
|
self.metadata
|
|
|
.insert(tid, (std::any::type_name::<CS>(), ch.clone()));
|
|
|
self.channels.insert(tid, ch);
|
|
|
- }
|
|
|
+ }*/
|
|
|
|
|
|
- pub fn channel<CS: ChannelSpec>(&self) -> &EventChannel<CS::Tag, Context, CS::Data> {
|
|
|
+ /*pub fn channel<CS: ChannelSpec>(&self) -> &EventChannel<CS::Tag, Context, CS::Data> {
|
|
|
let tid = std::any::TypeId::of::<EventChannel<CS::Tag, Context, CS::Data>>();
|
|
|
|
|
|
match self.channels.get(&tid) {
|
|
@@ -296,26 +242,20 @@ impl<Context: 'static + ?Sized> EventRoot<Context> {
|
|
|
)
|
|
|
}
|
|
|
}
|
|
|
- }
|
|
|
+ }*/
|
|
|
|
|
|
- pub fn priority_channel<CS: ChannelSpec>(
|
|
|
+ pub fn channel<CS: ChannelSpec>(
|
|
|
&self,
|
|
|
- ) -> &PriorityChannel<CS::Tag, Context, CS::Data> {
|
|
|
- let tid = std::any::TypeId::of::<PriorityChannel<CS::Tag, Context, CS::Data>>();
|
|
|
+ ) -> &Channel<CS::Tag, Context, CS::Data, CS::Priority> {
|
|
|
+ let tid = std::any::TypeId::of::<Channel<CS::Tag, Context, CS::Data, CS::Priority>>();
|
|
|
|
|
|
match self.channels.get(&tid) {
|
|
|
Some(ch) => ch.downcast_ref().expect("internal inconsistency"),
|
|
|
None => {
|
|
|
- if self.channels.contains_key(&std::any::TypeId::of::<
|
|
|
- EventChannel<CS::Tag, Context, CS::Data>,
|
|
|
- >()) {
|
|
|
- panic!("Asked for priority channel {}, which does not exist, but there is a non-priority channel!", std::any::type_name::<CS>())
|
|
|
- } else {
|
|
|
- panic!(
|
|
|
- "Asked for priority channel {} that has not been created!",
|
|
|
- std::any::type_name::<CS>()
|
|
|
- )
|
|
|
- }
|
|
|
+ panic!(
|
|
|
+ "Asked for channel {} that has not been created!",
|
|
|
+ std::any::type_name::<CS>()
|
|
|
+ )
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -345,7 +285,6 @@ mod tests {
|
|
|
};
|
|
|
|
|
|
use super::{Event, EventRoot};
|
|
|
- use crate::service::order;
|
|
|
|
|
|
struct Receiver {
|
|
|
int_count: Cell<usize>,
|
|
@@ -371,6 +310,14 @@ mod tests {
|
|
|
struct IntTag;
|
|
|
type IntChannel = (IntTag, i32);
|
|
|
|
|
|
+ #[derive(PartialEq, PartialOrd)]
|
|
|
+ enum IntPriority {
|
|
|
+ First,
|
|
|
+ Second,
|
|
|
+ Third
|
|
|
+ }
|
|
|
+ type IntPriorityChannel = (IntTag, i32, IntPriority);
|
|
|
+
|
|
|
#[test]
|
|
|
fn simple_fire() {
|
|
|
let mut root = EventRoot::default();
|
|
@@ -393,7 +340,7 @@ mod tests {
|
|
|
fn priority_fire() {
|
|
|
let mut root = EventRoot::default();
|
|
|
|
|
|
- root.create_priority_channel::<IntChannel>();
|
|
|
+ root.create_channel::<IntPriorityChannel>();
|
|
|
|
|
|
let order = Rc::new(RefCell::new(Vec::new()));
|
|
|
|
|
@@ -405,16 +352,20 @@ mod tests {
|
|
|
id: 2,
|
|
|
order: order.clone(),
|
|
|
});
|
|
|
+ let recv3 = Rc::new(OrderedReceiver {
|
|
|
+ id: 3,
|
|
|
+ order: order.clone(),
|
|
|
+ });
|
|
|
|
|
|
- root.priority_channel::<IntChannel>()
|
|
|
- .with::<order::First>()
|
|
|
- .subscribe(&recv1, OrderedReceiver::receive_i32);
|
|
|
- root.priority_channel::<IntChannel>()
|
|
|
- .with::<order::Last>()
|
|
|
- .subscribe(&recv2, OrderedReceiver::receive_i32);
|
|
|
- root.priority_channel::<IntChannel>().queue(0i32);
|
|
|
+ root.channel::<IntPriorityChannel>()
|
|
|
+ .subscribe(IntPriority::Second, &recv2, OrderedReceiver::receive_i32);
|
|
|
+ root.channel::<IntPriorityChannel>()
|
|
|
+ .subscribe(IntPriority::First, &recv1, OrderedReceiver::receive_i32);
|
|
|
+ root.channel::<IntPriorityChannel>()
|
|
|
+ .subscribe(IntPriority::Third, &recv3, OrderedReceiver::receive_i32);
|
|
|
+ root.channel::<IntPriorityChannel>().queue(0i32);
|
|
|
assert_eq!(order.borrow().deref(), &vec![]);
|
|
|
- root.priority_channel::<IntChannel>().do_fire_all(&());
|
|
|
- assert_eq!(order.borrow().deref(), &vec![1, 2]);
|
|
|
+ root.channel::<IntPriorityChannel>().do_fire_all(&());
|
|
|
+ assert_eq!(order.borrow().deref(), &vec![1, 2, 3]);
|
|
|
}
|
|
|
}
|