neweff production: Consider insufficient food
newe() and prod() duplicate parts of the update's do_feed(), except they round babies down instead of randomly, to get a stable, conservative forecast. Unlike the update, they assume sufficient food. Inaccurate for sectors that are going to starve or have suboptimal population growth. Not documented. Has always been that way. Eliminate the undocumented assumption by replacing the duplicate code by a call of do_feed(). Add a suitable parameter to do_feed() to preserve the different rounding. The update test shows the improvement. Signed-off-by: Markus Armbruster <armbru@pond.sub.org>
This commit is contained in:
parent
43fc32a365
commit
41a2f7d1df
6 changed files with 40 additions and 56 deletions
|
@ -42,17 +42,19 @@
|
|||
#include "update.h"
|
||||
#include "xy.h"
|
||||
|
||||
static int new_work(struct sctstr *, int);
|
||||
static int growfood(struct sctstr *, int, int);
|
||||
static int starve_some(short *, i_type, int);
|
||||
static void trunc_people(struct sctstr *, struct natstr *);
|
||||
static int grow_people(struct sctstr *, int, struct natstr *);
|
||||
static int babies(int, int, double, int, int);
|
||||
static int grow_people(struct sctstr *, int, struct natstr *, int);
|
||||
static int babies(int, int, double, int, int, int);
|
||||
|
||||
/*
|
||||
* feed the individual sector
|
||||
*/
|
||||
int
|
||||
do_feed(struct sctstr *sp, struct natstr *np, int etu)
|
||||
do_feed(struct sctstr *sp, struct natstr *np, int etu,
|
||||
int round_babies_down)
|
||||
{
|
||||
int work_avail;
|
||||
int starved, sctwork;
|
||||
|
@ -67,7 +69,6 @@ do_feed(struct sctstr *sp, struct natstr *np, int etu)
|
|||
sp->sct_item[I_MILIT],
|
||||
sp->sct_item[I_UW],
|
||||
maxworkers));
|
||||
|
||||
if (sp->sct_type != SCT_SANCT) {
|
||||
manna = 0;
|
||||
if (opt_NOFOOD == 0) {
|
||||
|
@ -100,7 +101,7 @@ do_feed(struct sctstr *sp, struct natstr *np, int etu)
|
|||
sctwork += 7 + roll(15);
|
||||
if (sctwork > 100)
|
||||
sctwork = 100;
|
||||
grow_people(sp, etu, np);
|
||||
grow_people(sp, etu, np, round_babies_down);
|
||||
work_avail = new_work(sp,
|
||||
total_work(sp->sct_work, etu,
|
||||
sp->sct_item[I_CIVIL],
|
||||
|
@ -124,7 +125,7 @@ do_feed(struct sctstr *sp, struct natstr *np, int etu)
|
|||
return work_avail;
|
||||
}
|
||||
|
||||
int
|
||||
static int
|
||||
new_work(struct sctstr *sp, int delta)
|
||||
{
|
||||
if (sp->sct_type == sp->sct_newtype)
|
||||
|
@ -231,17 +232,17 @@ trunc_people(struct sctstr *sp, struct natstr *np)
|
|||
* production? Maybe with just high education?
|
||||
*/
|
||||
static int
|
||||
grow_people(struct sctstr *sp, int etu, struct natstr *np)
|
||||
grow_people(struct sctstr *sp, int etu, struct natstr *np, int round_down)
|
||||
{
|
||||
int newciv;
|
||||
int newuw;
|
||||
int maxpop = max_pop(np->nat_level[NAT_RLEV], sp);
|
||||
|
||||
newciv = babies(sp->sct_item[I_CIVIL], etu, obrate,
|
||||
sp->sct_item[I_FOOD], maxpop);
|
||||
sp->sct_item[I_FOOD], maxpop, round_down);
|
||||
sp->sct_item[I_CIVIL] += newciv;
|
||||
newuw = babies(sp->sct_item[I_UW], etu, uwbrate,
|
||||
sp->sct_item[I_FOOD], maxpop);
|
||||
sp->sct_item[I_FOOD], maxpop, round_down);
|
||||
sp->sct_item[I_UW] += newuw;
|
||||
/*
|
||||
* subtract the baby eat food (if we are using FOOD) and return
|
||||
|
@ -257,24 +258,28 @@ grow_people(struct sctstr *sp, int etu, struct natstr *np)
|
|||
* @brate is the birth rate.
|
||||
* @food is the food available for growing babies.
|
||||
* @maxpop is the population limit.
|
||||
* If @round_down, discard fractions instead of rounding them
|
||||
* randomly.
|
||||
*/
|
||||
static int
|
||||
babies(int adults, int etu, double brate, int food, int maxpop)
|
||||
babies(int adults, int etu, double brate, int food, int maxpop,
|
||||
int round_down)
|
||||
{
|
||||
int new_birth, new_food, new;
|
||||
double new_birth;
|
||||
int new_food, new;
|
||||
|
||||
if (adults >= maxpop)
|
||||
return 0;
|
||||
|
||||
new_birth = roundavg(brate * etu * adults);
|
||||
if (opt_NOFOOD)
|
||||
new_food = new_birth;
|
||||
else
|
||||
new_food = (int)(food / (2.0 * babyeat));
|
||||
new_birth = brate * etu * adults;
|
||||
new = round_down ? (int)new_birth : roundavg(new_birth);
|
||||
|
||||
if (!opt_NOFOOD) {
|
||||
new_food = (int)(food / (2.0 * babyeat));
|
||||
if (new > new_food)
|
||||
new = new_food;
|
||||
}
|
||||
|
||||
new = new_birth;
|
||||
if (new > new_food)
|
||||
new = new_food;
|
||||
if (adults + new > maxpop)
|
||||
new = maxpop - adults;
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue