2003-10-30 16:48:26 +00:00
|
|
|
|
#include "eval.hh"
|
2006-09-04 21:06:23 +00:00
|
|
|
|
#include "hash.hh"
|
|
|
|
|
#include "util.hh"
|
2006-11-30 17:43:04 +00:00
|
|
|
|
#include "store-api.hh"
|
2006-10-16 15:55:34 +00:00
|
|
|
|
#include "derivations.hh"
|
2006-12-01 20:51:18 +00:00
|
|
|
|
#include "globals.hh"
|
2012-02-04 13:50:25 +00:00
|
|
|
|
#include "eval-inline.hh"
|
2020-04-06 21:57:28 +00:00
|
|
|
|
#include "filetransfer.hh"
|
2018-09-01 22:50:22 +00:00
|
|
|
|
#include "json.hh"
|
2019-11-13 16:38:37 +00:00
|
|
|
|
#include "function-trace.hh"
|
2004-02-04 16:03:29 +00:00
|
|
|
|
|
2014-03-29 23:49:23 +00:00
|
|
|
|
#include <algorithm>
|
2019-04-12 16:31:33 +00:00
|
|
|
|
#include <chrono>
|
2010-03-29 14:37:56 +00:00
|
|
|
|
#include <cstring>
|
2011-02-10 14:31:04 +00:00
|
|
|
|
#include <unistd.h>
|
2012-02-04 13:27:11 +00:00
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
#include <sys/resource.h>
|
2018-09-01 22:50:22 +00:00
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <fstream>
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2018-06-11 14:10:50 +00:00
|
|
|
|
#include <sys/resource.h>
|
|
|
|
|
|
2010-10-22 13:39:15 +00:00
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
|
2020-04-16 14:28:07 +00:00
|
|
|
|
#define GC_INCLUDE_NEW
|
|
|
|
|
|
2010-10-20 11:38:30 +00:00
|
|
|
|
#include <gc/gc.h>
|
|
|
|
|
#include <gc/gc_cpp.h>
|
|
|
|
|
|
2020-10-30 19:55:53 +00:00
|
|
|
|
#include <boost/coroutine2/coroutine.hpp>
|
|
|
|
|
#include <boost/coroutine2/protected_fixedsize_stack.hpp>
|
|
|
|
|
#include <boost/context/stack_context.hpp>
|
|
|
|
|
|
2010-10-22 13:39:15 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
namespace nix {
|
2010-10-24 00:41:29 +00:00
|
|
|
|
|
|
|
|
|
|
2015-03-19 13:11:35 +00:00
|
|
|
|
static char * dupString(const char * s)
|
|
|
|
|
{
|
|
|
|
|
char * t;
|
|
|
|
|
#if HAVE_BOEHMGC
|
2018-06-12 14:16:03 +00:00
|
|
|
|
t = GC_STRDUP(s);
|
2015-03-19 13:11:35 +00:00
|
|
|
|
#else
|
|
|
|
|
t = strdup(s);
|
|
|
|
|
#endif
|
|
|
|
|
if (!t) throw std::bad_alloc();
|
|
|
|
|
return t;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-12-05 18:11:09 +00:00
|
|
|
|
static char * dupStringWithLen(const char * s, size_t size)
|
|
|
|
|
{
|
|
|
|
|
char * t;
|
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
t = GC_STRNDUP(s, size);
|
|
|
|
|
#else
|
|
|
|
|
t = strndup(s, size);
|
|
|
|
|
#endif
|
|
|
|
|
if (!t) throw std::bad_alloc();
|
|
|
|
|
return t;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-16 14:28:07 +00:00
|
|
|
|
RootValue allocRootValue(Value * v)
|
|
|
|
|
{
|
2021-06-25 22:57:37 +00:00
|
|
|
|
#if HAVE_BOEHMGC
|
2020-04-16 14:28:07 +00:00
|
|
|
|
return std::allocate_shared<Value *>(traceable_allocator<Value *>(), v);
|
2021-06-25 22:57:37 +00:00
|
|
|
|
#else
|
|
|
|
|
return std::make_shared<Value *>(v);
|
|
|
|
|
#endif
|
2020-04-16 14:28:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-12-12 01:22:58 +00:00
|
|
|
|
void printValue(std::ostream & str, std::set<const Value *> & active, const Value & v)
|
2003-10-30 16:48:26 +00:00
|
|
|
|
{
|
2015-07-17 09:33:39 +00:00
|
|
|
|
checkInterrupt();
|
|
|
|
|
|
2019-10-09 13:51:52 +00:00
|
|
|
|
if (!active.insert(&v).second) {
|
2014-09-22 12:59:37 +00:00
|
|
|
|
str << "<CYCLE>";
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-17 13:42:52 +00:00
|
|
|
|
switch (v.internalType) {
|
2010-03-29 14:37:56 +00:00
|
|
|
|
case tInt:
|
|
|
|
|
str << v.integer;
|
|
|
|
|
break;
|
|
|
|
|
case tBool:
|
|
|
|
|
str << (v.boolean ? "true" : "false");
|
|
|
|
|
break;
|
|
|
|
|
case tString:
|
2010-04-08 11:41:19 +00: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";
|
2020-09-14 15:19:25 +00:00
|
|
|
|
else if (*i == '$' && *(i+1) == '{') str << "\\" << *i;
|
2010-04-08 11:41:19 +00:00
|
|
|
|
else str << *i;
|
|
|
|
|
str << "\"";
|
2010-03-29 14:37:56 +00:00
|
|
|
|
break;
|
2010-03-30 09:22:33 +00:00
|
|
|
|
case tPath:
|
|
|
|
|
str << v.path; // !!! escaping?
|
|
|
|
|
break;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
case tNull:
|
2010-08-27 12:10:56 +00:00
|
|
|
|
str << "null";
|
2010-03-29 14:37:56 +00:00
|
|
|
|
break;
|
2010-05-12 12:15:49 +00:00
|
|
|
|
case tAttrs: {
|
2010-03-29 14:37:56 +00:00
|
|
|
|
str << "{ ";
|
2017-01-25 15:06:50 +00:00
|
|
|
|
for (auto & i : v.attrs->lexicographicOrder()) {
|
|
|
|
|
str << i->name << " = ";
|
|
|
|
|
printValue(str, active, *i->value);
|
2014-09-22 12:59:37 +00:00
|
|
|
|
str << "; ";
|
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
str << "}";
|
|
|
|
|
break;
|
2010-05-12 12:15:49 +00:00
|
|
|
|
}
|
2015-07-23 20:05:09 +00:00
|
|
|
|
case tList1:
|
|
|
|
|
case tList2:
|
|
|
|
|
case tListN:
|
2010-03-29 14:37:56 +00:00
|
|
|
|
str << "[ ";
|
2015-07-23 20:05:09 +00:00
|
|
|
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
|
|
|
|
printValue(str, active, *v.listElems()[n]);
|
2014-09-22 12:59:37 +00:00
|
|
|
|
str << " ";
|
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
str << "]";
|
|
|
|
|
break;
|
|
|
|
|
case tThunk:
|
2010-06-02 09:43:04 +00:00
|
|
|
|
case tApp:
|
2010-03-29 14:37:56 +00:00
|
|
|
|
str << "<CODE>";
|
|
|
|
|
break;
|
|
|
|
|
case tLambda:
|
|
|
|
|
str << "<LAMBDA>";
|
|
|
|
|
break;
|
|
|
|
|
case tPrimOp:
|
|
|
|
|
str << "<PRIMOP>";
|
|
|
|
|
break;
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
str << "<PRIMOP-APP>";
|
|
|
|
|
break;
|
2014-11-30 18:16:19 +00:00
|
|
|
|
case tExternal:
|
|
|
|
|
str << *v.external;
|
|
|
|
|
break;
|
2016-01-04 23:40:40 +00:00
|
|
|
|
case tFloat:
|
|
|
|
|
str << v.fpoint;
|
2016-01-05 08:46:37 +00:00
|
|
|
|
break;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
default:
|
|
|
|
|
throw Error("invalid value");
|
|
|
|
|
}
|
2014-10-01 13:54:40 +00:00
|
|
|
|
|
|
|
|
|
active.erase(&v);
|
2014-09-22 12:59:37 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
std::ostream & operator << (std::ostream & str, const Value & v)
|
|
|
|
|
{
|
2014-10-01 13:54:40 +00:00
|
|
|
|
std::set<const Value *> active;
|
|
|
|
|
printValue(str, active, v);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return str;
|
|
|
|
|
}
|
2004-02-04 16:03:29 +00:00
|
|
|
|
|
2004-10-26 22:54:26 +00:00
|
|
|
|
|
2019-03-21 13:32:20 +00:00
|
|
|
|
const Value *getPrimOp(const Value &v) {
|
|
|
|
|
const Value * primOp = &v;
|
2020-12-12 01:15:11 +00:00
|
|
|
|
while (primOp->isPrimOpApp()) {
|
2019-03-21 13:32:20 +00:00
|
|
|
|
primOp = primOp->primOpApp.left;
|
|
|
|
|
}
|
2020-12-12 01:15:11 +00:00
|
|
|
|
assert(primOp->isPrimOp());
|
2019-03-21 13:32:20 +00:00
|
|
|
|
return primOp;
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-17 13:42:52 +00:00
|
|
|
|
string showType(ValueType type)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2019-09-09 14:34:44 +00:00
|
|
|
|
switch (type) {
|
2020-12-11 22:32:45 +00:00
|
|
|
|
case nInt: return "an integer";
|
|
|
|
|
case nBool: return "a Boolean";
|
|
|
|
|
case nString: return "a string";
|
|
|
|
|
case nPath: return "a path";
|
|
|
|
|
case nNull: return "null";
|
|
|
|
|
case nAttrs: return "a set";
|
|
|
|
|
case nList: return "a list";
|
|
|
|
|
case nFunction: return "a function";
|
|
|
|
|
case nExternal: return "an external value";
|
|
|
|
|
case nFloat: return "a float";
|
|
|
|
|
case nThunk: return "a thunk";
|
2019-09-09 14:34:44 +00:00
|
|
|
|
}
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string showType(const Value & v)
|
|
|
|
|
{
|
2020-12-17 13:42:52 +00:00
|
|
|
|
switch (v.internalType) {
|
2019-09-09 14:34:44 +00:00
|
|
|
|
case tString: return v.string.context ? "a string with context" : "a string";
|
2019-03-21 13:32:20 +00:00
|
|
|
|
case tPrimOp:
|
|
|
|
|
return fmt("the built-in function '%s'", string(v.primOp->name));
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
return fmt("the partially applied built-in function '%s'", string(getPrimOp(v)->primOp->name));
|
2014-11-30 18:16:19 +00:00
|
|
|
|
case tExternal: return v.external->showType();
|
2020-12-11 22:32:45 +00:00
|
|
|
|
case tThunk: return "a thunk";
|
|
|
|
|
case tApp: return "a function application";
|
|
|
|
|
case tBlackhole: return "a black hole";
|
2019-09-09 14:34:44 +00:00
|
|
|
|
default:
|
2020-12-17 13:45:45 +00:00
|
|
|
|
return showType(v.type());
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-08 21:27:00 +00:00
|
|
|
|
Pos Value::determinePos(const Pos &pos) const
|
|
|
|
|
{
|
|
|
|
|
switch (internalType) {
|
|
|
|
|
case tAttrs: return *attrs->pos;
|
|
|
|
|
case tLambda: return lambda.fun->pos;
|
|
|
|
|
case tApp: return app.left->determinePos(pos);
|
|
|
|
|
default: return pos;
|
|
|
|
|
}
|
|
|
|
|
}
|
2009-05-12 11:06:24 +00:00
|
|
|
|
|
2019-09-09 15:34:38 +00:00
|
|
|
|
bool Value::isTrivial() const
|
|
|
|
|
{
|
|
|
|
|
return
|
2020-12-17 13:42:52 +00:00
|
|
|
|
internalType != tApp
|
|
|
|
|
&& internalType != tPrimOpApp
|
|
|
|
|
&& (internalType != tThunk
|
2019-09-09 15:34:38 +00:00
|
|
|
|
|| (dynamic_cast<ExprAttrs *>(thunk.expr)
|
|
|
|
|
&& ((ExprAttrs *) thunk.expr)->dynamicAttrs.empty())
|
2020-10-26 19:37:11 +00:00
|
|
|
|
|| dynamic_cast<ExprLambda *>(thunk.expr)
|
|
|
|
|
|| dynamic_cast<ExprList *>(thunk.expr));
|
2019-09-09 15:34:38 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 15:53:52 +00:00
|
|
|
|
#if HAVE_BOEHMGC
|
2013-11-23 20:19:36 +00: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();
|
|
|
|
|
}
|
2020-10-30 19:55:53 +00:00
|
|
|
|
|
|
|
|
|
class BoehmGCStackAllocator : public StackAllocator {
|
BoehmGCStackAllocator: ignore stack protection page
This fixes a crash that looks like:
```
Thread 1 "nix-build" received signal SIGSEGV, Segmentation fault.
0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
(gdb) bt
0 0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
1 0x00007ffff7adeefb in GC_push_all_stacks () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
2 0x00007ffff7ad5ac7 in GC_mark_some () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
3 0x00007ffff7ad77bd in GC_stopped_mark () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
4 0x00007ffff7adbe3a in GC_try_to_collect_inner.part.0 () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
5 0x00007ffff7adc2a2 in GC_collect_or_expand () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
6 0x00007ffff7adc4f8 in GC_allocobj () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
7 0x00007ffff7adc88f in GC_generic_malloc_inner () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
8 0x00007ffff7ae1a04 in GC_generic_malloc_many () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
9 0x00007ffff7ae1c72 in GC_malloc_kind () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
10 0x00007ffff7e003d6 in nix::EvalState::allocValue() () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
11 0x00007ffff7e04b9c in nix::EvalState::callPrimOp(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
12 0x00007ffff7e0a773 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
13 0x00007ffff7e0a91d in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
14 0x00007ffff7e0a8f8 in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
15 0x00007ffff7e0e0e8 in nix::ExprOpNEq::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
16 0x00007ffff7e0d708 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
17 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
18 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
19 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
20 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
21 0x00007ffff7e09e19 in nix::ExprOpNot::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
22 0x00007ffff7e0a792 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
23 0x00007ffff7e8cba0 in nix::addPath(nix::EvalState&, nix::Pos const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Value*, nix::FileIngestionMethod, std::optional<nix::Hash>, nix::Value&)::{lambda(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)#1}::operator()(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&) const () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
24 0x00007ffff752e6f9 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
25 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
26 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
27 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
28 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
29 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
30 0x00007ffff757f8c0 in void boost::context::detail::fiber_entry<boost::context::detail::fiber_record<boost::context::fiber, nix::VirtualStackAllocator, boost::coroutines2::detail::pull_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >::control_block::control_block<nix::VirtualStackAllocator, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}>(boost::context::preallocated, nix::VirtualStackAllocator&&, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}&&)::{lambda(boost::context::fiber&&)#1}> >(boost::context::detail::transfer_t) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
31 0x00007ffff6f331ef in make_fcontext () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libboost_context.so.1.69.0
32 0x0000000000000000 in ?? ()
```
2020-11-16 10:03:53 +00:00
|
|
|
|
boost::coroutines2::protected_fixedsize_stack stack {
|
|
|
|
|
// We allocate 8 MB, the default max stack size on NixOS.
|
|
|
|
|
// A smaller stack might be quicker to allocate but reduces the stack
|
|
|
|
|
// depth available for source filter expressions etc.
|
|
|
|
|
std::max(boost::context::stack_traits::default_size(), static_cast<std::size_t>(8 * 1024 * 1024))
|
2020-10-30 22:18:24 +00:00
|
|
|
|
};
|
2020-10-30 19:55:53 +00:00
|
|
|
|
|
BoehmGCStackAllocator: ignore stack protection page
This fixes a crash that looks like:
```
Thread 1 "nix-build" received signal SIGSEGV, Segmentation fault.
0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
(gdb) bt
0 0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
1 0x00007ffff7adeefb in GC_push_all_stacks () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
2 0x00007ffff7ad5ac7 in GC_mark_some () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
3 0x00007ffff7ad77bd in GC_stopped_mark () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
4 0x00007ffff7adbe3a in GC_try_to_collect_inner.part.0 () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
5 0x00007ffff7adc2a2 in GC_collect_or_expand () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
6 0x00007ffff7adc4f8 in GC_allocobj () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
7 0x00007ffff7adc88f in GC_generic_malloc_inner () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
8 0x00007ffff7ae1a04 in GC_generic_malloc_many () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
9 0x00007ffff7ae1c72 in GC_malloc_kind () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
10 0x00007ffff7e003d6 in nix::EvalState::allocValue() () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
11 0x00007ffff7e04b9c in nix::EvalState::callPrimOp(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
12 0x00007ffff7e0a773 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
13 0x00007ffff7e0a91d in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
14 0x00007ffff7e0a8f8 in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
15 0x00007ffff7e0e0e8 in nix::ExprOpNEq::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
16 0x00007ffff7e0d708 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
17 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
18 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
19 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
20 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
21 0x00007ffff7e09e19 in nix::ExprOpNot::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
22 0x00007ffff7e0a792 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
23 0x00007ffff7e8cba0 in nix::addPath(nix::EvalState&, nix::Pos const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Value*, nix::FileIngestionMethod, std::optional<nix::Hash>, nix::Value&)::{lambda(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)#1}::operator()(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&) const () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
24 0x00007ffff752e6f9 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
25 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
26 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
27 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
28 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
29 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
30 0x00007ffff757f8c0 in void boost::context::detail::fiber_entry<boost::context::detail::fiber_record<boost::context::fiber, nix::VirtualStackAllocator, boost::coroutines2::detail::pull_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >::control_block::control_block<nix::VirtualStackAllocator, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}>(boost::context::preallocated, nix::VirtualStackAllocator&&, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}&&)::{lambda(boost::context::fiber&&)#1}> >(boost::context::detail::transfer_t) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
31 0x00007ffff6f331ef in make_fcontext () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libboost_context.so.1.69.0
32 0x0000000000000000 in ?? ()
```
2020-11-16 10:03:53 +00:00
|
|
|
|
// This is specific to boost::coroutines2::protected_fixedsize_stack.
|
|
|
|
|
// The stack protection page is included in sctx.size, so we have to
|
|
|
|
|
// subtract one page size from the stack size.
|
|
|
|
|
std::size_t pfss_usable_stack_size(boost::context::stack_context &sctx) {
|
|
|
|
|
return sctx.size - boost::context::stack_traits::page_size();
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-30 19:55:53 +00:00
|
|
|
|
public:
|
|
|
|
|
boost::context::stack_context allocate() override {
|
|
|
|
|
auto sctx = stack.allocate();
|
BoehmGCStackAllocator: ignore stack protection page
This fixes a crash that looks like:
```
Thread 1 "nix-build" received signal SIGSEGV, Segmentation fault.
0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
(gdb) bt
0 0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
1 0x00007ffff7adeefb in GC_push_all_stacks () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
2 0x00007ffff7ad5ac7 in GC_mark_some () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
3 0x00007ffff7ad77bd in GC_stopped_mark () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
4 0x00007ffff7adbe3a in GC_try_to_collect_inner.part.0 () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
5 0x00007ffff7adc2a2 in GC_collect_or_expand () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
6 0x00007ffff7adc4f8 in GC_allocobj () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
7 0x00007ffff7adc88f in GC_generic_malloc_inner () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
8 0x00007ffff7ae1a04 in GC_generic_malloc_many () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
9 0x00007ffff7ae1c72 in GC_malloc_kind () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
10 0x00007ffff7e003d6 in nix::EvalState::allocValue() () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
11 0x00007ffff7e04b9c in nix::EvalState::callPrimOp(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
12 0x00007ffff7e0a773 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
13 0x00007ffff7e0a91d in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
14 0x00007ffff7e0a8f8 in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
15 0x00007ffff7e0e0e8 in nix::ExprOpNEq::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
16 0x00007ffff7e0d708 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
17 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
18 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
19 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
20 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
21 0x00007ffff7e09e19 in nix::ExprOpNot::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
22 0x00007ffff7e0a792 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
23 0x00007ffff7e8cba0 in nix::addPath(nix::EvalState&, nix::Pos const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Value*, nix::FileIngestionMethod, std::optional<nix::Hash>, nix::Value&)::{lambda(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)#1}::operator()(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&) const () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
24 0x00007ffff752e6f9 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
25 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
26 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
27 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
28 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
29 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
30 0x00007ffff757f8c0 in void boost::context::detail::fiber_entry<boost::context::detail::fiber_record<boost::context::fiber, nix::VirtualStackAllocator, boost::coroutines2::detail::pull_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >::control_block::control_block<nix::VirtualStackAllocator, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}>(boost::context::preallocated, nix::VirtualStackAllocator&&, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}&&)::{lambda(boost::context::fiber&&)#1}> >(boost::context::detail::transfer_t) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
31 0x00007ffff6f331ef in make_fcontext () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libboost_context.so.1.69.0
32 0x0000000000000000 in ?? ()
```
2020-11-16 10:03:53 +00:00
|
|
|
|
|
|
|
|
|
// Stacks generally start at a high address and grow to lower addresses.
|
|
|
|
|
// Architectures that do the opposite are rare; in fact so rare that
|
|
|
|
|
// boost_routine does not implement it.
|
|
|
|
|
// So we subtract the stack size.
|
|
|
|
|
GC_add_roots(static_cast<char *>(sctx.sp) - pfss_usable_stack_size(sctx), sctx.sp);
|
2020-10-30 19:55:53 +00:00
|
|
|
|
return sctx;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void deallocate(boost::context::stack_context sctx) override {
|
BoehmGCStackAllocator: ignore stack protection page
This fixes a crash that looks like:
```
Thread 1 "nix-build" received signal SIGSEGV, Segmentation fault.
0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
(gdb) bt
0 0x00007ffff7ad22a0 in GC_push_all_eager () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
1 0x00007ffff7adeefb in GC_push_all_stacks () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
2 0x00007ffff7ad5ac7 in GC_mark_some () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
3 0x00007ffff7ad77bd in GC_stopped_mark () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
4 0x00007ffff7adbe3a in GC_try_to_collect_inner.part.0 () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
5 0x00007ffff7adc2a2 in GC_collect_or_expand () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
6 0x00007ffff7adc4f8 in GC_allocobj () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
7 0x00007ffff7adc88f in GC_generic_malloc_inner () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
8 0x00007ffff7ae1a04 in GC_generic_malloc_many () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
9 0x00007ffff7ae1c72 in GC_malloc_kind () from /nix/store/p1z58l18klf88iijpd0qi8yd2n9lhlk4-boehm-gc-8.0.4/lib/libgc.so.1
10 0x00007ffff7e003d6 in nix::EvalState::allocValue() () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
11 0x00007ffff7e04b9c in nix::EvalState::callPrimOp(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
12 0x00007ffff7e0a773 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
13 0x00007ffff7e0a91d in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
14 0x00007ffff7e0a8f8 in nix::ExprApp::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
15 0x00007ffff7e0e0e8 in nix::ExprOpNEq::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
16 0x00007ffff7e0d708 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
17 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
18 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
19 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
20 0x00007ffff7e0d695 in nix::ExprOpOr::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
21 0x00007ffff7e09e19 in nix::ExprOpNot::eval(nix::EvalState&, nix::Env&, nix::Value&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
22 0x00007ffff7e0a792 in nix::EvalState::callFunction(nix::Value&, nix::Value&, nix::Value&, nix::Pos const&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
23 0x00007ffff7e8cba0 in nix::addPath(nix::EvalState&, nix::Pos const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Value*, nix::FileIngestionMethod, std::optional<nix::Hash>, nix::Value&)::{lambda(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)#1}::operator()(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&) const () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixexpr.so
24 0x00007ffff752e6f9 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
25 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
26 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
27 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
28 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
29 0x00007ffff752e8e2 in nix::dump(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, nix::Sink&, std::function<bool (std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)>&) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
30 0x00007ffff757f8c0 in void boost::context::detail::fiber_entry<boost::context::detail::fiber_record<boost::context::fiber, nix::VirtualStackAllocator, boost::coroutines2::detail::pull_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >::control_block::control_block<nix::VirtualStackAllocator, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}>(boost::context::preallocated, nix::VirtualStackAllocator&&, nix::sinkToSource(std::function<void (nix::Sink&)>, std::function<void ()>)::SinkToSource::read(char*, unsigned long)::{lambda(boost::coroutines2::detail::push_coroutine<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >&)#1}&&)::{lambda(boost::context::fiber&&)#1}> >(boost::context::detail::transfer_t) () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libnixutil.so
31 0x00007ffff6f331ef in make_fcontext () from /nix/store/hzdzcv9d3bc8rlsaphh7x54zsf0x8nx6-nix-2.4pre20210601_5985b8b/lib/libboost_context.so.1.69.0
32 0x0000000000000000 in ?? ()
```
2020-11-16 10:03:53 +00:00
|
|
|
|
GC_remove_roots(static_cast<char *>(sctx.sp) - pfss_usable_stack_size(sctx), sctx.sp);
|
2020-10-30 19:55:53 +00:00
|
|
|
|
stack.deallocate(sctx);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static BoehmGCStackAllocator boehmGCStackAllocator;
|
|
|
|
|
|
2014-04-04 15:53:52 +00:00
|
|
|
|
#endif
|
2013-11-23 20:19:36 +00:00
|
|
|
|
|
|
|
|
|
|
2014-04-04 19:14:11 +00:00
|
|
|
|
static Symbol getName(const AttrName & name, EvalState & state, Env & env)
|
|
|
|
|
{
|
2013-12-31 23:56:26 +00: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);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-03-19 19:02:37 +00:00
|
|
|
|
static bool gcInitialised = false;
|
|
|
|
|
|
|
|
|
|
void initGC()
|
|
|
|
|
{
|
|
|
|
|
if (gcInitialised) return;
|
|
|
|
|
|
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
/* Initialise the Boehm garbage collector. */
|
2018-06-12 13:41:37 +00:00
|
|
|
|
|
|
|
|
|
/* Don't look for interior pointers. This reduces the odds of
|
|
|
|
|
misdetection a bit. */
|
2015-03-19 19:10:08 +00:00
|
|
|
|
GC_set_all_interior_pointers(0);
|
|
|
|
|
|
2018-06-12 13:41:37 +00:00
|
|
|
|
/* We don't have any roots in data segments, so don't scan from
|
|
|
|
|
there. */
|
|
|
|
|
GC_set_no_dls(1);
|
|
|
|
|
|
2015-03-19 19:02:37 +00:00
|
|
|
|
GC_INIT();
|
|
|
|
|
|
2017-04-14 12:42:20 +00:00
|
|
|
|
GC_set_oom_fn(oomHandler);
|
2015-03-19 19:02:37 +00:00
|
|
|
|
|
2020-10-30 19:55:53 +00:00
|
|
|
|
StackAllocator::defaultAllocator = &boehmGCStackAllocator;
|
|
|
|
|
|
2015-03-19 19:02:37 +00: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 overridden
|
|
|
|
|
through libgc's GC_INITIAL_HEAP_SIZE environment 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. */
|
2019-11-22 15:06:44 +00:00
|
|
|
|
if (!getEnv("GC_INITIAL_HEAP_SIZE")) {
|
2015-03-19 19:02:37 +00:00
|
|
|
|
size_t size = 32 * 1024 * 1024;
|
|
|
|
|
#if HAVE_SYSCONF && defined(_SC_PAGESIZE) && defined(_SC_PHYS_PAGES)
|
|
|
|
|
size_t maxSize = 384 * 1024 * 1024;
|
|
|
|
|
long pageSize = sysconf(_SC_PAGESIZE);
|
|
|
|
|
long pages = sysconf(_SC_PHYS_PAGES);
|
|
|
|
|
if (pageSize != -1)
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
gcInitialised = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-06-17 14:20:11 +00:00
|
|
|
|
/* Very hacky way to parse $NIX_PATH, which is colon-separated, but
|
|
|
|
|
can contain URLs (e.g. "nixpkgs=https://bla...:foo=https://"). */
|
2016-04-14 15:25:49 +00:00
|
|
|
|
static Strings parseNixPath(const string & s)
|
2015-06-17 14:20:11 +00:00
|
|
|
|
{
|
2016-04-14 15:25:49 +00:00
|
|
|
|
Strings res;
|
|
|
|
|
|
|
|
|
|
auto p = s.begin();
|
|
|
|
|
|
|
|
|
|
while (p != s.end()) {
|
|
|
|
|
auto start = p;
|
|
|
|
|
auto start2 = p;
|
|
|
|
|
|
|
|
|
|
while (p != s.end() && *p != ':') {
|
|
|
|
|
if (*p == '=') start2 = p + 1;
|
|
|
|
|
++p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (p == s.end()) {
|
|
|
|
|
if (p != start) res.push_back(std::string(start, p));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (*p == ':') {
|
|
|
|
|
if (isUri(std::string(start2, s.end()))) {
|
|
|
|
|
++p;
|
|
|
|
|
while (p != s.end() && *p != ':') ++p;
|
|
|
|
|
}
|
|
|
|
|
res.push_back(std::string(start, p));
|
|
|
|
|
if (p == s.end()) break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
++p;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-17 14:20:11 +00:00
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-06-29 19:09:48 +00:00
|
|
|
|
EvalState::EvalState(
|
|
|
|
|
const Strings & _searchPath,
|
|
|
|
|
ref<Store> store,
|
|
|
|
|
std::shared_ptr<Store> buildStore)
|
2010-04-13 12:25:42 +00: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 06:34:44 +00:00
|
|
|
|
, sValue(symbols.create("value"))
|
2010-04-21 15:08:58 +00:00
|
|
|
|
, sSystem(symbols.create("system"))
|
2010-05-15 08:10:12 +00:00
|
|
|
|
, sOverrides(symbols.create("__overrides"))
|
2012-11-28 12:49:44 +00:00
|
|
|
|
, sOutputs(symbols.create("outputs"))
|
2012-11-26 16:39:09 +00:00
|
|
|
|
, sOutputName(symbols.create("outputName"))
|
2012-11-27 14:01:32 +00:00
|
|
|
|
, sIgnoreNulls(symbols.create("__ignoreNulls"))
|
2013-11-18 19:14:54 +00:00
|
|
|
|
, sFile(symbols.create("file"))
|
|
|
|
|
, sLine(symbols.create("line"))
|
|
|
|
|
, sColumn(symbols.create("column"))
|
2014-10-16 02:04:48 +00:00
|
|
|
|
, sFunctor(symbols.create("__functor"))
|
2015-11-27 19:20:29 +00:00
|
|
|
|
, sToString(symbols.create("__toString"))
|
2016-08-29 15:28:20 +00:00
|
|
|
|
, sRight(symbols.create("right"))
|
|
|
|
|
, sWrong(symbols.create("wrong"))
|
Add support for passing structured data to builders
Previously, all derivation attributes had to be coerced into strings
so that they could be passed via the environment. This is lossy
(e.g. lists get flattened, necessitating configureFlags
vs. configureFlagsArray, of which the latter cannot be specified as an
attribute), doesn't support attribute sets at all, and has size
limitations (necessitating hacks like passAsFile).
This patch adds a new mode for passing attributes to builders, namely
encoded as a JSON file ".attrs.json" in the current directory of the
builder. This mode is activated via the special attribute
__structuredAttrs = true;
(The idea is that one day we can set this in stdenv.mkDerivation.)
For example,
stdenv.mkDerivation {
__structuredAttrs = true;
name = "foo";
buildInputs = [ pkgs.hello pkgs.cowsay ];
doCheck = true;
hardening.format = false;
}
results in a ".attrs.json" file containing (sans the indentation):
{
"buildInputs": [],
"builder": "/nix/store/ygl61ycpr2vjqrx775l1r2mw1g2rb754-bash-4.3-p48/bin/bash",
"configureFlags": [
"--with-foo",
"--with-bar=1 2"
],
"doCheck": true,
"hardening": {
"format": false
},
"name": "foo",
"nativeBuildInputs": [
"/nix/store/10h6li26i7g6z3mdpvra09yyf10mmzdr-hello-2.10",
"/nix/store/4jnvjin0r6wp6cv1hdm5jbkx3vinlcvk-cowsay-3.03"
],
"propagatedBuildInputs": [],
"propagatedNativeBuildInputs": [],
"stdenv": "/nix/store/f3hw3p8armnzy6xhd4h8s7anfjrs15n2-stdenv",
"system": "x86_64-linux"
}
"passAsFile" is ignored in this mode because it's not needed - large
strings are included directly in the JSON representation.
It is up to the builder to do something with the JSON
representation. For example, in bash-based builders, lists/attrsets of
string values could be mapped to bash (associative) arrays.
2017-01-25 15:42:07 +00:00
|
|
|
|
, sStructuredAttrs(symbols.create("__structuredAttrs"))
|
|
|
|
|
, sBuilder(symbols.create("builder"))
|
2017-03-04 13:24:06 +00:00
|
|
|
|
, sArgs(symbols.create("args"))
|
2020-07-22 23:59:25 +00:00
|
|
|
|
, sContentAddressed(symbols.create("__contentAddressed"))
|
2017-03-04 13:24:06 +00:00
|
|
|
|
, sOutputHash(symbols.create("outputHash"))
|
|
|
|
|
, sOutputHashAlgo(symbols.create("outputHashAlgo"))
|
|
|
|
|
, sOutputHashMode(symbols.create("outputHashMode"))
|
2020-06-18 11:44:40 +00:00
|
|
|
|
, sRecurseForDerivations(symbols.create("recurseForDerivations"))
|
2018-11-29 18:18:36 +00:00
|
|
|
|
, sDescription(symbols.create("description"))
|
2019-08-30 09:22:34 +00:00
|
|
|
|
, sSelf(symbols.create("self"))
|
2020-04-17 21:04:21 +00:00
|
|
|
|
, sEpsilon(symbols.create(""))
|
2017-06-28 16:11:01 +00:00
|
|
|
|
, repair(NoRepair)
|
Eliminate the "store" global variable
Also, move a few free-standing functions into StoreAPI and Derivation.
Also, introduce a non-nullable smart pointer, ref<T>, which is just a
wrapper around std::shared_ptr ensuring that the pointer is never
null. (For reference-counted values, this is better than passing a
"T&", because the latter doesn't maintain the refcount. Usually, the
caller will have a shared_ptr keeping the value alive, but that's not
always the case, e.g., when passing a reference to a std::thread via
std::bind.)
2016-02-04 13:28:26 +00:00
|
|
|
|
, store(store)
|
2021-06-29 19:09:48 +00:00
|
|
|
|
, buildStore(buildStore ? buildStore : store)
|
2020-09-21 16:22:45 +00:00
|
|
|
|
, regexCache(makeRegexCache())
|
2010-04-14 14:42:32 +00:00
|
|
|
|
, baseEnv(allocEnv(128))
|
2010-04-14 22:59:39 +00:00
|
|
|
|
, staticBaseEnv(false, 0)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2019-11-22 15:06:44 +00:00
|
|
|
|
countCalls = getEnv("NIX_COUNT_CALLS").value_or("0") != "0";
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2015-03-19 19:02:37 +00:00
|
|
|
|
assert(gcInitialised);
|
2011-08-06 16:05:24 +00:00
|
|
|
|
|
2018-05-30 15:47:30 +00:00
|
|
|
|
static_assert(sizeof(Env) <= 16, "environment must be <= 16 bytes");
|
2018-05-22 14:02:32 +00:00
|
|
|
|
|
2011-08-06 16:05:24 +00:00
|
|
|
|
/* Initialise the Nix expression search path. */
|
2018-03-27 17:02:22 +00:00
|
|
|
|
if (!evalSettings.pureEval) {
|
2018-01-16 17:50:38 +00:00
|
|
|
|
for (auto & i : _searchPath) addToSearchPath(i);
|
2019-11-22 22:07:35 +00:00
|
|
|
|
for (auto & i : evalSettings.nixPath.get()) addToSearchPath(i);
|
2018-01-16 17:50:38 +00:00
|
|
|
|
}
|
2020-09-16 14:56:28 +00:00
|
|
|
|
|
2018-03-27 17:02:22 +00:00
|
|
|
|
if (evalSettings.restrictEval || evalSettings.pureEval) {
|
2018-01-16 17:50:38 +00:00
|
|
|
|
allowedPaths = PathSet();
|
2018-05-09 13:45:05 +00:00
|
|
|
|
|
2018-01-16 17:50:38 +00:00
|
|
|
|
for (auto & i : searchPath) {
|
|
|
|
|
auto r = resolveSearchPathElem(i);
|
|
|
|
|
if (!r.first) continue;
|
2018-05-09 13:45:05 +00:00
|
|
|
|
|
|
|
|
|
auto path = r.second;
|
|
|
|
|
|
|
|
|
|
if (store->isInStore(r.second)) {
|
2020-08-24 17:15:17 +00:00
|
|
|
|
try {
|
|
|
|
|
StorePathSet closure;
|
|
|
|
|
store->computeFSClosure(store->toStorePath(r.second).first, closure);
|
|
|
|
|
for (auto & path : closure)
|
2021-10-07 10:11:00 +00:00
|
|
|
|
allowPath(store->printStorePath(path));
|
2020-08-24 17:15:17 +00:00
|
|
|
|
} catch (InvalidPath &) {
|
2021-10-07 10:11:00 +00:00
|
|
|
|
allowPath(r.second);
|
2020-08-24 17:15:17 +00:00
|
|
|
|
}
|
2018-05-09 13:45:05 +00:00
|
|
|
|
} else
|
2021-10-07 10:11:00 +00:00
|
|
|
|
allowPath(r.second);
|
2018-01-16 17:50:38 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-18 13:38:49 +00:00
|
|
|
|
vEmptySet.mkAttrs(allocBindings(0));
|
2015-07-23 21:11:08 +00:00
|
|
|
|
|
2012-01-03 14:01:47 +00:00
|
|
|
|
createBaseEnv();
|
2004-02-04 16:03:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-09 12:00:49 +00:00
|
|
|
|
EvalState::~EvalState()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-09-28 20:12:41 +00:00
|
|
|
|
void EvalState::requireExperimentalFeatureOnEvaluation(
|
|
|
|
|
const std::string & feature,
|
|
|
|
|
const std::string_view fName,
|
|
|
|
|
const Pos & pos)
|
|
|
|
|
{
|
|
|
|
|
if (!settings.isExperimentalFeatureEnabled(feature)) {
|
|
|
|
|
throw EvalError({
|
|
|
|
|
.msg = hintfmt(
|
|
|
|
|
"Cannot call '%2%' because experimental Nix feature '%1%' is disabled. You can enable it via '--extra-experimental-features %1%'.",
|
|
|
|
|
feature,
|
|
|
|
|
fName
|
|
|
|
|
),
|
|
|
|
|
.errPos = pos
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-07 10:11:00 +00:00
|
|
|
|
void EvalState::allowPath(const Path & path)
|
|
|
|
|
{
|
|
|
|
|
if (allowedPaths)
|
|
|
|
|
allowedPaths->insert(path);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-23 13:41:53 +00:00
|
|
|
|
Path EvalState::checkSourcePath(const Path & path_)
|
|
|
|
|
{
|
2018-01-16 17:50:38 +00:00
|
|
|
|
if (!allowedPaths) return path_;
|
|
|
|
|
|
2018-05-22 11:02:14 +00:00
|
|
|
|
auto i = resolvedPaths.find(path_);
|
|
|
|
|
if (i != resolvedPaths.end())
|
|
|
|
|
return i->second;
|
|
|
|
|
|
2018-01-16 17:50:38 +00:00
|
|
|
|
bool found = false;
|
|
|
|
|
|
2018-08-03 04:24:57 +00:00
|
|
|
|
/* First canonicalize the path without symlinks, so we make sure an
|
|
|
|
|
* attacker can't append ../../... to a path that would be in allowedPaths
|
|
|
|
|
* and thus leak symlink targets.
|
|
|
|
|
*/
|
|
|
|
|
Path abspath = canonPath(path_);
|
|
|
|
|
|
2020-12-22 13:43:20 +00:00
|
|
|
|
if (hasPrefix(abspath, corepkgsPrefix)) return abspath;
|
|
|
|
|
|
2018-01-16 17:50:38 +00:00
|
|
|
|
for (auto & i : *allowedPaths) {
|
2018-08-03 04:24:57 +00:00
|
|
|
|
if (isDirOrInDir(abspath, i)) {
|
2018-01-16 17:50:38 +00:00
|
|
|
|
found = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-19 13:58:26 +00:00
|
|
|
|
if (!found)
|
2018-08-03 04:24:57 +00:00
|
|
|
|
throw RestrictedPathError("access to path '%1%' is forbidden in restricted mode", abspath);
|
2015-02-23 13:41:53 +00:00
|
|
|
|
|
|
|
|
|
/* Resolve symlinks. */
|
2018-08-03 04:24:57 +00:00
|
|
|
|
debug(format("checking access to '%s'") % abspath);
|
|
|
|
|
Path path = canonPath(abspath, true);
|
2015-02-23 13:41:53 +00:00
|
|
|
|
|
2018-01-16 17:50:38 +00:00
|
|
|
|
for (auto & i : *allowedPaths) {
|
2018-05-22 11:02:14 +00:00
|
|
|
|
if (isDirOrInDir(path, i)) {
|
|
|
|
|
resolvedPaths[path_] = path;
|
2015-02-23 13:41:53 +00:00
|
|
|
|
return path;
|
2018-05-22 11:02:14 +00:00
|
|
|
|
}
|
2016-04-14 13:32:24 +00:00
|
|
|
|
}
|
2015-02-23 13:41:53 +00:00
|
|
|
|
|
2018-01-19 13:58:26 +00:00
|
|
|
|
throw RestrictedPathError("access to path '%1%' is forbidden in restricted mode", path);
|
2015-02-23 13:41:53 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2017-10-30 11:39:59 +00:00
|
|
|
|
void EvalState::checkURI(const std::string & uri)
|
|
|
|
|
{
|
2018-03-27 17:02:22 +00:00
|
|
|
|
if (!evalSettings.restrictEval) return;
|
2017-10-30 11:39:59 +00:00
|
|
|
|
|
|
|
|
|
/* 'uri' should be equal to a prefix, or in a subdirectory of a
|
|
|
|
|
prefix. Thus, the prefix https://github.co does not permit
|
|
|
|
|
access to https://github.com. Note: this allows 'http://' and
|
|
|
|
|
'https://' as prefixes for any http/https URI. */
|
2018-03-27 17:02:22 +00:00
|
|
|
|
for (auto & prefix : evalSettings.allowedUris.get())
|
2017-10-30 11:39:59 +00:00
|
|
|
|
if (uri == prefix ||
|
|
|
|
|
(uri.size() > prefix.size()
|
|
|
|
|
&& prefix.size() > 0
|
|
|
|
|
&& hasPrefix(uri, prefix)
|
|
|
|
|
&& (prefix[prefix.size() - 1] == '/' || uri[prefix.size()] == '/')))
|
|
|
|
|
return;
|
|
|
|
|
|
2018-02-06 13:35:14 +00:00
|
|
|
|
/* If the URI is a path, then check it against allowedPaths as
|
|
|
|
|
well. */
|
|
|
|
|
if (hasPrefix(uri, "/")) {
|
|
|
|
|
checkSourcePath(uri);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (hasPrefix(uri, "file://")) {
|
|
|
|
|
checkSourcePath(std::string(uri, 7));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-30 11:39:59 +00:00
|
|
|
|
throw RestrictedPathError("access to URI '%s' is forbidden in restricted mode", uri);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-01-12 16:31:08 +00:00
|
|
|
|
Path EvalState::toRealPath(const Path & path, const PathSet & context)
|
|
|
|
|
{
|
|
|
|
|
// FIXME: check whether 'path' is in 'context'.
|
|
|
|
|
return
|
|
|
|
|
!context.empty() && store->isInStore(path)
|
|
|
|
|
? store->toRealPath(path)
|
|
|
|
|
: path;
|
2019-11-10 16:23:35 +00:00
|
|
|
|
}
|
2018-01-12 16:31:08 +00:00
|
|
|
|
|
|
|
|
|
|
2018-02-08 18:00:53 +00:00
|
|
|
|
Value * EvalState::addConstant(const string & name, Value & v)
|
2010-03-30 14:39:27 +00:00
|
|
|
|
{
|
2010-10-22 14:47:42 +00:00
|
|
|
|
Value * v2 = allocValue();
|
|
|
|
|
*v2 = v;
|
2010-04-14 22:59:39 +00:00
|
|
|
|
staticBaseEnv.vars[symbols.create(name)] = baseEnvDispl;
|
2010-10-22 15:51:52 +00:00
|
|
|
|
baseEnv.values[baseEnvDispl++] = v2;
|
2010-03-30 14:39:27 +00:00
|
|
|
|
string name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
2010-10-24 00:41:29 +00:00
|
|
|
|
baseEnv.values[0]->attrs->push_back(Attr(symbols.create(name2), v2));
|
2018-02-08 18:00:53 +00:00
|
|
|
|
return v2;
|
2010-03-30 14:39:27 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-01-16 17:50:38 +00:00
|
|
|
|
Value * EvalState::addPrimOp(const string & name,
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t arity, PrimOpFun primOp)
|
2004-02-04 16:03:29 +00:00
|
|
|
|
{
|
2018-11-29 15:28:43 +00:00
|
|
|
|
auto name2 = string(name, 0, 2) == "__" ? string(name, 2) : name;
|
|
|
|
|
Symbol sym = symbols.create(name2);
|
|
|
|
|
|
|
|
|
|
/* Hack to make constants lazy: turn them into a application of
|
|
|
|
|
the primop to a dummy value. */
|
2018-02-08 18:00:53 +00:00
|
|
|
|
if (arity == 0) {
|
2018-11-29 15:28:43 +00:00
|
|
|
|
auto vPrimOp = allocValue();
|
2020-12-18 13:38:49 +00:00
|
|
|
|
vPrimOp->mkPrimOp(new PrimOp { .fun = primOp, .arity = 1, .name = sym });
|
2018-02-08 18:00:53 +00:00
|
|
|
|
Value v;
|
2018-11-29 15:28:43 +00:00
|
|
|
|
mkApp(v, *vPrimOp, *vPrimOp);
|
2018-02-08 18:00:53 +00:00
|
|
|
|
return addConstant(name, v);
|
|
|
|
|
}
|
2018-11-29 15:28:43 +00:00
|
|
|
|
|
2010-10-22 14:47:42 +00:00
|
|
|
|
Value * v = allocValue();
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v->mkPrimOp(new PrimOp { .fun = primOp, .arity = arity, .name = sym });
|
2010-10-24 19:52:33 +00:00
|
|
|
|
staticBaseEnv.vars[symbols.create(name)] = baseEnvDispl;
|
2010-10-22 15:51:52 +00:00
|
|
|
|
baseEnv.values[baseEnvDispl++] = v;
|
2010-10-24 19:52:33 +00:00
|
|
|
|
baseEnv.values[0]->attrs->push_back(Attr(sym, v));
|
2018-01-16 17:50:38 +00:00
|
|
|
|
return v;
|
2003-10-31 17:09:31 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-08-24 11:11:56 +00:00
|
|
|
|
Value * EvalState::addPrimOp(PrimOp && primOp)
|
|
|
|
|
{
|
|
|
|
|
/* Hack to make constants lazy: turn them into a application of
|
|
|
|
|
the primop to a dummy value. */
|
|
|
|
|
if (primOp.arity == 0) {
|
|
|
|
|
primOp.arity = 1;
|
|
|
|
|
auto vPrimOp = allocValue();
|
2020-12-18 13:38:49 +00:00
|
|
|
|
vPrimOp->mkPrimOp(new PrimOp(std::move(primOp)));
|
2020-08-24 11:11:56 +00:00
|
|
|
|
Value v;
|
|
|
|
|
mkApp(v, *vPrimOp, *vPrimOp);
|
|
|
|
|
return addConstant(primOp.name, v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Symbol envName = primOp.name;
|
|
|
|
|
if (hasPrefix(primOp.name, "__"))
|
|
|
|
|
primOp.name = symbols.create(std::string(primOp.name, 2));
|
|
|
|
|
|
|
|
|
|
Value * v = allocValue();
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v->mkPrimOp(new PrimOp(std::move(primOp)));
|
2020-08-24 11:11:56 +00:00
|
|
|
|
staticBaseEnv.vars[envName] = baseEnvDispl;
|
|
|
|
|
baseEnv.values[baseEnvDispl++] = v;
|
|
|
|
|
baseEnv.values[0]->attrs->push_back(Attr(primOp.name, v));
|
|
|
|
|
return v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-08-23 15:11:19 +00:00
|
|
|
|
Value & EvalState::getBuiltin(const string & name)
|
2013-09-03 13:45:32 +00:00
|
|
|
|
{
|
2016-08-23 15:11:19 +00:00
|
|
|
|
return *baseEnv.values[0]->attrs->find(symbols.create(name))->value;
|
2013-09-03 13:45:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-08-25 11:31:11 +00:00
|
|
|
|
std::optional<EvalState::Doc> EvalState::getDoc(Value & v)
|
|
|
|
|
{
|
2021-03-03 16:52:57 +00:00
|
|
|
|
if (v.isPrimOp()) {
|
2020-08-25 11:31:11 +00:00
|
|
|
|
auto v2 = &v;
|
|
|
|
|
if (v2->primOp->doc)
|
|
|
|
|
return Doc {
|
|
|
|
|
.pos = noPos,
|
|
|
|
|
.name = v2->primOp->name,
|
|
|
|
|
.arity = v2->primOp->arity,
|
|
|
|
|
.args = v2->primOp->args,
|
|
|
|
|
.doc = v2->primOp->doc,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2007-02-27 19:10:45 +00: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))
|
|
|
|
|
{
|
2020-04-21 23:07:07 +00:00
|
|
|
|
throw EvalError(s, s2);
|
2007-02-27 19:10:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const Pos & pos, const char * s, const string & s2))
|
2014-04-04 20:19:33 +00:00
|
|
|
|
{
|
2020-06-15 11:50:33 +00:00
|
|
|
|
throw EvalError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, s2),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 11:50:33 +00:00
|
|
|
|
});
|
2014-04-04 20:19:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-04-09 12:00:49 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const char * s, const string & s2, const string & s3))
|
|
|
|
|
{
|
2020-04-21 23:07:07 +00:00
|
|
|
|
throw EvalError(s, s2, s3);
|
2010-04-09 12:00:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const Pos & pos, const char * s, const string & s2, const string & s3))
|
2014-04-04 19:14:11 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw EvalError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, s2, s3),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2014-04-04 19:14:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwEvalError(const Pos & p1, const char * s, const Symbol & sym, const Pos & p2))
|
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 03:25:30 +00:00
|
|
|
|
{
|
2020-05-12 16:52:26 +00:00
|
|
|
|
// p1 is where the error occurred; p2 is a position mentioned in the message.
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw EvalError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, sym, p2),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = p1
|
2020-06-15 12:06:58 +00: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 03:25:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const Pos & pos, const char * s))
|
2014-04-04 20:19:33 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw TypeError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2014-04-04 20:19:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwTypeError(const Pos & pos, const char * s, const ExprLambda & fun, const Symbol & s2))
|
2013-11-07 17:04:36 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw TypeError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, fun.showNamePos(), s2),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2013-11-07 17:04:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwAssertionError(const Pos & pos, const char * s, const string & s1))
|
2010-04-09 12:00:49 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw AssertionError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, s1),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2010-04-09 12:00:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwUndefinedVarError(const Pos & pos, const char * s, const string & s1))
|
2013-10-08 12:45:36 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw UndefinedVarError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, s1),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2013-10-08 12:45:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-11 16:29:32 +00:00
|
|
|
|
LocalNoInlineNoReturn(void throwMissingArgumentError(const Pos & pos, const char * s, const string & s1))
|
|
|
|
|
{
|
|
|
|
|
throw MissingArgumentError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt(s, s1),
|
2020-11-11 16:29:32 +00:00
|
|
|
|
.errPos = pos
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-19 19:44:08 +00:00
|
|
|
|
LocalNoInline(void addErrorTrace(Error & e, const char * s, const string & s2))
|
2007-02-27 19:10:45 +00:00
|
|
|
|
{
|
2020-06-24 19:46:25 +00:00
|
|
|
|
e.addTrace(std::nullopt, s, s2);
|
2007-02-27 19:10:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-06-19 19:44:08 +00:00
|
|
|
|
LocalNoInline(void addErrorTrace(Error & e, const Pos & pos, const char * s, const string & s2))
|
2013-11-12 11:51:59 +00:00
|
|
|
|
{
|
2020-06-24 19:46:25 +00:00
|
|
|
|
e.addTrace(pos, s, s2);
|
2010-05-07 12:11:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2007-02-27 19:10:45 +00:00
|
|
|
|
|
2010-03-30 18:05:54 +00:00
|
|
|
|
void mkString(Value & v, const char * s)
|
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkString(dupString(s));
|
2010-03-30 18:05:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-12-05 18:11:09 +00:00
|
|
|
|
Value & mkString(Value & v, std::string_view s, const PathSet & context)
|
2010-03-30 18:05:54 +00:00
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkString(dupStringWithLen(s.data(), s.size()));
|
2010-03-31 19:52:29 +00:00
|
|
|
|
if (!context.empty()) {
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t n = 0;
|
2010-10-28 12:50:01 +00:00
|
|
|
|
v.string.context = (const char * *)
|
2015-03-19 13:11:35 +00:00
|
|
|
|
allocBytes((context.size() + 1) * sizeof(char *));
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : context)
|
|
|
|
|
v.string.context[n++] = dupString(i.c_str());
|
2010-03-31 19:52:29 +00:00
|
|
|
|
v.string.context[n] = 0;
|
|
|
|
|
}
|
2016-08-23 15:11:19 +00:00
|
|
|
|
return v;
|
2010-03-30 18:05:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void mkPath(Value & v, const char * s)
|
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkPath(dupString(s));
|
2010-03-30 18:05:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-10-08 12:24:53 +00:00
|
|
|
|
inline Value * EvalState::lookupVar(Env * env, const ExprVar & var, bool noEval)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t l = var.level; l; --l, env = env->up) ;
|
2013-07-31 10:44:21 +00:00
|
|
|
|
|
|
|
|
|
if (!var.fromWith) return env->values[var.displ];
|
|
|
|
|
|
|
|
|
|
while (1) {
|
2018-05-22 14:02:32 +00:00
|
|
|
|
if (env->type == Env::HasWithExpr) {
|
2013-07-31 10:44:21 +00:00
|
|
|
|
if (noEval) return 0;
|
2013-10-02 13:24:45 +00:00
|
|
|
|
Value * v = allocValue();
|
|
|
|
|
evalAttrs(*env->up, (Expr *) env->values[0], *v);
|
|
|
|
|
env->values[0] = v;
|
2018-05-22 14:02:32 +00:00
|
|
|
|
env->type = Env::HasWithAttrs;
|
2010-04-22 15:08:09 +00:00
|
|
|
|
}
|
2013-07-31 10:44:21 +00:00
|
|
|
|
Bindings::iterator j = env->values[0]->attrs->find(var.name);
|
|
|
|
|
if (j != env->values[0]->attrs->end()) {
|
2021-08-29 16:09:13 +00:00
|
|
|
|
if (countCalls) attrSelects[*j->pos]++;
|
2013-07-31 10:44:21 +00:00
|
|
|
|
return j->value;
|
|
|
|
|
}
|
|
|
|
|
if (!env->prevWith)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwUndefinedVarError(var.pos, "undefined variable '%1%'", var.name);
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t l = env->prevWith; l; --l, env = env->up) ;
|
2013-07-31 10:44:21 +00:00
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-22 14:47:42 +00:00
|
|
|
|
Value * EvalState::allocValue()
|
|
|
|
|
{
|
|
|
|
|
nrValues++;
|
2018-06-11 14:10:50 +00:00
|
|
|
|
auto v = (Value *) allocBytes(sizeof(Value));
|
|
|
|
|
return v;
|
2010-10-22 14:47:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-05-02 11:56:34 +00:00
|
|
|
|
Env & EvalState::allocEnv(size_t size)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
|
|
|
|
nrEnvs++;
|
2010-04-14 23:48:46 +00:00
|
|
|
|
nrValuesInEnvs += size;
|
2015-03-19 13:11:35 +00:00
|
|
|
|
Env * env = (Env *) allocBytes(sizeof(Env) + size * sizeof(Value *));
|
2018-05-22 14:02:32 +00:00
|
|
|
|
env->type = Env::Plain;
|
2010-10-24 14:20:02 +00:00
|
|
|
|
|
2018-02-13 03:00:17 +00:00
|
|
|
|
/* We assume that env->values has been cleared by the allocator; maybeThunk() and lookupVar fromWith expect this. */
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-04-14 14:42:32 +00:00
|
|
|
|
return *env;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-05-02 11:56:34 +00:00
|
|
|
|
void EvalState::mkList(Value & v, size_t size)
|
2010-03-30 14:39:27 +00:00
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkList(size);
|
|
|
|
|
if (size > 2)
|
|
|
|
|
v.bigList.elems = (Value * *) allocBytes(size * sizeof(Value *));
|
2015-07-23 20:05:09 +00:00
|
|
|
|
nrListElems += size;
|
2010-03-30 14:39:27 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-24 14:20:02 +00:00
|
|
|
|
unsigned long nrThunks = 0;
|
|
|
|
|
|
|
|
|
|
static inline void mkThunk(Value & v, Env & env, Expr * expr)
|
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkThunk(&env, expr);
|
2010-10-24 14:20:02 +00:00
|
|
|
|
nrThunks++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void EvalState::mkThunk_(Value & v, Expr * expr)
|
2010-04-07 15:47:06 +00:00
|
|
|
|
{
|
|
|
|
|
mkThunk(v, baseEnv, expr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-08-29 16:09:13 +00:00
|
|
|
|
void EvalState::mkPos(Value & v, ptr<Pos> pos)
|
2013-11-18 21:22:35 +00:00
|
|
|
|
{
|
2021-08-29 16:09:13 +00:00
|
|
|
|
if (pos->file.set()) {
|
2013-11-18 21:22:35 +00:00
|
|
|
|
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 14:20:02 +00: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 21:24:11 +00:00
|
|
|
|
Value * Expr::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
|
|
|
|
Value * v = state.allocValue();
|
|
|
|
|
mkThunk(*v, env, this);
|
|
|
|
|
return v;
|
|
|
|
|
}
|
2010-10-24 14:20:02 +00:00
|
|
|
|
|
|
|
|
|
|
2012-01-04 21:24:11 +00:00
|
|
|
|
Value * ExprVar::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2013-10-08 12:24:53 +00:00
|
|
|
|
Value * v = state.lookupVar(&env, *this, true);
|
2013-07-16 12:43:54 +00:00
|
|
|
|
/* The value might not be initialised in the environment yet.
|
|
|
|
|
In that case, ignore it. */
|
2021-07-21 22:31:08 +00:00
|
|
|
|
if (v) { state.nrAvoided++; return v; }
|
2012-01-04 21:24:11 +00:00
|
|
|
|
return Expr::maybeThunk(state, env);
|
2010-10-24 14:20:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-01-07 17:26:33 +00:00
|
|
|
|
Value * ExprString::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2021-07-21 22:31:08 +00:00
|
|
|
|
state.nrAvoided++;
|
2012-01-07 17:26:33 +00:00
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Value * ExprInt::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2021-07-21 22:31:08 +00:00
|
|
|
|
state.nrAvoided++;
|
2012-01-07 17:26:33 +00:00
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-04 23:40:40 +00:00
|
|
|
|
Value * ExprFloat::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2021-07-21 22:31:08 +00:00
|
|
|
|
state.nrAvoided++;
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-07 17:26:33 +00:00
|
|
|
|
Value * ExprPath::maybeThunk(EvalState & state, Env & env)
|
|
|
|
|
{
|
2021-07-21 22:31:08 +00:00
|
|
|
|
state.nrAvoided++;
|
2012-01-07 17:26:33 +00:00
|
|
|
|
return &v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-09-09 15:34:38 +00:00
|
|
|
|
void EvalState::evalFile(const Path & path_, Value & v, bool mustBeTrivial)
|
2010-03-30 09:22:33 +00:00
|
|
|
|
{
|
2018-01-16 17:50:38 +00:00
|
|
|
|
auto path = checkSourcePath(path_);
|
|
|
|
|
|
2013-10-23 11:16:46 +00:00
|
|
|
|
FileEvalCache::iterator i;
|
|
|
|
|
if ((i = fileEvalCache.find(path)) != fileEvalCache.end()) {
|
|
|
|
|
v = i->second;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2013-09-03 10:56:33 +00:00
|
|
|
|
|
2021-09-13 12:41:28 +00:00
|
|
|
|
Path resolvedPath = resolveExprPath(path);
|
|
|
|
|
if ((i = fileEvalCache.find(resolvedPath)) != fileEvalCache.end()) {
|
2011-08-06 19:45:43 +00:00
|
|
|
|
v = i->second;
|
2013-09-03 10:56:33 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-13 12:41:28 +00:00
|
|
|
|
printTalkative("evaluating file '%1%'", resolvedPath);
|
2018-06-11 14:06:01 +00:00
|
|
|
|
Expr * e = nullptr;
|
|
|
|
|
|
2021-09-13 12:41:28 +00:00
|
|
|
|
auto j = fileParseCache.find(resolvedPath);
|
2018-06-11 14:06:01 +00:00
|
|
|
|
if (j != fileParseCache.end())
|
|
|
|
|
e = j->second;
|
|
|
|
|
|
|
|
|
|
if (!e)
|
2021-09-13 12:41:28 +00:00
|
|
|
|
e = parseExprFromFile(checkSourcePath(resolvedPath));
|
|
|
|
|
|
|
|
|
|
cacheFile(path, resolvedPath, e, v, mustBeTrivial);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void EvalState::resetFileCache()
|
|
|
|
|
{
|
|
|
|
|
fileEvalCache.clear();
|
|
|
|
|
fileParseCache.clear();
|
|
|
|
|
}
|
2018-06-11 14:06:01 +00:00
|
|
|
|
|
2021-09-13 12:41:28 +00:00
|
|
|
|
|
|
|
|
|
void EvalState::cacheFile(
|
|
|
|
|
const Path & path,
|
|
|
|
|
const Path & resolvedPath,
|
|
|
|
|
Expr * e,
|
|
|
|
|
Value & v,
|
|
|
|
|
bool mustBeTrivial)
|
|
|
|
|
{
|
|
|
|
|
fileParseCache[resolvedPath] = e;
|
2018-06-11 14:06:01 +00:00
|
|
|
|
|
2013-09-03 10:56:33 +00:00
|
|
|
|
try {
|
2019-09-09 15:34:38 +00:00
|
|
|
|
// Enforce that 'flake.nix' is a direct attrset, not a
|
|
|
|
|
// computation.
|
|
|
|
|
if (mustBeTrivial &&
|
|
|
|
|
!(dynamic_cast<ExprAttrs *>(e)))
|
2021-06-29 21:28:43 +00:00
|
|
|
|
throw EvalError("file '%s' must be an attribute set", path);
|
2013-09-03 10:56:33 +00:00
|
|
|
|
eval(e, v);
|
|
|
|
|
} catch (Error & e) {
|
2021-09-13 12:41:28 +00:00
|
|
|
|
addErrorTrace(e, "while evaluating the file '%1%':", resolvedPath);
|
2013-09-03 10:56:33 +00:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2013-10-23 11:16:46 +00:00
|
|
|
|
|
2021-09-13 12:41:28 +00:00
|
|
|
|
fileEvalCache[resolvedPath] = v;
|
|
|
|
|
if (path != resolvedPath) fileEvalCache[path] = v;
|
2013-09-02 16:34:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void EvalState::eval(Expr * e, Value & v)
|
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
e->eval(*this, baseEnv, v);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
|
2014-03-05 15:18:13 +00:00
|
|
|
|
inline bool EvalState::evalBool(Env & env, Expr * e)
|
2010-04-12 18:30:11 +00:00
|
|
|
|
{
|
2014-03-05 15:18:13 +00:00
|
|
|
|
Value v;
|
2012-02-04 13:50:25 +00:00
|
|
|
|
e->eval(*this, env, v);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nBool)
|
2013-11-07 12:44:14 +00:00
|
|
|
|
throwTypeError("value is %1% while a Boolean was expected", v);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 20:43:52 +00:00
|
|
|
|
inline bool EvalState::evalBool(Env & env, Expr * e, const Pos & pos)
|
|
|
|
|
{
|
|
|
|
|
Value v;
|
|
|
|
|
e->eval(*this, env, v);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nBool)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while a Boolean was expected", v);
|
2014-04-04 20:43:52 +00:00
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2012-02-04 13:50:25 +00:00
|
|
|
|
inline void EvalState::evalAttrs(Env & env, Expr * e, Value & v)
|
2010-04-16 15:13:47 +00:00
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
e->eval(*this, env, v);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nAttrs)
|
2013-11-07 12:44:14 +00:00
|
|
|
|
throwTypeError("value is %1% while a set was expected", v);
|
2010-04-16 15:13:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 22:03:27 +00:00
|
|
|
|
void Expr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprInt::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 17:26:33 +00:00
|
|
|
|
v = this->v;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-01-04 23:40:40 +00:00
|
|
|
|
void ExprFloat::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
v = this->v;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprString::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 17:26:33 +00:00
|
|
|
|
v = this->v;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprPath::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-01-07 17:26:33 +00:00
|
|
|
|
v = this->v;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprAttrs::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-09-19 14:49:41 +00: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 03:25:30 +00:00
|
|
|
|
Env *dynamicEnv = &env;
|
2010-04-14 23:25:05 +00:00
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
if (recursive) {
|
|
|
|
|
/* Create a new environment that contains the attributes in
|
|
|
|
|
this `rec'. */
|
2010-10-24 19:52:33 +00:00
|
|
|
|
Env & env2(state.allocEnv(attrs.size()));
|
2010-04-12 18:30:11 +00: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 03:25:30 +00:00
|
|
|
|
dynamicEnv = &env2;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
|
2010-10-24 19:52:33 +00: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. */
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t displ = 0;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : attrs) {
|
2013-07-15 21:10:18 +00:00
|
|
|
|
Value * vAttr;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
if (hasOverrides && !i.second.inherited) {
|
2013-07-15 21:10:18 +00:00
|
|
|
|
vAttr = state.allocValue();
|
2015-07-17 17:24:28 +00:00
|
|
|
|
mkThunk(*vAttr, env2, i.second.e);
|
2013-07-15 21:10:18 +00:00
|
|
|
|
} else
|
2015-07-17 17:24:28 +00:00
|
|
|
|
vAttr = i.second.e->maybeThunk(state, i.second.inherited ? env : env2);
|
2013-07-15 21:10:18 +00:00
|
|
|
|
env2.values[displ++] = vAttr;
|
2021-08-29 16:09:13 +00:00
|
|
|
|
v.attrs->push_back(Attr(i.first, vAttr, ptr(&i.second.pos)));
|
2013-07-15 21:10:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-05-15 08:10:12 +00: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 19:52:33 +00:00
|
|
|
|
if (hasOverrides) {
|
|
|
|
|
Value * vOverrides = (*v.attrs)[overrides->second.displ].value;
|
|
|
|
|
state.forceAttrs(*vOverrides);
|
2019-11-25 12:37:14 +00:00
|
|
|
|
Bindings * newBnds = state.allocBindings(v.attrs->capacity() + vOverrides->attrs->size());
|
2014-09-19 14:49:41 +00:00
|
|
|
|
for (auto & i : *v.attrs)
|
|
|
|
|
newBnds->push_back(i);
|
|
|
|
|
for (auto & i : *vOverrides->attrs) {
|
|
|
|
|
AttrDefs::iterator j = attrs.find(i.name);
|
2010-10-24 19:52:33 +00:00
|
|
|
|
if (j != attrs.end()) {
|
2014-09-19 14:49:41 +00:00
|
|
|
|
(*newBnds)[j->second.displ] = i;
|
|
|
|
|
env2.values[j->second.displ] = i.value;
|
2010-10-24 19:52:33 +00:00
|
|
|
|
} else
|
2014-09-19 14:49:41 +00:00
|
|
|
|
newBnds->push_back(i);
|
2010-05-15 08:10:12 +00:00
|
|
|
|
}
|
2014-09-19 14:49:41 +00:00
|
|
|
|
newBnds->sort();
|
|
|
|
|
v.attrs = newBnds;
|
2010-05-15 08:10:12 +00:00
|
|
|
|
}
|
2010-04-12 18:30:11 +00: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 03:25:30 +00:00
|
|
|
|
else
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : attrs)
|
2021-08-29 16:09:13 +00:00
|
|
|
|
v.attrs->push_back(Attr(i.first, i.second.e->maybeThunk(state, env), ptr(&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 03:25:30 +00:00
|
|
|
|
|
2014-04-04 20:19:33 +00:00
|
|
|
|
/* Dynamic attrs apply *after* rec and __overrides. */
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : dynamicAttrs) {
|
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 03:25:30 +00:00
|
|
|
|
Value nameVal;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
i.nameExpr->eval(state, *dynamicEnv, nameVal);
|
2015-07-31 15:32:25 +00:00
|
|
|
|
state.forceValue(nameVal, i.pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (nameVal.type() == nNull)
|
2014-10-04 23:04:58 +00:00
|
|
|
|
continue;
|
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 03:25:30 +00:00
|
|
|
|
state.forceStringNoCtx(nameVal);
|
|
|
|
|
Symbol nameSym = state.symbols.create(nameVal.string.s);
|
|
|
|
|
Bindings::iterator j = v.attrs->find(nameSym);
|
|
|
|
|
if (j != v.attrs->end())
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(i.pos, "dynamic attribute '%1%' already defined at %2%", nameSym, *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 03:25:30 +00:00
|
|
|
|
|
2015-07-17 17:24:28 +00:00
|
|
|
|
i.valueExpr->setName(nameSym);
|
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 03:25:30 +00:00
|
|
|
|
/* Keep sorted order so find can catch duplicates */
|
2021-08-29 16:09:13 +00:00
|
|
|
|
v.attrs->push_back(Attr(nameSym, i.valueExpr->maybeThunk(state, *dynamicEnv), ptr(&i.pos)));
|
2014-09-19 14:49:41 +00:00
|
|
|
|
v.attrs->sort(); // FIXME: inefficient
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
2021-01-08 21:27:00 +00:00
|
|
|
|
|
2021-08-29 16:09:13 +00:00
|
|
|
|
v.attrs->pos = ptr(&pos);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-13 13:42:25 +00:00
|
|
|
|
void ExprLet::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
/* Create a new environment that contains the attributes in this
|
|
|
|
|
`let'. */
|
2010-10-24 19:52:33 +00:00
|
|
|
|
Env & env2(state.allocEnv(attrs->attrs.size()));
|
2010-04-13 13:42:25 +00:00
|
|
|
|
env2.up = &env;
|
2010-04-14 14:42:32 +00:00
|
|
|
|
|
2010-10-24 19:52:33 +00:00
|
|
|
|
/* The recursive attributes are evaluated in the new environment,
|
|
|
|
|
while the inherited attributes are evaluated in the original
|
2010-04-13 13:42:25 +00:00
|
|
|
|
environment. */
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t displ = 0;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : attrs->attrs)
|
|
|
|
|
env2.values[displ++] = i.second.e->maybeThunk(state, i.second.inherited ? env : env2);
|
2010-04-13 13:42:25 +00:00
|
|
|
|
|
2012-02-04 13:50:25 +00:00
|
|
|
|
body->eval(state, env2, v);
|
2010-04-13 13:42:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprList::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
state.mkList(v, elems.size());
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t n = 0; n < elems.size(); ++n)
|
2015-07-23 20:05:09 +00:00
|
|
|
|
v.listElems()[n] = elems[n]->maybeThunk(state, env);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprVar::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2013-10-08 12:24:53 +00:00
|
|
|
|
Value * v2 = state.lookupVar(&env, *this, false);
|
2015-07-31 15:32:25 +00:00
|
|
|
|
state.forceValue(*v2, pos);
|
2010-04-14 15:14:23 +00:00
|
|
|
|
v = *v2;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-03-06 13:24:08 +00:00
|
|
|
|
static string showAttrPath(EvalState & state, Env & env, const AttrPath & attrPath)
|
|
|
|
|
{
|
|
|
|
|
std::ostringstream out;
|
|
|
|
|
bool first = true;
|
|
|
|
|
for (auto & i : attrPath) {
|
|
|
|
|
if (!first) out << '.'; else first = false;
|
|
|
|
|
try {
|
|
|
|
|
out << getName(i, state, env);
|
|
|
|
|
} catch (Error & e) {
|
|
|
|
|
assert(!i.symbol.set());
|
|
|
|
|
out << "\"${" << *i.expr << "}\"";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return out.str();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprSelect::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2011-07-06 12:28:57 +00:00
|
|
|
|
Value vTmp;
|
2021-08-29 16:09:13 +00:00
|
|
|
|
ptr<Pos> pos2(&noPos);
|
2011-07-06 12:28:57 +00:00
|
|
|
|
Value * vAttrs = &vTmp;
|
|
|
|
|
|
2012-02-04 13:50:25 +00:00
|
|
|
|
e->eval(state, env, vTmp);
|
2011-07-06 12:28:57 +00:00
|
|
|
|
|
|
|
|
|
try {
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : attrPath) {
|
2021-07-21 22:31:08 +00:00
|
|
|
|
state.nrLookups++;
|
2011-07-06 12:28:57 +00:00
|
|
|
|
Bindings::iterator j;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
Symbol name = getName(i, state, env);
|
2011-07-13 12:19:57 +00:00
|
|
|
|
if (def) {
|
2015-07-31 15:32:25 +00:00
|
|
|
|
state.forceValue(*vAttrs, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (vAttrs->type() != nAttrs ||
|
2013-12-31 23:56:26 +00:00
|
|
|
|
(j = vAttrs->attrs->find(name)) == vAttrs->attrs->end())
|
2011-07-13 12:19:57 +00:00
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
def->eval(state, env, v);
|
2011-07-13 12:19:57 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2014-04-04 20:52:14 +00:00
|
|
|
|
state.forceAttrs(*vAttrs, pos);
|
2015-03-06 13:24:08 +00:00
|
|
|
|
if ((j = vAttrs->attrs->find(name)) == vAttrs->attrs->end())
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(pos, "attribute '%1%' missing", name);
|
2011-07-13 12:19:57 +00:00
|
|
|
|
}
|
2011-07-06 12:28:57 +00:00
|
|
|
|
vAttrs = j->value;
|
2014-04-04 20:52:14 +00:00
|
|
|
|
pos2 = j->pos;
|
2021-08-29 16:09:13 +00:00
|
|
|
|
if (state.countCalls) state.attrSelects[*pos2]++;
|
2011-07-06 12:28:57 +00:00
|
|
|
|
}
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2021-08-29 16:09:13 +00:00
|
|
|
|
state.forceValue(*vAttrs, (*pos2 != noPos ? *pos2 : this->pos ) );
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
} catch (Error & e) {
|
2021-08-29 16:09:13 +00:00
|
|
|
|
if (*pos2 != noPos && pos2->file != state.sDerivationNix)
|
2020-06-19 19:44:08 +00:00
|
|
|
|
addErrorTrace(e, *pos2, "while evaluating the attribute '%1%'",
|
|
|
|
|
showAttrPath(state, env, attrPath));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2011-07-06 12:28:57 +00:00
|
|
|
|
v = *vAttrs;
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:21:24 +00:00
|
|
|
|
void ExprOpHasAttr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2011-07-06 10:58:17 +00:00
|
|
|
|
Value vTmp;
|
|
|
|
|
Value * vAttrs = &vTmp;
|
|
|
|
|
|
2012-02-04 13:50:25 +00:00
|
|
|
|
e->eval(state, env, vTmp);
|
2011-07-06 10:58:17 +00:00
|
|
|
|
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : attrPath) {
|
2011-07-06 10:58:17 +00:00
|
|
|
|
state.forceValue(*vAttrs);
|
|
|
|
|
Bindings::iterator j;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
Symbol name = getName(i, state, env);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (vAttrs->type() != nAttrs ||
|
2013-12-31 23:56:26 +00:00
|
|
|
|
(j = vAttrs->attrs->find(name)) == vAttrs->attrs->end())
|
2011-07-06 10:58:17 +00:00
|
|
|
|
{
|
|
|
|
|
mkBool(v, false);
|
|
|
|
|
return;
|
|
|
|
|
} else {
|
|
|
|
|
vAttrs = j->value;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2011-07-06 10:58:17 +00:00
|
|
|
|
mkBool(v, true);
|
2010-04-12 21:21:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprLambda::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkLambda(&env, this);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprApp::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-02-27 20:47:59 +00:00
|
|
|
|
/* FIXME: vFun prevents GCC from doing tail call optimisation. */
|
|
|
|
|
Value vFun;
|
|
|
|
|
e1->eval(state, env, vFun);
|
2014-04-04 15:53:52 +00:00
|
|
|
|
state.callFunction(vFun, *(e2->maybeThunk(state, env)), v, pos);
|
2013-11-07 17:04:36 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 16:51:01 +00:00
|
|
|
|
void EvalState::callPrimOp(Value & fun, Value & arg, Value & v, const Pos & pos)
|
2013-11-07 17:04:36 +00:00
|
|
|
|
{
|
|
|
|
|
/* Figure out the number of arguments still needed. */
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t argsDone = 0;
|
2013-11-07 17:04:36 +00:00
|
|
|
|
Value * primOp = &fun;
|
2020-12-12 01:15:11 +00:00
|
|
|
|
while (primOp->isPrimOpApp()) {
|
2013-11-07 17:04:36 +00:00
|
|
|
|
argsDone++;
|
|
|
|
|
primOp = primOp->primOpApp.left;
|
|
|
|
|
}
|
2020-12-12 01:15:11 +00:00
|
|
|
|
assert(primOp->isPrimOp());
|
2018-05-02 11:56:34 +00:00
|
|
|
|
auto arity = primOp->primOp->arity;
|
|
|
|
|
auto argsLeft = arity - argsDone;
|
2013-11-07 17:04:36 +00:00
|
|
|
|
|
|
|
|
|
if (argsLeft == 1) {
|
|
|
|
|
/* We have all the arguments, so call the primop. */
|
|
|
|
|
|
|
|
|
|
/* Put all the arguments in an array. */
|
|
|
|
|
Value * vArgs[arity];
|
2018-05-02 11:56:34 +00:00
|
|
|
|
auto n = arity - 1;
|
2013-11-07 17:04:36 +00:00
|
|
|
|
vArgs[n--] = &arg;
|
2020-12-12 01:15:11 +00:00
|
|
|
|
for (Value * arg = &fun; arg->isPrimOpApp(); arg = arg->primOpApp.left)
|
2013-11-07 17:04:36 +00:00
|
|
|
|
vArgs[n--] = arg->primOpApp.right;
|
|
|
|
|
|
|
|
|
|
/* And call the primop. */
|
|
|
|
|
nrPrimOpCalls++;
|
|
|
|
|
if (countCalls) primOpCalls[primOp->primOp->name]++;
|
2014-04-04 16:51:01 +00:00
|
|
|
|
primOp->primOp->fun(*this, pos, vArgs, v);
|
2013-11-07 17:04:36 +00:00
|
|
|
|
} else {
|
|
|
|
|
Value * fun2 = allocValue();
|
|
|
|
|
*fun2 = fun;
|
2020-12-18 13:38:49 +00:00
|
|
|
|
v.mkPrimOpApp(fun2, &arg);
|
2013-11-07 17:04:36 +00:00
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 15:53:52 +00:00
|
|
|
|
void EvalState::callFunction(Value & fun, Value & arg, Value & v, const Pos & pos)
|
2010-03-30 13:47:59 +00:00
|
|
|
|
{
|
2020-01-05 15:21:34 +00:00
|
|
|
|
auto trace = evalSettings.traceFunctionCalls ? std::make_unique<FunctionCallTrace>(pos) : nullptr;
|
2019-04-12 16:31:33 +00:00
|
|
|
|
|
2018-07-06 21:49:51 +00:00
|
|
|
|
forceValue(fun, pos);
|
|
|
|
|
|
2020-12-12 01:15:11 +00:00
|
|
|
|
if (fun.isPrimOp() || fun.isPrimOpApp()) {
|
2014-04-04 16:51:01 +00:00
|
|
|
|
callPrimOp(fun, arg, v, pos);
|
2010-03-30 13:47:59 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (fun.type() == nAttrs) {
|
2014-10-16 02:04:48 +00:00
|
|
|
|
auto found = fun.attrs->find(sFunctor);
|
|
|
|
|
if (found != fun.attrs->end()) {
|
2016-08-29 11:36:28 +00:00
|
|
|
|
/* fun may be allocated on the stack of the calling function,
|
|
|
|
|
* but for functors we may keep a reference, so heap-allocate
|
|
|
|
|
* a copy and use that instead.
|
|
|
|
|
*/
|
|
|
|
|
auto & fun2 = *allocValue();
|
|
|
|
|
fun2 = fun;
|
|
|
|
|
/* !!! Should we use the attr pos here? */
|
|
|
|
|
Value v2;
|
|
|
|
|
callFunction(*found->value, fun2, v2, pos);
|
|
|
|
|
return callFunction(v2, arg, v, pos);
|
2014-10-16 02:04:48 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-12 01:15:11 +00:00
|
|
|
|
if (!fun.isLambda())
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "attempt to call something which is not a function but %1%", fun);
|
2010-03-30 13:47:59 +00:00
|
|
|
|
|
2013-11-12 11:51:59 +00:00
|
|
|
|
ExprLambda & lambda(*fun.lambda.fun);
|
|
|
|
|
|
2018-05-02 11:56:34 +00:00
|
|
|
|
auto size =
|
2013-11-12 11:51:59 +00:00
|
|
|
|
(lambda.arg.empty() ? 0 : 1) +
|
2021-10-06 15:08:08 +00:00
|
|
|
|
(lambda.hasFormals() ? lambda.formals->formals.size() : 0);
|
2010-04-14 14:42:32 +00:00
|
|
|
|
Env & env2(allocEnv(size));
|
2010-03-30 13:47:59 +00:00
|
|
|
|
env2.up = fun.lambda.env;
|
|
|
|
|
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t displ = 0;
|
2010-04-14 14:42:32 +00:00
|
|
|
|
|
2021-10-06 15:08:08 +00:00
|
|
|
|
if (!lambda.hasFormals())
|
2010-10-22 15:51:52 +00:00
|
|
|
|
env2.values[displ++] = &arg;
|
2010-03-30 13:47:59 +00:00
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
else {
|
2014-04-04 17:11:40 +00:00
|
|
|
|
forceAttrs(arg, pos);
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2013-11-12 11:51:59 +00:00
|
|
|
|
if (!lambda.arg.empty())
|
2010-10-22 15:51:52 +00:00
|
|
|
|
env2.values[displ++] = &arg;
|
2010-03-30 13:47:59 +00: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. */
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t attrsUsed = 0;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : lambda.formals->formals) {
|
|
|
|
|
Bindings::iterator j = arg.attrs->find(i.name);
|
2010-03-30 13:47:59 +00:00
|
|
|
|
if (j == arg.attrs->end()) {
|
2020-05-12 17:27:37 +00:00
|
|
|
|
if (!i.def) throwTypeError(pos, "%1% called without required argument '%2%'",
|
|
|
|
|
lambda, i.name);
|
2015-07-17 17:24:28 +00:00
|
|
|
|
env2.values[displ++] = i.def->maybeThunk(*this, env2);
|
2010-03-30 13:47:59 +00:00
|
|
|
|
} else {
|
|
|
|
|
attrsUsed++;
|
2010-10-24 00:41:29 +00:00
|
|
|
|
env2.values[displ++] = j->value;
|
2010-03-30 13:47:59 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check that each actual argument is listed as a formal
|
2013-05-16 15:56:14 +00:00
|
|
|
|
argument (unless the attribute match specifies a `...'). */
|
2013-11-12 11:51:59 +00:00
|
|
|
|
if (!lambda.formals->ellipsis && attrsUsed != arg.attrs->size()) {
|
2013-05-16 15:56:14 +00:00
|
|
|
|
/* Nope, so show the first unexpected argument to the
|
|
|
|
|
user. */
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : *arg.attrs)
|
|
|
|
|
if (lambda.formals->argNames.find(i.name) == lambda.formals->argNames.end())
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "%1% called with unexpected argument '%2%'", lambda, i.name);
|
2013-05-16 15:56:14 +00:00
|
|
|
|
abort(); // can't happen
|
|
|
|
|
}
|
2010-03-30 13:47:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
2012-08-13 03:41:48 +00:00
|
|
|
|
nrFunctionCalls++;
|
2013-11-12 11:51:59 +00:00
|
|
|
|
if (countCalls) incrFunctionCall(&lambda);
|
|
|
|
|
|
|
|
|
|
/* Evaluate the body. This is conditional on showTrace, because
|
|
|
|
|
catching exceptions makes this function not tail-recursive. */
|
2020-07-02 15:04:31 +00:00
|
|
|
|
if (loggerSettings.showTrace.get())
|
2013-11-12 11:51:59 +00:00
|
|
|
|
try {
|
|
|
|
|
lambda.body->eval(*this, env2, v);
|
|
|
|
|
} catch (Error & e) {
|
2020-08-05 13:41:51 +00:00
|
|
|
|
addErrorTrace(e, lambda.pos, "while evaluating %s",
|
|
|
|
|
(lambda.name.set()
|
|
|
|
|
? "'" + (string) lambda.name + "'"
|
|
|
|
|
: "anonymous lambda"));
|
2020-06-30 21:44:19 +00:00
|
|
|
|
addErrorTrace(e, pos, "from call site%s", "");
|
2013-11-12 11:51:59 +00:00
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
else
|
2012-02-04 13:50:25 +00:00
|
|
|
|
fun.lambda.fun->body->eval(*this, env2, v);
|
2013-11-07 17:04:36 +00: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 13:47:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-10-22 14:47:42 +00:00
|
|
|
|
void EvalState::autoCallFunction(Bindings & args, Value & fun, Value & res)
|
2010-04-07 15:47:06 +00:00
|
|
|
|
{
|
|
|
|
|
forceValue(fun);
|
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (fun.type() == nAttrs) {
|
2015-11-25 16:56:14 +00:00
|
|
|
|
auto found = fun.attrs->find(sFunctor);
|
|
|
|
|
if (found != fun.attrs->end()) {
|
|
|
|
|
Value * v = allocValue();
|
|
|
|
|
callFunction(*found->value, fun, *v, noPos);
|
|
|
|
|
forceValue(*v);
|
|
|
|
|
return autoCallFunction(args, *v, res);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-06 15:08:08 +00:00
|
|
|
|
if (!fun.isLambda() || !fun.lambda.fun->hasFormals()) {
|
2010-04-07 15:47:06 +00:00
|
|
|
|
res = fun;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-09 14:09:31 +00:00
|
|
|
|
Value * actualArgs = allocValue();
|
2020-09-11 17:21:40 +00:00
|
|
|
|
mkAttrs(*actualArgs, std::max(static_cast<uint32_t>(fun.lambda.fun->formals->formals.size()), args.size()));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
|
2020-08-27 16:28:12 +00:00
|
|
|
|
if (fun.lambda.fun->formals->ellipsis) {
|
|
|
|
|
// If the formals have an ellipsis (eg the function accepts extra args) pass
|
|
|
|
|
// all available automatic arguments (which includes arguments specified on
|
|
|
|
|
// the command line via --arg/--argstr)
|
|
|
|
|
for (auto& v : args) {
|
|
|
|
|
actualArgs->attrs->push_back(v);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// Otherwise, only pass the arguments that the function accepts
|
|
|
|
|
for (auto & i : fun.lambda.fun->formals->formals) {
|
|
|
|
|
Bindings::iterator j = args.find(i.name);
|
|
|
|
|
if (j != args.end()) {
|
|
|
|
|
actualArgs->attrs->push_back(*j);
|
|
|
|
|
} else if (!i.def) {
|
2020-11-11 18:05:21 +00:00
|
|
|
|
throwMissingArgumentError(i.pos, R"(cannot evaluate a function that has an argument without a value ('%1%')
|
|
|
|
|
|
2021-02-22 14:24:14 +00:00
|
|
|
|
Nix attempted to evaluate a function as a top level expression; in
|
|
|
|
|
this case it must have its arguments supplied either by default
|
|
|
|
|
values, or passed explicitly with '--arg' or '--argstr'. See
|
|
|
|
|
https://nixos.org/manual/nix/stable/#ss-functions.)", i.name);
|
2020-11-11 18:05:21 +00:00
|
|
|
|
|
2020-08-27 16:28:12 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2010-04-07 15:47:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
2012-11-09 14:09:31 +00:00
|
|
|
|
actualArgs->attrs->sort();
|
2010-10-24 19:52:33 +00:00
|
|
|
|
|
2014-04-04 15:53:52 +00:00
|
|
|
|
callFunction(fun, *actualArgs, res, noPos);
|
2010-04-07 15:47:06 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprWith::eval(EvalState & state, Env & env, Value & v)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2010-04-14 15:01:04 +00:00
|
|
|
|
Env & env2(state.allocEnv(1));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
env2.up = &env;
|
2010-04-22 15:08:09 +00:00
|
|
|
|
env2.prevWith = prevWith;
|
2018-05-22 14:02:32 +00:00
|
|
|
|
env2.type = Env::HasWithExpr;
|
2013-07-31 10:44:21 +00:00
|
|
|
|
env2.values[0] = (Value *) attrs;
|
2010-04-14 15:01:04 +00:00
|
|
|
|
|
2012-02-04 13:50:25 +00:00
|
|
|
|
body->eval(state, env2, v);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprIf::eval(EvalState & state, Env & env, Value & v)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2020-04-09 07:45:15 +00:00
|
|
|
|
(state.evalBool(env, cond, pos) ? then : else_)->eval(state, env, v);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-04-12 21:21:24 +00:00
|
|
|
|
void ExprAssert::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2020-01-11 14:06:57 +00:00
|
|
|
|
if (!state.evalBool(env, cond, pos)) {
|
|
|
|
|
std::ostringstream out;
|
|
|
|
|
cond->show(out);
|
2020-11-03 13:44:52 +00:00
|
|
|
|
throwAssertionError(pos, "assertion '%1%' failed", out.str());
|
2020-01-11 14:06:57 +00:00
|
|
|
|
}
|
2012-02-04 13:50:25 +00:00
|
|
|
|
body->eval(state, env, v);
|
2010-04-12 21:21:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-04-12 21:21:24 +00:00
|
|
|
|
void ExprOpNot::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
mkBool(v, !state.evalBool(env, e));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void ExprOpEq::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
mkBool(v, state.eqValues(v1, v2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpNEq::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
mkBool(v, !state.eqValues(v1, v2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpAnd::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 20:43:52 +00:00
|
|
|
|
mkBool(v, state.evalBool(env, e1, pos) && state.evalBool(env, e2, pos));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpOr::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 20:43:52 +00:00
|
|
|
|
mkBool(v, state.evalBool(env, e1, pos) || state.evalBool(env, e2, pos));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpImpl::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2014-04-04 20:43:52 +00:00
|
|
|
|
mkBool(v, !state.evalBool(env, e1, pos) || state.evalBool(env, e2, pos));
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpUpdate::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2010-08-02 16:31:05 +00:00
|
|
|
|
Value v1, v2;
|
|
|
|
|
state.evalAttrs(env, e1, v1);
|
2010-04-16 15:13:47 +00:00
|
|
|
|
state.evalAttrs(env, e2, v2);
|
2010-08-02 16:31:05 +00:00
|
|
|
|
|
2010-10-20 15:48:00 +00:00
|
|
|
|
state.nrOpUpdates++;
|
|
|
|
|
|
2010-08-02 16:31:05 +00:00
|
|
|
|
if (v1.attrs->size() == 0) { v = v2; return; }
|
|
|
|
|
if (v2.attrs->size() == 0) { v = v1; return; }
|
|
|
|
|
|
2010-10-24 20:09:37 +00:00
|
|
|
|
state.mkAttrs(v, v1.attrs->size() + v2.attrs->size());
|
2010-10-24 19:52:33 +00:00
|
|
|
|
|
2013-10-24 14:41:04 +00:00
|
|
|
|
/* Merge the sets, preferring values from the second set. Make
|
|
|
|
|
sure to keep the resulting vector in sorted order. */
|
2010-10-24 19:52:33 +00:00
|
|
|
|
Bindings::iterator i = v1.attrs->begin();
|
|
|
|
|
Bindings::iterator j = v2.attrs->begin();
|
2010-08-02 16:31:05 +00:00
|
|
|
|
|
2010-10-24 19:52:33 +00: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 15:48:00 +00:00
|
|
|
|
|
2010-10-24 19:52:33 +00:00
|
|
|
|
while (i != v1.attrs->end()) v.attrs->push_back(*i++);
|
|
|
|
|
while (j != v2.attrs->end()) v.attrs->push_back(*j++);
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-10-20 15:48:00 +00:00
|
|
|
|
state.nrOpUpdateValuesCopied += v.attrs->size();
|
2010-04-12 18:30:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void ExprOpConcatLists::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2012-02-04 13:50:25 +00:00
|
|
|
|
Value v1; e1->eval(state, env, v1);
|
|
|
|
|
Value v2; e2->eval(state, env, v2);
|
2012-08-13 05:53:10 +00:00
|
|
|
|
Value * lists[2] = { &v1, &v2 };
|
2014-04-04 20:43:52 +00:00
|
|
|
|
state.concatLists(v, 2, lists, pos);
|
2012-08-13 05:53:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2018-05-02 11:56:34 +00:00
|
|
|
|
void EvalState::concatLists(Value & v, size_t nrLists, Value * * lists, const Pos & pos)
|
2012-08-13 05:53:10 +00:00
|
|
|
|
{
|
|
|
|
|
nrListConcats++;
|
2012-08-13 18:58:54 +00:00
|
|
|
|
|
|
|
|
|
Value * nonEmpty = 0;
|
2018-05-02 11:56:34 +00:00
|
|
|
|
size_t len = 0;
|
|
|
|
|
for (size_t n = 0; n < nrLists; ++n) {
|
2014-04-04 20:43:52 +00:00
|
|
|
|
forceList(*lists[n], pos);
|
2018-05-02 11:56:34 +00:00
|
|
|
|
auto l = lists[n]->listSize();
|
2012-08-13 18:58:54 +00:00
|
|
|
|
len += l;
|
|
|
|
|
if (l) nonEmpty = lists[n];
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 20:05:09 +00:00
|
|
|
|
if (nonEmpty && len == nonEmpty->listSize()) {
|
2012-08-13 18:58:54 +00:00
|
|
|
|
v = *nonEmpty;
|
|
|
|
|
return;
|
2012-08-13 05:53:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mkList(v, len);
|
2015-07-23 20:05:09 +00:00
|
|
|
|
auto out = v.listElems();
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t n = 0, pos = 0; n < nrLists; ++n) {
|
|
|
|
|
auto l = lists[n]->listSize();
|
2018-03-15 03:44:02 +00:00
|
|
|
|
if (l)
|
|
|
|
|
memcpy(out + pos, lists[n]->listElems(), l * sizeof(Value *));
|
2012-08-13 05:53:10 +00:00
|
|
|
|
pos += l;
|
|
|
|
|
}
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-12 21:21:24 +00:00
|
|
|
|
void ExprConcatStrings::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
|
|
|
|
PathSet context;
|
|
|
|
|
std::ostringstream s;
|
2013-08-19 10:35:03 +00:00
|
|
|
|
NixInt n = 0;
|
2016-01-04 23:40:40 +00:00
|
|
|
|
NixFloat nf = 0;
|
2013-08-02 15:21:17 +00:00
|
|
|
|
|
2013-10-16 22:39:59 +00:00
|
|
|
|
bool first = !forceString;
|
2020-12-17 13:42:52 +00:00
|
|
|
|
ValueType firstType = nString;
|
2010-04-12 21:21:24 +00:00
|
|
|
|
|
2015-07-17 17:24:28 +00:00
|
|
|
|
for (auto & i : *es) {
|
2013-08-02 15:21:17 +00:00
|
|
|
|
Value vTmp;
|
2015-07-17 17:24:28 +00:00
|
|
|
|
i->eval(state, env, vTmp);
|
2010-04-12 21:21:24 +00: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) {
|
2020-12-17 13:45:45 +00:00
|
|
|
|
firstType = vTmp.type();
|
2010-04-12 21:21:24 +00:00
|
|
|
|
}
|
2012-08-13 19:10:29 +00:00
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
if (firstType == nInt) {
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (vTmp.type() == nInt) {
|
2016-01-04 23:40:40 +00:00
|
|
|
|
n += vTmp.integer;
|
2020-12-17 13:45:45 +00:00
|
|
|
|
} else if (vTmp.type() == nFloat) {
|
2016-01-04 23:40:40 +00:00
|
|
|
|
// Upgrade the type from int to float;
|
2020-12-12 01:09:10 +00:00
|
|
|
|
firstType = nFloat;
|
2016-01-04 23:40:40 +00:00
|
|
|
|
nf = n;
|
|
|
|
|
nf += vTmp.fpoint;
|
|
|
|
|
} else
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(pos, "cannot add %1% to an integer", showType(vTmp));
|
2020-12-12 01:09:10 +00:00
|
|
|
|
} else if (firstType == nFloat) {
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (vTmp.type() == nInt) {
|
2016-01-04 23:40:40 +00:00
|
|
|
|
nf += vTmp.integer;
|
2020-12-17 13:45:45 +00:00
|
|
|
|
} else if (vTmp.type() == nFloat) {
|
2016-01-04 23:40:40 +00:00
|
|
|
|
nf += vTmp.fpoint;
|
|
|
|
|
} else
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(pos, "cannot add %1% to a float", showType(vTmp));
|
2013-08-02 15:21:17 +00:00
|
|
|
|
} else
|
2021-07-29 16:03:07 +00:00
|
|
|
|
/* skip canonization of first path, which would only be not
|
|
|
|
|
canonized in the first place if it's coming from a ./${foo} type
|
|
|
|
|
path */
|
|
|
|
|
s << state.coerceToString(pos, vTmp, context, false, firstType == nString, !first);
|
|
|
|
|
|
|
|
|
|
first = false;
|
2010-04-12 21:21:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
if (firstType == nInt)
|
2013-08-02 15:21:17 +00:00
|
|
|
|
mkInt(v, n);
|
2020-12-12 01:09:10 +00:00
|
|
|
|
else if (firstType == nFloat)
|
2016-01-04 23:40:40 +00:00
|
|
|
|
mkFloat(v, nf);
|
2020-12-12 01:09:10 +00:00
|
|
|
|
else if (firstType == nPath) {
|
2013-08-02 15:21:17 +00:00
|
|
|
|
if (!context.empty())
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(pos, "a string that refers to a store path cannot be appended to a path");
|
2015-02-19 13:39:25 +00:00
|
|
|
|
auto path = canonPath(s.str());
|
|
|
|
|
mkPath(v, path.c_str());
|
2013-08-02 15:21:17 +00:00
|
|
|
|
} else
|
2010-04-12 21:21:24 +00:00
|
|
|
|
mkString(v, s.str(), context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-18 19:14:54 +00:00
|
|
|
|
void ExprPos::eval(EvalState & state, Env & env, Value & v)
|
|
|
|
|
{
|
2021-08-29 16:09:13 +00:00
|
|
|
|
state.mkPos(v, ptr(&pos));
|
2013-11-18 19:14:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-09-22 13:03:59 +00:00
|
|
|
|
void EvalState::forceValueDeep(Value & v)
|
2010-04-07 13:55:46 +00:00
|
|
|
|
{
|
2014-09-22 13:16:09 +00:00
|
|
|
|
std::set<const Value *> seen;
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2014-09-22 13:16:09 +00:00
|
|
|
|
std::function<void(Value & v)> recurse;
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2014-09-22 13:16:09 +00:00
|
|
|
|
recurse = [&](Value & v) {
|
2019-10-09 13:51:52 +00:00
|
|
|
|
if (!seen.insert(&v).second) return;
|
2014-09-22 13:16:09 +00:00
|
|
|
|
|
|
|
|
|
forceValue(v);
|
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nAttrs) {
|
2015-03-06 14:10:12 +00:00
|
|
|
|
for (auto & i : *v.attrs)
|
|
|
|
|
try {
|
|
|
|
|
recurse(*i.value);
|
|
|
|
|
} catch (Error & e) {
|
2020-06-19 19:44:08 +00:00
|
|
|
|
addErrorTrace(e, *i.pos, "while evaluating the attribute '%1%'", i.name);
|
2015-03-06 14:10:12 +00:00
|
|
|
|
throw;
|
|
|
|
|
}
|
2014-09-22 13:16:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-07-23 20:05:09 +00:00
|
|
|
|
else if (v.isList()) {
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t n = 0; n < v.listSize(); ++n)
|
2015-07-23 20:05:09 +00:00
|
|
|
|
recurse(*v.listElems()[n]);
|
2014-09-22 13:16:09 +00:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
recurse(v);
|
2010-04-07 13:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 16:58:15 +00:00
|
|
|
|
NixInt EvalState::forceInt(Value & v, const Pos & pos)
|
2010-03-29 14:37:56 +00:00
|
|
|
|
{
|
2015-07-31 15:32:25 +00:00
|
|
|
|
forceValue(v, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nInt)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while an integer was expected", v);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return v.integer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-01-04 23:40:40 +00:00
|
|
|
|
NixFloat EvalState::forceFloat(Value & v, const Pos & pos)
|
|
|
|
|
{
|
|
|
|
|
forceValue(v, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nInt)
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return v.integer;
|
2020-12-17 13:45:45 +00:00
|
|
|
|
else if (v.type() != nFloat)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while a float was expected", v);
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return v.fpoint;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-08-29 15:56:35 +00:00
|
|
|
|
bool EvalState::forceBool(Value & v, const Pos & pos)
|
2010-03-31 15:38:03 +00:00
|
|
|
|
{
|
2020-04-16 10:32:07 +00:00
|
|
|
|
forceValue(v, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nBool)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while a Boolean was expected", v);
|
2010-03-31 15:38:03 +00:00
|
|
|
|
return v.boolean;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-10-08 11:22:11 +00:00
|
|
|
|
bool EvalState::isFunctor(Value & fun)
|
|
|
|
|
{
|
2020-12-17 13:45:45 +00:00
|
|
|
|
return fun.type() == nAttrs && fun.attrs->find(sFunctor) != fun.attrs->end();
|
2015-10-08 11:22:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 17:05:36 +00:00
|
|
|
|
void EvalState::forceFunction(Value & v, const Pos & pos)
|
2010-03-30 13:47:59 +00:00
|
|
|
|
{
|
2020-04-16 10:32:07 +00:00
|
|
|
|
forceValue(v, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nFunction && !isFunctor(v))
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while a function was expected", v);
|
2010-03-30 13:47:59 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 19:14:11 +00:00
|
|
|
|
string EvalState::forceString(Value & v, const Pos & pos)
|
2010-03-30 18:05:54 +00:00
|
|
|
|
{
|
2015-07-31 15:32:25 +00:00
|
|
|
|
forceValue(v, pos);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nString) {
|
2014-04-04 19:14:11 +00:00
|
|
|
|
if (pos)
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "value is %1% while a string was expected", v);
|
2014-04-04 19:14:11 +00:00
|
|
|
|
else
|
|
|
|
|
throwTypeError("value is %1% while a string was expected", v);
|
|
|
|
|
}
|
2010-03-31 15:38:03 +00:00
|
|
|
|
return string(v.string.s);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-06-29 17:08:37 +00:00
|
|
|
|
/* Decode a context string ‘!<name>!<path>’ into a pair <path,
|
|
|
|
|
name>. */
|
|
|
|
|
std::pair<string, string> decodeContext(std::string_view s)
|
|
|
|
|
{
|
|
|
|
|
if (s.at(0) == '!') {
|
|
|
|
|
size_t index = s.find("!", 1);
|
|
|
|
|
return {std::string(s.substr(index + 1)), std::string(s.substr(1, index - 1))};
|
|
|
|
|
} else
|
|
|
|
|
return {s.at(0) == '/' ? std::string(s) : std::string(s.substr(1)), ""};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-06-10 10:29:50 +00:00
|
|
|
|
void copyContext(const Value & v, PathSet & context)
|
2010-03-31 19:52:29 +00:00
|
|
|
|
{
|
2010-04-09 12:00:49 +00:00
|
|
|
|
if (v.string.context)
|
2013-09-02 14:29:15 +00:00
|
|
|
|
for (const char * * p = v.string.context; *p; ++p)
|
2010-03-31 19:52:29 +00:00
|
|
|
|
context.insert(*p);
|
2010-06-10 10:29:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-06-29 17:08:37 +00:00
|
|
|
|
std::vector<std::pair<Path, std::string>> Value::getContext()
|
|
|
|
|
{
|
|
|
|
|
std::vector<std::pair<Path, std::string>> res;
|
2020-12-17 13:42:52 +00:00
|
|
|
|
assert(internalType == tString);
|
2020-06-29 17:08:37 +00:00
|
|
|
|
if (string.context)
|
|
|
|
|
for (const char * * p = string.context; *p; ++p)
|
|
|
|
|
res.push_back(decodeContext(*p));
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-11-25 09:23:36 +00:00
|
|
|
|
string EvalState::forceString(Value & v, PathSet & context, const Pos & pos)
|
2010-06-10 10:29:50 +00:00
|
|
|
|
{
|
2014-11-25 09:23:36 +00:00
|
|
|
|
string s = forceString(v, pos);
|
2010-06-10 10:29:50 +00:00
|
|
|
|
copyContext(v, context);
|
2010-03-31 19:52:29 +00:00
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 19:14:11 +00:00
|
|
|
|
string EvalState::forceStringNoCtx(Value & v, const Pos & pos)
|
2010-03-31 15:38:03 +00:00
|
|
|
|
{
|
2014-04-04 19:14:11 +00:00
|
|
|
|
string s = forceString(v, pos);
|
|
|
|
|
if (v.string.context) {
|
|
|
|
|
if (pos)
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throwEvalError(pos, "the string '%1%' is not allowed to refer to a store path (such as '%2%')",
|
2020-05-12 17:27:37 +00:00
|
|
|
|
v.string.s, v.string.context[0]);
|
2014-04-04 19:14:11 +00:00
|
|
|
|
else
|
2017-07-30 11:27:57 +00:00
|
|
|
|
throwEvalError("the string '%1%' is not allowed to refer to a store path (such as '%2%')",
|
2014-04-04 19:14:11 +00:00
|
|
|
|
v.string.s, v.string.context[0]);
|
|
|
|
|
}
|
2010-03-31 15:38:03 +00:00
|
|
|
|
return s;
|
2010-03-30 18:05:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-04-07 13:55:46 +00:00
|
|
|
|
bool EvalState::isDerivation(Value & v)
|
|
|
|
|
{
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() != nAttrs) return false;
|
2010-04-13 12:25:42 +00:00
|
|
|
|
Bindings::iterator i = v.attrs->find(sType);
|
2011-10-27 19:06:23 +00:00
|
|
|
|
if (i == v.attrs->end()) return false;
|
|
|
|
|
forceValue(*i->value);
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (i->value->type() != nString) return false;
|
2013-10-24 01:08:34 +00:00
|
|
|
|
return strcmp(i->value->string.s, "derivation") == 0;
|
2010-04-07 13:55:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-10-27 09:15:51 +00:00
|
|
|
|
std::optional<string> EvalState::tryAttrsToString(const Pos & pos, Value & v,
|
|
|
|
|
PathSet & context, bool coerceMore, bool copyToStore)
|
|
|
|
|
{
|
|
|
|
|
auto i = v.attrs->find(sToString);
|
|
|
|
|
if (i != v.attrs->end()) {
|
|
|
|
|
Value v1;
|
|
|
|
|
callFunction(*i->value, v, v1, pos);
|
|
|
|
|
return coerceToString(pos, v1, context, coerceMore, copyToStore);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return {};
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-04 20:19:33 +00:00
|
|
|
|
string EvalState::coerceToString(const Pos & pos, Value & v, PathSet & context,
|
2021-08-31 12:02:04 +00:00
|
|
|
|
bool coerceMore, bool copyToStore, bool canonicalizePath)
|
2010-03-30 09:22:33 +00:00
|
|
|
|
{
|
2020-04-16 10:32:07 +00:00
|
|
|
|
forceValue(v, pos);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
|
|
|
|
string s;
|
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nString) {
|
2010-06-10 10:29:50 +00:00
|
|
|
|
copyContext(v, context);
|
2010-03-31 19:52:29 +00:00
|
|
|
|
return v.string.s;
|
|
|
|
|
}
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nPath) {
|
2021-08-31 12:02:04 +00:00
|
|
|
|
Path path(canonicalizePath ? canonPath(v.path) : v.path);
|
2013-11-18 23:03:11 +00:00
|
|
|
|
return copyToStore ? copyPathToStore(context, path) : path;
|
2010-03-30 09:22:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nAttrs) {
|
2019-10-27 09:15:51 +00:00
|
|
|
|
auto maybeString = tryAttrsToString(pos, v, context, coerceMore, copyToStore);
|
|
|
|
|
if (maybeString) {
|
|
|
|
|
return *maybeString;
|
2015-11-27 19:20:29 +00:00
|
|
|
|
}
|
2019-10-27 09:15:51 +00:00
|
|
|
|
auto i = v.attrs->find(sOutPath);
|
2020-05-12 17:27:37 +00:00
|
|
|
|
if (i == v.attrs->end()) throwTypeError(pos, "cannot coerce a set to a string");
|
2014-04-04 20:19:33 +00:00
|
|
|
|
return coerceToString(pos, *i->value, context, coerceMore, copyToStore);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nExternal)
|
2014-11-30 18:16:19 +00:00
|
|
|
|
return v.external->coerceToString(pos, context, coerceMore, copyToStore);
|
|
|
|
|
|
2010-03-30 09:22:33 +00:00
|
|
|
|
if (coerceMore) {
|
|
|
|
|
|
|
|
|
|
/* Note that `false' is represented as an empty string for
|
|
|
|
|
shell scripting convenience, just like `null'. */
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v.type() == nBool && v.boolean) return "1";
|
|
|
|
|
if (v.type() == nBool && !v.boolean) return "";
|
|
|
|
|
if (v.type() == nInt) return std::to_string(v.integer);
|
|
|
|
|
if (v.type() == nFloat) return std::to_string(v.fpoint);
|
|
|
|
|
if (v.type() == nNull) return "";
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
2015-07-23 20:05:09 +00:00
|
|
|
|
if (v.isList()) {
|
2010-03-30 09:22:33 +00:00
|
|
|
|
string result;
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t n = 0; n < v.listSize(); ++n) {
|
2015-07-23 20:05:09 +00:00
|
|
|
|
result += coerceToString(pos, *v.listElems()[n],
|
2010-03-30 09:22:33 +00:00
|
|
|
|
context, coerceMore, copyToStore);
|
2015-07-23 20:05:09 +00:00
|
|
|
|
if (n < v.listSize() - 1
|
2010-04-01 14:35:03 +00:00
|
|
|
|
/* !!! not quite correct */
|
2015-07-23 20:05:09 +00:00
|
|
|
|
&& (!v.listElems()[n]->isList() || v.listElems()[n]->listSize() != 0))
|
2010-04-01 14:35:03 +00:00
|
|
|
|
result += " ";
|
2010-03-30 09:22:33 +00:00
|
|
|
|
}
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwTypeError(pos, "cannot coerce %1% to a string", v);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-11-18 23:03:11 +00:00
|
|
|
|
string EvalState::copyPathToStore(PathSet & context, const Path & path)
|
|
|
|
|
{
|
|
|
|
|
if (nix::isDerivation(path))
|
2017-07-30 11:27:57 +00:00
|
|
|
|
throwEvalError("file names are not allowed to end in '%1%'", drvExtension);
|
2013-11-18 23:03:11 +00:00
|
|
|
|
|
|
|
|
|
Path dstPath;
|
2019-12-05 18:11:09 +00:00
|
|
|
|
auto i = srcToStore.find(path);
|
|
|
|
|
if (i != srcToStore.end())
|
|
|
|
|
dstPath = store->printStorePath(i->second);
|
2013-11-18 23:03:11 +00:00
|
|
|
|
else {
|
2019-12-05 18:11:09 +00:00
|
|
|
|
auto p = settings.readOnlyMode
|
|
|
|
|
? store->computeStorePathForPath(std::string(baseNameOf(path)), checkSourcePath(path)).first
|
2020-03-29 05:04:55 +00:00
|
|
|
|
: store->addToStore(std::string(baseNameOf(path)), checkSourcePath(path), FileIngestionMethod::Recursive, htSHA256, defaultPathFilter, repair);
|
2019-12-05 18:11:09 +00:00
|
|
|
|
dstPath = store->printStorePath(p);
|
|
|
|
|
srcToStore.insert_or_assign(path, std::move(p));
|
|
|
|
|
printMsg(lvlChatty, "copied source '%1%' -> '%2%'", path, dstPath);
|
2013-11-18 23:03:11 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
context.insert(dstPath);
|
|
|
|
|
return dstPath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-04-04 20:19:33 +00:00
|
|
|
|
Path EvalState::coerceToPath(const Pos & pos, Value & v, PathSet & context)
|
2010-03-30 09:22:33 +00:00
|
|
|
|
{
|
2014-04-04 20:19:33 +00:00
|
|
|
|
string path = coerceToString(pos, v, context, false, false);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
if (path == "" || path[0] != '/')
|
2020-05-12 17:27:37 +00:00
|
|
|
|
throwEvalError(pos, "string '%1%' doesn't represent an absolute path", path);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
bool EvalState::eqValues(Value & v1, Value & v2)
|
|
|
|
|
{
|
|
|
|
|
forceValue(v1);
|
|
|
|
|
forceValue(v2);
|
|
|
|
|
|
2010-04-12 09:50:20 +00:00
|
|
|
|
/* !!! Hack to support some old broken code that relies on pointer
|
2013-10-24 14:41:04 +00:00
|
|
|
|
equality tests between sets. (Specifically, builderDefs calls
|
|
|
|
|
uniqList on a list of sets.) Will remove this eventually. */
|
2010-04-12 09:50:20 +00:00
|
|
|
|
if (&v1 == &v2) return true;
|
|
|
|
|
|
2016-01-04 23:40:40 +00:00
|
|
|
|
// Special case type-compatibility between float and int
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v1.type() == nInt && v2.type() == nFloat)
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return v1.integer == v2.fpoint;
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v1.type() == nFloat && v2.type() == nInt)
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return v1.fpoint == v2.integer;
|
|
|
|
|
|
|
|
|
|
// All other types are not compatible with each other.
|
2020-12-17 13:45:45 +00:00
|
|
|
|
if (v1.type() != v2.type()) return false;
|
2010-04-12 09:50:20 +00:00
|
|
|
|
|
2020-12-17 13:45:45 +00:00
|
|
|
|
switch (v1.type()) {
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nInt:
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return v1.integer == v2.integer;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nBool:
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return v1.boolean == v2.boolean;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nString:
|
2014-06-10 12:02:56 +00:00
|
|
|
|
return strcmp(v1.string.s, v2.string.s) == 0;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nPath:
|
2010-03-31 20:09:20 +00:00
|
|
|
|
return strcmp(v1.path, v2.path) == 0;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nNull:
|
2010-03-31 09:54:12 +00:00
|
|
|
|
return true;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nList:
|
2015-07-23 20:05:09 +00:00
|
|
|
|
if (v1.listSize() != v2.listSize()) return false;
|
2018-05-02 11:56:34 +00:00
|
|
|
|
for (size_t n = 0; n < v1.listSize(); ++n)
|
2015-07-23 20:05:09 +00:00
|
|
|
|
if (!eqValues(*v1.listElems()[n], *v2.listElems()[n])) return false;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return true;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nAttrs: {
|
2013-10-24 14:41:04 +00:00
|
|
|
|
/* If both sets denote a derivation (type = "derivation"),
|
|
|
|
|
then compare their outPaths. */
|
2012-01-19 23:08:47 +00: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 13:51:01 +00:00
|
|
|
|
if (v1.attrs->size() != v2.attrs->size()) return false;
|
2012-01-19 23:08:47 +00: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 00:41:29 +00:00
|
|
|
|
if (i->name != j->name || !eqValues(*i->value, *j->value))
|
2010-05-07 12:11:05 +00:00
|
|
|
|
return false;
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2010-04-01 10:55:36 +00:00
|
|
|
|
/* Functions are incomparable. */
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nFunction:
|
2010-04-01 10:55:36 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nExternal:
|
2014-11-30 18:16:19 +00:00
|
|
|
|
return *v1.external == *v2.external;
|
|
|
|
|
|
2020-12-12 01:09:10 +00:00
|
|
|
|
case nFloat:
|
2016-01-04 23:40:40 +00:00
|
|
|
|
return v1.fpoint == v2.fpoint;
|
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
default:
|
2010-04-09 12:00:49 +00:00
|
|
|
|
throwEvalError("cannot compare %1% with %2%", showType(v1), showType(v2));
|
2010-03-29 14:37:56 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-30 15:18:20 +00:00
|
|
|
|
void EvalState::printStats()
|
2003-10-31 17:09:31 +00:00
|
|
|
|
{
|
2019-11-22 15:06:44 +00:00
|
|
|
|
bool showStats = getEnv("NIX_SHOW_STATS").value_or("0") != "0";
|
2012-02-04 13:27:11 +00:00
|
|
|
|
|
|
|
|
|
struct rusage buf;
|
|
|
|
|
getrusage(RUSAGE_SELF, &buf);
|
|
|
|
|
float cpuTime = buf.ru_utime.tv_sec + ((float) buf.ru_utime.tv_usec / 1000000);
|
|
|
|
|
|
2014-10-04 22:39:28 +00:00
|
|
|
|
uint64_t bEnvs = nrEnvs * sizeof(Env) + nrValuesInEnvs * sizeof(Value *);
|
|
|
|
|
uint64_t bLists = nrListElems * sizeof(Value *);
|
|
|
|
|
uint64_t bValues = nrValues * sizeof(Value);
|
|
|
|
|
uint64_t bAttrsets = nrAttrsets * sizeof(Bindings) + nrAttrsInAttrsets * sizeof(Attr);
|
|
|
|
|
|
2015-03-18 15:24:54 +00:00
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
|
GC_word heapSize, totalBytes;
|
|
|
|
|
GC_get_heap_usage_safe(&heapSize, 0, 0, 0, &totalBytes);
|
|
|
|
|
#endif
|
2018-09-02 21:20:18 +00:00
|
|
|
|
if (showStats) {
|
2019-11-22 15:06:44 +00:00
|
|
|
|
auto outPath = getEnv("NIX_SHOW_STATS_PATH").value_or("-");
|
2018-09-01 22:50:22 +00:00
|
|
|
|
std::fstream fs;
|
2018-09-05 19:35:58 +00:00
|
|
|
|
if (outPath != "-")
|
2018-09-01 22:50:22 +00:00
|
|
|
|
fs.open(outPath, std::fstream::out);
|
2018-09-05 19:35:58 +00:00
|
|
|
|
JSONObject topObj(outPath == "-" ? std::cerr : fs, true);
|
2018-09-01 22:50:22 +00:00
|
|
|
|
topObj.attr("cpuTime",cpuTime);
|
|
|
|
|
{
|
|
|
|
|
auto envs = topObj.object("envs");
|
|
|
|
|
envs.attr("number", nrEnvs);
|
2018-09-02 21:20:18 +00:00
|
|
|
|
envs.attr("elements", nrValuesInEnvs);
|
2018-09-01 22:50:22 +00:00
|
|
|
|
envs.attr("bytes", bEnvs);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto lists = topObj.object("list");
|
|
|
|
|
lists.attr("elements", nrListElems);
|
|
|
|
|
lists.attr("bytes", bLists);
|
|
|
|
|
lists.attr("concats", nrListConcats);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto values = topObj.object("values");
|
|
|
|
|
values.attr("number", nrValues);
|
|
|
|
|
values.attr("bytes", bValues);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto syms = topObj.object("symbols");
|
|
|
|
|
syms.attr("number", symbols.size());
|
|
|
|
|
syms.attr("bytes", symbols.totalSize());
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto sets = topObj.object("sets");
|
|
|
|
|
sets.attr("number", nrAttrsets);
|
|
|
|
|
sets.attr("bytes", bAttrsets);
|
2018-09-02 21:20:18 +00:00
|
|
|
|
sets.attr("elements", nrAttrsInAttrsets);
|
2018-09-01 22:50:22 +00:00
|
|
|
|
}
|
|
|
|
|
{
|
2018-09-05 19:57:54 +00:00
|
|
|
|
auto sizes = topObj.object("sizes");
|
2018-09-01 22:50:22 +00:00
|
|
|
|
sizes.attr("Env", sizeof(Env));
|
|
|
|
|
sizes.attr("Value", sizeof(Value));
|
|
|
|
|
sizes.attr("Bindings", sizeof(Bindings));
|
|
|
|
|
sizes.attr("Attr", sizeof(Attr));
|
|
|
|
|
}
|
|
|
|
|
topObj.attr("nrOpUpdates", nrOpUpdates);
|
|
|
|
|
topObj.attr("nrOpUpdateValuesCopied", nrOpUpdateValuesCopied);
|
|
|
|
|
topObj.attr("nrThunks", nrThunks);
|
|
|
|
|
topObj.attr("nrAvoided", nrAvoided);
|
|
|
|
|
topObj.attr("nrLookups", nrLookups);
|
|
|
|
|
topObj.attr("nrPrimOpCalls", nrPrimOpCalls);
|
|
|
|
|
topObj.attr("nrFunctionCalls", nrFunctionCalls);
|
|
|
|
|
#if HAVE_BOEHMGC
|
2018-09-05 19:57:54 +00:00
|
|
|
|
{
|
|
|
|
|
auto gc = topObj.object("gc");
|
|
|
|
|
gc.attr("heapSize", heapSize);
|
|
|
|
|
gc.attr("totalBytes", totalBytes);
|
|
|
|
|
}
|
2018-09-01 22:50:22 +00:00
|
|
|
|
#endif
|
2019-04-11 21:04:13 +00:00
|
|
|
|
|
2018-09-05 19:57:54 +00:00
|
|
|
|
if (countCalls) {
|
|
|
|
|
{
|
|
|
|
|
auto obj = topObj.object("primops");
|
|
|
|
|
for (auto & i : primOpCalls)
|
|
|
|
|
obj.attr(i.first, i.second);
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto list = topObj.list("functions");
|
|
|
|
|
for (auto & i : functionCalls) {
|
|
|
|
|
auto obj = list.object();
|
|
|
|
|
if (i.first->name.set())
|
|
|
|
|
obj.attr("name", (const string &) i.first->name);
|
|
|
|
|
else
|
|
|
|
|
obj.attr("name", nullptr);
|
|
|
|
|
if (i.first->pos) {
|
|
|
|
|
obj.attr("file", (const string &) i.first->pos.file);
|
|
|
|
|
obj.attr("line", i.first->pos.line);
|
|
|
|
|
obj.attr("column", i.first->pos.column);
|
|
|
|
|
}
|
|
|
|
|
obj.attr("count", i.second);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
{
|
|
|
|
|
auto list = topObj.list("attributes");
|
|
|
|
|
for (auto & i : attrSelects) {
|
|
|
|
|
auto obj = list.object();
|
|
|
|
|
if (i.first) {
|
|
|
|
|
obj.attr("file", (const string &) i.first.file);
|
|
|
|
|
obj.attr("line", i.first.line);
|
|
|
|
|
obj.attr("column", i.first.column);
|
|
|
|
|
}
|
|
|
|
|
obj.attr("count", i.second);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-04-11 21:04:13 +00:00
|
|
|
|
|
2019-11-22 15:06:44 +00:00
|
|
|
|
if (getEnv("NIX_SHOW_SYMBOLS").value_or("0") != "0") {
|
2019-04-11 21:04:13 +00:00
|
|
|
|
auto list = topObj.list("symbols");
|
|
|
|
|
symbols.dump([&](const std::string & s) { list.elem(s); });
|
|
|
|
|
}
|
2012-08-13 03:29:28 +00:00
|
|
|
|
}
|
2003-10-31 17:09:31 +00:00
|
|
|
|
}
|
2006-09-04 21:06:23 +00:00
|
|
|
|
|
2010-04-07 13:55:46 +00:00
|
|
|
|
|
2014-12-02 15:02:03 +00:00
|
|
|
|
string ExternalValueBase::coerceToString(const Pos & pos, PathSet & context, bool copyMore, bool copyToStore) const
|
2014-11-30 18:16:19 +00:00
|
|
|
|
{
|
2020-06-15 12:06:58 +00:00
|
|
|
|
throw TypeError({
|
2021-01-20 23:27:36 +00:00
|
|
|
|
.msg = hintfmt("cannot coerce %1% to a string", showType()),
|
2020-06-23 21:30:13 +00:00
|
|
|
|
.errPos = pos
|
2020-06-15 12:06:58 +00:00
|
|
|
|
});
|
2014-11-30 18:16:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-12-02 15:02:03 +00:00
|
|
|
|
bool ExternalValueBase::operator==(const ExternalValueBase & b) const
|
2014-11-30 18:16:19 +00:00
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-12-02 15:02:03 +00:00
|
|
|
|
std::ostream & operator << (std::ostream & str, const ExternalValueBase & v) {
|
2014-11-30 18:16:19 +00:00
|
|
|
|
return v.print(str);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-11-22 22:07:35 +00:00
|
|
|
|
EvalSettings::EvalSettings()
|
|
|
|
|
{
|
|
|
|
|
auto var = getEnv("NIX_PATH");
|
|
|
|
|
if (var) nixPath = parseNixPath(*var);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Strings EvalSettings::getDefaultNixPath()
|
|
|
|
|
{
|
|
|
|
|
Strings res;
|
2020-11-12 15:46:08 +00:00
|
|
|
|
auto add = [&](const Path & p, const std::string & s = std::string()) {
|
|
|
|
|
if (pathExists(p)) {
|
|
|
|
|
if (s.empty()) {
|
|
|
|
|
res.push_back(p);
|
|
|
|
|
} else {
|
|
|
|
|
res.push_back(s + "=" + p);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2021-04-12 16:36:14 +00:00
|
|
|
|
if (!evalSettings.restrictEval && !evalSettings.pureEval) {
|
|
|
|
|
add(getHome() + "/.nix-defexpr/channels");
|
|
|
|
|
add(settings.nixStateDir + "/profiles/per-user/root/channels/nixpkgs", "nixpkgs");
|
|
|
|
|
add(settings.nixStateDir + "/profiles/per-user/root/channels");
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-22 22:07:35 +00:00
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-27 17:02:22 +00:00
|
|
|
|
EvalSettings evalSettings;
|
|
|
|
|
|
2020-10-06 11:36:55 +00:00
|
|
|
|
static GlobalConfig::Register rEvalSettings(&evalSettings);
|
2018-03-27 17:02:22 +00:00
|
|
|
|
|
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
}
|