2012-11-30 17:26:26 +00:00
//------------------------------------------------------------------------------
// CLING - the C++ LLVM-based InterpreterG :)
// version: $Id$
// author: Vassil Vassilev <vvasilev@cern.ch>
//------------------------------------------------------------------------------
# include "MetaParser.h"
# include "MetaLexer.h"
# include "MetaSema.h"
2012-12-02 19:01:02 +00:00
# include "cling/Interpreter/Interpreter.h"
# include "cling/Interpreter/InvocationOptions.h"
2013-06-10 15:14:36 +02:00
# include "cling/Interpreter/StoredValueRef.h"
2012-12-02 19:01:02 +00:00
2012-11-30 17:26:26 +00:00
# include "llvm/ADT/StringRef.h"
# include "llvm/Support/Path.h"
namespace cling {
MetaParser : : MetaParser ( MetaSema * Actions ) {
m_Lexer . reset ( 0 ) ;
m_Actions . reset ( Actions ) ;
2012-12-02 19:01:02 +00:00
const InvocationOptions & Opts = Actions - > getInterpreter ( ) . getOptions ( ) ;
MetaLexer metaSymbolLexer ( Opts . MetaString ) ;
2012-12-02 19:05:24 +00:00
Token Tok ;
2012-12-02 19:01:02 +00:00
while ( true ) {
metaSymbolLexer . Lex ( Tok ) ;
if ( Tok . is ( tok : : eof ) )
break ;
m_MetaSymbolCache . push_back ( Tok ) ;
}
2012-11-30 17:26:26 +00:00
}
void MetaParser : : enterNewInputLine ( llvm : : StringRef Line ) {
m_Lexer . reset ( new MetaLexer ( Line ) ) ;
m_TokenCache . clear ( ) ;
}
void MetaParser : : consumeToken ( ) {
if ( m_TokenCache . size ( ) )
m_TokenCache . erase ( m_TokenCache . begin ( ) ) ;
lookAhead ( 0 ) ;
}
void MetaParser : : consumeAnyStringToken ( tok : : TokenKind stopAt /*=tok::space*/ ) {
consumeToken ( ) ;
// we have to merge the tokens from the queue until we reach eof token or
// space token
2013-02-28 10:55:45 +00:00
skipWhitespace ( ) ;
2012-11-30 17:26:26 +00:00
// Add the new token in which we will merge the others.
Token & MergedTok = m_TokenCache . front ( ) ;
if ( MergedTok . is ( stopAt ) | | MergedTok . is ( tok : : eof )
| | MergedTok . is ( tok : : comment ) )
return ;
Token Tok = lookAhead ( 1 ) ;
2012-12-01 00:54:25 +00:00
while ( Tok . isNot ( stopAt ) & & Tok . isNot ( tok : : eof ) ) {
2012-11-30 17:26:26 +00:00
//MergedTok.setLength(MergedTok.getLength() + Tok.getLength());
m_TokenCache . erase ( m_TokenCache . begin ( ) + 1 ) ;
Tok = lookAhead ( 1 ) ;
}
MergedTok . setKind ( tok : : raw_ident ) ;
MergedTok . setLength ( Tok . getBufStart ( ) - MergedTok . getBufStart ( ) ) ;
}
const Token & MetaParser : : lookAhead ( unsigned N ) {
if ( N < m_TokenCache . size ( ) )
return m_TokenCache [ N ] ;
for ( unsigned C = N + 1 - m_TokenCache . size ( ) ; C > 0 ; - - C ) {
m_TokenCache . push_back ( Token ( ) ) ;
m_Lexer - > Lex ( m_TokenCache . back ( ) ) ;
}
return m_TokenCache . back ( ) ;
}
2013-02-28 10:55:45 +00:00
void MetaParser : : skipWhitespace ( ) {
2012-11-30 17:26:26 +00:00
while ( getCurTok ( ) . is ( tok : : space ) )
consumeToken ( ) ;
}
2013-06-10 15:14:36 +02:00
bool MetaParser : : isMetaCommand ( MetaSema : : ActionResult & actionResult ,
StoredValueRef * resultValue ) {
return isCommandSymbol ( ) & & isCommand ( actionResult , resultValue ) ;
2012-11-30 17:26:26 +00:00
}
2012-12-06 11:47:24 +00:00
bool MetaParser : : isQuitRequested ( ) const {
return m_Actions - > isQuitRequested ( ) ;
}
2012-11-30 17:26:26 +00:00
bool MetaParser : : isCommandSymbol ( ) {
2012-12-02 19:01:02 +00:00
for ( size_t i = 0 ; i < m_MetaSymbolCache . size ( ) ; + + i ) {
if ( getCurTok ( ) . getKind ( ) ! = m_MetaSymbolCache [ i ] . getKind ( ) )
return false ;
consumeToken ( ) ;
}
return true ;
2012-11-30 17:26:26 +00:00
}
2013-06-10 15:14:36 +02:00
bool MetaParser : : isCommand ( MetaSema : : ActionResult & actionResult ,
StoredValueRef * resultValue ) {
if ( resultValue )
* resultValue = StoredValueRef : : invalidValue ( ) ;
2013-05-24 17:50:13 +02:00
return isLCommand ( actionResult )
2013-06-10 15:14:36 +02:00
| | isXCommand ( actionResult , resultValue )
2013-08-13 11:06:58 +02:00
| | isqCommand ( ) | | isUCommand ( actionResult ) | | isICommand ( )
| | isOCommand ( ) | | israwInputCommand ( ) | | isprintASTCommand ( )
| | isdynamicExtensionsCommand ( )
| | ishelpCommand ( ) | | isfileExCommand ( ) | | isfilesCommand ( ) | | isClassCommand ( )
2013-06-17 20:49:08 +02:00
| | isgCommand ( ) | | isTypedefCommand ( ) | | isprintIRCommand ( )
2013-08-21 12:25:48 +02:00
| | isShellCommand ( actionResult , resultValue )
| | isstoreStateCommand ( ) | | iscompareStateCommand ( ) ;
2012-11-30 17:26:26 +00:00
}
// L := 'L' FilePath
// FilePath := AnyString
// AnyString := .*^(' ' | '\t')
2013-05-24 17:50:13 +02:00
bool MetaParser : : isLCommand ( MetaSema : : ActionResult & actionResult ) {
2012-11-30 17:26:26 +00:00
bool result = false ;
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " L " ) ) {
consumeAnyStringToken ( ) ;
if ( getCurTok ( ) . is ( tok : : raw_ident ) ) {
result = true ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnLCommand ( llvm : : sys : : Path ( getCurTok ( ) . getIdent ( ) ) ) ;
2012-11-30 17:26:26 +00:00
consumeToken ( ) ;
if ( getCurTok ( ) . is ( tok : : comment ) ) {
consumeAnyStringToken ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnComment ( getCurTok ( ) . getIdent ( ) ) ;
2012-11-30 17:26:26 +00:00
}
}
}
// TODO: Some fine grained diagnostics
return result ;
}
2012-12-01 16:48:33 +00:00
// XCommand := 'x' FilePath[ArgList] | 'X' FilePath[ArgList]
2012-11-30 17:26:26 +00:00
// FilePath := AnyString
// ArgList := (ExtraArgList) ' ' [ArgList]
// ExtraArgList := AnyString [, ExtraArgList]
2013-06-10 15:14:36 +02:00
bool MetaParser : : isXCommand ( MetaSema : : ActionResult & actionResult ,
StoredValueRef * resultValue ) {
if ( resultValue )
* resultValue = StoredValueRef : : invalidValue ( ) ;
2012-12-01 16:48:33 +00:00
const Token & Tok = getCurTok ( ) ;
if ( Tok . is ( tok : : ident ) & & ( Tok . getIdent ( ) . equals ( " x " )
| | Tok . getIdent ( ) . equals ( " X " ) ) ) {
2012-11-30 17:26:26 +00:00
// There might be ArgList
consumeAnyStringToken ( tok : : l_paren ) ;
llvm : : sys : : Path file ( getCurTok ( ) . getIdent ( ) ) ;
llvm : : StringRef args ;
consumeToken ( ) ;
if ( getCurTok ( ) . is ( tok : : l_paren ) & & isExtraArgList ( ) ) {
args = getCurTok ( ) . getIdent ( ) ;
consumeToken ( ) ; // consume the closing paren
}
2013-05-24 17:50:13 +02:00
actionResult = m_Actions - > actOnxCommand ( file , args , resultValue ) ;
2012-11-30 17:26:26 +00:00
if ( getCurTok ( ) . is ( tok : : comment ) ) {
consumeAnyStringToken ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnComment ( getCurTok ( ) . getIdent ( ) ) ;
2012-11-30 17:26:26 +00:00
}
2013-06-07 14:09:13 +02:00
return true ;
2012-11-30 17:26:26 +00:00
}
2013-06-07 14:09:13 +02:00
return false ;
2012-11-30 17:26:26 +00:00
}
// ExtraArgList := AnyString [, ExtraArgList]
bool MetaParser : : isExtraArgList ( ) {
// This might be expanded if we need better arg parsing.
consumeAnyStringToken ( tok : : r_paren ) ;
return getCurTok ( ) . is ( tok : : raw_ident ) ;
}
bool MetaParser : : isqCommand ( ) {
bool result = false ;
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " q " ) ) {
result = true ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnqCommand ( ) ;
2012-11-30 17:26:26 +00:00
}
return result ;
}
2013-05-24 17:50:13 +02:00
bool MetaParser : : isUCommand ( MetaSema : : ActionResult & actionResult ) {
actionResult = MetaSema : : AR_Failure ;
2012-12-02 19:31:08 +00:00
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " U " ) ) {
2013-05-24 17:50:13 +02:00
actionResult = m_Actions - > actOnUCommand ( ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
bool MetaParser : : isICommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " I " ) ) {
consumeAnyStringToken ( ) ;
llvm : : sys : : Path path ;
if ( getCurTok ( ) . is ( tok : : raw_ident ) )
path = getCurTok ( ) . getIdent ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnICommand ( path ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
2013-08-13 11:06:58 +02:00
bool MetaParser : : isOCommand ( ) {
const Token & currTok = getCurTok ( ) ;
if ( currTok . is ( tok : : ident ) ) {
llvm : : StringRef ident = currTok . getIdent ( ) ;
if ( ident . startswith ( " O " ) ) {
if ( ident . size ( ) > 1 ) {
int level = 0 ;
if ( ! ident . substr ( 1 ) . getAsInteger ( 10 , level ) & & level > = 0 ) {
consumeAnyStringToken ( tok : : eof ) ;
if ( getCurTok ( ) . is ( tok : : raw_ident ) )
return false ;
//TODO: Process .OXXX here as .O with level XXX.
return true ;
}
} else {
consumeAnyStringToken ( tok : : eof ) ;
const Token & lastStringToken = getCurTok ( ) ;
if ( lastStringToken . is ( tok : : raw_ident ) & & lastStringToken . getLength ( ) ) {
int level = 0 ;
if ( ! lastStringToken . getIdent ( ) . getAsInteger ( 10 , level ) & & level > = 0 ) {
//TODO: process .O XXX
return true ;
}
} else {
//TODO: process .O
return true ;
}
}
}
}
return false ;
}
2012-11-30 17:26:26 +00:00
bool MetaParser : : israwInputCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " rawInput " ) ) {
MetaSema : : SwitchMode mode = MetaSema : : kToggle ;
consumeToken ( ) ;
2013-02-28 10:55:45 +00:00
skipWhitespace ( ) ;
2012-11-30 17:26:26 +00:00
if ( getCurTok ( ) . is ( tok : : constant ) )
mode = ( MetaSema : : SwitchMode ) getCurTok ( ) . getConstant ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnrawInputCommand ( mode ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
bool MetaParser : : isprintASTCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " printAST " ) ) {
MetaSema : : SwitchMode mode = MetaSema : : kToggle ;
consumeToken ( ) ;
2013-02-28 10:55:45 +00:00
skipWhitespace ( ) ;
2012-11-30 17:26:26 +00:00
if ( getCurTok ( ) . is ( tok : : constant ) )
mode = ( MetaSema : : SwitchMode ) getCurTok ( ) . getConstant ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnprintASTCommand ( mode ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
2013-06-17 20:49:08 +02:00
bool MetaParser : : isprintIRCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " printIR " ) ) {
MetaSema : : SwitchMode mode = MetaSema : : kToggle ;
consumeToken ( ) ;
skipWhitespace ( ) ;
if ( getCurTok ( ) . is ( tok : : constant ) )
mode = ( MetaSema : : SwitchMode ) getCurTok ( ) . getConstant ( ) ;
m_Actions - > actOnprintIRCommand ( mode ) ;
return true ;
}
return false ;
}
2013-08-21 12:25:48 +02:00
bool MetaParser : : isstoreStateCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " storeState " ) ) {
//MetaSema::SwitchMode mode = MetaSema::kToggle;
consumeToken ( ) ;
skipWhitespace ( ) ;
if ( ! getCurTok ( ) . is ( tok : : quote ) )
return false ; // FIXME: Issue proper diagnostics
consumeToken ( ) ;
if ( ! getCurTok ( ) . is ( tok : : ident ) )
return false ; // FIXME: Issue proper diagnostics
std : : string ident = getCurTok ( ) . getIdent ( ) ;
consumeToken ( ) ;
if ( ! getCurTok ( ) . is ( tok : : quote ) )
return false ; // FIXME: Issue proper diagnostics
m_Actions - > actOnstoreStateCommand ( ident ) ;
return true ;
}
return false ;
}
bool MetaParser : : iscompareStateCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " compareState " ) ) {
//MetaSema::SwitchMode mode = MetaSema::kToggle;
consumeToken ( ) ;
skipWhitespace ( ) ;
if ( ! getCurTok ( ) . is ( tok : : quote ) )
return false ; // FIXME: Issue proper diagnostics
consumeToken ( ) ;
if ( ! getCurTok ( ) . is ( tok : : ident ) )
return false ; // FIXME: Issue proper diagnostics
std : : string ident = getCurTok ( ) . getIdent ( ) ;
consumeToken ( ) ;
if ( ! getCurTok ( ) . is ( tok : : quote ) )
return false ; // FIXME: Issue proper diagnostics
m_Actions - > actOncompareStateCommand ( ident ) ;
return true ;
}
return false ;
}
2012-11-30 17:26:26 +00:00
bool MetaParser : : isdynamicExtensionsCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & &
getCurTok ( ) . getIdent ( ) . equals ( " dynamicExtensions " ) ) {
MetaSema : : SwitchMode mode = MetaSema : : kToggle ;
consumeToken ( ) ;
2013-02-28 10:55:45 +00:00
skipWhitespace ( ) ;
2012-11-30 17:26:26 +00:00
if ( getCurTok ( ) . is ( tok : : constant ) )
mode = ( MetaSema : : SwitchMode ) getCurTok ( ) . getConstant ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOndynamicExtensionsCommand ( mode ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
bool MetaParser : : ishelpCommand ( ) {
2013-01-15 10:11:52 +00:00
const Token & Tok = getCurTok ( ) ;
if ( Tok . is ( tok : : quest_mark ) | |
( Tok . is ( tok : : ident ) & & Tok . getIdent ( ) . equals ( " help " ) ) ) {
2012-12-01 14:07:57 +00:00
m_Actions - > actOnhelpCommand ( ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
bool MetaParser : : isfileExCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " fileEx " ) ) {
2012-12-01 14:07:57 +00:00
m_Actions - > actOnfileExCommand ( ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
bool MetaParser : : isfilesCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " files " ) ) {
2012-12-01 14:07:57 +00:00
m_Actions - > actOnfilesCommand ( ) ;
2012-11-30 17:26:26 +00:00
return true ;
}
return false ;
}
2012-12-01 13:46:50 +00:00
bool MetaParser : : isClassCommand ( ) {
const Token & Tok = getCurTok ( ) ;
if ( Tok . is ( tok : : ident ) ) {
if ( Tok . getIdent ( ) . equals ( " class " ) ) {
2012-12-05 16:36:42 +00:00
consumeAnyStringToken ( tok : : eof ) ;
2012-12-01 13:46:50 +00:00
const Token & NextTok = getCurTok ( ) ;
llvm : : StringRef className ;
if ( NextTok . is ( tok : : raw_ident ) )
2012-12-05 16:36:42 +00:00
className = NextTok . getIdent ( ) ;
2012-12-01 14:07:57 +00:00
m_Actions - > actOnclassCommand ( className ) ;
2012-12-01 13:46:50 +00:00
return true ;
}
else if ( Tok . getIdent ( ) . equals ( " Class " ) ) {
2012-12-01 14:07:57 +00:00
m_Actions - > actOnClassCommand ( ) ;
2012-12-01 13:46:50 +00:00
return true ;
}
}
return false ;
}
2012-12-02 20:14:02 +00:00
bool MetaParser : : isgCommand ( ) {
if ( getCurTok ( ) . is ( tok : : ident ) & & getCurTok ( ) . getIdent ( ) . equals ( " g " ) ) {
consumeToken ( ) ;
2013-02-28 10:55:45 +00:00
skipWhitespace ( ) ;
2012-12-02 20:14:02 +00:00
llvm : : StringRef varName ;
if ( getCurTok ( ) . is ( tok : : ident ) )
varName = getCurTok ( ) . getIdent ( ) ;
m_Actions - > actOngCommand ( varName ) ;
return true ;
}
return false ;
}
2012-12-05 13:20:58 +00:00
bool MetaParser : : isTypedefCommand ( ) {
const Token & Tok = getCurTok ( ) ;
if ( Tok . is ( tok : : ident ) ) {
if ( Tok . getIdent ( ) . equals ( " typedef " ) ) {
2012-12-11 09:19:50 +00:00
consumeAnyStringToken ( tok : : eof ) ;
2012-12-05 13:20:58 +00:00
const Token & NextTok = getCurTok ( ) ;
llvm : : StringRef typedefName ;
if ( NextTok . is ( tok : : raw_ident ) )
typedefName = NextTok . getIdent ( ) ;
m_Actions - > actOnTypedefCommand ( typedefName ) ;
return true ;
}
}
return false ;
}
2012-12-11 09:19:50 +00:00
2013-06-10 15:14:36 +02:00
bool MetaParser : : isShellCommand ( MetaSema : : ActionResult & actionResult ,
StoredValueRef * resultValue ) {
if ( resultValue )
* resultValue = StoredValueRef : : invalidValue ( ) ;
2013-05-24 17:50:13 +02:00
actionResult = MetaSema : : AR_Failure ;
2012-12-11 09:19:50 +00:00
const Token & Tok = getCurTok ( ) ;
if ( Tok . is ( tok : : excl_mark ) ) {
consumeAnyStringToken ( tok : : eof ) ;
const Token & NextTok = getCurTok ( ) ;
if ( NextTok . is ( tok : : raw_ident ) ) {
llvm : : StringRef commandLine ( NextTok . getIdent ( ) ) ;
if ( ! commandLine . empty ( ) )
2013-05-24 17:50:13 +02:00
actionResult = m_Actions - > actOnShellCommand ( commandLine ,
resultValue ) ;
2012-12-11 09:19:50 +00:00
}
return true ;
}
return false ;
}
2012-12-02 20:14:02 +00:00
2012-11-30 17:26:26 +00:00
} // end namespace cling