ParsedExpression.cpp 13 KB
Newer Older
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
/* -------------------------------------------------------------------------- *
 *                                   Lepton                                   *
 * -------------------------------------------------------------------------- *
 * This is part of the Lepton expression parser originating from              *
 * Simbios, the NIH National Center for Physics-Based Simulation of           *
 * Biological Structures at Stanford, funded under the NIH Roadmap for        *
 * Medical Research, grant U54 GM072970. See https://simtk.org.               *
 *                                                                            *
 * Portions copyright (c) 2009 Stanford University and the Authors.           *
 * Authors: Peter Eastman                                                     *
 * Contributors:                                                              *
 *                                                                            *
 * Permission is hereby granted, free of charge, to any person obtaining a    *
 * copy of this software and associated documentation files (the "Software"), *
 * to deal in the Software without restriction, including without limitation  *
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,   *
 * and/or sell copies of the Software, and to permit persons to whom the      *
 * Software is furnished to do so, subject to the following conditions:       *
 *                                                                            *
 * The above copyright notice and this permission notice shall be included in *
 * all copies or substantial portions of the Software.                        *
 *                                                                            *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR *
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,   *
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL    *
 * THE AUTHORS, CONTRIBUTORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,    *
 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR      *
 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE  *
 * USE OR OTHER DEALINGS IN THE SOFTWARE.                                     *
 * -------------------------------------------------------------------------- */

32
33
34
#include "lepton/ParsedExpression.h"
#include "lepton/ExpressionProgram.h"
#include "lepton/Operation.h"
35
#include <limits>
36
37
38
39
40
#include <vector>

using namespace Lepton;
using namespace std;

Peter Eastman's avatar
Peter Eastman committed
41
ParsedExpression::ParsedExpression(const ExpressionTreeNode& rootNode) : rootNode(rootNode) {
42
43
44
45
46
47
48
}

const ExpressionTreeNode& ParsedExpression::getRootNode() const {
    return rootNode;
}

double ParsedExpression::evaluate() const {
49
    return evaluate(getRootNode(), map<string, double>());
50
51
52
}

double ParsedExpression::evaluate(const std::map<std::string, double>& variables) const {
53
    return evaluate(getRootNode(), variables);
54
55
}

56
double ParsedExpression::evaluate(const ExpressionTreeNode& node, const map<string, double>& variables) {
57
58
59
60
61
    vector<double> args(node.getChildren().size());
    for (int i = 0; i < args.size(); i++)
        args[i] = evaluate(node.getChildren()[i], variables);
    return node.getOperation().evaluate(&args[0], variables);
}
62
63

ParsedExpression ParsedExpression::optimize() const {
64
65
    ExpressionTreeNode result = precalculateConstantSubexpressions(getRootNode());
    result = substituteSimplerExpression(result);
66
67
68
69
    return result;
}

ParsedExpression ParsedExpression::optimize(const map<string, double>& variables) const {
70
71
72
73
    ExpressionTreeNode result = preevaluateVariables(getRootNode(), variables);
    result = precalculateConstantSubexpressions(result);
    result = substituteSimplerExpression(result);
    return ParsedExpression(result);
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
107
}

ExpressionTreeNode ParsedExpression::preevaluateVariables(const ExpressionTreeNode& node, const map<string, double>& variables) {
    if (node.getOperation().getId() == Operation::VARIABLE) {
        const Operation::Variable& var = dynamic_cast<const Operation::Variable&>(node.getOperation());
        map<string, double>::const_iterator iter = variables.find(var.getName());
        if (iter == variables.end())
            return node;
        return ExpressionTreeNode(new Operation::Constant(iter->second));
    }
    vector<ExpressionTreeNode> children(node.getChildren().size());
    for (int i = 0; i < children.size(); i++)
        children[i] = preevaluateVariables(node.getChildren()[i], variables);
    return ExpressionTreeNode(node.getOperation().clone(), children);
}

ExpressionTreeNode ParsedExpression::precalculateConstantSubexpressions(const ExpressionTreeNode& node) {
    vector<ExpressionTreeNode> children(node.getChildren().size());
    for (int i = 0; i < children.size(); i++)
        children[i] = precalculateConstantSubexpressions(node.getChildren()[i]);
    ExpressionTreeNode result = ExpressionTreeNode(node.getOperation().clone(), children);
    if (node.getOperation().getId() == Operation::VARIABLE)
        return result;
    for (int i = 0; i < children.size(); i++)
        if (children[i].getOperation().getId() != Operation::CONSTANT)
            return result;
    return ExpressionTreeNode(new Operation::Constant(evaluate(result, map<string, double>())));
}

