aboutsummaryrefslogtreecommitdiffstats
path: root/src/modelica_ast.rs
diff options
context:
space:
mode:
authorbnewbold <bnewbold@robocracy.org>2016-12-17 18:34:47 -0800
committerbnewbold <bnewbold@robocracy.org>2016-12-17 18:34:47 -0800
commit9f82aceb9fbdb42f332d68f4a423123bd0788b2c (patch)
treec082b9795be8e9e9d286c8f8f1345d22f3ec1b59 /src/modelica_ast.rs
parentf6364ebcac0d0a88a3cc6812fd2120c97b42cc26 (diff)
downloadmodelthing-9f82aceb9fbdb42f332d68f4a423123bd0788b2c.tar.gz
modelthing-9f82aceb9fbdb42f332d68f4a423123bd0788b2c.zip
refactor modelica parser into separate crate
Diffstat (limited to 'src/modelica_ast.rs')
-rw-r--r--src/modelica_ast.rs407
1 files changed, 0 insertions, 407 deletions
diff --git a/src/modelica_ast.rs b/src/modelica_ast.rs
deleted file mode 100644
index f6d32b8..0000000
--- a/src/modelica_ast.rs
+++ /dev/null
@@ -1,407 +0,0 @@
-
-use std::clone::Clone;
-use std::fmt::{Debug, Formatter, Error};
-use std::collections::HashMap;
-use std::collections::HashSet;
-
-#[derive(Clone, PartialEq)]
-pub struct ModelicaModel {
- pub name: String,
- pub components: Vec<Component>,
- pub equations: Vec<SimpleEquation>,
- pub connections: Vec<Connection>,
- pub extends: Vec<String>,
-}
-
-#[derive(Copy, Clone, PartialEq)]
-pub enum ComponentPrefix {
- // incomplete: eg, can be parameter and input
- Flow,
- Stream,
- Input,
- Output,
- Discrete,
- Parameter,
- Constant,
-}
-
-#[derive(Clone, PartialEq)]
-pub struct Component {
- pub prefix: Option<ComponentPrefix>,
- pub specifier: String,
- pub name: String,
- pub value: Option<Expr>,
- pub units: Option<String>,
- pub description: Option<String>,
-}
-
-#[derive(Clone, PartialEq)]
-pub struct Connection {
- pub a: String,
- pub b: String,
-}
-
-#[derive(Clone, PartialEq)]
-pub struct SimpleEquation {
- pub lhs: Expr,
- pub rhs: Expr,
-}
-
-#[derive(Clone, PartialEq)]
-pub enum Expr {
- Integer(i64),
- Float(f64),
- Ident(String),
- Der(Box<Expr>),
- Abs(Box<Expr>),
- BinExpr(BinOperator, Box<Expr>, Box<Expr>),
-}
-
-#[derive(Copy, Clone, PartialEq)]
-pub enum BinOperator {
- Multiply,
- Divide,
- Add,
- Subtract,
-}
-
-//// Helpers
-
-impl ModelicaModel {
-
- pub fn get_constant_vars(&self) -> HashMap<String,Option<Expr>> {
- let mut binds = HashMap::new();
- // XXX: actually implement this...
- for c in &self.components {
- match c.prefix {
- Some(ComponentPrefix::Constant) => { binds.insert(c.name.clone(), Some(Expr::Integer(123))); },
- Some(ComponentPrefix::Parameter) => { binds.insert(c.name.clone(), Some(Expr::Float(4.56))); },
- _ => (),
- }
- }
- binds
- }
-
- // This crude function finds "unbound" variables: those which are not constants, parameters, or
- // the sole element on the LHS of an equation.
- // Bugs:
- // if a var is on LHS and RHS of same equation
- pub fn get_free_vars(&self) -> Vec<String> {
- // Start with components, and remove constants and parameters
- let vars = self.components.iter().filter(|v| match v.prefix {
- Some(ComponentPrefix::Constant) | Some(ComponentPrefix::Parameter) => false,
- _ => true,
- });
-
- // Remove LHS (bound) vars
- let mut outputs = vec![];
- for eq in self.equations.iter() {
- // TODO:
- if let Expr::Ident(ref symb) = eq.lhs {
- outputs.push(symb.to_string());
- }
- }
- let vars = vars.filter(|v| !outputs.contains(&v.name));
-
- vars.map(|c| c.name.clone()).collect()
- }
-
- // V variables
- // Q constants (become kwargs)
- // P bound vars (independent, inputs/passed, become like constants)
- // M unknowns (dependent, outputs)
- // N' total equations
- // N equations with unknowns
- //
- // TODO: allow passing in Q
- pub fn solve_for(&self, indep_vars: Vec<String>, dep_vars: Vec<String>) -> Result<ModelicaModel,String> {
- let constants = self.get_constant_vars();
- let mut all_vars: HashSet<String> = HashSet::new();
- for eqn in &self.equations {
- all_vars.extend(eqn.identifiers());
- }
-
- // check that all dep and indep are in equations
- let mut passed_vars = indep_vars.clone();
- passed_vars.extend(dep_vars.clone());
- for var in passed_vars {
- if !all_vars.contains(&var) {
- return Err(format!("Variable not found in equations: {}", var));
- }
- }
-
- // check that V = Q + P + M
- if all_vars.len() != (constants.len() + indep_vars.len() + dep_vars.len()) {
- return Err(format!("Variable counts don't add up (V={} Q={} P={} M={})",
- all_vars.len(),
- constants.len(),
- indep_vars.len(),
- dep_vars.len()));
- }
-
- // check that all constants are bound and simple
- for (name, value) in &constants {
- match *value {
- None => return Err(format!("UnderSpecifiedConstant: {}", name)),
- Some(Expr::Integer(_)) | Some(Expr::Float(_)) => (), // Ok,
- Some(_) => return Err(format!("NaiveImplementation: can't handle constant: {}", name)),
- }
- }
-
- // check that there is a depdendent variable in each equation
- for eqn in &self.equations {
- if intersect_strings(&dep_vars, &eqn.identifiers()).len() == 0 {
- return Err("NaiveImplementation/OverConstrained: at least one equation is missing a dependent variable".to_string());
- }
- }
-
- // check N >= M
- if self.equations.len() < dep_vars.len() {
- return Err("UnderConstrained: more dependent variables than equations".to_string());
- }
-
- println!("Soliving for {:?} in terms of params {:?} and constants {:?}, with {} equations",
- dep_vars, indep_vars, constants, self.equations.len());
-
- let mut unsolved_eqns = self.equations.clone();
- let mut solved: Vec<SimpleEquation> = vec![];
- let mut unsolved_vars = dep_vars.clone();
- while unsolved_eqns.len() > 0 {
- let next_i = unsolved_eqns
- .iter()
- .position(|ref x| intersect_strings(&unsolved_vars, &x.identifiers()).len() == 1);
- let eqn = match next_i {
- None => { return Err("NaiveImplementation (or poor equation selection?)".to_string()); },
- Some(i) => unsolved_eqns.remove(i),
- };
- let ref var = intersect_strings(&unsolved_vars, &eqn.identifiers())[0];
- let eqn = eqn.rebalance_for(var.to_string()).expect("rebalance for success");
- solved.push(eqn);
- let var_i = unsolved_vars.iter().position(|ref x| x == &var).unwrap();
- unsolved_vars.remove(var_i);
- };
-
- // TODO: sort output equations by LHS
- Ok(ModelicaModel{
- name: self.name.clone(),
- components: self.components.clone(),
- connections: vec![],
- equations: solved,
- extends: vec![],
- })
- }
-}
-
-fn union_strings(a: &Vec<String>, b: &Vec<String>) -> Vec<String> {
- let mut u = a.clone();
- for e in b {
- if !(u.contains(&e)) {
- u.push(e.clone());
- }
- }
- u
-}
-
-fn intersect_strings(a: &Vec<String>, b: &Vec<String>) -> Vec<String> {
- let mut both = vec![];
- for e in a {
- if b.contains(&e) {
- both.push(e.clone());
- }
- }
- both
-}
-
-impl Expr {
-
- // Order is undefined
- // TODO: should return a HashSet, not a Vec
- pub fn identifiers(&self) -> Vec<String> {
- use self::Expr::*;
- match *self {
- Integer(_) | Float(_) => vec![],
- Ident(ref s) => vec![s.clone()],
- Der(ref e) | Abs(ref e) => e.identifiers(),
- BinExpr(_, ref e1, ref e2) => {
- union_strings(&e1.identifiers(), &e2.identifiers())
- },
- }
- }
-
- pub fn contains(&self, ident: &str) -> bool{
- self.identifiers().contains(&ident.to_string())
- }
-}
-
-impl SimpleEquation {
-
- // Order is undefined
- pub fn identifiers(&self) -> Vec<String> {
- union_strings(&self.lhs.identifiers(), &self.rhs.identifiers())
- }
-
- pub fn contains(&self, ident: &str) -> bool{
- let s = &ident.to_string();
- self.lhs.identifiers().contains(s) || self.rhs.identifiers().contains(s)
- }
-
- pub fn rebalance_for(&self, ident: String) -> Result<SimpleEquation,String> {
- let lvars = self.lhs.identifiers();
- let rvars = self.rhs.identifiers();
-
- let ret = match (lvars.contains(&ident), rvars.contains(&ident)) {
- (true, true) => Err("SymbolicError: NaiveImplementation".to_string()),
- (false, false) => Err("SymbolicError: VariableNotFound".to_string()),
- (true, false) => self.simplify_lhs(&ident),
- (false, true) =>
- SimpleEquation{lhs: self.rhs.clone(),
- rhs: self.lhs.clone()}.simplify_lhs(&ident),
- };
- match ret {
- Ok(eqn) => {
- if eqn.rhs.contains(&ident) {
- Err("SymbolicError: NaiveImplementation".to_string())
- } else {
- Ok(eqn)
- }},
- Err(_) => ret,
- }
- }
-
- pub fn simplify_lhs(&self, ident: &str) -> Result<SimpleEquation,String> {
- use self::Expr::*;
- use self::BinOperator::*;
- match self.lhs {
- Ident(ref s) if s == ident => Ok((*self).clone()),
- Ident(_) | Integer(_) | Float(_) =>
- Err("SymbolicError: InternalError: expected var on LHS".to_string()),
- Der(_) | Abs(_) =>
- Err("SymbolicError: NaiveImplementation: can't simplify der() or abs()".to_string()),
- // TODO: create a macro for the below...
- BinExpr(Multiply, ref a, ref b) if a.contains(ident) => {
- SimpleEquation{
- lhs: *a.clone(),
- rhs: BinExpr(Divide, Box::new(self.rhs.clone()), b.clone())}.simplify_lhs(&ident) },
- BinExpr(Multiply, ref a, ref b) if b.contains(ident) => {
- SimpleEquation{
- lhs: *b.clone(),
- rhs: BinExpr(Divide, Box::new(self.rhs.clone()), a.clone())}.simplify_lhs(&ident) },
- BinExpr(Divide, ref a, ref b) if a.contains(ident) => {
- SimpleEquation{
- lhs: *a.clone(),
- rhs: BinExpr(Multiply, Box::new(self.rhs.clone()), b.clone())}.simplify_lhs(&ident) },
- BinExpr(Divide, ref a, ref b) if b.contains(ident) => {
- SimpleEquation{
- lhs: *b.clone(),
- rhs: BinExpr(Divide, a.clone(), Box::new(self.rhs.clone()))}.simplify_lhs(&ident) },
- BinExpr(Add, ref a, ref b) if a.contains(ident) => {
- SimpleEquation{
- lhs: *a.clone(),
- rhs: BinExpr(Subtract, Box::new(self.rhs.clone()), b.clone())}.simplify_lhs(&ident) },
- BinExpr(Add, ref a, ref b) if b.contains(ident) => {
- SimpleEquation{
- lhs: *b.clone(),
- rhs: BinExpr(Subtract, Box::new(self.rhs.clone()), a.clone())}.simplify_lhs(&ident) },
- BinExpr(Subtract, ref a, ref b) if a.contains(ident) => {
- SimpleEquation{
- lhs: *a.clone(),
- rhs: BinExpr(Add, Box::new(self.rhs.clone()), b.clone())}.simplify_lhs(&ident) },
- BinExpr(Subtract, ref a, ref b) if b.contains(ident) => {
- SimpleEquation{
- lhs: *b.clone(),
- rhs: BinExpr(Subtract, a.clone(), Box::new(self.rhs.clone()))}.simplify_lhs(&ident) },
- BinExpr(_, _, _) => Err("SymbolicError: NotImplemented BinOperator (or else couldn't find var...)".to_string()),
- // in case we add opers: _ => Err("NotImplemented".to_string()),
- }
- }
-}
-
-
-//// Debug Implementations
-
-impl Debug for ModelicaModel {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- try!(write!(fmt, "model {}\n", self.name));
- for e in self.extends.iter() {
- try!(write!(fmt, " extends {};\n", e));
- }
- for v in self.components.iter() {
- try!(write!(fmt, " {:?};\n", v));
- }
- try!(write!(fmt, "equation\n"));
- for c in self.connections.iter() {
- try!(write!(fmt, " {:?};\n", c));
- }
- for e in self.equations.iter() {
- try!(write!(fmt, " {:?};\n", e));
- }
- write!(fmt, "end {};\n", self.name)
- }
-}
-
-impl Debug for ComponentPrefix {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- use self::ComponentPrefix::*;
- write!(fmt, "{}",
- match *self {
- Flow => "flow",
- Stream => "stream",
- Input => "input",
- Output => "output",
- Discrete => "discrete",
- Parameter => "parameter",
- Constant => "constant",
- })
- }
-}
-
-impl Debug for Component {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- write!(fmt, "{}{} {}",
- match self.prefix {
- Some(p) => format!("{:?} ", p),
- None => "".to_string(),
- },
- self.specifier,
- self.name,
- )
- }
-}
-
-impl Debug for Connection {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- write!(fmt, "connect({}, {})", self.a, self.b)
- }
-}
-
-impl Debug for SimpleEquation {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- write!(fmt, "{:?} = {:?}", self.lhs, self.rhs)
- }
-}
-
-impl Debug for Expr {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- use self::Expr::*;
- match *self {
- Integer(e) => write!(fmt, "{}", e),
- Float(e) => write!(fmt, "{}", e),
- Ident(ref e) => write!(fmt, "{}", e),
- Der(ref e) => write!(fmt, "der({:?})", e),
- Abs(ref e) => write!(fmt, "abs({:?})", e),
- BinExpr(op, ref l, ref r) => write!(fmt, "({:?} {:?} {:?})", l, op, r),
- }
- }
-}
-
-impl Debug for BinOperator {
- fn fmt(&self, fmt: &mut Formatter) -> Result<(), Error> {
- use self::BinOperator::*;
- match *self {
- Multiply => write!(fmt, "*"),
- Divide => write!(fmt, "/"),
- Add => write!(fmt, "+"),
- Subtract => write!(fmt, "-"),
- }
- }
-}