summaryrefslogtreecommitdiffstats
path: root/extensions/renderer
diff options
context:
space:
mode:
authorrob <rob@robwu.nl>2016-02-10 14:45:57 -0800
committerCommit bot <commit-bot@chromium.org>2016-02-10 22:47:21 +0000
commit5a15b72a270b514cd442872221a788a303bdaa88 (patch)
tree896c988afa920b82821fd9f863f2b4804e1e5427 /extensions/renderer
parent6acd65a66713446464ab641cd2e9dd3e19612ef6 (diff)
downloadchromium_src-5a15b72a270b514cd442872221a788a303bdaa88.zip
chromium_src-5a15b72a270b514cd442872221a788a303bdaa88.tar.gz
chromium_src-5a15b72a270b514cd442872221a788a303bdaa88.tar.bz2
Fix re-entrancy and lifetime issue in RenderFrameObserverNatives::OnDocumentElementCreated
BUG=585268,568130 Review URL: https://codereview.chromium.org/1684953002 Cr-Commit-Position: refs/heads/master@{#374758}
Diffstat (limited to 'extensions/renderer')
-rw-r--r--extensions/renderer/render_frame_observer_natives.cc82
-rw-r--r--extensions/renderer/render_frame_observer_natives.h8
2 files changed, 57 insertions, 33 deletions
diff --git a/extensions/renderer/render_frame_observer_natives.cc b/extensions/renderer/render_frame_observer_natives.cc
index c3a147d..1c1f06e8 100644
--- a/extensions/renderer/render_frame_observer_natives.cc
+++ b/extensions/renderer/render_frame_observer_natives.cc
@@ -19,43 +19,29 @@ namespace {
// Deletes itself when done.
class LoadWatcher : public content::RenderFrameObserver {
public:
- LoadWatcher(ScriptContext* context,
- content::RenderFrame* frame,
- v8::Local<v8::Function> cb)
- : content::RenderFrameObserver(frame),
- context_(context),
- callback_(context->isolate(), cb) {
- if (ExtensionFrameHelper::Get(frame)->
- did_create_current_document_element()) {
- // If the document element is already created, then we can call the
- // callback immediately (though post it to the message loop so as to not
- // call it re-entrantly).
- // The Unretained is safe because this class manages its own lifetime.
- base::MessageLoop::current()->PostTask(
- FROM_HERE,
- base::Bind(&LoadWatcher::CallbackAndDie, base::Unretained(this),
- true));
- }
+ LoadWatcher(content::RenderFrame* frame,
+ const base::Callback<void(bool)>& callback)
+ : content::RenderFrameObserver(frame), callback_(callback) {}
+
+ void DidCreateDocumentElement() override {
+ // The callback must be run as soon as the root element is available.
+ // Running the callback may trigger DidCreateDocumentElement or
+ // DidFailProvisionalLoad, so delete this before running the callback.
+ base::Callback<void(bool)> callback = callback_;
+ delete this;
+ callback.Run(true);
}
- void DidCreateDocumentElement() override { CallbackAndDie(true); }
-
void DidFailProvisionalLoad(const blink::WebURLError& error) override {
- CallbackAndDie(false);
- }
-
- private:
- void CallbackAndDie(bool succeeded) {
- v8::Isolate* isolate = context_->isolate();
- v8::HandleScope handle_scope(isolate);
- v8::Local<v8::Value> args[] = {v8::Boolean::New(isolate, succeeded)};
- context_->CallFunction(v8::Local<v8::Function>::New(isolate, callback_),
- arraysize(args), args);
+ // Use PostTask to avoid running user scripts while handling this
+ // DidFailProvisionalLoad notification.
+ base::MessageLoop::current()->PostTask(FROM_HERE,
+ base::Bind(callback_, false));
delete this;
}
- ScriptContext* context_;
- v8::Global<v8::Function> callback_;
+ private:
+ base::Callback<void(bool)> callback_;
DISALLOW_COPY_AND_ASSIGN(LoadWatcher);
};
@@ -63,13 +49,20 @@ class LoadWatcher : public content::RenderFrameObserver {
} // namespace
RenderFrameObserverNatives::RenderFrameObserverNatives(ScriptContext* context)
- : ObjectBackedNativeHandler(context) {
+ : ObjectBackedNativeHandler(context), weak_ptr_factory_(this) {
RouteFunction(
"OnDocumentElementCreated",
base::Bind(&RenderFrameObserverNatives::OnDocumentElementCreated,
base::Unretained(this)));
}
+RenderFrameObserverNatives::~RenderFrameObserverNatives() {}
+
+void RenderFrameObserverNatives::Invalidate() {
+ weak_ptr_factory_.InvalidateWeakPtrs();
+ ObjectBackedNativeHandler::Invalidate();
+}
+
void RenderFrameObserverNatives::OnDocumentElementCreated(
const v8::FunctionCallbackInfo<v8::Value>& args) {
CHECK(args.Length() == 2);
@@ -84,9 +77,32 @@ void RenderFrameObserverNatives::OnDocumentElementCreated(
return;
}
- new LoadWatcher(context(), frame, args[1].As<v8::Function>());
+ v8::Global<v8::Function> v8_callback(context()->isolate(),
+ args[1].As<v8::Function>());
+ base::Callback<void(bool)> callback(
+ base::Bind(&RenderFrameObserverNatives::InvokeCallback,
+ weak_ptr_factory_.GetWeakPtr(), base::Passed(&v8_callback)));
+ if (ExtensionFrameHelper::Get(frame)->did_create_current_document_element()) {
+ // If the document element is already created, then we can call the callback
+ // immediately (though use PostTask to ensure that the callback is called
+ // asynchronously).
+ base::MessageLoop::current()->PostTask(FROM_HERE,
+ base::Bind(callback, true));
+ } else {
+ new LoadWatcher(frame, callback);
+ }
args.GetReturnValue().Set(true);
}
+void RenderFrameObserverNatives::InvokeCallback(
+ v8::Global<v8::Function> callback,
+ bool succeeded) {
+ v8::Isolate* isolate = context()->isolate();
+ v8::HandleScope handle_scope(isolate);
+ v8::Local<v8::Value> args[] = {v8::Boolean::New(isolate, succeeded)};
+ context()->CallFunction(v8::Local<v8::Function>::New(isolate, callback),
+ arraysize(args), args);
+}
+
} // namespace extensions
diff --git a/extensions/renderer/render_frame_observer_natives.h b/extensions/renderer/render_frame_observer_natives.h
index 946377d..4e81e3b 100644
--- a/extensions/renderer/render_frame_observer_natives.h
+++ b/extensions/renderer/render_frame_observer_natives.h
@@ -6,6 +6,7 @@
#define EXTENSIONS_RENDERER_RENDER_FRAME_OBSERVER_NATIVES_H_
#include "base/macros.h"
+#include "base/memory/weak_ptr.h"
#include "extensions/renderer/object_backed_native_handler.h"
namespace extensions {
@@ -15,13 +16,20 @@ class ScriptContext;
class RenderFrameObserverNatives : public ObjectBackedNativeHandler {
public:
explicit RenderFrameObserverNatives(ScriptContext* context);
+ ~RenderFrameObserverNatives() override;
private:
+ void Invalidate() override;
+
// Runs a callback upon creation of new document element inside a render frame
// (document.documentElement).
void OnDocumentElementCreated(
const v8::FunctionCallbackInfo<v8::Value>& args);
+ void InvokeCallback(v8::Global<v8::Function> callback, bool succeeded);
+
+ base::WeakPtrFactory<RenderFrameObserverNatives> weak_ptr_factory_;
+
DISALLOW_COPY_AND_ASSIGN(RenderFrameObserverNatives);
};