blob: 6d06e873e4d07dca2db648431c9cb28b327ceda8 [file] [log] [blame] [edit]
//===----------------------------------------------------------------------===//
//
// 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
//
//===----------------------------------------------------------------------===//
//
// On arm64e, Clang emits ConstantPtrAuth expressions in global initializers
// to represent signed pointers. These are normally resolved by the dynamic
// linker, but LLDB's JIT does not run the linker, so they must be resolved
// manually. This pass replaces each ConstantPtrAuth in a global initializer
// with the unsigned pointer and emits a constructor function that signs the
// pointer at runtime using the ptrauth intrinsics.
//
// Example: given "static int (*fp)(int, int) = &mul;", Clang emits:
//
// @fp = internal global ptr ptrauth (ptr @mul, i32 0)
//
// This pass transforms it into:
//
// @fp = internal global ptr @mul
// @llvm.global_ctors = appending global [1 x { i32, ptr, ptr }]
// [{ i32, ptr, ptr } { i32 0, ptr @ptrauth.sign, ptr null }]
//
// define internal void @ptrauth.sign() {
// %1 = load ptr, ptr @fp, align 8
// %2 = ptrtoint ptr %1 to i64
// %3 = call i64 @llvm.ptrauth.sign(i64 %2, i32 0, i64 0)
// %4 = inttoptr i64 %3 to ptr
// store ptr %4, ptr @fp, align 8
// ret void
// }
//
//===----------------------------------------------------------------------===//
#include "InjectPointerSigningFixups.h"
#include "llvm/IR/Constants.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/IRBuilder.h"
#include "llvm/IR/Module.h"
#include "llvm/Support/raw_ostream.h"
#include "llvm/TargetParser/Triple.h"
using namespace llvm;
namespace {
struct PtrAuthFixup {
GlobalVariable *GV;
ConstantPtrAuth *CPA;
SmallVector<unsigned> Indices;
PtrAuthFixup(GlobalVariable *GV, ConstantPtrAuth *CPA,
const SmallVectorImpl<unsigned> &Indices)
: GV(GV), CPA(CPA), Indices(Indices.begin(), Indices.end()) {}
};
} // namespace
/// Recursively walk a constant looking for ConstantPtrAuth expressions.
/// When found, record the global variable containing the ConstantPtrAuth and
/// the index path to reach it within the initializer.
static void findPtrAuth(Constant *C, GlobalVariable &GV,
SmallVectorImpl<unsigned> &Indices,
SmallVectorImpl<PtrAuthFixup> &Fixups) {
if (auto *CPA = dyn_cast<ConstantPtrAuth>(C)) {
Fixups.emplace_back(&GV, CPA, Indices);
return;
}
for (unsigned I = 0, E = C->getNumOperands(); I != E; ++I) {
if (auto *COp = dyn_cast<Constant>(C->getOperand(I))) {
Indices.push_back(I);
findPtrAuth(COp, GV, Indices, Fixups);
Indices.pop_back();
}
}
}
namespace lldb_private {
Error InjectPointerSigningFixupCode(llvm::Module &M,
ExecutionPolicy execution_policy) {
// If we cannot execute fixups, don't insert them.
if (execution_policy == eExecutionPolicyNever)
return Error::success();
llvm::Triple T(M.getTargetTriple());
// Bail out if we don't need pointer signing fixups.
if (!T.isArm64e())
return Error::success();
// Collect all ConstantPtrAuth expressions in global initializers.
SmallVector<PtrAuthFixup> Fixups;
for (auto &G : M.globals()) {
if (!G.hasInitializer())
continue;
SmallVector<unsigned> Indices;
findPtrAuth(G.getInitializer(), G, Indices, Fixups);
}
if (Fixups.empty())
return Error::success();
// Set up types and intrinsics.
auto &Ctx = M.getContext();
Type *Int32Ty = Type::getInt32Ty(Ctx);
Type *IntPtrTy = Type::getInt64Ty(Ctx);
Function *BlendIntrinsic =
Intrinsic::getOrInsertDeclaration(&M, Intrinsic::ptrauth_blend);
Function *SignIntrinsic =
Intrinsic::getOrInsertDeclaration(&M, Intrinsic::ptrauth_sign);
// Create the fixup function.
Function *FixupFn =
Function::Create(FunctionType::get(Type::getVoidTy(Ctx), false),
GlobalValue::InternalLinkage, "ptrauth.sign", &M);
FixupFn->insert(FixupFn->end(), BasicBlock::Create(Ctx));
IRBuilder<> B(&FixupFn->back());
for (auto &Fixup : Fixups) {
GlobalVariable *GV = Fixup.GV;
ConstantPtrAuth *CPA = Fixup.CPA;
// Null pointers must remain zero.
if (isa<ConstantPointerNull>(CPA->getPointer())) {
CPA->replaceAllUsesWith(CPA->getPointer());
continue;
}
// Build a GEP to the location of the ConstantPtrAuth within the global.
Value *Loc;
if (Fixup.Indices.empty()) {
Loc = GV;
} else {
SmallVector<Value *> GEPIndices;
GEPIndices.push_back(ConstantInt::get(Int32Ty, 0));
for (unsigned Idx : Fixup.Indices)
GEPIndices.push_back(ConstantInt::get(Int32Ty, Idx));
Loc = B.CreateGEP(GV->getValueType(), GV, GEPIndices);
}
Type *PtrTy = CPA->getType();
// Load the raw (unsigned) pointer.
Value *RawPtr = B.CreateLoad(PtrTy, Loc);
// Compute the discriminator, blending with the address if needed.
Value *Disc = CPA->getDiscriminator();
if (CPA->hasAddressDiscriminator())
Disc = B.CreateCall(BlendIntrinsic,
{B.CreatePointerCast(Loc, IntPtrTy), Disc});
// Sign the pointer.
Value *SignedPtr =
B.CreateCall(SignIntrinsic, {B.CreatePointerCast(RawPtr, IntPtrTy),
CPA->getKey(), Disc});
// Store the signed pointer back.
B.CreateStore(B.CreateBitOrPointerCast(SignedPtr, PtrTy), Loc);
// Replace the ConstantPtrAuth in the initializer with the unsigned pointer.
CPA->replaceAllUsesWith(CPA->getPointer());
}
// Close off the fixup function.
B.CreateRetVoid();
// Update the global ctors list to call the pointer fixup function first.
auto *UInt8PtrTy = PointerType::getUnqual(Ctx);
StructType *CtorType =
StructType::get(Ctx, {Int32Ty, FixupFn->getType(), UInt8PtrTy});
Constant *PtrFixupCtor =
ConstantStruct::get(CtorType, {ConstantInt::get(Int32Ty, 0), FixupFn,
Constant::getNullValue(UInt8PtrTy)});
const char *LLVMGlobalCtorsName = "llvm.global_ctors";
GlobalVariable *OldCtorList = M.getNamedGlobal(LLVMGlobalCtorsName);
SmallVector<Constant *> CtorListArgs;
CtorListArgs.push_back(PtrFixupCtor);
if (OldCtorList) {
// If the old ctors list has any uses then bail out: we do not know how to
// rewrite them.
if (OldCtorList->getNumUses() != 0) {
std::string ErrStr;
raw_string_ostream S(ErrStr);
S << "Global ctors variable has users, so can not be rewritten to "
"include pointer fixups: '"
<< *OldCtorList << "'";
return make_error<StringError>(S.str(), inconvertibleErrorCode());
}
for (auto &Op : OldCtorList->getInitializer()->operands())
CtorListArgs.push_back(cast<Constant>(Op.get()));
}
ArrayType *CtorListType = ArrayType::get(CtorType, CtorListArgs.size());
Constant *CtorListInit = ConstantArray::get(CtorListType, CtorListArgs);
GlobalVariable *NewCtorList = new GlobalVariable(
M, CtorListType, false, GlobalValue::AppendingLinkage, CtorListInit);
if (OldCtorList) {
NewCtorList->takeName(OldCtorList);
OldCtorList->eraseFromParent();
} else
NewCtorList->setName(LLVMGlobalCtorsName);
return Error::success();
}
} // namespace lldb_private