Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SingleFrameAllocator and multiple world instances. #348

Open
programingman opened this issue Jul 10, 2023 · 1 comment
Open

SingleFrameAllocator and multiple world instances. #348

programingman opened this issue Jul 10, 2023 · 1 comment
Assignees

Comments

@programingman
Copy link

programingman commented Jul 10, 2023

Hi again!

For a while I've been running in to an issue with seemingly random crashes when allocating or freeing memory.
I think I've finally tracked it down to having multiple PhysicsWorld instances sharing a SingleFrameAllocator.

I know multiple worlds probably isn't a common use case, but here's what I've found, and potential solutions.

The problem is related to user code being able to cause allocations from the SingleFrameAllocator outside of PhysicsWorld::Update().
There may be multiple things that can trigger this, but most commonly, I'll use setTransform() and then testOverlap() or testCollision() and this will sometimes cause new elements to be added to mLostContactPairs via computeBroadPhase -> removeNonOverlappingPairs -> addLostContactPair.
Usually this is fine, because mLostContactPairs is processed in the next physics update before SingleFrameAllocator is cleared/reallocated.
But if another PhysicsWorld::Update() is called before then, the mLostContactPairs will be invalidated.

Here's an example timeline:

WORLD INSTANCE 1:
  -PhysicsWorld::update()
    -add lost contacts, process and clear lost contacts
    -resetFrameAllocator()
  -user code
    -setTransform(), testOverlap(), causes mLostContactPairs.add() <--------------------------------------------------------

WORLD INSTANCE 2:
  -PhysicsWorld::update()
    -add lost contacts, process and clear lost contacts
    -resetFrameAllocator() <--------------------------------------------------------
  -user code
    -move object, add lost contacts

WORLD INSTANCE 1:
  -PhysicsWorld::update()
    -lost contacts added in previous user code are now potentially being overwritten by other allocations, 
      or if the SingleFrameAllocator was resized, they will be in deallocated memory!

Potential solutions:

  1. Always call PhysicsWorld::update() immediately AFTER any user code related to that instance, rather than before which I'm doing.
  2. Have a separate SingleFrameAllocator for each PhysicsWorld instance.
  3. Add a flag to redirect SingleFrameAllocator to a different allocator when outside of PhysicWorld::update()
  4. Remove the resetFrameAllocator() from PhysicWorld::update() and call it manually after all updates are done.

No rush addressing this, I've implemented solution 3 as a workaround.
And apologies if this has been addressed already in develop branch, I haven't dug in deep enough to confirm if it's been fixed there.

Thanks again for your time and this excellent library!

@DanielChappuis
Copy link
Owner

Thanks a lot for taking the time to report this (with all the details). I don't have the time to improve this right now but I will take a look when I have some time.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

2 participants