#include "RuntimeManager.h" #include "preprocessor/llvm_includes_start.h" #include #include "preprocessor/llvm_includes_end.h" #include "Stack.h" #include "Utils.h" namespace dev { namespace eth { namespace jit { llvm::StructType* RuntimeManager::getRuntimeDataType() { static llvm::StructType* type = nullptr; if (!type) { llvm::Type* elems[] = { Type::Size, // gas Type::Size, // gasPrice Type::BytePtr, // callData Type::Size, // callDataSize Type::Word, // address Type::Word, // caller Type::Word, // origin Type::Word, // callValue Type::Word, // coinBase Type::Word, // difficulty Type::Word, // gasLimit Type::Size, // blockNumber Type::Size, // blockTimestamp Type::BytePtr, // code Type::Size, // codeSize }; type = llvm::StructType::create(elems, "RuntimeData"); } return type; } llvm::StructType* RuntimeManager::getRuntimeType() { static llvm::StructType* type = nullptr; if (!type) { llvm::Type* elems[] = { Type::RuntimeDataPtr, // data Type::EnvPtr, // Env* Array::getType() // memory }; type = llvm::StructType::create(elems, "Runtime"); } return type; } namespace { llvm::Twine getName(RuntimeData::Index _index) { switch (_index) { default: return "data"; case RuntimeData::Address: return "address"; case RuntimeData::Caller: return "caller"; case RuntimeData::Origin: return "origin"; case RuntimeData::CallValue: return "callvalue"; case RuntimeData::GasPrice: return "gasprice"; case RuntimeData::CoinBase: return "coinbase"; case RuntimeData::Difficulty: return "difficulty"; case RuntimeData::GasLimit: return "gaslimit"; case RuntimeData::CallData: return "callData"; case RuntimeData::Code: return "code"; case RuntimeData::CodeSize: return "code"; case RuntimeData::CallDataSize: return "callDataSize"; case RuntimeData::Gas: return "gas"; case RuntimeData::Number: return "number"; case RuntimeData::Timestamp: return "timestamp"; } } } RuntimeManager::RuntimeManager(llvm::IRBuilder<>& _builder, code_iterator _codeBegin, code_iterator _codeEnd): CompilerHelper(_builder), m_codeBegin(_codeBegin), m_codeEnd(_codeEnd) { m_longjmp = llvm::Intrinsic::getDeclaration(getModule(), llvm::Intrinsic::eh_sjlj_longjmp); // Unpack data auto rtPtr = getRuntimePtr(); m_dataPtr = m_builder.CreateLoad(m_builder.CreateStructGEP(rtPtr, 0), "data"); assert(m_dataPtr->getType() == Type::RuntimeDataPtr); m_gasPtr = m_builder.CreateStructGEP(m_dataPtr, 0, "gas"); assert(m_gasPtr->getType() == Type::Gas->getPointerTo()); m_memPtr = m_builder.CreateStructGEP(rtPtr, 2, "mem"); assert(m_memPtr->getType() == Array::getType()->getPointerTo()); m_envPtr = m_builder.CreateLoad(m_builder.CreateStructGEP(rtPtr, 1), "env"); assert(m_envPtr->getType() == Type::EnvPtr); m_stackSize = m_builder.CreateAlloca(Type::Size, nullptr, "stackSize"); m_builder.CreateStore(m_builder.getInt64(0), m_stackSize); llvm::Type* checkStackLimitArgs[] = {Type::Size->getPointerTo(), Type::Size, Type::Size, Type::BytePtr}; m_checkStackLimit = llvm::Function::Create(llvm::FunctionType::get(Type::Void, checkStackLimitArgs, false), llvm::Function::PrivateLinkage, "stack.checkSize", getModule()); m_checkStackLimit->setDoesNotThrow(); m_checkStackLimit->setDoesNotCapture(1); auto checkBB = llvm::BasicBlock::Create(_builder.getContext(), "Check", m_checkStackLimit); auto updateBB = llvm::BasicBlock::Create(_builder.getContext(), "Update", m_checkStackLimit); auto outOfStackBB = llvm::BasicBlock::Create(_builder.getContext(), "OutOfStack", m_checkStackLimit); auto currSizePtr = &m_checkStackLimit->getArgumentList().front(); currSizePtr->setName("currSize"); auto max = currSizePtr->getNextNode(); max->setName("max"); auto diff = max->getNextNode(); diff->setName("diff"); auto jmpBuf = diff->getNextNode(); jmpBuf->setName("jmpBuf"); InsertPointGuard guard{m_builder}; m_builder.SetInsertPoint(checkBB); auto currSize = m_builder.CreateLoad(currSizePtr, "cur"); auto maxSize = m_builder.CreateNUWAdd(currSize, max, "maxSize"); auto ok = m_builder.CreateICmpULE(maxSize, m_builder.getInt64(1024), "ok"); m_builder.CreateCondBr(ok, updateBB, outOfStackBB, Type::expectTrue); m_builder.SetInsertPoint(updateBB); auto newSize = m_builder.CreateNSWAdd(currSize, diff); m_builder.CreateStore(newSize, currSizePtr); m_builder.CreateRetVoid(); m_builder.SetInsertPoint(outOfStackBB); abort(jmpBuf); m_builder.CreateUnreachable(); } void RuntimeManager::checkStackLimit(size_t _max, int _diff) { createCall(m_checkStackLimit, {m_stackSize, m_builder.getInt64(_max), m_builder.getInt64(_diff), getJmpBuf()}); } llvm::Value* RuntimeManager::getRuntimePtr() { // Expect first argument of a function to be a pointer to Runtime auto func = m_builder.GetInsertBlock()->getParent(); auto rtPtr = &func->getArgumentList().front(); assert(rtPtr->getType() == Type::RuntimePtr); return rtPtr; } llvm::Value* RuntimeManager::getDataPtr() { if (getMainFunction()) return m_dataPtr; auto rtPtr = getRuntimePtr(); auto dataPtr = m_builder.CreateLoad(m_builder.CreateStructGEP(rtPtr, 0), "data"); assert(dataPtr->getType() == getRuntimeDataType()->getPointerTo()); return dataPtr; } llvm::Value* RuntimeManager::getEnvPtr() { assert(getMainFunction()); // Available only in main function return m_envPtr; } llvm::Value* RuntimeManager::getPtr(RuntimeData::Index _index) { auto ptr = getBuilder().CreateStructGEP(getDataPtr(), _index); assert(getRuntimeDataType()->getElementType(_index)->getPointerTo() == ptr->getType()); return ptr; } llvm::Value* RuntimeManager::get(RuntimeData::Index _index) { return getBuilder().CreateLoad(getPtr(_index), getName(_index)); } void RuntimeManager::set(RuntimeData::Index _index, llvm::Value* _value) { auto ptr = getPtr(_index); assert(ptr->getType() == _value->getType()->getPointerTo()); getBuilder().CreateStore(_value, ptr); } void RuntimeManager::registerReturnData(llvm::Value* _offset, llvm::Value* _size) { auto memPtr = m_builder.CreateBitCast(getMem(), Type::BytePtr->getPointerTo()); auto mem = getBuilder().CreateLoad(memPtr, "memory"); auto idx = m_builder.CreateTrunc(_offset, Type::Size, "idx"); // Never allow memory index be a type bigger than i64 // TODO: Report bug & fix to LLVM auto returnDataPtr = getBuilder().CreateGEP(mem, idx); set(RuntimeData::ReturnData, returnDataPtr); auto size64 = getBuilder().CreateTrunc(_size, Type::Size); set(RuntimeData::ReturnDataSize, size64); } void RuntimeManager::registerSuicide(llvm::Value* _balanceAddress) { set(RuntimeData::SuicideDestAddress, _balanceAddress); } void RuntimeManager::exit(ReturnCode _returnCode) { if (m_stack) m_stack->free(); m_builder.CreateRet(Constant::get(_returnCode)); } void RuntimeManager::abort(llvm::Value* _jmpBuf) { auto longjmp = llvm::Intrinsic::getDeclaration(getModule(), llvm::Intrinsic::eh_sjlj_longjmp); createCall(longjmp, {_jmpBuf}); } llvm::Value* RuntimeManager::get(Instruction _inst) { switch (_inst) { default: assert(false); return nullptr; case Instruction::ADDRESS: return get(RuntimeData::Address); case Instruction::CALLER: return get(RuntimeData::Caller); case Instruction::ORIGIN: return get(RuntimeData::Origin); case Instruction::CALLVALUE: return get(RuntimeData::CallValue); case Instruction::GASPRICE: return get(RuntimeData::GasPrice); case Instruction::COINBASE: return get(RuntimeData::CoinBase); case Instruction::DIFFICULTY: return get(RuntimeData::Difficulty); case Instruction::GASLIMIT: return get(RuntimeData::GasLimit); case Instruction::NUMBER: return get(RuntimeData::Number); case Instruction::TIMESTAMP: return get(RuntimeData::Timestamp); } } llvm::Value* RuntimeManager::getCallData() { return get(RuntimeData::CallData); } llvm::Value* RuntimeManager::getCode() { // OPT Check what is faster //return get(RuntimeData::Code); return m_builder.CreateGlobalStringPtr({reinterpret_cast(m_codeBegin), static_cast(m_codeEnd - m_codeBegin)}, "code"); } llvm::Value* RuntimeManager::getCodeSize() { return Constant::get(m_codeEnd - m_codeBegin); } llvm::Value* RuntimeManager::getCallDataSize() { auto value = get(RuntimeData::CallDataSize); assert(value->getType() == Type::Size); return getBuilder().CreateZExt(value, Type::Word); } llvm::Value* RuntimeManager::getGas() { auto gas = get(RuntimeData::Gas); assert(gas->getType() == Type::Gas); return gas; } llvm::Value* RuntimeManager::getGasPtr() { assert(getMainFunction()); return m_gasPtr; } llvm::Value* RuntimeManager::getMem() { assert(getMainFunction()); return m_memPtr; } void RuntimeManager::setGas(llvm::Value* _gas) { assert(_gas->getType() == Type::Gas); set(RuntimeData::Gas, _gas); } } } }