summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/lexer.cpp42
-rw-r--r--src/lexer.h8
-rw-r--r--src/options.cpp2
-rw-r--r--src/parser.cpp55
-rw-r--r--src/parser.h37
-rw-r--r--src/token.cpp6
-rw-r--r--src/token.h6
-rw-r--r--src/unitparser.cpp7
-rw-r--r--src/unitparser.h1
9 files changed, 155 insertions, 9 deletions
diff --git a/src/lexer.cpp b/src/lexer.cpp
index b0d0fd1..fbcaafb 100644
--- a/src/lexer.cpp
+++ b/src/lexer.cpp
@@ -11,27 +11,55 @@ Lexer::Lexer( Bu::Stream &rIn ) :
11 iRadix( 10 ), 11 iRadix( 10 ),
12 numRangeTop('9'), 12 numRangeTop('9'),
13 ascRangeTop('a'-1), 13 ascRangeTop('a'-1),
14 eMode( modeNormal ) 14 eMode( modeNormal ),
15 iLookAheadSize( 2 ),
16 iLookAheadUsed( 0 ),
17 iLookAheadStart( 0 ),
18 aLookAhead( 0 )
15{ 19{
20 aLookAhead = new Token[iLookAheadSize];
16} 21}
17 22
18Lexer::~Lexer() 23Lexer::~Lexer()
19{ 24{
20} 25}
21 26
22Token Lexer::nextToken() 27void Lexer::nextToken()
28{
29 if( iLookAheadSize <= 1 )
30 {
31 iLookAheadSize = 0;
32 iLookAheadStart = 0;
33 }
34 else
35 {
36 iLookAheadStart = (iLookAheadStart+1)%iLookAheadSize;
37 iLookAheadSize--;
38 }
39}
40
41void Lexer::fillToken()
23{ 42{
24 switch( eMode ) 43 switch( eMode )
25 { 44 {
26 case modeNormal: 45 case modeNormal:
27 return nextTokenNormal(); 46 aLookAhead[(iLookAheadUsed+iLookAheadStart)%iLookAheadSize] =
47 nextTokenNormal();
48 break;
28 49
29 case modeCommand: 50 case modeCommand:
30 return nextTokenCommand(); 51 aLookAhead[(iLookAheadUsed+iLookAheadStart)%iLookAheadSize] =
52 nextTokenCommand();
53 break;
31 54
32 default: 55 default:
33 throw Bu::ExceptionBase("Invalid mode."); 56 throw Bu::ExceptionBase("Invalid mode.");
34 } 57 }
58
59 Bu::sio << "read["
60 << ((iLookAheadUsed+iLookAheadStart)%iLookAheadSize)
61 << "]: "
62 << aLookAhead[(iLookAheadUsed+iLookAheadStart)%iLookAheadSize].eType;
35} 63}
36 64
37Token Lexer::nextTokenNormal() 65Token Lexer::nextTokenNormal()
@@ -235,5 +263,11 @@ void Lexer::setRadix( int i )
235 263
236Token &Lexer::operator[]( int iIdx ) 264Token &Lexer::operator[]( int iIdx )
237{ 265{
266 while( iIdx >= iLookAheadUsed )
267 {
268 fillToken();
269 iLookAheadUsed++;
270 }
271 return aLookAhead[(iLookAheadStart+iIdx)%iLookAheadSize];
238} 272}
239 273
diff --git a/src/lexer.h b/src/lexer.h
index d5631d4..193a22b 100644
--- a/src/lexer.h
+++ b/src/lexer.h
@@ -20,7 +20,7 @@ public:
20 void setMode( Mode e ) { eMode = e; } 20 void setMode( Mode e ) { eMode = e; }
21 Mode getMode() const { return eMode; } 21 Mode getMode() const { return eMode; }
22 22
23 Token nextToken(); 23 void nextToken();
24 24
25 int getScale() const { return iScale; } 25 int getScale() const { return iScale; }
26 void setScale( int i ) { iScale = i; } 26 void setScale( int i ) { iScale = i; }
@@ -31,6 +31,7 @@ public:
31 Token &operator[]( int iIdx ); 31 Token &operator[]( int iIdx );
32 32
33private: 33private:
34 void fillToken();
34 Token nextTokenNormal(); 35 Token nextTokenNormal();
35 Token nextTokenCommand(); 36 Token nextTokenCommand();
36 37
@@ -43,6 +44,11 @@ private:
43 char numRangeTop; 44 char numRangeTop;
44 char ascRangeTop; 45 char ascRangeTop;
45 Mode eMode; 46 Mode eMode;
47
48 int iLookAheadSize;
49 int iLookAheadUsed;
50 int iLookAheadStart;
51 Token *aLookAhead;
46}; 52};
47 53
48#endif 54#endif
diff --git a/src/options.cpp b/src/options.cpp
index c127022..3c016a6 100644
--- a/src/options.cpp
+++ b/src/options.cpp
@@ -191,7 +191,7 @@ int Options::execute( Bu::StringArray aArgs )
191 mbIn.write(" "); 191 mbIn.write(" ");
192 mbIn.write( aArgs[j] ); 192 mbIn.write( aArgs[j] );
193 } 193 }
194// Bu::println("eq: '''%1'''").arg( mbIn.getString() ); 194 Bu::println("eq: '''%1'''").arg( mbIn.getString() );
195 mbIn.setPos( 0 ); 195 mbIn.setPos( 0 );
196 Bu::MemBuf mbOut; 196 Bu::MemBuf mbOut;
197 Lexer lex( mbIn ); 197 Lexer lex( mbIn );
diff --git a/src/parser.cpp b/src/parser.cpp
index 153b6fe..1f9a193 100644
--- a/src/parser.cpp
+++ b/src/parser.cpp
@@ -22,6 +22,58 @@ void Parser::parse()
22{ 22{
23 for(;;) 23 for(;;)
24 { 24 {
25 lex.nextToken();
26 expr();
27
28 for( TokenStack::iterator i = tsStack.begin(); i; i++ )
29 {
30 Bu::sio << *i << " ";
31 }
32 }
33}
34
35void Parser::expr()
36{
37 if( lex[0].eType == Token::tVariable &&
38 lex[1].eType == Token::tEquals )
39 {
40 // Assignment!
41 expr();
42 return;
43 }
44 exprP();
45}
46
47void Parser::exprP()
48{
49 switch( lex[0].eType )
50 {
51 case Token::tNumber:
52 case Token::tVariable:
53 tsStack.push( lex[0] );
54 lex.nextToken();
55 break;
56
57 case Token::tOpenParen:
58 lex.nextToken();
59 expr();
60 if( lex[0].eType != Token::tCloseParen )
61 {
62 throw Bu::ExceptionBase("Expected close paren");
63 }
64 lex.nextToken();
65 break;
66
67 case Token::tMinus:
68 lex.nextToken();
69 exprP();
70 tsStack.push( Token( Token::tNegate ) );
71 break;
72 }
73}
74/*
75 for(;;)
76 {
25 Token t = lex.nextToken(); 77 Token t = lex.nextToken();
26 switch( t.eType ) 78 switch( t.eType )
27 { 79 {
@@ -158,6 +210,7 @@ void Parser::parse()
158 } 210 }
159 } 211 }
160} 212}
213*/
161 214
162Number Parser::getVariable( const Bu::String &sName ) 215Number Parser::getVariable( const Bu::String &sName )
163{ 216{
@@ -171,6 +224,7 @@ void Parser::setVariable( const Bu::String &sName, const Number &rValue )
171 224
172void Parser::unwind() 225void Parser::unwind()
173{ 226{
227 /*
174 for(;;) 228 for(;;)
175 { 229 {
176 DBS_START( PARSE ); 230 DBS_START( PARSE );
@@ -312,6 +366,7 @@ void Parser::unwind()
312 getPriority( t.eType ) ) 366 getPriority( t.eType ) )
313 return; 367 return;
314 } 368 }
369 */
315} 370}
316 371
317int Parser::reqTokens( Token::Type eType ) 372int Parser::reqTokens( Token::Type eType )
diff --git a/src/parser.h b/src/parser.h
index e163212..d50951b 100644
--- a/src/parser.h
+++ b/src/parser.h
@@ -13,6 +13,35 @@ namespace Bu
13 13
14class Lexer; 14class Lexer;
15 15
16/**
17 *
18 * expr: literal
19 * | variable
20 * | expr '+' expr
21 * | expr '-' expr
22 * | expr '*' expr
23 * | expr '/' expr
24 * | '(' expr ')'
25 * | '-' expr
26 * | variable '=' expr
27 * ;
28 *
29 * -----
30 *
31 * expr': literal
32 * | variable
33 * | '(' expr ')'
34 * | '-' expr'
35 * ;
36 *
37 * expr: expr' '+' expr
38 * | expr' '-' expr
39 * | expr' '*' expr
40 * | expr' '/' expr
41 * | expr'
42 * | variable '=' expr
43 * ;
44 */
16class Parser 45class Parser
17{ 46{
18public: 47public:
@@ -24,6 +53,10 @@ public:
24 void setVariable( const Bu::String &sName, const Number &rValue ); 53 void setVariable( const Bu::String &sName, const Number &rValue );
25 54
26private: 55private:
56 void expr();
57 void exprP();
58
59private:
27 void unwind(); 60 void unwind();
28 int reqTokens( Token::Type eType ); 61 int reqTokens( Token::Type eType );
29 int getPriority( Token::Type eType ); 62 int getPriority( Token::Type eType );
@@ -34,8 +67,8 @@ private:
34 Bu::Stream &rOut; 67 Bu::Stream &rOut;
35 typedef Bu::List<Token> TokenStack; 68 typedef Bu::List<Token> TokenStack;
36 typedef Bu::Hash<Bu::String, Number> VarHash; 69 typedef Bu::Hash<Bu::String, Number> VarHash;
37 TokenStack tsTerminal; 70 TokenStack tsStack;
38 TokenStack tsNonTerminal; 71// TokenStack tsNonTerminal;
39 Number nZero; 72 Number nZero;
40 VarHash hVars; 73 VarHash hVars;
41}; 74};
diff --git a/src/token.cpp b/src/token.cpp
index a03c821..30f3566 100644
--- a/src/token.cpp
+++ b/src/token.cpp
@@ -4,6 +4,12 @@
4#include <bu/formatter.h> 4#include <bu/formatter.h>
5#include <bu/string.h> 5#include <bu/string.h>
6 6
7Token::Token() :
8 eType( tUninitialized ),
9 sVal( 0 )
10{
11}
12
7Token::Token( Type eType ) : 13Token::Token( Type eType ) :
8 eType( eType ), 14 eType( eType ),
9 sVal( 0 ) 15 sVal( 0 )
diff --git a/src/token.h b/src/token.h
index 92a7bcc..12a4904 100644
--- a/src/token.h
+++ b/src/token.h
@@ -26,11 +26,15 @@ public:
26 tEquals, 26 tEquals,
27 tString, 27 tString,
28 28
29 tNegate,
29 tEndOfLine, 30 tEndOfLine,
30 31
31 tEndOfInput 32 tEndOfInput,
33
34 tUninitialized
32 }; 35 };
33 36
37 Token();
34 Token( Type eType ); 38 Token( Type eType );
35 Token( Type eType, Bu::String *s ); 39 Token( Type eType, Bu::String *s );
36 Token( Type eType, Number *n ); 40 Token( Type eType, Number *n );
diff --git a/src/unitparser.cpp b/src/unitparser.cpp
index f518fdc..09b5982 100644
--- a/src/unitparser.cpp
+++ b/src/unitparser.cpp
@@ -13,6 +13,8 @@ UnitParser::UnitParser()
13 "order1", Bu::UnitSuite::expectPass ); 13 "order1", Bu::UnitSuite::expectPass );
14 add( static_cast<Bu::UnitSuite::Test>(&UnitParser::assignment), 14 add( static_cast<Bu::UnitSuite::Test>(&UnitParser::assignment),
15 "assignment", Bu::UnitSuite::expectPass ); 15 "assignment", Bu::UnitSuite::expectPass );
16 add( static_cast<Bu::UnitSuite::Test>(&UnitParser::parse1),
17 "parse1", Bu::UnitSuite::expectPass );
16} 18}
17 19
18UnitParser::~UnitParser() 20UnitParser::~UnitParser()
@@ -55,3 +57,8 @@ void UnitParser::assignment()
55 unitTest( mbOut.getString().trimWhitespace() == parser.getVariable("test").toString() ); 57 unitTest( mbOut.getString().trimWhitespace() == parser.getVariable("test").toString() );
56} 58}
57 59
60void UnitParser::parse1()
61{
62 unitTest(parse("-5") == "-5");
63}
64
diff --git a/src/unitparser.h b/src/unitparser.h
index 744deef..c3b3687 100644
--- a/src/unitparser.h
+++ b/src/unitparser.h
@@ -11,6 +11,7 @@ public:
11 11
12 void order1(); 12 void order1();
13 void assignment(); 13 void assignment();
14 void parse1();
14}; 15};
15 16
16#endif 17#endif