-
Notifications
You must be signed in to change notification settings - Fork 481
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
Don't include raiding units in Units::isActive; make getPosition return invalid coords #4959
base: develop
Are you sure you want to change the base?
Conversation
Don't follow units that are out on raid.
return false; | ||
else if (unit->flags1.bits.move_state || unit->flags1.bits.can_swap) | ||
return true; // These are always unset when leaving the map | ||
return linear_index(world->units.active, &df::unit::id, unit->id) >= 0; |
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.
Units::isActive
is used in many many places. I'm worried that adding an O(N) search for every call will be a performance killer. I'm also concerned that changing the behavior of this function will break existing callers, both Lua and C++. I'd like to see some research done on those two points before merging this PR.
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.
I did a check on my fort, and all of my units in units.active
had at least one of inactive
, move_state
, or can_swap
set. There was an inactive
unit without the other two flags set that had its pos
directly above my well. It's possible the combination could represent a ghost or climbing unit? Probably not going to be a performance consideration given how infrequently those occur. Then there's the actual raiding units, which you won't hit while iterating units.active
, and for which the check is mandatory when iterating units.all
.
I went through all the uses of isActive
, getPosition
, getUnitsInBox
, and isUnitInBox
I could find, as well as the inactive
flag. There's no change in behavior to anything iterating the units.active
vector (or indirectly through forCitizens
/getCitizens
). isDead
is used instead of isActive
for the case of actual dead units. Anything iterating units.all
wasn't properly considering raiding units, which was just RemoteTools.cpp
/spectate
.
I don't fully understand plugins/preserve-rooms.cpp
but it didn't look like the changes break anything. You'd know better than me.
for more information, see https://pre-commit.ci
* Update Units.cpp * Update Items.h
if (Units::isActive(unit) && !Units::isDead(unit) && active_unit_ids.contains(unit->id)) { | ||
if (!Units::isDead(unit) && Units::isActive(unit) && active_unit_ids.contains(unit->id)) { |
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.
Moved isActive
after isDead
, which is a simple flag check. Not sure how isActive
compares to testing an unordered set. The latter is probably much faster for units that are actually raiding? In that case we should test isActive
afterwards.
Or maybe switch to directly testing the inactive
flag (since active_unit_ids
already handles the raiding units). Just curious how it performs first, though, as a stress test.
Raiding units are unflagged as
inactive
shortly after they leave the map, as they are removed fromunits.active
. Scan this vector to rule out raiding units. Themove_state
andcan_swap
flags are also unset, which can be used to optimize the check.Units::getPosition
now returns invalid coords for inactive units. This impactsUnits::isUnitInBox
, andUnits::getUnitsInBox
.It also impacts
Items::getPosition
, but an item initems.other.IN_PLAY
has its holder is guaranteed to be active (artifact theft bugs notwithstanding).Only iterate active units in
spectate
instead of all. (Mainly for efficiency.Units::isActive
change would've caught any potential issue with raiding units.)