1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/basictypes.h"
6 #include "base/memory/scoped_ptr.h"
7 #include "content/renderer/pepper/host_globals.h"
8 #include "content/renderer/pepper/host_var_tracker.h"
9 #include "content/renderer/pepper/mock_resource.h"
10 #include "content/renderer/pepper/pepper_plugin_instance_impl.h"
11 #include "content/renderer/pepper/pepper_try_catch.h"
12 #include "content/renderer/pepper/v8_var_converter.h"
13 #include "content/renderer/pepper/v8object_var.h"
14 #include "content/test/ppapi_unittest.h"
15 #include "gin/handle.h"
16 #include "gin/wrappable.h"
17 #include "ppapi/c/pp_var.h"
18 #include "ppapi/c/ppp_instance.h"
19 #include "third_party/WebKit/public/web/WebBindings.h"
21 using ppapi::V8ObjectVar
;
27 int g_v8objects_alive
= 0;
29 class MyObject
: public gin::Wrappable
<MyObject
> {
31 static gin::WrapperInfo kWrapperInfo
;
33 static v8::Local
<v8::Value
> Create(v8::Isolate
* isolate
) {
34 return gin::CreateHandle(isolate
, new MyObject()).ToV8();
38 MyObject() { ++g_v8objects_alive
; }
39 ~MyObject() override
{ --g_v8objects_alive
; }
41 DISALLOW_COPY_AND_ASSIGN(MyObject
);
44 gin::WrapperInfo
MyObject::kWrapperInfo
= {gin::kEmbedderNativeGin
};
46 class PepperTryCatchForTest
: public PepperTryCatch
{
48 PepperTryCatchForTest(PepperPluginInstanceImpl
* instance
,
49 V8VarConverter
* converter
)
50 : PepperTryCatch(instance
, converter
),
51 handle_scope_(instance
->GetIsolate()),
52 context_scope_(v8::Context::New(instance
->GetIsolate())) {}
54 void SetException(const char* message
) override
{ NOTREACHED(); }
55 bool HasException() override
{ return false; }
56 v8::Local
<v8::Context
> GetContext() override
{
57 return instance_
->GetIsolate()->GetCurrentContext();
61 v8::HandleScope handle_scope_
;
62 v8::Context::Scope context_scope_
;
64 DISALLOW_COPY_AND_ASSIGN(PepperTryCatchForTest
);
69 class HostVarTrackerTest
: public PpapiUnittest
{
71 HostVarTrackerTest() {}
73 void TearDown() override
{
74 v8::Isolate::GetCurrent()->RequestGarbageCollectionForTesting(
75 v8::Isolate::kFullGarbageCollection
);
76 EXPECT_EQ(0, g_v8objects_alive
);
77 PpapiUnittest::TearDown();
80 HostVarTracker
& tracker() { return *HostGlobals::Get()->host_var_tracker(); }
83 TEST_F(HostVarTrackerTest
, DeleteObjectVarWithInstance
) {
84 v8::Isolate
* test_isolate
= v8::Isolate::GetCurrent();
86 // Make a second instance (the test harness already creates & manages one).
87 scoped_refptr
<PepperPluginInstanceImpl
> instance2(
88 PepperPluginInstanceImpl::Create(NULL
, module(), NULL
, GURL()));
89 PP_Instance pp_instance2
= instance2
->pp_instance();
92 V8VarConverter
converter(
93 instance2
->pp_instance(), V8VarConverter::kAllowObjectVars
);
94 PepperTryCatchForTest
try_catch(instance2
.get(), &converter
);
95 // Make an object var.
96 ppapi::ScopedPPVar var
= try_catch
.FromV8(MyObject::Create(test_isolate
));
97 EXPECT_EQ(1, g_v8objects_alive
);
98 EXPECT_EQ(1, tracker().GetLiveV8ObjectVarsForTest(pp_instance2
));
99 // Purposely leak the var.
103 // Free the instance, this should release the ObjectVar.
105 EXPECT_EQ(0, tracker().GetLiveV8ObjectVarsForTest(pp_instance2
));
108 // Make sure that using the same v8 object should give the same PP_Var
110 TEST_F(HostVarTrackerTest
, ReuseVar
) {
111 V8VarConverter
converter(
112 instance()->pp_instance(), V8VarConverter::kAllowObjectVars
);
113 PepperTryCatchForTest
try_catch(instance(), &converter
);
115 v8::Local
<v8::Value
> v8_object
= MyObject::Create(v8::Isolate::GetCurrent());
116 ppapi::ScopedPPVar pp_object1
= try_catch
.FromV8(v8_object
);
117 ppapi::ScopedPPVar pp_object2
= try_catch
.FromV8(v8_object
);
119 // The two results should be the same.
120 EXPECT_EQ(pp_object1
.get().value
.as_id
, pp_object2
.get().value
.as_id
);
122 // The objects should be able to get us back to the associated v8 object.
124 scoped_refptr
<V8ObjectVar
> check_object(
125 V8ObjectVar::FromPPVar(pp_object1
.get()));
126 ASSERT_TRUE(check_object
.get());
127 EXPECT_EQ(instance(), check_object
->instance());
128 EXPECT_EQ(v8_object
, check_object
->GetHandle());
131 // Remove both of the refs we made above.
132 pp_object1
= ppapi::ScopedPPVar();
133 pp_object2
= ppapi::ScopedPPVar();
135 // Releasing the resource should free the internal ref, and so making a new
136 // one now should generate a new ID.
137 ppapi::ScopedPPVar pp_object3
= try_catch
.FromV8(v8_object
);
138 EXPECT_NE(pp_object1
.get().value
.as_id
, pp_object3
.get().value
.as_id
);
141 } // namespace content