-
-
Notifications
You must be signed in to change notification settings - Fork 3.7k
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
Share implementation of sort methods. #16203
base: main
Are you sure you want to change the base?
Conversation
crates/bevy_ecs/src/query/iter.rs
Outdated
) | ||
}; | ||
let mut keyed_query: Vec<_> = query_lens.collect(); | ||
keyed_query.sort_by(|(key_1, _), (key_2, _)| compare(key_1, key_2)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Query::sort_unstable_by
was calling sort_by
instead of sort_unstable_by
. That seemed like a bug, so I changed it. If we don't merge this PR, we should probably make one that fixes this line.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good catch! This is a bug.
fn sort_impl<L: ReadOnlyQueryData + 'w>( | ||
self, | ||
f: impl FnOnce(&mut Vec<(L::Item<'w>, NeutralOrd<Entity>)>), | ||
) -> QuerySortedIter< | ||
'w, | ||
's, | ||
D, | ||
F, | ||
impl ExactSizeIterator<Item = Entity> + DoubleEndedIterator + FusedIterator + 'w, | ||
> { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you see a way to preserve whether Entity
needs to be wrapped in NeutralOrd
?
Since this is no longer public, no need to restrict yourself to impl Trait
, you can just use plain generics here.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It seems simpler to use it everywhere. It's a simple newtype wrapper, so it won't affect performance. And the methods that weren't wrapping the Entity
don't actually use it during sorting, so it won't affect behavior, either.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fair, it should be okay then!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It'd be nice if this could be done without introducing an additonal level of closure, maybe with generics?
Another reason for punting on this was that while these do the same now, that may change when sort caching is introduced.
This might have to look different then.
What's wrong with using a closure here? Closures are just syntactic sugar for generics, and I don't see how to make things more clear here with named types. (There's no dynamic dispatch, if that's your worry; this is |
Oh, I didn't know there were plans for sort caching! This PR certainly isn't urgent, so it can wait if that's happening soon. Depending on your plans for caching, though, this change could make it easier. I wouldn't expect the caching to vary wildly based on the type of sort, so doing this first could mean that you have to add it to one place instead of six. |
While this might generate mildly different code, the main concern here instead would be capability to preserve the information that might be useful for caching: the sort kind, and the sort function used, if there is one (maybe as a fn pointer). However, caching is not yet in the process of being implemented and is still being discussed in #13464, so how it'll look like is up in the air. |
It shouldn't. The public user facing APIs for the cached variants will only allow |
Actually, the information I am talking about can and would better be passed as an additional parameter to |
Since #13443 merged first, this should do the same for the |
Done!
Not easily, sorry. I used https://github.com/pacak/cargo-show-asm, but I had a temporary function that called the sort methods, and I didn't keep it anywhere. I have the files with the output that I used for diffing, but they're 100 KB each and I don't remember which sort method they were calling. I think a lot of the changes were just reordering blocks of instructions, but there was enough changed that I didn't try to track down exactly what was happening. I just made sure there weren't any |
fair enough! |
Objective
The various
Query::sort()
methods have a lot of duplicated code between them, including some unsafe code. Reduce the duplication to make the code easier to read and maintain.Solution
Extract the duplicated code to a private method, and pass in the sorting strategy as a closure.
Testing
I used
cargo-show-asm
to verify that the closures were inlined, but I didn't run anything through a profiler. Thesort()
method itself even had identical assembly before and after this change, although the others did not.