mirror of
				https://bitbucket.org/chromiumembedded/cef
				synced 2025-06-05 21:39:12 +02:00 
			
		
		
		
	Improve timing of frame attach/detach (see #3664)
- Move frame attachment from RenderFrameCreated to
  DidCommitProvisionalLoad. This has a number of advantages:
  - Significantly reduces the frequency of disconnects by avoiding
    the GetInterface/DidCommitNavigation race condition.
  - Stops connecting temporary frames (created during cross-origin
    navigation), making callback behavior more consistent.
- Split frame detach and destruction notifications into separate
  callbacks. OnFrameDetached now reflects a potentially recoverable
  state. Add a new OnFrameDestroyed callback for the unrecoverable
  destruction state.
			
			
This commit is contained in:
		| @@ -33,7 +33,7 @@ | |||||||
| // by hand. See the translator.README.txt file in the tools directory for | // by hand. See the translator.README.txt file in the tools directory for | ||||||
| // more information. | // more information. | ||||||
| // | // | ||||||
| // $hash=fc6fbee765ce2b649f5293c8c4b076d36014e4aa$ | // $hash=d8d06ee3d6e749d864e585fce8011d113b397220$ | ||||||
| // | // | ||||||
|  |  | ||||||
| #ifndef CEF_INCLUDE_CAPI_CEF_FRAME_HANDLER_CAPI_H_ | #ifndef CEF_INCLUDE_CAPI_CEF_FRAME_HANDLER_CAPI_H_ | ||||||
| @@ -76,6 +76,8 @@ extern "C" { | |||||||
| ///   objects are detached at the same time then notifications will be sent for | ///   objects are detached at the same time then notifications will be sent for | ||||||
| ///   any sub-frame objects before the main frame object. Commands can no longer | ///   any sub-frame objects before the main frame object. Commands can no longer | ||||||
| ///   be routed and will be discarded. | ///   be routed and will be discarded. | ||||||
|  | /// - CefFremeHadler::OnFrameDestroyed => An existing main frame or sub-frame | ||||||
|  | ///   object has been destroyed. | ||||||
| /// - cef_frame_handler_t::OnMainFrameChanged => A new main frame object has | /// - cef_frame_handler_t::OnMainFrameChanged => A new main frame object has | ||||||
| ///   been assigned to the browser. This will only occur with cross-origin | ///   been assigned to the browser. This will only occur with cross-origin | ||||||
| ///   navigation or re-navigation after renderer process termination (due to | ///   navigation or re-navigation after renderer process termination (due to | ||||||
| @@ -85,42 +87,32 @@ extern "C" { | |||||||
| /// - cef_frame_handler_t::OnFrameDetached => Any sub-frame objects have lost | /// - cef_frame_handler_t::OnFrameDetached => Any sub-frame objects have lost | ||||||
| ///   their connection to the renderer process. Commands can no longer be routed | ///   their connection to the renderer process. Commands can no longer be routed | ||||||
| ///   and will be discarded. | ///   and will be discarded. | ||||||
|  | /// - CefFreameHandler::OnFrameDestroyed => Any sub-frame objects have been | ||||||
|  | ///   destroyed. | ||||||
| /// - cef_life_span_handler_t::OnBeforeClose => The browser has been destroyed. | /// - cef_life_span_handler_t::OnBeforeClose => The browser has been destroyed. | ||||||
| /// - cef_frame_handler_t::OnFrameDetached => The main frame object have lost | /// - cef_frame_handler_t::OnFrameDetached => The main frame object have lost | ||||||
| ///   its connection to the renderer process. Notifications will be sent for any | ///   its connection to the renderer process. Notifications will be sent for any | ||||||
| ///   sub-frame objects before the main frame object. Commands can no longer be | ///   sub-frame objects before the main frame object. Commands can no longer be | ||||||
| ///   routed and will be discarded. | ///   routed and will be discarded. | ||||||
|  | /// - CefFreameHandler::OnFrameDestroyed => The main frame object has been | ||||||
|  | ///   destroyed. | ||||||
| /// - cef_frame_handler_t::OnMainFrameChanged => The final main frame object has | /// - cef_frame_handler_t::OnMainFrameChanged => The final main frame object has | ||||||
| ///   been removed from the browser. | ///   been removed from the browser. | ||||||
| /// | /// | ||||||
| /// Cross-origin navigation and/or loading receives special handling. | /// Special handling applies for cross-origin loading on creation/navigation of | ||||||
|  | /// sub-frames, and cross-origin loading on creation of new popup browsers. A | ||||||
|  | /// temporary frame will first be created in the parent frame's renderer | ||||||
|  | /// process. This temporary frame will never attach and will be discarded after | ||||||
|  | /// the real cross-origin frame is created in the new/target renderer process. | ||||||
|  | /// The client will receive creation callbacks for the temporary frame, followed | ||||||
|  | /// by cross-origin navigation callbacks (2) for the transition from the | ||||||
|  | /// temporary frame to the real frame. The temporary frame will not receive or | ||||||
|  | /// execute commands during this transitional period (any sent commands will be | ||||||
|  | /// discarded). | ||||||
| /// | /// | ||||||
| /// When the main frame navigates to a different origin the OnMainFrameChanged | /// When the main frame navigates to a different origin the OnMainFrameChanged | ||||||
| /// callback (2) will be executed with the old and new main frame objects. | /// callback (2) will be executed with the old and new main frame objects. | ||||||
| /// | /// | ||||||
| /// When a new sub-frame is loaded in, or an existing sub-frame is navigated to, |  | ||||||
| /// a different origin from the parent frame, a temporary sub-frame object will |  | ||||||
| /// first be created in the parent's renderer process. That temporary sub-frame |  | ||||||
| /// will then be discarded after the real cross-origin sub-frame is created in |  | ||||||
| /// the new/target renderer process. The client will receive cross-origin |  | ||||||
| /// navigation callbacks (2) for the transition from the temporary sub-frame to |  | ||||||
| /// the real sub-frame. The temporary sub-frame will not receive or execute |  | ||||||
| /// commands during this transitional period (any sent commands will be |  | ||||||
| /// discarded). |  | ||||||
| /// |  | ||||||
| /// When a new popup browser is created in a different origin from the parent |  | ||||||
| /// browser, a temporary main frame object for the popup will first be created |  | ||||||
| /// in the parent's renderer process. That temporary main frame will then be |  | ||||||
| /// discarded after the real cross-origin main frame is created in the |  | ||||||
| /// new/target renderer process. The client will receive creation and initial |  | ||||||
| /// navigation callbacks (1) for the temporary main frame, followed by cross- |  | ||||||
| /// origin navigation callbacks (2) for the transition from the temporary main |  | ||||||
| /// frame to the real main frame. The temporary main frame may receive and |  | ||||||
| /// execute commands during this transitional period (any sent commands may be |  | ||||||
| /// executed, but the behavior is potentially undesirable since they execute in |  | ||||||
| /// the parent browser's renderer process and not the new/target renderer |  | ||||||
| /// process). |  | ||||||
| /// |  | ||||||
| /// Callbacks will not be executed for placeholders that may be created during | /// Callbacks will not be executed for placeholders that may be created during | ||||||
| /// pre-commit navigation for sub-frames that do not yet exist in the renderer | /// pre-commit navigation for sub-frames that do not yet exist in the renderer | ||||||
| /// process. Placeholders will have cef_frame_t::get_identifier() == -4. | /// process. Placeholders will have cef_frame_t::get_identifier() == -4. | ||||||
| @@ -138,17 +130,33 @@ typedef struct _cef_frame_handler_t { | |||||||
|   /// Called when a new frame is created. This will be the first notification |   /// Called when a new frame is created. This will be the first notification | ||||||
|   /// that references |frame|. Any commands that require transport to the |   /// that references |frame|. Any commands that require transport to the | ||||||
|   /// associated renderer process (LoadRequest, SendProcessMessage, GetSource, |   /// associated renderer process (LoadRequest, SendProcessMessage, GetSource, | ||||||
|   /// etc.) will be queued until OnFrameAttached is called for |frame|. |   /// etc.) will be queued. The queued commands will be sent before | ||||||
|  |   /// OnFrameAttached or discarded before OnFrameDestroyed if the frame never | ||||||
|  |   /// attaches. | ||||||
|   /// |   /// | ||||||
|   void(CEF_CALLBACK* on_frame_created)(struct _cef_frame_handler_t* self, |   void(CEF_CALLBACK* on_frame_created)(struct _cef_frame_handler_t* self, | ||||||
|                                        struct _cef_browser_t* browser, |                                        struct _cef_browser_t* browser, | ||||||
|                                        struct _cef_frame_t* frame); |                                        struct _cef_frame_t* frame); | ||||||
|  |  | ||||||
|  |   /// | ||||||
|  |   /// Called when an existing frame is destroyed. This will be the last | ||||||
|  |   /// notification that references |frame| and cef_frame_t::is_valid() will | ||||||
|  |   /// return false (0) for |frame|. If called during browser destruction and | ||||||
|  |   /// after cef_life_span_handler_t::on_before_close() then | ||||||
|  |   /// cef_browser_t::is_valid() will return false (0) for |browser|. Any queued | ||||||
|  |   /// commands that have not been sent will be discarded before this callback. | ||||||
|  |   /// | ||||||
|  |   void(CEF_CALLBACK* on_frame_destroyed)(struct _cef_frame_handler_t* self, | ||||||
|  |                                          struct _cef_browser_t* browser, | ||||||
|  |                                          struct _cef_frame_t* frame); | ||||||
|  |  | ||||||
|   /// |   /// | ||||||
|   /// Called when a frame can begin routing commands to/from the associated |   /// Called when a frame can begin routing commands to/from the associated | ||||||
|   /// renderer process. |reattached| will be true (1) if the frame was re- |   /// renderer process. |reattached| will be true (1) if the frame was re- | ||||||
|   /// attached after exiting the BackForwardCache. Any commands that were queued |   /// attached after exiting the BackForwardCache or after encountering a | ||||||
|   /// have now been dispatched. |   /// recoverable connection error. Any queued commands will now have been | ||||||
|  |   /// dispatched. This function will not be called for temporary frames created | ||||||
|  |   /// during cross-origin navigation. | ||||||
|   /// |   /// | ||||||
|   void(CEF_CALLBACK* on_frame_attached)(struct _cef_frame_handler_t* self, |   void(CEF_CALLBACK* on_frame_attached)(struct _cef_frame_handler_t* self, | ||||||
|                                         struct _cef_browser_t* browser, |                                         struct _cef_browser_t* browser, | ||||||
| @@ -156,12 +164,19 @@ typedef struct _cef_frame_handler_t { | |||||||
|                                         int reattached); |                                         int reattached); | ||||||
|  |  | ||||||
|   /// |   /// | ||||||
|   /// Called when a frame loses its connection to the renderer process and will |   /// Called when a frame loses its connection to the renderer process. This may | ||||||
|   /// be destroyed. Any pending or future commands will be discarded and |   /// occur when a frame is destroyed, enters the BackForwardCache, or | ||||||
|   /// cef_frame_t::is_valid() will now return false (0) for |frame|. If called |   /// encounters a rare connection error. In the case of frame destruction this | ||||||
|   /// after cef_life_span_handler_t::on_before_close() during browser |   /// call will be followed by a (potentially async) call to OnFrameDestroyed. | ||||||
|   /// destruction then cef_browser_t::is_valid() will return false (0) for |   /// If frame destruction is occuring synchronously then | ||||||
|   /// |browser|. |   /// cef_frame_t::is_valid() will return false (0) for |frame|. If called | ||||||
|  |   /// during browser destruction and after | ||||||
|  |   /// cef_life_span_handler_t::on_before_close() then cef_browser_t::is_valid() | ||||||
|  |   /// will return false (0) for |browser|. If, in the non-destruction case, the | ||||||
|  |   /// same frame later exits the BackForwardCache or recovers from a connection | ||||||
|  |   /// error then there will be a follow-up call to OnFrameAttached. This | ||||||
|  |   /// function will not be called for temporary frames created during cross- | ||||||
|  |   /// origin navigation. | ||||||
|   /// |   /// | ||||||
|   void(CEF_CALLBACK* on_frame_detached)(struct _cef_frame_handler_t* self, |   void(CEF_CALLBACK* on_frame_detached)(struct _cef_frame_handler_t* self, | ||||||
|                                         struct _cef_browser_t* browser, |                                         struct _cef_browser_t* browser, | ||||||
| @@ -173,14 +188,14 @@ typedef struct _cef_frame_handler_t { | |||||||
|   /// navigation after renderer process termination (due to crashes, etc). |   /// navigation after renderer process termination (due to crashes, etc). | ||||||
|   /// |old_frame| will be NULL and |new_frame| will be non-NULL when a main |   /// |old_frame| will be NULL and |new_frame| will be non-NULL when a main | ||||||
|   /// frame is assigned to |browser| for the first time. |old_frame| will be |   /// frame is assigned to |browser| for the first time. |old_frame| will be | ||||||
|   /// non-NULL and |new_frame| will be NULL and  when a main frame is removed |   /// non-NULL and |new_frame| will be NULL when a main frame is removed from | ||||||
|   /// from |browser| for the last time. Both |old_frame| and |new_frame| will be |   /// |browser| for the last time. Both |old_frame| and |new_frame| will be non- | ||||||
|   /// non-NULL for cross-origin navigations or re-navigation after renderer |   /// NULL for cross-origin navigations or re-navigation after renderer process | ||||||
|   /// process termination. This function will be called after on_frame_created() |   /// termination. This function will be called after on_frame_created() for | ||||||
|   /// for |new_frame| and/or after on_frame_detached() for |old_frame|. If |   /// |new_frame| and/or after on_frame_destroyed() for |old_frame|. If called | ||||||
|   /// called after cef_life_span_handler_t::on_before_close() during browser |   /// during browser destruction and after | ||||||
|   /// destruction then cef_browser_t::is_valid() will return false (0) for |   /// cef_life_span_handler_t::on_before_close() then cef_browser_t::is_valid() | ||||||
|   /// |browser|. |   /// will return false (0) for |browser|. | ||||||
|   /// |   /// | ||||||
|   void(CEF_CALLBACK* on_main_frame_changed)(struct _cef_frame_handler_t* self, |   void(CEF_CALLBACK* on_main_frame_changed)(struct _cef_frame_handler_t* self, | ||||||
|                                             struct _cef_browser_t* browser, |                                             struct _cef_browser_t* browser, | ||||||
|   | |||||||
| @@ -42,13 +42,13 @@ | |||||||
| // way that may cause binary incompatibility with other builds. The universal | // way that may cause binary incompatibility with other builds. The universal | ||||||
| // hash value will change if any platform is affected whereas the platform hash | // hash value will change if any platform is affected whereas the platform hash | ||||||
| // values will change only if that particular platform is affected. | // values will change only if that particular platform is affected. | ||||||
| #define CEF_API_HASH_UNIVERSAL "8d6d53c5732a12b4d663665e3745a93d1cfd742d" | #define CEF_API_HASH_UNIVERSAL "7ce563b3df3a227c06d331774d5fa51c8421a7f1" | ||||||
| #if defined(OS_WIN) | #if defined(OS_WIN) | ||||||
| #define CEF_API_HASH_PLATFORM "3593cc6344b391d992421dd985c4ebcc46d8314f" | #define CEF_API_HASH_PLATFORM "0b070263fe1c7304bd90bd7a8338fcc08410d299" | ||||||
| #elif defined(OS_MAC) | #elif defined(OS_MAC) | ||||||
| #define CEF_API_HASH_PLATFORM "c045e75415a6abc2c29a3e1e05baea7528e2ec28" | #define CEF_API_HASH_PLATFORM "e99938b49f333ccda55ea34dfb36dd65a6b2cacd" | ||||||
| #elif defined(OS_LINUX) | #elif defined(OS_LINUX) | ||||||
| #define CEF_API_HASH_PLATFORM "b1058e8b167cfaa2f0feaccf4b4f23a813db515a" | #define CEF_API_HASH_PLATFORM "45669721b97760d8cda9ea29a5bd070d38267424" | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #ifdef __cplusplus | #ifdef __cplusplus | ||||||
|   | |||||||
| @@ -68,6 +68,8 @@ | |||||||
| ///   objects are detached at the same time then notifications will be sent for | ///   objects are detached at the same time then notifications will be sent for | ||||||
| ///   any sub-frame objects before the main frame object. Commands can no longer | ///   any sub-frame objects before the main frame object. Commands can no longer | ||||||
| ///   be routed and will be discarded. | ///   be routed and will be discarded. | ||||||
|  | /// - CefFremeHadler::OnFrameDestroyed => An existing main frame or sub-frame | ||||||
|  | ///   object has been destroyed. | ||||||
| /// - CefFrameHandler::OnMainFrameChanged => A new main frame object has been | /// - CefFrameHandler::OnMainFrameChanged => A new main frame object has been | ||||||
| ///   assigned to the browser. This will only occur with cross-origin navigation | ///   assigned to the browser. This will only occur with cross-origin navigation | ||||||
| ///   or re-navigation after renderer process termination (due to crashes, etc). | ///   or re-navigation after renderer process termination (due to crashes, etc). | ||||||
| @@ -76,42 +78,32 @@ | |||||||
| /// - CefFrameHandler::OnFrameDetached => Any sub-frame objects have lost their | /// - CefFrameHandler::OnFrameDetached => Any sub-frame objects have lost their | ||||||
| ///   connection to the renderer process. Commands can no longer be routed and | ///   connection to the renderer process. Commands can no longer be routed and | ||||||
| ///   will be discarded. | ///   will be discarded. | ||||||
|  | /// - CefFreameHandler::OnFrameDestroyed => Any sub-frame objects have been | ||||||
|  | ///   destroyed. | ||||||
| /// - CefLifeSpanHandler::OnBeforeClose => The browser has been destroyed. | /// - CefLifeSpanHandler::OnBeforeClose => The browser has been destroyed. | ||||||
| /// - CefFrameHandler::OnFrameDetached => The main frame object have lost its | /// - CefFrameHandler::OnFrameDetached => The main frame object have lost its | ||||||
| ///   connection to the renderer process. Notifications will be sent for any | ///   connection to the renderer process. Notifications will be sent for any | ||||||
| ///   sub-frame objects before the main frame object. Commands can no longer be | ///   sub-frame objects before the main frame object. Commands can no longer be | ||||||
| ///   routed and will be discarded. | ///   routed and will be discarded. | ||||||
|  | /// - CefFreameHandler::OnFrameDestroyed => The main frame object has been | ||||||
|  | ///   destroyed. | ||||||
| /// - CefFrameHandler::OnMainFrameChanged => The final main frame object has | /// - CefFrameHandler::OnMainFrameChanged => The final main frame object has | ||||||
| ///   been removed from the browser. | ///   been removed from the browser. | ||||||
| /// | /// | ||||||
| /// Cross-origin navigation and/or loading receives special handling. | /// Special handling applies for cross-origin loading on creation/navigation of | ||||||
|  | /// sub-frames, and cross-origin loading on creation of new popup browsers. A | ||||||
|  | /// temporary frame will first be created in the parent frame's renderer | ||||||
|  | /// process. This temporary frame will never attach and will be discarded after | ||||||
|  | /// the real cross-origin frame is created in the new/target renderer process. | ||||||
|  | /// The client will receive creation callbacks for the temporary frame, followed | ||||||
|  | /// by cross-origin navigation callbacks (2) for the transition from the | ||||||
|  | /// temporary frame to the real frame. The temporary frame will not receive or | ||||||
|  | /// execute commands during this transitional period (any sent commands will be | ||||||
|  | /// discarded). | ||||||
| /// | /// | ||||||
| /// When the main frame navigates to a different origin the OnMainFrameChanged | /// When the main frame navigates to a different origin the OnMainFrameChanged | ||||||
| /// callback (2) will be executed with the old and new main frame objects. | /// callback (2) will be executed with the old and new main frame objects. | ||||||
| /// | /// | ||||||
| /// When a new sub-frame is loaded in, or an existing sub-frame is navigated to, |  | ||||||
| /// a different origin from the parent frame, a temporary sub-frame object will |  | ||||||
| /// first be created in the parent's renderer process. That temporary sub-frame |  | ||||||
| /// will then be discarded after the real cross-origin sub-frame is created in |  | ||||||
| /// the new/target renderer process. The client will receive cross-origin |  | ||||||
| /// navigation callbacks (2) for the transition from the temporary sub-frame to |  | ||||||
| /// the real sub-frame. The temporary sub-frame will not receive or execute |  | ||||||
| /// commands during this transitional period (any sent commands will be |  | ||||||
| /// discarded). |  | ||||||
| /// |  | ||||||
| /// When a new popup browser is created in a different origin from the parent |  | ||||||
| /// browser, a temporary main frame object for the popup will first be created |  | ||||||
| /// in the parent's renderer process. That temporary main frame will then be |  | ||||||
| /// discarded after the real cross-origin main frame is created in the |  | ||||||
| /// new/target renderer process. The client will receive creation and initial |  | ||||||
| /// navigation callbacks (1) for the temporary main frame, followed by |  | ||||||
| /// cross-origin navigation callbacks (2) for the transition from the temporary |  | ||||||
| /// main frame to the real main frame. The temporary main frame may receive and |  | ||||||
| /// execute commands during this transitional period (any sent commands may be |  | ||||||
| /// executed, but the behavior is potentially undesirable since they execute in |  | ||||||
| /// the parent browser's renderer process and not the new/target renderer |  | ||||||
| /// process). |  | ||||||
| /// |  | ||||||
| /// Callbacks will not be executed for placeholders that may be created during | /// Callbacks will not be executed for placeholders that may be created during | ||||||
| /// pre-commit navigation for sub-frames that do not yet exist in the renderer | /// pre-commit navigation for sub-frames that do not yet exist in the renderer | ||||||
| /// process. Placeholders will have CefFrame::GetIdentifier() == -4. | /// process. Placeholders will have CefFrame::GetIdentifier() == -4. | ||||||
| @@ -126,17 +118,33 @@ class CefFrameHandler : public virtual CefBaseRefCounted { | |||||||
|   /// Called when a new frame is created. This will be the first notification |   /// Called when a new frame is created. This will be the first notification | ||||||
|   /// that references |frame|. Any commands that require transport to the |   /// that references |frame|. Any commands that require transport to the | ||||||
|   /// associated renderer process (LoadRequest, SendProcessMessage, GetSource, |   /// associated renderer process (LoadRequest, SendProcessMessage, GetSource, | ||||||
|   /// etc.) will be queued until OnFrameAttached is called for |frame|. |   /// etc.) will be queued. The queued commands will be sent before | ||||||
|  |   /// OnFrameAttached or discarded before OnFrameDestroyed if the frame never | ||||||
|  |   /// attaches. | ||||||
|   /// |   /// | ||||||
|   /*--cef()--*/ |   /*--cef()--*/ | ||||||
|   virtual void OnFrameCreated(CefRefPtr<CefBrowser> browser, |   virtual void OnFrameCreated(CefRefPtr<CefBrowser> browser, | ||||||
|                               CefRefPtr<CefFrame> frame) {} |                               CefRefPtr<CefFrame> frame) {} | ||||||
|  |  | ||||||
|  |   /// | ||||||
|  |   /// Called when an existing frame is destroyed. This will be the last | ||||||
|  |   /// notification that references |frame| and CefFrame::IsValid() will return | ||||||
|  |   /// false for |frame|. If called during browser destruction and after | ||||||
|  |   /// CefLifeSpanHandler::OnBeforeClose() then CefBrowser::IsValid() will return | ||||||
|  |   /// false for |browser|. Any queued commands that have not been sent will be | ||||||
|  |   /// discarded before this callback. | ||||||
|  |   /// | ||||||
|  |   /*--cef()--*/ | ||||||
|  |   virtual void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                                 CefRefPtr<CefFrame> frame) {} | ||||||
|  |  | ||||||
|   /// |   /// | ||||||
|   /// Called when a frame can begin routing commands to/from the associated |   /// Called when a frame can begin routing commands to/from the associated | ||||||
|   /// renderer process. |reattached| will be true if the frame was re-attached |   /// renderer process. |reattached| will be true if the frame was re-attached | ||||||
|   /// after exiting the BackForwardCache. Any commands that were queued have now |   /// after exiting the BackForwardCache or after encountering a recoverable | ||||||
|   /// been dispatched. |   /// connection error. Any queued commands will now have been dispatched. This | ||||||
|  |   /// method will not be called for temporary frames created during cross-origin | ||||||
|  |   /// navigation. | ||||||
|   /// |   /// | ||||||
|   /*--cef()--*/ |   /*--cef()--*/ | ||||||
|   virtual void OnFrameAttached(CefRefPtr<CefBrowser> browser, |   virtual void OnFrameAttached(CefRefPtr<CefBrowser> browser, | ||||||
| @@ -144,11 +152,17 @@ class CefFrameHandler : public virtual CefBaseRefCounted { | |||||||
|                                bool reattached) {} |                                bool reattached) {} | ||||||
|  |  | ||||||
|   /// |   /// | ||||||
|   /// Called when a frame loses its connection to the renderer process and will |   /// Called when a frame loses its connection to the renderer process. This may | ||||||
|   /// be destroyed. Any pending or future commands will be discarded and |   /// occur when a frame is destroyed, enters the BackForwardCache, or | ||||||
|   /// CefFrame::IsValid() will now return false for |frame|. If called after |   /// encounters a rare connection error. In the case of frame destruction this | ||||||
|   /// CefLifeSpanHandler::OnBeforeClose() during browser destruction then |   /// call will be followed by a (potentially async) call to OnFrameDestroyed. | ||||||
|   /// CefBrowser::IsValid() will return false for |browser|. |   /// If frame destruction is occuring synchronously then CefFrame::IsValid() | ||||||
|  |   /// will return false for |frame|. If called during browser destruction and | ||||||
|  |   /// after CefLifeSpanHandler::OnBeforeClose() then CefBrowser::IsValid() will | ||||||
|  |   /// return false for |browser|. If, in the non-destruction case, the same | ||||||
|  |   /// frame later exits the BackForwardCache or recovers from a connection error | ||||||
|  |   /// then there will be a follow-up call to OnFrameAttached. This method will | ||||||
|  |   /// not be called for temporary frames created during cross-origin navigation. | ||||||
|   /// |   /// | ||||||
|   /*--cef()--*/ |   /*--cef()--*/ | ||||||
|   virtual void OnFrameDetached(CefRefPtr<CefBrowser> browser, |   virtual void OnFrameDetached(CefRefPtr<CefBrowser> browser, | ||||||
| @@ -160,13 +174,13 @@ class CefFrameHandler : public virtual CefBaseRefCounted { | |||||||
|   /// re-navigation after renderer process termination (due to crashes, etc). |   /// re-navigation after renderer process termination (due to crashes, etc). | ||||||
|   /// |old_frame| will be NULL and |new_frame| will be non-NULL when a main |   /// |old_frame| will be NULL and |new_frame| will be non-NULL when a main | ||||||
|   /// frame is assigned to |browser| for the first time. |old_frame| will be |   /// frame is assigned to |browser| for the first time. |old_frame| will be | ||||||
|   /// non-NULL and |new_frame| will be NULL and  when a main frame is removed |   /// non-NULL and |new_frame| will be NULL when a main frame is removed from | ||||||
|   /// from |browser| for the last time. Both |old_frame| and |new_frame| will be |   /// |browser| for the last time. Both |old_frame| and |new_frame| will be | ||||||
|   /// non-NULL for cross-origin navigations or re-navigation after renderer |   /// non-NULL for cross-origin navigations or re-navigation after renderer | ||||||
|   /// process termination. This method will be called after OnFrameCreated() for |   /// process termination. This method will be called after OnFrameCreated() for | ||||||
|   /// |new_frame| and/or after OnFrameDetached() for |old_frame|. If called |   /// |new_frame| and/or after OnFrameDestroyed() for |old_frame|. If called | ||||||
|   /// after CefLifeSpanHandler::OnBeforeClose() during browser destruction then |   /// during browser destruction and after CefLifeSpanHandler::OnBeforeClose() | ||||||
|   /// CefBrowser::IsValid() will return false for |browser|. |   /// then CefBrowser::IsValid() will return false for |browser|. | ||||||
|   /// |   /// | ||||||
|   /*--cef(optional_param=old_frame,optional_param=new_frame)--*/ |   /*--cef(optional_param=old_frame,optional_param=new_frame)--*/ | ||||||
|   virtual void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, |   virtual void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, | ||||||
|   | |||||||
| @@ -131,6 +131,9 @@ void CefBrowserInfo::MaybeCreateFrame(content::RenderFrameHost* host) { | |||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   DVLOG(1) << __func__ << ": " | ||||||
|  |            << frame_util::GetFrameDebugString(host->GetGlobalFrameToken()); | ||||||
|  |  | ||||||
|   const auto global_id = host->GetGlobalId(); |   const auto global_id = host->GetGlobalId(); | ||||||
|   const bool is_main_frame = (host->GetParent() == nullptr); |   const bool is_main_frame = (host->GetParent() == nullptr); | ||||||
|  |  | ||||||
| @@ -160,7 +163,7 @@ void CefBrowserInfo::MaybeCreateFrame(content::RenderFrameHost* host) { | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
|     // Update the associated RFH, which may have changed. |     // Update the associated RFH, which may have changed. | ||||||
|     info->frame_->MaybeReAttach(this, host, /*require_detached=*/false); |     info->frame_->MaybeAttach(this, host); | ||||||
|  |  | ||||||
|     if (info->is_speculative_ && !is_speculative) { |     if (info->is_speculative_ && !is_speculative) { | ||||||
|       // Upgrade the frame info from speculative to non-speculative. |       // Upgrade the frame info from speculative to non-speculative. | ||||||
| @@ -208,17 +211,20 @@ void CefBrowserInfo::FrameHostStateChanged( | |||||||
|     content::RenderFrameHost::LifecycleState new_state) { |     content::RenderFrameHost::LifecycleState new_state) { | ||||||
|   CEF_REQUIRE_UIT(); |   CEF_REQUIRE_UIT(); | ||||||
|  |  | ||||||
|  |   DVLOG(1) << __func__ << ": " | ||||||
|  |            << frame_util::GetFrameDebugString(host->GetGlobalFrameToken()); | ||||||
|  |  | ||||||
|   if ((old_state == content::RenderFrameHost::LifecycleState::kPrerendering || |   if ((old_state == content::RenderFrameHost::LifecycleState::kPrerendering || | ||||||
|        old_state == |        old_state == | ||||||
|            content::RenderFrameHost::LifecycleState::kInBackForwardCache) && |            content::RenderFrameHost::LifecycleState::kInBackForwardCache) && | ||||||
|       new_state == content::RenderFrameHost::LifecycleState::kActive) { |       new_state == content::RenderFrameHost::LifecycleState::kActive) { | ||||||
|     if (auto frame = GetFrameForHost(host)) { |     if (auto frame = GetFrameForHost(host)) { | ||||||
|       // Update the associated RFH, which may have changed. |       // Update the associated RFH, which may have changed. | ||||||
|       frame->MaybeReAttach(this, host, /*require_detached=*/true); |       frame->MaybeAttach(this, host); | ||||||
|  |  | ||||||
|       if (frame->IsMain()) { |       if (frame->IsMain()) { | ||||||
|         // Update the main frame object. |  | ||||||
|         NotificationStateLock lock_scope(this); |         NotificationStateLock lock_scope(this); | ||||||
|  |         // Update the main frame object. | ||||||
|         SetMainFrame(browser_, frame); |         SetMainFrame(browser_, frame); | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -251,6 +257,9 @@ void CefBrowserInfo::FrameHostStateChanged( | |||||||
| void CefBrowserInfo::RemoveFrame(content::RenderFrameHost* host) { | void CefBrowserInfo::RemoveFrame(content::RenderFrameHost* host) { | ||||||
|   CEF_REQUIRE_UIT(); |   CEF_REQUIRE_UIT(); | ||||||
|  |  | ||||||
|  |   DVLOG(1) << __func__ << ": " | ||||||
|  |            << frame_util::GetFrameDebugString(host->GetGlobalFrameToken()); | ||||||
|  |  | ||||||
|   NotificationStateLock lock_scope(this); |   NotificationStateLock lock_scope(this); | ||||||
|  |  | ||||||
|   const auto global_id = host->GetGlobalId(); |   const auto global_id = host->GetGlobalId(); | ||||||
| @@ -281,12 +290,17 @@ void CefBrowserInfo::RemoveFrame(content::RenderFrameHost* host) { | |||||||
|     const auto& other_frame_info = *it2; |     const auto& other_frame_info = *it2; | ||||||
|     if (other_frame_info->frame_) { |     if (other_frame_info->frame_) { | ||||||
|       const bool is_current_main_frame = other_frame_info->IsCurrentMainFrame(); |       const bool is_current_main_frame = other_frame_info->IsCurrentMainFrame(); | ||||||
|       if (other_frame_info->frame_->Detach( |       const auto [frame_detached, frame_destroyed] = | ||||||
|  |           other_frame_info->frame_->Detach( | ||||||
|               CefFrameHostImpl::DetachReason::RENDER_FRAME_DELETED, |               CefFrameHostImpl::DetachReason::RENDER_FRAME_DELETED, | ||||||
|               is_current_main_frame)) { |               is_current_main_frame); | ||||||
|         DCHECK(!is_current_main_frame); |       if (frame_detached) { | ||||||
|         MaybeNotifyFrameDetached(browser_, other_frame_info->frame_); |         MaybeNotifyFrameDetached(browser_, other_frame_info->frame_); | ||||||
|       } |       } | ||||||
|  |       if (frame_destroyed) { | ||||||
|  |         DCHECK(!is_current_main_frame); | ||||||
|  |         MaybeNotifyFrameDestroyed(browser_, other_frame_info->frame_); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     frame_info_set_.erase(it2); |     frame_info_set_.erase(it2); | ||||||
| @@ -477,13 +491,21 @@ void CefBrowserInfo::SetMainFrame(CefRefPtr<CefBrowserHostBase> browser, | |||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   DVLOG(1) << __func__ << ": " | ||||||
|  |            << (frame ? frame->GetIdentifier().ToString() : "null"); | ||||||
|  |  | ||||||
|   CefRefPtr<CefFrameHostImpl> old_frame; |   CefRefPtr<CefFrameHostImpl> old_frame; | ||||||
|   if (main_frame_) { |   if (main_frame_) { | ||||||
|     old_frame = main_frame_; |     old_frame = main_frame_; | ||||||
|     if (old_frame->Detach(CefFrameHostImpl::DetachReason::NEW_MAIN_FRAME, |     const auto [frame_detached, frame_destroyed] = | ||||||
|                           /*is_current_main_frame=*/false)) { |         old_frame->Detach(CefFrameHostImpl::DetachReason::NEW_MAIN_FRAME, | ||||||
|  |                           /*is_current_main_frame=*/false); | ||||||
|  |     if (frame_detached) { | ||||||
|       MaybeNotifyFrameDetached(browser, old_frame); |       MaybeNotifyFrameDetached(browser, old_frame); | ||||||
|     } |     } | ||||||
|  |     if (frame_destroyed) { | ||||||
|  |       MaybeNotifyFrameDestroyed(browser, old_frame); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   main_frame_ = frame; |   main_frame_ = frame; | ||||||
| @@ -526,6 +548,24 @@ void CefBrowserInfo::MaybeNotifyFrameDetached( | |||||||
|       browser, frame)); |       browser, frame)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Passing in |browser| here because |browser_| may already be cleared. | ||||||
|  | void CefBrowserInfo::MaybeNotifyFrameDestroyed( | ||||||
|  |     CefRefPtr<CefBrowserHostBase> browser, | ||||||
|  |     CefRefPtr<CefFrameHostImpl> frame) { | ||||||
|  |   CEF_REQUIRE_UIT(); | ||||||
|  |  | ||||||
|  |   // Never notify for temporary objects. | ||||||
|  |   DCHECK(!frame->is_temporary()); | ||||||
|  |  | ||||||
|  |   MaybeExecuteFrameNotification(base::BindOnce( | ||||||
|  |       [](CefRefPtr<CefBrowserHostBase> browser, | ||||||
|  |          CefRefPtr<CefFrameHostImpl> frame, | ||||||
|  |          CefRefPtr<CefFrameHandler> handler) { | ||||||
|  |         handler->OnFrameDestroyed(browser, frame); | ||||||
|  |       }, | ||||||
|  |       browser, frame)); | ||||||
|  | } | ||||||
|  |  | ||||||
| // Passing in |browser| here because |browser_| may already be cleared. | // Passing in |browser| here because |browser_| may already be cleared. | ||||||
| void CefBrowserInfo::MaybeNotifyMainFrameChanged( | void CefBrowserInfo::MaybeNotifyMainFrameChanged( | ||||||
|     CefRefPtr<CefBrowserHostBase> browser, |     CefRefPtr<CefBrowserHostBase> browser, | ||||||
| @@ -563,13 +603,13 @@ void CefBrowserInfo::RemoveAllFrames( | |||||||
|   // Explicitly Detach everything. |   // Explicitly Detach everything. | ||||||
|   for (auto& info : frame_info_set_) { |   for (auto& info : frame_info_set_) { | ||||||
|     if (info->frame_) { |     if (info->frame_) { | ||||||
|       const bool is_current_main_frame = info->IsCurrentMainFrame(); |       [[maybe_unused]] const auto [frame_detached, frame_destroyed] = | ||||||
|       if (info->frame_->Detach( |           info->frame_->Detach( | ||||||
|               CefFrameHostImpl::DetachReason::BROWSER_DESTROYED, |               CefFrameHostImpl::DetachReason::BROWSER_DESTROYED, | ||||||
|               is_current_main_frame)) { |               info->IsCurrentMainFrame()); | ||||||
|         DCHECK(!is_current_main_frame); |       // Shouldn't need to trigger any notifications at this point. | ||||||
|         MaybeNotifyFrameDetached(old_browser, info->frame_); |       DCHECK(!frame_detached); | ||||||
|       } |       DCHECK(!frame_destroyed); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -166,6 +166,8 @@ class CefBrowserInfo : public base::RefCountedThreadSafe<CefBrowserInfo> { | |||||||
|       CefRefPtr<CefBrowserHostBase> browser, |       CefRefPtr<CefBrowserHostBase> browser, | ||||||
|       CefRefPtr<CefFrameHostImpl> frame, |       CefRefPtr<CefFrameHostImpl> frame, | ||||||
|       std::vector<CefDraggableRegion> draggable_regions); |       std::vector<CefDraggableRegion> draggable_regions); | ||||||
|  |   void MaybeNotifyFrameDetached(CefRefPtr<CefBrowserHostBase> browser, | ||||||
|  |                                 CefRefPtr<CefFrameHostImpl> frame); | ||||||
|  |  | ||||||
|  private: |  private: | ||||||
|   friend class base::RefCountedThreadSafe<CefBrowserInfo>; |   friend class base::RefCountedThreadSafe<CefBrowserInfo>; | ||||||
| @@ -190,8 +192,8 @@ class CefBrowserInfo : public base::RefCountedThreadSafe<CefBrowserInfo> { | |||||||
|                     CefRefPtr<CefFrameHostImpl> frame); |                     CefRefPtr<CefFrameHostImpl> frame); | ||||||
|  |  | ||||||
|   void MaybeNotifyFrameCreated(CefRefPtr<CefFrameHostImpl> frame); |   void MaybeNotifyFrameCreated(CefRefPtr<CefFrameHostImpl> frame); | ||||||
|   void MaybeNotifyFrameDetached(CefRefPtr<CefBrowserHostBase> browser, |   void MaybeNotifyFrameDestroyed(CefRefPtr<CefBrowserHostBase> browser, | ||||||
|                                 CefRefPtr<CefFrameHostImpl> frame); |                                  CefRefPtr<CefFrameHostImpl> frame); | ||||||
|   void MaybeNotifyMainFrameChanged(CefRefPtr<CefBrowserHostBase> browser, |   void MaybeNotifyMainFrameChanged(CefRefPtr<CefBrowserHostBase> browser, | ||||||
|                                    CefRefPtr<CefFrameHostImpl> old_frame, |                                    CefRefPtr<CefFrameHostImpl> old_frame, | ||||||
|                                    CefRefPtr<CefFrameHostImpl> new_frame); |                                    CefRefPtr<CefFrameHostImpl> new_frame); | ||||||
|   | |||||||
| @@ -500,35 +500,18 @@ bool CefFrameHostImpl::IsDetached() const { | |||||||
|   return !GetRenderFrameHost(); |   return !GetRenderFrameHost(); | ||||||
| } | } | ||||||
|  |  | ||||||
| bool CefFrameHostImpl::Detach(DetachReason reason, bool is_current_main_frame) { | std::pair<bool, bool> CefFrameHostImpl::Detach(DetachReason reason, | ||||||
|  |                                                bool is_current_main_frame) { | ||||||
|   CEF_REQUIRE_UIT(); |   CEF_REQUIRE_UIT(); | ||||||
|  |  | ||||||
|   // This method may be called multiple times (e.g. from CefBrowserInfo |  | ||||||
|   // SetMainFrame and RemoveFrame). |  | ||||||
|   bool is_first_complete_detach = false; |  | ||||||
|  |  | ||||||
|   // Should not be called for temporary frames. |   // Should not be called for temporary frames. | ||||||
|   CHECK(!is_temporary()); |   CHECK(!is_temporary()); | ||||||
|  |  | ||||||
|   // Must be a main frame if |is_current_main_frame| is true. |   // Must be a main frame if |is_current_main_frame| is true. | ||||||
|   CHECK(!is_current_main_frame || is_main_frame_); |   CHECK(!is_current_main_frame || is_main_frame_); | ||||||
|  |  | ||||||
|   if (!is_current_main_frame) { |   const bool is_bound = render_frame_.is_bound(); | ||||||
|     { |   if (is_bound) { | ||||||
|       base::AutoLock lock_scope(state_lock_); |  | ||||||
|       if (browser_info_) { |  | ||||||
|         is_first_complete_detach = true; |  | ||||||
|         browser_info_ = nullptr; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // In case we never attached, clean up. |  | ||||||
|     while (!queued_renderer_actions_.empty()) { |  | ||||||
|       queued_renderer_actions_.pop(); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (render_frame_.is_bound()) { |  | ||||||
|     if (VLOG_IS_ON(1)) { |     if (VLOG_IS_ON(1)) { | ||||||
|       std::string reason_str; |       std::string reason_str; | ||||||
|       switch (reason) { |       switch (reason) { | ||||||
| @@ -554,7 +537,27 @@ bool CefFrameHostImpl::Detach(DetachReason reason, bool is_current_main_frame) { | |||||||
|     render_frame_host_ = nullptr; |     render_frame_host_ = nullptr; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return is_first_complete_detach; |   // This method may be called multiple times (e.g. from CefBrowserInfo | ||||||
|  |   // SetMainFrame and RemoveFrame). | ||||||
|  |   bool is_first_complete_detach = false; | ||||||
|  |  | ||||||
|  |   if (!is_current_main_frame) { | ||||||
|  |     { | ||||||
|  |       base::AutoLock lock_scope(state_lock_); | ||||||
|  |       if (browser_info_) { | ||||||
|  |         DVLOG(1) << __func__ << ": " << GetDebugString() << " invalidated"; | ||||||
|  |         is_first_complete_detach = true; | ||||||
|  |         browser_info_ = nullptr; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // In case we never attached, clean up. | ||||||
|  |     while (!queued_renderer_actions_.empty()) { | ||||||
|  |       queued_renderer_actions_.pop(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return std::make_pair(is_bound, is_first_complete_detach); | ||||||
| } | } | ||||||
|  |  | ||||||
| void CefFrameHostImpl::DetachRenderFrame() { | void CefFrameHostImpl::DetachRenderFrame() { | ||||||
| @@ -564,12 +567,11 @@ void CefFrameHostImpl::DetachRenderFrame() { | |||||||
|       static_cast<uint32_t>(frame_util::ResetReason::kDetached), "Detached"); |       static_cast<uint32_t>(frame_util::ResetReason::kDetached), "Detached"); | ||||||
| } | } | ||||||
|  |  | ||||||
| void CefFrameHostImpl::MaybeReAttach( | void CefFrameHostImpl::MaybeAttach( | ||||||
|     scoped_refptr<CefBrowserInfo> browser_info, |     scoped_refptr<CefBrowserInfo> browser_info, | ||||||
|     content::RenderFrameHost* render_frame_host, |     content::RenderFrameHost* render_frame_host) { | ||||||
|     bool require_detached) { |  | ||||||
|   CEF_REQUIRE_UIT(); |   CEF_REQUIRE_UIT(); | ||||||
|   if (render_frame_.is_bound() && render_frame_host_ == render_frame_host) { |   if (render_frame_host_ == render_frame_host) { | ||||||
|     // Nothing to do here. |     // Nothing to do here. | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
| @@ -577,16 +579,13 @@ void CefFrameHostImpl::MaybeReAttach( | |||||||
|   // Should not be called for temporary frames. |   // Should not be called for temporary frames. | ||||||
|   CHECK(!is_temporary()); |   CHECK(!is_temporary()); | ||||||
|  |  | ||||||
|   // If |require_detached| then we expect that Detach() was called previously. |   // We expect that either this frame has never attached (e.g. when swapping | ||||||
|   CHECK(!require_detached || !render_frame_.is_bound()); |   // from speculative to non-speculative) or Detach() was called previously | ||||||
|  |   // (e.g. when exiting the bfcache). | ||||||
|  |   CHECK(!render_frame_.is_bound()); | ||||||
|  |  | ||||||
|   if (render_frame_.is_bound()) { |   // Intentionally not clearing |queued_renderer_actions_|, as we may be | ||||||
|     // Intentionally not clearing |queued_renderer_actions_|, as we may be |   // changing RFH during initial browser navigation. | ||||||
|     // changing RFH during initial browser navigation. |  | ||||||
|     DVLOG(1) << __func__ << ": " << GetDebugString() |  | ||||||
|              << " detached (reason=RENDER_FRAME_CHANGED)"; |  | ||||||
|     DetachRenderFrame(); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // The RFH may change but the frame token should remain the same. |   // The RFH may change but the frame token should remain the same. | ||||||
|   CHECK(*frame_token_ == render_frame_host->GetGlobalFrameToken()); |   CHECK(*frame_token_ == render_frame_host->GetGlobalFrameToken()); | ||||||
| @@ -653,6 +652,14 @@ void CefFrameHostImpl::SendToRenderFrame(const std::string& function_name, | |||||||
| void CefFrameHostImpl::OnRenderFrameDisconnect() { | void CefFrameHostImpl::OnRenderFrameDisconnect() { | ||||||
|   CEF_REQUIRE_UIT(); |   CEF_REQUIRE_UIT(); | ||||||
|  |  | ||||||
|  |   DVLOG(1) << __func__ << ": " << GetDebugString(); | ||||||
|  |  | ||||||
|  |   if (auto browser_info = GetBrowserInfo()) { | ||||||
|  |     if (auto browser = browser_info->browser()) { | ||||||
|  |       browser_info->MaybeNotifyFrameDetached(browser, this); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // Reconnect, if any, will be triggered via FrameAttached(). |   // Reconnect, if any, will be triggered via FrameAttached(). | ||||||
|   render_frame_.reset(); |   render_frame_.reset(); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ | |||||||
| #include <optional> | #include <optional> | ||||||
| #include <queue> | #include <queue> | ||||||
| #include <string> | #include <string> | ||||||
|  | #include <utility> | ||||||
|  |  | ||||||
| #include "base/memory/raw_ptr.h" | #include "base/memory/raw_ptr.h" | ||||||
| #include "base/synchronization/lock.h" | #include "base/synchronization/lock.h" | ||||||
| @@ -140,15 +141,15 @@ class CefFrameHostImpl : public CefFrame, public cef::mojom::BrowserFrame { | |||||||
|   // implicitly via CefBrowserInfo::browser() returning nullptr. If |   // implicitly via CefBrowserInfo::browser() returning nullptr. If | ||||||
|   // |is_current_main_frame| is true then only the RenderFrameHost references |   // |is_current_main_frame| is true then only the RenderFrameHost references | ||||||
|   // will be released as we want the frame object itself to remain valid. |   // will be released as we want the frame object itself to remain valid. | ||||||
|   // Returns true if the frame is completely detached for the first time. |   // Returns (bool, bool) to indicate if frame detached and/or frame destroyed | ||||||
|   bool Detach(DetachReason reason, bool is_current_main_frame); |   // notifications should be triggered respectively. | ||||||
|  |   std::pair<bool, bool> Detach(DetachReason reason, bool is_current_main_frame); | ||||||
|  |  | ||||||
|   // A frame has swapped to active status from prerendering or the back-forward |   // A new frame was created or a frame has swapped to active status from | ||||||
|   // cache. We may need to re-attach if the RFH has changed. See |   // prerendering or the back-forward cache. Update internal state if the RFH | ||||||
|   // https://crbug.com/1179502#c8 for additional background. |   // has changed. See https://crbug.com/1179502#c8 for additional background. | ||||||
|   void MaybeReAttach(scoped_refptr<CefBrowserInfo> browser_info, |   void MaybeAttach(scoped_refptr<CefBrowserInfo> browser_info, | ||||||
|                      content::RenderFrameHost* render_frame_host, |                    content::RenderFrameHost* render_frame_host); | ||||||
|                      bool require_detached); |  | ||||||
|  |  | ||||||
|   // cef::mojom::BrowserFrame methods forwarded from CefBrowserFrame. |   // cef::mojom::BrowserFrame methods forwarded from CefBrowserFrame. | ||||||
|   void SendMessage(const std::string& name, |   void SendMessage(const std::string& name, | ||||||
|   | |||||||
| @@ -69,7 +69,10 @@ class CefFrameServiceBase : public Interface, | |||||||
|     DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); |     DCHECK_CALLED_ON_VALID_THREAD(thread_checker_); | ||||||
|  |  | ||||||
|     if (render_frame_host == render_frame_host_) { |     if (render_frame_host == render_frame_host_) { | ||||||
|       DVLOG(1) << __func__ << ": RenderFrameHost destroyed."; |       DVLOG(1) << __func__ << ": " | ||||||
|  |                << frame_util::GetFrameDebugString( | ||||||
|  |                       render_frame_host->GetGlobalFrameToken()) | ||||||
|  |                << " destroyed"; | ||||||
|       if (receiver_.is_bound()) { |       if (receiver_.is_bound()) { | ||||||
|         receiver_.ResetWithReason( |         receiver_.ResetWithReason( | ||||||
|             static_cast<uint32_t>(frame_util::ResetReason::kDeleted), |             static_cast<uint32_t>(frame_util::ResetReason::kDeleted), | ||||||
|   | |||||||
| @@ -318,20 +318,23 @@ void CefFrameImpl::SendProcessMessage(CefProcessId target_process, | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void CefFrameImpl::OnAttached() { |  | ||||||
|   // Called indirectly from RenderFrameCreated. |  | ||||||
|   ConnectBrowserFrame(ConnectReason::RENDER_FRAME_CREATED); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void CefFrameImpl::OnWasShown() { | void CefFrameImpl::OnWasShown() { | ||||||
|   if (browser_connection_state_ == ConnectionState::DISCONNECTED) { |   if (browser_connection_state_ == ConnectionState::DISCONNECTED && | ||||||
|     // Reconnect a frame that has exited the bfcache. |       did_commit_provisional_load_) { | ||||||
|  |     // Reconnect a frame that has exited the bfcache. We ignore temporary | ||||||
|  |     // frames that have never called DidCommitProvisionalLoad. | ||||||
|     ConnectBrowserFrame(ConnectReason::WAS_SHOWN); |     ConnectBrowserFrame(ConnectReason::WAS_SHOWN); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void CefFrameImpl::OnDidCommitProvisionalLoad() { | void CefFrameImpl::OnDidCommitProvisionalLoad() { | ||||||
|   did_commit_provisional_load_ = true; |   did_commit_provisional_load_ = true; | ||||||
|  |   if (browser_connection_state_ == ConnectionState::DISCONNECTED) { | ||||||
|  |     // Connect after RenderFrameImpl::DidCommitNavigation has potentially | ||||||
|  |     // reset the BrowserInterfaceBroker in the browser process. See related | ||||||
|  |     // comments in OnDisconnect. | ||||||
|  |     ConnectBrowserFrame(ConnectReason::DID_COMMIT); | ||||||
|  |   } | ||||||
|   MaybeInitializeScriptContext(); |   MaybeInitializeScriptContext(); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -475,8 +478,8 @@ void CefFrameImpl::ConnectBrowserFrame(ConnectReason reason) { | |||||||
|   if (VLOG_IS_ON(1)) { |   if (VLOG_IS_ON(1)) { | ||||||
|     std::string reason_str; |     std::string reason_str; | ||||||
|     switch (reason) { |     switch (reason) { | ||||||
|       case ConnectReason::RENDER_FRAME_CREATED: |       case ConnectReason::DID_COMMIT: | ||||||
|         reason_str = "RENDER_FRAME_CREATED"; |         reason_str = "DID_COMMIT"; | ||||||
|         break; |         break; | ||||||
|       case ConnectReason::WAS_SHOWN: |       case ConnectReason::WAS_SHOWN: | ||||||
|         reason_str = "WAS_SHOWN"; |         reason_str = "WAS_SHOWN"; | ||||||
|   | |||||||
| @@ -81,7 +81,6 @@ class CefFrameImpl | |||||||
|                           CefRefPtr<CefProcessMessage> message) override; |                           CefRefPtr<CefProcessMessage> message) override; | ||||||
|  |  | ||||||
|   // Forwarded from CefRenderFrameObserver. |   // Forwarded from CefRenderFrameObserver. | ||||||
|   void OnAttached(); |  | ||||||
|   void OnWasShown(); |   void OnWasShown(); | ||||||
|   void OnDidCommitProvisionalLoad(); |   void OnDidCommitProvisionalLoad(); | ||||||
|   void OnDidFinishLoad(); |   void OnDidFinishLoad(); | ||||||
| @@ -105,7 +104,7 @@ class CefFrameImpl | |||||||
|                            LocalFrameAction action); |                            LocalFrameAction action); | ||||||
|  |  | ||||||
|   enum class ConnectReason { |   enum class ConnectReason { | ||||||
|     RENDER_FRAME_CREATED, |     DID_COMMIT, | ||||||
|     WAS_SHOWN, |     WAS_SHOWN, | ||||||
|     RETRY, |     RETRY, | ||||||
|   }; |   }; | ||||||
|   | |||||||
| @@ -199,7 +199,6 @@ void CefRenderFrameObserver::AttachFrame(CefFrameImpl* frame) { | |||||||
|   DCHECK(frame); |   DCHECK(frame); | ||||||
|   DCHECK(!frame_); |   DCHECK(!frame_); | ||||||
|   frame_ = frame; |   frame_ = frame; | ||||||
|   frame_->OnAttached(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void CefRenderFrameObserver::OnLoadStart() { | void CefRenderFrameObserver::OnLoadStart() { | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
| // implementations. See the translator.README.txt file in the tools directory | // implementations. See the translator.README.txt file in the tools directory | ||||||
| // for more information. | // for more information. | ||||||
| // | // | ||||||
| // $hash=4a3d33abbaa00a373ea515338ed67d96708dbb9c$ | // $hash=51da21d569dd41e38cb2dc6e0f2dea0bd88dbdce$ | ||||||
| // | // | ||||||
|  |  | ||||||
| #include "libcef_dll/cpptoc/frame_handler_cpptoc.h" | #include "libcef_dll/cpptoc/frame_handler_cpptoc.h" | ||||||
| @@ -50,6 +50,34 @@ frame_handler_on_frame_created(struct _cef_frame_handler_t* self, | |||||||
|       CefBrowserCToCpp::Wrap(browser), CefFrameCToCpp::Wrap(frame)); |       CefBrowserCToCpp::Wrap(browser), CefFrameCToCpp::Wrap(frame)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void CEF_CALLBACK | ||||||
|  | frame_handler_on_frame_destroyed(struct _cef_frame_handler_t* self, | ||||||
|  |                                  cef_browser_t* browser, | ||||||
|  |                                  cef_frame_t* frame) { | ||||||
|  |   shutdown_checker::AssertNotShutdown(); | ||||||
|  |  | ||||||
|  |   // AUTO-GENERATED CONTENT - DELETE THIS COMMENT BEFORE MODIFYING | ||||||
|  |  | ||||||
|  |   DCHECK(self); | ||||||
|  |   if (!self) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   // Verify param: browser; type: refptr_diff | ||||||
|  |   DCHECK(browser); | ||||||
|  |   if (!browser) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   // Verify param: frame; type: refptr_diff | ||||||
|  |   DCHECK(frame); | ||||||
|  |   if (!frame) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Execute | ||||||
|  |   CefFrameHandlerCppToC::Get(self)->OnFrameDestroyed( | ||||||
|  |       CefBrowserCToCpp::Wrap(browser), CefFrameCToCpp::Wrap(frame)); | ||||||
|  | } | ||||||
|  |  | ||||||
| void CEF_CALLBACK | void CEF_CALLBACK | ||||||
| frame_handler_on_frame_attached(struct _cef_frame_handler_t* self, | frame_handler_on_frame_attached(struct _cef_frame_handler_t* self, | ||||||
|                                 cef_browser_t* browser, |                                 cef_browser_t* browser, | ||||||
| @@ -140,6 +168,7 @@ frame_handler_on_main_frame_changed(struct _cef_frame_handler_t* self, | |||||||
|  |  | ||||||
| CefFrameHandlerCppToC::CefFrameHandlerCppToC() { | CefFrameHandlerCppToC::CefFrameHandlerCppToC() { | ||||||
|   GetStruct()->on_frame_created = frame_handler_on_frame_created; |   GetStruct()->on_frame_created = frame_handler_on_frame_created; | ||||||
|  |   GetStruct()->on_frame_destroyed = frame_handler_on_frame_destroyed; | ||||||
|   GetStruct()->on_frame_attached = frame_handler_on_frame_attached; |   GetStruct()->on_frame_attached = frame_handler_on_frame_attached; | ||||||
|   GetStruct()->on_frame_detached = frame_handler_on_frame_detached; |   GetStruct()->on_frame_detached = frame_handler_on_frame_detached; | ||||||
|   GetStruct()->on_main_frame_changed = frame_handler_on_main_frame_changed; |   GetStruct()->on_main_frame_changed = frame_handler_on_main_frame_changed; | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
| // implementations. See the translator.README.txt file in the tools directory | // implementations. See the translator.README.txt file in the tools directory | ||||||
| // for more information. | // for more information. | ||||||
| // | // | ||||||
| // $hash=0074492ed580ccc06962a05b6c72bdabae182a51$ | // $hash=14e4a39489488582d7965ae71ed1ef174a4f3b08$ | ||||||
| // | // | ||||||
|  |  | ||||||
| #include "libcef_dll/ctocpp/frame_handler_ctocpp.h" | #include "libcef_dll/ctocpp/frame_handler_ctocpp.h" | ||||||
| @@ -48,6 +48,34 @@ void CefFrameHandlerCToCpp::OnFrameCreated(CefRefPtr<CefBrowser> browser, | |||||||
|                             CefFrameCppToC::Wrap(frame)); |                             CefFrameCppToC::Wrap(frame)); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | NO_SANITIZE("cfi-icall") | ||||||
|  | void CefFrameHandlerCToCpp::OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                                              CefRefPtr<CefFrame> frame) { | ||||||
|  |   shutdown_checker::AssertNotShutdown(); | ||||||
|  |  | ||||||
|  |   cef_frame_handler_t* _struct = GetStruct(); | ||||||
|  |   if (CEF_MEMBER_MISSING(_struct, on_frame_destroyed)) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // AUTO-GENERATED CONTENT - DELETE THIS COMMENT BEFORE MODIFYING | ||||||
|  |  | ||||||
|  |   // Verify param: browser; type: refptr_diff | ||||||
|  |   DCHECK(browser.get()); | ||||||
|  |   if (!browser.get()) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   // Verify param: frame; type: refptr_diff | ||||||
|  |   DCHECK(frame.get()); | ||||||
|  |   if (!frame.get()) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Execute | ||||||
|  |   _struct->on_frame_destroyed(_struct, CefBrowserCppToC::Wrap(browser), | ||||||
|  |                               CefFrameCppToC::Wrap(frame)); | ||||||
|  | } | ||||||
|  |  | ||||||
| NO_SANITIZE("cfi-icall") | NO_SANITIZE("cfi-icall") | ||||||
| void CefFrameHandlerCToCpp::OnFrameAttached(CefRefPtr<CefBrowser> browser, | void CefFrameHandlerCToCpp::OnFrameAttached(CefRefPtr<CefBrowser> browser, | ||||||
|                                             CefRefPtr<CefFrame> frame, |                                             CefRefPtr<CefFrame> frame, | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
| // implementations. See the translator.README.txt file in the tools directory | // implementations. See the translator.README.txt file in the tools directory | ||||||
| // for more information. | // for more information. | ||||||
| // | // | ||||||
| // $hash=a571fa8b3c173d78cfb67eb3e44c8f2c3fb2e089$ | // $hash=f33fedc6d7e0d692b03fe7f35319e93c5f31b9b1$ | ||||||
| // | // | ||||||
|  |  | ||||||
| #ifndef CEF_LIBCEF_DLL_CTOCPP_FRAME_HANDLER_CTOCPP_H_ | #ifndef CEF_LIBCEF_DLL_CTOCPP_FRAME_HANDLER_CTOCPP_H_ | ||||||
| @@ -36,6 +36,8 @@ class CefFrameHandlerCToCpp : public CefCToCppRefCounted<CefFrameHandlerCToCpp, | |||||||
|   // CefFrameHandler methods. |   // CefFrameHandler methods. | ||||||
|   void OnFrameCreated(CefRefPtr<CefBrowser> browser, |   void OnFrameCreated(CefRefPtr<CefBrowser> browser, | ||||||
|                       CefRefPtr<CefFrame> frame) override; |                       CefRefPtr<CefFrame> frame) override; | ||||||
|  |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                         CefRefPtr<CefFrame> frame) override; | ||||||
|   void OnFrameAttached(CefRefPtr<CefBrowser> browser, |   void OnFrameAttached(CefRefPtr<CefBrowser> browser, | ||||||
|                        CefRefPtr<CefFrame> frame, |                        CefRefPtr<CefFrame> frame, | ||||||
|                        bool reattached) override; |                        bool reattached) override; | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
| #include <map> | #include <map> | ||||||
| #include <memory> | #include <memory> | ||||||
| #include <queue> | #include <queue> | ||||||
|  | #include <set> | ||||||
| #include <sstream> | #include <sstream> | ||||||
| #include <string> | #include <string> | ||||||
|  |  | ||||||
| @@ -16,7 +17,8 @@ | |||||||
| #include "tests/ceftests/test_util.h" | #include "tests/ceftests/test_util.h" | ||||||
| #include "tests/gtest/include/gtest/gtest.h" | #include "tests/gtest/include/gtest/gtest.h" | ||||||
|  |  | ||||||
| // Set to 1 to enable verbose debugging info logging. | // Set to 1 and add `--enable-logging --vmodule=*frame*=1 --log-file=<path>` to | ||||||
|  | // the command-line to enable verbose debugging info logging. | ||||||
| #define VERBOSE_DEBUGGING 0 | #define VERBOSE_DEBUGGING 0 | ||||||
|  |  | ||||||
| namespace { | namespace { | ||||||
| @@ -33,8 +35,9 @@ struct FrameStatus { | |||||||
|     MAIN_FRAME_CHANGED_ASSIGNED, |     MAIN_FRAME_CHANGED_ASSIGNED, | ||||||
|     LOAD_START, |     LOAD_START, | ||||||
|     LOAD_END, |     LOAD_END, | ||||||
|     BEFORE_CLOSE, |  | ||||||
|     FRAME_DETACHED, |     FRAME_DETACHED, | ||||||
|  |     BEFORE_CLOSE, | ||||||
|  |     FRAME_DESTROYED, | ||||||
|     MAIN_FRAME_CHANGED_REMOVED, |     MAIN_FRAME_CHANGED_REMOVED, | ||||||
|     MAIN_FRAME_FINAL_REMOVED, |     MAIN_FRAME_FINAL_REMOVED, | ||||||
|  |  | ||||||
| @@ -57,10 +60,12 @@ struct FrameStatus { | |||||||
|         return "OnLoadStart"; |         return "OnLoadStart"; | ||||||
|       case LOAD_END: |       case LOAD_END: | ||||||
|         return "OnLoadEnd"; |         return "OnLoadEnd"; | ||||||
|       case BEFORE_CLOSE: |  | ||||||
|         return "OnBeforeClose"; |  | ||||||
|       case FRAME_DETACHED: |       case FRAME_DETACHED: | ||||||
|         return "OnFrameDetached"; |         return "OnFrameDetached"; | ||||||
|  |       case BEFORE_CLOSE: | ||||||
|  |         return "OnBeforeClose"; | ||||||
|  |       case FRAME_DESTROYED: | ||||||
|  |         return "OnFrameDestroyed"; | ||||||
|       case MAIN_FRAME_CHANGED_REMOVED: |       case MAIN_FRAME_CHANGED_REMOVED: | ||||||
|         return "OnMainFrameChanged(changed_removed)"; |         return "OnMainFrameChanged(changed_removed)"; | ||||||
|       case MAIN_FRAME_FINAL_REMOVED: |       case MAIN_FRAME_FINAL_REMOVED: | ||||||
| @@ -133,7 +138,9 @@ struct FrameStatus { | |||||||
| #endif | #endif | ||||||
|     return got_callback_[LOAD_END]; |     return got_callback_[LOAD_END]; | ||||||
|   } |   } | ||||||
|   bool IsDetached() const { return got_callback_[FRAME_DETACHED]; } |   bool IsDestroyed() const { return got_callback_[FRAME_DESTROYED]; } | ||||||
|  |  | ||||||
|  |   bool IsMain() const { return is_main_; } | ||||||
|  |  | ||||||
|   void SetIsFirstMain(bool val) { |   void SetIsFirstMain(bool val) { | ||||||
|     EXPECT_TRUE(is_main_); |     EXPECT_TRUE(is_main_); | ||||||
| @@ -148,17 +155,24 @@ struct FrameStatus { | |||||||
|     is_last_main_ = val; |     is_last_main_ = val; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void SetIsTemporary(bool val) { |   void SetIsTemporary(bool val) { is_temporary_ = val; } | ||||||
|     EXPECT_FALSE(is_main_); |  | ||||||
|     is_temporary_ = val; |  | ||||||
|   } |  | ||||||
|   bool IsTemporary() const { return is_temporary_; } |   bool IsTemporary() const { return is_temporary_; } | ||||||
|  |  | ||||||
|   void SetAdditionalDebugInfo(const std::string& debug_info) { |   void SetAdditionalDebugInfo(const std::string& debug_info) { | ||||||
|     debug_info_ = debug_info; |     debug_info_ = debug_info; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   std::string GetDebugString() const { return debug_info_ + ident_str_; } |   std::string GetDebugString(bool dump_state = false) const { | ||||||
|  |     std::string result = debug_info_ + ident_str_; | ||||||
|  |     if (dump_state) { | ||||||
|  |       std::stringstream ss; | ||||||
|  |       ss << "\nis_main=" << is_main_ << "\nis_first_main=" << is_first_main_ | ||||||
|  |          << "\nis_last_main=" << is_last_main_ | ||||||
|  |          << "\nis_temporary=" << is_temporary_; | ||||||
|  |       result += ss.str(); | ||||||
|  |     } | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   // The main frame will be reused for same-origin navigations. |   // The main frame will be reused for same-origin navigations. | ||||||
|   void ResetMainLoadStatus() { |   void ResetMainLoadStatus() { | ||||||
| @@ -200,12 +214,34 @@ struct FrameStatus { | |||||||
|                        CefRefPtr<CefFrame> frame) { |                        CefRefPtr<CefFrame> frame) { | ||||||
|     EXPECT_UI_THREAD(); |     EXPECT_UI_THREAD(); | ||||||
|     VerifyBrowser(__FUNCTION__, browser); |     VerifyBrowser(__FUNCTION__, browser); | ||||||
|     // A frame is never valid after it's detached. |  | ||||||
|     VerifyFrame(__FUNCTION__, frame, /*expect_valid=*/false); |     // Don't check IsValid() here for sub-frames. Invalidation will occur during | ||||||
|  |     // destruction and we may become detached either before or during | ||||||
|  |     // destruction, so the value is not guaranteed. | ||||||
|  |     std::optional<bool> expect_valid; | ||||||
|  |  | ||||||
|  |     if (frame->IsMain()) { | ||||||
|  |       // Check IsValid() here for main frames. It will only return true for the | ||||||
|  |       // current main frame. | ||||||
|  |       expect_valid = | ||||||
|  |           frame->GetIdentifier() == browser->GetMainFrame()->GetIdentifier(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     VerifyFrame(__FUNCTION__, frame, expect_valid); | ||||||
|  |  | ||||||
|     GotCallback(__FUNCTION__, FRAME_DETACHED); |     GotCallback(__FUNCTION__, FRAME_DETACHED); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                         CefRefPtr<CefFrame> frame) { | ||||||
|  |     EXPECT_UI_THREAD(); | ||||||
|  |     VerifyBrowser(__FUNCTION__, browser); | ||||||
|  |     // A frame is never valid after it's destroyed. | ||||||
|  |     VerifyFrame(__FUNCTION__, frame, /*expect_valid=*/false); | ||||||
|  |  | ||||||
|  |     GotCallback(__FUNCTION__, FRAME_DESTROYED); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, |   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, | ||||||
|                           CefRefPtr<CefFrame> old_frame, |                           CefRefPtr<CefFrame> old_frame, | ||||||
|                           CefRefPtr<CefFrame> new_frame) { |                           CefRefPtr<CefFrame> new_frame) { | ||||||
| @@ -257,7 +293,7 @@ struct FrameStatus { | |||||||
|  |  | ||||||
|   // Called for all existing frames, not just the target frame. |   // Called for all existing frames, not just the target frame. | ||||||
|   // We need to track this status to know if the browser should be valid in |   // We need to track this status to know if the browser should be valid in | ||||||
|   // following calls to OnFrameDetached. |   // following calls to OnFrameDetached/OnFrameDestroyed. | ||||||
|   void OnBeforeClose(CefRefPtr<CefBrowser> browser) { |   void OnBeforeClose(CefRefPtr<CefBrowser> browser) { | ||||||
|     EXPECT_UI_THREAD(); |     EXPECT_UI_THREAD(); | ||||||
|     VerifyBrowser(__FUNCTION__, browser); |     VerifyBrowser(__FUNCTION__, browser); | ||||||
| @@ -325,7 +361,6 @@ struct FrameStatus { | |||||||
|  |  | ||||||
|     if (is_temporary_) { |     if (is_temporary_) { | ||||||
|       // Should not receive any queries. |       // Should not receive any queries. | ||||||
|       EXPECT_FALSE(is_main_); |  | ||||||
|       EXPECT_EQ(0, delivered_query_ct_); |       EXPECT_EQ(0, delivered_query_ct_); | ||||||
|     } else { |     } else { | ||||||
|       // Verify that all expected messages have been sent and received. |       // Verify that all expected messages have been sent and received. | ||||||
| @@ -357,6 +392,14 @@ struct FrameStatus { | |||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     if (is_temporary_) { | ||||||
|  |       // Temporary frames should not connect or load. | ||||||
|  |       if (callback == FRAME_ATTACHED || callback == FRAME_DETACHED || | ||||||
|  |           callback == LOAD_START || callback == LOAD_END) { | ||||||
|  |         return false; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     if (is_main_) { |     if (is_main_) { | ||||||
|       if ((callback == MAIN_FRAME_INITIAL_ASSIGNED || |       if ((callback == MAIN_FRAME_INITIAL_ASSIGNED || | ||||||
|            callback == AFTER_CREATED) && |            callback == AFTER_CREATED) && | ||||||
| @@ -373,44 +416,43 @@ struct FrameStatus { | |||||||
|       if (callback == MAIN_FRAME_CHANGED_REMOVED && is_last_main_) { |       if (callback == MAIN_FRAME_CHANGED_REMOVED && is_last_main_) { | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|     } else if (is_temporary_) { |  | ||||||
|       // For cross-process sub-frame navigation a sub-frame is first created in |  | ||||||
|       // the parent's renderer process. That sub-frame is then discarded after |  | ||||||
|       // the real cross-origin sub-frame is created in a different renderer |  | ||||||
|       // process. These discarded sub-frames will get OnFrameCreated/ |  | ||||||
|       // OnFrameAttached immediately followed by OnFrameDetached. |  | ||||||
|       return callback == FRAME_CREATED || callback == FRAME_ATTACHED || |  | ||||||
|              callback == FRAME_DETACHED; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   bool IsFlakyCallbackOrder(int callback1, int callback2) const { | ||||||
|  |     if (callback1 == FRAME_ATTACHED && | ||||||
|  |         (callback2 == MAIN_FRAME_CHANGED_ASSIGNED || callback2 == LOAD_START || | ||||||
|  |          callback2 == LOAD_END)) { | ||||||
|  |       // Timing of OnFrameAttached is flaky. See issue #3817. | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void VerifyCallbackStatus(const std::string& func, |   void VerifyCallbackStatus(const std::string& func, | ||||||
|                             int current_callback) const { |                             int current_callback) const { | ||||||
|     EXPECT_UI_THREAD(); |     EXPECT_UI_THREAD(); | ||||||
|  |  | ||||||
|     for (int i = 0; i <= CALLBACK_LAST; ++i) { |     for (int i = 0; i <= CALLBACK_LAST; ++i) { | ||||||
|       if (i < current_callback && IsExpectedCallback(i)) { |       if (i < current_callback && IsExpectedCallback(i)) { | ||||||
|         if (i == FRAME_ATTACHED && |         if (IsFlakyCallbackOrder(i, current_callback)) { | ||||||
|             (current_callback == MAIN_FRAME_CHANGED_ASSIGNED || |  | ||||||
|              current_callback == LOAD_START || current_callback == LOAD_END)) { |  | ||||||
|           // Timing of OnFrameAttached is flaky. See issue #3817. |  | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         EXPECT_TRUE(got_callback_[i]) |         EXPECT_TRUE(got_callback_[i]) | ||||||
|             << "inside " << func << " should already have gotten " |             << "inside " << func << "\nfor " | ||||||
|             << GetCallbackName(i); |             << GetDebugString(/*dump_state=*/true) | ||||||
|  |             << "\nshould already have gotten " << GetCallbackName(i); | ||||||
|       } else { |       } else { | ||||||
|         if (current_callback == FRAME_ATTACHED && |         if (IsFlakyCallbackOrder(current_callback, i)) { | ||||||
|             (i == MAIN_FRAME_CHANGED_ASSIGNED || i == LOAD_START || |  | ||||||
|              i == LOAD_END)) { |  | ||||||
|           // Timing of OnFrameAttached is flaky. See issue #3817. |  | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         EXPECT_FALSE(got_callback_[i]) |         EXPECT_FALSE(got_callback_[i]) | ||||||
|             << "inside " << func << " should not already have gotten " |             << "inside " << func << "\nfor " | ||||||
|             << GetCallbackName(i); |             << GetDebugString(/*dump_state=*/true) | ||||||
|  |             << "\nshould NOT already have gotten " << GetCallbackName(i); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -436,7 +478,8 @@ struct FrameStatus { | |||||||
|     // nicely with the concept that "GetMainFrame() always returns a frame that |     // nicely with the concept that "GetMainFrame() always returns a frame that | ||||||
|     // can be used", which wouldn't be the case if we returned the old frame |     // can be used", which wouldn't be the case if we returned the old frame | ||||||
|     // when calling GetMainFrame() from inside OnFrameCreated (for the new |     // when calling GetMainFrame() from inside OnFrameCreated (for the new | ||||||
|     // frame), OnFrameDetached (for the old frame) or OnMainFrameChanged. |     // frame), OnFrameDetached/OnFrameDestoyed (for the old frame) or | ||||||
|  |     // OnMainFrameChanged. | ||||||
|     auto main_frame = browser->GetMainFrame(); |     auto main_frame = browser->GetMainFrame(); | ||||||
|     if (expect_valid) { |     if (expect_valid) { | ||||||
|       EXPECT_TRUE(main_frame) << func; |       EXPECT_TRUE(main_frame) << func; | ||||||
| @@ -452,11 +495,13 @@ struct FrameStatus { | |||||||
|  |  | ||||||
|   void VerifyFrame(const std::string& func, |   void VerifyFrame(const std::string& func, | ||||||
|                    CefRefPtr<CefFrame> frame, |                    CefRefPtr<CefFrame> frame, | ||||||
|                    bool expect_valid = true) const { |                    std::optional<bool> expect_valid = true) const { | ||||||
|     if (expect_valid) { |     if (expect_valid.has_value()) { | ||||||
|       EXPECT_TRUE(frame->IsValid()) << func; |       if (*expect_valid) { | ||||||
|     } else { |         EXPECT_TRUE(frame->IsValid()) << func; | ||||||
|       EXPECT_FALSE(frame->IsValid()) << func; |       } else { | ||||||
|  |         EXPECT_FALSE(frame->IsValid()) << func; | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // |frame| should be us. This checks the frame type and ID. |     // |frame| should be us. This checks the frame type and ID. | ||||||
| @@ -648,6 +693,13 @@ class OrderMainTestHandler : public RoutingTestHandler, public CefFrameHandler { | |||||||
|     current_main_frame_->OnFrameDetached(browser, frame); |     current_main_frame_->OnFrameDetached(browser, frame); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                         CefRefPtr<CefFrame> frame) override { | ||||||
|  |     EXPECT_UI_THREAD(); | ||||||
|  |     EXPECT_TRUE(current_main_frame_); | ||||||
|  |     current_main_frame_->OnFrameDestroyed(browser, frame); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, |   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, | ||||||
|                           CefRefPtr<CefFrame> old_frame, |                           CefRefPtr<CefFrame> old_frame, | ||||||
|                           CefRefPtr<CefFrame> new_frame) override { |                           CefRefPtr<CefFrame> new_frame) override { | ||||||
| @@ -1003,14 +1055,14 @@ class FrameStatusMap { | |||||||
|     return true; |     return true; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool AllFramesDetached() const { |   bool AllFramesDestroyed() const { | ||||||
|     if (size() != expected_frame_ct_) { |     if (size() != expected_frame_ct_) { | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Map::const_iterator it = frame_map_.begin(); |     Map::const_iterator it = frame_map_.begin(); | ||||||
|     for (; it != frame_map_.end(); ++it) { |     for (; it != frame_map_.end(); ++it) { | ||||||
|       if (!it->second->IsDetached()) { |       if (!it->second->IsDestroyed()) { | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -1125,21 +1177,33 @@ class OrderSubTestHandler : public NavigateOrderMainTestHandler { | |||||||
|  |  | ||||||
|   void OnFrameDetached(CefRefPtr<CefBrowser> browser, |   void OnFrameDetached(CefRefPtr<CefBrowser> browser, | ||||||
|                        CefRefPtr<CefFrame> frame) override { |                        CefRefPtr<CefFrame> frame) override { | ||||||
|  |     if (!frame->IsMain()) { | ||||||
|  |       auto map = GetFrameMap(frame); | ||||||
|  |       auto status = map->GetFrameStatus(frame); | ||||||
|  |       status->OnFrameDetached(browser, frame); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     NavigateOrderMainTestHandler::OnFrameDetached(browser, frame); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                         CefRefPtr<CefFrame> frame) override { | ||||||
|     if (!frame->IsMain()) { |     if (!frame->IsMain()) { | ||||||
|       // Potentially the last notification for an old sub-frame after |       // Potentially the last notification for an old sub-frame after | ||||||
|       // navigation. |       // navigation. | ||||||
|       auto map = GetFrameMap(frame); |       auto map = GetFrameMap(frame); | ||||||
|       auto status = map->GetFrameStatus(frame); |       auto status = map->GetFrameStatus(frame); | ||||||
|       status->OnFrameDetached(browser, frame); |       status->OnFrameDestroyed(browser, frame); | ||||||
|  |  | ||||||
|       if (map->AllFramesDetached()) { |       if (map->AllFramesDestroyed()) { | ||||||
|         // Verify results from the previous navigation. |         // Verify results from the previous navigation. | ||||||
|         VerifyAndClearSubFrameTestResults(map); |         VerifyAndClearSubFrameTestResults(map); | ||||||
|       } |       } | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     NavigateOrderMainTestHandler::OnFrameDetached(browser, frame); |     NavigateOrderMainTestHandler::OnFrameDestroyed(browser, frame); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void OnLoadStart(CefRefPtr<CefBrowser> browser, |   void OnLoadStart(CefRefPtr<CefBrowser> browser, | ||||||
| @@ -1390,23 +1454,45 @@ class CrossOriginOrderSubTestHandler : public OrderSubTestHandler { | |||||||
|                             mode, |                             mode, | ||||||
|                             /*expected_frame_ct=*/4U) {} |                             /*expected_frame_ct=*/4U) {} | ||||||
|  |  | ||||||
|   void OnFrameDetached(CefRefPtr<CefBrowser> browser, |   void OnFrameCreated(CefRefPtr<CefBrowser> browser, | ||||||
|                        CefRefPtr<CefFrame> frame) override { |                       CefRefPtr<CefFrame> frame) override { | ||||||
|  |     OrderSubTestHandler::OnFrameCreated(browser, frame); | ||||||
|  |  | ||||||
|  |     if (!frame->IsMain() && | ||||||
|  |         loaded_frame_child_ids_.find(ExtractChildId(frame->GetIdentifier())) != | ||||||
|  |             loaded_frame_child_ids_.end()) { | ||||||
|  |       // Mark sub-frames in the same process as a loaded frame as temporary. | ||||||
|  |       // See below comments in OnFrameDestroyed. | ||||||
|  |       auto map = GetFrameMap(frame); | ||||||
|  |       auto status = map->GetFrameStatus(frame); | ||||||
|  |       status->SetIsTemporary(true); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void OnLoadStart(CefRefPtr<CefBrowser> browser, | ||||||
|  |                    CefRefPtr<CefFrame> frame, | ||||||
|  |                    TransitionType transition_type) override { | ||||||
|  |     OrderSubTestHandler::OnLoadStart(browser, frame, transition_type); | ||||||
|  |  | ||||||
|  |     loaded_frame_child_ids_.insert(ExtractChildId(frame->GetIdentifier())); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|  |                         CefRefPtr<CefFrame> frame) override { | ||||||
|     // A sub-frame is first created in the parent's renderer process. That |     // A sub-frame is first created in the parent's renderer process. That | ||||||
|     // sub-frame is then discarded after the real cross-origin sub-frame is |     // sub-frame is then discarded after the real cross-origin sub-frame is | ||||||
|     // created in a different renderer process. These discarded sub-frames will |     // created in a different renderer process. These discarded sub-frames will | ||||||
|     // get OnFrameCreated/OnFrameAttached immediately followed by |     // get OnFrameCreated/OnFrameDestroyed. | ||||||
|     // OnFrameDetached. |  | ||||||
|     if (!frame->IsMain()) { |     if (!frame->IsMain()) { | ||||||
|       auto map = GetFrameMap(frame); |       auto map = GetFrameMap(frame); | ||||||
|       auto status = map->GetFrameStatus(frame); |       auto status = map->GetFrameStatus(frame); | ||||||
|       if (status && !status->DidGetCallback(FrameStatus::LOAD_START)) { |       if (status && !status->DidGetCallback(FrameStatus::LOAD_START)) { | ||||||
|         status->SetIsTemporary(true); |         EXPECT_TRUE(status->IsTemporary()); | ||||||
|         temp_frame_detached_ct_++; |         temp_frame_detached_ct_++; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     OrderSubTestHandler::OnFrameDetached(browser, frame); |     OrderSubTestHandler::OnFrameDestroyed(browser, frame); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  protected: |  protected: | ||||||
| @@ -1431,7 +1517,16 @@ class CrossOriginOrderSubTestHandler : public OrderSubTestHandler { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|  private: |  private: | ||||||
|  |   // Parse the format from frame_util::MakeFrameIdentifier to return |child_id|. | ||||||
|  |   static std::string ExtractChildId(const std::string& frame_id) { | ||||||
|  |     const int pos = frame_id.find('-'); | ||||||
|  |     CHECK_GT(pos, 0) << frame_id; | ||||||
|  |     return frame_id.substr(0, pos); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   size_t temp_frame_detached_ct_ = 0U; |   size_t temp_frame_detached_ct_ = 0U; | ||||||
|  |  | ||||||
|  |   std::set<std::string> loaded_frame_child_ids_; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| }  // namespace | }  // namespace | ||||||
| @@ -1523,6 +1618,7 @@ class PopupOrderMainTestHandler : public OrderMainTestHandler { | |||||||
|       temp_main_frame_->SetAdditionalDebugInfo(GetAdditionalDebugInfo() + |       temp_main_frame_->SetAdditionalDebugInfo(GetAdditionalDebugInfo() + | ||||||
|                                                "temp "); |                                                "temp "); | ||||||
|       temp_main_frame_->SetIsFirstMain(true); |       temp_main_frame_->SetIsFirstMain(true); | ||||||
|  |       temp_main_frame_->SetIsTemporary(true); | ||||||
|       temp_main_frame_->OnFrameCreated(browser, frame); |       temp_main_frame_->OnFrameCreated(browser, frame); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
| @@ -1558,6 +1654,17 @@ class PopupOrderMainTestHandler : public OrderMainTestHandler { | |||||||
|     OrderMainTestHandler::OnFrameAttached(browser, frame, reattached); |     OrderMainTestHandler::OnFrameAttached(browser, frame, reattached); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   void OnFrameDetached(CefRefPtr<CefBrowser> browser, | ||||||
|  |                        CefRefPtr<CefFrame> frame) override { | ||||||
|  |     if (temp_main_frame_ && temp_main_frame_->IsSame(frame)) { | ||||||
|  |       EXPECT_TRUE(cross_origin_); | ||||||
|  |       temp_main_frame_->OnFrameDetached(browser, frame); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     OrderMainTestHandler::OnFrameDetached(browser, frame); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, |   void OnMainFrameChanged(CefRefPtr<CefBrowser> browser, | ||||||
|                           CefRefPtr<CefFrame> old_frame, |                           CefRefPtr<CefFrame> old_frame, | ||||||
|                           CefRefPtr<CefFrame> new_frame) override { |                           CefRefPtr<CefFrame> new_frame) override { | ||||||
| @@ -1570,8 +1677,8 @@ class PopupOrderMainTestHandler : public OrderMainTestHandler { | |||||||
|     OrderMainTestHandler::OnMainFrameChanged(browser, old_frame, new_frame); |     OrderMainTestHandler::OnMainFrameChanged(browser, old_frame, new_frame); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   void OnFrameDetached(CefRefPtr<CefBrowser> browser, |   void OnFrameDestroyed(CefRefPtr<CefBrowser> browser, | ||||||
|                        CefRefPtr<CefFrame> frame) override { |                         CefRefPtr<CefFrame> frame) override { | ||||||
|     if (temp_main_frame_ && temp_main_frame_->IsSame(frame)) { |     if (temp_main_frame_ && temp_main_frame_->IsSame(frame)) { | ||||||
|       EXPECT_TRUE(cross_origin_); |       EXPECT_TRUE(cross_origin_); | ||||||
|       EXPECT_FALSE(got_temp_destroyed_); |       EXPECT_FALSE(got_temp_destroyed_); | ||||||
| @@ -1579,28 +1686,30 @@ class PopupOrderMainTestHandler : public OrderMainTestHandler { | |||||||
|  |  | ||||||
| #if VERBOSE_DEBUGGING | #if VERBOSE_DEBUGGING | ||||||
|       LOG(INFO) << temp_main_frame_->GetDebugString() |       LOG(INFO) << temp_main_frame_->GetDebugString() | ||||||
|                 << " callback OnFrameDetached(discarded)"; |                 << " callback OnFrameDestroyed(discarded)"; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|       // All of the initial main frame callbacks go to the proxy. |       // All of the initial main frame callbacks go to the proxy. | ||||||
|       EXPECT_TRUE(temp_main_frame_->DidGetCallback(FrameStatus::AFTER_CREATED)); |       EXPECT_TRUE(temp_main_frame_->DidGetCallback(FrameStatus::AFTER_CREATED)); | ||||||
|       EXPECT_TRUE(temp_main_frame_->DidGetCallback( |       EXPECT_TRUE(temp_main_frame_->DidGetCallback( | ||||||
|           FrameStatus::MAIN_FRAME_INITIAL_ASSIGNED)); |           FrameStatus::MAIN_FRAME_INITIAL_ASSIGNED)); | ||||||
|       EXPECT_TRUE(!temp_main_frame_->DidGetCallback(FrameStatus::LOAD_START)); |       EXPECT_FALSE(temp_main_frame_->DidGetCallback(FrameStatus::LOAD_START)); | ||||||
|  |       EXPECT_FALSE(temp_main_frame_->DidGetCallback(FrameStatus::LOAD_END)); | ||||||
|       EXPECT_TRUE(temp_main_frame_->DidGetCallback(FrameStatus::FRAME_CREATED)); |       EXPECT_TRUE(temp_main_frame_->DidGetCallback(FrameStatus::FRAME_CREATED)); | ||||||
|       EXPECT_TRUE( |       EXPECT_FALSE( | ||||||
|           temp_main_frame_->DidGetCallback(FrameStatus::FRAME_ATTACHED)); |           temp_main_frame_->DidGetCallback(FrameStatus::FRAME_ATTACHED)); | ||||||
|  |       EXPECT_FALSE( | ||||||
|  |           temp_main_frame_->DidGetCallback(FrameStatus::FRAME_DETACHED)); | ||||||
|  |  | ||||||
|       // Should receive queries for OnFrameCreated, OnAfterCreated, |       // Temporary frames never attach. | ||||||
|       // OnFrameAttached. |       EXPECT_EQ(temp_main_frame_->QueriesDeliveredCount(), 0); | ||||||
|       EXPECT_EQ(temp_main_frame_->QueriesDeliveredCount(), 3); |  | ||||||
|  |  | ||||||
|       delete temp_main_frame_; |       delete temp_main_frame_; | ||||||
|       temp_main_frame_ = nullptr; |       temp_main_frame_ = nullptr; | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     OrderMainTestHandler::OnFrameDetached(browser, frame); |     OrderMainTestHandler::OnFrameDestroyed(browser, frame); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   bool OnQuery(CefRefPtr<CefBrowser> browser, |   bool OnQuery(CefRefPtr<CefBrowser> browser, | ||||||
|   | |||||||
| @@ -3593,7 +3593,9 @@ class ExtraInfoNavTestHandler : public TestHandler { | |||||||
|                  CefRefPtr<CefFrame> frame, |                  CefRefPtr<CefFrame> frame, | ||||||
|                  int httpStatusCode) override { |                  int httpStatusCode) override { | ||||||
|     if (popup_opened_) { |     if (popup_opened_) { | ||||||
|       DestroyTest(); |       EXPECT_FALSE(got_load_end_popup_); | ||||||
|  |       got_load_end_popup_.yes(); | ||||||
|  |       MaybeDestroyTest(); | ||||||
|     } else { |     } else { | ||||||
|       GrantPopupPermission(browser->GetHost()->GetRequestContext(), |       GrantPopupPermission(browser->GetHost()->GetRequestContext(), | ||||||
|                            browser->GetMainFrame()->GetURL()); |                            browser->GetMainFrame()->GetURL()); | ||||||
| @@ -3641,9 +3643,12 @@ class ExtraInfoNavTestHandler : public TestHandler { | |||||||
|       EXPECT_TRUE(args->GetBool(0)); |       EXPECT_TRUE(args->GetBool(0)); | ||||||
|       if (popup_opened_) { |       if (popup_opened_) { | ||||||
|         EXPECT_TRUE(args->GetBool(1)); |         EXPECT_TRUE(args->GetBool(1)); | ||||||
|  |         EXPECT_FALSE(got_process_message_popup_); | ||||||
|         got_process_message_popup_.yes(); |         got_process_message_popup_.yes(); | ||||||
|  |         MaybeDestroyTest(); | ||||||
|       } else { |       } else { | ||||||
|         EXPECT_FALSE(args->GetBool(1)); |         EXPECT_FALSE(args->GetBool(1)); | ||||||
|  |         EXPECT_FALSE(got_process_message_main_); | ||||||
|         got_process_message_main_.yes(); |         got_process_message_main_.yes(); | ||||||
|       } |       } | ||||||
|       return true; |       return true; | ||||||
| @@ -3653,15 +3658,23 @@ class ExtraInfoNavTestHandler : public TestHandler { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  protected: |  private: | ||||||
|   bool popup_opened_ = false; |   bool popup_opened_ = false; | ||||||
|   TrackCallback got_process_message_main_; |   TrackCallback got_process_message_main_; | ||||||
|   TrackCallback got_process_message_popup_; |   TrackCallback got_process_message_popup_; | ||||||
|  |   TrackCallback got_load_end_popup_; | ||||||
|  |  | ||||||
|  |   void MaybeDestroyTest() { | ||||||
|  |     if (got_process_message_popup_ && got_load_end_popup_) { | ||||||
|  |       DestroyTest(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   void DestroyTest() override { |   void DestroyTest() override { | ||||||
|     // Verify test expectations. |     // Verify test expectations. | ||||||
|     EXPECT_TRUE(got_process_message_main_); |     EXPECT_TRUE(got_process_message_main_); | ||||||
|     EXPECT_TRUE(got_process_message_popup_); |     EXPECT_TRUE(got_process_message_popup_); | ||||||
|  |     EXPECT_TRUE(got_load_end_popup_); | ||||||
|  |  | ||||||
|     TestHandler::DestroyTest(); |     TestHandler::DestroyTest(); | ||||||
|   } |   } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user