This is a very simple code that emulates metaprogramming with C macros
to parse arguments from command line and store them in an array and
report them at the end with the right format for printf
.
Of course there are several limitations because metaprogramming is not a C feature. But it is a good example of how to exploit macros in C to produce scalable template-like function.
The project is very simple built with cmake. So you can add it as a cmake module or as a git submodule. Currently Linux, Max and Windows are supported.
The compilation generates the libargparser.a which is a static object you can link to in compilation time, so you don't need to redistribute it with your executable as it doesn't becomes a dependency.
The function definitions are all in argparser.h; but the implementations are distributed in different files. The final user don't really need to deal with anything else more than:
#include "argparser.h"
The api provides different functions for 3 different functionalities; most of the functions described here with TYPE are generated throw macros, so they provide exactly the same functionalities, but for different input and/or output types.
Most of the functions that receives a name
argument uses it to
report the parameter in the report. The name is expected to be a
const char[]
with no more than 64 characters.
There are some general purpose functions exposed to the user:
-
void init_args (int argc, char **argv)
: This one is needed before any call to tho the internal api. It allocates the internal datastructures and initializes the api. -
void report_args ()
: To print the report. This can be called several times if needed. If the first (or seconds) CL argument is-json
this function prints the output in a sort ofjson
format.Similarly, if the first (or seconds) CL argument is
-millis
the timers will be printed in milliseconds instead of the default nanoseconds. -
void free_args ()
: Release all the internal's library memory and datastructures. Use this at the very end of the program. -
int get_rest_args(char ***rest)
: If there are extra arguments (more than needed to parse) this function returns the number of remaining args; otherwise the function returns zero or a negative number. If the function returns a positive value andrest
is notNULL
; the pointer is set to the first non parsed argument so it can be used in the same way as argv.
The functionalities are:
-
Parse command line arguments in order and add them to the Final report and get the value converted to
TYPE
.-
const TYPE create_cl_TYPE (const char *name)
: Read a positional command line argument and convert it to TYPE. If no enough command line arguments are provided then the function fails and aborts the program. When TYPE isstring
then the function returnschar *
. -
const TYPE create_optional_cl_TYPE (const char *name, TYPE default)
: Read a positional command line argument and convert it to TYPE, if no enough command line arguments are provided then the function created the value with thedefault
value.
-
-
Add specific values to the final report.
const TYPE create_reportable_TYPE (const char *name, TYPE value)
: This creates areportable
value. This function add a variable withname
and default valueval
. The report values are reported AFTER the command line arguments.
-
Add timers that automatically are added to the report.
-
timer create_timer(const char *name)
: Creates a timer and start it. -
void start_timer(timer *out)
: start the timer to count (needed only after a call tostop
. -
void stop_timer(timer *out)
: stop the timer temporarily. Successive calls to stop and start are useful to accumulate time values as the timer accumulator isn't reset. -
void reset_timer(timer *out)
: resets a timer. this discards all the previous values stored/accumulated. It is recommended to use this only after having saved the values to some other variable. -
getNS_timer(const timer *in)
: Returns the number of nano-seconds the timer has measured (accumulated or stored).
-
For now only 4 different types are supported: int
, double
,
size_t
and string
. The last one is just a name for functions
returning char *
needed to provide a nice name for the macro. Extra
types can be added very easily.
To see how to use all of this there is a main.c file used to generate the tests where most of the API functions are used.
The string
stores strings in an internal buffer. Currently the
buffer size is limited to 256 chars (including the '\0') and in case
of overflow the string is truncated quietly.
This is a MWE that shows how to use the library. It requires 3 mandatory command line arguments and 3 other optional ones.
So this program may be called like:
./example 1 Hello1 3.14
./example 2 Hello2 2.71 5
./example 3 Hello3 0,22 6 World
A call with less than 3 arguments or with wrong input (not convertibles) will emit an error.
#include "argparser.h"
int main(int argc, char *argv[])
{
// Initialize
init_args (argc, argv);
// Mandatory CL arguments
int m_int = create_cl_int ("m_int");
const char *m_string = create_cl_string ("m_string");
double m_double = create_cl_double ("m_double");
// Optional CL arguments
int o_int = create_optional_cl_int ("o_int", -1);
double o_double = create_optional_cl_double ("o_double", 0.5);
const char *o_string = create_optional_cl_string ("o_string", "Hello");
// Declare a timer
timer t1 = create_timer("timer_1");
sleep(1);
stop_timer(&t1);
// Report the arguments
report_args();
// Release memory (finalize the library)
free_args ();
return 0;
}
A more complete use example is in the main.c
This project also includes some C++ bindings wrapping the functions with templates to expose the basic functionalities to C++. This is a clear example on how macros and templates should NEVER mix.