/*
- * Copyright (c) 2016 ARM Limited
+ * Copyright (c) 2016-2017 ARM Limited
* All rights reserved
*
* The license below extends only to copyright in the software and shall
#include "sim/sim_object.hh"
MathExprPowerModel::MathExprPowerModel(const Params *p)
- : PowerModelState(p), dyn_expr(p->dyn), st_expr(p->st)
+ : PowerModelState(p), dyn_expr(p->dyn), st_expr(p->st), failed(false)
{
// Calculate the name of the object we belong to
std::vector<std::string> path;
for (auto & i: Stats::statsList())
if (i->name.find(basename) == 0)
stats_map[i->name.substr(basename.size())] = i;
+
+ tryEval(st_expr);
+ const bool st_failed = failed;
+
+ tryEval(dyn_expr);
+ const bool dyn_failed = failed;
+
+ if (st_failed || dyn_failed) {
+ const auto *p = dynamic_cast<const Params *>(params());
+ assert(p);
+
+ fatal("Failed to evaluate power expressions:\n%s%s%s\n",
+ st_failed ? p->st : "",
+ st_failed && dyn_failed ? "\n" : "",
+ dyn_failed ? p->dyn : "");
+ }
}
+double
+MathExprPowerModel::eval(const MathExpr &expr) const
+{
+ const double value = tryEval(expr);
+
+ // This shouldn't happen unless something went wrong the equations
+ // were verified in startup().
+ panic_if(failed, "Failed to evaluate power expression '%s'\n",
+ expr.toStr());
+
+ return value;
+}
+
+double
+MathExprPowerModel::tryEval(const MathExpr &expr) const
+{
+ failed = false;
+ const double value = expr.eval(
+ std::bind(&MathExprPowerModel::getStatValue,
+ this, std::placeholders::_1)
+ );
+
+ return value;
+}
+
+
double
MathExprPowerModel::getStatValue(const std::string &name) const
{
return _temp;
// Try to cast the stat, only these are supported right now
- Info *info = stats_map.at(name);
+ const auto it = stats_map.find(name);
+ if (it == stats_map.cend()) {
+ warn("Failed to find stat '%s'\n", name);
+ failed = true;
+ return 0;
+ }
+
+ const Info *info = it->second;
- ScalarInfo *si = dynamic_cast<ScalarInfo*>(info);
+ auto si = dynamic_cast<const ScalarInfo *>(info);
if (si)
return si->value();
- FormulaInfo *fi = dynamic_cast<FormulaInfo*>(info);
+ auto fi = dynamic_cast<const FormulaInfo *>(info);
if (fi)
return fi->total();
/*
- * Copyright (c) 2016 ARM Limited
+ * Copyright (c) 2016-2017 ARM Limited
* All rights reserved
*
* The license below extends only to copyright in the software and shall
*
* @return Power (Watts) consumed by this object (dynamic component)
*/
- double getDynamicPower() const {
- return dyn_expr.eval(
- std::bind(&MathExprPowerModel::getStatValue,
- this, std::placeholders::_1)
- );
- }
+ double getDynamicPower() const { return eval(dyn_expr); }
/**
* Get the static power consumption.
*
* @return Power (Watts) consumed by this object (static component)
*/
- double getStaticPower() const {
- return st_expr.eval(
- std::bind(&MathExprPowerModel::getStatValue,
- this, std::placeholders::_1)
- );
- }
+ double getStaticPower() const { return eval(st_expr); }
/**
* Get the value for a variable (maps to a stat)
void regStats();
private:
+ /**
+ * Evaluate an expression in the context of this object, fatal if
+ * evaluation fails.
+ *
+ * @param expr Expression to evaluate
+ * @return Value of expression.
+ */
+ double eval(const MathExpr &expr) const;
+
+ /**
+ * Evaluate an expression in the context of this object, set
+ * failed if evaluation failed.
+ *
+ * @param expr Expression to evaluate
+ * @return Value of expression.
+ */
+ double tryEval(const MathExpr &expr) const;
// Math expressions for dynamic and static power
MathExpr dyn_expr, st_expr;
// Map that contains relevant stats for this power model
std::unordered_map<std::string, Stats::Info*> stats_map;
+
+ // Did the expression fail to evaluate (e.g., because a stat value
+ // can't be resolved)
+ mutable bool failed;
};
#endif