-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathprotocols.lisp
657 lines (554 loc) · 23.1 KB
/
protocols.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
;;a simple implementation of clojure protocols, and deftype.
;;this will help with building libraries, particularly
;;the seq libraries.
;;If we can bolt on a few fundamental operations, we can take
;;advantage of the bulk of the excellent bootstrapped clojure
;;defined in the clojurescript compiler.
(defpackage :clclojure.protocols
(:use :common-lisp :common-utils :clclojure.reader :clclojure.pvector
:clclojure.eval)
(:export :defprotocol
:extend-protocol
:extend-type
:satisfies?
:protocol-exists?
:list-protocols
:clojure-deftype))
(in-package :clclojure.protocols)
;;aux
;;bootstrapping hack!
(defun vector? (x) (typep x 'clclojure.pvector::pvec))
(defun vector-expr (x)
(and (listp x) (eq (first x) 'persistent-vector)))
;;this keeps args in order....we nreverse all over the place.
;;Since we prototyped using lists, and now the vector
;;reader is working well, we're in the middle of migrating
;;to vectors. For now, we allow backwards compat with both
;;(perhaps allowing CL to define protocols in their native
;;tongue, I dunno). In the future, we'll enforce
;;vectors....
;;TBD replace with seq
(defun as-list (xs)
(if (vector? xs) (vector-to-list xs)
(if (vector-expr xs) (rest xs)
xs)))
;;changed this since we have lists now...
(defun drop-literals (xs)
(filter (lambda (x) (not (or (literal? x) (stringp x)))) (as-list xs)))
;;Note-> we need to add support for variadic functions,
;;and variadic protocols members.
;;protocols are used extensively, as is deftype. There are a
;;few additional data types that we need to provide.
;Protocol specifications take on the form below:
;(protocolname
; (function-name1 (args) &optional doc)
; function-name2 (args) &optional doc))
;; (defparameter samplespec
;; '(ISeq
;; (next (coll)
;; "Gets the next element from the sequence")
;; (more (coll)
;; "Gets the rest of the sequence.")))
(defun spec-name (protocolspec)
(car protocolspec))
(defun spec-functions (protocolspec)
(remove-if-not #'listp (as-list protocolspec)))
;;bombing out since we can't extend SEQUENCE to
;;our own types (thanks hyperspec!)
;;We can, however, enforce that one must use persistent
;;vectors....or....we can coerce the vectors to lists, which
;;are acceptable sequences....
(defun function-names (protocolspec)
(mapcar #'first (spec-functions protocolspec)))
(defun function-info (protocolspec)
(mapcar #'first (spec-functions protocolspec)))
;;TODO replace this with a more stringent version based
;;on protocol-function implementations rather than the spec.
;;This has arity information derived, so can provide a stronger
;;structural check.
(defun make-satisfier (protocolspec)
"From a list of function specs, builds a function that compares a
list of function specs to ensure both specifications have the same
function names. If no function specs are provided, the identity
function is returned."
(let ((names (function-names protocolspec)))
(if (null names)
(lambda (x)
(declare (ignore x))
t)
;;We need to harden this to check arities for the functions.
(lambda (newspec)
(null (set-difference names (function-names newspec)))))))
;a sample implementation for ISeqs...
(comment
(defparameter sampleimp
'(ISeq
(next (coll) (car coll))
(more (coll) (cdr coll)))))
;;From stack overflow. It looks like the compiler needs a hint if we're
;;defining struct/class literals and using them as constants.
(EVAL-WHEN (:compile-toplevel :load-toplevel :execute)
;;A protocol is a name, a set of generic functions, and a set of
;;types that implement the protocol.
(defstruct protocol name functions satisfier doc (members (list)))
(defmethod make-load-form ((v protocol) &optional env)
(declare (ignore env))
(make-load-form-saving-slots v))
;;Protocol function carries information about the arities
;;documentation, and other meta derived from the spec.
(defstruct protocol-function name args bodies arities doc)
(defmethod make-load-form ((v protocol-function) &optional env)
(declare (ignore env))
(make-load-form-saving-slots v)))
(defun ->protocol (name functions &optional satisfier doc)
(make-protocol :name name :functions functions :satisfier satisfier :doc (or doc "Not Documented")))
;;not used.
(defun protocol-to-spec (p)
(with-slots (name functions) p
(list name functions)))
(defun args->arities (xs)
(mapcar (lambda (args)
(multiple-value-bind (l variadic?)
(common-utils::args-type (as-list args))
(list :args args :arity l :variadic? (when variadic? t))))
xs))
(defun function->info (name args-doc)
(let* ((args (remove-if #'stringp args-doc))
(doc (first (remove-if-not #'stringp args-doc)))
(arities (args->arities args)))
(make-protocol-function :name name :args args
:bodies (length arities) :arities arities :doc (or doc "not documented"))))
;;We won't keep a central listing of protocols. They'll be first class objects,
;;as in Clojure on the JVM.
;;So, for bookeeping, it'd be nice to hold onto function specifications.
;;Simple things like
;;{:name blah
;; :arities #{1 2 3 :variadic}}
;Debating whether to keep this around,
;I may not need it...
(defparameter *protocols* (make-hash-table :test #'eq))
;Probably deprecated soon...
(defun get-protocol (name)
(gethash name *protocols*))
;;we can replace this using CLOS. We just add a generic function that
;;tells us if an object satisfies a protocol.
;;like (satisfies-protocol? (protocol obj))
;;Since protocols are actual objects (structs in this case), we call
;;(satisfies? the-protocol-obj the-obj)
;;which delegates to
;;((get-slot 'satisfier the-protocol-obj) the-obj)
;;So we let the protocol tell us if an object satisfies its protocol.
;;When we do defprotocol then, we add an implementation of
(defun add-protocol-member (pname membername)
"Identifies membername as an implementor of protocol
pname"
(multiple-value-bind (p exists?) (get-protocol pname)
(when exists?
(push membername (protocol-members p)))))
;Probably deprecated soon...
(defun protocol-exists? (name)
(not (null (get-protocol name))))
(defun drop-protocol (name)
"Eliminates any bindings to the quoted protocol name,
including generic functions."
(if (protocol-exists? name)
(let ((p (get-protocol name)))
(progn (unintern (protocol-name p))
(dolist (n (protocol-functions p))
(unintern n))
(remhash name *protocols*)))))
(defun list-protocols ()
"Lists all known protocols."
(loop for k being the hash-keys in *protocols* collect k))
;;we should cache this....
(defgeneric satisfies? (p x))
(defmethod satisfies? ((p protocol) x)
(not (null (find (type-of x) (protocol-members p)))))
(defmethod satisfies? ((p protocol) x)
(not (null (find (type-of x) (protocol-members p)))))
(define-condition protocol-exists (error)
((text :initarg :text :reader text)))
(define-condition malformed-protocol (error)
((text :initarg :text :reader text)))
(define-condition missing-implementations (error)
((text :initarg :text :reader text)))
(define-condition name-collision (error)
((text :initarg :text :reader text)))
;when we add protocols, we just want to ensure that the
;right generic functions are implemented.
;We let Common Lisp sort out whether the generic functions
;are actually correct.
(defun add-protocol (p)
(with-slots (name) p
(if (null (get-protocol name)) (setf (gethash name *protocols*) p)
; (error 'protocol-exists))))
(progn (print (format nil "Overwriting existing protocol ~A" name))
(drop-protocol `(quote ,name))
(setf (gethash name *protocols*) p)))))
;;We need to add the multiple-dispatch function that's in bootstrap at the moment.
;;A multiple-body protocol implementation could be...
(defparameter proto-spec
'(defprotocol ITough
(get-toughness [x] [x y]
"gets the toughness of x, or if x is compared to y, the relative toughness")))
;;If we want to use generic functions to mirror single-dispatch implementations of
;;protocols, we have to allow for multiple-body functions.
;;So, there's probably a protocol dispatch function..
;;Just like our fn macro...
;;If we have multiple specs, [x] [x y], in this case, we need a generic function
;;that dispatches based on the first arg of the spec.
;;Alternately....we can just use the fn body from before...
;;This only ever matters if there are multiple function bodies. If there's only one,
;;we're golden (that's the current situation).
;;note: dealing with reader-literals and how macros parse stuff, like pvectors,
;;so we're just filtering them out of arglists.
;; (defun build-generic (functionspec)
;; (let* ((args (drop-literals (second functionspec)))
;; (name (first functionspec))
;; (docs (if (= (length functionspec) 3)
;; (third functionspec)
;; "Not Documented")))
;; `(progn (defgeneric ,name ,args (:documentation ,docs))
;; ;;lets us use protocol fns as values...
;; (defparameter ,name (function ,name))
;; (setf (symbol-function (quote ,name)) (symbol-value (quote ,name)))
;; ,name)))
(defun build-generic (functionspec)
(let* ((args (drop-literals (rest functionspec)))
(name (first functionspec))
(doc (first (last functionspec)))
(docs (if (stringp doc)
doc
"Not Documented")))
(case (length args)
(1 (let ((args (drop-literals (first args))))
`(progn (defgeneric ,name ,args (:documentation ,docs))
;;lets us use protocol fns as values...
(defparameter ,name (function ,name))
(setf (symbol-function (quote ,name)) (symbol-value (quote ,name)))
,name)))
;;variadic
(otherwise
`(progn (defgeneric ,name (,'this ,'&rest ,'args) (:documentation ,docs))
;;lets us use protocol fns as values...
(defparameter ,name (function ,name))
(setf (symbol-function (quote ,name)) (symbol-value (quote ,name)))
,name)))))
(defun quoted-names (xs)
(mapcar (lambda (x) (list 'quote x))
(function-names xs)))
(defun derive-protocol-functions (xs)
(let* ((funcs (remove-if #'stringp xs)))
(mapcar (lambda (spec) (function->info (first spec) (rest spec))) funcs)))
(defun spec-to-protocol (protocolspec)
(let ((pfs (gensym "protofuncs"))
(doc (last (last protocolspec))))
`(let ((,pfs (,'list ,@(derive-protocol-functions (rest protocolspec)))))
(progn ,@(mapcar #'build-generic (spec-functions protocolspec))
(->protocol (quote ,(spec-name protocolspec))
,pfs
(make-satisfier (quote ,protocolspec))
,(if (stringp doc) doc "Not Documented"))))))
;;we'll have to update this guy later, but for now it's okay.
;;Added that a symbol gets created in the current package.
;; (defmacro defprotocol (name &rest functions)
;; (let ((p (gensym))
;; (spec (cons name functions)))
;; `(let ((,p (eval (spec-to-protocol (quote ,spec)))))
;; (progn (add-protocol ,p)
;; (defparameter ,name ,p)))))
(defmacro defprotocol (name &rest functions)
(let ((p (gensym))
(spec (cons name functions)))
`(let ((,p ,(spec-to-protocol spec)))
(progn (add-protocol ,p)
(defparameter ,name ,p)))))
(comment
(defprotocol IComplex
(f2 [x] [x y] [x y & zs]))
(defparameter test-impl
'(common-lisp:cons
(f2 [x] x)
(f2 [x y] (cons y x))
(f2 [z y & zs] (append (cons y z) zs))))
(defparameter parsed
'((CONS
(F2 [X] X)
(F2 [X Y] (CONS Y X))
(F2 [Z Y & ZS] (APPEND (CONS Y Z) ZS)))))
)
;;once we have implementations, grouped by type..
;;we need to parse the functions within them.
;;We expect something of the form:
;;(type-name &rest implementations)
;;where
;;implementations conform to
;;(function-name arg-vector & body)
;;We have multiple possible implementations
;;E.g.
;;(function-name arg-vector1 & body)
;;(function-name arg-vector2 & body2)
;;Order is not guaranteed either.
;;So, we need to group-by function-name.
;;(function-name (arg-vector1 body1)
;; (arg-vector2 body2)
;; (arg-vector3 body3))
;;May have to use seql instead of eql for symbol equality..
;;collect-functions will coerce function definitions with multiple
;;implementations in the above normal form into a
;;variadic form that's compatible with/expected by emit*
;;and leave single implementations alone.
;;So we'd get
;; (common-lisp:CONS
;; (F2 [X] X)
;; (F2 [X Y] (CONS Y X))
;; (F2 [Z Y & ZS] (APPEND (CONS Y Z) ZS)))
;;parsed into the method
;; (CONS
;; (F2 ([X] X)
;; ([X Y] (CONS Y X))
;; ([Z Y & ZS]
;; (APPEND (CONS Y Z) ZS))))
;;while single-arg implementations, or
;;functions with only one implementation,
;;are parsed as single-arity functions.
(defun collect-functions (type-fns)
(let ((fns (group-by #'first (rest type-fns) ;:test #'common-utils:seql
)))
(cons (first type-fns)
(mapcar (lambda (kv)
(case (length (second kv))
(1 (first (second kv))) ;;leave it alone.
(otherwise ;;generate a variadic implementation.
(cons (first kv)
(nreverse (mapcar #'cdr (second kv)))))))
(hash-table->entries fns)))))
;;extends protocol defined by name to
;;each type in the typespecs, where typespecs
;;are of the form...
;;(typename1 (func1 (x) (body))
;; (func2 (x) (body))
;; typename2 (func1 (x) (body))
;; (func2 (x) (body)))
(defun parse-implementations (x)
(labels ((get-spec (acc specs)
(if (null specs)
acc
(let ((arg (first specs)))
(if (symbolp arg)
(get-spec (cons (list arg) acc) (rest specs))
(let ((currentspec (first acc)))
(get-spec (cons (cons arg currentspec) (rest acc))
(rest specs))))))))
(mapcar (lambda (x) (collect-functions (nreverse x))) ;#'nreverse
(get-spec (list) x))))
;; (defparameter samplext
;; '(pvec
;; (next (coll) (nth coll 0))
;; (more (coll) (subvec coll))
;; cons
;; (next (coll) (first coll))
;; (more (coll) (rest coll))))
(defun implement-function (typename spec)
(let* ((args (cond ((vector? (second spec))
(vector-to-list (second spec)))
;this is a crappy hack.
((vector-expr (second spec))
(rest (second spec)))
(t
(drop-literals (second spec)))))
(newargs (cons (list (first args) typename) (rest args)))
(body (third spec)))
`(defmethod ,(first spec) ,newargs ,body)))
;;given something like this
;; '((f [this x] (list x))
;; (f [this x y] (list x y)))
;;emit a lambda* dispatch function like
;;(lambda* ((f (this x) (list x))
;; (f (this x y) (list x y)))
;; '((f [this x] (list x))
;; (f [this x y] (list x y)))
;;On the implementation side, we need to replace
;;& - which is permissible in clj, with &rest for
;;common lisp.
(defun replace-ampersand (xs)
(substitute '&rest '& xs))
(defun emit-dispatch (specs)
`(lambda* ,@(mapcar (lambda (spec)
(let* ((args (replace-ampersand (vector-to-list (first spec))))
(body (rest spec)))
`(,args ,@body)))
specs)))
;;multiple arity protocol function implementation.
(defun implement-function* (typename specs)
(let ((dispatch (gensym "dispatch"))
(name (first specs))
(fns (rest specs)))
`(let ((,dispatch ,(emit-dispatch fns)))
(defmethod ,name ((,'obj ,typename) ,'&rest ,'args)
(apply ,dispatch ,'obj ,'args)))))
(defmacro/literal-walker emit-method (protoname typename imp)
`(progn (add-protocol-member (quote ,protoname) (quote ,typename))
,@(mapcar (lambda (spec)
(if (listp (second spec))
(implement-function* typename spec)
(implement-function typename spec)))
(rest imp))))
;;this is the one choke point where we're getting
;;[x] -> (persistent-vector x) transforms in practice.
(defun emit-implementation (name satvar imp)
(let* ((quoted-imp (gensym "quotedimp"))
(msg (str `(,name ,imp))))
`(let ((,quoted-imp (quote ,imp)))
(if (funcall ,satvar ,quoted-imp)
(emit-method ,name ,(first imp) ,imp)
(error 'missing-implementations ,msg)))))
(defmacro/literal-walker extend-protocol (name &rest typespecs)
(let* ((imps (parse-implementations typespecs))
(satisfies? (gensym))
(emits (mapcar (lambda (imp) (emit-implementation name satisfies? imp))
imps)))
`(let ((,satisfies? (protocol-satisfier (get-protocol (quote ,name)))))
,@emits)))
;Extend-type is also particularly useful.
;;Pending -> implement deftype.
(comment
;Testing....
(defprotocol INamed
(get-name (thing) "gets the name of the thing!")
(say-name (thing) "Says the name of the thing!"))
(extend-protocol
INamed
cons (get-name (thing) (car thing))
(say-name (thing) (pprint (format nil "The name is: ~A" (get-name thing)))))
(defun test ()
(let ((data '(:tom)))
(when (satisfies? INamed data)
(pprint (get-name data)))
(say-name data))
(defmacro cljmacro (name argvec & body)
(let ((args (if (vector? argvec argvec)
(eval `(clclojure.reader/quoted-children ,argvec)))))
`(,@body)))
)
;;Deftype implementation.
;;Once we have protocols, deftype is pretty easy.
;;deftype is a hook into the type definition or object system of
;;the host environment. We'll use it to generate CLOS classes via
;;defclass. I may include an option to use deftype to build structs
;;which would likely kick ass for performance.
;;A deftype form is pretty easy:
;;(deftype name-of-type (field1 field2 ... fieldn)
;; Protocol1
;; (function1 (args) body1)
;; Protocol2
;; (function2 (args) body2))
;;
;;Should expand into:
;;(progn
;; (defclass name-of-type
;; ((field1 :init-arg :field1)
;; (field2 :init-arg :field2)))
;; (extend-protocol Protocol1 name-of-type
;; (function1 (args) body1))
;; (extend-protocol Protool2 name-of-type
;; (function2 (args) body2)))
)
(defun symbolize (x) (read-from-string x))
(defun emit-class-field (nm s)
`(,s :initarg ,(make-keyword s)
:accessor ,(symbolize (str nm "-" s))))
(defun emit-protocol-extension (proto-name type-name imps)
`(extend-protocol ,proto-name ,type-name ,@imps))
;;impl has protocol (pfn ...) (pfn ...)
(defmacro extend-type (typename &rest impls)
(let ((imps (parse-implementations impls))
;(name (gensym))
;(the-imp (gensym))
)
`(progn ,@(mapcar (lambda (the-imp)
(let ((expr `(emit-protocol-extension (quote ,(first the-imp))
(quote ,typename)
(quote ,(rest the-imp)))))
(eval expr)))
imps))
;; (if (funcall ,satisfies? imp)
;; (progn (add-protocol-member (quote ,name) typename)
;; (dolist (spec (rest imp))
;; (eval (implement-function ,typename spec))))
;; (error 'missing-implementation))
))
;;the goal here is to define "instance-local" operations
;;at the method level, where fields refer to slots on the object.
;;so, we may have a object like {:a 2 :b 3}
;;fields [a b],
;;our implementations could be
;;(blah [obj] a) ;;undefined!
;;(blee [a] a) ;;defined, shadowing, poor form, but meh.
;;we need to extend the lexcial environment to include
;;field access...
;;(blah [obj] a) =>
;;(blah [obj]
;; (with-slots ((a obj))
; a))
;;we can be more efficient if we walk the implementations
;;to detect field usage. For NOW, we'll just
;;bind all the fields in the lexical environment
;;of body, less the field names that are shadowed
;;by protocol args.
;;TODO: walk the body and collect fields to determine
;;the final set of fields to use (tailored).
(defun with-fields (fields method args &rest body)
(let* ((arglist (as-list args))
(var (first arglist))
(flds (set-difference (as-list fields) arglist)))
;;naive implementation is just bind all the slots....
(if flds
`(,method ,args
(with-slots ,flds ,var
,@body))
`(,method ,args ,@body))))
;;we need to mod this. If the implementations refer to a field (and
;;the field is NOT shadowed as an argument to their method impl), we
;;need a call to with-slots to pull the referenced fields out to
;;mirror clojure's behavior.
(defmacro clojure-deftype (name fields &rest implementations)
(let* ((flds (cond ((vector? fields)
(vector-to-list fields))
((vector-expr fields)
(rest fields))
(t
fields)))
(impls (mapcar (lambda (impl)
(if (atom impl) impl
(apply #'with-fields (cons flds impl)))) implementations)))
`(progn
(defclass ,name ()
,(mapcar (lambda (f) (emit-class-field name f) ) flds))
;;we need to parse the implementations to provide
;;instance-level fields...
(extend-type ,name ,@impls)
;;debugging
(defun ,(symbolize (str "->" name)) ,flds
(make-instance ,`(quote ,name) ,@(flatten (mapcar (lambda (f) `(,(make-keyword f) ,f)) flds ))))
)))
;;Deftype exists in common lisp.
(comment
;;An experimental class-bassed approach; putting this on ice for now.
;;This is our interface, which is a base class all protocols will
;;derive from.
;; (defclass IProtocol ()
;; (name
;; functions
;; satisfier
;; (members
;; :initform (list))))
;;Defining a protocol is just a matter of defining a new class that inherits
;;from IProtocol.
;; (defmacro defprotocol-1 (name functions &optional satisifer)
;; `(defclass ,name (IProtocol)
;; ((name :initform ,name)
;; (functions :initform functions)
;; (
;; )
)