aboutsummaryrefslogtreecommitdiffstats
path: root/ethchain/vm.go
diff options
context:
space:
mode:
Diffstat (limited to 'ethchain/vm.go')
-rw-r--r--ethchain/vm.go461
1 files changed, 190 insertions, 271 deletions
diff --git a/ethchain/vm.go b/ethchain/vm.go
index 7e119ac99..126592b25 100644
--- a/ethchain/vm.go
+++ b/ethchain/vm.go
@@ -1,12 +1,12 @@
package ethchain
import (
- "bytes"
- "fmt"
+ _ "bytes"
+ _ "fmt"
"github.com/ethereum/eth-go/ethutil"
- "github.com/obscuren/secp256k1-go"
+ _ "github.com/obscuren/secp256k1-go"
"log"
- "math"
+ _ "math"
"math/big"
)
@@ -18,122 +18,102 @@ type Vm struct {
mem map[string]*big.Int
vars RuntimeVars
+
+ state *State
}
type RuntimeVars struct {
- address []byte
+ origin []byte
blockNumber uint64
- sender []byte
prevHash []byte
coinbase []byte
time int64
diff *big.Int
- txValue *big.Int
txData []string
}
-func (vm *Vm) Process(contract *Contract, state *State, vars RuntimeVars) {
- vm.mem = make(map[string]*big.Int)
- vm.stack = NewStack()
+func NewVm(state *State, vars RuntimeVars) *Vm {
+ return &Vm{vars: vars, state: state}
+}
- addr := vars.address // tx.Hash()[12:]
- // Instruction pointer
- pc := 0
+var Pow256 = ethutil.BigPow(2, 256)
- if contract == nil {
- fmt.Println("Contract not found")
- return
+func (vm *Vm) RunClosure(closure *Closure) []byte {
+ // If the amount of gas supplied is less equal to 0
+ if closure.Gas.Cmp(big.NewInt(0)) <= 0 {
+ // TODO Do something
}
- Pow256 := ethutil.BigPow(2, 256)
+ // Memory for the current closure
+ mem := &Memory{}
+ // New stack (should this be shared?)
+ stack := NewStack()
+ // Instruction pointer
+ pc := big.NewInt(0)
+ // Current step count
+ step := 0
+ // The base for all big integer arithmetic
+ base := new(big.Int)
if ethutil.Config.Debug {
ethutil.Config.Log.Debugf("# op\n")
}
- stepcount := 0
- totalFee := new(big.Int)
-
-out:
for {
- stepcount++
- // The base big int for all calculations. Use this for any results.
- base := new(big.Int)
- val := contract.GetMem(pc)
- //fmt.Printf("%x = %d, %v %x\n", r, len(r), v, nb)
+ step++
+ // Get the memory location of pc
+ val := closure.GetMem(pc)
+ // Get the opcode (it must be an opcode!)
op := OpCode(val.Uint())
-
- var fee *big.Int = new(big.Int)
- var fee2 *big.Int = new(big.Int)
- if stepcount > 16 {
- fee.Add(fee, StepFee)
- }
-
- // Calculate the fees
- switch op {
- case oSSTORE:
- y, x := vm.stack.Peekn()
- val := contract.Addr(ethutil.BigToBytes(x, 256))
- if val.IsEmpty() && len(y.Bytes()) > 0 {
- fee2.Add(DataFee, StoreFee)
- } else {
- fee2.Sub(DataFee, StoreFee)
- }
- case oSLOAD:
- fee.Add(fee, StoreFee)
- case oEXTRO, oBALANCE:
- fee.Add(fee, ExtroFee)
- case oSHA256, oRIPEMD160, oECMUL, oECADD, oECSIGN, oECRECOVER, oECVALID:
- fee.Add(fee, CryptoFee)
- case oMKTX:
- fee.Add(fee, ContractFee)
+ if ethutil.Config.Debug {
+ ethutil.Config.Log.Debugf("%-3d %-4s", pc, op.String())
}
- tf := new(big.Int).Add(fee, fee2)
- if contract.Amount.Cmp(tf) < 0 {
- fmt.Println("Insufficient fees to continue running the contract", tf, contract.Amount)
- break
- }
- // Add the fee to the total fee. It's subtracted when we're done looping
- totalFee.Add(totalFee, tf)
+ // TODO Get each instruction cost properly
+ fee := new(big.Int)
+ fee.Add(fee, big.NewInt(1000))
- if ethutil.Config.Debug {
- ethutil.Config.Log.Debugf("%-3d %-4s", pc, op.String())
+ if closure.Gas.Cmp(fee) < 0 {
+ return closure.Return(nil)
}
switch op {
- case oSTOP:
- fmt.Println("")
- break out
+ case oLOG:
+ stack.Print()
+ mem.Print()
+ case oSTOP: // Stop the closure
+ return closure.Return(nil)
+
+ // 0x20 range
case oADD:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// (x + y) % 2 ** 256
base.Add(x, y)
base.Mod(base, Pow256)
// Pop result back on the stack
- vm.stack.Push(base)
+ stack.Push(base)
case oSUB:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// (x - y) % 2 ** 256
base.Sub(x, y)
base.Mod(base, Pow256)
// Pop result back on the stack
- vm.stack.Push(base)
+ stack.Push(base)
case oMUL:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// (x * y) % 2 ** 256
base.Mul(x, y)
base.Mod(base, Pow256)
// Pop result back on the stack
- vm.stack.Push(base)
+ stack.Push(base)
case oDIV:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// floor(x / y)
base.Div(x, y)
// Pop result back on the stack
- vm.stack.Push(base)
+ stack.Push(base)
case oSDIV:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// n > 2**255
if x.Cmp(Pow256) > 0 {
x.Sub(Pow256, x)
@@ -147,13 +127,13 @@ out:
z.Sub(Pow256, z)
}
// Push result on to the stack
- vm.stack.Push(z)
+ stack.Push(z)
case oMOD:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
base.Mod(x, y)
- vm.stack.Push(base)
+ stack.Push(base)
case oSMOD:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// n > 2**255
if x.Cmp(Pow256) > 0 {
x.Sub(Pow256, x)
@@ -167,250 +147,189 @@ out:
z.Sub(Pow256, z)
}
// Push result on to the stack
- vm.stack.Push(z)
+ stack.Push(z)
case oEXP:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
base.Exp(x, y, Pow256)
- vm.stack.Push(base)
+ stack.Push(base)
case oNEG:
- base.Sub(Pow256, vm.stack.Pop())
- vm.stack.Push(base)
+ base.Sub(Pow256, stack.Pop())
+ stack.Push(base)
case oLT:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// x < y
if x.Cmp(y) < 0 {
- vm.stack.Push(ethutil.BigTrue)
- } else {
- vm.stack.Push(ethutil.BigFalse)
- }
- case oLE:
- x, y := vm.stack.Popn()
- // x <= y
- if x.Cmp(y) < 1 {
- vm.stack.Push(ethutil.BigTrue)
+ stack.Push(ethutil.BigTrue)
} else {
- vm.stack.Push(ethutil.BigFalse)
+ stack.Push(ethutil.BigFalse)
}
case oGT:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// x > y
if x.Cmp(y) > 0 {
- vm.stack.Push(ethutil.BigTrue)
- } else {
- vm.stack.Push(ethutil.BigFalse)
- }
- case oGE:
- x, y := vm.stack.Popn()
- // x >= y
- if x.Cmp(y) > -1 {
- vm.stack.Push(ethutil.BigTrue)
+ stack.Push(ethutil.BigTrue)
} else {
- vm.stack.Push(ethutil.BigFalse)
+ stack.Push(ethutil.BigFalse)
}
case oNOT:
- x, y := vm.stack.Popn()
+ x, y := stack.Popn()
// x != y
if x.Cmp(y) != 0 {
- vm.stack.Push(ethutil.BigTrue)
+ stack.Push(ethutil.BigTrue)
} else {
- vm.stack.Push(ethutil.BigFalse)
+ stack.Push(ethutil.BigFalse)
}
- case oMYADDRESS:
- vm.stack.Push(ethutil.BigD(addr))
- case oTXSENDER:
- vm.stack.Push(ethutil.BigD(vars.sender))
- case oTXVALUE:
- vm.stack.Push(vars.txValue)
- case oTXDATAN:
- vm.stack.Push(big.NewInt(int64(len(vars.txData))))
- case oTXDATA:
- v := vm.stack.Pop()
- // v >= len(data)
- if v.Cmp(big.NewInt(int64(len(vars.txData)))) >= 0 {
- vm.stack.Push(ethutil.Big("0"))
- } else {
- vm.stack.Push(ethutil.Big(vars.txData[v.Uint64()]))
- }
- case oBLK_PREVHASH:
- vm.stack.Push(ethutil.BigD(vars.prevHash))
- case oBLK_COINBASE:
- vm.stack.Push(ethutil.BigD(vars.coinbase))
- case oBLK_TIMESTAMP:
- vm.stack.Push(big.NewInt(vars.time))
- case oBLK_NUMBER:
- vm.stack.Push(big.NewInt(int64(vars.blockNumber)))
- case oBLK_DIFFICULTY:
- vm.stack.Push(vars.diff)
- case oBASEFEE:
- // e = 10^21
- e := big.NewInt(0).Exp(big.NewInt(10), big.NewInt(21), big.NewInt(0))
- d := new(big.Rat)
- d.SetInt(vars.diff)
- c := new(big.Rat)
- c.SetFloat64(0.5)
- // d = diff / 0.5
- d.Quo(d, c)
- // base = floor(d)
- base.Div(d.Num(), d.Denom())
- x := new(big.Int)
- x.Div(e, base)
+ // 0x10 range
+ case oAND:
+ case oOR:
+ case oXOR:
+ case oBYTE:
- // x = floor(10^21 / floor(diff^0.5))
- vm.stack.Push(x)
- case oSHA256, oSHA3, oRIPEMD160:
- // This is probably save
- // ceil(pop / 32)
- length := int(math.Ceil(float64(vm.stack.Pop().Uint64()) / 32.0))
- // New buffer which will contain the concatenated popped items
- data := new(bytes.Buffer)
- for i := 0; i < length; i++ {
- // Encode the number to bytes and have it 32bytes long
- num := ethutil.NumberToBytes(vm.stack.Pop().Bytes(), 256)
- data.WriteString(string(num))
- }
+ // 0x20 range
+ case oSHA3:
- if op == oSHA256 {
- vm.stack.Push(base.SetBytes(ethutil.Sha256Bin(data.Bytes())))
- } else if op == oSHA3 {
- vm.stack.Push(base.SetBytes(ethutil.Sha3Bin(data.Bytes())))
- } else {
- vm.stack.Push(base.SetBytes(ethutil.Ripemd160(data.Bytes())))
- }
- case oECMUL:
- y := vm.stack.Pop()
- x := vm.stack.Pop()
- //n := vm.stack.Pop()
+ // 0x30 range
+ case oADDRESS:
+ stack.Push(ethutil.BigD(closure.Object().Address()))
+ case oBALANCE:
+ stack.Push(closure.Value)
+ case oORIGIN:
+ stack.Push(ethutil.BigD(vm.vars.origin))
+ case oCALLER:
+ stack.Push(ethutil.BigD(closure.Callee().Address()))
+ case oCALLVALUE:
+ // FIXME: Original value of the call, not the current value
+ stack.Push(closure.Value)
+ case oCALLDATA:
+ offset := stack.Pop()
+ mem.Set(offset.Int64(), int64(len(closure.Args)), closure.Args)
+ case oCALLDATASIZE:
+ stack.Push(big.NewInt(int64(len(closure.Args))))
+ case oGASPRICE:
+ // TODO
- //if ethutil.Big(x).Cmp(ethutil.Big(y)) {
- data := new(bytes.Buffer)
- data.WriteString(x.String())
- data.WriteString(y.String())
- if secp256k1.VerifyPubkeyValidity(data.Bytes()) == 1 {
- // TODO
- } else {
- // Invalid, push infinity
- vm.stack.Push(ethutil.Big("0"))
- vm.stack.Push(ethutil.Big("0"))
- }
- //} else {
- // // Invalid, push infinity
- // vm.stack.Push("0")
- // vm.stack.Push("0")
- //}
+ // 0x40 range
+ case oPREVHASH:
+ stack.Push(ethutil.BigD(vm.vars.prevHash))
+ case oCOINBASE:
+ stack.Push(ethutil.BigD(vm.vars.coinbase))
+ case oTIMESTAMP:
+ stack.Push(big.NewInt(vm.vars.time))
+ case oNUMBER:
+ stack.Push(big.NewInt(int64(vm.vars.blockNumber)))
+ case oDIFFICULTY:
+ stack.Push(vm.vars.diff)
+ case oGASLIMIT:
+ // TODO
- case oECADD:
- case oECSIGN:
- case oECRECOVER:
- case oECVALID:
- case oPUSH:
- pc++
- vm.stack.Push(contract.GetMem(pc).BigInt())
+ // 0x50 range
+ case oPUSH: // Push PC+1 on to the stack
+ pc.Add(pc, ethutil.Big1)
+
+ val := closure.GetMem(pc).BigInt()
+ stack.Push(val)
case oPOP:
- // Pop current value of the stack
- vm.stack.Pop()
+ stack.Pop()
case oDUP:
- // Dup top stack
- x := vm.stack.Pop()
- vm.stack.Push(x)
- vm.stack.Push(x)
+ stack.Push(stack.Peek())
case oSWAP:
- // Swap two top most values
- x, y := vm.stack.Popn()
- vm.stack.Push(y)
- vm.stack.Push(x)
+ x, y := stack.Popn()
+ stack.Push(y)
+ stack.Push(x)
case oMLOAD:
- x := vm.stack.Pop()
- vm.stack.Push(vm.mem[x.String()])
- case oMSTORE:
- x, y := vm.stack.Popn()
- vm.mem[x.String()] = y
+ offset := stack.Pop()
+ stack.Push(ethutil.BigD(mem.Get(offset.Int64(), 32)))
+ case oMSTORE: // Store the value at stack top-1 in to memory at location stack top
+ // Pop value of the stack
+ val, mStart := stack.Popn()
+ mem.Set(mStart.Int64(), 32, ethutil.BigToBytes(val, 256))
+ 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:
- // Load the value in storage and push it on the stack
- x := vm.stack.Pop()
- // decode the object as a big integer
- decoder := contract.Addr(x.Bytes())
- if !decoder.IsNil() {
- vm.stack.Push(decoder.BigInt())
- } else {
- vm.stack.Push(ethutil.BigFalse)
- }
+ loc := stack.Pop()
+ val := closure.GetMem(loc)
+ stack.Push(val.BigInt())
case oSSTORE:
- // Store Y at index X
- y, x := vm.stack.Popn()
- addr := ethutil.BigToBytes(x, 256)
- fmt.Printf(" => %x (%v) @ %v", y.Bytes(), y, ethutil.BigD(addr))
- contract.SetAddr(addr, y)
- //contract.State().Update(string(idx), string(y))
- case oJMP:
- x := int(vm.stack.Pop().Uint64())
- // Set pc to x - 1 (minus one so the incrementing at the end won't effect it)
- pc = x
- pc--
- case oJMPI:
- x := vm.stack.Pop()
- // Set pc to x if it's non zero
- if x.Cmp(ethutil.BigFalse) != 0 {
- pc = int(x.Uint64())
- pc--
+ val, loc := stack.Popn()
+ closure.SetMem(loc, ethutil.NewValue(val))
+ case oJUMP:
+ pc = stack.Pop()
+ case oJUMPI:
+ pos, cond := stack.Popn()
+ if cond.Cmp(big.NewInt(0)) > 0 {
+ pc = pos
}
- case oIND:
- vm.stack.Push(big.NewInt(int64(pc)))
- case oEXTRO:
- memAddr := vm.stack.Pop()
- contractAddr := vm.stack.Pop().Bytes()
+ case oPC:
+ stack.Push(pc)
+ case oMSIZE:
+ stack.Push(big.NewInt(int64(mem.Len())))
+ // 0x60 range
+ case oCALL:
+ // Pop return size and offset
+ retSize, retOffset := stack.Popn()
+ // Pop input size and offset
+ inSize, inOffset := stack.Popn()
+ // Get the arguments from the memory
+ args := mem.Get(inOffset.Int64(), inSize.Int64())
+ // Pop gas and value of the stack.
+ gas, value := stack.Popn()
+ // Closure addr
+ addr := stack.Pop()
+ // Fetch the contract which will serve as the closure body
+ contract := vm.state.GetContract(addr.Bytes())
+ // Create a new callable closure
+ closure := NewClosure(closure, contract, vm.state, gas, value)
+ // Executer the closure and get the return value (if any)
+ ret := closure.Call(vm, args)
- // Push the contract's memory on to the stack
- vm.stack.Push(contractMemory(state, contractAddr, memAddr))
- case oBALANCE:
- // Pushes the balance of the popped value on to the stack
- account := state.GetAccount(vm.stack.Pop().Bytes())
- vm.stack.Push(account.Amount)
- case oMKTX:
- addr, value := vm.stack.Popn()
- from, length := vm.stack.Popn()
+ mem.Set(retOffset.Int64(), retSize.Int64(), ret)
+ case oRETURN:
+ size, offset := stack.Popn()
+ ret := mem.Get(offset.Int64(), size.Int64())
- makeInlineTx(addr.Bytes(), value, from, length, contract, state)
+ return closure.Return(ret)
case oSUICIDE:
- recAddr := vm.stack.Pop().Bytes()
- // Purge all memory
- deletedMemory := contract.state.Purge()
- // Add refunds to the pop'ed address
- refund := new(big.Int).Mul(StoreFee, big.NewInt(int64(deletedMemory)))
- account := state.GetAccount(recAddr)
- account.Amount.Add(account.Amount, refund)
- // Update the refunding address
- state.UpdateAccount(recAddr, account)
- // Delete the contract
- state.trie.Update(string(addr), "")
+ /*
+ recAddr := stack.Pop().Bytes()
+ // Purge all memory
+ deletedMemory := contract.state.Purge()
+ // Add refunds to the pop'ed address
+ refund := new(big.Int).Mul(StoreFee, big.NewInt(int64(deletedMemory)))
+ account := state.GetAccount(recAddr)
+ account.Amount.Add(account.Amount, refund)
+ // Update the refunding address
+ state.UpdateAccount(recAddr, account)
+ // Delete the contract
+ state.trie.Update(string(addr), "")
- ethutil.Config.Log.Debugf("(%d) => %x\n", deletedMemory, recAddr)
- break out
+ ethutil.Config.Log.Debugf("(%d) => %x\n", deletedMemory, recAddr)
+ break out
+ */
default:
- fmt.Printf("Invalid OPCODE: %x\n", op)
+ ethutil.Config.Log.Debugln("Invalid opcode", op)
}
- ethutil.Config.Log.Debugln("")
- //vm.stack.Print()
- pc++
- }
- state.UpdateContract(addr, contract)
+ pc.Add(pc, ethutil.Big1)
+ }
}
func makeInlineTx(addr []byte, value, from, length *big.Int, contract *Contract, state *State) {
ethutil.Config.Log.Debugf(" => creating inline tx %x %v %v %v", addr, value, from, length)
- j := 0
+ j := int64(0)
dataItems := make([]string, int(length.Uint64()))
- for i := from.Uint64(); i < length.Uint64(); i++ {
- dataItems[j] = contract.GetMem(j).Str()
+ for i := from.Int64(); i < length.Int64(); i++ {
+ dataItems[j] = contract.GetMem(big.NewInt(j)).Str()
j++
}
tx := NewTransaction(addr, value, dataItems)
if tx.IsContract() {
contract := MakeContract(tx, state)
- state.UpdateContract(tx.Hash()[12:], contract)
+ state.UpdateContract(contract)
} else {
account := state.GetAccount(tx.Recipient)
account.Amount.Add(account.Amount, tx.Value)