1 //===- lib/MC/MCStreamer.cpp - Streaming Machine Code Output --------------===// 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 10 #include "llvm/MC/MCStreamer.h" 11 #include "llvm/ADT/Optional.h" 12 #include "llvm/ADT/SmallString.h" 13 #include "llvm/ADT/StringRef.h" 14 #include "llvm/ADT/Twine.h" 15 #include "llvm/BinaryFormat/COFF.h" 16 #include "llvm/MC/MCAsmBackend.h" 17 #include "llvm/MC/MCAsmInfo.h" 18 #include "llvm/MC/MCCodeView.h" 19 #include "llvm/MC/MCContext.h" 20 #include "llvm/MC/MCDwarf.h" 21 #include "llvm/MC/MCExpr.h" 22 #include "llvm/MC/MCInst.h" 23 #include "llvm/MC/MCInstPrinter.h" 24 #include "llvm/MC/MCObjectFileInfo.h" 25 #include "llvm/MC/MCSection.h" 26 #include "llvm/MC/MCSectionCOFF.h" 27 #include "llvm/MC/MCSymbol.h" 28 #include "llvm/MC/MCWin64EH.h" 29 #include "llvm/MC/MCWinEH.h" 30 #include "llvm/Support/Casting.h" 31 #include "llvm/Support/ErrorHandling.h" 32 #include "llvm/Support/LEB128.h" 33 #include "llvm/Support/MathExtras.h" 34 #include "llvm/Support/raw_ostream.h" 35 #include <cassert> 36 #include <cstdint> 37 #include <cstdlib> 38 #include <utility> 39 40 using namespace llvm; 41 42 MCTargetStreamer::MCTargetStreamer(MCStreamer &S) : Streamer(S) { 43 S.setTargetStreamer(this); 44 } 45 46 // Pin the vtables to this file. 47 MCTargetStreamer::~MCTargetStreamer() = default; 48 49 void MCTargetStreamer::emitLabel(MCSymbol *Symbol) {} 50 51 void MCTargetStreamer::finish() {} 52 53 void MCTargetStreamer::changeSection(const MCSection *CurSection, 54 MCSection *Section, 55 const MCExpr *Subsection, 56 raw_ostream &OS) { 57 Section->PrintSwitchToSection( 58 *Streamer.getContext().getAsmInfo(), 59 Streamer.getContext().getObjectFileInfo()->getTargetTriple(), OS, 60 Subsection); 61 } 62 63 void MCTargetStreamer::emitDwarfFileDirective(StringRef Directive) { 64 Streamer.EmitRawText(Directive); 65 } 66 67 void MCTargetStreamer::emitValue(const MCExpr *Value) { 68 SmallString<128> Str; 69 raw_svector_ostream OS(Str); 70 71 Value->print(OS, Streamer.getContext().getAsmInfo()); 72 Streamer.EmitRawText(OS.str()); 73 } 74 75 void MCTargetStreamer::emitAssignment(MCSymbol *Symbol, const MCExpr *Value) {} 76 77 MCStreamer::MCStreamer(MCContext &Ctx) 78 : Context(Ctx), CurrentWinFrameInfo(nullptr), 79 UseAssemblerInfoForParsing(false) { 80 SectionStack.push_back(std::pair<MCSectionSubPair, MCSectionSubPair>()); 81 } 82 83 MCStreamer::~MCStreamer() {} 84 85 void MCStreamer::reset() { 86 DwarfFrameInfos.clear(); 87 CurrentWinFrameInfo = nullptr; 88 WinFrameInfos.clear(); 89 SymbolOrdering.clear(); 90 SectionStack.clear(); 91 SectionStack.push_back(std::pair<MCSectionSubPair, MCSectionSubPair>()); 92 } 93 94 raw_ostream &MCStreamer::GetCommentOS() { 95 // By default, discard comments. 96 return nulls(); 97 } 98 99 void MCStreamer::emitRawComment(const Twine &T, bool TabPrefix) {} 100 101 void MCStreamer::addExplicitComment(const Twine &T) {} 102 void MCStreamer::emitExplicitComments() {} 103 104 void MCStreamer::generateCompactUnwindEncodings(MCAsmBackend *MAB) { 105 for (auto &FI : DwarfFrameInfos) 106 FI.CompactUnwindEncoding = 107 (MAB ? MAB->generateCompactUnwindEncoding(FI.Instructions) : 0); 108 } 109 110 /// EmitIntValue - Special case of EmitValue that avoids the client having to 111 /// pass in a MCExpr for constant integers. 112 void MCStreamer::EmitIntValue(uint64_t Value, unsigned Size) { 113 assert(1 <= Size && Size <= 8 && "Invalid size"); 114 assert((isUIntN(8 * Size, Value) || isIntN(8 * Size, Value)) && 115 "Invalid size"); 116 char buf[8]; 117 const bool isLittleEndian = Context.getAsmInfo()->isLittleEndian(); 118 for (unsigned i = 0; i != Size; ++i) { 119 unsigned index = isLittleEndian ? i : (Size - i - 1); 120 buf[i] = uint8_t(Value >> (index * 8)); 121 } 122 EmitBytes(StringRef(buf, Size)); 123 } 124 125 /// EmitULEB128IntValue - Special case of EmitULEB128Value that avoids the 126 /// client having to pass in a MCExpr for constant integers. 127 void MCStreamer::EmitULEB128IntValue(uint64_t Value) { 128 SmallString<128> Tmp; 129 raw_svector_ostream OSE(Tmp); 130 encodeULEB128(Value, OSE); 131 EmitBytes(OSE.str()); 132 } 133 134 /// EmitSLEB128IntValue - Special case of EmitSLEB128Value that avoids the 135 /// client having to pass in a MCExpr for constant integers. 136 void MCStreamer::EmitSLEB128IntValue(int64_t Value) { 137 SmallString<128> Tmp; 138 raw_svector_ostream OSE(Tmp); 139 encodeSLEB128(Value, OSE); 140 EmitBytes(OSE.str()); 141 } 142 143 void MCStreamer::EmitValue(const MCExpr *Value, unsigned Size, SMLoc Loc) { 144 EmitValueImpl(Value, Size, Loc); 145 } 146 147 void MCStreamer::EmitSymbolValue(const MCSymbol *Sym, unsigned Size, 148 bool IsSectionRelative) { 149 assert((!IsSectionRelative || Size == 4) && 150 "SectionRelative value requires 4-bytes"); 151 152 if (!IsSectionRelative) 153 EmitValueImpl(MCSymbolRefExpr::create(Sym, getContext()), Size); 154 else 155 EmitCOFFSecRel32(Sym, /*Offset=*/0); 156 } 157 158 void MCStreamer::EmitDTPRel64Value(const MCExpr *Value) { 159 report_fatal_error("unsupported directive in streamer"); 160 } 161 162 void MCStreamer::EmitDTPRel32Value(const MCExpr *Value) { 163 report_fatal_error("unsupported directive in streamer"); 164 } 165 166 void MCStreamer::EmitTPRel64Value(const MCExpr *Value) { 167 report_fatal_error("unsupported directive in streamer"); 168 } 169 170 void MCStreamer::EmitTPRel32Value(const MCExpr *Value) { 171 report_fatal_error("unsupported directive in streamer"); 172 } 173 174 void MCStreamer::EmitGPRel64Value(const MCExpr *Value) { 175 report_fatal_error("unsupported directive in streamer"); 176 } 177 178 void MCStreamer::EmitGPRel32Value(const MCExpr *Value) { 179 report_fatal_error("unsupported directive in streamer"); 180 } 181 182 /// Emit NumBytes bytes worth of the value specified by FillValue. 183 /// This implements directives such as '.space'. 184 void MCStreamer::emitFill(uint64_t NumBytes, uint8_t FillValue) { 185 emitFill(*MCConstantExpr::create(NumBytes, getContext()), FillValue); 186 } 187 188 /// The implementation in this class just redirects to emitFill. 189 void MCStreamer::EmitZeros(uint64_t NumBytes) { 190 emitFill(NumBytes, 0); 191 } 192 193 Expected<unsigned> 194 MCStreamer::tryEmitDwarfFileDirective(unsigned FileNo, StringRef Directory, 195 StringRef Filename, 196 MD5::MD5Result *Checksum, 197 Optional<StringRef> Source, 198 unsigned CUID) { 199 return getContext().getDwarfFile(Directory, Filename, FileNo, Checksum, 200 Source, CUID); 201 } 202 203 void MCStreamer::emitDwarfFile0Directive(StringRef Directory, 204 StringRef Filename, 205 MD5::MD5Result *Checksum, 206 Optional<StringRef> Source, 207 unsigned CUID) { 208 getContext().setMCLineTableRootFile(CUID, Directory, Filename, Checksum, 209 Source); 210 } 211 212 void MCStreamer::EmitDwarfLocDirective(unsigned FileNo, unsigned Line, 213 unsigned Column, unsigned Flags, 214 unsigned Isa, 215 unsigned Discriminator, 216 StringRef FileName) { 217 getContext().setCurrentDwarfLoc(FileNo, Line, Column, Flags, Isa, 218 Discriminator); 219 } 220 221 MCSymbol *MCStreamer::getDwarfLineTableSymbol(unsigned CUID) { 222 MCDwarfLineTable &Table = getContext().getMCDwarfLineTable(CUID); 223 if (!Table.getLabel()) { 224 StringRef Prefix = Context.getAsmInfo()->getPrivateGlobalPrefix(); 225 Table.setLabel( 226 Context.getOrCreateSymbol(Prefix + "line_table_start" + Twine(CUID))); 227 } 228 return Table.getLabel(); 229 } 230 231 bool MCStreamer::hasUnfinishedDwarfFrameInfo() { 232 return !DwarfFrameInfos.empty() && !DwarfFrameInfos.back().End; 233 } 234 235 MCDwarfFrameInfo *MCStreamer::getCurrentDwarfFrameInfo() { 236 if (!hasUnfinishedDwarfFrameInfo()) { 237 getContext().reportError(SMLoc(), "this directive must appear between " 238 ".cfi_startproc and .cfi_endproc " 239 "directives"); 240 return nullptr; 241 } 242 return &DwarfFrameInfos.back(); 243 } 244 245 bool MCStreamer::EmitCVFileDirective(unsigned FileNo, StringRef Filename, 246 ArrayRef<uint8_t> Checksum, 247 unsigned ChecksumKind) { 248 return getContext().getCVContext().addFile(*this, FileNo, Filename, Checksum, 249 ChecksumKind); 250 } 251 252 bool MCStreamer::EmitCVFuncIdDirective(unsigned FunctionId) { 253 return getContext().getCVContext().recordFunctionId(FunctionId); 254 } 255 256 bool MCStreamer::EmitCVInlineSiteIdDirective(unsigned FunctionId, 257 unsigned IAFunc, unsigned IAFile, 258 unsigned IALine, unsigned IACol, 259 SMLoc Loc) { 260 if (getContext().getCVContext().getCVFunctionInfo(IAFunc) == nullptr) { 261 getContext().reportError(Loc, "parent function id not introduced by " 262 ".cv_func_id or .cv_inline_site_id"); 263 return true; 264 } 265 266 return getContext().getCVContext().recordInlinedCallSiteId( 267 FunctionId, IAFunc, IAFile, IALine, IACol); 268 } 269 270 void MCStreamer::EmitCVLocDirective(unsigned FunctionId, unsigned FileNo, 271 unsigned Line, unsigned Column, 272 bool PrologueEnd, bool IsStmt, 273 StringRef FileName, SMLoc Loc) { 274 CodeViewContext &CVC = getContext().getCVContext(); 275 MCCVFunctionInfo *FI = CVC.getCVFunctionInfo(FunctionId); 276 if (!FI) 277 return getContext().reportError( 278 Loc, "function id not introduced by .cv_func_id or .cv_inline_site_id"); 279 280 // Track the section 281 if (FI->Section == nullptr) 282 FI->Section = getCurrentSectionOnly(); 283 else if (FI->Section != getCurrentSectionOnly()) 284 return getContext().reportError( 285 Loc, 286 "all .cv_loc directives for a function must be in the same section"); 287 288 CVC.setCurrentCVLoc(FunctionId, FileNo, Line, Column, PrologueEnd, IsStmt); 289 } 290 291 void MCStreamer::EmitCVLinetableDirective(unsigned FunctionId, 292 const MCSymbol *Begin, 293 const MCSymbol *End) {} 294 295 void MCStreamer::EmitCVInlineLinetableDirective(unsigned PrimaryFunctionId, 296 unsigned SourceFileId, 297 unsigned SourceLineNum, 298 const MCSymbol *FnStartSym, 299 const MCSymbol *FnEndSym) {} 300 301 void MCStreamer::EmitCVDefRangeDirective( 302 ArrayRef<std::pair<const MCSymbol *, const MCSymbol *>> Ranges, 303 StringRef FixedSizePortion) {} 304 305 void MCStreamer::EmitEHSymAttributes(const MCSymbol *Symbol, 306 MCSymbol *EHSymbol) { 307 } 308 309 void MCStreamer::InitSections(bool NoExecStack) { 310 SwitchSection(getContext().getObjectFileInfo()->getTextSection()); 311 } 312 313 void MCStreamer::AssignFragment(MCSymbol *Symbol, MCFragment *Fragment) { 314 assert(Fragment); 315 Symbol->setFragment(Fragment); 316 317 // As we emit symbols into a section, track the order so that they can 318 // be sorted upon later. Zero is reserved to mean 'unemitted'. 319 SymbolOrdering[Symbol] = 1 + SymbolOrdering.size(); 320 } 321 322 void MCStreamer::EmitLabel(MCSymbol *Symbol, SMLoc Loc) { 323 Symbol->redefineIfPossible(); 324 325 if (!Symbol->isUndefined() || Symbol->isVariable()) 326 return getContext().reportError(Loc, "invalid symbol redefinition"); 327 328 assert(!Symbol->isVariable() && "Cannot emit a variable symbol!"); 329 assert(getCurrentSectionOnly() && "Cannot emit before setting section!"); 330 assert(!Symbol->getFragment() && "Unexpected fragment on symbol data!"); 331 assert(Symbol->isUndefined() && "Cannot define a symbol twice!"); 332 333 Symbol->setFragment(&getCurrentSectionOnly()->getDummyFragment()); 334 335 MCTargetStreamer *TS = getTargetStreamer(); 336 if (TS) 337 TS->emitLabel(Symbol); 338 } 339 340 void MCStreamer::EmitCFISections(bool EH, bool Debug) { 341 assert(EH || Debug); 342 } 343 344 void MCStreamer::EmitCFIStartProc(bool IsSimple) { 345 if (hasUnfinishedDwarfFrameInfo()) 346 getContext().reportError( 347 SMLoc(), "starting new .cfi frame before finishing the previous one"); 348 349 MCDwarfFrameInfo Frame; 350 Frame.IsSimple = IsSimple; 351 EmitCFIStartProcImpl(Frame); 352 353 const MCAsmInfo* MAI = Context.getAsmInfo(); 354 if (MAI) { 355 for (const MCCFIInstruction& Inst : MAI->getInitialFrameState()) { 356 if (Inst.getOperation() == MCCFIInstruction::OpDefCfa || 357 Inst.getOperation() == MCCFIInstruction::OpDefCfaRegister) { 358 Frame.CurrentCfaRegister = Inst.getRegister(); 359 } 360 } 361 } 362 363 DwarfFrameInfos.push_back(Frame); 364 } 365 366 void MCStreamer::EmitCFIStartProcImpl(MCDwarfFrameInfo &Frame) { 367 } 368 369 void MCStreamer::EmitCFIEndProc() { 370 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 371 if (!CurFrame) 372 return; 373 EmitCFIEndProcImpl(*CurFrame); 374 } 375 376 void MCStreamer::EmitCFIEndProcImpl(MCDwarfFrameInfo &Frame) { 377 // Put a dummy non-null value in Frame.End to mark that this frame has been 378 // closed. 379 Frame.End = (MCSymbol *)1; 380 } 381 382 MCSymbol *MCStreamer::EmitCFILabel() { 383 // Return a dummy non-null value so that label fields appear filled in when 384 // generating textual assembly. 385 return (MCSymbol *)1; 386 } 387 388 void MCStreamer::EmitCFIDefCfa(int64_t Register, int64_t Offset) { 389 MCSymbol *Label = EmitCFILabel(); 390 MCCFIInstruction Instruction = 391 MCCFIInstruction::createDefCfa(Label, Register, Offset); 392 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 393 if (!CurFrame) 394 return; 395 CurFrame->Instructions.push_back(Instruction); 396 CurFrame->CurrentCfaRegister = static_cast<unsigned>(Register); 397 } 398 399 void MCStreamer::EmitCFIDefCfaOffset(int64_t Offset) { 400 MCSymbol *Label = EmitCFILabel(); 401 MCCFIInstruction Instruction = 402 MCCFIInstruction::createDefCfaOffset(Label, Offset); 403 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 404 if (!CurFrame) 405 return; 406 CurFrame->Instructions.push_back(Instruction); 407 } 408 409 void MCStreamer::EmitCFIAdjustCfaOffset(int64_t Adjustment) { 410 MCSymbol *Label = EmitCFILabel(); 411 MCCFIInstruction Instruction = 412 MCCFIInstruction::createAdjustCfaOffset(Label, Adjustment); 413 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 414 if (!CurFrame) 415 return; 416 CurFrame->Instructions.push_back(Instruction); 417 } 418 419 void MCStreamer::EmitCFIDefCfaRegister(int64_t Register) { 420 MCSymbol *Label = EmitCFILabel(); 421 MCCFIInstruction Instruction = 422 MCCFIInstruction::createDefCfaRegister(Label, Register); 423 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 424 if (!CurFrame) 425 return; 426 CurFrame->Instructions.push_back(Instruction); 427 CurFrame->CurrentCfaRegister = static_cast<unsigned>(Register); 428 } 429 430 void MCStreamer::EmitCFIOffset(int64_t Register, int64_t Offset) { 431 MCSymbol *Label = EmitCFILabel(); 432 MCCFIInstruction Instruction = 433 MCCFIInstruction::createOffset(Label, Register, Offset); 434 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 435 if (!CurFrame) 436 return; 437 CurFrame->Instructions.push_back(Instruction); 438 } 439 440 void MCStreamer::EmitCFIRelOffset(int64_t Register, int64_t Offset) { 441 MCSymbol *Label = EmitCFILabel(); 442 MCCFIInstruction Instruction = 443 MCCFIInstruction::createRelOffset(Label, Register, Offset); 444 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 445 if (!CurFrame) 446 return; 447 CurFrame->Instructions.push_back(Instruction); 448 } 449 450 void MCStreamer::EmitCFIPersonality(const MCSymbol *Sym, 451 unsigned Encoding) { 452 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 453 if (!CurFrame) 454 return; 455 CurFrame->Personality = Sym; 456 CurFrame->PersonalityEncoding = Encoding; 457 } 458 459 void MCStreamer::EmitCFILsda(const MCSymbol *Sym, unsigned Encoding) { 460 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 461 if (!CurFrame) 462 return; 463 CurFrame->Lsda = Sym; 464 CurFrame->LsdaEncoding = Encoding; 465 } 466 467 void MCStreamer::EmitCFIRememberState() { 468 MCSymbol *Label = EmitCFILabel(); 469 MCCFIInstruction Instruction = MCCFIInstruction::createRememberState(Label); 470 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 471 if (!CurFrame) 472 return; 473 CurFrame->Instructions.push_back(Instruction); 474 } 475 476 void MCStreamer::EmitCFIRestoreState() { 477 // FIXME: Error if there is no matching cfi_remember_state. 478 MCSymbol *Label = EmitCFILabel(); 479 MCCFIInstruction Instruction = MCCFIInstruction::createRestoreState(Label); 480 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 481 if (!CurFrame) 482 return; 483 CurFrame->Instructions.push_back(Instruction); 484 } 485 486 void MCStreamer::EmitCFISameValue(int64_t Register) { 487 MCSymbol *Label = EmitCFILabel(); 488 MCCFIInstruction Instruction = 489 MCCFIInstruction::createSameValue(Label, Register); 490 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 491 if (!CurFrame) 492 return; 493 CurFrame->Instructions.push_back(Instruction); 494 } 495 496 void MCStreamer::EmitCFIRestore(int64_t Register) { 497 MCSymbol *Label = EmitCFILabel(); 498 MCCFIInstruction Instruction = 499 MCCFIInstruction::createRestore(Label, Register); 500 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 501 if (!CurFrame) 502 return; 503 CurFrame->Instructions.push_back(Instruction); 504 } 505 506 void MCStreamer::EmitCFIEscape(StringRef Values) { 507 MCSymbol *Label = EmitCFILabel(); 508 MCCFIInstruction Instruction = MCCFIInstruction::createEscape(Label, Values); 509 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 510 if (!CurFrame) 511 return; 512 CurFrame->Instructions.push_back(Instruction); 513 } 514 515 void MCStreamer::EmitCFIGnuArgsSize(int64_t Size) { 516 MCSymbol *Label = EmitCFILabel(); 517 MCCFIInstruction Instruction = 518 MCCFIInstruction::createGnuArgsSize(Label, Size); 519 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 520 if (!CurFrame) 521 return; 522 CurFrame->Instructions.push_back(Instruction); 523 } 524 525 void MCStreamer::EmitCFISignalFrame() { 526 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 527 if (!CurFrame) 528 return; 529 CurFrame->IsSignalFrame = true; 530 } 531 532 void MCStreamer::EmitCFIUndefined(int64_t Register) { 533 MCSymbol *Label = EmitCFILabel(); 534 MCCFIInstruction Instruction = 535 MCCFIInstruction::createUndefined(Label, Register); 536 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 537 if (!CurFrame) 538 return; 539 CurFrame->Instructions.push_back(Instruction); 540 } 541 542 void MCStreamer::EmitCFIRegister(int64_t Register1, int64_t Register2) { 543 MCSymbol *Label = EmitCFILabel(); 544 MCCFIInstruction Instruction = 545 MCCFIInstruction::createRegister(Label, Register1, Register2); 546 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 547 if (!CurFrame) 548 return; 549 CurFrame->Instructions.push_back(Instruction); 550 } 551 552 void MCStreamer::EmitCFIWindowSave() { 553 MCSymbol *Label = EmitCFILabel(); 554 MCCFIInstruction Instruction = 555 MCCFIInstruction::createWindowSave(Label); 556 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 557 if (!CurFrame) 558 return; 559 CurFrame->Instructions.push_back(Instruction); 560 } 561 562 void MCStreamer::EmitCFIReturnColumn(int64_t Register) { 563 MCDwarfFrameInfo *CurFrame = getCurrentDwarfFrameInfo(); 564 if (!CurFrame) 565 return; 566 CurFrame->RAReg = Register; 567 } 568 569 WinEH::FrameInfo *MCStreamer::EnsureValidWinFrameInfo(SMLoc Loc) { 570 const MCAsmInfo *MAI = Context.getAsmInfo(); 571 if (!MAI->usesWindowsCFI()) { 572 getContext().reportError( 573 Loc, ".seh_* directives are not supported on this target"); 574 return nullptr; 575 } 576 if (!CurrentWinFrameInfo || CurrentWinFrameInfo->End) { 577 getContext().reportError( 578 Loc, ".seh_ directive must appear within an active frame"); 579 return nullptr; 580 } 581 return CurrentWinFrameInfo; 582 } 583 584 void MCStreamer::EmitWinCFIStartProc(const MCSymbol *Symbol, SMLoc Loc) { 585 const MCAsmInfo *MAI = Context.getAsmInfo(); 586 if (!MAI->usesWindowsCFI()) 587 return getContext().reportError( 588 Loc, ".seh_* directives are not supported on this target"); 589 if (CurrentWinFrameInfo && !CurrentWinFrameInfo->End) 590 getContext().reportError( 591 Loc, "Starting a function before ending the previous one!"); 592 593 MCSymbol *StartProc = EmitCFILabel(); 594 595 WinFrameInfos.emplace_back( 596 llvm::make_unique<WinEH::FrameInfo>(Symbol, StartProc)); 597 CurrentWinFrameInfo = WinFrameInfos.back().get(); 598 CurrentWinFrameInfo->TextSection = getCurrentSectionOnly(); 599 } 600 601 void MCStreamer::EmitWinCFIEndProc(SMLoc Loc) { 602 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 603 if (!CurFrame) 604 return; 605 if (CurFrame->ChainedParent) 606 getContext().reportError(Loc, "Not all chained regions terminated!"); 607 608 MCSymbol *Label = EmitCFILabel(); 609 CurFrame->End = Label; 610 } 611 612 void MCStreamer::EmitWinCFIStartChained(SMLoc Loc) { 613 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 614 if (!CurFrame) 615 return; 616 617 MCSymbol *StartProc = EmitCFILabel(); 618 619 WinFrameInfos.emplace_back(llvm::make_unique<WinEH::FrameInfo>( 620 CurFrame->Function, StartProc, CurFrame)); 621 CurrentWinFrameInfo = WinFrameInfos.back().get(); 622 CurrentWinFrameInfo->TextSection = getCurrentSectionOnly(); 623 } 624 625 void MCStreamer::EmitWinCFIEndChained(SMLoc Loc) { 626 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 627 if (!CurFrame) 628 return; 629 if (!CurFrame->ChainedParent) 630 return getContext().reportError( 631 Loc, "End of a chained region outside a chained region!"); 632 633 MCSymbol *Label = EmitCFILabel(); 634 635 CurFrame->End = Label; 636 CurrentWinFrameInfo = const_cast<WinEH::FrameInfo *>(CurFrame->ChainedParent); 637 } 638 639 void MCStreamer::EmitWinEHHandler(const MCSymbol *Sym, bool Unwind, bool Except, 640 SMLoc Loc) { 641 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 642 if (!CurFrame) 643 return; 644 if (CurFrame->ChainedParent) 645 return getContext().reportError( 646 Loc, "Chained unwind areas can't have handlers!"); 647 CurFrame->ExceptionHandler = Sym; 648 if (!Except && !Unwind) 649 getContext().reportError(Loc, "Don't know what kind of handler this is!"); 650 if (Unwind) 651 CurFrame->HandlesUnwind = true; 652 if (Except) 653 CurFrame->HandlesExceptions = true; 654 } 655 656 void MCStreamer::EmitWinEHHandlerData(SMLoc Loc) { 657 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 658 if (!CurFrame) 659 return; 660 if (CurFrame->ChainedParent) 661 getContext().reportError(Loc, "Chained unwind areas can't have handlers!"); 662 } 663 664 void MCStreamer::emitCGProfileEntry(const MCSymbolRefExpr *From, 665 const MCSymbolRefExpr *To, uint64_t Count) { 666 } 667 668 static MCSection *getWinCFISection(MCContext &Context, unsigned *NextWinCFIID, 669 MCSection *MainCFISec, 670 const MCSection *TextSec) { 671 // If this is the main .text section, use the main unwind info section. 672 if (TextSec == Context.getObjectFileInfo()->getTextSection()) 673 return MainCFISec; 674 675 const auto *TextSecCOFF = cast<MCSectionCOFF>(TextSec); 676 auto *MainCFISecCOFF = cast<MCSectionCOFF>(MainCFISec); 677 unsigned UniqueID = TextSecCOFF->getOrAssignWinCFISectionID(NextWinCFIID); 678 679 // If this section is COMDAT, this unwind section should be COMDAT associative 680 // with its group. 681 const MCSymbol *KeySym = nullptr; 682 if (TextSecCOFF->getCharacteristics() & COFF::IMAGE_SCN_LNK_COMDAT) { 683 KeySym = TextSecCOFF->getCOMDATSymbol(); 684 685 // In a GNU environment, we can't use associative comdats. Instead, do what 686 // GCC does, which is to make plain comdat selectany section named like 687 // ".[px]data$_Z3foov". 688 if (!Context.getAsmInfo()->hasCOFFAssociativeComdats()) { 689 std::string SectionName = 690 (MainCFISecCOFF->getSectionName() + "$" + 691 TextSecCOFF->getSectionName().split('$').second) 692 .str(); 693 return Context.getCOFFSection( 694 SectionName, 695 MainCFISecCOFF->getCharacteristics() | COFF::IMAGE_SCN_LNK_COMDAT, 696 MainCFISecCOFF->getKind(), "", COFF::IMAGE_COMDAT_SELECT_ANY); 697 } 698 } 699 700 return Context.getAssociativeCOFFSection(MainCFISecCOFF, KeySym, UniqueID); 701 } 702 703 MCSection *MCStreamer::getAssociatedPDataSection(const MCSection *TextSec) { 704 return getWinCFISection(getContext(), &NextWinCFIID, 705 getContext().getObjectFileInfo()->getPDataSection(), 706 TextSec); 707 } 708 709 MCSection *MCStreamer::getAssociatedXDataSection(const MCSection *TextSec) { 710 return getWinCFISection(getContext(), &NextWinCFIID, 711 getContext().getObjectFileInfo()->getXDataSection(), 712 TextSec); 713 } 714 715 void MCStreamer::EmitSyntaxDirective() {} 716 717 void MCStreamer::EmitWinCFIPushReg(unsigned Register, SMLoc Loc) { 718 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 719 if (!CurFrame) 720 return; 721 722 MCSymbol *Label = EmitCFILabel(); 723 724 WinEH::Instruction Inst = Win64EH::Instruction::PushNonVol(Label, Register); 725 CurFrame->Instructions.push_back(Inst); 726 } 727 728 void MCStreamer::EmitWinCFISetFrame(unsigned Register, unsigned Offset, 729 SMLoc Loc) { 730 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 731 if (!CurFrame) 732 return; 733 if (CurFrame->LastFrameInst >= 0) 734 return getContext().reportError( 735 Loc, "frame register and offset can be set at most once"); 736 if (Offset & 0x0F) 737 return getContext().reportError(Loc, "offset is not a multiple of 16"); 738 if (Offset > 240) 739 return getContext().reportError( 740 Loc, "frame offset must be less than or equal to 240"); 741 742 MCSymbol *Label = EmitCFILabel(); 743 744 WinEH::Instruction Inst = 745 Win64EH::Instruction::SetFPReg(Label, Register, Offset); 746 CurFrame->LastFrameInst = CurFrame->Instructions.size(); 747 CurFrame->Instructions.push_back(Inst); 748 } 749 750 void MCStreamer::EmitWinCFIAllocStack(unsigned Size, SMLoc Loc) { 751 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 752 if (!CurFrame) 753 return; 754 if (Size == 0) 755 return getContext().reportError(Loc, 756 "stack allocation size must be non-zero"); 757 if (Size & 7) 758 return getContext().reportError( 759 Loc, "stack allocation size is not a multiple of 8"); 760 761 MCSymbol *Label = EmitCFILabel(); 762 763 WinEH::Instruction Inst = Win64EH::Instruction::Alloc(Label, Size); 764 CurFrame->Instructions.push_back(Inst); 765 } 766 767 void MCStreamer::EmitWinCFISaveReg(unsigned Register, unsigned Offset, 768 SMLoc Loc) { 769 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 770 if (!CurFrame) 771 return; 772 773 if (Offset & 7) 774 return getContext().reportError( 775 Loc, "register save offset is not 8 byte aligned"); 776 777 MCSymbol *Label = EmitCFILabel(); 778 779 WinEH::Instruction Inst = 780 Win64EH::Instruction::SaveNonVol(Label, Register, Offset); 781 CurFrame->Instructions.push_back(Inst); 782 } 783 784 void MCStreamer::EmitWinCFISaveXMM(unsigned Register, unsigned Offset, 785 SMLoc Loc) { 786 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 787 if (!CurFrame) 788 return; 789 if (Offset & 0x0F) 790 return getContext().reportError(Loc, "offset is not a multiple of 16"); 791 792 MCSymbol *Label = EmitCFILabel(); 793 794 WinEH::Instruction Inst = 795 Win64EH::Instruction::SaveXMM(Label, Register, Offset); 796 CurFrame->Instructions.push_back(Inst); 797 } 798 799 void MCStreamer::EmitWinCFIPushFrame(bool Code, SMLoc Loc) { 800 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 801 if (!CurFrame) 802 return; 803 if (!CurFrame->Instructions.empty()) 804 return getContext().reportError( 805 Loc, "If present, PushMachFrame must be the first UOP"); 806 807 MCSymbol *Label = EmitCFILabel(); 808 809 WinEH::Instruction Inst = Win64EH::Instruction::PushMachFrame(Label, Code); 810 CurFrame->Instructions.push_back(Inst); 811 } 812 813 void MCStreamer::EmitWinCFIEndProlog(SMLoc Loc) { 814 WinEH::FrameInfo *CurFrame = EnsureValidWinFrameInfo(Loc); 815 if (!CurFrame) 816 return; 817 818 MCSymbol *Label = EmitCFILabel(); 819 820 CurFrame->PrologEnd = Label; 821 } 822 823 void MCStreamer::EmitCOFFSafeSEH(MCSymbol const *Symbol) { 824 } 825 826 void MCStreamer::EmitCOFFSymbolIndex(MCSymbol const *Symbol) {} 827 828 void MCStreamer::EmitCOFFSectionIndex(MCSymbol const *Symbol) { 829 } 830 831 void MCStreamer::EmitCOFFSecRel32(MCSymbol const *Symbol, uint64_t Offset) {} 832 833 void MCStreamer::EmitCOFFImgRel32(MCSymbol const *Symbol, int64_t Offset) {} 834 835 /// EmitRawText - If this file is backed by an assembly streamer, this dumps 836 /// the specified string in the output .s file. This capability is 837 /// indicated by the hasRawTextSupport() predicate. 838 void MCStreamer::EmitRawTextImpl(StringRef String) { 839 errs() << "EmitRawText called on an MCStreamer that doesn't support it, " 840 " something must not be fully mc'ized\n"; 841 abort(); 842 } 843 844 void MCStreamer::EmitRawText(const Twine &T) { 845 SmallString<128> Str; 846 EmitRawTextImpl(T.toStringRef(Str)); 847 } 848 849 void MCStreamer::EmitWindowsUnwindTables() { 850 } 851 852 void MCStreamer::Finish() { 853 if ((!DwarfFrameInfos.empty() && !DwarfFrameInfos.back().End) || 854 (!WinFrameInfos.empty() && !WinFrameInfos.back()->End)) { 855 getContext().reportError(SMLoc(), "Unfinished frame!"); 856 return; 857 } 858 859 MCTargetStreamer *TS = getTargetStreamer(); 860 if (TS) 861 TS->finish(); 862 863 FinishImpl(); 864 } 865 866 void MCStreamer::EmitAssignment(MCSymbol *Symbol, const MCExpr *Value) { 867 visitUsedExpr(*Value); 868 Symbol->setVariableValue(Value); 869 870 MCTargetStreamer *TS = getTargetStreamer(); 871 if (TS) 872 TS->emitAssignment(Symbol, Value); 873 } 874 875 void MCTargetStreamer::prettyPrintAsm(MCInstPrinter &InstPrinter, raw_ostream &OS, 876 const MCInst &Inst, const MCSubtargetInfo &STI) { 877 InstPrinter.printInst(&Inst, OS, "", STI); 878 } 879 880 void MCStreamer::visitUsedSymbol(const MCSymbol &Sym) { 881 } 882 883 void MCStreamer::visitUsedExpr(const MCExpr &Expr) { 884 switch (Expr.getKind()) { 885 case MCExpr::Target: 886 cast<MCTargetExpr>(Expr).visitUsedExpr(*this); 887 break; 888 889 case MCExpr::Constant: 890 break; 891 892 case MCExpr::Binary: { 893 const MCBinaryExpr &BE = cast<MCBinaryExpr>(Expr); 894 visitUsedExpr(*BE.getLHS()); 895 visitUsedExpr(*BE.getRHS()); 896 break; 897 } 898 899 case MCExpr::SymbolRef: 900 visitUsedSymbol(cast<MCSymbolRefExpr>(Expr).getSymbol()); 901 break; 902 903 case MCExpr::Unary: 904 visitUsedExpr(*cast<MCUnaryExpr>(Expr).getSubExpr()); 905 break; 906 } 907 } 908 909 void MCStreamer::EmitInstruction(const MCInst &Inst, const MCSubtargetInfo &STI, 910 bool) { 911 // Scan for values. 912 for (unsigned i = Inst.getNumOperands(); i--;) 913 if (Inst.getOperand(i).isExpr()) 914 visitUsedExpr(*Inst.getOperand(i).getExpr()); 915 } 916 917 void MCStreamer::emitAbsoluteSymbolDiff(const MCSymbol *Hi, const MCSymbol *Lo, 918 unsigned Size) { 919 // Get the Hi-Lo expression. 920 const MCExpr *Diff = 921 MCBinaryExpr::createSub(MCSymbolRefExpr::create(Hi, Context), 922 MCSymbolRefExpr::create(Lo, Context), Context); 923 924 const MCAsmInfo *MAI = Context.getAsmInfo(); 925 if (!MAI->doesSetDirectiveSuppressReloc()) { 926 EmitValue(Diff, Size); 927 return; 928 } 929 930 // Otherwise, emit with .set (aka assignment). 931 MCSymbol *SetLabel = Context.createTempSymbol("set", true); 932 EmitAssignment(SetLabel, Diff); 933 EmitSymbolValue(SetLabel, Size); 934 } 935 936 void MCStreamer::emitAbsoluteSymbolDiffAsULEB128(const MCSymbol *Hi, 937 const MCSymbol *Lo) { 938 // Get the Hi-Lo expression. 939 const MCExpr *Diff = 940 MCBinaryExpr::createSub(MCSymbolRefExpr::create(Hi, Context), 941 MCSymbolRefExpr::create(Lo, Context), Context); 942 943 EmitULEB128Value(Diff); 944 } 945 946 void MCStreamer::EmitAssemblerFlag(MCAssemblerFlag Flag) {} 947 void MCStreamer::EmitThumbFunc(MCSymbol *Func) {} 948 void MCStreamer::EmitSymbolDesc(MCSymbol *Symbol, unsigned DescValue) {} 949 void MCStreamer::BeginCOFFSymbolDef(const MCSymbol *Symbol) { 950 llvm_unreachable("this directive only supported on COFF targets"); 951 } 952 void MCStreamer::EndCOFFSymbolDef() { 953 llvm_unreachable("this directive only supported on COFF targets"); 954 } 955 void MCStreamer::EmitFileDirective(StringRef Filename) {} 956 void MCStreamer::EmitCOFFSymbolStorageClass(int StorageClass) { 957 llvm_unreachable("this directive only supported on COFF targets"); 958 } 959 void MCStreamer::EmitCOFFSymbolType(int Type) { 960 llvm_unreachable("this directive only supported on COFF targets"); 961 } 962 void MCStreamer::emitELFSize(MCSymbol *Symbol, const MCExpr *Value) {} 963 void MCStreamer::emitELFSymverDirective(StringRef AliasName, 964 const MCSymbol *Aliasee) {} 965 void MCStreamer::EmitLocalCommonSymbol(MCSymbol *Symbol, uint64_t Size, 966 unsigned ByteAlignment) {} 967 void MCStreamer::EmitTBSSSymbol(MCSection *Section, MCSymbol *Symbol, 968 uint64_t Size, unsigned ByteAlignment) {} 969 void MCStreamer::ChangeSection(MCSection *, const MCExpr *) {} 970 void MCStreamer::EmitWeakReference(MCSymbol *Alias, const MCSymbol *Symbol) {} 971 void MCStreamer::EmitBytes(StringRef Data) {} 972 void MCStreamer::EmitBinaryData(StringRef Data) { EmitBytes(Data); } 973 void MCStreamer::EmitValueImpl(const MCExpr *Value, unsigned Size, SMLoc Loc) { 974 visitUsedExpr(*Value); 975 } 976 void MCStreamer::EmitULEB128Value(const MCExpr *Value) {} 977 void MCStreamer::EmitSLEB128Value(const MCExpr *Value) {} 978 void MCStreamer::emitFill(const MCExpr &NumBytes, uint64_t Value, SMLoc Loc) {} 979 void MCStreamer::emitFill(const MCExpr &NumValues, int64_t Size, int64_t Expr, 980 SMLoc Loc) {} 981 void MCStreamer::EmitValueToAlignment(unsigned ByteAlignment, int64_t Value, 982 unsigned ValueSize, 983 unsigned MaxBytesToEmit) {} 984 void MCStreamer::EmitCodeAlignment(unsigned ByteAlignment, 985 unsigned MaxBytesToEmit) {} 986 void MCStreamer::emitValueToOffset(const MCExpr *Offset, unsigned char Value, 987 SMLoc Loc) {} 988 void MCStreamer::EmitBundleAlignMode(unsigned AlignPow2) {} 989 void MCStreamer::EmitBundleLock(bool AlignToEnd) {} 990 void MCStreamer::FinishImpl() {} 991 void MCStreamer::EmitBundleUnlock() {} 992 993 void MCStreamer::SwitchSection(MCSection *Section, const MCExpr *Subsection) { 994 assert(Section && "Cannot switch to a null section!"); 995 MCSectionSubPair curSection = SectionStack.back().first; 996 SectionStack.back().second = curSection; 997 if (MCSectionSubPair(Section, Subsection) != curSection) { 998 ChangeSection(Section, Subsection); 999 SectionStack.back().first = MCSectionSubPair(Section, Subsection); 1000 assert(!Section->hasEnded() && "Section already ended"); 1001 MCSymbol *Sym = Section->getBeginSymbol(); 1002 if (Sym && !Sym->isInSection()) 1003 EmitLabel(Sym); 1004 } 1005 } 1006 1007 MCSymbol *MCStreamer::endSection(MCSection *Section) { 1008 // TODO: keep track of the last subsection so that this symbol appears in the 1009 // correct place. 1010 MCSymbol *Sym = Section->getEndSymbol(Context); 1011 if (Sym->isInSection()) 1012 return Sym; 1013 1014 SwitchSection(Section); 1015 EmitLabel(Sym); 1016 return Sym; 1017 } 1018 1019 void MCStreamer::EmitVersionForTarget(const Triple &Target) { 1020 if (!Target.isOSBinFormatMachO() || !Target.isOSDarwin()) 1021 return; 1022 // Do we even know the version? 1023 if (Target.getOSMajorVersion() == 0) 1024 return; 1025 1026 unsigned Major; 1027 unsigned Minor; 1028 unsigned Update; 1029 MCVersionMinType VersionType; 1030 if (Target.isWatchOS()) { 1031 VersionType = MCVM_WatchOSVersionMin; 1032 Target.getWatchOSVersion(Major, Minor, Update); 1033 } else if (Target.isTvOS()) { 1034 VersionType = MCVM_TvOSVersionMin; 1035 Target.getiOSVersion(Major, Minor, Update); 1036 } else if (Target.isMacOSX()) { 1037 VersionType = MCVM_OSXVersionMin; 1038 if (!Target.getMacOSXVersion(Major, Minor, Update)) 1039 Major = 0; 1040 } else { 1041 VersionType = MCVM_IOSVersionMin; 1042 Target.getiOSVersion(Major, Minor, Update); 1043 } 1044 if (Major != 0) 1045 EmitVersionMin(VersionType, Major, Minor, Update); 1046 } 1047