refactor(3p/nix/libexpr): Back Nix lists with std::vector
This change does away with the previous special-casing of lists of certain element sizes, and the use of raw C-style arrays. Lists are now backed by a std::vector of nix::Value*, which uses the traceable GC allocator. This change is unfortunately quite noisy because the accessor methods were updated/removed accordingly, so all callsites of Nix-related lists have changed. For some operations in primops.cc where keeping the previous code structure would have been more difficult with a "proper" vector, the implementation has been replaced with std::vector methods. For example, list concatenation now uses appropriate range inserts. Anecdotally the performance of this is about equal, to even slightly better, than the previous implementation. All language tests pass and the depot paths I've used for testing still evaluate. Change-Id: Ib5eca6c0207429cb323a330c838c3a2200b2c693 Reviewed-on: https://cl.tvl.fyi/c/depot/+/1266 Tested-by: BuildkiteCI Reviewed-by: isomer <isomer@tvl.fyi> Reviewed-by: Kane York <rikingcoding@gmail.com> Reviewed-by: glittershark <grfn@gws.fyi>
This commit is contained in:
parent
56614c75e4
commit
cdc1a0bb6e
16 changed files with 163 additions and 231 deletions
2
third_party/nix/src/libexpr/attr-path.cc
vendored
2
third_party/nix/src/libexpr/attr-path.cc
vendored
|
@ -99,7 +99,7 @@ Value* findAlongAttrPath(EvalState& state, const std::string& attrPath,
|
||||||
attrIndex % attrPath);
|
attrIndex % attrPath);
|
||||||
}
|
}
|
||||||
|
|
||||||
v = v->listElems()[attrIndex];
|
v = (*v->list)[attrIndex];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
89
third_party/nix/src/libexpr/eval.cc
vendored
89
third_party/nix/src/libexpr/eval.cc
vendored
|
@ -97,12 +97,10 @@ static void printValue(std::ostream& str, std::set<const Value*>& active,
|
||||||
str << "}";
|
str << "}";
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
|
||||||
case tListN:
|
|
||||||
str << "[ ";
|
str << "[ ";
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
printValue(str, active, *v.listElems()[n]);
|
printValue(str, active, *(*v.list)[n]);
|
||||||
str << " ";
|
str << " ";
|
||||||
}
|
}
|
||||||
str << "]";
|
str << "]";
|
||||||
|
@ -162,9 +160,7 @@ std::string showType(const Value& v) {
|
||||||
return "null";
|
return "null";
|
||||||
case tAttrs:
|
case tAttrs:
|
||||||
return "a set";
|
return "a set";
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
|
||||||
case tListN:
|
|
||||||
return "a list";
|
return "a list";
|
||||||
case tThunk:
|
case tThunk:
|
||||||
return "a thunk";
|
return "a thunk";
|
||||||
|
@ -641,19 +637,15 @@ Env& EvalState::allocEnv(size_t size) {
|
||||||
return *env;
|
return *env;
|
||||||
}
|
}
|
||||||
|
|
||||||
void EvalState::mkList(Value& v, size_t size) {
|
void EvalState::mkList(Value& v, NixList* list) {
|
||||||
clearValue(v);
|
clearValue(v);
|
||||||
if (size == 1) {
|
v.type = tList;
|
||||||
v.type = tList1;
|
v.list = list;
|
||||||
} else if (size == 2) {
|
nrListElems += list->size();
|
||||||
v.type = tList2;
|
}
|
||||||
} else {
|
|
||||||
v.type = tListN;
|
void EvalState::mkList(Value& v, size_t size) {
|
||||||
v.bigList.size = size;
|
EvalState::mkList(v, new (GC) NixList(size));
|
||||||
v.bigList.elems =
|
|
||||||
size != 0u ? (Value**)allocBytes(size * sizeof(Value*)) : nullptr;
|
|
||||||
}
|
|
||||||
nrListElems += size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long nrThunks = 0;
|
unsigned long nrThunks = 0;
|
||||||
|
@ -877,7 +869,7 @@ void ExprLet::eval(EvalState& state, Env& env, Value& v) {
|
||||||
void ExprList::eval(EvalState& state, Env& env, Value& v) {
|
void ExprList::eval(EvalState& state, Env& env, Value& v) {
|
||||||
state.mkList(v, elems.size());
|
state.mkList(v, elems.size());
|
||||||
for (size_t n = 0; n < elems.size(); ++n) {
|
for (size_t n = 0; n < elems.size(); ++n) {
|
||||||
v.listElems()[n] = elems[n]->maybeThunk(state, env);
|
(*v.list)[n] = elems[n]->maybeThunk(state, env);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1252,39 +1244,21 @@ void ExprOpConcatLists::eval(EvalState& state, Env& env, Value& v) {
|
||||||
e1->eval(state, env, v1);
|
e1->eval(state, env, v1);
|
||||||
Value v2;
|
Value v2;
|
||||||
e2->eval(state, env, v2);
|
e2->eval(state, env, v2);
|
||||||
Value* lists[2] = {&v1, &v2};
|
state.concatLists(v, {&v1, &v2}, pos);
|
||||||
state.concatLists(v, 2, lists, pos);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void EvalState::concatLists(Value& v, size_t nrLists, Value** lists,
|
void EvalState::concatLists(Value& v, const NixList& lists, const Pos& pos) {
|
||||||
const Pos& pos) {
|
|
||||||
nrListConcats++;
|
nrListConcats++;
|
||||||
|
|
||||||
Value* nonEmpty = nullptr;
|
auto outlist = new (GC) NixList();
|
||||||
|
|
||||||
size_t len = 0;
|
size_t len = 0;
|
||||||
for (size_t n = 0; n < nrLists; ++n) {
|
for (Value* list : lists) {
|
||||||
forceList(*lists[n], pos);
|
forceList(*list, pos);
|
||||||
auto l = lists[n]->listSize();
|
outlist->insert(outlist->end(), list->list->begin(), list->list->end());
|
||||||
len += l;
|
|
||||||
if (l != 0u) {
|
|
||||||
nonEmpty = lists[n];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((nonEmpty != nullptr) && len == nonEmpty->listSize()) {
|
mkList(v, outlist);
|
||||||
v = *nonEmpty;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mkList(v, len);
|
|
||||||
auto out = v.listElems();
|
|
||||||
for (size_t n = 0, pos = 0; n < nrLists; ++n) {
|
|
||||||
auto l = lists[n]->listSize();
|
|
||||||
if (l != 0u) {
|
|
||||||
memcpy(out + pos, lists[n]->listElems(), l * sizeof(Value*));
|
|
||||||
}
|
|
||||||
pos += l;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExprConcatStrings::eval(EvalState& state, Env& env, Value& v) {
|
void ExprConcatStrings::eval(EvalState& state, Env& env, Value& v) {
|
||||||
|
@ -1383,7 +1357,7 @@ void EvalState::forceValueDeep(Value& v) {
|
||||||
}
|
}
|
||||||
} else if (v.isList()) {
|
} else if (v.isList()) {
|
||||||
for (size_t n = 0; n < v.listSize(); ++n) {
|
for (size_t n = 0; n < v.listSize(); ++n) {
|
||||||
recurse(*v.listElems()[n]);
|
recurse(*(*v.list)[n]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -1563,12 +1537,11 @@ std::string EvalState::coerceToString(const Pos& pos, Value& v,
|
||||||
if (v.isList()) {
|
if (v.isList()) {
|
||||||
std::string result;
|
std::string result;
|
||||||
for (size_t n = 0; n < v.listSize(); ++n) {
|
for (size_t n = 0; n < v.listSize(); ++n) {
|
||||||
result += coerceToString(pos, *v.listElems()[n], context, coerceMore,
|
result += coerceToString(pos, *(*v.list)[n], context, coerceMore,
|
||||||
copyToStore);
|
copyToStore);
|
||||||
if (n < v.listSize() - 1
|
if (n < v.listSize() - 1
|
||||||
/* !!! not quite correct */
|
/* !!! not quite correct */
|
||||||
&& (!v.listElems()[n]->isList() ||
|
&& (!(*v.list)[n]->isList() || (*v.list)[n]->listSize() != 0)) {
|
||||||
v.listElems()[n]->listSize() != 0)) {
|
|
||||||
result += " ";
|
result += " ";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1653,14 +1626,12 @@ bool EvalState::eqValues(Value& v1, Value& v2) {
|
||||||
case tNull:
|
case tNull:
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
|
||||||
case tListN:
|
|
||||||
if (v1.listSize() != v2.listSize()) {
|
if (v1.listSize() != v2.listSize()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
for (size_t n = 0; n < v1.listSize(); ++n) {
|
for (size_t n = 0; n < v1.listSize(); ++n) {
|
||||||
if (!eqValues(*v1.listElems()[n], *v2.listElems()[n])) {
|
if (!eqValues(*(*v1.list)[n], *(*v2.list)[n])) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1883,14 +1854,12 @@ size_t valueSize(Value& v) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
if (seen.find(v.list) == seen.end()) {
|
||||||
case tListN:
|
seen.insert(v.list);
|
||||||
if (seen.find(v.listElems()) == seen.end()) {
|
|
||||||
seen.insert(v.listElems());
|
|
||||||
sz += v.listSize() * sizeof(Value*);
|
sz += v.listSize() * sizeof(Value*);
|
||||||
for (size_t n = 0; n < v.listSize(); ++n) {
|
for (size_t n = 0; n < v.listSize(); ++n) {
|
||||||
sz += doValue(*v.listElems()[n]);
|
sz += doValue(*(*v.list)[n]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
11
third_party/nix/src/libexpr/eval.hh
vendored
11
third_party/nix/src/libexpr/eval.hh
vendored
|
@ -260,12 +260,19 @@ class EvalState {
|
||||||
|
|
||||||
Value* allocAttr(Value& vAttrs, const Symbol& name);
|
Value* allocAttr(Value& vAttrs, const Symbol& name);
|
||||||
|
|
||||||
void mkList(Value& v, size_t size);
|
// Create a list value from the specified vector.
|
||||||
|
void mkList(Value& v, NixList* list);
|
||||||
|
|
||||||
|
// Create a list value, allocating as many elements as specified in
|
||||||
|
// size. This is used for the many cases in this codebase where
|
||||||
|
// assignment happens into the preallocated list.
|
||||||
|
void mkList(Value& v, size_t size = 0);
|
||||||
|
|
||||||
void mkAttrs(Value& v, size_t capacity);
|
void mkAttrs(Value& v, size_t capacity);
|
||||||
void mkThunk_(Value& v, Expr* expr);
|
void mkThunk_(Value& v, Expr* expr);
|
||||||
void mkPos(Value& v, Pos* pos);
|
void mkPos(Value& v, Pos* pos);
|
||||||
|
|
||||||
void concatLists(Value& v, size_t nrLists, Value** lists, const Pos& pos);
|
void concatLists(Value& v, const NixList& lists, const Pos& pos);
|
||||||
|
|
||||||
/* Print statistics. */
|
/* Print statistics. */
|
||||||
void printStats();
|
void printStats();
|
||||||
|
|
20
third_party/nix/src/libexpr/get-drvs.cc
vendored
20
third_party/nix/src/libexpr/get-drvs.cc
vendored
|
@ -99,8 +99,8 @@ DrvInfo::Outputs DrvInfo::queryOutputs(bool onlyOutputsToInstall) {
|
||||||
/* For each output... */
|
/* For each output... */
|
||||||
for (unsigned int j = 0; j < i->second.value->listSize(); ++j) {
|
for (unsigned int j = 0; j < i->second.value->listSize(); ++j) {
|
||||||
/* Evaluate the corresponding set. */
|
/* Evaluate the corresponding set. */
|
||||||
std::string name = state->forceStringNoCtx(
|
std::string name = state->forceStringNoCtx(*(*i->second.value->list)[j],
|
||||||
*i->second.value->listElems()[j], *i->second.pos);
|
*i->second.pos);
|
||||||
Bindings::iterator out = attrs->find(state->symbols.Create(name));
|
Bindings::iterator out = attrs->find(state->symbols.Create(name));
|
||||||
if (out == attrs->end()) {
|
if (out == attrs->end()) {
|
||||||
continue; // FIXME: throw error?
|
continue; // FIXME: throw error?
|
||||||
|
@ -136,12 +136,12 @@ DrvInfo::Outputs DrvInfo::queryOutputs(bool onlyOutputsToInstall) {
|
||||||
throw errMsg;
|
throw errMsg;
|
||||||
}
|
}
|
||||||
Outputs result;
|
Outputs result;
|
||||||
for (auto i = outTI->listElems(); i != outTI->listElems() + outTI->listSize();
|
|
||||||
++i) {
|
for (Value* i : *outTI->list) {
|
||||||
if ((*i)->type != tString) {
|
if (i->type != tString) {
|
||||||
throw errMsg;
|
throw errMsg;
|
||||||
}
|
}
|
||||||
auto out = outputs.find((*i)->string.s);
|
auto out = outputs.find(i->string.s);
|
||||||
if (out == outputs.end()) {
|
if (out == outputs.end()) {
|
||||||
throw errMsg;
|
throw errMsg;
|
||||||
}
|
}
|
||||||
|
@ -190,7 +190,7 @@ bool DrvInfo::checkMeta(Value& v) {
|
||||||
state->forceValue(v);
|
state->forceValue(v);
|
||||||
if (v.isList()) {
|
if (v.isList()) {
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
if (!checkMeta(*v.listElems()[n])) {
|
if (!checkMeta(*(*v.list)[n])) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -418,10 +418,10 @@ static void getDerivations(EvalState& state, Value& vIn,
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
std::string pathPrefix2 =
|
std::string pathPrefix2 =
|
||||||
addToPath(pathPrefix, (format("%1%") % n).str());
|
addToPath(pathPrefix, (format("%1%") % n).str());
|
||||||
if (getDerivation(state, *v.listElems()[n], pathPrefix2, drvs, done,
|
if (getDerivation(state, *(*v.list)[n], pathPrefix2, drvs, done,
|
||||||
ignoreAssertionFailures)) {
|
ignoreAssertionFailures)) {
|
||||||
getDerivations(state, *v.listElems()[n], pathPrefix2, autoArgs, drvs,
|
getDerivations(state, *(*v.list)[n], pathPrefix2, autoArgs, drvs, done,
|
||||||
done, ignoreAssertionFailures);
|
ignoreAssertionFailures);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
4
third_party/nix/src/libexpr/json-to-value.cc
vendored
4
third_party/nix/src/libexpr/json-to-value.cc
vendored
|
@ -63,7 +63,7 @@ static void parseJSON(EvalState& state, const char*& s, Value& v) {
|
||||||
|
|
||||||
if (*s == '[') {
|
if (*s == '[') {
|
||||||
s++;
|
s++;
|
||||||
ValueVector values;
|
NixList values;
|
||||||
values.reserve(128);
|
values.reserve(128);
|
||||||
skipWhitespace(s);
|
skipWhitespace(s);
|
||||||
while (true) {
|
while (true) {
|
||||||
|
@ -85,7 +85,7 @@ static void parseJSON(EvalState& state, const char*& s, Value& v) {
|
||||||
s++;
|
s++;
|
||||||
state.mkList(v, values.size());
|
state.mkList(v, values.size());
|
||||||
for (size_t n = 0; n < values.size(); ++n) {
|
for (size_t n = 0; n < values.size(); ++n) {
|
||||||
v.listElems()[n] = values[n];
|
(*v.list)[n] = values[n];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
158
third_party/nix/src/libexpr/primops.cc
vendored
158
third_party/nix/src/libexpr/primops.cc
vendored
|
@ -127,9 +127,10 @@ static void prim_scopedImport(EvalState& state, const Pos& pos, Value** args,
|
||||||
for (const auto& o : drv.outputs) {
|
for (const auto& o : drv.outputs) {
|
||||||
v2 = state.allocAttr(w, state.symbols.Create(o.first));
|
v2 = state.allocAttr(w, state.symbols.Create(o.first));
|
||||||
mkString(*v2, o.second.path, {"!" + o.first + "!" + path});
|
mkString(*v2, o.second.path, {"!" + o.first + "!" + path});
|
||||||
outputsVal->listElems()[outputs_index] = state.allocValue();
|
(*outputsVal->list)[outputs_index] = state.allocValue();
|
||||||
mkString(*(outputsVal->listElems()[outputs_index++]), o.first);
|
mkString(*((*outputsVal->list)[outputs_index++]), o.first);
|
||||||
}
|
}
|
||||||
|
|
||||||
Value fun;
|
Value fun;
|
||||||
state.evalFile(
|
state.evalFile(
|
||||||
settings.nixDataDir + "/nix/corepkgs/imported-drv-to-derivation.nix",
|
settings.nixDataDir + "/nix/corepkgs/imported-drv-to-derivation.nix",
|
||||||
|
@ -231,9 +232,7 @@ static void prim_typeOf(EvalState& state, const Pos& pos, Value** args,
|
||||||
case tAttrs:
|
case tAttrs:
|
||||||
t = "set";
|
t = "set";
|
||||||
break;
|
break;
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
|
||||||
case tListN:
|
|
||||||
t = "list";
|
t = "list";
|
||||||
break;
|
break;
|
||||||
case tLambda:
|
case tLambda:
|
||||||
|
@ -360,8 +359,8 @@ static void prim_genericClosure(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.forceList(*startSet->second.value, pos);
|
state.forceList(*startSet->second.value, pos);
|
||||||
|
|
||||||
ValueList workSet;
|
ValueList workSet;
|
||||||
for (unsigned int n = 0; n < startSet->second.value->listSize(); ++n) {
|
for (Value* elem : *startSet->second.value->list) {
|
||||||
workSet.push_back(startSet->second.value->listElems()[n]);
|
workSet.push_back(elem);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get the operator. */
|
/* Get the operator. */
|
||||||
|
@ -404,8 +403,8 @@ static void prim_genericClosure(EvalState& state, const Pos& pos, Value** args,
|
||||||
|
|
||||||
/* Add the values returned by the operator to the work set. */
|
/* Add the values returned by the operator to the work set. */
|
||||||
for (unsigned int n = 0; n < call.listSize(); ++n) {
|
for (unsigned int n = 0; n < call.listSize(); ++n) {
|
||||||
state.forceValue(*call.listElems()[n]);
|
state.forceValue(*(*call.list)[n]);
|
||||||
workSet.push_back(call.listElems()[n]);
|
workSet.push_back((*call.list)[n]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -413,7 +412,7 @@ static void prim_genericClosure(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(v, res.size());
|
state.mkList(v, res.size());
|
||||||
unsigned int n = 0;
|
unsigned int n = 0;
|
||||||
for (auto& i : res) {
|
for (auto& i : res) {
|
||||||
v.listElems()[n++] = i;
|
(*v.list)[n++] = i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -620,7 +619,7 @@ static void prim_derivationStrict(EvalState& state, const Pos& pos,
|
||||||
state.forceList(*i->value, pos);
|
state.forceList(*i->value, pos);
|
||||||
for (unsigned int n = 0; n < i->value->listSize(); ++n) {
|
for (unsigned int n = 0; n < i->value->listSize(); ++n) {
|
||||||
std::string s = state.coerceToString(
|
std::string s = state.coerceToString(
|
||||||
posDrvName, *i->value->listElems()[n], context, true);
|
posDrvName, *(*i->value->list)[n], context, true);
|
||||||
drv.args.push_back(s);
|
drv.args.push_back(s);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -651,8 +650,8 @@ static void prim_derivationStrict(EvalState& state, const Pos& pos,
|
||||||
state.forceList(*i->value, posDrvName);
|
state.forceList(*i->value, posDrvName);
|
||||||
Strings ss;
|
Strings ss;
|
||||||
for (unsigned int n = 0; n < i->value->listSize(); ++n) {
|
for (unsigned int n = 0; n < i->value->listSize(); ++n) {
|
||||||
ss.emplace_back(state.forceStringNoCtx(*i->value->listElems()[n],
|
ss.emplace_back(
|
||||||
posDrvName));
|
state.forceStringNoCtx(*(*i->value->list)[n], posDrvName));
|
||||||
}
|
}
|
||||||
handleOutputs(ss);
|
handleOutputs(ss);
|
||||||
}
|
}
|
||||||
|
@ -942,7 +941,7 @@ static void prim_findFile(EvalState& state, const Pos& pos, Value** args,
|
||||||
SearchPath searchPath;
|
SearchPath searchPath;
|
||||||
|
|
||||||
for (unsigned int n = 0; n < args[0]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[0]->listSize(); ++n) {
|
||||||
Value& v2(*args[0]->listElems()[n]);
|
Value& v2(*(*args[0]->list)[n]);
|
||||||
state.forceAttrs(v2, pos);
|
state.forceAttrs(v2, pos);
|
||||||
|
|
||||||
std::string prefix;
|
std::string prefix;
|
||||||
|
@ -1225,7 +1224,7 @@ static void prim_attrNames(EvalState& state, const Pos& pos, Value** args,
|
||||||
|
|
||||||
unsigned int n = 0;
|
unsigned int n = 0;
|
||||||
for (auto& [key, value] : *args[0]->attrs) {
|
for (auto& [key, value] : *args[0]->attrs) {
|
||||||
mkString(*(v.listElems()[n++] = state.allocValue()), key);
|
mkString(*((*v.list)[n++] = state.allocValue()), key);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1239,7 +1238,7 @@ static void prim_attrValues(EvalState& state, const Pos& pos, Value** input,
|
||||||
|
|
||||||
unsigned int n = 0;
|
unsigned int n = 0;
|
||||||
for (auto& [key, value] : *input[0]->attrs) {
|
for (auto& [key, value] : *input[0]->attrs) {
|
||||||
output.listElems()[n++] = value.value;
|
(*output.list)[n++] = value.value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1297,8 +1296,8 @@ static void prim_removeAttrs(EvalState& state, const Pos& pos, Value** args,
|
||||||
/* Get the attribute names to be removed. */
|
/* Get the attribute names to be removed. */
|
||||||
std::set<Symbol> names;
|
std::set<Symbol> names;
|
||||||
for (unsigned int i = 0; i < args[1]->listSize(); ++i) {
|
for (unsigned int i = 0; i < args[1]->listSize(); ++i) {
|
||||||
state.forceStringNoCtx(*args[1]->listElems()[i], pos);
|
state.forceStringNoCtx(*(*args[1]->list)[i], pos);
|
||||||
names.insert(state.symbols.Create(args[1]->listElems()[i]->string.s));
|
names.insert(state.symbols.Create((*args[1]->list)[i]->string.s));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Copy all attributes not in that set. Note that we don't need
|
/* Copy all attributes not in that set. Note that we don't need
|
||||||
|
@ -1326,7 +1325,7 @@ static void prim_listToAttrs(EvalState& state, const Pos& pos, Value** args,
|
||||||
std::set<Symbol> seen;
|
std::set<Symbol> seen;
|
||||||
|
|
||||||
for (unsigned int i = 0; i < args[0]->listSize(); ++i) {
|
for (unsigned int i = 0; i < args[0]->listSize(); ++i) {
|
||||||
Value& v2(*args[0]->listElems()[i]);
|
Value& v2(*(*args[0]->list)[i]);
|
||||||
state.forceAttrs(v2, pos);
|
state.forceAttrs(v2, pos);
|
||||||
|
|
||||||
Bindings::iterator j = v2.attrs->find(state.sName);
|
Bindings::iterator j = v2.attrs->find(state.sName);
|
||||||
|
@ -1391,7 +1390,7 @@ static void prim_catAttrs(EvalState& state, const Pos& pos, Value** args,
|
||||||
unsigned int found = 0;
|
unsigned int found = 0;
|
||||||
|
|
||||||
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
||||||
Value& v2(*args[1]->listElems()[n]);
|
Value& v2(*(*args[1]->list)[n]);
|
||||||
state.forceAttrs(v2, pos);
|
state.forceAttrs(v2, pos);
|
||||||
Bindings::iterator i = v2.attrs->find(attrName);
|
Bindings::iterator i = v2.attrs->find(attrName);
|
||||||
if (i != v2.attrs->end()) {
|
if (i != v2.attrs->end()) {
|
||||||
|
@ -1401,7 +1400,7 @@ static void prim_catAttrs(EvalState& state, const Pos& pos, Value** args,
|
||||||
|
|
||||||
state.mkList(v, found);
|
state.mkList(v, found);
|
||||||
for (unsigned int n = 0; n < found; ++n) {
|
for (unsigned int n = 0; n < found; ++n) {
|
||||||
v.listElems()[n] = res[n];
|
(*v.list)[n] = res[n];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1472,8 +1471,8 @@ static void elemAt(EvalState& state, const Pos& pos, Value& list, int n,
|
||||||
if (n < 0 || (unsigned int)n >= list.listSize()) {
|
if (n < 0 || (unsigned int)n >= list.listSize()) {
|
||||||
throw Error(format("list index %1% is out of bounds, at %2%") % n % pos);
|
throw Error(format("list index %1% is out of bounds, at %2%") % n % pos);
|
||||||
}
|
}
|
||||||
state.forceValue(*list.listElems()[n]);
|
state.forceValue(*(*list.list)[n]);
|
||||||
v = *list.listElems()[n];
|
v = *(*list.list)[n];
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the n-1'th element of a list. */
|
/* Return the n-1'th element of a list. */
|
||||||
|
@ -1499,7 +1498,7 @@ static void prim_tail(EvalState& state, const Pos& pos, Value** args,
|
||||||
}
|
}
|
||||||
state.mkList(v, args[0]->listSize() - 1);
|
state.mkList(v, args[0]->listSize() - 1);
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
v.listElems()[n] = args[0]->listElems()[n + 1];
|
(*v.list)[n] = (*args[0]->list)[n + 1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1510,8 +1509,7 @@ static void prim_map(EvalState& state, const Pos& pos, Value** args, Value& v) {
|
||||||
state.mkList(v, args[1]->listSize());
|
state.mkList(v, args[1]->listSize());
|
||||||
|
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
mkApp(*(v.listElems()[n] = state.allocValue()), *args[0],
|
mkApp(*((*v.list)[n] = state.allocValue()), *args[0], *(*args[1]->list)[n]);
|
||||||
*args[1]->listElems()[n]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1530,9 +1528,9 @@ static void prim_filter(EvalState& state, const Pos& pos, Value** args,
|
||||||
bool same = true;
|
bool same = true;
|
||||||
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
||||||
Value res;
|
Value res;
|
||||||
state.callFunction(*args[0], *args[1]->listElems()[n], res, noPos);
|
state.callFunction(*args[0], *(*args[1]->list)[n], res, noPos);
|
||||||
if (state.forceBool(res, pos)) {
|
if (state.forceBool(res, pos)) {
|
||||||
vs[k++] = args[1]->listElems()[n];
|
vs[k++] = (*args[1]->list)[n];
|
||||||
} else {
|
} else {
|
||||||
same = false;
|
same = false;
|
||||||
}
|
}
|
||||||
|
@ -1543,7 +1541,7 @@ static void prim_filter(EvalState& state, const Pos& pos, Value** args,
|
||||||
} else {
|
} else {
|
||||||
state.mkList(v, k);
|
state.mkList(v, k);
|
||||||
for (unsigned int n = 0; n < k; ++n) {
|
for (unsigned int n = 0; n < k; ++n) {
|
||||||
v.listElems()[n] = vs[n];
|
(*v.list)[n] = vs[n];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1554,7 +1552,7 @@ static void prim_elem(EvalState& state, const Pos& pos, Value** args,
|
||||||
bool res = false;
|
bool res = false;
|
||||||
state.forceList(*args[1], pos);
|
state.forceList(*args[1], pos);
|
||||||
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
||||||
if (state.eqValues(*args[0], *args[1]->listElems()[n])) {
|
if (state.eqValues(*args[0], *(*args[1]->list)[n])) {
|
||||||
res = true;
|
res = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1566,7 +1564,7 @@ static void prim_elem(EvalState& state, const Pos& pos, Value** args,
|
||||||
static void prim_concatLists(EvalState& state, const Pos& pos, Value** args,
|
static void prim_concatLists(EvalState& state, const Pos& pos, Value** args,
|
||||||
Value& v) {
|
Value& v) {
|
||||||
state.forceList(*args[0], pos);
|
state.forceList(*args[0], pos);
|
||||||
state.concatLists(v, args[0]->listSize(), args[0]->listElems(), pos);
|
state.concatLists(v, *args[0]->list, pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Return the length of a list. This is an O(1) time operation. */
|
/* Return the length of a list. This is an O(1) time operation. */
|
||||||
|
@ -1590,7 +1588,7 @@ static void prim_foldlStrict(EvalState& state, const Pos& pos, Value** args,
|
||||||
Value vTmp;
|
Value vTmp;
|
||||||
state.callFunction(*args[0], *vCur, vTmp, pos);
|
state.callFunction(*args[0], *vCur, vTmp, pos);
|
||||||
vCur = n == args[2]->listSize() - 1 ? &v : state.allocValue();
|
vCur = n == args[2]->listSize() - 1 ? &v : state.allocValue();
|
||||||
state.callFunction(vTmp, *args[2]->listElems()[n], *vCur, pos);
|
state.callFunction(vTmp, *(*args[2]->list)[n], *vCur, pos);
|
||||||
}
|
}
|
||||||
state.forceValue(v);
|
state.forceValue(v);
|
||||||
} else {
|
} else {
|
||||||
|
@ -1606,7 +1604,7 @@ static void anyOrAll(bool any, EvalState& state, const Pos& pos, Value** args,
|
||||||
|
|
||||||
Value vTmp;
|
Value vTmp;
|
||||||
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
||||||
state.callFunction(*args[0], *args[1]->listElems()[n], vTmp, pos);
|
state.callFunction(*args[0], *(*args[1]->list)[n], vTmp, pos);
|
||||||
bool res = state.forceBool(vTmp, pos);
|
bool res = state.forceBool(vTmp, pos);
|
||||||
if (res == any) {
|
if (res == any) {
|
||||||
mkBool(v, any);
|
mkBool(v, any);
|
||||||
|
@ -1639,7 +1637,7 @@ static void prim_genList(EvalState& state, const Pos& pos, Value** args,
|
||||||
for (unsigned int n = 0; n < (unsigned int)len; ++n) {
|
for (unsigned int n = 0; n < (unsigned int)len; ++n) {
|
||||||
Value* arg = state.allocValue();
|
Value* arg = state.allocValue();
|
||||||
mkInt(*arg, n);
|
mkInt(*arg, n);
|
||||||
mkApp(*(v.listElems()[n] = state.allocValue()), *args[0], *arg);
|
mkApp(*((*v.list)[n] = state.allocValue()), *args[0], *arg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1651,12 +1649,11 @@ static void prim_sort(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.forceFunction(*args[0], pos);
|
state.forceFunction(*args[0], pos);
|
||||||
state.forceList(*args[1], pos);
|
state.forceList(*args[1], pos);
|
||||||
|
|
||||||
auto len = args[1]->listSize();
|
// Copy of the input list which can be sorted in place.
|
||||||
state.mkList(v, len);
|
auto outlist = new (GC) NixList(*args[1]->list);
|
||||||
for (unsigned int n = 0; n < len; ++n) {
|
|
||||||
state.forceValue(*args[1]->listElems()[n]);
|
std::for_each(outlist->begin(), outlist->end(),
|
||||||
v.listElems()[n] = args[1]->listElems()[n];
|
[&](Value* val) { state.forceValue(*val); });
|
||||||
}
|
|
||||||
|
|
||||||
auto comparator = [&](Value* a, Value* b) {
|
auto comparator = [&](Value* a, Value* b) {
|
||||||
/* Optimization: if the comparator is lessThan, bypass
|
/* Optimization: if the comparator is lessThan, bypass
|
||||||
|
@ -1675,7 +1672,7 @@ static void prim_sort(EvalState& state, const Pos& pos, Value** args,
|
||||||
/* FIXME: std::sort can segfault if the comparator is not a strict
|
/* FIXME: std::sort can segfault if the comparator is not a strict
|
||||||
weak ordering. What to do? std::stable_sort() seems more
|
weak ordering. What to do? std::stable_sort() seems more
|
||||||
resilient, but no guarantees... */
|
resilient, but no guarantees... */
|
||||||
std::stable_sort(v.listElems(), v.listElems() + len, comparator);
|
std::stable_sort(v.list->begin(), v.list->end(), comparator);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void prim_partition(EvalState& state, const Pos& pos, Value** args,
|
static void prim_partition(EvalState& state, const Pos& pos, Value** args,
|
||||||
|
@ -1685,36 +1682,28 @@ static void prim_partition(EvalState& state, const Pos& pos, Value** args,
|
||||||
|
|
||||||
auto len = args[1]->listSize();
|
auto len = args[1]->listSize();
|
||||||
|
|
||||||
ValueVector right;
|
NixList* right = new (GC) NixList();
|
||||||
ValueVector wrong;
|
NixList* wrong = new (GC) NixList();
|
||||||
|
|
||||||
|
for (Value* elem : *args[1]->list) {
|
||||||
|
state.forceValue(*elem, pos);
|
||||||
|
|
||||||
for (unsigned int n = 0; n < len; ++n) {
|
|
||||||
auto vElem = args[1]->listElems()[n];
|
|
||||||
state.forceValue(*vElem);
|
|
||||||
Value res;
|
Value res;
|
||||||
state.callFunction(*args[0], *vElem, res, pos);
|
state.callFunction(*args[0], *elem, res, pos);
|
||||||
if (state.forceBool(res, pos)) {
|
if (state.forceBool(res, pos)) {
|
||||||
right.push_back(vElem);
|
right->push_back(elem);
|
||||||
} else {
|
} else {
|
||||||
wrong.push_back(vElem);
|
wrong->push_back(elem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
state.mkAttrs(v, 2);
|
state.mkAttrs(v, 2);
|
||||||
|
|
||||||
Value* vRight = state.allocAttr(v, state.sRight);
|
Value* vRight = state.allocAttr(v, state.sRight);
|
||||||
auto rsize = right.size();
|
state.mkList(*vRight, right);
|
||||||
state.mkList(*vRight, rsize);
|
|
||||||
if (rsize != 0u) {
|
|
||||||
memcpy(vRight->listElems(), right.data(), sizeof(Value*) * rsize);
|
|
||||||
}
|
|
||||||
|
|
||||||
Value* vWrong = state.allocAttr(v, state.sWrong);
|
Value* vWrong = state.allocAttr(v, state.sWrong);
|
||||||
auto wsize = wrong.size();
|
state.mkList(*vWrong, wrong);
|
||||||
state.mkList(*vWrong, wsize);
|
|
||||||
if (wsize != 0u) {
|
|
||||||
memcpy(vWrong->listElems(), wrong.data(), sizeof(Value*) * wsize);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* concatMap = f: list: concatLists (map f list); */
|
/* concatMap = f: list: concatLists (map f list); */
|
||||||
|
@ -1723,27 +1712,18 @@ static void prim_concatMap(EvalState& state, const Pos& pos, Value** args,
|
||||||
Value& v) {
|
Value& v) {
|
||||||
state.forceFunction(*args[0], pos);
|
state.forceFunction(*args[0], pos);
|
||||||
state.forceList(*args[1], pos);
|
state.forceList(*args[1], pos);
|
||||||
auto nrLists = args[1]->listSize();
|
|
||||||
|
|
||||||
Value lists[nrLists];
|
NixList* outlist = new (GC) NixList;
|
||||||
size_t len = 0;
|
|
||||||
|
|
||||||
for (unsigned int n = 0; n < nrLists; ++n) {
|
for (Value* elem : *args[1]->list) {
|
||||||
Value* vElem = args[1]->listElems()[n];
|
auto out = state.allocValue();
|
||||||
state.callFunction(*args[0], *vElem, lists[n], pos);
|
state.callFunction(*args[0], *elem, *out, pos);
|
||||||
state.forceList(lists[n], pos);
|
state.forceList(*out, pos);
|
||||||
len += lists[n].listSize();
|
|
||||||
|
outlist->insert(outlist->end(), out->list->begin(), out->list->end());
|
||||||
}
|
}
|
||||||
|
|
||||||
state.mkList(v, len);
|
state.mkList(v, outlist);
|
||||||
auto out = v.listElems();
|
|
||||||
for (unsigned int n = 0, pos = 0; n < nrLists; ++n) {
|
|
||||||
auto l = lists[n].listSize();
|
|
||||||
if (l != 0u) {
|
|
||||||
memcpy(out + pos, lists[n].listElems(), l * sizeof(Value*));
|
|
||||||
}
|
|
||||||
pos += l;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*************************************************************
|
/*************************************************************
|
||||||
|
@ -1908,9 +1888,9 @@ static void prim_match(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(v, len);
|
state.mkList(v, len);
|
||||||
for (size_t i = 0; i < len; ++i) {
|
for (size_t i = 0; i < len; ++i) {
|
||||||
if (!match[i + 1].matched) {
|
if (!match[i + 1].matched) {
|
||||||
mkNull(*(v.listElems()[i] = state.allocValue()));
|
mkNull(*((*v.list)[i] = state.allocValue()));
|
||||||
} else {
|
} else {
|
||||||
mkString(*(v.listElems()[i] = state.allocValue()),
|
mkString(*((*v.list)[i] = state.allocValue()),
|
||||||
match[i + 1].str().c_str());
|
match[i + 1].str().c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1947,7 +1927,7 @@ static void prim_split(EvalState& state, const Pos& pos, Value** args,
|
||||||
Value* elem;
|
Value* elem;
|
||||||
|
|
||||||
if (len == 0) {
|
if (len == 0) {
|
||||||
v.listElems()[idx++] = args[1];
|
(*v.list)[idx++] = args[1];
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1956,27 +1936,27 @@ static void prim_split(EvalState& state, const Pos& pos, Value** args,
|
||||||
std::smatch match = *i;
|
std::smatch match = *i;
|
||||||
|
|
||||||
// Add a string for non-matched characters.
|
// Add a string for non-matched characters.
|
||||||
elem = v.listElems()[idx++] = state.allocValue();
|
elem = (*v.list)[idx++] = state.allocValue();
|
||||||
mkString(*elem, match.prefix().str().c_str());
|
mkString(*elem, match.prefix().str().c_str());
|
||||||
|
|
||||||
// Add a list for matched substrings.
|
// Add a list for matched substrings.
|
||||||
const size_t slen = match.size() - 1;
|
const size_t slen = match.size() - 1;
|
||||||
elem = v.listElems()[idx++] = state.allocValue();
|
elem = (*v.list)[idx++] = state.allocValue();
|
||||||
|
|
||||||
// Start at 1, beacause the first match is the whole string.
|
// Start at 1, beacause the first match is the whole string.
|
||||||
state.mkList(*elem, slen);
|
state.mkList(*elem, slen);
|
||||||
for (size_t si = 0; si < slen; ++si) {
|
for (size_t si = 0; si < slen; ++si) {
|
||||||
if (!match[si + 1].matched) {
|
if (!match[si + 1].matched) {
|
||||||
mkNull(*(elem->listElems()[si] = state.allocValue()));
|
mkNull(*((*elem->list)[si] = state.allocValue()));
|
||||||
} else {
|
} else {
|
||||||
mkString(*(elem->listElems()[si] = state.allocValue()),
|
mkString(*((*elem->list)[si] = state.allocValue()),
|
||||||
match[si + 1].str().c_str());
|
match[si + 1].str().c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a string for non-matched suffix characters.
|
// Add a string for non-matched suffix characters.
|
||||||
if (idx == 2 * len) {
|
if (idx == 2 * len) {
|
||||||
elem = v.listElems()[idx++] = state.allocValue();
|
elem = (*v.list)[idx++] = state.allocValue();
|
||||||
mkString(*elem, match.suffix().str().c_str());
|
mkString(*elem, match.suffix().str().c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2010,7 +1990,7 @@ static void prim_concatStringSep(EvalState& state, const Pos& pos, Value** args,
|
||||||
res += sep;
|
res += sep;
|
||||||
}
|
}
|
||||||
|
|
||||||
res += state.coerceToString(pos, *args[1]->listElems()[n], context);
|
res += state.coerceToString(pos, *(*args[1]->list)[n], context);
|
||||||
}
|
}
|
||||||
|
|
||||||
mkString(v, res, context);
|
mkString(v, res, context);
|
||||||
|
@ -2029,14 +2009,14 @@ static void prim_replaceStrings(EvalState& state, const Pos& pos, Value** args,
|
||||||
std::vector<std::string> from;
|
std::vector<std::string> from;
|
||||||
from.reserve(args[0]->listSize());
|
from.reserve(args[0]->listSize());
|
||||||
for (unsigned int n = 0; n < args[0]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[0]->listSize(); ++n) {
|
||||||
from.push_back(state.forceString(*args[0]->listElems()[n], pos));
|
from.push_back(state.forceString(*(*args[0]->list)[n], pos));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<std::pair<std::string, PathSet>> to;
|
std::vector<std::pair<std::string, PathSet>> to;
|
||||||
to.reserve(args[1]->listSize());
|
to.reserve(args[1]->listSize());
|
||||||
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
for (unsigned int n = 0; n < args[1]->listSize(); ++n) {
|
||||||
PathSet ctx;
|
PathSet ctx;
|
||||||
auto s = state.forceString(*args[1]->listElems()[n], ctx, pos);
|
auto s = state.forceString(*(*args[1]->list)[n], ctx, pos);
|
||||||
to.emplace_back(std::move(s), std::move(ctx));
|
to.emplace_back(std::move(s), std::move(ctx));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2116,7 +2096,7 @@ static void prim_splitVersion(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(v, components.size());
|
state.mkList(v, components.size());
|
||||||
unsigned int n = 0;
|
unsigned int n = 0;
|
||||||
for (auto& component : components) {
|
for (auto& component : components) {
|
||||||
auto listElem = v.listElems()[n++] = state.allocValue();
|
auto listElem = (*v.list)[n++] = state.allocValue();
|
||||||
mkString(*listElem, component);
|
mkString(*listElem, component);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2386,7 +2366,7 @@ void EvalState::createBaseEnv() {
|
||||||
mkList(v, searchPath.size());
|
mkList(v, searchPath.size());
|
||||||
int n = 0;
|
int n = 0;
|
||||||
for (auto& i : searchPath) {
|
for (auto& i : searchPath) {
|
||||||
v2 = v.listElems()[n++] = allocValue();
|
v2 = (*v.list)[n++] = allocValue();
|
||||||
mkAttrs(*v2, 2);
|
mkAttrs(*v2, 2);
|
||||||
mkString(*allocAttr(*v2, symbols.Create("path")), i.second);
|
mkString(*allocAttr(*v2, symbols.Create("path")), i.second);
|
||||||
mkString(*allocAttr(*v2, symbols.Create("prefix")), i.first);
|
mkString(*allocAttr(*v2, symbols.Create("prefix")), i.first);
|
||||||
|
|
|
@ -123,7 +123,7 @@ static void prim_getContext(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(outputsVal, info.second.outputs.size());
|
state.mkList(outputsVal, info.second.outputs.size());
|
||||||
size_t i = 0;
|
size_t i = 0;
|
||||||
for (const auto& output : info.second.outputs) {
|
for (const auto& output : info.second.outputs) {
|
||||||
mkString(*(outputsVal.listElems()[i++] = state.allocValue()), output);
|
mkString(*((*outputsVal.list)[i++] = state.allocValue()), output);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -184,7 +184,7 @@ static void prim_appendContext(EvalState& state, const Pos& pos, Value** args,
|
||||||
i->name, i->pos);
|
i->name, i->pos);
|
||||||
}
|
}
|
||||||
for (unsigned int n = 0; n < iter->second.value->listSize(); ++n) {
|
for (unsigned int n = 0; n < iter->second.value->listSize(); ++n) {
|
||||||
auto name = state.forceStringNoCtx(*iter->second.value->listElems()[n],
|
auto name = state.forceStringNoCtx(*(*iter->second.value->list)[n],
|
||||||
*iter->second.pos);
|
*iter->second.pos);
|
||||||
context.insert("!" + name + "!" + std::string(i->name));
|
context.insert("!" + name + "!" + std::string(i->name));
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,7 +31,7 @@ static void prim_fromTOML(EvalState& state, const Pos& pos, Value** args,
|
||||||
size_t size2 = i2->get().size();
|
size_t size2 = i2->get().size();
|
||||||
state.mkList(v2, size2);
|
state.mkList(v2, size2);
|
||||||
for (size_t j = 0; j < size2; ++j)
|
for (size_t j = 0; j < size2; ++j)
|
||||||
visit(*(v2.listElems()[j] = state.allocValue()), i2->get()[j]);
|
visit(*((*v2.list)[j] = state.allocValue()), i2->get()[j]);
|
||||||
} else
|
} else
|
||||||
visit(v2, i.second);
|
visit(v2, i.second);
|
||||||
}
|
}
|
||||||
|
@ -43,7 +43,7 @@ static void prim_fromTOML(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(v, size);
|
state.mkList(v, size);
|
||||||
|
|
||||||
for (size_t i = 0; i < size; ++i)
|
for (size_t i = 0; i < size; ++i)
|
||||||
visit(*(v.listElems()[i] = state.allocValue()), t2->get()[i]);
|
visit(*((*v.list)[i] = state.allocValue()), t2->get()[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle cases like 'a = [[{ a = true }]]', which IMHO should be
|
// Handle cases like 'a = [[{ a = true }]]', which IMHO should be
|
||||||
|
@ -56,7 +56,7 @@ static void prim_fromTOML(EvalState& state, const Pos& pos, Value** args,
|
||||||
state.mkList(v, size);
|
state.mkList(v, size);
|
||||||
|
|
||||||
for (size_t j = 0; j < size; ++j)
|
for (size_t j = 0; j < size; ++j)
|
||||||
visit(*(v.listElems()[j] = state.allocValue()), t2->get()[j]);
|
visit(*((*v.list)[j] = state.allocValue()), t2->get()[j]);
|
||||||
}
|
}
|
||||||
|
|
||||||
else if (t->is_value()) {
|
else if (t->is_value()) {
|
||||||
|
|
7
third_party/nix/src/libexpr/value-to-json.cc
vendored
7
third_party/nix/src/libexpr/value-to-json.cc
vendored
|
@ -64,14 +64,11 @@ void printValueAsJSON(EvalState& state, bool strict, Value& v,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case tList1:
|
case tList: {
|
||||||
case tList2:
|
|
||||||
case tListN: {
|
|
||||||
auto list(out.list());
|
auto list(out.list());
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
auto placeholder(list.placeholder());
|
auto placeholder(list.placeholder());
|
||||||
printValueAsJSON(state, strict, *v.listElems()[n], placeholder,
|
printValueAsJSON(state, strict, *(*v.list)[n], placeholder, context);
|
||||||
context);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
8
third_party/nix/src/libexpr/value-to-xml.cc
vendored
8
third_party/nix/src/libexpr/value-to-xml.cc
vendored
|
@ -127,13 +127,11 @@ static void printValueAsXML(EvalState& state, bool strict, bool location,
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case tList1:
|
case tList: {
|
||||||
case tList2:
|
|
||||||
case tListN: {
|
|
||||||
XMLOpenElement _(doc, "list");
|
XMLOpenElement _(doc, "list");
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
printValueAsXML(state, strict, location, *v.listElems()[n], doc,
|
printValueAsXML(state, strict, location, *(*v.list)[n], doc, context,
|
||||||
context, drvsSeen);
|
drvsSeen);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
37
third_party/nix/src/libexpr/value.hh
vendored
37
third_party/nix/src/libexpr/value.hh
vendored
|
@ -1,6 +1,10 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <tuple>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include <gc/gc_allocator.h>
|
#include <gc/gc_allocator.h>
|
||||||
|
#include <gc/gc_cpp.h>
|
||||||
|
|
||||||
#include "libexpr/symbol-table.hh"
|
#include "libexpr/symbol-table.hh"
|
||||||
#include "libutil/types.hh"
|
#include "libutil/types.hh"
|
||||||
|
@ -14,9 +18,7 @@ typedef enum {
|
||||||
tPath,
|
tPath,
|
||||||
tNull,
|
tNull,
|
||||||
tAttrs,
|
tAttrs,
|
||||||
tList1,
|
tList,
|
||||||
tList2,
|
|
||||||
tListN,
|
|
||||||
tThunk,
|
tThunk,
|
||||||
tApp,
|
tApp,
|
||||||
tLambda,
|
tLambda,
|
||||||
|
@ -119,11 +121,6 @@ struct NixString {
|
||||||
const char** context; // must be in sorted order
|
const char** context; // must be in sorted order
|
||||||
};
|
};
|
||||||
|
|
||||||
struct NixBigList {
|
|
||||||
size_t size;
|
|
||||||
Value** elems;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct NixThunk {
|
struct NixThunk {
|
||||||
Env* env;
|
Env* env;
|
||||||
Expr* expr;
|
Expr* expr;
|
||||||
|
@ -142,7 +139,9 @@ struct NixPrimOpApp {
|
||||||
Value *left, *right;
|
Value *left, *right;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct Value {
|
using NixList = std::vector<Value*, traceable_allocator<Value*>>;
|
||||||
|
|
||||||
|
struct Value : public gc {
|
||||||
ValueType type;
|
ValueType type;
|
||||||
union { // TODO(tazjin): std::variant
|
union { // TODO(tazjin): std::variant
|
||||||
NixInt integer;
|
NixInt integer;
|
||||||
|
@ -150,8 +149,7 @@ struct Value {
|
||||||
NixString string;
|
NixString string;
|
||||||
const char* path;
|
const char* path;
|
||||||
Bindings* attrs;
|
Bindings* attrs;
|
||||||
NixBigList bigList;
|
NixList* list;
|
||||||
Value* smallList[2];
|
|
||||||
NixThunk thunk;
|
NixThunk thunk;
|
||||||
NixApp app; // TODO(tazjin): "app"?
|
NixApp app; // TODO(tazjin): "app"?
|
||||||
NixLambda lambda;
|
NixLambda lambda;
|
||||||
|
@ -161,21 +159,9 @@ struct Value {
|
||||||
NixFloat fpoint;
|
NixFloat fpoint;
|
||||||
};
|
};
|
||||||
|
|
||||||
bool isList() const {
|
bool isList() const { return type == tList; }
|
||||||
return type == tList1 || type == tList2 || type == tListN;
|
|
||||||
}
|
|
||||||
|
|
||||||
Value** listElems() {
|
size_t listSize() const { return list->size(); }
|
||||||
return type == tList1 || type == tList2 ? smallList : bigList.elems;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Value* const* listElems() const {
|
|
||||||
return type == tList1 || type == tList2 ? smallList : bigList.elems;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t listSize() const {
|
|
||||||
return type == tList1 ? 1 : type == tList2 ? 2 : bigList.size;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/* After overwriting an app node, be sure to clear pointers in the
|
/* After overwriting an app node, be sure to clear pointers in the
|
||||||
|
@ -242,7 +228,6 @@ void mkPath(Value& v, const char* s);
|
||||||
not included. */
|
not included. */
|
||||||
size_t valueSize(Value& v);
|
size_t valueSize(Value& v);
|
||||||
|
|
||||||
typedef std::vector<Value*, traceable_allocator<Value*>> ValueVector;
|
|
||||||
typedef std::map<Symbol, Value*, std::less<Symbol>,
|
typedef std::map<Symbol, Value*, std::less<Symbol>,
|
||||||
traceable_allocator<std::pair<const Symbol, Value*>>>
|
traceable_allocator<std::pair<const Symbol, Value*>>>
|
||||||
ValueMap;
|
ValueMap;
|
||||||
|
|
|
@ -33,8 +33,8 @@ typedef enum {
|
||||||
wopQuerySubstitutablePathInfo = 21,
|
wopQuerySubstitutablePathInfo = 21,
|
||||||
wopQueryDerivationOutputs = 22,
|
wopQueryDerivationOutputs = 22,
|
||||||
wopQueryAllValidPaths = 23,
|
wopQueryAllValidPaths = 23,
|
||||||
wopQueryFailedPaths = 24, // obsolete
|
wopQueryFailedPaths = 24, // obsolete
|
||||||
wopClearFailedPaths = 25, // obsolete
|
wopClearFailedPaths = 25, // obsolete
|
||||||
wopQueryPathInfo = 26,
|
wopQueryPathInfo = 26,
|
||||||
wopImportPaths = 27, // obsolete
|
wopImportPaths = 27, // obsolete
|
||||||
wopQueryDerivationOutputNames = 28,
|
wopQueryDerivationOutputNames = 28,
|
||||||
|
|
7
third_party/nix/src/nix-env/nix-env.cc
vendored
7
third_party/nix/src/nix-env/nix-env.cc
vendored
|
@ -158,8 +158,7 @@ static void loadSourceExpr(EvalState& state, const Path& path, Value& v) {
|
||||||
directory). */
|
directory). */
|
||||||
else if (S_ISDIR(st.st_mode)) {
|
else if (S_ISDIR(st.st_mode)) {
|
||||||
state.mkAttrs(v, 1024);
|
state.mkAttrs(v, 1024);
|
||||||
state.mkList(*state.allocAttr(v, state.symbols.Create("_combineChannels")),
|
state.mkList(*state.allocAttr(v, state.symbols.Create("_combineChannels")));
|
||||||
0);
|
|
||||||
StringSet attrs;
|
StringSet attrs;
|
||||||
getAllExprs(state, path, attrs, v);
|
getAllExprs(state, path, attrs, v);
|
||||||
}
|
}
|
||||||
|
@ -1199,11 +1198,11 @@ static void opQuery(Globals& globals, Strings opFlags, Strings opArgs) {
|
||||||
attrs2["type"] = "strings";
|
attrs2["type"] = "strings";
|
||||||
XMLOpenElement m(xml, "meta", attrs2);
|
XMLOpenElement m(xml, "meta", attrs2);
|
||||||
for (unsigned int j = 0; j < v->listSize(); ++j) {
|
for (unsigned int j = 0; j < v->listSize(); ++j) {
|
||||||
if (v->listElems()[j]->type != tString) {
|
if ((*v->list)[j]->type != tString) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
XMLAttrs attrs3;
|
XMLAttrs attrs3;
|
||||||
attrs3["value"] = v->listElems()[j]->string.s;
|
attrs3["value"] = (*v->list)[j]->string.s;
|
||||||
xml.writeEmptyElement("string", attrs3);
|
xml.writeEmptyElement("string", attrs3);
|
||||||
}
|
}
|
||||||
} else if (v->type == tAttrs) {
|
} else if (v->type == tAttrs) {
|
||||||
|
|
24
third_party/nix/src/nix-env/user-env.cc
vendored
24
third_party/nix/src/nix-env/user-env.cc
vendored
|
@ -51,29 +51,29 @@ bool createUserEnv(EvalState& state, DrvInfos& elems, const Path& profile,
|
||||||
as the meta attributes. */
|
as the meta attributes. */
|
||||||
Path drvPath = keepDerivations ? i.queryDrvPath() : "";
|
Path drvPath = keepDerivations ? i.queryDrvPath() : "";
|
||||||
|
|
||||||
Value& v(*state.allocValue());
|
Value* v = state.allocValue();
|
||||||
manifest.listElems()[n++] = &v;
|
(*manifest.list)[n++] = v;
|
||||||
state.mkAttrs(v, 16);
|
state.mkAttrs(*v, 16);
|
||||||
|
|
||||||
mkString(*state.allocAttr(v, state.sType), "derivation");
|
mkString(*state.allocAttr(*v, state.sType), "derivation");
|
||||||
mkString(*state.allocAttr(v, state.sName), i.queryName());
|
mkString(*state.allocAttr(*v, state.sName), i.queryName());
|
||||||
auto system = i.querySystem();
|
auto system = i.querySystem();
|
||||||
if (!system.empty()) {
|
if (!system.empty()) {
|
||||||
mkString(*state.allocAttr(v, state.sSystem), system);
|
mkString(*state.allocAttr(*v, state.sSystem), system);
|
||||||
}
|
}
|
||||||
mkString(*state.allocAttr(v, state.sOutPath), i.queryOutPath());
|
mkString(*state.allocAttr(*v, state.sOutPath), i.queryOutPath());
|
||||||
if (!drvPath.empty()) {
|
if (!drvPath.empty()) {
|
||||||
mkString(*state.allocAttr(v, state.sDrvPath), i.queryDrvPath());
|
mkString(*state.allocAttr(*v, state.sDrvPath), i.queryDrvPath());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy each output meant for installation.
|
// Copy each output meant for installation.
|
||||||
DrvInfo::Outputs outputs = i.queryOutputs(true);
|
DrvInfo::Outputs outputs = i.queryOutputs(true);
|
||||||
Value& vOutputs = *state.allocAttr(v, state.sOutputs);
|
Value& vOutputs = *state.allocAttr(*v, state.sOutputs);
|
||||||
state.mkList(vOutputs, outputs.size());
|
state.mkList(vOutputs, outputs.size());
|
||||||
unsigned int m = 0;
|
unsigned int m = 0;
|
||||||
for (auto& j : outputs) {
|
for (auto& j : outputs) {
|
||||||
mkString(*(vOutputs.listElems()[m++] = state.allocValue()), j.first);
|
mkString(*((*vOutputs.list)[m++] = state.allocValue()), j.first);
|
||||||
Value& vOutputs = *state.allocAttr(v, state.symbols.Create(j.first));
|
Value& vOutputs = *state.allocAttr(*v, state.symbols.Create(j.first));
|
||||||
state.mkAttrs(vOutputs, 2);
|
state.mkAttrs(vOutputs, 2);
|
||||||
mkString(*state.allocAttr(vOutputs, state.sOutPath), j.second);
|
mkString(*state.allocAttr(vOutputs, state.sOutPath), j.second);
|
||||||
|
|
||||||
|
@ -86,7 +86,7 @@ bool createUserEnv(EvalState& state, DrvInfos& elems, const Path& profile,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy the meta attributes.
|
// Copy the meta attributes.
|
||||||
Value& vMeta = *state.allocAttr(v, state.sMeta);
|
Value& vMeta = *state.allocAttr(*v, state.sMeta);
|
||||||
state.mkAttrs(vMeta, 16);
|
state.mkAttrs(vMeta, 16);
|
||||||
StringSet metaNames = i.queryMetaNames();
|
StringSet metaNames = i.queryMetaNames();
|
||||||
for (auto& j : metaNames) {
|
for (auto& j : metaNames) {
|
||||||
|
|
|
@ -50,8 +50,7 @@ std::string resolveMirrorUri(EvalState& state, std::string uri) {
|
||||||
throw Error(format("mirror URI '%1%' did not expand to anything") % uri);
|
throw Error(format("mirror URI '%1%' did not expand to anything") % uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string mirror =
|
std::string mirror = state.forceString(*(*mirrorList->second.value->list)[0]);
|
||||||
state.forceString(*mirrorList->second.value->listElems()[0]);
|
|
||||||
return mirror + (absl::EndsWith(mirror, "/") ? "" : "/") +
|
return mirror + (absl::EndsWith(mirror, "/") ? "" : "/") +
|
||||||
std::string(s, p + 1);
|
std::string(s, p + 1);
|
||||||
}
|
}
|
||||||
|
@ -137,7 +136,7 @@ static int _main(int argc, char** argv) {
|
||||||
if (attr->second.value->listSize() < 1) {
|
if (attr->second.value->listSize() < 1) {
|
||||||
throw Error("'urls' list is empty");
|
throw Error("'urls' list is empty");
|
||||||
}
|
}
|
||||||
uri = state->forceString(*attr->second.value->listElems()[0]);
|
uri = state->forceString(*(*attr->second.value->list)[0]);
|
||||||
|
|
||||||
/* Extract the hash mode. */
|
/* Extract the hash mode. */
|
||||||
attr = v.attrs->find(state->symbols.Create("outputHashMode"));
|
attr = v.attrs->find(state->symbols.Create("outputHashMode"));
|
||||||
|
|
8
third_party/nix/src/nix/repl.cc
vendored
8
third_party/nix/src/nix/repl.cc
vendored
|
@ -748,19 +748,17 @@ std::ostream& NixRepl::printValue(std::ostream& str, Value& v,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case tList1:
|
case tList:
|
||||||
case tList2:
|
|
||||||
case tListN:
|
|
||||||
seen.insert(&v);
|
seen.insert(&v);
|
||||||
|
|
||||||
str << "[ ";
|
str << "[ ";
|
||||||
if (maxDepth > 0) {
|
if (maxDepth > 0) {
|
||||||
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
for (unsigned int n = 0; n < v.listSize(); ++n) {
|
||||||
if (seen.find(v.listElems()[n]) != seen.end()) {
|
if (seen.find((*v.list)[n]) != seen.end()) {
|
||||||
str << "«repeated»";
|
str << "«repeated»";
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
printValue(str, *v.listElems()[n], maxDepth - 1, seen);
|
printValue(str, *(*v.list)[n], maxDepth - 1, seen);
|
||||||
} catch (AssertionError& e) {
|
} catch (AssertionError& e) {
|
||||||
str << ESC_RED "«error: " << e.msg() << "»" ESC_END;
|
str << ESC_RED "«error: " << e.msg() << "»" ESC_END;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue