annotate src/share/vm/opto/phaseX.cpp @ 3249:e1162778c1c8

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