Allow generation of nested target defined types in gdb target xml

This patch adds support to generate multiple nested architecture defined
data types in gdb target xml generated by openOCD. Architecture defined 
structs, unions, vectors nested in one or more architecture defined 
types can be generated now.

Example:

  <vector id="v2d" type="ieee_double" count="2"/>
  <vector id="v2u" type="uint64" count="2"/>
  <vector id="v2i" type="int64" count="2"/>

  <union id="vnd">
    <field name="f" type="v2d"/>
    <field name="u" type="v2u"/>
    <field name="s" type="v2i"/>
  </union>

Change-Id: I0f3c5c6daf3d22cde7e4b7b4165d2e97e25872f7
Signed-off-by: Omair Javaid <omair.javaid@linaro.org>
Reviewed-on: http://openocd.zylin.com/4372
Tested-by: jenkins
Reviewed-by: Matthias Welwarsky <matthias@welwarsky.de>
This commit is contained in:
Omair Javaid 2018-01-22 02:57:19 +05:00 committed by Matthias Welwarsky
parent db268a3115
commit b4a01f8cdc
1 changed files with 71 additions and 4 deletions

View File

@ -1946,12 +1946,45 @@ static const char *gdb_get_reg_type_name(enum reg_type type)
return "int"; /* "int" as default value */ return "int"; /* "int" as default value */
} }
static int lookup_add_arch_defined_types(char const **arch_defined_types_list[], const char *type_id,
int *num_arch_defined_types)
{
int tbl_sz = *num_arch_defined_types;
if (type_id != NULL && (strcmp(type_id, ""))) {
for (int j = 0; j < (tbl_sz + 1); j++) {
if (!((*arch_defined_types_list)[j])) {
(*arch_defined_types_list)[tbl_sz++] = type_id;
*arch_defined_types_list = realloc(*arch_defined_types_list,
sizeof(char *) * (tbl_sz + 1));
(*arch_defined_types_list)[tbl_sz] = NULL;
*num_arch_defined_types = tbl_sz;
return 1;
} else {
if (!strcmp((*arch_defined_types_list)[j], type_id))
return 0;
}
}
}
return -1;
}
static int gdb_generate_reg_type_description(struct target *target, static int gdb_generate_reg_type_description(struct target *target,
char **tdesc, int *pos, int *size, struct reg_data_type *type) char **tdesc, int *pos, int *size, struct reg_data_type *type,
char const **arch_defined_types_list[], int * num_arch_defined_types)
{ {
int retval = ERROR_OK; int retval = ERROR_OK;
if (type->type_class == REG_TYPE_CLASS_VECTOR) { if (type->type_class == REG_TYPE_CLASS_VECTOR) {
struct reg_data_type *data_type = type->reg_type_vector->type;
if (data_type->type == REG_TYPE_ARCH_DEFINED) {
if (lookup_add_arch_defined_types(arch_defined_types_list, data_type->id,
num_arch_defined_types))
gdb_generate_reg_type_description(target, tdesc, pos, size, data_type,
arch_defined_types_list,
num_arch_defined_types);
}
/* <vector id="id" type="type" count="count"/> */ /* <vector id="id" type="type" count="count"/> */
xml_printf(&retval, tdesc, pos, size, xml_printf(&retval, tdesc, pos, size,
"<vector id=\"%s\" type=\"%s\" count=\"%d\"/>\n", "<vector id=\"%s\" type=\"%s\" count=\"%d\"/>\n",
@ -1959,6 +1992,20 @@ static int gdb_generate_reg_type_description(struct target *target,
type->reg_type_vector->count); type->reg_type_vector->count);
} else if (type->type_class == REG_TYPE_CLASS_UNION) { } else if (type->type_class == REG_TYPE_CLASS_UNION) {
struct reg_data_type_union_field *field;
field = type->reg_type_union->fields;
while (field != NULL) {
struct reg_data_type *data_type = field->type;
if (data_type->type == REG_TYPE_ARCH_DEFINED) {
if (lookup_add_arch_defined_types(arch_defined_types_list, data_type->id,
num_arch_defined_types))
gdb_generate_reg_type_description(target, tdesc, pos, size, data_type,
arch_defined_types_list,
num_arch_defined_types);
}
field = field->next;
}
/* <union id="id"> /* <union id="id">
* <field name="name" type="type"/> ... * <field name="name" type="type"/> ...
* </union> */ * </union> */
@ -1966,7 +2013,6 @@ static int gdb_generate_reg_type_description(struct target *target,
"<union id=\"%s\">\n", "<union id=\"%s\">\n",
type->id); type->id);
struct reg_data_type_union_field *field;
field = type->reg_type_union->fields; field = type->reg_type_union->fields;
while (field != NULL) { while (field != NULL) {
xml_printf(&retval, tdesc, pos, size, xml_printf(&retval, tdesc, pos, size,
@ -1999,6 +2045,17 @@ static int gdb_generate_reg_type_description(struct target *target,
field = field->next; field = field->next;
} }
} else { } else {
while (field != NULL) {
struct reg_data_type *data_type = field->type;
if (data_type->type == REG_TYPE_ARCH_DEFINED) {
if (lookup_add_arch_defined_types(arch_defined_types_list, data_type->id,
num_arch_defined_types))
gdb_generate_reg_type_description(target, tdesc, pos, size, data_type,
arch_defined_types_list,
num_arch_defined_types);
}
}
/* <struct id="id"> /* <struct id="id">
* <field name="name" type="type"/> ... * <field name="name" type="type"/> ...
* </struct> */ * </struct> */
@ -2091,11 +2148,15 @@ static int gdb_generate_target_description(struct target *target, char **tdesc_o
struct reg **reg_list = NULL; struct reg **reg_list = NULL;
int reg_list_size; int reg_list_size;
char const **features = NULL; char const **features = NULL;
char const **arch_defined_types = NULL;
int feature_list_size = 0; int feature_list_size = 0;
int num_arch_defined_types = 0;
char *tdesc = NULL; char *tdesc = NULL;
int pos = 0; int pos = 0;
int size = 0; int size = 0;
arch_defined_types = calloc(1, sizeof(char *));
retval = target_get_gdb_reg_list(target, &reg_list, retval = target_get_gdb_reg_list(target, &reg_list,
&reg_list_size, REG_CLASS_ALL); &reg_list_size, REG_CLASS_ALL);
@ -2148,8 +2209,13 @@ static int gdb_generate_target_description(struct target *target, char **tdesc_o
if (reg_list[i]->reg_data_type != NULL) { if (reg_list[i]->reg_data_type != NULL) {
if (reg_list[i]->reg_data_type->type == REG_TYPE_ARCH_DEFINED) { if (reg_list[i]->reg_data_type->type == REG_TYPE_ARCH_DEFINED) {
/* generate <type... first, if there are architecture-defined types. */ /* generate <type... first, if there are architecture-defined types. */
if (lookup_add_arch_defined_types(&arch_defined_types,
reg_list[i]->reg_data_type->id,
&num_arch_defined_types))
gdb_generate_reg_type_description(target, &tdesc, &pos, &size, gdb_generate_reg_type_description(target, &tdesc, &pos, &size,
reg_list[i]->reg_data_type); reg_list[i]->reg_data_type,
&arch_defined_types,
&num_arch_defined_types);
type_str = reg_list[i]->reg_data_type->id; type_str = reg_list[i]->reg_data_type->id;
} else { } else {
@ -2199,6 +2265,7 @@ static int gdb_generate_target_description(struct target *target, char **tdesc_o
error: error:
free(features); free(features);
free(reg_list); free(reg_list);
free(arch_defined_types);
if (retval == ERROR_OK) if (retval == ERROR_OK)
*tdesc_out = tdesc; *tdesc_out = tdesc;