2010-05-24 08:28:51 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
2012-05-21 04:12:37 -07:00
|
|
|
* 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/. */
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
#include "LayerManagerD3D9.h"
|
|
|
|
|
|
|
|
#include "ThebesLayerD3D9.h"
|
|
|
|
#include "ContainerLayerD3D9.h"
|
|
|
|
#include "ImageLayerD3D9.h"
|
|
|
|
#include "ColorLayerD3D9.h"
|
|
|
|
#include "CanvasLayerD3D9.h"
|
2011-02-16 14:43:30 -08:00
|
|
|
#include "ReadbackLayerD3D9.h"
|
2010-09-07 20:27:36 -07:00
|
|
|
#include "gfxWindowsPlatform.h"
|
2010-09-16 09:45:19 -07:00
|
|
|
#include "nsIGfxInfo.h"
|
2010-12-21 02:52:00 -08:00
|
|
|
#include "nsServiceManagerUtils.h"
|
2011-02-09 11:28:39 -08:00
|
|
|
#include "gfxFailure.h"
|
2014-02-26 18:53:27 -08:00
|
|
|
#include "gfxPrefs.h"
|
2010-09-16 09:45:19 -07:00
|
|
|
|
2011-03-02 12:50:36 -08:00
|
|
|
#include "gfxCrashReporterUtils.h"
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
namespace mozilla {
|
|
|
|
namespace layers {
|
|
|
|
|
|
|
|
LayerManagerD3D9::LayerManagerD3D9(nsIWidget *aWidget)
|
2011-01-19 00:27:54 -08:00
|
|
|
: mWidget(aWidget)
|
|
|
|
, mDeviceResetCount(0)
|
2010-05-24 08:28:51 -07:00
|
|
|
{
|
2013-07-20 01:48:55 -07:00
|
|
|
mCurrentCallbackInfo.Callback = nullptr;
|
|
|
|
mCurrentCallbackInfo.CallbackData = nullptr;
|
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
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2012-02-27 13:33:19 -08:00
|
|
|
LayerManagerD3D9::Initialize(bool force)
|
2010-05-24 08:28:51 -07:00
|
|
|
{
|
2012-02-27 13:33:19 -08:00
|
|
|
ScopedGfxFeatureReporter reporter("D3D9 Layers", force);
|
2011-03-02 12:50:36 -08:00
|
|
|
|
2011-01-18 12:11:19 -08:00
|
|
|
/* XXX: this preference and blacklist code should move out of the layer manager */
|
2014-02-26 18:53:27 -08:00
|
|
|
bool forceAccelerate = gfxPrefs::LayersAccelerationForceEnabled();
|
2011-01-11 14:01:37 -08:00
|
|
|
|
2010-09-16 09:45:19 -07:00
|
|
|
nsCOMPtr<nsIGfxInfo> gfxInfo = do_GetService("@mozilla.org/gfx/info;1");
|
|
|
|
if (gfxInfo) {
|
2012-08-22 08:56:38 -07:00
|
|
|
int32_t status;
|
2010-09-16 09:45:19 -07:00
|
|
|
if (NS_SUCCEEDED(gfxInfo->GetFeatureStatus(nsIGfxInfo::FEATURE_DIRECT3D_9_LAYERS, &status))) {
|
2011-01-11 14:01:37 -08:00
|
|
|
if (status != nsIGfxInfo::FEATURE_NO_INFO && !forceAccelerate)
|
2010-09-16 09:45:19 -07:00
|
|
|
{
|
|
|
|
NS_WARNING("Direct3D 9-accelerated layers are not supported on this system.");
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2010-09-16 09:45:19 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-26 13:44:53 -07:00
|
|
|
|
2014-01-26 20:10:40 -08:00
|
|
|
mDeviceManager = gfxWindowsPlatform::GetPlatform()->GetD3D9DeviceManager();
|
|
|
|
if (!mDeviceManager) {
|
|
|
|
return false;
|
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) {
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
2011-03-02 12:50:36 -08:00
|
|
|
reporter.SetSuccessful();
|
2011-10-17 07:59:28 -07:00
|
|
|
return true;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
*/
|
2012-07-30 07:20:58 -07:00
|
|
|
mSwapChain = nullptr;
|
|
|
|
mDeviceManager = nullptr;
|
2010-09-15 15:15:49 -07:00
|
|
|
}
|
|
|
|
LayerManager::Destroy();
|
|
|
|
}
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::BeginTransaction()
|
|
|
|
{
|
2012-08-15 14:55:38 -07:00
|
|
|
mInTransaction = true;
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::BeginTransactionWithTarget(gfxContext *aTarget)
|
|
|
|
{
|
2012-08-15 14:55:38 -07:00
|
|
|
mInTransaction = true;
|
2010-05-24 08:28:51 -07:00
|
|
|
mTarget = aTarget;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
LayerManagerD3D9::EndConstruction()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-01-19 00:27:54 -08:00
|
|
|
bool
|
2012-08-13 03:10:10 -07:00
|
|
|
LayerManagerD3D9::EndEmptyTransaction(EndTransactionFlags aFlags)
|
2011-01-19 00:27:54 -08:00
|
|
|
{
|
2012-08-15 14:55:38 -07:00
|
|
|
mInTransaction = false;
|
|
|
|
|
2011-01-19 00:27:54 -08:00
|
|
|
// If the device reset count from our last EndTransaction doesn't match
|
|
|
|
// the current device reset count, the device must have been reset one or
|
|
|
|
// more times since our last transaction. In that case, an empty transaction
|
|
|
|
// is not possible, because layers may need to be rerendered.
|
|
|
|
if (!mRoot || mDeviceResetCount != mDeviceManager->GetDeviceResetCount())
|
|
|
|
return false;
|
|
|
|
|
2012-08-13 03:10:10 -07:00
|
|
|
EndTransaction(nullptr, nullptr, aFlags);
|
2011-01-19 00:27:54 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-12-13 16:14:07 -08:00
|
|
|
void
|
2010-05-24 08:28:51 -07:00
|
|
|
LayerManagerD3D9::EndTransaction(DrawThebesLayerCallback aCallback,
|
2011-09-26 06:20:42 -07:00
|
|
|
void* aCallbackData,
|
|
|
|
EndTransactionFlags aFlags)
|
2010-05-24 08:28:51 -07:00
|
|
|
{
|
2012-08-15 14:55:38 -07:00
|
|
|
mInTransaction = false;
|
|
|
|
|
2011-01-19 00:27:54 -08:00
|
|
|
mDeviceResetCount = mDeviceManager->GetDeviceResetCount();
|
|
|
|
|
2011-09-26 06:20:42 -07:00
|
|
|
if (mRoot && !(aFlags & END_NO_IMMEDIATE_REDRAW)) {
|
2011-02-10 00:58:11 -08:00
|
|
|
mCurrentCallbackInfo.Callback = aCallback;
|
|
|
|
mCurrentCallbackInfo.CallbackData = aCallbackData;
|
2010-11-08 01:06:15 -08:00
|
|
|
|
2012-10-02 22:55:50 -07:00
|
|
|
if (aFlags & END_NO_COMPOSITE) {
|
|
|
|
// Apply pending tree updates before recomputing effective
|
|
|
|
// properties.
|
|
|
|
mRoot->ApplyPendingUpdatesToSubtree();
|
|
|
|
}
|
|
|
|
|
2011-02-10 00:58:11 -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.
|
2014-01-27 07:28:04 -08:00
|
|
|
mRoot->ComputeEffectiveTransforms(gfx::Matrix4x4());
|
2011-02-10 00:58:11 -08:00
|
|
|
|
2012-08-13 03:10:10 -07:00
|
|
|
SetCompositingDisabled(aFlags & END_NO_COMPOSITE);
|
2011-02-10 00:58:11 -08:00
|
|
|
Render();
|
|
|
|
/* Clean this out for sanity */
|
2013-07-20 01:48:55 -07:00
|
|
|
mCurrentCallbackInfo.Callback = nullptr;
|
|
|
|
mCurrentCallbackInfo.CallbackData = nullptr;
|
2011-02-10 00:58:11 -08:00
|
|
|
}
|
2010-11-08 01:06:15 -08:00
|
|
|
|
2010-07-13 15:19:43 -07:00
|
|
|
// Clear mTarget, next transaction could have no target
|
2013-07-20 01:48:55 -07:00
|
|
|
mTarget = nullptr;
|
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();
|
|
|
|
}
|
|
|
|
|
2011-02-16 14:43:30 -08:00
|
|
|
already_AddRefed<ReadbackLayer>
|
|
|
|
LayerManagerD3D9::CreateReadbackLayer()
|
|
|
|
{
|
|
|
|
nsRefPtr<ReadbackLayer> layer = new ReadbackLayerD3D9(this);
|
|
|
|
return layer.forget();
|
|
|
|
}
|
|
|
|
|
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: ");
|
2012-08-22 08:56:38 -07:00
|
|
|
msg.AppendInt(uint32_t(aCode));
|
2010-10-08 09:02:53 -07:00
|
|
|
NS_WARNING(msg.BeginReading());
|
2011-02-09 11:28:39 -08:00
|
|
|
|
|
|
|
gfx::LogFailure(msg);
|
2010-10-08 09:02:53 -07:00
|
|
|
}
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
void
|
|
|
|
LayerManagerD3D9::Render()
|
|
|
|
{
|
2014-01-26 20:10:40 -08:00
|
|
|
if (mSwapChain->PrepareForRendering() != DeviceOK) {
|
2010-05-24 08:28:51 -07:00
|
|
|
return;
|
|
|
|
}
|
2014-01-26 20:10:40 -08:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
deviceManager()->SetupRenderState();
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
SetupPipeline();
|
2012-08-13 03:10:10 -07:00
|
|
|
|
|
|
|
if (CompositingDisabled()) {
|
|
|
|
static_cast<LayerD3D9*>(mRoot->ImplData())->RenderLayer();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-05-24 08:28:51 -07:00
|
|
|
nsIntRect rect;
|
2010-07-13 15:19:41 -07:00
|
|
|
mWidget->GetClientBounds(rect);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2013-07-20 01:48:55 -07:00
|
|
|
device()->Clear(0, nullptr, 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
|
|
|
|
2011-02-10 00:58:11 -08:00
|
|
|
const nsIntRect *clipRect = mRoot->GetClipRect();
|
|
|
|
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-05-24 08:28:51 -07:00
|
|
|
}
|
2011-02-10 00:58:11 -08:00
|
|
|
device()->SetScissorRect(&r);
|
|
|
|
|
|
|
|
static_cast<LayerD3D9*>(mRoot->ImplData())->RenderLayer();
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2014-02-12 16:20:41 -08:00
|
|
|
if (!mRegionToClear.IsEmpty()) {
|
|
|
|
D3DRECT* rects = new D3DRECT[mRegionToClear.GetNumRects()];
|
|
|
|
nsIntRegionRectIterator iter(mRegionToClear);
|
|
|
|
const nsIntRect *r;
|
|
|
|
size_t i = 0;
|
|
|
|
while ((r = iter.Next())) {
|
|
|
|
rects[i].x1 = r->x;
|
|
|
|
rects[i].y1 = r->y;
|
|
|
|
rects[i].x2 = r->x + r->width;
|
|
|
|
rects[i].y2 = r->y + r->height;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
device()->Clear(i, rects, D3DCLEAR_TARGET,
|
|
|
|
0x00000000, 0, 0);
|
|
|
|
|
|
|
|
delete [] rects;
|
|
|
|
}
|
|
|
|
|
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);
|
2012-07-30 07:20:58 -07:00
|
|
|
(r = iter.Next()) != nullptr;) {
|
2010-08-10 16:32:45 -07:00
|
|
|
mSwapChain->Present(*r);
|
2010-05-24 08:28:51 -07:00
|
|
|
}
|
2013-11-26 23:32:19 -08:00
|
|
|
RecordFrame();
|
|
|
|
PostPresent();
|
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;
|
2011-08-02 20:04:20 -07:00
|
|
|
viewMatrix._33 = 0.0f;
|
2010-10-01 15:24:58 -07:00
|
|
|
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,
|
2013-07-20 01:48:55 -07:00
|
|
|
getter_AddRefs(destSurf), nullptr);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
2010-08-10 16:32:45 -07:00
|
|
|
device()->GetRenderTargetData(backBuff, destSurf);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
D3DLOCKED_RECT rect;
|
2013-07-20 01:48:55 -07:00
|
|
|
destSurf->LockRect(&rect, nullptr, D3DLOCK_READONLY);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
nsRefPtr<gfxImageSurface> imageSurface =
|
|
|
|
new gfxImageSurface((unsigned char*)rect.pBits,
|
|
|
|
gfxIntSize(desc.Width, desc.Height),
|
|
|
|
rect.Pitch,
|
2014-01-23 10:26:40 -08:00
|
|
|
gfxImageFormat::ARGB32);
|
2010-05-24 08:28:51 -07:00
|
|
|
|
|
|
|
mTarget->SetSource(imageSurface);
|
|
|
|
mTarget->SetOperator(gfxContext::OPERATOR_OVER);
|
|
|
|
mTarget->Paint();
|
|
|
|
destSurf->UnlockRect();
|
|
|
|
}
|
|
|
|
|
|
|
|
LayerD3D9::LayerD3D9(LayerManagerD3D9 *aManager)
|
|
|
|
: mD3DManager(aManager)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
} /* namespace layers */
|
|
|
|
} /* namespace mozilla */
|