Skip to content

Commit 51fd539

Browse files
Handle complex local addresses in block morphing
In block morphing, "addrSpill" is used when the destination or source represent indirections of "complex" addresses. Unfortunately, some trees in the form of "IND(ADDR(LCL))" fall into this category. If such an "ADDR(LCL)" is used as an "addrSpill", the underlying local *must* be marked as address-exposed. Block morphing was using a very simplistic test for when that needs to happen, essentially only recognizing "ADDR(LCL_VAR/FLD)". But it is possible to have a more complicated pattern as "PrepareDst/Src" uses "IsLocalAddrExpr" to recognize indirect stores to locals. Currently it appears impossible to get a mismatch here as morph transforms "IND(ADD(ADDR(LCL_VAR), OFFSET))" into "LCL_FLD" (including for TYP_STRUCT indirections), but this is a very fragile invariant. Transforming TYP_STRUCT GT_FIELDs into GT_OBJs instead of GT_INDs breaks it, for example. Fix this by address-exposing the local obtained via "IsLocalAddrExpr".
1 parent 64c05a3 commit 51fd539

File tree

1 file changed

+11
-31
lines changed

1 file changed

+11
-31
lines changed

src/coreclr/jit/morphblock.cpp

+11-31
Original file line numberDiff line numberDiff line change
@@ -1048,9 +1048,9 @@ GenTree* MorphCopyBlockHelper::CopyFieldByField()
10481048
{
10491049
GenTreeOp* asgFields = nullptr;
10501050

1051-
GenTree* addrSpill = nullptr;
1052-
unsigned addrSpillTemp = BAD_VAR_NUM;
1053-
bool addrSpillIsStackDest = false; // true if 'addrSpill' represents the address in our local stack frame
1051+
GenTree* addrSpill = nullptr;
1052+
unsigned addrSpillSrcLclNum = BAD_VAR_NUM;
1053+
unsigned addrSpillTemp = BAD_VAR_NUM;
10541054

10551055
GenTree* addrSpillAsg = nullptr;
10561056

@@ -1095,7 +1095,8 @@ GenTree* MorphCopyBlockHelper::CopyFieldByField()
10951095
// We will spill m_srcAddr (i.e. assign to a temp "BlockOp address local")
10961096
// no need to clone a new copy as it is only used once
10971097
//
1098-
addrSpill = m_srcAddr; // addrSpill represents the 'm_srcAddr'
1098+
addrSpill = m_srcAddr; // addrSpill represents the 'm_srcAddr'
1099+
addrSpillSrcLclNum = m_srcLclNum;
10991100
}
11001101
}
11011102
}
@@ -1144,28 +1145,13 @@ GenTree* MorphCopyBlockHelper::CopyFieldByField()
11441145
// We will spill m_dstAddr (i.e. assign to a temp "BlockOp address local")
11451146
// no need to clone a new copy as it is only used once
11461147
//
1147-
addrSpill = m_dstAddr; // addrSpill represents the 'm_dstAddr'
1148+
addrSpill = m_dstAddr; // addrSpill represents the 'm_dstAddr'
1149+
addrSpillSrcLclNum = m_dstLclNum;
11481150
}
11491151
}
11501152
}
11511153
}
11521154

1153-
// TODO-CQ: this should be based on a more general
1154-
// "BaseAddress" method, that handles fields of structs, before or after
1155-
// morphing.
1156-
if ((addrSpill != nullptr) && addrSpill->OperIs(GT_ADDR))
1157-
{
1158-
GenTree* addrSpillOp = addrSpill->AsOp()->gtGetOp1();
1159-
if (addrSpillOp->IsLocal())
1160-
{
1161-
// We will *not* consider this to define the local, but rather have each individual field assign
1162-
// be a definition.
1163-
addrSpillOp->gtFlags &= ~(GTF_LIVENESS_MASK);
1164-
addrSpillIsStackDest = true; // addrSpill represents the address of LclVar[varNum] in our
1165-
// local stack frame
1166-
}
1167-
}
1168-
11691155
if (addrSpill != nullptr)
11701156
{
11711157
// 'addrSpill' is already morphed
@@ -1178,8 +1164,9 @@ GenTree* MorphCopyBlockHelper::CopyFieldByField()
11781164

11791165
addrSpillDsc->lvType = TYP_BYREF;
11801166

1181-
if (addrSpillIsStackDest)
1167+
if (addrSpillSrcLclNum != BAD_VAR_NUM)
11821168
{
1169+
// addrSpill represents the address of LclVar[varNum] in our local stack frame.
11831170
addrSpillDsc->lvStackByref = true;
11841171
}
11851172

@@ -1193,23 +1180,16 @@ GenTree* MorphCopyBlockHelper::CopyFieldByField()
11931180
// that we don't delete the definition for this LclVar
11941181
// as a dead store later on.
11951182
//
1196-
if (addrSpill->OperGet() == GT_ADDR)
1183+
if (addrSpillSrcLclNum != BAD_VAR_NUM)
11971184
{
1198-
GenTree* addrOp = addrSpill->AsOp()->gtOp1;
1199-
if (addrOp->IsLocal())
1200-
{
1201-
unsigned lclVarNum = addrOp->AsLclVarCommon()->GetLclNum();
1202-
m_comp->lvaGetDesc(lclVarNum)->SetAddressExposed(true DEBUGARG(AddressExposedReason::COPY_FLD_BY_FLD));
1203-
m_comp->lvaSetVarDoNotEnregister(lclVarNum DEBUGARG(DoNotEnregisterReason::AddrExposed));
1204-
}
1185+
m_comp->lvaSetVarAddrExposed(addrSpillSrcLclNum DEBUGARG(AddressExposedReason::COPY_FLD_BY_FLD));
12051186
}
12061187
}
12071188

12081189
// We may have allocated a temp above, and that may have caused the lvaTable to be expanded.
12091190
// So, beyond this point we cannot rely on the old values of 'm_srcVarDsc' and 'm_dstVarDsc'.
12101191
for (unsigned i = 0; i < fieldCnt; ++i)
12111192
{
1212-
12131193
GenTree* dstFld;
12141194
if (m_dstDoFldAsg)
12151195
{

0 commit comments

Comments
 (0)