friend class MCExpr;
friend class MCContext;
- typedef const StringMapEntry<bool> NameEntryTy;
- MCSymbol(SymbolKind Kind, NameEntryTy *Name, bool isTemporary)
+ /// \brief The name for a symbol.
+ /// MCSymbol contains a uint64_t so is probably aligned to 8. On a 32-bit
+ /// system, the name is a pointer so isn't going to satisfy the 8 byte
+ /// alignment of uint64_t. Account for that here.
+ typedef union {
+ const StringMapEntry<bool> *NameEntry;
+ uint64_t AlignmentPadding;
+ } NameEntryStorageTy;
+
+ MCSymbol(SymbolKind Kind, const StringMapEntry<bool> *Name, bool isTemporary)
: Value(nullptr), IsTemporary(isTemporary),
IsRedefinable(false), IsUsed(false), IsRegistered(false),
IsExternal(false), IsPrivateExtern(false), HasName(!!Name),
Kind(Kind) {
Offset = 0;
if (Name)
- getNameEntryPtr() = Name;
+ getNameEntryPtr().NameEntry = Name;
}
// Provide custom new/delete as we will only allocate space for a name
// if we need one.
- void *operator new(size_t s, NameEntryTy *Name, MCContext &Ctx);
+ void *operator new(size_t s, const StringMapEntry<bool> *Name,
+ MCContext &Ctx);
private:
}
/// \brief Get a reference to the name field. Requires that we have a name
- NameEntryTy *&getNameEntryPtr() {
+ NameEntryStorageTy &getNameEntryPtr() {
assert(HasName && "Name is required");
- NameEntryTy **Name = reinterpret_cast<NameEntryTy **>(this);
+ NameEntryStorageTy *Name = reinterpret_cast<NameEntryStorageTy *>(this);
return *(Name - 1);
}
- NameEntryTy *const &getNameEntryPtr() const {
+ const NameEntryStorageTy &getNameEntryPtr() const {
assert(HasName && "Name is required");
- NameEntryTy *const *Name = reinterpret_cast<NameEntryTy *const *>(this);
+ const auto *Name = reinterpret_cast<const NameEntryStorageTy *>(this);
return *(Name - 1);
}
if (!HasName)
return StringRef();
- return getNameEntryPtr()->first();
+ return getNameEntryPtr().NameEntry->first();
}
bool isRegistered() const { return IsRegistered; }
// Sentinel value for the absolute pseudo section.
MCSection *MCSymbol::AbsolutePseudoSection = reinterpret_cast<MCSection *>(1);
-void *MCSymbol::operator new(size_t s, NameEntryTy *Name, MCContext &Ctx) {
- size_t Size = s + (Name ? sizeof(Name) : 0);
+void *MCSymbol::operator new(size_t s, const StringMapEntry<bool> *Name,
+ MCContext &Ctx) {
+ // We may need more space for a Name to account for alignment. So allocate
+ // space for the storage type and not the name pointer.
+ size_t Size = s + (Name ? sizeof(NameEntryStorageTy) : 0);
// For safety, ensure that the alignment of a pointer is enough for an
// MCSymbol. This also ensures we don't need padding between the name and
// symbol.
- assert(alignOf<MCSymbol>() <= alignOf<NameEntryTy *>() &&
+ assert(alignOf<MCSymbol>() <= alignOf<NameEntryStorageTy>() &&
"Bad alignment of MCSymbol");
- void *Storage = Ctx.allocate(Size, alignOf<NameEntryTy *>());
- NameEntryTy **Start = static_cast<NameEntryTy**>(Storage);
- NameEntryTy **End = Start + (Name ? 1 : 0);
+ void *Storage = Ctx.allocate(Size, alignOf<NameEntryStorageTy>());
+ NameEntryStorageTy *Start = static_cast<NameEntryStorageTy*>(Storage);
+ NameEntryStorageTy *End = Start + (Name ? 1 : 0);
return End;
}