You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
247 lines
8.2 KiB
247 lines
8.2 KiB
#include "Memory.h"
|
|
|
|
#include <vector>
|
|
#include <iostream>
|
|
#include <iomanip>
|
|
#include <cstdint>
|
|
#include <cassert>
|
|
|
|
#include <llvm/IR/GlobalVariable.h>
|
|
#include <llvm/IR/Function.h>
|
|
|
|
#include <libdevcore/Common.h>
|
|
|
|
#include "Type.h"
|
|
#include "Runtime.h"
|
|
#include "GasMeter.h"
|
|
|
|
namespace evmcc
|
|
{
|
|
|
|
Memory::Memory(llvm::IRBuilder<>& _builder, llvm::Module* _module, GasMeter& _gasMeter):
|
|
m_builder(_builder)
|
|
{
|
|
auto i64Ty = m_builder.getInt64Ty();
|
|
llvm::Type* argTypes[] = {i64Ty, i64Ty};
|
|
auto dumpTy = llvm::FunctionType::get(m_builder.getVoidTy(), llvm::ArrayRef<llvm::Type*>(argTypes), false);
|
|
m_memDump = llvm::Function::Create(dumpTy, llvm::GlobalValue::LinkageTypes::ExternalLinkage,
|
|
"evmccrt_memory_dump", _module);
|
|
|
|
m_data = new llvm::GlobalVariable(*_module, Type::BytePtr, false, llvm::GlobalVariable::PrivateLinkage, llvm::UndefValue::get(Type::BytePtr), "mem.data");
|
|
m_data->setUnnamedAddr(true); // Address is not important
|
|
|
|
m_size = new llvm::GlobalVariable(*_module, Type::i256, false, llvm::GlobalVariable::PrivateLinkage, Constant::get(0), "mem.size");
|
|
m_size->setUnnamedAddr(true); // Address is not important
|
|
|
|
m_returnDataOffset = new llvm::GlobalVariable(*_module, Type::i256, false, llvm::GlobalVariable::ExternalLinkage, nullptr, "mem_returnDataOffset");
|
|
m_returnDataOffset->setUnnamedAddr(true); // Address is not important
|
|
|
|
m_returnDataSize = new llvm::GlobalVariable(*_module, Type::i256, false, llvm::GlobalVariable::ExternalLinkage, nullptr, "mem_returnDataSize");
|
|
m_returnDataSize->setUnnamedAddr(true); // Address is not important
|
|
|
|
m_resize = llvm::Function::Create(llvm::FunctionType::get(Type::BytePtr, Type::WordPtr, false), llvm::Function::ExternalLinkage, "mem_resize", _module);
|
|
m_loadWord = createFunc(false, Type::i256, _module, _gasMeter);
|
|
m_storeWord = createFunc(true, Type::i256, _module, _gasMeter);
|
|
m_storeByte = createFunc(true, Type::Byte, _module, _gasMeter);
|
|
|
|
m_require = createRequireFunc(_module, _gasMeter);
|
|
}
|
|
|
|
llvm::Function* Memory::createRequireFunc(llvm::Module* _module, GasMeter& _gasMeter)
|
|
{
|
|
auto func = llvm::Function::Create(llvm::FunctionType::get(Type::Void, Type::i256, false), llvm::Function::PrivateLinkage, "mem.require", _module);
|
|
|
|
auto checkBB = llvm::BasicBlock::Create(func->getContext(), "check", func);
|
|
auto resizeBB = llvm::BasicBlock::Create(func->getContext(), "resize", func);
|
|
auto returnBB = llvm::BasicBlock::Create(func->getContext(), "return", func);
|
|
|
|
// BB "check"
|
|
llvm::IRBuilder<> builder(checkBB);
|
|
llvm::Value* sizeRequired = func->arg_begin();
|
|
sizeRequired->setName("sizeRequired");
|
|
auto size = builder.CreateLoad(m_size, "size");
|
|
auto resizeNeeded = builder.CreateICmpULE(size, sizeRequired, "resizeNeeded");
|
|
builder.CreateCondBr(resizeNeeded, resizeBB, returnBB); // OPT branch weights?
|
|
|
|
// BB "resize"
|
|
builder.SetInsertPoint(resizeBB);
|
|
// Check gas first
|
|
auto wordsRequired = builder.CreateUDiv(builder.CreateAdd(sizeRequired, Constant::get(31)), Constant::get(32), "wordsRequired");
|
|
auto words = builder.CreateUDiv(builder.CreateAdd(size, Constant::get(31)), Constant::get(32), "words");
|
|
auto newWords = builder.CreateSub(wordsRequired, words, "addtionalWords");
|
|
_gasMeter.checkMemory(newWords, builder);
|
|
// Resize
|
|
builder.CreateStore(sizeRequired, m_size);
|
|
auto newData = builder.CreateCall(m_resize, m_size, "newData");
|
|
builder.CreateStore(newData, m_data);
|
|
builder.CreateBr(returnBB);
|
|
|
|
// BB "return"
|
|
builder.SetInsertPoint(returnBB);
|
|
builder.CreateRetVoid();
|
|
return func;
|
|
}
|
|
|
|
llvm::Function* Memory::createFunc(bool _isStore, llvm::Type* _valueType, llvm::Module* _module, GasMeter& _gasMeter)
|
|
{
|
|
auto isWord = _valueType == Type::i256;
|
|
|
|
llvm::Type* storeArgs[] = {Type::i256, _valueType};
|
|
auto name = _isStore ? isWord ? "mstore" : "mstore8" : "mload";
|
|
auto funcType = _isStore ? llvm::FunctionType::get(Type::Void, storeArgs, false) : llvm::FunctionType::get(Type::i256, Type::i256, false);
|
|
auto func = llvm::Function::Create(funcType, llvm::Function::PrivateLinkage, name, _module);
|
|
|
|
auto checkBB = llvm::BasicBlock::Create(func->getContext(), "check", func);
|
|
auto resizeBB = llvm::BasicBlock::Create(func->getContext(), "resize", func);
|
|
auto accessBB = llvm::BasicBlock::Create(func->getContext(), "access", func);
|
|
|
|
// BB "check"
|
|
llvm::IRBuilder<> builder(checkBB);
|
|
llvm::Value* index = func->arg_begin();
|
|
index->setName("index");
|
|
auto valueSize = _valueType->getPrimitiveSizeInBits() / 8;
|
|
auto sizeRequired = builder.CreateAdd(index, Constant::get(valueSize), "sizeRequired");
|
|
auto size = builder.CreateLoad(m_size, "size");
|
|
auto resizeNeeded = builder.CreateICmpULE(size, sizeRequired, "resizeNeeded");
|
|
builder.CreateCondBr(resizeNeeded, resizeBB, accessBB); // OPT branch weights?
|
|
|
|
// BB "resize"
|
|
builder.SetInsertPoint(resizeBB);
|
|
// Check gas first
|
|
auto wordsRequired = builder.CreateUDiv(builder.CreateAdd(sizeRequired, Constant::get(31)), Constant::get(32), "wordsRequired");
|
|
auto words = builder.CreateUDiv(builder.CreateAdd(size, Constant::get(31)), Constant::get(32), "words");
|
|
auto newWords = builder.CreateSub(wordsRequired, words, "addtionalWords");
|
|
_gasMeter.checkMemory(newWords, builder);
|
|
// Resize
|
|
builder.CreateStore(sizeRequired, m_size);
|
|
auto newData = builder.CreateCall(m_resize, m_size, "newData");
|
|
builder.CreateStore(newData, m_data);
|
|
builder.CreateBr(accessBB);
|
|
|
|
// BB "access"
|
|
builder.SetInsertPoint(accessBB);
|
|
auto data = builder.CreateLoad(m_data, "data");
|
|
auto ptr = builder.CreateGEP(data, index, "ptr");
|
|
if (isWord)
|
|
ptr = builder.CreateBitCast(ptr, Type::WordPtr, "wordPtr");
|
|
if (_isStore)
|
|
{
|
|
llvm::Value* value = ++func->arg_begin();
|
|
value->setName("value");
|
|
builder.CreateStore(value, ptr);
|
|
builder.CreateRetVoid();
|
|
}
|
|
else
|
|
{
|
|
auto ret = builder.CreateLoad(ptr);
|
|
builder.CreateRet(ret);
|
|
}
|
|
return func;
|
|
}
|
|
|
|
|
|
llvm::Value* Memory::loadWord(llvm::Value* _addr)
|
|
{
|
|
auto value = m_builder.CreateCall(m_loadWord, _addr);
|
|
|
|
dump(0);
|
|
return value;
|
|
}
|
|
|
|
void Memory::storeWord(llvm::Value* _addr, llvm::Value* _word)
|
|
{
|
|
m_builder.CreateCall2(m_storeWord, _addr, _word);
|
|
|
|
dump(0);
|
|
}
|
|
|
|
void Memory::storeByte(llvm::Value* _addr, llvm::Value* _word)
|
|
{
|
|
auto byte = m_builder.CreateTrunc(_word, Type::Byte, "byte");
|
|
m_builder.CreateCall2(m_storeByte, _addr, byte);
|
|
|
|
dump(0);
|
|
}
|
|
|
|
llvm::Value* Memory::getSize()
|
|
{
|
|
return m_builder.CreateLoad(m_size);
|
|
}
|
|
|
|
void Memory::require(llvm::Value* _size)
|
|
{
|
|
m_builder.CreateCall(m_require, _size);
|
|
}
|
|
|
|
void Memory::registerReturnData(llvm::Value* _index, llvm::Value* _size)
|
|
{
|
|
auto sizeRequired = m_builder.CreateAdd(_index, _size, "sizeRequired");
|
|
require(sizeRequired); // Make sure that memory is allocated and count gas
|
|
|
|
m_builder.CreateStore(_index, m_returnDataOffset);
|
|
m_builder.CreateStore(_size, m_returnDataSize);
|
|
}
|
|
|
|
void Memory::dump(uint64_t _begin, uint64_t _end)
|
|
{
|
|
if (getenv("EVMCC_DEBUG_MEMORY") == nullptr)
|
|
return;
|
|
|
|
auto beginVal = llvm::ConstantInt::get(m_builder.getInt64Ty(), _begin);
|
|
auto endVal = llvm::ConstantInt::get(m_builder.getInt64Ty(), _end);
|
|
|
|
std::vector<llvm::Value*> args = {beginVal, endVal};
|
|
m_builder.CreateCall(m_memDump, llvm::ArrayRef<llvm::Value*>(args));
|
|
}
|
|
|
|
} // namespace evmcc
|
|
|
|
extern "C"
|
|
{
|
|
using namespace evmcc;
|
|
|
|
EXPORT i256 mem_returnDataOffset;
|
|
EXPORT i256 mem_returnDataSize;
|
|
|
|
EXPORT uint8_t* mem_resize(i256* _size)
|
|
{
|
|
auto size = _size->a; // Trunc to 64-bit
|
|
auto& memory = Runtime::getMemory();
|
|
memory.resize(size);
|
|
return memory.data();
|
|
}
|
|
|
|
EXPORT void evmccrt_memory_dump(uint64_t _begin, uint64_t _end)
|
|
{
|
|
if (_end == 0)
|
|
_end = Runtime::getMemory().size();
|
|
|
|
std::cerr << "MEMORY: active size: " << std::dec
|
|
<< Runtime::getMemory().size() / 32 << " words\n";
|
|
std::cerr << "MEMORY: dump from " << std::dec
|
|
<< _begin << " to " << _end << ":";
|
|
if (_end <= _begin)
|
|
return;
|
|
|
|
_begin = _begin / 16 * 16;
|
|
for (size_t i = _begin; i < _end; i++)
|
|
{
|
|
if ((i - _begin) % 16 == 0)
|
|
std::cerr << '\n' << std::dec << i << ": ";
|
|
|
|
auto b = Runtime::getMemory()[i];
|
|
std::cerr << std::hex << std::setw(2) << static_cast<int>(b) << ' ';
|
|
}
|
|
std::cerr << std::endl;
|
|
}
|
|
|
|
} // extern "C"
|
|
|
|
dev::bytesConstRef evmcc::Memory::getReturnData()
|
|
{
|
|
// TODO: Handle large indexes
|
|
auto offset = static_cast<size_t>(llvm2eth(mem_returnDataOffset));
|
|
auto size = static_cast<size_t>(llvm2eth(mem_returnDataSize));
|
|
auto& memory = Runtime::getMemory();
|
|
return {memory.data() + offset, size};
|
|
}
|
|
|