2010-05-24 08:28:51 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is Mozilla Corporation code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Mozilla Foundation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Bas Schouten <bschouten@mozilla.com>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
#include "LayerManagerD3D9.h"
|
|
|
|
|
|
|
|
#include "ThebesLayerD3D9.h"
|
|
|
|
#include "ContainerLayerD3D9.h"
|
|
|
|
#include "ImageLayerD3D9.h"
|
|
|
|
#include "ColorLayerD3D9.h"
|
|
|
|
#include "CanvasLayerD3D9.h"
|
2010-08-26 13:44:53 -07:00
|
|
|
#include "nsIServiceManager.h"
|
|
|
|
#include "nsIPrefService.h"
|
2010-09-07 20:27:36 -07:00
|
|
|
#include "gfxWindowsPlatform.h"
|
2010-09-16 09:45:19 -07:00
|
|
|
#include "nsIGfxInfo.h"
|
|
|
|
|
2010-09-07 20:27:36 -07:00
|
|
|
#ifdef CAIRO_HAS_D2D_SURFACE
|
|
|
|
#include "gfxD2DSurface.h"
|
|
|
|
#endif
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace layers {
|
|
|
|
|
2010-09-15 15:15:56 -07:00
|
|
|
DeviceManagerD3D9 *LayerManagerD3D9::mDefaultDeviceManager = nsnull;
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
LayerManagerD3D9::LayerManagerD3D9(nsIWidget *aWidget)
|
2010-08-26 13:44:53 -07:00
|
|
|
: mIs3DEnabled(PR_FALSE)
|
2010-05-24 08:28:51 -07:00
|
|
|
{
|
2010-09-15 15:15:56 -07:00
|
|
|
mWidget = aWidget;
|
|
|
|
mCurrentCallbackInfo.Callback = NULL;
|
|
|
|
mCurrentCallbackInfo.CallbackData = NULL;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
LayerManagerD3D9::~LayerManagerD3D9()
|
|
|
|
{
|
2010-09-15 15:15:49 -07:00
|
|
|
Destroy();
|
2010-08-10 16:32:45 -07:00
|
|
|
}
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
PRBool
|
|
|
|
LayerManagerD3D9::Initialize()
|
|
|
|
{
|
2010-08-26 13:44:53 -07:00
|
|
|
/* Check the user preference for whether 3d video is enabled or not */
|
|
|
|
nsCOMPtr<nsIPrefBranch> prefs = do_GetService(NS_PREFSERVICE_CONTRACTID);
|
2010-09-16 09:45:19 -07:00
|
|
|
prefs->GetBoolPref("gfx.3d_video.enabled", &mIs3DEnabled);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIGfxInfo> gfxInfo = do_GetService("@mozilla.org/gfx/info;1");
|
|
|
|
if (gfxInfo) {
|
|
|
|
PRInt32 status;
|
|
|
|
if (NS_SUCCEEDED(gfxInfo->GetFeatureStatus(nsIGfxInfo::FEATURE_DIRECT3D_9_LAYERS, &status))) {
|
2010-10-06 21:40:08 -07:00
|
|
|
if (status != nsIGfxInfo::FEATURE_NO_INFO)
|
2010-09-16 09:45:19 -07:00
|
|
|
{
|
|
|
|
NS_WARNING("Direct3D 9-accelerated layers are not supported on this system.");
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-26 13:44:53 -07:00
|
|
|
|
2010-09-15 15:15:56 -07:00
|
|
|
if (!mDefaultDeviceManager) {
|
2010-08-10 16:32:45 -07:00
|
|
|
mDeviceManager = new DeviceManagerD3D9;
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
if (!mDeviceManager->Init()) {
|
|
|
|
mDeviceManager = nsnull;
|
2010-05-24 08:28:51 -07:00
|
|
|
return PR_FALSE;
|
|
|
|
}
|
2010-09-15 15:15:56 -07:00
|
|
|
|
|
|
|
mDefaultDeviceManager = mDeviceManager;
|
2010-09-03 20:21:26 -07:00
|
|
|
} else {
|
2010-09-15 15:15:56 -07:00
|
|
|
mDeviceManager = mDefaultDeviceManager;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
mSwapChain = mDeviceManager->
|
|
|
|
CreateSwapChain((HWND)mWidget->GetNativeData(NS_NATIVE_WINDOW));
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
if (!mSwapChain) {
|
2010-05-24 08:28:51 -07:00
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::SetClippingRegion(const nsIntRegion &aClippingRegion)
|
|
|
|
{
|
|
|
|
mClippingRegion = aClippingRegion;
|
|
|
|
}
|
|
|
|
|
2010-09-15 15:15:49 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::Destroy()
|
|
|
|
{
|
|
|
|
if (!IsDestroyed()) {
|
|
|
|
if (mRoot) {
|
|
|
|
static_cast<LayerD3D9*>(mRoot->ImplData())->LayerManagerDestroyed();
|
|
|
|
}
|
|
|
|
/* Important to release this first since it also holds a reference to the
|
|
|
|
* device manager
|
|
|
|
*/
|
|
|
|
mSwapChain = nsnull;
|
2010-09-15 15:15:56 -07:00
|
|
|
mDeviceManager = nsnull;
|
2010-09-15 15:15:49 -07:00
|
|
|
}
|
|
|
|
LayerManager::Destroy();
|
|
|
|
}
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::BeginTransaction()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::BeginTransactionWithTarget(gfxContext *aTarget)
|
|
|
|
{
|
|
|
|
mTarget = aTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::EndConstruction()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::EndTransaction(DrawThebesLayerCallback aCallback,
|
|
|
|
void* aCallbackData)
|
|
|
|
{
|
|
|
|
mCurrentCallbackInfo.Callback = aCallback;
|
|
|
|
mCurrentCallbackInfo.CallbackData = aCallbackData;
|
2010-11-08 01:06:15 -08:00
|
|
|
|
|
|
|
// The results of our drawing always go directly into a pixel buffer,
|
|
|
|
// so we don't need to pass any global transform here.
|
|
|
|
mRoot->ComputeEffectiveTransforms(gfx3DMatrix());
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
Render();
|
|
|
|
/* Clean this out for sanity */
|
|
|
|
mCurrentCallbackInfo.Callback = NULL;
|
|
|
|
mCurrentCallbackInfo.CallbackData = NULL;
|
2010-07-13 15:19:43 -07:00
|
|
|
// Clear mTarget, next transaction could have no target
|
|
|
|
mTarget = NULL;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::SetRoot(Layer *aLayer)
|
|
|
|
{
|
2010-09-02 20:50:29 -07:00
|
|
|
mRoot = aLayer;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ThebesLayer>
|
|
|
|
LayerManagerD3D9::CreateThebesLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ThebesLayer> layer = new ThebesLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ContainerLayer>
|
|
|
|
LayerManagerD3D9::CreateContainerLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ContainerLayer> layer = new ContainerLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ImageLayer>
|
|
|
|
LayerManagerD3D9::CreateImageLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ImageLayer> layer = new ImageLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ColorLayer>
|
|
|
|
LayerManagerD3D9::CreateColorLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ColorLayer> layer = new ColorLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<CanvasLayer>
|
|
|
|
LayerManagerD3D9::CreateCanvasLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<CanvasLayer> layer = new CanvasLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<ImageContainer>
|
|
|
|
LayerManagerD3D9::CreateImageContainer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ImageContainer> container = new ImageContainerD3D9(this);
|
|
|
|
return container.forget();
|
|
|
|
}
|
|
|
|
|
2010-09-07 20:27:36 -07:00
|
|
|
cairo_user_data_key_t gKeyD3D9Texture;
|
|
|
|
|
|
|
|
void ReleaseTexture(void *texture)
|
|
|
|
{
|
|
|
|
static_cast<IDirect3DTexture9*>(texture)->Release();
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<gfxASurface>
|
|
|
|
LayerManagerD3D9::CreateOptimalSurface(const gfxIntSize &aSize,
|
|
|
|
gfxASurface::gfxImageFormat aFormat)
|
|
|
|
{
|
|
|
|
#ifdef CAIRO_HAS_D2D_SURFACE
|
|
|
|
if ((aFormat != gfxASurface::ImageFormatRGB24 &&
|
|
|
|
aFormat != gfxASurface::ImageFormatARGB32) ||
|
|
|
|
gfxWindowsPlatform::GetPlatform()->GetRenderMode() !=
|
|
|
|
gfxWindowsPlatform::RENDER_DIRECT2D ||
|
|
|
|
!deviceManager()->IsD3D9Ex()) {
|
|
|
|
return LayerManager::CreateOptimalSurface(aSize, aFormat);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsRefPtr<IDirect3DTexture9> texture;
|
|
|
|
|
|
|
|
HANDLE sharedHandle = 0;
|
|
|
|
device()->CreateTexture(aSize.width, aSize.height, 1,
|
|
|
|
D3DUSAGE_RENDERTARGET, D3DFMT_A8R8G8B8,
|
|
|
|
D3DPOOL_DEFAULT, getter_AddRefs(texture), &sharedHandle);
|
|
|
|
|
|
|
|
nsRefPtr<gfxD2DSurface> surface =
|
|
|
|
new gfxD2DSurface(sharedHandle, aFormat == gfxASurface::ImageFormatRGB24 ?
|
|
|
|
gfxASurface::CONTENT_COLOR : gfxASurface::CONTENT_COLOR_ALPHA);
|
|
|
|
|
|
|
|
if (!surface || surface->CairoStatus()) {
|
|
|
|
return LayerManager::CreateOptimalSurface(aSize, aFormat);
|
|
|
|
}
|
|
|
|
|
|
|
|
surface->SetData(&gKeyD3D9Texture,
|
|
|
|
texture.forget().get(),
|
|
|
|
ReleaseTexture);
|
|
|
|
|
|
|
|
return surface.forget();
|
|
|
|
#else
|
|
|
|
return LayerManager::CreateOptimalSurface(aSize, aFormat);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2010-10-08 09:02:53 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::ReportFailure(const nsACString &aMsg, HRESULT aCode)
|
|
|
|
{
|
|
|
|
// We could choose to abort here when hr == E_OUTOFMEMORY.
|
|
|
|
nsCString msg;
|
|
|
|
msg.Append(aMsg);
|
|
|
|
msg.AppendLiteral(" Error code: ");
|
2010-11-06 07:24:19 -07:00
|
|
|
msg.AppendInt(PRUint32(aCode));
|
2010-10-08 09:02:53 -07:00
|
|
|
NS_WARNING(msg.BeginReading());
|
|
|
|
}
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::Render()
|
|
|
|
{
|
2010-08-10 16:32:45 -07:00
|
|
|
if (!mSwapChain->PrepareForRendering()) {
|
2010-05-24 08:28:51 -07:00
|
|
|
return;
|
|
|
|
}
|
2010-08-10 16:32:45 -07:00
|
|
|
deviceManager()->SetupRenderState();
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
SetupPipeline();
|
|
|
|
nsIntRect rect;
|
2010-07-13 15:19:41 -07:00
|
|
|
mWidget->GetClientBounds(rect);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->Clear(0, NULL, D3DCLEAR_TARGET, 0x00000000, 0, 0);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->BeginScene();
|
2010-05-28 20:27:03 -07:00
|
|
|
|
2010-09-02 20:50:29 -07:00
|
|
|
if (mRoot) {
|
|
|
|
const nsIntRect *clipRect = mRoot->GetClipRect();
|
2010-05-24 08:28:51 -07:00
|
|
|
RECT r;
|
|
|
|
if (clipRect) {
|
|
|
|
r.left = (LONG)clipRect->x;
|
|
|
|
r.top = (LONG)clipRect->y;
|
|
|
|
r.right = (LONG)(clipRect->x + clipRect->width);
|
|
|
|
r.bottom = (LONG)(clipRect->y + clipRect->height);
|
|
|
|
} else {
|
|
|
|
r.left = r.top = 0;
|
|
|
|
r.right = rect.width;
|
|
|
|
r.bottom = rect.height;
|
|
|
|
}
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->SetScissorRect(&r);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-11-08 01:06:15 -08:00
|
|
|
static_cast<LayerD3D9*>(mRoot->ImplData())->RenderLayer();
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->EndScene();
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
if (!mTarget) {
|
|
|
|
const nsIntRect *r;
|
|
|
|
for (nsIntRegionRectIterator iter(mClippingRegion);
|
|
|
|
(r = iter.Next()) != nsnull;) {
|
2010-08-10 16:32:45 -07:00
|
|
|
mSwapChain->Present(*r);
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
PaintToTarget();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::SetupPipeline()
|
|
|
|
{
|
|
|
|
nsIntRect rect;
|
2010-07-13 15:19:41 -07:00
|
|
|
mWidget->GetClientBounds(rect);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-10-01 15:24:58 -07:00
|
|
|
gfx3DMatrix viewMatrix;
|
2010-05-24 08:28:51 -07:00
|
|
|
/*
|
2010-05-28 20:27:03 -07:00
|
|
|
* Matrix to transform to viewport space ( <-1.0, 1.0> topleft,
|
2010-05-24 08:28:51 -07:00
|
|
|
* <1.0, -1.0> bottomright)
|
|
|
|
*/
|
2010-10-01 15:24:58 -07:00
|
|
|
viewMatrix._11 = 2.0f / rect.width;
|
|
|
|
viewMatrix._22 = -2.0f / rect.height;
|
|
|
|
viewMatrix._41 = -1.0f;
|
|
|
|
viewMatrix._42 = 1.0f;
|
2010-05-28 20:27:03 -07:00
|
|
|
|
2010-10-01 15:24:58 -07:00
|
|
|
HRESULT hr = device()->SetVertexShaderConstantF(CBmProjection,
|
|
|
|
&viewMatrix._11, 4);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
NS_WARNING("Failed to set projection shader constant!");
|
|
|
|
}
|
2010-09-07 20:27:38 -07:00
|
|
|
|
2010-10-01 15:24:58 -07:00
|
|
|
hr = device()->SetVertexShaderConstantF(CBvTextureCoords,
|
|
|
|
ShaderConstantRect(0, 0, 1.0f, 1.0f),
|
|
|
|
1);
|
2010-09-07 20:27:38 -07:00
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
NS_WARNING("Failed to set texCoords shader constant!");
|
|
|
|
}
|
2010-10-01 15:25:00 -07:00
|
|
|
|
|
|
|
float offset[] = { 0, 0, 0, 0 };
|
|
|
|
hr = device()->SetVertexShaderConstantF(CBvRenderTargetOffset, offset, 1);
|
|
|
|
|
|
|
|
if (FAILED(hr)) {
|
|
|
|
NS_WARNING("Failed to set RenderTargetOffset shader constant!");
|
|
|
|
}
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::PaintToTarget()
|
|
|
|
{
|
|
|
|
nsRefPtr<IDirect3DSurface9> backBuff;
|
|
|
|
nsRefPtr<IDirect3DSurface9> destSurf;
|
2010-08-23 20:56:49 -07:00
|
|
|
device()->GetRenderTarget(0, getter_AddRefs(backBuff));
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
D3DSURFACE_DESC desc;
|
|
|
|
backBuff->GetDesc(&desc);
|
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->CreateOffscreenPlainSurface(desc.Width, desc.Height,
|
2010-05-24 08:28:51 -07:00
|
|
|
D3DFMT_A8R8G8B8, D3DPOOL_SYSTEMMEM,
|
|
|
|
getter_AddRefs(destSurf), NULL);
|
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->GetRenderTargetData(backBuff, destSurf);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
D3DLOCKED_RECT rect;
|
|
|
|
destSurf->LockRect(&rect, NULL, D3DLOCK_READONLY);
|
|
|
|
|
|
|
|
nsRefPtr<gfxImageSurface> imageSurface =
|
|
|
|
new gfxImageSurface((unsigned char*)rect.pBits,
|
|
|
|
gfxIntSize(desc.Width, desc.Height),
|
|
|
|
rect.Pitch,
|
|
|
|
gfxASurface::ImageFormatARGB32);
|
|
|
|
|
|
|
|
mTarget->SetSource(imageSurface);
|
|
|
|
mTarget->SetOperator(gfxContext::OPERATOR_OVER);
|
|
|
|
mTarget->Paint();
|
|
|
|
destSurf->UnlockRect();
|
|
|
|
}
|
|
|
|
|
|
|
|
LayerD3D9::LayerD3D9(LayerManagerD3D9 *aManager)
|
|
|
|
: mD3DManager(aManager)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
} /* namespace layers */
|
|
|
|
} /* namespace mozilla */
|