Closure return, arguments fixed. Added proper tests
This commit is contained in:
parent
fa1db8d2dc
commit
2ea4c632d1
|
@ -6,7 +6,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type Account struct {
|
type Account struct {
|
||||||
Address []byte
|
address []byte
|
||||||
Amount *big.Int
|
Amount *big.Int
|
||||||
Nonce uint64
|
Nonce uint64
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,7 @@ func NewAccount(address []byte, amount *big.Int) *Account {
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAccountFromData(address, data []byte) *Account {
|
func NewAccountFromData(address, data []byte) *Account {
|
||||||
account := &Account{Address: address}
|
account := &Account{address: address}
|
||||||
account.RlpDecode(data)
|
account.RlpDecode(data)
|
||||||
|
|
||||||
return account
|
return account
|
||||||
|
@ -30,11 +30,15 @@ func (a *Account) AddFunds(funds *big.Int) {
|
||||||
a.Amount.Add(a.Amount, funds)
|
a.Amount.Add(a.Amount, funds)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a *Account) Address() []byte {
|
||||||
|
return a.address
|
||||||
|
}
|
||||||
|
|
||||||
// Implements Callee
|
// Implements Callee
|
||||||
func (a *Account) ReturnGas(value *big.Int, state *State) {
|
func (a *Account) ReturnGas(value *big.Int, state *State) {
|
||||||
// Return the value back to the sender
|
// Return the value back to the sender
|
||||||
a.AddFunds(value)
|
a.AddFunds(value)
|
||||||
state.UpdateAccount(a.Address, a)
|
state.UpdateAccount(a.address, a)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *Account) RlpEncode() []byte {
|
func (a *Account) RlpEncode() []byte {
|
||||||
|
|
|
@ -9,13 +9,13 @@ import (
|
||||||
|
|
||||||
type Callee interface {
|
type Callee interface {
|
||||||
ReturnGas(*big.Int, *State)
|
ReturnGas(*big.Int, *State)
|
||||||
|
Address() []byte
|
||||||
}
|
}
|
||||||
|
|
||||||
type ClosureBody interface {
|
type ClosureBody interface {
|
||||||
Callee
|
Callee
|
||||||
ethutil.RlpEncodable
|
ethutil.RlpEncodable
|
||||||
GetMem(int64) *ethutil.Value
|
GetMem(int64) *ethutil.Value
|
||||||
Address() []byte
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Basic inline closure object which implement the 'closure' interface
|
// Basic inline closure object which implement the 'closure' interface
|
||||||
|
@ -24,8 +24,8 @@ type Closure struct {
|
||||||
object ClosureBody
|
object ClosureBody
|
||||||
State *State
|
State *State
|
||||||
|
|
||||||
gas *big.Int
|
Gas *big.Int
|
||||||
val *big.Int
|
Value *big.Int
|
||||||
|
|
||||||
Args []byte
|
Args []byte
|
||||||
}
|
}
|
||||||
|
@ -45,6 +45,10 @@ func (c *Closure) GetMem(x int64) *ethutil.Value {
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Closure) Address() []byte {
|
||||||
|
return c.object.Address()
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Closure) Call(vm *Vm, args []byte) []byte {
|
func (c *Closure) Call(vm *Vm, args []byte) []byte {
|
||||||
c.Args = args
|
c.Args = args
|
||||||
|
|
||||||
|
@ -56,9 +60,9 @@ func (c *Closure) Return(ret []byte) []byte {
|
||||||
// If no callee is present return it to
|
// If no callee is present return it to
|
||||||
// the origin (i.e. contract or tx)
|
// the origin (i.e. contract or tx)
|
||||||
if c.callee != nil {
|
if c.callee != nil {
|
||||||
c.callee.ReturnGas(c.gas, c.State)
|
c.callee.ReturnGas(c.Gas, c.State)
|
||||||
} else {
|
} else {
|
||||||
c.object.ReturnGas(c.gas, c.State)
|
c.object.ReturnGas(c.Gas, c.State)
|
||||||
// TODO incase it's a POST contract we gotta serialise the contract again.
|
// TODO incase it's a POST contract we gotta serialise the contract again.
|
||||||
// But it's not yet defined
|
// But it's not yet defined
|
||||||
}
|
}
|
||||||
|
@ -69,9 +73,13 @@ func (c *Closure) Return(ret []byte) []byte {
|
||||||
// Implement the Callee interface
|
// Implement the Callee interface
|
||||||
func (c *Closure) ReturnGas(gas *big.Int, state *State) {
|
func (c *Closure) ReturnGas(gas *big.Int, state *State) {
|
||||||
// Return the gas to the closure
|
// Return the gas to the closure
|
||||||
c.gas.Add(c.gas, gas)
|
c.Gas.Add(c.Gas, gas)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Closure) GetGas() *big.Int {
|
func (c *Closure) Object() ClosureBody {
|
||||||
return c.gas
|
return c.object
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Closure) Callee() Callee {
|
||||||
|
return c.callee
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,17 +43,15 @@ const (
|
||||||
oCALLVALUE = 0x34
|
oCALLVALUE = 0x34
|
||||||
oCALLDATA = 0x35
|
oCALLDATA = 0x35
|
||||||
oCALLDATASIZE = 0x36
|
oCALLDATASIZE = 0x36
|
||||||
oRETURNDATASIZE = 0x37
|
oGASPRICE = 0x37
|
||||||
oTXGASPRICE = 0x38
|
|
||||||
|
|
||||||
// 0x40 range - block operations
|
// 0x40 range - block operations
|
||||||
oPREVHASH = 0x40
|
oPREVHASH = 0x40
|
||||||
oPREVNONCE = 0x41
|
oCOINBASE = 0x41
|
||||||
oCOINBASE = 0x42
|
oTIMESTAMP = 0x42
|
||||||
oTIMESTAMP = 0x43
|
oNUMBER = 0x43
|
||||||
oNUMBER = 0x44
|
oDIFFICULTY = 0x44
|
||||||
oDIFFICULTY = 0x45
|
oGASLIMIT = 0x45
|
||||||
oGASLIMIT = 0x46
|
|
||||||
|
|
||||||
// 0x50 range - 'storage' and execution
|
// 0x50 range - 'storage' and execution
|
||||||
oPUSH = 0x50
|
oPUSH = 0x50
|
||||||
|
@ -115,12 +113,10 @@ var opCodeToString = map[OpCode]string{
|
||||||
oCALLVALUE: "CALLVALUE",
|
oCALLVALUE: "CALLVALUE",
|
||||||
oCALLDATA: "CALLDATA",
|
oCALLDATA: "CALLDATA",
|
||||||
oCALLDATASIZE: "CALLDATASIZE",
|
oCALLDATASIZE: "CALLDATASIZE",
|
||||||
oRETURNDATASIZE: "RETURNDATASIZE",
|
oGASPRICE: "TXGASPRICE",
|
||||||
oTXGASPRICE: "TXGASPRICE",
|
|
||||||
|
|
||||||
// 0x40 range - block operations
|
// 0x40 range - block operations
|
||||||
oPREVHASH: "PREVHASH",
|
oPREVHASH: "PREVHASH",
|
||||||
oPREVNONCE: "PREVNONCE",
|
|
||||||
oCOINBASE: "COINBASE",
|
oCOINBASE: "COINBASE",
|
||||||
oTIMESTAMP: "TIMESTAMP",
|
oTIMESTAMP: "TIMESTAMP",
|
||||||
oNUMBER: "NUMBER",
|
oNUMBER: "NUMBER",
|
||||||
|
@ -244,7 +240,11 @@ func (m *Memory) Get(offset, size int64) []byte {
|
||||||
func (m *Memory) Print() {
|
func (m *Memory) Print() {
|
||||||
fmt.Println("### MEM ###")
|
fmt.Println("### MEM ###")
|
||||||
if len(m.store) > 0 {
|
if len(m.store) > 0 {
|
||||||
fmt.Println(m.store)
|
addr := 0
|
||||||
|
for i := 0; i+32 < len(m.store); i += 32 {
|
||||||
|
fmt.Printf("%03d %v\n", addr, m.store[i:i+32])
|
||||||
|
addr++
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
fmt.Println("-- empty --")
|
fmt.Println("-- empty --")
|
||||||
}
|
}
|
||||||
|
|
|
@ -308,7 +308,7 @@ func (sm *StateManager) ProcessContract(contract *Contract, tx *Transaction, blo
|
||||||
caller := sm.procState.GetAccount(tx.Sender())
|
caller := sm.procState.GetAccount(tx.Sender())
|
||||||
closure := NewClosure(caller, contract, sm.procState, tx.Gas, tx.Value)
|
closure := NewClosure(caller, contract, sm.procState, tx.Gas, tx.Value)
|
||||||
vm := NewVm(sm.procState, RuntimeVars{
|
vm := NewVm(sm.procState, RuntimeVars{
|
||||||
origin: caller.Address,
|
origin: caller.Address(),
|
||||||
blockNumber: block.BlockInfo().Number,
|
blockNumber: block.BlockInfo().Number,
|
||||||
prevHash: block.PrevHash,
|
prevHash: block.PrevHash,
|
||||||
coinbase: block.Coinbase,
|
coinbase: block.Coinbase,
|
||||||
|
|
|
@ -40,7 +40,7 @@ var Pow256 = ethutil.BigPow(2, 256)
|
||||||
|
|
||||||
func (vm *Vm) RunClosure(closure *Closure) []byte {
|
func (vm *Vm) RunClosure(closure *Closure) []byte {
|
||||||
// If the amount of gas supplied is less equal to 0
|
// If the amount of gas supplied is less equal to 0
|
||||||
if closure.GetGas().Cmp(big.NewInt(0)) <= 0 {
|
if closure.Gas.Cmp(big.NewInt(0)) <= 0 {
|
||||||
// TODO Do something
|
// TODO Do something
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -73,7 +73,7 @@ func (vm *Vm) RunClosure(closure *Closure) []byte {
|
||||||
fee := new(big.Int)
|
fee := new(big.Int)
|
||||||
fee.Add(fee, big.NewInt(1000))
|
fee.Add(fee, big.NewInt(1000))
|
||||||
|
|
||||||
if closure.GetGas().Cmp(fee) < 0 {
|
if closure.Gas.Cmp(fee) < 0 {
|
||||||
return closure.Return(nil)
|
return closure.Return(nil)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -192,25 +192,37 @@ func (vm *Vm) RunClosure(closure *Closure) []byte {
|
||||||
|
|
||||||
// 0x30 range
|
// 0x30 range
|
||||||
case oADDRESS:
|
case oADDRESS:
|
||||||
|
stack.Push(ethutil.BigD(closure.Object().Address()))
|
||||||
case oBALANCE:
|
case oBALANCE:
|
||||||
|
stack.Push(closure.Value)
|
||||||
case oORIGIN:
|
case oORIGIN:
|
||||||
|
stack.Push(ethutil.BigD(vm.vars.origin))
|
||||||
case oCALLER:
|
case oCALLER:
|
||||||
|
stack.Push(ethutil.BigD(closure.Callee().Address()))
|
||||||
case oCALLVALUE:
|
case oCALLVALUE:
|
||||||
|
// FIXME: Original value of the call, not the current value
|
||||||
|
stack.Push(closure.Value)
|
||||||
case oCALLDATA:
|
case oCALLDATA:
|
||||||
offset := stack.Pop()
|
offset := stack.Pop()
|
||||||
mem.Set(offset.Int64(), int64(len(closure.Args)), closure.Args)
|
mem.Set(offset.Int64(), int64(len(closure.Args)), closure.Args)
|
||||||
case oCALLDATASIZE:
|
case oCALLDATASIZE:
|
||||||
case oRETURNDATASIZE:
|
stack.Push(big.NewInt(int64(len(closure.Args))))
|
||||||
case oTXGASPRICE:
|
case oGASPRICE:
|
||||||
|
// TODO
|
||||||
|
|
||||||
// 0x40 range
|
// 0x40 range
|
||||||
case oPREVHASH:
|
case oPREVHASH:
|
||||||
case oPREVNONCE:
|
stack.Push(ethutil.BigD(vm.vars.prevHash))
|
||||||
case oCOINBASE:
|
case oCOINBASE:
|
||||||
|
stack.Push(ethutil.BigD(vm.vars.coinbase))
|
||||||
case oTIMESTAMP:
|
case oTIMESTAMP:
|
||||||
|
stack.Push(big.NewInt(vm.vars.time))
|
||||||
case oNUMBER:
|
case oNUMBER:
|
||||||
|
stack.Push(big.NewInt(int64(vm.vars.blockNumber)))
|
||||||
case oDIFFICULTY:
|
case oDIFFICULTY:
|
||||||
|
stack.Push(vm.vars.diff)
|
||||||
case oGASLIMIT:
|
case oGASLIMIT:
|
||||||
|
// TODO
|
||||||
|
|
||||||
// 0x50 range
|
// 0x50 range
|
||||||
case oPUSH: // Push PC+1 on to the stack
|
case oPUSH: // Push PC+1 on to the stack
|
||||||
|
@ -218,8 +230,13 @@ func (vm *Vm) RunClosure(closure *Closure) []byte {
|
||||||
val := closure.GetMem(pc).BigInt()
|
val := closure.GetMem(pc).BigInt()
|
||||||
stack.Push(val)
|
stack.Push(val)
|
||||||
case oPOP:
|
case oPOP:
|
||||||
|
stack.Pop()
|
||||||
case oDUP:
|
case oDUP:
|
||||||
|
stack.Push(stack.Peek())
|
||||||
case oSWAP:
|
case oSWAP:
|
||||||
|
x, y := stack.Popn()
|
||||||
|
stack.Push(y)
|
||||||
|
stack.Push(x)
|
||||||
case oMLOAD:
|
case oMLOAD:
|
||||||
offset := stack.Pop()
|
offset := stack.Pop()
|
||||||
stack.Push(ethutil.BigD(mem.Get(offset.Int64(), 32)))
|
stack.Push(ethutil.BigD(mem.Get(offset.Int64(), 32)))
|
||||||
|
@ -228,7 +245,13 @@ func (vm *Vm) RunClosure(closure *Closure) []byte {
|
||||||
val, mStart := stack.Popn()
|
val, mStart := stack.Popn()
|
||||||
mem.Set(mStart.Int64(), 32, ethutil.BigToBytes(val, 256))
|
mem.Set(mStart.Int64(), 32, ethutil.BigToBytes(val, 256))
|
||||||
case oMSTORE8:
|
case oMSTORE8:
|
||||||
|
val, mStart := stack.Popn()
|
||||||
|
base.And(val, new(big.Int).SetInt64(0xff))
|
||||||
|
mem.Set(mStart.Int64(), 32, ethutil.BigToBytes(base, 256))
|
||||||
case oSLOAD:
|
case oSLOAD:
|
||||||
|
loc := stack.Pop()
|
||||||
|
val := closure.GetMem(loc.Int64())
|
||||||
|
stack.Push(val.BigInt())
|
||||||
case oSSTORE:
|
case oSSTORE:
|
||||||
case oJUMP:
|
case oJUMP:
|
||||||
case oJUMPI:
|
case oJUMPI:
|
||||||
|
|
|
@ -126,7 +126,6 @@ func TestRun3(t *testing.T) {
|
||||||
|
|
||||||
"PUSH", "64",
|
"PUSH", "64",
|
||||||
"PUSH", "0",
|
"PUSH", "0",
|
||||||
"LOG",
|
|
||||||
"RETURN",
|
"RETURN",
|
||||||
})
|
})
|
||||||
tx := NewTransaction(ContractAddr, ethutil.Big("100000000000000000000000000000000000000000000000000"), script)
|
tx := NewTransaction(ContractAddr, ethutil.Big("100000000000000000000000000000000000000000000000000"), script)
|
||||||
|
@ -159,7 +158,7 @@ func TestRun3(t *testing.T) {
|
||||||
callerClosure := NewClosure(account, MakeContract(callerTx, state), state, big.NewInt(1000000000), new(big.Int))
|
callerClosure := NewClosure(account, MakeContract(callerTx, state), state, big.NewInt(1000000000), new(big.Int))
|
||||||
|
|
||||||
vm := NewVm(state, RuntimeVars{
|
vm := NewVm(state, RuntimeVars{
|
||||||
origin: account.Address,
|
origin: account.Address(),
|
||||||
blockNumber: 1,
|
blockNumber: 1,
|
||||||
prevHash: ethutil.FromHex("5e20a0453cecd065ea59c37ac63e079ee08998b6045136a8ce6635c7912ec0b6"),
|
prevHash: ethutil.FromHex("5e20a0453cecd065ea59c37ac63e079ee08998b6045136a8ce6635c7912ec0b6"),
|
||||||
coinbase: ethutil.FromHex("2adc25665018aa1fe0e6bc666dac8fc2697ff9ba"),
|
coinbase: ethutil.FromHex("2adc25665018aa1fe0e6bc666dac8fc2697ff9ba"),
|
||||||
|
|
|
@ -40,17 +40,15 @@ var OpCodes = map[string]byte{
|
||||||
"CALLVALUE": 0x34,
|
"CALLVALUE": 0x34,
|
||||||
"CALLDATA": 0x35,
|
"CALLDATA": 0x35,
|
||||||
"CALLDATASIZE": 0x36,
|
"CALLDATASIZE": 0x36,
|
||||||
"RETURNDATASIZE": 0x37,
|
"GASPRICE": 0x38,
|
||||||
"TXGASPRICE": 0x38,
|
|
||||||
|
|
||||||
// 0x40 range - block operations
|
// 0x40 range - block operations
|
||||||
"PREVHASH": 0x40,
|
"PREVHASH": 0x40,
|
||||||
"PREVNONCE": 0x41,
|
"COINBASE": 0x41,
|
||||||
"COINBASE": 0x42,
|
"TIMESTAMP": 0x42,
|
||||||
"TIMESTAMP": 0x43,
|
"NUMBER": 0x43,
|
||||||
"NUMBER": 0x44,
|
"DIFFICULTY": 0x44,
|
||||||
"DIFFICULTY": 0x45,
|
"GASLIMIT": 0x45,
|
||||||
"GASLIMIT": 0x46,
|
|
||||||
|
|
||||||
// 0x50 range - 'storage' and execution
|
// 0x50 range - 'storage' and execution
|
||||||
"PUSH": 0x50,
|
"PUSH": 0x50,
|
||||||
|
|
Loading…
Reference in New Issue