summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/constraints.dot27
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/constraints_diff.dot65
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/environment.dot1
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/environment_diff.dot3
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/store.dot1
-rw-r--r--clang/test/Analysis/exploded-graph-rewriter/store_diff.dot2
-rwxr-xr-xclang/utils/analyzer/exploded-graph-rewriter.py65
7 files changed, 162 insertions, 2 deletions
diff --git a/clang/test/Analysis/exploded-graph-rewriter/constraints.dot b/clang/test/Analysis/exploded-graph-rewriter/constraints.dot
new file mode 100644
index 00000000000..f9ffde74316
--- /dev/null
+++ b/clang/test/Analysis/exploded-graph-rewriter/constraints.dot
@@ -0,0 +1,27 @@
+// RUN: %exploded_graph_rewriter %s | FileCheck %s
+
+// FIXME: Substitution doesn't seem to work on Windows.
+// UNSUPPORTED: system-windows
+
+// CHECK: <tr><td align="left"><b>Ranges: </b></td></tr>
+// CHECK-SAME: <tr><td align="left"><table border="0">
+// CHECK-SAME: <tr>
+// CHECK-SAME: <td align="left">reg_$0<x></td>
+// CHECK-SAME: <td align="left">\{ [0, 0] \}</td>
+// CHECK-SAME: </tr>
+// CHECK-SAME: </table></td></tr>
+Node0x1 [shape=record,label=
+ "{
+ { "node_id": 1,
+ "pointer": "0x1",
+ "state_id": 2,
+ "program_points": [],
+ "program_state": {
+ "store": null,
+ "environment": null,
+ "constraints": [
+ { "symbol": "reg_$0<x>", "range": "{ [0, 0] }" }
+ ]
+ }
+ }
+\l}"];
diff --git a/clang/test/Analysis/exploded-graph-rewriter/constraints_diff.dot b/clang/test/Analysis/exploded-graph-rewriter/constraints_diff.dot
new file mode 100644
index 00000000000..a65a3a2464c
--- /dev/null
+++ b/clang/test/Analysis/exploded-graph-rewriter/constraints_diff.dot
@@ -0,0 +1,65 @@
+// RUN: %exploded_graph_rewriter -d %s | FileCheck %s
+
+// FIXME: Substitution doesn't seem to work on Windows.
+// UNSUPPORTED: system-windows
+
+Node0x1 [shape=record,label=
+ "{
+ { "node_id": 1,
+ "pointer": "0x1",
+ "state_id": 2,
+ "program_points": [],
+ "program_state": {
+ "store": null,
+ "environment": null,
+ "constraints": [
+ { "symbol": "reg_$0<x>", "range": "{ [0, 10] }" }
+ ]
+ }
+ }
+\l}"];
+
+Node0x1 -> Node0x3;
+
+// CHECK: Node0x3 [
+// CHECK-SAME: <tr>
+// CHECK-SAME: <td><font color="red">-</font></td>
+// CHECK-SAME: <td align="left">reg_$0<x></td>
+// CHECK-SAME: <td align="left">\{ [0, 10] \}</td>
+// CHECK-SAME: </tr>
+// CHECK-SAME: <tr>
+// CHECK-SAME: <td><font color="forestgreen">+</font></td>
+// CHECK-SAME: <td align="left">reg_$0<x></td>
+// CHECK-SAME: <td align="left">\{ [0, 5] \}</td>
+// CHECK-SAME: </tr>
+Node0x3 [shape=record,label=
+ "{
+ { "node_id": 3,
+ "pointer": "0x3",
+ "state_id": 4,
+ "program_points": [],
+ "program_state": {
+ "store": null,
+ "environment": null,
+ "constraints": [
+ { "symbol": "reg_$0<x>", "range": "{ [0, 5] }" }
+ ]
+ }
+ }
+\l}"];
+
+Node0x3 -> Node0x5;
+
+Node0x5 [shape=record,label=
+ "{
+ { "node_id": 5,
+ "pointer": "0x5",
+ "state_id": 6,
+ "program_points": [],
+ "program_state": {
+ "store": null,
+ "environment": null,
+ "constraints": null
+ }
+ }
+\l}"];
diff --git a/clang/test/Analysis/exploded-graph-rewriter/environment.dot b/clang/test/Analysis/exploded-graph-rewriter/environment.dot
index 7271684642a..e18cc7fc192 100644
--- a/clang/test/Analysis/exploded-graph-rewriter/environment.dot
+++ b/clang/test/Analysis/exploded-graph-rewriter/environment.dot
@@ -33,6 +33,7 @@ Node0x1 [shape=record,label=
"program_points": [],
"program_state": {
"store": null,
+ "constraints": null,
"environment": [
{
"location_context": "#0 Call",
diff --git a/clang/test/Analysis/exploded-graph-rewriter/environment_diff.dot b/clang/test/Analysis/exploded-graph-rewriter/environment_diff.dot
index 83cf0eb9471..33fe07d8789 100644
--- a/clang/test/Analysis/exploded-graph-rewriter/environment_diff.dot
+++ b/clang/test/Analysis/exploded-graph-rewriter/environment_diff.dot
@@ -12,6 +12,7 @@ Node0x1 [shape=record,label=
"program_points": [],
"program_state": {
"store": null,
+ "constraints": null,
"environment": [
{
"location_context": "#0 Call",
@@ -54,6 +55,7 @@ Node0x6 [shape=record,label=
"program_points": [],
"program_state": {
"store": null,
+ "constraints": null,
"environment": [
{
"location_context": "#0 Call",
@@ -90,6 +92,7 @@ Node0x9 [shape=record,label=
"program_points": [],
"program_state": {
"store": null,
+ "constraints": null,
"environment": [
{
"location_context": "#0 Call",
diff --git a/clang/test/Analysis/exploded-graph-rewriter/store.dot b/clang/test/Analysis/exploded-graph-rewriter/store.dot
index 3f1a937b737..22f4eba623c 100644
--- a/clang/test/Analysis/exploded-graph-rewriter/store.dot
+++ b/clang/test/Analysis/exploded-graph-rewriter/store.dot
@@ -28,6 +28,7 @@ Node0x1 [shape=record,label=
"program_points": [],
"program_state": {
"environment": null,
+ "constraints": null,
"store": [
{
"cluster": "x",
diff --git a/clang/test/Analysis/exploded-graph-rewriter/store_diff.dot b/clang/test/Analysis/exploded-graph-rewriter/store_diff.dot
index 47ea9af014b..d0a4e19817a 100644
--- a/clang/test/Analysis/exploded-graph-rewriter/store_diff.dot
+++ b/clang/test/Analysis/exploded-graph-rewriter/store_diff.dot
@@ -11,6 +11,7 @@ Node0x1 [shape=record,label=
"program_points": [],
"program_state": {
"environment": null,
+ "constraints": null,
"store": [
{
"cluster": "x",
@@ -52,6 +53,7 @@ Node0x4 [shape=record,label=
"program_points": [],
"program_state": {
"environment": null,
+ "constraints": null,
"store": [
{
"cluster": "x",
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">'
OpenPOWER on IntegriCloud