日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

EOS生产区块:解析插件producer_plugin

發(fā)布時間:2024/4/15 编程问答 60 豆豆
生活随笔 收集整理的這篇文章主要介紹了 EOS生产区块:解析插件producer_plugin 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

producer_plugin是控制區(qū)塊生產(chǎn)的關(guān)鍵插件。

關(guān)鍵字:producer_plugin,同步區(qū)塊的處理,pending區(qū)塊,生產(chǎn)區(qū)塊,最后不可逆區(qū)塊,生產(chǎn)循環(huán),生產(chǎn)安排,水印輪次,計時器,確認(rèn)數(shù)

producer_plugin生命周期

EOS的所有plugin都有共同的基類,因此每個plugin的研究都可以從生命周期入手。

①set_program_options

向config.ini文件中增加屬于producer_plugin的配置項。這些配置項如下表所示。

配置項解釋
enable-stale-production允許區(qū)塊生產(chǎn),即使鏈?zhǔn)顷惛?#xff0c;即鏈生產(chǎn)的區(qū)塊由于遲到未能被采納進(jìn)鏈。
pause-on-startup當(dāng)生產(chǎn)暫停時啟動這個節(jié)點(diǎn)
max-transaction-time執(zhí)行已推送事務(wù)代碼的最長時間限制,過期則判定為無效,默認(rèn)30毫秒
max-irreversible-block-age當(dāng)前節(jié)點(diǎn)生產(chǎn)區(qū)塊所在鏈的DPOS不可逆區(qū)塊的時間限制,按秒計算,默認(rèn)-1無限制。
producer-name當(dāng)前節(jié)點(diǎn)的生產(chǎn)者ID,可以被多次指定。
private-key(已被丟棄,使用以下signature-provider替代)
signature-providerKV元組使用格式為<public-key>=<provider-spec>,等號前為公鑰,等號后為KEY:私鑰或KEOSD:私鑰。前者為以上公鑰對應(yīng)的私鑰,后者為keosd的可用url并且相關(guān)錢包要被解鎖。
keosd-provider-timeout發(fā)送區(qū)塊到keosd簽名的最大時間,按毫秒計算。
greylist-account灰名單,記錄了無法繼承CPU/NET虛擬資源的賬戶列表。
produce-time-offset-us非最后一個區(qū)塊產(chǎn)生時間的偏移量,按微秒計算。負(fù)值會導(dǎo)致塊更早出去,正值會導(dǎo)致塊更晚出去。
last-block-time-offset-us最后一個區(qū)塊產(chǎn)生時間的偏移量,按微秒計算。負(fù)值會導(dǎo)致塊更早出去,正值會導(dǎo)致塊更晚出去。
incoming-defer-ratio當(dāng)兩者都被耗盡時,輸入交易和遞延交易之間的比率。
snapshots-dir快照目錄的位置(絕對路徑或data目錄相對路徑)

②initialize

插件初始化,第一個階段是通過現(xiàn)有配置項初始化設(shè)置插件。現(xiàn)有配置項來自于配置文件config.ini中producer_plugin相關(guān)配置項與命令行參數(shù)中producer_plugin相關(guān)配置項的交集,同樣的配置項以命令行為準(zhǔn)。現(xiàn)有配置以boost::program_options::variables_map&類型對象options為參數(shù)傳入初始化函數(shù)。配置過程操作的是producer_plugin的私有成員std::shared_ptr<class producer_plugin_impl> my。my指針擁有producer_plugin_impl對象的成員,這些成員都被設(shè)計為與傳入配置項對應(yīng),逐一設(shè)置即可。
第二個階段是4個遠(yuǎn)程異步調(diào)用的聲明:

  • 前兩個通訊模式是訂閱一個channel綁定一個執(zhí)行函數(shù),一旦嗅到該頻道被發(fā)布則執(zhí)行綁定的函數(shù)。
    • incoming::channels::block,接收區(qū)塊的頻道,該頻道將在bnet_plugin的on_message的on函數(shù)中被發(fā)布,觸發(fā)producer_plugin當(dāng)前的訂閱函數(shù)on_incoming_block,下面詳述。
    • incoming::channels::transaction,接收事務(wù)的頻道,該頻道與上面相同,也將在bnet_plugin的on_message的on函數(shù)中被發(fā)布,觸發(fā)producer_plugin當(dāng)前的訂閱函數(shù)on_incoming_transaction_async,下面詳述。
  • 后兩個通訊模式是注冊一個method,供外部程序調(diào)用。
    • incoming::methods::block_sync,接收區(qū)塊的同步方法。該method將在chain_plugin的read_write::push_block函數(shù)中被調(diào)用,這部分內(nèi)容在chain_plugin的文章中有專門的分析。實(shí)際上執(zhí)行的是producer_plugin當(dāng)下注冊的方法on_incoming_block,同上。
    • incoming::methods::transaction_async,接收事務(wù)的同步方法。該method將在chain_plugin的read_write::push_transaction函數(shù)中被調(diào)用,實(shí)際上執(zhí)行的是on_incoming_transaction_async,亦同上。

總結(jié)一下會發(fā)現(xiàn),在producer_plugin的初始化階段:

  • 有兩個處理對象,
    • 接收的區(qū)塊,針對該處理對象,均執(zhí)行函數(shù)on_incoming_block
    • 接收的事務(wù),針對該處理對象,均執(zhí)行函數(shù)on_incoming_transaction_async
  • 有兩個通訊模式,
    • channel的方式,對接的是bnet_plugin
    • method的方式,對接的是chain_plugin

③startup

進(jìn)入插件的啟動階段,首先設(shè)置日志,

const fc::string logger_name("producer_plugin"); fc::logger _log;const fc::string trx_trace_logger_name("transaction_tracing"); fc::logger _trx_trace_log;

