polysemy/main.rkt
Georges Dupéron 66aed0320f Initial commit
2017-05-09 19:41:54 +02:00

157 lines
6.1 KiB
Racket

#lang racket/base
(provide
;;; Require transformer (does not work correctly, for now)
#;poly-in
;; Another require transformer
poly-rename-in
;; Alternative require form which handles polysemic ids
poly-require
;; Definition of a polysemic id, and of a part of a polysemic id
define-poly)
(require racket/match
(for-syntax racket/base
racket/contract
racket/string
racket/require-transform
syntax/parse))
;; This scope is used to hide and later identify parts of polysemic identifiers.
;; Each part is stored in a separate identifier.
(define-for-syntax poly-scope (make-syntax-introducer))
;; Utilities
;; _____________________________________________________________________________
;; Escapes the identifier, so that it does not contain the separator character
(begin-for-syntax
(define/contract (escape-symbol sym separator escape)
(-> symbol? char? char? string?)
(let ()
(define s1 (symbol->string sym))
(define s2 (string-replace s1
(format "~a" escape)
(format "~a~a" escape escape)))
(define s3 (string-replace s1
(format "~a" separator)
(format "~a~a" separator escape)))
s3)))
;; Generates a single-meaning identifier from `id` and `meaning`, possibly
;; escaping some characters in `meaning` to remove ambiguities.
(begin-for-syntax
(define/contract (gen-id ctx meaning id)
(-> syntax? symbol? identifier? identifier?)
(let ()
(define s (format " polysemy_~a_~a"
(escape-symbol meaning #\_ #\\)
(symbol->string (syntax-e id))))
(datum->syntax ctx (string->symbol s) id id))))
;; Require transformer
;; _____________________________________________________________________________
;; Require transformer which allows selecting and renaming parts of polysemic
;; parts of identifiers.
#;(define-syntax poly-in
(make-require-transformer
(λ (stx)
(syntax-case stx ()
[(_ mod id ...)
(let ()
;; Works, but we cannot bind a syntax transformer that way.
(define idd (syntax-local-lift-expression #'42))
;; Too late, top-level uses of macros have already been prefixed
;; with #%app:
(syntax-local-lift-module-end-declaration
#'(begin (define-syntax id (λ (stx) #`'(#,stx 42))) ...))
;; Won't work because we have to run expand-import before the
;; module has a chance to be injected:
(syntax-local-lift-module
#'(module m racket/base
(provide id ...)
(define-syntax id (λ (stx) #`'(#,stx 42))) ...))
(define-values (a b) (expand-import #'(only-in mod id ...)))
(define a*
(let ([local-id (import-local-id (car a))]
[src-sym (import-src-sym (car a))]
[src-mod-path (import-src-mod-path (car a))]
[mode (import-mode (car a))]
[req-mode (import-req-mode (car a))]
[orig-mode (import-orig-mode (car a))]
[orig-stx (import-orig-stx (car a))])
(list (import idd
src-sym
src-mod-path
mode
req-mode
orig-mode
orig-stx))))
(values a* b))]))))
(define-syntax poly-rename-in
(make-require-transformer
(syntax-parser
[(_ mod [old-id:id meaning:id new-id:id] ...)
(with-syntax ([(old-generated-id ...)
(map gen-id
(syntax->list #'(old-id ...))
(map syntax-e (syntax->list #'(meaning ...)))
(syntax->list #'(old-id ...)))]
[(new-generated-id ...)
(map gen-id
(syntax->list #'(new-id ...))
(map syntax-e (syntax->list #'(meaning ...)))
(syntax->list #'(new-id ...)))])
(expand-import
#'(rename-in mod [old-generated-id new-generated-id] ...)))])))
;; polysemic require (experiment, nothing interesting for now)
(define-syntax poly-require
(λ (stx)
(syntax-case stx ()
[(_ mod id ...)
(with-syntax ([(tmp ...) (generate-temporaries #'(id ...))])
#'(begin
(require (only-in mod [id tmp] ...))
(define-syntax id (λ (stx) #'42))
...))])))
;; Definition of polysemic identifiers and parts of these
;; _____________________________________________________________________________
;; Definition of a new polysemic identifier
(define-syntax (define-poly stx)
(syntax-case stx ()
[(_ id)
#'(define-syntax id (polysemic #'id))]
[(_ id meaning value)
(with-syntax ([generated-id (gen-id #'id (syntax-e #'meaning) #'id)])
#'(define-syntax generated-id value))]))
;; Creates a wrapper for a prop:…, by extracting the the given `meaning`
;; for the identifier.
(define-for-syntax ((make-wrapper meaning) self stx)
((syntax-local-value (gen-id (car (syntax-e stx)) meaning (polysemic-id self))) stx))
;; Wrapper for prop:procedure on a transformer id.
;; Dispatches to
(define-for-syntax (macro-wrapper self stx)
(define id (polysemic-id self))
(if (syntax? stx)
(syntax-case stx (set!)
[x
(identifier? #'x)
((syntax-local-value (gen-id #'x 'identifier-macro id)) stx)]
[(set! v . _)
((syntax-local-value (gen-id #'v 'set!-macro id)) stx)]
[(self . _)
((syntax-local-value (gen-id #'self 'normal-macro id)) stx)])
(error "oops")#;((syntax-local-value (gen-id 'normal-macro id)) stx)))
;; Instances of this struct are bound (as transformer values) to polysemic ids.
(begin-for-syntax
(struct polysemic (id)
#:property prop:match-expander (make-wrapper 'match-expander)
#:property prop:procedure macro-wrapper))