Initial commit
This commit is contained in:
commit
2c8423ac16
6
.gitignore
vendored
Normal file
6
.gitignore
vendored
Normal file
|
@ -0,0 +1,6 @@
|
|||
*~
|
||||
\#*
|
||||
.\#*
|
||||
.DS_Store
|
||||
compiled/
|
||||
/doc/
|
58
.travis.yml
Normal file
58
.travis.yml
Normal file
|
@ -0,0 +1,58 @@
|
|||
language: c
|
||||
|
||||
# Based from: https://github.com/greghendershott/travis-racket
|
||||
|
||||
# Optional: Remove to use Travis CI's older infrastructure.
|
||||
sudo: false
|
||||
|
||||
env:
|
||||
global:
|
||||
# Supply a global RACKET_DIR environment variable. This is where
|
||||
# Racket will be installed. A good idea is to use ~/racket because
|
||||
# that doesn't require sudo to install and is therefore compatible
|
||||
# with Travis CI's newer container infrastructure.
|
||||
- RACKET_DIR=~/racket
|
||||
matrix:
|
||||
# Supply at least one RACKET_VERSION environment variable. This is
|
||||
# used by the install-racket.sh script (run at before_install,
|
||||
# below) to select the version of Racket to download and install.
|
||||
#
|
||||
# Supply more than one RACKET_VERSION (as in the example below) to
|
||||
# create a Travis-CI build matrix to test against multiple Racket
|
||||
# versions.
|
||||
- RACKET_VERSION=6.0
|
||||
- RACKET_VERSION=6.1
|
||||
- RACKET_VERSION=6.1.1
|
||||
- RACKET_VERSION=6.2
|
||||
- RACKET_VERSION=6.3
|
||||
- RACKET_VERSION=6.4
|
||||
- RACKET_VERSION=6.5
|
||||
- RACKET_VERSION=6.6
|
||||
- RACKET_VERSION=6.7
|
||||
- RACKET_VERSION=HEAD
|
||||
|
||||
matrix:
|
||||
allow_failures:
|
||||
# - env: RACKET_VERSION=HEAD
|
||||
fast_finish: true
|
||||
|
||||
before_install:
|
||||
- git clone https://github.com/greghendershott/travis-racket.git ~/travis-racket
|
||||
- cat ~/travis-racket/install-racket.sh | bash # pipe to bash not sh!
|
||||
- export PATH="${RACKET_DIR}/bin:${PATH}" #install-racket.sh can't set for us
|
||||
|
||||
install:
|
||||
- raco pkg install --deps search-auto
|
||||
|
||||
before_script:
|
||||
|
||||
# Here supply steps such as raco make, raco test, etc. You can run
|
||||
# `raco pkg install --deps search-auto` to install any required
|
||||
# packages without it getting stuck on a confirmation prompt.
|
||||
script:
|
||||
- raco test -x -p debug-scopes
|
||||
|
||||
after_success:
|
||||
- raco setup --check-pkg-deps --pkgs debug-scopes
|
||||
- raco pkg install --deps search-auto cover cover-coveralls
|
||||
- raco cover -b -f coveralls -d $TRAVIS_BUILD_DIR/coverage .
|
11
LICENSE.txt
Normal file
11
LICENSE.txt
Normal file
|
@ -0,0 +1,11 @@
|
|||
debug-scopes
|
||||
Copyright (c) 2016 georges
|
||||
|
||||
This package is distributed under the GNU Lesser General Public
|
||||
License (LGPL). This means that you can link debug-scopes into proprietary
|
||||
applications, provided you follow the rules stated in the LGPL. You
|
||||
can also modify this package; if you distribute a modified version,
|
||||
you must distribute it under the terms of the LGPL, which in
|
||||
particular means that you must release the source code for the
|
||||
modified software. See http://www.gnu.org/copyleft/lesser.html
|
||||
for more information.
|
9
info.rkt
Normal file
9
info.rkt
Normal file
|
@ -0,0 +1,9 @@
|
|||
#lang info
|
||||
(define collection "debug-scopes")
|
||||
(define deps '("base"
|
||||
"rackunit-lib"))
|
||||
(define build-deps '("scribble-lib" "racket-doc"))
|
||||
(define scribblings '(("scribblings/debug-scopes.scrbl" ())))
|
||||
(define pkg-desc "Description Here")
|
||||
(define version "0.0")
|
||||
(define pkg-authors '(georges))
|
35
main.rkt
Normal file
35
main.rkt
Normal file
|
@ -0,0 +1,35 @@
|
|||
#lang racket/base
|
||||
|
||||
(module+ test
|
||||
(require rackunit))
|
||||
|
||||
;; Notice
|
||||
;; To install (from within the package directory):
|
||||
;; $ raco pkg install
|
||||
;; To install (once uploaded to pkgs.racket-lang.org):
|
||||
;; $ raco pkg install <<name>>
|
||||
;; To uninstall:
|
||||
;; $ raco pkg remove <<name>>
|
||||
;; To view documentation:
|
||||
;; $ raco docs <<name>>
|
||||
;;
|
||||
;; For your convenience, we have included a LICENSE.txt file, which links to
|
||||
;; the GNU Lesser General Public License.
|
||||
;; If you would prefer to use a different license, replace LICENSE.txt with the
|
||||
;; desired license.
|
||||
;;
|
||||
;; Some users like to add a `private/` directory, place auxiliary files there,
|
||||
;; and require them in `main.rkt`.
|
||||
;;
|
||||
;; See the current version of the racket style guide here:
|
||||
;; http://docs.racket-lang.org/style/index.html
|
||||
|
||||
;; Code here
|
||||
|
||||
(module+ test
|
||||
;; Tests to be run with raco test
|
||||
)
|
||||
|
||||
(module+ main
|
||||
;; Main entry point, executed when run with the `racket` executable or DrRacket.
|
||||
)
|
7
named-scopes.rkt
Normal file
7
named-scopes.rkt
Normal file
|
@ -0,0 +1,7 @@
|
|||
#lang racket
|
||||
|
||||
(require debug-scopes/named-scopes/exptime)
|
||||
(require (for-template debug-scopes/named-scopes/override))
|
||||
|
||||
(provide (all-from-out debug-scopes/named-scopes/exptime)
|
||||
(for-template (all-from-out debug-scopes/named-scopes/override)))
|
10
named-scopes/dummy-lang.rkt
Normal file
10
named-scopes/dummy-lang.rkt
Normal file
|
@ -0,0 +1,10 @@
|
|||
#lang racket
|
||||
(provide (rename-out [my-module-begin #%module-begin]))
|
||||
(define-syntax (my-module-begin stx)
|
||||
(syntax-case stx ()
|
||||
[(_ real-lang body)
|
||||
(syntax-case (local-expand #'(module m real-lang body) 'top-level (list)) ()
|
||||
[(module nm lng (#%plain-module-begin . body2))
|
||||
#`(#%plain-module-begin
|
||||
(#%require real-lang)
|
||||
. #,(values #'body2))])]))
|
146
named-scopes/exptime.rkt
Normal file
146
named-scopes/exptime.rkt
Normal file
|
@ -0,0 +1,146 @@
|
|||
#lang racket
|
||||
|
||||
(require (for-template '#%kernel)
|
||||
type-expander/debug-scopes
|
||||
racket/syntax
|
||||
racket/struct
|
||||
type-expander/debug-scopes)
|
||||
|
||||
(provide make-named-scope
|
||||
named-transformer
|
||||
(rename-out [-syntax-local-introduce syntax-local-introduce]))
|
||||
|
||||
(define (use-site-context?)
|
||||
(not (bound-identifier=? (syntax-local-introduce #'here)
|
||||
(syntax-local-identifier-as-binding
|
||||
(syntax-local-introduce #'here)))))
|
||||
|
||||
(define (make-named-scope nm)
|
||||
(define name (if (symbol? nm) nm (string->symbol nm)))
|
||||
(define E1
|
||||
(local-expand (datum->syntax #f
|
||||
`(,#'module
|
||||
,name
|
||||
debug-scopes/named-scopes/dummy-lang
|
||||
'#%kernel
|
||||
list))
|
||||
'top-level
|
||||
(list)))
|
||||
(define/with-syntax (_module _name _lang (_modbeg (_#%require QK1) Body1)) E1)
|
||||
(define QK (datum->syntax #'QK1 'qk-sym))
|
||||
(define Body (datum->syntax #'Body1 'body-sym))
|
||||
(define Zero (datum->syntax #f 'zero))
|
||||
(define ΔBody (make-syntax-delta-introducer Body Zero))
|
||||
(define QK-Body (ΔBody QK 'remove))
|
||||
(define ΔQK-Body (make-syntax-delta-introducer QK-Body Zero))
|
||||
(define QK-rest (ΔQK-Body QK 'remove))
|
||||
(define named-scope (make-syntax-delta-introducer QK-rest Zero))
|
||||
named-scope)
|
||||
|
||||
(define ((has-scope scope) stx)
|
||||
(and (identifier? stx)
|
||||
(bound-identifier=? stx (scope stx 'add))))
|
||||
|
||||
(define (replace-scope old new)
|
||||
(define (replace e)
|
||||
(cond
|
||||
[(syntax? e)
|
||||
(datum->syntax (if ((has-scope old) e)
|
||||
(new (old e 'remove) 'add)
|
||||
e)
|
||||
(replace (syntax-e e))
|
||||
e
|
||||
e)]
|
||||
[(pair? e) (cons (replace (car e)) (replace (cdr e)))]
|
||||
[(vector? e) (list->vector (replace (vector->list e)))]
|
||||
[(hash? e)
|
||||
(cond [(hash-eq? e) (make-hasheq (replace (hash->list e)))]
|
||||
[(hash-eqv? e) (make-hasheqv (replace (hash->list e)))]
|
||||
[(hash-equal? e) (make-hash (replace (hash->list e)))]
|
||||
[else e])]
|
||||
[(prefab-struct-key e)
|
||||
=> (λ (k)
|
||||
(apply make-prefab-struct k (replace (struct->list e))))]
|
||||
[else e]))
|
||||
replace)
|
||||
|
||||
(define (deep-has-scope sc)
|
||||
(define (scan e)
|
||||
(cond
|
||||
[(syntax? e) (or ((has-scope sc) e) (scan (syntax-e e)))]
|
||||
[(pair? e) (or (scan (car e)) (scan (cdr e)))]
|
||||
[(vector? e) (scan (vector->list e))]
|
||||
[(hash? e) (scan (hash->list e))]
|
||||
[(prefab-struct-key e) (scan (struct->list e))]
|
||||
[else #f]))
|
||||
scan)
|
||||
|
||||
(define (old-macro-scope)
|
||||
(make-syntax-delta-introducer
|
||||
(syntax-local-identifier-as-binding
|
||||
(syntax-local-introduce
|
||||
(datum->syntax #f 'zero)))
|
||||
(datum->syntax #f 'zero)))
|
||||
|
||||
(define (old-use-site-scope)
|
||||
(make-syntax-delta-introducer
|
||||
((old-macro-scope) (syntax-local-introduce (datum->syntax #f 'zero)) 'remove)
|
||||
(datum->syntax #f 'zero)))
|
||||
|
||||
(define (convert-macro-scopes stx)
|
||||
(if (sli-scopes)
|
||||
(let* ([macro (car (sli-scopes))]
|
||||
[use-site (cdr (sli-scopes))]
|
||||
[old-macro (old-macro-scope)]
|
||||
[old-use (old-use-site-scope)])
|
||||
((compose (if (use-site-context?)
|
||||
(replace-scope old-use use-site)
|
||||
(λ (x) x))
|
||||
(replace-scope old-macro macro))
|
||||
stx))
|
||||
;; Otherwise leave unchanged.
|
||||
stx))
|
||||
|
||||
(define ((named-transformer-wrap name f) stx)
|
||||
(parameterize ([sli-scopes
|
||||
(cons (make-named-scope (format "macro: ~a" name))
|
||||
(if (use-site-context?)
|
||||
(make-named-scope (format "use-site: ~a" name))
|
||||
(make-syntax-delta-introducer
|
||||
(datum->syntax #f 'zero)
|
||||
(datum->syntax #f 'zero))))])
|
||||
;;; TODO: we should detect the presence of old-* here instead, and 'add them
|
||||
(displayln (+scopes stx))
|
||||
(displayln (use-site-context?))
|
||||
(displayln (+scopes (convert-macro-scopes stx)))
|
||||
(let ([res (f (convert-macro-scopes stx))])
|
||||
(when ((deep-has-scope (old-macro-scope)) res)
|
||||
(error (format "original macro scope appeared within the result of a named transformer: ~a\n~a\n~a"
|
||||
res
|
||||
(+scopes res)
|
||||
(with-output-to-string (λ () (print-full-scopes))))))
|
||||
(when (and (use-site-context?)
|
||||
((deep-has-scope (old-use-site-scope)) res))
|
||||
(error "original use-site scope appeared within the result of a named transformer"))
|
||||
(let* ([/mm ((car (sli-scopes)) res 'flip)]
|
||||
[/mm/uu (if (use-site-context?) ((cdr (sli-scopes)) /mm 'flip) /mm)]
|
||||
[/mm/uu+m ((old-macro-scope) /mm/uu 'add)])
|
||||
(if (use-site-context?)
|
||||
((old-use-site-scope) /mm/uu+m 'add)
|
||||
/mm/uu+m)))))
|
||||
|
||||
(define-syntax-rule (named-transformer (name stx) . body)
|
||||
(named-transformer-wrap 'name (λ (stx) . body)))
|
||||
|
||||
(define sli-scopes (make-parameter #f))
|
||||
|
||||
(define (-syntax-local-introduce stx)
|
||||
(if (sli-scopes)
|
||||
((cdr (sli-scopes)) ((car (sli-scopes)) stx 'flip)
|
||||
'flip)
|
||||
(syntax-local-introduce stx)))
|
||||
|
||||
(define (-syntax-local-identifier-as-binding stx)
|
||||
(if (and (sli-scopes) (use-site-context?))
|
||||
((cdr (sli-scopes)) stx 'flip)
|
||||
(syntax-local-introduce stx)))
|
12
named-scopes/override.rkt
Normal file
12
named-scopes/override.rkt
Normal file
|
@ -0,0 +1,12 @@
|
|||
#lang racket
|
||||
|
||||
(require (for-syntax "exptime.rkt"))
|
||||
|
||||
(provide (rename-out [-define-syntax define-syntax]))
|
||||
|
||||
(define-syntax (-define-syntax stx)
|
||||
(syntax-case stx ()
|
||||
[(_ (name arg) . body) #'(define-syntax name
|
||||
(named-transformer (name arg)
|
||||
. body))]
|
||||
[(_ name value) #'(define-syntax name value)]))
|
10
scribblings/debug-scopes.scrbl
Normal file
10
scribblings/debug-scopes.scrbl
Normal file
|
@ -0,0 +1,10 @@
|
|||
#lang scribble/manual
|
||||
@require[@for-label[debug-scopes
|
||||
racket/base]]
|
||||
|
||||
@title{debug-scopes}
|
||||
@author{georges}
|
||||
|
||||
@defmodule[debug-scopes]
|
||||
|
||||
Package Description Here
|
24
test/named-scopes-test-def.rkt
Normal file
24
test/named-scopes-test-def.rkt
Normal file
|
@ -0,0 +1,24 @@
|
|||
#lang racket
|
||||
|
||||
(require (for-syntax debug-scopes/named-scopes
|
||||
type-expander/debug-scopes ;;;
|
||||
syntax/stx))
|
||||
|
||||
(begin-for-syntax
|
||||
(define-syntax-rule (named-transformer (_ stx) . body) (λ (stx) . body))
|
||||
(define (make-named-scope _) (make-syntax-introducer)))
|
||||
|
||||
(provide foo-macro bar-macro baz-macro)
|
||||
|
||||
(define-syntax (foo-macro stx)
|
||||
(syntax-case stx ()
|
||||
[(_ a)
|
||||
(let ([foo-scope (make-named-scope 'my-foo-scope-wohoo)])
|
||||
(foo-scope #'a))]))
|
||||
|
||||
(define-syntax bar-macro
|
||||
(named-transformer (bar-macro stx)
|
||||
#`(let ([x 1]) . #,(stx-cdr stx))))
|
||||
|
||||
(define-syntax (baz-macro stx)
|
||||
#`(let ([x 5]) . #,(stx-cdr stx)))
|
32
test/named-scopes-test-use.rkt
Normal file
32
test/named-scopes-test-use.rkt
Normal file
|
@ -0,0 +1,32 @@
|
|||
#lang racket
|
||||
|
||||
(require ;"named-scopes-test-def.rkt"
|
||||
rackunit
|
||||
(for-syntax type-expander/debug-scopes
|
||||
;debug-scopes/named-scopes
|
||||
))
|
||||
|
||||
#|
|
||||
(define r1 (foo-macro +))
|
||||
(define r2 (let ([x 2])
|
||||
(bar-macro x)))
|
||||
(define r3 (let ([x 3])
|
||||
(baz-macro x)))
|
||||
|
||||
(define r4 (let ()
|
||||
(define-syntax (quux stx)
|
||||
(syntax-local-introduce #'+))
|
||||
(quux)))
|
||||
|
||||
(check-equal? (list r1 r2 r3 r4) (list + 2 3 +))
|
||||
|#
|
||||
|
||||
(define-syntax (quux stx)
|
||||
(syntax-case stx ()
|
||||
[(_ m)
|
||||
(let ()
|
||||
(displayln (+scopes #'m))
|
||||
(displayln (+scopes (syntax-local-introduce #'+)))
|
||||
(print-full-scopes)
|
||||
(syntax-local-introduce #'+))]))
|
||||
(quux -)
|
Loading…
Reference in New Issue
Block a user