2017-10-11 06:33:29 +08:00
|
|
|
//===- EHStreamer.h - Exception Handling Directive Streamer -----*- C++ -*-===//
|
2014-06-11 09:19:03 +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
|
2014-06-11 09:19:03 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains support for writing exception info into assembly files.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2014-08-14 00:26:38 +08:00
|
|
|
#ifndef LLVM_LIB_CODEGEN_ASMPRINTER_EHSTREAMER_H
|
|
|
|
#define LLVM_LIB_CODEGEN_ASMPRINTER_EHSTREAMER_H
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2018-12-19 07:10:17 +08:00
|
|
|
#include "llvm/CodeGen/AsmPrinterHandler.h"
|
2017-10-11 06:33:29 +08:00
|
|
|
#include "llvm/Support/Compiler.h"
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
namespace llvm {
|
2017-10-11 06:33:29 +08:00
|
|
|
|
|
|
|
class AsmPrinter;
|
2014-06-11 09:19:03 +08:00
|
|
|
struct LandingPadInfo;
|
|
|
|
class MachineInstr;
|
2017-10-11 06:33:29 +08:00
|
|
|
class MachineModuleInfo;
|
2015-01-14 09:05:27 +08:00
|
|
|
class MCSymbol;
|
2017-10-11 06:33:29 +08:00
|
|
|
template <typename T> class SmallVectorImpl;
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
/// Emits exception handling directives.
|
2015-07-02 00:18:16 +08:00
|
|
|
class LLVM_LIBRARY_VISIBILITY EHStreamer : public AsmPrinterHandler {
|
2014-06-11 09:19:03 +08:00
|
|
|
protected:
|
|
|
|
/// Target of directive emission.
|
|
|
|
AsmPrinter *Asm;
|
|
|
|
|
|
|
|
/// Collected machine module information.
|
|
|
|
MachineModuleInfo *MMI;
|
|
|
|
|
|
|
|
/// How many leading type ids two landing pads have in common.
|
|
|
|
static unsigned sharedTypeIDs(const LandingPadInfo *L,
|
|
|
|
const LandingPadInfo *R);
|
|
|
|
|
|
|
|
/// Structure holding a try-range and the associated landing pad.
|
|
|
|
struct PadRange {
|
|
|
|
// The index of the landing pad.
|
|
|
|
unsigned PadIndex;
|
2017-10-11 06:33:29 +08:00
|
|
|
|
2014-06-11 09:19:03 +08:00
|
|
|
// The index of the begin and end labels in the landing pad's label lists.
|
|
|
|
unsigned RangeIndex;
|
|
|
|
};
|
|
|
|
|
2017-10-11 06:33:29 +08:00
|
|
|
using RangeMapType = DenseMap<MCSymbol *, PadRange>;
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
/// Structure describing an entry in the actions table.
|
|
|
|
struct ActionEntry {
|
|
|
|
int ValueForTypeID; // The value to write - may not be equal to the type id.
|
|
|
|
int NextAction;
|
|
|
|
unsigned Previous;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Structure describing an entry in the call-site table.
|
|
|
|
struct CallSiteEntry {
|
|
|
|
// The 'try-range' is BeginLabel .. EndLabel.
|
2015-01-14 09:05:27 +08:00
|
|
|
MCSymbol *BeginLabel; // Null indicates the start of the function.
|
|
|
|
MCSymbol *EndLabel; // Null indicates the end of the function.
|
2014-06-11 09:19:03 +08:00
|
|
|
|
2015-01-14 09:05:27 +08:00
|
|
|
// LPad contains the landing pad start labels.
|
|
|
|
const LandingPadInfo *LPad; // Null indicates that there is no landing pad.
|
2017-10-11 06:33:29 +08:00
|
|
|
|
2014-06-11 09:19:03 +08:00
|
|
|
unsigned Action;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Compute the actions table and gather the first action index for each
|
|
|
|
/// landing pad site.
|
2018-07-17 02:51:40 +08:00
|
|
|
void computeActionsTable(const SmallVectorImpl<const LandingPadInfo *> &LandingPads,
|
2018-02-10 01:00:25 +08:00
|
|
|
SmallVectorImpl<ActionEntry> &Actions,
|
|
|
|
SmallVectorImpl<unsigned> &FirstActions);
|
2014-06-11 09:19:03 +08:00
|
|
|
|
2015-03-31 06:58:10 +08:00
|
|
|
void computePadMap(const SmallVectorImpl<const LandingPadInfo *> &LandingPads,
|
|
|
|
RangeMapType &PadMap);
|
|
|
|
|
2014-06-11 09:19:03 +08:00
|
|
|
/// Compute the call-site table. The entry for an invoke has a try-range
|
|
|
|
/// containing the call, a non-zero landing pad and an appropriate action.
|
|
|
|
/// The entry for an ordinary call has a try-range containing the call and
|
|
|
|
/// zero for the landing pad and the action. Calls marked 'nounwind' have
|
|
|
|
/// no entry and must not be contained in the try-range of any entry - they
|
|
|
|
/// form gaps in the table. Entries must be ordered by try-range address.
|
Reland "[WebAssembly] LSDA info generation"
Summary:
This adds support for LSDA (exception table) generation for wasm EH.
Wasm EH mostly follows the structure of Itanium-style exception tables,
with one exception: a call site table entry in wasm EH corresponds to
not a call site but a landing pad.
In wasm EH, the VM is responsible for stack unwinding. After an
exception occurs and the stack is unwound, the control flow is
transferred to wasm 'catch' instruction by the VM, after which the
personality function is called from the compiler-generated code. (Refer
to WasmEHPrepare pass for more information on this part.)
This patch:
- Changes wasm.landingpad.index intrinsic to take a token argument, to
make this 1:1 match with a catchpad instruction
- Stores landingpad index info and catch type info MachineFunction in
before instruction selection
- Lowers wasm.lsda intrinsic to an MCSymbol pointing to the start of an
exception table
- Adds WasmException class with overridden methods for table generation
- Adds support for LSDA section in Wasm object writer
Reviewers: dschuff, sbc100, rnk
Subscribers: mgorny, jgravelle-google, sunfish, llvm-commits
Differential Revision: https://reviews.llvm.org/D52748
llvm-svn: 345345
2018-10-26 07:55:10 +08:00
|
|
|
virtual void computeCallSiteTable(
|
|
|
|
SmallVectorImpl<CallSiteEntry> &CallSites,
|
|
|
|
const SmallVectorImpl<const LandingPadInfo *> &LandingPads,
|
|
|
|
const SmallVectorImpl<unsigned> &FirstActions);
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
/// Emit landing pads and actions.
|
|
|
|
///
|
|
|
|
/// The general organization of the table is complex, but the basic concepts
|
|
|
|
/// are easy. First there is a header which describes the location and
|
|
|
|
/// organization of the three components that follow.
|
|
|
|
/// 1. The landing pad site information describes the range of code covered
|
|
|
|
/// by the try. In our case it's an accumulation of the ranges covered
|
|
|
|
/// by the invokes in the try. There is also a reference to the landing
|
|
|
|
/// pad that handles the exception once processed. Finally an index into
|
|
|
|
/// the actions table.
|
|
|
|
/// 2. The action table, in our case, is composed of pairs of type ids
|
|
|
|
/// and next action offset. Starting with the action index from the
|
|
|
|
/// landing pad site, each type Id is checked for a match to the current
|
|
|
|
/// exception. If it matches then the exception and type id are passed
|
|
|
|
/// on to the landing pad. Otherwise the next action is looked up. This
|
|
|
|
/// chain is terminated with a next action of zero. If no type id is
|
|
|
|
/// found the frame is unwound and handling continues.
|
|
|
|
/// 3. Type id table contains references to all the C++ typeinfo for all
|
|
|
|
/// catches in the function. This tables is reversed indexed base 1.
|
Reland "[WebAssembly] LSDA info generation"
Summary:
This adds support for LSDA (exception table) generation for wasm EH.
Wasm EH mostly follows the structure of Itanium-style exception tables,
with one exception: a call site table entry in wasm EH corresponds to
not a call site but a landing pad.
In wasm EH, the VM is responsible for stack unwinding. After an
exception occurs and the stack is unwound, the control flow is
transferred to wasm 'catch' instruction by the VM, after which the
personality function is called from the compiler-generated code. (Refer
to WasmEHPrepare pass for more information on this part.)
This patch:
- Changes wasm.landingpad.index intrinsic to take a token argument, to
make this 1:1 match with a catchpad instruction
- Stores landingpad index info and catch type info MachineFunction in
before instruction selection
- Lowers wasm.lsda intrinsic to an MCSymbol pointing to the start of an
exception table
- Adds WasmException class with overridden methods for table generation
- Adds support for LSDA section in Wasm object writer
Reviewers: dschuff, sbc100, rnk
Subscribers: mgorny, jgravelle-google, sunfish, llvm-commits
Differential Revision: https://reviews.llvm.org/D52748
llvm-svn: 345345
2018-10-26 07:55:10 +08:00
|
|
|
///
|
|
|
|
/// Returns the starting symbol of an exception table.
|
|
|
|
MCSymbol *emitExceptionTable();
|
2014-06-11 09:19:03 +08:00
|
|
|
|
2018-02-10 01:00:25 +08:00
|
|
|
virtual void emitTypeInfos(unsigned TTypeEncoding, MCSymbol *TTBaseLabel);
|
2014-06-11 09:19:03 +08:00
|
|
|
|
2018-01-26 16:15:29 +08:00
|
|
|
// Helpers for identifying what kind of clause an EH typeid or selector
|
2015-01-14 09:05:27 +08:00
|
|
|
// corresponds to. Negative selectors are for filter clauses, the zero
|
|
|
|
// selector is for cleanups, and positive selectors are for catch clauses.
|
|
|
|
static bool isFilterEHSelector(int Selector) { return Selector < 0; }
|
|
|
|
static bool isCleanupEHSelector(int Selector) { return Selector == 0; }
|
|
|
|
static bool isCatchEHSelector(int Selector) { return Selector > 0; }
|
|
|
|
|
2014-06-11 09:19:03 +08:00
|
|
|
public:
|
|
|
|
EHStreamer(AsmPrinter *A);
|
2015-04-11 10:11:45 +08:00
|
|
|
~EHStreamer() override;
|
2014-06-11 09:19:03 +08:00
|
|
|
|
|
|
|
// Unused.
|
|
|
|
void setSymbolSize(const MCSymbol *Sym, uint64_t Size) override {}
|
|
|
|
void beginInstruction(const MachineInstr *MI) override {}
|
|
|
|
void endInstruction() override {}
|
2015-10-02 05:38:24 +08:00
|
|
|
|
|
|
|
/// Return `true' if this is a call to a function marked `nounwind'. Return
|
|
|
|
/// `false' otherwise.
|
|
|
|
static bool callToNoUnwindFunction(const MachineInstr *MI);
|
2014-06-11 09:19:03 +08:00
|
|
|
};
|
|
|
|
|
2017-10-11 06:33:29 +08:00
|
|
|
} // end namespace llvm
|
2014-06-11 09:19:03 +08:00
|
|
|
|
2017-10-11 06:33:29 +08:00
|
|
|
#endif // LLVM_LIB_CODEGEN_ASMPRINTER_EHSTREAMER_H
|