From b5ea42c39e2bdbe8ffbc9d85de4e1b8180aa92ed Mon Sep 17 00:00:00 2001 From: Marko Lindqvist Date: Thu, 8 Apr 2021 18:00:49 +0300 Subject: [PATCH 04/18] Wait until city is in sane state before sending info packets Don't send city info packets to client in the middle of the city processing when city might be in inconsistent state. See osdn #41851 Signed-off-by: Marko Lindqvist --- common/city.h | 8 +++++++- server/citytools.c | 30 +++++++++++++++++++++++++----- server/citytools.h | 1 + server/cityturn.c | 15 ++++++++++++--- server/sanitycheck.c | 2 +- 5 files changed, 46 insertions(+), 10 deletions(-) diff --git a/common/city.h b/common/city.h index b26b581e8a..dd60110418 100644 --- a/common/city.h +++ b/common/city.h @@ -298,6 +298,12 @@ enum city_build_result { CB_NO_MIN_DIST }; +enum city_needs_arrange { + CNA_NOT = 0, + CNA_NORMAL, + CNA_BROADCAST_PENDING +}; + struct tile_cache; /* defined and only used within city.c */ struct adv_city; /* defined in ./server/advisors/infracache.h */ @@ -404,7 +410,7 @@ struct city { /* If set, workers need to be arranged when the city is unfrozen. * Set inside auto_arrange_workers() and city_freeze_workers_queue(). */ - bool needs_arrange; + enum city_needs_arrange needs_arrange; /* If set, city needs to be refreshed at a later time. * Set inside city_refresh() and city_refresh_queue_add(). */ diff --git a/server/citytools.c b/server/citytools.c index d71be66aa6..f3e11fdedb 100644 --- a/server/citytools.c +++ b/server/citytools.c @@ -141,7 +141,8 @@ void city_thaw_workers(struct city *pcity) { pcity->server.workers_frozen--; fc_assert(pcity->server.workers_frozen >= 0); - if (pcity->server.workers_frozen == 0 && pcity->server.needs_arrange) { + if (pcity->server.workers_frozen == 0 + && pcity->server.needs_arrange != CNA_NOT) { city_refresh(pcity); /* Citizen count sanity */ auto_arrange_workers(pcity); } @@ -160,7 +161,9 @@ void city_freeze_workers_queue(struct city *pcity) city_list_prepend(arrange_workers_queue, pcity); city_freeze_workers(pcity); - pcity->server.needs_arrange = TRUE; + if (pcity->server.needs_arrange == CNA_NOT) { + pcity->server.needs_arrange = CNA_NORMAL; + } } /**************************************************************************** @@ -2079,14 +2082,23 @@ void refresh_dumb_city(struct city *pcity) (Split off from send_city_info_at_tile() because that was getting too difficult for me to understand... --dwp) **************************************************************************/ -static void broadcast_city_info(struct city *pcity) +void broadcast_city_info(struct city *pcity) { struct packet_city_info packet; struct packet_city_short_info sc_pack; struct player *powner = city_owner(pcity); /* Send to everyone who can see the city. */ - package_city(pcity, &packet, FALSE); + if (pcity->server.needs_arrange == CNA_NOT) { + /* Send city only when it's in sane state. + * If it's not, it will be sent to everyone once + * rearrangement has been finished. */ + package_city(pcity, &packet, FALSE); + } else { + pcity->server.needs_arrange = CNA_BROADCAST_PENDING; + + return; + } players_iterate(pplayer) { if (can_player_see_city_internals(pplayer, pcity)) { if (!send_city_suppressed || pplayer != powner) { @@ -2217,6 +2229,11 @@ void send_city_info_at_tile(struct player *pviewer, struct conn_list *dest, if (!pcity) { pcity = tile_city(ptile); } + if (pcity->server.needs_arrange != CNA_NOT) { + pcity->server.needs_arrange = CNA_BROADCAST_PENDING; + + return; + } if (pcity) { powner = city_owner(pcity); } @@ -2224,7 +2241,10 @@ void send_city_info_at_tile(struct player *pviewer, struct conn_list *dest, /* send info to owner */ /* This case implies powner non-NULL which means pcity non-NULL */ if (!send_city_suppressed) { - /* send all info to the owner */ + /* Send all info to the owner. + * Wait that city has been rearranged, if it's currently + * not in sane state. + */ update_dumb_city(powner, pcity); package_city(pcity, &packet, FALSE); lsend_packet_city_info(dest, &packet, FALSE); diff --git a/server/citytools.h b/server/citytools.h index d7a13fb38e..7fa0af9dc8 100644 --- a/server/citytools.h +++ b/server/citytools.h @@ -43,6 +43,7 @@ void send_city_info_at_tile(struct player *pviewer, struct conn_list *dest, struct city *pcity, struct tile *ptile); void send_all_known_cities(struct conn_list *dest); void send_player_cities(struct player *pplayer); +void broadcast_city_info(struct city *pcity); void package_city(struct city *pcity, struct packet_city_info *packet, bool dipl_invest); diff --git a/server/cityturn.c b/server/cityturn.c index 161e8362ed..81e0d22bf0 100644 --- a/server/cityturn.c +++ b/server/cityturn.c @@ -301,24 +301,29 @@ void auto_arrange_workers(struct city *pcity) { struct cm_parameter cmp; struct cm_result *cmr; + bool broadcast_needed; /* See comment in freeze_workers(): we can't rearrange while * workers are frozen (i.e. multiple updates need to be done). */ if (pcity->server.workers_frozen > 0) { - pcity->server.needs_arrange = TRUE; + if (pcity->server.needs_arrange == CNA_NOT) { + pcity->server.needs_arrange = CNA_NORMAL; + } return; } TIMING_LOG(AIT_CITIZEN_ARRANGE, TIMER_START); + broadcast_needed = (pcity->server.needs_arrange == CNA_BROADCAST_PENDING); + /* Freeze the workers and make sure all the tiles around the city * are up to date. Then thaw, but hackishly make sure that thaw * doesn't call us recursively, which would waste time. */ city_freeze_workers(pcity); - pcity->server.needs_arrange = FALSE; + pcity->server.needs_arrange = CNA_NOT; city_map_update_all(pcity); - pcity->server.needs_arrange = FALSE; + pcity->server.needs_arrange = CNA_NOT; city_thaw_workers(pcity); /* Now start actually rearranging. */ @@ -416,6 +421,10 @@ void auto_arrange_workers(struct city *pcity) } sanity_check_city(pcity); + if (broadcast_needed) { + broadcast_city_info(pcity); + } + cm_result_destroy(cmr); TIMING_LOG(AIT_CITIZEN_ARRANGE, TIMER_STOP); } diff --git a/server/sanitycheck.c b/server/sanitycheck.c index 467e999123..43737b8faf 100644 --- a/server/sanitycheck.c +++ b/server/sanitycheck.c @@ -336,7 +336,7 @@ static void check_city_feelings(const struct city *pcity, const char *file, * savegame despite the fact that city workers_frozen level of the city * is above zero -> can't limit sanitycheck callpoints by that. Instead * we check even more relevant needs_arrange. */ - SANITY_CITY(pcity, !pcity->server.needs_arrange); + SANITY_CITY(pcity, pcity->server.needs_arrange == CNA_NOT); SANITY_CITY(pcity, city_size_get(pcity) - spe == sum); } -- 2.30.2