source: git/src/cavern.h @ 3830dc0

debug-cidebug-ci-sanitisersfaster-cavernloglog-selectstereo-2025walls-datawalls-data-hanging-as-warningwarn-only-for-hanging-survey
Last change on this file since 3830dc0 was 3830dc0, checked in by Olly Betts <olly@…>, 4 years ago

Rename declination location variables to be clearer

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