MDLLoader.h 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. /*
  2. Open Asset Import Library (assimp)
  3. ----------------------------------------------------------------------
  4. Copyright (c) 2006-2025, assimp team
  5. All rights reserved.
  6. Redistribution and use of this software in source and binary forms,
  7. with or without modification, are permitted provided that the
  8. following conditions are met:
  9. * Redistributions of source code must retain the above
  10. copyright notice, this list of conditions and the
  11. following disclaimer.
  12. * Redistributions in binary form must reproduce the above
  13. copyright notice, this list of conditions and the
  14. following disclaimer in the documentation and/or other
  15. materials provided with the distribution.
  16. * Neither the name of the assimp team, nor the names of its
  17. contributors may be used to endorse or promote products
  18. derived from this software without specific prior
  19. written permission of the assimp team.
  20. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  21. "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  22. LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  23. A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  24. OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  25. SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  26. LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  27. DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  28. THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  29. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  30. OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  31. ----------------------------------------------------------------------
  32. */
  33. /// @file MDLLoader.h
  34. /// @brief Declaration of the loader for MDL files
  35. #pragma once
  36. #ifndef AI_MDLLOADER_H_INCLUDED
  37. #define AI_MDLLOADER_H_INCLUDED
  38. #include <assimp/BaseImporter.h>
  39. #include "MDLFileData.h"
  40. #include "AssetLib/HMP/HalfLifeFileData.h"
  41. #include "HalfLife/HL1ImportSettings.h"
  42. struct aiNode;
  43. struct aiTexture;
  44. namespace Assimp {
  45. using namespace MDL;
  46. // --------------------------------------------------------------------------------------
  47. // Include file/line information in debug builds
  48. #ifdef ASSIMP_BUILD_DEBUG
  49. # define VALIDATE_FILE_SIZE(msg) SizeCheck(msg,__FILE__,__LINE__)
  50. #else
  51. # define VALIDATE_FILE_SIZE(msg) SizeCheck(msg)
  52. #endif
  53. // --------------------------------------------------------------------------------------
  54. /** @brief Class to load MDL files.
  55. *
  56. * Several subformats exist:
  57. * <ul>
  58. * <li>Quake I</li>
  59. * <li>3D Game Studio MDL3, MDL4</li>
  60. * <li>3D Game Studio MDL5</li>
  61. * <li>3D Game Studio MDL7</li>
  62. * <li>Halflife 1</li>
  63. * <li>Halflife 2</li>
  64. * </ul>
  65. * These formats are partially identical and it would be possible to load
  66. * them all with a single 1000-line function-beast. However, it has been
  67. * split into several code paths to make the code easier to read and maintain.
  68. */
  69. class MDLImporter : public BaseImporter {
  70. public:
  71. MDLImporter();
  72. ~MDLImporter() override = default;
  73. // -------------------------------------------------------------------
  74. /** Returns whether the class can handle the format of the given file.
  75. * See BaseImporter::CanRead() for details. */
  76. bool CanRead( const std::string& pFile, IOSystem* pIOHandler,
  77. bool checkSig) const override;
  78. // -------------------------------------------------------------------
  79. /** Called prior to ReadFile().
  80. * The function is a request to the importer to update its configuration
  81. * basing on the Importer's configuration property list.
  82. */
  83. void SetupProperties(const Importer* pImp) override;
  84. protected:
  85. // -------------------------------------------------------------------
  86. /** Return importer meta information.
  87. * See #BaseImporter::GetInfo for the details
  88. */
  89. const aiImporterDesc* GetInfo () const override;
  90. // -------------------------------------------------------------------
  91. /** Imports the given file into the given scene structure.
  92. * See BaseImporter::InternReadFile() for details
  93. */
  94. void InternReadFile( const std::string& pFile, aiScene* pScene,
  95. IOSystem* pIOHandler) override;
  96. // -------------------------------------------------------------------
  97. /** Import a quake 1 MDL file (IDPO)
  98. */
  99. void InternReadFile_Quake1( );
  100. // -------------------------------------------------------------------
  101. /** Import a GameStudio A4/A5 file (MDL 3,4,5)
  102. */
  103. void InternReadFile_3DGS_MDL345( );
  104. // -------------------------------------------------------------------
  105. /** Import a GameStudio A7 file (MDL 7)
  106. */
  107. void InternReadFile_3DGS_MDL7( );
  108. // -------------------------------------------------------------------
  109. /** Import a Half-Life 1 MDL file
  110. */
  111. void InternReadFile_HL1(const std::string& pFile, const uint32_t iMagicWord);
  112. // -------------------------------------------------------------------
  113. /** Import a CS:S/HL2 MDL file (not fully implemented)
  114. */
  115. AI_WONT_RETURN void InternReadFile_HL2( ) AI_WONT_RETURN_SUFFIX;
  116. // -------------------------------------------------------------------
  117. /** Check whether a given position is inside the valid range
  118. * Throw a DeadlyImportError if it is not
  119. * \param szPos Cursor position
  120. * \param szFile Name of the source file from which the function was called
  121. * \param iLine Source code line from which the function was called
  122. */
  123. void SizeCheck(const void* szPos);
  124. void SizeCheck(const void* szPos, const char* szFile, unsigned int iLine);
  125. bool IsPosValid(const void* szPos) const;
  126. // -------------------------------------------------------------------
  127. /** Validate the header data structure of a game studio MDL7 file
  128. * \param pcHeader Input header to be validated
  129. */
  130. void ValidateHeader_3DGS_MDL7(const MDL::Header_MDL7* pcHeader);
  131. // -------------------------------------------------------------------
  132. /** Validate the header data structure of a Quake 1 model
  133. * \param pcHeader Input header to be validated
  134. */
  135. void ValidateHeader_Quake1(const MDL::Header* pcHeader);
  136. // -------------------------------------------------------------------
  137. /** Try to load a palette from the current directory (colormap.lmp)
  138. * If it is not found the default palette of Quake1 is returned
  139. */
  140. void SearchPalette(const unsigned char** pszColorMap);
  141. // -------------------------------------------------------------------
  142. /** Free a palette created with a previous call to SearchPalette()
  143. */
  144. void FreePalette(const unsigned char* pszColorMap);
  145. // -------------------------------------------------------------------
  146. /** Load a palletized texture from the file and convert it to 32bpp
  147. */
  148. void CreateTextureARGB8_3DGS_MDL3(const unsigned char* szData);
  149. // -------------------------------------------------------------------
  150. /** Used to load textures from MDL3/4
  151. * \param szData Input data
  152. * \param iType Color data type
  153. * \param piSkip Receive: Size to skip, in bytes
  154. */
  155. void CreateTexture_3DGS_MDL4(const unsigned char* szData,
  156. unsigned int iType,
  157. unsigned int* piSkip);
  158. // -------------------------------------------------------------------
  159. /** Used to load textures from MDL5
  160. * \param szData Input data
  161. * \param iType Color data type
  162. * \param piSkip Receive: Size to skip, in bytes
  163. */
  164. void CreateTexture_3DGS_MDL5(const unsigned char* szData,
  165. unsigned int iType,
  166. unsigned int* piSkip);
  167. // -------------------------------------------------------------------
  168. /** Checks whether a texture can be replaced with a single color
  169. * This is useful for all file formats before MDL7 (all those
  170. * that are not containing material colors separate from textures).
  171. * MED seems to write dummy 8x8 monochrome images instead.
  172. * \param pcTexture Input texture
  173. * \return aiColor.r is set to qnan if the function fails and no
  174. * color can be found.
  175. */
  176. aiColor4D ReplaceTextureWithColor(const aiTexture* pcTexture);
  177. // -------------------------------------------------------------------
  178. /** Converts the absolute texture coordinates in MDL5 files to
  179. * relative in a range between 0 and 1
  180. */
  181. void CalculateUVCoordinates_MDL5();
  182. // -------------------------------------------------------------------
  183. /** Read an UV coordinate from the file. If the file format is not
  184. * MDL5, the function calculates relative texture coordinates
  185. * \param vOut Receives the output UV coord
  186. * \param pcSrc UV coordinate buffer
  187. * \param UV coordinate index
  188. */
  189. void ImportUVCoordinate_3DGS_MDL345( aiVector3D& vOut,
  190. const MDL::TexCoord_MDL3* pcSrc,
  191. unsigned int iIndex);
  192. // -------------------------------------------------------------------
  193. /** Setup the material properties for Quake and MDL<7 models.
  194. * These formats don't support more than one material per mesh,
  195. * therefore the method processes only ONE skin and removes
  196. * all others.
  197. */
  198. void SetupMaterialProperties_3DGS_MDL5_Quake1( );
  199. // -------------------------------------------------------------------
  200. /** Parse a skin lump in a MDL7/HMP7 file with all of its features
  201. * variant 1: Current cursor position is the beginning of the skin header
  202. * \param szCurrent Current data pointer
  203. * \param szCurrentOut Output data pointer
  204. * \param pcMats Material list for this group. To be filled ...
  205. */
  206. void ParseSkinLump_3DGS_MDL7(
  207. const unsigned char* szCurrent,
  208. const unsigned char** szCurrentOut,
  209. std::vector<aiMaterial*>& pcMats);
  210. // -------------------------------------------------------------------
  211. /** Parse a skin lump in a MDL7/HMP7 file with all of its features
  212. * variant 2: Current cursor position is the beginning of the skin data
  213. * \param szCurrent Current data pointer
  214. * \param szCurrentOut Output data pointer
  215. * \param pcMatOut Output material
  216. * \param iType header.typ
  217. * \param iWidth header.width
  218. * \param iHeight header.height
  219. */
  220. void ParseSkinLump_3DGS_MDL7(
  221. const unsigned char* szCurrent,
  222. const unsigned char** szCurrentOut,
  223. aiMaterial* pcMatOut,
  224. unsigned int iType,
  225. unsigned int iWidth,
  226. unsigned int iHeight);
  227. // -------------------------------------------------------------------
  228. /** Skip a skin lump in a MDL7/HMP7 file
  229. * \param szCurrent Current data pointer
  230. * \param szCurrentOut Output data pointer. Points to the byte just
  231. * behind the last byte of the skin.
  232. * \param iType header.typ
  233. * \param iWidth header.width
  234. * \param iHeight header.height
  235. */
  236. void SkipSkinLump_3DGS_MDL7(const unsigned char* szCurrent,
  237. const unsigned char** szCurrentOut,
  238. unsigned int iType,
  239. unsigned int iWidth,
  240. unsigned int iHeight);
  241. // -------------------------------------------------------------------
  242. /** Parse texture color data for MDL5, MDL6 and MDL7 formats
  243. * \param szData Current data pointer
  244. * \param iType type of the texture data. No DDS or external
  245. * \param piSkip Receive the number of bytes to skip
  246. * \param pcNew Must point to fully initialized data. Width and
  247. * height must be set. If pcNew->pcData is set to UINT_MAX,
  248. * piSkip will receive the size of the texture, in bytes, but no
  249. * color data will be read.
  250. */
  251. void ParseTextureColorData(const unsigned char* szData,
  252. unsigned int iType,
  253. unsigned int* piSkip,
  254. aiTexture* pcNew);
  255. // -------------------------------------------------------------------
  256. /** Join two materials / skins. Setup UV source ... etc
  257. * \param pcMat1 First input material
  258. * \param pcMat2 Second input material
  259. * \param pcMatOut Output material instance to be filled. Must be empty
  260. */
  261. void JoinSkins_3DGS_MDL7(aiMaterial* pcMat1,
  262. aiMaterial* pcMat2,
  263. aiMaterial* pcMatOut);
  264. // -------------------------------------------------------------------
  265. /** Add a bone transformation key to an animation
  266. * \param iTrafo Index of the transformation (always==frame index?)
  267. * No need to validate this index, it is always valid.
  268. * \param pcBoneTransforms Bone transformation for this index
  269. * \param apcOutBones Output bones array
  270. */
  271. void AddAnimationBoneTrafoKey_3DGS_MDL7(unsigned int iTrafo,
  272. const MDL::BoneTransform_MDL7* pcBoneTransforms,
  273. MDL::IntBone_MDL7** apcBonesOut);
  274. // -------------------------------------------------------------------
  275. /** Load the bone list of a MDL7 file
  276. * \return If the bones could be loaded successfully, a valid
  277. * array containing pointers to a temporary bone
  278. * representation. nullptr if the bones could not be loaded.
  279. */
  280. MDL::IntBone_MDL7** LoadBones_3DGS_MDL7();
  281. // -------------------------------------------------------------------
  282. /** Load bone transformation keyframes from a file chunk
  283. * \param groupInfo -> doc of data structure
  284. * \param frame -> doc of data structure
  285. * \param shared -> doc of data structure
  286. */
  287. void ParseBoneTrafoKeys_3DGS_MDL7(
  288. const MDL::IntGroupInfo_MDL7& groupInfo,
  289. IntFrameInfo_MDL7& frame,
  290. MDL::IntSharedData_MDL7& shared);
  291. // -------------------------------------------------------------------
  292. /** Calculate absolute bone animation matrices for each bone
  293. * \param apcOutBones Output bones array
  294. */
  295. void CalcAbsBoneMatrices_3DGS_MDL7(MDL::IntBone_MDL7** apcOutBones);
  296. // -------------------------------------------------------------------
  297. /** Add all bones to the nodegraph (as children of the root node)
  298. * \param apcBonesOut List of bones
  299. * \param pcParent Parent node. New nodes will be added to this node
  300. * \param iParentIndex Index of the parent bone
  301. */
  302. void AddBonesToNodeGraph_3DGS_MDL7(const MDL::IntBone_MDL7** apcBonesOut,
  303. aiNode* pcParent,uint16_t iParentIndex);
  304. // -------------------------------------------------------------------
  305. /** Build output animations
  306. * \param apcBonesOut List of bones
  307. */
  308. void BuildOutputAnims_3DGS_MDL7(const MDL::IntBone_MDL7** apcBonesOut);
  309. // -------------------------------------------------------------------
  310. /** Handles materials that are just referencing another material
  311. * There is no test file for this feature, but Conitec's doc
  312. * say it is used.
  313. */
  314. void HandleMaterialReferences_3DGS_MDL7();
  315. // -------------------------------------------------------------------
  316. /** Copies only the material that are referenced by at least one
  317. * mesh to the final output material list. All other materials
  318. * will be discarded.
  319. * \param shared -> doc of data structure
  320. */
  321. void CopyMaterials_3DGS_MDL7(MDL::IntSharedData_MDL7 &shared);
  322. // -------------------------------------------------------------------
  323. /** Process the frame section at the end of a group
  324. * \param groupInfo -> doc of data structure
  325. * \param shared -> doc of data structure
  326. * \param szCurrent Pointer to the start of the frame section
  327. * \param szCurrentOut Receives a pointer to the first byte of the
  328. * next data section.
  329. * \return false to read no further groups (a small workaround for
  330. * some tiny and unsolved problems ... )
  331. */
  332. bool ProcessFrames_3DGS_MDL7(const MDL::IntGroupInfo_MDL7& groupInfo,
  333. MDL::IntGroupData_MDL7& groupData,
  334. MDL::IntSharedData_MDL7& shared,
  335. const unsigned char* szCurrent,
  336. const unsigned char** szCurrentOut);
  337. // -------------------------------------------------------------------
  338. /** Sort all faces by their materials. If the mesh is using
  339. * multiple materials per face (that are blended together) the function
  340. * might create new materials.
  341. * \param groupInfo -> doc of data structure
  342. * \param groupData -> doc of data structure
  343. * \param splitGroupData -> doc of data structure
  344. */
  345. void SortByMaterials_3DGS_MDL7(
  346. const MDL::IntGroupInfo_MDL7& groupInfo,
  347. MDL::IntGroupData_MDL7& groupData,
  348. MDL::IntSplitGroupData_MDL7& splitGroupData);
  349. // -------------------------------------------------------------------
  350. /** Read all faces and vertices from a MDL7 group. The function fills
  351. * preallocated memory buffers.
  352. * \param groupInfo -> doc of data structure
  353. * \param groupData -> doc of data structure
  354. */
  355. void ReadFaces_3DGS_MDL7(const MDL::IntGroupInfo_MDL7& groupInfo,
  356. MDL::IntGroupData_MDL7& groupData);
  357. // -------------------------------------------------------------------
  358. /** Generate the final output meshes for a7 models
  359. * \param groupData -> doc of data structure
  360. * \param splitGroupData -> doc of data structure
  361. */
  362. void GenerateOutputMeshes_3DGS_MDL7(
  363. MDL::IntGroupData_MDL7& groupData,
  364. MDL::IntSplitGroupData_MDL7& splitGroupData);
  365. protected:
  366. /** Configuration option: frame to be loaded */
  367. unsigned int configFrameID;
  368. /** Configuration option: palette to be used to decode palletized images*/
  369. std::string configPalette;
  370. /** Buffer to hold the loaded file */
  371. unsigned char* mBuffer;
  372. /** For GameStudio MDL files: The number in the magic word, either 3,4 or 5
  373. * (MDL7 doesn't need this, the format has a separate loader) */
  374. unsigned int iGSFileVersion;
  375. /** Output I/O handler. used to load external lmp files */
  376. IOSystem* mIOHandler;
  377. /** Output scene to be filled */
  378. aiScene* pScene;
  379. /** Size of the input file in bytes */
  380. unsigned int iFileSize;
  381. /* Configuration for HL1 MDL */
  382. HalfLife::HL1ImportSettings mHL1ImportSettings;
  383. };
  384. } // end of namespace Assimp
  385. #endif // AI_3DSIMPORTER_H_INC