llvm-project/llvm/lib/MC/MCAsmInfoXCOFF.cpp
Fangrui Song 87c73f498d Move MCSection::printSwitchToSection to MCAsmInfo
This removes the only virtual function of MCSection.

NVPTXTargetStreamer::changeSection uses the MCSectionELF print method.
Change it to just print the section name.
2025-07-26 15:42:05 -07:00

179 lines
6.0 KiB
C++

//===- 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.");
}