1
2
3 container duplex-list:_elem [
4 value:_elem
5 next:&:duplex-list:_elem
6 prev:&:duplex-list:_elem
7 ]
8
9
10 def push x:_elem, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
11 local-scope
12 load-ingredients
13 result:&:duplex-list:_elem <- new {(duplex-list _elem): type}
14 *result <- merge x, in, 0
15 {
16 ¦ break-unless in
17 ¦ *in <- put *in, prev:offset, result
18 }
19 return result
20 ]
21
22 def first in:&:duplex-list:_elem -> result:_elem [
23 local-scope
24 load-ingredients
25 return-unless in, 0
26 result <- get *in, value:offset
27 ]
28
29 def next in:&:duplex-list:_elem -> result:&:duplex-list:_elem/contained-in:in [
30 local-scope
31 load-ingredients
32 return-unless in, 0
33 result <- get *in, next:offset
34 ]
35
36 def prev in:&:duplex-list:_elem -> result:&:duplex-list:_elem/contained-in:in [
37 local-scope
38 load-ingredients
39 return-unless in, 0
40 result <- get *in, prev:offset
41 return result
42 ]
43
44 scenario duplex-list-handling [
45 run [
46 ¦ local-scope
47 ¦
48 ¦ 10:num/raw <- copy 34
49 ¦ 11:num/raw <- copy 35
50 ¦ list:&:duplex-list:char <- push 3, 0
51 ¦ list <- push 4, list
52 ¦ list <- push 5, list
53 ¦ list2:&:duplex-list:char <- copy list
54 ¦ 20:char/raw <- first list2
55 ¦ list2 <- next list2
56 ¦ 21:char/raw <- first list2
57 ¦ list2 <- next list2
58 ¦ 22:char/raw <- first list2
59 ¦ 30:&:duplex-list:char/raw <- next list2
60 ¦ 31:char/raw <- first 30:&:duplex-list:char/raw
61 ¦ 32:&:duplex-list:char/raw <- next 30:&:duplex-list:char/raw
62 ¦ 33:&:duplex-list:char/raw <- prev 30:&:duplex-list:char/raw
63 ¦ list2 <- prev list2
64 ¦ 40:char/raw <- first list2
65 ¦ list2 <- prev list2
66 ¦ 41:char/raw <- first list2
67 ¦ 50:bool/raw <- equal list, list2
68 ]
69 memory-should-contain [
70 ¦ 0 <- 0
71 ¦ 10 <- 34
72 ¦ 11 <- 35
73 ¦ 20 <- 5
74 ¦ 21 <- 4
75 ¦ 22 <- 3
76 ¦ 30 <- 0
77 ¦ 31 <- 0
78 ¦ 32 <- 0
79 ¦ 33 <- 0
80 ¦ 40 <- 4
81 ¦ 41 <- 5
82 ¦ 50 <- 1
83 ]
84 ]
85
86 def length l:&:duplex-list:_elem -> result:num [
87 local-scope
88 load-ingredients
89 result <- copy 0
90 {
91 ¦ break-unless l
92 ¦ result <- add result, 1
93 ¦ l <- next l
94 ¦ loop
95 }
96 ]
97
98
99 def insert x:_elem, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
100 local-scope
101 load-ingredients
102 new-node:&:duplex-list:_elem <- new {(duplex-list _elem): type}
103 *new-node <- put *new-node, value:offset, x
104
105 next-node:&:duplex-list:_elem <- get *in, next:offset
106 *in <- put *in, next:offset, new-node
107 *new-node <- put *new-node, prev:offset, in
108 *new-node <- put *new-node, next:offset, next-node
109 return-unless next-node
110 *next-node <- put *next-node, prev:offset, new-node
111 ]
112
113 scenario inserting-into-duplex-list [
114 local-scope
115 list:&:duplex-list:char <- push 3, 0
116 list <- push 4, list
117 list <- push 5, list
118 run [
119 ¦ list2:&:duplex-list:char <- next list
120 ¦ list2 <- insert 6, list2
121 ¦
122 ¦ list2 <- copy list
123 ¦ 10:char/raw <- first list2
124 ¦ list2 <- next list2
125 ¦ 11:char/raw <- first list2
126 ¦ list2 <- next list2
127 ¦ 12:char/raw <- first list2
128 ¦ list2 <- next list2
129 ¦ 13:char/raw <- first list2
130 ¦ list2 <- prev list2
131 ¦ 20:char/raw <- first list2
132 ¦ list2 <- prev list2
133 ¦ 21:char/raw <- first list2
134 ¦ list2 <- prev list2
135 ¦ 22:char/raw <- first list2
136 ¦ 30:bool/raw <- equal list, list2
137 ]
138 memory-should-contain [
139 ¦ 10 <- 5
140 ¦ 11 <- 4
141 ¦ 12 <- 6
142 ¦ 13 <- 3
143 ¦ 20 <- 6
144 ¦ 21 <- 4
145 ¦ 22 <- 5
146 ¦ 30 <- 1
147 ]
148 ]
149
150 scenario inserting-at-end-of-duplex-list [
151 local-scope
152 list:&:duplex-list:char <- push 3, 0
153 list <- push 4, list
154 list <- push 5, list
155 run [
156 ¦ list2:&:duplex-list:char <- next list
157 ¦ list2 <- next list2
158 ¦ list2 <- insert 6, list2
159 ¦
160 ¦ list2 <- copy list
161 ¦ 10:char/raw <- first list2
162 ¦ list2 <- next list2
163 ¦ 11:char/raw <- first list2
164 ¦ list2 <- next list2
165 ¦ 12:char/raw <- first list2
166 ¦ list2 <- next list2
167 ¦ 13:char/raw <- first list2
168 ¦ list2 <- prev list2
169 ¦ 20:char/raw <- first list2
170 ¦ list2 <- prev list2
171 ¦ 21:char/raw <- first list2
172 ¦ list2 <- prev list2
173 ¦ 22:char/raw <- first list2
174 ¦ 30:bool/raw <- equal list, list2
175 ]
176 memory-should-contain [
177 ¦ 10 <- 5
178 ¦ 11 <- 4
179 ¦ 12 <- 3
180 ¦ 13 <- 6
181 ¦ 20 <- 3
182 ¦ 21 <- 4
183 ¦ 22 <- 5
184 ¦ 30 <- 1
185 ]
186 ]
187
188 scenario inserting-after-start-of-duplex-list [
189 local-scope
190 list:&:duplex-list:char <- push 3, 0
191 list <- push 4, list
192 list <- push 5, list
193 run [
194 ¦ list <- insert 6, list
195 ¦
196 ¦ list2:&:duplex-list:char <- copy list
197 ¦ 10:char/raw <- first list2
198 ¦ list2 <- next list2
199 ¦ 11:char/raw <- first list2
200 ¦ list2 <- next list2
201 ¦ 12:char/raw <- first list2
202 ¦ list2 <- next list2
203 ¦ 13:char/raw <- first list2
204 ¦ list2 <- prev list2
205 ¦ 20:char/raw <- first list2
206 ¦ list2 <- prev list2
207 ¦ 21:char/raw <- first list2
208 ¦ list2 <- prev list2
209 ¦ 22:char/raw <- first list2
210 ¦ 30:bool/raw <- equal list, list2
211 ]
212 memory-should-contain [
213 ¦ 10 <- 5
214 ¦ 11 <- 6
215 ¦ 12 <- 4
216 ¦ 13 <- 3
217 ¦ 20 <- 4
218 ¦ 21 <- 6
219 ¦ 22 <- 5
220 ¦ 30 <- 1
221 ]
222 ]
223
224
225
226
227
228 def remove x:&:duplex-list:_elem/contained-in:in, in:&:duplex-list:_elem -> in:&:duplex-list:_elem [
229 local-scope
230 load-ingredients
231
232 return-unless x
233 next-node:&:duplex-list:_elem <- get *x, next:offset
234 prev-node:&:duplex-list:_elem <- get *x, prev:offset
235
236 *x <- put *x, next:offset, 0
237 *x <- put *x, prev:offset, 0
238
239 {
240 ¦ break-unless next-node
241 ¦ *next-node <- put *next-node, prev:offset, prev-node
242 }
243
244 {
245 ¦ break-unless prev-node
246 ¦ *prev-node <- put *prev-node, next:offset, next-node
247 ¦ return
248 }
249
250
251 return next-node
252 ]
253
254 scenario removing-from-duplex-list [
255 local-scope
256 list:&:duplex-list:char <- push 3, 0
257 list <- push 4, list
258 list <- push 5, list
259 run [
260 ¦ list2:&:duplex-list:char <- next list
261 ¦ list <- remove list2, list
262 ¦ 10:bool/raw <- equal list2, 0
263 ¦
264 ¦ list2 <- copy list
265 ¦ 11:char/raw <- first list2
266 ¦ list2 <- next list2
267 ¦ 12:char/raw <- first list2
268 ¦ 20:&:duplex-list:char/raw <- next list2
269 ¦ list2 <- prev list2
270 ¦ 30:char/raw <- first list2
271 ¦ 40:bool/raw <- equal list, list2
272 ]
273 memory-should-contain [
274 ¦ 10 <- 0
275 ¦ 11 <- 5
276 ¦ 12 <- 3
277 ¦ 20 <- 0
278 ¦ 30 <- 5
279 ¦ 40 <- 1
280 ]
281 ]
282
283 scenario removing-from-start-of-duplex-list [
284 local-scope
285 list:&:duplex-list:char <- push 3, 0
286 list <- push 4, list
287 list <- push 5, list
288 run [
289 ¦ list <- remove list, list
290 ¦
291 ¦ list2:&:duplex-list:char <- copy list
292 ¦ 10:char/raw <- first list2
293 ¦ list2 <- next list2
294 ¦ 11:char/raw <- first list2
295 ¦ 20:&:duplex-list:char/raw <- next list2
296 ¦ list2 <- prev list2
297 ¦ 30:char/raw <- first list2
298 ¦ 40:bool/raw <- equal list, list2
299 ]
300 memory-should-contain [
301 ¦ 10 <- 4
302 ¦ 11 <- 3
303 ¦ 20 <- 0
304 ¦ 30 <- 4
305 ¦ 40 <- 1
306 ]
307 ]
308
309 scenario removing-from-end-of-duplex-list [
310 local-scope
311 list:&:duplex-list:char <- push 3, 0
312 list <- push 4, list
313 list <- push 5, list
314 run [
315 ¦
316 ¦ list2:&:duplex-list:char <- next list
317 ¦ list2 <- next list2
318 ¦ list <- remove list2, list
319 ¦ 10:bool/raw <- equal list2, 0
320 ¦
321 ¦ list2 <- copy list
322 ¦ 11:char/raw <- first list2
323 ¦ list2 <- next list2
324 ¦ 12:char/raw <- first list2
325 ¦ 20:&:duplex-list:char/raw <- next list2
326 ¦ list2 <- prev list2
327 ¦ 30:char/raw <- first list2
328 ¦ 40:bool/raw <- equal list, list2
329 ]
330 memory-should-contain [
331 ¦ 10 <- 0
332 ¦ 11 <- 5
333 ¦ 12 <- 4
334 ¦ 20 <- 0
335 ¦ 30 <- 5
336 ¦ 40 <- 1
337 ]
338 ]
339
340 scenario removing-from-singleton-duplex-list [
341 local-scope
342 list:&:duplex-list:char <- push 3, 0
343 run [
344 ¦ list <- remove list, list
345 ¦ 1:num/raw <- copy list
346 ]
347 memory-should-contain [
348 ¦ 1 <- 0
349 ]
350 ]
351
352
353
354
355
356
357 def remove-between start:&:duplex-list:_elem, end:&:duplex-list:_elem/contained-in:start -> start:&:duplex-list:_elem [
358 local-scope
359 load-ingredients
360 next:&:duplex-list:_elem <- get *start, next:offset
361 nothing-to-delete?:bool <- equal next, end
362 return-if nothing-to-delete?
363 assert next, [malformed duplex list]
364
365
366 *next <- put *next, prev:offset, 0
367 *start <- put *start, next:offset, end
368 return-unless end
369
370
371 prev:&:duplex-list:_elem <- get *end, prev:offset
372 assert prev, [malformed duplex list - 2]
373 *prev <- put *prev, next:offset, 0
374 *end <- put *end, prev:offset, start
375 ]
376
377 scenario remove-range [
378
379 local-scope
380 list:&:duplex-list:char <- push 18, 0
381 list <- push 17, list
382 list <- push 16, list
383 list <- push 15, list
384 list <- push 14, list
385 list <- push 13, list
386 run [
387 ¦
388 ¦
389 ¦ list2:&:duplex-list:char <- next list
390 ¦ list2 <- next list2
391 ¦ list2 <- remove-between list2, 0
392 ¦
393 ¦ 10:char/raw <- get *list, value:offset
394 ¦ list <- next list
395 ¦ 11:char/raw <- get *list, value:offset
396 ¦ list <- next list
397 ¦ 12:char/raw <- get *list, value:offset
398 ¦ 20:&:duplex-list:char/raw <- next list
399 ]
400 memory-should-contain [
401 ¦ 10 <- 13
402 ¦ 11 <- 14
403 ¦ 12 <- 15
404 ¦ 20 <- 0
405 ]
406 ]
407
408 scenario remove-range-to-final [
409 local-scope
410
411 list:&:duplex-list:char <- push 18, 0
412 list <- push 17, list
413 list <- push 16, list
414 list <- push 15, list
415 list <- push 14, list
416 list <- push 13, list
417 run [
418 ¦
419 ¦
420 ¦ list2:&:duplex-list:char <- next list
421 ¦
422 ¦ end:&:duplex-list:char <- next list2
423 ¦ end <- next end
424 ¦ end <- next end
425 ¦ end <- next end
426 ¦ remove-between list2, end
427 ¦
428 ¦ 10:char/raw <- get *list, value:offset
429 ¦ list <- next list
430 ¦ 11:char/raw <- get *list, value:offset
431 ¦ list <- next list
432 ¦ 12:char/raw <- get *list, value:offset
433 ¦ 20:&:duplex-list:char/raw <- next list
434 ]
435 memory-should-contain [
436 ¦ 10 <- 13
437 ¦ 11 <- 14
438 ¦ 12 <- 18
439 ¦ 20 <- 0
440 ]
441 ]
442
443 scenario remove-range-empty [
444 local-scope
445
446 list:&:duplex-list:char <- push 15, 0
447 list <- push 14, list
448 list <- push 13, list
449 run [
450 ¦
451 ¦ list2:&:duplex-list:char <- next list
452 ¦ remove-between list, list2
453 ¦
454 ¦ 10:char/raw <- get *list, value:offset
455 ¦ list <- next list
456 ¦ 11:char/raw <- get *list, value:offset
457 ¦ list <- next list
458 ¦ 12:char/raw <- get *list, value:offset
459 ¦ 20:&:duplex-list:char/raw <- next list
460 ]
461
462 memory-should-contain [
463 ¦ 10 <- 13
464 ¦ 11 <- 14
465 ¦ 12 <- 15
466 ¦ 20 <- 0
467 ]
468 ]
469
470 scenario remove-range-to-end [
471 local-scope
472
473 list:&:duplex-list:char <- push 18, 0
474 list <- push 17, list
475 list <- push 16, list
476 list <- push 15, list
477 list <- push 14, list
478 list <- push 13, list
479 run [
480 ¦
481 ¦ list2:&:duplex-list:char <- next list
482 ¦ remove-between list2, 0
483 ¦
484 ¦ 10:char/raw <- get *list, value:offset
485 ¦ list <- next list
486 ¦ 11:char/raw <- get *list, value:offset
487 ¦ 20:&:duplex-list:char/raw <- next list
488 ]
489 memory-should-contain [
490 ¦ 10 <- 13
491 ¦ 11 <- 14
492 ¦ 20 <- 0
493 ]
494 ]
495
496
497 def splice in:&:duplex-list:_elem, start:&:duplex-list:_elem/contained-in:in -> in:&:duplex-list:_elem [
498 local-scope
499 load-ingredients
500 return-unless in
501 return-unless start
502 end:&:duplex-list:_elem <- last start
503 next:&:duplex-list:_elem <- next in
504 {
505 ¦ break-unless next
506 ¦ *end <- put *end, next:offset, next
507 ¦ *next <- put *next, prev:offset, end
508 }
509 *in <- put *in, next:offset, start
510 *start <- put *start, prev:offset, in
511 ]
512
513
514 def insert in:&:duplex-list:_elem, new:&:@:_elem -> in:&:duplex-list:_elem [
515 local-scope
516 load-ingredients
517 return-unless in
518 return-unless new
519 len:num <- length *new
520 return-unless len
521 curr:&:duplex-list:_elem <- copy in
522 idx:num <- copy 0
523 {
524 ¦ done?:bool <- greater-or-equal idx, len
525 ¦ break-if done?
526 ¦ c:_elem <- index *new, idx
527 ¦ insert c, curr
528 ¦
529 ¦ curr <- next curr
530 ¦ idx <- add idx, 1
531 ¦ loop
532 }
533 ]
534
535 def append in:&:duplex-list:_elem, new:&:duplex-list:_elem/contained-in:in -> in:&:duplex-list:_elem [
536 local-scope
537 load-ingredients
538 last:&:duplex-list:_elem <- last in
539 *last <- put *last, next:offset, new
540 return-unless new
541 *new <- put *new, prev:offset, last
542 ]
543
544 def last in:&:duplex-list:_elem -> result:&:duplex-list:_elem [
545 local-scope
546 load-ingredients
547 result <- copy in
548 {
549 ¦ next:&:duplex-list:_elem <- next result
550 ¦ break-unless next
551 ¦ result <- copy next
552 ¦ loop
553 }
554 ]
555
556
557 def dump-from x:&:duplex-list:_elem [
558 local-scope
559 load-ingredients
560 $print x, [: ]
561 {
562 ¦ break-unless x
563 ¦ c:_elem <- get *x, value:offset
564 ¦ $print c, [ ]
565 ¦ x <- next x
566 ¦ {
567 ¦ ¦ is-newline?:bool <- equal c, 10/newline
568 ¦ ¦ break-unless is-newline?
569 ¦ ¦ $print 10/newline
570 ¦ ¦ $print x, [: ]
571 ¦ }
572 ¦ loop
573 }
574 $print 10/newline, [---], 10/newline
575 ]
576
577 scenario stash-duplex-list [
578 local-scope
579 list:&:duplex-list:num <- push 1, 0
580 list <- push 2, list
581 list <- push 3, list
582 run [
583 ¦ stash [list:], list
584 ]
585 trace-should-contain [
586 ¦ app: list: 3 <-> 2 <-> 1
587 ]
588 ]
589
590 def to-text in:&:duplex-list:_elem -> result:text [
591 local-scope
592 load-ingredients
593 buf:&:buffer:char <- new-buffer 80
594 buf <- to-buffer in, buf
595 result <- buffer-to-array buf
596 ]
597
598
599 def to-text-line in:&:duplex-list:_elem -> result:text [
600 local-scope
601 load-ingredients
602 buf:&:buffer:char <- new-buffer 80
603 buf <- to-buffer in, buf, 6
604 result <- buffer-to-array buf
605 ]
606
607 def to-buffer in:&:duplex-list:_elem, buf:&:buffer:char -> buf:&:buffer:char [
608 local-scope
609 load-ingredients
610 {
611 ¦ break-if in
612 ¦ buf <- append buf, [[]]
613 ¦ return
614 }
615
616 val:_elem <- get *in, value:offset
617 buf <- append buf, val
618
619 next:&:duplex-list:_elem <- next in
620 nextn:num <- copy next
621 return-unless next
622 buf <- append buf, [ <-> ]
623
624 remaining:num, optional-ingredient-found?:bool <- next-ingredient
625 {
626 ¦ break-if optional-ingredient-found?
627 ¦
628 ¦ buf <- to-buffer next, buf
629 ¦ return
630 }
631 {
632 ¦ break-unless remaining
633 ¦
634 ¦ remaining <- subtract remaining, 1
635 ¦ buf <- to-buffer next, buf, remaining
636 ¦ return
637 }
638
639 append buf, [...]
640 ]
641
642 scenario stash-empty-duplex-list [
643 local-scope
644 x:&:duplex-list:num <- copy 0
645 run [
646 ¦ stash x
647 ]
648 trace-should-contain [
649 ¦ app: []
650 ]
651 ]