| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2012 Sam Lantinga <slouken@libsdl.org>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "SDL_config.h"
- #if SDL_VIDEO_RENDER_D3D11 && !SDL_RENDER_DISABLED
- #ifdef __WINRT__
- #include <windows.ui.core.h>
- #include <windows.foundation.h>
- #if WINAPI_FAMILY == WINAPI_FAMILY_APP
- #include <windows.ui.xaml.media.dxinterop.h>
- #endif
- #endif
- extern "C" {
- #include "../../core/windows/SDL_windows.h"
- #include "SDL_hints.h"
- //#include "SDL_loadso.h"
- #include "SDL_system.h"
- #include "SDL_syswm.h"
- #include "../SDL_sysrender.h"
- #include "SDL_log.h"
- #include "../../video/SDL_sysvideo.h"
- //#include "stdio.h"
- }
- #include <fstream>
- #include <string>
- #include <vector>
- #include "SDL_render_d3d11_cpp.h"
- using namespace DirectX;
- using namespace Microsoft::WRL;
- using namespace std;
- #ifdef __WINRT__
- using namespace Windows::Graphics::Display;
- using namespace Windows::UI::Core;
- #endif
- /* Texture sampling types */
- static const D3D11_FILTER SDL_D3D11_NEAREST_PIXEL_FILTER = D3D11_FILTER_MIN_MAG_MIP_POINT;
- static const D3D11_FILTER SDL_D3D11_LINEAR_FILTER = D3D11_FILTER_MIN_MAG_MIP_LINEAR;
- /* Direct3D 11.1 renderer implementation */
- static SDL_Renderer *D3D11_CreateRenderer(SDL_Window * window, Uint32 flags);
- static void D3D11_WindowEvent(SDL_Renderer * renderer,
- const SDL_WindowEvent *event);
- static int D3D11_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture);
- static int D3D11_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * rect, const void *srcPixels,
- int srcPitch);
- static int D3D11_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * rect, void **pixels, int *pitch);
- static void D3D11_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture);
- static int D3D11_SetRenderTarget(SDL_Renderer * renderer, SDL_Texture * texture);
- static int D3D11_UpdateViewport(SDL_Renderer * renderer);
- static int D3D11_UpdateClipRect(SDL_Renderer * renderer);
- static int D3D11_RenderClear(SDL_Renderer * renderer);
- static int D3D11_RenderDrawPoints(SDL_Renderer * renderer,
- const SDL_FPoint * points, int count);
- static int D3D11_RenderDrawLines(SDL_Renderer * renderer,
- const SDL_FPoint * points, int count);
- static int D3D11_RenderFillRects(SDL_Renderer * renderer,
- const SDL_FRect * rects, int count);
- static int D3D11_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * srcrect, const SDL_FRect * dstrect);
- static int D3D11_RenderCopyEx(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * srcrect, const SDL_FRect * dstrect,
- const double angle, const SDL_FPoint * center, const SDL_RendererFlip flip);
- static int D3D11_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
- Uint32 format, void * pixels, int pitch);
- static void D3D11_RenderPresent(SDL_Renderer * renderer);
- static void D3D11_DestroyTexture(SDL_Renderer * renderer,
- SDL_Texture * texture);
- static void D3D11_DestroyRenderer(SDL_Renderer * renderer);
- /* Direct3D 11.1 Internal Functions */
- HRESULT D3D11_CreateDeviceResources(SDL_Renderer * renderer);
- HRESULT D3D11_CreateWindowSizeDependentResources(SDL_Renderer * renderer);
- HRESULT D3D11_UpdateForWindowSizeChange(SDL_Renderer * renderer);
- HRESULT D3D11_HandleDeviceLost(SDL_Renderer * renderer);
- extern "C" SDL_RenderDriver D3D11_RenderDriver = {
- D3D11_CreateRenderer,
- {
- "direct3d 11.1",
- (
- SDL_RENDERER_ACCELERATED |
- SDL_RENDERER_PRESENTVSYNC |
- SDL_RENDERER_TARGETTEXTURE
- ), // flags. see SDL_RendererFlags
- 2, // num_texture_formats
- { // texture_formats
- SDL_PIXELFORMAT_RGB888,
- SDL_PIXELFORMAT_ARGB8888
- },
- 0, // max_texture_width: will be filled in later
- 0 // max_texture_height: will be filled in later
- }
- };
- static Uint32
- DXGIFormatToSDLPixelFormat(DXGI_FORMAT dxgiFormat) {
- switch (dxgiFormat) {
- case DXGI_FORMAT_B8G8R8A8_UNORM:
- return SDL_PIXELFORMAT_ARGB8888;
- case DXGI_FORMAT_B8G8R8X8_UNORM:
- return SDL_PIXELFORMAT_RGB888;
- default:
- return SDL_PIXELFORMAT_UNKNOWN;
- }
- }
- static DXGI_FORMAT
- SDLPixelFormatToDXGIFormat(Uint32 sdlFormat)
- {
- switch (sdlFormat) {
- case SDL_PIXELFORMAT_ARGB8888:
- return DXGI_FORMAT_B8G8R8A8_UNORM;
- case SDL_PIXELFORMAT_RGB888:
- return DXGI_FORMAT_B8G8R8X8_UNORM;
- default:
- return DXGI_FORMAT_UNKNOWN;
- }
- }
- //typedef struct
- //{
- // float x, y, z;
- // DWORD color;
- // float u, v;
- //} Vertex;
- SDL_Renderer *
- D3D11_CreateRenderer(SDL_Window * window, Uint32 flags)
- {
- SDL_Renderer *renderer;
- D3D11_RenderData *data;
- renderer = (SDL_Renderer *) SDL_calloc(1, sizeof(*renderer));
- if (!renderer) {
- SDL_OutOfMemory();
- return NULL;
- }
- SDL_zerop(renderer);
- data = new D3D11_RenderData; // Use the C++ 'new' operator to make sure the struct's members initialize using C++ rules
- if (!data) {
- SDL_OutOfMemory();
- return NULL;
- }
- data->featureLevel = (D3D_FEATURE_LEVEL) 0;
- data->windowSizeInDIPs = XMFLOAT2(0, 0);
- data->renderTargetSize = XMFLOAT2(0, 0);
- renderer->WindowEvent = D3D11_WindowEvent;
- renderer->CreateTexture = D3D11_CreateTexture;
- renderer->UpdateTexture = D3D11_UpdateTexture;
- renderer->LockTexture = D3D11_LockTexture;
- renderer->UnlockTexture = D3D11_UnlockTexture;
- renderer->SetRenderTarget = D3D11_SetRenderTarget;
- renderer->UpdateViewport = D3D11_UpdateViewport;
- renderer->UpdateClipRect = D3D11_UpdateClipRect;
- renderer->RenderClear = D3D11_RenderClear;
- renderer->RenderDrawPoints = D3D11_RenderDrawPoints;
- renderer->RenderDrawLines = D3D11_RenderDrawLines;
- renderer->RenderFillRects = D3D11_RenderFillRects;
- renderer->RenderCopy = D3D11_RenderCopy;
- renderer->RenderCopyEx = D3D11_RenderCopyEx;
- renderer->RenderReadPixels = D3D11_RenderReadPixels;
- renderer->RenderPresent = D3D11_RenderPresent;
- renderer->DestroyTexture = D3D11_DestroyTexture;
- renderer->DestroyRenderer = D3D11_DestroyRenderer;
- renderer->info = D3D11_RenderDriver.info;
- renderer->driverdata = data;
- // HACK: make sure the SDL_Renderer references the SDL_Window data now, in
- // order to give init functions access to the underlying window handle:
- renderer->window = window;
- /* Initialize Direct3D resources */
- if (FAILED(D3D11_CreateDeviceResources(renderer))) {
- D3D11_DestroyRenderer(renderer);
- return NULL;
- }
- if (FAILED(D3D11_CreateWindowSizeDependentResources(renderer))) {
- D3D11_DestroyRenderer(renderer);
- return NULL;
- }
- // TODO, WinRT: fill in renderer->info.texture_formats where appropriate
- return renderer;
- }
- static void
- D3D11_DestroyRenderer(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- if (data) {
- delete data;
- data = NULL;
- }
- }
- static bool
- D3D11_ReadFileContents(const wstring & fileName, vector<char> & out)
- {
- ifstream in(fileName, ios::in | ios::binary);
- if (!in) {
- return false;
- }
- in.seekg(0, ios::end);
- out.resize((size_t) in.tellg());
- in.seekg(0, ios::beg);
- in.read(&out[0], out.size());
- return in.good();
- }
- static bool
- D3D11_ReadShaderContents(const wstring & shaderName, vector<char> & out)
- {
- wstring fileName;
- #if WINAPI_FAMILY == WINAPI_FAMILY_APP
- fileName = SDL_WinRTGetFSPathUNICODE(SDL_WINRT_PATH_INSTALLED_LOCATION);
- fileName += L"\\SDL_VS2012_WinRT\\";
- #elif WINAPI_FAMILY == WINAPI_PHONE_APP
- fileName = SDL_WinRTGetFSPathUNICODE(SDL_WINRT_PATH_INSTALLED_LOCATION);
- fileName += L"\\";
- #endif
- // TODO, WinRT: test Direct3D 11.1 shader loading on Win32
- fileName += shaderName;
- return D3D11_ReadFileContents(fileName, out);
- }
- static HRESULT
- D3D11_LoadPixelShader(SDL_Renderer * renderer,
- const wstring & shaderName,
- ID3D11PixelShader ** shaderOutput)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- vector<char> fileData;
-
- if (!D3D11_ReadShaderContents(shaderName, fileData)) {
- SDL_SetError("Unable to open SDL's pixel shader file.");
- return E_FAIL;
- }
- result = data->d3dDevice->CreatePixelShader(
- &fileData[0],
- fileData.size(),
- nullptr,
- shaderOutput
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- return S_OK;
- }
- static HRESULT
- D3D11_CreateBlendMode(SDL_Renderer * renderer,
- BOOL enableBlending,
- D3D11_BLEND srcBlend,
- D3D11_BLEND destBlend,
- ID3D11BlendState ** blendStateOutput)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- D3D11_BLEND_DESC blendDesc;
- memset(&blendDesc, 0, sizeof(blendDesc));
- blendDesc.AlphaToCoverageEnable = FALSE;
- blendDesc.IndependentBlendEnable = FALSE;
- blendDesc.RenderTarget[0].BlendEnable = enableBlending;
- blendDesc.RenderTarget[0].SrcBlend = srcBlend;
- blendDesc.RenderTarget[0].DestBlend = destBlend;
- blendDesc.RenderTarget[0].BlendOp = D3D11_BLEND_OP_ADD;
- blendDesc.RenderTarget[0].SrcBlendAlpha = D3D11_BLEND_ONE;
- blendDesc.RenderTarget[0].DestBlendAlpha = D3D11_BLEND_ZERO;
- blendDesc.RenderTarget[0].BlendOpAlpha = D3D11_BLEND_OP_ADD;
- blendDesc.RenderTarget[0].RenderTargetWriteMask = D3D11_COLOR_WRITE_ENABLE_ALL;
- result = data->d3dDevice->CreateBlendState(&blendDesc, blendStateOutput);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- return S_OK;
- }
- // Create resources that depend on the device.
- HRESULT
- D3D11_CreateDeviceResources(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- // This flag adds support for surfaces with a different color channel ordering
- // than the API default. It is required for compatibility with Direct2D.
- UINT creationFlags = D3D11_CREATE_DEVICE_BGRA_SUPPORT;
- // Make sure Direct3D's debugging feature gets used, if the app requests it.
- const char *hint = SDL_GetHint(SDL_HINT_RENDER_DIRECT3D11_DEBUG);
- if (hint) {
- if (*hint == '1') {
- creationFlags |= D3D11_CREATE_DEVICE_DEBUG;
- }
- }
- // This array defines the set of DirectX hardware feature levels this app will support.
- // Note the ordering should be preserved.
- // Don't forget to declare your application's minimum required feature level in its
- // description. All applications are assumed to support 9.1 unless otherwise stated.
- D3D_FEATURE_LEVEL featureLevels[] =
- {
- D3D_FEATURE_LEVEL_11_1,
- D3D_FEATURE_LEVEL_11_0,
- D3D_FEATURE_LEVEL_10_1,
- D3D_FEATURE_LEVEL_10_0,
- D3D_FEATURE_LEVEL_9_3,
- D3D_FEATURE_LEVEL_9_2,
- D3D_FEATURE_LEVEL_9_1
- };
- // Create the Direct3D 11 API device object and a corresponding context.
- ComPtr<ID3D11Device> device;
- ComPtr<ID3D11DeviceContext> context;
- HRESULT result = S_OK;
- result = D3D11CreateDevice(
- nullptr, // Specify nullptr to use the default adapter.
- D3D_DRIVER_TYPE_HARDWARE,
- nullptr,
- creationFlags, // Set set debug and Direct2D compatibility flags.
- featureLevels, // List of feature levels this app can support.
- ARRAYSIZE(featureLevels),
- D3D11_SDK_VERSION, // Always set this to D3D11_SDK_VERSION for Windows Store apps.
- &device, // Returns the Direct3D device created.
- &data->featureLevel, // Returns feature level of device created.
- &context // Returns the device immediate context.
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- // Get the Direct3D 11.1 API device and context interfaces.
- Microsoft::WRL::ComPtr<ID3D11Device1> d3dDevice1;
- result = device.As(&(data->d3dDevice));
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- result = context.As(&data->d3dContext);
- if (FAILED(result)) {
- return result;
- }
- //
- // Make note of the maximum texture size
- // Max texture sizes are documented on MSDN, at:
- // http://msdn.microsoft.com/en-us/library/windows/apps/ff476876.aspx
- //
- switch (data->d3dDevice->GetFeatureLevel()) {
- case D3D_FEATURE_LEVEL_11_1:
- case D3D_FEATURE_LEVEL_11_0:
- renderer->info.max_texture_width = renderer->info.max_texture_height = 16384;
- break;
- case D3D_FEATURE_LEVEL_10_1:
- case D3D_FEATURE_LEVEL_10_0:
- renderer->info.max_texture_width = renderer->info.max_texture_height = 8192;
- break;
- case D3D_FEATURE_LEVEL_9_3:
- renderer->info.max_texture_width = renderer->info.max_texture_height = 4096;
- break;
- case D3D_FEATURE_LEVEL_9_2:
- case D3D_FEATURE_LEVEL_9_1:
- renderer->info.max_texture_width = renderer->info.max_texture_height = 2048;
- break;
- }
- //
- // Load in SDL's one and only vertex shader:
- //
- vector<char> fileData;
- if (!D3D11_ReadShaderContents(L"SDL_D3D11_VertexShader_Default.cso", fileData)) {
- SDL_SetError("Unable to open SDL's vertex shader file.");
- return E_FAIL;
- }
- result = data->d3dDevice->CreateVertexShader(
- &fileData[0],
- fileData.size(),
- nullptr,
- &data->vertexShader
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- //
- // Create an input layout for SDL's vertex shader:
- //
- const D3D11_INPUT_ELEMENT_DESC vertexDesc[] =
- {
- { "POSITION", 0, DXGI_FORMAT_R32G32B32_FLOAT, 0, 0, D3D11_INPUT_PER_VERTEX_DATA, 0 },
- { "TEXCOORD", 0, DXGI_FORMAT_R32G32_FLOAT, 0, 12, D3D11_INPUT_PER_VERTEX_DATA, 0 },
- { "COLOR", 0, DXGI_FORMAT_R32G32B32A32_FLOAT, 0, 20, D3D11_INPUT_PER_VERTEX_DATA, 0 },
- };
- result = data->d3dDevice->CreateInputLayout(
- vertexDesc,
- ARRAYSIZE(vertexDesc),
- &fileData[0],
- fileData.size(),
- &data->inputLayout
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- //
- // Load in SDL's pixel shaders
- //
- result = D3D11_LoadPixelShader(renderer, L"SDL_D3D11_PixelShader_TextureColored.cso", &data->texturePixelShader);
- if (FAILED(result)) {
- // D3D11_LoadPixelShader will have aleady set the SDL error
- return result;
- }
- result = D3D11_LoadPixelShader(renderer, L"SDL_D3D11_PixelShader_FixedColor.cso", &data->colorPixelShader);
- if (FAILED(result)) {
- // D3D11_LoadPixelShader will have aleady set the SDL error
- return result;
- }
- //
- // Setup space to hold vertex shader constants:
- //
- CD3D11_BUFFER_DESC constantBufferDesc(sizeof(SDL_VertexShaderConstants), D3D11_BIND_CONSTANT_BUFFER);
- result = data->d3dDevice->CreateBuffer(
- &constantBufferDesc,
- nullptr,
- &data->vertexShaderConstants
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- //
- // Make sure that the vertex buffer, if already created, gets freed.
- // It will be recreated later.
- //
- data->vertexBuffer = nullptr;
- //
- // Create samplers to use when drawing textures:
- //
- D3D11_SAMPLER_DESC samplerDesc;
- samplerDesc.Filter = SDL_D3D11_NEAREST_PIXEL_FILTER;
- samplerDesc.AddressU = D3D11_TEXTURE_ADDRESS_CLAMP;
- samplerDesc.AddressV = D3D11_TEXTURE_ADDRESS_CLAMP;
- samplerDesc.AddressW = D3D11_TEXTURE_ADDRESS_CLAMP;
- samplerDesc.MipLODBias = 0.0f;
- samplerDesc.MaxAnisotropy = 1;
- samplerDesc.ComparisonFunc = D3D11_COMPARISON_ALWAYS;
- samplerDesc.BorderColor[0] = 0.0f;
- samplerDesc.BorderColor[1] = 0.0f;
- samplerDesc.BorderColor[2] = 0.0f;
- samplerDesc.BorderColor[3] = 0.0f;
- samplerDesc.MinLOD = 0.0f;
- samplerDesc.MaxLOD = D3D11_FLOAT32_MAX;
- result = data->d3dDevice->CreateSamplerState(
- &samplerDesc,
- &data->nearestPixelSampler
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- samplerDesc.Filter = SDL_D3D11_LINEAR_FILTER;
- result = data->d3dDevice->CreateSamplerState(
- &samplerDesc,
- &data->linearSampler
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- //
- // Setup the Direct3D rasterizer
- //
- D3D11_RASTERIZER_DESC rasterDesc;
- memset(&rasterDesc, 0, sizeof(rasterDesc));
- rasterDesc.AntialiasedLineEnable = false;
- rasterDesc.CullMode = D3D11_CULL_NONE;
- rasterDesc.DepthBias = 0;
- rasterDesc.DepthBiasClamp = 0.0f;
- rasterDesc.DepthClipEnable = true;
- rasterDesc.FillMode = D3D11_FILL_SOLID;
- rasterDesc.FrontCounterClockwise = false;
- rasterDesc.MultisampleEnable = false;
- rasterDesc.ScissorEnable = false;
- rasterDesc.SlopeScaledDepthBias = 0.0f;
- result = data->d3dDevice->CreateRasterizerState(&rasterDesc, &data->mainRasterizer);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- //
- // Create blending states:
- //
- result = D3D11_CreateBlendMode(
- renderer,
- TRUE,
- D3D11_BLEND_SRC_ALPHA,
- D3D11_BLEND_INV_SRC_ALPHA,
- &data->blendModeBlend);
- if (FAILED(result)) {
- // D3D11_CreateBlendMode will set the SDL error, if it fails
- return result;
- }
- result = D3D11_CreateBlendMode(
- renderer,
- TRUE,
- D3D11_BLEND_SRC_ALPHA,
- D3D11_BLEND_ONE,
- &data->blendModeAdd);
- if (FAILED(result)) {
- // D3D11_CreateBlendMode will set the SDL error, if it fails
- return result;
- }
- result = D3D11_CreateBlendMode(
- renderer,
- TRUE,
- D3D11_BLEND_ZERO,
- D3D11_BLEND_SRC_COLOR,
- &data->blendModeMod);
- if (FAILED(result)) {
- // D3D11_CreateBlendMode will set the SDL error, if it fails
- return result;
- }
- //
- // All done!
- //
- return S_OK;
- }
- #ifdef __WINRT__
- static ABI::Windows::UI::Core::ICoreWindow *
- D3D11_GetCoreWindowFromSDLRenderer(SDL_Renderer * renderer)
- {
- SDL_Window * sdlWindow = renderer->window;
- if ( ! renderer->window ) {
- return nullptr;
- }
- SDL_SysWMinfo sdlWindowInfo;
- SDL_VERSION(&sdlWindowInfo.version);
- if ( ! SDL_GetWindowWMInfo(sdlWindow, &sdlWindowInfo) ) {
- return nullptr;
- }
- if (sdlWindowInfo.subsystem != SDL_SYSWM_WINRT) {
- return nullptr;
- }
- if ( ! sdlWindowInfo.info.winrt.window ) {
- return nullptr;
- }
- ABI::Windows::UI::Core::ICoreWindow * coreWindow = nullptr;
- if (FAILED(sdlWindowInfo.info.winrt.window->QueryInterface(&coreWindow))) {
- return nullptr;
- }
- return coreWindow;
- }
- // Method to convert a length in device-independent pixels (DIPs) to a length in physical pixels.
- static float
- D3D11_ConvertDipsToPixels(float dips)
- {
- static const float dipsPerInch = 96.0f;
- return floor(dips * DisplayProperties::LogicalDpi / dipsPerInch + 0.5f); // Round to nearest integer.
- }
- #endif
- #if WINAPI_FAMILY == WINAPI_FAMILY_APP
- // TODO, WinRT, XAML: get the ISwapChainBackgroundPanelNative from something other than a global var
- extern ISwapChainBackgroundPanelNative * WINRT_GlobalSwapChainBackgroundPanelNative;
- #endif
- static DXGI_MODE_ROTATION
- D3D11_GetRotationForOrientation(Windows::Graphics::Display::DisplayOrientations orientation)
- {
- switch (orientation)
- {
- #if WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- //
- // Windows Phone rotations
- //
- case DisplayOrientations::Landscape:
- return DXGI_MODE_ROTATION_ROTATE90;
- case DisplayOrientations::Portrait:
- return DXGI_MODE_ROTATION_IDENTITY;
- case DisplayOrientations::LandscapeFlipped:
- return DXGI_MODE_ROTATION_ROTATE270;
- case DisplayOrientations::PortraitFlipped:
- return DXGI_MODE_ROTATION_ROTATE180;
- #else
- //
- // Non-Windows-Phone rotations (ex: Windows 8, Windows RT)
- //
- case DisplayOrientations::Landscape:
- return DXGI_MODE_ROTATION_IDENTITY;
- case DisplayOrientations::Portrait:
- return DXGI_MODE_ROTATION_ROTATE270;
- case DisplayOrientations::LandscapeFlipped:
- return DXGI_MODE_ROTATION_ROTATE180;
- case DisplayOrientations::PortraitFlipped:
- return DXGI_MODE_ROTATION_ROTATE90;
- #endif // WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- default:
- return DXGI_MODE_ROTATION_UNSPECIFIED;
- }
- }
- // Initialize all resources that change when the window's size changes.
- // TODO, WinRT: get D3D11_CreateWindowSizeDependentResources working on Win32
- HRESULT
- D3D11_CreateWindowSizeDependentResources(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- ABI::Windows::UI::Core::ICoreWindow * coreWindow = D3D11_GetCoreWindowFromSDLRenderer(renderer);
- // Store the window bounds so the next time we get a SizeChanged event we can
- // avoid rebuilding everything if the size is identical.
- ABI::Windows::Foundation::Rect nativeWindowBounds;
- if (coreWindow) {
- result = coreWindow->get_Bounds(&nativeWindowBounds);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__", Get Window Bounds", result);
- return result;
- }
- } else {
- // TODO, WinRT, XAML: clean up window-bounds code in D3D11_CreateWindowSizeDependentResources
- SDL_DisplayMode displayMode;
- if (SDL_GetDesktopDisplayMode(0, &displayMode) < 0) {
- SDL_SetError(__FUNCTION__", Get Window Bounds (XAML): Unable to retrieve the native window's size");
- return E_FAIL;
- }
- nativeWindowBounds.Width = (FLOAT) displayMode.w;
- nativeWindowBounds.Height = (FLOAT) displayMode.h;
- }
- // TODO, WinRT, XAML: see if window/control sizes are in DIPs, or something else. If something else, then adjust renderer size tracking accordingly.
- data->windowSizeInDIPs.x = nativeWindowBounds.Width;
- data->windowSizeInDIPs.y = nativeWindowBounds.Height;
- // Calculate the necessary swap chain and render target size in pixels.
- float windowWidth = D3D11_ConvertDipsToPixels(data->windowSizeInDIPs.x);
- float windowHeight = D3D11_ConvertDipsToPixels(data->windowSizeInDIPs.y);
- // The width and height of the swap chain must be based on the window's
- // landscape-oriented width and height. If the window is in a portrait
- // orientation, the dimensions must be reversed.
- data->orientation = DisplayProperties::CurrentOrientation;
- #if WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- const bool swapDimensions = false;
- #else
- const bool swapDimensions =
- data->orientation == DisplayOrientations::Portrait ||
- data->orientation == DisplayOrientations::PortraitFlipped;
- #endif
- data->renderTargetSize.x = swapDimensions ? windowHeight : windowWidth;
- data->renderTargetSize.y = swapDimensions ? windowWidth : windowHeight;
- if(data->swapChain != nullptr)
- {
- // If the swap chain already exists, resize it.
- result = data->swapChain->ResizeBuffers(
- 2, // Double-buffered swap chain.
- static_cast<UINT>(data->renderTargetSize.x),
- static_cast<UINT>(data->renderTargetSize.y),
- DXGI_FORMAT_B8G8R8A8_UNORM,
- 0
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- }
- else
- {
- const bool usingXAML = (coreWindow == nullptr);
- // Otherwise, create a new one using the same adapter as the existing Direct3D device.
- DXGI_SWAP_CHAIN_DESC1 swapChainDesc = {0};
- swapChainDesc.Width = static_cast<UINT>(data->renderTargetSize.x); // Match the size of the window.
- swapChainDesc.Height = static_cast<UINT>(data->renderTargetSize.y);
- swapChainDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; // This is the most common swap chain format.
- swapChainDesc.Stereo = false;
- swapChainDesc.SampleDesc.Count = 1; // Don't use multi-sampling.
- swapChainDesc.SampleDesc.Quality = 0;
- swapChainDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT;
- swapChainDesc.BufferCount = 2; // Use double-buffering to minimize latency.
- #if WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- swapChainDesc.Scaling = DXGI_SCALING_STRETCH; // On phone, only stretch and aspect-ratio stretch scaling are allowed.
- swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD; // On phone, no swap effects are supported.
- #else
- if (usingXAML) {
- swapChainDesc.Scaling = DXGI_SCALING_STRETCH;
- } else {
- swapChainDesc.Scaling = DXGI_SCALING_NONE;
- }
- swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL; // All Windows Store apps must use this SwapEffect.
- #endif
- swapChainDesc.Flags = 0;
- ComPtr<IDXGIDevice1> dxgiDevice;
- result = data->d3dDevice.As(&dxgiDevice);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- ComPtr<IDXGIAdapter> dxgiAdapter;
- result = dxgiDevice->GetAdapter(&dxgiAdapter);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- ComPtr<IDXGIFactory2> dxgiFactory;
- result = dxgiAdapter->GetParent(
- __uuidof(IDXGIFactory2),
- &dxgiFactory
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- if (usingXAML) {
- result = dxgiFactory->CreateSwapChainForComposition(
- data->d3dDevice.Get(),
- &swapChainDesc,
- nullptr,
- &data->swapChain);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", CreateSwapChainForComposition", result);
- return result;
- }
- #if WINAPI_FAMILY == WINAPI_FAMILY_APP
- result = WINRT_GlobalSwapChainBackgroundPanelNative->SetSwapChain(data->swapChain.Get());
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", ISwapChainBackgroundPanelNative::SetSwapChain", result);
- return result;
- }
- #else
- SDL_SetError(__FUNCTION__ ", XAML support is not yet available for Windows Phone");
- return E_FAIL;
- #endif
- } else {
- IUnknown * coreWindowAsIUnknown = nullptr;
- result = coreWindow->QueryInterface(&coreWindowAsIUnknown);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", CoreWindow to IUnknown", result);
- return result;
- }
- result = dxgiFactory->CreateSwapChainForCoreWindow(
- data->d3dDevice.Get(),
- coreWindowAsIUnknown,
- &swapChainDesc,
- nullptr, // Allow on all displays.
- &data->swapChain
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- }
-
- // Ensure that DXGI does not queue more than one frame at a time. This both reduces latency and
- // ensures that the application will only render after each VSync, minimizing power consumption.
- result = dxgiDevice->SetMaximumFrameLatency(1);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- }
-
- #if WINAPI_FAMILY != WINAPI_FAMILY_PHONE_APP
- // Set the proper orientation for the swap chain, and generate the
- // 3D matrix transformation for rendering to the rotated swap chain.
- //
- // To note, the call for this, IDXGISwapChain1::SetRotation, is not necessary
- // on Windows Phone, nor is it supported there. It's only needed in Windows 8/RT.
- DXGI_MODE_ROTATION rotation = D3D11_GetRotationForOrientation(data->orientation);
- result = data->swapChain->SetRotation(rotation);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- #endif
- // Create a render target view of the swap chain back buffer.
- ComPtr<ID3D11Texture2D> backBuffer;
- result = data->swapChain->GetBuffer(
- 0,
- __uuidof(ID3D11Texture2D),
- &backBuffer
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- result = data->d3dDevice->CreateRenderTargetView(
- backBuffer.Get(),
- nullptr,
- &data->mainRenderTargetView
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- if (D3D11_UpdateViewport(renderer) != 0) {
- // D3D11_UpdateViewport will set the SDL error if it fails.
- return E_FAIL;
- }
- return S_OK;
- }
- // This method is called when the window's size changes.
- HRESULT
- D3D11_UpdateForWindowSizeChange(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- ABI::Windows::UI::Core::ICoreWindow * coreWindow = D3D11_GetCoreWindowFromSDLRenderer(renderer);
- ABI::Windows::Foundation::Rect coreWindowBounds;
- result = coreWindow->get_Bounds(&coreWindowBounds);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Get Window Bounds", result);
- return result;
- }
- if (coreWindowBounds.Width != data->windowSizeInDIPs.x ||
- coreWindowBounds.Height != data->windowSizeInDIPs.y ||
- data->orientation != DisplayProperties::CurrentOrientation)
- {
- ID3D11RenderTargetView* nullViews[] = {nullptr};
- data->d3dContext->OMSetRenderTargets(ARRAYSIZE(nullViews), nullViews, nullptr);
- data->mainRenderTargetView = nullptr;
- data->d3dContext->Flush();
- result = D3D11_CreateWindowSizeDependentResources(renderer);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- }
- return S_OK;
- }
- HRESULT
- D3D11_HandleDeviceLost(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- // Reset these member variables to ensure that D3D11_UpdateForWindowSizeChange recreates all resources.
- data->windowSizeInDIPs.x = 0;
- data->windowSizeInDIPs.y = 0;
- data->swapChain = nullptr;
- result = D3D11_CreateDeviceResources(renderer);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- result = D3D11_UpdateForWindowSizeChange(renderer);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return result;
- }
- return S_OK;
- }
- static void
- D3D11_WindowEvent(SDL_Renderer * renderer, const SDL_WindowEvent *event)
- {
- //D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- if (event->event == SDL_WINDOWEVENT_SIZE_CHANGED) {
- D3D11_UpdateForWindowSizeChange(renderer);
- }
- }
- static D3D11_FILTER
- GetScaleQuality(void)
- {
- const char *hint = SDL_GetHint(SDL_HINT_RENDER_SCALE_QUALITY);
- if (!hint || *hint == '0' || SDL_strcasecmp(hint, "nearest") == 0) {
- return SDL_D3D11_NEAREST_PIXEL_FILTER;
- } else /* if (*hint == '1' || SDL_strcasecmp(hint, "linear") == 0) */ {
- return SDL_D3D11_LINEAR_FILTER;
- }
- }
- static int
- D3D11_CreateTexture(SDL_Renderer * renderer, SDL_Texture * texture)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData;
- HRESULT result;
- DXGI_FORMAT textureFormat = SDLPixelFormatToDXGIFormat(texture->format);
- if (textureFormat == SDL_PIXELFORMAT_UNKNOWN) {
- SDL_SetError("%s, An unsupported SDL pixel format (0x%x) was specified",
- __FUNCTION__, texture->format);
- return -1;
- }
- textureData = new D3D11_TextureData;
- if (!textureData) {
- SDL_OutOfMemory();
- return -1;
- }
- textureData->pixelFormat = SDL_AllocFormat(texture->format);
- textureData->lockedTexturePosition = XMINT2(0, 0);
- textureData->scaleMode = GetScaleQuality();
- texture->driverdata = textureData;
- D3D11_TEXTURE2D_DESC textureDesc = {0};
- textureDesc.Width = texture->w;
- textureDesc.Height = texture->h;
- textureDesc.MipLevels = 1;
- textureDesc.ArraySize = 1;
- textureDesc.Format = textureFormat;
- textureDesc.SampleDesc.Count = 1;
- textureDesc.SampleDesc.Quality = 0;
- textureDesc.MiscFlags = 0;
- if (texture->access == SDL_TEXTUREACCESS_STREAMING) {
- textureDesc.Usage = D3D11_USAGE_DYNAMIC;
- textureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
- } else {
- textureDesc.Usage = D3D11_USAGE_DEFAULT;
- textureDesc.CPUAccessFlags = 0;
- }
- if (texture->access == SDL_TEXTUREACCESS_TARGET) {
- textureDesc.BindFlags = D3D11_BIND_SHADER_RESOURCE | D3D11_BIND_RENDER_TARGET;
- } else {
- textureDesc.BindFlags = D3D11_BIND_SHADER_RESOURCE;
- }
- #if 0
- // Fill the texture with a non-black color, for debugging purposes:
- const int numPixels = textureDesc.Width * textureDesc.Height;
- const int pixelSizeInBytes = textureData->pixelFormat->BytesPerPixel;
- std::vector<uint8> initialTexturePixels(numPixels * pixelSizeInBytes, 0x00);
- for (int i = 0; i < (numPixels * pixelSizeInBytes); i += pixelSizeInBytes) {
- initialTexturePixels[i+0] = 0xff;
- initialTexturePixels[i+1] = 0xff;
- initialTexturePixels[i+2] = 0x00;
- initialTexturePixels[i+3] = 0xff;
- }
- D3D11_SUBRESOURCE_DATA initialTextureData = {0};
- initialTextureData.pSysMem = (void *)&(initialTexturePixels[0]);
- initialTextureData.SysMemPitch = textureDesc.Width * pixelSizeInBytes;
- initialTextureData.SysMemSlicePitch = numPixels * pixelSizeInBytes;
- #endif
- result = rendererData->d3dDevice->CreateTexture2D(
- &textureDesc,
- NULL, // &initialTextureData,
- &textureData->mainTexture
- );
- if (FAILED(result)) {
- D3D11_DestroyTexture(renderer, texture);
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return -1;
- }
- if (texture->access & SDL_TEXTUREACCESS_TARGET) {
- D3D11_RENDER_TARGET_VIEW_DESC renderTargetViewDesc;
- renderTargetViewDesc.Format = textureDesc.Format;
- renderTargetViewDesc.ViewDimension = D3D11_RTV_DIMENSION_TEXTURE2D;
- renderTargetViewDesc.Texture2D.MipSlice = 0;
- result = rendererData->d3dDevice->CreateRenderTargetView(
- textureData->mainTexture.Get(),
- &renderTargetViewDesc,
- &textureData->mainTextureRenderTargetView);
- if (FAILED(result)) {
- D3D11_DestroyTexture(renderer, texture);
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return -1;
- }
- }
- D3D11_SHADER_RESOURCE_VIEW_DESC resourceViewDesc;
- resourceViewDesc.Format = textureDesc.Format;
- resourceViewDesc.ViewDimension = D3D11_SRV_DIMENSION_TEXTURE2D;
- resourceViewDesc.Texture2D.MostDetailedMip = 0;
- resourceViewDesc.Texture2D.MipLevels = textureDesc.MipLevels;
- result = rendererData->d3dDevice->CreateShaderResourceView(
- textureData->mainTexture.Get(),
- &resourceViewDesc,
- &textureData->mainTextureResourceView
- );
- if (FAILED(result)) {
- D3D11_DestroyTexture(renderer, texture);
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return -1;
- }
- return 0;
- }
- static void
- D3D11_DestroyTexture(SDL_Renderer * renderer,
- SDL_Texture * texture)
- {
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- if (textureData) {
- if (textureData->pixelFormat) {
- SDL_FreeFormat(textureData->pixelFormat);
- textureData->pixelFormat = NULL;
- }
- delete textureData;
- texture->driverdata = NULL;
- }
- }
- static int
- D3D11_UpdateTexture(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * rect, const void * srcPixels,
- int srcPitch)
- {
- // Lock the texture, retrieving a buffer to write pixel data to:
- void * destPixels = NULL;
- int destPitch = 0;
- if (D3D11_LockTexture(renderer, texture, rect, &destPixels, &destPitch) != 0) {
- // An error is already set. Attach some info to it, then return to
- // the caller.
- std::string errorMessage = string(__FUNCTION__ ", Lock Texture Failed: ") + SDL_GetError();
- SDL_SetError(errorMessage.c_str());
- return -1;
- }
- // Copy pixel data to the locked texture's memory:
- for (int y = 0; y < rect->h; ++y) {
- memcpy(
- ((Uint8 *)destPixels) + (destPitch * y),
- ((Uint8 *)srcPixels) + (srcPitch * y),
- srcPitch
- );
- }
- // Commit the texture's memory back to Direct3D:
- D3D11_UnlockTexture(renderer, texture);
- // Return to the caller:
- return 0;
- }
- static int
- D3D11_LockTexture(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * rect, void **pixels, int *pitch)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- HRESULT result = S_OK;
- if (textureData->stagingTexture) {
- SDL_SetError("texture is already locked");
- return -1;
- }
-
- // Create a 'staging' texture, which will be used to write to a portion
- // of the main texture. This is necessary, as Direct3D 11.1 does not
- // have the ability to write a CPU-bound pixel buffer to a rectangular
- // subrect of a texture. Direct3D 11.1 can, however, write a pixel
- // buffer to an entire texture, hence the use of a staging texture.
- D3D11_TEXTURE2D_DESC stagingTextureDesc;
- textureData->mainTexture->GetDesc(&stagingTextureDesc);
- stagingTextureDesc.Width = rect->w;
- stagingTextureDesc.Height = rect->h;
- stagingTextureDesc.BindFlags = 0;
- stagingTextureDesc.MiscFlags = 0;
- stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
- stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
- result = rendererData->d3dDevice->CreateTexture2D(
- &stagingTextureDesc,
- NULL,
- &textureData->stagingTexture);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Create Staging Texture", result);
- return -1;
- }
- // Get a write-only pointer to data in the staging texture:
- D3D11_MAPPED_SUBRESOURCE textureMemory = {0};
- result = rendererData->d3dContext->Map(
- textureData->stagingTexture.Get(),
- D3D11CalcSubresource(0, 0, 0),
- D3D11_MAP_WRITE,
- 0,
- &textureMemory
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Map Staging Texture", result);
- textureData->stagingTexture = nullptr;
- return -1;
- }
- // Make note of where the staging texture will be written to (on a
- // call to SDL_UnlockTexture):
- textureData->lockedTexturePosition = XMINT2(rect->x, rect->y);
- // Make sure the caller has information on the texture's pixel buffer,
- // then return:
- *pixels = textureMemory.pData;
- *pitch = textureMemory.RowPitch;
- return 0;
- }
- static void
- D3D11_UnlockTexture(SDL_Renderer * renderer, SDL_Texture * texture)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- // Commit the pixel buffer's changes back to the staging texture:
- rendererData->d3dContext->Unmap(
- textureData->stagingTexture.Get(),
- 0);
- // Copy the staging texture's contents back to the main texture:
- rendererData->d3dContext->CopySubresourceRegion(
- textureData->mainTexture.Get(),
- D3D11CalcSubresource(0, 0, 0),
- textureData->lockedTexturePosition.x,
- textureData->lockedTexturePosition.y,
- 0,
- textureData->stagingTexture.Get(),
- D3D11CalcSubresource(0, 0, 0),
- NULL);
- // Clean up and return:
- textureData->stagingTexture = nullptr;
- textureData->lockedTexturePosition = XMINT2(0, 0);
- }
- static int
- D3D11_SetRenderTarget(SDL_Renderer * renderer, SDL_Texture * texture)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- if (texture == NULL) {
- rendererData->currentOffscreenRenderTargetView = nullptr;
- return 0;
- }
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- if (!textureData->mainTextureRenderTargetView) {
- std::string errorMessage = string(__FUNCTION__) + ": specified texture is not a render target";
- SDL_SetError(errorMessage.c_str());
- return -1;
- }
- rendererData->currentOffscreenRenderTargetView = textureData->mainTextureRenderTargetView;
- return 0;
- }
- static int
- D3D11_UpdateViewport(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- if (renderer->viewport.w == 0 || renderer->viewport.h == 0) {
- // If the viewport is empty, assume that it is because
- // SDL_CreateRenderer is calling it, and will call it again later
- // with a non-empty viewport.
- return 0;
- }
- // Make sure the SDL viewport gets rotated to that of the physical display's orientation.
- // Keep in mind here that the Y-axis will be been inverted (from Direct3D's
- // default coordinate system) so rotations will be done in the opposite
- // direction of the DXGI_MODE_ROTATION enumeration.
- switch (D3D11_GetRotationForOrientation(data->orientation))
- {
- case DXGI_MODE_ROTATION_IDENTITY:
- XMStoreFloat4x4(&data->vertexShaderConstantsData.projection, XMMatrixIdentity());
- break;
- case DXGI_MODE_ROTATION_ROTATE270:
- XMStoreFloat4x4(&data->vertexShaderConstantsData.projection, XMMatrixRotationZ(XM_PIDIV2));
- break;
- case DXGI_MODE_ROTATION_ROTATE180:
- XMStoreFloat4x4(&data->vertexShaderConstantsData.projection, XMMatrixRotationZ(XM_PI));
- break;
- case DXGI_MODE_ROTATION_ROTATE90:
- XMStoreFloat4x4(&data->vertexShaderConstantsData.projection, XMMatrixRotationZ(-XM_PIDIV2));
- break;
- default:
- SDL_SetError("An unknown DisplayOrientation is being used");
- return -1;
- }
- //
- // Update the view matrix
- //
- float viewportWidth = (float) renderer->viewport.w;
- float viewportHeight = (float) renderer->viewport.h;
- XMStoreFloat4x4(&data->vertexShaderConstantsData.view,
- XMMatrixMultiply(
- XMMatrixScaling(2.0f / viewportWidth, 2.0f / viewportHeight, 1.0f),
- XMMatrixMultiply(
- XMMatrixTranslation(-1, -1, 0),
- XMMatrixRotationX(XM_PI)
- )));
- #if 0
- data->vertexShaderConstantsData.view = XMMatrixIdentity();
- #endif
- //
- // Reset the model matrix
- //
- XMStoreFloat4x4(&data->vertexShaderConstantsData.model, XMMatrixIdentity());
- //
- // Update the Direct3D viewport, which seems to be aligned to the
- // swap buffer's coordinate space, which is always in either
- // a landscape mode, for all Windows 8/RT devices, or a portrait mode,
- // for Windows Phone devices.
- //
- SDL_FRect orientationAlignedViewport;
- #if WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- const bool swapDimensions =
- data->orientation == DisplayOrientations::Landscape ||
- data->orientation == DisplayOrientations::LandscapeFlipped;
- #else
- const bool swapDimensions =
- data->orientation == DisplayOrientations::Portrait ||
- data->orientation == DisplayOrientations::PortraitFlipped;
- #endif
- if (swapDimensions) {
- orientationAlignedViewport.x = (float) renderer->viewport.y;
- orientationAlignedViewport.y = (float) renderer->viewport.x;
- orientationAlignedViewport.w = (float) renderer->viewport.h;
- orientationAlignedViewport.h = (float) renderer->viewport.w;
- } else {
- orientationAlignedViewport.x = (float) renderer->viewport.x;
- orientationAlignedViewport.y = (float) renderer->viewport.y;
- orientationAlignedViewport.w = (float) renderer->viewport.w;
- orientationAlignedViewport.h = (float) renderer->viewport.h;
- }
- // TODO, WinRT: get custom viewports working with non-Landscape modes (Portrait, PortraitFlipped, and LandscapeFlipped)
- D3D11_VIEWPORT viewport;
- memset(&viewport, 0, sizeof(viewport));
- viewport.TopLeftX = orientationAlignedViewport.x;
- viewport.TopLeftY = orientationAlignedViewport.y;
- viewport.Width = orientationAlignedViewport.w;
- viewport.Height = orientationAlignedViewport.h;
- viewport.MinDepth = 0.0f;
- viewport.MaxDepth = 1.0f;
- data->d3dContext->RSSetViewports(1, &viewport);
- #if 0
- SDL_Log("%s, oav={%.0f,%.0f,%.0f,%.0f}, rend={%.0f,%.0f}\n",
- __FUNCTION__,
- orientationAlignedViewport.x,
- orientationAlignedViewport.y,
- orientationAlignedViewport.w,
- orientationAlignedViewport.h,
- data->renderTargetSize.x,
- data->renderTargetSize.y);
- #endif
- return 0;
- }
- static int
- D3D11_UpdateClipRect(SDL_Renderer * renderer)
- {
- // TODO, WinRT: implement D3D11_UpdateClipRect
- return 0;
- }
- static ComPtr<ID3D11RenderTargetView> &
- D3D11_GetCurrentRenderTargetView(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- if (data->currentOffscreenRenderTargetView) {
- return data->currentOffscreenRenderTargetView;
- } else {
- return data->mainRenderTargetView;
- }
- }
- static int
- D3D11_RenderClear(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- const float colorRGBA[] = {
- (renderer->r / 255.0f),
- (renderer->g / 255.0f),
- (renderer->b / 255.0f),
- (renderer->a / 255.0f)
- };
- data->d3dContext->ClearRenderTargetView(
- D3D11_GetCurrentRenderTargetView(renderer).Get(),
- colorRGBA
- );
- return 0;
- }
- static int
- D3D11_UpdateVertexBuffer(SDL_Renderer *renderer,
- const void * vertexData, unsigned int dataSizeInBytes)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- D3D11_BUFFER_DESC vertexBufferDesc;
- if (rendererData->vertexBuffer) {
- rendererData->vertexBuffer->GetDesc(&vertexBufferDesc);
- } else {
- memset(&vertexBufferDesc, 0, sizeof(vertexBufferDesc));
- }
- if (vertexBufferDesc.ByteWidth >= dataSizeInBytes) {
- rendererData->d3dContext->UpdateSubresource(rendererData->vertexBuffer.Get(), 0, NULL, vertexData, dataSizeInBytes, 0);
- } else {
- vertexBufferDesc.ByteWidth = dataSizeInBytes;
- vertexBufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER;
- D3D11_SUBRESOURCE_DATA vertexBufferData = {0};
- vertexBufferData.pSysMem = vertexData;
- vertexBufferData.SysMemPitch = 0;
- vertexBufferData.SysMemSlicePitch = 0;
- result = rendererData->d3dDevice->CreateBuffer(
- &vertexBufferDesc,
- &vertexBufferData,
- &rendererData->vertexBuffer
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, result);
- return -1;
- }
- }
- UINT stride = sizeof(VertexPositionColor);
- UINT offset = 0;
- rendererData->d3dContext->IASetVertexBuffers(
- 0,
- 1,
- rendererData->vertexBuffer.GetAddressOf(),
- &stride,
- &offset
- );
- return 0;
- }
- static void
- D3D11_RenderStartDrawOp(SDL_Renderer * renderer)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- rendererData->d3dContext->OMSetRenderTargets(
- 1,
- D3D11_GetCurrentRenderTargetView(renderer).GetAddressOf(),
- nullptr
- );
- }
- static void
- D3D11_RenderSetBlendMode(SDL_Renderer * renderer, SDL_BlendMode blendMode)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- switch (blendMode) {
- case SDL_BLENDMODE_BLEND:
- rendererData->d3dContext->OMSetBlendState(rendererData->blendModeBlend.Get(), 0, 0xFFFFFFFF);
- break;
- case SDL_BLENDMODE_ADD:
- rendererData->d3dContext->OMSetBlendState(rendererData->blendModeAdd.Get(), 0, 0xFFFFFFFF);
- break;
- case SDL_BLENDMODE_MOD:
- rendererData->d3dContext->OMSetBlendState(rendererData->blendModeMod.Get(), 0, 0xFFFFFFFF);
- break;
- case SDL_BLENDMODE_NONE:
- rendererData->d3dContext->OMSetBlendState(NULL, 0, 0xFFFFFFFF);
- break;
- }
- }
- static void
- D3D11_SetPixelShader(SDL_Renderer * renderer,
- ID3D11PixelShader * shader,
- ID3D11ShaderResourceView * shaderResource,
- ID3D11SamplerState * sampler)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- rendererData->d3dContext->PSSetShader(shader, nullptr, 0);
- rendererData->d3dContext->PSSetShaderResources(0, 1, &shaderResource);
- rendererData->d3dContext->PSSetSamplers(0, 1, &sampler);
- }
- static void
- D3D11_RenderFinishDrawOp(SDL_Renderer * renderer,
- D3D11_PRIMITIVE_TOPOLOGY primitiveTopology,
- UINT vertexCount)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- rendererData->d3dContext->UpdateSubresource(
- rendererData->vertexShaderConstants.Get(),
- 0,
- NULL,
- &rendererData->vertexShaderConstantsData,
- 0,
- 0
- );
- rendererData->d3dContext->IASetPrimitiveTopology(primitiveTopology);
- rendererData->d3dContext->IASetInputLayout(rendererData->inputLayout.Get());
- rendererData->d3dContext->VSSetShader(rendererData->vertexShader.Get(), nullptr, 0);
- rendererData->d3dContext->VSSetConstantBuffers(0, 1, rendererData->vertexShaderConstants.GetAddressOf());
- rendererData->d3dContext->RSSetState(rendererData->mainRasterizer.Get());
- rendererData->d3dContext->Draw(vertexCount, 0);
- }
- static int
- D3D11_RenderDrawPoints(SDL_Renderer * renderer,
- const SDL_FPoint * points, int count)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- float r, g, b, a;
- r = (float)(renderer->r / 255.0f);
- g = (float)(renderer->g / 255.0f);
- b = (float)(renderer->b / 255.0f);
- a = (float)(renderer->a / 255.0f);
- VertexPositionColor * vertices = SDL_stack_alloc(VertexPositionColor, count);
- for (int i = 0; i < min(count, 128); ++i) {
- const VertexPositionColor v = {XMFLOAT3(points[i].x, points[i].y, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)};
- vertices[i] = v;
- }
- D3D11_RenderStartDrawOp(renderer);
- D3D11_RenderSetBlendMode(renderer, renderer->blendMode);
- if (D3D11_UpdateVertexBuffer(renderer, vertices, (unsigned int)count * sizeof(VertexPositionColor)) != 0) {
- SDL_stack_free(vertices);
- return -1;
- }
- D3D11_SetPixelShader(
- renderer,
- rendererData->colorPixelShader.Get(),
- nullptr,
- nullptr);
- D3D11_RenderFinishDrawOp(renderer, D3D11_PRIMITIVE_TOPOLOGY_POINTLIST, count);
- SDL_stack_free(vertices);
- return 0;
- }
- static int
- D3D11_RenderDrawLines(SDL_Renderer * renderer,
- const SDL_FPoint * points, int count)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- float r, g, b, a;
- r = (float)(renderer->r / 255.0f);
- g = (float)(renderer->g / 255.0f);
- b = (float)(renderer->b / 255.0f);
- a = (float)(renderer->a / 255.0f);
- VertexPositionColor * vertices = SDL_stack_alloc(VertexPositionColor, count);
- for (int i = 0; i < count; ++i) {
- const VertexPositionColor v = {XMFLOAT3(points[i].x, points[i].y, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)};
- vertices[i] = v;
- }
- D3D11_RenderStartDrawOp(renderer);
- D3D11_RenderSetBlendMode(renderer, renderer->blendMode);
- if (D3D11_UpdateVertexBuffer(renderer, vertices, (unsigned int)count * sizeof(VertexPositionColor)) != 0) {
- SDL_stack_free(vertices);
- return -1;
- }
- D3D11_SetPixelShader(
- renderer,
- rendererData->colorPixelShader.Get(),
- nullptr,
- nullptr);
- D3D11_RenderFinishDrawOp(renderer, D3D11_PRIMITIVE_TOPOLOGY_LINESTRIP, count);
- SDL_stack_free(vertices);
- return 0;
- }
- static int
- D3D11_RenderFillRects(SDL_Renderer * renderer,
- const SDL_FRect * rects, int count)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- float r, g, b, a;
- r = (float)(renderer->r / 255.0f);
- g = (float)(renderer->g / 255.0f);
- b = (float)(renderer->b / 255.0f);
- a = (float)(renderer->a / 255.0f);
- #if 0
- // Set up a test pattern:
- SDL_FRect _rects[] = {
- {-1.1f, 1.1f, 1.1f, -1.1f},
- {-1.0f, 1.0f, 1.0f, -1.0f}, // red
- {0.0f, 1.0f, 1.0f, -1.0f}, // green
- {-1.0f, 0.0f, 1.0f, -1.0f}, // blue
- {0.0f, 0.0f, 1.0f, -1.0f} // white
- };
- count = sizeof(_rects) / sizeof(SDL_FRect);
- #define rects _rects
- #endif
- for (int i = 0; i < count; ++i) {
- D3D11_RenderStartDrawOp(renderer);
- D3D11_RenderSetBlendMode(renderer, renderer->blendMode);
- #if 0
- // Set colors for the test pattern:
- a = 1.0f;
- switch (i) {
- case 0: r = 1.0f; g = 1.0f; b = 0.0f; break;
- case 1: r = 1.0f; g = 0.0f; b = 0.0f; break;
- case 2: r = 0.0f; g = 1.0f; b = 0.0f; break;
- case 3: r = 0.0f; g = 0.0f; b = 1.0f; break;
- case 4: r = 1.0f; g = 1.0f; b = 1.0f; break;
- }
- #endif
- VertexPositionColor vertices[] = {
- {XMFLOAT3(rects[i].x, rects[i].y, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(rects[i].x, rects[i].y + rects[i].h, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(rects[i].x + rects[i].w, rects[i].y, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(rects[i].x + rects[i].w, rects[i].y + rects[i].h, 0.0f), XMFLOAT2(0.0f, 0.0f), XMFLOAT4(r, g, b, a)},
- };
- if (D3D11_UpdateVertexBuffer(renderer, vertices, sizeof(vertices)) != 0) {
- return -1;
- }
- D3D11_SetPixelShader(
- renderer,
- rendererData->colorPixelShader.Get(),
- nullptr,
- nullptr);
- D3D11_RenderFinishDrawOp(renderer, D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP, sizeof(vertices) / sizeof(VertexPositionColor));
- }
- return 0;
- }
- static ID3D11SamplerState *
- D3D11_RenderGetSampler(SDL_Renderer * renderer, SDL_Texture * texture)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- switch (textureData->scaleMode) {
- case SDL_D3D11_NEAREST_PIXEL_FILTER:
- return rendererData->nearestPixelSampler.Get();
- case SDL_D3D11_LINEAR_FILTER:
- return rendererData->linearSampler.Get();
- default:
- return NULL;
- }
- }
- static int
- D3D11_RenderCopy(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * srcrect, const SDL_FRect * dstrect)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- D3D11_RenderStartDrawOp(renderer);
- D3D11_RenderSetBlendMode(renderer, texture->blendMode);
- float minu = (float) srcrect->x / texture->w;
- float maxu = (float) (srcrect->x + srcrect->w) / texture->w;
- float minv = (float) srcrect->y / texture->h;
- float maxv = (float) (srcrect->y + srcrect->h) / texture->h;
- float r = 1.0f;
- float g = 1.0f;
- float b = 1.0f;
- float a = 1.0f;
- if (texture->modMode & SDL_TEXTUREMODULATE_COLOR) {
- r = (float)(texture->r / 255.0f);
- g = (float)(texture->g / 255.0f);
- b = (float)(texture->b / 255.0f);
- }
- if (texture->modMode & SDL_TEXTUREMODULATE_ALPHA) {
- a = (float)(texture->a / 255.0f);
- }
- VertexPositionColor vertices[] = {
- {XMFLOAT3(dstrect->x, dstrect->y, 0.0f), XMFLOAT2(minu, minv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(dstrect->x, dstrect->y + dstrect->h, 0.0f), XMFLOAT2(minu, maxv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(dstrect->x + dstrect->w, dstrect->y, 0.0f), XMFLOAT2(maxu, minv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(dstrect->x + dstrect->w, dstrect->y + dstrect->h, 0.0f), XMFLOAT2(maxu, maxv), XMFLOAT4(r, g, b, a)},
- };
- if (D3D11_UpdateVertexBuffer(renderer, vertices, sizeof(vertices)) != 0) {
- return -1;
- }
- ID3D11SamplerState *textureSampler = D3D11_RenderGetSampler(renderer, texture);
- D3D11_SetPixelShader(
- renderer,
- rendererData->texturePixelShader.Get(),
- textureData->mainTextureResourceView.Get(),
- textureSampler);
- D3D11_RenderFinishDrawOp(renderer, D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP, sizeof(vertices) / sizeof(VertexPositionColor));
- return 0;
- }
- static int
- D3D11_RenderCopyEx(SDL_Renderer * renderer, SDL_Texture * texture,
- const SDL_Rect * srcrect, const SDL_FRect * dstrect,
- const double angle, const SDL_FPoint * center, const SDL_RendererFlip flip)
- {
- D3D11_RenderData *rendererData = (D3D11_RenderData *) renderer->driverdata;
- D3D11_TextureData *textureData = (D3D11_TextureData *) texture->driverdata;
- D3D11_RenderStartDrawOp(renderer);
- D3D11_RenderSetBlendMode(renderer, texture->blendMode);
- float minu = (float) srcrect->x / texture->w;
- float maxu = (float) (srcrect->x + srcrect->w) / texture->w;
- float minv = (float) srcrect->y / texture->h;
- float maxv = (float) (srcrect->y + srcrect->h) / texture->h;
- float r = 1.0f;
- float g = 1.0f;
- float b = 1.0f;
- float a = 1.0f;
- if (texture->modMode & SDL_TEXTUREMODULATE_COLOR) {
- r = (float)(texture->r / 255.0f);
- g = (float)(texture->g / 255.0f);
- b = (float)(texture->b / 255.0f);
- }
- if (texture->modMode & SDL_TEXTUREMODULATE_ALPHA) {
- a = (float)(texture->a / 255.0f);
- }
- if (flip & SDL_FLIP_HORIZONTAL) {
- float tmp = maxu;
- maxu = minu;
- minu = tmp;
- }
- if (flip & SDL_FLIP_VERTICAL) {
- float tmp = maxv;
- maxv = minv;
- minv = tmp;
- }
- XMFLOAT4X4 oldModelMatrix = rendererData->vertexShaderConstantsData.model;
- XMStoreFloat4x4(
- &rendererData->vertexShaderConstantsData.model,
- XMMatrixMultiply(
- XMMatrixRotationZ((float)(XM_PI * (float) angle / 180.0f)),
- XMMatrixTranslation(dstrect->x + center->x, dstrect->y + center->y, 0)
- ));
- const float minx = -center->x;
- const float maxx = dstrect->w - center->x;
- const float miny = -center->y;
- const float maxy = dstrect->h - center->y;
- VertexPositionColor vertices[] = {
- {XMFLOAT3(minx, miny, 0.0f), XMFLOAT2(minu, minv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(minx, maxy, 0.0f), XMFLOAT2(minu, maxv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(maxx, miny, 0.0f), XMFLOAT2(maxu, minv), XMFLOAT4(r, g, b, a)},
- {XMFLOAT3(maxx, maxy, 0.0f), XMFLOAT2(maxu, maxv), XMFLOAT4(r, g, b, a)},
- };
- if (D3D11_UpdateVertexBuffer(renderer, vertices, sizeof(vertices)) != 0) {
- return -1;
- }
- ID3D11SamplerState *textureSampler = D3D11_RenderGetSampler(renderer, texture);
- D3D11_SetPixelShader(
- renderer,
- rendererData->texturePixelShader.Get(),
- textureData->mainTextureResourceView.Get(),
- textureSampler);
- D3D11_RenderFinishDrawOp(renderer, D3D11_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP, sizeof(vertices) / sizeof(VertexPositionColor));
- rendererData->vertexShaderConstantsData.model = oldModelMatrix;
- return 0;
- }
- static int
- D3D11_RenderReadPixels(SDL_Renderer * renderer, const SDL_Rect * rect,
- Uint32 format, void * pixels, int pitch)
- {
- D3D11_RenderData * data = (D3D11_RenderData *) renderer->driverdata;
- HRESULT result = S_OK;
- // Retrieve a pointer to the back buffer:
- ComPtr<ID3D11Texture2D> backBuffer;
- result = data->swapChain->GetBuffer(
- 0,
- __uuidof(ID3D11Texture2D),
- &backBuffer
- );
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Get Back Buffer", result);
- return -1;
- }
- // Create a staging texture to copy the screen's data to:
- ComPtr<ID3D11Texture2D> stagingTexture;
- D3D11_TEXTURE2D_DESC stagingTextureDesc;
- backBuffer->GetDesc(&stagingTextureDesc);
- stagingTextureDesc.Width = rect->w;
- stagingTextureDesc.Height = rect->h;
- stagingTextureDesc.BindFlags = 0;
- stagingTextureDesc.MiscFlags = 0;
- stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
- stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
- result = data->d3dDevice->CreateTexture2D(
- &stagingTextureDesc,
- NULL,
- &stagingTexture);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Create Staging Texture", result);
- return -1;
- }
- // Copy the desired portion of the back buffer to the staging texture:
- D3D11_BOX srcBox;
- switch (D3D11_GetRotationForOrientation(data->orientation)) {
- case DXGI_MODE_ROTATION_IDENTITY:
- srcBox.left = rect->x;
- srcBox.right = rect->x + rect->w;
- srcBox.top = rect->y;
- srcBox.bottom = rect->y + rect->h;
- break;
- case DXGI_MODE_ROTATION_ROTATE270:
- srcBox.left = rect->y;
- srcBox.right = rect->y + rect->h;
- srcBox.top = renderer->viewport.w - rect->x - rect->w;
- srcBox.bottom = renderer->viewport.w - rect->x;
- break;
- case DXGI_MODE_ROTATION_ROTATE180:
- srcBox.left = renderer->viewport.w - rect->x - rect->w;
- srcBox.right = renderer->viewport.w - rect->x;
- srcBox.top = renderer->viewport.h - rect->y - rect->h;
- srcBox.bottom = renderer->viewport.h - rect->y;
- break;
- case DXGI_MODE_ROTATION_ROTATE90:
- srcBox.left = renderer->viewport.h - rect->y - rect->h;
- srcBox.right = renderer->viewport.h - rect->y;
- srcBox.top = rect->x;
- srcBox.bottom = rect->x + rect->h;
- break;
- default:
- return SDL_SetError("The physical display is in an unknown or unsupported orientation");
- }
- srcBox.front = 0;
- srcBox.back = 1;
- data->d3dContext->CopySubresourceRegion(
- stagingTexture.Get(),
- D3D11CalcSubresource(0, 0, 0),
- 0, 0, 0,
- backBuffer.Get(),
- D3D11CalcSubresource(0, 0, 0),
- &srcBox);
- // Map the staging texture's data to CPU-accessible memory:
- D3D11_MAPPED_SUBRESOURCE textureMemory = {0};
- result = data->d3dContext->Map(
- stagingTexture.Get(),
- D3D11CalcSubresource(0, 0, 0),
- D3D11_MAP_READ,
- 0,
- &textureMemory);
- if (FAILED(result)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__ ", Map Staging Texture to CPU Memory", result);
- return -1;
- }
- // Copy the data into the desired buffer, converting pixels to the
- // desired format at the same time:
- if (SDL_ConvertPixels(
- rect->w, rect->h,
- DXGIFormatToSDLPixelFormat(stagingTextureDesc.Format),
- textureMemory.pData,
- textureMemory.RowPitch,
- format,
- pixels,
- pitch) != 0)
- {
- // When SDL_ConvertPixels fails, it'll have already set the format.
- // Get the error message, and attach some extra data to it.
- std::string errorMessage = string(__FUNCTION__ ", Convert Pixels failed: ") + SDL_GetError();
- SDL_SetError(errorMessage.c_str());
- return -1;
- }
- // Unmap the texture:
- data->d3dContext->Unmap(
- stagingTexture.Get(),
- D3D11CalcSubresource(0, 0, 0));
- // All done. The staging texture will be cleaned up in it's container
- // ComPtr<>'s destructor.
- return 0;
- }
- static void
- D3D11_RenderPresent(SDL_Renderer * renderer)
- {
- D3D11_RenderData *data = (D3D11_RenderData *) renderer->driverdata;
- #if WINAPI_FAMILY == WINAPI_FAMILY_PHONE_APP
- // The first argument instructs DXGI to block until VSync, putting the application
- // to sleep until the next VSync. This ensures we don't waste any cycles rendering
- // frames that will never be displayed to the screen.
- HRESULT hr = data->swapChain->Present(1, 0);
- #else
- // The application may optionally specify "dirty" or "scroll"
- // rects to improve efficiency in certain scenarios.
- // This option is not available on Windows Phone 8, to note.
- DXGI_PRESENT_PARAMETERS parameters = {0};
- parameters.DirtyRectsCount = 0;
- parameters.pDirtyRects = nullptr;
- parameters.pScrollRect = nullptr;
- parameters.pScrollOffset = nullptr;
-
- // The first argument instructs DXGI to block until VSync, putting the application
- // to sleep until the next VSync. This ensures we don't waste any cycles rendering
- // frames that will never be displayed to the screen.
- HRESULT hr = data->swapChain->Present1(1, 0, ¶meters);
- #endif
- // Discard the contents of the render target.
- // This is a valid operation only when the existing contents will be entirely
- // overwritten. If dirty or scroll rects are used, this call should be removed.
- data->d3dContext->DiscardView(data->mainRenderTargetView.Get());
- // If the device was removed either by a disconnect or a driver upgrade, we
- // must recreate all device resources.
- //
- // TODO, WinRT: consider throwing an exception if D3D11_RenderPresent fails, especially if there is a way to salvedge debug info from users' machines
- if (hr == DXGI_ERROR_DEVICE_REMOVED)
- {
- hr = D3D11_HandleDeviceLost(renderer);
- if (FAILED(hr)) {
- WIN_SetErrorFromHRESULT(__FUNCTION__, hr);
- }
- }
- else
- {
- WIN_SetErrorFromHRESULT(__FUNCTION__, hr);
- }
- }
- #endif /* SDL_VIDEO_RENDER_D3D && !SDL_RENDER_DISABLED */
- /* vi: set ts=4 sw=4 expandtab: */
|