Commit b0868e34 authored by Kenton Varda's avatar Kenton Varda

Replace some STL maps with KJ maps in Cap'n Proto implementation.

So far this is only a small subset of all the STL uses.
parent e443282b
...@@ -98,11 +98,10 @@ SegmentReader* ReaderArena::tryGetSegment(SegmentId id) { ...@@ -98,11 +98,10 @@ SegmentReader* ReaderArena::tryGetSegment(SegmentId id) {
SegmentMap* segments = nullptr; SegmentMap* segments = nullptr;
KJ_IF_MAYBE(s, *lock) { KJ_IF_MAYBE(s, *lock) {
auto iter = s->get()->find(id.value); KJ_IF_MAYBE(segment, s->find(id.value)) {
if (iter != s->get()->end()) { return *segment;
return iter->second;
} }
segments = *s; segments = s;
} }
kj::ArrayPtr<const word> newSegment = message->getSegment(id.value); kj::ArrayPtr<const word> newSegment = message->getSegment(id.value);
...@@ -114,15 +113,13 @@ SegmentReader* ReaderArena::tryGetSegment(SegmentId id) { ...@@ -114,15 +113,13 @@ SegmentReader* ReaderArena::tryGetSegment(SegmentId id) {
if (*lock == nullptr) { if (*lock == nullptr) {
// OK, the segment exists, so allocate the map. // OK, the segment exists, so allocate the map.
auto s = kj::heap<SegmentMap>(); segments = &lock->emplace();
segments = s;
*lock = kj::mv(s);
} }
auto segment = kj::heap<SegmentReader>( auto segment = kj::heap<SegmentReader>(
this, id, newSegment.begin(), newSegmentSize, &readLimiter); this, id, newSegment.begin(), newSegmentSize, &readLimiter);
SegmentReader* result = segment; SegmentReader* result = segment;
segments->insert(std::make_pair(id.value, mv(segment))); segments->insert(id.value, kj::mv(segment));
return result; return result;
} }
......
...@@ -37,7 +37,7 @@ ...@@ -37,7 +37,7 @@
#include "common.h" #include "common.h"
#include "message.h" #include "message.h"
#include "layout.h" #include "layout.h"
#include <unordered_map> #include <kj/map.h>
#if !CAPNP_LITE #if !CAPNP_LITE
#include "capability.h" #include "capability.h"
...@@ -241,8 +241,8 @@ private: ...@@ -241,8 +241,8 @@ private:
// Optimize for single-segment messages so that small messages are handled quickly. // Optimize for single-segment messages so that small messages are handled quickly.
SegmentReader segment0; SegmentReader segment0;
typedef std::unordered_map<uint, kj::Own<SegmentReader>> SegmentMap; typedef kj::HashMap<uint, kj::Own<SegmentReader>> SegmentMap;
kj::MutexGuarded<kj::Maybe<kj::Own<SegmentMap>>> moreSegments; kj::MutexGuarded<kj::Maybe<SegmentMap>> moreSegments;
// We need to mutex-guard the segment map because we lazily initialize segments when they are // We need to mutex-guard the segment map because we lazily initialize segments when they are
// first requested, but a Reader is allowed to be used concurrently in multiple threads. Luckily // first requested, but a Reader is allowed to be used concurrently in multiple threads. Luckily
// this only applies to large messages. // this only applies to large messages.
......
...@@ -46,6 +46,7 @@ ...@@ -46,6 +46,7 @@
#include <capnp/compat/json.h> #include <capnp/compat/json.h>
#include <errno.h> #include <errno.h>
#include <stdlib.h> #include <stdlib.h>
#include <kj/map.h>
#if _WIN32 #if _WIN32
#include <process.h> #include <process.h>
...@@ -1767,17 +1768,16 @@ private: ...@@ -1767,17 +1768,16 @@ private:
// require to function. // require to function.
struct SourceDirectory { struct SourceDirectory {
kj::Path path;
kj::Own<const kj::ReadableDirectory> dir; kj::Own<const kj::ReadableDirectory> dir;
bool isSourcePrefix; bool isSourcePrefix;
}; };
std::map<kj::PathPtr, SourceDirectory> sourceDirectories; kj::HashMap<kj::Path, SourceDirectory> sourceDirectories;
// For each import path and source prefix, tracks the directory object we opened for it. // For each import path and source prefix, tracks the directory object we opened for it.
// //
// Use via getSourceDirectory(). // Use via getSourceDirectory().
std::map<const kj::ReadableDirectory*, kj::String> dirPrefixes; kj::HashMap<const kj::ReadableDirectory*, kj::String> dirPrefixes;
// For each open directory object, maps to a path prefix to add when displaying this path in // For each open directory object, maps to a path prefix to add when displaying this path in
// error messages. This keeps track of the original directory name as given by the user, before // error messages. This keeps track of the original directory name as given by the user, before
// canonicalization. // canonicalization.
...@@ -1827,10 +1827,9 @@ private: ...@@ -1827,10 +1827,9 @@ private:
if (path.size() == 0) return disk->getRoot(); if (path.size() == 0) return disk->getRoot();
auto iter = sourceDirectories.find(path); KJ_IF_MAYBE(sdir, sourceDirectories.find(path)) {
if (iter != sourceDirectories.end()) { sdir->isSourcePrefix = sdir->isSourcePrefix || isSourcePrefix;
iter->second.isSourcePrefix = iter->second.isSourcePrefix || isSourcePrefix; return *sdir->dir;
return *iter->second.dir;
} }
if (path == cwd) { if (path == cwd) {
...@@ -1843,26 +1842,22 @@ private: ...@@ -1843,26 +1842,22 @@ private:
// getDisplayName(). // getDisplayName().
auto& result = disk->getCurrent(); auto& result = disk->getCurrent();
if (isSourcePrefix) { if (isSourcePrefix) {
kj::PathPtr key = path;
kj::Own<const kj::ReadableDirectory> fakeOwn(&result, kj::NullDisposer::instance); kj::Own<const kj::ReadableDirectory> fakeOwn(&result, kj::NullDisposer::instance);
KJ_ASSERT(sourceDirectories.insert(std::make_pair(key, sourceDirectories.insert(kj::mv(path), { kj::mv(fakeOwn), isSourcePrefix });
SourceDirectory { kj::mv(path), kj::mv(fakeOwn), isSourcePrefix })).second);
} }
return result; return result;
} }
KJ_IF_MAYBE(dir, disk->getRoot().tryOpenSubdir(path)) { KJ_IF_MAYBE(dir, disk->getRoot().tryOpenSubdir(path)) {
auto& result = *dir->get(); auto& result = *dir->get();
kj::PathPtr key = path; sourceDirectories.insert(kj::mv(path), { kj::mv(*dir), isSourcePrefix });
KJ_ASSERT(sourceDirectories.insert(std::make_pair(key,
SourceDirectory { kj::mv(path), kj::mv(*dir), isSourcePrefix })).second);
#if _WIN32 #if _WIN32
kj::String prefix = pathStr.endsWith("/") || pathStr.endsWith("\\") kj::String prefix = pathStr.endsWith("/") || pathStr.endsWith("\\")
? kj::str(pathStr) : kj::str(pathStr, '\\'); ? kj::str(pathStr) : kj::str(pathStr, '\\');
#else #else
kj::String prefix = pathStr.endsWith("/") ? kj::str(pathStr) : kj::str(pathStr, '/'); kj::String prefix = pathStr.endsWith("/") ? kj::str(pathStr) : kj::str(pathStr, '/');
#endif #endif
KJ_ASSERT(dirPrefixes.insert(std::make_pair(&result, kj::mv(prefix))).second); dirPrefixes.insert(&result, kj::mv(prefix));
return result; return result;
} else { } else {
return nullptr; return nullptr;
...@@ -1883,9 +1878,8 @@ private: ...@@ -1883,9 +1878,8 @@ private:
auto prefix = path.slice(0, i); auto prefix = path.slice(0, i);
auto remainder = path.slice(i, path.size()); auto remainder = path.slice(i, path.size());
auto iter = sourceDirectories.find(prefix); KJ_IF_MAYBE(sdir, sourceDirectories.find(prefix)) {
if (iter != sourceDirectories.end() && iter->second.isSourcePrefix) { return { *sdir->dir, remainder.clone() };
return { *iter->second.dir, remainder.clone() };
} }
} }
...@@ -1914,9 +1908,8 @@ private: ...@@ -1914,9 +1908,8 @@ private:
} }
kj::String getDisplayName(const kj::ReadableDirectory& dir, kj::PathPtr path) { kj::String getDisplayName(const kj::ReadableDirectory& dir, kj::PathPtr path) {
auto iter = dirPrefixes.find(&dir); KJ_IF_MAYBE(prefix, dirPrefixes.find(&dir)) {
if (iter != dirPrefixes.end()) { return kj::str(*prefix, path.toNativeString());
return kj::str(iter->second, path.toNativeString());
} else if (&dir == &disk->getRoot()) { } else if (&dir == &disk->getRoot()) {
return path.toNativeString(true); return path.toNativeString(true);
} else if (&dir == &disk->getCurrent()) { } else if (&dir == &disk->getCurrent()) {
......
...@@ -126,9 +126,9 @@ private: ...@@ -126,9 +126,9 @@ private:
// Space in which we can construct a ReaderArena. We don't use ReaderArena directly here // Space in which we can construct a ReaderArena. We don't use ReaderArena directly here
// because we don't want clients to have to #include arena.h, which itself includes a bunch of // because we don't want clients to have to #include arena.h, which itself includes a bunch of
// big STL headers. We don't use a pointer to a ReaderArena because that would require an // other headers. We don't use a pointer to a ReaderArena because that would require an
// extra malloc on every message which could be expensive when processing small messages. // extra malloc on every message which could be expensive when processing small messages.
void* arenaSpace[15 + sizeof(kj::MutexGuarded<void*>) / sizeof(void*)]; void* arenaSpace[17 + sizeof(kj::MutexGuarded<void*>) / sizeof(void*)];
bool allocatedArena; bool allocatedArena;
_::ReaderArena* arena() { return reinterpret_cast<_::ReaderArena*>(arenaSpace); } _::ReaderArena* arena() { return reinterpret_cast<_::ReaderArena*>(arenaSpace); }
......
This diff is collapsed.
...@@ -26,6 +26,7 @@ ...@@ -26,6 +26,7 @@
#include <inttypes.h> #include <inttypes.h>
#include "time.h" #include "time.h"
#include "function.h" #include "function.h"
#include "hash.h"
namespace kj { namespace kj {
...@@ -156,6 +157,9 @@ public: ...@@ -156,6 +157,9 @@ public:
bool operator>=(PathPtr other) const; bool operator>=(PathPtr other) const;
// Compare path components lexically. // Compare path components lexically.
uint hashCode() const;
// Can use in HashMap.
bool startsWith(PathPtr prefix) const; bool startsWith(PathPtr prefix) const;
bool endsWith(PathPtr suffix) const; bool endsWith(PathPtr suffix) const;
// Compare prefix / suffix. // Compare prefix / suffix.
...@@ -264,6 +268,7 @@ public: ...@@ -264,6 +268,7 @@ public:
bool operator> (PathPtr other) const; bool operator> (PathPtr other) const;
bool operator<=(PathPtr other) const; bool operator<=(PathPtr other) const;
bool operator>=(PathPtr other) const; bool operator>=(PathPtr other) const;
uint hashCode() const;
bool startsWith(PathPtr prefix) const; bool startsWith(PathPtr prefix) const;
bool endsWith(PathPtr suffix) const; bool endsWith(PathPtr suffix) const;
Path evalWin32(StringPtr pathText) const; Path evalWin32(StringPtr pathText) const;
...@@ -991,6 +996,7 @@ inline bool Path::operator< (PathPtr other) const { return PathPtr(*this) < oth ...@@ -991,6 +996,7 @@ inline bool Path::operator< (PathPtr other) const { return PathPtr(*this) < oth
inline bool Path::operator> (PathPtr other) const { return PathPtr(*this) > other; } inline bool Path::operator> (PathPtr other) const { return PathPtr(*this) > other; }
inline bool Path::operator<=(PathPtr other) const { return PathPtr(*this) <= other; } inline bool Path::operator<=(PathPtr other) const { return PathPtr(*this) <= other; }
inline bool Path::operator>=(PathPtr other) const { return PathPtr(*this) >= other; } inline bool Path::operator>=(PathPtr other) const { return PathPtr(*this) >= other; }
inline uint Path::hashCode() const { return kj::hashCode(parts); }
inline bool Path::startsWith(PathPtr prefix) const { return PathPtr(*this).startsWith(prefix); } inline bool Path::startsWith(PathPtr prefix) const { return PathPtr(*this).startsWith(prefix); }
inline bool Path::endsWith (PathPtr suffix) const { return PathPtr(*this).endsWith (suffix); } inline bool Path::endsWith (PathPtr suffix) const { return PathPtr(*this).endsWith (suffix); }
inline String Path::toString(bool absolute) const { return PathPtr(*this).toString(absolute); } inline String Path::toString(bool absolute) const { return PathPtr(*this).toString(absolute); }
...@@ -1020,6 +1026,7 @@ inline bool PathPtr::operator!=(PathPtr other) const { return !(*this == other); ...@@ -1020,6 +1026,7 @@ inline bool PathPtr::operator!=(PathPtr other) const { return !(*this == other);
inline bool PathPtr::operator> (PathPtr other) const { return other < *this; } inline bool PathPtr::operator> (PathPtr other) const { return other < *this; }
inline bool PathPtr::operator<=(PathPtr other) const { return !(other < *this); } inline bool PathPtr::operator<=(PathPtr other) const { return !(other < *this); }
inline bool PathPtr::operator>=(PathPtr other) const { return !(*this < other); } inline bool PathPtr::operator>=(PathPtr other) const { return !(*this < other); }
inline uint PathPtr::hashCode() const { return kj::hashCode(parts); }
inline String PathPtr::toWin32String(bool absolute) const { inline String PathPtr::toWin32String(bool absolute) const {
return toWin32StringImpl(absolute, false); return toWin32StringImpl(absolute, false);
} }
......
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