Browse Source

Set/get values in Ext store - SSTORE & SLOAD

cl-refactor
Paweł Bylica 10 years ago
parent
commit
362acacfa2
  1. 16
      evmcc/Compiler.cpp
  2. 35
      evmcc/Ext.cpp
  3. 9
      evmcc/Ext.h
  4. 11
      evmcc/Stack.cpp
  5. 34
      evmcc/Utils.cpp
  6. 24
      evmcc/Utils.h
  7. 1
      evmcc/bytecode/store_test.evm
  8. 26
      evmcc/lll/stack_test.lll
  9. 14
      evmcc/lll/store_test.lll
  10. 4
      windows/evmcc.vcxproj
  11. 4
      windows/evmcc.vcxproj.filters

16
evmcc/Compiler.cpp

@ -247,6 +247,22 @@ std::unique_ptr<llvm::Module> Compiler::compile(const dev::bytes& bytecode)
break; break;
} }
case Instruction::SLOAD:
{
auto index = stack.pop();
auto value = ext.store(index);
stack.push(value);
break;
}
case Instruction::SSTORE:
{
auto index = stack.pop();
auto value = stack.pop();
ext.setStore(index, value);
break;
}
} }
} }

35
evmcc/Ext.cpp

@ -4,6 +4,8 @@
#include <llvm/IR/Function.h> #include <llvm/IR/Function.h>
#include <llvm/IR/TypeBuilder.h> #include <llvm/IR/TypeBuilder.h>
#include "Utils.h"
#ifdef _MSC_VER #ifdef _MSC_VER
#define EXPORT __declspec(dllexport) #define EXPORT __declspec(dllexport)
#else #else
@ -30,21 +32,42 @@ Ext::Ext(llvm::IRBuilder<>& _builder)
auto module = m_builder.GetInsertBlock()->getParent()->getParent(); auto module = m_builder.GetInsertBlock()->getParent()->getParent();
auto&& ctx = _builder.getContext(); auto&& ctx = _builder.getContext();
Function::Create(TypeBuilder<void(i<256>*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_store", module); m_args[0] = m_builder.CreateAlloca(m_builder.getIntNTy(256), nullptr, "ext.index");
Function::Create(TypeBuilder<void(i<256>*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_setStore", module); m_args[1] = m_builder.CreateAlloca(m_builder.getIntNTy(256), nullptr, "ext.value");
m_store = Function::Create(TypeBuilder<void(i<256>*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_store", module);
m_setStore = Function::Create(TypeBuilder<void(i<256>*, i<256>*), true>::get(ctx), Linkage::ExternalLinkage, "ext_setStore", module);
} }
extern "C" 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]);
}
EXPORT void ext_store(void* _index, void* _value) 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);
} }
EXPORT void ext_setStore(void* _index, void* _value) extern "C"
{ {
EXPORT void ext_store(i256* _index, i256* _value)
{
auto index = llvm2eth(*_index);
auto value = g_ext->store(index);
*_value = eth2llvm(value);
}
EXPORT void ext_setStore(i256* _index, i256* _value)
{
auto index = llvm2eth(*_index);
auto value = llvm2eth(*_value);
g_ext->setStore(index, value);
} }
} }

9
evmcc/Ext.h

@ -1,4 +1,6 @@
#pragma once
#include <llvm/IR/IRBuilder.h> #include <llvm/IR/IRBuilder.h>
#include <libevm/ExtVMFace.h> #include <libevm/ExtVMFace.h>
@ -14,8 +16,15 @@ public:
Ext(llvm::IRBuilder<>& _builder); Ext(llvm::IRBuilder<>& _builder);
static void init(std::unique_ptr<dev::eth::ExtVMFace> _ext); static void init(std::unique_ptr<dev::eth::ExtVMFace> _ext);
llvm::Value* store(llvm::Value* _index);
void setStore(llvm::Value* _index, llvm::Value* _value);
private: private:
llvm::IRBuilder<>& m_builder; llvm::IRBuilder<>& m_builder;
llvm::Value* m_args[2];
llvm::Function* m_store;
llvm::Function* m_setStore;
}; };

11
evmcc/Stack.cpp

