add a simulator #3

Merged
programmerjake merged 58 commits from adding-simulator into master 2024-12-16 04:06:48 +00:00
Showing only changes of commit 698b8adc23 - Show all commits

View file

@ -10,16 +10,13 @@ use petgraph::{
use crate::{ use crate::{
bundle::{BundleField, BundleType}, bundle::{BundleField, BundleType},
enum_::{EnumType, EnumVariant}, enum_::{EnumType, EnumVariant},
expr::{ expr::{ops, ExprEnum, Flow},
ops,
target::{TargetBase, TargetPathArrayElement, TargetPathBundleField, TargetPathElement},
Flow,
},
intern::{Intern, Interned, Memoize}, intern::{Intern, Interned, Memoize},
module::{ module::{
AnnotatedModuleIO, Block, ExprInInstantiatedModule, ExternModuleBody, InstantiatedModule, transform::visit::{Visit, Visitor},
ModuleBody, ModuleIO, NormalModuleBody, Stmt, StmtConnect, StmtDeclaration, StmtFormal, AnnotatedModuleIO, Block, ExternModuleBody, InstantiatedModule, ModuleBody, ModuleIO,
StmtIf, StmtInstance, StmtMatch, StmtReg, StmtWire, TargetInInstantiatedModule, NormalModuleBody, Stmt, StmtConnect, StmtDeclaration, StmtFormal, StmtIf, StmtInstance,
StmtMatch, StmtReg, StmtWire, TargetInInstantiatedModule,
}, },
prelude::*, prelude::*,
}; };
@ -51,7 +48,7 @@ impl From<DeduceResetsError> for std::io::Error {
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
enum ResetTarget { enum ResetTarget {
Base { Base {
base: Interned<TargetBase>, base: Expr<CanonicalType>,
}, },
BundleField { BundleField {
parent: Interned<ResetTarget>, parent: Interned<ResetTarget>,
@ -75,7 +72,7 @@ enum ResetTarget {
impl ResetTarget { impl ResetTarget {
fn canonical_ty(self) -> CanonicalType { fn canonical_ty(self) -> CanonicalType {
match self { match self {
ResetTarget::Base { base } => base.canonical_ty(), ResetTarget::Base { base } => Expr::ty(base),
ResetTarget::BundleField { field_ty, .. } => field_ty, ResetTarget::BundleField { field_ty, .. } => field_ty,
ResetTarget::EnumVariant { variant_ty, .. } => variant_ty, ResetTarget::EnumVariant { variant_ty, .. } => variant_ty,
ResetTarget::ArraysElements { element_ty, .. } => element_ty, ResetTarget::ArraysElements { element_ty, .. } => element_ty,
@ -89,7 +86,7 @@ impl ResetTarget {
| ResetTarget::ArraysElements { parent, .. } => Some(parent), | ResetTarget::ArraysElements { parent, .. } => Some(parent),
} }
} }
fn base(mut self) -> Interned<TargetBase> { fn base(mut self) -> Expr<CanonicalType> {
loop { loop {
match self { match self {
ResetTarget::Base { base } => break base, ResetTarget::Base { base } => break base,
@ -150,14 +147,6 @@ impl ResetTarget {
} }
} }
impl<T: Into<TargetBase>> From<T> for ResetTarget {
fn from(base: T) -> Self {
ResetTarget::Base {
base: TargetBase::intern_sized(base.into()),
}
}
}
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)] #[derive(Copy, Clone, PartialEq, Eq, Hash, Debug)]
struct ResetTargetInInstantiatedModule { struct ResetTargetInInstantiatedModule {
instantiated_module: InstantiatedModule, instantiated_module: InstantiatedModule,
@ -289,6 +278,23 @@ impl<C: FromIterator<A>, A, Step: ProcessStep> FromIterator<Processed<A, Step>>
} }
} }
fn for_each_subexpr_exclusive<E>(
expr: Expr<CanonicalType>,
f: impl FnMut(Expr<CanonicalType>) -> Result<(), E>,
) -> Result<(), E> {
struct MyVisitor<F> {
f: F,
}
impl<F: FnMut(Expr<CanonicalType>) -> Result<(), E>, E> Visitor for MyVisitor<F> {
type Error = E;
fn visit_expr_enum(&mut self, v: &ExprEnum) -> Result<(), Self::Error> {
(self.f)(v.to_expr())?;
v.default_visit(self)
}
}
Expr::expr_enum(expr).default_visit(&mut MyVisitor { f })
}
struct AddNodesToGraphStep; struct AddNodesToGraphStep;
impl ProcessStep for AddNodesToGraphStep { impl ProcessStep for AddNodesToGraphStep {
@ -342,6 +348,41 @@ impl RunProcessStep<AddNodesToGraphStep> for ResetTarget {
} }
} }
impl RunProcessStep<AddEdgesToGraphStep> for ResetTarget {
fn run_process_step(
self,
_instantiated_module: InstantiatedModule,
_state: &mut State,
_step: &mut AddEdgesToGraphStep,
) -> Result<Processed<Self, AddEdgesToGraphStep>, Infallible> {
Ok(Processed(()))
}
}
impl<Step: ProcessStep> RunProcessStep<Step> for Expr<CanonicalType>
where
ResetTarget: RunProcessStep<Step>,
{
fn run_process_step(
self,
instantiated_module: InstantiatedModule,
state: &mut State,
step: &mut Step,
) -> Result<Processed<Self, Step>, Step::Error> {
let retval = ResetTarget::Base { base: self }
.run_process_step(instantiated_module, state, step)?
.map(|target| match target {
ResetTarget::Base { base } => base,
_ => unreachable!(),
});
for_each_subexpr_exclusive(self, |base| {
ResetTarget::Base { base }.run_process_step(instantiated_module, state, step)?;
Ok(())
})?;
Ok(retval)
}
}
struct ConnectAndLhsInstantiatedModule { struct ConnectAndLhsInstantiatedModule {
lhs_instantiated_module: InstantiatedModule, lhs_instantiated_module: InstantiatedModule,
lhs: Expr<CanonicalType>, lhs: Expr<CanonicalType>,
@ -349,24 +390,49 @@ struct ConnectAndLhsInstantiatedModule {
source_location: SourceLocation, source_location: SourceLocation,
} }
impl<Step: ProcessStep> RunProcessStep<Step> for ConnectAndLhsInstantiatedModule { impl RunProcessStep<AddNodesToGraphStep> for ConnectAndLhsInstantiatedModule {
fn run_process_step(
self,
_instantiated_module: InstantiatedModule,
_state: &mut State,
_step: &mut AddNodesToGraphStep,
) -> Result<Processed<Self, AddNodesToGraphStep>, Infallible> {
Ok(Processed(()))
}
}
impl RunProcessStep<AddEdgesToGraphStep> for ConnectAndLhsInstantiatedModule {
fn run_process_step( fn run_process_step(
self, self,
rhs_instantiated_module: InstantiatedModule, rhs_instantiated_module: InstantiatedModule,
state: &mut State, state: &mut State,
step: &mut Step, step: &mut AddEdgesToGraphStep,
) -> Result<Processed<Self, Step>, Step::Error> { ) -> Result<Processed<Self, AddEdgesToGraphStep>, Infallible> {
let Self { let Self {
lhs_instantiated_module, lhs_instantiated_module,
lhs, lhs,
rhs, rhs,
source_location, source_location,
} = self; } = self;
todo!(); match Expr::ty(lhs) {
CanonicalType::UInt(ty) => todo!(),
CanonicalType::SInt(ty) => todo!(),
CanonicalType::Bool(ty) => todo!(),
CanonicalType::AsyncReset(ty) => todo!(),
CanonicalType::SyncReset(ty) => todo!(),
CanonicalType::Reset(ty) => todo!(),
CanonicalType::Clock(ty) => todo!(),
CanonicalType::Array(ty) => todo!(),
CanonicalType::Enum(ty) => todo!(),
CanonicalType::Bundle(ty) => todo!(),
}
} }
} }
impl<Step: ProcessStep> RunProcessStep<Step> for StmtConnect { impl<Step: ProcessStep> RunProcessStep<Step> for StmtConnect
where
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{
fn run_process_step( fn run_process_step(
self, self,
instantiated_module: InstantiatedModule, instantiated_module: InstantiatedModule,
@ -444,16 +510,12 @@ where
state: &mut State, state: &mut State,
step: &mut Step, step: &mut Step,
) -> Result<Processed<Self, Step>, Step::Error> { ) -> Result<Processed<Self, Step>, Step::Error> {
Ok(ResetTarget::from(self) Ok(self
.to_expr()
.run_process_step(instantiated_module, state, step)? .run_process_step(instantiated_module, state, step)?
.map(|target| { .map(|expr| match *Expr::expr_enum(expr) {
let ResetTarget::Base { base } = target else { ExprEnum::ModuleIO(module_io) => module_io,
unreachable!(); _ => unreachable!(),
};
let TargetBase::ModuleIO(module_io) = *base else {
unreachable!();
};
module_io
})) }))
} }
} }
@ -469,16 +531,12 @@ where
step: &mut Step, step: &mut Step,
) -> Result<Processed<Self, Step>, Step::Error> { ) -> Result<Processed<Self, Step>, Step::Error> {
let Self { annotations, wire } = self; let Self { annotations, wire } = self;
Ok(ResetTarget::from(wire) Ok(wire
.to_expr()
.run_process_step(instantiated_module, state, step)? .run_process_step(instantiated_module, state, step)?
.map(|target| { .map(|expr| match *Expr::expr_enum(expr) {
let ResetTarget::Base { base } = target else { ExprEnum::Wire(wire) => Self { annotations, wire },
unreachable!(); _ => unreachable!(),
};
let TargetBase::Wire(wire) = *base else {
unreachable!();
};
Self { annotations, wire }
})) }))
} }
} }
@ -494,16 +552,12 @@ where
step: &mut Step, step: &mut Step,
) -> Result<Processed<Self, Step>, Step::Error> { ) -> Result<Processed<Self, Step>, Step::Error> {
let Self { annotations, reg } = self; let Self { annotations, reg } = self;
Ok(ResetTarget::from(reg) Ok(reg
.to_expr()
.run_process_step(instantiated_module, state, step)? .run_process_step(instantiated_module, state, step)?
.map(|target| { .map(|expr| match *Expr::expr_enum(expr) {
let ResetTarget::Base { base } = target else { ExprEnum::Reg(reg) => Self { annotations, reg },
unreachable!(); _ => unreachable!(),
};
let TargetBase::Reg(reg) = *base else {
unreachable!();
};
Self { annotations, reg }
})) }))
} }
} }
@ -511,6 +565,7 @@ where
impl<Step: ProcessStep> RunProcessStep<Step> for StmtInstance impl<Step: ProcessStep> RunProcessStep<Step> for StmtInstance
where where
ResetTarget: RunProcessStep<Step>, ResetTarget: RunProcessStep<Step>,
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{ {
fn run_process_step( fn run_process_step(
self, self,
@ -563,19 +618,14 @@ where
} }
.run_process_step(rhs_instantiated_module, state, step)?; .run_process_step(rhs_instantiated_module, state, step)?;
} }
Ok(ResetTarget::from(instance) Ok(Expr::canonical(instance.to_expr())
.run_process_step(instantiated_module, state, step)? .run_process_step(instantiated_module, state, step)?
.map(|target| { .map(|expr| match *Expr::expr_enum(expr) {
let ResetTarget::Base { base } = target else { ExprEnum::Instance(instance) => Self {
unreachable!();
};
let TargetBase::Instance(instance) = *base else {
unreachable!();
};
Self {
annotations, annotations,
instance, instance,
} },
_ => unreachable!(),
})) }))
} }
} }
@ -583,6 +633,7 @@ where
impl<Step: ProcessStep> RunProcessStep<Step> for StmtDeclaration impl<Step: ProcessStep> RunProcessStep<Step> for StmtDeclaration
where where
ResetTarget: RunProcessStep<Step>, ResetTarget: RunProcessStep<Step>,
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{ {
fn run_process_step( fn run_process_step(
self, self,
@ -607,6 +658,7 @@ where
impl<Step: ProcessStep> RunProcessStep<Step> for Stmt impl<Step: ProcessStep> RunProcessStep<Step> for Stmt
where where
ResetTarget: RunProcessStep<Step>, ResetTarget: RunProcessStep<Step>,
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{ {
fn run_process_step( fn run_process_step(
self, self,
@ -637,6 +689,7 @@ where
impl<Step: ProcessStep> RunProcessStep<Step> for Block impl<Step: ProcessStep> RunProcessStep<Step> for Block
where where
ResetTarget: RunProcessStep<Step>, ResetTarget: RunProcessStep<Step>,
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{ {
fn run_process_step( fn run_process_step(
self, self,
@ -649,7 +702,11 @@ where
// so always just use the old `memories` value. we add the ports to the graph anyway to make the other logic easier. // so always just use the old `memories` value. we add the ports to the graph anyway to make the other logic easier.
for memory in memories { for memory in memories {
for port in memory.ports() { for port in memory.ports() {
ResetTarget::from(port).run_process_step(instantiated_module, state, step)?; Expr::canonical(port.to_expr()).run_process_step(
instantiated_module,
state,
step,
)?;
} }
} }
let stmts = Result::<Processed<_, _>, _>::from_iter( let stmts = Result::<Processed<_, _>, _>::from_iter(
@ -664,6 +721,7 @@ where
impl<Step: ProcessStep> RunProcessStep<Step> for Module<Bundle> impl<Step: ProcessStep> RunProcessStep<Step> for Module<Bundle>
where where
ResetTarget: RunProcessStep<Step>, ResetTarget: RunProcessStep<Step>,
ConnectAndLhsInstantiatedModule: RunProcessStep<Step>,
{ {
fn run_process_step( fn run_process_step(
self, self,
@ -722,7 +780,12 @@ pub fn deduce_resets(
&mut state, &mut state,
&mut AddNodesToGraphStep, &mut AddNodesToGraphStep,
); );
todo!("add edges"); let mut step = AddEdgesToGraphStep {
union_find: UnionFind::new(state.graph.node_count()),
};
let Ok(Processed(())) =
module.run_process_step(InstantiatedModule::Base(module), &mut state, &mut step);
let AddEdgesToGraphStep { union_find } = step;
todo!("deduce types"); todo!("deduce types");
Ok(todo!("transform module")) Ok(todo!("transform module"))
} }