b811d2c292
gdb/ChangeLog: Update copyright year range in all GDB files.
357 lines
12 KiB
C++
357 lines
12 KiB
C++
/* CLI options framework, for GDB.
|
|
|
|
Copyright (C) 2017-2020 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 "gdbsupport/gdb_optional.h"
|
|
#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_,
|
|
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_),
|
|
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;
|
|
|
|
/* 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);
|
|
char **(*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,
|
|
(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 *),
|
|
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,
|
|
(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>;
|
|
}
|
|
};
|
|
|
|
/* A var_zuinteger_unlimited command line option. */
|
|
|
|
template<typename Context>
|
|
struct zuinteger_unlimited_option_def : option_def
|
|
{
|
|
zuinteger_unlimited_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)
|
|
: option_def (long_option_, var_zuinteger_unlimited,
|
|
(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>;
|
|
}
|
|
};
|
|
|
|
/* 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,
|
|
(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_,
|
|
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_string,
|
|
(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>;
|
|
}
|
|
};
|
|
|
|
/* 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 */
|