]> git.pond.sub.org Git - empserver/blob - src/lib/update/deliver.c
include: Drop update.h
[empserver] / src / lib / update / deliver.c
1 /*
2  *  Empire - A multi-player, client/server Internet based war game.
3  *  Copyright (C) 1986-2016, Dave Pare, Jeff Bailey, Thomas Ruschak,
4  *                Ken Stevens, Steve McClure, Markus Armbruster
5  *
6  *  Empire 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 3 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, see <http://www.gnu.org/licenses/>.
18  *
19  *  ---
20  *
21  *  See files README, COPYING and CREDITS in the root of the source
22  *  tree for related information and legal notices.  It is expected
23  *  that future projects/authors will amend these files as needed.
24  *
25  *  ---
26  *
27  *  deliver.c: Deliver commodities to neighboring sector
28  *
29  *  Known contributors to this file:
30  *
31  */
32
33 #include <config.h>
34
35 #include "file.h"
36 #include "item.h"
37 #include "optlist.h"
38 #include "path.h"
39 #include "plague.h"
40 #include "sect.h"
41 #include "prototypes.h"
42
43 #define DELIVER_BONUS 4.0
44
45 static int
46 deliver(struct sctstr *from, struct ichrstr *ip, int dir,
47         int thresh, int amt_src, int plague, enum i_packing packing)
48 {
49     struct sctstr *to;
50     i_type vtype;               /* item vartype */
51     int amt_moved;
52     int amt_dst;
53     int mobility;
54     double mcost;
55     int n;
56
57     if (dir <= 0 || dir > DIR_UL)
58         return 0;
59     if (amt_src <= 0)
60         return 0;
61     if ((amt_moved = amt_src - thresh) <= 0)
62         return 0;
63     /*
64      * make sure delivery looks ok.  Check where its going,
65      * where its coming from, and see if there is more than
66      * the threshold amount
67      */
68     if (!military_control(from))
69         return 0;
70     to = getsectp(from->sct_x + diroff[dir][0],
71                   from->sct_y + diroff[dir][1]);
72     if (to->sct_own != from->sct_own) {
73         wu(0, from->sct_own, "%s delivery walkout at %s\n",
74            ip->i_name, ownxy(from));
75         return 0;
76     }
77     vtype = ip->i_uid;
78     mobility = from->sct_mobil / 2;
79     if (vtype == I_CIVIL) {
80         if (from->sct_own != from->sct_oldown) {
81             wu(0, from->sct_own,
82                "The conquered populace in %s refuses to relocate!\n",
83                ownxy(from));
84             return 0;
85         }
86         if (to->sct_own != to->sct_oldown) {
87             wu(0, from->sct_own,
88                "Citizens in %s refuse to relocate!\n", ownxy(from));
89             return 0;
90         }
91     }
92     /*
93      * disallow delivery into prohibited sectors.
94      * calculate unit movement cost; decrease amount if
95      * there isn't enough mobility.
96      */
97     mcost = sector_mcost(to, MOB_MOVE) * ip->i_lbs / ip->i_pkg[packing];
98     mcost /= DELIVER_BONUS;
99
100     if (mobility < mcost * amt_moved) {
101         /* XXX can mcost be == 0? */
102         amt_moved = (int)(mobility / mcost);
103         if (amt_moved <= 0)
104             return 0;
105     }
106     amt_dst = to->sct_item[vtype];
107     if (amt_moved > ITEM_MAX - amt_dst) {
108         /* delivery backlog */
109         amt_moved = ITEM_MAX - amt_dst;
110     }
111     to->sct_item[vtype] = amt_moved + amt_dst;
112     /* deliver the plague too! */
113     if (plague == PLG_INFECT && to->sct_pstage == PLG_HEALTHY)
114         to->sct_pstage = PLG_EXPOSED;
115     n = from->sct_mobil - (int)(mcost * amt_moved);
116     if (n < 0)
117         n = 0;
118     from->sct_mobil = n;
119     return amt_moved;
120 }
121
122 void
123 dodeliver(struct sctstr *sp)
124 {
125     i_type i;
126     int thresh;
127     int dir;
128     int plague;
129     enum i_packing packing;
130     int n;
131
132     if (sp->sct_mobil <= 0)
133         return;
134     plague = sp->sct_pstage;
135     packing = sp->sct_effic >= 60 ? dchr[sp->sct_type].d_pkg : IPKG;
136     for (i = I_NONE + 1; i <= I_MAX; i++) {
137         if (sp->sct_del[i] == 0)
138             continue;
139         thresh = sp->sct_del[i] & ~0x7;
140         dir = sp->sct_del[i] & 0x7;
141         n = deliver(sp, &ichr[i], dir, thresh, sp->sct_item[i],
142                     plague, packing);
143         if (n > 0) {
144             sp->sct_item[i] -= n;
145             if (sp->sct_mobil <= 0)
146                 break;
147         }
148     }
149 }