2012-07-18 21:59:03 +03:00
|
|
|
#pragma once
|
2003-10-30 18:48:26 +02:00
|
|
|
|
2012-01-07 19:26:33 +02:00
|
|
|
#include "value.hh"
|
2003-11-18 14:06:07 +02:00
|
|
|
#include "nixexpr.hh"
|
2010-04-13 15:25:42 +03:00
|
|
|
#include "symbol-table.hh"
|
2010-10-04 13:51:16 +03:00
|
|
|
#include "hash.hh"
|
2003-10-30 18:48:26 +02:00
|
|
|
|
2010-10-04 20:55:38 +03:00
|
|
|
#include <map>
|
|
|
|
|
2010-10-22 16:39:15 +03:00
|
|
|
#if HAVE_BOEHMGC
|
2010-10-20 14:38:30 +03:00
|
|
|
#include <gc/gc_allocator.h>
|
2010-10-22 16:39:15 +03:00
|
|
|
#endif
|
2010-10-20 14:38:30 +03:00
|
|
|
|
2003-10-30 18:48:26 +02:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
class EvalState;
|
2010-05-07 15:11:05 +03:00
|
|
|
struct Attr;
|
2010-03-29 17:37:56 +03:00
|
|
|
|
2010-10-24 03:41:29 +03:00
|
|
|
|
|
|
|
/* Attribute sets are represented as a vector of attributes, sorted by
|
|
|
|
symbol (i.e. pointer to the attribute name in the symbol table). */
|
2010-10-22 16:39:15 +03:00
|
|
|
#if HAVE_BOEHMGC
|
2010-10-24 03:41:29 +03:00
|
|
|
typedef std::vector<Attr, gc_allocator<Attr> > BindingsBase;
|
2010-10-22 16:39:15 +03:00
|
|
|
#else
|
2010-10-24 03:41:29 +03:00
|
|
|
typedef std::vector<Attr> BindingsBase;
|
2010-10-22 16:39:15 +03:00
|
|
|
#endif
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
|
|
|
2010-10-24 03:41:29 +03:00
|
|
|
class Bindings : public BindingsBase
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
iterator find(const Symbol & name);
|
2010-10-24 22:52:33 +03:00
|
|
|
void sort();
|
2010-10-24 03:41:29 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-10-23 23:07:47 +03:00
|
|
|
typedef void (* PrimOpFun) (EvalState & state, Value * * args, Value & v);
|
|
|
|
|
|
|
|
|
|
|
|
struct PrimOp
|
|
|
|
{
|
|
|
|
PrimOpFun fun;
|
|
|
|
unsigned int arity;
|
|
|
|
Symbol name;
|
|
|
|
PrimOp(PrimOpFun fun, unsigned int arity, Symbol name)
|
|
|
|
: fun(fun), arity(arity), name(name) { }
|
|
|
|
};
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
|
|
|
2010-04-14 17:42:32 +03:00
|
|
|
struct Env
|
|
|
|
{
|
|
|
|
Env * up;
|
2013-07-31 13:44:21 +03:00
|
|
|
unsigned short prevWith; // nr of levels up to next `with' environment
|
|
|
|
bool haveWithAttrs;
|
2010-10-22 18:51:52 +03:00
|
|
|
Value * values[0];
|
2010-04-14 17:42:32 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-05-07 15:11:05 +03:00
|
|
|
struct Attr
|
|
|
|
{
|
2010-10-24 03:41:29 +03:00
|
|
|
Symbol name;
|
2010-10-22 17:47:42 +03:00
|
|
|
Value * value;
|
2010-05-07 15:11:05 +03:00
|
|
|
Pos * pos;
|
2010-10-24 03:41:29 +03:00
|
|
|
Attr(Symbol name, Value * value, Pos * pos = &noPos)
|
|
|
|
: name(name), value(value), pos(pos) { };
|
2010-05-07 15:11:05 +03:00
|
|
|
Attr() : pos(&noPos) { };
|
2010-10-24 22:52:33 +03:00
|
|
|
bool operator < (const Attr & a) const
|
|
|
|
{
|
|
|
|
return name < a.name;
|
|
|
|
}
|
2010-05-07 15:11:05 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-03-31 01:39:48 +03:00
|
|
|
void mkString(Value & v, const string & s, const PathSet & context = PathSet());
|
2010-03-30 12:22:33 +03:00
|
|
|
|
2010-06-10 13:29:50 +03:00
|
|
|
void copyContext(const Value & v, PathSet & context);
|
|
|
|
|
2010-03-30 12:22:33 +03:00
|
|
|
|
2006-03-09 17:09:18 +02:00
|
|
|
/* Cache for calls to addToStore(); maps source paths to the store
|
|
|
|
paths. */
|
2006-09-05 00:06:23 +03:00
|
|
|
typedef std::map<Path, Path> SrcToStore;
|
2006-03-09 17:09:18 +02:00
|
|
|
|
2004-02-04 18:03:29 +02:00
|
|
|
struct EvalState;
|
2004-08-04 13:59:20 +03:00
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
|
2010-05-18 13:36:37 +03:00
|
|
|
std::ostream & operator << (std::ostream & str, const Value & v);
|
2004-02-04 18:03:29 +02:00
|
|
|
|
|
|
|
|
2013-09-02 17:29:15 +03:00
|
|
|
class EvalState
|
2003-10-30 18:48:26 +02:00
|
|
|
{
|
2010-03-31 18:38:03 +03:00
|
|
|
public:
|
2010-04-13 15:25:42 +03:00
|
|
|
SymbolTable symbols;
|
|
|
|
|
2010-05-15 11:10:12 +03:00
|
|
|
const Symbol sWith, sOutPath, sDrvPath, sType, sMeta, sName,
|
2012-11-28 14:49:44 +02:00
|
|
|
sSystem, sOverrides, sOutputs, sOutputName, sIgnoreNulls;
|
2013-10-17 12:47:38 +03:00
|
|
|
Symbol sDerivationNix;
|
2010-04-13 15:25:42 +03:00
|
|
|
|
2012-10-03 22:09:18 +03:00
|
|
|
/* If set, force copying files to the Nix store even if they
|
|
|
|
already exist there. */
|
|
|
|
bool repair;
|
|
|
|
|
2010-03-31 18:38:03 +03:00
|
|
|
private:
|
2013-09-02 17:29:15 +03:00
|
|
|
SrcToStore srcToStore;
|
2003-10-30 18:48:26 +02:00
|
|
|
|
2011-08-06 22:45:43 +03:00
|
|
|
/* A cache from path names to values. */
|
|
|
|
#if HAVE_BOEHMGC
|
|
|
|
typedef std::map<Path, Value, std::less<Path>, gc_allocator<std::pair<const Path, Value> > > FileEvalCache;
|
|
|
|
#else
|
|
|
|
typedef std::map<Path, Value> FileEvalCache;
|
|
|
|
#endif
|
|
|
|
FileEvalCache fileEvalCache;
|
|
|
|
|
2011-08-06 20:48:57 +03:00
|
|
|
typedef list<std::pair<string, Path> > SearchPath;
|
|
|
|
SearchPath searchPath;
|
|
|
|
SearchPath::iterator searchPathInsertionPoint;
|
2011-08-06 19:05:24 +03:00
|
|
|
|
2010-03-30 18:18:20 +03:00
|
|
|
public:
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2003-10-30 18:48:26 +02:00
|
|
|
EvalState();
|
2010-04-09 15:00:49 +03:00
|
|
|
~EvalState();
|
2004-02-04 18:03:29 +02:00
|
|
|
|
2011-08-06 19:05:24 +03:00
|
|
|
void addToSearchPath(const string & s);
|
|
|
|
|
2013-09-03 13:56:33 +03:00
|
|
|
/* Parse a Nix expression from the specified file. */
|
|
|
|
Expr * parseExprFromFile(const Path & path);
|
2011-08-06 16:02:55 +03:00
|
|
|
|
|
|
|
/* Parse a Nix expression from the specified string. */
|
2013-09-02 19:34:04 +03:00
|
|
|
Expr * parseExprFromString(const string & s, const Path & basePath, StaticEnv & staticEnv);
|
2011-08-06 16:02:55 +03:00
|
|
|
Expr * parseExprFromString(const string & s, const Path & basePath);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2010-03-30 12:22:33 +03:00
|
|
|
/* Evaluate an expression read from the given file to normal
|
|
|
|
form. */
|
|
|
|
void evalFile(const Path & path, Value & v);
|
|
|
|
|
2013-09-02 19:34:04 +03:00
|
|
|
void resetFileCache();
|
|
|
|
|
2011-08-06 19:05:24 +03:00
|
|
|
/* Look up a file in the search path. */
|
|
|
|
Path findFile(const string & path);
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
/* Evaluate an expression to normal form, storing the result in
|
|
|
|
value `v'. */
|
2010-04-12 21:30:11 +03:00
|
|
|
void eval(Expr * e, Value & v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
|
|
/* Evaluation the expression, then verify that it has the expected
|
|
|
|
type. */
|
2012-02-04 15:50:25 +02:00
|
|
|
inline bool evalBool(Env & env, Expr * e);
|
|
|
|
inline void evalAttrs(Env & env, Expr * e, Value & v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
|
|
/* If `v' is a thunk, enter it and overwrite `v' with the result
|
2010-03-30 16:47:59 +03:00
|
|
|
of the evaluation of the thunk. If `v' is a delayed function
|
|
|
|
application, call the function and overwrite `v' with the
|
|
|
|
result. Otherwise, this is a no-op. */
|
2012-02-04 15:50:25 +02:00
|
|
|
inline void forceValue(Value & v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
2010-04-07 16:55:46 +03:00
|
|
|
/* Force a value, then recursively force list elements and
|
|
|
|
attributes. */
|
|
|
|
void strictForceValue(Value & v);
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
/* Force `v', and then verify that it has the expected type. */
|
2013-08-19 13:35:03 +03:00
|
|
|
NixInt forceInt(Value & v);
|
2010-03-31 18:38:03 +03:00
|
|
|
bool forceBool(Value & v);
|
2012-02-04 15:50:25 +02:00
|
|
|
inline void forceAttrs(Value & v);
|
|
|
|
inline void forceList(Value & v);
|
2010-03-30 16:47:59 +03:00
|
|
|
void forceFunction(Value & v); // either lambda or primop
|
2010-03-31 18:38:03 +03:00
|
|
|
string forceString(Value & v);
|
2010-03-31 22:52:29 +03:00
|
|
|
string forceString(Value & v, PathSet & context);
|
2010-03-30 21:05:54 +03:00
|
|
|
string forceStringNoCtx(Value & v);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
2010-04-07 16:55:46 +03:00
|
|
|
/* Return true iff the value `v' denotes a derivation (i.e. a
|
|
|
|
set with attribute `type = "derivation"'). */
|
|
|
|
bool isDerivation(Value & v);
|
|
|
|
|
2010-03-30 12:22:33 +03:00
|
|
|
/* String coercion. Converts strings, paths and derivations to a
|
|
|
|
string. If `coerceMore' is set, also converts nulls, integers,
|
|
|
|
booleans and lists to a string. If `copyToStore' is set,
|
2013-08-14 23:32:49 +03:00
|
|
|
referenced paths are copied to the Nix store as a side effect. */
|
2010-03-30 12:22:33 +03:00
|
|
|
string coerceToString(Value & v, PathSet & context,
|
|
|
|
bool coerceMore = false, bool copyToStore = true);
|
|
|
|
|
|
|
|
/* Path coercion. Converts strings, paths and derivations to a
|
|
|
|
path. The result is guaranteed to be a canonicalised, absolute
|
|
|
|
path. Nothing is copied to the store. */
|
|
|
|
Path coerceToPath(Value & v, PathSet & context);
|
|
|
|
|
2013-09-02 19:34:04 +03:00
|
|
|
public:
|
2010-03-29 17:37:56 +03:00
|
|
|
|
|
|
|
/* The base environment, containing the builtin functions and
|
|
|
|
values. */
|
|
|
|
Env & baseEnv;
|
|
|
|
|
2010-04-15 01:59:39 +03:00
|
|
|
/* The same, but used during parsing to resolve variables. */
|
|
|
|
StaticEnv staticBaseEnv; // !!! should be private
|
|
|
|
|
|
|
|
private:
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2013-09-02 19:34:04 +03:00
|
|
|
unsigned int baseEnvDispl;
|
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
void createBaseEnv();
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2010-03-30 17:39:27 +03:00
|
|
|
void addConstant(const string & name, Value & v);
|
|
|
|
|
2004-08-04 13:59:20 +03:00
|
|
|
void addPrimOp(const string & name,
|
2010-10-23 23:07:47 +03:00
|
|
|
unsigned int arity, PrimOpFun primOp);
|
2010-03-29 17:37:56 +03:00
|
|
|
|
2013-09-03 16:45:32 +03:00
|
|
|
public:
|
|
|
|
|
|
|
|
void getBuiltin(const string & name, Value & v);
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
2013-10-08 15:24:53 +03:00
|
|
|
inline Value * lookupVar(Env * env, const ExprVar & var, bool noEval);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2010-04-13 15:25:42 +03:00
|
|
|
friend class ExprVar;
|
|
|
|
friend class ExprAttrs;
|
2010-04-13 16:42:25 +03:00
|
|
|
friend class ExprLet;
|
2010-04-13 15:25:42 +03:00
|
|
|
|
2013-09-02 19:34:04 +03:00
|
|
|
Expr * parse(const char * text, const Path & path,
|
|
|
|
const Path & basePath, StaticEnv & staticEnv);
|
2011-08-06 16:02:55 +03:00
|
|
|
|
2010-04-12 21:30:11 +03:00
|
|
|
public:
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
/* Do a deep equality test between two values. That is, list
|
|
|
|
elements and attributes are compared recursively. */
|
|
|
|
bool eqValues(Value & v1, Value & v2);
|
|
|
|
|
2010-04-07 16:55:46 +03:00
|
|
|
void callFunction(Value & fun, Value & arg, Value & v);
|
|
|
|
|
2010-04-07 18:47:06 +03:00
|
|
|
/* Automatically call a function for which each argument has a
|
|
|
|
default value or has a binding in the `args' map. */
|
2010-10-22 17:47:42 +03:00
|
|
|
void autoCallFunction(Bindings & args, Value & fun, Value & res);
|
2013-09-02 17:29:15 +03:00
|
|
|
|
2010-03-29 17:37:56 +03:00
|
|
|
/* Allocation primitives. */
|
2010-10-22 17:47:42 +03:00
|
|
|
Value * allocValue();
|
2010-04-14 17:42:32 +03:00
|
|
|
Env & allocEnv(unsigned int size);
|
2010-03-30 17:39:27 +03:00
|
|
|
|
2010-10-22 17:47:42 +03:00
|
|
|
Value * allocAttr(Value & vAttrs, const Symbol & name);
|
|
|
|
|
2010-03-30 17:39:27 +03:00
|
|
|
void mkList(Value & v, unsigned int length);
|
2010-10-24 23:09:37 +03:00
|
|
|
void mkAttrs(Value & v, unsigned int expected);
|
2010-04-12 21:30:11 +03:00
|
|
|
void mkThunk_(Value & v, Expr * expr);
|
2010-10-24 17:20:02 +03:00
|
|
|
|
2012-08-13 08:53:10 +03:00
|
|
|
void concatLists(Value & v, unsigned int nrLists, Value * * lists);
|
|
|
|
|
2010-03-30 18:18:20 +03:00
|
|
|
/* Print statistics. */
|
|
|
|
void printStats();
|
2010-04-09 15:00:49 +03:00
|
|
|
|
|
|
|
private:
|
2012-08-13 06:41:48 +03:00
|
|
|
|
2010-04-09 15:00:49 +03:00
|
|
|
unsigned long nrEnvs;
|
2010-04-15 02:48:46 +03:00
|
|
|
unsigned long nrValuesInEnvs;
|
|
|
|
unsigned long nrValues;
|
2010-04-15 03:37:36 +03:00
|
|
|
unsigned long nrListElems;
|
2010-10-20 18:48:00 +03:00
|
|
|
unsigned long nrAttrsets;
|
|
|
|
unsigned long nrOpUpdates;
|
|
|
|
unsigned long nrOpUpdateValuesCopied;
|
2012-08-13 06:41:48 +03:00
|
|
|
unsigned long nrListConcats;
|
|
|
|
unsigned long nrPrimOpCalls;
|
|
|
|
unsigned long nrFunctionCalls;
|
2012-08-13 06:29:28 +03:00
|
|
|
|
|
|
|
bool countCalls;
|
|
|
|
|
|
|
|
typedef std::map<Symbol, unsigned int> PrimOpCalls;
|
|
|
|
PrimOpCalls primOpCalls;
|
|
|
|
|
2013-08-02 21:29:23 +03:00
|
|
|
typedef std::map<ExprLambda *, unsigned int> FunctionCalls;
|
2012-08-13 06:29:28 +03:00
|
|
|
FunctionCalls functionCalls;
|
|
|
|
|
|
|
|
typedef std::map<Pos, unsigned int> AttrSelects;
|
|
|
|
AttrSelects attrSelects;
|
|
|
|
|
2010-10-20 14:38:30 +03:00
|
|
|
friend class ExprOpUpdate;
|
2012-08-13 06:41:48 +03:00
|
|
|
friend class ExprOpConcatLists;
|
2012-08-13 06:29:28 +03:00
|
|
|
friend class ExprSelect;
|
|
|
|
friend void prim_getAttr(EvalState & state, Value * * args, Value & v);
|
2003-10-30 18:48:26 +02:00
|
|
|
};
|
2010-04-07 16:55:46 +03:00
|
|
|
|
|
|
|
|
|
|
|
/* Return a string representing the type of the value `v'. */
|
2010-04-21 18:57:11 +03:00
|
|
|
string showType(const Value & v);
|
2003-10-30 18:48:26 +02:00
|
|
|
|
|
|
|
|
2013-09-03 13:56:33 +03:00
|
|
|
/* If `path' refers to a directory, then append "/default.nix". */
|
|
|
|
Path resolveExprPath(Path path);
|
|
|
|
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
}
|