以太坊源码解读(20)EVM 转账或调用合约
在了解合约调用之前,我们需要知道调用合约的本质是什么。在我们创建合约的时候,由run函数初始化的智能合约code(ret)储存在stateDB中。也就是说在内存中并没有Contract这个对象,而只是存在智能合约code。那我们如何调用合约呢?本质上,调用合约实际上是从合约账户中取出合约代码,然后NewContract()创建出一个临时的contract对象(如下图),然后执行contract的SetCallCode()或其他方法,确定智能合约的执行环境,然后执行run()函数,返回执行后的代码。
知道了这个过程,我们再来看看下面这个函数,智能合约的调用或普通交易——Call()。
Call()的主要功能是执行一笔交易,具体的步骤如下:
1、交易执行前的检查:深度判断和余额状况; 2、如果世界状态中不存在这个账号,则创建这个账号; 3、进行转账; 4、创建一个待执行的合约对象,并执行; 5、处理交易执行返回值
一、Call()的参数
func (evm *EVM) Call(caller ContractRef, addr common.Address, input []byte, gas uint64, value *big.Int)
caller:转出方地址; addr:转入方地址,如果是调用智能合约,那就是智能合约的地址; input:调用函数的参数 gas:当前交易的剩余gas; value:转账额度;
二、Call的执行步骤
首先,执行前检查。
// 如果不允许递归深度大于0,直接退出 if evm.vmConfig.NoRecursion && evm.depth > 0 { return nil, gas, nil } // 如果递归深度大于1024,直接退出 if evm.depth > int(params.CallCreateDepth) { return nil, gas, ErrDepth } // 如果余额不够,直接退出 if !evm.Context.CanTransfer(evm.StateDB, caller.Address(), value) { return nil, gas, ErrInsufficientBalance }
然后判断世界状态中是否存在这个账号,如果不存在则创建账号;
var ( to = AccountRef(addr) snapshot = evm.StateDB.Snapshot() ) if !evm.StateDB.Exist(addr) { precompiles := PrecompiledContractsHomestead if evm.ChainConfig().IsByzantium(evm.BlockNumber) { precompiles = PrecompiledContractsByzantium } if precompiles[addr] == nil && evm.ChainConfig().IsEIP158(evm.BlockNumber) && value.Sign() == 0 { // Calling a non existing account, don't do anything, but ping the tracer if evm.vmConfig.Debug && evm.depth == 0 { evm.vmConfig.Tracer.CaptureStart(caller.Address(), addr, false, input, gas, value) evm.vmConfig.Tracer.CaptureEnd(ret, 0, 0, nil) } return nil, gas, nil } evm.StateDB.CreateAccount(addr) }
第三步进行转账。
evm.Transfer(evm.StateDB, caller.Address(), to.Address(), value)
第四步创建一个待执行的合约对象,并执行。
contract := NewContract(caller, to, value, gas) contract.SetCallCode(&addr, evm.StateDB.GetCodeHash(addr), evm.StateDB.GetCode(addr)) // Even if the account has no code, we need to continue because it might be a precompile start := time.Now() // Capture the tracer start/end events in debug mode if evm.vmConfig.Debug && evm.depth == 0 { evm.vmConfig.Tracer.CaptureStart(caller.Address(), addr, false, input, gas, value) defer func() { // Lazy evaluation of the parameters evm.vmConfig.Tracer.CaptureEnd(ret, gas-contract.Gas, time.Since(start), err) }() } ret, err = run(evm, contract, input, false)
这里create()不同的是构建新contract对象的时候,create()调用了SetCodeOptionalHash(&address, codeAndHash),而这里我们调用的是SetCallCode(&addr, evm.StateDB.GetCodeHash(addr), evm.StateDB.GetCode(addr))。create()中构建的contract对象中的Code是来自原始transaction中的Payload,而在Call()构建的contract对象中的Code则是create()中初始化智能合约即执行run()之后返回的ret,这两者在结构上是有区别的。
当我们写完智能合约通过solidity编译器生成合约代码时,最前面是一段合约部署代码,用来引导合约的部署,transaction中的Payload就包含这一段,它可以引导执行合约构造函数。当我们部署完成后,在智能合约地址中储存的Code却不包含合约部署代码部分,而是只有后面的部分,即ret。所以当我们调用智能合约的时候,构建的contract的code只有ret,直接从合约函数引导的代码开始执行。
三、智能合约的连环调用
我们说过智能合约EVM的递归调用深度为1024,也就是指通过一个合约调用另一个合约,像这样的调用可以递归1024次。
为什么说是“递归”?因为从一个智能合约调用另一个智能合约,比如通过Call()方法,都要重新构建contract实例,然后执行run()。而run()的执行是通过EVMinterpreter.Run()进行的。而在EVMInterpreter结构体中又传入了*EVM的地址,然后执行了evm.depth++。所以实际上每一次调用都是在同一个EVM内进行的。
连环调用的方法有下面几种:1、Call();2、CallCode();3、DelegateCall()。
Call()
to = AccountRef(addr) contract := NewContract(caller, to, value, gas) // 假设有外部账户A,合约账户B和合约账户C A Call B ——> ContractB CallerAddress: A Caller: A self: B B Call C ——> ContractC CallerAddress: B Caller: B self: C
CallCode()
to = AccountRef(caller.Address()) contract := NewContract(caller, to, value, gas) // 假设有外部账户A,合约账户B和合约账户C A Call B ——> ContractB CallerAddress: A Caller: A self: B B Callcode C ——> ContractC CallerAddress: B Caller: B self: B
delegateCall()
to = AccountRef(caller.Address()) contract := NewContract(caller, to, nil, gas).AsDelegate() func (c *Contract) AsDelegate() *Contract { parent := c.caller.(*Contract) c.CallerAddress = parent.CallerAddress c.value = parent.value return c } // 假设有外部账户A,合约账户B和合约账户C A Call B ——> ContractB CallerAddress: A Caller: A self: B B DelegateCall C ——> ContractC CallerAddress: A Caller: B self: B
从代码上看,这三者的主要区别就是这一点,主要就是在contract结构中的callerAddress、caller和self这三个的值不同。
如果外部账户A的某个操作通过Call方法调用B合约,而B合约又通过Call方法调用了C合约,那么最后实际上修改的是合约C账户的值;
如果外部账户A的某个操作通过Call方法调用B合约,而B合约通过CallCode方法调用了C合约,那么B只是调用了C中的函数代码,而最终改变的还是合约B账户的值。
DelegateCall其实跟CallCode方法的目的类似,都是只调用指定地址(合约C)的代码,而操作B的值。只不过它明确了CallerAddress是来自A,而不是B。所以这两种方法都可以用来实现动态库:即你调用我的函数和方法改动的都是你自己的数据)。
下一章:以太坊源码解读(21)EVM 解释器
之前我们说到EVM解释器是面对Contract对象的,不论是Contract的创建还是调用,都会通过run()函数来调用Interpreter的Run()方法。该方法初始化执行过程中所需要的一些变量,然 ...