charon_lib/ast/
llbc_ast_utils.rsuse crate::llbc_ast::*;
use crate::meta;
use crate::meta::Span;
use derive_generic_visitor::*;
pub fn combine_switch_targets_span(targets: &Switch) -> Span {
match targets {
Switch::If(_, st1, st2) => meta::combine_span(&st1.span, &st2.span),
Switch::SwitchInt(_, _, branches, otherwise) => {
let branches = branches.iter().map(|b| &b.1.span);
let mbranches = meta::combine_span_iter(branches);
meta::combine_span(&mbranches, &otherwise.span)
}
Switch::Match(_, branches, otherwise) => {
let branches = branches.iter().map(|b| &b.1.span);
let mbranches = meta::combine_span_iter(branches);
if let Some(otherwise) = otherwise {
meta::combine_span(&mbranches, &otherwise.span)
} else {
mbranches
}
}
}
}
impl Switch {
pub fn iter_targets(&self) -> impl Iterator<Item = &Block> {
use itertools::Either;
match self {
Switch::If(_, exp1, exp2) => Either::Left([exp1, exp2].into_iter()),
Switch::SwitchInt(_, _, targets, otherwise) => Either::Right(Either::Left(
targets.iter().map(|(_, tgt)| tgt).chain([otherwise]),
)),
Switch::Match(_, targets, otherwise) => Either::Right(Either::Right(
targets.iter().map(|(_, tgt)| tgt).chain(otherwise.as_ref()),
)),
}
}
pub fn iter_targets_mut(&mut self) -> impl Iterator<Item = &mut Block> {
use itertools::Either;
match self {
Switch::If(_, exp1, exp2) => Either::Left([exp1, exp2].into_iter()),
Switch::SwitchInt(_, _, targets, otherwise) => Either::Right(Either::Left(
targets.iter_mut().map(|(_, tgt)| tgt).chain([otherwise]),
)),
Switch::Match(_, targets, otherwise) => Either::Right(Either::Right(
targets
.iter_mut()
.map(|(_, tgt)| tgt)
.chain(otherwise.as_mut()),
)),
}
}
}
impl Statement {
pub fn new(span: Span, content: RawStatement) -> Self {
Statement {
span,
content,
comments_before: vec![],
}
}
pub fn into_box(self) -> Box<Self> {
Box::new(self)
}
pub fn into_block(self) -> Block {
Block {
span: self.span,
statements: vec![self],
}
}
}
impl Block {
pub fn from_seq(seq: Vec<Statement>) -> Option<Self> {
if seq.is_empty() {
None
} else {
let span = seq
.iter()
.map(|st| st.span)
.reduce(|a, b| meta::combine_span(&a, &b))
.unwrap();
Some(Block {
span,
statements: seq,
})
}
}
pub fn merge(mut self, mut other: Self) -> Self {
self.span = meta::combine_span(&self.span, &other.span);
self.statements.append(&mut other.statements);
self
}
pub fn then(mut self, r: Statement) -> Self {
self.span = meta::combine_span(&self.span, &r.span);
self.statements.push(r);
self
}
pub fn then_opt(self, other: Option<Statement>) -> Self {
if let Some(other) = other {
self.then(other)
} else {
self
}
}
pub fn visit_statements<F: FnMut(&mut Statement)>(&mut self, f: F) {
BlockVisitor::new(|_| {}, f).visit(self);
}
pub fn transform<F: FnMut(&mut Statement) -> Vec<Statement>>(&mut self, mut f: F) {
self.transform_sequences(|slice| f(&mut slice[0]));
}
pub fn transform_sequences<F: FnMut(&mut [Statement]) -> Vec<Statement>>(&mut self, mut f: F) {
self.visit_blocks_bwd(|blk: &mut Block| {
for i in (0..blk.statements.len()).rev() {
let prefix_to_insert = f(&mut blk.statements[i..]);
if !prefix_to_insert.is_empty() {
blk.statements.splice(i..i, prefix_to_insert);
}
}
})
}
pub fn visit_blocks_bwd<F: FnMut(&mut Block)>(&mut self, f: F) {
BlockVisitor::new(f, |_| {}).visit(self);
}
}
#[derive(Visitor)]
pub struct BlockVisitor<F: FnMut(&mut Block), G: FnMut(&mut Statement)> {
exit_blk: F,
enter_stmt: G,
}
impl<F: FnMut(&mut Block), G: FnMut(&mut Statement)> BlockVisitor<F, G> {
pub fn new(exit_blk: F, enter_stmt: G) -> Self {
Self {
exit_blk,
enter_stmt,
}
}
}
impl<F: FnMut(&mut Block), G: FnMut(&mut Statement)> VisitBodyMut for BlockVisitor<F, G> {
fn exit_llbc_block(&mut self, x: &mut Block) {
(self.exit_blk)(x)
}
fn enter_llbc_statement(&mut self, x: &mut Statement) {
(self.enter_stmt)(x)
}
}