LayoutGraph.java revision 1472:c18cbe5936b8
1/*
2 * Copyright (c) 2008, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.
8 *
9 * This code is distributed in the hope that it will be useful, but WITHOUT
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
12 * version 2 for more details (a copy is included in the LICENSE file that
13 * accompanied this code).
14 *
15 * You should have received a copy of the GNU General Public License version
16 * 2 along with this work; if not, write to the Free Software Foundation,
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18 *
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20 * or visit www.oracle.com if you need additional information or have any
21 * questions.
22 *
23 */
24package com.sun.hotspot.igv.layout;
25
26import java.util.HashSet;
27import java.util.HashMap;
28import java.util.Set;
29import java.util.SortedSet;
30import java.util.TreeSet;
31
32/**
33 *
34 * @author Thomas Wuerthinger
35 */
36public class LayoutGraph {
37
38    private Set<? extends Link> links;
39    private SortedSet<Vertex> vertices;
40    private HashMap<Vertex, Set<Port>> inputPorts;
41    private HashMap<Vertex, Set<Port>> outputPorts;
42    private HashMap<Port, Set<Link>> portLinks;
43
44    public LayoutGraph(Set<? extends Link> links) {
45        this(links, new HashSet<Vertex>());
46    }
47
48    public LayoutGraph(Set<? extends Link> links, Set<? extends Vertex> additionalVertices) {
49        this.links = links;
50        assert verify();
51
52        vertices = new TreeSet<Vertex>();
53        portLinks = new HashMap<Port, Set<Link>>();
54        inputPorts = new HashMap<Vertex, Set<Port>>();
55        outputPorts = new HashMap<Vertex, Set<Port>>();
56
57        for (Link l : links) {
58            Port p = l.getFrom();
59            Port p2 = l.getTo();
60            Vertex v1 = p.getVertex();
61            Vertex v2 = p2.getVertex();
62
63            if (!vertices.contains(v1)) {
64
65                outputPorts.put(v1, new HashSet<Port>(1));
66                inputPorts.put(v1, new HashSet<Port>(3));
67                vertices.add(v1);
68                assert vertices.contains(v1);
69            }
70
71            if (!vertices.contains(v2)) {
72                vertices.add(v2);
73                assert vertices.contains(v2);
74                outputPorts.put(v2, new HashSet<Port>(1));
75                inputPorts.put(v2, new HashSet<Port>(3));
76            }
77
78            if (!portLinks.containsKey(p)) {
79                HashSet<Link> hashSet = new HashSet<Link>(3);
80                portLinks.put(p, hashSet);
81            }
82
83            if (!portLinks.containsKey(p2)) {
84                portLinks.put(p2, new HashSet<Link>(3));
85            }
86
87            outputPorts.get(v1).add(p);
88            inputPorts.get(v2).add(p2);
89
90            portLinks.get(p).add(l);
91            portLinks.get(p2).add(l);
92        }
93
94        for (Vertex v : additionalVertices) {
95            if (!vertices.contains(v)) {
96                outputPorts.put(v, new HashSet<Port>(1));
97                inputPorts.put(v, new HashSet<Port>(3));
98                vertices.add(v);
99                vertices.contains(v);
100            }
101        }
102    }
103
104    public Set<Port> getInputPorts(Vertex v) {
105        return this.inputPorts.get(v);
106    }
107
108    public Set<Port> getOutputPorts(Vertex v) {
109        return this.outputPorts.get(v);
110    }
111
112    public Set<Link> getPortLinks(Port p) {
113        return portLinks.get(p);
114    }
115
116    public Set<? extends Link> getLinks() {
117        return links;
118    }
119
120    public boolean verify() {
121        return true;
122    }
123
124    public SortedSet<Vertex> getVertices() {
125        return vertices;
126    }
127
128    private void markNotRoot(Set<Vertex> notRootSet, Vertex v, Vertex startingVertex) {
129
130        if (notRootSet.contains(v)) {
131            return;
132        }
133        if (v != startingVertex) {
134            notRootSet.add(v);
135        }
136        Set<Port> outPorts = getOutputPorts(v);
137        for (Port p : outPorts) {
138            Set<Link> portLinks = getPortLinks(p);
139            for (Link l : portLinks) {
140                Port other = l.getTo();
141                Vertex otherVertex = other.getVertex();
142                if (otherVertex != startingVertex) {
143                    markNotRoot(notRootSet, otherVertex, startingVertex);
144                }
145            }
146        }
147    }
148
149    // Returns a set of vertices with the following properties:
150    // - All Vertices in the set startingRoots are elements of the set.
151    // - When starting a DFS at every vertex in the set, every vertex of the
152    //   whole graph is visited.
153    public Set<Vertex> findRootVertices(Set<Vertex> startingRoots) {
154
155        Set<Vertex> notRootSet = new HashSet<Vertex>();
156        for (Vertex v : startingRoots) {
157            if (!notRootSet.contains(v)) {
158                markNotRoot(notRootSet, v, v);
159            }
160        }
161
162        Set<Vertex> tmpVertices = getVertices();
163        for (Vertex v : tmpVertices) {
164            if (!notRootSet.contains(v)) {
165                if (this.getInputPorts(v).size() == 0) {
166                    markNotRoot(notRootSet, v, v);
167                }
168            }
169        }
170
171        for (Vertex v : tmpVertices) {
172            if (!notRootSet.contains(v)) {
173                markNotRoot(notRootSet, v, v);
174            }
175        }
176
177        Set<Vertex> result = new HashSet<Vertex>();
178        for (Vertex v : tmpVertices) {
179            if (!notRootSet.contains(v)) {
180                result.add(v);
181            }
182        }
183        assert tmpVertices.size() == 0 || result.size() > 0;
184        return result;
185    }
186
187    public Set<Vertex> findRootVertices() {
188        return findRootVertices(new HashSet<Vertex>());
189    }
190
191    public SortedSet<Cluster> getClusters() {
192
193        SortedSet<Cluster> clusters = new TreeSet<Cluster>();
194        for (Vertex v : getVertices()) {
195            if (v.getCluster() != null) {
196                clusters.add(v.getCluster());
197            }
198        }
199
200        return clusters;
201    }
202}
203