Path: blob/main/contrib/llvm-project/llvm/lib/MC/MCParser/AsmLexer.cpp
35269 views
//===- AsmLexer.cpp - Lexer for Assembly Files ----------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//7//8// This class implements the lexer for assembly files.9//10//===----------------------------------------------------------------------===//1112#include "llvm/MC/MCParser/AsmLexer.h"13#include "llvm/ADT/APInt.h"14#include "llvm/ADT/ArrayRef.h"15#include "llvm/ADT/StringExtras.h"16#include "llvm/ADT/StringRef.h"17#include "llvm/ADT/StringSwitch.h"18#include "llvm/MC/MCAsmInfo.h"19#include "llvm/MC/MCParser/MCAsmLexer.h"20#include "llvm/Support/Compiler.h"21#include "llvm/Support/SMLoc.h"22#include "llvm/Support/SaveAndRestore.h"23#include <cassert>24#include <cctype>25#include <cstdio>26#include <cstring>27#include <string>28#include <tuple>29#include <utility>3031using namespace llvm;3233AsmLexer::AsmLexer(const MCAsmInfo &MAI) : MAI(MAI) {34AllowAtInIdentifier = !StringRef(MAI.getCommentString()).starts_with("@");35LexMotorolaIntegers = MAI.shouldUseMotorolaIntegers();36}3738AsmLexer::~AsmLexer() = default;3940void AsmLexer::setBuffer(StringRef Buf, const char *ptr,41bool EndStatementAtEOF) {42CurBuf = Buf;4344if (ptr)45CurPtr = ptr;46else47CurPtr = CurBuf.begin();4849TokStart = nullptr;50this->EndStatementAtEOF = EndStatementAtEOF;51}5253/// ReturnError - Set the error to the specified string at the specified54/// location. This is defined to always return AsmToken::Error.55AsmToken AsmLexer::ReturnError(const char *Loc, const std::string &Msg) {56SetError(SMLoc::getFromPointer(Loc), Msg);5758return AsmToken(AsmToken::Error, StringRef(Loc, CurPtr - Loc));59}6061int AsmLexer::getNextChar() {62if (CurPtr == CurBuf.end())63return EOF;64return (unsigned char)*CurPtr++;65}6667int AsmLexer::peekNextChar() {68if (CurPtr == CurBuf.end())69return EOF;70return (unsigned char)*CurPtr;71}7273/// The leading integral digit sequence and dot should have already been74/// consumed, some or all of the fractional digit sequence *can* have been75/// consumed.76AsmToken AsmLexer::LexFloatLiteral() {77// Skip the fractional digit sequence.78while (isDigit(*CurPtr))79++CurPtr;8081if (*CurPtr == '-' || *CurPtr == '+')82return ReturnError(CurPtr, "invalid sign in float literal");8384// Check for exponent85if ((*CurPtr == 'e' || *CurPtr == 'E')) {86++CurPtr;8788if (*CurPtr == '-' || *CurPtr == '+')89++CurPtr;9091while (isDigit(*CurPtr))92++CurPtr;93}9495return AsmToken(AsmToken::Real,96StringRef(TokStart, CurPtr - TokStart));97}9899/// LexHexFloatLiteral matches essentially (.[0-9a-fA-F]*)?[pP][+-]?[0-9a-fA-F]+100/// while making sure there are enough actual digits around for the constant to101/// be valid.102///103/// The leading "0x[0-9a-fA-F]*" (i.e. integer part) has already been consumed104/// before we get here.105AsmToken AsmLexer::LexHexFloatLiteral(bool NoIntDigits) {106assert((*CurPtr == 'p' || *CurPtr == 'P' || *CurPtr == '.') &&107"unexpected parse state in floating hex");108bool NoFracDigits = true;109110// Skip the fractional part if there is one111if (*CurPtr == '.') {112++CurPtr;113114const char *FracStart = CurPtr;115while (isHexDigit(*CurPtr))116++CurPtr;117118NoFracDigits = CurPtr == FracStart;119}120121if (NoIntDigits && NoFracDigits)122return ReturnError(TokStart, "invalid hexadecimal floating-point constant: "123"expected at least one significand digit");124125// Make sure we do have some kind of proper exponent part126if (*CurPtr != 'p' && *CurPtr != 'P')127return ReturnError(TokStart, "invalid hexadecimal floating-point constant: "128"expected exponent part 'p'");129++CurPtr;130131if (*CurPtr == '+' || *CurPtr == '-')132++CurPtr;133134// N.b. exponent digits are *not* hex135const char *ExpStart = CurPtr;136while (isDigit(*CurPtr))137++CurPtr;138139if (CurPtr == ExpStart)140return ReturnError(TokStart, "invalid hexadecimal floating-point constant: "141"expected at least one exponent digit");142143return AsmToken(AsmToken::Real, StringRef(TokStart, CurPtr - TokStart));144}145146/// LexIdentifier: [a-zA-Z_$.@?][a-zA-Z0-9_$.@#?]*147static bool isIdentifierChar(char C, bool AllowAt, bool AllowHash) {148return isAlnum(C) || C == '_' || C == '$' || C == '.' || C == '?' ||149(AllowAt && C == '@') || (AllowHash && C == '#');150}151152AsmToken AsmLexer::LexIdentifier() {153// Check for floating point literals.154if (CurPtr[-1] == '.' && isDigit(*CurPtr)) {155// Disambiguate a .1243foo identifier from a floating literal.156while (isDigit(*CurPtr))157++CurPtr;158159if (!isIdentifierChar(*CurPtr, AllowAtInIdentifier,160AllowHashInIdentifier) ||161*CurPtr == 'e' || *CurPtr == 'E')162return LexFloatLiteral();163}164165while (isIdentifierChar(*CurPtr, AllowAtInIdentifier, AllowHashInIdentifier))166++CurPtr;167168// Handle . as a special case.169if (CurPtr == TokStart+1 && TokStart[0] == '.')170return AsmToken(AsmToken::Dot, StringRef(TokStart, 1));171172return AsmToken(AsmToken::Identifier, StringRef(TokStart, CurPtr - TokStart));173}174175/// LexSlash: Slash: /176/// C-Style Comment: /* ... */177/// C-style Comment: // ...178AsmToken AsmLexer::LexSlash() {179if (!MAI.shouldAllowAdditionalComments()) {180IsAtStartOfStatement = false;181return AsmToken(AsmToken::Slash, StringRef(TokStart, 1));182}183184switch (*CurPtr) {185case '*':186IsAtStartOfStatement = false;187break; // C style comment.188case '/':189++CurPtr;190return LexLineComment();191default:192IsAtStartOfStatement = false;193return AsmToken(AsmToken::Slash, StringRef(TokStart, 1));194}195196// C Style comment.197++CurPtr; // skip the star.198const char *CommentTextStart = CurPtr;199while (CurPtr != CurBuf.end()) {200switch (*CurPtr++) {201case '*':202// End of the comment?203if (*CurPtr != '/')204break;205// If we have a CommentConsumer, notify it about the comment.206if (CommentConsumer) {207CommentConsumer->HandleComment(208SMLoc::getFromPointer(CommentTextStart),209StringRef(CommentTextStart, CurPtr - 1 - CommentTextStart));210}211++CurPtr; // End the */.212return AsmToken(AsmToken::Comment,213StringRef(TokStart, CurPtr - TokStart));214}215}216return ReturnError(TokStart, "unterminated comment");217}218219/// LexLineComment: Comment: #[^\n]*220/// : //[^\n]*221AsmToken AsmLexer::LexLineComment() {222// Mark This as an end of statement with a body of the223// comment. While it would be nicer to leave this two tokens,224// backwards compatability with TargetParsers makes keeping this in this form225// better.226const char *CommentTextStart = CurPtr;227int CurChar = getNextChar();228while (CurChar != '\n' && CurChar != '\r' && CurChar != EOF)229CurChar = getNextChar();230const char *NewlinePtr = CurPtr;231if (CurChar == '\r' && CurPtr != CurBuf.end() && *CurPtr == '\n')232++CurPtr;233234// If we have a CommentConsumer, notify it about the comment.235if (CommentConsumer) {236CommentConsumer->HandleComment(237SMLoc::getFromPointer(CommentTextStart),238StringRef(CommentTextStart, NewlinePtr - 1 - CommentTextStart));239}240241IsAtStartOfLine = true;242// This is a whole line comment. leave newline243if (IsAtStartOfStatement)244return AsmToken(AsmToken::EndOfStatement,245StringRef(TokStart, CurPtr - TokStart));246IsAtStartOfStatement = true;247248return AsmToken(AsmToken::EndOfStatement,249StringRef(TokStart, CurPtr - 1 - TokStart));250}251252static void SkipIgnoredIntegerSuffix(const char *&CurPtr) {253// Skip case-insensitive ULL, UL, U, L and LL suffixes.254if (CurPtr[0] == 'U' || CurPtr[0] == 'u')255++CurPtr;256if (CurPtr[0] == 'L' || CurPtr[0] == 'l')257++CurPtr;258if (CurPtr[0] == 'L' || CurPtr[0] == 'l')259++CurPtr;260}261262// Look ahead to search for first non-hex digit, if it's [hH], then we treat the263// integer as a hexadecimal, possibly with leading zeroes.264static unsigned doHexLookAhead(const char *&CurPtr, unsigned DefaultRadix,265bool LexHex) {266const char *FirstNonDec = nullptr;267const char *LookAhead = CurPtr;268while (true) {269if (isDigit(*LookAhead)) {270++LookAhead;271} else {272if (!FirstNonDec)273FirstNonDec = LookAhead;274275// Keep going if we are looking for a 'h' suffix.276if (LexHex && isHexDigit(*LookAhead))277++LookAhead;278else279break;280}281}282bool isHex = LexHex && (*LookAhead == 'h' || *LookAhead == 'H');283CurPtr = isHex || !FirstNonDec ? LookAhead : FirstNonDec;284if (isHex)285return 16;286return DefaultRadix;287}288289static const char *findLastDigit(const char *CurPtr, unsigned DefaultRadix) {290while (hexDigitValue(*CurPtr) < DefaultRadix) {291++CurPtr;292}293return CurPtr;294}295296static AsmToken intToken(StringRef Ref, APInt &Value) {297if (Value.isIntN(64))298return AsmToken(AsmToken::Integer, Ref, Value);299return AsmToken(AsmToken::BigNum, Ref, Value);300}301302static std::string radixName(unsigned Radix) {303switch (Radix) {304case 2:305return "binary";306case 8:307return "octal";308case 10:309return "decimal";310case 16:311return "hexadecimal";312default:313return "base-" + std::to_string(Radix);314}315}316317/// LexDigit: First character is [0-9].318/// Local Label: [0-9][:]319/// Forward/Backward Label: [0-9][fb]320/// Binary integer: 0b[01]+321/// Octal integer: 0[0-7]+322/// Hex integer: 0x[0-9a-fA-F]+ or [0x]?[0-9][0-9a-fA-F]*[hH]323/// Decimal integer: [1-9][0-9]*324AsmToken AsmLexer::LexDigit() {325// MASM-flavor binary integer: [01]+[yY] (if DefaultRadix < 16, [bByY])326// MASM-flavor octal integer: [0-7]+[oOqQ]327// MASM-flavor decimal integer: [0-9]+[tT] (if DefaultRadix < 16, [dDtT])328// MASM-flavor hexadecimal integer: [0-9][0-9a-fA-F]*[hH]329if (LexMasmIntegers && isdigit(CurPtr[-1])) {330const char *FirstNonBinary =331(CurPtr[-1] != '0' && CurPtr[-1] != '1') ? CurPtr - 1 : nullptr;332const char *FirstNonDecimal =333(CurPtr[-1] < '0' || CurPtr[-1] > '9') ? CurPtr - 1 : nullptr;334const char *OldCurPtr = CurPtr;335while (isHexDigit(*CurPtr)) {336switch (*CurPtr) {337default:338if (!FirstNonDecimal) {339FirstNonDecimal = CurPtr;340}341[[fallthrough]];342case '9':343case '8':344case '7':345case '6':346case '5':347case '4':348case '3':349case '2':350if (!FirstNonBinary) {351FirstNonBinary = CurPtr;352}353break;354case '1':355case '0':356break;357}358++CurPtr;359}360if (*CurPtr == '.') {361// MASM float literals (other than hex floats) always contain a ".", and362// are always written in decimal.363++CurPtr;364return LexFloatLiteral();365}366367if (LexMasmHexFloats && (*CurPtr == 'r' || *CurPtr == 'R')) {368++CurPtr;369return AsmToken(AsmToken::Real, StringRef(TokStart, CurPtr - TokStart));370}371372unsigned Radix = 0;373if (*CurPtr == 'h' || *CurPtr == 'H') {374// hexadecimal number375++CurPtr;376Radix = 16;377} else if (*CurPtr == 't' || *CurPtr == 'T') {378// decimal number379++CurPtr;380Radix = 10;381} else if (*CurPtr == 'o' || *CurPtr == 'O' || *CurPtr == 'q' ||382*CurPtr == 'Q') {383// octal number384++CurPtr;385Radix = 8;386} else if (*CurPtr == 'y' || *CurPtr == 'Y') {387// binary number388++CurPtr;389Radix = 2;390} else if (FirstNonDecimal && FirstNonDecimal + 1 == CurPtr &&391DefaultRadix < 14 &&392(*FirstNonDecimal == 'd' || *FirstNonDecimal == 'D')) {393Radix = 10;394} else if (FirstNonBinary && FirstNonBinary + 1 == CurPtr &&395DefaultRadix < 12 &&396(*FirstNonBinary == 'b' || *FirstNonBinary == 'B')) {397Radix = 2;398}399400if (Radix) {401StringRef Result(TokStart, CurPtr - TokStart);402APInt Value(128, 0, true);403404if (Result.drop_back().getAsInteger(Radix, Value))405return ReturnError(TokStart, "invalid " + radixName(Radix) + " number");406407// MSVC accepts and ignores type suffices on integer literals.408SkipIgnoredIntegerSuffix(CurPtr);409410return intToken(Result, Value);411}412413// default-radix integers, or floating point numbers, fall through414CurPtr = OldCurPtr;415}416417// MASM default-radix integers: [0-9a-fA-F]+418// (All other integer literals have a radix specifier.)419if (LexMasmIntegers && UseMasmDefaultRadix) {420CurPtr = findLastDigit(CurPtr, 16);421StringRef Result(TokStart, CurPtr - TokStart);422423APInt Value(128, 0, true);424if (Result.getAsInteger(DefaultRadix, Value)) {425return ReturnError(TokStart,426"invalid " + radixName(DefaultRadix) + " number");427}428429return intToken(Result, Value);430}431432// Motorola hex integers: $[0-9a-fA-F]+433if (LexMotorolaIntegers && CurPtr[-1] == '$') {434const char *NumStart = CurPtr;435while (isHexDigit(CurPtr[0]))436++CurPtr;437438APInt Result(128, 0);439if (StringRef(NumStart, CurPtr - NumStart).getAsInteger(16, Result))440return ReturnError(TokStart, "invalid hexadecimal number");441442return intToken(StringRef(TokStart, CurPtr - TokStart), Result);443}444445// Motorola binary integers: %[01]+446if (LexMotorolaIntegers && CurPtr[-1] == '%') {447const char *NumStart = CurPtr;448while (*CurPtr == '0' || *CurPtr == '1')449++CurPtr;450451APInt Result(128, 0);452if (StringRef(NumStart, CurPtr - NumStart).getAsInteger(2, Result))453return ReturnError(TokStart, "invalid binary number");454455return intToken(StringRef(TokStart, CurPtr - TokStart), Result);456}457458// Decimal integer: [1-9][0-9]*459// HLASM-flavour decimal integer: [0-9][0-9]*460// FIXME: Later on, support for fb for HLASM has to be added in461// as they probably would be needed for asm goto462if (LexHLASMIntegers || CurPtr[-1] != '0' || CurPtr[0] == '.') {463unsigned Radix = doHexLookAhead(CurPtr, 10, LexMasmIntegers);464465if (!LexHLASMIntegers) {466bool IsHex = Radix == 16;467// Check for floating point literals.468if (!IsHex && (*CurPtr == '.' || *CurPtr == 'e' || *CurPtr == 'E')) {469if (*CurPtr == '.')470++CurPtr;471return LexFloatLiteral();472}473}474475StringRef Result(TokStart, CurPtr - TokStart);476477APInt Value(128, 0, true);478if (Result.getAsInteger(Radix, Value))479return ReturnError(TokStart, "invalid " + radixName(Radix) + " number");480481if (!LexHLASMIntegers)482// The darwin/x86 (and x86-64) assembler accepts and ignores type483// suffices on integer literals.484SkipIgnoredIntegerSuffix(CurPtr);485486return intToken(Result, Value);487}488489if (!LexMasmIntegers && ((*CurPtr == 'b') || (*CurPtr == 'B'))) {490++CurPtr;491// See if we actually have "0b" as part of something like "jmp 0b\n"492if (!isDigit(CurPtr[0])) {493--CurPtr;494StringRef Result(TokStart, CurPtr - TokStart);495return AsmToken(AsmToken::Integer, Result, 0);496}497const char *NumStart = CurPtr;498while (CurPtr[0] == '0' || CurPtr[0] == '1')499++CurPtr;500501// Requires at least one binary digit.502if (CurPtr == NumStart)503return ReturnError(TokStart, "invalid binary number");504505StringRef Result(TokStart, CurPtr - TokStart);506507APInt Value(128, 0, true);508if (Result.substr(2).getAsInteger(2, Value))509return ReturnError(TokStart, "invalid binary number");510511// The darwin/x86 (and x86-64) assembler accepts and ignores ULL and LL512// suffixes on integer literals.513SkipIgnoredIntegerSuffix(CurPtr);514515return intToken(Result, Value);516}517518if ((*CurPtr == 'x') || (*CurPtr == 'X')) {519++CurPtr;520const char *NumStart = CurPtr;521while (isHexDigit(CurPtr[0]))522++CurPtr;523524// "0x.0p0" is valid, and "0x0p0" (but not "0xp0" for example, which will be525// diagnosed by LexHexFloatLiteral).526if (CurPtr[0] == '.' || CurPtr[0] == 'p' || CurPtr[0] == 'P')527return LexHexFloatLiteral(NumStart == CurPtr);528529// Otherwise requires at least one hex digit.530if (CurPtr == NumStart)531return ReturnError(CurPtr-2, "invalid hexadecimal number");532533APInt Result(128, 0);534if (StringRef(TokStart, CurPtr - TokStart).getAsInteger(0, Result))535return ReturnError(TokStart, "invalid hexadecimal number");536537// Consume the optional [hH].538if (LexMasmIntegers && (*CurPtr == 'h' || *CurPtr == 'H'))539++CurPtr;540541// The darwin/x86 (and x86-64) assembler accepts and ignores ULL and LL542// suffixes on integer literals.543SkipIgnoredIntegerSuffix(CurPtr);544545return intToken(StringRef(TokStart, CurPtr - TokStart), Result);546}547548// Either octal or hexadecimal.549APInt Value(128, 0, true);550unsigned Radix = doHexLookAhead(CurPtr, 8, LexMasmIntegers);551StringRef Result(TokStart, CurPtr - TokStart);552if (Result.getAsInteger(Radix, Value))553return ReturnError(TokStart, "invalid " + radixName(Radix) + " number");554555// Consume the [hH].556if (Radix == 16)557++CurPtr;558559// The darwin/x86 (and x86-64) assembler accepts and ignores ULL and LL560// suffixes on integer literals.561SkipIgnoredIntegerSuffix(CurPtr);562563return intToken(Result, Value);564}565566/// LexSingleQuote: Integer: 'b'567AsmToken AsmLexer::LexSingleQuote() {568int CurChar = getNextChar();569570if (LexHLASMStrings)571return ReturnError(TokStart, "invalid usage of character literals");572573if (LexMasmStrings) {574while (CurChar != EOF) {575if (CurChar != '\'') {576CurChar = getNextChar();577} else if (peekNextChar() == '\'') {578// In MASM single-quote strings, doubled single-quotes mean an escaped579// single quote, so should be lexed in.580(void)getNextChar();581CurChar = getNextChar();582} else {583break;584}585}586if (CurChar == EOF)587return ReturnError(TokStart, "unterminated string constant");588return AsmToken(AsmToken::String, StringRef(TokStart, CurPtr - TokStart));589}590591if (CurChar == '\\')592CurChar = getNextChar();593594if (CurChar == EOF)595return ReturnError(TokStart, "unterminated single quote");596597CurChar = getNextChar();598599if (CurChar != '\'')600return ReturnError(TokStart, "single quote way too long");601602// The idea here being that 'c' is basically just an integral603// constant.604StringRef Res = StringRef(TokStart,CurPtr - TokStart);605long long Value;606607if (Res.starts_with("\'\\")) {608char theChar = Res[2];609switch (theChar) {610default: Value = theChar; break;611case '\'': Value = '\''; break;612case 't': Value = '\t'; break;613case 'n': Value = '\n'; break;614case 'b': Value = '\b'; break;615case 'f': Value = '\f'; break;616case 'r': Value = '\r'; break;617}618} else619Value = TokStart[1];620621return AsmToken(AsmToken::Integer, Res, Value);622}623624/// LexQuote: String: "..."625AsmToken AsmLexer::LexQuote() {626int CurChar = getNextChar();627if (LexHLASMStrings)628return ReturnError(TokStart, "invalid usage of string literals");629630if (LexMasmStrings) {631while (CurChar != EOF) {632if (CurChar != '"') {633CurChar = getNextChar();634} else if (peekNextChar() == '"') {635// In MASM double-quoted strings, doubled double-quotes mean an escaped636// double quote, so should be lexed in.637(void)getNextChar();638CurChar = getNextChar();639} else {640break;641}642}643if (CurChar == EOF)644return ReturnError(TokStart, "unterminated string constant");645return AsmToken(AsmToken::String, StringRef(TokStart, CurPtr - TokStart));646}647648while (CurChar != '"') {649if (CurChar == '\\') {650// Allow \", etc.651CurChar = getNextChar();652}653654if (CurChar == EOF)655return ReturnError(TokStart, "unterminated string constant");656657CurChar = getNextChar();658}659660return AsmToken(AsmToken::String, StringRef(TokStart, CurPtr - TokStart));661}662663StringRef AsmLexer::LexUntilEndOfStatement() {664TokStart = CurPtr;665666while (!isAtStartOfComment(CurPtr) && // Start of line comment.667!isAtStatementSeparator(CurPtr) && // End of statement marker.668*CurPtr != '\n' && *CurPtr != '\r' && CurPtr != CurBuf.end()) {669++CurPtr;670}671return StringRef(TokStart, CurPtr-TokStart);672}673674StringRef AsmLexer::LexUntilEndOfLine() {675TokStart = CurPtr;676677while (*CurPtr != '\n' && *CurPtr != '\r' && CurPtr != CurBuf.end()) {678++CurPtr;679}680return StringRef(TokStart, CurPtr-TokStart);681}682683size_t AsmLexer::peekTokens(MutableArrayRef<AsmToken> Buf,684bool ShouldSkipSpace) {685SaveAndRestore SavedTokenStart(TokStart);686SaveAndRestore SavedCurPtr(CurPtr);687SaveAndRestore SavedAtStartOfLine(IsAtStartOfLine);688SaveAndRestore SavedAtStartOfStatement(IsAtStartOfStatement);689SaveAndRestore SavedSkipSpace(SkipSpace, ShouldSkipSpace);690SaveAndRestore SavedIsPeeking(IsPeeking, true);691std::string SavedErr = getErr();692SMLoc SavedErrLoc = getErrLoc();693694size_t ReadCount;695for (ReadCount = 0; ReadCount < Buf.size(); ++ReadCount) {696AsmToken Token = LexToken();697698Buf[ReadCount] = Token;699700if (Token.is(AsmToken::Eof))701break;702}703704SetError(SavedErrLoc, SavedErr);705return ReadCount;706}707708bool AsmLexer::isAtStartOfComment(const char *Ptr) {709if (MAI.getRestrictCommentStringToStartOfStatement() && !IsAtStartOfStatement)710return false;711712StringRef CommentString = MAI.getCommentString();713714if (CommentString.size() == 1)715return CommentString[0] == Ptr[0];716717// Allow # preprocessor comments also be counted as comments for "##" cases718if (CommentString[1] == '#')719return CommentString[0] == Ptr[0];720721return strncmp(Ptr, CommentString.data(), CommentString.size()) == 0;722}723724bool AsmLexer::isAtStatementSeparator(const char *Ptr) {725return strncmp(Ptr, MAI.getSeparatorString(),726strlen(MAI.getSeparatorString())) == 0;727}728729AsmToken AsmLexer::LexToken() {730TokStart = CurPtr;731// This always consumes at least one character.732int CurChar = getNextChar();733734if (!IsPeeking && CurChar == '#' && IsAtStartOfStatement) {735// If this starts with a '#', this may be a cpp736// hash directive and otherwise a line comment.737AsmToken TokenBuf[2];738MutableArrayRef<AsmToken> Buf(TokenBuf, 2);739size_t num = peekTokens(Buf, true);740// There cannot be a space preceding this741if (IsAtStartOfLine && num == 2 && TokenBuf[0].is(AsmToken::Integer) &&742TokenBuf[1].is(AsmToken::String)) {743CurPtr = TokStart; // reset curPtr;744StringRef s = LexUntilEndOfLine();745UnLex(TokenBuf[1]);746UnLex(TokenBuf[0]);747return AsmToken(AsmToken::HashDirective, s);748}749750if (MAI.shouldAllowAdditionalComments())751return LexLineComment();752}753754if (isAtStartOfComment(TokStart))755return LexLineComment();756757if (isAtStatementSeparator(TokStart)) {758CurPtr += strlen(MAI.getSeparatorString()) - 1;759IsAtStartOfLine = true;760IsAtStartOfStatement = true;761return AsmToken(AsmToken::EndOfStatement,762StringRef(TokStart, strlen(MAI.getSeparatorString())));763}764765// If we're missing a newline at EOF, make sure we still get an766// EndOfStatement token before the Eof token.767if (CurChar == EOF && !IsAtStartOfStatement && EndStatementAtEOF) {768IsAtStartOfLine = true;769IsAtStartOfStatement = true;770return AsmToken(AsmToken::EndOfStatement, StringRef(TokStart, 0));771}772IsAtStartOfLine = false;773bool OldIsAtStartOfStatement = IsAtStartOfStatement;774IsAtStartOfStatement = false;775switch (CurChar) {776default:777// Handle identifier: [a-zA-Z_.$@#?][a-zA-Z0-9_.$@#?]*778// Whether or not the lexer accepts '$', '@', '#' and '?' at the start of779// an identifier is target-dependent. These characters are handled in the780// respective switch cases.781if (isalpha(CurChar) || CurChar == '_' || CurChar == '.')782return LexIdentifier();783784// Unknown character, emit an error.785return ReturnError(TokStart, "invalid character in input");786case EOF:787if (EndStatementAtEOF) {788IsAtStartOfLine = true;789IsAtStartOfStatement = true;790}791return AsmToken(AsmToken::Eof, StringRef(TokStart, 0));792case 0:793case ' ':794case '\t':795IsAtStartOfStatement = OldIsAtStartOfStatement;796while (*CurPtr == ' ' || *CurPtr == '\t')797CurPtr++;798if (SkipSpace)799return LexToken(); // Ignore whitespace.800else801return AsmToken(AsmToken::Space, StringRef(TokStart, CurPtr - TokStart));802case '\r': {803IsAtStartOfLine = true;804IsAtStartOfStatement = true;805// If this is a CR followed by LF, treat that as one token.806if (CurPtr != CurBuf.end() && *CurPtr == '\n')807++CurPtr;808return AsmToken(AsmToken::EndOfStatement,809StringRef(TokStart, CurPtr - TokStart));810}811case '\n':812IsAtStartOfLine = true;813IsAtStartOfStatement = true;814return AsmToken(AsmToken::EndOfStatement, StringRef(TokStart, 1));815case ':': return AsmToken(AsmToken::Colon, StringRef(TokStart, 1));816case '+': return AsmToken(AsmToken::Plus, StringRef(TokStart, 1));817case '~': return AsmToken(AsmToken::Tilde, StringRef(TokStart, 1));818case '(': return AsmToken(AsmToken::LParen, StringRef(TokStart, 1));819case ')': return AsmToken(AsmToken::RParen, StringRef(TokStart, 1));820case '[': return AsmToken(AsmToken::LBrac, StringRef(TokStart, 1));821case ']': return AsmToken(AsmToken::RBrac, StringRef(TokStart, 1));822case '{': return AsmToken(AsmToken::LCurly, StringRef(TokStart, 1));823case '}': return AsmToken(AsmToken::RCurly, StringRef(TokStart, 1));824case '*': return AsmToken(AsmToken::Star, StringRef(TokStart, 1));825case ',': return AsmToken(AsmToken::Comma, StringRef(TokStart, 1));826case '$': {827if (LexMotorolaIntegers && isHexDigit(*CurPtr))828return LexDigit();829if (MAI.doesAllowDollarAtStartOfIdentifier())830return LexIdentifier();831return AsmToken(AsmToken::Dollar, StringRef(TokStart, 1));832}833case '@':834if (MAI.doesAllowAtAtStartOfIdentifier())835return LexIdentifier();836return AsmToken(AsmToken::At, StringRef(TokStart, 1));837case '#':838if (MAI.doesAllowHashAtStartOfIdentifier())839return LexIdentifier();840return AsmToken(AsmToken::Hash, StringRef(TokStart, 1));841case '?':842if (MAI.doesAllowQuestionAtStartOfIdentifier())843return LexIdentifier();844return AsmToken(AsmToken::Question, StringRef(TokStart, 1));845case '\\': return AsmToken(AsmToken::BackSlash, StringRef(TokStart, 1));846case '=':847if (*CurPtr == '=') {848++CurPtr;849return AsmToken(AsmToken::EqualEqual, StringRef(TokStart, 2));850}851return AsmToken(AsmToken::Equal, StringRef(TokStart, 1));852case '-':853if (*CurPtr == '>') {854++CurPtr;855return AsmToken(AsmToken::MinusGreater, StringRef(TokStart, 2));856}857return AsmToken(AsmToken::Minus, StringRef(TokStart, 1));858case '|':859if (*CurPtr == '|') {860++CurPtr;861return AsmToken(AsmToken::PipePipe, StringRef(TokStart, 2));862}863return AsmToken(AsmToken::Pipe, StringRef(TokStart, 1));864case '^': return AsmToken(AsmToken::Caret, StringRef(TokStart, 1));865case '&':866if (*CurPtr == '&') {867++CurPtr;868return AsmToken(AsmToken::AmpAmp, StringRef(TokStart, 2));869}870return AsmToken(AsmToken::Amp, StringRef(TokStart, 1));871case '!':872if (*CurPtr == '=') {873++CurPtr;874return AsmToken(AsmToken::ExclaimEqual, StringRef(TokStart, 2));875}876return AsmToken(AsmToken::Exclaim, StringRef(TokStart, 1));877case '%':878if (LexMotorolaIntegers && (*CurPtr == '0' || *CurPtr == '1')) {879return LexDigit();880}881882if (MAI.hasMipsExpressions()) {883AsmToken::TokenKind Operator;884unsigned OperatorLength;885886std::tie(Operator, OperatorLength) =887StringSwitch<std::pair<AsmToken::TokenKind, unsigned>>(888StringRef(CurPtr))889.StartsWith("call16", {AsmToken::PercentCall16, 7})890.StartsWith("call_hi", {AsmToken::PercentCall_Hi, 8})891.StartsWith("call_lo", {AsmToken::PercentCall_Lo, 8})892.StartsWith("dtprel_hi", {AsmToken::PercentDtprel_Hi, 10})893.StartsWith("dtprel_lo", {AsmToken::PercentDtprel_Lo, 10})894.StartsWith("got_disp", {AsmToken::PercentGot_Disp, 9})895.StartsWith("got_hi", {AsmToken::PercentGot_Hi, 7})896.StartsWith("got_lo", {AsmToken::PercentGot_Lo, 7})897.StartsWith("got_ofst", {AsmToken::PercentGot_Ofst, 9})898.StartsWith("got_page", {AsmToken::PercentGot_Page, 9})899.StartsWith("gottprel", {AsmToken::PercentGottprel, 9})900.StartsWith("got", {AsmToken::PercentGot, 4})901.StartsWith("gp_rel", {AsmToken::PercentGp_Rel, 7})902.StartsWith("higher", {AsmToken::PercentHigher, 7})903.StartsWith("highest", {AsmToken::PercentHighest, 8})904.StartsWith("hi", {AsmToken::PercentHi, 3})905.StartsWith("lo", {AsmToken::PercentLo, 3})906.StartsWith("neg", {AsmToken::PercentNeg, 4})907.StartsWith("pcrel_hi", {AsmToken::PercentPcrel_Hi, 9})908.StartsWith("pcrel_lo", {AsmToken::PercentPcrel_Lo, 9})909.StartsWith("tlsgd", {AsmToken::PercentTlsgd, 6})910.StartsWith("tlsldm", {AsmToken::PercentTlsldm, 7})911.StartsWith("tprel_hi", {AsmToken::PercentTprel_Hi, 9})912.StartsWith("tprel_lo", {AsmToken::PercentTprel_Lo, 9})913.Default({AsmToken::Percent, 1});914915if (Operator != AsmToken::Percent) {916CurPtr += OperatorLength - 1;917return AsmToken(Operator, StringRef(TokStart, OperatorLength));918}919}920return AsmToken(AsmToken::Percent, StringRef(TokStart, 1));921case '/':922IsAtStartOfStatement = OldIsAtStartOfStatement;923return LexSlash();924case '\'': return LexSingleQuote();925case '"': return LexQuote();926case '0': case '1': case '2': case '3': case '4':927case '5': case '6': case '7': case '8': case '9':928return LexDigit();929case '<':930switch (*CurPtr) {931case '<':932++CurPtr;933return AsmToken(AsmToken::LessLess, StringRef(TokStart, 2));934case '=':935++CurPtr;936return AsmToken(AsmToken::LessEqual, StringRef(TokStart, 2));937case '>':938++CurPtr;939return AsmToken(AsmToken::LessGreater, StringRef(TokStart, 2));940default:941return AsmToken(AsmToken::Less, StringRef(TokStart, 1));942}943case '>':944switch (*CurPtr) {945case '>':946++CurPtr;947return AsmToken(AsmToken::GreaterGreater, StringRef(TokStart, 2));948case '=':949++CurPtr;950return AsmToken(AsmToken::GreaterEqual, StringRef(TokStart, 2));951default:952return AsmToken(AsmToken::Greater, StringRef(TokStart, 1));953}954955// TODO: Quoted identifiers (objc methods etc)956// local labels: [0-9][:]957// Forward/backward labels: [0-9][fb]958// Integers, fp constants, character constants.959}960}961962963