1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
use std::fmt;

#[derive(Debug, Clone)]
pub enum BinaryOpcode {
    Or,
    Xor,
    And,
    Shl,
    Shr,
    Add,
    Sub,
    Mul,
    Div,
    Mod,
}

#[derive(Debug, Clone)]
pub enum UnaryOpcode {
    Inv,
    Neg,
}

#[derive(Debug)]
pub enum Type {
    Pattern,
    Sequence,
    Range,
    Sample,
    Unique,
    WeightedSample,
    Expand,
    Done,
    Once,
}

#[derive(Debug)]
pub enum VariableMethod {
    Next,
    Prev,
    Copy,
}

#[derive(Debug)]
pub enum Node {
    Number(u32),
    UnaryOperation(UnaryOpcode, Box<Node>),
    BinaryOperation(Box<Node>, BinaryOpcode, Box<Node>),
    Variable(String, Box<Node>),
    Enum(String, Vec<Box<Node>>),
    EnumItem(String, Option<Box<Node>>),
    EnumInst(String),
    EnumItemInst(String, String),
    Type(Type, Vec<Box<Node>>),
    WeightedPair(u32, Box<Node>),
    VariableInst(String, VariableMethod),
}

/// An abstraction above Node to implement `import`
#[derive(Debug)]
pub enum Item {
    /// A single item in the current file
    ///
    /// E.g. Node::Assignment
    Single(Box<Node>),

    /// The expansion of a `import` statement
    ///
    /// Contains all items from the `import`d file.
    Multiple(Vec<Item>),

    /// Encapsulates errors on `import`
    ///
    /// We can't use normal Rust error handling techniques due to abstraction by rust-peg.
    /// Instead, embed an Item::ImportErrors on a import error.
    ImportError(::std::path::PathBuf, ::std::io::Error),
}

impl fmt::Display for BinaryOpcode {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let operator = match *self {
            BinaryOpcode::Or => "|",
            BinaryOpcode::Xor => "^",
            BinaryOpcode::And => "&",
            BinaryOpcode::Shl => "<<",
            BinaryOpcode::Shr => ">>",
            BinaryOpcode::Add => "+",
            BinaryOpcode::Sub => "-",
            BinaryOpcode::Mul => "*",
            BinaryOpcode::Div => "/",
            BinaryOpcode::Mod => "%",
        };

        write!(f, "{}", operator)
    }
}

impl fmt::Display for UnaryOpcode {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let operator = match *self {
            UnaryOpcode::Inv => "~",
            UnaryOpcode::Neg => "-",
        };

        write!(f, "{}", operator)
    }
}