]> git.pond.sub.org Git - empserver/blob - src/lib/global/nsc.c
Virtual selectors
[empserver] / src / lib / global / nsc.c
1 /*
2  *  Empire - A multi-player, client/server Internet based war game.
3  *  Copyright (C) 1986-2008, Dave Pare, Jeff Bailey, Thomas Ruschak,
4  *                           Ken Stevens, Steve McClure
5  *
6  *  This program is free software; you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation; either version 2 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  *  ---
21  *
22  *  See files README, COPYING and CREDITS in the root of the source
23  *  tree for related information and legal notices.  It is expected
24  *  that future projects/authors will amend these files as needed.
25  *
26  *  ---
27  *
28  *  nsc.c: Empire selection global structures
29  * 
30  *  Known contributors to this file:
31  *     Markus Armbruster, 2004-2007
32  */
33
34 /*
35  * Convention: uid selector comes first.  Table TYPE has an uid
36  * selector if empfile[TYPE].cadef[0].ca_table == TYPE.
37  */
38
39 #include <config.h>
40
41 #include <stdlib.h>
42 #include "empobj.h"
43 #include "file.h"
44 #include "optlist.h"
45 #include "nsc.h"
46 #include "product.h"
47
48 static void *nsc_ver(struct valstr *, natid, void *);
49
50 /* Ugly hack to improve legibility by avoid long lines */
51 #define fldoff(fld) offsetof(CURSTR, fld)
52 #define empobjoff(fld) offsetof(struct empobj, fld)
53
54 #define NSC_IELT(name, pfx, sfx, base, itype)   \
55     {sizeof(sfx) == 1 ? name : pfx sfx,         \
56      ((base) + (itype)*sizeof(unsigned short)), \
57      NSC_SHORT, 0, NULL, EF_BAD, 0}
58
59 #define NSC_IVEC(base, sfx)                     \
60 NSC_IELT("civil", "c", sfx, base, I_CIVIL),     \
61 NSC_IELT("milit", "m", sfx, base, I_MILIT),     \
62 NSC_IELT("shell", "s", sfx, base, I_SHELL),     \
63 NSC_IELT("gun", "g", sfx, base, I_GUN),         \
64 NSC_IELT("petrol", "p", sfx, base, I_PETROL),   \
65 NSC_IELT("iron", "i", sfx, base, I_IRON),       \
66 NSC_IELT("dust", "d", sfx, base, I_DUST),       \
67 NSC_IELT("bar", "b", sfx, base, I_BAR),         \
68 NSC_IELT("food", "f", sfx, base, I_FOOD),       \
69 NSC_IELT("oil", "o", sfx, base, I_OIL),         \
70 NSC_IELT("lcm", "l", sfx, base, I_LCM),         \
71 NSC_IELT("hcm", "h", sfx, base, I_HCM),         \
72 NSC_IELT("uw", "u", sfx, base, I_UW),           \
73 NSC_IELT("rad", "r", sfx, base, I_RAD)
74
75 struct castr ichr_ca[] = {
76 #define CURSTR struct ichrstr
77     {"uid", fldoff(i_uid), NSC_SITYPE(i_type), 0, NULL, EF_ITEM, 0},
78     {"name", fldoff(i_name), NSC_STRING, 0, NULL, EF_BAD, 0},
79     {"mnem", fldoff(i_mnem), NSC_STRINGY, 1, NULL, EF_BAD, NSC_CONST},
80     {"value", fldoff(i_value), NSC_INT, 0, NULL, EF_BAD, 0},
81     {"sell", fldoff(i_sell), NSC_INT, 0, NULL, EF_BAD, 0},
82     {"lbs", fldoff(i_lbs), NSC_INT, 0, NULL, EF_BAD, 0},
83     {"pkg", fldoff(i_pkg), NSC_INT, NUMPKG, NULL, EF_BAD, 0},
84     {"melt_denom", fldoff(i_melt_denom), NSC_INT, 0, NULL, EF_BAD, 0},
85     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
86 #undef CURSTR
87 };
88
89 struct castr pchr_ca[] = {
90 #define CURSTR struct pchrstr
91     {"uid", fldoff(p_uid), NSC_INT, 0, NULL, EF_PRODUCT, 0},
92     {"name", fldoff(p_name), NSC_STRING, 0, NULL, EF_BAD, 0},
93     {"sname", fldoff(p_sname), NSC_STRING, 0, NULL, EF_BAD, NSC_CONST},
94     {"ctype", fldoff(p_ctype), NSC_SITYPE(i_type), MAXPRCON, NULL,
95      EF_ITEM, 0},
96     {"camt", fldoff(p_camt), NSC_USHORT, MAXPRCON, NULL, EF_BAD, 0},
97     {"type", fldoff(p_type), NSC_SITYPE(i_type), 0, NULL, EF_ITEM, 0},
98     {"level", fldoff(p_level), NSC_INT, 0, NULL, EF_LEVEL, 0},
99     {"cost", fldoff(p_cost), NSC_INT, 0, NULL, EF_BAD, 0},
100     {"nrndx", fldoff(p_nrndx), NSC_INT, 0, NULL, EF_RESOURCES, 0},
101     {"nrdep", fldoff(p_nrdep), NSC_INT, 0, NULL, EF_BAD, 0},
102     {"nlndx", fldoff(p_nlndx), NSC_INT, 0, NULL, EF_LEVEL, 0},
103     {"nlmin", fldoff(p_nlmin), NSC_INT, 0, NULL, EF_BAD, 0},
104     {"nllag", fldoff(p_nllag), NSC_INT, 0, NULL, EF_BAD, 0},
105     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
106 #undef CURSTR
107 };
108
109 struct castr sect_ca[] = {
110 #define CURSTR struct sctstr
111     /* uid needs to be NSC_DEITY because it discloses true origin */
112     {"uid", fldoff(sct_uid), NSC_SHORT, 0, NULL,
113      EF_SECTOR, NSC_DEITY | NSC_EXTRA},
114     {"owner", fldoff(sct_own), NSC_NATID, 0, NULL, EF_NATION, 0},
115     {"xloc", fldoff(sct_x), NSC_XCOORD, 0, NULL, EF_BAD, NSC_CONST},
116     {"yloc", fldoff(sct_y), NSC_YCOORD, 0, NULL, EF_BAD, NSC_CONST},
117     {"des", fldoff(sct_type), NSC_CHAR, 0, NULL, EF_SECTOR_CHR, 0},
118     {"effic", fldoff(sct_effic), NSC_UCHAR, 0, NULL, EF_BAD, 0},
119     {"mobil", fldoff(sct_mobil), NSC_SHORT, 0, NULL, EF_BAD, 0},
120     {"loyal", fldoff(sct_loyal), NSC_UCHAR, 0, NULL, EF_BAD, NSC_DEITY},
121     {"terr", fldoff(sct_terr), NSC_UCHAR, 0, NULL, EF_BAD, 0},
122     {"terr0", fldoff(sct_terr), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
123     {"terr1", fldoff(sct_terr1), NSC_UCHAR, 0, NULL, EF_BAD, 0},
124     {"terr2", fldoff(sct_terr2), NSC_UCHAR, 0, NULL, EF_BAD, 0},
125     {"terr3", fldoff(sct_terr3), NSC_UCHAR, 0, NULL, EF_BAD, 0},
126     {"dterr", fldoff(sct_dterr), NSC_UCHAR, 0, NULL, EF_BAD, NSC_DEITY},
127     {"xdist", fldoff(sct_dist_x), NSC_XCOORD, 0, NULL, EF_BAD, 0},
128     {"ydist", fldoff(sct_dist_y), NSC_YCOORD, 0, NULL, EF_BAD, 0},
129     {"avail", fldoff(sct_avail), NSC_SHORT, 0, NULL, EF_BAD, 0},
130     {"elev", fldoff(sct_elev), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
131     {"work", fldoff(sct_work), NSC_UCHAR, 0, NULL, EF_BAD, 0},
132     {"coastal", fldoff(sct_coastal), NSC_UCHAR, 0, NULL, EF_BAD, 0},
133     {"newdes", fldoff(sct_newtype), NSC_CHAR, 0, NULL, EF_SECTOR_CHR, 0},
134     {"min", fldoff(sct_min), NSC_UCHAR, 0, NULL, EF_BAD, 0},
135     {"gold", fldoff(sct_gmin), NSC_UCHAR, 0, NULL, EF_BAD, 0},
136     {"fert", fldoff(sct_fertil), NSC_UCHAR, 0, NULL, EF_BAD, 0},
137     {"ocontent", fldoff(sct_oil), NSC_UCHAR, 0, NULL, EF_BAD, 0},
138     {"uran", fldoff(sct_uran), NSC_UCHAR, 0, NULL, EF_BAD, 0},
139     {"oldown", fldoff(sct_oldown), NSC_NATID, 0, NULL, EF_NATION, 0},
140     {"off", fldoff(sct_off), NSC_UCHAR, 0, NULL, EF_BAD, 0},
141     NSC_IVEC(fldoff(sct_item), ""),
142     NSC_IVEC(fldoff(sct_dist), "_dist"),
143     NSC_IVEC(fldoff(sct_del), "_del"),
144     /* should let old owner access mines, but can't express that: */
145     {"mines", fldoff(sct_mines), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
146     {"pstage", fldoff(sct_pstage), NSC_SHORT, 0, NULL,
147      EF_PLAGUE_STAGES, NSC_DEITY},
148     {"ptime", fldoff(sct_ptime), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
149     {"che", fldoff(sct_che), NSC_UCHAR, 0, NULL, EF_BAD, NSC_DEITY},
150     {"che_target", fldoff(sct_che_target), NSC_NATID, 0, NULL,
151      EF_NATION, NSC_DEITY},
152     {"fallout", fldoff(sct_fallout), NSC_USHORT, 0, NULL, EF_BAD, 0},
153     {"access", fldoff(sct_access), NSC_SHORT, 0, NULL, EF_BAD, 0},
154     {"road", fldoff(sct_road), NSC_UCHAR, 0, NULL, EF_BAD, 0},
155     {"rail", fldoff(sct_rail), NSC_UCHAR, 0, NULL, EF_BAD, 0},
156     {"dfense", fldoff(sct_defense), NSC_UCHAR, 0, NULL, EF_BAD, 0},
157     {"timestamp", fldoff(sct_timestamp), NSC_TIME, 0, NULL,
158      EF_BAD, NSC_EXTRA},
159     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
160 #undef CURSTR
161 };
162
163 struct castr dchr_ca[] = {
164 #define CURSTR struct dchrstr
165     {"uid", fldoff(d_uid), NSC_UCHAR, 0, NULL, EF_SECTOR_CHR, 0},
166     {"name", fldoff(d_name), NSC_STRING, 0, NULL, EF_BAD, 0},
167     {"mnem", fldoff(d_mnem), NSC_STRINGY, 1, NULL, EF_BAD, NSC_CONST},
168     {"prd", fldoff(d_prd), NSC_INT, 0, NULL, EF_PRODUCT, 0},
169     {"peffic", fldoff(d_peffic), NSC_INT, 0, NULL, EF_BAD, 0},
170     {"mob0", fldoff(d_mob0), NSC_FLOAT, 0, NULL, EF_BAD, 0},
171     {"mob1", fldoff(d_mob1), NSC_FLOAT, 0, NULL, EF_BAD, 0},
172     {"nav", fldoff(d_nav), NSC_SITYPE(d_navigation), 0, NULL,
173      EF_SECTOR_NAVIGATION, 0},
174     {"pkg", fldoff(d_pkg), NSC_SITYPE(i_packing), 0, NULL, EF_PACKING, 0},
175     {"ostr", fldoff(d_ostr), NSC_FLOAT, 0, NULL, EF_BAD, 0},
176     {"dstr", fldoff(d_dstr), NSC_FLOAT, 0, NULL, EF_BAD, 0},
177     {"value", fldoff(d_value), NSC_INT, 0, NULL, EF_BAD, 0},
178     {"cost", fldoff(d_cost), NSC_INT, 0, NULL, EF_BAD, 0},
179     {"build", fldoff(d_build), NSC_INT, 0, NULL, EF_BAD, 0},
180     {"lcms", fldoff(d_lcms), NSC_INT, 0, NULL, EF_BAD, 0},
181     {"hcms", fldoff(d_hcms), NSC_INT, 0, NULL, EF_BAD, 0},
182     {"maxpop", fldoff(d_maxpop), NSC_INT, 0, NULL, EF_BAD, 0},
183     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
184 #undef CURSTR
185 };
186
187 #define NSC_GENITEM(ef_type, ef_chr)                                    \
188     {"uid", empobjoff(uid),  NSC_SHORT, 0, NULL, ef_type, 0},           \
189     {"owner", empobjoff(own),  NSC_NATID, 0, NULL, EF_NATION, 0},       \
190     {"xloc", empobjoff(x),  NSC_XCOORD, 0, NULL, EF_BAD, 0},            \
191     {"yloc", empobjoff(y),  NSC_YCOORD, 0, NULL, EF_BAD, 0},            \
192     {"type", empobjoff(type),  NSC_CHAR, 0, NULL, ef_chr, 0},           \
193     {"effic", empobjoff(effic),  NSC_CHAR, 0, NULL, EF_BAD, 0},         \
194     {"mobil", empobjoff(mobil),  NSC_CHAR , 0, NULL, EF_BAD, 0},        \
195     {"off", empobjoff(off),  NSC_UCHAR , 0, NULL, EF_BAD, 0},           \
196     {"tech", empobjoff(tech),  NSC_SHORT, 0, NULL, EF_BAD, 0},          \
197     {"group", empobjoff(group),  NSC_STRINGY, 1, NULL, EF_BAD, NSC_EXTRA}, \
198     {"opx", empobjoff(opx),  NSC_XCOORD, 0, NULL, EF_BAD, 0},           \
199     {"opy", empobjoff(opy),  NSC_YCOORD, 0, NULL, EF_BAD, 0},           \
200     {"mission", empobjoff(mission),  NSC_SHORT, 0, NULL, EF_MISSIONS, 0}, \
201     {"radius", empobjoff(radius),  NSC_SHORT, 0, NULL, EF_BAD, 0}
202
203 struct castr ship_ca[] = {
204 #define CURSTR struct shpstr
205     NSC_GENITEM(EF_SHIP, EF_SHIP_CHR),
206     {"fleet", fldoff(shp_fleet), NSC_STRINGY, 1, NULL, EF_BAD, 0},
207     {"nplane", fldoff(shp_nplane), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
208     {"nland", fldoff(shp_nland), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
209     {"xstart", fldoff(shp_destx[0]), NSC_XCOORD, 0, NULL, EF_BAD, 0},
210     {"xend", fldoff(shp_destx[1]), NSC_XCOORD, 0, NULL, EF_BAD, 0},
211     {"ystart", fldoff(shp_desty[0]), NSC_YCOORD, 0, NULL, EF_BAD, 0},
212     {"yend", fldoff(shp_desty[1]), NSC_YCOORD, 0, NULL, EF_BAD, 0},
213     {"cargostart", fldoff(shp_tstart), NSC_SITYPE(i_type), TMAX, NULL,
214      EF_ITEM, 0},
215     {"cargoend", fldoff(shp_tend), NSC_SITYPE(i_type), TMAX, NULL,
216      EF_ITEM, 0},
217     {"amtstart", fldoff(shp_lstart), NSC_SHORT, TMAX, NULL, EF_BAD, 0},
218     {"amtend", fldoff(shp_lend), NSC_SHORT, TMAX, NULL, EF_BAD, 0},
219     {"autonav", fldoff(shp_autonav), NSC_UCHAR, 0, NULL, EF_BAD, 0},
220     NSC_IVEC(fldoff(shp_item), ""),
221     {"pstage", fldoff(shp_pstage), NSC_SHORT, 0, NULL,
222      EF_PLAGUE_STAGES, NSC_DEITY},
223     {"ptime", fldoff(shp_ptime), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
224     {"access", fldoff(shp_access), NSC_SHORT, 0, NULL, EF_BAD, 0},
225     {"timestamp", fldoff(shp_timestamp), NSC_TIME, 0, NULL,
226      EF_BAD, NSC_EXTRA},
227     {"mquota", fldoff(shp_mobquota), NSC_UCHAR, 0, NULL, EF_BAD, 0},
228     {"path", fldoff(shp_path), NSC_STRINGY, MAXSHPPATH, NULL, EF_BAD, 0},
229     {"follow", fldoff(shp_follow), NSC_SHORT, 0, NULL, EF_BAD, 0},
230     {"name", fldoff(shp_name), NSC_STRINGY, MAXSHPNAMLEN, NULL,
231      EF_BAD, 0},
232     {"fuel", fldoff(shp_fuel), NSC_UCHAR, 0, NULL, EF_BAD, 0},
233     {"nchoppers", fldoff(shp_nchoppers), NSC_UCHAR, 0, NULL,
234      EF_BAD, NSC_EXTRA},
235     {"nxlight", fldoff(shp_nxlight), NSC_UCHAR, 0, NULL,
236      EF_BAD, NSC_EXTRA},
237     /* should let builder access xbuilt, ybuilt, but can't express that: */
238     {"xbuilt", fldoff(shp_orig_x), NSC_XCOORD, 0, NULL,
239      EF_BAD, NSC_DEITY},
240     {"ybuilt", fldoff(shp_orig_y), NSC_YCOORD, 0, NULL,
241      EF_BAD, NSC_DEITY},
242     {"builder", fldoff(shp_orig_own), NSC_NATID, 0, NULL,
243      EF_NATION, NSC_DEITY},
244     {"rflags", fldoff(shp_rflags), NSC_INT, 0, NULL,
245      EF_RETREAT_FLAGS, NSC_BITS},
246     {"rpath", fldoff(shp_rpath), NSC_STRINGY, RET_LEN, NULL, EF_BAD, 0},
247     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
248 #undef CURSTR
249 };
250
251 struct castr mchr_ca[] = {
252 #define CURSTR struct mchrstr
253     {"type", fldoff(m_type), NSC_CHAR, 0, NULL, EF_SHIP_CHR, 0},
254     {"name", fldoff(m_name), NSC_STRING, 0, NULL, EF_BAD, 0},
255     NSC_IVEC(fldoff(m_item), ""),
256     {"l_build", fldoff(m_lcm), NSC_INT, 0, NULL, EF_BAD, 0},
257     {"h_build", fldoff(m_hcm), NSC_INT, 0, NULL, EF_BAD, 0},
258     {"armor", fldoff(m_armor), NSC_INT, 0, NULL, EF_BAD, 0},
259     {"speed", fldoff(m_speed), NSC_INT, 0, NULL, EF_BAD, 0},
260     {"visib", fldoff(m_visib), NSC_INT, 0, NULL, EF_BAD, 0},
261     {"vrnge", fldoff(m_vrnge), NSC_INT, 0, NULL, EF_BAD, 0},
262     {"frnge", fldoff(m_frnge), NSC_INT, 0, NULL, EF_BAD, 0},
263     {"glim", fldoff(m_glim), NSC_INT, 0, NULL, EF_BAD, 0},
264     {"nxlight", fldoff(m_nxlight), NSC_UCHAR, 0, NULL, EF_BAD, 0},
265     {"nchoppers", fldoff(m_nchoppers), NSC_UCHAR, 0, NULL, EF_BAD, 0},
266     {"fuelc", fldoff(m_fuelc), NSC_UCHAR, 0, NULL, EF_BAD, 0},
267     {"fuelu", fldoff(m_fuelu), NSC_UCHAR, 0, NULL, EF_BAD, 0},
268     {"tech", fldoff(m_tech), NSC_INT, 0, NULL, EF_BAD, 0},
269     {"cost", fldoff(m_cost), NSC_INT, 0, NULL, EF_BAD, 0},
270     {"flags", fldoff(m_flags), NSC_LONG, 0, NULL,
271      EF_SHIP_CHR_FLAGS, NSC_BITS},
272     {"nplanes", fldoff(m_nplanes), NSC_UCHAR, 0, NULL, EF_BAD, 0},
273     {"nland", fldoff(m_nland), NSC_UCHAR, 0, NULL, EF_BAD, 0},
274     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
275 #undef CURSTR
276 };
277
278 struct castr plane_ca[] = {
279 #define CURSTR struct plnstr
280     NSC_GENITEM(EF_PLANE, EF_PLANE_CHR),
281     {"wing", fldoff(pln_wing), NSC_STRINGY, 1, NULL, EF_BAD, 0},
282     {"range", fldoff(pln_range), NSC_UCHAR, 0, NULL, EF_BAD, 0},
283     {"ship", fldoff(pln_ship), NSC_SHORT, 0, NULL, EF_BAD, 0},
284     {"land", fldoff(pln_land), NSC_SHORT, 0, NULL, EF_BAD, 0},
285     {"att", fldoff(pln_att), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
286     {"def", fldoff(pln_def), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
287     {"harden", fldoff(pln_harden), NSC_CHAR, 0, NULL, EF_BAD, 0},
288     {"nuketype", fldoff(pln_nuketype), NSC_CHAR, 0, NULL, EF_BAD, 0},
289     {"flags", fldoff(pln_flags), NSC_CHAR, 0, NULL,
290      EF_PLANE_FLAGS, NSC_BITS},
291     {"access", fldoff(pln_access), NSC_SHORT, 0, NULL, EF_BAD, 0},
292     {"timestamp", fldoff(pln_timestamp), NSC_TIME, 0, NULL,
293      EF_BAD, NSC_EXTRA},
294     {"theta", fldoff(pln_theta), NSC_FLOAT, 0, NULL, EF_BAD, 0},
295     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
296 #undef CURSTR
297 };
298
299 struct castr plchr_ca[] = {
300 #define CURSTR struct plchrstr
301     {"type", fldoff(pl_type), NSC_CHAR, 0, NULL, EF_PLANE_CHR, 0},
302     {"name", fldoff(pl_name), NSC_STRING, 0, NULL, EF_BAD, 0},
303     {"l_build", fldoff(pl_lcm), NSC_INT, 0, NULL, EF_BAD, 0},
304     {"h_build", fldoff(pl_hcm), NSC_INT, 0, NULL, EF_BAD, 0},
305     {"cost", fldoff(pl_cost), NSC_INT, 0, NULL, EF_BAD, 0},
306     {"tech", fldoff(pl_tech), NSC_INT, 0, NULL, EF_BAD, 0},
307     {"acc", fldoff(pl_acc), NSC_INT, 0, NULL, EF_BAD, 0},
308     {"load", fldoff(pl_load), NSC_INT, 0, NULL, EF_BAD, 0},
309     {"att", fldoff(pl_att), NSC_INT, 0, NULL, EF_BAD, 0},
310     {"def", fldoff(pl_def), NSC_INT, 0, NULL, EF_BAD, 0},
311     {"range", fldoff(pl_range), NSC_INT, 0, NULL, EF_BAD, 0},
312     {"crew", fldoff(pl_crew), NSC_INT, 0, NULL, EF_BAD, 0},
313     {"fuel", fldoff(pl_fuel), NSC_INT, 0, NULL, EF_BAD, 0},
314     {"stealth", fldoff(pl_stealth), NSC_INT, 0, NULL, EF_BAD, 0},
315     {"flags", fldoff(pl_flags), NSC_INT, 0, NULL,
316      EF_PLANE_CHR_FLAGS, NSC_BITS},
317     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
318 #undef CURSTR
319 };
320
321 struct castr land_ca[] = {
322 #define CURSTR struct lndstr
323     NSC_GENITEM(EF_LAND, EF_LAND_CHR),
324     {"army", fldoff(lnd_army), NSC_STRINGY, 1, NULL, EF_BAD, 0},
325     {"ship", fldoff(lnd_ship), NSC_SHORT, 0, NULL, EF_BAD, 0},
326     {"harden", fldoff(lnd_harden), NSC_CHAR, 0, NULL, EF_BAD, 0},
327     {"retreat", fldoff(lnd_retreat), NSC_SHORT, 0, NULL, EF_BAD, 0},
328     {"fuel", fldoff(lnd_fuel), NSC_UCHAR, 0, NULL, EF_BAD, 0},
329     {"nxlight", fldoff(lnd_nxlight), NSC_UCHAR, 0, NULL,
330      EF_BAD, NSC_EXTRA},
331     {"rflags", fldoff(lnd_rflags), NSC_INT, 0, NULL,
332      EF_RETREAT_FLAGS, NSC_BITS},
333     {"rpath", fldoff(lnd_rpath), NSC_STRINGY, RET_LEN, NULL, EF_BAD, 0},
334     {"react", fldoff(lnd_rad_max), NSC_UCHAR, 0, NULL, EF_BAD, 0},
335     NSC_IVEC(fldoff(lnd_item), ""),
336     {"pstage", fldoff(lnd_pstage), NSC_SHORT, 0, NULL,
337      EF_PLAGUE_STAGES, NSC_DEITY},
338     {"ptime", fldoff(lnd_ptime), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
339     {"land", fldoff(lnd_land), NSC_SHORT, 0, NULL, EF_BAD, 0},
340     {"nland", fldoff(lnd_nland), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
341     {"access", fldoff(lnd_access), NSC_SHORT, 0, NULL, EF_BAD, 0},
342     {"att", fldoff(lnd_att), NSC_FLOAT, 0, NULL, EF_BAD, NSC_EXTRA},
343     {"def", fldoff(lnd_def), NSC_FLOAT, 0, NULL, EF_BAD, NSC_EXTRA},
344     {"vul", fldoff(lnd_vul), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
345     {"spd", fldoff(lnd_spd), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
346     {"vis", fldoff(lnd_vis), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
347     {"spy", fldoff(lnd_spy), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
348     {"rmax", fldoff(lnd_rad), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
349     {"frg", fldoff(lnd_frg), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
350     {"acc", fldoff(lnd_acc), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
351     {"dam", fldoff(lnd_dam), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
352     {"ammo", fldoff(lnd_ammo), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
353     {"aaf", fldoff(lnd_aaf), NSC_INT, 0, NULL, EF_BAD, NSC_EXTRA},
354     {"fuelc", fldoff(lnd_fuelc), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
355     {"fuelu", fldoff(lnd_fuelu), NSC_UCHAR, 0, NULL, EF_BAD, NSC_EXTRA},
356     {"maxlight", fldoff(lnd_maxlight), NSC_UCHAR, 0, NULL,
357      EF_BAD, NSC_EXTRA},
358     {"timestamp", fldoff(lnd_timestamp), NSC_TIME, 0, NULL,
359      EF_BAD, NSC_EXTRA},
360     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
361 #undef CURSTR
362 };
363
364 struct castr lchr_ca[] = {
365 #define CURSTR struct lchrstr
366     {"type", fldoff(l_type), NSC_CHAR, 0, NULL, EF_LAND_CHR, 0},
367     {"name", fldoff(l_name), NSC_STRING, 0, NULL, EF_BAD, 0},
368     NSC_IVEC(fldoff(l_item), ""),
369     {"l_build", fldoff(l_lcm), NSC_INT, 0, NULL, EF_BAD, 0},
370     {"h_build", fldoff(l_hcm), NSC_INT, 0, NULL, EF_BAD, 0},
371     {"tech", fldoff(l_tech), NSC_INT, 0, NULL, EF_BAD, 0},
372     {"cost", fldoff(l_cost), NSC_INT, 0, NULL, EF_BAD, 0},
373     {"att", fldoff(l_att), NSC_FLOAT, 0, NULL, EF_BAD, 0},
374     {"def", fldoff(l_def), NSC_FLOAT, 0, NULL, EF_BAD, 0},
375     {"vul", fldoff(l_vul), NSC_INT, 0, NULL, EF_BAD, 0},
376     {"spd", fldoff(l_spd), NSC_INT, 0, NULL, EF_BAD, 0},
377     {"vis", fldoff(l_vis), NSC_INT, 0, NULL, EF_BAD, 0},
378     {"spy", fldoff(l_spy), NSC_INT, 0, NULL, EF_BAD, 0},
379     {"rmax", fldoff(l_rad), NSC_INT, 0, NULL, EF_BAD, 0},
380     {"frg", fldoff(l_frg), NSC_INT, 0, NULL, EF_BAD, 0},
381     {"acc", fldoff(l_acc), NSC_INT, 0, NULL, EF_BAD, 0},
382     {"dam", fldoff(l_dam), NSC_INT, 0, NULL, EF_BAD, 0},
383     {"ammo", fldoff(l_ammo), NSC_INT, 0, NULL, EF_BAD, 0},
384     {"aaf", fldoff(l_aaf), NSC_INT, 0, NULL, EF_BAD, 0},
385     {"fuelc", fldoff(l_fuelc), NSC_UCHAR, 0, NULL, EF_BAD, 0},
386     {"fuelu", fldoff(l_fuelu), NSC_UCHAR, 0, NULL, EF_BAD, 0},
387     {"nxlight", fldoff(l_nxlight), NSC_UCHAR, 0, NULL, EF_BAD, 0},
388     {"nland", fldoff(l_nland), NSC_UCHAR, 0, NULL, EF_BAD, 0},
389     {"flags", fldoff(l_flags), NSC_LONG, 0, NULL,
390      EF_LAND_CHR_FLAGS, NSC_BITS},
391     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
392 #undef CURSTR
393 };
394
395 struct castr nuke_ca[] = {
396 #define CURSTR struct nukstr
397     NSC_GENITEM(EF_NUKE, EF_NUKE_CHR),
398     {"plane", fldoff(nuk_plane), NSC_SHORT, 0, NULL, EF_BAD, 0},
399     {"timestamp", fldoff(nuk_timestamp), NSC_TIME, 0, NULL,
400      EF_BAD, NSC_EXTRA},
401     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
402 #undef CURSTR
403 };
404
405 struct castr nchr_ca[] = {
406 #define CURSTR struct nchrstr
407     {"type", fldoff(n_type), NSC_CHAR, 0, NULL, EF_NUKE_CHR, 0},
408     {"name", fldoff(n_name), NSC_STRING, 0, NULL, EF_BAD, 0},
409     {"l_build", fldoff(n_lcm), NSC_INT, 0, NULL, EF_BAD, 0},
410     {"h_build", fldoff(n_hcm), NSC_INT, 0, NULL, EF_BAD, 0},
411     {"o_build", fldoff(n_oil), NSC_INT, 0, NULL, EF_BAD, 0},
412     {"r_build", fldoff(n_rad), NSC_INT, 0, NULL, EF_BAD, 0},
413     {"blast", fldoff(n_blast), NSC_INT, 0, NULL, EF_BAD, 0},
414     {"dam", fldoff(n_dam), NSC_INT, 0, NULL, EF_BAD, 0},
415     {"cost", fldoff(n_cost), NSC_INT, 0, NULL, EF_BAD, 0},
416     {"tech", fldoff(n_tech), NSC_INT, 0, NULL, EF_BAD, 0},
417     {"weight", fldoff(n_weight), NSC_INT, 0, NULL, EF_BAD, 0},
418     {"flags", fldoff(n_flags), NSC_INT, 0, NULL,
419      EF_NUKE_CHR_FLAGS, NSC_BITS},
420     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
421 #undef CURSTR
422 };
423
424 struct castr treaty_ca[] = {
425 #define CURSTR struct trtstr
426     {"uid", fldoff(trt_uid), NSC_SHORT, 0, NULL, EF_TREATY, 0},
427     {"cna", fldoff(trt_cna), NSC_NATID, 0, NULL, EF_NATION, 0},
428     {"cnb", fldoff(trt_cnb), NSC_NATID, 0, NULL, EF_NATION, 0},
429     {"status", fldoff(trt_status), NSC_CHAR, 0, NULL,
430      EF_AGREEMENT_STATUS, 0},
431     {"acond", fldoff(trt_acond), NSC_SHORT, 0, NULL,
432      EF_TREATY_FLAGS, NSC_BITS},
433     {"bcond", fldoff(trt_bcond), NSC_SHORT, 0, NULL,
434      EF_TREATY_FLAGS, NSC_BITS},
435     {"exp", fldoff(trt_exp), NSC_TIME, 0, NULL, EF_BAD, 0},
436     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
437 #undef CURSTR
438 };
439
440 struct castr loan_ca[] = {
441 #define CURSTR struct lonstr
442     {"uid", fldoff(l_uid), NSC_SHORT, 0, NULL, EF_LOAN, 0},
443     {"loaner", fldoff(l_loner), NSC_NATID, 0, NULL, EF_NATION, 0},
444     {"loanee", fldoff(l_lonee), NSC_NATID, 0, NULL, EF_NATION, 0},
445     {"status", fldoff(l_status), NSC_CHAR, 0, NULL,
446      EF_AGREEMENT_STATUS, 0},
447     {"irate", fldoff(l_irate), NSC_INT, 0, NULL, EF_BAD, 0},
448     {"ldur", fldoff(l_ldur), NSC_INT, 0, NULL, EF_BAD, 0},
449     {"amtpaid", fldoff(l_amtpaid), NSC_LONG, 0, NULL, EF_BAD, 0},
450     {"amtdue", fldoff(l_amtdue), NSC_LONG, 0, NULL, EF_BAD, 0},
451     {"lastpay", fldoff(l_lastpay), NSC_TIME, 0, NULL, EF_BAD, 0},
452     {"duedate", fldoff(l_duedate), NSC_TIME, 0, NULL, EF_BAD, 0},
453     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
454 #undef CURSTR
455 };
456
457 struct castr news_ca[] = {
458 #define CURSTR struct nwsstr
459     /* no need for uid as long as it's not referenced from other tables */
460     {"actor", fldoff(nws_ano), NSC_NATID, 0, NULL, EF_NATION, 0},
461     {"action", fldoff(nws_vrb), NSC_UCHAR, 0, NULL, EF_NEWS_CHR, 0},
462     {"victim", fldoff(nws_vno), NSC_NATID, 0, NULL, EF_NATION, 0},
463     {"times", fldoff(nws_ntm), NSC_CHAR, 0, NULL, EF_BAD, 0},
464     {"time", fldoff(nws_when), NSC_TIME, 0, NULL, EF_BAD, 0},
465     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
466 #undef CURSTR
467 };
468
469 struct castr lost_ca[] = {
470 #define CURSTR struct loststr
471    /* no need for uid as long as it's not referenced from other tables */
472     {"owner", fldoff(lost_owner), NSC_NATID, 0, NULL, EF_NATION, 0},
473     {"type", fldoff(lost_type), NSC_CHAR, 0, NULL, EF_TABLE, 0},
474     {"id", fldoff(lost_id), NSC_SHORT, 0, NULL, EF_BAD, 0},
475     {"x", fldoff(lost_x), NSC_XCOORD, 0, NULL, EF_BAD, 0},
476     {"y", fldoff(lost_y), NSC_YCOORD, 0, NULL, EF_BAD, 0},
477     {"timestamp", fldoff(lost_timestamp), NSC_TIME, 0, NULL, EF_BAD, 0},
478     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
479 #undef CURSTR
480 };
481
482 struct castr commodity_ca[] = {
483 #define CURSTR struct comstr
484     {"uid", fldoff(com_uid), NSC_SHORT, 0, NULL, EF_COMM, 0},
485     {"owner", fldoff(com_owner), NSC_NATID, 0, NULL, EF_NATION, 0},
486     {"type", fldoff(com_type), NSC_SITYPE(i_type), 0, NULL, EF_ITEM, 0},
487     {"amount", fldoff(com_amount), NSC_INT, 0, NULL, EF_BAD, 0},
488     {"price", fldoff(com_price), NSC_FLOAT, 0, NULL, EF_BAD, 0},
489     {"maxbidder", fldoff(com_maxbidder), NSC_INT, 0, NULL, EF_NATION, 0},
490     {"markettime", fldoff(com_markettime), NSC_TIME, 0, NULL, EF_BAD, 0},
491     /* should let maxbidder access xbuy, ybuy, but can't express that: */
492     {"xbuy", fldoff(com_x), NSC_XCOORD, 0, NULL, EF_BAD, NSC_DEITY},
493     {"ybuy", fldoff(com_y), NSC_XCOORD, 0, NULL, EF_BAD, NSC_DEITY},
494     /* should let owner access xsell, ysell, but can't express that: */
495     {"xsell", fldoff(sell_x), NSC_XCOORD, 0, NULL, EF_BAD, NSC_DEITY},
496     {"ysell", fldoff(sell_y), NSC_YCOORD, 0, NULL, EF_BAD, NSC_DEITY},
497     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
498 #undef CURSTR
499 };
500
501 struct castr trade_ca[] = {
502 #define CURSTR struct trdstr
503     {"uid", fldoff(trd_uid), NSC_SHORT, 0, NULL, EF_TRADE, 0},
504     {"owner", fldoff(trd_owner), NSC_NATID, 0, NULL, EF_NATION, 0},
505     {"type", fldoff(trd_type), NSC_CHAR, 0, NULL, EF_TABLE, 0},
506     {"unitid", fldoff(trd_unitid), NSC_SHORT, 0, NULL, EF_BAD, 0},
507     {"price", fldoff(trd_price), NSC_LONG, 0, NULL, EF_BAD, 0},
508     {"maxbidder", fldoff(trd_maxbidder), NSC_INT, 0, NULL, EF_NATION, 0},
509     {"markettime", fldoff(trd_markettime), NSC_TIME, 0, NULL, EF_BAD, 0},
510     /* should let maxbidder access xloc, yloc, but can't express that: */
511     {"xloc", fldoff(trd_x), NSC_XCOORD, 0, NULL, EF_BAD, NSC_DEITY},
512     {"yloc", fldoff(trd_y), NSC_YCOORD, 0, NULL, EF_BAD, NSC_DEITY},
513     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
514 #undef CURSTR
515 };
516
517 struct castr cou_ca[] = {
518 #define CURSTR struct natstr
519     /*
520      * This is the owner's view, i.e. it applies only to the own
521      * nation.  The public view nat_ca[], which applies to all
522      * nations, has the same selectors with different flags: NSC_DEITY
523      * is set except for cnum (which must come first) and all
524      * NSC_EXTRA selectors, NSC_EXTRA is cleared.
525      * nat_ca[] should also make tech, research, education and
526      * happiness available, but we can't express the obfuscation
527      * necessary for foreign levels.
528      */
529     {"cnum", fldoff(nat_cnum), NSC_NATID, 0, NULL, EF_NATION, 0},
530     {"stat", fldoff(nat_stat), NSC_SITYPE(nat_status), 0, NULL,
531      EF_NATION_STATUS, NSC_EXTRA},
532     {"cname", fldoff(nat_cnam), NSC_STRINGY, 20, NULL, EF_BAD, NSC_EXTRA},
533     {"passwd", fldoff(nat_pnam), NSC_STRINGY, 20, NULL,
534      EF_BAD, NSC_DEITY | NSC_EXTRA},
535     {"ip", fldoff(nat_hostaddr), NSC_STRINGY, 32, NULL, EF_BAD, 0},
536     {"hostname", fldoff(nat_hostname), NSC_STRINGY, 512, NULL, EF_BAD, 0},
537     {"userid", fldoff(nat_userid), NSC_STRINGY, 32, NULL, EF_BAD, 0},
538     {"xcap", fldoff(nat_xcap), NSC_XCOORD, 0, NULL, EF_BAD, 0},
539     {"ycap", fldoff(nat_ycap), NSC_YCOORD, 0, NULL, EF_BAD, 0},
540     {"xorg", fldoff(nat_xorg), NSC_XCOORD, 0, NULL,
541      EF_BAD, NSC_DEITY | NSC_EXTRA},
542     {"yorg", fldoff(nat_yorg), NSC_YCOORD, 0, NULL,
543      EF_BAD, NSC_DEITY | NSC_EXTRA},
544     {"dayno", fldoff(nat_dayno), NSC_CHAR, 0, NULL, EF_BAD, 0},
545     {"update", fldoff(nat_update), NSC_CHAR, 0, NULL, EF_BAD, 0},
546     {"tgms", fldoff(nat_tgms), NSC_USHORT, 0, NULL, EF_BAD, 0},
547     {"ann", fldoff(nat_ann), NSC_USHORT, 0, NULL, EF_BAD, 0},
548     {"minused", fldoff(nat_minused), NSC_USHORT, 0, NULL, EF_BAD, 0},
549     {"btu", fldoff(nat_btu), NSC_SHORT, 0, NULL, EF_BAD, 0},
550     {"access", fldoff(nat_access), NSC_SHORT, 0, NULL, EF_BAD, 0},
551     {"milreserve", fldoff(nat_reserve), NSC_LONG, 0, NULL, EF_BAD, 0},
552     {"money", fldoff(nat_money), NSC_LONG, 0, NULL, EF_BAD, 0},
553     {"login", fldoff(nat_last_login), NSC_TIME, 0, NULL, EF_BAD, 0},
554     {"logout", fldoff(nat_last_logout), NSC_TIME, 0, NULL, EF_BAD, 0},
555     {"newstim", fldoff(nat_newstim), NSC_TIME, 0, NULL, EF_BAD, 0},
556     {"annotim", fldoff(nat_annotim), NSC_TIME, 0, NULL, EF_BAD, 0},
557     {"tech", fldoff(nat_level[NAT_TLEV]), NSC_FLOAT, 0, NULL, EF_BAD, 0},
558     {"research", fldoff(nat_level[NAT_RLEV]), NSC_FLOAT, 0, NULL,
559      EF_BAD, 0},
560     {"education", fldoff(nat_level[NAT_ELEV]), NSC_FLOAT, 0, NULL,
561      EF_BAD, 0},
562     {"happiness", fldoff(nat_level[NAT_HLEV]), NSC_FLOAT, 0, NULL,
563      EF_BAD, 0},
564     {"relations", fldoff(nat_relate), NSC_HIDDEN, MAXNOC, NULL,
565      EF_NATION_RELATIONS, NSC_EXTRA},
566     /* mortals know there's contact (relations show), but not how strong */
567     {"contacts", fldoff(nat_contact), NSC_UCHAR, MAXNOC, NULL,
568      EF_BAD, NSC_DEITY | NSC_EXTRA},
569     {"rejects", fldoff(nat_rejects), NSC_UCHAR, MAXNOC, NULL,
570      EF_NATION_REJECTS, NSC_EXTRA | NSC_BITS},
571     {"flags", fldoff(nat_flags), NSC_LONG, 0, NULL,
572      EF_NATION_FLAGS, NSC_BITS},
573     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
574 #undef CURSTR
575 };
576
577 struct castr nat_ca[sizeof(cou_ca) / sizeof(*cou_ca)];
578 /* initialized by nsc_init() */
579
580 struct castr realm_ca[] = {
581 #define CURSTR struct realmstr
582     /* uid is encoded in cnum, realm */
583     {"cnum", fldoff(r_cnum), NSC_NATID, 0, NULL, EF_NATION, NSC_CONST},
584     {"realm", fldoff(r_realm), NSC_USHORT, 0, NULL, EF_BAD, NSC_CONST},
585     {"xl", fldoff(r_xl), NSC_SHORT, 0, NULL, EF_BAD, 0},
586     {"xh", fldoff(r_xh), NSC_SHORT, 0, NULL, EF_BAD, 0},
587     {"yl", fldoff(r_yl), NSC_SHORT, 0, NULL, EF_BAD, 0},
588     {"yh", fldoff(r_yh), NSC_SHORT, 0, NULL, EF_BAD, 0},
589     {"timestamp", fldoff(r_timestamp), NSC_TIME, 0, NULL,
590      EF_BAD, NSC_EXTRA},
591     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
592 #undef CURSTR
593 };
594
595 struct castr game_ca[] = {
596 #define CURSTR struct gamestr
597     /* no need for uid */
598     {"upd_disable", fldoff(game_upd_disable), NSC_CHAR, 0, NULL,
599      EF_BAD, 0},
600     {"turn", fldoff(game_turn), NSC_SHORT, 0, NULL, EF_BAD, 0},
601     {"tick", fldoff(game_tick), NSC_SHORT, 0, NULL, EF_BAD, NSC_DEITY},
602     {"rt", fldoff(game_rt), NSC_TIME, 0, NULL, EF_BAD, NSC_DEITY},
603     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
604 #undef CURSTR
605 };
606
607 struct castr intrchr_ca[] = {
608 #define CURSTR struct sctintrins
609     /* no need for uid as long as it's not referenced from other tables */
610     {"name", fldoff(in_name), NSC_STRING, 0, NULL, EF_BAD, NSC_CONST},
611     {"lcms", fldoff(in_lcms), NSC_UCHAR, 0, NULL, EF_BAD, 0},
612     {"hcms", fldoff(in_hcms), NSC_UCHAR, 0, NULL, EF_BAD, 0},
613     {"dcost", fldoff(in_dcost), NSC_UCHAR, 0, NULL, EF_BAD, 0},
614     {"mcost", fldoff(in_mcost), NSC_UCHAR, 0, NULL, EF_BAD, 0},
615     {"enable", fldoff(in_enable), NSC_UCHAR, 0, NULL, EF_BAD, 0},
616     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
617 #undef CURSTR
618 };
619
620 struct castr rpt_ca[] = {
621 #define CURSTR struct rptstr
622     {"uid", fldoff(r_uid), NSC_CHAR, 0, NULL, EF_NEWS_CHR, 0},
623     {"newstory", fldoff(r_newstory), NSC_STRING, NUM_RPTS, NULL,
624      EF_BAD, 0},
625     {"good_will", fldoff(r_good_will), NSC_INT, 0, NULL, EF_BAD, 0},
626     {"newspage", fldoff(r_newspage), NSC_INT, 0, NULL,
627      EF_PAGE_HEADINGS, 0},
628     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
629 #undef CURSTR
630 };
631
632 struct castr update_ca[] = {
633     {"time", 0, NSC_TIME, 0, NULL, EF_BAD, 0},
634     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
635 };
636
637 struct castr empfile_ca[] = {
638 #define CURSTR struct empfile
639     {"uid", fldoff(uid), NSC_INT, 0, NULL, EF_TABLE, 0},
640     {"name", fldoff(name), NSC_STRING, 0, NULL, EF_BAD, NSC_CONST},
641     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
642 #undef CURSTR
643 };
644
645 struct castr symbol_ca[] = {
646 #define CURSTR struct symbol
647     /*
648      * value is is const because it has to match what is compiled into
649      * the server.  name is const because clients key on it.
650      */
651     {"value", fldoff(value), NSC_INT, 0, NULL, EF_BAD, NSC_CONST},
652     {"name", fldoff(name), NSC_STRING, 0, NULL, EF_BAD, NSC_CONST},
653     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
654 #undef CURSTR
655 };
656
657 struct castr mdchr_ca[] = {
658 #define CURSTR struct castr
659     /* no need for uid */
660     /* name must come first, clients may rely on it */
661     {"name", fldoff(ca_name), NSC_STRING, 0, NULL, EF_BAD, NSC_CONST},
662     {"type", fldoff(ca_type), NSC_CHAR, 0, NULL, EF_META_TYPE, NSC_CONST},
663     {"flags", fldoff(ca_flags), NSC_UCHAR, 0, NULL,
664      EF_META_FLAGS, NSC_CONST | NSC_BITS},
665     {"len", fldoff(ca_len), NSC_USHORT, 0, NULL, EF_BAD, NSC_CONST},
666     {"table", fldoff(ca_table), NSC_INT, 0, NULL, EF_BAD, NSC_CONST},
667     {NULL, 0, NSC_NOTYPE, 0, NULL, EF_BAD, 0}
668 #undef CURSTR
669 };
670
671 void
672 nsc_init(void)
673 {
674     static struct castr version_ca0 = {
675         "version", 0, NSC_STRINGY, sizeof(PACKAGE_STRING), NULL, EF_BAD, 0
676     };
677     static struct castr *ca;
678     struct keymatch *kp;
679     int n, i;
680     unsigned flags;
681
682     /* derive empfile[EF_VERSION].cadef from configkeys[] */
683     n = 0;
684     for (kp = configkeys; kp->km_key; ++kp) {
685         if (kp->km_type != NSC_NOTYPE && !(kp->km_flags & KM_INTERNAL))
686             n++;
687     }
688     ca = calloc(n + 2, sizeof(*ca));
689     ca[0] = version_ca0;
690     i = 1;
691     for (kp = configkeys; kp->km_key; ++kp) {
692         if (kp->km_type != NSC_NOTYPE && !(kp->km_flags & KM_INTERNAL)) {
693             ca[i].ca_type = kp->km_type;
694             ca[i].ca_off = kp - configkeys;
695             ca[i].ca_name = kp->km_key;
696             ca[i].ca_table = EF_BAD;
697             ca[i].ca_get = nsc_ver;
698             i++;
699         }
700     }
701     empfile[EF_VERSION].cadef = ca;
702
703     /* derive nat_ca[] from cou_ca[] */
704     for (i = 0; cou_ca[i].ca_name; i++) {
705         nat_ca[i] = cou_ca[i];
706         flags = nat_ca[i].ca_flags;
707         if (flags & NSC_EXTRA)
708             flags &= ~NSC_EXTRA;
709         else if (i != 0)
710             flags |= NSC_DEITY;
711         nat_ca[i].ca_flags = flags;
712     }
713     nat_ca[i] = cou_ca[i];
714 }
715
716 /*
717  * Virtual selectors
718  */
719
720 static void *
721 nsc_ver(struct valstr *val, natid cnum, void *ptr)
722 {
723     struct keymatch *kp = &configkeys[val->val_as.sym.off];
724     val->val_as.sym.off = 0;
725     val->val_as.sym.get = NULL;
726     return kp->km_data;
727 }