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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Cronet android 设计与实现分析——备选服务机制

發布時間:2024/4/11 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Cronet android 设计与实现分析——备选服务机制 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

前面我們分析到,在URLRequestHttpJob::StartTransactionInternal()中,會通過URLRequestContext的HttpTransactionFactory創建HttpTransaction,在URLRequestContextBuilder::Build()中創建HttpTransactionFactory的過程如下:

storage->set_http_network_session(base::WrapUnique(new HttpNetworkSession(network_session_params)));std::unique_ptr<HttpTransactionFactory> http_transaction_factory;if (http_cache_enabled_) {std::unique_ptr<HttpCache::BackendFactory> http_cache_backend;if (http_cache_params_.type != HttpCacheParams::IN_MEMORY) {BackendType backend_type =http_cache_params_.type == HttpCacheParams::DISK? CACHE_BACKEND_DEFAULT: CACHE_BACKEND_SIMPLE;http_cache_backend.reset(new HttpCache::DefaultBackend(DISK_CACHE, backend_type, http_cache_params_.path,http_cache_params_.max_size, context->GetFileTaskRunner()));} else {http_cache_backend =HttpCache::DefaultBackend::InMemory(http_cache_params_.max_size);}LOG(INFO) << "Cache is enabled, to create HttpCache";http_transaction_factory.reset(new HttpCache(storage->http_network_session(), std::move(http_cache_backend), true));} else {LOG(INFO) << "Cache is disabled, to create HttpNetworkLayer";http_transaction_factory.reset(new HttpNetworkLayer(storage->http_network_session()));}storage->set_http_transaction_factory(std::move(http_transaction_factory));

Chromium net中有兩個HttpTransactionFactory的實現,分別是HttpCache和HttpNetworkLayer,它們分別在cache被打開和被關閉時用到。這里還會創建HttpNetworkSession。而在cache打開時,在創建HttpCache的同時,還會為它創建http_cache_backend。

HttpCache的創建過程(chromium_android/src/net/http/http_cache.cc)如下:

//----------------------------------------------------------------------------- HttpCache::HttpCache(HttpNetworkSession* session,std::unique_ptr<BackendFactory> backend_factory,bool set_up_quic_server_info): HttpCache(base::WrapUnique(new HttpNetworkLayer(session)),std::move(backend_factory),set_up_quic_server_info) {}HttpCache::HttpCache(std::unique_ptr<HttpTransactionFactory> network_layer,std::unique_ptr<BackendFactory> backend_factory,bool set_up_quic_server_info): net_log_(nullptr),backend_factory_(std::move(backend_factory)),building_backend_(false),bypass_lock_for_test_(false),fail_conditionalization_for_test_(false),mode_(NORMAL),network_layer_(std::move(network_layer)),clock_(new base::DefaultClock()),weak_factory_(this) {HttpNetworkSession* session = network_layer_->GetSession();// Session may be NULL in unittests.// TODO(mmenke): Seems like tests could be changed to provide a session,// rather than having logic only used in unit tests here.if (session) {net_log_ = session->net_log();if (set_up_quic_server_info &&!session->quic_stream_factory()->has_quic_server_info_factory()) {// QuicStreamFactory takes ownership of QuicServerInfoFactoryAdaptor.session->quic_stream_factory()->set_quic_server_info_factory(new QuicServerInfoFactoryAdaptor(this));}} }

這里還是會創建HttpNetworkLayer。HttpTransactionFactory相關的幾個類之間的關系如下:


HttpTransactionFactory

HttpNetworkTransaction表示一個直接的網絡事務,可以理解為一個網絡連接。HttpNetworkSession用于管理網絡連接。HttpNetworkLayer主要用于創建HttpNetworkTransactionHttpCacheHttpCache::Transaction用于處理緩存。HttpCache::Transaction表示一個啟用了緩存的網絡事務,它會借助于HttpCache保存的HttpNetworkLayer引用創建HttpNetworkTransaction,借助于HttpNetworkTransaction訪問網絡,并根據需要將結果緩存起來。HttpCache則對緩存進行管理。HttpNetworkLayerHttpCache都是HttpTransactionFactory,而HttpNetworkTransactionHttpCache::Transaction都是HttpTransaction

我們先不關心啟用cache時,HTTP請求的處理流程,來看HttpNetworkLayer。則在URLRequestHttpJob::StartTransactionInternal()中將通過HttpNetworkLayer創建類型為HttpNetworkTransactionHttpTransaction

HttpNetworkLayer::HttpNetworkLayer(HttpNetworkSession* session): session_(session),suspended_(false) {DCHECK(session_); #if defined(OS_WIN)base::PowerMonitor* power_monitor = base::PowerMonitor::Get();if (power_monitor)power_monitor->AddObserver(this); #endif }HttpNetworkLayer::~HttpNetworkLayer() { #if defined(OS_WIN)base::PowerMonitor* power_monitor = base::PowerMonitor::Get();if (power_monitor)power_monitor->RemoveObserver(this); #endif }int HttpNetworkLayer::CreateTransaction(RequestPriority priority,std::unique_ptr<HttpTransaction>* trans) {if (suspended_)return ERR_NETWORK_IO_SUSPENDED;trans->reset(new HttpNetworkTransaction(priority, GetSession()));return OK; }HttpCache* HttpNetworkLayer::GetCache() {return NULL; }HttpNetworkSession* HttpNetworkLayer::GetSession() {return session_; }void HttpNetworkLayer::OnSuspend() {suspended_ = true;session_->CloseIdleConnections(); }void HttpNetworkLayer::OnResume() {suspended_ = false; }

URLRequestHttpJob::StartTransactionInternal()創建了HttpTransaction之后,它會執行HttpTransaction的Start()來啟動對HTTP事務的處理,HttpNetworkTransaction的Start()定義如下:

int HttpNetworkTransaction::Start(const HttpRequestInfo* request_info,const CompletionCallback& callback,const BoundNetLog& net_log) {net_log_ = net_log;request_ = request_info;// Now that we have an HttpRequestInfo object, update server_ssl_config_.session_->GetSSLConfig(*request_, &server_ssl_config_, &proxy_ssl_config_);if (request_->load_flags & LOAD_DISABLE_CERT_REVOCATION_CHECKING) {server_ssl_config_.rev_checking_enabled = false;proxy_ssl_config_.rev_checking_enabled = false;}if (request_->load_flags & LOAD_PREFETCH)response_.unused_since_prefetch = true;next_state_ = STATE_NOTIFY_BEFORE_CREATE_STREAM;int rv = DoLoop(OK);if (rv == ERR_IO_PENDING)callback_ = callback;return rv; }

Chromium net將所有HTTP請求的處理抽象為幾個步驟,并通過一個循環DoLoop()來一步一步地執行。DoLoop()的定義 (chromium_android/src/net/http/http_network_transaction.cc) 如下:

int HttpNetworkTransaction::DoLoop(int result) {DCHECK(next_state_ != STATE_NONE);int rv = result;do {State state = next_state_;next_state_ = STATE_NONE;switch (state) {case STATE_NOTIFY_BEFORE_CREATE_STREAM:DCHECK_EQ(OK, rv);rv = DoNotifyBeforeCreateStream();break;case STATE_CREATE_STREAM:DCHECK_EQ(OK, rv);rv = DoCreateStream();break;case STATE_CREATE_STREAM_COMPLETE:rv = DoCreateStreamComplete(rv);break;case STATE_INIT_STREAM:DCHECK_EQ(OK, rv);rv = DoInitStream();break;case STATE_INIT_STREAM_COMPLETE:rv = DoInitStreamComplete(rv);break;case STATE_GENERATE_PROXY_AUTH_TOKEN:DCHECK_EQ(OK, rv);rv = DoGenerateProxyAuthToken();break;case STATE_GENERATE_PROXY_AUTH_TOKEN_COMPLETE:rv = DoGenerateProxyAuthTokenComplete(rv);break;case STATE_GENERATE_SERVER_AUTH_TOKEN:DCHECK_EQ(OK, rv);rv = DoGenerateServerAuthToken();break;case STATE_GENERATE_SERVER_AUTH_TOKEN_COMPLETE:rv = DoGenerateServerAuthTokenComplete(rv);break;case STATE_GET_PROVIDED_TOKEN_BINDING_KEY:DCHECK_EQ(OK, rv);rv = DoGetProvidedTokenBindingKey();break;case STATE_GET_PROVIDED_TOKEN_BINDING_KEY_COMPLETE:rv = DoGetProvidedTokenBindingKeyComplete(rv);break;case STATE_GET_REFERRED_TOKEN_BINDING_KEY:DCHECK_EQ(OK, rv);rv = DoGetReferredTokenBindingKey();break;case STATE_GET_REFERRED_TOKEN_BINDING_KEY_COMPLETE:rv = DoGetReferredTokenBindingKeyComplete(rv);break;case STATE_INIT_REQUEST_BODY:DCHECK_EQ(OK, rv);rv = DoInitRequestBody();break;case STATE_INIT_REQUEST_BODY_COMPLETE:rv = DoInitRequestBodyComplete(rv);break;case STATE_BUILD_REQUEST:DCHECK_EQ(OK, rv);net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_SEND_REQUEST);rv = DoBuildRequest();break;case STATE_BUILD_REQUEST_COMPLETE:rv = DoBuildRequestComplete(rv);break;case STATE_SEND_REQUEST:DCHECK_EQ(OK, rv);rv = DoSendRequest();break;case STATE_SEND_REQUEST_COMPLETE:rv = DoSendRequestComplete(rv);net_log_.EndEventWithNetErrorCode(NetLog::TYPE_HTTP_TRANSACTION_SEND_REQUEST, rv);break;case STATE_READ_HEADERS:DCHECK_EQ(OK, rv);net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_READ_HEADERS);rv = DoReadHeaders();break;case STATE_READ_HEADERS_COMPLETE:rv = DoReadHeadersComplete(rv);net_log_.EndEventWithNetErrorCode(NetLog::TYPE_HTTP_TRANSACTION_READ_HEADERS, rv);break;case STATE_READ_BODY:DCHECK_EQ(OK, rv);net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_READ_BODY);rv = DoReadBody();break;case STATE_READ_BODY_COMPLETE:rv = DoReadBodyComplete(rv);net_log_.EndEventWithNetErrorCode(NetLog::TYPE_HTTP_TRANSACTION_READ_BODY, rv);break;case STATE_DRAIN_BODY_FOR_AUTH_RESTART:DCHECK_EQ(OK, rv);net_log_.BeginEvent(NetLog::TYPE_HTTP_TRANSACTION_DRAIN_BODY_FOR_AUTH_RESTART);rv = DoDrainBodyForAuthRestart();break;case STATE_DRAIN_BODY_FOR_AUTH_RESTART_COMPLETE:rv = DoDrainBodyForAuthRestartComplete(rv);net_log_.EndEventWithNetErrorCode(NetLog::TYPE_HTTP_TRANSACTION_DRAIN_BODY_FOR_AUTH_RESTART, rv);break;default:NOTREACHED() << "bad state";rv = ERR_FAILED;break;}} while (rv != ERR_IO_PENDING && next_state_ != STATE_NONE);return rv; }

