/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- * vim: sw=4 ts=4 et : * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "PluginInstanceParent.h" #include "BrowserStreamParent.h" #include "PluginBackgroundDestroyer.h" #include "PluginModuleParent.h" #include "PluginStreamParent.h" #include "StreamNotifyParent.h" #include "npfunctions.h" #include "nsAutoPtr.h" #include "gfxASurface.h" #include "gfxContext.h" #include "gfxPlatform.h" #include "gfxSharedImageSurface.h" #include "nsNPAPIPluginInstance.h" #include "mozilla/StandardInteger.h" // for intptr_t #ifdef MOZ_X11 #include "gfxXlibSurface.h" #endif #include "gfxContext.h" #include "gfxColor.h" #include "gfxUtils.h" #include "nsNPAPIPluginInstance.h" #include "Layers.h" #if defined(OS_WIN) #include #include "gfxWindowsPlatform.h" #include "mozilla/plugins/PluginSurfaceParent.h" // Plugin focus event for widget. extern const PRUnichar* kOOPPPluginFocusEventId; UINT gOOPPPluginFocusEvent = RegisterWindowMessage(kOOPPPluginFocusEventId); extern const PRUnichar* kFlashFullscreenClass; #elif defined(MOZ_WIDGET_GTK) #include #elif defined(XP_MACOSX) #include #endif // defined(XP_MACOSX) using namespace mozilla::plugins; bool StreamNotifyParent::RecvRedirectNotifyResponse(const bool& allow) { PluginInstanceParent* instance = static_cast(Manager()); instance->mNPNIface->urlredirectresponse(instance->mNPP, this, static_cast(allow)); return true; } PluginInstanceParent::PluginInstanceParent(PluginModuleParent* parent, NPP npp, const nsCString& aMimeType, const NPNetscapeFuncs* npniface) : mParent(parent) , mNPP(npp) , mNPNIface(npniface) , mWindowType(NPWindowTypeWindow) , mDrawingModel(kDefaultDrawingModel) #if defined(OS_WIN) , mPluginHWND(NULL) , mPluginWndProc(NULL) , mNestedEventState(false) #endif // defined(XP_WIN) #if defined(XP_MACOSX) , mShWidth(0) , mShHeight(0) , mShColorSpace(nullptr) #endif { #ifdef OS_WIN mTextureMap.Init(); #endif } PluginInstanceParent::~PluginInstanceParent() { if (mNPP) mNPP->pdata = NULL; #if defined(OS_WIN) NS_ASSERTION(!(mPluginHWND || mPluginWndProc), "Subclass was not reset correctly before the dtor was reached!"); #endif #if defined(MOZ_WIDGET_COCOA) if (mShWidth != 0 && mShHeight != 0) { DeallocShmem(mShSurface); } if (mShColorSpace) ::CGColorSpaceRelease(mShColorSpace); #endif if (mRemoteImageDataShmem.IsWritable()) { ImageContainer *container = GetImageContainer(); if (container) { container->SetRemoteImageData(nullptr, nullptr); container->SetCompositionNotifySink(nullptr); DeallocShmem(mRemoteImageDataShmem); } } } bool PluginInstanceParent::Init() { mScriptableObjects.Init(); return true; } void PluginInstanceParent::ActorDestroy(ActorDestroyReason why) { #if defined(OS_WIN) if (why == AbnormalShutdown) { // If the plugin process crashes, this is the only // chance we get to destroy resources. SharedSurfaceRelease(); UnsubclassPluginWindow(); } #endif // After this method, the data backing the remote surface may no // longer be calid. The X surface may be destroyed, or the shared // memory backing this surface may no longer be valid. The right // way to inform the nsObjectFrame that the surface is no longer // valid is with an invalidate call. if (mFrontSurface) { mFrontSurface = NULL; const NPRect rect = {0, 0, 0, 0}; RecvNPN_InvalidateRect(rect); #ifdef MOZ_X11 FinishX(DefaultXDisplay()); #endif } } NPError PluginInstanceParent::Destroy() { NPError retval; if (!CallNPP_Destroy(&retval)) retval = NPERR_GENERIC_ERROR; #if defined(OS_WIN) SharedSurfaceRelease(); UnsubclassPluginWindow(); #endif return retval; } PBrowserStreamParent* PluginInstanceParent::AllocPBrowserStream(const nsCString& url, const uint32_t& length, const uint32_t& lastmodified, PStreamNotifyParent* notifyData, const nsCString& headers, const nsCString& mimeType, const bool& seekable, NPError* rv, uint16_t *stype) { NS_RUNTIMEABORT("Not reachable"); return NULL; } bool PluginInstanceParent::DeallocPBrowserStream(PBrowserStreamParent* stream) { delete stream; return true; } PPluginStreamParent* PluginInstanceParent::AllocPPluginStream(const nsCString& mimeType, const nsCString& target, NPError* result) { return new PluginStreamParent(this, mimeType, target, result); } bool PluginInstanceParent::DeallocPPluginStream(PPluginStreamParent* stream) { delete stream; return true; } bool PluginInstanceParent::AnswerNPN_GetValue_NPNVnetscapeWindow(NativeWindowHandle* value, NPError* result) { #ifdef XP_WIN HWND id; #elif defined(MOZ_X11) XID id; #elif defined(XP_MACOSX) intptr_t id; #elif defined(ANDROID) // TODO: Need Android impl int id; #elif defined(MOZ_WIDGET_QT) // TODO: Need Qt non X impl int id; #else #warning Implement me #endif *result = mNPNIface->getvalue(mNPP, NPNVnetscapeWindow, &id); *value = id; return true; } bool PluginInstanceParent::InternalGetValueForNPObject( NPNVariable aVariable, PPluginScriptableObjectParent** aValue, NPError* aResult) { NPObject* npobject; NPError result = mNPNIface->getvalue(mNPP, aVariable, (void*)&npobject); if (result == NPERR_NO_ERROR) { NS_ASSERTION(npobject, "Shouldn't return null and NPERR_NO_ERROR!"); PluginScriptableObjectParent* actor = GetActorForNPObject(npobject); mNPNIface->releaseobject(npobject); if (actor) { *aValue = actor; *aResult = NPERR_NO_ERROR; return true; } NS_ERROR("Failed to get actor!"); result = NPERR_GENERIC_ERROR; } *aValue = nullptr; *aResult = result; return true; } bool PluginInstanceParent::IsAsyncDrawing() { return IsDrawingModelAsync(mDrawingModel); } bool PluginInstanceParent::AnswerNPN_GetValue_NPNVWindowNPObject( PPluginScriptableObjectParent** aValue, NPError* aResult) { return InternalGetValueForNPObject(NPNVWindowNPObject, aValue, aResult); } bool PluginInstanceParent::AnswerNPN_GetValue_NPNVPluginElementNPObject( PPluginScriptableObjectParent** aValue, NPError* aResult) { return InternalGetValueForNPObject(NPNVPluginElementNPObject, aValue, aResult); } bool PluginInstanceParent::AnswerNPN_GetValue_NPNVprivateModeBool(bool* value, NPError* result) { NPBool v; *result = mNPNIface->getvalue(mNPP, NPNVprivateModeBool, &v); *value = v; return true; } bool PluginInstanceParent::AnswerNPN_GetValue_DrawingModelSupport(const NPNVariable& model, bool* value) { *value = false; #ifdef XP_WIN switch (model) { case NPNVsupportsAsyncWindowsDXGISurfaceBool: { if (gfxWindowsPlatform::GetPlatform()->GetRenderMode() == gfxWindowsPlatform::RENDER_DIRECT2D) { *value = true; } } } #endif return true; } bool PluginInstanceParent::AnswerNPN_GetValue_NPNVdocumentOrigin(nsCString* value, NPError* result) { void *v = nullptr; *result = mNPNIface->getvalue(mNPP, NPNVdocumentOrigin, &v); if (*result == NPERR_NO_ERROR && v) { value->Adopt(static_cast(v)); } return true; } bool PluginInstanceParent::AnswerNPN_SetValue_NPPVpluginWindow( const bool& windowed, NPError* result) { // Yes, we are passing a boolean as a void*. We have to cast to intptr_t // first to avoid gcc warnings about casting to a pointer from a // non-pointer-sized integer. *result = mNPNIface->setvalue(mNPP, NPPVpluginWindowBool, (void*)(intptr_t)windowed); return true; } bool PluginInstanceParent::AnswerNPN_SetValue_NPPVpluginTransparent( const bool& transparent, NPError* result) { *result = mNPNIface->setvalue(mNPP, NPPVpluginTransparentBool, (void*)(intptr_t)transparent); return true; } bool PluginInstanceParent::AnswerNPN_SetValue_NPPVpluginUsesDOMForCursor( const bool& useDOMForCursor, NPError* result) { *result = mNPNIface->setvalue(mNPP, NPPVpluginUsesDOMForCursorBool, (void*)(intptr_t)useDOMForCursor); return true; } class NotificationSink : public CompositionNotifySink { public: NotificationSink(PluginInstanceParent *aInstance) : mInstance(aInstance) { } virtual void DidComposite() { mInstance->DidComposite(); } private: PluginInstanceParent *mInstance; }; bool PluginInstanceParent::AnswerNPN_SetValue_NPPVpluginDrawingModel( const int& drawingModel, OptionalShmem *shmem, CrossProcessMutexHandle *mutex, NPError* result) { *shmem = null_t(); #ifdef XP_MACOSX if (drawingModel == NPDrawingModelCoreAnimation || drawingModel == NPDrawingModelInvalidatingCoreAnimation) { // We need to request CoreGraphics otherwise // the nsObjectFrame will try to draw a CALayer // that can not be shared across process. mDrawingModel = drawingModel; *result = mNPNIface->setvalue(mNPP, NPPVpluginDrawingModel, (void*)NPDrawingModelCoreGraphics); } else #endif if (drawingModel == NPDrawingModelAsyncBitmapSurface #ifdef XP_WIN || drawingModel == NPDrawingModelAsyncWindowsDXGISurface #endif ) { ImageContainer *container = GetImageContainer(); if (!container) { *result = NPERR_GENERIC_ERROR; return true; } #ifdef XP_WIN if (drawingModel == NPDrawingModelAsyncWindowsDXGISurface && gfxWindowsPlatform::GetPlatform()->GetRenderMode() != gfxWindowsPlatform::RENDER_DIRECT2D) { *result = NPERR_GENERIC_ERROR; return true; } #endif mDrawingModel = drawingModel; *result = mNPNIface->setvalue(mNPP, NPPVpluginDrawingModel, reinterpret_cast(static_cast(drawingModel))); if (*result != NPERR_NO_ERROR) { return true; } AllocUnsafeShmem(sizeof(RemoteImageData), SharedMemory::TYPE_BASIC, &mRemoteImageDataShmem); *shmem = mRemoteImageDataShmem; mRemoteImageDataMutex = new CrossProcessMutex("PluginInstanceParent.mRemoteImageDataMutex"); *mutex = mRemoteImageDataMutex->ShareToProcess(OtherProcess()); container->SetRemoteImageData(mRemoteImageDataShmem.get(), mRemoteImageDataMutex); mNotifySink = new NotificationSink(this); container->SetCompositionNotifySink(mNotifySink); } else if ( #if defined(XP_WIN) drawingModel == NPDrawingModelSyncWin #elif defined(XP_MACOSX) drawingModel == NPDrawingModelOpenGL || drawingModel == NPDrawingModelCoreGraphics #elif defined(MOZ_X11) drawingModel == NPDrawingModelSyncX #else false #endif ) { *shmem = null_t(); ImageContainer *container = GetImageContainer(); if (!container) { *result = NPERR_GENERIC_ERROR; return true; } mDrawingModel = drawingModel; *result = mNPNIface->setvalue(mNPP, NPPVpluginDrawingModel, (void*)(intptr_t)drawingModel); if (mRemoteImageDataShmem.IsWritable()) { container->SetRemoteImageData(nullptr, nullptr); container->SetCompositionNotifySink(nullptr); DeallocShmem(mRemoteImageDataShmem); mRemoteImageDataMutex = NULL; } } else { *result = NPERR_GENERIC_ERROR; } return true; } bool PluginInstanceParent::AnswerNPN_SetValue_NPPVpluginEventModel( const int& eventModel, NPError* result) { #ifdef XP_MACOSX *result = mNPNIface->setvalue(mNPP, NPPVpluginEventModel, (void*)eventModel); return true; #else *result = NPERR_GENERIC_ERROR; return true; #endif } bool PluginInstanceParent::AnswerNPN_GetURL(const nsCString& url, const nsCString& target, NPError* result) { *result = mNPNIface->geturl(mNPP, NullableStringGet(url), NullableStringGet(target)); return true; } bool PluginInstanceParent::AnswerNPN_PostURL(const nsCString& url, const nsCString& target, const nsCString& buffer, const bool& file, NPError* result) { *result = mNPNIface->posturl(mNPP, url.get(), NullableStringGet(target), buffer.Length(), buffer.get(), file); return true; } PStreamNotifyParent* PluginInstanceParent::AllocPStreamNotify(const nsCString& url, const nsCString& target, const bool& post, const nsCString& buffer, const bool& file, NPError* result) { return new StreamNotifyParent(); } bool PluginInstanceParent::AnswerPStreamNotifyConstructor(PStreamNotifyParent* actor, const nsCString& url, const nsCString& target, const bool& post, const nsCString& buffer, const bool& file, NPError* result) { bool streamDestroyed = false; static_cast(actor)-> SetDestructionFlag(&streamDestroyed); if (!post) { *result = mNPNIface->geturlnotify(mNPP, NullableStringGet(url), NullableStringGet(target), actor); } else { *result = mNPNIface->posturlnotify(mNPP, NullableStringGet(url), NullableStringGet(target), buffer.Length(), NullableStringGet(buffer), file, actor); } if (streamDestroyed) { // If the stream was destroyed, we must return an error code in the // constructor. *result = NPERR_GENERIC_ERROR; } else { static_cast(actor)->ClearDestructionFlag(); if (*result != NPERR_NO_ERROR) return PStreamNotifyParent::Send__delete__(actor, NPERR_GENERIC_ERROR); } return true; } bool PluginInstanceParent::DeallocPStreamNotify(PStreamNotifyParent* notifyData) { delete notifyData; return true; } bool PluginInstanceParent::RecvNPN_InvalidateRect(const NPRect& rect) { mNPNIface->invalidaterect(mNPP, const_cast(&rect)); return true; } bool PluginInstanceParent::RecvShow(const NPRect& updatedRect, const SurfaceDescriptor& newSurface, SurfaceDescriptor* prevSurface) { PLUGIN_LOG_DEBUG( ("[InstanceParent][%p] RecvShow for ", this, updatedRect.left, updatedRect.top, updatedRect.right - updatedRect.left, updatedRect.bottom - updatedRect.top)); nsRefPtr surface; if (newSurface.type() == SurfaceDescriptor::TShmem) { if (!newSurface.get_Shmem().IsReadable()) { NS_WARNING("back surface not readable"); return false; } surface = gfxSharedImageSurface::Open(newSurface.get_Shmem()); } #ifdef XP_MACOSX else if (newSurface.type() == SurfaceDescriptor::TIOSurfaceDescriptor) { IOSurfaceDescriptor iodesc = newSurface.get_IOSurfaceDescriptor(); RefPtr newIOSurface = MacIOSurface::LookupSurface(iodesc.surfaceId()); if (!newIOSurface) { NS_WARNING("Got bad IOSurfaceDescriptor in RecvShow"); return false; } if (mFrontIOSurface) *prevSurface = IOSurfaceDescriptor(mFrontIOSurface->GetIOSurfaceID()); else *prevSurface = null_t(); mFrontIOSurface = newIOSurface; RecvNPN_InvalidateRect(updatedRect); PLUGIN_LOG_DEBUG((" (RecvShow invalidated for surface %p)", mFrontSurface.get())); return true; } #endif #ifdef MOZ_X11 else if (newSurface.type() == SurfaceDescriptor::TSurfaceDescriptorX11) { surface = newSurface.get_SurfaceDescriptorX11().OpenForeign(); } #endif #ifdef XP_WIN else if (newSurface.type() == SurfaceDescriptor::TPPluginSurfaceParent) { PluginSurfaceParent* s = static_cast(newSurface.get_PPluginSurfaceParent()); surface = s->Surface(); } #endif #ifdef MOZ_X11 if (mFrontSurface && mFrontSurface->GetType() == gfxASurface::SurfaceTypeXlib) // This is the "old front buffer" we're about to hand back to // the plugin. We might still have drawing operations // referencing it, so we XSync here to let them finish before // the plugin starts scribbling on it again, or worse, // destroys it. FinishX(DefaultXDisplay()); #endif if (mFrontSurface && gfxSharedImageSurface::IsSharedImage(mFrontSurface)) *prevSurface = static_cast(mFrontSurface.get())->GetShmem(); else *prevSurface = null_t(); if (surface) { // Notify the cairo backend that this surface has changed behind // its back. gfxRect ur(updatedRect.left, updatedRect.top, updatedRect.right - updatedRect.left, updatedRect.bottom - updatedRect.top); surface->MarkDirty(ur); surface->Flush(); } mFrontSurface = surface; if (!surface) { ImageContainer* container = GetImageContainer(); if (container) { container->SetCurrentImage(nullptr); } } RecvNPN_InvalidateRect(updatedRect); PLUGIN_LOG_DEBUG((" (RecvShow invalidated for surface %p)", mFrontSurface.get())); return true; } nsresult PluginInstanceParent::AsyncSetWindow(NPWindow* aWindow) { NPRemoteWindow window; mWindowType = aWindow->type; window.window = reinterpret_cast(aWindow->window); window.x = aWindow->x; window.y = aWindow->y; window.width = aWindow->width; window.height = aWindow->height; window.clipRect = aWindow->clipRect; window.type = aWindow->type; if (!SendAsyncSetWindow(gfxPlatform::GetPlatform()->ScreenReferenceSurface()->GetType(), window)) return NS_ERROR_FAILURE; return NS_OK; } #if defined(MOZ_WIDGET_QT) && (MOZ_PLATFORM_MAEMO == 6) nsresult PluginInstanceParent::HandleGUIEvent(const nsGUIEvent& anEvent, bool* handled) { switch (anEvent.eventStructType) { case NS_KEY_EVENT: if (!CallHandleKeyEvent(static_cast(anEvent), handled)) { return NS_ERROR_FAILURE; } break; case NS_TEXT_EVENT: if (!CallHandleTextEvent(static_cast(anEvent), handled)) { return NS_ERROR_FAILURE; } break; default: NS_ERROR("Not implemented for this event type"); return NS_ERROR_FAILURE; } return NS_OK; } #endif nsresult PluginInstanceParent::GetImageContainer(ImageContainer** aContainer) { #ifdef XP_MACOSX MacIOSurface* ioSurface = NULL; if (mFrontIOSurface) { ioSurface = mFrontIOSurface; } else if (mIOSurface) { ioSurface = mIOSurface; } if (!mFrontSurface && !ioSurface) #else if (!mFrontSurface) #endif return NS_ERROR_NOT_AVAILABLE; ImageFormat format = CAIRO_SURFACE; #ifdef XP_MACOSX if (ioSurface) { format = MAC_IO_SURFACE; } #endif ImageContainer *container = GetImageContainer(); if (!container) { return NS_ERROR_FAILURE; } if (IsAsyncDrawing()) { NS_IF_ADDREF(container); *aContainer = container; return NS_OK; } nsRefPtr image; image = container->CreateImage(&format, 1); if (!image) { return NS_ERROR_FAILURE; } #ifdef XP_MACOSX if (ioSurface) { NS_ASSERTION(image->GetFormat() == MAC_IO_SURFACE, "Wrong format?"); MacIOSurfaceImage* ioImage = static_cast(image.get()); MacIOSurfaceImage::Data ioData; ioData.mIOSurface = ioSurface; ioImage->SetData(ioData); container->SetCurrentImageInTransaction(ioImage); NS_IF_ADDREF(container); *aContainer = container; return NS_OK; } #endif NS_ASSERTION(image->GetFormat() == CAIRO_SURFACE, "Wrong format?"); CairoImage* pluginImage = static_cast(image.get()); CairoImage::Data cairoData; cairoData.mSurface = mFrontSurface; cairoData.mSize = mFrontSurface->GetSize(); pluginImage->SetData(cairoData); container->SetCurrentImageInTransaction(pluginImage); NS_IF_ADDREF(container); *aContainer = container; return NS_OK; } nsresult PluginInstanceParent::GetImageSize(nsIntSize* aSize) { if (mFrontSurface) { gfxIntSize size = mFrontSurface->GetSize(); *aSize = nsIntSize(size.width, size.height); return NS_OK; } #ifdef XP_MACOSX if (mFrontIOSurface) { *aSize = nsIntSize(mFrontIOSurface->GetWidth(), mFrontIOSurface->GetHeight()); return NS_OK; } else if (mIOSurface) { *aSize = nsIntSize(mIOSurface->GetWidth(), mIOSurface->GetHeight()); return NS_OK; } #endif return NS_ERROR_NOT_AVAILABLE; } #ifdef XP_MACOSX nsresult PluginInstanceParent::IsRemoteDrawingCoreAnimation(bool *aDrawing) { *aDrawing = (NPDrawingModelCoreAnimation == (NPDrawingModel)mDrawingModel || NPDrawingModelInvalidatingCoreAnimation == (NPDrawingModel)mDrawingModel); return NS_OK; } #endif nsresult PluginInstanceParent::SetBackgroundUnknown() { PLUGIN_LOG_DEBUG(("[InstanceParent][%p] SetBackgroundUnknown", this)); if (mBackground) { DestroyBackground(); NS_ABORT_IF_FALSE(!mBackground, "Background not destroyed"); } return NS_OK; } nsresult PluginInstanceParent::BeginUpdateBackground(const nsIntRect& aRect, gfxContext** aCtx) { PLUGIN_LOG_DEBUG( ("[InstanceParent][%p] BeginUpdateBackground for ", this, aRect.x, aRect.y, aRect.width, aRect.height)); if (!mBackground) { // XXX if we failed to create a background surface on one // update, there's no guarantee that later updates will be for // the entire background area until successful. We might want // to fix that eventually. NS_ABORT_IF_FALSE(aRect.TopLeft() == nsIntPoint(0, 0), "Expecting rect for whole frame"); if (!CreateBackground(aRect.Size())) { *aCtx = nullptr; return NS_OK; } } #ifdef DEBUG gfxIntSize sz = mBackground->GetSize(); NS_ABORT_IF_FALSE(nsIntRect(0, 0, sz.width, sz.height).Contains(aRect), "Update outside of background area"); #endif nsRefPtr ctx = new gfxContext(mBackground); *aCtx = ctx.forget().get(); return NS_OK; } nsresult PluginInstanceParent::EndUpdateBackground(gfxContext* aCtx, const nsIntRect& aRect) { PLUGIN_LOG_DEBUG( ("[InstanceParent][%p] EndUpdateBackground for ", this, aRect.x, aRect.y, aRect.width, aRect.height)); #ifdef MOZ_X11 // Have to XSync here to avoid the plugin trying to draw with this // surface racing with its creation in the X server. We also want // to avoid the plugin drawing onto stale pixels, then handing us // back a front surface from those pixels that we might // recomposite for "a while" until the next update. This XSync // still doesn't guarantee that the plugin draws onto a consistent // view of its background, but it does mean that the plugin is // drawing onto pixels no older than those in the latest // EndUpdateBackground(). XSync(DefaultXDisplay(), False); #endif unused << SendUpdateBackground(BackgroundDescriptor(), aRect); return NS_OK; } bool PluginInstanceParent::CreateBackground(const nsIntSize& aSize) { NS_ABORT_IF_FALSE(!mBackground, "Already have a background"); // XXX refactor me #if defined(MOZ_X11) Screen* screen = DefaultScreenOfDisplay(DefaultXDisplay()); Visual* visual = DefaultVisualOfScreen(screen); mBackground = gfxXlibSurface::Create(screen, visual, gfxIntSize(aSize.width, aSize.height)); return !!mBackground; #elif defined(XP_WIN) // We have chosen to create an unsafe surface in which the plugin // can read from the region while we're writing to it. mBackground = gfxSharedImageSurface::CreateUnsafe( this, gfxIntSize(aSize.width, aSize.height), gfxASurface::ImageFormatRGB24); return !!mBackground; #else return nullptr; #endif } void PluginInstanceParent::DestroyBackground() { if (!mBackground) { return; } // Relinquish ownership of |mBackground| to its destroyer PPluginBackgroundDestroyerParent* pbd = new PluginBackgroundDestroyerParent(mBackground); mBackground = nullptr; // If this fails, there's no problem: |bd| will be destroyed along // with the old background surface. unused << SendPPluginBackgroundDestroyerConstructor(pbd); } SurfaceDescriptor PluginInstanceParent::BackgroundDescriptor() { NS_ABORT_IF_FALSE(mBackground, "Need a background here"); // XXX refactor me #ifdef MOZ_X11 gfxXlibSurface* xsurf = static_cast(mBackground.get()); return SurfaceDescriptorX11(xsurf); #endif #ifdef XP_WIN NS_ABORT_IF_FALSE(gfxSharedImageSurface::IsSharedImage(mBackground), "Expected shared image surface"); gfxSharedImageSurface* shmem = static_cast(mBackground.get()); return shmem->GetShmem(); #endif // If this is ever used, which it shouldn't be, it will trigger a // hard assertion in IPDL-generated code. return SurfaceDescriptor(); } ImageContainer* PluginInstanceParent::GetImageContainer() { if (mImageContainer) { return mImageContainer; } mImageContainer = LayerManager::CreateImageContainer(); return mImageContainer; } PPluginBackgroundDestroyerParent* PluginInstanceParent::AllocPPluginBackgroundDestroyer() { NS_RUNTIMEABORT("'Power-user' ctor is used exclusively"); return nullptr; } bool PluginInstanceParent::DeallocPPluginBackgroundDestroyer( PPluginBackgroundDestroyerParent* aActor) { delete aActor; return true; } NPError PluginInstanceParent::NPP_SetWindow(const NPWindow* aWindow) { PLUGIN_LOG_DEBUG(("%s (aWindow=%p)", FULLFUNCTION, (void*) aWindow)); NS_ENSURE_TRUE(aWindow, NPERR_GENERIC_ERROR); NPRemoteWindow window; mWindowType = aWindow->type; #if defined(OS_WIN) // On windowless controls, reset the shared memory surface as needed. if (mWindowType == NPWindowTypeDrawable) { // SharedSurfaceSetWindow will take care of NPRemoteWindow. if (!SharedSurfaceSetWindow(aWindow, window)) { return NPERR_OUT_OF_MEMORY_ERROR; } } else { SubclassPluginWindow(reinterpret_cast(aWindow->window)); window.window = reinterpret_cast(aWindow->window); window.x = aWindow->x; window.y = aWindow->y; window.width = aWindow->width; window.height = aWindow->height; window.type = aWindow->type; } #else window.window = reinterpret_cast(aWindow->window); window.x = aWindow->x; window.y = aWindow->y; window.width = aWindow->width; window.height = aWindow->height; window.clipRect = aWindow->clipRect; // MacOS specific window.type = aWindow->type; #endif #if defined(XP_MACOSX) if (mShWidth != window.width || mShHeight != window.height) { if (mDrawingModel == NPDrawingModelCoreAnimation || mDrawingModel == NPDrawingModelInvalidatingCoreAnimation) { mIOSurface = MacIOSurface::CreateIOSurface(window.width, window.height); } else if (uint32_t(mShWidth * mShHeight) != window.width * window.height) { if (mShWidth != 0 && mShHeight != 0) { DeallocShmem(mShSurface); mShWidth = 0; mShHeight = 0; } if (window.width != 0 && window.height != 0) { if (!AllocShmem(window.width * window.height*4, SharedMemory::TYPE_BASIC, &mShSurface)) { PLUGIN_LOG_DEBUG(("Shared memory could not be allocated.")); return NPERR_GENERIC_ERROR; } } } mShWidth = window.width; mShHeight = window.height; } #endif #if defined(MOZ_X11) && defined(XP_UNIX) && !defined(XP_MACOSX) const NPSetWindowCallbackStruct* ws_info = static_cast(aWindow->ws_info); window.visualID = ws_info->visual ? ws_info->visual->visualid : None; window.colormap = ws_info->colormap; #endif if (!CallNPP_SetWindow(window)) return NPERR_GENERIC_ERROR; return NPERR_NO_ERROR; } NPError PluginInstanceParent::NPP_GetValue(NPPVariable aVariable, void* _retval) { switch (aVariable) { case NPPVpluginWantsAllNetworkStreams: { bool wantsAllStreams; NPError rv; if (!CallNPP_GetValue_NPPVpluginWantsAllNetworkStreams(&wantsAllStreams, &rv)) { return NPERR_GENERIC_ERROR; } if (NPERR_NO_ERROR != rv) { return rv; } (*(NPBool*)_retval) = wantsAllStreams; return NPERR_NO_ERROR; } #ifdef MOZ_X11 case NPPVpluginNeedsXEmbed: { bool needsXEmbed; NPError rv; if (!CallNPP_GetValue_NPPVpluginNeedsXEmbed(&needsXEmbed, &rv)) { return NPERR_GENERIC_ERROR; } if (NPERR_NO_ERROR != rv) { return rv; } (*(NPBool*)_retval) = needsXEmbed; return NPERR_NO_ERROR; } #endif case NPPVpluginScriptableNPObject: { PPluginScriptableObjectParent* actor; NPError rv; if (!CallNPP_GetValue_NPPVpluginScriptableNPObject(&actor, &rv)) { return NPERR_GENERIC_ERROR; } if (NPERR_NO_ERROR != rv) { return rv; } if (!actor) { NS_ERROR("NPPVpluginScriptableNPObject succeeded but null."); return NPERR_GENERIC_ERROR; } const NPNetscapeFuncs* npn = mParent->GetNetscapeFuncs(); if (!npn) { NS_WARNING("No netscape functions?!"); return NPERR_GENERIC_ERROR; } NPObject* object = static_cast(actor)->GetObject(true); NS_ASSERTION(object, "This shouldn't ever be null!"); (*(NPObject**)_retval) = npn->retainobject(object); return NPERR_NO_ERROR; } #ifdef MOZ_ACCESSIBILITY_ATK case NPPVpluginNativeAccessibleAtkPlugId: { nsCString plugId; NPError rv; if (!CallNPP_GetValue_NPPVpluginNativeAccessibleAtkPlugId(&plugId, &rv)) { return NPERR_GENERIC_ERROR; } if (NPERR_NO_ERROR != rv) { return rv; } (*(nsCString*)_retval) = plugId; return NPERR_NO_ERROR; } #endif default: PR_LOG(gPluginLog, PR_LOG_WARNING, ("In PluginInstanceParent::NPP_GetValue: Unhandled NPPVariable %i (%s)", (int) aVariable, NPPVariableToString(aVariable))); return NPERR_GENERIC_ERROR; } } NPError PluginInstanceParent::NPP_SetValue(NPNVariable variable, void* value) { switch (variable) { case NPNVprivateModeBool: NPError result; if (!CallNPP_SetValue_NPNVprivateModeBool(*static_cast(value), &result)) return NPERR_GENERIC_ERROR; return result; default: NS_ERROR("Unhandled NPNVariable in NPP_SetValue"); PR_LOG(gPluginLog, PR_LOG_WARNING, ("In PluginInstanceParent::NPP_SetValue: Unhandled NPNVariable %i (%s)", (int) variable, NPNVariableToString(variable))); return NPERR_GENERIC_ERROR; } } void PluginInstanceParent::NPP_URLRedirectNotify(const char* url, int32_t status, void* notifyData) { if (!notifyData) return; PStreamNotifyParent* streamNotify = static_cast(notifyData); unused << streamNotify->SendRedirectNotify(NullableString(url), status); } int16_t PluginInstanceParent::NPP_HandleEvent(void* event) { PLUGIN_LOG_DEBUG_FUNCTION; #if defined(XP_MACOSX) NPCocoaEvent* npevent = reinterpret_cast(event); #else NPEvent* npevent = reinterpret_cast(event); #endif NPRemoteEvent npremoteevent; npremoteevent.event = *npevent; int16_t handled = 0; #if defined(OS_WIN) if (mWindowType == NPWindowTypeDrawable) { if (IsAsyncDrawing()) { if (npevent->event == WM_PAINT || npevent->event == DoublePassRenderingEvent()) { // This plugin maintains its own async drawing. return handled; } } if (DoublePassRenderingEvent() == npevent->event) { return CallPaint(npremoteevent, &handled) && handled; } switch (npevent->event) { case WM_PAINT: { RECT rect; SharedSurfaceBeforePaint(rect, npremoteevent); if (!CallPaint(npremoteevent, &handled)) { handled = false; } SharedSurfaceAfterPaint(npevent); return handled; } break; case WM_KILLFOCUS: { // When the user selects fullscreen mode in Flash video players, // WM_KILLFOCUS will be delayed by deferred event processing: // WM_LBUTTONUP results in a call to CreateWindow within Flash, // which fires WM_KILLFOCUS. Delayed delivery causes Flash to // misinterpret the event, dropping back out of fullscreen. Trap // this event and drop it. PRUnichar szClass[26]; HWND hwnd = GetForegroundWindow(); if (hwnd && hwnd != mPluginHWND && GetClassNameW(hwnd, szClass, sizeof(szClass)/sizeof(PRUnichar)) && !wcscmp(szClass, kFlashFullscreenClass)) { return 0; } } break; case WM_WINDOWPOSCHANGED: { // We send this in nsObjectFrame just before painting return SendWindowPosChanged(npremoteevent); } break; } } #endif #if defined(MOZ_X11) switch (npevent->type) { case GraphicsExpose: PLUGIN_LOG_DEBUG((" schlepping drawable 0x%lx across the pipe\n", npevent->xgraphicsexpose.drawable)); // Make sure the X server has created the Drawable and completes any // drawing before the plugin draws on top. // // XSync() waits for the X server to complete. Really this parent // process does not need to wait; the child is the process that needs // to wait. A possibly-slightly-better alternative would be to send // an X event to the child that the child would wait for. FinishX(DefaultXDisplay()); return CallPaint(npremoteevent, &handled) ? handled : 0; case ButtonPress: // Release any active pointer grab so that the plugin X client can // grab the pointer if it wishes. Display *dpy = DefaultXDisplay(); # ifdef MOZ_WIDGET_GTK // GDK attempts to (asynchronously) track whether there is an active // grab so ungrab through GDK. gdk_pointer_ungrab(npevent->xbutton.time); # else XUngrabPointer(dpy, npevent->xbutton.time); # endif // Wait for the ungrab to complete. XSync(dpy, False); break; } #endif #ifdef XP_MACOSX if (npevent->type == NPCocoaEventDrawRect) { if (mDrawingModel == NPDrawingModelCoreAnimation || mDrawingModel == NPDrawingModelInvalidatingCoreAnimation) { if (!mIOSurface) { NS_ERROR("No IOSurface allocated."); return false; } if (!CallNPP_HandleEvent_IOSurface(npremoteevent, mIOSurface->GetIOSurfaceID(), &handled)) return false; // no good way to handle errors here... CGContextRef cgContext = npevent->data.draw.context; if (!mShColorSpace) { mShColorSpace = CreateSystemColorSpace(); } if (!mShColorSpace) { PLUGIN_LOG_DEBUG(("Could not allocate ColorSpace.")); return false; } if (cgContext) { nsCARenderer::DrawSurfaceToCGContext(cgContext, mIOSurface, mShColorSpace, npevent->data.draw.x, npevent->data.draw.y, npevent->data.draw.width, npevent->data.draw.height); } return true; } else if (mFrontIOSurface) { CGContextRef cgContext = npevent->data.draw.context; if (!mShColorSpace) { mShColorSpace = CreateSystemColorSpace(); } if (!mShColorSpace) { PLUGIN_LOG_DEBUG(("Could not allocate ColorSpace.")); return false; } if (cgContext) { nsCARenderer::DrawSurfaceToCGContext(cgContext, mFrontIOSurface, mShColorSpace, npevent->data.draw.x, npevent->data.draw.y, npevent->data.draw.width, npevent->data.draw.height); } return true; } else { if (mShWidth == 0 && mShHeight == 0) { PLUGIN_LOG_DEBUG(("NPCocoaEventDrawRect on window of size 0.")); return false; } if (!mShSurface.IsReadable()) { PLUGIN_LOG_DEBUG(("Shmem is not readable.")); return false; } if (!CallNPP_HandleEvent_Shmem(npremoteevent, mShSurface, &handled, &mShSurface)) return false; // no good way to handle errors here... if (!mShSurface.IsReadable()) { PLUGIN_LOG_DEBUG(("Shmem not returned. Either the plugin crashed " "or we have a bug.")); return false; } char* shContextByte = mShSurface.get(); if (!mShColorSpace) { mShColorSpace = CreateSystemColorSpace(); } if (!mShColorSpace) { PLUGIN_LOG_DEBUG(("Could not allocate ColorSpace.")); return false; } CGContextRef shContext = ::CGBitmapContextCreate(shContextByte, mShWidth, mShHeight, 8, mShWidth*4, mShColorSpace, kCGImageAlphaPremultipliedFirst | kCGBitmapByteOrder32Host); if (!shContext) { PLUGIN_LOG_DEBUG(("Could not allocate CGBitmapContext.")); return false; } CGImageRef shImage = ::CGBitmapContextCreateImage(shContext); if (shImage) { CGContextRef cgContext = npevent->data.draw.context; ::CGContextDrawImage(cgContext, CGRectMake(0,0,mShWidth,mShHeight), shImage); ::CGImageRelease(shImage); } else { ::CGContextRelease(shContext); return false; } ::CGContextRelease(shContext); return true; } } #endif if (!CallNPP_HandleEvent(npremoteevent, &handled)) return 0; // no good way to handle errors here... return handled; } NPError PluginInstanceParent::NPP_NewStream(NPMIMEType type, NPStream* stream, NPBool seekable, uint16_t* stype) { PLUGIN_LOG_DEBUG(("%s (type=%s, stream=%p, seekable=%i)", FULLFUNCTION, (char*) type, (void*) stream, (int) seekable)); BrowserStreamParent* bs = new BrowserStreamParent(this, stream); NPError err; if (!CallPBrowserStreamConstructor(bs, NullableString(stream->url), stream->end, stream->lastmodified, static_cast(stream->notifyData), NullableString(stream->headers), NullableString(type), seekable, &err, stype)) return NPERR_GENERIC_ERROR; if (NPERR_NO_ERROR != err) unused << PBrowserStreamParent::Send__delete__(bs); return err; } NPError PluginInstanceParent::NPP_DestroyStream(NPStream* stream, NPReason reason) { PLUGIN_LOG_DEBUG(("%s (stream=%p, reason=%i)", FULLFUNCTION, (void*) stream, (int) reason)); AStream* s = static_cast(stream->pdata); if (s->IsBrowserStream()) { BrowserStreamParent* sp = static_cast(s); if (sp->mNPP != this) NS_RUNTIMEABORT("Mismatched plugin data"); sp->NPP_DestroyStream(reason); return NPERR_NO_ERROR; } else { PluginStreamParent* sp = static_cast(s); if (sp->mInstance != this) NS_RUNTIMEABORT("Mismatched plugin data"); return PPluginStreamParent::Call__delete__(sp, reason, false) ? NPERR_NO_ERROR : NPERR_GENERIC_ERROR; } } void PluginInstanceParent::NPP_Print(NPPrint* platformPrint) { // TODO: implement me NS_ERROR("Not implemented"); } PPluginScriptableObjectParent* PluginInstanceParent::AllocPPluginScriptableObject() { return new PluginScriptableObjectParent(Proxy); } #ifdef DEBUG namespace { struct ActorSearchData { PluginScriptableObjectParent* actor; bool found; }; PLDHashOperator ActorSearch(NPObject* aKey, PluginScriptableObjectParent* aData, void* aUserData) { ActorSearchData* asd = reinterpret_cast(aUserData); if (asd->actor == aData) { asd->found = true; return PL_DHASH_STOP; } return PL_DHASH_NEXT; } } // anonymous namespace #endif // DEBUG bool PluginInstanceParent::DeallocPPluginScriptableObject( PPluginScriptableObjectParent* aObject) { PluginScriptableObjectParent* actor = static_cast(aObject); NPObject* object = actor->GetObject(false); if (object) { NS_ASSERTION(mScriptableObjects.Get(object, nullptr), "NPObject not in the hash!"); mScriptableObjects.Remove(object); } #ifdef DEBUG else { ActorSearchData asd = { actor, false }; mScriptableObjects.EnumerateRead(ActorSearch, &asd); NS_ASSERTION(!asd.found, "Actor in the hash with a null NPObject!"); } #endif delete actor; return true; } bool PluginInstanceParent::RecvPPluginScriptableObjectConstructor( PPluginScriptableObjectParent* aActor) { // This is only called in response to the child process requesting the // creation of an actor. This actor will represent an NPObject that is // created by the plugin and returned to the browser. PluginScriptableObjectParent* actor = static_cast(aActor); NS_ASSERTION(!actor->GetObject(false), "Actor already has an object?!"); actor->InitializeProxy(); NS_ASSERTION(actor->GetObject(false), "Actor should have an object!"); return true; } void PluginInstanceParent::NPP_URLNotify(const char* url, NPReason reason, void* notifyData) { PLUGIN_LOG_DEBUG(("%s (%s, %i, %p)", FULLFUNCTION, url, (int) reason, notifyData)); PStreamNotifyParent* streamNotify = static_cast(notifyData); unused << PStreamNotifyParent::Send__delete__(streamNotify, reason); } bool PluginInstanceParent::RegisterNPObjectForActor( NPObject* aObject, PluginScriptableObjectParent* aActor) { NS_ASSERTION(aObject && aActor, "Null pointers!"); NS_ASSERTION(mScriptableObjects.IsInitialized(), "Hash not initialized!"); NS_ASSERTION(!mScriptableObjects.Get(aObject, nullptr), "Duplicate entry!"); mScriptableObjects.Put(aObject, aActor); return true; } void PluginInstanceParent::UnregisterNPObject(NPObject* aObject) { NS_ASSERTION(aObject, "Null pointer!"); NS_ASSERTION(mScriptableObjects.IsInitialized(), "Hash not initialized!"); NS_ASSERTION(mScriptableObjects.Get(aObject, nullptr), "Unknown entry!"); mScriptableObjects.Remove(aObject); } PluginScriptableObjectParent* PluginInstanceParent::GetActorForNPObject(NPObject* aObject) { NS_ASSERTION(aObject, "Null pointer!"); if (aObject->_class == PluginScriptableObjectParent::GetClass()) { // One of ours! ParentNPObject* object = static_cast(aObject); NS_ASSERTION(object->parent, "Null actor!"); return object->parent; } PluginScriptableObjectParent* actor; if (mScriptableObjects.Get(aObject, &actor)) { return actor; } actor = new PluginScriptableObjectParent(LocalObject); if (!actor) { NS_ERROR("Out of memory!"); return nullptr; } if (!SendPPluginScriptableObjectConstructor(actor)) { NS_WARNING("Failed to send constructor message!"); return nullptr; } actor->InitializeLocal(aObject); return actor; } PPluginSurfaceParent* PluginInstanceParent::AllocPPluginSurface(const WindowsSharedMemoryHandle& handle, const gfxIntSize& size, const bool& transparent) { #ifdef XP_WIN return new PluginSurfaceParent(handle, size, transparent); #else NS_ERROR("This shouldn't be called!"); return NULL; #endif } bool PluginInstanceParent::DeallocPPluginSurface(PPluginSurfaceParent* s) { #ifdef XP_WIN delete s; return true; #else return false; #endif } bool PluginInstanceParent::AnswerNPN_PushPopupsEnabledState(const bool& aState) { mNPNIface->pushpopupsenabledstate(mNPP, aState ? 1 : 0); return true; } bool PluginInstanceParent::AnswerNPN_PopPopupsEnabledState() { mNPNIface->poppopupsenabledstate(mNPP); return true; } bool PluginInstanceParent::AnswerNPN_GetValueForURL(const NPNURLVariable& variable, const nsCString& url, nsCString* value, NPError* result) { char* v; uint32_t len; *result = mNPNIface->getvalueforurl(mNPP, (NPNURLVariable) variable, url.get(), &v, &len); if (NPERR_NO_ERROR == *result) value->Adopt(v, len); return true; } bool PluginInstanceParent::AnswerNPN_SetValueForURL(const NPNURLVariable& variable, const nsCString& url, const nsCString& value, NPError* result) { *result = mNPNIface->setvalueforurl(mNPP, (NPNURLVariable) variable, url.get(), value.get(), value.Length()); return true; } bool PluginInstanceParent::AnswerNPN_GetAuthenticationInfo(const nsCString& protocol, const nsCString& host, const int32_t& port, const nsCString& scheme, const nsCString& realm, nsCString* username, nsCString* password, NPError* result) { char* u; uint32_t ulen; char* p; uint32_t plen; *result = mNPNIface->getauthenticationinfo(mNPP, protocol.get(), host.get(), port, scheme.get(), realm.get(), &u, &ulen, &p, &plen); if (NPERR_NO_ERROR == *result) { username->Adopt(u, ulen); password->Adopt(p, plen); } return true; } bool PluginInstanceParent::AnswerNPN_ConvertPoint(const double& sourceX, const bool& ignoreDestX, const double& sourceY, const bool& ignoreDestY, const NPCoordinateSpace& sourceSpace, const NPCoordinateSpace& destSpace, double *destX, double *destY, bool *result) { *result = mNPNIface->convertpoint(mNPP, sourceX, sourceY, sourceSpace, ignoreDestX ? nullptr : destX, ignoreDestY ? nullptr : destY, destSpace); return true; } bool PluginInstanceParent::AnswerNPN_InitAsyncSurface(const gfxIntSize& size, const NPImageFormat& format, NPRemoteAsyncSurface* surfData, bool* result) { if (!IsAsyncDrawing()) { *result = false; return true; } switch (mDrawingModel) { case NPDrawingModelAsyncBitmapSurface: { Shmem sharedMem; if (!AllocUnsafeShmem(size.width * size.height * 4, SharedMemory::TYPE_BASIC, &sharedMem)) { *result = false; return true; } surfData->size() = size; surfData->hostPtr() = (uintptr_t)sharedMem.get(); surfData->stride() = size.width * 4; surfData->format() = format; surfData->data() = sharedMem; *result = true; return true; } #ifdef XP_WIN case NPDrawingModelAsyncWindowsDXGISurface: { ID3D10Device1 *device = gfxWindowsPlatform::GetPlatform()->GetD3D10Device(); nsRefPtr texture; CD3D10_TEXTURE2D_DESC desc(DXGI_FORMAT_B8G8R8A8_UNORM, size.width, size.height, 1, 1); desc.MiscFlags = D3D10_RESOURCE_MISC_SHARED_KEYEDMUTEX; desc.BindFlags = D3D10_BIND_RENDER_TARGET | D3D10_BIND_SHADER_RESOURCE; if (FAILED(device->CreateTexture2D(&desc, NULL, getter_AddRefs(texture)))) { *result = false; return true; } nsRefPtr resource; if (FAILED(texture->QueryInterface(IID_IDXGIResource, getter_AddRefs(resource)))) { *result = false; return true; } HANDLE sharedHandle; if (FAILED(resource->GetSharedHandle(&sharedHandle))) { *result = false; return true; } surfData->size() = size; surfData->data() = sharedHandle; surfData->format() = format; mTextureMap.Put(sharedHandle, texture); *result = true; } #endif } return true; } bool PluginInstanceParent::RecvRedrawPlugin() { nsNPAPIPluginInstance *inst = static_cast(mNPP->ndata); if (!inst) { return false; } inst->RedrawPlugin(); return true; } bool PluginInstanceParent::RecvNegotiatedCarbon() { nsNPAPIPluginInstance *inst = static_cast(mNPP->ndata); if (!inst) { return false; } inst->CarbonNPAPIFailure(); return true; } bool PluginInstanceParent::RecvReleaseDXGISharedSurface(const DXGISharedSurfaceHandle &aHandle) { #ifdef XP_WIN mTextureMap.Remove(aHandle); #endif return true; } #if defined(OS_WIN) /* plugin focus changes between processes focus from dom -> child: Focus manager calls on widget to set the focus on the window. We pick up the resulting wm_setfocus event here, and forward that over ipc to the child which calls set focus on itself. focus from child -> focus manager: Child picks up the local wm_setfocus and sends it via ipc over here. We then post a custom event to widget/windows/nswindow which fires off a gui event letting the browser know. */ static const PRUnichar kPluginInstanceParentProperty[] = L"PluginInstanceParentProperty"; // static LRESULT CALLBACK PluginInstanceParent::PluginWindowHookProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { PluginInstanceParent* self = reinterpret_cast( ::GetPropW(hWnd, kPluginInstanceParentProperty)); if (!self) { NS_NOTREACHED("PluginInstanceParent::PluginWindowHookProc null this ptr!"); return DefWindowProc(hWnd, message, wParam, lParam); } NS_ASSERTION(self->mPluginHWND == hWnd, "Wrong window!"); switch (message) { case WM_SETFOCUS: // Let the child plugin window know it should take focus. unused << self->CallSetPluginFocus(); break; case WM_CLOSE: self->UnsubclassPluginWindow(); break; } if (self->mPluginWndProc == PluginWindowHookProc) { NS_NOTREACHED( "PluginWindowHookProc invoking mPluginWndProc w/" "mPluginWndProc == PluginWindowHookProc????"); return DefWindowProc(hWnd, message, wParam, lParam); } return ::CallWindowProc(self->mPluginWndProc, hWnd, message, wParam, lParam); } void PluginInstanceParent::SubclassPluginWindow(HWND aWnd) { NS_ASSERTION(!(mPluginHWND && aWnd != mPluginHWND), "PluginInstanceParent::SubclassPluginWindow hwnd is not our window!"); if (!mPluginHWND) { mPluginHWND = aWnd; mPluginWndProc = (WNDPROC)::SetWindowLongPtrA(mPluginHWND, GWLP_WNDPROC, reinterpret_cast(PluginWindowHookProc)); DebugOnly bRes = ::SetPropW(mPluginHWND, kPluginInstanceParentProperty, this); NS_ASSERTION(mPluginWndProc, "PluginInstanceParent::SubclassPluginWindow failed to set subclass!"); NS_ASSERTION(bRes, "PluginInstanceParent::SubclassPluginWindow failed to set prop!"); } } void PluginInstanceParent::UnsubclassPluginWindow() { if (mPluginHWND && mPluginWndProc) { ::SetWindowLongPtrA(mPluginHWND, GWLP_WNDPROC, reinterpret_cast(mPluginWndProc)); ::RemovePropW(mPluginHWND, kPluginInstanceParentProperty); mPluginWndProc = NULL; mPluginHWND = NULL; } } /* windowless drawing helpers */ /* * Origin info: * * windowless, offscreen: * * WM_WINDOWPOSCHANGED: origin is relative to container * setwindow: origin is 0,0 * WM_PAINT: origin is 0,0 * * windowless, native: * * WM_WINDOWPOSCHANGED: origin is relative to container * setwindow: origin is relative to container * WM_PAINT: origin is relative to container * * PluginInstanceParent: * * painting: mPluginPort (nsIntRect, saved in SetWindow) */ void PluginInstanceParent::SharedSurfaceRelease() { mSharedSurfaceDib.Close(); } bool PluginInstanceParent::SharedSurfaceSetWindow(const NPWindow* aWindow, NPRemoteWindow& aRemoteWindow) { aRemoteWindow.window = 0; aRemoteWindow.x = aWindow->x; aRemoteWindow.y = aWindow->y; aRemoteWindow.width = aWindow->width; aRemoteWindow.height = aWindow->height; aRemoteWindow.type = aWindow->type; nsIntRect newPort(aWindow->x, aWindow->y, aWindow->width, aWindow->height); // save the the rect location within the browser window. mPluginPort = newPort; // move the port to our shared surface origin newPort.MoveTo(0,0); // check to see if we have the room in shared surface if (mSharedSurfaceDib.IsValid() && mSharedSize.Contains(newPort)) { // ok to paint aRemoteWindow.surfaceHandle = 0; return true; } // allocate a new shared surface SharedSurfaceRelease(); if (NS_FAILED(mSharedSurfaceDib.Create(reinterpret_cast(aWindow->window), newPort.width, newPort.height, false))) return false; // save the new shared surface size we just allocated mSharedSize = newPort; base::SharedMemoryHandle handle; if (NS_FAILED(mSharedSurfaceDib.ShareToProcess(mParent->ChildProcessHandle(), &handle))) return false; aRemoteWindow.surfaceHandle = handle; return true; } void PluginInstanceParent::SharedSurfaceBeforePaint(RECT& rect, NPRemoteEvent& npremoteevent) { RECT* dr = (RECT*)npremoteevent.event.lParam; HDC parentHdc = (HDC)npremoteevent.event.wParam; nsIntRect dirtyRect(dr->left, dr->top, dr->right-dr->left, dr->bottom-dr->top); dirtyRect.MoveBy(-mPluginPort.x, -mPluginPort.y); // should always be smaller than dirtyRect ::BitBlt(mSharedSurfaceDib.GetHDC(), dirtyRect.x, dirtyRect.y, dirtyRect.width, dirtyRect.height, parentHdc, dr->left, dr->top, SRCCOPY); // setup the translated dirty rect we'll send to the child rect.left = dirtyRect.x; rect.top = dirtyRect.y; rect.right = dirtyRect.x + dirtyRect.width; rect.bottom = dirtyRect.y + dirtyRect.height; npremoteevent.event.wParam = WPARAM(0); npremoteevent.event.lParam = LPARAM(&rect); } void PluginInstanceParent::SharedSurfaceAfterPaint(NPEvent* npevent) { RECT* dr = (RECT*)npevent->lParam; HDC parentHdc = (HDC)npevent->wParam; nsIntRect dirtyRect(dr->left, dr->top, dr->right-dr->left, dr->bottom-dr->top); dirtyRect.MoveBy(-mPluginPort.x, -mPluginPort.y); // src copy the shared dib into the parent surface we are handed. ::BitBlt(parentHdc, dr->left, dr->top, dirtyRect.width, dirtyRect.height, mSharedSurfaceDib.GetHDC(), dirtyRect.x, dirtyRect.y, SRCCOPY); } #endif // defined(OS_WIN) bool PluginInstanceParent::AnswerPluginFocusChange(const bool& gotFocus) { PLUGIN_LOG_DEBUG(("%s", FULLFUNCTION)); // Currently only in use on windows - an rpc event we receive from the // child when it's plugin window (or one of it's children) receives keyboard // focus. We forward the event down to widget so the dom/focus manager can // be updated. #if defined(OS_WIN) ::SendMessage(mPluginHWND, gOOPPPluginFocusEvent, gotFocus ? 1 : 0, 0); return true; #else NS_NOTREACHED("PluginInstanceParent::AnswerPluginFocusChange not implemented!"); return false; #endif }