summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/station_cmd.cpp34
1 files changed, 20 insertions, 14 deletions
diff --git a/src/station_cmd.cpp b/src/station_cmd.cpp
index ae2ba9588..44cf45f5a 100644
--- a/src/station_cmd.cpp
+++ b/src/station_cmd.cpp
@@ -4101,27 +4101,33 @@ void FlowStat::ChangeShare(StationID st, int flow)
* be empty. In that case the whole flow stat must be deleted then. */
assert(!this->shares.empty());
- int32 added_shares = 0;
- uint32 last_share = 0;
+ uint removed_shares = 0;
+ uint added_shares = 0;
+ uint last_share = 0;
SharesMap new_shares;
for (SharesMap::iterator it(this->shares.begin()); it != this->shares.end(); ++it) {
if (it->second == st) {
- uint share = it->first - last_share;
- int change = flow - added_shares;
- uint new_share = (change < 0 && (uint)(-change) > share) ? 0 : share + change;
- added_shares -= share;
- added_shares += new_share;
- if (new_share > 0) {
- new_shares[it->first + added_shares] = it->second;
+ if (flow < 0) {
+ uint share = it->first - last_share;
+ if (flow == INT_MIN || (uint)(-flow) >= share) {
+ removed_shares += share;
+ if (flow != INT_MIN) flow += share;
+ last_share = it->first;
+ continue; // remove the whole share
+ }
+ removed_shares += (uint)(-flow);
+ } else {
+ added_shares += (uint)(flow);
}
- } else {
- new_shares[it->first + added_shares] = it->second;
+
+ /* If we don't continue above the whole flow has been added or
+ * removed. */
+ flow = 0;
}
+ new_shares[it->first + added_shares - removed_shares] = it->second;
last_share = it->first;
}
- if (flow > added_shares) {
- new_shares[last_share + flow - added_shares] = st;
- }
+ if (flow > 0) new_shares[last_share + (uint)flow] = st;
this->shares.swap(new_shares);
}