15c96de3aSDylan McKay //===-- AVRTargetObjectFile.cpp - AVR Object Files ------------------------===//
25c96de3aSDylan McKay //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65c96de3aSDylan McKay //
75c96de3aSDylan McKay //===----------------------------------------------------------------------===//
85c96de3aSDylan McKay 
95c96de3aSDylan McKay #include "AVRTargetObjectFile.h"
10*c1dd6074SBen Shi #include "AVRTargetMachine.h"
115c96de3aSDylan McKay 
12264b5d9eSZachary Turner #include "llvm/BinaryFormat/ELF.h"
135c96de3aSDylan McKay #include "llvm/IR/DerivedTypes.h"
145c96de3aSDylan McKay #include "llvm/IR/GlobalValue.h"
155c96de3aSDylan McKay #include "llvm/IR/Mangler.h"
165c96de3aSDylan McKay #include "llvm/MC/MCContext.h"
175c96de3aSDylan McKay #include "llvm/MC/MCSectionELF.h"
185c96de3aSDylan McKay 
195c96de3aSDylan McKay #include "AVR.h"
205c96de3aSDylan McKay 
215c96de3aSDylan McKay namespace llvm {
Initialize(MCContext & Ctx,const TargetMachine & TM)225c96de3aSDylan McKay void AVRTargetObjectFile::Initialize(MCContext &Ctx, const TargetMachine &TM) {
235c96de3aSDylan McKay   Base::Initialize(Ctx, TM);
245c96de3aSDylan McKay   ProgmemDataSection =
255c96de3aSDylan McKay       Ctx.getELFSection(".progmem.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
26*c1dd6074SBen Shi   Progmem1DataSection =
27*c1dd6074SBen Shi       Ctx.getELFSection(".progmem1.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
28*c1dd6074SBen Shi   Progmem2DataSection =
29*c1dd6074SBen Shi       Ctx.getELFSection(".progmem2.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
30*c1dd6074SBen Shi   Progmem3DataSection =
31*c1dd6074SBen Shi       Ctx.getELFSection(".progmem3.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
32*c1dd6074SBen Shi   Progmem4DataSection =
33*c1dd6074SBen Shi       Ctx.getELFSection(".progmem4.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
34*c1dd6074SBen Shi   Progmem5DataSection =
35*c1dd6074SBen Shi       Ctx.getELFSection(".progmem5.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC);
365c96de3aSDylan McKay }
375c96de3aSDylan McKay 
SelectSectionForGlobal(const GlobalObject * GO,SectionKind Kind,const TargetMachine & TM) const385449d2daSShivam Gupta MCSection *AVRTargetObjectFile::SelectSectionForGlobal(
395449d2daSShivam Gupta     const GlobalObject *GO, SectionKind Kind, const TargetMachine &TM) const {
40*c1dd6074SBen Shi   // Global values in flash memory are placed in the progmem*.data section
415c96de3aSDylan McKay   // unless they already have a user assigned section.
42*c1dd6074SBen Shi   const auto &AVRTM = static_cast<const AVRTargetMachine &>(TM);
43*c1dd6074SBen Shi   if (AVR::isProgramMemoryAddress(GO) && !GO->hasSection() &&
44*c1dd6074SBen Shi       Kind.isReadOnly()) {
45*c1dd6074SBen Shi     // The AVR subtarget should support LPM to access section '.progmem*.data'.
46*c1dd6074SBen Shi     if (!AVRTM.getSubtargetImpl()->hasLPM()) {
47*c1dd6074SBen Shi       // TODO: Get the global object's location in source file.
48*c1dd6074SBen Shi       getContext().reportError(
49*c1dd6074SBen Shi           SMLoc(),
50*c1dd6074SBen Shi           "Current AVR subtarget does not support accessing program memory");
51*c1dd6074SBen Shi       return Base::SelectSectionForGlobal(GO, Kind, TM);
52*c1dd6074SBen Shi     }
53*c1dd6074SBen Shi     // The AVR subtarget should support ELPM to access section
54*c1dd6074SBen Shi     // '.progmem[1|2|3|4|5].data'.
55*c1dd6074SBen Shi     if (!AVRTM.getSubtargetImpl()->hasELPM() &&
56*c1dd6074SBen Shi         AVR::getAddressSpace(GO) != AVR::ProgramMemory) {
57*c1dd6074SBen Shi       // TODO: Get the global object's location in source file.
58*c1dd6074SBen Shi       getContext().reportError(SMLoc(),
59*c1dd6074SBen Shi                                "Current AVR subtarget does not support "
60*c1dd6074SBen Shi                                "accessing extended program memory");
615c96de3aSDylan McKay       return ProgmemDataSection;
62*c1dd6074SBen Shi     }
63*c1dd6074SBen Shi     switch (AVR::getAddressSpace(GO)) {
64*c1dd6074SBen Shi     case AVR::ProgramMemory: // address space 1
65*c1dd6074SBen Shi       return ProgmemDataSection;
66*c1dd6074SBen Shi     case AVR::ProgramMemory1: // address space 2
67*c1dd6074SBen Shi       return Progmem1DataSection;
68*c1dd6074SBen Shi     case AVR::ProgramMemory2: // address space 3
69*c1dd6074SBen Shi       return Progmem2DataSection;
70*c1dd6074SBen Shi     case AVR::ProgramMemory3: // address space 4
71*c1dd6074SBen Shi       return Progmem3DataSection;
72*c1dd6074SBen Shi     case AVR::ProgramMemory4: // address space 5
73*c1dd6074SBen Shi       return Progmem4DataSection;
74*c1dd6074SBen Shi     case AVR::ProgramMemory5: // address space 6
75*c1dd6074SBen Shi       return Progmem5DataSection;
76*c1dd6074SBen Shi     default:
77*c1dd6074SBen Shi       llvm_unreachable("unexpected program memory index");
78*c1dd6074SBen Shi     }
79*c1dd6074SBen Shi   }
805c96de3aSDylan McKay 
815c96de3aSDylan McKay   // Otherwise, we work the same way as ELF.
826733564eSPeter Collingbourne   return Base::SelectSectionForGlobal(GO, Kind, TM);
835c96de3aSDylan McKay }
845c96de3aSDylan McKay } // end of namespace llvm
85