分別創(chuàng)建以producer_plugin插件為主的日志對象,以及事務(wù)追蹤"transaction_tracing"為主的日志對象。除了這兩個在插件內(nèi)部新建的日志,還有程序自身日志,例如nodeos,日志信息將打印在nodeos的輸出位置,輸出插件啟動日志。接下來的工作列舉如下:

  • 校驗chain的db讀取模式以及本地生產(chǎn)者集合是否為空,根據(jù)不同情況輸出對應(yīng)日志用于提示用戶。
  • 使用【信號槽技術(shù)】分別連接信號accepted_block(綁定本地處理函數(shù)on_block和信號irreversible_block(綁定本地處理函數(shù)on_irreversible_block,這兩個信號將在controller中被發(fā)射,從而觸發(fā)當(dāng)前信號槽,這兩個處理函數(shù)將在下面詳述。

信號槽的方式,對接的都是controller。

之前討論過多次,由于解耦的模式,信號發(fā)射方和信號槽處理方互不認(rèn)識,因此一個信號被發(fā)射,可以擁有多個信號槽處理方。

  • 是針對最后不可逆區(qū)塊的討論,下面詳述。
  • 如果本地生產(chǎn)者集合不為空時,輸出日志在當(dāng)前為這些生產(chǎn)者啟動區(qū)塊生產(chǎn)工作。如果本地具備生產(chǎn)能力_production_enabled,如果當(dāng)前鏈的頭區(qū)塊號為0,則調(diào)用new_chain_banner(chain),該函數(shù)下面詳述。
  • 執(zhí)行定時生產(chǎn)循環(huán)函數(shù)schedule_production_loop,下面詳述。

④shutdown

釋放資源,代碼不多如下:

void producer_plugin::plugin_shutdown() {try {my->_timer.cancel(); // 停止倒計時器} catch(fc::exception& e) {edump((e.to_detail_string())); // 輸出錯誤日志}// 重置釋放連接槽my->_accepted_block_connection.reset();my->_irreversible_block_connection.reset(); }

插件關(guān)閉階段,取消計時器,后面會展開對計時器basic_deadline_timer的研究,重置(調(diào)用析構(gòu)函數(shù))清除上面startup階段啟動的兩個信號槽。

on_incoming_block 函數(shù)

/*** 處理incoming接收到的區(qū)塊。* @param block 已簽名區(qū)塊*/ void on_incoming_block(const signed_block_ptr& block) {fc_dlog(_log, "received incoming block ${id}", ("id", block->id()));// 判斷區(qū)塊時間是否在當(dāng)前節(jié)點(diǎn)的未來7秒之內(nèi),如果不是,則證明這個區(qū)塊還沒到處理的時間。EOS_ASSERT( block->timestamp < (fc::time_point::now() + fc::seconds(7)), block_from_the_future, "received a block from the future, ignoring it" );// 獲取鏈對象。chain::controller& chain = app().get_plugin<chain_plugin>().chain();/* 如果本地已經(jīng)存在接收的區(qū)塊了,則不必處理,直接返回。*/auto id = block->id();auto existing = chain.fetch_block_by_id( id );if( existing ) { return; }// 啟動多線程驗證區(qū)塊。這個函數(shù)在下面有解釋auto bsf = chain.create_block_state_future( block );// 丟棄pending區(qū)塊chain.abort_block();// 拋出異常,保證重啟定時生產(chǎn)循環(huán)auto ensure = fc::make_scoped_exit([this](){schedule_production_loop();});// 向本地鏈推送新區(qū)塊bool except = false;try {chain.push_block(block);//推送區(qū)塊} catch ( const guard_exception& e ) {// 打印詳細(xì)錯誤日志,并跳出循環(huán)。app().get_plugin<chain_plugin>().handle_guard_exception(e);return;} catch( const fc::exception& e ) {elog((e.to_detail_string()));except = true;} catch ( boost::interprocess::bad_alloc& ) {chain_plugin::handle_db_exhaustion();return;}if( except ) {// rejected_block頻道發(fā)布某區(qū)塊已被拒絕的消息,該頻道已在bnet插件被訂閱,當(dāng)消息發(fā)布,bnet插件會調(diào)用函數(shù)on_bad_block處理被拒區(qū)塊。app().get_channel<channels::rejected_block>().publish( block );return;}// 當(dāng)鏈的頭塊狀態(tài)中時間戳的下一個點(diǎn)大于等于當(dāng)前時間時,本地則具備生產(chǎn)能力。if( chain.head_block_state()->header.timestamp.next().to_time_point() >= fc::time_point::now() ) {_production_enabled = true;}if( fc::time_point::now() - block->timestamp < fc::minutes(5) || (block->block_num() % 1000 == 0) ) {//區(qū)塊時間點(diǎn)已流逝的時間在5分鐘之內(nèi)的情況,或者區(qū)塊號是整千時。輸出日志模板并替換變量的值。ilog("Received block ${id}... #${n} @ ${t} signed by ${p} [trxs: ${count}, lib: ${lib}, conf: ${confs}, latency: ${latency} ms]",// p是生產(chǎn)者,// id是區(qū)塊id截取中間的8到16位輸出,// n是區(qū)塊號,t是區(qū)塊時間,// count是區(qū)塊中事務(wù)的數(shù)量,// lib是鏈最后一個不可逆區(qū)塊號,// confs是區(qū)塊的確認(rèn)數(shù)("p",block->producer)("id",fc::variant(block->id()).as_string().substr(8,16))("n",block_header::num_from_id(block->id()))("t",block->timestamp)// confirmed,是生產(chǎn)者在簽名一個區(qū)塊時向前確認(rèn)的區(qū)塊數(shù)量,默認(rèn)是1,則只確認(rèn)前一個區(qū)塊。// latency,潛伏因素的字面含義。值為當(dāng)前區(qū)塊時間點(diǎn)已流逝的時間。// count是時間庫中的一個特殊函數(shù),返回某個時間按照某個單位來計數(shù)時的字面值,可以用做跨單位的運(yùn)算。// block_timestamp_type類型定義了區(qū)塊鏈的時間戳的默認(rèn)間隔是500ms,一個周期是2000年。("count",block->transactions.size())("lib",chain.last_irreversible_block_num())("confs", block->confirmed)("latency", (fc::time_point::now() - block->timestamp).count()/1000 ) );} }

關(guān)于block_timestamp_type類型的定義,源碼如下:

typedef block_timestamp<config::block_interval_ms,config::block_timestamp_epoch> block_timestamp_type; ... const static int block_interval_ms = 500; const static uint64_t block_timestamp_epoch = 946684800000ll; // epoch is year 2000.

接著進(jìn)入函數(shù)create_block_state_future,

std::future<block_state_ptr> create_block_state_future( const signed_block_ptr& b ) {EOS_ASSERT( b, block_validate_exception, "null block" );//不能為空塊auto id = b->id();// 已存在區(qū)塊,終止并提示auto existing = fork_db.get_block( id );EOS_ASSERT( !existing, fork_database_exception, "we already know about this block: ${id}", ("id", id) );auto prev = fork_db.get_block( b->previous );// 獲得前一個區(qū)塊,不存在則報錯。EOS_ASSERT( prev, unlinkable_block_exception, "unlinkable block ${id}", ("id", id)("previous", b->previous) );return async_thread_pool( [b, prev]() {// 傳入具體task到異步線程池。const bool skip_validate_signee = false;return std::make_shared<block_state>( *prev, move( b ), skip_validate_signee );} ); }

異步線程池async_thread_pool。傳入task,由當(dāng)前同步的待驗證區(qū)塊以及前一個區(qū)塊組成,返回的是block_state對象。

template<typename F> auto async_thread_pool( F&& f ) {auto task = std::make_shared<std::packaged_task<decltype( f() )()>>( std::forward<F>( f ) );boost::asio::post( *thread_pool, [task]() { (*task)(); } );// 將任務(wù)上傳到線程池,通過boost::asio庫異步分配線程并行處理。return task->get_future(); }

on_incoming_transaction_async 函數(shù)

該函數(shù)的工作是處理接收到的事務(wù)的本地同步,聲明如下:

/*** 處理接收到的事務(wù)的本地同步工作* @param trx 接收的事務(wù),是打包狀態(tài)的* @param persist_until_expired 標(biāo)志位:事務(wù)是否在過期前被持久化了,bool類型* @param next 回調(diào)函數(shù)next方法。*/ void on_incoming_transaction_async(const packed_transaction_ptr& trx, bool persist_until_expired, next_function<transaction_trace_ptr> next) {}

可以分為三個部分,第一部分是校驗工作。

如果鏈不存在pending區(qū)塊狀態(tài),則在pending接收事務(wù)結(jié)合中增加接收的事務(wù)待start_block中處理,并中止函數(shù)返回。

接收到的事務(wù)要打包在本地的pending區(qū)塊中,如果不存在pending區(qū)塊,說明本地節(jié)點(diǎn)未開始生產(chǎn)區(qū)塊,所以要插入到pending事務(wù)集合_pending_incoming_transactions中等待start_block來處理。這部分的校驗代碼如下:

chain::controller& chain = app().get_plugin<chain_plugin>().chain();if (!chain.pending_block_state()) {_pending_incoming_transactions.emplace_back(trx, persist_until_expired, next);return; }

第二部分是該函數(shù)定義了一個lambda的內(nèi)部函數(shù)send_response,用于異步發(fā)送響應(yīng),該內(nèi)部函數(shù)源碼如下:

auto send_response = [this, &trx, &chain, &next](const fc::static_variant<fc::exception_ptr, transaction_trace_ptr>& response) {next(response);if (response.contains<fc::exception_ptr>()) {// 如果響應(yīng)中包含異常指針,則發(fā)布異常信息以及事務(wù)對象到channels::transaction_ack_transaction_ack_channel.publish(std::pair<fc::exception_ptr, packed_transaction_ptr>(response.get<fc::exception_ptr>(), trx));if (_pending_block_mode == pending_block_mode::producing) {// 如果pending區(qū)塊的模式為生產(chǎn)中,則打印出對應(yīng)的debug日志:區(qū)塊被拒絕。fc_dlog(_trx_trace_log, "[TRX_TRACE] Block ${block_num} for producer ${prod} is REJECTING tx: ${txid} : ${why} ",("block_num", chain.head_block_num() + 1)("prod", chain.pending_block_state()->header.producer)("txid", trx->id())("why",response.get<fc::exception_ptr>()->what()));} else {// 如果pending區(qū)塊的模式為投機(jī)中,則打印出對應(yīng)的debug日志:投機(jī)行為被拒絕。fc_dlog(_trx_trace_log, "[TRX_TRACE] Speculative execution is REJECTING tx: ${txid} : ${why} ",("txid", trx->id())("why",response.get<fc::exception_ptr>()->what()));}} else {// 響應(yīng)中無異常。發(fā)布空異常信息以及事務(wù)對象到channels::transaction_ack_transaction_ack_channel.publish(std::pair<fc::exception_ptr, packed_transaction_ptr>(nullptr, trx));if (_pending_block_mode == pending_block_mode::producing) {// 仍舊區(qū)分pending區(qū)塊狀態(tài)生產(chǎn)中與投機(jī)行為的不同日志輸出。fc_dlog(_trx_trace_log, "[TRX_TRACE] Block ${block_num} for producer ${prod} is ACCEPTING tx: ${txid}",("block_num", chain.head_block_num() + 1)("prod", chain.pending_block_state()->header.producer)("txid", trx->id()));} else {fc_dlog(_trx_trace_log, "[TRX_TRACE] Speculative execution is ACCEPTING tx: ${txid}",("txid", trx->id()));}}};

這部分代碼很容易拆解,通過publish/subscribe通訊模式,本地發(fā)布頻道信息,交由頻道訂閱者異步處理。
頻道:channels::transaction_ack
publisher:on_incoming_transaction_async ->send_response
subscriber:net_plugin::plugin_startup
binding function:net_plugin_impl::transaction_ack
dispatcher:rejected_transaction/bcast_transaction

在這個異步通訊過程中,要加入校驗代碼。函數(shù)體被調(diào)用時,send_response已經(jīng)收到了處理后的事務(wù)響應(yīng),同時捕獲了事務(wù)源對象,鏈對象。鏈對象在當(dāng)前程序中應(yīng)該是單例的,不必在此校驗。校驗響應(yīng)事務(wù)是否存在異常信息,如果存在則將異常信息附屬發(fā)布到頻道消息,如果不存在則附屬空異常。

if (response.contains<fc::exception_ptr>()) {_transaction_ack_channel.publish(std::pair<fc::exception_ptr, packed_transaction_ptr>(response.get<fc::exception_ptr>(), trx)); } else {_transaction_ack_channel.publish(std::pair<fc::exception_ptr, packed_transaction_ptr>(nullptr, trx)); }

注意,在發(fā)布完頻道消息以后,要給前臺輸出事務(wù)跟蹤日志。

producer_plugin的startup啟動階段分析過,該插件包含三種日志,事務(wù)跟蹤日志就是其中之一。輸出日志要判斷pending區(qū)塊性質(zhì)是否是正在生產(chǎn),如果是生產(chǎn)中的區(qū)塊,則打印區(qū)塊號,生產(chǎn)者以及事務(wù)id,如果不是生產(chǎn)中的區(qū)塊而是投機(jī)區(qū)塊(可能被生產(chǎn)也可能被丟棄),則只打印事務(wù)id。

publish的消息是trx源事務(wù)對象,而不是響應(yīng)對象response。

前兩部分完成以后,本地存在pending區(qū)塊有打包事務(wù)的條件,且發(fā)送響應(yīng)的函數(shù)也有了,準(zhǔn)備工作已經(jīng)做好了。接下來進(jìn)入第三部分,正式開始本地打包接收事務(wù)的工作。工作開始之前,仍舊要先校驗:

  • 接收的事務(wù)是否過期,通過比較待打包區(qū)塊時間和接收事務(wù)時間確定事務(wù)是否過期,如果過期則發(fā)送事務(wù)已過期的響應(yīng)信息并終止程序。
  • 接收事的務(wù)是否已存在,在本地查找該事務(wù)如果查到則發(fā)送事務(wù)已存在的響應(yīng)信息并終止程序。

這兩個校驗的源碼如下:

auto block_time = chain.pending_block_state()->header.timestamp.to_time_point();//獲得待打包區(qū)塊時間,即鏈pending區(qū)塊頭的時間戳轉(zhuǎn)換而來。 auto id = trx->id(); if( fc::time_point(trx->expiration()) < block_time ) {//如果事務(wù)的過期時間小于區(qū)塊時間,說明區(qū)塊開始打包時事務(wù)已過期。報錯并中止。send_response(std::static_pointer_cast<fc::exception>(std::make_shared<expired_tx_exception>(FC_LOG_MESSAGE(error, "expired transaction ${id}", ("id", id)) )));return; }if( chain.is_known_unexpired_transaction(id) ) {// 如果在鏈db中找到了該事務(wù),說明已存在,報錯并中止。send_response(std::static_pointer_cast<fc::exception>(std::make_shared<tx_duplicate>(FC_LOG_MESSAGE(error, "duplicate transaction ${id}", ("id", id)) )));return; }

兩個校驗工作結(jié)束以后,要確定接收事務(wù)的code執(zhí)行截止時間。初始化的值是當(dāng)前時間加上本地設(shè)置的最大事務(wù)執(zhí)行時間。但如果本地設(shè)置未限制最大事務(wù)執(zhí)行時間或者pending區(qū)塊是本地正在生產(chǎn)且區(qū)塊時間小于截止時間的,事務(wù)截止時間改為區(qū)塊時間。這段代碼如下:

auto block_time = chain.pending_block_state()->header.timestamp.to_time_point();//獲得待打包區(qū)塊時間,即鏈pending區(qū)塊頭的時間戳轉(zhuǎn)換而來。 auto deadline = fc::time_point::now() + fc::milliseconds(_max_transaction_time_ms);// 算出事務(wù)的code執(zhí)行的截止時間。 bool deadline_is_subjective = false; // 主觀截止日期標(biāo)志位,事務(wù)截止時間為區(qū)塊時間 if (_max_transaction_time_ms < 0 || (_pending_block_mode == pending_block_mode::producing && block_time < deadline) ) {deadline_is_subjective = true; // 主觀截止日期標(biāo)志位設(shè)置為true。deadline = block_time;// 截止時間改為區(qū)塊時間 }

接下來,確認(rèn)了事務(wù)截止時間以后,執(zhí)行推送接收的事務(wù)到區(qū)塊鏈。

// 調(diào)用chain推送事務(wù),接收結(jié)果儲存在trace對象 auto trace = chain.push_transaction(std::make_shared<transaction_metadata>(*trx), deadline);

trace對象接收了chain的推送事務(wù)的處理結(jié)果。如果判斷該結(jié)果沒有異常則證明處理成功,則要先判斷標(biāo)志位persist_until_expired是否為true,如果為true說明該事務(wù)在過期前已被成功持久化,需要在本地持久化事務(wù)集合對象中插入事務(wù)id,用來保證也能應(yīng)用在未來的投機(jī)區(qū)塊。最后,將trace對象作為響應(yīng)信息發(fā)送出去。源碼如下:

if (persist_until_expired) {// 標(biāo)志位:事務(wù)過期前被持久化// 存儲事務(wù)ID,從而保證它也能應(yīng)用在未來的投機(jī)區(qū)塊(可逆區(qū)塊)。_persistent_transactions.insert(transaction_id_with_expiry{trx->id(), trx->expiration()}); } send_response(trace);// 將事務(wù)推送結(jié)果發(fā)送響應(yīng)。

如果trace結(jié)果包含異常,則要判斷該異常是否是主觀異常。如果是的話,采用上面不存在pending區(qū)塊的處理方式,將事務(wù)插入到pending接收事務(wù)集合中,等待start_block處理,同時按照pending區(qū)塊性質(zhì)輸出日志。如果不是主觀失敗,則直接丟棄事務(wù),發(fā)送異常信息作為響應(yīng)內(nèi)容。源碼如下:

if (trace->except) {// 異常處理if (failure_is_subjective(*trace->except, deadline_is_subjective)) {// 主觀失敗,在pending接收事務(wù)結(jié)合中增加接收的事務(wù)待start_block中處理,并中止函數(shù)返回。_pending_incoming_transactions.emplace_back(trx, persist_until_expired, next);// 仍舊區(qū)分pending區(qū)塊狀態(tài)生產(chǎn)中與投機(jī)行為的不同日志輸出。if (_pending_block_mode == pending_block_mode::producing) {fc_dlog(_trx_trace_log, "[TRX_TRACE] Block ${block_num} for producer ${prod} COULD NOT FIT, tx: ${txid} RETRYING ",("block_num", chain.head_block_num() + 1)("prod", chain.pending_block_state()->header.producer)("txid", trx->id()));} else {fc_dlog(_trx_trace_log, "[TRX_TRACE] Speculative execution COULD NOT FIT tx: ${txid} RETRYING",("txid", trx->id()));}} else {// 增加異常信息,發(fā)送響應(yīng)。auto e_ptr = trace->except->dynamic_copy_exception();send_response(e_ptr);} }

on_block 函數(shù)

該函數(shù)在controller accepted_block信號處理時有過介紹。下面進(jìn)行詳細(xì)分析,首先是三個校驗:

if( bsp->header.timestamp <= _last_signed_block_time ) return; if( bsp->header.timestamp <= _start_time ) return; if( bsp->block_num <= _last_signed_block_num ) return;
  • 如果區(qū)塊時間小于等于最后簽名區(qū)塊時間,則終止退出當(dāng)前函數(shù)。
  • 如果區(qū)塊時間小于等于開始時間(初始化為當(dāng)前時間),則終止退出當(dāng)前函數(shù)。
  • 如果區(qū)塊號小于等于最后簽名區(qū)塊號,則退出當(dāng)前函數(shù)。

通過以上三個校驗,可以得知新區(qū)塊要在最后簽名區(qū)塊之后(生產(chǎn)時間要在它之后,區(qū)塊號也要在它之后),另外新區(qū)塊的生產(chǎn)時間不能是比現(xiàn)在早的時間,必須是之后的時間才可能被當(dāng)下所處理。校驗通過以后,新建活躍生產(chǎn)者賬戶集合active_producers,插入計劃出塊的生產(chǎn)者。

const auto& active_producer_to_signing_key = bsp->active_schedule.producers;flat_set<account_name> active_producers; active_producers.reserve(bsp->active_schedule.producers.size()); for (const auto& p: bsp->active_schedule.producers) {active_producers.insert(p.producer_name); }

接下來處理接收到的由它節(jié)點(diǎn)生產(chǎn)的區(qū)塊。

// 利用set\_intersection取本地生產(chǎn)者與集合active\_producers的交集 std::set_intersection( _producers.begin(), _producers.end(),active_producers.begin(), active_producers.end(),// 將結(jié)果存入一個迭代器make_function_output_iterator,迭代執(zhí)行內(nèi)部函數(shù)boost::make_function_output_iterator( [&]( const chain::account_name& producer )// 如果結(jié)果為空,說明本地生產(chǎn)者沒有出塊權(quán)利不屬于活躍生產(chǎn)者的一份子 {if( producer != bsp->header.producer ) { // 如果交集生產(chǎn)者不等于接收區(qū)塊的生產(chǎn)者,說明是校驗別人生產(chǎn)的區(qū)塊,如果是相等的不必做特殊處理。// 在活躍生產(chǎn)者的key中找到匹配的key(本地生產(chǎn)者賬戶公鑰)auto itr = std::find_if( active_producer_to_signing_key.begin(), active_producer_to_signing_key.end(),[&](const producer_key& k){ return k.producer_name == producer; } );if( itr != active_producer_to_signing_key.end() ) {// 成功找到,否則說明該區(qū)塊不是合法生產(chǎn)者簽名拋棄不處理。auto private_key_itr = _signature_providers.find( itr->block_signing_key );// 獲取本地生產(chǎn)者私鑰if( private_key_itr != _signature_providers.end() ) {auto d = bsp->sig_digest();auto sig = private_key_itr->second( d );// 更新producer插件本地標(biāo)志位_last_signed_block_time = bsp->header.timestamp;_last_signed_block_num = bsp->block_num;// 組裝生產(chǎn)確認(rèn)數(shù)據(jù)字段,包括區(qū)塊id,區(qū)塊摘要,生產(chǎn)者,簽名。發(fā)射信號confirmed\_block。但經(jīng)過搜索,項目中目前沒有對該信號設(shè)置槽connection_self->confirmed_block( { bsp->id, d, producer, sig } );}}} } ) );

在區(qū)塊創(chuàng)建之前要為該區(qū)塊的生產(chǎn)者設(shè)置水印用來標(biāo)示該區(qū)塊的生產(chǎn)者是誰。水印就是一個kv結(jié)構(gòu)對象,例如 _producer_watermarks[new_producer] = new_block_num;

chain::controller& chain = app().get_plugin<chain_plugin>().chain(); const auto hbn = bsp->block_num; // 設(shè)置新區(qū)塊頭信息,水印信息,包括時間戳 auto new_block_header = bsp->header; new_block_header.timestamp = new_block_header.timestamp.next(); new_block_header.previous = bsp->id; auto new_bs = bsp->generate_next(new_block_header.timestamp);

接下來,對于新安裝的生產(chǎn)者,可以設(shè)置他們的水印使他們變?yōu)榛钴S生產(chǎn)者。

if (new_bs.maybe_promote_pending() && bsp->active_schedule.version != new_bs.active_schedule.version) {flat_set<account_name> new_producers;new_producers.reserve(new_bs.active_schedule.producers.size());for( const auto& p: new_bs.active_schedule.producers) {if (_producers.count(p.producer_name) > 0)new_producers.insert(p.producer_name);}for( const auto& p: bsp->active_schedule.producers) {new_producers.erase(p.producer_name);}for (const auto& new_producer: new_producers) {_producer_watermarks[new_producer] = hbn;// 水印map,本地變量,用于指揮計劃出塊的生產(chǎn)者。} }

on_irreversible_block 函數(shù)

在producer_plugin中,該函數(shù)是用來更新不可逆區(qū)塊時間的,這個時間在系統(tǒng)中由一個時間變量_irreversible_block_time控制。

void on_irreversible_block( const signed_block_ptr& lib ) {_irreversible_block_time = lib->timestamp.to_time_point(); }

這個時間變量將用來計算不可逆區(qū)塊時間的流逝時間,即當(dāng)前時間減去該時間變量的結(jié)果,如果結(jié)果為正數(shù)且不小說明很久沒有出現(xiàn)不可逆區(qū)塊了,反之則是剛剛出現(xiàn)不可逆區(qū)塊。

fc::microseconds get_irreversible_block_age() {auto now = fc::time_point::now();if (now < _irreversible_block_time) {return fc::microseconds(0);} else {return now - _irreversible_block_time;} }

last_irreversible 的討論

在producer_plugin的啟動階段,包含一段關(guān)于最后不可逆區(qū)塊的代碼:

const auto lib_num = chain.last_irreversible_block_num();// 獲取當(dāng)前最后不可逆區(qū)塊號 const auto lib = chain.fetch_block_by_number(lib_num); // 獲取最后不可逆區(qū)塊 if (lib) { // 如果最后不可逆區(qū)塊存在my->on_irreversible_block(lib); // 執(zhí)行函數(shù)同步更新本地區(qū)塊的不可逆時間 } else { // 如果最后不可逆區(qū)塊不存在my->_irreversible_block_time = fc::time_point::maximum();// 區(qū)塊不可逆時間設(shè)置為最大值。 }

通常來講,最后不可逆區(qū)塊的存在是被用來定位本地事務(wù)被打包至某個區(qū)塊后是否成功上鏈變?yōu)椴豢赡鏍顟B(tài),只需要這個區(qū)塊號小于最后不可逆區(qū)塊即可確定。

以上代碼段中if-else語句比較容易理解,是根據(jù)最后不可逆區(qū)塊是否存在對本地區(qū)塊不可逆時間變量_irreversible_block_time的設(shè)置,存在則更新為最后不可逆區(qū)塊的時間,不存在則將其設(shè)置為時間最大值。不存在最后不可逆區(qū)塊意味著鏈數(shù)據(jù)完全是孤立的未經(jīng)任何確認(rèn)的,區(qū)塊鏈的特性也不再存在,因此本地時間變量設(shè)置為了時間的最大值。那么令人費(fèi)解的是上面兩行代碼,首先獲取最后不可逆區(qū)塊號,接著通過該區(qū)塊號獲得區(qū)塊。

controller::last_irreversible_block_num

uint32_t controller::last_irreversible_block_num() const {return std::max(std::max(my->head->bft_irreversible_blocknum, my->head->dpos_irreversible_blocknum), my->snapshot_head_block); }

以上獲取最后不可逆區(qū)塊號函數(shù)的源碼,可以看出是從當(dāng)前區(qū)塊頭的bft不可逆區(qū)塊號、dpos不可逆區(qū)塊號以及快照頭塊的區(qū)塊號三者中選擇最大的一個作為結(jié)果返回。分別來看這三個區(qū)塊號的含義:

  • bft不可逆區(qū)塊號,在區(qū)塊頭狀態(tài)結(jié)構(gòu)中的generate_next函數(shù)中有初始化的操作,這個函數(shù)主要是用來通過一個給定的時間生成一個模板的區(qū)塊頭狀態(tài)對象,不包含事務(wù)Merkle根、action Merkle根以及新生產(chǎn)者字段數(shù)據(jù),因為這些組件是派生自鏈狀態(tài)的。總之,在代碼中查找,發(fā)現(xiàn)bft不可逆區(qū)塊號只有一個初始化為0的賦值動作,原因可能與EOS計劃引入bft而目前還沒有bft有關(guān)系。因此該值為0。
  • dpos不可逆區(qū)塊號,controller初始化為0。仍舊在generate_next函數(shù)中找到該字段的初始化值為calc_dpos_last_irreversible()函數(shù)的結(jié)果。
  • 快照的頭塊號,初始化是0,如果有快照讀入的話,就是快照的頭區(qū)塊號。

calc_dpos_last_irreversible函數(shù)

該函數(shù)用來計算dpos最后不可逆區(qū)塊。

uint32_t block_header_state::calc_dpos_last_irreversible()const {vector<uint32_t> blocknums; blocknums.reserve( producer_to_last_implied_irb.size() );for( auto& i : producer_to_last_implied_irb ) {blocknums.push_back(i.second);}if( blocknums.size() == 0 ) return 0;std::sort( blocknums.begin(), blocknums.end() );//默認(rèn)從小到大排序。less<int>()return blocknums[ (blocknums.size()-1) / 3 ];// dpos最后不可逆區(qū)塊的判斷條件是必須在池子里面保持有2/3個區(qū)塊號是大于自己的。 }

fetch_block_by_number

signed_block_ptr controller::fetch_block_by_number( uint32_t block_num )const { try {auto blk_state = my->fork_db.get_block_in_current_chain_by_num( block_num );// 從分叉庫中根據(jù)塊號獲取狀態(tài)區(qū)塊。if( blk_state && blk_state->block ) {//狀態(tài)區(qū)塊存在且其block成員也存在return blk_state->block; // 返回其block成員對象。}return my->blog.read_block_by_num(block_num);// 否則的話從block.log日志中獲取區(qū)塊返回。 } FC_CAPTURE_AND_RETHROW( (block_num) ) }

重新回到producer_plugin的啟動階段的last_irreversible 的討論。首先通過函數(shù)last_irreversible_block_num從bft和dpos以及快照三個區(qū)塊號中獲取最大的一個,由于目前未引進(jìn)bft且有快照進(jìn)入的概率不高,所以暫定該最后不可逆區(qū)塊號為dpos的那個號。接著用這個區(qū)塊號通過fetch_block_by_number中查找,先在fork_db中查找,如果沒有則在block.log中查找獲得區(qū)塊對象。不過一般fork_db中不應(yīng)該存在不可逆區(qū)塊,如果區(qū)塊變?yōu)椴豢赡鏍顟B(tài)應(yīng)該被立即持久化到block.log,并從fork_db中刪除。

new_chain_banner(chain)

該函數(shù)翻譯過來就是新鏈的條幅,條幅是顯示在日志中的,源碼如下:

void new_chain_banner(const eosio::chain::controller& db) {std::cerr << "\n""*******************************\n""* *\n""* ------ NEW CHAIN ------ *\n""* - Welcome to EOSIO! - *\n""* ----------------------- *\n""* *\n""*******************************\n""\n";if( db.head_block_state()->header.timestamp.to_time_point() < (fc::time_point::now() - fc::milliseconds(200 * config::block_interval_ms))){std::cerr << "Your genesis seems to have an old timestamp\n""Please consider using the --genesis-timestamp option to give your genesis a recent timestamp\n""\n";}return; }

傳入一個鏈對象(controller實(shí)例),輸出一個字符圖案在日志中,接著校驗genesis的時間戳,如果小于當(dāng)前時間200個間隔周期,則報錯重新設(shè)置genesis的時間戳配置為一個就近的時間。

schedule_production_loop

這是一個對于producer_plugin非常重要的函數(shù),是出塊節(jié)點(diǎn)按計劃出塊的循環(huán)函數(shù)。在系統(tǒng)多個功能函數(shù)中涉及處理恢復(fù)繼續(xù)按計劃出塊時,多次被調(diào)用到。該函數(shù)中大量使用到了_timer對象,下面先研究_timer。

basic_deadline_timer 的研究

對producer_plugin_impl類的共有成員_timer的追蹤,可以發(fā)現(xiàn)它是basic_deadline_timer類的對象。

該函數(shù)提供了可等待的計時器功能。basic_deadline_timer類模板提供了執(zhí)行阻塞(blocking)或異步等待(asynchronous wait)定時器期滿的能力。截止日期計時器總是處于兩種狀態(tài)之一:“過期”或“未過期”。如果在過期計時器上調(diào)用wait()或async_wait()函數(shù),則等待操作將立即完成。

使用實(shí)例:

①阻塞等待(blocking wait)

為計時器設(shè)置一個相對時間。

timer.expires_from_now(boost::posix_time::seconds(5));// 從現(xiàn)在開始計時5秒鐘。

等待計時器過期。

timer.wait();

②異步等待(asynchronous wait)

首先要創(chuàng)建一個處理器handler。

void handler(const boost::system::error_code& error) {if (!error){// Timer expired.} }

構(gòu)建一個絕對過期時間的計時器。

boost::asio::deadline_timer timer(io_context, boost::posix_time::time_from_string("2005-12-07 23:59:59.000"));

啟動一個異步等待。

timer.async_wait(handler);

③改變過期時間

當(dāng)存在掛起的異步等待時,更改計時器的過期時間會導(dǎo)致這些等待操作被取消。要確保與計時器關(guān)聯(lián)的操作只執(zhí)行一次,請使用類似的方法:

// boost::asio::basic\_deadline\_timer::expires\_from\_now() 函數(shù)取消任何掛起的異步等待,并返回已取消的異步等待的數(shù)量。如果返回0,則太遲了,且等待處理器已經(jīng)被執(zhí)行,或者即將被執(zhí)行。如果它返回1,那么等待程序會被成功取消。 void on_some_event() // 模擬某事件處理函數(shù) {if (my_timer.expires_from_now(seconds(5)) > 0) {// 取消計時器,啟動一個新的異步等待my_timer.async_wait(on_timeout);} else {// 計時器已過期。} } // 如果一個等待處理程序被取消,傳遞給它的boost::system::error\_code包含值boost::asio::error::operation\_aborted。 void on_timeout(const boost::system::error_code& e) // 超時事件處理函數(shù) {if (e != boost::asio::error::operation_aborted) {計時器未取消,繼續(xù)執(zhí)行操作。} }

回到producer_plugin的shutdown階段中_timer的使用。

my->_timer.cancel();

進(jìn)入basic_deadline_timer::cancel()函數(shù):

std::size_t cancel() {boost::system::error_code ec;std::size_t s = this->get_service().cancel(this->get_implementation(), ec);boost::asio::detail::throw_error(ec, "cancel");return s; }

該函數(shù)將取消所有正在等待計時器的異步操作。

回到schedule_production_loop函數(shù)。

這一部分是對計時器的設(shè)置。首先重置計時器,獲得鏈對象chain以及弱指針producer_plugin_impl實(shí)例。執(zhí)行start_block,并接收結(jié)果,根據(jù)結(jié)果的不同做不同的處理,該結(jié)果為一個枚舉類型:

enum class start_block_result {succeeded, // 成功failed, // 失敗waiting, // 等待exhausted // 耗盡,該狀態(tài)在producer插件中并沒有顯式使用,而是其他狀態(tài)處理完畢剩余的情況。 };
  • 如果是failed,啟動區(qū)塊的返回值是失敗的,那么要輸出提醒日志,同時計時器啟動50毫秒倒計時,異步等待到期以后再次嘗試重新調(diào)用自己schedule_production_loop函數(shù)。
  • 如果是waiting,等待中。判斷生產(chǎn)者如果不是空且啟用了生產(chǎn)能力,則調(diào)用延時計劃生產(chǎn)循環(huán)schedule_delayed_production_loop函數(shù)。
bool production_disabled_by_policy() { // 確定生產(chǎn)能力是否被禁用的方式。有以下三種判斷條件,滿足其一即可。return !_production_enabled || _pause_production || (_max_irreversible_block_age_us.count() >= 0 && get_irreversible_block_age() >= _max_irreversible_block_age_us); }

延時計劃生產(chǎn)循環(huán)schedule_delayed_production_loop函數(shù),主要操作對象是wake_up_time,即該延時操作的喚醒時間。一些列校驗判斷探測出喚醒時間已到達(dá)時,就會調(diào)用回schedule_production_loop函數(shù)。

  • 接下來,start_block結(jié)果其他的狀態(tài)情況,即succeeded或者exhausted。當(dāng)pending區(qū)塊模式為生產(chǎn)中時,pending區(qū)塊的模式分為:
enum class pending_block_mode {producing, // 本地生產(chǎn)speculating // 外部確認(rèn)有可能確認(rèn)失敗不一定能成為不可逆,因此是投機(jī)性的。 };

到目前這個分支下,換句話講就是啟動區(qū)塊start_block已經(jīng)成功succeeded了,但是也有可能耗盡exhausted,這兩種情況要通過另外一種判斷,即是否pending區(qū)塊處于生產(chǎn)中的狀態(tài)來做區(qū)分。

pending區(qū)塊模式為生產(chǎn)中producing

  • start_block成功succeeded。這部分代碼的工作主要是用來保證區(qū)塊要在截止時間之前被裝運(yùn)上鏈。先校驗一下是否存在pending區(qū)塊。接著計算截止時間并按照該時間啟動計時器:
// 計算截止時間,epoch默認(rèn)是從1970/1/1,從epoch開始計算到pending區(qū)塊的時間加上預(yù)設(shè)的區(qū)塊生產(chǎn)時間偏移量。 auto deadline = chain.pending_block_time().time_since_epoch().count() + (last_block ? _last_block_time_offset_us : _produce_time_offset_us); _timer.expires_at( epoch + boost::posix_time::microseconds( deadline ));// 截止時間加上epoch的時間初始量,按此時間啟動計時器。

現(xiàn)行西歷即格里歷,又譯國瑞歷、額我略歷、格列高利歷、格里高利歷,稱西元。地球每天的自轉(zhuǎn)是有些不規(guī)則的,而且正在緩慢減速。所以,格林尼治時間已經(jīng)不再被作為標(biāo)準(zhǔn)時間使用。現(xiàn)在的標(biāo)準(zhǔn)時間──協(xié)調(diào)世界時(UTC)──由原子鐘提供。

  • start_block成功exhausted。仍舊要先檢查是否存在pending區(qū)塊。接著計算預(yù)期時間expect_time,是penging區(qū)塊時間減去一個區(qū)塊間隔時間0.5秒(現(xiàn)在是設(shè)置的0.5秒出一個塊,在config.hpp中可以查到)
auto expect_time = chain.pending_block_time() - fc::microseconds(config::block_interval_us);... config.hpp const static int block_interval_ms = 500; const static int block_interval_us = block_interval_ms*1000;

下面是判斷預(yù)期時間和現(xiàn)在時間的對比,如果預(yù)期時間已過,則將計時器時間調(diào)節(jié)為0(立即執(zhí)行出塊)。如果預(yù)期時間未到,則設(shè)置計時器到預(yù)期時間,等待計時完成。

if (fc::time_point::now() >= expect_time) { // 預(yù)期時間已過_timer.expires_from_now( boost::posix_time::microseconds( 0 )); // 將計時器時間調(diào)節(jié)為0fc_dlog(_log, "Scheduling Block Production on Exhausted Block #${num} immediately", ("num", chain.pending_block_state()->block_num)); } else { // 預(yù)期時間未到_timer.expires_at(epoch + boost::posix_time::microseconds(expect_time.time_since_epoch().count()));fc_dlog(_log, "Scheduling Block Production on Exhausted Block #${num} at ${time}", ("num", chain.pending_block_state()->block_num)("time",expect_time)); // 設(shè)置計時器到預(yù)期時間 }

分別將succeeded以及exhausted狀態(tài)的_timer設(shè)置完畢以后,下面要處理當(dāng)計時器到時的事件處理,即_timer.async_wait函數(shù)。該函數(shù)的參數(shù)為匿名內(nèi)部類組成的異步回調(diào)函數(shù)。

_timer.async_wait([&chain,weak_this,cid=++_timer_corelation_id](const boost::system::error_code& ec) {auto self = weak_this.lock(); // 獲得鎖。if (self && ec != boost::asio::error::operation_aborted && cid == self->_timer_corelation_id) { // 滿足生產(chǎn)區(qū)塊的條件:有鎖且操作未被終止且計時器關(guān)聯(lián)id匹配。// 內(nèi)部要校驗一遍pending區(qū)塊是否存在。auto block_num = chain.pending_block_state() ? chain.pending_block_state()->block_num : 0; // 區(qū)塊號的設(shè)置,若pending區(qū)塊存在則設(shè)置為pending區(qū)塊號,若不存在,則設(shè)置為0。auto res = self->maybe_produce_block(); // 調(diào)用maybe_produce_block()函數(shù)(下面分析)執(zhí)行區(qū)塊的生產(chǎn),返回生產(chǎn)結(jié)果。fc_dlog(_log, "Producing Block #${num} returned: ${res}", ("num", block_num)("res", res));} });

計時器關(guān)聯(lián)id匹配。_timer_corelation_id的存在源自一個攻擊警報:Boost計時器可能處于一個處理程序尚未執(zhí)行但不可中止的狀態(tài),這個狀態(tài)給外部攻擊提供了可能。關(guān)聯(lián)id的設(shè)置可以有效防止,處理程序被改變。在處理程序捕獲相關(guān)性ID設(shè)置時,他們必須執(zhí)行檢查匹配全局變量_timer_corelation_id。如果不匹配,則意味著該方法已被調(diào)用,處理程序處于應(yīng)該取消但無法取消的狀態(tài)。

pending區(qū)塊模式為投機(jī)中speculating

這個狀態(tài)下,分兩種情況處理:

  • 如果生產(chǎn)者存在且具備生產(chǎn)能力(有可能是備用節(jié)點(diǎn))時,校驗一番以后最終會調(diào)用延時計劃出塊循環(huán)schedule_delayed_production_loop。
  • 其他情況則只打印日志,說明創(chuàng)建了投機(jī)區(qū)塊。

maybe_produce_block()函數(shù)

前面提到,schedule_production_loop函數(shù)是出塊者生產(chǎn)區(qū)塊時,調(diào)用start_block函數(shù)并根據(jù)返回結(jié)果設(shè)置計時器_timer,并處理計時完成的處理程序,而最終只有start_block結(jié)果為succeeded以及exhausted狀態(tài),計時完成以后同時滿足有鎖且操作未被終止且計時器關(guān)聯(lián)id匹配。這全部條件的滿足,最后調(diào)用區(qū)塊生產(chǎn)執(zhí)行函數(shù)maybe_produce_block。

簡單來講,schedule_production_loop函數(shù)就是通過調(diào)用start_block設(shè)置timer,計時完成執(zhí)行maybe_produce_block。所以schedule_production_loop函數(shù)的核心是處理_timer。

下面分析函數(shù)maybe_produce_block:

bool producer_plugin_impl::maybe_produce_block() {// 當(dāng)前作用域退出時回調(diào)schedule_production_loop()繼續(xù)循環(huán)處理出塊工作。auto reschedule = fc::make_scoped_exit([this]{ schedule_production_loop();});try {try {produce_block(); // 實(shí)際調(diào)用函數(shù)produce_block生產(chǎn)區(qū)塊。return true; // 返回true,代表區(qū)塊生產(chǎn)成功,其他異常狀態(tài)均返回false,代表出塊失敗。} catch ( const guard_exception& e ) { // 處理守衛(wèi)異常app().get_plugin<chain_plugin>().handle_guard_exception(e);return false;} FC_LOG_AND_DROP();} catch ( boost::interprocess::bad_alloc&) { // 處理內(nèi)部線程內(nèi)存錯誤異常raise(SIGUSR1);return false;}// 區(qū)塊生產(chǎn)出錯,丟其區(qū)塊。fc_dlog(_log, "Aborting block due to produce_block error");chain::controller& chain = app().get_plugin<chain_plugin>().chain();chain.abort_block(); // 丟其區(qū)塊。return false; }

produce_block函數(shù)

區(qū)塊生產(chǎn)函數(shù)用于處理區(qū)塊生產(chǎn),前面maybe_produce_block函數(shù)的主要功能集中在“maybe”,所以實(shí)際出塊任務(wù)仍舊交由produce_block處理。

void producer_plugin_impl::produce_block() {// 區(qū)塊生產(chǎn)必須是pending區(qū)塊狀態(tài)為producing,否則輸出錯誤日志:實(shí)際上并沒有真正生產(chǎn)區(qū)塊。EOS_ASSERT(_pending_block_mode == pending_block_mode::producing, producer_exception, "called produce_block while not actually producing");chain::controller& chain = app().get_plugin<chain_plugin>().chain(); // 獲取chain實(shí)例const auto& pbs = chain.pending_block_state(); // 從chain實(shí)例獲取當(dāng)前pending區(qū)塊,如果獲取為空,則輸出錯誤日志:不存在pending區(qū)塊,可能被其他插件毀壞。const auto& hbs = chain.head_block_state(); // 從chain實(shí)例獲取當(dāng)前頭區(qū)塊EOS_ASSERT(pbs, missing_pending_block_state, "pending_block_state does not exist but it should, another plugin may have corrupted it");auto signature_provider_itr = _signature_providers.find( pbs->block_signing_key ); // 通過pending區(qū)塊的區(qū)塊簽名公鑰去內(nèi)存多索引表_signature_providers中差找signature_provider。// 如果未查到有效signature_provider,則輸出錯誤日志:正在嘗試生產(chǎn)一個區(qū)塊,是由一個我們不擁有的私鑰所簽名。EOS_ASSERT(signature_provider_itr != _signature_providers.end(), producer_priv_key_not_found, "Attempting to produce a block for which we don't have the private key");chain.finalize_block(); // 執(zhí)行chain的區(qū)塊完成操作,重置資源(調(diào)用的為controller的finalize_block函數(shù))。chain.sign_block( [&]( const digest_type& d ) { // 調(diào)用controller的sign_block函數(shù)進(jìn)行函數(shù)簽名,參數(shù)為一個回調(diào)函數(shù)。區(qū)塊簽名最終是由block_header_state來做的實(shí)際工作。auto debug_logger = maybe_make_debug_time_logger();return signature_provider_itr->second(d);} );chain.commit_block(); // 仍舊是執(zhí)行controller的commit_block函數(shù)進(jìn)行區(qū)塊提交。block_state_ptr new_bs = chain.head_block_state(); _producer_watermarks[new_bs->header.producer] = chain.head_block_num(); // 設(shè)置水印// 打印生產(chǎn)結(jié)果日志。ilog("Produced block ${id}... #${n} @ ${t} signed by ${p} [trxs: ${count}, lib: ${lib}, confirmed: ${confs}]",("p",new_bs->header.producer)("id",fc::variant(new_bs->id).as_string().substr(0,16))("n",new_bs->block_num)("t",new_bs->header.timestamp)("count",new_bs->block->transactions.size())("lib",chain.last_irreversible_block_num())("confs", new_bs->header.confirmed));}

水印的意義重申一下,是用來給計劃出塊預(yù)先設(shè)置出塊安排的,即安排下一個區(qū)塊的生產(chǎn)者,管理出塊輪次。

produce_block函數(shù)屬于producer_plugin,然而其中核心區(qū)塊處理,例如重置資源準(zhǔn)備、區(qū)塊簽名、提交區(qū)塊都時通過chain_plugin調(diào)用了controller的相關(guān)函數(shù),而controller只是負(fù)責(zé)管理與數(shù)據(jù)層的交互,數(shù)據(jù)層包括block.log以及及與chainbase的db,區(qū)塊簽名的內(nèi)容是區(qū)塊頭block_header_state來處理。結(jié)構(gòu)拆分如下圖所示:

start_block 函數(shù)

該函數(shù)是producer插件對出塊管理的核心函數(shù)。該函數(shù)通過對時間的控制管理了出塊節(jié)奏,管理出塊輪次。到這里可以得出producer插件的操作對象是pending區(qū)塊,所以該函數(shù)對pending區(qū)塊是本地生產(chǎn)還是外部同步進(jìn)來的做了區(qū)分處理。這其中涉及到一個區(qū)塊同步確認(rèn)的處理,即生產(chǎn)者生產(chǎn)當(dāng)前區(qū)塊時要確認(rèn)多少個區(qū)塊:

  • 如果區(qū)塊的生產(chǎn)者不是當(dāng)前節(jié)點(diǎn)的,則假設(shè)沒有確認(rèn)(丟棄這個塊)。
  • 如果區(qū)塊的生產(chǎn)者是當(dāng)前節(jié)點(diǎn)上從未產(chǎn)生過的生產(chǎn)者,那么保守的方法就是假定沒有確認(rèn),確保不會在crash之后重復(fù)簽名。(不過此處有個問題是crash的話,是否要保證水印持久化?否則crash會丟失,答案是肯定的)
  • 如果區(qū)塊的生產(chǎn)者是這個節(jié)點(diǎn)上的生產(chǎn)者,這個節(jié)點(diǎn)是知道它生成的最后一個塊的,則安全地設(shè)置它:unless
  • 如果區(qū)塊的生產(chǎn)者在該節(jié)點(diǎn)的最后水印中的位置較高,則意味著該區(qū)塊時在一個不同的分叉上。

本函數(shù)大約包含三百多行代碼,用于處理pending區(qū)塊不同情況下的校驗以及動作,包括對區(qū)塊中打包事務(wù)的校驗和處理,最終返回的時start_block_result狀態(tài),前面有介紹過。

start_block的代碼不在此詳細(xì)分析,但總結(jié)下來可以得出是對pending區(qū)塊的區(qū)塊頭校驗,包括是否是本地生產(chǎn)抑或是外部同步,然后是對pending區(qū)塊內(nèi)事務(wù)的處理,包括如何重置打包接收的事務(wù)。到這部分相當(dāng)于將一個區(qū)塊的頭部信息構(gòu)成以及校驗工作和區(qū)塊體的事務(wù)打包內(nèi)容工作完成了。最后返回一個處理狀態(tài),如果通過了層層校驗以及無異常的順利處理,則返回啟動區(qū)塊成功的狀態(tài),如果是時間超時,耗盡了規(guī)定時間則返回exhausted,其他情況則時failed。

總結(jié)

本文分析介紹了producer_plugin的重點(diǎn)功能,研究了其大量內(nèi)部函數(shù)。最初的研究路先是分析該插件的生命周期,然后引申到各個未知或以前未仔細(xì)研究過的調(diào)用的函數(shù)細(xì)節(jié)。其中,涉及到了出塊安排水印、pending區(qū)塊處理、區(qū)塊生產(chǎn)循環(huán)、區(qū)塊的生產(chǎn)者校驗、是否本地或是同步、計時器的相關(guān)知識和應(yīng)用、最后不可逆塊的研究、區(qū)塊生產(chǎn)、區(qū)塊簽名等,另外還涉及到新版本的多線程校驗簽名區(qū)塊的內(nèi)容。研究過程中,也梳理了producer插件與chain插件的交互以及延伸到controller的內(nèi)部函數(shù)的使用。總之,內(nèi)容較多篇幅較長,整體研究脈絡(luò)似乎仍舊不算清晰,但也算是自身知識圖譜的“大數(shù)據(jù)”的一部分,量變引發(fā)質(zhì)變。

更多文章請轉(zhuǎn)到醒者呆的博客園。

轉(zhuǎn)載于:https://www.cnblogs.com/Evsward/p/producerPlugin.html

總結(jié)

以上是生活随笔為你收集整理的EOS生产区块:解析插件producer_plugin的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

免费高清在线观看电视网站 | 久久久久久久网 | 一区二区三区在线视频111 | 黄av免费| 丁香六月激情婷婷 | 国产精品免费看久久久8精臀av | 国产视频美女 | 国产福利一区二区在线 | 国产精品一区二区三区电影 | 精品a在线 | 国产精品久久久久久久久费观看 | 久久日本视频 | 亚洲成av人片在线观看香蕉 | 欧美成人中文字幕 | 欧美综合在线视频 | 婷婷av网站| 亚洲美女视频网 | 美女视频a美女大全免费下载蜜臀 | 国产精品原创 | 丰满少妇在线观看网站 | 精品欧美一区二区在线观看 | 久久精品爱爱视频 | 亚洲精品国产综合久久 | 国产成人精品电影久久久 | 成片免费观看视频999 | 国产成人精品久 | 日韩三级av | 欧美精品久久久久久久久老牛影院 | 国产一区91| 国产一区二区三区 在线 | av3级在线 | 在线免费观看国产精品 | 久久久久成人精品亚洲国产 | 欧美久久久久久久久久久久久 | 精品一区二区在线免费观看 | 91精品系列 | 日韩欧美一级二级 | 日韩www在线 | 久久久久成人免费 | 黄色一级大片在线免费看产 | 欧美激情视频在线免费观看 | 日韩中文在线视频 | 国产一区在线视频 | 亚洲精品乱码久久久久v最新版 | 天天操综 | 国产色影院 | 最新国产在线视频 | 97天堂 | 免费三级大片 | 超碰在线日韩 | 中文区中文字幕免费看 | 天天操天天操天天操天天操 | 在线观看亚洲a | 日韩三级免费观看 | 国产精品久久久免费 | 亚洲女裸体 | 免费视频18| 成人精品视频久久久久 | 色婷婷av一区 | 久久精品99国产国产 | 欧美ⅹxxxxxx| 五月天激情综合 | 视频在线观看入口黄最新永久免费国产 | 99精品国产亚洲 | 韩日电影在线观看 | av在线小说| 国产成人福利在线观看 | 久久综合九色综合欧美就去吻 | 欧美日韩国产精品一区 | 亚洲高清视频一区二区三区 | 玖玖在线观看视频 | 国产精品免费观看网站 | 99re8这里有精品热视频免费 | 国产亚洲成人网 | 91麻豆精品| 欧美日韩视频免费 | 日韩av在线一区二区 | 久久伊人操 | 成人毛片网 | 国产免费小视频 | 日韩免费电影一区二区 | 色多视频在线观看 | 日本高清dvd | 国产精品女同一区二区三区久久夜 | 成人av电影免费在线观看 | 日韩激情影院 | 中文字幕 国产精品 | 99久久精品国产网站 | 天天超碰| 伊人五月婷| 国产一区精品在线 | 国产精品一级在线 | 成人亚洲综合 | 美女精品在线观看 | 日韩av进入| 免费在线播放视频 | 日韩电影一区二区在线观看 | www.色综合.com | www.玖玖玖| 成人久久电影 | 亚洲激精日韩激精欧美精品 | 91大神dom调教在线观看 | 久久精品99 | 久久免费视频在线 | 免费观看视频黄 | 亚洲欧美日韩精品久久久 | 国产美女精品视频 | 色偷偷888欧美精品久久久 | 天天操天天射天天 | 懂色av一区二区三区蜜臀 | 亚洲免费视频在线观看 | 午夜久久久久久久 | 国产女v资源在线观看 | 91精品国自产在线 | 日韩在线视频看看 | 99在线播放 | 中文在线中文资源 | 免费色网站 | 日韩免费观看一区二区三区 | 日韩 国产| 能在线观看的日韩av | 麻豆果冻剧传媒在线播放 | 在线亚洲午夜片av大片 | 久久99精品久久久久蜜臀 | 超碰97免费观看 | 国产视频精品久久 | 日本午夜在线亚洲.国产 | 91福利社在线观看 | 欧美成人精品欧美一级乱 | 日韩av在线高清 | 国产精品久久久影视 | 免费激情网 | 天天射天天干天天爽 | 久久久久中文 | 亚洲精品国产视频 | 欧美日韩中文在线 | 久久久久婷 | 一区免费在线 | 成年人在线观看网站 | 91在线看| 色视频 在线 | 欧美 激情在线 | 97在线视频免费观看 | 亚洲理论在线观看电影 | 99爱在线 | 四虎免费av| 久久精品中文视频 | 免费日韩精品 | 超碰在线98 | 天天干天天操天天射 | 91精品国自产拍天天拍 | 最近中文字幕国语免费av | 亚洲区另类春色综合小说校园片 | 国产精品视频内 | 色综合天天综合在线视频 | 91九色国产 | 五月天国产 | 99久久99久久免费精品蜜臀 | 中文字幕永久免费 | 国产精品美女在线观看 | 日韩在观看线 | 久久9999久久免费精品国产 | 91热这里只有精品 | 国产精品1区2区在线观看 | 国产91丝袜在线播放动漫 | 国产一区福利在线 | 久久婷婷精品视频 | 日韩三级视频 | 国产成人av网址 | 亚洲乱亚洲乱亚洲 | 中文十次啦 | 黄色三级视频片 | 四虎视频 | 久久国产电影院 | 精品久久久久一区二区国产 | 国产精国产精品 | 国产日韩中文字幕在线 | 国产精品久99 | 亚洲精品日韩av | 涩涩网站在线观看 | 亚洲精品自在在线观看 | 成人一级影视 | 欧美资源在线观看 | 夜夜看av| 国产成人黄色av | 特级西西444www高清大视频 | 国产第页 | 人人干在线观看 | 91黄在线看| 日韩av电影网站在线观看 | 国产一级大片在线观看 | 国产综合香蕉五月婷在线 | 亚洲91中文字幕无线码三区 | 亚洲 综合 国产 精品 | 久草在线欧美 | 久久久久国产精品一区 | 日本三级大片 | 91av免费看 | 成人aⅴ视频 | 亚洲黄色免费在线看 | 五月婷婷伊人网 | 色网免费观看 | 中文字幕av在线电影 | 中文字幕视频免费观看 | 黄色三级免费观看 | 亚洲美女视频在线观看 | 久久免费影院 | 97成人超碰| 少妇高潮流白浆在线观看 | 在线播放国产一区二区三区 | 成人国产精品电影 | 亚洲专区路线二 | 日韩1级片 | av在线收看| 五月天综合激情网 | 欧美一级日韩三级 | 成人在线观看资源 | 最新国产在线视频 | 丁香视频| 99久久久国产精品 | 免费十分钟 | 亚洲国产理论片 | 手机在线看a | 日日色综合 | 成人av网页 | 国产精品资源在线 | 91精品国产综合久久婷婷香蕉 | 在线婷婷 | 国产一二三区在线观看 | 午夜精品久久一牛影视 | 久草com | 天天做天天爱天天爽综合网 | 五月天久久久 | 色爱区综合激月婷婷 | 国产成人1区 | 成人网中文字幕 | 欧美日韩性视频 | 人人爽人人爽人人片av | 伊人伊成久久人综合网站 | 在线看成人片 | 日日摸日日爽 | 日韩三级免费观看 | 人人澡人人爽欧一区 | 色婷婷久久久 | 伊人狠狠色 | 成人黄色小说在线观看 | 99爱在线观看 | 高清av免费观看 | 色视频在线观看 | 怡春院av | 2019中文字幕网站 | 免费毛片一区二区三区久久久 | 亚洲最大色| 六月丁香婷婷网 | 日韩va在线观看 | 国产原创中文在线 | 国产精品国产精品 | 国产成年人av | 国产日韩精品在线 | 国产一区观看 | 欧美精品一区在线 | 免费在线激情电影 | 久久久精品一区二区 | 天天操天天射天天插 | 97福利在线 | 日韩av免费在线看 | 欧美射射射 | 超碰97av在线 | 一本一道久久a久久综合蜜桃 | 国产一级免费av | 色婷婷丁香 | 久久久久精 | 91九色国产视频 | 日韩在线一级 | 欧美精品国产综合久久 | 精品亚洲成a人在线观看 | 日韩videos高潮hd | 狠狠做深爱婷婷综合一区 | 色网免费观看 | www视频在线播放 | 成人黄色小说网 | a级国产乱理伦片在线播放 久久久久国产精品一区 | 国产精品18久久久久白浆 | 国产亚洲精品久久久久动 | 99久久久久成人国产免费 | 激情中文字幕 | 久久网站最新地址 | 久久久久久看片 | 波多野结衣动态图 | 免费在线观看av网址 | 69热国产视频 | 日日干日日色 | 九九影视理伦片 | 日韩欧美在线观看一区 | 亚洲精品动漫在线 | 精品国产一区二区久久 | 国产资源精品在线观看 | 色妞久久福利网 | 亚洲最新精品 | 日韩免费电影 | 久久精品艹| 丁香婷婷综合网 | 国产日产欧美在线观看 | 五月婷婷六月丁香在线观看 | 狠狠干夜夜 | 一区二区三区免费在线观看视频 | 91视频88av | 久久激情五月激情 | 麻豆免费在线播放 | 婷婷六月天在线 | 四虎5151久久欧美毛片 | 亚洲伊人第一页 | 免费网站观看www在线观看 | 99久久日韩精品免费热麻豆美女 | 国产精品一区二区白浆 | 国产成人精品一区二区三区在线观看 | 免费看片色 | 亚洲精品乱码久久久久久按摩 | 国产成人精品女人久久久 | 国产成人精品av | 欧美日韩在线观看一区二区三区 | 天天色棕合合合合合合 | 五月婷婷丁香在线观看 | 成人一级电影在线观看 | 国产精品久久久区三区天天噜 | 久久免费视频国产 | www.com.黄| 国产一区av在线 | 欧美日产在线观看 | 激情综合婷婷 | 亚洲欧美一区二区三区孕妇写真 | 在线国产能看的 | 国产精品久久久 | 成人午夜电影在线播放 | 免费看成人片 | 激情综合网色播五月 | 激情av在线资源 | 91精品一区二区在线观看 | 国产成人精品999在线观看 | 国产三级视频 | 国产色资源 | 成人av网址大全 | 天海翼一区二区三区免费 | 国产中出在线观看 | 国产精品成人一区二区 | 五月天视频网站 | 精品一区二三区 | 精品久久久久久一区二区里番 | 在线视频一二三 | 免费观看国产视频 | 精品超碰 | 国产成人一区二区三区在线观看 | 国产精品大全 | 久草视频中文在线 | av一区二区三区在线播放 | 国产精品门事件 | 国产精品久久久久一区二区三区共 | 24小时日本在线www免费的 | 在线视频 成人 | 免费黄色在线网站 | 2018亚洲男人天堂 | 午夜在线看 | 三级在线国产 | 天天射综合网视频 | 欧美在一区| 国产流白浆高潮在线观看 | 91色影院| 成人免费在线电影 | 五月婷婷色综合 | 国产精品欧美久久久久三级 | 久草网首页| 国产在线精品视频 | 亚洲精品动漫成人3d无尽在线 | 天堂在线一区二区三区 | 91人网站| 在线观看久久久久久 | 日韩在线精品一区 | 亚洲综合成人av | 97av影院| 最近中文字幕mv免费高清在线 | 福利网在线 | 黄色一二级片 | 99热最新精品 | 午夜美女wwww | 日日干影院| 最近免费中文视频 | 美女精品| 国产一区二区三区四区在线 | 人人干人人超 | 在线看国产 | 日韩精品视频久久 | 99久久99视频只有精品 | 91一区二区三区在线观看 | 九9热这里真品2 | 欧美成年黄网站色视频 | 美女在线免费视频 | 色资源二区在线视频 | 丁香六月天 | 人人爽久久久噜噜噜电影 | 国产91精品一区二区麻豆网站 | 综合在线观看 | 亚洲人在线视频 | 午夜精品久久久久久久99热影院 | 中文字幕资源在线 | 精品欧美小视频在线观看 | 亚洲一区黄色 | 国产精品乱码在线 | 在线中文字幕av观看 | 欧美美女视频在线观看 | 91精品视频免费在线观看 | 日日爱网址 | a成人v | 一区二区三区在线视频观看58 | 51精品国自产在线 | 欧美日韩精品在线一区二区 | japanesefreesex中国少妇 | 国产精品一区免费观看 | 久久精品国产v日韩v亚洲 | 免费91麻豆精品国产自产在线观看 | 欧美日韩精品免费观看 | 久久婷婷丁香 | 超碰人人做 | 九草在线视频 | 99精品国产在热久久 | 精品国产一区二区久久 | 日韩在线观看中文 | 五月天亚洲精品 | 热九九精品 | 美女免费网视频 | 国产一级在线看 | 成年人视频在线免费 | 国产欧美中文字幕 | 最新日韩电影 | 国产中文字幕在线视频 | 操久在线 | 色黄www小说 | 久久色亚洲| av黄网站 | 精品国产a | 中文字幕一区二区三区久久蜜桃 | 一区二区三区污 | 色夜影院 | 久久综合狠狠综合久久综合88 | 99精品影视 | 狠狠色丁香婷婷综合久小说久 | 久久久久免费视频 | 97视频在线看 | 五月天婷婷免费视频 | 97在线观看视频 | 精品免费在线视频 | 中文字幕中文字幕在线中文字幕三区 | 人人干人人超 | 日韩黄视频 | 日韩av资源站 | 2019精品手机国产品在线 | 日韩视频一区二区三区在线播放免费观看 | 在线草| 成年人毛片在线观看 | 在线看岛国av | 久久国产系列 | 国产精品久久久久久久久久免费看 | 国产日韩精品一区二区三区 | 狠狠躁夜夜a产精品视频 | 久久精品香蕉 | 国产98色在线 | 日韩 | 国产专区视频在线观看 | 蜜臀av性久久久久蜜臀aⅴ涩爱 | 激情综合网在线观看 | 日韩二区三区在线 | 天天操天天操天天操天天操天天操天天操 | 国产精品爽爽爽 | 成人免费色| 午夜精品久久久久久久99水蜜桃 | 超级av在线 | 日韩网站在线播放 | 日韩免费电影一区二区 | 欧美激情精品久久久久久免费印度 | 久草a视频| 成人啊 v | 国产一区网 | 亚洲播播| 免费在线观看av不卡 | 日本中文字幕电影在线免费观看 | 7777精品伊人久久久大香线蕉 | 国产精品青草综合久久久久99 | 狠狠躁夜夜a产精品视频 | 精品久久久久久国产偷窥 | 中文字幕av在线免费 | 久久久亚洲国产精品麻豆综合天堂 | 国产在线精品一区二区不卡了 | 三级av小说| 天天搞天天 | 九九热免费在线视频 | 丁香激情网 | 国产精品扒开做爽爽的视频 | 精品一区二区在线播放 | 精品国自产在线观看 | 在线免费av电影 | 色婷婷视频在线观看 | 日韩最新在线 | 国产精品久久久久免费 | 久久九九视频 | 99视频网站| 欧美激情xxxx | 久草视频手机在线 | 国产亚洲精品美女 | 99亚洲国产 | 国产精品久久婷婷六月丁香 | 亚洲在线网址 | 久久99国产精品 | 九九九九色 | 国产一级电影网 | 一区三区在线欧 | 久久在线免费观看视频 | 欧美成人h版在线观看 | 中文字幕成人在线观看 | 国产区免费 | 91成人在线免费观看 | 伊人国产在线观看 | 日韩精品观看 | 国产一区二区视频在线播放 | 日韩欧美国产精品 | 成人禁用看黄a在线 | 狠狠做六月爱婷婷综合aⅴ 日本高清免费中文字幕 | 国产精品综合在线观看 | 久久精品综合一区 | 亚洲成人黄色网址 | 美女黄频在线观看 | 日韩网页 | 国产美女免费看 | 精品伊人久久久 | 婷婷六月天天 | 欧美专区亚洲专区 | 成人免费视频观看 | 激情综合一区 | 99re8这里有精品热视频免费 | 狠狠色丁香久久婷婷综合丁香 | 在线电影 一区 | 欧美日韩免费网站 | 国产91在线 | 美洲 | 九九亚洲视频 | 国产中文字幕在线免费观看 | 96亚洲精品久久久蜜桃 | 亚洲国产av精品毛片鲁大师 | 午夜av免费看 | 成年人免费观看国产 | 欧美成人一区二区 | 日韩精品偷拍 | 久草在线视频在线观看 | 成人一级在线观看 | 人人看人人做人人澡 | 国产日韩视频在线观看 | 黄色www免费 | 成人av一级片 | 成人精品福利 | 在线观看一区 | 99精品国产一区二区三区麻豆 | 99精品一级欧美片免费播放 | 中文国产在线观看 | 波多野结衣在线播放视频 | 亚洲最大av在线播放 | 久久女同性恋中文字幕 | 一区二区理论片 | 亚洲在线网址 | 黄色一及电影 | 国产成人av片 | 中文字幕亚洲综合久久五月天色无吗'' | 久久国产精品久久久久 | 国产精品精品国产 | 99激情网 | 91一区二区三区久久久久国产乱 | 91最新在线视频 | 久久久免费看片 | 亚洲无吗视频在线 | 久久久久久久久久久久久影院 | 国产免费资源 | 亚洲国产精品999 | 欧美激情xxxx性bbbb | 日韩免费成人 | 久久久鲁| 超级碰碰碰碰 | 国产一区二区在线视频观看 | 午夜视频福利 | 色a资源在线 | 日韩av看片| 奇米影视777影音先锋 | 欧美日韩电影在线播放 | 亚洲一区二区视频 | 久草视频在线看 | 九九热有精品 | 欧美精品少妇xxxxx喷水 | 国产精品自产拍在线观看中文 | 欧美精品三级在线观看 | 亚洲a免费 | 久久久国产精品久久久 | 久久精品五月 | 欧美韩国日本在线 | 免费在线成人av电影 | 亚洲日本国产 | 国内精品久久久久久久久久 | 日本aaaa级毛片在线看 | 最近中文字幕高清字幕免费mv | 二区三区中文字幕 | 丁香婷婷成人 | 日韩精品免费一区二区在线观看 | 国内精品免费久久影院 | 正在播放国产91 | 免费日p视频 | 中文字幕在线免费观看视频 | 狠狠久久| 成片免费观看视频 | 国产一区国产二区在线观看 | 丁香花在线视频观看免费 | 国产原创在线观看 | 丁香六月婷婷综合 | 欧美日韩国产精品一区 | 夜夜躁狠狠躁日日躁 | 超碰在线个人 | 在线观看aaa | 国产精品女人久久久 | 国产精品成人一区二区三区吃奶 | 亚洲网站在线看 | 99精品亚洲 | 国产v欧美 | 欧美成人aa | 狠狠躁夜夜躁人人爽超碰97香蕉 | 久久精品9 | 九九免费在线观看视频 | 成人毛片在线视频 | 欧美少妇18p | 97电影在线看视频 | 人人草人人做 | 6699私人影院 | 久久免费播放 | 91网免费看 | 欧美analxxxx | 手机看片1042 | 免费精品久久久 | 国产黄色特级片 | 99国产在线 | 免费网站在线观看人 | 国产日女人 | 亚洲成人av电影在线 | 日韩91av | 久久久久久国产精品亚洲78 | 制服丝袜在线91 | 国产人成在线观看 | 欧美日韩亚洲在线观看 | 三级av免费观看 | 久久久久久久久毛片精品 | 天天天插| 亚洲区视频在线观看 | 久久久久久久久久久黄色 | 久久999精品 | 免费在线观看毛片网站 | 伊人影院在线观看 | 亚洲视频 一区 | 日韩欧美精品在线观看视频 | 日本久久免费视频 | 一区二区三区在线免费观看 | 日韩欧美在线视频一区二区 | 00av视频| 免费在线看v | 免费看黄网站在线 | av成人免费在线 | 黄色在线视频网址 | 国产亚洲一区二区三区 | 91视频网址入口 | 久久激情视频网 | 麻豆视频免费网站 | 韩日av在线 | 国产精品自拍在线 | 欧美一级免费片 | 好看的国产精品视频 | 国产精品18久久久久久首页狼 | 狠狠色综合欧美激情 | 四虎在线观看精品视频 | 九九热国产 | 免费在线成人 | 日韩欧美在线观看一区二区 | 国产精品视频免费在线观看 | 日韩欧美一区二区三区在线观看 | 精品中文字幕在线播放 | 91高清视频 | 一二三久久久 | 99免费观看视频 | 国产精品2区 | 成人免费一级片 | 亚洲专区视频在线观看 | 成人黄色在线电影 | 91久久电影 | 国产精品久久久久久久久久久久久 | 97视频在线观看播放 | 国产xvideos免费视频播放 | 玖玖爱国产在线 | 少妇bbb | 九色免费视频 | 又爽又黄又刺激的视频 | 久久国产剧场电影 | 欧美一级日韩三级 | 99爱精品视频 | 在线观看激情av | 久久网站最新地址 | 六月色婷婷 | 成人黄色影片在线 | 国产成人精品午夜在线播放 | 国产女做a爱免费视频 | 色偷偷88888欧美精品久久久 | 六月激情久久 | 国产精品久久久久久影院 | 国产天天爽| 日韩精品中文字幕在线播放 | 射综合网| 丝袜美腿在线 | 国产91精品在线观看 | 国产黄色片一级三级 | 天天添夜夜操 | 黄色av一级 | 在线观看亚洲 | 久久成人一区 | 麻豆久久 | 激情综合久久 | 久草视频中文在线 | 亚洲专区路线二 | 精品视频在线观看 | 亚洲最新在线视频 | 在线观看黄色国产 | 国产精品久久一区二区三区, | 免费欧美精品 | 国内精品久久久久影院一蜜桃 | 天天干人人 | www.久久色| 精品国产91亚洲一区二区三区www | 久久精品一区二区三区四区 | av在线免费播放 | 99精品在线观看视频 | 久久成人国产精品入口 | 国产福利网站 | 亚洲国内精品 | 精品视频国产 | 欧美日韩免费观看一区=区三区 | 成人网中文字幕 | 久久er99热精品一区二区 | 免费福利影院 | 欧美伊人网 | www.亚洲黄色 | 亚洲1级片 | 高清不卡一区二区三区 | 精品视频123区在线观看 | 狠狠干狠狠艹 | 亚洲精品成人在线 | 国产精品免费久久久久久久久久中文 | 久久九九影院 | 99re国产视频 | 午夜精品久久久久久久99水蜜桃 | 中文字幕精品久久 | 五月天欧美精品 | 99久久精品网 | 日本三级全黄少妇三2023 | 天天综合色 | 色婷婷激情综合 | 久久精品8| 欧美一级免费黄色片 | 五月婷婷在线观看 | 国产日韩欧美在线观看视频 | 天天干天天上 | 免费观看成人网 | 免费精品久久久 | 国产在线播放不卡 | 久久久综合色 | 美女网站免费福利视频 | 欧美精品久久久久久久亚洲调教 | 在线观看一区视频 | 日韩视频一区二区 | 日韩动态视频 | 国产精品久久久久久一二三四五 | 三级午夜片 | 国产精品视频永久免费播放 | 国产精品av免费 | 色综合五月天 | 国产免费高清 | 欧美大片aaa | 人人干人人上 | 国产精品九九九九九 | av电影一区二区 | 91丝袜美腿 | 久久精品国产v日韩v亚洲 | 在线观看黄网站 | 国内精品久久天天躁人人爽 | 久久久久久久久精 | 92国产精品久久久久首页 | 天堂av官网 | 久久久精品亚洲 | av视屏在线| 91自拍视频在线观看 | 视频在线99 | 在线观看视频国产 | 国际精品久久 | 精品久久一二三区 | 亚洲最新视频在线 | 亚洲精品国产精品乱码不99热 | 97超碰资源网 | 96在线| 天天色天天搞 | 狠狠干天天操 | 一区二区三区在线不卡 | 91传媒在线看| 久久国产亚洲 | 天天操天天干天天爱 | 天天综合婷婷 | 色综合天天在线 | av免费看av | 亚洲午夜久久久久久久久电影网 | 久操中文字幕在线观看 | 天天曰天天 | www.91国产 | 伊人看片 | 久久精品一 | 夜夜天天干 | 日本巨乳在线 | 中文字幕第一页在线 | 国产麻豆视频在线观看 | 国产一级二级在线观看 | 日韩簧片在线观看 | 国产黄色电影 | 欧美巨乳网 | 99热免费在线 | 伊人电影天堂 | 日韩精品久久久久久 | 欧美激情视频一二区 | 在线精品视频免费观看 | 在线 视频 亚洲 | 国产成人精品一区二区三区福利 | 日本中文字幕观看 | 一区二区三区 中文字幕 | 999久久国精品免费观看网站 | 日韩在线观看精品 | 色婷婷综合久久久 | 亚洲精品国产日韩 | 亚洲视频 在线观看 | 亚洲国产精品500在线观看 | 99精品在线视频播放 | 中文字幕在线观看免费高清完整版 | 午夜精品福利一区二区三区蜜桃 | 亚洲va在线va天堂 | 日韩精品在线免费观看 | 精品国产综合区久久久久久 | www.香蕉视频| 99日精品 | 久久久高清一区二区三区 | 国产精品资源在线 | 五月天久久激情 | 久久久精品国产一区二区电影四季 | 国产视频在线免费 | 人人躁 | 激情深爱.com | 亚洲国产mv| 久久激情五月激情 | 久久久免费 | 欧洲精品一区二区 | 日韩免费播放 | 久久久91精品国产 | 欧美电影黄色 | 91成人精品一区在线播放 | 欧美作爱视频 | 欧美另类tv | 免费视频97 | 日韩欧美视频在线 | 色五丁香 | av免费观看在线 | 日韩精品中文字幕在线观看 | 中文字幕中文 | 国产精品乱码久久 | 国产亚洲久久 | 日韩有色 | 亚洲极色 | 高清av免费一区中文字幕 | 国产精品黄 | 久久久久黄 | 国产美女在线免费观看 | 久草在线中文视频 | 国产精品视频大全 | 在线国产欧美 | 日韩精品一区二区三区免费视频观看 | 免费影视大全推荐 | www视频在线播放 | 亚洲专区一二三 | av三级在线免费观看 | 99一区二区三区 | 国产在线观看午夜 | 久久草草热国产精品直播 | 免费看的黄网站软件 | 国产一区在线视频 | www.av免费 | 一区二区三区四区五区在线 | 超碰人人超| 天天干.com | 狠狠色噜噜狠狠狠狠2021天天 | 久久你懂的 | 九月婷婷综合网 | 久久国产午夜精品理论片最新版本 | 在线天堂日本 | 在线观看免费国产小视频 | 超碰免费在线公开 | 国产精品私人影院 | 99久久久| 高清av中文字幕 | 国产在线永久 | 国产69久久 | 久久精品视频免费 | 超碰在线天天 | 天天爽夜夜爽人人爽曰av | 免费精品视频在线观看 | 久久电影色 | 97在线超碰 | 色窝资源 | 欧美色图视频一区 | 日韩毛片在线播放 | 在线免费色视频 | 91中文字幕网 | 最新av观看 | 亚洲精品视频在线播放 | 午夜久久福利视频 | www.成人久久 | www亚洲精品 | 伊人中文字幕在线 | 综合久色| 成av人电影 | 成人作爱视频 | 一区二区精 | 成人欧美一区二区三区在线观看 | 麻豆视频免费入口 | 欧美一级乱黄 | 亚洲热久久 | 丁香花在线视频观看免费 | 黄网站色视频 | 3d黄动漫免费看 | 午夜视频在线瓜伦 | 婷香五月 | 国产精品网址在线观看 | 亚洲精品合集 | 日本精品久久久久影院 | 夜夜爽天天爽 | 69精品人人人人 | 黄色亚洲大片免费在线观看 | 国产日韩三级 | 久久久精品网站 | 亚洲乱码久久 | 久久免费视频国产 | 国产美女精品人人做人人爽 | 国产免费久久久久 | 久操视频在线观看 | 91色九色| 国产不卡视频 | 又湿又紧又大又爽a视频国产 | 婷婷六月综合网 | 久久亚洲二区 | 69国产成人综合久久精品欧美 | 国产一区成人 | wwwwww国产| 92精品国产成人观看免费 | 综合五月婷婷 | 看片的网址 | 在线观看一区视频 | 69绿帽绿奴3pvideos | 波多野结衣在线中文字幕 | 国产精品久久久免费 | 久久久av电影 | 国产成人精品久 | 久久系列| 国产91成人在在线播放 | 视频在线一区二区三区 | 天天摸夜夜添 | www日韩在线| 在线亚洲成人 | 亚洲精品天天 | 国产精品手机看片 | 综合网久久 | 日韩欧美一区二区三区视频 | 四虎影院在线观看av | 777视频在线观看 | 欧美一区二区三区在线 | 中文字幕资源网 国产 | 国产精品高清免费在线观看 | 免费在线观看亚洲视频 | 二区在线播放 | 超碰大片 | 开心色停停 | 91精品久久久久久综合五月天 | 成年人国产在线观看 | 精品国产伦一区二区三区观看说明 | 91九色视频观看 | 亚洲美女免费精品视频在线观看 | 国产在线国产 | 国产中文欧美日韩在线 | 五月婷网站 | 国产精品久久久久永久免费看 | 日韩久久精品一区 |