forked from lix-project/lix
0746951be1
* Finish converting existing comments for internal API docs 99% of this was just reformatting existing comments. Only two exceptions: - Expanded upon `BuildResult::status` compat note - Split up file-level `symbol-table.hh` doc comments to get per-definition docs Also fixed a few whitespace goofs, turning leading tabs to spaces and removing trailing spaces. Picking up from #8133 * Fix two things from comments * Use triple-backtick not indent for `dumpPath` * Convert GNU-style `\`..'` quotes to markdown style in API docs This will render correctly.
171 lines
4 KiB
C++
171 lines
4 KiB
C++
#pragma once
|
|
///@file
|
|
|
|
#include "nixexpr.hh"
|
|
#include "symbol-table.hh"
|
|
|
|
#include <algorithm>
|
|
#include <optional>
|
|
|
|
namespace nix {
|
|
|
|
|
|
class EvalState;
|
|
struct Value;
|
|
|
|
/**
|
|
* Map one attribute name to its value.
|
|
*/
|
|
struct Attr
|
|
{
|
|
/* the placement of `name` and `pos` in this struct is important.
|
|
both of them are uint32 wrappers, they are next to each other
|
|
to make sure that Attr has no padding on 64 bit machines. that
|
|
way we keep Attr size at two words with no wasted space. */
|
|
Symbol name;
|
|
PosIdx pos;
|
|
Value * value;
|
|
Attr(Symbol name, Value * value, PosIdx pos = noPos)
|
|
: name(name), pos(pos), value(value) { };
|
|
Attr() { };
|
|
bool operator < (const Attr & a) const
|
|
{
|
|
return name < a.name;
|
|
}
|
|
};
|
|
|
|
static_assert(sizeof(Attr) == 2 * sizeof(uint32_t) + sizeof(Value *),
|
|
"performance of the evaluator is highly sensitive to the size of Attr. "
|
|
"avoid introducing any padding into Attr if at all possible, and do not "
|
|
"introduce new fields that need not be present for almost every instance.");
|
|
|
|
/**
|
|
* Bindings contains all the attributes of an attribute set. It is defined
|
|
* by its size and its capacity, the capacity being the number of Attr
|
|
* elements allocated after this structure, while the size corresponds to
|
|
* the number of elements already inserted in this structure.
|
|
*/
|
|
class Bindings
|
|
{
|
|
public:
|
|
typedef uint32_t size_t;
|
|
PosIdx pos;
|
|
|
|
private:
|
|
size_t size_, capacity_;
|
|
Attr attrs[0];
|
|
|
|
Bindings(size_t capacity) : size_(0), capacity_(capacity) { }
|
|
Bindings(const Bindings & bindings) = delete;
|
|
|
|
public:
|
|
size_t size() const { return size_; }
|
|
|
|
bool empty() const { return !size_; }
|
|
|
|
typedef Attr * iterator;
|
|
|
|
void push_back(const Attr & attr)
|
|
{
|
|
assert(size_ < capacity_);
|
|
attrs[size_++] = attr;
|
|
}
|
|
|
|
iterator find(Symbol name)
|
|
{
|
|
Attr key(name, 0);
|
|
iterator i = std::lower_bound(begin(), end(), key);
|
|
if (i != end() && i->name == name) return i;
|
|
return end();
|
|
}
|
|
|
|
Attr * get(Symbol name)
|
|
{
|
|
Attr key(name, 0);
|
|
iterator i = std::lower_bound(begin(), end(), key);
|
|
if (i != end() && i->name == name) return &*i;
|
|
return nullptr;
|
|
}
|
|
|
|
iterator begin() { return &attrs[0]; }
|
|
iterator end() { return &attrs[size_]; }
|
|
|
|
Attr & operator[](size_t pos)
|
|
{
|
|
return attrs[pos];
|
|
}
|
|
|
|
void sort();
|
|
|
|
size_t capacity() { return capacity_; }
|
|
|
|
/**
|
|
* Returns the attributes in lexicographically sorted order.
|
|
*/
|
|
std::vector<const Attr *> lexicographicOrder(const SymbolTable & symbols) const
|
|
{
|
|
std::vector<const Attr *> res;
|
|
res.reserve(size_);
|
|
for (size_t n = 0; n < size_; n++)
|
|
res.emplace_back(&attrs[n]);
|
|
std::sort(res.begin(), res.end(), [&](const Attr * a, const Attr * b) {
|
|
std::string_view sa = symbols[a->name], sb = symbols[b->name];
|
|
return sa < sb;
|
|
});
|
|
return res;
|
|
}
|
|
|
|
friend class EvalState;
|
|
};
|
|
|
|
/**
|
|
* A wrapper around Bindings that ensures that its always in sorted
|
|
* order at the end. The only way to consume a BindingsBuilder is to
|
|
* call finish(), which sorts the bindings.
|
|
*/
|
|
class BindingsBuilder
|
|
{
|
|
Bindings * bindings;
|
|
|
|
public:
|
|
// needed by std::back_inserter
|
|
using value_type = Attr;
|
|
|
|
EvalState & state;
|
|
|
|
BindingsBuilder(EvalState & state, Bindings * bindings)
|
|
: bindings(bindings), state(state)
|
|
{ }
|
|
|
|
void insert(Symbol name, Value * value, PosIdx pos = noPos)
|
|
{
|
|
insert(Attr(name, value, pos));
|
|
}
|
|
|
|
void insert(const Attr & attr)
|
|
{
|
|
push_back(attr);
|
|
}
|
|
|
|
void push_back(const Attr & attr)
|
|
{
|
|
bindings->push_back(attr);
|
|
}
|
|
|
|
Value & alloc(Symbol name, PosIdx pos = noPos);
|
|
|
|
Value & alloc(std::string_view name, PosIdx pos = noPos);
|
|
|
|
Bindings * finish()
|
|
{
|
|
bindings->sort();
|
|
return bindings;
|
|
}
|
|
|
|
Bindings * alreadySorted()
|
|
{
|
|
return bindings;
|
|
}
|
|
};
|
|
|
|
}
|