Paweł Bylica
10 years ago
6 changed files with 138 additions and 7 deletions
@ -0,0 +1,79 @@ |
|||
/*
|
|||
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/>.
|
|||
*/ |
|||
|
|||
// SmartVM is only available if EVM JIT is enabled
|
|||
#if ETH_EVMJIT |
|||
|
|||
#include "SmartVM.h" |
|||
#include <unordered_map> |
|||
#include <libdevcore/Log.h> |
|||
#include <libdevcrypto/SHA3.h> |
|||
#include <evmjit/JIT.h> |
|||
#include <evmjit/libevmjit-cpp/Utils.h> |
|||
#include "VMFactory.h" |
|||
|
|||
namespace dev |
|||
{ |
|||
namespace eth |
|||
{ |
|||
namespace |
|||
{ |
|||
using HitMap = std::unordered_map<h256, uint64_t>; |
|||
|
|||
HitMap& getHitMap() |
|||
{ |
|||
static HitMap s_hitMap; |
|||
return s_hitMap; |
|||
} |
|||
} |
|||
|
|||
bytesConstRef SmartVM::go(ExtVMFace& _ext, OnOpFunc const& _onOp, uint64_t _steps) |
|||
{ |
|||
auto codeHash = sha3(_ext.code); |
|||
auto vmKind = VMKind::Interpreter; // default VM
|
|||
|
|||
// Jitted EVM code already in memory?
|
|||
if (evmjit::JIT::isCodeReady(eth2llvm(codeHash))) |
|||
{ |
|||
cnote << "Jitted"; |
|||
vmKind = VMKind::JIT; |
|||
} |
|||
else |
|||
{ |
|||
// Check EVM code hit count
|
|||
static const uint64_t c_hitTreshold = 1; |
|||
auto& hits = getHitMap()[codeHash]; |
|||
++hits; |
|||
if (hits > c_hitTreshold) |
|||
{ |
|||
cnote << "JIT selected"; |
|||
vmKind = VMKind::JIT; |
|||
} |
|||
} |
|||
|
|||
// TODO: Selected VM must be kept only because it returns reference to its internal memory.
|
|||
// VM implementations should be stateless, without gas counter and escaping memory reference.
|
|||
m_selectedVM = VMFactory::create(vmKind, gas()); |
|||
auto out = m_selectedVM->go(_ext, _onOp, _steps); |
|||
m_gas = m_selectedVM->gas(); |
|||
return out; |
|||
} |
|||
|
|||
} |
|||
} |
|||
|
|||
#endif |
@ -0,0 +1,43 @@ |
|||
/*
|
|||
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/>.
|
|||
*/ |
|||
#pragma once |
|||
|
|||
#include "VMFace.h" |
|||
|
|||
namespace dev |
|||
{ |
|||
namespace eth |
|||
{ |
|||
|
|||
/// Smart VM proxy.
|
|||
///
|
|||
/// This class is a strategy pattern implementation for VM. For every EVM code
|
|||
/// execution request it tries to select the best VM implementation (Interpreter or JIT)
|
|||
/// by analyzing available information like: code size, hit count, JIT status, etc.
|
|||
class SmartVM: public VMFace |
|||
{ |
|||
public: |
|||
SmartVM(u256 _gas): VMFace(_gas) {} |
|||
|
|||
virtual bytesConstRef go(ExtVMFace& _ext, OnOpFunc const& _onOp = {}, uint64_t _steps = (uint64_t)-1) override final; |
|||
|
|||
private: |
|||
std::unique_ptr<VMFace> m_selectedVM; |
|||
}; |
|||
|
|||
} |
|||
} |
Loading…
Reference in new issue