3.1. Variables

Most expressions that we see in practice involve variables, such as

b * b + 4 * a * c

No parser is much good without being able to deal with them. Fortunately, it's also quite easy to do.

Remember that in our parser as it currently stands, there are two kinds of factors allowed: integer constants and expressions within parentheses. In BNF notation,

<factor> ::= <number> | (<expression>)

The '|' stands for "or", meaning of course that either form is a legal form for a factor. Remember, too, that we had no trouble knowing which was which … the lookahead character is a left paren ( in one case, and a digit in the other.

It probably won't come as too much of a surprise that a variable is just another kind of factor. So we extend the BNF above to read:

<factor> ::= <number> | (<expression>) | <variable>

Again, there is no ambiguity: if the lookahead character is a letter, we have a variable; if a digit, we have a number. Back when we translated the number, we just issued code to load the number, as immediate data, into D0. Now we do the same, only we load a variable.

A minor complication in the code generation arises from the fact that most 68000 operating systems, including the SK*DOS™ that I'm using, require the code to be written in “position-independent” form, which basically means that everything is PC-relative. The format for a load in this language is

        MOVE X(PC),D0

where X is, of course, the variable name. Armed with that, let's modify the current version of Factor to read:

{ Parse and Translate a Math Factor }
procedure Expression; Forward;
procedure Factor;
   if Look = '(' then begin
   else if IsAlpha(Look) then
      EmitLn('MOVE ' + GetName + '(PC),D0')
      EmitLn('MOVE #' + GetNum + ',D0');

I've remarked before how easy it is to add extensions to the parser, because of the way it's structured. You can see that this still holds true here. This time it cost us all of two extra lines of code. Notice, too, how the if-else-else structure exactly parallels the BNF syntax equation.

OK, compile and test this new version of the parser. That didn't hurt too badly, did it?