/* $NetBSD: hack.end.c,v 1.18 2020/02/07 20:34:18 fox Exp $ */ /* * Copyright (c) 1985, Stichting Centrum voor Wiskunde en Informatica, * Amsterdam * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * - Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * - Neither the name of the Stichting Centrum voor Wiskunde en * Informatica, nor the names of its contributors may be used to endorse or * promote products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /* * Copyright (c) 1982 Jay Fenlason * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #ifndef lint __RCSID("$NetBSD: hack.end.c,v 1.18 2020/02/07 20:34:18 fox Exp $"); #endif /* not lint */ #include #include #include #include "hack.h" #include "extern.h" #define Snprintf (void) snprintf xchar maxdlevel = 1; struct toptenentry; static void topten(void); static void outheader(void); static int outentry(int, struct toptenentry *, int); static char *itoa(int); static const char *ordin(int); int dodone(void) { done1(0); return 0; } /*ARGSUSED*/ void done1(int n __unused) { (void) signal(SIGINT, SIG_IGN); pline("Really quit?"); if (readchar() != 'y') { (void) signal(SIGINT, done1); clrlin(); (void) fflush(stdout); if (multi > 0) nomul(0); return; } done("quit"); /* NOTREACHED */ } static int done_stopprint; static int done_hup; /*ARGSUSED*/ static void done_intr(int n __unused) { done_stopprint++; (void) signal(SIGINT, SIG_IGN); (void) signal(SIGQUIT, SIG_IGN); } static void done_hangup(int n) { done_hup++; (void) signal(SIGHUP, SIG_IGN); done_intr(n); } void done_in_by(struct monst *mtmp) { static char buf[BUFSZ]; pline("You die ..."); if (mtmp->data->mlet == ' ') { Snprintf(buf, sizeof(buf), "the ghost of %s", (char *) mtmp->mextra); killer = buf; } else if (mtmp->mnamelth) { Snprintf(buf, sizeof(buf), "%s called %s", mtmp->data->mname, NAME(mtmp)); killer = buf; } else if (mtmp->minvis) { Snprintf(buf, sizeof(buf), "invisible %s", mtmp->data->mname); killer = buf; } else killer = mtmp->data->mname; done("died"); } /* * called with arg "died", "drowned", "escaped", "quit", "choked", * "panicked", "burned", "starved" or "tricked" */ /* Be careful not to call panic from here! */ void done(const char *st1) { #ifdef WIZARD if (wizard && *st1 == 'd') { u.uswldtim = 0; if (u.uhpmax < 0) u.uhpmax = 100; /* arbitrary */ u.uhp = u.uhpmax; pline("For some reason you are still alive."); flags.move = 0; if (multi > 0) multi = 0; else multi = -1; flags.botl = 1; return; } #endif /* WIZARD */ (void) signal(SIGINT, done_intr); (void) signal(SIGQUIT, done_intr); (void) signal(SIGHUP, done_hangup); if (*st1 == 'q' && u.uhp < 1) { st1 = "died"; killer = "quit while already on Charon's boat"; } if (*st1 == 's') killer = "starvation"; else if (*st1 == 'd' && st1[1] == 'r') killer = "drowning"; else if (*st1 == 'p') killer = "panic"; else if (*st1 == 't') killer = "trickery"; else if (!strchr("bcd", *st1)) killer = st1; paybill(); clearlocks(); if (flags.toplin == 1) more(); if (strchr("bcds", *st1)) { #ifdef WIZARD if (!wizard) #endif /* WIZARD */ savebones(); if (!flags.notombstone) outrip(); } if (*st1 == 'c') killer = st1; /* after outrip() */ settty(NULL); /* does a clear_screen() */ if (!done_stopprint) printf("Goodbye %s %s...\n\n", pl_character, plname); { long int tmp; tmp = u.ugold - u.ugold0; if (tmp < 0) tmp = 0; if (*st1 == 'd' || *st1 == 'b') tmp -= tmp / 10; u.urexp += tmp; u.urexp += 50 * maxdlevel; if (maxdlevel > 20) u.urexp += 1000 * ((maxdlevel > 30) ? 10 : maxdlevel - 20); } if (*st1 == 'e') { struct monst *mtmp; struct obj *otmp; int i; unsigned worthlessct = 0; boolean has_amulet = FALSE; killer = st1; keepdogs(); mtmp = mydogs; if (mtmp) { if (!done_stopprint) printf("You"); while (mtmp) { if (!done_stopprint) printf(" and %s", monnam(mtmp)); if (mtmp->mtame) u.urexp += mtmp->mhp; mtmp = mtmp->nmon; } if (!done_stopprint) printf("\nescaped from the dungeon with %ld points,\n", u.urexp); } else if (!done_stopprint) printf("You escaped from the dungeon with %ld points,\n", u.urexp); for (otmp = invent; otmp; otmp = otmp->nobj) { if (otmp->olet == GEM_SYM) { objects[otmp->otyp].oc_name_known = 1; i = otmp->quan * objects[otmp->otyp].g_val; if (i == 0) { worthlessct += otmp->quan; continue; } u.urexp += i; if (!done_stopprint) printf("\t%s (worth %d Zorkmids),\n", doname(otmp), i); } else if (otmp->olet == AMULET_SYM) { otmp->known = 1; i = (otmp->spe < 0) ? 2 : 5000; u.urexp += i; if (!done_stopprint) printf("\t%s (worth %d Zorkmids),\n", doname(otmp), i); if (otmp->spe >= 0) { has_amulet = TRUE; killer = "escaped (with amulet)"; } } } if (worthlessct) if (!done_stopprint) printf("\t%u worthless piece%s of coloured glass,\n", worthlessct, plur(worthlessct)); if (has_amulet) u.urexp *= 2; } else if (!done_stopprint) printf("You %s on dungeon level %d with %ld points,\n", st1, dlevel, u.urexp); if (!done_stopprint) printf("and %ld piece%s of gold, after %ld move%s.\n", u.ugold, plur(u.ugold), moves, plur(moves)); if (!done_stopprint) printf("You were level %u with a maximum of %d hit points when you %s.\n", u.ulevel, u.uhpmax, st1); if (*st1 == 'e' && !done_stopprint) { getret(); /* all those pieces of coloured glass ... */ cls(); } #ifdef WIZARD if (!wizard) #endif /* WIZARD */ topten(); if (done_stopprint) printf("\n\n"); exit(0); } #define newttentry() ((struct toptenentry *) alloc(sizeof(struct toptenentry))) #define NAMSZ 8 #define DTHSZ 40 #define PERSMAX 1 #define POINTSMIN 1 /* must be > 0 */ #define ENTRYMAX 100 /* must be >= 10 */ #define PERS_IS_UID /* delete for PERSMAX per name; now per uid */ struct toptenentry { struct toptenentry *tt_next; long int points; int level, maxlvl, hp, maxhp; int uid; char plchar; char sex; char name[NAMSZ + 1]; char death[DTHSZ + 1]; char date[7];/* yymmdd */ }; static struct toptenentry *tt_head; static void topten(void) { int uid = getuid(); int rank, rank0 = -1, rank1 = 0; int occ_cnt = PERSMAX; struct toptenentry *t0, *t1, *tprev; const char *recfile = RECORD; const char *reclock = "record_lock"; int sleepct = 300; FILE *rfile; int flg = 0; #define HUP if(!done_hup) while (link(recfile, reclock) == -1) { HUP perror(reclock); if (!sleepct--) { HUP puts("I give up. Sorry."); HUP puts("Perhaps there is an old record_lock around?"); return; } HUP printf("Waiting for access to record file. (%d)\n", sleepct); HUP(void) fflush(stdout); sleep(1); } if (!(rfile = fopen(recfile, "r"))) { HUP puts("Cannot open record file!"); goto unlock; } HUP(void) putchar('\n'); /* create a new 'topten' entry */ t0 = newttentry(); t0->level = dlevel; t0->maxlvl = maxdlevel; t0->hp = u.uhp; t0->maxhp = u.uhpmax; t0->points = u.urexp; t0->plchar = pl_character[0]; t0->sex = (flags.female ? 'F' : 'M'); t0->uid = uid; (void) strlcpy(t0->name, plname, NAMSZ); (t0->name)[NAMSZ] = 0; (void) strncpy(t0->death, killer, DTHSZ); (t0->death)[DTHSZ] = 0; (void) strcpy(t0->date, getdatestr()); /* assure minimum number of points */ if (t0->points < POINTSMIN) t0->points = 0; t1 = tt_head = newttentry(); tprev = 0; /* rank0: -1 undefined, 0 not_on_list, n n_th on list */ for (rank = 1;;) { if (fscanf(rfile, "%6s %d %d %d %d %d %ld %c%c %[^,],%[^\n]", t1->date, &t1->uid, &t1->level, &t1->maxlvl, &t1->hp, &t1->maxhp, &t1->points, &t1->plchar, &t1->sex, t1->name, t1->death) != 11 || t1->points < POINTSMIN) t1->points = 0; if (rank0 < 0 && t1->points < t0->points) { rank0 = rank++; if (tprev == 0) tt_head = t0; else tprev->tt_next = t0; t0->tt_next = t1; occ_cnt--; flg++; /* ask for a rewrite */ } else tprev = t1; if (t1->points == 0) break; if ( #ifdef PERS_IS_UID t1->uid == t0->uid && #else strncmp(t1->name, t0->name, NAMSZ) == 0 && #endif /* PERS_IS_UID */ t1->plchar == t0->plchar && --occ_cnt <= 0) { if (rank0 < 0) { rank0 = 0; rank1 = rank; HUP printf("You didn't beat your previous score of %ld points.\n\n", t1->points); } if (occ_cnt < 0) { flg++; continue; } } if (rank <= ENTRYMAX) { t1 = t1->tt_next = newttentry(); rank++; } if (rank > ENTRYMAX) { t1->points = 0; break; } } if (flg) { /* rewrite record file */ (void) fclose(rfile); if (!(rfile = fopen(recfile, "w"))) { HUP puts("Cannot write record file\n"); goto unlock; } if (!done_stopprint) if (rank0 > 0) { if (rank0 <= 10) puts("You made the top ten list!\n"); else printf("You reached the %d%s place on the top %d list.\n\n", rank0, ordin(rank0), ENTRYMAX); } } if (rank0 == 0) rank0 = rank1; if (rank0 <= 0) rank0 = rank; if (!done_stopprint) outheader(); t1 = tt_head; for (rank = 1; t1->points != 0; rank++, t1 = t1->tt_next) { if (flg) fprintf(rfile, "%6s %d %d %d %d %d %ld %c%c %s,%s\n", t1->date, t1->uid, t1->level, t1->maxlvl, t1->hp, t1->maxhp, t1->points, t1->plchar, t1->sex, t1->name, t1->death); if (done_stopprint) continue; if (rank > (int)flags.end_top && (rank < rank0 - (int)flags.end_around || rank > rank0 + (int)flags.end_around) && (!flags.end_own || #ifdef PERS_IS_UID t1->uid != t0->uid)) #else strncmp(t1->name, t0->name, NAMSZ))) #endif /* PERS_IS_UID */ continue; if (rank == rank0 - (int)flags.end_around && rank0 > (int)flags.end_top + (int)flags.end_around + 1 && !flags.end_own) (void) putchar('\n'); if (rank != rank0) (void) outentry(rank, t1, 0); else if (!rank1) (void) outentry(rank, t1, 1); else { int t0lth = outentry(0, t0, -1); int t1lth = outentry(rank, t1, t0lth); if (t1lth > t0lth) t0lth = t1lth; (void) outentry(0, t0, t0lth); } } if (rank0 >= rank) if (!done_stopprint) (void) outentry(0, t0, 1); (void) fclose(rfile); free(t0); unlock: (void) unlink(reclock); } static void outheader(void) { char linebuf[BUFSZ]; char *bp; (void) strcpy(linebuf, "Number Points Name"); bp = eos(linebuf); while (bp < linebuf + COLNO - 9) *bp++ = ' '; (void) strcpy(bp, "Hp [max]"); puts(linebuf); } /* so>0: standout line; so=0: ordinary line; so<0: no output, return length */ static int outentry(int rank, struct toptenentry *t1, int so) { boolean quit = FALSE, gotkilled = FALSE, starv = FALSE; char linebuf[BUFSZ]; size_t pos; linebuf[0] = '\0'; pos = 0; if (rank) Snprintf(linebuf+pos, sizeof(linebuf)-pos, "%3d", rank); else Snprintf(linebuf+pos, sizeof(linebuf)-pos, " "); pos = strlen(linebuf); Snprintf(linebuf+pos, sizeof(linebuf)-pos, " %6ld %8s", t1->points, t1->name); pos = strlen(linebuf); if (t1->plchar == 'X') Snprintf(linebuf+pos, sizeof(linebuf)-pos, " "); else Snprintf(linebuf+pos, sizeof(linebuf)-pos, "-%c ", t1->plchar); pos = strlen(linebuf); if (!strncmp("escaped", t1->death, 7)) { if (!strcmp(" (with amulet)", t1->death + 7)) Snprintf(linebuf+pos, sizeof(linebuf)-pos, "escaped the dungeon with amulet"); else Snprintf(linebuf+pos, sizeof(linebuf)-pos, "escaped the dungeon [max level %d]", t1->maxlvl); pos = strlen(linebuf); } else { if (!strncmp(t1->death, "quit", 4)) { quit = TRUE; if (t1->maxhp < 3 * t1->hp && t1->maxlvl < 4) Snprintf(linebuf+pos, sizeof(linebuf)-pos, "cravenly gave up"); else Snprintf(linebuf+pos, sizeof(linebuf)-pos, "quit"); } else if (!strcmp(t1->death, "choked")) { Snprintf(linebuf+pos, sizeof(linebuf)-pos, "choked on %s food", (t1->sex == 'F') ? "her" : "his"); } else if (!strncmp(t1->death, "starv", 5)) { Snprintf(linebuf+pos, sizeof(linebuf)-pos, "starved to death"); starv = TRUE; } else { Snprintf(linebuf+pos, sizeof(linebuf)-pos, "was killed"); gotkilled = TRUE; } pos = strlen(linebuf); Snprintf(linebuf+pos, sizeof(linebuf)-pos, " on%s level %d", (gotkilled || starv) ? "" : " dungeon", t1->level); pos = strlen(linebuf); if (t1->maxlvl != t1->level) Snprintf(linebuf+pos, sizeof(linebuf)-pos, " [max %d]", t1->maxlvl); pos = strlen(linebuf); if (quit && t1->death[4]) Snprintf(linebuf+pos, sizeof(linebuf)-pos, "%s", t1->death + 4); pos = strlen(linebuf); } if (gotkilled) { Snprintf(linebuf+pos, sizeof(linebuf)-pos, " by %s%s", (!strncmp(t1->death, "trick", 5) || !strncmp(t1->death, "the ", 4)) ? "" : strchr(vowels, *t1->death) ? "an " : "a ", t1->death); pos = strlen(linebuf); } strlcat(linebuf, ".", sizeof(linebuf)); pos = strlen(linebuf); if (t1->maxhp) { char hpbuf[10]; unsigned hppos; strlcpy(hpbuf, (t1->hp > 0) ? itoa(t1->hp) : "-", sizeof(hpbuf)); hppos = COLNO - 7 - strlen(hpbuf); if (pos <= hppos) { while (pos < hppos) linebuf[pos++] = ' '; (void) strlcpy(linebuf+pos, hpbuf, sizeof(linebuf)-pos); pos = strlen(linebuf); Snprintf(linebuf+pos, sizeof(linebuf)-pos, " [%d]", t1->maxhp); pos = strlen(linebuf); } } if (so == 0) puts(linebuf); else if (so > 0) { if (so >= COLNO) so = COLNO - 1; while (pos < (unsigned)so) linebuf[pos++] = ' '; linebuf[pos] = '\0'; standoutbeg(); fputs(linebuf, stdout); standoutend(); (void) putchar('\n'); } return /*(strlen(linebuf))*/ pos; } static char * itoa(int a) { static char buf[12]; Snprintf(buf, sizeof(buf), "%d", a); return (buf); } static const char * ordin(int n) { int dg = n % 10; return ((dg == 0 || dg > 3 || n / 10 == 1) ? "th" : (dg == 1) ? "st" : (dg == 2) ? "nd" : "rd"); } void clearlocks(void) { int x; (void) signal(SIGHUP, SIG_IGN); for (x = maxdlevel; x >= 0; x--) { glo(x); (void) unlink(lock); /* not all levels need be present */ } } #ifdef NOSAVEONHANGUP /*ARGSUSED*/ void hang_up(int n __unused) { (void) signal(SIGINT, SIG_IGN); clearlocks(); exit(1); } #endif /* NOSAVEONHANGUP */ char * eos(char *s) { while (*s) s++; return (s); } /* it is the callers responsibility to check that there is room for c */ void charcat(char *s, int c) { while (*s) s++; *s++ = c; *s = 0; } /* * Called with args from main if argc >= 0. In this case, list scores as * requested. Otherwise, find scores for the current player (and list them * if argc == -1). */ void prscore(int argc, char **argv) { char **players = NULL; int playerct; int rank; struct toptenentry *t1, *t2; const char *recfile = RECORD; FILE *rfile; int flg = 0; int i; #ifdef nonsense long total_score = 0L; char totchars[10]; int totcharct = 0; #endif /* nonsense */ int outflg = (argc >= -1); #ifdef PERS_IS_UID int uid = -1; #else char *player0; #endif /* PERS_IS_UID */ if (!(rfile = fopen(recfile, "r"))) { puts("Cannot open record file!"); return; } if (argc > 1 && !strncmp(argv[1], "-s", 2)) { if (!argv[1][2]) { argc--; argv++; } else if (!argv[1][3] && strchr("CFKSTWX", argv[1][2])) { argv[1]++; argv[1][0] = '-'; } else argv[1] += 2; } if (argc <= 1) { #ifdef PERS_IS_UID uid = getuid(); playerct = 0; #else player0 = plname; if (!*player0) player0 = "hackplayer"; playerct = 1; players = &player0; #endif /* PERS_IS_UID */ } else { playerct = --argc; players = ++argv; } if (outflg) putchar('\n'); t1 = tt_head = newttentry(); for (rank = 1;; rank++) { if (fscanf(rfile, "%6s %d %d %d %d %d %ld %c%c %[^,],%[^\n]", t1->date, &t1->uid, &t1->level, &t1->maxlvl, &t1->hp, &t1->maxhp, &t1->points, &t1->plchar, &t1->sex, t1->name, t1->death) != 11) t1->points = 0; if (t1->points == 0) break; #ifdef PERS_IS_UID if (!playerct && t1->uid == uid) flg++; else #endif /* PERS_IS_UID */ for (i = 0; i < playerct; i++) { if (strcmp(players[i], "all") == 0 || strncmp(t1->name, players[i], NAMSZ) == 0 || (players[i][0] == '-' && players[i][1] == t1->plchar && players[i][2] == 0) || (digit(players[i][0]) && rank <= atoi(players[i]))) flg++; } t1 = t1->tt_next = newttentry(); } (void) fclose(rfile); if (!flg) { if (outflg) { printf("Cannot find any entries for "); if (playerct < 1) printf("you.\n"); else { if (playerct > 1) printf("any of "); for (i = 0; i < playerct; i++) printf("%s%s", players[i], (i < playerct - 1) ? ", " : ".\n"); printf("Call is: %s -s [playernames]\n", hname); } } return; } if (outflg) outheader(); t1 = tt_head; for (rank = 1; t1->points != 0; rank++, t1 = t2) { t2 = t1->tt_next; #ifdef PERS_IS_UID if (!playerct && t1->uid == uid) goto outwithit; else #endif /* PERS_IS_UID */ for (i = 0; i < playerct; i++) { if (strcmp(players[i], "all") == 0 || strncmp(t1->name, players[i], NAMSZ) == 0 || (players[i][0] == '-' && players[i][1] == t1->plchar && players[i][2] == 0) || (digit(players[i][0]) && rank <= atoi(players[i]))) { outwithit: if (outflg) (void) outentry(rank, t1, 0); #ifdef nonsense total_score += t1->points; if (totcharct < sizeof(totchars) - 1) totchars[totcharct++] = t1->plchar; #endif /* nonsense */ break; } } free(t1); } #ifdef nonsense totchars[totcharct] = 0; /* * We would like to determine whether he is experienced. However, the * information collected here only tells about the scores/roles that * got into the topten (top 100?). We should maintain a .hacklog or * something in his home directory. */ flags.beginner = (total_score < 6000); for (i = 0; i < 6; i++) if (!strchr(totchars, "CFKSTWX"[i])) { flags.beginner = 1; if (!pl_character[0]) pl_character[0] = "CFKSTWX"[i]; break; } #endif /* nonsense */ }