ExpressionTreeNode ParsedExpression::substituteSimplerExpression(const ExpressionTreeNode& node) {
    vector<ExpressionTreeNode> children(node.getChildren().size());
    for (int i = 0; i < children.size(); i++)
        children[i] = substituteSimplerExpression(node.getChildren()[i]);
    switch (node.getOperation().getId()) {
108
        case Operation::ADD:
Peter Eastman's avatar
Peter Eastman committed
109
        {
110
111
            double first = getConstantValue(children[0]);
            double second = getConstantValue(children[1]);
Peter Eastman's avatar
Peter Eastman committed
112
            if (first == 0.0) // Add 0
113
                return children[1];
Peter Eastman's avatar
Peter Eastman committed
114
            if (first == 1.0) // Add 1
115
                return ExpressionTreeNode(new Operation::Increment(), children[1]);
Peter Eastman's avatar
Peter Eastman committed
116
            if (second == 0.0) // Add 0
117
                return children[0];
Peter Eastman's avatar
Peter Eastman committed
118
            if (second == 1.0) // Add 1
119
120
                return ExpressionTreeNode(new Operation::Increment(), children[0]);
            break;
Peter Eastman's avatar
Peter Eastman committed
121
        }
122
        case Operation::SUBTRACT:
Peter Eastman's avatar
Peter Eastman committed
123
124
        {
            double first = getConstantValue(children[0]);
Peter Eastman's avatar
Peter Eastman committed
125
            if (first == 0.0) // Subtract from 0
126
                return ExpressionTreeNode(new Operation::Negate(), children[1]);
Peter Eastman's avatar
Peter Eastman committed
127
            double second = getConstantValue(children[1]);
Peter Eastman's avatar
Peter Eastman committed
128
            if (second == 0.0) // Subtract 0
129
                return children[0];
Peter Eastman's avatar
Peter Eastman committed
130
            if (second == 1.0) // Subtract 1
131
132
                return ExpressionTreeNode(new Operation::Decrement(), children[0]);
            break;
Peter Eastman's avatar
Peter Eastman committed
133
        }
134
        case Operation::MULTIPLY:
Peter Eastman's avatar
Peter Eastman committed
135
136
137
        {
            double first = getConstantValue(children[0]);
            double second = getConstantValue(children[1]);
Peter Eastman's avatar
Peter Eastman committed
138
            if (first == 0.0 || second == 0.0) // Multiply by 0
139
                return ExpressionTreeNode(new Operation::Constant(0.0));
Peter Eastman's avatar
Peter Eastman committed
140
            if (first == 1.0) // Multiply by 1
141
                return children[1];
Peter Eastman's avatar
Peter Eastman committed
142
            if (second == 1.0) // Multiply by 1
143
                return children[0];
Peter Eastman's avatar
Peter Eastman committed
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
            if (children[0].getOperation().getId() == Operation::CONSTANT) {
                if (children[1].getOperation().getId() == Operation::MULTIPLY) {
                    if (children[1].getChildren()[0].getOperation().getId() == Operation::CONSTANT) // Combine two multiplies into a single one
                        return ExpressionTreeNode(new Operation::Multiply(), children[1].getChildren()[1], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[1].getChildren()[0])*first)));
                    if (children[1].getChildren()[1].getOperation().getId() == Operation::CONSTANT) // Combine two multiplies into a single one
                        return ExpressionTreeNode(new Operation::Multiply(), children[1].getChildren()[0], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[1].getChildren()[1])*first)));
                }
            }
            if (children[1].getOperation().getId() == Operation::CONSTANT) {
                if (children[0].getOperation().getId() == Operation::MULTIPLY) {
                    if (children[0].getChildren()[0].getOperation().getId() == Operation::CONSTANT) // Combine two multiplies into a single one
                        return ExpressionTreeNode(new Operation::Multiply(), children[0].getChildren()[1], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[0].getChildren()[0])*second)));
                    if (children[0].getChildren()[1].getOperation().getId() == Operation::CONSTANT) // Combine two multiplies into a single one
                        return ExpressionTreeNode(new Operation::Multiply(), children[0].getChildren()[0], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[0].getChildren()[1])*second)));
                }
            }
160
            break;
Peter Eastman's avatar
Peter Eastman committed
161
        }
162
        case Operation::DIVIDE:
