From 9dd29b53b9a9fddc062de9e45616a87f61ef9b6a Mon Sep 17 00:00:00 2001 From: Marko Lindqvist Date: Sun, 30 Oct 2022 06:18:23 +0200 Subject: [PATCH 48/48] AI: Do not count defense value for a police when there's no danger Danger was considered to be high when there was none, but defensive unit was considered just for Martial Law effect. See osdn #45595 Signed-off-by: Marko Lindqvist --- ai/default/daicity.c | 3 +- ai/default/daimilitary.c | 165 +++++++++++++++++++++++---------------- ai/default/daimilitary.h | 2 +- 3 files changed, 100 insertions(+), 70 deletions(-) diff --git a/ai/default/daicity.c b/ai/default/daicity.c index 8cd96e54b1..7d6cf94b8f 100644 --- a/ai/default/daicity.c +++ b/ai/default/daicity.c @@ -291,7 +291,8 @@ static void dai_city_choose_build(struct ai_type *ait, struct player *pplayer, unsigned int our_def = assess_defense_quadratic(ait, pcity); if (our_def == 0 - && dai_process_defender_want(ait, pplayer, pcity, 1, &(city_data->choice))) { + && dai_process_defender_want(ait, pplayer, pcity, 1, + &(city_data->choice), 0)) { adv_choice_set_use(&(city_data->choice), "fallback defender"); CITY_LOG(LOG_DEBUG, pcity, "Building fallback defender"); } else if (best_role_unit(pcity, UTYF_SETTLERS)) { diff --git a/ai/default/daimilitary.c b/ai/default/daimilitary.c index ec286e0878..b6b81b459b 100644 --- a/ai/default/daimilitary.c +++ b/ai/default/daimilitary.c @@ -822,7 +822,7 @@ int dai_unit_attack_desirability(struct ai_type *ait, **************************************************************************/ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, struct city *pcity, unsigned int danger, - struct adv_choice *choice) + struct adv_choice *choice, adv_want extra_want) { const struct research *presearch = research_get(pplayer); /* FIXME: We check if the city has *some* defensive structure, @@ -838,6 +838,7 @@ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, int best_unit_cost = 1; struct ai_city *city_data = def_ai_city_data(pcity, ait); struct ai_plr *plr_data = def_ai_player_data(pplayer, ait); + adv_want total_want = danger + extra_want; memset(tech_desire, 0, sizeof(tech_desire)); @@ -854,62 +855,70 @@ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, desire = dai_unit_defence_desirability(ait, punittype); if (!utype_has_role(punittype, L_DEFEND_GOOD)) { - desire /= 2; /* not good, just ok */ + desire /= 2; /* Not good, just ok */ } if (utype_has_flag(punittype, UTYF_FIELDUNIT)) { /* Causes unhappiness even when in defense, so not a good - * idea for a defender, unless it is _really_ good */ - desire /= 2; - } + * idea for a defender, unless it is _really_ good. + * Downright counter productive, if we want unit just for + * maintaining peace. */ + if (danger == 0) { + desire = -50; + } else { + desire /= 2; + } + } - desire /= POWER_DIVIDER / 2; /* Good enough, no rounding errors. */ - desire *= desire; + if (desire > 0) { + desire /= POWER_DIVIDER / 2; /* Good enough, no rounding errors. */ + desire *= desire; - if (can_city_build_unit_now(pcity, punittype)) { - /* We can build the unit now... */ + if (can_city_build_unit_now(pcity, punittype)) { + /* We can build the unit now... */ - int build_cost = utype_build_shield_cost(pcity, NULL, punittype); - int limit_cost = pcity->shield_stock + 40; + int build_cost = utype_build_shield_cost(pcity, NULL, punittype); + int limit_cost = pcity->shield_stock + 40; - if (walls && !utype_has_flag(punittype, UTYF_BADCITYDEFENDER)) { - desire *= city_data->wallvalue; - /* TODO: More use of POWER_FACTOR ! */ - desire /= POWER_FACTOR; - } + if (walls && !utype_has_flag(punittype, UTYF_BADCITYDEFENDER)) { + desire *= city_data->wallvalue; + /* TODO: More use of POWER_FACTOR ! */ + desire /= POWER_FACTOR; + } - if ((best_unit_cost > limit_cost - && build_cost < best_unit_cost) - || ((desire > best - || (desire == best && build_cost <= best_unit_cost)) - && (best_unit_type == NULL - /* In case all units are more expensive than limit_cost */ - || limit_cost <= pcity->shield_stock + 40))) { - best = desire; - best_unit_type = punittype; - best_unit_cost = build_cost; - } - } else if (can_city_build_unit_later(pcity, punittype)) { - /* We first need to develop the tech required by the unit... */ + if ((best_unit_cost > limit_cost + && build_cost < best_unit_cost) + || ((desire > best + || (desire == best && build_cost <= best_unit_cost)) + && (best_unit_type == NULL + /* In case all units are more expensive than limit_cost */ + || limit_cost <= pcity->shield_stock + 40))) { + best = desire; + best_unit_type = punittype; + best_unit_cost = build_cost; + } + } else if (can_city_build_unit_later(pcity, punittype)) { + /* We first need to develop the tech required by the unit... */ + + /* Cost (shield equivalent) of gaining these techs. */ + /* FIXME? Katvrr advises that this should be weighted more heavily in + * big danger. */ + int tech_cost = research_goal_bulbs_required(presearch, + advance_number(punittype->require_advance)) / 4 + / city_list_size(pplayer->cities); + + /* Contrary to the above, we don't care if walls are actually built + * - we're looking into the future now. */ + if (!utype_has_flag(punittype, UTYF_BADCITYDEFENDER)) { + desire *= city_data->wallvalue; + desire /= POWER_FACTOR; + } - /* Cost (shield equivalent) of gaining these techs. */ - /* FIXME? Katvrr advises that this should be weighted more heavily in - * big danger. */ - int tech_cost = research_goal_bulbs_required(presearch, - advance_number(punittype->require_advance)) / 4 - / city_list_size(pplayer->cities); - - /* Contrary to the above, we don't care if walls are actually built - * - we're looking into the future now. */ - if (!utype_has_flag(punittype, UTYF_BADCITYDEFENDER)) { - desire *= city_data->wallvalue; - desire /= POWER_FACTOR; + /* Yes, there's some similarity with kill_desire(). */ + /* TODO: Explain what shield cost has to do with tech want. */ + tech_desire[utype_index(punittype)] = + (desire * total_want / (utype_build_shield_cost(pcity, NULL, punittype) + tech_cost)); } - - /* Yes, there's some similarity with kill_desire(). */ - /* TODO: Explain what shield cost has to do with tech want. */ - tech_desire[utype_index(punittype)] = - (desire * danger / (utype_build_shield_cost(pcity, NULL, punittype) + tech_cost)); } } simple_ai_unit_type_iterate_end; @@ -917,7 +926,7 @@ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, CITY_LOG(LOG_DEBUG, pcity, "Ooops - we cannot build any defender!"); } - if (best_unit_type) { + if (best_unit_type != NULL) { if (!walls && !utype_has_flag(best_unit_type, UTYF_BADCITYDEFENDER)) { best *= city_data->wallvalue; best /= POWER_FACTOR; @@ -928,7 +937,9 @@ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, * first defender type. */ } - if (best <= 0) best = 1; /* Avoid division by zero below. */ + if (best <= 0) { + best = 1; /* Avoid division by zero below. */ + } /* Update tech_want for appropriate techs for units we want to build. */ simple_ai_unit_type_iterate(punittype) { @@ -952,8 +963,13 @@ bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, } choice->value.utype = best_unit_type; - choice->want = danger; + choice->want = danger; /* FIXME: Not 'total_want' because of the way callers + * are constructed. They may overwrite this value, + * and then the value will NOT contain 'extra_want'. + * Later we may add 'extra_want', and don't want it + * already included in case this was NOT overwritten. */ choice->type = CT_DEFENDER; + return TRUE; } @@ -1555,32 +1571,40 @@ struct adv_choice *military_advisor_choose_build(struct ai_type *ait, bool def_unit_selected = FALSE; int qdanger = city_data->danger * city_data->danger; - /* First determine the danger. It is measured in percents of our - * defensive strength, capped at 200 + urgency */ - if (qdanger >= our_def) { - if (urgency == 0) { - /* don't waste money */ - danger = 100; - } else if (our_def == 0) { - danger = 200 + urgency; + if (qdanger <= 0) { + /* We only need these defenders because of Martial Law value */ + danger = 0; + build_walls = FALSE; /* Walls don't provide Martial Law */ + } else { + /* First determine the danger. It is measured in percents of our + * defensive strength, capped at 200 + urgency */ + if (qdanger >= our_def) { + if (urgency == 0) { + /* Don't waste money */ + danger = 100; + } else if (our_def == 0) { + danger = 200 + urgency; + } else { + danger = MIN(200, 100 * qdanger / our_def) + urgency; + } } else { - danger = MIN(200, 100 * qdanger / our_def) + urgency; + danger = 100 * qdanger / our_def; + } + + if (pcity->surplus[O_SHIELD] <= 0 && our_def != 0) { + /* Won't be able to support anything */ + danger = 0; } - } else { - danger = 100 * qdanger / our_def; - } - if (pcity->surplus[O_SHIELD] <= 0 && our_def != 0) { - /* Won't be able to support anything */ - danger = 0; } CITY_LOG(LOG_DEBUG, pcity, "m_a_c_d urgency=%d danger=%d num_def=%d " "our_def=%d", urgency, danger, num_defenders, our_def); - if (our_def == 0) { + if (our_def == 0 && danger > 0) { /* Build defensive unit first! Walls will help nothing if there's * nobody behind them. */ - if (dai_process_defender_want(ait, pplayer, pcity, danger, choice)) { + if (dai_process_defender_want(ait, pplayer, pcity, danger, choice, + martial_value)) { choice->want = 100 + danger; adv_choice_set_use(choice, "first defender"); build_walls = FALSE; @@ -1640,7 +1664,8 @@ struct adv_choice *military_advisor_choose_build(struct ai_type *ait, adv_init_choice(&uchoice); /* Consider building defensive units */ - if (dai_process_defender_want(ait, pplayer, pcity, danger, &uchoice)) { + if (dai_process_defender_want(ait, pplayer, pcity, danger, &uchoice, + martial_value)) { /* Potential defender found */ if (urgency == 0 && uchoice.value.utype->defense_strength == 1) { @@ -1656,7 +1681,11 @@ struct adv_choice *military_advisor_choose_build(struct ai_type *ait, } uchoice.want += martial_value; - adv_choice_set_use(&uchoice, "defender"); + if (danger > 0) { + adv_choice_set_use(&uchoice, "defender"); + } else { + adv_choice_set_use(&uchoice, "police"); + } if (!build_walls || uchoice.want > choice->want) { adv_choice_copy(choice, &uchoice); diff --git a/ai/default/daimilitary.h b/ai/default/daimilitary.h index 6f7e04a20c..3612fdb09f 100644 --- a/ai/default/daimilitary.h +++ b/ai/default/daimilitary.h @@ -52,6 +52,6 @@ int dai_unit_attack_desirability(struct ai_type *ait, const struct unit_type *punittype); bool dai_process_defender_want(struct ai_type *ait, struct player *pplayer, struct city *pcity, unsigned int danger, - struct adv_choice *choice); + struct adv_choice *choice, adv_want extra_want); #endif /* FC__DAIMILITARY_H */ -- 2.35.1