// Copyright 2015 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. #include "base/bind.h" #include "base/memory/weak_ptr.h" #include "base/message_loop/message_loop.h" #include "extensions/common/extension.h" #include "extensions/common/extension_set.h" #include "extensions/common/features/feature.h" #include "extensions/renderer/gc_callback.h" #include "extensions/renderer/scoped_web_frame.h" #include "extensions/renderer/script_context.h" #include "extensions/renderer/script_context_set.h" #include "gin/function_template.h" #include "gin/public/context_holder.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/WebKit/public/web/WebFrame.h" #include "v8/include/v8.h" namespace extensions { namespace { void SetToTrue(bool* value) { if (*value) ADD_FAILURE() << "Value is already true"; *value = true; } class GCCallbackTest : public testing::Test { public: GCCallbackTest() : script_context_set_(&active_extensions_) {} protected: base::MessageLoop& message_loop() { return message_loop_; } ScriptContextSet& script_context_set() { return script_context_set_; } v8::Local v8_context() { return v8::Local::New(v8::Isolate::GetCurrent(), v8_context_); } ScriptContext* RegisterScriptContext() { // No extension group or world ID. return script_context_set_.Register( web_frame_.frame(), v8::Local::New(v8::Isolate::GetCurrent(), v8_context_), 0, 0); } void RequestGarbageCollection() { v8::Isolate::GetCurrent()->RequestGarbageCollectionForTesting( v8::Isolate::kFullGarbageCollection); } private: void SetUp() override { v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::HandleScope handle_scope(isolate); v8::Local local_v8_context = v8::Context::New(isolate); v8_context_.Reset(isolate, local_v8_context); // ScriptContexts rely on gin. gin_context_holder_.reset(new gin::ContextHolder(isolate)); gin_context_holder_->SetContext(local_v8_context); } void TearDown() override { gin_context_holder_.reset(); v8_context_.Reset(); RequestGarbageCollection(); } base::MessageLoop message_loop_; ScopedWebFrame web_frame_; // (this will construct the v8::Isolate) ExtensionIdSet active_extensions_; ScriptContextSet script_context_set_; v8::Global v8_context_; scoped_ptr gin_context_holder_; DISALLOW_COPY_AND_ASSIGN(GCCallbackTest); }; TEST_F(GCCallbackTest, GCBeforeContextInvalidated) { v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::HandleScope handle_scope(isolate); v8::Context::Scope context_scope(v8_context()); ScriptContext* script_context = RegisterScriptContext(); bool callback_invoked = false; bool fallback_invoked = false; { // Nest another HandleScope so that |object| and |unreachable_function|'s // handles will be garbage collected. v8::HandleScope handle_scope(isolate); v8::Local object = v8::Object::New(isolate); v8::Local unreachable_function = gin::CreateFunctionTemplate(isolate, base::Bind(SetToTrue, &callback_invoked)); // The GCCallback will delete itself, or memory tests will complain. new GCCallback(script_context, object, unreachable_function->GetFunction(), base::Bind(SetToTrue, &fallback_invoked)); } // Trigger a GC. Only the callback should be invoked. RequestGarbageCollection(); message_loop().RunUntilIdle(); EXPECT_TRUE(callback_invoked); EXPECT_FALSE(fallback_invoked); // Invalidate the context. The fallback should not be invoked because the // callback was already invoked. script_context_set().Remove(script_context); message_loop().RunUntilIdle(); EXPECT_FALSE(fallback_invoked); } TEST_F(GCCallbackTest, ContextInvalidatedBeforeGC) { v8::Isolate* isolate = v8::Isolate::GetCurrent(); v8::HandleScope handle_scope(isolate); v8::Context::Scope context_scope(v8_context()); ScriptContext* script_context = RegisterScriptContext(); bool callback_invoked = false; bool fallback_invoked = false; { // Nest another HandleScope so that |object| and |unreachable_function|'s // handles will be garbage collected. v8::HandleScope handle_scope(isolate); v8::Local object = v8::Object::New(isolate); v8::Local unreachable_function = gin::CreateFunctionTemplate(isolate, base::Bind(SetToTrue, &callback_invoked)); // The GCCallback will delete itself, or memory tests will complain. new GCCallback(script_context, object, unreachable_function->GetFunction(), base::Bind(SetToTrue, &fallback_invoked)); } // Invalidate the context. Only the fallback should be invoked. script_context_set().Remove(script_context); message_loop().RunUntilIdle(); EXPECT_FALSE(callback_invoked); EXPECT_TRUE(fallback_invoked); // Trigger a GC. The callback should not be invoked because the fallback was // already invoked. RequestGarbageCollection(); message_loop().RunUntilIdle(); EXPECT_FALSE(callback_invoked); } } // namespace } // namespace extensions