forked from lix-project/lix
* After parsing, compute level/displacement pairs for each variable
use site, allowing environments to be stores as vectors of values rather than maps. This should speed up evaluation and reduce the number of allocations.
This commit is contained in:
parent
816dd3f061
commit
9985230c00
7 changed files with 258 additions and 143 deletions
|
@ -55,7 +55,8 @@ void run(Strings args)
|
||||||
doTest(state, "123");
|
doTest(state, "123");
|
||||||
doTest(state, "{ x = 1; y = 2; }");
|
doTest(state, "{ x = 1; y = 2; }");
|
||||||
doTest(state, "{ x = 1; y = 2; }.y");
|
doTest(state, "{ x = 1; y = 2; }.y");
|
||||||
doTest(state, "rec { x = 1; y = x; }.y");
|
doTest(state, "let x = 1; y = 2; z = 3; in let a = 4; in y");
|
||||||
|
doTest(state, "rec { x = 1; y = x; }.x");
|
||||||
doTest(state, "(x: x) 1");
|
doTest(state, "(x: x) 1");
|
||||||
doTest(state, "(x: y: y) 1 2");
|
doTest(state, "(x: y: y) 1 2");
|
||||||
doTest(state, "x: x");
|
doTest(state, "x: x");
|
||||||
|
|
|
@ -98,7 +98,7 @@ EvalState::EvalState()
|
||||||
, sType(symbols.create("type"))
|
, sType(symbols.create("type"))
|
||||||
, sMeta(symbols.create("meta"))
|
, sMeta(symbols.create("meta"))
|
||||||
, sName(symbols.create("name"))
|
, sName(symbols.create("name"))
|
||||||
, baseEnv(allocEnv())
|
, baseEnv(allocEnv(128))
|
||||||
{
|
{
|
||||||
nrValues = nrEnvs = nrEvaluated = recursionDepth = maxRecursionDepth = 0;
|
nrValues = nrEnvs = nrEvaluated = recursionDepth = maxRecursionDepth = 0;
|
||||||
deepestStack = (char *) -1;
|
deepestStack = (char *) -1;
|
||||||
|
@ -117,16 +117,19 @@ EvalState::~EvalState()
|
||||||
|
|
||||||
void EvalState::addConstant(const string & name, Value & v)
|
void EvalState::addConstant(const string & name, Value & v)
|
||||||
{
|
{
|
||||||
|
#if 0
|
||||||
baseEnv.bindings[symbols.create(name)] = v;
|
baseEnv.bindings[symbols.create(name)] = v;
|
||||||
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
||||||
(*baseEnv.bindings[symbols.create("builtins")].attrs)[symbols.create(name2)] = v;
|
(*baseEnv.bindings[symbols.create("builtins")].attrs)[symbols.create(name2)] = v;
|
||||||
nrValues += 2;
|
nrValues += 2;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void EvalState::addPrimOp(const string & name,
|
void EvalState::addPrimOp(const string & name,
|
||||||
unsigned int arity, PrimOp primOp)
|
unsigned int arity, PrimOp primOp)
|
||||||
{
|
{
|
||||||
|
#if 0
|
||||||
Value v;
|
Value v;
|
||||||
v.type = tPrimOp;
|
v.type = tPrimOp;
|
||||||
v.primOp.arity = arity;
|
v.primOp.arity = arity;
|
||||||
|
@ -135,6 +138,7 @@ void EvalState::addPrimOp(const string & name,
|
||||||
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
||||||
(*baseEnv.bindings[symbols.create("builtins")].attrs)[symbols.create(name2)] = v;
|
(*baseEnv.bindings[symbols.create("builtins")].attrs)[symbols.create(name2)] = v;
|
||||||
nrValues += 2;
|
nrValues += 2;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -234,8 +238,8 @@ void mkPath(Value & v, const char * s)
|
||||||
|
|
||||||
Value * EvalState::lookupVar(Env * env, const Symbol & name)
|
Value * EvalState::lookupVar(Env * env, const Symbol & name)
|
||||||
{
|
{
|
||||||
|
#if 0
|
||||||
/* First look for a regular variable binding for `name'. */
|
/* First look for a regular variable binding for `name'. */
|
||||||
for (Env * env2 = env; env2; env2 = env2->up) {
|
|
||||||
Bindings::iterator i = env2->bindings.find(name);
|
Bindings::iterator i = env2->bindings.find(name);
|
||||||
if (i != env2->bindings.end()) return &i->second;
|
if (i != env2->bindings.end()) return &i->second;
|
||||||
}
|
}
|
||||||
|
@ -250,7 +254,8 @@ Value * EvalState::lookupVar(Env * env, const Symbol & name)
|
||||||
if (j != i->second.attrs->end()) return &j->second;
|
if (j != i->second.attrs->end()) return &j->second;
|
||||||
}
|
}
|
||||||
|
|
||||||
throwEvalError("undefined variable `%1%'", name);
|
throwEvalError("urgh! undefined variable `%1%'", name);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -261,10 +266,11 @@ Value * EvalState::allocValues(unsigned int count)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Env & EvalState::allocEnv()
|
Env & EvalState::allocEnv(unsigned int size)
|
||||||
{
|
{
|
||||||
nrEnvs++;
|
nrEnvs++;
|
||||||
return *(new Env);
|
Env * env = (Env *) malloc(sizeof(Env) + size * sizeof(Value));
|
||||||
|
return *env;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -343,7 +349,7 @@ void EvalState::eval(Env & env, Expr * e, Value & v)
|
||||||
char x;
|
char x;
|
||||||
if (&x < deepestStack) deepestStack = &x;
|
if (&x < deepestStack) deepestStack = &x;
|
||||||
|
|
||||||
//debug(format("eval: %1%") % *e);
|
debug(format("eval: %1%") % *e);
|
||||||
|
|
||||||
checkInterrupt();
|
checkInterrupt();
|
||||||
|
|
||||||
|
@ -396,28 +402,33 @@ void ExprPath::eval(EvalState & state, Env & env, Value & v)
|
||||||
void ExprAttrs::eval(EvalState & state, Env & env, Value & v)
|
void ExprAttrs::eval(EvalState & state, Env & env, Value & v)
|
||||||
{
|
{
|
||||||
if (recursive) {
|
if (recursive) {
|
||||||
|
|
||||||
/* Create a new environment that contains the attributes in
|
/* Create a new environment that contains the attributes in
|
||||||
this `rec'. */
|
this `rec'. */
|
||||||
Env & env2(state.allocEnv());
|
Env & env2(state.allocEnv(attrs.size() + inherited.size()));
|
||||||
env2.up = &env;
|
env2.up = &env;
|
||||||
|
|
||||||
v.type = tAttrs;
|
v.type = tAttrs;
|
||||||
v.attrs = &env2.bindings;
|
v.attrs = new Bindings;
|
||||||
|
|
||||||
|
unsigned int displ = 0;
|
||||||
|
|
||||||
/* The recursive attributes are evaluated in the new
|
/* The recursive attributes are evaluated in the new
|
||||||
environment. */
|
environment. */
|
||||||
foreach (Attrs::iterator, i, attrs) {
|
foreach (Attrs::iterator, i, attrs) {
|
||||||
Value & v2 = env2.bindings[i->first];
|
Value & v2 = (*v.attrs)[i->first];
|
||||||
mkThunk(v2, env2, i->second);
|
mkCopy(v2, env2.values[displ]);
|
||||||
|
mkThunk(env2.values[displ++], env2, i->second);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
/* The inherited attributes, on the other hand, are
|
/* The inherited attributes, on the other hand, are
|
||||||
evaluated in the original environment. */
|
evaluated in the original environment. */
|
||||||
foreach (list<Symbol>::iterator, i, inherited) {
|
foreach (list<Symbol>::iterator, i, inherited) {
|
||||||
Value & v2 = env2.bindings[*i];
|
Value & v2 = env2.bindings[*i];
|
||||||
mkCopy(v2, *state.lookupVar(&env, *i));
|
mkCopy(v2, *state.lookupVar(&env, *i));
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
else {
|
||||||
|
@ -439,22 +450,24 @@ void ExprLet::eval(EvalState & state, Env & env, Value & v)
|
||||||
{
|
{
|
||||||
/* Create a new environment that contains the attributes in this
|
/* Create a new environment that contains the attributes in this
|
||||||
`let'. */
|
`let'. */
|
||||||
Env & env2(state.allocEnv());
|
Env & env2(state.allocEnv(attrs->attrs.size() + attrs->inherited.size()));
|
||||||
env2.up = &env;
|
env2.up = &env;
|
||||||
|
|
||||||
|
unsigned int displ = 0;
|
||||||
|
|
||||||
/* The recursive attributes are evaluated in the new
|
/* The recursive attributes are evaluated in the new
|
||||||
environment. */
|
environment. */
|
||||||
foreach (ExprAttrs::Attrs::iterator, i, attrs->attrs) {
|
foreach (ExprAttrs::Attrs::iterator, i, attrs->attrs)
|
||||||
Value & v2 = env2.bindings[i->first];
|
mkThunk(env2.values[displ++], env2, i->second);
|
||||||
mkThunk(v2, env2, i->second);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
#if 0
|
||||||
/* The inherited attributes, on the other hand, are evaluated in
|
/* The inherited attributes, on the other hand, are evaluated in
|
||||||
the original environment. */
|
the original environment. */
|
||||||
foreach (list<Symbol>::iterator, i, attrs->inherited) {
|
foreach (list<Symbol>::iterator, i, attrs->inherited) {
|
||||||
Value & v2 = env2.bindings[*i];
|
Value & v2 = env2.bindings[*i];
|
||||||
mkCopy(v2, *state.lookupVar(&env, *i));
|
mkCopy(v2, *state.lookupVar(&env, *i));
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
state.eval(env2, body, v);
|
state.eval(env2, body, v);
|
||||||
}
|
}
|
||||||
|
@ -470,9 +483,16 @@ void ExprList::eval(EvalState & state, Env & env, Value & v)
|
||||||
|
|
||||||
void ExprVar::eval(EvalState & state, Env & env, Value & v)
|
void ExprVar::eval(EvalState & state, Env & env, Value & v)
|
||||||
{
|
{
|
||||||
Value * v2 = state.lookupVar(&env, name);
|
printMsg(lvlError, format("eval var %1% %2% %3%") % fromWith % level % displ);
|
||||||
state.forceValue(*v2);
|
|
||||||
v = *v2;
|
if (fromWith) {
|
||||||
|
abort();
|
||||||
|
} else {
|
||||||
|
Env * env2 = &env;
|
||||||
|
for (unsigned int l = level; l; --l, env2 = env2->up) ;
|
||||||
|
state.forceValue(env2->values[displ]);
|
||||||
|
v = env2->values[displ];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -559,22 +579,22 @@ void EvalState::callFunction(Value & fun, Value & arg, Value & v)
|
||||||
throwTypeError("attempt to call something which is neither a function nor a primop (built-in operation) but %1%",
|
throwTypeError("attempt to call something which is neither a function nor a primop (built-in operation) but %1%",
|
||||||
showType(fun));
|
showType(fun));
|
||||||
|
|
||||||
Env & env2(allocEnv());
|
unsigned int size =
|
||||||
|
(fun.lambda.fun->arg.empty() ? 0 : 1) +
|
||||||
|
(fun.lambda.fun->matchAttrs ? fun.lambda.fun->formals->formals.size() : 0);
|
||||||
|
Env & env2(allocEnv(size));
|
||||||
env2.up = fun.lambda.env;
|
env2.up = fun.lambda.env;
|
||||||
|
|
||||||
if (!fun.lambda.fun->matchAttrs) {
|
unsigned int displ = 0;
|
||||||
Value & vArg = env2.bindings[fun.lambda.fun->arg];
|
|
||||||
nrValues++;
|
if (!fun.lambda.fun->matchAttrs)
|
||||||
vArg = arg;
|
env2.values[displ++] = arg;
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
else {
|
||||||
forceAttrs(arg);
|
forceAttrs(arg);
|
||||||
|
|
||||||
if (!fun.lambda.fun->arg.empty()) {
|
if (!fun.lambda.fun->arg.empty())
|
||||||
env2.bindings[fun.lambda.fun->arg] = arg;
|
env2.values[displ++] = arg;
|
||||||
nrValues++;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* For each formal argument, get the actual argument. If
|
/* For each formal argument, get the actual argument. If
|
||||||
there is no matching actual argument but the formal
|
there is no matching actual argument but the formal
|
||||||
|
@ -582,17 +602,13 @@ void EvalState::callFunction(Value & fun, Value & arg, Value & v)
|
||||||
unsigned int attrsUsed = 0;
|
unsigned int attrsUsed = 0;
|
||||||
foreach (Formals::Formals_::iterator, i, fun.lambda.fun->formals->formals) {
|
foreach (Formals::Formals_::iterator, i, fun.lambda.fun->formals->formals) {
|
||||||
Bindings::iterator j = arg.attrs->find(i->name);
|
Bindings::iterator j = arg.attrs->find(i->name);
|
||||||
|
|
||||||
Value & v = env2.bindings[i->name];
|
|
||||||
nrValues++;
|
|
||||||
|
|
||||||
if (j == arg.attrs->end()) {
|
if (j == arg.attrs->end()) {
|
||||||
if (!i->def) throwTypeError("function at %1% called without required argument `%2%'",
|
if (!i->def) throwTypeError("function at %1% called without required argument `%2%'",
|
||||||
fun.lambda.fun->pos, i->name);
|
fun.lambda.fun->pos, i->name);
|
||||||
mkThunk(v, env2, i->def);
|
mkThunk(env2.values[displ++], env2, i->def);
|
||||||
} else {
|
} else {
|
||||||
attrsUsed++;
|
attrsUsed++;
|
||||||
mkCopy(v, j->second);
|
mkCopy(env2.values[displ++], j->second);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -639,6 +655,8 @@ void EvalState::autoCallFunction(const Bindings & args, Value & fun, Value & res
|
||||||
|
|
||||||
void ExprWith::eval(EvalState & state, Env & env, Value & v)
|
void ExprWith::eval(EvalState & state, Env & env, Value & v)
|
||||||
{
|
{
|
||||||
|
abort();
|
||||||
|
#if 0
|
||||||
Env & env2(state.allocEnv());
|
Env & env2(state.allocEnv());
|
||||||
env2.up = &env;
|
env2.up = &env;
|
||||||
|
|
||||||
|
@ -647,6 +665,7 @@ void ExprWith::eval(EvalState & state, Env & env, Value & v)
|
||||||
state.forceAttrs(vAttrs);
|
state.forceAttrs(vAttrs);
|
||||||
|
|
||||||
state.eval(env2, body, v);
|
state.eval(env2, body, v);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -18,13 +18,6 @@ struct Value;
|
||||||
typedef std::map<Symbol, Value> Bindings;
|
typedef std::map<Symbol, Value> Bindings;
|
||||||
|
|
||||||
|
|
||||||
struct Env
|
|
||||||
{
|
|
||||||
Env * up;
|
|
||||||
Bindings bindings;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
tInt = 1,
|
tInt = 1,
|
||||||
tBool,
|
tBool,
|
||||||
|
@ -109,6 +102,13 @@ struct Value
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
struct Env
|
||||||
|
{
|
||||||
|
Env * up;
|
||||||
|
Value values[0];
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
static inline void mkInt(Value & v, int n)
|
static inline void mkInt(Value & v, int n)
|
||||||
{
|
{
|
||||||
v.type = tInt;
|
v.type = tInt;
|
||||||
|
@ -258,7 +258,7 @@ public:
|
||||||
|
|
||||||
/* Allocation primitives. */
|
/* Allocation primitives. */
|
||||||
Value * allocValues(unsigned int count);
|
Value * allocValues(unsigned int count);
|
||||||
Env & allocEnv();
|
Env & allocEnv(unsigned int size);
|
||||||
|
|
||||||
void mkList(Value & v, unsigned int length);
|
void mkList(Value & v, unsigned int length);
|
||||||
void mkAttrs(Value & v);
|
void mkAttrs(Value & v);
|
||||||
|
|
|
@ -8,13 +8,14 @@
|
||||||
namespace nix {
|
namespace nix {
|
||||||
|
|
||||||
|
|
||||||
|
/* Displaying abstract syntax trees. */
|
||||||
|
|
||||||
std::ostream & operator << (std::ostream & str, Expr & e)
|
std::ostream & operator << (std::ostream & str, Expr & e)
|
||||||
{
|
{
|
||||||
e.show(str);
|
e.show(str);
|
||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Expr::show(std::ostream & str)
|
void Expr::show(std::ostream & str)
|
||||||
{
|
{
|
||||||
abort();
|
abort();
|
||||||
|
@ -137,101 +138,160 @@ std::ostream & operator << (std::ostream & str, const Pos & pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#if 0
|
/* Computing levels/displacements for variables. */
|
||||||
static void varsBoundByPattern(ATermMap & map, Pattern pat)
|
|
||||||
|
void Expr::bindVars(const StaticEnv & env)
|
||||||
{
|
{
|
||||||
ATerm name;
|
abort();
|
||||||
ATermList formals;
|
}
|
||||||
ATermBool ellipsis;
|
|
||||||
/* Use makeRemoved() so that it can be used directly in
|
void ExprInt::bindVars(const StaticEnv & env)
|
||||||
substitute(). */
|
{
|
||||||
if (matchVarPat(pat, name))
|
}
|
||||||
map.set(name, makeRemoved());
|
|
||||||
else if (matchAttrsPat(pat, formals, ellipsis, name)) {
|
void ExprString::bindVars(const StaticEnv & env)
|
||||||
if (name != sNoAlias) map.set(name, makeRemoved());
|
{
|
||||||
for (ATermIterator i(formals); i; ++i) {
|
}
|
||||||
ATerm d1;
|
|
||||||
if (!matchFormal(*i, name, d1)) abort();
|
void ExprPath::bindVars(const StaticEnv & env)
|
||||||
map.set(name, makeRemoved());
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprVar::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
/* Check whether the variable appears in the environment. If so,
|
||||||
|
set its level and displacement. */
|
||||||
|
const StaticEnv * curEnv;
|
||||||
|
unsigned int level;
|
||||||
|
int withLevel = -1;
|
||||||
|
for (curEnv = &env, level = 0; curEnv; curEnv = curEnv->up, level++) {
|
||||||
|
if (curEnv->isWith) {
|
||||||
|
if (withLevel == -1) withLevel = level;
|
||||||
|
} else {
|
||||||
|
StaticEnv::Vars::const_iterator i = curEnv->vars.find(name);
|
||||||
|
if (i != curEnv->vars.end()) {
|
||||||
|
fromWith = false;
|
||||||
|
this->level = level;
|
||||||
|
displ = i->second;
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else abort();
|
|
||||||
|
/* Otherwise, the variable must be obtained from the nearest
|
||||||
|
enclosing `with'. If there is no `with', then we can issue an
|
||||||
|
"undefined variable" error now. */
|
||||||
|
if (withLevel == -1) throw EvalError(format("undefined variable `%1%'") % name);
|
||||||
|
|
||||||
|
fromWith = true;
|
||||||
|
this->level = withLevel;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ExprSelect::bindVars(const StaticEnv & env)
|
||||||
/* We use memoisation to prevent exponential complexity on heavily
|
|
||||||
shared ATerms (remember, an ATerm is a graph, not a tree!). Note
|
|
||||||
that using an STL set is fine here wrt to ATerm garbage collection
|
|
||||||
since all the ATerms in the set are already reachable from
|
|
||||||
somewhere else. */
|
|
||||||
static void checkVarDefs2(set<Expr> & done, const ATermMap & defs, Expr e)
|
|
||||||
{
|
{
|
||||||
if (done.find(e) != done.end()) return;
|
e->bindVars(env);
|
||||||
done.insert(e);
|
|
||||||
|
|
||||||
ATerm name, pos, value;
|
|
||||||
ATerm with, body;
|
|
||||||
ATermList rbnds, nrbnds;
|
|
||||||
Pattern pat;
|
|
||||||
|
|
||||||
/* Closed terms don't have free variables, so we don't have to
|
|
||||||
check by definition. */
|
|
||||||
if (matchClosed(e, value)) return;
|
|
||||||
|
|
||||||
else if (matchVar(e, name)) {
|
|
||||||
if (!defs.get(name))
|
|
||||||
throw EvalError(format("undefined variable `%1%'")
|
|
||||||
% aterm2String(name));
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (matchFunction(e, pat, body, pos)) {
|
|
||||||
ATermMap defs2(defs);
|
|
||||||
varsBoundByPattern(defs2, pat);
|
|
||||||
set<Expr> done2;
|
|
||||||
checkVarDefs2(done2, defs2, pat);
|
|
||||||
checkVarDefs2(done2, defs2, body);
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (matchRec(e, rbnds, nrbnds)) {
|
|
||||||
checkVarDefs2(done, defs, (ATerm) nrbnds);
|
|
||||||
ATermMap defs2(defs);
|
|
||||||
for (ATermIterator i(rbnds); i; ++i) {
|
|
||||||
if (!matchBind(*i, name, value, pos)) abort(); /* can't happen */
|
|
||||||
defs2.set(name, (ATerm) ATempty);
|
|
||||||
}
|
|
||||||
for (ATermIterator i(nrbnds); i; ++i) {
|
|
||||||
if (!matchBind(*i, name, value, pos)) abort(); /* can't happen */
|
|
||||||
defs2.set(name, (ATerm) ATempty);
|
|
||||||
}
|
|
||||||
set<Expr> done2;
|
|
||||||
checkVarDefs2(done2, defs2, (ATerm) rbnds);
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (matchWith(e, with, body, pos)) {
|
|
||||||
/* We can't check the body without evaluating the definitions
|
|
||||||
(which is an arbitrary expression), so we don't do that
|
|
||||||
here but only when actually evaluating the `with'. */
|
|
||||||
checkVarDefs2(done, defs, with);
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (ATgetType(e) == AT_APPL) {
|
|
||||||
int arity = ATgetArity(ATgetAFun(e));
|
|
||||||
for (int i = 0; i < arity; ++i)
|
|
||||||
checkVarDefs2(done, defs, ATgetArgument(e, i));
|
|
||||||
}
|
|
||||||
|
|
||||||
else if (ATgetType(e) == AT_LIST)
|
|
||||||
for (ATermIterator i((ATermList) e); i; ++i)
|
|
||||||
checkVarDefs2(done, defs, *i);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ExprOpHasAttr::bindVars(const StaticEnv & env)
|
||||||
void checkVarDefs(const ATermMap & defs, Expr e)
|
|
||||||
{
|
{
|
||||||
set<Expr> done;
|
e->bindVars(env);
|
||||||
checkVarDefs2(done, defs, e);
|
}
|
||||||
|
|
||||||
|
void ExprAttrs::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
if (recursive) {
|
||||||
|
StaticEnv newEnv(false, &env);
|
||||||
|
|
||||||
|
unsigned int displ = 0;
|
||||||
|
|
||||||
|
foreach (ExprAttrs::Attrs::iterator, i, attrs)
|
||||||
|
newEnv.vars[i->first] = displ++;
|
||||||
|
|
||||||
|
foreach (list<Symbol>::iterator, i, inherited)
|
||||||
|
newEnv.vars[*i] = displ++;
|
||||||
|
|
||||||
|
foreach (ExprAttrs::Attrs::iterator, i, attrs)
|
||||||
|
i->second->bindVars(newEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
else
|
||||||
|
foreach (ExprAttrs::Attrs::iterator, i, attrs)
|
||||||
|
i->second->bindVars(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprList::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
foreach (vector<Expr *>::iterator, i, elems)
|
||||||
|
(*i)->bindVars(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprLambda::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
StaticEnv newEnv(false, &env);
|
||||||
|
|
||||||
|
unsigned int displ = 0;
|
||||||
|
|
||||||
|
if (!arg.empty()) newEnv.vars[arg] = displ++;
|
||||||
|
|
||||||
|
if (matchAttrs) {
|
||||||
|
foreach (Formals::Formals_::iterator, i, formals->formals)
|
||||||
|
newEnv.vars[i->name] = displ++;
|
||||||
|
|
||||||
|
foreach (Formals::Formals_::iterator, i, formals->formals)
|
||||||
|
if (i->def) i->def->bindVars(newEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
body->bindVars(newEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprLet::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
StaticEnv newEnv(false, &env);
|
||||||
|
|
||||||
|
unsigned int displ = 0;
|
||||||
|
|
||||||
|
foreach (ExprAttrs::Attrs::iterator, i, attrs->attrs)
|
||||||
|
newEnv.vars[i->first] = displ++;
|
||||||
|
|
||||||
|
foreach (list<Symbol>::iterator, i, attrs->inherited)
|
||||||
|
newEnv.vars[*i] = displ++;
|
||||||
|
|
||||||
|
foreach (ExprAttrs::Attrs::iterator, i, attrs->attrs)
|
||||||
|
i->second->bindVars(newEnv);
|
||||||
|
|
||||||
|
body->bindVars(newEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprWith::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
attrs->bindVars(env);
|
||||||
|
StaticEnv newEnv(true, &env);
|
||||||
|
body->bindVars(newEnv);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprIf::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
cond->bindVars(env);
|
||||||
|
then->bindVars(env);
|
||||||
|
else_->bindVars(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprAssert::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
cond->bindVars(env);
|
||||||
|
body->bindVars(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprOpNot::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
e->bindVars(env);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ExprConcatStrings::bindVars(const StaticEnv & env)
|
||||||
|
{
|
||||||
|
foreach (vector<Expr *>::iterator, i, *es)
|
||||||
|
(*i)->bindVars(env);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,25 +17,29 @@ MakeError(Abort, EvalError)
|
||||||
MakeError(TypeError, EvalError)
|
MakeError(TypeError, EvalError)
|
||||||
|
|
||||||
|
|
||||||
|
/* Position objects. */
|
||||||
|
|
||||||
struct Pos
|
struct Pos
|
||||||
{
|
{
|
||||||
string file;
|
string file;
|
||||||
unsigned int line, column;
|
unsigned int line, column;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
std::ostream & operator << (std::ostream & str, const Pos & pos);
|
std::ostream & operator << (std::ostream & str, const Pos & pos);
|
||||||
|
|
||||||
|
|
||||||
/* Abstract syntax of Nix expressions. */
|
|
||||||
|
|
||||||
struct Env;
|
struct Env;
|
||||||
struct Value;
|
struct Value;
|
||||||
struct EvalState;
|
struct EvalState;
|
||||||
|
struct StaticEnv;
|
||||||
|
|
||||||
|
|
||||||
|
/* Abstract syntax of Nix expressions. */
|
||||||
|
|
||||||
struct Expr
|
struct Expr
|
||||||
{
|
{
|
||||||
virtual void show(std::ostream & str);
|
virtual void show(std::ostream & str);
|
||||||
|
virtual void bindVars(const StaticEnv & env);
|
||||||
virtual void eval(EvalState & state, Env & env, Value & v);
|
virtual void eval(EvalState & state, Env & env, Value & v);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -43,7 +47,8 @@ std::ostream & operator << (std::ostream & str, Expr & e);
|
||||||
|
|
||||||
#define COMMON_METHODS \
|
#define COMMON_METHODS \
|
||||||
void show(std::ostream & str); \
|
void show(std::ostream & str); \
|
||||||
void eval(EvalState & state, Env & env, Value & v);
|
void eval(EvalState & state, Env & env, Value & v); \
|
||||||
|
void bindVars(const StaticEnv & env);
|
||||||
|
|
||||||
struct ExprInt : Expr
|
struct ExprInt : Expr
|
||||||
{
|
{
|
||||||
|
@ -76,6 +81,20 @@ struct ExprPath : Expr
|
||||||
struct ExprVar : Expr
|
struct ExprVar : Expr
|
||||||
{
|
{
|
||||||
Symbol name;
|
Symbol name;
|
||||||
|
|
||||||
|
/* Whether the variable comes from an environment (e.g. a rec, let
|
||||||
|
or function argument) or from a "with". */
|
||||||
|
bool fromWith;
|
||||||
|
|
||||||
|
/* In the former case, the value is obtained by going `level'
|
||||||
|
levels up from the current environment and getting the
|
||||||
|
`displ'th value in that environment. In the latter case, the
|
||||||
|
value is obtained by getting the attribute named `name' from
|
||||||
|
the attribute set stored in the environment that is `level'
|
||||||
|
levels up from the current one.*/
|
||||||
|
unsigned int level;
|
||||||
|
unsigned int displ;
|
||||||
|
|
||||||
ExprVar(const Symbol & name) : name(name) { };
|
ExprVar(const Symbol & name) : name(name) { };
|
||||||
COMMON_METHODS
|
COMMON_METHODS
|
||||||
};
|
};
|
||||||
|
@ -186,6 +205,10 @@ struct ExprOpNot : Expr
|
||||||
{ \
|
{ \
|
||||||
str << *e1 << " " s " " << *e2; \
|
str << *e1 << " " s " " << *e2; \
|
||||||
} \
|
} \
|
||||||
|
void bindVars(const StaticEnv & env) \
|
||||||
|
{ \
|
||||||
|
e1->bindVars(env); e2->bindVars(env); \
|
||||||
|
} \
|
||||||
void eval(EvalState & state, Env & env, Value & v); \
|
void eval(EvalState & state, Env & env, Value & v); \
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -206,11 +229,18 @@ struct ExprConcatStrings : Expr
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#if 0
|
/* Static environments are used to map variable names onto (level,
|
||||||
/* Check whether all variables are defined in the given expression.
|
displacement) pairs used to obtain the value of the variable at
|
||||||
Throw an exception if this isn't the case. */
|
runtime. */
|
||||||
void checkVarDefs(const ATermMap & def, Expr e);
|
struct StaticEnv
|
||||||
#endif
|
{
|
||||||
|
bool isWith;
|
||||||
|
const StaticEnv * up;
|
||||||
|
typedef std::map<Symbol, unsigned int> Vars;
|
||||||
|
Vars vars;
|
||||||
|
StaticEnv(bool isWith, const StaticEnv * up) : isWith(isWith), up(up) { };
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -461,7 +461,8 @@ static Expr * parse(EvalState & state, const char * text,
|
||||||
if (res) throw ParseError(data.error);
|
if (res) throw ParseError(data.error);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// !!! checkVarDefs(state.primOps, data.result);
|
StaticEnv env(false, 0);
|
||||||
|
data.result->bindVars(env);
|
||||||
} catch (Error & e) {
|
} catch (Error & e) {
|
||||||
throw ParseError(format("%1%, in `%2%'") % e.msg() % path);
|
throw ParseError(format("%1%, in `%2%'") % e.msg() % path);
|
||||||
}
|
}
|
||||||
|
|
|
@ -999,9 +999,11 @@ void EvalState::createBaseEnv()
|
||||||
{
|
{
|
||||||
baseEnv.up = 0;
|
baseEnv.up = 0;
|
||||||
|
|
||||||
|
#if 0
|
||||||
Value & builtins = baseEnv.bindings[symbols.create("builtins")];
|
Value & builtins = baseEnv.bindings[symbols.create("builtins")];
|
||||||
builtins.type = tAttrs;
|
builtins.type = tAttrs;
|
||||||
builtins.attrs = new Bindings;
|
builtins.attrs = new Bindings;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Add global constants such as `true' to the base environment. */
|
/* Add global constants such as `true' to the base environment. */
|
||||||
Value v;
|
Value v;
|
||||||
|
@ -1023,9 +1025,11 @@ void EvalState::createBaseEnv()
|
||||||
|
|
||||||
/* Add a wrapper around the derivation primop that computes the
|
/* Add a wrapper around the derivation primop that computes the
|
||||||
`drvPath' and `outPath' attributes lazily. */
|
`drvPath' and `outPath' attributes lazily. */
|
||||||
|
#if 0
|
||||||
string s = "attrs: let res = derivationStrict attrs; in attrs // { drvPath = res.drvPath; outPath = res.outPath; type = \"derivation\"; }";
|
string s = "attrs: let res = derivationStrict attrs; in attrs // { drvPath = res.drvPath; outPath = res.outPath; type = \"derivation\"; }";
|
||||||
mkThunk(v, baseEnv, parseExprFromString(*this, s, "/"));
|
mkThunk(v, baseEnv, parseExprFromString(*this, s, "/"));
|
||||||
addConstant("derivation", v);
|
addConstant("derivation", v);
|
||||||
|
#endif
|
||||||
|
|
||||||
// Miscellaneous
|
// Miscellaneous
|
||||||
addPrimOp("import", 1, prim_import);
|
addPrimOp("import", 1, prim_import);
|
||||||
|
|
Loading…
Reference in a new issue