annotate src/share/vm/opto/phaseX.cpp @ 14649:f6301b007a16

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