forked from lix-project/lix
* Fix NIX-23: quadratic complexity in maintaining the referers
mapping. The referer table is replaced by a referrer table (note spelling fix) that stores each referrer separately. That is, instead of having referer[P] = {Q_1, Q_2, Q_3, ...} we store referer[(P, Q_1)] = "" referer[(P, Q_2)] = "" referer[(P, Q_3)] = "" ... To find the referrers of P, we enumerate over the keys with a value lexicographically greater than P. This requires the referrer table to be stored as a B-Tree rather than a hash table. (The tuples (P, Q) are stored as P + null-byte + Q.) Old Nix databases are upgraded automatically to the new schema.
This commit is contained in:
parent
18bbcb1214
commit
8463f27d8c
4 changed files with 107 additions and 40 deletions
|
@ -257,9 +257,8 @@ void Database::open(const string & path)
|
||||||
|
|
||||||
if (e.get_errno() == DB_VERSION_MISMATCH) {
|
if (e.get_errno() == DB_VERSION_MISMATCH) {
|
||||||
/* Remove the environment while we are holding the global
|
/* Remove the environment while we are holding the global
|
||||||
lock. If things go wrong there, we bail out. !!!
|
lock. If things go wrong there, we bail out.
|
||||||
there is some leakage here op DbEnv and lock
|
!!! argh, we abolished the global lock :-( */
|
||||||
handles. */
|
|
||||||
open2(path, true);
|
open2(path, true);
|
||||||
|
|
||||||
/* Try again. */
|
/* Try again. */
|
||||||
|
@ -311,7 +310,7 @@ void Database::close()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
TableId Database::openTable(const string & tableName)
|
TableId Database::openTable(const string & tableName, bool sorted)
|
||||||
{
|
{
|
||||||
requireEnv();
|
requireEnv();
|
||||||
TableId table = nextId++;
|
TableId table = nextId++;
|
||||||
|
@ -322,7 +321,8 @@ TableId Database::openTable(const string & tableName)
|
||||||
|
|
||||||
try {
|
try {
|
||||||
db->open(0, tableName.c_str(), 0,
|
db->open(0, tableName.c_str(), 0,
|
||||||
DB_HASH, DB_CREATE | DB_AUTO_COMMIT, 0666);
|
sorted ? DB_BTREE : DB_HASH,
|
||||||
|
DB_CREATE | DB_AUTO_COMMIT, 0666);
|
||||||
} catch (...) {
|
} catch (...) {
|
||||||
delete db;
|
delete db;
|
||||||
throw;
|
throw;
|
||||||
|
@ -410,7 +410,7 @@ void Database::delPair(const Transaction & txn, TableId table,
|
||||||
|
|
||||||
|
|
||||||
void Database::enumTable(const Transaction & txn, TableId table,
|
void Database::enumTable(const Transaction & txn, TableId table,
|
||||||
Strings & keys)
|
Strings & keys, const string & keyPrefix)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
Db * db = getDb(table);
|
Db * db = getDb(table);
|
||||||
|
@ -420,10 +420,21 @@ void Database::enumTable(const Transaction & txn, TableId table,
|
||||||
DestroyDbc destroyDbc(dbc);
|
DestroyDbc destroyDbc(dbc);
|
||||||
|
|
||||||
Dbt kt, dt;
|
Dbt kt, dt;
|
||||||
while (dbc->get(&kt, &dt, DB_NEXT) != DB_NOTFOUND) {
|
u_int32_t flags = DB_NEXT;
|
||||||
|
|
||||||
|
if (!keyPrefix.empty()) {
|
||||||
|
flags = DB_SET_RANGE;
|
||||||
|
kt = Dbt((void *) keyPrefix.c_str(), keyPrefix.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
while (dbc->get(&kt, &dt, flags) != DB_NOTFOUND) {
|
||||||
checkInterrupt();
|
checkInterrupt();
|
||||||
keys.push_back(
|
string data((char *) kt.get_data(), kt.get_size());
|
||||||
string((char *) kt.get_data(), kt.get_size()));
|
if (!keyPrefix.empty() &&
|
||||||
|
data.compare(0, keyPrefix.size(), keyPrefix) != 0)
|
||||||
|
break;
|
||||||
|
keys.push_back(data);
|
||||||
|
flags = DB_NEXT;
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (DbException e) { rethrow(e); }
|
} catch (DbException e) { rethrow(e); }
|
||||||
|
|
|
@ -64,7 +64,7 @@ public:
|
||||||
void open(const string & path);
|
void open(const string & path);
|
||||||
void close();
|
void close();
|
||||||
|
|
||||||
TableId openTable(const string & table);
|
TableId openTable(const string & table, bool sorted = false);
|
||||||
|
|
||||||
bool queryString(const Transaction & txn, TableId table,
|
bool queryString(const Transaction & txn, TableId table,
|
||||||
const string & key, string & data);
|
const string & key, string & data);
|
||||||
|
@ -83,7 +83,7 @@ public:
|
||||||
const string & key);
|
const string & key);
|
||||||
|
|
||||||
void enumTable(const Transaction & txn, TableId table,
|
void enumTable(const Transaction & txn, TableId table,
|
||||||
Strings & keys);
|
Strings & keys, const string & keyPrefix = "");
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -34,10 +34,12 @@ static TableId dbValidPaths = 0;
|
||||||
paths. */
|
paths. */
|
||||||
static TableId dbReferences = 0;
|
static TableId dbReferences = 0;
|
||||||
|
|
||||||
/* dbReferers :: Path -> [Path]
|
/* dbReferrers :: Path -> Path
|
||||||
|
|
||||||
This table is just the reverse mapping of dbReferences. */
|
This table is just the reverse mapping of dbReferences. This table
|
||||||
static TableId dbReferers = 0;
|
can have duplicate keys, each corresponding value denoting a single
|
||||||
|
referrer. */
|
||||||
|
static TableId dbReferrers = 0;
|
||||||
|
|
||||||
/* dbSubstitutes :: Path -> [[Path]]
|
/* dbSubstitutes :: Path -> [[Path]]
|
||||||
|
|
||||||
|
@ -70,7 +72,8 @@ bool Substitute::operator == (const Substitute & sub) const
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void upgradeStore();
|
static void upgradeStore07();
|
||||||
|
static void upgradeStore09();
|
||||||
|
|
||||||
|
|
||||||
void openDB()
|
void openDB()
|
||||||
|
@ -86,7 +89,7 @@ void openDB()
|
||||||
}
|
}
|
||||||
dbValidPaths = nixDB.openTable("validpaths");
|
dbValidPaths = nixDB.openTable("validpaths");
|
||||||
dbReferences = nixDB.openTable("references");
|
dbReferences = nixDB.openTable("references");
|
||||||
dbReferers = nixDB.openTable("referers");
|
dbReferrers = nixDB.openTable("referrers", true); /* must be sorted */
|
||||||
dbSubstitutes = nixDB.openTable("substitutes");
|
dbSubstitutes = nixDB.openTable("substitutes");
|
||||||
dbDerivers = nixDB.openTable("derivers");
|
dbDerivers = nixDB.openTable("derivers");
|
||||||
|
|
||||||
|
@ -103,7 +106,10 @@ void openDB()
|
||||||
% curSchema % nixSchemaVersion);
|
% curSchema % nixSchemaVersion);
|
||||||
|
|
||||||
if (curSchema < nixSchemaVersion) {
|
if (curSchema < nixSchemaVersion) {
|
||||||
upgradeStore();
|
if (curSchema <= 1)
|
||||||
|
upgradeStore07();
|
||||||
|
if (curSchema == 2)
|
||||||
|
upgradeStore09();
|
||||||
writeFile(schemaFN, (format("%1%") % nixSchemaVersion).str());
|
writeFile(schemaFN, (format("%1%") % nixSchemaVersion).str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -286,11 +292,31 @@ static bool isRealisablePath(const Transaction & txn, const Path & path)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static string addPrefix(const string & prefix, const string & s)
|
||||||
|
{
|
||||||
|
return prefix + string(1, 0) + s;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static string stripPrefix(const string & prefix, const string & s)
|
||||||
|
{
|
||||||
|
if (s.size() <= prefix.size() ||
|
||||||
|
s.compare(0, prefix.size(), prefix) != 0 ||
|
||||||
|
s[prefix.size()] != 0)
|
||||||
|
throw Error(format("string `%1%' is missing prefix `%2%'")
|
||||||
|
% s % prefix);
|
||||||
|
return string(s, prefix.size() + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
static PathSet getReferers(const Transaction & txn, const Path & storePath)
|
static PathSet getReferers(const Transaction & txn, const Path & storePath)
|
||||||
{
|
{
|
||||||
Paths referers;
|
PathSet referrers;
|
||||||
nixDB.queryStrings(txn, dbReferers, storePath, referers);
|
Strings keys;
|
||||||
return PathSet(referers.begin(), referers.end());
|
nixDB.enumTable(txn, dbReferrers, keys, storePath + string(1, 0));
|
||||||
|
for (Strings::iterator i = keys.begin(); i != keys.end(); ++i)
|
||||||
|
referrers.insert(stripPrefix(storePath, *i));
|
||||||
|
return referrers;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -312,26 +338,18 @@ void setReferences(const Transaction & txn, const Path & storePath,
|
||||||
nixDB.setStrings(txn, dbReferences, storePath,
|
nixDB.setStrings(txn, dbReferences, storePath,
|
||||||
Paths(references.begin(), references.end()));
|
Paths(references.begin(), references.end()));
|
||||||
|
|
||||||
/* Update the referers mappings of all referenced paths. */
|
/* Update the referers mappings of all new referenced paths. */
|
||||||
for (PathSet::const_iterator i = references.begin();
|
for (PathSet::const_iterator i = references.begin();
|
||||||
i != references.end(); ++i)
|
i != references.end(); ++i)
|
||||||
{
|
if (oldReferences2.find(*i) == oldReferences2.end())
|
||||||
PathSet referers = getReferers(txn, *i);
|
nixDB.setString(txn, dbReferrers, addPrefix(*i, storePath), "");
|
||||||
referers.insert(storePath);
|
|
||||||
nixDB.setStrings(txn, dbReferers, *i,
|
|
||||||
Paths(referers.begin(), referers.end()));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Remove referer mappings from paths that are no longer
|
/* Remove referer mappings from paths that are no longer
|
||||||
references. */
|
references. */
|
||||||
for (Paths::iterator i = oldReferences.begin();
|
for (Paths::iterator i = oldReferences.begin();
|
||||||
i != oldReferences.end(); ++i)
|
i != oldReferences.end(); ++i)
|
||||||
if (references.find(*i) == references.end()) {
|
if (references.find(*i) == references.end())
|
||||||
PathSet referers = getReferers(txn, *i);
|
nixDB.delPair(txn, dbReferrers, addPrefix(*i, storePath));
|
||||||
referers.erase(storePath);
|
|
||||||
nixDB.setStrings(txn, dbReferers, *i,
|
|
||||||
Paths(referers.begin(), referers.end()));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -840,13 +858,11 @@ void verifyStore(bool checkContents)
|
||||||
for (PathSet::iterator j = references.begin();
|
for (PathSet::iterator j = references.begin();
|
||||||
j != references.end(); ++j)
|
j != references.end(); ++j)
|
||||||
{
|
{
|
||||||
PathSet referers = getReferers(txn, *j);
|
string dummy;
|
||||||
if (referers.find(*i) == referers.end()) {
|
if (!nixDB.queryString(txn, dbReferrers, addPrefix(*j, *i), dummy)) {
|
||||||
printMsg(lvlError, format("missing referer mapping from `%1%' to `%2%'")
|
printMsg(lvlError, format("missing referer mapping from `%1%' to `%2%'")
|
||||||
% *j % *i);
|
% *j % *i);
|
||||||
referers.insert(*i);
|
nixDB.setString(txn, dbReferrers, addPrefix(*j, *i), "");
|
||||||
nixDB.setStrings(txn, dbReferers, *j,
|
|
||||||
Paths(referers.begin(), referers.end()));
|
|
||||||
}
|
}
|
||||||
if (isValid && validPaths.find(*j) == validPaths.end()) {
|
if (isValid && validPaths.find(*j) == validPaths.end()) {
|
||||||
printMsg(lvlError, format("incomplete closure: `%1%' needs missing `%2%'")
|
printMsg(lvlError, format("incomplete closure: `%1%' needs missing `%2%'")
|
||||||
|
@ -856,6 +872,7 @@ void verifyStore(bool checkContents)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if 0 // !!!
|
||||||
/* Check the `referers' table. */
|
/* Check the `referers' table. */
|
||||||
Paths referersKeys;
|
Paths referersKeys;
|
||||||
nixDB.enumTable(txn, dbReferers, referersKeys);
|
nixDB.enumTable(txn, dbReferers, referersKeys);
|
||||||
|
@ -892,6 +909,7 @@ void verifyStore(bool checkContents)
|
||||||
Paths(newReferers.begin(), newReferers.end()));
|
Paths(newReferers.begin(), newReferers.end()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
txn.commit();
|
txn.commit();
|
||||||
}
|
}
|
||||||
|
@ -902,7 +920,7 @@ void verifyStore(bool checkContents)
|
||||||
|
|
||||||
|
|
||||||
/* Upgrade from schema 1 (Nix <= 0.7) to schema 2 (Nix >= 0.8). */
|
/* Upgrade from schema 1 (Nix <= 0.7) to schema 2 (Nix >= 0.8). */
|
||||||
static void upgradeStore()
|
static void upgradeStore07()
|
||||||
{
|
{
|
||||||
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
||||||
|
|
||||||
|
@ -986,3 +1004,38 @@ static void upgradeStore()
|
||||||
/* !!! maybe this transaction is way too big */
|
/* !!! maybe this transaction is way too big */
|
||||||
txn.commit();
|
txn.commit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Upgrade from schema 2 (0.8 <= Nix <= 0.9) to schema 3 (Nix >=
|
||||||
|
0.10). The only thing to do here is to upgrade the old `referer'
|
||||||
|
table (which causes quadratic complexity in some cases) to the new
|
||||||
|
(and properly spelled) `referrer' table. */
|
||||||
|
static void upgradeStore09()
|
||||||
|
{
|
||||||
|
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
||||||
|
|
||||||
|
if (!pathExists(nixDBPath + "/referers")) return;
|
||||||
|
|
||||||
|
Transaction txn(nixDB);
|
||||||
|
|
||||||
|
cerr << "converting referers to referrers...";
|
||||||
|
|
||||||
|
TableId dbReferers = nixDB.openTable("referers"); /* sic! */
|
||||||
|
|
||||||
|
Paths referersKeys;
|
||||||
|
nixDB.enumTable(txn, dbReferers, referersKeys);
|
||||||
|
for (Paths::iterator i = referersKeys.begin();
|
||||||
|
i != referersKeys.end(); ++i)
|
||||||
|
{
|
||||||
|
Paths referers;
|
||||||
|
nixDB.queryStrings(txn, dbReferers, *i, referers);
|
||||||
|
for (Paths::iterator j = referers.begin();
|
||||||
|
j != referers.end(); ++j)
|
||||||
|
nixDB.setString(txn, dbReferrers, addPrefix(*i, *j), "");
|
||||||
|
cerr << ".";
|
||||||
|
}
|
||||||
|
|
||||||
|
cerr << "\n";
|
||||||
|
|
||||||
|
txn.commit();
|
||||||
|
}
|
||||||
|
|
|
@ -9,7 +9,10 @@
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
const int nixSchemaVersion = 2;
|
/* Nix store and database schema version. Version 1 (or 0) was Nix <=
|
||||||
|
0.7. Version 2 was Nix 0.8 and 0.8. Version 3 is Nix 0.10 and
|
||||||
|
up. */
|
||||||
|
const int nixSchemaVersion = 3;
|
||||||
|
|
||||||
|
|
||||||
/* A substitute is a program invocation that constructs some store
|
/* A substitute is a program invocation that constructs some store
|
||||||
|
|
Loading…
Reference in a new issue