1 #include "precompiled.h"
3 // Copyright (c) 2012 The ANGLE Project Authors. All rights reserved.
4 // Use of this source code is governed by a BSD-style license that can be
5 // found in the LICENSE file.
8 // RenderTarget11.cpp: Implements a DX11-specific wrapper for ID3D11View pointers
9 // retained by Renderbuffers.
11 #include "libGLESv2/renderer/d3d/d3d11/RenderTarget11.h"
12 #include "libGLESv2/renderer/d3d/d3d11/Renderer11.h"
14 #include "libGLESv2/renderer/d3d/d3d11/renderer11_utils.h"
15 #include "libGLESv2/renderer/d3d/d3d11/formatutils11.h"
16 #include "libGLESv2/main.h"
21 static bool getTextureProperties(ID3D11Resource *resource, unsigned int *mipLevels, unsigned int *samples)
23 ID3D11Texture1D *texture1D = d3d11::DynamicCastComObject<ID3D11Texture1D>(resource);
26 D3D11_TEXTURE1D_DESC texDesc;
27 texture1D->GetDesc(&texDesc);
28 SafeRelease(texture1D);
30 *mipLevels = texDesc.MipLevels;
36 ID3D11Texture2D *texture2D = d3d11::DynamicCastComObject<ID3D11Texture2D>(resource);
39 D3D11_TEXTURE2D_DESC texDesc;
40 texture2D->GetDesc(&texDesc);
41 SafeRelease(texture2D);
43 *mipLevels = texDesc.MipLevels;
44 *samples = texDesc.SampleDesc.Count > 1 ? texDesc.SampleDesc.Count : 0;
49 ID3D11Texture3D *texture3D = d3d11::DynamicCastComObject<ID3D11Texture3D>(resource);
52 D3D11_TEXTURE3D_DESC texDesc;
53 texture3D->GetDesc(&texDesc);
54 SafeRelease(texture3D);
56 *mipLevels = texDesc.MipLevels;
65 static unsigned int getRTVSubresourceIndex(ID3D11Resource *resource, ID3D11RenderTargetView *view)
67 D3D11_RENDER_TARGET_VIEW_DESC rtvDesc;
68 view->GetDesc(&rtvDesc);
70 unsigned int mipSlice = 0;
71 unsigned int arraySlice = 0;
73 switch (rtvDesc.ViewDimension)
75 case D3D11_RTV_DIMENSION_TEXTURE1D:
76 mipSlice = rtvDesc.Texture1D.MipSlice;
80 case D3D11_RTV_DIMENSION_TEXTURE1DARRAY:
81 mipSlice = rtvDesc.Texture1DArray.MipSlice;
82 arraySlice = rtvDesc.Texture1DArray.FirstArraySlice;
85 case D3D11_RTV_DIMENSION_TEXTURE2D:
86 mipSlice = rtvDesc.Texture2D.MipSlice;
90 case D3D11_RTV_DIMENSION_TEXTURE2DARRAY:
91 mipSlice = rtvDesc.Texture2DArray.MipSlice;
92 arraySlice = rtvDesc.Texture2DArray.FirstArraySlice;
95 case D3D11_RTV_DIMENSION_TEXTURE2DMS:
100 case D3D11_RTV_DIMENSION_TEXTURE2DMSARRAY:
102 arraySlice = rtvDesc.Texture2DMSArray.FirstArraySlice;
105 case D3D11_RTV_DIMENSION_TEXTURE3D:
106 mipSlice = rtvDesc.Texture3D.MipSlice;
110 case D3D11_RTV_DIMENSION_UNKNOWN:
111 case D3D11_RTV_DIMENSION_BUFFER:
120 unsigned int mipLevels, samples;
121 getTextureProperties(resource, &mipLevels, &samples);
123 return D3D11CalcSubresource(mipSlice, arraySlice, mipLevels);
126 static unsigned int getDSVSubresourceIndex(ID3D11Resource *resource, ID3D11DepthStencilView *view)
128 D3D11_DEPTH_STENCIL_VIEW_DESC dsvDesc;
129 view->GetDesc(&dsvDesc);
131 unsigned int mipSlice = 0;
132 unsigned int arraySlice = 0;
134 switch (dsvDesc.ViewDimension)
136 case D3D11_DSV_DIMENSION_TEXTURE1D:
137 mipSlice = dsvDesc.Texture1D.MipSlice;
141 case D3D11_DSV_DIMENSION_TEXTURE1DARRAY:
142 mipSlice = dsvDesc.Texture1DArray.MipSlice;
143 arraySlice = dsvDesc.Texture1DArray.FirstArraySlice;
146 case D3D11_DSV_DIMENSION_TEXTURE2D:
147 mipSlice = dsvDesc.Texture2D.MipSlice;
151 case D3D11_DSV_DIMENSION_TEXTURE2DARRAY:
152 mipSlice = dsvDesc.Texture2DArray.MipSlice;
153 arraySlice = dsvDesc.Texture2DArray.FirstArraySlice;
156 case D3D11_DSV_DIMENSION_TEXTURE2DMS:
161 case D3D11_DSV_DIMENSION_TEXTURE2DMSARRAY:
163 arraySlice = dsvDesc.Texture2DMSArray.FirstArraySlice;
166 case D3D11_DSV_DIMENSION_UNKNOWN:
175 unsigned int mipLevels, samples;
176 getTextureProperties(resource, &mipLevels, &samples);
178 return D3D11CalcSubresource(mipSlice, arraySlice, mipLevels);
181 RenderTarget11::RenderTarget11(Renderer *renderer, ID3D11RenderTargetView *rtv, ID3D11Resource *resource,
182 ID3D11ShaderResourceView *srv, GLsizei width, GLsizei height, GLsizei depth)
184 mRenderer = Renderer11::makeRenderer11(renderer);
195 mRenderTarget->AddRef();
198 mDepthStencil = NULL;
200 mShaderResource = srv;
203 mShaderResource->AddRef();
206 mSubresourceIndex = 0;
208 if (mRenderTarget && mTexture)
210 D3D11_RENDER_TARGET_VIEW_DESC desc;
211 mRenderTarget->GetDesc(&desc);
213 unsigned int mipLevels, samples;
214 getTextureProperties(mTexture, &mipLevels, &samples);
216 mSubresourceIndex = getRTVSubresourceIndex(mTexture, mRenderTarget);
222 const d3d11::DXGIFormat &dxgiFormatInfo = d3d11::GetDXGIFormatInfo(desc.Format);
223 mInternalFormat = dxgiFormatInfo.internalFormat;
224 mActualFormat = dxgiFormatInfo.internalFormat;
228 RenderTarget11::RenderTarget11(Renderer *renderer, ID3D11DepthStencilView *dsv, ID3D11Resource *resource,
229 ID3D11ShaderResourceView *srv, GLsizei width, GLsizei height, GLsizei depth)
231 mRenderer = Renderer11::makeRenderer11(renderer);
239 mRenderTarget = NULL;
244 mDepthStencil->AddRef();
247 mShaderResource = srv;
250 mShaderResource->AddRef();
253 mSubresourceIndex = 0;
255 if (mDepthStencil && mTexture)
257 D3D11_DEPTH_STENCIL_VIEW_DESC desc;
258 mDepthStencil->GetDesc(&desc);
260 unsigned int mipLevels, samples;
261 getTextureProperties(mTexture, &mipLevels, &samples);
263 mSubresourceIndex = getDSVSubresourceIndex(mTexture, mDepthStencil);
269 const d3d11::DXGIFormat &dxgiFormatInfo = d3d11::GetDXGIFormatInfo(desc.Format);
270 mInternalFormat = dxgiFormatInfo.internalFormat;
271 mActualFormat = dxgiFormatInfo.internalFormat;
275 RenderTarget11::RenderTarget11(Renderer *renderer, GLsizei width, GLsizei height, GLenum internalFormat, GLsizei samples)
277 mRenderer = Renderer11::makeRenderer11(renderer);
279 mRenderTarget = NULL;
280 mDepthStencil = NULL;
281 mShaderResource = NULL;
283 const d3d11::TextureFormat &formatInfo = d3d11::GetTextureFormatInfo(internalFormat);
284 const d3d11::DXGIFormat &dxgiFormatInfo = d3d11::GetDXGIFormatInfo(formatInfo.texFormat);
286 const gl::TextureCaps &textureCaps = mRenderer->getRendererTextureCaps().get(internalFormat);
287 GLuint supportedSamples = textureCaps.getNearestSamples(samples);
289 if (width > 0 && height > 0)
291 // Create texture resource
292 D3D11_TEXTURE2D_DESC desc;
294 desc.Height = height;
297 desc.Format = formatInfo.texFormat;
298 desc.SampleDesc.Count = (supportedSamples == 0) ? 1 : supportedSamples;
299 desc.SampleDesc.Quality = 0;
300 desc.Usage = D3D11_USAGE_DEFAULT;
301 desc.CPUAccessFlags = 0;
304 // If a rendertarget or depthstencil format exists for this texture format,
305 // we'll flag it to allow binding that way. Shader resource views are a little
307 bool bindRTV = false, bindDSV = false, bindSRV = false;
308 bindRTV = (formatInfo.rtvFormat != DXGI_FORMAT_UNKNOWN);
309 bindDSV = (formatInfo.dsvFormat != DXGI_FORMAT_UNKNOWN);
310 if (formatInfo.srvFormat != DXGI_FORMAT_UNKNOWN)
312 // Multisample targets flagged for binding as depth stencil cannot also be
313 // flagged for binding as SRV, so make certain not to add the SRV flag for
315 bindSRV = !(formatInfo.dsvFormat != DXGI_FORMAT_UNKNOWN && desc.SampleDesc.Count > 1);
318 desc.BindFlags = (bindRTV ? D3D11_BIND_RENDER_TARGET : 0) |
319 (bindDSV ? D3D11_BIND_DEPTH_STENCIL : 0) |
320 (bindSRV ? D3D11_BIND_SHADER_RESOURCE : 0);
322 ID3D11Device *device = mRenderer->getDevice();
323 ID3D11Texture2D *texture = NULL;
324 HRESULT result = device->CreateTexture2D(&desc, NULL, &texture);
327 if (result == E_OUTOFMEMORY)
329 gl::error(GL_OUT_OF_MEMORY);
332 ASSERT(SUCCEEDED(result));
336 D3D11_SHADER_RESOURCE_VIEW_DESC srvDesc;
337 srvDesc.Format = formatInfo.srvFormat;
338 srvDesc.ViewDimension = (supportedSamples == 0) ? D3D11_SRV_DIMENSION_TEXTURE2D : D3D11_SRV_DIMENSION_TEXTURE2DMS;
339 srvDesc.Texture2D.MostDetailedMip = 0;
340 srvDesc.Texture2D.MipLevels = 1;
341 result = device->CreateShaderResourceView(mTexture, &srvDesc, &mShaderResource);
343 if (result == E_OUTOFMEMORY)
345 SafeRelease(mTexture);
346 gl::error(GL_OUT_OF_MEMORY);
349 ASSERT(SUCCEEDED(result));
354 D3D11_DEPTH_STENCIL_VIEW_DESC dsvDesc;
355 dsvDesc.Format = formatInfo.dsvFormat;
356 dsvDesc.ViewDimension = (supportedSamples == 0) ? D3D11_DSV_DIMENSION_TEXTURE2D : D3D11_DSV_DIMENSION_TEXTURE2DMS;
357 dsvDesc.Texture2D.MipSlice = 0;
359 result = device->CreateDepthStencilView(mTexture, &dsvDesc, &mDepthStencil);
361 if (result == E_OUTOFMEMORY)
363 SafeRelease(mTexture);
364 SafeRelease(mShaderResource);
365 gl::error(GL_OUT_OF_MEMORY);
368 ASSERT(SUCCEEDED(result));
373 D3D11_RENDER_TARGET_VIEW_DESC rtvDesc;
374 rtvDesc.Format = formatInfo.rtvFormat;
375 rtvDesc.ViewDimension = (supportedSamples == 0) ? D3D11_RTV_DIMENSION_TEXTURE2D : D3D11_RTV_DIMENSION_TEXTURE2DMS;
376 rtvDesc.Texture2D.MipSlice = 0;
377 result = device->CreateRenderTargetView(mTexture, &rtvDesc, &mRenderTarget);
379 if (result == E_OUTOFMEMORY)
381 SafeRelease(mTexture);
382 SafeRelease(mShaderResource);
383 SafeRelease(mDepthStencil);
384 gl::error(GL_OUT_OF_MEMORY);
387 ASSERT(SUCCEEDED(result));
389 if (formatInfo.dataInitializerFunction != NULL)
391 ID3D11DeviceContext *context = mRenderer->getDeviceContext();
393 const float clearValues[4] = { 0.0f, 0.0f, 0.0f, 1.0f };
394 context->ClearRenderTargetView(mRenderTarget, clearValues);
403 mInternalFormat = internalFormat;
404 mSamples = supportedSamples;
405 mActualFormat = dxgiFormatInfo.internalFormat;
406 mSubresourceIndex = D3D11CalcSubresource(0, 0, 1);
409 RenderTarget11::~RenderTarget11()
411 SafeRelease(mTexture);
412 SafeRelease(mRenderTarget);
413 SafeRelease(mDepthStencil);
414 SafeRelease(mShaderResource);
417 RenderTarget11 *RenderTarget11::makeRenderTarget11(RenderTarget *target)
419 ASSERT(HAS_DYNAMIC_TYPE(rx::RenderTarget11*, target));
420 return static_cast<rx::RenderTarget11*>(target);
423 void RenderTarget11::invalidate(GLint x, GLint y, GLsizei width, GLsizei height)
428 ID3D11Resource *RenderTarget11::getTexture() const
433 ID3D11RenderTargetView *RenderTarget11::getRenderTargetView() const
435 return mRenderTarget;
438 ID3D11DepthStencilView *RenderTarget11::getDepthStencilView() const
440 return mDepthStencil;
443 ID3D11ShaderResourceView *RenderTarget11::getShaderResourceView() const
445 return mShaderResource;
448 unsigned int RenderTarget11::getSubresourceIndex() const
450 return mSubresourceIndex;