|  | //===- MC/MCAsmInfoXCOFF.cpp - XCOFF asm properties ------------ *- C++ -*-===// | 
|  | // | 
|  | // 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 "llvm/MC/MCAsmInfoXCOFF.h" | 
|  | #include "llvm/ADT/StringExtras.h" | 
|  | #include "llvm/MC/MCAsmInfo.h" | 
|  | #include "llvm/MC/MCSectionXCOFF.h" | 
|  | #include "llvm/Support/CommandLine.h" | 
|  | #include "llvm/Support/Format.h" | 
|  | #include "llvm/Support/raw_ostream.h" | 
|  |  | 
|  | using namespace llvm; | 
|  |  | 
|  | namespace llvm { | 
|  | extern cl::opt<cl::boolOrDefault> UseLEB128Directives; | 
|  | } | 
|  |  | 
|  | MCAsmInfoXCOFF::MCAsmInfoXCOFF() { | 
|  | IsAIX = true; | 
|  | IsLittleEndian = false; | 
|  |  | 
|  | PrivateGlobalPrefix = "L.."; | 
|  | PrivateLabelPrefix = "L.."; | 
|  | SupportsQuotedNames = false; | 
|  | if (UseLEB128Directives == cl::BOU_UNSET) | 
|  | HasLEB128Directives = false; | 
|  | ZeroDirective = "\t.space\t"; | 
|  | AsciiDirective = nullptr; // not supported | 
|  | AscizDirective = nullptr; // not supported | 
|  | CharacterLiteralSyntax = ACLS_SingleQuotePrefix; | 
|  |  | 
|  | // Use .vbyte for data definition to avoid directives that apply an implicit | 
|  | // alignment. | 
|  | Data16bitsDirective = "\t.vbyte\t2, "; | 
|  | Data32bitsDirective = "\t.vbyte\t4, "; | 
|  |  | 
|  | COMMDirectiveAlignmentIsInBytes = false; | 
|  | LCOMMDirectiveAlignmentType = LCOMM::Log2Alignment; | 
|  | HasDotTypeDotSizeDirective = false; | 
|  | ParseInlineAsmUsingAsmParser = true; | 
|  |  | 
|  | ExceptionsType = ExceptionHandling::AIX; | 
|  | } | 
|  |  | 
|  | bool MCAsmInfoXCOFF::isAcceptableChar(char C) const { | 
|  | // QualName is allowed for a MCSymbolXCOFF, and | 
|  | // QualName contains '[' and ']'. | 
|  | if (C == '[' || C == ']') | 
|  | return true; | 
|  |  | 
|  | // For AIX assembler, symbols may consist of numeric digits, | 
|  | // underscores, periods, uppercase or lowercase letters, or | 
|  | // any combination of these. | 
|  | return isAlnum(C) || C == '_' || C == '.'; | 
|  | } | 
|  |  | 
|  | bool MCAsmInfoXCOFF::useCodeAlign(const MCSection &Sec) const { | 
|  | return static_cast<const MCSectionXCOFF &>(Sec).getKind().isText(); | 
|  | } | 
|  |  | 
|  | MCSectionXCOFF::~MCSectionXCOFF() = default; | 
|  |  | 
|  | void MCSectionXCOFF::printCsectDirective(raw_ostream &OS) const { | 
|  | OS << "\t.csect " << QualName->getName() << "," << Log2(getAlign()) << '\n'; | 
|  | } | 
|  |  | 
|  | void MCAsmInfoXCOFF::printSwitchToSection(const MCSection &Section, uint32_t, | 
|  | const Triple &T, | 
|  | raw_ostream &OS) const { | 
|  | auto &Sec = static_cast<const MCSectionXCOFF &>(Section); | 
|  | if (Sec.getKind().isText()) { | 
|  | if (Sec.getMappingClass() != XCOFF::XMC_PR) | 
|  | report_fatal_error("Unhandled storage-mapping class for .text csect"); | 
|  |  | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (Sec.getKind().isReadOnly()) { | 
|  | if (Sec.getMappingClass() != XCOFF::XMC_RO && | 
|  | Sec.getMappingClass() != XCOFF::XMC_TD) | 
|  | report_fatal_error("Unhandled storage-mapping class for .rodata csect."); | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (Sec.getKind().isReadOnlyWithRel()) { | 
|  | if (Sec.getMappingClass() != XCOFF::XMC_RW && | 
|  | Sec.getMappingClass() != XCOFF::XMC_RO && | 
|  | Sec.getMappingClass() != XCOFF::XMC_TD) | 
|  | report_fatal_error( | 
|  | "Unexepected storage-mapping class for ReadOnlyWithRel kind"); | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Initialized TLS data. | 
|  | if (Sec.getKind().isThreadData()) { | 
|  | // We only expect XMC_TL here for initialized TLS data. | 
|  | if (Sec.getMappingClass() != XCOFF::XMC_TL) | 
|  | report_fatal_error("Unhandled storage-mapping class for .tdata csect."); | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (Sec.getKind().isData()) { | 
|  | switch (Sec.getMappingClass()) { | 
|  | case XCOFF::XMC_RW: | 
|  | case XCOFF::XMC_DS: | 
|  | case XCOFF::XMC_TD: | 
|  | Sec.printCsectDirective(OS); | 
|  | break; | 
|  | case XCOFF::XMC_TC: | 
|  | case XCOFF::XMC_TE: | 
|  | break; | 
|  | case XCOFF::XMC_TC0: | 
|  | OS << "\t.toc\n"; | 
|  | break; | 
|  | default: | 
|  | report_fatal_error("Unhandled storage-mapping class for .data csect."); | 
|  | } | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (Sec.isCsect() && Sec.getMappingClass() == XCOFF::XMC_TD) { | 
|  | // Common csect type (uninitialized storage) does not have to print | 
|  | // csect directive for section switching unless it is local. | 
|  | if (Sec.getKind().isCommon() && !Sec.getKind().isBSSLocal()) | 
|  | return; | 
|  |  | 
|  | assert(Sec.getKind().isBSS() && "Unexpected section kind for toc-data"); | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  | // Common csect type (uninitialized storage) does not have to print csect | 
|  | // directive for section switching. | 
|  | if (Sec.isCsect() && Sec.getCSectType() == XCOFF::XTY_CM) { | 
|  | assert((Sec.getMappingClass() == XCOFF::XMC_RW || | 
|  | Sec.getMappingClass() == XCOFF::XMC_BS || | 
|  | Sec.getMappingClass() == XCOFF::XMC_UL) && | 
|  | "Generated a storage-mapping class for a common/bss/tbss csect we " | 
|  | "don't " | 
|  | "understand how to switch to."); | 
|  | // Common symbols and local zero-initialized symbols for TLS and Non-TLS are | 
|  | // eligible for .bss/.tbss csect, getKind().isThreadBSS() is used to | 
|  | // cover TLS common and zero-initialized local symbols since linkage type | 
|  | // (in the GlobalVariable) is not accessible in this class. | 
|  | assert((Sec.getKind().isBSSLocal() || Sec.getKind().isCommon() || | 
|  | Sec.getKind().isThreadBSS()) && | 
|  | "wrong symbol type for .bss/.tbss csect"); | 
|  | // Don't have to print a directive for switching to section for commons | 
|  | // and zero-initialized TLS data. The '.comm' and '.lcomm' directives of the | 
|  | // variable will create the needed csect. | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Zero-initialized TLS data with weak or external linkage are not eligible to | 
|  | // be put into common csect. | 
|  | if (Sec.getKind().isThreadBSS()) { | 
|  | Sec.printCsectDirective(OS); | 
|  | return; | 
|  | } | 
|  |  | 
|  | // XCOFF debug sections. | 
|  | if (Sec.getKind().isMetadata() && Sec.isDwarfSect()) { | 
|  | OS << "\n\t.dwsect " << format("0x%" PRIx32, *Sec.getDwarfSubtypeFlags()) | 
|  | << '\n'; | 
|  | OS << Sec.getName() << ':' << '\n'; | 
|  | return; | 
|  | } | 
|  |  | 
|  | report_fatal_error("Printing for this SectionKind is unimplemented."); | 
|  | } |