Commit 44cc0500 authored by Kenton Varda's avatar Kenton Varda

Fix SchemaParser for threadsafe filesystem API.

parent deb65e9a
...@@ -43,14 +43,14 @@ public: ...@@ -43,14 +43,14 @@ public:
->writeAll(content); ->writeAll(content);
} }
kj::Directory& getRoot() override { return *root; } const kj::Directory& getRoot() const override { return *root; }
kj::Directory& getCurrent() override { return *current; } const kj::Directory& getCurrent() const override { return *current; }
kj::PathPtr getCurrentPath() override { return cwd; } kj::PathPtr getCurrentPath() const override { return cwd; }
private: private:
kj::Own<kj::Directory> root = kj::newInMemoryDirectory(kj::nullClock()); kj::Own<const kj::Directory> root = kj::newInMemoryDirectory(kj::nullClock());
kj::Path cwd = kj::Path({}).evalNative(ABS("path/to/current/dir")); kj::Path cwd = kj::Path({}).evalNative(ABS("path/to/current/dir"));
kj::Own<kj::Directory> current = root->openSubdir(cwd, kj::Own<const kj::Directory> current = root->openSubdir(cwd,
kj::WriteMode::CREATE | kj::WriteMode::CREATE_PARENT); kj::WriteMode::CREATE | kj::WriteMode::CREATE_PARENT);
}; };
......
...@@ -171,11 +171,11 @@ struct SchemaParser::DiskFileCompat { ...@@ -171,11 +171,11 @@ struct SchemaParser::DiskFileCompat {
struct ImportDir { struct ImportDir {
kj::String pathStr; kj::String pathStr;
kj::Path path; kj::Path path;
kj::Own<kj::ReadableDirectory> dir; kj::Own<const kj::ReadableDirectory> dir;
}; };
std::map<kj::StringPtr, ImportDir> cachedImportDirs; std::map<kj::StringPtr, ImportDir> cachedImportDirs;
std::map<std::pair<const kj::StringPtr*, size_t>, kj::Array<kj::ReadableDirectory*>> std::map<std::pair<const kj::StringPtr*, size_t>, kj::Array<const kj::ReadableDirectory*>>
cachedImportPaths; cachedImportPaths;
DiskFileCompat(): ownFs(kj::newDiskFilesystem()), fs(*ownFs) {} DiskFileCompat(): ownFs(kj::newDiskFilesystem()), fs(*ownFs) {}
...@@ -195,8 +195,8 @@ SchemaParser::SchemaParser(): impl(kj::heap<Impl>()) {} ...@@ -195,8 +195,8 @@ SchemaParser::SchemaParser(): impl(kj::heap<Impl>()) {}
SchemaParser::~SchemaParser() noexcept(false) {} SchemaParser::~SchemaParser() noexcept(false) {}
ParsedSchema SchemaParser::parseFromDirectory( ParsedSchema SchemaParser::parseFromDirectory(
kj::ReadableDirectory& baseDir, kj::Path path, const kj::ReadableDirectory& baseDir, kj::Path path,
kj::ArrayPtr<kj::ReadableDirectory* const> importPath) const { kj::ArrayPtr<const kj::ReadableDirectory* const> importPath) const {
return parseFile(SchemaFile::newFromDirectory(baseDir, kj::mv(path), importPath)); return parseFile(SchemaFile::newFromDirectory(baseDir, kj::mv(path), importPath));
} }
...@@ -214,24 +214,24 @@ ParsedSchema SchemaParser::parseDiskFile( ...@@ -214,24 +214,24 @@ ParsedSchema SchemaParser::parseDiskFile(
auto& root = compat->fs.getRoot(); auto& root = compat->fs.getRoot();
auto cwd = compat->fs.getCurrentPath(); auto cwd = compat->fs.getCurrentPath();
kj::ReadableDirectory* baseDir = &root; const kj::ReadableDirectory* baseDir = &root;
kj::Path path = cwd.evalNative(diskPath); kj::Path path = cwd.evalNative(diskPath);
kj::ArrayPtr<kj::ReadableDirectory* const> translatedImportPath = nullptr; kj::ArrayPtr<const kj::ReadableDirectory* const> translatedImportPath = nullptr;
if (importPath.size() > 0) { if (importPath.size() > 0) {
auto importPathKey = std::make_pair(importPath.begin(), importPath.size()); auto importPathKey = std::make_pair(importPath.begin(), importPath.size());
auto& slot = compat->cachedImportPaths[importPathKey]; auto& slot = compat->cachedImportPaths[importPathKey];
if (slot == nullptr) { if (slot == nullptr) {
slot = KJ_MAP(path, importPath) -> kj::ReadableDirectory* { slot = KJ_MAP(path, importPath) -> const kj::ReadableDirectory* {
auto iter = compat->cachedImportDirs.find(path); auto iter = compat->cachedImportDirs.find(path);
if (iter != compat->cachedImportDirs.end()) { if (iter != compat->cachedImportDirs.end()) {
return iter->second.dir; return iter->second.dir;
} }
auto parsed = cwd.evalNative(path); auto parsed = cwd.evalNative(path);
kj::Own<kj::ReadableDirectory> dir; kj::Own<const kj::ReadableDirectory> dir;
KJ_IF_MAYBE(d, root.tryOpenSubdir(parsed)) { KJ_IF_MAYBE(d, root.tryOpenSubdir(parsed)) {
dir = kj::mv(*d); dir = kj::mv(*d);
} else { } else {
...@@ -239,7 +239,7 @@ ParsedSchema SchemaParser::parseDiskFile( ...@@ -239,7 +239,7 @@ ParsedSchema SchemaParser::parseDiskFile(
dir = kj::newInMemoryDirectory(kj::nullClock()); dir = kj::newInMemoryDirectory(kj::nullClock());
} }
kj::ReadableDirectory* result = dir; const kj::ReadableDirectory* result = dir;
kj::StringPtr pathRef = path; kj::StringPtr pathRef = path;
KJ_ASSERT(compat->cachedImportDirs.insert(std::make_pair(pathRef, KJ_ASSERT(compat->cachedImportDirs.insert(std::make_pair(pathRef,
...@@ -336,9 +336,9 @@ schema::Node::SourceInfo::Reader ParsedSchema::getSourceInfo() const { ...@@ -336,9 +336,9 @@ schema::Node::SourceInfo::Reader ParsedSchema::getSourceInfo() const {
class SchemaFile::DiskSchemaFile final: public SchemaFile { class SchemaFile::DiskSchemaFile final: public SchemaFile {
public: public:
DiskSchemaFile(kj::ReadableDirectory& baseDir, kj::Path pathParam, DiskSchemaFile(const kj::ReadableDirectory& baseDir, kj::Path pathParam,
kj::ArrayPtr<kj::ReadableDirectory* const> importPath, kj::ArrayPtr<const kj::ReadableDirectory* const> importPath,
kj::Own<kj::ReadableFile> file, kj::Own<const kj::ReadableFile> file,
kj::Maybe<kj::String> displayNameOverride) kj::Maybe<kj::String> displayNameOverride)
: baseDir(baseDir), path(kj::mv(pathParam)), importPath(importPath), file(kj::mv(file)) { : baseDir(baseDir), path(kj::mv(pathParam)), importPath(importPath), file(kj::mv(file)) {
KJ_IF_MAYBE(dn, displayNameOverride) { KJ_IF_MAYBE(dn, displayNameOverride) {
...@@ -355,12 +355,7 @@ public: ...@@ -355,12 +355,7 @@ public:
} }
kj::Array<const char> readContent() const override { kj::Array<const char> readContent() const override {
auto lock = file.lockExclusive(); return file->mmap(0, file->stat().size).releaseAsChars();
// TODO(soon): Should we say that KJ files must be thread-safe and mark all the methods const?
// The disk-based implementations are already thread-safe. The in-memory implementations
// are not thread-safe, in fact they are thread-hostile currently due to the non-threadsafe
// refcounting.
return lock->get()->mmap(0, lock->get()->stat().size).releaseAsChars();
} }
kj::Maybe<kj::Own<SchemaFile>> import(kj::StringPtr target) const override { kj::Maybe<kj::Own<SchemaFile>> import(kj::StringPtr target) const override {
...@@ -422,17 +417,17 @@ public: ...@@ -422,17 +417,17 @@ public:
} }
private: private:
kj::ReadableDirectory& baseDir; const kj::ReadableDirectory& baseDir;
kj::Path path; kj::Path path;
kj::ArrayPtr<kj::ReadableDirectory* const> importPath; kj::ArrayPtr<const kj::ReadableDirectory* const> importPath;
kj::MutexGuarded<kj::Own<kj::ReadableFile>> file; kj::Own<const kj::ReadableFile> file;
kj::String displayName; kj::String displayName;
bool displayNameOverridden; bool displayNameOverridden;
}; };
kj::Own<SchemaFile> SchemaFile::newFromDirectory( kj::Own<SchemaFile> SchemaFile::newFromDirectory(
kj::ReadableDirectory& baseDir, kj::Path path, const kj::ReadableDirectory& baseDir, kj::Path path,
kj::ArrayPtr<kj::ReadableDirectory* const> importPath, kj::ArrayPtr<const kj::ReadableDirectory* const> importPath,
kj::Maybe<kj::String> displayNameOverride) { kj::Maybe<kj::String> displayNameOverride) {
return kj::heap<DiskSchemaFile>(baseDir, kj::mv(path), importPath, baseDir.openFile(path), return kj::heap<DiskSchemaFile>(baseDir, kj::mv(path), importPath, baseDir.openFile(path),
kj::mv(displayNameOverride)); kj::mv(displayNameOverride));
......
...@@ -44,8 +44,9 @@ public: ...@@ -44,8 +44,9 @@ public:
SchemaParser(); SchemaParser();
~SchemaParser() noexcept(false); ~SchemaParser() noexcept(false);
ParsedSchema parseFromDirectory(kj::ReadableDirectory& baseDir, kj::Path path, ParsedSchema parseFromDirectory(
kj::ArrayPtr<kj::ReadableDirectory* const> importPath) const; const kj::ReadableDirectory& baseDir, kj::Path path,
kj::ArrayPtr<const kj::ReadableDirectory* const> importPath) const;
// Parse a file from the KJ filesystem API. Throws an exception if the file dosen't exist. // Parse a file from the KJ filesystem API. Throws an exception if the file dosen't exist.
// //
// `baseDir` and `path` are used together to resolve relative imports. `path` is the source // `baseDir` and `path` are used together to resolve relative imports. `path` is the source
...@@ -197,8 +198,8 @@ public: ...@@ -197,8 +198,8 @@ public:
// kj::ReadableDirectory, or using kj::newInMemoryDirectory(). // kj::ReadableDirectory, or using kj::newInMemoryDirectory().
static kj::Own<SchemaFile> newFromDirectory( static kj::Own<SchemaFile> newFromDirectory(
kj::ReadableDirectory& baseDir, kj::Path path, const kj::ReadableDirectory& baseDir, kj::Path path,
kj::ArrayPtr<kj::ReadableDirectory* const> importPath, kj::ArrayPtr<const kj::ReadableDirectory* const> importPath,
kj::Maybe<kj::String> displayNameOverride = nullptr); kj::Maybe<kj::String> displayNameOverride = nullptr);
// Construct a SchemaFile representing a file in a kj::ReadableDirectory. This is used to // Construct a SchemaFile representing a file in a kj::ReadableDirectory. This is used to
// implement SchemaParser::parseFromDirectory(); see there for details. // implement SchemaParser::parseFromDirectory(); see there for details.
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment