annotate src/share/vm/opto/phaseX.cpp @ 8804:91bf0bdae37b

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