blob: 92a1e1db7d28f793b48eed52cebbef64d3ec75f5 [file] [log] [blame]
Sean Callanane88f5522010-01-23 02:43:15 +00001//===-- X86AsmLexer.cpp - Tokenize X86 assembly to AsmTokens --------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Sean Callanancf2e3d12010-01-26 00:08:25 +000010#include "llvm/ADT/SmallVector.h"
Sean Callanan7dcef4c2010-01-26 01:00:10 +000011#include "llvm/ADT/StringExtras.h"
Sean Callanane88f5522010-01-23 02:43:15 +000012#include "llvm/Target/TargetAsmLexer.h"
13#include "llvm/Target/TargetRegistry.h"
Sean Callanan436c4842010-01-25 21:59:20 +000014#include "llvm/MC/MCAsmInfo.h"
Sean Callanane88f5522010-01-23 02:43:15 +000015#include "llvm/MC/MCParser/MCAsmLexer.h"
16#include "llvm/MC/MCParser/MCParsedAsmOperand.h"
17#include "X86.h"
18
19using namespace llvm;
20
21namespace {
22
23class X86AsmLexer : public TargetAsmLexer {
24 const MCAsmInfo &AsmInfo;
Sean Callanancf2e3d12010-01-26 00:08:25 +000025
26 bool tentativeIsValid;
27 AsmToken tentativeToken;
28
29 const AsmToken &lexTentative() {
Sean Callanan894c1af2010-01-31 02:28:18 +000030 tentativeToken = getLexer()->Lex();
Sean Callanancf2e3d12010-01-26 00:08:25 +000031 tentativeIsValid = true;
32 return tentativeToken;
33 }
34
35 const AsmToken &lexDefinite() {
36 if(tentativeIsValid) {
37 tentativeIsValid = false;
38 return tentativeToken;
39 }
40 else {
Sean Callanan894c1af2010-01-31 02:28:18 +000041 return getLexer()->Lex();
Sean Callanancf2e3d12010-01-26 00:08:25 +000042 }
43 }
Sean Callanan436c4842010-01-25 21:59:20 +000044
45 AsmToken LexTokenATT();
46 AsmToken LexTokenIntel();
Sean Callanane88f5522010-01-23 02:43:15 +000047protected:
Sean Callanan436c4842010-01-25 21:59:20 +000048 AsmToken LexToken() {
Sean Callanancf2e3d12010-01-26 00:08:25 +000049 if (!Lexer) {
50 SetError(SMLoc(), "No MCAsmLexer installed");
51 return AsmToken(AsmToken::Error, "", 0);
52 }
53
Sean Callanan436c4842010-01-25 21:59:20 +000054 switch (AsmInfo.getAssemblerDialect()) {
55 default:
56 SetError(SMLoc(), "Unhandled dialect");
57 return AsmToken(AsmToken::Error, "", 0);
58 case 0:
59 return LexTokenATT();
60 case 1:
61 return LexTokenIntel();
62 }
63 }
Sean Callanane88f5522010-01-23 02:43:15 +000064public:
65 X86AsmLexer(const Target &T, const MCAsmInfo &MAI)
Sean Callanan894c1af2010-01-31 02:28:18 +000066 : TargetAsmLexer(T), AsmInfo(MAI), tentativeIsValid(false) {
Sean Callanane88f5522010-01-23 02:43:15 +000067 }
68};
69
70}
71
Chris Lattnerb8d6e982010-02-09 00:34:28 +000072static unsigned MatchRegisterName(StringRef Name);
Sean Callanancf2e3d12010-01-26 00:08:25 +000073
Sean Callanan436c4842010-01-25 21:59:20 +000074AsmToken X86AsmLexer::LexTokenATT() {
Sean Callanan3bdbdd02010-02-08 22:50:23 +000075 const AsmToken lexedToken = lexDefinite();
Sean Callanancf2e3d12010-01-26 00:08:25 +000076
77 switch (lexedToken.getKind()) {
78 default:
79 return AsmToken(lexedToken);
80 case AsmToken::Error:
81 SetError(Lexer->getErrLoc(), Lexer->getErr());
82 return AsmToken(lexedToken);
Chris Lattner02758a42010-06-24 07:16:25 +000083 case AsmToken::Percent: {
Sean Callanancf2e3d12010-01-26 00:08:25 +000084 const AsmToken &nextToken = lexTentative();
Chris Lattner02758a42010-06-24 07:16:25 +000085 if (nextToken.getKind() != AsmToken::Identifier)
Sean Callanancf2e3d12010-01-26 00:08:25 +000086 return AsmToken(lexedToken);
Chris Lattner02758a42010-06-24 07:16:25 +000087
88
89 if (unsigned regID = MatchRegisterName(nextToken.getString())) {
90 lexDefinite();
91
92 StringRef regStr(lexedToken.getString().data(),
93 lexedToken.getString().size() +
94 nextToken.getString().size());
95
96 return AsmToken(AsmToken::Register,
97 regStr,
98 static_cast<int64_t>(regID));
Sean Callanancf2e3d12010-01-26 00:08:25 +000099 }
Chris Lattner02758a42010-06-24 07:16:25 +0000100
101 return AsmToken(lexedToken);
Sean Callanancf2e3d12010-01-26 00:08:25 +0000102 }
103 }
Sean Callanan436c4842010-01-25 21:59:20 +0000104}
105
106AsmToken X86AsmLexer::LexTokenIntel() {
Sean Callanan7dcef4c2010-01-26 01:00:10 +0000107 const AsmToken &lexedToken = lexDefinite();
108
109 switch(lexedToken.getKind()) {
110 default:
111 return AsmToken(lexedToken);
112 case AsmToken::Error:
113 SetError(Lexer->getErrLoc(), Lexer->getErr());
114 return AsmToken(lexedToken);
115 case AsmToken::Identifier:
116 {
117 std::string upperCase = lexedToken.getString().str();
118 std::string lowerCase = LowercaseString(upperCase);
119 StringRef lowerRef(lowerCase);
120
121 unsigned regID = MatchRegisterName(lowerRef);
122
123 if (regID) {
124 return AsmToken(AsmToken::Register,
125 lexedToken.getString(),
126 static_cast<int64_t>(regID));
127 }
128 else {
129 return AsmToken(lexedToken);
130 }
131 }
132 }
Sean Callanane88f5522010-01-23 02:43:15 +0000133}
134
135extern "C" void LLVMInitializeX86AsmLexer() {
136 RegisterAsmLexer<X86AsmLexer> X(TheX86_32Target);
137 RegisterAsmLexer<X86AsmLexer> Y(TheX86_64Target);
138}
139
Sean Callanancf2e3d12010-01-26 00:08:25 +0000140#define REGISTERS_ONLY
Daniel Dunbarc690aab2010-01-26 03:56:22 +0000141#include "X86GenAsmMatcher.inc"
Sean Callanancf2e3d12010-01-26 00:08:25 +0000142#undef REGISTERS_ONLY