mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-25 20:23:11 +01:00
1243bb96b5
Summary: Fix the computation of the offsets present in the scopetable when using the SEH (__except_handler4). This patch added an intrinsic to track the position of the allocation on the stack of the EHGuard. This position is needed when producing the ScopeTable. ``` struct _EH4_SCOPETABLE { DWORD GSCookieOffset; DWORD GSCookieXOROffset; DWORD EHCookieOffset; DWORD EHCookieXOROffset; _EH4_SCOPETABLE_RECORD ScopeRecord[1]; }; struct _EH4_SCOPETABLE_RECORD { DWORD EnclosingLevel; long (*FilterFunc)(); union { void (*HandlerAddress)(); void (*FinallyFunc)(); }; }; ``` The code to generate the EHCookie is added in `X86WinEHState.cpp`. Which is adding these instructions when using SEH4. ``` Lfunc_begin0: # BB#0: # %entry pushl %ebp movl %esp, %ebp pushl %ebx pushl %edi pushl %esi subl $28, %esp movl %ebp, %eax <<-- Loading FramePtr movl %esp, -36(%ebp) movl $-2, -16(%ebp) movl $L__ehtable$use_except_handler4_ssp, %ecx xorl ___security_cookie, %ecx movl %ecx, -20(%ebp) xorl ___security_cookie, %eax <<-- XOR FramePtr and Cookie movl %eax, -40(%ebp) <<-- Storing EHGuard leal -28(%ebp), %eax movl $__except_handler4, -24(%ebp) movl %fs:0, %ecx movl %ecx, -28(%ebp) movl %eax, %fs:0 movl $0, -16(%ebp) calll _may_throw_or_crash LBB1_1: # %cont movl -28(%ebp), %eax movl %eax, %fs:0 addl $28, %esp popl %esi popl %edi popl %ebx popl %ebp retl ``` And the corresponding offset is computed: ``` Luse_except_handler4_ssp$parent_frame_offset = -36 .p2align 2 L__ehtable$use_except_handler4_ssp: .long -2 # GSCookieOffset .long 0 # GSCookieXOROffset .long -40 # EHCookieOffset <<---- .long 0 # EHCookieXOROffset .long -2 # ToState .long _catchall_filt # FilterFunction .long LBB1_2 # ExceptionHandler ``` Clang is not yet producing function using SEH4, but it's a work in progress. This patch is a step toward having a valid implementation of SEH4. Unfortunately, it is not yet fully working. The EH registration block is not allocated at the right offset on the stack. Reviewers: rnk, majnemer Subscribers: llvm-commits, chrisha Differential Revision: http://reviews.llvm.org/D21231 llvm-svn: 273281
130 lines
3.9 KiB
C++
130 lines
3.9 KiB
C++
//===-- llvm/CodeGen/WinEHFuncInfo.h ----------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// Data structures and associated state for Windows exception handling schemes.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_CODEGEN_WINEHFUNCINFO_H
|
|
#define LLVM_CODEGEN_WINEHFUNCINFO_H
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
#include "llvm/ADT/PointerUnion.h"
|
|
#include "llvm/ADT/SmallVector.h"
|
|
#include "llvm/ADT/TinyPtrVector.h"
|
|
#include "llvm/IR/Instructions.h"
|
|
|
|
namespace llvm {
|
|
class AllocaInst;
|
|
class BasicBlock;
|
|
class CatchReturnInst;
|
|
class Constant;
|
|
class Function;
|
|
class GlobalVariable;
|
|
class InvokeInst;
|
|
class IntrinsicInst;
|
|
class LandingPadInst;
|
|
class MCExpr;
|
|
class MCSymbol;
|
|
class MachineBasicBlock;
|
|
class Value;
|
|
|
|
// The following structs respresent the .xdata tables for various
|
|
// Windows-related EH personalities.
|
|
|
|
typedef PointerUnion<const BasicBlock *, MachineBasicBlock *> MBBOrBasicBlock;
|
|
|
|
struct CxxUnwindMapEntry {
|
|
int ToState;
|
|
MBBOrBasicBlock Cleanup;
|
|
};
|
|
|
|
/// Similar to CxxUnwindMapEntry, but supports SEH filters.
|
|
struct SEHUnwindMapEntry {
|
|
/// If unwinding continues through this handler, transition to the handler at
|
|
/// this state. This indexes into SEHUnwindMap.
|
|
int ToState = -1;
|
|
|
|
bool IsFinally = false;
|
|
|
|
/// Holds the filter expression function.
|
|
const Function *Filter = nullptr;
|
|
|
|
/// Holds the __except or __finally basic block.
|
|
MBBOrBasicBlock Handler;
|
|
};
|
|
|
|
struct WinEHHandlerType {
|
|
int Adjectives;
|
|
/// The CatchObj starts out life as an LLVM alloca and is eventually turned
|
|
/// frame index.
|
|
union {
|
|
const AllocaInst *Alloca;
|
|
int FrameIndex;
|
|
} CatchObj = {};
|
|
GlobalVariable *TypeDescriptor;
|
|
MBBOrBasicBlock Handler;
|
|
};
|
|
|
|
struct WinEHTryBlockMapEntry {
|
|
int TryLow = -1;
|
|
int TryHigh = -1;
|
|
int CatchHigh = -1;
|
|
SmallVector<WinEHHandlerType, 1> HandlerArray;
|
|
};
|
|
|
|
enum class ClrHandlerType { Catch, Finally, Fault, Filter };
|
|
|
|
struct ClrEHUnwindMapEntry {
|
|
MBBOrBasicBlock Handler;
|
|
uint32_t TypeToken;
|
|
int HandlerParentState; ///< Outer handler enclosing this entry's handler
|
|
int TryParentState; ///< Outer try region enclosing this entry's try region,
|
|
///< treating later catches on same try as "outer"
|
|
ClrHandlerType HandlerType;
|
|
};
|
|
|
|
struct WinEHFuncInfo {
|
|
DenseMap<const Instruction *, int> EHPadStateMap;
|
|
DenseMap<const FuncletPadInst *, int> FuncletBaseStateMap;
|
|
DenseMap<const InvokeInst *, int> InvokeStateMap;
|
|
DenseMap<MCSymbol *, std::pair<int, MCSymbol *>> LabelToStateMap;
|
|
SmallVector<CxxUnwindMapEntry, 4> CxxUnwindMap;
|
|
SmallVector<WinEHTryBlockMapEntry, 4> TryBlockMap;
|
|
SmallVector<SEHUnwindMapEntry, 4> SEHUnwindMap;
|
|
SmallVector<ClrEHUnwindMapEntry, 4> ClrEHUnwindMap;
|
|
int UnwindHelpFrameIdx = INT_MAX;
|
|
int PSPSymFrameIdx = INT_MAX;
|
|
|
|
int getLastStateNumber() const { return CxxUnwindMap.size() - 1; }
|
|
|
|
void addIPToStateRange(const InvokeInst *II, MCSymbol *InvokeBegin,
|
|
MCSymbol *InvokeEnd);
|
|
|
|
int EHRegNodeFrameIndex = INT_MAX;
|
|
int EHRegNodeEndOffset = INT_MAX;
|
|
int EHGuardFrameIndex = INT_MAX;
|
|
int SEHSetFrameOffset = INT_MAX;
|
|
|
|
WinEHFuncInfo();
|
|
};
|
|
|
|
/// Analyze the IR in ParentFn and it's handlers to build WinEHFuncInfo, which
|
|
/// describes the state numbers and tables used by __CxxFrameHandler3. This
|
|
/// analysis assumes that WinEHPrepare has already been run.
|
|
void calculateWinCXXEHStateNumbers(const Function *ParentFn,
|
|
WinEHFuncInfo &FuncInfo);
|
|
|
|
void calculateSEHStateNumbers(const Function *ParentFn,
|
|
WinEHFuncInfo &FuncInfo);
|
|
|
|
void calculateClrEHStateNumbers(const Function *Fn, WinEHFuncInfo &FuncInfo);
|
|
}
|
|
#endif // LLVM_CODEGEN_WINEHFUNCINFO_H
|