DDSLoader.cpp 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027
  1. /*
  2. * Copyright (c) 2021, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/Debug.h>
  7. #include <AK/DeprecatedString.h>
  8. #include <AK/Endian.h>
  9. #include <AK/Error.h>
  10. #include <AK/MemoryStream.h>
  11. #include <AK/StringBuilder.h>
  12. #include <AK/Try.h>
  13. #include <AK/Vector.h>
  14. #include <LibGfx/DDSLoader.h>
  15. #include <fcntl.h>
  16. #include <math.h>
  17. #include <stdio.h>
  18. #include <string.h>
  19. #include <sys/mman.h>
  20. #include <sys/stat.h>
  21. #include <unistd.h>
  22. namespace Gfx {
  23. struct DDSLoadingContext {
  24. enum State {
  25. NotDecoded = 0,
  26. Error,
  27. BitmapDecoded,
  28. };
  29. State state { State::NotDecoded };
  30. u8 const* data { nullptr };
  31. size_t data_size { 0 };
  32. DDSHeader header;
  33. DDSHeaderDXT10 header10;
  34. RefPtr<Gfx::Bitmap> bitmap;
  35. void dump_debug();
  36. };
  37. static constexpr u32 create_four_cc(char c0, char c1, char c2, char c3)
  38. {
  39. return c0 | c1 << 8 | c2 << 16 | c3 << 24;
  40. }
  41. static bool is_planar(DXGIFormat format)
  42. {
  43. switch (format) {
  44. case DXGI_FORMAT_NV12:
  45. case DXGI_FORMAT_420_OPAQUE:
  46. case DXGI_FORMAT_P208:
  47. case DXGI_FORMAT_P010:
  48. case DXGI_FORMAT_P016:
  49. return true;
  50. default:
  51. return false;
  52. }
  53. }
  54. static bool is_packed(DXGIFormat format)
  55. {
  56. switch (format) {
  57. case DXGI_FORMAT_R8G8_B8G8_UNORM:
  58. case DXGI_FORMAT_G8R8_G8B8_UNORM:
  59. case DXGI_FORMAT_YUY2:
  60. case DXGI_FORMAT_Y210:
  61. case DXGI_FORMAT_Y216:
  62. return true;
  63. default:
  64. return false;
  65. }
  66. }
  67. static u64 get_width(DDSHeader header, size_t mipmap_level)
  68. {
  69. if (mipmap_level >= header.mip_map_count) {
  70. return header.width;
  71. }
  72. return header.width >> mipmap_level;
  73. }
  74. static u64 get_height(DDSHeader header, size_t mipmap_level)
  75. {
  76. if (mipmap_level >= header.mip_map_count) {
  77. return header.height;
  78. }
  79. return header.height >> mipmap_level;
  80. }
  81. static constexpr bool has_bitmask(DDSPixelFormat format, u32 r, u32 g, u32 b, u32 a)
  82. {
  83. return format.r_bit_mask == r && format.g_bit_mask == g && format.b_bit_mask == b && format.a_bit_mask == a;
  84. }
  85. static DXGIFormat get_format(DDSPixelFormat format)
  86. {
  87. if ((format.flags & PixelFormatFlags::DDPF_RGB) == PixelFormatFlags::DDPF_RGB) {
  88. switch (format.rgb_bit_count) {
  89. case 32: {
  90. if (has_bitmask(format, 0x000000FF, 0x0000FF00, 0x00FF0000, 0xFF000000))
  91. return DXGI_FORMAT_R8G8B8A8_UNORM;
  92. if (has_bitmask(format, 0x00FF0000, 0x0000FF00, 0x000000FF, 0xFF000000))
  93. return DXGI_FORMAT_B8G8R8A8_UNORM;
  94. if (has_bitmask(format, 0x00FF0000, 0x0000FF00, 0x000000FF, 0x00000000))
  95. return DXGI_FORMAT_B8G8R8X8_UNORM;
  96. if (has_bitmask(format, 0x3FF00000, 0x000FFC00, 0x000003FF, 0xC0000000))
  97. return DXGI_FORMAT_R10G10B10A2_UNORM;
  98. if (has_bitmask(format, 0x0000FFFF, 0xFFFF0000, 0x00000000, 0x00000000))
  99. return DXGI_FORMAT_R16G16_UNORM;
  100. if (has_bitmask(format, 0xFFFFFFFF, 0x00000000, 0x00000000, 0x00000000))
  101. return DXGI_FORMAT_R32_FLOAT;
  102. break;
  103. }
  104. case 24:
  105. break;
  106. case 16: {
  107. if (has_bitmask(format, 0x7C00, 0x03E0, 0x001F, 0x8000))
  108. return DXGI_FORMAT_B5G5R5A1_UNORM;
  109. if (has_bitmask(format, 0xF800, 0x07E0, 0x001F, 0x0000))
  110. return DXGI_FORMAT_B5G6R5_UNORM;
  111. if (has_bitmask(format, 0xF800, 0x07E0, 0x001F, 0x0000))
  112. return DXGI_FORMAT_B5G6R5_UNORM;
  113. if (has_bitmask(format, 0x0F00, 0x00F0, 0x000F, 0xF000))
  114. return DXGI_FORMAT_B4G4R4A4_UNORM;
  115. if (has_bitmask(format, 0x00FF, 0x0000, 0x0000, 0xFF00))
  116. return DXGI_FORMAT_R8G8_UNORM;
  117. if (has_bitmask(format, 0xFFFF, 0x0000, 0x0000, 0x0000))
  118. return DXGI_FORMAT_R16_UNORM;
  119. break;
  120. }
  121. case 8: {
  122. if (has_bitmask(format, 0xFF, 0x00, 0x00, 0x00))
  123. return DXGI_FORMAT_R8_UNORM;
  124. break;
  125. }
  126. }
  127. } else if ((format.flags & PixelFormatFlags::DDPF_LUMINANCE) == PixelFormatFlags::DDPF_LUMINANCE) {
  128. switch (format.rgb_bit_count) {
  129. case 16: {
  130. if (has_bitmask(format, 0xFFFF, 0x0000, 0x0000, 0x0000))
  131. return DXGI_FORMAT_R16_UNORM;
  132. if (has_bitmask(format, 0x00FF, 0x0000, 0x0000, 0xFF00))
  133. return DXGI_FORMAT_R8G8_UNORM;
  134. break;
  135. }
  136. case 8: {
  137. if (has_bitmask(format, 0xFF, 0x00, 0x00, 0x00))
  138. return DXGI_FORMAT_R8_UNORM;
  139. // Some writers mistakenly write this as 8 bpp.
  140. if (has_bitmask(format, 0x00FF, 0x0000, 0x0000, 0xFF00))
  141. return DXGI_FORMAT_R8G8_UNORM;
  142. break;
  143. }
  144. }
  145. } else if ((format.flags & PixelFormatFlags::DDPF_ALPHA) == PixelFormatFlags::DDPF_ALPHA) {
  146. if (format.rgb_bit_count == 8)
  147. return DXGI_FORMAT_A8_UNORM;
  148. } else if ((format.flags & PixelFormatFlags::DDPF_BUMPDUDV) == PixelFormatFlags::DDPF_BUMPDUDV) {
  149. switch (format.rgb_bit_count) {
  150. case 32: {
  151. if (has_bitmask(format, 0x000000FF, 0x0000FF00, 0x00FF0000, 0xFF000000))
  152. return DXGI_FORMAT_R8G8B8A8_SNORM;
  153. if (has_bitmask(format, 0x0000FFFF, 0xFFFF0000, 0x00000000, 0x00000000))
  154. return DXGI_FORMAT_R16G16_SNORM;
  155. break;
  156. }
  157. case 16: {
  158. if (has_bitmask(format, 0x00FF, 0xFF00, 0x0000, 0x0000))
  159. return DXGI_FORMAT_R8G8_SNORM;
  160. break;
  161. }
  162. }
  163. } else if ((format.flags & PixelFormatFlags::DDPF_FOURCC) == PixelFormatFlags::DDPF_FOURCC) {
  164. if (format.four_cc == create_four_cc('D', 'X', 'T', '1'))
  165. return DXGI_FORMAT_BC1_UNORM;
  166. if (format.four_cc == create_four_cc('D', 'X', 'T', '2'))
  167. return DXGI_FORMAT_BC2_UNORM;
  168. if (format.four_cc == create_four_cc('D', 'X', 'T', '3'))
  169. return DXGI_FORMAT_BC2_UNORM;
  170. if (format.four_cc == create_four_cc('D', 'X', 'T', '4'))
  171. return DXGI_FORMAT_BC3_UNORM;
  172. if (format.four_cc == create_four_cc('D', 'X', 'T', '5'))
  173. return DXGI_FORMAT_BC3_UNORM;
  174. if (format.four_cc == create_four_cc('A', 'T', 'I', '1'))
  175. return DXGI_FORMAT_BC4_UNORM;
  176. if (format.four_cc == create_four_cc('B', 'C', '4', 'U'))
  177. return DXGI_FORMAT_BC4_UNORM;
  178. if (format.four_cc == create_four_cc('B', 'C', '4', 'S'))
  179. return DXGI_FORMAT_BC4_SNORM;
  180. if (format.four_cc == create_four_cc('A', 'T', 'I', '2'))
  181. return DXGI_FORMAT_BC5_UNORM;
  182. if (format.four_cc == create_four_cc('B', 'C', '5', 'U'))
  183. return DXGI_FORMAT_BC5_UNORM;
  184. if (format.four_cc == create_four_cc('B', 'C', '5', 'S'))
  185. return DXGI_FORMAT_BC5_SNORM;
  186. if (format.four_cc == create_four_cc('R', 'G', 'B', 'G'))
  187. return DXGI_FORMAT_R8G8_B8G8_UNORM;
  188. if (format.four_cc == create_four_cc('G', 'R', 'G', 'B'))
  189. return DXGI_FORMAT_G8R8_G8B8_UNORM;
  190. if (format.four_cc == create_four_cc('Y', 'U', 'Y', '2'))
  191. return DXGI_FORMAT_YUY2;
  192. switch (format.four_cc) {
  193. case 36:
  194. return DXGI_FORMAT_R16G16B16A16_UNORM;
  195. case 110:
  196. return DXGI_FORMAT_R16G16B16A16_SNORM;
  197. case 111:
  198. return DXGI_FORMAT_R16_FLOAT;
  199. case 112:
  200. return DXGI_FORMAT_R16G16_FLOAT;
  201. case 113:
  202. return DXGI_FORMAT_R16G16B16A16_FLOAT;
  203. case 114:
  204. return DXGI_FORMAT_R32_FLOAT;
  205. case 115:
  206. return DXGI_FORMAT_R32G32_FLOAT;
  207. case 116:
  208. return DXGI_FORMAT_R32G32B32A32_FLOAT;
  209. }
  210. }
  211. return DXGI_FORMAT_UNKNOWN;
  212. }
  213. static bool is_block_compressed(DXGIFormat format)
  214. {
  215. switch (format) {
  216. case DXGI_FORMAT_BC1_TYPELESS:
  217. case DXGI_FORMAT_BC1_UNORM:
  218. case DXGI_FORMAT_BC1_UNORM_SRGB:
  219. case DXGI_FORMAT_BC4_TYPELESS:
  220. case DXGI_FORMAT_BC4_UNORM:
  221. case DXGI_FORMAT_BC4_SNORM:
  222. case DXGI_FORMAT_BC2_TYPELESS:
  223. case DXGI_FORMAT_BC2_UNORM:
  224. case DXGI_FORMAT_BC2_UNORM_SRGB:
  225. case DXGI_FORMAT_BC3_TYPELESS:
  226. case DXGI_FORMAT_BC3_UNORM:
  227. case DXGI_FORMAT_BC3_UNORM_SRGB:
  228. case DXGI_FORMAT_BC5_TYPELESS:
  229. case DXGI_FORMAT_BC5_UNORM:
  230. case DXGI_FORMAT_BC5_SNORM:
  231. case DXGI_FORMAT_BC6H_TYPELESS:
  232. case DXGI_FORMAT_BC6H_UF16:
  233. case DXGI_FORMAT_BC6H_SF16:
  234. case DXGI_FORMAT_BC7_TYPELESS:
  235. case DXGI_FORMAT_BC7_UNORM:
  236. case DXGI_FORMAT_BC7_UNORM_SRGB:
  237. return true;
  238. default:
  239. return false;
  240. }
  241. }
  242. static size_t block_size(DXGIFormat format)
  243. {
  244. switch (format) {
  245. case DXGI_FORMAT_BC2_TYPELESS:
  246. case DXGI_FORMAT_BC2_UNORM:
  247. case DXGI_FORMAT_BC2_UNORM_SRGB:
  248. case DXGI_FORMAT_BC3_TYPELESS:
  249. case DXGI_FORMAT_BC3_UNORM:
  250. case DXGI_FORMAT_BC3_UNORM_SRGB:
  251. case DXGI_FORMAT_BC5_TYPELESS:
  252. case DXGI_FORMAT_BC5_UNORM:
  253. case DXGI_FORMAT_BC5_SNORM:
  254. case DXGI_FORMAT_BC6H_TYPELESS:
  255. case DXGI_FORMAT_BC6H_UF16:
  256. case DXGI_FORMAT_BC6H_SF16:
  257. case DXGI_FORMAT_BC7_TYPELESS:
  258. case DXGI_FORMAT_BC7_UNORM:
  259. case DXGI_FORMAT_BC7_UNORM_SRGB:
  260. return 16;
  261. case DXGI_FORMAT_BC1_TYPELESS:
  262. case DXGI_FORMAT_BC1_UNORM:
  263. case DXGI_FORMAT_BC1_UNORM_SRGB:
  264. case DXGI_FORMAT_BC4_TYPELESS:
  265. case DXGI_FORMAT_BC4_UNORM:
  266. case DXGI_FORMAT_BC4_SNORM:
  267. case DXGI_FORMAT_Y210:
  268. case DXGI_FORMAT_Y216:
  269. return 8;
  270. case DXGI_FORMAT_R8G8_B8G8_UNORM:
  271. case DXGI_FORMAT_G8R8_G8B8_UNORM:
  272. case DXGI_FORMAT_YUY2:
  273. case DXGI_FORMAT_P010:
  274. case DXGI_FORMAT_P016:
  275. return 4;
  276. case DXGI_FORMAT_NV12:
  277. case DXGI_FORMAT_420_OPAQUE:
  278. case DXGI_FORMAT_P208:
  279. return 2;
  280. default:
  281. return 0;
  282. }
  283. }
  284. static size_t bits_per_pixel(DXGIFormat format)
  285. {
  286. switch (format) {
  287. case DXGI_FORMAT_R32G32B32A32_TYPELESS:
  288. case DXGI_FORMAT_R32G32B32A32_FLOAT:
  289. case DXGI_FORMAT_R32G32B32A32_UINT:
  290. case DXGI_FORMAT_R32G32B32A32_SINT:
  291. return 128;
  292. case DXGI_FORMAT_R32G32B32_TYPELESS:
  293. case DXGI_FORMAT_R32G32B32_FLOAT:
  294. case DXGI_FORMAT_R32G32B32_UINT:
  295. case DXGI_FORMAT_R32G32B32_SINT:
  296. return 96;
  297. case DXGI_FORMAT_R16G16B16A16_TYPELESS:
  298. case DXGI_FORMAT_R16G16B16A16_FLOAT:
  299. case DXGI_FORMAT_R16G16B16A16_UNORM:
  300. case DXGI_FORMAT_R16G16B16A16_UINT:
  301. case DXGI_FORMAT_R16G16B16A16_SNORM:
  302. case DXGI_FORMAT_R16G16B16A16_SINT:
  303. case DXGI_FORMAT_R32G32_TYPELESS:
  304. case DXGI_FORMAT_R32G32_FLOAT:
  305. case DXGI_FORMAT_R32G32_UINT:
  306. case DXGI_FORMAT_R32G32_SINT:
  307. case DXGI_FORMAT_R32G8X24_TYPELESS:
  308. case DXGI_FORMAT_D32_FLOAT_S8X24_UINT:
  309. case DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS:
  310. case DXGI_FORMAT_X32_TYPELESS_G8X24_UINT:
  311. case DXGI_FORMAT_Y416:
  312. case DXGI_FORMAT_Y210:
  313. case DXGI_FORMAT_Y216:
  314. return 64;
  315. case DXGI_FORMAT_R10G10B10A2_TYPELESS:
  316. case DXGI_FORMAT_R10G10B10A2_UNORM:
  317. case DXGI_FORMAT_R10G10B10A2_UINT:
  318. case DXGI_FORMAT_R11G11B10_FLOAT:
  319. case DXGI_FORMAT_R8G8B8A8_TYPELESS:
  320. case DXGI_FORMAT_R8G8B8A8_UNORM:
  321. case DXGI_FORMAT_R8G8B8A8_UNORM_SRGB:
  322. case DXGI_FORMAT_R8G8B8A8_UINT:
  323. case DXGI_FORMAT_R8G8B8A8_SNORM:
  324. case DXGI_FORMAT_R8G8B8A8_SINT:
  325. case DXGI_FORMAT_R16G16_TYPELESS:
  326. case DXGI_FORMAT_R16G16_FLOAT:
  327. case DXGI_FORMAT_R16G16_UNORM:
  328. case DXGI_FORMAT_R16G16_UINT:
  329. case DXGI_FORMAT_R16G16_SNORM:
  330. case DXGI_FORMAT_R16G16_SINT:
  331. case DXGI_FORMAT_R32_TYPELESS:
  332. case DXGI_FORMAT_D32_FLOAT:
  333. case DXGI_FORMAT_R32_FLOAT:
  334. case DXGI_FORMAT_R32_UINT:
  335. case DXGI_FORMAT_R32_SINT:
  336. case DXGI_FORMAT_R24G8_TYPELESS:
  337. case DXGI_FORMAT_D24_UNORM_S8_UINT:
  338. case DXGI_FORMAT_R24_UNORM_X8_TYPELESS:
  339. case DXGI_FORMAT_X24_TYPELESS_G8_UINT:
  340. case DXGI_FORMAT_R9G9B9E5_SHAREDEXP:
  341. case DXGI_FORMAT_R8G8_B8G8_UNORM:
  342. case DXGI_FORMAT_G8R8_G8B8_UNORM:
  343. case DXGI_FORMAT_B8G8R8A8_UNORM:
  344. case DXGI_FORMAT_B8G8R8X8_UNORM:
  345. case DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM:
  346. case DXGI_FORMAT_B8G8R8A8_TYPELESS:
  347. case DXGI_FORMAT_B8G8R8A8_UNORM_SRGB:
  348. case DXGI_FORMAT_B8G8R8X8_TYPELESS:
  349. case DXGI_FORMAT_B8G8R8X8_UNORM_SRGB:
  350. case DXGI_FORMAT_AYUV:
  351. case DXGI_FORMAT_Y410:
  352. case DXGI_FORMAT_YUY2:
  353. return 32;
  354. case DXGI_FORMAT_P010:
  355. case DXGI_FORMAT_P016:
  356. case DXGI_FORMAT_V408:
  357. return 24;
  358. case DXGI_FORMAT_R8G8_TYPELESS:
  359. case DXGI_FORMAT_R8G8_UNORM:
  360. case DXGI_FORMAT_R8G8_UINT:
  361. case DXGI_FORMAT_R8G8_SNORM:
  362. case DXGI_FORMAT_R8G8_SINT:
  363. case DXGI_FORMAT_R16_TYPELESS:
  364. case DXGI_FORMAT_R16_FLOAT:
  365. case DXGI_FORMAT_D16_UNORM:
  366. case DXGI_FORMAT_R16_UNORM:
  367. case DXGI_FORMAT_R16_UINT:
  368. case DXGI_FORMAT_R16_SNORM:
  369. case DXGI_FORMAT_R16_SINT:
  370. case DXGI_FORMAT_B5G6R5_UNORM:
  371. case DXGI_FORMAT_B5G5R5A1_UNORM:
  372. case DXGI_FORMAT_A8P8:
  373. case DXGI_FORMAT_B4G4R4A4_UNORM:
  374. case DXGI_FORMAT_P208:
  375. case DXGI_FORMAT_V208:
  376. return 16;
  377. case DXGI_FORMAT_NV12:
  378. case DXGI_FORMAT_420_OPAQUE:
  379. case DXGI_FORMAT_NV11:
  380. return 12;
  381. case DXGI_FORMAT_R8_TYPELESS:
  382. case DXGI_FORMAT_R8_UNORM:
  383. case DXGI_FORMAT_R8_UINT:
  384. case DXGI_FORMAT_R8_SNORM:
  385. case DXGI_FORMAT_R8_SINT:
  386. case DXGI_FORMAT_A8_UNORM:
  387. case DXGI_FORMAT_BC2_TYPELESS:
  388. case DXGI_FORMAT_BC2_UNORM:
  389. case DXGI_FORMAT_BC2_UNORM_SRGB:
  390. case DXGI_FORMAT_BC3_TYPELESS:
  391. case DXGI_FORMAT_BC3_UNORM:
  392. case DXGI_FORMAT_BC3_UNORM_SRGB:
  393. case DXGI_FORMAT_BC5_TYPELESS:
  394. case DXGI_FORMAT_BC5_UNORM:
  395. case DXGI_FORMAT_BC5_SNORM:
  396. case DXGI_FORMAT_BC6H_TYPELESS:
  397. case DXGI_FORMAT_BC6H_UF16:
  398. case DXGI_FORMAT_BC6H_SF16:
  399. case DXGI_FORMAT_BC7_TYPELESS:
  400. case DXGI_FORMAT_BC7_UNORM:
  401. case DXGI_FORMAT_BC7_UNORM_SRGB:
  402. case DXGI_FORMAT_AI44:
  403. case DXGI_FORMAT_IA44:
  404. case DXGI_FORMAT_P8:
  405. return 8;
  406. case DXGI_FORMAT_R1_UNORM:
  407. return 1;
  408. case DXGI_FORMAT_BC1_TYPELESS:
  409. case DXGI_FORMAT_BC1_UNORM:
  410. case DXGI_FORMAT_BC1_UNORM_SRGB:
  411. case DXGI_FORMAT_BC4_TYPELESS:
  412. case DXGI_FORMAT_BC4_UNORM:
  413. case DXGI_FORMAT_BC4_SNORM:
  414. return 4;
  415. default:
  416. return 0;
  417. }
  418. }
  419. static void decode_dx5_alpha_block(InputMemoryStream& stream, DDSLoadingContext& context, u64 bitmap_x, u64 bitmap_y)
  420. {
  421. LittleEndian<u8> color0 {}, color1 {};
  422. LittleEndian<u8> code0 {}, code1 {}, code2 {}, code3 {}, code4 {}, code5 {};
  423. stream >> color0;
  424. stream >> color1;
  425. stream >> code0;
  426. stream >> code1;
  427. stream >> code2;
  428. stream >> code3;
  429. stream >> code4;
  430. stream >> code5;
  431. u32 codes[6] = { 0 };
  432. codes[0] = code0 + 256 * (code1 + 256);
  433. codes[1] = code1 + 256 * (code2 + 256);
  434. codes[2] = code2 + 256 * (code3 + 256);
  435. codes[3] = code3 + 256 * (code4 + 256);
  436. codes[4] = code4 + 256 * code5;
  437. codes[5] = code5;
  438. u32 color[8] = { 0 };
  439. if (color0 > 128) {
  440. color[0] = color0;
  441. }
  442. if (color1 > 128) {
  443. color[1] = color1;
  444. }
  445. if (color0 > color1) {
  446. color[2] = (6 * color[0] + 1 * color[1]) / 7;
  447. color[3] = (5 * color[0] + 2 * color[1]) / 7;
  448. color[4] = (4 * color[0] + 3 * color[1]) / 7;
  449. color[5] = (3 * color[0] + 4 * color[1]) / 7;
  450. color[6] = (2 * color[0] + 5 * color[1]) / 7;
  451. color[7] = (1 * color[0] + 6 * color[1]) / 7;
  452. } else {
  453. color[2] = (4 * color[0] + 1 * color[1]) / 5;
  454. color[3] = (3 * color[0] + 2 * color[1]) / 5;
  455. color[4] = (2 * color[0] + 3 * color[1]) / 5;
  456. color[5] = (1 * color[0] + 4 * color[1]) / 5;
  457. color[6] = 0;
  458. color[7] = 255;
  459. }
  460. for (size_t y = 0; y < 4; y++) {
  461. for (size_t x = 0; x < 4; x++) {
  462. u8 index = 3 * (4 * y + x);
  463. u8 bit_location = floor(index / 8.0);
  464. u8 adjusted_index = index - (bit_location * 8);
  465. u8 code = (codes[bit_location] >> adjusted_index) & 7;
  466. u8 alpha = color[code];
  467. Color color = Color(0, 0, 0, alpha);
  468. context.bitmap->set_pixel(bitmap_x + x, bitmap_y + y, color);
  469. }
  470. }
  471. }
  472. static void decode_dx3_alpha_block(InputMemoryStream& stream, DDSLoadingContext& context, u64 bitmap_x, u64 bitmap_y)
  473. {
  474. LittleEndian<u8> a0 {}, a1 {}, a2 {}, a3 {}, a4 {}, a5 {}, a6 {}, a7 {};
  475. stream >> a0;
  476. stream >> a1;
  477. stream >> a2;
  478. stream >> a3;
  479. stream >> a4;
  480. stream >> a5;
  481. stream >> a6;
  482. stream >> a7;
  483. u64 alpha_0 = a0 + 256u * (a1 + 256u * (a2 + 256u * (a3 + 256u)));
  484. u64 alpha_1 = a4 + 256u * (a5 + 256u * (a6 + 256u * a7));
  485. for (size_t y = 0; y < 4; y++) {
  486. for (size_t x = 0; x < 4; x++) {
  487. u8 code = 4 * (4 * y + x);
  488. if (code >= 32) {
  489. code = code - 32;
  490. u8 alpha = ((alpha_1 >> code) & 0x0F) * 17;
  491. Color color = Color(0, 0, 0, alpha);
  492. context.bitmap->set_pixel(bitmap_x + x, bitmap_y + y, color);
  493. } else {
  494. u8 alpha = ((alpha_0 >> code) & 0x0F) * 17;
  495. Color color = Color(0, 0, 0, alpha);
  496. context.bitmap->set_pixel(bitmap_x + x, bitmap_y + y, color);
  497. }
  498. }
  499. }
  500. }
  501. static void unpack_rbg_565(u32 rgb, u8* output)
  502. {
  503. u8 r = (rgb >> 11) & 0x1F;
  504. u8 g = (rgb >> 5) & 0x3F;
  505. u8 b = rgb & 0x1F;
  506. output[0] = (r << 3) | (r >> 2);
  507. output[1] = (g << 2) | (g >> 4);
  508. output[2] = (b << 3) | (b >> 2);
  509. output[3] = 255;
  510. }
  511. static void decode_color_block(InputMemoryStream& stream, DDSLoadingContext& context, bool dxt1, u64 bitmap_x, u64 bitmap_y)
  512. {
  513. LittleEndian<u8> c0_low {}, c0_high {}, c1_low {}, c1_high {};
  514. LittleEndian<u8> codes_0 {}, codes_1 {}, codes_2 {}, codes_3 {};
  515. stream >> c0_low;
  516. stream >> c0_high;
  517. stream >> c1_low;
  518. stream >> c1_high;
  519. stream >> codes_0;
  520. stream >> codes_1;
  521. stream >> codes_2;
  522. stream >> codes_3;
  523. u64 code = codes_0 + 256 * (codes_1 + 256 * (codes_2 + 256 * codes_3));
  524. u32 color_0 = c0_low + (c0_high * 256);
  525. u32 color_1 = c1_low + (c1_high * 256);
  526. u8 rgba[4][4];
  527. unpack_rbg_565(color_0, rgba[0]);
  528. unpack_rbg_565(color_1, rgba[1]);
  529. if (color_0 > color_1) {
  530. for (size_t i = 0; i < 3; i++) {
  531. rgba[2][i] = (2 * rgba[0][i] + rgba[1][i]) / 3;
  532. rgba[3][i] = (rgba[0][i] + 2 * rgba[1][i]) / 3;
  533. }
  534. rgba[2][3] = 255;
  535. rgba[3][3] = 255;
  536. } else {
  537. for (size_t i = 0; i < 3; i++) {
  538. rgba[2][i] = (rgba[0][i] + rgba[1][i]) / 2;
  539. rgba[3][i] = 0;
  540. }
  541. rgba[2][3] = 255;
  542. rgba[3][3] = dxt1 ? 0 : 255;
  543. }
  544. size_t i = 0;
  545. for (size_t y = 0; y < 4; y++) {
  546. for (size_t x = 0; x < 4; x++) {
  547. u8 code_byte = (code >> (i * 2)) & 3;
  548. u8 r = rgba[code_byte][0];
  549. u8 g = rgba[code_byte][1];
  550. u8 b = rgba[code_byte][2];
  551. u8 a = dxt1 ? rgba[code_byte][3] : context.bitmap->get_pixel(bitmap_x + x, bitmap_y + y).alpha();
  552. Color color = Color(r, g, b, a);
  553. context.bitmap->set_pixel(bitmap_x + x, bitmap_y + y, color);
  554. i++;
  555. }
  556. }
  557. }
  558. static void decode_dxt(InputMemoryStream& stream, DDSLoadingContext& context, DXGIFormat format, u64 width, u64 y)
  559. {
  560. if (format == DXGI_FORMAT_BC1_UNORM) {
  561. for (size_t x = 0; x < width; x += 4) {
  562. decode_color_block(stream, context, true, x, y);
  563. }
  564. }
  565. if (format == DXGI_FORMAT_BC2_UNORM) {
  566. for (size_t x = 0; x < width; x += 4) {
  567. decode_dx3_alpha_block(stream, context, x, y);
  568. decode_color_block(stream, context, false, x, y);
  569. }
  570. }
  571. if (format == DXGI_FORMAT_BC3_UNORM) {
  572. for (size_t x = 0; x < width; x += 4) {
  573. decode_dx5_alpha_block(stream, context, x, y);
  574. decode_color_block(stream, context, false, x, y);
  575. }
  576. }
  577. }
  578. static void decode_bitmap(InputMemoryStream& stream, DDSLoadingContext& context, DXGIFormat format, u64 width, u64 height)
  579. {
  580. Vector<u32> dxt_formats = { DXGI_FORMAT_BC1_UNORM, DXGI_FORMAT_BC2_UNORM, DXGI_FORMAT_BC3_UNORM };
  581. if (dxt_formats.contains_slow(format)) {
  582. for (u64 y = 0; y < height; y += 4) {
  583. decode_dxt(stream, context, format, width, y);
  584. }
  585. }
  586. // FIXME: Support more encodings (ATI, YUV, RAW, etc...).
  587. }
  588. static size_t get_minimum_bytes_for_mipmap(DXGIFormat format, u64 width, u64 height)
  589. {
  590. u64 row_bytes {};
  591. u64 row_count {};
  592. if (is_block_compressed(format)) {
  593. u64 width_in_blocks {};
  594. u64 height_in_blocks {};
  595. if (width > 0) {
  596. width_in_blocks = max(static_cast<u64>(1), (width + 3u) / 4u);
  597. }
  598. if (height > 0) {
  599. height_in_blocks = max(static_cast<u64>(1), (height + 3u) / 4u);
  600. }
  601. row_bytes = width_in_blocks * block_size(format);
  602. row_count = height_in_blocks;
  603. return row_bytes * row_count;
  604. } else if (is_packed(format)) {
  605. row_bytes = ((width + 1u) >> 1) * block_size(format);
  606. row_count = height;
  607. return row_bytes * row_count;
  608. } else if (format == DXGI_FORMAT_NV11) {
  609. row_bytes = ((width + 3u) >> 2) * 4u;
  610. row_count = height * 2u;
  611. return row_bytes * row_count;
  612. } else if (is_planar(format)) {
  613. row_bytes = ((width + 1u) >> 1) * block_size(format);
  614. row_count = height + ((height + 1u) >> 1);
  615. return (row_bytes * row_count) + (((row_bytes * row_count) + 1) >> 1);
  616. } else {
  617. u32 bpp = bits_per_pixel(format);
  618. row_bytes = (width * bpp + 7u) / 8u;
  619. row_count = height;
  620. return row_bytes * row_count;
  621. }
  622. }
  623. static ErrorOr<void> decode_dds(DDSLoadingContext& context)
  624. {
  625. InputMemoryStream stream({ context.data, context.data_size });
  626. // All valid DDS files are at least 128 bytes long.
  627. if (stream.remaining() < 128) {
  628. dbgln_if(DDS_DEBUG, "File is too short for DDS");
  629. context.state = DDSLoadingContext::State::Error;
  630. return Error::from_string_literal("File is too short for DDS");
  631. }
  632. u32 magic;
  633. stream >> magic;
  634. if (magic != create_four_cc('D', 'D', 'S', ' ')) {
  635. dbgln_if(DDS_DEBUG, "Missing magic number");
  636. context.state = DDSLoadingContext::State::Error;
  637. return Error::from_string_literal("Missing magic number");
  638. }
  639. stream >> context.header.size;
  640. stream >> context.header.flags;
  641. stream >> context.header.height;
  642. stream >> context.header.width;
  643. stream >> context.header.pitch;
  644. stream >> context.header.depth;
  645. stream >> context.header.mip_map_count;
  646. // The bytes in context.header.reserved are unused, so we just skip over them (11 * 4 bytes).
  647. stream.discard_or_error(44);
  648. stream >> context.header.pixel_format.size;
  649. stream >> context.header.pixel_format.flags;
  650. stream >> context.header.pixel_format.four_cc;
  651. stream >> context.header.pixel_format.rgb_bit_count;
  652. stream >> context.header.pixel_format.r_bit_mask;
  653. stream >> context.header.pixel_format.g_bit_mask;
  654. stream >> context.header.pixel_format.b_bit_mask;
  655. stream >> context.header.pixel_format.a_bit_mask;
  656. stream >> context.header.caps1;
  657. stream >> context.header.caps2;
  658. stream >> context.header.caps3;
  659. stream >> context.header.caps4;
  660. stream >> context.header.reserved2;
  661. if (context.header.size != 124) {
  662. dbgln_if(DDS_DEBUG, "Header size is malformed");
  663. context.state = DDSLoadingContext::State::Error;
  664. return Error::from_string_literal("Header size is malformed");
  665. }
  666. if (context.header.pixel_format.size != 32) {
  667. dbgln_if(DDS_DEBUG, "Pixel format size is malformed");
  668. context.state = DDSLoadingContext::State::Error;
  669. return Error::from_string_literal("Pixel format size is malformed");
  670. }
  671. if ((context.header.pixel_format.flags & PixelFormatFlags::DDPF_FOURCC) == PixelFormatFlags::DDPF_FOURCC) {
  672. if (context.header.pixel_format.four_cc == create_four_cc('D', 'X', '1', '0')) {
  673. if (stream.bytes().size() < 148) {
  674. dbgln_if(DDS_DEBUG, "DX10 header is too short");
  675. context.state = DDSLoadingContext::State::Error;
  676. return Error::from_string_literal("DX10 header is too short");
  677. }
  678. u32 format {};
  679. stream >> format;
  680. context.header10.format = static_cast<DXGIFormat>(format);
  681. stream >> context.header10.resource_dimension;
  682. stream >> context.header10.misc_flag;
  683. stream >> context.header10.array_size;
  684. stream >> context.header10.misc_flag2;
  685. }
  686. }
  687. if constexpr (DDS_DEBUG) {
  688. context.dump_debug();
  689. }
  690. DXGIFormat format = get_format(context.header.pixel_format);
  691. Vector<u32> supported_formats = { DXGI_FORMAT_BC1_UNORM, DXGI_FORMAT_BC2_UNORM, DXGI_FORMAT_BC3_UNORM };
  692. if (!supported_formats.contains_slow(format)) {
  693. dbgln_if(DDS_DEBUG, "Format of type {} is not supported at the moment", static_cast<u32>(format));
  694. context.state = DDSLoadingContext::State::Error;
  695. return Error::from_string_literal("Format type is not supported at the moment");
  696. }
  697. // We support parsing mipmaps, but we only care about the largest one :^) (At least for now)
  698. if (size_t mipmap_level = 0; mipmap_level < max(context.header.mip_map_count, 1u)) {
  699. u64 width = get_width(context.header, mipmap_level);
  700. u64 height = get_height(context.header, mipmap_level);
  701. u64 needed_bytes = get_minimum_bytes_for_mipmap(format, width, height);
  702. dbgln_if(DDS_DEBUG, "There are {} bytes remaining, we need {} for mipmap level {} of the image", stream.remaining(), needed_bytes, mipmap_level);
  703. VERIFY(stream.remaining() >= needed_bytes);
  704. context.bitmap = TRY(Bitmap::try_create(BitmapFormat::BGRA8888, { width, height }));
  705. decode_bitmap(stream, context, format, width, height);
  706. }
  707. context.state = DDSLoadingContext::State::BitmapDecoded;
  708. return {};
  709. }
  710. void DDSLoadingContext::dump_debug()
  711. {
  712. StringBuilder builder;
  713. builder.append("\nDDS:\n"sv);
  714. builder.appendff("\tHeader Size: {}\n", header.size);
  715. builder.append("\tFlags:"sv);
  716. if ((header.flags & DDSFlags::DDSD_CAPS) == DDSFlags::DDSD_CAPS)
  717. builder.append(" DDSD_CAPS"sv);
  718. if ((header.flags & DDSFlags::DDSD_HEIGHT) == DDSFlags::DDSD_HEIGHT)
  719. builder.append(" DDSD_HEIGHT"sv);
  720. if ((header.flags & DDSFlags::DDSD_WIDTH) == DDSFlags::DDSD_WIDTH)
  721. builder.append(" DDSD_WIDTH"sv);
  722. if ((header.flags & DDSFlags::DDSD_PITCH) == DDSFlags::DDSD_PITCH)
  723. builder.append(" DDSD_PITCH"sv);
  724. if ((header.flags & DDSFlags::DDSD_PIXELFORMAT) == DDSFlags::DDSD_PIXELFORMAT)
  725. builder.append(" DDSD_PIXELFORMAT"sv);
  726. if ((header.flags & DDSFlags::DDSD_MIPMAPCOUNT) == DDSFlags::DDSD_MIPMAPCOUNT)
  727. builder.append(" DDSD_MIPMAPCOUNT"sv);
  728. if ((header.flags & DDSFlags::DDSD_LINEARSIZE) == DDSFlags::DDSD_LINEARSIZE)
  729. builder.append(" DDSD_LINEARSIZE"sv);
  730. if ((header.flags & DDSFlags::DDSD_DEPTH) == DDSFlags::DDSD_DEPTH)
  731. builder.append(" DDSD_DEPTH"sv);
  732. builder.append("\n"sv);
  733. builder.appendff("\tHeight: {}\n", header.height);
  734. builder.appendff("\tWidth: {}\n", header.width);
  735. builder.appendff("\tPitch: {}\n", header.pitch);
  736. builder.appendff("\tDepth: {}\n", header.depth);
  737. builder.appendff("\tMipmap Count: {}\n", header.mip_map_count);
  738. builder.append("\tCaps:"sv);
  739. if ((header.caps1 & Caps1Flags::DDSCAPS_COMPLEX) == Caps1Flags::DDSCAPS_COMPLEX)
  740. builder.append(" DDSCAPS_COMPLEX"sv);
  741. if ((header.caps1 & Caps1Flags::DDSCAPS_MIPMAP) == Caps1Flags::DDSCAPS_MIPMAP)
  742. builder.append(" DDSCAPS_MIPMAP"sv);
  743. if ((header.caps1 & Caps1Flags::DDSCAPS_TEXTURE) == Caps1Flags::DDSCAPS_TEXTURE)
  744. builder.append(" DDSCAPS_TEXTURE"sv);
  745. builder.append("\n"sv);
  746. builder.append("\tCaps2:"sv);
  747. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP) == Caps2Flags::DDSCAPS2_CUBEMAP)
  748. builder.append(" DDSCAPS2_CUBEMAP"sv);
  749. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEX) == Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEX)
  750. builder.append(" DDSCAPS2_CUBEMAP_POSITIVEX"sv);
  751. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEX) == Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEX)
  752. builder.append(" DDSCAPS2_CUBEMAP_NEGATIVEX"sv);
  753. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEY) == Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEY)
  754. builder.append(" DDSCAPS2_CUBEMAP_POSITIVEY"sv);
  755. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEY) == Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEY)
  756. builder.append(" DDSCAPS2_CUBEMAP_NEGATIVEY"sv);
  757. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEZ) == Caps2Flags::DDSCAPS2_CUBEMAP_POSITIVEZ)
  758. builder.append(" DDSCAPS2_CUBEMAP_POSITIVEZ"sv);
  759. if ((header.caps2 & Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEZ) == Caps2Flags::DDSCAPS2_CUBEMAP_NEGATIVEZ)
  760. builder.append(" DDSCAPS2_CUBEMAP_NEGATIVEZ"sv);
  761. if ((header.caps2 & Caps2Flags::DDSCAPS2_VOLUME) == Caps2Flags::DDSCAPS2_VOLUME)
  762. builder.append(" DDSCAPS2_VOLUME"sv);
  763. builder.append("\n"sv);
  764. builder.append("Pixel Format:\n"sv);
  765. builder.appendff("\tStruct Size: {}\n", header.pixel_format.size);
  766. builder.append("\tFlags:"sv);
  767. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_ALPHAPIXELS) == PixelFormatFlags::DDPF_ALPHAPIXELS)
  768. builder.append(" DDPF_ALPHAPIXELS"sv);
  769. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_ALPHA) == PixelFormatFlags::DDPF_ALPHA)
  770. builder.append(" DDPF_ALPHA"sv);
  771. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_FOURCC) == PixelFormatFlags::DDPF_FOURCC)
  772. builder.append(" DDPF_FOURCC"sv);
  773. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_PALETTEINDEXED8) == PixelFormatFlags::DDPF_PALETTEINDEXED8)
  774. builder.append(" DDPF_PALETTEINDEXED8"sv);
  775. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_RGB) == PixelFormatFlags::DDPF_RGB)
  776. builder.append(" DDPF_RGB"sv);
  777. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_YUV) == PixelFormatFlags::DDPF_YUV)
  778. builder.append(" DDPF_YUV"sv);
  779. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_LUMINANCE) == PixelFormatFlags::DDPF_LUMINANCE)
  780. builder.append(" DDPF_LUMINANCE"sv);
  781. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_BUMPDUDV) == PixelFormatFlags::DDPF_BUMPDUDV)
  782. builder.append(" DDPF_BUMPDUDV"sv);
  783. if ((header.pixel_format.flags & PixelFormatFlags::DDPF_NORMAL) == PixelFormatFlags::DDPF_NORMAL)
  784. builder.append(" DDPF_NORMAL"sv);
  785. builder.append("\n"sv);
  786. builder.append("\tFour CC: "sv);
  787. builder.appendff("{:c}", (header.pixel_format.four_cc >> (8 * 0)) & 0xFF);
  788. builder.appendff("{:c}", (header.pixel_format.four_cc >> (8 * 1)) & 0xFF);
  789. builder.appendff("{:c}", (header.pixel_format.four_cc >> (8 * 2)) & 0xFF);
  790. builder.appendff("{:c}", (header.pixel_format.four_cc >> (8 * 3)) & 0xFF);
  791. builder.append("\n"sv);
  792. builder.appendff("\tRGB Bit Count: {}\n", header.pixel_format.rgb_bit_count);
  793. builder.appendff("\tR Bit Mask: {}\n", header.pixel_format.r_bit_mask);
  794. builder.appendff("\tG Bit Mask: {}\n", header.pixel_format.g_bit_mask);
  795. builder.appendff("\tB Bit Mask: {}\n", header.pixel_format.b_bit_mask);
  796. builder.appendff("\tA Bit Mask: {}\n", header.pixel_format.a_bit_mask);
  797. builder.append("DDS10:\n"sv);
  798. builder.appendff("\tFormat: {}\n", static_cast<u32>(header10.format));
  799. builder.append("\tResource Dimension:"sv);
  800. if ((header10.resource_dimension & ResourceDimensions::DDS_DIMENSION_UNKNOWN) == ResourceDimensions::DDS_DIMENSION_UNKNOWN)
  801. builder.append(" DDS_DIMENSION_UNKNOWN"sv);
  802. if ((header10.resource_dimension & ResourceDimensions::DDS_DIMENSION_BUFFER) == ResourceDimensions::DDS_DIMENSION_BUFFER)
  803. builder.append(" DDS_DIMENSION_BUFFER"sv);
  804. if ((header10.resource_dimension & ResourceDimensions::DDS_DIMENSION_TEXTURE1D) == ResourceDimensions::DDS_DIMENSION_TEXTURE1D)
  805. builder.append(" DDS_DIMENSION_TEXTURE1D"sv);
  806. if ((header10.resource_dimension & ResourceDimensions::DDS_DIMENSION_TEXTURE2D) == ResourceDimensions::DDS_DIMENSION_TEXTURE2D)
  807. builder.append(" DDS_DIMENSION_TEXTURE2D"sv);
  808. if ((header10.resource_dimension & ResourceDimensions::DDS_DIMENSION_TEXTURE3D) == ResourceDimensions::DDS_DIMENSION_TEXTURE3D)
  809. builder.append(" DDS_DIMENSION_TEXTURE3D"sv);
  810. builder.append("\n"sv);
  811. builder.appendff("\tArray Size: {}\n", header10.array_size);
  812. builder.append("\tMisc Flags:"sv);
  813. if ((header10.misc_flag & MiscFlags::DDS_RESOURCE_MISC_TEXTURECUBE) == MiscFlags::DDS_RESOURCE_MISC_TEXTURECUBE)
  814. builder.append(" DDS_RESOURCE_MISC_TEXTURECUBE"sv);
  815. builder.append("\n"sv);
  816. builder.append("\tMisc Flags 2:"sv);
  817. if ((header10.misc_flag2 & Misc2Flags::DDS_ALPHA_MODE_UNKNOWN) == Misc2Flags::DDS_ALPHA_MODE_UNKNOWN)
  818. builder.append(" DDS_ALPHA_MODE_UNKNOWN"sv);
  819. if ((header10.misc_flag2 & Misc2Flags::DDS_ALPHA_MODE_STRAIGHT) == Misc2Flags::DDS_ALPHA_MODE_STRAIGHT)
  820. builder.append(" DDS_ALPHA_MODE_STRAIGHT"sv);
  821. if ((header10.misc_flag2 & Misc2Flags::DDS_ALPHA_MODE_PREMULTIPLIED) == Misc2Flags::DDS_ALPHA_MODE_PREMULTIPLIED)
  822. builder.append(" DDS_ALPHA_MODE_PREMULTIPLIED"sv);
  823. if ((header10.misc_flag2 & Misc2Flags::DDS_ALPHA_MODE_OPAQUE) == Misc2Flags::DDS_ALPHA_MODE_OPAQUE)
  824. builder.append(" DDS_ALPHA_MODE_OPAQUE"sv);
  825. if ((header10.misc_flag2 & Misc2Flags::DDS_ALPHA_MODE_CUSTOM) == Misc2Flags::DDS_ALPHA_MODE_CUSTOM)
  826. builder.append(" DDS_ALPHA_MODE_CUSTOM"sv);
  827. builder.append("\n"sv);
  828. dbgln("{}", builder.to_deprecated_string());
  829. }
  830. DDSImageDecoderPlugin::DDSImageDecoderPlugin(u8 const* data, size_t size)
  831. {
  832. m_context = make<DDSLoadingContext>();
  833. m_context->data = data;
  834. m_context->data_size = size;
  835. }
  836. DDSImageDecoderPlugin::~DDSImageDecoderPlugin() = default;
  837. IntSize DDSImageDecoderPlugin::size()
  838. {
  839. if (m_context->state == DDSLoadingContext::State::Error)
  840. return {};
  841. if (m_context->state == DDSLoadingContext::State::BitmapDecoded)
  842. return { m_context->header.width, m_context->header.height };
  843. return {};
  844. }
  845. void DDSImageDecoderPlugin::set_volatile()
  846. {
  847. if (m_context->bitmap)
  848. m_context->bitmap->set_volatile();
  849. }
  850. bool DDSImageDecoderPlugin::set_nonvolatile(bool& was_purged)
  851. {
  852. if (!m_context->bitmap)
  853. return false;
  854. return m_context->bitmap->set_nonvolatile(was_purged);
  855. }
  856. bool DDSImageDecoderPlugin::initialize()
  857. {
  858. // The header is always at least 128 bytes, so if the file is smaller, it can't be a DDS.
  859. return m_context->data_size > 128
  860. && m_context->data[0] == 0x44
  861. && m_context->data[1] == 0x44
  862. && m_context->data[2] == 0x53
  863. && m_context->data[3] == 0x20;
  864. }
  865. ErrorOr<bool> DDSImageDecoderPlugin::sniff(ReadonlyBytes data)
  866. {
  867. // The header is always at least 128 bytes, so if the file is smaller, it can't be a DDS.
  868. return data.size() > 128
  869. && data.data()[0] == 0x44
  870. && data.data()[1] == 0x44
  871. && data.data()[2] == 0x53
  872. && data.data()[3] == 0x20;
  873. }
  874. ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> DDSImageDecoderPlugin::create(ReadonlyBytes data)
  875. {
  876. return adopt_nonnull_own_or_enomem(new (nothrow) DDSImageDecoderPlugin(data.data(), data.size()));
  877. }
  878. bool DDSImageDecoderPlugin::is_animated()
  879. {
  880. return false;
  881. }
  882. size_t DDSImageDecoderPlugin::loop_count()
  883. {
  884. return 0;
  885. }
  886. size_t DDSImageDecoderPlugin::frame_count()
  887. {
  888. return 1;
  889. }
  890. ErrorOr<ImageFrameDescriptor> DDSImageDecoderPlugin::frame(size_t index)
  891. {
  892. if (index > 0)
  893. return Error::from_string_literal("DDSImageDecoderPlugin: Invalid frame index");
  894. if (m_context->state == DDSLoadingContext::State::Error)
  895. return Error::from_string_literal("DDSImageDecoderPlugin: Decoding failed");
  896. if (m_context->state < DDSLoadingContext::State::BitmapDecoded) {
  897. TRY(decode_dds(*m_context));
  898. }
  899. VERIFY(m_context->bitmap);
  900. return ImageFrameDescriptor { m_context->bitmap, 0 };
  901. }
  902. }