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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

vsomeip源码梳理 -- Event订阅流程

發(fā)布時間:2023/12/3 综合教程 57 生活家
生活随笔 收集整理的這篇文章主要介紹了 vsomeip源码梳理 -- Event订阅流程 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

本文基于vsomeip 3.1.20.3總結(jié)而成
源碼地址:https://github.com/GENIVI/vsomeip.git

本文主要涉及vsomeip庫中的如下代碼:

在vsomeip中,提供了一個event類來實現(xiàn)SOME/IP協(xié)議中的事件所包含的信息與功能,在之前的寫的demo篇中有介紹過事件的基礎(chǔ)使用,包括事件的注冊,訂閱以及發(fā)送三個功能,那這一篇文章繼續(xù)來看看event的處理具體處理流程。

事件訂閱(request_event & subscribe )

在跟蹤源碼之前,先看下一個客戶端訂閱某個事件組的API介紹,主要是兩個函數(shù),request_event與subscribe, 函數(shù)定義在application.hpp中

//注冊該應(yīng)用模塊為純事件或者屬性事件的使用者
virtual void request_event(service_t _service, instance_t _instance,event_t _event, const std::set<eventgroup_t> &_eventgroups,event_type_e _type = event_type_e::ET_EVENT,reliability_type_e _reliability = reliability_type_e::RT_UNKNOWN) = 0;//訂閱事件組,該函數(shù)需在request_event后調(diào)用。
virtual void subscribe(service_t _service, instance_t _instance,eventgroup_t _eventgroup, major_version_t _major = DEFAULT_MAJOR,event_t _event = ANY_EVENT) = 0;

這兩個函數(shù)就是這個模塊跟蹤的入口了,看代碼,兩個函數(shù)的實現(xiàn)在application_impl.cpp中,按照注釋的順序,我們先從request_event函數(shù)開始看

void application_impl::request_event(service_t _service, instance_t _instance,event_t _event, const std::set<eventgroup_t> &_eventgroups,event_type_e _type, reliability_type_e _reliability) {//這里的routing_在之前的init流程分析過,可能指向host模式的rtm_impl,也//可能指向proxy模式的rtm_proxyif (routing_)routing_->register_event(client_,_service, _instance,_event,_eventgroups, _type, _reliability,std::chrono::milliseconds::zero(), false, true,nullptr,false);
}

還是先跟蹤host模式的路由實現(xiàn),因為proxy的實現(xiàn)相對來說比較簡單。app_模塊中的request_event啥也沒干,直接交給路由模塊的register_event函數(shù)處理了。

host路由中的register_event實現(xiàn)

/** app調(diào)用register_event傳入的參數(shù):* _change_resets_cycle= false* _update_on_change = true* _epsilon_change_func = null* _is_provided = false* _is_shadow = false* _is_cache_placeholder = false*/
void routing_manager_impl::register_event(client_t _client,service_t _service, instance_t _instance,event_t _notifier,const std::set<eventgroup_t> &_eventgroups, const event_type_e _type,reliability_type_e _reliability,std::chrono::milliseconds _cycle, bool _change_resets_cycle,bool _update_on_change,epsilon_change_func_t _epsilon_change_func,bool _is_provided, bool _is_shadow, bool _is_cache_placeholder) {//從APP中的event緩存map中查找是否已經(jīng)存在對應(yīng)的event實例auto its_event = find_event(_service, _instance, _notifier);bool is_first(false);//判斷是否為首次注冊if (its_event) {if (!its_event->has_ref(_client, _is_provided)) {is_first = true;}} else {is_first = true;}if (is_first) {//首次注冊的情況下,調(diào)用父類的register_event方法routing_manager_base::register_event(_client,_service, _instance,_notifier,_eventgroups, _type, _reliability,_cycle, _change_resets_cycle, _update_on_change,_epsilon_change_func, _is_provided, _is_shadow,_is_cache_placeholder);}//忽略日志打印代碼
}

上面的流程也比較簡單,就是根據(jù)服務(wù)實例以及事件ID來判斷事件是否已經(jīng)注冊過,已經(jīng)注冊過的情況下,就不處理該次注冊動作了,接著看rtm_base中的流程,因為rtm_base ::register_event中源碼挺多的,而且host與proxy共用邏輯,這里先對整個流程畫了一個流程圖簡述一下

