llvm-project/llvm/lib/MC/MCAsmInfoGOFF.cpp

162 lines
4.9 KiB
C++

//===- MCAsmInfoGOFF.cpp - MCGOFFAsmInfo properties -----------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
///
/// \file
/// This file defines certain target specific asm properties for GOFF (z/OS)
/// based targets.
///
//===----------------------------------------------------------------------===//
#include "llvm/MC/MCAsmInfoGOFF.h"
#include "llvm/BinaryFormat/GOFF.h"
#include "llvm/MC/MCSectionGOFF.h"
#include "llvm/Support/raw_ostream.h"
using namespace llvm;
MCAsmInfoGOFF::MCAsmInfoGOFF() {
Data64bitsDirective = "\t.quad\t";
HasDotTypeDotSizeDirective = false;
PrivateGlobalPrefix = "L#";
PrivateLabelPrefix = "L#";
ZeroDirective = "\t.space\t";
}
static void emitCATTR(raw_ostream &OS, StringRef Name, GOFF::ESDRmode Rmode,
GOFF::ESDAlignment Alignment,
GOFF::ESDLoadingBehavior LoadBehavior,
GOFF::ESDExecutable Executable, bool IsReadOnly,
uint32_t SortKey, uint8_t FillByteValue,
StringRef PartName) {
OS << Name << " CATTR ";
OS << "ALIGN(" << static_cast<unsigned>(Alignment) << "),"
<< "FILL(" << static_cast<unsigned>(FillByteValue) << ")";
switch (LoadBehavior) {
case GOFF::ESD_LB_Deferred:
OS << ",DEFLOAD";
break;
case GOFF::ESD_LB_NoLoad:
OS << ",NOLOAD";
break;
default:
break;
}
switch (Executable) {
case GOFF::ESD_EXE_CODE:
OS << ",EXECUTABLE";
break;
case GOFF::ESD_EXE_DATA:
OS << ",NOTEXECUTABLE";
break;
default:
break;
}
if (IsReadOnly)
OS << ",READONLY";
if (Rmode != GOFF::ESD_RMODE_None) {
OS << ',';
OS << "RMODE(";
switch (Rmode) {
case GOFF::ESD_RMODE_None:
llvm_unreachable("");
case GOFF::ESD_RMODE_24:
OS << "24";
break;
case GOFF::ESD_RMODE_31:
OS << "31";
break;
case GOFF::ESD_RMODE_64:
OS << "64";
break;
}
OS << ')';
}
if (SortKey)
OS << ",PRIORITY(" << SortKey << ")";
if (!PartName.empty())
OS << ",PART(" << PartName << ")";
OS << '\n';
}
static void emitXATTR(raw_ostream &OS, StringRef Name,
GOFF::ESDLinkageType Linkage,
GOFF::ESDExecutable Executable,
GOFF::ESDBindingScope BindingScope) {
OS << Name << " XATTR ";
OS << "LINKAGE(" << (Linkage == GOFF::ESD_LT_OS ? "OS" : "XPLINK") << "),";
if (Executable != GOFF::ESD_EXE_Unspecified)
OS << "REFERENCE(" << (Executable == GOFF::ESD_EXE_CODE ? "CODE" : "DATA")
<< "),";
if (BindingScope != GOFF::ESD_BSC_Unspecified) {
OS << "SCOPE(";
switch (BindingScope) {
case GOFF::ESD_BSC_Section:
OS << "SECTION";
break;
case GOFF::ESD_BSC_Module:
OS << "MODULE";
break;
case GOFF::ESD_BSC_Library:
OS << "LIBRARY";
break;
case GOFF::ESD_BSC_ImportExport:
OS << "EXPORT";
break;
default:
break;
}
OS << ')';
}
OS << '\n';
}
void MCAsmInfoGOFF::printSwitchToSection(const MCSection &Section,
uint32_t Subsection, const Triple &T,
raw_ostream &OS) const {
auto &Sec =
const_cast<MCSectionGOFF &>(static_cast<const MCSectionGOFF &>(Section));
switch (Sec.SymbolType) {
case GOFF::ESD_ST_SectionDefinition: {
OS << Sec.getName() << " CSECT\n";
Sec.Emitted = true;
break;
}
case GOFF::ESD_ST_ElementDefinition: {
printSwitchToSection(*Sec.getParent(), Subsection, T, OS);
if (!Sec.Emitted) {
emitCATTR(OS, Sec.getName(), Sec.EDAttributes.Rmode,
Sec.EDAttributes.Alignment, Sec.EDAttributes.LoadBehavior,
GOFF::ESD_EXE_Unspecified, Sec.EDAttributes.IsReadOnly, 0,
Sec.EDAttributes.FillByteValue, StringRef());
Sec.Emitted = true;
} else
OS << Sec.getName() << " CATTR\n";
break;
}
case GOFF::ESD_ST_PartReference: {
MCSectionGOFF *ED = Sec.getParent();
printSwitchToSection(*ED->getParent(), Subsection, T, OS);
if (!Sec.Emitted) {
emitCATTR(OS, ED->getName(), ED->getEDAttributes().Rmode,
ED->EDAttributes.Alignment, ED->EDAttributes.LoadBehavior,
Sec.PRAttributes.Executable, ED->EDAttributes.IsReadOnly,
Sec.PRAttributes.SortKey, ED->EDAttributes.FillByteValue,
Sec.getName());
emitXATTR(OS, Sec.getName(), Sec.PRAttributes.Linkage,
Sec.PRAttributes.Executable, Sec.PRAttributes.BindingScope);
ED->Emitted = true;
Sec.Emitted = true;
} else
OS << ED->getName() << " CATTR PART(" << Sec.getName() << ")\n";
break;
}
default:
llvm_unreachable("Wrong section type");
}
}