diff --git a/compiler-rt/cmake/Modules/AllSupportedArchDefs.cmake b/compiler-rt/cmake/Modules/AllSupportedArchDefs.cmake --- a/compiler-rt/cmake/Modules/AllSupportedArchDefs.cmake +++ b/compiler-rt/cmake/Modules/AllSupportedArchDefs.cmake @@ -85,7 +85,7 @@ set(ALL_SHADOWCALLSTACK_SUPPORTED_ARCH ${ARM64}) if (UNIX) -set(ALL_ORC_SUPPORTED_ARCH ${X86_64} ${ARM64} ${ARM32}) +set(ALL_ORC_SUPPORTED_ARCH ${X86_64} ${ARM64} ${ARM32} ${PPC64}) endif() if (WIN32) diff --git a/compiler-rt/lib/orc/CMakeLists.txt b/compiler-rt/lib/orc/CMakeLists.txt --- a/compiler-rt/lib/orc/CMakeLists.txt +++ b/compiler-rt/lib/orc/CMakeLists.txt @@ -24,6 +24,7 @@ macho_tlv.arm64.S elfnix_tls.x86-64.S elfnix_tls.aarch64.S + elfnix_tls.ppc64.S ) # Common implementation headers will go here. @@ -149,6 +150,7 @@ set(ORC_ASM_SOURCES elfnix_tls.x86-64.S elfnix_tls.aarch64.S + elfnix_tls.ppc64.S ) endif() diff --git a/compiler-rt/lib/orc/elfnix_tls.ppc64.S b/compiler-rt/lib/orc/elfnix_tls.ppc64.S new file mode 100644 --- /dev/null +++ b/compiler-rt/lib/orc/elfnix_tls.ppc64.S @@ -0,0 +1,35 @@ +//===-- orc_rt_elfnix_tls.ppc64.s -------------------------------*- ASM -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// This file is a part of the ORC runtime support library. +// +//===----------------------------------------------------------------------===// + +// The content of this file is PowerPC64 only. +#if defined(__powerpc64__) + +#define REGISTER_SAVE_SPACE_SIZE 32 + + .text + // Return address of TLV in r3, all other registers are preserved. + // TODO: Add fast-path for repeating access. + .global ___orc_rt_elfnix_tls_get_addr +___orc_rt_elfnix_tls_get_addr: + addis 2, 12, .TOC.-___orc_rt_elfnix_tls_get_addr@ha + addi 2, 2, .TOC.-___orc_rt_elfnix_tls_get_addr@l + mflr 0 + std 0, 16(1) + stdu 1, -32(1) + bl __orc_rt_elfnix_tls_get_addr_impl + nop + addi 1, 1, 32 + ld 0, 16(1) + mtlr 0 + blr + +#endif diff --git a/llvm/include/llvm/ExecutionEngine/JITLink/ppc64.h b/llvm/include/llvm/ExecutionEngine/JITLink/ppc64.h --- a/llvm/include/llvm/ExecutionEngine/JITLink/ppc64.h +++ b/llvm/include/llvm/ExecutionEngine/JITLink/ppc64.h @@ -40,6 +40,9 @@ RequestCall, // Request calling function without TOC. RequestCallNoTOC, + RequestTLSDescInGOTAndTransformToTOCDelta16HA, + RequestTLSDescInGOTAndTransformToTOCDelta16LO, + }; enum PLTCallStubKind { diff --git a/llvm/lib/ExecutionEngine/JITLink/ELF_ppc64.cpp b/llvm/lib/ExecutionEngine/JITLink/ELF_ppc64.cpp --- a/llvm/lib/ExecutionEngine/JITLink/ELF_ppc64.cpp +++ b/llvm/lib/ExecutionEngine/JITLink/ELF_ppc64.cpp @@ -31,6 +31,70 @@ constexpr StringRef ELFTOCSymbolName = ".TOC."; constexpr StringRef TOCSymbolAliasIdent = "__TOC__"; constexpr uint64_t ELFTOCBaseOffset = 0x8000; +constexpr StringRef ELFTLSInfoSectionName = "$__TLSINFO"; + +template +class TLSInfoTableManager_ELF_ppc64 + : public TableManager> { +public: + static const uint8_t TLSInfoEntryContent[16]; + + static StringRef getSectionName() { return ELFTLSInfoSectionName; } + + bool visitEdge(LinkGraph &G, Block *B, Edge &E) { + Edge::Kind K = E.getKind(); + if (K == ppc64::RequestTLSDescInGOTAndTransformToTOCDelta16HA) { + E.setKind(ppc64::TOCDelta16HA); + E.setTarget(this->getEntryForTarget(G, E.getTarget())); + return true; + } + if (K == ppc64::RequestTLSDescInGOTAndTransformToTOCDelta16LO) { + E.setKind(ppc64::TOCDelta16LO); + E.setTarget(this->getEntryForTarget(G, E.getTarget())); + return true; + } + return false; + } + + Symbol &createEntry(LinkGraph &G, Symbol &Target) { + // The TLS Info entry's key value will be written by + // `fixTLVSectionsAndEdges`, so create mutable content. + auto &TLSInfoEntry = G.createMutableContentBlock( + getTLSInfoSection(G), G.allocateContent(getTLSInfoEntryContent()), + orc::ExecutorAddr(), 8, 0); + TLSInfoEntry.addEdge(ppc64::Pointer64, 8, Target, 0); + return G.addAnonymousSymbol(TLSInfoEntry, 0, 16, false, false); + } + +private: + Section &getTLSInfoSection(LinkGraph &G) { + if (!TLSInfoTable) + TLSInfoTable = + &G.createSection(ELFTLSInfoSectionName, orc::MemProt::Read); + return *TLSInfoTable; + } + + ArrayRef getTLSInfoEntryContent() const { + return {reinterpret_cast(TLSInfoEntryContent), + sizeof(TLSInfoEntryContent)}; + } + + Section *TLSInfoTable = nullptr; +}; + +template <> +const uint8_t TLSInfoTableManager_ELF_ppc64< + support::endianness::little>::TLSInfoEntryContent[16] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /*pthread key */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 /*data address*/ +}; + +template <> +const uint8_t TLSInfoTableManager_ELF_ppc64< + support::endianness::big>::TLSInfoEntryContent[16] = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, /*pthread key */ + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 /*data address*/ +}; template Symbol &createELFGOTHeader(LinkGraph &G, @@ -92,8 +156,8 @@ registerExistingGOTEntries(G, TOC); ppc64::PLTTableManager PLT(TOC); - visitExistingEdges(G, TOC, PLT); - // TODO: Add TLS support. + TLSInfoTableManager_ELF_ppc64 TLSInfo; + visitExistingEdges(G, TOC, PLT, TLSInfo); // After visiting edges in LinkGraph, we have GOT entries built in the // synthesized section. @@ -165,6 +229,10 @@ if (LLVM_UNLIKELY(ELFReloc == ELF::R_PPC64_NONE)) return Error::success(); + // Ignore TLS model marker. We currently only support global-dynamic model. + if (ELFReloc == ELF::R_PPC64_TLSGD) + return Error::success(); + auto ObjSymbol = Base::Obj.getRelocationSymbol(Rel, Base::SymTabSec); if (!ObjSymbol) return ObjSymbol.takeError(); @@ -193,6 +261,9 @@ case ELF::R_PPC64_ADDR64: Kind = ppc64::Pointer64; break; + case ELF::R_PPC64_ADDR32: + Kind = ppc64::Pointer32; + break; case ELF::R_PPC64_TOC16_HA: Kind = ppc64::TOCDelta16HA; break; @@ -227,6 +298,12 @@ case ELF::R_PPC64_REL64: Kind = ppc64::Delta64; break; + case ELF::R_PPC64_GOT_TLSGD16_HA: + Kind = ppc64::RequestTLSDescInGOTAndTransformToTOCDelta16HA; + break; + case ELF::R_PPC64_GOT_TLSGD16_LO: + Kind = ppc64::RequestTLSDescInGOTAndTransformToTOCDelta16LO; + break; } Edge GE(Kind, Offset, *GraphSymbol, Addend); diff --git a/llvm/lib/ExecutionEngine/JITLink/ppc64.cpp b/llvm/lib/ExecutionEngine/JITLink/ppc64.cpp --- a/llvm/lib/ExecutionEngine/JITLink/ppc64.cpp +++ b/llvm/lib/ExecutionEngine/JITLink/ppc64.cpp @@ -92,6 +92,10 @@ return "RequestCall"; case RequestCallNoTOC: return "RequestCallNoTOC"; + case RequestTLSDescInGOTAndTransformToTOCDelta16HA: + return "RequestTLSDescInGOTAndTransformToTOCDelta16HA"; + case RequestTLSDescInGOTAndTransformToTOCDelta16LO: + return "RequestTLSDescInGOTAndTransformToTOCDelta16LO"; default: return getGenericEdgeKindName(static_cast(K)); } diff --git a/llvm/lib/ExecutionEngine/Orc/ELFNixPlatform.cpp b/llvm/lib/ExecutionEngine/Orc/ELFNixPlatform.cpp --- a/llvm/lib/ExecutionEngine/Orc/ELFNixPlatform.cpp +++ b/llvm/lib/ExecutionEngine/Orc/ELFNixPlatform.cpp @@ -11,6 +11,7 @@ #include "llvm/BinaryFormat/ELF.h" #include "llvm/ExecutionEngine/JITLink/ELF_x86_64.h" #include "llvm/ExecutionEngine/JITLink/aarch64.h" +#include "llvm/ExecutionEngine/JITLink/ppc64.h" #include "llvm/ExecutionEngine/JITLink/x86_64.h" #include "llvm/ExecutionEngine/Orc/DebugUtils.h" #include "llvm/ExecutionEngine/Orc/ExecutionUtils.h" @@ -54,6 +55,16 @@ Endianness = support::endianness::little; EdgeKind = jitlink::aarch64::Pointer64; break; + case Triple::ppc64: + PointerSize = 8; + Endianness = support::endianness::big; + EdgeKind = jitlink::ppc64::Pointer64; + break; + case Triple::ppc64le: + PointerSize = 8; + Endianness = support::endianness::little; + EdgeKind = jitlink::ppc64::Pointer64; + break; default: llvm_unreachable("Unrecognized architecture"); } @@ -238,6 +249,8 @@ switch (TT.getArch()) { case Triple::x86_64: case Triple::aarch64: + case Triple::ppc64: + case Triple::ppc64le: return true; default: return false;