forked from lix-project/lix
Make roots a map of store paths to pinning links
This new structure makes more sense as there may be many sources rooting the same store path. Many profiles can reference the same path but this is even more true with /proc/<pid>/maps where distinct pids can and often do map the same store path. This implementation is also more efficient as the `Roots` map contains only one entry per rooted store path.
This commit is contained in:
parent
a17f86ce3a
commit
ebc86550f9
5 changed files with 39 additions and 35 deletions
|
@ -130,7 +130,7 @@ Path LocalFSStore::addPermRoot(const Path & _storePath,
|
||||||
gcroots directory. */
|
gcroots directory. */
|
||||||
if (settings.checkRootReachability) {
|
if (settings.checkRootReachability) {
|
||||||
Roots roots = findRoots();
|
Roots roots = findRoots();
|
||||||
if (roots.find(gcRoot) == roots.end())
|
if (roots[storePath].count(gcRoot) == 0)
|
||||||
printError(
|
printError(
|
||||||
format(
|
format(
|
||||||
"warning: '%1%' is not in a directory where the garbage collector looks for roots; "
|
"warning: '%1%' is not in a directory where the garbage collector looks for roots; "
|
||||||
|
@ -266,7 +266,7 @@ void LocalStore::findRoots(const Path & path, unsigned char type, Roots & roots)
|
||||||
auto foundRoot = [&](const Path & path, const Path & target) {
|
auto foundRoot = [&](const Path & path, const Path & target) {
|
||||||
Path storePath = toStorePath(target);
|
Path storePath = toStorePath(target);
|
||||||
if (isStorePath(storePath) && isValidPath(storePath))
|
if (isStorePath(storePath) && isValidPath(storePath))
|
||||||
roots[path] = storePath;
|
roots[storePath].emplace(path);
|
||||||
else
|
else
|
||||||
printInfo(format("skipping invalid root from '%1%' to '%2%'") % path % storePath);
|
printInfo(format("skipping invalid root from '%1%' to '%2%'") % path % storePath);
|
||||||
};
|
};
|
||||||
|
@ -306,7 +306,7 @@ void LocalStore::findRoots(const Path & path, unsigned char type, Roots & roots)
|
||||||
else if (type == DT_REG) {
|
else if (type == DT_REG) {
|
||||||
Path storePath = storeDir + "/" + baseNameOf(path);
|
Path storePath = storeDir + "/" + baseNameOf(path);
|
||||||
if (isStorePath(storePath) && isValidPath(storePath))
|
if (isStorePath(storePath) && isValidPath(storePath))
|
||||||
roots[path] = storePath;
|
roots[storePath].emplace(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -346,10 +346,10 @@ Roots LocalStore::findRoots()
|
||||||
FDs fds;
|
FDs fds;
|
||||||
pid_t prev = -1;
|
pid_t prev = -1;
|
||||||
size_t n = 0;
|
size_t n = 0;
|
||||||
for (auto & root : readTempRoots(fds)) {
|
for (auto & [pid, root] : readTempRoots(fds)) {
|
||||||
if (prev != root.first) n = 0;
|
if (prev != pid) n = 0;
|
||||||
prev = root.first;
|
prev = pid;
|
||||||
roots[fmt("{temp:%d:%d}", root.first, n++)] = root.second;
|
roots[root].emplace(fmt("{temp:%d:%d}", pid, n++));
|
||||||
}
|
}
|
||||||
|
|
||||||
return roots;
|
return roots;
|
||||||
|
@ -374,8 +374,8 @@ try_again:
|
||||||
goto try_again;
|
goto try_again;
|
||||||
}
|
}
|
||||||
if (res > 0 && buf[0] == '/')
|
if (res > 0 && buf[0] == '/')
|
||||||
roots.emplace((format("{memory:%1%") % file).str(),
|
roots[std::string(static_cast<char *>(buf), res)]
|
||||||
std::string(static_cast<char *>(buf), res));
|
.emplace((format("{memory:%1%") % file).str());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -388,7 +388,7 @@ static string quoteRegexChars(const string & raw)
|
||||||
static void readFileRoots(const char * path, Roots & roots)
|
static void readFileRoots(const char * path, Roots & roots)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
roots.emplace(path, readFile(path));
|
roots[readFile(path)].emplace(path);
|
||||||
} catch (SysError & e) {
|
} catch (SysError & e) {
|
||||||
if (e.errNo != ENOENT && e.errNo != EACCES)
|
if (e.errNo != ENOENT && e.errNo != EACCES)
|
||||||
throw;
|
throw;
|
||||||
|
@ -434,19 +434,17 @@ void LocalStore::findRuntimeRoots(Roots & roots)
|
||||||
try {
|
try {
|
||||||
auto mapFile = (format("/proc/%1%/maps") % ent->d_name).str();
|
auto mapFile = (format("/proc/%1%/maps") % ent->d_name).str();
|
||||||
auto mapLines = tokenizeString<std::vector<string>>(readFile(mapFile, true), "\n");
|
auto mapLines = tokenizeString<std::vector<string>>(readFile(mapFile, true), "\n");
|
||||||
int n = 0;
|
|
||||||
for (const auto& line : mapLines) {
|
for (const auto& line : mapLines) {
|
||||||
auto match = std::smatch{};
|
auto match = std::smatch{};
|
||||||
if (std::regex_match(line, match, mapRegex))
|
if (std::regex_match(line, match, mapRegex))
|
||||||
unchecked.emplace((format("{memory:%1%:%2%}") % mapFile % n++).str(), match[1]);
|
unchecked[match[1]].emplace((format("{memory:%1%}") % mapFile).str());
|
||||||
}
|
}
|
||||||
|
|
||||||
auto envFile = (format("/proc/%1%/environ") % ent->d_name).str();
|
auto envFile = (format("/proc/%1%/environ") % ent->d_name).str();
|
||||||
auto envString = readFile(envFile, true);
|
auto envString = readFile(envFile, true);
|
||||||
auto env_end = std::sregex_iterator{};
|
auto env_end = std::sregex_iterator{};
|
||||||
n = 0;
|
|
||||||
for (auto i = std::sregex_iterator{envString.begin(), envString.end(), storePathRegex}; i != env_end; ++i)
|
for (auto i = std::sregex_iterator{envString.begin(), envString.end(), storePathRegex}; i != env_end; ++i)
|
||||||
unchecked.emplace((format("{memory:%1%:%2%}") % envFile % n++).str(), i->str());
|
unchecked[i->str()].emplace((format("{memory:%1%}") % envFile).str());
|
||||||
} catch (SysError & e) {
|
} catch (SysError & e) {
|
||||||
if (errno == ENOENT || errno == EACCES || errno == ESRCH)
|
if (errno == ENOENT || errno == EACCES || errno == ESRCH)
|
||||||
continue;
|
continue;
|
||||||
|
@ -463,11 +461,10 @@ void LocalStore::findRuntimeRoots(Roots & roots)
|
||||||
std::regex lsofRegex(R"(^n(/.*)$)");
|
std::regex lsofRegex(R"(^n(/.*)$)");
|
||||||
auto lsofLines =
|
auto lsofLines =
|
||||||
tokenizeString<std::vector<string>>(runProgram(LSOF, true, { "-n", "-w", "-F", "n" }), "\n");
|
tokenizeString<std::vector<string>>(runProgram(LSOF, true, { "-n", "-w", "-F", "n" }), "\n");
|
||||||
int n = 0;
|
|
||||||
for (const auto & line : lsofLines) {
|
for (const auto & line : lsofLines) {
|
||||||
std::smatch match;
|
std::smatch match;
|
||||||
if (std::regex_match(line, match, lsofRegex))
|
if (std::regex_match(line, match, lsofRegex))
|
||||||
unchecked.emplace((format("{memory:%1%:%2%}" % LSOF % n++).str(), match[1]);
|
unchecked[match[1]].emplace((format("{memory:%1%}" % LSOF).str());
|
||||||
}
|
}
|
||||||
} catch (ExecError & e) {
|
} catch (ExecError & e) {
|
||||||
/* lsof not installed, lsof failed */
|
/* lsof not installed, lsof failed */
|
||||||
|
@ -480,12 +477,12 @@ void LocalStore::findRuntimeRoots(Roots & roots)
|
||||||
readFileRoots("/proc/sys/kernel/poweroff_cmd", unchecked);
|
readFileRoots("/proc/sys/kernel/poweroff_cmd", unchecked);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for (auto & root : unchecked) {
|
for (auto & [target, links] : unchecked) {
|
||||||
if (isInStore(root.second)) {
|
if (isInStore(target)) {
|
||||||
Path path = toStorePath(root.second);
|
Path path = toStorePath(target);
|
||||||
if (isStorePath(path) && isValidPath(path)) {
|
if (isStorePath(path) && isValidPath(path)) {
|
||||||
debug(format("got additional root '%1%'") % path);
|
debug(format("got additional root '%1%'") % path);
|
||||||
roots.emplace(root.first, path);
|
roots[path].insert(links.begin(), links.end());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -757,7 +754,7 @@ void LocalStore::collectGarbage(const GCOptions & options, GCResults & results)
|
||||||
printError(format("finding garbage collector roots..."));
|
printError(format("finding garbage collector roots..."));
|
||||||
Roots rootMap = options.ignoreLiveness ? Roots() : findRootsNoTemp();
|
Roots rootMap = options.ignoreLiveness ? Roots() : findRootsNoTemp();
|
||||||
|
|
||||||
for (auto & i : rootMap) state.roots.insert(i.second);
|
for (auto & i : rootMap) state.roots.insert(i.first);
|
||||||
|
|
||||||
/* Read the temporary roots. This acquires read locks on all
|
/* Read the temporary roots. This acquires read locks on all
|
||||||
per-process temporary root files. So after this point no paths
|
per-process temporary root files. So after this point no paths
|
||||||
|
|
|
@ -606,7 +606,7 @@ Roots RemoteStore::findRoots()
|
||||||
while (count--) {
|
while (count--) {
|
||||||
Path link = readString(conn->from);
|
Path link = readString(conn->from);
|
||||||
Path target = readStorePath(*this, conn->from);
|
Path target = readStorePath(*this, conn->from);
|
||||||
result[link] = target;
|
result[target].emplace(link);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,7 +47,7 @@ const size_t storePathHashLen = 32; // i.e. 160 bits
|
||||||
const uint32_t exportMagic = 0x4558494e;
|
const uint32_t exportMagic = 0x4558494e;
|
||||||
|
|
||||||
|
|
||||||
typedef std::map<Path, Path> Roots;
|
typedef std::map<Path, std::set<std::string>> Roots;
|
||||||
|
|
||||||
|
|
||||||
struct GCOptions
|
struct GCOptions
|
||||||
|
|
|
@ -477,14 +477,19 @@ static void performOp(TunnelLogger * logger, ref<Store> store,
|
||||||
logger->startWork();
|
logger->startWork();
|
||||||
Roots roots = store->findRoots();
|
Roots roots = store->findRoots();
|
||||||
logger->stopWork();
|
logger->stopWork();
|
||||||
to << roots.size();
|
size_t total_length = 0;
|
||||||
|
for (auto & root : roots)
|
||||||
|
total_length += root.second.size();
|
||||||
|
to << total_length;
|
||||||
int n = 0;
|
int n = 0;
|
||||||
for (auto & i : roots) {
|
for (auto & [target, links] : roots) {
|
||||||
// Obfuscate 'memory' roots as they exposes information about other users,
|
for (auto & link : links) {
|
||||||
if (i.first.rfind("{memory:", 0) == 0) {
|
// Obfuscate 'memory' roots as they expose information about other users,
|
||||||
to << fmt("{memory:%d}", n++) << i.second;
|
if (link.rfind("{memory:", 0) == 0) {
|
||||||
} else {
|
to << fmt("{memory:%d}", n++) << target;
|
||||||
to << i.first << i.second;
|
} else {
|
||||||
|
to << link << target;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -428,9 +428,10 @@ static void opQuery(Strings opFlags, Strings opArgs)
|
||||||
referrers, true, settings.gcKeepOutputs, settings.gcKeepDerivations);
|
referrers, true, settings.gcKeepOutputs, settings.gcKeepDerivations);
|
||||||
}
|
}
|
||||||
Roots roots = store->findRoots();
|
Roots roots = store->findRoots();
|
||||||
for (auto & i : roots)
|
for (auto & [path, roots] : roots)
|
||||||
if (referrers.find(i.second) != referrers.end())
|
if (referrers.find(path) != referrers.end())
|
||||||
cout << format("%1%\n") % i.first;
|
for (auto & root : roots)
|
||||||
|
cout << format("%1% -> %2%\n") % root % path;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -591,8 +592,9 @@ static void opGC(Strings opFlags, Strings opArgs)
|
||||||
|
|
||||||
if (printRoots) {
|
if (printRoots) {
|
||||||
Roots roots = store->findRoots();
|
Roots roots = store->findRoots();
|
||||||
for (auto & i : roots)
|
for (auto & [path, roots] : roots)
|
||||||
cout << i.first << " -> " << i.second << std::endl;
|
for (auto & root : roots)
|
||||||
|
cout << root << " -> " << path << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
else {
|
||||||
|
|
Loading…
Reference in a new issue