diff options
Diffstat (limited to 'src/algebra/stream.spad.pamphlet')
-rw-r--r-- | src/algebra/stream.spad.pamphlet | 1296 |
1 files changed, 1296 insertions, 0 deletions
diff --git a/src/algebra/stream.spad.pamphlet b/src/algebra/stream.spad.pamphlet new file mode 100644 index 00000000..4edd8509 --- /dev/null +++ b/src/algebra/stream.spad.pamphlet @@ -0,0 +1,1296 @@ +\documentclass{article} +\usepackage{axiom} +\begin{document} +\title{\$SPAD/src/algebra stream.spad} +\author{Clifton J. Williamson, William Burge, Stephen M. Watt} +\maketitle +\begin{abstract} +\end{abstract} +\eject +\tableofcontents +\eject +\section{category LZSTAGG LazyStreamAggregate} +<<category LZSTAGG LazyStreamAggregate>>= +)abbrev category LZSTAGG LazyStreamAggregate +++ Category of streams with lazy evaluation +++ Author: Clifton J. Williamson +++ Date Created: 22 November 1989 +++ Date Last Updated: 20 July 1990 +++ Keywords: stream, infinite list, infinite sequence +++ Description: +++ LazyStreamAggregate is the category of streams with lazy +++ evaluation. It is understood that the function 'empty?' will +++ cause lazy evaluation if necessary to determine if there are +++ entries. Functions which call 'empty?', e.g. 'first' and 'rest', +++ will also cause lazy evaluation if necessary. + +LazyStreamAggregate(S:Type): Category == StreamAggregate(S) with + remove: (S -> Boolean,%) -> % + ++ remove(f,st) returns a stream consisting of those elements of stream + ++ st which do not satisfy the predicate f. + ++ Note: \spad{remove(f,st) = [x for x in st | not f(x)]}. + select: (S -> Boolean,%) -> % + ++ select(f,st) returns a stream consisting of those elements of stream + ++ st satisfying the predicate f. + ++ Note: \spad{select(f,st) = [x for x in st | f(x)]}. + explicitEntries?: % -> Boolean + ++ explicitEntries?(s) returns true if the stream s has + ++ explicitly computed entries, and false otherwise. + explicitlyEmpty?: % -> Boolean + ++ explicitlyEmpty?(s) returns true if the stream is an + ++ (explicitly) empty stream. + ++ Note: this is a null test which will not cause lazy evaluation. + lazy?: % -> Boolean + ++ lazy?(s) returns true if the first node of the stream s + ++ is a lazy evaluation mechanism which could produce an + ++ additional entry to s. + lazyEvaluate: % -> % + ++ lazyEvaluate(s) causes one lazy evaluation of stream s. + ++ Caution: the first node must be a lazy evaluation mechanism + ++ (satisfies \spad{lazy?(s) = true}) as there is no error check. + ++ Note: a call to this function may + ++ or may not produce an explicit first entry + frst: % -> S + ++ frst(s) returns the first element of stream s. + ++ Caution: this function should only be called after a \spad{empty?} test + ++ has been made since there no error check. + rst: % -> % + ++ rst(s) returns a pointer to the next node of stream s. + ++ Caution: this function should only be called after a \spad{empty?} test + ++ has been made since there no error check. + numberOfComputedEntries: % -> NonNegativeInteger + ++ numberOfComputedEntries(st) returns the number of explicitly + ++ computed entries of stream st which exist immediately prior to the time + ++ this function is called. + extend: (%,Integer) -> % + ++ extend(st,n) causes entries to be computed, if necessary, + ++ so that 'st' will have at least 'n' explicit entries or so + ++ that all entries of 'st' will be computed if 'st' is finite + ++ with length <= n. + complete: % -> % + ++ complete(st) causes all entries of 'st' to be computed. + ++ this function should only be called on streams which are + ++ known to be finite. + + add + + MIN ==> 1 -- minimal stream index + + I ==> Integer + NNI ==> NonNegativeInteger + L ==> List + U ==> UniversalSegment Integer + + indexx? : (Integer,%) -> Boolean + cycleElt : % -> Union(%,"failed") + computeCycleLength : % -> NNI + computeCycleEntry : (%,%) -> % + +--% SETCAT functions + + if S has SetCategory then + + x = y == + eq?(x,y) => true + explicitlyFinite? x and explicitlyFinite? y => + entries x = entries y + explicitEntries? x and explicitEntries? y => + frst x = frst y and EQ(rst x, rst y)$Lisp + -- treat cyclic streams + false + +--% HOAGG functions + + --null x == empty? x + + less?(x,n) == + n = 0 => false + empty? x => true + less?(rst x,(n-1) :: NNI) + + more?(x,n) == + empty? x => false + n = 0 => true + more?(rst x,(n-1) :: NNI) + + size?(x,n) == + empty? x => n = 0 + size?(rst x,(n-1) :: NNI) + + # x == + -- error if stream is not finite + y := x + for i in 0.. repeat + explicitlyEmpty? y => return i + lazy? y => error "#: infinite stream" + y := rst y + if odd? i then x := rst x + eq?(x,y) => error "#: infinite stream" + +--% CLAGG functions + + any?(f,x) == + -- error message only when x is a stream with lazy + -- evaluation and f(s) = false for all stream elements + -- 's' which have been computed when the function is + -- called + y := x + for i in 0.. repeat + explicitlyEmpty? y => return false + lazy? y => error "any?: infinite stream" + f frst y => return true + y := rst y + if odd? i then x := rst x + eq?(x,y) => return false + + every?(f,x) == + -- error message only when x is a stream with lazy + -- evaluation and f(s) = true for all stream elements + -- 's' which have been computed when the function is + -- called + y := x + for i in 0.. repeat + explicitlyEmpty? y => return true + lazy? y => error "every?: infinite stream" + not f frst y => return false + y := rst y + if odd? i then x := rst x + eq?(x,y) => return true + +-- following ops count and member? are only exported if $ has finiteAggregate + +-- count(f:S -> Boolean,x:%) == +-- -- error if stream is not finite +-- count : NNI := 0 +-- y := x +-- for i in 0.. repeat +-- explicitlyEmpty? y => return count +-- lazy? y => error "count: infinite stream" +-- if f frst y then count := count + 1 +-- y := rst y +-- if odd? i then x := rst x +-- eq?(x,y) => error "count: infinite stream" + + +-- if S has SetCategory then + +-- count(s:S,x:%) == count(#1 = s,x) +-- -- error if stream is not finite + +-- member?(s,x) == +-- -- error message only when x is a stream with lazy +-- -- evaluation and 's' is not among the stream elements +-- -- which have been computed when the function is called +-- y := x +-- for i in 0.. repeat +-- explicitlyEmpty? y => return false +-- lazy? y => error "member?: infinite stream" +-- frst y = s => return true +-- y := rst y +-- if odd? i then x := rst x +-- eq?(x,y) => return false + + entries x == + -- returns a list of elements which have been computed + -- error if infinite + y := x + l : L S := empty() + for i in 0.. repeat + explicitlyEmpty? y => return reverse_! l + lazy? y => error "infinite stream" + l := concat(frst y,l) + y := rst y + if odd? i then x := rst x + eq?(x,y) => error "infinite stream" + +--% CNAGG functions + + construct l == + empty? l => empty() + concat(first l, construct rest l) + + --entries x == + -- returns a list of the stream elements + -- error if the stream is not finite + --members x + +--% ELTAGG functions + + elt(x:%,n:I) == + n < MIN or empty? x => error "elt: no such element" + n = MIN => frst x + elt(rst x,n - 1) + + elt(x:%,n:I,s:S) == + n < MIN or empty? x => s + n = MIN => frst x + elt(rst x,n - 1) + +--% IXAGG functions + +-- following assumes % has finiteAggregate and S has SetCategory +-- entry?(s,x) == +-- -- error message only when x is a stream with lazy +-- -- evaluation and 's' is not among the stream elements +-- -- which have been computed when the function is called +-- member?(s,x) + + --entries x == + -- error if the stream is not finite + --members x + + indexx?(n,x) == + empty? x => false + n = MIN => true + indexx?(n-1,rst x) + + index?(n,x) == + -- returns 'true' iff 'n' is the index of an entry which + -- may or may not have been computed when the function is + -- called + -- additional entries are computed if necessary + n < MIN => false + indexx?(n,x) + + indices x == + -- error if stream is not finite + y := x + l : L I := empty() + for i in MIN.. repeat + explicitlyEmpty? y => return reverse_! l + lazy? y => error "indices: infinite stream" + l := concat(i,l) + y := rst y + if odd? i then x := rst x + eq?(x,y) => error "indices: infinite stream" + + maxIndex x == + -- error if stream is not finite + empty? x => + error "maxIndex: no maximal index for empty stream" + y := rst x + for i in MIN.. repeat + explicitlyEmpty? y => return i + lazy? y => error "maxIndex: infinite stream" + y := rst y + if odd? i then x := rst x + eq?(x,y) => error "maxIndex: infinite stream" + + minIndex x == + empty? x => error "minIndex: no minimal index for empty stream" + MIN + +--% LNAGG functions + + delete(x:%,n:I) == + -- non-destructive + not index?(n,x) => error "delete: index out of range" + concat(first(x,(n - MIN) :: NNI), rest(x,(n - MIN + 1) :: NNI)) + + delete(x:%,seg:U) == + low := lo seg + hasHi seg => + high := hi seg + high < low => copy x + (not index?(low,x)) or (not index?(high,x)) => + error "delete: index out of range" + concat(first(x,(low - MIN) :: NNI),rest(x,(high - MIN + 1) :: NNI)) + not index?(low,x) => error "delete: index out of range" + first(x,(low - MIN) :: NNI) + + elt(x:%,seg:U) == + low := lo seg + hasHi seg => + high := hi seg + high < low => empty() + (not index?(low,x)) or (not index?(high,x)) => + error "elt: index out of range" + first(rest(x,(low - MIN) :: NNI),(high - low + 1) :: NNI) + not index?(low,x) => error "elt: index out of range" + rest(x,(low - MIN) :: NNI) + + insert(s:S,x:%,n:I) == + not index?(n,x) => error "insert: index out of range" + nn := (n - MIN) :: NNI + concat([first(x,nn), concat(s, empty()), rest(x,nn)]) + + insert(y:%,x:%,n:I) == + not index?(n,x) => error "insert: index out of range" + nn := (n - MIN) :: NNI + concat([first(x,nn), y, rest(x,nn)]) + +--% RCAGG functions + + cycleElt x == cycleElt(x)$CyclicStreamTools(S,%) + + cyclic? x == + cycleElt(x) case "failed" => false + true + + if S has SetCategory then + child?(x,y) == + empty? y => error "child: no children" + x = rst y + + children x == + empty? x => error "children: no children" + [rst x] + + distance(x,z) == + y := x + for i in 0.. repeat + eq?(y,z) => return i + (explicitlyEmpty? y) or (lazy? y) => + error "distance: 2nd arg not a descendent of the 1st" + y := rst y + if odd? i then x := rst x + eq?(x,y) => + error "distance: 2nd arg not a descendent of the 1st" + + if S has SetCategory then + node?(z,x) == + -- error message only when x is a stream with lazy + -- evaluation and 'y' is not a node of 'x' + -- which has been computed when the function is called + y := x + for i in 0.. repeat + z = y => return true + explicitlyEmpty? y => return false + lazy? y => error "node?: infinite stream" + y := rst y + if odd? i then x := rst x + eq?(x,y) => return false + + nodes x == + y := x + l : L % := [] + for i in 0.. repeat + explicitlyEmpty? y => return reverse_! l + lazy? y => error "nodes: infinite stream" + l := concat(y,l) + y := rst y + if odd? i then x := rst x + eq?(x,y) => error "nodes: infinite stream" + l -- @#$%^& compiler + + leaf? x == empty? rest x + + value x == first x + +--% URAGG functions + + computeCycleLength cycElt == + computeCycleLength(cycElt)$CyclicStreamTools(S,%) + + computeCycleEntry(x,cycElt) == + computeCycleEntry(x,cycElt)$CyclicStreamTools(S,%) + + cycleEntry x == + cycElt := cycleElt x + cycElt case "failed" => + error "cycleEntry: non-cyclic stream" + computeCycleEntry(x,cycElt::%) + + cycleLength x == + cycElt := cycleElt x + cycElt case "failed" => + error "cycleLength: non-cyclic stream" + computeCycleLength(cycElt::%) + + cycleTail x == + cycElt := cycleElt x + cycElt case "failed" => + error "cycleTail: non-cyclic stream" + y := x := computeCycleEntry(x,cycElt::%) + z := rst x + repeat + eq?(x,z) => return y + y := z ; z := rst z + + elt(x,"first") == first x + + first(x,n) == + -- former name: take + n = 0 or empty? x => empty() + concat(frst x, first(rst x,(n-1) :: NNI)) + + rest x == + empty? x => error "Can't take the rest of an empty stream." + rst x + + elt(x,"rest") == rest x + + rest(x,n) == + -- former name: drop + n = 0 or empty? x => x + rest(rst x,(n-1) :: NNI) + + last x == + -- error if stream is not finite + empty? x => error "last: empty stream" + y1 := x + y2 := rst x + for i in 0.. repeat + explicitlyEmpty? y2 => return frst y1 + lazy? y2 => error "last: infinite stream" + y1 := y2 + y2 := rst y2 + if odd? i then x := rst x + eq?(x,y2) => error "last: infinite stream" + + if % has finiteAggregate then -- # is only defined for finiteAggregates + last(x,n) == + possiblyInfinite? x => error "last: infinite stream" + m := # x + m < n => error "last: index out of range" + copy rest(x,(m-n)::NNI) + + elt(x,"last") == last x + + tail x == + -- error if stream is not finite + empty? x => error "tail: empty stream" + y1 := x + y2 := rst x + for i in 0.. repeat + explicitlyEmpty? y2 => return y1 + lazy? y2 => error "tail: infinite stream" + y1 := y2 + y2 := rst y2 + if odd? i then x := rst x + eq?(x,y2) => error "tail: infinite stream" + +--% STAGG functions + + possiblyInfinite? x == + y := x + for i in 0.. repeat + explicitlyEmpty? y => return false + lazy? y => return true + if odd? i then x := rst x + y := rst y + eq?(x,y) => return true + + explicitlyFinite? x == not possiblyInfinite? x + +--% LZSTAGG functions + + extend(x,n) == + y := x + for i in 1..n while not empty? y repeat y := rst y + x + + complete x == + y := x + while not empty? y repeat y := rst y + x + +@ +\section{package CSTTOOLS CyclicStreamTools} +<<package CSTTOOLS CyclicStreamTools>>= +)abbrev package CSTTOOLS CyclicStreamTools +++ Functions for dealing with cyclic streams +++ Author: Clifton J. Williamson +++ Date Created: 5 December 1989 +++ Date Last Updated: 5 December 1989 +++ Keywords: stream, cyclic +++ Description: +++ This package provides tools for working with cyclic streams. +CyclicStreamTools(S,ST): Exports == Implementation where + S : Type + ST : LazyStreamAggregate S + + Exports ==> with + + cycleElt: ST -> Union(ST,"failed") + ++ cycleElt(s) returns a pointer to a node in the cycle if the stream s is + ++ cyclic and returns "failed" if s is not cyclic + computeCycleLength: ST -> NonNegativeInteger + ++ computeCycleLength(s) returns the length of the cycle of a + ++ cyclic stream t, where s is a pointer to a node in the + ++ cyclic part of t. + computeCycleEntry: (ST,ST) -> ST + ++ computeCycleEntry(x,cycElt), where cycElt is a pointer to a + ++ node in the cyclic part of the cyclic stream x, returns a + ++ pointer to the first node in the cycle + + Implementation ==> add + + cycleElt x == + y := x + for i in 0.. repeat + (explicitlyEmpty? y) or (lazy? y) => return "failed" + y := rst y + if odd? i then x := rst x + eq?(x,y) => return y + + computeCycleLength cycElt == + i : NonNegativeInteger + y := cycElt + for i in 1.. repeat + y := rst y + eq?(y,cycElt) => return i + + computeCycleEntry(x,cycElt) == + y := rest(x, computeCycleLength cycElt) + repeat + eq?(x,y) => return x + x := rst x ; y := rst y + +@ +\section{domain STREAM Stream} +<<domain STREAM Stream>>= +)abbrev domain STREAM Stream +++ Implementation of streams via lazy evaluation +++ Authors: Burge, Watt; updated by Clifton J. Williamson +++ Date Created: July 1986 +++ Date Last Updated: 30 March 1990 +++ Keywords: stream, infinite list, infinite sequence +++ Examples: +++ References: +++ Description: +++ A stream is an implementation of an infinite sequence using +++ a list of terms that have been computed and a function closure +++ to compute additional terms when needed. + +Stream(S): Exports == Implementation where +-- problems: +-- 1) dealing with functions which basically want a finite structure +-- 2) 'map' doesn't deal with cycles very well + + S : Type + B ==> Boolean + OUT ==> OutputForm + I ==> Integer + L ==> List + NNI ==> NonNegativeInteger + U ==> UniversalSegment I + + Exports ==> LazyStreamAggregate(S) with + shallowlyMutable + ++ one may destructively alter a stream by assigning new + ++ values to its entries. + + coerce: L S -> % + ++ coerce(l) converts a list l to a stream. + repeating: L S -> % + ++ repeating(l) is a repeating stream whose period is the list l. + if S has SetCategory then + repeating?: (L S,%) -> B + ++ repeating?(l,s) returns true if a stream s is periodic + ++ with period l, and false otherwise. + findCycle: (NNI,%) -> Record(cycle?: B, prefix: NNI, period: NNI) + ++ findCycle(n,st) determines if st is periodic within n. + delay: (() -> %) -> % + ++ delay(f) creates a stream with a lazy evaluation defined by function f. + ++ Caution: This function can only be called in compiled code. + cons: (S,%) -> % + ++ cons(a,s) returns a stream whose \spad{first} is \spad{a} + ++ and whose \spad{rest} is s. + ++ Note: \spad{cons(a,s) = concat(a,s)}. + if S has SetCategory then + output: (I, %) -> Void + ++ output(n,st) computes and displays the first n entries + ++ of st. + showAllElements: % -> OUT + ++ showAllElements(s) creates an output form which displays all + ++ computed elements. + showAll?: () -> B + ++ showAll?() returns true if all computed entries of streams + ++ will be displayed. + --!! this should be a function of one argument + setrest_!: (%,I,%) -> % + ++ setrest!(x,n,y) sets rest(x,n) to y. The function will expand + ++ cycles if necessary. + generate: (() -> S) -> % + ++ generate(f) creates an infinite stream all of whose elements are + ++ equal to \spad{f()}. + ++ Note: \spad{generate(f) = [f(),f(),f(),...]}. + generate: (S -> S,S) -> % + ++ generate(f,x) creates an infinite stream whose first element is + ++ x and whose nth element (\spad{n > 1}) is f applied to the previous + ++ element. Note: \spad{generate(f,x) = [x,f(x),f(f(x)),...]}. + filterWhile: (S -> Boolean,%) -> % + ++ filterWhile(p,s) returns \spad{[x0,x1,...,x(n-1)]} where + ++ \spad{s = [x0,x1,x2,..]} and + ++ n is the smallest index such that \spad{p(xn) = false}. + filterUntil: (S -> Boolean,%) -> % + ++ filterUntil(p,s) returns \spad{[x0,x1,...,x(n)]} where + ++ \spad{s = [x0,x1,x2,..]} and + ++ n is the smallest index such that \spad{p(xn) = true}. +-- if S has SetCategory then +-- map: ((S,S) -> S,%,%,S) -> % +-- ++ map(f,x,y,a) is equivalent to map(f,x,y) +-- ++ If z = map(f,x,y,a), then z = map(f,x,y) except if +-- ++ x.n = a and rest(rest(x,n)) = rest(x,n) in which case +-- ++ rest(z,n) = rest(y,n) or if y.m = a and rest(rest(y,m)) = +-- ++ rest(y,m) in which case rest(z,n) = rest(x,n). +-- ++ Think of the case where f(xi,yi) = xi + yi and a = 0. + + Implementation ==> add + MIN ==> 1 -- minimal stream index; see also the defaults in LZSTAGG + x:% + + import CyclicStreamTools(S,%) + +--% representation + + -- This description of the rep is not quite true. + -- The Rep is a pair of one of three forms: + -- [value: S, rest: %] + -- [nullstream: Magic, NIL ] + -- [nonnullstream: Magic, fun: () -> %] + -- Could use a record of unions if we could guarantee no tags. + + NullStream: S := _$NullStream$Lisp pretend S + NonNullStream: S := _$NonNullStream$Lisp pretend S + + Rep := Record(firstElt: S, restOfStream: %) + + explicitlyEmpty? x == EQ(frst x,NullStream)$Lisp + lazy? x == EQ(frst x,NonNullStream)$Lisp + +--% signatures of local functions + + setfrst_! : (%,S) -> S + setrst_! : (%,%) -> % + setToNil_! : % -> % + setrestt_! : (%,I,%) -> % + lazyEval : % -> % + expand_! : (%,I) -> % + +--% functions to access or change record fields without lazy evaluation + + frst x == x.firstElt + rst x == x.restOfStream + + setfrst_!(x,s) == x.firstElt := s + setrst_!(x,y) == x.restOfStream := y + + setToNil_! x == + -- destructively changes x to a null stream + setfrst_!(x,NullStream); setrst_!(x,NIL$Lisp) + x + +--% SETCAT functions + + if S has SetCategory then + + getm : (%,L OUT,I) -> L OUT + streamCountCoerce : % -> OUT + listm : (%,L OUT,I) -> L OUT + + getm(x,le,n) == + explicitlyEmpty? x => le + lazy? x => + n > 0 => + empty? x => le + getm(rst x,concat(frst(x) :: OUT,le),n - 1) + concat(message("..."),le) + eq?(x,rst x) => concat(overbar(frst(x) :: OUT),le) + n > 0 => getm(rst x,concat(frst(x) :: OUT,le),n - 1) + concat(message("..."),le) + + streamCountCoerce x == + -- this will not necessarily display all stream elements + -- which have been computed + count := _$streamCount$Lisp + -- compute count elements + y := x + for i in 1..count while not empty? y repeat y := rst y + fc := findCycle(count,x) + not fc.cycle? => bracket reverse_! getm(x,empty(),count) + le : L OUT := empty() + for i in 1..fc.prefix repeat + le := concat(first(x) :: OUT,le) + x := rest x + pp : OUT := + fc.period = 1 => overbar(frst(x) :: OUT) + pl : L OUT := empty() + for i in 1..fc.period repeat + pl := concat(frst(x) :: OUT,pl) + x := rest x + overbar commaSeparate reverse_! pl + bracket reverse_! concat(pp,le) + + listm(x,le,n) == + explicitlyEmpty? x => le + lazy? x => + n > 0 => + empty? x => le + listm(rst x, concat(frst(x) :: OUT,le),n-1) + concat(message("..."),le) + listm(rst x,concat(frst(x) :: OUT,le),n-1) + + showAllElements x == + -- this will display all stream elements which have been computed + -- and will display at least n elements with n = streamCount$Lisp + extend(x,_$streamCount$Lisp) + cycElt := cycleElt x + cycElt case "failed" => + le := listm(x,empty(),_$streamCount$Lisp) + bracket reverse_! le + cycEnt := computeCycleEntry(x,cycElt :: %) + le : L OUT := empty() + while not eq?(x,cycEnt) repeat + le := concat(frst(x) :: OUT,le) + x := rst x + len := computeCycleLength(cycElt :: %) + pp : OUT := + len = 1 => overbar(frst(x) :: OUT) + pl : L OUT := [] + for i in 1..len repeat + pl := concat(frst(x) :: OUT,pl) + x := rst x + overbar commaSeparate reverse_! pl + bracket reverse_! concat(pp,le) + + showAll?() == + NULL(_$streamsShowAll$Lisp)$Lisp => false + true + + coerce(x):OUT == + showAll?() => showAllElements x + streamCountCoerce x + +--% AGG functions + + lazyCopy:% -> % + lazyCopy x == delay + empty? x => empty() + concat(frst x, copy rst x) + + copy x == + cycElt := cycleElt x + cycElt case "failed" => lazyCopy x + ce := cycElt :: % + len := computeCycleLength(ce) + e := computeCycleEntry(x,ce) + d := distance(x,e) + cycle := complete first(e,len) + setrst_!(tail cycle,cycle) + d = 0 => cycle + head := complete first(x,d::NNI) + setrst_!(tail head,cycle) + head + +--% CNAGG functions + + construct l == + -- copied from defaults to avoid loading defaults + empty? l => empty() + concat(first l, construct rest l) + +--% ELTAGG functions + + elt(x:%,n:I) == + -- copied from defaults to avoid loading defaults + n < MIN or empty? x => error "elt: no such element" + n = MIN => frst x + elt(rst x,n - 1) + + seteltt:(%,I,S) -> S + seteltt(x,n,s) == + n = MIN => setfrst_!(x,s) + seteltt(rst x,n - 1,s) + + setelt(x,n:I,s:S) == + n < MIN or empty? x => error "setelt: no such element" + x := expand_!(x,n - MIN + 1) + seteltt(x,n,s) + +--% IXAGG functions + + removee: ((S -> Boolean),%) -> % + removee(p,x) == delay + empty? x => empty() + p(frst x) => remove(p,rst x) + concat(frst x,remove(p,rst x)) + + remove(p,x) == + explicitlyEmpty? x => empty() + eq?(x,rst x) => + p(frst x) => empty() + x + removee(p,x) + + selectt: ((S -> Boolean),%) -> % + selectt(p,x) == delay + empty? x => empty() + not p(frst x) => select(p, rst x) + concat(frst x,select(p,rst x)) + + select(p,x) == + explicitlyEmpty? x => empty() + eq?(x,rst x) => + p(frst x) => x + empty() + selectt(p,x) + + map(f,x) == + map(f,x pretend Stream(S))$StreamFunctions2(S,S) pretend % + + map(g,x,y) == + xs := x pretend Stream(S); ys := y pretend Stream(S) + map(g,xs,ys)$StreamFunctions3(S,S,S) pretend % + + fill_!(x,s) == + setfrst_!(x,s) + setrst_!(x,x) + + map_!(f,x) == + -- too many problems with map_! on a lazy stream, so + -- in this case, an error message is returned + cyclic? x => + tail := cycleTail x ; y := x + until y = tail repeat + setfrst_!(y,f frst y) + y := rst y + x + explicitlyFinite? x => + y := x + while not empty? y repeat + setfrst_!(y,f frst y) + y := rst y + x + error "map!: stream with lazy evaluation" + + swap_!(x,m,n) == + (not index?(m,x)) or (not index?(n,x)) => + error "swap!: no such elements" + x := expand_!(x,max(m,n) - MIN + 1) + xm := elt(x,m); xn := elt(x,n) + setelt(x,m,xn); setelt(x,n,xm) + x + +--% LNAGG functions + + concat(x:%,s:S) == delay + empty? x => concat(s,empty()) + concat(frst x,concat(rst x,s)) + + concat(x:%,y:%) == delay + empty? x => copy y + concat(frst x,concat(rst x, y)) + + concat l == delay + empty? l => empty() + empty?(x := first l) => concat rest l + concat(frst x,concat(rst x,concat rest l)) + + setelt(x,seg:U,s:S) == + low := lo seg + hasHi seg => + high := hi seg + high < low => s + (not index?(low,x)) or (not index?(high,x)) => + error "setelt: index out of range" + x := expand_!(x,high - MIN + 1) + y := rest(x,(low - MIN) :: NNI) + for i in 0..(high-low) repeat + setfrst_!(y,s) + y := rst y + s + not index?(low,x) => error "setelt: index out of range" + x := rest(x,(low - MIN) :: NNI) + setrst_!(x,x) + setfrst_!(x,s) + +--% RCAGG functions + + empty() == [NullStream, NIL$Lisp] + + lazyEval x == (rst(x):(()-> %)) () + + lazyEvaluate x == + st := lazyEval x + setfrst_!(x, frst st) + setrst_!(x,if EQ(rst st,st)$Lisp then x else rst st) + x + + -- empty? is the only function that explicitly causes evaluation + -- of a stream element + empty? x == + while lazy? x repeat + st := lazyEval x + setfrst_!(x, frst st) + setrst_!(x,if EQ(rst st,st)$Lisp then x else rst st) + explicitlyEmpty? x + + --setvalue(x,s) == setfirst_!(x,s) + + --setchildren(x,l) == + --empty? l => error "setchildren: empty list of children" + --not(empty? rest l) => error "setchildren: wrong number of children" + --setrest_!(x,first l) + +--% URAGG functions + + first(x,n) == delay + -- former name: take + n = 0 or empty? x => empty() + (concat(frst x, first(rst x,(n-1) :: NNI))) + + concat(s:S,x:%) == [s,x] + cons(s,x) == concat(s,x) + + cycleSplit_! x == + cycElt := cycleElt x + cycElt case "failed" => + error "cycleSplit_!: non-cyclic stream" + y := computeCycleEntry(x,cycElt :: %) + eq?(x,y) => (setToNil_! x; return y) + z := rst x + repeat + eq?(y,z) => (setrest_!(x,empty()); return y) + x := z ; z := rst z + + expand_!(x,n) == + -- expands cycles (if necessary) so that the first n + -- elements of x will not be part of a cycle + n < 1 => x + y := x + for i in 1..n while not empty? y repeat y := rst y + cycElt := cycleElt x + cycElt case "failed" => x + e := computeCycleEntry(x,cycElt :: %) + d : I := distance(x,e) + d >= n => x + if d = 0 then + -- roll the cycle 1 entry + d := 1 + t := cycleTail e + if eq?(t,e) then + t := concat(frst t,empty()) + e := setrst_!(t,t) + setrst_!(x,e) + else + setrst_!(t,concat(frst e,rst e)) + e := rst e + nLessD := (n-d) :: NNI + y := complete first(e,nLessD) + e := rest(e,nLessD) + setrst_!(tail y,e) + setrst_!(rest(x,(d-1) :: NNI),y) + x + + first x == + empty? x => error "Can't take the first of an empty stream." + frst x + + concat_!(x:%,y:%) == + empty? x => y + setrst_!(tail x,y) + + concat_!(x:%,s:S) == + concat_!(x,concat(s,empty())) + + setfirst_!(x,s) == setelt(x,0,s) + setelt(x,"first",s) == setfirst_!(x,s) + setrest_!(x,y) == + empty? x => error "setrest!: empty stream" + setrst_!(x,y) + setelt(x,"rest",y) == setrest_!(x,y) + + setlast_!(x,s) == + empty? x => error "setlast!: empty stream" + setfrst_!(tail x, s) + setelt(x,"last",s) == setlast_!(x,s) + + split_!(x,n) == + n < MIN => error "split!: index out of range" + n = MIN => + y : % := empty() + setfrst_!(y,frst x) + setrst_!(y,rst x) + setToNil_! x + y + x := expand_!(x,n - MIN) + x := rest(x,(n - MIN - 1) :: NNI) + y := rest x + setrst_!(x,empty()) + y + +--% STREAM functions + + coerce(l: L S) == construct l + + repeating l == + empty? l => + error "Need a non-null list to make a repeating stream." + x0 : % := x := construct l + while not empty? rst x repeat x := rst x + setrst_!(x,x0) + + if S has SetCategory then + + repeating?(l, x) == + empty? l => + error "Need a non-empty? list to make a repeating stream." + empty? rest l => + not empty? x and frst x = first l and x = rst x + x0 := x + for s in l repeat + empty? x or s ^= frst x => return false + x := rst x + eq?(x,x0) + + findCycle(n, x) == + hd := x + -- Determine whether periodic within n. + tl := rest(x, n) + explicitlyEmpty? tl => [false, 0, 0] + i := 0; while not eq?(x,tl) repeat (x := rst x; i := i + 1) + i = n => [false, 0, 0] + -- Find period. Now x=tl, so step over and find it again. + x := rst x; per := 1 + while not eq?(x,tl) repeat (x := rst x; per := per + 1) + -- Find non-periodic part. + x := hd; xp := rest(hd, per); npp := 0 + while not eq?(x,xp) repeat (x := rst x; xp := rst xp; npp := npp+1) + [true, npp, per] + + delay(fs:()->%) == [NonNullStream, fs pretend %] + +-- explicitlyEmpty? x == markedNull? x + + explicitEntries? x == + not explicitlyEmpty? x and not lazy? x + + numberOfComputedEntries x == + explicitEntries? x => numberOfComputedEntries(rst x) + 1 + 0 + + if S has SetCategory then + + output(n,x) == + (not(n>0))or empty? x => void() + mathPrint(frst(x)::OUT)$Lisp + output(n-1, rst x) + + setrestt_!(x,n,y) == + n = 0 => setrst_!(x,y) + setrestt_!(rst x,n-1,y) + + setrest_!(x,n,y) == + n < 0 or empty? x => error "setrest!: no such rest" + x := expand_!(x,n+1) + setrestt_!(x,n,y) + + generate f == delay concat(f(), generate f) + gen:(S -> S,S) -> % + gen(f,s) == delay(ss:=f s; concat(ss, gen(f,ss))) + generate(f,s)==concat(s,gen(f,s)) + + concat(x:%,y:%) ==delay + empty? x => y + concat(frst x,concat(rst x,y)) + + swhilee:(S -> Boolean,%) -> % + swhilee(p,x) == delay + empty? x => empty() + not p(frst x) => empty() + concat(frst x,filterWhile(p,rst x)) + filterWhile(p,x)== + explicitlyEmpty? x => empty() + eq?(x,rst x) => + p(frst x) => x + empty() + swhilee(p,x) + + suntill: (S -> Boolean,%) -> % + suntill(p,x) == delay + empty? x => empty() + p(frst x) => concat(frst x,empty()) + concat(frst x, filterUntil(p, rst x)) + + filterUntil(p,x)== + explicitlyEmpty? x => empty() + eq?(x,rst x) => + p(frst x) => concat(frst x,empty()) + x + suntill(p,x) + +-- if S has SetCategory then +-- mapp: ((S,S) -> S,%,%,S) -> % +-- mapp(f,x,y,a) == delay +-- empty? x or empty? y => empty() +-- concat(f(frst x,frst y), map(f,rst x,rst y,a)) +-- map(f,x,y,a) == +-- explicitlyEmpty? x => empty() +-- eq?(x,rst x) => +-- frst x=a => y +-- map(f(frst x,#1),y) +-- explicitlyEmpty? y => empty() +-- eq?(y,rst y) => +-- frst y=a => x +-- p(f(#1,frst y),x) +-- mapp(f,x,y,a) + +@ +\section{package STREAM1 StreamFunctions1} +<<package STREAM1 StreamFunctions1>>= +)abbrev package STREAM1 StreamFunctions1 +++ Authors: Burge, Watt; updated by Clifton J. Williamson +++ Date Created: July 1986 +++ Date Last Updated: 29 January 1990 +++ Keywords: stream, infinite list, infinite sequence +StreamFunctions1(S:Type): Exports == Implementation where + ++ Functions defined on streams with entries in one set. + ST ==> Stream + + Exports ==> with + concat: ST ST S -> ST S + ++ concat(u) returns the left-to-right concatentation of the streams in u. + ++ Note: \spad{concat(u) = reduce(concat,u)}. + + Implementation ==> add + + concat z == delay + empty? z => empty() + empty?(x := frst z) => concat rst z + concat(frst x,concat(rst x,concat rst z)) + +@ +\section{package STREAM2 StreamFunctions2} +<<package STREAM2 StreamFunctions2>>= +)abbrev package STREAM2 StreamFunctions2 +++ Authors: Burge, Watt; updated by Clifton J. Williamson +++ Date Created: July 1986 +++ Date Last Updated: 29 January 1990 +++ Keywords: stream, infinite list, infinite sequence +StreamFunctions2(A:Type,B:Type): Exports == Implementation where + ++ Functions defined on streams with entries in two sets. + ST ==> Stream + + Exports ==> with + map: ((A -> B),ST A) -> ST B + ++ map(f,s) returns a stream whose elements are the function f applied + ++ to the corresponding elements of s. + ++ Note: \spad{map(f,[x0,x1,x2,...]) = [f(x0),f(x1),f(x2),..]}. + scan: (B,((A,B) -> B),ST A) -> ST B + ++ scan(b,h,[x0,x1,x2,...]) returns \spad{[y0,y1,y2,...]}, where + ++ \spad{y0 = h(x0,b)}, + ++ \spad{y1 = h(x1,y0)},\spad{...} + ++ \spad{yn = h(xn,y(n-1))}. + reduce: (B,(A,B) -> B,ST A) -> B + ++ reduce(b,f,u), where u is a finite stream \spad{[x0,x1,...,xn]}, + ++ returns the value \spad{r(n)} computed as follows: + ++ \spad{r0 = f(x0,b), + ++ r1 = f(x1,r0),..., + ++ r(n) = f(xn,r(n-1))}. +-- rreduce: (B,(A,B) -> B,ST A) -> B +-- ++ reduce(b,h,[x0,x1,..,xn]) = h(x1,h(x2(..,h(x(n-1),h(xn,b))..) +-- reshape: (ST B,ST A) -> ST B +-- ++ reshape(y,x) = y + + Implementation ==> add + + mapp: (A -> B,ST A) -> ST B + mapp(f,x)== delay + empty? x => empty() + concat(f frst x, map(f,rst x)) + + map(f,x) == + explicitlyEmpty? x => empty() + eq?(x,rst x) => repeating([f frst x]) + mapp(f, x) + +-- reshape(y,x) == y + + scan(b,h,x) == delay + empty? x => empty() + c := h(frst x,b) + concat(c,scan(c,h,rst x)) + + reduce(b,h,x) == + empty? x => b + reduce(h(frst x,b),h,rst x) +-- rreduce(b,h,x) == +-- empty? x => b +-- h(frst x,rreduce(b,h,rst x)) + +@ +\section{package STREAM3 StreamFunctions3} +<<package STREAM3 StreamFunctions3>>= +)abbrev package STREAM3 StreamFunctions3 +++ Authors: Burge, Watt; updated by Clifton J. Williamson +++ Date Created: July 1986 +++ Date Last Updated: 29 January 1990 +++ Keywords: stream, infinite list, infinite sequence +StreamFunctions3(A,B,C): Exports == Implementation where + ++ Functions defined on streams with entries in three sets. + A : Type + B : Type + C : Type + ST ==> Stream + + Exports ==> with + map: ((A,B) -> C,ST A,ST B) -> ST C + ++ map(f,st1,st2) returns the stream whose elements are the + ++ function f applied to the corresponding elements of st1 and st2. + ++ Note: \spad{map(f,[x0,x1,x2,..],[y0,y1,y2,..]) = [f(x0,y0),f(x1,y1),..]}. + + Implementation ==> add + + mapp:((A,B) -> C,ST A,ST B) -> ST C + mapp(g,x,y) == delay + empty? x or empty? y => empty() + concat(g(frst x,frst y), map(g,rst x,rst y)) + + map(g,x,y) == + explicitlyEmpty? x => empty() + eq?(x,rst x) => map(g(frst x,#1),y)$StreamFunctions2(B,C) + explicitlyEmpty? y => empty() + eq?(y,rst y) => map(g(#1,frst y),x)$StreamFunctions2(A,C) + mapp(g,x,y) + +@ +\section{License} +<<license>>= +--Copyright (c) 1991-2002, The Numerical ALgorithms Group Ltd. +--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 Numerical ALgorithms Group Ltd. 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 THE COPYRIGHT OWNER +--OR CONTRIBUTORS 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. +@ +<<*>>= +<<license>> + +<<category LZSTAGG LazyStreamAggregate>> +<<package CSTTOOLS CyclicStreamTools>> +<<domain STREAM Stream>> +<<package STREAM1 StreamFunctions1>> +<<package STREAM2 StreamFunctions2>> +<<package STREAM3 StreamFunctions3>> +@ +\eject +\begin{thebibliography}{99} +\bibitem{1} nothing +\end{thebibliography} +\end{document} |