annotate src/share/vm/opto/addnode.cpp @ 1941:79d04223b8a5

Added caching for resolved types and resolved fields. This is crucial, because the local load elimination will lead to wrong results, if field equality (of two RiField objects with the same object and the same RiType) is not given. The caching makes sure that the default equals implementation is sufficient.
author Thomas Wuerthinger <wuerthinger@ssw.jku.at>
date Tue, 28 Dec 2010 18:33:26 +0100
parents d6f45b55c972
children f95d63e2154a
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
1552
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1541
diff changeset
2 * Copyright (c) 1997, 2009, 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: 1541
diff changeset
19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
c18cbe5936b8 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1541
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: 1541
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
a61af66fc99e Initial load
duke
parents:
diff changeset
25 // Portions of code courtesy of Clifford Click
a61af66fc99e Initial load
duke
parents:
diff changeset
26
a61af66fc99e Initial load
duke
parents:
diff changeset
27 #include "incls/_precompiled.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
28 #include "incls/_addnode.cpp.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
29
a61af66fc99e Initial load
duke
parents:
diff changeset
30 #define MAXFLOAT ((float)3.40282346638528860e+38)
a61af66fc99e Initial load
duke
parents:
diff changeset
31
a61af66fc99e Initial load
duke
parents:
diff changeset
32 // Classic Add functionality. This covers all the usual 'add' behaviors for
a61af66fc99e Initial load
duke
parents:
diff changeset
33 // an algebraic ring. Add-integer, add-float, add-double, and binary-or are
a61af66fc99e Initial load
duke
parents:
diff changeset
34 // all inherited from this class. The various identity values are supplied
a61af66fc99e Initial load
duke
parents:
diff changeset
35 // by virtual functions.
a61af66fc99e Initial load
duke
parents:
diff changeset
36
a61af66fc99e Initial load
duke
parents:
diff changeset
37
a61af66fc99e Initial load
duke
parents:
diff changeset
38 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
39 //------------------------------hash-------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
40 // Hash function over AddNodes. Needs to be commutative; i.e., I swap
a61af66fc99e Initial load
duke
parents:
diff changeset
41 // (commute) inputs to AddNodes willy-nilly so the hash function must return
a61af66fc99e Initial load
duke
parents:
diff changeset
42 // the same value in the presence of edge swapping.
a61af66fc99e Initial load
duke
parents:
diff changeset
43 uint AddNode::hash() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
44 return (uintptr_t)in(1) + (uintptr_t)in(2) + Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
45 }
a61af66fc99e Initial load
duke
parents:
diff changeset
46
a61af66fc99e Initial load
duke
parents:
diff changeset
47 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
48 // If either input is a constant 0, return the other input.
a61af66fc99e Initial load
duke
parents:
diff changeset
49 Node *AddNode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
50 const Type *zero = add_id(); // The additive identity
a61af66fc99e Initial load
duke
parents:
diff changeset
51 if( phase->type( in(1) )->higher_equal( zero ) ) return in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
52 if( phase->type( in(2) )->higher_equal( zero ) ) return in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
53 return this;
a61af66fc99e Initial load
duke
parents:
diff changeset
54 }
a61af66fc99e Initial load
duke
parents:
diff changeset
55
a61af66fc99e Initial load
duke
parents:
diff changeset
56 //------------------------------commute----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
57 // Commute operands to move loads and constants to the right.
a61af66fc99e Initial load
duke
parents:
diff changeset
58 static bool commute( Node *add, int con_left, int con_right ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
59 Node *in1 = add->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
60 Node *in2 = add->in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
61
a61af66fc99e Initial load
duke
parents:
diff changeset
62 // Convert "1+x" into "x+1".
a61af66fc99e Initial load
duke
parents:
diff changeset
63 // Right is a constant; leave it
a61af66fc99e Initial load
duke
parents:
diff changeset
64 if( con_right ) return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
65 // Left is a constant; move it right.
a61af66fc99e Initial load
duke
parents:
diff changeset
66 if( con_left ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
67 add->swap_edges(1, 2);
a61af66fc99e Initial load
duke
parents:
diff changeset
68 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
69 }
a61af66fc99e Initial load
duke
parents:
diff changeset
70
a61af66fc99e Initial load
duke
parents:
diff changeset
71 // Convert "Load+x" into "x+Load".
a61af66fc99e Initial load
duke
parents:
diff changeset
72 // Now check for loads
99
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
73 if (in2->is_Load()) {
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
74 if (!in1->is_Load()) {
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
75 // already x+Load to return
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
76 return false;
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
77 }
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
78 // both are loads, so fall through to sort inputs by idx
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
79 } else if( in1->is_Load() ) {
8a4ef4e001d3 6680594: Load + Load isn't canonicalized leading to missed GVN opportunities
never
parents: 32
diff changeset
80 // Left is a Load and Right is not; move it right.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
81 add->swap_edges(1, 2);
a61af66fc99e Initial load
duke
parents:
diff changeset
82 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
83 }
a61af66fc99e Initial load
duke
parents:
diff changeset
84
a61af66fc99e Initial load
duke
parents:
diff changeset
85 PhiNode *phi;
a61af66fc99e Initial load
duke
parents:
diff changeset
86 // Check for tight loop increments: Loop-phi of Add of loop-phi
a61af66fc99e Initial load
duke
parents:
diff changeset
87 if( in1->is_Phi() && (phi = in1->as_Phi()) && !phi->is_copy() && phi->region()->is_Loop() && phi->in(2)==add)
a61af66fc99e Initial load
duke
parents:
diff changeset
88 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
89 if( in2->is_Phi() && (phi = in2->as_Phi()) && !phi->is_copy() && phi->region()->is_Loop() && phi->in(2)==add){
a61af66fc99e Initial load
duke
parents:
diff changeset
90 add->swap_edges(1, 2);
a61af66fc99e Initial load
duke
parents:
diff changeset
91 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
92 }
a61af66fc99e Initial load
duke
parents:
diff changeset
93
a61af66fc99e Initial load
duke
parents:
diff changeset
94 // Otherwise, sort inputs (commutativity) to help value numbering.
a61af66fc99e Initial load
duke
parents:
diff changeset
95 if( in1->_idx > in2->_idx ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
96 add->swap_edges(1, 2);
a61af66fc99e Initial load
duke
parents:
diff changeset
97 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
98 }
a61af66fc99e Initial load
duke
parents:
diff changeset
99 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
100 }
a61af66fc99e Initial load
duke
parents:
diff changeset
101
a61af66fc99e Initial load
duke
parents:
diff changeset
102 //------------------------------Idealize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
103 // If we get here, we assume we are associative!
a61af66fc99e Initial load
duke
parents:
diff changeset
104 Node *AddNode::Ideal(PhaseGVN *phase, bool can_reshape) {
a61af66fc99e Initial load
duke
parents:
diff changeset
105 const Type *t1 = phase->type( in(1) );
a61af66fc99e Initial load
duke
parents:
diff changeset
106 const Type *t2 = phase->type( in(2) );
a61af66fc99e Initial load
duke
parents:
diff changeset
107 int con_left = t1->singleton();
a61af66fc99e Initial load
duke
parents:
diff changeset
108 int con_right = t2->singleton();
a61af66fc99e Initial load
duke
parents:
diff changeset
109
a61af66fc99e Initial load
duke
parents:
diff changeset
110 // Check for commutative operation desired
a61af66fc99e Initial load
duke
parents:
diff changeset
111 if( commute(this,con_left,con_right) ) return this;
a61af66fc99e Initial load
duke
parents:
diff changeset
112
a61af66fc99e Initial load
duke
parents:
diff changeset
113 AddNode *progress = NULL; // Progress flag
a61af66fc99e Initial load
duke
parents:
diff changeset
114
a61af66fc99e Initial load
duke
parents:
diff changeset
115 // Convert "(x+1)+2" into "x+(1+2)". If the right input is a
a61af66fc99e Initial load
duke
parents:
diff changeset
116 // constant, and the left input is an add of a constant, flatten the
a61af66fc99e Initial load
duke
parents:
diff changeset
117 // expression tree.
a61af66fc99e Initial load
duke
parents:
diff changeset
118 Node *add1 = in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
119 Node *add2 = in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
120 int add1_op = add1->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
121 int this_op = Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
122 if( con_right && t2 != Type::TOP && // Right input is a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
123 add1_op == this_op ) { // Left input is an Add?
a61af66fc99e Initial load
duke
parents:
diff changeset
124
a61af66fc99e Initial load
duke
parents:
diff changeset
125 // Type of left _in right input
a61af66fc99e Initial load
duke
parents:
diff changeset
126 const Type *t12 = phase->type( add1->in(2) );
a61af66fc99e Initial load
duke
parents:
diff changeset
127 if( t12->singleton() && t12 != Type::TOP ) { // Left input is an add of a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
128 // Check for rare case of closed data cycle which can happen inside
a61af66fc99e Initial load
duke
parents:
diff changeset
129 // unreachable loops. In these cases the computation is undefined.
a61af66fc99e Initial load
duke
parents:
diff changeset
130 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
131 Node *add11 = add1->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
132 int add11_op = add11->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
133 if( (add1 == add1->in(1))
a61af66fc99e Initial load
duke
parents:
diff changeset
134 || (add11_op == this_op && add11->in(1) == add1) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
135 assert(false, "dead loop in AddNode::Ideal");
a61af66fc99e Initial load
duke
parents:
diff changeset
136 }
a61af66fc99e Initial load
duke
parents:
diff changeset
137 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
138 // The Add of the flattened expression
a61af66fc99e Initial load
duke
parents:
diff changeset
139 Node *x1 = add1->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
140 Node *x2 = phase->makecon( add1->as_Add()->add_ring( t2, t12 ));
a61af66fc99e Initial load
duke
parents:
diff changeset
141 PhaseIterGVN *igvn = phase->is_IterGVN();
a61af66fc99e Initial load
duke
parents:
diff changeset
142 if( igvn ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
143 set_req_X(2,x2,igvn);
a61af66fc99e Initial load
duke
parents:
diff changeset
144 set_req_X(1,x1,igvn);
a61af66fc99e Initial load
duke
parents:
diff changeset
145 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
146 set_req(2,x2);
a61af66fc99e Initial load
duke
parents:
diff changeset
147 set_req(1,x1);
a61af66fc99e Initial load
duke
parents:
diff changeset
148 }
a61af66fc99e Initial load
duke
parents:
diff changeset
149 progress = this; // Made progress
a61af66fc99e Initial load
duke
parents:
diff changeset
150 add1 = in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
151 add1_op = add1->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
152 }
a61af66fc99e Initial load
duke
parents:
diff changeset
153 }
a61af66fc99e Initial load
duke
parents:
diff changeset
154
a61af66fc99e Initial load
duke
parents:
diff changeset
155 // Convert "(x+1)+y" into "(x+y)+1". Push constants down the expression tree.
a61af66fc99e Initial load
duke
parents:
diff changeset
156 if( add1_op == this_op && !con_right ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
157 Node *a12 = add1->in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
158 const Type *t12 = phase->type( a12 );
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
159 if( t12->singleton() && t12 != Type::TOP && (add1 != add1->in(1)) &&
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
160 !(add1->in(1)->is_Phi() && add1->in(1)->as_Phi()->is_tripcount()) ) {
320
2b73d212b1fd 6676462: JVM sometimes would suddenly consume significant amount of memory
kvn
parents: 306
diff changeset
161 assert(add1->in(1) != this, "dead loop in AddNode::Ideal");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
162 add2 = add1->clone();
a61af66fc99e Initial load
duke
parents:
diff changeset
163 add2->set_req(2, in(2));
a61af66fc99e Initial load
duke
parents:
diff changeset
164 add2 = phase->transform(add2);
a61af66fc99e Initial load
duke
parents:
diff changeset
165 set_req(1, add2);
a61af66fc99e Initial load
duke
parents:
diff changeset
166 set_req(2, a12);
a61af66fc99e Initial load
duke
parents:
diff changeset
167 progress = this;
a61af66fc99e Initial load
duke
parents:
diff changeset
168 add2 = a12;
a61af66fc99e Initial load
duke
parents:
diff changeset
169 }
a61af66fc99e Initial load
duke
parents:
diff changeset
170 }
a61af66fc99e Initial load
duke
parents:
diff changeset
171
a61af66fc99e Initial load
duke
parents:
diff changeset
172 // Convert "x+(y+1)" into "(x+y)+1". Push constants down the expression tree.
a61af66fc99e Initial load
duke
parents:
diff changeset
173 int add2_op = add2->Opcode();
a61af66fc99e Initial load
duke
parents:
diff changeset
174 if( add2_op == this_op && !con_left ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
175 Node *a22 = add2->in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
176 const Type *t22 = phase->type( a22 );
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
177 if( t22->singleton() && t22 != Type::TOP && (add2 != add2->in(1)) &&
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
178 !(add2->in(1)->is_Phi() && add2->in(1)->as_Phi()->is_tripcount()) ) {
320
2b73d212b1fd 6676462: JVM sometimes would suddenly consume significant amount of memory
kvn
parents: 306
diff changeset
179 assert(add2->in(1) != this, "dead loop in AddNode::Ideal");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
180 Node *addx = add2->clone();
a61af66fc99e Initial load
duke
parents:
diff changeset
181 addx->set_req(1, in(1));
a61af66fc99e Initial load
duke
parents:
diff changeset
182 addx->set_req(2, add2->in(1));
a61af66fc99e Initial load
duke
parents:
diff changeset
183 addx = phase->transform(addx);
a61af66fc99e Initial load
duke
parents:
diff changeset
184 set_req(1, addx);
a61af66fc99e Initial load
duke
parents:
diff changeset
185 set_req(2, a22);
a61af66fc99e Initial load
duke
parents:
diff changeset
186 progress = this;
a61af66fc99e Initial load
duke
parents:
diff changeset
187 }
a61af66fc99e Initial load
duke
parents:
diff changeset
188 }
a61af66fc99e Initial load
duke
parents:
diff changeset
189
a61af66fc99e Initial load
duke
parents:
diff changeset
190 return progress;
a61af66fc99e Initial load
duke
parents:
diff changeset
191 }
a61af66fc99e Initial load
duke
parents:
diff changeset
192
a61af66fc99e Initial load
duke
parents:
diff changeset
193 //------------------------------Value-----------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
194 // An add node sums it's two _in. If one input is an RSD, we must mixin
a61af66fc99e Initial load
duke
parents:
diff changeset
195 // the other input's symbols.
a61af66fc99e Initial load
duke
parents:
diff changeset
196 const Type *AddNode::Value( PhaseTransform *phase ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
197 // Either input is TOP ==> the result is TOP
a61af66fc99e Initial load
duke
parents:
diff changeset
198 const Type *t1 = phase->type( in(1) );
a61af66fc99e Initial load
duke
parents:
diff changeset
199 const Type *t2 = phase->type( in(2) );
a61af66fc99e Initial load
duke
parents:
diff changeset
200 if( t1 == Type::TOP ) return Type::TOP;
a61af66fc99e Initial load
duke
parents:
diff changeset
201 if( t2 == Type::TOP ) return Type::TOP;
a61af66fc99e Initial load
duke
parents:
diff changeset
202
a61af66fc99e Initial load
duke
parents:
diff changeset
203 // Either input is BOTTOM ==> the result is the local BOTTOM
a61af66fc99e Initial load
duke
parents:
diff changeset
204 const Type *bot = bottom_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
205 if( (t1 == bot) || (t2 == bot) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
206 (t1 == Type::BOTTOM) || (t2 == Type::BOTTOM) )
a61af66fc99e Initial load
duke
parents:
diff changeset
207 return bot;
a61af66fc99e Initial load
duke
parents:
diff changeset
208
a61af66fc99e Initial load
duke
parents:
diff changeset
209 // Check for an addition involving the additive identity
a61af66fc99e Initial load
duke
parents:
diff changeset
210 const Type *tadd = add_of_identity( t1, t2 );
a61af66fc99e Initial load
duke
parents:
diff changeset
211 if( tadd ) return tadd;
a61af66fc99e Initial load
duke
parents:
diff changeset
212
a61af66fc99e Initial load
duke
parents:
diff changeset
213 return add_ring(t1,t2); // Local flavor of type addition
a61af66fc99e Initial load
duke
parents:
diff changeset
214 }
a61af66fc99e Initial load
duke
parents:
diff changeset
215
a61af66fc99e Initial load
duke
parents:
diff changeset
216 //------------------------------add_identity-----------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
217 // Check for addition of the identity
a61af66fc99e Initial load
duke
parents:
diff changeset
218 const Type *AddNode::add_of_identity( const Type *t1, const Type *t2 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
219 const Type *zero = add_id(); // The additive identity
a61af66fc99e Initial load
duke
parents:
diff changeset
220 if( t1->higher_equal( zero ) ) return t2;
a61af66fc99e Initial load
duke
parents:
diff changeset
221 if( t2->higher_equal( zero ) ) return t1;
a61af66fc99e Initial load
duke
parents:
diff changeset
222
a61af66fc99e Initial load
duke
parents:
diff changeset
223 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
224 }
a61af66fc99e Initial load
duke
parents:
diff changeset
225
a61af66fc99e Initial load
duke
parents:
diff changeset
226
a61af66fc99e Initial load
duke
parents:
diff changeset
227 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
228 //------------------------------Idealize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
229 Node *AddINode::Ideal(PhaseGVN *phase, bool can_reshape) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
230 Node* in1 = in(1);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
231 Node* in2 = in(2);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
232 int op1 = in1->Opcode();
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
233 int op2 = in2->Opcode();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
234 // Fold (con1-x)+con2 into (con1+con2)-x
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
235 if ( op1 == Op_AddI && op2 == Op_SubI ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
236 // Swap edges to try optimizations below
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
237 in1 = in2;
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
238 in2 = in(1);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
239 op1 = op2;
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
240 op2 = in2->Opcode();
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
241 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
242 if( op1 == Op_SubI ) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
243 const Type *t_sub1 = phase->type( in1->in(1) );
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
244 const Type *t_2 = phase->type( in2 );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
245 if( t_sub1->singleton() && t_2->singleton() && t_sub1 != Type::TOP && t_2 != Type::TOP )
a61af66fc99e Initial load
duke
parents:
diff changeset
246 return new (phase->C, 3) SubINode(phase->makecon( add_ring( t_sub1, t_2 ) ),
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
247 in1->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
248 // Convert "(a-b)+(c-d)" into "(a+c)-(b+d)"
a61af66fc99e Initial load
duke
parents:
diff changeset
249 if( op2 == Op_SubI ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
250 // Check for dead cycle: d = (a-b)+(c-d)
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
251 assert( in1->in(2) != this && in2->in(2) != this,
0
a61af66fc99e Initial load
duke
parents:
diff changeset
252 "dead loop in AddINode::Ideal" );
a61af66fc99e Initial load
duke
parents:
diff changeset
253 Node *sub = new (phase->C, 3) SubINode(NULL, NULL);
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
254 sub->init_req(1, phase->transform(new (phase->C, 3) AddINode(in1->in(1), in2->in(1) ) ));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
255 sub->init_req(2, phase->transform(new (phase->C, 3) AddINode(in1->in(2), in2->in(2) ) ));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
256 return sub;
a61af66fc99e Initial load
duke
parents:
diff changeset
257 }
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
258 // Convert "(a-b)+(b+c)" into "(a+c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
259 if( op2 == Op_AddI && in1->in(2) == in2->in(1) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
260 assert(in1->in(1) != this && in2->in(2) != this,"dead loop in AddINode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
261 return new (phase->C, 3) AddINode(in1->in(1), in2->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
262 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
263 // Convert "(a-b)+(c+b)" into "(a+c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
264 if( op2 == Op_AddI && in1->in(2) == in2->in(2) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
265 assert(in1->in(1) != this && in2->in(1) != this,"dead loop in AddINode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
266 return new (phase->C, 3) AddINode(in1->in(1), in2->in(1));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
267 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
268 // Convert "(a-b)+(b-c)" into "(a-c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
269 if( op2 == Op_SubI && in1->in(2) == in2->in(1) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
270 assert(in1->in(1) != this && in2->in(2) != this,"dead loop in AddINode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
271 return new (phase->C, 3) SubINode(in1->in(1), in2->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
272 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
273 // Convert "(a-b)+(c-a)" into "(c-b)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
274 if( op2 == Op_SubI && in1->in(1) == in2->in(2) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
275 assert(in1->in(2) != this && in2->in(1) != this,"dead loop in AddINode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
276 return new (phase->C, 3) SubINode(in2->in(1), in1->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
277 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
278 }
a61af66fc99e Initial load
duke
parents:
diff changeset
279
a61af66fc99e Initial load
duke
parents:
diff changeset
280 // Convert "x+(0-y)" into "(x-y)"
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
281 if( op2 == Op_SubI && phase->type(in2->in(1)) == TypeInt::ZERO )
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
282 return new (phase->C, 3) SubINode(in1, in2->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
283
a61af66fc99e Initial load
duke
parents:
diff changeset
284 // Convert "(0-y)+x" into "(x-y)"
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
285 if( op1 == Op_SubI && phase->type(in1->in(1)) == TypeInt::ZERO )
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
286 return new (phase->C, 3) SubINode( in2, in1->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
287
a61af66fc99e Initial load
duke
parents:
diff changeset
288 // Convert (x>>>z)+y into (x+(y<<z))>>>z for small constant z and y.
a61af66fc99e Initial load
duke
parents:
diff changeset
289 // Helps with array allocation math constant folding
a61af66fc99e Initial load
duke
parents:
diff changeset
290 // See 4790063:
a61af66fc99e Initial load
duke
parents:
diff changeset
291 // Unrestricted transformation is unsafe for some runtime values of 'x'
a61af66fc99e Initial load
duke
parents:
diff changeset
292 // ( x == 0, z == 1, y == -1 ) fails
a61af66fc99e Initial load
duke
parents:
diff changeset
293 // ( x == -5, z == 1, y == 1 ) fails
a61af66fc99e Initial load
duke
parents:
diff changeset
294 // Transform works for small z and small negative y when the addition
a61af66fc99e Initial load
duke
parents:
diff changeset
295 // (x + (y << z)) does not cross zero.
a61af66fc99e Initial load
duke
parents:
diff changeset
296 // Implement support for negative y and (x >= -(y << z))
a61af66fc99e Initial load
duke
parents:
diff changeset
297 // Have not observed cases where type information exists to support
a61af66fc99e Initial load
duke
parents:
diff changeset
298 // positive y and (x <= -(y << z))
a61af66fc99e Initial load
duke
parents:
diff changeset
299 if( op1 == Op_URShiftI && op2 == Op_ConI &&
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
300 in1->in(2)->Opcode() == Op_ConI ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
301 jint z = phase->type( in1->in(2) )->is_int()->get_con() & 0x1f; // only least significant 5 bits matter
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
302 jint y = phase->type( in2 )->is_int()->get_con();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
303
a61af66fc99e Initial load
duke
parents:
diff changeset
304 if( z < 5 && -5 < y && y < 0 ) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
305 const Type *t_in11 = phase->type(in1->in(1));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
306 if( t_in11 != Type::TOP && (t_in11->is_int()->_lo >= -(y << z)) ) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
307 Node *a = phase->transform( new (phase->C, 3) AddINode( in1->in(1), phase->intcon(y<<z) ) );
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
308 return new (phase->C, 3) URShiftINode( a, in1->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
309 }
a61af66fc99e Initial load
duke
parents:
diff changeset
310 }
a61af66fc99e Initial load
duke
parents:
diff changeset
311 }
a61af66fc99e Initial load
duke
parents:
diff changeset
312
a61af66fc99e Initial load
duke
parents:
diff changeset
313 return AddNode::Ideal(phase, can_reshape);
a61af66fc99e Initial load
duke
parents:
diff changeset
314 }
a61af66fc99e Initial load
duke
parents:
diff changeset
315
a61af66fc99e Initial load
duke
parents:
diff changeset
316
a61af66fc99e Initial load
duke
parents:
diff changeset
317 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
318 // Fold (x-y)+y OR y+(x-y) into x
a61af66fc99e Initial load
duke
parents:
diff changeset
319 Node *AddINode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
320 if( in(1)->Opcode() == Op_SubI && phase->eqv(in(1)->in(2),in(2)) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
321 return in(1)->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
322 }
a61af66fc99e Initial load
duke
parents:
diff changeset
323 else if( in(2)->Opcode() == Op_SubI && phase->eqv(in(2)->in(2),in(1)) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
324 return in(2)->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
325 }
a61af66fc99e Initial load
duke
parents:
diff changeset
326 return AddNode::Identity(phase);
a61af66fc99e Initial load
duke
parents:
diff changeset
327 }
a61af66fc99e Initial load
duke
parents:
diff changeset
328
a61af66fc99e Initial load
duke
parents:
diff changeset
329
a61af66fc99e Initial load
duke
parents:
diff changeset
330 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
331 // Supplied function returns the sum of the inputs. Guaranteed never
a61af66fc99e Initial load
duke
parents:
diff changeset
332 // to be passed a TOP or BOTTOM type, these are filtered out by
a61af66fc99e Initial load
duke
parents:
diff changeset
333 // pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
334 const Type *AddINode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
335 const TypeInt *r0 = t0->is_int(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
336 const TypeInt *r1 = t1->is_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
337 int lo = r0->_lo + r1->_lo;
a61af66fc99e Initial load
duke
parents:
diff changeset
338 int hi = r0->_hi + r1->_hi;
a61af66fc99e Initial load
duke
parents:
diff changeset
339 if( !(r0->is_con() && r1->is_con()) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
340 // Not both constants, compute approximate result
a61af66fc99e Initial load
duke
parents:
diff changeset
341 if( (r0->_lo & r1->_lo) < 0 && lo >= 0 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
342 lo = min_jint; hi = max_jint; // Underflow on the low side
a61af66fc99e Initial load
duke
parents:
diff changeset
343 }
a61af66fc99e Initial load
duke
parents:
diff changeset
344 if( (~(r0->_hi | r1->_hi)) < 0 && hi < 0 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
345 lo = min_jint; hi = max_jint; // Overflow on the high side
a61af66fc99e Initial load
duke
parents:
diff changeset
346 }
a61af66fc99e Initial load
duke
parents:
diff changeset
347 if( lo > hi ) { // Handle overflow
a61af66fc99e Initial load
duke
parents:
diff changeset
348 lo = min_jint; hi = max_jint;
a61af66fc99e Initial load
duke
parents:
diff changeset
349 }
a61af66fc99e Initial load
duke
parents:
diff changeset
350 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
351 // both constants, compute precise result using 'lo' and 'hi'
a61af66fc99e Initial load
duke
parents:
diff changeset
352 // Semantics define overflow and underflow for integer addition
a61af66fc99e Initial load
duke
parents:
diff changeset
353 // as expected. In particular: 0x80000000 + 0x80000000 --> 0x0
a61af66fc99e Initial load
duke
parents:
diff changeset
354 }
a61af66fc99e Initial load
duke
parents:
diff changeset
355 return TypeInt::make( lo, hi, MAX2(r0->_widen,r1->_widen) );
a61af66fc99e Initial load
duke
parents:
diff changeset
356 }
a61af66fc99e Initial load
duke
parents:
diff changeset
357
a61af66fc99e Initial load
duke
parents:
diff changeset
358
a61af66fc99e Initial load
duke
parents:
diff changeset
359 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
360 //------------------------------Idealize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
361 Node *AddLNode::Ideal(PhaseGVN *phase, bool can_reshape) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
362 Node* in1 = in(1);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
363 Node* in2 = in(2);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
364 int op1 = in1->Opcode();
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
365 int op2 = in2->Opcode();
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
366 // Fold (con1-x)+con2 into (con1+con2)-x
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
367 if ( op1 == Op_AddL && op2 == Op_SubL ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
368 // Swap edges to try optimizations below
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
369 in1 = in2;
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
370 in2 = in(1);
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
371 op1 = op2;
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
372 op2 = in2->Opcode();
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
373 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
374 // Fold (con1-x)+con2 into (con1+con2)-x
a61af66fc99e Initial load
duke
parents:
diff changeset
375 if( op1 == Op_SubL ) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
376 const Type *t_sub1 = phase->type( in1->in(1) );
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
377 const Type *t_2 = phase->type( in2 );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
378 if( t_sub1->singleton() && t_2->singleton() && t_sub1 != Type::TOP && t_2 != Type::TOP )
a61af66fc99e Initial load
duke
parents:
diff changeset
379 return new (phase->C, 3) SubLNode(phase->makecon( add_ring( t_sub1, t_2 ) ),
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
380 in1->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
381 // Convert "(a-b)+(c-d)" into "(a+c)-(b+d)"
a61af66fc99e Initial load
duke
parents:
diff changeset
382 if( op2 == Op_SubL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
383 // Check for dead cycle: d = (a-b)+(c-d)
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
384 assert( in1->in(2) != this && in2->in(2) != this,
0
a61af66fc99e Initial load
duke
parents:
diff changeset
385 "dead loop in AddLNode::Ideal" );
a61af66fc99e Initial load
duke
parents:
diff changeset
386 Node *sub = new (phase->C, 3) SubLNode(NULL, NULL);
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
387 sub->init_req(1, phase->transform(new (phase->C, 3) AddLNode(in1->in(1), in2->in(1) ) ));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
388 sub->init_req(2, phase->transform(new (phase->C, 3) AddLNode(in1->in(2), in2->in(2) ) ));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
389 return sub;
a61af66fc99e Initial load
duke
parents:
diff changeset
390 }
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
391 // Convert "(a-b)+(b+c)" into "(a+c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
392 if( op2 == Op_AddL && in1->in(2) == in2->in(1) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
393 assert(in1->in(1) != this && in2->in(2) != this,"dead loop in AddLNode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
394 return new (phase->C, 3) AddLNode(in1->in(1), in2->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
395 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
396 // Convert "(a-b)+(c+b)" into "(a+c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
397 if( op2 == Op_AddL && in1->in(2) == in2->in(2) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
398 assert(in1->in(1) != this && in2->in(1) != this,"dead loop in AddLNode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
399 return new (phase->C, 3) AddLNode(in1->in(1), in2->in(1));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
400 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
401 // Convert "(a-b)+(b-c)" into "(a-c)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
402 if( op2 == Op_SubL && in1->in(2) == in2->in(1) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
403 assert(in1->in(1) != this && in2->in(2) != this,"dead loop in AddLNode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
404 return new (phase->C, 3) SubLNode(in1->in(1), in2->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
405 }
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
406 // Convert "(a-b)+(c-a)" into "(c-b)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
407 if( op2 == Op_SubL && in1->in(1) == in1->in(2) ) {
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
408 assert(in1->in(2) != this && in2->in(1) != this,"dead loop in AddLNode::Ideal");
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
409 return new (phase->C, 3) SubLNode(in2->in(1), in1->in(2));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
410 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
411 }
a61af66fc99e Initial load
duke
parents:
diff changeset
412
a61af66fc99e Initial load
duke
parents:
diff changeset
413 // Convert "x+(0-y)" into "(x-y)"
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
414 if( op2 == Op_SubL && phase->type(in2->in(1)) == TypeLong::ZERO )
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
415 return new (phase->C, 3) SubLNode( in1, in2->in(2) );
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
416
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
417 // Convert "(0-y)+x" into "(x-y)"
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
418 if( op1 == Op_SubL && phase->type(in1->in(1)) == TypeInt::ZERO )
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
419 return new (phase->C, 3) SubLNode( in2, in1->in(2) );
0
a61af66fc99e Initial load
duke
parents:
diff changeset
420
a61af66fc99e Initial load
duke
parents:
diff changeset
421 // Convert "X+X+X+X+X...+X+Y" into "k*X+Y" or really convert "X+(X+Y)"
a61af66fc99e Initial load
duke
parents:
diff changeset
422 // into "(X<<1)+Y" and let shift-folding happen.
a61af66fc99e Initial load
duke
parents:
diff changeset
423 if( op2 == Op_AddL &&
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
424 in2->in(1) == in1 &&
0
a61af66fc99e Initial load
duke
parents:
diff changeset
425 op1 != Op_ConL &&
a61af66fc99e Initial load
duke
parents:
diff changeset
426 0 ) {
400
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
427 Node *shift = phase->transform(new (phase->C, 3) LShiftLNode(in1,phase->intcon(1)));
cc80376deb0c 6667595: Set probability FAIR for pre-, post- loops and ALWAYS for main loop
kvn
parents: 320
diff changeset
428 return new (phase->C, 3) AddLNode(shift,in2->in(2));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
429 }
a61af66fc99e Initial load
duke
parents:
diff changeset
430
a61af66fc99e Initial load
duke
parents:
diff changeset
431 return AddNode::Ideal(phase, can_reshape);
a61af66fc99e Initial load
duke
parents:
diff changeset
432 }
a61af66fc99e Initial load
duke
parents:
diff changeset
433
a61af66fc99e Initial load
duke
parents:
diff changeset
434
a61af66fc99e Initial load
duke
parents:
diff changeset
435 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
436 // Fold (x-y)+y OR y+(x-y) into x
a61af66fc99e Initial load
duke
parents:
diff changeset
437 Node *AddLNode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
438 if( in(1)->Opcode() == Op_SubL && phase->eqv(in(1)->in(2),in(2)) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
439 return in(1)->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
440 }
a61af66fc99e Initial load
duke
parents:
diff changeset
441 else if( in(2)->Opcode() == Op_SubL && phase->eqv(in(2)->in(2),in(1)) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
442 return in(2)->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
443 }
a61af66fc99e Initial load
duke
parents:
diff changeset
444 return AddNode::Identity(phase);
a61af66fc99e Initial load
duke
parents:
diff changeset
445 }
a61af66fc99e Initial load
duke
parents:
diff changeset
446
a61af66fc99e Initial load
duke
parents:
diff changeset
447
a61af66fc99e Initial load
duke
parents:
diff changeset
448 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
449 // Supplied function returns the sum of the inputs. Guaranteed never
a61af66fc99e Initial load
duke
parents:
diff changeset
450 // to be passed a TOP or BOTTOM type, these are filtered out by
a61af66fc99e Initial load
duke
parents:
diff changeset
451 // pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
452 const Type *AddLNode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
453 const TypeLong *r0 = t0->is_long(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
454 const TypeLong *r1 = t1->is_long();
a61af66fc99e Initial load
duke
parents:
diff changeset
455 jlong lo = r0->_lo + r1->_lo;
a61af66fc99e Initial load
duke
parents:
diff changeset
456 jlong hi = r0->_hi + r1->_hi;
a61af66fc99e Initial load
duke
parents:
diff changeset
457 if( !(r0->is_con() && r1->is_con()) ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
458 // Not both constants, compute approximate result
a61af66fc99e Initial load
duke
parents:
diff changeset
459 if( (r0->_lo & r1->_lo) < 0 && lo >= 0 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
460 lo =min_jlong; hi = max_jlong; // Underflow on the low side
a61af66fc99e Initial load
duke
parents:
diff changeset
461 }
a61af66fc99e Initial load
duke
parents:
diff changeset
462 if( (~(r0->_hi | r1->_hi)) < 0 && hi < 0 ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
463 lo = min_jlong; hi = max_jlong; // Overflow on the high side
a61af66fc99e Initial load
duke
parents:
diff changeset
464 }
a61af66fc99e Initial load
duke
parents:
diff changeset
465 if( lo > hi ) { // Handle overflow
a61af66fc99e Initial load
duke
parents:
diff changeset
466 lo = min_jlong; hi = max_jlong;
a61af66fc99e Initial load
duke
parents:
diff changeset
467 }
a61af66fc99e Initial load
duke
parents:
diff changeset
468 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
469 // both constants, compute precise result using 'lo' and 'hi'
a61af66fc99e Initial load
duke
parents:
diff changeset
470 // Semantics define overflow and underflow for integer addition
a61af66fc99e Initial load
duke
parents:
diff changeset
471 // as expected. In particular: 0x80000000 + 0x80000000 --> 0x0
a61af66fc99e Initial load
duke
parents:
diff changeset
472 }
a61af66fc99e Initial load
duke
parents:
diff changeset
473 return TypeLong::make( lo, hi, MAX2(r0->_widen,r1->_widen) );
a61af66fc99e Initial load
duke
parents:
diff changeset
474 }
a61af66fc99e Initial load
duke
parents:
diff changeset
475
a61af66fc99e Initial load
duke
parents:
diff changeset
476
a61af66fc99e Initial load
duke
parents:
diff changeset
477 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
478 //------------------------------add_of_identity--------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
479 // Check for addition of the identity
a61af66fc99e Initial load
duke
parents:
diff changeset
480 const Type *AddFNode::add_of_identity( const Type *t1, const Type *t2 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
481 // x ADD 0 should return x unless 'x' is a -zero
a61af66fc99e Initial load
duke
parents:
diff changeset
482 //
a61af66fc99e Initial load
duke
parents:
diff changeset
483 // const Type *zero = add_id(); // The additive identity
a61af66fc99e Initial load
duke
parents:
diff changeset
484 // jfloat f1 = t1->getf();
a61af66fc99e Initial load
duke
parents:
diff changeset
485 // jfloat f2 = t2->getf();
a61af66fc99e Initial load
duke
parents:
diff changeset
486 //
a61af66fc99e Initial load
duke
parents:
diff changeset
487 // if( t1->higher_equal( zero ) ) return t2;
a61af66fc99e Initial load
duke
parents:
diff changeset
488 // if( t2->higher_equal( zero ) ) return t1;
a61af66fc99e Initial load
duke
parents:
diff changeset
489
a61af66fc99e Initial load
duke
parents:
diff changeset
490 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
491 }
a61af66fc99e Initial load
duke
parents:
diff changeset
492
a61af66fc99e Initial load
duke
parents:
diff changeset
493 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
494 // Supplied function returns the sum of the inputs.
a61af66fc99e Initial load
duke
parents:
diff changeset
495 // This also type-checks the inputs for sanity. Guaranteed never to
a61af66fc99e Initial load
duke
parents:
diff changeset
496 // be passed a TOP or BOTTOM type, these are filtered out by pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
497 const Type *AddFNode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
498 // We must be adding 2 float constants.
a61af66fc99e Initial load
duke
parents:
diff changeset
499 return TypeF::make( t0->getf() + t1->getf() );
a61af66fc99e Initial load
duke
parents:
diff changeset
500 }
a61af66fc99e Initial load
duke
parents:
diff changeset
501
a61af66fc99e Initial load
duke
parents:
diff changeset
502 //------------------------------Ideal------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
503 Node *AddFNode::Ideal(PhaseGVN *phase, bool can_reshape) {
a61af66fc99e Initial load
duke
parents:
diff changeset
504 if( IdealizedNumerics && !phase->C->method()->is_strict() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
505 return AddNode::Ideal(phase, can_reshape); // commutative and associative transforms
a61af66fc99e Initial load
duke
parents:
diff changeset
506 }
a61af66fc99e Initial load
duke
parents:
diff changeset
507
a61af66fc99e Initial load
duke
parents:
diff changeset
508 // Floating point additions are not associative because of boundary conditions (infinity)
a61af66fc99e Initial load
duke
parents:
diff changeset
509 return commute(this,
a61af66fc99e Initial load
duke
parents:
diff changeset
510 phase->type( in(1) )->singleton(),
a61af66fc99e Initial load
duke
parents:
diff changeset
511 phase->type( in(2) )->singleton() ) ? this : NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
512 }
a61af66fc99e Initial load
duke
parents:
diff changeset
513
a61af66fc99e Initial load
duke
parents:
diff changeset
514
a61af66fc99e Initial load
duke
parents:
diff changeset
515 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
516 //------------------------------add_of_identity--------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
517 // Check for addition of the identity
a61af66fc99e Initial load
duke
parents:
diff changeset
518 const Type *AddDNode::add_of_identity( const Type *t1, const Type *t2 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
519 // x ADD 0 should return x unless 'x' is a -zero
a61af66fc99e Initial load
duke
parents:
diff changeset
520 //
a61af66fc99e Initial load
duke
parents:
diff changeset
521 // const Type *zero = add_id(); // The additive identity
a61af66fc99e Initial load
duke
parents:
diff changeset
522 // jfloat f1 = t1->getf();
a61af66fc99e Initial load
duke
parents:
diff changeset
523 // jfloat f2 = t2->getf();
a61af66fc99e Initial load
duke
parents:
diff changeset
524 //
a61af66fc99e Initial load
duke
parents:
diff changeset
525 // if( t1->higher_equal( zero ) ) return t2;
a61af66fc99e Initial load
duke
parents:
diff changeset
526 // if( t2->higher_equal( zero ) ) return t1;
a61af66fc99e Initial load
duke
parents:
diff changeset
527
a61af66fc99e Initial load
duke
parents:
diff changeset
528 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
529 }
a61af66fc99e Initial load
duke
parents:
diff changeset
530 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
531 // Supplied function returns the sum of the inputs.
a61af66fc99e Initial load
duke
parents:
diff changeset
532 // This also type-checks the inputs for sanity. Guaranteed never to
a61af66fc99e Initial load
duke
parents:
diff changeset
533 // be passed a TOP or BOTTOM type, these are filtered out by pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
534 const Type *AddDNode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
535 // We must be adding 2 double constants.
a61af66fc99e Initial load
duke
parents:
diff changeset
536 return TypeD::make( t0->getd() + t1->getd() );
a61af66fc99e Initial load
duke
parents:
diff changeset
537 }
a61af66fc99e Initial load
duke
parents:
diff changeset
538
a61af66fc99e Initial load
duke
parents:
diff changeset
539 //------------------------------Ideal------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
540 Node *AddDNode::Ideal(PhaseGVN *phase, bool can_reshape) {
a61af66fc99e Initial load
duke
parents:
diff changeset
541 if( IdealizedNumerics && !phase->C->method()->is_strict() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
542 return AddNode::Ideal(phase, can_reshape); // commutative and associative transforms
a61af66fc99e Initial load
duke
parents:
diff changeset
543 }
a61af66fc99e Initial load
duke
parents:
diff changeset
544
a61af66fc99e Initial load
duke
parents:
diff changeset
545 // Floating point additions are not associative because of boundary conditions (infinity)
a61af66fc99e Initial load
duke
parents:
diff changeset
546 return commute(this,
a61af66fc99e Initial load
duke
parents:
diff changeset
547 phase->type( in(1) )->singleton(),
a61af66fc99e Initial load
duke
parents:
diff changeset
548 phase->type( in(2) )->singleton() ) ? this : NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
549 }
a61af66fc99e Initial load
duke
parents:
diff changeset
550
a61af66fc99e Initial load
duke
parents:
diff changeset
551
a61af66fc99e Initial load
duke
parents:
diff changeset
552 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
553 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
554 // If one input is a constant 0, return the other input.
a61af66fc99e Initial load
duke
parents:
diff changeset
555 Node *AddPNode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
556 return ( phase->type( in(Offset) )->higher_equal( TypeX_ZERO ) ) ? in(Address) : this;
a61af66fc99e Initial load
duke
parents:
diff changeset
557 }
a61af66fc99e Initial load
duke
parents:
diff changeset
558
a61af66fc99e Initial load
duke
parents:
diff changeset
559 //------------------------------Idealize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
560 Node *AddPNode::Ideal(PhaseGVN *phase, bool can_reshape) {
a61af66fc99e Initial load
duke
parents:
diff changeset
561 // Bail out if dead inputs
a61af66fc99e Initial load
duke
parents:
diff changeset
562 if( phase->type( in(Address) ) == Type::TOP ) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
563
a61af66fc99e Initial load
duke
parents:
diff changeset
564 // If the left input is an add of a constant, flatten the expression tree.
a61af66fc99e Initial load
duke
parents:
diff changeset
565 const Node *n = in(Address);
a61af66fc99e Initial load
duke
parents:
diff changeset
566 if (n->is_AddP() && n->in(Base) == in(Base)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
567 const AddPNode *addp = n->as_AddP(); // Left input is an AddP
a61af66fc99e Initial load
duke
parents:
diff changeset
568 assert( !addp->in(Address)->is_AddP() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
569 addp->in(Address)->as_AddP() != addp,
a61af66fc99e Initial load
duke
parents:
diff changeset
570 "dead loop in AddPNode::Ideal" );
a61af66fc99e Initial load
duke
parents:
diff changeset
571 // Type of left input's right input
a61af66fc99e Initial load
duke
parents:
diff changeset
572 const Type *t = phase->type( addp->in(Offset) );
a61af66fc99e Initial load
duke
parents:
diff changeset
573 if( t == Type::TOP ) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
574 const TypeX *t12 = t->is_intptr_t();
a61af66fc99e Initial load
duke
parents:
diff changeset
575 if( t12->is_con() ) { // Left input is an add of a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
576 // If the right input is a constant, combine constants
a61af66fc99e Initial load
duke
parents:
diff changeset
577 const Type *temp_t2 = phase->type( in(Offset) );
a61af66fc99e Initial load
duke
parents:
diff changeset
578 if( temp_t2 == Type::TOP ) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
579 const TypeX *t2 = temp_t2->is_intptr_t();
32
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
580 Node* address;
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
581 Node* offset;
0
a61af66fc99e Initial load
duke
parents:
diff changeset
582 if( t2->is_con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
583 // The Add of the flattened expression
32
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
584 address = addp->in(Address);
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
585 offset = phase->MakeConX(t2->get_con() + t12->get_con());
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
586 } else {
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
587 // Else move the constant to the right. ((A+con)+B) into ((A+B)+con)
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
588 address = phase->transform(new (phase->C, 4) AddPNode(in(Base),addp->in(Address),in(Offset)));
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
589 offset = addp->in(Offset);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
590 }
32
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
591 PhaseIterGVN *igvn = phase->is_IterGVN();
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
592 if( igvn ) {
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
593 set_req_X(Address,address,igvn);
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
594 set_req_X(Offset,offset,igvn);
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
595 } else {
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
596 set_req(Address,address);
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
597 set_req(Offset,offset);
4d428c5b4cb3 6667573: Use set_req_X() in AddPNode::Ideal() for Iterative GVN
kvn
parents: 17
diff changeset
598 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
599 return this;
a61af66fc99e Initial load
duke
parents:
diff changeset
600 }
a61af66fc99e Initial load
duke
parents:
diff changeset
601 }
a61af66fc99e Initial load
duke
parents:
diff changeset
602
a61af66fc99e Initial load
duke
parents:
diff changeset
603 // Raw pointers?
a61af66fc99e Initial load
duke
parents:
diff changeset
604 if( in(Base)->bottom_type() == Type::TOP ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
605 // If this is a NULL+long form (from unsafe accesses), switch to a rawptr.
a61af66fc99e Initial load
duke
parents:
diff changeset
606 if (phase->type(in(Address)) == TypePtr::NULL_PTR) {
a61af66fc99e Initial load
duke
parents:
diff changeset
607 Node* offset = in(Offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
608 return new (phase->C, 2) CastX2PNode(offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
609 }
a61af66fc99e Initial load
duke
parents:
diff changeset
610 }
a61af66fc99e Initial load
duke
parents:
diff changeset
611
a61af66fc99e Initial load
duke
parents:
diff changeset
612 // If the right is an add of a constant, push the offset down.
a61af66fc99e Initial load
duke
parents:
diff changeset
613 // Convert: (ptr + (offset+con)) into (ptr+offset)+con.
a61af66fc99e Initial load
duke
parents:
diff changeset
614 // The idea is to merge array_base+scaled_index groups together,
a61af66fc99e Initial load
duke
parents:
diff changeset
615 // and only have different constant offsets from the same base.
a61af66fc99e Initial load
duke
parents:
diff changeset
616 const Node *add = in(Offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
617 if( add->Opcode() == Op_AddX && add->in(1) != add ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
618 const Type *t22 = phase->type( add->in(2) );
a61af66fc99e Initial load
duke
parents:
diff changeset
619 if( t22->singleton() && (t22 != Type::TOP) ) { // Right input is an add of a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
620 set_req(Address, phase->transform(new (phase->C, 4) AddPNode(in(Base),in(Address),add->in(1))));
a61af66fc99e Initial load
duke
parents:
diff changeset
621 set_req(Offset, add->in(2));
a61af66fc99e Initial load
duke
parents:
diff changeset
622 return this; // Made progress
a61af66fc99e Initial load
duke
parents:
diff changeset
623 }
a61af66fc99e Initial load
duke
parents:
diff changeset
624 }
a61af66fc99e Initial load
duke
parents:
diff changeset
625
a61af66fc99e Initial load
duke
parents:
diff changeset
626 return NULL; // No progress
a61af66fc99e Initial load
duke
parents:
diff changeset
627 }
a61af66fc99e Initial load
duke
parents:
diff changeset
628
a61af66fc99e Initial load
duke
parents:
diff changeset
629 //------------------------------bottom_type------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
630 // Bottom-type is the pointer-type with unknown offset.
a61af66fc99e Initial load
duke
parents:
diff changeset
631 const Type *AddPNode::bottom_type() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
632 if (in(Address) == NULL) return TypePtr::BOTTOM;
a61af66fc99e Initial load
duke
parents:
diff changeset
633 const TypePtr *tp = in(Address)->bottom_type()->isa_ptr();
a61af66fc99e Initial load
duke
parents:
diff changeset
634 if( !tp ) return Type::TOP; // TOP input means TOP output
a61af66fc99e Initial load
duke
parents:
diff changeset
635 assert( in(Offset)->Opcode() != Op_ConP, "" );
a61af66fc99e Initial load
duke
parents:
diff changeset
636 const Type *t = in(Offset)->bottom_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
637 if( t == Type::TOP )
a61af66fc99e Initial load
duke
parents:
diff changeset
638 return tp->add_offset(Type::OffsetTop);
a61af66fc99e Initial load
duke
parents:
diff changeset
639 const TypeX *tx = t->is_intptr_t();
a61af66fc99e Initial load
duke
parents:
diff changeset
640 intptr_t txoffset = Type::OffsetBot;
a61af66fc99e Initial load
duke
parents:
diff changeset
641 if (tx->is_con()) { // Left input is an add of a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
642 txoffset = tx->get_con();
a61af66fc99e Initial load
duke
parents:
diff changeset
643 }
a61af66fc99e Initial load
duke
parents:
diff changeset
644 return tp->add_offset(txoffset);
a61af66fc99e Initial load
duke
parents:
diff changeset
645 }
a61af66fc99e Initial load
duke
parents:
diff changeset
646
a61af66fc99e Initial load
duke
parents:
diff changeset
647 //------------------------------Value------------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
648 const Type *AddPNode::Value( PhaseTransform *phase ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
649 // Either input is TOP ==> the result is TOP
a61af66fc99e Initial load
duke
parents:
diff changeset
650 const Type *t1 = phase->type( in(Address) );
a61af66fc99e Initial load
duke
parents:
diff changeset
651 const Type *t2 = phase->type( in(Offset) );
a61af66fc99e Initial load
duke
parents:
diff changeset
652 if( t1 == Type::TOP ) return Type::TOP;
a61af66fc99e Initial load
duke
parents:
diff changeset
653 if( t2 == Type::TOP ) return Type::TOP;
a61af66fc99e Initial load
duke
parents:
diff changeset
654
a61af66fc99e Initial load
duke
parents:
diff changeset
655 // Left input is a pointer
a61af66fc99e Initial load
duke
parents:
diff changeset
656 const TypePtr *p1 = t1->isa_ptr();
a61af66fc99e Initial load
duke
parents:
diff changeset
657 // Right input is an int
a61af66fc99e Initial load
duke
parents:
diff changeset
658 const TypeX *p2 = t2->is_intptr_t();
a61af66fc99e Initial load
duke
parents:
diff changeset
659 // Add 'em
a61af66fc99e Initial load
duke
parents:
diff changeset
660 intptr_t p2offset = Type::OffsetBot;
a61af66fc99e Initial load
duke
parents:
diff changeset
661 if (p2->is_con()) { // Left input is an add of a constant?
a61af66fc99e Initial load
duke
parents:
diff changeset
662 p2offset = p2->get_con();
a61af66fc99e Initial load
duke
parents:
diff changeset
663 }
a61af66fc99e Initial load
duke
parents:
diff changeset
664 return p1->add_offset(p2offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
665 }
a61af66fc99e Initial load
duke
parents:
diff changeset
666
a61af66fc99e Initial load
duke
parents:
diff changeset
667 //------------------------Ideal_base_and_offset--------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
668 // Split an oop pointer into a base and offset.
a61af66fc99e Initial load
duke
parents:
diff changeset
669 // (The offset might be Type::OffsetBot in the case of an array.)
a61af66fc99e Initial load
duke
parents:
diff changeset
670 // Return the base, or NULL if failure.
a61af66fc99e Initial load
duke
parents:
diff changeset
671 Node* AddPNode::Ideal_base_and_offset(Node* ptr, PhaseTransform* phase,
a61af66fc99e Initial load
duke
parents:
diff changeset
672 // second return value:
a61af66fc99e Initial load
duke
parents:
diff changeset
673 intptr_t& offset) {
a61af66fc99e Initial load
duke
parents:
diff changeset
674 if (ptr->is_AddP()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
675 Node* base = ptr->in(AddPNode::Base);
a61af66fc99e Initial load
duke
parents:
diff changeset
676 Node* addr = ptr->in(AddPNode::Address);
a61af66fc99e Initial load
duke
parents:
diff changeset
677 Node* offs = ptr->in(AddPNode::Offset);
a61af66fc99e Initial load
duke
parents:
diff changeset
678 if (base == addr || base->is_top()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
679 offset = phase->find_intptr_t_con(offs, Type::OffsetBot);
a61af66fc99e Initial load
duke
parents:
diff changeset
680 if (offset != Type::OffsetBot) {
a61af66fc99e Initial load
duke
parents:
diff changeset
681 return addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
682 }
a61af66fc99e Initial load
duke
parents:
diff changeset
683 }
a61af66fc99e Initial load
duke
parents:
diff changeset
684 }
a61af66fc99e Initial load
duke
parents:
diff changeset
685 offset = Type::OffsetBot;
a61af66fc99e Initial load
duke
parents:
diff changeset
686 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
687 }
a61af66fc99e Initial load
duke
parents:
diff changeset
688
17
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
689 //------------------------------unpack_offsets----------------------------------
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
690 // Collect the AddP offset values into the elements array, giving up
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
691 // if there are more than length.
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
692 int AddPNode::unpack_offsets(Node* elements[], int length) {
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
693 int count = 0;
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
694 Node* addr = this;
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
695 Node* base = addr->in(AddPNode::Base);
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
696 while (addr->is_AddP()) {
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
697 if (addr->in(AddPNode::Base) != base) {
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
698 // give up
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
699 return -1;
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
700 }
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
701 elements[count++] = addr->in(AddPNode::Offset);
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
702 if (count == length) {
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
703 // give up
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
704 return -1;
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
705 }
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
706 addr = addr->in(AddPNode::Address);
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
707 }
1763
d6f45b55c972 4809552: Optimize Arrays.fill(...)
never
parents: 1552
diff changeset
708 if (addr != base) {
d6f45b55c972 4809552: Optimize Arrays.fill(...)
never
parents: 1552
diff changeset
709 return -1;
d6f45b55c972 4809552: Optimize Arrays.fill(...)
never
parents: 1552
diff changeset
710 }
17
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
711 return count;
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
712 }
ff5961f4c095 6395208: Elide autoboxing for calls to HashMap.get(int) and HashMap.get(long)
never
parents: 0
diff changeset
713
0
a61af66fc99e Initial load
duke
parents:
diff changeset
714 //------------------------------match_edge-------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
715 // Do we Match on this edge index or not? Do not match base pointer edge
a61af66fc99e Initial load
duke
parents:
diff changeset
716 uint AddPNode::match_edge(uint idx) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
717 return idx > Base;
a61af66fc99e Initial load
duke
parents:
diff changeset
718 }
a61af66fc99e Initial load
duke
parents:
diff changeset
719
a61af66fc99e Initial load
duke
parents:
diff changeset
720 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
721 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
722 Node *OrINode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
723 // x | x => x
a61af66fc99e Initial load
duke
parents:
diff changeset
724 if (phase->eqv(in(1), in(2))) {
a61af66fc99e Initial load
duke
parents:
diff changeset
725 return in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
726 }
a61af66fc99e Initial load
duke
parents:
diff changeset
727
a61af66fc99e Initial load
duke
parents:
diff changeset
728 return AddNode::Identity(phase);
a61af66fc99e Initial load
duke
parents:
diff changeset
729 }
a61af66fc99e Initial load
duke
parents:
diff changeset
730
a61af66fc99e Initial load
duke
parents:
diff changeset
731 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
732 // Supplied function returns the sum of the inputs IN THE CURRENT RING. For
a61af66fc99e Initial load
duke
parents:
diff changeset
733 // the logical operations the ring's ADD is really a logical OR function.
a61af66fc99e Initial load
duke
parents:
diff changeset
734 // This also type-checks the inputs for sanity. Guaranteed never to
a61af66fc99e Initial load
duke
parents:
diff changeset
735 // be passed a TOP or BOTTOM type, these are filtered out by pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
736 const Type *OrINode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
737 const TypeInt *r0 = t0->is_int(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
738 const TypeInt *r1 = t1->is_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
739
a61af66fc99e Initial load
duke
parents:
diff changeset
740 // If both args are bool, can figure out better types
a61af66fc99e Initial load
duke
parents:
diff changeset
741 if ( r0 == TypeInt::BOOL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
742 if ( r1 == TypeInt::ONE) {
a61af66fc99e Initial load
duke
parents:
diff changeset
743 return TypeInt::ONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
744 } else if ( r1 == TypeInt::BOOL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
745 return TypeInt::BOOL;
a61af66fc99e Initial load
duke
parents:
diff changeset
746 }
a61af66fc99e Initial load
duke
parents:
diff changeset
747 } else if ( r0 == TypeInt::ONE ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
748 if ( r1 == TypeInt::BOOL ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
749 return TypeInt::ONE;
a61af66fc99e Initial load
duke
parents:
diff changeset
750 }
a61af66fc99e Initial load
duke
parents:
diff changeset
751 }
a61af66fc99e Initial load
duke
parents:
diff changeset
752
a61af66fc99e Initial load
duke
parents:
diff changeset
753 // If either input is not a constant, just return all integers.
a61af66fc99e Initial load
duke
parents:
diff changeset
754 if( !r0->is_con() || !r1->is_con() )
a61af66fc99e Initial load
duke
parents:
diff changeset
755 return TypeInt::INT; // Any integer, but still no symbols.
a61af66fc99e Initial load
duke
parents:
diff changeset
756
a61af66fc99e Initial load
duke
parents:
diff changeset
757 // Otherwise just OR them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
758 return TypeInt::make( r0->get_con() | r1->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
759 }
a61af66fc99e Initial load
duke
parents:
diff changeset
760
a61af66fc99e Initial load
duke
parents:
diff changeset
761 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
762 //------------------------------Identity---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
763 Node *OrLNode::Identity( PhaseTransform *phase ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
764 // x | x => x
a61af66fc99e Initial load
duke
parents:
diff changeset
765 if (phase->eqv(in(1), in(2))) {
a61af66fc99e Initial load
duke
parents:
diff changeset
766 return in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
767 }
a61af66fc99e Initial load
duke
parents:
diff changeset
768
a61af66fc99e Initial load
duke
parents:
diff changeset
769 return AddNode::Identity(phase);
a61af66fc99e Initial load
duke
parents:
diff changeset
770 }
a61af66fc99e Initial load
duke
parents:
diff changeset
771
a61af66fc99e Initial load
duke
parents:
diff changeset
772 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
773 const Type *OrLNode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
774 const TypeLong *r0 = t0->is_long(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
775 const TypeLong *r1 = t1->is_long();
a61af66fc99e Initial load
duke
parents:
diff changeset
776
a61af66fc99e Initial load
duke
parents:
diff changeset
777 // If either input is not a constant, just return all integers.
a61af66fc99e Initial load
duke
parents:
diff changeset
778 if( !r0->is_con() || !r1->is_con() )
a61af66fc99e Initial load
duke
parents:
diff changeset
779 return TypeLong::LONG; // Any integer, but still no symbols.
a61af66fc99e Initial load
duke
parents:
diff changeset
780
a61af66fc99e Initial load
duke
parents:
diff changeset
781 // Otherwise just OR them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
782 return TypeLong::make( r0->get_con() | r1->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
783 }
a61af66fc99e Initial load
duke
parents:
diff changeset
784
a61af66fc99e Initial load
duke
parents:
diff changeset
785 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
786 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
787 // Supplied function returns the sum of the inputs IN THE CURRENT RING. For
a61af66fc99e Initial load
duke
parents:
diff changeset
788 // the logical operations the ring's ADD is really a logical OR function.
a61af66fc99e Initial load
duke
parents:
diff changeset
789 // This also type-checks the inputs for sanity. Guaranteed never to
a61af66fc99e Initial load
duke
parents:
diff changeset
790 // be passed a TOP or BOTTOM type, these are filtered out by pre-check.
a61af66fc99e Initial load
duke
parents:
diff changeset
791 const Type *XorINode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
792 const TypeInt *r0 = t0->is_int(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
793 const TypeInt *r1 = t1->is_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
794
a61af66fc99e Initial load
duke
parents:
diff changeset
795 // Complementing a boolean?
a61af66fc99e Initial load
duke
parents:
diff changeset
796 if( r0 == TypeInt::BOOL && ( r1 == TypeInt::ONE
a61af66fc99e Initial load
duke
parents:
diff changeset
797 || r1 == TypeInt::BOOL))
a61af66fc99e Initial load
duke
parents:
diff changeset
798 return TypeInt::BOOL;
a61af66fc99e Initial load
duke
parents:
diff changeset
799
a61af66fc99e Initial load
duke
parents:
diff changeset
800 if( !r0->is_con() || !r1->is_con() ) // Not constants
a61af66fc99e Initial load
duke
parents:
diff changeset
801 return TypeInt::INT; // Any integer, but still no symbols.
a61af66fc99e Initial load
duke
parents:
diff changeset
802
a61af66fc99e Initial load
duke
parents:
diff changeset
803 // Otherwise just XOR them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
804 return TypeInt::make( r0->get_con() ^ r1->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
805 }
a61af66fc99e Initial load
duke
parents:
diff changeset
806
a61af66fc99e Initial load
duke
parents:
diff changeset
807 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
808 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
809 const Type *XorLNode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
810 const TypeLong *r0 = t0->is_long(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
811 const TypeLong *r1 = t1->is_long();
a61af66fc99e Initial load
duke
parents:
diff changeset
812
a61af66fc99e Initial load
duke
parents:
diff changeset
813 // If either input is not a constant, just return all integers.
a61af66fc99e Initial load
duke
parents:
diff changeset
814 if( !r0->is_con() || !r1->is_con() )
a61af66fc99e Initial load
duke
parents:
diff changeset
815 return TypeLong::LONG; // Any integer, but still no symbols.
a61af66fc99e Initial load
duke
parents:
diff changeset
816
a61af66fc99e Initial load
duke
parents:
diff changeset
817 // Otherwise just OR them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
818 return TypeLong::make( r0->get_con() ^ r1->get_con() );
a61af66fc99e Initial load
duke
parents:
diff changeset
819 }
a61af66fc99e Initial load
duke
parents:
diff changeset
820
a61af66fc99e Initial load
duke
parents:
diff changeset
821 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
822 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
823 // Supplied function returns the sum of the inputs.
a61af66fc99e Initial load
duke
parents:
diff changeset
824 const Type *MaxINode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
825 const TypeInt *r0 = t0->is_int(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
826 const TypeInt *r1 = t1->is_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
827
a61af66fc99e Initial load
duke
parents:
diff changeset
828 // Otherwise just MAX them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
829 return TypeInt::make( MAX2(r0->_lo,r1->_lo), MAX2(r0->_hi,r1->_hi), MAX2(r0->_widen,r1->_widen) );
a61af66fc99e Initial load
duke
parents:
diff changeset
830 }
a61af66fc99e Initial load
duke
parents:
diff changeset
831
a61af66fc99e Initial load
duke
parents:
diff changeset
832 //=============================================================================
a61af66fc99e Initial load
duke
parents:
diff changeset
833 //------------------------------Idealize---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
834 // MINs show up in range-check loop limit calculations. Look for
a61af66fc99e Initial load
duke
parents:
diff changeset
835 // "MIN2(x+c0,MIN2(y,x+c1))". Pick the smaller constant: "MIN2(x+c0,y)"
a61af66fc99e Initial load
duke
parents:
diff changeset
836 Node *MinINode::Ideal(PhaseGVN *phase, bool can_reshape) {
a61af66fc99e Initial load
duke
parents:
diff changeset
837 Node *progress = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
838 // Force a right-spline graph
a61af66fc99e Initial load
duke
parents:
diff changeset
839 Node *l = in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
840 Node *r = in(2);
a61af66fc99e Initial load
duke
parents:
diff changeset
841 // Transform MinI1( MinI2(a,b), c) into MinI1( a, MinI2(b,c) )
a61af66fc99e Initial load
duke
parents:
diff changeset
842 // to force a right-spline graph for the rest of MinINode::Ideal().
a61af66fc99e Initial load
duke
parents:
diff changeset
843 if( l->Opcode() == Op_MinI ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
844 assert( l != l->in(1), "dead loop in MinINode::Ideal" );
a61af66fc99e Initial load
duke
parents:
diff changeset
845 r = phase->transform(new (phase->C, 3) MinINode(l->in(2),r));
a61af66fc99e Initial load
duke
parents:
diff changeset
846 l = l->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
847 set_req(1, l);
a61af66fc99e Initial load
duke
parents:
diff changeset
848 set_req(2, r);
a61af66fc99e Initial load
duke
parents:
diff changeset
849 return this;
a61af66fc99e Initial load
duke
parents:
diff changeset
850 }
a61af66fc99e Initial load
duke
parents:
diff changeset
851
a61af66fc99e Initial load
duke
parents:
diff changeset
852 // Get left input & constant
a61af66fc99e Initial load
duke
parents:
diff changeset
853 Node *x = l;
a61af66fc99e Initial load
duke
parents:
diff changeset
854 int x_off = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
855 if( x->Opcode() == Op_AddI && // Check for "x+c0" and collect constant
a61af66fc99e Initial load
duke
parents:
diff changeset
856 x->in(2)->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
857 const Type *t = x->in(2)->bottom_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
858 if( t == Type::TOP ) return NULL; // No progress
a61af66fc99e Initial load
duke
parents:
diff changeset
859 x_off = t->is_int()->get_con();
a61af66fc99e Initial load
duke
parents:
diff changeset
860 x = x->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
861 }
a61af66fc99e Initial load
duke
parents:
diff changeset
862
a61af66fc99e Initial load
duke
parents:
diff changeset
863 // Scan a right-spline-tree for MINs
a61af66fc99e Initial load
duke
parents:
diff changeset
864 Node *y = r;
a61af66fc99e Initial load
duke
parents:
diff changeset
865 int y_off = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
866 // Check final part of MIN tree
a61af66fc99e Initial load
duke
parents:
diff changeset
867 if( y->Opcode() == Op_AddI && // Check for "y+c1" and collect constant
a61af66fc99e Initial load
duke
parents:
diff changeset
868 y->in(2)->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
869 const Type *t = y->in(2)->bottom_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
870 if( t == Type::TOP ) return NULL; // No progress
a61af66fc99e Initial load
duke
parents:
diff changeset
871 y_off = t->is_int()->get_con();
a61af66fc99e Initial load
duke
parents:
diff changeset
872 y = y->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
873 }
a61af66fc99e Initial load
duke
parents:
diff changeset
874 if( x->_idx > y->_idx && r->Opcode() != Op_MinI ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
875 swap_edges(1, 2);
a61af66fc99e Initial load
duke
parents:
diff changeset
876 return this;
a61af66fc99e Initial load
duke
parents:
diff changeset
877 }
a61af66fc99e Initial load
duke
parents:
diff changeset
878
a61af66fc99e Initial load
duke
parents:
diff changeset
879
a61af66fc99e Initial load
duke
parents:
diff changeset
880 if( r->Opcode() == Op_MinI ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
881 assert( r != r->in(2), "dead loop in MinINode::Ideal" );
a61af66fc99e Initial load
duke
parents:
diff changeset
882 y = r->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
883 // Check final part of MIN tree
a61af66fc99e Initial load
duke
parents:
diff changeset
884 if( y->Opcode() == Op_AddI &&// Check for "y+c1" and collect constant
a61af66fc99e Initial load
duke
parents:
diff changeset
885 y->in(2)->is_Con() ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
886 const Type *t = y->in(2)->bottom_type();
a61af66fc99e Initial load
duke
parents:
diff changeset
887 if( t == Type::TOP ) return NULL; // No progress
a61af66fc99e Initial load
duke
parents:
diff changeset
888 y_off = t->is_int()->get_con();
a61af66fc99e Initial load
duke
parents:
diff changeset
889 y = y->in(1);
a61af66fc99e Initial load
duke
parents:
diff changeset
890 }
a61af66fc99e Initial load
duke
parents:
diff changeset
891
a61af66fc99e Initial load
duke
parents:
diff changeset
892 if( x->_idx > y->_idx )
a61af66fc99e Initial load
duke
parents:
diff changeset
893 return new (phase->C, 3) MinINode(r->in(1),phase->transform(new (phase->C, 3) MinINode(l,r->in(2))));
a61af66fc99e Initial load
duke
parents:
diff changeset
894
a61af66fc99e Initial load
duke
parents:
diff changeset
895 // See if covers: MIN2(x+c0,MIN2(y+c1,z))
a61af66fc99e Initial load
duke
parents:
diff changeset
896 if( !phase->eqv(x,y) ) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
897 // If (y == x) transform MIN2(x+c0, MIN2(x+c1,z)) into
a61af66fc99e Initial load
duke
parents:
diff changeset
898 // MIN2(x+c0 or x+c1 which less, z).
a61af66fc99e Initial load
duke
parents:
diff changeset
899 return new (phase->C, 3) MinINode(phase->transform(new (phase->C, 3) AddINode(x,phase->intcon(MIN2(x_off,y_off)))),r->in(2));
a61af66fc99e Initial load
duke
parents:
diff changeset
900 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
901 // See if covers: MIN2(x+c0,y+c1)
a61af66fc99e Initial load
duke
parents:
diff changeset
902 if( !phase->eqv(x,y) ) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
903 // If (y == x) transform MIN2(x+c0,x+c1) into x+c0 or x+c1 which less.
a61af66fc99e Initial load
duke
parents:
diff changeset
904 return new (phase->C, 3) AddINode(x,phase->intcon(MIN2(x_off,y_off)));
a61af66fc99e Initial load
duke
parents:
diff changeset
905 }
a61af66fc99e Initial load
duke
parents:
diff changeset
906
a61af66fc99e Initial load
duke
parents:
diff changeset
907 }
a61af66fc99e Initial load
duke
parents:
diff changeset
908
a61af66fc99e Initial load
duke
parents:
diff changeset
909 //------------------------------add_ring---------------------------------------
a61af66fc99e Initial load
duke
parents:
diff changeset
910 // Supplied function returns the sum of the inputs.
a61af66fc99e Initial load
duke
parents:
diff changeset
911 const Type *MinINode::add_ring( const Type *t0, const Type *t1 ) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
912 const TypeInt *r0 = t0->is_int(); // Handy access
a61af66fc99e Initial load
duke
parents:
diff changeset
913 const TypeInt *r1 = t1->is_int();
a61af66fc99e Initial load
duke
parents:
diff changeset
914
a61af66fc99e Initial load
duke
parents:
diff changeset
915 // Otherwise just MIN them bits.
a61af66fc99e Initial load
duke
parents:
diff changeset
916 return TypeInt::make( MIN2(r0->_lo,r1->_lo), MIN2(r0->_hi,r1->_hi), MAX2(r0->_widen,r1->_widen) );
a61af66fc99e Initial load
duke
parents:
diff changeset
917 }