2012-07-18 18:59:03 +00:00
|
|
|
|
#pragma once
|
2003-10-30 16:48:26 +00:00
|
|
|
|
|
2015-07-14 17:18:56 +00:00
|
|
|
|
#include "attr-set.hh"
|
2012-01-07 17:26:33 +00:00
|
|
|
|
#include "value.hh"
|
2003-11-18 12:06:07 +00:00
|
|
|
|
#include "nixexpr.hh"
|
2010-04-13 12:25:42 +00:00
|
|
|
|
#include "symbol-table.hh"
|
2010-10-04 10:51:16 +00:00
|
|
|
|
#include "hash.hh"
|
2003-10-30 16:48:26 +00:00
|
|
|
|
|
2010-10-04 17:55:38 +00:00
|
|
|
|
#include <map>
|
|
|
|
|
|
2003-10-30 16:48:26 +00:00
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
|
|
2016-02-04 13:48:42 +00:00
|
|
|
|
class Store;
|
2010-03-29 14:37:56 +00:00
|
|
|
|
class EvalState;
|
|
|
|
|
|
2010-10-24 00:41:29 +00:00
|
|
|
|
|
2014-04-04 16:51:01 +00:00
|
|
|
|
typedef void (* PrimOpFun) (EvalState & state, const Pos & pos, Value * * args, Value & v);
|
2010-10-23 20:07:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct PrimOp
|
|
|
|
|
{
|
|
|
|
|
PrimOpFun fun;
|
2016-04-13 09:15:45 +00:00
|
|
|
|
size_t arity;
|
2010-10-23 20:07:47 +00:00
|
|
|
|
Symbol name;
|
2016-04-13 09:15:45 +00:00
|
|
|
|
PrimOp(PrimOpFun fun, size_t arity, Symbol name)
|
2010-10-23 20:07:47 +00:00
|
|
|
|
: fun(fun), arity(arity), name(name) { }
|
|
|
|
|
};
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
|
|
|
|
|
2010-04-14 14:42:32 +00:00
|
|
|
|
struct Env
|
|
|
|
|
{
|
|
|
|
|
Env * up;
|
2014-09-22 12:46:42 +00:00
|
|
|
|
unsigned short size; // used by ‘valueSize’
|
|
|
|
|
unsigned short prevWith:15; // nr of levels up to next `with' environment
|
|
|
|
|
unsigned short haveWithAttrs:1;
|
2010-10-22 15:51:52 +00:00
|
|
|
|
Value * values[0];
|
2010-04-14 14:42:32 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2016-08-23 15:11:19 +00:00
|
|
|
|
Value & mkString(Value & v, const string & s, const PathSet & context = PathSet());
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
2010-06-10 10:29:50 +00:00
|
|
|
|
void copyContext(const Value & v, PathSet & context);
|
|
|
|
|
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
2006-03-09 15:09:18 +00:00
|
|
|
|
/* Cache for calls to addToStore(); maps source paths to the store
|
|
|
|
|
paths. */
|
2006-09-04 21:06:23 +00:00
|
|
|
|
typedef std::map<Path, Path> SrcToStore;
|
2006-03-09 15:09:18 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2010-05-18 10:36:37 +00:00
|
|
|
|
std::ostream & operator << (std::ostream & str, const Value & v);
|
2004-02-04 16:03:29 +00:00
|
|
|
|
|
|
|
|
|
|
2016-04-14 13:32:24 +00:00
|
|
|
|
typedef std::pair<std::string, std::string> SearchPathElem;
|
|
|
|
|
typedef std::list<SearchPathElem> SearchPath;
|
2014-05-26 15:02:22 +00:00
|
|
|
|
|
|
|
|
|
|
2015-03-19 19:02:37 +00:00
|
|
|
|
/* Initialise the Boehm GC, if applicable. */
|
|
|
|
|
void initGC();
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 14:29:15 +00:00
|
|
|
|
class EvalState
|
2003-10-30 16:48:26 +00:00
|
|
|
|
{
|
2010-03-31 15:38:03 +00:00
|
|
|
|
public:
|
2010-04-13 12:25:42 +00:00
|
|
|
|
SymbolTable symbols;
|
|
|
|
|
|
2013-10-28 06:34:44 +00:00
|
|
|
|
const Symbol sWith, sOutPath, sDrvPath, sType, sMeta, sName, sValue,
|
2013-11-18 19:14:54 +00:00
|
|
|
|
sSystem, sOverrides, sOutputs, sOutputName, sIgnoreNulls,
|
2016-08-29 15:28:20 +00:00
|
|
|
|
sFile, sLine, sColumn, sFunctor, sToString,
|
|
|
|
|
sRight, sWrong;
|
2013-10-17 09:47:38 +00:00
|
|
|
|
Symbol sDerivationNix;
|
2010-04-13 12:25:42 +00:00
|
|
|
|
|
2012-10-03 19:09:18 +00:00
|
|
|
|
/* If set, force copying files to the Nix store even if they
|
|
|
|
|
already exist there. */
|
2015-07-23 21:14:07 +00:00
|
|
|
|
bool repair = false;
|
2012-10-03 19:09:18 +00:00
|
|
|
|
|
2015-02-23 13:41:53 +00:00
|
|
|
|
/* If set, don't allow access to files outside of the Nix search
|
|
|
|
|
path or to environment variables. */
|
|
|
|
|
bool restricted;
|
|
|
|
|
|
2015-07-23 21:11:08 +00:00
|
|
|
|
Value vEmptySet;
|
|
|
|
|
|
2016-02-04 13:48:42 +00:00
|
|
|
|
const ref<Store> store;
|
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
|
|
|
|
|
2010-03-31 15:38:03 +00:00
|
|
|
|
private:
|
2013-09-02 14:29:15 +00:00
|
|
|
|
SrcToStore srcToStore;
|
2003-10-30 16:48:26 +00:00
|
|
|
|
|
2011-08-06 19:45:43 +00:00
|
|
|
|
/* A cache from path names to values. */
|
|
|
|
|
#if HAVE_BOEHMGC
|
2014-08-20 22:05:17 +00:00
|
|
|
|
typedef std::map<Path, Value, std::less<Path>, traceable_allocator<std::pair<const Path, Value> > > FileEvalCache;
|
2011-08-06 19:45:43 +00:00
|
|
|
|
#else
|
|
|
|
|
typedef std::map<Path, Value> FileEvalCache;
|
|
|
|
|
#endif
|
|
|
|
|
FileEvalCache fileEvalCache;
|
|
|
|
|
|
2011-08-06 17:48:57 +00:00
|
|
|
|
SearchPath searchPath;
|
2011-08-06 16:05:24 +00:00
|
|
|
|
|
2016-04-14 13:32:24 +00:00
|
|
|
|
std::map<std::string, std::pair<bool, std::string>> searchPathResolved;
|
|
|
|
|
|
2010-03-30 15:18:20 +00:00
|
|
|
|
public:
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2016-02-04 13:48:42 +00:00
|
|
|
|
EvalState(const Strings & _searchPath, ref<Store> store);
|
2010-04-09 12:00:49 +00:00
|
|
|
|
~EvalState();
|
2004-02-04 16:03:29 +00:00
|
|
|
|
|
2016-04-14 13:32:24 +00:00
|
|
|
|
void addToSearchPath(const string & s);
|
2011-08-06 16:05:24 +00:00
|
|
|
|
|
2016-08-23 15:11:19 +00:00
|
|
|
|
SearchPath getSearchPath() { return searchPath; }
|
|
|
|
|
|
2015-02-23 13:41:53 +00:00
|
|
|
|
Path checkSourcePath(const Path & path);
|
|
|
|
|
|
2013-09-03 10:56:33 +00:00
|
|
|
|
/* Parse a Nix expression from the specified file. */
|
|
|
|
|
Expr * parseExprFromFile(const Path & path);
|
Add primop ‘scopedImport’
‘scopedImport’ works like ‘import’, except that it takes a set of
attributes to be added to the lexical scope of the expression,
essentially extending or overriding the builtin variables. For
instance, the expression
scopedImport { x = 1; } ./foo.nix
where foo.nix contains ‘x’, will evaluate to 1.
This has a few applications:
* It allows getting rid of function argument specifications in package
expressions. For instance, a package expression like:
{ stdenv, fetchurl, libfoo }:
stdenv.mkDerivation { ... buildInputs = [ libfoo ]; }
can now we written as just
stdenv.mkDerivation { ... buildInputs = [ libfoo ]; }
and imported in all-packages.nix as:
bar = scopedImport pkgs ./bar.nix;
So whereas we once had dependencies listed in three places
(buildInputs, the function, and the call site), they now only need
to appear in one place.
* It allows overriding builtin functions. For instance, to trace all
calls to ‘map’:
let
overrides = {
map = f: xs: builtins.trace "map called!" (map f xs);
# Ensure that our override gets propagated by calls to
# import/scopedImport.
import = fn: scopedImport overrides fn;
scopedImport = attrs: fn: scopedImport (overrides // attrs) fn;
# Also update ‘builtins’.
builtins = builtins // overrides;
};
in scopedImport overrides ./bla.nix
* Similarly, it allows extending the set of builtin functions. For
instance, during Nixpkgs/NixOS evaluation, the Nixpkgs library
functions could be added to the default scope.
There is a downside: calls to scopedImport are not memoized, unlike
import. So importing a file multiple times leads to multiple parsings
/ evaluations. It would be possible to construct the AST only once,
but that would require careful handling of variables/environments.
2014-05-26 11:46:11 +00:00
|
|
|
|
Expr * parseExprFromFile(const Path & path, StaticEnv & staticEnv);
|
2011-08-06 13:02:55 +00:00
|
|
|
|
|
|
|
|
|
/* Parse a Nix expression from the specified string. */
|
2013-09-02 16:34:04 +00:00
|
|
|
|
Expr * parseExprFromString(const string & s, const Path & basePath, StaticEnv & staticEnv);
|
2011-08-06 13:02:55 +00:00
|
|
|
|
Expr * parseExprFromString(const string & s, const Path & basePath);
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-03-30 09:22:33 +00:00
|
|
|
|
/* Evaluate an expression read from the given file to normal
|
|
|
|
|
form. */
|
|
|
|
|
void evalFile(const Path & path, Value & v);
|
|
|
|
|
|
2013-09-02 16:34:04 +00:00
|
|
|
|
void resetFileCache();
|
|
|
|
|
|
2011-08-06 16:05:24 +00:00
|
|
|
|
/* Look up a file in the search path. */
|
|
|
|
|
Path findFile(const string & path);
|
2015-01-07 12:43:55 +00:00
|
|
|
|
Path findFile(SearchPath & searchPath, const string & path, const Pos & pos = noPos);
|
2011-08-06 16:05:24 +00:00
|
|
|
|
|
2016-04-14 13:32:24 +00:00
|
|
|
|
/* If the specified search path element is a URI, download it. */
|
|
|
|
|
std::pair<bool, std::string> resolveSearchPathElem(const SearchPathElem & elem);
|
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
/* Evaluate an expression to normal form, storing the result in
|
|
|
|
|
value `v'. */
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void eval(Expr * e, Value & v);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
|
|
|
|
/* Evaluation the expression, then verify that it has the expected
|
|
|
|
|
type. */
|
2012-02-04 13:50:25 +00:00
|
|
|
|
inline bool evalBool(Env & env, Expr * e);
|
2014-04-04 20:43:52 +00:00
|
|
|
|
inline bool evalBool(Env & env, Expr * e, const Pos & pos);
|
2012-02-04 13:50:25 +00:00
|
|
|
|
inline void evalAttrs(Env & env, Expr * e, Value & v);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
|
|
|
|
/* If `v' is a thunk, enter it and overwrite `v' with the result
|
2010-03-30 13:47:59 +00: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. */
|
2015-07-31 15:32:25 +00:00
|
|
|
|
inline void forceValue(Value & v, const Pos & pos = noPos);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2010-04-07 13:55:46 +00:00
|
|
|
|
/* Force a value, then recursively force list elements and
|
|
|
|
|
attributes. */
|
2014-09-22 13:03:59 +00:00
|
|
|
|
void forceValueDeep(Value & v);
|
2010-04-07 13:55:46 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
/* Force `v', and then verify that it has the expected type. */
|
2014-04-04 16:58:15 +00:00
|
|
|
|
NixInt forceInt(Value & v, const Pos & pos);
|
2016-01-04 23:40:40 +00:00
|
|
|
|
NixFloat forceFloat(Value & v, const Pos & pos);
|
2016-08-29 15:56:35 +00:00
|
|
|
|
bool forceBool(Value & v, const Pos & pos);
|
2012-02-04 13:50:25 +00:00
|
|
|
|
inline void forceAttrs(Value & v);
|
2014-04-04 17:11:40 +00:00
|
|
|
|
inline void forceAttrs(Value & v, const Pos & pos);
|
2012-02-04 13:50:25 +00:00
|
|
|
|
inline void forceList(Value & v);
|
2014-04-04 17:05:36 +00:00
|
|
|
|
inline void forceList(Value & v, const Pos & pos);
|
|
|
|
|
void forceFunction(Value & v, const Pos & pos); // either lambda or primop
|
2014-04-04 19:14:11 +00:00
|
|
|
|
string forceString(Value & v, const Pos & pos = noPos);
|
2014-11-25 09:23:36 +00:00
|
|
|
|
string forceString(Value & v, PathSet & context, const Pos & pos = noPos);
|
2014-04-04 19:14:11 +00:00
|
|
|
|
string forceStringNoCtx(Value & v, const Pos & pos = noPos);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2010-04-07 13:55:46 +00: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 09:22:33 +00: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 20:32:49 +00:00
|
|
|
|
referenced paths are copied to the Nix store as a side effect. */
|
2014-04-04 20:19:33 +00:00
|
|
|
|
string coerceToString(const Pos & pos, Value & v, PathSet & context,
|
2010-03-30 09:22:33 +00:00
|
|
|
|
bool coerceMore = false, bool copyToStore = true);
|
|
|
|
|
|
2013-11-18 23:03:11 +00:00
|
|
|
|
string copyPathToStore(PathSet & context, const Path & path);
|
|
|
|
|
|
2010-03-30 09:22:33 +00:00
|
|
|
|
/* 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. */
|
2014-04-04 20:19:33 +00:00
|
|
|
|
Path coerceToPath(const Pos & pos, Value & v, PathSet & context);
|
2010-03-30 09:22:33 +00:00
|
|
|
|
|
2013-09-02 16:34:04 +00:00
|
|
|
|
public:
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
|
|
|
|
/* The base environment, containing the builtin functions and
|
|
|
|
|
values. */
|
|
|
|
|
Env & baseEnv;
|
|
|
|
|
|
2010-04-14 22:59:39 +00:00
|
|
|
|
/* The same, but used during parsing to resolve variables. */
|
|
|
|
|
StaticEnv staticBaseEnv; // !!! should be private
|
|
|
|
|
|
|
|
|
|
private:
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2015-07-23 21:14:07 +00:00
|
|
|
|
unsigned int baseEnvDispl = 0;
|
2013-09-02 16:34:04 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
void createBaseEnv();
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-03-30 14:39:27 +00:00
|
|
|
|
void addConstant(const string & name, Value & v);
|
|
|
|
|
|
2004-08-04 10:59:20 +00:00
|
|
|
|
void addPrimOp(const string & name,
|
2010-10-23 20:07:47 +00:00
|
|
|
|
unsigned int arity, PrimOpFun primOp);
|
2010-03-29 14:37:56 +00:00
|
|
|
|
|
2013-09-03 13:45:32 +00:00
|
|
|
|
public:
|
|
|
|
|
|
2016-08-23 15:11:19 +00:00
|
|
|
|
Value & getBuiltin(const string & name);
|
2013-09-03 13:45:32 +00:00
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
2013-10-08 12:24:53 +00:00
|
|
|
|
inline Value * lookupVar(Env * env, const ExprVar & var, bool noEval);
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2014-01-21 17:29:55 +00:00
|
|
|
|
friend struct ExprVar;
|
|
|
|
|
friend struct ExprAttrs;
|
|
|
|
|
friend struct ExprLet;
|
2010-04-13 12:25:42 +00:00
|
|
|
|
|
2013-09-02 16:34:04 +00:00
|
|
|
|
Expr * parse(const char * text, const Path & path,
|
|
|
|
|
const Path & basePath, StaticEnv & staticEnv);
|
2011-08-06 13:02:55 +00:00
|
|
|
|
|
2010-04-12 18:30:11 +00:00
|
|
|
|
public:
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
/* Do a deep equality test between two values. That is, list
|
|
|
|
|
elements and attributes are compared recursively. */
|
|
|
|
|
bool eqValues(Value & v1, Value & v2);
|
|
|
|
|
|
2015-09-06 23:03:23 +00:00
|
|
|
|
bool isFunctor(Value & fun);
|
|
|
|
|
|
2014-04-04 15:53:52 +00:00
|
|
|
|
void callFunction(Value & fun, Value & arg, Value & v, const Pos & pos);
|
2014-04-04 16:51:01 +00:00
|
|
|
|
void callPrimOp(Value & fun, Value & arg, Value & v, const Pos & pos);
|
2010-04-07 13:55:46 +00:00
|
|
|
|
|
2010-04-07 15:47:06 +00:00
|
|
|
|
/* Automatically call a function for which each argument has a
|
|
|
|
|
default value or has a binding in the `args' map. */
|
2010-10-22 14:47:42 +00:00
|
|
|
|
void autoCallFunction(Bindings & args, Value & fun, Value & res);
|
2013-09-02 14:29:15 +00:00
|
|
|
|
|
2010-03-29 14:37:56 +00:00
|
|
|
|
/* Allocation primitives. */
|
2010-10-22 14:47:42 +00:00
|
|
|
|
Value * allocValue();
|
2010-04-14 14:42:32 +00:00
|
|
|
|
Env & allocEnv(unsigned int size);
|
2010-03-30 14:39:27 +00:00
|
|
|
|
|
2010-10-22 14:47:42 +00:00
|
|
|
|
Value * allocAttr(Value & vAttrs, const Symbol & name);
|
|
|
|
|
|
2014-09-19 14:49:41 +00:00
|
|
|
|
Bindings * allocBindings(Bindings::size_t capacity);
|
|
|
|
|
|
2010-03-30 14:39:27 +00:00
|
|
|
|
void mkList(Value & v, unsigned int length);
|
2015-07-23 21:11:08 +00:00
|
|
|
|
void mkAttrs(Value & v, unsigned int capacity);
|
2010-04-12 18:30:11 +00:00
|
|
|
|
void mkThunk_(Value & v, Expr * expr);
|
2013-11-18 21:22:35 +00:00
|
|
|
|
void mkPos(Value & v, Pos * pos);
|
2010-10-24 14:20:02 +00:00
|
|
|
|
|
2014-04-04 20:43:52 +00:00
|
|
|
|
void concatLists(Value & v, unsigned int nrLists, Value * * lists, const Pos & pos);
|
2012-08-13 05:53:10 +00:00
|
|
|
|
|
2010-03-30 15:18:20 +00:00
|
|
|
|
/* Print statistics. */
|
|
|
|
|
void printStats();
|
2010-04-09 12:00:49 +00:00
|
|
|
|
|
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
|
|
|
|
void realiseContext(const PathSet & context);
|
|
|
|
|
|
2010-04-09 12:00:49 +00:00
|
|
|
|
private:
|
2012-08-13 03:41:48 +00:00
|
|
|
|
|
2015-07-23 21:14:07 +00:00
|
|
|
|
unsigned long nrEnvs = 0;
|
|
|
|
|
unsigned long nrValuesInEnvs = 0;
|
|
|
|
|
unsigned long nrValues = 0;
|
|
|
|
|
unsigned long nrListElems = 0;
|
|
|
|
|
unsigned long nrAttrsets = 0;
|
|
|
|
|
unsigned long nrAttrsInAttrsets = 0;
|
|
|
|
|
unsigned long nrOpUpdates = 0;
|
|
|
|
|
unsigned long nrOpUpdateValuesCopied = 0;
|
|
|
|
|
unsigned long nrListConcats = 0;
|
|
|
|
|
unsigned long nrPrimOpCalls = 0;
|
|
|
|
|
unsigned long nrFunctionCalls = 0;
|
2012-08-13 03:29:28 +00:00
|
|
|
|
|
|
|
|
|
bool countCalls;
|
|
|
|
|
|
|
|
|
|
typedef std::map<Symbol, unsigned int> PrimOpCalls;
|
|
|
|
|
PrimOpCalls primOpCalls;
|
|
|
|
|
|
2013-08-02 18:29:23 +00:00
|
|
|
|
typedef std::map<ExprLambda *, unsigned int> FunctionCalls;
|
2012-08-13 03:29:28 +00:00
|
|
|
|
FunctionCalls functionCalls;
|
|
|
|
|
|
2013-11-07 17:04:36 +00:00
|
|
|
|
void incrFunctionCall(ExprLambda * fun);
|
|
|
|
|
|
2012-08-13 03:29:28 +00:00
|
|
|
|
typedef std::map<Pos, unsigned int> AttrSelects;
|
|
|
|
|
AttrSelects attrSelects;
|
|
|
|
|
|
2014-01-21 17:29:55 +00:00
|
|
|
|
friend struct ExprOpUpdate;
|
|
|
|
|
friend struct ExprOpConcatLists;
|
|
|
|
|
friend struct ExprSelect;
|
2014-04-04 16:51:01 +00:00
|
|
|
|
friend void prim_getAttr(EvalState & state, const Pos & pos, Value * * args, Value & v);
|
2003-10-30 16:48:26 +00:00
|
|
|
|
};
|
2010-04-07 13:55:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a string representing the type of the value `v'. */
|
2010-04-21 15:57:11 +00:00
|
|
|
|
string showType(const Value & v);
|
2003-10-30 16:48:26 +00:00
|
|
|
|
|
|
|
|
|
|
2013-09-03 10:56:33 +00:00
|
|
|
|
/* If `path' refers to a directory, then append "/default.nix". */
|
|
|
|
|
Path resolveExprPath(Path path);
|
|
|
|
|
|
2014-10-18 02:15:09 +00:00
|
|
|
|
struct InvalidPathError : EvalError
|
|
|
|
|
{
|
|
|
|
|
Path path;
|
|
|
|
|
InvalidPathError(const Path & path);
|
2014-10-20 16:15:50 +00:00
|
|
|
|
#ifdef EXCEPTION_NEEDS_THROW_SPEC
|
|
|
|
|
~InvalidPathError() throw () { };
|
|
|
|
|
#endif
|
2014-10-18 02:15:09 +00:00
|
|
|
|
};
|
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
}
|