use crate::{
exec::{Executable, InterpreterState},
gc::{empty_trace, Finalize, Trace},
syntax::ast::node::Node,
BoaProfiler, Context, Result, Value,
};
use std::{fmt, ops::Deref, rc::Rc};
#[cfg(feature = "deser")]
use serde::{Deserialize, Serialize};
#[cfg(feature = "vm")]
use crate::vm::{compilation::CodeGen, Compiler};
#[cfg_attr(feature = "deser", derive(Serialize, Deserialize))]
#[derive(Clone, Debug, Trace, Finalize, PartialEq)]
pub struct StatementList {
#[cfg_attr(feature = "deser", serde(flatten))]
items: Box<[Node]>,
}
impl StatementList {
pub fn items(&self) -> &[Node] {
&self.items
}
pub(in crate::syntax::ast::node) fn display(
&self,
f: &mut fmt::Formatter<'_>,
indentation: usize,
) -> fmt::Result {
let indent = " ".repeat(indentation);
for node in self.items.iter() {
f.write_str(&indent)?;
node.display(f, indentation + 1)?;
match node {
Node::Block(_) | Node::If(_) | Node::Switch(_) | Node::WhileLoop(_) => {}
_ => write!(f, ";")?,
}
writeln!(f)?;
}
Ok(())
}
}
impl Executable for StatementList {
fn run(&self, context: &mut Context) -> Result<Value> {
let _timer = BoaProfiler::global().start_event("StatementList", "exec");
let mut obj = Value::default();
context
.executor()
.set_current_state(InterpreterState::Executing);
for (i, item) in self.items().iter().enumerate() {
let val = item.run(context)?;
match context.executor().get_current_state() {
InterpreterState::Return => {
obj = val;
break;
}
InterpreterState::Break(_label) => {
break;
}
InterpreterState::Continue(_label) => {
break;
}
InterpreterState::Executing => {
}
}
if i + 1 == self.items().len() {
obj = val;
}
}
Ok(obj)
}
}
#[cfg(feature = "vm")]
impl CodeGen for StatementList {
fn compile(&self, compiler: &mut Compiler) {
let _timer = BoaProfiler::global().start_event("StatementList - Code Gen", "codeGen");
for item in self.items().iter() {
item.compile(compiler);
}
}
}
impl<T> From<T> for StatementList
where
T: Into<Box<[Node]>>,
{
fn from(stm: T) -> Self {
Self { items: stm.into() }
}
}
impl fmt::Display for StatementList {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.display(f, 0)
}
}
#[derive(Clone, Debug, Finalize, PartialEq)]
pub struct RcStatementList(Rc<StatementList>);
impl Deref for RcStatementList {
type Target = StatementList;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl From<StatementList> for RcStatementList {
#[inline]
fn from(statementlist: StatementList) -> Self {
Self(Rc::from(statementlist))
}
}
unsafe impl Trace for RcStatementList {
empty_trace!();
}