|  | /* CLI options framework, for GDB. | 
|  |  | 
|  | Copyright (C) 2017-2024 Free Software Foundation, Inc. | 
|  |  | 
|  | This file is part of GDB. | 
|  |  | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3 of the License, or | 
|  | (at your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | #ifndef CLI_OPTION_H | 
|  | #define CLI_OPTION_H 1 | 
|  |  | 
|  | #include <optional> | 
|  | #include "gdbsupport/array-view.h" | 
|  | #include "completer.h" | 
|  | #include <string> | 
|  | #include "command.h" | 
|  |  | 
|  | namespace gdb { | 
|  | namespace option { | 
|  |  | 
|  | /* A type-erased option definition.  The actual type of the option is | 
|  | stored in the TYPE field.  Client code cannot define objects of | 
|  | this type directly (the ctor is protected).  Instead, one of the | 
|  | wrapper types below that extends this (boolean_option_def, | 
|  | flag_option_def, uinteger_option_def, etc.) should be defined.  */ | 
|  | struct option_def | 
|  | { | 
|  | /* The ctor is protected because you're supposed to construct using | 
|  | one of bool_option_def, etc. below.  */ | 
|  | protected: | 
|  | typedef void *(erased_get_var_address_ftype) (); | 
|  |  | 
|  | /* Construct an option.  NAME_ is the option's name.  VAR_TYPE_ | 
|  | defines the option's type.  ERASED_GET_VAR_ADDRESS_ is a pointer | 
|  | to a function that returns the option's control variable. | 
|  | SHOW_CMD_CB_ is a pointer to callback for the "show" command that | 
|  | is installed for this option.  SET_DOC_, SHOW_DOC_, HELP_DOC_ are | 
|  | used to create the option's "set/show" commands.  */ | 
|  | constexpr option_def (const char *name_, | 
|  | var_types var_type_, | 
|  | const literal_def *extra_literals_, | 
|  | erased_get_var_address_ftype *erased_get_var_address_, | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_, | 
|  | const char *help_doc_) | 
|  | : name (name_), type (var_type_), extra_literals (extra_literals_), | 
|  | erased_get_var_address (erased_get_var_address_), | 
|  | var_address {}, | 
|  | show_cmd_cb (show_cmd_cb_), | 
|  | set_doc (set_doc_), show_doc (show_doc_), help_doc (help_doc_) | 
|  | {} | 
|  |  | 
|  | public: | 
|  | /* The option's name.  */ | 
|  | const char *name; | 
|  |  | 
|  | /* The option's type.  */ | 
|  | var_types type; | 
|  |  | 
|  | /* Extra literals, such as `unlimited', accepted in lieu of a number.  */ | 
|  | const literal_def *extra_literals; | 
|  |  | 
|  | /* A function that gets the controlling variable's address, type | 
|  | erased.  */ | 
|  | erased_get_var_address_ftype *erased_get_var_address; | 
|  |  | 
|  | /* Get the controlling variable's address.  Each type of variable | 
|  | uses a different union member.  We do this instead of having a | 
|  | single hook that return a "void *", for better type safety.  This | 
|  | way, actual instances of concrete option_def types | 
|  | (boolean_option_def, etc.) fail to compile if you pass in a | 
|  | function with incorrect return type.  CTX here is the aggregate | 
|  | object that groups the option variables from which the callback | 
|  | returns the address of some member.  */ | 
|  | union | 
|  | { | 
|  | bool *(*boolean) (const option_def &, void *ctx); | 
|  | unsigned int *(*uinteger) (const option_def &, void *ctx); | 
|  | int *(*integer) (const option_def &, void *ctx); | 
|  | const char **(*enumeration) (const option_def &, void *ctx); | 
|  | std::string *(*string) (const option_def &, void *ctx); | 
|  | } | 
|  | var_address; | 
|  |  | 
|  | /* Pointer to null terminated list of enumerated values (like argv). | 
|  | Only used by var_enum options.  */ | 
|  | const char *const *enums = nullptr; | 
|  |  | 
|  | /* True if the option takes an argument.  */ | 
|  | bool have_argument = true; | 
|  |  | 
|  | /* The "show" callback to use in the associated "show" command. | 
|  | E.g., "show print elements".  */ | 
|  | show_value_ftype *show_cmd_cb; | 
|  |  | 
|  | /* The set/show/help strings.  These are shown in both the help of | 
|  | commands that use the option group this option belongs to (e.g., | 
|  | "help print"), and in the associated commands (e.g., "set/show | 
|  | print elements", "help set print elements").  */ | 
|  | const char *set_doc; | 
|  | const char *show_doc; | 
|  | const char *help_doc; | 
|  |  | 
|  | /* Convenience method that returns THIS as an option_def.  Useful | 
|  | when you're putting an option_def subclass in an option_def | 
|  | array_view.  */ | 
|  | const option_def &def () const | 
|  | { | 
|  | return *this; | 
|  | } | 
|  | }; | 
|  |  | 
|  | namespace detail | 
|  | { | 
|  |  | 
|  | /* Get the address of the option's value, cast to the right type. | 
|  | RetType is the restored type of the variable, and Context is the | 
|  | restored type of the context pointer.  */ | 
|  | template<typename RetType, typename Context> | 
|  | static inline RetType * | 
|  | get_var_address (const option_def &option, void *ctx) | 
|  | { | 
|  | using unerased_ftype = RetType *(Context *); | 
|  | unerased_ftype *fun = (unerased_ftype *) option.erased_get_var_address; | 
|  | return fun ((Context *) ctx); | 
|  | } | 
|  |  | 
|  | /* Convenience identity helper that just returns SELF.  */ | 
|  |  | 
|  | template<typename T> | 
|  | static T * | 
|  | return_self (T *self) | 
|  | { | 
|  | return self; | 
|  | } | 
|  |  | 
|  | } /* namespace detail */ | 
|  |  | 
|  | /* Follows the definitions of the option types that client code should | 
|  | define.  Note that objects of these types are placed in option_def | 
|  | arrays, by design, so they must not have data fields of their | 
|  | own.  */ | 
|  |  | 
|  | /* A var_boolean command line option.  */ | 
|  |  | 
|  | template<typename Context> | 
|  | struct boolean_option_def : option_def | 
|  | { | 
|  | boolean_option_def (const char *long_option_, | 
|  | bool *(*get_var_address_cb_) (Context *), | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : option_def (long_option_, var_boolean, nullptr, | 
|  | (erased_get_var_address_ftype *) get_var_address_cb_, | 
|  | show_cmd_cb_, | 
|  | set_doc_, show_doc_, help_doc_) | 
|  | { | 
|  | var_address.boolean = detail::get_var_address<bool, Context>; | 
|  | } | 
|  | }; | 
|  |  | 
|  | /* A flag command line option.  This is a var_boolean option under the | 
|  | hood, but unlike boolean options, flag options don't take an on/off | 
|  | argument.  */ | 
|  |  | 
|  | template<typename Context = bool> | 
|  | struct flag_option_def : boolean_option_def<Context> | 
|  | { | 
|  | flag_option_def (const char *long_option_, | 
|  | bool *(*var_address_cb_) (Context *), | 
|  | const char *set_doc_, | 
|  | const char *help_doc_ = nullptr) | 
|  | : boolean_option_def<Context> (long_option_, | 
|  | var_address_cb_, | 
|  | NULL, | 
|  | set_doc_, NULL, help_doc_) | 
|  | { | 
|  | this->have_argument = false; | 
|  | } | 
|  |  | 
|  | flag_option_def (const char *long_option_, | 
|  | const char *set_doc_, | 
|  | const char *help_doc_ = nullptr) | 
|  | : boolean_option_def<Context> (long_option_, | 
|  | gdb::option::detail::return_self, | 
|  | NULL, | 
|  | set_doc_, nullptr, help_doc_) | 
|  | { | 
|  | this->have_argument = false; | 
|  | } | 
|  | }; | 
|  |  | 
|  | /* A var_uinteger command line option.  */ | 
|  |  | 
|  | template<typename Context> | 
|  | struct uinteger_option_def : option_def | 
|  | { | 
|  | uinteger_option_def (const char *long_option_, | 
|  | unsigned int *(*get_var_address_cb_) (Context *), | 
|  | const literal_def *extra_literals_, | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : option_def (long_option_, var_uinteger, extra_literals_, | 
|  | (erased_get_var_address_ftype *) get_var_address_cb_, | 
|  | show_cmd_cb_, | 
|  | set_doc_, show_doc_, help_doc_) | 
|  | { | 
|  | var_address.uinteger = detail::get_var_address<unsigned int, Context>; | 
|  | } | 
|  |  | 
|  | uinteger_option_def (const char *long_option_, | 
|  | unsigned int *(*get_var_address_cb_) (Context *), | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : uinteger_option_def (long_option_, get_var_address_cb_, nullptr, | 
|  | show_cmd_cb_, set_doc_, show_doc_, help_doc_) | 
|  | { /* Nothing.  */ } | 
|  | }; | 
|  |  | 
|  | /* A var_pinteger command line option.  */ | 
|  |  | 
|  | template<typename Context> | 
|  | struct pinteger_option_def : option_def | 
|  | { | 
|  | pinteger_option_def (const char *long_option_, | 
|  | int *(*get_var_address_cb_) (Context *), | 
|  | const literal_def *extra_literals_, | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : option_def (long_option_, var_pinteger, extra_literals_, | 
|  | (erased_get_var_address_ftype *) get_var_address_cb_, | 
|  | show_cmd_cb_, | 
|  | set_doc_, show_doc_, help_doc_) | 
|  | { | 
|  | var_address.integer = detail::get_var_address<int, Context>; | 
|  | } | 
|  |  | 
|  | pinteger_option_def (const char *long_option_, | 
|  | int *(*get_var_address_cb_) (Context *), | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : pinteger_option_def (long_option_, get_var_address_cb_, nullptr, | 
|  | show_cmd_cb_, set_doc_, show_doc_, help_doc_) | 
|  | { /* Nothing.  */ } | 
|  | }; | 
|  |  | 
|  | /* An var_enum command line option.  */ | 
|  |  | 
|  | template<typename Context> | 
|  | struct enum_option_def : option_def | 
|  | { | 
|  | enum_option_def (const char *long_option_, | 
|  | const char *const *enumlist, | 
|  | const char **(*get_var_address_cb_) (Context *), | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : option_def (long_option_, var_enum, nullptr, | 
|  | (erased_get_var_address_ftype *) get_var_address_cb_, | 
|  | show_cmd_cb_, | 
|  | set_doc_, show_doc_, help_doc_) | 
|  | { | 
|  | var_address.enumeration = detail::get_var_address<const char *, Context>; | 
|  | this->enums = enumlist; | 
|  | } | 
|  | }; | 
|  |  | 
|  | /* A var_string command line option.  */ | 
|  |  | 
|  | template<typename Context> | 
|  | struct string_option_def : option_def | 
|  | { | 
|  | string_option_def (const char *long_option_, | 
|  | std::string *(*get_var_address_cb_) (Context *), | 
|  | show_value_ftype *show_cmd_cb_, | 
|  | const char *set_doc_, | 
|  | const char *show_doc_ = nullptr, | 
|  | const char *help_doc_ = nullptr) | 
|  | : option_def (long_option_, var_string, nullptr, | 
|  | (erased_get_var_address_ftype *) get_var_address_cb_, | 
|  | show_cmd_cb_, | 
|  | set_doc_, show_doc_, help_doc_) | 
|  | { | 
|  | var_address.string = detail::get_var_address<std::string, Context>; | 
|  | } | 
|  | }; | 
|  |  | 
|  | /* A group of options that all share the same context pointer to pass | 
|  | to the options' get-current-value callbacks.  */ | 
|  | struct option_def_group | 
|  | { | 
|  | /* The list of options.  */ | 
|  | gdb::array_view<const option_def> options; | 
|  |  | 
|  | /* The context pointer to pass to the options' get-current-value | 
|  | callbacks.  */ | 
|  | void *ctx; | 
|  | }; | 
|  |  | 
|  | /* Modes of operation for process_options.  */ | 
|  | enum process_options_mode | 
|  | { | 
|  | /* In this mode, options are only processed if we find a "--" | 
|  | delimiter.  Throws an error if unknown options are found.  */ | 
|  | PROCESS_OPTIONS_REQUIRE_DELIMITER, | 
|  |  | 
|  | /* In this mode, a "--" delimiter is not required.  Throws an error | 
|  | if unknown options are found, regardless of whether a delimiter | 
|  | is present.  */ | 
|  | PROCESS_OPTIONS_UNKNOWN_IS_ERROR, | 
|  |  | 
|  | /* In this mode, a "--" delimiter is not required.  If an unknown | 
|  | option is found, assume it is the command's argument/operand.  */ | 
|  | PROCESS_OPTIONS_UNKNOWN_IS_OPERAND, | 
|  | }; | 
|  |  | 
|  | /* Process ARGS, using OPTIONS_GROUP as valid options.  Returns true | 
|  | if the string has been fully parsed and there are no operands to | 
|  | handle by the caller.  Return false if options were parsed, and | 
|  | *ARGS now points at the first operand.  */ | 
|  | extern bool process_options | 
|  | (const char **args, | 
|  | process_options_mode mode, | 
|  | gdb::array_view<const option_def_group> options_group); | 
|  |  | 
|  | /* Complete ARGS on options listed by OPTIONS_GROUP.  Returns true if | 
|  | the string has been fully parsed and there are no operands to | 
|  | handle by the caller.  Return false if options were parsed, and | 
|  | *ARGS now points at the first operand.  */ | 
|  | extern bool complete_options | 
|  | (completion_tracker &tracker, | 
|  | const char **args, | 
|  | process_options_mode mode, | 
|  | gdb::array_view<const option_def_group> options_group); | 
|  |  | 
|  | /* Complete on all options listed by OPTIONS_GROUP.  */ | 
|  | extern void | 
|  | complete_on_all_options (completion_tracker &tracker, | 
|  | gdb::array_view<const option_def_group> options_group); | 
|  |  | 
|  | /* Return a string with the result of replacing %OPTIONS% in HELP_TMLP | 
|  | with an auto-generated "help" string fragment for all the options | 
|  | in OPTIONS_GROUP.  */ | 
|  | extern std::string build_help | 
|  | (const char *help_tmpl, | 
|  | gdb::array_view<const option_def_group> options_group); | 
|  |  | 
|  | /* Install set/show commands for options defined in OPTIONS.  DATA is | 
|  | a pointer to the structure that holds the data associated with the | 
|  | OPTIONS array.  */ | 
|  | extern void add_setshow_cmds_for_options (command_class cmd_class, void *data, | 
|  | gdb::array_view<const option_def> options, | 
|  | struct cmd_list_element **set_list, | 
|  | struct cmd_list_element **show_list); | 
|  |  | 
|  | } /* namespace option */ | 
|  | } /* namespace gdb */ | 
|  |  | 
|  | #endif /* CLI_OPTION_H */ |