Index: webkit/plugins/npapi/test/plugin_delete_plugin_in_deallocate_test.cc |
diff --git a/webkit/plugins/npapi/test/plugin_delete_plugin_in_deallocate_test.cc b/webkit/plugins/npapi/test/plugin_delete_plugin_in_deallocate_test.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a1053be2e0ac12dbb9f387f018c817c4c733c02b |
--- /dev/null |
+++ b/webkit/plugins/npapi/test/plugin_delete_plugin_in_deallocate_test.cc |
@@ -0,0 +1,151 @@ |
+// Copyright (c) 2011 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 "webkit/plugins/npapi/test/plugin_delete_plugin_in_deallocate_test.h" |
+ |
+#include "base/basictypes.h" |
+#include "base/compiler_specific.h" |
+ |
+namespace { |
+ |
+NPAPIClient::DeletePluginInDeallocateTest* g_signalling_instance_ = NULL; |
+ |
+class DeletePluginInDeallocateTestNPObject : public NPObject { |
+ public: |
+ DeletePluginInDeallocateTestNPObject() |
+ : NPObject(), id_(NULL), host_functions_(NULL), deallocate_count_(0) {} |
+ |
+ static NPObject* Allocate(NPP npp, NPClass* npclass) { |
+ return new DeletePluginInDeallocateTestNPObject(); |
+ } |
+ |
+ static void Deallocate(NPObject* npobject) { |
+ DeletePluginInDeallocateTestNPObject* object = |
+ reinterpret_cast<DeletePluginInDeallocateTestNPObject*>(npobject); |
+ ++object->deallocate_count_; |
+ |
+ // Call window.deletePlugin to tear-down our plugin from inside deallocate. |
+ if (object->deallocate_count_ == 1) { |
+ NPIdentifier delete_id = |
+ object->host_functions_->getstringidentifier("deletePlugin"); |
+ NPObject* window_obj = NULL; |
+ object->host_functions_->getvalue(object->id_, NPNVWindowNPObject, |
+ &window_obj); |
+ NPVariant rv; |
+ object->host_functions_->invoke(object->id_, window_obj, delete_id, NULL, |
+ 0, &rv); |
+ } |
+ } |
+ |
+ NPP id_; |
+ NPNetscapeFuncs* host_functions_; |
+ int deallocate_count_; |
+}; |
+ |
+NPClass* GetDeletePluginInDeallocateTestClass() { |
+ static NPClass plugin_class = { |
+ NP_CLASS_STRUCT_VERSION, |
+ DeletePluginInDeallocateTestNPObject::Allocate, |
+ DeletePluginInDeallocateTestNPObject::Deallocate, |
+ NULL, // Invalidate |
+ NULL, // HasMethod |
+ NULL, // Invoke |
+ NULL, // InvokeDefault |
+ NULL, // HasProperty |
+ NULL, // GetProperty |
+ NULL, // SetProperty |
+ NULL, // RemoveProperty |
+ }; |
+ return &plugin_class; |
+} |
+ |
+} // namespace |
+ |
+namespace NPAPIClient { |
+ |
+DeletePluginInDeallocateTest::DeletePluginInDeallocateTest( |
+ NPP id, NPNetscapeFuncs* host_functions) |
+ : PluginTest(id, host_functions), npobject_(NULL), test_started_(false) { |
+} |
+ |
+NPError DeletePluginInDeallocateTest::SetWindow(NPWindow* pNPWindow) { |
+ if (pNPWindow->window == NULL) |
+ return NPERR_NO_ERROR; |
+ |
+ // Ensure that we run the test once, even if SetWindow is called again. |
+ if (test_started_) |
+ return NPERR_NO_ERROR; |
+ test_started_ = true; |
+ |
+ // Because of http://crbug.com/94829, we have to have a second plugin |
+ // instance that we can use to signal completion. |
+ if (test_id() == "signaller") { |
+ g_signalling_instance_ = this; |
+ return NPERR_NO_ERROR; |
+ } |
+ |
+ // Create a custom NPObject and give our Id and the Netscape function table. |
+ npobject_ = HostFunctions()->createobject( |
+ id(), GetDeletePluginInDeallocateTestClass()); |
+ DeletePluginInDeallocateTestNPObject* test_object = |
+ reinterpret_cast<DeletePluginInDeallocateTestNPObject*>(npobject_); |
+ test_object->id_ = id(); |
+ test_object->host_functions_ = HostFunctions(); |
+ |
+ // Fetch the window script object for our page. |
+ NPObject* window = NULL; |
+ HostFunctions()->getvalue(id(), NPNVWindowNPObject, &window); |
+ |
+ // Pass it to the window.setTestObject function, which will later release it. |
+ NPIdentifier set_test_object_id = |
+ HostFunctions()->getstringidentifier("setTestObject"); |
+ NPVariant func_var; |
+ NULL_TO_NPVARIANT(func_var); |
+ HostFunctions()->getproperty(id(), window, set_test_object_id, &func_var); |
+ |
+ NPObject* func = NPVARIANT_TO_OBJECT(func_var); |
+ NPVariant func_arg; |
+ OBJECT_TO_NPVARIANT(npobject_, func_arg); |
+ NPVariant func_result; |
+ HostFunctions()->invokeDefault(id(), func, &func_arg, 1, |
+ &func_result); |
+ |
+ // Release the object - the page's reference should keep it alive. |
+ HostFunctions()->releaseobject(npobject_); |
+ |
+ return NPERR_NO_ERROR; |
+} |
+ |
+NPError DeletePluginInDeallocateTest::Destroy() { |
+ // Because of http://crbug.com/94829, we can't signal completion from within |
+ // the NPP_Destroy of the plugin that is being destroyed. We work-around |
+ // that by testing using a second instance, and signalling though the main |
+ // test instance. |
+ |
+ // There should always be a signalling instance by the time we reach here. |
+ if (!g_signalling_instance_) |
+ return NPERR_NO_ERROR; |
+ |
+ // If we're the signalling instance, do nothing. |
+ if (g_signalling_instance_ == this) |
+ return NPERR_NO_ERROR; |
+ |
+ if (!npobject_) { |
+ g_signalling_instance_->SetError("SetWindow was not invoked."); |
+ } else { |
+ // Verify that our object was deallocated exactly once. |
+ DeletePluginInDeallocateTestNPObject* test_object = |
+ reinterpret_cast<DeletePluginInDeallocateTestNPObject*>(npobject_); |
+ if (test_object->deallocate_count_ != 1) |
+ g_signalling_instance_->SetError( |
+ "Object was not deallocated exactly once."); |
+ delete test_object; |
+ } |
+ |
+ g_signalling_instance_->SignalTestCompleted(); |
+ |
+ return NPERR_NO_ERROR; |
+} |
+ |
+} // namespace NPAPIClient |