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) {
|
||||
/* Remove the environment while we are holding the global
|
||||
lock. If things go wrong there, we bail out. !!!
|
||||
there is some leakage here op DbEnv and lock
|
||||
handles. */
|
||||
lock. If things go wrong there, we bail out.
|
||||
!!! argh, we abolished the global lock :-( */
|
||||
open2(path, true);
|
||||
|
||||
/* Try again. */
|
||||
|
@ -311,7 +310,7 @@ void Database::close()
|
|||
}
|
||||
|
||||
|
||||
TableId Database::openTable(const string & tableName)
|
||||
TableId Database::openTable(const string & tableName, bool sorted)
|
||||
{
|
||||
requireEnv();
|
||||
TableId table = nextId++;
|
||||
|
@ -322,7 +321,8 @@ TableId Database::openTable(const string & tableName)
|
|||
|
||||
try {
|
||||
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 (...) {
|
||||
delete db;
|
||||
throw;
|
||||
|
@ -410,7 +410,7 @@ void Database::delPair(const Transaction & txn, TableId table,
|
|||
|
||||
|
||||
void Database::enumTable(const Transaction & txn, TableId table,
|
||||
Strings & keys)
|
||||
Strings & keys, const string & keyPrefix)
|
||||
{
|
||||
try {
|
||||
Db * db = getDb(table);
|
||||
|
@ -420,10 +420,21 @@ void Database::enumTable(const Transaction & txn, TableId table,
|
|||
DestroyDbc destroyDbc(dbc);
|
||||
|
||||
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();
|
||||
keys.push_back(
|
||||
string((char *) kt.get_data(), kt.get_size()));
|
||||
string data((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); }
|
||||
|
|
|
@ -64,7 +64,7 @@ public:
|
|||
void open(const string & path);
|
||||
void close();
|
||||
|
||||
TableId openTable(const string & table);
|
||||
TableId openTable(const string & table, bool sorted = false);
|
||||
|
||||
bool queryString(const Transaction & txn, TableId table,
|
||||
const string & key, string & data);
|
||||
|
@ -83,7 +83,7 @@ public:
|
|||
const string & key);
|
||||
|
||||
void enumTable(const Transaction & txn, TableId table,
|
||||
Strings & keys);
|
||||
Strings & keys, const string & keyPrefix = "");
|
||||
};
|
||||
|
||||
|
||||
|
|
|
@ -34,10 +34,12 @@ static TableId dbValidPaths = 0;
|
|||
paths. */
|
||||
static TableId dbReferences = 0;
|
||||
|
||||
/* dbReferers :: Path -> [Path]
|
||||
/* dbReferrers :: Path -> Path
|
||||
|
||||
This table is just the reverse mapping of dbReferences. */
|
||||
static TableId dbReferers = 0;
|
||||
This table is just the reverse mapping of dbReferences. This table
|
||||
can have duplicate keys, each corresponding value denoting a single
|
||||
referrer. */
|
||||
static TableId dbReferrers = 0;
|
||||
|
||||
/* 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()
|
||||
|
@ -86,7 +89,7 @@ void openDB()
|
|||
}
|
||||
dbValidPaths = nixDB.openTable("validpaths");
|
||||
dbReferences = nixDB.openTable("references");
|
||||
dbReferers = nixDB.openTable("referers");
|
||||
dbReferrers = nixDB.openTable("referrers", true); /* must be sorted */
|
||||
dbSubstitutes = nixDB.openTable("substitutes");
|
||||
dbDerivers = nixDB.openTable("derivers");
|
||||
|
||||
|
@ -103,7 +106,10 @@ void openDB()
|
|||
% curSchema % nixSchemaVersion);
|
||||
|
||||
if (curSchema < nixSchemaVersion) {
|
||||
upgradeStore();
|
||||
if (curSchema <= 1)
|
||||
upgradeStore07();
|
||||
if (curSchema == 2)
|
||||
upgradeStore09();
|
||||
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)
|
||||
{
|
||||
Paths referers;
|
||||
nixDB.queryStrings(txn, dbReferers, storePath, referers);
|
||||
return PathSet(referers.begin(), referers.end());
|
||||
PathSet referrers;
|
||||
Strings keys;
|
||||
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,
|
||||
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();
|
||||
i != references.end(); ++i)
|
||||
{
|
||||
PathSet referers = getReferers(txn, *i);
|
||||
referers.insert(storePath);
|
||||
nixDB.setStrings(txn, dbReferers, *i,
|
||||
Paths(referers.begin(), referers.end()));
|
||||
}
|
||||
if (oldReferences2.find(*i) == oldReferences2.end())
|
||||
nixDB.setString(txn, dbReferrers, addPrefix(*i, storePath), "");
|
||||
|
||||
/* Remove referer mappings from paths that are no longer
|
||||
references. */
|
||||
for (Paths::iterator i = oldReferences.begin();
|
||||
i != oldReferences.end(); ++i)
|
||||
if (references.find(*i) == references.end()) {
|
||||
PathSet referers = getReferers(txn, *i);
|
||||
referers.erase(storePath);
|
||||
nixDB.setStrings(txn, dbReferers, *i,
|
||||
Paths(referers.begin(), referers.end()));
|
||||
}
|
||||
if (references.find(*i) == references.end())
|
||||
nixDB.delPair(txn, dbReferrers, addPrefix(*i, storePath));
|
||||
}
|
||||
|
||||
|
||||
|
@ -840,13 +858,11 @@ void verifyStore(bool checkContents)
|
|||
for (PathSet::iterator j = references.begin();
|
||||
j != references.end(); ++j)
|
||||
{
|
||||
PathSet referers = getReferers(txn, *j);
|
||||
if (referers.find(*i) == referers.end()) {
|
||||
string dummy;
|
||||
if (!nixDB.queryString(txn, dbReferrers, addPrefix(*j, *i), dummy)) {
|
||||
printMsg(lvlError, format("missing referer mapping from `%1%' to `%2%'")
|
||||
% *j % *i);
|
||||
referers.insert(*i);
|
||||
nixDB.setStrings(txn, dbReferers, *j,
|
||||
Paths(referers.begin(), referers.end()));
|
||||
nixDB.setString(txn, dbReferrers, addPrefix(*j, *i), "");
|
||||
}
|
||||
if (isValid && validPaths.find(*j) == validPaths.end()) {
|
||||
printMsg(lvlError, format("incomplete closure: `%1%' needs missing `%2%'")
|
||||
|
@ -856,6 +872,7 @@ void verifyStore(bool checkContents)
|
|||
}
|
||||
}
|
||||
|
||||
#if 0 // !!!
|
||||
/* Check the `referers' table. */
|
||||
Paths referersKeys;
|
||||
nixDB.enumTable(txn, dbReferers, referersKeys);
|
||||
|
@ -892,6 +909,7 @@ void verifyStore(bool checkContents)
|
|||
Paths(newReferers.begin(), newReferers.end()));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
txn.commit();
|
||||
}
|
||||
|
@ -902,7 +920,7 @@ void verifyStore(bool checkContents)
|
|||
|
||||
|
||||
/* 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)...");
|
||||
|
||||
|
@ -986,3 +1004,38 @@ static void upgradeStore()
|
|||
/* !!! maybe this transaction is way too big */
|
||||
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;
|
||||
|
||||
|
||||
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
|
||||
|
|
Loading…
Reference in a new issue