forked from OSchip/llvm-project
[XCOFF][AIX] Differentiate usage of label symbol and csect symbol
Summary: We are using symbols to represent label and csect interchangeably before, and that could be a problem. There are cases we would need to add storage mapping class to the symbol if that symbol is actually the name of a csect, but it's hard for us to figure out whether that symbol is a label or csect. This patch intend to do the following: 1. Construct a QualName (A name include the storage mapping class) MCSymbolXCOFF for every MCSectionXCOFF. 2. Keep a pointer to that QualName inside of MCSectionXCOFF. 3. Use that QualName whenever we need a symbol refers to that MCSectionXCOFF. 4. Adapt the snowball effect from the above changes in XCOFFObjectWriter.cpp. Reviewers: xingxue, DiggerLin, sfertile, daltenty, hubert.reinterpretcast Reviewed By: DiggerLin, daltenty Subscribers: wuzish, nemanjai, mgorny, hiraditya, kbarton, jsji, llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D69633
This commit is contained in:
parent
e25bc5e024
commit
0dc0572b48
|
@ -13,6 +13,7 @@
|
||||||
#ifndef LLVM_BINARYFORMAT_XCOFF_H
|
#ifndef LLVM_BINARYFORMAT_XCOFF_H
|
||||||
#define LLVM_BINARYFORMAT_XCOFF_H
|
#define LLVM_BINARYFORMAT_XCOFF_H
|
||||||
|
|
||||||
|
#include "llvm/ADT/StringRef.h"
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
@ -251,6 +252,8 @@ enum CFileCpuId : uint8_t {
|
||||||
TCPU_970 = 19 ///< PPC970 - PowerPC 64-bit architecture.
|
TCPU_970 = 19 ///< PPC970 - PowerPC 64-bit architecture.
|
||||||
};
|
};
|
||||||
|
|
||||||
|
StringRef getMappingClassString(XCOFF::StorageMappingClass SMC);
|
||||||
|
|
||||||
} // end namespace XCOFF
|
} // end namespace XCOFF
|
||||||
} // end namespace llvm
|
} // end namespace llvm
|
||||||
|
|
||||||
|
|
|
@ -454,6 +454,9 @@ public:
|
||||||
unsigned Encoding,
|
unsigned Encoding,
|
||||||
MCStreamer &Streamer) const;
|
MCStreamer &Streamer) const;
|
||||||
|
|
||||||
|
/// Return true if C is an acceptable character inside a symbol name.
|
||||||
|
virtual bool isAcceptableChar(char C) const;
|
||||||
|
|
||||||
/// Return true if the identifier \p Name does not need quotes to be
|
/// Return true if the identifier \p Name does not need quotes to be
|
||||||
/// syntactically correct.
|
/// syntactically correct.
|
||||||
virtual bool isValidUnquotedName(StringRef Name) const;
|
virtual bool isValidUnquotedName(StringRef Name) const;
|
||||||
|
|
|
@ -20,9 +20,9 @@ protected:
|
||||||
MCAsmInfoXCOFF();
|
MCAsmInfoXCOFF();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Return true only when the identifier Name does not need quotes to be
|
// Return true only when C is an acceptable character inside a
|
||||||
// syntactically correct for XCOFF.
|
// MCSymbolXCOFF.
|
||||||
bool isValidUnquotedName(StringRef Name) const override;
|
bool isAcceptableChar(char C) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // end namespace llvm
|
} // end namespace llvm
|
||||||
|
|
|
@ -16,11 +16,10 @@
|
||||||
#include "llvm/ADT/Twine.h"
|
#include "llvm/ADT/Twine.h"
|
||||||
#include "llvm/BinaryFormat/XCOFF.h"
|
#include "llvm/BinaryFormat/XCOFF.h"
|
||||||
#include "llvm/MC/MCSection.h"
|
#include "llvm/MC/MCSection.h"
|
||||||
|
#include "llvm/MC/MCSymbolXCOFF.h"
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
|
|
||||||
class MCSymbol;
|
|
||||||
|
|
||||||
// This class represents an XCOFF `Control Section`, more commonly referred to
|
// This class represents an XCOFF `Control Section`, more commonly referred to
|
||||||
// as a csect. A csect represents the smallest possible unit of data/code which
|
// as a csect. A csect represents the smallest possible unit of data/code which
|
||||||
// will be relocated as a single block. A csect can either be:
|
// will be relocated as a single block. A csect can either be:
|
||||||
|
@ -38,14 +37,18 @@ class MCSectionXCOFF final : public MCSection {
|
||||||
XCOFF::StorageMappingClass MappingClass;
|
XCOFF::StorageMappingClass MappingClass;
|
||||||
XCOFF::SymbolType Type;
|
XCOFF::SymbolType Type;
|
||||||
XCOFF::StorageClass StorageClass;
|
XCOFF::StorageClass StorageClass;
|
||||||
|
MCSymbolXCOFF *const QualName;
|
||||||
|
|
||||||
MCSectionXCOFF(StringRef Section, XCOFF::StorageMappingClass SMC,
|
MCSectionXCOFF(StringRef Section, XCOFF::StorageMappingClass SMC,
|
||||||
XCOFF::SymbolType ST, XCOFF::StorageClass SC, SectionKind K,
|
XCOFF::SymbolType ST, XCOFF::StorageClass SC, SectionKind K,
|
||||||
MCSymbol *Begin)
|
MCSymbolXCOFF *QualName, MCSymbol *Begin)
|
||||||
: MCSection(SV_XCOFF, K, Begin), Name(Section), MappingClass(SMC),
|
: MCSection(SV_XCOFF, K, Begin), Name(Section), MappingClass(SMC),
|
||||||
Type(ST), StorageClass(SC) {
|
Type(ST), StorageClass(SC), QualName(QualName) {
|
||||||
assert((ST == XCOFF::XTY_SD || ST == XCOFF::XTY_CM) &&
|
assert((ST == XCOFF::XTY_SD || ST == XCOFF::XTY_CM) &&
|
||||||
"Invalid or unhandled type for csect.");
|
"Invalid or unhandled type for csect.");
|
||||||
|
assert(QualName != nullptr && "QualName is needed.");
|
||||||
|
QualName->setStorageClass(SC);
|
||||||
|
QualName->setContainingCsect(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -59,6 +62,7 @@ public:
|
||||||
XCOFF::StorageMappingClass getMappingClass() const { return MappingClass; }
|
XCOFF::StorageMappingClass getMappingClass() const { return MappingClass; }
|
||||||
XCOFF::StorageClass getStorageClass() const { return StorageClass; }
|
XCOFF::StorageClass getStorageClass() const { return StorageClass; }
|
||||||
XCOFF::SymbolType getCSectType() const { return Type; }
|
XCOFF::SymbolType getCSectType() const { return Type; }
|
||||||
|
MCSymbolXCOFF *getQualNameSymbol() const { return QualName; }
|
||||||
|
|
||||||
void PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
void PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
||||||
raw_ostream &OS,
|
raw_ostream &OS,
|
||||||
|
|
|
@ -546,11 +546,13 @@ public:
|
||||||
|
|
||||||
/// Emits an lcomm directive with XCOFF csect information.
|
/// Emits an lcomm directive with XCOFF csect information.
|
||||||
///
|
///
|
||||||
/// \param Symbol - The symbol we are emiting.
|
/// \param LabelSym - Label on the block of storage.
|
||||||
/// \param Size - The size of the block of storage.
|
/// \param Size - The size of the block of storage.
|
||||||
/// \param ByteAlignment - The alignment of the symbol in bytes. Must be a power
|
/// \param CsectSym - Csect name for the block of storage.
|
||||||
/// of 2.
|
/// \param ByteAlignment - The alignment of the symbol in bytes. Must be a
|
||||||
virtual void EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
/// power of 2.
|
||||||
|
virtual void EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym, uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlignment);
|
unsigned ByteAlignment);
|
||||||
|
|
||||||
/// Emit an ELF .size directive.
|
/// Emit an ELF .size directive.
|
||||||
|
|
|
@ -26,7 +26,8 @@ public:
|
||||||
uint64_t Size = 0, unsigned ByteAlignment = 0,
|
uint64_t Size = 0, unsigned ByteAlignment = 0,
|
||||||
SMLoc Loc = SMLoc()) override;
|
SMLoc Loc = SMLoc()) override;
|
||||||
void EmitInstToData(const MCInst &Inst, const MCSubtargetInfo &) override;
|
void EmitInstToData(const MCInst &Inst, const MCSubtargetInfo &) override;
|
||||||
void EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
void EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym, uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlign) override;
|
unsigned ByteAlign) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -8,6 +8,7 @@ add_llvm_library(LLVMBinaryFormat
|
||||||
MsgPackReader.cpp
|
MsgPackReader.cpp
|
||||||
MsgPackWriter.cpp
|
MsgPackWriter.cpp
|
||||||
Wasm.cpp
|
Wasm.cpp
|
||||||
|
XCOFF.cpp
|
||||||
|
|
||||||
ADDITIONAL_HEADER_DIRS
|
ADDITIONAL_HEADER_DIRS
|
||||||
${LLVM_MAIN_INCLUDE_DIR}/llvm/BinaryFormat
|
${LLVM_MAIN_INCLUDE_DIR}/llvm/BinaryFormat
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
//===-- llvm/BinaryFormat/XCOFF.cpp - The XCOFF file format -----*- 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/BinaryFormat/XCOFF.h"
|
||||||
|
|
||||||
|
using namespace llvm;
|
||||||
|
|
||||||
|
StringRef XCOFF::getMappingClassString(XCOFF::StorageMappingClass SMC) {
|
||||||
|
switch (SMC) {
|
||||||
|
case XCOFF::XMC_DS:
|
||||||
|
return "DS";
|
||||||
|
case XCOFF::XMC_RW:
|
||||||
|
return "RW";
|
||||||
|
case XCOFF::XMC_PR:
|
||||||
|
return "PR";
|
||||||
|
case XCOFF::XMC_TC0:
|
||||||
|
return "TC0";
|
||||||
|
case XCOFF::XMC_BS:
|
||||||
|
return "BS";
|
||||||
|
default:
|
||||||
|
report_fatal_error("Unhandled storage-mapping class.");
|
||||||
|
}
|
||||||
|
}
|
|
@ -100,7 +100,7 @@ MCAsmInfo::getExprForFDESymbol(const MCSymbol *Sym,
|
||||||
return MCBinaryExpr::createSub(Res, PC, Context);
|
return MCBinaryExpr::createSub(Res, PC, Context);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool isAcceptableChar(char C) {
|
bool MCAsmInfo::isAcceptableChar(char C) const {
|
||||||
return (C >= 'a' && C <= 'z') || (C >= 'A' && C <= 'Z') ||
|
return (C >= 'a' && C <= 'z') || (C >= 'A' && C <= 'Z') ||
|
||||||
(C >= '0' && C <= '9') || C == '_' || C == '$' || C == '.' || C == '@';
|
(C >= '0' && C <= '9') || C == '_' || C == '$' || C == '.' || C == '@';
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,10 +26,11 @@ MCAsmInfoXCOFF::MCAsmInfoXCOFF() {
|
||||||
SupportsQuotedNames = false;
|
SupportsQuotedNames = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MCAsmInfoXCOFF::isValidUnquotedName(StringRef Name) const {
|
bool MCAsmInfoXCOFF::isAcceptableChar(char C) const {
|
||||||
// FIXME: Remove this function when we stop using "TOC[TC0]" as a symbol name.
|
// QualName is allowed for a MCSymbolXCOFF, and
|
||||||
if (Name.equals("TOC[TC0]"))
|
// QualName contains '[' and ']'.
|
||||||
|
if (C == '[' || C == ']')
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
return MCAsmInfo::isValidUnquotedName(Name);
|
return MCAsmInfo::isAcceptableChar(C);
|
||||||
}
|
}
|
||||||
|
|
|
@ -164,7 +164,8 @@ public:
|
||||||
void EmitCOFFSectionIndex(MCSymbol const *Symbol) override;
|
void EmitCOFFSectionIndex(MCSymbol const *Symbol) override;
|
||||||
void EmitCOFFSecRel32(MCSymbol const *Symbol, uint64_t Offset) override;
|
void EmitCOFFSecRel32(MCSymbol const *Symbol, uint64_t Offset) override;
|
||||||
void EmitCOFFImgRel32(MCSymbol const *Symbol, int64_t Offset) override;
|
void EmitCOFFImgRel32(MCSymbol const *Symbol, int64_t Offset) override;
|
||||||
void EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
void EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym, uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlign) override;
|
unsigned ByteAlign) override;
|
||||||
void emitELFSize(MCSymbol *Symbol, const MCExpr *Value) override;
|
void emitELFSize(MCSymbol *Symbol, const MCExpr *Value) override;
|
||||||
void EmitCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
void EmitCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
||||||
|
@ -765,16 +766,18 @@ void MCAsmStreamer::EmitCOFFImgRel32(MCSymbol const *Symbol, int64_t Offset) {
|
||||||
// We need an XCOFF-specific version of this directive as the AIX syntax
|
// We need an XCOFF-specific version of this directive as the AIX syntax
|
||||||
// requires a QualName argument identifying the csect name and storage mapping
|
// requires a QualName argument identifying the csect name and storage mapping
|
||||||
// class to appear before the alignment if we are specifying it.
|
// class to appear before the alignment if we are specifying it.
|
||||||
void MCAsmStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
void MCAsmStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym,
|
||||||
|
uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlignment) {
|
unsigned ByteAlignment) {
|
||||||
assert(MAI->getLCOMMDirectiveAlignmentType() == LCOMM::Log2Alignment &&
|
assert(MAI->getLCOMMDirectiveAlignmentType() == LCOMM::Log2Alignment &&
|
||||||
"We only support writing log base-2 alignment format with XCOFF.");
|
"We only support writing log base-2 alignment format with XCOFF.");
|
||||||
assert(isPowerOf2_32(ByteAlignment) && "Alignment must be a power of 2.");
|
assert(isPowerOf2_32(ByteAlignment) && "Alignment must be a power of 2.");
|
||||||
|
|
||||||
OS << "\t.lcomm\t";
|
OS << "\t.lcomm\t";
|
||||||
Symbol->print(OS, MAI);
|
LabelSym->print(OS, MAI);
|
||||||
OS << ',' << Size;
|
OS << ',' << Size << ',';
|
||||||
OS << ',' << Symbol->getName();
|
CsectSym->print(OS, MAI);
|
||||||
OS << ',' << Log2_32(ByteAlignment);
|
OS << ',' << Log2_32(ByteAlignment);
|
||||||
|
|
||||||
EmitEOL();
|
EmitEOL();
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
#include "llvm/ADT/Twine.h"
|
#include "llvm/ADT/Twine.h"
|
||||||
#include "llvm/BinaryFormat/COFF.h"
|
#include "llvm/BinaryFormat/COFF.h"
|
||||||
#include "llvm/BinaryFormat/ELF.h"
|
#include "llvm/BinaryFormat/ELF.h"
|
||||||
|
#include "llvm/BinaryFormat/XCOFF.h"
|
||||||
#include "llvm/MC/MCAsmInfo.h"
|
#include "llvm/MC/MCAsmInfo.h"
|
||||||
#include "llvm/MC/MCCodeView.h"
|
#include "llvm/MC/MCCodeView.h"
|
||||||
#include "llvm/MC/MCDwarf.h"
|
#include "llvm/MC/MCDwarf.h"
|
||||||
|
@ -550,13 +551,15 @@ MCSectionXCOFF *MCContext::getXCOFFSection(StringRef Section,
|
||||||
|
|
||||||
// Otherwise, return a new section.
|
// Otherwise, return a new section.
|
||||||
StringRef CachedName = Entry.first.SectionName;
|
StringRef CachedName = Entry.first.SectionName;
|
||||||
|
MCSymbol *QualName = getOrCreateSymbol(
|
||||||
|
CachedName + "[" + XCOFF::getMappingClassString(SMC) + "]");
|
||||||
|
|
||||||
MCSymbol *Begin = nullptr;
|
MCSymbol *Begin = nullptr;
|
||||||
if (BeginSymName)
|
if (BeginSymName)
|
||||||
Begin = createTempSymbol(BeginSymName, false);
|
Begin = createTempSymbol(BeginSymName, false);
|
||||||
|
|
||||||
MCSectionXCOFF *Result = new (XCOFFAllocator.Allocate())
|
MCSectionXCOFF *Result = new (XCOFFAllocator.Allocate()) MCSectionXCOFF(
|
||||||
MCSectionXCOFF(CachedName, SMC, Type, SC, Kind, Begin);
|
CachedName, SMC, Type, SC, Kind, cast<MCSymbolXCOFF>(QualName), Begin);
|
||||||
Entry.second = Result;
|
Entry.second = Result;
|
||||||
|
|
||||||
auto *F = new MCDataFragment();
|
auto *F = new MCDataFragment();
|
||||||
|
|
|
@ -15,18 +15,6 @@ using namespace llvm;
|
||||||
|
|
||||||
MCSectionXCOFF::~MCSectionXCOFF() = default;
|
MCSectionXCOFF::~MCSectionXCOFF() = default;
|
||||||
|
|
||||||
static StringRef getMappingClassString(XCOFF::StorageMappingClass SMC) {
|
|
||||||
switch (SMC) {
|
|
||||||
case XCOFF::XMC_DS:
|
|
||||||
return "DS";
|
|
||||||
case XCOFF::XMC_RW:
|
|
||||||
return "RW";
|
|
||||||
case XCOFF::XMC_PR:
|
|
||||||
return "PR";
|
|
||||||
default:
|
|
||||||
report_fatal_error("Unhandled storage-mapping class.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void MCSectionXCOFF::PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
void MCSectionXCOFF::PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
||||||
raw_ostream &OS,
|
raw_ostream &OS,
|
||||||
|
@ -35,9 +23,7 @@ void MCSectionXCOFF::PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
||||||
if (getMappingClass() != XCOFF::XMC_PR)
|
if (getMappingClass() != XCOFF::XMC_PR)
|
||||||
report_fatal_error("Unhandled storage-mapping class for .text csect");
|
report_fatal_error("Unhandled storage-mapping class for .text csect");
|
||||||
|
|
||||||
OS << "\t.csect " << getSectionName() << "["
|
OS << "\t.csect " << QualName->getName() << '\n';
|
||||||
<< getMappingClassString(getMappingClass())
|
|
||||||
<< "]" << '\n';
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -45,8 +31,7 @@ void MCSectionXCOFF::PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T,
|
||||||
switch (getMappingClass()) {
|
switch (getMappingClass()) {
|
||||||
case XCOFF::XMC_RW:
|
case XCOFF::XMC_RW:
|
||||||
case XCOFF::XMC_DS:
|
case XCOFF::XMC_DS:
|
||||||
OS << "\t.csect " << getSectionName() << "["
|
OS << "\t.csect " << QualName->getName() << '\n';
|
||||||
<< getMappingClassString(getMappingClass()) << "]" << '\n';
|
|
||||||
break;
|
break;
|
||||||
case XCOFF::XMC_TC0:
|
case XCOFF::XMC_TC0:
|
||||||
OS << "\t.toc\n";
|
OS << "\t.toc\n";
|
||||||
|
|
|
@ -1063,7 +1063,8 @@ void MCStreamer::EmitCOFFSymbolStorageClass(int StorageClass) {
|
||||||
void MCStreamer::EmitCOFFSymbolType(int Type) {
|
void MCStreamer::EmitCOFFSymbolType(int Type) {
|
||||||
llvm_unreachable("this directive only supported on COFF targets");
|
llvm_unreachable("this directive only supported on COFF targets");
|
||||||
}
|
}
|
||||||
void MCStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size,
|
void MCStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym, uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlign) {
|
unsigned ByteAlign) {
|
||||||
llvm_unreachable("this directive only supported on XCOFF targets");
|
llvm_unreachable("this directive only supported on XCOFF targets");
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,8 +94,9 @@ MCStreamer *llvm::createXCOFFStreamer(MCContext &Context,
|
||||||
return S;
|
return S;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MCXCOFFStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *Symbol,
|
void MCXCOFFStreamer::EmitXCOFFLocalCommonSymbol(MCSymbol *LabelSym,
|
||||||
uint64_t Size,
|
uint64_t Size,
|
||||||
|
MCSymbol *CsectSym,
|
||||||
unsigned ByteAlignment) {
|
unsigned ByteAlignment) {
|
||||||
EmitCommonSymbol(Symbol, Size, ByteAlignment);
|
EmitCommonSymbol(CsectSym, Size, ByteAlignment);
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,15 +78,7 @@ struct ControlSection {
|
||||||
// (approximately) the same storage mapping class. For example all the csects
|
// (approximately) the same storage mapping class. For example all the csects
|
||||||
// with a storage mapping class of `xmc_pr` will get placed into the same
|
// with a storage mapping class of `xmc_pr` will get placed into the same
|
||||||
// container.
|
// container.
|
||||||
struct CsectGroup {
|
using CsectGroup = std::deque<ControlSection>;
|
||||||
enum LabelDefinitionSupport : bool {
|
|
||||||
LabelDefSupported = true,
|
|
||||||
LabelDefUnsupported = false
|
|
||||||
};
|
|
||||||
|
|
||||||
const LabelDefinitionSupport SupportLabelDef;
|
|
||||||
std::deque<ControlSection> Csects;
|
|
||||||
};
|
|
||||||
|
|
||||||
using CsectGroups = std::deque<CsectGroup *>;
|
using CsectGroups = std::deque<CsectGroup *>;
|
||||||
|
|
||||||
|
@ -132,7 +124,7 @@ struct Section {
|
||||||
Index = UninitializedIndex;
|
Index = UninitializedIndex;
|
||||||
// Clear any csects we have stored.
|
// Clear any csects we have stored.
|
||||||
for (auto *Group : Groups)
|
for (auto *Group : Groups)
|
||||||
Group->Csects.clear();
|
Group->clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
Section(const char *N, XCOFF::SectionTypeFlags Flags, bool IsVirtual,
|
Section(const char *N, XCOFF::SectionTypeFlags Flags, bool IsVirtual,
|
||||||
|
@ -157,9 +149,9 @@ class XCOFFObjectWriter : public MCObjectWriter {
|
||||||
// CsectGroups. These store the csects which make up different parts of
|
// CsectGroups. These store the csects which make up different parts of
|
||||||
// the sections. Should have one for each set of csects that get mapped into
|
// the sections. Should have one for each set of csects that get mapped into
|
||||||
// the same section and get handled in a 'similar' way.
|
// the same section and get handled in a 'similar' way.
|
||||||
CsectGroup ProgramCodeCsects{CsectGroup::LabelDefSupported, {}};
|
CsectGroup ProgramCodeCsects;
|
||||||
CsectGroup DataCsects{CsectGroup::LabelDefSupported, {}};
|
CsectGroup DataCsects;
|
||||||
CsectGroup BSSCsects{CsectGroup::LabelDefUnsupported, {}};
|
CsectGroup BSSCsects;
|
||||||
|
|
||||||
// The Predefined sections.
|
// The Predefined sections.
|
||||||
Section Text;
|
Section Text;
|
||||||
|
@ -294,8 +286,8 @@ void XCOFFObjectWriter::executePostLayoutBinding(MCAssembler &Asm,
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
CsectGroup &Group = getCsectGroup(MCSec);
|
CsectGroup &Group = getCsectGroup(MCSec);
|
||||||
Group.Csects.emplace_back(MCSec);
|
Group.emplace_back(MCSec);
|
||||||
WrapperMap[MCSec] = &Group.Csects.back();
|
WrapperMap[MCSec] = &Group.back();
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const MCSymbol &S : Asm.symbols()) {
|
for (const MCSymbol &S : Asm.symbols()) {
|
||||||
|
@ -309,6 +301,11 @@ void XCOFFObjectWriter::executePostLayoutBinding(MCAssembler &Asm,
|
||||||
assert(WrapperMap.find(ContainingCsect) != WrapperMap.end() &&
|
assert(WrapperMap.find(ContainingCsect) != WrapperMap.end() &&
|
||||||
"Expected containing csect to exist in map");
|
"Expected containing csect to exist in map");
|
||||||
|
|
||||||
|
// If the symbol is the Csect itself, we don't need to put the symbol
|
||||||
|
// into Csect's Syms.
|
||||||
|
if (XSym == ContainingCsect->getQualNameSymbol())
|
||||||
|
continue;
|
||||||
|
|
||||||
// Lookup the containing csect and add the symbol to it.
|
// Lookup the containing csect and add the symbol to it.
|
||||||
WrapperMap[ContainingCsect]->Syms.emplace_back(XSym);
|
WrapperMap[ContainingCsect]->Syms.emplace_back(XSym);
|
||||||
|
|
||||||
|
@ -339,7 +336,7 @@ void XCOFFObjectWriter::writeSections(const MCAssembler &Asm,
|
||||||
assert(CurrentAddressLocation == Section->Address &&
|
assert(CurrentAddressLocation == Section->Address &&
|
||||||
"We should have no padding between sections.");
|
"We should have no padding between sections.");
|
||||||
for (const auto *Group : Section->Groups) {
|
for (const auto *Group : Section->Groups) {
|
||||||
for (const auto &Csect : Group->Csects) {
|
for (const auto &Csect : *Group) {
|
||||||
if (uint32_t PaddingSize = Csect.Address - CurrentAddressLocation)
|
if (uint32_t PaddingSize = Csect.Address - CurrentAddressLocation)
|
||||||
W.OS.write_zeros(PaddingSize);
|
W.OS.write_zeros(PaddingSize);
|
||||||
Asm.writeSectionData(W.OS, Csect.MCCsect, Layout);
|
Asm.writeSectionData(W.OS, Csect.MCCsect, Layout);
|
||||||
|
@ -527,20 +524,14 @@ void XCOFFObjectWriter::writeSymbolTable(const MCAsmLayout &Layout) {
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
for (const auto *Group : Section->Groups) {
|
for (const auto *Group : Section->Groups) {
|
||||||
if (Group->Csects.empty())
|
if (Group->empty())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
const bool SupportLabelDef = Group->SupportLabelDef;
|
|
||||||
const int16_t SectionIndex = Section->Index;
|
const int16_t SectionIndex = Section->Index;
|
||||||
for (const auto &Csect : Group->Csects) {
|
for (const auto &Csect : *Group) {
|
||||||
// Write out the control section first and then each symbol in it.
|
// Write out the control section first and then each symbol in it.
|
||||||
writeSymbolTableEntryForControlSection(
|
writeSymbolTableEntryForControlSection(
|
||||||
Csect, SectionIndex, Csect.MCCsect->getStorageClass());
|
Csect, SectionIndex, Csect.MCCsect->getStorageClass());
|
||||||
if (!SupportLabelDef) {
|
|
||||||
assert(Csect.Syms.size() == 1 && "Csect should only contain 1 symbol "
|
|
||||||
"which is its label definition.");
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const auto Sym : Csect.Syms)
|
for (const auto Sym : Csect.Syms)
|
||||||
writeSymbolTableEntryForCsectMemberLabel(
|
writeSymbolTableEntryForCsectMemberLabel(
|
||||||
|
@ -563,9 +554,8 @@ void XCOFFObjectWriter::assignAddressesAndIndices(const MCAsmLayout &Layout) {
|
||||||
|
|
||||||
for (auto *Section : Sections) {
|
for (auto *Section : Sections) {
|
||||||
const bool IsEmpty =
|
const bool IsEmpty =
|
||||||
llvm::all_of(Section->Groups, [](const CsectGroup *Group) {
|
llvm::all_of(Section->Groups,
|
||||||
return Group->Csects.empty();
|
[](const CsectGroup *Group) { return Group->empty(); });
|
||||||
});
|
|
||||||
if (IsEmpty)
|
if (IsEmpty)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -576,11 +566,10 @@ void XCOFFObjectWriter::assignAddressesAndIndices(const MCAsmLayout &Layout) {
|
||||||
|
|
||||||
bool SectionAddressSet = false;
|
bool SectionAddressSet = false;
|
||||||
for (auto *Group : Section->Groups) {
|
for (auto *Group : Section->Groups) {
|
||||||
if (Group->Csects.empty())
|
if (Group->empty())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
const bool SupportLabelDef = Group->SupportLabelDef;
|
for (auto &Csect : *Group) {
|
||||||
for (auto &Csect : Group->Csects) {
|
|
||||||
const MCSectionXCOFF *MCSec = Csect.MCCsect;
|
const MCSectionXCOFF *MCSec = Csect.MCCsect;
|
||||||
Csect.Address = alignTo(Address, MCSec->getAlignment());
|
Csect.Address = alignTo(Address, MCSec->getAlignment());
|
||||||
Csect.Size = Layout.getSectionAddressSize(MCSec);
|
Csect.Size = Layout.getSectionAddressSize(MCSec);
|
||||||
|
@ -588,10 +577,7 @@ void XCOFFObjectWriter::assignAddressesAndIndices(const MCAsmLayout &Layout) {
|
||||||
Csect.SymbolTableIndex = SymbolTableIndex;
|
Csect.SymbolTableIndex = SymbolTableIndex;
|
||||||
// 1 main and 1 auxiliary symbol table entry for the csect.
|
// 1 main and 1 auxiliary symbol table entry for the csect.
|
||||||
SymbolTableIndex += 2;
|
SymbolTableIndex += 2;
|
||||||
|
|
||||||
if (!SupportLabelDef)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
for (auto &Sym : Csect.Syms) {
|
for (auto &Sym : Csect.Syms) {
|
||||||
Sym.SymbolTableIndex = SymbolTableIndex;
|
Sym.SymbolTableIndex = SymbolTableIndex;
|
||||||
// 1 main and 1 auxiliary symbol table entry for each contained
|
// 1 main and 1 auxiliary symbol table entry for each contained
|
||||||
|
@ -601,7 +587,7 @@ void XCOFFObjectWriter::assignAddressesAndIndices(const MCAsmLayout &Layout) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!SectionAddressSet) {
|
if (!SectionAddressSet) {
|
||||||
Section->Address = Group->Csects.front().Address;
|
Section->Address = Group->front().Address;
|
||||||
SectionAddressSet = true;
|
SectionAddressSet = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1752,15 +1752,15 @@ void PPCAIXAsmPrinter::EmitGlobalVariable(const GlobalVariable *GV) {
|
||||||
"not supported yet.");
|
"not supported yet.");
|
||||||
|
|
||||||
// Create the containing csect and switch to it.
|
// Create the containing csect and switch to it.
|
||||||
MCSectionXCOFF *CSect = cast<MCSectionXCOFF>(
|
MCSectionXCOFF *Csect = cast<MCSectionXCOFF>(
|
||||||
getObjFileLowering().SectionForGlobal(GV, GVKind, TM));
|
getObjFileLowering().SectionForGlobal(GV, GVKind, TM));
|
||||||
OutStreamer->SwitchSection(CSect);
|
OutStreamer->SwitchSection(Csect);
|
||||||
|
|
||||||
// Create the symbol, set its storage class, and emit it.
|
// Create the symbol, set its storage class, and emit it.
|
||||||
MCSymbolXCOFF *GVSym = cast<MCSymbolXCOFF>(getSymbol(GV));
|
MCSymbolXCOFF *GVSym = cast<MCSymbolXCOFF>(getSymbol(GV));
|
||||||
GVSym->setStorageClass(
|
GVSym->setStorageClass(
|
||||||
TargetLoweringObjectFileXCOFF::getStorageClassForGlobal(GV));
|
TargetLoweringObjectFileXCOFF::getStorageClassForGlobal(GV));
|
||||||
GVSym->setContainingCsect(CSect);
|
GVSym->setContainingCsect(Csect);
|
||||||
|
|
||||||
const DataLayout &DL = GV->getParent()->getDataLayout();
|
const DataLayout &DL = GV->getParent()->getDataLayout();
|
||||||
|
|
||||||
|
@ -1771,9 +1771,10 @@ void PPCAIXAsmPrinter::EmitGlobalVariable(const GlobalVariable *GV) {
|
||||||
uint64_t Size = DL.getTypeAllocSize(GV->getType()->getElementType());
|
uint64_t Size = DL.getTypeAllocSize(GV->getType()->getElementType());
|
||||||
|
|
||||||
if (GVKind.isBSSLocal())
|
if (GVKind.isBSSLocal())
|
||||||
OutStreamer->EmitXCOFFLocalCommonSymbol(GVSym, Size, Align);
|
OutStreamer->EmitXCOFFLocalCommonSymbol(
|
||||||
|
GVSym, Size, Csect->getQualNameSymbol(), Align);
|
||||||
else
|
else
|
||||||
OutStreamer->EmitCommonSymbol(GVSym, Size, Align);
|
OutStreamer->EmitCommonSymbol(Csect->getQualNameSymbol(), Size, Align);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1813,11 +1814,9 @@ void PPCAIXAsmPrinter::EmitEndOfAsmFile(Module &M) {
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Emit TOC base.
|
// Emit TOC base.
|
||||||
MCSymbol *TOCBaseSym = OutContext.getOrCreateSymbol(StringRef("TOC[TC0]"));
|
|
||||||
MCSectionXCOFF *TOCBaseSection = OutStreamer->getContext().getXCOFFSection(
|
MCSectionXCOFF *TOCBaseSection = OutStreamer->getContext().getXCOFFSection(
|
||||||
StringRef("TOC"), XCOFF::XMC_TC0, XCOFF::XTY_SD, XCOFF::C_HIDEXT,
|
StringRef("TOC"), XCOFF::XMC_TC0, XCOFF::XTY_SD, XCOFF::C_HIDEXT,
|
||||||
SectionKind::getData());
|
SectionKind::getData());
|
||||||
cast<MCSymbolXCOFF>(TOCBaseSym)->setContainingCsect(TOCBaseSection);
|
|
||||||
// Switch to section to emit TOC base.
|
// Switch to section to emit TOC base.
|
||||||
OutStreamer->SwitchSection(TOCBaseSection);
|
OutStreamer->SwitchSection(TOCBaseSection);
|
||||||
}
|
}
|
||||||
|
|
|
@ -84,13 +84,13 @@
|
||||||
; CHECK-NEXT: .llong 4613937818241073152
|
; CHECK-NEXT: .llong 4613937818241073152
|
||||||
; CHECK-NEXT: .llong 4616189618054758400
|
; CHECK-NEXT: .llong 4616189618054758400
|
||||||
|
|
||||||
; CHECK-NEXT: .comm a,4,2
|
; CHECK-NEXT: .comm a[RW],4,2
|
||||||
; CHECK-NEXT: .comm b,8,3
|
; CHECK-NEXT: .comm b[RW],8,3
|
||||||
; CHECK-NEXT: .comm c,2,1
|
; CHECK-NEXT: .comm c[RW],2,1
|
||||||
; CHECK-NEXT: .comm d,8,3
|
; CHECK-NEXT: .comm d[RW],8,3
|
||||||
; CHECK-NEXT: .comm f,4,2
|
; CHECK-NEXT: .comm f[RW],4,2
|
||||||
; CHECK-NEXT: .comm over_aligned_comm,8,5
|
; CHECK-NEXT: .comm over_aligned_comm[RW],8,5
|
||||||
; CHECK-NEXT: .comm array,33,0
|
; CHECK-NEXT: .comm array[RW],33,0
|
||||||
|
|
||||||
; OBJ: File: {{.*}}aix-xcoff-data.ll.tmp.o
|
; OBJ: File: {{.*}}aix-xcoff-data.ll.tmp.o
|
||||||
; OBJ-NEXT: Format: aixcoff-rs6000
|
; OBJ-NEXT: Format: aixcoff-rs6000
|
||||||
|
|
|
@ -14,9 +14,9 @@
|
||||||
@b = internal global i64 0, align 8
|
@b = internal global i64 0, align 8
|
||||||
@c = internal global i16 0, align 2
|
@c = internal global i16 0, align 2
|
||||||
|
|
||||||
; CHECK: .lcomm a,4,a,2
|
; CHECK: .lcomm a,4,a[BS],2
|
||||||
; CHECK-NEXT: .lcomm b,8,b,3
|
; CHECK-NEXT: .lcomm b,8,b[BS],3
|
||||||
; CHECK-NEXT: .lcomm c,2,c,1
|
; CHECK-NEXT: .lcomm c,2,c[BS],1
|
||||||
|
|
||||||
; OBJ: File: {{.*}}aix-xcoff-lcomm.ll.tmp.o
|
; OBJ: File: {{.*}}aix-xcoff-lcomm.ll.tmp.o
|
||||||
; OBJ-NEXT: Format: aixcoff-rs6000
|
; OBJ-NEXT: Format: aixcoff-rs6000
|
||||||
|
|
Loading…
Reference in New Issue