2020-04-28 18:25:24 +00:00
|
|
|
use crate::System;
|
|
|
|
use legion::prelude::Schedule;
|
2020-05-01 08:50:07 +00:00
|
|
|
use std::{
|
|
|
|
borrow::Cow,
|
|
|
|
cmp::Ordering,
|
|
|
|
collections::{HashMap, HashSet},
|
|
|
|
};
|
2020-04-16 02:40:24 +00:00
|
|
|
|
2020-04-06 08:57:00 +00:00
|
|
|
#[derive(Default)]
|
|
|
|
pub struct SchedulePlan {
|
|
|
|
stages: HashMap<String, Vec<System>>,
|
|
|
|
stage_order: Vec<String>,
|
2020-05-01 07:55:32 +00:00
|
|
|
system_names: HashSet<Cow<'static, str>>,
|
2020-04-06 08:57:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SchedulePlan {
|
2020-04-16 19:13:05 +00:00
|
|
|
pub fn build(&mut self) -> Schedule {
|
2020-04-06 08:57:00 +00:00
|
|
|
let mut schedule_builder = Schedule::builder();
|
|
|
|
|
|
|
|
for stage in self.stage_order.drain(..) {
|
|
|
|
if let Some((_, mut systems)) = self.stages.remove_entry(&stage) {
|
|
|
|
let system_count = systems.len();
|
|
|
|
for system in systems.drain(..) {
|
|
|
|
match system {
|
|
|
|
System::Schedulable(schedulable) => {
|
|
|
|
schedule_builder = schedule_builder.add_system(schedulable);
|
|
|
|
}
|
|
|
|
System::ThreadLocal(runnable) => {
|
|
|
|
schedule_builder = schedule_builder.add_thread_local(runnable);
|
|
|
|
}
|
2020-05-01 07:55:32 +00:00
|
|
|
System::ThreadLocalFn((_name, thread_local)) => {
|
2020-04-06 08:57:00 +00:00
|
|
|
schedule_builder = schedule_builder.add_thread_local_fn(thread_local);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if system_count > 0 {
|
|
|
|
schedule_builder = schedule_builder.flush();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
schedule_builder.build()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage(&mut self, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
} else {
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.push(stage.to_string());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage_after(&mut self, target: &str, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
let target_index = self
|
|
|
|
.stage_order
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_i, stage)| stage.as_str() == target)
|
|
|
|
.map(|(i, _)| i)
|
|
|
|
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
|
|
|
|
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.insert(target_index + 1, stage.to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_stage_before(&mut self, target: &str, stage: &str) {
|
|
|
|
if let Some(_) = self.stages.get(stage) {
|
|
|
|
panic!("Stage already exists: {}", stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
let target_index = self
|
|
|
|
.stage_order
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|(_i, stage)| stage.as_str() == target)
|
|
|
|
.map(|(i, _)| i)
|
|
|
|
.unwrap_or_else(|| panic!("Target stage does not exist: {}", target));
|
|
|
|
|
|
|
|
self.stages.insert(stage.to_string(), Vec::new());
|
|
|
|
self.stage_order.insert(target_index, stage.to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn add_system_to_stage(
|
|
|
|
&mut self,
|
|
|
|
stage_name: &str,
|
2020-04-16 02:40:24 +00:00
|
|
|
system: impl Into<System>,
|
2020-04-06 08:57:00 +00:00
|
|
|
) -> &mut Self {
|
|
|
|
let systems = self
|
|
|
|
.stages
|
|
|
|
.get_mut(stage_name)
|
|
|
|
.unwrap_or_else(|| panic!("Stage does not exist: {}", stage_name));
|
2020-05-01 07:55:32 +00:00
|
|
|
let system = system.into();
|
|
|
|
let system_name = system.name();
|
|
|
|
if self.system_names.contains(&system_name) {
|
|
|
|
panic!("System with name {} already exists", system_name);
|
|
|
|
}
|
|
|
|
self.system_names.insert(system_name);
|
2020-04-16 02:40:24 +00:00
|
|
|
systems.push(system.into());
|
2020-04-06 08:57:00 +00:00
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// working around the famous "rust float ordering" problem
|
|
|
|
#[derive(PartialOrd)]
|
|
|
|
struct FloatOrd(f32);
|
|
|
|
|
|
|
|
impl Ord for FloatOrd {
|
|
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
|
|
self.0.partial_cmp(&other.0).unwrap_or_else(|| {
|
|
|
|
if self.0.is_nan() && !other.0.is_nan() {
|
|
|
|
Ordering::Less
|
|
|
|
} else if !self.0.is_nan() && other.0.is_nan() {
|
|
|
|
Ordering::Greater
|
|
|
|
} else {
|
|
|
|
Ordering::Equal
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for FloatOrd {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
if self.0.is_nan() && other.0.is_nan() {
|
|
|
|
true
|
|
|
|
} else {
|
|
|
|
self.0 == other.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for FloatOrd {}
|