cdecl(1) | General Commands Manual | cdecl(1) |
NAME¶
cdecl, c++decl - compose and decipher C & C++ declarations and casts
SYNOPSIS¶
cdecl [options]
[command ...]
c++decl [options] [command
...]
explain [options] gibberish
DESCRIPTION¶
``I'm still uncertain about the language declaration syntax, where in declarations, syntax is used that mimics the use of the variables being declared. It is one of the things that draws strong criticism, but it has a certain logic to it.''
- — Dennis M. Ritchie, Creator of C
``I consider the C declarator syntax an experiment that failed.''
- — Bjarne Stroustrup, Creator of C++
cdecl is a program for composing and deciphering C (or C++) declarations or casts, aka ``gibberish.'' Additionally, cdecl is also for developing and debugging C preprocessor macros by performing expansion step-by-step.
cdecl can be used interactively on a terminal or accept input from either the command-line or standard input.
INVOCATION¶
Executable Name¶
cdecl may be invoked under a number of different names (by either renaming the executable or creating either a symbolic or hard link to it). If it is invoked as:
- cdecl
- Runs with the latest supported version of C as the default language.
- c++decl | cppdecl | cxxdecl
- Runs with the latest supported version of C++ as the default language.
- explain
- Runs with the latest supported version of C as the default language interpreting the rest of the command-line (if given) as gibberish, performs the conversion to pseudo-English, and exits.
For example (where $ is the shell prompt):
$ explain 'int *const (*p)[4]' declare p as pointer to array 4 of constant pointer to integer
For declarations given on the command line, care must be taken either to escape or quote shell metacharacters. The default language can be specified via the cdeclrc file (see CONFIGURATION FILES).
First Argument¶
cdecl will also behave as above if the first non-option argument on the command-line or the first whitespace separated word thereof is a cdecl command (see Commands):
$ cdecl declare p as pointer to array 4 of int int (*p)[4]; $ c++decl 'reinterpret cast n into pointer to unsigned' reinterpret_cast<unsigned*>(n)
Standard Input¶
If no arguments are given, standard input will be read until end-of-file is encountered or one of the exit, q, or quit commands is read.
You can use cdecl as you create a C (or C++) program from within an editor. For example, in vi(1), type the declaration in pseudo-English, then filter the line through cdecl:
!!cdecl<CR>
where <CR> is the return key.
Interactive Input¶
If standard input is coming from a terminal, a prompt of either cdecl> or c++decl> will be written to the terminal before each line of input. The prompt can be turned off by either the --no-prompt or -P option or the set noprompt command.
cdecl uses the GNU readline(3) library (if available and compiled in) to provide keyword completion, command-line history, and editing.
OPTIONS¶
An option argument f means file, n means unsigned integer, and s means string.
- --alt-tokens | -a
- Turns on alternative token output. The alternative tokens are: and (&&), and_eq (&=), bitand (&), bitor (|), compl (~), not (!), not_eq (!=), or (||), or_eq (|=), xor (^), and xor_eq (^=). (Supported only in C95 and later.)
- --bison-debug | -B
- Turns on bison(1) debugging output (if compiled in). The additional output is of grammar productions as they are being reduced by the parser. (This is a debugging aid for developers of cdecl itself.)
- --color=s | -k s
- Sets when to colorize output to s (which is case-insensitive). (See the CDECL_COLORS environment variable about colorization.) The choices for s are:
- always
- Always colorize.
- auto | isatty | tty
- Colorize only when standard output is connected to a TTY (terminal).
- never
- Never colorize.
- not_file | not_isreg
- Colorize only when not writing to a regular file (hence writing to a TTY or pipe). (Specifically, fstat(3) is called on standard output: colorize only if S_IFREG is false.)
- This is more useful than isatty since it accommodates the common case of allowing color output to be piped to less(1) and still display in color since less understands SGR (``Select Graphic Rendition'') ASCII terminal escape sequences that produce color.
- The default is not_file.
- --commands | -K
- Prints a sorted list of all cdecl commands for the current language that may be used as the first argument on a command-line. (See First Argument.) This is intended for consumption by a shell completion function. (See also --options or -O.)
- --config=f | -c f
- Specifies the configuration file f to read (see CONFIGURATION FILES).
- --debug[=s] | -d[s]
- Turns on cdecl debugging output:
- While parsing a command, prints additional JSON5 output of an abstract syntax tree (AST) as it is being constructed from user input while the parser is reducing grammar productions.
- While expanding a macro, prints additional JSON5 output of the tokens.
- If an error or warning message is printed, includes the [file:line] of the code that generated said message.
- For syntax errors, includes the numeric [id] of the token that caused the error.
- (This is a debugging aid for developers of cdecl itself.)
- Valid formats for s are:
- u
- Include AST node unique_id values in JSON5 output as an additional debugging aid.
- Alternatively, * may be given to mean ``all'' or - may be given to mean ``none.''
- --digraphs | -2
- Turns on digraph token output. The digraph tokens are: <: ([) and :> (]). (Supported only in C95 and later.)
- --east-const | -e
- Turns on ``east const'' output where const (and volatile) are printed to the right (``east'') of the type, e.g.:
const int i; // west int const j; // east
- (Not supported in K&R C.)
- --echo-commands | -E
- Echoes commands before their corresponding output, but only when not interactive. (This is primarily useful for associating command output with input in test scripts.)
- --explicit-ecsu=s | -S s
- For C++ only, sets the type keywords to s (which is case-insensitive) that should be explicitly included in declarations. Valid formats for s are:
- Multiple format may be given, one immediately after the other. Alternatively, * may be given to mean ``all'' or - may be given to mean ``none.'' The default is su. For example, using the default:
c++decl> declare ps as pointer to struct S struct S *ps; c++decl> declare pt as pointer to class T T *pt;
- (See Other Caveats.) For C, enum, struct, and union are always included in declarations.
- --explicit-int=s | -i s
- Sets the integer types to s (which is case-insensitive) that should have int explicitly included in declarations. (Normally, int is omitted for short, long, long long, or unsigned declarations where int is implied.) Valid formats for s are:
- Multiple formats may be given, one immediately after the other, e.g., usl means unsigned short and (signed) long. Parsing is greedy so commas may be used to separate formats. For example, ulll is parsed as unsigned long long and long whereas ul,ll is parsed as unsigned long and long long. Note that since u is both a shorthand and a modifier, iu means all signed and unsigned integer types whereas ui means just unsigned int. Alternatively, * may be given to mean ``all'' or - may be given to mean ``none.''
- --file=f | -f f
- Reads commands from file f.
- --flex-debug | -F
- Turns on flex(1) debugging output (if compiled in). The additional output is of lexer rules as they are being tokenized by the lexer. (This is a debugging aid for developers of cdecl itself.)
- --help | -h
- Prints a help message for command-line options and exits.
- --infer-command | -I
- Tries to infer a command when an input line doesn't start with any. If an input line starts with a:
- Macro name, infers expand;
- C (or C++) declaration, infers explain.
- --language=s | -x s
- Specifies which version of what language s (which is case-insensitive) to use. (See C AND C++ LANGUAGE VERSIONS for valid languages.) The default is C23 (for cdecl) and C++23 (for c++decl).
- --lineno=n | -L n
- Specifies an integer n to add to all line numbers in error and warning messages. (This is a debugging aid for developers of cdecl itself.)
- This is useful when cdecl is called from a shell script with a here document for input. Consider a test-script (showing line numbers) like:
1 #! /usr/bin/env bash 2 cdecl --echo-commands --lineno=$LINENO <<END 3 explain int 4 END
- where $LINENO (the shell environment variable containing the current the line number of the script, in this case, the one cdecl is called on — in this example: 2) is given. When executed, the script will print:
$ test-script cdecl> explain int
^ 3,12: error: declaration expected
- where the line number printed is 3 (the starting line number 1 plus 2 from $LINENO) which is the absolute line number within the script as opposed to the here document that makes locating offending lines easier. (See Other Caveats.)
- --no-buffer-stdout | -b
- Sets standard output to unbuffered. (This is a debugging aid for developers of cdecl itself.)
- --no-config | -C
- Suppresses reading of any configuration file, even one explicitly specified via either --config or -c.
- --no-english-types | -T
- Prints types in C/C++ (e.g., int), not pseudo-English (e.g., integer) when explaining gibberish.
- --no-prompt | -P
- Suppresses printing of the prompt.
- --no-semicolon | -s
- Suppresses printing of a final semicolon for C (and C++) declarations.
- --no-typedefs | -t
- Suppresses predefining standard types, e.g., size_t, uint8_t, etc. (See PREDEFINED TYPES.)
- --no-using | -u
- Always declares types with typedef rather than using in C++11 and later.
- --options | -O
- Prints a sorted list of all cdecl options that may be used on a command-line in an easily parsable form. This is intended for consumption by a shell completion function. (See also --commands or -K.)
- --output=f | -o f
- Sends all non-error output to file f.
- --permissive-types | -p
- Permits keywords in language versions other than the current language as types in pseudo-English. By default, a declaration in C like:
declare p as pointer to class
- would result in an ``unsupported type in C'' error even though class could be a valid user-defined type in C. This option permits such declarations. (See Permissive Types.)
- --trigraphs | -3
- Turns on trigraph token output. The trigraph tokens are: ??( ([), ??) (]), ??' (^), ??'= (^=), ??! (|), ??!= (|=), ??!??! (||), and ??- (~). (Supported only between C89 and C17 and between C++03 and C++14.)
- --trailing-return | -r
- Declares functions and operators using the trailing return type syntax in C++11 and later.
- --version | -v
- Prints the version number and exits. If given twice, additionally prints the set of configure feature & package options and whether GNU readline(3) (if compiled in) is genuine.
- --west-decl=s | -w s
- Sets when to print the * for pointer and & and && for references adjacent to the type (``west'') versus adjacent to the name (``east'') to s (which is case-insensitive), e.g.:
int *p; // east int* q; // west
- Valid formats for s are:
- Multiple formats may be given, one immediately after the other. Alternatively, * may be given to mean ``all'' or - may be given to mean ``none.'' The default is r.
- However, when more than one name is given in the same declare command for anything other than a structured binding, the *, &, and && are always printed adjacent to the name (``east''):
cdecl> declare f, g as function returning pointer to char char *f(), *g();
C AND C++ LANGUAGE VERSIONS¶
The argument to the --language or -x option or the set command is one of the following versions (which are case-insensitive):
- C
- Use the latest supported version of the C language.
- CK&R | CKNR | CKR | K&R | K&RC | KNR | KNRC | KR | KRC | C78
- Use the pre-ANSI Kernighan & Ritchie version of the C language as given in the first edition of The C Programming Language.
- C89 | C90
- Use the C89 (first ANSI C) version of the C language. Adds support for const, enum, long double, signed, unsigned char, unsigned long, unsigned short, void, volatile, and function prototypes. Adds support for C preprocessor token concatenation via ## and stringification via #.
- C95
- Use the C95 version of the C language. Adds support for wchar_t.
- C99
- Use the C99 version of the C language. Adds support for _Bool, _Complex, _Imaginary, inline functions, long long, restrict, and static, type-qualified, and variable length array function parameters. Adds support for C preprocessor variadic macros.
- C11
- Use the C11 version of the C language. Adds support for _Alignas, _Atomic, char16_t, char32_t, _Noreturn, and _Thread_local.
- C17 | C18
- Use the C17 version of the C language. (Minor revision: no new features. Equivalent to C11.)
- C23
- Use the C23 version of the C language. Adds support for alignas, auto (as a deduced type), _BitInt, bool, char8_t, constexpr, [[deprecated]], false, fixed-type enumerations, [[maybe_unused]], [[nodiscard]], nullptr, [[reproducible]], thread_local, true, typeof, [[unsequenced]], and the __VA_OPT__ preprocessor macro.
- C++
- Use the latest supported version of the C++ language.
- C++98
- Use the C++98 version of the C++ language. Adds support for class, constructors, destructors, exception specifications (throw), mutable data members, namespace, new-style casts, overloaded operators, references, pointers to class members, user-defined conversion, and virtual functions.
- C++03
- Use the C++03 version of the C++ language. (Minor revision; no new features. Equivalent to C++98.)
- C++11
- Use the C++11 version of the C++ language. Adds support for alignas, auto (as a deduced type), [[carries_dependency]], char16_t, char32_t, constexpr, default and delete for member functions, enum class, final, fixed-type enumerations, function trailing return-types, inline namespaces, lambdas, long long, member function ref-qualfiers, noexcept, [[noreturn]], override, rvalue references, thread_local, using (as a typedef synonym), and user-defined literals.
- C++14
- Use the C++14 version of the C++ language. Adds support for auto and constexpr return types and [[deprecated]].
- C++17
- Use the C++17 version of the C++ language. Adds support for inline variables, [[maybe_unused]], nested namespace declarations, [[nodiscard]], and structured bindings.
- C++20
- Use the C++20 version of the C++ language. Adds support for auto parameters and parameter packs, char8_t, consteval, constinit, default relational operators, export, nested inline namespaces, [[no_unique_address]], operator<=>, and the __VA_OPT__ preprocessor macro.
- C++23
- Use the C++23 version of the C++ language. Adds support for _Atomic, explicit object parameters, static operator(), and zero or more parameters for operator[].
- C++26
- Use the C++26 version of the C++ language. Adds support for = delete("reason").
CDECL LANGUAGE¶
Commands¶
In what follows, [] means zero or one, * means zero or more, {} means one of, and | means alternate.
cdecl has the following commands:
- cast [s-name] {as|[in]to} english
- Composes a C (or C++) cast from pseudo-English.
- {const|dynamic|reinterpret|static} cast s-name {as|[in]to} english
- Composes a C++ new-style cast from pseudo-English.
- declare s-name [, s-name]* as english [width number [bits]]
- Composes C (or C++) declarations from pseudo-English with an optional bit-field width (for integral types only).
- declare s-name [, s-name]* as english aligned [as|to] { number [bytes] | english }
- Composes C (or C++) declarations from pseudo-English aligned to either a specific number of bytes or the same alignment as english.
- declare operator as english
- For C++ only, composes an overloaded operator declaration from pseudo-English.
- declare store* lambda [[captures] [captures]] [([args])] [returning english]
- For C++ only, composes a lambda declaration from pseudo-English.
- declare store* user-defined conversion [operator] [of scope-e s-name]* returning english
- For C++ only, composes a user-defined conversion operator from pseudo-English.
- define s-name as english
- Defines a type (typedef) from pseudo-English.
- #define name [pp-tokens]
- Defines an object-like macro that expands into pp-tokens.
- #define name([pp-params]) [pp-tokens]
- Defines a function-like macro that expands into pp-tokens. Note that the ( must be adjacent to name.
- enum [class] s-name [: type]
- Defines s-name as an enum (or an enum class in C++) type optionally of a fixed underlying type. In C, this is equivalent to:
typedef enum s-name s-name
- expand name [([pp-args])] [pp-tokens]
- Expands a previously defined macro step by step using the supplied arguments (if given). The additional pp-tokens (if given) are appended and expanded.
- explain gibberish [, gibberish]*
- Deciphers C (or C++) declarations or a new-style cast (C++ only) into pseudo-English.
- explain (gibberish)[s-name]
- Deciphers a C (or C++) cast into pseudo-English.
- include "path"
- Includes the file denoted by path and performs the contained commands. (This is typically used in a configuration file to include another configuration file.) Shell meta characters in path, e.g., ~, are expanded.
- Note that include is recognized as a cdecl command only when immediately followed by a string literal.
- #include "path"
- Same as include, except #include may not end with ; and must be on a line by itself.
- { struct | union | class } s-name
- Defines s-name as a struct, union, or class (C++ only) type. In C, this is equivalent to:
typedef { struct | union } s-name s-name
- scope-c s-name { [{ scope-c | typedef | using } ;]* }
- For C++ only, defines s-name as a scope-c (struct, union, or class) type; or begins a namespace. Also executes zero or more class, struct, typedef, union, or using commands within the scope of s-name thus declaring type(s) within that scope.
- set [option [= value] | options | lang]*
- In the first form, sets a particular option (see Set Options for valid options); in the second form, prints the current value of all options; in the third form, sets the current language to lang. If no argument is given, it's equivalent to the second form.
- show [ s-name | [all] [predefined | user] [glob]] [[as] {english | typedef | using}]
- For s-name, shows the definition for a previously defined type (via define, typedef, or using) having that name.
- For all only, shows all predefined and user-defined types.
- For predefined, shows only predefined types that are valid in the current language or later.
- For user, shows only user-defined types that were defined in the current language or later.
- For either all predefined or all user, shows their respective set of types regardless of the current language.
- For none of s-name, all, predefined, user, or glob, equivalent to show user.
- By default, types are shown as they were defined. If typedef is given, types are shown as typedef declarations. For C++11 or later only, if using is given, types are shown as using declarations.
- For name, shows the definition for a previously defined macro (via #define) having that name.
- For predefined, shows only predefined macros that are valid in the current language or later.
- For user, shows only user-defined macros.
- For neither predefined nor user, equivalent to show user.
- type[def] gibberish [, gibberish]*
- Defines types via a C (or C++) typedef declaration.
- #undef name
- Undefines a previously defined macro.
- using name = gibberish
- For C++11 or later only, defines a type via a using declaration.
- { help | ? } [ command[s] | command | english | options ]
- Prints help that's sensitive to the current programming language (C or C++). By default or when command or commands is given, prints help on cdecl's commands (this section); if command is given, prints help only for that command; if english is given, prints help on pseudo-English (see English below); if options is given, prints help on cdecl's options (see Set Options below).
- exit | q[uit]
- Quits cdecl. Note that q is recognized as a synonym for quit only when it's the only thing on a line other than whitespace.
where:
- args
- A comma-separated list of s-name, english, s-name as english; or one of varargs, variadic, or ... (ellipsis).
- captures
- Optionally either copy by default (or =) or reference by default (or &) followed by a comma-separated list of name or reference to name.
- name
- A valid C (or C++) identifier.
- operator
- A valid C++ operator.
- s-name
- For C, is the same as name; for C++, is either the same as name or is a scoped name that may always be specified using ::, e.g., S::T::x, or in an english context, may alternatively be specified as s-name [of scope-e s-name]*.
- glob
- For C, a name; or for C++, a scoped name that may contain ::. Either may contain * as a wildcard, e.g., T* (any type name starting with T), *::T* (any type name starting with T in any top-level scope). As a special case for C++, glob may start with **:: to match a type name in any scope.
- pp-args
- A comma-separated list of valid C (or C++) tokens for macro arguments.
- pp-params
- A comma-separated list of valid C (or C++) identifiers for macro parameters. In C99 and later, the last parameter may be ....
- pp-tokens
- A set of valid C (or C++) tokens.
- scope-c
- A C or C++ scope-creating keyword, one of: class, namespace, struct, or union.
- scope-e
- An extended scope-creating keyword, either one of scope-c or the cdecl-specific scope when the particular scope-c is either unknown when T wasn't previously declared or when it doesn't matter. For example:
c++decl> explain int T::x declare x of scope T as integer
- cdecl knows T is one of scope-c, but it has no way to know which one so it uses the generic scope. Similarly, you may use scope when you don't care which:
c++decl> declare x of scope T as int int T::x;
Commands are terminated by either a semicolon or newline. However, commands may be given that span multiple lines when newlines are escaped via \. When a newline is escaped, the next prompt (if enabled) changes to either cdecl+ or c++decl+ to indicate that the current line will be a continuation of the previons line.
English¶
In what follows, [] means zero or one, * means zero or more, {} means one of, and | means alternate. The only punctuation characters used in pseudo-English are hyphens in hyphenated words and parentheses around and commas between constructor, function, operator, user-defined literal, or block parameters, and brackets around lambda captures.
English is one of:
store* ar-qual* variable [length] array of english
block [([args])] [returning english]
cv-qual* concept s-name [parameter pack]
store* constructor [([args])]
[virtual] destructor [()]
store* fn-qual* [[non-]member] function [([args])] [returning english]
store* fn-qual* [[non-]member] operator [([args])] [returning english]
cv-qual* pointer to [member of class s-name] english
[rvalue] reference to english
structured binding
store* user-defined literal [([args])] [returning english]
store* modifier* [C-type]
{ enum [class|struct] [of [type] english] | class | namespace | struct | union } s-name
where:
- ar-qual
- One of: non-empty, const, restrict, or volatile.
- C-type
- One of: auto (C23 or C++11 or later), bool, char, char8_t, char16_t, char32_t, wchar_t, int, float, double, parameter pack, or void; or for C99 only: _Accum or _Fract.
- cv-qual
- One of: _Atomic, const, restrict, or volatile.
- fn-qual
- One of: const, final, override, reference, restrict, rvalue reference, or volatile.
- modifier
- One of: short, long, signed, unsigned, _Complex, _Imaginary, or _Sat (C99 only).
- number
- One of a decimal, octal (if starting with 0), hexadecimal (if starting with either 0x or 0X), or binary (if starting with either 0b or 0B) number.
- store
- One of: auto (C17 or earlier, or C++03 or earlier), block, carries-dependency, consteval, constexpr, deprecated, explicit, export, extern, extern "C", final, friend, inline, maybe-unused, mutable, nodiscard, non-throwing, noreturn, override, register, reproducible, static, thread-local, typedef, unsequenced, virtual, or pure virtual.
If returning english is omitted, it's equivalent to returning int in C95 and earlier or returning void in C99 and later or C++.
Synonyms¶
Some synonyms are permitted within pseudo-English. The terms on the left are synonyms for what's on the right:
accum _Accum
atomic _Atomic
automatic auto
bit precise integer _BitInt
bit-precise integer _BitInt
bool _Bool
boolean _Bool
Boolean _Bool
capture capturing
captures capturing
carries dependency carries_dependency
carries-dependency carries_dependency
char 8 char8_t
char 16 char16_t
char 32 char32_t
character char
command commands
complex _Complex
constant const
constant evaluation consteval
constant-evaluation consteval
const-eval consteval
constant expression constexpr
constant-expression constexpr
const-expr constexpr constant initialization constinit constant-initialization constinit
const-init constinit
conv conversion
ctor constructor
double precision double
double-precision double
dtor destructor
enumeration enum
eval evaluation
exported export
expr expression
external extern
floating point float
floating-point float
fract _Fract
func function
imaginary _Imaginary
init initialization
integer int
len length
maybe unused maybe_unused
maybe-unused maybe_unused
mbr member
no discard nodiscard
no-discard nodiscard
non-discardable nodiscard
no except noexcept
no-except noexcept
no-exception noexcept
no unique address no_unique_address
no-unique-address no_unique_address
non-unique-address no_unique_address
non-mbr non-member
noreturn _Noreturn
no return _Noreturn
no-return _Noreturn
non-returning _Noreturn
non-throwing throw()
oper operator
overridden override
predef predefined
ptr pointer
ref reference
restricted restrict
ret returning
sat _Sat
saturated _Sat
structure struct
type typedef
thread local thread_local
thread-local thread_local
thread_local _Thread_local
user defined user-defined
user-def user-defined
var variable
varargs ...
variadic ...
vector array
wide character wchar_t
Gibberish¶
Gibberish is any supported C (for cdecl) or C++ (for c++decl) declaration of a variable, constant, array, macro, pointer, reference, rvalue reference, function, constructor, destructor, or overloaded operator; or user-defined type, conversion, or literal; or type cast. (See EXAMPLES for examples and CAVEATS for unsupported declarations.)
Gibberish also includes support for the following:
- Apple's ``blocks'' syntax and the __block storage class.
- For C99 only, Embedded C's _Accum and _Fract types, and the _Sat modifier.
- For C99 only, Unified Parallel C's relaxed, shared, and strict qualifiers.
- GNU C's __auto_type, __complex, __complex__, __const, __inline, __inline__, __restrict, __restrict__, __signed, __signed__, __thread, __typeof__, __volatile, and __volatile__.
- GNU C's __attribute__ syntax, but all attributes are ignored.
- Microsoft C's __declspec syntax, but all attributes are ignored.
- Microsoft C's calling conventions __cdecl, __clrcall, __fastcall, __stdcall (also WINAPI), __thiscall, and __vectorcall. Pseudo-English also allows the same convention names but without the leading __.
- Microsoft C's __forceinline, but it's treated as a synonym for inline.
- Microsoft C's _asm, _cdecl, _declspec, _fastcall, _forceinline, _inline, _restrict, _stdcall, and _vectorcall that are synonyms for their respective __ counterparts.
Set Options¶
The set command takes several options (which ignore hyphens). Unambiguous option abbreviations may be used.
- [no]alt-tokens
- Turns [off] on alternative token output — default is off. (Supported only in C95 and later.)
- [no]bison-debug
- Turns [off] on bison(1) debugging output (if compiled in) — default is off.
- debug[=s]
- Turns on cdecl debugging output — default is off. (See the --debug or -d option for details.)
- nodebug
- Turns off cdecl debugging output.
- nographs
- Turns off either digraph or trigraph output, i.e., reverts to emitting all characters as-is.
- digraphs
- Turns on digraph output for [ and ] — default is off. (Supported only in C95 and later.)
- [no]east-const
- Turns [off] on ``east const'' output where const (and volatile) are printed to the right (``east'') of the type — default is off. (Not supported in K&R C.)
- [no]echo-commands
- Turns [off] on echoing commands before corresponding output, but only when not interactive — default is off.
- [no]english-types
- Turns [off] on printing types in pseudo-English (e.g., integer), not C/C++ (e.g., int) when explaining gibberish — default is on.
- explicit-ecsu=s
- For C++ only, turns on explicit enum, class, struct, or union in declarations — default is su (for struct and union). (See the --explicit-ecsu or -S option for details.)
- explicit-int=s
- Turns on explicit int for the integer types s — default is none. (See the --explicit-int or -i option for details.)
- noexplicit-ecsu
- For C++ only, turns off explicit enum, class, struct, and union in declarations.
- noexplicit-int
- Turns off explicit int for all integer types.
- [no]flex-debug
- Turns [off] on flex(1) debugging output (if compiled in) — default is off.
- [no]infer-command
- Turns [off] on trying to infer a command when an input line doesn't start with any — default is off. (See the --infer-command or -I option for details.)
- language=s
- Use s (which is case-insensitive) as the current language. (See C AND C++ LANGUAGE VERSIONS for valid languages.)
- lang
- Shorthand for language=lang.
- options
- Prints the current value of all options.
- [no]permissive-types
- Turns [off] on permitting keywords in language versions other than the current language as types — default is off. (See the --permissive-types or -p option for details.)
- [no]prompt
- Turns [off] on the prompt — default is on.
- [no]semicolon
- Turns [off] on printing a semicolon at the end of a C (or C++) declaration — default is on.
- trigraphs
- Turns on trigraph output for [, ], ^, |, and ~ — default is off. (Supported only between C89 and C17 and between C++03 and C++14.)
- [no]trailing-return
- Turns [off] on declaring functions and operators using the trailing return type syntax in C++11 and later — default is off.
- [no]using
- Turns [off] on declaring types with using rather than typedef in C++11 and later — default is on.
- west-decl=s
- Turns on printing the * for pointer and & and && for reference types adjacent to the type (``west'') to s — default is r (only all return types). (See the --west-decl or -w option for details.)
- nowest-decl
- Turns off printing the * for pointer and & and && for reference types adjacent to the type (``west'') for all types.
PREDEFINED TYPES¶
The following types are predefined (unless either the --no-typedefs or -t option is given) in the specified language and later. However, the types std::partial_ordering, std::strong_ordering, and std::weak_ordering, are always defined in C++20 and later since they are required by operator<=>().
K&R C¶
caddr_t, daddr_t, dev_t, FILE, ino_t, jmp_buf, off_t, time_t, tm
C89¶
blkcnt_t, blksize_t, cc_t, clockid_t, clock_t, DIR, div_t, double_t, errno_t, fd_set, femode_t, fenv_t, fexcept_t, float_t, fpos_t, fsblkcnt_t, fsfilcnt_t, gid_t, iconv_t, id_t, imaxdiv_t, in_addr_t, in_port_t, key_t, lconv, ldiv_t, lldiv_t, locale_t, long_double_t, mbstate_t, mode_t, nfds_t, nlink_t, pid_t, posix_spawnattr_t, posix_spawn_file_actions_t, ptrdiff_t, regex_t, regmatch_t, regoff_t, rlim_t, sa_family_t, sig_atomic_t, sighandler_t, sigset_t, sig_t, size_t, socklen_t, ssize_t, suseconds_t, timer_r, uid_t, useconds_t, va_list,
_Decimal32, _Decimal32_t, _Decimal64, _Decimal64_t, _Decimal64x, _Decimal128, _Decimal128_t, _Decimal128x,
_Float16, _Float16_t, _Float32, _Float32_t, _Float32x, _Float64, _Float64_t, _Float64x, _Float128, _Float128_t, _Float128x
C95¶
pthread_t, pthread_barrier_t, pthread_barrierattr_t, pthread_cond_t, pthread_condattr_t, pthread_key_t, pthread_mutex_t, pthread_mutexattr_t, pthread_once_t, pthread_rwlock_t, pthread_rwlockattr_t, pthread_spinlock_t, wctrans_t, wctype_t, wint_t,
__float80, __float128, __fp16, __ibm128, __int128, __m128, __m128d, __m128i, __m64,
ATOM, BOOL, BOOLEAN, BYTE, CCHAR, CHAR, COLORREF, DWORD, DWORD32, DWORD64, DWORDLONG, DWORD_PTR, FLOAT, HALF_PTR, HANDLE, HBITMAP, HBRUSH, HCOLORSPACE, HCONV, HCONVLIST, HCURSOR, HDC, HDDEDATA, HDESK, HDROP, HDWP, HENHMETAFILE, HFILE, HFONT, HGDIOBJ, HGLOBAL, HHOOK, HICON, HINSTANCE, HKEY, HKL, HLOCAL, HMENU, HMETAFILE, HMODULE, HMONITOR, HPALETTE, HPEN, HRESULT, HRGN, HRSRC, HSZ, HWINSTA, HWND, INT, _int8, _int16, _int32, _int64, __int8, __int16, __int32, __int64, INT_PTR, LANGID, LARGE_INTEGER, LCID, LCTYPE, LGRPID, LONG, LONG32, LONG64, LONGLONG, LONG_PTR, LPBOOL, LPBYTE, LPCHAR, LPCOLORREF, LPCSTR, LPCTSTR, LPCVOID, LPCWSTR, LPDWORD, LPHANDLE, LPINT, LPLONG, LPSTR, LPTSTR, LPVOID, LPWORD, LPWSTR, LRESULT, PBOOL, PBOOLEAN, PBYTE, PCHAR, PCSTR, PCTSTR, PCWSTR, PDWORD, PDWORD32, PDWORD64, PDWORDLONG, PDWORD_PTR, PFLOAT, PHALF_PTR, PHANDLE, PHKEY, PINT, PINT16, PINT32, PINT64, PINT8, PINT_PTR, PLCID, PLONG, PLONG32, PLONG64, PLONGLONG, PLONG_PTR, PSHORT, PSIZE_T, PSSIZE_T, PSTR, PTBYTE, PTCHAR, PTSTR, PUCHAR, PUHALF_PTR, PUINT, PUINT16, PUINT32, PUINT64, PUINT8, PUINT_PTR, PULONG, PULONG32, PULONG64, PULONGLONG, PULONG_PTR, PUSHORT, PVOID, PWCHAR, PWORD, PWSTR, QWORD, SC_HANDLE, SC_LOCK, SERVICE_STATUS_HANDLE, SHORT, SIZE_T, SSIZE_T, TBYTE, TCHAR, UCHAR, UHALF_PTR, UINT, UINT16, UINT32, UINT64, UINT8, UINT_PTR, ULARGE_INTEGER, ULONG, ULONG32, ULONG64, ULONGLONG, ULONG_PTR, UNICODE_STRING, USHORT, USN, WCHAR, __wchar_t, WORD, WPARAM
C99¶
int8_t, int16_t, int32_t, int64_t, intmax_t, intptr_t, uint8_t, uint16_t, uint32_t, uint64_t, uintmax_t, uintptr_t,
int_fast8_t, int_fast16_t, int_fast32_t, int_fast64_t, int_least8_t, int_least16_t, int_least32_t, int_least64_t, uint_fast8_t, uint_fast16_t, uint_fast32_t, uint_fast64_t, uint_least8_t, uint_least16_t, uint_least32_t, uint_least64_t,
int_hk_t, int_hr_t, int_k_t, int_lk_t, int_lr_t, int_r_t, uint_uhk_t, uint_uhr_t, uint_uk_t, uint_ulk_t, uint_ulr_t, uint_ur_t
C11¶
atomic_bool, atomic_char16_t, atomic_char32_t, atomic_char, atomic_flag, atomic_int, atomic_intmax_t, atomic_intptr_t, atomic_llong, atomic_long, atomic_ptrdiff_t, atomic_schar, atomic_short, atomic_size_t, atomic_uchar, atomic_uint, atomic_uintmax_t, atomic_uintptr_t, atomic_ullong, atomic_ulong, atomic_ushort, atomic_wchar_t,
atomic_int_fast8_t, atomic_int_fast16_t, atomic_int_fast32_t, atomic_int_fast64_t, atomic_int_least8_t, atomic_int_least16_t, atomic_int_least32_t, atomic_int_least64_t, atomic_uint_fast8_t, atomic_uint_fast16_t, atomic_uint_fast32_t, atomic_uint_fast64_t, atomic_uint_least8_t, atomic_uint_least16_t, atomic_uint_least32_t, atomic_uint_least64_t,
cnd_t, constraint_handler_t, max_align_t, memory_order, mtx_t, once_flag, rsize_t, thrd_start_t, thrd_t, tss_dtor_t, tss_t
C++98¶
std::bad_alloc, std::bad_cast, std::bad_exception, std::bad_type_id, std::codecvt_base, std::ctype_base, std::ctype_base::mask, std::div_t, std::domain_error, std::exception, std::filebuf, std::fstream, std::ifstream, std::invalid_argument, std::ios, std::ios_base::Init, std::ios_base::event, std::ios_base::event_callback, std::ios_base::fmtflags, std::ios_base::iostate, std::ios_base::openmode, std::ios_base::seekdir, std::ios_base, std::iostream, std::istream, std::istringstream, std::lconv, std::ldiv_t, std::length_error, std::locale, std::logic_error, std::messages_base, std::money_base, std::new_handler, std::nothrow_t, std::ofstream, std::ostream, std::ostringstream, std::osyncstream, std::out_of_range, std::overflow_error, std::ptrdiff_t, std::range_error, std::runtime_error, std::sig_atomic_t, std::size_t, std::streambuf, std::streamoff, std::streamsize, std::string, std::stringbuf, std::stringstream, std::syncbuf, std::time_base, std::tm, std::underflow_error, std::wfilebuf, std::wfstream, std::wifstream, std::wios, std::wiostream, std::wistream, std::wistringstream, std::wofstream, std::wostream, std::wostringstream, std::wosyncstream, std::wstreambuf, std::wstring, std::wstringbuf, std::wstringstream, std::wsyncbuf
C++11¶
std::atomic_bool, std::atomic_char16_t, std::atomic_char32_t, std::atomic_char8_t, std::atomic_char, std::atomic_flag, std::atomic_int16_t, std::atomic_int32_t, std::atomic_int64_t, std::atomic_int8_t, std::atomic_int_fast16_t, std::atomic_int_fast32_t, std::atomic_int_fast64_t, std::atomic_int_fast8_t, std::atomic_int, std::atomic_int_least16_t, std::atomic_int_least32_t, std::atomic_int_least64_t, std::atomic_int_least8_t, std::atomic_intmax_t, std::atomic_intptr_t, std::atomic_llong, std::atomic_long, std::atomic_ptrdiff_t, std::atomic_schar, std::atomic_short, std::atomic_signed_lock_free, std::atomic_size_t, std::atomic_uchar, std::atomic_uint16_t, std::atomic_uint32_t, std::atomic_uint64_t, std::atomic_uint8_t, std::atomic_uint_fast16_t, std::atomic_uint_fast32_t, std::atomic_uint_fast64_t, std::atomic_uint_fast8_t, std::atomic_uint, std::atomic_uint_least16_t, std::atomic_uint_least32_t, std::atomic_uint_least64_t, std::atomic_uint_least8_t, std::atomic_uintmax_t, std::atomic_uintptr_t, std::atomic_ullong, std::atomic_ulong, std::atomic_unsigned_lock_free, std::atomic_ushort, std::atomic_wchar_t,
std::adopt_lock_t, std::allocator_arg_t, std::bad_array_new_length, std::bad_function_call, std::bad_weak_ptr, std::bernoulli_distribution, std::chrono::high_resolution_clock, std::chrono::steady_clock, std::condition_variable, std::condition_variable_any, std::cv_status, std::defer_lock_t, std::error_category, std::error_code, std::error_condition, std::future_errc, std::future_error, std::future_status, std::imaxdiv_t, std::ios_base::failure, std::launch, std::lldiv_t, std::max_align_t, std::mutex, std::nullptr_t, std::random_device, std::recursive_mutex, std::recursive_timed_mutex, std::regex, std::regex_constants::error_type, std::regex_constants::match_flag_type, std::regex_constants::syntax_option_type, std::regex_error, std::shared_mutex, std::shared_timed_mutex, std::system_error, std::thread, std::timed_mutex, std::try_to_lock_t, std::u32string, std::u32string_view, std::wregex
C++17¶
std::align_val_t, std::any, std::bad_any_cast, std::bad_optional_access, std::bad_variant_access, std::byte, std::chars_format, std::from_chars_result, std::memory_resource, std::mono_state, std::string_view, std::to_chars_result, std::u16string, std::u16string_view, std::wstring_view,
std::filesystem::copy_options, std::filesystem::directory_entry, std::filesystem::directory_iterator, std::filesystem::directory_options, std::filesystem::file_status, std::filesystem::file_type, std::filesystem::filesystem_error, std::filesystem::path, std::filesystem::perm_options, std::filesystem::perms, std::filesystem::recursive_directory_iterator, std::filesystem::space_info,
std::pmr::memory_resource std::pmr::monotonic_buffer_resource std::pmr::pool_options std::pmr::synchronized_pool_resource std::pmr::unsynchronized_pool_resource
C++20¶
std::ambiguous_local_time, std::compare_three_way, std::endian, std::latch, std::noop_coroutine_promise, std::strong_equality, std::strong_ordering, std::suspend_always, std::suspend_never, std::nonstopstate_t, std::weak_equality, std::stop_source, std::stop_token, std::destroying_delete_t, std::format_error, std::jthread, std::partial_ordering, std::u8string_view, std::weak_ordering,
std::chrono::choose, std::chrono::day, std::chrono::file_clock, std::chrono::gps_clock, std::chrono::is_clock, std::chrono::last_spec, std::chrono::leap_second, std::chrono::local_info, std::chrono::local_t, std::chrono::month, std::chrono::month_day, std::chrono::month_day_last, std::chrono::month_weekday, std::chrono::month_weekday_last, std::chrono::nonexistent_local_time, std::chrono::sys_info, std::chrono::system_clock, std::chrono::tai_clock, std::chrono::time_zone, std::chrono::time_zone_link, std::chrono::tzdb, std::chrono::tzdb_list, std::chrono::utc_clock, std::chrono::weekday, std::chrono::weekday_indexed, std::chrono::weekday_last, std::chrono::year, std::chrono::year_month, std::chrono::year_month_day, std::chrono::year_month_day_last, std::chrono::year_month_weekday, std::chrono::year_month_weekday_last
C++23¶
std::bfloat16_t, std::float128_t, std::float16_t, std::float32_t, std::float64_t, std::ispanstream, std::ospanstream, std::range_format, std::spanbuf, std::spanstream, std::stacktrace, std::stacktrace_entry, std::unexpect_t, std::wispanstream std::wospanstream, std::wspanbuf, std::wspanstream
PREDEFINED MACROS¶
The following macros are predefined in the specified language and later.
C89¶
- __DATE__
- The local date in the form "Mmm dd yyyy".
- __FILE__
- The current file name, if any, or "stdin".
- __LINE__
- The current line number within __FILE__, if any, or an arbitrary line number.
- __STDC__
- 1. (Not defined in C++.)
- __STDC_VERSION__
- The current C version, e.g., 202311L for C23. (Not defined in C++.)
- __TIME__
- The local time in the form "hh:mm:ss".
C99¶
- __VA_ARGS__
- The variable arguments of a macro, if any.
C++¶
- __cplusplus
- The current C++ version e.g., 202302L for C++23. (Not defined in C.)
C23 | C++20¶
- __VA_OPT__
- Optional preprocessor tokens enclosed between parentheses.
NOTES¶
main()¶
In C, a function named main has its parameters and return type checked for those required by main; in C++, this occurs only if it is not declared const, consteval, constexpr, default, delete, final, inline, override, __restrict, static, virtual, nor volatile. If it is so declared, it's assumed to be a member function.
Permissive Types¶
Permissive types via the --permissive-types or -p option is not the default because always permitting keywords in language versions other than the current language as types can result in confusing errors. For example, if permissive types were the default, then you would get the following in C:
cdecl> declare D as virtual destructor
^ 14: warning: "virtual" is a keyword in C++ virtual D;
^ 22: syntax error: "destructor": unexpected token ...
Here, virtual, not being a keyword in C and therefore a valid name for a user-defined type, would be taken to be a type name, so cdecl would interpret that to mean you want to declare D as a variable of type virtual — and cdecl would do so by printing virtual D (but still warning that virtual is a keyword in C++). But then destructor would be unexpectedly encountered and generate an error. (It could easily be the case that you simply forgot to set the current language to C++ instead of C.)
With the default non-permissive behavior, you instead get:
cdecl> declare D as virtual destructor
^ 14: error: "virtual": unsupported keyword in C
which is clearer, but at the cost of not permitting valid declarations that use or keywords in language versions other than the current language as types.
CONFIGURATION FILES¶
cdeclrc¶
The cdeclrc file is used to configure cdecl by executing the contained commands on start-up (unless either the --no-config or -C option is given). The full path of this file can be specified by either the --config or -c option; or, if not, the path is taken from the value of the CDECLRC environment variable unless it is either unset or empty in which case the path defaults to ~/.cdeclrc.
The commands useful within a configuration file are:
- class, define, enum, namespace, struct, typedef, union, or using to pre-define user-specific types so that they may be subsequently used when either composing or deciphering declarations.
- #define to pre-define macros so that they may be subsequently expanded.
- include to include other configuration files.
- set to set the language or other options initially.
Configuration files may include blank lines, C-style /* */ comments, and C++-style // comments, all of which are ignored.
EXAMPLES¶
C Declarations¶
To declare an array of pointers to functions that are like malloc(3):
cdecl> declare fptab as array of pointer to function \ cdecl+ returning pointer to void void *(*fptab[])();
When you see this declaration in someone else's code, you can make sense out of it by doing:
cdecl> explain void *(*fptab[])()
The function prototype for a function such as _exit(2) would be declared with:
cdecl> declare _exit as function (retval as int) returning void void _exit(int retval);
As a more complex example, signal(2) would be fully defined as:
cdecl> declare signal as function \ cdecl+ (sig as int, \ cdecl+ f as pointer to function (int) returning void) \ cdecl+ returning pointer to function (int) returning void void (*signal(int sig, void (*f)(int)))(int);
This is made more comprehensible with one of define, typedef, or using:
cdecl> define pfi_v as pointer to function (int) returning void cdecl> declare signal as function \ cdecl+ (sig as int, f as pfi_v) returning pfi_v pfi_v signal(int sig, pfi_v f);
cdecl can help figure out where to put const and volatile qualifiers:
cdecl> declare pci as pointer to const int const int *pci; cdecl> declare cpi as const pointer to int int *const cpi;
C++ Declarations¶
c++decl can help with declaring references:
c++decl> declare rpc as reference to pointer to char char *&rpc;
c++decl can help with pointers to member of classes:
c++decl> declare p as pointer to member of class C int int C::*p;
and:
c++decl> declare p as pointer to member of class C \ c++decl+ function (i as int, j as int) \ c++decl+ returning pointer to class D D *(C::*p)(int i, int j)
To define types within scopes:
c++decl> define A::B::T1 as int c++decl> define T2 of scope A as int c++decl> define T3 of scope B of scope A as int c++decl> define T4 of scope A::B as int c++decl> define T5 of class C::D as int c++decl> class C { typedef int T; } c++decl> class C1 { class C2 { typedef int T; }; } c++decl> struct S { typedef int T; } c++decl> namespace N { typedef int T; } c++decl> namespace N::M { typedef int T; } c++decl> union U { typedef int T; }
Preprocessor Macros¶
cdecl can help with understanding how macros expand:
cdecl> #define NAME2_HELPER(A,B) A ## B cdecl> #define NAME2(A,B) NAME2_HELPER(A,B) cdecl> expand NAME2(var_, __LINE__) NAME2(var_, __LINE__) => NAME2_HELPER(A,B) | A => var_ | B => __LINE__ | | __LINE__ => 42 | B => 42 NAME2(var_, 42) => NAME2_HELPER(var_,42) | NAME2_HELPER(var_, 42) => A ## B | NAME2_HELPER(var_, 42) => var_ ## 42 | NAME2_HELPER(var_, 42) => var_42 NAME2(var_, 42) => var_42
EXIT STATUS¶
- 0
- Success.
- 64
- Command-line usage error.
- 65
- Syntax or semantic error.
- 66
- Open file error.
- 69
- System resource unavailable.
- 70
- Internal software error. (Please report the bug.)
- 71
- System error.
- 73
- Create file error.
- 74
- I/O error.
ENVIRONMENT¶
- CDECL_COLORS
- This variable specifies the colors and other attributes used to highlight various parts of the output in a manner similar to the GCC_COLORS variable used by gcc.
- As with gcc, the value is composed of a colon-separated sequence of capabilities. Each capability is of the form name[=SGR] where name is a capability name and SGR, if present, is a ``Select Graphic Rendition'' value that is a semicolon-separated list of integers in the range 0-255. An example SGR value is 31;1 that specifies a bright red foreground on the terminal's default background.
- Capability names containing upper-case letters are unique to cdecl; those in all lower-case are upwards compatibile with gcc.
- caret=SGR
- SGR for the caret pointing to the error on the line above (as with gcc). The default is 32;1 (bright green foreground over current terminal background).
- error=SGR
- SGR for the word ``error.'' The default is 31;1 (bright red foreground over current terminal background).
- HELP-keyword=SGR
- SGR for keywords in help output. The default is 1 (bold terminal foreground over current terminal background).
- HELP-nonterm=SGR
- SGR for nonterminals in help output. The default is 36 (cyan foreground over current terminal background).
- HELP-punct=SGR
- SGR for punctuation in help output. The default is 30;1 (dark dray forgreound over current terminal background).
- HELP-title=SGR
- SGR for titles in help output. The default is 34;1 (bright blue foreground over current terminal background).
- locus=SGR
- SGR for location information in error and warning messages. The default is 1 (bold current foreground over current terminal background).
- MACRO-no-expand=SGR
- SGR for a macro that can not expand. The default is 35 (magenta foreground over current terminal background).
- MACRO-punct=SGR
- SGR for macro expansion punctuation. The default is 32;1 (bright green foreground over current terminal background).
- MACRO-subst=SGR
- SGR for macro substituted tokens. The default is 36 (cyan foreground over current terminal background).
- PROMPT=SGR
- SGR for the prompt. The default is 32 (green foreground over current terminal background).
- warning=SGR
- SGR for the word ``warning.'' The default is 33;1 (bright yellow foreground over current terminal background).
- The term ``color'' is used loosely. In addition to colors, other character attributes such as bold, underlined, reverse video, etc., may be possible depending on the capabilities of the terminal.
- CDECL_DEBUG
- If set to an ``affirmative'' value (one of 1, t, true, y, or yes, case-insensitive), then cdecl will print its process ID and a ``waiting for debugger to attach'' message to standard error and wait indefinitely for a debugger to attach. (This is a debugging aid for developers of cdecl itself.) It is enabled only when cdecl was compiled with NDEBUG not defined.
- CDECL_TEST
- If set to an ``affirmative'' value (one of 1, t, true, y, or yes, case-insensitive), then cdecl will (not) do the following to facilitate testing:
- Not read any cdeclrc file unless specified explicitly via either the --config or -c option.
- Use 80 for the number of columns of the terminal ignoring its actual width and the COLUMNS variable.
- Use constant values for the __DATE__, __FILE__, __LINE__, and __TIME__ macros.
- CDECLRC
- The full path to the user-specific configuration file (see CONFIGURATION FILES). Used only if not empty and none of the --config, -c, --no-config, or -C options are given.
- COLUMNS
- The number of columns of the terminal on which cdecl is being run. Used to get the terminal's width for limiting error and warning messages' length. Takes precedence over the number of columns specified by the TERM variable.
- HOME
- The user's home directory: used to locate the default configuration file. If unset, the home directory is obtained from the password database entry for the effective user. If that fails, no default configuration file is read.
- TERM
- The type of the terminal on which cdecl is being run.
FILES¶
- ~/.cdeclrc
- The default cdecl configuration file (see CONFIGURATION FILES).
- ~/.editrc
- Individual editline(3) initialization file. On systems where the readline(3) API is provided but is just a wrapper around libedit (e.g., macOS), the ~/.editrc file, if present, is read instead of ~/.inputrc. If present, add the following (undocumented) command:
bind ^I rl_complete
- to make tab-completion work in cdecl.
- ~/.inputrc
- Individual readline(3) initialization file.
BUGS¶
Readline Wrapper Around Editline¶
On systems where the readline(3) API is provided but is just a wrapper around the Editline Library, (e.g., macOS), there are a few issues:
- The wrapper has a bug that prevents color prompts from working correctly. Therefore, the PROMPT color cabapility is ignored on systems that do not provide genuine GNU readline(3).
- Hitting tab when there are no completion matches ordinarily rings the terminal's bell. However, older versions of the wrapper don't provide the rl_ding() function needed to ring the bell so cdecl provides a substitute. However, the substitute doesn't respect the user's preferred bell style (none, audible, or visual) and always does an audible bell.
- In some cases, hitting tab causes the wrapper to suggest only a single completion rather than simply inserting it.
To avoid these issues, compile cdecl against the genuine GNU Readline Library.
See Also¶
CAVEATS¶
Unsupported Declarations¶
The following types of declarations are not currently supported:
- •
- Only enum, class, struct, and union names and scoped type declarations are supported; complete declarations are not:
struct S s; // supported struct S { typedef int Int; }; // supported (C++ only) struct S { int i; char c; } s; // not supported
- For array sizes, only integer literals, names, and * are supported; arbitrary expressions are not.
- In C23 and later, auto as a storage-class specifier is no longer supported.
- For C23 typeof and typeof_unqual declarations, only types are supported; arbitrary expressions are not:
typeof(int*) x // supported typeof(x) y // not supported
- •
- Only C++ lambda signature declarations are supported; complete lambda declarations are not:
[n](int x) // supported [n](int x) { return n * x; } // not supported
- C++ decltype, abbreviated function template, default argument, and template declarations are not supported.
- For C++ function exception specifications, only noexcept, noexcept(true), noexcept(false), and throw(), are supported; arbitrary expressions for noexcept or types for throw are not.
- C++ namespace alias declarations are not supported:
namespace ALN = A_Long_Name; // not supported
- Multiple _Alignas or alignas specifiers in the same declaration are not supported.
- For the argument to either the _Alignas or alignas specifier, only integer literals or types are supported; arbitrary expressions are not.
- Only simple C++ attribute specifiers like [[this]] are supported; attribute specifiers with namespaces are not. Additionally, optional arguments for deprecated and nodiscard are ignored.
- C++20 contracts ([[assert]], [[ensures]], and [[expects]]) are not supported.
- The C++20 explicit specifier with an expression is not supported.
Other Caveats¶
- •
- Macros are expanded only via the expand command and can't be used elsewhere:
cdecl> #define exp explain cdecl> exp int *p // error: can't alias commands cdecl> #define FILE_PTR FILE* cdecl> explain FILE_PTR f // error: can't use as type
- •
- When converting from pseudo-English to a C++ declaration, any enum, class, struct, or union type keyword omitted from a declaration (via omission from either the --explicit-ecsu or -S option, or the set explicit-ecsu command) makes the declaration not ``round-trippable'':
c++decl> declare pt as pointer to class T T *pt; c++decl> explain T *pt
^ 18: error: "T": unknown name
- This is because, when going from a C++ declaration to pseudo-English, cdecl doesn't know that an arbitrary name, in this example, T, is a class name.
- To include the type keywords explicitly and thus make the declarations ``round-trippable,'' include them via either the --explicit-ecsu or -S option, or the set explicit-ecsu command.
- Alternatively, declare the type via one of class, define, enum, struct, typedef, union, or using:
c++decl> class T c++decl> explain T *pt declare pt as pointer to T
- •
- While class, enum, explain, namespace, struct, typedef, union, and using can accept names that are cdecl keywords, cast, declare, and define can not; hence, not all explanations are ``round-trippable'':
cdecl> explain int explain declare explain as integer cdecl> declare explain as integer
^ 9: syntax error: "explain": name expected
- •
- When converting from one of the C++ overloaded operators &, *, +, ++, -, or --, to pseudo-English when declared as:
T operator OP(U);
- i.e., taking one parameter, it's ambiguous (to cdecl) between being a member or non-member operator since cdecl doesn't have the context in which the operator is declared. If it were declared in-class, e.g.:
class T { public:
// ...
T operator OP(U); };
- then clearly it's a member operator; if it were declared at file scope, then clearly it's a non-member operator; but cdecl doesn't have this context. In such cases, cdecl omits either member or non-member from its output.
- •
- When converting from pseudo-English to a C23 or C++11 declaration for auto (or __auto_type in GNU C), or a const, constinit, reference, or an rvalue reference variable that is not a function parameter, the output doesn't include an initializer:
c++decl> declare x as auto auto x; c++decl> declare x, y as structured binding auto [x, y]; c++decl> declare r as reference to int int &r;
- These are illegal C++ declarations since such declarations must be initialized.
- Only casting a name is supported; casting an expression is not.
- When converting from or to a C++ new-style cast, only some semantic validation is performed to determine whether the type of cast is legal.
- When a predefined type, e.g., size_t, uint16_t, etc., is shown (via the show command), the underlying type is merely typical and does not necessarily match the underlying type on any particular platform or even the platform on which cdecl is running.
- An integer literal given as the argument for an alignment specifier is only checked to ensure it's either zero or a power of two; it is not checked to see whether it meets the minimum alignment for the type.
- In GNU C, the type __int128 is a distinct type; in Microsoft C, the types __int8, __int16, __int32, __int64, and __wchar_t are keyword synonyms. These types can take modifiers:
unsigned __int128 x128; // legal in GNU C unsigned __int32 x32; // legal in Microsoft C
- In cdecl, these types are typedefs and can't take modifiers since that's illegal in C.
- •
- When using the --lineno or -L option with a here document containing escaped newlines (lines ending in \), the \s also need to be escaped to get the line numbers correct because the shell ordinarily strips escaped newlines; for example:
#define IDENT_CHARS "ABCDEFGHIJKLMNOPQRSTUVWXYZ_" \\
"abcdefghijklmnopqrstuvwxyz" \\
"0123456789"
AUTHORS¶
cdecl has been around since the mid-1980s and there have been many versions of cdecl, some with different subsets of authors. This list is a best-effort at a union of all authors. In reverse chronological order:
Paul J. Lucas <paul@lucasmail.org>
Peter Ammon <cdecl@ridiculousfish.com>
David R. Conrad <conrad@detroit.freenet.org>
Alexander Dupuy <dupuy@cs.columbia.edu>
Merlyn LeRoy <merlyn@rose3.rosemount.com>
Tony Hansen <tony@attmail.com>
David Wolverton <david_wolverton@att.com>
Graham Ross
SEE ALSO¶
bison(1), clang(1), flex(1), gcc(1), less(1), vi(1), yacc(1), isatty(3), readline(3), sysexits(3), editrc(5)
September 1, 2024 | PJL TOOLS |