矿工出块分析+阅读收获
首先声明一下看的版本,geth1.10.10
矿工出块流程分析
简单说明一下数据流向: 收到交易事件:newWorkLoop->基于父块生成几个新的密封任务:mainLoop的commitNewWork->运行任何事务后的状态修改,并组装最后一个块:commitNewWork的commit->进行挖矿taskLoop的Seal->数据上链resultLoop的WriteBlockWithState
具体函数内容说明: commitNewWork(基于父块生成几个新的密封任务): 获得当前最新区块、时间、区块号。根据这些设置并生成区块头。整理区块头,检查coinbase等。进行叔块处理。在块未执行完毕的情况下,预先密封并用所有可用的挂起事务填充该块。将挂起的事务拆分为本地事务和远程事务。对本地事务,执行NewTransactionsByPriceAndNonce后,执行commitTransactions。然后对远程事务重新来一遍。最后,执行commit提交结果。
commit(运行任何事务后状态修改,组装最后一个块,如果共识引擎正在运行,则提交新工作): 首先FinalizeAndAssemble计算区块奖励,并组装并返回最终块进行密封。然后在挖矿运行的情况下,执行fullTaskHook(方法在推送完全密封任务之前调用),然后传递消息给taskCh,此时激活taskLoop。然后执行updateSnapshot更新快照更新挂起的快照块和状态。
taskLoop (用于从生成器获取密封任务并将其推送到一致性引擎)进行SealHash,激活resultLoop的resultCh,进行区块结果上链。
阅读收获
看了这部分源码之后的一个感受就是以太坊对于go channel的运用非常的频繁和有趣。里面大量的代码都使用了一种结构:
for{
select {
case task := <-w.taskCh:
......
case <-w.exitCh:
}
}
当然,我觉得最有收获的还是收获一个知识点:
select {
case w.taskCh <- &task{......}:
case <-w.exitCh:
}
```
|