xref: /llvm-project/llvm/lib/CodeGen/AsmPrinter/AIXException.cpp (revision 4169338e75cdce73d34063532db598c95ee82ae4)
1a65d8c5dSjasonliu //===-- CodeGen/AsmPrinter/AIXException.cpp - AIX Exception Impl ----------===//
2a65d8c5dSjasonliu //
3a65d8c5dSjasonliu // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4a65d8c5dSjasonliu // See https://llvm.org/LICENSE.txt for license information.
5a65d8c5dSjasonliu // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6a65d8c5dSjasonliu //
7a65d8c5dSjasonliu //===----------------------------------------------------------------------===//
8a65d8c5dSjasonliu //
9a65d8c5dSjasonliu // This file contains support for writing AIX exception info into asm files.
10a65d8c5dSjasonliu //
11a65d8c5dSjasonliu //===----------------------------------------------------------------------===//
12a65d8c5dSjasonliu 
13a65d8c5dSjasonliu #include "DwarfException.h"
14a65d8c5dSjasonliu #include "llvm/CodeGen/AsmPrinter.h"
15a65d8c5dSjasonliu #include "llvm/CodeGen/MachineModuleInfo.h"
16a1e1dcabSdiggerlin #include "llvm/CodeGen/TargetLoweringObjectFileImpl.h"
17*4169338eSNikita Popov #include "llvm/IR/Module.h"
18e5926906SXing Xue #include "llvm/MC/MCSectionXCOFF.h"
19a65d8c5dSjasonliu #include "llvm/MC/MCStreamer.h"
20a65d8c5dSjasonliu #include "llvm/Target/TargetLoweringObjectFile.h"
21a65d8c5dSjasonliu #include "llvm/Target/TargetMachine.h"
22a65d8c5dSjasonliu 
23a65d8c5dSjasonliu namespace llvm {
24a65d8c5dSjasonliu 
AIXException(AsmPrinter * A)25648ce3d3SJames Y Knight AIXException::AIXException(AsmPrinter *A) : EHStreamer(A) {}
26a2505bd0SSean Fertile 
emitExceptionInfoTable(const MCSymbol * LSDA,const MCSymbol * PerSym)27a65d8c5dSjasonliu void AIXException::emitExceptionInfoTable(const MCSymbol *LSDA,
28a65d8c5dSjasonliu                                           const MCSymbol *PerSym) {
29a65d8c5dSjasonliu   // Generate EH Info Table.
30a65d8c5dSjasonliu   // The EH Info Table, aka, 'compat unwind section' on AIX, have the following
31a65d8c5dSjasonliu   // format: struct eh_info_t {
32a65d8c5dSjasonliu   //   unsigned version;           /* EH info verion 0 */
33a65d8c5dSjasonliu   // #if defined(__64BIT__)
34a65d8c5dSjasonliu   //   char _pad[4];               /* padding */
35a65d8c5dSjasonliu   // #endif
36a65d8c5dSjasonliu   //   unsigned long lsda;         /* Pointer to LSDA */
37a65d8c5dSjasonliu   //   unsigned long personality;  /* Pointer to the personality routine */
38a65d8c5dSjasonliu   //   }
39a65d8c5dSjasonliu 
40e5926906SXing Xue   auto *EHInfo =
41e5926906SXing Xue       cast<MCSectionXCOFF>(Asm->getObjFileLowering().getCompactUnwindSection());
42e5926906SXing Xue   if (Asm->TM.getFunctionSections()) {
43e5926906SXing Xue     // If option -ffunction-sections is on, append the function name to the
44e5926906SXing Xue     // name of EH Info Table csect so that each function has its own EH Info
45e5926906SXing Xue     // Table csect. This helps the linker to garbage-collect EH info of unused
46e5926906SXing Xue     // functions.
47e5926906SXing Xue     SmallString<128> NameStr = EHInfo->getName();
48e5926906SXing Xue     raw_svector_ostream(NameStr) << '.' << Asm->MF->getFunction().getName();
49e5926906SXing Xue     EHInfo = Asm->OutContext.getXCOFFSection(NameStr, EHInfo->getKind(),
50e5926906SXing Xue                                              EHInfo->getCsectProp());
51e5926906SXing Xue   }
52adf4142fSFangrui Song   Asm->OutStreamer->switchSection(EHInfo);
53a1e1dcabSdiggerlin   MCSymbol *EHInfoLabel =
54a1e1dcabSdiggerlin       TargetLoweringObjectFileXCOFF::getEHInfoTableSymbol(Asm->MF);
55a65d8c5dSjasonliu   Asm->OutStreamer->emitLabel(EHInfoLabel);
56a65d8c5dSjasonliu 
57a65d8c5dSjasonliu   // Version number.
58a65d8c5dSjasonliu   Asm->emitInt32(0);
59a65d8c5dSjasonliu 
60a65d8c5dSjasonliu   const DataLayout &DL = MMI->getModule()->getDataLayout();
61a65d8c5dSjasonliu   const unsigned PointerSize = DL.getPointerSize();
62a65d8c5dSjasonliu 
63a65d8c5dSjasonliu   // Add necessary paddings in 64 bit mode.
646c09ea3fSGuillaume Chatelet   Asm->OutStreamer->emitValueToAlignment(Align(PointerSize));
65a65d8c5dSjasonliu 
66a65d8c5dSjasonliu   // LSDA location.
67a65d8c5dSjasonliu   Asm->OutStreamer->emitValue(MCSymbolRefExpr::create(LSDA, Asm->OutContext),
68a65d8c5dSjasonliu                               PointerSize);
69a65d8c5dSjasonliu 
70a65d8c5dSjasonliu   // Personality routine.
71a65d8c5dSjasonliu   Asm->OutStreamer->emitValue(MCSymbolRefExpr::create(PerSym, Asm->OutContext),
72a65d8c5dSjasonliu                               PointerSize);
73a65d8c5dSjasonliu }
74a65d8c5dSjasonliu 
endFunction(const MachineFunction * MF)75a65d8c5dSjasonliu void AIXException::endFunction(const MachineFunction *MF) {
76bd240b3dSzhijian   // There is no easy way to access register information in `AIXException`
77bd240b3dSzhijian   // class. when ShouldEmitEHBlock is false and VRs are saved, A dumy eh info
78bd240b3dSzhijian   // table are emitted in PPCAIXAsmPrinter::emitFunctionBodyEnd.
79a1e1dcabSdiggerlin   if (!TargetLoweringObjectFileXCOFF::ShouldEmitEHBlock(MF))
80a65d8c5dSjasonliu     return;
81a65d8c5dSjasonliu 
82a65d8c5dSjasonliu   const MCSymbol *LSDALabel = emitExceptionTable();
83a1e1dcabSdiggerlin 
84a1e1dcabSdiggerlin   const Function &F = MF->getFunction();
85a1e1dcabSdiggerlin   assert(F.hasPersonalityFn() &&
86a1e1dcabSdiggerlin          "Landingpads are presented, but no personality routine is found.");
87ae95f291SSimon Pilgrim   const auto *Per =
88ae95f291SSimon Pilgrim       cast<GlobalValue>(F.getPersonalityFn()->stripPointerCasts());
89a65d8c5dSjasonliu   const MCSymbol *PerSym = Asm->TM.getSymbol(Per);
90a65d8c5dSjasonliu 
91a65d8c5dSjasonliu   emitExceptionInfoTable(LSDALabel, PerSym);
92a65d8c5dSjasonliu }
93a65d8c5dSjasonliu 
94a65d8c5dSjasonliu } // End of namespace llvm
95