2014-01-04 07:12:02 +08:00
|
|
|
//===- lib/ReaderWriter/MachO/Atoms.h -------------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLD_READER_WRITER_MACHO_ATOMS_H
|
|
|
|
#define LLD_READER_WRITER_MACHO_ATOMS_H
|
|
|
|
|
2014-06-12 05:47:51 +08:00
|
|
|
#include "lld/Core/Simple.h"
|
2014-01-04 07:12:02 +08:00
|
|
|
|
|
|
|
namespace lld {
|
|
|
|
namespace mach_o {
|
|
|
|
class MachODefinedAtom : public SimpleDefinedAtom {
|
|
|
|
public:
|
2014-05-16 07:03:50 +08:00
|
|
|
MachODefinedAtom(const File &f, const StringRef name, Scope scope,
|
2014-08-22 06:18:30 +08:00
|
|
|
ContentType type, Merge merge, bool thumb, bool noDeadStrip,
|
2014-11-18 08:30:25 +08:00
|
|
|
const ArrayRef<uint8_t> content, Alignment align)
|
2014-05-16 07:03:50 +08:00
|
|
|
: SimpleDefinedAtom(f), _name(name), _content(content),
|
2014-11-18 08:30:25 +08:00
|
|
|
_align(align), _contentType(type), _scope(scope), _merge(merge),
|
|
|
|
_thumb(thumb), _noDeadStrip(noDeadStrip) {}
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-05-16 07:03:50 +08:00
|
|
|
// Constructor for zero-fill content
|
|
|
|
MachODefinedAtom(const File &f, const StringRef name, Scope scope,
|
2014-11-18 08:30:25 +08:00
|
|
|
uint64_t size, bool noDeadStrip, Alignment align)
|
2014-05-16 07:03:50 +08:00
|
|
|
: SimpleDefinedAtom(f), _name(name),
|
2014-11-18 08:30:25 +08:00
|
|
|
_content(ArrayRef<uint8_t>(nullptr, size)), _align(align),
|
2014-05-28 09:16:35 +08:00
|
|
|
_contentType(DefinedAtom::typeZeroFill),
|
2014-08-22 06:18:30 +08:00
|
|
|
_scope(scope), _merge(mergeNo), _thumb(false),
|
|
|
|
_noDeadStrip(noDeadStrip) {}
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-05-16 07:03:50 +08:00
|
|
|
uint64_t size() const override { return _content.size(); }
|
|
|
|
|
|
|
|
ContentType contentType() const override { return _contentType; }
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-11-18 08:30:25 +08:00
|
|
|
Alignment alignment() const override { return _align; }
|
|
|
|
|
2014-03-29 05:36:33 +08:00
|
|
|
StringRef name() const override { return _name; }
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-03-29 05:36:33 +08:00
|
|
|
Scope scope() const override { return _scope; }
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-05-28 09:16:35 +08:00
|
|
|
Merge merge() const override { return _merge; }
|
|
|
|
|
2014-05-28 07:20:52 +08:00
|
|
|
DeadStripKind deadStrip() const override {
|
|
|
|
if (_contentType == DefinedAtom::typeInitializerPtr)
|
|
|
|
return deadStripNever;
|
|
|
|
if (_contentType == DefinedAtom::typeTerminatorPtr)
|
|
|
|
return deadStripNever;
|
2014-08-22 06:18:30 +08:00
|
|
|
if (_noDeadStrip)
|
|
|
|
return deadStripNever;
|
2014-05-28 07:20:52 +08:00
|
|
|
return deadStripNormal;
|
|
|
|
}
|
|
|
|
|
2014-05-16 07:03:50 +08:00
|
|
|
ArrayRef<uint8_t> rawContent() const override {
|
2014-06-28 02:25:01 +08:00
|
|
|
// Note: Zerofill atoms have a content pointer which is null.
|
2014-05-16 07:03:50 +08:00
|
|
|
return _content;
|
|
|
|
}
|
2014-01-04 07:12:02 +08:00
|
|
|
|
2014-07-04 08:11:09 +08:00
|
|
|
bool isThumb() const { return _thumb; }
|
|
|
|
|
2014-08-22 06:18:30 +08:00
|
|
|
void addReference(uint32_t offsetInAtom, uint16_t relocType,
|
|
|
|
const Atom *target, Reference::Addend addend,
|
|
|
|
Reference::KindArch arch = Reference::KindArch::x86_64,
|
|
|
|
Reference::KindNamespace ns
|
|
|
|
= Reference::KindNamespace::mach_o) {
|
|
|
|
SimpleDefinedAtom::addReference(ns, arch, relocType, offsetInAtom, target,
|
|
|
|
addend);
|
2014-06-28 02:25:01 +08:00
|
|
|
}
|
2014-08-22 06:18:30 +08:00
|
|
|
|
2014-01-04 07:12:02 +08:00
|
|
|
private:
|
|
|
|
const StringRef _name;
|
|
|
|
const ArrayRef<uint8_t> _content;
|
2014-11-18 08:30:25 +08:00
|
|
|
const DefinedAtom::Alignment _align;
|
2014-05-16 07:03:50 +08:00
|
|
|
const ContentType _contentType;
|
2014-01-14 06:28:02 +08:00
|
|
|
const Scope _scope;
|
2014-05-28 09:16:35 +08:00
|
|
|
const Merge _merge;
|
2014-07-04 08:11:09 +08:00
|
|
|
const bool _thumb;
|
2014-08-22 06:18:30 +08:00
|
|
|
const bool _noDeadStrip;
|
2014-01-04 07:12:02 +08:00
|
|
|
};
|
2014-05-16 04:59:23 +08:00
|
|
|
|
2014-05-31 06:51:04 +08:00
|
|
|
class MachODefinedCustomSectionAtom : public MachODefinedAtom {
|
|
|
|
public:
|
2014-08-22 06:18:30 +08:00
|
|
|
MachODefinedCustomSectionAtom(const File &f, const StringRef name,
|
2014-05-31 06:51:04 +08:00
|
|
|
Scope scope, ContentType type, Merge merge,
|
2014-08-22 06:18:30 +08:00
|
|
|
bool thumb, bool noDeadStrip,
|
|
|
|
const ArrayRef<uint8_t> content,
|
2014-11-18 08:30:25 +08:00
|
|
|
StringRef sectionName, Alignment align)
|
2014-08-22 06:18:30 +08:00
|
|
|
: MachODefinedAtom(f, name, scope, type, merge, thumb, noDeadStrip,
|
2014-11-18 08:30:25 +08:00
|
|
|
content, align),
|
2014-05-31 06:51:04 +08:00
|
|
|
_sectionName(sectionName) {}
|
|
|
|
|
|
|
|
SectionChoice sectionChoice() const override {
|
|
|
|
return DefinedAtom::sectionCustomRequired;
|
|
|
|
}
|
2014-08-22 06:18:30 +08:00
|
|
|
|
2014-05-31 06:51:04 +08:00
|
|
|
StringRef customSectionName() const override {
|
|
|
|
return _sectionName;
|
|
|
|
}
|
2014-08-22 06:18:30 +08:00
|
|
|
private:
|
2014-05-31 06:51:04 +08:00
|
|
|
StringRef _sectionName;
|
|
|
|
};
|
|
|
|
|
2014-05-16 04:59:23 +08:00
|
|
|
|
|
|
|
class MachOTentativeDefAtom : public SimpleDefinedAtom {
|
|
|
|
public:
|
|
|
|
MachOTentativeDefAtom(const File &f, const StringRef name, Scope scope,
|
|
|
|
uint64_t size, DefinedAtom::Alignment align)
|
|
|
|
: SimpleDefinedAtom(f), _name(name), _scope(scope), _size(size),
|
|
|
|
_align(align) {}
|
|
|
|
|
|
|
|
uint64_t size() const override { return _size; }
|
|
|
|
|
|
|
|
Merge merge() const override { return DefinedAtom::mergeAsTentative; }
|
|
|
|
|
|
|
|
ContentType contentType() const override { return DefinedAtom::typeZeroFill; }
|
|
|
|
|
|
|
|
Alignment alignment() const override { return _align; }
|
|
|
|
|
|
|
|
StringRef name() const override { return _name; }
|
|
|
|
|
|
|
|
Scope scope() const override { return _scope; }
|
|
|
|
|
|
|
|
ArrayRef<uint8_t> rawContent() const override { return ArrayRef<uint8_t>(); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
const StringRef _name;
|
|
|
|
const Scope _scope;
|
|
|
|
const uint64_t _size;
|
|
|
|
const DefinedAtom::Alignment _align;
|
|
|
|
};
|
|
|
|
|
2014-06-30 17:11:38 +08:00
|
|
|
class MachOSharedLibraryAtom : public SharedLibraryAtom {
|
|
|
|
public:
|
|
|
|
MachOSharedLibraryAtom(const File &file, StringRef name,
|
2014-08-14 07:55:41 +08:00
|
|
|
StringRef dylibInstallName, bool weakDef)
|
2014-06-30 17:11:38 +08:00
|
|
|
: SharedLibraryAtom(), _file(file), _name(name),
|
|
|
|
_dylibInstallName(dylibInstallName) {}
|
|
|
|
virtual ~MachOSharedLibraryAtom() {}
|
|
|
|
|
|
|
|
virtual StringRef loadName() const override {
|
|
|
|
return _dylibInstallName;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool canBeNullAtRuntime() const override {
|
|
|
|
// FIXME: this may actually be changeable. For now, all symbols are strongly
|
|
|
|
// defined though.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const File& file() const override {
|
|
|
|
return _file;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual StringRef name() const override {
|
|
|
|
return _name;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual Type type() const override {
|
|
|
|
// Unused in MachO (I think).
|
|
|
|
return Type::Unknown;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uint64_t size() const override {
|
|
|
|
// Unused in MachO (I think)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const File &_file;
|
|
|
|
StringRef _name;
|
|
|
|
StringRef _dylibInstallName;
|
|
|
|
};
|
|
|
|
|
2014-05-16 04:59:23 +08:00
|
|
|
|
2014-01-04 07:12:02 +08:00
|
|
|
} // mach_o
|
|
|
|
} // lld
|
|
|
|
|
|
|
|
#endif
|