https://github.com/ysyeda updated https://github.com/llvm/llvm-project/pull/65407:
>From 16ecf1be6925403d34d52edad2263aa2f3c3c0b3 Mon Sep 17 00:00:00 2001 From: Yusra Syeda <yusra.sy...@ibm.com> Date: Tue, 5 Sep 2023 15:43:24 -0400 Subject: [PATCH] [SystemZ][z/OS] This change adds support for the PPA2 section in zOS --- clang/include/clang/Basic/LangStandard.h | 1 + clang/lib/Basic/LangStandards.cpp | 39 ++++ clang/lib/CodeGen/CodeGenModule.cpp | 15 ++ clang/lib/Driver/ToolChains/Clang.cpp | 13 +- clang/lib/Driver/ToolChains/Clang.h | 3 +- clang/test/CodeGen/SystemZ/systemz-ppa2.c | 25 +++ llvm/include/llvm/BinaryFormat/GOFF.h | 1 + llvm/include/llvm/MC/MCObjectFileInfo.h | 4 + llvm/include/llvm/Support/Chrono.h | 26 +++ llvm/lib/MC/MCObjectFileInfo.cpp | 5 + llvm/lib/Support/Chrono.cpp | 91 +++++--- llvm/lib/Target/SystemZ/SystemZAsmPrinter.cpp | 197 +++++++++++++++++- llvm/lib/Target/SystemZ/SystemZAsmPrinter.h | 7 +- llvm/test/CodeGen/SystemZ/zos-ppa2.ll | 27 +++ 14 files changed, 422 insertions(+), 32 deletions(-) create mode 100644 clang/test/CodeGen/SystemZ/systemz-ppa2.c create mode 100644 llvm/test/CodeGen/SystemZ/zos-ppa2.ll diff --git a/clang/include/clang/Basic/LangStandard.h b/clang/include/clang/Basic/LangStandard.h index 6356f16acc811e..d94567adf2bfb9 100644 --- a/clang/include/clang/Basic/LangStandard.h +++ b/clang/include/clang/Basic/LangStandard.h @@ -43,6 +43,7 @@ enum class Language : uint8_t { HLSL, ///@} }; +const char *LanguageToString(Language L); enum LangFeatures { LineComment = (1 << 0), diff --git a/clang/lib/Basic/LangStandards.cpp b/clang/lib/Basic/LangStandards.cpp index af9cf4f273920e..e9b75d78e820a6 100644 --- a/clang/lib/Basic/LangStandards.cpp +++ b/clang/lib/Basic/LangStandards.cpp @@ -10,9 +10,48 @@ #include "clang/Config/config.h" #include "llvm/ADT/StringSwitch.h" #include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/FormatVariadic.h" #include "llvm/TargetParser/Triple.h" using namespace clang; +const char *clang::LanguageToString(Language L) { + // I would like to make this function and the definition of Language + // in the .h file simply expand the contents of a .def file. + // However, in the .h the members of the enum have doxygen annotations + // and/or comments which would be lost. + switch (L) { + case Language::Unknown: + return "Unknown"; + case Language::Asm: + return "Asm"; + case Language::LLVM_IR: + return "LLVM_IR"; + case Language::C: + return "C"; + case Language::CXX: + return "CXX"; + case Language::ObjC: + return "ObjC"; + case Language::ObjCXX: + return "ObjCXX"; + case Language::OpenCL: + return "OpenCL"; + case Language::OpenCLCXX: + return "OpenCLCXX"; + case Language::CUDA: + return "CUDA"; + case Language::RenderScript: + return "RenderScript"; + case Language::HIP: + return "HIP"; + case Language::HLSL: + return "HLSL"; + } + + std::string msg = llvm::formatv("Unknown value ({0}) passed to LanguageToString", (unsigned int) L); + llvm_unreachable(msg.c_str()); +} + #define LANGSTANDARD(id, name, lang, desc, features) \ static const LangStandard Lang_##id = {name, desc, features, Language::lang}; #include "clang/Basic/LangStandards.def" diff --git a/clang/lib/CodeGen/CodeGenModule.cpp b/clang/lib/CodeGen/CodeGenModule.cpp index f5513217bebfba..239e3d2bcd1625 100644 --- a/clang/lib/CodeGen/CodeGenModule.cpp +++ b/clang/lib/CodeGen/CodeGenModule.cpp @@ -975,6 +975,21 @@ void CodeGenModule::Release() { Context.getTypeSizeInChars(Context.getWideCharType()).getQuantity(); getModule().addModuleFlag(llvm::Module::Error, "wchar_size", WCharWidth); + if (getTriple().isOSzOS()) { + int32_t ProductVersion, ProductRelease, ProductPatch; + ProductVersion = LLVM_VERSION_MAJOR, + ProductRelease = LLVM_VERSION_MINOR, ProductPatch = LLVM_VERSION_PATCH; + getModule().addModuleFlag(llvm::Module::Warning, "Product Major Version", ProductVersion); + getModule().addModuleFlag(llvm::Module::Warning, "Product Minor Version", ProductRelease); + getModule().addModuleFlag(llvm::Module::Warning, "Product Patchlevel", ProductPatch); + + // Record the language because we need it for the PPA2. + const char *lang_str = LanguageToString( + LangStandard::getLangStandardForKind(LangOpts.LangStd).Language); + getModule().addModuleFlag(llvm::Module::Error, "zos_cu_language", + llvm::MDString::get(VMContext, lang_str)); + } + llvm::Triple::ArchType Arch = Context.getTargetInfo().getTriple().getArch(); if ( Arch == llvm::Triple::arm || Arch == llvm::Triple::armeb diff --git a/clang/lib/Driver/ToolChains/Clang.cpp b/clang/lib/Driver/ToolChains/Clang.cpp index da5edda6b385bc..0196b17a7a33ae 100644 --- a/clang/lib/Driver/ToolChains/Clang.cpp +++ b/clang/lib/Driver/ToolChains/Clang.cpp @@ -1735,7 +1735,7 @@ void Clang::RenderTargetOptions(const llvm::Triple &EffectiveTriple, break; case llvm::Triple::systemz: - AddSystemZTargetArgs(Args, CmdArgs); + AddSystemZTargetArgs(EffectiveTriple, Args, CmdArgs); break; case llvm::Triple::x86: @@ -2232,7 +2232,8 @@ void Clang::AddSparcTargetArgs(const ArgList &Args, } } -void Clang::AddSystemZTargetArgs(const ArgList &Args, +void Clang::AddSystemZTargetArgs(const llvm::Triple &Triple, + const ArgList &Args, ArgStringList &CmdArgs) const { if (const Arg *A = Args.getLastArg(options::OPT_mtune_EQ)) { CmdArgs.push_back("-tune-cpu"); @@ -2264,6 +2265,14 @@ void Clang::AddSystemZTargetArgs(const ArgList &Args, CmdArgs.push_back("-mfloat-abi"); CmdArgs.push_back("soft"); } + + if (Triple.isOSzOS()) { + CmdArgs.push_back("-mllvm"); + CmdArgs.push_back( + Args.MakeArgString(llvm::Twine("-translation-time=") + .concat(llvm::Twine(std::time(nullptr))) + .str())); + } } void Clang::AddX86TargetArgs(const ArgList &Args, diff --git a/clang/lib/Driver/ToolChains/Clang.h b/clang/lib/Driver/ToolChains/Clang.h index 0f503c4bd1c4fe..9f065f846b4cf3 100644 --- a/clang/lib/Driver/ToolChains/Clang.h +++ b/clang/lib/Driver/ToolChains/Clang.h @@ -69,7 +69,8 @@ class LLVM_LIBRARY_VISIBILITY Clang : public Tool { llvm::opt::ArgStringList &CmdArgs) const; void AddSparcTargetArgs(const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CmdArgs) const; - void AddSystemZTargetArgs(const llvm::opt::ArgList &Args, + void AddSystemZTargetArgs(const llvm::Triple &Triple, + const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CmdArgs) const; void AddX86TargetArgs(const llvm::opt::ArgList &Args, llvm::opt::ArgStringList &CmdArgs) const; diff --git a/clang/test/CodeGen/SystemZ/systemz-ppa2.c b/clang/test/CodeGen/SystemZ/systemz-ppa2.c new file mode 100644 index 00000000000000..8f396fc12c75fc --- /dev/null +++ b/clang/test/CodeGen/SystemZ/systemz-ppa2.c @@ -0,0 +1,25 @@ +// Please note the following: +// + we are checking that the first bytes of the PPA2 are 0x3 0x0 +// for C, and 0x3 0x1 for C++ +// + the label for the PPA2 seems to vary on different versions. +// We try to cover all cases, and use substitution blocks to +// help write the tests. The contents of the PPA2 itself should +// not be different. +// + the [[:space:]] combines the two .byte lines into one pattern. +// This is necessary because if the lines were separated, the first +// .byte (i.e., the one for the 3) would, it seems, also match +// the .byte line below for the 34. + +// RUN: %clang --target=s390x-ibm-zos -xc -S -o - %s | FileCheck %s --check-prefix CHECK-C +// CHECK-C: [[PPA2:(.L)|(@@)PPA2]]: +// CHECK-C-NEXT: .byte 3{{[[:space:]]*}}.byte 0 +// CHECK-C-NEXT: .byte 34{{$}} +// CHECK-C-NEXT: .byte {{4}} +// CHECK-C-NEXT: .long {{(CELQSTRT)}}-[[PPA2]] + +// RUN: %clang --target=s390x-ibm-zos -xc++ -S -o - %s | FileCheck %s --check-prefix CHECK-CXX +// CHECK-CXX: [[PPA2:(.L)|(@@)PPA2]]: +// CHECK-CXX-NEXT: .byte 3{{[[:space:]]*}}.byte 1 +// CHECK-CXX-NEXT: .byte 34{{$}} +// CHECK-CXX-NEXT: .byte {{4}} +// CHECK-CXX-NEXT: .long {{(CELQSTRT)}}-[[PPA2]] diff --git a/llvm/include/llvm/BinaryFormat/GOFF.h b/llvm/include/llvm/BinaryFormat/GOFF.h index b4ddbabdf1e4e7..7837776f21ee58 100644 --- a/llvm/include/llvm/BinaryFormat/GOFF.h +++ b/llvm/include/llvm/BinaryFormat/GOFF.h @@ -154,6 +154,7 @@ enum ENDEntryPointRequest : uint8_t { // \brief Subsections of the primary C_CODE section in the object file. enum SubsectionKind : uint8_t { SK_PPA1 = 2, + SK_PPA2 = 4, }; } // end namespace GOFF diff --git a/llvm/include/llvm/MC/MCObjectFileInfo.h b/llvm/include/llvm/MC/MCObjectFileInfo.h index 54f696cb795fbc..2b2adf5012defa 100644 --- a/llvm/include/llvm/MC/MCObjectFileInfo.h +++ b/llvm/include/llvm/MC/MCObjectFileInfo.h @@ -227,7 +227,9 @@ class MCObjectFileInfo { // GOFF specific sections. MCSection *PPA1Section = nullptr; + MCSection *PPA2Section = nullptr; MCSection *ADASection = nullptr; + MCSection *IDRLSection = nullptr; // XCOFF specific sections MCSection *TOCBaseSection = nullptr; @@ -431,7 +433,9 @@ class MCObjectFileInfo { // GOFF specific sections. MCSection *getPPA1Section() const { return PPA1Section; } + MCSection *getPPA2Section() const { return PPA2Section; } MCSection *getADASection() const { return ADASection; } + MCSection *getIDRLSection() const { return IDRLSection; } // XCOFF specific sections MCSection *getTOCBaseSection() const { return TOCBaseSection; } diff --git a/llvm/include/llvm/Support/Chrono.h b/llvm/include/llvm/Support/Chrono.h index 9c2bd45d2803e5..71859af7c7e4a5 100644 --- a/llvm/include/llvm/Support/Chrono.h +++ b/llvm/include/llvm/Support/Chrono.h @@ -33,6 +33,19 @@ namespace sys { template <typename D = std::chrono::nanoseconds> using TimePoint = std::chrono::time_point<std::chrono::system_clock, D>; +// utc_clock and utc_time are only available since C++20. Add enough code to +// support formatting date/time in UTC. +class UtcClock : public std::chrono::system_clock {}; + +template <typename D = std::chrono::nanoseconds> +using UtcTime = std::chrono::time_point<UtcClock, D>; + +/// Convert a std::time_t to a UtcTime +inline UtcTime<std::chrono::seconds> toUtcTime(std::time_t T) { + using namespace std::chrono; + return UtcTime<seconds>(seconds(T)); +} + /// Convert a TimePoint to std::time_t inline std::time_t toTimeT(TimePoint<> TP) { using namespace std::chrono; @@ -40,6 +53,13 @@ inline std::time_t toTimeT(TimePoint<> TP) { time_point_cast<system_clock::time_point::duration>(TP)); } +/// Convert a UtcTime to std::time_t +inline std::time_t toTimeT(UtcTime<> TP) { + using namespace std::chrono; + return system_clock::to_time_t(time_point<system_clock, seconds>( + duration_cast<seconds>(TP.time_since_epoch()))); +} + /// Convert a std::time_t to a TimePoint inline TimePoint<std::chrono::seconds> toTimePoint(std::time_t T) { @@ -58,6 +78,7 @@ toTimePoint(std::time_t T, uint32_t nsec) { } // namespace sys raw_ostream &operator<<(raw_ostream &OS, sys::TimePoint<> TP); +raw_ostream &operator<<(raw_ostream &OS, sys::UtcTime<> TP); /// Format provider for TimePoint<> /// @@ -73,6 +94,11 @@ struct format_provider<sys::TimePoint<>> { StringRef Style); }; +template <> struct format_provider<sys::UtcTime<std::chrono::seconds>> { + static void format(const sys::UtcTime<std::chrono::seconds> &TP, + llvm::raw_ostream &OS, StringRef Style); +}; + namespace detail { template <typename Period> struct unit { static const char value[]; }; template <typename Period> const char unit<Period>::value[] = ""; diff --git a/llvm/lib/MC/MCObjectFileInfo.cpp b/llvm/lib/MC/MCObjectFileInfo.cpp index 0b5109e41e7171..1b30645cea3c1c 100644 --- a/llvm/lib/MC/MCObjectFileInfo.cpp +++ b/llvm/lib/MC/MCObjectFileInfo.cpp @@ -547,8 +547,13 @@ void MCObjectFileInfo::initGOFFMCObjectFileInfo(const Triple &T) { PPA1Section = Ctx->getGOFFSection(".ppa1", SectionKind::getMetadata(), TextSection, MCConstantExpr::create(GOFF::SK_PPA1, *Ctx)); + PPA2Section = + Ctx->getGOFFSection(".ppa2", SectionKind::getMetadata(), TextSection, + MCConstantExpr::create(GOFF::SK_PPA2, *Ctx)); ADASection = Ctx->getGOFFSection(".ada", SectionKind::getData(), nullptr, nullptr); + IDRLSection = + Ctx->getGOFFSection("B_IDRL", SectionKind::getData(), nullptr, nullptr); } void MCObjectFileInfo::initCOFFMCObjectFileInfo(const Triple &T) { diff --git a/llvm/lib/Support/Chrono.cpp b/llvm/lib/Support/Chrono.cpp index 859ece8f550080..4e64f03eea224e 100644 --- a/llvm/lib/Support/Chrono.cpp +++ b/llvm/lib/Support/Chrono.cpp @@ -40,6 +40,24 @@ static inline struct tm getStructTM(TimePoint<> TP) { return Storage; } +static inline struct tm getStructTMUtc(UtcTime<> TP) { + struct tm Storage; + std::time_t OurTime = toTimeT(TP); + +#if defined(LLVM_ON_UNIX) + struct tm *LT = ::gmtime_r(&OurTime, &Storage); + assert(LT); + (void)LT; +#endif +#if defined(_WIN32) + int Error = ::gmtime_s(&Storage, &OurTime); + assert(!Error); + (void)Error; +#endif + + return Storage; +} + raw_ostream &operator<<(raw_ostream &OS, TimePoint<> TP) { struct tm LT = getStructTM(TP); char Buffer[sizeof("YYYY-MM-DD HH:MM:SS")]; @@ -50,44 +68,63 @@ raw_ostream &operator<<(raw_ostream &OS, TimePoint<> TP) { .count())); } -void format_provider<TimePoint<>>::format(const TimePoint<> &T, raw_ostream &OS, - StringRef Style) { +template <class T> +static void format(const T &Fractional, struct tm <, raw_ostream &OS, + StringRef Style) { using namespace std::chrono; - TimePoint<seconds> Truncated = time_point_cast<seconds>(T); - auto Fractional = T - Truncated; - struct tm LT = getStructTM(Truncated); // Handle extensions first. strftime mangles unknown %x on some platforms. - if (Style.empty()) Style = "%Y-%m-%d %H:%M:%S.%N"; + if (Style.empty()) + Style = "%Y-%m-%d %H:%M:%S.%N"; std::string Format; raw_string_ostream FStream(Format); for (unsigned I = 0; I < Style.size(); ++I) { - if (Style[I] == '%' && Style.size() > I + 1) switch (Style[I + 1]) { - case 'L': // Milliseconds, from Ruby. - FStream << llvm::format( - "%.3lu", (long)duration_cast<milliseconds>(Fractional).count()); - ++I; - continue; - case 'f': // Microseconds, from Python. - FStream << llvm::format( - "%.6lu", (long)duration_cast<microseconds>(Fractional).count()); - ++I; - continue; - case 'N': // Nanoseconds, from date(1). - FStream << llvm::format( - "%.9lu", (long)duration_cast<nanoseconds>(Fractional).count()); - ++I; - continue; - case '%': // Consume %%, so %%f parses as (%%)f not %(%f) - FStream << "%%"; - ++I; - continue; + if (Style[I] == '%' && Style.size() > I + 1) + switch (Style[I + 1]) { + case 'L': // Milliseconds, from Ruby. + FStream << llvm::format( + "%.3lu", (long)duration_cast<milliseconds>(Fractional).count()); + ++I; + continue; + case 'f': // Microseconds, from Python. + FStream << llvm::format( + "%.6lu", (long)duration_cast<microseconds>(Fractional).count()); + ++I; + continue; + case 'N': // Nanoseconds, from date(1). + FStream << llvm::format( + "%.9lu", (long)duration_cast<nanoseconds>(Fractional).count()); + ++I; + continue; + case '%': // Consume %%, so %%f parses as (%%)f not %(%f) + FStream << "%%"; + ++I; + continue; } FStream << Style[I]; } FStream.flush(); - char Buffer[256]; // Should be enough for anywhen. + char Buffer[256]; // Should be enough for anywhen. size_t Len = strftime(Buffer, sizeof(Buffer), Format.c_str(), <); OS << (Len ? Buffer : "BAD-DATE-FORMAT"); } +void format_provider<UtcTime<std::chrono::seconds>>::format( + const UtcTime<std::chrono::seconds> &T, raw_ostream &OS, StringRef Style) { + using namespace std::chrono; + UtcTime<seconds> Truncated = + UtcTime<seconds>(duration_cast<seconds>(T.time_since_epoch())); + auto Fractional = T - Truncated; + struct tm LT = getStructTMUtc(Truncated); + llvm::format(Fractional, LT, OS, Style); +} + +void format_provider<TimePoint<>>::format(const TimePoint<> &T, raw_ostream &OS, + StringRef Style) { + using namespace std::chrono; + TimePoint<seconds> Truncated = time_point_cast<seconds>(T); + auto Fractional = T - Truncated; + struct tm LT = getStructTM(Truncated); + llvm::format(Fractional, LT, OS, Style); +} + } // namespace llvm diff --git a/llvm/lib/Target/SystemZ/SystemZAsmPrinter.cpp b/llvm/lib/Target/SystemZ/SystemZAsmPrinter.cpp index b3075c150ebb36..d963009c6c158f 100644 --- a/llvm/lib/Target/SystemZ/SystemZAsmPrinter.cpp +++ b/llvm/lib/Target/SystemZ/SystemZAsmPrinter.cpp @@ -27,10 +27,20 @@ #include "llvm/MC/MCSectionELF.h" #include "llvm/MC/MCStreamer.h" #include "llvm/MC/TargetRegistry.h" +#include "llvm/Support/Chrono.h" +#include "llvm/Support/CommandLine.h" #include "llvm/Support/ConvertEBCDIC.h" +#include "llvm/Support/CommandLine.h" +#include "llvm/Support/FormatProviders.h" +#include "llvm/Support/FormatVariadic.h" using namespace llvm; +cl::opt<uint64_t> TranslationTime( + "translation-time", + cl::desc("sets the time of compilation in seconds since epoch"), + cl::init(0)); + // Return an RI instruction like MI with opcode Opcode, but with the // GR64 register operands turned into GR32s. static MCInst lowerRILow(const MachineInstr *MI, unsigned Opcode) { @@ -953,6 +963,7 @@ void SystemZAsmPrinter::emitEndOfAsmFile(Module &M) { auto TT = OutContext.getTargetTriple(); if (TT.isOSzOS()) { emitADASection(); + emitIDRLSection(M); } emitAttributes(M); } @@ -1026,6 +1037,68 @@ void SystemZAsmPrinter::emitADASection() { OutStreamer->popSection(); } +static uint32_t getProductVersion(Module &M) { + if (auto *VersionVal = cast_or_null<ConstantAsMetadata>( + M.getModuleFlag("Product Major Version"))) + return cast<ConstantInt>(VersionVal->getValue())->getZExtValue(); + return LLVM_VERSION_MAJOR; +} + +static uint32_t getProductRelease(Module &M) { + if (auto *ReleaseVal = cast_or_null<ConstantAsMetadata>( + M.getModuleFlag("Product Minor Version"))) + return cast<ConstantInt>(ReleaseVal->getValue())->getZExtValue(); + return LLVM_VERSION_MINOR; +} + +static uint32_t getProductPatch(Module &M) { + if (auto *PatchVal = cast_or_null<ConstantAsMetadata>( + M.getModuleFlag("Product Patchlevel"))) + return cast<ConstantInt>(PatchVal->getValue())->getZExtValue(); + return LLVM_VERSION_PATCH; +} + +void SystemZAsmPrinter::emitIDRLSection(Module &M) { + OutStreamer->pushSection(); + OutStreamer->switchSection(getObjFileLowering().getIDRLSection()); + constexpr unsigned IDRLDataLength = 30; + std::time_t Time = TranslationTime; + + uint32_t ProductVersion = getProductVersion(M); + uint32_t ProductRelease = getProductRelease(M); + + std::string ProductID; + if (auto *MD = M.getModuleFlag("Product Id")) + ProductID = cast<MDString>(MD)->getString().str(); + + if (ProductID.empty()) { + char ProductIDFormatted[11]; // 10 + null. + snprintf(ProductIDFormatted, sizeof(ProductIDFormatted), "LLVM %02d%02d", + ProductVersion, ProductRelease); + ProductID = ProductIDFormatted; + } + + // Remove - from Product Id, which makes it consistent with legacy. + // The binder expects alphanumeric characters only. + std::size_t DashFound = ProductID.find("-"); + if (DashFound != std::string::npos) + ProductID.erase(ProductID.begin() + DashFound); + + SmallString<IDRLDataLength + 1> TempStr; + raw_svector_ostream O(TempStr); + O << formatv("{0}{1,0-2:d}{2,0-2:d}{3:%Y-%m-%d %H:%M:%S}", ProductID.c_str(), + ProductVersion, ProductRelease, llvm::sys::toUtcTime(Time)); + SmallString<IDRLDataLength> Data; + + ConverterEBCDIC::convertToEBCDIC(TempStr, Data); + + OutStreamer->emitInt8(0); // Reserved. + OutStreamer->emitInt8(3); // Format. + OutStreamer->emitInt16(IDRLDataLength); // Length. + OutStreamer->emitBytes(Data.str()); + OutStreamer->popSection(); +} + void SystemZAsmPrinter::emitFunctionBodyEnd() { if (TM.getTargetTriple().isOSzOS()) { // Emit symbol for the end of function if the z/OS target streamer @@ -1150,6 +1223,8 @@ static void emitPPA1Name(std::unique_ptr<MCStreamer> &OutStreamer, } void SystemZAsmPrinter::emitPPA1(MCSymbol *FnEndSym) { + assert(PPA2Sym != nullptr && "PPA2 Symbol not defined"); + const TargetRegisterInfo *TRI = MF->getRegInfo().getTargetRegisterInfo(); const SystemZSubtarget &Subtarget = MF->getSubtarget<SystemZSubtarget>(); const auto TargetHasVector = Subtarget.hasVector(); @@ -1239,6 +1314,8 @@ void SystemZAsmPrinter::emitPPA1(MCSymbol *FnEndSym) { OutStreamer->emitInt8(0xCE); // CEL signature. OutStreamer->AddComment("Saved GPR Mask"); OutStreamer->emitInt16(SavedGPRMask); + OutStreamer->AddComment("Offset to PPA2"); + OutStreamer->emitAbsoluteSymbolDiff(PPA2Sym, CurrentFnPPA1Sym, 4); bool HasName = MF->getFunction().hasName() && MF->getFunction().getName().size() > 0; @@ -1296,6 +1373,124 @@ void SystemZAsmPrinter::emitPPA1(MCSymbol *FnEndSym) { 4); } +void SystemZAsmPrinter::emitStartOfAsmFile(Module &M) { + if (TM.getTargetTriple().isOSzOS()) + emitPPA2(M); + AsmPrinter::emitStartOfAsmFile(M); +} + +void SystemZAsmPrinter::emitPPA2(Module &M) { + OutStreamer->pushSection(); + OutStreamer->switchSection(getObjFileLowering().getPPA2Section()); + MCContext &OutContext = OutStreamer->getContext(); + // Make CELQSTRT symbol. + const char *StartSymbolName = "CELQSTRT"; + MCSymbol *CELQSTRT = OutContext.getOrCreateSymbol(StartSymbolName); + + // Create symbol and assign to class field for use in PPA1. + PPA2Sym = OutContext.createTempSymbol("PPA2", false); + MCSymbol *DateVersionSym = OutContext.createTempSymbol("DVS", false); + + std::time_t Time = TranslationTime; + SmallString<15> CompilationTime; // 14 + null + raw_svector_ostream O(CompilationTime); + O << formatv("{0:%Y%m%d%H%M%S}", llvm::sys::toUtcTime(Time)); + + uint32_t ProductVersion = getProductVersion(M), + ProductRelease = getProductRelease(M), + ProductPatch = getProductPatch(M); + + SmallString<7> Version; // 6 + null + raw_svector_ostream ostr(Version); + ostr << formatv("{0,0-2:d}{1,0-2:d}{2,0-2:d}", ProductVersion, + ProductRelease, ProductPatch); + + // Drop 0 during conversion. + SmallString<sizeof(CompilationTime) - 1> CompilationTimeStr; + SmallString<sizeof(Version) - 1> VersionStr; + + ConverterEBCDIC::convertToEBCDIC(CompilationTime, CompilationTimeStr); + ConverterEBCDIC::convertToEBCDIC(Version, VersionStr); + + enum class PPA2MemberId : uint8_t { + // See z/OS Language Environment Vendor Interfaces v2r5, p.23, for + // complete list. Only the C runtime is supported by this backend. + LE_C_Runtime = 3, + }; + enum class PPA2MemberSubId : uint8_t { + // List of languages using the LE C runtime implementation. + C = 0x00, + CXX = 0x01, + Swift = 0x03, + Go = 0x60, + LLVMBasedLang = 0xe7, + }; + // PPA2 Flags + enum class PPA2Flags : uint8_t { + CompileForBinaryFloatingPoint = 0x80, + CompiledWithXPLink = 0x01, + CompiledUnitASCII = 0x04, + HasServiceInfo = 0x20, + }; + + PPA2MemberSubId MemberSubId = PPA2MemberSubId::LLVMBasedLang; + if (auto *MD = M.getModuleFlag("zos_cu_language")) { + StringRef Language = cast<MDString>(MD)->getString(); + MemberSubId = StringSwitch<PPA2MemberSubId>(Language) + .Case("C", PPA2MemberSubId::C) + .Case("CXX", PPA2MemberSubId::CXX) + .Case("Swift", PPA2MemberSubId::Swift) + .Case("Go", PPA2MemberSubId::Go) + .Default(PPA2MemberSubId::LLVMBasedLang); + } + + // Emit PPA2 section. + OutStreamer->emitLabel(PPA2Sym); + OutStreamer->emitInt8(static_cast<uint8_t>(PPA2MemberId::LE_C_Runtime)); + OutStreamer->emitInt8(static_cast<uint8_t>(MemberSubId)); + OutStreamer->emitInt8(0x22); // Member defined, c370_plist+c370_env + OutStreamer->emitInt8(0x04); // Control level 4 (XPLink) + OutStreamer->emitAbsoluteSymbolDiff(CELQSTRT, PPA2Sym, 4); + OutStreamer->emitInt32(0x00000000); + OutStreamer->emitAbsoluteSymbolDiff(DateVersionSym, PPA2Sym, 4); + OutStreamer->emitInt32( + 0x00000000); // Offset to main entry point, always 0 (so says TR). + uint8_t Flgs = static_cast<uint8_t>(PPA2Flags::CompileForBinaryFloatingPoint); + Flgs |= static_cast<uint8_t>(PPA2Flags::CompiledWithXPLink); + + if (auto *MD = M.getModuleFlag("zos_le_char_mode")) { + const StringRef &CharMode = cast<MDString>(MD)->getString(); + if (CharMode == "ascii") { + Flgs |= static_cast<uint8_t>( + PPA2Flags::CompiledUnitASCII); // Setting bit for ASCII char. mode. + } else if (CharMode != "ebcdic") { + report_fatal_error( + "Only ascii or ebcdic are valid values for zos_le_char_mode " + "metadata"); + } + } + + OutStreamer->emitInt8(Flgs); + OutStreamer->emitInt8(0x00); // Reserved. + // No MD5 signature before timestamp. + // No FLOAT(AFP(VOLATILE)). + // Remaining 5 flag bits reserved. + OutStreamer->emitInt16(0x0000); // 16 Reserved flag bits. + + // Emit date and version section. + OutStreamer->emitLabel(DateVersionSym); + OutStreamer->emitBytes(CompilationTimeStr.str()); + OutStreamer->emitBytes(VersionStr.str()); + + OutStreamer->emitInt16(0x0000); // Service level string length. + + // Emit 8 byte alignment. + // Emit pointer to PPA2 label. + OutStreamer->AddComment("A(PPA2-CELQSTRT)"); + OutStreamer->emitAbsoluteSymbolDiff(PPA2Sym, CELQSTRT, 8); + OutStreamer->popSection(); +} + void SystemZAsmPrinter::emitFunctionEntryLabel() { const SystemZSubtarget &Subtarget = MF->getSubtarget<SystemZSubtarget>(); @@ -1318,7 +1513,7 @@ void SystemZAsmPrinter::emitFunctionEntryLabel() { uint32_t DSASize = MFFrame.getStackSize(); bool IsLeaf = DSASize == 0 && MFFrame.getCalleeSavedInfo().empty(); - // Set Flags + // Set Flags. uint8_t Flags = 0; if (IsLeaf) Flags |= 0x08; diff --git a/llvm/lib/Target/SystemZ/SystemZAsmPrinter.h b/llvm/lib/Target/SystemZ/SystemZAsmPrinter.h index c9dbbfd0b4c433..303cce1a1b6581 100644 --- a/llvm/lib/Target/SystemZ/SystemZAsmPrinter.h +++ b/llvm/lib/Target/SystemZ/SystemZAsmPrinter.h @@ -27,6 +27,7 @@ class LLVM_LIBRARY_VISIBILITY SystemZAsmPrinter : public AsmPrinter { private: MCSymbol *CurrentFnPPA1Sym; // PPA1 Symbol. MCSymbol *CurrentFnEPMarkerSym; // Entry Point Marker. + MCSymbol *PPA2Sym; SystemZTargetStreamer *getTargetStreamer() { MCTargetStreamer *TS = OutStreamer->getTargetStreamer(); @@ -90,12 +91,15 @@ class LLVM_LIBRARY_VISIBILITY SystemZAsmPrinter : public AsmPrinter { AssociatedDataAreaTable ADATable; void emitPPA1(MCSymbol *FnEndSym); + void emitPPA2(Module &M); void emitADASection(); + void emitIDRLSection(Module &M); public: SystemZAsmPrinter(TargetMachine &TM, std::unique_ptr<MCStreamer> Streamer) : AsmPrinter(TM, std::move(Streamer)), CurrentFnPPA1Sym(nullptr), - CurrentFnEPMarkerSym(nullptr), ADATable(TM.getPointerSize(0)) {} + CurrentFnEPMarkerSym(nullptr), PPA2Sym(nullptr), + ADATable(TM.getPointerSize(0)) {} // Override AsmPrinter. StringRef getPassName() const override { return "SystemZ Assembly Printer"; } @@ -113,6 +117,7 @@ class LLVM_LIBRARY_VISIBILITY SystemZAsmPrinter : public AsmPrinter { } void emitFunctionEntryLabel() override; void emitFunctionBodyEnd() override; + void emitStartOfAsmFile(Module &M) override; private: void emitCallInformation(CallType CT); diff --git a/llvm/test/CodeGen/SystemZ/zos-ppa2.ll b/llvm/test/CodeGen/SystemZ/zos-ppa2.ll new file mode 100644 index 00000000000000..c06134040f4682 --- /dev/null +++ b/llvm/test/CodeGen/SystemZ/zos-ppa2.ll @@ -0,0 +1,27 @@ +; RUN: llc -mtriple s390x-ibm-zos -mcpu=z15 -asm-verbose=true < %s | FileCheck %s +; REQUIRES: systemz-registered-target + +; CHECK: .section ".ppa2" +; CHECK: @@PPA2: +; CHECK: .byte 3 +; CHECK: .byte 231 +; CHECK: .byte 34 +; CHECK: .byte 4 +; CHECK: .long CELQSTRT-@@PPA2 +; CHECK: .long 0 +; CHECK: .long @@DVS-@@PPA2 +; CHECK: .long 0 +; CHECK: .byte 129 +; CHECK: .byte 0 +; CHECK: .short 0 +; CHECK: @@DVS: +; CHECK: .ascii "\361\371\367\360\360\361\360\361\360\360\360\360\360\360" +; CHECK: .ascii "\361\367\360\360\360\360" +; CHECK: .short 0 +; CHECK: .quad @@PPA2-CELQSTRT * A(PPA2-CELQSTRT) +; CHECK: @@PPA1_void_test_0: +; CHECK: .long @@PPA2-@@PPA1_void_test_0 * Offset to PPA2 +define void @void_test() { +entry: + ret void +} _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits