10b57cec5SDimitry Andric //===- DarwinAsmParser.cpp - Darwin (Mach-O) Assembly Parser --------------===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric 90b57cec5SDimitry Andric #include "llvm/ADT/STLExtras.h" 100b57cec5SDimitry Andric #include "llvm/ADT/SmallVector.h" 110b57cec5SDimitry Andric #include "llvm/ADT/StringRef.h" 120b57cec5SDimitry Andric #include "llvm/ADT/StringSwitch.h" 130b57cec5SDimitry Andric #include "llvm/ADT/Triple.h" 140b57cec5SDimitry Andric #include "llvm/ADT/Twine.h" 150b57cec5SDimitry Andric #include "llvm/BinaryFormat/MachO.h" 160b57cec5SDimitry Andric #include "llvm/MC/MCContext.h" 170b57cec5SDimitry Andric #include "llvm/MC/MCDirectives.h" 180b57cec5SDimitry Andric #include "llvm/MC/MCObjectFileInfo.h" 190b57cec5SDimitry Andric #include "llvm/MC/MCParser/MCAsmLexer.h" 200b57cec5SDimitry Andric #include "llvm/MC/MCParser/MCAsmParser.h" 210b57cec5SDimitry Andric #include "llvm/MC/MCParser/MCAsmParserExtension.h" 220b57cec5SDimitry Andric #include "llvm/MC/MCSectionMachO.h" 230b57cec5SDimitry Andric #include "llvm/MC/MCStreamer.h" 240b57cec5SDimitry Andric #include "llvm/MC/MCSymbol.h" 250b57cec5SDimitry Andric #include "llvm/MC/SectionKind.h" 260b57cec5SDimitry Andric #include "llvm/Support/FileSystem.h" 270b57cec5SDimitry Andric #include "llvm/Support/MemoryBuffer.h" 280b57cec5SDimitry Andric #include "llvm/Support/SMLoc.h" 290b57cec5SDimitry Andric #include "llvm/Support/SourceMgr.h" 300b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h" 310b57cec5SDimitry Andric #include <algorithm> 320b57cec5SDimitry Andric #include <cstddef> 330b57cec5SDimitry Andric #include <cstdint> 340b57cec5SDimitry Andric #include <string> 350b57cec5SDimitry Andric #include <system_error> 360b57cec5SDimitry Andric #include <utility> 370b57cec5SDimitry Andric 380b57cec5SDimitry Andric using namespace llvm; 390b57cec5SDimitry Andric 400b57cec5SDimitry Andric namespace { 410b57cec5SDimitry Andric 420b57cec5SDimitry Andric /// Implementation of directive handling which is shared across all 430b57cec5SDimitry Andric /// Darwin targets. 440b57cec5SDimitry Andric class DarwinAsmParser : public MCAsmParserExtension { 450b57cec5SDimitry Andric template<bool (DarwinAsmParser::*HandlerMethod)(StringRef, SMLoc)> 460b57cec5SDimitry Andric void addDirectiveHandler(StringRef Directive) { 470b57cec5SDimitry Andric MCAsmParser::ExtensionDirectiveHandler Handler = std::make_pair( 480b57cec5SDimitry Andric this, HandleDirective<DarwinAsmParser, HandlerMethod>); 490b57cec5SDimitry Andric getParser().addDirectiveHandler(Directive, Handler); 500b57cec5SDimitry Andric } 510b57cec5SDimitry Andric 520b57cec5SDimitry Andric bool parseSectionSwitch(StringRef Segment, StringRef Section, 530b57cec5SDimitry Andric unsigned TAA = 0, unsigned ImplicitAlign = 0, 540b57cec5SDimitry Andric unsigned StubSize = 0); 550b57cec5SDimitry Andric 560b57cec5SDimitry Andric SMLoc LastVersionDirective; 570b57cec5SDimitry Andric 580b57cec5SDimitry Andric public: 590b57cec5SDimitry Andric DarwinAsmParser() = default; 600b57cec5SDimitry Andric 610b57cec5SDimitry Andric void Initialize(MCAsmParser &Parser) override { 620b57cec5SDimitry Andric // Call the base implementation. 630b57cec5SDimitry Andric this->MCAsmParserExtension::Initialize(Parser); 640b57cec5SDimitry Andric 650b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveAltEntry>(".alt_entry"); 660b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveDesc>(".desc"); 670b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveIndirectSymbol>( 680b57cec5SDimitry Andric ".indirect_symbol"); 690b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveLsym>(".lsym"); 700b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveSubsectionsViaSymbols>( 710b57cec5SDimitry Andric ".subsections_via_symbols"); 720b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveDumpOrLoad>(".dump"); 730b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveDumpOrLoad>(".load"); 740b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveSection>(".section"); 750b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectivePushSection>( 760b57cec5SDimitry Andric ".pushsection"); 770b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectivePopSection>( 780b57cec5SDimitry Andric ".popsection"); 790b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectivePrevious>(".previous"); 800b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveSecureLogUnique>( 810b57cec5SDimitry Andric ".secure_log_unique"); 820b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveSecureLogReset>( 830b57cec5SDimitry Andric ".secure_log_reset"); 840b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveTBSS>(".tbss"); 850b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveZerofill>(".zerofill"); 860b57cec5SDimitry Andric 870b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveDataRegion>( 880b57cec5SDimitry Andric ".data_region"); 890b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveDataRegionEnd>( 900b57cec5SDimitry Andric ".end_data_region"); 910b57cec5SDimitry Andric 920b57cec5SDimitry Andric // Special section directives. 930b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveBss>(".bss"); 940b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveConst>(".const"); 950b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveConstData>( 960b57cec5SDimitry Andric ".const_data"); 970b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveConstructor>( 980b57cec5SDimitry Andric ".constructor"); 990b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveCString>( 1000b57cec5SDimitry Andric ".cstring"); 1010b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveData>(".data"); 1020b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveDestructor>( 1030b57cec5SDimitry Andric ".destructor"); 1040b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveDyld>(".dyld"); 1050b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveFVMLibInit0>( 1060b57cec5SDimitry Andric ".fvmlib_init0"); 1070b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveFVMLibInit1>( 1080b57cec5SDimitry Andric ".fvmlib_init1"); 1090b57cec5SDimitry Andric addDirectiveHandler< 1100b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveLazySymbolPointers>( 1110b57cec5SDimitry Andric ".lazy_symbol_pointer"); 1120b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseDirectiveLinkerOption>( 1130b57cec5SDimitry Andric ".linker_option"); 1140b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveLiteral16>( 1150b57cec5SDimitry Andric ".literal16"); 1160b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveLiteral4>( 1170b57cec5SDimitry Andric ".literal4"); 1180b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveLiteral8>( 1190b57cec5SDimitry Andric ".literal8"); 1200b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveModInitFunc>( 1210b57cec5SDimitry Andric ".mod_init_func"); 1220b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveModTermFunc>( 1230b57cec5SDimitry Andric ".mod_term_func"); 1240b57cec5SDimitry Andric addDirectiveHandler< 1250b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveNonLazySymbolPointers>( 1260b57cec5SDimitry Andric ".non_lazy_symbol_pointer"); 1270b57cec5SDimitry Andric addDirectiveHandler< 1280b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveThreadLocalVariablePointers>( 1290b57cec5SDimitry Andric ".thread_local_variable_pointer"); 1300b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCCatClsMeth>( 1310b57cec5SDimitry Andric ".objc_cat_cls_meth"); 1320b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCCatInstMeth>( 1330b57cec5SDimitry Andric ".objc_cat_inst_meth"); 1340b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCCategory>( 1350b57cec5SDimitry Andric ".objc_category"); 1360b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCClass>( 1370b57cec5SDimitry Andric ".objc_class"); 1380b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCClassNames>( 1390b57cec5SDimitry Andric ".objc_class_names"); 1400b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCClassVars>( 1410b57cec5SDimitry Andric ".objc_class_vars"); 1420b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCClsMeth>( 1430b57cec5SDimitry Andric ".objc_cls_meth"); 1440b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCClsRefs>( 1450b57cec5SDimitry Andric ".objc_cls_refs"); 1460b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCInstMeth>( 1470b57cec5SDimitry Andric ".objc_inst_meth"); 1480b57cec5SDimitry Andric addDirectiveHandler< 1490b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveObjCInstanceVars>( 1500b57cec5SDimitry Andric ".objc_instance_vars"); 1510b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCMessageRefs>( 1520b57cec5SDimitry Andric ".objc_message_refs"); 1530b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCMetaClass>( 1540b57cec5SDimitry Andric ".objc_meta_class"); 1550b57cec5SDimitry Andric addDirectiveHandler< 1560b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveObjCMethVarNames>( 1570b57cec5SDimitry Andric ".objc_meth_var_names"); 1580b57cec5SDimitry Andric addDirectiveHandler< 1590b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveObjCMethVarTypes>( 1600b57cec5SDimitry Andric ".objc_meth_var_types"); 1610b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCModuleInfo>( 1620b57cec5SDimitry Andric ".objc_module_info"); 1630b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCProtocol>( 1640b57cec5SDimitry Andric ".objc_protocol"); 1650b57cec5SDimitry Andric addDirectiveHandler< 1660b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveObjCSelectorStrs>( 1670b57cec5SDimitry Andric ".objc_selector_strs"); 1680b57cec5SDimitry Andric addDirectiveHandler< 1690b57cec5SDimitry Andric &DarwinAsmParser::parseSectionDirectiveObjCStringObject>( 1700b57cec5SDimitry Andric ".objc_string_object"); 1710b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveObjCSymbols>( 1720b57cec5SDimitry Andric ".objc_symbols"); 1730b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectivePICSymbolStub>( 1740b57cec5SDimitry Andric ".picsymbol_stub"); 1750b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveStaticConst>( 1760b57cec5SDimitry Andric ".static_const"); 1770b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveStaticData>( 1780b57cec5SDimitry Andric ".static_data"); 1790b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveSymbolStub>( 1800b57cec5SDimitry Andric ".symbol_stub"); 1810b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveTData>(".tdata"); 1820b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveText>(".text"); 1830b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveThreadInitFunc>( 1840b57cec5SDimitry Andric ".thread_init_func"); 1850b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveTLV>(".tlv"); 1860b57cec5SDimitry Andric 1870b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseSectionDirectiveIdent>(".ident"); 1880b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseWatchOSVersionMin>( 1890b57cec5SDimitry Andric ".watchos_version_min"); 1900b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseTvOSVersionMin>( 1910b57cec5SDimitry Andric ".tvos_version_min"); 1920b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseIOSVersionMin>( 1930b57cec5SDimitry Andric ".ios_version_min"); 1940b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseMacOSXVersionMin>( 1950b57cec5SDimitry Andric ".macosx_version_min"); 1960b57cec5SDimitry Andric addDirectiveHandler<&DarwinAsmParser::parseBuildVersion>(".build_version"); 1970b57cec5SDimitry Andric 1980b57cec5SDimitry Andric LastVersionDirective = SMLoc(); 1990b57cec5SDimitry Andric } 2000b57cec5SDimitry Andric 2010b57cec5SDimitry Andric bool parseDirectiveAltEntry(StringRef, SMLoc); 2020b57cec5SDimitry Andric bool parseDirectiveDesc(StringRef, SMLoc); 2030b57cec5SDimitry Andric bool parseDirectiveIndirectSymbol(StringRef, SMLoc); 2040b57cec5SDimitry Andric bool parseDirectiveDumpOrLoad(StringRef, SMLoc); 2050b57cec5SDimitry Andric bool parseDirectiveLsym(StringRef, SMLoc); 2060b57cec5SDimitry Andric bool parseDirectiveLinkerOption(StringRef, SMLoc); 2070b57cec5SDimitry Andric bool parseDirectiveSection(StringRef, SMLoc); 2080b57cec5SDimitry Andric bool parseDirectivePushSection(StringRef, SMLoc); 2090b57cec5SDimitry Andric bool parseDirectivePopSection(StringRef, SMLoc); 2100b57cec5SDimitry Andric bool parseDirectivePrevious(StringRef, SMLoc); 2110b57cec5SDimitry Andric bool parseDirectiveSecureLogReset(StringRef, SMLoc); 2120b57cec5SDimitry Andric bool parseDirectiveSecureLogUnique(StringRef, SMLoc); 2130b57cec5SDimitry Andric bool parseDirectiveSubsectionsViaSymbols(StringRef, SMLoc); 2140b57cec5SDimitry Andric bool parseDirectiveTBSS(StringRef, SMLoc); 2150b57cec5SDimitry Andric bool parseDirectiveZerofill(StringRef, SMLoc); 2160b57cec5SDimitry Andric bool parseDirectiveDataRegion(StringRef, SMLoc); 2170b57cec5SDimitry Andric bool parseDirectiveDataRegionEnd(StringRef, SMLoc); 2180b57cec5SDimitry Andric 2190b57cec5SDimitry Andric // Named Section Directive 2200b57cec5SDimitry Andric bool parseSectionDirectiveBss(StringRef, SMLoc) { 2210b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__bss"); 2220b57cec5SDimitry Andric } 2230b57cec5SDimitry Andric 2240b57cec5SDimitry Andric bool parseSectionDirectiveConst(StringRef, SMLoc) { 2250b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__const"); 2260b57cec5SDimitry Andric } 2270b57cec5SDimitry Andric 2280b57cec5SDimitry Andric bool parseSectionDirectiveStaticConst(StringRef, SMLoc) { 2290b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__static_const"); 2300b57cec5SDimitry Andric } 2310b57cec5SDimitry Andric 2320b57cec5SDimitry Andric bool parseSectionDirectiveCString(StringRef, SMLoc) { 2330b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__cstring", 2340b57cec5SDimitry Andric MachO::S_CSTRING_LITERALS); 2350b57cec5SDimitry Andric } 2360b57cec5SDimitry Andric 2370b57cec5SDimitry Andric bool parseSectionDirectiveLiteral4(StringRef, SMLoc) { 2380b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__literal4", 2390b57cec5SDimitry Andric MachO::S_4BYTE_LITERALS, 4); 2400b57cec5SDimitry Andric } 2410b57cec5SDimitry Andric 2420b57cec5SDimitry Andric bool parseSectionDirectiveLiteral8(StringRef, SMLoc) { 2430b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__literal8", 2440b57cec5SDimitry Andric MachO::S_8BYTE_LITERALS, 8); 2450b57cec5SDimitry Andric } 2460b57cec5SDimitry Andric 2470b57cec5SDimitry Andric bool parseSectionDirectiveLiteral16(StringRef, SMLoc) { 2480b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__literal16", 2490b57cec5SDimitry Andric MachO::S_16BYTE_LITERALS, 16); 2500b57cec5SDimitry Andric } 2510b57cec5SDimitry Andric 2520b57cec5SDimitry Andric bool parseSectionDirectiveConstructor(StringRef, SMLoc) { 2530b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__constructor"); 2540b57cec5SDimitry Andric } 2550b57cec5SDimitry Andric 2560b57cec5SDimitry Andric bool parseSectionDirectiveDestructor(StringRef, SMLoc) { 2570b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__destructor"); 2580b57cec5SDimitry Andric } 2590b57cec5SDimitry Andric 2600b57cec5SDimitry Andric bool parseSectionDirectiveFVMLibInit0(StringRef, SMLoc) { 2610b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__fvmlib_init0"); 2620b57cec5SDimitry Andric } 2630b57cec5SDimitry Andric 2640b57cec5SDimitry Andric bool parseSectionDirectiveFVMLibInit1(StringRef, SMLoc) { 2650b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__fvmlib_init1"); 2660b57cec5SDimitry Andric } 2670b57cec5SDimitry Andric 2680b57cec5SDimitry Andric bool parseSectionDirectiveSymbolStub(StringRef, SMLoc) { 2690b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__symbol_stub", 2700b57cec5SDimitry Andric MachO::S_SYMBOL_STUBS | 2710b57cec5SDimitry Andric MachO::S_ATTR_PURE_INSTRUCTIONS, 2720b57cec5SDimitry Andric // FIXME: Different on PPC and ARM. 2730b57cec5SDimitry Andric 0, 16); 2740b57cec5SDimitry Andric } 2750b57cec5SDimitry Andric 2760b57cec5SDimitry Andric bool parseSectionDirectivePICSymbolStub(StringRef, SMLoc) { 2770b57cec5SDimitry Andric return parseSectionSwitch("__TEXT","__picsymbol_stub", 2780b57cec5SDimitry Andric MachO::S_SYMBOL_STUBS | 2790b57cec5SDimitry Andric MachO::S_ATTR_PURE_INSTRUCTIONS, 0, 26); 2800b57cec5SDimitry Andric } 2810b57cec5SDimitry Andric 2820b57cec5SDimitry Andric bool parseSectionDirectiveData(StringRef, SMLoc) { 2830b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__data"); 2840b57cec5SDimitry Andric } 2850b57cec5SDimitry Andric 2860b57cec5SDimitry Andric bool parseSectionDirectiveStaticData(StringRef, SMLoc) { 2870b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__static_data"); 2880b57cec5SDimitry Andric } 2890b57cec5SDimitry Andric 2900b57cec5SDimitry Andric bool parseSectionDirectiveNonLazySymbolPointers(StringRef, SMLoc) { 2910b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__nl_symbol_ptr", 2920b57cec5SDimitry Andric MachO::S_NON_LAZY_SYMBOL_POINTERS, 4); 2930b57cec5SDimitry Andric } 2940b57cec5SDimitry Andric 2950b57cec5SDimitry Andric bool parseSectionDirectiveLazySymbolPointers(StringRef, SMLoc) { 2960b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__la_symbol_ptr", 2970b57cec5SDimitry Andric MachO::S_LAZY_SYMBOL_POINTERS, 4); 2980b57cec5SDimitry Andric } 2990b57cec5SDimitry Andric 3000b57cec5SDimitry Andric bool parseSectionDirectiveThreadLocalVariablePointers(StringRef, SMLoc) { 3010b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__thread_ptr", 3020b57cec5SDimitry Andric MachO::S_THREAD_LOCAL_VARIABLE_POINTERS, 4); 3030b57cec5SDimitry Andric } 3040b57cec5SDimitry Andric 3050b57cec5SDimitry Andric bool parseSectionDirectiveDyld(StringRef, SMLoc) { 3060b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__dyld"); 3070b57cec5SDimitry Andric } 3080b57cec5SDimitry Andric 3090b57cec5SDimitry Andric bool parseSectionDirectiveModInitFunc(StringRef, SMLoc) { 3100b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__mod_init_func", 3110b57cec5SDimitry Andric MachO::S_MOD_INIT_FUNC_POINTERS, 4); 3120b57cec5SDimitry Andric } 3130b57cec5SDimitry Andric 3140b57cec5SDimitry Andric bool parseSectionDirectiveModTermFunc(StringRef, SMLoc) { 3150b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__mod_term_func", 3160b57cec5SDimitry Andric MachO::S_MOD_TERM_FUNC_POINTERS, 4); 3170b57cec5SDimitry Andric } 3180b57cec5SDimitry Andric 3190b57cec5SDimitry Andric bool parseSectionDirectiveConstData(StringRef, SMLoc) { 3200b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__const"); 3210b57cec5SDimitry Andric } 3220b57cec5SDimitry Andric 3230b57cec5SDimitry Andric bool parseSectionDirectiveObjCClass(StringRef, SMLoc) { 3240b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__class", 3250b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3260b57cec5SDimitry Andric } 3270b57cec5SDimitry Andric 3280b57cec5SDimitry Andric bool parseSectionDirectiveObjCMetaClass(StringRef, SMLoc) { 3290b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__meta_class", 3300b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3310b57cec5SDimitry Andric } 3320b57cec5SDimitry Andric 3330b57cec5SDimitry Andric bool parseSectionDirectiveObjCCatClsMeth(StringRef, SMLoc) { 3340b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__cat_cls_meth", 3350b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3360b57cec5SDimitry Andric } 3370b57cec5SDimitry Andric 3380b57cec5SDimitry Andric bool parseSectionDirectiveObjCCatInstMeth(StringRef, SMLoc) { 3390b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__cat_inst_meth", 3400b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3410b57cec5SDimitry Andric } 3420b57cec5SDimitry Andric 3430b57cec5SDimitry Andric bool parseSectionDirectiveObjCProtocol(StringRef, SMLoc) { 3440b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__protocol", 3450b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3460b57cec5SDimitry Andric } 3470b57cec5SDimitry Andric 3480b57cec5SDimitry Andric bool parseSectionDirectiveObjCStringObject(StringRef, SMLoc) { 3490b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__string_object", 3500b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3510b57cec5SDimitry Andric } 3520b57cec5SDimitry Andric 3530b57cec5SDimitry Andric bool parseSectionDirectiveObjCClsMeth(StringRef, SMLoc) { 3540b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__cls_meth", 3550b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3560b57cec5SDimitry Andric } 3570b57cec5SDimitry Andric 3580b57cec5SDimitry Andric bool parseSectionDirectiveObjCInstMeth(StringRef, SMLoc) { 3590b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__inst_meth", 3600b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3610b57cec5SDimitry Andric } 3620b57cec5SDimitry Andric 3630b57cec5SDimitry Andric bool parseSectionDirectiveObjCClsRefs(StringRef, SMLoc) { 3640b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__cls_refs", 3650b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP | 3660b57cec5SDimitry Andric MachO::S_LITERAL_POINTERS, 4); 3670b57cec5SDimitry Andric } 3680b57cec5SDimitry Andric 3690b57cec5SDimitry Andric bool parseSectionDirectiveObjCMessageRefs(StringRef, SMLoc) { 3700b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__message_refs", 3710b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP | 3720b57cec5SDimitry Andric MachO::S_LITERAL_POINTERS, 4); 3730b57cec5SDimitry Andric } 3740b57cec5SDimitry Andric 3750b57cec5SDimitry Andric bool parseSectionDirectiveObjCSymbols(StringRef, SMLoc) { 3760b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__symbols", 3770b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3780b57cec5SDimitry Andric } 3790b57cec5SDimitry Andric 3800b57cec5SDimitry Andric bool parseSectionDirectiveObjCCategory(StringRef, SMLoc) { 3810b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__category", 3820b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3830b57cec5SDimitry Andric } 3840b57cec5SDimitry Andric 3850b57cec5SDimitry Andric bool parseSectionDirectiveObjCClassVars(StringRef, SMLoc) { 3860b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__class_vars", 3870b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3880b57cec5SDimitry Andric } 3890b57cec5SDimitry Andric 3900b57cec5SDimitry Andric bool parseSectionDirectiveObjCInstanceVars(StringRef, SMLoc) { 3910b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__instance_vars", 3920b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3930b57cec5SDimitry Andric } 3940b57cec5SDimitry Andric 3950b57cec5SDimitry Andric bool parseSectionDirectiveObjCModuleInfo(StringRef, SMLoc) { 3960b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__module_info", 3970b57cec5SDimitry Andric MachO::S_ATTR_NO_DEAD_STRIP); 3980b57cec5SDimitry Andric } 3990b57cec5SDimitry Andric 4000b57cec5SDimitry Andric bool parseSectionDirectiveObjCClassNames(StringRef, SMLoc) { 4010b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__cstring", 4020b57cec5SDimitry Andric MachO::S_CSTRING_LITERALS); 4030b57cec5SDimitry Andric } 4040b57cec5SDimitry Andric 4050b57cec5SDimitry Andric bool parseSectionDirectiveObjCMethVarTypes(StringRef, SMLoc) { 4060b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__cstring", 4070b57cec5SDimitry Andric MachO::S_CSTRING_LITERALS); 4080b57cec5SDimitry Andric } 4090b57cec5SDimitry Andric 4100b57cec5SDimitry Andric bool parseSectionDirectiveObjCMethVarNames(StringRef, SMLoc) { 4110b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__cstring", 4120b57cec5SDimitry Andric MachO::S_CSTRING_LITERALS); 4130b57cec5SDimitry Andric } 4140b57cec5SDimitry Andric 4150b57cec5SDimitry Andric bool parseSectionDirectiveObjCSelectorStrs(StringRef, SMLoc) { 4160b57cec5SDimitry Andric return parseSectionSwitch("__OBJC", "__selector_strs", 4170b57cec5SDimitry Andric MachO::S_CSTRING_LITERALS); 4180b57cec5SDimitry Andric } 4190b57cec5SDimitry Andric 4200b57cec5SDimitry Andric bool parseSectionDirectiveTData(StringRef, SMLoc) { 4210b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__thread_data", 4220b57cec5SDimitry Andric MachO::S_THREAD_LOCAL_REGULAR); 4230b57cec5SDimitry Andric } 4240b57cec5SDimitry Andric 4250b57cec5SDimitry Andric bool parseSectionDirectiveText(StringRef, SMLoc) { 4260b57cec5SDimitry Andric return parseSectionSwitch("__TEXT", "__text", 4270b57cec5SDimitry Andric MachO::S_ATTR_PURE_INSTRUCTIONS); 4280b57cec5SDimitry Andric } 4290b57cec5SDimitry Andric 4300b57cec5SDimitry Andric bool parseSectionDirectiveTLV(StringRef, SMLoc) { 4310b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__thread_vars", 4320b57cec5SDimitry Andric MachO::S_THREAD_LOCAL_VARIABLES); 4330b57cec5SDimitry Andric } 4340b57cec5SDimitry Andric 4350b57cec5SDimitry Andric bool parseSectionDirectiveIdent(StringRef, SMLoc) { 4360b57cec5SDimitry Andric // Darwin silently ignores the .ident directive. 4370b57cec5SDimitry Andric getParser().eatToEndOfStatement(); 4380b57cec5SDimitry Andric return false; 4390b57cec5SDimitry Andric } 4400b57cec5SDimitry Andric 4410b57cec5SDimitry Andric bool parseSectionDirectiveThreadInitFunc(StringRef, SMLoc) { 4420b57cec5SDimitry Andric return parseSectionSwitch("__DATA", "__thread_init", 4430b57cec5SDimitry Andric MachO::S_THREAD_LOCAL_INIT_FUNCTION_POINTERS); 4440b57cec5SDimitry Andric } 4450b57cec5SDimitry Andric 4460b57cec5SDimitry Andric bool parseWatchOSVersionMin(StringRef Directive, SMLoc Loc) { 4470b57cec5SDimitry Andric return parseVersionMin(Directive, Loc, MCVM_WatchOSVersionMin); 4480b57cec5SDimitry Andric } 4490b57cec5SDimitry Andric bool parseTvOSVersionMin(StringRef Directive, SMLoc Loc) { 4500b57cec5SDimitry Andric return parseVersionMin(Directive, Loc, MCVM_TvOSVersionMin); 4510b57cec5SDimitry Andric } 4520b57cec5SDimitry Andric bool parseIOSVersionMin(StringRef Directive, SMLoc Loc) { 4530b57cec5SDimitry Andric return parseVersionMin(Directive, Loc, MCVM_IOSVersionMin); 4540b57cec5SDimitry Andric } 4550b57cec5SDimitry Andric bool parseMacOSXVersionMin(StringRef Directive, SMLoc Loc) { 4560b57cec5SDimitry Andric return parseVersionMin(Directive, Loc, MCVM_OSXVersionMin); 4570b57cec5SDimitry Andric } 4580b57cec5SDimitry Andric 4590b57cec5SDimitry Andric bool parseBuildVersion(StringRef Directive, SMLoc Loc); 4600b57cec5SDimitry Andric bool parseVersionMin(StringRef Directive, SMLoc Loc, MCVersionMinType Type); 4610b57cec5SDimitry Andric bool parseMajorMinorVersionComponent(unsigned *Major, unsigned *Minor, 4620b57cec5SDimitry Andric const char *VersionName); 4630b57cec5SDimitry Andric bool parseOptionalTrailingVersionComponent(unsigned *Component, 4640b57cec5SDimitry Andric const char *ComponentName); 4650b57cec5SDimitry Andric bool parseVersion(unsigned *Major, unsigned *Minor, unsigned *Update); 4660b57cec5SDimitry Andric bool parseSDKVersion(VersionTuple &SDKVersion); 4670b57cec5SDimitry Andric void checkVersion(StringRef Directive, StringRef Arg, SMLoc Loc, 4680b57cec5SDimitry Andric Triple::OSType ExpectedOS); 4690b57cec5SDimitry Andric }; 4700b57cec5SDimitry Andric 4710b57cec5SDimitry Andric } // end anonymous namespace 4720b57cec5SDimitry Andric 4730b57cec5SDimitry Andric bool DarwinAsmParser::parseSectionSwitch(StringRef Segment, StringRef Section, 4740b57cec5SDimitry Andric unsigned TAA, unsigned Align, 4750b57cec5SDimitry Andric unsigned StubSize) { 4760b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 4770b57cec5SDimitry Andric return TokError("unexpected token in section switching directive"); 4780b57cec5SDimitry Andric Lex(); 4790b57cec5SDimitry Andric 4800b57cec5SDimitry Andric // FIXME: Arch specific. 4810b57cec5SDimitry Andric bool isText = TAA & MachO::S_ATTR_PURE_INSTRUCTIONS; 4820b57cec5SDimitry Andric getStreamer().SwitchSection(getContext().getMachOSection( 4830b57cec5SDimitry Andric Segment, Section, TAA, StubSize, 4840b57cec5SDimitry Andric isText ? SectionKind::getText() : SectionKind::getData())); 4850b57cec5SDimitry Andric 4860b57cec5SDimitry Andric // Set the implicit alignment, if any. 4870b57cec5SDimitry Andric // 4880b57cec5SDimitry Andric // FIXME: This isn't really what 'as' does; I think it just uses the implicit 4890b57cec5SDimitry Andric // alignment on the section (e.g., if one manually inserts bytes into the 4900b57cec5SDimitry Andric // section, then just issuing the section switch directive will not realign 4910b57cec5SDimitry Andric // the section. However, this is arguably more reasonable behavior, and there 4920b57cec5SDimitry Andric // is no good reason for someone to intentionally emit incorrectly sized 4930b57cec5SDimitry Andric // values into the implicitly aligned sections. 4940b57cec5SDimitry Andric if (Align) 4955ffd83dbSDimitry Andric getStreamer().emitValueToAlignment(Align); 4960b57cec5SDimitry Andric 4970b57cec5SDimitry Andric return false; 4980b57cec5SDimitry Andric } 4990b57cec5SDimitry Andric 5000b57cec5SDimitry Andric /// parseDirectiveAltEntry 5010b57cec5SDimitry Andric /// ::= .alt_entry identifier 5020b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveAltEntry(StringRef, SMLoc) { 5030b57cec5SDimitry Andric StringRef Name; 5040b57cec5SDimitry Andric if (getParser().parseIdentifier(Name)) 5050b57cec5SDimitry Andric return TokError("expected identifier in directive"); 5060b57cec5SDimitry Andric 5070b57cec5SDimitry Andric // Look up symbol. 5080b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(Name); 5090b57cec5SDimitry Andric 5100b57cec5SDimitry Andric if (Sym->isDefined()) 5110b57cec5SDimitry Andric return TokError(".alt_entry must preceed symbol definition"); 5120b57cec5SDimitry Andric 5135ffd83dbSDimitry Andric if (!getStreamer().emitSymbolAttribute(Sym, MCSA_AltEntry)) 5140b57cec5SDimitry Andric return TokError("unable to emit symbol attribute"); 5150b57cec5SDimitry Andric 5160b57cec5SDimitry Andric Lex(); 5170b57cec5SDimitry Andric return false; 5180b57cec5SDimitry Andric } 5190b57cec5SDimitry Andric 5200b57cec5SDimitry Andric /// parseDirectiveDesc 5210b57cec5SDimitry Andric /// ::= .desc identifier , expression 5220b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveDesc(StringRef, SMLoc) { 5230b57cec5SDimitry Andric StringRef Name; 5240b57cec5SDimitry Andric if (getParser().parseIdentifier(Name)) 5250b57cec5SDimitry Andric return TokError("expected identifier in directive"); 5260b57cec5SDimitry Andric 5270b57cec5SDimitry Andric // Handle the identifier as the key symbol. 5280b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(Name); 5290b57cec5SDimitry Andric 5300b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 5310b57cec5SDimitry Andric return TokError("unexpected token in '.desc' directive"); 5320b57cec5SDimitry Andric Lex(); 5330b57cec5SDimitry Andric 5340b57cec5SDimitry Andric int64_t DescValue; 5350b57cec5SDimitry Andric if (getParser().parseAbsoluteExpression(DescValue)) 5360b57cec5SDimitry Andric return true; 5370b57cec5SDimitry Andric 5380b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 5390b57cec5SDimitry Andric return TokError("unexpected token in '.desc' directive"); 5400b57cec5SDimitry Andric 5410b57cec5SDimitry Andric Lex(); 5420b57cec5SDimitry Andric 5430b57cec5SDimitry Andric // Set the n_desc field of this Symbol to this DescValue 5445ffd83dbSDimitry Andric getStreamer().emitSymbolDesc(Sym, DescValue); 5450b57cec5SDimitry Andric 5460b57cec5SDimitry Andric return false; 5470b57cec5SDimitry Andric } 5480b57cec5SDimitry Andric 5490b57cec5SDimitry Andric /// parseDirectiveIndirectSymbol 5500b57cec5SDimitry Andric /// ::= .indirect_symbol identifier 5510b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveIndirectSymbol(StringRef, SMLoc Loc) { 5520b57cec5SDimitry Andric const MCSectionMachO *Current = static_cast<const MCSectionMachO *>( 5530b57cec5SDimitry Andric getStreamer().getCurrentSectionOnly()); 5540b57cec5SDimitry Andric MachO::SectionType SectionType = Current->getType(); 5550b57cec5SDimitry Andric if (SectionType != MachO::S_NON_LAZY_SYMBOL_POINTERS && 5560b57cec5SDimitry Andric SectionType != MachO::S_LAZY_SYMBOL_POINTERS && 5570b57cec5SDimitry Andric SectionType != MachO::S_THREAD_LOCAL_VARIABLE_POINTERS && 5580b57cec5SDimitry Andric SectionType != MachO::S_SYMBOL_STUBS) 5590b57cec5SDimitry Andric return Error(Loc, "indirect symbol not in a symbol pointer or stub " 5600b57cec5SDimitry Andric "section"); 5610b57cec5SDimitry Andric 5620b57cec5SDimitry Andric StringRef Name; 5630b57cec5SDimitry Andric if (getParser().parseIdentifier(Name)) 5640b57cec5SDimitry Andric return TokError("expected identifier in .indirect_symbol directive"); 5650b57cec5SDimitry Andric 5660b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(Name); 5670b57cec5SDimitry Andric 5680b57cec5SDimitry Andric // Assembler local symbols don't make any sense here. Complain loudly. 5690b57cec5SDimitry Andric if (Sym->isTemporary()) 5700b57cec5SDimitry Andric return TokError("non-local symbol required in directive"); 5710b57cec5SDimitry Andric 5725ffd83dbSDimitry Andric if (!getStreamer().emitSymbolAttribute(Sym, MCSA_IndirectSymbol)) 5730b57cec5SDimitry Andric return TokError("unable to emit indirect symbol attribute for: " + Name); 5740b57cec5SDimitry Andric 5750b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 5760b57cec5SDimitry Andric return TokError("unexpected token in '.indirect_symbol' directive"); 5770b57cec5SDimitry Andric 5780b57cec5SDimitry Andric Lex(); 5790b57cec5SDimitry Andric 5800b57cec5SDimitry Andric return false; 5810b57cec5SDimitry Andric } 5820b57cec5SDimitry Andric 5830b57cec5SDimitry Andric /// parseDirectiveDumpOrLoad 5840b57cec5SDimitry Andric /// ::= ( .dump | .load ) "filename" 5850b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveDumpOrLoad(StringRef Directive, 5860b57cec5SDimitry Andric SMLoc IDLoc) { 5870b57cec5SDimitry Andric bool IsDump = Directive == ".dump"; 5880b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::String)) 5890b57cec5SDimitry Andric return TokError("expected string in '.dump' or '.load' directive"); 5900b57cec5SDimitry Andric 5910b57cec5SDimitry Andric Lex(); 5920b57cec5SDimitry Andric 5930b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 5940b57cec5SDimitry Andric return TokError("unexpected token in '.dump' or '.load' directive"); 5950b57cec5SDimitry Andric 5960b57cec5SDimitry Andric Lex(); 5970b57cec5SDimitry Andric 5980b57cec5SDimitry Andric // FIXME: If/when .dump and .load are implemented they will be done in the 5990b57cec5SDimitry Andric // the assembly parser and not have any need for an MCStreamer API. 6000b57cec5SDimitry Andric if (IsDump) 6010b57cec5SDimitry Andric return Warning(IDLoc, "ignoring directive .dump for now"); 6020b57cec5SDimitry Andric else 6030b57cec5SDimitry Andric return Warning(IDLoc, "ignoring directive .load for now"); 6040b57cec5SDimitry Andric } 6050b57cec5SDimitry Andric 6060b57cec5SDimitry Andric /// ParseDirectiveLinkerOption 6070b57cec5SDimitry Andric /// ::= .linker_option "string" ( , "string" )* 6080b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveLinkerOption(StringRef IDVal, SMLoc) { 6090b57cec5SDimitry Andric SmallVector<std::string, 4> Args; 6100b57cec5SDimitry Andric while (true) { 6110b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::String)) 6120b57cec5SDimitry Andric return TokError("expected string in '" + Twine(IDVal) + "' directive"); 6130b57cec5SDimitry Andric 6140b57cec5SDimitry Andric std::string Data; 6150b57cec5SDimitry Andric if (getParser().parseEscapedString(Data)) 6160b57cec5SDimitry Andric return true; 6170b57cec5SDimitry Andric 6180b57cec5SDimitry Andric Args.push_back(Data); 6190b57cec5SDimitry Andric 6200b57cec5SDimitry Andric if (getLexer().is(AsmToken::EndOfStatement)) 6210b57cec5SDimitry Andric break; 6220b57cec5SDimitry Andric 6230b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 6240b57cec5SDimitry Andric return TokError("unexpected token in '" + Twine(IDVal) + "' directive"); 6250b57cec5SDimitry Andric Lex(); 6260b57cec5SDimitry Andric } 6270b57cec5SDimitry Andric 6285ffd83dbSDimitry Andric getStreamer().emitLinkerOptions(Args); 6290b57cec5SDimitry Andric return false; 6300b57cec5SDimitry Andric } 6310b57cec5SDimitry Andric 6320b57cec5SDimitry Andric /// parseDirectiveLsym 6330b57cec5SDimitry Andric /// ::= .lsym identifier , expression 6340b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveLsym(StringRef, SMLoc) { 6350b57cec5SDimitry Andric StringRef Name; 6360b57cec5SDimitry Andric if (getParser().parseIdentifier(Name)) 6370b57cec5SDimitry Andric return TokError("expected identifier in directive"); 6380b57cec5SDimitry Andric 6390b57cec5SDimitry Andric // Handle the identifier as the key symbol. 6400b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(Name); 6410b57cec5SDimitry Andric 6420b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 6430b57cec5SDimitry Andric return TokError("unexpected token in '.lsym' directive"); 6440b57cec5SDimitry Andric Lex(); 6450b57cec5SDimitry Andric 6460b57cec5SDimitry Andric const MCExpr *Value; 6470b57cec5SDimitry Andric if (getParser().parseExpression(Value)) 6480b57cec5SDimitry Andric return true; 6490b57cec5SDimitry Andric 6500b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 6510b57cec5SDimitry Andric return TokError("unexpected token in '.lsym' directive"); 6520b57cec5SDimitry Andric 6530b57cec5SDimitry Andric Lex(); 6540b57cec5SDimitry Andric 6550b57cec5SDimitry Andric // We don't currently support this directive. 6560b57cec5SDimitry Andric // 6570b57cec5SDimitry Andric // FIXME: Diagnostic location! 6580b57cec5SDimitry Andric (void) Sym; 6590b57cec5SDimitry Andric return TokError("directive '.lsym' is unsupported"); 6600b57cec5SDimitry Andric } 6610b57cec5SDimitry Andric 6620b57cec5SDimitry Andric /// parseDirectiveSection: 6630b57cec5SDimitry Andric /// ::= .section identifier (',' identifier)* 6640b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveSection(StringRef, SMLoc) { 6650b57cec5SDimitry Andric SMLoc Loc = getLexer().getLoc(); 6660b57cec5SDimitry Andric 6670b57cec5SDimitry Andric StringRef SectionName; 6680b57cec5SDimitry Andric if (getParser().parseIdentifier(SectionName)) 6690b57cec5SDimitry Andric return Error(Loc, "expected identifier after '.section' directive"); 6700b57cec5SDimitry Andric 6710b57cec5SDimitry Andric // Verify there is a following comma. 6720b57cec5SDimitry Andric if (!getLexer().is(AsmToken::Comma)) 6730b57cec5SDimitry Andric return TokError("unexpected token in '.section' directive"); 6740b57cec5SDimitry Andric 6755ffd83dbSDimitry Andric std::string SectionSpec = std::string(SectionName); 6760b57cec5SDimitry Andric SectionSpec += ","; 6770b57cec5SDimitry Andric 6780b57cec5SDimitry Andric // Add all the tokens until the end of the line, ParseSectionSpecifier will 6790b57cec5SDimitry Andric // handle this. 6800b57cec5SDimitry Andric StringRef EOL = getLexer().LexUntilEndOfStatement(); 6810b57cec5SDimitry Andric SectionSpec.append(EOL.begin(), EOL.end()); 6820b57cec5SDimitry Andric 6830b57cec5SDimitry Andric Lex(); 6840b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 6850b57cec5SDimitry Andric return TokError("unexpected token in '.section' directive"); 6860b57cec5SDimitry Andric Lex(); 6870b57cec5SDimitry Andric 6880b57cec5SDimitry Andric StringRef Segment, Section; 6890b57cec5SDimitry Andric unsigned StubSize; 6900b57cec5SDimitry Andric unsigned TAA; 6910b57cec5SDimitry Andric bool TAAParsed; 6920b57cec5SDimitry Andric std::string ErrorStr = 6930b57cec5SDimitry Andric MCSectionMachO::ParseSectionSpecifier(SectionSpec, Segment, Section, 6940b57cec5SDimitry Andric TAA, TAAParsed, StubSize); 6950b57cec5SDimitry Andric 6960b57cec5SDimitry Andric if (!ErrorStr.empty()) 6970b57cec5SDimitry Andric return Error(Loc, ErrorStr); 6980b57cec5SDimitry Andric 6990b57cec5SDimitry Andric // Issue a warning if the target is not powerpc and Section is a *coal* section. 7000b57cec5SDimitry Andric Triple TT = getParser().getContext().getObjectFileInfo()->getTargetTriple(); 7010b57cec5SDimitry Andric Triple::ArchType ArchTy = TT.getArch(); 7020b57cec5SDimitry Andric 7030b57cec5SDimitry Andric if (ArchTy != Triple::ppc && ArchTy != Triple::ppc64) { 7040b57cec5SDimitry Andric StringRef NonCoalSection = StringSwitch<StringRef>(Section) 7050b57cec5SDimitry Andric .Case("__textcoal_nt", "__text") 7060b57cec5SDimitry Andric .Case("__const_coal", "__const") 7070b57cec5SDimitry Andric .Case("__datacoal_nt", "__data") 7080b57cec5SDimitry Andric .Default(Section); 7090b57cec5SDimitry Andric 7100b57cec5SDimitry Andric if (!Section.equals(NonCoalSection)) { 7110b57cec5SDimitry Andric StringRef SectionVal(Loc.getPointer()); 7120b57cec5SDimitry Andric size_t B = SectionVal.find(',') + 1, E = SectionVal.find(',', B); 7130b57cec5SDimitry Andric SMLoc BLoc = SMLoc::getFromPointer(SectionVal.data() + B); 7140b57cec5SDimitry Andric SMLoc ELoc = SMLoc::getFromPointer(SectionVal.data() + E); 7150b57cec5SDimitry Andric getParser().Warning(Loc, "section \"" + Section + "\" is deprecated", 7160b57cec5SDimitry Andric SMRange(BLoc, ELoc)); 7170b57cec5SDimitry Andric getParser().Note(Loc, "change section name to \"" + NonCoalSection + 7180b57cec5SDimitry Andric "\"", SMRange(BLoc, ELoc)); 7190b57cec5SDimitry Andric } 7200b57cec5SDimitry Andric } 7210b57cec5SDimitry Andric 7220b57cec5SDimitry Andric // FIXME: Arch specific. 7230b57cec5SDimitry Andric bool isText = Segment == "__TEXT"; // FIXME: Hack. 7240b57cec5SDimitry Andric getStreamer().SwitchSection(getContext().getMachOSection( 7250b57cec5SDimitry Andric Segment, Section, TAA, StubSize, 7260b57cec5SDimitry Andric isText ? SectionKind::getText() : SectionKind::getData())); 7270b57cec5SDimitry Andric return false; 7280b57cec5SDimitry Andric } 7290b57cec5SDimitry Andric 7300b57cec5SDimitry Andric /// ParseDirectivePushSection: 7310b57cec5SDimitry Andric /// ::= .pushsection identifier (',' identifier)* 7320b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectivePushSection(StringRef S, SMLoc Loc) { 7330b57cec5SDimitry Andric getStreamer().PushSection(); 7340b57cec5SDimitry Andric 7350b57cec5SDimitry Andric if (parseDirectiveSection(S, Loc)) { 7360b57cec5SDimitry Andric getStreamer().PopSection(); 7370b57cec5SDimitry Andric return true; 7380b57cec5SDimitry Andric } 7390b57cec5SDimitry Andric 7400b57cec5SDimitry Andric return false; 7410b57cec5SDimitry Andric } 7420b57cec5SDimitry Andric 7430b57cec5SDimitry Andric /// ParseDirectivePopSection: 7440b57cec5SDimitry Andric /// ::= .popsection 7450b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectivePopSection(StringRef, SMLoc) { 7460b57cec5SDimitry Andric if (!getStreamer().PopSection()) 7470b57cec5SDimitry Andric return TokError(".popsection without corresponding .pushsection"); 7480b57cec5SDimitry Andric return false; 7490b57cec5SDimitry Andric } 7500b57cec5SDimitry Andric 7510b57cec5SDimitry Andric /// ParseDirectivePrevious: 7520b57cec5SDimitry Andric /// ::= .previous 7530b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectivePrevious(StringRef DirName, SMLoc) { 7540b57cec5SDimitry Andric MCSectionSubPair PreviousSection = getStreamer().getPreviousSection(); 7550b57cec5SDimitry Andric if (!PreviousSection.first) 7560b57cec5SDimitry Andric return TokError(".previous without corresponding .section"); 7570b57cec5SDimitry Andric getStreamer().SwitchSection(PreviousSection.first, PreviousSection.second); 7580b57cec5SDimitry Andric return false; 7590b57cec5SDimitry Andric } 7600b57cec5SDimitry Andric 7610b57cec5SDimitry Andric /// ParseDirectiveSecureLogUnique 7620b57cec5SDimitry Andric /// ::= .secure_log_unique ... message ... 7630b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveSecureLogUnique(StringRef, SMLoc IDLoc) { 7640b57cec5SDimitry Andric StringRef LogMessage = getParser().parseStringToEndOfStatement(); 7650b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 7660b57cec5SDimitry Andric return TokError("unexpected token in '.secure_log_unique' directive"); 7670b57cec5SDimitry Andric 7680b57cec5SDimitry Andric if (getContext().getSecureLogUsed()) 7690b57cec5SDimitry Andric return Error(IDLoc, ".secure_log_unique specified multiple times"); 7700b57cec5SDimitry Andric 7710b57cec5SDimitry Andric // Get the secure log path. 7720b57cec5SDimitry Andric const char *SecureLogFile = getContext().getSecureLogFile(); 7730b57cec5SDimitry Andric if (!SecureLogFile) 7740b57cec5SDimitry Andric return Error(IDLoc, ".secure_log_unique used but AS_SECURE_LOG_FILE " 7750b57cec5SDimitry Andric "environment variable unset."); 7760b57cec5SDimitry Andric 7770b57cec5SDimitry Andric // Open the secure log file if we haven't already. 7780b57cec5SDimitry Andric raw_fd_ostream *OS = getContext().getSecureLog(); 7790b57cec5SDimitry Andric if (!OS) { 7800b57cec5SDimitry Andric std::error_code EC; 7818bcb0991SDimitry Andric auto NewOS = std::make_unique<raw_fd_ostream>( 7828bcb0991SDimitry Andric StringRef(SecureLogFile), EC, sys::fs::OF_Append | sys::fs::OF_Text); 7830b57cec5SDimitry Andric if (EC) 7840b57cec5SDimitry Andric return Error(IDLoc, Twine("can't open secure log file: ") + 7850b57cec5SDimitry Andric SecureLogFile + " (" + EC.message() + ")"); 7860b57cec5SDimitry Andric OS = NewOS.get(); 7870b57cec5SDimitry Andric getContext().setSecureLog(std::move(NewOS)); 7880b57cec5SDimitry Andric } 7890b57cec5SDimitry Andric 7900b57cec5SDimitry Andric // Write the message. 7910b57cec5SDimitry Andric unsigned CurBuf = getSourceManager().FindBufferContainingLoc(IDLoc); 7920b57cec5SDimitry Andric *OS << getSourceManager().getBufferInfo(CurBuf).Buffer->getBufferIdentifier() 7930b57cec5SDimitry Andric << ":" << getSourceManager().FindLineNumber(IDLoc, CurBuf) << ":" 7940b57cec5SDimitry Andric << LogMessage + "\n"; 7950b57cec5SDimitry Andric 7960b57cec5SDimitry Andric getContext().setSecureLogUsed(true); 7970b57cec5SDimitry Andric 7980b57cec5SDimitry Andric return false; 7990b57cec5SDimitry Andric } 8000b57cec5SDimitry Andric 8010b57cec5SDimitry Andric /// ParseDirectiveSecureLogReset 8020b57cec5SDimitry Andric /// ::= .secure_log_reset 8030b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveSecureLogReset(StringRef, SMLoc IDLoc) { 8040b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 8050b57cec5SDimitry Andric return TokError("unexpected token in '.secure_log_reset' directive"); 8060b57cec5SDimitry Andric 8070b57cec5SDimitry Andric Lex(); 8080b57cec5SDimitry Andric 8090b57cec5SDimitry Andric getContext().setSecureLogUsed(false); 8100b57cec5SDimitry Andric 8110b57cec5SDimitry Andric return false; 8120b57cec5SDimitry Andric } 8130b57cec5SDimitry Andric 8140b57cec5SDimitry Andric /// parseDirectiveSubsectionsViaSymbols 8150b57cec5SDimitry Andric /// ::= .subsections_via_symbols 8160b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveSubsectionsViaSymbols(StringRef, SMLoc) { 8170b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 8180b57cec5SDimitry Andric return TokError("unexpected token in '.subsections_via_symbols' directive"); 8190b57cec5SDimitry Andric 8200b57cec5SDimitry Andric Lex(); 8210b57cec5SDimitry Andric 8225ffd83dbSDimitry Andric getStreamer().emitAssemblerFlag(MCAF_SubsectionsViaSymbols); 8230b57cec5SDimitry Andric 8240b57cec5SDimitry Andric return false; 8250b57cec5SDimitry Andric } 8260b57cec5SDimitry Andric 8270b57cec5SDimitry Andric /// ParseDirectiveTBSS 8280b57cec5SDimitry Andric /// ::= .tbss identifier, size, align 8290b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveTBSS(StringRef, SMLoc) { 8300b57cec5SDimitry Andric SMLoc IDLoc = getLexer().getLoc(); 8310b57cec5SDimitry Andric StringRef Name; 8320b57cec5SDimitry Andric if (getParser().parseIdentifier(Name)) 8330b57cec5SDimitry Andric return TokError("expected identifier in directive"); 8340b57cec5SDimitry Andric 8350b57cec5SDimitry Andric // Handle the identifier as the key symbol. 8360b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(Name); 8370b57cec5SDimitry Andric 8380b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 8390b57cec5SDimitry Andric return TokError("unexpected token in directive"); 8400b57cec5SDimitry Andric Lex(); 8410b57cec5SDimitry Andric 8420b57cec5SDimitry Andric int64_t Size; 8430b57cec5SDimitry Andric SMLoc SizeLoc = getLexer().getLoc(); 8440b57cec5SDimitry Andric if (getParser().parseAbsoluteExpression(Size)) 8450b57cec5SDimitry Andric return true; 8460b57cec5SDimitry Andric 8470b57cec5SDimitry Andric int64_t Pow2Alignment = 0; 8480b57cec5SDimitry Andric SMLoc Pow2AlignmentLoc; 8490b57cec5SDimitry Andric if (getLexer().is(AsmToken::Comma)) { 8500b57cec5SDimitry Andric Lex(); 8510b57cec5SDimitry Andric Pow2AlignmentLoc = getLexer().getLoc(); 8520b57cec5SDimitry Andric if (getParser().parseAbsoluteExpression(Pow2Alignment)) 8530b57cec5SDimitry Andric return true; 8540b57cec5SDimitry Andric } 8550b57cec5SDimitry Andric 8560b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 8570b57cec5SDimitry Andric return TokError("unexpected token in '.tbss' directive"); 8580b57cec5SDimitry Andric 8590b57cec5SDimitry Andric Lex(); 8600b57cec5SDimitry Andric 8610b57cec5SDimitry Andric if (Size < 0) 8620b57cec5SDimitry Andric return Error(SizeLoc, "invalid '.tbss' directive size, can't be less than" 8630b57cec5SDimitry Andric "zero"); 8640b57cec5SDimitry Andric 8650b57cec5SDimitry Andric // FIXME: Diagnose overflow. 8660b57cec5SDimitry Andric if (Pow2Alignment < 0) 8670b57cec5SDimitry Andric return Error(Pow2AlignmentLoc, "invalid '.tbss' alignment, can't be less" 8680b57cec5SDimitry Andric "than zero"); 8690b57cec5SDimitry Andric 8700b57cec5SDimitry Andric if (!Sym->isUndefined()) 8710b57cec5SDimitry Andric return Error(IDLoc, "invalid symbol redefinition"); 8720b57cec5SDimitry Andric 8735ffd83dbSDimitry Andric getStreamer().emitTBSSSymbol( 8745ffd83dbSDimitry Andric getContext().getMachOSection("__DATA", "__thread_bss", 8755ffd83dbSDimitry Andric MachO::S_THREAD_LOCAL_ZEROFILL, 0, 8765ffd83dbSDimitry Andric SectionKind::getThreadBSS()), 8770b57cec5SDimitry Andric Sym, Size, 1 << Pow2Alignment); 8780b57cec5SDimitry Andric 8790b57cec5SDimitry Andric return false; 8800b57cec5SDimitry Andric } 8810b57cec5SDimitry Andric 8820b57cec5SDimitry Andric /// ParseDirectiveZerofill 8830b57cec5SDimitry Andric /// ::= .zerofill segname , sectname [, identifier , size_expression [ 8840b57cec5SDimitry Andric /// , align_expression ]] 8850b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveZerofill(StringRef, SMLoc) { 8860b57cec5SDimitry Andric StringRef Segment; 8870b57cec5SDimitry Andric if (getParser().parseIdentifier(Segment)) 8880b57cec5SDimitry Andric return TokError("expected segment name after '.zerofill' directive"); 8890b57cec5SDimitry Andric 8900b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 8910b57cec5SDimitry Andric return TokError("unexpected token in directive"); 8920b57cec5SDimitry Andric Lex(); 8930b57cec5SDimitry Andric 8940b57cec5SDimitry Andric StringRef Section; 8950b57cec5SDimitry Andric SMLoc SectionLoc = getLexer().getLoc(); 8960b57cec5SDimitry Andric if (getParser().parseIdentifier(Section)) 8970b57cec5SDimitry Andric return TokError("expected section name after comma in '.zerofill' " 8980b57cec5SDimitry Andric "directive"); 8990b57cec5SDimitry Andric 9000b57cec5SDimitry Andric // If this is the end of the line all that was wanted was to create the 9010b57cec5SDimitry Andric // the section but with no symbol. 9020b57cec5SDimitry Andric if (getLexer().is(AsmToken::EndOfStatement)) { 9030b57cec5SDimitry Andric // Create the zerofill section but no symbol 9045ffd83dbSDimitry Andric getStreamer().emitZerofill( 9050b57cec5SDimitry Andric getContext().getMachOSection(Segment, Section, MachO::S_ZEROFILL, 0, 9060b57cec5SDimitry Andric SectionKind::getBSS()), 9070b57cec5SDimitry Andric /*Symbol=*/nullptr, /*Size=*/0, /*ByteAlignment=*/0, SectionLoc); 9080b57cec5SDimitry Andric return false; 9090b57cec5SDimitry Andric } 9100b57cec5SDimitry Andric 9110b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 9120b57cec5SDimitry Andric return TokError("unexpected token in directive"); 9130b57cec5SDimitry Andric Lex(); 9140b57cec5SDimitry Andric 9150b57cec5SDimitry Andric SMLoc IDLoc = getLexer().getLoc(); 9160b57cec5SDimitry Andric StringRef IDStr; 9170b57cec5SDimitry Andric if (getParser().parseIdentifier(IDStr)) 9180b57cec5SDimitry Andric return TokError("expected identifier in directive"); 9190b57cec5SDimitry Andric 9200b57cec5SDimitry Andric // handle the identifier as the key symbol. 9210b57cec5SDimitry Andric MCSymbol *Sym = getContext().getOrCreateSymbol(IDStr); 9220b57cec5SDimitry Andric 9230b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 9240b57cec5SDimitry Andric return TokError("unexpected token in directive"); 9250b57cec5SDimitry Andric Lex(); 9260b57cec5SDimitry Andric 9270b57cec5SDimitry Andric int64_t Size; 9280b57cec5SDimitry Andric SMLoc SizeLoc = getLexer().getLoc(); 9290b57cec5SDimitry Andric if (getParser().parseAbsoluteExpression(Size)) 9300b57cec5SDimitry Andric return true; 9310b57cec5SDimitry Andric 9320b57cec5SDimitry Andric int64_t Pow2Alignment = 0; 9330b57cec5SDimitry Andric SMLoc Pow2AlignmentLoc; 9340b57cec5SDimitry Andric if (getLexer().is(AsmToken::Comma)) { 9350b57cec5SDimitry Andric Lex(); 9360b57cec5SDimitry Andric Pow2AlignmentLoc = getLexer().getLoc(); 9370b57cec5SDimitry Andric if (getParser().parseAbsoluteExpression(Pow2Alignment)) 9380b57cec5SDimitry Andric return true; 9390b57cec5SDimitry Andric } 9400b57cec5SDimitry Andric 9410b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 9420b57cec5SDimitry Andric return TokError("unexpected token in '.zerofill' directive"); 9430b57cec5SDimitry Andric 9440b57cec5SDimitry Andric Lex(); 9450b57cec5SDimitry Andric 9460b57cec5SDimitry Andric if (Size < 0) 9470b57cec5SDimitry Andric return Error(SizeLoc, "invalid '.zerofill' directive size, can't be less " 9480b57cec5SDimitry Andric "than zero"); 9490b57cec5SDimitry Andric 9500b57cec5SDimitry Andric // NOTE: The alignment in the directive is a power of 2 value, the assembler 9510b57cec5SDimitry Andric // may internally end up wanting an alignment in bytes. 9520b57cec5SDimitry Andric // FIXME: Diagnose overflow. 9530b57cec5SDimitry Andric if (Pow2Alignment < 0) 9540b57cec5SDimitry Andric return Error(Pow2AlignmentLoc, "invalid '.zerofill' directive alignment, " 9550b57cec5SDimitry Andric "can't be less than zero"); 9560b57cec5SDimitry Andric 9570b57cec5SDimitry Andric if (!Sym->isUndefined()) 9580b57cec5SDimitry Andric return Error(IDLoc, "invalid symbol redefinition"); 9590b57cec5SDimitry Andric 9600b57cec5SDimitry Andric // Create the zerofill Symbol with Size and Pow2Alignment 9610b57cec5SDimitry Andric // 9620b57cec5SDimitry Andric // FIXME: Arch specific. 9635ffd83dbSDimitry Andric getStreamer().emitZerofill(getContext().getMachOSection( 9640b57cec5SDimitry Andric Segment, Section, MachO::S_ZEROFILL, 9650b57cec5SDimitry Andric 0, SectionKind::getBSS()), 9660b57cec5SDimitry Andric Sym, Size, 1 << Pow2Alignment, SectionLoc); 9670b57cec5SDimitry Andric 9680b57cec5SDimitry Andric return false; 9690b57cec5SDimitry Andric } 9700b57cec5SDimitry Andric 9710b57cec5SDimitry Andric /// ParseDirectiveDataRegion 9720b57cec5SDimitry Andric /// ::= .data_region [ ( jt8 | jt16 | jt32 ) ] 9730b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveDataRegion(StringRef, SMLoc) { 9740b57cec5SDimitry Andric if (getLexer().is(AsmToken::EndOfStatement)) { 9750b57cec5SDimitry Andric Lex(); 9765ffd83dbSDimitry Andric getStreamer().emitDataRegion(MCDR_DataRegion); 9770b57cec5SDimitry Andric return false; 9780b57cec5SDimitry Andric } 9790b57cec5SDimitry Andric StringRef RegionType; 9800b57cec5SDimitry Andric SMLoc Loc = getParser().getTok().getLoc(); 9810b57cec5SDimitry Andric if (getParser().parseIdentifier(RegionType)) 9820b57cec5SDimitry Andric return TokError("expected region type after '.data_region' directive"); 9830b57cec5SDimitry Andric int Kind = StringSwitch<int>(RegionType) 9840b57cec5SDimitry Andric .Case("jt8", MCDR_DataRegionJT8) 9850b57cec5SDimitry Andric .Case("jt16", MCDR_DataRegionJT16) 9860b57cec5SDimitry Andric .Case("jt32", MCDR_DataRegionJT32) 9870b57cec5SDimitry Andric .Default(-1); 9880b57cec5SDimitry Andric if (Kind == -1) 9890b57cec5SDimitry Andric return Error(Loc, "unknown region type in '.data_region' directive"); 9900b57cec5SDimitry Andric Lex(); 9910b57cec5SDimitry Andric 9925ffd83dbSDimitry Andric getStreamer().emitDataRegion((MCDataRegionType)Kind); 9930b57cec5SDimitry Andric return false; 9940b57cec5SDimitry Andric } 9950b57cec5SDimitry Andric 9960b57cec5SDimitry Andric /// ParseDirectiveDataRegionEnd 9970b57cec5SDimitry Andric /// ::= .end_data_region 9980b57cec5SDimitry Andric bool DarwinAsmParser::parseDirectiveDataRegionEnd(StringRef, SMLoc) { 9990b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::EndOfStatement)) 10000b57cec5SDimitry Andric return TokError("unexpected token in '.end_data_region' directive"); 10010b57cec5SDimitry Andric 10020b57cec5SDimitry Andric Lex(); 10035ffd83dbSDimitry Andric getStreamer().emitDataRegion(MCDR_DataRegionEnd); 10040b57cec5SDimitry Andric return false; 10050b57cec5SDimitry Andric } 10060b57cec5SDimitry Andric 10070b57cec5SDimitry Andric static bool isSDKVersionToken(const AsmToken &Tok) { 10080b57cec5SDimitry Andric return Tok.is(AsmToken::Identifier) && Tok.getIdentifier() == "sdk_version"; 10090b57cec5SDimitry Andric } 10100b57cec5SDimitry Andric 10110b57cec5SDimitry Andric /// parseMajorMinorVersionComponent ::= major, minor 10120b57cec5SDimitry Andric bool DarwinAsmParser::parseMajorMinorVersionComponent(unsigned *Major, 10130b57cec5SDimitry Andric unsigned *Minor, 10140b57cec5SDimitry Andric const char *VersionName) { 10150b57cec5SDimitry Andric // Get the major version number. 10160b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Integer)) 10170b57cec5SDimitry Andric return TokError(Twine("invalid ") + VersionName + 10180b57cec5SDimitry Andric " major version number, integer expected"); 10190b57cec5SDimitry Andric int64_t MajorVal = getLexer().getTok().getIntVal(); 10200b57cec5SDimitry Andric if (MajorVal > 65535 || MajorVal <= 0) 10210b57cec5SDimitry Andric return TokError(Twine("invalid ") + VersionName + " major version number"); 10220b57cec5SDimitry Andric *Major = (unsigned)MajorVal; 10230b57cec5SDimitry Andric Lex(); 10240b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 10250b57cec5SDimitry Andric return TokError(Twine(VersionName) + 10260b57cec5SDimitry Andric " minor version number required, comma expected"); 10270b57cec5SDimitry Andric Lex(); 10280b57cec5SDimitry Andric // Get the minor version number. 10290b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Integer)) 10300b57cec5SDimitry Andric return TokError(Twine("invalid ") + VersionName + 10310b57cec5SDimitry Andric " minor version number, integer expected"); 10320b57cec5SDimitry Andric int64_t MinorVal = getLexer().getTok().getIntVal(); 10330b57cec5SDimitry Andric if (MinorVal > 255 || MinorVal < 0) 10340b57cec5SDimitry Andric return TokError(Twine("invalid ") + VersionName + " minor version number"); 10350b57cec5SDimitry Andric *Minor = MinorVal; 10360b57cec5SDimitry Andric Lex(); 10370b57cec5SDimitry Andric return false; 10380b57cec5SDimitry Andric } 10390b57cec5SDimitry Andric 10400b57cec5SDimitry Andric /// parseOptionalTrailingVersionComponent ::= , version_number 10410b57cec5SDimitry Andric bool DarwinAsmParser::parseOptionalTrailingVersionComponent( 10420b57cec5SDimitry Andric unsigned *Component, const char *ComponentName) { 10430b57cec5SDimitry Andric assert(getLexer().is(AsmToken::Comma) && "comma expected"); 10440b57cec5SDimitry Andric Lex(); 10450b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Integer)) 10460b57cec5SDimitry Andric return TokError(Twine("invalid ") + ComponentName + 10470b57cec5SDimitry Andric " version number, integer expected"); 10480b57cec5SDimitry Andric int64_t Val = getLexer().getTok().getIntVal(); 10490b57cec5SDimitry Andric if (Val > 255 || Val < 0) 10500b57cec5SDimitry Andric return TokError(Twine("invalid ") + ComponentName + " version number"); 10510b57cec5SDimitry Andric *Component = Val; 10520b57cec5SDimitry Andric Lex(); 10530b57cec5SDimitry Andric return false; 10540b57cec5SDimitry Andric } 10550b57cec5SDimitry Andric 10560b57cec5SDimitry Andric /// parseVersion ::= parseMajorMinorVersionComponent 10570b57cec5SDimitry Andric /// parseOptionalTrailingVersionComponent 10580b57cec5SDimitry Andric bool DarwinAsmParser::parseVersion(unsigned *Major, unsigned *Minor, 10590b57cec5SDimitry Andric unsigned *Update) { 10600b57cec5SDimitry Andric if (parseMajorMinorVersionComponent(Major, Minor, "OS")) 10610b57cec5SDimitry Andric return true; 10620b57cec5SDimitry Andric 10630b57cec5SDimitry Andric // Get the update level, if specified 10640b57cec5SDimitry Andric *Update = 0; 10650b57cec5SDimitry Andric if (getLexer().is(AsmToken::EndOfStatement) || 10660b57cec5SDimitry Andric isSDKVersionToken(getLexer().getTok())) 10670b57cec5SDimitry Andric return false; 10680b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 10690b57cec5SDimitry Andric return TokError("invalid OS update specifier, comma expected"); 10700b57cec5SDimitry Andric if (parseOptionalTrailingVersionComponent(Update, "OS update")) 10710b57cec5SDimitry Andric return true; 10720b57cec5SDimitry Andric return false; 10730b57cec5SDimitry Andric } 10740b57cec5SDimitry Andric 10750b57cec5SDimitry Andric bool DarwinAsmParser::parseSDKVersion(VersionTuple &SDKVersion) { 10760b57cec5SDimitry Andric assert(isSDKVersionToken(getLexer().getTok()) && "expected sdk_version"); 10770b57cec5SDimitry Andric Lex(); 10780b57cec5SDimitry Andric unsigned Major, Minor; 10790b57cec5SDimitry Andric if (parseMajorMinorVersionComponent(&Major, &Minor, "SDK")) 10800b57cec5SDimitry Andric return true; 10810b57cec5SDimitry Andric SDKVersion = VersionTuple(Major, Minor); 10820b57cec5SDimitry Andric 10830b57cec5SDimitry Andric // Get the subminor version, if specified. 10840b57cec5SDimitry Andric if (getLexer().is(AsmToken::Comma)) { 10850b57cec5SDimitry Andric unsigned Subminor; 10860b57cec5SDimitry Andric if (parseOptionalTrailingVersionComponent(&Subminor, "SDK subminor")) 10870b57cec5SDimitry Andric return true; 10880b57cec5SDimitry Andric SDKVersion = VersionTuple(Major, Minor, Subminor); 10890b57cec5SDimitry Andric } 10900b57cec5SDimitry Andric return false; 10910b57cec5SDimitry Andric } 10920b57cec5SDimitry Andric 10930b57cec5SDimitry Andric void DarwinAsmParser::checkVersion(StringRef Directive, StringRef Arg, 10940b57cec5SDimitry Andric SMLoc Loc, Triple::OSType ExpectedOS) { 10950b57cec5SDimitry Andric const Triple &Target = getContext().getObjectFileInfo()->getTargetTriple(); 10960b57cec5SDimitry Andric if (Target.getOS() != ExpectedOS) 10970b57cec5SDimitry Andric Warning(Loc, Twine(Directive) + 10980b57cec5SDimitry Andric (Arg.empty() ? Twine() : Twine(' ') + Arg) + 10990b57cec5SDimitry Andric " used while targeting " + Target.getOSName()); 11000b57cec5SDimitry Andric 11010b57cec5SDimitry Andric if (LastVersionDirective.isValid()) { 11020b57cec5SDimitry Andric Warning(Loc, "overriding previous version directive"); 11030b57cec5SDimitry Andric Note(LastVersionDirective, "previous definition is here"); 11040b57cec5SDimitry Andric } 11050b57cec5SDimitry Andric LastVersionDirective = Loc; 11060b57cec5SDimitry Andric } 11070b57cec5SDimitry Andric 11080b57cec5SDimitry Andric static Triple::OSType getOSTypeFromMCVM(MCVersionMinType Type) { 11090b57cec5SDimitry Andric switch (Type) { 11100b57cec5SDimitry Andric case MCVM_WatchOSVersionMin: return Triple::WatchOS; 11110b57cec5SDimitry Andric case MCVM_TvOSVersionMin: return Triple::TvOS; 11120b57cec5SDimitry Andric case MCVM_IOSVersionMin: return Triple::IOS; 11130b57cec5SDimitry Andric case MCVM_OSXVersionMin: return Triple::MacOSX; 11140b57cec5SDimitry Andric } 11150b57cec5SDimitry Andric llvm_unreachable("Invalid mc version min type"); 11160b57cec5SDimitry Andric } 11170b57cec5SDimitry Andric 11180b57cec5SDimitry Andric /// parseVersionMin 11190b57cec5SDimitry Andric /// ::= .ios_version_min parseVersion parseSDKVersion 11200b57cec5SDimitry Andric /// | .macosx_version_min parseVersion parseSDKVersion 11210b57cec5SDimitry Andric /// | .tvos_version_min parseVersion parseSDKVersion 11220b57cec5SDimitry Andric /// | .watchos_version_min parseVersion parseSDKVersion 11230b57cec5SDimitry Andric bool DarwinAsmParser::parseVersionMin(StringRef Directive, SMLoc Loc, 11240b57cec5SDimitry Andric MCVersionMinType Type) { 11250b57cec5SDimitry Andric unsigned Major; 11260b57cec5SDimitry Andric unsigned Minor; 11270b57cec5SDimitry Andric unsigned Update; 11280b57cec5SDimitry Andric if (parseVersion(&Major, &Minor, &Update)) 11290b57cec5SDimitry Andric return true; 11300b57cec5SDimitry Andric 11310b57cec5SDimitry Andric VersionTuple SDKVersion; 11320b57cec5SDimitry Andric if (isSDKVersionToken(getLexer().getTok()) && parseSDKVersion(SDKVersion)) 11330b57cec5SDimitry Andric return true; 11340b57cec5SDimitry Andric 11350b57cec5SDimitry Andric if (parseToken(AsmToken::EndOfStatement)) 11360b57cec5SDimitry Andric return addErrorSuffix(Twine(" in '") + Directive + "' directive"); 11370b57cec5SDimitry Andric 11380b57cec5SDimitry Andric Triple::OSType ExpectedOS = getOSTypeFromMCVM(Type); 11390b57cec5SDimitry Andric checkVersion(Directive, StringRef(), Loc, ExpectedOS); 11405ffd83dbSDimitry Andric getStreamer().emitVersionMin(Type, Major, Minor, Update, SDKVersion); 11410b57cec5SDimitry Andric return false; 11420b57cec5SDimitry Andric } 11430b57cec5SDimitry Andric 11440b57cec5SDimitry Andric static Triple::OSType getOSTypeFromPlatform(MachO::PlatformType Type) { 11450b57cec5SDimitry Andric switch (Type) { 11460b57cec5SDimitry Andric case MachO::PLATFORM_MACOS: return Triple::MacOSX; 11470b57cec5SDimitry Andric case MachO::PLATFORM_IOS: return Triple::IOS; 11480b57cec5SDimitry Andric case MachO::PLATFORM_TVOS: return Triple::TvOS; 11490b57cec5SDimitry Andric case MachO::PLATFORM_WATCHOS: return Triple::WatchOS; 11500b57cec5SDimitry Andric case MachO::PLATFORM_BRIDGEOS: /* silence warning */ break; 11510b57cec5SDimitry Andric case MachO::PLATFORM_MACCATALYST: return Triple::IOS; 11520b57cec5SDimitry Andric case MachO::PLATFORM_IOSSIMULATOR: /* silence warning */ break; 11530b57cec5SDimitry Andric case MachO::PLATFORM_TVOSSIMULATOR: /* silence warning */ break; 11540b57cec5SDimitry Andric case MachO::PLATFORM_WATCHOSSIMULATOR: /* silence warning */ break; 1155*e8d8bef9SDimitry Andric case MachO::PLATFORM_DRIVERKIT: /* silence warning */ break; 11560b57cec5SDimitry Andric } 11570b57cec5SDimitry Andric llvm_unreachable("Invalid mach-o platform type"); 11580b57cec5SDimitry Andric } 11590b57cec5SDimitry Andric 11600b57cec5SDimitry Andric /// parseBuildVersion 11610b57cec5SDimitry Andric /// ::= .build_version (macos|ios|tvos|watchos), parseVersion parseSDKVersion 11620b57cec5SDimitry Andric bool DarwinAsmParser::parseBuildVersion(StringRef Directive, SMLoc Loc) { 11630b57cec5SDimitry Andric StringRef PlatformName; 11640b57cec5SDimitry Andric SMLoc PlatformLoc = getTok().getLoc(); 11650b57cec5SDimitry Andric if (getParser().parseIdentifier(PlatformName)) 11660b57cec5SDimitry Andric return TokError("platform name expected"); 11670b57cec5SDimitry Andric 11680b57cec5SDimitry Andric unsigned Platform = StringSwitch<unsigned>(PlatformName) 11690b57cec5SDimitry Andric .Case("macos", MachO::PLATFORM_MACOS) 11700b57cec5SDimitry Andric .Case("ios", MachO::PLATFORM_IOS) 11710b57cec5SDimitry Andric .Case("tvos", MachO::PLATFORM_TVOS) 11720b57cec5SDimitry Andric .Case("watchos", MachO::PLATFORM_WATCHOS) 11730b57cec5SDimitry Andric .Case("macCatalyst", MachO::PLATFORM_MACCATALYST) 11740b57cec5SDimitry Andric .Default(0); 11750b57cec5SDimitry Andric if (Platform == 0) 11760b57cec5SDimitry Andric return Error(PlatformLoc, "unknown platform name"); 11770b57cec5SDimitry Andric 11780b57cec5SDimitry Andric if (getLexer().isNot(AsmToken::Comma)) 11790b57cec5SDimitry Andric return TokError("version number required, comma expected"); 11800b57cec5SDimitry Andric Lex(); 11810b57cec5SDimitry Andric 11820b57cec5SDimitry Andric unsigned Major; 11830b57cec5SDimitry Andric unsigned Minor; 11840b57cec5SDimitry Andric unsigned Update; 11850b57cec5SDimitry Andric if (parseVersion(&Major, &Minor, &Update)) 11860b57cec5SDimitry Andric return true; 11870b57cec5SDimitry Andric 11880b57cec5SDimitry Andric VersionTuple SDKVersion; 11890b57cec5SDimitry Andric if (isSDKVersionToken(getLexer().getTok()) && parseSDKVersion(SDKVersion)) 11900b57cec5SDimitry Andric return true; 11910b57cec5SDimitry Andric 11920b57cec5SDimitry Andric if (parseToken(AsmToken::EndOfStatement)) 11930b57cec5SDimitry Andric return addErrorSuffix(" in '.build_version' directive"); 11940b57cec5SDimitry Andric 11950b57cec5SDimitry Andric Triple::OSType ExpectedOS 11960b57cec5SDimitry Andric = getOSTypeFromPlatform((MachO::PlatformType)Platform); 11970b57cec5SDimitry Andric checkVersion(Directive, PlatformName, Loc, ExpectedOS); 11985ffd83dbSDimitry Andric getStreamer().emitBuildVersion(Platform, Major, Minor, Update, SDKVersion); 11990b57cec5SDimitry Andric return false; 12000b57cec5SDimitry Andric } 12010b57cec5SDimitry Andric 12020b57cec5SDimitry Andric 12030b57cec5SDimitry Andric namespace llvm { 12040b57cec5SDimitry Andric 12050b57cec5SDimitry Andric MCAsmParserExtension *createDarwinAsmParser() { 12060b57cec5SDimitry Andric return new DarwinAsmParser; 12070b57cec5SDimitry Andric } 12080b57cec5SDimitry Andric 12090b57cec5SDimitry Andric } // end llvm namespace 1210