mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-21 01:12:32 +08:00
d2f8a107b7
There is a single declaration of set_tdesc_osabi that is shared between gdbserver/ and gdb/, this declaration takes a 'const char *' argument which is the string representing an osabi. Then in gdb/ we have an overload of set_tdesc_osabi which takes an 'enum gdb_osabi'. In this commit I change the shared set_tdesc_osabi to be the version which takes an 'enum gdb_osabi', and I remove the version which takes a 'const char *'. All users of set_tdesc_osabi are updated to pass an 'enum gdb_osabi'. The features/ code, which is generated from the xml files, requires a new function to be added to osabi.{c,h} which can return a string representation of an 'enum gdb_osabi'. With that new function in place the features/ code is regenerated. This change is being made to support the next commit. In the next commit gdbserver will be updated to call set_tdesc_osabi in more cases. The problem is that gdbserver stores the osabi as a string. The issue here is that a typo in the gdbserver/ code might go unnoticed and result in gdbserver sending back an invalid osabi string. To fix this we want gdbserver to pass an 'enum gdb_osabi' to the set_tdesc_osabi function. With that requirement in place it seems to make sense if all calls to set_tdesc_osabi pass an 'enum gdb_osabi'. There should be no user visible changes after this commit. Approved-By: Luis Machado <luis.machado@arm.com> Approved-By: Simon Marchi <simon.marchi@efficios.com>
468 lines
13 KiB
C++
468 lines
13 KiB
C++
/* Copyright (C) 2006-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 COMMON_TDESC_H
|
|
#define COMMON_TDESC_H
|
|
|
|
#include "gdbsupport/osabi.h"
|
|
|
|
struct tdesc_feature;
|
|
struct tdesc_type;
|
|
struct tdesc_type_builtin;
|
|
struct tdesc_type_vector;
|
|
struct tdesc_type_with_fields;
|
|
struct tdesc_reg;
|
|
struct target_desc;
|
|
|
|
/* The interface to visit different elements of target description. */
|
|
|
|
class tdesc_element_visitor
|
|
{
|
|
public:
|
|
virtual void visit_pre (const target_desc *e)
|
|
{}
|
|
|
|
virtual void visit_post (const target_desc *e)
|
|
{}
|
|
|
|
virtual void visit_pre (const tdesc_feature *e)
|
|
{}
|
|
|
|
virtual void visit_post (const tdesc_feature *e)
|
|
{}
|
|
|
|
virtual void visit (const tdesc_type_builtin *e)
|
|
{}
|
|
|
|
virtual void visit (const tdesc_type_vector *e)
|
|
{}
|
|
|
|
virtual void visit (const tdesc_type_with_fields *e)
|
|
{}
|
|
|
|
virtual void visit (const tdesc_reg *e)
|
|
{}
|
|
};
|
|
|
|
class tdesc_element
|
|
{
|
|
public:
|
|
virtual void accept (tdesc_element_visitor &v) const = 0;
|
|
};
|
|
|
|
/* An individual register from a target description. */
|
|
|
|
struct tdesc_reg : tdesc_element
|
|
{
|
|
tdesc_reg (struct tdesc_feature *feature, const std::string &name_,
|
|
int regnum, int save_restore_, const char *group_,
|
|
int bitsize_, const char *type_);
|
|
|
|
virtual ~tdesc_reg () = default;
|
|
|
|
DISABLE_COPY_AND_ASSIGN (tdesc_reg);
|
|
|
|
/* The name of this register. In standard features, it may be
|
|
recognized by the architecture support code, or it may be purely
|
|
for the user. */
|
|
std::string name;
|
|
|
|
/* The register number used by this target to refer to this
|
|
register. This is used for remote p/P packets and to determine
|
|
the ordering of registers in the remote g/G packets. */
|
|
long target_regnum;
|
|
|
|
/* If this flag is set, GDB should save and restore this register
|
|
around calls to an inferior function. */
|
|
int save_restore;
|
|
|
|
/* The name of the register group containing this register, or empty
|
|
if the group should be automatically determined from the
|
|
register's type. If this is "general", "float", or "vector", the
|
|
corresponding "info" command should display this register's
|
|
value. It can be an arbitrary string, but should be limited to
|
|
alphanumeric characters and internal hyphens. Currently other
|
|
strings are ignored (treated as empty). */
|
|
std::string group;
|
|
|
|
/* The size of the register, in bits. */
|
|
int bitsize;
|
|
|
|
/* The type of the register. This string corresponds to either
|
|
a named type from the target description or a predefined
|
|
type from GDB. */
|
|
std::string type;
|
|
|
|
/* The target-described type corresponding to TYPE, if found. */
|
|
struct tdesc_type *tdesc_type;
|
|
|
|
void accept (tdesc_element_visitor &v) const override
|
|
{
|
|
v.visit (this);
|
|
}
|
|
|
|
bool operator== (const tdesc_reg &other) const
|
|
{
|
|
return (name == other.name
|
|
&& target_regnum == other.target_regnum
|
|
&& save_restore == other.save_restore
|
|
&& bitsize == other.bitsize
|
|
&& group == other.group
|
|
&& type == other.type);
|
|
}
|
|
|
|
bool operator!= (const tdesc_reg &other) const
|
|
{
|
|
return !(*this == other);
|
|
}
|
|
};
|
|
|
|
typedef std::unique_ptr<tdesc_reg> tdesc_reg_up;
|
|
|
|
/* Declaration of a structure that holds information about one
|
|
"compatibility" entry within a target description. */
|
|
|
|
struct tdesc_compatible_info;
|
|
|
|
/* A pointer to a single piece of compatibility information. */
|
|
|
|
typedef std::unique_ptr<tdesc_compatible_info> tdesc_compatible_info_up;
|
|
|
|
/* Return a vector of compatibility information pointers from the target
|
|
description TARGET_DESC. */
|
|
|
|
const std::vector<tdesc_compatible_info_up> &tdesc_compatible_info_list
|
|
(const target_desc *target_desc);
|
|
|
|
/* Return the architecture name from a compatibility information
|
|
COMPATIBLE. */
|
|
|
|
const char *tdesc_compatible_info_arch_name
|
|
(const tdesc_compatible_info_up &compatible);
|
|
|
|
enum tdesc_type_kind
|
|
{
|
|
/* Predefined types. */
|
|
TDESC_TYPE_BOOL,
|
|
TDESC_TYPE_INT8,
|
|
TDESC_TYPE_INT16,
|
|
TDESC_TYPE_INT32,
|
|
TDESC_TYPE_INT64,
|
|
TDESC_TYPE_INT128,
|
|
TDESC_TYPE_UINT8,
|
|
TDESC_TYPE_UINT16,
|
|
TDESC_TYPE_UINT32,
|
|
TDESC_TYPE_UINT64,
|
|
TDESC_TYPE_UINT128,
|
|
TDESC_TYPE_CODE_PTR,
|
|
TDESC_TYPE_DATA_PTR,
|
|
TDESC_TYPE_IEEE_HALF,
|
|
TDESC_TYPE_IEEE_SINGLE,
|
|
TDESC_TYPE_IEEE_DOUBLE,
|
|
TDESC_TYPE_ARM_FPA_EXT,
|
|
TDESC_TYPE_I387_EXT,
|
|
TDESC_TYPE_BFLOAT16,
|
|
|
|
/* Types defined by a target feature. */
|
|
TDESC_TYPE_VECTOR,
|
|
TDESC_TYPE_STRUCT,
|
|
TDESC_TYPE_UNION,
|
|
TDESC_TYPE_FLAGS,
|
|
TDESC_TYPE_ENUM
|
|
};
|
|
|
|
struct tdesc_type : tdesc_element
|
|
{
|
|
tdesc_type (const std::string &name_, enum tdesc_type_kind kind_)
|
|
: name (name_), kind (kind_)
|
|
{}
|
|
|
|
virtual ~tdesc_type () = default;
|
|
|
|
DISABLE_COPY_AND_ASSIGN (tdesc_type);
|
|
|
|
/* The name of this type. */
|
|
std::string name;
|
|
|
|
/* Identify the kind of this type. */
|
|
enum tdesc_type_kind kind;
|
|
|
|
bool operator== (const tdesc_type &other) const
|
|
{
|
|
return name == other.name && kind == other.kind;
|
|
}
|
|
|
|
bool operator!= (const tdesc_type &other) const
|
|
{
|
|
return !(*this == other);
|
|
}
|
|
};
|
|
|
|
typedef std::unique_ptr<tdesc_type> tdesc_type_up;
|
|
|
|
struct tdesc_type_builtin : tdesc_type
|
|
{
|
|
tdesc_type_builtin (const std::string &name, enum tdesc_type_kind kind)
|
|
: tdesc_type (name, kind)
|
|
{}
|
|
|
|
void accept (tdesc_element_visitor &v) const override
|
|
{
|
|
v.visit (this);
|
|
}
|
|
};
|
|
|
|
/* tdesc_type for vector types. */
|
|
|
|
struct tdesc_type_vector : tdesc_type
|
|
{
|
|
tdesc_type_vector (const std::string &name, tdesc_type *element_type_,
|
|
int count_)
|
|
: tdesc_type (name, TDESC_TYPE_VECTOR),
|
|
element_type (element_type_), count (count_)
|
|
{}
|
|
|
|
void accept (tdesc_element_visitor &v) const override
|
|
{
|
|
v.visit (this);
|
|
}
|
|
|
|
struct tdesc_type *element_type;
|
|
int count;
|
|
};
|
|
|
|
/* A named type from a target description. */
|
|
|
|
struct tdesc_type_field
|
|
{
|
|
tdesc_type_field (const std::string &name_, tdesc_type *type_,
|
|
int start_, int end_)
|
|
: name (name_), type (type_), start (start_), end (end_)
|
|
{}
|
|
|
|
std::string name;
|
|
struct tdesc_type *type;
|
|
/* For non-enum-values, either both are -1 (non-bitfield), or both are
|
|
not -1 (bitfield). For enum values, start is the value (which could be
|
|
-1), end is -1. */
|
|
int start, end;
|
|
};
|
|
|
|
/* tdesc_type for struct, union, flags, and enum types. */
|
|
|
|
struct tdesc_type_with_fields : tdesc_type
|
|
{
|
|
tdesc_type_with_fields (const std::string &name, tdesc_type_kind kind,
|
|
int size_ = 0)
|
|
: tdesc_type (name, kind), size (size_)
|
|
{}
|
|
|
|
void accept (tdesc_element_visitor &v) const override
|
|
{
|
|
v.visit (this);
|
|
}
|
|
|
|
std::vector<tdesc_type_field> fields;
|
|
int size;
|
|
};
|
|
|
|
/* A feature from a target description. Each feature is a collection
|
|
of other elements, e.g. registers and types. */
|
|
|
|
struct tdesc_feature : tdesc_element
|
|
{
|
|
tdesc_feature (const std::string &name_)
|
|
: name (name_)
|
|
{}
|
|
|
|
virtual ~tdesc_feature () = default;
|
|
|
|
DISABLE_COPY_AND_ASSIGN (tdesc_feature);
|
|
|
|
/* The name of this feature. It may be recognized by the architecture
|
|
support code. */
|
|
std::string name;
|
|
|
|
/* The registers associated with this feature. */
|
|
std::vector<tdesc_reg_up> registers;
|
|
|
|
/* The types associated with this feature. */
|
|
std::vector<tdesc_type_up> types;
|
|
|
|
void accept (tdesc_element_visitor &v) const override;
|
|
|
|
bool operator== (const tdesc_feature &other) const;
|
|
|
|
bool operator!= (const tdesc_feature &other) const
|
|
{
|
|
return !(*this == other);
|
|
}
|
|
};
|
|
|
|
typedef std::unique_ptr<tdesc_feature> tdesc_feature_up;
|
|
|
|
/* A deleter adapter for a target_desc. There are different
|
|
implementations of this deleter class in gdb and gdbserver because even
|
|
though the target_desc name is shared between the two projects, the
|
|
actual implementations of target_desc are completely different. */
|
|
|
|
struct target_desc_deleter
|
|
{
|
|
void operator() (struct target_desc *desc) const;
|
|
};
|
|
|
|
/* A unique pointer specialization that holds a target_desc. */
|
|
|
|
typedef std::unique_ptr<target_desc, target_desc_deleter> target_desc_up;
|
|
|
|
/* Allocate a new target_desc. */
|
|
target_desc_up allocate_target_description (void);
|
|
|
|
/* Set TARGET_DESC's architecture by NAME. */
|
|
void set_tdesc_architecture (target_desc *target_desc,
|
|
const char *name);
|
|
|
|
/* Return the architecture associated with this target description as a string,
|
|
or NULL if no architecture was specified. */
|
|
const char *tdesc_architecture_name (const struct target_desc *target_desc);
|
|
|
|
/* Set TARGET_DESC's osabi to OSABI. */
|
|
void set_tdesc_osabi (target_desc *target_desc, enum gdb_osabi osabi);
|
|
|
|
/* Return the osabi associated with this target description as a string,
|
|
or NULL if no osabi was specified. */
|
|
const char *tdesc_osabi_name (const struct target_desc *target_desc);
|
|
|
|
/* Return the type associated with ID in the context of FEATURE, or
|
|
NULL if none. */
|
|
struct tdesc_type *tdesc_named_type (const struct tdesc_feature *feature,
|
|
const char *id);
|
|
|
|
/* Return the created feature named NAME in target description TDESC. */
|
|
struct tdesc_feature *tdesc_create_feature (struct target_desc *tdesc,
|
|
const char *name);
|
|
|
|
/* Return the created vector tdesc_type named NAME in FEATURE. */
|
|
struct tdesc_type *tdesc_create_vector (struct tdesc_feature *feature,
|
|
const char *name,
|
|
struct tdesc_type *field_type,
|
|
int count);
|
|
|
|
/* Return the created struct tdesc_type named NAME in FEATURE. */
|
|
tdesc_type_with_fields *tdesc_create_struct (struct tdesc_feature *feature,
|
|
const char *name);
|
|
|
|
/* Return the created union tdesc_type named NAME in FEATURE. */
|
|
tdesc_type_with_fields *tdesc_create_union (struct tdesc_feature *feature,
|
|
const char *name);
|
|
|
|
/* Return the created flags tdesc_type named NAME in FEATURE. */
|
|
tdesc_type_with_fields *tdesc_create_flags (struct tdesc_feature *feature,
|
|
const char *name,
|
|
int size);
|
|
|
|
/* Return the created enum tdesc_type named NAME in FEATURE. */
|
|
tdesc_type_with_fields *tdesc_create_enum (struct tdesc_feature *feature,
|
|
const char *name,
|
|
int size);
|
|
|
|
/* Add a new field to TYPE. FIELD_NAME is its name, and FIELD_TYPE is
|
|
its type. */
|
|
void tdesc_add_field (tdesc_type_with_fields *type, const char *field_name,
|
|
struct tdesc_type *field_type);
|
|
|
|
/* Add a new bitfield to TYPE, with range START to END. FIELD_NAME is its name,
|
|
and FIELD_TYPE is its type. */
|
|
void tdesc_add_typed_bitfield (tdesc_type_with_fields *type,
|
|
const char *field_name,
|
|
int start, int end,
|
|
struct tdesc_type *field_type);
|
|
|
|
/* Set the total length of TYPE. Structs which contain bitfields may
|
|
omit the reserved bits, so the end of the last field may not
|
|
suffice. */
|
|
void tdesc_set_struct_size (tdesc_type_with_fields *type, int size);
|
|
|
|
/* Add a new untyped bitfield to TYPE.
|
|
Untyped bitfields become either uint32 or uint64 depending on the size
|
|
of the underlying type. */
|
|
void tdesc_add_bitfield (tdesc_type_with_fields *type, const char *field_name,
|
|
int start, int end);
|
|
|
|
/* A flag is just a typed(bool) single-bit bitfield.
|
|
This function is kept to minimize changes in generated files. */
|
|
void tdesc_add_flag (tdesc_type_with_fields *type, int start,
|
|
const char *flag_name);
|
|
|
|
/* Add field with VALUE and NAME to the enum TYPE. */
|
|
void tdesc_add_enum_value (tdesc_type_with_fields *type, int value,
|
|
const char *name);
|
|
|
|
/* Create a register in feature FEATURE. */
|
|
void tdesc_create_reg (struct tdesc_feature *feature, const char *name,
|
|
int regnum, int save_restore, const char *group,
|
|
int bitsize, const char *type);
|
|
|
|
/* Return the tdesc in string XML format. */
|
|
|
|
const char *tdesc_get_features_xml (const target_desc *tdesc);
|
|
|
|
/* Print target description as xml. */
|
|
|
|
class print_xml_feature : public tdesc_element_visitor
|
|
{
|
|
public:
|
|
print_xml_feature (std::string *buffer_)
|
|
: m_buffer (buffer_),
|
|
m_depth (0)
|
|
{}
|
|
|
|
void visit_pre (const target_desc *e) override;
|
|
void visit_post (const target_desc *e) override;
|
|
void visit_pre (const tdesc_feature *e) override;
|
|
void visit_post (const tdesc_feature *e) override;
|
|
void visit (const tdesc_type_builtin *type) override;
|
|
void visit (const tdesc_type_vector *type) override;
|
|
void visit (const tdesc_type_with_fields *type) override;
|
|
void visit (const tdesc_reg *reg) override;
|
|
|
|
private:
|
|
|
|
/* Called with a positive value of ADJUST when we move inside an element,
|
|
for example inside <target>, and with a negative value when we leave
|
|
the element. In this class this function does nothing, but a
|
|
sub-class can override this to track the current level of nesting. */
|
|
void indent (int adjust)
|
|
{
|
|
m_depth += (adjust * 2);
|
|
}
|
|
|
|
/* Functions to add lines to the output buffer M_BUFFER. Each of these
|
|
functions appends a newline, so don't include one in the strings being
|
|
passed. */
|
|
void add_line (const std::string &str);
|
|
void add_line (const char *fmt, ...) ATTRIBUTE_PRINTF (2, 3);
|
|
|
|
/* The buffer we are writing too. */
|
|
std::string *m_buffer;
|
|
|
|
/* The current indentation depth. */
|
|
int m_depth;
|
|
};
|
|
|
|
#endif /* COMMON_TDESC_H */
|