1 //===- PrettyClassDefinitionDumper.cpp --------------------------*- C++ -*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "PrettyClassDefinitionDumper.h" 10 11 #include "PrettyClassLayoutGraphicalDumper.h" 12 #include "llvm-pdbutil.h" 13 14 #include "llvm/ADT/APFloat.h" 15 #include "llvm/ADT/SmallString.h" 16 #include "llvm/DebugInfo/PDB/IPDBLineNumber.h" 17 #include "llvm/DebugInfo/PDB/PDBSymbolTypeBaseClass.h" 18 #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionSig.h" 19 #include "llvm/DebugInfo/PDB/PDBSymbolTypeUDT.h" 20 #include "llvm/DebugInfo/PDB/UDTLayout.h" 21 22 #include "llvm/Support/Format.h" 23 24 using namespace llvm; 25 using namespace llvm::pdb; 26 27 ClassDefinitionDumper::ClassDefinitionDumper(LinePrinter &P) 28 : PDBSymDumper(true), Printer(P) {} 29 30 void ClassDefinitionDumper::start(const PDBSymbolTypeUDT &Class) { 31 assert(opts::pretty::ClassFormat != 32 opts::pretty::ClassDefinitionFormat::None); 33 34 ClassLayout Layout(Class); 35 start(Layout); 36 } 37 38 void ClassDefinitionDumper::start(const ClassLayout &Layout) { 39 prettyPrintClassIntro(Layout); 40 41 PrettyClassLayoutGraphicalDumper Dumper(Printer, 1, 0); 42 DumpedAnything |= Dumper.start(Layout); 43 44 prettyPrintClassOutro(Layout); 45 } 46 47 void ClassDefinitionDumper::prettyPrintClassIntro(const ClassLayout &Layout) { 48 DumpedAnything = false; 49 Printer.NewLine(); 50 51 uint32_t Size = Layout.getSize(); 52 const PDBSymbolTypeUDT &Class = Layout.getClass(); 53 54 if (Layout.getClass().isConstType()) 55 WithColor(Printer, PDB_ColorItem::Keyword).get() << "const "; 56 if (Layout.getClass().isVolatileType()) 57 WithColor(Printer, PDB_ColorItem::Keyword).get() << "volatile "; 58 if (Layout.getClass().isUnalignedType()) 59 WithColor(Printer, PDB_ColorItem::Keyword).get() << "unaligned "; 60 61 WithColor(Printer, PDB_ColorItem::Keyword).get() << Class.getUdtKind() << " "; 62 WithColor(Printer, PDB_ColorItem::Type).get() << Class.getName(); 63 WithColor(Printer, PDB_ColorItem::Comment).get() << " [sizeof = " << Size 64 << "]"; 65 uint32_t BaseCount = Layout.bases().size(); 66 if (BaseCount > 0) { 67 Printer.Indent(); 68 char NextSeparator = ':'; 69 for (auto BC : Layout.bases()) { 70 const auto &Base = BC->getBase(); 71 if (Base.isIndirectVirtualBaseClass()) 72 continue; 73 74 Printer.NewLine(); 75 Printer << NextSeparator << " "; 76 WithColor(Printer, PDB_ColorItem::Keyword).get() << Base.getAccess(); 77 if (BC->isVirtualBase()) 78 WithColor(Printer, PDB_ColorItem::Keyword).get() << " virtual"; 79 80 WithColor(Printer, PDB_ColorItem::Type).get() << " " << Base.getName(); 81 NextSeparator = ','; 82 } 83 84 Printer.Unindent(); 85 } 86 87 Printer << " {"; 88 Printer.Indent(); 89 } 90 91 void ClassDefinitionDumper::prettyPrintClassOutro(const ClassLayout &Layout) { 92 Printer.Unindent(); 93 if (DumpedAnything) 94 Printer.NewLine(); 95 Printer << "}"; 96 Printer.NewLine(); 97 if (Layout.deepPaddingSize() > 0) { 98 APFloat Pct(100.0 * (double)Layout.deepPaddingSize() / 99 (double)Layout.getSize()); 100 SmallString<8> PctStr; 101 Pct.toString(PctStr, 4); 102 WithColor(Printer, PDB_ColorItem::Padding).get() 103 << "Total padding " << Layout.deepPaddingSize() << " bytes (" << PctStr 104 << "% of class size)"; 105 Printer.NewLine(); 106 APFloat Pct2(100.0 * (double)Layout.immediatePadding() / 107 (double)Layout.getSize()); 108 PctStr.clear(); 109 Pct2.toString(PctStr, 4); 110 WithColor(Printer, PDB_ColorItem::Padding).get() 111 << "Immediate padding " << Layout.immediatePadding() << " bytes (" 112 << PctStr << "% of class size)"; 113 Printer.NewLine(); 114 } 115 } 116