2
0

OgreBinarySerializer.cpp 32 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109
  1. /*
  2. Open Asset Import Library (assimp)
  3. ----------------------------------------------------------------------
  4. Copyright (c) 2006-2016, 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. #include "OgreBinarySerializer.h"
  34. #include "OgreXmlSerializer.h"
  35. #include "OgreParsingUtils.h"
  36. #include "TinyFormatter.h"
  37. #include <assimp/DefaultLogger.hpp>
  38. #ifndef ASSIMP_BUILD_NO_OGRE_IMPORTER
  39. // Define as 1 to get verbose logging.
  40. #define OGRE_BINARY_SERIALIZER_DEBUG 0
  41. namespace Assimp
  42. {
  43. namespace Ogre
  44. {
  45. const std::string MESH_VERSION_1_8 = "[MeshSerializer_v1.8]";
  46. const std::string SKELETON_VERSION_1_8 = "[Serializer_v1.80]";
  47. const std::string SKELETON_VERSION_1_1 = "[Serializer_v1.10]";
  48. const unsigned short HEADER_CHUNK_ID = 0x1000;
  49. const long MSTREAM_OVERHEAD_SIZE = sizeof(uint16_t) + sizeof(uint32_t);
  50. const long MSTREAM_BONE_SIZE_WITHOUT_SCALE = MSTREAM_OVERHEAD_SIZE + sizeof(unsigned short) + (sizeof(float) * 7);
  51. const long MSTREAM_KEYFRAME_SIZE_WITHOUT_SCALE = MSTREAM_OVERHEAD_SIZE + (sizeof(float) * 8);
  52. template<>
  53. inline bool OgreBinarySerializer::Read<bool>()
  54. {
  55. return (m_reader->GetU1() > 0);
  56. }
  57. template<>
  58. inline char OgreBinarySerializer::Read<char>()
  59. {
  60. return static_cast<char>(m_reader->GetU1());
  61. }
  62. template<>
  63. inline uint8_t OgreBinarySerializer::Read<uint8_t>()
  64. {
  65. return m_reader->GetU1();
  66. }
  67. template<>
  68. inline uint16_t OgreBinarySerializer::Read<uint16_t>()
  69. {
  70. return m_reader->GetU2();
  71. }
  72. template<>
  73. inline uint32_t OgreBinarySerializer::Read<uint32_t>()
  74. {
  75. return m_reader->GetU4();
  76. }
  77. template<>
  78. inline float OgreBinarySerializer::Read<float>()
  79. {
  80. return m_reader->GetF4();
  81. }
  82. void OgreBinarySerializer::ReadBytes(char *dest, size_t numBytes)
  83. {
  84. ReadBytes(static_cast<void*>(dest), numBytes);
  85. }
  86. void OgreBinarySerializer::ReadBytes(uint8_t *dest, size_t numBytes)
  87. {
  88. ReadBytes(static_cast<void*>(dest), numBytes);
  89. }
  90. void OgreBinarySerializer::ReadBytes(void *dest, size_t numBytes)
  91. {
  92. m_reader->CopyAndAdvance(dest, numBytes);
  93. }
  94. uint8_t *OgreBinarySerializer::ReadBytes(size_t numBytes)
  95. {
  96. uint8_t *bytes = new uint8_t[numBytes];
  97. ReadBytes(bytes, numBytes);
  98. return bytes;
  99. }
  100. void OgreBinarySerializer::ReadVector(aiVector3D &vec)
  101. {
  102. m_reader->CopyAndAdvance(&vec.x, sizeof(float)*3);
  103. }
  104. void OgreBinarySerializer::ReadQuaternion(aiQuaternion &quat)
  105. {
  106. float temp[4];
  107. m_reader->CopyAndAdvance(temp, sizeof(float)*4);
  108. quat.x = temp[0];
  109. quat.y = temp[1];
  110. quat.z = temp[2];
  111. quat.w = temp[3];
  112. }
  113. bool OgreBinarySerializer::AtEnd() const
  114. {
  115. return (m_reader->GetRemainingSize() == 0);
  116. }
  117. std::string OgreBinarySerializer::ReadString(size_t len)
  118. {
  119. std::string str;
  120. str.resize(len);
  121. ReadBytes(&str[0], len);
  122. return str;
  123. }
  124. std::string OgreBinarySerializer::ReadLine()
  125. {
  126. std::string str;
  127. while(!AtEnd())
  128. {
  129. char c = Read<char>();
  130. if (c == '\n')
  131. break;
  132. str += c;
  133. }
  134. return str;
  135. }
  136. uint16_t OgreBinarySerializer::ReadHeader(bool readLen)
  137. {
  138. uint16_t id = Read<uint16_t>();
  139. if (readLen)
  140. m_currentLen = Read<uint32_t>();
  141. #if (OGRE_BINARY_SERIALIZER_DEBUG == 1)
  142. if (id != HEADER_CHUNK_ID)
  143. {
  144. DefaultLogger::get()->debug(Formatter::format() << (assetMode == AM_Mesh
  145. ? MeshHeaderToString(static_cast<MeshChunkId>(id)) : SkeletonHeaderToString(static_cast<SkeletonChunkId>(id))));
  146. }
  147. #endif
  148. return id;
  149. }
  150. void OgreBinarySerializer::RollbackHeader()
  151. {
  152. m_reader->IncPtr(-MSTREAM_OVERHEAD_SIZE);
  153. }
  154. void OgreBinarySerializer::SkipBytes(size_t numBytes)
  155. {
  156. #if (OGRE_BINARY_SERIALIZER_DEBUG == 1)
  157. DefaultLogger::get()->debug(Formatter::format() << "Skipping " << numBytes << " bytes");
  158. #endif
  159. m_reader->IncPtr(numBytes);
  160. }
  161. // Mesh
  162. Mesh *OgreBinarySerializer::ImportMesh(MemoryStreamReader *stream)
  163. {
  164. OgreBinarySerializer serializer(stream, OgreBinarySerializer::AM_Mesh);
  165. uint16_t id = serializer.ReadHeader(false);
  166. if (id != HEADER_CHUNK_ID) {
  167. throw DeadlyExportError("Invalid Ogre Mesh file header.");
  168. }
  169. /// @todo Check what we can actually support.
  170. std::string version = serializer.ReadLine();
  171. if (version != MESH_VERSION_1_8)
  172. {
  173. throw DeadlyExportError(Formatter::format() << "Mesh version " << version << " not supported by this importer. Run OgreMeshUpgrader tool on the file and try again."
  174. << " Supported versions: " << MESH_VERSION_1_8);
  175. }
  176. Mesh *mesh = new Mesh();
  177. while (!serializer.AtEnd())
  178. {
  179. id = serializer.ReadHeader();
  180. switch(id)
  181. {
  182. case M_MESH:
  183. {
  184. serializer.ReadMesh(mesh);
  185. break;
  186. }
  187. }
  188. }
  189. return mesh;
  190. }
  191. void OgreBinarySerializer::ReadMesh(Mesh *mesh)
  192. {
  193. mesh->hasSkeletalAnimations = Read<bool>();
  194. DefaultLogger::get()->debug("Reading Mesh");
  195. DefaultLogger::get()->debug(Formatter::format() << " - Skeletal animations: " << (mesh->hasSkeletalAnimations ? "true" : "false"));
  196. if (!AtEnd())
  197. {
  198. uint16_t id = ReadHeader();
  199. while (!AtEnd() &&
  200. (id == M_GEOMETRY ||
  201. id == M_SUBMESH ||
  202. id == M_MESH_SKELETON_LINK ||
  203. id == M_MESH_BONE_ASSIGNMENT ||
  204. id == M_MESH_LOD ||
  205. id == M_MESH_BOUNDS ||
  206. id == M_SUBMESH_NAME_TABLE ||
  207. id == M_EDGE_LISTS ||
  208. id == M_POSES ||
  209. id == M_ANIMATIONS ||
  210. id == M_TABLE_EXTREMES))
  211. {
  212. switch(id)
  213. {
  214. case M_GEOMETRY:
  215. {
  216. mesh->sharedVertexData = new VertexData();
  217. ReadGeometry(mesh->sharedVertexData);
  218. break;
  219. }
  220. case M_SUBMESH:
  221. {
  222. ReadSubMesh(mesh);
  223. break;
  224. }
  225. case M_MESH_SKELETON_LINK:
  226. {
  227. ReadMeshSkeletonLink(mesh);
  228. break;
  229. }
  230. case M_MESH_BONE_ASSIGNMENT:
  231. {
  232. ReadBoneAssignment(mesh->sharedVertexData);
  233. break;
  234. }
  235. case M_MESH_LOD:
  236. {
  237. ReadMeshLodInfo(mesh);
  238. break;
  239. }
  240. case M_MESH_BOUNDS:
  241. {
  242. ReadMeshBounds(mesh);
  243. break;
  244. }
  245. case M_SUBMESH_NAME_TABLE:
  246. {
  247. ReadSubMeshNames(mesh);
  248. break;
  249. }
  250. case M_EDGE_LISTS:
  251. {
  252. ReadEdgeList(mesh);
  253. break;
  254. }
  255. case M_POSES:
  256. {
  257. ReadPoses(mesh);
  258. break;
  259. }
  260. case M_ANIMATIONS:
  261. {
  262. ReadAnimations(mesh);
  263. break;
  264. }
  265. case M_TABLE_EXTREMES:
  266. {
  267. ReadMeshExtremes(mesh);
  268. break;
  269. }
  270. }
  271. if (!AtEnd())
  272. id = ReadHeader();
  273. }
  274. if (!AtEnd())
  275. RollbackHeader();
  276. }
  277. NormalizeBoneWeights(mesh->sharedVertexData);
  278. }
  279. void OgreBinarySerializer::ReadMeshLodInfo(Mesh *mesh)
  280. {
  281. // Assimp does not acknowledge LOD levels as far as I can see it. This info is just skipped.
  282. // @todo Put this stuff to scene/mesh custom properties. If manual mesh the app can use the information.
  283. ReadLine(); // strategy name
  284. uint16_t numLods = Read<uint16_t>();
  285. bool manual = Read<bool>();
  286. /// @note Main mesh is considered as LOD 0, start from index 1.
  287. for (size_t i=1; i<numLods; ++i)
  288. {
  289. uint16_t id = ReadHeader();
  290. if (id != M_MESH_LOD_USAGE) {
  291. throw DeadlyImportError("M_MESH_LOD does not contain a M_MESH_LOD_USAGE for each LOD level");
  292. }
  293. m_reader->IncPtr(sizeof(float)); // user value
  294. if (manual)
  295. {
  296. id = ReadHeader();
  297. if (id != M_MESH_LOD_MANUAL) {
  298. throw DeadlyImportError("Manual M_MESH_LOD_USAGE does not contain M_MESH_LOD_MANUAL");
  299. }
  300. ReadLine(); // manual mesh name (ref to another mesh)
  301. }
  302. else
  303. {
  304. for(size_t si=0, silen=mesh->NumSubMeshes(); si<silen; ++si)
  305. {
  306. id = ReadHeader();
  307. if (id != M_MESH_LOD_GENERATED) {
  308. throw DeadlyImportError("Generated M_MESH_LOD_USAGE does not contain M_MESH_LOD_GENERATED");
  309. }
  310. uint32_t indexCount = Read<uint32_t>();
  311. bool is32bit = Read<bool>();
  312. if (indexCount > 0)
  313. {
  314. uint32_t len = indexCount * (is32bit ? sizeof(uint32_t) : sizeof(uint16_t));
  315. m_reader->IncPtr(len);
  316. }
  317. }
  318. }
  319. }
  320. }
  321. void OgreBinarySerializer::ReadMeshSkeletonLink(Mesh *mesh)
  322. {
  323. mesh->skeletonRef = ReadLine();
  324. }
  325. void OgreBinarySerializer::ReadMeshBounds(Mesh * /*mesh*/)
  326. {
  327. // Skip bounds, not compatible with Assimp.
  328. // 2x float vec3 + 1x float sphere radius
  329. SkipBytes(sizeof(float) * 7);
  330. }
  331. void OgreBinarySerializer::ReadMeshExtremes(Mesh * /*mesh*/)
  332. {
  333. // Skip extremes, not compatible with Assimp.
  334. size_t numBytes = m_currentLen - MSTREAM_OVERHEAD_SIZE;
  335. SkipBytes(numBytes);
  336. }
  337. void OgreBinarySerializer::ReadBoneAssignment(VertexData *dest)
  338. {
  339. if (!dest) {
  340. throw DeadlyImportError("Cannot read bone assignments, vertex data is null.");
  341. }
  342. VertexBoneAssignment ba;
  343. ba.vertexIndex = Read<uint32_t>();
  344. ba.boneIndex = Read<uint16_t>();
  345. ba.weight = Read<float>();
  346. dest->boneAssignments.push_back(ba);
  347. }
  348. void OgreBinarySerializer::ReadSubMesh(Mesh *mesh)
  349. {
  350. uint16_t id = 0;
  351. SubMesh *submesh = new SubMesh();
  352. submesh->materialRef = ReadLine();
  353. submesh->usesSharedVertexData = Read<bool>();
  354. submesh->indexData->count = Read<uint32_t>();
  355. submesh->indexData->faceCount = static_cast<uint32_t>(submesh->indexData->count / 3);
  356. submesh->indexData->is32bit = Read<bool>();
  357. DefaultLogger::get()->debug(Formatter::format() << "Reading SubMesh " << mesh->subMeshes.size());
  358. DefaultLogger::get()->debug(Formatter::format() << " - Material: '" << submesh->materialRef << "'");
  359. DefaultLogger::get()->debug(Formatter::format() << " - Uses shared geometry: " << (submesh->usesSharedVertexData ? "true" : "false"));
  360. // Index buffer
  361. if (submesh->indexData->count > 0)
  362. {
  363. uint32_t numBytes = submesh->indexData->count * (submesh->indexData->is32bit ? sizeof(uint32_t) : sizeof(uint16_t));
  364. uint8_t *indexBuffer = ReadBytes(numBytes);
  365. submesh->indexData->buffer = MemoryStreamPtr(new Assimp::MemoryIOStream(indexBuffer, numBytes, true));
  366. DefaultLogger::get()->debug(Formatter::format() << " - " << submesh->indexData->faceCount
  367. << " faces from " << submesh->indexData->count << (submesh->indexData->is32bit ? " 32bit" : " 16bit")
  368. << " indexes of " << numBytes << " bytes");
  369. }
  370. // Vertex buffer if not referencing the shared geometry
  371. if (!submesh->usesSharedVertexData)
  372. {
  373. id = ReadHeader();
  374. if (id != M_GEOMETRY) {
  375. throw DeadlyImportError("M_SUBMESH does not contain M_GEOMETRY, but shader geometry is set to false");
  376. }
  377. submesh->vertexData = new VertexData();
  378. ReadGeometry(submesh->vertexData);
  379. }
  380. // Bone assignment, submesh operation and texture aliases
  381. if (!AtEnd())
  382. {
  383. id = ReadHeader();
  384. while (!AtEnd() &&
  385. (id == M_SUBMESH_OPERATION ||
  386. id == M_SUBMESH_BONE_ASSIGNMENT ||
  387. id == M_SUBMESH_TEXTURE_ALIAS))
  388. {
  389. switch(id)
  390. {
  391. case M_SUBMESH_OPERATION:
  392. {
  393. ReadSubMeshOperation(submesh);
  394. break;
  395. }
  396. case M_SUBMESH_BONE_ASSIGNMENT:
  397. {
  398. ReadBoneAssignment(submesh->vertexData);
  399. break;
  400. }
  401. case M_SUBMESH_TEXTURE_ALIAS:
  402. {
  403. ReadSubMeshTextureAlias(submesh);
  404. break;
  405. }
  406. }
  407. if (!AtEnd())
  408. id = ReadHeader();
  409. }
  410. if (!AtEnd())
  411. RollbackHeader();
  412. }
  413. NormalizeBoneWeights(submesh->vertexData);
  414. submesh->index = static_cast<unsigned int>(mesh->subMeshes.size());
  415. mesh->subMeshes.push_back(submesh);
  416. }
  417. void OgreBinarySerializer::NormalizeBoneWeights(VertexData *vertexData) const
  418. {
  419. if (!vertexData || vertexData->boneAssignments.empty())
  420. return;
  421. std::set<uint32_t> influencedVertices;
  422. for (VertexBoneAssignmentList::const_iterator baIter=vertexData->boneAssignments.begin(), baEnd=vertexData->boneAssignments.end(); baIter != baEnd; ++baIter) {
  423. influencedVertices.insert(baIter->vertexIndex);
  424. }
  425. /** Normalize bone weights.
  426. Some exporters wont care if the sum of all bone weights
  427. for a single vertex equals 1 or not, so validate here. */
  428. const float epsilon = 0.05f;
  429. for (const uint32_t vertexIndex : influencedVertices)
  430. {
  431. float sum = 0.0f;
  432. for (VertexBoneAssignmentList::const_iterator baIter=vertexData->boneAssignments.begin(), baEnd=vertexData->boneAssignments.end(); baIter != baEnd; ++baIter)
  433. {
  434. if (baIter->vertexIndex == vertexIndex)
  435. sum += baIter->weight;
  436. }
  437. if ((sum < (1.0f - epsilon)) || (sum > (1.0f + epsilon)))
  438. {
  439. for (auto &boneAssign : vertexData->boneAssignments)
  440. {
  441. if (boneAssign.vertexIndex == vertexIndex)
  442. boneAssign.weight /= sum;
  443. }
  444. }
  445. }
  446. }
  447. void OgreBinarySerializer::ReadSubMeshOperation(SubMesh *submesh)
  448. {
  449. submesh->operationType = static_cast<SubMesh::OperationType>(Read<uint16_t>());
  450. }
  451. void OgreBinarySerializer::ReadSubMeshTextureAlias(SubMesh *submesh)
  452. {
  453. submesh->textureAliasName = ReadLine();
  454. submesh->textureAliasRef = ReadLine();
  455. }
  456. void OgreBinarySerializer::ReadSubMeshNames(Mesh *mesh)
  457. {
  458. uint16_t id = 0;
  459. if (!AtEnd())
  460. {
  461. id = ReadHeader();
  462. while (!AtEnd() && id == M_SUBMESH_NAME_TABLE_ELEMENT)
  463. {
  464. uint16_t submeshIndex = Read<uint16_t>();
  465. SubMesh *submesh = mesh->GetSubMesh(submeshIndex);
  466. if (!submesh) {
  467. throw DeadlyImportError(Formatter::format() << "Ogre Mesh does not include submesh " << submeshIndex << " referenced in M_SUBMESH_NAME_TABLE_ELEMENT. Invalid mesh file.");
  468. }
  469. submesh->name = ReadLine();
  470. DefaultLogger::get()->debug(Formatter::format() << " - SubMesh " << submesh->index << " name '" << submesh->name << "'");
  471. if (!AtEnd())
  472. id = ReadHeader();
  473. }
  474. if (!AtEnd())
  475. RollbackHeader();
  476. }
  477. }
  478. void OgreBinarySerializer::ReadGeometry(VertexData *dest)
  479. {
  480. dest->count = Read<uint32_t>();
  481. DefaultLogger::get()->debug(Formatter::format() << " - Reading geometry of " << dest->count << " vertices");
  482. if (!AtEnd())
  483. {
  484. uint16_t id = ReadHeader();
  485. while (!AtEnd() &&
  486. (id == M_GEOMETRY_VERTEX_DECLARATION ||
  487. id == M_GEOMETRY_VERTEX_BUFFER))
  488. {
  489. switch(id)
  490. {
  491. case M_GEOMETRY_VERTEX_DECLARATION:
  492. {
  493. ReadGeometryVertexDeclaration(dest);
  494. break;
  495. }
  496. case M_GEOMETRY_VERTEX_BUFFER:
  497. {
  498. ReadGeometryVertexBuffer(dest);
  499. break;
  500. }
  501. }
  502. if (!AtEnd())
  503. id = ReadHeader();
  504. }
  505. if (!AtEnd())
  506. RollbackHeader();
  507. }
  508. }
  509. void OgreBinarySerializer::ReadGeometryVertexDeclaration(VertexData *dest)
  510. {
  511. if (!AtEnd())
  512. {
  513. uint16_t id = ReadHeader();
  514. while (!AtEnd() && id == M_GEOMETRY_VERTEX_ELEMENT)
  515. {
  516. ReadGeometryVertexElement(dest);
  517. if (!AtEnd())
  518. id = ReadHeader();
  519. }
  520. if (!AtEnd())
  521. RollbackHeader();
  522. }
  523. }
  524. void OgreBinarySerializer::ReadGeometryVertexElement(VertexData *dest)
  525. {
  526. VertexElement element;
  527. element.source = Read<uint16_t>();
  528. element.type = static_cast<VertexElement::Type>(Read<uint16_t>());
  529. element.semantic = static_cast<VertexElement::Semantic>(Read<uint16_t>());
  530. element.offset = Read<uint16_t>();
  531. element.index = Read<uint16_t>();
  532. DefaultLogger::get()->debug(Formatter::format() << " - Vertex element " << element.SemanticToString() << " of type "
  533. << element.TypeToString() << " index=" << element.index << " source=" << element.source);
  534. dest->vertexElements.push_back(element);
  535. }
  536. void OgreBinarySerializer::ReadGeometryVertexBuffer(VertexData *dest)
  537. {
  538. uint16_t bindIndex = Read<uint16_t>();
  539. uint16_t vertexSize = Read<uint16_t>();
  540. uint16_t id = ReadHeader();
  541. if (id != M_GEOMETRY_VERTEX_BUFFER_DATA)
  542. throw DeadlyImportError("M_GEOMETRY_VERTEX_BUFFER_DATA not found in M_GEOMETRY_VERTEX_BUFFER");
  543. if (dest->VertexSize(bindIndex) != vertexSize)
  544. throw DeadlyImportError("Vertex buffer size does not agree with vertex declaration in M_GEOMETRY_VERTEX_BUFFER");
  545. size_t numBytes = dest->count * vertexSize;
  546. uint8_t *vertexBuffer = ReadBytes(numBytes);
  547. dest->vertexBindings[bindIndex] = MemoryStreamPtr(new Assimp::MemoryIOStream(vertexBuffer, numBytes, true));
  548. DefaultLogger::get()->debug(Formatter::format() << " - Read vertex buffer for source " << bindIndex << " of " << numBytes << " bytes");
  549. }
  550. void OgreBinarySerializer::ReadEdgeList(Mesh * /*mesh*/)
  551. {
  552. // Assimp does not acknowledge LOD levels as far as I can see it. This info is just skipped.
  553. if (!AtEnd())
  554. {
  555. uint16_t id = ReadHeader();
  556. while (!AtEnd() && id == M_EDGE_LIST_LOD)
  557. {
  558. m_reader->IncPtr(sizeof(uint16_t)); // lod index
  559. bool manual = Read<bool>();
  560. if (!manual)
  561. {
  562. m_reader->IncPtr(sizeof(uint8_t));
  563. uint32_t numTriangles = Read<uint32_t>();
  564. uint32_t numEdgeGroups = Read<uint32_t>();
  565. size_t skipBytes = (sizeof(uint32_t) * 8 + sizeof(float) * 4) * numTriangles;
  566. m_reader->IncPtr(skipBytes);
  567. for (size_t i=0; i<numEdgeGroups; ++i)
  568. {
  569. uint16_t id = ReadHeader();
  570. if (id != M_EDGE_GROUP)
  571. throw DeadlyImportError("M_EDGE_GROUP not found in M_EDGE_LIST_LOD");
  572. m_reader->IncPtr(sizeof(uint32_t) * 3);
  573. uint32_t numEdges = Read<uint32_t>();
  574. for (size_t j=0; j<numEdges; ++j)
  575. {
  576. m_reader->IncPtr(sizeof(uint32_t) * 6 + sizeof(uint8_t));
  577. }
  578. }
  579. }
  580. if (!AtEnd())
  581. id = ReadHeader();
  582. }
  583. if (!AtEnd())
  584. RollbackHeader();
  585. }
  586. }
  587. void OgreBinarySerializer::ReadPoses(Mesh *mesh)
  588. {
  589. if (!AtEnd())
  590. {
  591. uint16_t id = ReadHeader();
  592. while (!AtEnd() && id == M_POSE)
  593. {
  594. Pose *pose = new Pose();
  595. pose->name = ReadLine();
  596. pose->target = Read<uint16_t>();
  597. pose->hasNormals = Read<bool>();
  598. ReadPoseVertices(pose);
  599. mesh->poses.push_back(pose);
  600. if (!AtEnd())
  601. id = ReadHeader();
  602. }
  603. if (!AtEnd())
  604. RollbackHeader();
  605. }
  606. }
  607. void OgreBinarySerializer::ReadPoseVertices(Pose *pose)
  608. {
  609. if (!AtEnd())
  610. {
  611. uint16_t id = ReadHeader();
  612. while (!AtEnd() && id == M_POSE_VERTEX)
  613. {
  614. Pose::Vertex v;
  615. v.index = Read<uint32_t>();
  616. ReadVector(v.offset);
  617. if (pose->hasNormals)
  618. ReadVector(v.normal);
  619. pose->vertices[v.index] = v;
  620. if (!AtEnd())
  621. id = ReadHeader();
  622. }
  623. if (!AtEnd())
  624. RollbackHeader();
  625. }
  626. }
  627. void OgreBinarySerializer::ReadAnimations(Mesh *mesh)
  628. {
  629. if (!AtEnd())
  630. {
  631. uint16_t id = ReadHeader();
  632. while (!AtEnd() && id == M_ANIMATION)
  633. {
  634. Animation *anim = new Animation(mesh);
  635. anim->name = ReadLine();
  636. anim->length = Read<float>();
  637. ReadAnimation(anim);
  638. mesh->animations.push_back(anim);
  639. if (!AtEnd())
  640. id = ReadHeader();
  641. }
  642. if (!AtEnd())
  643. RollbackHeader();
  644. }
  645. }
  646. void OgreBinarySerializer::ReadAnimation(Animation *anim)
  647. {
  648. if (!AtEnd())
  649. {
  650. uint16_t id = ReadHeader();
  651. if (id == M_ANIMATION_BASEINFO)
  652. {
  653. anim->baseName = ReadLine();
  654. anim->baseTime = Read<float>();
  655. // Advance to first track
  656. id = ReadHeader();
  657. }
  658. while (!AtEnd() && id == M_ANIMATION_TRACK)
  659. {
  660. VertexAnimationTrack track;
  661. track.type = static_cast<VertexAnimationTrack::Type>(Read<uint16_t>());
  662. track.target = Read<uint16_t>();
  663. ReadAnimationKeyFrames(anim, &track);
  664. anim->tracks.push_back(track);
  665. if (!AtEnd())
  666. id = ReadHeader();
  667. }
  668. if (!AtEnd())
  669. RollbackHeader();
  670. }
  671. }
  672. void OgreBinarySerializer::ReadAnimationKeyFrames(Animation *anim, VertexAnimationTrack *track)
  673. {
  674. if (!AtEnd())
  675. {
  676. uint16_t id = ReadHeader();
  677. while (!AtEnd() &&
  678. (id == M_ANIMATION_MORPH_KEYFRAME ||
  679. id == M_ANIMATION_POSE_KEYFRAME))
  680. {
  681. if (id == M_ANIMATION_MORPH_KEYFRAME)
  682. {
  683. MorphKeyFrame kf;
  684. kf.timePos = Read<float>();
  685. bool hasNormals = Read<bool>();
  686. size_t vertexCount = anim->AssociatedVertexData(track)->count;
  687. size_t vertexSize = sizeof(float) * (hasNormals ? 6 : 3);
  688. size_t numBytes = vertexCount * vertexSize;
  689. uint8_t *morphBuffer = ReadBytes(numBytes);
  690. kf.buffer = MemoryStreamPtr(new Assimp::MemoryIOStream(morphBuffer, numBytes, true));
  691. track->morphKeyFrames.push_back(kf);
  692. }
  693. else if (id == M_ANIMATION_POSE_KEYFRAME)
  694. {
  695. PoseKeyFrame kf;
  696. kf.timePos = Read<float>();
  697. if (!AtEnd())
  698. {
  699. id = ReadHeader();
  700. while (!AtEnd() && id == M_ANIMATION_POSE_REF)
  701. {
  702. PoseRef pr;
  703. pr.index = Read<uint16_t>();
  704. pr.influence = Read<float>();
  705. kf.references.push_back(pr);
  706. if (!AtEnd())
  707. id = ReadHeader();
  708. }
  709. if (!AtEnd())
  710. RollbackHeader();
  711. }
  712. track->poseKeyFrames.push_back(kf);
  713. }
  714. if (!AtEnd())
  715. id = ReadHeader();
  716. }
  717. if (!AtEnd())
  718. RollbackHeader();
  719. }
  720. }
  721. // Skeleton
  722. bool OgreBinarySerializer::ImportSkeleton(Assimp::IOSystem *pIOHandler, Mesh *mesh)
  723. {
  724. if (!mesh || mesh->skeletonRef.empty())
  725. return false;
  726. // Highly unusual to see in read world cases but support
  727. // binary mesh referencing a XML skeleton file.
  728. if (EndsWith(mesh->skeletonRef, ".skeleton.xml", false))
  729. {
  730. OgreXmlSerializer::ImportSkeleton(pIOHandler, mesh);
  731. return false;
  732. }
  733. MemoryStreamReaderPtr reader = OpenReader(pIOHandler, mesh->skeletonRef);
  734. Skeleton *skeleton = new Skeleton();
  735. OgreBinarySerializer serializer(reader.get(), OgreBinarySerializer::AM_Skeleton);
  736. serializer.ReadSkeleton(skeleton);
  737. mesh->skeleton = skeleton;
  738. return true;
  739. }
  740. bool OgreBinarySerializer::ImportSkeleton(Assimp::IOSystem *pIOHandler, MeshXml *mesh)
  741. {
  742. if (!mesh || mesh->skeletonRef.empty())
  743. return false;
  744. MemoryStreamReaderPtr reader = OpenReader(pIOHandler, mesh->skeletonRef);
  745. if (!reader.get())
  746. return false;
  747. Skeleton *skeleton = new Skeleton();
  748. OgreBinarySerializer serializer(reader.get(), OgreBinarySerializer::AM_Skeleton);
  749. serializer.ReadSkeleton(skeleton);
  750. mesh->skeleton = skeleton;
  751. return true;
  752. }
  753. MemoryStreamReaderPtr OgreBinarySerializer::OpenReader(Assimp::IOSystem *pIOHandler, const std::string &filename)
  754. {
  755. if (!EndsWith(filename, ".skeleton", false))
  756. {
  757. DefaultLogger::get()->error("Imported Mesh is referencing to unsupported '" + filename + "' skeleton file.");
  758. return MemoryStreamReaderPtr();
  759. }
  760. if (!pIOHandler->Exists(filename))
  761. {
  762. DefaultLogger::get()->error("Failed to find skeleton file '" + filename + "' that is referenced by imported Mesh.");
  763. return MemoryStreamReaderPtr();
  764. }
  765. IOStream *f = pIOHandler->Open(filename, "rb");
  766. if (!f) {
  767. throw DeadlyImportError("Failed to open skeleton file " + filename);
  768. }
  769. return MemoryStreamReaderPtr(new MemoryStreamReader(f));
  770. }
  771. void OgreBinarySerializer::ReadSkeleton(Skeleton *skeleton)
  772. {
  773. uint16_t id = ReadHeader(false);
  774. if (id != HEADER_CHUNK_ID) {
  775. throw DeadlyExportError("Invalid Ogre Skeleton file header.");
  776. }
  777. // This deserialization supports both versions of the skeleton spec
  778. std::string version = ReadLine();
  779. if (version != SKELETON_VERSION_1_8 && version != SKELETON_VERSION_1_1)
  780. {
  781. throw DeadlyExportError(Formatter::format() << "Skeleton version " << version << " not supported by this importer."
  782. << " Supported versions: " << SKELETON_VERSION_1_8 << " and " << SKELETON_VERSION_1_1);
  783. }
  784. DefaultLogger::get()->debug("Reading Skeleton");
  785. bool firstBone = true;
  786. bool firstAnim = true;
  787. while (!AtEnd())
  788. {
  789. id = ReadHeader();
  790. switch(id)
  791. {
  792. case SKELETON_BLENDMODE:
  793. {
  794. skeleton->blendMode = static_cast<Skeleton::BlendMode>(Read<uint16_t>());
  795. break;
  796. }
  797. case SKELETON_BONE:
  798. {
  799. if (firstBone)
  800. {
  801. DefaultLogger::get()->debug(" - Bones");
  802. firstBone = false;
  803. }
  804. ReadBone(skeleton);
  805. break;
  806. }
  807. case SKELETON_BONE_PARENT:
  808. {
  809. ReadBoneParent(skeleton);
  810. break;
  811. }
  812. case SKELETON_ANIMATION:
  813. {
  814. if (firstAnim)
  815. {
  816. DefaultLogger::get()->debug(" - Animations");
  817. firstAnim = false;
  818. }
  819. ReadSkeletonAnimation(skeleton);
  820. break;
  821. }
  822. case SKELETON_ANIMATION_LINK:
  823. {
  824. ReadSkeletonAnimationLink(skeleton);
  825. break;
  826. }
  827. }
  828. }
  829. // Calculate bone matrices for root bones. Recursively calculates their children.
  830. for (size_t i=0, len=skeleton->bones.size(); i<len; ++i)
  831. {
  832. Bone *bone = skeleton->bones[i];
  833. if (!bone->IsParented())
  834. bone->CalculateWorldMatrixAndDefaultPose(skeleton);
  835. }
  836. }
  837. void OgreBinarySerializer::ReadBone(Skeleton *skeleton)
  838. {
  839. Bone *bone = new Bone();
  840. bone->name = ReadLine();
  841. bone->id = Read<uint16_t>();
  842. // Pos and rot
  843. ReadVector(bone->position);
  844. ReadQuaternion(bone->rotation);
  845. // Scale (optional)
  846. if (m_currentLen > MSTREAM_BONE_SIZE_WITHOUT_SCALE)
  847. ReadVector(bone->scale);
  848. // Bone indexes need to start from 0 and be contiguous
  849. if (bone->id != skeleton->bones.size()) {
  850. throw DeadlyImportError(Formatter::format() << "Ogre Skeleton bone indexes not contiguous. Error at bone index " << bone->id);
  851. }
  852. DefaultLogger::get()->debug(Formatter::format() << " " << bone->id << " " << bone->name);
  853. skeleton->bones.push_back(bone);
  854. }
  855. void OgreBinarySerializer::ReadBoneParent(Skeleton *skeleton)
  856. {
  857. uint16_t childId = Read<uint16_t>();
  858. uint16_t parentId = Read<uint16_t>();
  859. Bone *child = skeleton->BoneById(childId);
  860. Bone *parent = skeleton->BoneById(parentId);
  861. if (child && parent)
  862. parent->AddChild(child);
  863. else
  864. throw DeadlyImportError(Formatter::format() << "Failed to find bones for parenting: Child id " << childId << " for parent id " << parentId);
  865. }
  866. void OgreBinarySerializer::ReadSkeletonAnimation(Skeleton *skeleton)
  867. {
  868. Animation *anim = new Animation(skeleton);
  869. anim->name = ReadLine();
  870. anim->length = Read<float>();
  871. if (!AtEnd())
  872. {
  873. uint16_t id = ReadHeader();
  874. if (id == SKELETON_ANIMATION_BASEINFO)
  875. {
  876. anim->baseName = ReadLine();
  877. anim->baseTime = Read<float>();
  878. // Advance to first track
  879. id = ReadHeader();
  880. }
  881. while (!AtEnd() && id == SKELETON_ANIMATION_TRACK)
  882. {
  883. ReadSkeletonAnimationTrack(skeleton, anim);
  884. if (!AtEnd())
  885. id = ReadHeader();
  886. }
  887. if (!AtEnd())
  888. RollbackHeader();
  889. }
  890. skeleton->animations.push_back(anim);
  891. DefaultLogger::get()->debug(Formatter::format() << " " << anim->name << " (" << anim->length << " sec, " << anim->tracks.size() << " tracks)");
  892. }
  893. void OgreBinarySerializer::ReadSkeletonAnimationTrack(Skeleton * /*skeleton*/, Animation *dest)
  894. {
  895. uint16_t boneId = Read<uint16_t>();
  896. Bone *bone = dest->parentSkeleton->BoneById(boneId);
  897. if (!bone) {
  898. throw DeadlyImportError(Formatter::format() << "Cannot read animation track, target bone " << boneId << " not in target Skeleton");
  899. }
  900. VertexAnimationTrack track;
  901. track.type = VertexAnimationTrack::VAT_TRANSFORM;
  902. track.boneName = bone->name;
  903. uint16_t id = ReadHeader();
  904. while (!AtEnd() && id == SKELETON_ANIMATION_TRACK_KEYFRAME)
  905. {
  906. ReadSkeletonAnimationKeyFrame(&track);
  907. if (!AtEnd())
  908. id = ReadHeader();
  909. }
  910. if (!AtEnd())
  911. RollbackHeader();
  912. dest->tracks.push_back(track);
  913. }
  914. void OgreBinarySerializer::ReadSkeletonAnimationKeyFrame(VertexAnimationTrack *dest)
  915. {
  916. TransformKeyFrame keyframe;
  917. keyframe.timePos = Read<float>();
  918. // Rot and pos
  919. ReadQuaternion(keyframe.rotation);
  920. ReadVector(keyframe.position);
  921. // Scale (optional)
  922. if (m_currentLen > MSTREAM_KEYFRAME_SIZE_WITHOUT_SCALE)
  923. ReadVector(keyframe.scale);
  924. dest->transformKeyFrames.push_back(keyframe);
  925. }
  926. void OgreBinarySerializer::ReadSkeletonAnimationLink(Skeleton * /*skeleton*/)
  927. {
  928. // Skip bounds, not compatible with Assimp.
  929. ReadLine(); // skeleton name
  930. SkipBytes(sizeof(float) * 3); // scale
  931. }
  932. } // Ogre
  933. } // Assimp
  934. #endif // ASSIMP_BUILD_NO_OGRE_IMPORTER