1 1.16 rillig /* $NetBSD: vocab.c,v 1.16 2021/05/02 12:50:43 rillig Exp $ */ 2 1.2 cgd 3 1.1 jtc /*- 4 1.1 jtc * Copyright (c) 1991, 1993 5 1.1 jtc * The Regents of the University of California. All rights reserved. 6 1.1 jtc * 7 1.1 jtc * The game adventure was originally written in Fortran by Will Crowther 8 1.1 jtc * and Don Woods. It was later translated to C and enhanced by Jim 9 1.1 jtc * Gillogly. This code is derived from software contributed to Berkeley 10 1.1 jtc * by Jim Gillogly at The Rand Corporation. 11 1.1 jtc * 12 1.1 jtc * Redistribution and use in source and binary forms, with or without 13 1.1 jtc * modification, are permitted provided that the following conditions 14 1.1 jtc * are met: 15 1.1 jtc * 1. Redistributions of source code must retain the above copyright 16 1.1 jtc * notice, this list of conditions and the following disclaimer. 17 1.1 jtc * 2. Redistributions in binary form must reproduce the above copyright 18 1.1 jtc * notice, this list of conditions and the following disclaimer in the 19 1.1 jtc * documentation and/or other materials provided with the distribution. 20 1.11 agc * 3. Neither the name of the University nor the names of its contributors 21 1.1 jtc * may be used to endorse or promote products derived from this software 22 1.1 jtc * without specific prior written permission. 23 1.1 jtc * 24 1.1 jtc * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 25 1.1 jtc * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 26 1.1 jtc * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 27 1.1 jtc * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 28 1.1 jtc * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 29 1.1 jtc * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 30 1.1 jtc * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 31 1.1 jtc * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 32 1.1 jtc * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 33 1.1 jtc * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 34 1.1 jtc * SUCH DAMAGE. 35 1.1 jtc */ 36 1.1 jtc 37 1.3 christos #include <sys/cdefs.h> 38 1.1 jtc #ifndef lint 39 1.2 cgd #if 0 40 1.1 jtc static char sccsid[] = "@(#)vocab.c 8.1 (Berkeley) 5/31/93"; 41 1.2 cgd #else 42 1.16 rillig __RCSID("$NetBSD: vocab.c,v 1.16 2021/05/02 12:50:43 rillig Exp $"); 43 1.2 cgd #endif 44 1.5 lukem #endif /* not lint */ 45 1.1 jtc 46 1.1 jtc /* Re-coding of advent in C: data structure routines */ 47 1.1 jtc 48 1.8 hubertf #include <err.h> 49 1.3 christos #include <stdio.h> 50 1.3 christos #include <stdlib.h> 51 1.3 christos #include "hdr.h" 52 1.3 christos #include "extern.h" 53 1.1 jtc 54 1.3 christos void 55 1.15 dholland destroy(int object) 56 1.5 lukem { 57 1.5 lukem move(object, 0); 58 1.1 jtc } 59 1.1 jtc 60 1.3 christos void 61 1.12 jmc juggle(int object) 62 1.5 lukem { 63 1.5 lukem int i, j; 64 1.5 lukem 65 1.5 lukem i = place[object]; 66 1.5 lukem j = fixed[object]; 67 1.5 lukem move(object, i); 68 1.5 lukem move(object + 100, j); 69 1.1 jtc } 70 1.1 jtc 71 1.3 christos void 72 1.12 jmc move(int object, int where) 73 1.5 lukem { 74 1.5 lukem int from; 75 1.1 jtc 76 1.5 lukem if (object <= 100) 77 1.5 lukem from = place[object]; 78 1.1 jtc else 79 1.5 lukem from = fixed[object - 100]; 80 1.5 lukem if (from > 0 && from <= 300) 81 1.5 lukem carry(object, from); 82 1.5 lukem drop(object, where); 83 1.1 jtc } 84 1.1 jtc 85 1.3 christos int 86 1.12 jmc put(int object, int where, int pval) 87 1.5 lukem { 88 1.5 lukem move(object, where); 89 1.5 lukem return (-1 - pval); 90 1.1 jtc } 91 1.1 jtc 92 1.3 christos void 93 1.12 jmc carry(int object, int where) 94 1.5 lukem { 95 1.5 lukem int temp; 96 1.5 lukem 97 1.5 lukem if (object <= 100) { 98 1.5 lukem if (place[object] == -1) 99 1.5 lukem return; 100 1.1 jtc place[object] = -1; 101 1.15 dholland holding++; 102 1.1 jtc } 103 1.5 lukem if (atloc[where] == object) { 104 1.5 lukem atloc[where] = links[object]; 105 1.1 jtc return; 106 1.1 jtc } 107 1.5 lukem for (temp = atloc[where]; links[temp] != object; temp = links[temp]); 108 1.5 lukem links[temp] = links[object]; 109 1.1 jtc } 110 1.1 jtc 111 1.1 jtc 112 1.3 christos void 113 1.12 jmc drop(int object, int where) 114 1.5 lukem { 115 1.5 lukem if (object > 100) 116 1.5 lukem fixed[object - 100] = where; 117 1.5 lukem else { 118 1.5 lukem if (place[object] == -1) 119 1.15 dholland holding--; 120 1.5 lukem place[object] = where; 121 1.1 jtc } 122 1.5 lukem if (where <= 0) 123 1.5 lukem return; 124 1.5 lukem links[object] = atloc[where]; 125 1.5 lukem atloc[where] = object; 126 1.1 jtc } 127 1.1 jtc 128 1.12 jmc /* look up or store a word */ 129 1.12 jmc /* -2 for store, -1 for user word, >=0 for canned lookup */ 130 1.12 jmc /* used for storing only */ 131 1.3 christos int 132 1.12 jmc vocab(const char *word, int type, int value) 133 1.5 lukem { 134 1.5 lukem int adr; 135 1.7 hubertf const char *s; 136 1.7 hubertf char *t; 137 1.5 lukem int hash, i; 138 1.1 jtc struct hashtab *h; 139 1.1 jtc 140 1.12 jmc for (hash = 0, s = word, i = 0; i < 5 && *s; i++) /* some kind of hash*/ 141 1.5 lukem hash += *s++; /* add all chars in the word */ 142 1.5 lukem hash = (hash * 3719) & 077777; /* pulled that one out of a hat */ 143 1.5 lukem hash %= HTSIZE; /* put it into range of table */ 144 1.5 lukem 145 1.5 lukem for (adr = hash;; adr++) { /* look for entry in table */ 146 1.5 lukem if (adr == HTSIZE) 147 1.5 lukem adr = 0;/* wrap around */ 148 1.5 lukem h = &voc[adr]; /* point at the entry */ 149 1.5 lukem switch (type) { 150 1.5 lukem case -2: /* fill in entry */ 151 1.5 lukem if (h->val) /* already got an entry? */ 152 1.1 jtc goto exitloop2; 153 1.5 lukem h->val = value; 154 1.5 lukem h->atab = malloc(length(word)); 155 1.8 hubertf if (h->atab == NULL) 156 1.10 jsm err(1, NULL); 157 1.5 lukem for (s = word, t = h->atab; *s;) 158 1.1 jtc *t++ = *s++ ^ '='; 159 1.5 lukem *t = 0 ^ '='; 160 1.1 jtc /* encrypt slightly to thwart core reader */ 161 1.5 lukem /* printf("Stored \"%s\" (%d ch) as entry %d\n", */ 162 1.5 lukem /* word, length(word), adr); */ 163 1.5 lukem return (0); /* entry unused */ 164 1.5 lukem case -1: /* looking up user word */ 165 1.5 lukem if (h->val == 0) 166 1.5 lukem return (-1); /* not found */ 167 1.5 lukem for (s = word, t = h->atab; *t ^ '=';) 168 1.1 jtc if ((*s++ ^ '=') != *t++) 169 1.1 jtc goto exitloop2; 170 1.5 lukem if ((*s ^ '=') != *t && s - word < 5) 171 1.5 lukem goto exitloop2; 172 1.1 jtc /* the word matched o.k. */ 173 1.5 lukem return (h->val); 174 1.5 lukem default: /* looking up known word */ 175 1.9 hubertf if (h->val == 0) 176 1.9 hubertf errx(1,"Unable to find %s in vocab", word); 177 1.5 lukem for (s = word, t = h->atab; *t ^ '=';) 178 1.5 lukem if ((*s++ ^ '=') != *t++) 179 1.5 lukem goto exitloop2; 180 1.1 jtc /* the word matched o.k. */ 181 1.5 lukem if (h->val / 1000 != type) 182 1.5 lukem continue; 183 1.5 lukem return (h->val % 1000); 184 1.1 jtc } 185 1.1 jtc 186 1.5 lukem exitloop2: /* hashed entry does not match */ 187 1.13 christos if (adr + 1 == hash || hash == 0) 188 1.9 hubertf errx(1,"Hash table overflow"); 189 1.1 jtc } 190 1.1 jtc } 191 1.1 jtc 192 1.14 dholland /* print hash table (for debugging) */ 193 1.14 dholland static __unused void 194 1.12 jmc prht(void) 195 1.16 rillig { 196 1.5 lukem int i, j, l; 197 1.5 lukem char *c; 198 1.1 jtc struct hashtab *h; 199 1.5 lukem for (i = 0; i < HTSIZE / 10 + 1; i++) { 200 1.5 lukem printf("%4d", i * 10); 201 1.5 lukem for (j = 0; j < 10; j++) { 202 1.5 lukem if (i * 10 + j >= HTSIZE) 203 1.5 lukem break; 204 1.5 lukem h = &voc[i * 10 + j]; 205 1.1 jtc putchar(' '); 206 1.5 lukem if (h->val == 0) { 207 1.5 lukem printf("-----"); 208 1.1 jtc continue; 209 1.1 jtc } 210 1.5 lukem for (l = 0, c = h->atab; l < 5; l++) 211 1.5 lukem if ((*c ^ '=')) 212 1.5 lukem putchar(*c++ ^ '='); 213 1.5 lukem else 214 1.5 lukem putchar(' '); 215 1.1 jtc } 216 1.1 jtc putchar('\n'); 217 1.1 jtc } 218 1.1 jtc } 219