Box3D provides minimal base functionality for allocation hooks and vector math. The C interface allows most runtime data and types to be defined internally in the src folder.
Box3D will assert on bad input. This includes things like sending in NaN or infinity for values. It will assert if you use negative values for things that should only be positive, such as density.
Box3D will also assert if an internal bug is detected. For this reason, it is advisable to build Box3D from source. The library compiles in about a second.
You may wish to capture assertions in your application. In that case use b3SetAssertFcn(). This lets you override the debugger break and/or perform your own error handling.
Box3D uses memory efficiently and minimizes per-frame allocations by pooling memory. The engine quickly adapts to the simulation size. After the first step or two of simulation there should be no further per-frame allocations.
As bodies, shapes, and joints are created and destroyed, their memory is recycled. Internally all this data is stored in contiguous arrays. When an object is destroyed, the array element is marked empty. When an object is created it fills an empty slot via an efficient free list.
Once the internal memory pools are initially filled, the only allocations should be for sleeping islands, since their data is copied out of the main simulation. Those allocations are generally infrequent.
You can provide a custom allocator using b3SetAllocator() and query the total bytes currently allocated using b3GetByteCount().
The b3Version structure holds the current version so you can query it at run-time using b3GetVersion().
Box3D includes a vector math library covering types b3Vec3, b3Quat, b3Transform, b3Matrix3, and b3AABB. The library is designed to suit the internal needs of Box3D and its interface. All members are exposed, so you can use them freely in your application.
Three-component float vector with fields x, y, z. Useful constants and operations:
Unit quaternion representing orientation. Stored as a vector part q.v (x, y, z) and a scalar part q.s. The identity quaternion is b3Quat_identity.
Useful operations:
Because orientation in 3D is three-dimensional, there is no single scalar angle as there was in 2D. Always work with the full quaternion or the derived matrix.
A rigid transform: a position vector t.p (b3Vec3) combined with an orientation t.q (b3Quat). The identity transform is b3Transform_identity.
3×3 matrix stored as three column vectors cx, cy, cz. Primarily used for inertia tensors and rotation matrices. Useful operations include b3MulMV (matrix-vector multiply), b3MulMM (matrix-matrix multiply), b3Transpose, b3InvertMatrix, and b3MakeMatrixFromQuat.
Axis-aligned bounding box with lowerBound and upperBound as b3Vec3. Helpers include b3AABB_Overlaps, b3AABB_Contains, b3AABB_ContainsPoint, b3AABB_Union, b3AABB_Center, b3AABB_Extents, b3AABB_Inflate, and b3AABB_Transform.
Box3D has been highly optimized for multithreading. Multithreading is not required and by default Box3D runs single-threaded. If performance matters, you should consider wiring up the multithreading interface.
Box3D does not create threads itself. You supply your own thread pool and hook it up through the world definition before calling b3CreateWorld. See b3EnqueueTaskCallback, b3FinishTaskCallback, b3WorldDef::workerCount, b3WorldDef::enqueueTask, and b3WorldDef::finishTask:
Your MyEnqueueTask callback receives a b3TaskCallback* function pointer plus a context pointer that must be forwarded to the task exactly once on a worker thread. Return a non-null void* handle representing your pending task object, or null if the work was executed serially inside the callback (Box3D then skips the corresponding b3FinishTaskCallback).
MyFinishTask must block until the task has completed. Because the step blocks here on the tasks it spawned, b3World_Step holds its stack across every fork and join inside the step. This has a consequence for how you drive it. Run b3World_Step from a thread you can dedicate to the step, or from a fiber that MyFinishTask can park so the underlying thread is freed to run the queued tasks. In a job system that cannot park a job's stack, do not call b3World_Step from inside a job: a job that blocks on the sub-jobs it spawned without yielding its thread starves the workers it is waiting on and can deadlock. The alternative is help-while-wait, where the waiting thread runs other pending tasks itself. This is what the in-tree scheduler does, which is why it needs no fiber.
Box3D ships an in-tree scheduler (src/scheduler.h) used by the test suite and samples. You can adapt it or use it directly:
The multithreading design for Box3D is focused on data parallelism. The goal is to use multiple cores to finish the world simulation as fast as possible. Box3D multithreading is not designed for task parallelism. Often in games you have a render thread or an audio thread doing work in isolation from the main thread. Those are examples of task parallelism.
So when you design your game loop, you should let Box3D go wide and use multiple cores to finish its work quickly, without other threads interacting with the Box3D world at the same time.
In a multithreaded environment you must be careful to avoid race conditions. Modifying the world while it is simulating will lead to unpredictable behavior and is never safe. It is also not safe to read data from a Box3D world while it is simulating; Box3D may move data structures to improve cache performance, so you could easily read garbage.
Caution: Do not perform read or write operations on a Box3D world during b3World_Step()
Caution: Do not write to the Box3D world from multiple threads
It is safe to do ray-casts, shape-casts, and overlap tests from multiple threads outside of b3World_Step(). Generally any read-only operation is safe to do multithreaded outside of b3World_Step(). This can be very useful if you have multithreaded game logic.
Some applications may wish to create multiple Box3D worlds and simulate them on different threads. This works fine because Box3D has very limited use of globals.
There are a few caveats: