Destruction using Blast API

Summary
For my specialization i decided to implement Destruction using Nvidia's Blast API, this decision was made as i implemented PhysX in FM HPE 102.4 and i find destruction in games to be interesting as to how it can improve the player experience.
The first step was setting up the project and building the Blast Library, after that i decided that i set up version control using GitHub.
​
After everything was set up i was able to create a wrapper for handling the creation and usage of a TKActor. This was achieved by using the High level API and authoring extensions to create a destructible asset that would successfully split once damaged.
After this i decided to create both a manager to handle all the Blast Assets and the creation of them as well as setting up a framework class to handle external functions such as creation of PhysX Actors.
In the end it was working and i was able to visually se the mesh hit the ground and break apart.
​​​​
I don´t consider the project finished as the creation of the assets as well as fracturing them currently takes a lot of time and i want to both optimize and add more features. (All Improvements)
But the current version servers a good solid base to continue with and the experience gained by planning, executing and even reading documentation has been valuable.
For more in depth info see below.
Process
The project started by reading the documentation for the API and setting up the project.
The next step was setting up a Blast Asset class that would work as the main wrapper for handling Blast.
This was a process itself as i started by defining the data used to create the assets, as well as familiarizing myself with the API and Library.
As i started to implement the wrapper i decided to use the High level TK(Toolkit) API instead of the Low Level one as it provides a better entry point to using blast, this also lead me to use the available authoring extensions to create the meshes and chunks.
​
Using a mesh cleaner object and the "NvBlastExtAuthoringCreateMesh" method i created a Blast::Mesh i could assign it to a fracture tool. The next step was to generate fracture points inside the mesh for this i used the available Voronoi site generator to generate a number of fractures contained inside the mesh.
By doing this i was able to fracture any input mesh into usable chunks, the next step was to create the bonds connecting the chunks and actually create a TKAsset .
​This is the point were i encountered some major issues with the API and i will keep it short in this segment as they are discussed in more detail in the Issues section.
​​
​The first one was that certain functions did not exist in the latest version of the API that existed in previous ones and they were still referenced in comments for functions. This was particularly in for the generation of bonds between chunks. ​
After this was successfully resolved I immediately encountered another issue that being that the function calls made through the API were not calling the correct function, this had to be resolved by recreating the function outside of the API thus giving more functionality for debugging but also more control.
Once this was finally resolved i could generate geometry for the chunks to use this then started the process of creating PhysX actors for the chunks and putting them i the scene. This was done by saving the calculated geometry for a assets chunks and passing it to create shape in PhysX and attaching the created shapes to a new actor representing the chunk.
(A little note: I discovered is that PhysX does not do well with dynamic rigid body's with multiple shapes especially if they are large)
​
Now i had something that was visible and the next step was to actually damage the original mesh and have it split into its chunks.
TKActors contain a damage function that takes a damage shader and arguments, these can be customized and you can create your own shaders as long as the function has the same signature. I decided to use the default radial shader to start with as it was the most intuitive way to think about it. As i was calling the damage function i had to define something that would listen to the callback events sent by it. I decided that to simplify the usage and debugging i would have each wrapper object listen to its own actors callbacks.
Next i implemented the necessary functions for catching the split event, all i had to do now was remove the PhysX actor of the actor that was split and add the PhysX actors corresponding to the chunks this was a relatively simple task as all the necessary data was available. A part of this was since the actor data sent for both the chunk being split and the chunks to be created contained a unique index we could easily store active PhysX actors in a unordered map to keep track of them.
At this point i added the manager to keep track of all of the Blast wrapper objects as well as to make it easier to apply damage on contact, as it contains a function with the same signature as the PhysX on contact callback, when called it checks the user data of the shape and with that being assigned to the index of the wrapper object we can now apply damage to the object.
​
At this point i had run out of time to make any meaningful progress but when conversing with my mentor i still felt that the point i got to was satisfactory ​for the issues i encountered and the timespan of the project. Being able to use it as a starting point for further improvements also contributed to this sentiment.

Fractured cube

Fracturing a mesh into chunks works then PhysX takes over
Solving issues
I encountered many problems during the early stages one of them being that a lot of the authoring functions required classes that inherited and overloaded certain functions one such being the ConvexMeshGenerator. According to the comments in the header there was a function to create a default one but when i tried it it did not exist, that is how i found out that that function was deprecated and no longer existed in the latest version yet was still referenced and required to use the tools available.
In the end i created my own by using PhysX convex mesh cooking to generate the convex collision hulls.
​
The next major hurdle was found to be the existing function for finalizing the fracturing and generating the collision hulls and meshes. The function in the API was not executing correctly i figured this out after i put breakpoints i the ConvexMeshGnerator i created the function for generating collision hulls was never called as it should instead according to the API when the function was called it was release for the object that was executed. As this was not something i knew how i could fix i decided to recreate the API code manually in my own cpp files, this was to make both debugging easier as well as giving me more control over how the process was executed. It was also a good moment for me to reflect on the progress i made and how i underestimated the kind of undertaking this task would be.
​
A general issue i had to deal with was the behavior of PhysX, the most important thing i discovered was that id really did not enjoy the center of mass and center point the shapes not being aligned with each other. As well as the behavior of Dynamic Actors with multiple shapes not being the most predictable often clipping through the ground or flying off into the distance.
Final Result
The final result at least for the moment of writing this is a cube that is fractured into pieces and breaks apart once it hits the ground. A cube is not the only available mesh as it works with all kinds of meshes but i decided on a cube for simplicity and it was also the one i found was easiest to debug.
​
Even if it is not completely finished i have a good starting point for continuing development and integration of this system.

Lessons Learned
-
I have learned quite a lot during this project, how to gather information when documentation is lacking ,how to expand on existing functionality and how to work with a API when functionality is missing.
-
I also learned a lot about the process of planning out a project from start to finish, as well as time management and how to best use the resources available to me.
-
The expectations at the planning phase might not always be the same as a attainable goal once work starts and there are a lot of unexpected things that easily get overlooked when planning a project.
-
This project probably would have been easier to implement in a already existing engine such as Unity as i had to implement certain features but i also think that it was a valuable experience nonetheless.
Future Improvements
-
Implement Serialization functionality as convex mesh decomposition is expensive and takes a long time.
-
Implement multithreading to creation of assets, decompositions of separate chunks don't depend on one another so this can be separated into threads.
-
More types of damage shaders to use, currently only radial is implemented.
-
Improve stability with how the shapes and colliders act in PhysX though this might be the hardest point.​