annotate src/share/vm/opto/phaseX.cpp @ 3992:d1bdeef3e3e2

7098282: G1: assert(interval >= 0) failed: Sanity check, referencePolicy.cpp: 76 Summary: There is a race between one thread successfully forwarding and copying the klass mirror for the SoftReference class (including the static master clock) and another thread attempting to use the master clock while attempting to discover a soft reference object. Maintain a shadow copy of the soft reference master clock and use the shadow during reference discovery and reference processing. Reviewed-by: tonyp, brutisso, ysr
author johnc
date Wed, 12 Oct 2011 10:25:51 -0700
parents 075ea0ed9e7c
children 670a74b863fc
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
3958
075ea0ed9e7c 7081842: assert(Compile::current()->unique() < (uint)MaxNodeLimit) failed: Node limit exceeded
kvn
parents: 1972
diff changeset
2 * Copyright (c) 1997, 2011, Oracle and/or its affiliates. All rights reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1017
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1017
diff changeset
20 * or visit www.oracle.com if you need additional information or have any
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1017
diff changeset
21 * questions.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
1972
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
25 #include "precompiled.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
26 #include "memory/allocation.inline.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
27 #include "opto/block.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
28 #include "opto/callnode.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
29 #include "opto/cfgnode.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
30 #include "opto/connode.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
31 #include "opto/idealGraphPrinter.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
32 #include "opto/loopnode.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
33 #include "opto/machnode.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
34 #include "opto/opcodes.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
35 #include "opto/phaseX.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
36 #include "opto/regalloc.hpp"
f95d63e2154a 6989984: Use standard include model for Hospot
stefank
parents: 1826
diff changeset
37 #include "opto/rootnode.hpp"
0
a61af66fc99e Initial load
duke
parents:
diff changeset
38
a61af66fc99e Initial load
duke
parents:
diff changeset
39 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
40 #define NODE_HASH_MINIMUM_SIZE 255
a61af66fc99e Initial load
duke
parents:
diff changeset
41 //------------------------------NodeHash---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
42 NodeHash::NodeHash(uint est_max_size) :
a61af66fc99e Initial load
duke
parents:
diff changeset
43 _max( round_up(est_max_size < NODE_HASH_MINIMUM_SIZE ? NODE_HASH_MINIMUM_SIZE : est_max_size) ),
a61af66fc99e Initial load
duke
parents:
diff changeset
44 _a(Thread::current()->resource_area()),
a61af66fc99e Initial load
duke
parents:
diff changeset
45 _table( NEW_ARENA_ARRAY( _a , Node* , _max ) ), // (Node**)_a->Amalloc(_max * sizeof(Node*)) ),
a61af66fc99e Initial load
duke
parents:
diff changeset
46 _inserts(0), _insert_limit( insert_limit() ),
a61af66fc99e Initial load
duke
parents:
diff changeset
47 _look_probes(0), _lookup_hits(0), _lookup_misses(0),
a61af66fc99e Initial load
duke
parents:
diff changeset
48 _total_insert_probes(0), _total_inserts(0),
a61af66fc99e Initial load
duke
parents:
diff changeset
49 _insert_probes(0), _grows(0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
50 // _sentinel must be in the current node space
a61af66fc99e Initial load
duke
parents:
diff changeset
51 _sentinel = new (Compile::current(), 1) ProjNode(NULL, TypeFunc::Control);
a61af66fc99e Initial load
duke
parents:
diff changeset
52 memset(_table,0,sizeof(Node*)*_max);
a61af66fc99e Initial load
duke
parents:
diff changeset
53 }
a61af66fc99e Initial load
duke
parents:
diff changeset
54
a61af66fc99e Initial load
duke
parents:
diff changeset
55 //------------------------------NodeHash---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
56 NodeHash::NodeHash(Arena *arena, uint est_max_size) :
a61af66fc99e Initial load
duke
parents:
diff changeset
57 _max( round_up(est_max_size < NODE_HASH_MINIMUM_SIZE ? NODE_HASH_MINIMUM_SIZE : est_max_size) ),
a61af66fc99e Initial load
duke
parents:
diff changeset
58 _a(arena),
a61af66fc99e Initial load
duke
parents:
diff changeset
59 _table( NEW_ARENA_ARRAY( _a , Node* , _max ) ),
a61af66fc99e Initial load
duke
parents:
diff changeset
60 _inserts(0), _insert_limit( insert_limit() ),
a61af66fc99e Initial load
duke
parents:
diff changeset
61 _look_probes(0), _lookup_hits(0), _lookup_misses(0),
a61af66fc99e Initial load
duke
parents:
diff changeset
62 _delete_probes(0), _delete_hits(0), _delete_misses(0),
a61af66fc99e Initial load
duke
parents:
diff changeset
63 _total_insert_probes(0), _total_inserts(0),
a61af66fc99e Initial load
duke
parents:
diff changeset
64 _insert_probes(0), _grows(0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
65 // _sentinel must be in the current node space
a61af66fc99e Initial load
duke
parents:
diff changeset
66 _sentinel = new (Compile::current(), 1) ProjNode(NULL, TypeFunc::Control);
a61af66fc99e Initial load
duke
parents:
diff changeset
67 memset(_table,0,sizeof(Node*)*_max);
a61af66fc99e Initial load
duke
parents:
diff changeset
68 }
a61af66fc99e Initial load
duke
parents:
diff changeset
69
a61af66fc99e Initial load
duke
parents:
diff changeset
70 //------------------------------NodeHash---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
71 NodeHash::NodeHash(NodeHash *nh) {
a61af66fc99e Initial load
duke
parents:
diff changeset
72 debug_only(_table = (Node**)badAddress); // interact correctly w/ operator=
a61af66fc99e Initial load
duke
parents:
diff changeset
73 // just copy in all the fields
a61af66fc99e Initial load
duke
parents:
diff changeset
74 *this = *nh;
a61af66fc99e Initial load
duke
parents:
diff changeset
75 // nh->_sentinel must be in the current node space
a61af66fc99e Initial load
duke
parents:
diff changeset
76 }
a61af66fc99e Initial load
duke
parents:
diff changeset
77
a61af66fc99e Initial load
duke
parents:
diff changeset
78 //------------------------------hash_find--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
79 // Find in hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
80 Node *NodeHash::hash_find( const Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
81 // ((Node*)n)->set_hash( n->hash() );
a61af66fc99e Initial load
duke
parents:
diff changeset
82 uint hash = n->hash();
a61af66fc99e Initial load
duke
parents:
diff changeset
83 if (hash == Node::NO_HASH) {
a61af66fc99e Initial load
duke
parents:
diff changeset
84 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
85 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
86 }
a61af66fc99e Initial load
duke
parents:
diff changeset
87 uint key = hash & (_max-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
88 uint stride = key | 0x01;
a61af66fc99e Initial load
duke
parents:
diff changeset
89 debug_only( _look_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
90 Node *k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
91 if( !k ) { // ?Miss?
a61af66fc99e Initial load
duke
parents:
diff changeset
92 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
93 return NULL; // Miss!
a61af66fc99e Initial load
duke
parents:
diff changeset
94 }
a61af66fc99e Initial load
duke
parents:
diff changeset
95
a61af66fc99e Initial load
duke
parents:
diff changeset
96 int op = n->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
97 uint req = n->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
98 while( 1 ) { // While probing hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
99 if( k->req() == req && // Same count of inputs
a61af66fc99e Initial load
duke
parents:
diff changeset
100 k->Opcode() == op ) { // Same Opcode
a61af66fc99e Initial load
duke
parents:
diff changeset
101 for( uint i=0; i<req; i++ )
a61af66fc99e Initial load
duke
parents:
diff changeset
102 if( n->in(i)!=k->in(i)) // Different inputs?
a61af66fc99e Initial load
duke
parents:
diff changeset
103 goto collision; // "goto" is a speed hack...
a61af66fc99e Initial load
duke
parents:
diff changeset
104 if( n->cmp(*k) ) { // Check for any special bits
a61af66fc99e Initial load
duke
parents:
diff changeset
105 debug_only( _lookup_hits++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
106 return k; // Hit!
a61af66fc99e Initial load
duke
parents:
diff changeset
107 }
a61af66fc99e Initial load
duke
parents:
diff changeset
108 }
a61af66fc99e Initial load
duke
parents:
diff changeset
109 collision:
a61af66fc99e Initial load
duke
parents:
diff changeset
110 debug_only( _look_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
111 key = (key + stride/*7*/) & (_max-1); // Stride through table with relative prime
a61af66fc99e Initial load
duke
parents:
diff changeset
112 k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
113 if( !k ) { // ?Miss?
a61af66fc99e Initial load
duke
parents:
diff changeset
114 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
115 return NULL; // Miss!
a61af66fc99e Initial load
duke
parents:
diff changeset
116 }
a61af66fc99e Initial load
duke
parents:
diff changeset
117 }
a61af66fc99e Initial load
duke
parents:
diff changeset
118 ShouldNotReachHere();
a61af66fc99e Initial load
duke
parents:
diff changeset
119 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
120 }
a61af66fc99e Initial load
duke
parents:
diff changeset
121
a61af66fc99e Initial load
duke
parents:
diff changeset
122 //------------------------------hash_find_insert-------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
123 // Find in hash table, insert if not already present
a61af66fc99e Initial load
duke
parents:
diff changeset
124 // Used to preserve unique entries in hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
125 Node *NodeHash::hash_find_insert( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
126 // n->set_hash( );
a61af66fc99e Initial load
duke
parents:
diff changeset
127 uint hash = n->hash();
a61af66fc99e Initial load
duke
parents:
diff changeset
128 if (hash == Node::NO_HASH) {
a61af66fc99e Initial load
duke
parents:
diff changeset
129 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
130 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
131 }
a61af66fc99e Initial load
duke
parents:
diff changeset
132 uint key = hash & (_max-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
133 uint stride = key | 0x01; // stride must be relatively prime to table siz
a61af66fc99e Initial load
duke
parents:
diff changeset
134 uint first_sentinel = 0; // replace a sentinel if seen.
a61af66fc99e Initial load
duke
parents:
diff changeset
135 debug_only( _look_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
136 Node *k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
137 if( !k ) { // ?Miss?
a61af66fc99e Initial load
duke
parents:
diff changeset
138 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
139 _table[key] = n; // Insert into table!
a61af66fc99e Initial load
duke
parents:
diff changeset
140 debug_only(n->enter_hash_lock()); // Lock down the node while in the table.
a61af66fc99e Initial load
duke
parents:
diff changeset
141 check_grow(); // Grow table if insert hit limit
a61af66fc99e Initial load
duke
parents:
diff changeset
142 return NULL; // Miss!
a61af66fc99e Initial load
duke
parents:
diff changeset
143 }
a61af66fc99e Initial load
duke
parents:
diff changeset
144 else if( k == _sentinel ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
145 first_sentinel = key; // Can insert here
a61af66fc99e Initial load
duke
parents:
diff changeset
146 }
a61af66fc99e Initial load
duke
parents:
diff changeset
147
a61af66fc99e Initial load
duke
parents:
diff changeset
148 int op = n->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
149 uint req = n->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
150 while( 1 ) { // While probing hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
151 if( k->req() == req && // Same count of inputs
a61af66fc99e Initial load
duke
parents:
diff changeset
152 k->Opcode() == op ) { // Same Opcode
a61af66fc99e Initial load
duke
parents:
diff changeset
153 for( uint i=0; i<req; i++ )
a61af66fc99e Initial load
duke
parents:
diff changeset
154 if( n->in(i)!=k->in(i)) // Different inputs?
a61af66fc99e Initial load
duke
parents:
diff changeset
155 goto collision; // "goto" is a speed hack...
a61af66fc99e Initial load
duke
parents:
diff changeset
156 if( n->cmp(*k) ) { // Check for any special bits
a61af66fc99e Initial load
duke
parents:
diff changeset
157 debug_only( _lookup_hits++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
158 return k; // Hit!
a61af66fc99e Initial load
duke
parents:
diff changeset
159 }
a61af66fc99e Initial load
duke
parents:
diff changeset
160 }
a61af66fc99e Initial load
duke
parents:
diff changeset
161 collision:
a61af66fc99e Initial load
duke
parents:
diff changeset
162 debug_only( _look_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
163 key = (key + stride) & (_max-1); // Stride through table w/ relative prime
a61af66fc99e Initial load
duke
parents:
diff changeset
164 k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
165 if( !k ) { // ?Miss?
a61af66fc99e Initial load
duke
parents:
diff changeset
166 debug_only( _lookup_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
167 key = (first_sentinel == 0) ? key : first_sentinel; // ?saw sentinel?
a61af66fc99e Initial load
duke
parents:
diff changeset
168 _table[key] = n; // Insert into table!
a61af66fc99e Initial load
duke
parents:
diff changeset
169 debug_only(n->enter_hash_lock()); // Lock down the node while in the table.
a61af66fc99e Initial load
duke
parents:
diff changeset
170 check_grow(); // Grow table if insert hit limit
a61af66fc99e Initial load
duke
parents:
diff changeset
171 return NULL; // Miss!
a61af66fc99e Initial load
duke
parents:
diff changeset
172 }
a61af66fc99e Initial load
duke
parents:
diff changeset
173 else if( first_sentinel == 0 && k == _sentinel ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
174 first_sentinel = key; // Can insert here
a61af66fc99e Initial load
duke
parents:
diff changeset
175 }
a61af66fc99e Initial load
duke
parents:
diff changeset
176
a61af66fc99e Initial load
duke
parents:
diff changeset
177 }
a61af66fc99e Initial load
duke
parents:
diff changeset
178 ShouldNotReachHere();
a61af66fc99e Initial load
duke
parents:
diff changeset
179 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
180 }
a61af66fc99e Initial load
duke
parents:
diff changeset
181
a61af66fc99e Initial load
duke
parents:
diff changeset
182 //------------------------------hash_insert------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
183 // Insert into hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
184 void NodeHash::hash_insert( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
185 // // "conflict" comments -- print nodes that conflict
a61af66fc99e Initial load
duke
parents:
diff changeset
186 // bool conflict = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
187 // n->set_hash();
a61af66fc99e Initial load
duke
parents:
diff changeset
188 uint hash = n->hash();
a61af66fc99e Initial load
duke
parents:
diff changeset
189 if (hash == Node::NO_HASH) {
a61af66fc99e Initial load
duke
parents:
diff changeset
190 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
191 }
a61af66fc99e Initial load
duke
parents:
diff changeset
192 check_grow();
a61af66fc99e Initial load
duke
parents:
diff changeset
193 uint key = hash & (_max-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
194 uint stride = key | 0x01;
a61af66fc99e Initial load
duke
parents:
diff changeset
195
a61af66fc99e Initial load
duke
parents:
diff changeset
196 while( 1 ) { // While probing hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
197 debug_only( _insert_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
198 Node *k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
199 if( !k || (k == _sentinel) ) break; // Found a slot
a61af66fc99e Initial load
duke
parents:
diff changeset
200 assert( k != n, "already inserted" );
a61af66fc99e Initial load
duke
parents:
diff changeset
201 // if( PrintCompilation && PrintOptoStatistics && Verbose ) { tty->print(" conflict: "); k->dump(); conflict = true; }
a61af66fc99e Initial load
duke
parents:
diff changeset
202 key = (key + stride) & (_max-1); // Stride through table w/ relative prime
a61af66fc99e Initial load
duke
parents:
diff changeset
203 }
a61af66fc99e Initial load
duke
parents:
diff changeset
204 _table[key] = n; // Insert into table!
a61af66fc99e Initial load
duke
parents:
diff changeset
205 debug_only(n->enter_hash_lock()); // Lock down the node while in the table.
a61af66fc99e Initial load
duke
parents:
diff changeset
206 // if( conflict ) { n->dump(); }
a61af66fc99e Initial load
duke
parents:
diff changeset
207 }
a61af66fc99e Initial load
duke
parents:
diff changeset
208
a61af66fc99e Initial load
duke
parents:
diff changeset
209 //------------------------------hash_delete------------------------------------
605
98cb887364d3 6810672: Comment typos
twisti
parents: 305
diff changeset
210 // Replace in hash table with sentinel
0
a61af66fc99e Initial load
duke
parents:
diff changeset
211 bool NodeHash::hash_delete( const Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
212 Node *k;
a61af66fc99e Initial load
duke
parents:
diff changeset
213 uint hash = n->hash();
a61af66fc99e Initial load
duke
parents:
diff changeset
214 if (hash == Node::NO_HASH) {
a61af66fc99e Initial load
duke
parents:
diff changeset
215 debug_only( _delete_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
216 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
217 }
a61af66fc99e Initial load
duke
parents:
diff changeset
218 uint key = hash & (_max-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
219 uint stride = key | 0x01;
a61af66fc99e Initial load
duke
parents:
diff changeset
220 debug_only( uint counter = 0; );
605
98cb887364d3 6810672: Comment typos
twisti
parents: 305
diff changeset
221 for( ; /* (k != NULL) && (k != _sentinel) */; ) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
222 debug_only( counter++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
223 debug_only( _delete_probes++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
224 k = _table[key]; // Get hashed value
a61af66fc99e Initial load
duke
parents:
diff changeset
225 if( !k ) { // Miss?
a61af66fc99e Initial load
duke
parents:
diff changeset
226 debug_only( _delete_misses++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
227 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
228 if( VerifyOpto ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
229 for( uint i=0; i < _max; i++ )
a61af66fc99e Initial load
duke
parents:
diff changeset
230 assert( _table[i] != n, "changed edges with rehashing" );
a61af66fc99e Initial load
duke
parents:
diff changeset
231 }
a61af66fc99e Initial load
duke
parents:
diff changeset
232 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
233 return false; // Miss! Not in chain
a61af66fc99e Initial load
duke
parents:
diff changeset
234 }
a61af66fc99e Initial load
duke
parents:
diff changeset
235 else if( n == k ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
236 debug_only( _delete_hits++ );
a61af66fc99e Initial load
duke
parents:
diff changeset
237 _table[key] = _sentinel; // Hit! Label as deleted entry
a61af66fc99e Initial load
duke
parents:
diff changeset
238 debug_only(((Node*)n)->exit_hash_lock()); // Unlock the node upon removal from table.
a61af66fc99e Initial load
duke
parents:
diff changeset
239 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
240 }
a61af66fc99e Initial load
duke
parents:
diff changeset
241 else {
a61af66fc99e Initial load
duke
parents:
diff changeset
242 // collision: move through table with prime offset
a61af66fc99e Initial load
duke
parents:
diff changeset
243 key = (key + stride/*7*/) & (_max-1);
a61af66fc99e Initial load
duke
parents:
diff changeset
244 assert( counter <= _insert_limit, "Cycle in hash-table");
a61af66fc99e Initial load
duke
parents:
diff changeset
245 }
a61af66fc99e Initial load
duke
parents:
diff changeset
246 }
a61af66fc99e Initial load
duke
parents:
diff changeset
247 ShouldNotReachHere();
a61af66fc99e Initial load
duke
parents:
diff changeset
248 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
249 }
a61af66fc99e Initial load
duke
parents:
diff changeset
250
a61af66fc99e Initial load
duke
parents:
diff changeset
251 //------------------------------round_up---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
252 // Round up to nearest power of 2
a61af66fc99e Initial load
duke
parents:
diff changeset
253 uint NodeHash::round_up( uint x ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
254 x += (x>>2); // Add 25% slop
a61af66fc99e Initial load
duke
parents:
diff changeset
255 if( x <16 ) return 16; // Small stuff
a61af66fc99e Initial load
duke
parents:
diff changeset
256 uint i=16;
a61af66fc99e Initial load
duke
parents:
diff changeset
257 while( i < x ) i <<= 1; // Double to fit
a61af66fc99e Initial load
duke
parents:
diff changeset
258 return i; // Return hash table size
a61af66fc99e Initial load
duke
parents:
diff changeset
259 }
a61af66fc99e Initial load
duke
parents:
diff changeset
260
a61af66fc99e Initial load
duke
parents:
diff changeset
261 //------------------------------grow-------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
262 // Grow _table to next power of 2 and insert old entries
a61af66fc99e Initial load
duke
parents:
diff changeset
263 void NodeHash::grow() {
a61af66fc99e Initial load
duke
parents:
diff changeset
264 // Record old state
a61af66fc99e Initial load
duke
parents:
diff changeset
265 uint old_max = _max;
a61af66fc99e Initial load
duke
parents:
diff changeset
266 Node **old_table = _table;
a61af66fc99e Initial load
duke
parents:
diff changeset
267 // Construct new table with twice the space
a61af66fc99e Initial load
duke
parents:
diff changeset
268 _grows++;
a61af66fc99e Initial load
duke
parents:
diff changeset
269 _total_inserts += _inserts;
a61af66fc99e Initial load
duke
parents:
diff changeset
270 _total_insert_probes += _insert_probes;
a61af66fc99e Initial load
duke
parents:
diff changeset
271 _inserts = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
272 _insert_probes = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
273 _max = _max << 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
274 _table = NEW_ARENA_ARRAY( _a , Node* , _max ); // (Node**)_a->Amalloc( _max * sizeof(Node*) );
a61af66fc99e Initial load
duke
parents:
diff changeset
275 memset(_table,0,sizeof(Node*)*_max);
a61af66fc99e Initial load
duke
parents:
diff changeset
276 _insert_limit = insert_limit();
a61af66fc99e Initial load
duke
parents:
diff changeset
277 // Insert old entries into the new table
a61af66fc99e Initial load
duke
parents:
diff changeset
278 for( uint i = 0; i < old_max; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
279 Node *m = *old_table++;
a61af66fc99e Initial load
duke
parents:
diff changeset
280 if( !m || m == _sentinel ) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
281 debug_only(m->exit_hash_lock()); // Unlock the node upon removal from old table.
a61af66fc99e Initial load
duke
parents:
diff changeset
282 hash_insert(m);
a61af66fc99e Initial load
duke
parents:
diff changeset
283 }
a61af66fc99e Initial load
duke
parents:
diff changeset
284 }
a61af66fc99e Initial load
duke
parents:
diff changeset
285
a61af66fc99e Initial load
duke
parents:
diff changeset
286 //------------------------------clear------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
287 // Clear all entries in _table to NULL but keep storage
a61af66fc99e Initial load
duke
parents:
diff changeset
288 void NodeHash::clear() {
a61af66fc99e Initial load
duke
parents:
diff changeset
289 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
290 // Unlock all nodes upon removal from table.
a61af66fc99e Initial load
duke
parents:
diff changeset
291 for (uint i = 0; i < _max; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
292 Node* n = _table[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
293 if (!n || n == _sentinel) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
294 n->exit_hash_lock();
a61af66fc99e Initial load
duke
parents:
diff changeset
295 }
a61af66fc99e Initial load
duke
parents:
diff changeset
296 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
297
a61af66fc99e Initial load
duke
parents:
diff changeset
298 memset( _table, 0, _max * sizeof(Node*) );
a61af66fc99e Initial load
duke
parents:
diff changeset
299 }
a61af66fc99e Initial load
duke
parents:
diff changeset
300
a61af66fc99e Initial load
duke
parents:
diff changeset
301 //-----------------------remove_useless_nodes----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
302 // Remove useless nodes from value table,
a61af66fc99e Initial load
duke
parents:
diff changeset
303 // implementation does not depend on hash function
a61af66fc99e Initial load
duke
parents:
diff changeset
304 void NodeHash::remove_useless_nodes(VectorSet &useful) {
a61af66fc99e Initial load
duke
parents:
diff changeset
305
a61af66fc99e Initial load
duke
parents:
diff changeset
306 // Dead nodes in the hash table inherited from GVN should not replace
a61af66fc99e Initial load
duke
parents:
diff changeset
307 // existing nodes, remove dead nodes.
a61af66fc99e Initial load
duke
parents:
diff changeset
308 uint max = size();
a61af66fc99e Initial load
duke
parents:
diff changeset
309 Node *sentinel_node = sentinel();
a61af66fc99e Initial load
duke
parents:
diff changeset
310 for( uint i = 0; i < max; ++i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
311 Node *n = at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
312 if(n != NULL && n != sentinel_node && !useful.test(n->_idx)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
313 debug_only(n->exit_hash_lock()); // Unlock the node when removed
a61af66fc99e Initial load
duke
parents:
diff changeset
314 _table[i] = sentinel_node; // Replace with placeholder
a61af66fc99e Initial load
duke
parents:
diff changeset
315 }
a61af66fc99e Initial load
duke
parents:
diff changeset
316 }
a61af66fc99e Initial load
duke
parents:
diff changeset
317 }
a61af66fc99e Initial load
duke
parents:
diff changeset
318
a61af66fc99e Initial load
duke
parents:
diff changeset
319 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
320 //------------------------------dump-------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
321 // Dump statistics for the hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
322 void NodeHash::dump() {
a61af66fc99e Initial load
duke
parents:
diff changeset
323 _total_inserts += _inserts;
a61af66fc99e Initial load
duke
parents:
diff changeset
324 _total_insert_probes += _insert_probes;
a61af66fc99e Initial load
duke
parents:
diff changeset
325 if( PrintCompilation && PrintOptoStatistics && Verbose && (_inserts > 0) ) { // PrintOptoGVN
a61af66fc99e Initial load
duke
parents:
diff changeset
326 if( PrintCompilation2 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
327 for( uint i=0; i<_max; i++ )
a61af66fc99e Initial load
duke
parents:
diff changeset
328 if( _table[i] )
a61af66fc99e Initial load
duke
parents:
diff changeset
329 tty->print("%d/%d/%d ",i,_table[i]->hash()&(_max-1),_table[i]->_idx);
a61af66fc99e Initial load
duke
parents:
diff changeset
330 }
a61af66fc99e Initial load
duke
parents:
diff changeset
331 tty->print("\nGVN Hash stats: %d grows to %d max_size\n", _grows, _max);
a61af66fc99e Initial load
duke
parents:
diff changeset
332 tty->print(" %d/%d (%8.1f%% full)\n", _inserts, _max, (double)_inserts/_max*100.0);
a61af66fc99e Initial load
duke
parents:
diff changeset
333 tty->print(" %dp/(%dh+%dm) (%8.2f probes/lookup)\n", _look_probes, _lookup_hits, _lookup_misses, (double)_look_probes/(_lookup_hits+_lookup_misses));
a61af66fc99e Initial load
duke
parents:
diff changeset
334 tty->print(" %dp/%di (%8.2f probes/insert)\n", _total_insert_probes, _total_inserts, (double)_total_insert_probes/_total_inserts);
a61af66fc99e Initial load
duke
parents:
diff changeset
335 // sentinels increase lookup cost, but not insert cost
a61af66fc99e Initial load
duke
parents:
diff changeset
336 assert((_lookup_misses+_lookup_hits)*4+100 >= _look_probes, "bad hash function");
a61af66fc99e Initial load
duke
parents:
diff changeset
337 assert( _inserts+(_inserts>>3) < _max, "table too full" );
a61af66fc99e Initial load
duke
parents:
diff changeset
338 assert( _inserts*3+100 >= _insert_probes, "bad hash function" );
a61af66fc99e Initial load
duke
parents:
diff changeset
339 }
a61af66fc99e Initial load
duke
parents:
diff changeset
340 }
a61af66fc99e Initial load
duke
parents:
diff changeset
341
a61af66fc99e Initial load
duke
parents:
diff changeset
342 Node *NodeHash::find_index(uint idx) { // For debugging
a61af66fc99e Initial load
duke
parents:
diff changeset
343 // Find an entry by its index value
a61af66fc99e Initial load
duke
parents:
diff changeset
344 for( uint i = 0; i < _max; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
345 Node *m = _table[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
346 if( !m || m == _sentinel ) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
347 if( m->_idx == (uint)idx ) return m;
a61af66fc99e Initial load
duke
parents:
diff changeset
348 }
a61af66fc99e Initial load
duke
parents:
diff changeset
349 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
350 }
a61af66fc99e Initial load
duke
parents:
diff changeset
351 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
352
a61af66fc99e Initial load
duke
parents:
diff changeset
353 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
354 NodeHash::~NodeHash() {
a61af66fc99e Initial load
duke
parents:
diff changeset
355 // Unlock all nodes upon destruction of table.
a61af66fc99e Initial load
duke
parents:
diff changeset
356 if (_table != (Node**)badAddress) clear();
a61af66fc99e Initial load
duke
parents:
diff changeset
357 }
a61af66fc99e Initial load
duke
parents:
diff changeset
358
a61af66fc99e Initial load
duke
parents:
diff changeset
359 void NodeHash::operator=(const NodeHash& nh) {
a61af66fc99e Initial load
duke
parents:
diff changeset
360 // Unlock all nodes upon replacement of table.
a61af66fc99e Initial load
duke
parents:
diff changeset
361 if (&nh == this) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
362 if (_table != (Node**)badAddress) clear();
a61af66fc99e Initial load
duke
parents:
diff changeset
363 memcpy(this, &nh, sizeof(*this));
a61af66fc99e Initial load
duke
parents:
diff changeset
364 // Do not increment hash_lock counts again.
a61af66fc99e Initial load
duke
parents:
diff changeset
365 // Instead, be sure we never again use the source table.
a61af66fc99e Initial load
duke
parents:
diff changeset
366 ((NodeHash*)&nh)->_table = (Node**)badAddress;
a61af66fc99e Initial load
duke
parents:
diff changeset
367 }
a61af66fc99e Initial load
duke
parents:
diff changeset
368
a61af66fc99e Initial load
duke
parents:
diff changeset
369
a61af66fc99e Initial load
duke
parents:
diff changeset
370 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
371
a61af66fc99e Initial load
duke
parents:
diff changeset
372
a61af66fc99e Initial load
duke
parents:
diff changeset
373 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
374 //------------------------------PhaseRemoveUseless-----------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
375 // 1) Use a breadthfirst walk to collect useful nodes reachable from root.
a61af66fc99e Initial load
duke
parents:
diff changeset
376 PhaseRemoveUseless::PhaseRemoveUseless( PhaseGVN *gvn, Unique_Node_List *worklist ) : Phase(Remove_Useless),
a61af66fc99e Initial load
duke
parents:
diff changeset
377 _useful(Thread::current()->resource_area()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
378
a61af66fc99e Initial load
duke
parents:
diff changeset
379 // Implementation requires 'UseLoopSafepoints == true' and an edge from root
a61af66fc99e Initial load
duke
parents:
diff changeset
380 // to each SafePointNode at a backward branch. Inserted in add_safepoint().
a61af66fc99e Initial load
duke
parents:
diff changeset
381 if( !UseLoopSafepoints || !OptoRemoveUseless ) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
382
a61af66fc99e Initial load
duke
parents:
diff changeset
383 // Identify nodes that are reachable from below, useful.
a61af66fc99e Initial load
duke
parents:
diff changeset
384 C->identify_useful_nodes(_useful);
a61af66fc99e Initial load
duke
parents:
diff changeset
385
a61af66fc99e Initial load
duke
parents:
diff changeset
386 // Remove all useless nodes from PhaseValues' recorded types
a61af66fc99e Initial load
duke
parents:
diff changeset
387 // Must be done before disconnecting nodes to preserve hash-table-invariant
a61af66fc99e Initial load
duke
parents:
diff changeset
388 gvn->remove_useless_nodes(_useful.member_set());
a61af66fc99e Initial load
duke
parents:
diff changeset
389
a61af66fc99e Initial load
duke
parents:
diff changeset
390 // Remove all useless nodes from future worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
391 worklist->remove_useless_nodes(_useful.member_set());
a61af66fc99e Initial load
duke
parents:
diff changeset
392
a61af66fc99e Initial load
duke
parents:
diff changeset
393 // Disconnect 'useless' nodes that are adjacent to useful nodes
a61af66fc99e Initial load
duke
parents:
diff changeset
394 C->remove_useless_nodes(_useful);
a61af66fc99e Initial load
duke
parents:
diff changeset
395
a61af66fc99e Initial load
duke
parents:
diff changeset
396 // Remove edges from "root" to each SafePoint at a backward branch.
a61af66fc99e Initial load
duke
parents:
diff changeset
397 // They were inserted during parsing (see add_safepoint()) to make infinite
a61af66fc99e Initial load
duke
parents:
diff changeset
398 // loops without calls or exceptions visible to root, i.e., useful.
a61af66fc99e Initial load
duke
parents:
diff changeset
399 Node *root = C->root();
a61af66fc99e Initial load
duke
parents:
diff changeset
400 if( root != NULL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
401 for( uint i = root->req(); i < root->len(); ++i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
402 Node *n = root->in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
403 if( n != NULL && n->is_SafePoint() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
404 root->rm_prec(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
405 --i;
a61af66fc99e Initial load
duke
parents:
diff changeset
406 }
a61af66fc99e Initial load
duke
parents:
diff changeset
407 }
a61af66fc99e Initial load
duke
parents:
diff changeset
408 }
a61af66fc99e Initial load
duke
parents:
diff changeset
409 }
a61af66fc99e Initial load
duke
parents:
diff changeset
410
a61af66fc99e Initial load
duke
parents:
diff changeset
411
a61af66fc99e Initial load
duke
parents:
diff changeset
412 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
413 //------------------------------PhaseTransform---------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
414 PhaseTransform::PhaseTransform( PhaseNumber pnum ) : Phase(pnum),
a61af66fc99e Initial load
duke
parents:
diff changeset
415 _arena(Thread::current()->resource_area()),
a61af66fc99e Initial load
duke
parents:
diff changeset
416 _nodes(_arena),
a61af66fc99e Initial load
duke
parents:
diff changeset
417 _types(_arena)
a61af66fc99e Initial load
duke
parents:
diff changeset
418 {
a61af66fc99e Initial load
duke
parents:
diff changeset
419 init_con_caches();
a61af66fc99e Initial load
duke
parents:
diff changeset
420 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
421 clear_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
422 clear_transforms();
a61af66fc99e Initial load
duke
parents:
diff changeset
423 set_allow_progress(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
424 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
425 // Force allocation for currently existing nodes
a61af66fc99e Initial load
duke
parents:
diff changeset
426 _types.map(C->unique(), NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
427 }
a61af66fc99e Initial load
duke
parents:
diff changeset
428
a61af66fc99e Initial load
duke
parents:
diff changeset
429 //------------------------------PhaseTransform---------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
430 PhaseTransform::PhaseTransform( Arena *arena, PhaseNumber pnum ) : Phase(pnum),
a61af66fc99e Initial load
duke
parents:
diff changeset
431 _arena(arena),
a61af66fc99e Initial load
duke
parents:
diff changeset
432 _nodes(arena),
a61af66fc99e Initial load
duke
parents:
diff changeset
433 _types(arena)
a61af66fc99e Initial load
duke
parents:
diff changeset
434 {
a61af66fc99e Initial load
duke
parents:
diff changeset
435 init_con_caches();
a61af66fc99e Initial load
duke
parents:
diff changeset
436 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
437 clear_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
438 clear_transforms();
a61af66fc99e Initial load
duke
parents:
diff changeset
439 set_allow_progress(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
440 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
441 // Force allocation for currently existing nodes
a61af66fc99e Initial load
duke
parents:
diff changeset
442 _types.map(C->unique(), NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
443 }
a61af66fc99e Initial load
duke
parents:
diff changeset
444
a61af66fc99e Initial load
duke
parents:
diff changeset
445 //------------------------------PhaseTransform---------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
446 // Initialize with previously generated type information
a61af66fc99e Initial load
duke
parents:
diff changeset
447 PhaseTransform::PhaseTransform( PhaseTransform *pt, PhaseNumber pnum ) : Phase(pnum),
a61af66fc99e Initial load
duke
parents:
diff changeset
448 _arena(pt->_arena),
a61af66fc99e Initial load
duke
parents:
diff changeset
449 _nodes(pt->_nodes),
a61af66fc99e Initial load
duke
parents:
diff changeset
450 _types(pt->_types)
a61af66fc99e Initial load
duke
parents:
diff changeset
451 {
a61af66fc99e Initial load
duke
parents:
diff changeset
452 init_con_caches();
a61af66fc99e Initial load
duke
parents:
diff changeset
453 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
454 clear_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
455 clear_transforms();
a61af66fc99e Initial load
duke
parents:
diff changeset
456 set_allow_progress(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
457 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
458 }
a61af66fc99e Initial load
duke
parents:
diff changeset
459
a61af66fc99e Initial load
duke
parents:
diff changeset
460 void PhaseTransform::init_con_caches() {
a61af66fc99e Initial load
duke
parents:
diff changeset
461 memset(_icons,0,sizeof(_icons));
a61af66fc99e Initial load
duke
parents:
diff changeset
462 memset(_lcons,0,sizeof(_lcons));
a61af66fc99e Initial load
duke
parents:
diff changeset
463 memset(_zcons,0,sizeof(_zcons));
a61af66fc99e Initial load
duke
parents:
diff changeset
464 }
a61af66fc99e Initial load
duke
parents:
diff changeset
465
a61af66fc99e Initial load
duke
parents:
diff changeset
466
a61af66fc99e Initial load
duke
parents:
diff changeset
467 //--------------------------------find_int_type--------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
468 const TypeInt* PhaseTransform::find_int_type(Node* n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
469 if (n == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
470 // Call type_or_null(n) to determine node's type since we might be in
a61af66fc99e Initial load
duke
parents:
diff changeset
471 // parse phase and call n->Value() may return wrong type.
a61af66fc99e Initial load
duke
parents:
diff changeset
472 // (For example, a phi node at the beginning of loop parsing is not ready.)
a61af66fc99e Initial load
duke
parents:
diff changeset
473 const Type* t = type_or_null(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
474 if (t == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
475 return t->isa_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
476 }
a61af66fc99e Initial load
duke
parents:
diff changeset
477
a61af66fc99e Initial load
duke
parents:
diff changeset
478
a61af66fc99e Initial load
duke
parents:
diff changeset
479 //-------------------------------find_long_type--------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
480 const TypeLong* PhaseTransform::find_long_type(Node* n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
481 if (n == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
482 // (See comment above on type_or_null.)
a61af66fc99e Initial load
duke
parents:
diff changeset
483 const Type* t = type_or_null(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
484 if (t == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
485 return t->isa_long();
a61af66fc99e Initial load
duke
parents:
diff changeset
486 }
a61af66fc99e Initial load
duke
parents:
diff changeset
487
a61af66fc99e Initial load
duke
parents:
diff changeset
488
a61af66fc99e Initial load
duke
parents:
diff changeset
489 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
490 void PhaseTransform::dump_old2new_map() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
491 _nodes.dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
492 }
a61af66fc99e Initial load
duke
parents:
diff changeset
493
a61af66fc99e Initial load
duke
parents:
diff changeset
494 void PhaseTransform::dump_new( uint nidx ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
495 for( uint i=0; i<_nodes.Size(); i++ )
a61af66fc99e Initial load
duke
parents:
diff changeset
496 if( _nodes[i] && _nodes[i]->_idx == nidx ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
497 _nodes[i]->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
498 tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
499 tty->print_cr("Old index= %d",i);
a61af66fc99e Initial load
duke
parents:
diff changeset
500 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
501 }
a61af66fc99e Initial load
duke
parents:
diff changeset
502 tty->print_cr("Node %d not found in the new indices", nidx);
a61af66fc99e Initial load
duke
parents:
diff changeset
503 }
a61af66fc99e Initial load
duke
parents:
diff changeset
504
a61af66fc99e Initial load
duke
parents:
diff changeset
505 //------------------------------dump_types-------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
506 void PhaseTransform::dump_types( ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
507 _types.dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
508 }
a61af66fc99e Initial load
duke
parents:
diff changeset
509
a61af66fc99e Initial load
duke
parents:
diff changeset
510 //------------------------------dump_nodes_and_types---------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
511 void PhaseTransform::dump_nodes_and_types(const Node *root, uint depth, bool only_ctrl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
512 VectorSet visited(Thread::current()->resource_area());
a61af66fc99e Initial load
duke
parents:
diff changeset
513 dump_nodes_and_types_recur( root, depth, only_ctrl, visited );
a61af66fc99e Initial load
duke
parents:
diff changeset
514 }
a61af66fc99e Initial load
duke
parents:
diff changeset
515
a61af66fc99e Initial load
duke
parents:
diff changeset
516 //------------------------------dump_nodes_and_types_recur---------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
517 void PhaseTransform::dump_nodes_and_types_recur( const Node *n, uint depth, bool only_ctrl, VectorSet &visited) {
a61af66fc99e Initial load
duke
parents:
diff changeset
518 if( !n ) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
519 if( depth == 0 ) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
520 if( visited.test_set(n->_idx) ) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
521 for( uint i=0; i<n->len(); i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
522 if( only_ctrl && !(n->is_Region()) && i != TypeFunc::Control ) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
523 dump_nodes_and_types_recur( n->in(i), depth-1, only_ctrl, visited );
a61af66fc99e Initial load
duke
parents:
diff changeset
524 }
a61af66fc99e Initial load
duke
parents:
diff changeset
525 n->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
526 if (type_or_null(n) != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
527 tty->print(" "); type(n)->dump(); tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
528 }
a61af66fc99e Initial load
duke
parents:
diff changeset
529 }
a61af66fc99e Initial load
duke
parents:
diff changeset
530
a61af66fc99e Initial load
duke
parents:
diff changeset
531 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
532
a61af66fc99e Initial load
duke
parents:
diff changeset
533
a61af66fc99e Initial load
duke
parents:
diff changeset
534 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
535 //------------------------------PhaseValues------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
536 // Set minimum table size to "255"
a61af66fc99e Initial load
duke
parents:
diff changeset
537 PhaseValues::PhaseValues( Arena *arena, uint est_max_size ) : PhaseTransform(arena, GVN), _table(arena, est_max_size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
538 NOT_PRODUCT( clear_new_values(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
539 }
a61af66fc99e Initial load
duke
parents:
diff changeset
540
a61af66fc99e Initial load
duke
parents:
diff changeset
541 //------------------------------PhaseValues------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
542 // Set minimum table size to "255"
a61af66fc99e Initial load
duke
parents:
diff changeset
543 PhaseValues::PhaseValues( PhaseValues *ptv ) : PhaseTransform( ptv, GVN ),
a61af66fc99e Initial load
duke
parents:
diff changeset
544 _table(&ptv->_table) {
a61af66fc99e Initial load
duke
parents:
diff changeset
545 NOT_PRODUCT( clear_new_values(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
546 }
a61af66fc99e Initial load
duke
parents:
diff changeset
547
a61af66fc99e Initial load
duke
parents:
diff changeset
548 //------------------------------PhaseValues------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
549 // Used by +VerifyOpto. Clear out hash table but copy _types array.
a61af66fc99e Initial load
duke
parents:
diff changeset
550 PhaseValues::PhaseValues( PhaseValues *ptv, const char *dummy ) : PhaseTransform( ptv, GVN ),
a61af66fc99e Initial load
duke
parents:
diff changeset
551 _table(ptv->arena(),ptv->_table.size()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
552 NOT_PRODUCT( clear_new_values(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
553 }
a61af66fc99e Initial load
duke
parents:
diff changeset
554
a61af66fc99e Initial load
duke
parents:
diff changeset
555 //------------------------------~PhaseValues-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
556 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
557 PhaseValues::~PhaseValues() {
a61af66fc99e Initial load
duke
parents:
diff changeset
558 _table.dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
559
a61af66fc99e Initial load
duke
parents:
diff changeset
560 // Statistics for value progress and efficiency
a61af66fc99e Initial load
duke
parents:
diff changeset
561 if( PrintCompilation && Verbose && WizardMode ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
562 tty->print("\n%sValues: %d nodes ---> %d/%d (%d)",
a61af66fc99e Initial load
duke
parents:
diff changeset
563 is_IterGVN() ? "Iter" : " ", C->unique(), made_progress(), made_transforms(), made_new_values());
a61af66fc99e Initial load
duke
parents:
diff changeset
564 if( made_transforms() != 0 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
565 tty->print_cr(" ratio %f", made_progress()/(float)made_transforms() );
a61af66fc99e Initial load
duke
parents:
diff changeset
566 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
567 tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
568 }
a61af66fc99e Initial load
duke
parents:
diff changeset
569 }
a61af66fc99e Initial load
duke
parents:
diff changeset
570 }
a61af66fc99e Initial load
duke
parents:
diff changeset
571 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
572
a61af66fc99e Initial load
duke
parents:
diff changeset
573 //------------------------------makecon----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
574 ConNode* PhaseTransform::makecon(const Type *t) {
a61af66fc99e Initial load
duke
parents:
diff changeset
575 assert(t->singleton(), "must be a constant");
a61af66fc99e Initial load
duke
parents:
diff changeset
576 assert(!t->empty() || t == Type::TOP, "must not be vacuous range");
a61af66fc99e Initial load
duke
parents:
diff changeset
577 switch (t->base()) { // fast paths
a61af66fc99e Initial load
duke
parents:
diff changeset
578 case Type::Half:
a61af66fc99e Initial load
duke
parents:
diff changeset
579 case Type::Top: return (ConNode*) C->top();
a61af66fc99e Initial load
duke
parents:
diff changeset
580 case Type::Int: return intcon( t->is_int()->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
581 case Type::Long: return longcon( t->is_long()->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
582 }
a61af66fc99e Initial load
duke
parents:
diff changeset
583 if (t->is_zero_type())
a61af66fc99e Initial load
duke
parents:
diff changeset
584 return zerocon(t->basic_type());
a61af66fc99e Initial load
duke
parents:
diff changeset
585 return uncached_makecon(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
586 }
a61af66fc99e Initial load
duke
parents:
diff changeset
587
a61af66fc99e Initial load
duke
parents:
diff changeset
588 //--------------------------uncached_makecon-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
589 // Make an idealized constant - one of ConINode, ConPNode, etc.
a61af66fc99e Initial load
duke
parents:
diff changeset
590 ConNode* PhaseValues::uncached_makecon(const Type *t) {
a61af66fc99e Initial load
duke
parents:
diff changeset
591 assert(t->singleton(), "must be a constant");
a61af66fc99e Initial load
duke
parents:
diff changeset
592 ConNode* x = ConNode::make(C, t);
a61af66fc99e Initial load
duke
parents:
diff changeset
593 ConNode* k = (ConNode*)hash_find_insert(x); // Value numbering
a61af66fc99e Initial load
duke
parents:
diff changeset
594 if (k == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
595 set_type(x, t); // Missed, provide type mapping
a61af66fc99e Initial load
duke
parents:
diff changeset
596 GrowableArray<Node_Notes*>* nna = C->node_note_array();
a61af66fc99e Initial load
duke
parents:
diff changeset
597 if (nna != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
598 Node_Notes* loc = C->locate_node_notes(nna, x->_idx, true);
a61af66fc99e Initial load
duke
parents:
diff changeset
599 loc->clear(); // do not put debug info on constants
a61af66fc99e Initial load
duke
parents:
diff changeset
600 }
a61af66fc99e Initial load
duke
parents:
diff changeset
601 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
602 x->destruct(); // Hit, destroy duplicate constant
a61af66fc99e Initial load
duke
parents:
diff changeset
603 x = k; // use existing constant
a61af66fc99e Initial load
duke
parents:
diff changeset
604 }
a61af66fc99e Initial load
duke
parents:
diff changeset
605 return x;
a61af66fc99e Initial load
duke
parents:
diff changeset
606 }
a61af66fc99e Initial load
duke
parents:
diff changeset
607
a61af66fc99e Initial load
duke
parents:
diff changeset
608 //------------------------------intcon-----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
609 // Fast integer constant. Same as "transform(new ConINode(TypeInt::make(i)))"
a61af66fc99e Initial load
duke
parents:
diff changeset
610 ConINode* PhaseTransform::intcon(int i) {
a61af66fc99e Initial load
duke
parents:
diff changeset
611 // Small integer? Check cache! Check that cached node is not dead
a61af66fc99e Initial load
duke
parents:
diff changeset
612 if (i >= _icon_min && i <= _icon_max) {
a61af66fc99e Initial load
duke
parents:
diff changeset
613 ConINode* icon = _icons[i-_icon_min];
a61af66fc99e Initial load
duke
parents:
diff changeset
614 if (icon != NULL && icon->in(TypeFunc::Control) != NULL)
a61af66fc99e Initial load
duke
parents:
diff changeset
615 return icon;
a61af66fc99e Initial load
duke
parents:
diff changeset
616 }
a61af66fc99e Initial load
duke
parents:
diff changeset
617 ConINode* icon = (ConINode*) uncached_makecon(TypeInt::make(i));
a61af66fc99e Initial load
duke
parents:
diff changeset
618 assert(icon->is_Con(), "");
a61af66fc99e Initial load
duke
parents:
diff changeset
619 if (i >= _icon_min && i <= _icon_max)
a61af66fc99e Initial load
duke
parents:
diff changeset
620 _icons[i-_icon_min] = icon; // Cache small integers
a61af66fc99e Initial load
duke
parents:
diff changeset
621 return icon;
a61af66fc99e Initial load
duke
parents:
diff changeset
622 }
a61af66fc99e Initial load
duke
parents:
diff changeset
623
a61af66fc99e Initial load
duke
parents:
diff changeset
624 //------------------------------longcon----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
625 // Fast long constant.
a61af66fc99e Initial load
duke
parents:
diff changeset
626 ConLNode* PhaseTransform::longcon(jlong l) {
a61af66fc99e Initial load
duke
parents:
diff changeset
627 // Small integer? Check cache! Check that cached node is not dead
a61af66fc99e Initial load
duke
parents:
diff changeset
628 if (l >= _lcon_min && l <= _lcon_max) {
a61af66fc99e Initial load
duke
parents:
diff changeset
629 ConLNode* lcon = _lcons[l-_lcon_min];
a61af66fc99e Initial load
duke
parents:
diff changeset
630 if (lcon != NULL && lcon->in(TypeFunc::Control) != NULL)
a61af66fc99e Initial load
duke
parents:
diff changeset
631 return lcon;
a61af66fc99e Initial load
duke
parents:
diff changeset
632 }
a61af66fc99e Initial load
duke
parents:
diff changeset
633 ConLNode* lcon = (ConLNode*) uncached_makecon(TypeLong::make(l));
a61af66fc99e Initial load
duke
parents:
diff changeset
634 assert(lcon->is_Con(), "");
a61af66fc99e Initial load
duke
parents:
diff changeset
635 if (l >= _lcon_min && l <= _lcon_max)
a61af66fc99e Initial load
duke
parents:
diff changeset
636 _lcons[l-_lcon_min] = lcon; // Cache small integers
a61af66fc99e Initial load
duke
parents:
diff changeset
637 return lcon;
a61af66fc99e Initial load
duke
parents:
diff changeset
638 }
a61af66fc99e Initial load
duke
parents:
diff changeset
639
a61af66fc99e Initial load
duke
parents:
diff changeset
640 //------------------------------zerocon-----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
641 // Fast zero or null constant. Same as "transform(ConNode::make(Type::get_zero_type(bt)))"
a61af66fc99e Initial load
duke
parents:
diff changeset
642 ConNode* PhaseTransform::zerocon(BasicType bt) {
a61af66fc99e Initial load
duke
parents:
diff changeset
643 assert((uint)bt <= _zcon_max, "domain check");
a61af66fc99e Initial load
duke
parents:
diff changeset
644 ConNode* zcon = _zcons[bt];
a61af66fc99e Initial load
duke
parents:
diff changeset
645 if (zcon != NULL && zcon->in(TypeFunc::Control) != NULL)
a61af66fc99e Initial load
duke
parents:
diff changeset
646 return zcon;
a61af66fc99e Initial load
duke
parents:
diff changeset
647 zcon = (ConNode*) uncached_makecon(Type::get_zero_type(bt));
a61af66fc99e Initial load
duke
parents:
diff changeset
648 _zcons[bt] = zcon;
a61af66fc99e Initial load
duke
parents:
diff changeset
649 return zcon;
a61af66fc99e Initial load
duke
parents:
diff changeset
650 }
a61af66fc99e Initial load
duke
parents:
diff changeset
651
a61af66fc99e Initial load
duke
parents:
diff changeset
652
a61af66fc99e Initial load
duke
parents:
diff changeset
653
a61af66fc99e Initial load
duke
parents:
diff changeset
654 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
655 //------------------------------transform--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
656 // Return a node which computes the same function as this node, but in a
41
874b2c4f43d1 6667605: (Escape Analysis) inline java constructors when EA is on
kvn
parents: 0
diff changeset
657 // faster or cheaper fashion.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
658 Node *PhaseGVN::transform( Node *n ) {
41
874b2c4f43d1 6667605: (Escape Analysis) inline java constructors when EA is on
kvn
parents: 0
diff changeset
659 return transform_no_reclaim(n);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
660 }
a61af66fc99e Initial load
duke
parents:
diff changeset
661
a61af66fc99e Initial load
duke
parents:
diff changeset
662 //------------------------------transform--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
663 // Return a node which computes the same function as this node, but
a61af66fc99e Initial load
duke
parents:
diff changeset
664 // in a faster or cheaper fashion.
a61af66fc99e Initial load
duke
parents:
diff changeset
665 Node *PhaseGVN::transform_no_reclaim( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
666 NOT_PRODUCT( set_transforms(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
667
a61af66fc99e Initial load
duke
parents:
diff changeset
668 // Apply the Ideal call in a loop until it no longer applies
a61af66fc99e Initial load
duke
parents:
diff changeset
669 Node *k = n;
a61af66fc99e Initial load
duke
parents:
diff changeset
670 NOT_PRODUCT( uint loop_count = 0; )
a61af66fc99e Initial load
duke
parents:
diff changeset
671 while( 1 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
672 Node *i = k->Ideal(this, /*can_reshape=*/false);
a61af66fc99e Initial load
duke
parents:
diff changeset
673 if( !i ) break;
a61af66fc99e Initial load
duke
parents:
diff changeset
674 assert( i->_idx >= k->_idx, "Idealize should return new nodes, use Identity to return old nodes" );
a61af66fc99e Initial load
duke
parents:
diff changeset
675 k = i;
a61af66fc99e Initial load
duke
parents:
diff changeset
676 assert(loop_count++ < K, "infinite loop in PhaseGVN::transform");
a61af66fc99e Initial load
duke
parents:
diff changeset
677 }
a61af66fc99e Initial load
duke
parents:
diff changeset
678 NOT_PRODUCT( if( loop_count != 0 ) { set_progress(); } )
a61af66fc99e Initial load
duke
parents:
diff changeset
679
a61af66fc99e Initial load
duke
parents:
diff changeset
680
a61af66fc99e Initial load
duke
parents:
diff changeset
681 // If brand new node, make space in type array.
a61af66fc99e Initial load
duke
parents:
diff changeset
682 ensure_type_or_null(k);
a61af66fc99e Initial load
duke
parents:
diff changeset
683
a61af66fc99e Initial load
duke
parents:
diff changeset
684 // Since I just called 'Value' to compute the set of run-time values
a61af66fc99e Initial load
duke
parents:
diff changeset
685 // for this Node, and 'Value' is non-local (and therefore expensive) I'll
a61af66fc99e Initial load
duke
parents:
diff changeset
686 // cache Value. Later requests for the local phase->type of this Node can
a61af66fc99e Initial load
duke
parents:
diff changeset
687 // use the cached Value instead of suffering with 'bottom_type'.
a61af66fc99e Initial load
duke
parents:
diff changeset
688 const Type *t = k->Value(this); // Get runtime Value set
a61af66fc99e Initial load
duke
parents:
diff changeset
689 assert(t != NULL, "value sanity");
a61af66fc99e Initial load
duke
parents:
diff changeset
690 if (type_or_null(k) != t) {
a61af66fc99e Initial load
duke
parents:
diff changeset
691 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
692 // Do not count initial visit to node as a transformation
a61af66fc99e Initial load
duke
parents:
diff changeset
693 if (type_or_null(k) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
694 inc_new_values();
a61af66fc99e Initial load
duke
parents:
diff changeset
695 set_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
696 }
a61af66fc99e Initial load
duke
parents:
diff changeset
697 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
698 set_type(k, t);
a61af66fc99e Initial load
duke
parents:
diff changeset
699 // If k is a TypeNode, capture any more-precise type permanently into Node
a61af66fc99e Initial load
duke
parents:
diff changeset
700 k->raise_bottom_type(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
701 }
a61af66fc99e Initial load
duke
parents:
diff changeset
702
a61af66fc99e Initial load
duke
parents:
diff changeset
703 if( t->singleton() && !k->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
704 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
705 return makecon(t); // Turn into a constant
a61af66fc99e Initial load
duke
parents:
diff changeset
706 }
a61af66fc99e Initial load
duke
parents:
diff changeset
707
a61af66fc99e Initial load
duke
parents:
diff changeset
708 // Now check for Identities
a61af66fc99e Initial load
duke
parents:
diff changeset
709 Node *i = k->Identity(this); // Look for a nearby replacement
a61af66fc99e Initial load
duke
parents:
diff changeset
710 if( i != k ) { // Found? Return replacement!
a61af66fc99e Initial load
duke
parents:
diff changeset
711 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
712 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
713 }
a61af66fc99e Initial load
duke
parents:
diff changeset
714
a61af66fc99e Initial load
duke
parents:
diff changeset
715 // Global Value Numbering
a61af66fc99e Initial load
duke
parents:
diff changeset
716 i = hash_find_insert(k); // Insert if new
a61af66fc99e Initial load
duke
parents:
diff changeset
717 if( i && (i != k) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
718 // Return the pre-existing node
a61af66fc99e Initial load
duke
parents:
diff changeset
719 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
720 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
721 }
a61af66fc99e Initial load
duke
parents:
diff changeset
722
a61af66fc99e Initial load
duke
parents:
diff changeset
723 // Return Idealized original
a61af66fc99e Initial load
duke
parents:
diff changeset
724 return k;
a61af66fc99e Initial load
duke
parents:
diff changeset
725 }
a61af66fc99e Initial load
duke
parents:
diff changeset
726
a61af66fc99e Initial load
duke
parents:
diff changeset
727 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
728 //------------------------------dead_loop_check--------------------------------
605
98cb887364d3 6810672: Comment typos
twisti
parents: 305
diff changeset
729 // Check for a simple dead loop when a data node references itself directly
0
a61af66fc99e Initial load
duke
parents:
diff changeset
730 // or through an other data node excluding cons and phis.
a61af66fc99e Initial load
duke
parents:
diff changeset
731 void PhaseGVN::dead_loop_check( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
732 // Phi may reference itself in a loop
a61af66fc99e Initial load
duke
parents:
diff changeset
733 if (n != NULL && !n->is_dead_loop_safe() && !n->is_CFG()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
734 // Do 2 levels check and only data inputs.
a61af66fc99e Initial load
duke
parents:
diff changeset
735 bool no_dead_loop = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
736 uint cnt = n->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
737 for (uint i = 1; i < cnt && no_dead_loop; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
738 Node *in = n->in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
739 if (in == n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
740 no_dead_loop = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
741 } else if (in != NULL && !in->is_dead_loop_safe()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
742 uint icnt = in->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
743 for (uint j = 1; j < icnt && no_dead_loop; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
744 if (in->in(j) == n || in->in(j) == in)
a61af66fc99e Initial load
duke
parents:
diff changeset
745 no_dead_loop = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
746 }
a61af66fc99e Initial load
duke
parents:
diff changeset
747 }
a61af66fc99e Initial load
duke
parents:
diff changeset
748 }
a61af66fc99e Initial load
duke
parents:
diff changeset
749 if (!no_dead_loop) n->dump(3);
a61af66fc99e Initial load
duke
parents:
diff changeset
750 assert(no_dead_loop, "dead loop detected");
a61af66fc99e Initial load
duke
parents:
diff changeset
751 }
a61af66fc99e Initial load
duke
parents:
diff changeset
752 }
a61af66fc99e Initial load
duke
parents:
diff changeset
753 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
754
a61af66fc99e Initial load
duke
parents:
diff changeset
755 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
756 //------------------------------PhaseIterGVN-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
757 // Initialize hash table to fresh and clean for +VerifyOpto
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
758 PhaseIterGVN::PhaseIterGVN( PhaseIterGVN *igvn, const char *dummy ) : PhaseGVN(igvn,dummy), _worklist( ),
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
759 _delay_transform(false) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
760 }
a61af66fc99e Initial load
duke
parents:
diff changeset
761
a61af66fc99e Initial load
duke
parents:
diff changeset
762 //------------------------------PhaseIterGVN-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
763 // Initialize with previous PhaseIterGVN info; used by PhaseCCP
a61af66fc99e Initial load
duke
parents:
diff changeset
764 PhaseIterGVN::PhaseIterGVN( PhaseIterGVN *igvn ) : PhaseGVN(igvn),
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
765 _worklist( igvn->_worklist ),
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
766 _delay_transform(igvn->_delay_transform)
0
a61af66fc99e Initial load
duke
parents:
diff changeset
767 {
a61af66fc99e Initial load
duke
parents:
diff changeset
768 }
a61af66fc99e Initial load
duke
parents:
diff changeset
769
a61af66fc99e Initial load
duke
parents:
diff changeset
770 //------------------------------PhaseIterGVN-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
771 // Initialize with previous PhaseGVN info from Parser
a61af66fc99e Initial load
duke
parents:
diff changeset
772 PhaseIterGVN::PhaseIterGVN( PhaseGVN *gvn ) : PhaseGVN(gvn),
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
773 _worklist(*C->for_igvn()),
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
774 _delay_transform(false)
0
a61af66fc99e Initial load
duke
parents:
diff changeset
775 {
a61af66fc99e Initial load
duke
parents:
diff changeset
776 uint max;
a61af66fc99e Initial load
duke
parents:
diff changeset
777
a61af66fc99e Initial load
duke
parents:
diff changeset
778 // Dead nodes in the hash table inherited from GVN were not treated as
a61af66fc99e Initial load
duke
parents:
diff changeset
779 // roots during def-use info creation; hence they represent an invisible
a61af66fc99e Initial load
duke
parents:
diff changeset
780 // use. Clear them out.
a61af66fc99e Initial load
duke
parents:
diff changeset
781 max = _table.size();
a61af66fc99e Initial load
duke
parents:
diff changeset
782 for( uint i = 0; i < max; ++i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
783 Node *n = _table.at(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
784 if(n != NULL && n != _table.sentinel() && n->outcnt() == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
785 if( n->is_top() ) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
786 assert( false, "Parse::remove_useless_nodes missed this node");
a61af66fc99e Initial load
duke
parents:
diff changeset
787 hash_delete(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
788 }
a61af66fc99e Initial load
duke
parents:
diff changeset
789 }
a61af66fc99e Initial load
duke
parents:
diff changeset
790
a61af66fc99e Initial load
duke
parents:
diff changeset
791 // Any Phis or Regions on the worklist probably had uses that could not
a61af66fc99e Initial load
duke
parents:
diff changeset
792 // make more progress because the uses were made while the Phis and Regions
a61af66fc99e Initial load
duke
parents:
diff changeset
793 // were in half-built states. Put all uses of Phis and Regions on worklist.
a61af66fc99e Initial load
duke
parents:
diff changeset
794 max = _worklist.size();
a61af66fc99e Initial load
duke
parents:
diff changeset
795 for( uint j = 0; j < max; j++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
796 Node *n = _worklist.at(j);
a61af66fc99e Initial load
duke
parents:
diff changeset
797 uint uop = n->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
798 if( uop == Op_Phi || uop == Op_Region ||
a61af66fc99e Initial load
duke
parents:
diff changeset
799 n->is_Type() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
800 n->is_Mem() )
a61af66fc99e Initial load
duke
parents:
diff changeset
801 add_users_to_worklist(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
802 }
a61af66fc99e Initial load
duke
parents:
diff changeset
803 }
a61af66fc99e Initial load
duke
parents:
diff changeset
804
a61af66fc99e Initial load
duke
parents:
diff changeset
805
a61af66fc99e Initial load
duke
parents:
diff changeset
806 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
807 void PhaseIterGVN::verify_step(Node* n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
808 _verify_window[_verify_counter % _verify_window_size] = n;
a61af66fc99e Initial load
duke
parents:
diff changeset
809 ++_verify_counter;
a61af66fc99e Initial load
duke
parents:
diff changeset
810 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
811 ResourceArea *area = Thread::current()->resource_area();
a61af66fc99e Initial load
duke
parents:
diff changeset
812 VectorSet old_space(area), new_space(area);
a61af66fc99e Initial load
duke
parents:
diff changeset
813 if (C->unique() < 1000 ||
a61af66fc99e Initial load
duke
parents:
diff changeset
814 0 == _verify_counter % (C->unique() < 10000 ? 10 : 100)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
815 ++_verify_full_passes;
a61af66fc99e Initial load
duke
parents:
diff changeset
816 Node::verify_recur(C->root(), -1, old_space, new_space);
a61af66fc99e Initial load
duke
parents:
diff changeset
817 }
a61af66fc99e Initial load
duke
parents:
diff changeset
818 const int verify_depth = 4;
a61af66fc99e Initial load
duke
parents:
diff changeset
819 for ( int i = 0; i < _verify_window_size; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
820 Node* n = _verify_window[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
821 if ( n == NULL ) continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
822 if( n->in(0) == NodeSentinel ) { // xform_idom
a61af66fc99e Initial load
duke
parents:
diff changeset
823 _verify_window[i] = n->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
824 --i; continue;
a61af66fc99e Initial load
duke
parents:
diff changeset
825 }
a61af66fc99e Initial load
duke
parents:
diff changeset
826 // Typical fanout is 1-2, so this call visits about 6 nodes.
a61af66fc99e Initial load
duke
parents:
diff changeset
827 Node::verify_recur(n, verify_depth, old_space, new_space);
a61af66fc99e Initial load
duke
parents:
diff changeset
828 }
a61af66fc99e Initial load
duke
parents:
diff changeset
829 }
a61af66fc99e Initial load
duke
parents:
diff changeset
830 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
831
a61af66fc99e Initial load
duke
parents:
diff changeset
832
a61af66fc99e Initial load
duke
parents:
diff changeset
833 //------------------------------init_worklist----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
834 // Initialize worklist for each node.
a61af66fc99e Initial load
duke
parents:
diff changeset
835 void PhaseIterGVN::init_worklist( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
836 if( _worklist.member(n) ) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
837 _worklist.push(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
838 uint cnt = n->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
839 for( uint i =0 ; i < cnt; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
840 Node *m = n->in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
841 if( m ) init_worklist(m);
a61af66fc99e Initial load
duke
parents:
diff changeset
842 }
a61af66fc99e Initial load
duke
parents:
diff changeset
843 }
a61af66fc99e Initial load
duke
parents:
diff changeset
844
a61af66fc99e Initial load
duke
parents:
diff changeset
845 //------------------------------optimize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
846 void PhaseIterGVN::optimize() {
a61af66fc99e Initial load
duke
parents:
diff changeset
847 debug_only(uint num_processed = 0;);
a61af66fc99e Initial load
duke
parents:
diff changeset
848 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
849 {
a61af66fc99e Initial load
duke
parents:
diff changeset
850 _verify_counter = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
851 _verify_full_passes = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
852 for ( int i = 0; i < _verify_window_size; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
853 _verify_window[i] = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
854 }
a61af66fc99e Initial load
duke
parents:
diff changeset
855 }
a61af66fc99e Initial load
duke
parents:
diff changeset
856 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
857
1826
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
858 #ifdef ASSERT
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
859 Node* prev = NULL;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
860 uint rep_cnt = 0;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
861 #endif
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
862 uint loop_count = 0;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
863
0
a61af66fc99e Initial load
duke
parents:
diff changeset
864 // Pull from worklist; transform node;
a61af66fc99e Initial load
duke
parents:
diff changeset
865 // If node has changed: update edge info and put uses on worklist.
a61af66fc99e Initial load
duke
parents:
diff changeset
866 while( _worklist.size() ) {
3958
075ea0ed9e7c 7081842: assert(Compile::current()->unique() < (uint)MaxNodeLimit) failed: Node limit exceeded
kvn
parents: 1972
diff changeset
867 if (C->check_node_count(NodeLimitFudgeFactor * 2,
075ea0ed9e7c 7081842: assert(Compile::current()->unique() < (uint)MaxNodeLimit) failed: Node limit exceeded
kvn
parents: 1972
diff changeset
868 "out of nodes optimizing method")) {
075ea0ed9e7c 7081842: assert(Compile::current()->unique() < (uint)MaxNodeLimit) failed: Node limit exceeded
kvn
parents: 1972
diff changeset
869 return;
075ea0ed9e7c 7081842: assert(Compile::current()->unique() < (uint)MaxNodeLimit) failed: Node limit exceeded
kvn
parents: 1972
diff changeset
870 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
871 Node *n = _worklist.pop();
1826
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
872 if (++loop_count >= K * C->unique()) {
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
873 debug_only(n->dump(4);)
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
874 assert(false, "infinite loop in PhaseIterGVN::optimize");
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
875 C->record_method_not_compilable("infinite loop in PhaseIterGVN::optimize");
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
876 return;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
877 }
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
878 #ifdef ASSERT
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
879 if (n == prev) {
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
880 if (++rep_cnt > 3) {
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
881 n->dump(4);
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
882 assert(false, "loop in Ideal transformation");
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
883 }
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
884 } else {
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
885 rep_cnt = 0;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
886 }
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
887 prev = n;
56601ef83436 6916062: assert(_inserts <= _insert_limit,"hash table overflow") in NodeHash::hash_insert
kvn
parents: 1621
diff changeset
888 #endif
0
a61af66fc99e Initial load
duke
parents:
diff changeset
889 if (TraceIterativeGVN && Verbose) {
a61af66fc99e Initial load
duke
parents:
diff changeset
890 tty->print(" Pop ");
a61af66fc99e Initial load
duke
parents:
diff changeset
891 NOT_PRODUCT( n->dump(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
892 debug_only(if( (num_processed++ % 100) == 0 ) _worklist.print_set();)
a61af66fc99e Initial load
duke
parents:
diff changeset
893 }
a61af66fc99e Initial load
duke
parents:
diff changeset
894
a61af66fc99e Initial load
duke
parents:
diff changeset
895 if (n->outcnt() != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
896
a61af66fc99e Initial load
duke
parents:
diff changeset
897 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
898 uint wlsize = _worklist.size();
a61af66fc99e Initial load
duke
parents:
diff changeset
899 const Type* oldtype = type_or_null(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
900 #endif //PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
901
a61af66fc99e Initial load
duke
parents:
diff changeset
902 Node *nn = transform_old(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
903
a61af66fc99e Initial load
duke
parents:
diff changeset
904 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
905 if (TraceIterativeGVN) {
a61af66fc99e Initial load
duke
parents:
diff changeset
906 const Type* newtype = type_or_null(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
907 if (nn != n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
908 // print old node
a61af66fc99e Initial load
duke
parents:
diff changeset
909 tty->print("< ");
a61af66fc99e Initial load
duke
parents:
diff changeset
910 if (oldtype != newtype && oldtype != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
911 oldtype->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
912 }
a61af66fc99e Initial load
duke
parents:
diff changeset
913 do { tty->print("\t"); } while (tty->position() < 16);
a61af66fc99e Initial load
duke
parents:
diff changeset
914 tty->print("<");
a61af66fc99e Initial load
duke
parents:
diff changeset
915 n->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
916 }
a61af66fc99e Initial load
duke
parents:
diff changeset
917 if (oldtype != newtype || nn != n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
918 // print new node and/or new type
a61af66fc99e Initial load
duke
parents:
diff changeset
919 if (oldtype == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
920 tty->print("* ");
a61af66fc99e Initial load
duke
parents:
diff changeset
921 } else if (nn != n) {
a61af66fc99e Initial load
duke
parents:
diff changeset
922 tty->print("> ");
a61af66fc99e Initial load
duke
parents:
diff changeset
923 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
924 tty->print("= ");
a61af66fc99e Initial load
duke
parents:
diff changeset
925 }
a61af66fc99e Initial load
duke
parents:
diff changeset
926 if (newtype == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
927 tty->print("null");
a61af66fc99e Initial load
duke
parents:
diff changeset
928 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
929 newtype->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
930 }
a61af66fc99e Initial load
duke
parents:
diff changeset
931 do { tty->print("\t"); } while (tty->position() < 16);
a61af66fc99e Initial load
duke
parents:
diff changeset
932 nn->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
933 }
a61af66fc99e Initial load
duke
parents:
diff changeset
934 if (Verbose && wlsize < _worklist.size()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
935 tty->print(" Push {");
a61af66fc99e Initial load
duke
parents:
diff changeset
936 while (wlsize != _worklist.size()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
937 Node* pushed = _worklist.at(wlsize++);
a61af66fc99e Initial load
duke
parents:
diff changeset
938 tty->print(" %d", pushed->_idx);
a61af66fc99e Initial load
duke
parents:
diff changeset
939 }
a61af66fc99e Initial load
duke
parents:
diff changeset
940 tty->print_cr(" }");
a61af66fc99e Initial load
duke
parents:
diff changeset
941 }
a61af66fc99e Initial load
duke
parents:
diff changeset
942 }
a61af66fc99e Initial load
duke
parents:
diff changeset
943 if( VerifyIterativeGVN && nn != n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
944 verify_step((Node*) NULL); // ignore n, it might be subsumed
a61af66fc99e Initial load
duke
parents:
diff changeset
945 }
a61af66fc99e Initial load
duke
parents:
diff changeset
946 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
947 } else if (!n->is_top()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
948 remove_dead_node(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
949 }
a61af66fc99e Initial load
duke
parents:
diff changeset
950 }
a61af66fc99e Initial load
duke
parents:
diff changeset
951
a61af66fc99e Initial load
duke
parents:
diff changeset
952 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
953 C->verify_graph_edges();
a61af66fc99e Initial load
duke
parents:
diff changeset
954 if( VerifyOpto && allow_progress() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
955 // Must turn off allow_progress to enable assert and break recursion
a61af66fc99e Initial load
duke
parents:
diff changeset
956 C->root()->verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
957 { // Check if any progress was missed using IterGVN
a61af66fc99e Initial load
duke
parents:
diff changeset
958 // Def-Use info enables transformations not attempted in wash-pass
a61af66fc99e Initial load
duke
parents:
diff changeset
959 // e.g. Region/Phi cleanup, ...
a61af66fc99e Initial load
duke
parents:
diff changeset
960 // Null-check elision -- may not have reached fixpoint
a61af66fc99e Initial load
duke
parents:
diff changeset
961 // do not propagate to dominated nodes
a61af66fc99e Initial load
duke
parents:
diff changeset
962 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
963 PhaseIterGVN igvn2(this,"Verify"); // Fresh and clean!
a61af66fc99e Initial load
duke
parents:
diff changeset
964 // Fill worklist completely
a61af66fc99e Initial load
duke
parents:
diff changeset
965 igvn2.init_worklist(C->root());
a61af66fc99e Initial load
duke
parents:
diff changeset
966
a61af66fc99e Initial load
duke
parents:
diff changeset
967 igvn2.set_allow_progress(false);
a61af66fc99e Initial load
duke
parents:
diff changeset
968 igvn2.optimize();
a61af66fc99e Initial load
duke
parents:
diff changeset
969 igvn2.set_allow_progress(true);
a61af66fc99e Initial load
duke
parents:
diff changeset
970 }
a61af66fc99e Initial load
duke
parents:
diff changeset
971 }
a61af66fc99e Initial load
duke
parents:
diff changeset
972 if ( VerifyIterativeGVN && PrintOpto ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
973 if ( _verify_counter == _verify_full_passes )
a61af66fc99e Initial load
duke
parents:
diff changeset
974 tty->print_cr("VerifyIterativeGVN: %d transforms and verify passes",
a61af66fc99e Initial load
duke
parents:
diff changeset
975 _verify_full_passes);
a61af66fc99e Initial load
duke
parents:
diff changeset
976 else
a61af66fc99e Initial load
duke
parents:
diff changeset
977 tty->print_cr("VerifyIterativeGVN: %d transforms, %d full verify passes",
a61af66fc99e Initial load
duke
parents:
diff changeset
978 _verify_counter, _verify_full_passes);
a61af66fc99e Initial load
duke
parents:
diff changeset
979 }
a61af66fc99e Initial load
duke
parents:
diff changeset
980 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
981 }
a61af66fc99e Initial load
duke
parents:
diff changeset
982
a61af66fc99e Initial load
duke
parents:
diff changeset
983
a61af66fc99e Initial load
duke
parents:
diff changeset
984 //------------------register_new_node_with_optimizer---------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
985 // Register a new node with the optimizer. Update the types array, the def-use
a61af66fc99e Initial load
duke
parents:
diff changeset
986 // info. Put on worklist.
a61af66fc99e Initial load
duke
parents:
diff changeset
987 Node* PhaseIterGVN::register_new_node_with_optimizer(Node* n, Node* orig) {
a61af66fc99e Initial load
duke
parents:
diff changeset
988 set_type_bottom(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
989 _worklist.push(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
990 if (orig != NULL) C->copy_node_notes_to(n, orig);
a61af66fc99e Initial load
duke
parents:
diff changeset
991 return n;
a61af66fc99e Initial load
duke
parents:
diff changeset
992 }
a61af66fc99e Initial load
duke
parents:
diff changeset
993
a61af66fc99e Initial load
duke
parents:
diff changeset
994 //------------------------------transform--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
995 // Non-recursive: idealize Node 'n' with respect to its inputs and its value
a61af66fc99e Initial load
duke
parents:
diff changeset
996 Node *PhaseIterGVN::transform( Node *n ) {
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
997 if (_delay_transform) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
998 // Register the node but don't optimize for now
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
999 register_new_node_with_optimizer(n);
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
1000 return n;
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
1001 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 65
diff changeset
1002
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1003 // If brand new node, make space in type array, and give it a type.
a61af66fc99e Initial load
duke
parents:
diff changeset
1004 ensure_type_or_null(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1005 if (type_or_null(n) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1006 set_type_bottom(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1007 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1008
a61af66fc99e Initial load
duke
parents:
diff changeset
1009 return transform_old(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1010 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1011
a61af66fc99e Initial load
duke
parents:
diff changeset
1012 //------------------------------transform_old----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1013 Node *PhaseIterGVN::transform_old( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1014 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1015 debug_only(uint loop_count = 0;);
a61af66fc99e Initial load
duke
parents:
diff changeset
1016 set_transforms();
a61af66fc99e Initial load
duke
parents:
diff changeset
1017 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1018 // Remove 'n' from hash table in case it gets modified
a61af66fc99e Initial load
duke
parents:
diff changeset
1019 _table.hash_delete(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1020 if( VerifyIterativeGVN ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1021 assert( !_table.find_index(n->_idx), "found duplicate entry in table");
a61af66fc99e Initial load
duke
parents:
diff changeset
1022 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1023
a61af66fc99e Initial load
duke
parents:
diff changeset
1024 // Apply the Ideal call in a loop until it no longer applies
a61af66fc99e Initial load
duke
parents:
diff changeset
1025 Node *k = n;
a61af66fc99e Initial load
duke
parents:
diff changeset
1026 DEBUG_ONLY(dead_loop_check(k);)
305
ab075d07f1ba 6736417: Fastdebug C2 crashes in StoreBNode::Ideal
kvn
parents: 196
diff changeset
1027 DEBUG_ONLY(bool is_new = (k->outcnt() == 0);)
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1028 Node *i = k->Ideal(this, /*can_reshape=*/true);
305
ab075d07f1ba 6736417: Fastdebug C2 crashes in StoreBNode::Ideal
kvn
parents: 196
diff changeset
1029 assert(i != k || is_new || i->outcnt() > 0, "don't return dead nodes");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1030 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1031 if( VerifyIterativeGVN )
a61af66fc99e Initial load
duke
parents:
diff changeset
1032 verify_step(k);
a61af66fc99e Initial load
duke
parents:
diff changeset
1033 if( i && VerifyOpto ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1034 if( !allow_progress() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1035 if (i->is_Add() && i->outcnt() == 1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1036 // Switched input to left side because this is the only use
a61af66fc99e Initial load
duke
parents:
diff changeset
1037 } else if( i->is_If() && (i->in(0) == NULL) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1038 // This IF is dead because it is dominated by an equivalent IF When
a61af66fc99e Initial load
duke
parents:
diff changeset
1039 // dominating if changed, info is not propagated sparsely to 'this'
a61af66fc99e Initial load
duke
parents:
diff changeset
1040 // Propagating this info further will spuriously identify other
a61af66fc99e Initial load
duke
parents:
diff changeset
1041 // progress.
a61af66fc99e Initial load
duke
parents:
diff changeset
1042 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1043 } else
a61af66fc99e Initial load
duke
parents:
diff changeset
1044 set_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
1045 } else
a61af66fc99e Initial load
duke
parents:
diff changeset
1046 set_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
1047 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1048 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1049
a61af66fc99e Initial load
duke
parents:
diff changeset
1050 while( i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1051 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1052 debug_only( if( loop_count >= K ) i->dump(4); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1053 assert(loop_count < K, "infinite loop in PhaseIterGVN::transform");
a61af66fc99e Initial load
duke
parents:
diff changeset
1054 debug_only( loop_count++; )
a61af66fc99e Initial load
duke
parents:
diff changeset
1055 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1056 assert((i->_idx >= k->_idx) || i->is_top(), "Idealize should return new nodes, use Identity to return old nodes");
a61af66fc99e Initial load
duke
parents:
diff changeset
1057 // Made a change; put users of original Node on worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1058 add_users_to_worklist( k );
a61af66fc99e Initial load
duke
parents:
diff changeset
1059 // Replacing root of transform tree?
a61af66fc99e Initial load
duke
parents:
diff changeset
1060 if( k != i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1061 // Make users of old Node now use new.
a61af66fc99e Initial load
duke
parents:
diff changeset
1062 subsume_node( k, i );
a61af66fc99e Initial load
duke
parents:
diff changeset
1063 k = i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1064 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1065 DEBUG_ONLY(dead_loop_check(k);)
a61af66fc99e Initial load
duke
parents:
diff changeset
1066 // Try idealizing again
305
ab075d07f1ba 6736417: Fastdebug C2 crashes in StoreBNode::Ideal
kvn
parents: 196
diff changeset
1067 DEBUG_ONLY(is_new = (k->outcnt() == 0);)
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1068 i = k->Ideal(this, /*can_reshape=*/true);
305
ab075d07f1ba 6736417: Fastdebug C2 crashes in StoreBNode::Ideal
kvn
parents: 196
diff changeset
1069 assert(i != k || is_new || i->outcnt() > 0, "don't return dead nodes");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1070 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1071 if( VerifyIterativeGVN )
a61af66fc99e Initial load
duke
parents:
diff changeset
1072 verify_step(k);
a61af66fc99e Initial load
duke
parents:
diff changeset
1073 if( i && VerifyOpto ) set_progress();
a61af66fc99e Initial load
duke
parents:
diff changeset
1074 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1075 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1076
a61af66fc99e Initial load
duke
parents:
diff changeset
1077 // If brand new node, make space in type array.
a61af66fc99e Initial load
duke
parents:
diff changeset
1078 ensure_type_or_null(k);
a61af66fc99e Initial load
duke
parents:
diff changeset
1079
a61af66fc99e Initial load
duke
parents:
diff changeset
1080 // See what kind of values 'k' takes on at runtime
a61af66fc99e Initial load
duke
parents:
diff changeset
1081 const Type *t = k->Value(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
1082 assert(t != NULL, "value sanity");
a61af66fc99e Initial load
duke
parents:
diff changeset
1083
a61af66fc99e Initial load
duke
parents:
diff changeset
1084 // Since I just called 'Value' to compute the set of run-time values
a61af66fc99e Initial load
duke
parents:
diff changeset
1085 // for this Node, and 'Value' is non-local (and therefore expensive) I'll
a61af66fc99e Initial load
duke
parents:
diff changeset
1086 // cache Value. Later requests for the local phase->type of this Node can
a61af66fc99e Initial load
duke
parents:
diff changeset
1087 // use the cached Value instead of suffering with 'bottom_type'.
a61af66fc99e Initial load
duke
parents:
diff changeset
1088 if (t != type_or_null(k)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1089 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1090 NOT_PRODUCT( inc_new_values();)
a61af66fc99e Initial load
duke
parents:
diff changeset
1091 set_type(k, t);
a61af66fc99e Initial load
duke
parents:
diff changeset
1092 // If k is a TypeNode, capture any more-precise type permanently into Node
a61af66fc99e Initial load
duke
parents:
diff changeset
1093 k->raise_bottom_type(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
1094 // Move users of node to worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1095 add_users_to_worklist( k );
a61af66fc99e Initial load
duke
parents:
diff changeset
1096 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1097
a61af66fc99e Initial load
duke
parents:
diff changeset
1098 // If 'k' computes a constant, replace it with a constant
a61af66fc99e Initial load
duke
parents:
diff changeset
1099 if( t->singleton() && !k->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1100 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1101 Node *con = makecon(t); // Make a constant
a61af66fc99e Initial load
duke
parents:
diff changeset
1102 add_users_to_worklist( k );
a61af66fc99e Initial load
duke
parents:
diff changeset
1103 subsume_node( k, con ); // Everybody using k now uses con
a61af66fc99e Initial load
duke
parents:
diff changeset
1104 return con;
a61af66fc99e Initial load
duke
parents:
diff changeset
1105 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1106
a61af66fc99e Initial load
duke
parents:
diff changeset
1107 // Now check for Identities
a61af66fc99e Initial load
duke
parents:
diff changeset
1108 i = k->Identity(this); // Look for a nearby replacement
a61af66fc99e Initial load
duke
parents:
diff changeset
1109 if( i != k ) { // Found? Return replacement!
a61af66fc99e Initial load
duke
parents:
diff changeset
1110 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1111 add_users_to_worklist( k );
a61af66fc99e Initial load
duke
parents:
diff changeset
1112 subsume_node( k, i ); // Everybody using k now uses i
a61af66fc99e Initial load
duke
parents:
diff changeset
1113 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1114 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1115
a61af66fc99e Initial load
duke
parents:
diff changeset
1116 // Global Value Numbering
a61af66fc99e Initial load
duke
parents:
diff changeset
1117 i = hash_find_insert(k); // Check for pre-existing node
a61af66fc99e Initial load
duke
parents:
diff changeset
1118 if( i && (i != k) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1119 // Return the pre-existing node if it isn't dead
a61af66fc99e Initial load
duke
parents:
diff changeset
1120 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1121 add_users_to_worklist( k );
a61af66fc99e Initial load
duke
parents:
diff changeset
1122 subsume_node( k, i ); // Everybody using k now uses i
a61af66fc99e Initial load
duke
parents:
diff changeset
1123 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1124 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1125
a61af66fc99e Initial load
duke
parents:
diff changeset
1126 // Return Idealized original
a61af66fc99e Initial load
duke
parents:
diff changeset
1127 return k;
a61af66fc99e Initial load
duke
parents:
diff changeset
1128 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1129
a61af66fc99e Initial load
duke
parents:
diff changeset
1130 //---------------------------------saturate------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1131 const Type* PhaseIterGVN::saturate(const Type* new_type, const Type* old_type,
a61af66fc99e Initial load
duke
parents:
diff changeset
1132 const Type* limit_type) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1133 return new_type->narrow(old_type);
a61af66fc99e Initial load
duke
parents:
diff changeset
1134 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1135
a61af66fc99e Initial load
duke
parents:
diff changeset
1136 //------------------------------remove_globally_dead_node----------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1137 // Kill a globally dead Node. All uses are also globally dead and are
a61af66fc99e Initial load
duke
parents:
diff changeset
1138 // aggressively trimmed.
a61af66fc99e Initial load
duke
parents:
diff changeset
1139 void PhaseIterGVN::remove_globally_dead_node( Node *dead ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1140 assert(dead != C->root(), "killing root, eh?");
a61af66fc99e Initial load
duke
parents:
diff changeset
1141 if (dead->is_top()) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
1142 NOT_PRODUCT( set_progress(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1143 // Remove from iterative worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1144 _worklist.remove(dead);
a61af66fc99e Initial load
duke
parents:
diff changeset
1145 if (!dead->is_Con()) { // Don't kill cons but uses
a61af66fc99e Initial load
duke
parents:
diff changeset
1146 // Remove from hash table
a61af66fc99e Initial load
duke
parents:
diff changeset
1147 _table.hash_delete( dead );
a61af66fc99e Initial load
duke
parents:
diff changeset
1148 // Smash all inputs to 'dead', isolating him completely
a61af66fc99e Initial load
duke
parents:
diff changeset
1149 for( uint i = 0; i < dead->req(); i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1150 Node *in = dead->in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1151 if( in ) { // Points to something?
a61af66fc99e Initial load
duke
parents:
diff changeset
1152 dead->set_req(i,NULL); // Kill the edge
a61af66fc99e Initial load
duke
parents:
diff changeset
1153 if (in->outcnt() == 0 && in != C->top()) {// Made input go dead?
a61af66fc99e Initial load
duke
parents:
diff changeset
1154 remove_dead_node(in); // Recursively remove
a61af66fc99e Initial load
duke
parents:
diff changeset
1155 } else if (in->outcnt() == 1 &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1156 in->has_special_unique_user()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1157 _worklist.push(in->unique_out());
a61af66fc99e Initial load
duke
parents:
diff changeset
1158 } else if (in->outcnt() <= 2 && dead->is_Phi()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1159 if( in->Opcode() == Op_Region )
a61af66fc99e Initial load
duke
parents:
diff changeset
1160 _worklist.push(in);
a61af66fc99e Initial load
duke
parents:
diff changeset
1161 else if( in->is_Store() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1162 DUIterator_Fast imax, i = in->fast_outs(imax);
a61af66fc99e Initial load
duke
parents:
diff changeset
1163 _worklist.push(in->fast_out(i));
a61af66fc99e Initial load
duke
parents:
diff changeset
1164 i++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1165 if(in->outcnt() == 2) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1166 _worklist.push(in->fast_out(i));
a61af66fc99e Initial load
duke
parents:
diff changeset
1167 i++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1168 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1169 assert(!(i < imax), "sanity");
a61af66fc99e Initial load
duke
parents:
diff changeset
1170 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1171 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1172 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1173 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1174
a61af66fc99e Initial load
duke
parents:
diff changeset
1175 if (dead->is_macro()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1176 C->remove_macro_node(dead);
a61af66fc99e Initial load
duke
parents:
diff changeset
1177 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1178 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1179 // Aggressively kill globally dead uses
a61af66fc99e Initial load
duke
parents:
diff changeset
1180 // (Cannot use DUIterator_Last because of the indefinite number
a61af66fc99e Initial load
duke
parents:
diff changeset
1181 // of edge deletions per loop trip.)
a61af66fc99e Initial load
duke
parents:
diff changeset
1182 while (dead->outcnt() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1183 remove_globally_dead_node(dead->raw_out(0));
a61af66fc99e Initial load
duke
parents:
diff changeset
1184 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1185 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1186
a61af66fc99e Initial load
duke
parents:
diff changeset
1187 //------------------------------subsume_node-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1188 // Remove users from node 'old' and add them to node 'nn'.
a61af66fc99e Initial load
duke
parents:
diff changeset
1189 void PhaseIterGVN::subsume_node( Node *old, Node *nn ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1190 assert( old != hash_find(old), "should already been removed" );
a61af66fc99e Initial load
duke
parents:
diff changeset
1191 assert( old != C->top(), "cannot subsume top node");
a61af66fc99e Initial load
duke
parents:
diff changeset
1192 // Copy debug or profile information to the new version:
a61af66fc99e Initial load
duke
parents:
diff changeset
1193 C->copy_node_notes_to(nn, old);
a61af66fc99e Initial load
duke
parents:
diff changeset
1194 // Move users of node 'old' to node 'nn'
a61af66fc99e Initial load
duke
parents:
diff changeset
1195 for (DUIterator_Last imin, i = old->last_outs(imin); i >= imin; ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1196 Node* use = old->last_out(i); // for each use...
a61af66fc99e Initial load
duke
parents:
diff changeset
1197 // use might need re-hashing (but it won't if it's a new node)
a61af66fc99e Initial load
duke
parents:
diff changeset
1198 bool is_in_table = _table.hash_delete( use );
a61af66fc99e Initial load
duke
parents:
diff changeset
1199 // Update use-def info as well
a61af66fc99e Initial load
duke
parents:
diff changeset
1200 // We remove all occurrences of old within use->in,
a61af66fc99e Initial load
duke
parents:
diff changeset
1201 // so as to avoid rehashing any node more than once.
a61af66fc99e Initial load
duke
parents:
diff changeset
1202 // The hash table probe swamps any outer loop overhead.
a61af66fc99e Initial load
duke
parents:
diff changeset
1203 uint num_edges = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1204 for (uint jmax = use->len(), j = 0; j < jmax; j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1205 if (use->in(j) == old) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1206 use->set_req(j, nn);
a61af66fc99e Initial load
duke
parents:
diff changeset
1207 ++num_edges;
a61af66fc99e Initial load
duke
parents:
diff changeset
1208 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1209 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1210 // Insert into GVN hash table if unique
a61af66fc99e Initial load
duke
parents:
diff changeset
1211 // If a duplicate, 'use' will be cleaned up when pulled off worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1212 if( is_in_table ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1213 hash_find_insert(use);
a61af66fc99e Initial load
duke
parents:
diff changeset
1214 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1215 i -= num_edges; // we deleted 1 or more copies of this edge
a61af66fc99e Initial load
duke
parents:
diff changeset
1216 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1217
a61af66fc99e Initial load
duke
parents:
diff changeset
1218 // Smash all inputs to 'old', isolating him completely
a61af66fc99e Initial load
duke
parents:
diff changeset
1219 Node *temp = new (C, 1) Node(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
1220 temp->init_req(0,nn); // Add a use to nn to prevent him from dying
a61af66fc99e Initial load
duke
parents:
diff changeset
1221 remove_dead_node( old );
a61af66fc99e Initial load
duke
parents:
diff changeset
1222 temp->del_req(0); // Yank bogus edge
a61af66fc99e Initial load
duke
parents:
diff changeset
1223 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1224 if( VerifyIterativeGVN ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1225 for ( int i = 0; i < _verify_window_size; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1226 if ( _verify_window[i] == old )
a61af66fc99e Initial load
duke
parents:
diff changeset
1227 _verify_window[i] = nn;
a61af66fc99e Initial load
duke
parents:
diff changeset
1228 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1229 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1230 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1231 _worklist.remove(temp); // this can be necessary
a61af66fc99e Initial load
duke
parents:
diff changeset
1232 temp->destruct(); // reuse the _idx of this little guy
a61af66fc99e Initial load
duke
parents:
diff changeset
1233 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1234
a61af66fc99e Initial load
duke
parents:
diff changeset
1235 //------------------------------add_users_to_worklist--------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1236 void PhaseIterGVN::add_users_to_worklist0( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1237 for (DUIterator_Fast imax, i = n->fast_outs(imax); i < imax; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1238 _worklist.push(n->fast_out(i)); // Push on worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1239 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1240 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1241
a61af66fc99e Initial load
duke
parents:
diff changeset
1242 void PhaseIterGVN::add_users_to_worklist( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1243 add_users_to_worklist0(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1244
a61af66fc99e Initial load
duke
parents:
diff changeset
1245 // Move users of node to worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1246 for (DUIterator_Fast imax, i = n->fast_outs(imax); i < imax; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1247 Node* use = n->fast_out(i); // Get use
a61af66fc99e Initial load
duke
parents:
diff changeset
1248
a61af66fc99e Initial load
duke
parents:
diff changeset
1249 if( use->is_Multi() || // Multi-definer? Push projs on worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1250 use->is_Store() ) // Enable store/load same address
a61af66fc99e Initial load
duke
parents:
diff changeset
1251 add_users_to_worklist0(use);
a61af66fc99e Initial load
duke
parents:
diff changeset
1252
a61af66fc99e Initial load
duke
parents:
diff changeset
1253 // If we changed the receiver type to a call, we need to revisit
a61af66fc99e Initial load
duke
parents:
diff changeset
1254 // the Catch following the call. It's looking for a non-NULL
a61af66fc99e Initial load
duke
parents:
diff changeset
1255 // receiver to know when to enable the regular fall-through path
a61af66fc99e Initial load
duke
parents:
diff changeset
1256 // in addition to the NullPtrException path.
a61af66fc99e Initial load
duke
parents:
diff changeset
1257 if (use->is_CallDynamicJava() && n == use->in(TypeFunc::Parms)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1258 Node* p = use->as_CallDynamicJava()->proj_out(TypeFunc::Control);
a61af66fc99e Initial load
duke
parents:
diff changeset
1259 if (p != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1260 add_users_to_worklist0(p);
a61af66fc99e Initial load
duke
parents:
diff changeset
1261 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1262 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1263
a61af66fc99e Initial load
duke
parents:
diff changeset
1264 if( use->is_Cmp() ) { // Enable CMP/BOOL optimization
a61af66fc99e Initial load
duke
parents:
diff changeset
1265 add_users_to_worklist(use); // Put Bool on worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1266 // Look for the 'is_x2logic' pattern: "x ? : 0 : 1" and put the
a61af66fc99e Initial load
duke
parents:
diff changeset
1267 // phi merging either 0 or 1 onto the worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1268 if (use->outcnt() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1269 Node* bol = use->raw_out(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1270 if (bol->outcnt() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1271 Node* iff = bol->raw_out(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1272 if (iff->outcnt() == 2) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1273 Node* ifproj0 = iff->raw_out(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1274 Node* ifproj1 = iff->raw_out(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
1275 if (ifproj0->outcnt() > 0 && ifproj1->outcnt() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1276 Node* region0 = ifproj0->raw_out(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1277 Node* region1 = ifproj1->raw_out(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1278 if( region0 == region1 )
a61af66fc99e Initial load
duke
parents:
diff changeset
1279 add_users_to_worklist0(region0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1280 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1281 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1282 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1283 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1284 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1285
a61af66fc99e Initial load
duke
parents:
diff changeset
1286 uint use_op = use->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
1287 // If changed Cast input, check Phi users for simple cycles
65
99269dbf4ba8 6674588: (Escape Analysis) Improve Escape Analysis code
kvn
parents: 41
diff changeset
1288 if( use->is_ConstraintCast() || use->is_CheckCastPP() ) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1289 for (DUIterator_Fast i2max, i2 = use->fast_outs(i2max); i2 < i2max; i2++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1290 Node* u = use->fast_out(i2);
a61af66fc99e Initial load
duke
parents:
diff changeset
1291 if (u->is_Phi())
a61af66fc99e Initial load
duke
parents:
diff changeset
1292 _worklist.push(u);
a61af66fc99e Initial load
duke
parents:
diff changeset
1293 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1294 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1295 // If changed LShift inputs, check RShift users for useless sign-ext
a61af66fc99e Initial load
duke
parents:
diff changeset
1296 if( use_op == Op_LShiftI ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1297 for (DUIterator_Fast i2max, i2 = use->fast_outs(i2max); i2 < i2max; i2++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1298 Node* u = use->fast_out(i2);
a61af66fc99e Initial load
duke
parents:
diff changeset
1299 if (u->Opcode() == Op_RShiftI)
a61af66fc99e Initial load
duke
parents:
diff changeset
1300 _worklist.push(u);
a61af66fc99e Initial load
duke
parents:
diff changeset
1301 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1302 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1303 // If changed AddP inputs, check Stores for loop invariant
a61af66fc99e Initial load
duke
parents:
diff changeset
1304 if( use_op == Op_AddP ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1305 for (DUIterator_Fast i2max, i2 = use->fast_outs(i2max); i2 < i2max; i2++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1306 Node* u = use->fast_out(i2);
a61af66fc99e Initial load
duke
parents:
diff changeset
1307 if (u->is_Mem())
a61af66fc99e Initial load
duke
parents:
diff changeset
1308 _worklist.push(u);
a61af66fc99e Initial load
duke
parents:
diff changeset
1309 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1310 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1311 // If changed initialization activity, check dependent Stores
a61af66fc99e Initial load
duke
parents:
diff changeset
1312 if (use_op == Op_Allocate || use_op == Op_AllocateArray) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1313 InitializeNode* init = use->as_Allocate()->initialization();
a61af66fc99e Initial load
duke
parents:
diff changeset
1314 if (init != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1315 Node* imem = init->proj_out(TypeFunc::Memory);
a61af66fc99e Initial load
duke
parents:
diff changeset
1316 if (imem != NULL) add_users_to_worklist0(imem);
a61af66fc99e Initial load
duke
parents:
diff changeset
1317 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1318 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1319 if (use_op == Op_Initialize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1320 Node* imem = use->as_Initialize()->proj_out(TypeFunc::Memory);
a61af66fc99e Initial load
duke
parents:
diff changeset
1321 if (imem != NULL) add_users_to_worklist0(imem);
a61af66fc99e Initial load
duke
parents:
diff changeset
1322 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1323 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1324 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1325
a61af66fc99e Initial load
duke
parents:
diff changeset
1326 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
1327 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1328 uint PhaseCCP::_total_invokes = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1329 uint PhaseCCP::_total_constants = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1330 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1331 //------------------------------PhaseCCP---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1332 // Conditional Constant Propagation, ala Wegman & Zadeck
a61af66fc99e Initial load
duke
parents:
diff changeset
1333 PhaseCCP::PhaseCCP( PhaseIterGVN *igvn ) : PhaseIterGVN(igvn) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1334 NOT_PRODUCT( clear_constants(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1335 assert( _worklist.size() == 0, "" );
a61af66fc99e Initial load
duke
parents:
diff changeset
1336 // Clear out _nodes from IterGVN. Must be clear to transform call.
a61af66fc99e Initial load
duke
parents:
diff changeset
1337 _nodes.clear(); // Clear out from IterGVN
a61af66fc99e Initial load
duke
parents:
diff changeset
1338 analyze();
a61af66fc99e Initial load
duke
parents:
diff changeset
1339 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1340
a61af66fc99e Initial load
duke
parents:
diff changeset
1341 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1342 //------------------------------~PhaseCCP--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1343 PhaseCCP::~PhaseCCP() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1344 inc_invokes();
a61af66fc99e Initial load
duke
parents:
diff changeset
1345 _total_constants += count_constants();
a61af66fc99e Initial load
duke
parents:
diff changeset
1346 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1347 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1348
a61af66fc99e Initial load
duke
parents:
diff changeset
1349
a61af66fc99e Initial load
duke
parents:
diff changeset
1350 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
1351 static bool ccp_type_widens(const Type* t, const Type* t0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1352 assert(t->meet(t0) == t, "Not monotonic");
a61af66fc99e Initial load
duke
parents:
diff changeset
1353 switch (t->base() == t0->base() ? t->base() : Type::Top) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1354 case Type::Int:
a61af66fc99e Initial load
duke
parents:
diff changeset
1355 assert(t0->isa_int()->_widen <= t->isa_int()->_widen, "widen increases");
a61af66fc99e Initial load
duke
parents:
diff changeset
1356 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1357 case Type::Long:
a61af66fc99e Initial load
duke
parents:
diff changeset
1358 assert(t0->isa_long()->_widen <= t->isa_long()->_widen, "widen increases");
a61af66fc99e Initial load
duke
parents:
diff changeset
1359 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1360 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1361 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1362 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1363 #endif //ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
1364
a61af66fc99e Initial load
duke
parents:
diff changeset
1365 //------------------------------analyze----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1366 void PhaseCCP::analyze() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1367 // Initialize all types to TOP, optimistic analysis
a61af66fc99e Initial load
duke
parents:
diff changeset
1368 for (int i = C->unique() - 1; i >= 0; i--) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1369 _types.map(i,Type::TOP);
a61af66fc99e Initial load
duke
parents:
diff changeset
1370 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1371
a61af66fc99e Initial load
duke
parents:
diff changeset
1372 // Push root onto worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1373 Unique_Node_List worklist;
a61af66fc99e Initial load
duke
parents:
diff changeset
1374 worklist.push(C->root());
a61af66fc99e Initial load
duke
parents:
diff changeset
1375
a61af66fc99e Initial load
duke
parents:
diff changeset
1376 // Pull from worklist; compute new value; push changes out.
a61af66fc99e Initial load
duke
parents:
diff changeset
1377 // This loop is the meat of CCP.
a61af66fc99e Initial load
duke
parents:
diff changeset
1378 while( worklist.size() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1379 Node *n = worklist.pop();
a61af66fc99e Initial load
duke
parents:
diff changeset
1380 const Type *t = n->Value(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
1381 if (t != type(n)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1382 assert(ccp_type_widens(t, type(n)), "ccp type must widen");
a61af66fc99e Initial load
duke
parents:
diff changeset
1383 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1384 if( TracePhaseCCP ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1385 t->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
1386 do { tty->print("\t"); } while (tty->position() < 16);
a61af66fc99e Initial load
duke
parents:
diff changeset
1387 n->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
1388 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1389 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1390 set_type(n, t);
a61af66fc99e Initial load
duke
parents:
diff changeset
1391 for (DUIterator_Fast imax, i = n->fast_outs(imax); i < imax; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1392 Node* m = n->fast_out(i); // Get user
a61af66fc99e Initial load
duke
parents:
diff changeset
1393 if( m->is_Region() ) { // New path to Region? Must recheck Phis too
a61af66fc99e Initial load
duke
parents:
diff changeset
1394 for (DUIterator_Fast i2max, i2 = m->fast_outs(i2max); i2 < i2max; i2++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1395 Node* p = m->fast_out(i2); // Propagate changes to uses
a61af66fc99e Initial load
duke
parents:
diff changeset
1396 if( p->bottom_type() != type(p) ) // If not already bottomed out
a61af66fc99e Initial load
duke
parents:
diff changeset
1397 worklist.push(p); // Propagate change to user
a61af66fc99e Initial load
duke
parents:
diff changeset
1398 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1399 }
605
98cb887364d3 6810672: Comment typos
twisti
parents: 305
diff changeset
1400 // If we changed the receiver type to a call, we need to revisit
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1401 // the Catch following the call. It's looking for a non-NULL
a61af66fc99e Initial load
duke
parents:
diff changeset
1402 // receiver to know when to enable the regular fall-through path
a61af66fc99e Initial load
duke
parents:
diff changeset
1403 // in addition to the NullPtrException path
a61af66fc99e Initial load
duke
parents:
diff changeset
1404 if (m->is_Call()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1405 for (DUIterator_Fast i2max, i2 = m->fast_outs(i2max); i2 < i2max; i2++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1406 Node* p = m->fast_out(i2); // Propagate changes to uses
a61af66fc99e Initial load
duke
parents:
diff changeset
1407 if (p->is_Proj() && p->as_Proj()->_con == TypeFunc::Control && p->outcnt() == 1)
a61af66fc99e Initial load
duke
parents:
diff changeset
1408 worklist.push(p->unique_out());
a61af66fc99e Initial load
duke
parents:
diff changeset
1409 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1410 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1411 if( m->bottom_type() != type(m) ) // If not already bottomed out
a61af66fc99e Initial load
duke
parents:
diff changeset
1412 worklist.push(m); // Propagate change to user
a61af66fc99e Initial load
duke
parents:
diff changeset
1413 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1414 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1415 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1416 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1417
a61af66fc99e Initial load
duke
parents:
diff changeset
1418 //------------------------------do_transform-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1419 // Top level driver for the recursive transformer
a61af66fc99e Initial load
duke
parents:
diff changeset
1420 void PhaseCCP::do_transform() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1421 // Correct leaves of new-space Nodes; they point to old-space.
a61af66fc99e Initial load
duke
parents:
diff changeset
1422 C->set_root( transform(C->root())->as_Root() );
a61af66fc99e Initial load
duke
parents:
diff changeset
1423 assert( C->top(), "missing TOP node" );
a61af66fc99e Initial load
duke
parents:
diff changeset
1424 assert( C->root(), "missing root" );
a61af66fc99e Initial load
duke
parents:
diff changeset
1425 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1426
a61af66fc99e Initial load
duke
parents:
diff changeset
1427 //------------------------------transform--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1428 // Given a Node in old-space, clone him into new-space.
a61af66fc99e Initial load
duke
parents:
diff changeset
1429 // Convert any of his old-space children into new-space children.
a61af66fc99e Initial load
duke
parents:
diff changeset
1430 Node *PhaseCCP::transform( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1431 Node *new_node = _nodes[n->_idx]; // Check for transformed node
a61af66fc99e Initial load
duke
parents:
diff changeset
1432 if( new_node != NULL )
a61af66fc99e Initial load
duke
parents:
diff changeset
1433 return new_node; // Been there, done that, return old answer
a61af66fc99e Initial load
duke
parents:
diff changeset
1434 new_node = transform_once(n); // Check for constant
a61af66fc99e Initial load
duke
parents:
diff changeset
1435 _nodes.map( n->_idx, new_node ); // Flag as having been cloned
a61af66fc99e Initial load
duke
parents:
diff changeset
1436
a61af66fc99e Initial load
duke
parents:
diff changeset
1437 // Allocate stack of size _nodes.Size()/2 to avoid frequent realloc
a61af66fc99e Initial load
duke
parents:
diff changeset
1438 GrowableArray <Node *> trstack(C->unique() >> 1);
a61af66fc99e Initial load
duke
parents:
diff changeset
1439
a61af66fc99e Initial load
duke
parents:
diff changeset
1440 trstack.push(new_node); // Process children of cloned node
a61af66fc99e Initial load
duke
parents:
diff changeset
1441 while ( trstack.is_nonempty() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1442 Node *clone = trstack.pop();
a61af66fc99e Initial load
duke
parents:
diff changeset
1443 uint cnt = clone->req();
a61af66fc99e Initial load
duke
parents:
diff changeset
1444 for( uint i = 0; i < cnt; i++ ) { // For all inputs do
a61af66fc99e Initial load
duke
parents:
diff changeset
1445 Node *input = clone->in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1446 if( input != NULL ) { // Ignore NULLs
a61af66fc99e Initial load
duke
parents:
diff changeset
1447 Node *new_input = _nodes[input->_idx]; // Check for cloned input node
a61af66fc99e Initial load
duke
parents:
diff changeset
1448 if( new_input == NULL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1449 new_input = transform_once(input); // Check for constant
a61af66fc99e Initial load
duke
parents:
diff changeset
1450 _nodes.map( input->_idx, new_input );// Flag as having been cloned
a61af66fc99e Initial load
duke
parents:
diff changeset
1451 trstack.push(new_input);
a61af66fc99e Initial load
duke
parents:
diff changeset
1452 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1453 assert( new_input == clone->in(i), "insanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
1454 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1455 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1456 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1457 return new_node;
a61af66fc99e Initial load
duke
parents:
diff changeset
1458 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1459
a61af66fc99e Initial load
duke
parents:
diff changeset
1460
a61af66fc99e Initial load
duke
parents:
diff changeset
1461 //------------------------------transform_once---------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1462 // For PhaseCCP, transformation is IDENTITY unless Node computed a constant.
a61af66fc99e Initial load
duke
parents:
diff changeset
1463 Node *PhaseCCP::transform_once( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1464 const Type *t = type(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1465 // Constant? Use constant Node instead
a61af66fc99e Initial load
duke
parents:
diff changeset
1466 if( t->singleton() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1467 Node *nn = n; // Default is to return the original constant
a61af66fc99e Initial load
duke
parents:
diff changeset
1468 if( t == Type::TOP ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1469 // cache my top node on the Compile instance
a61af66fc99e Initial load
duke
parents:
diff changeset
1470 if( C->cached_top_node() == NULL || C->cached_top_node()->in(0) == NULL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1471 C->set_cached_top_node( ConNode::make(C, Type::TOP) );
a61af66fc99e Initial load
duke
parents:
diff changeset
1472 set_type(C->top(), Type::TOP);
a61af66fc99e Initial load
duke
parents:
diff changeset
1473 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1474 nn = C->top();
a61af66fc99e Initial load
duke
parents:
diff changeset
1475 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1476 if( !n->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1477 if( t != Type::TOP ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1478 nn = makecon(t); // ConNode::make(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
1479 NOT_PRODUCT( inc_constants(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1480 } else if( n->is_Region() ) { // Unreachable region
a61af66fc99e Initial load
duke
parents:
diff changeset
1481 // Note: nn == C->top()
a61af66fc99e Initial load
duke
parents:
diff changeset
1482 n->set_req(0, NULL); // Cut selfreference
a61af66fc99e Initial load
duke
parents:
diff changeset
1483 // Eagerly remove dead phis to avoid phis copies creation.
a61af66fc99e Initial load
duke
parents:
diff changeset
1484 for (DUIterator i = n->outs(); n->has_out(i); i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1485 Node* m = n->out(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1486 if( m->is_Phi() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1487 assert(type(m) == Type::TOP, "Unreachable region should not have live phis.");
1621
6027dddc26c6 6677629: PhaseIterGVN::subsume_node() should call hash_delete() and add_users_to_worklist()
kvn
parents: 1552
diff changeset
1488 replace_node(m, nn);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1489 --i; // deleted this phi; rescan starting with next position
a61af66fc99e Initial load
duke
parents:
diff changeset
1490 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1491 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1492 }
1621
6027dddc26c6 6677629: PhaseIterGVN::subsume_node() should call hash_delete() and add_users_to_worklist()
kvn
parents: 1552
diff changeset
1493 replace_node(n,nn); // Update DefUse edges for new constant
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1494 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1495 return nn;
a61af66fc99e Initial load
duke
parents:
diff changeset
1496 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1497
a61af66fc99e Initial load
duke
parents:
diff changeset
1498 // If x is a TypeNode, capture any more-precise type permanently into Node
a61af66fc99e Initial load
duke
parents:
diff changeset
1499 if (t != n->bottom_type()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1500 hash_delete(n); // changing bottom type may force a rehash
a61af66fc99e Initial load
duke
parents:
diff changeset
1501 n->raise_bottom_type(t);
a61af66fc99e Initial load
duke
parents:
diff changeset
1502 _worklist.push(n); // n re-enters the hash table via the worklist
a61af66fc99e Initial load
duke
parents:
diff changeset
1503 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1504
a61af66fc99e Initial load
duke
parents:
diff changeset
1505 // Idealize graph using DU info. Must clone() into new-space.
a61af66fc99e Initial load
duke
parents:
diff changeset
1506 // DU info is generally used to show profitability, progress or safety
a61af66fc99e Initial load
duke
parents:
diff changeset
1507 // (but generally not needed for correctness).
a61af66fc99e Initial load
duke
parents:
diff changeset
1508 Node *nn = n->Ideal_DU_postCCP(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
1509
a61af66fc99e Initial load
duke
parents:
diff changeset
1510 // TEMPORARY fix to ensure that 2nd GVN pass eliminates NULL checks
a61af66fc99e Initial load
duke
parents:
diff changeset
1511 switch( n->Opcode() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1512 case Op_FastLock: // Revisit FastLocks for lock coarsening
a61af66fc99e Initial load
duke
parents:
diff changeset
1513 case Op_If:
a61af66fc99e Initial load
duke
parents:
diff changeset
1514 case Op_CountedLoopEnd:
a61af66fc99e Initial load
duke
parents:
diff changeset
1515 case Op_Region:
a61af66fc99e Initial load
duke
parents:
diff changeset
1516 case Op_Loop:
a61af66fc99e Initial load
duke
parents:
diff changeset
1517 case Op_CountedLoop:
a61af66fc99e Initial load
duke
parents:
diff changeset
1518 case Op_Conv2B:
a61af66fc99e Initial load
duke
parents:
diff changeset
1519 case Op_Opaque1:
a61af66fc99e Initial load
duke
parents:
diff changeset
1520 case Op_Opaque2:
a61af66fc99e Initial load
duke
parents:
diff changeset
1521 _worklist.push(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1522 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1523 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1524 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1525 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1526 if( nn ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1527 _worklist.push(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1528 // Put users of 'n' onto worklist for second igvn transform
a61af66fc99e Initial load
duke
parents:
diff changeset
1529 add_users_to_worklist(n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1530 return nn;
a61af66fc99e Initial load
duke
parents:
diff changeset
1531 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1532
a61af66fc99e Initial load
duke
parents:
diff changeset
1533 return n;
a61af66fc99e Initial load
duke
parents:
diff changeset
1534 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1535
a61af66fc99e Initial load
duke
parents:
diff changeset
1536 //---------------------------------saturate------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1537 const Type* PhaseCCP::saturate(const Type* new_type, const Type* old_type,
a61af66fc99e Initial load
duke
parents:
diff changeset
1538 const Type* limit_type) const {
1009
03b336640699 6885584: A particular class structure causes large allocation spike for jit
never
parents: 927
diff changeset
1539 const Type* wide_type = new_type->widen(old_type, limit_type);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1540 if (wide_type != new_type) { // did we widen?
a61af66fc99e Initial load
duke
parents:
diff changeset
1541 // If so, we may have widened beyond the limit type. Clip it back down.
a61af66fc99e Initial load
duke
parents:
diff changeset
1542 new_type = wide_type->filter(limit_type);
a61af66fc99e Initial load
duke
parents:
diff changeset
1543 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1544 return new_type;
a61af66fc99e Initial load
duke
parents:
diff changeset
1545 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1546
a61af66fc99e Initial load
duke
parents:
diff changeset
1547 //------------------------------print_statistics-------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1548 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1549 void PhaseCCP::print_statistics() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1550 tty->print_cr("CCP: %d constants found: %d", _total_invokes, _total_constants);
a61af66fc99e Initial load
duke
parents:
diff changeset
1551 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1552 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1553
a61af66fc99e Initial load
duke
parents:
diff changeset
1554
a61af66fc99e Initial load
duke
parents:
diff changeset
1555 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
1556 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1557 uint PhasePeephole::_total_peepholes = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1558 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1559 //------------------------------PhasePeephole----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1560 // Conditional Constant Propagation, ala Wegman & Zadeck
a61af66fc99e Initial load
duke
parents:
diff changeset
1561 PhasePeephole::PhasePeephole( PhaseRegAlloc *regalloc, PhaseCFG &cfg )
a61af66fc99e Initial load
duke
parents:
diff changeset
1562 : PhaseTransform(Peephole), _regalloc(regalloc), _cfg(cfg) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1563 NOT_PRODUCT( clear_peepholes(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1564 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1565
a61af66fc99e Initial load
duke
parents:
diff changeset
1566 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1567 //------------------------------~PhasePeephole---------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1568 PhasePeephole::~PhasePeephole() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1569 _total_peepholes += count_peepholes();
a61af66fc99e Initial load
duke
parents:
diff changeset
1570 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1571 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1572
a61af66fc99e Initial load
duke
parents:
diff changeset
1573 //------------------------------transform--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1574 Node *PhasePeephole::transform( Node *n ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1575 ShouldNotCallThis();
a61af66fc99e Initial load
duke
parents:
diff changeset
1576 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1577 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1578
a61af66fc99e Initial load
duke
parents:
diff changeset
1579 //------------------------------do_transform-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1580 void PhasePeephole::do_transform() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1581 bool method_name_not_printed = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1582
a61af66fc99e Initial load
duke
parents:
diff changeset
1583 // Examine each basic block
a61af66fc99e Initial load
duke
parents:
diff changeset
1584 for( uint block_number = 1; block_number < _cfg._num_blocks; ++block_number ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1585 Block *block = _cfg._blocks[block_number];
a61af66fc99e Initial load
duke
parents:
diff changeset
1586 bool block_not_printed = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
1587
a61af66fc99e Initial load
duke
parents:
diff changeset
1588 // and each instruction within a block
a61af66fc99e Initial load
duke
parents:
diff changeset
1589 uint end_index = block->_nodes.size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1590 // block->end_idx() not valid after PhaseRegAlloc
a61af66fc99e Initial load
duke
parents:
diff changeset
1591 for( uint instruction_index = 1; instruction_index < end_index; ++instruction_index ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1592 Node *n = block->_nodes.at(instruction_index);
a61af66fc99e Initial load
duke
parents:
diff changeset
1593 if( n->is_Mach() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1594 MachNode *m = n->as_Mach();
a61af66fc99e Initial load
duke
parents:
diff changeset
1595 int deleted_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1596 // check for peephole opportunities
a61af66fc99e Initial load
duke
parents:
diff changeset
1597 MachNode *m2 = m->peephole( block, instruction_index, _regalloc, deleted_count, C );
a61af66fc99e Initial load
duke
parents:
diff changeset
1598 if( m2 != NULL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1599 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1600 if( PrintOptoPeephole ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1601 // Print method, first time only
a61af66fc99e Initial load
duke
parents:
diff changeset
1602 if( C->method() && method_name_not_printed ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1603 C->method()->print_short_name(); tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
1604 method_name_not_printed = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1605 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1606 // Print this block
a61af66fc99e Initial load
duke
parents:
diff changeset
1607 if( Verbose && block_not_printed) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1608 tty->print_cr("in block");
a61af66fc99e Initial load
duke
parents:
diff changeset
1609 block->dump();
a61af66fc99e Initial load
duke
parents:
diff changeset
1610 block_not_printed = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1611 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1612 // Print instructions being deleted
a61af66fc99e Initial load
duke
parents:
diff changeset
1613 for( int i = (deleted_count - 1); i >= 0; --i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1614 block->_nodes.at(instruction_index-i)->as_Mach()->format(_regalloc); tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
1615 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1616 tty->print_cr("replaced with");
a61af66fc99e Initial load
duke
parents:
diff changeset
1617 // Print new instruction
a61af66fc99e Initial load
duke
parents:
diff changeset
1618 m2->format(_regalloc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1619 tty->print("\n\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
1620 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1621 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1622 // Remove old nodes from basic block and update instruction_index
a61af66fc99e Initial load
duke
parents:
diff changeset
1623 // (old nodes still exist and may have edges pointing to them
a61af66fc99e Initial load
duke
parents:
diff changeset
1624 // as register allocation info is stored in the allocator using
a61af66fc99e Initial load
duke
parents:
diff changeset
1625 // the node index to live range mappings.)
a61af66fc99e Initial load
duke
parents:
diff changeset
1626 uint safe_instruction_index = (instruction_index - deleted_count);
a61af66fc99e Initial load
duke
parents:
diff changeset
1627 for( ; (instruction_index > safe_instruction_index); --instruction_index ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1628 block->_nodes.remove( instruction_index );
a61af66fc99e Initial load
duke
parents:
diff changeset
1629 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1630 // install new node after safe_instruction_index
a61af66fc99e Initial load
duke
parents:
diff changeset
1631 block->_nodes.insert( safe_instruction_index + 1, m2 );
a61af66fc99e Initial load
duke
parents:
diff changeset
1632 end_index = block->_nodes.size() - 1; // Recompute new block size
a61af66fc99e Initial load
duke
parents:
diff changeset
1633 NOT_PRODUCT( inc_peepholes(); )
a61af66fc99e Initial load
duke
parents:
diff changeset
1634 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1635 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1636 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1637 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1638 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1639
a61af66fc99e Initial load
duke
parents:
diff changeset
1640 //------------------------------print_statistics-------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1641 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1642 void PhasePeephole::print_statistics() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1643 tty->print_cr("Peephole: peephole rules applied: %d", _total_peepholes);
a61af66fc99e Initial load
duke
parents:
diff changeset
1644 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1645 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1646
a61af66fc99e Initial load
duke
parents:
diff changeset
1647
a61af66fc99e Initial load
duke
parents:
diff changeset
1648 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
1649 //------------------------------set_req_X--------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1650 void Node::set_req_X( uint i, Node *n, PhaseIterGVN *igvn ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1651 assert( is_not_dead(n), "can not use dead node");
a61af66fc99e Initial load
duke
parents:
diff changeset
1652 assert( igvn->hash_find(this) != this, "Need to remove from hash before changing edges" );
a61af66fc99e Initial load
duke
parents:
diff changeset
1653 Node *old = in(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1654 set_req(i, n);
a61af66fc99e Initial load
duke
parents:
diff changeset
1655
a61af66fc99e Initial load
duke
parents:
diff changeset
1656 // old goes dead?
a61af66fc99e Initial load
duke
parents:
diff changeset
1657 if( old ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1658 switch (old->outcnt()) {
927
662f330d7275 6866651: Regression: simple int sum crashes jvm (build 1.6.0_14-b08 and 1.7.0-ea-b59)
cfang
parents: 605
diff changeset
1659 case 0:
662f330d7275 6866651: Regression: simple int sum crashes jvm (build 1.6.0_14-b08 and 1.7.0-ea-b59)
cfang
parents: 605
diff changeset
1660 // Put into the worklist to kill later. We do not kill it now because the
662f330d7275 6866651: Regression: simple int sum crashes jvm (build 1.6.0_14-b08 and 1.7.0-ea-b59)
cfang
parents: 605
diff changeset
1661 // recursive kill will delete the current node (this) if dead-loop exists
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1662 if (!old->is_top())
927
662f330d7275 6866651: Regression: simple int sum crashes jvm (build 1.6.0_14-b08 and 1.7.0-ea-b59)
cfang
parents: 605
diff changeset
1663 igvn->_worklist.push( old );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1664 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1665 case 1:
a61af66fc99e Initial load
duke
parents:
diff changeset
1666 if( old->is_Store() || old->has_special_unique_user() )
a61af66fc99e Initial load
duke
parents:
diff changeset
1667 igvn->add_users_to_worklist( old );
a61af66fc99e Initial load
duke
parents:
diff changeset
1668 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1669 case 2:
a61af66fc99e Initial load
duke
parents:
diff changeset
1670 if( old->is_Store() )
a61af66fc99e Initial load
duke
parents:
diff changeset
1671 igvn->add_users_to_worklist( old );
a61af66fc99e Initial load
duke
parents:
diff changeset
1672 if( old->Opcode() == Op_Region )
a61af66fc99e Initial load
duke
parents:
diff changeset
1673 igvn->_worklist.push(old);
a61af66fc99e Initial load
duke
parents:
diff changeset
1674 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1675 case 3:
a61af66fc99e Initial load
duke
parents:
diff changeset
1676 if( old->Opcode() == Op_Region ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1677 igvn->_worklist.push(old);
a61af66fc99e Initial load
duke
parents:
diff changeset
1678 igvn->add_users_to_worklist( old );
a61af66fc99e Initial load
duke
parents:
diff changeset
1679 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1680 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1681 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
1682 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
1683 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1684 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1685
a61af66fc99e Initial load
duke
parents:
diff changeset
1686 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1687
a61af66fc99e Initial load
duke
parents:
diff changeset
1688 //-------------------------------replace_by-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1689 // Using def-use info, replace one node for another. Follow the def-use info
a61af66fc99e Initial load
duke
parents:
diff changeset
1690 // to all users of the OLD node. Then make all uses point to the NEW node.
a61af66fc99e Initial load
duke
parents:
diff changeset
1691 void Node::replace_by(Node *new_node) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1692 assert(!is_top(), "top node has no DU info");
a61af66fc99e Initial load
duke
parents:
diff changeset
1693 for (DUIterator_Last imin, i = last_outs(imin); i >= imin; ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1694 Node* use = last_out(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
1695 uint uses_found = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1696 for (uint j = 0; j < use->len(); j++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1697 if (use->in(j) == this) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1698 if (j < use->req())
a61af66fc99e Initial load
duke
parents:
diff changeset
1699 use->set_req(j, new_node);
a61af66fc99e Initial load
duke
parents:
diff changeset
1700 else use->set_prec(j, new_node);
a61af66fc99e Initial load
duke
parents:
diff changeset
1701 uses_found++;
a61af66fc99e Initial load
duke
parents:
diff changeset
1702 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1703 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1704 i -= uses_found; // we deleted 1 or more copies of this edge
a61af66fc99e Initial load
duke
parents:
diff changeset
1705 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1706 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1707
a61af66fc99e Initial load
duke
parents:
diff changeset
1708 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
1709 //-----------------------------------------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1710 void Type_Array::grow( uint i ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1711 if( !_max ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1712 _max = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
1713 _types = (const Type**)_a->Amalloc( _max * sizeof(Type*) );
a61af66fc99e Initial load
duke
parents:
diff changeset
1714 _types[0] = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1715 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1716 uint old = _max;
a61af66fc99e Initial load
duke
parents:
diff changeset
1717 while( i >= _max ) _max <<= 1; // Double to fit
a61af66fc99e Initial load
duke
parents:
diff changeset
1718 _types = (const Type**)_a->Arealloc( _types, old*sizeof(Type*),_max*sizeof(Type*));
a61af66fc99e Initial load
duke
parents:
diff changeset
1719 memset( &_types[old], 0, (_max-old)*sizeof(Type*) );
a61af66fc99e Initial load
duke
parents:
diff changeset
1720 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1721
a61af66fc99e Initial load
duke
parents:
diff changeset
1722 //------------------------------dump-------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
1723 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1724 void Type_Array::dump() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1725 uint max = Size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1726 for( uint i = 0; i < max; i++ ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1727 if( _types[i] != NULL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1728 tty->print(" %d\t== ", i); _types[i]->dump(); tty->cr();
a61af66fc99e Initial load
duke
parents:
diff changeset
1729 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1730 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1731 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1732 #endif