diff options
author | Joel Brobecker <brobecker@gnat.com> | 2012-02-29 19:46:48 +0000 |
---|---|---|
committer | Joel Brobecker <brobecker@gnat.com> | 2012-02-29 19:46:48 +0000 |
commit | 1b61134393256db280b968d24ddf7bb22b6aabcc (patch) | |
tree | 9c4bf66e9db573f2eb507eedb0fa8a6b4b8bb41b /gdb/ada-lang.c | |
parent | 41246937ec25e4d69802ee153ce8c562882126a1 (diff) | |
download | binutils-1b61134393256db280b968d24ddf7bb22b6aabcc.zip binutils-1b61134393256db280b968d24ddf7bb22b6aabcc.tar.gz binutils-1b61134393256db280b968d24ddf7bb22b6aabcc.tar.bz2 |
[Ada] avoid error message pollution with uninitialized tagged variable
Consider the following function...
3 procedure Foo is
4 I : Integer := Ident (10);
5 Obj : Base;
6 begin
7 Obj.X := I;
8 Do_Nothing (Obj.X'Address);
9 end Foo;
... where type "Base" is defined as a plain tagged record. If the user
stops execution before "Obj" gets initialized (for example, by inserting
a breakpoint "on" the function - or in other words, by inserting a
breakpoint using the function name as the location), one might get
the following of output if you try printing the value of obj:
(gdb) p obj
object size is larger than varsize-limit
object size is larger than varsize-limit
object size is larger than varsize-limit
$1 = object size is larger than varsize-limit
(x => 4204154)
Same thing with "info locals":
(gdb) info locals
i = 0
obj = object size is larger than varsize-limit
(x => 4204154)
We have also seen different error messages such as "Cannot read
memory at 0x...".
The error happens because we are trying to read the dispatch table
of a tagged type variable before it gets initialized. So the errors
might legitimately occur, and are supposed to be be contained.
However, the way things are written in ada-lang.c:ada_tag_name,
although the exception is in fact contained, the error message still
gets to be printed out.
This patch prevents this from happening by eliminating the use of
catch_errors, and using a TRY_CATCH block instead. Doing this removed
the need to use functions specifically fitted for catch_errors, and
thus some other simplifications could me made. In the end, the code
got reorganized a bit to better show the logic behind it, as well as
the common patterns.
gdb/ChangeLog:
* ada-lang.c (struct tag_args): Delete.
(ada_get_tsd_type): Function body moved up in source file.
(ada_tag_name_1, ada_tag_name_2): Delete.
(ada_get_tsd_from_tag): New function.
(ada_tag_name_from_tsd): New function.
(ada_tag_name): Use a TRY_CATCH block instead of catch_errors
to determine the tag name.
gdb/testsuite/ChangeLog:
* gdb.ada/tagged_not_init: New testcase.
Diffstat (limited to 'gdb/ada-lang.c')
-rw-r--r-- | gdb/ada-lang.c | 139 |
1 files changed, 72 insertions, 67 deletions
diff --git a/gdb/ada-lang.c b/gdb/ada-lang.c index 430e706..1b46778 100644 --- a/gdb/ada-lang.c +++ b/gdb/ada-lang.c @@ -6029,105 +6029,110 @@ type_from_tag (struct value *tag) return NULL; } -struct tag_args +/* Return the "ada__tags__type_specific_data" type. */ + +static struct type * +ada_get_tsd_type (struct inferior *inf) { - struct value *tag; - char *name; -}; + struct ada_inferior_data *data = get_ada_inferior_data (inf); + if (data->tsd_type == 0) + data->tsd_type = ada_find_any_type ("ada__tags__type_specific_data"); + return data->tsd_type; +} -static int ada_tag_name_1 (void *); -static int ada_tag_name_2 (struct tag_args *); +/* Return the TSD (type-specific data) associated to the given TAG. + TAG is assumed to be the tag of a tagged-type entity. -/* Wrapper function used by ada_tag_name. Given a struct tag_args* - value ARGS, sets ARGS->name to the tag name of ARGS->tag. - The value stored in ARGS->name is valid until the next call to - ada_tag_name_1. */ + May return NULL if we are unable to get the TSD. */ -static int -ada_tag_name_1 (void *args0) +static struct value * +ada_get_tsd_from_tag (struct value *tag) { - struct tag_args *args = (struct tag_args *) args0; - static char name[1024]; - char *p; struct value *val; + struct type *type; - args->name = NULL; - val = ada_value_struct_elt (args->tag, "tsd", 1); - if (val == NULL) - return ada_tag_name_2 (args); - val = ada_value_struct_elt (val, "expanded_name", 1); - if (val == NULL) - return 0; - read_memory_string (value_as_address (val), name, sizeof (name) - 1); - for (p = name; *p != '\0'; p += 1) - if (isalpha (*p)) - *p = tolower (*p); - args->name = name; - return 0; -} + /* First option: The TSD is simply stored as a field of our TAG. + Only older versions of GNAT would use this format, but we have + to test it first, because there are no visible markers for + the current approach except the absence of that field. */ -/* Return the "ada__tags__type_specific_data" type. */ + val = ada_value_struct_elt (tag, "tsd", 1); + if (val) + return val; -static struct type * -ada_get_tsd_type (struct inferior *inf) -{ - struct ada_inferior_data *data = get_ada_inferior_data (inf); + /* Try the second representation for the dispatch table (in which + there is no explicit 'tsd' field in the referent of the tag pointer, + and instead the tsd pointer is stored just before the dispatch + table. */ - if (data->tsd_type == 0) - data->tsd_type = ada_find_any_type ("ada__tags__type_specific_data"); - return data->tsd_type; + type = ada_get_tsd_type (current_inferior()); + if (type == NULL) + return NULL; + type = lookup_pointer_type (lookup_pointer_type (type)); + val = value_cast (type, tag); + if (val == NULL) + return NULL; + return value_ind (value_ptradd (val, -1)); } -/* Utility function for ada_tag_name_1 that tries the second - representation for the dispatch table (in which there is no - explicit 'tsd' field in the referent of the tag pointer, and instead - the tsd pointer is stored just before the dispatch table. */ - -static int -ada_tag_name_2 (struct tag_args *args) +/* Given the TSD of a tag (type-specific data), return a string + containing the name of the associated type. + + The returned value is good until the next call. May return NULL + if we are unable to determine the tag name. */ + +static char * +ada_tag_name_from_tsd (struct value *tsd) { - struct type *info_type; static char name[1024]; char *p; - struct value *val, *valp; + struct value *val; - args->name = NULL; - info_type = ada_get_tsd_type (current_inferior()); - if (info_type == NULL) - return 0; - info_type = lookup_pointer_type (lookup_pointer_type (info_type)); - valp = value_cast (info_type, args->tag); - if (valp == NULL) - return 0; - val = value_ind (value_ptradd (valp, -1)); - if (val == NULL) - return 0; - val = ada_value_struct_elt (val, "expanded_name", 1); + val = ada_value_struct_elt (tsd, "expanded_name", 1); if (val == NULL) - return 0; + return NULL; read_memory_string (value_as_address (val), name, sizeof (name) - 1); for (p = name; *p != '\0'; p += 1) if (isalpha (*p)) *p = tolower (*p); - args->name = name; - return 0; + return name; } /* The type name of the dynamic type denoted by the 'tag value TAG, as - a C string. */ + a C string. + + Return NULL if the TAG is not an Ada tag, or if we were unable to + determine the name of that tag. The result is good until the next + call. */ const char * ada_tag_name (struct value *tag) { - struct tag_args args; + volatile struct gdb_exception e; + char *name = NULL; if (!ada_is_tag_type (value_type (tag))) return NULL; - args.tag = tag; - args.name = NULL; - catch_errors (ada_tag_name_1, &args, NULL, RETURN_MASK_ALL); - return args.name; + + /* It is perfectly possible that an exception be raised while trying + to determine the TAG's name, even under normal circumstances: + The associated variable may be uninitialized or corrupted, for + instance. We do not let any exception propagate past this point. + instead we return NULL. + + We also do not print the error message either (which often is very + low-level (Eg: "Cannot read memory at 0x[...]"), but instead let + the caller print a more meaningful message if necessary. */ + TRY_CATCH (e, RETURN_MASK_ERROR) + { + struct value *tsd = ada_get_tsd_from_tag (tag); + + if (tsd != NULL) + name = ada_tag_name_from_tsd (tsd); + } + + return name; } /* The parent type of TYPE, or NULL if none. */ |