mirror of
git://gcc.gnu.org/git/gcc.git
synced 2025-04-05 23:31:21 +08:00
analyzer: make summarized dumps more comprehensive
The previous implementation of summarized dumps within region_model::dump_to_pp showed only the "top-level" keys within the current frame and for globals, and thus didn't e.g. show the values of fields of structs, or elements of arrays. This patch rewrites it to gather a vec of representative path_vars for all regions, using this to generate the dump, so that all expressible lvalues ought to make it to the summarized dump. gcc/analyzer/ChangeLog: * region-model.cc: Include "stor-layout.h". (region_model::dump_to_pp): Rather than calling dump_summary_of_map on each of the current frame and the globals, instead get a vec of representative path_vars for all regions, and then dump a summary of all of them. (region_model::dump_summary_of_map): Delete, rewriting into... (region_model::dump_summary_of_rep_path_vars): ...this new function, working on a vec of path_vars. (region_model::set_value): New overload. (region_model::get_representative_path_var): Rename "parent_region" local to "parent_reg" and consolidate with other local. Guard test for grandparent being stack on parent_reg being non-NULL. Move handling for parent being an array_region to within guard for parent_reg being non-NULL. (selftest::make_test_compound_type): New function. (selftest::test_dump_2): New selftest. (selftest::test_dump_3): New selftest. (selftest::test_stack_frames): Update expected output from simplified dump to show "a" and "b" from parent frame and "y" in child frame. (selftest::analyzer_region_model_cc_tests): Call test_dump_2 and test_dump_3. * region-model.h (region_model::set_value): New overload decl. (region_model::dump_summary_of_map): Delete. (region_model::dump_summary_of_rep_path_vars): New.
This commit is contained in:
parent
f665beeba6
commit
884d914111
@ -1,3 +1,31 @@
|
||||
2020-03-18 David Malcolm <dmalcolm@redhat.com>
|
||||
|
||||
* region-model.cc: Include "stor-layout.h".
|
||||
(region_model::dump_to_pp): Rather than calling
|
||||
dump_summary_of_map on each of the current frame and the globals,
|
||||
instead get a vec of representative path_vars for all regions,
|
||||
and then dump a summary of all of them.
|
||||
(region_model::dump_summary_of_map): Delete, rewriting into...
|
||||
(region_model::dump_summary_of_rep_path_vars): ...this new
|
||||
function, working on a vec of path_vars.
|
||||
(region_model::set_value): New overload.
|
||||
(region_model::get_representative_path_var): Rename
|
||||
"parent_region" local to "parent_reg" and consolidate with other
|
||||
local. Guard test for grandparent being stack on parent_reg being
|
||||
non-NULL. Move handling for parent being an array_region to
|
||||
within guard for parent_reg being non-NULL.
|
||||
(selftest::make_test_compound_type): New function.
|
||||
(selftest::test_dump_2): New selftest.
|
||||
(selftest::test_dump_3): New selftest.
|
||||
(selftest::test_stack_frames): Update expected output from
|
||||
simplified dump to show "a" and "b" from parent frame and "y" in
|
||||
child frame.
|
||||
(selftest::analyzer_region_model_cc_tests): Call test_dump_2 and
|
||||
test_dump_3.
|
||||
* region-model.h (region_model::set_value): New overload decl.
|
||||
(region_model::dump_summary_of_map): Delete.
|
||||
(region_model::dump_summary_of_rep_path_vars): New.
|
||||
|
||||
2020-03-18 David Malcolm <dmalcolm@redhat.com>
|
||||
|
||||
* region-model.h (class noop_region_model_context): New subclass
|
||||
|
@ -59,6 +59,7 @@ along with GCC; see the file COPYING3. If not see
|
||||
#include "analyzer/sm.h"
|
||||
#include "analyzer/pending-diagnostic.h"
|
||||
#include "analyzer/analyzer-selftests.h"
|
||||
#include "stor-layout.h"
|
||||
|
||||
#if ENABLE_ANALYZER
|
||||
|
||||
@ -3538,7 +3539,7 @@ region_model::dump_dot (const char *path) const
|
||||
/* Dump a multiline representation of this model to PP, showing the
|
||||
region hierarchy, the svalues, and any constraints.
|
||||
|
||||
If SUMMARIZE is true, show only the most pertient information,
|
||||
If SUMMARIZE is true, show only the most pertinent information,
|
||||
in a form that attempts to be less verbose.
|
||||
Otherwise, show all information. */
|
||||
|
||||
@ -3547,18 +3548,23 @@ region_model::dump_to_pp (pretty_printer *pp, bool summarize) const
|
||||
{
|
||||
if (summarize)
|
||||
{
|
||||
bool is_first = true;
|
||||
region_id frame_id = get_current_frame_id ();
|
||||
frame_region *frame = get_region <frame_region> (frame_id);
|
||||
if (frame)
|
||||
dump_summary_of_map (pp, frame, &is_first);
|
||||
|
||||
region_id globals_id = get_globals_region_id ();
|
||||
map_region *globals = get_region <map_region> (globals_id);
|
||||
if (globals)
|
||||
dump_summary_of_map (pp, globals, &is_first);
|
||||
auto_vec<path_var> rep_path_vars;
|
||||
|
||||
unsigned i;
|
||||
region *reg;
|
||||
FOR_EACH_VEC_ELT (m_regions, i, reg)
|
||||
{
|
||||
region_id rid = region_id::from_int (i);
|
||||
path_var pv = get_representative_path_var (rid);
|
||||
if (pv.m_tree)
|
||||
rep_path_vars.safe_push (pv);
|
||||
}
|
||||
bool is_first = true;
|
||||
|
||||
/* Work with a copy in case the get_lvalue calls change anything
|
||||
(they shouldn't). */
|
||||
region_model copy (*this);
|
||||
copy.dump_summary_of_rep_path_vars (pp, &rep_path_vars, &is_first);
|
||||
|
||||
equiv_class *ec;
|
||||
FOR_EACH_VEC_ELT (m_constraints->m_equiv_classes, i, ec)
|
||||
@ -3680,37 +3686,28 @@ dump_vec_of_tree (pretty_printer *pp,
|
||||
pp_printf (pp, "}: %s", label);
|
||||
}
|
||||
|
||||
/* Dump *MAP_REGION to PP in compact form, updating *IS_FIRST.
|
||||
Subroutine of region_model::dump_to_pp for use on stack frames and for
|
||||
the "globals" region. */
|
||||
/* Dump all *REP_PATH_VARS to PP in compact form, updating *IS_FIRST.
|
||||
Subroutine of region_model::dump_to_pp. */
|
||||
|
||||
void
|
||||
region_model::dump_summary_of_map (pretty_printer *pp,
|
||||
map_region *map_region,
|
||||
bool *is_first) const
|
||||
region_model::dump_summary_of_rep_path_vars (pretty_printer *pp,
|
||||
auto_vec<path_var> *rep_path_vars,
|
||||
bool *is_first)
|
||||
{
|
||||
/* Get the keys, sorted by tree_cmp. In particular, this ought
|
||||
to alphabetize any decls. */
|
||||
auto_vec<tree> keys (map_region->elements ());
|
||||
for (map_region::iterator_t iter = map_region->begin ();
|
||||
iter != map_region->end ();
|
||||
++iter)
|
||||
{
|
||||
tree key_a = (*iter).first;
|
||||
keys.quick_push (key_a);
|
||||
}
|
||||
keys.qsort (tree_cmp);
|
||||
|
||||
/* Print pointers, constants, and poisoned values that aren't "uninit";
|
||||
gather keys for unknown and uninit values. */
|
||||
unsigned i;
|
||||
tree key;
|
||||
auto_vec<tree> unknown_keys;
|
||||
auto_vec<tree> uninit_keys;
|
||||
FOR_EACH_VEC_ELT (keys, i, key)
|
||||
path_var *pv;
|
||||
auto_vec<tree> unknown_trees;
|
||||
auto_vec<tree> uninit_trees;
|
||||
FOR_EACH_VEC_ELT (*rep_path_vars, i, pv)
|
||||
{
|
||||
region_id child_rid = *map_region->get (key);
|
||||
|
||||
if (TREE_CODE (pv->m_tree) == STRING_CST)
|
||||
continue;
|
||||
tentative_region_model_context ctxt;
|
||||
region_id child_rid = get_lvalue (*pv, &ctxt);
|
||||
if (ctxt.had_errors_p ())
|
||||
continue;
|
||||
region *child_region = get_region (child_rid);
|
||||
if (!child_region)
|
||||
continue;
|
||||
@ -3729,7 +3726,7 @@ region_model::dump_summary_of_map (pretty_printer *pp,
|
||||
gcc_assert (!pointee_rid.null_p ());
|
||||
tree pointee = get_representative_path_var (pointee_rid).m_tree;
|
||||
dump_separator (pp, is_first);
|
||||
dump_tree (pp, key);
|
||||
dump_tree (pp, pv->m_tree);
|
||||
pp_string (pp, ": ");
|
||||
pp_character (pp, '&');
|
||||
if (pointee)
|
||||
@ -3740,23 +3737,23 @@ region_model::dump_summary_of_map (pretty_printer *pp,
|
||||
break;
|
||||
case SK_CONSTANT:
|
||||
dump_separator (pp, is_first);
|
||||
dump_tree (pp, key);
|
||||
dump_tree (pp, pv->m_tree);
|
||||
pp_string (pp, ": ");
|
||||
dump_tree (pp, sval->dyn_cast_constant_svalue ()->get_constant ());
|
||||
break;
|
||||
case SK_UNKNOWN:
|
||||
unknown_keys.safe_push (key);
|
||||
unknown_trees.safe_push (pv->m_tree);
|
||||
break;
|
||||
case SK_POISONED:
|
||||
{
|
||||
poisoned_svalue *poisoned_sval = as_a <poisoned_svalue *> (sval);
|
||||
enum poison_kind pkind = poisoned_sval->get_poison_kind ();
|
||||
if (pkind == POISON_KIND_UNINIT)
|
||||
uninit_keys.safe_push (key);
|
||||
uninit_trees.safe_push (pv->m_tree);
|
||||
else
|
||||
{
|
||||
dump_separator (pp, is_first);
|
||||
dump_tree (pp, key);
|
||||
dump_tree (pp, pv->m_tree);
|
||||
pp_printf (pp, ": %s", poison_kind_to_str (pkind));
|
||||
}
|
||||
}
|
||||
@ -3770,8 +3767,8 @@ region_model::dump_summary_of_map (pretty_printer *pp,
|
||||
}
|
||||
|
||||
/* Print unknown and uninitialized values in consolidated form. */
|
||||
dump_vec_of_tree (pp, is_first, unknown_keys, "unknown");
|
||||
dump_vec_of_tree (pp, is_first, uninit_keys, "uninit");
|
||||
dump_vec_of_tree (pp, is_first, unknown_trees, "unknown");
|
||||
dump_vec_of_tree (pp, is_first, uninit_trees, "uninit");
|
||||
}
|
||||
|
||||
/* Assert that this object is valid. */
|
||||
@ -5355,6 +5352,19 @@ region_model::set_value (region_id lhs_rid, svalue_id rhs_sid,
|
||||
get_region (lhs_rid)->set_value (*this, lhs_rid, rhs_sid, ctxt);
|
||||
}
|
||||
|
||||
/* Set the value of the region given by LHS to the value given
|
||||
by RHS. */
|
||||
|
||||
void
|
||||
region_model::set_value (tree lhs, tree rhs, region_model_context *ctxt)
|
||||
{
|
||||
region_id lhs_rid = get_lvalue (lhs, ctxt);
|
||||
svalue_id rhs_sid = get_rvalue (rhs, ctxt);
|
||||
gcc_assert (!lhs_rid.null_p ());
|
||||
gcc_assert (!rhs_sid.null_p ());
|
||||
set_value (lhs_rid, rhs_sid, ctxt);
|
||||
}
|
||||
|
||||
/* Determine what is known about the condition "LHS_SID OP RHS_SID" within
|
||||
this model. */
|
||||
|
||||
@ -5735,12 +5745,12 @@ path_var
|
||||
region_model::get_representative_path_var (region_id rid) const
|
||||
{
|
||||
region *reg = get_region (rid);
|
||||
region *parent_region = get_region (reg->get_parent ());
|
||||
region *parent_reg = get_region (reg->get_parent ());
|
||||
region_id stack_rid = get_stack_region_id ();
|
||||
if (!stack_rid.null_p ())
|
||||
if (parent_region->get_parent () == stack_rid)
|
||||
if (parent_reg && parent_reg->get_parent () == stack_rid)
|
||||
{
|
||||
frame_region *parent_frame = (frame_region *)parent_region;
|
||||
frame_region *parent_frame = (frame_region *)parent_reg;
|
||||
tree t = parent_frame->get_tree_for_child_region (rid);
|
||||
return path_var (t, parent_frame->get_depth ());
|
||||
}
|
||||
@ -5753,7 +5763,6 @@ region_model::get_representative_path_var (region_id rid) const
|
||||
|
||||
/* Handle e.g. fields of a local by recursing. */
|
||||
region_id parent_rid = reg->get_parent ();
|
||||
region *parent_reg = get_region (parent_rid);
|
||||
if (parent_reg)
|
||||
{
|
||||
if (reg->is_view_p ())
|
||||
@ -5782,25 +5791,25 @@ region_model::get_representative_path_var (region_id rid) const
|
||||
parent_pv.m_stack_depth);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* Handle elements within an array. */
|
||||
if (array_region *array_reg = parent_region->dyn_cast_array_region ())
|
||||
{
|
||||
array_region::key_t key;
|
||||
if (array_reg->get_key_for_child_region (rid, &key))
|
||||
{
|
||||
path_var parent_pv = get_representative_path_var (parent_rid);
|
||||
if (parent_pv.m_tree && reg->get_type ())
|
||||
{
|
||||
tree index = array_reg->constant_from_key (key);
|
||||
return path_var (build4 (ARRAY_REF,
|
||||
reg->get_type (),
|
||||
parent_pv.m_tree, index,
|
||||
NULL_TREE, NULL_TREE),
|
||||
parent_pv.m_stack_depth);
|
||||
}
|
||||
}
|
||||
/* Handle elements within an array. */
|
||||
if (array_region *array_reg = parent_reg->dyn_cast_array_region ())
|
||||
{
|
||||
array_region::key_t key;
|
||||
if (array_reg->get_key_for_child_region (rid, &key))
|
||||
{
|
||||
path_var parent_pv = get_representative_path_var (parent_rid);
|
||||
if (parent_pv.m_tree && reg->get_type ())
|
||||
{
|
||||
tree index = array_reg->constant_from_key (key);
|
||||
return path_var (build4 (ARRAY_REF,
|
||||
reg->get_type (),
|
||||
parent_pv.m_tree, index,
|
||||
NULL_TREE, NULL_TREE),
|
||||
parent_pv.m_stack_depth);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* Handle string literals. */
|
||||
@ -7400,6 +7409,124 @@ test_dump ()
|
||||
ASSERT_DUMP_EQ (model, true, "");
|
||||
}
|
||||
|
||||
/* Helper function for selftests. Create a struct or union type named NAME,
|
||||
with the fields given by the FIELD_DECLS in FIELDS.
|
||||
If IS_STRUCT is true create a RECORD_TYPE (aka a struct), otherwise
|
||||
create a UNION_TYPE. */
|
||||
|
||||
static tree
|
||||
make_test_compound_type (const char *name, bool is_struct,
|
||||
const auto_vec<tree> *fields)
|
||||
{
|
||||
tree t = make_node (is_struct ? RECORD_TYPE : UNION_TYPE);
|
||||
TYPE_NAME (t) = get_identifier (name);
|
||||
TYPE_SIZE (t) = 0;
|
||||
|
||||
tree fieldlist = NULL;
|
||||
int i;
|
||||
tree field;
|
||||
FOR_EACH_VEC_ELT (*fields, i, field)
|
||||
{
|
||||
gcc_assert (TREE_CODE (field) == FIELD_DECL);
|
||||
DECL_CONTEXT (field) = t;
|
||||
fieldlist = chainon (field, fieldlist);
|
||||
}
|
||||
fieldlist = nreverse (fieldlist);
|
||||
TYPE_FIELDS (t) = fieldlist;
|
||||
|
||||
layout_type (t);
|
||||
return t;
|
||||
}
|
||||
|
||||
/* Verify that dumps can show struct fields. */
|
||||
|
||||
static void
|
||||
test_dump_2 ()
|
||||
{
|
||||
auto_vec<tree> fields;
|
||||
tree x_field = build_decl (UNKNOWN_LOCATION, FIELD_DECL,
|
||||
get_identifier ("x"), integer_type_node);
|
||||
fields.safe_push (x_field);
|
||||
tree y_field = build_decl (UNKNOWN_LOCATION, FIELD_DECL,
|
||||
get_identifier ("y"), integer_type_node);
|
||||
fields.safe_push (y_field);
|
||||
tree coord_type = make_test_compound_type ("coord", true, &fields);
|
||||
|
||||
tree c = build_global_decl ("c", coord_type);
|
||||
tree c_x = build3 (COMPONENT_REF, TREE_TYPE (x_field),
|
||||
c, x_field, NULL_TREE);
|
||||
tree c_y = build3 (COMPONENT_REF, TREE_TYPE (y_field),
|
||||
c, y_field, NULL_TREE);
|
||||
|
||||
tree int_17 = build_int_cst (integer_type_node, 17);
|
||||
tree int_m3 = build_int_cst (integer_type_node, -3);
|
||||
|
||||
region_model model;
|
||||
model.set_value (c_x, int_17, NULL);
|
||||
model.set_value (c_y, int_m3, NULL);
|
||||
|
||||
/* Simplified dump. */
|
||||
ASSERT_DUMP_EQ (model, true, "c.x: 17, c.y: -3");
|
||||
|
||||
/* Full dump. */
|
||||
ASSERT_DUMP_EQ
|
||||
(model, false,
|
||||
"r0: {kind: `root', parent: null, sval: null}\n"
|
||||
"`-globals: r1: {kind: `globals', parent: r0, sval: null, map: {`c': r2}}\n"
|
||||
" `-`c': r2: {kind: `struct', parent: r1, sval: null, type: `struct coord', map: {`x': r3, `y': r4}}\n"
|
||||
" |: type: `struct coord'\n"
|
||||
" |-`x': r3: {kind: `primitive', parent: r2, sval: sv0, type: `int'}\n"
|
||||
" | |: sval: sv0: {type: `int', `17'}\n"
|
||||
" | |: type: `int'\n"
|
||||
" `-`y': r4: {kind: `primitive', parent: r2, sval: sv1, type: `int'}\n"
|
||||
" |: sval: sv1: {type: `int', `-3'}\n"
|
||||
" |: type: `int'\n"
|
||||
"svalues:\n"
|
||||
" sv0: {type: `int', `17'}\n"
|
||||
" sv1: {type: `int', `-3'}\n"
|
||||
"constraint manager:\n"
|
||||
" equiv classes:\n"
|
||||
" constraints:\n");
|
||||
}
|
||||
|
||||
/* Verify that dumps can show array elements. */
|
||||
|
||||
static void
|
||||
test_dump_3 ()
|
||||
{
|
||||
tree tlen = size_int (10);
|
||||
tree arr_type = build_array_type (char_type_node, build_index_type (tlen));
|
||||
|
||||
tree a = build_global_decl ("a", arr_type);
|
||||
|
||||
region_model model;
|
||||
tree int_0 = build_int_cst (integer_type_node, 0);
|
||||
tree a_0 = build4 (ARRAY_REF, char_type_node,
|
||||
a, int_0, NULL_TREE, NULL_TREE);
|
||||
tree char_A = build_int_cst (char_type_node, 'A');
|
||||
model.set_value (a_0, char_A, NULL);
|
||||
|
||||
/* Simplified dump. */
|
||||
ASSERT_DUMP_EQ (model, true, "a[0]: 65");
|
||||
|
||||
/* Full dump. */
|
||||
ASSERT_DUMP_EQ
|
||||
(model, false,
|
||||
"r0: {kind: `root', parent: null, sval: null}\n"
|
||||
"`-globals: r1: {kind: `globals', parent: r0, sval: null, map: {`a': r2}}\n"
|
||||
" `-`a': r2: {kind: `array', parent: r1, sval: null, type: `char[11]', array: {[0]: r3}}\n"
|
||||
" |: type: `char[11]'\n"
|
||||
" `-[0]: r3: {kind: `primitive', parent: r2, sval: sv1, type: `char'}\n"
|
||||
" |: sval: sv1: {type: `char', `65'}\n"
|
||||
" |: type: `char'\n"
|
||||
"svalues:\n"
|
||||
" sv0: {type: `int', `0'}\n"
|
||||
" sv1: {type: `char', `65'}\n"
|
||||
"constraint manager:\n"
|
||||
" equiv classes:\n"
|
||||
" constraints:\n");
|
||||
}
|
||||
|
||||
/* Verify that region_model::get_representative_tree works as expected. */
|
||||
|
||||
static void
|
||||
@ -7834,7 +7961,7 @@ test_stack_frames ()
|
||||
#endif
|
||||
|
||||
ASSERT_DUMP_EQ (model, true,
|
||||
"x: 0, {y}: unknown, p: &x, q: &p, b < 10, y != 5");
|
||||
"a: 42, x: 0, p: &x, q: &p, {b, y}: unknown, b < 10, y != 5");
|
||||
|
||||
/* Pop the "child_fn" frame from the stack. */
|
||||
purge_stats purged;
|
||||
@ -8475,6 +8602,8 @@ analyzer_region_model_cc_tests ()
|
||||
{
|
||||
test_tree_cmp_on_constants ();
|
||||
test_dump ();
|
||||
test_dump_2 ();
|
||||
test_dump_3 ();
|
||||
test_get_representative_tree ();
|
||||
test_unique_constants ();
|
||||
test_svalue_equality ();
|
||||
|
@ -1771,6 +1771,7 @@ class region_model
|
||||
|
||||
void set_value (region_id lhs_rid, svalue_id rhs_sid,
|
||||
region_model_context *ctxt);
|
||||
void set_value (tree lhs, tree rhs, region_model_context *ctxt);
|
||||
svalue_id set_to_new_unknown_value (region_id dst_rid, tree type,
|
||||
region_model_context *ctxt);
|
||||
|
||||
@ -1884,8 +1885,9 @@ class region_model
|
||||
void poison_any_pointers_to_bad_regions (const region_id_set &bad_regions,
|
||||
enum poison_kind pkind);
|
||||
|
||||
void dump_summary_of_map (pretty_printer *pp, map_region *map_region,
|
||||
bool *is_first) const;
|
||||
void dump_summary_of_rep_path_vars (pretty_printer *pp,
|
||||
auto_vec<path_var> *rep_path_vars,
|
||||
bool *is_first);
|
||||
|
||||
auto_delete_vec<svalue> m_svalues;
|
||||
auto_delete_vec<region> m_regions;
|
||||
|
Loading…
x
Reference in New Issue
Block a user