DdsSubfile.cpp 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. ///
  2. // Created by Иван_Архипов on 24.11.2017.
  3. //
  4. #include "DdsSubfile.h"
  5. #include "../BinaryData.h"
  6. #include "../DatFile.h"
  7. #include "../Common/DatException.h"
  8. #include "../Database.h"
  9. #include "../Common/CommonFunctions.h"
  10. #include <algorithm>
  11. namespace LOTRO_DAT {
  12. DdsSubfile::DdsSubfile() {}
  13. DdsSubfile::DdsSubfile(DatFile *dat, long long fragments_count, long long unknown1,
  14. long long file_id, long long file_offset, long long file_size,
  15. long long timestamp,
  16. long long version, long long block_size)
  17. : Subfile(dat, fragments_count, unknown1, file_id, file_offset, file_size, timestamp, version, block_size) {
  18. BinaryData header(20);
  19. dat_->ReadData(header, 10, file_offset_ + 8);
  20. compressed_ = header.CheckCompression();
  21. }
  22. FILE_TYPE DdsSubfile::FileType() const {
  23. return DDS;
  24. }
  25. std::string DdsSubfile::Extension() const {
  26. return ".dds";
  27. }
  28. bool DdsSubfile::PrepareAsBinary(BinaryData &data) {
  29. if (compressed_)
  30. data = data.DecompressData(4);
  31. BinaryData ddsData(data.size() - 24 + 128);
  32. for (int i = 0; i < 128; i++)
  33. ddsData[i] = 0;
  34. memcpy(ddsData.data() + 128, data.data() + 24, data.size() - 24);
  35. ddsData[0] = 0x44; // D
  36. ddsData[1] = 0x44; // D
  37. ddsData[2] = 0x53; // S
  38. ddsData[3] = 0x20;
  39. ddsData[4] = 0x7C;
  40. ddsData[8] = 7;
  41. ddsData[9] = 0x10;
  42. // width, height
  43. ddsData[12] = data[12];
  44. ddsData[13] = data[13];
  45. ddsData[14] = data[14];
  46. ddsData[15] = data[15];
  47. ddsData[16] = data[8];
  48. ddsData[17] = data[9];
  49. ddsData[18] = data[10];
  50. ddsData[19] = data[11];
  51. long long compression = data.ToNumber<4>(0x10);
  52. switch (compression) {
  53. case 20: // 14 00 00 00 - 888 (R8G8B8)
  54. ddsData[0x4C] = 0x20; // ?
  55. ddsData[0x50] = 0x40; // compressed or not
  56. ddsData[0x58] = 0x18; // bytes per pixel
  57. ddsData[0x5E] = 0xFF;
  58. ddsData[0x61] = 0xFF;
  59. ddsData[0x64] = 0xFF;
  60. break;
  61. case 21: // 15 00 00 00 - 8888 (R8G8B8A8)
  62. ddsData[0x4C] = 0x20; // ?
  63. ddsData[0x50] = 0x40; // compressed or not
  64. ddsData[0x58] = 0x20; // bytes per pixel
  65. ddsData[0x5E] = 0xFF;
  66. ddsData[0x61] = 0xFF;
  67. ddsData[0x64] = 0xFF;
  68. ddsData[0x6B] = 0xFF;
  69. break;
  70. case 28: // 1C 00 00 00 - 332 (?)
  71. ddsData[0x4C] = 0x20; // ?
  72. ddsData[0x50] = 0x40; // compressed or not
  73. ddsData[0x58] = 0x08; // bytes per pixel
  74. ddsData[0x5E] = 0xFF;
  75. ddsData[0x61] = 0xFF;
  76. ddsData[0x64] = 0xFF;
  77. break;
  78. case 827611204: // 44 58 54 31 - DXT1
  79. ddsData[76] = 32;
  80. ddsData[80] = 4;
  81. ddsData[84] = 68;
  82. ddsData[85] = 88;
  83. ddsData[86] = 84;
  84. ddsData[87] = 49;
  85. break;
  86. case 861165636: // 44 58 54 33 - DXT3
  87. ddsData[22] = 1;
  88. ddsData[76] = 32;
  89. ddsData[80] = 4;
  90. ddsData[84] = 68;
  91. ddsData[85] = 88;
  92. ddsData[86] = 84;
  93. ddsData[87] = 51;
  94. ddsData[108] = 8;
  95. ddsData[109] = 16;
  96. ddsData[110] = 64;
  97. break;
  98. case 894720068: // 44 58 54 35 - DXT5
  99. ddsData[10] = 8;
  100. ddsData[22] = 1;
  101. ddsData[28] = 1;
  102. ddsData[76] = 32;
  103. ddsData[80] = 4;
  104. ddsData[84] = 68;
  105. ddsData[85] = 88;
  106. ddsData[86] = 84;
  107. ddsData[87] = 53;
  108. ddsData[88] = 32;
  109. ddsData[94] = 255;
  110. ddsData[97] = 255;
  111. ddsData[100] = 255;
  112. ddsData[107] = 255;
  113. ddsData[109] = 16;
  114. break;
  115. default:
  116. throw DatException("Bad DdsSubfile::PrepareAsBinary() - unknown header format.", EXPORT_EXCEPTION);
  117. }
  118. data = ddsData;
  119. return true;
  120. }
  121. BinaryData DdsSubfile::MakeFromBinary(const BinaryData &data) {
  122. return data.CutData(128).CompressData();
  123. }
  124. };