以太坊源码解读(18)以太坊交易执行流程

之前分析挖矿模块,miner从TxPool拿来的交易,交给worker对象。后者要调用commitTransaction在本地执行交易,生成receipt,更改世界状态,打包成挖矿的block最后递交给engine进行挖矿。而这一节我们关注的是在commitTransaction中,如何在本地执行交易。

func (w *worker) commitTransaction(tx *types.Transaction, coinbase common.Address) ([]*types.Log, error) {
	snap := w.current.state.Snapshot()

	receipt, _, err := core.ApplyTransaction(w.config, w.chain, &coinbase, w.current.gasPool, w.current.state, w.current.header, tx, &w.current.header.GasUsed, vm.Config{})
	if err != nil {
		w.current.state.RevertToSnapshot(snap)
		return nil, err
	}
	w.current.txs = append(w.current.txs, tx)
	w.current.receipts = append(w.current.receipts, receipt)

	return receipt.Logs, nil
}

core.ApplyTransaction就是执行交易的入口,而交易的执行就离不开大名鼎鼎的以太坊虚拟机。

一、ApplyTransaction函数

该函数的调用有两种情况: 1、是在将区块插入区块链前需要验证区块合法性bc.insertChain ——> bc.processor.Process  ——> stateProcessor.Process ——> ApplyTransaction 2、是worker挖矿过程中执行交易时Worker.commitTransaction ——> ApplyTransaction

主要功能是:将交易转化成Message,创建EVM对象,调用ApplyMessage执行交易,生成日志对象; 1、将交易转换成Message; 2、初始化一个EVM的执行环境; 3、执行交易,改变stateDB世界状态,然后生成收据

func ApplyTransaction(config *params.ChainConfig, bc ChainContext, author *common.Address, gp *GasPool, statedb *state.StateDB, header *types.Header, tx *types.Transaction, usedGas *uint64, cfg vm.Config) (*types.Receipt, uint64, error) {
        // 转换一个Message对象
	msg, err := tx.AsMessage(types.MakeSigner(config, header.Number))
	if err != nil {
		return nil, 0, err
	}
	// 使用message初始化一个执行上下文
	context := NewEVMContext(msg, header, bc, author)

	// 使用上下文初始化一个EVM执行环境
	vmenv := vm.NewEVM(context, statedb, config, cfg)

	// 执行交易,返回使用的gas数量
	_, gas, failed, err := ApplyMessage(vmenv, msg, gp)
	if err != nil {
		return nil, 0, err
	}
	// 更改stateDB世界状态
	var root []byte
        // 如果是拜占庭硬分叉,清理世界状态
	if config.IsByzantium(header.Number) {
		statedb.Finalise(true)
	} else {
        // 否则计算状态树根,用于产生收据
		root = statedb.IntermediateRoot(config.IsEIP158(header.Number)).Bytes()
	}
        // 增加header中的usedGas
	*usedGas += gas

	// 用root和usedGas产生收据,即收据里包含了该交易执行时的世界状态,这样方便进行校验
	receipt := types.NewReceipt(root, failed, *usedGas)
	receipt.TxHash = tx.Hash()
	receipt.GasUsed = gas
	// 如果交易创建了一个合约,要把合约地址放在收据里
	if msg.To() == nil {
		receipt.ContractAddress = crypto.CreateAddress(vmenv.Context.Origin, tx.Nonce())
	}
	// Set the receipt logs and create a bloom for filtering
	receipt.Logs = statedb.GetLogs(tx.Hash())
	receipt.Bloom = types.CreateBloom(types.Receipts{receipt})

	return receipt, gas, err
}

二、StateTransition.TransitionDb

交易工作环境(StateTransition)的数据结构如下:

type StateTransition struct {
	gp         *GasPool   // 区块工作环境中的gas剩余额度,就是header中的gasLimit
	msg        Message    // 交易转化的message
	gas        uint64     // 交易的gas余额,最开始等于initialGas,随着交易执行会递减
	gasPrice   *big.Int   
	initialGas uint64     // 初始gas,等于交易的gasLimit
	value      *big.Int   // 交易转账额度
	data       []byte     // 交易的input,如果是合约创建,data就是合约代码
	state      vm.StateDB // 状态树
	evm        *vm.EVM    // evm对象
}

这里gp是整个区块所有交易可用的gas,其实就是来自于header的gaslimit,而header的gasLimit是通过父区块的gasUsed推算出来的。initialGas是交易的gasLimit,gas是余额(等于initialGas减去交易usedGas)。

在ApplyMessage中,首先新建一个交易工作环境,然后紧接着调用TransitionDb方法:

func ApplyMessage(evm *vm.EVM, msg Message, gp *GasPool) ([]byte, uint64, bool, error) {
	return NewStateTransition(evm, msg, gp).TransitionDb()
}

TransitionDb()的主要功能是初始化交易工作环境,执行交易,然后处理交易执行前后的gas增减。

