/* Low-level support for variables and properties in Xconq GDL. Copyright (C) 1991-1996, 1998-2000 Stanley T. Shebs. Xconq 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 2, or (at your option) any later version. See the file COPYING. */ #include "config.h" #include "misc.h" #include "lisp.h" #include "game.h" /* The total number of unit/material/terrain/advance types. */ short numutypes; short nummtypes; short numttypes; short numatypes; /* These variables indicate whether new types can still be defined. Once a table or list of types is manipulated, these are turned off. */ short canaddutype = TRUE; short canaddmtype = TRUE; short canaddttype = TRUE; short canaddatype = TRUE; short first_bord_type; short *next_bord_type; short first_conn_type; short *next_conn_type; /* It would be useful for these to mention the slot or table access that resulted in the error, but we don't want to have to pass a bunch of parameters to these routines, since they're referenced a lot. */ void utype_error(int u) { run_warning("Bad utype %d", u); } void mtype_error(int m) { run_warning("Bad mtype %d", m); } void ttype_error(int t) { run_warning("Bad ttype %d", t); } void atype_error(int a) { run_warning("Bad atype %d", a); } /* Declarations of the type definitions themselves. */ Utype *utypes; Mtype *mtypes; Ttype *ttypes; Atype *atypes; Globals globals; /* This prepares the type definitions to be filled in, doing initial allocations, etc. */ int curmaxutypes = MAXUTYPES; int curmaxmtypes = MAXMTYPES; int curmaxttypes = MAXTTYPES; int curmaxatypes = MAXATYPES; void init_types(void) { numutypes = nummtypes = numttypes = numatypes = 0; Dprintf("Utype is %d bytes, mtype is %d bytes, ttype is %d bytes, atype is %d bytes.\n", sizeof(Utype), sizeof(Mtype), sizeof(Ttype), sizeof(Atype)); utypes = (Utype *) xmalloc(sizeof(Utype) * curmaxutypes); mtypes = (Mtype *) xmalloc(sizeof(Mtype) * curmaxmtypes); ttypes = (Ttype *) xmalloc(sizeof(Ttype) * curmaxttypes); atypes = (Atype *) xmalloc(sizeof(Atype) * curmaxatypes); } VarDefn vardefns[] = { #undef DEF_VAR_I #define DEF_VAR_I(NAME,FNAME,SETFNAME,DOC,var,LO,DFLT,HI) \ { NAME, FNAME, NULL, NULL, SETFNAME, NULL, NULL, DOC, DFLT, NULL, NULL, LO, HI }, #undef DEF_VAR_S #define DEF_VAR_S(NAME,FNAME,SETFNAME,DOC,var,DFLT) \ { NAME, NULL, FNAME, NULL, NULL, SETFNAME, NULL, DOC, 0, DFLT, NULL, 0, 0 }, #undef DEF_VAR_L #define DEF_VAR_L(NAME,FNAME,SETFNAME,DOC,var,DFLT) \ { NAME, NULL, NULL, FNAME, NULL, NULL, SETFNAME, DOC, 0, NULL, DFLT, 0, 0 }, #include "gvar.def" { NULL } }; /* Define all the global-getting and -setting functions. */ #undef DEF_VAR_I #define DEF_VAR_I(str,FNAME,SETFNAME,doc,VAR,lo,dflt,hi) \ int FNAME(void) { return globals.VAR; } \ void SETFNAME(int v) { globals.VAR = v; } #undef DEF_VAR_S #define DEF_VAR_S(str,FNAME,SETFNAME,doc,VAR,dflt) \ char *FNAME(void) { return globals.VAR; } \ void SETFNAME(char *v) { globals.VAR = v; } #undef DEF_VAR_L #define DEF_VAR_L(str,FNAME,SETFNAME,doc,VAR,DFLT) \ Obj *FNAME(void) { return globals.VAR; } \ void SETFNAME(Obj *v) { globals.VAR = v; } #include "gvar.def" /* Set the globals to their default values. For Lisp values, the default is an initialization function rather than a value. */ void init_globals(void) { #undef DEF_VAR_I #define DEF_VAR_I(str,fname,SETFNAME,doc,var,lo,DFLT,hi) \ SETFNAME(DFLT); #undef DEF_VAR_S #define DEF_VAR_S(str,fname,SETFNAME,doc,var,DFLT) \ SETFNAME(DFLT); #undef DEF_VAR_L #define DEF_VAR_L(str,fname,SETFNAME,doc,var,DFLT) \ if ((DFLT) != NULL) { \ void (*fn)(void) = (DFLT); \ if (fn != NULL) (*fn)(); \ } else { \ SETFNAME(lispnil); \ } #include "gvar.def" } #define TYPEPROP(TYPES, N, DEFNS, I, TYPE) \ ((TYPE *) &(((char *) (&(TYPES[N])))[DEFNS[I].offset]))[0] /* This sets all the defaults in a unit type definition. Note that all type structures get blasted with zeros initially, so we really only need to do default settings of nonzero values, thus the test. (It helps if the compiler is smart enough to remove dead code.) */ void default_unit_type(int u) { int i; for (i = 0; utypedefns[i].name != NULL; ++i) { if (utypedefns[i].intgetter) { if (utypedefns[i].dflt != 0) TYPEPROP(utypes, u, utypedefns, i, short) = utypedefns[i].dflt; } else if (utypedefns[i].strgetter) { if (utypedefns[i].dfltstr != NULL) TYPEPROP(utypes, u, utypedefns, i, char *) = (char *) utypedefns[i].dfltstr; } else { TYPEPROP(utypes, u, utypedefns, i, Obj *) = lispnil; } } } /* This sets all the defaults in a material type definition. */ void default_material_type(int m) { int i; for (i = 0; mtypedefns[i].name != NULL; ++i) { if (mtypedefns[i].intgetter) { if (mtypedefns[i].dflt != 0) TYPEPROP(mtypes, m, mtypedefns, i, short) = mtypedefns[i].dflt; } else if (mtypedefns[i].strgetter) { if (mtypedefns[i].dfltstr != NULL) TYPEPROP(mtypes, m, mtypedefns, i, char *) = (char *) mtypedefns[i].dfltstr; } else { TYPEPROP(mtypes, m, mtypedefns, i, Obj *) = lispnil; } } } /* This sets all the defaults in a terrain type definition. */ void default_terrain_type(int t) { int i; for (i = 0; ttypedefns[i].name != NULL; ++i) { if (ttypedefns[i].intgetter) { if (ttypedefns[i].dflt != 0) TYPEPROP(ttypes, t, ttypedefns, i, short) = ttypedefns[i].dflt; } else if (ttypedefns[i].strgetter) { if (ttypedefns[i].dfltstr != 0) TYPEPROP(ttypes, t, ttypedefns, i, char *) = (char *) ttypedefns[i].dfltstr; } else { TYPEPROP(ttypes, t, ttypedefns, i, Obj *) = lispnil; } } } /* This sets all the defaults in an advance type definition. */ void default_advance_type(int a) { int i; for (i = 0; atypedefns[i].name != NULL; ++i) { if (atypedefns[i].intgetter) { if (atypedefns[i].dflt != 0) TYPEPROP(atypes, a, atypedefns, i, short) = atypedefns[i].dflt; } else if (atypedefns[i].strgetter) { if (atypedefns[i].dfltstr != 0) TYPEPROP(atypes, a, atypedefns, i, char *) = (char *) atypedefns[i].dfltstr; } else { TYPEPROP(atypes, a, atypedefns, i, Obj *) = lispnil; } } } char * index_type_name(int x) { return ((x) == UTYP ? "unit" : ((x) == MTYP ? "material" : ((x) == TTYP ? "terrain" : "advance"))); } /* This function allocates a parameter table and fills it with a default. */ void allocate_table(int tbl, int reset) { int i, lim1, lim2, dflt = tabledefns[tbl].dflt; short *rslt; if (reset) *(tabledefns[tbl].table) = NULL; if (*tabledefns[tbl].table == NULL) { lim1 = numtypes_from_index_type(tabledefns[tbl].index1); lim2 = numtypes_from_index_type(tabledefns[tbl].index2); if (lim1 == 0) { run_warning("Can't allocate the %s table, no %s types defined", tabledefns[tbl].name, index_type_name(tabledefns[tbl].index1)); return; } if (lim2 == 0) { run_warning("Can't allocate the %s table, no %s types defined", tabledefns[tbl].name, index_type_name(tabledefns[tbl].index2)); return; } /* Allocate the table itself. */ rslt = (short *) xmalloc(lim1 * lim2 * sizeof(short)); /* Put the table's default everywhere in the table. */ for (i = 0; i < lim1 * lim2; ++i) rslt[i] = dflt; *(tabledefns[tbl].table) = rslt; /* For each index, flag that no more types of that sort allowed. */ switch (tabledefns[tbl].index1) { case UTYP: canaddutype = FALSE; break; case MTYP: canaddmtype = FALSE; break; case TTYP: canaddttype = FALSE; break; case ATYP: canaddatype = FALSE; break; } switch (tabledefns[tbl].index2) { case UTYP: canaddutype = FALSE; break; case MTYP: canaddmtype = FALSE; break; case TTYP: canaddttype = FALSE; break; case ATYP: canaddatype = FALSE; break; } } } int numtypes_from_index_type(int x) { return ((x) == UTYP ? numutypes : ((x) == MTYP ? nummtypes : ((x) == TTYP ? numttypes : numatypes))); }