separate task and scope tasks

This commit is contained in:
Evan Almloff 2024-03-04 13:04:22 -06:00
parent 78b9b157dc
commit c4b8ebc1cf
4 changed files with 135 additions and 63 deletions

View file

@ -2,7 +2,7 @@ use std::ops::{Deref, DerefMut};
use crate::{
any_props::AnyProps,
innerlude::{DirtyScope, ElementRef, MountId, VComponent, WriteMutations},
innerlude::{DirtyScopes,ScopeOrder, ElementRef, MountId, VComponent, WriteMutations},
nodes::RenderReturn,
nodes::VNode,
scopes::ScopeId,
@ -91,7 +91,7 @@ impl VNode {
dom.diff_scope(to, scope_id, new);
let height = dom.runtime.get_state(scope_id).unwrap().height;
dom.dirty_scopes.remove(&DirtyScope::new(height, scope_id));
dom.dirty_scopes.remove(&ScopeOrder::new(height, scope_id));
}
fn replace_vcomponent(

View file

@ -4,8 +4,9 @@ use std::borrow::Borrow;
use std::cell::Cell;
use std::cell::RefCell;
use std::hash::Hash;
use std::collections::BTreeSet;
#[derive(Debug, Clone, Eq)]
#[derive(Debug, Clone, Copy, Eq)]
pub struct ScopeOrder {
pub(crate) height: u32,
pub(crate) id: ScopeId,
@ -41,62 +42,149 @@ impl Hash for ScopeOrder {
}
}
#[derive(Debug, Default)]
pub struct DirtyScopes {
pub(crate) scopes: BTreeSet<ScopeOrder>,
pub(crate) tasks: BTreeSet<DirtyTasks>,
}
impl DirtyScopes {
pub fn queue_task(&mut self, task: Task, order: ScopeOrder) {
match self.tasks.get(&order) {
Some(scope) => scope.queue_task(task),
None => {
let mut scope = DirtyTasks::from(order);
scope.queue_task(task);
self.tasks.insert(scope);
}
}
}
pub fn queue_scope(&mut self, order: ScopeOrder) {
self.scopes.insert(order);
}
pub fn has_dirty_scopes(&self) -> bool {
!self.scopes.is_empty()
}
pub fn pop_task(&mut self) -> Option<DirtyTasks> {
self.tasks.pop_first()
}
pub fn pop_scope(&mut self) -> Option<ScopeOrder> {
self.scopes.pop_first()
}
pub fn pop_work(&mut self) -> Option<Work> {
let dirty_scope = self.scopes.first();
let dirty_task = self.tasks.first();
match (dirty_scope, dirty_task) {
(Some(scope), Some(task)) => {
let tasks_order = task.borrow();
if scope > tasks_order{
let scope = self.scopes.pop_first().unwrap();
Some(Work{
scope: scope,
rerun_scope: true,
tasks: Vec::new(),
})
} else if tasks_order> scope {
let task = self.tasks.pop_first().unwrap();
Some(Work{
scope: task.order,
rerun_scope: false,
tasks: task.tasks_queued.into_inner(),
})
}
else {
let scope = self.scopes.pop_first().unwrap();
let task = self.tasks.pop_first().unwrap();
Some(Work{
scope: scope,
rerun_scope: true,
tasks: task.tasks_queued.into_inner(),
})
}
}
(Some(scope), None) => {
let scope = self.scopes.pop_first().unwrap();
Some(Work{
scope: scope,
rerun_scope: true,
tasks: Vec::new(),
})
}
(None, Some(task)) => {
let task = self.tasks.pop_first().unwrap();
Some(Work{
scope: task.order,
rerun_scope: false,
tasks: task.tasks_queued.into_inner(),
})
}
(None, None) => None
}
}
pub fn remove(&mut self, scope: &ScopeOrder) {
self.scopes.remove(scope);
}
}
pub struct Work {
pub scope: ScopeOrder,
pub rerun_scope: bool,
pub tasks: Vec<Task>,
}
#[derive(Debug, Clone, Eq)]
pub struct DirtyScope {
pub(crate) struct DirtyTasks {
pub order: ScopeOrder,
pub rerun_queued: Cell<bool>,
pub tasks_queued: RefCell<Vec<Task>>,
}
impl From<ScopeOrder> for DirtyScope {
impl From<ScopeOrder> for DirtyTasks {
fn from(order: ScopeOrder) -> Self {
Self {
order,
rerun_queued: false.into(),
tasks_queued: Vec::new().into(),
}
}
}
impl DirtyScope {
pub fn new(height: u32, id: ScopeId) -> Self {
ScopeOrder { height, id }.into()
}
impl DirtyTasks {
pub fn queue_task(&self, task: Task) {
self.tasks_queued.borrow_mut().push(task);
}
pub fn queue_rerun(&self) {
self.rerun_queued.set(true);
}
}
impl Borrow<ScopeOrder> for DirtyScope {
impl Borrow<ScopeOrder> for DirtyTasks {
fn borrow(&self) -> &ScopeOrder {
&self.order
}
}
impl PartialOrd for DirtyScope {
impl PartialOrd for DirtyTasks {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
Some(self.order.cmp(&other.order))
}
}
impl Ord for DirtyScope {
impl Ord for DirtyTasks {
fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.order.cmp(&other.order)
}
}
impl PartialEq for DirtyScope {
impl PartialEq for DirtyTasks {
fn eq(&self, other: &Self) -> bool {
self.order == other.order
}
}
impl Hash for DirtyScope {
impl Hash for DirtyTasks {
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
self.order.hash(state);
}

View file

@ -8,7 +8,7 @@ use crate::{
any_props::AnyProps,
arena::ElementId,
innerlude::{
DirtyScope, ElementRef, ErrorBoundary, NoOpMutations, SchedulerMsg, ScopeState, VNodeMount,
DirtyScopes, ElementRef, ErrorBoundary, NoOpMutations, SchedulerMsg, ScopeState, VNodeMount,
VProps, WriteMutations,
},
nodes::RenderReturn,
@ -184,7 +184,7 @@ use std::{any::Any, collections::BTreeSet, rc::Rc};
pub struct VirtualDom {
pub(crate) scopes: Slab<ScopeState>,
pub(crate) dirty_scopes: BTreeSet<DirtyScope>,
pub(crate) dirty_scopes: DirtyScopes,
pub(crate) scopes_need_rerun: bool,
// Maps a template path to a map of byte indexes to templates
@ -380,16 +380,7 @@ impl VirtualDom {
tracing::trace!("Marking scope {:?} as dirty", id);
self.scopes_need_rerun = true;
let order = ScopeOrder::new(scope.height(), id);
match self.dirty_scopes.get(&order) {
Some(dirty) => {
dirty.queue_rerun();
}
None => {
let dirty: DirtyScope = order.into();
dirty.queue_rerun();
self.dirty_scopes.insert(dirty);
}
}
self.dirty_scopes.queue_scope(order);
}
/// Mark a task as dirty
@ -401,16 +392,7 @@ impl VirtualDom {
return;
};
let order = ScopeOrder::new(scope.height(), scope.id);
match self.dirty_scopes.get(&order) {
Some(dirty) => {
dirty.queue_task(task);
}
None => {
let dirty: DirtyScope = order.into();
dirty.queue_task(task);
self.dirty_scopes.insert(dirty);
}
}
self.dirty_scopes.queue_task(task, order);
}
/// Call a listener inside the VirtualDom with data from outside the VirtualDom. **The ElementId passed in must be the id of an element with a listener, not a static node or a text node.**
@ -509,15 +491,16 @@ impl VirtualDom {
// Next, run any queued tasks
// We choose not to poll the deadline since we complete pretty quickly anyways
while let Some(dirty) = self.dirty_scopes.pop_first() {
while let Some(task) = self.dirty_scopes.pop_task() {
// If the scope doesn't exist for whatever reason, then we should skip it
if !self.scopes.contains(dirty.order.id.0) {
if !self.scopes.contains(task.order.id.0) {
continue;
}
// Then poll any tasks that might be pending
for task in dirty.tasks_queued.borrow().iter() {
let _ = self.runtime.handle_task_wakeup(*task);
let tasks = std::mem::take(&mut *task.tasks_queued.borrow_mut());
for task in tasks {
let _ = self.runtime.handle_task_wakeup(task);
// Running that task, may mark a scope higher up as dirty. If it does, return from the function early
self.queue_events();
if self.scopes_need_rerun {
@ -536,18 +519,19 @@ impl VirtualDom {
pub fn replace_template(&mut self, template: Template) {
self.register_template_first_byte_index(template);
// iterating a slab is very inefficient, but this is a rare operation that will only happen during development so it's fine
for (_, scope) in self.scopes.iter() {
let mut dirty = Vec::new();
for (id, scope) in self.scopes.iter() {
if let Some(RenderReturn::Ready(sync)) = scope.try_root_node() {
if sync.template.get().name.rsplit_once(':').unwrap().0
== template.name.rsplit_once(':').unwrap().0
{
let context = scope.state();
let height = context.height;
self.dirty_scopes
.insert(DirtyScope::new(height, context.id));
dirty.push(ScopeId(id));
}
}
}
for dirty in dirty {
self.mark_dirty(dirty);
}
}
/// Rebuild the virtualdom without handling any of the mutations
@ -608,9 +592,9 @@ impl VirtualDom {
// Next, diff any dirty scopes
// We choose not to poll the deadline since we complete pretty quickly anyways
while let Some(dirty) = self.dirty_scopes.pop_first() {
while let Some(work) = self.dirty_scopes.pop_work() {
// If the scope doesn't exist for whatever reason, then we should skip it
if !self.scopes.contains(dirty.order.id.0) {
if !self.scopes.contains(work.scope.id.0) {
continue;
}
@ -618,14 +602,14 @@ impl VirtualDom {
let _runtime = RuntimeGuard::new(self.runtime.clone());
// Then, poll any tasks that might be pending in the scope
// This will run effects, so this **must** be done after the scope is diffed
for task in dirty.tasks_queued.borrow().iter() {
let _ = self.runtime.handle_task_wakeup(*task);
for task in work.tasks{
let _ = self.runtime.handle_task_wakeup(task);
}
// If the scope is dirty, run the scope and get the mutations
if dirty.rerun_queued.get() {
let new_nodes = self.run_scope(dirty.order.id);
if work.rerun_scope {
let new_nodes = self.run_scope(work.scope.id);
self.diff_scope(to, dirty.order.id, new_nodes);
self.diff_scope(to, work.scope.id, new_nodes);
}
}
}

View file

@ -105,9 +105,9 @@ async fn flushing() {
use_hook(|| {
spawn(async move {
for _ in 0..10 {
flush_sync().await;
BROADCAST.with(|b| b.1.resubscribe()).recv().await.unwrap();
println!("Task 1 recved");
flush_sync().await;
println!("Task 1");
SEQUENCE.with(|s| s.borrow_mut().push(1));
}
@ -117,9 +117,9 @@ async fn flushing() {
use_hook(|| {
spawn(async move {
for _ in 0..10 {
flush_sync().await;
BROADCAST.with(|b| b.1.resubscribe()).recv().await.unwrap();
println!("Task 2 recved");
flush_sync().await;
println!("Task 2");
SEQUENCE.with(|s| s.borrow_mut().push(2));
}
@ -135,15 +135,15 @@ async fn flushing() {
let fut = async {
// Trigger the flush by waiting for work
for _ in 0..30 {
for i in 0..30 {
BROADCAST.with(|b| b.0.send(()).unwrap());
dom.mark_dirty(ScopeId(0));
tokio::select! {
_ = dom.wait_for_work() => {}
_ = tokio::time::sleep(Duration::from_millis(10)) => {}
}
dom.mark_dirty(ScopeId(0));
dom.render_immediate(&mut dioxus_core::NoOpMutations);
println!("Flushed");
println!("Flushed {}", i);
}
};