ModelNode.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. // Copyright (C) 2009-2020, Panagiotis Christopoulos Charitos and contributors.
  2. // All rights reserved.
  3. // Code licensed under the BSD License.
  4. // http://www.anki3d.org/LICENSE
  5. #include <anki/scene/ModelNode.h>
  6. #include <anki/scene/SceneGraph.h>
  7. #include <anki/scene/DebugDrawer.h>
  8. #include <anki/scene/components/BodyComponent.h>
  9. #include <anki/scene/components/SkinComponent.h>
  10. #include <anki/scene/components/RenderComponent.h>
  11. #include <anki/resource/ModelResource.h>
  12. #include <anki/resource/ResourceManager.h>
  13. #include <anki/resource/SkeletonResource.h>
  14. #include <anki/physics/PhysicsWorld.h>
  15. namespace anki
  16. {
  17. /// Feedback component.
  18. class ModelNode::MoveFeedbackComponent : public SceneComponent
  19. {
  20. public:
  21. MoveFeedbackComponent()
  22. : SceneComponent(SceneComponentType::NONE)
  23. {
  24. }
  25. ANKI_USE_RESULT Error update(SceneNode& node, Second prevTime, Second crntTime, Bool& updated) override
  26. {
  27. updated = false;
  28. const MoveComponent& move = node.getFirstComponentOfType<MoveComponent>();
  29. const SkinComponent* skin = node.tryGetFirstComponentOfType<SkinComponent>();
  30. if(move.getTimestamp() == node.getGlobalTimestamp()
  31. || (skin && skin->getTimestamp() == node.getGlobalTimestamp()))
  32. {
  33. ModelNode& mnode = static_cast<ModelNode&>(node);
  34. mnode.updateSpatialComponent(move);
  35. }
  36. return Error::NONE;
  37. }
  38. };
  39. /// Feedback component.
  40. class ModelNode::SkinFeedbackComponent : public SceneComponent
  41. {
  42. public:
  43. SkinFeedbackComponent()
  44. : SceneComponent(SceneComponentType::NONE)
  45. {
  46. }
  47. ANKI_USE_RESULT Error update(SceneNode& node, Second prevTime, Second crntTime, Bool& updated) override
  48. {
  49. updated = false;
  50. const SkinComponent& skin = node.getFirstComponentOfType<SkinComponent>();
  51. if(skin.getTimestamp() == node.getGlobalTimestamp())
  52. {
  53. ModelNode& mnode = static_cast<ModelNode&>(node);
  54. const Aabb& box = skin.getBoneBoundingVolume();
  55. mnode.m_obbLocal.setCenter((box.getMin() + box.getMax()) / 2.0f);
  56. mnode.m_obbLocal.setExtend(box.getMax() - mnode.m_obbLocal.getCenter());
  57. mnode.m_obbLocal.setRotation(Mat3x4::getIdentity());
  58. }
  59. return Error::NONE;
  60. }
  61. };
  62. ModelNode::ModelNode(SceneGraph* scene, CString name)
  63. : SceneNode(scene, name)
  64. {
  65. }
  66. ModelNode::~ModelNode()
  67. {
  68. }
  69. Error ModelNode::init(ModelResourcePtr resource, U32 modelPatchIdx)
  70. {
  71. ANKI_ASSERT(modelPatchIdx < resource->getModelPatches().getSize());
  72. ANKI_CHECK(m_dbgDrawer.init(&getResourceManager()));
  73. m_model = resource;
  74. m_modelPatchIdx = modelPatchIdx;
  75. // Merge key
  76. Array<U64, 2> toHash;
  77. toHash[0] = modelPatchIdx;
  78. toHash[1] = resource->getUuid();
  79. m_mergeKey = computeHash(&toHash[0], sizeof(toHash));
  80. // Components
  81. if(m_model->getSkeleton().isCreated())
  82. {
  83. newComponent<SkinComponent>(this, m_model->getSkeleton());
  84. newComponent<SkinFeedbackComponent>();
  85. }
  86. newComponent<MoveComponent>();
  87. newComponent<MoveFeedbackComponent>();
  88. newComponent<SpatialComponent>(this, &m_obbWorld);
  89. RenderComponent* rcomp = newComponent<RenderComponent>();
  90. rcomp->initRaster(
  91. [](RenderQueueDrawContext& ctx, ConstWeakArray<void*> userData) {
  92. const ModelNode& self = *static_cast<const ModelNode*>(userData[0]);
  93. self.draw(ctx, userData);
  94. },
  95. this, m_mergeKey);
  96. rcomp->setFlagsFromMaterial(m_model->getModelPatches()[m_modelPatchIdx].getMaterial());
  97. if(m_model->getModelPatches()[m_modelPatchIdx].getSupportedRayTracingTypes() != RayTypeBit::NONE)
  98. {
  99. rcomp->initRayTracing(setupRayTracingInstanceQueueElement, this);
  100. }
  101. m_obbLocal = m_model->getModelPatches()[m_modelPatchIdx].getBoundingShape();
  102. return Error::NONE;
  103. }
  104. Error ModelNode::init(const CString& modelFname)
  105. {
  106. ModelResourcePtr model;
  107. ANKI_CHECK(getResourceManager().loadResource(modelFname, model));
  108. // Init this
  109. ANKI_CHECK(init(model, 0));
  110. // Create separate nodes for the model patches and make the children
  111. for(U32 i = 1; i < model->getModelPatches().getSize(); ++i)
  112. {
  113. ModelNode* other;
  114. ANKI_CHECK(getSceneGraph().newSceneNode(CString(), other, model, i));
  115. addChild(other);
  116. }
  117. return Error::NONE;
  118. }
  119. void ModelNode::updateSpatialComponent(const MoveComponent& move)
  120. {
  121. m_obbWorld = m_obbLocal.getTransformed(move.getWorldTransform());
  122. SpatialComponent& sp = getFirstComponentOfType<SpatialComponent>();
  123. sp.markForUpdate();
  124. sp.setSpatialOrigin(move.getWorldTransform().getOrigin());
  125. }
  126. void ModelNode::draw(RenderQueueDrawContext& ctx, ConstWeakArray<void*> userData) const
  127. {
  128. ANKI_ASSERT(userData.getSize() > 0 && userData.getSize() <= MAX_INSTANCES);
  129. ANKI_ASSERT(ctx.m_key.getInstanceCount() == userData.getSize());
  130. CommandBufferPtr& cmdb = ctx.m_commandBuffer;
  131. if(ANKI_LIKELY(!ctx.m_debugDraw))
  132. {
  133. const ModelPatch& patch = m_model->getModelPatches()[m_modelPatchIdx];
  134. // That will not work on multi-draw and instanced at the same time. Make sure that there is no multi-draw
  135. // anywhere
  136. ANKI_ASSERT(patch.getSubMeshCount() == 1);
  137. // Transforms
  138. Array<Mat4, MAX_INSTANCES> trfs;
  139. Array<Mat4, MAX_INSTANCES> prevTrfs;
  140. const MoveComponent& movec = getFirstComponentOfType<MoveComponent>();
  141. trfs[0] = Mat4(movec.getWorldTransform());
  142. prevTrfs[0] = Mat4(movec.getPreviousWorldTransform());
  143. Bool moved = trfs[0] != prevTrfs[0];
  144. for(U32 i = 1; i < userData.getSize(); ++i)
  145. {
  146. const ModelNode& self2 = *static_cast<const ModelNode*>(userData[i]);
  147. const MoveComponent& movec = self2.getFirstComponentOfType<MoveComponent>();
  148. trfs[i] = Mat4(movec.getWorldTransform());
  149. prevTrfs[i] = Mat4(movec.getPreviousWorldTransform());
  150. moved = moved || (trfs[i] != prevTrfs[i]);
  151. }
  152. ctx.m_key.setVelocity(moved && ctx.m_key.getPass() == Pass::GB);
  153. ModelRenderingInfo modelInf;
  154. patch.getRenderingInfo(ctx.m_key, WeakArray<U8>(), modelInf);
  155. // Bones storage
  156. if(m_model->getSkeleton())
  157. {
  158. const SkinComponent& skinc = getComponentAt<SkinComponent>(0);
  159. const U32 boneCount = skinc.getBoneTransforms().getSize();
  160. StagingGpuMemoryToken token, tokenPrev;
  161. void* trfs = ctx.m_stagingGpuAllocator->allocateFrame(boneCount * sizeof(Mat4),
  162. StagingGpuMemoryType::STORAGE, token);
  163. memcpy(trfs, &skinc.getBoneTransforms()[0], boneCount * sizeof(Mat4));
  164. trfs = ctx.m_stagingGpuAllocator->allocateFrame(boneCount * sizeof(Mat4), StagingGpuMemoryType::STORAGE,
  165. tokenPrev);
  166. memcpy(trfs, &skinc.getPreviousFrameBoneTransforms()[0], boneCount * sizeof(Mat4));
  167. ANKI_ASSERT(modelInf.m_boneTransformsBinding < MAX_U32);
  168. cmdb->bindStorageBuffer(patch.getMaterial()->getDescriptorSetIndex(), modelInf.m_boneTransformsBinding,
  169. token.m_buffer, token.m_offset, token.m_range);
  170. ANKI_ASSERT(modelInf.m_prevFrameBoneTransformsBinding < MAX_U32);
  171. cmdb->bindStorageBuffer(patch.getMaterial()->getDescriptorSetIndex(),
  172. modelInf.m_prevFrameBoneTransformsBinding, tokenPrev.m_buffer, tokenPrev.m_offset,
  173. tokenPrev.m_range);
  174. }
  175. // Program
  176. cmdb->bindShaderProgram(modelInf.m_program);
  177. // Uniforms
  178. RenderComponent::allocateAndSetupUniforms(m_model->getModelPatches()[m_modelPatchIdx].getMaterial(), ctx,
  179. ConstWeakArray<Mat4>(&trfs[0], userData.getSize()),
  180. ConstWeakArray<Mat4>(&prevTrfs[0], userData.getSize()),
  181. *ctx.m_stagingGpuAllocator);
  182. // Set attributes
  183. for(U i = 0; i < modelInf.m_vertexAttributeCount; ++i)
  184. {
  185. const VertexAttributeInfo& attrib = modelInf.m_vertexAttributes[i];
  186. ANKI_ASSERT(attrib.m_format != Format::NONE);
  187. cmdb->setVertexAttribute(U32(attrib.m_location), attrib.m_bufferBinding, attrib.m_format,
  188. attrib.m_relativeOffset);
  189. }
  190. // Set vertex buffers
  191. for(U32 i = 0; i < modelInf.m_vertexBufferBindingCount; ++i)
  192. {
  193. const VertexBufferBinding& binding = modelInf.m_vertexBufferBindings[i];
  194. cmdb->bindVertexBuffer(i, binding.m_buffer, binding.m_offset, binding.m_stride, VertexStepRate::VERTEX);
  195. }
  196. // Index buffer
  197. cmdb->bindIndexBuffer(modelInf.m_indexBuffer, 0, IndexType::U16);
  198. // Draw
  199. cmdb->drawElements(PrimitiveTopology::TRIANGLES, modelInf.m_indicesCountArray[0], userData.getSize(),
  200. U32(modelInf.m_indicesOffsetArray[0] / sizeof(U16)), 0, 0);
  201. }
  202. else
  203. {
  204. // Draw the bounding volumes
  205. Mat4* const mvps = ctx.m_frameAllocator.newArray<Mat4>(userData.getSize());
  206. for(U32 i = 0; i < userData.getSize(); ++i)
  207. {
  208. const ModelNode& self2 = *static_cast<const ModelNode*>(userData[i]);
  209. const Mat3 rot = self2.m_obbWorld.getRotation().getRotationPart();
  210. const Vec4 tsl = self2.m_obbWorld.getCenter().xyz1();
  211. const Vec3 scale = self2.m_obbWorld.getExtend().xyz();
  212. // Set non uniform scale. Add a margin to avoid flickering
  213. Mat3 nonUniScale = Mat3::getZero();
  214. constexpr F32 MARGIN = 1.02f;
  215. nonUniScale(0, 0) = scale.x() * MARGIN;
  216. nonUniScale(1, 1) = scale.y() * MARGIN;
  217. nonUniScale(2, 2) = scale.z() * MARGIN;
  218. mvps[i] = ctx.m_viewProjectionMatrix * Mat4(tsl, rot * nonUniScale, 1.0f);
  219. }
  220. const Bool enableDepthTest = ctx.m_debugDrawFlags.get(RenderQueueDebugDrawFlag::DEPTH_TEST_ON);
  221. if(enableDepthTest)
  222. {
  223. cmdb->setDepthCompareOperation(CompareOperation::LESS);
  224. }
  225. else
  226. {
  227. cmdb->setDepthCompareOperation(CompareOperation::ALWAYS);
  228. }
  229. m_dbgDrawer.drawCubes(ConstWeakArray<Mat4>(mvps, userData.getSize()), Vec4(1.0f, 0.0f, 1.0f, 1.0f), 2.0f,
  230. ctx.m_debugDrawFlags.get(RenderQueueDebugDrawFlag::DITHERED_DEPTH_TEST_ON), 2.0f,
  231. *ctx.m_stagingGpuAllocator, cmdb);
  232. ctx.m_frameAllocator.deleteArray(mvps, userData.getSize());
  233. // Bones
  234. if(m_model->getSkeleton())
  235. {
  236. const SkinComponent& skinc = getComponentAt<SkinComponent>(0);
  237. SkeletonResourcePtr skeleton = skinc.getSkeleronResource();
  238. const U32 boneCount = skinc.getBoneTransforms().getSize();
  239. DynamicArrayAuto<Vec3> lines(ctx.m_frameAllocator);
  240. lines.resizeStorage(boneCount * 2);
  241. DynamicArrayAuto<Vec3> chidlessLines(ctx.m_frameAllocator);
  242. for(U32 i = 0; i < boneCount; ++i)
  243. {
  244. const Bone& bone = skeleton->getBones()[i];
  245. ANKI_ASSERT(bone.getIndex() == i);
  246. const Vec4 point(0.0f, 0.0f, 0.0f, 1.0f);
  247. const Bone* parent = bone.getParent();
  248. Mat4 m = (parent)
  249. ? skinc.getBoneTransforms()[parent->getIndex()] * parent->getVertexTransform().getInverse()
  250. : Mat4::getIdentity();
  251. const Vec3 a = (m * point).xyz();
  252. m = skinc.getBoneTransforms()[i] * bone.getVertexTransform().getInverse();
  253. const Vec3 b = (m * point).xyz();
  254. lines.emplaceBack(a);
  255. lines.emplaceBack(b);
  256. if(bone.getChildren().getSize() == 0)
  257. {
  258. // If there are not children try to draw something for that bone as well
  259. chidlessLines.emplaceBack(b);
  260. const F32 len = (b - a).getLength();
  261. const Vec3 c = b + (b - a).getNormalized() * len;
  262. chidlessLines.emplaceBack(c);
  263. }
  264. }
  265. const Mat4 mvp =
  266. ctx.m_viewProjectionMatrix * Mat4(getFirstComponentOfType<MoveComponent>().getWorldTransform());
  267. m_dbgDrawer.drawLines(ConstWeakArray<Mat4>(&mvp, 1), Vec4(1.0f), 20.0f,
  268. ctx.m_debugDrawFlags.get(RenderQueueDebugDrawFlag::DITHERED_DEPTH_TEST_ON), lines,
  269. *ctx.m_stagingGpuAllocator, cmdb);
  270. m_dbgDrawer.drawLines(ConstWeakArray<Mat4>(&mvp, 1), Vec4(0.7f, 0.7f, 0.7f, 1.0f), 5.0f,
  271. ctx.m_debugDrawFlags.get(RenderQueueDebugDrawFlag::DITHERED_DEPTH_TEST_ON),
  272. chidlessLines, *ctx.m_stagingGpuAllocator, cmdb);
  273. }
  274. // Restore state
  275. if(!enableDepthTest)
  276. {
  277. cmdb->setDepthCompareOperation(CompareOperation::LESS);
  278. }
  279. }
  280. }
  281. void ModelNode::setupRayTracingInstanceQueueElement(U32 lod, const void* userData, RayTracingInstanceQueueElement& el)
  282. {
  283. const ModelNode& self = *static_cast<const ModelNode*>(userData);
  284. const ModelPatch& patch = self.m_model->getModelPatches()[self.m_modelPatchIdx];
  285. ModelRayTracingInfo info;
  286. patch.getRayTracingInfo(lod, info);
  287. memset(&el, 0, sizeof(el));
  288. // AS
  289. el.m_bottomLevelAccelerationStructure = info.m_bottomLevelAccelerationStructure.get();
  290. info.m_bottomLevelAccelerationStructure->getRefcount().fetchAdd(1);
  291. // Set the descriptor
  292. el.m_modelDescriptor = info.m_descriptor;
  293. const MoveComponent& movec = self.getFirstComponentOfType<MoveComponent>();
  294. const Mat3x4 worldTrf(movec.getWorldTransform());
  295. memcpy(&el.m_modelDescriptor.m_worldTransform, &worldTrf, sizeof(worldTrf));
  296. el.m_modelDescriptor.m_worldRotation = movec.getWorldTransform().getRotation().getRotationPart();
  297. // Handles
  298. for(RayType type : EnumIterable<RayType>())
  299. {
  300. if(!(patch.getMaterial()->getSupportedRayTracingTypes() & RayTypeBit(1 << type)))
  301. {
  302. continue;
  303. }
  304. el.m_shaderGroupHandles[type] = &info.m_shaderGroupHandles[type][0];
  305. }
  306. // References
  307. ANKI_ASSERT(info.m_grObjectReferenceCount <= el.m_grObjects.getSize());
  308. el.m_grObjectCount = info.m_grObjectReferenceCount;
  309. for(U32 i = 0; i < info.m_grObjectReferenceCount; ++i)
  310. {
  311. el.m_grObjects[i] = info.m_grObjectReferences[i].get();
  312. el.m_grObjects[i]->getRefcount().fetchAdd(1);
  313. }
  314. }
  315. } // end namespace anki