接下來我們逐個地分析這些步驟。

Stream的創建

DoNotifyBeforeCreateStream()執行before_network_start_callback:

int HttpNetworkTransaction::DoNotifyBeforeCreateStream() {next_state_ = STATE_CREATE_STREAM;bool defer = false;if (!before_network_start_callback_.is_null())before_network_start_callback_.Run(&defer);if (!defer)return OK;return ERR_IO_PENDING; }

在DoCreateStream()中創建Stream:

int HttpNetworkTransaction::DoCreateStream() {// TODO(mmenke): Remove ScopedTracker below once crbug.com/424359 is fixed.tracked_objects::ScopedTracker tracking_profile(FROM_HERE_WITH_EXPLICIT_FUNCTION("424359 HttpNetworkTransaction::DoCreateStream"));response_.network_accessed = true;next_state_ = STATE_CREATE_STREAM_COMPLETE;if (ForWebSocketHandshake()) {stream_request_.reset(session_->http_stream_factory_for_websocket()->RequestWebSocketHandshakeStream(*request_,priority_,server_ssl_config_,proxy_ssl_config_,this,websocket_handshake_stream_base_create_helper_,net_log_));} else {stream_request_.reset(session_->http_stream_factory()->RequestStream(*request_,priority_,server_ssl_config_,proxy_ssl_config_,this,net_log_));}DCHECK(stream_request_.get());return ERR_IO_PENDING; }......bool HttpNetworkTransaction::ForWebSocketHandshake() const {return websocket_handshake_stream_base_create_helper_ &&request_->url.SchemeIsWSOrWSS(); }

當請求是一個Websocket請求時,通過HttpNetworkSession的http_stream_factory_for_websocket創建Stream,而其他情況下,則會通過HttpNetworkSession的http_stream_factory創建Stream。

HttpNetworkSession的創建過程 (chromium_android/src/net/http/http_network_session.cc) 可以看到,http_stream_factory_for_websocket和http_stream_factory都是HttpStreamFactoryImpl

http_stream_factory_(new HttpStreamFactoryImpl(this, false)),http_stream_factory_for_websocket_(new HttpStreamFactoryImpl(this, true)),

為Websocket之外的其它請求創建Stream的過程 (chromium_android/src/net/http/http_stream_factory_impl.cc) 為:

HttpStreamFactoryImpl::HttpStreamFactoryImpl(HttpNetworkSession* session,bool for_websockets): session_(session),job_factory_(new DefaultJobFactory()),for_websockets_(for_websockets) {}HttpStreamRequest* HttpStreamFactoryImpl::RequestStream(const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HttpStreamRequest::Delegate* delegate,const BoundNetLog& net_log) {DCHECK(!for_websockets_);return RequestStreamInternal(request_info, priority, server_ssl_config,proxy_ssl_config, delegate, nullptr,HttpStreamRequest::HTTP_STREAM, net_log); }......HttpStreamRequest* HttpStreamFactoryImpl::RequestStreamInternal(const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HttpStreamRequest::Delegate* delegate,WebSocketHandshakeStreamBase::CreateHelper*websocket_handshake_stream_create_helper,HttpStreamRequest::StreamType stream_type,const BoundNetLog& net_log) {JobController* job_controller =new JobController(this, delegate, session_, job_factory_.get());job_controller_set_.insert(base::WrapUnique(job_controller));Request* request = job_controller->Start(request_info, delegate, websocket_handshake_stream_create_helper, net_log,stream_type, priority, server_ssl_config, proxy_ssl_config);return request; }

在HttpStreamFactoryImpl::RequestStreamInternal()中,主要是創建了一個JobController,然后用job_controller->Start()創建了Request,也就是HttpStreamRequest。

由HttpStreamFactoryImpl的構造函數可以看到,job_factory_是DefaultJobFactory,這個類的實現也相當簡單(chromium_android/src/net/http/http_stream_factory_impl.cc) :

namespace { // Default JobFactory for creating HttpStreamFactoryImpl::Jobs. class DefaultJobFactory : public HttpStreamFactoryImpl::JobFactory {public:DefaultJobFactory() {}~DefaultJobFactory() override {}HttpStreamFactoryImpl::Job* CreateJob(HttpStreamFactoryImpl::Job::Delegate* delegate,HttpStreamFactoryImpl::JobType job_type,HttpNetworkSession* session,const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HostPortPair destination,GURL origin_url,NetLog* net_log) override {return new HttpStreamFactoryImpl::Job(delegate, job_type, session, request_info, priority, server_ssl_config,proxy_ssl_config, destination, origin_url, net_log);}HttpStreamFactoryImpl::Job* CreateJob(HttpStreamFactoryImpl::Job::Delegate* delegate,HttpStreamFactoryImpl::JobType job_type,HttpNetworkSession* session,const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HostPortPair destination,GURL origin_url,AlternativeService alternative_service,NetLog* net_log) override {return new HttpStreamFactoryImpl::Job(delegate, job_type, session, request_info, priority, server_ssl_config,proxy_ssl_config, destination, origin_url, alternative_service,net_log);} }; } // anonymous namespace

JobController的Start()定義 (chromium_android/src/net/http/http_stream_factory_impl_job_controller.cc) 如下:

HttpStreamFactoryImpl::Request* HttpStreamFactoryImpl::JobController::Start(const HttpRequestInfo& request_info,HttpStreamRequest::Delegate* delegate,WebSocketHandshakeStreamBase::CreateHelper*websocket_handshake_stream_create_helper,const BoundNetLog& net_log,HttpStreamRequest::StreamType stream_type,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config) {DCHECK(factory_);DCHECK(!request_);request_ = new Request(request_info.url, this, delegate,websocket_handshake_stream_create_helper, net_log,stream_type);CreateJobs(request_info, priority, server_ssl_config, proxy_ssl_config,delegate, stream_type, net_log);return request_; }

在這里主要是創建了一個Request,然后調用CreateJobs()創建了一些Jobs:

void HttpStreamFactoryImpl::JobController::CreateJobs(const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HttpStreamRequest::Delegate* delegate,HttpStreamRequest::StreamType stream_type,const BoundNetLog& net_log) {DCHECK(!main_job_);DCHECK(!alternative_job_);HostPortPair destination(HostPortPair::FromURL(request_info.url));GURL origin_url = ApplyHostMappingRules(request_info.url, &destination);main_job_.reset(job_factory_->CreateJob(this, MAIN, session_, request_info, priority, server_ssl_config,proxy_ssl_config, destination, origin_url, net_log.net_log()));AttachJob(main_job_.get());// Create an alternative job if alternative service is set up for this domain.const AlternativeService alternative_service =GetAlternativeServiceFor(request_info, delegate, stream_type);if (alternative_service.protocol != UNINITIALIZED_ALTERNATE_PROTOCOL) {// Never share connection with other jobs for FTP requests.DVLOG(1) << "Selected alternative service (host: "<< alternative_service.host_port_pair().host()<< " port: " << alternative_service.host_port_pair().port() << ")";DCHECK(!request_info.url.SchemeIs("ftp"));HostPortPair alternative_destination(alternative_service.host_port_pair());ignore_result(ApplyHostMappingRules(request_info.url, &alternative_destination));alternative_job_.reset(job_factory_->CreateJob(this, ALTERNATIVE, session_, request_info, priority, server_ssl_config,proxy_ssl_config, alternative_destination, origin_url,alternative_service, net_log.net_log()));AttachJob(alternative_job_.get());main_job_is_blocked_ = true;alternative_job_->Start(request_->stream_type());}// Even if |alternative_job| has already finished, it will not have notified// the request yet, since we defer that to the next iteration of the// MessageLoop, so starting |main_job_| is always safe.main_job_->Start(request_->stream_type()); }......GURL HttpStreamFactoryImpl::JobController::ApplyHostMappingRules(const GURL& url,HostPortPair* endpoint) {const HostMappingRules* mapping_rules = session_->params().host_mapping_rules;if (mapping_rules && mapping_rules->RewriteHost(endpoint)) {url::Replacements<char> replacements;const std::string port_str = base::UintToString(endpoint->port());replacements.SetPort(port_str.c_str(), url::Component(0, port_str.size()));replacements.SetHost(endpoint->host().c_str(),url::Component(0, endpoint->host().size()));return url.ReplaceComponents(replacements);}return url; }

這里的過程如下:

  • 應用主機映射規則,對url做修飾。
  • 通過job_factory創建main_job。
  • 查找備選服務。
  • 找到了備選服務,則創建alternative_job,并Start它。
  • Start main_job。
  • 我們前面提到的一些Jobs主要是指main_job,和可能會創建的alternative_job。

    接著我們來看Job的Start()方法(chromium_android/src/net/http/http_stream_factory_impl_job.cc) :

    void HttpStreamFactoryImpl::Job::Start(HttpStreamRequest::StreamType stream_type) {stream_type_ = stream_type;StartInternal(); }.......int HttpStreamFactoryImpl::Job::RunLoop(int result) {TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("net"),"HttpStreamFactoryImpl::Job::RunLoop");LOG(INFO) << "HttpStreamFactoryImpl Job DoLoop start " << "job type " << job_type_;result = DoLoop(result);LOG(INFO) << "HttpStreamFactoryImpl Job DoLoop end " << "result " << result;if (result == ERR_IO_PENDING)return result;if (job_type_ == PRECONNECT) {base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&HttpStreamFactoryImpl::Job::OnPreconnectsComplete,ptr_factory_.GetWeakPtr()));return ERR_IO_PENDING;}if (IsCertificateError(result)) {// Retrieve SSL information from the socket.GetSSLInfo();next_state_ = STATE_WAITING_USER_ACTION;base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&HttpStreamFactoryImpl::Job::OnCertificateErrorCallback,ptr_factory_.GetWeakPtr(), result, ssl_info_));return ERR_IO_PENDING;}switch (result) {case ERR_PROXY_AUTH_REQUESTED: {UMA_HISTOGRAM_BOOLEAN("Net.ProxyAuthRequested.HasConnection",connection_.get() != NULL);if (!connection_.get())return ERR_PROXY_AUTH_REQUESTED_WITH_NO_CONNECTION;CHECK(connection_->socket());CHECK(establishing_tunnel_);next_state_ = STATE_WAITING_USER_ACTION;ProxyClientSocket* proxy_socket =static_cast<ProxyClientSocket*>(connection_->socket());base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&Job::OnNeedsProxyAuthCallback, ptr_factory_.GetWeakPtr(),*proxy_socket->GetConnectResponseInfo(),base::RetainedRef(proxy_socket->GetAuthController())));return ERR_IO_PENDING;}case ERR_SSL_CLIENT_AUTH_CERT_NEEDED:base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&Job::OnNeedsClientAuthCallback, ptr_factory_.GetWeakPtr(),base::RetainedRef(connection_->ssl_error_response_info().cert_request_info)));return ERR_IO_PENDING;case ERR_HTTPS_PROXY_TUNNEL_RESPONSE: {DCHECK(connection_.get());DCHECK(connection_->socket());DCHECK(establishing_tunnel_);ProxyClientSocket* proxy_socket =static_cast<ProxyClientSocket*>(connection_->socket());base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, base::Bind(&Job::OnHttpsProxyTunnelResponseCallback,ptr_factory_.GetWeakPtr(),*proxy_socket->GetConnectResponseInfo(),proxy_socket->CreateConnectResponseStream()));return ERR_IO_PENDING;}case OK:job_status_ = STATUS_SUCCEEDED;MaybeMarkAlternativeServiceBroken();next_state_ = STATE_DONE;if (new_spdy_session_.get()) {base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, base::Bind(&Job::OnNewSpdySessionReadyCallback,ptr_factory_.GetWeakPtr()));} else if (delegate_->for_websockets()) {DCHECK(websocket_stream_);base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, base::Bind(&Job::OnWebSocketHandshakeStreamReadyCallback,ptr_factory_.GetWeakPtr()));} else if (stream_type_ == HttpStreamRequest::BIDIRECTIONAL_STREAM) {if (!bidirectional_stream_impl_) {base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, base::Bind(&Job::OnStreamFailedCallback,ptr_factory_.GetWeakPtr(), ERR_FAILED));} else {base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&Job::OnBidirectionalStreamImplReadyCallback,ptr_factory_.GetWeakPtr()));}} else {DCHECK(stream_.get());job_stream_ready_start_time_ = base::TimeTicks::Now();base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE,base::Bind(&Job::OnStreamReadyCallback, ptr_factory_.GetWeakPtr()));}return ERR_IO_PENDING;default:if (job_status_ != STATUS_BROKEN) {DCHECK_EQ(STATUS_RUNNING, job_status_);job_status_ = STATUS_FAILED;MaybeMarkAlternativeServiceBroken();}base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, base::Bind(&Job::OnStreamFailedCallback,ptr_factory_.GetWeakPtr(), result));return ERR_IO_PENDING;} }......int HttpStreamFactoryImpl::Job::StartInternal() {CHECK_EQ(STATE_NONE, next_state_);next_state_ = STATE_START;int rv = RunLoop(OK);DCHECK_EQ(ERR_IO_PENDING, rv);return rv; }

    執行調用流程大體如下:


    HttpStreamFactoryImpl_Job

    HttpStreamFactoryImpl::Job::RunLoop()中,主要是調用了HttpStreamFactoryImpl::Job::DoLoop(),并針對其執行結果,調用響應的回調函數,如:

    void HttpStreamFactoryImpl::Job::OnStreamFailedCallback(int result) {DCHECK_NE(job_type_, PRECONNECT);MaybeCopyConnectionAttemptsFromSocketOrHandle();delegate_->OnStreamFailed(this, result, server_ssl_config_);// |this| may be deleted after this call. }void HttpStreamFactoryImpl::Job::OnCertificateErrorCallback(int result, const SSLInfo& ssl_info) {DCHECK_NE(job_type_, PRECONNECT);MaybeCopyConnectionAttemptsFromSocketOrHandle();delegate_->OnCertificateError(this, result, server_ssl_config_, ssl_info);// |this| may be deleted after this call. }

    從前面的HttpStreamFactoryImpl::JobController::CreateJobs()中可以看到,delegate_正是HttpStreamFactoryImpl::JobController

    而在HttpStreamFactoryImpl::Job::DoLoop()中,則是處理Stream建立的事情。與HttpNetworkTransactionStart() 執行的DoLoop()類似,HttpStreamFactoryImpl::Job::DoLoop()也是將Stream創建的過程抽象為一系列的步驟,通過一個循環,以一種類似于狀態機模式的方式逐步驟執行:

    int HttpStreamFactoryImpl::Job::DoLoop(int result) {DCHECK_NE(next_state_, STATE_NONE);int rv = result;do {State state = next_state_;next_state_ = STATE_NONE;switch (state) {case STATE_START:DCHECK_EQ(OK, rv);rv = DoStart();break;case STATE_RESOLVE_PROXY:DCHECK_EQ(OK, rv);rv = DoResolveProxy();break;case STATE_RESOLVE_PROXY_COMPLETE:rv = DoResolveProxyComplete(rv);break;case STATE_WAIT:DCHECK_EQ(OK, rv);rv = DoWait();break;case STATE_WAIT_COMPLETE:rv = DoWaitComplete(rv);break;case STATE_INIT_CONNECTION:DCHECK_EQ(OK, rv);rv = DoInitConnection();break;case STATE_INIT_CONNECTION_COMPLETE:rv = DoInitConnectionComplete(rv);break;case STATE_WAITING_USER_ACTION:rv = DoWaitingUserAction(rv);break;case STATE_RESTART_TUNNEL_AUTH:DCHECK_EQ(OK, rv);rv = DoRestartTunnelAuth();break;case STATE_RESTART_TUNNEL_AUTH_COMPLETE:rv = DoRestartTunnelAuthComplete(rv);break;case STATE_CREATE_STREAM:DCHECK_EQ(OK, rv);rv = DoCreateStream();break;case STATE_CREATE_STREAM_COMPLETE:rv = DoCreateStreamComplete(rv);break;default:NOTREACHED() << "bad state";rv = ERR_FAILED;break;}} while (rv != ERR_IO_PENDING && next_state_ != STATE_NONE);return rv; }

    以執行一個QUIC請求為例,創建Stream的整個執行流程大體如下:


    CreateStream

    備選服務機制

    HttpStreamFactoryImpl::JobControllerCreateJobs()中我們看到,在通過job_factory創建main_job之后,會查找備選服務,在找到了備選服務時,還會為它創建job,并Start。那備選服務又是一套什么樣的機制呢?

    我們從兩個方面來探究這套機制究竟是什么樣的,及它又被用來做什么,一是備選服務的信息是從哪里及如何獲取的,二是備選服務對HttpStreamFactoryImpl::Job::Job的操作的影響。

    獲取備選服務信息

    我們先來看備選服務信息的獲取。在HttpStreamFactoryImpl::JobControllerCreateJobs()中,通過GetAlternativeServiceFor()來獲取備選服務的信息:

    AlternativeService HttpStreamFactoryImpl::JobController::GetAlternativeServiceFor(const HttpRequestInfo& request_info,HttpStreamRequest::Delegate* delegate,HttpStreamRequest::StreamType stream_type) {AlternativeService alternative_service =GetAlternativeServiceForInternal(request_info, delegate, stream_type);AlternativeServiceType type;if (alternative_service.protocol == UNINITIALIZED_ALTERNATE_PROTOCOL) {type = NO_ALTERNATIVE_SERVICE;} else if (alternative_service.protocol == QUIC) {if (request_info.url.host() == alternative_service.host) {type = QUIC_SAME_DESTINATION;} else {type = QUIC_DIFFERENT_DESTINATION;}} else {if (request_info.url.host() == alternative_service.host) {type = NOT_QUIC_SAME_DESTINATION;} else {type = NOT_QUIC_DIFFERENT_DESTINATION;}}UMA_HISTOGRAM_ENUMERATION("Net.AlternativeServiceTypeForRequest", type,MAX_ALTERNATIVE_SERVICE_TYPE);return alternative_service; }

    這里主要通過GetAlternativeServiceForInternal()獲取備選服務的信息:

    AlternativeService HttpStreamFactoryImpl::JobController::GetAlternativeServiceForInternal(const HttpRequestInfo& request_info,HttpStreamRequest::Delegate* delegate,HttpStreamRequest::StreamType stream_type) {GURL original_url = request_info.url;if (!original_url.SchemeIs("https"))return AlternativeService();url::SchemeHostPort origin(original_url);HttpServerProperties& http_server_properties =*session_->http_server_properties();const AlternativeServiceVector alternative_service_vector =http_server_properties.GetAlternativeServices(origin);if (alternative_service_vector.empty())return AlternativeService();bool quic_advertised = false;bool quic_all_broken = true;// First Alt-Svc that is not marked as broken.AlternativeService first_alternative_service;for (const AlternativeService& alternative_service :alternative_service_vector) {DCHECK(IsAlternateProtocolValid(alternative_service.protocol));if (!quic_advertised && alternative_service.protocol == QUIC)quic_advertised = true;if (http_server_properties.IsAlternativeServiceBroken(alternative_service)) {HistogramAlternateProtocolUsage(ALTERNATE_PROTOCOL_USAGE_BROKEN);continue;}// Some shared unix systems may have user home directories (like// http://foo.com/~mike) which allow users to emit headers. This is a bad// idea already, but with Alternate-Protocol, it provides the ability for a// single user on a multi-user system to hijack the alternate protocol.// These systems also enforce ports <1024 as restricted ports. So don't// allow protocol upgrades to user-controllable ports.const int kUnrestrictedPort = 1024;if (!session_->params().enable_user_alternate_protocol_ports &&(alternative_service.port >= kUnrestrictedPort &&origin.port() < kUnrestrictedPort))continue;if (alternative_service.protocol >= NPN_SPDY_MINIMUM_VERSION &&alternative_service.protocol <= NPN_SPDY_MAXIMUM_VERSION) {if (origin.host() != alternative_service.host &&!session_->params().enable_http2_alternative_service_with_different_host) {continue;}// Cache this entry if we don't have a non-broken Alt-Svc yet.if (first_alternative_service.protocol ==UNINITIALIZED_ALTERNATE_PROTOCOL)first_alternative_service = alternative_service;continue;}DCHECK_EQ(QUIC, alternative_service.protocol);if (origin.host() != alternative_service.host &&!session_->params().enable_quic_alternative_service_with_different_host) {continue;}quic_all_broken = false;if (!session_->params().enable_quic)continue;if (!IsQuicWhitelistedForHost(origin.host()))continue;if (stream_type == HttpStreamRequest::BIDIRECTIONAL_STREAM &&session_->params().quic_disable_bidirectional_streams) {continue;}if (session_->quic_stream_factory()->IsQuicDisabled(alternative_service.port))continue;if (!original_url.SchemeIs("https"))continue;// Check whether there is an existing QUIC session to use for this origin.HostPortPair mapped_origin(origin.host(), origin.port());ignore_result(ApplyHostMappingRules(original_url, &mapped_origin));QuicServerId server_id(mapped_origin, request_info.privacy_mode);HostPortPair destination(alternative_service.host_port_pair());ignore_result(ApplyHostMappingRules(original_url, &destination));if (session_->quic_stream_factory()->CanUseExistingSession(server_id,destination)) {return alternative_service;}// Cache this entry if we don't have a non-broken Alt-Svc yet.if (first_alternative_service.protocol == UNINITIALIZED_ALTERNATE_PROTOCOL)first_alternative_service = alternative_service;}// Ask delegate to mark QUIC as broken for the origin.if (quic_advertised && quic_all_broken && delegate != nullptr)delegate->OnQuicBroken();return first_alternative_service; }

    這個函數的執行流程如下:

  • 檢查Url的scheme是否為https,若不是,直接返回空的AlternativeService。即備選服務只用于https。
  • 從session_獲取HttpServerProperties http_server_properties。
  • 從http_server_properties獲取所有的備選服務信息。
  • 遍歷上一步找到的備選服務,找到一個可用的。
  • 在Chromium net中,以https為scheme的請求有多種,一是常規的HTTP/1.1 + TLS的請求,二是SPDY/HTTP2請求,三是QUIC協議的請求。備選服務主要用于后兩種協議的請求。這里會根據同源策略、端口、協議是否打開及主機是否在白名單等判斷一個備選服務是否可用。

    我們可以看到,備選服務的所有信息,都來源于http_server_properties。http_server_properties來源于HttpNetworkSession。HttpNetworkSession的http_server_properties在URLRequestContextBuilder::Build()中創建:

    void URLRequestContextBuilder::SetHttpNetworkSessionComponents(const URLRequestContext* context,HttpNetworkSession::Params* params) {params->host_resolver = context->host_resolver();params->cert_verifier = context->cert_verifier();params->transport_security_state = context->transport_security_state();params->cert_transparency_verifier = context->cert_transparency_verifier();params->ct_policy_enforcer = context->ct_policy_enforcer();params->proxy_service = context->proxy_service();params->ssl_config_service = context->ssl_config_service();params->http_auth_handler_factory = context->http_auth_handler_factory();params->http_server_properties = context->http_server_properties();params->net_log = context->net_log();params->channel_id_service = context->channel_id_service(); }std::unique_ptr<URLRequestContext> URLRequestContextBuilder::Build() { ......if (http_server_properties_) {storage->set_http_server_properties(std::move(http_server_properties_));} else {storage->set_http_server_properties(std::unique_ptr<HttpServerProperties>(new HttpServerPropertiesImpl()));} ......storage->set_http_network_session(base::WrapUnique(new HttpNetworkSession(network_session_params)));

    Cronet庫的初始化過程中,會執行CronetURLRequestContextAdapter::InitializeOnNetworkThread(),在這個方法中,通過URLRequestContextBuilder構建了URLRequestContext之后,會向其中添加備選服務的信息:

    void CronetURLRequestContextAdapter::InitializeOnNetworkThread(std::unique_ptr<URLRequestContextConfig> config,const base::android::ScopedJavaGlobalRef<jobject>&jcronet_url_request_context) { ......if (config->enable_quic) {for (auto hint = config->quic_hints.begin();hint != config->quic_hints.end(); ++hint) {const URLRequestContextConfig::QuicHint& quic_hint = **hint;if (quic_hint.host.empty()) {LOG(ERROR) << "Empty QUIC hint host: " << quic_hint.host;continue;}url::CanonHostInfo host_info;std::string canon_host(net::CanonicalizeHost(quic_hint.host, &host_info));if (!host_info.IsIPAddress() &&!net::IsCanonicalizedHostCompliant(canon_host)) {LOG(ERROR) << "Invalid QUIC hint host: " << quic_hint.host;continue;}if (quic_hint.port <= std::numeric_limits<uint16_t>::min() ||quic_hint.port > std::numeric_limits<uint16_t>::max()) {LOG(ERROR) << "Invalid QUIC hint port: "<< quic_hint.port;continue;}if (quic_hint.alternate_port <= std::numeric_limits<uint16_t>::min() ||quic_hint.alternate_port > std::numeric_limits<uint16_t>::max()) {LOG(ERROR) << "Invalid QUIC hint alternate port: "<< quic_hint.alternate_port;continue;}url::SchemeHostPort quic_server("https", canon_host, quic_hint.port);net::AlternativeService alternative_service(net::AlternateProtocol::QUIC, "",static_cast<uint16_t>(quic_hint.alternate_port));context_->http_server_properties()->SetAlternativeService(quic_server, alternative_service, base::Time::Max());}}

    這里更是限定了只允許給QUIC協議添加備選服務。而這里添加的備選服務的信息都來自于URLRequestContextConfig。

    繼續來看給HttpServerProperties添加備選服務信息的過程 (chromium_android/src/net/http/http_server_properties_impl.cc):

    bool HttpServerPropertiesImpl::SetAlternativeService(const url::SchemeHostPort& origin,const AlternativeService& alternative_service,base::Time expiration) {return SetAlternativeServices(origin,AlternativeServiceInfoVector(/*size=*/1, AlternativeServiceInfo(alternative_service, expiration))); }bool HttpServerPropertiesImpl::SetAlternativeServices(const url::SchemeHostPort& origin,const AlternativeServiceInfoVector& alternative_service_info_vector) {AlternativeServiceMap::iterator it = alternative_service_map_.Peek(origin);if (alternative_service_info_vector.empty()) {RemoveCanonicalHost(origin);if (it == alternative_service_map_.end())return false;alternative_service_map_.Erase(it);return true;}bool changed = true;if (it != alternative_service_map_.end()) {DCHECK(!it->second.empty());if (it->second.size() == alternative_service_info_vector.size()) {const base::Time now = base::Time::Now();changed = false;auto new_it = alternative_service_info_vector.begin();for (const auto& old : it->second) {// Persist to disk immediately if new entry has different scheme, host,// or port.if (old.alternative_service != new_it->alternative_service) {changed = true;break;}// Also persist to disk if new expiration it more that twice as far or// less than half as far in the future.base::Time old_time = old.expiration;base::Time new_time = new_it->expiration;if (new_time - now > 2 * (old_time - now) ||2 * (new_time - now) < (old_time - now)) {changed = true;break;}++new_it;}}}const bool previously_no_alternative_services =(GetAlternateProtocolIterator(origin) == alternative_service_map_.end());alternative_service_map_.Put(origin, alternative_service_info_vector);if (previously_no_alternative_services &&!GetAlternativeServices(origin).empty()) {// TODO(rch): Consider the case where multiple requests are started// before the first completes. In this case, only one of the jobs// would reach this code, whereas all of them should should have.HistogramAlternateProtocolUsage(ALTERNATE_PROTOCOL_USAGE_MAPPING_MISSING);}// If this host ends with a canonical suffix, then set it as the// canonical host.const char* kCanonicalScheme = "https";if (origin.scheme() == kCanonicalScheme) {const std::string* canonical_suffix = GetCanonicalSuffix(origin.host());if (canonical_suffix != nullptr) {url::SchemeHostPort canonical_server(kCanonicalScheme, *canonical_suffix,origin.port());canonical_host_to_origin_map_[canonical_server] = origin;}}return changed; }

    HttpServerPropertiesImpl用一個Map來管理備選服務的信息,key為原始服務的scheme+host+port,用url::SchemeHostPort來表示,而value則為AlternativeServiceInfoVector,即備選服務信息的列表。

    用戶添加備選服務信息

    在CronetEngine.Builder中 (chromium_android/src/components/cronet/android/api/src/org/chromium/net/CronetEngine.java),提供了接口,來添加QUIC服務器的一些信息:

    public abstract class CronetEngine {/*** A builder for {@link CronetEngine}s, which allows runtime configuration of* {@code CronetEngine}. Configuration options are set on the builder and* then {@link #build} is called to create the {@code CronetEngine}.*/public static class Builder {....../*** A hint that a host supports QUIC.* @hide only used by internal implementation.*/public static class QuicHint {// The host.public final String mHost;// Port of the server that supports QUIC.public final int mPort;// Alternate protocol port.public final int mAlternatePort;QuicHint(String host, int port, int alternatePort) {mHost = host;mPort = port;mAlternatePort = alternatePort;}}....../*** Adds hint that {@code host} supports QUIC.* Note that {@link #enableHttpCache enableHttpCache}* ({@link #HTTP_CACHE_DISK}) is needed to take advantage of 0-RTT* connection establishment between sessions.** @param host hostname of the server that supports QUIC.* @param port host of the server that supports QUIC.* @param alternatePort alternate port to use for QUIC.* @return the builder to facilitate chaining.*/public Builder addQuicHint(String host, int port, int alternatePort) {if (host.contains("/")) {throw new IllegalArgumentException("Illegal QUIC Hint Host: " + host);}mQuicHints.add(new QuicHint(host, port, alternatePort));return this;}/*** @hide only used by internal implementation.*/public List<QuicHint> quicHints() {return mQuicHints;}

    在CronetUrlRequestContext創建中,創建native UrlRequestContextConfig時會將所有的QUIC hint信息傳遞給native層。

    @VisibleForTestingpublic static long createNativeUrlRequestContextConfig(final Context context, CronetEngine.Builder builder) {final long urlRequestContextConfig = nativeCreateRequestContextConfig(builder.getUserAgent(), builder.storagePath(), builder.quicEnabled(),builder.getDefaultQuicUserAgentId(context), builder.http2Enabled(),builder.sdchEnabled(), builder.dataReductionProxyKey(),builder.dataReductionProxyPrimaryProxy(), builder.dataReductionProxyFallbackProxy(),builder.dataReductionProxySecureProxyCheckUrl(), builder.cacheDisabled(),builder.httpCacheMode(), builder.httpCacheMaxSize(), builder.experimentalOptions(),builder.mockCertVerifier(), builder.networkQualityEstimatorEnabled(),builder.publicKeyPinningBypassForLocalTrustAnchorsEnabled(),builder.certVerifierData());for (Builder.QuicHint quicHint : builder.quicHints()) {nativeAddQuicHint(urlRequestContextConfig, quicHint.mHost, quicHint.mPort,quicHint.mAlternatePort);}for (Builder.Pkp pkp : builder.publicKeyPins()) {nativeAddPkp(urlRequestContextConfig, pkp.mHost, pkp.mHashes, pkp.mIncludeSubdomains,pkp.mExpirationDate.getTime());}return urlRequestContextConfig;}

    nativeAddQuicHint()在chromium_android/src/components/cronet/android/cronet_url_request_context_adapter.cc中定義:

    // Add a QUIC hint to a URLRequestContextConfig. static void AddQuicHint(JNIEnv* env,const JavaParamRef<jclass>& jcaller,jlong jurl_request_context_config,const JavaParamRef<jstring>& jhost,jint jport,jint jalternate_port) {URLRequestContextConfig* config =reinterpret_cast<URLRequestContextConfig*>(jurl_request_context_config);config->quic_hints.push_back(base::WrapUnique(new URLRequestContextConfig::QuicHint(base::android::ConvertJavaStringToUTF8(env, jhost), jport,jalternate_port))); }

    備選服務對HttpStreamFactoryImpl::Job::Job的操作的影響

    為備選服務和為常規服務會以略有不同的方式創建Job:

    HttpStreamFactoryImpl::Job::Job(Delegate* delegate,JobType job_type,HttpNetworkSession* session,const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HostPortPair destination,GURL origin_url,NetLog* net_log): Job(delegate,job_type,session,request_info,priority,server_ssl_config,proxy_ssl_config,destination,origin_url,AlternativeService(),net_log) {}HttpStreamFactoryImpl::Job::Job(Delegate* delegate,JobType job_type,HttpNetworkSession* session,const HttpRequestInfo& request_info,RequestPriority priority,const SSLConfig& server_ssl_config,const SSLConfig& proxy_ssl_config,HostPortPair destination,GURL origin_url,AlternativeService alternative_service,NetLog* net_log): request_info_(request_info),priority_(priority),server_ssl_config_(server_ssl_config),proxy_ssl_config_(proxy_ssl_config),net_log_(BoundNetLog::Make(net_log, NetLog::SOURCE_HTTP_STREAM_JOB)),io_callback_(base::Bind(&Job::OnIOComplete, base::Unretained(this))),connection_(new ClientSocketHandle),session_(session),next_state_(STATE_NONE),pac_request_(NULL),destination_(destination),origin_url_(origin_url),alternative_service_(alternative_service),delegate_(delegate),job_type_(job_type),using_ssl_(false),using_spdy_(false),using_quic_(false),quic_request_(session_->quic_stream_factory()),using_existing_quic_session_(false),spdy_certificate_error_(OK),establishing_tunnel_(false),was_npn_negotiated_(false),protocol_negotiated_(kProtoUnknown),num_streams_(0),spdy_session_direct_(false),job_status_(STATUS_RUNNING),other_job_status_(STATUS_RUNNING),stream_type_(HttpStreamRequest::BIDIRECTIONAL_STREAM),ptr_factory_(this) {DCHECK(session);if (IsSpdyAlternative()) {DCHECK(origin_url_.SchemeIs("https"));}if (IsQuicAlternative()) {DCHECK(session_->params().enable_quic);using_quic_ = true;} }......bool HttpStreamFactoryImpl::Job::IsQuicAlternative() const {return alternative_service_.protocol == QUIC; }

    為QUIC備選服務創建的Job,在創建時期,就會將using_quic_置為true。這個標記的設置將對后續創建Stream的過程產生決定性的影響。

    總結一下,備選服務機制像是一種過渡方案,用于在協議開發早期,還沒有確定協議協商機制的情況下。在chromium net中,scheme為https的請求所用的協議有可能是HTTP/1.1+TLS、HTTP2和QUIC這三種的任一種,其中前兩種都是基于TCP的,而QUIC是基于UDP的。當前前面的兩種協議已經有了NPN和ALPN這樣的協議協商的機制,而傳給chromium net一個scheme為https的QUIC請求的URL,它也是不知道要用QUIC協議來做請求的。而備選服務機制,則允許chromium net的用戶指定,對某些主機的訪問采用特定的協議進行。此外,在HttpStreamFactoryImpl::JobController的CreateJobs()中,在為備選服務創建Job之外,還是會創建main_job,即是說,傳給chromium net一個以https為scheme的Url,它一定會嘗試用TCP的方式建立連接的,只是對于請求QUIC協議的服務,這個連接將會失敗,而真正取回數據的將是alternative_job。

    總結

    以上是生活随笔為你收集整理的Cronet android 设计与实现分析——备选服务机制的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    久久国产麻豆 | 男女激情麻豆 | 日本精品视频一区 | 成年人免费看的视频 | 伊人成人激情 | 91精品久久香蕉国产线看观看 | 中文字幕在线观看免费高清电影 | 西西大胆免费视频 | 国产一级特黄毛片在线毛片 | 国产一级a毛片视频爆浆 | 激情影院在线观看 | 免费看片网址 | 日韩毛片一区 | 色婷婷福利视频 | 色婷婷啪啪免费在线电影观看 | 91亚洲精品在线观看 | 在线观看久久久久久 | 亚洲精品在 | 国产97碰免费视频 | 久久久久婷 | 又色又爽又黄高潮的免费视频 | 欧美成人xxxxx| 成人高清av在线 | 久草网视频 | 91污在线观看 | 夜夜视频 | 最新中文字幕在线观看视频 | 五月天久久精品 | 麻豆传媒视频在线免费观看 | 在线观看色网 | 天天干天天草 | www.午夜| 91在线在线观看 | 超碰97在线资源 | 91在线小视频| 黄色字幕网 | 91漂亮少妇露脸在线播放 | 午夜少妇一区二区三区 | 在线视频18在线视频4k | 久久久久二区 | 久操视频在线播放 | 综合国产在线 | 91福利免费| 久久免费视频观看 | 国产不卡免费av | 国产中文字幕在线免费观看 | 综合激情伊人 | 成人免费一区二区三区在线观看 | 四虎成人在线 | 精品毛片久久久久久 | 色99中文字幕 | 精品在线99| 国产一区在线观看免费 | 日韩精品一区二区三区免费视频观看 | 欧美aaaxxxx做受视频 | 久久久国产精品久久久 | 狠狠狠色丁香综合久久天下网 | 麻花传媒mv免费观看 | 精品国产aⅴ麻豆 | 欧美在线观看小视频 | 免费在线黄色av | 爱av在线网 | 亚洲一区二区视频在线 | 免费精品在线视频 | 夜夜摸夜夜爽 | 免费国产在线视频 | 婷婷激情5月天 | 日韩三级免费 | 夜夜爱av | 国产精品毛片久久久久久久久久99999999 | 亚洲国产精品va在线看黑人动漫 | 精品久久视频 | 久草网视频在线观看 | 精品久久久久久久久久久久久久久久 | 午夜国产一区二区三区四区 | 欧美作爱视频 | 精品国产免费看 | 久草视频观看 | 国产成人精品综合久久久 | 天天射狠狠干 | 四虎在线免费观看视频 | 在线观看免费观看在线91 | 91丨九色丨首页 | 精品久久国产一区 | 玖草在线观看 | 在线a人片免费观看视频 | 久久久精品欧美一区二区免费 | av免费在线播放 | 四虎成人网 | 99久久精品国产一区二区成人 | 成人免费在线观看入口 | 人人干免费| 久草观看 | 国产欧美在线一区二区三区 | 香蕉久久久久久av成人 | 日韩高清久久 | 国产色视频一区二区三区qq号 | 日韩伦理片一区二区三区 | 五月婷婷综合在线视频 | 91在线视频网址 | 免费色黄 | 久久久久久久久久免费 | 日韩欧美在线影院 | 欧美成人xxx | 一区二区精 | 亚洲欧洲精品一区二区精品久久久 | 麻豆国产网站 | 欧美日韩中文字幕综合视频 | 99热国产在线 | 午夜精品剧场 | 日韩理论片中文字幕 | 成人欧美在线 | 国产高清在线精品 | 极品国产91在线网站 | 一区久久久 | 国产成人久久精品77777综合 | 麻豆91网站 | 色美女在线 | 国产精品麻豆视频 | 欧美久久久久久久久中文字幕 | 狠狠狠狠狠狠狠 | 国内精品国产三级国产aⅴ久 | 深夜福利视频一区二区 | 午夜12点 | 精品久久久久久久久久久院品网 | 日韩欧美视频一区二区 | 免费av网站在线看 | 日本精品免费看 | 国产色小视频 | 91天堂素人约啪 | 国产美女精品视频免费观看 | 四虎影视精品成人 | 99久久婷婷国产一区二区三区 | 国产小视频精品 | av超碰在线观看 | 日韩免费观看高清 | 久久久国产一区二区三区 | 亚洲精品黄色 | 中文字幕a∨在线乱码免费看 | 天堂在线视频免费观看 | 国产一区免费视频 | 久久99网 | 成人永久在线 | 在线 成人 | 国产黄色片一级三级 | 日韩在线视频播放 | 在线观看国产永久免费视频 | 97碰碰精品嫩模在线播放 | 日本精品久久久久中文字幕5 | 91视频观看免费 | 97福利社 | 国产精品免费观看久久 | 久久综合色一综合色88 | 天天天插 | 婷婷综合 | 久草在线视频精品 | 日本精品视频在线 | 久久人人爽人人爽人人片av软件 | 99r国产精品 | 久久久久久久久久久免费av | 午夜久操 | 国产精品免费视频网站 | 国产一级免费在线观看 | 在线观看v片 | 国产网红在线观看 | 国产精品成人在线 | 蜜臀精品久久久久久蜜臀 | 99这里只有精品视频 | 日韩大片在线播放 | 日日弄天天弄美女bbbb | 最近高清中文字幕 | 激情五月在线观看 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 九九免费视频 | 免费又黄又爽视频 | 91黄色在线看 | 国产精品美女久久久久久网站 | 国产美女视频网站 | 丁香六月天 | 国产精品一区二区三区免费看 | 天堂av在线 | 亚洲免费观看视频 | 精品国产一区二区三区av性色 | 国产裸体视频bbbbb | 色综合久久99 | 91在线在线观看 | 亚洲资源视频 | 99精品视频免费观看 | 九九视频这里只有精品 | 日韩精品1区2区 | 日韩精品一区二区三区在线播放 | 欧美另类网站 | 亚洲午夜激情网 | 网站你懂的 | 99精品国产99久久久久久97 | av在线不卡观看 | 国产黄a三级三级三级三级三级 | 欧美日韩中文字幕视频 | 97免费在线观看视频 | 久久99精品久久久久蜜臀 | 91九色视频导航 | 国产免费高清 | 欧美色精品天天在线观看视频 | 精品一区二区视频 | 日日夜夜天天综合 | 日韩在线观看中文 | 久久另类小说 | 九九涩涩av台湾日本热热 | av在线免费观看不卡 | 欧美精品免费在线观看 | 久久伦理 | 在线成人一区二区 | 人人澡超碰碰 | 免费看污在线观看 | 国产精品久久一卡二卡 | 日韩videos高潮hd | 国产成年人av | 欧美精品久久久久久久久久丰满 | 国产欧美精品一区aⅴ影院 99视频国产精品免费观看 | 18做爰免费视频网站 | av免费看看 | 黄色免费看片网站 | 婷婷 综合 色 | 国产精品高潮呻吟久久久久 | 成人黄大片 | 国产精品乱码一区二三区 | 亚洲精品视频在线看 | 最新av网址在线 | 激情视频在线高清看 | 欧美性受极品xxxx喷水 | 日韩视频一区二区三区在线播放免费观看 | 热久久国产精品 | 日韩免费二区 | www黄色av| 超碰97在线资源 | 黄色视屏免费在线观看 | 91自拍视频在线观看 | 中文字幕一区二区三区乱码不卡 | 天天射天天干天天操 | 黄色一级在线观看 | 国产精品精品 | av电影在线免费观看 | 久久久精品免费观看 | 国产精品a级| 最新高清无码专区 | 999在线视频 | 色噜噜狠狠狠狠色综合 | 成人欧美日韩国产 | 国产 字幕 制服 中文 在线 | 91香蕉视频好色先生 | 婷婷在线观看视频 | 一级黄色免费网站 | 中文字幕av免费在线观看 | 日本精品久久 | 中文字幕av播放 | 久久 地址 | 欧美一级欧美一级 | 日韩影视在线观看 | 中文字幕资源网在线观看 | av网站有哪些 | 久久国产精品99久久人人澡 | 亚洲天堂va | 亚洲高清视频在线观看 | 亚洲精品黄| 日本成人中文字幕在线观看 | 天天爱天天草 | 日韩精品视频免费专区在线播放 | 99人久久精品视频最新地址 | 国产一二区视频 | 天天天天天天天操 | 丁香婷婷色综合亚洲电影 | 久草在线视频网站 | 久久韩国免费视频 | 91久久国产综合精品女同国语 | 在线中文字幕av观看 | 国产美女精品视频 | 国内少妇自拍视频一区 | 麻豆va一区二区三区久久浪 | 色视频在线免费 | 中午字幕在线 | 久久人人添人人爽添人人88v | 国产视频不卡一区 | 亚洲一区二区精品视频 | 香蕉成人在线视频 | 奇米网8888 | 亚洲狠狠操 | 又黄又爽又刺激视频 | 国产精品18久久久久久不卡孕妇 | 精品在线观| 亚洲综合五月 | 久久精品99国产 | 欧美性大战 | 欧美一级激情 | 久久电影国产免费久久电影 | 午夜精品久久久99热福利 | 亚洲午夜久久久久久久久电影网 | 国产精品欧美精品 | 狠狠色噜噜狠狠狠狠2021天天 | 亚洲国产精品激情在线观看 | 天天爽网站 | 黄a在线看 | 亚洲成人黄色av | 亚洲欧美日韩一级 | 久久天堂亚洲 | 插婷婷| 国产福利电影网址 | 波多野结衣视频一区 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 日韩av午夜在线观看 | 免费久久久久久 | 免费午夜视频在线观看 | 日本成人中文字幕在线观看 | 免费高清在线观看成人 | 成人久久久精品国产乱码一区二区 | 97精品国产97久久久久久春色 | 国产精品久久电影网 | 四虎国产精品免费观看视频优播 | 成人在线播放视频 | 亚洲免费在线观看视频 | 久久9999久久免费精品国产 | 中文字幕在线观看第一页 | 久久九九久久精品 | 日本精品视频网站 | 国产精品完整版 | 青草草在线视频 | av电影免费观看 | 超碰在线免费97 | 精品国产一区二区三区久久久久久 | 国产精品一区二区在线播放 | 国产精品mv| 中文字幕在线网址 | 婷婷丁香在线 | 在线有码中文 | 91av电影在线 | 91理论片午午伦夜理片久久 | 国产一区二区在线观看视频 | 色婷婷综合成人av | 色综合久久66 | 国产精品一区二区三区免费看 | 久久视频国产精品免费视频在线 | 久久久91精品国产 | 在线免费看黄网站 | 久色婷婷| 精品久久久久久久久久久久 | 久久激情小视频 | 日韩三级不卡 | 色伊人网| 人人看人人艹 | 中文字幕丝袜一区二区 | 国产91av视频在线观看 | 精品色综合 | 美女久久99| 国产精品成人免费一区久久羞羞 | www色| 欧美a级在线 | 国产亚洲一级高清 | 超碰在线日本 | 日本精品久久久久 | 国产亚洲无| 免费在线观看午夜视频 | 日韩av中文字幕在线免费观看 | 国产成人一区二区三区 | 国产亚洲免费的视频看 | 天天舔天天射天天操 | 免费看搞黄视频网站 | 波多野结衣小视频 | 午夜私人影院久久久久 | 国产999精品久久久影片官网 | 国产福利免费在线观看 | 亚洲色图av| 天天射天天爱天天干 | 欧美另类高潮 | 亚洲黄色成人网 | 丝袜少妇在线 | 91理论电影 | 亚洲国产手机在线 | 免费看一级片 | 亚洲精品乱码久久久久久蜜桃动漫 | 精品视频久久久 | 欧美大片aaa | 亚洲午夜精品福利 | 国产高清在线视频 | 九色在线| 91精品视频网站 | 在线播放国产精品 | 日日干天天 | 狠狠躁日日躁狂躁夜夜躁 | 97久久精品午夜一区二区 | 91精品国产欧美一区二区 | 国产精品久久一卡二卡 | 高清av免费看| 亚洲精品国产自产拍在线观看 | www.久久99 | 精品亚洲免费 | 精品亚洲一区二区 | 91精品在线麻豆 | 国产手机视频精品 | 国产色一区| 欧美日韩aaaa | 日韩电影在线一区 | 91人人澡| 国产美女视频免费观看的网站 | 免费看污污视频的网站 | 久久久免费毛片 | 2019天天干天天色 | 久久兔费看a级 | 91精品在线视频观看 | 亚洲 综合 国产 精品 | 国产麻豆视频在线观看 | 国产资源在线播放 | 国产专区在线视频 | 亚洲精品影院在线观看 | 中文字幕在线免费观看视频 | 最新真实国产在线视频 | 高清不卡毛片 | 亚洲精品美女在线 | 狠狠综合网 | 久青草视频在线观看 | 亚洲精品天天 | 欧美另类sm图片 | 亚洲欧洲日韩在线观看 | 久久私人影院 | 国产高清第一页 | 在线免费观看视频一区 | 成人动漫一区二区三区 | 国产你懂的在线 | 欧美黄污视频 | 日韩特级片 | 超碰.com| 日本在线观看黄色 | 精品国产人成亚洲区 | 亚州精品天堂中文字幕 | 18网站在线观看 | 国产亚洲欧美精品久久久久久 | 91精品国产麻豆国产自产影视 | 久久理论影院 | 日本乱视频 | 天堂在线一区二区 | 日韩精品一区在线播放 | 久久综合九色综合欧美狠狠 | 日韩一级理论片 | 欧美视频一区二 | 精品国产伦一区二区三区 | 国产高清在线a视频大全 | 久久成人亚洲欧美电影 | 国产精品大尺度 | 久久福利小视频 | 黄色91在线 | 日韩欧美视频在线免费观看 | 国产午夜精品理论片在线 | 一本一本久久a久久精品综合妖精 | 国产精品色在线 | 欧美日本中文字幕 | 日韩高清在线一区 | 97视频入口免费观看 | 国产视频九色蝌蚪 | 欧美成人xxxxxxxx | 久久久蜜桃一区二区 | www免费网站在线观看 | 在线影院 国内精品 | 一区二区免费不卡在线 | 97狠狠操| 国产精品久久一 | 91插插视频| 日韩系列 | 成人91在线| 欧美一二区在线 | 日本黄色免费观看 | 手机av看片| 久久免费看 | aaa毛片视频 | 欧洲亚洲女同hd | 国产麻豆果冻传媒在线观看 | 国产拍揄自揄精品视频麻豆 | 蜜桃av人人夜夜澡人人爽 | 色噜噜日韩精品欧美一区二区 | 精品99在线 | 国产69精品久久久久9999apgf | 丁香六月伊人 | 日韩精品中文字幕一区二区 | 欧美,日韩| 日本不卡一区二区 | 在线观看亚洲成人 | 国产操在线 | 在线精品一区二区 | 国产精品美女久久久久久 | 国产生活一级片 | 日韩在线中文字幕视频 | 在线免费观看视频a | 嫩草91影院| 99久久综合狠狠综合久久 | 91亚洲精品久久久蜜桃网站 | 日本久久免费电影 | 欧美日韩精品影院 | 午夜视频免费在线观看 | 碰天天操天天 | 国产成人精品免高潮在线观看 | 狠狠色伊人亚洲综合网站色 | 激情五月婷婷网 | 美女网站一区 | 国产成人免费高清 | 国产成人精品一区二区三区福利 | 精品久久久久久久久久久久久久久久久久 | 爱色av.com| 日韩电影一区二区在线观看 | 欧美日韩国产精品一区二区 | 五月天堂网 | 亚洲亚洲精品在线观看 | 国产精品一区二区三区在线播放 | 91亚洲精品国产 | 日本性xxxxx 亚洲精品午夜久久久 | 99热这里是精品 | 欧美激情视频在线免费观看 | 胖bbbb搡bbbb擦bbbb| 欧美a级免费视频 | 日韩mv欧美mv国产精品 | 日本女人在线观看 | 在线观看日韩精品视频 | 国产精品免费久久 | 激情五月在线视频 | 日韩视频中文字幕在线观看 | 国语精品久久 | 特级aaa毛片 | 国产婷婷精品 | 99爱这里只有精品 | 久久一级电影 | 91成人免费电影 | 色综合天天 | 国产又粗又长的视频 | 日韩大片在线免费观看 | 国产视频网站在线观看 | 国产乱码精品一区二区三区介绍 | av免费看网站 | 麻豆免费在线视频 | 欧美性生活免费看 | 九九爱免费视频 | 免费a v网站 | 伊人天堂av | 久久久久国产一区二区三区 | 日韩精品一区二区电影 | 国产在线欧美日韩 | 日日夜夜精品视频天天综合网 | 在线观影网站 | 婷婷日 | 欧美另类tv| 免费在线a| 成人久久18免费网站麻豆 | 日日日日日 | 婷婷丁香在线视频 | 国产精品每日更新 | 久久久久欧美精品999 | 国产91精品一区二区麻豆网站 | 国产视频资源在线观看 | 国产精品99久久99久久久二8 | av不卡网站 | 在线日韩精品视频 | 九九九九免费视频 | 亚洲国产经典视频 | 婷婷激情影院 | 色一级片| 亚洲欧美日韩中文在线 | 国产剧情一区二区 | 中文字幕在线观看的网站 | 揉bbb玩bbb少妇bbb | 国产高清专区 | 丁香花在线视频观看免费 | 蜜桃视频日本 | 在线视频 成人 | 成人精品福利 | 在线v片| 成人精品99 | 亚洲精品国产品国语在线 | 亚av在线 | 中文字幕在线视频精品 | 操操操夜夜操 | 久日精品| 蜜桃av久久久亚洲精品 | 欧美日韩久久一区 | 毛片一级免费一级 | 视频二区在线 | 国产视频一区二区在线观看 | 六月丁香色婷婷 | 最新高清无码专区 | 天天操,夜夜操 | 亚洲免费激情 | 国产五月婷婷 | 成人a级网站| 婷婷综合五月 | 亚洲春色综合另类校园电影 | 在线看欧美| 波多野结衣一区三区 | 婷婷九月激情 | 天天爽夜夜爽精品视频婷婷 | 久久久黄色av | 亚洲开心激情 | 久久久久久免费网 | 婷婷五天天在线视频 | 国产精品免费久久久久久久久久中文 | 久久久久久久99精品免费观看 | 国产a国产| 亚洲成人黄色在线观看 | 91精品国自产在线观看欧美 | 成人在线观看av | 日韩色区| 99热9| www日韩欧美 | 久久99精品国产 | www日韩高清| 天天插一插 | 色综合中文字幕 | 日韩超碰在线 | 349k.cc看片app| 久久欧美综合 | 97超视频免费观看 | 成人av直播 | 日韩三级视频在线看 | 久艹在线播放 | 国产91免费在线 | 夜夜干天天操 | 久久毛片视频 | 午夜精品婷婷 | 亚洲黄色免费网站 | 狠狠色丁香婷婷综合最新地址 | 国产视频资源 | 欧美精品久久久久久久久久白贞 | 中文字幕av网站 | 五月婷婷丁香网 | 欧美日韩中文国产一区发布 | 中文字幕日本电影 | 国产在线综合视频 | 日日干av| 欧美久草网 | 成人免费xyz网站 | 天天色天天操天天爽 | 成人中文字幕在线观看 | 欧美一区二区在线免费看 | 在线视频麻豆 | 欧美 日韩 国产 中文字幕 | 日韩一级片大全 | 91在线视频播放 | 99精品国自产在线 | 九九激情视频 | 狠狠干综合 | 99色婷婷 | 国产精品区在线观看 | 国产一二区视频 | 亚洲精品999| 91麻豆精品国产自产在线 | 日韩在线观看精品 | 狠狠干综合网 | 国产精品久久99综合免费观看尤物 | 不卡av电影在线观看 | 狠狠色丁香婷婷综合久小说久 | 青青五月天 | 成人免费看片网址 | 国产一区二区久久 | 欧美日韩不卡在线观看 | 黄色免费高清视频 | 在线免费观看视频你懂的 | 国产91精品看黄网站在线观看动漫 | www操操| 91免费高清在线观看 | 成人在线免费看视频 | 日韩黄色在线电影 | 精品久久久久久亚洲综合网站 | 色在线网 | www91在线观看| 午夜精品婷婷 | 97精品欧美91久久久久久 | 色丁香婷婷 | 久久久久久蜜桃一区二区 | 免费福利在线播放 | 人人爱在线视频 | 国产精品少妇 | 国产精品涩涩屋www在线观看 | 99久久9 | 免费福利片 | 久久亚洲热 | 日本狠狠色 | 亚洲国产精品va在线看黑人动漫 | www.五月天婷婷.com | 日韩xxx视频 | 91人人爽人人爽人人精88v | 亚洲一区二区三区精品在线观看 | 久久99亚洲精品久久 | 天天射天天干天天插 | 美女黄视频免费 | 黄色特级毛片 | 国产盗摄精品一区二区 | 在线免费亚洲 | 国产91勾搭技师精品 | 国产98色在线 | 日韩 | 久久国产精品99久久久久久老狼 | 国产在线观看免费 | 国产视频精品久久 | 婷婷丁香在线视频 | 欧美特一级片 | 欧美 亚洲 另类 激情 另类 | jizz999 | 91亚州 | 日日夜夜天天久久 | 久久久久久久久久久久久久免费看 | 国产精品国产三级国产aⅴ9色 | 黄色三级av| 在线观看岛国av | 亚洲国产中文在线观看 | 又爽又黄又无遮挡网站动态图 | 国产精品自产拍在线观看蜜 | 亚洲精品自在在线观看 | 色爱区综合激月婷婷 | 国产视频99| 成人黄色在线电影 | 日韩av免费一区二区 | 午夜精品视频免费在线观看 | 成人av免费在线观看 | 99一区二区三区 | 色操插 | 黄色午夜网站 | 国产精品每日更新 | 色视频在线看 | 国产成人一区二区三区在线观看 | 久久成人人人人精品欧 | 国产日韩欧美在线看 | 国产美女视频 | 综合久久精品 | 99色在线播放 | 国产成人中文字幕 | 99婷婷狠狠成为人免费视频 | 日韩丝袜在线观看 | 亚洲激情网站免费观看 | 天天摸天天操天天舔 | 中文字幕第一页在线 | 国产91av视频在线观看 | 国产精品久久久久免费观看 | 中文字幕在线久一本久 | 久草亚洲视频 | 久久超碰在线 | 天天爽天天做 | 最近2019中文免费高清视频观看www99 | 操操操影院 | 激情综合网五月婷婷 | 午夜体验区 | 成年人在线观看视频免费 | 欧美成人一二区 | 99热只有精品在线观看 | 免费观看av网站 | 91人人人 | 日日爽天天操 | 国产精品一区二区在线观看免费 | 色综合天天综合网国产成人网 | 人人爱天天操 | 色视频在线看 | 丁香久久五月 | 337p日本欧洲亚洲大胆裸体艺术 | 碰碰影院 | 亚洲国产午夜 | 国产精品av久久久久久无 | 午夜私人影院久久久久 | 美女久久久久久久久久久 | 97视频免费观看 | 久久专区 | 精品久久久久久久久久久久久久久久 | 99视频在线观看一区三区 | 最新影院 | 国产又粗又硬又爽视频 | 午夜一级免费电影 | www视频免费在线观看 | 国产在线一卡 | 香蕉视频导航 | 欧美日韩国产精品久久 | 92国产精品久久久久首页 | 青青河边草免费直播 | 五月婷婷网站 | 午夜婷婷在线播放 | 国产这里只有精品 | 国产亚洲精品bv在线观看 | 亚洲欧美日韩不卡 | 日韩免费一区二区在线观看 | 免费久久精品视频 | 日韩av中文字幕在线免费观看 | 亚洲婷婷在线视频 | av在线免费在线观看 | 高潮久久久久久久久 | 国产精品一二 | 免费色视频在线 | 国产黄av| 全久久久久久久久久久电影 | 国产中文字幕三区 | 午夜国产福利视频 | 亚洲视频 视频在线 | av高清在线观看 | 欧美另类网站 | 久草在线视频免费资源观看 | 中文字幕资源在线观看 | 国产精品久久嫩一区二区免费 | 99精品久久久久 | 日本精品在线视频 | 久久久久亚洲精品中文字幕 | 日韩av不卡在线 | 久久视频中文字幕 | 久久新 | www.久久久| 麻豆一区二区三区视频 | 国产在线精品一区二区 | 毛片无卡免费无播放器 | 97电影在线 | 国内外成人在线 | 久久国际影院 | av大全在线免费观看 | 手机在线看片日韩 | av高清不卡| 天天操比 | 久久超级碰 | 亚洲精品网站 | 波多野结衣一区三区 | 国产啊v在线观看 | 91在线视频| 日韩色在线 | 五月婷婷.com | 美女免费视频观看网站 | 超碰人人在线 | 美女精品网站 | 一级黄色在线视频 | 久久久久久国产一区二区三区 | 99爱视频在线观看 | 色婷婷啪啪免费在线电影观看 | 97超视频在线观看 | 27xxoo无遮挡动态视频 | 久久亚洲综合国产精品99麻豆的功能介绍 | 天天操天天射天天舔 | 亚洲aaa毛片 | 狠狠色丁香婷婷综合 | 国产精品黄色 | 欧美性大战久久久久 | 精品在线你懂的 | 免费观看久久久 | 欧美精品久久久久久久 | 视频一区在线免费观看 | 久久一级电影 | 亚洲电影网站 | 国模精品一区二区三区 | 成人在线免费视频观看 | 超碰人人做 | 91精品国产入口 | 视频91在线 | 精品国产乱码久久久久久三级人 | 色综合狠狠干 | 日韩动态视频 | 99久久999久久久精玫瑰 | 欧美日韩中文在线视频 | 黄色最新网址 | 亚洲欧美国产日韩在线观看 | 久久三级视频 | 色www.| 国产特级毛片 | 国产少妇在线观看 | 亚洲dvd| 成人黄色资源 | 免费在线国产黄色 | 国产精品久久久久影院日本 | 天堂麻豆 | 久久久久国产一区二区三区四区 | 中文字幕中文字幕在线中文字幕三区 | 日韩高清在线看 | 国产精品青草综合久久久久99 | 特级黄色电影 | 亚洲欧美日韩不卡 | 香蕉视频一级 | 综合久久2023 | 美女视频又黄又免费 | 青草视频在线免费 | 中文在线免费观看 | 久久撸在线视频 | 免费人做人爱www的视 | 日韩在线视频网 | 一区二区三区 中文字幕 | 色婷婷久久久综合中文字幕 | 在线观看视频免费播放 | 久久久久 | 五月天综合网站 | 欧美一区二区在线免费观看 | 日韩午夜大片 | 欧美一区二区伦理片 | 日韩免费高清 | 麻豆国产精品va在线观看不卡 | 天天干夜夜擦 | 蜜臀久久99精品久久久无需会员 | 精品久久久久久综合日本 | 永久免费看av | 免费观看91视频 | 91精品视频免费看 | 日韩在线网址 | 国产一区高清在线 | 久久99热精品这里久久精品 | 久久国产精品久久w女人spa | 69国产成人综合久久精品欧美 | 毛片一二区| 国产黄色大片 | 国产二区精品 | 国产视频一区二区三区在线 | 日韩高清一区在线 | 9在线观看免费高清完整版在线观看明 | 亚洲免费不卡 | 久久久精品亚洲 | 特级西西444www高清大视频 | 二区在线播放 | 2024国产在线 | 日韩精品国产一区 | 久久精彩| av888.com| 日日狠狠| 国产小视频在线播放 | 国产精品二区在线 | 很黄很黄的网站免费的 | 性色av一区二区三区在线观看 | 91九色porny蝌蚪主页 | 日韩欧美亚州 | 亚洲精品系列 | 久久久久免费精品视频 | 亚洲视频高清 | 久久视频这里有久久精品视频11 | 在线观看av免费 | 最新影院| 国产69精品久久久久99尤 | 亚洲免费在线视频 | 免费在线国产视频 | 成全在线视频免费观看 | 天天操人人要 | 国产在线va | 在线观看免费观看在线91 | 亚洲成人资源网 | 啪嗒啪嗒免费观看完整版 | 丁香视频在线观看 | 成人免费视频观看 | 欧美伦理一区 | 黄色在线观看www | 日韩免费三区 | 国产精品亚州 | 国产涩图 | 中文资源在线官网 | 久久久鲁 | 99国产精品久久久久久久久久 | 色综合色综合色综合 | 久久久精品久久日韩一区综合 | av超碰在线观看 | 日本公妇在线观看高清 | 久久午夜电影院 | 国产精品久久久久久妇 | 久久视频网 | 中文免费在线观看 | 色偷偷88欧美精品久久久 | 精品视频久久久 | 日韩精品一区二区免费 | 久久午夜剧场 | 国产精品久久久久久欧美 | 蜜臀av夜夜澡人人爽人人 | 久久狠狠一本精品综合网 | 337p欧美| 综合久色| 成人免费一区二区三区在线观看 | 丁香六月久久综合狠狠色 | 亚洲高清av| 五月天国产 | 精一区二区 | 亚洲天堂网在线视频观看 | 丁香久久婷婷 | 91porny九色在线播放 | 欧美精品久久久久a | 人人爽人人乐 | 色老板在线视频 | 日韩av在线免费看 | 中文字幕亚洲欧美 | 在线看小早川怜子av | 人人干免费 | 亚洲国产综合在线 | 国产一级视频在线免费观看 | 久久国产成人午夜av影院宅 | 国产又粗又硬又爽的视频 | 久久久久久久久久久免费视频 | 伊人狠狠色 | 99久久久久久 | 国产精品久久久久一区二区国产 | 国产在线观看中文字幕 | 久久精品亚洲综合专区 | 日韩欧美在线观看一区二区 | 免费在线观看日韩 | 国产中文视 | 日韩一级电影在线观看 | 国内精品在线一区 | 亚洲专区免费观看 | 亚洲另类视频在线 | 在线亚洲欧美视频 | 亚洲精品视频在线观看免费视频 | 久久久久久亚洲精品 | 色综合久久天天 | 成年人视频在线免费观看 | www.午夜 | 日韩有码欧美 |