source: git/src/cavern.h @ 9914d6b

walls-datawalls-data-hanging-as-warning
Last change on this file since 9914d6b was 87d7f7c, checked in by Olly Betts <olly@…>, 2 months ago

Merge branch 'master' into walls-data

  • Property mode set to 100644
File size: 15.4 KB
Line 
1/* cavern.h
2 * SURVEX Cave surveying software - header file
3 * Copyright (C) 1991-2024 Olly Betts
4 * Copyright (C) 2004 Simeon Warner
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., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
19 */
20
21#ifndef CAVERN_H
22#define CAVERN_H
23
24/* Using covariances increases the memory required somewhat - may be
25 * desirable to disable this for small memory machines */
26
27/* #define NO_COVARIANCES 1 */
28
29#include <stdio.h>
30#include <stdlib.h>
31#include <string.h>
32#include <ctype.h>
33#include <math.h>
34#include <float.h>
35
36#include <proj.h>
37
38#include "img_hosted.h"
39#include "str.h"
40#include "useful.h"
41
42/* Set EXPLICIT_FIXED_FLAG to 1 to force an explicit fixed flag to be used
43 * in each pos struct, rather than using p[0]==UNFIXED_VAL to indicate
44 * unfixed-ness.  This may be slightly faster, but uses more memory.
45 */
46#ifndef EXPLICIT_FIXED_FLAG
47# define EXPLICIT_FIXED_FLAG 0
48#endif
49
50typedef double real; /* so we can change the precision used easily */
51#define HUGE_REAL HUGE_VAL
52#define REAL_EPSILON DBL_EPSILON
53
54#if (!EXPLICIT_FIXED_FLAG)
55# define UNFIXED_VAL HUGE_VAL /* if p[0]==UNFIXED_VAL, station is unfixed */
56#endif
57
58#define SPECIAL_EOL             0x0001
59#define SPECIAL_BLANK           0x0002
60#define SPECIAL_KEYWORD         0x0004
61#define SPECIAL_COMMENT         0x0008
62#define SPECIAL_OMIT            0x0010
63#ifndef NO_DEPRECATED
64#define SPECIAL_ROOT            0x0020
65#endif
66#define SPECIAL_SEPARATOR       0x0040
67#define SPECIAL_NAMES           0x0080
68#define SPECIAL_DECIMAL         0x0100
69#define SPECIAL_MINUS           0x0200
70#define SPECIAL_PLUS            0x0400
71#define SPECIAL_OPEN            0x0800
72#define SPECIAL_CLOSE           0x1000
73#define SPECIAL_ANON            0x2000
74
75extern char *fnm_output_base;
76extern int fnm_output_base_is_dir;
77
78extern bool fExportUsed;
79
80extern int current_days_since_1900;
81
82/* Types */
83
84typedef enum {
85   Q_NULL = -1, Q_DEFAULT, Q_POS, Q_PLUMB, Q_LEVEL,
86   Q_GRADIENT, Q_BACKGRADIENT, Q_BEARING, Q_BACKBEARING,
87   Q_LENGTH, Q_BACKLENGTH, Q_DEPTH, Q_DX, Q_DY, Q_DZ, Q_COUNT, Q_DECLINATION,
88   Q_LEFT, Q_RIGHT, Q_UP, Q_DOWN,
89   Q_MAC
90} q_quantity;
91
92typedef enum {
93   INFER_NULL = -1,
94   INFER_EQUATES,
95   INFER_EXPORTS,
96   INFER_PLUMBS,
97   INFER_SUBSURVEYS,
98   /* In Compass DAT files a dummy zero-length leg from a station to itself is
99    * used to provide a place to specify LRUD for the start or end of a
100    * traverse (depending if dimensions are measured at the from or to
101    * station), so we shouldn't warn about equating a station to itself.
102    * This should be set *as well as* INFER_EQUATES.
103    */
104   INFER_EQUATES_SELF_OK
105} infer_what;
106
107/* unsigned long to cope with 16-bit int-s */
108#define BIT(N) (1UL << (N))
109#define BITA(N) (1UL << ((N) - 'a'))
110
111#define TSTBIT(W, N) (((W)>>(N))&1)
112
113/* masks for quantities which are length and angles respectively */
114#define LEN_QMASK (BIT(Q_LENGTH) | BIT(Q_BACKLENGTH) | BIT(Q_DEPTH) |\
115   BIT(Q_DX) | BIT(Q_DY) | BIT(Q_DZ) | BIT(Q_POS) | BIT(Q_COUNT) |\
116   BIT(Q_LEFT) | BIT(Q_RIGHT) | BIT(Q_UP) | BIT(Q_DOWN))
117#define ANG_QMASK (BIT(Q_BEARING) | BIT(Q_BACKBEARING) |\
118   BIT(Q_GRADIENT) | BIT(Q_BACKGRADIENT) | BIT(Q_PLUMB) | BIT(Q_LEVEL) |\
119   BIT(Q_DECLINATION))
120
121/* if you add/change the order, check factor_tab in commands.c */
122typedef enum {
123   UNITS_NULL = -1, UNITS_METRES, UNITS_FEET, UNITS_YARDS,
124   UNITS_DEGS, UNITS_QUADRANTS, UNITS_GRADS, UNITS_PERCENT, UNITS_MINUTES,
125   UNITS_MAC, UNITS_DEPRECATED_ALIAS_FOR_GRADS
126} u_units;
127
128/* don't reorder these values!  They need to match with img.h too */
129typedef enum {
130   FLAGS_NOT = -2, FLAGS_UNKNOWN = -1, FLAGS_SURFACE, FLAGS_DUPLICATE,
131   FLAGS_SPLAY,
132#if 0
133   /* underground, but through rock (e.g. radiolocation).  Want to hide from
134    * plots by default (so not cave) but don't want to include in surface
135    * triangulation nets (so not surface) */
136   FLAGS_SKELETAL, /* FIXME */
137#endif
138   /* Don't need to match img.h: */
139   FLAGS_ANON_ONE_END,
140   FLAGS_IMPLICIT_SPLAY,
141   FLAGS_STYLE_BIT0, FLAGS_STYLE_BIT1, FLAGS_STYLE_BIT2
142} flags;
143
144/* flags are currently stored in an unsigned char */
145typedef int compiletimeassert_flags0[FLAGS_STYLE_BIT2 <= 7 ? 1 : -1];
146
147/* Mask to AND with to get bits to pass to img library. */
148#define FLAGS_MASK \
149    (BIT(FLAGS_SURFACE) | BIT(FLAGS_DUPLICATE) | BIT(FLAGS_SPLAY))
150
151typedef int compiletimeassert_flags1[BIT(FLAGS_SURFACE) == img_FLAG_SURFACE ? 1 : -1];
152typedef int compiletimeassert_flags2[BIT(FLAGS_DUPLICATE) == img_FLAG_DUPLICATE ? 1 : -1];
153typedef int compiletimeassert_flags3[BIT(FLAGS_SPLAY) == img_FLAG_SPLAY ? 1 : -1];
154
155typedef enum {
156   /* Don't reorder these values!  They need to match with img.h too. */
157   SFLAGS_SURFACE = 0, SFLAGS_UNDERGROUND, SFLAGS_ENTRANCE, SFLAGS_EXPORTED,
158   SFLAGS_FIXED, SFLAGS_ANON, SFLAGS_WALL,
159   /* These values don't need to match img.h, but mustn't clash. */
160   SFLAGS_USED = 11,
161   SFLAGS_SOLVED = 12, SFLAGS_SUSPECTTYPO = 13, SFLAGS_SURVEY = 14, SFLAGS_PREFIX_ENTERED = 15
162} sflags;
163
164/* Mask to AND with to get bits to pass to img library. */
165#define SFLAGS_MASK (BIT(SFLAGS_SURFACE) | BIT(SFLAGS_UNDERGROUND) |\
166        BIT(SFLAGS_ENTRANCE) | BIT(SFLAGS_EXPORTED) | BIT(SFLAGS_FIXED) |\
167        BIT(SFLAGS_ANON) | BIT(SFLAGS_WALL))
168
169typedef int compiletimeassert_sflags1[BIT(SFLAGS_SURFACE) == img_SFLAG_SURFACE ? 1 : -1];
170typedef int compiletimeassert_sflags2[BIT(SFLAGS_UNDERGROUND) == img_SFLAG_UNDERGROUND ? 1 : -1];
171typedef int compiletimeassert_sflags3[BIT(SFLAGS_ENTRANCE) == img_SFLAG_ENTRANCE ? 1 : -1];
172typedef int compiletimeassert_sflags4[BIT(SFLAGS_EXPORTED) == img_SFLAG_EXPORTED ? 1 : -1];
173typedef int compiletimeassert_sflags5[BIT(SFLAGS_FIXED) == img_SFLAG_FIXED ? 1 : -1];
174typedef int compiletimeassert_sflags6[BIT(SFLAGS_ANON) == img_SFLAG_ANON ? 1 : -1];
175typedef int compiletimeassert_sflags7[BIT(SFLAGS_WALL) == img_SFLAG_WALL ? 1 : -1];
176
177/* enumeration of field types */
178typedef enum {
179   End = 0, Tape, Comp, Clino, BackTape, BackComp, BackClino,
180   Left, Right, Up, Down,
181   FrDepth, ToDepth, Dx, Dy, Dz, FrCount, ToCount,
182   /* Up to here are readings are allowed multiple values
183    * and have slot in the value[] array in datain.c.
184    * (Depth, DepthChange, and Count can have multiple
185    * readings, but are actually handled using tokens
186    * above rather than as themselves).
187    *
188    * Fr must be the first reading after this comment!
189    */
190   Fr, To, Station, Depth, DepthChange, Count, Dir,
191   Newline, IgnoreAllAndNewLine, Ignore, IgnoreAll,
192   /* IgnoreAll must be the last reading before this comment!
193    *
194    * Readings after this comment are only used in datain.c
195    * so can have enum values >= 32 because we only use a
196    * bitmask for those readings used in commands.c.
197    */
198   CompassDATFr, CompassDATTo,
199   CompassDATComp, CompassDATClino, CompassDATBackComp, CompassDATBackClino,
200   CompassDATLeft, CompassDATRight, CompassDATUp, CompassDATDown,
201   CompassDATFlags,
202
203   WallsSRVFr, WallsSRVTo, WallsSRVTape, WallsSRVComp, WallsSRVClino
204} reading;
205
206/* if IgnoreAll is >= 32, the compiler will choke on this */
207typedef char compiletimeassert_reading[IgnoreAll < 32 ? 1 : -1];
208
209/* position or length vector */
210typedef real delta[3];
211
212/* variance */
213#ifdef NO_COVARIANCES
214typedef real var[3];
215typedef var svar;
216#else
217typedef real var[3][3];
218typedef real svar[6];
219#endif
220
221/* station name */
222typedef struct Prefix {
223   struct Prefix *up, *down, *right;
224   struct Node *stn;
225   struct Pos *pos;
226   const char *ident;
227   const char *filename;
228   unsigned int line;
229   /* If (min_export == 0) then max_export is max # levels above is this
230    * prefix is used (and so needs to be exported) (0 == parent only).
231    * If (min_export > 0) then max_export is max # levels above this
232    * prefix has been exported, and min_export is how far down the exports
233    * have got (if min_export > 1 after a run, this prefix hasn't been
234    * exported from below enough).
235    * If INFER_EXPORTS is active when a station is encountered, we
236    * set min_export = USHRT_MAX and max_export gets set as usual.  Then at
237    * the end of the run, we also mark stations with min_export == USHRT_MAX
238    * and max_export > 0 as exported. */
239   unsigned short max_export, min_export;
240   /* stn flags - e.g. surface, underground, entrance
241    * also suspecttypo and survey */
242   unsigned short sflags;
243   short shape;
244} prefix;
245
246/* survey metadata */
247typedef struct Meta_data {
248    size_t ref_count;
249    /* Days since 1900 for start and end date of survey, or -1 if undated. */
250    int days1, days2;
251} meta_data;
252
253/* stuff stored for both forward & reverse legs */
254typedef struct {
255   struct Node *to;
256   /* bits 0..1 = reverse leg number; bit7 is fFullLeg */
257   /* bit6 = fReplacementLeg (by reduction rules) */
258   /* bit5 = articulation leg (i.e. carries no error) */
259   unsigned char reverse;
260   /* flags - e.g. surface, duplicate survey
261    * only used if (FLAG_DATAHERE & !(FLAG_REPLACEMENTLEG|FLAG_FAKE))
262    * This could be only in linkfor, but this is actually more space
263    * efficient.
264    */
265   unsigned char flags;
266} linkcommon;
267
268#define FLAG_DATAHERE 0x80
269#define FLAG_REPLACEMENTLEG 0x40
270#define FLAG_ARTICULATION 0x20
271#define FLAG_FAKE 0x10 /* an equate or leg inside an sdfix */
272#define MASK_REVERSEDIRN 0x03
273
274/* reverse leg - deltas & vars stored on other dirn */
275typedef struct LinkRev {
276   linkcommon l;
277} linkrev;
278
279/* forward leg - deltas & vars stored here */
280typedef struct Link {
281   linkcommon l;
282   delta d; /* Delta */
283   svar v; /* Variances */
284   meta_data *meta;
285} linkfor;
286
287/* node - like a station, except several nodes are used to represent a
288 * station with more than 3 legs connected to it
289 */
290typedef struct Node {
291   struct Prefix *name;
292   struct Link *leg[3];
293   struct Node *prev, *next;
294   long colour;
295} node;
296
297/* station position */
298typedef struct Pos {
299   delta p; /* Position */
300#if EXPLICIT_FIXED_FLAG
301   unsigned char fFixed; /* flag indicating if station is a fixed point */
302#endif
303} pos;
304
305/*
306typedef struct Inst {
307   real zero, scale, units;
308} inst;
309*/
310
311/* Survey data styles */
312#define STYLE_NORMAL     0
313#define STYLE_DIVING     1
314#define STYLE_CARTESIAN  2
315#define STYLE_CYLPOLAR   3
316#define STYLE_NOSURVEY   4
317#define STYLE_PASSAGE    5
318#define STYLE_IGNORE     6
319
320typedef int compiletimeassert_style1[STYLE_NORMAL == img_STYLE_NORMAL ? 1 : -1];
321typedef int compiletimeassert_style2[STYLE_DIVING == img_STYLE_DIVING ? 1 : -1];
322typedef int compiletimeassert_style3[STYLE_CARTESIAN == img_STYLE_CARTESIAN ? 1 : -1];
323typedef int compiletimeassert_style4[STYLE_CYLPOLAR == img_STYLE_CYLPOLAR ? 1 : -1];
324typedef int compiletimeassert_style5[STYLE_NOSURVEY == img_STYLE_NOSURVEY ? 1 : -1];
325
326/* various settings preserved by *BEGIN and *END */
327typedef struct Settings {
328   struct Settings *next;
329   unsigned int Truncate;
330   bool f_clino_percent;
331   bool f_backclino_percent;
332   bool f_bearing_quadrants;
333   bool f_backbearing_quadrants;
334   bool dash_for_anon_wall_station;
335   unsigned char infer;
336   enum {OFF, LOWER, UPPER} Case;
337   /* STYLE_xxx value to process data as. */
338   int style;
339   /* STYLE_xxx value to put in 3d file (different for Compass DAT diving
340    * data, as the data in the DAT file is always presented in the format
341    * tape,compass,clino even if that isn't how it was really measured).
342    */
343   int recorded_style;
344   prefix *Prefix;
345   prefix *begin_survey; /* used to check BEGIN and END match */
346   short *Translate; /* if short is >= 16 bits, which ANSI requires */
347   real Var[Q_MAC];
348   real z[Q_MAC];
349   real sc[Q_MAC];
350   real units[Q_MAC];
351   const reading *ordering;
352   int begin_lineno; /* 0 means no block started in this file */
353   int flags;
354   char* proj_str;
355   /* Location at which we calculate the declination if
356    * z[Q_DECLINATION] == HUGE_REAL.
357    *
358    * Latitude and longitude are in radians; altitude is in metres above the
359    * ellipsoid.
360    */
361   real dec_lat, dec_lon, dec_alt;
362   /* Cached auto-declination in radians, or HUGE_REAL for no cached value.
363    * Only meaningful if date1 != -1.
364    */
365   real declination;
366   double min_declination, max_declination;
367   int min_declination_days, max_declination_days;
368   const char* dec_filename;
369   int dec_line;
370   /* Copy of the text of the `*declination auto ...` line (malloced). */
371   char* dec_context;
372   /* Grid convergence in radians. */
373   real convergence;
374   meta_data * meta;
375} settings;
376
377/* global variables */
378extern settings *pcs;
379extern prefix *root;
380extern prefix *anon_list;
381extern node *stnlist;
382extern unsigned long optimize;
383extern char * proj_str_out;
384extern PJ * pj_cached;
385
386extern string survey_title;
387
388extern bool fExplicitTitle;
389extern long cLegs, cStns, cComponents;
390extern FILE *fhErrStat;
391extern img *pimg;
392extern real totadj, total, totplan, totvert;
393extern real min[6], max[6];
394extern prefix *pfxHi[6], *pfxLo[6];
395extern bool fQuiet; /* just show brief summary + errors */
396extern bool fMute; /* just show errors */
397extern bool fSuppress; /* only output 3d file */
398
399/* macros */
400
401#define POS(S, D) ((S)->name->pos->p[(D)])
402#define POSD(S) ((S)->name->pos->p)
403
404#define data_here(L) ((L)->l.reverse & FLAG_DATAHERE)
405#define reverse_leg_dirn(L) ((L)->l.reverse & MASK_REVERSEDIRN)
406#define reverse_leg(L) ((L)->l.to->leg[reverse_leg_dirn(L)])
407
408#if EXPLICIT_FIXED_FLAG
409# define pfx_fixed(N) ((N)->pos->fFixed)
410# define pos_fixed(P) ((P)->fFixed)
411# define fix(S) (S)->name->pos->fFixed = (char)true
412# define fixpos(P) (P)->fFixed = (char)true
413# define unfix(S) (S)->name->pos->fFixed = (char)false
414#else
415# define pfx_fixed(N) ((N)->pos->p[0] != UNFIXED_VAL)
416# define pos_fixed(P) ((P)->p[0] != UNFIXED_VAL)
417# define fix(S) NOP
418# define fixpos(P) NOP
419# define unfix(S) POS((S), 0) = UNFIXED_VAL
420#endif
421#define fixed(S) pfx_fixed((S)->name)
422
423/* macros for special chars */
424
425#define isEol(c)    (pcs->Translate[(c)] & SPECIAL_EOL)
426#define isBlank(c)  (pcs->Translate[(c)] & SPECIAL_BLANK)
427#define isKeywd(c)  (pcs->Translate[(c)] & SPECIAL_KEYWORD)
428#define isComm(c)   (pcs->Translate[(c)] & SPECIAL_COMMENT)
429#define isOmit(c)   (pcs->Translate[(c)] & SPECIAL_OMIT)
430#ifndef NO_DEPRECATED
431#define isRoot(c)   (pcs->Translate[(c)] & SPECIAL_ROOT)
432#endif
433#define isSep(c)    (pcs->Translate[(c)] & SPECIAL_SEPARATOR)
434#define isNames(c)  (pcs->Translate[(c)] & SPECIAL_NAMES)
435#define isDecimal(c) (pcs->Translate[(c)] & SPECIAL_DECIMAL)
436#define isMinus(c)  (pcs->Translate[(c)] & SPECIAL_MINUS)
437#define isPlus(c)   (pcs->Translate[(c)] & SPECIAL_PLUS)
438#define isOpen(c)   (pcs->Translate[(c)] & SPECIAL_OPEN)
439#define isClose(c)  (pcs->Translate[(c)] & SPECIAL_CLOSE)
440#define isAnon(c)   (pcs->Translate[(c)] & SPECIAL_ANON)
441
442#define isSign(c)   (pcs->Translate[(c)] & (SPECIAL_PLUS | SPECIAL_MINUS))
443#define isData(c)   (pcs->Translate[(c)] & (SPECIAL_OMIT | SPECIAL_ROOT|\
444   SPECIAL_SEPARATOR | SPECIAL_NAMES | SPECIAL_DECIMAL | SPECIAL_PLUS |\
445   SPECIAL_MINUS))
446
447typedef struct nosurveylink {
448   node *fr, *to;
449   int flags;
450   meta_data *meta;
451   struct nosurveylink *next;
452} nosurveylink;
453
454extern nosurveylink *nosurveyhead;
455
456typedef struct lrud {
457    struct lrud * next;
458    prefix *stn;
459    meta_data *meta;
460    real l, r, u, d;
461} lrud;
462
463typedef struct lrudlist {
464    lrud * tube;
465    struct lrudlist * next;
466} lrudlist;
467
468extern lrudlist * model;
469
470extern lrud ** next_lrud;
471
472extern char output_separator;
473
474#endif /* CAVERN_H */
Note: See TracBrowser for help on using the repository browser.