summaryrefslogtreecommitdiffstats
path: root/chrome_frame/buggy_bho_handling.h
diff options
context:
space:
mode:
authortommi@chromium.org <tommi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-07-21 16:07:24 +0000
committertommi@chromium.org <tommi@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-07-21 16:07:24 +0000
commit0535809cc310e18eefdf3afbd32ff35b593b182f (patch)
tree269d36f76b6ce9e69ab4d88254871d26366dbfa5 /chrome_frame/buggy_bho_handling.h
parent2e4c50cd49a81bef973b934c236b110e660f3746 (diff)
downloadchromium_src-0535809cc310e18eefdf3afbd32ff35b593b182f.zip
chromium_src-0535809cc310e18eefdf3afbd32ff35b593b182f.tar.gz
chromium_src-0535809cc310e18eefdf3afbd32ff35b593b182f.tar.bz2
A different approach to avoid crashes in buggy 3rd party BHOs.This time we're more preceise and only target the buggy components.Behaviour for components that handle browser events correctly, is unchanged even in the presence of buggy DLLs.The core class here is the BuggyBhoTls class and here's the comment for that class:
// Construct an instance of this class on the stack when firing web browser // events that can be sent to buggy BHOs. This class will intercept those // BHOs (see list in cc file) and ignore notifications to those components // for as long as the BuggyBhoTls instance on the stack lives. TEST=A better fix for bug 44463 but also fixes bug 49373 and a number of crashes that haven't been associated with bug reports yet. BUG=44463, 49373 Review URL: http://codereview.chromium.org/3031009 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@53194 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome_frame/buggy_bho_handling.h')
-rw-r--r--chrome_frame/buggy_bho_handling.h86
1 files changed, 86 insertions, 0 deletions
diff --git a/chrome_frame/buggy_bho_handling.h b/chrome_frame/buggy_bho_handling.h
new file mode 100644
index 0000000..5fb26b1
--- /dev/null
+++ b/chrome_frame/buggy_bho_handling.h
@@ -0,0 +1,86 @@
+// Copyright (c) 2010 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_FRAME_BUGGY_BHO_HANDLING_H_
+#define CHROME_FRAME_BUGGY_BHO_HANDLING_H_
+
+#include <atlbase.h>
+#include <atlcom.h>
+#include <exdisp.h>
+
+#include <vector>
+
+#include "base/thread_local.h"
+
+namespace buggy_bho {
+
+// Method prototype for IDispatch::Invoke.
+typedef HRESULT (__stdcall* InvokeFunc)(IDispatch* me, DISPID dispid,
+ REFIID riid, LCID lcid, WORD flags,
+ DISPPARAMS* params, VARIANT* result,
+ EXCEPINFO* ei, UINT* err);
+
+// Construct an instance of this class on the stack when firing web browser
+// events that can be sent to buggy BHOs. This class will intercept those
+// BHOs (see list in cc file) and ignore notifications to those components
+// for as long as the BuggyBhoTls instance on the stack lives.
+class BuggyBhoTls {
+ public:
+ BuggyBhoTls();
+ ~BuggyBhoTls();
+
+ // Call after instantiating an instance of BuggyBhoTls. This method traverses
+ // the list of DWebBrowserEvents and DWebBrowserEvents2 subscribers and checks
+ // if any of the sinks belong to a list of known-to-be-buggy BHOs.
+ // For each of those, a patch will be applied that temporarily ignores certain
+ // invokes.
+ static HRESULT PatchBuggyBHOs(IWebBrowser2* browser);
+
+ protected:
+ // internal implementation:
+
+ // Called when a buggy instance is found to be subscribing to browser events.
+ void AddBuggyObject(IDispatch* obj);
+
+ // Called from our patch to check if calls for this object should be ignored.
+ // The reason we do this check is because there might be more than one browser
+ // object running on the same thread (e.g. IE6) with one running CF and the
+ // other MSHTML. We don't want to drop events being fired by MSHTML, only
+ // events fired by CF since these BHOs should handle MSHTML correctly.
+ bool IsBuggyObject(IDispatch* obj) const;
+
+ // Static, protected member methods
+
+ // Returns the currently registered (TLS) BuggyBhoTls instance or NULL.
+ static BuggyBhoTls* FromCurrentThread();
+
+ // Patches a subscriber if it belongs to a buggy dll.
+ static bool PatchIfBuggy(CONNECTDATA* cd, const IID& diid);
+
+ // Patches the IDispatch::Invoke method.
+ static HRESULT PatchInvokeMethod(PROC* invoke);
+
+ // This is our substitute function that is called instead of the buggy DLLs.
+ // Here we call IsBuggyObject to check if we should ignore invokes or allow
+ // them to go through.
+ static STDMETHODIMP BuggyBhoInvoke(InvokeFunc original, IDispatch* me,
+ DISPID dispid, REFIID riid, LCID lcid, WORD flags, DISPPARAMS* params,
+ VARIANT* result, EXCEPINFO* ei, UINT* err);
+
+ protected:
+ // List of buggy subscribers.
+ std::vector<IDispatch*> bad_objects_;
+
+ // Pointer to a previous instance of BuggyBhoTls on this thread if any.
+ // Under regular circumstances, this will be NULL. However, there's a chance
+ // that we could get reentrant calls, hence we maintain a stack.
+ BuggyBhoTls* previous_instance_;
+
+ // Where we store the current thread's instance.
+ static base::ThreadLocalPointer<BuggyBhoTls> s_bad_object_tls_;
+};
+
+} // end namespace buggy_bho
+
+#endif // CHROME_FRAME_BUGGY_BHO_HANDLING_H_