@ -9,6 +9,8 @@
#include <llvm/IR/Function.h> #include <llvm/IR/Function.h>
#include "Utils.h"
#ifdef _MSC_VER #ifdef _MSC_VER
#define EXPORT __declspec(dllexport) #define EXPORT __declspec(dllexport)
#else #else
@ -18,15 +20,6 @@
namespace evmcc namespace evmcc
{ {
struct i256
{
uint64_t a;
uint64_t b;
uint64_t c;
uint64_t d;
};
static_assert(sizeof(i256) == 32, "Wrong i265 size");
using StackImpl = std::vector<i256>; using StackImpl = std::vector<i256>;

34
evmcc/Utils.cpp

@ -0,0 +1,34 @@
#include "Utils.h"
namespace evmcc
{
dev::u256 llvm2eth(i256 _i)
{
dev::u256 u = 0;
u |= _i.d;
u <<= 64;
u |= _i.c;
u <<= 64;
u |= _i.b;
u <<= 64;
u |= _i.a;
return u;
}
i256 eth2llvm(dev::u256 _u)
{
i256 i;
dev::u256 mask = 0xFFFFFFFFFFFFFFFF;
i.a = static_cast<uint64_t>(_u & mask);
_u >>= 64;
i.b = static_cast<uint64_t>(_u & mask);
_u >>= 64;
i.c = static_cast<uint64_t>(_u & mask);
_u >>= 64;
i.d = static_cast<uint64_t>(_u & mask);
return i;
}
}

24
evmcc/Utils.h

@ -0,0 +1,24 @@
#pragma once
#include <cstdint>
#include <libdevcore/Common.h>
namespace evmcc
{
/// Representation of 256-bit value binary compatible with LLVM i256
struct i256
{
uint64_t a;
uint64_t b;
uint64_t c;
uint64_t d;
};
static_assert(sizeof(i256) == 32, "Wrong i265 size");
dev::u256 llvm2eth(i256);
i256 eth2llvm(dev::u256);
}

1
evmcc/bytecode/store_test.evm

@ -0,0 +1 @@
607b607c60015760005760015660005603

26
evmcc/lll/stack_test.lll

@ -1,24 +1,8 @@
(asm (asm
60666666666606 123
7077777777777707 SSTORE
DUP1 SLOAD
DUP3 123
DUP1 SUB
DUP5
DUP2
DUP5
POP
POP
POP
POP
POP
POP
POP
1111
2222
DUP2
DUP2
SWAP3
SWAP1
) )

14
evmcc/lll/store_test.lll

@ -0,0 +1,14 @@
(asm
123
124
1
SSTORE
0
SSTORE
1
SLOAD
0
SLOAD
SUB
)

4
windows/evmcc.vcxproj

@ -23,13 +23,17 @@
<ClCompile Include="..\evmcc\evmcc.cpp" /> <ClCompile Include="..\evmcc\evmcc.cpp" />
<ClCompile Include="..\evmcc\ExecutionEngine.cpp" /> <ClCompile Include="..\evmcc\ExecutionEngine.cpp" />
<ClCompile Include="..\evmcc\Ext.cpp" /> <ClCompile Include="..\evmcc\Ext.cpp" />
<ClCompile Include="..\evmcc\Memory.cpp" />
<ClCompile Include="..\evmcc\Stack.cpp" /> <ClCompile Include="..\evmcc\Stack.cpp" />
<ClCompile Include="..\evmcc\Utils.cpp" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ClInclude Include="..\evmcc\Compiler.h" /> <ClInclude Include="..\evmcc\Compiler.h" />
<ClInclude Include="..\evmcc\ExecutionEngine.h" /> <ClInclude Include="..\evmcc\ExecutionEngine.h" />
<ClInclude Include="..\evmcc\Ext.h" /> <ClInclude Include="..\evmcc\Ext.h" />
<ClInclude Include="..\evmcc\Memory.h" />
<ClInclude Include="..\evmcc\Stack.h" /> <ClInclude Include="..\evmcc\Stack.h" />
<ClInclude Include="..\evmcc\Utils.h" />
</ItemGroup> </ItemGroup>
<PropertyGroup Label="Globals"> <PropertyGroup Label="Globals">
<ProjectGuid>{D51A4898-BD9E-4961-BCAE-1FE7F854BB4D}</ProjectGuid> <ProjectGuid>{D51A4898-BD9E-4961-BCAE-1FE7F854BB4D}</ProjectGuid>

4
windows/evmcc.vcxproj.filters

@ -6,11 +6,15 @@
<ClCompile Include="..\evmcc\ExecutionEngine.cpp" /> <ClCompile Include="..\evmcc\ExecutionEngine.cpp" />
<ClCompile Include="..\evmcc\Stack.cpp" /> <ClCompile Include="..\evmcc\Stack.cpp" />
<ClCompile Include="..\evmcc\Ext.cpp" /> <ClCompile Include="..\evmcc\Ext.cpp" />
<ClCompile Include="..\evmcc\Memory.cpp" />
<ClCompile Include="..\evmcc\Utils.cpp" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ClInclude Include="..\evmcc\Compiler.h" /> <ClInclude Include="..\evmcc\Compiler.h" />
<ClInclude Include="..\evmcc\ExecutionEngine.h" /> <ClInclude Include="..\evmcc\ExecutionEngine.h" />
<ClInclude Include="..\evmcc\Stack.h" /> <ClInclude Include="..\evmcc\Stack.h" />
<ClInclude Include="..\evmcc\Ext.h" /> <ClInclude Include="..\evmcc\Ext.h" />
<ClInclude Include="..\evmcc\Memory.h" />
<ClInclude Include="..\evmcc\Utils.h" />
</ItemGroup> </ItemGroup>
</Project> </Project>
Loading…
Cancel
Save