annotate src/share/vm/opto/addnode.cpp @ 9126:bc26f978b0ce

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