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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > HTML >内容正文

HTML

谷歌chrome浏览器的源码分析(五)

發布時間:2025/3/21 HTML 56 豆豆
生活随笔 收集整理的這篇文章主要介紹了 谷歌chrome浏览器的源码分析(五) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.


上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。





上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。




上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash。

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?









上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。


上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash。

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?


繼續上一次來分析LoadRequest的代碼,在分析這個函數代碼之前,先看看WebFrame類的繼承層次關系,如下:

class WebFrame : public base::RefCounted<WebFrame> {

WebFrame是一個接口類,但它先繼承引用計數類RefCounted,這樣對于這個對象多次訪問,就可以使用引用計數來判斷對象的生命周期了。對于base::RefCounted<WebFrame>的語法,其實它是一種模板實現的多態特性,這種方案是最高效的實現方式,比使用虛函數更少占內存,并且運行的速度也更快。它就是解決如下的問題:

??void Release() {

????if (subtle::RefCountedBase::Release()) {

??????delete static_cast<T*>(this);

????}

??}

上面的函數里static_cast<T*>(this),它就是一種多態的實現方法,由于base::RefCounted類并沒有聲明為虛析構函數,如下:

template <class T>

class RefCounted : public subtle::RefCountedBase {

?public:

??RefCounted() { }

??~RefCounted() { }

?

既然沒有把類RefCounted聲明為虛析構函數,又想在基類里調用派生類的析構函數,只好使用static_cast和類型轉換了,這是一種比較好的模板使用方法,在WTL里就大量使用這種技術。

接著可以看到:

class WebFrameImpl : public WebFrame {

?public:

??WebFrameImpl();

??~WebFrameImpl();

類WebFrameImpl是繼承接口類WebFrame,這里是使用接口與實現分析的設計模式,這樣更方便代碼靈活地復用。可見設計Chrome的設計師和寫代碼的程序員,都是頂尖的模板高手,大部的思想與WTL庫的設計是一脈相承。也難怪Chrome的瀏覽器使用WTL庫來設計界面。

?

#001??void WebFrameImpl::LoadRequest(WebRequest* request) {

#002????SubstituteData data;

#003????InternalLoadRequest(request, data, false);

#004??}

在WebFrame里調用函數LoadRequest,實際上是調用實現類WebFrameImpl函數LoadRequest,而在這個函數又是調用InternalLoadRequest來實現的,它的代碼如下:

#001??void WebFrameImpl::InternalLoadRequest(const WebRequest* request,

#002????????????????????????????????????????const SubstituteData& data,

#003????????????????????????????????????????bool replace) {

?

//轉換請求參數。

#004????const WebRequestImpl* request_impl =

#005????????static_cast<const WebRequestImpl*>(request);

#006?

?

獲取請求的資源。

#007????const ResourceRequest& resource_request =

#008????????request_impl->frame_load_request().resourceRequest();

#009?

#010????// Special-case?JavaScript?URLs.??Do not interrupt the existing load when

#011????// asked to load a javascript URL unless the script generates a result.

#012????// We can't just use FrameLoader::executeIfJavaScriptURL because it doesn't

#013????// handle redirects properly.

?

獲取需要下載網頁的地址。

#014????const KURL& kurl = resource_request.url();

?

處理加載javascript的連接情況。

#015????if (!data.isValid() && kurl.protocol() == "javascript") {

#016??????// Don't attempt to reload javascript URLs.

#017??????if (resource_request.cachePolicy() == ReloadIgnoringCacheData)

#018????????return;

#019?

#020??????// We can't load a javascript: URL if there is no Document!

#021??????if (!frame_->document())

#022????????return;

#023?

#024??????// TODO(darin): Is this the best API to use here???It works and seems good,

#025??????// but will it change out from under us?

#026??????DeprecatedString script =

#027?????????KURL::decode_string(kurl.deprecatedString().mid(sizeof("javascript:")-1));

#028??????bool succ = false;

?

加載執行腳本。

#029??????WebCore::String value =

#030??????????frame_->loader()->executeScript(script, &succ, true);

#031??????if (succ && !frame_->loader()->isScheduledLocationChangePending()) {

#032????????// TODO(darin): We need to figure out how to represent this in session

#033????????// history.??Hint: don't re-eval script when the user or script navigates

#034????????// back-n-forth (instead store the script result somewhere).

#035????????LoadDocumentData(kurl, value, String("text/html"), String());

#036??????}

#037??????return;

#038????}

#039?

?

停止上一次沒有完成的加載情況。

#040????StopLoading();??// make sure existing activity stops

#041?

#042????// Keep track of the request temporarily.??This is effectively a way of

#043????// passing the request to callbacks that may need it.??See

#044????// WebFrameLoaderClient::createDocumentLoader.

?

保存當前的請求連接。

#045????currently_loading_request_ = request;

#046?

#047????// If we have a current datasource, save the request info on it immediately.

#048????// This is because WebCore may not actually initiate a load on the toplevel

#049????// frame for some subframe navigations, so we want to update its request.

?

獲取當前數據源,如果已經存在就可以保存它。

#050????WebDataSourceImpl* datasource = GetDataSourceImpl();

#051????if (datasource)

#052??????CacheCurrentRequestInfo(datasource);

#053?

?

如果數據有效就可以直接替換就行了。

#054????if (data.isValid()) {

#055??????frame_->loader()->load(resource_request, data);

#056??????if (replace) {

#057????????// Do this to force WebKit to treat the load as replacing the currently

#058????????// loaded page.

#059????????frame_->loader()->setReplacing();

#060??????}

?

如果是歷史網頁選擇,就判斷是否出錯的加載處理。

#061????} else if (request_impl->history_item()) {

#062??????// Use the history item if we have one, otherwise fall back to standard

#063??????// load.

#064??????RefPtr<HistoryItem> current_item = frame_->loader()->currentHistoryItem();

#065?

#066??????// If there is no current_item, which happens when we are navigating in

#067??????// session history after a crash, we need to manufacture one otherwise

#068??????// WebKit hoarks. This is probably the wrong thing to do, but it seems to

#069??????// work.

#070??????if (!current_item) {

#071????????current_item = new HistoryItem(KURL("about:blank"), "");

#072????????frame_->loader()->setCurrentHistoryItem(current_item);

#073????????frame_->page()->backForwardList()->setCurrentItem(current_item.get());

#074?

#075????????// Mark the item as fake, so that we don't attempt to save its state and

#076????????// end up with about:blank in the navigation history.

#077????????frame_->page()->backForwardList()->setCurrentItemFake(true);

#078??????}

#079?

#080??????frame_->loader()->goToItem(request_impl->history_item().get(),

#081????????????????????????????????WebCore::FrameLoadTypeIndexedBackForward);

?

重新加載網頁。

#082????} else if (resource_request.cachePolicy() == ReloadIgnoringCacheData) {

#083??????frame_->loader()->reload();

?

?

下面開始調用load來加載新下載的網頁資源。

#084????} else {

#085??????frame_->loader()->load(resource_request);

#086????}

#087?

#088????currently_loading_request_ = NULL;

#089??}

?

上面通過幾種情況來分別實現了加載javascript網頁的處理,還有歷史選項處理,還有重新加載網頁和加載新網頁的處理。下一次再來分析加載新網頁的函數frame_->loader()->load的實現。







上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。


上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash。

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?


繼續上一次的分析,這里開始把連接址和其它相關的信息傳送frame_->loader()->load函數里面,那么在這個函數里面到底是怎么樣處理的呢,只有去分析它的代碼,我們才能找到它的答案,現在就來開始看吧,如下:

#001??void FrameLoader::load(const ResourceRequest& request)

#002??{

#003??????load(request, SubstituteData());

#004??}

在這個函數也只是一個中間者,它又調用函數load函數的重載函數來實現了。

#001??void FrameLoader::load(const ResourceRequest& request, const SubstituteData& substituteData)

#002??{

#003??????if (m_inStopAllLoaders)

#004??????????return;

#005?????????

#006??????// FIXME: is this the right place to reset loadType? Perhaps this should be done after loading is finished or aborted.

#007??????m_loadType = FrameLoadTypeStandard;

#008??????load(m_client->createDocumentLoader(request, substituteData).get());

#009??}

#010?

在這個函數里,第一個參數request是連接相關的信息,第二個參數substituteData是一些狀態數據。然后在第7行里設置加載的類型,第8行里調用WebFrameLoaderClient::createDocumentLoader函數來創建WebDocumentLoaderImpl對象,然后再通過get函數返回來,這樣就知道load函數又調用那個重載函數了,原來它是調用這個函數,如下:

#001??void FrameLoader::load(DocumentLoader* newDocumentLoader)

#002??{

#003??????ResourceRequest& r = newDocumentLoader->request();

#004??????addExtraFieldsToRequest(r, true, false);

#005??????FrameLoadType type;

#006?

#007??????if (shouldTreatURLAsSameAsCurrent(newDocumentLoader->originalRequest().url())) {

#008??????????r.setCachePolicy(ReloadIgnoringCacheData);

#009??????????type = FrameLoadTypeSame;

#010??????} else

#011??????????type = FrameLoadTypeStandard;

#012?

#013??????// Do not use original encoding override since it is not loaded by user

#014??????// selecting encoding.

#015??????if (m_documentLoader)

#016??????????newDocumentLoader->setOverrideEncoding(String());

#017?????

#018??????// When we loading alternate content for an unreachable URL that we're

#019??????// visiting in the b/f list, we treat it as a reload so the b/f list

#020??????// is appropriately maintained.

#021??????if (shouldReloadToHandleUnreachableURL(newDocumentLoader)) {

#022??????????ASSERT(type == FrameLoadTypeStandard);

#023??????????type = FrameLoadTypeReload;

#024??????}

#025?

#026??????load(newDocumentLoader, type, 0);

#027??}

上面只對newDocumentLoader做一些準備工作,并沒有真正地去加載任何東西,接著又調用函數:

void FrameLoader::load(DocumentLoader* loader, FrameLoadType type, PassRefPtr<FormState> formState)

在上面這個函數進行安全策略的處理,然后再經過N個函數處理之后,就調用下面的函數:

void FrameLoader::continueLoadAfterWillSubmitForm(PolicyAction)

在這個函數開始使用類DocumentLoader來設置下載請求,主要通過函數

bool DocumentLoader::startLoadingMainResource(unsigned long identifier)

來實現的,緊跟后面調用加載函數:

bool MainResourceLoader::load(const ResourceRequest& r, const SubstituteData&??substituteData)

在這個函數里又開始分為兩種情況處理,一種是延進加載數據,一種是立即加載數據,下面主要介紹立即加載數據函數:

bool MainResourceLoader::loadNow(ResourceRequest& r)

在類MainResourceLoader是主要資源下載的管理類,loadNow函數是把資源請求ResourceRequest變成一個IPC消息又發送給資源下載進程去處理。它的簡略代碼如下:

#001??bool MainResourceLoader::loadNow(ResourceRequest& r)

#002??{

......

#011??????willSendRequest(r, ResourceResponse());

#012?

......

?

#015??????if (!frameLoader())

#016??????????return false;

#017?????

......

#023?

#024??????if (m_substituteData.isValid())

#025??????????handleDataLoadSoon(r);

#026??????else if (shouldLoadEmpty || frameLoader()->representationExistsForURLScheme(url.protocol()))

#027??????????handleEmptyLoad(url, !shouldLoadEmpty);

#028??????else

#029??????????m_handle = ResourceHandle::create(r, this, m_frame.get(), false, true, true);

#030?

#031??????return false;

#032??}

在這個函數的第29行里,就會通過ResourceHandle::create函數創建一個資源消息,并把這個消息發送出去,到底它是怎么樣實現的呢?下一次再來分析它。





上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。


上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash。

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?


上一次說到需要把顯示的網絡連接地址變成一個資源的消息發送出去,它是通過函數ResourceHandle::create來實現的,但這個函數到底是怎么樣實現的呢?現在就分析它的實現代碼,了解它怎么樣把資源變換成消息,并且通過IPC機制把消息發送到資源下載進程去。數ResourceHandle::create的代碼如下:

#001??PassRefPtr<ResourceHandle> ResourceHandle::create(const ResourceRequest& request,

#002???????????????????????????????????????????????????ResourceHandleClient* client,

#003???????????????????????????????????????????????????Frame* deprecated,

#004???????????????????????????????????????????????????bool defersLoading,

#005???????????????????????????????????????????????????bool shouldContentSniff,

#006???????????????????????????????????????????????????bool mightDownloadFromHandle) {

上面的參數request是把所有請求網絡連接地址信息傳進來了。

?

#007????RefPtr<ResourceHandle> newHandle(

#008????????new ResourceHandle(request, client, defersLoading, shouldContentSniff,

#009??????????????????????????mightDownloadFromHandle));

這里創建資源類ResourceHandle對象,通過它來生成一個消息發送出去。

?

#010?

#011????if (newHandle->start(NULL))

#012??????return newHandle.release();

上面的代碼里,調用函數start來處理資源請求下載。

?

#013?

#014????return NULL;

#015??}

?

在這個函數里調用newHandle->start函數來處理,其實它是調用下面的函數來工作的:

bool ResourceHandle::start(Frame* deprecated) {

??return d->Start(NULL);

}

那么這里的d實例是什么呢?可以通過ResourceHandle的構造函數來看到它的類,如下:

ResourceHandle::ResourceHandle(const ResourceRequest& request,

???????????????????????????????ResourceHandleClient* client,

???????????????????????????????bool defersLoading,

???????????????????????????????bool shouldContentSniff,

???????????????????????????????bool mightDownloadFromHandle)

#pragma warning(suppress: 4355)??// it's okay to pass |this| here!

??????:?d(new ResourceHandleInternal(this, request, client))?{

??// TODO(darin): figure out what to do with the two bool params

}???????????????

可以看到d是類ResourceHandleInternal的實例,這就是說調用d->Start函數,其實就是調用下面的函數:

#001??bool ResourceHandleInternal::Start(

#002??????ResourceLoaderBridge::SyncLoadResponse* sync_load_response) {

#003????DCHECK(!bridge_.get());

#004?

#005????// The WebFrame is the Frame's FrameWinClient

#006????WebFrameImpl* webframe =

#007????????request_.frame() ? WebFrameImpl::FromFrame(request_.frame()) : NULL;

......

#154?

#155????if (sync_load_response) {

#156??????bridge_->SyncLoad(sync_load_response);

#157??????return true;

#158????}

#159?

?

通過上面的處理,然后就調用橋連接成員bridge_來創建消息。

#160????bool rv = bridge_->Start(this);

#161????if (rv) {

#162??????pending_ = true;

#163??????job_->ref();??// to be released when we get a OnCompletedRequest.

#164????} else {

#165??????bridge_.reset();

#166????}

#167?

#168????return rv;

#169??}

在這里使用一個設計模式,叫橋連接模式。函數bridge_->Start的代碼如下:

// Writes a footer on the message and sends it

bool IPCResourceLoaderBridge::Start(Peer* peer) {

??if (request_id_ != -1) {

????NOTREACHED() << "Starting a request twice";

????return false;

??}

?

??RESOURCE_LOG("Starting request for " << url_);

?

?

保存當前接收的連接端點。

??peer_ = peer;

?

?

生成請求ID,以便返回數據時可以找到相應的顯示進程和窗口。

??// generate the request ID, and append it to the message

??request_id_ = dispatcher_->AddPendingRequest(peer_, request_.resource_type,

??????????????????????????????????????????????request_.mixed_content);

?

?

找到IPC的消息發送對象,然后創建ViewHostMsg_RequestResource消息并發送出去。

??IPC::Message::Sender* sender = dispatcher_->message_sender();

??bool ret = false;

??if (sender)

????ret = sender->Send(new ViewHostMsg_RequestResource(MSG_ROUTING_NONE,

??????????????????????????????????????????????????????request_id_,

??????????????????????????????????????????????????????request_));

??return ret;

}

?

通過上面漫長的分析,總算搞清楚了這個過程:

從界面開始輸入URL地址,然后界面把URL發送到渲染進程,渲染進程再進行處理,把這個URL連接請求再次發送到資源下載進程去處理。串起來是一個極其簡單的過程,但在這個瀏覽器里比較復雜的,因為它是多進程的瀏覽器,進程之間相互消息傳送,就比其它瀏覽器復雜,并且它還有很多安全策略的使用和優化處理,導致這個處理過程是比較復雜的。

OK,資源下載請求消息已經發送出去,那么這個消息又往何處而去呢?又怎么樣通過網絡連接下載回來呢?欲知后事如何,請繼續看下一篇!



from:?

上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。


上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash。

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?


http://blog.csdn.net/caimouse/article/details/3071368

上一次說到類RenderThread和類RenderView把消息處理,那么這兩個類是怎么樣處理消息的呢?又是怎么樣處理瀏覽的消息呢?現在就帶著這兩個問題去分析它的源碼,理解它處理消息的方法。類RenderThread處理消息的代碼如下:

#001??void RenderThread::OnMessageReceived(const IPC::Message& msg) {

#002????// NOTE: We could subclass router_ to intercept OnControlMessageReceived, but

#003????// it seems simpler to just process any control messages that we care about

#004????// up-front and then send the rest of the messages onto router_.

#005?

?

下面判斷是控制消息,如果是控制消息就在本類里處理,否則就分發到別的地方處理,主要是轉到類RenderView處理。

#006????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#007??????IPC_BEGIN_MESSAGE_MAP(RenderThread, msg)

#008????????IPC_MESSAGE_HANDLER(ViewMsg_VisitedLink_NewTable, OnUpdateVisitedLinks)

#009????????IPC_MESSAGE_HANDLER(ViewMsg_SetNextPageID, OnSetNextPageID)

#010????????IPC_MESSAGE_HANDLER(ViewMsg_New, OnCreateNewView)

#011????????IPC_MESSAGE_HANDLER(ViewMsg_SetCacheCapacities, OnSetCacheCapacities)

#012????????IPC_MESSAGE_HANDLER(ViewMsg_GetCacheResourceStats,

#013???????????????????????????OnGetCacheResourceStats)

#014????????// send the rest to the router

#015????????IPC_MESSAGE_UNHANDLED(router_.OnMessageReceived(msg))

#016??????IPC_END_MESSAGE_MAP()

#017????} else {

?

這里是分發消息到別的地方處理。

#018??????router_.OnMessageReceived(msg);

#019????}

#020??}

?

在瀏覽器里,消息分為兩大類:控制消息和路由消息。像使用IPC_MESSAGE_CONTROL宏定義的消息,就是控制消息;使用IPC_MESSAGE_ROUTED宏定義的消息,就是路由消息。

路由消息分發是由類MessageRouter來負責的,主要處理的代碼如下:

#001?

#002??void MessageRouter::OnMessageReceived(const IPC::Message& msg) {

#003????if (msg.routing_id() == MSG_ROUTING_CONTROL) {

#004??????OnControlMessageReceived(msg);

#005????} else {

#006??????RouteMessage(msg);

#007????}

#008??}

在這里又分為MSG_ROUTING_CONTROL消息和其它路由消息,再一次通過函數RouteMessage分發之后,如下:

#001??bool MessageRouter::RouteMessage(const IPC::Message& msg) {

#002????IPC::Channel::Listener* listener = routes_.Lookup(msg.routing_id());

#003????if (!listener)

#004??????return false;

#005?

#006????listener->OnMessageReceived(msg);

#007????return true;

#008??}

上面這個函數里又把消息通過發送到listener里去,其實listener是根據消息的目標routing_id來選擇的,那么就是說它是選擇發送到不同的窗口里去,因為每個TAB一個窗口。消息經過這樣的處理之后,就到達了終點地---?RenderView::OnMessageReceived函數。下一次再來分析RenderView::OnMessageReceived函數的代碼和后繼處理。

上一次說到消息轉發,并分析了RenderThread類里處理消息的函數,其實大部份的消息都是在RenderView類里的OnMessageReceived函數處理,比如瀏覽的消息也是在這里處理。它的代碼如下:

#001??void RenderView::OnMessageReceived(const IPC::Message& message) {

#002????// Let the resource dispatcher intercept resource messages first.

?

如果是資源消息,就直接分發去處理,以便提高效率。

#003????if (resource_dispatcher_->OnMessageReceived(message))

#004??????return;

?

下面開始處理RenderView類里所有的消息。

#005????IPC_BEGIN_MESSAGE_MAP(RenderView, message)

#006??????IPC_MESSAGE_HANDLER(ViewMsg_CreatingNew_ACK, OnCreatingNewAck)

#007??????IPC_MESSAGE_HANDLER(ViewMsg_CaptureThumbnail, SendThumbnail)

#008??????IPC_MESSAGE_HANDLER(ViewMsg_GetPrintedPagesCount, OnGetPrintedPagesCount)

#009??????IPC_MESSAGE_HANDLER(ViewMsg_PrintPages, OnPrintPages)

#010??????IPC_MESSAGE_HANDLER(ViewMsg_Navigate, OnNavigate)

#011??????IPC_MESSAGE_HANDLER(ViewMsg_Stop, OnStop)

#012??????IPC_MESSAGE_HANDLER(ViewMsg_LoadAlternateHTMLText, OnLoadAlternateHTMLText)

#013??????IPC_MESSAGE_HANDLER(ViewMsg_StopFinding, OnStopFinding)

#014??????IPC_MESSAGE_HANDLER(ViewMsg_Undo, OnUndo)

#015??????IPC_MESSAGE_HANDLER(ViewMsg_Redo, OnRedo)

#016??????IPC_MESSAGE_HANDLER(ViewMsg_Cut, OnCut)

#017??????IPC_MESSAGE_HANDLER(ViewMsg_Copy, OnCopy)

#018??????IPC_MESSAGE_HANDLER(ViewMsg_Paste, OnPaste)

#019??????IPC_MESSAGE_HANDLER(ViewMsg_Replace, OnReplace)

#020??????IPC_MESSAGE_HANDLER(ViewMsg_Delete, OnDelete)

#021??????IPC_MESSAGE_HANDLER(ViewMsg_SelectAll, OnSelectAll)

#022??????IPC_MESSAGE_HANDLER(ViewMsg_CopyImageAt, OnCopyImageAt)

#023??????IPC_MESSAGE_HANDLER(ViewMsg_Find, OnFind)

#024??????IPC_MESSAGE_HANDLER(ViewMsg_AlterTextSize, OnAlterTextSize)

#025??????IPC_MESSAGE_HANDLER(ViewMsg_SetPageEncoding, OnSetPageEncoding)

#026??????IPC_MESSAGE_HANDLER(ViewMsg_InspectElement, OnInspectElement)

#027??????IPC_MESSAGE_HANDLER(ViewMsg_ShowJavaScriptConsole, OnShowJavaScriptConsole)

#028??????IPC_MESSAGE_HANDLER(ViewMsg_DownloadImage, OnDownloadImage)

#029??????IPC_MESSAGE_HANDLER(ViewMsg_ScriptEvalRequest, OnScriptEvalRequest)

#030??????IPC_MESSAGE_HANDLER(ViewMsg_AddMessageToConsole, OnAddMessageToConsole)

#031??????IPC_MESSAGE_HANDLER(ViewMsg_DebugAttach, OnDebugAttach)

#032??????IPC_MESSAGE_HANDLER(ViewMsg_DebugDetach, OnDebugDetach)

#033??????IPC_MESSAGE_HANDLER(ViewMsg_ReservePageIDRange, OnReservePageIDRange)

#034??????IPC_MESSAGE_HANDLER(ViewMsg_UploadFile, OnUploadFileRequest)

#035??????IPC_MESSAGE_HANDLER(ViewMsg_FormFill, OnFormFill)

#036??????IPC_MESSAGE_HANDLER(ViewMsg_FillPasswordForm, OnFillPasswordForm)

#037??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragEnter, OnDragTargetDragEnter)

#038??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragOver, OnDragTargetDragOver)

#039??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDragLeave, OnDragTargetDragLeave)

#040??????IPC_MESSAGE_HANDLER(ViewMsg_DragTargetDrop, OnDragTargetDrop)

#041??????IPC_MESSAGE_HANDLER(ViewMsg_AllowDomAutomationBindings,

#042?????????????????????????OnAllowDomAutomationBindings)

#043??????IPC_MESSAGE_HANDLER(ViewMsg_AllowBindings, OnAllowBindings)

#044??????IPC_MESSAGE_HANDLER(ViewMsg_SetDOMUIProperty, OnSetDOMUIProperty)

#045??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceEndedOrMoved, OnDragSourceEndedOrMoved)

#046??????IPC_MESSAGE_HANDLER(ViewMsg_DragSourceSystemDragEnded,

#047?????????????????????????OnDragSourceSystemDragEnded)

#048??????IPC_MESSAGE_HANDLER(ViewMsg_SetInitialFocus, OnSetInitialFocus)

#049??????IPC_MESSAGE_HANDLER(ViewMsg_FindReplyACK, OnFindReplyAck)

#050??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateTargetURL_ACK, OnUpdateTargetURLAck)

#051??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateWebPreferences, OnUpdateWebPreferences)

#052??????IPC_MESSAGE_HANDLER(ViewMsg_SetAltErrorPageURL, OnSetAltErrorPageURL)

#053??????IPC_MESSAGE_HANDLER(ViewMsg_InstallMissingPlugin, OnInstallMissingPlugin)

#054??????IPC_MESSAGE_HANDLER(ViewMsg_RunFileChooserResponse, OnFileChooserResponse)

#055??????IPC_MESSAGE_HANDLER(ViewMsg_EnableViewSourceMode, OnEnableViewSourceMode)

#056??????IPC_MESSAGE_HANDLER(ViewMsg_UpdateBackForwardListCount,

#057?????????????????????????OnUpdateBackForwardListCount)

#058?????IPC_MESSAGE_HANDLER(ViewMsg_GetAllSavableResourceLinksForCurrentPage,

#059?????????????????????????OnGetAllSavableResourceLinksForCurrentPage)

#060?????IPC_MESSAGE_HANDLER(ViewMsg_GetSerializedHtmlDataForCurrentPageWithLocalLinks,

#061?????????????????????????OnGetSerializedHtmlDataForCurrentPageWithLocalLinks)

#062??????IPC_MESSAGE_HANDLER(ViewMsg_GetApplicationInfo, OnGetApplicationInfo)

#063??????IPC_MESSAGE_HANDLER(ViewMsg_ShouldClose, OnMsgShouldClose)

#064??????IPC_MESSAGE_HANDLER(ViewMsg_ClosePage, OnClosePage)

#065??????IPC_MESSAGE_HANDLER(ViewMsg_ThemeChanged, OnThemeChanged)

#066??#ifdef CHROME_PERSONALIZATION

#067??????IPC_MESSAGE_HANDLER(ViewMsg_PersonalizationEvent, OnPersonalizationEvent)

#068??#endif

#069??????IPC_MESSAGE_HANDLER(ViewMsg_HandleMessageFromExternalHost,

#070??????????????????????????OnMessageFromExternalHost)

?

這里對于沒有處理的消息進行提示。

#071??????// Have the super handle all other messages.

#072??????IPC_MESSAGE_UNHANDLED(RenderWidget::OnMessageReceived(message))

#073????IPC_END_MESSAGE_MAP()

#074??}

?

從上面這個函數可以看到,它的消息處理是非常多的,下面來分析一個瀏覽網絡連接的消息,它就是ViewMsg_Navigate,可以看到這個消息后面響應函數是OnNavigate,也就是說,當你輸入網絡地址之后按回車,就會通過上說過的IPC機制把消息發送到這里,接著來看這個函數OnNavigate的代碼,如下:

#001??void RenderView::OnNavigate(const ViewMsg_Navigate_Params& params) {

?

判斷窗口是否關閉,如果關閉就不用去打開連接地址了。

#002????if (!webview())

#003??????return;

#004?

?

處理一些about的連接處理,比如about:crash

#005????AboutHandler::MaybeHandle(params.url);

#006?

?

保存是否重新加載網頁。

#007????bool is_reload = params.reload;

#008?

?

獲取WEB的顯示框架。

#009????WebFrame* main_frame = webview()->GetMainFrame();

?

判斷當是重新加載時,而當前又不是歷史網頁的情況。

#010????if (is_reload && !main_frame->HasCurrentState()) {

#011??????// We cannot reload if we do not have any history state.??This happens, for

#012??????// example, when recovering from a crash.??Our workaround here is a bit of

#013??????// a hack since it means that reload after a crashed tab does not cause an

#014??????// end-to-end cache validation.

#015??????is_reload = false;

#016????}

#017?

?

下面設置緩沖策略。

#018????WebRequestCachePolicy cache_policy;

#019????if (is_reload) {

#020??????cache_policy = WebRequestReloadIgnoringCacheData;

#021????} else if (params.page_id != -1 || main_frame->GetInViewSourceMode()) {

#022??????cache_policy = WebRequestReturnCacheDataElseLoad;

#023????} else {

#024??????cache_policy = WebRequestUseProtocolCachePolicy;

#025????}

#026?

?

下面創建一個下載請求,并把相關參數設置到請求里面。

#027????scoped_ptr<WebRequest> request(WebRequest::Create(params.url));

#028????request->SetCachePolicy(cache_policy);

#029????request->SetExtraData(new RenderViewExtraRequestData(

#030????????params.page_id, params.transition, params.url));

#031?

?

設置WEBKIT的請求狀態。

#032????// If we are reloading, then WebKit will use the state of the current page.

#033????// Otherwise, we give it the state to navigate to.

#034????if (!is_reload)

#035??????request->SetHistoryState(params.state);

#036?

?

讓主WEB顯示框架去下載請求顯示。

#037????main_frame->LoadRequest(request.get());

#038??}

?

分析這個函數,就可以知道處理瀏覽消息的過程,下一次來分析WebFrame里的接口函數LoadRequest處理過程,到底它是怎么樣處理JavaScript腳本網頁下載的呢?

總結

以上是生活随笔為你收集整理的谷歌chrome浏览器的源码分析(五)的全部內容,希望文章能夠幫你解決所遇到的問題。

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

91成人黄色 | 国产一级免费播放 | 国产一区av在线 | 国产成人精品一区二区三区免费 | 麻豆极品| a天堂中文在线 | 天天干com| 日韩三级视频在线观看 | 一本到视频在线观看 | 日日夜夜狠狠干 | 久久久国产毛片 | 狠狠色噜噜狠狠狠合久 | 中文在线√天堂 | 99热精品免费观看 | 丁香国产视频 | 在线视频 你懂得 | 天堂中文在线播放 | wwwav视频 | 久久精品一区二区三区视频 | 日韩欧美高清一区二区三区 | 在线视频你懂 | 日韩在线短视频 | 国产va在线 | 国产伦精品一区二区三区四区视频 | 黄色av成人在线观看 | 国内精品久久久久影院男同志 | 国产福利中文字幕 | 最新av电影网站 | 日韩免费福利 | 日韩在线观看视频网站 | 日日碰狠狠添天天爽超碰97久久 | 中文字幕在线观 | 国产日韩欧美在线一区 | 日韩毛片在线播放 | 在线导航av| 色婷婷五 | 久久精品视频观看 | 涩涩爱夜夜爱 | 欧美黑人巨大xxxxx | 免费视频你懂的 | 成人动漫一区二区三区 | 在线观看黄网站 | 黄色软件网站在线观看 | 久久久久五月 | 国内精自线一二区永久 | 国产精品久久久久久69 | 国产一级二级在线 | 视频高清| 91精品资源 | 99久久精品免费看 | 91福利社在线观看 | 国产精品视频地址 | 日韩美女久久 | 亚洲成人黄 | 91试看| aa级黄色大片 | 欧美久久久久久久 | 欧美乱淫视频 | 国产中文字幕视频在线观看 | 成人精品亚洲 | 顶级欧美色妇4khd | 国产精品国产三级国产aⅴ无密码 | 一区二区三区福利 | 久久久久黄 | 亚洲jizzjizz日本少妇 | 在线精品视频免费播放 | www.国产在线观看 | 日韩色视频在线观看 | 久久国产经典 | 夜夜摸夜夜爽 | 成人97视频一区二区 | 国内精品久久久久久中文字幕 | 精品久久久一区二区 | 久久成人18免费网站 | 国产大陆亚洲精品国产 | 黄色大片入口 | 99久久精品午夜一区二区小说 | 国内外成人免费在线视频 | 国产黑丝袜在线 | 久久精品久久精品久久 | 97视频免费观看 | 亚洲精品国产成人 | 91黄在线看 | 精品免费久久久久 | 免费在线91 | 日本在线视频一区二区三区 | 亚洲精品88欧美一区二区 | 色噜噜狠狠色综合中国 | 久久视精品 | 欧美精品免费在线 | 亚洲最新在线视频 | 久久国产视屏 | 国产精品久久久久影院 | 亚洲天堂精品视频在线观看 | 黄色影院在线播放 | 亚洲一区日韩精品 | 99999精品视频 | 夜色资源站国产www在线视频 | 在线播放 日韩专区 | 婷婷六月激情 | 99久久www免费 | 国产日本高清 | 亚一亚二国产专区 | 国产精品入口麻豆 | 天天玩天天干 | 中文字幕一区二区三区乱码不卡 | 国产色视频网站2 | 6080yy午夜一二三区久久 | 欧美老女人xx | 日日爽视频 | 美女国产精品 | 中文字幕中文字幕 | 久久精品国产美女 | 国产精品999久久久 久产久精国产品 | 日韩综合第一页 | h视频日本| 欧美日韩在线电影 | 免费的成人av | 97精品国自产拍在线观看 | 欧美激情视频一二区 | 亚洲精品乱码久久久一二三 | 色综合天天在线 | 亚洲午夜精品一区 | 三级午夜片 | 国产精品毛片一区二区 | 婷婷丁香在线视频 | av在线播放国产 | av最新资源 | 手机看片99 | 蜜桃视频成人在线观看 | 免费成人在线网站 | 久久精品久久久久电影 | 亚洲天堂网在线播放 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 日本3级在线观看 | 成人a在线观看高清电影 | 国产资源免费 | 一级黄色免费网站 | 国产一区二三区好的 | 天天看天天干天天操 | 久久久久免费看 | 亚洲黄色在线观看 | 在线观看资源 | 99久久激情视频 | 免费成人av | 免费观看特级毛片 | 国产国语在线 | 精品国产日本 | 国产在线免费 | 国产精品久久久久久久av大片 | 欧美韩国日本在线观看 | 天堂av免费 | 国产亚洲欧美日韩高清 | av超碰免费在线 | 日韩一区精品 | 日本不卡一区二区 | 久久精品久久精品久久 | 亚洲 欧洲 国产 日本 综合 | 日韩欧美综合在线视频 | www操操操| 亚洲国产色一区 | 色综合婷婷 | 黄色免费网站 | jizz18欧美18 | 国产麻豆精品95视频 | 超碰在线97免费 | 免费国产一区二区 | 国产成人精品日本亚洲999 | 一区二区三区在线播放 | 亚洲精色| 国产亚洲视频在线免费观看 | 欧美狠狠色 | 在线精品视频免费播放 | 国产精品成人久久久久 | 国产精品专区一 | 亚洲专区免费观看 | 香蕉视频啪啪 | 日日操网站 | 91精品国产高清自在线观看 | 91女神的呻吟细腰翘臀美女 | 亚洲黄色小说网址 | 成年人免费在线观看网站 | 亚洲欧美在线观看视频 | 国产中文字幕三区 | 日本精品视频在线观看 | 欧美日韩中文字幕在线视频 | 懂色av一区二区三区蜜臀 | 精品国产_亚洲人成在线 | 97精品国产97久久久久久免费 | 成人综合婷婷国产精品久久免费 | 日韩精品不卡 | 欧美国产日韩在线视频 | 丁香九月婷婷综合 | 欧美另类一二三四区 | 免费在线观看a v | 亚洲四虎| 国产高清久久 | 91视频啪 | 97人人人人 | 男女全黄一级一级高潮免费看 | 亚洲精品影视 | 日韩成人在线免费观看 | 综合精品在线 | 综合中文字幕 | 亚洲一区美女视频在线观看免费 | 久久久久久久久久久高潮一区二区 | 免费碰碰 | 中文资源在线官网 | 在线免费观看国产视频 | 久草资源在线观看 | 日韩精品久久一区二区三区 | 天天干天天插伊人网 | 日本中文字幕在线 | 亚洲视频久久久久 | 久久精品二区 | 五月开心激情 | 国产精品久久久影视 | 亚洲视频六区 | 国产黄色片在线 | 91在线在线观看 | 国产毛片久久 | 成人黄大片视频在线观看 | 精品久久网站 | 亚洲黄色免费电影 | 欧洲性视频 | 亚洲黄色在线观看 | 91传媒在线观看 | 国产精品综合av一区二区国产馆 | 99久久精品久久久久久清纯 | av电影中文字幕 | 国产精品久久在线 | 91福利在线观看 | 精品在线观看视频 | 一区二区三区四区精品视频 | 97精品国产一二三产区 | 精品免费久久久久 | 日本中文字幕观看 | 在线只有精品 | 亚洲精品激情 | 久草视频视频在线播放 | 久久免费看毛片 | 国产视频综合在线 | 国产精品高清av | 伊人婷婷激情 | 亚洲欧美视频在线 | 免费看黄色小说的网站 | 久久精品中文 | 国产精品系列在线 | 在线黄色av电影 | 国产精品久久久久永久免费看 | 成全在线视频免费观看 | 欧美男男激情videos | 午夜少妇一区二区三区 | 亚洲精品国精品久久99热一 | 性色视频在线 | 91欧美日韩国产 | 国产日韩三级 | 亚洲成人高清在线 | 人人玩人人添人人 | 国产精品成人av电影 | 日韩在线视频看看 | 久久精品网址 | 久久理论片 | 天天插天天操天天干 | 久久精品视频在线看 | 在线观看黄av | 超碰在线9 | 色婷婷狠狠五月综合天色拍 | 一区二区精品在线观看 | 中文字幕888 | 一区二区视频在线播放 | 中文字幕国产精品一区二区 | 成人99免费视频 | 超碰97中文 | 91麻豆操| 三级黄免费看 | 久久久久久久久久久国产精品 | 91桃色免费观看 | 国产99久久久久久免费看 | 国产精品 中文字幕 亚洲 欧美 | 亚洲国产中文在线观看 | 国产午夜小视频 | 日本视频高清 | 911国产在线观看 | 99色人 | 免费网站黄 | 国产在线观看一区 | h视频在线看 | www.激情五月.com | 黄色一级在线免费观看 | 午夜精品久久久久久久久久 | 日韩中文字幕国产精品 | 欧美精品亚洲精品日韩精品 | 99在线精品视频在线观看 | av日韩av| 免费三级大片 | 婷婷中文字幕 | 人人揉人人揉人人揉人人揉97 | 色婷婷综合久久久 | 久久免费视频在线观看 | 免费在线观看午夜视频 | 免费av的网站 | 亚洲专区路线二 | 激情av资源| 国产精品视频一二三 | 久久伊99综合婷婷久久伊 | 日本大片免费观看在线 | 日韩在线一级 | 久草视频中文在线 | 91在线免费看片 | 九七视频在线观看 | av免费在线网站 | a天堂免费 | 久久精品一级片 | 国产色婷婷在线 | 99re视频在线观看 | 麻豆传媒在线免费看 | 亚洲精区二区三区四区麻豆 | 久久久蜜桃一区二区 | 亚洲2019精品 | 欧美视频99 | 久久精久久精 | 97成人啪啪网 | 成人黄色电影在线观看 | 97av影院 | 黄色免费高清视频 | 亚洲精品久久在线 | 国产精品99久久久久久有的能看 | 麻豆小视频在线观看 | 三级黄色欧美 | 国产高清不卡一区二区三区 | 国产午夜精品理论片在线 | 久久久久久中文字幕 | 久久看片| 日韩久久一区二区 | 人人爽人人爽人人片av | 欧美九九视频 | 国产在线观看国语版免费 | 91亚洲精品国偷拍自产在线观看 | 国产精品一区电影 | 国产麻豆视频免费观看 | 国内精品久久久久久久 | 欧美色伊人 | 国产精品久久久久9999吃药 | 日本中文字幕影院 | 国产美女在线免费观看 | 成人午夜电影在线观看 | 三三级黄色片之日韩 | 国产精品成人自产拍在线观看 | 91网页版在线观看 | 免费观看www视频 | 国产不卡精品 | 91视频链接| 日本精品久久久久影院 | 欧美午夜性生活 | 日韩国产欧美在线播放 | 2022中文字幕在线观看 | 免费看黄在线网站 | 国产一级在线观看视频 | 久久免费国产精品1 | 伊人色综合久久天天网 | 91丨九色丨国产女 | 丁香婷婷久久久综合精品国产 | 我爱av激情网 | 美女av免费| 日韩va亚洲va欧美va久久 | 国产色黄网站 | 日本精品久久久久久 | 婷婷成人在线 | 狠狠躁夜夜a产精品视频 | 超碰精品在线观看 | av在线超碰 | 国产精品69久久久久 | 99999精品视频 | 国产91大片 | 最近日本mv字幕免费观看 | 天天操天天舔天天干 | 99视频这里有精品 | 欧美精品在线观看 | 亚洲欧美婷婷六月色综合 | 日韩毛片在线免费观看 | 色综合天天综合 | 久久久久欧美精品999 | 久久久久五月 | 久久久久激情视频 | 五月婷婷六月丁香 | 日韩欧美精品一区二区 | 激情网色 | 久久久这里有精品 | 男女免费视频观看 | 午夜久久网站 | 国产韩国精品一区二区三区 | 草久在线观看 | 日韩午夜电影网 | 99精品国产aⅴ | 13日本xxxxxⅹxxx20 | 在线观看黄色免费视频 | 亚洲婷婷在线视频 | 国产精品成人一区二区 | 国产一区视频导航 | 亚洲aaa毛片 | 色综合中文字幕 | 国产精品激情偷乱一区二区∴ | 五月天天色 | 在线观看中文字幕第一页 | 国产成人黄色网址 | 91看片成人 | 中文字幕丰满人伦在线 | 久久精品视频99 | 亚洲精品ww| 久久小视频 | 9992tv成人免费看片 | 久久久久免费视频 | 91福利社区在线观看 | 国产成人精品av在线 | 精品视频免费 | 国产福利专区 | 国产午夜精品福利视频 | 亚洲精品乱码久久久久久 | 久久婷婷影视 | 成人福利在线观看 | 成人免费观看网站 | 偷拍区另类综合在线 | 久久精品com| 中文在线a∨在线 | 91精品一区二区在线观看 | 亚洲,国产成人av | av在线免费观看黄 | 色婷婷九月 | 一区二区精品在线视频 | 伊人黄色网 | 97在线视频免费看 | 免费观看的黄色片 | 久久成| 日韩免费成人av | 婷婷九月丁香 | 午夜精品一二三区 | 亚洲精品久久久蜜臀下载官网 | 成人9ⅰ免费影视网站 | 久久精品五月 | 99热在线观看 | 97超视频| 欧美日韩在线视频一区 | 国产精品一区二区吃奶在线观看 | 亚洲理论视频 | www.夜夜操.com | 亚洲精品国产综合99久久夜夜嗨 | 亚洲国产mv| 视频二区在线 | 在线观看福利网站 | 国产精品亚州 | 日韩亚洲欧美中文字幕 | 国产特黄色片 | 国产精品免费在线播放 | 成 人 黄 色 视频 免费观看 | 国产综合精品一区二区三区 | 色综合久久中文综合久久牛 | 天天干,天天射,天天操,天天摸 | 日日夜夜噜 | 天天操夜夜叫 | 中文伊人 | 三级黄色网址 | 久久精品一区二区三区国产主播 | 国产精品99页 | 欧美一级片免费 | 激情综合电影网 | 伊人五月天.com | 高清不卡一区二区在线 | 97福利 | 欧美一区二区三区激情视频 | 麻豆国产精品一区二区三区 | 黄色一级免费网站 | 视频二区在线 | 中文字幕资源在线 | 色播五月激情五月 | 亚洲免费国产 | 嫩嫩影院理论片 | 人人爽人人乐 | 中文字幕在线观看免费观看 | 波多野结衣精品视频 | 免费观看丰满少妇做爰 | 97国产精品久久 | 日韩欧美一区二区三区黑寡妇 | 亚洲欧美国产精品va在线观看 | 91av在线电影 | 国产精品午夜av | 特级毛片在线观看 | 亚洲国产精品成人精品 | 亚洲欧美成人综合 | 欧美日韩一区二区久久 | 国产精品系列在线播放 | 人人澡视频| 日日爽天天爽 | 久久精品久久精品久久 | 狠狠狠狠狠狠狠狠 | 在线免费观看黄色av | 国产亚洲综合性久久久影院 | 亚洲免费高清视频 | 国产18精品乱码免费看 | 天天曰夜夜爽 | 久久精品在线免费观看 | 免费网站看av片 | 日韩av在线免费看 | 国产99在线免费 | 天天操夜夜操国产精品 | 日韩视频 一区 | 激情综合网五月激情 | 成人在线播放网站 | 69久久久| 日韩精品视频第一页 | 国产精品96久久久久久吹潮 | 国产群p视频 | 精品免费久久久久久 | 狠狠ri| 日韩av女优视频 | 色综合久久久久 | 午夜精品一区二区三区在线播放 | www.色午夜,com | 91资源在线 | 麻豆一区在线观看 | 黄网站app在线观看免费视频 | 丁香在线观看完整电影视频 | 色视频在线观看免费 | 精品免费久久久久久 | 日韩中文字幕在线看 | 亚洲一级片av| 日韩精品免费一区二区三区 | 一区电影| 亚洲乱码国产乱码精品天美传媒 | 日韩精品一区二区三区免费观看 | 欧美一区二区三区在线播放 | 91精品老司机久久一区啪 | 国产日韩欧美精品在线观看 | 欧美不卡视频在线 | 少妇搡bbbb搡bbb搡忠贞 | 91网站在线视频 | 午夜三级大片 | 国产不卡在线播放 | 最近高清中文字幕在线国语5 | 69人人| 国产一级片直播 | 日韩视频一区二区三区 | 91免费网址 | 岛国av在线不卡 | 久久久免费电影 | av 一区二区三区四区 | 亚州黄色一级 | 中文字幕在线观看第一区 | 国产精品美女久久久久久网站 | 久久久久欧美精品999 | 日本美女xx | 在线观看日韩 | 色在线视频 | 日韩 国产 | 久久久久久久久综合 | 亚洲国产日韩一区 | 国产精品资源网 | 97精品一区 | 99久久国产免费,99久久国产免费大片 | 激情小说 五月 | 一区二区电影在线观看 | 日本特黄特色aaa大片免费 | 91av视频免费观看 | 亚洲国内在线 | 久青草电影 | 黄色大片免费播放 | 亚洲欧美日韩一二三区 | 97在线播放| 国产精品岛国久久久久久久久红粉 | 欧美性黑人 | 欧美成亚洲 | 婷婷色资源| 狠狠色丁香久久婷婷综合五月 | 91成人精品一区在线播放69 | 男女男视频| 99精品欧美一区二区三区黑人哦 | 久草视频中文在线 | 亚洲 欧美日韩 国产 中文 | 91视频在线免费看 | 日本99干网 | 日日干视频 | 久草视频在线新免费 | 国产亚洲午夜高清国产拍精品 | 91麻豆精品国产91久久久更新时间 | 国产日韩精品一区二区三区在线 | 久久免费看视频 | 国产又黄又硬又爽 | 亚洲第二色 | 99热这里只有精品久久 | 国产不卡免费视频 | 国模精品一区二区三区 | 国产精品久久在线观看 | 久久超| 天天色天天色天天色 | 在线成人性视频 | 中文字幕专区高清在线观看 | 狠狠伊人 | 3d黄动漫免费看 | 人人插人人艹 | 日韩精品在线一区 | 国产不卡精品视频 | 国内精品在线观看视频 | 国产成人精品日本亚洲999 | 国产精品视频一二三 | 亚洲精品在线免费观看视频 | 亚洲精品9 | 999久久久久久 | av片在线看 | 久久天天躁狠狠躁亚洲综合公司 | 日韩精品一区在线观看 | 高清国产在线一区 | 激情黄色av| 五月婷婷伊人网 | 奇米网在线观看 | 狠狠色伊人亚洲综合网站野外 | 午夜12点| 国产精品区在线观看 | 伊人中文在线 | 久久视频免费观看 | 久久狠狠亚洲综合 | 欧美视频国产视频 | 久草在线免费播放 | 国产一区免费观看 | 国内精品久久久久久久久久久久 | 精品亚洲男同gayvideo网站 | 99精品欧美一区二区三区黑人哦 | 欧美日韩视频一区二区三区 | 九九久久成人 | 国产一级在线观看 | 久久99国产精品自在自在app | 精品99免费 | 香蕉久久久久久av成人 | 国产无遮挡又黄又爽在线观看 | 精品视频久久久久久 | 国产二区精品 | 美女网色| 国产成人精品一区二区三区在线观看 | 国产久草在线观看 | 免费网站v| 日色在线视频 | 色片网站在线观看 | 99久久精品国产亚洲 | 国产日韩视频在线播放 | 黄色中文字幕 | 久久久三级视频 | 国产在线观看二区 | 中午字幕在线 | 免费在线观看av网址 | 高清在线一区 | 国产亚洲91 | 国产真实精品久久二三区 | 四虎国产精品免费观看视频优播 | 亚洲专区视频在线观看 | av免费电影在线 | 欧美一级性生活 | 中午字幕在线 | 中文字幕在线人 | 日韩国产精品一区 | 99在线观看免费视频精品观看 | 国产在线观看午夜 | 很黄很污的视频网站 | 日韩日韩日韩日韩 | 国产不卡片 | 91精品国产九九九久久久亚洲 | 精品999久久久 | 六月婷婷久香在线视频 | 国产一级片观看 | 丁香婷婷激情国产高清秒播 | 精品国产伦一区二区三区观看体验 | 亚洲视频在线看 | 久久美女精品 | 国产成人三级在线观看 | 婷婷网址 | 国产原创中文在线 | 国产涩图 | 精品欧美小视频在线观看 | 国产精品一区二区在线免费观看 | 91av视频免费观看 | 日本在线视频网址 | 欧美一区二视频在线免费观看 | 国产精品观看在线亚洲人成网 | 黄在线免费看 | 亚洲精品福利在线观看 | 国产精品久久久久久电影 | 日韩av电影中文字幕在线观看 | 国产一区二区三精品久久久无广告 | 久久久精品国产一区二区 | 中国老女人日b | 中文字幕在线观看你懂的 | 精品国产一区二区三区四 | 00av视频| 超碰人人干人人 | www.狠狠| 欧美一进一出抽搐大尺度视频 | 最新av在线网站 | 国产精品乱码久久久 | 在线免费观看国产黄色 | www.av免费| 国产大陆亚洲精品国产 | 免费看国产曰批40分钟 | 人人模人人爽 | 天堂va欧美va亚洲va老司机 | 在线国产精品一区 | 天天干,天天射,天天操,天天摸 | 亚洲久草在线视频 | 国产极品尤物在线 | 久久久国产精品久久久 | 在线观看色网站 | 国产四虎在线 | 成年人在线观看 | 欧美精品久久久久 | 色射色 | 9热精品| 久久成人人人人精品欧 | 免费看污污视频的网站 | 午夜美女视频 | 欧美国产视频在线 | 五月天综合网 | 五月婷婷国产 | 国产精品久久嫩一区二区免费 | 亚洲清纯国产 | 人人搞人人干 | 天天草天天色 | 亚洲成人动漫在线观看 | 视频三区| 国产亚洲精品久久久久久久久久 | 人人爽人人爽人人爽学生一级 | 亚洲男男gaygay无套 | 色悠悠久久综合 | 久久艹在线 | 国产精彩视频一区二区 | a黄色片| 欧美亚洲国产一卡 | 中文字幕乱码视频 | 99久在线精品99re8热视频 | 色偷偷人人澡久久超碰69 | 在线看一级片 | 久爱精品在线 | 亚洲精品资源在线观看 | 亚洲国产视频网站 | 亚洲区视频在线观看 | 一区二区三区免费播放 | 久久免费试看 | 国产在线超碰 | 国产精品乱码久久久久久1区2区 | av在线电影免费观看 | 色就干| 亚洲精品一区二区三区在线观看 | 精品国产乱码一区二 | 一级电影免费在线观看 | 色多多在线观看 | 日韩精品在线一区 | 精品影院 | 91精品区 | 六月色丁香 | 美女网站黄免费 | 91麻豆免费视频 | 日韩一级黄色av | 高清日韩一区二区 | 一级黄色片在线观看 | 精品在线观看一区二区三区 | 最近中文字幕大全 | 丁香影院在线 | 99精品视频免费在线观看 | 人人玩人人爽 | 国产精品一区二区久久精品 | 视频成人免费 | 国产亚洲精品久久久久秋 | 麻豆高清免费国产一区 | 国产精品精品久久久久久 | 免费a v网站 | 中文字幕 在线 一 二 | 狠狠躁日日躁狂躁夜夜躁 | 久久99九九99精品 | 天天爱天天插 | 黄色av电影一级片 | 激情网站网址 | 日韩午夜精品 | 久久精品影片 | 91av在线视频免费观看 | 亚洲国产999| 日韩欧美一二三 | 久久久久免费观看 | 国产一卡二卡在线 | 免费在线观看亚洲视频 | 亚洲成人网在线 | 丁香六月中文字幕 | 久久精品欧美日韩精品 | 亚洲激情 欧美激情 | 欧美久草网| 青草视频在线 | 97超碰在线免费观看 | 日韩中文字幕视频在线观看 | 超碰公开在线观看 | 欧美日韩高清一区二区 国产亚洲免费看 | 欧美日韩精品电影 | 六月婷婷色 | 麻豆你懂的 | 婷婷夜夜 | 久久久久久不卡 | 91精品免费在线视频 | 日韩av视屏 | 中文字幕在线观看免费 | 国产免费一区二区三区最新6 | 最新日韩在线观看视频 | 一本到视频在线观看 | 亚洲国产成人av网 | www.国产毛片 | 九九综合久久 | 国产精品久久久久久久电影 | 欧美另类高清 | 久草爱 | 激情五月网站 | 天天综合色天天综合 | 婷婷深爱 | 成人av亚洲| 国产91精品看黄网站 | 欧美一区二区三区免费观看 | 美女视频黄网站 | 免费三级av | 激情综合婷婷 | 欧美另类高潮 | 久久色视频 | 国产精品麻豆三级一区视频 | 欧美日韩在线精品 | 久久久91精品国产 | 免费在线观看视频一区 | 久久乐九色婷婷综合色狠狠182 | 亚洲国内在线 | 久久久久久片 | 国产一区二区精 | 久久综合狠狠综合久久综合88 | 日本黄色免费观看 | 亚洲欧美一区二区三区孕妇写真 | 中文字幕日韩一区二区三区不卡 | 久久99深爱久久99精品 | 欧美成人手机版 | 亚洲成人资源网 | 久久国产午夜精品理论片最新版本 | 亚洲国内精品 | 久久久影片 | 国产日韩欧美在线看 | 插婷婷| 二区精品视频 | 国产精品久久久久三级 | 久久99精品久久久久久久久久久久 | 欧美性视频网站 | 天天操夜夜逼 | 久久精品亚洲国产 | 中文字幕a∨在线乱码免费看 | 中文日韩在线视频 | 99视频在线精品免费观看2 | 欧美精品在线视频观看 | 亚洲一区二区视频在线 | 国产一区不卡在线 | 黄色tv视频 | 婷婷六月色| 日日干网 | 成年人电影毛片 | 三级黄色大片在线观看 | 99热精品在线 | 国产免费久久 | 96超碰在线 | 麻豆影视在线观看 | 色多多视频在线观看 | 亚洲欧洲精品一区二区精品久久久 | 日韩久久久久久久 | 啪嗒啪嗒免费观看完整版 | 久久精品久久精品久久 | 国产精品手机在线播放 | 国色综合 | 最新午夜 | 免费色网站 | 中文字幕 国产视频 | 国产三级在线播放 | av丝袜天堂 | 8x成人免费视频 | 日韩在线高清视频 | 在线免费观看国产精品 | 香蕉在线观看 | 日韩免费高清 | 中文字幕电影网 | 伊人色播| 色91av| 精品视频专区 | 最新99热 | 久久国产精品免费观看 | 亚洲视频在线观看网站 | 欧美日韩在线免费观看 | 国产精选在线观看 | 99久久久久免费精品国产 | 久久久成人精品 | 欧美视频www| 久久精品老司机 | 麻豆视频免费在线 | 精品久久久久久亚洲综合网 | 伊人天天 | 最新国产精品久久精品 | 欧美成人免费在线 | 午夜三级理论 | 五月婷婷久久丁香 | www.夜夜干.com | 中国一区二区视频 | 日韩激情在线视频 | 日韩在线视频看看 | 成人免费在线视频观看 | 在线观看国产 | 中文字幕av免费观看 | 欧美a在线免费观看 | 天天操夜夜做 | 国产精品成人一区二区三区 | 久久99精品国产麻豆婷婷 | 三级小视频在线观看 | 久久成人欧美 | 一区二区高清在线 | 国模一区二区三区四区 | 日韩av黄| 成 人 黄 色 视频 免费观看 | 久久的色| 91桃花视频 | 国产精品免费成人 | 国产在线观看一 | 少妇搡bbbb搡bbb搡aa | 97av精品| 精品国产aⅴ麻豆 | 91视频链接 | www色av | 免费av的网站 | 成人a视频在线观看 | 精品一区二区三区香蕉蜜桃 | 免费成人在线视频网站 | 美腿丝袜一区二区三区 | 日韩欧美精品在线 | 99热这里有精品 | www天天操 | 人人添人人澡人人澡人人人爽 | 视频国产精品 | 久久久免费少妇 | 亚洲成a人片在线www | 在线观看日韩一区 | 亚洲天堂网在线视频观看 | 亚州精品天堂中文字幕 | 91在线www | 国产一区二区三区免费视频 | 久久久久久久久久久久国产精品 | 天天射,天天干 | 久久成人毛片 | 精品久久久国产 | 国产这里只有精品 | av中文在线观看 | 国产视频精品免费 | 狠狠做深爱婷婷综合一区 | a级片韩国| 国产视频97 | 亚洲激情p | 久久午夜精品 | 日韩a在线看| av综合av | 日韩中文字幕第一页 | 99免费精品视频 | 亚洲国产欧美在线人成大黄瓜 | 国产黄a三级三级三级三级三级 | 成年人免费在线观看 | 久久毛片网 | 国产成人一区二区三区久久精品 | 欧美日本日韩aⅴ在线视频 插插插色综合 | 亚洲九九精品 | av官网| 91视频最新网址 | 国产精品久久久999 国产91九色视频 | 波多在线视频 | 国产亚洲字幕 | 欧美坐爱视频 | 久久艹中文字幕 | 久操中文字幕在线观看 | 久久成人精品电影 | 91自拍视频在线观看 | 国内精品视频在线 | 欧美疯狂性受xxxxx另类 | av免费电影在线观看 | 在线观看的a站 | 中文字幕在线视频一区 | 亚洲欧美日韩精品久久奇米一区 | 麻豆超碰| 伊人久久精品久久亚洲一区 | 99精品偷拍视频一区二区三区 | 综合色中文 | 欧美做受高潮 | 亚洲综合爱| av手机在线播放 | 中国一级片在线播放 | 久久久久久久久久国产精品 | 狠狠色丁香久久综合网 | 亚洲一区av| www.777奇米| 91麻豆国产 | 激情五月播播久久久精品 | 亚洲一区二区观看 |