-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeclarations.lisp
145 lines (137 loc) · 5.47 KB
/
declarations.lisp
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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
(cl:in-package #:ecclesia)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; Take a list of declaration specifiers and return a list of
;;; canonicalized declaration specifiers.
;;;
;;; A canonicalized declaration specifier is one of following:
;;;
;;; * (declaration name)
;;; * (dynamic-extent var)
;;; * (dynamic-extent (function fn))
;;; * (ftype type function-name)
;;; * (ignore var)
;;; * (ignore (function fn))
;;; * (ignorable var)
;;; * (ignorable (function fn))
;;; * (inline function-name)
;;; * (notinline function-name)
;;; * (optimize (quality value))
;;; * (special var)
;;; * (type typespec var)
;;;
;;; In other words, the following work is done here:
;;;
;;; * Each declaration specifier is divided up so that each one
;;; concerns a single entity (name, variable, function name,
;;; quality).
;;;
;;; * Abbreviations for TYPE declaration specifiers are expanded so
;;; that they contain the word TYPE.
;;;
;;; * A declaration specifier with an empty list of entities is
;;; removed.
;;;
;;; We do NOT check whether the type in a TYPE declaration specifier
;;; is a valid type.
;;;
;;; FIXME:
;;; At the moment, we assume that a declaration specifier where the
;;; CAR is not one of the symbols DECLARATION, DYNAMIC-EXTENT, FTYPE,
;;; IGNORE, IGNORABLE, INLINE, NOTINLINE, OPTIMIZE, SPECIAL, or TYPE
;;; is an abbreviated type specifier. This is not quite correct,
;;; because it could be a symbol introduced by DECLARATION.
;;; FIXME: use specific conditions
(defun check-declaration-specifier (declaration-specifier)
(unless (proper-list-p declaration-specifier)
(error "declaration specifier must be a proper list"))
(when (null declaration-specifier)
(error "declaration specifier can not be nil"))
(when (eq (car declaration-specifier) 'type)
(unless (>= (length declaration-specifier) 2)
(error "TYPE declaration specifier must have a type specifier"))))
;;; Take a single declaration specifier and return a list of
;;; canonicalized declaration specifiers.
(defun canonicalize-declaration-specifier (declaration-specifier
alien-identifiers)
(cond ((member (car declaration-specifier)
'(declaration dynamic-extent ignore ignorable
inline notinline special))
(loop for entity in (cdr declaration-specifier)
collect `(,(car declaration-specifier) ,entity)))
((eq (car declaration-specifier) 'optimize)
(loop for entity in (cdr declaration-specifier)
collect `(optimize ,entity)))
((member (car declaration-specifier) '(type ftype))
(loop for entity in (cddr declaration-specifier)
collect `(,(car declaration-specifier)
,(cadr declaration-specifier) ,entity)))
((member (car declaration-specifier) alien-identifiers)
;; This means that the declaration is one specified by
;; (declaim (declaration ...))
;; It has some user- or implementation- specified meaning
;; that we don't care about, so we ignore it.
nil)
(t
(loop for entity in (cdr declaration-specifier)
collect `(type ,(car declaration-specifier) ,entity)))))
(defun canonicalize-declaration-specifiers (declaration-specifiers
alien-identifiers)
(unless (proper-list-p declaration-specifiers)
(error "declaration specifiers must be a proper list"))
(reduce #'append
(mapcar (lambda (spec)
(canonicalize-declaration-specifier
spec alien-identifiers))
declaration-specifiers)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Separate an ordinary body such as a let or let* body that may
;;; contain declarations (but no documentation) into the declarations
;;; and the executable forms.
;;;
;;; If there are declarations after the first executable form (which
;;; is a syntax error), then those declarations will be considered
;;; part of the executable forms.
(defun separate-ordinary-body (body)
(unless (proper-list-p body)
(error 'ordinary-body-must-be-proper-list
:body body))
(let ((pos (position-if-not (lambda (item)
(and (consp item)
(eq (car item) 'declare)))
body)))
(if (null pos)
(values body '())
(values (subseq body 0 pos) (subseq body pos)))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Separate a body such as a defun, flet, or lables that may contain
;;; both declarations and a documentation string into the
;;; declarations, the documentation, and the executable forms.
;;;
;;; Return three values. The first value is a list of declarations.
;;; Each element of the list is a complete declaration, including the
;;; symbol DECLARE. The second value is a the documentation as a
;;; string, or NIL if no documentation was found. The last value is a
;;; list of forms.
(defun separate-function-body (body)
(unless (proper-list-p body)
(error 'function-body-must-be-proper-list
:body body))
(let ((declarations '())
(documentation nil)
(forms '()))
(loop for (expr . rest) on body
do (cond ((not (null forms))
(push expr forms))
((and (consp expr) (eq (car expr) 'declare))
(push expr declarations))
((stringp expr)
(if (or (null rest)
(not (null documentation))
(not (null forms)))
(push expr forms)
(setf documentation expr)))
(t
(push expr forms))))
(values (nreverse declarations) documentation (nreverse forms))))