forked from OSchip/llvm-project
152 lines
5.0 KiB
C++
152 lines
5.0 KiB
C++
//===- WasmAsmParser.cpp - Wasm Assembly Parser -----------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
// --
|
|
//
|
|
// Note, this is for wasm, the binary format (analogous to ELF), not wasm,
|
|
// the instruction set (analogous to x86), for which parsing code lives in
|
|
// WebAssemblyAsmParser.
|
|
//
|
|
// This file contains processing for generic directives implemented using
|
|
// MCTargetStreamer, the ones that depend on WebAssemblyTargetStreamer are in
|
|
// WebAssemblyAsmParser.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/BinaryFormat/Wasm.h"
|
|
#include "llvm/MC/MCContext.h"
|
|
#include "llvm/MC/MCParser/MCAsmLexer.h"
|
|
#include "llvm/MC/MCParser/MCAsmParser.h"
|
|
#include "llvm/MC/MCParser/MCAsmParserExtension.h"
|
|
#include "llvm/MC/MCSectionWasm.h"
|
|
#include "llvm/MC/MCStreamer.h"
|
|
#include "llvm/MC/MCSymbol.h"
|
|
#include "llvm/MC/MCSymbolWasm.h"
|
|
#include "llvm/Support/MachineValueType.h"
|
|
|
|
using namespace llvm;
|
|
|
|
namespace {
|
|
|
|
class WasmAsmParser : public MCAsmParserExtension {
|
|
MCAsmParser *Parser;
|
|
MCAsmLexer *Lexer;
|
|
|
|
template<bool (WasmAsmParser::*HandlerMethod)(StringRef, SMLoc)>
|
|
void addDirectiveHandler(StringRef Directive) {
|
|
MCAsmParser::ExtensionDirectiveHandler Handler = std::make_pair(
|
|
this, HandleDirective<WasmAsmParser, HandlerMethod>);
|
|
|
|
getParser().addDirectiveHandler(Directive, Handler);
|
|
}
|
|
|
|
public:
|
|
WasmAsmParser() : Parser(nullptr), Lexer(nullptr) {
|
|
BracketExpressionsSupported = true;
|
|
}
|
|
|
|
void Initialize(MCAsmParser &P) override {
|
|
Parser = &P;
|
|
Lexer = &Parser->getLexer();
|
|
// Call the base implementation.
|
|
this->MCAsmParserExtension::Initialize(*Parser);
|
|
|
|
addDirectiveHandler<&WasmAsmParser::parseSectionDirectiveText>(".text");
|
|
addDirectiveHandler<&WasmAsmParser::parseSectionDirective>(".section");
|
|
addDirectiveHandler<&WasmAsmParser::parseDirectiveSize>(".size");
|
|
addDirectiveHandler<&WasmAsmParser::parseDirectiveType>(".type");
|
|
}
|
|
|
|
bool Error(const StringRef &msg, const AsmToken &tok) {
|
|
return Parser->Error(tok.getLoc(), msg + tok.getString());
|
|
}
|
|
|
|
bool IsNext(AsmToken::TokenKind Kind) {
|
|
auto ok = Lexer->is(Kind);
|
|
if (ok) Lex();
|
|
return ok;
|
|
}
|
|
|
|
bool Expect(AsmToken::TokenKind Kind, const char *KindName) {
|
|
if (!IsNext(Kind))
|
|
return Error(std::string("Expected ") + KindName + ", instead got: ",
|
|
Lexer->getTok());
|
|
return false;
|
|
}
|
|
|
|
bool parseSectionDirectiveText(StringRef, SMLoc) {
|
|
// FIXME: .text currently no-op.
|
|
return false;
|
|
}
|
|
|
|
bool parseSectionDirective(StringRef, SMLoc) {
|
|
StringRef Name;
|
|
if (Parser->parseIdentifier(Name))
|
|
return TokError("expected identifier in directive");
|
|
// FIXME: currently requiring this very fixed format.
|
|
if (Expect(AsmToken::Comma, ",") || Expect(AsmToken::String, "string") ||
|
|
Expect(AsmToken::Comma, ",") || Expect(AsmToken::At, "@") ||
|
|
Expect(AsmToken::EndOfStatement, "eol"))
|
|
return true;
|
|
auto WS = getContext().getWasmSection(Name, SectionKind::getText());
|
|
getStreamer().SwitchSection(WS);
|
|
return false;
|
|
}
|
|
|
|
// TODO: This function is almost the same as ELFAsmParser::ParseDirectiveSize
|
|
// so maybe could be shared somehow.
|
|
bool parseDirectiveSize(StringRef, SMLoc) {
|
|
StringRef Name;
|
|
if (Parser->parseIdentifier(Name))
|
|
return TokError("expected identifier in directive");
|
|
auto Sym = getContext().getOrCreateSymbol(Name);
|
|
if (Expect(AsmToken::Comma, ","))
|
|
return true;
|
|
const MCExpr *Expr;
|
|
if (Parser->parseExpression(Expr))
|
|
return true;
|
|
if (Expect(AsmToken::EndOfStatement, "eol"))
|
|
return true;
|
|
// MCWasmStreamer implements this.
|
|
getStreamer().emitELFSize(Sym, Expr);
|
|
return false;
|
|
}
|
|
|
|
bool parseDirectiveType(StringRef, SMLoc) {
|
|
// This could be the start of a function, check if followed by
|
|
// "label,@function"
|
|
if (!Lexer->is(AsmToken::Identifier))
|
|
return Error("Expected label after .type directive, got: ",
|
|
Lexer->getTok());
|
|
auto WasmSym = cast<MCSymbolWasm>(
|
|
getStreamer().getContext().getOrCreateSymbol(
|
|
Lexer->getTok().getString()));
|
|
Lex();
|
|
if (!(IsNext(AsmToken::Comma) && IsNext(AsmToken::At) &&
|
|
Lexer->is(AsmToken::Identifier)))
|
|
return Error("Expected label,@type declaration, got: ", Lexer->getTok());
|
|
auto TypeName = Lexer->getTok().getString();
|
|
if (TypeName == "function")
|
|
WasmSym->setType(wasm::WASM_SYMBOL_TYPE_FUNCTION);
|
|
else if (TypeName == "global")
|
|
WasmSym->setType(wasm::WASM_SYMBOL_TYPE_GLOBAL);
|
|
else
|
|
return Error("Unknown WASM symbol type: ", Lexer->getTok());
|
|
Lex();
|
|
return Expect(AsmToken::EndOfStatement, "EOL");
|
|
}
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
namespace llvm {
|
|
|
|
MCAsmParserExtension *createWasmAsmParser() {
|
|
return new WasmAsmParser;
|
|
}
|
|
|
|
} // end namespace llvm
|