up
This commit is contained in:
parent
895560bb75
commit
d86eb1951b
150
remix/module.rkt
Normal file
150
remix/module.rkt
Normal file
|
@ -0,0 +1,150 @@
|
|||
;; XXX implement @lang+ and @lang*+
|
||||
(module module '#%kernel
|
||||
(#%require racket/private/more-scheme
|
||||
racket/private/modbeg
|
||||
(for-syntax '#%kernel
|
||||
racket/private/stxcase-scheme
|
||||
racket/private/more-scheme
|
||||
racket/private/letstx-scheme
|
||||
racket/private/qqstx))
|
||||
|
||||
(#%provide module+
|
||||
define-module+
|
||||
define-module*+)
|
||||
|
||||
(begin-for-syntax
|
||||
(define-values (expect-identifier-for)
|
||||
(lambda (whole-stx stx where can-be-false?)
|
||||
(define-values (v) (syntax-e stx))
|
||||
(unless (or (and can-be-false? (not v))
|
||||
(symbol? v))
|
||||
(raise-syntax-error #f
|
||||
(format
|
||||
"expected an identifier for ~a, found something else"
|
||||
where)
|
||||
whole-stx
|
||||
stx))))
|
||||
|
||||
(define-values (do-define-module+)
|
||||
(lambda (this which-module stx)
|
||||
(case (syntax-local-context)
|
||||
[(module-begin)
|
||||
(quasisyntax/loc stx (begin #,stx))]
|
||||
[(module)
|
||||
(syntax-case stx ()
|
||||
[(_ the-module+ the-submodule the-module-lang)
|
||||
(begin
|
||||
(expect-identifier-for stx #'the-module+ "the module+ form" #f)
|
||||
(expect-identifier-for stx #'the-submodule "a submodule" #f)
|
||||
(expect-identifier-for stx #'the-module-lang "the module language" #t)
|
||||
(quasisyntax/loc stx
|
||||
(define-syntaxes (the-module+)
|
||||
(lambda (module+-stx)
|
||||
(do-module+-for #'#,which-module
|
||||
#'the-submodule
|
||||
(syntax-local-introduce #'the-module-lang)
|
||||
(syntax-local-introduce #'the-module-lang)
|
||||
module+-stx)))))])])))
|
||||
|
||||
(define-values (do-module+-for)
|
||||
(lambda (which-module-stx the-submodule-stx the-module-lang-stx context-stx stx)
|
||||
(case (syntax-local-context)
|
||||
[(module-begin)
|
||||
(quasisyntax/loc stx (begin #,stx))]
|
||||
[(module)
|
||||
(syntax-case stx ()
|
||||
[(_ #:declared)
|
||||
(quasisyntax/loc stx
|
||||
(define-module
|
||||
#,which-module-stx
|
||||
#,the-submodule-stx
|
||||
#,the-module-lang-stx))]
|
||||
[(_ e ...)
|
||||
(begin
|
||||
(when (hash-has-key? submodule->defined? (syntax-e the-submodule-stx))
|
||||
(raise-syntax-error #f "submodule is already declared" stx))
|
||||
;; This looks it up the first time and is allowed to create a
|
||||
;; list and lift a module-end declaration if necessary:
|
||||
(let ([stxs-box (get-stxs-box which-module-stx
|
||||
context-stx
|
||||
the-submodule-stx
|
||||
the-module-lang-stx
|
||||
#t)])
|
||||
(set-box!
|
||||
stxs-box
|
||||
(append (reverse (syntax->list (syntax-local-introduce #'(e ...))))
|
||||
(unbox stxs-box))))
|
||||
(syntax/loc stx (begin)))])]
|
||||
[else
|
||||
(raise-syntax-error #f
|
||||
"allowed only in a module body"
|
||||
stx)]))))
|
||||
|
||||
(define-syntaxes (define-module+)
|
||||
(lambda (stx)
|
||||
(do-define-module+ 'define-module+ #'module stx)))
|
||||
|
||||
(define-syntaxes (define-module*+)
|
||||
(lambda (stx)
|
||||
(do-define-module+ 'define-module*+ #'module* stx)))
|
||||
|
||||
(define-syntaxes (module+)
|
||||
(lambda (stx)
|
||||
(syntax-case stx ()
|
||||
[(_ the-submodule e ...)
|
||||
(begin
|
||||
(expect-identifier-for stx #'the-submodule "a submodule" #f)
|
||||
(do-module+-for #'module* #'the-submodule #'#f
|
||||
stx
|
||||
#'(fake-the-submodule+ e ...)))])))
|
||||
|
||||
(begin-for-syntax
|
||||
;; The following table is newly instantiated for each module
|
||||
;; expansion that uses `module+', so it is effectively
|
||||
;; module-local:
|
||||
(define-values (submodule->stxs-box) (make-weak-hash))
|
||||
(define-values (get-stxs-box)
|
||||
(lambda (which-module-stx form-stx the-submodule-stx the-module-lang-stx lift?)
|
||||
(hash-ref! submodule->stxs-box (syntax-e the-submodule-stx)
|
||||
(lambda ()
|
||||
(when lift?
|
||||
(syntax-local-lift-module-end-declaration
|
||||
;; Use the lexical context of the first `module+'
|
||||
;; form as the context of the implicit `#%module-begin':
|
||||
(datum->syntax
|
||||
form-stx
|
||||
(list #'define-module
|
||||
which-module-stx
|
||||
the-submodule-stx
|
||||
the-module-lang-stx)
|
||||
form-stx)))
|
||||
(box null)))))
|
||||
|
||||
(define-values (submodule->defined?) (make-weak-hash))
|
||||
(define-values (defined-or-define!)
|
||||
(lambda (the-submodule-stx)
|
||||
(let-values ([(k) (syntax-e the-submodule-stx)])
|
||||
(begin0 (hash-ref submodule->defined? k #t)
|
||||
(hash-ref! submodule->defined? k #f))))))
|
||||
|
||||
;; A use of this form is lifted to the end of the enclosing module
|
||||
;; for each submodule created by `module+':
|
||||
(define-syntaxes (define-module)
|
||||
(lambda (stx)
|
||||
(syntax-case stx ()
|
||||
[(_ which-module the-submodule the-module-lang)
|
||||
(cond
|
||||
[(defined-or-define! #'the-submodule)
|
||||
(let ([stxs-box (get-stxs-box #f #f #'the-submodule #f #f)])
|
||||
;; Propagate the lexical context of the first `module+'
|
||||
;; for the implicit `#%module-begin':
|
||||
(datum->syntax
|
||||
stx
|
||||
(list*
|
||||
#'which-module
|
||||
#'the-submodule
|
||||
#'the-module-lang
|
||||
(map syntax-local-introduce (reverse (unbox stxs-box))))
|
||||
stx))]
|
||||
[else
|
||||
(syntax/loc stx (begin))])]))))
|
63
remix/tests/module.rkt
Normal file
63
remix/tests/module.rkt
Normal file
|
@ -0,0 +1,63 @@
|
|||
#lang racket/base
|
||||
(require "../module.rkt")
|
||||
|
||||
;; A normal `module` declaration can use a different module-language
|
||||
(module og-duck racket/base
|
||||
(provide num-eggs quack)
|
||||
(define num-eggs 2)
|
||||
(define (quack n)
|
||||
(unless (zero? n)
|
||||
(printf "quack\n")
|
||||
(quack (sub1 n)))))
|
||||
;; And be required by the parent module
|
||||
(require (prefix-in og: 'og-duck))
|
||||
(og:quack og:num-eggs)
|
||||
|
||||
;; module+ can't do this though, because there's nowhere to write down
|
||||
;; that you don't want to use (module* .... #f ....)
|
||||
(module+ tests
|
||||
(printf "Old module+..."))
|
||||
|
||||
(module+ tests
|
||||
(printf "works!\n"))
|
||||
|
||||
;; define-module+ lets you define a new `module+` like form
|
||||
;; specialized for a particular module. This, however, is like
|
||||
;; `module` and not `module*`, so it can't require the parent
|
||||
(define-module+ duck+
|
||||
;; You tell it what the name of the module to define is
|
||||
duck
|
||||
;; And what the module language is
|
||||
racket/base)
|
||||
|
||||
;; Now we start filling in the module
|
||||
(duck+ (provide num-eggs quack)
|
||||
(define num-eggs 2))
|
||||
(duck+ (define (quack n)
|
||||
(unless (zero? n)
|
||||
(printf "quack\n")
|
||||
(quack (sub1 n)))))
|
||||
|
||||
;; Since this is a 'module', we need to finish declaring it at some
|
||||
;; point if we want the parent to get it. If we didn't want that, we
|
||||
;; wouldn't need to use this form.
|
||||
(duck+ #:declared)
|
||||
|
||||
;; So, here we can get at it:
|
||||
(require 'duck)
|
||||
(quack num-eggs)
|
||||
|
||||
;; This is a syntax error because `duck` is declared
|
||||
#;(duck+ (printf "Yo!\n"))
|
||||
|
||||
(define nine 9)
|
||||
(provide nine)
|
||||
|
||||
;; We can, of course, defined a `module*` like this.
|
||||
(define-module*+ main+
|
||||
main racket/base)
|
||||
(main+ (require (submod "..")))
|
||||
(main+ (displayln nine))
|
||||
;; We aren't required to put in #:declared and probably never would
|
||||
;; for `define-module*+`
|
||||
|
Loading…
Reference in New Issue
Block a user