]> git.pond.sub.org Git - empserver/blob - src/lib/commands/budg.c
Remove budget priorities:
[empserver] / src / lib / commands / budg.c
1 /*
2  *  Empire - A multi-player, client/server Internet based war game.
3  *  Copyright (C) 1986-2006, Dave Pare, Jeff Bailey, Thomas Ruschak,
4  *                           Ken Stevens, Steve McClure
5  *
6  *  This program is free software; you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation; either version 2 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  *  ---
21  *
22  *  See files README, COPYING and CREDITS in the root of the source
23  *  tree for related information and legal notices.  It is expected
24  *  that future projects/authors will amend these files as needed.
25  *
26  *  ---
27  *
28  *  budg.c: Calculate production levels, prioritize
29  * 
30  *  Known contributors to this file:
31  *     Thomas Ruschak, 1992
32  *     Ville Virrankoski, 1995
33  *     Steve McClure, 1997-2000
34  *     Markus Armbruster, 2004-2006
35  */
36
37 #include <config.h>
38
39 #include <string.h>
40 #include "misc.h"
41 #include "player.h"
42 #include "xy.h"
43 #include "nsc.h"
44 #include "sect.h"
45 #include "product.h"
46 #include "nat.h"
47 #include "item.h"
48 #include "file.h"
49 #include "ship.h"
50 #include "land.h"
51 #include "plane.h"
52 #include "optlist.h"
53 #include "budg.h"
54 #include "commands.h"
55
56 static void calc_all(long (*p_sect)[2], int *taxes, int *Ncivs,
57                      int *Nuws, int *bars, int *Nbars, int *mil,
58                      int *ships, int *sbuild, int *nsbuild, int *smaint,
59                      int *units, int *lbuild, int *nlbuild, int *lmaint,
60                      int *planes, int *pbuild, int *npbuild, int *pmaint);
61 static char *dotsprintf(char *buf, char *format, int data);
62 static void prexpense(long cash, int *expensesp, int amount);
63
64 int
65 budg(void)
66 {
67     int i, res;
68     long p_sect[SCT_MAXDEF+1][2];
69     int taxes, Ncivs, Nuws, bars, Nbars, mil;
70     int ships, sbuild, nsbuild, smaint;
71     int units, lbuild, nlbuild, lmaint;
72     int planes, pbuild, npbuild, pmaint;
73     int n, etu;
74     int income, expenses;
75     struct natstr *np;
76     char buf[1024];
77     char in[80];
78
79     etu = etu_per_update;
80
81     np = getnatp(player->cnum);
82
83     player->simulation = 1;
84     calc_all(p_sect,
85              &taxes, &Ncivs, &Nuws, &bars, &Nbars, &mil,
86              &ships, &sbuild, &nsbuild, &smaint,
87              &units, &lbuild, &nlbuild, &lmaint,
88              &planes, &pbuild, &npbuild, &pmaint);
89     player->simulation = 0;
90
91     income = taxes + bars;
92     expenses = 0;
93     pr("Sector Type\t\t\tProduction\t\t\t    Cost\n");
94     for (i = 0; i <= SCT_MAXDEF; i++) {
95         if (!p_sect[i][1] || i == SCT_CAPIT)
96             continue;
97         pr("%-17s\t\t", dchr[i].d_name);
98         if (i == SCT_ENLIST)
99             pr("%ld mil    \t", p_sect[i][0]);
100         else if (pchr[dchr[i].d_prd].p_cost != 0)
101             pr("%ld %-7s\t", p_sect[i][0], pchr[dchr[i].d_prd].p_sname);
102         else
103             pr("\t\t");
104         prexpense(np->nat_money + income, &expenses, p_sect[i][1]);
105     }
106
107     if (sbuild) {
108         sprintf(buf, "%d ship%s", nsbuild, splur(nsbuild));
109         pr("Ship building\t\t\t%-16s", buf);
110         prexpense(np->nat_money + income, &expenses, -sbuild);
111     }
112
113     if (smaint) {
114         sprintf(buf, "%d ship%s", ships, splur(ships));
115         pr("Ship maintenance\t\t%-16s", buf);
116         prexpense(np->nat_money + income, &expenses, -smaint);
117     }
118
119     if (pbuild) {
120         sprintf(buf, "%d plane%s", npbuild, splur(npbuild));
121         pr("Plane building\t\t\t%-16s", buf);
122         prexpense(np->nat_money + income, &expenses, -pbuild);
123     }
124
125     if (pmaint) {
126         sprintf(buf, "%d plane%s", planes, splur(planes));
127         pr("Plane maintenance\t\t%-16s", buf);
128         prexpense(np->nat_money + income, &expenses, -pmaint);
129     }
130     if (lbuild) {
131         sprintf(buf, "%d unit%s", nlbuild, splur(nlbuild));
132         pr("Unit building\t\t\t%-16s", buf);
133         prexpense(np->nat_money + income, &expenses, -lbuild);
134     }
135
136     if (lmaint) {
137         sprintf(buf, "%d unit%s", units, splur(units));
138         pr("Unit maintenance\t\t%-16s", buf);
139         prexpense(np->nat_money + income, &expenses, -lmaint);
140     }
141
142     if (p_sect[SCT_EFFIC][1]) {
143         pr("Sector building\t\t\t\t%8ld sct(s)\t\t%8ld\n",
144            p_sect[SCT_EFFIC][0], p_sect[SCT_EFFIC][1]);
145         expenses += p_sect[SCT_EFFIC][1];
146     }
147     if (mil) {
148         n = (mil - np->nat_reserve * money_res * etu) / (etu * money_mil);
149         sprintf(in, "%d mil, %d res", n, (int)np->nat_reserve);
150         pr("Military payroll\t\t%-32s%8d\n", in, -mil);
151         expenses -= mil;
152     }
153     if (p_sect[SCT_CAPIT][0]) {
154         pr("%c%s maintenance\t\t",
155            toupper(dchr[SCT_CAPIT].d_name[0]),
156            dchr[SCT_CAPIT].d_name + 1);
157         n = p_sect[SCT_CAPIT][0];
158         sprintf(in, "%d %s", n, dchr[SCT_CAPIT].d_name);
159         plurize(in, sizeof(in), n);
160         pr("%-32s%8ld\n", in, p_sect[SCT_CAPIT][1]);
161         expenses += p_sect[SCT_CAPIT][1];
162     }
163     pr("Total expenses%s\n", dotsprintf(buf, "%58d", expenses));
164     if (taxes) {
165         sprintf(in, "%d civ%s, %d uw%s",
166                 Ncivs, splur(Ncivs), Nuws, splur(Nuws));
167         pr("Income from taxes\t\t%-32s%+8d\n", in, taxes);
168     }
169     if (bars) {
170         sprintf(in, "%d bar%s", Nbars, splur(Nbars));
171         pr("Income from bars\t\t%-32s%+8d\n", in, bars);
172     }
173     pr("Total income%s\n", dotsprintf(buf, "%+60d", income));
174     pr("Balance forward\t\t\t\t\t\t      %10ld\n", np->nat_money);
175     pr("Estimated delta\t\t\t\t\t\t      %+10d\n", income - expenses);
176     pr("Estimated new treasury%s\n",
177        dotsprintf(buf, "%50d", np->nat_money + income - expenses));
178     if (((np->nat_money + income - expenses) < 0) && !player->god) {
179         pr("After processsing sectors, you will be broke!\n");
180         pr("Sectors will not produce, distribute, or deliver!\n\n");
181     }
182
183     return RET_OK;
184 }
185
186 static void
187 calc_all(long p_sect[][2],
188          int *taxes, int *Ncivs, int *Nuws, int *bars, int *Nbars, int *mil,
189          int *ships, int *sbuild, int *nsbuild, int *smaint,
190          int *units, int *lbuild, int *nlbuild, int *lmaint,
191          int *planes, int *pbuild, int *npbuild, int *pmaint)
192 {
193     int y, z;
194     struct natstr *np;
195     int *bp;
196     long pop = 0;
197     int n, civ_tax, uw_tax, mil_pay;
198     struct sctstr *sp;
199     int etu = etu_per_update;
200
201     mil_dbl_pay = 0;
202     memset(p_sect, 0, sizeof(**p_sect) * (SCT_MAXDEF+1) * 2);
203     *taxes = *Ncivs = *Nuws = *bars = *Nbars = *mil = 0;
204     *ships = *sbuild = *nsbuild = *smaint = 0;
205     *units = *lbuild = *nlbuild = *lmaint = 0;
206     *planes = *pbuild = *npbuild = *pmaint = 0;
207     
208     np = getnatp(player->cnum);
209     bp = calloc(WORLD_X * WORLD_Y * 8, sizeof(int));
210     for (n = 0; NULL != (sp = getsectid(n)); n++) {
211         fill_update_array(bp, sp);
212         if (sp->sct_own == player->cnum) {
213             sp->sct_updated = 0;
214             tax(sp, np, etu, &pop, &civ_tax, &uw_tax, &mil_pay);
215             *Ncivs += sp->sct_item[I_CIVIL];
216             *Nuws += sp->sct_item[I_UW];
217             *taxes += civ_tax + uw_tax;
218             *mil += mil_pay;
219             if (sp->sct_type == SCT_BANK) {
220                 *bars += bank_income(sp, etu);
221                 *Nbars += sp->sct_item[I_BAR];
222             }
223         }
224     }
225     tpops[player->cnum] = pop;
226     *mil += (int)(np->nat_reserve * money_res * etu);
227
228     *mil += upd_slmilcosts(np->nat_cnum, etu);
229
230     /* Maintain and build ships */
231     sea_money[player->cnum] = 0;
232     *ships = prod_ship(etu, player->cnum, bp, 0);
233     *smaint = sea_money[player->cnum];
234     sea_money[player->cnum] = 0;
235     *nsbuild = prod_ship(etu, player->cnum, bp, 1);
236     *sbuild = sea_money[player->cnum];
237     sea_money[player->cnum] = 0;
238
239     /* Maintain and build planes */
240     air_money[player->cnum] = 0;
241     *planes = prod_plane(etu, player->cnum, bp, 0);
242     *pmaint = air_money[player->cnum];
243     air_money[player->cnum] = 0;
244     *npbuild = prod_plane(etu, player->cnum, bp, 1);
245     *pbuild = air_money[player->cnum];
246     air_money[player->cnum] = 0;
247
248     /* Maintain and build land units */
249     lnd_money[player->cnum] = 0;
250     *units = prod_land(etu, player->cnum, bp, 0);
251     *lmaint = lnd_money[player->cnum];
252     lnd_money[player->cnum] = 0;
253     *nlbuild = prod_land(etu, player->cnum, bp, 1);
254     *lbuild = lnd_money[player->cnum];
255     lnd_money[player->cnum] = 0;
256
257     /* Produce */
258     produce_sect(player->cnum, etu, bp, p_sect);
259
260     free(bp);
261 }
262
263 static char *
264 dotsprintf(char *buf, char *format, int data)
265 {
266     sprintf(buf, format, data);
267     return memset(buf, '.', strspn(buf, " "));
268 }
269
270 static void
271 prexpense(long cash, int *expensesp, int amount)
272 {
273     if (cash > *expensesp) {
274         pr("\t\t%8d\n", amount);
275         *expensesp += amount;
276     } else {
277         pr("\t\t[%7d]\n", amount);
278         *expensesp += amount;
279     }
280 }