@@ -1098,6 +1098,17 @@ Node* LoadNode::can_see_arraycopy_value(Node* st, PhaseGVN* phase) const {
1098
1098
return nullptr ;
1099
1099
}
1100
1100
1101
+ static Node* see_through_inline_type (PhaseValues* phase, const MemNode* load, Node* base, int offset) {
1102
+ if (!load->is_mismatched_access () && base != nullptr && base->is_InlineType () && offset > oopDesc::klass_offset_in_bytes ()) {
1103
+ InlineTypeNode* vt = base->as_InlineType ();
1104
+ assert (!vt->is_larval (), " must not load from a larval object" );
1105
+ Node* value = vt->field_value_by_offset (offset, true );
1106
+ assert (value != nullptr , " must see some value" );
1107
+ return value;
1108
+ }
1109
+
1110
+ return nullptr ;
1111
+ }
1101
1112
1102
1113
// ---------------------------can_see_stored_value------------------------------
1103
1114
// This routine exists to make sure this set of tests is done the same
@@ -1110,6 +1121,16 @@ Node* MemNode::can_see_stored_value(Node* st, PhaseValues* phase) const {
1110
1121
Node* ld_adr = in (MemNode::Address);
1111
1122
intptr_t ld_off = 0 ;
1112
1123
Node* ld_base = AddPNode::Ideal_base_and_offset (ld_adr, phase, ld_off);
1124
+
1125
+ // Try to see through an InlineTypeNode
1126
+ // LoadN is special because the input is not compressed
1127
+ if (Opcode () != Op_LoadN) {
1128
+ Node* value = see_through_inline_type (phase, this , ld_base, ld_off);
1129
+ if (value != nullptr ) {
1130
+ return value;
1131
+ }
1132
+ }
1133
+
1113
1134
Node* ld_alloc = AllocateNode::Ideal_allocation (ld_base);
1114
1135
const TypeInstPtr* tp = phase->type (ld_adr)->isa_instptr ();
1115
1136
Compile::AliasType* atp = (tp != nullptr ) ? phase->C ->alias_type (tp) : nullptr ;
@@ -1283,23 +1304,6 @@ bool LoadNode::is_instance_field_load_with_local_phi(Node* ctrl) {
1283
1304
// ------------------------------Identity---------------------------------------
1284
1305
// Loads are identity if previous store is to same address
1285
1306
Node* LoadNode::Identity (PhaseGVN* phase) {
1286
- // Loading from an InlineType? The InlineType has the values of
1287
- // all fields as input. Look for the field with matching offset.
1288
- Node* addr = in (Address);
1289
- intptr_t offset;
1290
- Node* base = AddPNode::Ideal_base_and_offset (addr, phase, offset);
1291
- if (!is_mismatched_access () && base != nullptr && base->is_InlineType () && offset > oopDesc::klass_offset_in_bytes ()) {
1292
- Node* value = base->as_InlineType ()->field_value_by_offset ((int )offset, true );
1293
- if (value != nullptr ) {
1294
- if (Opcode () == Op_LoadN) {
1295
- // Encode oop value if we are loading a narrow oop
1296
- assert (!phase->type (value)->isa_narrowoop (), " should already be decoded" );
1297
- value = phase->transform (new EncodePNode (value, bottom_type ()));
1298
- }
1299
- return value;
1300
- }
1301
- }
1302
-
1303
1307
// If the previous store-maker is the right kind of Store, and the store is
1304
1308
// to the same address, then we are equal to the value stored.
1305
1309
Node* mem = in (Memory);
@@ -2430,6 +2434,19 @@ const Type* LoadSNode::Value(PhaseGVN* phase) const {
2430
2434
return LoadNode::Value (phase);
2431
2435
}
2432
2436
2437
+ Node* LoadNNode::Ideal (PhaseGVN* phase, bool can_reshape) {
2438
+ // Loading from an InlineType, find the input and make an EncodeP
2439
+ Node* addr = in (Address);
2440
+ intptr_t offset;
2441
+ Node* base = AddPNode::Ideal_base_and_offset (addr, phase, offset);
2442
+ Node* value = see_through_inline_type (phase, this , base, offset);
2443
+ if (value != nullptr ) {
2444
+ return new EncodePNode (value, type ());
2445
+ }
2446
+
2447
+ return LoadNode::Ideal (phase, can_reshape);
2448
+ }
2449
+
2433
2450
// =============================================================================
2434
2451
// ----------------------------LoadKlassNode::make------------------------------
2435
2452
// Polymorphic factory method:
0 commit comments