-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlambda-list-parsing.lisp
1506 lines (1424 loc) · 54.4 KB
/
lambda-list-parsing.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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
(cl:in-package #:ecclesia)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Lambda list utilities.
;;; There is much to say about lambda lists.
;;;
;;; There are 10 different types of lambda lists and they vary both
;;; with respect to syntax and semantics. It gets pretty messy in
;;; fact.
;;;
;;; Lambda lists admit something known as "lambda list keywords".
;;; They are not keywords in the normal sense of the word (i.e.,
;;; symbols in the :KEYWORD package), but just ordinary symbols in
;;; the COMMON-LISP package that happen to have names that start with
;;; the `&' character (ampersand).
;;;
;;; The lambda list keywords that are allowed in each different type
;;; of lambda list are clearly indicated in the CLHS for that type of
;;; lambda list. However, the CLHS also allows for
;;; implementation-specific lambda list keywords. The complete list
;;; of lambda list keywords that a particular implementation
;;; recognizes is available as the value of the variable
;;; LAMBDA-LIST-KEYWORDS. However, there is no way to determine what
;;; an implementation-specific lambda list keyword means, nor how it
;;; is used or even in which type of lambda list it is allowed. There
;;; is also no indication as to whether implementation-specific lambda
;;; list keywords must begin with `&'.
;;;
;;; The lambda list keywords that are recognized by the CLHS are:
;;; &allow-other-keys, &aux, &body, &environment, &key, &optional,
;;; &rest, and &whole. The lambda list keywords &body and &rest are
;;; synonymous, but good style gives preference to one rather than the
;;; other according to the type of lambda list it occurs in.
;;;
;;; To make things more complicated, the CLHS does not tell us how to
;;; handle occurrences of a particular lambda list keyword in a lambda
;;; list of a type that does not recognize it. One interpretation
;;; could be that such a lambda list keyword should be treated as just
;;; any symbol, so that it becomes the name of a parameter. But with
;;; this interpretation, a program can have some subtle bugs just
;;; because a programmer incorrectly believes that a particular
;;; lambda-list keyword is acceptable in a type of lambda list where
;;; in fact it is not. According to another interpretation, a
;;; program-error should be signaled in this case. At the very least,
;;; it seems reasonable to give a style-warning in that case.
;;;
;;; Similarly, the CLHS does not indicate how to handle occurrences of
;;; symbols that do not occur in LAMBDA-LIST-KEYWORDS, but that happen
;;; to start with `&'. Again, some subtle bugs could result if such a
;;; situation were not to be flagged to the programmer. Again, at the
;;; very least, a style warning seems to be appropriate.
;;;
;;; Lambda list keywords have different arities, i.e., the number of
;;; items that can follow it in a lambda list. The good news is that
;;; each lambda list keyword has the same arity no matter what type of
;;; lambda list it occurs in. Thus &allow-other-keys always has arity
;;; 0 (zero), &rest, &body, &whole, and &environment always have arity
;;; 1 (one), and the remaining ones (&aux, &key, and &optional) can
;;; take any number of items, so have arbitrary arity.
;;;
;;; Another piece of relatively good news is that the order in which
;;; lambda list keywords can occur in a lambda list is independent of
;;; the type of lambda list in which they occur, and that the relative
;;; order between two lambda list keywords is fixed, with &environment
;;; being the only exception, because it can occur anywhere (except
;;; before &whole) in the lambda lists in which it is allowed.
;;;
;;; A piece of not-so-good news is that &whole, whenever it is
;;; allowed, must appear first in the lambda list. That is, not only
;;; first as in the first lambda list keyword, but as the first item
;;; in the lambda list, before the list of required variables. This
;;; rule messes up syntax checking a bit.
;;; A list of lambda list keywords in the order that they can occur in
;;; a lambda list (except &environment, which can occur anywhere, and
;;; except &rest and &body which do not have any relative order
;;; because they cannot both occur). For each keyword, we indicate
;;; its min and max arity, where NIL means unbounded.
(defparameter *lambda-list-keywords*
`((&whole 1 1)
(&environment 1 nil)
(&optional 0 nil)
(&rest 1 1)
(&body 1 1)
(&key 0 nil)
(&allow-other-keys 0 0)
(&aux 0 nil)))
(defun potential-lambda-list-keyword-p (object)
(and (symbolp object)
(plusp (length (symbol-name object)))
(eql (char (symbol-name object) 0) #\&)))
;;; Use this function for lambda lists that can be proper or dotted.
(defun check-lambda-list-not-circular (lambda-list)
(when (eq (nth-value 1 (list-structure lambda-list)) :circular)
(error 'lambda-list-must-not-be-circular
:code lambda-list)))
;;; Use this function for lambda lists that must be proper lists.
(defun check-lambda-list-proper (lambda-list)
(unless (eq (nth-value 1 (list-structure lambda-list)) :proper)
(error 'lambda-list-must-be-proper-list
:code lambda-list)))
;;; Check for restrictions common to all lambda lists.
;;;
;;; Before calling this function, individual parsers must check the
;;; structure of the lambda list, in particular that it is a list, and
;;; that it is not circular. This function can deal with proper and
;;; dotted lists.
;;;
;;; We do the following checks:
;;;
;;; * check for lambda list keywords not allowed
;;;
;;; * check and warn if a symbol starting with & apperas, but
;;; it is not a recognized lambda list keyword.
;;;
;;; * check that each keyword appears with the correct arity.
;;;
;;; * check that each keyword appears at most once.
;;;
;;; * check that the keywords appear in the right order.
;;;
;;; * check that if &whole appears, it appears first. This is safe,
;;; because we have first checked whether &whole is allowed at all,
;;; so there is no risk that we will give an error message about
;;; &whole for a lambda list that does not allow it.
;;;
;;; We do NOT do any of the following checks:
;;;
;;; * We do not check the restrictions on keywords that must be
;;; respected for dotted lists, because it would look funny if such
;;; an error were reported for a lambda list that is not allowed to
;;; be dotted in the first place.
;;;
;;; * We do not check the nature of the arguments to the lambda list
;;; keywords. The parser for each type of lambda list must do that.
(defun check-lambda-list-keywords (lambda-list keywords)
;; We assume that KEYWORDS is a subset of LAMBDA-LIST-KEYWORDS, in
;; other words that we are given only valid lambda list keywords as
;; defined by the system.
(let* (;; All symbols in the lambda list that look like they might
;; be lambda-list keywords, in the order that the occur in
;; the lambda list. Multiple occurrences are preserved.
(potential (loop for remaining = lambda-list then (cdr remaining)
while (consp remaining)
when (potential-lambda-list-keyword-p (car remaining))
collect (car remaining)))
;; All symbols in the lambda list that are also lambda-list
;; keywords as defined by the system, in the order that they
;; occur in the lambda list.
(real (remove-if-not (lambda (x) (member x lambda-list-keywords))
potential))
;; All symbols in the lambda list that look like they might
;; be lambda-list keywords, but that are not lambda list
;; keywords defined by the system, in any old order.
(suspect (set-difference potential lambda-list-keywords))
;; All symbols in the lambda list that are also lambda-list
;; keywords as defined by the system, but that are not in the
;; list of lambda list keywords allowed for this type of
;; lambda list, in any old order.
(forbidden (set-difference real keywords))
;; All symbols in the lambda list that are also in the list
;; of valid keywords for this lambda list, in the order that
;; they appear in the lambda list. Multiple occurrences are
;; preserved.
(to-process (remove-if-not (lambda (x) (member x keywords))
potential)))
;; Check for forbidden keywords.
(unless (null forbidden)
(error 'lambda-list-keyword-not-allowed
:code lambda-list
:keyword (car forbidden)))
;; Check for suspect keywords.
(unless (null suspect)
(warn 'suspect-lambda-list-keyword
:code lambda-list
:keyword (car suspect)))
;; Check for multiple occurrences.
(loop for keyword in to-process
do (when (> (count keyword to-process) 1)
(error 'multiple-occurrences-of-lambda-list-keyword
:code lambda-list
:keyword keyword)))
(when (> (+ (count '&body to-process) (count '&rest to-process)) 1)
(error 'both-rest-and-body-occur-in-lambda-list
:code lambda-list))
;; Check the order of keywords.
(loop for rem = to-process then (cdr rem)
until (null (cdr rem))
do (when (and (not (eq (car rem) '&environment))
(not (eq (cadr rem) '&environment))
(> (position (car rem) *lambda-list-keywords* :key #'car)
(position (cadr rem) *lambda-list-keywords* :key #'car)))
(error 'incorrect-keyword-order
:code lambda-list
:keyword1 (car rem)
:keyword2 (cadr rem))))
;; Check arities.
(flet ((check-arity (keyword number-of-args)
(if (eq keyword '&whole)
(when (zerop number-of-args)
(error 'whole-must-be-followed-by-variable
:code lambda-list))
(let ((arities (cdr (assoc keyword *lambda-list-keywords*))))
(when (or (< number-of-args (car arities))
(and (not (null (cadr arities)))
(> number-of-args (cadr arities))))
(error "wrong arity for ~s" keyword))))))
(loop with positions = (mapcar (lambda (x) (position x lambda-list))
to-process)
for keyword in to-process
for (pos next-pos) on (append positions
(list (list-structure lambda-list)))
do (check-arity keyword (- next-pos pos 1))))
;; Check that if &whole is present, it appears first.
(when (and (member '&whole to-process)
(not (eq (car lambda-list) '&whole)))
(error 'whole-must-appear-first
:code lambda-list))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; A pattern is either:
;;;
;;; * a tree (a symbol or a CONS cell), or
;;; * an instance of the class LAMBDA-LIST.
;;;
;;; An &optional entry (after canonicalization) is one of:
;;;
;;; * (pattern init-form)
;;; * (pattern init-form supplied-p-parameter)
;;;
;;; A &key entry (after canonicalization) is one of:
;;;
;;; * ((keyword pattern) init-form)
;;; * ((keyword pattern) init-form supplied-p-parameter)
;;;
;;; An &aux entry (after canonicalization) is of the form:
;;;
;;; * (var init-form)
(defgeneric required (parsed-lambda-list))
(defgeneric (setf required) (required parsed-lambda-list))
(defgeneric environment (parsed-lambda-list))
(defgeneric (setf environment) (environment parsed-lambda-list))
(defgeneric whole (parsed-lambda-list))
(defgeneric (setf whole) (whole parsed-lambda-list))
(defgeneric optionals (parsed-lambda-list))
(defgeneric (setf optionals) (optionals parsed-lambda-list))
(defgeneric rest-body (parsed-lambda-list))
(defgeneric (setf rest-body) (rest-body parsed-lambda-list))
(defgeneric keys (parsed-lambda-list))
(defgeneric (setf keys) (keys parsed-lambda-list))
(defgeneric allow-other-keys (parsed-lambda-list))
(defgeneric (setf allow-other-keys) (allow-other-keys parsed-lambda-list))
(defgeneric aux (parsed-lambda-list))
(defgeneric (setf aux) (aux parsed-lambda-list))
(defclass lambda-list ()
(;; A possibly empty list of patterns.
(%required :initform '() :initarg :required :accessor required)
;; Either:
;; * :none, meaning &environment was not given, or
;; * a single variable, represented as a symbol.
(%environment :initform :none :initarg :environment :accessor environment)
;; Either:
;; * :none, meaning &whole was not given, or
;; * a single variable, represented as a symbol.
(%whole :initform :none :initarg :whole :accessor whole)
;; Either:
;; * :none, meaning &optional was not given at all,
;; * a possibly empty list of &optional entries.
(%optionals :initform :none :initarg :optionals :accessor optionals)
;; Either:
;; * :none, meaning &rest or &body was not given at all, or
;; * a single pattern.
(%rest-body :initform :none :initarg :rest-body :accessor rest-body)
;; Either:
;; * :none, meaning &key was not given at all,
;; * a possibly empty list of &key entries.
(%keys :initform :none :initarg :keys :accessor keys)
;; Either:
;; * nil, meaning &allow-other-keys was not given at all,
;; * t, meaning &allow-other-keys was given.
(%allow-other-keys :initform nil
:initarg :allow-other-keys
:accessor allow-other-keys)
;; Either:
;; * :none, meaning &aux was not given at all,
;; * a possibly empty list of &aux entries.
(%aux :initform '() :initarg :aux :accessor aux)))
(defun list-has-keyword-p (list)
(loop for rest = list then (cdr rest)
while (consp rest)
when (member (car rest) *lambda-list-keywords* :key #'car)
return t))
;;; We only check that the tree doesn't have any illegal atoms in it.
;;; At this point, we do not check for multiple occurrences of
;;; variables.
(defun check-tree (tree)
(labels ((check-aux (subtree)
(cond ((or (null subtree)
(and (symbolp subtree)
(not (constantp subtree))))
nil)
((consp subtree)
(check-aux (car subtree))
(check-aux (cdr subtree)))
(t
(error 'malformed-destructuring-tree
:code tree)))))
(check-aux tree)))
(defun parse-pattern (tree-or-lambda-list)
(cond ((and (symbolp tree-or-lambda-list)
(not (constantp tree-or-lambda-list)))
tree-or-lambda-list)
((consp tree-or-lambda-list)
(cond ((list-has-keyword-p tree-or-lambda-list)
(parse-destructuring-lambda-list tree-or-lambda-list))
(t
(check-tree tree-or-lambda-list)
tree-or-lambda-list)))
(t
(error 'malformed-lambda-list-pattern :code nil))))
(defun parse-ordinary-required (required)
(unless (and (symbolp required)
(not (constantp required)))
(error 'required-must-be-variable
:code required))
required)
(defun parse-destructuring-required (required)
;; A required argument can be any pattern.
(parse-pattern required))
;;; Parse a specialized required parameter.
;;; We canonicalize it, so that instead of having the original
;;; 3 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * (var specializer)
;;;
;;; we boil it down to just 1:
;;;
;;; * (var specializer)
;;;
;;; by replacing var or (var) by (var t)
(defun parse-specialized-required (required)
(if (consp required)
(progn
(unless (and (symbolp (car required))
(not (constantp (car required)))
(or (null (cdr required))
(and (null (cddr required))
(or (symbolp (cadr required))
(and (consp (cadr required))
(consp (cdadr required))
(null (cddadr required))
(eq (caadr required) 'eql))))))
(error 'malformed-specialized-required
:code required))
`(,(car required) ,(if (null (cdr required)) t (cadr required))))
(progn
(unless (and (symbolp required)
(not (constantp required)))
(error 'malformed-specialized-required
:code required))
`(,required t))))
(defun parse-all-required (lambda-list start end item-parser)
(loop for i from start below end
for rest = (nthcdr start lambda-list) then (cdr rest)
for required = (car rest)
collect (funcall item-parser required)))
;;; Parse an ordinary &optional item.
;;; We canonicalize it a bit, so that instead of having the original
;;; 4 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * (var init-form)
;;; * (var init-form supplied-p-parameter)
;;;
;;; we boil it down to 2:
;;;
;;; * (var init-form)
;;; * (var init-form supplied-p-parameter)
;;;
;;; by replacing var or (var) by (var nil)
(defun parse-ordinary-optional (optional)
(if (consp optional)
(multiple-value-bind (length structure)
(list-structure optional)
(unless (and (eq structure :proper)
(<= 1 length 3)
(symbolp (car optional))
(not (constantp (car optional)))
(or (< length 3)
(symbolp (caddr optional))
(not (constantp (caddr optional)))))
(error 'malformed-ordinary-optional
:code optional))
`(,(car optional)
,(if (> length 1) (cadr optional) nil)
. ,(cddr optional)))
(progn
(unless (and (symbolp optional)
(not (constantp optional)))
(error 'malformed-ordinary-optional
:code optional))
`(,optional nil))))
;;; Parse a defgeneric &optional item.
;;; We canonicalize it, so that instead of having the original
;;; 2 different possible forms:
;;;
;;; * var
;;; * (var)
;;;
;;; we boil it down to just 1:
;;;
;;; * var
;;;
;;; by replacing (var) by var.
(defun parse-defgeneric-optional (optional)
(if (consp optional)
(progn
(unless (and (null (cdr optional))
(symbolp (car optional))
(not (constantp (car optional))))
(error 'malformed-defgeneric-optional
:code optional))
(car optional))
(progn
(unless (and (symbolp optional)
(not (constantp optional)))
(error 'malformed-defgeneric-optional
:code optional))
optional)))
;;; Parse a destructuring &optional item.
;;; We canonicalize it a bit, so that instead of having the original
;;; 4 different possible forms:
;;;
;;; * var
;;; * (pattern)
;;; * (pattern init-form)
;;; * (pattern init-form supplied-p-parameter)
;;;
;;; we boil it down to 2:
;;;
;;; * (pattern init-form)
;;; * (pattern init-form supplied-p-parameter)
;;;
;;; by replacing var by (var <default>) and (pattern) by (pattern nil).
(defun parse-destructuring/deftype-optional (optional default)
(if (consp optional)
(multiple-value-bind (length structure)
(list-structure optional)
(unless (and (eq structure :proper)
(<= 1 length 3)
(or (< length 3)
(symbolp (caddr optional))
(not (constantp (caddr optional)))))
(error 'malformed-destructuring-optional
:code optional))
`(,(car optional)
,(if (> length 1) (cadr optional) `',default)
. ,(cddr optional)))
(progn
(unless (and (symbolp optional)
(not (constantp optional)))
(error 'malformed-destructuring-optional
:code optional))
`(,optional ',default))))
(defun parse-destructuring-optional (optional)
(parse-destructuring/deftype-optional optional nil))
(defun parse-deftype-optional (optional)
(parse-destructuring/deftype-optional optional '*))
(defun parse-all-optionals
(lambda-list positions item-parser)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &optional.
(eq (elt lambda-list (car positions)) '&optional))
(values (loop for i from (1+ (car positions)) below (cadr positions)
for optional in (nthcdr (1+ (car positions)) lambda-list)
collect (funcall item-parser optional))
(cdr positions)))
(t
(values :none positions))))
;;; Parse an ordinary &key item.
;;; We canonicalize it a bit, so that instead of having the original
;;; 7 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * (var init-form)
;;; * (var init-form supplied-p-parameter)
;;; * ((keyword var))
;;; * ((keyword var) init-form)
;;; * ((keyword var) init-form supplied-p-parameter)
;;;
;;; we boil it down to 2:
;;;
;;; * ((keyword var) init-form)
;;; * ((keyword var) init-form supplied-p-parameter)
;;;
;;; by replacing var or (var) by ((:var var) nil),
;;; by replacing (var init-form) by ((:var var) init-form), and
;;; by replacing (var init-form supplied-p-parameter) by
;;; ((:var var) init-form supplied-p-parameter).
(defun parse-ordinary-key (key)
(if (consp key)
(multiple-value-bind (length structure)
(list-structure key)
(unless (and (eq structure :proper)
(<= 1 length 3)
(or (and (symbolp (car key))
(not (constantp (car key))))
(and (consp (car key))
(symbolp (caar key))
(consp (cdar key))
(symbolp (cadar key))
(not (constantp (cadar key)))
(null (cddar key))))
(or (< length 3)
(symbolp (caddr key))
(not (constantp (caddr key)))))
(error 'malformed-ordinary-key
:code key))
`(,(if (symbolp (car key))
`(,(intern (symbol-name (car key)) :keyword) ,(car key))
(car key))
,(if (> length 1) (cadr key) nil)
. ,(cddr key)))
(progn
(unless (and (symbolp key)
(not (constantp key)))
(error 'malformed-ordinary-key
:code key))
`((,(intern (symbol-name key) :keyword) ,key) nil))))
;;; Parse a defgeneric &key item.
;;; We canonicalize it, so that instead of having the original
;;; 3 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * ((keyword var))
;;;
;;; we boil it down to just 1:
;;;
;;; * ((keyword var))
;;;
;;; by replacing var and (var) by ((:var var))
(defun parse-defgeneric-key (key)
(if (consp key)
(progn
(unless (and (null (cdr key))
(or (and (symbolp (car key))
(not (constantp (car key))))
(and (consp (car key))
(symbolp (caar key))
(consp (cdar key))
(symbolp (cadar key))
(not (constantp (cadar key)))
(null (cddar key)))))
(error 'malformed-defgeneric-key
:code key))
`(,(if (symbolp (car key))
`(,(intern (symbol-name (car key)) :keyword) ,(car key))
(car key))))
(progn
(unless (and (symbolp key)
(not (constantp key)))
(error 'malformed-defgeneric-key
:code key))
`(,(intern (symbol-name key) :keyword) ,key))))
;;; Parse a destructuring &key item.
;;; We canonicalize it a bit, so that instead of having the original
;;; 7 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * (var init-form)
;;; * (var init-form supplied-p-parameter)
;;; * ((keyword pattern))
;;; * ((keyword pattern) init-form)
;;; * ((keyword pattern) init-form supplied-p-parameter)
;;;
;;; we boil it down to 2:
;;;
;;; * ((keyword pattern) init-form)
;;; * ((keyword pattern) init-form supplied-p-parameter)
;;;
;;; by replacing var or (var) by ((:var var) <default>),
;;; by replacing (var init-form) by ((:var var) init-form), and
;;; by replacing (var init-form supplied-p-parameter) by
;;; ((:var var) init-form supplied-p-parameter).
(defun parse-destructuring/deftype-key (key default)
(if (consp key)
(multiple-value-bind (length structure)
(list-structure key)
(unless (and (eq structure :proper)
(<= 1 length 3)
(or (and (symbolp (car key))
(not (constantp (car key))))
(and (consp (car key))
(symbolp (caar key))
(consp (cdar key))
(null (cddar key))))
(or (< length 3)
(symbolp (caddr key))
(not (constantp (caddr key)))))
(error 'malformed-ordinary-key
:code key))
`(,(if (symbolp (car key))
`(,(intern (symbol-name (car key)) :keyword) ,(car key))
`(,(caar key) ,(parse-pattern (cadar key))))
,(if (> length 1) (cadr key) default)
. ,(cddr key)))
(progn
(unless (and (symbolp key)
(not (constantp key)))
(error 'malformed-ordinary-key
:code key))
`((,(intern (symbol-name key) :keyword) ,key) ,default))))
(defun parse-destructuring-key (key)
(parse-destructuring/deftype-key key nil))
(defun parse-deftype-key (key)
(parse-destructuring/deftype-key key '*))
(defun parse-all-keys
(lambda-list positions item-parser)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &key.
(eq (elt lambda-list (car positions)) '&key))
(values (loop for i from (1+ (car positions)) below (cadr positions)
for key in (nthcdr (1+ (car positions)) lambda-list)
collect (funcall item-parser key))
(cdr positions)))
(t
(values :none positions))))
;;; Parse an &aux item.
;;; We canonicalize it, so that instead of having the original
;;; 3 different possible forms:
;;;
;;; * var
;;; * (var)
;;; * (var intitform)
;;;
;;; we boil it down to just 1:
;;;
;;; * (var initform)
;;;
;;; by replacing var and (var) by (var nil)
(defun parse-aux (aux)
(if (consp aux)
(progn
(unless (and (symbolp (car aux))
(not (constantp (car aux)))
(or (null (cdr aux))
(null (cddr aux))))
(error 'malformed-aux
:code aux))
`(,(car aux) ,(if (null (cdr aux)) nil (cadr aux))))
(progn
(unless (and (symbolp aux)
(not (constantp aux)))
(error 'malformed-aux
:code aux))
`(,aux nil))))
(defun parse-all-aux (lambda-list positions)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &aux.
(eq (elt lambda-list (car positions)) '&aux))
(values (loop for i from (1+ (car positions)) below (cadr positions)
for aux in (nthcdr (1+ (car positions)) lambda-list)
collect (parse-aux aux))
(cdr positions)))
(t
(values :none positions))))
(defun parse-allow-other-keys (lambda-list positions)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &allow-other-keys.
(eq (elt lambda-list (car positions)) '&allow-other-keys))
(values t (cdr positions)))
(t
(values nil positions))))
(defun parse-environment (lambda-list positions)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &environment.
(eq (elt lambda-list (car positions)) '&environment))
;; The arity has already been checked so we know there is
;; something after it, but we don't know what.
(let ((arg (elt lambda-list (1+ (car positions)))))
(unless (and (symbolp arg)
(not (constantp arg)))
(error 'environment-must-be-followed-by-variable
:code lambda-list))
(values arg (cdr positions))))
(t
(values :none positions))))
(defun parse-rest/body (lambda-list positions)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &rest or &body.
(or (eq (elt lambda-list (car positions)) '&rest)
(eq (elt lambda-list (car positions)) '&body)))
;; The arity has already been checked so we know there is
;; something after it, but we don't know what.
(let ((arg (elt lambda-list (1+ (car positions)))))
(unless (and (symbolp arg)
(not (constantp arg)))
(error 'rest/body-must-be-followed-by-variable
:code lambda-list))
(values arg (cdr positions))))
(t
(values :none positions))))
(defun parse-whole (lambda-list positions)
(cond ((and
;; there is a keyword yet to be processed.
(not (null (cdr positions)))
;; that keyword is &whole
(eq (elt lambda-list (car positions)) '&whole))
;; The arity has already been checked so we know there is
;; something after it, but we don't know what.
(let ((arg (elt lambda-list (1+ (car positions)))))
(unless (and (symbolp arg)
(not (constantp arg)))
(error 'whole-must-be-followed-by-variable
:code lambda-list))
(values arg (cdr positions))))
(t
(values :none positions))))
;;; Compute the position of each of the allowed keywords
;;; that appears in the lambda list, and add the length
;;; of the lambda list (i.e., the number of CONS cells it has)
;;; at the end of the computed list.
(defun compute-keyword-positions (lambda-list allowed)
(loop for rest = lambda-list then (cdr rest)
for i from 0
unless (consp rest) collect i
while (consp rest)
when (member (car rest) allowed)
collect i))
(defun parse-ordinary-lambda-list (lambda-list)
(let ((allowed '(&optional &rest &key &allow-other-keys &aux)))
(check-lambda-list-proper lambda-list)
(check-lambda-list-keywords lambda-list allowed)
(let ((positions (compute-keyword-positions lambda-list allowed))
(result (make-instance 'lambda-list)))
(setf (required result)
(parse-all-required
lambda-list 0 (car positions) #'parse-ordinary-required))
(setf (values (optionals result) positions)
(parse-all-optionals
lambda-list positions #'parse-ordinary-optional))
(setf (values (rest-body result) positions)
(parse-rest/body lambda-list positions))
(setf (values (keys result) positions)
(parse-all-keys
lambda-list positions #'parse-ordinary-key))
(setf (values (allow-other-keys result) positions)
(parse-allow-other-keys lambda-list positions))
(setf (values (aux result) positions)
(parse-all-aux lambda-list positions))
;; We should have run out of parameters now.
(unless (null (cdr positions))
(error 'lambda-list-too-many-parameters :parameters (cdr positions)))
result)))
(defun parse-generic-function-lambda-list (lambda-list)
(let ((allowed '(&optional &rest &key &allow-other-keys)))
(check-lambda-list-proper lambda-list)
(check-lambda-list-keywords lambda-list allowed)
(let ((positions (compute-keyword-positions lambda-list allowed))
(result (make-instance 'lambda-list)))
(setf (required result)
(parse-all-required
lambda-list 0 (car positions) #'parse-ordinary-required))
(setf (values (optionals result) positions)
(parse-all-optionals
lambda-list positions #'parse-defgeneric-optional))
(setf (values (rest-body result) positions)
(parse-rest/body lambda-list positions))
(setf (values (keys result) positions)
(parse-all-keys
lambda-list positions #'parse-defgeneric-key))
(setf (values (allow-other-keys result) positions)
(parse-allow-other-keys lambda-list positions))
;; We should have run out of parameters now.
(unless (null (cdr positions))
(error 'lambda-list-too-many-parameters :parameters (cdr positions)))
result)))
(defun parse-specialized-lambda-list (lambda-list)
(let ((allowed '(&optional &rest &key &allow-other-keys &aux)))
(check-lambda-list-proper lambda-list)
(check-lambda-list-keywords lambda-list allowed)
(let ((positions (compute-keyword-positions lambda-list allowed))
(result (make-instance 'lambda-list)))
(setf (required result)
(parse-all-required
lambda-list 0 (car positions) #'parse-specialized-required))
(setf (values (optionals result) positions)
(parse-all-optionals
lambda-list positions #'parse-ordinary-optional))
(setf (values (rest-body result) positions)
(parse-rest/body lambda-list positions))
(setf (values (keys result) positions)
(parse-all-keys
lambda-list positions #'parse-ordinary-key))
(setf (values (allow-other-keys result) positions)
(parse-allow-other-keys lambda-list positions))
(setf (values (aux result) positions)
(parse-all-aux lambda-list positions))
;; We should have run out of parameters now.
(unless (null (cdr positions))
(error 'lambda-list-too-many-parameters :parameters (cdr positions)))
result)))
(defun parse-macro-lambda-list (lambda-list)
(multiple-value-bind (length structure) (list-structure lambda-list)
(when (eq structure :circular)
(error 'lambda-list-must-not-be-circular
:code lambda-list))
(if (eq structure :dotted)
(progn
(when (zerop length)
(error 'lambda-list-must-be-list
:code lambda-list))
(let ((allowed '(&whole &environment &optional)))
(check-lambda-list-keywords lambda-list allowed)
(let ((positions (compute-keyword-positions lambda-list allowed))
(result (make-instance 'lambda-list)))
(if (eq (car lambda-list) '&whole)
(progn
(setf (values (whole result) positions)
(parse-whole lambda-list positions))
(if (eq (caddr lambda-list) '&environment)
(progn
(setf (values (environment result) positions)
(parse-environment lambda-list positions))
(setf (required result)
(parse-all-required lambda-list
4 (car positions)
#'parse-pattern)))
(setf (required result)
(parse-all-required lambda-list
2 (car positions)
#'parse-pattern))))
(if (eq (car lambda-list) '&environment)
(progn
(setf (values (environment result) positions)
(parse-environment lambda-list positions))
(setf (required result)
(parse-all-required lambda-list
2 (car positions)
#'parse-pattern)))
(setf (required result)
(parse-all-required lambda-list
0 (car positions)
#'parse-pattern))))
;; The environment may follow the required.
(when (eq (environment result) :none)
(setf (values (environment result) positions)
(parse-environment lambda-list positions)))
(setf (values (optionals result) positions)
(parse-all-optionals
lambda-list positions #'parse-destructuring-optional))
;; The environment may follow the optionals.
(when (eq (environment result) :none)
(setf (values (environment result) positions)
(parse-environment lambda-list positions)))
;; We should have run out of parameters now.
(unless (null (cdr positions))
(error 'lambda-list-too-many-parameters :parameters (cdr positions)))
;; All that remains is to deal with the dotted end
;; of the list.
(let ((rest (cdr (last lambda-list))))
(unless (and (symbolp rest)
(not (constantp rest)))
(error 'atomic-lambda-list-tail-must-be-variable
:code lambda-list))
(setf (rest-body result) rest))
result)))
(progn
(let ((allowed '(&whole &environment &optional &rest &body
&key &allow-other-keys &aux)))
(check-lambda-list-keywords lambda-list allowed)
(let ((positions (compute-keyword-positions lambda-list allowed))
(result (make-instance 'lambda-list)))
(if (eq (car lambda-list) '&whole)
(progn
(setf (values (whole result) positions)
(parse-whole lambda-list positions))
(if (eq (caddr lambda-list) '&environment)
(progn
(setf (values (environment result) positions)
(parse-environment lambda-list positions))
(setf (required result)
(parse-all-required lambda-list
4 (car positions)
#'parse-pattern)))
(setf (required result)
(parse-all-required lambda-list
2 (car positions)
#'parse-pattern))))
(if (eq (car lambda-list) '&environment)
(progn
(setf (values (environment result) positions)
(parse-environment lambda-list positions))
(setf (required result)
(parse-all-required lambda-list
2 (car positions)
#'parse-pattern)))
(setf (required result)
(parse-all-required lambda-list
0 (car positions)
#'parse-pattern))))
;; The environment may follow the required.
(when (eq (environment result) :none)
(setf (values (environment result) positions)
(parse-environment lambda-list positions)))
(setf (values (optionals result) positions)
(parse-all-optionals
lambda-list positions #'parse-destructuring-optional))
;; The environment may follow the optionals.
(when (eq (environment result) :none)
(setf (values (environment result) positions)
(parse-environment lambda-list positions)))
(setf (values (rest-body result) positions)
(parse-rest/body lambda-list positions))
;; The environment may follow the rest/body.
(when (eq (environment result) :none)
(setf (values (environment result) positions)
(parse-environment lambda-list positions)))
(setf (values (keys result) positions)
(parse-all-keys
lambda-list positions #'parse-destructuring-key))
(setf (values (allow-other-keys result) positions)
(parse-allow-other-keys lambda-list positions))
;; The environment may follow the keys.
(when (eq (environment result) :none)
(setf (values (environment result) positions)