1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
|
==============
Option parsing
==============
Characteristics:
* short options:
* bundling of argument-free options (``-a -b -c`` -> ``-abc``)
* squashing of argument-requiring options (``-a foo`` -> ``-afoo``)
* GNU long options with space or equals sign (``--foo bar``, ``--foo=bar``)
* proper recognition of the lone dash (often used to indicate stdin/stdout)
* recognition of the double dash as option list terminator
* offers POSIX strictness where the option list terminates at the first
non-option argument
* option passthrough
* the parse function is one-shot; there is no context object (like popt),
no global state (like getopt) and no ``while`` loop (either of the two others)
* exclusively uses an option table
* value storing is performed through pointers in the option table
* or user-provided callbacks can be invoked per option
Synopsis
========
.. code-block:: c
#include <libHX/option.h>
struct HXoption {
const char *ln;
char sh;
unsigned int type;
void *ptr, *uptr;
void (*cb)(const struct HXoptcb *);
int val;
const char *help, *htyp;
};
int HX_getopt5(const struct HXoption *options_table, char **argv, int *new_argc, char ***new_argv, unsigned int flags);
The various fields of ``struct HXoption`` are:
``ln``
The long option name, if any. May be ``NULL`` if none is to be assigned
for this entry.
``sh``
The short option name/character, if any. May be '``\0``' if none is to
be assigned for this entry.
``type``
The type of the entry, essentially denoting the type of the target
variable (``ptr``).
``val``
An integer value to be stored into ``*(int *)ptr`` if the option type
is ``HXTYPE_IVAL``.
``ptr``
A pointer to the variable so that the option parser can store the
requested data in it. The pointer may be ``NULL``, in which case no
data is stored (but ``cb`` is still called if defined, with the data).
``uptr``
A user-supplied pointer. Its value is passed verbatim to the callback,
and may be used for any purpose the user wishes. If the option type is
``HXTYPE_SVAL``, it is the value in uptr that will be used to populate
``*(const char **)ptr``.
``cb``
If not ``NULL``, call out to the referenced function after the option
has been parsed (and the results possibly be stored in ``ptr``).
``help``
A help string that is shown for the option when the option table is
dumped by request (e.g. ``yourprgram --help``).
``htyp``
String containing a keyword to aid the user in understanding the
available options during dump. See examples.
Due to the amount of fields, it is advised to use C99/C++20 named initializers
to populate a struct, as they allow to omit unspecified fields, and assume no
specific order of the members:
.. code-block:: c
struct HXoption e = {.sh = 'f', .help = "Force"};
Type map
========
``HXTYPE_NONE``
The option does not take any argument, but the presence of the option
may be record by setting ``*(int *)ptr`` to ``1``. Other rules apply
when ``HXOPT_INC`` or ``HXOPT_DEC`` are specified as flags.
``HXTYPE_VAL``
Use the integer value specified by ``ival`` and store it in
``*(int *)ptr``.
``HXTYPE_SVAL``
Use the memory location specified by ``sval`` and store it in ``*(const
char **)ptr``. OUTDATED.
``HXTYPE_BOOL``
Interpret the supplied argument as a boolean descriptive (must be
``yes``, ``no``, ``on``, ``off``, ``true``, ``false``, ``0`` or ``1``)
and store the result in ``*(int *)ptr``.
``HXTYPE_STRING``
The argument string is duplicated to a new memory region and the
resulting pointer stored into ``*(char **)ptr``. This incurs an
allocation so that subsequently modifying the original argument string
in any way will not falsely propagate.
``HXTYPE_STRDQ``
The argument string is duplicated to a new memory region and the
resulting pointer is added to the given HXdeque. Note that you often
need to use deferred initialization of the options table to avoid
putting ``NULL`` into the entry. See section about pitfalls.
The following overview lists the types that map to the common integral and
floating-point types. Signed and unsigned integeral types are processed using
``strtol`` and ``strtoul``, respectively. ``strtol`` and ``strtoul`` will be
called with automatic base detection. This usually means that a leading ``0``
indicates the string is given in octal base, a leading ``0x`` indicates
hexadecimal base, and decimal otherwise. ``HXTYPE_LLONG``, ``HXTYPE_ULLONG``,
``HXTYPE_INT64`` and ``HXTYPE_UINT64`` use ``strtoll`` and/or ``strtoull``,
which may not be available on all platforms.
``HXTYPE_CHAR``
maps to ``char``
``HXTYPE_UCHAR``
maps to ``unsigned char``
``HXTYPE_SHORT``
maps to ``short``
``HXTYPE_USHORT``
maps to ``unsigned short``
``HXTYPE_INT``
maps to ``int``
``HXTYPE_UINT``
maps to ``unsigned int``
``HXTYPE_LONG``
maps to ``long``
``HXTYPE_ULONG``
maps to ``unsigned long``
``HXTYPE_LLONG``
maps to ``long long``
``HXTYPE_ULLONG``
maps to ``unsigned long long``
``HXTYPE_SIZE_T``
maps to ``size_t``
``HXTYPE_FLOAT``
maps to ``float``
``HXTYPE_DOUBLE``
maps to ``double``
``HXTYPE_INT8``
maps to ``int8_t``
``HXTYPE_UINT8``
maps to ``uint8_t``
``HXTYPE_INT16``
maps to ``int16_t``
``HXTYPE_UINT16``
maps to ``uint16_t``
``HXTYPE_INT32``
maps to ``int32_t``
``HXTYPE_UINT32``
maps to ``uint32_t``
``HXTYPE_INT64``
maps to ``int64_t``
``HXTYPE_UINT64``
maps to ``uint64_t``
``HXTYPE_FLOAT`` and ``HXTYPE_DOUBLE`` make use of ``strtod`` (``strtof`` is
not used). A corresponding type for the ``long double`` format is not
specified, but may be implemented on behalf of the user via a callback.
Flags
=====
Flags can be combined into the type parameter by OR-ing them. It is valid to
not specify any flags at all, but most flags collide with one another.
``HXOPT_INC``
Perform an increment on the memory location specified by the
``*(int *)ptr`` pointer. The referenced variable must be
initialized.
``HXOPT_DEC``
Perform a decrement on the pointee. Same requirements as ``HXOPT_INC``.
Only one of ``HXOPT_INC`` and ``HXOPT_DEC`` may be specified at a time,
and they require that the base type is ``HXTYPE_NONE``, or they will
have no effect. An example may be found below.
``HXOPT_NOT``
Binary negation of the argument directly after reading it from the
command line into memory. Any of the three following operations are
executed with the already-negated value.
``HXOPT_OR``
Apply bitwise OR on the pointee with the specified/transformed value.
``HXOPT_AND``
Apply bitwise AND on the pointee with the specified/transformed value.
``HXOPT_XOR``
Apply bitwise XOR on the pointee with the specified/transformed value.
Only one of ``HXOPT_OR``, ``HXOPT_AND`` and ``HXOPT_XOR`` may be specified at
a time, but they can be used with any integral type (``HXTYPE_UINT``,
``HXTYPE_ULONG``, etc.). An example can be found below.
``HXOPT_OPTIONAL``
This flag allows for an option to take zero or one argument. Needless
to say that this can be confusing to the user. iptables's ``-L`` option
for example is one of this kind (though it does not use the libHX
option parser). When this flag is used, ``-f -b`` is interpreted as
``-f`` without an argument, as is ``-f --bar`` — things that look like
an option take precedence over an option with an optional argument.
``-f -`` of course denotes an option with an argument, as ``-`` is
often used to indicate standard input/output.
Special entries
===============
HXopt provides two special entries via macros:
``HXOPT_AUTOHELP``
Adds entries to recognize ``-?`` and ``--help`` that will display the
(long-format) help screen, and ``--usage`` that will display the short
option syntax overview. All three options will exit the program
afterwards.
``HXOPT_TABLEEND``
This sentinel marks the end of the table and is required on all tables.
(See examples for details.)
Invoking the parser
===================
.. code-block:: c
int HX_getopt5(const struct HXoption *options_table, char **argv, int *new_argc, char **new_argv, unsigned int flags);
int HX_getopt(const struct HXoption *options_table, int *argc, char ***argv, unsigned int flags);
``HX_getopt5`` is the central parsing function. ``options_table`` specifies
the options that the parser will recognize. ``argv`` must be a NULL-terminated
array of C strings.
If ``new_argv`` is non-NULL, the leftover arguments will be output as a new
string vector on success. (That array can be freed with ``HX_zvecfree``). If
``new_argc`` is non-NULL, the argument count for new_argv will be output too.
The ``flags`` argument control the general behavior of ``HX_getopt``:
``HXOPT_PTHRU``
“Passthrough mode”. Any unknown options are passed through into
``new_argv``.
``HXOPT_QUIET``
Do not print any diagnostics when encountering errors in the user's
input.
``HXOPT_HELPONERR``
Display the (long-format) help when an error, such as an unknown option
or a violation of syntax, is encountered.
``HXOPT_USAGEONERR``
Display the short-format usage syntax when an error is encountered.
``HXOPT_RQ_ORDER``
Specifying this option terminates option processing when the first
non-option argument in argv is encountered. This behavior is also
implicit when the environment variable ``POSIXLY_CORRECT`` is set.
``HXOPT_KEEP_ARGV``
Do not set ``*new_argc`` and ``*new_argv`` at all.
The return value can be one of the following:
``HXOPT_ERR_SUCCESS``
Parsing was successful.
``HXOPT_ERR_UNKN``
An unknown option was encountered.
``HXOPT_ERR_VOID``
An argument was given for an option which does not allow one. In
practice this only happens with ``--foo=bar`` when ``--foo`` is of type
``HXTYPE_NONE``, ``HXTYPE_VAL`` or ``HXTYPE_SVAL``. This does not
affect ``--foo bar``, because this can be unambiguously interpreted as
``bar`` being a remaining argument to the program.
``HXOPT_ERR_MIS``
Missing argument for an option that requires one.
``HXOPT_ERR_AMBIG``
An abbreviation of a long option was ambiguous.
negative non-zero
Failure on behalf of lower-level calls; errno.
``HX_getopt`` is an older API where ``argv`` is both used for input and output.
It recognizes additional flags/has additional behavior:
``HXOPT_KEEP_ARGV``
``argc`` and ``argv`` is not updated.
``HXOPT_DESTROY_OLD``
Call ``HX_zvecfree`` on ``argv`` before updating it.
Pitfalls
========
Staticness of tables
--------------------
The following is an example of a possible pitfall regarding ``HXTYPE_STRDQ``:
.. code-block:: c
static struct HXdeque *dq;
int main(int argc, char **argv)
{
dq = HXdeque_init();
static const struct HXoption options_table[] = {
{.sh = 'N', .type = HXTYPE_STRDQ, .ptr = dq,
.help = "Add name"},
HXOPT_TABLEEND,
};
if (HX_getopt5(options_table, *argv, &argc, &argv,
HXOPT_USAGEONERR) != HXOPT_ERR_SUCCESS)
return EXIT_FAILURE;
/* ... */
HX_zvecfree(argv);
return EXIT_SUCCESS;
}
The problem here is that ``options_table`` is, due to the static keyword,
initialized at compile-time when ``dq`` is still ``NULL``. To counter this
problem and have it doing the right thing, the ``static`` qualifier on the
options table must be removed, so that the table is built when that line
executes.
Limitations
-----------
The HX option parser has been influenced by both popt and Getopt::Long, but
eventually, there are differences:
* Long options with a single dash (``-foo bar``). This unsupported
syntax clashes easily with support for option bundling or squashing. In case
of bundling, ``-foo`` might actually be ``-f -o -o``, or ``-f oo`` in case of
squashing. It also introduces redundant ways to specify options, which is not
in the spirit of the author.
* Options using a ``+`` as a prefix, as in ``+foo``. Xterm for
example uses it as a way to negate an option. In the author's opinion, using
one character to specify options is enough — by GNU standards, a negator is
named ``--no-foo``.
* Table nesting like implemented in popt. HXopt has no provision for nested
tables, as the need has not come up yet. It does however support chained
processing. You cannot do nested tables even with callbacks, as the new argv
array is only put in place shortly before ``HX_getopt`` returns.
Examples
========
Basic example
-------------
The following code snippet should provide an equivalent of the
GNU getopt sample.[#f5]
.. [#f5] http://www.gnu.org/software/libtool/manual/libc/Example-of-Getopt.html\#Example-of-Getopt
.. code-block:: c
#include <stdio.h>
#include <stdilb.h>
#include <libHX/option.h>
int main(int argc, char **argv)
{
int aflag = 0;
int bflag = 0;
char *cflag = NULL;
struct HXoption options_table[] = {
{.sh = 'a', .type = HXTYPE_NONE, .ptr = &aflag},
{.sh = 'b', .type = HXTYPE_NONE, .ptr = &bflag},
{.sh = 'c', .type = HXTYPE_STRING, .ptr = &cflag},
HXOPT_AUTOHELP,
HXOPT_TABLEEND,
};
if (HX_getopt5(options_table, argv, &argc, &argv,
HXOPT_USAGEONERR) != HXOPT_ERR_SUCCESS)
return EXIT_FAILURE;
printf("aflag = %d, bflag = %d, cvalue = %s\n",
aflag, bflag, cvalue);
while (*++argv != NULL)
printf("Non-option argument %s\n", *argv);
HX_zvecfree(argv);
return EXIT_SUCCESS;
}
Verbosity levels
----------------
.. code-block:: c
static int verbosity = 1; /* somewhat silent by default */
static const struct HXoption options_table[] = {
{.sh = 'q', .type = HXTYPE_NONE | HXOPT_DEC, .ptr = &verbosity,
.help = "Reduce verbosity"},
{.sh = 'v', .type = HXTYPE_NONE | HXOPT_INC, .ptr = &verbosity,
.help = "Increase verbosity"},
HXOPT_TABLEEND,
};
This sample option table makes it possible to turn the verbosity of the program
up or down, depending on whether the user specified ``-q`` or ``-v``. By passing
multiple ``-v`` flags, the verbosity can be turned up even more. The range depends
on the ``int`` data type for your particular platform and compiler; if you want
to have the verbosity capped at a specific level, you will need to use an extra
callback:
.. code-block:: c
static int verbosity = 1;
static void v_check(const struct HXoptcb *cbi)
{
if (verbosity < 0)
verbosity = 0;
else if (verbosity > 4)
verbosity = 4;
}
static const struct HXoption options_table[] = {
{.sh = 'q', .type = HXTYPE_NONE | HXOPT_DEC, .ptr = &verbosity,
.cb = v_check, .help = "Lower verbosity"},
{.sh = 'v', .type = HXTYPE_NONE | HXOPT_INC, .ptr = &verbosity,
.cb = v_check, .help = "Raise verbosity"},
HXOPT_TABLEEND,
};
Mask operations
---------------
.. code-block:: c
/* run on all CPU cores by default */
static unsigned int cpu_mask = ~0U;
/* use no network connections by default */
static unsigned int net_mask = 0;
static struct HXoption options_table[] = {
{.sh = 'c', .type = HXTYPE_UINT | HXOPT_NOT | HXOPT_AND, .ptr = &cpu_mask,
.help = "Mask of cores to exclude", .htyp = "cpu_mask"},
{.sh = 'n', .type = HXTYPE_UINT | HXOPT_OR, .ptr = &net_mask,
.help = "Mask of network channels to additionally use",
.htyp = "channel_mask"},
HXOPT_TABLEEND,
};
What this options table does is ``cpu_mask &= ~x`` and ``net_mask |= y``, the
classic operations of clearing and setting bits.
Support for non-standard actions
--------------------------------
Supporting additional types or custom storage formats is easy, by simply using
``HXTYPE_STRING``, ``NULL`` as the data pointer (usually by not specifying it
at all), the pointer to your data in the user-specified pointer ``uptr``, and
the callback function in ``cb``.
.. code-block:: c
struct fixed_point {
int integral;
unsigned int fraction;
};
static struct fixed_point number;
static void fixed_point_parse(const struct HXoptcb *cbi)
{
char *end;
number.integral = strtol(cbi->data, &end, 0);
if (*end == '\0')
number.fraction = 0;
else if (*end == '.')
number.fraction = strtoul(end + 1, NULL, 0);
else
fprintf(stderr, "Illegal input.\n");
}
static const struct HXoption options_table[] = {
{.sh = 'n', .type = HXTYPE_STRING, .cb = fixed_point_parse,
.uptr = &number, .help = "Do this or that",
HXOPT_TABLEEND,
};
Chained argument processing
---------------------------
On the first run, only ``--cake`` and ``--fruit`` is considered, which is then
used to select the next set of accepted options.
.. code-block:: c
static int get_cakes(int *argc, char ***argv)
{
struct HXoption cake_table[] = {
...
};
if (HX_getopt5(cake_table, *argv, &argc, &argv,
HXOPT_USAGEONERR) != HXOPT_ERR_SUCCESS)
return EXIT_FAILURE;
/* ... */
HX_zvecfree(argv);
return EXIT_SUCCESS;
}
static int fruit_main(int argc, char **argv)
{
struct HXoption fruit_table[] = {
...
};
if (HX_getopt5(fruit_table, *argv, &argc, &argv,
HXOPT_PTHRU) != HXOPT_ERR_SUCCESS)
return EXIT_FAILURE;
/* ... */
HX_zvecfree(argv);
return EXIT_SUCCESS;
}
int main(int argc, char **argv)
{
int cake = 0, fruit = 0;
struct HXoption option_table[] = {
{.ln = "cake", .type = HXTYPE_NONE, .ptr = &cake},
{.ln = "fruit", .type = HXTYPE_NONE, .ptr = &fruit},
HXOPT_TABLEEND,
};
if (HX_getopt5(option_table, *argv, &argc, &argv,
HXOPT_PTHRU) != HXOPT_ERR_SUCCESS)
return EXIT_FAILURE;
int ret = cake ? cake_main(argc, argv) : fruit_main(argc, argv);
HX_zvecfree(argv);
return EXIT_FAILURE;
}
|