Freeciv-3.1
Loading...
Searching...
No Matches
caravan.c
Go to the documentation of this file.
1/***********************************************************************
2 Freeciv - Copyright (C) 1996 - A Kjeldberg, L Gregersen, P Unold
3 This program is free software; you can redistribute it and/or modify
4 it under the terms of the GNU General Public License as published by
5 the Free Software Foundation; either version 2, or (at your option)
6 any later version.
7
8 This program is distributed in the hope that it will be useful,
9 but WITHOUT ANY WARRANTY; without even the implied warranty of
10 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 GNU General Public License for more details.
12***********************************************************************/
13
14#ifdef HAVE_CONFIG_H
15#include <fc_config.h>
16#endif
17
18#include <math.h>
19
20/* utility */
21#include "log.h"
22
23/* common */
24#include "game.h"
25#include "traderoutes.h"
26
27/* aicore */
28#include "path_finding.h"
29#include "pf_tools.h"
30
31#include "caravan.h"
32
33/************************************************************************/
37{
38 parameter->horizon = FC_INFINITY;
39 parameter->discount = 0.95;
40 parameter->consider_windfall = TRUE;
41 parameter->consider_trade = TRUE;
42 parameter->consider_wonders = TRUE; /* see also init_from_unit */
45 parameter->ignore_transit_time = FALSE;
46 parameter->convert_trade = FALSE;
47 parameter->callback = NULL;
48}
49
50/************************************************************************/
54 const struct unit *caravan)
55{
57 if (!unit_can_do_action(caravan, ACTION_TRADE_ROUTE)) {
58 parameter->consider_trade = FALSE;
59 }
60 if (!unit_can_do_action(caravan, ACTION_MARKETPLACE)
61 && !unit_can_do_action(caravan, ACTION_TRADE_ROUTE)) {
62 parameter->consider_windfall = FALSE;
63 }
64 if (!unit_can_do_action(caravan, ACTION_HELP_WONDER)) {
65 parameter->consider_wonders = FALSE;
66 }
67}
68
69/************************************************************************/
72bool caravan_parameter_is_legal(const struct caravan_parameter *parameter)
73{
74 /* a discount > 1.0 means money later is worth more than money now,
75 which is ridiculous. */
76 if (parameter->discount > 1.0) {
77 return FALSE;
78 }
79
80 /* a negative discount doesn't converge */
81 if (parameter->discount < 0.0) {
82 return FALSE;
83 }
84
85 /* infinite horizon with no discount gives infinite reward. */
86 if (parameter->horizon == FC_INFINITY && parameter->discount == 1.0) {
87 return FALSE;
88 }
89
90 return TRUE;
91}
92
93/************************************************************************/
96static const char *ftl_name(enum foreign_trade_limit ftl)
97{
98 switch (ftl) {
100 return "no";
101 case FTL_ALLIED:
102 return "allied";
103 case FTL_PEACEFUL:
104 return "peaceful";
105 case FTL_NONWAR:
106 return "anything but enemies";
107 }
108
109 return "<illegal>";
110}
111
112/************************************************************************/
116 enum log_level level, const char *file,
117 const char *function, int line)
118{
119 do_log(file, function, line, FALSE, level,
120 "parameter {\n"
121 " horizon = %d\n"
122 " discount = %g\n"
123 " objective = <%s,%s,%s>\n"
124 " account-broken = %s\n"
125 " allow-foreign = %s\n"
126 " ignore-transit = %s\n"
127 " convert-trade = %s\n"
128 "}\n",
129 parameter->horizon,
130 parameter->discount,
131 parameter->consider_windfall ? "windfall" : "-",
132 parameter->consider_trade ? "trade" : "-",
133 parameter->consider_wonders ? "wonders" : "-",
134 parameter->account_for_broken_routes ? "yes" : "no",
135 ftl_name(parameter->allow_foreign_trade),
136 parameter->ignore_transit_time ? "yes" : "no",
137 parameter->convert_trade ? "yes" : "no");
138}
139
140
141/************************************************************************/
145{
146 result->src = result->dest = NULL;
147 result->arrival_time = 0;
148 result->value = 0;
149 result->help_wonder = FALSE;
150 result->required_boat = FALSE;
151}
152
153/************************************************************************/
157static void caravan_result_init(struct caravan_result *result,
158 const struct city *src,
159 const struct city *dest,
160 int arrival_time)
161{
162 result->src = src;
163 result->dest = dest;
164 result->arrival_time = arrival_time;
165
166 result->value = 0;
167 result->help_wonder = FALSE;
168 /* FIXME: required_boat field is never used. */
169 if ((src != NULL) && (dest != NULL)) {
170 if (tile_continent(src->tile) != tile_continent(dest->tile)) {
171 result->required_boat = TRUE;
172 } else {
173 result->required_boat = FALSE;
174 }
175 } else {
176 result->required_boat = FALSE;
177 }
178}
179
180/************************************************************************/
184 const struct caravan_result *b)
185{
186 if (a->value > b->value) {
187 return 1;
188 } else if (a->value < b->value) {
189 return -1;
190 } else {
191 /* faster time is better, so reverse-sort on time. */
192 return b->arrival_time - a->arrival_time ;
193 }
194}
195
196/************************************************************************/
204typedef bool (*search_callback) (const struct civ_map *nmap,
205 void *data, const struct city *pcity,
206 int arrival_turn, int arrival_moves_left);
207
208static void caravan_search_from(const struct civ_map *nmap,
209 const struct unit *caravan,
210 const struct caravan_parameter *param,
211 struct tile *start_tile,
212 int turns_before, int moves_left_before,
213 bool omniscient,
215 void *callback_data)
216{
217 struct pf_map *pfm;
218 struct pf_parameter pfparam;
219 int end_time;
220
221 end_time = param->horizon - turns_before;
222
223 /* Initialize the pf run. */
224 pft_fill_unit_parameter(&pfparam, nmap, (struct unit *) caravan);
225 pfparam.start_tile = start_tile;
226 pfparam.moves_left_initially = moves_left_before;
227 pfparam.omniscience = omniscient;
228 pfm = pf_map_new(&pfparam);
229
230 /* For every tile in distance order:
231 quit if we've exceeded the maximum number of turns
232 otherwise, run the callback if we're on a city.
233 Do-while loop rather than while loop to make sure to process the
234 start tile.
235 */
237 struct city *pcity;
238
239 if (pos.turn > end_time) {
240 break;
241 }
242
243 pcity = tile_city(pos.tile);
244 if (pcity && callback(nmap, callback_data, pcity, turns_before + pos.turn,
245 pos.moves_left)) {
246 break;
247 }
249
250 pf_map_destroy(pfm);
251}
252
253/************************************************************************/
257static double windfall_benefit(const struct unit *caravan,
258 const struct city *src,
259 const struct city *dest,
260 const struct caravan_parameter *param)
261{
262 if (!param->consider_windfall || !can_cities_trade(src, dest)) {
263 return 0;
264 } else {
265 bool can_establish = (unit_can_do_action(caravan, ACTION_TRADE_ROUTE)
266 && can_establish_trade_route(src, dest));
267 int bonus = get_caravan_enter_city_trade_bonus(src, dest, NULL,
268 can_establish);
269
270 /* when bonus goes to both sci and gold, double it */
271 if (TBONUS_BOTH == trade_route_settings_by_type
272 (cities_trade_route_type(src, dest))->bonus_type) {
273 bonus *= 2;
274 }
275
276 return bonus;
277 }
278}
279
280/****************************************************************************
281 Compute the change in the per-turn trade.
282****************************************************************************/
283
284/************************************************************************/
288static int one_city_trade_benefit(const struct city *pcity,
289 const struct player *pplayer,
290 bool countloser, int newtrade)
291{
292 int losttrade = 0;
293
294 /* if the city is owned by someone else, we don't benefit from the
295 new trade (but we might still lose from a broken trade route) */
296 if (city_owner(pcity) != pplayer) {
297 newtrade = 0;
298 }
299
300 if (city_num_trade_routes(pcity) < max_trade_routes(pcity)) {
301 /* if the city can handle this route, we don't break any old routes */
302 losttrade = 0;
303 } else {
304 struct trade_route_list *would_remove = (countloser ? trade_route_list_new() : NULL);
305 int oldtrade = city_trade_removable(pcity, would_remove);
306
307 /* if we own the city, the trade benefit is only by how much
308 better we are than the old trade route */
309 if (city_owner(pcity) == pplayer) {
310 newtrade -= oldtrade;
311 }
312
313 /* if the cities that lost a trade route is one of ours, and if we
314 care about accounting for the lost trade, count it. */
315 if (countloser) {
316 trade_route_list_iterate(would_remove, plost) {
317 struct city *losercity = game_city_by_number(plost->partner);
318
319 if (city_owner(losercity) == pplayer) {
320 trade_routes_iterate(losercity, pback) {
321 if (pback->partner == pcity->id) {
322 losttrade += pback->value;
323 }
325 }
327 trade_route_list_destroy(would_remove);
328 }
329 }
330
331 /* find the increase or decrease in trade we benefit from */
332 return newtrade - losttrade;
333}
334
335/************************************************************************/
340static double trade_benefit(const struct player *caravan_owner,
341 const struct city *src,
342 const struct city *dest,
343 const struct caravan_parameter *param)
344{
345 /* do we care about trade at all? */
346 if (!param->consider_trade) {
347 return 0;
348 }
349
350 /* first, see if a new route is made. */
351 if (!can_cities_trade(src, dest) || !can_establish_trade_route(src, dest)) {
352 return 0;
353 }
354 if (max_trade_routes(src) <= 0 || max_trade_routes(dest) <= 0) {
355 /* Can't create new trade routes even by replacing old ones if
356 * there's no slots at all. */
357 return 0;
358 }
359
360 if (!param->convert_trade) {
361 bool countloser = param->account_for_broken_routes;
362 int newtrade = trade_base_between_cities(src, dest);
363
364 return one_city_trade_benefit(src, caravan_owner, countloser, newtrade)
365 + one_city_trade_benefit(dest, caravan_owner, countloser, newtrade);
366 } else {
367 /* Always fails. */
369 "Unimplemented functionality: "
370 "using CM to calculate trade.");
371 return 0;
372 }
373}
374
375/************************************************************************/
382static double wonder_benefit(const struct unit *caravan, int arrival_time,
383 const struct city *dest,
384 const struct caravan_parameter *param)
385{
386 int costwithout, costwith;
387 int shields_at_arrival;
388
389 if (!param->consider_wonders
390 /* TODO: Should helping an ally to build be considered when legal? */
391 || unit_owner(caravan) != city_owner(dest)
393 /* TODO: Should helping to build a unit be considered when legal? */
394 || VUT_UTYPE == dest->production.kind
395 /* TODO: Should helping to build an improvement be considered when
396 * legal? */
397 || !is_wonder(dest->production.value.building)) {
398 return 0;
399 }
400
401 shields_at_arrival = dest->shield_stock
402 + arrival_time * dest->surplus[O_SHIELD];
403
404 costwithout = impr_buy_gold_cost(dest, dest->production.value.building,
405 shields_at_arrival);
406 costwith = impr_buy_gold_cost(dest, dest->production.value.building,
407 shields_at_arrival + unit_build_shield_cost_base(caravan));
408
409 fc_assert_ret_val(costwithout >= costwith, -1.0);
410 return costwithout - costwith;
411}
412
413/************************************************************************/
417static double presentvalue(double payment, int term, double rate)
418{
419 return payment * pow(rate, term);
420}
421
422/************************************************************************/
426static double perpetuity(double payment, double rate)
427{
428 return payment / (1.0 - rate);
429}
430
431/************************************************************************/
435static double annuity(double payment, int term, double rate)
436{
437 return perpetuity(payment, rate) * (1.0 - 1.0 / pow(rate, term));
438}
439
440/************************************************************************/
443static bool does_foreign_trade_param_allow(const struct caravan_parameter *param,
444 struct player *src, struct player *dest)
445{
446 switch (param->allow_foreign_trade) {
448 return (src == dest);
449 break;
450 case FTL_ALLIED:
451 return pplayers_allied(src, dest);
452 break;
453 case FTL_PEACEFUL:
454 return pplayers_in_peace(src, dest);
455 break;
456 case FTL_NONWAR:
457 return !pplayers_at_war(src, dest);
458 }
459
461 return FALSE;
462}
463
464/************************************************************************/
470static bool get_discounted_reward(const struct unit *caravan,
471 const struct caravan_parameter *parameter,
472 struct caravan_result *result)
473{
474 double trade;
475 double windfall;
476 double wonder;
477 const struct city *src = result->src;
478 const struct city *dest = result->dest;
479 int arrival_time = result->arrival_time;
480 double discount = parameter->discount;
481 struct player *pplayer_src = city_owner(src);
482 struct player *pplayer_dest = city_owner(dest);
483 int cost = unit_build_shield_cost(src, caravan);
484 bool consider_wonder;
485 bool consider_trade;
486 bool consider_windfall;
487
488 /* if no foreign trade is allowed, just quit. */
489 if (!does_foreign_trade_param_allow(parameter, pplayer_src, pplayer_dest)) {
491 return FALSE;
492 }
493
494 consider_wonder = parameter->consider_wonders
496 action_speculate_unit_on_city(ACTION_HELP_WONDER,
497 caravan, src, city_tile(dest),
498 TRUE, dest));
499 consider_trade = parameter->consider_trade
501 action_speculate_unit_on_city(ACTION_TRADE_ROUTE,
502 caravan, src, city_tile(dest),
503 TRUE, dest));
504 consider_windfall = parameter->consider_windfall
506 action_speculate_unit_on_city(ACTION_MARKETPLACE,
507 caravan, src, city_tile(dest),
508 TRUE, dest));
509
510 if (!consider_wonder && !consider_trade && !consider_windfall) {
511 /* No caravan action is possible against this target. */
513
514 return FALSE;
515 }
516
517 if (consider_wonder) {
518 wonder = wonder_benefit(caravan, arrival_time, dest, parameter);
519 /* we want to aid for wonder building */
520 wonder *= 2;
521
522 wonder = presentvalue(wonder, arrival_time, discount);
523 } else {
524 wonder = -1.0;
525 }
526
527 if (consider_trade) {
528 trade = trade_benefit(pplayer_src, src, dest, parameter);
529 if (parameter->horizon == FC_INFINITY) {
530 trade = perpetuity(trade, discount);
531 } else {
532 trade = annuity(trade, parameter->horizon - arrival_time, discount);
533 }
534 trade = presentvalue(trade, arrival_time, discount);
535 } else {
536 trade = 0.0;
537 }
538
539 if (consider_windfall) {
540 windfall = windfall_benefit(caravan, src, dest, parameter);
541 windfall = presentvalue(windfall, arrival_time, discount);
542 } else {
543 windfall = 0.0;
544 }
545
546 if ((consider_trade
547 || (consider_windfall
548 && (!parameter->consider_trade /* can only enter marketplaces */
549 /* (FIXME: test any extra restrictions for trade routes) */
550 /* or the result is so big that we are still in plus */
551 /* (consider having produced IF_GOLD instead) */
552 || windfall
554 / MAX(src->surplus[O_SHIELD], 2), discount)))
555 ) && trade + windfall >= wonder) {
556 result->value = trade + windfall;
557 result->help_wonder = FALSE;
558 } else if (consider_wonder) {
559 result->value = wonder;
560 result->help_wonder = TRUE;
561 } else {
563 return FALSE;
564 }
565
566 if (parameter->callback) {
567 parameter->callback(result, parameter->callback_data);
568 }
569
570 return TRUE;
571}
572
573/****************************************************************************
574 Functions to compute the benefit of moving the caravan to dest.
575****************************************************************************/
576
577/************************************************************************/
581static void caravan_evaluate_notransit(const struct unit *caravan,
582 const struct city *dest,
583 const struct caravan_parameter *param,
584 struct caravan_result *result)
585{
586 const struct city *src = game_city_by_number(caravan->homecity);
587
588 caravan_result_init(result, src, dest, 0);
589 get_discounted_reward(caravan, param, result);
590}
591
592/************************************************************************/
596struct cewt_data {
597 const struct unit *caravan;
600};
601
602static bool cewt_callback(const struct civ_map *nmap,
603 void *vdata, const struct city *dest,
604 int arrival_time, int moves_left)
605{
606 struct cewt_data *data = vdata;
607
609
610 if (dest == data->result->dest) {
611 data->result->arrival_time = arrival_time;
612 get_discounted_reward(data->caravan, data->param, data->result);
613 return TRUE;
614 } else {
615 return FALSE;
616 }
617}
618
619/************************************************************************/
623static void caravan_evaluate_withtransit(const struct civ_map *nmap,
624 const struct unit *caravan,
625 const struct city *dest,
626 const struct caravan_parameter *param,
627 struct caravan_result *result,
628 bool omniscient)
629{
630 struct cewt_data data;
631
632 data.caravan = caravan;
633 data.param = param;
635 dest, 0);
636 data.result = result;
638 caravan->moves_left, omniscient, cewt_callback, &data);
639}
640
641/************************************************************************/
644void caravan_evaluate(const struct unit *caravan,
645 const struct city *dest,
646 const struct caravan_parameter *param,
647 struct caravan_result *result, bool omniscient)
648{
649 const struct civ_map *nmap = &(wld.map);
650
651 if (param->ignore_transit_time) {
652 caravan_evaluate_notransit(caravan, dest, param, result);
653 } else {
654 caravan_evaluate_withtransit(nmap, caravan, dest, param, result, omniscient);
655 }
656}
657
658
659/************************************************************************/
662static void caravan_find_best_destination_notransit(const struct unit *caravan,
663 const struct caravan_parameter *param,
664 struct caravan_result *best)
665{
666 struct caravan_result current;
667 struct city *pcity = game_city_by_number(caravan->homecity);
668 struct player *src_owner = city_owner(pcity);
669
670 caravan_result_init(best, pcity, NULL, 0);
671 current = *best;
672
673 players_iterate(dest_owner) {
674 if (does_foreign_trade_param_allow(param, src_owner, dest_owner)) {
675 city_list_iterate(dest_owner->cities, dest) {
676 caravan_result_init(&current, pcity, dest, 0);
677 get_discounted_reward(caravan, param, &current);
678
679 if (caravan_result_compare(&current, best) > 0) {
680 *best = current;
681 }
683 }
685}
686
687/************************************************************************/
693 const struct unit *caravan;
695};
696
697static bool cfbdw_callback(const struct civ_map *nmap,
698 void *vdata, const struct city *dest,
699 int arrival_time, int moves_left)
700{
701 struct cfbdw_data *data = vdata;
702 struct caravan_result current;
703
704 caravan_result_init(&current, data->best->src, dest, arrival_time);
705
706 get_discounted_reward(data->caravan, data->param, &current);
707 if (caravan_result_compare(&current, data->best) > 0) {
708 *data->best = current;
709 }
710
711 return FALSE; /* don't stop. */
712}
713
714/************************************************************************/
718 const struct civ_map *nmap,
719 const struct unit *caravan,
720 const struct caravan_parameter *param,
721 const struct city *src,
722 int turns_before,
723 int moves_left,
724 bool omniscient,
725 struct caravan_result *result)
726{
727 struct tile *start_tile;
728 struct cfbdw_data data;
729
730 data.param = param;
731 data.caravan = caravan;
732 data.best = result;
733 caravan_result_init(data.best, src, NULL, 0);
734
735 if (src->id != caravan->homecity) {
736 start_tile = src->tile;
737 } else {
738 start_tile = unit_tile(caravan);
739 }
740
741 caravan_search_from(nmap, caravan, param, start_tile, turns_before,
742 caravan->moves_left, omniscient, cfbdw_callback, &data);
743}
744
745/************************************************************************/
751 const struct unit *caravan,
752 const struct caravan_parameter *parameter,
753 struct caravan_result *result, bool omniscient)
754{
755 if (parameter->ignore_transit_time) {
757 } else {
758 const struct city *src = game_city_by_number(caravan->homecity);
759
760 fc_assert(src != NULL);
761
762 caravan_find_best_destination_withtransit(nmap, caravan, parameter, src, 0,
763 caravan->moves_left, omniscient, result);
764 }
765}
766
767/************************************************************************/
770static void caravan_optimize_notransit(const struct unit *caravan,
771 const struct caravan_parameter *param,
772 struct caravan_result *best)
773{
774 struct player *pplayer = unit_owner(caravan);
775
776 /* Iterate over all cities we own (since the caravan could change its
777 * home city); iterate over all cities we know about (places the caravan
778 * can go to); pick out the best trade route. */
779 city_list_iterate(pplayer->cities, src) {
780 players_iterate(dest_owner) {
781 if (does_foreign_trade_param_allow(param, pplayer, dest_owner)) {
782 city_list_iterate(dest_owner->cities, dest) {
783 struct caravan_result current;
784
785 caravan_result_init(&current, src, dest, 0);
786 get_discounted_reward(caravan, param, &current);
787 if (caravan_result_compare(&current, best) > 0) {
788 *best = current;
789 }
791 }
794}
795
796/****************************************************************************
797 Struct for the caravan_search invocation in
798 caravan_optimize_withtransit.
799****************************************************************************/
800struct cowt_data {
802 const struct unit *caravan;
805};
806
807/************************************************************************/
814static bool cowt_callback(const struct civ_map *nmap, void *vdata,
815 const struct city *pcity,
816 int arrival_time, int moves_left)
817{
818 struct cowt_data *data = vdata;
819 const struct unit *caravan = data->caravan;
820 struct caravan_result current;
821
823 pcity, arrival_time);
824
825 /* first, see what benefit we'd get from not changing home city */
826 get_discounted_reward(caravan, data->param, &current);
827 if (caravan_result_compare(&current, data->best) > 0) {
828 *data->best = current;
829 }
830
831 /* next, try changing home city (if we're allowed to) */
832 if (city_owner(pcity) == unit_owner(caravan)) {
834 nmap, caravan, data->param, pcity, arrival_time, moves_left, data->omniscient,
835 &current);
836 if (caravan_result_compare(&current, data->best) > 0) {
837 *data->best = current;
838 }
839 }
840
841 return FALSE; /* don't stop searching */
842}
843
844/************************************************************************/
848static void caravan_optimize_withtransit(const struct civ_map *nmap,
849 const struct unit *caravan,
850 const struct caravan_parameter *param,
851 struct caravan_result *result,
852 bool omniscient)
853{
854 struct cowt_data data;
855
856 data.param = param;
857 data.caravan = caravan;
858 data.best = result;
859 data.omniscient = omniscient;
863}
864
865/************************************************************************/
871 const struct caravan_parameter *param,
872 struct caravan_result *result, bool omniscient)
873{
874 const struct civ_map *nmap = &(wld.map);
875
876 if (param->ignore_transit_time) {
877 caravan_optimize_notransit(caravan, param, result);
878 } else {
879 caravan_optimize_withtransit(nmap, caravan, param, result, omniscient);
880 }
881}
bool action_prob_possible(const struct act_prob probability)
Definition actions.c:6707
struct act_prob action_speculate_unit_on_city(const action_id act_id, const struct unit *actor, const struct city *actor_home, const struct tile *actor_tile, const bool omniscient_cheat, const struct city *target)
Definition actions.c:6459
static double wonder_benefit(const struct unit *caravan, int arrival_time, const struct city *dest, const struct caravan_parameter *param)
Definition caravan.c:382
static double perpetuity(double payment, double rate)
Definition caravan.c:426
void caravan_find_best_destination(const struct civ_map *nmap, const struct unit *caravan, const struct caravan_parameter *parameter, struct caravan_result *result, bool omniscient)
Definition caravan.c:750
static bool does_foreign_trade_param_allow(const struct caravan_parameter *param, struct player *src, struct player *dest)
Definition caravan.c:443
static bool cowt_callback(const struct civ_map *nmap, void *vdata, const struct city *pcity, int arrival_time, int moves_left)
Definition caravan.c:814
static void caravan_search_from(const struct civ_map *nmap, const struct unit *caravan, const struct caravan_parameter *param, struct tile *start_tile, int turns_before, int moves_left_before, bool omniscient, search_callback callback, void *callback_data)
Definition caravan.c:208
bool caravan_parameter_is_legal(const struct caravan_parameter *parameter)
Definition caravan.c:72
static void caravan_find_best_destination_notransit(const struct unit *caravan, const struct caravan_parameter *param, struct caravan_result *best)
Definition caravan.c:662
static void caravan_evaluate_notransit(const struct unit *caravan, const struct city *dest, const struct caravan_parameter *param, struct caravan_result *result)
Definition caravan.c:581
static int one_city_trade_benefit(const struct city *pcity, const struct player *pplayer, bool countloser, int newtrade)
Definition caravan.c:288
static void caravan_optimize_withtransit(const struct civ_map *nmap, const struct unit *caravan, const struct caravan_parameter *param, struct caravan_result *result, bool omniscient)
Definition caravan.c:848
void caravan_result_init_zero(struct caravan_result *result)
Definition caravan.c:144
static bool cewt_callback(const struct civ_map *nmap, void *vdata, const struct city *dest, int arrival_time, int moves_left)
Definition caravan.c:602
static void caravan_optimize_notransit(const struct unit *caravan, const struct caravan_parameter *param, struct caravan_result *best)
Definition caravan.c:770
static double annuity(double payment, int term, double rate)
Definition caravan.c:435
static bool cfbdw_callback(const struct civ_map *nmap, void *vdata, const struct city *dest, int arrival_time, int moves_left)
Definition caravan.c:697
static const char * ftl_name(enum foreign_trade_limit ftl)
Definition caravan.c:96
bool(* search_callback)(const struct civ_map *nmap, void *data, const struct city *pcity, int arrival_turn, int arrival_moves_left)
Definition caravan.c:204
static void caravan_evaluate_withtransit(const struct civ_map *nmap, const struct unit *caravan, const struct city *dest, const struct caravan_parameter *param, struct caravan_result *result, bool omniscient)
Definition caravan.c:623
int caravan_result_compare(const struct caravan_result *a, const struct caravan_result *b)
Definition caravan.c:183
static void caravan_find_best_destination_withtransit(const struct civ_map *nmap, const struct unit *caravan, const struct caravan_parameter *param, const struct city *src, int turns_before, int moves_left, bool omniscient, struct caravan_result *result)
Definition caravan.c:717
void caravan_parameter_log_real(const struct caravan_parameter *parameter, enum log_level level, const char *file, const char *function, int line)
Definition caravan.c:115
void caravan_evaluate(const struct unit *caravan, const struct city *dest, const struct caravan_parameter *param, struct caravan_result *result, bool omniscient)
Definition caravan.c:644
static bool get_discounted_reward(const struct unit *caravan, const struct caravan_parameter *parameter, struct caravan_result *result)
Definition caravan.c:470
static double trade_benefit(const struct player *caravan_owner, const struct city *src, const struct city *dest, const struct caravan_parameter *param)
Definition caravan.c:340
void caravan_parameter_init_default(struct caravan_parameter *parameter)
Definition caravan.c:36
static double windfall_benefit(const struct unit *caravan, const struct city *src, const struct city *dest, const struct caravan_parameter *param)
Definition caravan.c:257
static double presentvalue(double payment, int term, double rate)
Definition caravan.c:417
void caravan_parameter_init_from_unit(struct caravan_parameter *parameter, const struct unit *caravan)
Definition caravan.c:53
static void caravan_result_init(struct caravan_result *result, const struct city *src, const struct city *dest, int arrival_time)
Definition caravan.c:157
void caravan_optimize_allpairs(const struct unit *caravan, const struct caravan_parameter *param, struct caravan_result *result, bool omniscient)
Definition caravan.c:870
foreign_trade_limit
Definition caravan.h:29
@ FTL_ALLIED
Definition caravan.h:31
@ FTL_NATIONAL_ONLY
Definition caravan.h:30
@ FTL_NONWAR
Definition caravan.h:33
@ FTL_PEACEFUL
Definition caravan.h:32
bool city_production_gets_caravan_shields(const struct universal *tgt)
Definition city.c:1829
#define city_list_iterate(citylist, pcity)
Definition city.h:488
#define city_tile(_pcity_)
Definition city.h:544
#define city_owner(_pcity_)
Definition city.h:543
#define city_list_iterate_end
Definition city.h:490
struct unit struct city struct unit struct tile struct extra_type const struct act_prob *act_probs int actor_unit_id struct unit struct unit int cost
Definition dialogs_g.h:73
@ O_SHIELD
Definition fc_types.h:91
struct world wld
Definition game.c:58
struct city * game_city_by_number(int id)
Definition game.c:102
static struct tile * pos
Definition finddlg.c:53
int impr_buy_gold_cost(const struct city *pcity, const struct impr_type *pimprove, int shields_in_stock)
bool is_wonder(const struct impr_type *pimprove)
void do_log(const char *file, const char *function, int line, bool print_from_where, enum log_level level, const char *message,...)
Definition log.c:514
#define fc_assert_msg(condition, message,...)
Definition log.h:181
#define fc_assert(condition)
Definition log.h:176
#define fc_assert_ret_val(condition, val)
Definition log.h:194
log_level
Definition log.h:28
struct pf_map * pf_map_new(const struct pf_parameter *parameter)
void pf_map_destroy(struct pf_map *pfm)
#define pf_map_positions_iterate_end
#define pf_map_positions_iterate(ARG_pfm, NAME_pos, COND_from_start)
void pft_fill_unit_parameter(struct pf_parameter *parameter, const struct civ_map *nmap, const struct unit *punit)
Definition pf_tools.c:840
bool pplayers_at_war(const struct player *pplayer, const struct player *pplayer2)
Definition player.c:1364
bool pplayers_allied(const struct player *pplayer, const struct player *pplayer2)
Definition player.c:1381
bool pplayers_in_peace(const struct player *pplayer, const struct player *pplayer2)
Definition player.c:1402
#define players_iterate_end
Definition player.h:535
#define players_iterate(_pplayer)
Definition player.h:530
struct setting_list * level[OLEVELS_NUM]
Definition settings.c:183
#define FC_INFINITY
Definition shared.h:36
#define MAX(x, y)
Definition shared.h:54
bool consider_trade
Definition caravan.h:86
bool consider_windfall
Definition caravan.h:85
void(* callback)(const struct caravan_result *result, void *data)
Definition caravan.h:124
enum foreign_trade_limit allow_foreign_trade
Definition caravan.h:99
bool consider_wonders
Definition caravan.h:87
bool ignore_transit_time
Definition caravan.h:108
double discount
Definition caravan.h:77
bool account_for_broken_routes
Definition caravan.h:93
void * callback_data
Definition caravan.h:125
const struct city * src
Definition caravan.h:52
const struct city * dest
Definition caravan.h:53
int arrival_time
Definition caravan.h:54
bool help_wonder
Definition caravan.h:57
bool required_boat
Definition caravan.h:58
double value
Definition caravan.h:56
const struct unit * caravan
Definition caravan.c:597
const struct caravan_parameter * param
Definition caravan.c:599
struct caravan_result * result
Definition caravan.c:598
struct caravan_result * best
Definition caravan.c:694
const struct unit * caravan
Definition caravan.c:693
const struct caravan_parameter * param
Definition caravan.c:692
Definition city.h:309
int surplus[O_LAST]
Definition city.h:343
int id
Definition city.h:315
struct universal production
Definition city.h:382
int bonus[O_LAST]
Definition city.h:351
struct tile * tile
Definition city.h:311
int shield_stock
Definition city.h:355
const struct caravan_parameter * param
Definition caravan.c:801
struct caravan_result * best
Definition caravan.c:803
const struct unit * caravan
Definition caravan.c:802
bool omniscient
Definition caravan.c:804
int moves_left_initially
struct tile * start_tile
struct city_list * cities
Definition player.h:281
Definition tile.h:49
Definition unit.h:138
int moves_left
Definition unit.h:150
int homecity
Definition unit.h:146
enum universals_n kind
Definition fc_types.h:758
universals_u value
Definition fc_types.h:757
struct civ_map map
#define TRUE
Definition support.h:46
#define FALSE
Definition support.h:47
#define bool
Definition support.h:61
struct city * tile_city(const struct tile *ptile)
Definition tile.c:83
#define tile_continent(_tile)
Definition tile.h:91
bool can_cities_trade(const struct city *pc1, const struct city *pc2)
enum trade_route_type cities_trade_route_type(const struct city *pcity1, const struct city *pcity2)
Definition traderoutes.c:58
int trade_base_between_cities(const struct city *pc1, const struct city *pc2)
int city_num_trade_routes(const struct city *pcity)
struct trade_route_settings * trade_route_settings_by_type(enum trade_route_type type)
unsigned max_trade_routes(const struct city *pcity)
Definition traderoutes.c:48
int get_caravan_enter_city_trade_bonus(const struct city *pc1, const struct city *pc2, struct goods_type *pgood, const bool establish_trade)
int city_trade_removable(const struct city *pcity, struct trade_route_list *would_remove)
bool can_establish_trade_route(const struct city *pc1, const struct city *pc2)
#define trade_route_list_iterate(trade_route_list, proute)
Definition traderoutes.h:95
#define trade_routes_iterate_end
#define trade_route_list_iterate_end
Definition traderoutes.h:97
#define trade_routes_iterate(c, proute)
const struct impr_type * building
Definition fc_types.h:598
bool unit_can_do_action(const struct unit *punit, const action_id act_id)
Definition unit.c:331
#define unit_tile(_pu)
Definition unit.h:388
#define unit_owner(_pu)
Definition unit.h:387
int unit_build_shield_cost(const struct city *pcity, const struct unit *punit)
Definition unittype.c:1528
int unit_build_shield_cost_base(const struct unit *punit)
Definition unittype.c:1536