aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorEelco Dolstra <e.dolstra@tudelft.nl>2005-12-12 18:24:42 +0000
committerEelco Dolstra <e.dolstra@tudelft.nl>2005-12-12 18:24:42 +0000
commit8463f27d8cd09d54648de21c747f895cdb558f83 (patch)
tree75bb48b23486d01c8891e8b259b344dbdafc6b86
parent18bbcb1214caf75c0190e610d7eb34e971366c7c (diff)
* 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.
-rw-r--r--src/libstore/db.cc29
-rw-r--r--src/libstore/db.hh4
-rw-r--r--src/libstore/store.cc109
-rw-r--r--src/libstore/store.hh5
4 files changed, 107 insertions, 40 deletions
diff --git a/src/libstore/db.cc b/src/libstore/db.cc
index 072b0bb97..74c933d0e 100644
--- a/src/libstore/db.cc
+++ b/src/libstore/db.cc
@@ -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); }
diff --git a/src/libstore/db.hh b/src/libstore/db.hh
index 76afc1d3b..b7ebd4467 100644
--- a/src/libstore/db.hh
+++ b/src/libstore/db.hh
@@ -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 = "");
};
diff --git a/src/libstore/store.cc b/src/libstore/store.cc
index 4e163605a..fbbf12269 100644
--- a/src/libstore/store.cc
+++ b/src/libstore/store.cc
@@ -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();
+}
diff --git a/src/libstore/store.hh b/src/libstore/store.hh
index bcf11ce96..2b83bd7fe 100644
--- a/src/libstore/store.hh
+++ b/src/libstore/store.hh
@@ -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