2003-10-30 18:48:26 +02:00
|
|
|
|
#include "eval.hh"
|
2006-09-05 00:06:23 +03:00
|
|
|
|
#include "hash.hh"
|
|
|
|
|
#include "util.hh"
|
2006-11-30 19:43:04 +02:00
|
|
|
|
#include "store-api.hh"
|
2006-10-16 18:55:34 +03:00
|
|
|
|
#include "derivations.hh"
|
2006-12-01 22:51:18 +02:00
|
|
|
|
#include "globals.hh"
|
2012-02-04 15:50:25 +02:00
|
|
|
|
#include "eval-inline.hh"
|
2004-02-04 18:03:29 +02:00
|
|
|
|
|
2014-03-30 01:49:23 +02:00
|
|
|
|
#include <algorithm>
|
2010-03-29 17:37:56 +03:00
|
|
|
|
#include <cstring>
|
2011-02-10 16:31:04 +02:00
|
|
|
|
#include <unistd.h>
|
2012-02-04 15:27:11 +02:00
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
#include <sys/resource.h>
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
2010-10-22 16:39:15 +03:00
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
|
2010-10-20 14:38:30 +03:00
|
|
|
|
#include <gc/gc.h>
|
|
|
|
|
#include <gc/gc_cpp.h>
|
|
|
|
|
|
2010-10-24 03:41:29 +03:00
|
|
|
|
#define NEW new (UseGC)
|
2010-10-24 01:58:24 +03:00
|
|
|
|
|
2010-10-22 16:39:15 +03:00
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
#define GC_STRDUP strdup
|
|
|
|
|
#define GC_MALLOC malloc
|
|
|
|
|
|
2010-10-24 01:58:24 +03:00
|
|
|
|
#define NEW new
|
|
|
|
|
|
2010-10-22 16:39:15 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
2004-02-04 18:03:29 +02:00
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
|
namespace nix {
|
2010-10-24 03:41:29 +03:00
|
|
|
|
|
|
|
|
|
|
2010-10-24 22:52:33 +03:00
|
|
|
|
void Bindings::sort()
|
2010-10-24 03:41:29 +03:00
|
|
|
|
{
|
2010-10-24 22:52:33 +03:00
|
|
|
|
std::sort(begin(), end());
|
2010-10-24 03:41:29 +03:00
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
|
|
2014-09-22 15:59:37 +03:00
|
|
|
|
static void printValue(std::ostream & str, std::set<const Value *> & seen, const Value & v)
|
2003-10-30 18:48:26 +02:00
|
|
|
|
{
|
2014-09-22 15:59:37 +03:00
|
|
|
|
if (seen.find(&v) != seen.end()) {
|
|
|
|
|
str << "<CYCLE>";
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
seen.insert(&v);
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
switch (v.type) {
|
|
|
|
|
case tInt:
|
|
|
|
|
str << v.integer;
|
|
|
|
|
break;
|
|
|
|
|
case tBool:
|
|
|
|
|
str << (v.boolean ? "true" : "false");
|
|
|
|
|
break;
|
|
|
|
|
case tString:
|
2010-04-08 14:41:19 +03:00
|
|
|
|
str << "\"";
|
|
|
|
|
for (const char * i = v.string.s; *i; i++)
|
|
|
|
|
if (*i == '\"' || *i == '\\') str << "\\" << *i;
|
|
|
|
|
else if (*i == '\n') str << "\\n";
|
|
|
|
|
else if (*i == '\r') str << "\\r";
|
|
|
|
|
else if (*i == '\t') str << "\\t";
|
|
|
|
|
else str << *i;
|
|
|
|
|
str << "\"";
|
2010-03-29 17:37:56 +03:00
|
|
|
|
break;
|
2010-03-30 12:22:33 +03:00
|
|
|
|
case tPath:
|
|
|
|
|
str << v.path; // !!! escaping?
|
|
|
|
|
break;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
case tNull:
|
2010-08-27 15:10:56 +03:00
|
|
|
|
str << "null";
|
2010-03-29 17:37:56 +03:00
|
|
|
|
break;
|
2010-05-12 15:15:49 +03:00
|
|
|
|
case tAttrs: {
|
2010-03-29 17:37:56 +03:00
|
|
|
|
str << "{ ";
|
2010-05-12 15:15:49 +03:00
|
|
|
|
typedef std::map<string, Value *> Sorted;
|
|
|
|
|
Sorted sorted;
|
2010-04-15 01:59:39 +03:00
|
|
|
|
foreach (Bindings::iterator, i, *v.attrs)
|
2010-10-24 03:41:29 +03:00
|
|
|
|
sorted[i->name] = i->value;
|
2014-09-22 15:59:37 +03:00
|
|
|
|
for (auto & i : sorted) {
|
|
|
|
|
str << i.first << " = ";
|
|
|
|
|
printValue(str, seen, *i.second);
|
|
|
|
|
str << "; ";
|
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
str << "}";
|
|
|
|
|
break;
|
2010-05-12 15:15:49 +03:00
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
case tList:
|
|
|
|
|
str << "[ ";
|
2014-09-22 15:59:37 +03:00
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n) {
|
|
|
|
|
printValue(str, seen, *v.list.elems[n]);
|
|
|
|
|
str << " ";
|
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
str << "]";
|
|
|
|
|
break;
|
|
|
|
|
case tThunk:
|
2010-06-02 12:43:04 +03:00
|
|
|
|
case tApp:
|
2010-03-29 17:37:56 +03:00
|
|
|
|
str << "<CODE>";
|
|
|
|
|
break;
|
|
|
|
|
case tLambda:
|
|
|
|
|
str << "<LAMBDA>";
|
|
|
|
|
break;
|
|
|
|
|
case tPrimOp:
|
|
|
|
|
str << "<PRIMOP>";
|
|
|
|
|
break;
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
str << "<PRIMOP-APP>";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
throw Error("invalid value");
|
|
|
|
|
}
|
2014-09-22 15:59:37 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::ostream & operator << (std::ostream & str, const Value & v)
|
|
|
|
|
{
|
|
|
|
|
std::set<const Value *> seen;
|
|
|
|
|
printValue(str, seen, v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
return str;
|
|
|
|
|
}
|
2004-02-04 18:03:29 +02:00
|
|
|
|
|
2004-10-27 01:54:26 +03:00
|
|
|
|
|
2010-04-21 18:57:11 +03:00
|
|
|
|
string showType(const Value & v)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
|
|
|
|
switch (v.type) {
|
|
|
|
|
case tInt: return "an integer";
|
|
|
|
|
case tBool: return "a boolean";
|
2010-04-01 13:55:36 +03:00
|
|
|
|
case tString: return "a string";
|
|
|
|
|
case tPath: return "a path";
|
2010-04-21 18:57:11 +03:00
|
|
|
|
case tNull: return "null";
|
2013-10-24 17:41:04 +03:00
|
|
|
|
case tAttrs: return "a set";
|
2010-03-29 17:37:56 +03:00
|
|
|
|
case tList: return "a list";
|
2010-04-21 18:57:11 +03:00
|
|
|
|
case tThunk: return "a thunk";
|
|
|
|
|
case tApp: return "a function application";
|
2010-04-01 13:55:36 +03:00
|
|
|
|
case tLambda: return "a function";
|
2010-04-21 18:57:11 +03:00
|
|
|
|
case tBlackhole: return "a black hole";
|
2010-04-01 13:55:36 +03:00
|
|
|
|
case tPrimOp: return "a built-in function";
|
2010-03-29 17:37:56 +03:00
|
|
|
|
case tPrimOpApp: return "a partially applied built-in function";
|
|
|
|
|
}
|
2010-04-21 18:57:11 +03:00
|
|
|
|
abort();
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
2009-05-12 14:06:24 +03:00
|
|
|
|
|
2014-04-04 18:53:52 +03:00
|
|
|
|
#if HAVE_BOEHMGC
|
2013-11-23 22:19:36 +02:00
|
|
|
|
/* Called when the Boehm GC runs out of memory. */
|
|
|
|
|
static void * oomHandler(size_t requested)
|
|
|
|
|
{
|
|
|
|
|
/* Convert this to a proper C++ exception. */
|
|
|
|
|
throw std::bad_alloc();
|
|
|
|
|
}
|
2014-04-04 18:53:52 +03:00
|
|
|
|
#endif
|
2013-11-23 22:19:36 +02:00
|
|
|
|
|
|
|
|
|
|
2014-04-04 22:14:11 +03:00
|
|
|
|
static Symbol getName(const AttrName & name, EvalState & state, Env & env)
|
|
|
|
|
{
|
2014-01-01 01:56:26 +02:00
|
|
|
|
if (name.symbol.set()) {
|
|
|
|
|
return name.symbol;
|
|
|
|
|
} else {
|
|
|
|
|
Value nameValue;
|
|
|
|
|
name.expr->eval(state, env, nameValue);
|
|
|
|
|
state.forceStringNoCtx(nameValue);
|
|
|
|
|
return state.symbols.create(nameValue.string.s);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-05-26 17:50:36 +03:00
|
|
|
|
EvalState::EvalState(const Strings & _searchPath)
|
2010-04-13 15:25:42 +03:00
|
|
|
|
: sWith(symbols.create("<with>"))
|
|
|
|
|
, sOutPath(symbols.create("outPath"))
|
|
|
|
|
, sDrvPath(symbols.create("drvPath"))
|
|
|
|
|
, sType(symbols.create("type"))
|
|
|
|
|
, sMeta(symbols.create("meta"))
|
|
|
|
|
, sName(symbols.create("name"))
|
2013-10-28 08:34:44 +02:00
|
|
|
|
, sValue(symbols.create("value"))
|
2010-04-21 18:08:58 +03:00
|
|
|
|
, sSystem(symbols.create("system"))
|
2010-05-15 11:10:12 +03:00
|
|
|
|
, sOverrides(symbols.create("__overrides"))
|
2012-11-28 14:49:44 +02:00
|
|
|
|
, sOutputs(symbols.create("outputs"))
|
2012-11-26 18:39:09 +02:00
|
|
|
|
, sOutputName(symbols.create("outputName"))
|
2012-11-27 16:01:32 +02:00
|
|
|
|
, sIgnoreNulls(symbols.create("__ignoreNulls"))
|
2013-11-18 21:14:54 +02:00
|
|
|
|
, sFile(symbols.create("file"))
|
|
|
|
|
, sLine(symbols.create("line"))
|
|
|
|
|
, sColumn(symbols.create("column"))
|
2013-05-16 18:48:19 +03:00
|
|
|
|
, repair(false)
|
2010-04-14 17:42:32 +03:00
|
|
|
|
, baseEnv(allocEnv(128))
|
2010-04-15 01:59:39 +03:00
|
|
|
|
, staticBaseEnv(false, 0)
|
2013-09-02 19:34:04 +03:00
|
|
|
|
, baseEnvDispl(0)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
2010-04-15 03:37:36 +03:00
|
|
|
|
nrEnvs = nrValuesInEnvs = nrValues = nrListElems = 0;
|
2014-09-19 17:49:41 +03:00
|
|
|
|
nrAttrsets = nrAttrsInAttrsets = nrOpUpdates = nrOpUpdateValuesCopied = 0;
|
2012-08-13 06:41:48 +03:00
|
|
|
|
nrListConcats = nrPrimOpCalls = nrFunctionCalls = 0;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
countCalls = getEnv("NIX_COUNT_CALLS", "0") != "0";
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
2011-02-10 00:59:50 +02:00
|
|
|
|
#if HAVE_BOEHMGC
|
2013-11-23 22:15:57 +02:00
|
|
|
|
static bool gcInitialised = false;
|
|
|
|
|
if (!gcInitialised) {
|
2013-11-23 22:19:36 +02:00
|
|
|
|
|
|
|
|
|
/* Initialise the Boehm garbage collector. This isn't
|
|
|
|
|
necessary on most platforms, but for portability we do it
|
|
|
|
|
anyway. */
|
|
|
|
|
GC_INIT();
|
|
|
|
|
|
|
|
|
|
GC_oom_fn = oomHandler;
|
|
|
|
|
|
2011-02-10 16:31:04 +02:00
|
|
|
|
/* Set the initial heap size to something fairly big (25% of
|
|
|
|
|
physical RAM, up to a maximum of 384 MiB) so that in most
|
|
|
|
|
cases we don't need to garbage collect at all. (Collection
|
|
|
|
|
has a fairly significant overhead.) The heap size can be
|
2013-08-11 00:36:16 +03:00
|
|
|
|
overridden through libgc's GC_INITIAL_HEAP_SIZE environment
|
2011-02-10 16:31:04 +02:00
|
|
|
|
variable. We should probably also provide a nix.conf
|
|
|
|
|
setting for this. Note that GC_expand_hp() causes a lot of
|
|
|
|
|
virtual, but not physical (resident) memory to be
|
|
|
|
|
allocated. This might be a problem on systems that don't
|
|
|
|
|
overcommit. */
|
|
|
|
|
if (!getenv("GC_INITIAL_HEAP_SIZE")) {
|
|
|
|
|
size_t maxSize = 384 * 1024 * 1024;
|
|
|
|
|
size_t size = 32 * 1024 * 1024;
|
|
|
|
|
#if HAVE_SYSCONF && defined(_SC_PAGESIZE) && defined(_SC_PHYS_PAGES)
|
|
|
|
|
long pageSize = sysconf(_SC_PAGESIZE);
|
2011-10-27 22:06:23 +03:00
|
|
|
|
long pages = sysconf(_SC_PHYS_PAGES);
|
|
|
|
|
if (pageSize != -1)
|
2011-02-10 16:31:04 +02:00
|
|
|
|
size = (pageSize * pages) / 4; // 25% of RAM
|
|
|
|
|
if (size > maxSize) size = maxSize;
|
|
|
|
|
#endif
|
|
|
|
|
debug(format("setting initial heap size to %1% bytes") % size);
|
|
|
|
|
GC_expand_hp(size);
|
|
|
|
|
}
|
2013-11-23 22:19:36 +02:00
|
|
|
|
|
2011-02-10 00:59:50 +02:00
|
|
|
|
gcInitialised = true;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2011-08-06 19:05:24 +03:00
|
|
|
|
|
|
|
|
|
/* Initialise the Nix expression search path. */
|
2012-09-19 22:43:23 +03:00
|
|
|
|
Strings paths = tokenizeString<Strings>(getEnv("NIX_PATH", ""), ":");
|
2014-08-13 03:57:59 +03:00
|
|
|
|
for (auto & i : _searchPath) addToSearchPath(i, true);
|
2014-05-26 17:50:36 +03:00
|
|
|
|
for (auto & i : paths) addToSearchPath(i);
|
2012-07-31 02:55:41 +03:00
|
|
|
|
addToSearchPath("nix=" + settings.nixDataDir + "/nix/corepkgs");
|
2012-01-03 16:01:47 +02:00
|
|
|
|
|
|
|
|
|
createBaseEnv();
|
2004-02-04 18:03:29 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-09 15:00:49 +03:00
|
|
|
|
EvalState::~EvalState()
|
|
|
|
|
{
|
2014-09-17 16:19:07 +03:00
|
|
|
|
fileEvalCache.clear();
|
|
|
|
|
printCanaries();
|
2010-04-09 15:00:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-30 17:39:27 +03:00
|
|
|
|
void EvalState::addConstant(const string & name, Value & v)
|
|
|
|
|
{
|
2010-10-22 17:47:42 +03:00
|
|
|
|
Value * v2 = allocValue();
|
|
|
|
|
*v2 = v;
|
2010-04-15 01:59:39 +03:00
|
|
|
|
staticBaseEnv.vars[symbols.create(name)] = baseEnvDispl;
|
2010-10-22 18:51:52 +03:00
|
|
|
|
baseEnv.values[baseEnvDispl++] = v2;
|
2010-03-30 17:39:27 +03:00
|
|
|
|
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
2010-10-24 03:41:29 +03:00
|
|
|
|
baseEnv.values[0]->attrs->push_back(Attr(symbols.create(name2), v2));
|
2010-03-30 17:39:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2004-08-04 13:59:20 +03:00
|
|
|
|
void EvalState::addPrimOp(const string & name,
|
2010-10-23 23:07:47 +03:00
|
|
|
|
unsigned int arity, PrimOpFun primOp)
|
2004-02-04 18:03:29 +02:00
|
|
|
|
{
|
2010-10-22 17:47:42 +03:00
|
|
|
|
Value * v = allocValue();
|
2010-04-21 18:08:58 +03:00
|
|
|
|
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
2010-10-24 22:52:33 +03:00
|
|
|
|
Symbol sym = symbols.create(name2);
|
2010-10-22 17:47:42 +03:00
|
|
|
|
v->type = tPrimOp;
|
2010-10-24 01:58:24 +03:00
|
|
|
|
v->primOp = NEW PrimOp(primOp, arity, sym);
|
2010-10-24 22:52:33 +03:00
|
|
|
|
staticBaseEnv.vars[symbols.create(name)] = baseEnvDispl;
|
2010-10-22 18:51:52 +03:00
|
|
|
|
baseEnv.values[baseEnvDispl++] = v;
|
2010-10-24 22:52:33 +03:00
|
|
|
|
baseEnv.values[0]->attrs->push_back(Attr(sym, v));
|
2003-10-31 19:09:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-03 16:45:32 +03:00
|
|
|
|
void EvalState::getBuiltin(const string & name, Value & v)
|
|
|
|
|
{
|
|
|
|
|
v = *baseEnv.values[0]->attrs->find(symbols.create(name))->value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2007-02-27 21:10:45 +02:00
|
|
|
|
/* Every "format" object (even temporary) takes up a few hundred bytes
|
|
|
|
|
of stack space, which is a real killer in the recursive
|
|
|
|
|
evaluator. So here are some helper functions for throwing
|
|
|
|
|
exceptions. */
|
|
|
|
|
|
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const string & s2))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % s2);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % pos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const string & s2, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % s2 % pos);
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-09 15:00:49 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const string & s2, const string & s3))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % s2 % s3);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 22:14:11 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const string & s2, const string & s3, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % s2 % s3 % pos);
|
|
|
|
|
}
|
|
|
|
|
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const Symbol & sym, const Pos & p1, const Pos & p2))
|
|
|
|
|
{
|
|
|
|
|
throw EvalError(format(s) % sym % p1 % p2);
|
|
|
|
|
}
|
|
|
|
|
|
2008-07-24 17:52:25 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const char * s))
|
|
|
|
|
{
|
|
|
|
|
throw TypeError(s);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const char * s, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
throw TypeError(format(s) % pos);
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-07 14:44:14 +02:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const char * s, const string & s1))
|
|
|
|
|
{
|
|
|
|
|
throw TypeError(format(s) % s1);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-16 20:08:02 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const char * s, const string & s1, const string & s2))
|
|
|
|
|
{
|
|
|
|
|
throw TypeError(format(s) % s1 % s2);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 18:53:52 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const char * s, const ExprLambda & fun, const Symbol & s2, const Pos & pos))
|
2013-11-07 19:04:36 +02:00
|
|
|
|
{
|
2014-04-04 18:53:52 +03:00
|
|
|
|
throw TypeError(format(s) % fun.showNamePos() % s2 % pos);
|
2013-11-07 19:04:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
2010-04-13 00:21:24 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwAssertionError(const char * s, const Pos & pos))
|
2010-04-09 15:00:49 +03:00
|
|
|
|
{
|
2010-04-13 00:21:24 +03:00
|
|
|
|
throw AssertionError(format(s) % pos);
|
2010-04-09 15:00:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
2013-10-08 15:45:36 +03:00
|
|
|
|
LocalNoInlineNoReturn(void throwUndefinedVarError(const char * s, const string & s1, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
throw UndefinedVarError(format(s) % s1 % pos);
|
|
|
|
|
}
|
|
|
|
|
|
2007-02-27 21:10:45 +02:00
|
|
|
|
LocalNoInline(void addErrorPrefix(Error & e, const char * s, const string & s2))
|
|
|
|
|
{
|
|
|
|
|
e.addPrefix(format(s) % s2);
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 18:53:52 +03:00
|
|
|
|
LocalNoInline(void addErrorPrefix(Error & e, const char * s, const ExprLambda & fun, const Pos & pos))
|
2013-11-12 13:51:59 +02:00
|
|
|
|
{
|
2014-04-04 18:53:52 +03:00
|
|
|
|
e.addPrefix(format(s) % fun.showNamePos() % pos);
|
2013-11-12 13:51:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2010-05-07 15:11:05 +03:00
|
|
|
|
LocalNoInline(void addErrorPrefix(Error & e, const char * s, const string & s2, const Pos & pos))
|
|
|
|
|
{
|
|
|
|
|
e.addPrefix(format(s) % s2 % pos);
|
|
|
|
|
}
|
|
|
|
|
|
2007-02-27 21:10:45 +02:00
|
|
|
|
|
2010-03-30 21:05:54 +03:00
|
|
|
|
void mkString(Value & v, const char * s)
|
|
|
|
|
{
|
2012-01-07 19:26:33 +02:00
|
|
|
|
mkStringNoCopy(v, GC_STRDUP(s));
|
2010-03-30 21:05:54 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void mkString(Value & v, const string & s, const PathSet & context)
|
|
|
|
|
{
|
|
|
|
|
mkString(v, s.c_str());
|
2010-03-31 22:52:29 +03:00
|
|
|
|
if (!context.empty()) {
|
2010-04-01 12:55:57 +03:00
|
|
|
|
unsigned int n = 0;
|
2010-10-28 15:50:01 +03:00
|
|
|
|
v.string.context = (const char * *)
|
|
|
|
|
GC_MALLOC((context.size() + 1) * sizeof(char *));
|
2013-09-02 17:29:15 +03:00
|
|
|
|
foreach (PathSet::const_iterator, i, context)
|
2010-10-22 16:39:15 +03:00
|
|
|
|
v.string.context[n++] = GC_STRDUP(i->c_str());
|
2010-03-31 22:52:29 +03:00
|
|
|
|
v.string.context[n] = 0;
|
|
|
|
|
}
|
2010-03-30 21:05:54 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void mkPath(Value & v, const char * s)
|
|
|
|
|
{
|
2012-01-07 19:26:33 +02:00
|
|
|
|
mkPathNoCopy(v, GC_STRDUP(s));
|
2010-03-30 21:05:54 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-10-08 15:24:53 +03:00
|
|
|
|
inline Value * EvalState::lookupVar(Env * env, const ExprVar & var, bool noEval)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
2010-04-14 18:14:23 +03:00
|
|
|
|
for (unsigned int l = var.level; l; --l, env = env->up) ;
|
2013-07-31 13:44:21 +03:00
|
|
|
|
|
|
|
|
|
if (!var.fromWith) return env->values[var.displ];
|
|
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
|
if (!env->haveWithAttrs) {
|
|
|
|
|
if (noEval) return 0;
|
2013-10-02 16:24:45 +03:00
|
|
|
|
Value * v = allocValue();
|
|
|
|
|
evalAttrs(*env->up, (Expr *) env->values[0], *v);
|
|
|
|
|
env->values[0] = v;
|
2013-07-31 13:44:21 +03:00
|
|
|
|
env->haveWithAttrs = true;
|
2010-04-22 18:08:09 +03:00
|
|
|
|
}
|
2013-07-31 13:44:21 +03:00
|
|
|
|
Bindings::iterator j = env->values[0]->attrs->find(var.name);
|
|
|
|
|
if (j != env->values[0]->attrs->end()) {
|
|
|
|
|
if (countCalls && j->pos) attrSelects[*j->pos]++;
|
|
|
|
|
return j->value;
|
|
|
|
|
}
|
|
|
|
|
if (!env->prevWith)
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwUndefinedVarError("undefined variable ‘%1%’ at %2%", var.name, var.pos);
|
2013-07-31 13:44:21 +03:00
|
|
|
|
for (unsigned int l = env->prevWith; l; --l, env = env->up) ;
|
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-22 17:47:42 +03:00
|
|
|
|
Value * EvalState::allocValue()
|
|
|
|
|
{
|
|
|
|
|
nrValues++;
|
2010-10-28 15:50:01 +03:00
|
|
|
|
return (Value *) GC_MALLOC(sizeof(Value));
|
2010-10-22 17:47:42 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-14 17:42:32 +03:00
|
|
|
|
Env & EvalState::allocEnv(unsigned int size)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
2014-09-22 15:46:42 +03:00
|
|
|
|
assert(size <= std::numeric_limits<decltype(Env::size)>::max());
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
nrEnvs++;
|
2010-04-15 02:48:46 +03:00
|
|
|
|
nrValuesInEnvs += size;
|
2010-10-22 18:51:52 +03:00
|
|
|
|
Env * env = (Env *) GC_MALLOC(sizeof(Env) + size * sizeof(Value *));
|
2014-09-22 15:46:42 +03:00
|
|
|
|
env->size = size;
|
2010-10-24 17:20:02 +03:00
|
|
|
|
|
2013-07-15 22:53:14 +03:00
|
|
|
|
/* Clear the values because maybeThunk() and lookupVar fromWith expects this. */
|
2010-10-24 17:20:02 +03:00
|
|
|
|
for (unsigned i = 0; i < size; ++i)
|
|
|
|
|
env->values[i] = 0;
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-04-14 17:42:32 +03:00
|
|
|
|
return *env;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-22 17:47:42 +03:00
|
|
|
|
Value * EvalState::allocAttr(Value & vAttrs, const Symbol & name)
|
|
|
|
|
{
|
2010-10-24 03:41:29 +03:00
|
|
|
|
Value * v = allocValue();
|
|
|
|
|
vAttrs.attrs->push_back(Attr(name, v));
|
|
|
|
|
return v;
|
2010-10-22 17:47:42 +03:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2014-09-19 17:49:41 +03:00
|
|
|
|
Bindings * EvalState::allocBindings(Bindings::size_t capacity)
|
|
|
|
|
{
|
|
|
|
|
return new (GC_MALLOC(sizeof(Bindings) + sizeof(Attr) * capacity)) Bindings(capacity);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-30 17:39:27 +03:00
|
|
|
|
void EvalState::mkList(Value & v, unsigned int length)
|
|
|
|
|
{
|
|
|
|
|
v.type = tList;
|
|
|
|
|
v.list.length = length;
|
2012-08-13 22:02:09 +03:00
|
|
|
|
v.list.elems = length ? (Value * *) GC_MALLOC(length * sizeof(Value *)) : 0;
|
2010-04-15 03:37:36 +03:00
|
|
|
|
nrListElems += length;
|
2010-03-30 17:39:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-24 23:09:37 +03:00
|
|
|
|
void EvalState::mkAttrs(Value & v, unsigned int expected)
|
2010-03-31 01:39:48 +03:00
|
|
|
|
{
|
2010-10-23 23:07:47 +03:00
|
|
|
|
clearValue(v);
|
2010-03-31 01:39:48 +03:00
|
|
|
|
v.type = tAttrs;
|
2014-09-19 17:49:41 +03:00
|
|
|
|
v.attrs = allocBindings(expected);
|
2010-10-20 18:48:00 +03:00
|
|
|
|
nrAttrsets++;
|
2014-09-19 17:49:41 +03:00
|
|
|
|
nrAttrsInAttrsets += expected;
|
2010-03-31 01:39:48 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-24 17:20:02 +03:00
|
|
|
|
unsigned long nrThunks = 0;
|
|
|
|
|
|
|
|
|
|
static inline void mkThunk(Value & v, Env & env, Expr * expr)
|
|
|
|
|
{
|
|
|
|
|
v.type = tThunk;
|
|
|
|
|
v.thunk.env = &env;
|
|
|
|
|
v.thunk.expr = expr;
|
|
|
|
|
nrThunks++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void EvalState::mkThunk_(Value & v, Expr * expr)
|
2010-04-07 18:47:06 +03:00
|
|
|
|
{
|
|
|
|
|
mkThunk(v, baseEnv, expr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-18 23:22:35 +02:00
|
|
|
|
void EvalState::mkPos(Value & v, Pos * pos)
|
|
|
|
|
{
|
|
|
|
|
if (pos) {
|
|
|
|
|
mkAttrs(v, 3);
|
|
|
|
|
mkString(*allocAttr(v, sFile), pos->file);
|
|
|
|
|
mkInt(*allocAttr(v, sLine), pos->line);
|
|
|
|
|
mkInt(*allocAttr(v, sColumn), pos->column);
|
|
|
|
|
v.attrs->sort();
|
|
|
|
|
} else
|
|
|
|
|
mkNull(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-24 17:20:02 +03:00
|
|
|
|
/* Create a thunk for the delayed computation of the given expression
|
|
|
|
|
in the given environment. But if the expression is a variable,
|
|
|
|
|
then look it up right away. This significantly reduces the number
|
|
|
|
|
of thunks allocated. */
|
2012-01-04 23:24:11 +02:00
|
|
|
|
Value * Expr::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
|
|
|
|
Value * v = state.allocValue();
|
|
|
|
|
mkThunk(*v, env, this);
|
|
|
|
|
return v;
|
|
|
|
|
}
|
2010-10-24 17:20:02 +03:00
|
|
|
|
|
|
|
|
|
|
2012-01-04 23:24:11 +02:00
|
|
|
|
unsigned long nrAvoided = 0;
|
|
|
|
|
|
|
|
|
|
Value * ExprVar::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2013-10-08 15:24:53 +03:00
|
|
|
|
Value * v = state.lookupVar(&env, *this, true);
|
2013-07-16 15:43:54 +03:00
|
|
|
|
/* The value might not be initialised in the environment yet.
|
|
|
|
|
In that case, ignore it. */
|
|
|
|
|
if (v) { nrAvoided++; return v; }
|
2012-01-04 23:24:11 +02:00
|
|
|
|
return Expr::maybeThunk(state, env);
|
2010-10-24 17:20:02 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-01-07 19:26:33 +02:00
|
|
|
|
Value * ExprString::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
|
|
|
|
nrAvoided++;
|
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Value * ExprInt::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
|
|
|
|
nrAvoided++;
|
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Value * ExprPath::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
|
|
|
|
nrAvoided++;
|
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-30 12:22:33 +03:00
|
|
|
|
void EvalState::evalFile(const Path & path, Value & v)
|
|
|
|
|
{
|
2013-10-23 14:16:46 +03:00
|
|
|
|
FileEvalCache::iterator i;
|
|
|
|
|
if ((i = fileEvalCache.find(path)) != fileEvalCache.end()) {
|
|
|
|
|
v = i->second;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2013-09-03 13:56:33 +03:00
|
|
|
|
|
2013-10-23 14:16:46 +03:00
|
|
|
|
Path path2 = resolveExprPath(path);
|
|
|
|
|
if ((i = fileEvalCache.find(path2)) != fileEvalCache.end()) {
|
2011-08-06 22:45:43 +03:00
|
|
|
|
v = i->second;
|
2013-09-03 13:56:33 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-20 18:00:17 +03:00
|
|
|
|
startNest(nest, lvlTalkative, format("evaluating file ‘%1%’") % path2);
|
2013-09-03 13:56:33 +03:00
|
|
|
|
Expr * e = parseExprFromFile(path2);
|
|
|
|
|
try {
|
|
|
|
|
eval(e, v);
|
|
|
|
|
} catch (Error & e) {
|
2014-08-20 18:00:17 +03:00
|
|
|
|
addErrorPrefix(e, "while evaluating the file ‘%1%’:\n", path2);
|
2013-09-03 13:56:33 +03:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2013-10-23 14:16:46 +03:00
|
|
|
|
|
2013-09-03 13:56:33 +03:00
|
|
|
|
fileEvalCache[path2] = v;
|
2013-10-23 14:16:46 +03:00
|
|
|
|
if (path != path2) fileEvalCache[path] = v;
|
2010-03-30 12:22:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 19:34:04 +03:00
|
|
|
|
void EvalState::resetFileCache()
|
|
|
|
|
{
|
|
|
|
|
fileEvalCache.clear();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void EvalState::eval(Expr * e, Value & v)
|
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
e->eval(*this, baseEnv, v);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
|
2014-03-05 17:18:13 +02:00
|
|
|
|
inline bool EvalState::evalBool(Env & env, Expr * e)
|
2010-04-12 21:30:11 +03:00
|
|
|
|
{
|
2014-03-05 17:18:13 +02:00
|
|
|
|
Value v;
|
2012-02-04 15:50:25 +02:00
|
|
|
|
e->eval(*this, env, v);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
if (v.type != tBool)
|
2013-11-07 14:44:14 +02:00
|
|
|
|
throwTypeError("value is %1% while a Boolean was expected", v);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 23:43:52 +03:00
|
|
|
|
inline bool EvalState::evalBool(Env & env, Expr * e, const Pos & pos)
|
|
|
|
|
{
|
|
|
|
|
Value v;
|
|
|
|
|
e->eval(*this, env, v);
|
|
|
|
|
if (v.type != tBool)
|
|
|
|
|
throwTypeError("value is %1% while a Boolean was expected, at %2%", v, pos);
|
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-02-04 15:50:25 +02:00
|
|
|
|
inline void EvalState::evalAttrs(Env & env, Expr * e, Value & v)
|
2010-04-16 18:13:47 +03:00
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
e->eval(*this, env, v);
|
2010-04-16 18:13:47 +03:00
|
|
|
|
if (v.type != tAttrs)
|
2013-11-07 14:44:14 +02:00
|
|
|
|
throwTypeError("value is %1% while a set was expected", v);
|
2010-04-16 18:13:47 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-13 01:03:27 +03:00
|
|
|
|
void Expr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprInt::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 19:26:33 +02:00
|
|
|
|
v = this->v;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprString::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 19:26:33 +02:00
|
|
|
|
v = this->v;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprPath::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 19:26:33 +02:00
|
|
|
|
v = this->v;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprAttrs::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-09-19 17:49:41 +03:00
|
|
|
|
state.mkAttrs(v, attrs.size() + dynamicAttrs.size());
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
Env *dynamicEnv = &env;
|
2010-04-15 02:25:05 +03:00
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
if (recursive) {
|
|
|
|
|
/* Create a new environment that contains the attributes in
|
|
|
|
|
this `rec'. */
|
2010-10-24 22:52:33 +03:00
|
|
|
|
Env & env2(state.allocEnv(attrs.size()));
|
2010-04-12 21:30:11 +03:00
|
|
|
|
env2.up = &env;
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
dynamicEnv = &env2;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
|
2010-10-24 22:52:33 +03:00
|
|
|
|
AttrDefs::iterator overrides = attrs.find(state.sOverrides);
|
|
|
|
|
bool hasOverrides = overrides != attrs.end();
|
|
|
|
|
|
|
|
|
|
/* The recursive attributes are evaluated in the new
|
|
|
|
|
environment, while the inherited attributes are evaluated
|
|
|
|
|
in the original environment. */
|
2010-04-14 17:42:32 +03:00
|
|
|
|
unsigned int displ = 0;
|
2013-07-16 00:10:18 +03:00
|
|
|
|
foreach (AttrDefs::iterator, i, attrs) {
|
|
|
|
|
Value * vAttr;
|
|
|
|
|
if (hasOverrides && !i->second.inherited) {
|
|
|
|
|
vAttr = state.allocValue();
|
|
|
|
|
mkThunk(*vAttr, env2, i->second.e);
|
|
|
|
|
} else
|
|
|
|
|
vAttr = i->second.e->maybeThunk(state, i->second.inherited ? env : env2);
|
|
|
|
|
env2.values[displ++] = vAttr;
|
|
|
|
|
v.attrs->push_back(Attr(i->first, vAttr, &i->second.pos));
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-15 11:10:12 +03:00
|
|
|
|
/* If the rec contains an attribute called `__overrides', then
|
|
|
|
|
evaluate it, and add the attributes in that set to the rec.
|
|
|
|
|
This allows overriding of recursive attributes, which is
|
|
|
|
|
otherwise not possible. (You can use the // operator to
|
|
|
|
|
replace an attribute, but other attributes in the rec will
|
|
|
|
|
still reference the original value, because that value has
|
|
|
|
|
been substituted into the bodies of the other attributes.
|
|
|
|
|
Hence we need __overrides.) */
|
2010-10-24 22:52:33 +03:00
|
|
|
|
if (hasOverrides) {
|
|
|
|
|
Value * vOverrides = (*v.attrs)[overrides->second.displ].value;
|
|
|
|
|
state.forceAttrs(*vOverrides);
|
2014-09-19 17:49:41 +03:00
|
|
|
|
Bindings * newBnds = state.allocBindings(v.attrs->size() + vOverrides->attrs->size());
|
|
|
|
|
for (auto & i : *v.attrs)
|
|
|
|
|
newBnds->push_back(i);
|
|
|
|
|
for (auto & i : *vOverrides->attrs) {
|
|
|
|
|
AttrDefs::iterator j = attrs.find(i.name);
|
2010-10-24 22:52:33 +03:00
|
|
|
|
if (j != attrs.end()) {
|
2014-09-19 17:49:41 +03:00
|
|
|
|
(*newBnds)[j->second.displ] = i;
|
|
|
|
|
env2.values[j->second.displ] = i.value;
|
2010-10-24 22:52:33 +03:00
|
|
|
|
} else
|
2014-09-19 17:49:41 +03:00
|
|
|
|
newBnds->push_back(i);
|
2010-05-15 11:10:12 +03:00
|
|
|
|
}
|
2014-09-19 17:49:41 +03:00
|
|
|
|
newBnds->sort();
|
|
|
|
|
v.attrs = newBnds;
|
2010-05-15 11:10:12 +03:00
|
|
|
|
}
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
else
|
2010-10-24 22:52:33 +03:00
|
|
|
|
foreach (AttrDefs::iterator, i, attrs)
|
2013-07-16 00:10:18 +03:00
|
|
|
|
v.attrs->push_back(Attr(i->first, i->second.e->maybeThunk(state, env), &i->second.pos));
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
/* Dynamic attrs apply *after* rec and __overrides. */
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
foreach (DynamicAttrDefs::iterator, i, dynamicAttrs) {
|
|
|
|
|
Value nameVal;
|
2014-03-09 20:41:02 +02:00
|
|
|
|
if (i->nameExpr->es->size() == 1) {
|
|
|
|
|
i->nameExpr->es->front()->eval(state, *dynamicEnv, nameVal);
|
2014-03-09 20:24:47 +02:00
|
|
|
|
state.forceValue(nameVal);
|
|
|
|
|
if (nameVal.type == tNull)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2014-03-09 20:41:02 +02:00
|
|
|
|
i->nameExpr->eval(state, *dynamicEnv, nameVal);
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
state.forceStringNoCtx(nameVal);
|
|
|
|
|
Symbol nameSym = state.symbols.create(nameVal.string.s);
|
|
|
|
|
Bindings::iterator j = v.attrs->find(nameSym);
|
|
|
|
|
if (j != v.attrs->end())
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwEvalError("dynamic attribute ‘%1%’ at %2% already defined at %3%", nameSym, i->pos, *j->pos);
|
Dynamic attrs
This adds new syntax for attribute names:
* attrs."${name}" => getAttr name attrs
* attrs ? "${name}" => isAttrs attrs && hasAttr attrs name
* attrs."${name}" or def => if attrs ? "${name}" then attrs."${name}" else def
* { "${name}" = value; } => listToAttrs [{ inherit name value; }]
Of course, it's a bit more complicated than that. The attribute chains
can be arbitrarily long and contain combinations of static and dynamic
parts (e.g. attrs."${foo}".bar."${baz}" or qux), which is relatively
straightforward for the getAttrs/hasAttrs cases but is more complex for
the listToAttrs case due to rules about duplicate attribute definitions.
For attribute sets with dynamic attribute names, duplicate static
attributes are detected at parse time while duplicate dynamic attributes
are detected when the attribute set is forced. So, for example, { a =
null; a.b = null; "${"c"}" = true; } will be a parse-time error, while
{ a = {}; "${"a"}".b = null; c = true; } will be an eval-time error
(technically that case could theoretically be detected at parse time,
but the general case would require full evaluation). Moreover, duplicate
dynamic attributes are not allowed even in cases where they would be
with static attributes ({ a.b.d = true; a.b.c = false; } is legal, but {
a."${"b"}".d = true; a."${"b"}".c = false; } is not). This restriction
might be relaxed in the future in cases where the static variant would
not be an error, but it is not obvious that that is desirable.
Finally, recursive attribute sets with dynamic attributes have the
static attributes in scope but not the dynamic ones. So rec { a = true;
"${"b"}" = a; } is equivalent to { a = true; b = true; } but rec {
"${"a"}" = true; b = a; } would be an error or use a from the
surrounding scope if it exists.
Note that the getAttr, getAttr or default, and hasAttr are all
implemented purely in the parser as syntactic sugar, while attribute
sets with dynamic attribute names required changes to the AST to be
implemented cleanly.
This is an alternative solution to and closes #167
Signed-off-by: Shea Levy <shea@shealevy.com>
2013-09-21 06:25:30 +03:00
|
|
|
|
|
|
|
|
|
i->valueExpr->setName(nameSym);
|
|
|
|
|
/* Keep sorted order so find can catch duplicates */
|
2014-09-19 17:49:41 +03:00
|
|
|
|
v.attrs->push_back(Attr(nameSym, i->valueExpr->maybeThunk(state, *dynamicEnv), &i->pos));
|
|
|
|
|
v.attrs->sort(); // FIXME: inefficient
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-13 16:42:25 +03:00
|
|
|
|
void ExprLet::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
/* Create a new environment that contains the attributes in this
|
|
|
|
|
`let'. */
|
2010-10-24 22:52:33 +03:00
|
|
|
|
Env & env2(state.allocEnv(attrs->attrs.size()));
|
2010-04-13 16:42:25 +03:00
|
|
|
|
env2.up = &env;
|
2010-04-14 17:42:32 +03:00
|
|
|
|
|
2010-10-24 22:52:33 +03:00
|
|
|
|
/* The recursive attributes are evaluated in the new environment,
|
|
|
|
|
while the inherited attributes are evaluated in the original
|
2010-04-13 16:42:25 +03:00
|
|
|
|
environment. */
|
2010-10-24 22:52:33 +03:00
|
|
|
|
unsigned int displ = 0;
|
|
|
|
|
foreach (ExprAttrs::AttrDefs::iterator, i, attrs->attrs)
|
2013-07-16 00:10:18 +03:00
|
|
|
|
env2.values[displ++] = i->second.e->maybeThunk(state, i->second.inherited ? env : env2);
|
2010-04-13 16:42:25 +03:00
|
|
|
|
|
2012-02-04 15:50:25 +02:00
|
|
|
|
body->eval(state, env2, v);
|
2010-04-13 16:42:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprList::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
state.mkList(v, elems.size());
|
2010-10-23 21:18:07 +03:00
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n)
|
2012-01-04 23:24:11 +02:00
|
|
|
|
v.list.elems[n] = elems[n]->maybeThunk(state, env);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprVar::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2013-10-08 15:24:53 +03:00
|
|
|
|
Value * v2 = state.lookupVar(&env, *this, false);
|
2010-04-14 18:14:23 +03:00
|
|
|
|
state.forceValue(*v2);
|
|
|
|
|
v = *v2;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-24 17:20:02 +03:00
|
|
|
|
unsigned long nrLookups = 0;
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprSelect::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2011-07-06 15:28:57 +03:00
|
|
|
|
Value vTmp;
|
2014-04-04 23:52:14 +03:00
|
|
|
|
Pos * pos2 = 0;
|
2011-07-06 15:28:57 +03:00
|
|
|
|
Value * vAttrs = &vTmp;
|
|
|
|
|
|
2012-02-04 15:50:25 +02:00
|
|
|
|
e->eval(state, env, vTmp);
|
2011-07-06 15:28:57 +03:00
|
|
|
|
|
|
|
|
|
try {
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2011-07-06 15:28:57 +03:00
|
|
|
|
foreach (AttrPath::const_iterator, i, attrPath) {
|
|
|
|
|
nrLookups++;
|
|
|
|
|
Bindings::iterator j;
|
2014-01-01 01:56:26 +02:00
|
|
|
|
Symbol name = getName(*i, state, env);
|
2011-07-13 15:19:57 +03:00
|
|
|
|
if (def) {
|
|
|
|
|
state.forceValue(*vAttrs);
|
|
|
|
|
if (vAttrs->type != tAttrs ||
|
2014-01-01 01:56:26 +02:00
|
|
|
|
(j = vAttrs->attrs->find(name)) == vAttrs->attrs->end())
|
2011-07-13 15:19:57 +03:00
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
def->eval(state, env, v);
|
2011-07-13 15:19:57 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2014-04-04 23:52:14 +03:00
|
|
|
|
state.forceAttrs(*vAttrs, pos);
|
2014-05-15 18:30:46 +03:00
|
|
|
|
if ((j = vAttrs->attrs->find(name)) == vAttrs->attrs->end()) {
|
|
|
|
|
AttrPath staticPath;
|
|
|
|
|
AttrPath::const_iterator j;
|
|
|
|
|
for (j = attrPath.begin(); j != i; ++j)
|
|
|
|
|
staticPath.push_back(AttrName(getName(*j, state, env)));
|
|
|
|
|
staticPath.push_back(AttrName(getName(*j, state, env)));
|
|
|
|
|
for (j = j + 1; j != attrPath.end(); ++j)
|
|
|
|
|
staticPath.push_back(*j);
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwEvalError("attribute ‘%1%’ missing, at %2%", showAttrPath(staticPath), pos);
|
2014-05-15 18:30:46 +03:00
|
|
|
|
}
|
2011-07-13 15:19:57 +03:00
|
|
|
|
}
|
2011-07-06 15:28:57 +03:00
|
|
|
|
vAttrs = j->value;
|
2014-04-04 23:52:14 +03:00
|
|
|
|
pos2 = j->pos;
|
|
|
|
|
if (state.countCalls && pos2) state.attrSelects[*pos2]++;
|
2011-07-06 15:28:57 +03:00
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2011-07-06 15:28:57 +03:00
|
|
|
|
state.forceValue(*vAttrs);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
} catch (Error & e) {
|
2014-04-04 23:52:14 +03:00
|
|
|
|
if (pos2 && pos2->file != state.sDerivationNix)
|
2014-08-20 18:00:17 +03:00
|
|
|
|
addErrorPrefix(e, "while evaluating the attribute ‘%1%’ at %2%:\n",
|
2014-04-04 23:52:14 +03:00
|
|
|
|
showAttrPath(attrPath), *pos2);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2011-07-06 15:28:57 +03:00
|
|
|
|
v = *vAttrs;
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-13 00:21:24 +03:00
|
|
|
|
void ExprOpHasAttr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2011-07-06 13:58:17 +03:00
|
|
|
|
Value vTmp;
|
|
|
|
|
Value * vAttrs = &vTmp;
|
|
|
|
|
|
2012-02-04 15:50:25 +02:00
|
|
|
|
e->eval(state, env, vTmp);
|
2011-07-06 13:58:17 +03:00
|
|
|
|
|
|
|
|
|
foreach (AttrPath::const_iterator, i, attrPath) {
|
|
|
|
|
state.forceValue(*vAttrs);
|
|
|
|
|
Bindings::iterator j;
|
2014-01-01 01:56:26 +02:00
|
|
|
|
Symbol name = getName(*i, state, env);
|
2011-07-06 13:58:17 +03:00
|
|
|
|
if (vAttrs->type != tAttrs ||
|
2014-01-01 01:56:26 +02:00
|
|
|
|
(j = vAttrs->attrs->find(name)) == vAttrs->attrs->end())
|
2011-07-06 13:58:17 +03:00
|
|
|
|
{
|
|
|
|
|
mkBool(v, false);
|
|
|
|
|
return;
|
|
|
|
|
} else {
|
|
|
|
|
vAttrs = j->value;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2011-07-06 13:58:17 +03:00
|
|
|
|
mkBool(v, true);
|
2010-04-13 00:21:24 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprLambda::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
v.type = tLambda;
|
|
|
|
|
v.lambda.env = &env;
|
|
|
|
|
v.lambda.fun = this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprApp::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-02-27 22:47:59 +02:00
|
|
|
|
/* FIXME: vFun prevents GCC from doing tail call optimisation. */
|
|
|
|
|
Value vFun;
|
|
|
|
|
e1->eval(state, env, vFun);
|
2014-04-04 18:53:52 +03:00
|
|
|
|
state.callFunction(vFun, *(e2->maybeThunk(state, env)), v, pos);
|
2013-11-07 19:04:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 19:51:01 +03:00
|
|
|
|
void EvalState::callPrimOp(Value & fun, Value & arg, Value & v, const Pos & pos)
|
2013-11-07 19:04:36 +02:00
|
|
|
|
{
|
|
|
|
|
/* Figure out the number of arguments still needed. */
|
|
|
|
|
unsigned int argsDone = 0;
|
|
|
|
|
Value * primOp = &fun;
|
|
|
|
|
while (primOp->type == tPrimOpApp) {
|
|
|
|
|
argsDone++;
|
|
|
|
|
primOp = primOp->primOpApp.left;
|
|
|
|
|
}
|
|
|
|
|
assert(primOp->type == tPrimOp);
|
|
|
|
|
unsigned int arity = primOp->primOp->arity;
|
|
|
|
|
unsigned int argsLeft = arity - argsDone;
|
|
|
|
|
|
|
|
|
|
if (argsLeft == 1) {
|
|
|
|
|
/* We have all the arguments, so call the primop. */
|
|
|
|
|
|
|
|
|
|
/* Put all the arguments in an array. */
|
|
|
|
|
Value * vArgs[arity];
|
|
|
|
|
unsigned int n = arity - 1;
|
|
|
|
|
vArgs[n--] = &arg;
|
|
|
|
|
for (Value * arg = &fun; arg->type == tPrimOpApp; arg = arg->primOpApp.left)
|
|
|
|
|
vArgs[n--] = arg->primOpApp.right;
|
|
|
|
|
|
|
|
|
|
/* And call the primop. */
|
|
|
|
|
nrPrimOpCalls++;
|
|
|
|
|
if (countCalls) primOpCalls[primOp->primOp->name]++;
|
2014-04-04 19:51:01 +03:00
|
|
|
|
primOp->primOp->fun(*this, pos, vArgs, v);
|
2013-11-07 19:04:36 +02:00
|
|
|
|
} else {
|
|
|
|
|
Value * fun2 = allocValue();
|
|
|
|
|
*fun2 = fun;
|
|
|
|
|
v.type = tPrimOpApp;
|
|
|
|
|
v.primOpApp.left = fun2;
|
|
|
|
|
v.primOpApp.right = &arg;
|
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 18:53:52 +03:00
|
|
|
|
void EvalState::callFunction(Value & fun, Value & arg, Value & v, const Pos & pos)
|
2010-03-30 16:47:59 +03:00
|
|
|
|
{
|
|
|
|
|
if (fun.type == tPrimOp || fun.type == tPrimOpApp) {
|
2014-04-04 19:51:01 +03:00
|
|
|
|
callPrimOp(fun, arg, v, pos);
|
2010-03-30 16:47:59 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-03-30 16:47:59 +03:00
|
|
|
|
if (fun.type != tLambda)
|
2014-04-04 18:53:52 +03:00
|
|
|
|
throwTypeError("attempt to call something which is not a function but %1%, at %2%", fun, pos);
|
2010-03-30 16:47:59 +03:00
|
|
|
|
|
2013-11-12 13:51:59 +02:00
|
|
|
|
ExprLambda & lambda(*fun.lambda.fun);
|
|
|
|
|
|
2010-04-14 17:42:32 +03:00
|
|
|
|
unsigned int size =
|
2013-11-12 13:51:59 +02:00
|
|
|
|
(lambda.arg.empty() ? 0 : 1) +
|
|
|
|
|
(lambda.matchAttrs ? lambda.formals->formals.size() : 0);
|
2010-04-14 17:42:32 +03:00
|
|
|
|
Env & env2(allocEnv(size));
|
2010-03-30 16:47:59 +03:00
|
|
|
|
env2.up = fun.lambda.env;
|
|
|
|
|
|
2010-04-14 17:42:32 +03:00
|
|
|
|
unsigned int displ = 0;
|
|
|
|
|
|
2013-11-12 13:51:59 +02:00
|
|
|
|
if (!lambda.matchAttrs)
|
2010-10-22 18:51:52 +03:00
|
|
|
|
env2.values[displ++] = &arg;
|
2010-03-30 16:47:59 +03:00
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
else {
|
2014-04-04 20:11:40 +03:00
|
|
|
|
forceAttrs(arg, pos);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2013-11-12 13:51:59 +02:00
|
|
|
|
if (!lambda.arg.empty())
|
2010-10-22 18:51:52 +03:00
|
|
|
|
env2.values[displ++] = &arg;
|
2010-03-30 16:47:59 +03:00
|
|
|
|
|
|
|
|
|
/* For each formal argument, get the actual argument. If
|
|
|
|
|
there is no matching actual argument but the formal
|
|
|
|
|
argument has a default, use the default. */
|
|
|
|
|
unsigned int attrsUsed = 0;
|
2013-11-12 13:51:59 +02:00
|
|
|
|
foreach (Formals::Formals_::iterator, i, lambda.formals->formals) {
|
2010-04-12 21:30:11 +03:00
|
|
|
|
Bindings::iterator j = arg.attrs->find(i->name);
|
2010-03-30 16:47:59 +03:00
|
|
|
|
if (j == arg.attrs->end()) {
|
2014-08-20 18:00:17 +03:00
|
|
|
|
if (!i->def) throwTypeError("%1% called without required argument ‘%2%’, at %3%",
|
2014-04-04 18:53:52 +03:00
|
|
|
|
lambda, i->name, pos);
|
2012-01-04 23:24:11 +02:00
|
|
|
|
env2.values[displ++] = i->def->maybeThunk(*this, env2);
|
2010-03-30 16:47:59 +03:00
|
|
|
|
} else {
|
|
|
|
|
attrsUsed++;
|
2010-10-24 03:41:29 +03:00
|
|
|
|
env2.values[displ++] = j->value;
|
2010-03-30 16:47:59 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check that each actual argument is listed as a formal
|
2013-05-16 18:56:14 +03:00
|
|
|
|
argument (unless the attribute match specifies a `...'). */
|
2013-11-12 13:51:59 +02:00
|
|
|
|
if (!lambda.formals->ellipsis && attrsUsed != arg.attrs->size()) {
|
2013-05-16 18:56:14 +03:00
|
|
|
|
/* Nope, so show the first unexpected argument to the
|
|
|
|
|
user. */
|
|
|
|
|
foreach (Bindings::iterator, i, *arg.attrs)
|
2013-11-12 13:51:59 +02:00
|
|
|
|
if (lambda.formals->argNames.find(i->name) == lambda.formals->argNames.end())
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwTypeError("%1% called with unexpected argument ‘%2%’, at %3%", lambda, i->name, pos);
|
2013-05-16 18:56:14 +03:00
|
|
|
|
abort(); // can't happen
|
|
|
|
|
}
|
2010-03-30 16:47:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
2012-08-13 06:41:48 +03:00
|
|
|
|
nrFunctionCalls++;
|
2013-11-12 13:51:59 +02:00
|
|
|
|
if (countCalls) incrFunctionCall(&lambda);
|
|
|
|
|
|
|
|
|
|
/* Evaluate the body. This is conditional on showTrace, because
|
|
|
|
|
catching exceptions makes this function not tail-recursive. */
|
|
|
|
|
if (settings.showTrace)
|
|
|
|
|
try {
|
|
|
|
|
lambda.body->eval(*this, env2, v);
|
|
|
|
|
} catch (Error & e) {
|
2014-04-04 18:53:52 +03:00
|
|
|
|
addErrorPrefix(e, "while evaluating %1%, called from %2%:\n", lambda, pos);
|
2013-11-12 13:51:59 +02:00
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
else
|
2012-02-04 15:50:25 +02:00
|
|
|
|
fun.lambda.fun->body->eval(*this, env2, v);
|
2013-11-07 19:04:36 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Lifted out of callFunction() because it creates a temporary that
|
|
|
|
|
// prevents tail-call optimisation.
|
|
|
|
|
void EvalState::incrFunctionCall(ExprLambda * fun)
|
|
|
|
|
{
|
|
|
|
|
functionCalls[fun]++;
|
2010-03-30 16:47:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-22 17:47:42 +03:00
|
|
|
|
void EvalState::autoCallFunction(Bindings & args, Value & fun, Value & res)
|
2010-04-07 18:47:06 +03:00
|
|
|
|
{
|
|
|
|
|
forceValue(fun);
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
if (fun.type != tLambda || !fun.lambda.fun->matchAttrs) {
|
2010-04-07 18:47:06 +03:00
|
|
|
|
res = fun;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-09 16:09:31 +02:00
|
|
|
|
Value * actualArgs = allocValue();
|
|
|
|
|
mkAttrs(*actualArgs, fun.lambda.fun->formals->formals.size());
|
2010-04-12 21:30:11 +03:00
|
|
|
|
|
|
|
|
|
foreach (Formals::Formals_::iterator, i, fun.lambda.fun->formals->formals) {
|
2010-10-22 17:47:42 +03:00
|
|
|
|
Bindings::iterator j = args.find(i->name);
|
2010-04-07 18:47:06 +03:00
|
|
|
|
if (j != args.end())
|
2012-11-09 16:09:31 +02:00
|
|
|
|
actualArgs->attrs->push_back(*j);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
else if (!i->def)
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwTypeError("cannot auto-call a function that has an argument without a default value (‘%1%’)", i->name);
|
2010-04-07 18:47:06 +03:00
|
|
|
|
}
|
|
|
|
|
|
2012-11-09 16:09:31 +02:00
|
|
|
|
actualArgs->attrs->sort();
|
2010-10-24 22:52:33 +03:00
|
|
|
|
|
2014-04-04 18:53:52 +03:00
|
|
|
|
callFunction(fun, *actualArgs, res, noPos);
|
2010-04-07 18:47:06 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprWith::eval(EvalState & state, Env & env, Value & v)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
2010-04-14 18:01:04 +03:00
|
|
|
|
Env & env2(state.allocEnv(1));
|
2010-04-12 21:30:11 +03:00
|
|
|
|
env2.up = &env;
|
2010-04-22 18:08:09 +03:00
|
|
|
|
env2.prevWith = prevWith;
|
2013-07-31 13:44:21 +03:00
|
|
|
|
env2.haveWithAttrs = false;
|
|
|
|
|
env2.values[0] = (Value *) attrs;
|
2010-04-14 18:01:04 +03:00
|
|
|
|
|
2012-02-04 15:50:25 +02:00
|
|
|
|
body->eval(state, env2, v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprIf::eval(EvalState & state, Env & env, Value & v)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
2014-03-05 17:18:13 +02:00
|
|
|
|
(state.evalBool(env, cond) ? then : else_)->eval(state, env, v);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-04-13 00:21:24 +03:00
|
|
|
|
void ExprAssert::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 23:43:52 +03:00
|
|
|
|
if (!state.evalBool(env, cond, pos))
|
2010-04-13 00:21:24 +03:00
|
|
|
|
throwAssertionError("assertion failed at %1%", pos);
|
2012-02-04 15:50:25 +02:00
|
|
|
|
body->eval(state, env, v);
|
2010-04-13 00:21:24 +03:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-04-13 00:21:24 +03:00
|
|
|
|
void ExprOpNot::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
mkBool(v, !state.evalBool(env, e));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
|
void ExprOpEq::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
mkBool(v, state.eqValues(v1, v2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpNEq::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2010-04-12 21:30:11 +03:00
|
|
|
|
mkBool(v, !state.eqValues(v1, v2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpAnd::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 23:43:52 +03:00
|
|
|
|
mkBool(v, state.evalBool(env, e1, pos) && state.evalBool(env, e2, pos));
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpOr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 23:43:52 +03:00
|
|
|
|
mkBool(v, state.evalBool(env, e1, pos) || state.evalBool(env, e2, pos));
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpImpl::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 23:43:52 +03:00
|
|
|
|
mkBool(v, !state.evalBool(env, e1, pos) || state.evalBool(env, e2, pos));
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpUpdate::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2010-08-02 19:31:05 +03:00
|
|
|
|
Value v1, v2;
|
|
|
|
|
state.evalAttrs(env, e1, v1);
|
2010-04-16 18:13:47 +03:00
|
|
|
|
state.evalAttrs(env, e2, v2);
|
2010-08-02 19:31:05 +03:00
|
|
|
|
|
2010-10-20 18:48:00 +03:00
|
|
|
|
state.nrOpUpdates++;
|
|
|
|
|
|
2010-08-02 19:31:05 +03:00
|
|
|
|
if (v1.attrs->size() == 0) { v = v2; return; }
|
|
|
|
|
if (v2.attrs->size() == 0) { v = v1; return; }
|
|
|
|
|
|
2010-10-24 23:09:37 +03:00
|
|
|
|
state.mkAttrs(v, v1.attrs->size() + v2.attrs->size());
|
2010-10-24 22:52:33 +03:00
|
|
|
|
|
2013-10-24 17:41:04 +03:00
|
|
|
|
/* Merge the sets, preferring values from the second set. Make
|
|
|
|
|
sure to keep the resulting vector in sorted order. */
|
2010-10-24 22:52:33 +03:00
|
|
|
|
Bindings::iterator i = v1.attrs->begin();
|
|
|
|
|
Bindings::iterator j = v2.attrs->begin();
|
2010-08-02 19:31:05 +03:00
|
|
|
|
|
2010-10-24 22:52:33 +03:00
|
|
|
|
while (i != v1.attrs->end() && j != v2.attrs->end()) {
|
|
|
|
|
if (i->name == j->name) {
|
|
|
|
|
v.attrs->push_back(*j);
|
|
|
|
|
++i; ++j;
|
|
|
|
|
}
|
|
|
|
|
else if (i->name < j->name)
|
|
|
|
|
v.attrs->push_back(*i++);
|
|
|
|
|
else
|
|
|
|
|
v.attrs->push_back(*j++);
|
|
|
|
|
}
|
2010-10-20 18:48:00 +03:00
|
|
|
|
|
2010-10-24 22:52:33 +03:00
|
|
|
|
while (i != v1.attrs->end()) v.attrs->push_back(*i++);
|
|
|
|
|
while (j != v2.attrs->end()) v.attrs->push_back(*j++);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-10-20 18:48:00 +03:00
|
|
|
|
state.nrOpUpdateValuesCopied += v.attrs->size();
|
2010-04-12 21:30:11 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpConcatLists::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 15:50:25 +02:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2012-08-13 08:53:10 +03:00
|
|
|
|
Value * lists[2] = { &v1, &v2 };
|
2014-04-04 23:43:52 +03:00
|
|
|
|
state.concatLists(v, 2, lists, pos);
|
2012-08-13 08:53:10 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 23:43:52 +03:00
|
|
|
|
void EvalState::concatLists(Value & v, unsigned int nrLists, Value * * lists, const Pos & pos)
|
2012-08-13 08:53:10 +03:00
|
|
|
|
{
|
|
|
|
|
nrListConcats++;
|
2012-08-13 21:58:54 +03:00
|
|
|
|
|
|
|
|
|
Value * nonEmpty = 0;
|
2012-08-13 08:53:10 +03:00
|
|
|
|
unsigned int len = 0;
|
|
|
|
|
for (unsigned int n = 0; n < nrLists; ++n) {
|
2014-04-04 23:43:52 +03:00
|
|
|
|
forceList(*lists[n], pos);
|
2012-08-13 21:58:54 +03:00
|
|
|
|
unsigned int l = lists[n]->list.length;
|
|
|
|
|
len += l;
|
|
|
|
|
if (l) nonEmpty = lists[n];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (nonEmpty && len == nonEmpty->list.length) {
|
|
|
|
|
v = *nonEmpty;
|
|
|
|
|
return;
|
2012-08-13 08:53:10 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mkList(v, len);
|
|
|
|
|
for (unsigned int n = 0, pos = 0; n < nrLists; ++n) {
|
|
|
|
|
unsigned int l = lists[n]->list.length;
|
|
|
|
|
memcpy(v.list.elems + pos, lists[n]->list.elems, l * sizeof(Value *));
|
|
|
|
|
pos += l;
|
|
|
|
|
}
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-13 00:21:24 +03:00
|
|
|
|
void ExprConcatStrings::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
PathSet context;
|
|
|
|
|
std::ostringstream s;
|
2013-08-19 13:35:03 +03:00
|
|
|
|
NixInt n = 0;
|
2013-08-02 18:21:17 +03:00
|
|
|
|
|
2013-10-17 01:39:59 +03:00
|
|
|
|
bool first = !forceString;
|
|
|
|
|
ValueType firstType = tString;
|
2010-04-13 00:21:24 +03:00
|
|
|
|
|
|
|
|
|
foreach (vector<Expr *>::iterator, i, *es) {
|
2013-08-02 18:21:17 +03:00
|
|
|
|
Value vTmp;
|
|
|
|
|
(*i)->eval(state, env, vTmp);
|
2010-04-13 00:21:24 +03:00
|
|
|
|
|
|
|
|
|
/* If the first element is a path, then the result will also
|
|
|
|
|
be a path, we don't copy anything (yet - that's done later,
|
|
|
|
|
since paths are copied when they are used in a derivation),
|
|
|
|
|
and none of the strings are allowed to have contexts. */
|
|
|
|
|
if (first) {
|
2013-08-02 18:21:17 +03:00
|
|
|
|
firstType = vTmp.type;
|
2010-04-13 00:21:24 +03:00
|
|
|
|
first = false;
|
|
|
|
|
}
|
2012-08-13 22:10:29 +03:00
|
|
|
|
|
2013-10-17 01:39:59 +03:00
|
|
|
|
if (firstType == tInt) {
|
2013-08-02 18:21:17 +03:00
|
|
|
|
if (vTmp.type != tInt)
|
2014-04-04 23:19:33 +03:00
|
|
|
|
throwEvalError("cannot add %1% to an integer, at %2%", showType(vTmp), pos);
|
2013-08-02 18:21:17 +03:00
|
|
|
|
n += vTmp.integer;
|
|
|
|
|
} else
|
2014-04-04 23:19:33 +03:00
|
|
|
|
s << state.coerceToString(pos, vTmp, context, false, firstType == tString);
|
2010-04-13 00:21:24 +03:00
|
|
|
|
}
|
|
|
|
|
|
2013-08-02 18:21:17 +03:00
|
|
|
|
if (firstType == tInt)
|
|
|
|
|
mkInt(v, n);
|
|
|
|
|
else if (firstType == tPath) {
|
|
|
|
|
if (!context.empty())
|
2014-04-04 23:19:33 +03:00
|
|
|
|
throwEvalError("a string that refers to a store path cannot be appended to a path, at %1%", pos);
|
2010-04-13 00:21:24 +03:00
|
|
|
|
mkPath(v, s.str().c_str());
|
2013-08-02 18:21:17 +03:00
|
|
|
|
} else
|
2010-04-13 00:21:24 +03:00
|
|
|
|
mkString(v, s.str(), context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-18 21:14:54 +02:00
|
|
|
|
void ExprPos::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2013-11-18 23:22:35 +02:00
|
|
|
|
state.mkPos(v, &pos);
|
2013-11-18 21:14:54 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-09-22 16:03:59 +03:00
|
|
|
|
void EvalState::forceValueDeep(Value & v)
|
2010-04-07 16:55:46 +03:00
|
|
|
|
{
|
2014-09-22 16:16:09 +03:00
|
|
|
|
std::set<const Value *> seen;
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2014-09-22 16:16:09 +03:00
|
|
|
|
std::function<void(Value & v)> recurse;
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2014-09-22 16:16:09 +03:00
|
|
|
|
recurse = [&](Value & v) {
|
|
|
|
|
if (seen.find(&v) != seen.end()) return;
|
|
|
|
|
seen.insert(&v);
|
|
|
|
|
|
|
|
|
|
forceValue(v);
|
|
|
|
|
|
|
|
|
|
if (v.type == tAttrs) {
|
|
|
|
|
foreach (Bindings::iterator, i, *v.attrs)
|
|
|
|
|
recurse(*i->value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (v.type == tList) {
|
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n)
|
|
|
|
|
recurse(*v.list.elems[n]);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
recurse(v);
|
2010-04-07 16:55:46 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 19:58:15 +03:00
|
|
|
|
NixInt EvalState::forceInt(Value & v, const Pos & pos)
|
2010-03-29 17:37:56 +03:00
|
|
|
|
{
|
|
|
|
|
forceValue(v);
|
|
|
|
|
if (v.type != tInt)
|
2014-04-04 19:58:15 +03:00
|
|
|
|
throwTypeError("value is %1% while an integer was expected, at %2%", v, pos);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
return v.integer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-31 18:38:03 +03:00
|
|
|
|
bool EvalState::forceBool(Value & v)
|
|
|
|
|
{
|
|
|
|
|
forceValue(v);
|
|
|
|
|
if (v.type != tBool)
|
2013-11-07 14:44:14 +02:00
|
|
|
|
throwTypeError("value is %1% while a Boolean was expected", v);
|
2010-03-31 18:38:03 +03:00
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 20:05:36 +03:00
|
|
|
|
void EvalState::forceFunction(Value & v, const Pos & pos)
|
2010-03-30 16:47:59 +03:00
|
|
|
|
{
|
|
|
|
|
forceValue(v);
|
|
|
|
|
if (v.type != tLambda && v.type != tPrimOp && v.type != tPrimOpApp)
|
2014-04-04 20:05:36 +03:00
|
|
|
|
throwTypeError("value is %1% while a function was expected, at %2%", v, pos);
|
2010-03-30 16:47:59 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 22:14:11 +03:00
|
|
|
|
string EvalState::forceString(Value & v, const Pos & pos)
|
2010-03-30 21:05:54 +03:00
|
|
|
|
{
|
|
|
|
|
forceValue(v);
|
2014-04-04 22:14:11 +03:00
|
|
|
|
if (v.type != tString) {
|
|
|
|
|
if (pos)
|
|
|
|
|
throwTypeError("value is %1% while a string was expected, at %2%", v, pos);
|
|
|
|
|
else
|
|
|
|
|
throwTypeError("value is %1% while a string was expected", v);
|
|
|
|
|
}
|
2010-03-31 18:38:03 +03:00
|
|
|
|
return string(v.string.s);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-06-10 13:29:50 +03:00
|
|
|
|
void copyContext(const Value & v, PathSet & context)
|
2010-03-31 22:52:29 +03:00
|
|
|
|
{
|
2010-04-09 15:00:49 +03:00
|
|
|
|
if (v.string.context)
|
2013-09-02 17:29:15 +03:00
|
|
|
|
for (const char * * p = v.string.context; *p; ++p)
|
2010-03-31 22:52:29 +03:00
|
|
|
|
context.insert(*p);
|
2010-06-10 13:29:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string EvalState::forceString(Value & v, PathSet & context)
|
|
|
|
|
{
|
|
|
|
|
string s = forceString(v);
|
|
|
|
|
copyContext(v, context);
|
2010-03-31 22:52:29 +03:00
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 22:14:11 +03:00
|
|
|
|
string EvalState::forceStringNoCtx(Value & v, const Pos & pos)
|
2010-03-31 18:38:03 +03:00
|
|
|
|
{
|
2014-04-04 22:14:11 +03:00
|
|
|
|
string s = forceString(v, pos);
|
|
|
|
|
if (v.string.context) {
|
|
|
|
|
if (pos)
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwEvalError("the string ‘%1%’ is not allowed to refer to a store path (such as ‘%2%’), at %3%",
|
2014-04-04 22:14:11 +03:00
|
|
|
|
v.string.s, v.string.context[0], pos);
|
|
|
|
|
else
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwEvalError("the string ‘%1%’ is not allowed to refer to a store path (such as ‘%2%’)",
|
2014-04-04 22:14:11 +03:00
|
|
|
|
v.string.s, v.string.context[0]);
|
|
|
|
|
}
|
2010-03-31 18:38:03 +03:00
|
|
|
|
return s;
|
2010-03-30 21:05:54 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-07 16:55:46 +03:00
|
|
|
|
bool EvalState::isDerivation(Value & v)
|
|
|
|
|
{
|
|
|
|
|
if (v.type != tAttrs) return false;
|
2010-04-13 15:25:42 +03:00
|
|
|
|
Bindings::iterator i = v.attrs->find(sType);
|
2011-10-27 22:06:23 +03:00
|
|
|
|
if (i == v.attrs->end()) return false;
|
|
|
|
|
forceValue(*i->value);
|
|
|
|
|
if (i->value->type != tString) return false;
|
2013-10-24 04:08:34 +03:00
|
|
|
|
return strcmp(i->value->string.s, "derivation") == 0;
|
2010-04-07 16:55:46 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
string EvalState::coerceToString(const Pos & pos, Value & v, PathSet & context,
|
2010-03-30 12:22:33 +03:00
|
|
|
|
bool coerceMore, bool copyToStore)
|
|
|
|
|
{
|
|
|
|
|
forceValue(v);
|
|
|
|
|
|
|
|
|
|
string s;
|
|
|
|
|
|
2010-03-31 22:52:29 +03:00
|
|
|
|
if (v.type == tString) {
|
2010-06-10 13:29:50 +03:00
|
|
|
|
copyContext(v, context);
|
2010-03-31 22:52:29 +03:00
|
|
|
|
return v.string.s;
|
|
|
|
|
}
|
2010-03-30 12:22:33 +03:00
|
|
|
|
|
|
|
|
|
if (v.type == tPath) {
|
|
|
|
|
Path path(canonPath(v.path));
|
2013-11-19 01:03:11 +02:00
|
|
|
|
return copyToStore ? copyPathToStore(context, path) : path;
|
2010-03-30 12:22:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (v.type == tAttrs) {
|
2010-04-13 15:25:42 +03:00
|
|
|
|
Bindings::iterator i = v.attrs->find(sOutPath);
|
2014-04-04 23:19:33 +03:00
|
|
|
|
if (i == v.attrs->end()) throwTypeError("cannot coerce a set to a string, at %1%", pos);
|
|
|
|
|
return coerceToString(pos, *i->value, context, coerceMore, copyToStore);
|
2010-03-30 12:22:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (coerceMore) {
|
|
|
|
|
|
|
|
|
|
/* Note that `false' is represented as an empty string for
|
|
|
|
|
shell scripting convenience, just like `null'. */
|
|
|
|
|
if (v.type == tBool && v.boolean) return "1";
|
|
|
|
|
if (v.type == tBool && !v.boolean) return "";
|
|
|
|
|
if (v.type == tInt) return int2String(v.integer);
|
|
|
|
|
if (v.type == tNull) return "";
|
|
|
|
|
|
|
|
|
|
if (v.type == tList) {
|
|
|
|
|
string result;
|
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n) {
|
2014-04-04 23:19:33 +03:00
|
|
|
|
result += coerceToString(pos, *v.list.elems[n],
|
2010-03-30 12:22:33 +03:00
|
|
|
|
context, coerceMore, copyToStore);
|
2010-04-01 17:35:03 +03:00
|
|
|
|
if (n < v.list.length - 1
|
|
|
|
|
/* !!! not quite correct */
|
2010-04-15 03:37:36 +03:00
|
|
|
|
&& (v.list.elems[n]->type != tList || v.list.elems[n]->list.length != 0))
|
2010-04-01 17:35:03 +03:00
|
|
|
|
result += " ";
|
2010-03-30 12:22:33 +03:00
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
throwTypeError("cannot coerce %1% to a string, at %2%", v, pos);
|
2010-03-30 12:22:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-19 01:03:11 +02:00
|
|
|
|
string EvalState::copyPathToStore(PathSet & context, const Path & path)
|
|
|
|
|
{
|
|
|
|
|
if (nix::isDerivation(path))
|
2014-08-20 18:00:17 +03:00
|
|
|
|
throwEvalError("file names are not allowed to end in ‘%1%’", drvExtension);
|
2013-11-19 01:03:11 +02:00
|
|
|
|
|
|
|
|
|
Path dstPath;
|
|
|
|
|
if (srcToStore[path] != "")
|
|
|
|
|
dstPath = srcToStore[path];
|
|
|
|
|
else {
|
|
|
|
|
dstPath = settings.readOnlyMode
|
|
|
|
|
? computeStorePathForPath(path).first
|
|
|
|
|
: store->addToStore(path, true, htSHA256, defaultPathFilter, repair);
|
|
|
|
|
srcToStore[path] = dstPath;
|
2014-08-20 18:00:17 +03:00
|
|
|
|
printMsg(lvlChatty, format("copied source ‘%1%’ -> ‘%2%’")
|
2013-11-19 01:03:11 +02:00
|
|
|
|
% path % dstPath);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.insert(dstPath);
|
|
|
|
|
return dstPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 23:19:33 +03:00
|
|
|
|
Path EvalState::coerceToPath(const Pos & pos, Value & v, PathSet & context)
|
2010-03-30 12:22:33 +03:00
|
|
|
|
{
|
2014-04-04 23:19:33 +03:00
|
|
|
|
string path = coerceToString(pos, v, context, false, false);
|
2010-03-30 12:22:33 +03:00
|
|
|
|
if (path == "" || path[0] != '/')
|
2014-09-02 23:53:01 +03:00
|
|
|
|
throwEvalError("string ‘%1%’ doesn't represent an absolute path, at %2%", path, pos);
|
2010-03-30 12:22:33 +03:00
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
bool EvalState::eqValues(Value & v1, Value & v2)
|
|
|
|
|
{
|
|
|
|
|
forceValue(v1);
|
|
|
|
|
forceValue(v2);
|
|
|
|
|
|
2010-04-12 12:50:20 +03:00
|
|
|
|
/* !!! Hack to support some old broken code that relies on pointer
|
2013-10-24 17:41:04 +03:00
|
|
|
|
equality tests between sets. (Specifically, builderDefs calls
|
|
|
|
|
uniqList on a list of sets.) Will remove this eventually. */
|
2010-04-12 12:50:20 +03:00
|
|
|
|
if (&v1 == &v2) return true;
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
if (v1.type != v2.type) return false;
|
2010-04-12 12:50:20 +03:00
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
switch (v1.type) {
|
|
|
|
|
|
|
|
|
|
case tInt:
|
|
|
|
|
return v1.integer == v2.integer;
|
|
|
|
|
|
|
|
|
|
case tBool:
|
|
|
|
|
return v1.boolean == v2.boolean;
|
|
|
|
|
|
2014-06-10 15:02:56 +03:00
|
|
|
|
case tString:
|
|
|
|
|
return strcmp(v1.string.s, v2.string.s) == 0;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
2010-03-31 23:09:20 +03:00
|
|
|
|
case tPath:
|
|
|
|
|
return strcmp(v1.path, v2.path) == 0;
|
|
|
|
|
|
2010-03-31 12:54:12 +03:00
|
|
|
|
case tNull:
|
|
|
|
|
return true;
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
case tList:
|
2010-04-16 16:51:01 +03:00
|
|
|
|
if (v1.list.length != v2.list.length) return false;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
for (unsigned int n = 0; n < v1.list.length; ++n)
|
2010-04-15 03:37:36 +03:00
|
|
|
|
if (!eqValues(*v1.list.elems[n], *v2.list.elems[n])) return false;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
case tAttrs: {
|
2013-10-24 17:41:04 +03:00
|
|
|
|
/* If both sets denote a derivation (type = "derivation"),
|
|
|
|
|
then compare their outPaths. */
|
2012-01-20 01:08:47 +02:00
|
|
|
|
if (isDerivation(v1) && isDerivation(v2)) {
|
|
|
|
|
Bindings::iterator i = v1.attrs->find(sOutPath);
|
|
|
|
|
Bindings::iterator j = v2.attrs->find(sOutPath);
|
|
|
|
|
if (i != v1.attrs->end() && j != v2.attrs->end())
|
|
|
|
|
return eqValues(*i->value, *j->value);
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-16 16:51:01 +03:00
|
|
|
|
if (v1.attrs->size() != v2.attrs->size()) return false;
|
2012-01-20 01:08:47 +02:00
|
|
|
|
|
|
|
|
|
/* Otherwise, compare the attributes one by one. */
|
|
|
|
|
Bindings::iterator i, j;
|
|
|
|
|
for (i = v1.attrs->begin(), j = v2.attrs->begin(); i != v1.attrs->end(); ++i, ++j)
|
2010-10-24 03:41:29 +03:00
|
|
|
|
if (i->name != j->name || !eqValues(*i->value, *j->value))
|
2010-05-07 15:11:05 +03:00
|
|
|
|
return false;
|
2013-09-02 17:29:15 +03:00
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-01 13:55:36 +03:00
|
|
|
|
/* Functions are incomparable. */
|
|
|
|
|
case tLambda:
|
|
|
|
|
case tPrimOp:
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
return false;
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
default:
|
2010-04-09 15:00:49 +03:00
|
|
|
|
throwEvalError("cannot compare %1% with %2%", showType(v1), showType(v2));
|
2010-03-29 17:37:56 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-30 18:18:20 +03:00
|
|
|
|
void EvalState::printStats()
|
2003-10-31 19:09:31 +02:00
|
|
|
|
{
|
2006-05-08 13:00:37 +03:00
|
|
|
|
bool showStats = getEnv("NIX_SHOW_STATS", "0") != "0";
|
2010-04-09 15:00:49 +03:00
|
|
|
|
Verbosity v = showStats ? lvlInfo : lvlDebug;
|
|
|
|
|
printMsg(v, "evaluation statistics:");
|
2012-02-04 15:27:11 +02:00
|
|
|
|
|
|
|
|
|
struct rusage buf;
|
|
|
|
|
getrusage(RUSAGE_SELF, &buf);
|
|
|
|
|
float cpuTime = buf.ru_utime.tv_sec + ((float) buf.ru_utime.tv_usec / 1000000);
|
|
|
|
|
|
|
|
|
|
printMsg(v, format(" time elapsed: %1%") % cpuTime);
|
2010-10-20 14:38:30 +03:00
|
|
|
|
printMsg(v, format(" size of a value: %1%") % sizeof(Value));
|
2010-04-15 02:48:46 +03:00
|
|
|
|
printMsg(v, format(" environments allocated: %1% (%2% bytes)")
|
2010-10-24 01:58:24 +03:00
|
|
|
|
% nrEnvs % (nrEnvs * sizeof(Env) + nrValuesInEnvs * sizeof(Value *)));
|
2010-04-15 03:37:36 +03:00
|
|
|
|
printMsg(v, format(" list elements: %1% (%2% bytes)")
|
|
|
|
|
% nrListElems % (nrListElems * sizeof(Value *)));
|
2012-08-13 06:41:48 +03:00
|
|
|
|
printMsg(v, format(" list concatenations: %1%") % nrListConcats);
|
2010-10-24 01:58:24 +03:00
|
|
|
|
printMsg(v, format(" values allocated: %1% (%2% bytes)")
|
2010-04-15 02:48:46 +03:00
|
|
|
|
% nrValues % (nrValues * sizeof(Value)));
|
2014-09-19 17:49:41 +03:00
|
|
|
|
printMsg(v, format(" sets allocated: %1% (%2% bytes)")
|
|
|
|
|
% nrAttrsets % (nrAttrsets * sizeof(Bindings) + nrAttrsInAttrsets * sizeof(Attr)));
|
2010-10-20 18:48:00 +03:00
|
|
|
|
printMsg(v, format(" right-biased unions: %1%") % nrOpUpdates);
|
|
|
|
|
printMsg(v, format(" values copied in right-biased unions: %1%") % nrOpUpdateValuesCopied);
|
2010-04-13 17:34:11 +03:00
|
|
|
|
printMsg(v, format(" symbols in symbol table: %1%") % symbols.size());
|
2013-10-08 16:34:57 +03:00
|
|
|
|
printMsg(v, format(" size of symbol table: %1%") % symbols.totalSize());
|
2010-10-24 17:20:02 +03:00
|
|
|
|
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);
|
2012-08-13 06:41:48 +03:00
|
|
|
|
printMsg(v, format(" number of primop calls: %1%") % nrPrimOpCalls);
|
|
|
|
|
printMsg(v, format(" number of function calls: %1%") % nrFunctionCalls);
|
2012-08-13 06:29:28 +03:00
|
|
|
|
|
|
|
|
|
if (countCalls) {
|
2013-10-24 03:20:54 +03:00
|
|
|
|
v = lvlInfo;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
|
|
|
|
|
printMsg(v, format("calls to %1% primops:") % primOpCalls.size());
|
|
|
|
|
typedef std::multimap<unsigned int, Symbol> PrimOpCalls_;
|
2013-08-02 21:29:23 +03:00
|
|
|
|
PrimOpCalls_ primOpCalls_;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
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());
|
2013-08-02 21:29:23 +03:00
|
|
|
|
typedef std::multimap<unsigned int, ExprLambda *> FunctionCalls_;
|
|
|
|
|
FunctionCalls_ functionCalls_;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
foreach (FunctionCalls::iterator, i, functionCalls)
|
2013-08-02 21:29:23 +03:00
|
|
|
|
functionCalls_.insert(std::pair<unsigned int, ExprLambda *>(i->second, i->first));
|
2012-08-13 06:29:28 +03:00
|
|
|
|
foreach_reverse (FunctionCalls_::reverse_iterator, i, functionCalls_)
|
2013-08-02 21:29:23 +03:00
|
|
|
|
printMsg(v, format("%1$10d %2%") % i->first % i->second->showNamePos());
|
2012-08-13 06:29:28 +03:00
|
|
|
|
|
|
|
|
|
printMsg(v, format("evaluations of %1% attributes:") % attrSelects.size());
|
|
|
|
|
typedef std::multimap<unsigned int, Pos> AttrSelects_;
|
2013-08-02 21:29:23 +03:00
|
|
|
|
AttrSelects_ attrSelects_;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
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);
|
|
|
|
|
|
|
|
|
|
}
|
2003-10-31 19:09:31 +02:00
|
|
|
|
}
|
2006-09-05 00:06:23 +03:00
|
|
|
|
|
2010-04-07 16:55:46 +03:00
|
|
|
|
|
2014-09-17 16:19:07 +03:00
|
|
|
|
void EvalState::printCanaries()
|
|
|
|
|
{
|
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
if (!settings.get("debug-gc", false)) return;
|
|
|
|
|
|
|
|
|
|
GC_gcollect();
|
|
|
|
|
|
|
|
|
|
if (gcCanaries.empty()) {
|
|
|
|
|
printMsg(lvlError, "all canaries have been garbage-collected");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
printMsg(lvlError, "the following canaries have not been garbage-collected:");
|
|
|
|
|
|
|
|
|
|
for (auto i : gcCanaries)
|
|
|
|
|
printMsg(lvlError, format(" %1%") % i->string.s);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-09-22 15:46:42 +03:00
|
|
|
|
size_t valueSize(Value & v)
|
|
|
|
|
{
|
|
|
|
|
std::set<const void *> seen;
|
|
|
|
|
|
|
|
|
|
auto doString = [&](const char * s) -> size_t {
|
|
|
|
|
if (seen.find(s) != seen.end()) return 0;
|
|
|
|
|
seen.insert(s);
|
|
|
|
|
return strlen(s) + 1;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
std::function<size_t(Value & v)> doValue;
|
|
|
|
|
std::function<size_t(Env & v)> doEnv;
|
|
|
|
|
|
|
|
|
|
doValue = [&](Value & v) -> size_t {
|
|
|
|
|
if (seen.find(&v) != seen.end()) return 0;
|
|
|
|
|
seen.insert(&v);
|
|
|
|
|
|
|
|
|
|
size_t sz = sizeof(Value);
|
|
|
|
|
|
|
|
|
|
switch (v.type) {
|
|
|
|
|
case tString:
|
|
|
|
|
sz += doString(v.string.s);
|
|
|
|
|
if (v.string.context)
|
|
|
|
|
for (const char * * p = v.string.context; *p; ++p)
|
|
|
|
|
sz += doString(*p);
|
|
|
|
|
break;
|
|
|
|
|
case tPath:
|
|
|
|
|
sz += doString(v.path);
|
|
|
|
|
break;
|
|
|
|
|
case tAttrs:
|
|
|
|
|
for (auto & i : *v.attrs)
|
|
|
|
|
sz += doValue(*i.value);
|
|
|
|
|
break;
|
|
|
|
|
case tList:
|
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n)
|
|
|
|
|
sz += doValue(*v.list.elems[n]);
|
|
|
|
|
break;
|
|
|
|
|
case tThunk:
|
|
|
|
|
sz += doEnv(*v.thunk.env);
|
|
|
|
|
break;
|
|
|
|
|
case tApp:
|
|
|
|
|
sz += doValue(*v.app.left);
|
|
|
|
|
sz += doValue(*v.app.right);
|
|
|
|
|
break;
|
|
|
|
|
case tLambda:
|
|
|
|
|
sz += doEnv(*v.lambda.env);
|
|
|
|
|
break;
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
sz += doValue(*v.primOpApp.left);
|
|
|
|
|
sz += doValue(*v.primOpApp.right);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sz;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
doEnv = [&](Env & env) -> size_t {
|
|
|
|
|
if (seen.find(&env) != seen.end()) return 0;
|
|
|
|
|
seen.insert(&env);
|
|
|
|
|
|
|
|
|
|
size_t sz = sizeof(Env);
|
|
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < env.size; ++i)
|
|
|
|
|
if (env.values[i])
|
|
|
|
|
sz += doValue(*env.values[i]);
|
|
|
|
|
|
|
|
|
|
if (env.up) sz += doEnv(*env.up);
|
|
|
|
|
|
|
|
|
|
return sz;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return doValue(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
|
}
|