[yaml2obj/obj2yaml] - Add support for .stack_sizes sections.
[llvm-complete.git] / lib / MC / MCParser / WasmAsmParser.cpp
blob0c242aed706d1be1d85e888c717f6c791c5c1dea
1 //===- WasmAsmParser.cpp - Wasm Assembly Parser -----------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 // --
8 //
9 // Note, this is for wasm, the binary format (analogous to ELF), not wasm,
10 // the instruction set (analogous to x86), for which parsing code lives in
11 // WebAssemblyAsmParser.
13 // This file contains processing for generic directives implemented using
14 // MCTargetStreamer, the ones that depend on WebAssemblyTargetStreamer are in
15 // WebAssemblyAsmParser.
17 //===----------------------------------------------------------------------===//
19 #include "llvm/BinaryFormat/Wasm.h"
20 #include "llvm/MC/MCContext.h"
21 #include "llvm/MC/MCParser/MCAsmLexer.h"
22 #include "llvm/MC/MCParser/MCAsmParser.h"
23 #include "llvm/MC/MCParser/MCAsmParserExtension.h"
24 #include "llvm/MC/MCSectionWasm.h"
25 #include "llvm/MC/MCStreamer.h"
26 #include "llvm/MC/MCSymbol.h"
27 #include "llvm/MC/MCSymbolWasm.h"
28 #include "llvm/Support/MachineValueType.h"
30 using namespace llvm;
32 namespace {
34 class WasmAsmParser : public MCAsmParserExtension {
35 MCAsmParser *Parser = nullptr;
36 MCAsmLexer *Lexer = nullptr;
38 template<bool (WasmAsmParser::*HandlerMethod)(StringRef, SMLoc)>
39 void addDirectiveHandler(StringRef Directive) {
40 MCAsmParser::ExtensionDirectiveHandler Handler = std::make_pair(
41 this, HandleDirective<WasmAsmParser, HandlerMethod>);
43 getParser().addDirectiveHandler(Directive, Handler);
46 public:
47 WasmAsmParser() { BracketExpressionsSupported = true; }
49 void Initialize(MCAsmParser &P) override {
50 Parser = &P;
51 Lexer = &Parser->getLexer();
52 // Call the base implementation.
53 this->MCAsmParserExtension::Initialize(*Parser);
55 addDirectiveHandler<&WasmAsmParser::parseSectionDirectiveText>(".text");
56 addDirectiveHandler<&WasmAsmParser::parseSectionDirective>(".section");
57 addDirectiveHandler<&WasmAsmParser::parseDirectiveSize>(".size");
58 addDirectiveHandler<&WasmAsmParser::parseDirectiveType>(".type");
59 addDirectiveHandler<&WasmAsmParser::ParseDirectiveIdent>(".ident");
60 addDirectiveHandler<
61 &WasmAsmParser::ParseDirectiveSymbolAttribute>(".weak");
62 addDirectiveHandler<
63 &WasmAsmParser::ParseDirectiveSymbolAttribute>(".local");
64 addDirectiveHandler<
65 &WasmAsmParser::ParseDirectiveSymbolAttribute>(".internal");
66 addDirectiveHandler<
67 &WasmAsmParser::ParseDirectiveSymbolAttribute>(".hidden");
70 bool error(const StringRef &Msg, const AsmToken &Tok) {
71 return Parser->Error(Tok.getLoc(), Msg + Tok.getString());
74 bool isNext(AsmToken::TokenKind Kind) {
75 auto Ok = Lexer->is(Kind);
76 if (Ok)
77 Lex();
78 return Ok;
81 bool expect(AsmToken::TokenKind Kind, const char *KindName) {
82 if (!isNext(Kind))
83 return error(std::string("Expected ") + KindName + ", instead got: ",
84 Lexer->getTok());
85 return false;
88 bool parseSectionDirectiveText(StringRef, SMLoc) {
89 // FIXME: .text currently no-op.
90 return false;
93 bool parseSectionFlags(StringRef FlagStr, bool &Passive) {
94 SmallVector<StringRef, 2> Flags;
95 // If there are no flags, keep Flags empty
96 FlagStr.split(Flags, ",", -1, false);
97 for (auto &Flag : Flags) {
98 if (Flag == "passive")
99 Passive = true;
100 else
101 return error("Expected section flags, instead got: ", Lexer->getTok());
103 return false;
106 bool parseSectionDirective(StringRef, SMLoc) {
107 StringRef Name;
108 if (Parser->parseIdentifier(Name))
109 return TokError("expected identifier in directive");
111 if (expect(AsmToken::Comma, ","))
112 return true;
114 if (Lexer->isNot(AsmToken::String))
115 return error("expected string in directive, instead got: ", Lexer->getTok());
117 auto Kind = StringSwitch<Optional<SectionKind>>(Name)
118 .StartsWith(".data", SectionKind::getData())
119 .StartsWith(".rodata", SectionKind::getReadOnly())
120 .StartsWith(".text", SectionKind::getText())
121 .StartsWith(".custom_section", SectionKind::getMetadata())
122 .StartsWith(".bss", SectionKind::getBSS())
123 // See use of .init_array in WasmObjectWriter and
124 // TargetLoweringObjectFileWasm
125 .StartsWith(".init_array", SectionKind::getData())
126 .StartsWith(".debug_", SectionKind::getMetadata())
127 .Default(Optional<SectionKind>());
128 if (!Kind.hasValue())
129 return Parser->Error(Lexer->getLoc(), "unknown section kind: " + Name);
131 MCSectionWasm *Section = getContext().getWasmSection(Name, Kind.getValue());
133 // Update section flags if present in this .section directive
134 bool Passive = false;
135 if (parseSectionFlags(getTok().getStringContents(), Passive))
136 return true;
138 if (Passive) {
139 if (!Section->isWasmData())
140 return Parser->Error(getTok().getLoc(),
141 "Only data sections can be passive");
142 Section->setPassive();
145 Lex();
147 if (expect(AsmToken::Comma, ",") || expect(AsmToken::At, "@") ||
148 expect(AsmToken::EndOfStatement, "eol"))
149 return true;
151 auto WS = getContext().getWasmSection(Name, Kind.getValue());
152 getStreamer().SwitchSection(WS);
153 return false;
156 // TODO: This function is almost the same as ELFAsmParser::ParseDirectiveSize
157 // so maybe could be shared somehow.
158 bool parseDirectiveSize(StringRef, SMLoc) {
159 StringRef Name;
160 if (Parser->parseIdentifier(Name))
161 return TokError("expected identifier in directive");
162 auto Sym = getContext().getOrCreateSymbol(Name);
163 if (expect(AsmToken::Comma, ","))
164 return true;
165 const MCExpr *Expr;
166 if (Parser->parseExpression(Expr))
167 return true;
168 if (expect(AsmToken::EndOfStatement, "eol"))
169 return true;
170 // This is done automatically by the assembler for functions currently,
171 // so this is only currently needed for data sections:
172 getStreamer().emitELFSize(Sym, Expr);
173 return false;
176 bool parseDirectiveType(StringRef, SMLoc) {
177 // This could be the start of a function, check if followed by
178 // "label,@function"
179 if (!Lexer->is(AsmToken::Identifier))
180 return error("Expected label after .type directive, got: ",
181 Lexer->getTok());
182 auto WasmSym = cast<MCSymbolWasm>(
183 getStreamer().getContext().getOrCreateSymbol(
184 Lexer->getTok().getString()));
185 Lex();
186 if (!(isNext(AsmToken::Comma) && isNext(AsmToken::At) &&
187 Lexer->is(AsmToken::Identifier)))
188 return error("Expected label,@type declaration, got: ", Lexer->getTok());
189 auto TypeName = Lexer->getTok().getString();
190 if (TypeName == "function")
191 WasmSym->setType(wasm::WASM_SYMBOL_TYPE_FUNCTION);
192 else if (TypeName == "global")
193 WasmSym->setType(wasm::WASM_SYMBOL_TYPE_GLOBAL);
194 else if (TypeName == "object")
195 WasmSym->setType(wasm::WASM_SYMBOL_TYPE_DATA);
196 else
197 return error("Unknown WASM symbol type: ", Lexer->getTok());
198 Lex();
199 return expect(AsmToken::EndOfStatement, "EOL");
202 // FIXME: Shared with ELF.
203 /// ParseDirectiveIdent
204 /// ::= .ident string
205 bool ParseDirectiveIdent(StringRef, SMLoc) {
206 if (getLexer().isNot(AsmToken::String))
207 return TokError("unexpected token in '.ident' directive");
208 StringRef Data = getTok().getIdentifier();
209 Lex();
210 if (getLexer().isNot(AsmToken::EndOfStatement))
211 return TokError("unexpected token in '.ident' directive");
212 Lex();
213 getStreamer().EmitIdent(Data);
214 return false;
217 // FIXME: Shared with ELF.
218 /// ParseDirectiveSymbolAttribute
219 /// ::= { ".local", ".weak", ... } [ identifier ( , identifier )* ]
220 bool ParseDirectiveSymbolAttribute(StringRef Directive, SMLoc) {
221 MCSymbolAttr Attr = StringSwitch<MCSymbolAttr>(Directive)
222 .Case(".weak", MCSA_Weak)
223 .Case(".local", MCSA_Local)
224 .Case(".hidden", MCSA_Hidden)
225 .Case(".internal", MCSA_Internal)
226 .Case(".protected", MCSA_Protected)
227 .Default(MCSA_Invalid);
228 assert(Attr != MCSA_Invalid && "unexpected symbol attribute directive!");
229 if (getLexer().isNot(AsmToken::EndOfStatement)) {
230 while (true) {
231 StringRef Name;
232 if (getParser().parseIdentifier(Name))
233 return TokError("expected identifier in directive");
234 MCSymbol *Sym = getContext().getOrCreateSymbol(Name);
235 getStreamer().EmitSymbolAttribute(Sym, Attr);
236 if (getLexer().is(AsmToken::EndOfStatement))
237 break;
238 if (getLexer().isNot(AsmToken::Comma))
239 return TokError("unexpected token in directive");
240 Lex();
243 Lex();
244 return false;
248 } // end anonymous namespace
250 namespace llvm {
252 MCAsmParserExtension *createWasmAsmParser() {
253 return new WasmAsmParser;
256 } // end namespace llvm