|  | //===- DarwinAsmParser.cpp - Darwin (Mach-O) Assembly Parser --------------===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "llvm/MC/MCParser/MCAsmParserExtension.h" | 
|  | #include "llvm/ADT/StringRef.h" | 
|  | #include "llvm/ADT/StringSwitch.h" | 
|  | #include "llvm/ADT/Twine.h" | 
|  | #include "llvm/MC/MCContext.h" | 
|  | #include "llvm/MC/MCParser/MCAsmLexer.h" | 
|  | #include "llvm/MC/MCParser/MCAsmParser.h" | 
|  | #include "llvm/MC/MCSectionMachO.h" | 
|  | #include "llvm/MC/MCStreamer.h" | 
|  | #include "llvm/MC/MCSymbol.h" | 
|  | #include "llvm/Support/MemoryBuffer.h" | 
|  | #include "llvm/Support/SourceMgr.h" | 
|  | using namespace llvm; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | /// \brief Implementation of directive handling which is shared across all | 
|  | /// Darwin targets. | 
|  | class DarwinAsmParser : public MCAsmParserExtension { | 
|  | template<bool (DarwinAsmParser::*HandlerMethod)(StringRef, SMLoc)> | 
|  | void addDirectiveHandler(StringRef Directive) { | 
|  | MCAsmParser::ExtensionDirectiveHandler Handler = std::make_pair( | 
|  | this, HandleDirective<DarwinAsmParser, HandlerMethod>); | 
|  | getParser().addDirectiveHandler(Directive, Handler); | 
|  | } | 
|  |  | 
|  | bool ParseSectionSwitch(const char *Segment, const char *Section, | 
|  | unsigned TAA = 0, unsigned ImplicitAlign = 0, | 
|  | unsigned StubSize = 0); | 
|  |  | 
|  | public: | 
|  | DarwinAsmParser() {} | 
|  |  | 
|  | virtual void Initialize(MCAsmParser &Parser) { | 
|  | // Call the base implementation. | 
|  | this->MCAsmParserExtension::Initialize(Parser); | 
|  |  | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveDesc>(".desc"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveIndirectSymbol>( | 
|  | ".indirect_symbol"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveLsym>(".lsym"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveSubsectionsViaSymbols>( | 
|  | ".subsections_via_symbols"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveDumpOrLoad>(".dump"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveDumpOrLoad>(".load"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveSection>(".section"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectivePushSection>( | 
|  | ".pushsection"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectivePopSection>( | 
|  | ".popsection"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectivePrevious>(".previous"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveSecureLogUnique>( | 
|  | ".secure_log_unique"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveSecureLogReset>( | 
|  | ".secure_log_reset"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveTBSS>(".tbss"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveZerofill>(".zerofill"); | 
|  |  | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveDataRegion>( | 
|  | ".data_region"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveDataRegionEnd>( | 
|  | ".end_data_region"); | 
|  |  | 
|  | // Special section directives. | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveBss>(".bss"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveConst>(".const"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveConstData>( | 
|  | ".const_data"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveConstructor>( | 
|  | ".constructor"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveCString>( | 
|  | ".cstring"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveData>(".data"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveDestructor>( | 
|  | ".destructor"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveDyld>(".dyld"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveFVMLibInit0>( | 
|  | ".fvmlib_init0"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveFVMLibInit1>( | 
|  | ".fvmlib_init1"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveLazySymbolPointers>( | 
|  | ".lazy_symbol_pointer"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseDirectiveLinkerOption>( | 
|  | ".linker_option"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveLiteral16>( | 
|  | ".literal16"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveLiteral4>( | 
|  | ".literal4"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveLiteral8>( | 
|  | ".literal8"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveModInitFunc>( | 
|  | ".mod_init_func"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveModTermFunc>( | 
|  | ".mod_term_func"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveNonLazySymbolPointers>( | 
|  | ".non_lazy_symbol_pointer"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCCatClsMeth>( | 
|  | ".objc_cat_cls_meth"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCCatInstMeth>( | 
|  | ".objc_cat_inst_meth"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCCategory>( | 
|  | ".objc_category"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCClass>( | 
|  | ".objc_class"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCClassNames>( | 
|  | ".objc_class_names"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCClassVars>( | 
|  | ".objc_class_vars"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCClsMeth>( | 
|  | ".objc_cls_meth"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCClsRefs>( | 
|  | ".objc_cls_refs"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCInstMeth>( | 
|  | ".objc_inst_meth"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveObjCInstanceVars>( | 
|  | ".objc_instance_vars"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCMessageRefs>( | 
|  | ".objc_message_refs"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCMetaClass>( | 
|  | ".objc_meta_class"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveObjCMethVarNames>( | 
|  | ".objc_meth_var_names"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveObjCMethVarTypes>( | 
|  | ".objc_meth_var_types"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCModuleInfo>( | 
|  | ".objc_module_info"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCProtocol>( | 
|  | ".objc_protocol"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveObjCSelectorStrs>( | 
|  | ".objc_selector_strs"); | 
|  | addDirectiveHandler< | 
|  | &DarwinAsmParser::ParseSectionDirectiveObjCStringObject>( | 
|  | ".objc_string_object"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveObjCSymbols>( | 
|  | ".objc_symbols"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectivePICSymbolStub>( | 
|  | ".picsymbol_stub"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveStaticConst>( | 
|  | ".static_const"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveStaticData>( | 
|  | ".static_data"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveSymbolStub>( | 
|  | ".symbol_stub"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveTData>(".tdata"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveText>(".text"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveThreadInitFunc>( | 
|  | ".thread_init_func"); | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveTLV>(".tlv"); | 
|  |  | 
|  | addDirectiveHandler<&DarwinAsmParser::ParseSectionDirectiveIdent>(".ident"); | 
|  | } | 
|  |  | 
|  | bool ParseDirectiveDesc(StringRef, SMLoc); | 
|  | bool ParseDirectiveIndirectSymbol(StringRef, SMLoc); | 
|  | bool ParseDirectiveDumpOrLoad(StringRef, SMLoc); | 
|  | bool ParseDirectiveLsym(StringRef, SMLoc); | 
|  | bool ParseDirectiveLinkerOption(StringRef, SMLoc); | 
|  | bool ParseDirectiveSection(StringRef, SMLoc); | 
|  | bool ParseDirectivePushSection(StringRef, SMLoc); | 
|  | bool ParseDirectivePopSection(StringRef, SMLoc); | 
|  | bool ParseDirectivePrevious(StringRef, SMLoc); | 
|  | bool ParseDirectiveSecureLogReset(StringRef, SMLoc); | 
|  | bool ParseDirectiveSecureLogUnique(StringRef, SMLoc); | 
|  | bool ParseDirectiveSubsectionsViaSymbols(StringRef, SMLoc); | 
|  | bool ParseDirectiveTBSS(StringRef, SMLoc); | 
|  | bool ParseDirectiveZerofill(StringRef, SMLoc); | 
|  | bool ParseDirectiveDataRegion(StringRef, SMLoc); | 
|  | bool ParseDirectiveDataRegionEnd(StringRef, SMLoc); | 
|  |  | 
|  | // Named Section Directive | 
|  | bool ParseSectionDirectiveBss(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__bss"); | 
|  | } | 
|  |  | 
|  | bool ParseSectionDirectiveConst(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__const"); | 
|  | } | 
|  | bool ParseSectionDirectiveStaticConst(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__static_const"); | 
|  | } | 
|  | bool ParseSectionDirectiveCString(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__cstring", | 
|  | MCSectionMachO::S_CSTRING_LITERALS); | 
|  | } | 
|  | bool ParseSectionDirectiveLiteral4(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__literal4", | 
|  | MCSectionMachO::S_4BYTE_LITERALS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveLiteral8(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__literal8", | 
|  | MCSectionMachO::S_8BYTE_LITERALS, 8); | 
|  | } | 
|  | bool ParseSectionDirectiveLiteral16(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__literal16", | 
|  | MCSectionMachO::S_16BYTE_LITERALS, 16); | 
|  | } | 
|  | bool ParseSectionDirectiveConstructor(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__constructor"); | 
|  | } | 
|  | bool ParseSectionDirectiveDestructor(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__destructor"); | 
|  | } | 
|  | bool ParseSectionDirectiveFVMLibInit0(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__fvmlib_init0"); | 
|  | } | 
|  | bool ParseSectionDirectiveFVMLibInit1(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__fvmlib_init1"); | 
|  | } | 
|  | bool ParseSectionDirectiveSymbolStub(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__symbol_stub", | 
|  | MCSectionMachO::S_SYMBOL_STUBS | | 
|  | MCSectionMachO::S_ATTR_PURE_INSTRUCTIONS, | 
|  | // FIXME: Different on PPC and ARM. | 
|  | 0, 16); | 
|  | } | 
|  | bool ParseSectionDirectivePICSymbolStub(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT","__picsymbol_stub", | 
|  | MCSectionMachO::S_SYMBOL_STUBS | | 
|  | MCSectionMachO::S_ATTR_PURE_INSTRUCTIONS, 0, 26); | 
|  | } | 
|  | bool ParseSectionDirectiveData(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__data"); | 
|  | } | 
|  | bool ParseSectionDirectiveStaticData(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__static_data"); | 
|  | } | 
|  | bool ParseSectionDirectiveNonLazySymbolPointers(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__nl_symbol_ptr", | 
|  | MCSectionMachO::S_NON_LAZY_SYMBOL_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveLazySymbolPointers(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__la_symbol_ptr", | 
|  | MCSectionMachO::S_LAZY_SYMBOL_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveDyld(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__dyld"); | 
|  | } | 
|  | bool ParseSectionDirectiveModInitFunc(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__mod_init_func", | 
|  | MCSectionMachO::S_MOD_INIT_FUNC_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveModTermFunc(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__mod_term_func", | 
|  | MCSectionMachO::S_MOD_TERM_FUNC_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveConstData(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__const"); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCClass(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__class", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCMetaClass(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__meta_class", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCCatClsMeth(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__cat_cls_meth", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCCatInstMeth(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__cat_inst_meth", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCProtocol(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__protocol", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCStringObject(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__string_object", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCClsMeth(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__cls_meth", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCInstMeth(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__inst_meth", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCClsRefs(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__cls_refs", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP | | 
|  | MCSectionMachO::S_LITERAL_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCMessageRefs(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__message_refs", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP | | 
|  | MCSectionMachO::S_LITERAL_POINTERS, 4); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCSymbols(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__symbols", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCCategory(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__category", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCClassVars(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__class_vars", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCInstanceVars(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__instance_vars", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCModuleInfo(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__module_info", | 
|  | MCSectionMachO::S_ATTR_NO_DEAD_STRIP); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCClassNames(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__cstring", | 
|  | MCSectionMachO::S_CSTRING_LITERALS); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCMethVarTypes(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__cstring", | 
|  | MCSectionMachO::S_CSTRING_LITERALS); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCMethVarNames(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__cstring", | 
|  | MCSectionMachO::S_CSTRING_LITERALS); | 
|  | } | 
|  | bool ParseSectionDirectiveObjCSelectorStrs(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__OBJC", "__selector_strs", | 
|  | MCSectionMachO::S_CSTRING_LITERALS); | 
|  | } | 
|  | bool ParseSectionDirectiveTData(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__thread_data", | 
|  | MCSectionMachO::S_THREAD_LOCAL_REGULAR); | 
|  | } | 
|  | bool ParseSectionDirectiveText(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__TEXT", "__text", | 
|  | MCSectionMachO::S_ATTR_PURE_INSTRUCTIONS); | 
|  | } | 
|  | bool ParseSectionDirectiveTLV(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__thread_vars", | 
|  | MCSectionMachO::S_THREAD_LOCAL_VARIABLES); | 
|  | } | 
|  | bool ParseSectionDirectiveIdent(StringRef, SMLoc) { | 
|  | // Darwin silently ignores the .ident directive. | 
|  | getParser().eatToEndOfStatement(); | 
|  | return false; | 
|  | } | 
|  | bool ParseSectionDirectiveThreadInitFunc(StringRef, SMLoc) { | 
|  | return ParseSectionSwitch("__DATA", "__thread_init", | 
|  | MCSectionMachO::S_THREAD_LOCAL_INIT_FUNCTION_POINTERS); | 
|  | } | 
|  |  | 
|  | }; | 
|  |  | 
|  | } // end anonymous namespace | 
|  |  | 
|  | bool DarwinAsmParser::ParseSectionSwitch(const char *Segment, | 
|  | const char *Section, | 
|  | unsigned TAA, unsigned Align, | 
|  | unsigned StubSize) { | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in section switching directive"); | 
|  | Lex(); | 
|  |  | 
|  | // FIXME: Arch specific. | 
|  | bool isText = TAA & MCSectionMachO::S_ATTR_PURE_INSTRUCTIONS; | 
|  | getStreamer().SwitchSection(getContext().getMachOSection( | 
|  | Segment, Section, TAA, StubSize, | 
|  | isText ? SectionKind::getText() | 
|  | : SectionKind::getDataRel())); | 
|  |  | 
|  | // Set the implicit alignment, if any. | 
|  | // | 
|  | // FIXME: This isn't really what 'as' does; I think it just uses the implicit | 
|  | // alignment on the section (e.g., if one manually inserts bytes into the | 
|  | // section, then just issuing the section switch directive will not realign | 
|  | // the section. However, this is arguably more reasonable behavior, and there | 
|  | // is no good reason for someone to intentionally emit incorrectly sized | 
|  | // values into the implicitly aligned sections. | 
|  | if (Align) | 
|  | getStreamer().EmitValueToAlignment(Align); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveDesc | 
|  | ///  ::= .desc identifier , expression | 
|  | bool DarwinAsmParser::ParseDirectiveDesc(StringRef, SMLoc) { | 
|  | StringRef Name; | 
|  | if (getParser().parseIdentifier(Name)) | 
|  | return TokError("expected identifier in directive"); | 
|  |  | 
|  | // Handle the identifier as the key symbol. | 
|  | MCSymbol *Sym = getContext().GetOrCreateSymbol(Name); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in '.desc' directive"); | 
|  | Lex(); | 
|  |  | 
|  | int64_t DescValue; | 
|  | if (getParser().parseAbsoluteExpression(DescValue)) | 
|  | return true; | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.desc' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | // Set the n_desc field of this Symbol to this DescValue | 
|  | getStreamer().EmitSymbolDesc(Sym, DescValue); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveIndirectSymbol | 
|  | ///  ::= .indirect_symbol identifier | 
|  | bool DarwinAsmParser::ParseDirectiveIndirectSymbol(StringRef, SMLoc Loc) { | 
|  | const MCSectionMachO *Current = static_cast<const MCSectionMachO*>( | 
|  | getStreamer().getCurrentSection().first); | 
|  | unsigned SectionType = Current->getType(); | 
|  | if (SectionType != MCSectionMachO::S_NON_LAZY_SYMBOL_POINTERS && | 
|  | SectionType != MCSectionMachO::S_LAZY_SYMBOL_POINTERS && | 
|  | SectionType != MCSectionMachO::S_SYMBOL_STUBS) | 
|  | return Error(Loc, "indirect symbol not in a symbol pointer or stub " | 
|  | "section"); | 
|  |  | 
|  | StringRef Name; | 
|  | if (getParser().parseIdentifier(Name)) | 
|  | return TokError("expected identifier in .indirect_symbol directive"); | 
|  |  | 
|  | MCSymbol *Sym = getContext().GetOrCreateSymbol(Name); | 
|  |  | 
|  | // Assembler local symbols don't make any sense here. Complain loudly. | 
|  | if (Sym->isTemporary()) | 
|  | return TokError("non-local symbol required in directive"); | 
|  |  | 
|  | if (!getStreamer().EmitSymbolAttribute(Sym, MCSA_IndirectSymbol)) | 
|  | return TokError("unable to emit indirect symbol attribute for: " + Name); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.indirect_symbol' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveDumpOrLoad | 
|  | ///  ::= ( .dump | .load ) "filename" | 
|  | bool DarwinAsmParser::ParseDirectiveDumpOrLoad(StringRef Directive, | 
|  | SMLoc IDLoc) { | 
|  | bool IsDump = Directive == ".dump"; | 
|  | if (getLexer().isNot(AsmToken::String)) | 
|  | return TokError("expected string in '.dump' or '.load' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.dump' or '.load' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | // FIXME: If/when .dump and .load are implemented they will be done in the | 
|  | // the assembly parser and not have any need for an MCStreamer API. | 
|  | if (IsDump) | 
|  | return Warning(IDLoc, "ignoring directive .dump for now"); | 
|  | else | 
|  | return Warning(IDLoc, "ignoring directive .load for now"); | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveLinkerOption | 
|  | ///  ::= .linker_option "string" ( , "string" )* | 
|  | bool DarwinAsmParser::ParseDirectiveLinkerOption(StringRef IDVal, SMLoc) { | 
|  | SmallVector<std::string, 4> Args; | 
|  | for (;;) { | 
|  | if (getLexer().isNot(AsmToken::String)) | 
|  | return TokError("expected string in '" + Twine(IDVal) + "' directive"); | 
|  |  | 
|  | std::string Data; | 
|  | if (getParser().parseEscapedString(Data)) | 
|  | return true; | 
|  |  | 
|  | Args.push_back(Data); | 
|  |  | 
|  | Lex(); | 
|  | if (getLexer().is(AsmToken::EndOfStatement)) | 
|  | break; | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in '" + Twine(IDVal) + "' directive"); | 
|  | Lex(); | 
|  | } | 
|  |  | 
|  | getStreamer().EmitLinkerOptions(Args); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveLsym | 
|  | ///  ::= .lsym identifier , expression | 
|  | bool DarwinAsmParser::ParseDirectiveLsym(StringRef, SMLoc) { | 
|  | StringRef Name; | 
|  | if (getParser().parseIdentifier(Name)) | 
|  | return TokError("expected identifier in directive"); | 
|  |  | 
|  | // Handle the identifier as the key symbol. | 
|  | MCSymbol *Sym = getContext().GetOrCreateSymbol(Name); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in '.lsym' directive"); | 
|  | Lex(); | 
|  |  | 
|  | const MCExpr *Value; | 
|  | if (getParser().parseExpression(Value)) | 
|  | return true; | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.lsym' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | // We don't currently support this directive. | 
|  | // | 
|  | // FIXME: Diagnostic location! | 
|  | (void) Sym; | 
|  | return TokError("directive '.lsym' is unsupported"); | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveSection: | 
|  | ///   ::= .section identifier (',' identifier)* | 
|  | bool DarwinAsmParser::ParseDirectiveSection(StringRef, SMLoc) { | 
|  | SMLoc Loc = getLexer().getLoc(); | 
|  |  | 
|  | StringRef SectionName; | 
|  | if (getParser().parseIdentifier(SectionName)) | 
|  | return Error(Loc, "expected identifier after '.section' directive"); | 
|  |  | 
|  | // Verify there is a following comma. | 
|  | if (!getLexer().is(AsmToken::Comma)) | 
|  | return TokError("unexpected token in '.section' directive"); | 
|  |  | 
|  | std::string SectionSpec = SectionName; | 
|  | SectionSpec += ","; | 
|  |  | 
|  | // Add all the tokens until the end of the line, ParseSectionSpecifier will | 
|  | // handle this. | 
|  | StringRef EOL = getLexer().LexUntilEndOfStatement(); | 
|  | SectionSpec.append(EOL.begin(), EOL.end()); | 
|  |  | 
|  | Lex(); | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.section' directive"); | 
|  | Lex(); | 
|  |  | 
|  |  | 
|  | StringRef Segment, Section; | 
|  | unsigned StubSize; | 
|  | unsigned TAA; | 
|  | bool TAAParsed; | 
|  | std::string ErrorStr = | 
|  | MCSectionMachO::ParseSectionSpecifier(SectionSpec, Segment, Section, | 
|  | TAA, TAAParsed, StubSize); | 
|  |  | 
|  | if (!ErrorStr.empty()) | 
|  | return Error(Loc, ErrorStr.c_str()); | 
|  |  | 
|  | // FIXME: Arch specific. | 
|  | bool isText = Segment == "__TEXT";  // FIXME: Hack. | 
|  | getStreamer().SwitchSection(getContext().getMachOSection( | 
|  | Segment, Section, TAA, StubSize, | 
|  | isText ? SectionKind::getText() | 
|  | : SectionKind::getDataRel())); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectivePushSection: | 
|  | ///   ::= .pushsection identifier (',' identifier)* | 
|  | bool DarwinAsmParser::ParseDirectivePushSection(StringRef S, SMLoc Loc) { | 
|  | getStreamer().PushSection(); | 
|  |  | 
|  | if (ParseDirectiveSection(S, Loc)) { | 
|  | getStreamer().PopSection(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectivePopSection: | 
|  | ///   ::= .popsection | 
|  | bool DarwinAsmParser::ParseDirectivePopSection(StringRef, SMLoc) { | 
|  | if (!getStreamer().PopSection()) | 
|  | return TokError(".popsection without corresponding .pushsection"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectivePrevious: | 
|  | ///   ::= .previous | 
|  | bool DarwinAsmParser::ParseDirectivePrevious(StringRef DirName, SMLoc) { | 
|  | MCSectionSubPair PreviousSection = getStreamer().getPreviousSection(); | 
|  | if (PreviousSection.first == NULL) | 
|  | return TokError(".previous without corresponding .section"); | 
|  | getStreamer().SwitchSection(PreviousSection.first, PreviousSection.second); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveSecureLogUnique | 
|  | ///  ::= .secure_log_unique ... message ... | 
|  | bool DarwinAsmParser::ParseDirectiveSecureLogUnique(StringRef, SMLoc IDLoc) { | 
|  | StringRef LogMessage = getParser().parseStringToEndOfStatement(); | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.secure_log_unique' directive"); | 
|  |  | 
|  | if (getContext().getSecureLogUsed() != false) | 
|  | return Error(IDLoc, ".secure_log_unique specified multiple times"); | 
|  |  | 
|  | // Get the secure log path. | 
|  | const char *SecureLogFile = getContext().getSecureLogFile(); | 
|  | if (SecureLogFile == NULL) | 
|  | return Error(IDLoc, ".secure_log_unique used but AS_SECURE_LOG_FILE " | 
|  | "environment variable unset."); | 
|  |  | 
|  | // Open the secure log file if we haven't already. | 
|  | raw_ostream *OS = getContext().getSecureLog(); | 
|  | if (OS == NULL) { | 
|  | std::string Err; | 
|  | OS = new raw_fd_ostream(SecureLogFile, Err, | 
|  | sys::fs::F_Append | sys::fs::F_Text); | 
|  | if (!Err.empty()) { | 
|  | delete OS; | 
|  | return Error(IDLoc, Twine("can't open secure log file: ") + | 
|  | SecureLogFile + " (" + Err + ")"); | 
|  | } | 
|  | getContext().setSecureLog(OS); | 
|  | } | 
|  |  | 
|  | // Write the message. | 
|  | int CurBuf = getSourceManager().FindBufferContainingLoc(IDLoc); | 
|  | *OS << getSourceManager().getBufferInfo(CurBuf).Buffer->getBufferIdentifier() | 
|  | << ":" << getSourceManager().FindLineNumber(IDLoc, CurBuf) << ":" | 
|  | << LogMessage + "\n"; | 
|  |  | 
|  | getContext().setSecureLogUsed(true); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveSecureLogReset | 
|  | ///  ::= .secure_log_reset | 
|  | bool DarwinAsmParser::ParseDirectiveSecureLogReset(StringRef, SMLoc IDLoc) { | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.secure_log_reset' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | getContext().setSecureLogUsed(false); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveSubsectionsViaSymbols | 
|  | ///  ::= .subsections_via_symbols | 
|  | bool DarwinAsmParser::ParseDirectiveSubsectionsViaSymbols(StringRef, SMLoc) { | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.subsections_via_symbols' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | getStreamer().EmitAssemblerFlag(MCAF_SubsectionsViaSymbols); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveTBSS | 
|  | ///  ::= .tbss identifier, size, align | 
|  | bool DarwinAsmParser::ParseDirectiveTBSS(StringRef, SMLoc) { | 
|  | SMLoc IDLoc = getLexer().getLoc(); | 
|  | StringRef Name; | 
|  | if (getParser().parseIdentifier(Name)) | 
|  | return TokError("expected identifier in directive"); | 
|  |  | 
|  | // Handle the identifier as the key symbol. | 
|  | MCSymbol *Sym = getContext().GetOrCreateSymbol(Name); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in directive"); | 
|  | Lex(); | 
|  |  | 
|  | int64_t Size; | 
|  | SMLoc SizeLoc = getLexer().getLoc(); | 
|  | if (getParser().parseAbsoluteExpression(Size)) | 
|  | return true; | 
|  |  | 
|  | int64_t Pow2Alignment = 0; | 
|  | SMLoc Pow2AlignmentLoc; | 
|  | if (getLexer().is(AsmToken::Comma)) { | 
|  | Lex(); | 
|  | Pow2AlignmentLoc = getLexer().getLoc(); | 
|  | if (getParser().parseAbsoluteExpression(Pow2Alignment)) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.tbss' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | if (Size < 0) | 
|  | return Error(SizeLoc, "invalid '.tbss' directive size, can't be less than" | 
|  | "zero"); | 
|  |  | 
|  | // FIXME: Diagnose overflow. | 
|  | if (Pow2Alignment < 0) | 
|  | return Error(Pow2AlignmentLoc, "invalid '.tbss' alignment, can't be less" | 
|  | "than zero"); | 
|  |  | 
|  | if (!Sym->isUndefined()) | 
|  | return Error(IDLoc, "invalid symbol redefinition"); | 
|  |  | 
|  | getStreamer().EmitTBSSSymbol(getContext().getMachOSection( | 
|  | "__DATA", "__thread_bss", | 
|  | MCSectionMachO::S_THREAD_LOCAL_ZEROFILL, | 
|  | 0, SectionKind::getThreadBSS()), | 
|  | Sym, Size, 1 << Pow2Alignment); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveZerofill | 
|  | ///  ::= .zerofill segname , sectname [, identifier , size_expression [ | 
|  | ///      , align_expression ]] | 
|  | bool DarwinAsmParser::ParseDirectiveZerofill(StringRef, SMLoc) { | 
|  | StringRef Segment; | 
|  | if (getParser().parseIdentifier(Segment)) | 
|  | return TokError("expected segment name after '.zerofill' directive"); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in directive"); | 
|  | Lex(); | 
|  |  | 
|  | StringRef Section; | 
|  | if (getParser().parseIdentifier(Section)) | 
|  | return TokError("expected section name after comma in '.zerofill' " | 
|  | "directive"); | 
|  |  | 
|  | // If this is the end of the line all that was wanted was to create the | 
|  | // the section but with no symbol. | 
|  | if (getLexer().is(AsmToken::EndOfStatement)) { | 
|  | // Create the zerofill section but no symbol | 
|  | getStreamer().EmitZerofill(getContext().getMachOSection( | 
|  | Segment, Section, MCSectionMachO::S_ZEROFILL, | 
|  | 0, SectionKind::getBSS())); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in directive"); | 
|  | Lex(); | 
|  |  | 
|  | SMLoc IDLoc = getLexer().getLoc(); | 
|  | StringRef IDStr; | 
|  | if (getParser().parseIdentifier(IDStr)) | 
|  | return TokError("expected identifier in directive"); | 
|  |  | 
|  | // handle the identifier as the key symbol. | 
|  | MCSymbol *Sym = getContext().GetOrCreateSymbol(IDStr); | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::Comma)) | 
|  | return TokError("unexpected token in directive"); | 
|  | Lex(); | 
|  |  | 
|  | int64_t Size; | 
|  | SMLoc SizeLoc = getLexer().getLoc(); | 
|  | if (getParser().parseAbsoluteExpression(Size)) | 
|  | return true; | 
|  |  | 
|  | int64_t Pow2Alignment = 0; | 
|  | SMLoc Pow2AlignmentLoc; | 
|  | if (getLexer().is(AsmToken::Comma)) { | 
|  | Lex(); | 
|  | Pow2AlignmentLoc = getLexer().getLoc(); | 
|  | if (getParser().parseAbsoluteExpression(Pow2Alignment)) | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.zerofill' directive"); | 
|  |  | 
|  | Lex(); | 
|  |  | 
|  | if (Size < 0) | 
|  | return Error(SizeLoc, "invalid '.zerofill' directive size, can't be less " | 
|  | "than zero"); | 
|  |  | 
|  | // NOTE: The alignment in the directive is a power of 2 value, the assembler | 
|  | // may internally end up wanting an alignment in bytes. | 
|  | // FIXME: Diagnose overflow. | 
|  | if (Pow2Alignment < 0) | 
|  | return Error(Pow2AlignmentLoc, "invalid '.zerofill' directive alignment, " | 
|  | "can't be less than zero"); | 
|  |  | 
|  | if (!Sym->isUndefined()) | 
|  | return Error(IDLoc, "invalid symbol redefinition"); | 
|  |  | 
|  | // Create the zerofill Symbol with Size and Pow2Alignment | 
|  | // | 
|  | // FIXME: Arch specific. | 
|  | getStreamer().EmitZerofill(getContext().getMachOSection( | 
|  | Segment, Section, MCSectionMachO::S_ZEROFILL, | 
|  | 0, SectionKind::getBSS()), | 
|  | Sym, Size, 1 << Pow2Alignment); | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveDataRegion | 
|  | ///  ::= .data_region [ ( jt8 | jt16 | jt32 ) ] | 
|  | bool DarwinAsmParser::ParseDirectiveDataRegion(StringRef, SMLoc) { | 
|  | if (getLexer().is(AsmToken::EndOfStatement)) { | 
|  | Lex(); | 
|  | getStreamer().EmitDataRegion(MCDR_DataRegion); | 
|  | return false; | 
|  | } | 
|  | StringRef RegionType; | 
|  | SMLoc Loc = getParser().getTok().getLoc(); | 
|  | if (getParser().parseIdentifier(RegionType)) | 
|  | return TokError("expected region type after '.data_region' directive"); | 
|  | int Kind = StringSwitch<int>(RegionType) | 
|  | .Case("jt8", MCDR_DataRegionJT8) | 
|  | .Case("jt16", MCDR_DataRegionJT16) | 
|  | .Case("jt32", MCDR_DataRegionJT32) | 
|  | .Default(-1); | 
|  | if (Kind == -1) | 
|  | return Error(Loc, "unknown region type in '.data_region' directive"); | 
|  | Lex(); | 
|  |  | 
|  | getStreamer().EmitDataRegion((MCDataRegionType)Kind); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /// ParseDirectiveDataRegionEnd | 
|  | ///  ::= .end_data_region | 
|  | bool DarwinAsmParser::ParseDirectiveDataRegionEnd(StringRef, SMLoc) { | 
|  | if (getLexer().isNot(AsmToken::EndOfStatement)) | 
|  | return TokError("unexpected token in '.end_data_region' directive"); | 
|  |  | 
|  | Lex(); | 
|  | getStreamer().EmitDataRegion(MCDR_DataRegionEnd); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | namespace llvm { | 
|  |  | 
|  | MCAsmParserExtension *createDarwinAsmParser() { | 
|  | return new DarwinAsmParser; | 
|  | } | 
|  |  | 
|  | } // end llvm namespace |