forked from ijp/ijputils
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsyntax.sls
89 lines (78 loc) · 2.23 KB
/
syntax.sls
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
#!r6rs
(library (ijputils syntax)
;; These should probably be exported at levels 0 and 1 as we most
;; likely want to use them in a syntax-case macro
;; TODO: find out a way to write `syntax?'
(export syntax->list
syntax->dotted-list
syntax-e
gensym
make-id
with-syntax*
)
(import (rnrs base)
(rnrs syntax-case)
(only (srfi :1) dotted-list? cons*)
(ijputils symbols)
)
(define (map-if predicate mapper list) ;from (familiars lists)
(map (lambda (x)
(if (predicate x)
(mapper x)
x))
list))
(define (syntax->list stxobj)
(define (inner stx)
(syntax-case stx ()
[() '()]
[(x . rest)
(cons #'x (inner #'rest))]))
(assert (list? (syntax->datum stxobj)))
(inner stxobj))
(define (syntax->dotted-list dlist)
(define (inner stx)
(syntax-case stx ()
[(x . rest)
(cons #'x (inner #'rest))]
[x #'x]))
(assert (dotted-list? (syntax->datum dlist)))
(inner dlist))
(define (syntax-e obj)
(syntax-case obj ()
[(x ...)
(apply list #'(x ...))]
[#(value ...)
(apply vector #'(value ...))]
[a (syntax->dotted-list #'a)]))
;; Other ideas,
;; with-syntax*
;; like with-syntax but nested
;; syntax-map
;; like (syntax-map f l) => (map f (syntax->list l))
(define (syntax-map proc l)
(assert (list? (syntax->datum l)))
(let loop ((stx-list l) (return-list '()))
(syntax-case stx-list ()
[() (reverse return-list)]
[(first . rest)
(loop #'rest (cons (proc #'first) return-list))])))
;; block
;; allows interleaving of definitions and expressions
;; syntax-parameters
(define (gensym)
(syntax->datum (car (generate-temporaries (list 'g)))))
;; may come in handy
(define (make-id stx . syms&ids)
(datum->syntax stx
(apply symbol-append
(map-if identifier? syntax->datum syms&ids))))
(define-syntax with-syntax*
(syntax-rules ()
[(with-syntax* () body rest ...)
(with-syntax () body rest ...)]
[(with-syntax* (one) body rest ...)
(with-syntax (one) body rest ...)]
[(with-syntax* (car . cdr) body rest ...)
(with-syntax (car)
(with-syntax* cdr body rest ...))]))
)