Peter Eastman's avatar
Peter Eastman committed
163
        {
164
            double numerator = getConstantValue(children[0]);
Peter Eastman's avatar
Peter Eastman committed
165
            if (numerator = 0.0) // 0 divided by something
166
                return ExpressionTreeNode(new Operation::Constant(0.0));
Peter Eastman's avatar
Peter Eastman committed
167
            if (numerator == 1.0) // 1 divided by something
168
169
                return ExpressionTreeNode(new Operation::Reciprocal(), children[1]);
            double denominator = getConstantValue(children[1]);
Peter Eastman's avatar
Peter Eastman committed
170
            if (denominator == 1.0) // Divide by 1
171
                return children[0];
Peter Eastman's avatar
Peter Eastman committed
172
173
174
175
176
177
178
179
180
            if (children[1].getOperation().getId() == Operation::CONSTANT) {
                if (children[0].getOperation().getId() == Operation::MULTIPLY) {
                    if (children[0].getChildren()[0].getOperation().getId() == Operation::CONSTANT) // Combine a multiply and a divide into one multiply
                        return ExpressionTreeNode(new Operation::Multiply(), children[0].getChildren()[1], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[0].getChildren()[0])/denominator)));
                    if (children[0].getChildren()[1].getOperation().getId() == Operation::CONSTANT) // Combine a multiply and a divide into one multiply
                        return ExpressionTreeNode(new Operation::Multiply(), children[0].getChildren()[0], ExpressionTreeNode(new Operation::Constant(getConstantValue(children[0].getChildren()[1])/denominator)));
                }
                return ExpressionTreeNode(new Operation::Multiply(), children[0], ExpressionTreeNode(new Operation::Constant(1.0/denominator))); // Replace a divide with a multiply
            }
181
            break;
Peter Eastman's avatar
Peter Eastman committed
182
        }
183
        case Operation::POWER:
Peter Eastman's avatar
Peter Eastman committed
184
        {
185
            double base = getConstantValue(children[0]);
Peter Eastman's avatar
Peter Eastman committed
186
            if (base == 0.0) // 0 to any power is 0
187
                return ExpressionTreeNode(new Operation::Constant(0.0));
Peter Eastman's avatar
Peter Eastman committed
188
            if (base == 1.0) // 1 to any power is 1
189
190
                return ExpressionTreeNode(new Operation::Constant(1.0));
            double exponent = getConstantValue(children[1]);
Peter Eastman's avatar
Peter Eastman committed
191
            if (exponent == 1.0) // x^1 = x
192
                return children[0];
Peter Eastman's avatar
Peter Eastman committed
193
            if (exponent == -1.0) // x^-1 = recip(x)
194
                return ExpressionTreeNode(new Operation::Reciprocal(), children[0]);
Peter Eastman's avatar
Peter Eastman committed
195
            if (exponent == 2.0) // x^2 = square(x)
196
                return ExpressionTreeNode(new Operation::Square(), children[0]);
Peter Eastman's avatar
Peter Eastman committed
197
            if (exponent == 3.0) // x^3 = cube(x)
198
                return ExpressionTreeNode(new Operation::Cube(), children[0]);
Peter Eastman's avatar
Peter Eastman committed
199
            if (exponent == 0.5) // x^0.5 = sqrt(x)
200
                return ExpressionTreeNode(new Operation::Sqrt(), children[0]);
201
            break;
Peter Eastman's avatar
Peter Eastman committed
202
        }
203
204
205
    }
    return ExpressionTreeNode(node.getOperation().clone(), children);
}
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223

ParsedExpression ParsedExpression::differentiate(const std::string& variable) const {
    return differentiate(getRootNode(), variable);
}

ExpressionTreeNode ParsedExpression::differentiate(const ExpressionTreeNode& node, const std::string& variable) {
    vector<ExpressionTreeNode> childDerivs(node.getChildren().size());
    for (int i = 0; i < childDerivs.size(); i++)
        childDerivs[i] = differentiate(node.getChildren()[i], variable);
    return node.getOperation().differentiate(node.getChildren(),childDerivs, variable);
}

double ParsedExpression::getConstantValue(const ExpressionTreeNode& node) {
    if (node.getOperation().getId() == Operation::CONSTANT)
        return dynamic_cast<const Operation::Constant&>(node.getOperation()).getValue();
    return numeric_limits<double>::quiet_NaN();
}

Peter Eastman's avatar
Peter Eastman committed
224
225
226
227
ExpressionProgram ParsedExpression::createProgram() const {
    return ExpressionProgram(*this);
}

228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
ostream& Lepton::operator<<(ostream& out, const ExpressionTreeNode& node) {
    out << node.getOperation().getName();
    if (node.getChildren().size() > 0) {
        out << "(";
        for (int i = 0; i < node.getChildren().size(); i++) {
            if (i > 0)
                out << ", ";
            out << node.getChildren()[i];
        }
        out << ")";
    }
    return out;
}

ostream& Lepton::operator<<(ostream& out, const ParsedExpression& exp) {
    out << exp.getRootNode();
    return out;
}