mirror of
https://github.com/NixOS/nix.git
synced 2025-11-09 03:56:01 +01:00
Merge pull request #13936 from xokdvium/empty-list-bindings
libexpr: Make constant Values global constants, move out of EvalState
This commit is contained in:
commit
ef5fedbc0d
13 changed files with 84 additions and 50 deletions
|
|
@ -110,8 +110,8 @@ std::pair<SourcePath, uint32_t> findPackageFilename(EvalState & state, Value & v
|
||||||
{
|
{
|
||||||
Value * v2;
|
Value * v2;
|
||||||
try {
|
try {
|
||||||
auto dummyArgs = state.allocBindings(0);
|
auto & dummyArgs = Bindings::emptyBindings;
|
||||||
v2 = findAlongAttrPath(state, "meta.position", *dummyArgs, v).first;
|
v2 = findAlongAttrPath(state, "meta.position", dummyArgs, v).first;
|
||||||
} catch (Error &) {
|
} catch (Error &) {
|
||||||
throw NoPositionInfo("package '%s' has no source location information", what);
|
throw NoPositionInfo("package '%s' has no source location information", what);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -5,13 +5,15 @@
|
||||||
|
|
||||||
namespace nix {
|
namespace nix {
|
||||||
|
|
||||||
|
Bindings Bindings::emptyBindings;
|
||||||
|
|
||||||
/* Allocate a new array of attributes for an attribute set with a specific
|
/* Allocate a new array of attributes for an attribute set with a specific
|
||||||
capacity. The space is implicitly reserved after the Bindings
|
capacity. The space is implicitly reserved after the Bindings
|
||||||
structure. */
|
structure. */
|
||||||
Bindings * EvalState::allocBindings(size_t capacity)
|
Bindings * EvalState::allocBindings(size_t capacity)
|
||||||
{
|
{
|
||||||
if (capacity == 0)
|
if (capacity == 0)
|
||||||
return &emptyBindings;
|
return &Bindings::emptyBindings;
|
||||||
if (capacity > std::numeric_limits<Bindings::size_t>::max())
|
if (capacity > std::numeric_limits<Bindings::size_t>::max())
|
||||||
throw Error("attribute set of size %d is too big", capacity);
|
throw Error("attribute set of size %d is too big", capacity);
|
||||||
nrAttrsets++;
|
nrAttrsets++;
|
||||||
|
|
|
||||||
|
|
@ -202,7 +202,6 @@ EvalState::EvalState(
|
||||||
, settings{settings}
|
, settings{settings}
|
||||||
, symbols(StaticEvalSymbols::staticSymbolTable())
|
, symbols(StaticEvalSymbols::staticSymbolTable())
|
||||||
, repair(NoRepair)
|
, repair(NoRepair)
|
||||||
, emptyBindings(Bindings())
|
|
||||||
, storeFS(makeMountedSourceAccessor({
|
, storeFS(makeMountedSourceAccessor({
|
||||||
{CanonPath::root, makeEmptySourceAccessor()},
|
{CanonPath::root, makeEmptySourceAccessor()},
|
||||||
/* In the pure eval case, we can simply require
|
/* In the pure eval case, we can simply require
|
||||||
|
|
@ -285,10 +284,6 @@ EvalState::EvalState(
|
||||||
|
|
||||||
static_assert(sizeof(Env) <= 16, "environment must be <= 16 bytes");
|
static_assert(sizeof(Env) <= 16, "environment must be <= 16 bytes");
|
||||||
|
|
||||||
vEmptyList.mkList(buildList(0));
|
|
||||||
vNull.mkNull();
|
|
||||||
vTrue.mkBool(true);
|
|
||||||
vFalse.mkBool(false);
|
|
||||||
vStringRegular.mkStringNoCopy("regular");
|
vStringRegular.mkStringNoCopy("regular");
|
||||||
vStringDirectory.mkStringNoCopy("directory");
|
vStringDirectory.mkStringNoCopy("directory");
|
||||||
vStringSymlink.mkStringNoCopy("symlink");
|
vStringSymlink.mkStringNoCopy("symlink");
|
||||||
|
|
@ -895,7 +890,7 @@ ListBuilder::ListBuilder(EvalState & state, size_t size)
|
||||||
|
|
||||||
Value * EvalState::getBool(bool b)
|
Value * EvalState::getBool(bool b)
|
||||||
{
|
{
|
||||||
return b ? &vTrue : &vFalse;
|
return b ? &Value::vTrue : &Value::vFalse;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long nrThunks = 0;
|
unsigned long nrThunks = 0;
|
||||||
|
|
@ -1301,7 +1296,7 @@ void ExprList::eval(EvalState & state, Env & env, Value & v)
|
||||||
Value * ExprList::maybeThunk(EvalState & state, Env & env)
|
Value * ExprList::maybeThunk(EvalState & state, Env & env)
|
||||||
{
|
{
|
||||||
if (elems.empty()) {
|
if (elems.empty()) {
|
||||||
return &state.vEmptyList;
|
return &Value::vEmptyList;
|
||||||
}
|
}
|
||||||
return Expr::maybeThunk(state, env);
|
return Expr::maybeThunk(state, env);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -54,6 +54,12 @@ public:
|
||||||
typedef uint32_t size_t;
|
typedef uint32_t size_t;
|
||||||
PosIdx pos;
|
PosIdx pos;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An instance of bindings objects with 0 attributes.
|
||||||
|
* This object must never be modified.
|
||||||
|
*/
|
||||||
|
static Bindings emptyBindings;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
size_t size_ = 0;
|
size_t size_ = 0;
|
||||||
Attr attrs[0];
|
Attr attrs[0];
|
||||||
|
|
|
||||||
|
|
@ -313,34 +313,6 @@ public:
|
||||||
*/
|
*/
|
||||||
RepairFlag repair;
|
RepairFlag repair;
|
||||||
|
|
||||||
Bindings emptyBindings;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Empty list constant.
|
|
||||||
*/
|
|
||||||
Value vEmptyList;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* `null` constant.
|
|
||||||
*
|
|
||||||
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
|
||||||
*/
|
|
||||||
Value vNull;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* `true` constant.
|
|
||||||
*
|
|
||||||
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
|
||||||
*/
|
|
||||||
Value vTrue;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* `true` constant.
|
|
||||||
*
|
|
||||||
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
|
||||||
*/
|
|
||||||
Value vFalse;
|
|
||||||
|
|
||||||
/** `"regular"` */
|
/** `"regular"` */
|
||||||
Value vStringRegular;
|
Value vStringRegular;
|
||||||
/** `"directory"` */
|
/** `"directory"` */
|
||||||
|
|
|
||||||
|
|
@ -833,6 +833,35 @@ struct Value : public ValueStorage<sizeof(void *)>
|
||||||
{
|
{
|
||||||
friend std::string showType(const Value & v);
|
friend std::string showType(const Value & v);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Empty list constant.
|
||||||
|
*
|
||||||
|
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
||||||
|
*/
|
||||||
|
static Value vEmptyList;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* `null` constant.
|
||||||
|
*
|
||||||
|
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
||||||
|
*/
|
||||||
|
static Value vNull;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* `true` constant.
|
||||||
|
*
|
||||||
|
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
||||||
|
*/
|
||||||
|
static Value vTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* `true` constant.
|
||||||
|
*
|
||||||
|
* This is _not_ a singleton. Pointer equality is _not_ sufficient.
|
||||||
|
*/
|
||||||
|
static Value vFalse;
|
||||||
|
|
||||||
|
private:
|
||||||
template<InternalType... discriminator>
|
template<InternalType... discriminator>
|
||||||
bool isa() const noexcept
|
bool isa() const noexcept
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -163,6 +163,7 @@ sources = files(
|
||||||
'search-path.cc',
|
'search-path.cc',
|
||||||
'value-to-json.cc',
|
'value-to-json.cc',
|
||||||
'value-to-xml.cc',
|
'value-to-xml.cc',
|
||||||
|
'value.cc',
|
||||||
'value/context.cc',
|
'value/context.cc',
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -1075,11 +1075,11 @@ static void prim_tryEval(EvalState & state, const PosIdx pos, Value ** args, Val
|
||||||
try {
|
try {
|
||||||
state.forceValue(*args[0], pos);
|
state.forceValue(*args[0], pos);
|
||||||
attrs.insert(state.s.value, args[0]);
|
attrs.insert(state.s.value, args[0]);
|
||||||
attrs.insert(state.symbols.create("success"), &state.vTrue);
|
attrs.insert(state.symbols.create("success"), &Value::vTrue);
|
||||||
} catch (AssertionError & e) {
|
} catch (AssertionError & e) {
|
||||||
// `value = false;` is unfortunate but removing it is a breaking change.
|
// `value = false;` is unfortunate but removing it is a breaking change.
|
||||||
attrs.insert(state.s.value, &state.vFalse);
|
attrs.insert(state.s.value, &Value::vFalse);
|
||||||
attrs.insert(state.symbols.create("success"), &state.vFalse);
|
attrs.insert(state.symbols.create("success"), &Value::vFalse);
|
||||||
}
|
}
|
||||||
|
|
||||||
// restore the debugRepl pointer if we saved it earlier.
|
// restore the debugRepl pointer if we saved it earlier.
|
||||||
|
|
@ -3326,14 +3326,14 @@ static void prim_functionArgs(EvalState & state, const PosIdx pos, Value ** args
|
||||||
{
|
{
|
||||||
state.forceValue(*args[0], pos);
|
state.forceValue(*args[0], pos);
|
||||||
if (args[0]->isPrimOpApp() || args[0]->isPrimOp()) {
|
if (args[0]->isPrimOpApp() || args[0]->isPrimOp()) {
|
||||||
v.mkAttrs(&state.emptyBindings);
|
v.mkAttrs(&Bindings::emptyBindings);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!args[0]->isLambda())
|
if (!args[0]->isLambda())
|
||||||
state.error<TypeError>("'functionArgs' requires a function").atPos(pos).debugThrow();
|
state.error<TypeError>("'functionArgs' requires a function").atPos(pos).debugThrow();
|
||||||
|
|
||||||
if (!args[0]->lambda().fun->hasFormals()) {
|
if (!args[0]->lambda().fun->hasFormals()) {
|
||||||
v.mkAttrs(&state.emptyBindings);
|
v.mkAttrs(&Bindings::emptyBindings);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -4613,7 +4613,7 @@ void prim_match(EvalState & state, const PosIdx pos, Value ** args, Value & v)
|
||||||
auto list = state.buildList(match.size() - 1);
|
auto list = state.buildList(match.size() - 1);
|
||||||
for (const auto & [i, v2] : enumerate(list))
|
for (const auto & [i, v2] : enumerate(list))
|
||||||
if (!match[i + 1].matched)
|
if (!match[i + 1].matched)
|
||||||
v2 = &state.vNull;
|
v2 = &Value::vNull;
|
||||||
else
|
else
|
||||||
v2 = mkString(state, match[i + 1]);
|
v2 = mkString(state, match[i + 1]);
|
||||||
v.mkList(list);
|
v.mkList(list);
|
||||||
|
|
@ -4705,7 +4705,7 @@ void prim_split(EvalState & state, const PosIdx pos, Value ** args, Value & v)
|
||||||
auto list2 = state.buildList(slen);
|
auto list2 = state.buildList(slen);
|
||||||
for (const auto & [si, v2] : enumerate(list2)) {
|
for (const auto & [si, v2] : enumerate(list2)) {
|
||||||
if (!match[si + 1].matched)
|
if (!match[si + 1].matched)
|
||||||
v2 = &state.vNull;
|
v2 = &Value::vNull;
|
||||||
else
|
else
|
||||||
v2 = mkString(state, match[si + 1]);
|
v2 = mkString(state, match[si + 1]);
|
||||||
}
|
}
|
||||||
|
|
@ -5059,7 +5059,7 @@ void EvalState::createBaseEnv(const EvalSettings & evalSettings)
|
||||||
|
|
||||||
addConstant(
|
addConstant(
|
||||||
"null",
|
"null",
|
||||||
&vNull,
|
&Value::vNull,
|
||||||
{
|
{
|
||||||
.type = nNull,
|
.type = nNull,
|
||||||
.doc = R"(
|
.doc = R"(
|
||||||
|
|
|
||||||
29
src/libexpr/value.cc
Normal file
29
src/libexpr/value.cc
Normal file
|
|
@ -0,0 +1,29 @@
|
||||||
|
#include "nix/expr/value.hh"
|
||||||
|
|
||||||
|
namespace nix {
|
||||||
|
|
||||||
|
Value Value::vEmptyList = []() {
|
||||||
|
Value res;
|
||||||
|
res.setStorage(List{.size = 0, .elems = nullptr});
|
||||||
|
return res;
|
||||||
|
}();
|
||||||
|
|
||||||
|
Value Value::vNull = []() {
|
||||||
|
Value res;
|
||||||
|
res.mkNull();
|
||||||
|
return res;
|
||||||
|
}();
|
||||||
|
|
||||||
|
Value Value::vTrue = []() {
|
||||||
|
Value res;
|
||||||
|
res.mkBool(true);
|
||||||
|
return res;
|
||||||
|
}();
|
||||||
|
|
||||||
|
Value Value::vFalse = []() {
|
||||||
|
Value res;
|
||||||
|
res.mkBool(false);
|
||||||
|
return res;
|
||||||
|
}();
|
||||||
|
|
||||||
|
} // namespace nix
|
||||||
|
|
@ -522,7 +522,7 @@ struct CmdFlakeCheck : FlakeCommand
|
||||||
auto checkNixOSConfiguration = [&](const std::string & attrPath, Value & v, const PosIdx pos) {
|
auto checkNixOSConfiguration = [&](const std::string & attrPath, Value & v, const PosIdx pos) {
|
||||||
try {
|
try {
|
||||||
Activity act(*logger, lvlInfo, actUnknown, fmt("checking NixOS configuration '%s'", attrPath));
|
Activity act(*logger, lvlInfo, actUnknown, fmt("checking NixOS configuration '%s'", attrPath));
|
||||||
Bindings & bindings(*state->allocBindings(0));
|
Bindings & bindings = Bindings::emptyBindings;
|
||||||
auto vToplevel = findAlongAttrPath(*state, "config.system.build.toplevel", bindings, v).first;
|
auto vToplevel = findAlongAttrPath(*state, "config.system.build.toplevel", bindings, v).first;
|
||||||
state->forceValue(*vToplevel, pos);
|
state->forceValue(*vToplevel, pos);
|
||||||
if (!state->isDerivation(*vToplevel))
|
if (!state->isDerivation(*vToplevel))
|
||||||
|
|
|
||||||
|
|
@ -158,7 +158,7 @@ static void loadSourceExpr(EvalState & state, const SourcePath & path, Value & v
|
||||||
directory). */
|
directory). */
|
||||||
else if (st.type == SourceAccessor::tDirectory) {
|
else if (st.type == SourceAccessor::tDirectory) {
|
||||||
auto attrs = state.buildBindings(maxAttrs);
|
auto attrs = state.buildBindings(maxAttrs);
|
||||||
attrs.insert(state.symbols.create("_combineChannels"), &state.vEmptyList);
|
attrs.insert(state.symbols.create("_combineChannels"), &Value::vEmptyList);
|
||||||
StringSet seen;
|
StringSet seen;
|
||||||
getAllExprs(state, path, seen, attrs);
|
getAllExprs(state, path, seen, attrs);
|
||||||
v.mkAttrs(attrs);
|
v.mkAttrs(attrs);
|
||||||
|
|
|
||||||
|
|
@ -24,7 +24,7 @@ PackageInfos queryInstalled(EvalState & state, const Path & userEnv)
|
||||||
if (pathExists(manifestFile)) {
|
if (pathExists(manifestFile)) {
|
||||||
Value v;
|
Value v;
|
||||||
state.evalFile(state.rootPath(CanonPath(manifestFile)).resolveSymlinks(), v);
|
state.evalFile(state.rootPath(CanonPath(manifestFile)).resolveSymlinks(), v);
|
||||||
Bindings & bindings(*state.allocBindings(0));
|
Bindings & bindings = Bindings::emptyBindings;
|
||||||
getDerivations(state, v, "", bindings, elems, false);
|
getDerivations(state, v, "", bindings, elems, false);
|
||||||
}
|
}
|
||||||
return elems;
|
return elems;
|
||||||
|
|
|
||||||
|
|
@ -162,7 +162,7 @@ struct CmdUpgradeNix : MixDryRun, StoreCommand
|
||||||
auto state = std::make_unique<EvalState>(LookupPath{}, store, fetchSettings, evalSettings);
|
auto state = std::make_unique<EvalState>(LookupPath{}, store, fetchSettings, evalSettings);
|
||||||
auto v = state->allocValue();
|
auto v = state->allocValue();
|
||||||
state->eval(state->parseExprFromString(res.data, state->rootPath(CanonPath("/no-such-path"))), *v);
|
state->eval(state->parseExprFromString(res.data, state->rootPath(CanonPath("/no-such-path"))), *v);
|
||||||
Bindings & bindings(*state->allocBindings(0));
|
Bindings & bindings = Bindings::emptyBindings;
|
||||||
auto v2 = findAlongAttrPath(*state, settings.thisSystem, bindings, *v).first;
|
auto v2 = findAlongAttrPath(*state, settings.thisSystem, bindings, *v).first;
|
||||||
|
|
||||||
return store->parseStorePath(
|
return store->parseStorePath(
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue