/*
* Empire - A multi-player, client/server Internet based war game.
- * Copyright (C) 1986-2000, Dave Pare, Jeff Bailey, Thomas Ruschak,
- * Ken Stevens, Steve McClure
+ * Copyright (C) 1986-2014, Dave Pare, Jeff Bailey, Thomas Ruschak,
+ * Ken Stevens, Steve McClure, Markus Armbruster
*
- * This program is free software; you can redistribute it and/or modify
+ * 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 2 of the License, or
+ * 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,
* 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, write to the Free Software
- * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* ---
*
- * See the "LEGAL", "LICENSE", "CREDITS" and "README" files for all the
- * related information and legal notices. It is expected that any future
- * projects/authors will amend these files as needed.
+ * 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.
*
* ---
*
* nsc.h: Definitions for Empire conditionals
- *
+ *
* Known contributors to this file:
* Dave Pare, 1989
+ * Markus Armbruster, 2004-2010
*/
-#ifndef _NSC_H_
-#define _NSC_H_
+#ifndef NSC_H
+#define NSC_H
+#include <stddef.h>
+#include "misc.h"
#include "xy.h"
#define NS_LSIZE 128
#define NS_NCOND 16
-enum { /* masks for struct nscstr member oper */
- NSC_OPMASK = 0x0ff, /* operator */
- NSC_ISNUM1 = 0x100, /* is left operand a number? */
- NSC_ISNUM2 = 0x200 /* is right operand a number? */
+/* Value type */
+enum nsc_type {
+ NSC_NOTYPE,
+ /* promoted types */
+ NSC_LONG, /* long */
+ NSC_DOUBLE, /* double */
+ NSC_STRING, /* character string */
+ /* unpromoted types */
+ NSC_CHAR, /* signed char */
+ NSC_UCHAR, /* unsigned char */
+ NSC_SHORT, /* short */
+ NSC_USHORT, /* unsigned short */
+ NSC_INT, /* int */
+ NSC_XCOORD, /* coord that needs x conversion */
+ NSC_YCOORD, /* coord that needs y conversion */
+ NSC_HIDDEN, /* unsigned char in struct natstr that
+ may need hiding */
+ NSC_TIME, /* time_t */
+ NSC_FLOAT, /* float */
+ NSC_STRINGY, /* char[] */
+ /* aliases, must match typedefs */
+ NSC_NATID = NSC_UCHAR /* nation id */
+};
+
+/* Is TYPE a promoted value type? */
+#define NSC_IS_PROMOTED(type) (NSC_LONG <= (type) && (type) <= NSC_STRING)
+
+/* Return nsc_type for a signed integer with the same size as TYPE. */
+#define NSC_SITYPE(type) \
+ (sizeof(type) == 1 ? NSC_CHAR \
+ : sizeof(type) == sizeof(short) ? NSC_SHORT \
+ : sizeof(type) == sizeof(int) ? NSC_INT \
+ : sizeof(type) == sizeof(long) ? NSC_LONG \
+ : 1/0)
+
+/* Value category */
+enum nsc_cat {
+ NSC_NOCAT,
+ NSC_VAL, /* evaluated value */
+ NSC_OFF, /* symbolic value: at offset in object */
+ NSC_ID /* unresolved identifier (internal use) */
+};
+
+/*
+ * Value, possibly symbolic
+ *
+ * If type is NSC_NOTYPE, it's an error value.
+ * If category is NSC_VAL, the value is in val_as, and the type is a
+ * promoted type.
+ * If category is NSC_OFF, the value is in a context object, and
+ * val_as.sym specifies how to get it, as follows.
+ * If sym.get is null, and type is NSC_STRINGY, the value is an array
+ * of sym.len characters starting at sym.offs in the context object.
+ * sym.idx must be zero.
+ * Else if sym.get is null, and sym.len is zero, the value is in the
+ * context object at offset sym.off. sym.idx must be zero.
+ * Else if sym.get is null, sym.len is non-zero, and the value has
+ * index sym.idx in an array of sym.len elements at offset sym.off in
+ * in the context object. I.e. the value is at sym.off + sym.idx *
+ * SZ, where SZ is the size of the value.
+ * If sym.get is not null, you obtain the value by calling get() like
+ * VAL->get(VAL, NP, CTXO), where NP points to the country to use for
+ * coordinate translation and access control (null for none), and CTXO
+ * is the context object. get() either returns a null pointer and
+ * sets VAL->val_as to the value, as appropriate for the type. Or it
+ * returns another context object and sets VAL->val_as.sym for it.
+ */
+struct valstr {
+ enum nsc_type val_type; /* type of value */
+ enum nsc_cat val_cat; /* category of value */
+ union {
+ struct { /* cat NSC_OFF */
+ ptrdiff_t off;
+ int len;
+ int idx;
+ void *(*get)(struct valstr *, struct natstr *, void *);
+ } sym;
+ double dbl; /* cat NSC_VAL, type NSC_DOUBLE */
+ struct { /* cat NSC_VAL, type NSC_STRING, cat NSC_ID */
+ char *base;
+ size_t maxsz;
+ } str;
+ long lng; /* cat NSC_VAL, type NSC_LONG */
+ } val_as;
};
+/* Compiled condition */
struct nscstr {
- long fld1; /* first commodity or number */
- long fld2; /* second commodity or number */
- int oper; /* required relationship operator */
+ char operator; /* '<', '=', '>', '#' */
+ enum nsc_type optype; /* operator type */
+ struct valstr lft; /* left operand */
+ struct valstr rgt; /* right operand */
+};
+
+/* Selection type */
+enum ns_seltype {
+ NS_UNDEF, /* error value */
+ NS_LIST, /* list of IDs */
+ NS_DIST, /* circular area */
+ NS_AREA, /* rectangular area */
+ NS_ALL, /* everything */
+ NS_XY, /* one sector area */
+ NS_GROUP, /* group, i.e. fleet, wing, army */
+ NS_CARGO /* loaded on the same carrier */
};
+/* Sector iterator */
struct nstr_sect {
coord x, y; /* current x-y */
coord dx, dy; /* accumlated x,y travel */
int id; /* return value of sctoff */
- int type; /* type of query */
- int curdist; /* dist query: current range */
+ enum ns_seltype type; /* selection type: NS_AREA or NS_DIST */
+ int curdist; /* NS_DIST: current range */
struct range range; /* area of coverage */
- int dist; /* dist query: range */
- coord cx, cy; /* dist query: center x-y */
- int (*read)(int type, int id, caddr_t ptr); /* read function */
+ int dist; /* NS_DIST: range */
+ coord cx, cy; /* NS_DIST: center x-y */
int ncond; /* # of selection conditions */
- struct nscstr cond[NS_NCOND]; /* selection conditions */
+ struct nscstr cond[NS_NCOND]; /* selection conditions */
};
+/* Item iterator */
struct nstr_item {
int cur; /* current item */
- int sel; /* selection type */
+ enum ns_seltype sel; /* selection type, any but NS_UNDEF */
int type; /* item type being selected */
int curdist; /* if NS_DIST, current item's dist */
struct range range; /* NS_AREA/NS_DIST: range selector */
int dist; /* NS_DIST: distance selector */
coord cx, cy; /* NS_DIST: center x-y, NS_XY: xy */
- int group; /* NS_GROUP: fleet/wing match */
+ char group; /* NS_GROUP: fleet/wing match */
+ int next; /* NS_CARGO: next item */
int size; /* NS_LIST: size of list */
int index; /* NS_LIST: index */
int list[NS_LSIZE]; /* NS_LIST: item list */
- int (*read)(int type, int id, caddr_t ptr); /* read function */
- int flags; /* EFF_ flags */
int ncond; /* # of selection conditions */
- struct nscstr cond[NS_NCOND]; /* selection conditions */
+ struct nscstr cond[NS_NCOND]; /* selection conditions */
};
-#define NS_UNDEF 0
-#define NS_LIST 1
-#define NS_DIST 2
-#define NS_AREA 3
-#define NS_ALL 4
-#define NS_XY 5
-#define NS_GROUP 6
-
/*
- * codes looks like this:
- * D: is access restricted to deity?
- * T: type of member addressed by offset, if category NSC_OFF
- * C: category of value
- * V: value
- *
- * 2 2 1 1 1
- * 2 0 8 6 2 8 4 0
- * xxx xxxx xxxx xxxx xxxx xxxx
- * DTT TTCC VVVV VVVV VVVV VVVV
+ * Symbol binding: associate NAME with VALUE.
*/
+struct symbol {
+ int value;
+ char *name;
+};
-/*
- * categories
- */
-#define NSC_VAL (0) /* value is plain number */
-#define NSC_VAR (1<<16) /* value is a vtype */
-#define NSC_OFF (2<<16) /* value is an offset */
-#define NSC_CMASK (3<<16)
+/* Selector flags */
+enum {
+ NSC_DEITY = bit(0), /* access restricted to deity */
+ NSC_EXTRA = bit(1), /* computable from other selectors */
+ NSC_CONST = bit(2), /* field cannot be changed */
+ NSC_BITS = bit(3) /* value consists of flag bits */
+};
/*
- * how to interpret "offset" fields
+ * Selector descriptor
+ *
+ * A selector describes an attribute of some context object.
+ * A selector with ca_type NSC_NOTYPE is invalid.
+ * If ca_get is null, the selector describes a datum of type ca_type
+ * at offset ca_offs in the context object.
+ * A datum of type NSC_STRINGY is an array of ca_len characters.
+ * A datum of any other type is either a scalar of that type (if
+ * ca_len is zero), or an array of ca_len elements of that type.
+ * If ca_get is not null, the selector is virtual. Values can be
+ * obtained by calling ca_get(VAL, NP, CTXO), where VAL has been
+ * initialized from the selector and an index by nstr_mksymval(),
+ * NP points to the country to use for coordinate translation and
+ * access control (null for none), and CTXO is the context object.
+ * See struct valstr for details.
+ * Because virtual selectors don't have a setter method, xundump must
+ * be made to ignore them, by setting NSC_EXTRA.
+ * If flag NSC_DEITY is set, only to deities can use this selector.
+ * If flag NSC_EXTRA is set, xdump and xundump ignore this selector.
+ * If flag NSC_CONST is set, the datum can't be changed from its
+ * initial value (xundump obeys that).
+ * If ca_table is not EF_BAD, the datum refers to that Empire table;
+ * ca_type must be an integer type. If flag NSC_BITS is set, the
+ * datum consists of flag bits, and the referred table must be a
+ * symbol table defining those bits.
*/
-#define NSC_CHAR (1<<18) /* offset of s_char member */
-#define NSC_UCHAR (2<<18) /* offset of unsigned char member */
-#define NSC_SHORT (3<<18) /* offset of short */
-#define NSC_USHORT (4<<18) /* offset of unsigned short member */
-#define NSC_INT (5<<18) /* offset of int member */
-#define NSC_LONG (6<<18) /* offset of long member */
-#define NSC_XCOORD (7<<18) /* offset of coord that needs x conversion */
-#define NSC_YCOORD (8<<18) /* offset of coord that needs y conversion */
-#define NSC_FLOAT (9<<18) /* offset of float member */
-#define NSC_CHARP (10<<18) /* offset of char *member */
-#define NSC_TIME (11<<18) /* offset of time_t member */
-#define NSC_TMASK (15<<18)
-
-#define NSC_NATID NSC_UCHAR /* must match natid */
-
-#define NSC_MASK (0xffff0000)
-
-#define NSC_DEITY (1<<22)
-
struct castr {
- long ca_code; /* encoded form */
- s_char *ca_name; /* name used for matches */
- u_short ca_len; /* Used for arrays */
+ char *ca_name;
+ ptrdiff_t ca_off;
+ enum nsc_type ca_type;
+ unsigned short ca_len;
+ void *(*ca_get)(struct valstr *, struct natstr *, void *);
+ int ca_table;
+ int ca_flags;
};
/* variables using the above */
-extern struct castr var_ca[];
+extern struct castr ichr_ca[];
+extern struct castr pchr_ca[];
extern struct castr sect_ca[];
+extern struct castr dchr_ca[];
extern struct castr ship_ca[];
+extern struct castr mchr_ca[];
extern struct castr plane_ca[];
+extern struct castr plchr_ca[];
extern struct castr land_ca[];
+extern struct castr lchr_ca[];
extern struct castr nuke_ca[];
-extern struct castr news_ca[];
-extern struct castr nat_ca[];
+extern struct castr nchr_ca[];
extern struct castr treaty_ca[];
-extern struct castr trade_ca[];
extern struct castr loan_ca[];
-extern struct castr genitem_ca[];
-extern struct castr map_ca[];
+extern struct castr news_ca[];
extern struct castr lost_ca[];
extern struct castr commodity_ca[];
+extern struct castr trade_ca[];
+extern struct castr nat_ca[];
+extern struct castr cou_ca[];
+extern struct castr realm_ca[];
+extern struct castr game_ca[];
+extern struct castr intrchr_ca[];
+extern struct castr rpt_ca[];
+extern struct castr update_ca[];
+extern struct castr empfile_ca[];
+extern struct castr symbol_ca[];
+extern struct symbol ship_chr_flags[];
+extern struct symbol plane_chr_flags[];
+extern struct symbol land_chr_flags[];
+extern struct symbol nuke_chr_flags[];
+extern struct symbol treaty_flags[];
+extern struct castr mdchr_ca[];
+extern struct symbol meta_type[];
+extern struct symbol meta_flags[];
+extern struct symbol missions[];
+extern struct symbol plane_flags[];
+extern struct symbol retreat_flags[];
+extern struct symbol nation_status[];
+extern struct symbol nation_flags[];
+extern struct symbol nation_rejects[];
+extern struct symbol nation_relations[];
+extern struct symbol level[];
+extern struct symbol agreement_statuses[];
+extern struct symbol plague_stages[];
+extern struct symbol packing[];
+extern struct symbol resources[];
+extern struct symbol sector_navigation[];
+/* src/lib/common/nstreval.c */
+extern struct valstr *nstr_mksymval(struct valstr *, struct castr *, int);
+extern struct valstr *nstr_exec_val(struct valstr *, natid, void *,
+ enum nsc_type);
+extern int nstr_promote(int);
+extern char *symbol_by_value(int, struct symbol *);
+/* src/lib/global/nsc.c */
+extern void nsc_init(void);
+/* src/lib/subs/nxtitem.c */
+extern int nxtitem(struct nstr_item *, void *);
+/* src/lib/subs/nxtsct.c */
+extern int nxtsct(struct nstr_sect *, struct sctstr *);
+/* src/lib/subs/snxtitem.c */
+extern int snxtitem(struct nstr_item *, int, char *, char *);
+extern void snxtitem_area(struct nstr_item *, int, struct range *);
+extern void snxtitem_dist(struct nstr_item *, int, int, int, int);
+extern void snxtitem_xy(struct nstr_item *, int, coord, coord);
+extern void snxtitem_all(struct nstr_item *, int);
+extern void snxtitem_group(struct nstr_item *, int, char);
+extern void snxtitem_rewind(struct nstr_item *);
+extern int snxtitem_list(struct nstr_item *, int, int *, int);
+extern void snxtitem_cargo(struct nstr_item *, int, int, int);
+extern int snxtitem_use_condarg(struct nstr_item *);
+/* src/lib/subs/snxtsct.c */
+extern int snxtsct(struct nstr_sect *, char *);
+extern void snxtsct_area(struct nstr_sect *, struct range *);
+extern void snxtsct_all(struct nstr_sect *);
+extern void snxtsct_rewind(struct nstr_sect *);
+extern void snxtsct_dist(struct nstr_sect *, coord, coord, int);
+extern int snxtsct_use_condarg(struct nstr_sect *);
/* src/lib/subs/nstr.c */
-extern s_char *nstr_comp(struct nscstr *, int *, int, s_char *);
-extern int encode(register s_char *, long *, int);
-
-
-extern s_char *decodep(long, void *);
-extern int decode(natid, long, void *, int);
-extern int nstr_exec(struct nscstr *, register int, void *, int);
+extern int nstr_comp(struct nscstr *np, int len, int type, char *str);
+extern char *nstr_comp_val(char *, struct valstr *, int);
+extern int nstr_exec(struct nscstr *, int, void *);
+/* src/lib/update/nxtitemp.c */
+extern void *nxtitemp(struct nstr_item *);
-#endif /* _NSC_H_ */
+#endif