//===- ELFAttribute.h -----------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#ifndef MCLD_TARGET_ELFATTRIBUTE_H_
#define MCLD_TARGET_ELFATTRIBUTE_H_
#include "mcld/Support/MemoryRegion.h"
#include "mcld/Target/ELFAttributeData.h"
#include <llvm/ADT/SmallVector.h>
#include <llvm/ADT/StringRef.h>
namespace mcld {
class ELFAttributeData;
class GNULDBackend;
class Input;
class LDSection;
class LinkerConfig;
/** \class ELFAttribute
* \brief ELFAttribute is the attribute section in an ELF file.
*/
class ELFAttribute {
public:
// ARM [ABI-addenda], 2.2.3.
static const char FormatVersion = 'A';
static const size_t FormatVersionFieldSize = sizeof(FormatVersion); // a byte
static const size_t SubsectionLengthFieldSize = 4; // a 4-byte integer
// MinimalELFAttributeSubsectionSize is the minimal number of bytes a valid
// subsection in ELF attribute section should have.
static const size_t MinimalELFAttributeSubsectionSize =
1 /* Tag_File, see ARM [ABI-addenda], 2.2.4 */ +
4 /* byte-size, see ARM [ABI-addenda], 2.2.4 */;
// MinimalELFAttributeSectionSize is the minimal number of bytes a valid ELF
// attribute section should have.
static const size_t MinimalELFAttributeSectionSize =
FormatVersionFieldSize + SubsectionLengthFieldSize +
2 /* vendor-name, a char plus '\0', see ARM [ABI-addenda], 2.2.3 */ +
1 * MinimalELFAttributeSubsectionSize;
public:
ELFAttribute(const GNULDBackend& pBackend, const LinkerConfig& pConfig)
: m_Backend(pBackend), m_Config(pConfig) {}
~ELFAttribute();
public:
/// merge - merge attributes from input (attribute) section
bool merge(const Input& pInput, LDSection& pInputAttrSectHdr);
/// sizeOutput - calculate the number of bytes required to encode this
/// attribute data section
size_t sizeOutput() const;
/// emit - encode and write out this attribute section
size_t emit(MemoryRegion& pRegion) const;
inline const GNULDBackend& backend() const { return m_Backend; }
inline const LinkerConfig& config() const { return m_Config; }
// Place vendor's attribute data under the management.
void registerAttributeData(ELFAttributeData& pAttrData);
private:
/** \class Subsection
* \brief A helper class to wrap ELFAttributeData and to provide general
* interfaces for ELFAttribute to operate on
*/
class Subsection {
public:
Subsection(ELFAttribute& pParent, ELFAttributeData& pAttrData)
: m_Parent(pParent), m_AttrData(pAttrData) {}
public:
bool isMyAttribute(llvm::StringRef pVendorName) const {
return (m_AttrData.getVendorName() == pVendorName);
}
/// merge - Merge the attributes from the section in the input data.
bool merge(const Input& pInput, ConstAddress pData, size_t pSize);
/// sizeOutput - calculate the number of bytes required to encode this
/// subsection
size_t sizeOutput() const;
/// emit - write out this attribute subsection to the buffer.
size_t emit(char* pBuf) const;
private:
// The attribute section this subsection belongs to
ELFAttribute& m_Parent;
// The attribute data containing in this subsection
ELFAttributeData& m_AttrData;
};
// Obtain the corresponding subsection of the specified vendor
Subsection* getSubsection(llvm::StringRef pVendorName) const;
private:
const GNULDBackend& m_Backend;
const LinkerConfig& m_Config;
// There is at most two subsections ("aeabi" and "gnu") in most cases.
llvm::SmallVector<Subsection*, 2> m_Subsections;
};
} // namespace mcld
#endif // MCLD_TARGET_ELFATTRIBUTE_H_