-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathgeneric-sequences.lisp
110 lines (94 loc) · 3.91 KB
/
generic-sequences.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
;; Copyright (c) 2011-2012, Krzysztof Drewniak
;; All rights reserved.
;; Redistribution and use in source and binary forms, with or without
;; modification, are permitted provided that the following conditions are met:
;; * Redistributions of source code must retain the above copyright
;; notice, this list of conditions and the following disclaimer.
;; * Redistributions in binary form must reproduce the above copyright
;; notice, this list of conditions and the following disclaimer in the
;; documentation and/or other materials provided with the distribution.
;; * Neither the name of the <organization> nor the
;; names of its contributors may be used to endorse or promote products
;; derived from this software without specific prior written permission.
;; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
;; ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
;; WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
;; DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
;; DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
;; (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
;; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
;; ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
;; (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
;; SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
(in-package #:dlist)
#+generic-sequences
(progn
(defmethod sequence:length ((x dlist))
(dlist-length x))
(defmethod sequence:elt ((x dlist) n)
(dlist-nth n x))
(defmethod (setf sequence:elt) (value (seq dlist) i)
(setf (dlist-nth i seq) value))
(defmethod sequence:adjust-sequence
((seq dlist) length &key initial-element (initial-contents nil ic-sp))
(let ((delta (- (dlist-length seq) length)))
(loop repeat delta do (dlist-pop seq :from-end t))
(cond
((= length 0) nil)
(ic-sp
(loop
repeat length
for i = initial-contents then (cdr i)
for j = (dlist-first seq) then (next j)
while i do
(if j (setf (data j) (car i)) (dlist-push (car i) seq :at-end t))))
(t
(loop repeat delta do
(dlist-push initial-element seq :at-end t))))
seq))
(defmethod sequence:make-sequence-like ((seq dlist) length &key initial-element initial-contents)
(cond
((= length 0) nil)
(initial-contents (apply #'dlist (coerce initial-contents 'list)))
(t (make-dlist length :initial-element initial-element))))
(defmethod sequence:reverse ((seq dlist))
(dlist-reverse seq))
(defmethod sequence:make-sequence-iterator ((seq dlist) &key from-end (start 0) end)
(let ((begin-val
(if from-end
(if end
(nthdcons (1- (- (dlist-length seq) end)) seq :from-end t)
(dlist-last seq))
(if start
(nthdcons start seq)
(dlist-first seq)))))
(values
begin-val ;;iterator
(if from-end ;;limit
(if (> start 0)
(nthdcons (1- start) seq)
nil)
(if end
(nthdcons end seq)
nil))
from-end ;;from-end
#'sequence:iterator-step
#'sequence:iterator-endp
#'sequence:iterator-element
#'(setf sequence:iterator-element)
#'sequence:iterator-index
#'sequence:iterator-copy)))
(defmethod sequence:iterator-step ((seq dlist) iter from-end)
(funcall (if from-end #'prev #'next) iter))
(defmethod sequence:iterator-endp ((seq dlist) iter lim (from-end t))
(eql iter lim))
(defmethod sequence:iterator-element ((seq dlist) iter)
(data iter))
(defmethod (setf sequence:iterator-element) (val (seq dlist) iter)
(setf (data iter) val))
(defmethod sequence:iterator-index ((seq dlist) iter)
(let ((ret 0))
(dodcons (i seq ret)
(when (eql iter i) (return ret))
(incf ret))))
);; end progn