Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 1 | //===- lib/MC/MCSectionELF.cpp - ELF Code Section Representation ----------===// |
| 2 | // |
| 3 | // The LLVM Compiler Infrastructure |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
| 9 | |
Chandler Carruth | e3e43d9 | 2017-06-06 11:49:48 +0000 | [diff] [blame] | 10 | #include "llvm/MC/MCSectionELF.h" |
Eugene Zelenko | 7211c53 | 2017-02-14 00:33:36 +0000 | [diff] [blame] | 11 | #include "llvm/ADT/Triple.h" |
Zachary Turner | 19ca2b0 | 2017-06-07 03:48:56 +0000 | [diff] [blame] | 12 | #include "llvm/BinaryFormat/ELF.h" |
Chris Lattner | af76e59 | 2009-08-22 20:48:53 +0000 | [diff] [blame] | 13 | #include "llvm/MC/MCAsmInfo.h" |
Peter Collingbourne | df39be6 | 2013-04-17 21:18:16 +0000 | [diff] [blame] | 14 | #include "llvm/MC/MCExpr.h" |
Simon Atanasyan | d1a4b1d | 2017-03-10 08:22:13 +0000 | [diff] [blame] | 15 | #include "llvm/Support/ErrorHandling.h" |
Chris Lattner | 4d2419d | 2010-01-13 21:21:29 +0000 | [diff] [blame] | 16 | #include "llvm/Support/raw_ostream.h" |
Eugene Zelenko | 7211c53 | 2017-02-14 00:33:36 +0000 | [diff] [blame] | 17 | #include <cassert> |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 18 | |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 19 | using namespace llvm; |
| 20 | |
Eugene Zelenko | 7211c53 | 2017-02-14 00:33:36 +0000 | [diff] [blame] | 21 | MCSectionELF::~MCSectionELF() = default; // anchor. |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 22 | |
Sanjay Patel | c160a9a | 2014-10-09 21:23:39 +0000 | [diff] [blame] | 23 | // Decides whether a '.section' directive |
| 24 | // should be printed before the section name. |
Benjamin Kramer | a46918d | 2010-01-22 18:21:23 +0000 | [diff] [blame] | 25 | bool MCSectionELF::ShouldOmitSectionDirective(StringRef Name, |
Chris Lattner | 4d2419d | 2010-01-13 21:21:29 +0000 | [diff] [blame] | 26 | const MCAsmInfo &MAI) const { |
Rafael Espindola | 903f4a2 | 2015-04-04 18:02:01 +0000 | [diff] [blame] | 27 | if (isUnique()) |
Rafael Espindola | e0a2541 | 2015-02-17 20:48:01 +0000 | [diff] [blame] | 28 | return false; |
| 29 | |
Tom Stellard | 68f9d1c | 2015-09-25 21:41:14 +0000 | [diff] [blame] | 30 | return MAI.shouldOmitSectionDirective(Name); |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 31 | } |
| 32 | |
Rafael Espindola | de9a1a2 | 2013-11-13 14:01:59 +0000 | [diff] [blame] | 33 | static void printName(raw_ostream &OS, StringRef Name) { |
| 34 | if (Name.find_first_not_of("0123456789_." |
| 35 | "abcdefghijklmnopqrstuvwxyz" |
| 36 | "ABCDEFGHIJKLMNOPQRSTUVWXYZ") == Name.npos) { |
| 37 | OS << Name; |
| 38 | return; |
| 39 | } |
| 40 | OS << '"'; |
| 41 | for (const char *B = Name.begin(), *E = Name.end(); B < E; ++B) { |
| 42 | if (*B == '"') // Unquoted " |
| 43 | OS << "\\\""; |
| 44 | else if (*B != '\\') // Neither " or backslash |
| 45 | OS << *B; |
| 46 | else if (B + 1 == E) // Trailing backslash |
| 47 | OS << "\\\\"; |
| 48 | else { |
| 49 | OS << B[0] << B[1]; // Quoted character |
| 50 | ++B; |
| 51 | } |
| 52 | } |
| 53 | OS << '"'; |
| 54 | } |
| 55 | |
Rafael Espindola | d5702b7 | 2017-01-30 15:38:43 +0000 | [diff] [blame] | 56 | void MCSectionELF::PrintSwitchToSection(const MCAsmInfo &MAI, const Triple &T, |
Peter Collingbourne | df39be6 | 2013-04-17 21:18:16 +0000 | [diff] [blame] | 57 | raw_ostream &OS, |
| 58 | const MCExpr *Subsection) const { |
Benjamin Kramer | a46918d | 2010-01-22 18:21:23 +0000 | [diff] [blame] | 59 | if (ShouldOmitSectionDirective(SectionName, MAI)) { |
Peter Collingbourne | df39be6 | 2013-04-17 21:18:16 +0000 | [diff] [blame] | 60 | OS << '\t' << getSectionName(); |
Matt Arsenault | d99ce2f | 2015-06-09 00:31:39 +0000 | [diff] [blame] | 61 | if (Subsection) { |
| 62 | OS << '\t'; |
| 63 | Subsection->print(OS, &MAI); |
| 64 | } |
Peter Collingbourne | df39be6 | 2013-04-17 21:18:16 +0000 | [diff] [blame] | 65 | OS << '\n'; |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 66 | return; |
| 67 | } |
| 68 | |
Rafael Espindola | de9a1a2 | 2013-11-13 14:01:59 +0000 | [diff] [blame] | 69 | OS << "\t.section\t"; |
| 70 | printName(OS, getSectionName()); |
Joerg Sonnenberger | ea83b13 | 2011-03-03 22:31:08 +0000 | [diff] [blame] | 71 | |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 72 | // Handle the weird solaris syntax if desired. |
Jim Grosbach | 2684d9e | 2012-05-11 01:41:30 +0000 | [diff] [blame] | 73 | if (MAI.usesSunStyleELFSectionSwitchSyntax() && |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 74 | !(Flags & ELF::SHF_MERGE)) { |
| 75 | if (Flags & ELF::SHF_ALLOC) |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 76 | OS << ",#alloc"; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 77 | if (Flags & ELF::SHF_EXECINSTR) |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 78 | OS << ",#execinstr"; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 79 | if (Flags & ELF::SHF_WRITE) |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 80 | OS << ",#write"; |
Benjamin Kramer | 766f253 | 2013-09-15 19:53:20 +0000 | [diff] [blame] | 81 | if (Flags & ELF::SHF_EXCLUDE) |
| 82 | OS << ",#exclude"; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 83 | if (Flags & ELF::SHF_TLS) |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 84 | OS << ",#tls"; |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 85 | OS << '\n'; |
| 86 | return; |
| 87 | } |
Jim Grosbach | 2684d9e | 2012-05-11 01:41:30 +0000 | [diff] [blame] | 88 | |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 89 | OS << ",\""; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 90 | if (Flags & ELF::SHF_ALLOC) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 91 | OS << 'a'; |
Benjamin Kramer | 766f253 | 2013-09-15 19:53:20 +0000 | [diff] [blame] | 92 | if (Flags & ELF::SHF_EXCLUDE) |
| 93 | OS << 'e'; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 94 | if (Flags & ELF::SHF_EXECINSTR) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 95 | OS << 'x'; |
Rafael Espindola | 5d618ef | 2011-02-14 22:23:49 +0000 | [diff] [blame] | 96 | if (Flags & ELF::SHF_GROUP) |
| 97 | OS << 'G'; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 98 | if (Flags & ELF::SHF_WRITE) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 99 | OS << 'w'; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 100 | if (Flags & ELF::SHF_MERGE) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 101 | OS << 'M'; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 102 | if (Flags & ELF::SHF_STRINGS) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 103 | OS << 'S'; |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 104 | if (Flags & ELF::SHF_TLS) |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 105 | OS << 'T'; |
Evgeniy Stepanov | c67a9ef | 2017-03-14 19:28:51 +0000 | [diff] [blame] | 106 | if (Flags & ELF::SHF_LINK_ORDER) |
Evgeniy Stepanov | 8719ab1 | 2017-04-04 22:35:08 +0000 | [diff] [blame] | 107 | OS << 'o'; |
Jim Grosbach | 2684d9e | 2012-05-11 01:41:30 +0000 | [diff] [blame] | 108 | |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 109 | // If there are target-specific flags, print them. |
Rafael Espindola | d5702b7 | 2017-01-30 15:38:43 +0000 | [diff] [blame] | 110 | Triple::ArchType Arch = T.getArch(); |
| 111 | if (Arch == Triple::xcore) { |
| 112 | if (Flags & ELF::XCORE_SHF_CP_SECTION) |
| 113 | OS << 'c'; |
| 114 | if (Flags & ELF::XCORE_SHF_DP_SECTION) |
| 115 | OS << 'd'; |
Florian Hahn | ace970e | 2017-08-12 17:40:18 +0000 | [diff] [blame] | 116 | } else if (T.isARM() || T.isThumb()) { |
Rafael Espindola | d5702b7 | 2017-01-30 15:38:43 +0000 | [diff] [blame] | 117 | if (Flags & ELF::SHF_ARM_PURECODE) |
| 118 | OS << 'y'; |
Krzysztof Parzyszek | 3927975 | 2018-11-09 14:17:27 +0000 | [diff] [blame] | 119 | } else if (Arch == Triple::hexagon) { |
| 120 | if (Flags & ELF::SHF_HEX_GPREL) |
| 121 | OS << 's'; |
Rafael Espindola | d5702b7 | 2017-01-30 15:38:43 +0000 | [diff] [blame] | 122 | } |
Jim Grosbach | 2684d9e | 2012-05-11 01:41:30 +0000 | [diff] [blame] | 123 | |
Chris Lattner | 74aae47 | 2010-04-08 21:26:26 +0000 | [diff] [blame] | 124 | OS << '"'; |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 125 | |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 126 | OS << ','; |
| 127 | |
| 128 | // If comment string is '@', e.g. as on ARM - use '%' instead |
| 129 | if (MAI.getCommentString()[0] == '@') |
| 130 | OS << '%'; |
| 131 | else |
| 132 | OS << '@'; |
| 133 | |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 134 | if (Type == ELF::SHT_INIT_ARRAY) |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 135 | OS << "init_array"; |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 136 | else if (Type == ELF::SHT_FINI_ARRAY) |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 137 | OS << "fini_array"; |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 138 | else if (Type == ELF::SHT_PREINIT_ARRAY) |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 139 | OS << "preinit_array"; |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 140 | else if (Type == ELF::SHT_NOBITS) |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 141 | OS << "nobits"; |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 142 | else if (Type == ELF::SHT_NOTE) |
Rafael Espindola | 9897661 | 2010-12-26 21:30:59 +0000 | [diff] [blame] | 143 | OS << "note"; |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 144 | else if (Type == ELF::SHT_PROGBITS) |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 145 | OS << "progbits"; |
Rafael Espindola | 012d959 | 2015-11-06 15:30:45 +0000 | [diff] [blame] | 146 | else if (Type == ELF::SHT_X86_64_UNWIND) |
| 147 | OS << "unwind"; |
Simon Atanasyan | b479d37 | 2017-03-10 08:22:20 +0000 | [diff] [blame] | 148 | else if (Type == ELF::SHT_MIPS_DWARF) |
| 149 | // Print hex value of the flag while we do not have |
| 150 | // any standard symbolic representation of the flag. |
| 151 | OS << "0x7000001e"; |
Peter Collingbourne | 6bd5b27 | 2017-06-14 18:52:12 +0000 | [diff] [blame] | 152 | else if (Type == ELF::SHT_LLVM_ODRTAB) |
| 153 | OS << "llvm_odrtab"; |
Saleem Abdulrasool | e7676fe | 2018-01-30 16:29:29 +0000 | [diff] [blame] | 154 | else if (Type == ELF::SHT_LLVM_LINKER_OPTIONS) |
| 155 | OS << "llvm_linker_options"; |
Michael J. Spencer | 56500c7 | 2018-06-02 16:33:01 +0000 | [diff] [blame] | 156 | else if (Type == ELF::SHT_LLVM_CALL_GRAPH_PROFILE) |
| 157 | OS << "llvm_call_graph_profile"; |
Simon Atanasyan | d1a4b1d | 2017-03-10 08:22:13 +0000 | [diff] [blame] | 158 | else |
| 159 | report_fatal_error("unsupported type 0x" + Twine::utohexstr(Type) + |
| 160 | " for section " + getSectionName()); |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 161 | |
| 162 | if (EntrySize) { |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 163 | assert(Flags & ELF::SHF_MERGE); |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 164 | OS << "," << EntrySize; |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 165 | } |
Rafael Espindola | 34be396 | 2010-11-09 23:42:07 +0000 | [diff] [blame] | 166 | |
Rafael Espindola | de9a1a2 | 2013-11-13 14:01:59 +0000 | [diff] [blame] | 167 | if (Flags & ELF::SHF_GROUP) { |
| 168 | OS << ","; |
| 169 | printName(OS, Group->getName()); |
| 170 | OS << ",comdat"; |
| 171 | } |
Rafael Espindola | e0a2541 | 2015-02-17 20:48:01 +0000 | [diff] [blame] | 172 | |
Evgeniy Stepanov | c67a9ef | 2017-03-14 19:28:51 +0000 | [diff] [blame] | 173 | if (Flags & ELF::SHF_LINK_ORDER) { |
| 174 | assert(AssociatedSymbol); |
| 175 | OS << ","; |
| 176 | printName(OS, AssociatedSymbol->getName()); |
| 177 | } |
| 178 | |
Rafael Espindola | 903f4a2 | 2015-04-04 18:02:01 +0000 | [diff] [blame] | 179 | if (isUnique()) |
Rafael Espindola | 6ba6e55 | 2015-04-06 16:34:41 +0000 | [diff] [blame] | 180 | OS << ",unique," << UniqueID; |
Rafael Espindola | e0a2541 | 2015-02-17 20:48:01 +0000 | [diff] [blame] | 181 | |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 182 | OS << '\n'; |
Peter Collingbourne | df39be6 | 2013-04-17 21:18:16 +0000 | [diff] [blame] | 183 | |
Matt Arsenault | d99ce2f | 2015-06-09 00:31:39 +0000 | [diff] [blame] | 184 | if (Subsection) { |
| 185 | OS << "\t.subsection\t"; |
| 186 | Subsection->print(OS, &MAI); |
| 187 | OS << '\n'; |
| 188 | } |
Bruno Cardoso Lopes | b808588 | 2009-08-13 05:07:35 +0000 | [diff] [blame] | 189 | } |
| 190 | |
Jan Wen Voung | 083cf15 | 2010-10-04 17:32:41 +0000 | [diff] [blame] | 191 | bool MCSectionELF::UseCodeAlign() const { |
Rafael Espindola | 1c13026 | 2011-01-23 04:43:11 +0000 | [diff] [blame] | 192 | return getFlags() & ELF::SHF_EXECINSTR; |
Jan Wen Voung | 083cf15 | 2010-10-04 17:32:41 +0000 | [diff] [blame] | 193 | } |
| 194 | |
Rafael Espindola | f2dc4aa | 2010-11-17 20:03:54 +0000 | [diff] [blame] | 195 | bool MCSectionELF::isVirtualSection() const { |
Rafael Espindola | c85dca6 | 2011-01-23 04:28:49 +0000 | [diff] [blame] | 196 | return getType() == ELF::SHT_NOBITS; |
Rafael Espindola | f2dc4aa | 2010-11-17 20:03:54 +0000 | [diff] [blame] | 197 | } |