#[macro_use]
use crate::*;
use nom::{
multi::separated_list0,
sequence::tuple as nom_tuple,
};
pub fn guard_operator(input: ParseString) -> ParseResult<()> {
let (input, _) = whitespace0(input)?;
let (input, _) = alt((tag("|"),tag("│"),tag("├"),tag("└")))(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, ()))
}
pub fn fsm_implementation(input: ParseString) -> ParseResult<FsmImplementation> {
let (input, _) = hashtag(input)?;
let (input, name) = identifier(input)?;
let (input, _) = left_parenthesis(input)?;
let (input, input_vars) = separated_list0(list_separator, var)(input)?;
let (input, _) = right_parenthesis(input)?;
let (input, _) = transition_operator(input)?;
let (input, start) = pattern(input)?;
let (input, _) = whitespace0(input)?;
let (input, arms) = many1(fsm_arm)(input)?;
let (input, _) = period(input)?;
Ok((input, FsmImplementation{name,input: input_vars,start,arms}))
}
pub fn fsm_arm(input: ParseString) -> ParseResult<FsmArm> {
let (input, arm) = alt((fsm_guard_arm,fsm_transition,fsm_comment_arm))(input)?;
let (input, _) = whitespace0(input)?;
Ok((input, arm))
}
pub fn fsm_guard_arm(input: ParseString) -> ParseResult<FsmArm> {
let (input, start) = pattern(input)?;
let (input, grds) = many1(fsm_guard)(input)?;
Ok((input, FsmArm::Guard(start, grds)))
}
pub fn fsm_comment_arm(input: ParseString) -> ParseResult<FsmArm> {
let (input, comment) = comment(input)?;
Ok((input, FsmArm::Comment(comment)))
}
pub fn fsm_guard(input: ParseString) -> ParseResult<Guard> {
let (input, _) = guard_operator(input)?;
let (input, cnd) = pattern(input)?;
let (input, trns) = many1(alt((
fsm_statement_transition,
fsm_state_transition,
fsm_output,
fsm_async_transition,
fsm_block_transition)))(input)?;
Ok((input, Guard{condition: cnd, transitions: trns}))
}
pub fn fsm_transition(input: ParseString) -> ParseResult<FsmArm> {
let (input, start) = pattern(input)?;
let (input, trns) = many1(alt((
fsm_state_transition,
fsm_output,
fsm_async_transition,
fsm_statement_transition,
fsm_block_transition)))(input)?;
Ok((input, FsmArm::Transition(start, trns)))
}
pub fn fsm_state_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = transition_operator(input)?;
let (input, ptrn) = pattern(input)?;
Ok((input, Transition::Next(ptrn)))
}
pub fn fsm_async_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = async_transition_operator(input)?;
let (input, ptrn) = pattern(input)?;
Ok((input, Transition::Async(ptrn)))
}
pub fn fsm_statement_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = transition_operator(input)?;
let (input, stmnt) = statement(input)?;
Ok((input, Transition::Statement(stmnt)))
}
pub fn fsm_block_transition(input: ParseString) -> ParseResult<Transition> {
let (input, _) = transition_operator(input)?;
let (input, _) = left_brace(input)?;
let (input, code) = mech_code(input)?;
let (input, _) = right_brace(input)?;
Ok((input, Transition::CodeBlock(code)))
}
pub fn fsm_output(input: ParseString) -> ParseResult<Transition> {
let (input, _) = output_operator(input)?;
let ((input, ptrn)) = pattern(input)?;
Ok((input, Transition::Output(ptrn)))
}
pub fn fsm_specification(input: ParseString) -> ParseResult<FsmSpecification> {
let (input, _) = hashtag(input)?;
let (input, name) = identifier(input)?;
let (input, _) = left_parenthesis(input)?;
let (input, input_vars) = separated_list0(list_separator, var)(input)?;
let (input, _) = right_parenthesis(input)?;
let (input, _) = opt(output_operator)(input)?;
let (input, output) = opt(kind_annotation)(input)?;
let (input, _) = opt(define_operator)(input)?;
let (input, states) = many1(fsm_state_definition)(input)?;
let (input, _) = period(input)?;
Ok((input, FsmSpecification{name,input: input_vars, output, states}))
}
pub fn fsm_state_definition(input: ParseString) -> ParseResult<StateDefinition> {
let (input, _) = guard_operator(input)?;
let (input, _) = whitespace0(input)?;
let (input, state_atom) = atom(input)?;
let (input, vars) = opt(fsm_state_definition_variables)(input)?;
Ok((input, StateDefinition{name: state_atom.name,state_variables: vars}))
}
pub fn fsm_state_definition_variables(input: ParseString) -> ParseResult<Vec<Var>> {
let (input, _) = left_parenthesis(input)?;
let (input, names) = separated_list1(list_separator, var)(input)?;
let (input, _) = right_parenthesis(input)?;
Ok((input, names))
}
pub fn fsm_pipe(input: ParseString) -> ParseResult<FsmPipe> {
let (input, start) = fsm_instance(input)?;
let (input, trns) = many0(alt((fsm_state_transition,fsm_async_transition,fsm_output)))(input)?;
Ok((input, FsmPipe{start, transitions: trns}))
}
pub fn fsm_declare(input: ParseString) -> ParseResult<FsmDeclare> {
let (input, fsm) = fsm(input)?;
let (input, _) = define_operator(input)?;
let (input, pipe) = fsm_pipe(input)?;
Ok((input, FsmDeclare{fsm,pipe}))
}
pub fn fsm(input: ParseString) -> ParseResult<Fsm> {
let ((input, _)) = hashtag(input)?;
let ((input, name)) = identifier(input)?;
let ((input, args)) = opt(argument_list)(input)?;
let ((input, kind)) = opt(kind_annotation)(input)?;
Ok((input, Fsm{ name, args, kind }))
}
pub fn fsm_instance(input: ParseString) -> ParseResult<FsmInstance> {
let ((input, _)) = hashtag(input)?;
let (input, name) = identifier(input)?;
let (input, args) = opt(fsm_args)(input)?;
Ok((input, FsmInstance{name,args} ))
}
pub fn fsm_args(input: ParseString) -> ParseResult<Vec<(Option<Identifier>,Expression)>> {
let (input, _) = left_parenthesis(input)?;
let (input, args) = separated_list0(list_separator, alt((call_arg_with_binding,call_arg)))(input)?;
let (input, _) = right_parenthesis(input)?;
Ok((input, args))
}