/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* 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 "WMF.h" #include "D3D11ShareHandleImage.h" #include "gfxImageSurface.h" #include "gfxWindowsPlatform.h" #include "mozilla/layers/TextureClient.h" #include "mozilla/layers/TextureD3D11.h" #include "mozilla/layers/CompositableClient.h" #include "mozilla/layers/CompositableForwarder.h" #include "mozilla/gfx/DeviceManagerDx.h" #include "d3d11.h" #include "gfxPrefs.h" #include "DXVA2Manager.h" #include namespace mozilla { namespace layers { using namespace gfx; D3D11ShareHandleImage::D3D11ShareHandleImage(const gfx::IntSize& aSize, const gfx::IntRect& aRect) : Image(nullptr, ImageFormat::D3D11_SHARE_HANDLE_TEXTURE), mSize(aSize), mPictureRect(aRect) { } bool D3D11ShareHandleImage::AllocateTexture(D3D11RecycleAllocator* aAllocator, ID3D11Device* aDevice) { if (aAllocator) { if (gfxPrefs::PDMWMFUseNV12Format() && gfx::DeviceManagerDx::Get()->CanUseNV12()) { mTextureClient = aAllocator->CreateOrRecycleClient(gfx::SurfaceFormat::NV12, mSize); } else { mTextureClient = aAllocator->CreateOrRecycleClient(gfx::SurfaceFormat::B8G8R8A8, mSize); } if (mTextureClient) { mTexture = static_cast(mTextureClient->GetInternalData())->GetD3D11Texture(); return true; } return false; } else { MOZ_ASSERT(aDevice); CD3D11_TEXTURE2D_DESC newDesc(DXGI_FORMAT_B8G8R8A8_UNORM, mSize.width, mSize.height, 1, 1, D3D11_BIND_RENDER_TARGET | D3D11_BIND_SHADER_RESOURCE); newDesc.MiscFlags = D3D11_RESOURCE_MISC_SHARED; HRESULT hr = aDevice->CreateTexture2D(&newDesc, nullptr, getter_AddRefs(mTexture)); return SUCCEEDED(hr); } } gfx::IntSize D3D11ShareHandleImage::GetSize() const { return mSize; } TextureClient* D3D11ShareHandleImage::GetTextureClient(KnowsCompositor* aForwarder) { return mTextureClient; } already_AddRefed D3D11ShareHandleImage::GetAsSourceSurface() { RefPtr texture = GetTexture(); if (!texture) { gfxWarning() << "Cannot readback from shared texture because no texture is available."; return nullptr; } RefPtr device; texture->GetDevice(getter_AddRefs(device)); D3D11_TEXTURE2D_DESC desc; texture->GetDesc(&desc); HRESULT hr; if (desc.Format == DXGI_FORMAT_NV12) { nsAutoCString error; std::unique_ptr manager(DXVA2Manager::CreateD3D11DXVA(nullptr, error, device)); if (!manager) { gfxWarning() << "Failed to create DXVA2 manager!"; return nullptr; } RefPtr outTexture; hr = manager->CopyToBGRATexture(texture, getter_AddRefs(outTexture)); if (FAILED(hr)) { gfxWarning() << "Failed to copy NV12 to BGRA texture."; return nullptr; } texture = outTexture; texture->GetDesc(&desc); } CD3D11_TEXTURE2D_DESC softDesc(desc.Format, desc.Width, desc.Height); softDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ; softDesc.BindFlags = 0; softDesc.MiscFlags = 0; softDesc.MipLevels = 1; softDesc.Usage = D3D11_USAGE_STAGING; RefPtr softTexture; hr = device->CreateTexture2D(&softDesc, NULL, static_cast(getter_AddRefs(softTexture))); if (FAILED(hr)) { NS_WARNING("Failed to create 2D staging texture."); return nullptr; } RefPtr context; device->GetImmediateContext(getter_AddRefs(context)); if (!context) { gfxCriticalError() << "Failed to get immediate context."; return nullptr; } RefPtr mutex; hr = texture->QueryInterface((IDXGIKeyedMutex**)getter_AddRefs(mutex)); if (SUCCEEDED(hr) && mutex) { hr = mutex->AcquireSync(0, 2000); if (hr != S_OK) { NS_WARNING("Acquire sync didn't manage to return within 2 seconds."); } } context->CopyResource(softTexture, texture); if (SUCCEEDED(hr) && mutex) { mutex->ReleaseSync(0); } RefPtr surface = gfx::Factory::CreateDataSourceSurface(mSize, gfx::SurfaceFormat::B8G8R8A8); if (NS_WARN_IF(!surface)) { return nullptr; } gfx::DataSourceSurface::MappedSurface mappedSurface; if (!surface->Map(gfx::DataSourceSurface::WRITE, &mappedSurface)) { return nullptr; } D3D11_MAPPED_SUBRESOURCE map; hr = context->Map(softTexture, 0, D3D11_MAP_READ, 0, &map); if (!SUCCEEDED(hr)) { surface->Unmap(); return nullptr; } for (int y = 0; y < mSize.height; y++) { memcpy(mappedSurface.mData + mappedSurface.mStride * y, (unsigned char*)(map.pData) + map.RowPitch * y, mSize.width * 4); } context->Unmap(softTexture, 0); surface->Unmap(); return surface.forget(); } ID3D11Texture2D* D3D11ShareHandleImage::GetTexture() const { return mTexture; } already_AddRefed D3D11RecycleAllocator::Allocate(gfx::SurfaceFormat aFormat, gfx::IntSize aSize, BackendSelector aSelector, TextureFlags aTextureFlags, TextureAllocationFlags aAllocFlags) { return CreateD3D11TextureClientWithDevice(aSize, aFormat, aTextureFlags, aAllocFlags, mDevice, mSurfaceAllocator->GetTextureForwarder()); } already_AddRefed D3D11RecycleAllocator::CreateOrRecycleClient(gfx::SurfaceFormat aFormat, const gfx::IntSize& aSize) { TextureAllocationFlags allocFlags = TextureAllocationFlags::ALLOC_DEFAULT; if (gfxPrefs::PDMWMFUseSyncTexture() || mDevice == DeviceManagerDx::Get()->GetCompositorDevice()) { // If our device is the compositor device, we don't need any synchronization in practice. allocFlags = TextureAllocationFlags::ALLOC_MANUAL_SYNCHRONIZATION; } RefPtr textureClient = CreateOrRecycle(aFormat, aSize, BackendSelector::Content, layers::TextureFlags::DEFAULT, allocFlags); return textureClient.forget(); } } // namespace layers } // namespace mozilla