-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmacro-utils.sls
99 lines (88 loc) · 3.41 KB
/
macro-utils.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
90
91
92
93
94
95
96
97
98
99
#!r6rs
;; Copyright 2009 Derick Eddington. My MIT-style license is in the file named
;; LICENSE from the original collection this file is distributed with.
(library (xitomatl macro-utils)
(export
gen-temp syntax->list with-syntax*
duplicate-id unique-ids? unique-ids?/raise formals-ok?/raise
identifier-append name=? identifier?/name=?
free-identifier-bound?)
(import
(rnrs)
(only (xitomatl predicates) name=?)
(xitomatl macro-utils fib))
(define (gen-temp)
(with-syntax (((t) (generate-temporaries '(1))))
#'t))
(define (syntax->list ls)
(syntax-case ls ()
((ls ...) #'(ls ...))
(_ (assertion-violation 'syntax->list "not a syntax list" ls))))
(define-syntax with-syntax*
(syntax-rules ()
((_ (pc0 pc1 pc* ...) b b* ...)
(with-syntax (pc0)
(with-syntax* (pc1 pc* ...) b b* ...)))
((_ pc b b* ...)
(with-syntax pc b b* ...))))
(define (duplicate-id ids)
(unless (and (list? ids) (for-all identifier? ids))
(assertion-violation 'duplicate-id "not a list of identifiers" ids))
(let recur ((ls ids))
(and (pair? ls)
(let ((id (car ls)) (rest (cdr ls)))
(if (memp (lambda (x) (bound-identifier=? x id)) rest)
id
(recur (cdr ls)))))))
(define (unique-ids? ls)
(not (duplicate-id ls)))
(define unique-ids?/raise
(case-lambda
((ids orig-stx msg)
(let ((dup (duplicate-id ids)))
(if dup
(syntax-violation #F msg orig-stx dup)
#T)))
((ids orig-stx)
(unique-ids?/raise ids orig-stx "duplicate identifier"))))
(define (formals-ok?/raise frmls-stx orig-stx)
(syntax-case frmls-stx ()
((arg* ... . rest)
(and (or (null? (syntax->datum #'rest))
(identifier? #'rest)
(syntax-violation #F "not an identifier" orig-stx #'rest))
(for-all (lambda (id)
(or (identifier? id)
(syntax-violation #F "not an identifier" orig-stx id)))
#'(arg* ...))
(unique-ids?/raise
(append
#'(arg* ...)
(if (identifier? #'rest) (list #'rest) '()))
orig-stx)))))
(define (identifier-append ctxt . ids)
(define who 'identifier-append)
(unless (identifier? ctxt) (assertion-violation who "not an identifier" ctxt))
(let ((rs
(apply string-append
(map
(lambda (id)
(cond ((identifier? id) (symbol->string (syntax->datum id)))
((symbol? id) (symbol->string id))
((string? id) id)
(else (assertion-violation who
"not an identifier, symbol, or string" id))))
ids))))
(unless (positive? (string-length rs))
(assertion-violation who "result length zero" rs))
(datum->syntax ctxt (string->symbol rs))))
(define (identifier?/name=? id name)
(and (identifier? id)
(name=? id name)))
#;(define (syntax-object->source-location-expression stx)
(let ((file-name (host:source-location-file-name stx))
(char-pos (host:source-location-char-pos stx)))
(assert (or (not file-name) (string? file-name)))
(assert (or (not char-pos) (number? char-pos)))
#`(make-source-location #,file-name #,char-pos)))
)