| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/renderer/pepper/host_var_tracker.h" |
| |
| #include <memory> |
| |
| #include "content/public/test/unittest_test_suite.h" |
| #include "content/renderer/pepper/host_globals.h" |
| #include "content/renderer/pepper/mock_resource.h" |
| #include "content/renderer/pepper/pepper_plugin_instance_impl.h" |
| #include "content/renderer/pepper/pepper_try_catch.h" |
| #include "content/renderer/pepper/v8_var_converter.h" |
| #include "content/renderer/pepper/v8object_var.h" |
| #include "content/test/ppapi_unittest.h" |
| #include "gin/handle.h" |
| #include "gin/wrappable.h" |
| #include "ppapi/c/pp_var.h" |
| #include "ppapi/c/ppp_instance.h" |
| |
| using ppapi::V8ObjectVar; |
| |
| namespace content { |
| |
| namespace { |
| |
| int g_v8objects_alive = 0; |
| |
| class MyObject : public gin::Wrappable<MyObject> { |
| public: |
| static gin::WrapperInfo kWrapperInfo; |
| |
| MyObject(const MyObject&) = delete; |
| MyObject& operator=(const MyObject&) = delete; |
| |
| static v8::Local<v8::Value> Create(v8::Isolate* isolate) { |
| return gin::CreateHandle(isolate, new MyObject()).ToV8(); |
| } |
| |
| private: |
| MyObject() { ++g_v8objects_alive; } |
| ~MyObject() override { --g_v8objects_alive; } |
| }; |
| |
| gin::WrapperInfo MyObject::kWrapperInfo = {gin::kEmbedderNativeGin}; |
| |
| class PepperTryCatchForTest : public PepperTryCatch { |
| public: |
| PepperTryCatchForTest(PepperPluginInstanceImpl* instance, |
| V8VarConverter* converter) |
| : PepperTryCatch(instance, converter), |
| handle_scope_(instance->GetIsolate()), |
| context_scope_(v8::Context::New(instance->GetIsolate())) {} |
| |
| PepperTryCatchForTest(const PepperTryCatchForTest&) = delete; |
| PepperTryCatchForTest& operator=(const PepperTryCatchForTest&) = delete; |
| |
| void SetException(const char* message) override { NOTREACHED(); } |
| bool HasException() override { return false; } |
| v8::Local<v8::Context> GetContext() override { |
| return instance_->GetIsolate()->GetCurrentContext(); |
| } |
| |
| private: |
| v8::HandleScope handle_scope_; |
| v8::Context::Scope context_scope_; |
| }; |
| |
| } // namespace |
| |
| class HostVarTrackerTest : public PpapiUnittest { |
| public: |
| HostVarTrackerTest() {} |
| |
| void TearDown() override { |
| v8::Isolate::GetCurrent()->RequestGarbageCollectionForTesting( |
| v8::Isolate::kFullGarbageCollection); |
| EXPECT_EQ(0, g_v8objects_alive); |
| PpapiUnittest::TearDown(); |
| } |
| |
| HostVarTracker& tracker() { return *HostGlobals::Get()->host_var_tracker(); } |
| }; |
| |
| TEST_F(HostVarTrackerTest, DeleteObjectVarWithInstance) { |
| v8::Isolate* test_isolate = v8::Isolate::GetCurrent(); |
| |
| // Make a second instance (the test harness already creates & manages one). |
| scoped_refptr<PepperPluginInstanceImpl> instance2( |
| PepperPluginInstanceImpl::Create( |
| nullptr, module(), nullptr, GURL(), |
| UnitTestTestSuite::MainThreadIsolateForUnitTestSuite())); |
| PP_Instance pp_instance2 = instance2->pp_instance(); |
| |
| { |
| V8VarConverter converter( |
| instance2->pp_instance(), V8VarConverter::kAllowObjectVars); |
| PepperTryCatchForTest try_catch(instance2.get(), &converter); |
| // Make an object var. |
| ppapi::ScopedPPVar var = try_catch.FromV8(MyObject::Create(test_isolate)); |
| EXPECT_EQ(1, g_v8objects_alive); |
| EXPECT_EQ(1, tracker().GetLiveV8ObjectVarsForTest(pp_instance2)); |
| // Purposely leak the var. |
| var.Release(); |
| } |
| |
| // Free the instance, this should release the ObjectVar. |
| instance2 = nullptr; |
| EXPECT_EQ(0, tracker().GetLiveV8ObjectVarsForTest(pp_instance2)); |
| } |
| |
| // Make sure that using the same v8 object should give the same PP_Var |
| // each time. |
| TEST_F(HostVarTrackerTest, ReuseVar) { |
| V8VarConverter converter( |
| instance()->pp_instance(), V8VarConverter::kAllowObjectVars); |
| PepperTryCatchForTest try_catch(instance(), &converter); |
| |
| v8::Local<v8::Value> v8_object = MyObject::Create(v8::Isolate::GetCurrent()); |
| ppapi::ScopedPPVar pp_object1 = try_catch.FromV8(v8_object); |
| ppapi::ScopedPPVar pp_object2 = try_catch.FromV8(v8_object); |
| |
| // The two results should be the same. |
| EXPECT_EQ(pp_object1.get().value.as_id, pp_object2.get().value.as_id); |
| |
| // The objects should be able to get us back to the associated v8 object. |
| { |
| scoped_refptr<V8ObjectVar> check_object( |
| V8ObjectVar::FromPPVar(pp_object1.get())); |
| ASSERT_TRUE(check_object.get()); |
| EXPECT_EQ(instance(), check_object->instance()); |
| EXPECT_EQ(v8_object, check_object->GetHandle()); |
| } |
| |
| // Remove both of the refs we made above. |
| pp_object1 = ppapi::ScopedPPVar(); |
| pp_object2 = ppapi::ScopedPPVar(); |
| |
| // Releasing the resource should free the internal ref, and so making a new |
| // one now should generate a new ID. |
| ppapi::ScopedPPVar pp_object3 = try_catch.FromV8(v8_object); |
| EXPECT_NE(pp_object1.get().value.as_id, pp_object3.get().value.as_id); |
| } |
| |
| } // namespace content |