xref: /netbsd-src/external/bsd/ntp/dist/sntp/libopts/init.c (revision 80d9064ac03cbb6a4174695f0d5b237c8766d3d0)
1 /*	$NetBSD: init.c,v 1.2 2013/12/28 03:20:15 christos Exp $	*/
2 
3 /**
4  * \file initialize.c
5  *
6  *  initialize the libopts data structures.
7  *
8  * @addtogroup autoopts
9  * @{
10  */
11 /*
12  *  This file is part of AutoOpts, a companion to AutoGen.
13  *  AutoOpts is free software.
14  *  AutoOpts is Copyright (C) 1992-2013 by Bruce Korb - all rights reserved
15  *
16  *  AutoOpts is available under any one of two licenses.  The license
17  *  in use must be one of these two and the choice is under the control
18  *  of the user of the license.
19  *
20  *   The GNU Lesser General Public License, version 3 or later
21  *      See the files "COPYING.lgplv3" and "COPYING.gplv3"
22  *
23  *   The Modified Berkeley Software Distribution License
24  *      See the file "COPYING.mbsd"
25  *
26  *  These files have the following sha256 sums:
27  *
28  *  8584710e9b04216a394078dc156b781d0b47e1729104d666658aecef8ee32e95  COPYING.gplv3
29  *  4379e7444a0e2ce2b12dd6f5a52a27a4d02d39d247901d3285c88cf0d37f477b  COPYING.lgplv3
30  *  13aa749a5b0a454917a944ed8fffc530b784f5ead522b1aacaf4ec8aa55a6239  COPYING.mbsd
31  */
32 
33 /* = = = START-STATIC-FORWARD = = = */
34 static tSuccess
35 do_presets(tOptions * opts);
36 /* = = = END-STATIC-FORWARD = = = */
37 
38 /**
39  *  Make sure the option descriptor is there and that we understand it.
40  *  This should be called from any user entry point where one needs to
41  *  worry about validity.  (Some entry points are free to assume that
42  *  the call is not the first to the library and, thus, that this has
43  *  already been called.)
44  *
45  *  Upon successful completion, pzProgName and pzProgPath are set.
46  *
47  *  @param[in,out] opts   program options descriptor
48  *  @param[in]     pname  name of program, from argv[]
49  *  @returns SUCCESS or FAILURE
50  */
51 LOCAL tSuccess
52 validate_struct(tOptions * opts, char const * pname)
53 {
54     if (opts == NULL) {
55         fputs(zno_opt_arg, stderr);
56         return FAILURE;
57     }
58     print_exit = ((opts->fOptSet & OPTPROC_SHELL_OUTPUT) != 0);
59 
60     /*
61      *  IF the client has enabled translation and the translation procedure
62      *  is available, then go do it.
63      */
64     if (  ((opts->fOptSet & OPTPROC_TRANSLATE) != 0)
65        && (opts->pTransProc != NULL)
66        && (option_xlateable_txt.field_ct != 0) ) {
67         /*
68          *  If option names are not to be translated at all, then do not do
69          *  it for configuration parsing either.  (That is the bit that really
70          *  gets tested anyway.)
71          */
72         if ((opts->fOptSet & OPTPROC_NO_XLAT_MASK) == OPTPROC_NXLAT_OPT)
73             opts->fOptSet |= OPTPROC_NXLAT_OPT_CFG;
74         (*opts->pTransProc)();
75     }
76 
77     /*
78      *  IF the struct version is not the current, and also
79      *     either too large (?!) or too small,
80      *  THEN emit error message and fail-exit
81      */
82     if (  ( opts->structVersion  != OPTIONS_STRUCT_VERSION  )
83        && (  (opts->structVersion > OPTIONS_STRUCT_VERSION  )
84           || (opts->structVersion < OPTIONS_MINIMUM_VERSION )
85        )  )  {
86 
87         static char const ao_ver_string[] =
88             STR(AO_CURRENT)":"STR(AO_REVISION)":"STR(AO_AGE)"\n";
89 
90         fprintf(stderr, zwrong_ver, pname, NUM_TO_VER(opts->structVersion));
91         if (opts->structVersion > OPTIONS_STRUCT_VERSION )
92             fputs(ztoo_new, stderr);
93         else
94             fputs(ztoo_old, stderr);
95 
96         fwrite(ao_ver_string, sizeof(ao_ver_string) - 1, 1, stderr);
97         return FAILURE;
98     }
99 
100     /*
101      *  If the program name hasn't been set, then set the name and the path
102      *  and the set of equivalent characters.
103      */
104     if (opts->pzProgName == NULL) {
105         char const *  pz = strrchr(pname, DIRCH);
106         char const ** pp =
107             (char const **)(void **)(intptr_t)&(opts->pzProgName);
108 
109         if (pz != NULL)
110             *pp = pz+1;
111         else
112             *pp = pname;
113 
114         pz = pathfind(getenv("PATH"), (char *)(intptr_t)pname, "rx");
115         if (pz != NULL)
116             pname = (void *)(intptr_t)pz;
117 
118         pp  = (char const **)(void **)(intptr_t)&(opts->pzProgPath);
119         *pp = pname;
120 
121         /*
122          *  when comparing long names, these are equivalent
123          */
124         strequate(zSepChars);
125     }
126 
127     return SUCCESS;
128 }
129 
130 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
131  *
132  *  DO PRESETS
133  *
134  *  The next several routines do the immediate action pass on the command
135  *  line options, then the environment variables, then the config files in
136  *  reverse order.  Once done with that, the order is reversed and all
137  *  the config files and environment variables are processed again, this
138  *  time only processing the non-immediate action options.  do_presets()
139  *  will then return for optionProcess() to do the final pass on the command
140  *  line arguments.
141  */
142 
143 /**
144  *  scan the command line for immediate action options.
145  *  This is only called the first time through.
146  *  While this procedure is active, the OPTPROC_IMMEDIATE is true.
147  *
148  *  @param pOpts   program options descriptor
149  *  @returns SUCCESS or FAILURE
150  */
151 LOCAL tSuccess
152 immediate_opts(tOptions * opts)
153 {
154     tSuccess  res;
155 
156     opts->fOptSet  |= OPTPROC_IMMEDIATE;
157     opts->curOptIdx = 1;     /* start by skipping program name */
158     opts->pzCurOpt  = NULL;
159 
160     /*
161      *  Examine all the options from the start.  We process any options that
162      *  are marked for immediate processing.
163      */
164     for (;;) {
165         tOptState opt_st = OPTSTATE_INITIALIZER(PRESET);
166 
167         res = next_opt(opts, &opt_st);
168         switch (res) {
169         case FAILURE: goto   failed_option;
170         case PROBLEM: res = SUCCESS; goto leave;
171         case SUCCESS: break;
172         }
173 
174         /*
175          *  IF this is an immediate-attribute option, then do it.
176          */
177         if (! DO_IMMEDIATELY(opt_st.flags))
178             continue;
179 
180         if (! SUCCESSFUL(handle_opt(opts, &opt_st)))
181             break;
182     } failed_option:;
183 
184     if ((opts->fOptSet & OPTPROC_ERRSTOP) != 0)
185         (*opts->pUsageProc)(opts, EXIT_FAILURE);
186 
187  leave:
188 
189     opts->fOptSet &= ~OPTPROC_IMMEDIATE;
190     return res;
191 }
192 
193 /**
194  *  check for preset values from a config files or envrionment variables
195  *
196  * @param[in,out] opts  the structure with the option names to check
197  */
198 static tSuccess
199 do_presets(tOptions * opts)
200 {
201     tOptDesc * od = NULL;
202 
203     if (! SUCCESSFUL(immediate_opts(opts)))
204         return FAILURE;
205 
206     /*
207      *  IF this option set has a --save-opts option, then it also
208      *  has a --load-opts option.  See if a command line option has disabled
209      *  option presetting.
210      */
211     if (  (opts->specOptIdx.save_opts != NO_EQUIVALENT)
212        && (opts->specOptIdx.save_opts != 0)) {
213         od = opts->pOptDesc + opts->specOptIdx.save_opts + 1;
214         if (DISABLED_OPT(od))
215             return SUCCESS;
216     }
217 
218     /*
219      *  Until we return from this procedure, disable non-presettable opts
220      */
221     opts->fOptSet |= OPTPROC_PRESETTING;
222     /*
223      *  IF there are no config files,
224      *  THEN do any environment presets and leave.
225      */
226     if (opts->papzHomeList == NULL) {
227         env_presets(opts, ENV_ALL);
228     }
229     else {
230         env_presets(opts, ENV_IMM);
231 
232         /*
233          *  Check to see if environment variables have disabled presetting.
234          */
235         if ((od != NULL) && ! DISABLED_OPT(od))
236             intern_file_load(opts);
237 
238         /*
239          *  ${PROGRAM_LOAD_OPTS} value of "no" cannot disable other environment
240          *  variable options.  Only the loading of .rc files.
241          */
242         env_presets(opts, ENV_NON_IMM);
243     }
244     opts->fOptSet &= ~OPTPROC_PRESETTING;
245 
246     return SUCCESS;
247 }
248 
249 /**
250  * AutoOpts initialization
251  *
252  * @param[in,out] opts  the structure to initialize
253  * @param[in]     a_ct  program argument count
254  * @param[in]     a_v   program argument vector
255  */
256 LOCAL bool
257 ao_initialize(tOptions * opts, int a_ct, char ** a_v)
258 {
259     if ((opts->fOptSet & OPTPROC_INITDONE) != 0)
260         return true;
261 
262     opts->origArgCt   = (unsigned int)a_ct;
263     opts->origArgVect = a_v;
264     opts->fOptSet    |= OPTPROC_INITDONE;
265 
266     if (HAS_pzPkgDataDir(opts))
267         program_pkgdatadir = opts->pzPkgDataDir;
268 
269     if (! SUCCESSFUL(do_presets(opts)))
270         return false;
271 
272     /*
273      *  IF option name conversion was suppressed but it is not suppressed
274      *  for the command line, then it's time to translate option names.
275      *  Usage text will not get retranslated.
276      */
277     if (  ((opts->fOptSet & OPTPROC_TRANSLATE) != 0)
278        && (opts->pTransProc != NULL)
279        && ((opts->fOptSet & OPTPROC_NO_XLAT_MASK) == OPTPROC_NXLAT_OPT_CFG)
280        )  {
281         opts->fOptSet &= ~OPTPROC_NXLAT_OPT_CFG;
282         (*opts->pTransProc)();
283     }
284 
285     if ((opts->fOptSet & OPTPROC_REORDER) != 0)
286         optionSort(opts);
287 
288     opts->curOptIdx   = 1;
289     opts->pzCurOpt    = NULL;
290     return true;
291 }
292 
293 /** @}
294  *
295  * Local Variables:
296  * mode: C
297  * c-file-style: "stroustrup"
298  * indent-tabs-mode: nil
299  * End:
300  * end of autoopts/initialize.c */
301