void routing_manager_base::register_event(client_t _client,service_t _service, instance_t _instance,event_t _notifier,const std::set<eventgroup_t> &_eventgroups,const event_type_e _type,reliability_type_e _reliability,std::chrono::milliseconds _cycle, bool _change_resets_cycle,bool _update_on_change,epsilon_change_func_t _epsilon_change_func,bool _is_provided, bool _is_shadow, bool _is_cache_placeholder) {std::lock_guard<std::mutex> its_registration_lock(event_registration_mutex_);auto determine_event_reliability = [this, &_service, &_instance,&_notifier, &_reliability]() {reliability_type_e its_reliability =configuration_->get_event_reliability(_service, _instance, _notifier);if (its_reliability != reliability_type_e::RT_UNKNOWN) {// event was explicitly configured -> overwrite value passed via APIreturn its_reliability;} else if (_reliability != reliability_type_e::RT_UNKNOWN) {// use value provided via APIreturn _reliability;} else { // automatic mode, user service' reliabilityreturn configuration_->get_service_reliability(_service, _instance);}};//從已注冊的事件表中查找當前事件是否已經(jīng)存在std::shared_ptr<event> its_event = find_event(_service, _instance, _notifier);bool transfer_subscriptions_from_any_event(false);if (its_event) {//事件已經(jīng)注冊過,判斷已注冊的事件是否是占位事件if (!its_event->is_cache_placeholder()) {if (_type == its_event->get_type()|| its_event->get_type() == event_type_e::ET_UNKNOWN
#ifdef VSOMEIP_ENABLE_COMPAT|| (its_event->get_type() == event_type_e::ET_EVENT&& _type == event_type_e::ET_SELECTIVE_EVENT)|| (its_event->get_type() == event_type_e::ET_SELECTIVE_EVENT&& _type == event_type_e::ET_EVENT && _is_provided)
#endif) {//非占位事件,且事件類型一致則根據(jù)傳入的參數(shù)更新事件信息
#ifdef VSOMEIP_ENABLE_COMPATif (its_event->get_type() == event_type_e::ET_EVENT&& _type == event_type_e::ET_SELECTIVE_EVENT) {its_event->set_type(_type);}
#endifif (_is_provided) {its_event->set_provided(true);its_event->set_reliability(determine_event_reliability());}if (_is_shadow && _is_provided) {its_event->set_shadow(_is_shadow);}//注冊該事件的客戶端為host路由應(yīng)用,強制將事件標記為非影子事件if (_client == host_->get_client() && _is_provided) {its_event->set_shadow(false);its_event->set_update_on_change(_update_on_change);}//更新事件的事件組信息for (auto eg : _eventgroups) {its_event->add_eventgroup(eg);}transfer_subscriptions_from_any_event = true;} else {
#ifdef VSOMEIP_ENABLE_COMPATif (!(its_event->get_type() == event_type_e::ET_SELECTIVE_EVENT&& _type == event_type_e::ET_EVENT))
#endifVSOMEIP_ERROR << "Event registration update failed. ""Specified arguments do not match existing registration.";}} else {//該事件之前已經(jīng)作為占位事件注冊過,這里將占位事件變?yōu)檎鎸嵶允录?/并更新事件信息if (_type != event_type_e::ET_FIELD) {// don't cache payload for non-fieldsits_event->unset_payload(true);}if (_is_shadow && _is_provided) {its_event->set_shadow(_is_shadow);}if (_client == host_->get_client() && _is_provided) {its_event->set_shadow(false);its_event->set_update_on_change(_update_on_change);}its_event->set_type(_type);its_event->set_reliability(determine_event_reliability());its_event->set_provided(_is_provided);its_event->set_cache_placeholder(false);std::shared_ptr<serviceinfo> its_service = find_service(_service, _instance);if (its_service) {its_event->set_version(its_service->get_major());}if (_eventgroups.size() == 0) { // No eventgroup specifiedstd::set<eventgroup_t> its_eventgroups;its_eventgroups.insert(_notifier);its_event->set_eventgroups(its_eventgroups);} else {for (auto eg : _eventgroups) {its_event->add_eventgroup(eg);}}its_event->set_epsilon_change_function(_epsilon_change_func);its_event->set_change_resets_cycle(_change_resets_cycle);its_event->set_update_cycle(_cycle);}} else {//該事件之前沒有注冊過,則創(chuàng)建新的event對象its_event = std::make_shared<event>(this, _is_shadow);its_event->set_service(_service);its_event->set_instance(_instance);its_event->set_event(_notifier);its_event->set_type(_type);its_event->set_reliability(determine_event_reliability());its_event->set_provided(_is_provided);its_event->set_cache_placeholder(_is_cache_placeholder);std::shared_ptr<serviceinfo> its_service = find_service(_service, _instance);if (its_service) {its_event->set_version(its_service->get_major());}if (_eventgroups.size() == 0) { // No eventgroup specifiedstd::set<eventgroup_t> its_eventgroups;its_eventgroups.insert(_notifier);its_event->set_eventgroups(its_eventgroups);} else {its_event->set_eventgroups(_eventgroups);}//當前注冊的是影子事件且epsilon變化事件處理函數(shù)為空,epsilon變化的意思是:僅當與最后一個值的差異大于某個閾值時才發(fā)送更新。if (_is_shadow && !_epsilon_change_func) {std::shared_ptr<cfg::debounce> its_debounce= configuration_->get_debounce(_service, _instance, _notifier);if (its_debounce) {//省略了部分日志代碼//根據(jù)配置文件中的debounce配置信息構(gòu)建新的_epsilon_change_func函數(shù)_epsilon_change_func = [its_debounce](const std::shared_ptr<payload> &_old,const std::shared_ptr<payload> &_new) {bool is_changed(false), is_elapsed(false);// Check whether we should forward because of changed dataif (its_debounce->on_change_) {length_t its_min_length, its_max_length;if (_old->get_length() < _new->get_length()) {its_min_length = _old->get_length();its_max_length = _new->get_length();} else {its_min_length = _new->get_length();its_max_length = _old->get_length();}// Check whether all additional bytes (if any) are excludedfor (length_t i = its_min_length; i < its_max_length; i++) {auto j = its_debounce->ignore_.find(i);// A change is detected when an additional byte is not// excluded at all or if its exclusion does not cover// all its bits.if (j == its_debounce->ignore_.end() || j->second != 0xFF) {is_changed = true;break;}}if (!is_changed) {const byte_t *its_old = _old->get_data();const byte_t *its_new = _new->get_data();for (length_t i = 0; i < its_min_length; i++) {auto j = its_debounce->ignore_.find(i);if (j == its_debounce->ignore_.end()) {if (its_old[i] != its_new[i]) {is_changed = true;break;}} else if (j->second != 0xFF) {if ((its_old[i] & ~(j->second)) != (its_new[i] & ~(j->second))) {is_changed = true;break;}}}}}if (its_debounce->interval_ > -1) {// Check whether we should forward because of the elapsed time since// we did last timestd::chrono::steady_clock::time_point its_current= std::chrono::steady_clock::now();long elapsed = std::chrono::duration_cast<std::chrono::milliseconds>(its_current - its_debounce->last_forwarded_).count();is_elapsed = (its_debounce->last_forwarded_ == (std::chrono::steady_clock::time_point::max)()|| elapsed >= its_debounce->interval_);if (is_elapsed || (is_changed && its_debounce->on_change_resets_interval_))its_debounce->last_forwarded_ = its_current;}return (is_changed || is_elapsed);};} else {//構(gòu)造一個空的函數(shù)_epsilon_change_func = [](const std::shared_ptr<payload> &_old,const std::shared_ptr<payload> &_new) {(void)_old;(void)_new;return true;};}}//設(shè)置觸發(fā)事件信息its_event->set_epsilon_change_function(_epsilon_change_func);its_event->set_change_resets_cycle(_change_resets_cycle);its_event->set_update_cycle(_cycle);its_event->set_update_on_change(_update_on_change);if (_is_provided) {transfer_subscriptions_from_any_event = true;}}if (transfer_subscriptions_from_any_event) {// check if someone subscribed to ANY_EVENT and the subscription// was stored in the cache placeholder. Move the subscribers// into new event//獲取指定服務(wù)實例中的任意事件的event對象std::shared_ptr<event> its_any_event =find_event(_service, _instance, ANY_EVENT);if (its_any_event) {//當前緩存中存在任意事件的event,獲取該event所在的事件組std::set<eventgroup_t> any_events_eventgroups =its_any_event->get_eventgroups();//遍歷當前注冊事件的所在事件組for (eventgroup_t eventgroup : _eventgroups) {//任意事件的事件組集中包含了當前注冊事件的事件組auto found_eg = any_events_eventgroups.find(eventgroup);if (found_eg != any_events_eventgroups.end()) {//獲取訂閱任意事件所在事件組的客戶端ID集std::set<client_t> its_any_event_subscribers =its_any_event->get_subscribers(eventgroup);//更新當前事件的訂閱器信息for (const client_t subscriber : its_any_event_subscribers) {its_event->add_subscriber(eventgroup, subscriber, true);}}}}}//事件為真實事件,添加該客戶端的引用if (!its_event->is_cache_placeholder()) {its_event->add_ref(_client, _is_provided);}//更新事件組信息for (auto eg : _eventgroups) {std::shared_ptr<eventgroupinfo> its_eventgroupinfo= find_eventgroup(_service, _instance, eg);if (!its_eventgroupinfo) {its_eventgroupinfo = std::make_shared<eventgroupinfo>();its_eventgroupinfo->set_service(_service);its_eventgroupinfo->set_instance(_instance);its_eventgroupinfo->set_eventgroup(eg);std::lock_guard<std::mutex> its_lock(eventgroups_mutex_);eventgroups_[_service][_instance][eg] = its_eventgroupinfo;}its_eventgroupinfo->add_event(its_event);}std::lock_guard<std::mutex> its_lock(events_mutex_);//更新已注冊事件信息events_[_service][_instance][_notifier] = its_event;
}

到這里,我們基本就清楚了host模式的vsomeip app中的request_event的主要作用是創(chuàng)建了eventgroupinfo, event兩個實例的共享指針 ,并將其添加到模塊中的eventgroups_ 以及events_表中

proxy路由中的request_event實現(xiàn)

proxy中的request_event流程比較簡單,首先是app_模塊中的request_event什么也沒做,直接調(diào)用了rtm_proxy中的register_event事件:

 void routing_manager_proxy::register_event(client_t _client,service_t _service, instance_t _instance,event_t _notifier,const std::set<eventgroup_t> &_eventgroups, const event_type_e _type,reliability_type_e _reliability,std::chrono::milliseconds _cycle, bool _change_resets_cycle,bool _update_on_change,  epsilon_change_func_t _epsilon_change_func,bool _is_provided, bool _is_shadow, bool _is_cache_placeholder) {(void)_is_shadow;(void)_is_cache_placeholder;//創(chuàng)建一個事件注冊器const event_data_t registration = {_service,_instance,_notifier,_type,_reliability,_is_provided,_eventgroups};bool is_first(false);{//從當前已經(jīng)pending的事件注冊器中年查找對應(yīng)的事件注冊器是否已經(jīng)存在,如果不存在的情況下,就是該事件的首次注冊std::lock_guard<std::mutex> its_lock(state_mutex_);is_first = pending_event_registrations_.find(registration)== pending_event_registrations_.end();
#ifndef VSOMEIP_ENABLE_COMPATif (is_first) {pending_event_registrations_.insert(registration);}
#else......
#endif}//首次注冊的情況下,調(diào)用rtm_base創(chuàng)建event實例與eventgroup實例,并將其加入到緩存map中,rtm_base::register_event中的邏輯參考host章節(jié)中的流程。if (is_first || _is_provided) {routing_manager_base::register_event(_client,_service, _instance,_notifier,_eventgroups, _type, _reliability,_cycle, _change_resets_cycle, _update_on_change,_epsilon_change_func,_is_provided);}{std::lock_guard<std::mutex> its_lock(state_mutex_);//如果當前應(yīng)用狀態(tài)已注冊,且事件為第一次注冊,則發(fā)送命令到host端實現(xiàn)register_event流程if (state_ == inner_state_type_e::ST_REGISTERED && is_first) {send_register_event(client_, _service, _instance,_notifier, _eventgroups, _type, _reliability, _is_provided);}}
}繼續(xù)跟蹤send_register_event函數(shù),發(fā)現(xiàn)其中的邏輯比較簡單,就是根據(jù)傳入的事件信息拼包,然后通過unix域socket的方式發(fā)送類型為VSOMEIP_REGISTER_EVENT命令到host端:```cpp
void routing_manager_proxy::send_register_event(client_t _client,service_t _service, instance_t _instance,event_t _notifier,const std::set<eventgroup_t> &_eventgroups, const event_type_e _type,reliability_type_e _reliability,bool _is_provided) {......byte_t *its_command = new byte_t[its_eventgroups_size];uint32_t its_size = static_cast<std::uint32_t>(its_eventgroups_size)- VSOMEIP_COMMAND_HEADER_SIZE;its_command[VSOMEIP_COMMAND_TYPE_POS] = VSOMEIP_REGISTER_EVENT;//省略拼包邏輯std::size_t i = 9;for (auto eg : _eventgroups) {std::memcpy(&its_command[VSOMEIP_COMMAND_PAYLOAD_POS + i], &eg,sizeof(eventgroup_t));i += sizeof(eventgroup_t);}{std::lock_guard<std::mutex> its_lock(sender_mutex_);//這個sender_在init流程的分析中跟過,它就是一個類型為local_client_endpoint的對象if (sender_) {sender_->send(its_command, static_cast<std::uint32_t>(its_eventgroups_size));}}if (_is_provided) {//打印日志}delete[] its_command;
}

總結(jié)一下:vsomeip的應(yīng)用如果需要在某個事件觸發(fā)的時候得到消息的回調(diào),就必須通過其提供的request_event方法來請求事件,請求事件的邏輯又會根據(jù)路由的host角色與proxy角色的不同來處理邏輯,兩種角色的相同部分邏輯就是會在自己的模塊中創(chuàng)建需要關(guān)聯(lián)的event與eventgroup的信息,不同的部分在于proxy端還需要將它所訂閱的事件信息發(fā)送給host路由,當host路由收到網(wǎng)段內(nèi)其他端的事件通知時,它就會將其通知給對應(yīng)的proxy端。

走完request_event的流程,接下來看subscribe,客戶端訂閱事件需要request_event與subscribe搭配才能收到事件。

Host路由的subscribe實現(xiàn)

host路由的subscribe函數(shù)實現(xiàn)再routing_maanger_impl.cpp文件中,函數(shù)定義如下:

void routing_manager_impl::subscribe(client_t _client, uid_t _uid, gid_t _gid,service_t _service, instance_t _instance, eventgroup_t _eventgroup,major_version_t _major, event_t _event)

該函數(shù)實現(xiàn),首先根據(jù)服務(wù)實例信息查看本地服務(wù)中是否有提供的該服務(wù)實例的客戶端ID

  const client_t its_local_client = find_local_client(_service, _instance);

這里有幾個判斷條件,匯總?cè)缦?#xff1a;

  1. 根據(jù)訂閱的事件信息,從本地服務(wù)的緩存中查找是否存在提供該事件的客戶端
  2. 提供事件信息的客戶端是當前的app模塊(get_client() == its_local_client),那么直接通過app中的on_subscription觸發(fā)回調(diào),告知訂閱成功,app模塊的回調(diào)函數(shù)如果允許訂閱,則通過rtm_stub發(fā)送subscribe_ack命令,否則發(fā)送subscribe_nack
  3. 如果提供事件信息的客戶端不是當前app模塊,且本地服務(wù)中沒有找到有提供該事件的實例,那么就通過SD模塊去發(fā)起遠程訂閱(網(wǎng)段內(nèi)廣播訂閱)
  4. 如果提供事件信息的客戶端不是當前app模塊,但是在本地服務(wù)中找到存在了事件的服務(wù)實例,通過rtm_stub告知對應(yīng)的app端有其他的客戶端向它的服務(wù)中包含的事件發(fā)起了訂閱(unix域通信)。

下面是代碼

//簡化后的代碼:訂閱的服務(wù)實例類型為本地服務(wù)
void routing_manager_impl::subscribe(client_t _client, uid_t _uid, gid_t _gid,service_t _service, instance_t _instance, eventgroup_t _eventgroup,major_version_t _major, event_t _event) {const client_t its_local_client = find_local_client(_service, _instance);if (get_client() == its_local_client) {auto self = shared_from_this();//調(diào)用application模塊的on_subscription方法host_->on_subscription(_service, _instance, _eventgroup, _client, _uid, _gid, true,[this, self, _client, _uid, _gid, _service, _instance, _eventgroup,_event, _major](const bool _subscription_accepted) {(void) ep_mgr_->find_or_create_local(_client);//如果當前app拒絕訂閱,則通過stub發(fā)送nack給到訂閱事件請求的客戶端,并返回if (!_subscription_accepted) {stub_->send_subscribe_nack(_client, _service, _instance, _eventgroup, _event);return;} else {//如果application中接受訂閱,則發(fā)送ackstub_->send_subscribe_ack(_client, _service, _instance, _eventgroup, _event);}//如果application中接受訂閱,調(diào)用rtm_base的subscribe,創(chuàng)建事件的訂閱器routing_manager_base::subscribe(_client, _uid, _gid, _service, _instance, _eventgroup, _major, _event);});} 

提供事件信息的客戶端不是當前app模塊

if (discovery_) {std::set<event_t> its_already_subscribed_events;std::unique_lock<std::mutex> its_critical(remote_subscription_state_mutex_);//添加訂閱器bool inserted = insert_subscription(_service, _instance, _eventgroup,_event, _client, &its_already_subscribed_events);if (inserted) {//當前為路由模塊if (0 == its_local_client) {handle_subscription_state(_client, _service, _instance, _eventgroup, _event);its_critical.unlock();static const ttl_t configured_ttl(configuration_->get_sd_ttl());//觸發(fā)一次事件通知notify_one_current_value(_client, _service, _instance,_eventgroup, _event, its_already_subscribed_events);//通過SD模塊,廣播遠程訂閱信息auto its_info = find_eventgroup(_service, _instance, _eventgroup);if (its_info) {discovery_->subscribe(_service, _instance, _eventgroup,_major, configured_ttl,its_info->is_selective() ? _client : VSOMEIP_ROUTING_CLIENT,its_info);}} else {//非路由模塊its_critical.unlock();if (is_available(_service, _instance, _major)) {//發(fā)送命令類型為VSOMEIP_SUBSCRIBE的報文給到代理端,rtm_proxy端的on_message函數(shù)最終//會收到該報文stub_->send_subscribe(ep_mgr_->find_local(_service, _instance),_client, _service, _instance, _eventgroup, _major, _event,PENDING_SUBSCRIPTION_ID);}}}if (get_client() == _client) {std::lock_guard<std::mutex> ist_lock(pending_subscription_mutex_);subscription_data_t subscription = {_service, _instance, _eventgroup, _major, _event, _uid, _gid};pending_subscriptions_.insert(subscription);}} else {VSOMEIP_ERROR<< "SOME/IP eventgroups require SD to be enabled!";}}

上面的代碼中,我們看到幾個子流程:

  • 給對應(yīng)客戶端發(fā)subscribe nack命令:routing_manager_stub::send_subscribe_nack
  • 給對應(yīng)客戶端發(fā)subscribe ack命令:routing_manager_stub::send_subscribe_ack
  • 給對應(yīng)客戶端發(fā)訂閱命令:routing_manager_stub::send_subscribe
  • 創(chuàng)建訂閱器:routing_manager_base::subscribe
  • 調(diào)用sd模塊的subscribe方法

我們摘出來一個個的單個看
routing_manager_stub::send_subscribe_nack 這個函數(shù)調(diào)用的場景在訂閱時提供事件信息的客戶端是當前的app模塊,且app模塊中通過register_subscription_handler注冊了訂閱操作函數(shù),此時在訂閱回調(diào)觸發(fā)時,如果app模塊拒絕客戶端訂閱,則會發(fā)送一個nack給到訂閱的客戶端。該函數(shù)定義如下:

void routing_manager_stub::send_subscribe_nack(client_t _client, service_t _service,instance_t _instance, eventgroup_t _eventgroup, event_t _event) {//找到目標client的endpoint對象std::shared_ptr<endpoint> its_endpoint = host_->find_local(_client);if (its_endpoint) {byte_t its_command[VSOMEIP_SUBSCRIBE_NACK_COMMAND_SIZE];uint32_t its_size = VSOMEIP_SUBSCRIBE_NACK_COMMAND_SIZE- VSOMEIP_COMMAND_HEADER_SIZE;client_t this_client = get_client();its_command[VSOMEIP_COMMAND_TYPE_POS] = VSOMEIP_SUBSCRIBE_NACK;//省略了拼包邏輯//將該命令報文發(fā)送給到對應(yīng)的client端its_endpoint->send(&its_command[0], sizeof(its_command));}
}

這個消息會被客戶端的routing_manager_proxy中消化:

void routing_manager_proxy::on_message(const byte_t *_data, length_t _size,endpoint *_receiver, const boost::asio::ip::address &_destination,client_t _bound_client,credentials_t _credentials,const boost::asio::ip::address &_remote_address,std::uint16_t _remote_port) {//省略代碼case VSOMEIP_SUBSCRIBE_NACK://省略代碼//直接轉(zhuǎn)到了該函數(shù)進行處理on_subscribe_nack(its_subscriber, its_service, its_instance, its_eventgroup, its_event);
}void routing_manager_proxy::on_subscribe_nack(client_t _client,service_t _service, instance_t _instance, eventgroup_t _eventgroup, event_t _event) {(void)_client;//如果是任意類型的事件訂閱被拒絕,則通知對應(yīng)的事件組中所有事件的服務(wù)模塊該事件訂閱被拒絕if (_event == ANY_EVENT) {auto its_eventgroup = find_eventgroup(_service, _instance, _eventgroup);if (its_eventgroup) {for (const auto& its_event : its_eventgroup->get_events()) {host_->on_subscription_status(_service, _instance, _eventgroup, its_event->get_event(), 0x7 /*Rejected*/);}}} else {//通知client端的app模塊,該事件被拒絕host_->on_subscription_status(_service, _instance, _eventgroup, _event, 0x7 /*Rejected*/);}
}

最終,如果client的app有注冊訂閱狀態(tài)的操作函數(shù),則能夠監(jiān)聽到訂閱狀態(tài)的回調(diào)

app_->register_subscription_status_handler

routing_manager_stub::send_subscribe_ack的流程與nack的流程基本大同小異, 這里不再復(fù)述了。

routing_manager_stub::send_subscribe : 這個給客戶端發(fā)送訂閱的調(diào)用場景客戶端訂閱時,提供訂閱事件的服務(wù)實例由另外一個代理客戶端提供的,所以這里拼了一個VSOMEIP_SUBSCRIBE的包發(fā)給rtm_proxy去處理

調(diào)用sd模塊的subscribe方法:SD模塊的業(yè)務(wù)實現(xiàn)在service_discovery_impl中,刨去細枝末節(jié),來跟一下整個subscribe的流程如下:

void
service_discovery_impl::subscribe(service_t _service, instance_t _instance,eventgroup_t _eventgroup, major_version_t _major,ttl_t _ttl, client_t _client,const std::shared_ptr<eventgroupinfo> &_info) {......send_subscription(its_subscription,_service, _instance, _eventgroup,_client);
}void
service_discovery_impl::send_subscription(const std::shared_ptr<subscription> &_subscription,const service_t _service, const instance_t _instance,const eventgroup_t _eventgroup,const client_t _client) {auto its_reliable = _subscription->get_endpoint(true);auto its_unreliable = _subscription->get_endpoint(false);boost::asio::ip::address its_address;get_subscription_address(its_reliable, its_unreliable, its_address);if (!its_address.is_unspecified()) {......if (its_data.entry_) {auto its_current_message = std::make_shared<message_impl>();std::vector<std::shared_ptr<message_impl> > its_messages;its_messages.push_back(its_current_message);add_entry_data(its_messages, its_data);//序列化數(shù)據(jù),然后發(fā)送報文serialize_and_send(its_messages, its_address);} }   
}//該函數(shù)中省略了部分代碼
bool
service_discovery_impl::serialize_and_send(const std::vector<std::shared_ptr<message_impl> > &_messages,const boost::asio::ip::address &_address) {if (!_address.is_unspecified()) {std::lock_guard<std::mutex> its_lock(serialize_mutex_);for (const auto &m : _messages) {if (m->has_entry()) {......//序列化報文數(shù)據(jù)if (serializer_->serialize(m.get())) {if (host_->send_via_sd(endpoint_definition::get(_address, port_,reliable_, m->get_service(), m->get_instance()),serializer_->get_data(), serializer_->get_size(),port_)) {//新增session idincrement_session(_address);}} }}}
}
//上面函數(shù)走完了,業(yè)務(wù)層面的就下完了,后面是網(wǎng)絡(luò)發(fā)送相關(guān)的邏輯,host_指針類型是routing_manager_impl, send_via_sd實現(xiàn)在rtm_impl中。
bool routing_manager_impl::send_via_sd(const std::shared_ptr<endpoint_definition> &_target,const byte_t *_data, uint32_t _size, uint16_t _sd_port) {std::shared_ptr<endpoint> its_endpoint =ep_mgr_impl_->find_server_endpoint(_sd_port,_target->is_reliable());return its_endpoint->send_to(_target, _data, _size);
}//上面的its_endpoint實際是udp_server_endpoint_impl類型的共享指針
bool udp_server_endpoint_impl::send_to(const std::shared_ptr<endpoint_definition> _target,const byte_t *_data, uint32_t _size) {std::lock_guard<std::mutex> its_lock(mutex_);endpoint_type its_target(_target->get_address(), _target->get_port());return send_intern(its_target, _data, _size);
}//send_intern實現(xiàn)在父類server_endpoint_impl中,是一個模板函數(shù)
template<typename Protocol>
bool server_endpoint_impl<Protocol>::send_intern(endpoint_type _target, const byte_t *_data, uint32_t _size) {......// STEP 10: restart timer with current departure timetarget_train->departure_timer_->expires_from_now(target_train->departure_);target_train->departure_timer_->async_wait(std::bind(&server_endpoint_impl<Protocol>::flush_cbk,this->shared_from_this(), _target,target_train, std::placeholders::_1));
}//接下來flush_cbk觸發(fā)執(zhí)行
template<typename Protocol>
void server_endpoint_impl<Protocol>::flush_cbk(endpoint_type _target,const std::shared_ptr<train>& _train, const boost::system::error_code &_error_code) {if (!_error_code) {(void) flush(_target, _train);}
}//調(diào)用了flush函數(shù)template<typename Protocol>
bool server_endpoint_impl<Protocol>::flush(endpoint_type _target,const std::shared_ptr<train>& _train){//buffer不為空的情況下,循環(huán)取出數(shù)據(jù)加入隊列if (!_train->buffer_->empty()){const queue_iterator_type target_queue_iterator = queues_.find(_target);if (target_queue_iterator != queues_.end()) {const bool queue_size_zero_on_entry(target_queue_iterator->second.second.empty());queue_train(target_queue_iterator, _train, queue_size_zero_on_entry);is_flushed = true;} }
}//將數(shù)據(jù)不斷發(fā)送到報文
template<typename Protocol>
void server_endpoint_impl<Protocol>::queue_train(const queue_iterator_type _queue_iterator,const std::shared_ptr<train>& _train,bool _queue_size_zero_on_entry) {...send_queued(_queue_iterator);
}//send_queued是一個虛函數(shù),由子類自行實現(xiàn),這里又走到了udp_server_endpoint_impl中, 發(fā)送流程基本就走完了。
void udp_server_endpoint_impl::send_queued(const queue_iterator_type _queue_iterator) {message_buffer_ptr_t its_buffer = _queue_iterator->second.second.front();......std::lock_guard<std::mutex> its_lock(unicast_mutex_);unicast_socket_.async_send_to(boost::asio::buffer(*its_buffer),_queue_iterator->first,std::bind(&udp_server_endpoint_base_impl::send_cbk,shared_from_this(),_queue_iterator,std::placeholders::_1,std::placeholders::_2));
}

proxy路由的subscribe實現(xiàn)

跟以往所有流程一樣,proxy的實現(xiàn)相對host來說要簡單很多了,proxy中的subscribe實現(xiàn)在routing_manager_proxy中

void routing_manager_proxy::subscribe(client_t _client, uid_t _uid, gid_t _gid, service_t _service,instance_t _instance, eventgroup_t _eventgroup, major_version_t _major,event_t _event) {......std::lock_guard<std::mutex> its_lock(state_mutex_);//服務(wù)可用,且當前app狀態(tài)為已注冊if (state_ == inner_state_type_e::ST_REGISTERED && is_available(_service, _instance, _major)) {send_subscribe(client_, _service, _instance, _eventgroup, _major, _event );}subscription_data_t subscription = { _service, _instance, _eventgroup, _major, _event, _uid, _gid};pending_subscriptions_.insert(subscription);
}void routing_manager_proxy::send_subscribe(client_t _client, service_t _service,instance_t _instance, eventgroup_t _eventgroup, major_version_t _major,event_t _event) {...its_command[VSOMEIP_COMMAND_TYPE_POS] = VSOMEIP_SUBSCRIBE;...client_t target_client = find_local_client(_service, _instance);if (target_client != VSOMEIP_ROUTING_CLIENT) {//發(fā)送命令包給該訂閱事件所在服務(wù)所屬的客戶端,存在于這種情況,同一個進程內(nèi)存在//一個路由app與多個proxy app, 其中某個proxy app提供了該訂閱所需的事件。//那么這個報文就發(fā)送到了另外一個app模塊的routing_manager_proxy中的on_message方法//處理auto its_target = ep_mgr_->find_or_create_local(target_client);its_target->send(its_command, sizeof(its_command));} else {//發(fā)送命令給到路由app,該命令由routing_manager_stub處理。std::lock_guard<std::mutex> its_lock(sender_mutex_);if (sender_) {sender_->send(its_command, sizeof(its_command));}}
}//第一種情況,proxy->proxy
void routing_manager_proxy::on_message(const byte_t *_data, length_t _size,endpoint *_receiver, const boost::asio::ip::address &_destination,client_t _bound_client,credentials_t _credentials,const boost::asio::ip::address &_remote_address,std::uint16_t _remote_port) {.......case VSOMEIP_SUBSCRIBE://1.觸發(fā)app中的subscription status回調(diào),如果app模塊允許訂閱,則回復(fù)ack, 否則回復(fù)nack//2.創(chuàng)建訂閱器
}//第二種情況,proxy->stub
void routing_manager_stub::on_message(const byte_t *_data, length_t _size,endpoint *_receiver, const boost::asio::ip::address &_destination,client_t _bound_client,credentials_t _credentials,const boost::asio::ip::address &_remote_address,std::uint16_t _remote_port) {case VSOMEIP_SUBSCRIBE://調(diào)用rtm_impl中的subscribe流程  host_->subscribe(its_client, its_sender_uid, its_sender_gid, its_service, its_instance,its_eventgroup, its_major, its_notifier);
}

總結(jié)

以上是生活随笔為你收集整理的vsomeip源码梳理 -- Event订阅流程的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

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

免费人成在线观看网站 | 午夜精品电影 | av 一区二区三区 | 国产又黄又爽无遮挡 | 久久毛片网 | 欧美精品九九99久久 | 91在线国产观看 | 91亚洲精品久久久中文字幕 | 97电影院网 | 中文字幕在线播放av | 日韩区欠美精品av视频 | 久草久草在线 | 精品九九九九 | 美女网站色免费 | 国产免费视频在线 | 国产亚洲精品久久久久久久久久 | 在线观看亚洲国产 | 激情五月综合网 | 亚洲精品乱码久久久久久蜜桃欧美 | 欧美日韩综合在线观看 | 99亚洲精品视频 | av在线成人 | 91免费高清观看 | 99精品影视 | 欧美激精品 | 久久艹久久 | 日韩免费观看视频 | 色中文字幕在线观看 | 亚洲精品国偷拍自产在线观看蜜桃 | 一区二区伦理电影 | 久久在线 | 天天干亚洲 | 三级av免费 | 中国一级片视频 | 91九色成人| 96av视频 | 免费观看mv大片高清 | 欧美色图东方 | 国产日韩精品一区二区三区在线 | 精品久久影院 | 久久国产综合视频 | 97色狠狠| 国产在线国偷精品产拍免费yy | 久久国产福利 | 日韩欧美视频在线观看免费 | 91精品国产91久久久久久三级 | 精品国产99国产精品 | 91久久国产综合精品女同国语 | 亚洲一区二区三区精品在线观看 | 国产精品资源在线观看 | 日本xxxx裸体xxxx17 | 特黄一级毛片 | 亚洲国产播放 | 日韩av免费在线看 | 国产破处在线视频 | 五月婷婷网站 | 成年人在线看片 | 亚洲精品视频偷拍 | 久久伊人八月婷婷综合激情 | 香蕉影院在线 | 国产 中文 日韩 欧美 | 97av超碰| a v在线视频 | 欧美日韩另类在线观看 | 久草视频在线新免费 | 韩国一区视频 | 亚洲第一香蕉视频 | 看片网站黄 | 91精品人成在线观看 | 国产精品久久久久久久久久尿 | 97精品超碰一区二区三区 | 午夜精品福利一区二区 | 三级在线国产 | 久久久久久久久久久久久久av | 国产精品99久久久久久宅男 | 四虎天堂 | 久久精品视频一 | 午夜aaaa| 999视频网| 中文在线a在线 | 国产午夜免费视频 | 四虎影视4hu4虎成人 | 日日夜夜狠狠 | 丝袜足交在线 | 五月天免费网站 | 日本少妇高清做爰视频 | 国产69精品久久99不卡的观看体验 | av不卡在线看 | 成人a视频 | 精品女同一区二区三区在线观看 | 久久99久久99精品免视看婷婷 | 五月天激情视频 | 四虎免费在线观看视频 | 精品国产一区二区三区四区vr | 天天天天色射综合 | 成人中文字幕+乱码+中文字幕 | 久久尤物电影视频在线观看 | 色搞搞 | 国产免费不卡av | 日韩高清一二区 | 成人精品一区二区三区电影免费 | av日韩av | 欧美大片第1页 | 久久精品免费观看 | 日韩国产高清在线 | 超碰在线人人草 | 综合精品在线 | 免费黄色特级片 | 精品久久久久久综合日本 | 午夜在线观看影院 | 国语对白少妇爽91 | 亚洲一区二区三区在线看 | 激情丁香综合 | 日韩欧美在线观看一区二区 | 久久久首页 | 国产在线永久 | 日韩欧美99 | 亚洲免费在线观看视频 | 男女激情片在线观看 | 黄色在线观看免费网站 | 国产91在线观看 | 免费日韩 精品中文字幕视频在线 | 欧美福利视频 | 亚洲欧洲视频 | 亚洲第一区在线播放 | 国产一区91 | 天天干天天操 | 九色免费视频 | 久久情爱 | 国产人成在线观看 | 丁香婷五月 | 精品国产一二三四区 | 久久成人视屏 | 婷婷六月中文字幕 | 九色琪琪久久综合网天天 | 在线蜜桃视频 | 在线亚洲播放 | 韩国av电影网 | av高清免费在线 | 国产免费一区二区三区网站免费 | 中文字幕高清视频 | 精品久久久免费 | 99久久婷婷国产综合精品 | 欧美日韩国产页 | 99r精品视频在线观看 | 欧美日韩国内在线 | 天天翘av| 一区 二区电影免费在线观看 | 豆豆色资源网xfplay | 人人干人人上 | 丁香激情婷婷 | av中文字幕网站 | 精品久久久久久久久久久久久久久久久久 | 国产黄色成人 | 久久久久国产一区二区 | 成人国产在线 | 欧美久久影院 | 深爱激情开心 | 久久久高清 | 国产91精品一区二区 | 日韩一区二区免费在线观看 | 狠狠色丁香婷综合久久 | 欧美激情综合网 | 精品久久久久久综合 | 日韩美女久久 | 久久婷婷一区二区三区 | 成年人在线看视频 | 久久毛片网站 | 久草影视在线 | 欧美福利在线播放 | 一区二区三区在线观看中文字幕 | 国产免费黄色 | 久久五月情影视 | 亚洲成人精品久久久 | 国产又粗又猛又爽又黄的视频免费 | 国产色在线观看 | 成人国产精品av | 成人黄色短片 | 日韩欧美大片免费观看 | 免费人成在线观看 | 久久精品99| 精品亚洲免费 | 五月天婷婷丁香花 | 国产成人精品午夜在线播放 | 久草五月| 亚洲jizzjizz日本少妇 | 亚洲一一在线 | 不卡国产视频 | 天天干天天操天天入 | 国产精品免费久久久 | 伊人国产在线观看 | 午夜三级影院 | 中文字幕 二区 | 中文视频一区二区 | 999久久久国产精品 高清av免费观看 | 国产精品99久久久久久小说 | 国产成人精品久久久久蜜臀 | 欧美成人基地 | 韩国av永久免费 | 精品国产99 | 欧美激情精品久久久久久变态 | 麻豆国产精品永久免费视频 | 日本精品一区二区三区在线播放视频 | 欧美成人基地 | 久久人人97超碰com | 天天操天天干天天干 | 三级黄色a | 欧美精品资源 | 国产亚洲片 | 色婷婷福利视频 | 综合色中色 | 国产成人高清 | 九九九免费视频 | 81精品国产乱码久久久久久 | 91精品欧美 | 96精品高清视频在线观看软件特色 | 久久综合九色综合97婷婷女人 | 日韩免费av片 | www在线免费观看 | 91亚洲精品国偷拍 | 精品国内自产拍在线观看视频 | 激情五月在线观看 | 日日噜噜噜噜夜夜爽亚洲精品 | aaaaaa毛片 | 中文免费在线观看 | 日韩av成人在线观看 | 国产日韩中文字幕 | 久久久久久久久久久免费 | 欧美一区二区三区不卡 | 精品久久久久久亚洲综合网站 | caobi视频 | a√国产免费a | 免费精品在线观看 | 在线久草视频 | av在线免费观看不卡 | 国产在线观看99 | av片一区二区 | 欧美日韩另类视频 | 日韩久久久久久久久久 | 中文字幕999| 亚洲色图美腿丝袜 | 麻豆激情电影 | 亚洲资源网 | 日韩中文字幕在线不卡 | 国产剧情一区二区在线观看 | 久久综合九色综合97婷婷女人 | 美国av片在线观看 | 99精品黄色| 中国一级片视频 | 精品影院一区二区久久久 | 伊人久久电影网 | 国产精品专区在线 | 久久久免费看视频 | 韩日电影在线观看 | 国产999精品 | 一区二区三区日韩精品 | 成人av在线一区二区 | 久草在线视频精品 | 国产喷水在线 | 久久国产欧美日韩精品 | 成人久久精品视频 | 一级黄色a视频 | 四虎在线永久免费观看 | 日本中出在线观看 | 在线蜜桃视频 | 国产理论片在线观看 | 九九精品在线观看 | 国产96在线观看 | 日本精品视频网站 | 日日操操 | 永久免费视频国产 | 五月婷网站 | 国产精品久久久亚洲 | .国产精品成人自产拍在线观看6 | 四虎在线免费 | 午夜电影一区 | 国产精品 中文字幕 亚洲 欧美 | 亚洲天堂网视频在线观看 | 久久久久久久久黄色 | 一区二区三区在线视频111 | 久久久久激情 | 91视频在线网址 | 亚洲精品乱码久久久久久蜜桃欧美 | 久久精品高清视频 | 亚洲一区黄色 | 1024手机基地在线观看 | 91香蕉久久 | 东方av在 | 999久久久久 | 天天色天天干天天色 | 男女啪啪免费网站 | 亚洲黄色app | 精品在线不卡 | 99在线热播精品免费99热 | 久久激情综合 | 欧美精品一级视频 | 91精品一区二区三区久久久久久 | www.婷婷com| 九七视频在线观看 | 色吊丝在线永久观看最新版本 | 国产黄色片在线 | 91福利社区在线观看 | 天天干天天干天天射 | 久久一区二区免费视频 | 国产精品18毛片一区二区 | 欧美精品国产综合久久 | 男女拍拍免费视频 | 综合激情婷婷 | 国产精品第 | 久久不射网站 | 国产高清黄 | 国产中文字幕一区二区三区 | 欧美日韩在线电影 | 久久免费精品 | 亚洲视频久久久 | 日韩av三区| 国产精品av免费在线观看 | 久久免费大片 | 色视频在线观看 | 五月天久久综合 | 免费国产黄线在线观看视频 | 精品一区二区三区香蕉蜜桃 | 免费日韩精品 | 伊人天天操 | 五月婷婷影视 | 欧美一区在线观看视频 | 视频二区在线 | 国产精品色在线 | www.天天干.com| 久久久免费看 | 91精品久久久久久久99蜜桃 | 午夜久久影院 | 欧美男女爱爱视频 | 日韩免费在线一区 | 欧美性粗大hdvideo | 水蜜桃亚洲一二三四在线 | 伊人五月 | 久久成人亚洲欧美电影 | 国产免费观看久久 | 欧美-第1页-屁屁影院 | 久久伊人婷婷 | 国产精品日韩在线 | 成人免费观看网站 | 国产高清永久免费 | 国产精品国内免费一区二区三区 | 亚洲一片黄 | 中文字幕二区三区 | 大胆欧美gogo免费视频一二区 | 热re99久久精品国产66热 | 国产五月色婷婷六月丁香视频 | 毛片无卡免费无播放器 | 久久看免费视频 | 91精品影视| 久草免费在线 | 日韩视频专区 | 中文在线最新版天堂 | 午夜精品影院 | 免费成人结看片 | 丁香婷婷网| 黄色精品一区 | 91香蕉视频 mp4 | 日韩欧美视频一区 | 国产裸体视频网站 | 久久9精品| 99热亚洲精品 | 91九色精品 | 欧美成人亚洲成人 | 福利一区二区三区四区 | 最近2019好看的中文字幕免费 | 免费在线黄色av | 在线免费观看国产 | 国产精品99蜜臀久久不卡二区 | 草久在线 | 一区 二区电影免费在线观看 | 久久大视频 | 8090yy亚洲精品久久 | 国产91丝袜在线播放动漫 | 国产视频在线观看免费 | 91在线视频 | 免费网址在线播放 | 成年人视频在线观看免费 | 激情深爱 | 日韩av资源站 | 国产精品视频永久免费播放 | 日韩va欧美va亚洲va久久 | 天海翼一区二区三区免费 | 色中文字幕在线观看 | 日韩v在线91成人自拍 | 激情小说久久 | 人人插超碰 | bbbbb女女女女女bbbbb国产 | 中文字幕亚洲精品日韩 | 亚洲国产成人久久综合 | 日韩精品专区在线影院重磅 | 久久久久亚洲精品成人网小说 | 日本xxxx.com | 91久久久久久久一区二区 | av免费网站 | 97看片吧| 一区二区三区四区影院 | 久久久久久久久久久免费av | 最新免费av在线 | 欧美 高跟鞋交 xxxxhd | 男女免费av | 久久av免费| 99精品国产一区二区三区不卡 | 男女啪啪视屏 | 天天搞天天| 色噜噜狠狠狠狠色综合 | 久久久99精品免费观看app | 国产精品久久艹 | 成人影视免费 | 亚洲三级性片 | 91亚洲精品国偷拍自产在线观看 | 免费在线观看污网站 | 在线国产日韩 | 久久国产欧美日韩精品 | 久久成人国产精品一区二区 | 天天色天天操天天爽 | 免费日韩av电影 | 美女久久视频 | 4438全国亚洲精品观看视频 | 激情视频综合网 | 欧美一级日韩三级 | 深爱激情五月婷婷 | 91麻豆免费版 | 处女av在线| 高清在线一区 | 亚洲一区二区精品3399 | 国产精品中文 | 99亚洲精品在线 | 久久久久久国产精品久久 | 天天操天天谢 | 激情丁香婷婷 | 蜜桃av人人夜夜澡人人爽 | 国产区精品视频 | 99视频在线看 | 欧美日韩免费观看一区=区三区 | 视频一区二区在线观看 | 欧美日韩伦理在线 | 亚洲综合色网站 | 狠狠狠操 | 国产黄影院色大全免费 | 国产视频色 | 国产精品18久久久久久久 | 久久精品国产免费看久久精品 | 亚洲高清视频一区二区三区 | 在线视频 国产 日韩 | 国产一区二区电影在线观看 | 婷婷综合导航 | 成人免费观看a | 精品女同一区二区三区在线观看 | 成人一区二区在线观看 | 亚洲成人第一区 | 99爱精品视频 | 国内精品99 | 色亚洲网 | 国产视频一区在线免费观看 | 天天色天天操综合网 | 欧美日韩一区二区久久 | 91大神精品视频在线观看 | 久久a v视频| 久草久热 | 人人狠狠综合久久亚洲婷 | 在线国产欧美 | 一级特黄av | 国产精品 国产精品 | 天天激情| 91九色国产在线 | av福利网址导航大全 | 欧美一级视频免费看 | 有码中文字幕 | 国产精品成人免费精品自在线观看 | 国产精久久久久久久 | 一区二区三区在线电影 | 亚洲久草在线 | 精品一区在线看 | 日韩在线电影一区二区 | 国产一线在线 | 在线观看日本韩国电影 | 六月激情 | 欧美日韩高清一区二区 国产亚洲免费看 | 日韩中文在线播放 | 美女免费视频黄 | 狠狠干成人综合网 | 在线精品视频免费播放 | 亚洲综合国产精品 | 天天操,夜夜操 | 激情欧美丁香 | 国产午夜一区 | 成人av电影免费在线播放 | 天天草综合 | 国产欧美日韩精品一区二区免费 | 久久桃花网| 麻豆传媒视频在线 | 国产精品乱码高清在线看 | 一区二区三区不卡在线 | 中文字幕亚洲综合久久五月天色无吗'' | 日日干天天操 | 国产高清av | 日本资源中文字幕在线 | 久久久久久久久久影视 | 最新av电影网站 | 亚洲精品视频免费观看 | adn—256中文在线观看 | 成年人app网址 | 黄色亚洲免费 | 亚洲天堂精品视频在线观看 | 国产精品区一区 | 亚洲国产精品成人女人久久 | 国产在线精品国自产拍影院 | 香蕉在线影院 | 亚洲电影自拍 | 亚洲精品看片 | 99精品国产亚洲 | 日韩精品第一区 | 日韩av免费在线电影 | 2022久久国产露脸精品国产 | 视频1区2区| 午夜影视剧场 | 91精品啪在线观看国产线免费 | 天天干天天拍天天操 | 精品视频区| 免费一级黄色 | 欧美极品裸体 | 日韩av网页 | 精品国产免费观看 | 日本中文字幕一二区观 | 狠狠色综合欧美激情 | 国产精久久久久久妇女av | 九九热精品视频在线播放 | 一级欧美一级日韩 | 色婷久久| 国产精品成人自产拍在线观看 | 在线免费国产 | 国产九九精品 | 2019中文 | 久久艹人人 | a色视频| 成人免费网站在线观看 | 国产黄色片免费在线观看 | 在线观看国产成人av片 | 亚洲精品国产精品国产 | 日韩欧美国产成人 | 成年人国产精品 | 国产小视频你懂的 | 97视频人人澡人人爽 | 亚洲 欧美 91 | 国产精品18毛片一区二区 | 色偷偷网站视频 | 国产精品久久久久9999吃药 | 91日韩免费 | 一级片免费观看 | 国产亚洲日 | 中文在线a√在线 | 国产精品高 | 欧美成人精品三级在线观看播放 | 蜜桃视频在线观看一区 | 精品美女视频 | 亚洲精品福利在线观看 | 日韩欧美一区二区三区视频 | 尤物97国产精品久久精品国产 | av成人亚洲 | 成人a视频在线观看 | 日韩免费电影一区二区 | 久久亚洲在线 | 国产精品高 | 十八岁以下禁止观看的1000个网站 | 91麻豆精品国产91久久久久久 | 天天操天天舔天天干 | 国产a级片免费观看 | 国产免费a| 久久高清国产 | 五月婷婷在线播放 | 日韩免费在线一区 | 在线免费av电影 | 成人网在线免费视频 | 日韩视频在线不卡 | 国产xvideos免费视频播放 | 91伊人| 久久精品视频网 | 成人亚洲免费 | av免费在线播放 | 97色涩 | 国产精品 9999 | 久久久久中文 | 午夜12点 | 国产精品女主播一区二区三区 | 四虎影视成人永久免费观看视频 | 久久久伦理 | 国产中文视频 | 久久6精品 | 国产精品99久久久精品免费观看 | 五月婷婷激情六月 | a级片网站| 色婷婷啪啪免费在线电影观看 | 最新的av网站 | 久久综合视频网 | 色播五月激情五月 | 91九色精品女同系列 | 日韩免费电影一区二区 | 天海翼一区二区三区免费 | av日韩国产 | www.天天色 | 亚洲欧美视频在线观看 | 香蕉久久久久 | 婷婷精品视频 | 久久成人国产精品 | 久久久久久久av麻豆果冻 | 欧美在线视频日韩 | 国产精品黑丝在线观看 | 午夜.dj高清免费观看视频 | 色综合久久久久久久 | 在线视频在线观看 | 综合色播| 99精品国产一区二区 | 国产在线高清精品 | 久草网视频在线观看 | 99色99| 91成人看片 | 在线视频 一区二区 | 日韩精品一区二区在线观看视频 | 国产中文字幕亚洲 | 天天摸天天舔天天操 | 500部大龄熟乱视频 欧美日本三级 | .精品久久久麻豆国产精品 亚洲va欧美 | 香蕉视频在线播放 | 国产精品福利视频 | 性色大片在线观看 | 国产精品第一页在线 | 男女激情片在线观看 | 久久久久免费视频 | 精品国产一区二区三区噜噜噜 | 精品在线视频一区 | 黄色免费网站下载 | 成人av高清在线 | 国产在线污 | 日韩欧美视频免费看 | 国产精品成人国产乱一区 | 欧美日bb | 精品一二 | 久久精品99国产精品亚洲最刺激 | www.玖玖玖 | 九九热有精品 | 国内视频在线 | 国产成人久久精品一区二区三区 | 亚洲国产日韩欧美 | 午夜视频一区二区三区 | 日韩深夜在线观看 | 久久99精品国产麻豆宅宅 | 日日久视频 | 在线观看的黄色 | 久久亚洲影视 | 在线 国产 日韩 | 91精品国产三级a在线观看 | 超碰在线97观看 | 国产区精品在线观看 | 在线观看av黄色 | 国产精品一区二区三区久久久 | 国产精品视频地址 | 久久久久五月天 | 亚洲精品1234区 | 99亚洲精品| 中文字幕在线不卡国产视频 | 日韩免费不卡视频 | 91视频免费国产 | 96久久精品| 13日本xxxxxⅹxxx20 | 国产成人精品久久久久 | 久草网视频| 久久最新网址 | avcom在线 | 一区二区三区精品在线视频 | 亚洲激情 在线 | 91亚洲精品久久久蜜桃网站 | 国内精品毛片 | 中文字字幕在线 | 这里只有精品视频在线 | 视频在线精品 | 亚洲va欧美va人人爽春色影视 | 在线免费观看麻豆 | 91在线免费观看网站 | 99久久精品免费看国产四区 | 69xx视频 | 黄色软件大全网站 | 91人人射 | 一区二区视频电影在线观看 | 91视频啊啊啊 | 国产精品二区在线 | 青青久视频 | 2000xxx影视 | 狠狠干干 | 六月丁香六月婷婷 | 免费日韩 精品中文字幕视频在线 | av网在线观看 | 久久亚洲热 | 在线视频观看亚洲 | 欧美性高跟鞋xxxxhd | 99久久爱 | 久久黄色影视 | 国内精品免费久久影院 | 日本久久精品 | 国产女人40精品一区毛片视频 | 狠狠五月天 | 五月在线 | 国产免费观看久久黄 | 久草在线视频看看 | 国产原创在线观看 | 欧美性爽爽| 国产精品综合av一区二区国产馆 | 亚州免费视频 | 伊人狠狠色丁香婷婷综合 | 99在线精品观看 | 在线成人av | 狠狠操夜夜操 | 91探花在线 | 日韩国产欧美在线播放 | 亚洲国产欧美一区二区三区丁香婷 | 国产精品爽爽久久久久久蜜臀 | av免费试看 | 国产又粗又猛又黄视频 | 四虎成人免费观看 | 久久成人国产精品入口 | av福利超碰网站 | 午夜精品一区二区三区在线观看 | 久久人人爽爽人人爽人人片av | 99精品一区二区 | 亚洲成av人影院 | 国产无遮挡猛进猛出免费软件 | 欧美日韩高清在线 | 欧美国产日韩一区二区三区 | 成人一级免费视频 | 免费看的黄色网 | 国产精品麻豆一区二区三区 | 日韩欧美91| 欧美一区二区伦理片 | 观看免费av | 天天视频色 | 婷婷色5月| 国产在线观看二区 | 欧美日韩精品影院 | 国产精品麻豆欧美日韩ww | 免费高清在线观看电视网站 | 97精品视频在线播放 | 综合网中文字幕 | 国产精品成人av电影 | 国产黄色观看 | 国产精品99久久免费黑人 | 伊人色综合久久天天 | 91精品国自产在线观看欧美 | 天天干亚洲 | 日本久久精 | 久久久免费播放 | 国产91全国探花系列在线播放 | 日本aa在线 | 日韩一区二区三区在线看 | 国产精品久久久久永久免费观看 | 亚洲专区一二三 | 亚洲一级久久 | 婷婷久操 | 波多野结衣资源 | 在线天堂中文在线资源网 | 九九视频精品在线 | 久久国产福利 | 天天操天天摸天天射 | 亚洲国产视频在线 | 最近中文字幕在线播放 | 国模精品一区二区三区 | 天天操天天舔天天干 | 色综合久久五月 | 亚洲精品视频在线播放 | 国产视频欧美视频 | 色狠狠干 | 国产专区在线 | 免费观看xxxx9999片 | 久章草在线 | 国产专区一 | 精品久久九九 | 在线免费观看黄色大片 | 久久影视一区二区 | 午夜精品久久久久久久99 | 国产免费亚洲高清 | 精品久久精品 | 天天干夜夜 | 久久视频在线观看 | 国产精久久久久久妇女av | 久久久久久久久黄色 | 日本中文字幕电影在线免费观看 | 久久艹国产| 欧美成人性网 | 国模精品一区二区三区 | www.狠狠插.com| 999国产精品视频 | 在线观看免费 | www.伊人网| 亚洲欧美日韩国产一区二区三区 | 亚洲手机av | 综合网久久 | 夜夜夜夜猛噜噜噜噜噜初音未来 | 日日操操操 | 日韩特黄一级欧美毛片特黄 | ww视频在线观看 | 精品美女在线视频 | 久久色视频| 91精品久久久久久久久久入口 | 久久久久成人精品亚洲国产 | 香蕉网在线| 黄色不卡av | 国产一区在线视频观看 | 伊人官网 | 手机在线小视频 | 夜夜夜夜操 | 国产精品入口66mio女同 | 欧美色图另类 | 久久久天天操 | 国产成人福利片 | 久草在线电影网 | www.婷婷色 | 不卡的一区二区三区 | 免费91麻豆精品国产自产在线观看 | 国产小视频在线免费观看视频 | 久久综合狠狠综合久久综合88 | 最新高清无码专区 | 久久人人做 | 精品久久久久久久久久国产 | 日韩av电影一区 | 一区二区不卡视频在线观看 | 日韩一级成人av | 日本黄色大片儿 | 亚洲成人资源 | 97在线观看 | 成人资源在线观看 | 成人午夜在线电影 | 国产精品不卡一区 | 成全免费观看视频 | 久久成人18免费网站 | 免费黄色特级片 | 91在线国内视频 | 在线视频 91 | 91在线视频精品 | 精品国产自在精品国产精野外直播 | 波多野结衣在线中文字幕 | 中文字幕高清在线播放 | 中文字幕亚洲不卡 | 激情婷婷六月 | 韩国在线视频一区 | 天天综合网国产 | 久久久久久久久国产 | 日日夜日日干 | 久久无码av一区二区三区电影网 | www.久热 | 黄色资源在线 | 国产精品视频免费在线观看 | 国产精品午夜8888 | 国产精品永久久久久久久久久 | 亚洲精品国偷拍自产在线观看蜜桃 | 日韩av在线资源 | 免费在线观看一级片 | 久久久免费精品国产一区二区 | 久久人人爽 | 中文字幕第一页在线播放 | 亚洲一区二区精品视频 | 丁香五月网久久综合 | 最新av电影网址 | 人人澡人 | 在线观看理论 | 麻豆传媒视频在线免费观看 | 国产精品原创av片国产免费 | 中文字幕xxxx| 久久久久成人免费 | 国产精品久久三 | 少妇av网| 国产精品18videosex性欧美 | 国产精品久久久久久模特 | 日韩资源在线播放 | 91最新视频在线观看 | 国产手机精品视频 | av性在线| 国产一区二区三区久久久 | 亚洲午夜久久久久久久久电影网 | 丁香电影小说免费视频观看 | 欧美污污视频 | 日韩av中文在线观看 | 色综合久久88色综合天天 | 免费av在线| 久久亚洲福利视频 | 日本在线视频一区二区三区 | 亚洲久草网 | 欧美日韩在线视频一区 | 中日韩免费视频 | 日韩三级视频在线观看 | 成人av网站在线播放 | av成人在线电影 | 天天操天天曰 | 99热最新地址 | 在线综合 亚洲 欧美在线视频 | 久久中文字幕在线视频 | 久久精品99久久 | 亚洲一级黄色av | 99精品国产免费久久久久久下载 | 日本久久久久久 | 天天射天天操天天干 | 国产午夜视频在线观看 | 国产高清久久久 | 青青啪 | 久久久国产一区二区三区四区小说 | 91av在线视频播放 | av成人免费 | 亚洲三级影院 | 日本中文字幕视频 | 视频在线99| 国产只有精品 | 日韩精品观看 | 天天操天天是 | 在线视频成人 | 亚洲成人在线免费 | 色视频在线观看免费 | 91丨精品丨蝌蚪丨白丝jk | 国产精品高潮呻吟久久av无 | a级一a一级在线观看 | av片在线观看免费 | 999久久久久久久久6666 | 国产色区| 国产亚州精品视频 | 日韩com| 国产99久久久欧美黑人 | 久久激情视频免费观看 | 国产精品淫片 | 精品免费一区二区三区 | 菠萝菠萝在线精品视频 | 免费观看福利视频 | 亚州免费视频 | 日日操天天操夜夜操 | 五月婷婷视频在线 | 日韩在线视频观看免费 | 国产手机视频在线 | 99免费| 亚洲国产精品传媒在线观看 | 99这里只有精品视频 | 中文字幕资源在线观看 | 西西444www大胆无视频 | av网站免费看 | 伊人久久精品久久亚洲一区 | 日韩二区三区在线 | 国产精品短视频 | av一级二级 | 色99中文字幕 | 激情图片久久 | 午夜精品一区二区三区可下载 | 久草在线视频网 | 天天干天天做 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 久久精品一区二区三区四区 | av在线播放国产 | 91精品免费在线观看 | 欧美黑人性爽 | 一区二区视频免费在线观看 | 国产精品久久婷婷六月丁香 | 九九热在线观看 | 日韩字幕在线观看 | 99精品国产一区二区三区麻豆 | 丁香激情视频 | 国产专区日韩专区 | 久久人人精品 | 国产美女精品久久久 | 亚洲国产小视频在线观看 | 精品免费 | 久久 精品一区 | 在线播放国产一区二区三区 | 天天综合网~永久入口 | 国产精品视频大全 | 国产高清视频免费最新在线 | 在线中文字幕观看 | 久久影院亚洲 | 超碰在线日韩 | 色香蕉在线 | 99爱爱 | 天天操夜夜干 | 欧美日韩免费观看一区=区三区 | 亚洲成人国产精品 | 国产中文字幕91 | 精品中文字幕在线观看 | 中文字幕91视频 | 成人黄色电影视频 | 日韩欧美91| 中文字幕av在线电影 | 国产精彩视频一区二区 | 香蕉在线视频播放网站 | 日日夜色 | 国产免费xvideos视频入口 | 黄色a一级视频 | 久久综合久久综合这里只有精品 | 97成人啪啪网 | 久久深爱网 | 在线观看色网 | 日色在线视频 | 超碰99在线 | 亚洲精品在线观看视频 |