2017-04-25 07:21:38 +08:00
|
|
|
//===- ELFObjectFile.cpp - ELF object file implementation -----------------===//
|
2011-01-20 14:38:47 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2011-01-20 14:38:47 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2012-02-12 14:12:10 +08:00
|
|
|
// Part of the ELFObjectFile class implementation.
|
2011-01-20 14:38:47 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/Object/ELFObjectFile.h"
|
2017-04-25 07:21:38 +08:00
|
|
|
#include "llvm/ADT/Triple.h"
|
2017-06-07 11:48:56 +08:00
|
|
|
#include "llvm/BinaryFormat/ELF.h"
|
2018-08-24 23:21:56 +08:00
|
|
|
#include "llvm/MC/MCInstrAnalysis.h"
|
2017-04-25 07:21:38 +08:00
|
|
|
#include "llvm/MC/SubtargetFeature.h"
|
|
|
|
#include "llvm/Object/ELF.h"
|
|
|
|
#include "llvm/Object/ELFTypes.h"
|
|
|
|
#include "llvm/Object/Error.h"
|
2017-01-18 23:52:11 +08:00
|
|
|
#include "llvm/Support/ARMAttributeParser.h"
|
2017-06-06 19:49:48 +08:00
|
|
|
#include "llvm/Support/ARMBuildAttributes.h"
|
2017-04-25 07:21:38 +08:00
|
|
|
#include "llvm/Support/Endian.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2013-01-05 04:36:28 +08:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2018-08-24 23:21:56 +08:00
|
|
|
#include "llvm/Support/TargetRegistry.h"
|
2017-04-25 07:21:38 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <system_error>
|
|
|
|
#include <utility>
|
2011-01-20 14:38:47 +08:00
|
|
|
|
2017-04-25 07:21:38 +08:00
|
|
|
using namespace llvm;
|
2012-02-12 14:12:10 +08:00
|
|
|
using namespace object;
|
2011-09-09 04:52:17 +08:00
|
|
|
|
2019-03-09 06:00:50 +08:00
|
|
|
const EnumEntry<unsigned> llvm::object::ElfSymbolTypes[NumElfSymbolTypes] = {
|
|
|
|
{"None", "NOTYPE", ELF::STT_NOTYPE},
|
|
|
|
{"Object", "OBJECT", ELF::STT_OBJECT},
|
|
|
|
{"Function", "FUNC", ELF::STT_FUNC},
|
|
|
|
{"Section", "SECTION", ELF::STT_SECTION},
|
|
|
|
{"File", "FILE", ELF::STT_FILE},
|
|
|
|
{"Common", "COMMON", ELF::STT_COMMON},
|
|
|
|
{"TLS", "TLS", ELF::STT_TLS},
|
|
|
|
{"GNU_IFunc", "IFUNC", ELF::STT_GNU_IFUNC}};
|
|
|
|
|
2014-08-20 02:44:46 +08:00
|
|
|
ELFObjectFileBase::ELFObjectFileBase(unsigned int Type, MemoryBufferRef Source)
|
|
|
|
: ObjectFile(Type, Source) {}
|
2014-08-18 01:52:10 +08:00
|
|
|
|
2017-10-11 05:21:16 +08:00
|
|
|
template <class ELFT>
|
|
|
|
static Expected<std::unique_ptr<ELFObjectFile<ELFT>>>
|
|
|
|
createPtr(MemoryBufferRef Object) {
|
|
|
|
auto Ret = ELFObjectFile<ELFT>::create(Object);
|
|
|
|
if (Error E = Ret.takeError())
|
|
|
|
return std::move(E);
|
|
|
|
return make_unique<ELFObjectFile<ELFT>>(std::move(*Ret));
|
|
|
|
}
|
|
|
|
|
2017-10-11 04:00:07 +08:00
|
|
|
Expected<std::unique_ptr<ObjectFile>>
|
2014-08-20 02:44:46 +08:00
|
|
|
ObjectFile::createELFObjectFile(MemoryBufferRef Obj) {
|
2014-07-05 19:38:52 +08:00
|
|
|
std::pair<unsigned char, unsigned char> Ident =
|
2014-08-20 02:44:46 +08:00
|
|
|
getElfArchType(Obj.getBuffer());
|
2013-01-05 04:36:28 +08:00
|
|
|
std::size_t MaxAlignment =
|
2014-08-20 02:44:46 +08:00
|
|
|
1ULL << countTrailingZeros(uintptr_t(Obj.getBufferStart()));
|
2013-01-05 04:36:28 +08:00
|
|
|
|
2015-06-02 20:05:27 +08:00
|
|
|
if (MaxAlignment < 2)
|
2017-10-11 04:00:07 +08:00
|
|
|
return createError("Insufficient alignment");
|
2015-06-02 20:05:27 +08:00
|
|
|
|
|
|
|
if (Ident.first == ELF::ELFCLASS32) {
|
|
|
|
if (Ident.second == ELF::ELFDATA2LSB)
|
2017-10-11 05:21:16 +08:00
|
|
|
return createPtr<ELF32LE>(Obj);
|
2015-06-02 20:05:27 +08:00
|
|
|
else if (Ident.second == ELF::ELFDATA2MSB)
|
2017-10-11 05:21:16 +08:00
|
|
|
return createPtr<ELF32BE>(Obj);
|
2015-06-02 20:05:27 +08:00
|
|
|
else
|
2017-10-11 04:00:07 +08:00
|
|
|
return createError("Invalid ELF data");
|
2015-06-05 07:14:43 +08:00
|
|
|
} else if (Ident.first == ELF::ELFCLASS64) {
|
2015-06-02 20:05:27 +08:00
|
|
|
if (Ident.second == ELF::ELFDATA2LSB)
|
2017-10-11 05:21:16 +08:00
|
|
|
return createPtr<ELF64LE>(Obj);
|
2015-06-02 20:05:27 +08:00
|
|
|
else if (Ident.second == ELF::ELFDATA2MSB)
|
2017-10-11 05:21:16 +08:00
|
|
|
return createPtr<ELF64BE>(Obj);
|
2015-06-02 20:05:27 +08:00
|
|
|
else
|
2017-10-11 04:00:07 +08:00
|
|
|
return createError("Invalid ELF data");
|
2011-10-11 11:18:58 +08:00
|
|
|
}
|
2017-10-11 05:21:16 +08:00
|
|
|
return createError("Invalid ELF class");
|
2011-09-09 04:52:17 +08:00
|
|
|
}
|
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
SubtargetFeatures ELFObjectFileBase::getMIPSFeatures() const {
|
|
|
|
SubtargetFeatures Features;
|
2018-01-30 02:27:30 +08:00
|
|
|
unsigned PlatformFlags = getPlatformFlags();
|
2017-01-18 23:52:11 +08:00
|
|
|
|
|
|
|
switch (PlatformFlags & ELF::EF_MIPS_ARCH) {
|
|
|
|
case ELF::EF_MIPS_ARCH_1:
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_2:
|
|
|
|
Features.AddFeature("mips2");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_3:
|
|
|
|
Features.AddFeature("mips3");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_4:
|
|
|
|
Features.AddFeature("mips4");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_5:
|
|
|
|
Features.AddFeature("mips5");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_32:
|
|
|
|
Features.AddFeature("mips32");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_64:
|
|
|
|
Features.AddFeature("mips64");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_32R2:
|
|
|
|
Features.AddFeature("mips32r2");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_64R2:
|
|
|
|
Features.AddFeature("mips64r2");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_32R6:
|
|
|
|
Features.AddFeature("mips32r6");
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_ARCH_64R6:
|
|
|
|
Features.AddFeature("mips64r6");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown EF_MIPS_ARCH value");
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (PlatformFlags & ELF::EF_MIPS_MACH) {
|
|
|
|
case ELF::EF_MIPS_MACH_NONE:
|
|
|
|
// No feature associated with this value.
|
|
|
|
break;
|
|
|
|
case ELF::EF_MIPS_MACH_OCTEON:
|
|
|
|
Features.AddFeature("cnmips");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
llvm_unreachable("Unknown EF_MIPS_ARCH value");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PlatformFlags & ELF::EF_MIPS_ARCH_ASE_M16)
|
|
|
|
Features.AddFeature("mips16");
|
|
|
|
if (PlatformFlags & ELF::EF_MIPS_MICROMIPS)
|
|
|
|
Features.AddFeature("micromips");
|
|
|
|
|
|
|
|
return Features;
|
|
|
|
}
|
2016-06-16 17:17:03 +08:00
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
SubtargetFeatures ELFObjectFileBase::getARMFeatures() const {
|
|
|
|
SubtargetFeatures Features;
|
|
|
|
ARMAttributeParser Attributes;
|
2019-05-10 18:19:08 +08:00
|
|
|
if (Error E = getBuildAttributes(Attributes))
|
2017-01-18 23:52:11 +08:00
|
|
|
return SubtargetFeatures();
|
|
|
|
|
|
|
|
// both ARMv7-M and R have to support thumb hardware div
|
|
|
|
bool isV7 = false;
|
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::CPU_arch))
|
|
|
|
isV7 = Attributes.getAttributeValue(ARMBuildAttrs::CPU_arch)
|
|
|
|
== ARMBuildAttrs::v7;
|
|
|
|
|
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::CPU_arch_profile)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::CPU_arch_profile)) {
|
|
|
|
case ARMBuildAttrs::ApplicationProfile:
|
|
|
|
Features.AddFeature("aclass");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::RealTimeProfile:
|
|
|
|
Features.AddFeature("rclass");
|
|
|
|
if (isV7)
|
|
|
|
Features.AddFeature("hwdiv");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::MicroControllerProfile:
|
|
|
|
Features.AddFeature("mclass");
|
|
|
|
if (isV7)
|
|
|
|
Features.AddFeature("hwdiv");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::THUMB_ISA_use)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::THUMB_ISA_use)) {
|
|
|
|
default:
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::Not_Allowed:
|
|
|
|
Features.AddFeature("thumb", false);
|
|
|
|
Features.AddFeature("thumb2", false);
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowThumb32:
|
|
|
|
Features.AddFeature("thumb2");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::FP_arch)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::FP_arch)) {
|
|
|
|
default:
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::Not_Allowed:
|
[ARM] Replace fp-only-sp and d16 with fp64 and d32.
Those two subtarget features were awkward because their semantics are
reversed: each one indicates the _lack_ of support for something in
the architecture, rather than the presence. As a consequence, you
don't get the behavior you want if you combine two sets of feature
bits.
Each SubtargetFeature for an FP architecture version now comes in four
versions, one for each combination of those options. So you can still
say (for example) '+vfp2' in a feature string and it will mean what
it's always meant, but there's a new string '+vfp2d16sp' meaning the
version without those extra options.
A lot of this change is just mechanically replacing positive checks
for the old features with negative checks for the new ones. But one
more interesting change is that I've rearranged getFPUFeatures() so
that the main FPU feature is appended to the output list *before*
rather than after the features derived from the Restriction field, so
that -fp64 and -d32 can override defaults added by the main feature.
Reviewers: dmgreen, samparker, SjoerdMeijer
Subscribers: srhines, javed.absar, eraman, kristof.beyls, hiraditya, zzheng, Petar.Avramovic, cfe-commits, llvm-commits
Tags: #clang, #llvm
Differential Revision: https://reviews.llvm.org/D60691
llvm-svn: 361845
2019-05-29 00:13:20 +08:00
|
|
|
Features.AddFeature("vfp2d16sp", false);
|
|
|
|
Features.AddFeature("vfp3d16sp", false);
|
|
|
|
Features.AddFeature("vfp4d16sp", false);
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowFPv2:
|
|
|
|
Features.AddFeature("vfp2");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowFPv3A:
|
|
|
|
case ARMBuildAttrs::AllowFPv3B:
|
|
|
|
Features.AddFeature("vfp3");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowFPv4A:
|
|
|
|
case ARMBuildAttrs::AllowFPv4B:
|
|
|
|
Features.AddFeature("vfp4");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-01-18 23:52:11 +08:00
|
|
|
}
|
2016-06-16 17:17:03 +08:00
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::Advanced_SIMD_arch)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::Advanced_SIMD_arch)) {
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::Not_Allowed:
|
|
|
|
Features.AddFeature("neon", false);
|
|
|
|
Features.AddFeature("fp16", false);
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowNeon:
|
|
|
|
Features.AddFeature("neon");
|
2016-06-16 17:17:03 +08:00
|
|
|
break;
|
2017-01-18 23:52:11 +08:00
|
|
|
case ARMBuildAttrs::AllowNeon2:
|
|
|
|
Features.AddFeature("neon");
|
|
|
|
Features.AddFeature("fp16");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-30 20:57:04 +08:00
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::MVE_arch)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::MVE_arch)) {
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::Not_Allowed:
|
|
|
|
Features.AddFeature("mve", false);
|
|
|
|
Features.AddFeature("mve.fp", false);
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::AllowMVEInteger:
|
|
|
|
Features.AddFeature("mve.fp", false);
|
|
|
|
Features.AddFeature("mve");
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::AllowMVEIntegerAndFloat:
|
|
|
|
Features.AddFeature("mve.fp");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::DIV_use)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::DIV_use)) {
|
2016-06-16 17:17:03 +08:00
|
|
|
default:
|
2017-01-18 23:52:11 +08:00
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::DisallowDIV:
|
|
|
|
Features.AddFeature("hwdiv", false);
|
|
|
|
Features.AddFeature("hwdiv-arm", false);
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::AllowDIVExt:
|
|
|
|
Features.AddFeature("hwdiv");
|
|
|
|
Features.AddFeature("hwdiv-arm");
|
|
|
|
break;
|
2016-06-16 17:17:03 +08:00
|
|
|
}
|
2017-01-18 23:52:11 +08:00
|
|
|
}
|
2016-06-16 17:17:03 +08:00
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
return Features;
|
|
|
|
}
|
2016-06-16 17:17:03 +08:00
|
|
|
|
2018-02-02 14:01:02 +08:00
|
|
|
SubtargetFeatures ELFObjectFileBase::getRISCVFeatures() const {
|
|
|
|
SubtargetFeatures Features;
|
|
|
|
unsigned PlatformFlags = getPlatformFlags();
|
|
|
|
|
|
|
|
if (PlatformFlags & ELF::EF_RISCV_RVC) {
|
|
|
|
Features.AddFeature("c");
|
|
|
|
}
|
|
|
|
|
|
|
|
return Features;
|
|
|
|
}
|
|
|
|
|
2017-01-18 23:52:11 +08:00
|
|
|
SubtargetFeatures ELFObjectFileBase::getFeatures() const {
|
|
|
|
switch (getEMachine()) {
|
|
|
|
case ELF::EM_MIPS:
|
|
|
|
return getMIPSFeatures();
|
|
|
|
case ELF::EM_ARM:
|
|
|
|
return getARMFeatures();
|
2018-02-02 14:01:02 +08:00
|
|
|
case ELF::EM_RISCV:
|
|
|
|
return getRISCVFeatures();
|
2016-06-16 17:17:03 +08:00
|
|
|
default:
|
|
|
|
return SubtargetFeatures();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-18 21:52:12 +08:00
|
|
|
// FIXME Encode from a tablegen description or target parser.
|
|
|
|
void ELFObjectFileBase::setARMSubArch(Triple &TheTriple) const {
|
|
|
|
if (TheTriple.getSubArch() != Triple::NoSubArch)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ARMAttributeParser Attributes;
|
2019-05-10 18:19:08 +08:00
|
|
|
if (Error E = getBuildAttributes(Attributes))
|
2017-01-18 21:52:12 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
std::string Triple;
|
|
|
|
// Default to ARM, but use the triple if it's been set.
|
2017-08-13 01:40:18 +08:00
|
|
|
if (TheTriple.isThumb())
|
2017-01-18 21:52:12 +08:00
|
|
|
Triple = "thumb";
|
|
|
|
else
|
|
|
|
Triple = "arm";
|
|
|
|
|
|
|
|
if (Attributes.hasAttribute(ARMBuildAttrs::CPU_arch)) {
|
|
|
|
switch(Attributes.getAttributeValue(ARMBuildAttrs::CPU_arch)) {
|
|
|
|
case ARMBuildAttrs::v4:
|
|
|
|
Triple += "v4";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v4T:
|
|
|
|
Triple += "v4t";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v5T:
|
|
|
|
Triple += "v5t";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v5TE:
|
|
|
|
Triple += "v5te";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v5TEJ:
|
|
|
|
Triple += "v5tej";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6:
|
|
|
|
Triple += "v6";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6KZ:
|
|
|
|
Triple += "v6kz";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6T2:
|
|
|
|
Triple += "v6t2";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6K:
|
|
|
|
Triple += "v6k";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v7:
|
|
|
|
Triple += "v7";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6_M:
|
|
|
|
Triple += "v6m";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v6S_M:
|
|
|
|
Triple += "v6sm";
|
|
|
|
break;
|
|
|
|
case ARMBuildAttrs::v7E_M:
|
|
|
|
Triple += "v7em";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!isLittleEndian())
|
|
|
|
Triple += "eb";
|
|
|
|
|
|
|
|
TheTriple.setArchName(Triple);
|
|
|
|
}
|
2018-08-24 23:21:56 +08:00
|
|
|
|
|
|
|
std::vector<std::pair<DataRefImpl, uint64_t>>
|
|
|
|
ELFObjectFileBase::getPltAddresses() const {
|
|
|
|
std::string Err;
|
|
|
|
const auto Triple = makeTriple();
|
|
|
|
const auto *T = TargetRegistry::lookupTarget(Triple.str(), Err);
|
|
|
|
if (!T)
|
|
|
|
return {};
|
|
|
|
uint64_t JumpSlotReloc = 0;
|
|
|
|
switch (Triple.getArch()) {
|
|
|
|
case Triple::x86:
|
|
|
|
JumpSlotReloc = ELF::R_386_JUMP_SLOT;
|
|
|
|
break;
|
|
|
|
case Triple::x86_64:
|
|
|
|
JumpSlotReloc = ELF::R_X86_64_JUMP_SLOT;
|
|
|
|
break;
|
|
|
|
case Triple::aarch64:
|
|
|
|
JumpSlotReloc = ELF::R_AARCH64_JUMP_SLOT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return {};
|
|
|
|
}
|
2018-08-25 05:03:35 +08:00
|
|
|
std::unique_ptr<const MCInstrInfo> MII(T->createMCInstrInfo());
|
2018-08-25 03:40:35 +08:00
|
|
|
std::unique_ptr<const MCInstrAnalysis> MIA(
|
2018-08-25 05:03:35 +08:00
|
|
|
T->createMCInstrAnalysis(MII.get()));
|
2018-08-24 23:21:56 +08:00
|
|
|
if (!MIA)
|
|
|
|
return {};
|
|
|
|
Optional<SectionRef> Plt = None, RelaPlt = None, GotPlt = None;
|
|
|
|
for (const SectionRef &Section : sections()) {
|
|
|
|
StringRef Name;
|
|
|
|
if (Section.getName(Name))
|
|
|
|
continue;
|
|
|
|
if (Name == ".plt")
|
|
|
|
Plt = Section;
|
|
|
|
else if (Name == ".rela.plt" || Name == ".rel.plt")
|
|
|
|
RelaPlt = Section;
|
|
|
|
else if (Name == ".got.plt")
|
|
|
|
GotPlt = Section;
|
|
|
|
}
|
|
|
|
if (!Plt || !RelaPlt || !GotPlt)
|
|
|
|
return {};
|
2019-05-16 21:24:04 +08:00
|
|
|
Expected<StringRef> PltContents = Plt->getContents();
|
|
|
|
if (!PltContents) {
|
|
|
|
consumeError(PltContents.takeError());
|
2018-08-24 23:21:56 +08:00
|
|
|
return {};
|
2019-05-16 21:24:04 +08:00
|
|
|
}
|
|
|
|
auto PltEntries = MIA->findPltEntries(Plt->getAddress(),
|
|
|
|
arrayRefFromStringRef(*PltContents),
|
2018-08-24 23:21:56 +08:00
|
|
|
GotPlt->getAddress(), Triple);
|
|
|
|
// Build a map from GOT entry virtual address to PLT entry virtual address.
|
|
|
|
DenseMap<uint64_t, uint64_t> GotToPlt;
|
|
|
|
for (const auto &Entry : PltEntries)
|
|
|
|
GotToPlt.insert(std::make_pair(Entry.second, Entry.first));
|
|
|
|
// Find the relocations in the dynamic relocation table that point to
|
|
|
|
// locations in the GOT for which we know the corresponding PLT entry.
|
|
|
|
std::vector<std::pair<DataRefImpl, uint64_t>> Result;
|
|
|
|
for (const auto &Relocation : RelaPlt->relocations()) {
|
|
|
|
if (Relocation.getType() != JumpSlotReloc)
|
|
|
|
continue;
|
|
|
|
auto PltEntryIter = GotToPlt.find(Relocation.getOffset());
|
|
|
|
if (PltEntryIter != GotToPlt.end())
|
|
|
|
Result.push_back(std::make_pair(
|
|
|
|
Relocation.getSymbol()->getRawDataRefImpl(), PltEntryIter->second));
|
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
}
|