mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2025-05-28 18:46:00 +00:00

Summary: A new function pass (Transforms/CFGuard/CFGuard.cpp) inserts CFGuard checks on indirect function calls, using either the check mechanism (X86, ARM, AArch64) or or the dispatch mechanism (X86-64). The check mechanism requires a new calling convention for the supported targets. The dispatch mechanism adds the target as an operand bundle, which is processed by SelectionDAG. Another pass (CodeGen/CFGuardLongjmp.cpp) identifies and emits valid longjmp targets, as required by /guard:cf. This feature is enabled using the `cfguard` CC1 option. Reviewers: thakis, rnk, theraven, pcc Subscribers: ychen, hans, metalcanine, dmajor, tomrittervg, alex, mehdi_amini, mgorny, javed.absar, kristof.beyls, hiraditya, steven_wu, dexonsmith, cfe-commits, llvm-commits Tags: #clang, #llvm Differential Revision: https://reviews.llvm.org/D65761
63 lines
2.1 KiB
C++
63 lines
2.1 KiB
C++
//===-- CodeGen/AsmPrinter/WinCFGuard.cpp - Control Flow Guard Impl ------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains support for writing the metadata for Windows Control Flow
|
|
// Guard, including address-taken functions, and valid longjmp targets.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "WinCFGuard.h"
|
|
#include "llvm/CodeGen/AsmPrinter.h"
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
|
#include "llvm/CodeGen/MachineModuleInfo.h"
|
|
#include "llvm/CodeGen/MachineOperand.h"
|
|
#include "llvm/IR/Constants.h"
|
|
#include "llvm/IR/Metadata.h"
|
|
#include "llvm/MC/MCAsmInfo.h"
|
|
#include "llvm/MC/MCObjectFileInfo.h"
|
|
#include "llvm/MC/MCStreamer.h"
|
|
|
|
#include <vector>
|
|
|
|
using namespace llvm;
|
|
|
|
WinCFGuard::WinCFGuard(AsmPrinter *A) : AsmPrinterHandler(), Asm(A) {}
|
|
|
|
WinCFGuard::~WinCFGuard() {}
|
|
|
|
void WinCFGuard::endFunction(const MachineFunction *MF) {
|
|
|
|
// Skip functions without any longjmp targets.
|
|
if (MF->getLongjmpTargets().empty())
|
|
return;
|
|
|
|
// Copy the function's longjmp targets to a module-level list.
|
|
LongjmpTargets.insert(LongjmpTargets.end(), MF->getLongjmpTargets().begin(),
|
|
MF->getLongjmpTargets().end());
|
|
}
|
|
|
|
void WinCFGuard::endModule() {
|
|
const Module *M = Asm->MMI->getModule();
|
|
std::vector<const Function *> Functions;
|
|
for (const Function &F : *M)
|
|
if (F.hasAddressTaken())
|
|
Functions.push_back(&F);
|
|
if (Functions.empty() && LongjmpTargets.empty())
|
|
return;
|
|
auto &OS = *Asm->OutStreamer;
|
|
OS.SwitchSection(Asm->OutContext.getObjectFileInfo()->getGFIDsSection());
|
|
for (const Function *F : Functions)
|
|
OS.EmitCOFFSymbolIndex(Asm->getSymbol(F));
|
|
|
|
// Emit the symbol index of each longjmp target.
|
|
OS.SwitchSection(Asm->OutContext.getObjectFileInfo()->getGLJMPSection());
|
|
for (const MCSymbol *S : LongjmpTargets) {
|
|
OS.EmitCOFFSymbolIndex(S);
|
|
}
|
|
}
|