-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlalr-test.sc
57 lines (52 loc) · 2.17 KB
/
lalr-test.sc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
;;; lalr-test.ss - An example file showing how the lalr parser can be used.
;;;
;;; An example using the lalr parser. This file defines the function
;;; eval-string, which takes a string representing a simple arithmetic
;;; expression and returns its value. E.g.
;;;
;;; (eval-string " 45 + 76 * 2 ") ==> 197
;;;
;;; To produce an expression defining the parse table which you could compile,
;;; change the backquote to a quote in the definition of expr-grammar,
;;; and evaluate
;;;
;;; `(define table ,(list 'quasiquote (lalr-table expr-grammar expr-terminals #f)))
;;;
;;; Be sure to include any procedures referenced in expr-grammar (in this
;;; example, binary-apply and identity)
(define (binary-apply expr1 op expr2) (op expr1 expr2))
(define (identity expr) expr)
(define expr-grammar
`((expr --> expr expr-op term ,binary-apply) ;;; change ` to '
(expr --> term ,identity) ;;; if you want to generate
(term --> term term-op term0 ,binary-apply) ;;; a Scheme program table
(term --> term0 ,identity)
(term0 --> lparen expr rparen ,(lambda (lp expr rp) expr))
(term0 --> number ,identity)
(number --> number digit ,(lambda (n d) (+ (* 10 n) d)))
(number --> digit ,identity)))
(define expr-terminals '(expr-op term-op lparen rparen digit))
(define table (lalr-table expr-grammar expr-terminals #f))
(define (eval-string string)
(let ((pos 0))
(define (lexical-analyser)
(if (< pos (string-length string))
(let ((char (string-ref string pos)))
(set! pos (+ pos 1))
(if (char=? char #\ )
(lexical-analyser)
(case char
((#\+) `(expr-op . ,+))
((#\-) `(expr-op . ,-))
((#\*) `(term-op . ,*))
((#\/) `(term-op . ,/))
((#\() '(lparen . #f))
((#\)) '(rparen . #f))
((#\0 #\1 #\2 #\3 #\4 #\5 #\6 #\7 #\8 #\9)
`(digit . ,(- (char->integer char) (char->integer #\0))))
(else (parse-error)))))))
(define (parse-error)
(display "Error somewhere in ")
(write (substring string 0 pos))
(newline))
(lalr-parser table lexical-analyser parse-error)))