racket/collects/profj/ast.ss

314 lines
11 KiB
Scheme

(module ast mzscheme
;Macro to allow structure definition and provision
(define-syntax p-define-struct
(syntax-rules ()
[(_ (name inherit) fields)
(begin
(provide (struct name fields))
(define-struct (name inherit) fields (make-inspector)))]
[(_ name fields)
(begin
(provide (struct name fields))
(define-struct name fields (make-inspector)))]))
;(make-src int int int int loc)
(p-define-struct src (line col pos span file))
(provide src->list)
(define (src->list src)
(list (src-file src) (src-line src) (src-col src) (src-pos src) (src-span src)))
;;(make-package (U #f name) (list import) (list (U class-def interface-def)))
(p-define-struct package (name imports defs))
;;(make-name id (list id) src)
(p-define-struct name (id path src))
;;(make-id string src)
(p-define-struct id (string src))
;; import java.lang.* -> (make-import (make-name "lang" `("java") ...) #t ...)
;;(make-import name bool src src string)
(p-define-struct import (name star key-src src file))
(provide def-name)
(define (def-name d)
(header-id (def-header d)))
;;(make-def header (list member) src src string symbol (list req) symbol (list id))
(p-define-struct def (header members key-src src file level uses kind closure-args))
;;(make-class-def header (list member) src src string symbol (list req) symbol)
;; members in file order
(p-define-struct (class-def def) ())
;;(make-interface-def header (list member) src src string symbol (list req) symbol)
(p-define-struct (interface-def def) ())
;(make-test-def test-header (list member) src src string symbol (list req) symbol)
(p-define-struct (test-def def) ())
;;(make-req string (list string))
(p-define-struct req (class path))
;;(make-header id (list modifier) (list name) (list name) (list gj-info) src)
(p-define-struct header (id modifiers extends implements type-parms src))
;;(make-test-header id (list modifier) (list name) (list name) (list gj-info) src (list name))
(p-define-struct (test-header header) (tests))
;;(make-modifier symbol src)
(p-define-struct modifier (kind src))
;;member = var-decl
;; | var-init
;; | method
;; | initialize
;; | class-def
;; | interface-def
;;(make-type-spec (U name type-var symbol) int src)
;; dim is for array dimensions
(p-define-struct type-spec (name dim src))
;;(make-type-var symbol (U #f type-spec) src)
(p-define-struct type-var (name bound src))
;;Code for accessing fields: var-decl and var-init
(provide field? field-name field-modifiers field-type-spec field-type set-field-type! field-src)
(define (field? v) (or (var-decl? v) (var-init? v)))
(define (field-name v) (var-decl-name (if (var-init? v) (var-init-var-decl v) v)))
(define (field-modifiers v) (var-decl-modifiers (if (var-init? v) (var-init-var-decl v) v)))
(define (field-type-spec v) (var-decl-type-spec (if (var-init? v) (var-init-var-decl v) v)))
(define (field-type v) (var-decl-type (if (var-init? v) (var-init-var-decl v) v)))
(define (set-field-type! v t) (set-var-decl-type! (if (var-init? v) (var-init-var-decl v) v) t))
(define (field-src v) (var-decl-src (if (var-init? v) (var-init-var-decl v) v)))
;;(make-var-decl id (list modifier) type-spec (U #f type) src)
(p-define-struct var-decl (name modifiers type-spec type src))
;;(make-var-init var-decl (U array-init expression) src)
(p-define-struct var-init (var-decl init src))
;;(make-array-init (list (U expression array-init)) src)
(p-define-struct array-init (vals src))
;;(make-method (list modifier) type-spec null id (list var-decl) (list name) Statement bool method-record src)
(p-define-struct method (modifiers type type-parms name parms throws body all-tail? rec src))
;;(make-test-method (list modifier) type-spec null id (list var-decl) (list name) Statement bool method-record src)
(p-define-struct (test-method method) ())
;;(make-initialize bool block src)
(p-define-struct initialize (static block src))
(provide statement?)
;statement? 'a -> bool
(define (statement? stmt)
(or (ifS? stmt) (throw? stmt) (return? stmt) (while? stmt) (doS? stmt)
(for? stmt) (try? stmt) (switch? stmt) (block? stmt) (break? stmt)
(continue? stmt) (label? stmt) (synchronized? stmt) (statement-expression? stmt)))
;statement => if
; | throw
; | return
; | while
; | do
; | for
; | try
; | switch
; | block
; | break
; | continue
; | label
; | synchronized
; | StatementExpression
;StatementExpression => call
; | post-expr
; | preExpr
; | assignment
; | class-alloc
;(make-ifS Expression Statement Statement src src)
(p-define-struct ifS (cond then else key-src src))
;(make-throw Expression src src)
(p-define-struct throw (expr key-src src))
;(make-return Expression (U #f type) boolean src)
(p-define-struct return (expr exp-type in-tail? src))
;(make-while Expression Statement src)
(p-define-struct while (cond loop src))
;(make-do Statement Expression src)
(p-define-struct doS (loop cond src))
;(make-for forInit Expression (list Expression) Statement src)
(p-define-struct for (init cond incr loop src))
;forInit => (list (U var-init var-decl))
; | (list StatementExpression)
;(make-try Block (list Catch) (U #f statement) src src)
(p-define-struct try (body catches finally key-src src))
;(make-catch var-decl statement src)
(p-define-struct catch (cond body src))
;(make-switch Expression CaseStatements src)
(p-define-struct switch (expr cases src))
;CaseStatements = (list case)
;(make-case (list (U ConstantExpression `default)) (list (U var-decl var-init Statement)) src)
(p-define-struct caseS (constant body src))
;(make-block (list (U var-decl var-init Statement)) src)
(p-define-struct block (stmts src))
;(make-break (U #f id) src)
(p-define-struct break (label src))
;(make-continue (U #f id) src)
(p-define-struct continue (label src))
;(make-label id statement src)
(p-define-struct label (label stmt src))
;(make-synchronized expression statement src)
(p-define-struct synchronized (expr stmt src))
(provide statement-expression?)
;statement-expression?: StatementExpression -> bool
(define (statement-expression? stmt)
(or (call? stmt)
(post-expr? stmt)
(pre-expr? stmt)
(unary? stmt)
(assignment? stmt)
(class-alloc? stmt)
(inner-alloc? stmt)))
;(make-expr (U #f type) src)
(p-define-struct expr (types src))
;Expression => literal
; | bin-op
; | access
; | special-name
; | specified-this
; | call
; | class-alloc
; | array-alloc
; | cond-expression
; | array-access
; | post-expr
; | pre-expr
; | unary
; | cast
; | instanceof
; | assignment
; | check
;(make-literal (U #f type) src value)
(p-define-struct (literal expr) (val))
;value => number | string
;(make-bin-op (U #f type) src binary-op Expression Expression src)
(p-define-struct (bin-op expr) (op left right key-src))
;binary-op => + - * / % << >> >>> < > <= >= == != & ^ or && oror
;(make-access (U #f type) src (U (list id) field-access local-access))
;Types before check
;After check, (list id) -> (U field-access local-access)
(p-define-struct (access expr) (name))
;(make-field-access (U Expression #f) id var-access)
(p-define-struct field-access (object field access))
;;(make-var-access bool bool bool symbol string)
(p-define-struct var-access (static? final? init? access class))
;(make-local-access id)
(p-define-struct local-access (name))
;(make-special-name (U #f type) src string)
(p-define-struct (special-name expr) (name))
;(make-specified-this (U #f type) src name (U string #f))
(p-define-struct (specified-this expr) (class var))
;(make-call (U #f type) src (U #f expression) MethodName (list Expression) (U #f method-record))
(p-define-struct (call expr) (expr method-name args method-record))
;MethodName => special-name
; | id
;(make-class-alloc (U #f type) src name (list Expression) (U #f method-record) bool)
(p-define-struct (class-alloc expr) (name args ctor-record class-inner? local-inner?))
;(make-inner-alloc (U #f type) src expr name (list Expression) (U #f method-record))
(p-define-struct (inner-alloc expr) (obj name args ctor-record))
;(make-array-alloc (U #f type) src type-spec (list Expression) int)
(p-define-struct (array-alloc expr) (name size dim))
;;(make-array-alloc-init (U #f type) src type-spec int array-init)
(p-define-struct (array-alloc-init expr) (name dim init))
;(make-cond-expression (U #f type) src Expression Expression Expression src)
(p-define-struct (cond-expression expr) (cond then else key-src))
;(make-array-access (U #f type) src expression Expression)
(p-define-struct (array-access expr) (name index))
;(make-post-expr (U #f type) src Expression PrePost src)
(p-define-struct (post-expr expr) (expr op key-src))
;PrePost => ++ --
;(make-pre-expr (U #f type) src PrePost Expression src)
(p-define-struct (pre-expr expr) (op expr key-src))
;(make-unary (U #f type) src UnaryOp Expression src)
(p-define-struct (unary expr) (op expr key-src))
;UnaryOp => + - ~ !
;(make-cast (U #f type) src type-spec Expression)
(p-define-struct (cast expr) (type expr))
;(make-instanceof (U #f type) src Expression type-spec src)
(p-define-struct (instanceof expr) (expr type key))
;Note: lefthand side might be incorrect
;(make-assignment (U #f type) src (U access array-access) symbol Expression src)
(p-define-struct (assignment expr) (left op right key-src))
;Op -> = *= /= %= += -= <<= >>= >>>= &= ^= or=
(p-define-struct (check expr) ())
;(make-check-expect (U #f type) src Expression Expression (U #f Expression) src)
(p-define-struct (check-expect check) (test actual range ta-src))
;(make-check-rand (U #f type) src Expression Expression src)
(p-define-struct (check-rand check) (test range ta-src))
;(make-check-catch (U #f type) src Expression type-spec)
(p-define-struct (check-catch check) (test exn))
;(make-check-by (U #f type) src Expression Expression (U '== Name))
(p-define-struct (check-by check) (test actual compare))
;(make-check-mutate (U #f type) src Expression Expression src)
(p-define-struct (check-mutate check) (mutate check op-src))
)