NAME
CodeGen::Cpppp::Enum - Helper for enumerations and generating related utility functions
SYNOPSIS
## my $enum= CodeGen::Cpppp::Enum->new(
## prefix => 'MYENUM_', type => 'int', values => q(
## VAL_1 1
## VAL_2
## ALT_VAL
## SOME_VAL
## OTHER_VAL 1<<10
## NONE -1
## ));
## section PUBLIC;
${{ $enum->generate_declaration }}
extern const char * myenum_name(int value);
extern bool myenum_parse(const char *str, size_t len, int *value_out);
## section PRIVATE;
${{ $enum->generate_static_tables }}
const char * myenum_name(int value) {
${{ $enum->generate_lookup_by_value }}
}
bool myenum_parse(const char *str, size_t len, int *value_out) {
${{ $enum->generate_lookup_by_name }}
}
DESCRIPTION
This utility module helps you generate C code for enumerations.
First, you create an instance of this object and load it with a list of enum values. These can come from a supplied list, or you can parse a header to find them. This module supports multiple names with the same value, but not multiple values with the same name.
Next, you can generate the definition of the enum as either #define
macros or a C++ enum { ... }
syntax.
This also can generate code that looks up the name by the value, or parses a name to get the value. The header would generally look like:
#define EXAMPLE_1 1
#define EXAMPLE_2 2
// Returns true if found a match
extern bool example_parse(const char *ch, int len, int *value_out);
extern const char *example_name(int enum_val);
There are several implementations to choose from, and a sensible one will be chosen by default based on the patterns in your enum values.
CONSTRUCTOR
new
Standard constructor. Pass values for any of the non-readonly attributes below.
ATTRIBUTES
prefix
String to be prefixed onto each name of the enum. You may then optionally allow name lookups that match strings without the prefix as well as ones that include it.
type
Defaults to 'int'.
values
Returns a list of [ $name, $value ]
. You can initialize it with a list or arrayref containing $name
or [ $name, $value ]
. Any element without a value will get the next sequential value from the previous entry, starting from 0.
value_table_var
C Variable name of the constant which holds the official list of enum values.
indent
Set to a literal string to use for each level of indent in generated code.
num_format
Controls formatting of integers, as decimal or hex. Can be any single placeholder known to sprintf, like '%d', '%x', or '%X'.
max_waste_ratio
Permissible ratio of empty lookup table elements vs. populated elements. If there would be more empty cells than that, a different algorithm will be chosen. The default is 2, meaning that the table must be at least 1/3 populated.
algorithm
One of 'bsearch', 'hashtable', 'switch'.
For each, the enum is stored in an array of string/value pairs (unless the values match the array index, then they get omitted). If this happens to be the sorted order of the names or sorted order of the values, those optimizations are taken into account. Otherwise, the algorithm operates as follows:
- bsearch
-
Secondary arrays of integers are built that refer back to the enum array. These arrays are searched during name or value lookups using the C library
bsearch
function. This results in the least space usage. - hashtable
-
Hash tables will be built that refer back to the enum array. Hash function constants will be chosen to provide the fewest number of collisions possible. This results in good performance without much added code.
- switch
-
The lookup functions will be built using a combination of C 'switch' statements and binary character-comparison 'if' statements. This results in the absolute fastest performance, but may generate a fair amount of parsing code.
is_symbolic
True if one if your values is an expression instead of a constant integer. In this case, only the switch algorithm can be used for lookups.
is_sequential
Read-only, calculated by whether all the values have adjacent numeric values.
is_sequentialish
Read-only, calculated by whether all the values can be stored in an array at the offset of their numeric value without wasting more than max_waste_factor
elements.
generate_declaration
@code_lines= $enum->generate_declaration(%options);
Return either a list of #define NAME ...
or the lines of a C++ enum Foo { ... }
.
generate_static_tables
@code_lines= $enum->generate_static_tables(%options);
Return lines of code that declare the tables needed for the lookups. This always includes a table in the same order as the declaration of the enum values, but may also include additional hash tables or lookup tables to represent the values in alphabetical order.
generate_lookup_by_value
@code_lines= $enum->generate_lookup_by_value(%options);
Return lines of code that return a const char *
of the name associated with a value. The variable holding the value is named 'value' by default, and the implementation returns NULL if this value does not have a name.
generate_lookup_by_name
@code_lines= $enum->generate_lookup_by_name(%options);
Return lines of code that take a name in a variable str
, len
bytes long, and store the enum value for that name into a variable value_out
and then return true (if found) or false if not found.
AUTHOR
Michael Conrad <mike@nrdvana.net>
VERSION
version 0.004
COPYRIGHT AND LICENSE
This software is copyright (c) 2024 by Michael Conrad.
This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself.