#include "../win/Win32Common.hpp" #include "logvisor/logvisor.hpp" #include "boo/graphicsdev/D3D.hpp" #include "boo/IGraphicsContext.hpp" #include #include #include #include #include #include #include #include #include #undef min #undef max extern pD3DCompile D3DCompilePROC; namespace boo { static logvisor::Module Log("boo::D3D11"); static inline void ThrowIfFailed(HRESULT hr) { if (FAILED(hr)) { // Set a breakpoint on this line to catch Win32 API errors. _com_error err(hr); LPCTSTR errMsg = err.ErrorMessage(); Log.report(logvisor::Fatal, errMsg); } } struct D3D11Data : IGraphicsData { std::vector> m_SPs; std::vector> m_SBinds; std::vector> m_SBufs; std::vector> m_DBufs; std::vector> m_STexs; std::vector> m_SATexs; std::vector> m_DTexs; std::vector> m_RTexs; std::vector> m_VFmts; }; class D3D11GraphicsBufferD; struct D3D11Pool : IGraphicsBufferPool { std::unordered_map> m_DBufs; }; static const D3D11_BIND_FLAG USE_TABLE[] = { D3D11_BIND_VERTEX_BUFFER, D3D11_BIND_VERTEX_BUFFER, D3D11_BIND_INDEX_BUFFER, D3D11_BIND_CONSTANT_BUFFER }; struct D3D11GraphicsBuffer { std::weak_ptr m_selfPtr; static std::shared_ptr getPtr(IGraphicsBuffer* buf); }; class D3D11GraphicsBufferS : public IGraphicsBufferS, public D3D11GraphicsBuffer { friend class D3D11DataFactory; friend struct D3D11CommandQueue; size_t m_sz; D3D11GraphicsBufferS(BufferUse use, D3D11Context* ctx, const void* data, size_t stride, size_t count) : m_stride(stride), m_count(count), m_sz(stride * count) { D3D11_SUBRESOURCE_DATA iData = {data}; ThrowIfFailed(ctx->m_dev->CreateBuffer(&CD3D11_BUFFER_DESC(m_sz, USE_TABLE[int(use)], D3D11_USAGE_IMMUTABLE), &iData, &m_buf)); } public: size_t m_stride; size_t m_count; ComPtr m_buf; ~D3D11GraphicsBufferS() = default; }; class D3D11GraphicsBufferD : public IGraphicsBufferD, public D3D11GraphicsBuffer { friend class D3D11DataFactory; friend struct D3D11CommandQueue; D3D11CommandQueue* m_q; std::unique_ptr m_cpuBuf; size_t m_cpuSz; int m_validSlots = 0; D3D11GraphicsBufferD(D3D11CommandQueue* q, BufferUse use, D3D11Context* ctx, size_t stride, size_t count) : m_q(q), m_stride(stride), m_count(count) { m_cpuSz = stride * count; m_cpuBuf.reset(new uint8_t[m_cpuSz]); for (int i=0 ; i<3 ; ++i) ThrowIfFailed(ctx->m_dev->CreateBuffer(&CD3D11_BUFFER_DESC(m_cpuSz, USE_TABLE[int(use)], D3D11_USAGE_DYNAMIC, D3D11_CPU_ACCESS_WRITE), nullptr, &m_bufs[i])); } void update(ID3D11DeviceContext* ctx, int b); public: size_t m_stride; size_t m_count; ComPtr m_bufs[3]; ~D3D11GraphicsBufferD() = default; void load(const void* data, size_t sz); void* map(size_t sz); void unmap(); }; std::shared_ptr D3D11GraphicsBuffer::getPtr(IGraphicsBuffer* buf) { if (!buf) return {}; if (buf->dynamic()) return static_cast(buf)->m_selfPtr.lock(); else return static_cast(buf)->m_selfPtr.lock(); } struct D3D11Texture { std::weak_ptr m_selfPtr; static std::shared_ptr getPtr(ITexture* tex); }; class D3D11TextureS : public ITextureS, public D3D11Texture { friend class D3D11DataFactory; size_t m_sz; D3D11TextureS(D3D11Context* ctx, size_t width, size_t height, size_t mips, TextureFormat fmt, const void* data, size_t sz) : m_sz(sz) { DXGI_FORMAT pfmt; int pxPitchNum = 1; int pxPitchDenom = 1; bool compressed = false; switch (fmt) { case TextureFormat::RGBA8: pfmt = DXGI_FORMAT_R8G8B8A8_UNORM; pxPitchNum = 4; break; case TextureFormat::I8: pfmt = DXGI_FORMAT_R8_UNORM; break; case TextureFormat::DXT1: pfmt = DXGI_FORMAT_BC1_UNORM; compressed = true; pxPitchNum = 1; pxPitchDenom = 2; break; default: Log.report(logvisor::Fatal, "unsupported tex format"); } CD3D11_TEXTURE2D_DESC desc(pfmt, width, height, 1, mips, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_IMMUTABLE); const uint8_t* dataIt = static_cast(data); D3D11_SUBRESOURCE_DATA upData[16] = {}; for (size_t i=0 ; im_dev->CreateTexture2D(&desc, upData, &m_tex)); ThrowIfFailed(ctx->m_dev->CreateShaderResourceView(m_tex.Get(), &CD3D11_SHADER_RESOURCE_VIEW_DESC(m_tex.Get(), D3D_SRV_DIMENSION_TEXTURE2D, pfmt, 0, mips), &m_srv)); } public: ComPtr m_tex; ComPtr m_srv; ~D3D11TextureS() = default; }; class D3D11TextureSA : public ITextureSA, public D3D11Texture { friend class D3D11DataFactory; size_t m_sz; D3D11TextureSA(D3D11Context* ctx, size_t width, size_t height, size_t layers, TextureFormat fmt, const void* data, size_t sz) : m_sz(sz) { size_t pixelPitch; DXGI_FORMAT pixelFmt; if (fmt == TextureFormat::RGBA8) { pixelPitch = 4; pixelFmt = DXGI_FORMAT_R8G8B8A8_UNORM; } else if (fmt == TextureFormat::I8) { pixelPitch = 1; pixelFmt = DXGI_FORMAT_R8_UNORM; } CD3D11_TEXTURE2D_DESC desc(pixelFmt, width, height, layers, 1, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_IMMUTABLE); const uint8_t* dataIt = static_cast(data); if (layers > 16) { std::unique_ptr upData(new D3D11_SUBRESOURCE_DATA[layers]); for (size_t i=0 ; im_dev->CreateTexture2D(&desc, upData.get(), &m_tex)); } else { D3D11_SUBRESOURCE_DATA upData[16] = {}; for (size_t i=0 ; im_dev->CreateTexture2D(&desc, upData, &m_tex)); } ThrowIfFailed(ctx->m_dev->CreateShaderResourceView(m_tex.Get(), &CD3D11_SHADER_RESOURCE_VIEW_DESC(m_tex.Get(), D3D_SRV_DIMENSION_TEXTURE2DARRAY, pixelFmt), &m_srv)); } public: ComPtr m_tex; ComPtr m_srv; ~D3D11TextureSA() = default; }; class D3D11TextureD : public ITextureD, public D3D11Texture { friend class D3D11DataFactory; friend struct D3D11CommandQueue; size_t m_width = 0; size_t m_height = 0; D3D11CommandQueue* m_q; std::unique_ptr m_cpuBuf; size_t m_cpuSz; int m_validSlots = 0; D3D11TextureD(D3D11CommandQueue* q, D3D11Context* ctx, size_t width, size_t height, TextureFormat fmt) : m_q(q) { DXGI_FORMAT pixelFmt; size_t pxPitch; switch (fmt) { case TextureFormat::RGBA8: pixelFmt = DXGI_FORMAT_R8G8B8A8_UNORM; pxPitch = 4; break; case TextureFormat::I8: pixelFmt = DXGI_FORMAT_R8_UNORM; pxPitch = 1; break; default: Log.report(logvisor::Fatal, "unsupported tex format"); } m_cpuSz = width * height * pxPitch; m_cpuBuf.reset(new uint8_t[m_cpuSz]); CD3D11_TEXTURE2D_DESC desc(pixelFmt, width, height, 1, 1, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_DYNAMIC, D3D11_CPU_ACCESS_WRITE); for (int i=0 ; i<3 ; ++i) { ThrowIfFailed(ctx->m_dev->CreateTexture2D(&desc, nullptr, &m_texs[i])); ThrowIfFailed(ctx->m_dev->CreateShaderResourceView(m_texs[i].Get(), &CD3D11_SHADER_RESOURCE_VIEW_DESC(m_texs[i].Get(), D3D_SRV_DIMENSION_TEXTURE2D, pixelFmt), &m_srvs[i])); } } void update(ID3D11DeviceContext* ctx, int b); public: ComPtr m_texs[3]; ComPtr m_srvs[3]; ~D3D11TextureD() = default; void load(const void* data, size_t sz); void* map(size_t sz); void unmap(); }; class D3D11TextureR : public ITextureR, public D3D11Texture { friend class D3D11DataFactory; friend struct D3D11CommandQueue; size_t m_width = 0; size_t m_height = 0; size_t m_samples = 0; bool m_enableShaderColorBind; bool m_enableShaderDepthBind; void Setup(D3D11Context* ctx, size_t width, size_t height, size_t samples, bool enableShaderColorBind, bool enableShaderDepthBind) { ThrowIfFailed(ctx->m_dev->CreateTexture2D(&CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R8G8B8A8_UNORM, width, height, 1, 1, D3D11_BIND_RENDER_TARGET, D3D11_USAGE_DEFAULT, 0, samples), nullptr, &m_colorTex)); ThrowIfFailed(ctx->m_dev->CreateTexture2D(&CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_D24_UNORM_S8_UINT, width, height, 1, 1, D3D11_BIND_DEPTH_STENCIL, D3D11_USAGE_DEFAULT, 0, samples), nullptr, &m_depthTex)); D3D11_RTV_DIMENSION rtvDim; D3D11_DSV_DIMENSION dsvDim; D3D11_SRV_DIMENSION srvDim; if (samples > 1) { rtvDim = D3D11_RTV_DIMENSION_TEXTURE2DMS; dsvDim = D3D11_DSV_DIMENSION_TEXTURE2DMS; srvDim = D3D11_SRV_DIMENSION_TEXTURE2DMS; } else { rtvDim = D3D11_RTV_DIMENSION_TEXTURE2D; dsvDim = D3D11_DSV_DIMENSION_TEXTURE2D; srvDim = D3D11_SRV_DIMENSION_TEXTURE2D; } ThrowIfFailed(ctx->m_dev->CreateRenderTargetView(m_colorTex.Get(), &CD3D11_RENDER_TARGET_VIEW_DESC(m_colorTex.Get(), rtvDim), &m_rtv)); ThrowIfFailed(ctx->m_dev->CreateDepthStencilView(m_depthTex.Get(), &CD3D11_DEPTH_STENCIL_VIEW_DESC(m_depthTex.Get(), dsvDim), &m_dsv)); if (enableShaderColorBind) { ThrowIfFailed(ctx->m_dev->CreateTexture2D(&CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R8G8B8A8_UNORM, width, height, 1, 1, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_DEFAULT, 0, samples), nullptr, &m_colorBindTex)); ThrowIfFailed(ctx->m_dev->CreateShaderResourceView(m_colorBindTex.Get(), &CD3D11_SHADER_RESOURCE_VIEW_DESC(m_colorBindTex.Get(), srvDim), &m_colorSrv)); } if (enableShaderDepthBind) { ThrowIfFailed(ctx->m_dev->CreateTexture2D(&CD3D11_TEXTURE2D_DESC(DXGI_FORMAT_R24G8_TYPELESS, width, height, 1, 1, D3D11_BIND_SHADER_RESOURCE, D3D11_USAGE_DEFAULT, 0, samples), nullptr, &m_depthBindTex)); ThrowIfFailed(ctx->m_dev->CreateShaderResourceView(m_depthBindTex.Get(), &CD3D11_SHADER_RESOURCE_VIEW_DESC(m_depthBindTex.Get(), srvDim, DXGI_FORMAT_R24G8_TYPELESS), &m_depthSrv)); } } D3D11TextureR(D3D11Context* ctx, size_t width, size_t height, size_t samples, bool enableShaderColorBind, bool enableShaderDepthBind) : m_width(width), m_height(height), m_samples(samples), m_enableShaderColorBind(enableShaderColorBind), m_enableShaderDepthBind(enableShaderDepthBind) { if (samples == 0) m_samples = 1; Setup(ctx, width, height, samples, enableShaderColorBind, enableShaderDepthBind); } public: size_t samples() const {return m_samples;} ComPtr m_colorTex; ComPtr m_rtv; ComPtr m_depthTex; ComPtr m_dsv; ComPtr m_colorBindTex; ComPtr m_colorSrv; ComPtr m_depthBindTex; ComPtr m_depthSrv; ~D3D11TextureR() = default; void resize(D3D11Context* ctx, size_t width, size_t height) { if (width < 1) width = 1; if (height < 1) height = 1; m_width = width; m_height = height; Setup(ctx, width, height, m_samples, m_enableShaderColorBind, m_enableShaderDepthBind); } }; std::shared_ptr D3D11Texture::getPtr(ITexture* tex) { if (!tex) return {}; switch (tex->type()) { case TextureType::Static: return static_cast(tex)->m_selfPtr.lock(); case TextureType::StaticArray: return static_cast(tex)->m_selfPtr.lock(); case TextureType::Dynamic: return static_cast(tex)->m_selfPtr.lock(); case TextureType::Render: return static_cast(tex)->m_selfPtr.lock(); } return {}; } static const size_t SEMANTIC_SIZE_TABLE[] = { 0, 12, 16, 12, 16, 16, 4, 8, 16, 16, 16 }; static const char* SEMANTIC_NAME_TABLE[] = { nullptr, "POSITION", "POSITION", "NORMAL", "NORMAL", "COLOR", "COLOR", "UV", "UV", "WEIGHT", "MODELVIEW" }; static const DXGI_FORMAT SEMANTIC_TYPE_TABLE[] = { DXGI_FORMAT_UNKNOWN, DXGI_FORMAT_R32G32B32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R32G32B32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_FORMAT_R32G32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R32G32B32A32_FLOAT }; struct D3D11VertexFormat : IVertexFormat { std::weak_ptr m_selfPtr; size_t m_elementCount; std::unique_ptr m_elements; size_t m_stride = 0; size_t m_instStride = 0; D3D11VertexFormat(size_t elementCount, const VertexElementDescriptor* elements) : m_elementCount(elementCount), m_elements(new D3D11_INPUT_ELEMENT_DESC[elementCount]) { memset(m_elements.get(), 0, elementCount * sizeof(D3D11_INPUT_ELEMENT_DESC)); for (size_t i=0 ; isemantic & boo::VertexSemantic::SemanticMask); elem.SemanticName = SEMANTIC_NAME_TABLE[semantic]; elem.SemanticIndex = elemin->semanticIdx; elem.Format = SEMANTIC_TYPE_TABLE[semantic]; if ((elemin->semantic & boo::VertexSemantic::Instanced) != boo::VertexSemantic::None) { elem.InputSlotClass = D3D11_INPUT_PER_INSTANCE_DATA; elem.InstanceDataStepRate = 1; elem.InputSlot = 1; elem.AlignedByteOffset = m_instStride; m_instStride += SEMANTIC_SIZE_TABLE[semantic]; } else { elem.InputSlotClass = D3D11_INPUT_PER_VERTEX_DATA; elem.AlignedByteOffset = m_stride; m_stride += SEMANTIC_SIZE_TABLE[semantic]; } } } }; static const D3D11_PRIMITIVE_TOPOLOGY PRIMITIVE_TABLE[] = { D3D10_PRIMITIVE_TOPOLOGY_TRIANGLELIST, D3D10_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP }; static const D3D11_BLEND BLEND_FACTOR_TABLE[] = { D3D11_BLEND_ZERO, D3D11_BLEND_ONE, D3D11_BLEND_SRC_COLOR, D3D11_BLEND_INV_SRC_COLOR, D3D11_BLEND_DEST_COLOR, D3D11_BLEND_INV_DEST_COLOR, D3D11_BLEND_SRC_ALPHA, D3D11_BLEND_INV_SRC_ALPHA, D3D11_BLEND_DEST_ALPHA, D3D11_BLEND_INV_DEST_ALPHA, D3D11_BLEND_SRC1_COLOR, D3D11_BLEND_INV_SRC1_COLOR }; class D3D11ShaderPipeline : public IShaderPipeline { friend class D3D11DataFactory; friend struct D3D11ShaderDataBinding; std::weak_ptr m_selfPtr; std::shared_ptr m_vtxFmt; D3D11ShaderPipeline(D3D11Context* ctx, ID3DBlob* vert, ID3DBlob* pixel, const D3D11VertexFormat* vtxFmt, BlendFactor srcFac, BlendFactor dstFac, Primitive prim, bool depthTest, bool depthWrite, bool backfaceCulling) : m_vtxFmt(vtxFmt->m_selfPtr.lock()), m_topology(PRIMITIVE_TABLE[int(prim)]) { ThrowIfFailed(ctx->m_dev->CreateVertexShader(vert->GetBufferPointer(), vert->GetBufferSize(), nullptr, &m_vShader)); ThrowIfFailed(ctx->m_dev->CreatePixelShader(pixel->GetBufferPointer(), pixel->GetBufferSize(), nullptr, &m_pShader)); CD3D11_RASTERIZER_DESC rasDesc(D3D11_FILL_SOLID, backfaceCulling ? D3D11_CULL_BACK : D3D11_CULL_NONE, true, D3D11_DEFAULT_DEPTH_BIAS, D3D11_DEFAULT_DEPTH_BIAS_CLAMP, D3D11_DEFAULT_SLOPE_SCALED_DEPTH_BIAS, true, true, false, false); ThrowIfFailed(ctx->m_dev->CreateRasterizerState(&rasDesc, &m_rasState)); CD3D11_DEPTH_STENCIL_DESC dsDesc(D3D11_DEFAULT); dsDesc.DepthEnable = depthTest; dsDesc.DepthWriteMask = D3D11_DEPTH_WRITE_MASK(depthWrite); dsDesc.DepthFunc = D3D11_COMPARISON_GREATER_EQUAL; ThrowIfFailed(ctx->m_dev->CreateDepthStencilState(&dsDesc, &m_dsState)); CD3D11_BLEND_DESC blDesc(D3D11_DEFAULT); blDesc.RenderTarget[0].BlendEnable = (dstFac != BlendFactor::Zero); blDesc.RenderTarget[0].SrcBlend = BLEND_FACTOR_TABLE[int(srcFac)]; blDesc.RenderTarget[0].DestBlend = BLEND_FACTOR_TABLE[int(dstFac)]; ThrowIfFailed(ctx->m_dev->CreateBlendState(&blDesc, &m_blState)); ThrowIfFailed(ctx->m_dev->CreateInputLayout(vtxFmt->m_elements.get(), vtxFmt->m_elementCount, vert->GetBufferPointer(), vert->GetBufferSize(), &m_inLayout)); } public: ComPtr m_vShader; ComPtr m_pShader; ComPtr m_rasState; ComPtr m_dsState; ComPtr m_blState; ComPtr m_inLayout; D3D11_PRIMITIVE_TOPOLOGY m_topology; ~D3D11ShaderPipeline() = default; D3D11ShaderPipeline& operator=(const D3D11ShaderPipeline&) = delete; D3D11ShaderPipeline(const D3D11ShaderPipeline&) = delete; void bind(ID3D11DeviceContext* ctx) { ctx->VSSetShader(m_vShader.Get(), nullptr, 0); ctx->PSSetShader(m_pShader.Get(), nullptr, 0); ctx->RSSetState(m_rasState.Get()); ctx->OMSetDepthStencilState(m_dsState.Get(), 0); ctx->OMSetBlendState(m_blState.Get(), nullptr, 0xffffffff); ctx->IASetInputLayout(m_inLayout.Get()); ctx->IASetPrimitiveTopology(m_topology); } }; struct D3D11ShaderDataBinding : IShaderDataBinding { std::weak_ptr m_selfPtr; std::shared_ptr m_pipeline; std::shared_ptr m_vbuf; std::shared_ptr m_instVbuf; std::shared_ptr m_ibuf; std::vector> m_ubufs; std::unique_ptr m_ubufFirstConsts; std::unique_ptr m_ubufNumConsts; std::unique_ptr m_pubufs; std::vector> m_texs; UINT m_baseOffsets[2]; D3D11ShaderDataBinding(D3D11Context* ctx, IShaderPipeline* pipeline, IGraphicsBuffer* vbuf, IGraphicsBuffer* instVbuf, IGraphicsBuffer* ibuf, size_t ubufCount, IGraphicsBuffer** ubufs, const PipelineStage* ubufStages, const size_t* ubufOffs, const size_t* ubufSizes, size_t texCount, ITexture** texs, size_t baseVert, size_t baseInst) : m_pipeline(static_cast(pipeline)->m_selfPtr), m_vbuf(D3D11GraphicsBuffer::getPtr(vbuf)), m_instVbuf(D3D11GraphicsBuffer::getPtr(instVbuf)), m_ibuf(D3D11GraphicsBuffer::getPtr(ibuf)) { m_ubufs.reserve(ubufCount); m_texs.reserve(texCount); m_baseOffsets[0] = UINT(baseVert * m_pipeline->m_vtxFmt->m_stride); m_baseOffsets[1] = UINT(baseInst * m_pipeline->m_vtxFmt->m_instStride); if (ubufStages) { m_pubufs.reset(new bool[ubufCount]); for (size_t i=0 ; ibind(ctx); ID3D11Buffer* bufs[2] = {}; UINT strides[2] = {}; if (m_vbuf) { if (m_vbuf->dynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_vbuf.get()); bufs[0] = cbuf->m_bufs[b].Get(); strides[0] = UINT(cbuf->m_stride); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_vbuf.get()); bufs[0] = cbuf->m_buf.Get(); strides[0] = UINT(cbuf->m_stride); } } if (m_instVbuf) { if (m_instVbuf->dynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_instVbuf.get()); bufs[1] = cbuf->m_bufs[b].Get(); strides[1] = UINT(cbuf->m_stride); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_instVbuf.get()); bufs[1] = cbuf->m_buf.Get(); strides[1] = UINT(cbuf->m_stride); } } ctx->IASetVertexBuffers(0, 2, bufs, strides, m_baseOffsets); if (m_ibuf) { if (m_ibuf->dynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_ibuf.get()); ctx->IASetIndexBuffer(cbuf->m_bufs[b].Get(), DXGI_FORMAT_R32_UINT, 0); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_ibuf.get()); ctx->IASetIndexBuffer(cbuf->m_buf.Get(), DXGI_FORMAT_R32_UINT, 0); } } if (m_ubufs.size()) { if (m_ubufFirstConsts) { ID3D11Buffer* constBufs[8] = {}; ctx->VSSetConstantBuffers(0, m_ubufs.size(), constBufs); for (int i=0 ; i<8 && idynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_bufs[b].Get(); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_buf.Get(); } } ctx->VSSetConstantBuffers1(0, m_ubufs.size(), constBufs, m_ubufFirstConsts.get(), m_ubufNumConsts.get()); if (m_pubufs) { ID3D11Buffer* constBufs[8] = {}; ctx->PSSetConstantBuffers(0, m_ubufs.size(), constBufs); for (int i=0 ; i<8 && idynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_bufs[b].Get(); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_buf.Get(); } } ctx->PSSetConstantBuffers1(0, m_ubufs.size(), constBufs, m_ubufFirstConsts.get(), m_ubufNumConsts.get()); } } else { ID3D11Buffer* constBufs[8] = {}; for (int i=0 ; i<8 && idynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_bufs[b].Get(); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_buf.Get(); } } ctx->VSSetConstantBuffers(0, m_ubufs.size(), constBufs); if (m_pubufs) { ID3D11Buffer* constBufs[8] = {}; for (int i=0 ; i<8 && idynamic()) { D3D11GraphicsBufferD* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_bufs[b].Get(); } else { D3D11GraphicsBufferS* cbuf = static_cast(m_ubufs[i].get()); constBufs[i] = cbuf->m_buf.Get(); } } ctx->PSSetConstantBuffers(0, m_ubufs.size(), constBufs); } } } if (m_texs.size()) { ID3D11ShaderResourceView* srvs[8] = {}; for (int i=0 ; i<8 && itype()) { case TextureType::Dynamic: { D3D11TextureD* ctex = static_cast(m_texs[i].get()); srvs[i] = ctex->m_srvs[b].Get(); break; } case TextureType::Static: { D3D11TextureS* ctex = static_cast(m_texs[i].get()); srvs[i] = ctex->m_srv.Get(); break; } case TextureType::StaticArray: { D3D11TextureSA* ctex = static_cast(m_texs[i].get()); srvs[i] = ctex->m_srv.Get(); break; } case TextureType::Render: { D3D11TextureR* ctex = static_cast(m_texs[i].get()); srvs[i] = ctex->m_colorSrv.Get(); break; } } } } ctx->PSSetShaderResources(0, m_texs.size(), srvs); } } }; struct D3D11CommandQueue : IGraphicsCommandQueue { Platform platform() const {return IGraphicsDataFactory::Platform::D3D11;} const SystemChar* platformName() const {return _S("D3D11");} D3D11Context* m_ctx; D3D11Context::Window* m_windowCtx; IGraphicsContext* m_parent; ComPtr m_deferredCtx; size_t m_fillBuf = 0; size_t m_completeBuf = 0; size_t m_drawBuf = 0; bool m_running = true; std::mutex m_mt; std::condition_variable m_cv; std::mutex m_initmt; std::condition_variable m_initcv; std::unique_lock m_initlk; std::thread m_thr; struct CommandList { ComPtr list; std::vector> bindings; std::shared_ptr workDoPresent; void reset() { list.Reset(); bindings.clear(); workDoPresent.reset(); } }; CommandList m_cmdLists[3]; std::recursive_mutex m_dynamicLock; void ProcessDynamicLoads(ID3D11DeviceContext* ctx); static void RenderingWorker(D3D11CommandQueue* self) { { std::unique_lock lk(self->m_initmt); } self->m_initcv.notify_one(); while (self->m_running) { { std::unique_lock lk(self->m_mt); self->m_cv.wait(lk); if (!self->m_running) break; self->m_drawBuf = self->m_completeBuf; auto& CmdList = self->m_cmdLists[self->m_drawBuf]; self->ProcessDynamicLoads(self->m_ctx->m_devCtx.Get()); if (self->m_texResizes.size()) { for (const auto& resize : self->m_texResizes) resize.first->resize(self->m_ctx, resize.second.first, resize.second.second); self->m_texResizes.clear(); CmdList.reset(); continue; } if (self->m_windowCtx->m_needsFSTransition) { if (self->m_windowCtx->m_fs) { self->m_windowCtx->m_swapChain->SetFullscreenState(true, nullptr); self->m_windowCtx->m_swapChain->ResizeTarget(&self->m_windowCtx->m_fsdesc); } else self->m_windowCtx->m_swapChain->SetFullscreenState(false, nullptr); self->m_windowCtx->m_needsFSTransition = false; CmdList.reset(); continue; } if (self->m_windowCtx->m_needsResize) { self->m_windowCtx->m_swapChain->ResizeBuffers(2, self->m_windowCtx->width, self->m_windowCtx->height, DXGI_FORMAT_R8G8B8A8_UNORM, DXGI_SWAP_CHAIN_FLAG_ALLOW_MODE_SWITCH); self->m_windowCtx->m_needsResize = false; CmdList.reset(); continue; } } auto& CmdList = self->m_cmdLists[self->m_drawBuf]; ID3D11CommandList* list = CmdList.list.Get(); self->m_ctx->m_devCtx->ExecuteCommandList(list, false); D3D11TextureR* csource = CmdList.workDoPresent.get(); if (csource) { ComPtr dest; ThrowIfFailed(self->m_windowCtx->m_swapChain->GetBuffer(0, __uuidof(ID3D11Texture2D), &dest)); ID3D11Texture2D* src = csource->m_colorTex.Get(); if (csource->m_samples > 1) self->m_ctx->m_devCtx->ResolveSubresource(dest.Get(), 0, src, 0, DXGI_FORMAT_R8G8B8A8_UNORM); else self->m_ctx->m_devCtx->CopyResource(dest.Get(), src); self->m_windowCtx->m_swapChain->Present(1, 0); } CmdList.reset(); } } D3D11CommandQueue(D3D11Context* ctx, D3D11Context::Window* windowCtx, IGraphicsContext* parent) : m_ctx(ctx), m_windowCtx(windowCtx), m_parent(parent), m_initlk(m_initmt), m_thr(RenderingWorker, this) { m_initcv.wait(m_initlk); m_initlk.unlock(); ThrowIfFailed(ctx->m_dev->CreateDeferredContext1(0, &m_deferredCtx)); } void stopRenderer() { m_running = false; m_cv.notify_one(); m_thr.join(); } ~D3D11CommandQueue() { if (m_running) stopRenderer(); } void setShaderDataBinding(IShaderDataBinding* binding) { D3D11ShaderDataBinding* cbind = static_cast(binding); cbind->bind(m_deferredCtx.Get(), m_fillBuf); m_cmdLists[m_fillBuf].bindings.push_back(cbind->m_selfPtr.lock()); ID3D11SamplerState* samp[] = {m_ctx->m_ss.Get()}; m_deferredCtx->PSSetSamplers(0, 1, samp); } std::shared_ptr m_boundTarget; void setRenderTarget(ITextureR* target) { D3D11TextureR* ctarget = static_cast(target); ID3D11RenderTargetView* view[] = {ctarget->m_rtv.Get()}; m_deferredCtx->OMSetRenderTargets(1, view, ctarget->m_dsv.Get()); m_boundTarget = std::static_pointer_cast(ctarget->m_selfPtr.lock()); } void setViewport(const SWindowRect& rect, float znear, float zfar) { if (m_boundTarget) { int boundHeight = m_boundTarget->m_height; D3D11_VIEWPORT vp = {FLOAT(rect.location[0]), FLOAT(boundHeight - rect.location[1] - rect.size[1]), FLOAT(rect.size[0]), FLOAT(rect.size[1]), znear, zfar}; m_deferredCtx->RSSetViewports(1, &vp); } } void setScissor(const SWindowRect& rect) { if (m_boundTarget) { int boundHeight = m_boundTarget->m_height; D3D11_RECT d3drect = {LONG(rect.location[0]), LONG(boundHeight - rect.location[1] - rect.size[1]), LONG(rect.location[0] + rect.size[0]), LONG(boundHeight - rect.location[1])}; m_deferredCtx->RSSetScissorRects(1, &d3drect); } } std::unordered_map, std::pair> m_texResizes; void resizeRenderTexture(ITextureR* tex, size_t width, size_t height) { D3D11TextureR* ctex = static_cast(tex); std::unique_lock lk(m_mt); m_texResizes[std::static_pointer_cast(ctex->m_selfPtr.lock())] = std::make_pair(width, height); } void schedulePostFrameHandler(std::function&& func) { func(); } float m_clearColor[4] = {0.0,0.0,0.0,1.0}; void setClearColor(const float rgba[4]) { m_clearColor[0] = rgba[0]; m_clearColor[1] = rgba[1]; m_clearColor[2] = rgba[2]; m_clearColor[3] = rgba[3]; } void clearTarget(bool render=true, bool depth=true) { if (!m_boundTarget) return; if (render) m_deferredCtx->ClearRenderTargetView(m_boundTarget->m_rtv.Get(), m_clearColor); if (depth) m_deferredCtx->ClearDepthStencilView(m_boundTarget->m_dsv.Get(), D3D11_CLEAR_DEPTH, 0.0f, 0); } void draw(size_t start, size_t count) { m_deferredCtx->Draw(count, start); } void drawIndexed(size_t start, size_t count) { m_deferredCtx->DrawIndexed(count, start, 0); } void drawInstances(size_t start, size_t count, size_t instCount) { m_deferredCtx->DrawInstanced(count, instCount, start, 0); } void drawInstancesIndexed(size_t start, size_t count, size_t instCount) { m_deferredCtx->DrawIndexedInstanced(count, instCount, start, 0, 0); } void resolveBindTexture(ITextureR* texture, const SWindowRect& rect, bool tlOrigin, bool color, bool depth) { const D3D11TextureR* tex = static_cast(texture); if (color && tex->m_enableShaderColorBind) { if (tex->m_samples > 1) { m_deferredCtx->CopyResource(tex->m_colorBindTex.Get(), tex->m_colorTex.Get()); } else { SWindowRect intersectRect = rect.intersect(SWindowRect(0, 0, tex->m_width, tex->m_height)); int y = tlOrigin ? intersectRect.location[1] : (tex->m_height - intersectRect.size[1] - intersectRect.location[1]); D3D11_BOX box = {UINT(intersectRect.location[0]), UINT(y), 0, UINT(intersectRect.location[0] + intersectRect.size[0]), UINT(y + intersectRect.size[1]), 1}; m_deferredCtx->CopySubresourceRegion1(tex->m_colorBindTex.Get(), 0, box.left, box.top, 0, tex->m_colorTex.Get(), 0, &box, D3D11_COPY_DISCARD); } } if (depth && tex->m_enableShaderDepthBind) { m_deferredCtx->CopyResource(tex->m_depthBindTex.Get(), tex->m_depthTex.Get()); } } std::shared_ptr m_doPresent; void resolveDisplay(ITextureR* source) { m_doPresent = std::static_pointer_cast( static_cast(source)->m_selfPtr.lock()); } void execute(); }; void D3D11GraphicsBufferD::update(ID3D11DeviceContext* ctx, int b) { int slot = 1 << b; if ((slot & m_validSlots) == 0) { ID3D11Buffer* res = m_bufs[b].Get(); D3D11_MAPPED_SUBRESOURCE d; if (SUCCEEDED(ctx->Map(res, 0, D3D11_MAP_WRITE_DISCARD, 0, &d))) { memcpy(d.pData, m_cpuBuf.get(), m_cpuSz); ctx->Unmap(res, 0); } m_validSlots |= slot; } } void D3D11GraphicsBufferD::load(const void* data, size_t sz) { std::unique_lock lk(m_q->m_dynamicLock); size_t bufSz = std::min(sz, m_cpuSz); memcpy(m_cpuBuf.get(), data, bufSz); m_validSlots = 0; } void* D3D11GraphicsBufferD::map(size_t sz) { if (sz > m_cpuSz) return nullptr; m_q->m_dynamicLock.lock(); return m_cpuBuf.get(); } void D3D11GraphicsBufferD::unmap() { m_validSlots = 0; m_q->m_dynamicLock.unlock(); } void D3D11TextureD::update(ID3D11DeviceContext* ctx, int b) { int slot = 1 << b; if ((slot & m_validSlots) == 0) { ID3D11Texture2D* res = m_texs[b].Get(); D3D11_MAPPED_SUBRESOURCE d; ctx->Map(res, 0, D3D11_MAP_WRITE_DISCARD, 0, &d); memcpy(d.pData, m_cpuBuf.get(), m_cpuSz); ctx->Unmap(res, 0); m_validSlots |= slot; } } void D3D11TextureD::load(const void* data, size_t sz) { std::unique_lock lk(m_q->m_dynamicLock); size_t bufSz = std::min(sz, m_cpuSz); memcpy(m_cpuBuf.get(), data, bufSz); m_validSlots = 0; } void* D3D11TextureD::map(size_t sz) { if (sz > m_cpuSz) return nullptr; m_q->m_dynamicLock.lock(); return m_cpuBuf.get(); } void D3D11TextureD::unmap() { m_validSlots = 0; m_q->m_dynamicLock.unlock(); } class D3D11DataFactory : public ID3DDataFactory { friend struct D3D11CommandQueue; IGraphicsContext* m_parent; static thread_local D3D11Data* m_deferredData; struct D3D11Context* m_ctx; std::unordered_set m_committedData; std::unordered_set m_committedPools; std::mutex m_committedMutex; uint32_t m_sampleCount; void destroyData(IGraphicsData* d) { std::unique_lock lk(m_committedMutex); D3D11Data* data = static_cast(d); m_committedData.erase(data); delete data; } void destroyAllData() { std::unique_lock lk(m_committedMutex); for (IGraphicsData* data : m_committedData) delete static_cast(data); for (IGraphicsBufferPool* pool : m_committedPools) delete static_cast(pool); m_committedData.clear(); m_committedPools.clear(); } void destroyPool(IGraphicsBufferPool* p) { std::unique_lock lk(m_committedMutex); D3D11Pool* pool = static_cast(p); m_committedPools.erase(pool); delete pool; } IGraphicsBufferD* newPoolBuffer(IGraphicsBufferPool* p, BufferUse use, size_t stride, size_t count) { D3D11CommandQueue* q = static_cast(m_parent->getCommandQueue()); D3D11Pool* pool = static_cast(p); D3D11GraphicsBufferD* retval = new D3D11GraphicsBufferD(q, use, m_ctx, stride, count); retval->m_selfPtr = pool->m_DBufs.emplace(std::make_pair(retval, retval)).first->second; return retval; } void deletePoolBuffer(IGraphicsBufferPool* p, IGraphicsBufferD* buf) { D3D11Pool* pool = static_cast(p); pool->m_DBufs.erase(static_cast(buf)); } public: D3D11DataFactory(IGraphicsContext* parent, D3D11Context* ctx, uint32_t sampleCount) : m_parent(parent), m_ctx(ctx), m_sampleCount(sampleCount) {} ~D3D11DataFactory() {destroyAllData();} Platform platform() const {return Platform::D3D11;} const SystemChar* platformName() const {return _S("D3D11");} class Context : public ID3DDataFactory::Context { friend class D3D11DataFactory; D3D11DataFactory& m_parent; Context(D3D11DataFactory& parent) : m_parent(parent) {} public: Platform platform() const {return Platform::D3D11;} const SystemChar* platformName() const {return _S("D3D11");} IGraphicsBufferS* newStaticBuffer(BufferUse use, const void* data, size_t stride, size_t count) { D3D11Data* d = static_cast(m_deferredData); D3D11GraphicsBufferS* retval = new D3D11GraphicsBufferS(use, m_parent.m_ctx, data, stride, count); d->m_SBufs.emplace_back(retval); retval->m_selfPtr = d->m_SBufs.back(); return retval; } IGraphicsBufferD* newDynamicBuffer(BufferUse use, size_t stride, size_t count) { D3D11Data* d = static_cast(m_deferredData); D3D11CommandQueue* q = static_cast(m_parent.m_parent->getCommandQueue()); D3D11GraphicsBufferD* retval = new D3D11GraphicsBufferD(q, use, m_parent.m_ctx, stride, count); d->m_DBufs.emplace_back(retval); retval->m_selfPtr = d->m_DBufs.back(); return retval; } ITextureS* newStaticTexture(size_t width, size_t height, size_t mips, TextureFormat fmt, const void* data, size_t sz) { D3D11Data* d = static_cast(m_deferredData); D3D11TextureS* retval = new D3D11TextureS(m_parent.m_ctx, width, height, mips, fmt, data, sz); d->m_STexs.emplace_back(retval); retval->m_selfPtr = d->m_STexs.back(); return retval; } ITextureSA* newStaticArrayTexture(size_t width, size_t height, size_t layers, TextureFormat fmt, const void* data, size_t sz) { D3D11Data* d = static_cast(m_deferredData); D3D11TextureSA* retval = new D3D11TextureSA(m_parent.m_ctx, width, height, layers, fmt, data, sz); d->m_SATexs.emplace_back(retval); retval->m_selfPtr = d->m_SATexs.back(); return retval; } ITextureD* newDynamicTexture(size_t width, size_t height, TextureFormat fmt) { D3D11Data* d = static_cast(m_deferredData); D3D11CommandQueue* q = static_cast(m_parent.m_parent->getCommandQueue()); D3D11TextureD* retval = new D3D11TextureD(q, m_parent.m_ctx, width, height, fmt); d->m_DTexs.emplace_back(retval); retval->m_selfPtr = d->m_DTexs.back(); return retval; } ITextureR* newRenderTexture(size_t width, size_t height, bool enableShaderColorBind, bool enableShaderDepthBind) { D3D11Data* d = static_cast(m_deferredData); D3D11CommandQueue* q = static_cast(m_parent.m_parent->getCommandQueue()); D3D11TextureR* retval = new D3D11TextureR(m_parent.m_ctx, width, height, m_parent.m_sampleCount, enableShaderColorBind, enableShaderDepthBind); d->m_RTexs.emplace_back(retval); retval->m_selfPtr = d->m_RTexs.back(); return retval; } IVertexFormat* newVertexFormat(size_t elementCount, const VertexElementDescriptor* elements, size_t baseVert, size_t baseInst) { D3D11Data* d = static_cast(m_deferredData); D3D11CommandQueue* q = static_cast(m_parent.m_parent->getCommandQueue()); D3D11VertexFormat* retval = new struct D3D11VertexFormat(elementCount, elements); d->m_VFmts.emplace_back(retval); retval->m_selfPtr = d->m_VFmts.back(); return retval; } #if _DEBUG #define BOO_D3DCOMPILE_FLAG D3DCOMPILE_DEBUG | D3DCOMPILE_OPTIMIZATION_LEVEL0 #else #define BOO_D3DCOMPILE_FLAG D3DCOMPILE_OPTIMIZATION_LEVEL3 #endif IShaderPipeline* newShaderPipeline (const char* vertSource, const char* fragSource, ComPtr& vertBlobOut, ComPtr& fragBlobOut, ComPtr& pipelineBlob, IVertexFormat* vtxFmt, BlendFactor srcFac, BlendFactor dstFac, Primitive prim, bool depthTest, bool depthWrite, bool backfaceCulling) { ComPtr errBlob; if (!vertBlobOut) { if (FAILED(D3DCompilePROC(vertSource, strlen(vertSource), "HECL Vert Source", nullptr, nullptr, "main", "vs_5_0", BOO_D3DCOMPILE_FLAG, 0, &vertBlobOut, &errBlob))) { Log.report(logvisor::Fatal, "error compiling vert shader: %s", errBlob->GetBufferPointer()); return nullptr; } } if (!fragBlobOut) { if (FAILED(D3DCompilePROC(fragSource, strlen(fragSource), "HECL Pixel Source", nullptr, nullptr, "main", "ps_5_0", BOO_D3DCOMPILE_FLAG, 0, &fragBlobOut, &errBlob))) { Log.report(logvisor::Fatal, "error compiling pixel shader: %s", errBlob->GetBufferPointer()); return nullptr; } } D3D11Data* d = static_cast(m_deferredData); D3D11ShaderPipeline* retval = new D3D11ShaderPipeline(m_parent.m_ctx, vertBlobOut.Get(), fragBlobOut.Get(), static_cast(vtxFmt), srcFac, dstFac, prim, depthTest, depthWrite, backfaceCulling); d->m_SPs.emplace_back(retval); retval->m_selfPtr = d->m_SPs.back(); return retval; } IShaderDataBinding* newShaderDataBinding(IShaderPipeline* pipeline, IVertexFormat* vtxFormat, IGraphicsBuffer* vbuf, IGraphicsBuffer* instVbo, IGraphicsBuffer* ibuf, size_t ubufCount, IGraphicsBuffer** ubufs, const PipelineStage* ubufStages, const size_t* ubufOffs, const size_t* ubufSizes, size_t texCount, ITexture** texs, size_t baseVert, size_t baseInst) { D3D11Data* d = static_cast(m_deferredData); D3D11ShaderDataBinding* retval = new D3D11ShaderDataBinding(m_parent.m_ctx, pipeline, vbuf, instVbo, ibuf, ubufCount, ubufs, ubufStages, ubufOffs, ubufSizes, texCount, texs, baseVert, baseInst); d->m_SBinds.emplace_back(retval); retval->m_selfPtr = d->m_SBinds.back(); return retval; } }; GraphicsDataToken commitTransaction(const FactoryCommitFunc& trans) { if (m_deferredData) Log.report(logvisor::Fatal, "nested commitTransaction usage detected"); m_deferredData = new D3D11Data(); D3D11DataFactory::Context ctx(*this); if (!trans(ctx)) { delete m_deferredData; m_deferredData = nullptr; return GraphicsDataToken(this, nullptr); } std::unique_lock lk(m_committedMutex); D3D11Data* retval = m_deferredData; m_deferredData = nullptr; m_committedData.insert(retval); lk.unlock(); return GraphicsDataToken(this, retval); } GraphicsBufferPoolToken newBufferPool() { std::unique_lock lk(m_committedMutex); D3D11Pool* retval = new D3D11Pool; m_committedPools.insert(retval); return GraphicsBufferPoolToken(this, retval); } }; thread_local D3D11Data* D3D11DataFactory::m_deferredData; void D3D11CommandQueue::execute() { /* Finish command list */ auto& CmdList = m_cmdLists[m_fillBuf]; ThrowIfFailed(m_deferredCtx->FinishCommandList(false, &CmdList.list)); CmdList.workDoPresent = std::move(m_doPresent); m_doPresent.reset(); /* Wait for worker thread to become ready */ std::unique_lock lk(m_mt); /* Ready for next frame */ m_completeBuf = m_fillBuf; for (size_t i=0 ; i<3 ; ++i) { if (i == m_completeBuf || i == m_drawBuf) continue; m_fillBuf = i; break; } /* Return control to worker thread */ lk.unlock(); m_cv.notify_one(); } void D3D11CommandQueue::ProcessDynamicLoads(ID3D11DeviceContext* ctx) { D3D11DataFactory* gfxF = static_cast(m_parent->getDataFactory()); std::unique_lock lk(m_dynamicLock); std::unique_lock datalk(gfxF->m_committedMutex); for (D3D11Data* d : gfxF->m_committedData) { for (std::shared_ptr& b : d->m_DBufs) b->update(ctx, m_drawBuf); for (std::shared_ptr& t : d->m_DTexs) t->update(ctx, m_drawBuf); } for (D3D11Pool* p : gfxF->m_committedPools) { for (auto& b : p->m_DBufs) b.second->update(ctx, m_drawBuf); } } IGraphicsCommandQueue* _NewD3D11CommandQueue(D3D11Context* ctx, D3D11Context::Window* windowCtx, IGraphicsContext* parent) { return new D3D11CommandQueue(ctx, windowCtx, parent); } IGraphicsDataFactory* _NewD3D11DataFactory(D3D11Context* ctx, IGraphicsContext* parent, uint32_t sampleCount) { return new D3D11DataFactory(parent, ctx, sampleCount); } }