aboutsummaryrefslogtreecommitdiff
path: root/src/libexpr/eval.cc
diff options
context:
space:
mode:
authorEelco Dolstra <eelco.dolstra@logicblox.com>2012-08-12 23:29:28 -0400
committerEelco Dolstra <eelco.dolstra@logicblox.com>2012-08-12 23:29:28 -0400
commite82767910c649f160d6701e47f606f3b8dde4b29 (patch)
treea951e469b37adfe3db1485a0bc8996ac4cc51416 /src/libexpr/eval.cc
parent325d1cfebf6c8ad391dc318f984feb3e5731aa5a (diff)
Add some basic profiling support to the evaluator
Setting the environment variable NIX_COUNT_CALLS to 1 enables some basic profiling in the evaluator. It will count calls to functions and primops as well as evaluations of attributes. For example, to see where evaluation of a NixOS configuration spends its time: $ NIX_SHOW_STATS=1 NIX_COUNT_CALLS=1 ./src/nix-instantiate/nix-instantiate '<nixos>' -A system --readonly-mode ... calls to 39 primops: 239532 head 233962 tail 191252 hasAttr ... calls to 1595 functions: 224157 `/nix/var/nix/profiles/per-user/root/channels/nixos/nixpkgs/pkgs/lib/lists.nix:17:19' 221767 `/nix/var/nix/profiles/per-user/root/channels/nixos/nixpkgs/pkgs/lib/lists.nix:17:14' 221767 `/nix/var/nix/profiles/per-user/root/channels/nixos/nixpkgs/pkgs/lib/lists.nix:17:10' ... evaluations of 7088 attributes: 167377 undefined position 132459 `/nix/var/nix/profiles/per-user/root/channels/nixos/nixpkgs/pkgs/lib/attrsets.nix:119:41' 47322 `/nix/var/nix/profiles/per-user/root/channels/nixos/nixpkgs/pkgs/lib/attrsets.nix:13:21' ...
Diffstat (limited to 'src/libexpr/eval.cc')
-rw-r--r--src/libexpr/eval.cc38
1 files changed, 37 insertions, 1 deletions
diff --git a/src/libexpr/eval.cc b/src/libexpr/eval.cc
index cf7c62ad2..2d41eba0e 100644
--- a/src/libexpr/eval.cc
+++ b/src/libexpr/eval.cc
@@ -144,6 +144,7 @@ EvalState::EvalState()
{
nrEnvs = nrValuesInEnvs = nrValues = nrListElems = 0;
nrAttrsets = nrOpUpdates = nrOpUpdateValuesCopied = 0;
+ countCalls = getEnv("NIX_COUNT_CALLS", "0") != "0";
#if HAVE_BOEHMGC
static bool gcInitialised = true;
@@ -300,8 +301,10 @@ inline Value * EvalState::lookupVar(Env * env, const VarRef & var)
if (var.fromWith) {
while (1) {
Bindings::iterator j = env->values[0]->attrs->find(var.name);
- if (j != env->values[0]->attrs->end())
+ if (j != env->values[0]->attrs->end()) {
+ if (countCalls && j->pos) attrSelects[*j->pos]++;
return j->value;
+ }
if (env->prevWith == 0)
throwEvalError("undefined variable `%1%'", var.name);
for (unsigned int l = env->prevWith; l; --l, env = env->up) ;
@@ -619,8 +622,10 @@ void ExprSelect::eval(EvalState & state, Env & env, Value & v)
}
vAttrs = j->value;
pos = j->pos;
+ if (state.countCalls && pos) state.attrSelects[*pos]++;
}
+
state.forceValue(*vAttrs);
} catch (Error & e) {
@@ -700,6 +705,7 @@ void EvalState::callFunction(Value & fun, Value & arg, Value & v)
vArgs[n--] = arg->primOpApp.right;
/* And call the primop. */
+ if (countCalls) primOpCalls[primOp->primOp->name]++;
try {
primOp->primOp->fun(*this, vArgs, v);
} catch (Error & e) {
@@ -760,6 +766,8 @@ void EvalState::callFunction(Value & fun, Value & arg, Value & v)
throwTypeError("function at %1% called with unexpected argument", fun.lambda.fun->pos);
}
+ if (countCalls) functionCalls[fun.lambda.fun->pos]++;
+
try {
fun.lambda.fun->body->eval(*this, env2, v);
} catch (Error & e) {
@@ -1216,6 +1224,34 @@ void EvalState::printStats()
printMsg(v, format(" number of thunks: %1%") % nrThunks);
printMsg(v, format(" number of thunks avoided: %1%") % nrAvoided);
printMsg(v, format(" number of attr lookups: %1%") % nrLookups);
+
+ if (countCalls) {
+
+ printMsg(v, format("calls to %1% primops:") % primOpCalls.size());
+ typedef std::multimap<unsigned int, Symbol> PrimOpCalls_;
+ std::multimap<unsigned int, Symbol> primOpCalls_;
+ foreach (PrimOpCalls::iterator, i, primOpCalls)
+ primOpCalls_.insert(std::pair<unsigned int, Symbol>(i->second, i->first));
+ foreach_reverse (PrimOpCalls_::reverse_iterator, i, primOpCalls_)
+ printMsg(v, format("%1$10d %2%") % i->first % i->second);
+
+ printMsg(v, format("calls to %1% functions:") % functionCalls.size());
+ typedef std::multimap<unsigned int, Pos> FunctionCalls_;
+ std::multimap<unsigned int, Pos> functionCalls_;
+ foreach (FunctionCalls::iterator, i, functionCalls)
+ functionCalls_.insert(std::pair<unsigned int, Pos>(i->second, i->first));
+ foreach_reverse (FunctionCalls_::reverse_iterator, i, functionCalls_)
+ printMsg(v, format("%1$10d %2%") % i->first % i->second);
+
+ printMsg(v, format("evaluations of %1% attributes:") % attrSelects.size());
+ typedef std::multimap<unsigned int, Pos> AttrSelects_;
+ std::multimap<unsigned int, Pos> attrSelects_;
+ foreach (AttrSelects::iterator, i, attrSelects)
+ attrSelects_.insert(std::pair<unsigned int, Pos>(i->second, i->first));
+ foreach_reverse (AttrSelects_::reverse_iterator, i, attrSelects_)
+ printMsg(v, format("%1$10d %2%") % i->first % i->second);
+
+ }
}