Skip to content

Commit 03eb433

Browse files
committed
Update accessStaticItem to use loadAddressConstant on POWER10
The MemoryReference::accessStaticItem method is in charge of handling any loads/stores to static addresses. Previously, on 64-bit systems, the pTOC would be used to load the address of the static to access. However, under POWER10 the pTOC will be disabled in favour of using ConstantDataSnippet with PC-relative offsets for these purposes. In line with this, this method has been updated to use loadAddressConstant when possible when running on POWER10 or later. Signed-off-by: Ben Thomas <ben@benthomas.ca>
1 parent d9b194d commit 03eb433

File tree

1 file changed

+68
-26
lines changed

1 file changed

+68
-26
lines changed

compiler/p/codegen/OMRMemoryReference.cpp

+68-26
Original file line numberDiff line numberDiff line change
@@ -1617,37 +1617,79 @@ void OMR::Power::MemoryReference::accessStaticItem(TR::Node *node, TR::SymbolRef
16171617
symbol->getStaticSymbol()->setTOCIndex(tocIndex);
16181618
}
16191619

1620-
// We need a snippet for unresolved or AOT if:
1621-
// 1. the load hasn't been resolved yet
1622-
// (otherwise optimizer will remove the ResolveCHK and we can be sure pTOC will contain the resolved address),
1623-
// 2. we don't have a PTOC slot, we must always take the slow path
1624-
1625-
if ((ref->isUnresolved() || useUnresSnippetToAvoidRelo) &&
1626-
(topNode->getOpCodeValue() == TR::ResolveCHK || tocIndex == PTOC_FULL_INDEX))
1620+
if (cg->comp()->target().cpu.isAtLeast(OMR_PROCESSOR_PPC_P10))
16271621
{
1628-
snippet = new (cg->trHeapMemory()) TR::UnresolvedDataSnippet(cg, node, ref, isStore, false);
1629-
cg->addSnippet(snippet);
1630-
}
1622+
TR::Register *addrReg = cg->allocateRegister();
1623+
_baseRegister = addrReg;
1624+
self()->setBaseModifiable();
16311625

1632-
// TODO: Improve the code sequence for cases when we know pTOC is full.
1633-
TR::MemoryReference *tocRef = new (cg->trHeapMemory()) TR::MemoryReference(cg->getTOCBaseRegister(), 0, sizeof(uintptr_t), cg);
1634-
tocRef->setSymbol(symbol, cg);
1635-
tocRef->getSymbolReference()->copyFlags(ref);
1636-
tocRef->setUsingStaticTOC();
1637-
if (snippet != NULL)
1638-
{
1639-
tocRef->setUnresolvedSnippet(snippet);
1640-
tocRef->adjustForResolution(cg);
1626+
// For now, P10 PC-relative loads and stores are not supported for UnresolvedDataSnippet
1627+
// and for anything requiring a TR_ClassAddress relocation. To make things work, we must
1628+
// emit the 5 instruction load address by faking that the pTOC was full.
1629+
if (ref->isUnresolved() || useUnresSnippetToAvoidRelo || (cg->comp()->compileRelocatableCode() && symbol->isStatic() && symbol->isClassObject()))
1630+
{
1631+
if (ref->isUnresolved() || useUnresSnippetToAvoidRelo)
1632+
{
1633+
snippet = new (cg->trHeapMemory()) TR::UnresolvedDataSnippet(cg, node, ref, isStore, false);
1634+
cg->addSnippet(snippet);
1635+
}
1636+
1637+
TR::MemoryReference *fakeTocRef = new (cg->trHeapMemory()) TR::MemoryReference(cg->getTOCBaseRegister(), 0, sizeof(uintptr_t), cg);
1638+
fakeTocRef->setSymbol(symbol, cg);
1639+
fakeTocRef->getSymbolReference()->copyFlags(ref);
1640+
fakeTocRef->setUsingStaticTOC();
1641+
1642+
if (snippet != NULL)
1643+
{
1644+
fakeTocRef->setUnresolvedSnippet(snippet);
1645+
fakeTocRef->adjustForResolution(cg);
1646+
}
1647+
1648+
symbol->getStaticSymbol()->setTOCIndex(PTOC_FULL_INDEX);
1649+
1650+
generateTrg1MemInstruction(cg, TR::InstOpCode::ld, node==NULL?topNode:node, addrReg, fakeTocRef);
1651+
if (snippet != NULL)
1652+
cg->stopUsingRegister(fakeTocRef->getModBase());
1653+
}
1654+
else
1655+
{
1656+
loadAddressConstant(cg, false, nodeForSymbol, reinterpret_cast<intptr_t>(symbol->getStaticSymbol()->getStaticAddress()), addrReg);
1657+
}
16411658
}
1659+
else
1660+
{
1661+
// We need a snippet for unresolved or AOT if:
1662+
// 1. the load hasn't been resolved yet
1663+
// (otherwise optimizer will remove the ResolveCHK and we can be sure pTOC will contain the resolved address),
1664+
// 2. we don't have a PTOC slot, we must always take the slow path
1665+
1666+
if ((ref->isUnresolved() || useUnresSnippetToAvoidRelo) &&
1667+
(topNode->getOpCodeValue() == TR::ResolveCHK || tocIndex == PTOC_FULL_INDEX))
1668+
{
1669+
snippet = new (cg->trHeapMemory()) TR::UnresolvedDataSnippet(cg, node, ref, isStore, false);
1670+
cg->addSnippet(snippet);
1671+
}
16421672

1643-
TR::Register *addrReg = cg->allocateRegister();
1644-
TR::InstOpCode::Mnemonic loadOp = TR::InstOpCode::ld;
1673+
// TODO: Improve the code sequence for cases when we know pTOC is full.
1674+
TR::MemoryReference *tocRef = new (cg->trHeapMemory()) TR::MemoryReference(cg->getTOCBaseRegister(), 0, sizeof(uintptr_t), cg);
1675+
tocRef->setSymbol(symbol, cg);
1676+
tocRef->getSymbolReference()->copyFlags(ref);
1677+
tocRef->setUsingStaticTOC();
1678+
if (snippet != NULL)
1679+
{
1680+
tocRef->setUnresolvedSnippet(snippet);
1681+
tocRef->adjustForResolution(cg);
1682+
}
16451683

1646-
generateTrg1MemInstruction(cg, loadOp, node==NULL?topNode:node, addrReg, tocRef);
1647-
if (snippet != NULL)
1648-
cg->stopUsingRegister(tocRef->getModBase());
1649-
_baseRegister = addrReg;
1650-
self()->setBaseModifiable();
1684+
TR::Register *addrReg = cg->allocateRegister();
1685+
TR::InstOpCode::Mnemonic loadOp = TR::InstOpCode::ld;
1686+
1687+
generateTrg1MemInstruction(cg, loadOp, node==NULL?topNode:node, addrReg, tocRef);
1688+
if (snippet != NULL)
1689+
cg->stopUsingRegister(tocRef->getModBase());
1690+
_baseRegister = addrReg;
1691+
self()->setBaseModifiable();
1692+
}
16511693
}
16521694
else
16531695
{

0 commit comments

Comments
 (0)