Upload
lamdung
View
266
Download
0
Embed Size (px)
Citation preview
Compiler Design
Scanner
Hwansoo Han
Scanner
Source code
Scanner IR
Parser
Errors
tokens
Front end
Machine code
Back
End
2
Scanner Generator
We want to avoid writing scanners by hand
The scanner is the first stage in the front end
Specifications can be expressed using regular expressions
Build tables and code from a DFA
Scanner
Scanner
Generator
specifications
source code tokens
tables or code
Specifications written as “regular expressions”
Represent words as indices into a global table
3
Regular Expressions
Regular Expression (over alphabet )
is a RE denoting the set {}
If a is in , then a is a RE denoting {a}
If x and y are REs denoting L(x) and L(y) then
x |y is an RE denoting L(x) L(y)
xy is an RE denoting L(x)L(y)
x* is an RE denoting L(x)*
4
5
RE for recognizing register names
Register r (0|1|2| … | 9) (0|1|2| … | 9)*
Allows registers of arbitrary number
Requires at least one digit
RE corresponds to a recognizer (or DFA)
Transitions on other inputs go to an error state, se
Regular Expression – Example
S0 S2 S1
r
(0|1|2| … 9)
accepting state
(0|1|2| … 9)
Recognizer for Register
6
Non-deterministic Finite Automata (NFA)
Each RE corresponds to a deterministic finite automaton (DFA)
May be hard to directly construct the right DFA
NFA for RE such as ( a | b )* abb
NFA is a little different from DFA
S0 has a transition on
S1 has two transitions on a
a | b
S0 S1 S4 S2 S3 a b b
Token Recognizer
Tokens are recognized by NFA
0
1
0 9
8 7 6
3 2 1 4
1
1
5
letter
digit
digit .
digit
digit
digit digit
letter | digit
other
other
other
id_or_keyword
num (integer)
num (real)
main
recognizer
9
7
…
=
…
9
8
9
9
other equal
0
7 1 2
8 9 1
5
…
…
…
keyword
recognizer
p r m
other
r n i
9
9
program
id
integer …
re-read the string stored in the buffer
7
Automating Scanner Construction
RE NFA (Thompson’s construction)
Build an NFA for each term
Combine them with -moves
NFA DFA (subset construction)
Build the simulation
DFA Minimal DFA
Hopcroft’s algorithm
DFA RE (Not part of the scanner construction)
All pairs, all paths problem
Take the union of all paths from s0 to an accepting state
minimal
DFA RE NFA DFA
The Cycle of Constructions
8
9
RE NFA using Thompson’s Construction
Key idea
NFA pattern for each symbol & each operator
Join them with moves in precedence order
S0 S1 a
NFA for a
S0 S1 a
S3 S4 b
NFA for ab
NFA for a | b
S0
S1 S2 a
S3 S4 b
S5
S0 S1
S3 S4
NFA for a*
a
Ken Thompson, CACM, 1968
10
Example of Thompson’s Construction
Let’s try a ( b | c )*
1. a, b, & c
2. b | c
3. ( b | c )*
S0 S1 a
S0 S1 b
S0 S1 c
S2 S3 b
S4 S5 c
S1 S6 S0 S7
S1 S2 b
S3 S4 c
S0 S5
11
Example of Thompson’s Construction (con’t)
4. a ( b | c )*
Of course, a human would design something simpler ...
S0 S1 a
b | c
But, we can automate production of
the more complex one ...
S0 S1 a
S4 S5 b
S6 S7 c
S3 S8 S2 S9
12
NFA DFA with Subset Construction
Need to build a simulation of the NFA
Two key functions
Move(si , a) is set of states reachable from si by a
-closure(si) is set of states reachable from si by
The algorithm: Start state derived from s0 of the NFA
Take its -closure S0 = -closure(s0)
Take the image of S0, Move(S0, ) for each , and take
its -closure
Iterate until no more states are added
Sounds more complex than it is…
Conversion NFA to DFA
What about a ( b | c )* ?
First, the subset construction: NFA → DFA
q0 q1 a
q4 q5 b
q6 q7 c
q3 q8 q2 q9
-closure(move(s,*))
NFA states a b c
s0 q0 q1, q2, q3, q4, q6, q9
none none
s1 q1, q2, q3,q4, q6, q9
none q5, q8, q9,q3, q4, q6
q7, q8, q9,q3, q4, q6
s2 q5, q8, q9,q3, q4, q6
none s2 s3
s3 q7, q8, q9,q3, q4, q6
none s2 s3
s3
s2
s0 s1
c
b
a
b
b
c
c
Final states
13
DFA Minimization
Then, apply the minimization algorithm
To produce the minimal DFA
s3
s2
s0 s1
c
b
a
b
b
c
c
Split on
Current Partition a b c
P0 { s1, s2, s3} {s0} none none none
s0 s1 a
b | c
final states
14
Another Example
Remember ( a | b )* abb ?
Applying the subset construction:
Iteration 3 adds nothing to S, so the algorithm halts
a | b
q0 q1 q4 q2 q3 a b b
Iter. State Contains-closure(
move(si,a))
-closure(
move(si,b))
0 s0 q0, q1 q1, q2 q1
1 s1 q1, q2 q1, q2 q1, q3
s2 q1 q1, q2 q1
2 s3 q1, q3 q1, q2 q1, q4
3 s4 q1, q4 q1, q2 q1
contains q4 (final state)
Our first NFA
15
Another Example (cont’d)
The DFA for ( a | b )* abb
Not much bigger than the original
All transitions are deterministic
s0 a
s1
b
s3 b
s4
s2
a
b
b
a
a
a
b
a b
s0 s1 s2
s1 s1 s3
s2 s1 s2
s3 s1 s4
s4 s1 s2
16
Another Example (cont’d)
Applying the minimization algorithm to the DFA
Current Partition Worklist s Split on a Split on b
P0 {s4} {s0,s1,s2,s3} {s4}
{s0,s1,s2,s3} {s4} none
{s0, s1, s2}
{s3}
P1 {s4} {s3} {s0,s1,s2} {s0,s1,s2}
{s3} {s3} none
{s0, s2}{s1}
P2 {s4} {s3} {s1} {s0,s2} {s0,s2}{s1} {s1} none none
s0 a
s1
b
s3 b
s4
s2
a
b
b
a
a
a
b
s0 , s2
a s1
b
s3 b
s4 b
a
a
a
b
final state
17
Building Faster Scanners from the DFA
Table-driven recognizers waste effort
Read (& classify) the next character
Find the next state
Assign to the state variable
Trip through case logic in δ() & action()
Branch back to the top
We can do better
Encode state & actions in the code
Do transition tests locally
Generate ugly, spaghetti-like code
Takes (many) fewer operations per input character
char next character; state s0 ; call action(state,char); while (char eof) state (state,char); call action(state,char); char next character; if (state) = final then report acceptance; else report failure;
18
Building Faster Scanners from the DFA
A direct-coded recognizer for r Digit Digit*
Many fewer operations per character
Almost no memory operations
Even faster with careful use of fall-through cases
goto s0; s0: word Ø ; char next character; if (char = ‘r’) then goto s1; else goto se; s1: word word + char; char next character; if (‘0’ ≤ char ≤ ‘9’) then goto s2; else goto se;
s2: word word + char; char next character; if (‘0’ ≤ char ≤ ‘9’) then goto s2; else if (char = eof) then report success; else goto se;
se: print error message; return failure;
19
Summary
Building scanner
All this technology automates scanner construction
Implementer writes down the regular expressions
Scanner generator builds NFA, DFA, minimal DFA, and then writes out the (table-driven or direct-coded) code
This reliably produces fast, robust scanners
For most modern language features, this works
You should think twice before introducing a feature that defeats a DFA-based scanner
insignificant blanks (Fortran: anint = an int = an int)
non-reserved keywords (e.g. int if = 1;)
20