use alloc::{boxed::Box, vec::Vec};
use core::fmt;
use miden_crypto::{Felt, Word};
use miden_formatting::prettier::{Document, PrettyPrint, const_text, nl};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use super::{MastNodeErrorContext, MastNodeExt};
use crate::{
OPCODE_DYN, OPCODE_DYNCALL,
mast::{DecoratedOpLink, DecoratorId, MastForest, MastNodeId, Remapping},
};
#[derive(Debug, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
pub struct DynNode {
is_dyncall: bool,
#[cfg_attr(feature = "serde", serde(default, skip_serializing_if = "Vec::is_empty"))]
before_enter: Vec<DecoratorId>,
#[cfg_attr(feature = "serde", serde(default, skip_serializing_if = "Vec::is_empty"))]
after_exit: Vec<DecoratorId>,
}
impl DynNode {
pub const DYN_DOMAIN: Felt = Felt::new(OPCODE_DYN as u64);
pub const DYNCALL_DOMAIN: Felt = Felt::new(OPCODE_DYNCALL as u64);
}
impl DynNode {
pub fn new_dyn() -> Self {
Self {
is_dyncall: false,
before_enter: Vec::new(),
after_exit: Vec::new(),
}
}
pub fn new_dyncall() -> Self {
Self {
is_dyncall: true,
before_enter: Vec::new(),
after_exit: Vec::new(),
}
}
pub fn is_dyncall(&self) -> bool {
self.is_dyncall
}
pub fn domain(&self) -> Felt {
if self.is_dyncall() {
Self::DYNCALL_DOMAIN
} else {
Self::DYN_DOMAIN
}
}
}
impl MastNodeErrorContext for DynNode {
fn decorators(&self) -> impl Iterator<Item = DecoratedOpLink> {
self.before_enter.iter().chain(&self.after_exit).copied().enumerate()
}
}
impl DynNode {
pub(super) fn to_display<'a>(&'a self, mast_forest: &'a MastForest) -> impl fmt::Display + 'a {
DynNodePrettyPrint { node: self, mast_forest }
}
pub(super) fn to_pretty_print<'a>(
&'a self,
mast_forest: &'a MastForest,
) -> impl PrettyPrint + 'a {
DynNodePrettyPrint { node: self, mast_forest }
}
}
struct DynNodePrettyPrint<'a> {
node: &'a DynNode,
mast_forest: &'a MastForest,
}
impl DynNodePrettyPrint<'_> {
fn concatenate_decorators(
&self,
decorator_ids: &[DecoratorId],
prepend: Document,
append: Document,
) -> Document {
let decorators = decorator_ids
.iter()
.map(|&decorator_id| self.mast_forest[decorator_id].render())
.reduce(|acc, doc| acc + const_text(" ") + doc)
.unwrap_or_default();
if decorators.is_empty() {
decorators
} else {
prepend + decorators + append
}
}
fn single_line_pre_decorators(&self) -> Document {
self.concatenate_decorators(self.node.before_enter(), Document::Empty, const_text(" "))
}
fn single_line_post_decorators(&self) -> Document {
self.concatenate_decorators(self.node.after_exit(), const_text(" "), Document::Empty)
}
fn multi_line_pre_decorators(&self) -> Document {
self.concatenate_decorators(self.node.before_enter(), Document::Empty, nl())
}
fn multi_line_post_decorators(&self) -> Document {
self.concatenate_decorators(self.node.after_exit(), nl(), Document::Empty)
}
}
impl crate::prettier::PrettyPrint for DynNodePrettyPrint<'_> {
fn render(&self) -> crate::prettier::Document {
let dyn_text = if self.node.is_dyncall() {
const_text("dyncall")
} else {
const_text("dyn")
};
let single_line = self.single_line_pre_decorators()
+ dyn_text.clone()
+ self.single_line_post_decorators();
let multi_line =
self.multi_line_pre_decorators() + dyn_text + self.multi_line_post_decorators();
single_line | multi_line
}
}
impl fmt::Display for DynNodePrettyPrint<'_> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.pretty_print(f)
}
}
impl MastNodeExt for DynNode {
fn digest(&self) -> Word {
if self.is_dyncall {
Word::new([
Felt::new(8751004906421739448),
Felt::new(13469709002495534233),
Felt::new(12584249374630430826),
Felt::new(7624899870831503004),
])
} else {
Word::new([
Felt::new(8115106948140260551),
Felt::new(13491227816952616836),
Felt::new(15015806788322198710),
Felt::new(16575543461540527115),
])
}
}
fn before_enter(&self) -> &[DecoratorId] {
&self.before_enter
}
fn after_exit(&self) -> &[DecoratorId] {
&self.after_exit
}
fn append_before_enter(&mut self, decorator_ids: &[DecoratorId]) {
self.before_enter.extend_from_slice(decorator_ids);
}
fn append_after_exit(&mut self, decorator_ids: &[DecoratorId]) {
self.after_exit.extend_from_slice(decorator_ids);
}
fn remove_decorators(&mut self) {
self.before_enter.truncate(0);
self.after_exit.truncate(0);
}
fn to_display<'a>(&'a self, mast_forest: &'a MastForest) -> Box<dyn fmt::Display + 'a> {
Box::new(DynNode::to_display(self, mast_forest))
}
fn to_pretty_print<'a>(&'a self, mast_forest: &'a MastForest) -> Box<dyn PrettyPrint + 'a> {
Box::new(DynNode::to_pretty_print(self, mast_forest))
}
fn remap_children(&self, _remapping: &Remapping) -> Self {
self.clone()
}
fn has_children(&self) -> bool {
false
}
fn append_children_to(&self, _target: &mut Vec<MastNodeId>) {
}
fn for_each_child<F>(&self, _f: F)
where
F: FnMut(MastNodeId),
{
}
fn domain(&self) -> Felt {
self.domain()
}
}
#[cfg(test)]
mod tests {
use miden_crypto::hash::rpo::Rpo256;
use super::*;
#[test]
pub fn test_dyn_node_digest() {
assert_eq!(
DynNode::new_dyn().digest(),
Rpo256::merge_in_domain(&[Word::default(), Word::default()], DynNode::DYN_DOMAIN)
);
assert_eq!(
DynNode::new_dyncall().digest(),
Rpo256::merge_in_domain(&[Word::default(), Word::default()], DynNode::DYNCALL_DOMAIN)
);
}
}