1、预先检查nonce和gas值,初始化交易工作环境的gas初始值; 2、计算并扣除固定gas消耗; 3、调用evm创建或执行交易; 4、奖励旷工;

func (st *StateTransition) TransitionDb() (ret []byte, usedGas uint64, failed bool, err error) {
        // 先检查交易的nonce和当前状态nonce是否一致,然后调用buyGas;
        // buyGas是要从区块的gasPool中取出一定的gas用于执行交易
	if err = st.preCheck(); err != nil {
		return
	}
	msg := st.msg
	sender := vm.AccountRef(msg.From())
	homestead := st.evm.ChainConfig().IsHomestead(st.evm.BlockNumber)
	contractCreation := msg.To() == nil

	// 计算并扣除固定的gas消耗
        // 固定的gas消耗(21000) + 非0值gas + 0值gas
	gas, err := IntrinsicGas(st.data, contractCreation, homestead)
	if err != nil {
		return nil, 0, false, err
	}
        // 更新剩余的gas值
	if err = st.useGas(gas); err != nil {
		return nil, 0, false, err
	}

	var (
		evm = st.evm
		vmerr error
	)
	if contractCreation {
                // 如果是合约创建建议,调用evm.Create
		ret, _, st.gas, vmerr = evm.Create(sender, st.data, st.gas, st.value)
	} else {
		// 如果是交易或合约调用,则先设置交易发送方的nonce值+1
		st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
                // 调用evm.Call执行交易
		ret, st.gas, vmerr = evm.Call(sender, st.to(), st.data, st.gas, st.value)
	}
	if vmerr != nil {
		log.Debug("VM returned with error", "err", vmerr)
		// evm的错误不影响共识,当且仅当vmerr是ErrInsufficientBalance时,return的gasUsed等于0,此时不影响账户状态;
                // 否则,即使交易执行失败,gas还是会被扣除
		if vmerr == vm.ErrInsufficientBalance {
			return nil, 0, false, vmerr
		}
	}
        // 返回余额给发起方,它同时调整了StateTransition中gasPool的余额
	st.refundGas()
        // 奖励矿工,gas消耗
	st.state.AddBalance(st.evm.Coinbase, new(big.Int).Mul(new(big.Int).SetUint64(st.gasUsed()), st.gasPrice))

	return ret, st.gasUsed(), vmerr != nil, err
}

交易执行之前,要调用bugGas()将initialGas和gas都设置为交易的gasLimit,并从总的gaspool里减去预支的gas(即gasLimit),交易发起者的账户也要减去相应的价值。

func (st *StateTransition) buyGas() error {
	mgval := new(big.Int).Mul(new(big.Int).SetUint64(st.msg.Gas()), st.gasPrice)
	if st.state.GetBalance(st.msg.From()).Cmp(mgval) < 0 {
		return errInsufficientBalanceForGas
	}
	if err := st.gp.SubGas(st.msg.Gas()); err != nil {
		return err
	}
	st.gas += st.msg.Gas()

	st.initialGas = st.msg.Gas()
	st.state.SubBalance(st.msg.From(), mgval)
	return nil
}

这里实际上是preCheck中的内容,后面计算交易的具体gas后,还要调整gas的值:

func (st *StateTransition) useGas(amount uint64) error {
	if st.gas < amount {
		return vm.ErrOutOfGas
	}
	st.gas -= amount

	return nil
}

最后,如果实际使用的gas超过了gasLimit,即initialGas,就返回错误;如果顺利执行,则将剩余的gas返回原来的账户,同时可能还有一笔奖励给原账户(st.gas + refund)。最后,将st.gas即交易执行后的gas余额返回给交易执行环境(st.gp)。

func (st *StateTransition) refundGas() {
	// Apply refund counter, capped to half of the used gas.
	refund := st.gasUsed() / 2
	if refund > st.state.GetRefund() {
		refund = st.state.GetRefund()
	}
	st.gas += refund

	// Return ETH for remaining gas, exchanged at the original rate.
	remaining := new(big.Int).Mul(new(big.Int).SetUint64(st.gas), st.gasPrice)
	st.state.AddBalance(st.msg.From(), remaining)

	// Also return remaining gas to the block gas counter so it is
	// available for the next transaction.
	st.gp.AddGas(st.gas)
}

执行过程中如果是合约创建交易,则调用evm.Create();如果是普通交易或调用智能合约,则调用evm.Call()方法。

if contractCreation {
    // 如果是合约创建建议,调用evm.Create
    ret, _, st.gas, vmerr = evm.Create(sender, st.data, st.gas, st.value)
} else {
    // 如果是交易或合约调用,则先设置交易发送方的nonce值+1
    st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
    // 调用evm.Call执行交易
    ret, st.gas, vmerr = evm.Call(sender, st.to(), st.data, st.gas, st.value)

下一章:以太坊源码解读(19)EVM 创建智能合约

我们要知道,evm解释器的执行上下文是stateTransition,是交易,但evm的服务对象是智能合约。智能合约与evm的解释器有紧密的联系,所以我们这一节先从智能合约的创建和执行开始学习。 ...