2020-05-25 20:48:33 +00:00
|
|
|
(*
|
2019-03-11 08:59:55 +00:00
|
|
|
BSD 2-Clause License
|
|
|
|
|
2020-10-13 07:58:51 +00:00
|
|
|
Copyright (c) 2018-2020, Anton Krotov
|
2019-03-11 08:59:55 +00:00
|
|
|
All rights reserved.
|
|
|
|
*)
|
|
|
|
|
|
|
|
MODULE LISTS;
|
|
|
|
|
|
|
|
IMPORT C := COLLECTIONS;
|
|
|
|
|
|
|
|
|
|
|
|
TYPE
|
|
|
|
|
|
|
|
ITEM* = POINTER TO RECORD (C.ITEM)
|
|
|
|
|
|
|
|
prev*, next*: ITEM
|
|
|
|
|
|
|
|
END;
|
|
|
|
|
|
|
|
LIST* = POINTER TO RECORD
|
|
|
|
|
|
|
|
first*, last*: ITEM
|
|
|
|
|
|
|
|
END;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE push* (list: LIST; item: ITEM);
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
ASSERT(item # NIL);
|
|
|
|
|
|
|
|
IF list.first = NIL THEN
|
|
|
|
list.first := item;
|
2020-10-13 07:58:51 +00:00
|
|
|
item.prev := NIL
|
2019-03-11 08:59:55 +00:00
|
|
|
ELSE
|
|
|
|
ASSERT(list.last # NIL);
|
|
|
|
item.prev := list.last;
|
2020-10-13 07:58:51 +00:00
|
|
|
list.last.next := item
|
|
|
|
END;
|
|
|
|
list.last := item;
|
|
|
|
item.next := NIL
|
2019-03-11 08:59:55 +00:00
|
|
|
END push;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE pop* (list: LIST): ITEM;
|
|
|
|
VAR
|
|
|
|
last: ITEM;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
|
|
|
|
last := list.last;
|
|
|
|
|
|
|
|
IF last # NIL THEN
|
|
|
|
IF last = list.first THEN
|
|
|
|
list.first := NIL;
|
|
|
|
list.last := NIL
|
|
|
|
ELSE
|
|
|
|
list.last := last.prev;
|
|
|
|
list.last.next := NIL
|
|
|
|
END;
|
|
|
|
|
|
|
|
last.next := NIL;
|
|
|
|
last.prev := NIL
|
|
|
|
END
|
|
|
|
|
|
|
|
RETURN last
|
|
|
|
END pop;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE insert* (list: LIST; cur, nov: ITEM);
|
|
|
|
VAR
|
|
|
|
next: ITEM;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
ASSERT(nov # NIL);
|
|
|
|
ASSERT(cur # NIL);
|
|
|
|
|
|
|
|
next := cur.next;
|
|
|
|
|
|
|
|
IF next # NIL THEN
|
|
|
|
next.prev := nov;
|
|
|
|
nov.next := next;
|
|
|
|
cur.next := nov;
|
|
|
|
nov.prev := cur
|
|
|
|
ELSE
|
|
|
|
push(list, nov)
|
|
|
|
END
|
|
|
|
|
|
|
|
END insert;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE insertL* (list: LIST; cur, nov: ITEM);
|
|
|
|
VAR
|
|
|
|
prev: ITEM;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
ASSERT(nov # NIL);
|
|
|
|
ASSERT(cur # NIL);
|
|
|
|
|
|
|
|
prev := cur.prev;
|
|
|
|
|
|
|
|
IF prev # NIL THEN
|
|
|
|
prev.next := nov;
|
2020-10-13 07:58:51 +00:00
|
|
|
nov.prev := prev
|
2019-03-11 08:59:55 +00:00
|
|
|
ELSE
|
|
|
|
nov.prev := NIL;
|
|
|
|
list.first := nov
|
2020-10-13 07:58:51 +00:00
|
|
|
END;
|
|
|
|
cur.prev := nov;
|
|
|
|
nov.next := cur
|
2019-03-11 08:59:55 +00:00
|
|
|
END insertL;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE delete* (list: LIST; item: ITEM);
|
|
|
|
VAR
|
|
|
|
prev, next: ITEM;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
ASSERT(item # NIL);
|
|
|
|
|
|
|
|
prev := item.prev;
|
|
|
|
next := item.next;
|
|
|
|
|
|
|
|
IF (next # NIL) & (prev # NIL) THEN
|
|
|
|
prev.next := next;
|
|
|
|
next.prev := prev
|
|
|
|
ELSIF (next = NIL) & (prev = NIL) THEN
|
|
|
|
list.first := NIL;
|
|
|
|
list.last := NIL
|
|
|
|
ELSIF (next = NIL) & (prev # NIL) THEN
|
|
|
|
prev.next := NIL;
|
|
|
|
list.last := prev
|
|
|
|
ELSIF (next # NIL) & (prev = NIL) THEN
|
|
|
|
next.prev := NIL;
|
|
|
|
list.first := next
|
|
|
|
END
|
|
|
|
|
|
|
|
END delete;
|
|
|
|
|
|
|
|
|
|
|
|
PROCEDURE count* (list: LIST): INTEGER;
|
|
|
|
VAR
|
|
|
|
item: ITEM;
|
|
|
|
res: INTEGER;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
res := 0;
|
|
|
|
|
|
|
|
item := list.first;
|
|
|
|
WHILE item # NIL DO
|
|
|
|
INC(res);
|
|
|
|
item := item.next
|
|
|
|
END
|
|
|
|
|
|
|
|
RETURN res
|
|
|
|
END count;
|
|
|
|
|
|
|
|
|
2019-09-26 20:23:06 +00:00
|
|
|
PROCEDURE getidx* (list: LIST; idx: INTEGER): ITEM;
|
|
|
|
VAR
|
|
|
|
item: ITEM;
|
|
|
|
|
|
|
|
BEGIN
|
|
|
|
ASSERT(list # NIL);
|
|
|
|
ASSERT(idx >= 0);
|
|
|
|
|
|
|
|
item := list.first;
|
|
|
|
WHILE (item # NIL) & (idx > 0) DO
|
|
|
|
item := item.next;
|
|
|
|
DEC(idx)
|
|
|
|
END
|
|
|
|
|
|
|
|
RETURN item
|
|
|
|
END getidx;
|
|
|
|
|
|
|
|
|
2019-03-11 08:59:55 +00:00
|
|
|
PROCEDURE create* (list: LIST): LIST;
|
|
|
|
BEGIN
|
|
|
|
IF list = NIL THEN
|
|
|
|
NEW(list)
|
|
|
|
END;
|
|
|
|
|
2019-09-26 20:23:06 +00:00
|
|
|
list.first := NIL;
|
|
|
|
list.last := NIL
|
2019-03-11 08:59:55 +00:00
|
|
|
|
|
|
|
RETURN list
|
|
|
|
END create;
|
|
|
|
|
|
|
|
|
2020-05-25 20:48:33 +00:00
|
|
|
END LISTS.
|