/* * Empire - A multi-player, client/server Internet based war game. * Copyright (C) 1986-2016, Dave Pare, Jeff Bailey, Thomas Ruschak, * Ken Stevens, Steve McClure, Markus Armbruster * * Empire is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * --- * * See files README, COPYING and CREDITS in the root of the source * tree for related information and legal notices. It is expected * that future projects/authors will amend these files as needed. * * --- * * finish.c: Finish the update * * Known contributors to this file: * Dave Pare, 1986 * Thomas Ruschak, 1993 * Steve McClure, 1998 * Markus Armbruster, 2004-2016 */ #include #include #include #include "nat.h" #include "optlist.h" #include "path.h" #include "sect.h" #include "prototypes.h" #include "update.h" static void assemble_dist_paths(double *); void finish_sects(int etu) { static double *import_cost; struct sctstr *sp; int n; struct rusage rus1, rus2; if (import_cost == NULL) { logerror("First update since reboot, allocating buffer\n"); import_cost = malloc(WORLD_SZ() * sizeof(*import_cost)); if (import_cost == NULL) { logerror("malloc failed in finish_sects.\n"); return; } } logerror("delivering...\n"); /* Do deliveries */ for (n = 0; NULL != (sp = getsectid(n)); n++) { if (!sp->sct_own || sp->sct_type == SCT_SANCT) continue; if (nat_budget[sp->sct_own].money < 0) continue; dodeliver(sp); } logerror("done delivering\n"); logerror("assembling paths...\n"); getrusage(RUSAGE_SELF, &rus1); assemble_dist_paths(import_cost); getrusage(RUSAGE_SELF, &rus2); logerror("done assembling paths %g user %g system", rus2.ru_utime.tv_sec + rus2.ru_utime.tv_usec / 1e6 - (rus1.ru_utime.tv_sec + rus1.ru_utime.tv_usec / 1e6), rus2.ru_stime.tv_sec + rus2.ru_stime.tv_usec / 1e6 - (rus1.ru_stime.tv_sec + rus1.ru_stime.tv_usec / 1e6)); logerror("exporting..."); for (n = 0; NULL != (sp = getsectid(n)); n++) { if (!sp->sct_own || sp->sct_type == SCT_SANCT) continue; if (nat_budget[sp->sct_own].money < 0) continue; dodistribute(sp, EXPORT, import_cost[n]); } logerror("done exporting\n"); logerror("importing..."); for (n = 0; NULL != (sp = getsectid(n)); n++) { sp->sct_off = 0; if (!sp->sct_own || sp->sct_type == SCT_SANCT) continue; if (nat_budget[sp->sct_own].money < 0) continue; dodistribute(sp, IMPORT, import_cost[n]); } logerror("done importing\n"); } static int distcmp(const void *p, const void *q) { int a = *(int *)p; int b = *(int *)q; struct sctstr *sp = (void *)empfile[EF_SECTOR].cache; int d; d = sp[b].sct_dist_y - sp[a].sct_dist_y; if (d) return d; d = sp[b].sct_dist_x - sp[a].sct_dist_x; if (d) return d; return b - a; } static void assemble_dist_paths(double *import_cost) { struct sctstr *sp; struct sctstr *dist; int n; static int *job; int uid, i; coord dx = 1, dy = 0; /* invalid */ if (!job) job = malloc(WORLD_SZ() * sizeof(*job)); n = 0; for (uid = 0; NULL != (sp = getsectid(uid)); uid++) { import_cost[uid] = -1; if (sp->sct_dist_x == sp->sct_x && sp->sct_dist_y == sp->sct_y) continue; job[n++] = uid; } #ifdef PATH_FIND_STATS printf("dist path reuse %zu bytes, %d/%d used\n", WORLD_SZ() * sizeof(*job), n, WORLD_SZ()); #endif qsort(job, n, sizeof(*job), distcmp); for (i = 0; i < n; i++) { uid = job[i]; sp = getsectid(uid); dist = getsectp(sp->sct_dist_x, sp->sct_dist_y); if (CANT_HAPPEN(!dist)) continue; if (sp->sct_own != dist->sct_own) continue; if (sp->sct_dist_x != dx || sp->sct_dist_y != dy) { dx = sp->sct_dist_x; dy = sp->sct_dist_y; path_find_from(dx, dy, dist->sct_own, MOB_MOVE); } import_cost[uid] = path_find_to(sp->sct_x, sp->sct_y); } path_find_print_stats(); }