#include "Ext.h" #include #include #include #include "Runtime.h" #ifdef _MSC_VER #define EXPORT __declspec(dllexport) #else #define EXPORT #endif using namespace llvm; using llvm::types::i; using Linkage = llvm::GlobalValue::LinkageTypes; using dev::h256; namespace evmcc { // TODO: Copy of dev::eth::fromAddress in VM.h inline dev::u256 fromAddress(dev::Address _a) { return (dev::u160)_a; } struct ExtData { i256 address; i256 caller; i256 origin; i256 callvalue; i256 calldatasize; i256 gasprice; i256 prevhash; i256 coinbase; i256 timestamp; i256 number; i256 difficulty; i256 gaslimit; const byte* calldata; }; Ext::Ext(llvm::IRBuilder<>& _builder, llvm::Module* module) : m_builder(_builder) { auto&& ctx = _builder.getContext(); auto i256Ty = m_builder.getIntNTy(256); m_args[0] = m_builder.CreateAlloca(i256Ty, nullptr, "ext.index"); m_args[1] = m_builder.CreateAlloca(i256Ty, nullptr, "ext.value"); Type* elements[] = { i256Ty, // i256 address; i256Ty, // i256 caller; i256Ty, // i256 origin; i256Ty, // i256 callvalue; i256Ty, // i256 calldatasize; i256Ty, // i256 gasprice; i256Ty, // i256 prevhash; i256Ty, // i256 coinbase; i256Ty, // i256 timestamp; i256Ty, // i256 number; i256Ty, // i256 difficulty; i256Ty, // i256 gaslimit; //m_builder.getInt8PtrTy() }; auto extDataTy = StructType::create(elements, "ext.Data"); m_data = m_builder.CreateAlloca(extDataTy, nullptr, "ext.data"); m_init = Function::Create(FunctionType::get(m_builder.getVoidTy(), extDataTy->getPointerTo(), false), Linkage::ExternalLinkage, "ext_init", module); m_store = Function::Create(TypeBuilder*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_store", module); m_setStore = Function::Create(TypeBuilder*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_setStore", module); m_calldataload = Function::Create(TypeBuilder*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_calldataload", module); m_balance = Function::Create(TypeBuilder*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_balance", module); m_bswap = Intrinsic::getDeclaration(module, Intrinsic::bswap, i256Ty); m_builder.CreateCall(m_init, m_data); } llvm::Value* Ext::store(llvm::Value* _index) { m_builder.CreateStore(_index, m_args[0]); m_builder.CreateCall(m_store, m_args); return m_builder.CreateLoad(m_args[1]); } void Ext::setStore(llvm::Value* _index, llvm::Value* _value) { m_builder.CreateStore(_index, m_args[0]); m_builder.CreateStore(_value, m_args[1]); m_builder.CreateCall(m_setStore, m_args); } Value* Ext::getDataElem(unsigned _index, const Twine& _name) { auto valuePtr = m_builder.CreateStructGEP(m_data, _index, _name); return m_builder.CreateLoad(valuePtr); } Value* Ext::address() { return getDataElem(0, "address"); } Value* Ext::caller() { return getDataElem(1, "caller"); } Value* Ext::origin() { return getDataElem(2, "origin"); } Value* Ext::callvalue() { return getDataElem(3, "callvalue"); } Value* Ext::calldatasize() { return getDataElem(4, "calldatasize"); } Value* Ext::gasprice() { return getDataElem(5, "gasprice"); } Value* Ext::prevhash() { return getDataElem(6, "prevhash"); } Value* Ext::coinbase() { return getDataElem(7, "coinbase"); } Value* Ext::timestamp() { return getDataElem(8, "timestamp"); } Value* Ext::number() { return getDataElem(9, "number"); } Value* Ext::difficulty() { return getDataElem(10, "difficulty"); } Value* Ext::gaslimit() { return getDataElem(11, "gaslimit"); } Value* Ext::calldataload(Value* _index) { m_builder.CreateStore(_index, m_args[0]); m_builder.CreateCall(m_calldataload, m_args); return m_builder.CreateLoad(m_args[1]); } Value* Ext::bswap(Value* _value) { return m_builder.CreateCall(m_bswap, _value); } Value* Ext::balance(Value* _address) { auto address = bswap(_address); // to BigEndian // TODO: I don't get it. It seems not needed m_builder.CreateStore(address, m_args[0]); m_builder.CreateCall(m_balance, m_args); return m_builder.CreateLoad(m_args[1]); } extern "C" { EXPORT void ext_init(ExtData* _extData) { auto&& ext = Runtime::getExt(); _extData->address = eth2llvm(fromAddress(ext.myAddress)); _extData->caller = eth2llvm(fromAddress(ext.caller)); _extData->origin = eth2llvm(fromAddress(ext.origin)); _extData->callvalue = eth2llvm(ext.value); _extData->gasprice = eth2llvm(ext.gasPrice); _extData->calldatasize = eth2llvm(ext.data.size()); _extData->prevhash = eth2llvm(ext.previousBlock.hash); _extData->coinbase = eth2llvm(fromAddress(ext.currentBlock.coinbaseAddress)); _extData->timestamp = eth2llvm(ext.currentBlock.timestamp); _extData->number = eth2llvm(ext.currentBlock.number); _extData->difficulty = eth2llvm(ext.currentBlock.difficulty); _extData->gaslimit = eth2llvm(ext.currentBlock.gasLimit); //_extData->calldata = ext.data.data(); } EXPORT void ext_store(i256* _index, i256* _value) { auto index = llvm2eth(*_index); auto value = Runtime::getExt().store(index); *_value = eth2llvm(value); } EXPORT void ext_setStore(i256* _index, i256* _value) { auto index = llvm2eth(*_index); auto value = llvm2eth(*_value); Runtime::getExt().setStore(index, value); } EXPORT void ext_calldataload(i256* _index, i256* _value) { auto index = static_cast(llvm2eth(*_index)); assert(index + 31 > index); // TODO: Handle large index auto b = reinterpret_cast(_value); for (size_t i = index, j = 31; i <= index + 31; ++i, --j) b[j] = i < Runtime::getExt().data.size() ? Runtime::getExt().data[i] : 0; // TODO: It all can be done by adding padding to data or by using min() algorithm without branch } EXPORT void ext_balance(h256* _address, i256* _value) { auto u = Runtime::getExt().balance(dev::right160(*_address)); *_value = eth2llvm(u); } } }