xref: /freebsd/contrib/llvm-project/llvm/lib/Target/ARM/ARMTargetObjectFile.cpp (revision 0b57cec536236d46e3dba9bd041533462f33dbb7)
1*0b57cec5SDimitry Andric //===-- llvm/Target/ARMTargetObjectFile.cpp - ARM Object Info Impl --------===//
2*0b57cec5SDimitry Andric //
3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*0b57cec5SDimitry Andric //
7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
8*0b57cec5SDimitry Andric 
9*0b57cec5SDimitry Andric #include "ARMTargetObjectFile.h"
10*0b57cec5SDimitry Andric #include "ARMSubtarget.h"
11*0b57cec5SDimitry Andric #include "ARMTargetMachine.h"
12*0b57cec5SDimitry Andric #include "llvm/BinaryFormat/Dwarf.h"
13*0b57cec5SDimitry Andric #include "llvm/BinaryFormat/ELF.h"
14*0b57cec5SDimitry Andric #include "llvm/MC/MCAsmInfo.h"
15*0b57cec5SDimitry Andric #include "llvm/MC/MCContext.h"
16*0b57cec5SDimitry Andric #include "llvm/MC/MCExpr.h"
17*0b57cec5SDimitry Andric #include "llvm/MC/MCSectionELF.h"
18*0b57cec5SDimitry Andric #include "llvm/MC/MCTargetOptions.h"
19*0b57cec5SDimitry Andric #include "llvm/MC/SectionKind.h"
20*0b57cec5SDimitry Andric #include "llvm/Target/TargetMachine.h"
21*0b57cec5SDimitry Andric #include <cassert>
22*0b57cec5SDimitry Andric 
23*0b57cec5SDimitry Andric using namespace llvm;
24*0b57cec5SDimitry Andric using namespace dwarf;
25*0b57cec5SDimitry Andric 
26*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
27*0b57cec5SDimitry Andric //                               ELF Target
28*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
29*0b57cec5SDimitry Andric 
30*0b57cec5SDimitry Andric void ARMElfTargetObjectFile::Initialize(MCContext &Ctx,
31*0b57cec5SDimitry Andric                                         const TargetMachine &TM) {
32*0b57cec5SDimitry Andric   const ARMBaseTargetMachine &ARM_TM = static_cast<const ARMBaseTargetMachine &>(TM);
33*0b57cec5SDimitry Andric   bool isAAPCS_ABI = ARM_TM.TargetABI == ARMBaseTargetMachine::ARMABI::ARM_ABI_AAPCS;
34*0b57cec5SDimitry Andric   bool genExecuteOnly =
35*0b57cec5SDimitry Andric       ARM_TM.getMCSubtargetInfo()->hasFeature(ARM::FeatureExecuteOnly);
36*0b57cec5SDimitry Andric 
37*0b57cec5SDimitry Andric   TargetLoweringObjectFileELF::Initialize(Ctx, TM);
38*0b57cec5SDimitry Andric   InitializeELF(isAAPCS_ABI);
39*0b57cec5SDimitry Andric 
40*0b57cec5SDimitry Andric   if (isAAPCS_ABI) {
41*0b57cec5SDimitry Andric     LSDASection = nullptr;
42*0b57cec5SDimitry Andric   }
43*0b57cec5SDimitry Andric 
44*0b57cec5SDimitry Andric   // Make code section unreadable when in execute-only mode
45*0b57cec5SDimitry Andric   if (genExecuteOnly) {
46*0b57cec5SDimitry Andric     unsigned Type = ELF::SHT_PROGBITS;
47*0b57cec5SDimitry Andric     unsigned Flags =
48*0b57cec5SDimitry Andric         ELF::SHF_EXECINSTR | ELF::SHF_ALLOC | ELF::SHF_ARM_PURECODE;
49*0b57cec5SDimitry Andric     // Since we cannot modify flags for an existing section, we create a new
50*0b57cec5SDimitry Andric     // section with the right flags, and use 0 as the unique ID for
51*0b57cec5SDimitry Andric     // execute-only text
52*0b57cec5SDimitry Andric     TextSection = Ctx.getELFSection(".text", Type, Flags, 0, "", 0U);
53*0b57cec5SDimitry Andric   }
54*0b57cec5SDimitry Andric }
55*0b57cec5SDimitry Andric 
56*0b57cec5SDimitry Andric const MCExpr *ARMElfTargetObjectFile::getTTypeGlobalReference(
57*0b57cec5SDimitry Andric     const GlobalValue *GV, unsigned Encoding, const TargetMachine &TM,
58*0b57cec5SDimitry Andric     MachineModuleInfo *MMI, MCStreamer &Streamer) const {
59*0b57cec5SDimitry Andric   if (TM.getMCAsmInfo()->getExceptionHandlingType() != ExceptionHandling::ARM)
60*0b57cec5SDimitry Andric     return TargetLoweringObjectFileELF::getTTypeGlobalReference(
61*0b57cec5SDimitry Andric         GV, Encoding, TM, MMI, Streamer);
62*0b57cec5SDimitry Andric 
63*0b57cec5SDimitry Andric   assert(Encoding == DW_EH_PE_absptr && "Can handle absptr encoding only");
64*0b57cec5SDimitry Andric 
65*0b57cec5SDimitry Andric   return MCSymbolRefExpr::create(TM.getSymbol(GV),
66*0b57cec5SDimitry Andric                                  MCSymbolRefExpr::VK_ARM_TARGET2, getContext());
67*0b57cec5SDimitry Andric }
68*0b57cec5SDimitry Andric 
69*0b57cec5SDimitry Andric const MCExpr *ARMElfTargetObjectFile::
70*0b57cec5SDimitry Andric getDebugThreadLocalSymbol(const MCSymbol *Sym) const {
71*0b57cec5SDimitry Andric   return MCSymbolRefExpr::create(Sym, MCSymbolRefExpr::VK_ARM_TLSLDO,
72*0b57cec5SDimitry Andric                                  getContext());
73*0b57cec5SDimitry Andric }
74*0b57cec5SDimitry Andric 
75*0b57cec5SDimitry Andric static bool isExecuteOnlyFunction(const GlobalObject *GO, SectionKind SK,
76*0b57cec5SDimitry Andric                                   const TargetMachine &TM) {
77*0b57cec5SDimitry Andric   if (const Function *F = dyn_cast<Function>(GO))
78*0b57cec5SDimitry Andric     if (TM.getSubtarget<ARMSubtarget>(*F).genExecuteOnly() && SK.isText())
79*0b57cec5SDimitry Andric       return true;
80*0b57cec5SDimitry Andric   return false;
81*0b57cec5SDimitry Andric }
82*0b57cec5SDimitry Andric 
83*0b57cec5SDimitry Andric MCSection *ARMElfTargetObjectFile::getExplicitSectionGlobal(
84*0b57cec5SDimitry Andric     const GlobalObject *GO, SectionKind SK, const TargetMachine &TM) const {
85*0b57cec5SDimitry Andric   // Set execute-only access for the explicit section
86*0b57cec5SDimitry Andric   if (isExecuteOnlyFunction(GO, SK, TM))
87*0b57cec5SDimitry Andric     SK = SectionKind::getExecuteOnly();
88*0b57cec5SDimitry Andric 
89*0b57cec5SDimitry Andric   return TargetLoweringObjectFileELF::getExplicitSectionGlobal(GO, SK, TM);
90*0b57cec5SDimitry Andric }
91*0b57cec5SDimitry Andric 
92*0b57cec5SDimitry Andric MCSection *ARMElfTargetObjectFile::SelectSectionForGlobal(
93*0b57cec5SDimitry Andric     const GlobalObject *GO, SectionKind SK, const TargetMachine &TM) const {
94*0b57cec5SDimitry Andric   // Place the global in the execute-only text section
95*0b57cec5SDimitry Andric   if (isExecuteOnlyFunction(GO, SK, TM))
96*0b57cec5SDimitry Andric     SK = SectionKind::getExecuteOnly();
97*0b57cec5SDimitry Andric 
98*0b57cec5SDimitry Andric   return TargetLoweringObjectFileELF::SelectSectionForGlobal(GO, SK, TM);
99*0b57cec5SDimitry Andric }
100