|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <csetjmp>
|
|
|
|
|
|
|
|
#include <libevm/ExtVMFace.h>
|
|
|
|
|
|
|
|
#include "CompilerHelper.h"
|
|
|
|
#include "Utils.h"
|
|
|
|
#include "Type.h"
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define EXPORT __declspec(dllexport)
|
|
|
|
#else
|
|
|
|
#define EXPORT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace dev
|
|
|
|
{
|
|
|
|
namespace eth
|
|
|
|
{
|
|
|
|
namespace jit
|
|
|
|
{
|
|
|
|
|
|
|
|
struct RuntimeData
|
|
|
|
{
|
|
|
|
enum Index: unsigned
|
|
|
|
{
|
|
|
|
Gas,
|
|
|
|
Address,
|
|
|
|
Caller,
|
|
|
|
Origin,
|
|
|
|
CallValue,
|
|
|
|
CallDataSize,
|
|
|
|
GasPrice,
|
|
|
|
PrevHash,
|
|
|
|
CoinBase,
|
|
|
|
TimeStamp,
|
|
|
|
Number,
|
|
|
|
Difficulty,
|
|
|
|
GasLimit,
|
|
|
|
CodeSize,
|
|
|
|
|
|
|
|
_size,
|
|
|
|
|
|
|
|
ReturnDataOffset = CallValue, // Reuse 2 fields for return data reference
|
|
|
|
ReturnDataSize = CallDataSize
|
|
|
|
};
|
|
|
|
|
|
|
|
i256 elems[_size];
|
|
|
|
byte const* callData;
|
|
|
|
byte const* code;
|
|
|
|
decltype(&jmp_buf{}[0]) jmpBuf;
|
|
|
|
|
|
|
|
static llvm::StructType* getType();
|
|
|
|
};
|
|
|
|
|
|
|
|
using StackImpl = std::vector<i256>;
|
|
|
|
using MemoryImpl = bytes;
|
|
|
|
|
|
|
|
class Runtime
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Runtime(u256 _gas, ExtVMFace& _ext, jmp_buf _jmpBuf, bool _outputLogs);
|
|
|
|
|
|
|
|
Runtime(const Runtime&) = delete;
|
|
|
|
void operator=(const Runtime&) = delete;
|
|
|
|
|
|
|
|
RuntimeData* getDataPtr() { return &m_data; }
|
|
|
|
|
|
|
|
StackImpl& getStack() { return m_stack; }
|
|
|
|
MemoryImpl& getMemory() { return m_memory; }
|
|
|
|
ExtVMFace& getExt() { return m_ext; }
|
|
|
|
|
|
|
|
u256 getGas() const;
|
|
|
|
bytesConstRef getReturnData() const;
|
|
|
|
decltype(&jmp_buf{}[0]) getJmpBuf() { return m_data.jmpBuf; }
|
|
|
|
bool outputLogs() const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
void set(RuntimeData::Index _index, u256 _value);
|
|
|
|
|
|
|
|
/// @internal Must be the first element to asure Runtime* === RuntimeData*
|
|
|
|
RuntimeData m_data;
|
|
|
|
StackImpl m_stack;
|
|
|
|
MemoryImpl m_memory;
|
|
|
|
ExtVMFace& m_ext;
|
|
|
|
bool m_outputLogs; ///< write LOG statements to console
|
|
|
|
};
|
|
|
|
|
|
|
|
class RuntimeManager: public CompilerHelper
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
RuntimeManager(llvm::IRBuilder<>& _builder);
|
|
|
|
|
|
|
|
llvm::Value* getRuntimePtr();
|
|
|
|
|
|
|
|
llvm::Value* get(RuntimeData::Index _index);
|
|
|
|
llvm::Value* get(Instruction _inst);
|
|
|
|
llvm::Value* getGas(); // TODO: Remove
|
|
|
|
llvm::Value* getCallData();
|
|
|
|
llvm::Value* getCode();
|
|
|
|
void setGas(llvm::Value* _gas);
|
|
|
|
|
|
|
|
void registerReturnData(llvm::Value* _index, llvm::Value* _size);
|
|
|
|
|
|
|
|
void raiseException(ReturnCode _returnCode);
|
|
|
|
|
|
|
|
private:
|
|
|
|
llvm::Value* getPtr(RuntimeData::Index _index);
|
|
|
|
void set(RuntimeData::Index _index, llvm::Value* _value);
|
|
|
|
llvm::Value* getJmpBuf();
|
|
|
|
|
|
|
|
llvm::GlobalVariable* m_dataPtr = nullptr;
|
|
|
|
llvm::Function* m_longjmp = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|