(module yacc mzscheme (require-for-syntax "private-yacc/parser-builder.ss" "private-yacc/yacc-helper.ss") (require "private-yacc/array2d.ss" "private-lex/token.ss" "private-yacc/parser-actions.ss" (lib "etc.ss") (lib "pretty.ss") (lib "readerr.ss" "syntax")) (provide parser) (define-syntax (parser stx) (syntax-case stx () ((_ args ...) (let ((arg-list (syntax->list (syntax (args ...)))) (src-pos #f) (debug #f) (error #f) (tokens #f) (start #f) (end #f) (precs #f) (suppress #f) (grammar #f) (yacc-output #f)) (for-each (lambda (arg) (syntax-case* arg (debug error tokens start end precs grammar suppress src-pos yacc-output) (lambda (a b) (eq? (syntax-object->datum a) (syntax-object->datum b))) ((debug filename) (cond ((not (string? (syntax-object->datum (syntax filename)))) (raise-syntax-error 'parser-debug "Debugging filename must be a string" (syntax filename))) (debug (raise-syntax-error #f "Multiple debug declarations" stx)) (else (set! debug (syntax-object->datum (syntax filename)))))) ((suppress) (set! suppress #t)) ((src-pos) (set! src-pos #t)) ((error expression) (if error (raise-syntax-error #f "Multiple error declarations" stx) (set! error (syntax expression)))) ((tokens def ...) (if tokens (raise-syntax-error #f "Multiple tokens declarations" stx) (set! tokens arg))) ((start symbol ...) (let ((symbols (syntax->list (syntax (symbol ...))))) (for-each (lambda (sym) (unless (identifier? sym) (raise-syntax-error 'parser-start "Start symbol must be a symbol" sym))) symbols) (when start (raise-syntax-error #f "Multiple start declarations" stx)) (when (null? symbols) (raise-syntax-error 'parser-start "Missing start symbol" stx)) (set! start symbols))) ((end symbols ...) (let ((symbols (syntax->list (syntax (symbols ...))))) (for-each (lambda (sym) (unless (identifier? sym) (raise-syntax-error 'parser-end "End token must be a symbol" sym))) symbols) (let ((d (duplicate-list? (map syntax-object->datum symbols)))) (when d (raise-syntax-error 'parser-end (format "Duplicate end token definition for ~a" d) arg)) (when (null? symbols) (raise-syntax-error 'parser-end "end declaration must contain at least 1 token" arg)) (when end (raise-syntax-error #f "Multiple end declarations" stx)) (set! end symbols)))) ((precs decls ...) (if precs (raise-syntax-error #f "Multiple precs declarations" stx) (set! precs arg))) ((grammar prods ...) (if grammar (raise-syntax-error #f "Multiple grammar declarations" stx) (set! grammar arg))) ((yacc-output filename) (cond ((not (string? (syntax-object->datum (syntax filename)))) (raise-syntax-error 'parser-yacc-output "Yacc-output filename must be a string" (syntax filename))) (yacc-output (raise-syntax-error #f "Multiple yacc-output declarations" stx)) (else (set! yacc-output (syntax-object->datum (syntax filename)))))) (_ (raise-syntax-error 'parser-args "argument must match (debug filename), (error expression), (tokens def ...), (start non-term), (end tokens ...), (precs decls ...), or (grammar prods ...)" arg)))) (syntax->list (syntax (args ...)))) (unless tokens (raise-syntax-error #f "missing tokens declaration" stx)) (unless error (raise-syntax-error #f "missing error declaration" stx)) (unless grammar (raise-syntax-error #f "missing grammar declaration" stx)) (unless end (raise-syntax-error #f "missing end declaration" stx)) (unless start (raise-syntax-error #f "missing start declaration" stx)) (let-values (((table term-sym->index actions check-syntax-fix) (build-parser (if debug debug "") src-pos suppress tokens start end precs grammar))) (when (and yacc-output (not (string=? yacc-output ""))) (with-handlers [(exn:fail:filesystem? (lambda (e) (fprintf (current-error-port) "Cannot write yacc-output to file \"~a\"~n" yacc-output)))] (call-with-output-file yacc-output (lambda (port) (display-yacc (syntax-object->datum grammar) tokens (syntax-object->datum start) (if precs (syntax-object->datum precs) #f) port))))) (with-syntax ((check-syntax-fix check-syntax-fix) (err error) (ends end) (starts start) (debug debug) (table table) (term-sym->index term-sym->index) (actions actions) (src-pos src-pos)) (syntax (begin check-syntax-fix (parser-body debug err (quote starts) (quote ends) table term-sym->index actions src-pos))))))) (_ (raise-syntax-error #f "parser must have the form (parser args ...)" stx)))) (define (reduce-stack stack num ret-vals src-pos) (cond ((> num 0) (let* ((top-frame (car stack)) (ret-vals (if src-pos (cons (stack-frame-value top-frame) (cons (stack-frame-start-pos top-frame) (cons (stack-frame-end-pos top-frame) ret-vals))) (cons (stack-frame-value top-frame) ret-vals)))) (reduce-stack (cdr stack) (sub1 num) ret-vals src-pos))) (else (values stack ret-vals)))) ;; extract-helper : (symbol or make-token) any any -> symbol any any any (define (extract-helper tok v1 v2) (cond ((symbol? tok) (values tok #f v1 v2)) ((token? tok) (values (real-token-name tok) (real-token-value tok) v1 v2)) (else (raise-type-error 'parser "symbol or struct:token" 0 tok)))) ;; extract-src-pos : position-token -> symbol any any any (define (extract-src-pos ip) (cond ((position-token? ip) (extract-helper (position-token-token ip) (position-token-start-pos ip) (position-token-end-pos ip))) (else (raise-type-error 'parser "struct:position-token" 0 ip)))) ;; extract-no-src-pos : (symbol or make-token) -> symbol any any any (define (extract-no-src-pos ip) (extract-helper ip #f #f)) (define-struct stack-frame (state value start-pos end-pos) (make-inspector)) (define (make-empty-stack i) (list (make-stack-frame i #f #f #f))) ;; The table format is an array2d that maps each state/term pair to either ;; an accept, shift or reduce structure - or a #f. Except that we will encode ;; by changing (make-accept) -> 'accept, (make-shift i) -> i and ;; (make-reduce i1 i2 i3) -> #(i1 i2 i3) (define (parser-body debug? err starts ends table term-sym->index actions src-pos) (local ((define extract (if src-pos extract-src-pos extract-no-src-pos)) (define (fix-error stack tok val start-pos end-pos get-token) (when debug? (pretty-print stack)) (local ((define (remove-input tok val start-pos end-pos) (if (memq tok ends) (raise-read-error "parser: Cannot continue after error" #f #f #f #f #f) (let ((a (find-action stack tok val start-pos end-pos))) (cond ((shift? a) ;; (printf "shift:~a~n" (shift-state a)) (cons (make-stack-frame (shift-state a) val start-pos end-pos) stack)) (else ;; (printf "discard input:~a~n" tok) (let-values (((tok val start-pos end-pos) (extract (get-token)))) (remove-input tok val start-pos end-pos)))))))) (let remove-states () (let ((a (find-action stack 'error #f start-pos end-pos))) (cond ((shift? a) ;; (printf "shift:~a~n" (shift-state a)) (set! stack (cons (make-stack-frame (shift-state a) #f start-pos end-pos) stack)) (remove-input tok val start-pos end-pos)) (else ;; (printf "discard state:~a~n" (car stack)) (cond ((< (length stack) 2) (raise-read-error "parser: Cannot continue after error" #f #f #f #f #f)) (else (set! stack (cdr stack)) (remove-states))))))))) (define (find-action stack tok val start-pos end-pos) (let ((token-index (hash-table-get term-sym->index tok (lambda () #f)))) (if token-index (array2d-ref table (stack-frame-state (car stack)) token-index) (begin (if src-pos (err #f tok val start-pos end-pos) (err #f tok val)) (raise-read-error (format "parser: got token of unknown type ~a" tok) #f #f #f #f #f))))) (define (make-parser start-number) (lambda (get-token) (let parsing-loop ((stack (make-empty-stack start-number)) (ip (get-token))) (let-values (((tok val start-pos end-pos) (extract ip))) (let ((action (find-action stack tok val start-pos end-pos))) (cond ((shift? action) ;; (printf "shift:~a~n" (shift-state action)) (parsing-loop (cons (make-stack-frame (shift-state action) val start-pos end-pos) stack) (get-token))) ((reduce? action) ;; (printf "reduce:~a~n" (reduce-prod-num action)) (let-values (((new-stack args) (reduce-stack stack (reduce-rhs-length action) null src-pos))) (let* ((A (reduce-lhs-num action)) (goto (array2d-ref table (stack-frame-state (car new-stack)) A))) (parsing-loop (cons (if src-pos (make-stack-frame goto (apply (vector-ref actions (reduce-prod-num action)) args) (if (null? args) start-pos (cadr args)) (if (null? args) end-pos (list-ref args (- (* (reduce-rhs-length action) 3) 1)))) (make-stack-frame goto (apply (vector-ref actions (reduce-prod-num action)) args) #f #f)) new-stack) ip)))) ((accept? action) ;; (printf "accept~n") (stack-frame-value (car stack))) (else (if src-pos (err #t tok val start-pos end-pos) (err #t tok val)) (parsing-loop (fix-error stack tok val start-pos end-pos get-token) (get-token)))))))))) (cond ((null? (cdr starts)) (make-parser 0)) (else (let loop ((l starts) (i 0)) (cond ((null? l) null) (else (cons (make-parser i) (loop (cdr l) (add1 i)))))))))) )