SDL_render_d3d11.c 112 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911
  1. /*
  2. Simple DirectMedia Layer
  3. Copyright (C) 1997-2025 Sam Lantinga <slouken@libsdl.org>
  4. This software is provided 'as-is', without any express or implied
  5. warranty. In no event will the authors be held liable for any damages
  6. arising from the use of this software.
  7. Permission is granted to anyone to use this software for any purpose,
  8. including commercial applications, and to alter it and redistribute it
  9. freely, subject to the following restrictions:
  10. 1. The origin of this software must not be misrepresented; you must not
  11. claim that you wrote the original software. If you use this software
  12. in a product, an acknowledgment in the product documentation would be
  13. appreciated but is not required.
  14. 2. Altered source versions must be plainly marked as such, and must not be
  15. misrepresented as being the original software.
  16. 3. This notice may not be removed or altered from any source distribution.
  17. */
  18. #include "SDL_internal.h"
  19. #ifdef SDL_VIDEO_RENDER_D3D11
  20. #define COBJMACROS
  21. #include "../../core/windows/SDL_windows.h"
  22. #include "../../video/windows/SDL_windowswindow.h"
  23. #include "../SDL_sysrender.h"
  24. #include "../SDL_d3dmath.h"
  25. #include "../../video/SDL_pixels_c.h"
  26. #include <d3d11_1.h>
  27. #ifdef HAVE_DXGI1_5_H
  28. #include <dxgi1_5.h>
  29. #else
  30. #include <dxgi1_4.h>
  31. #endif
  32. #include <dxgidebug.h>
  33. #include "SDL_shaders_d3d11.h"
  34. #if defined(_MSC_VER) && !defined(__clang__)
  35. #define SDL_COMPOSE_ERROR(str) __FUNCTION__ ", " str
  36. #else
  37. #define SDL_COMPOSE_ERROR(str) SDL_STRINGIFY_ARG(__FUNCTION__) ", " str
  38. #endif
  39. #define SAFE_RELEASE(X) \
  40. if ((X)) { \
  41. IUnknown_Release(SDL_static_cast(IUnknown *, X)); \
  42. X = NULL; \
  43. }
  44. /* !!! FIXME: vertex buffer bandwidth could be lower; only use UV coords when
  45. !!! FIXME: textures are needed. */
  46. // Vertex shader, common values
  47. typedef struct
  48. {
  49. Float4X4 model;
  50. Float4X4 projectionAndView;
  51. } D3D11_VertexShaderConstants;
  52. // These should mirror the definitions in D3D11_PixelShader_Common.hlsli
  53. static const float TONEMAP_NONE = 0;
  54. //static const float TONEMAP_LINEAR = 1;
  55. static const float TONEMAP_CHROME = 2;
  56. //static const float TEXTURETYPE_NONE = 0;
  57. static const float TEXTURETYPE_RGB = 1;
  58. static const float TEXTURETYPE_RGB_PIXELART = 2;
  59. static const float TEXTURETYPE_PALETTE = 3;
  60. static const float TEXTURETYPE_PALETTE_PIXELART = 4;
  61. static const float TEXTURETYPE_NV12 = 5;
  62. static const float TEXTURETYPE_NV21 = 6;
  63. static const float TEXTURETYPE_YUV = 7;
  64. static const float INPUTTYPE_UNSPECIFIED = 0;
  65. static const float INPUTTYPE_SRGB = 1;
  66. static const float INPUTTYPE_SCRGB = 2;
  67. static const float INPUTTYPE_HDR10 = 3;
  68. typedef struct
  69. {
  70. float scRGB_output;
  71. float texture_type;
  72. float input_type;
  73. float color_scale;
  74. float texel_width;
  75. float texel_height;
  76. float texture_width;
  77. float texture_height;
  78. float tonemap_method;
  79. float tonemap_factor1;
  80. float tonemap_factor2;
  81. float sdr_white_point;
  82. float YCbCr_matrix[16];
  83. } D3D11_PixelShaderConstants;
  84. typedef struct
  85. {
  86. ID3D11Buffer *constants;
  87. D3D11_PixelShaderConstants shader_constants;
  88. } D3D11_PixelShaderState;
  89. // Per-vertex data
  90. typedef struct
  91. {
  92. Float2 pos;
  93. Float2 tex;
  94. SDL_FColor color;
  95. } D3D11_VertexPositionColor;
  96. // Per-texture data
  97. typedef struct
  98. {
  99. int w, h;
  100. ID3D11Texture2D *mainTexture;
  101. ID3D11ShaderResourceView *mainTextureResourceView;
  102. ID3D11RenderTargetView *mainTextureRenderTargetView;
  103. ID3D11Texture2D *paletteTexture;
  104. ID3D11ShaderResourceView *paletteTextureResourceView;
  105. ID3D11Texture2D *stagingTexture;
  106. int lockedTexturePositionX;
  107. int lockedTexturePositionY;
  108. const float *YCbCr_matrix;
  109. #ifdef SDL_HAVE_YUV
  110. // YV12 texture support
  111. bool yuv;
  112. ID3D11Texture2D *mainTextureU;
  113. ID3D11ShaderResourceView *mainTextureResourceViewU;
  114. ID3D11Texture2D *mainTextureV;
  115. ID3D11ShaderResourceView *mainTextureResourceViewV;
  116. // NV12 texture support
  117. bool nv12;
  118. ID3D11ShaderResourceView *mainTextureResourceViewNV;
  119. Uint8 *pixels;
  120. int pitch;
  121. SDL_Rect locked_rect;
  122. #endif
  123. } D3D11_TextureData;
  124. // Blend mode data
  125. typedef struct
  126. {
  127. SDL_BlendMode blendMode;
  128. ID3D11BlendState *blendState;
  129. } D3D11_BlendMode;
  130. // Private renderer data
  131. typedef struct
  132. {
  133. SDL_SharedObject *hDXGIMod;
  134. SDL_SharedObject *hD3D11Mod;
  135. IDXGIFactory2 *dxgiFactory;
  136. IDXGIDebug *dxgiDebug;
  137. ID3D11Device1 *d3dDevice;
  138. ID3D11DeviceContext1 *d3dContext;
  139. IDXGISwapChain1 *swapChain;
  140. DXGI_SWAP_EFFECT swapEffect;
  141. UINT swapChainFlags;
  142. UINT syncInterval;
  143. UINT presentFlags;
  144. ID3D11RenderTargetView *mainRenderTargetView;
  145. ID3D11RenderTargetView *currentOffscreenRenderTargetView;
  146. ID3D11InputLayout *inputLayout;
  147. ID3D11Buffer *vertexBuffers[8];
  148. size_t vertexBufferSizes[8];
  149. ID3D11VertexShader *vertexShader;
  150. ID3D11PixelShader *pixelShaders[NUM_SHADERS];
  151. int blendModesCount;
  152. D3D11_BlendMode *blendModes;
  153. ID3D11SamplerState *samplers[RENDER_SAMPLER_COUNT];
  154. D3D_FEATURE_LEVEL featureLevel;
  155. bool pixelSizeChanged;
  156. // Rasterizers
  157. ID3D11RasterizerState *mainRasterizer;
  158. ID3D11RasterizerState *clippedRasterizer;
  159. // Vertex buffer constants
  160. D3D11_VertexShaderConstants vertexShaderConstantsData;
  161. ID3D11Buffer *vertexShaderConstants;
  162. // Cached renderer properties
  163. DXGI_MODE_ROTATION rotation;
  164. ID3D11RenderTargetView *currentRenderTargetView;
  165. ID3D11RasterizerState *currentRasterizerState;
  166. ID3D11BlendState *currentBlendState;
  167. D3D11_Shader currentShader;
  168. D3D11_PixelShaderState currentShaderState[NUM_SHADERS];
  169. int numCurrentShaderResources;
  170. ID3D11ShaderResourceView *currentShaderResource;
  171. int numCurrentShaderSamplers;
  172. ID3D11SamplerState *currentShaderSampler;
  173. bool cliprectDirty;
  174. bool currentCliprectEnabled;
  175. SDL_Rect currentCliprect;
  176. SDL_Rect currentViewport;
  177. int currentViewportRotation;
  178. bool viewportDirty;
  179. Float4X4 identity;
  180. int currentVertexBuffer;
  181. } D3D11_RenderData;
  182. // Define D3D GUIDs here so we don't have to include uuid.lib.
  183. #ifdef HAVE_GCC_DIAGNOSTIC_PRAGMA
  184. #pragma GCC diagnostic push
  185. #pragma GCC diagnostic ignored "-Wunused-const-variable"
  186. #endif
  187. #ifdef HAVE_DXGI1_5_H
  188. static const GUID SDL_IID_IDXGIFactory5 = { 0x7632e1f5, 0xee65, 0x4dca, { 0x87, 0xfd, 0x84, 0xcd, 0x75, 0xf8, 0x83, 0x8d } };
  189. #endif
  190. static const GUID SDL_IID_IDXGIFactory2 = { 0x50c83a1c, 0xe072, 0x4c48, { 0x87, 0xb0, 0x36, 0x30, 0xfa, 0x36, 0xa6, 0xd0 } };
  191. static const GUID SDL_IID_IDXGIDevice1 = { 0x77db970f, 0x6276, 0x48ba, { 0xba, 0x28, 0x07, 0x01, 0x43, 0xb4, 0x39, 0x2c } };
  192. static const GUID SDL_IID_ID3D11Texture2D = { 0x6f15aaf2, 0xd208, 0x4e89, { 0x9a, 0xb4, 0x48, 0x95, 0x35, 0xd3, 0x4f, 0x9c } };
  193. static const GUID SDL_IID_ID3D11Device1 = { 0xa04bfb29, 0x08ef, 0x43d6, { 0xa4, 0x9c, 0xa9, 0xbd, 0xbd, 0xcb, 0xe6, 0x86 } };
  194. static const GUID SDL_IID_ID3D11DeviceContext1 = { 0xbb2c6faa, 0xb5fb, 0x4082, { 0x8e, 0x6b, 0x38, 0x8b, 0x8c, 0xfa, 0x90, 0xe1 } };
  195. static const GUID SDL_IID_IDXGISwapChain2 = { 0x94d99bdb, 0xf1f8, 0x4ab0, { 0xb2, 0x36, 0x7d, 0xa0, 0x17, 0x0e, 0xda, 0xb1 } };
  196. static const GUID SDL_IID_IDXGIDebug1 = { 0xc5a05f0c, 0x16f2, 0x4adf, { 0x9f, 0x4d, 0xa8, 0xc4, 0xd5, 0x8a, 0xc5, 0x50 } };
  197. static const GUID SDL_IID_IDXGIInfoQueue = { 0xD67441C7, 0x672A, 0x476f, { 0x9E, 0x82, 0xCD, 0x55, 0xB4, 0x49, 0x49, 0xCE } };
  198. static const GUID SDL_DXGI_DEBUG_ALL = { 0xe48ae283, 0xda80, 0x490b, { 0x87, 0xe6, 0x43, 0xe9, 0xa9, 0xcf, 0xda, 0x8 } };
  199. #ifdef HAVE_GCC_DIAGNOSTIC_PRAGMA
  200. #pragma GCC diagnostic pop
  201. #endif
  202. SDL_PixelFormat D3D11_DXGIFormatToSDLPixelFormat(DXGI_FORMAT dxgiFormat)
  203. {
  204. switch (dxgiFormat) {
  205. case DXGI_FORMAT_B8G8R8A8_UNORM:
  206. case DXGI_FORMAT_B8G8R8A8_UNORM_SRGB:
  207. return SDL_PIXELFORMAT_ARGB8888;
  208. case DXGI_FORMAT_R8G8B8A8_UNORM:
  209. case DXGI_FORMAT_R8G8B8A8_UNORM_SRGB:
  210. return SDL_PIXELFORMAT_ABGR8888;
  211. case DXGI_FORMAT_B8G8R8X8_UNORM:
  212. case DXGI_FORMAT_B8G8R8X8_UNORM_SRGB:
  213. return SDL_PIXELFORMAT_XRGB8888;
  214. case DXGI_FORMAT_R10G10B10A2_UNORM:
  215. return SDL_PIXELFORMAT_ABGR2101010;
  216. case DXGI_FORMAT_R16G16B16A16_FLOAT:
  217. return SDL_PIXELFORMAT_RGBA64_FLOAT;
  218. default:
  219. return SDL_PIXELFORMAT_UNKNOWN;
  220. }
  221. }
  222. static DXGI_FORMAT SDLPixelFormatToDXGITextureFormat(Uint32 format, Uint32 output_colorspace)
  223. {
  224. switch (format) {
  225. case SDL_PIXELFORMAT_RGBA64_FLOAT:
  226. return DXGI_FORMAT_R16G16B16A16_FLOAT;
  227. case SDL_PIXELFORMAT_ABGR2101010:
  228. return DXGI_FORMAT_R10G10B10A2_UNORM;
  229. case SDL_PIXELFORMAT_ARGB8888:
  230. if (output_colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  231. return DXGI_FORMAT_B8G8R8A8_UNORM_SRGB;
  232. }
  233. return DXGI_FORMAT_B8G8R8A8_UNORM;
  234. case SDL_PIXELFORMAT_ABGR8888:
  235. if (output_colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  236. return DXGI_FORMAT_R8G8B8A8_UNORM_SRGB;
  237. }
  238. return DXGI_FORMAT_R8G8B8A8_UNORM;
  239. case SDL_PIXELFORMAT_XRGB8888:
  240. if (output_colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  241. return DXGI_FORMAT_B8G8R8X8_UNORM_SRGB;
  242. }
  243. return DXGI_FORMAT_B8G8R8X8_UNORM;
  244. case SDL_PIXELFORMAT_INDEX8:
  245. case SDL_PIXELFORMAT_YV12:
  246. case SDL_PIXELFORMAT_IYUV:
  247. return DXGI_FORMAT_R8_UNORM;
  248. case SDL_PIXELFORMAT_NV12:
  249. case SDL_PIXELFORMAT_NV21:
  250. return DXGI_FORMAT_NV12;
  251. case SDL_PIXELFORMAT_P010:
  252. return DXGI_FORMAT_P010;
  253. default:
  254. return DXGI_FORMAT_UNKNOWN;
  255. }
  256. }
  257. static DXGI_FORMAT SDLPixelFormatToDXGIMainResourceViewFormat(Uint32 format, Uint32 colorspace)
  258. {
  259. switch (format) {
  260. case SDL_PIXELFORMAT_RGBA64_FLOAT:
  261. return DXGI_FORMAT_R16G16B16A16_FLOAT;
  262. case SDL_PIXELFORMAT_ABGR2101010:
  263. return DXGI_FORMAT_R10G10B10A2_UNORM;
  264. case SDL_PIXELFORMAT_ARGB8888:
  265. if (colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  266. return DXGI_FORMAT_B8G8R8A8_UNORM_SRGB;
  267. }
  268. return DXGI_FORMAT_B8G8R8A8_UNORM;
  269. case SDL_PIXELFORMAT_ABGR8888:
  270. if (colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  271. return DXGI_FORMAT_R8G8B8A8_UNORM_SRGB;
  272. }
  273. return DXGI_FORMAT_R8G8B8A8_UNORM;
  274. case SDL_PIXELFORMAT_XRGB8888:
  275. if (colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  276. return DXGI_FORMAT_B8G8R8X8_UNORM_SRGB;
  277. }
  278. return DXGI_FORMAT_B8G8R8X8_UNORM;
  279. case SDL_PIXELFORMAT_YV12:
  280. case SDL_PIXELFORMAT_IYUV:
  281. case SDL_PIXELFORMAT_NV12: // For the Y texture
  282. case SDL_PIXELFORMAT_NV21: // For the Y texture
  283. return DXGI_FORMAT_R8_UNORM;
  284. case SDL_PIXELFORMAT_P010: // For the Y texture
  285. return DXGI_FORMAT_R16_UNORM;
  286. default:
  287. return DXGI_FORMAT_UNKNOWN;
  288. }
  289. }
  290. static void D3D11_DestroyTexture(SDL_Renderer *renderer, SDL_Texture *texture);
  291. static void D3D11_ReleaseAll(SDL_Renderer *renderer)
  292. {
  293. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  294. // Release all textures
  295. for (SDL_Texture *texture = renderer->textures; texture; texture = texture->next) {
  296. D3D11_DestroyTexture(renderer, texture);
  297. }
  298. // Release/reset everything else
  299. if (data) {
  300. int i;
  301. // Make sure the swap chain is fully released
  302. if (data->d3dContext) {
  303. ID3D11DeviceContext_ClearState(data->d3dContext);
  304. ID3D11DeviceContext_Flush(data->d3dContext);
  305. }
  306. SAFE_RELEASE(data->vertexShaderConstants);
  307. SAFE_RELEASE(data->clippedRasterizer);
  308. SAFE_RELEASE(data->mainRasterizer);
  309. for (i = 0; i < SDL_arraysize(data->samplers); ++i) {
  310. SAFE_RELEASE(data->samplers[i]);
  311. }
  312. if (data->blendModesCount > 0) {
  313. for (i = 0; i < data->blendModesCount; ++i) {
  314. SAFE_RELEASE(data->blendModes[i].blendState);
  315. }
  316. SDL_free(data->blendModes);
  317. data->blendModes = NULL;
  318. data->blendModesCount = 0;
  319. }
  320. for (i = 0; i < SDL_arraysize(data->pixelShaders); ++i) {
  321. SAFE_RELEASE(data->pixelShaders[i]);
  322. }
  323. for (i = 0; i < SDL_arraysize(data->currentShaderState); ++i) {
  324. SAFE_RELEASE(data->currentShaderState[i].constants);
  325. }
  326. SAFE_RELEASE(data->vertexShader);
  327. for (i = 0; i < SDL_arraysize(data->vertexBuffers); ++i) {
  328. SAFE_RELEASE(data->vertexBuffers[i]);
  329. }
  330. SAFE_RELEASE(data->inputLayout);
  331. SAFE_RELEASE(data->mainRenderTargetView);
  332. SAFE_RELEASE(data->swapChain);
  333. SAFE_RELEASE(data->d3dContext);
  334. SAFE_RELEASE(data->d3dDevice);
  335. SAFE_RELEASE(data->dxgiFactory);
  336. data->swapEffect = (DXGI_SWAP_EFFECT)0;
  337. data->rotation = DXGI_MODE_ROTATION_UNSPECIFIED;
  338. data->currentOffscreenRenderTargetView = NULL;
  339. data->currentRenderTargetView = NULL;
  340. data->currentRasterizerState = NULL;
  341. data->currentBlendState = NULL;
  342. data->currentShader = SHADER_NONE;
  343. SDL_zero(data->currentShaderState);
  344. data->numCurrentShaderResources = 0;
  345. data->currentShaderResource = NULL;
  346. data->numCurrentShaderSamplers = 0;
  347. data->currentShaderSampler = NULL;
  348. // Check for any leaks if in debug mode
  349. if (data->dxgiDebug) {
  350. DXGI_DEBUG_RLO_FLAGS rloFlags = (DXGI_DEBUG_RLO_FLAGS)(DXGI_DEBUG_RLO_DETAIL | DXGI_DEBUG_RLO_IGNORE_INTERNAL);
  351. IDXGIDebug_ReportLiveObjects(data->dxgiDebug, SDL_DXGI_DEBUG_ALL, rloFlags);
  352. SAFE_RELEASE(data->dxgiDebug);
  353. }
  354. /* Unload the D3D libraries. This should be done last, in order
  355. * to prevent IUnknown::Release() calls from crashing.
  356. */
  357. if (data->hD3D11Mod) {
  358. SDL_UnloadObject(data->hD3D11Mod);
  359. data->hD3D11Mod = NULL;
  360. }
  361. if (data->hDXGIMod) {
  362. SDL_UnloadObject(data->hDXGIMod);
  363. data->hDXGIMod = NULL;
  364. }
  365. }
  366. }
  367. static void D3D11_DestroyRenderer(SDL_Renderer *renderer)
  368. {
  369. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  370. if (data) {
  371. D3D11_ReleaseAll(renderer);
  372. SDL_free(data);
  373. }
  374. }
  375. static D3D11_BLEND GetBlendFunc(SDL_BlendFactor factor)
  376. {
  377. switch (factor) {
  378. case SDL_BLENDFACTOR_ZERO:
  379. return D3D11_BLEND_ZERO;
  380. case SDL_BLENDFACTOR_ONE:
  381. return D3D11_BLEND_ONE;
  382. case SDL_BLENDFACTOR_SRC_COLOR:
  383. return D3D11_BLEND_SRC_COLOR;
  384. case SDL_BLENDFACTOR_ONE_MINUS_SRC_COLOR:
  385. return D3D11_BLEND_INV_SRC_COLOR;
  386. case SDL_BLENDFACTOR_SRC_ALPHA:
  387. return D3D11_BLEND_SRC_ALPHA;
  388. case SDL_BLENDFACTOR_ONE_MINUS_SRC_ALPHA:
  389. return D3D11_BLEND_INV_SRC_ALPHA;
  390. case SDL_BLENDFACTOR_DST_COLOR:
  391. return D3D11_BLEND_DEST_COLOR;
  392. case SDL_BLENDFACTOR_ONE_MINUS_DST_COLOR:
  393. return D3D11_BLEND_INV_DEST_COLOR;
  394. case SDL_BLENDFACTOR_DST_ALPHA:
  395. return D3D11_BLEND_DEST_ALPHA;
  396. case SDL_BLENDFACTOR_ONE_MINUS_DST_ALPHA:
  397. return D3D11_BLEND_INV_DEST_ALPHA;
  398. default:
  399. return (D3D11_BLEND)0;
  400. }
  401. }
  402. static D3D11_BLEND_OP GetBlendEquation(SDL_BlendOperation operation)
  403. {
  404. switch (operation) {
  405. case SDL_BLENDOPERATION_ADD:
  406. return D3D11_BLEND_OP_ADD;
  407. case SDL_BLENDOPERATION_SUBTRACT:
  408. return D3D11_BLEND_OP_SUBTRACT;
  409. case SDL_BLENDOPERATION_REV_SUBTRACT:
  410. return D3D11_BLEND_OP_REV_SUBTRACT;
  411. case SDL_BLENDOPERATION_MINIMUM:
  412. return D3D11_BLEND_OP_MIN;
  413. case SDL_BLENDOPERATION_MAXIMUM:
  414. return D3D11_BLEND_OP_MAX;
  415. default:
  416. return (D3D11_BLEND_OP)0;
  417. }
  418. }
  419. static ID3D11BlendState *D3D11_CreateBlendState(SDL_Renderer *renderer, SDL_BlendMode blendMode)
  420. {
  421. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  422. SDL_BlendFactor srcColorFactor = SDL_GetBlendModeSrcColorFactor(blendMode);
  423. SDL_BlendFactor srcAlphaFactor = SDL_GetBlendModeSrcAlphaFactor(blendMode);
  424. SDL_BlendOperation colorOperation = SDL_GetBlendModeColorOperation(blendMode);
  425. SDL_BlendFactor dstColorFactor = SDL_GetBlendModeDstColorFactor(blendMode);
  426. SDL_BlendFactor dstAlphaFactor = SDL_GetBlendModeDstAlphaFactor(blendMode);
  427. SDL_BlendOperation alphaOperation = SDL_GetBlendModeAlphaOperation(blendMode);
  428. ID3D11BlendState *blendState = NULL;
  429. D3D11_BlendMode *blendModes;
  430. HRESULT result = S_OK;
  431. D3D11_BLEND_DESC blendDesc;
  432. SDL_zero(blendDesc);
  433. blendDesc.AlphaToCoverageEnable = FALSE;
  434. blendDesc.IndependentBlendEnable = FALSE;
  435. blendDesc.RenderTarget[0].BlendEnable = TRUE;
  436. blendDesc.RenderTarget[0].SrcBlend = GetBlendFunc(srcColorFactor);
  437. blendDesc.RenderTarget[0].DestBlend = GetBlendFunc(dstColorFactor);
  438. blendDesc.RenderTarget[0].BlendOp = GetBlendEquation(colorOperation);
  439. blendDesc.RenderTarget[0].SrcBlendAlpha = GetBlendFunc(srcAlphaFactor);
  440. blendDesc.RenderTarget[0].DestBlendAlpha = GetBlendFunc(dstAlphaFactor);
  441. blendDesc.RenderTarget[0].BlendOpAlpha = GetBlendEquation(alphaOperation);
  442. blendDesc.RenderTarget[0].RenderTargetWriteMask = D3D11_COLOR_WRITE_ENABLE_ALL;
  443. result = ID3D11Device_CreateBlendState(data->d3dDevice, &blendDesc, &blendState);
  444. if (FAILED(result)) {
  445. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateBlendState"), result);
  446. return NULL;
  447. }
  448. blendModes = (D3D11_BlendMode *)SDL_realloc(data->blendModes, (data->blendModesCount + 1) * sizeof(*blendModes));
  449. if (!blendModes) {
  450. SAFE_RELEASE(blendState);
  451. return NULL;
  452. }
  453. blendModes[data->blendModesCount].blendMode = blendMode;
  454. blendModes[data->blendModesCount].blendState = blendState;
  455. data->blendModes = blendModes;
  456. ++data->blendModesCount;
  457. return blendState;
  458. }
  459. // Create resources that depend on the device.
  460. static HRESULT D3D11_CreateDeviceResources(SDL_Renderer *renderer)
  461. {
  462. typedef HRESULT (WINAPI *pfnCreateDXGIFactory)(REFIID riid, void **ppFactory);
  463. typedef HRESULT (WINAPI *pfnCreateDXGIFactory2)(UINT flags, REFIID riid, void **ppFactory);
  464. pfnCreateDXGIFactory pCreateDXGIFactory = NULL;
  465. pfnCreateDXGIFactory2 pCreateDXGIFactory2 = NULL;
  466. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  467. PFN_D3D11_CREATE_DEVICE pD3D11CreateDevice;
  468. IDXGIAdapter *dxgiAdapter = NULL;
  469. ID3D11Device *d3dDevice = NULL;
  470. ID3D11DeviceContext *d3dContext = NULL;
  471. IDXGIDevice1 *dxgiDevice = NULL;
  472. HRESULT result = S_OK;
  473. D3D_DRIVER_TYPE driverType = D3D_DRIVER_TYPE_UNKNOWN;
  474. UINT creationFlags = 0;
  475. bool createDebug;
  476. #ifdef HAVE_DXGI1_5_H
  477. IDXGIFactory5 *dxgiFactory5 = NULL;
  478. #endif
  479. /* This array defines the set of DirectX hardware feature levels this app will support.
  480. * Note the ordering should be preserved.
  481. * Don't forget to declare your application's minimum required feature level in its
  482. * description. All applications are assumed to support 9.1 unless otherwise stated.
  483. */
  484. D3D_FEATURE_LEVEL featureLevels[] = {
  485. D3D_FEATURE_LEVEL_11_1,
  486. D3D_FEATURE_LEVEL_11_0,
  487. D3D_FEATURE_LEVEL_10_1,
  488. D3D_FEATURE_LEVEL_10_0,
  489. D3D_FEATURE_LEVEL_9_3,
  490. D3D_FEATURE_LEVEL_9_2,
  491. D3D_FEATURE_LEVEL_9_1
  492. };
  493. D3D11_BUFFER_DESC constantBufferDesc;
  494. D3D11_RASTERIZER_DESC rasterDesc;
  495. // See if we need debug interfaces
  496. createDebug = SDL_GetHintBoolean(SDL_HINT_RENDER_DIRECT3D11_DEBUG, false);
  497. data->hDXGIMod = SDL_LoadObject("dxgi.dll");
  498. if (!data->hDXGIMod) {
  499. result = E_FAIL;
  500. goto done;
  501. }
  502. pCreateDXGIFactory2 = (pfnCreateDXGIFactory2)SDL_LoadFunction(data->hDXGIMod, "CreateDXGIFactory2");
  503. if (!pCreateDXGIFactory2) {
  504. pCreateDXGIFactory = (pfnCreateDXGIFactory)SDL_LoadFunction(data->hDXGIMod, "CreateDXGIFactory");
  505. if (!pCreateDXGIFactory) {
  506. result = E_FAIL;
  507. goto done;
  508. }
  509. }
  510. data->hD3D11Mod = SDL_LoadObject("d3d11.dll");
  511. if (!data->hD3D11Mod) {
  512. result = E_FAIL;
  513. goto done;
  514. }
  515. pD3D11CreateDevice = (PFN_D3D11_CREATE_DEVICE)SDL_LoadFunction(data->hD3D11Mod, "D3D11CreateDevice");
  516. if (!pD3D11CreateDevice) {
  517. result = E_FAIL;
  518. goto done;
  519. }
  520. if (createDebug) {
  521. #ifdef __IDXGIInfoQueue_INTERFACE_DEFINED__
  522. IDXGIInfoQueue *dxgiInfoQueue = NULL;
  523. pfnCreateDXGIFactory2 pDXGIGetDebugInterface1;
  524. // If the debug hint is set, also create the DXGI factory in debug mode
  525. pDXGIGetDebugInterface1 = (pfnCreateDXGIFactory2)SDL_LoadFunction(data->hDXGIMod, "DXGIGetDebugInterface1");
  526. if (!pDXGIGetDebugInterface1) {
  527. result = E_FAIL;
  528. goto done;
  529. }
  530. result = pDXGIGetDebugInterface1(0, &SDL_IID_IDXGIDebug1, (void **)&data->dxgiDebug);
  531. if (FAILED(result)) {
  532. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("DXGIGetDebugInterface1"), result);
  533. goto done;
  534. }
  535. result = pDXGIGetDebugInterface1(0, &SDL_IID_IDXGIInfoQueue, (void **)&dxgiInfoQueue);
  536. if (FAILED(result)) {
  537. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("DXGIGetDebugInterface1"), result);
  538. goto done;
  539. }
  540. IDXGIInfoQueue_SetBreakOnSeverity(dxgiInfoQueue, SDL_DXGI_DEBUG_ALL, DXGI_INFO_QUEUE_MESSAGE_SEVERITY_ERROR, TRUE);
  541. IDXGIInfoQueue_SetBreakOnSeverity(dxgiInfoQueue, SDL_DXGI_DEBUG_ALL, DXGI_INFO_QUEUE_MESSAGE_SEVERITY_CORRUPTION, TRUE);
  542. SAFE_RELEASE(dxgiInfoQueue);
  543. #endif // __IDXGIInfoQueue_INTERFACE_DEFINED__
  544. creationFlags = DXGI_CREATE_FACTORY_DEBUG;
  545. }
  546. if (pCreateDXGIFactory2) {
  547. result = pCreateDXGIFactory2(creationFlags, &SDL_IID_IDXGIFactory2, (void **)&data->dxgiFactory);
  548. } else {
  549. result = pCreateDXGIFactory(&SDL_IID_IDXGIFactory2, (void **)&data->dxgiFactory);
  550. }
  551. if (FAILED(result)) {
  552. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("CreateDXGIFactory"), result);
  553. goto done;
  554. }
  555. #ifdef HAVE_DXGI1_5_H
  556. // Check for tearing support, which requires the IDXGIFactory5 interface.
  557. data->swapChainFlags = 0;
  558. if (!(SDL_GetWindowFlags(renderer->window) & SDL_WINDOW_TRANSPARENT)) {
  559. result = IDXGIFactory2_QueryInterface(data->dxgiFactory, &SDL_IID_IDXGIFactory5, (void **)&dxgiFactory5);
  560. if (SUCCEEDED(result)) {
  561. BOOL allowTearing = FALSE;
  562. result = IDXGIFactory5_CheckFeatureSupport(dxgiFactory5, DXGI_FEATURE_PRESENT_ALLOW_TEARING, &allowTearing, sizeof(allowTearing));
  563. if (SUCCEEDED(result) && allowTearing) {
  564. data->swapChainFlags = DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING;
  565. }
  566. IDXGIFactory5_Release(dxgiFactory5);
  567. }
  568. }
  569. #endif // HAVE_DXGI1_5_H
  570. if (SDL_GetHintBoolean(SDL_HINT_RENDER_DIRECT3D11_WARP, false)) {
  571. driverType = D3D_DRIVER_TYPE_WARP;
  572. dxgiAdapter = NULL;
  573. } else {
  574. driverType = D3D_DRIVER_TYPE_UNKNOWN;
  575. // FIXME: Should we use the default adapter?
  576. result = IDXGIFactory2_EnumAdapters(data->dxgiFactory, 0, &dxgiAdapter);
  577. if (FAILED(result)) {
  578. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("EnumAdapters"), result);
  579. goto done;
  580. }
  581. }
  582. /* This flag adds support for surfaces with a different color channel ordering
  583. * than the API default. It is required for compatibility with Direct2D.
  584. */
  585. creationFlags = D3D11_CREATE_DEVICE_BGRA_SUPPORT;
  586. // Make sure Direct3D's debugging feature gets used, if the app requests it.
  587. if (createDebug) {
  588. creationFlags |= D3D11_CREATE_DEVICE_DEBUG;
  589. }
  590. // Create a single-threaded device unless the app requests otherwise.
  591. if (!SDL_GetHintBoolean(SDL_HINT_RENDER_DIRECT3D_THREADSAFE, false)) {
  592. creationFlags |= D3D11_CREATE_DEVICE_SINGLETHREADED;
  593. }
  594. // Create the Direct3D 11 API device object and a corresponding context.
  595. result = pD3D11CreateDevice(
  596. dxgiAdapter,
  597. driverType,
  598. NULL,
  599. creationFlags, // Set set debug and Direct2D compatibility flags.
  600. featureLevels, // List of feature levels this app can support.
  601. SDL_arraysize(featureLevels),
  602. D3D11_SDK_VERSION, // Always set this to D3D11_SDK_VERSION for Windows Store apps.
  603. &d3dDevice, // Returns the Direct3D device created.
  604. &data->featureLevel, // Returns feature level of device created.
  605. &d3dContext // Returns the device immediate context.
  606. );
  607. if (FAILED(result)) {
  608. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("D3D11CreateDevice"), result);
  609. goto done;
  610. }
  611. result = ID3D11Device_QueryInterface(d3dDevice, &SDL_IID_ID3D11Device1, (void **)&data->d3dDevice);
  612. if (FAILED(result)) {
  613. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device to ID3D11Device1"), result);
  614. goto done;
  615. }
  616. result = ID3D11DeviceContext_QueryInterface(d3dContext, &SDL_IID_ID3D11DeviceContext1, (void **)&data->d3dContext);
  617. if (FAILED(result)) {
  618. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext to ID3D11DeviceContext1"), result);
  619. goto done;
  620. }
  621. result = ID3D11Device_QueryInterface(d3dDevice, &SDL_IID_IDXGIDevice1, (void **)&dxgiDevice);
  622. if (FAILED(result)) {
  623. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device to IDXGIDevice1"), result);
  624. goto done;
  625. }
  626. /* Ensure that DXGI does not queue more than one frame at a time. This both reduces latency and
  627. * ensures that the application will only render after each VSync, minimizing power consumption.
  628. */
  629. result = IDXGIDevice1_SetMaximumFrameLatency(dxgiDevice, 1);
  630. if (FAILED(result)) {
  631. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGIDevice1::SetMaximumFrameLatency"), result);
  632. goto done;
  633. }
  634. /* Make note of the maximum texture size
  635. * Max texture sizes are documented on MSDN, at:
  636. * http://msdn.microsoft.com/en-us/library/windows/apps/ff476876.aspx
  637. */
  638. switch (data->featureLevel) {
  639. case D3D_FEATURE_LEVEL_11_1:
  640. case D3D_FEATURE_LEVEL_11_0:
  641. SDL_SetNumberProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_MAX_TEXTURE_SIZE_NUMBER, 16384);
  642. break;
  643. case D3D_FEATURE_LEVEL_10_1:
  644. case D3D_FEATURE_LEVEL_10_0:
  645. SDL_SetNumberProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_MAX_TEXTURE_SIZE_NUMBER, 8192);
  646. break;
  647. case D3D_FEATURE_LEVEL_9_3:
  648. SDL_SetNumberProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_MAX_TEXTURE_SIZE_NUMBER, 4096);
  649. break;
  650. case D3D_FEATURE_LEVEL_9_2:
  651. case D3D_FEATURE_LEVEL_9_1:
  652. SDL_SetNumberProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_MAX_TEXTURE_SIZE_NUMBER, 2048);
  653. break;
  654. default:
  655. SDL_SetError("%s, Unexpected feature level: %d", __FUNCTION__, data->featureLevel);
  656. result = E_FAIL;
  657. goto done;
  658. }
  659. if (!D3D11_CreateVertexShader(data->d3dDevice, &data->vertexShader, &data->inputLayout)) {
  660. goto done;
  661. }
  662. // Setup space to hold vertex shader constants:
  663. SDL_zero(constantBufferDesc);
  664. constantBufferDesc.ByteWidth = sizeof(D3D11_VertexShaderConstants);
  665. constantBufferDesc.Usage = D3D11_USAGE_DEFAULT;
  666. constantBufferDesc.BindFlags = D3D11_BIND_CONSTANT_BUFFER;
  667. result = ID3D11Device_CreateBuffer(data->d3dDevice,
  668. &constantBufferDesc,
  669. NULL,
  670. &data->vertexShaderConstants);
  671. if (FAILED(result)) {
  672. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateBuffer [vertex shader constants]"), result);
  673. goto done;
  674. }
  675. // Setup Direct3D rasterizer states
  676. SDL_zero(rasterDesc);
  677. rasterDesc.AntialiasedLineEnable = FALSE;
  678. rasterDesc.CullMode = D3D11_CULL_NONE;
  679. rasterDesc.DepthBias = 0;
  680. rasterDesc.DepthBiasClamp = 0.0f;
  681. rasterDesc.DepthClipEnable = TRUE;
  682. rasterDesc.FillMode = D3D11_FILL_SOLID;
  683. rasterDesc.FrontCounterClockwise = FALSE;
  684. rasterDesc.MultisampleEnable = FALSE;
  685. rasterDesc.ScissorEnable = FALSE;
  686. rasterDesc.SlopeScaledDepthBias = 0.0f;
  687. result = ID3D11Device_CreateRasterizerState(data->d3dDevice, &rasterDesc, &data->mainRasterizer);
  688. if (FAILED(result)) {
  689. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateRasterizerState [main rasterizer]"), result);
  690. goto done;
  691. }
  692. rasterDesc.ScissorEnable = TRUE;
  693. result = ID3D11Device_CreateRasterizerState(data->d3dDevice, &rasterDesc, &data->clippedRasterizer);
  694. if (FAILED(result)) {
  695. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateRasterizerState [clipped rasterizer]"), result);
  696. goto done;
  697. }
  698. // Create blending states:
  699. if (!D3D11_CreateBlendState(renderer, SDL_BLENDMODE_BLEND)) {
  700. // D3D11_CreateBlendState will set the SDL error, if it fails
  701. goto done;
  702. }
  703. // Setup render state that doesn't change
  704. ID3D11DeviceContext_IASetInputLayout(data->d3dContext, data->inputLayout);
  705. ID3D11DeviceContext_VSSetShader(data->d3dContext, data->vertexShader, NULL, 0);
  706. ID3D11DeviceContext_VSSetConstantBuffers(data->d3dContext, 0, 1, &data->vertexShaderConstants);
  707. SDL_SetPointerProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_D3D11_DEVICE_POINTER, data->d3dDevice);
  708. done:
  709. SAFE_RELEASE(dxgiAdapter);
  710. SAFE_RELEASE(d3dDevice);
  711. SAFE_RELEASE(d3dContext);
  712. SAFE_RELEASE(dxgiDevice);
  713. return result;
  714. }
  715. static DXGI_MODE_ROTATION D3D11_GetCurrentRotation(void)
  716. {
  717. // FIXME
  718. return DXGI_MODE_ROTATION_IDENTITY;
  719. }
  720. static BOOL D3D11_IsDisplayRotated90Degrees(DXGI_MODE_ROTATION rotation)
  721. {
  722. switch (rotation) {
  723. case DXGI_MODE_ROTATION_ROTATE90:
  724. case DXGI_MODE_ROTATION_ROTATE270:
  725. return TRUE;
  726. default:
  727. return FALSE;
  728. }
  729. }
  730. static int D3D11_GetRotationForCurrentRenderTarget(SDL_Renderer *renderer)
  731. {
  732. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  733. if (data->currentOffscreenRenderTargetView) {
  734. return DXGI_MODE_ROTATION_IDENTITY;
  735. } else {
  736. return data->rotation;
  737. }
  738. }
  739. static bool D3D11_GetViewportAlignedD3DRect(SDL_Renderer *renderer, const SDL_Rect *sdlRect, D3D11_RECT *outRect, BOOL includeViewportOffset)
  740. {
  741. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  742. const int rotation = D3D11_GetRotationForCurrentRenderTarget(renderer);
  743. const SDL_Rect *viewport = &data->currentViewport;
  744. switch (rotation) {
  745. case DXGI_MODE_ROTATION_IDENTITY:
  746. outRect->left = sdlRect->x;
  747. outRect->right = (LONG)sdlRect->x + sdlRect->w;
  748. outRect->top = sdlRect->y;
  749. outRect->bottom = (LONG)sdlRect->y + sdlRect->h;
  750. if (includeViewportOffset) {
  751. outRect->left += viewport->x;
  752. outRect->right += viewport->x;
  753. outRect->top += viewport->y;
  754. outRect->bottom += viewport->y;
  755. }
  756. break;
  757. case DXGI_MODE_ROTATION_ROTATE270:
  758. outRect->left = sdlRect->y;
  759. outRect->right = (LONG)sdlRect->y + sdlRect->h;
  760. outRect->top = viewport->w - sdlRect->x - sdlRect->w;
  761. outRect->bottom = viewport->w - sdlRect->x;
  762. break;
  763. case DXGI_MODE_ROTATION_ROTATE180:
  764. outRect->left = viewport->w - sdlRect->x - sdlRect->w;
  765. outRect->right = viewport->w - sdlRect->x;
  766. outRect->top = viewport->h - sdlRect->y - sdlRect->h;
  767. outRect->bottom = viewport->h - sdlRect->y;
  768. break;
  769. case DXGI_MODE_ROTATION_ROTATE90:
  770. outRect->left = viewport->h - sdlRect->y - sdlRect->h;
  771. outRect->right = viewport->h - sdlRect->y;
  772. outRect->top = sdlRect->x;
  773. outRect->bottom = (LONG)sdlRect->x + sdlRect->h;
  774. break;
  775. default:
  776. return SDL_SetError("The physical display is in an unknown or unsupported rotation");
  777. }
  778. return true;
  779. }
  780. static HRESULT D3D11_CreateSwapChain(SDL_Renderer *renderer, int w, int h)
  781. {
  782. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  783. IUnknown *coreWindow = NULL;
  784. IDXGISwapChain3 *swapChain3 = NULL;
  785. HRESULT result = S_OK;
  786. // Create a swap chain using the same adapter as the existing Direct3D device.
  787. DXGI_SWAP_CHAIN_DESC1 swapChainDesc;
  788. SDL_zero(swapChainDesc);
  789. swapChainDesc.Width = w;
  790. swapChainDesc.Height = h;
  791. switch (renderer->output_colorspace) {
  792. case SDL_COLORSPACE_SRGB_LINEAR:
  793. swapChainDesc.Format = DXGI_FORMAT_R16G16B16A16_FLOAT;
  794. break;
  795. case SDL_COLORSPACE_HDR10:
  796. swapChainDesc.Format = DXGI_FORMAT_R10G10B10A2_UNORM;
  797. break;
  798. default:
  799. swapChainDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; // This is the most common swap chain format.
  800. break;
  801. }
  802. swapChainDesc.Stereo = FALSE;
  803. swapChainDesc.SampleDesc.Count = 1; // Don't use multi-sampling.
  804. swapChainDesc.SampleDesc.Quality = 0;
  805. swapChainDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT;
  806. swapChainDesc.BufferCount = 2; // Use double-buffering to minimize latency.
  807. if (WIN_IsWindows8OrGreater()) {
  808. swapChainDesc.Scaling = DXGI_SCALING_NONE;
  809. } else {
  810. swapChainDesc.Scaling = DXGI_SCALING_STRETCH;
  811. }
  812. if (SDL_GetWindowFlags(renderer->window) & SDL_WINDOW_TRANSPARENT) {
  813. swapChainDesc.Scaling = DXGI_SCALING_STRETCH;
  814. swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_DISCARD;
  815. } else {
  816. swapChainDesc.SwapEffect = DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL; // All Windows Store apps must use this SwapEffect.
  817. }
  818. swapChainDesc.Flags = data->swapChainFlags;
  819. if (coreWindow) {
  820. result = IDXGIFactory2_CreateSwapChainForCoreWindow(data->dxgiFactory,
  821. (IUnknown *)data->d3dDevice,
  822. coreWindow,
  823. &swapChainDesc,
  824. NULL, // Allow on all displays.
  825. &data->swapChain);
  826. if (FAILED(result)) {
  827. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGIFactory2::CreateSwapChainForCoreWindow"), result);
  828. goto done;
  829. }
  830. } else {
  831. #if defined(SDL_PLATFORM_WIN32) || defined(SDL_PLATFORM_WINGDK)
  832. HWND hwnd = (HWND)SDL_GetPointerProperty(SDL_GetWindowProperties(renderer->window), SDL_PROP_WINDOW_WIN32_HWND_POINTER, NULL);
  833. if (!hwnd) {
  834. SDL_SetError("Couldn't get window handle");
  835. result = E_FAIL;
  836. goto done;
  837. }
  838. result = IDXGIFactory2_CreateSwapChainForHwnd(data->dxgiFactory,
  839. (IUnknown *)data->d3dDevice,
  840. hwnd,
  841. &swapChainDesc,
  842. NULL,
  843. NULL, // Allow on all displays.
  844. &data->swapChain);
  845. if (FAILED(result)) {
  846. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGIFactory2::CreateSwapChainForHwnd"), result);
  847. goto done;
  848. }
  849. IDXGIFactory_MakeWindowAssociation(data->dxgiFactory, hwnd, DXGI_MWA_NO_WINDOW_CHANGES);
  850. #else
  851. SDL_SetError(__FUNCTION__ ", Unable to find something to attach a swap chain to");
  852. goto done;
  853. #endif // defined(SDL_PLATFORM_WIN32) || defined(SDL_PLATFORM_WINGDK) / else
  854. }
  855. data->swapEffect = swapChainDesc.SwapEffect;
  856. if (SUCCEEDED(IDXGISwapChain1_QueryInterface(data->swapChain, &SDL_IID_IDXGISwapChain2, (void **)&swapChain3))) {
  857. UINT colorspace_support = 0;
  858. DXGI_COLOR_SPACE_TYPE colorspace;
  859. switch (renderer->output_colorspace) {
  860. case SDL_COLORSPACE_SRGB_LINEAR:
  861. colorspace = DXGI_COLOR_SPACE_RGB_FULL_G10_NONE_P709;
  862. break;
  863. case SDL_COLORSPACE_HDR10:
  864. colorspace = DXGI_COLOR_SPACE_RGB_FULL_G2084_NONE_P2020;
  865. break;
  866. default:
  867. // sRGB
  868. colorspace = DXGI_COLOR_SPACE_RGB_FULL_G22_NONE_P709;
  869. break;
  870. }
  871. if (SUCCEEDED(IDXGISwapChain3_CheckColorSpaceSupport(swapChain3, colorspace, &colorspace_support)) &&
  872. (colorspace_support & DXGI_SWAP_CHAIN_COLOR_SPACE_SUPPORT_FLAG_PRESENT)) {
  873. result = IDXGISwapChain3_SetColorSpace1(swapChain3, colorspace);
  874. if (FAILED(result)) {
  875. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain3::SetColorSpace1"), result);
  876. goto done;
  877. }
  878. } else if (colorspace != DXGI_COLOR_SPACE_RGB_FULL_G10_NONE_P709) {
  879. // Not the default, we're not going to be able to present in this colorspace
  880. SDL_SetError("Unsupported output colorspace");
  881. result = DXGI_ERROR_UNSUPPORTED;
  882. }
  883. }
  884. SDL_SetPointerProperty(SDL_GetRendererProperties(renderer), SDL_PROP_RENDERER_D3D11_SWAPCHAIN_POINTER, data->swapChain);
  885. done:
  886. SAFE_RELEASE(swapChain3);
  887. SAFE_RELEASE(coreWindow);
  888. return result;
  889. }
  890. static void D3D11_ReleaseMainRenderTargetView(SDL_Renderer *renderer)
  891. {
  892. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  893. ID3D11DeviceContext_OMSetRenderTargets(data->d3dContext, 0, NULL, NULL);
  894. SAFE_RELEASE(data->mainRenderTargetView);
  895. }
  896. static void D3D11_UpdatePresentFlags(SDL_Renderer *renderer)
  897. {
  898. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  899. if (data->syncInterval > 0) {
  900. data->presentFlags = 0;
  901. } else {
  902. data->presentFlags = DXGI_PRESENT_DO_NOT_WAIT;
  903. #ifdef HAVE_DXGI1_5_H
  904. // Present tearing requires sync interval 0, a swap chain flag, and not in exclusive fullscreen mode.
  905. if ((data->swapChainFlags & DXGI_SWAP_CHAIN_FLAG_ALLOW_TEARING)) {
  906. HRESULT result = S_OK;
  907. BOOL fullscreenState = FALSE;
  908. result = IDXGISwapChain_GetFullscreenState(data->swapChain, &fullscreenState, NULL);
  909. if (SUCCEEDED(result) && !fullscreenState) {
  910. data->presentFlags = DXGI_PRESENT_ALLOW_TEARING;
  911. }
  912. }
  913. #endif // HAVE_DXGI1_5_H
  914. }
  915. }
  916. // Initialize all resources that change when the window's size changes.
  917. static HRESULT D3D11_CreateWindowSizeDependentResources(SDL_Renderer *renderer)
  918. {
  919. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  920. ID3D11Texture2D *backBuffer = NULL;
  921. HRESULT result = S_OK;
  922. int w, h;
  923. // Release the previous render target view
  924. D3D11_ReleaseMainRenderTargetView(renderer);
  925. /* The width and height of the swap chain must be based on the display's
  926. * non-rotated size.
  927. */
  928. SDL_GetWindowSizeInPixels(renderer->window, &w, &h);
  929. data->rotation = D3D11_GetCurrentRotation();
  930. // SDL_Log("%s: windowSize={%d,%d}, orientation=%d", __FUNCTION__, w, h, (int)data->rotation);
  931. if (D3D11_IsDisplayRotated90Degrees(data->rotation)) {
  932. int tmp = w;
  933. w = h;
  934. h = tmp;
  935. }
  936. if (data->swapChain) {
  937. // If the swap chain already exists, resize it.
  938. result = IDXGISwapChain_ResizeBuffers(data->swapChain,
  939. 0,
  940. w, h,
  941. DXGI_FORMAT_UNKNOWN,
  942. data->swapChainFlags);
  943. if (FAILED(result)) {
  944. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain::ResizeBuffers"), result);
  945. goto done;
  946. }
  947. } else {
  948. result = D3D11_CreateSwapChain(renderer, w, h);
  949. if (FAILED(result) || !data->swapChain) {
  950. goto done;
  951. }
  952. }
  953. D3D11_UpdatePresentFlags(renderer);
  954. // Set the proper rotation for the swap chain.
  955. if (WIN_IsWindows8OrGreater()) {
  956. if (data->swapEffect == DXGI_SWAP_EFFECT_FLIP_SEQUENTIAL) {
  957. result = IDXGISwapChain1_SetRotation(data->swapChain, data->rotation);
  958. if (FAILED(result)) {
  959. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain1::SetRotation"), result);
  960. goto done;
  961. }
  962. }
  963. }
  964. result = IDXGISwapChain_GetBuffer(data->swapChain,
  965. 0,
  966. &SDL_IID_ID3D11Texture2D,
  967. (void **)&backBuffer);
  968. if (FAILED(result)) {
  969. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain::GetBuffer [back-buffer]"), result);
  970. goto done;
  971. }
  972. // Create a render target view of the swap chain back buffer.
  973. result = ID3D11Device_CreateRenderTargetView(data->d3dDevice,
  974. (ID3D11Resource *)backBuffer,
  975. NULL,
  976. &data->mainRenderTargetView);
  977. if (FAILED(result)) {
  978. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device::CreateRenderTargetView"), result);
  979. goto done;
  980. }
  981. /* Set the swap chain target immediately, so that a target is always set
  982. * even before we get to SetDrawState. Without this it's possible to hit
  983. * null references in places like ReadPixels!
  984. */
  985. ID3D11DeviceContext_OMSetRenderTargets(data->d3dContext,
  986. 1,
  987. &data->mainRenderTargetView,
  988. NULL);
  989. data->viewportDirty = true;
  990. done:
  991. SAFE_RELEASE(backBuffer);
  992. return result;
  993. }
  994. static bool D3D11_HandleDeviceLost(SDL_Renderer *renderer)
  995. {
  996. bool recovered = false;
  997. D3D11_ReleaseAll(renderer);
  998. if (SUCCEEDED(D3D11_CreateDeviceResources(renderer)) &&
  999. SUCCEEDED(D3D11_CreateWindowSizeDependentResources(renderer))) {
  1000. recovered = true;
  1001. } else {
  1002. SDL_LogError(SDL_LOG_CATEGORY_RENDER, "Renderer couldn't recover from device lost: %s", SDL_GetError());
  1003. D3D11_ReleaseAll(renderer);
  1004. }
  1005. // Let the application know that the device has been reset or lost
  1006. SDL_Event event;
  1007. SDL_zero(event);
  1008. event.type = recovered ? SDL_EVENT_RENDER_DEVICE_RESET : SDL_EVENT_RENDER_DEVICE_LOST;
  1009. event.render.windowID = SDL_GetWindowID(SDL_GetRenderWindow(renderer));
  1010. SDL_PushEvent(&event);
  1011. return recovered;
  1012. }
  1013. // This method is called when the window's size changes.
  1014. static HRESULT D3D11_UpdateForWindowSizeChange(SDL_Renderer *renderer)
  1015. {
  1016. return D3D11_CreateWindowSizeDependentResources(renderer);
  1017. }
  1018. static void D3D11_WindowEvent(SDL_Renderer *renderer, const SDL_WindowEvent *event)
  1019. {
  1020. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  1021. if (event->type == SDL_EVENT_WINDOW_PIXEL_SIZE_CHANGED) {
  1022. data->pixelSizeChanged = true;
  1023. }
  1024. }
  1025. static bool D3D11_SupportsBlendMode(SDL_Renderer *renderer, SDL_BlendMode blendMode)
  1026. {
  1027. SDL_BlendFactor srcColorFactor = SDL_GetBlendModeSrcColorFactor(blendMode);
  1028. SDL_BlendFactor srcAlphaFactor = SDL_GetBlendModeSrcAlphaFactor(blendMode);
  1029. SDL_BlendOperation colorOperation = SDL_GetBlendModeColorOperation(blendMode);
  1030. SDL_BlendFactor dstColorFactor = SDL_GetBlendModeDstColorFactor(blendMode);
  1031. SDL_BlendFactor dstAlphaFactor = SDL_GetBlendModeDstAlphaFactor(blendMode);
  1032. SDL_BlendOperation alphaOperation = SDL_GetBlendModeAlphaOperation(blendMode);
  1033. if (!GetBlendFunc(srcColorFactor) || !GetBlendFunc(srcAlphaFactor) ||
  1034. !GetBlendEquation(colorOperation) ||
  1035. !GetBlendFunc(dstColorFactor) || !GetBlendFunc(dstAlphaFactor) ||
  1036. !GetBlendEquation(alphaOperation)) {
  1037. return false;
  1038. }
  1039. return true;
  1040. }
  1041. static bool GetTextureProperty(SDL_PropertiesID props, const char *name, ID3D11Texture2D **texture)
  1042. {
  1043. IUnknown *unknown = SDL_GetPointerProperty(props, name, NULL);
  1044. if (unknown) {
  1045. HRESULT result = IUnknown_QueryInterface(unknown, &SDL_IID_ID3D11Texture2D, (void **)texture);
  1046. if (FAILED(result)) {
  1047. return WIN_SetErrorFromHRESULT(name, result);
  1048. }
  1049. }
  1050. return true;
  1051. }
  1052. static bool D3D11_CreateTexture(SDL_Renderer *renderer, SDL_Texture *texture, SDL_PropertiesID create_props)
  1053. {
  1054. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1055. D3D11_TextureData *textureData;
  1056. HRESULT result;
  1057. DXGI_FORMAT textureFormat = SDLPixelFormatToDXGITextureFormat(texture->format, renderer->output_colorspace);
  1058. D3D11_TEXTURE2D_DESC textureDesc;
  1059. D3D11_SHADER_RESOURCE_VIEW_DESC resourceViewDesc;
  1060. if (!rendererData->d3dDevice) {
  1061. return SDL_SetError("Device lost and couldn't be recovered");
  1062. }
  1063. if (textureFormat == DXGI_FORMAT_UNKNOWN) {
  1064. return SDL_SetError("%s, An unsupported SDL pixel format (0x%x) was specified",
  1065. __FUNCTION__, texture->format);
  1066. }
  1067. textureData = (D3D11_TextureData *)SDL_calloc(1, sizeof(*textureData));
  1068. if (!textureData) {
  1069. return false;
  1070. }
  1071. texture->internal = textureData;
  1072. SDL_zero(textureDesc);
  1073. textureDesc.Width = texture->w;
  1074. textureDesc.Height = texture->h;
  1075. textureDesc.MipLevels = 1;
  1076. textureDesc.ArraySize = 1;
  1077. textureDesc.Format = textureFormat;
  1078. textureDesc.SampleDesc.Count = 1;
  1079. textureDesc.SampleDesc.Quality = 0;
  1080. textureDesc.MiscFlags = 0;
  1081. // NV12 textures must have even width and height
  1082. if (texture->format == SDL_PIXELFORMAT_NV12 ||
  1083. texture->format == SDL_PIXELFORMAT_NV21 ||
  1084. texture->format == SDL_PIXELFORMAT_P010) {
  1085. textureDesc.Width = (textureDesc.Width + 1) & ~1;
  1086. textureDesc.Height = (textureDesc.Height + 1) & ~1;
  1087. }
  1088. textureData->w = (int)textureDesc.Width;
  1089. textureData->h = (int)textureDesc.Height;
  1090. if (texture->access == SDL_TEXTUREACCESS_STREAMING) {
  1091. textureDesc.Usage = D3D11_USAGE_DYNAMIC;
  1092. textureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
  1093. } else {
  1094. textureDesc.Usage = D3D11_USAGE_DEFAULT;
  1095. textureDesc.CPUAccessFlags = 0;
  1096. }
  1097. if (texture->access == SDL_TEXTUREACCESS_TARGET) {
  1098. textureDesc.BindFlags = D3D11_BIND_SHADER_RESOURCE | D3D11_BIND_RENDER_TARGET;
  1099. } else {
  1100. textureDesc.BindFlags = D3D11_BIND_SHADER_RESOURCE;
  1101. }
  1102. if (!GetTextureProperty(create_props, SDL_PROP_TEXTURE_CREATE_D3D11_TEXTURE_POINTER, &textureData->mainTexture)) {
  1103. return false;
  1104. }
  1105. if (!textureData->mainTexture) {
  1106. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1107. &textureDesc,
  1108. NULL,
  1109. &textureData->mainTexture);
  1110. if (FAILED(result)) {
  1111. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D"), result);
  1112. }
  1113. }
  1114. SDL_SetPointerProperty(SDL_GetTextureProperties(texture), SDL_PROP_TEXTURE_D3D11_TEXTURE_POINTER, textureData->mainTexture);
  1115. if (texture->format == SDL_PIXELFORMAT_INDEX8) {
  1116. textureDesc.Width = 256;
  1117. textureDesc.Height = 1;
  1118. if (renderer->output_colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  1119. textureDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM_SRGB;
  1120. } else {
  1121. textureDesc.Format = DXGI_FORMAT_R8G8B8A8_UNORM;
  1122. }
  1123. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1124. &textureDesc,
  1125. NULL,
  1126. &textureData->paletteTexture);
  1127. if (FAILED(result)) {
  1128. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D"), result);
  1129. }
  1130. }
  1131. #ifdef SDL_HAVE_YUV
  1132. if (texture->format == SDL_PIXELFORMAT_YV12 ||
  1133. texture->format == SDL_PIXELFORMAT_IYUV) {
  1134. textureData->yuv = true;
  1135. textureDesc.Width = (textureDesc.Width + 1) / 2;
  1136. textureDesc.Height = (textureDesc.Height + 1) / 2;
  1137. if (!GetTextureProperty(create_props, SDL_PROP_TEXTURE_CREATE_D3D11_TEXTURE_U_POINTER, &textureData->mainTextureU)) {
  1138. return false;
  1139. }
  1140. if (!textureData->mainTextureU) {
  1141. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1142. &textureDesc,
  1143. NULL,
  1144. &textureData->mainTextureU);
  1145. if (FAILED(result)) {
  1146. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D"), result);
  1147. }
  1148. }
  1149. SDL_SetPointerProperty(SDL_GetTextureProperties(texture), SDL_PROP_TEXTURE_D3D11_TEXTURE_U_POINTER, textureData->mainTextureU);
  1150. if (!GetTextureProperty(create_props, SDL_PROP_TEXTURE_CREATE_D3D11_TEXTURE_V_POINTER, &textureData->mainTextureV)) {
  1151. return false;
  1152. }
  1153. if (!textureData->mainTextureV) {
  1154. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1155. &textureDesc,
  1156. NULL,
  1157. &textureData->mainTextureV);
  1158. if (FAILED(result)) {
  1159. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D"), result);
  1160. }
  1161. }
  1162. SDL_SetPointerProperty(SDL_GetTextureProperties(texture), SDL_PROP_TEXTURE_D3D11_TEXTURE_V_POINTER, textureData->mainTextureV);
  1163. textureData->YCbCr_matrix = SDL_GetYCbCRtoRGBConversionMatrix(texture->colorspace, texture->w, texture->h, 8);
  1164. if (!textureData->YCbCr_matrix) {
  1165. return SDL_SetError("Unsupported YUV colorspace");
  1166. }
  1167. }
  1168. if (texture->format == SDL_PIXELFORMAT_NV12 ||
  1169. texture->format == SDL_PIXELFORMAT_NV21 ||
  1170. texture->format == SDL_PIXELFORMAT_P010) {
  1171. int bits_per_pixel;
  1172. textureData->nv12 = true;
  1173. switch (texture->format) {
  1174. case SDL_PIXELFORMAT_P010:
  1175. bits_per_pixel = 10;
  1176. break;
  1177. default:
  1178. bits_per_pixel = 8;
  1179. break;
  1180. }
  1181. textureData->YCbCr_matrix = SDL_GetYCbCRtoRGBConversionMatrix(texture->colorspace, texture->w, texture->h, bits_per_pixel);
  1182. if (!textureData->YCbCr_matrix) {
  1183. return SDL_SetError("Unsupported YUV colorspace");
  1184. }
  1185. }
  1186. #endif // SDL_HAVE_YUV
  1187. SDL_zero(resourceViewDesc);
  1188. resourceViewDesc.Format = SDLPixelFormatToDXGIMainResourceViewFormat(texture->format, renderer->output_colorspace);
  1189. resourceViewDesc.ViewDimension = D3D11_SRV_DIMENSION_TEXTURE2D;
  1190. resourceViewDesc.Texture2D.MostDetailedMip = 0;
  1191. resourceViewDesc.Texture2D.MipLevels = textureDesc.MipLevels;
  1192. result = ID3D11Device_CreateShaderResourceView(rendererData->d3dDevice,
  1193. (ID3D11Resource *)textureData->mainTexture,
  1194. &resourceViewDesc,
  1195. &textureData->mainTextureResourceView);
  1196. if (FAILED(result)) {
  1197. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateShaderResourceView"), result);
  1198. }
  1199. if (texture->format == SDL_PIXELFORMAT_INDEX8) {
  1200. resourceViewDesc.Format = textureDesc.Format;
  1201. result = ID3D11Device_CreateShaderResourceView(rendererData->d3dDevice,
  1202. (ID3D11Resource *)textureData->paletteTexture,
  1203. &resourceViewDesc,
  1204. &textureData->paletteTextureResourceView);
  1205. if (FAILED(result)) {
  1206. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateShaderResourceView"), result);
  1207. }
  1208. }
  1209. #ifdef SDL_HAVE_YUV
  1210. if (textureData->yuv) {
  1211. result = ID3D11Device_CreateShaderResourceView(rendererData->d3dDevice,
  1212. (ID3D11Resource *)textureData->mainTextureU,
  1213. &resourceViewDesc,
  1214. &textureData->mainTextureResourceViewU);
  1215. if (FAILED(result)) {
  1216. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateShaderResourceView"), result);
  1217. }
  1218. result = ID3D11Device_CreateShaderResourceView(rendererData->d3dDevice,
  1219. (ID3D11Resource *)textureData->mainTextureV,
  1220. &resourceViewDesc,
  1221. &textureData->mainTextureResourceViewV);
  1222. if (FAILED(result)) {
  1223. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateShaderResourceView"), result);
  1224. }
  1225. }
  1226. if (textureData->nv12) {
  1227. D3D11_SHADER_RESOURCE_VIEW_DESC nvResourceViewDesc = resourceViewDesc;
  1228. if (texture->format == SDL_PIXELFORMAT_NV12 || texture->format == SDL_PIXELFORMAT_NV21) {
  1229. nvResourceViewDesc.Format = DXGI_FORMAT_R8G8_UNORM;
  1230. } else if (texture->format == SDL_PIXELFORMAT_P010) {
  1231. nvResourceViewDesc.Format = DXGI_FORMAT_R16G16_UNORM;
  1232. }
  1233. result = ID3D11Device_CreateShaderResourceView(rendererData->d3dDevice,
  1234. (ID3D11Resource *)textureData->mainTexture,
  1235. &nvResourceViewDesc,
  1236. &textureData->mainTextureResourceViewNV);
  1237. if (FAILED(result)) {
  1238. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateShaderResourceView"), result);
  1239. }
  1240. }
  1241. #endif // SDL_HAVE_YUV
  1242. if (texture->access & SDL_TEXTUREACCESS_TARGET) {
  1243. D3D11_RENDER_TARGET_VIEW_DESC renderTargetViewDesc;
  1244. SDL_zero(renderTargetViewDesc);
  1245. renderTargetViewDesc.Format = textureDesc.Format;
  1246. renderTargetViewDesc.ViewDimension = D3D11_RTV_DIMENSION_TEXTURE2D;
  1247. renderTargetViewDesc.Texture2D.MipSlice = 0;
  1248. result = ID3D11Device_CreateRenderTargetView(rendererData->d3dDevice,
  1249. (ID3D11Resource *)textureData->mainTexture,
  1250. &renderTargetViewDesc,
  1251. &textureData->mainTextureRenderTargetView);
  1252. if (FAILED(result)) {
  1253. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateRenderTargetView"), result);
  1254. }
  1255. }
  1256. return true;
  1257. }
  1258. static void D3D11_DestroyTexture(SDL_Renderer *renderer,
  1259. SDL_Texture *texture)
  1260. {
  1261. D3D11_TextureData *data = (D3D11_TextureData *)texture->internal;
  1262. if (!data) {
  1263. return;
  1264. }
  1265. SAFE_RELEASE(data->mainTexture);
  1266. SAFE_RELEASE(data->mainTextureResourceView);
  1267. SAFE_RELEASE(data->mainTextureRenderTargetView);
  1268. SAFE_RELEASE(data->paletteTexture);
  1269. SAFE_RELEASE(data->paletteTextureResourceView);
  1270. SAFE_RELEASE(data->stagingTexture);
  1271. #ifdef SDL_HAVE_YUV
  1272. SAFE_RELEASE(data->mainTextureU);
  1273. SAFE_RELEASE(data->mainTextureResourceViewU);
  1274. SAFE_RELEASE(data->mainTextureV);
  1275. SAFE_RELEASE(data->mainTextureResourceViewV);
  1276. SAFE_RELEASE(data->mainTextureResourceViewNV);
  1277. SDL_free(data->pixels);
  1278. #endif
  1279. SDL_free(data);
  1280. texture->internal = NULL;
  1281. }
  1282. static bool D3D11_UpdateTextureInternal(D3D11_RenderData *rendererData, ID3D11Texture2D *texture, int bpp, int x, int y, int w, int h, const void *pixels, int pitch)
  1283. {
  1284. ID3D11Texture2D *stagingTexture;
  1285. const Uint8 *src;
  1286. Uint8 *dst;
  1287. int row;
  1288. UINT length;
  1289. HRESULT result;
  1290. D3D11_TEXTURE2D_DESC stagingTextureDesc;
  1291. D3D11_MAPPED_SUBRESOURCE textureMemory;
  1292. // Create a 'staging' texture, which will be used to write to a portion of the main texture.
  1293. ID3D11Texture2D_GetDesc(texture, &stagingTextureDesc);
  1294. stagingTextureDesc.Width = w;
  1295. stagingTextureDesc.Height = h;
  1296. stagingTextureDesc.BindFlags = 0;
  1297. stagingTextureDesc.MiscFlags = 0;
  1298. stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
  1299. stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
  1300. if (stagingTextureDesc.Format == DXGI_FORMAT_NV12 ||
  1301. stagingTextureDesc.Format == DXGI_FORMAT_P010) {
  1302. stagingTextureDesc.Width = (stagingTextureDesc.Width + 1) & ~1;
  1303. stagingTextureDesc.Height = (stagingTextureDesc.Height + 1) & ~1;
  1304. }
  1305. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1306. &stagingTextureDesc,
  1307. NULL,
  1308. &stagingTexture);
  1309. if (FAILED(result)) {
  1310. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D [create staging texture]"), result);
  1311. }
  1312. // Get a write-only pointer to data in the staging texture:
  1313. result = ID3D11DeviceContext_Map(rendererData->d3dContext,
  1314. (ID3D11Resource *)stagingTexture,
  1315. 0,
  1316. D3D11_MAP_WRITE,
  1317. 0,
  1318. &textureMemory);
  1319. if (FAILED(result)) {
  1320. SAFE_RELEASE(stagingTexture);
  1321. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext1::Map [map staging texture]"), result);
  1322. }
  1323. src = (const Uint8 *)pixels;
  1324. dst = (Uint8 *)textureMemory.pData;
  1325. length = w * bpp;
  1326. if (length == (UINT)pitch && length == textureMemory.RowPitch) {
  1327. SDL_memcpy(dst, src, (size_t)length * h);
  1328. } else {
  1329. if (length > (UINT)pitch) {
  1330. length = pitch;
  1331. }
  1332. if (length > textureMemory.RowPitch) {
  1333. length = textureMemory.RowPitch;
  1334. }
  1335. for (row = 0; row < h; ++row) {
  1336. SDL_memcpy(dst, src, length);
  1337. src += pitch;
  1338. dst += textureMemory.RowPitch;
  1339. }
  1340. }
  1341. if (stagingTextureDesc.Format == DXGI_FORMAT_NV12 ||
  1342. stagingTextureDesc.Format == DXGI_FORMAT_P010) {
  1343. // Copy the UV plane as well
  1344. h = (h + 1) / 2;
  1345. if (stagingTextureDesc.Format == DXGI_FORMAT_P010) {
  1346. length = (length + 3) & ~3;
  1347. pitch = (pitch + 3) & ~3;
  1348. } else {
  1349. length = (length + 1) & ~1;
  1350. pitch = (pitch + 1) & ~1;
  1351. }
  1352. dst = (Uint8 *)textureMemory.pData + stagingTextureDesc.Height * textureMemory.RowPitch;
  1353. for (row = 0; row < h; ++row) {
  1354. SDL_memcpy(dst, src, length);
  1355. src += pitch;
  1356. dst += textureMemory.RowPitch;
  1357. }
  1358. }
  1359. // Commit the pixel buffer's changes back to the staging texture:
  1360. ID3D11DeviceContext_Unmap(rendererData->d3dContext,
  1361. (ID3D11Resource *)stagingTexture,
  1362. 0);
  1363. // Copy the staging texture's contents back to the texture:
  1364. ID3D11DeviceContext_CopySubresourceRegion(rendererData->d3dContext,
  1365. (ID3D11Resource *)texture,
  1366. 0,
  1367. x,
  1368. y,
  1369. 0,
  1370. (ID3D11Resource *)stagingTexture,
  1371. 0,
  1372. NULL);
  1373. SAFE_RELEASE(stagingTexture);
  1374. return true;
  1375. }
  1376. static bool D3D11_UpdateTexturePalette(SDL_Renderer *renderer, SDL_Texture *texture)
  1377. {
  1378. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1379. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1380. SDL_Palette *palette = texture->palette;
  1381. if (!textureData) {
  1382. return SDL_SetError("Texture is not currently available");
  1383. }
  1384. return D3D11_UpdateTextureInternal(rendererData, textureData->paletteTexture, 4, 0, 0, palette->ncolors, 1, palette->colors, palette->ncolors * sizeof(*palette->colors));
  1385. }
  1386. #ifdef SDL_HAVE_YUV
  1387. static bool D3D11_UpdateTextureNV(SDL_Renderer *renderer, SDL_Texture *texture,
  1388. const SDL_Rect *rect,
  1389. const Uint8 *Yplane, int Ypitch,
  1390. const Uint8 *UVplane, int UVpitch);
  1391. static bool D3D11_UpdateTextureYUV(SDL_Renderer *renderer, SDL_Texture *texture,
  1392. const SDL_Rect *rect,
  1393. const Uint8 *Yplane, int Ypitch,
  1394. const Uint8 *Uplane, int Upitch,
  1395. const Uint8 *Vplane, int Vpitch);
  1396. #endif
  1397. static bool D3D11_UpdateTexture(SDL_Renderer *renderer, SDL_Texture *texture,
  1398. const SDL_Rect *rect, const void *srcPixels,
  1399. int srcPitch)
  1400. {
  1401. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1402. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1403. if (!textureData) {
  1404. return SDL_SetError("Texture is not currently available");
  1405. }
  1406. #ifdef SDL_HAVE_YUV
  1407. if (textureData->nv12) {
  1408. const Uint8 *Yplane = (const Uint8 *)srcPixels;
  1409. const Uint8 *UVplane = Yplane + rect->h * srcPitch;
  1410. return D3D11_UpdateTextureNV(renderer, texture, rect, Yplane, srcPitch, UVplane, srcPitch);
  1411. } else if (textureData->yuv) {
  1412. int Ypitch = srcPitch;
  1413. int UVpitch = ((Ypitch + 1) / 2);
  1414. const Uint8 *Yplane = (const Uint8 *)srcPixels;
  1415. const Uint8 *Uplane = Yplane + rect->h * Ypitch;
  1416. const Uint8 *Vplane = Uplane + ((rect->h + 1) / 2) * UVpitch;
  1417. if (texture->format == SDL_PIXELFORMAT_YV12) {
  1418. return D3D11_UpdateTextureYUV(renderer, texture, rect, Yplane, Ypitch, Vplane, UVpitch, Uplane, UVpitch);
  1419. } else {
  1420. return D3D11_UpdateTextureYUV(renderer, texture, rect, Yplane, Ypitch, Uplane, UVpitch, Vplane, UVpitch);
  1421. }
  1422. }
  1423. #endif
  1424. if (!D3D11_UpdateTextureInternal(rendererData, textureData->mainTexture, SDL_BYTESPERPIXEL(texture->format), rect->x, rect->y, rect->w, rect->h, srcPixels, srcPitch)) {
  1425. return false;
  1426. }
  1427. return true;
  1428. }
  1429. #ifdef SDL_HAVE_YUV
  1430. static bool D3D11_UpdateTextureYUV(SDL_Renderer *renderer, SDL_Texture *texture,
  1431. const SDL_Rect *rect,
  1432. const Uint8 *Yplane, int Ypitch,
  1433. const Uint8 *Uplane, int Upitch,
  1434. const Uint8 *Vplane, int Vpitch)
  1435. {
  1436. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1437. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1438. if (!textureData) {
  1439. return SDL_SetError("Texture is not currently available");
  1440. }
  1441. if (!D3D11_UpdateTextureInternal(rendererData, textureData->mainTexture, SDL_BYTESPERPIXEL(texture->format), rect->x, rect->y, rect->w, rect->h, Yplane, Ypitch)) {
  1442. return false;
  1443. }
  1444. if (!D3D11_UpdateTextureInternal(rendererData, textureData->mainTextureU, SDL_BYTESPERPIXEL(texture->format), rect->x / 2, rect->y / 2, (rect->w + 1) / 2, (rect->h + 1) / 2, Uplane, Upitch)) {
  1445. return false;
  1446. }
  1447. if (!D3D11_UpdateTextureInternal(rendererData, textureData->mainTextureV, SDL_BYTESPERPIXEL(texture->format), rect->x / 2, rect->y / 2, (rect->w + 1) / 2, (rect->h + 1) / 2, Vplane, Vpitch)) {
  1448. return false;
  1449. }
  1450. return true;
  1451. }
  1452. static bool D3D11_UpdateTextureNV(SDL_Renderer *renderer, SDL_Texture *texture,
  1453. const SDL_Rect *rect,
  1454. const Uint8 *Yplane, int Ypitch,
  1455. const Uint8 *UVplane, int UVpitch)
  1456. {
  1457. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1458. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1459. ID3D11Texture2D *stagingTexture;
  1460. const Uint8 *src;
  1461. Uint8 *dst;
  1462. int w, h, row;
  1463. UINT length;
  1464. HRESULT result;
  1465. D3D11_TEXTURE2D_DESC stagingTextureDesc;
  1466. D3D11_MAPPED_SUBRESOURCE textureMemory;
  1467. if (!textureData) {
  1468. return SDL_SetError("Texture is not currently available");
  1469. }
  1470. w = rect->w;
  1471. h = rect->h;
  1472. // Create a 'staging' texture, which will be used to write to a portion of the main texture.
  1473. ID3D11Texture2D_GetDesc(textureData->mainTexture, &stagingTextureDesc);
  1474. stagingTextureDesc.Width = w;
  1475. stagingTextureDesc.Height = h;
  1476. stagingTextureDesc.BindFlags = 0;
  1477. stagingTextureDesc.MiscFlags = 0;
  1478. stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
  1479. stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
  1480. if (stagingTextureDesc.Format == DXGI_FORMAT_NV12 ||
  1481. stagingTextureDesc.Format == DXGI_FORMAT_P010) {
  1482. stagingTextureDesc.Width = (stagingTextureDesc.Width + 1) & ~1;
  1483. stagingTextureDesc.Height = (stagingTextureDesc.Height + 1) & ~1;
  1484. }
  1485. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1486. &stagingTextureDesc,
  1487. NULL,
  1488. &stagingTexture);
  1489. if (FAILED(result)) {
  1490. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D [create staging texture]"), result);
  1491. }
  1492. // Get a write-only pointer to data in the staging texture:
  1493. result = ID3D11DeviceContext_Map(rendererData->d3dContext,
  1494. (ID3D11Resource *)stagingTexture,
  1495. 0,
  1496. D3D11_MAP_WRITE,
  1497. 0,
  1498. &textureMemory);
  1499. if (FAILED(result)) {
  1500. SAFE_RELEASE(stagingTexture);
  1501. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext1::Map [map staging texture]"), result);
  1502. }
  1503. src = Yplane;
  1504. dst = (Uint8 *)textureMemory.pData;
  1505. length = w;
  1506. if (length == (UINT)Ypitch && length == textureMemory.RowPitch) {
  1507. SDL_memcpy(dst, src, (size_t)length * h);
  1508. } else {
  1509. if (length > (UINT)Ypitch) {
  1510. length = Ypitch;
  1511. }
  1512. if (length > textureMemory.RowPitch) {
  1513. length = textureMemory.RowPitch;
  1514. }
  1515. for (row = 0; row < h; ++row) {
  1516. SDL_memcpy(dst, src, length);
  1517. src += Ypitch;
  1518. dst += textureMemory.RowPitch;
  1519. }
  1520. }
  1521. src = UVplane;
  1522. length = w;
  1523. h = (h + 1) / 2;
  1524. if (stagingTextureDesc.Format == DXGI_FORMAT_P010) {
  1525. length = (length + 3) & ~3;
  1526. UVpitch = (UVpitch + 3) & ~3;
  1527. } else {
  1528. length = (length + 1) & ~1;
  1529. UVpitch = (UVpitch + 1) & ~1;
  1530. }
  1531. dst = (Uint8 *)textureMemory.pData + stagingTextureDesc.Height * textureMemory.RowPitch;
  1532. for (row = 0; row < h; ++row) {
  1533. SDL_memcpy(dst, src, length);
  1534. src += UVpitch;
  1535. dst += textureMemory.RowPitch;
  1536. }
  1537. // Commit the pixel buffer's changes back to the staging texture:
  1538. ID3D11DeviceContext_Unmap(rendererData->d3dContext,
  1539. (ID3D11Resource *)stagingTexture,
  1540. 0);
  1541. // Copy the staging texture's contents back to the texture:
  1542. ID3D11DeviceContext_CopySubresourceRegion(rendererData->d3dContext,
  1543. (ID3D11Resource *)textureData->mainTexture,
  1544. 0,
  1545. rect->x,
  1546. rect->y,
  1547. 0,
  1548. (ID3D11Resource *)stagingTexture,
  1549. 0,
  1550. NULL);
  1551. SAFE_RELEASE(stagingTexture);
  1552. return true;
  1553. }
  1554. #endif
  1555. static bool D3D11_LockTexture(SDL_Renderer *renderer, SDL_Texture *texture,
  1556. const SDL_Rect *rect, void **pixels, int *pitch)
  1557. {
  1558. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1559. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1560. HRESULT result = S_OK;
  1561. D3D11_TEXTURE2D_DESC stagingTextureDesc;
  1562. D3D11_MAPPED_SUBRESOURCE textureMemory;
  1563. if (!textureData) {
  1564. return SDL_SetError("Texture is not currently available");
  1565. }
  1566. #ifdef SDL_HAVE_YUV
  1567. if (textureData->yuv || textureData->nv12) {
  1568. // It's more efficient to upload directly...
  1569. if (!textureData->pixels) {
  1570. textureData->pitch = texture->w;
  1571. textureData->pixels = (Uint8 *)SDL_malloc((texture->h * textureData->pitch * 3) / 2);
  1572. if (!textureData->pixels) {
  1573. return false;
  1574. }
  1575. }
  1576. textureData->locked_rect = *rect;
  1577. *pixels =
  1578. (void *)(textureData->pixels + rect->y * textureData->pitch +
  1579. rect->x * SDL_BYTESPERPIXEL(texture->format));
  1580. *pitch = textureData->pitch;
  1581. return true;
  1582. }
  1583. #endif
  1584. if (textureData->stagingTexture) {
  1585. return SDL_SetError("texture is already locked");
  1586. }
  1587. /* Create a 'staging' texture, which will be used to write to a portion
  1588. * of the main texture. This is necessary, as Direct3D 11.1 does not
  1589. * have the ability to write a CPU-bound pixel buffer to a rectangular
  1590. * subrect of a texture. Direct3D 11.1 can, however, write a pixel
  1591. * buffer to an entire texture, hence the use of a staging texture.
  1592. */
  1593. ID3D11Texture2D_GetDesc(textureData->mainTexture, &stagingTextureDesc);
  1594. stagingTextureDesc.Width = rect->w;
  1595. stagingTextureDesc.Height = rect->h;
  1596. stagingTextureDesc.BindFlags = 0;
  1597. stagingTextureDesc.MiscFlags = 0;
  1598. stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
  1599. stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
  1600. result = ID3D11Device_CreateTexture2D(rendererData->d3dDevice,
  1601. &stagingTextureDesc,
  1602. NULL,
  1603. &textureData->stagingTexture);
  1604. if (FAILED(result)) {
  1605. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D [create staging texture]"), result);
  1606. }
  1607. // Get a write-only pointer to data in the staging texture:
  1608. result = ID3D11DeviceContext_Map(rendererData->d3dContext,
  1609. (ID3D11Resource *)textureData->stagingTexture,
  1610. 0,
  1611. D3D11_MAP_WRITE,
  1612. 0,
  1613. &textureMemory);
  1614. if (FAILED(result)) {
  1615. SAFE_RELEASE(textureData->stagingTexture);
  1616. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext1::Map [map staging texture]"), result);
  1617. }
  1618. /* Make note of where the staging texture will be written to
  1619. * (on a call to SDL_UnlockTexture):
  1620. */
  1621. textureData->lockedTexturePositionX = rect->x;
  1622. textureData->lockedTexturePositionY = rect->y;
  1623. /* Make sure the caller has information on the texture's pixel buffer,
  1624. * then return:
  1625. */
  1626. *pixels = textureMemory.pData;
  1627. *pitch = textureMemory.RowPitch;
  1628. return true;
  1629. }
  1630. static void D3D11_UnlockTexture(SDL_Renderer *renderer, SDL_Texture *texture)
  1631. {
  1632. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1633. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1634. if (!textureData) {
  1635. return;
  1636. }
  1637. #ifdef SDL_HAVE_YUV
  1638. if (textureData->yuv || textureData->nv12) {
  1639. const SDL_Rect *rect = &textureData->locked_rect;
  1640. void *pixels =
  1641. (void *)(textureData->pixels + rect->y * textureData->pitch +
  1642. rect->x * SDL_BYTESPERPIXEL(texture->format));
  1643. D3D11_UpdateTexture(renderer, texture, rect, pixels, textureData->pitch);
  1644. return;
  1645. }
  1646. #endif
  1647. // Commit the pixel buffer's changes back to the staging texture:
  1648. ID3D11DeviceContext_Unmap(rendererData->d3dContext,
  1649. (ID3D11Resource *)textureData->stagingTexture,
  1650. 0);
  1651. // Copy the staging texture's contents back to the main texture:
  1652. ID3D11DeviceContext_CopySubresourceRegion(rendererData->d3dContext,
  1653. (ID3D11Resource *)textureData->mainTexture,
  1654. 0,
  1655. textureData->lockedTexturePositionX,
  1656. textureData->lockedTexturePositionY,
  1657. 0,
  1658. (ID3D11Resource *)textureData->stagingTexture,
  1659. 0,
  1660. NULL);
  1661. SAFE_RELEASE(textureData->stagingTexture);
  1662. }
  1663. static bool D3D11_SetRenderTarget(SDL_Renderer *renderer, SDL_Texture *texture)
  1664. {
  1665. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1666. D3D11_TextureData *textureData = NULL;
  1667. if (!texture) {
  1668. rendererData->currentOffscreenRenderTargetView = NULL;
  1669. return true;
  1670. }
  1671. textureData = (D3D11_TextureData *)texture->internal;
  1672. if (!textureData->mainTextureRenderTargetView) {
  1673. return SDL_SetError("specified texture is not a render target");
  1674. }
  1675. rendererData->currentOffscreenRenderTargetView = textureData->mainTextureRenderTargetView;
  1676. return true;
  1677. }
  1678. static bool D3D11_QueueNoOp(SDL_Renderer *renderer, SDL_RenderCommand *cmd)
  1679. {
  1680. return true; // nothing to do in this backend.
  1681. }
  1682. static bool D3D11_QueueDrawPoints(SDL_Renderer *renderer, SDL_RenderCommand *cmd, const SDL_FPoint *points, int count)
  1683. {
  1684. D3D11_VertexPositionColor *verts = (D3D11_VertexPositionColor *)SDL_AllocateRenderVertices(renderer, count * sizeof(D3D11_VertexPositionColor), 0, &cmd->data.draw.first);
  1685. int i;
  1686. SDL_FColor color = cmd->data.draw.color;
  1687. bool convert_color = SDL_RenderingLinearSpace(renderer);
  1688. if (!verts) {
  1689. return false;
  1690. }
  1691. cmd->data.draw.count = count;
  1692. if (convert_color) {
  1693. SDL_ConvertToLinear(&color);
  1694. }
  1695. for (i = 0; i < count; i++) {
  1696. verts->pos.x = points[i].x + 0.5f;
  1697. verts->pos.y = points[i].y + 0.5f;
  1698. verts->tex.x = 0.0f;
  1699. verts->tex.y = 0.0f;
  1700. verts->color = color;
  1701. verts++;
  1702. }
  1703. return true;
  1704. }
  1705. static bool D3D11_QueueGeometry(SDL_Renderer *renderer, SDL_RenderCommand *cmd, SDL_Texture *texture,
  1706. const float *xy, int xy_stride, const SDL_FColor *color, int color_stride, const float *uv, int uv_stride,
  1707. int num_vertices, const void *indices, int num_indices, int size_indices,
  1708. float scale_x, float scale_y)
  1709. {
  1710. int i;
  1711. int count = indices ? num_indices : num_vertices;
  1712. D3D11_VertexPositionColor *verts = (D3D11_VertexPositionColor *)SDL_AllocateRenderVertices(renderer, count * sizeof(D3D11_VertexPositionColor), 0, &cmd->data.draw.first);
  1713. bool convert_color = SDL_RenderingLinearSpace(renderer);
  1714. D3D11_TextureData *textureData = texture ? (D3D11_TextureData *)texture->internal : NULL;
  1715. float u_scale = textureData ? (float)texture->w / textureData->w : 0.0f;
  1716. float v_scale = textureData ? (float)texture->h / textureData->h : 0.0f;
  1717. if (!verts) {
  1718. return false;
  1719. }
  1720. cmd->data.draw.count = count;
  1721. size_indices = indices ? size_indices : 0;
  1722. for (i = 0; i < count; i++) {
  1723. int j;
  1724. float *xy_;
  1725. if (size_indices == 4) {
  1726. j = ((const Uint32 *)indices)[i];
  1727. } else if (size_indices == 2) {
  1728. j = ((const Uint16 *)indices)[i];
  1729. } else if (size_indices == 1) {
  1730. j = ((const Uint8 *)indices)[i];
  1731. } else {
  1732. j = i;
  1733. }
  1734. xy_ = (float *)((char *)xy + j * xy_stride);
  1735. verts->pos.x = xy_[0] * scale_x;
  1736. verts->pos.y = xy_[1] * scale_y;
  1737. verts->color = *(SDL_FColor *)((char *)color + j * color_stride);
  1738. if (convert_color) {
  1739. SDL_ConvertToLinear(&verts->color);
  1740. }
  1741. if (texture) {
  1742. float *uv_ = (float *)((char *)uv + j * uv_stride);
  1743. verts->tex.x = uv_[0] * u_scale;
  1744. verts->tex.y = uv_[1] * v_scale;
  1745. } else {
  1746. verts->tex.x = 0.0f;
  1747. verts->tex.y = 0.0f;
  1748. }
  1749. verts += 1;
  1750. }
  1751. return true;
  1752. }
  1753. static bool D3D11_UpdateVertexBuffer(SDL_Renderer *renderer,
  1754. const void *vertexData, size_t dataSizeInBytes)
  1755. {
  1756. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1757. HRESULT result = S_OK;
  1758. const int vbidx = rendererData->currentVertexBuffer;
  1759. const UINT stride = sizeof(D3D11_VertexPositionColor);
  1760. const UINT offset = 0;
  1761. if (dataSizeInBytes == 0) {
  1762. return true; // nothing to do.
  1763. }
  1764. if (rendererData->vertexBuffers[vbidx] && rendererData->vertexBufferSizes[vbidx] >= dataSizeInBytes) {
  1765. D3D11_MAPPED_SUBRESOURCE mappedResource;
  1766. result = ID3D11DeviceContext_Map(rendererData->d3dContext,
  1767. (ID3D11Resource *)rendererData->vertexBuffers[vbidx],
  1768. 0,
  1769. D3D11_MAP_WRITE_DISCARD,
  1770. 0,
  1771. &mappedResource);
  1772. if (FAILED(result)) {
  1773. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext1::Map [vertex buffer]"), result);
  1774. }
  1775. SDL_memcpy(mappedResource.pData, vertexData, dataSizeInBytes);
  1776. ID3D11DeviceContext_Unmap(rendererData->d3dContext, (ID3D11Resource *)rendererData->vertexBuffers[vbidx], 0);
  1777. } else {
  1778. D3D11_BUFFER_DESC vertexBufferDesc;
  1779. D3D11_SUBRESOURCE_DATA vertexBufferData;
  1780. SAFE_RELEASE(rendererData->vertexBuffers[vbidx]);
  1781. SDL_zero(vertexBufferDesc);
  1782. vertexBufferDesc.ByteWidth = (UINT)dataSizeInBytes;
  1783. vertexBufferDesc.Usage = D3D11_USAGE_DYNAMIC;
  1784. vertexBufferDesc.BindFlags = D3D11_BIND_VERTEX_BUFFER;
  1785. vertexBufferDesc.CPUAccessFlags = D3D11_CPU_ACCESS_WRITE;
  1786. SDL_zero(vertexBufferData);
  1787. vertexBufferData.pSysMem = vertexData;
  1788. vertexBufferData.SysMemPitch = 0;
  1789. vertexBufferData.SysMemSlicePitch = 0;
  1790. result = ID3D11Device_CreateBuffer(rendererData->d3dDevice,
  1791. &vertexBufferDesc,
  1792. &vertexBufferData,
  1793. &rendererData->vertexBuffers[vbidx]);
  1794. if (FAILED(result)) {
  1795. return WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateBuffer [vertex buffer]"), result);
  1796. }
  1797. rendererData->vertexBufferSizes[vbidx] = dataSizeInBytes;
  1798. }
  1799. ID3D11DeviceContext_IASetVertexBuffers(rendererData->d3dContext,
  1800. 0,
  1801. 1,
  1802. &rendererData->vertexBuffers[vbidx],
  1803. &stride,
  1804. &offset);
  1805. rendererData->currentVertexBuffer++;
  1806. if (rendererData->currentVertexBuffer >= SDL_arraysize(rendererData->vertexBuffers)) {
  1807. rendererData->currentVertexBuffer = 0;
  1808. }
  1809. return true;
  1810. }
  1811. static bool D3D11_UpdateViewport(SDL_Renderer *renderer)
  1812. {
  1813. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  1814. const SDL_Rect *viewport = &data->currentViewport;
  1815. Float4X4 projection;
  1816. Float4X4 view;
  1817. SDL_FRect orientationAlignedViewport;
  1818. BOOL swapDimensions;
  1819. D3D11_VIEWPORT d3dviewport;
  1820. const int rotation = D3D11_GetRotationForCurrentRenderTarget(renderer);
  1821. if (viewport->w == 0 || viewport->h == 0) {
  1822. /* If the viewport is empty, assume that it is because
  1823. * SDL_CreateRenderer is calling it, and will call it again later
  1824. * with a non-empty viewport.
  1825. */
  1826. // SDL_Log("%s, no viewport was set!", __FUNCTION__);
  1827. return false;
  1828. }
  1829. /* Make sure the SDL viewport gets rotated to that of the physical display's rotation.
  1830. * Keep in mind here that the Y-axis will be been inverted (from Direct3D's
  1831. * default coordinate system) so rotations will be done in the opposite
  1832. * direction of the DXGI_MODE_ROTATION enumeration.
  1833. */
  1834. switch (rotation) {
  1835. case DXGI_MODE_ROTATION_IDENTITY:
  1836. projection = MatrixIdentity();
  1837. break;
  1838. case DXGI_MODE_ROTATION_ROTATE270:
  1839. projection = MatrixRotationZ(SDL_PI_F * 0.5f);
  1840. break;
  1841. case DXGI_MODE_ROTATION_ROTATE180:
  1842. projection = MatrixRotationZ(SDL_PI_F);
  1843. break;
  1844. case DXGI_MODE_ROTATION_ROTATE90:
  1845. projection = MatrixRotationZ(-SDL_PI_F * 0.5f);
  1846. break;
  1847. default:
  1848. return SDL_SetError("An unknown DisplayOrientation is being used");
  1849. }
  1850. // Update the view matrix
  1851. SDL_zero(view);
  1852. view.m[0][0] = 2.0f / viewport->w;
  1853. view.m[1][1] = -2.0f / viewport->h;
  1854. view.m[2][2] = 1.0f;
  1855. view.m[3][0] = -1.0f;
  1856. view.m[3][1] = 1.0f;
  1857. view.m[3][3] = 1.0f;
  1858. /* Combine the projection + view matrix together now, as both only get
  1859. * set here (as of this writing, on Dec 26, 2013). When done, store it
  1860. * for eventual transfer to the GPU.
  1861. */
  1862. data->vertexShaderConstantsData.projectionAndView = MatrixMultiply(
  1863. view,
  1864. projection);
  1865. /* Update the Direct3D viewport, which seems to be aligned to the
  1866. * swap buffer's coordinate space, which is always in either
  1867. * a landscape mode, for all Windows 8/RT devices, or a portrait mode,
  1868. * for Windows Phone devices.
  1869. */
  1870. swapDimensions = D3D11_IsDisplayRotated90Degrees(rotation);
  1871. if (swapDimensions) {
  1872. orientationAlignedViewport.x = (float)viewport->y;
  1873. orientationAlignedViewport.y = (float)viewport->x;
  1874. orientationAlignedViewport.w = (float)viewport->h;
  1875. orientationAlignedViewport.h = (float)viewport->w;
  1876. } else {
  1877. orientationAlignedViewport.x = (float)viewport->x;
  1878. orientationAlignedViewport.y = (float)viewport->y;
  1879. orientationAlignedViewport.w = (float)viewport->w;
  1880. orientationAlignedViewport.h = (float)viewport->h;
  1881. }
  1882. d3dviewport.TopLeftX = orientationAlignedViewport.x;
  1883. d3dviewport.TopLeftY = orientationAlignedViewport.y;
  1884. d3dviewport.Width = orientationAlignedViewport.w;
  1885. d3dviewport.Height = orientationAlignedViewport.h;
  1886. d3dviewport.MinDepth = 0.0f;
  1887. d3dviewport.MaxDepth = 1.0f;
  1888. // SDL_Log("%s: D3D viewport = {%f,%f,%f,%f}", __FUNCTION__, d3dviewport.TopLeftX, d3dviewport.TopLeftY, d3dviewport.Width, d3dviewport.Height);
  1889. ID3D11DeviceContext_RSSetViewports(data->d3dContext, 1, &d3dviewport);
  1890. data->viewportDirty = false;
  1891. return true;
  1892. }
  1893. static ID3D11RenderTargetView *D3D11_GetCurrentRenderTargetView(SDL_Renderer *renderer)
  1894. {
  1895. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  1896. if (data->currentOffscreenRenderTargetView) {
  1897. return data->currentOffscreenRenderTargetView;
  1898. } else {
  1899. return data->mainRenderTargetView;
  1900. }
  1901. }
  1902. static void D3D11_SetupShaderConstants(SDL_Renderer *renderer, const SDL_RenderCommand *cmd, const SDL_Texture *texture, D3D11_PixelShaderConstants *constants)
  1903. {
  1904. float output_headroom;
  1905. SDL_zerop(constants);
  1906. constants->scRGB_output = (float)SDL_RenderingLinearSpace(renderer);
  1907. constants->color_scale = cmd->data.draw.color_scale;
  1908. if (texture) {
  1909. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  1910. switch (texture->format) {
  1911. case SDL_PIXELFORMAT_INDEX8:
  1912. if (cmd->data.draw.texture_scale_mode == SDL_SCALEMODE_PIXELART) {
  1913. constants->texture_type = TEXTURETYPE_PALETTE_PIXELART;
  1914. constants->texture_width = texture->w;
  1915. constants->texture_height = texture->h;
  1916. constants->texel_width = 1.0f / constants->texture_width;
  1917. constants->texel_height = 1.0f / constants->texture_height;
  1918. } else {
  1919. constants->texture_type = TEXTURETYPE_PALETTE;
  1920. }
  1921. constants->input_type = INPUTTYPE_UNSPECIFIED;
  1922. break;
  1923. case SDL_PIXELFORMAT_YV12:
  1924. case SDL_PIXELFORMAT_IYUV:
  1925. constants->texture_type = TEXTURETYPE_YUV;
  1926. constants->input_type = INPUTTYPE_SRGB;
  1927. break;
  1928. case SDL_PIXELFORMAT_NV12:
  1929. constants->texture_type = TEXTURETYPE_NV12;
  1930. constants->input_type = INPUTTYPE_SRGB;
  1931. break;
  1932. case SDL_PIXELFORMAT_NV21:
  1933. constants->texture_type = TEXTURETYPE_NV21;
  1934. constants->input_type = INPUTTYPE_SRGB;
  1935. break;
  1936. case SDL_PIXELFORMAT_P010:
  1937. constants->texture_type = TEXTURETYPE_NV12;
  1938. constants->input_type = INPUTTYPE_HDR10;
  1939. break;
  1940. default:
  1941. if (cmd->data.draw.texture_scale_mode == SDL_SCALEMODE_PIXELART) {
  1942. constants->texture_type = TEXTURETYPE_RGB_PIXELART;
  1943. constants->texture_width = texture->w;
  1944. constants->texture_height = texture->h;
  1945. constants->texel_width = 1.0f / constants->texture_width;
  1946. constants->texel_height = 1.0f / constants->texture_height;
  1947. } else {
  1948. constants->texture_type = TEXTURETYPE_RGB;
  1949. }
  1950. if (texture->colorspace == SDL_COLORSPACE_SRGB_LINEAR) {
  1951. constants->input_type = INPUTTYPE_SCRGB;
  1952. } else if (texture->colorspace == SDL_COLORSPACE_HDR10) {
  1953. constants->input_type = INPUTTYPE_HDR10;
  1954. } else {
  1955. // The sampler will convert from sRGB to linear on load if working in linear colorspace
  1956. constants->input_type = INPUTTYPE_UNSPECIFIED;
  1957. }
  1958. break;
  1959. }
  1960. constants->sdr_white_point = texture->SDR_white_point;
  1961. if (renderer->target) {
  1962. output_headroom = renderer->target->HDR_headroom;
  1963. } else {
  1964. output_headroom = renderer->HDR_headroom;
  1965. }
  1966. if (texture->HDR_headroom > output_headroom) {
  1967. constants->tonemap_method = TONEMAP_CHROME;
  1968. constants->tonemap_factor1 = (output_headroom / (texture->HDR_headroom * texture->HDR_headroom));
  1969. constants->tonemap_factor2 = (1.0f / output_headroom);
  1970. }
  1971. if (textureData->YCbCr_matrix) {
  1972. SDL_memcpy(constants->YCbCr_matrix, textureData->YCbCr_matrix, sizeof(constants->YCbCr_matrix));
  1973. }
  1974. }
  1975. }
  1976. static D3D11_Shader SelectShader(const D3D11_PixelShaderConstants *shader_constants)
  1977. {
  1978. if (!shader_constants) {
  1979. return SHADER_SOLID;
  1980. }
  1981. if (shader_constants->texture_type == TEXTURETYPE_RGB &&
  1982. shader_constants->input_type == INPUTTYPE_UNSPECIFIED &&
  1983. shader_constants->tonemap_method == TONEMAP_NONE) {
  1984. return SHADER_RGB;
  1985. }
  1986. return SHADER_ADVANCED;
  1987. }
  1988. static bool D3D11_SetDrawState(SDL_Renderer *renderer, const SDL_RenderCommand *cmd,
  1989. const D3D11_PixelShaderConstants *shader_constants,
  1990. int numShaderResources, ID3D11ShaderResourceView **shaderResources,
  1991. int numShaderSamplers, ID3D11SamplerState **shaderSamplers, const Float4X4 *matrix)
  1992. {
  1993. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  1994. const Float4X4 *newmatrix = matrix ? matrix : &rendererData->identity;
  1995. ID3D11RasterizerState *rasterizerState;
  1996. ID3D11RenderTargetView *renderTargetView = D3D11_GetCurrentRenderTargetView(renderer);
  1997. const SDL_BlendMode blendMode = cmd->data.draw.blend;
  1998. ID3D11BlendState *blendState = NULL;
  1999. bool updateSubresource = false;
  2000. D3D11_Shader shader = SelectShader(shader_constants);
  2001. D3D11_PixelShaderState *shader_state = &rendererData->currentShaderState[shader];
  2002. D3D11_PixelShaderConstants solid_constants;
  2003. bool shaderResourcesChanged = false;
  2004. if (numShaderResources != rendererData->numCurrentShaderResources ||
  2005. (numShaderResources > 0 && shaderResources[0] != rendererData->currentShaderResource)) {
  2006. shaderResourcesChanged = true;
  2007. }
  2008. bool shaderSamplersChanged = false;
  2009. if (numShaderSamplers != rendererData->numCurrentShaderSamplers ||
  2010. (numShaderSamplers > 0 && shaderSamplers[0] != rendererData->currentShaderSampler)) {
  2011. shaderSamplersChanged = true;
  2012. }
  2013. // Make sure the render target isn't bound to a shader
  2014. if (shaderResourcesChanged) {
  2015. ID3D11ShaderResourceView *pNullResource = NULL;
  2016. ID3D11DeviceContext_PSSetShaderResources(rendererData->d3dContext, 0, 1, &pNullResource);
  2017. }
  2018. if (renderTargetView != rendererData->currentRenderTargetView) {
  2019. ID3D11DeviceContext_OMSetRenderTargets(rendererData->d3dContext,
  2020. 1,
  2021. &renderTargetView,
  2022. NULL);
  2023. rendererData->currentRenderTargetView = renderTargetView;
  2024. }
  2025. if (rendererData->viewportDirty) {
  2026. if (D3D11_UpdateViewport(renderer)) {
  2027. // vertexShaderConstantsData.projectionAndView has changed
  2028. updateSubresource = true;
  2029. }
  2030. }
  2031. if (rendererData->cliprectDirty) {
  2032. if (!rendererData->currentCliprectEnabled) {
  2033. ID3D11DeviceContext_RSSetScissorRects(rendererData->d3dContext, 0, NULL);
  2034. } else {
  2035. D3D11_RECT scissorRect;
  2036. if (!D3D11_GetViewportAlignedD3DRect(renderer, &rendererData->currentCliprect, &scissorRect, TRUE)) {
  2037. // D3D11_GetViewportAlignedD3DRect will have set the SDL error
  2038. return false;
  2039. }
  2040. ID3D11DeviceContext_RSSetScissorRects(rendererData->d3dContext, 1, &scissorRect);
  2041. }
  2042. rendererData->cliprectDirty = false;
  2043. }
  2044. if (!rendererData->currentCliprectEnabled) {
  2045. rasterizerState = rendererData->mainRasterizer;
  2046. } else {
  2047. rasterizerState = rendererData->clippedRasterizer;
  2048. }
  2049. if (rasterizerState != rendererData->currentRasterizerState) {
  2050. ID3D11DeviceContext_RSSetState(rendererData->d3dContext, rasterizerState);
  2051. rendererData->currentRasterizerState = rasterizerState;
  2052. }
  2053. if (blendMode != SDL_BLENDMODE_NONE) {
  2054. int i;
  2055. for (i = 0; i < rendererData->blendModesCount; ++i) {
  2056. if (blendMode == rendererData->blendModes[i].blendMode) {
  2057. blendState = rendererData->blendModes[i].blendState;
  2058. break;
  2059. }
  2060. }
  2061. if (!blendState) {
  2062. blendState = D3D11_CreateBlendState(renderer, blendMode);
  2063. if (!blendState) {
  2064. return false;
  2065. }
  2066. }
  2067. }
  2068. if (blendState != rendererData->currentBlendState) {
  2069. ID3D11DeviceContext_OMSetBlendState(rendererData->d3dContext, blendState, 0, 0xFFFFFFFF);
  2070. rendererData->currentBlendState = blendState;
  2071. }
  2072. if (!shader_constants) {
  2073. D3D11_SetupShaderConstants(renderer, cmd, NULL, &solid_constants);
  2074. shader_constants = &solid_constants;
  2075. }
  2076. if (!shader_state->constants ||
  2077. SDL_memcmp(shader_constants, &shader_state->shader_constants, sizeof(*shader_constants)) != 0) {
  2078. SAFE_RELEASE(shader_state->constants);
  2079. D3D11_BUFFER_DESC desc;
  2080. SDL_zero(desc);
  2081. desc.Usage = D3D11_USAGE_DEFAULT;
  2082. desc.ByteWidth = sizeof(*shader_constants);
  2083. desc.BindFlags = D3D11_BIND_CONSTANT_BUFFER;
  2084. D3D11_SUBRESOURCE_DATA data;
  2085. SDL_zero(data);
  2086. data.pSysMem = shader_constants;
  2087. HRESULT result = ID3D11Device_CreateBuffer(rendererData->d3dDevice, &desc, &data, &shader_state->constants);
  2088. if (FAILED(result)) {
  2089. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device::CreateBuffer [create shader constants]"), result);
  2090. return false;
  2091. }
  2092. SDL_memcpy(&shader_state->shader_constants, shader_constants, sizeof(*shader_constants));
  2093. // Force the shader parameters to be re-set
  2094. rendererData->currentShader = SHADER_NONE;
  2095. }
  2096. if (shader != rendererData->currentShader) {
  2097. if (!rendererData->pixelShaders[shader]) {
  2098. if (!D3D11_CreatePixelShader(rendererData->d3dDevice, shader, &rendererData->pixelShaders[shader])) {
  2099. return false;
  2100. }
  2101. }
  2102. ID3D11DeviceContext_PSSetShader(rendererData->d3dContext, rendererData->pixelShaders[shader], NULL, 0);
  2103. if (shader_state->constants) {
  2104. ID3D11DeviceContext_PSSetConstantBuffers(rendererData->d3dContext, 0, 1, &shader_state->constants);
  2105. }
  2106. rendererData->currentShader = shader;
  2107. }
  2108. if (shaderResourcesChanged) {
  2109. ID3D11DeviceContext_PSSetShaderResources(rendererData->d3dContext, 0, numShaderResources, shaderResources);
  2110. rendererData->numCurrentShaderResources = numShaderResources;
  2111. if (numShaderResources > 0) {
  2112. rendererData->currentShaderResource = shaderResources[0];
  2113. }
  2114. }
  2115. if (shaderSamplersChanged) {
  2116. ID3D11DeviceContext_PSSetSamplers(rendererData->d3dContext, 0, numShaderSamplers, shaderSamplers);
  2117. rendererData->numCurrentShaderSamplers = numShaderSamplers;
  2118. if (numShaderSamplers) {
  2119. rendererData->currentShaderSampler = shaderSamplers[0];
  2120. }
  2121. }
  2122. if (updateSubresource == true || SDL_memcmp(&rendererData->vertexShaderConstantsData.model, newmatrix, sizeof(*newmatrix)) != 0) {
  2123. SDL_copyp(&rendererData->vertexShaderConstantsData.model, newmatrix);
  2124. ID3D11DeviceContext_UpdateSubresource(rendererData->d3dContext,
  2125. (ID3D11Resource *)rendererData->vertexShaderConstants,
  2126. 0,
  2127. NULL,
  2128. &rendererData->vertexShaderConstantsData,
  2129. 0,
  2130. 0);
  2131. }
  2132. return true;
  2133. }
  2134. static ID3D11SamplerState *D3D11_GetSamplerState(D3D11_RenderData *data, SDL_ScaleMode scale_mode, SDL_TextureAddressMode address_u, SDL_TextureAddressMode address_v)
  2135. {
  2136. Uint32 key = RENDER_SAMPLER_HASHKEY(scale_mode, address_u, address_v);
  2137. SDL_assert(key < SDL_arraysize(data->samplers));
  2138. if (!data->samplers[key]) {
  2139. D3D11_SAMPLER_DESC samplerDesc;
  2140. SDL_zero(samplerDesc);
  2141. samplerDesc.AddressW = D3D11_TEXTURE_ADDRESS_CLAMP;
  2142. samplerDesc.MipLODBias = 0.0f;
  2143. samplerDesc.MaxAnisotropy = 1;
  2144. samplerDesc.ComparisonFunc = D3D11_COMPARISON_ALWAYS;
  2145. samplerDesc.MinLOD = 0.0f;
  2146. samplerDesc.MaxLOD = D3D11_FLOAT32_MAX;
  2147. switch (scale_mode) {
  2148. case SDL_SCALEMODE_NEAREST:
  2149. samplerDesc.Filter = D3D11_FILTER_MIN_MAG_MIP_POINT;
  2150. break;
  2151. case SDL_SCALEMODE_PIXELART: // Uses linear sampling
  2152. case SDL_SCALEMODE_LINEAR:
  2153. samplerDesc.Filter = D3D11_FILTER_MIN_MAG_MIP_LINEAR;
  2154. break;
  2155. default:
  2156. SDL_SetError("Unknown scale mode: %d", scale_mode);
  2157. return NULL;
  2158. }
  2159. switch (address_u) {
  2160. case SDL_TEXTURE_ADDRESS_CLAMP:
  2161. samplerDesc.AddressU = D3D11_TEXTURE_ADDRESS_CLAMP;
  2162. break;
  2163. case SDL_TEXTURE_ADDRESS_WRAP:
  2164. samplerDesc.AddressU = D3D11_TEXTURE_ADDRESS_WRAP;
  2165. break;
  2166. default:
  2167. SDL_SetError("Unknown texture address mode: %d", address_u);
  2168. return NULL;
  2169. }
  2170. switch (address_v) {
  2171. case SDL_TEXTURE_ADDRESS_CLAMP:
  2172. samplerDesc.AddressV = D3D11_TEXTURE_ADDRESS_CLAMP;
  2173. break;
  2174. case SDL_TEXTURE_ADDRESS_WRAP:
  2175. samplerDesc.AddressV = D3D11_TEXTURE_ADDRESS_WRAP;
  2176. break;
  2177. default:
  2178. SDL_SetError("Unknown texture address mode: %d", address_v);
  2179. return NULL;
  2180. }
  2181. HRESULT result = ID3D11Device_CreateSamplerState(data->d3dDevice,
  2182. &samplerDesc,
  2183. &data->samplers[key]);
  2184. if (FAILED(result)) {
  2185. WIN_SetErrorFromHRESULT("ID3D11Device::CreateSamplerState", result);
  2186. return NULL;
  2187. }
  2188. }
  2189. return data->samplers[key];
  2190. }
  2191. static bool D3D11_SetCopyState(SDL_Renderer *renderer, const SDL_RenderCommand *cmd, const Float4X4 *matrix)
  2192. {
  2193. SDL_Texture *texture = cmd->data.draw.texture;
  2194. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  2195. D3D11_TextureData *textureData = (D3D11_TextureData *)texture->internal;
  2196. int numShaderResources = 0;
  2197. ID3D11ShaderResourceView *shaderResources[3];
  2198. int numShaderSamplers = 0;
  2199. ID3D11SamplerState *shaderSamplers[2];
  2200. D3D11_PixelShaderConstants constants;
  2201. if (!textureData) {
  2202. return SDL_SetError("Texture is not currently available");
  2203. }
  2204. D3D11_SetupShaderConstants(renderer, cmd, texture, &constants);
  2205. shaderResources[numShaderResources++] = textureData->mainTextureResourceView;
  2206. shaderSamplers[numShaderSamplers] = D3D11_GetSamplerState(rendererData, cmd->data.draw.texture_scale_mode, cmd->data.draw.texture_address_mode_u, cmd->data.draw.texture_address_mode_v);
  2207. if (!shaderSamplers[numShaderSamplers]) {
  2208. return false;
  2209. }
  2210. ++numShaderSamplers;
  2211. if (texture->format == SDL_PIXELFORMAT_INDEX8) {
  2212. shaderResources[numShaderResources++] = textureData->paletteTextureResourceView;
  2213. shaderSamplers[numShaderSamplers] = D3D11_GetSamplerState(rendererData, SDL_SCALEMODE_NEAREST, SDL_TEXTURE_ADDRESS_CLAMP, SDL_TEXTURE_ADDRESS_CLAMP);
  2214. if (!shaderSamplers[numShaderSamplers]) {
  2215. return false;
  2216. }
  2217. ++numShaderSamplers;
  2218. }
  2219. #ifdef SDL_HAVE_YUV
  2220. if (textureData->yuv) {
  2221. shaderResources[numShaderResources++] = textureData->mainTextureResourceViewU;
  2222. shaderResources[numShaderResources++] = textureData->mainTextureResourceViewV;
  2223. } else if (textureData->nv12) {
  2224. shaderResources[numShaderResources++] = textureData->mainTextureResourceViewNV;
  2225. }
  2226. #endif // SDL_HAVE_YUV
  2227. return D3D11_SetDrawState(renderer, cmd, &constants, numShaderResources, shaderResources, numShaderSamplers, shaderSamplers, matrix);
  2228. }
  2229. static void D3D11_DrawPrimitives(SDL_Renderer *renderer, D3D11_PRIMITIVE_TOPOLOGY primitiveTopology, const size_t vertexStart, const size_t vertexCount)
  2230. {
  2231. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  2232. ID3D11DeviceContext_IASetPrimitiveTopology(rendererData->d3dContext, primitiveTopology);
  2233. ID3D11DeviceContext_Draw(rendererData->d3dContext, (UINT)vertexCount, (UINT)vertexStart);
  2234. }
  2235. static void D3D11_InvalidateCachedState(SDL_Renderer *renderer)
  2236. {
  2237. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  2238. data->currentRenderTargetView = NULL;
  2239. data->currentRasterizerState = NULL;
  2240. data->currentBlendState = NULL;
  2241. data->currentShader = SHADER_NONE;
  2242. data->numCurrentShaderResources = 0;
  2243. data->currentShaderResource = NULL;
  2244. data->numCurrentShaderSamplers = 0;
  2245. data->currentShaderSampler = NULL;
  2246. data->cliprectDirty = true;
  2247. data->viewportDirty = true;
  2248. }
  2249. static bool D3D11_RunCommandQueue(SDL_Renderer *renderer, SDL_RenderCommand *cmd, void *vertices, size_t vertsize)
  2250. {
  2251. D3D11_RenderData *rendererData = (D3D11_RenderData *)renderer->internal;
  2252. const int viewportRotation = D3D11_GetRotationForCurrentRenderTarget(renderer);
  2253. if (!rendererData->d3dDevice) {
  2254. return SDL_SetError("Device lost and couldn't be recovered");
  2255. }
  2256. if (rendererData->pixelSizeChanged) {
  2257. D3D11_UpdateForWindowSizeChange(renderer);
  2258. rendererData->pixelSizeChanged = false;
  2259. }
  2260. if (rendererData->currentViewportRotation != viewportRotation) {
  2261. rendererData->currentViewportRotation = viewportRotation;
  2262. rendererData->viewportDirty = true;
  2263. }
  2264. if (!D3D11_UpdateVertexBuffer(renderer, vertices, vertsize)) {
  2265. return false;
  2266. }
  2267. while (cmd) {
  2268. switch (cmd->command) {
  2269. case SDL_RENDERCMD_SETDRAWCOLOR:
  2270. {
  2271. break; // this isn't currently used in this render backend.
  2272. }
  2273. case SDL_RENDERCMD_SETVIEWPORT:
  2274. {
  2275. SDL_Rect *viewport = &rendererData->currentViewport;
  2276. if (SDL_memcmp(viewport, &cmd->data.viewport.rect, sizeof(cmd->data.viewport.rect)) != 0) {
  2277. SDL_copyp(viewport, &cmd->data.viewport.rect);
  2278. rendererData->viewportDirty = true;
  2279. rendererData->cliprectDirty = true;
  2280. }
  2281. break;
  2282. }
  2283. case SDL_RENDERCMD_SETCLIPRECT:
  2284. {
  2285. const SDL_Rect *rect = &cmd->data.cliprect.rect;
  2286. if (rendererData->currentCliprectEnabled != cmd->data.cliprect.enabled) {
  2287. rendererData->currentCliprectEnabled = cmd->data.cliprect.enabled;
  2288. rendererData->cliprectDirty = true;
  2289. }
  2290. if (SDL_memcmp(&rendererData->currentCliprect, rect, sizeof(*rect)) != 0) {
  2291. SDL_copyp(&rendererData->currentCliprect, rect);
  2292. rendererData->cliprectDirty = true;
  2293. }
  2294. break;
  2295. }
  2296. case SDL_RENDERCMD_CLEAR:
  2297. {
  2298. bool convert_color = SDL_RenderingLinearSpace(renderer);
  2299. SDL_FColor color = cmd->data.color.color;
  2300. if (convert_color) {
  2301. SDL_ConvertToLinear(&color);
  2302. }
  2303. color.r *= cmd->data.color.color_scale;
  2304. color.g *= cmd->data.color.color_scale;
  2305. color.b *= cmd->data.color.color_scale;
  2306. ID3D11DeviceContext_ClearRenderTargetView(rendererData->d3dContext, D3D11_GetCurrentRenderTargetView(renderer), &color.r);
  2307. break;
  2308. }
  2309. case SDL_RENDERCMD_DRAW_POINTS:
  2310. {
  2311. const size_t count = cmd->data.draw.count;
  2312. const size_t first = cmd->data.draw.first;
  2313. const size_t start = first / sizeof(D3D11_VertexPositionColor);
  2314. D3D11_SetDrawState(renderer, cmd, NULL, 0, NULL, 0, NULL, NULL);
  2315. D3D11_DrawPrimitives(renderer, D3D11_PRIMITIVE_TOPOLOGY_POINTLIST, start, count);
  2316. break;
  2317. }
  2318. case SDL_RENDERCMD_DRAW_LINES:
  2319. {
  2320. const size_t count = cmd->data.draw.count;
  2321. const size_t first = cmd->data.draw.first;
  2322. const size_t start = first / sizeof(D3D11_VertexPositionColor);
  2323. const D3D11_VertexPositionColor *verts = (D3D11_VertexPositionColor *)(((Uint8 *)vertices) + first);
  2324. D3D11_SetDrawState(renderer, cmd, NULL, 0, NULL, 0, NULL, NULL);
  2325. D3D11_DrawPrimitives(renderer, D3D11_PRIMITIVE_TOPOLOGY_LINESTRIP, start, count);
  2326. if (verts[0].pos.x != verts[count - 1].pos.x || verts[0].pos.y != verts[count - 1].pos.y) {
  2327. D3D11_DrawPrimitives(renderer, D3D11_PRIMITIVE_TOPOLOGY_POINTLIST, start + (count - 1), 1);
  2328. }
  2329. break;
  2330. }
  2331. case SDL_RENDERCMD_FILL_RECTS: // unused
  2332. break;
  2333. case SDL_RENDERCMD_COPY: // unused
  2334. break;
  2335. case SDL_RENDERCMD_COPY_EX: // unused
  2336. break;
  2337. case SDL_RENDERCMD_GEOMETRY:
  2338. {
  2339. SDL_Texture *texture = cmd->data.draw.texture;
  2340. const size_t count = cmd->data.draw.count;
  2341. const size_t first = cmd->data.draw.first;
  2342. const size_t start = first / sizeof(D3D11_VertexPositionColor);
  2343. if (texture) {
  2344. D3D11_SetCopyState(renderer, cmd, NULL);
  2345. } else {
  2346. D3D11_SetDrawState(renderer, cmd, NULL, 0, NULL, 0, NULL, NULL);
  2347. }
  2348. D3D11_DrawPrimitives(renderer, D3D11_PRIMITIVE_TOPOLOGY_TRIANGLELIST, start, count);
  2349. break;
  2350. }
  2351. case SDL_RENDERCMD_NO_OP:
  2352. break;
  2353. }
  2354. cmd = cmd->next;
  2355. }
  2356. return true;
  2357. }
  2358. static SDL_Surface *D3D11_RenderReadPixels(SDL_Renderer *renderer, const SDL_Rect *rect)
  2359. {
  2360. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  2361. ID3D11RenderTargetView *renderTargetView = NULL;
  2362. ID3D11Texture2D *backBuffer = NULL;
  2363. ID3D11Texture2D *stagingTexture = NULL;
  2364. HRESULT result;
  2365. D3D11_TEXTURE2D_DESC stagingTextureDesc;
  2366. D3D11_RECT srcRect = { 0, 0, 0, 0 };
  2367. D3D11_BOX srcBox;
  2368. D3D11_MAPPED_SUBRESOURCE textureMemory;
  2369. SDL_Surface *output = NULL;
  2370. renderTargetView = D3D11_GetCurrentRenderTargetView(renderer);
  2371. if (!renderTargetView) {
  2372. SDL_SetError("%s, ID3D11DeviceContext::OMGetRenderTargets failed", __FUNCTION__);
  2373. goto done;
  2374. }
  2375. ID3D11View_GetResource(renderTargetView, (ID3D11Resource **)&backBuffer);
  2376. if (!backBuffer) {
  2377. SDL_SetError("%s, ID3D11View::GetResource failed", __FUNCTION__);
  2378. goto done;
  2379. }
  2380. // Create a staging texture to copy the screen's data to:
  2381. ID3D11Texture2D_GetDesc(backBuffer, &stagingTextureDesc);
  2382. stagingTextureDesc.Width = rect->w;
  2383. stagingTextureDesc.Height = rect->h;
  2384. stagingTextureDesc.BindFlags = 0;
  2385. stagingTextureDesc.MiscFlags = 0;
  2386. stagingTextureDesc.CPUAccessFlags = D3D11_CPU_ACCESS_READ;
  2387. stagingTextureDesc.Usage = D3D11_USAGE_STAGING;
  2388. result = ID3D11Device_CreateTexture2D(data->d3dDevice,
  2389. &stagingTextureDesc,
  2390. NULL,
  2391. &stagingTexture);
  2392. if (FAILED(result)) {
  2393. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11Device1::CreateTexture2D [create staging texture]"), result);
  2394. goto done;
  2395. }
  2396. // Copy the desired portion of the back buffer to the staging texture:
  2397. if (!D3D11_GetViewportAlignedD3DRect(renderer, rect, &srcRect, FALSE)) {
  2398. // D3D11_GetViewportAlignedD3DRect will have set the SDL error
  2399. goto done;
  2400. }
  2401. srcBox.left = srcRect.left;
  2402. srcBox.right = srcRect.right;
  2403. srcBox.top = srcRect.top;
  2404. srcBox.bottom = srcRect.bottom;
  2405. srcBox.front = 0;
  2406. srcBox.back = 1;
  2407. ID3D11DeviceContext_CopySubresourceRegion(data->d3dContext,
  2408. (ID3D11Resource *)stagingTexture,
  2409. 0,
  2410. 0, 0, 0,
  2411. (ID3D11Resource *)backBuffer,
  2412. 0,
  2413. &srcBox);
  2414. // Map the staging texture's data to CPU-accessible memory:
  2415. result = ID3D11DeviceContext_Map(data->d3dContext,
  2416. (ID3D11Resource *)stagingTexture,
  2417. 0,
  2418. D3D11_MAP_READ,
  2419. 0,
  2420. &textureMemory);
  2421. if (FAILED(result)) {
  2422. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("ID3D11DeviceContext1::Map [map staging texture]"), result);
  2423. goto done;
  2424. }
  2425. output = SDL_DuplicatePixels(
  2426. rect->w, rect->h,
  2427. D3D11_DXGIFormatToSDLPixelFormat(stagingTextureDesc.Format),
  2428. renderer->target ? renderer->target->colorspace : renderer->output_colorspace,
  2429. textureMemory.pData,
  2430. textureMemory.RowPitch);
  2431. // Unmap the texture:
  2432. ID3D11DeviceContext_Unmap(data->d3dContext,
  2433. (ID3D11Resource *)stagingTexture,
  2434. 0);
  2435. done:
  2436. SAFE_RELEASE(backBuffer);
  2437. SAFE_RELEASE(stagingTexture);
  2438. return output;
  2439. }
  2440. static bool D3D11_RenderPresent(SDL_Renderer *renderer)
  2441. {
  2442. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  2443. HRESULT result;
  2444. DXGI_PRESENT_PARAMETERS parameters;
  2445. if (!data->d3dDevice) {
  2446. return SDL_SetError("Device lost and couldn't be recovered");
  2447. }
  2448. SDL_zero(parameters);
  2449. /* The application may optionally specify "dirty" or "scroll"
  2450. * rects to improve efficiency in certain scenarios.
  2451. */
  2452. result = IDXGISwapChain1_Present1(data->swapChain, data->syncInterval, data->presentFlags, &parameters);
  2453. /* Discard the contents of the render target.
  2454. * This is a valid operation only when the existing contents will be entirely
  2455. * overwritten. If dirty or scroll rects are used, this call should be removed.
  2456. */
  2457. ID3D11DeviceContext1_DiscardView(data->d3dContext, (ID3D11View *)data->mainRenderTargetView);
  2458. // When the present flips, it unbinds the current view, so bind it again on the next draw call
  2459. data->currentRenderTargetView = NULL;
  2460. if (FAILED(result) && result != DXGI_ERROR_WAS_STILL_DRAWING) {
  2461. /* If the device was removed either by a disconnect or a driver upgrade, we
  2462. * must recreate all device resources.
  2463. */
  2464. if (result == DXGI_ERROR_DEVICE_REMOVED) {
  2465. if (D3D11_HandleDeviceLost(renderer)) {
  2466. SDL_SetError("Present failed, device lost");
  2467. } else {
  2468. // Recovering from device lost failed, error is already set
  2469. }
  2470. } else if (result == DXGI_ERROR_INVALID_CALL) {
  2471. // We probably went through a fullscreen <-> windowed transition
  2472. D3D11_CreateWindowSizeDependentResources(renderer);
  2473. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain::Present"), result);
  2474. } else {
  2475. WIN_SetErrorFromHRESULT(SDL_COMPOSE_ERROR("IDXGISwapChain::Present"), result);
  2476. }
  2477. return false;
  2478. }
  2479. return true;
  2480. }
  2481. static bool D3D11_SetVSync(SDL_Renderer *renderer, const int vsync)
  2482. {
  2483. D3D11_RenderData *data = (D3D11_RenderData *)renderer->internal;
  2484. if (vsync < 0) {
  2485. return SDL_Unsupported();
  2486. }
  2487. if (vsync > 0) {
  2488. data->syncInterval = vsync;
  2489. } else {
  2490. data->syncInterval = 0;
  2491. }
  2492. D3D11_UpdatePresentFlags(renderer);
  2493. return true;
  2494. }
  2495. static bool D3D11_CreateRenderer(SDL_Renderer *renderer, SDL_Window *window, SDL_PropertiesID create_props)
  2496. {
  2497. D3D11_RenderData *data;
  2498. HWND hwnd = (HWND)SDL_GetPointerProperty(SDL_GetWindowProperties(window), SDL_PROP_WINDOW_WIN32_HWND_POINTER, NULL);
  2499. if (!hwnd) {
  2500. return SDL_SetError("Couldn't get window handle");
  2501. }
  2502. SDL_SetupRendererColorspace(renderer, create_props);
  2503. if (renderer->output_colorspace != SDL_COLORSPACE_SRGB &&
  2504. renderer->output_colorspace != SDL_COLORSPACE_SRGB_LINEAR
  2505. /*&& renderer->output_colorspace != SDL_COLORSPACE_HDR10*/) {
  2506. return SDL_SetError("Unsupported output colorspace");
  2507. }
  2508. data = (D3D11_RenderData *)SDL_calloc(1, sizeof(*data));
  2509. if (!data) {
  2510. return false;
  2511. }
  2512. data->identity = MatrixIdentity();
  2513. renderer->WindowEvent = D3D11_WindowEvent;
  2514. renderer->SupportsBlendMode = D3D11_SupportsBlendMode;
  2515. renderer->CreateTexture = D3D11_CreateTexture;
  2516. renderer->UpdateTexturePalette = D3D11_UpdateTexturePalette;
  2517. renderer->UpdateTexture = D3D11_UpdateTexture;
  2518. #ifdef SDL_HAVE_YUV
  2519. renderer->UpdateTextureYUV = D3D11_UpdateTextureYUV;
  2520. renderer->UpdateTextureNV = D3D11_UpdateTextureNV;
  2521. #endif
  2522. renderer->LockTexture = D3D11_LockTexture;
  2523. renderer->UnlockTexture = D3D11_UnlockTexture;
  2524. renderer->SetRenderTarget = D3D11_SetRenderTarget;
  2525. renderer->QueueSetViewport = D3D11_QueueNoOp;
  2526. renderer->QueueSetDrawColor = D3D11_QueueNoOp;
  2527. renderer->QueueDrawPoints = D3D11_QueueDrawPoints;
  2528. renderer->QueueDrawLines = D3D11_QueueDrawPoints; // lines and points queue vertices the same way.
  2529. renderer->QueueGeometry = D3D11_QueueGeometry;
  2530. renderer->InvalidateCachedState = D3D11_InvalidateCachedState;
  2531. renderer->RunCommandQueue = D3D11_RunCommandQueue;
  2532. renderer->RenderReadPixels = D3D11_RenderReadPixels;
  2533. renderer->RenderPresent = D3D11_RenderPresent;
  2534. renderer->DestroyTexture = D3D11_DestroyTexture;
  2535. renderer->DestroyRenderer = D3D11_DestroyRenderer;
  2536. renderer->SetVSync = D3D11_SetVSync;
  2537. renderer->internal = data;
  2538. D3D11_InvalidateCachedState(renderer);
  2539. renderer->name = D3D11_RenderDriver.name;
  2540. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ARGB8888);
  2541. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ABGR8888);
  2542. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_XRGB8888);
  2543. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_ABGR2101010);
  2544. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_RGBA64_FLOAT);
  2545. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_INDEX8);
  2546. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_YV12);
  2547. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_IYUV);
  2548. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_NV12);
  2549. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_NV21);
  2550. SDL_AddSupportedTextureFormat(renderer, SDL_PIXELFORMAT_P010);
  2551. data->swapChainFlags = 0;
  2552. data->syncInterval = 0;
  2553. data->presentFlags = DXGI_PRESENT_DO_NOT_WAIT;
  2554. /* HACK: make sure the SDL_Renderer references the SDL_Window data now, in
  2555. * order to give init functions access to the underlying window handle:
  2556. */
  2557. renderer->window = window;
  2558. // Initialize Direct3D resources
  2559. if (FAILED(D3D11_CreateDeviceResources(renderer))) {
  2560. return false;
  2561. }
  2562. if (FAILED(D3D11_CreateWindowSizeDependentResources(renderer))) {
  2563. return false;
  2564. }
  2565. return true;
  2566. }
  2567. SDL_RenderDriver D3D11_RenderDriver = {
  2568. D3D11_CreateRenderer, "direct3d11"
  2569. };
  2570. #endif // SDL_VIDEO_RENDER_D3D11