/* 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 . */ /** @file MemoryDB.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include #include #include #include "SHA3.h" namespace dev { struct DBChannel: public LogChannel { static const char* name(); static const int verbosity = 18; }; struct DBWarn: public LogChannel { static const char* name(); static const int verbosity = 1; }; #define dbdebug clog(DBChannel) #define dbwarn clog(DBWarn) class MemoryDB { friend class EnforceRefs; public: MemoryDB() {} MemoryDB(MemoryDB const& _c) { operator=(_c); } MemoryDB& operator=(MemoryDB const& _c); void clear() { m_main.clear(); } // WARNING !!!! didn't originally clear m_refCount!!! std::unordered_map get() const; std::string lookup(h256 const& _h) const; bool exists(h256 const& _h) const; void insert(h256 const& _h, bytesConstRef _v); bool kill(h256 const& _h); void purge(); bytes lookupAux(h256 const& _h) const { ReadGuard l(x_this); auto it = m_aux.find(_h); if (it != m_aux.end() && (!m_enforceRefs || it->second.second)) return it->second.first; return bytes(); } void removeAux(h256 const& _h) { WriteGuard l(x_this); m_aux[_h].second = false; } void insertAux(h256 const& _h, bytesConstRef _v) { WriteGuard l(x_this); m_aux[_h] = make_pair(_v.toBytes(), true); } h256Hash keys() const; protected: mutable SharedMutex x_this; std::unordered_map> m_main; std::unordered_map> m_aux; mutable bool m_enforceRefs = false; }; class EnforceRefs { public: EnforceRefs(MemoryDB const& _o, bool _r): m_o(_o), m_r(_o.m_enforceRefs) { _o.m_enforceRefs = _r; } ~EnforceRefs() { m_o.m_enforceRefs = m_r; } private: MemoryDB const& m_o; bool m_r; }; inline std::ostream& operator<<(std::ostream& _out, MemoryDB const& _m) { for (auto const& i: _m.get()) { _out << i.first << ": "; _out << RLP(i.second); _out << " " << toHex(i.second); _out << std::endl; } return _out; } }