diff options
author | Artem Dergachev <artem.dergachev@gmail.com> | 2019-06-25 02:16:53 +0000 |
---|---|---|
committer | Artem Dergachev <artem.dergachev@gmail.com> | 2019-06-25 02:16:53 +0000 |
commit | beb85ad66de5af11e45bae236a3dd5a6a4762150 (patch) | |
tree | 49433d897ee48c183699cb06861ab1894a663b5d /clang/utils/analyzer/exploded-graph-rewriter.py | |
parent | b9c94f946f2ca45734af666d6afba0234b21411b (diff) | |
download | bcm5719-llvm-beb85ad66de5af11e45bae236a3dd5a6a4762150.tar.gz bcm5719-llvm-beb85ad66de5af11e45bae236a3dd5a6a4762150.zip |
[analyzer] exploded-graph-rewriter: Add support for range constraints.
Diff support included.
A cheap solution is implemented that treats range constraints as
"some sort of key-value map", so it's going to be trivial
to add support for other such maps later, such as dynamic type info.
Differential Revision: https://reviews.llvm.org/D63685
llvm-svn: 364268
Diffstat (limited to 'clang/utils/analyzer/exploded-graph-rewriter.py')
-rwxr-xr-x | clang/utils/analyzer/exploded-graph-rewriter.py | 65 |
1 files changed, 63 insertions, 2 deletions
diff --git a/clang/utils/analyzer/exploded-graph-rewriter.py b/clang/utils/analyzer/exploded-graph-rewriter.py index 1eb2f896bee..4aae72c0da6 100755 --- a/clang/utils/analyzer/exploded-graph-rewriter.py +++ b/clang/utils/analyzer/exploded-graph-rewriter.py @@ -25,6 +25,19 @@ def diff_dicts(curr, prev): return (removed, added) +# Represents any program state trait that is a dictionary of key-value pairs. +class GenericMap(object): + def __init__(self, generic_map): + self.generic_map = generic_map + + def diff(self, prev): + return diff_dicts(self.generic_map, prev.generic_map) + + def is_different(self, prev): + removed, added = self.diff(prev) + return len(removed) != 0 or len(added) != 0 + + # A deserialized source location. class SourceLocation(object): def __init__(self, json_loc): @@ -203,8 +216,10 @@ class ProgramState(object): if json_ps['store'] is not None else None self.environment = Environment(json_ps['environment']) \ if json_ps['environment'] is not None else None + self.constraints = GenericMap(collections.OrderedDict([ + (c['symbol'], c['range']) for c in json_ps['constraints'] + ])) if json_ps['constraints'] is not None else None # TODO: Objects under construction. - # TODO: Constraint ranges. # TODO: Dynamic types of objects. # TODO: Checker messages. @@ -479,11 +494,57 @@ class DotDumpVisitor(object): else: self._dump('</td></tr><tr><td align="left">') self.visit_store(s.store) - self._dump('</td></tr><hr />') + self._dump('</td></tr>') + + def visit_generic_map(self, m, prev_m=None): + self._dump('<table border="0">') + + def dump_pair(m, k, is_added=None): + self._dump('<tr><td>%s</td>' + '<td align="left">%s</td>' + '<td align="left">%s</td></tr>' + % (self._diff_plus_minus(is_added), + k, m.generic_map[k])) + + if prev_m is not None: + removed, added = m.diff(prev_m) + for k in removed: + dump_pair(prev_m, k, False) + for k in added: + dump_pair(m, k, True) + else: + for k in m.generic_map: + dump_pair(m, k, None) + + self._dump('</table>') + + def visit_generic_map_in_state(self, selector, s, prev_s=None): + self._dump('<tr><td align="left">' + '<b>Ranges: </b>') + m = getattr(s, selector) + if m is None: + self._dump('<i> Nothing!</i>') + else: + prev_m = None + if prev_s is not None: + prev_m = getattr(prev_s, selector) + if prev_m is not None: + if m.is_different(prev_m): + self._dump('</td></tr><tr><td align="left">') + self.visit_generic_map(m, prev_m) + else: + self._dump('<i> No changes!</i>') + if prev_m is None: + self._dump('</td></tr><tr><td align="left">') + self.visit_generic_map(m) + self._dump('</td></tr>') def visit_state(self, s, prev_s): self.visit_store_in_state(s, prev_s) + self._dump('<hr />') self.visit_environment_in_state(s, prev_s) + self._dump('<hr />') + self.visit_generic_map_in_state('constraints', s, prev_s) def visit_node(self, node): self._dump('%s [shape=record,label=<<table border="0">' |