-
-
Notifications
You must be signed in to change notification settings - Fork 21.5k
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
Optimized support function for large meshes #64382
Conversation
Please squash the commits as required by our pipeline. |
This is very interesting. I was planning to do a different type of optimization for these kind of shapes which should hopefully be faster as its O(1). My idea is to use octahedral encoding for all possible directions. This way, a support function maps a normalized 2D unit vector to a 2D position in a 2D texture, then it only needs to check the 4 surrounding points found and one of them is going to be the correct result. There are some quirks in the implementation that are not so obvious if you are not familiar with type of encoding that I would need to look up if anyone is interested in doing. |
In any case, if you are interested in physics engine discussion, I would be happy to invite you to the #physics channel of our rocketchat instance. https://chat.godotengine.org |
Is your idea to encode the support function as an image? I think that's equivalent to replacing the high resolution mesh with a simplified one whose vertices correspond to the pixels in the image? Or you could think of it as applying a lowpass filter to the support function. It should work well for smooth meshes, less well for ones with fine details. Think of a long thin object like a spear. If the tip of the spear doesn't exactly line up with a pixel, it will get shortened. Another possibility is to combine the two approaches. Use the image to get the initial guess for the vertex, then follow with the hill climbing stage to find the exact support point. That would trade off more work at object creation time (depending on the size of the image) to reduce the number of iterations of hill climbing when a collision happens.
That would be great! |
For #63702 I will most likely do a combination of Gauss map discarding of edge pairs and GJKEpa for convexes with too many vertices (which is already integrated in the codebase and works well) instead of MPR (because I fear based on previous experience and experience with Bullet that the fact its approximate collision may introduce problems in some corner cases). But this one with the changes noted I think should be fine and is likely simpler/better than what I wanted to do. |
I went ahead and fixed the above in the more verbose way. Let me know if it looks ok now. |
CI is failing due to sign-compare warnings raised by both GCC and Clang:
Please also squash the commits and this should be ready to merge. |
I did a basic test, adding convex shapes (low and high vertices) to a scene and checking the FPS, but I don't see any improvement. Could you please share your test project? |
A test scene is attached. It involves a collision between two object that each have a few hundred vertices. It's unplayably slow either with or without this change, but much faster with the change than without. Because of the way the SAT code works (it calls |
I ran the workflows to see if it passes CI but please note that this still requires to be squashed before it can be merged. |
There was one more signed/unsigned comparison that needed fixing first. I just made the change. I can squash it once all tests are passing. Github has a really useful feature that can do that automatically. In the repository settings, go to the General tab and make sure "Allow squash merging" is checking. Once it is, you can click on the arrow next to the "Merge" button for a PR and select "Squash and merge". A benefit of doing it that way is that it generates a name for the squash commit that contains the name and number of the PR. When you look through the commit history, every commit becomes a hyperlink to the corresponding PR. Just something to consider for the future. |
We don't use the squash-and-merge feature GitHub provides for various reasons. We prefer that PRs are squashed before merging so that the history we merge matches the one in the PR. |
Ok, no problem. I'll squash once we get a clean build. Is there a flag I can pass to scons so it will report warnings when building locally? |
Yes, you can use |
Thanks! No problems reported when I build with that option, so hopefully CI will be happy this time. |
I guess cl is a bit more picky than gcc... |
All tests passed, and I squashed the commits. It should be ready to merge. |
Thanks and congratulations on your first merged contribution! I'm really forward to the physics optimizations you have in mind |
This is the second piece of addressing #48587.
get_support()
andproject_range()
are both currently implemented with full scans over all vertices. For a large mesh, that's very slow. I used the approach suggested in https://graphics.stanford.edu/courses/cs468-01-fall/Papers/cameron.pdf, which is to search along the surface, moving between adjacent vertices to increase the support value.To make this fast, we want a good starting point to search from. I precompute the extreme vertices along 26 directions. That gives us a short list of vertices to start with, one of which should be close to the true support point.
To benchmark it, I measured the time for 1 million calls to
get_support()
along random directions. I did that for meshes of varying size with the current implementation and with this PR. Here are the times in seconds for 1 million calls, or equivalently the average time in microseconds for a single call.When combined with #63702, the total cost of collision detection scales very weakly with the number of vertices, roughly as O(sqrt(N)) in the limit of very large meshes. For two meshes that each have several thousand vertices, the total time to compute a collision is measured in microseconds.