//===-- AVRTargetObjectFile.cpp - AVR Object Files ------------------------===// // // 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 // //===----------------------------------------------------------------------===// #include "AVRTargetObjectFile.h" #include "AVRTargetMachine.h" #include "llvm/BinaryFormat/ELF.h" #include "llvm/IR/DerivedTypes.h" #include "llvm/IR/GlobalValue.h" #include "llvm/IR/Mangler.h" #include "llvm/MC/MCContext.h" #include "llvm/MC/MCSectionELF.h" #include "AVR.h" namespace llvm { void AVRTargetObjectFile::Initialize(MCContext &Ctx, const TargetMachine &TM) { Base::Initialize(Ctx, TM); ProgmemDataSection = Ctx.getELFSection(".progmem.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); Progmem1DataSection = Ctx.getELFSection(".progmem1.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); Progmem2DataSection = Ctx.getELFSection(".progmem2.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); Progmem3DataSection = Ctx.getELFSection(".progmem3.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); Progmem4DataSection = Ctx.getELFSection(".progmem4.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); Progmem5DataSection = Ctx.getELFSection(".progmem5.data", ELF::SHT_PROGBITS, ELF::SHF_ALLOC); } MCSection *AVRTargetObjectFile::SelectSectionForGlobal( const GlobalObject *GO, SectionKind Kind, const TargetMachine &TM) const { // Global values in flash memory are placed in the progmem*.data section // unless they already have a user assigned section. const auto &AVRTM = static_cast(TM); if (AVR::isProgramMemoryAddress(GO) && !GO->hasSection() && Kind.isReadOnly()) { // The AVR subtarget should support LPM to access section '.progmem*.data'. if (!AVRTM.getSubtargetImpl()->hasLPM()) { // TODO: Get the global object's location in source file. getContext().reportError( SMLoc(), "Current AVR subtarget does not support accessing program memory"); return Base::SelectSectionForGlobal(GO, Kind, TM); } // The AVR subtarget should support ELPM to access section // '.progmem[1|2|3|4|5].data'. if (!AVRTM.getSubtargetImpl()->hasELPM() && AVR::getAddressSpace(GO) != AVR::ProgramMemory) { // TODO: Get the global object's location in source file. getContext().reportError(SMLoc(), "Current AVR subtarget does not support " "accessing extended program memory"); return ProgmemDataSection; } switch (AVR::getAddressSpace(GO)) { case AVR::ProgramMemory: // address space 1 return ProgmemDataSection; case AVR::ProgramMemory1: // address space 2 return Progmem1DataSection; case AVR::ProgramMemory2: // address space 3 return Progmem2DataSection; case AVR::ProgramMemory3: // address space 4 return Progmem3DataSection; case AVR::ProgramMemory4: // address space 5 return Progmem4DataSection; case AVR::ProgramMemory5: // address space 6 return Progmem5DataSection; default: llvm_unreachable("unexpected program memory index"); } } // Otherwise, we work the same way as ELF. return Base::SelectSectionForGlobal(GO, Kind, TM); } } // end of namespace llvm