aboutsummaryrefslogtreecommitdiffstats
path: root/MemTrie.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'MemTrie.cpp')
-rw-r--r--MemTrie.cpp479
1 files changed, 479 insertions, 0 deletions
diff --git a/MemTrie.cpp b/MemTrie.cpp
new file mode 100644
index 00000000..0a7d0ebe
--- /dev/null
+++ b/MemTrie.cpp
@@ -0,0 +1,479 @@
+/*
+ This file is part of cpp-ethereum.
+
+ cpp-ethereum is free software: you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation, either version 3 of the License, or
+ (at your option) any later version.
+
+ cpp-ethereum is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
+*/
+/** @file MemTrie.cpp
+ * @author Gav Wood <i@gavwood.com>
+ * @date 2014
+ */
+
+#include "MemTrie.h"
+
+#include <CommonEth.h>
+#include <TrieCommon.h>
+using namespace std;
+using namespace eth;
+
+namespace eth
+{
+
+#define ENABLE_DEBUG_PRINT 0
+
+/*/
+#define APPEND_CHILD appendData
+/*/
+#define APPEND_CHILD appendRaw
+/**/
+
+class MemTrieNode
+{
+public:
+ MemTrieNode() {}
+ virtual ~MemTrieNode() {}
+
+ virtual std::string const& at(bytesConstRef _key) const = 0;
+ virtual MemTrieNode* insert(bytesConstRef _key, std::string const& _value) = 0;
+ virtual MemTrieNode* remove(bytesConstRef _key) = 0;
+ void putRLP(RLPStream& _parentStream) const;
+
+#if ENABLE_DEBUG_PRINT
+ void debugPrint(std::string const& _indent = "") const { std::cerr << std::hex << hash256() << ":" << std::dec << std::endl; debugPrintBody(_indent); }
+#endif
+
+ /// 256-bit hash of the node - this is a SHA-3/256 hash of the RLP of the node.
+ h256 hash256() const { RLPStream s; makeRLP(s); return eth::sha3(s.out()); }
+ bytes rlp() const { RLPStream s; makeRLP(s); return s.out(); }
+ void mark() { m_hash256 = h256(); }
+
+protected:
+ virtual void makeRLP(RLPStream& _intoStream) const = 0;
+
+#if ENABLE_DEBUG_PRINT
+ virtual void debugPrintBody(std::string const& _indent = "") const = 0;
+#endif
+
+ static MemTrieNode* newBranch(bytesConstRef _k1, std::string const& _v1, bytesConstRef _k2, std::string const& _v2);
+
+private:
+ mutable h256 m_hash256;
+};
+
+static const std::string c_nullString;
+
+class TrieExtNode: public MemTrieNode
+{
+public:
+ TrieExtNode(bytesConstRef _bytes): m_ext(_bytes.begin(), _bytes.end()) {}
+
+ bytes m_ext;
+};
+
+class TrieBranchNode: public MemTrieNode
+{
+public:
+ TrieBranchNode(std::string const& _value): m_value(_value)
+ {
+ memset(m_nodes.data(), 0, sizeof(MemTrieNode*) * 16);
+ }
+
+ TrieBranchNode(byte _i1, MemTrieNode* _n1, std::string const& _value = std::string()): m_value(_value)
+ {
+ memset(m_nodes.data(), 0, sizeof(MemTrieNode*) * 16);
+ m_nodes[_i1] = _n1;
+ }
+
+ TrieBranchNode(byte _i1, MemTrieNode* _n1, byte _i2, MemTrieNode* _n2)
+ {
+ memset(m_nodes.data(), 0, sizeof(MemTrieNode*) * 16);
+ m_nodes[_i1] = _n1;
+ m_nodes[_i2] = _n2;
+ }
+
+ virtual ~TrieBranchNode()
+ {
+ for (auto i: m_nodes)
+ delete i;
+ }
+
+#if ENABLE_DEBUG_PRINT
+ virtual void debugPrintBody(std::string const& _indent) const
+ {
+
+ if (m_value.size())
+ std::cerr << _indent << "@: " << m_value << std::endl;
+ for (auto i = 0; i < 16; ++i)
+ if (m_nodes[i])
+ {
+ std::cerr << _indent << std::hex << i << ": " << std::dec;
+ m_nodes[i]->debugPrint(_indent + " ");
+ }
+ }
+#endif
+
+ virtual std::string const& at(bytesConstRef _key) const override;
+ virtual MemTrieNode* insert(bytesConstRef _key, std::string const& _value) override;
+ virtual MemTrieNode* remove(bytesConstRef _key) override;
+ virtual void makeRLP(RLPStream& _parentStream) const override;
+
+private:
+ /// @returns (byte)-1 when no active branches, 16 when multiple active and the index of the active branch otherwise.
+ byte activeBranch() const;
+
+ MemTrieNode* rejig();
+
+ std::array<MemTrieNode*, 16> m_nodes;
+ std::string m_value;
+};
+
+class TrieLeafNode: public TrieExtNode
+{
+public:
+ TrieLeafNode(bytesConstRef _key, std::string const& _value): TrieExtNode(_key), m_value(_value) {}
+
+#if ENABLE_DEBUG_PRINT
+ virtual void debugPrintBody(std::string const& _indent) const
+ {
+ assert(m_value.size());
+ std::cerr << _indent;
+ if (m_ext.size())
+ std::cerr << toHex(m_ext, 1) << ": ";
+ else
+ std::cerr << "@: ";
+ std::cerr << m_value << std::endl;
+ }
+#endif
+
+ virtual std::string const& at(bytesConstRef _key) const override { return contains(_key) ? m_value : c_nullString; }
+ virtual MemTrieNode* insert(bytesConstRef _key, std::string const& _value) override;
+ virtual MemTrieNode* remove(bytesConstRef _key) override;
+ virtual void makeRLP(RLPStream& _parentStream) const override;
+
+private:
+ bool contains(bytesConstRef _key) const { return _key.size() == m_ext.size() && !memcmp(_key.data(), m_ext.data(), _key.size()); }
+
+ std::string m_value;
+};
+
+class TrieInfixNode: public TrieExtNode
+{
+public:
+ TrieInfixNode(bytesConstRef _key, MemTrieNode* _next): TrieExtNode(_key), m_next(_next) {}
+ virtual ~TrieInfixNode() { delete m_next; }
+
+#if ENABLE_DEBUG_PRINT
+ virtual void debugPrintBody(std::string const& _indent) const
+ {
+ std::cerr << _indent << toHex(m_ext, 1) << ": ";
+ m_next->debugPrint(_indent + " ");
+ }
+#endif
+
+ virtual std::string const& at(bytesConstRef _key) const override { assert(m_next); return contains(_key) ? m_next->at(_key.cropped(m_ext.size())) : c_nullString; }
+ virtual MemTrieNode* insert(bytesConstRef _key, std::string const& _value) override;
+ virtual MemTrieNode* remove(bytesConstRef _key) override;
+ virtual void makeRLP(RLPStream& _parentStream) const override;
+
+private:
+ bool contains(bytesConstRef _key) const { return _key.size() >= m_ext.size() && !memcmp(_key.data(), m_ext.data(), m_ext.size()); }
+
+ MemTrieNode* m_next;
+};
+
+void MemTrieNode::putRLP(RLPStream& _parentStream) const
+{
+ RLPStream s;
+ makeRLP(s);
+ if (s.out().size() < 32)
+ _parentStream.APPEND_CHILD(s.out());
+ else
+ _parentStream << eth::sha3(s.out());
+}
+
+void TrieBranchNode::makeRLP(RLPStream& _intoStream) const
+{
+ _intoStream.appendList(17);
+ for (auto i: m_nodes)
+ if (i)
+ i->putRLP(_intoStream);
+ else
+ _intoStream << "";
+ _intoStream << m_value;
+}
+
+void TrieLeafNode::makeRLP(RLPStream& _intoStream) const
+{
+ _intoStream.appendList(2) << hexPrefixEncode(m_ext, true) << m_value;
+}
+
+void TrieInfixNode::makeRLP(RLPStream& _intoStream) const
+{
+ assert(m_next);
+ _intoStream.appendList(2);
+ _intoStream << hexPrefixEncode(m_ext, false);
+ m_next->putRLP(_intoStream);
+}
+
+MemTrieNode* MemTrieNode::newBranch(bytesConstRef _k1, std::string const& _v1, bytesConstRef _k2, std::string const& _v2)
+{
+ uint prefix = commonPrefix(_k1, _k2);
+
+ MemTrieNode* ret;
+ if (_k1.size() == prefix)
+ ret = new TrieBranchNode(_k2[prefix], new TrieLeafNode(_k2.cropped(prefix + 1), _v2), _v1);
+ else if (_k2.size() == prefix)
+ ret = new TrieBranchNode(_k1[prefix], new TrieLeafNode(_k1.cropped(prefix + 1), _v1), _v2);
+ else // both continue after split
+ ret = new TrieBranchNode(_k1[prefix], new TrieLeafNode(_k1.cropped(prefix + 1), _v1), _k2[prefix], new TrieLeafNode(_k2.cropped(prefix + 1), _v2));
+
+ if (prefix)
+ // have shared prefix - split.
+ ret = new TrieInfixNode(_k1.cropped(0, prefix), ret);
+
+ return ret;
+}
+
+std::string const& TrieBranchNode::at(bytesConstRef _key) const
+{
+ if (_key.empty())
+ return m_value;
+ else if (m_nodes[_key[0]] != nullptr)
+ return m_nodes[_key[0]]->at(_key.cropped(1));
+ return c_nullString;
+}
+
+MemTrieNode* TrieBranchNode::insert(bytesConstRef _key, std::string const& _value)
+{
+ assert(_value.size());
+ mark();
+ if (_key.empty())
+ m_value = _value;
+ else
+ if (!m_nodes[_key[0]])
+ m_nodes[_key[0]] = new TrieLeafNode(_key.cropped(1), _value);
+ else
+ m_nodes[_key[0]] = m_nodes[_key[0]]->insert(_key.cropped(1), _value);
+ return this;
+}
+
+MemTrieNode* TrieBranchNode::remove(bytesConstRef _key)
+{
+ if (_key.empty())
+ if (m_value.size())
+ {
+ m_value.clear();
+ return rejig();
+ }
+ else {}
+ else if (m_nodes[_key[0]] != nullptr)
+ {
+ m_nodes[_key[0]] = m_nodes[_key[0]]->remove(_key.cropped(1));
+ return rejig();
+ }
+ return this;
+}
+
+MemTrieNode* TrieBranchNode::rejig()
+{
+ mark();
+ byte n = activeBranch();
+
+ if (n == (byte)-1 && m_value.size())
+ {
+ // switch to leaf
+ auto r = new TrieLeafNode(bytesConstRef(), m_value);
+ delete this;
+ return r;
+ }
+ else if (n < 16 && m_value.empty())
+ {
+ // only branching to n...
+ if (auto b = dynamic_cast<TrieBranchNode*>(m_nodes[n]))
+ {
+ // switch to infix
+ m_nodes[n] = nullptr;
+ delete this;
+ return new TrieInfixNode(bytesConstRef(&n, 1), b);
+ }
+ else
+ {
+ auto x = dynamic_cast<TrieExtNode*>(m_nodes[n]);
+ assert(x);
+ // include in child
+ pushFront(x->m_ext, n);
+ m_nodes[n] = nullptr;
+ delete this;
+ return x;
+ }
+ }
+
+ return this;
+}
+
+byte TrieBranchNode::activeBranch() const
+{
+ byte n = (byte)-1;
+ for (int i = 0; i < 16; ++i)
+ if (m_nodes[i] != nullptr)
+ {
+ if (n == (byte)-1)
+ n = (byte)i;
+ else
+ return 16;
+ }
+ return n;
+}
+
+MemTrieNode* TrieInfixNode::insert(bytesConstRef _key, std::string const& _value)
+{
+ assert(_value.size());
+ mark();
+ if (contains(_key))
+ {
+ m_next = m_next->insert(_key.cropped(m_ext.size()), _value);
+ return this;
+ }
+ else
+ {
+ uint prefix = commonPrefix(_key, m_ext);
+ if (prefix)
+ {
+ // one infix becomes two infixes, then insert into the second
+ // instead of pop_front()...
+ trimFront(m_ext, prefix);
+
+ return new TrieInfixNode(_key.cropped(0, prefix), insert(_key.cropped(prefix), _value));
+ }
+ else
+ {
+ // split here.
+ auto f = m_ext[0];
+ trimFront(m_ext, 1);
+ MemTrieNode* n = m_ext.empty() ? m_next : this;
+ if (n != this)
+ {
+ m_next = nullptr;
+ delete this;
+ }
+ TrieBranchNode* ret = new TrieBranchNode(f, n);
+ ret->insert(_key, _value);
+ return ret;
+ }
+ }
+}
+
+MemTrieNode* TrieInfixNode::remove(bytesConstRef _key)
+{
+ if (contains(_key))
+ {
+ mark();
+ m_next = m_next->remove(_key.cropped(m_ext.size()));
+ if (auto p = dynamic_cast<TrieExtNode*>(m_next))
+ {
+ // merge with child...
+ m_ext.reserve(m_ext.size() + p->m_ext.size());
+ for (auto i: p->m_ext)
+ m_ext.push_back(i);
+ p->m_ext = m_ext;
+ p->mark();
+ m_next = nullptr;
+ delete this;
+ return p;
+ }
+ if (!m_next)
+ {
+ delete this;
+ return nullptr;
+ }
+ }
+ return this;
+}
+
+MemTrieNode* TrieLeafNode::insert(bytesConstRef _key, std::string const& _value)
+{
+ assert(_value.size());
+ mark();
+ if (contains(_key))
+ {
+ m_value = _value;
+ return this;
+ }
+ else
+ {
+ // create new trie.
+ auto n = MemTrieNode::newBranch(_key, _value, bytesConstRef(&m_ext), m_value);
+ delete this;
+ return n;
+ }
+}
+
+MemTrieNode* TrieLeafNode::remove(bytesConstRef _key)
+{
+ if (contains(_key))
+ {
+ delete this;
+ return nullptr;
+ }
+ return this;
+}
+
+MemTrie::~MemTrie()
+{
+ delete m_root;
+}
+
+h256 MemTrie::hash256() const
+{
+ return m_root ? m_root->hash256() : h256();
+}
+
+bytes MemTrie::rlp() const
+{
+ return m_root ? m_root->rlp() : bytes();
+}
+
+void MemTrie::debugPrint()
+{
+#if ENABLE_DEBUG_PRINT
+ if (m_root)
+ m_root->debugPrint();
+#endif
+}
+
+std::string const& MemTrie::at(std::string const& _key) const
+{
+ if (!m_root)
+ return c_nullString;
+ auto h = asNibbles(_key);
+ return m_root->at(bytesConstRef(&h));
+}
+
+void MemTrie::insert(std::string const& _key, std::string const& _value)
+{
+ if (_value.empty())
+ remove(_key);
+ auto h = asNibbles(_key);
+ m_root = m_root ? m_root->insert(&h, _value) : new TrieLeafNode(bytesConstRef(&h), _value);
+}
+
+void MemTrie::remove(std::string const& _key)
+{
+ if (m_root)
+ {
+ auto h = asNibbles(_key);
+ m_root = m_root->remove(&h);
+ }
+}
+
+}