123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 |
- ;;; STREAM -- LIBRARY OF SYNTAX AND FUNCTIONS TO MANIPULATE STREAMS
- ;;; A stream is a new data type, disjoint from all other data types, that
- ;;; contains a promise that, when forced, is either nil (a single object
- ;;; distinguishable from all other objects) or consists of an object
- ;;; (the stream element) followed by a stream. Each stream element is
- ;;; evaluated exactly once, when it is first retrieved (not when it is
- ;;; created); once evaluated its value is saved to be returned by
- ;;; subsequent retrievals without being evaluated again.
- ; Copyright (C) 2003 by Philip L. Bewig of Saint Louis, Missouri,
- ; United States of America. All rights reserved.
- ; This document and translations of it may be copied and furnished to
- ; others, and derivative works that comment on or otherwise explain it
- ; or assist in its implementation may be prepared, copied, published
- ; and distributed, in whole or in part, without restriction of any
- ; kind, provided that the above copyright notice and this paragraph
- ; are included on all such copies and derivative works. However, this
- ; document itself may not be modified in any way, such as by removing
- ; the copyright notice or references to the Scheme Request For
- ; Implementation process or editors, except as needed for the purpose
- ; of developing SRFIs in which case the procedures for copyrights
- ; defined in the SRFI process must be followed, or as required to
- ; translate it into languages other than English.
- ; The limited permissions granted above are perpetual and will not be
- ; revoked by the authors or their successors or assigns.
- ; This document and the information contained herein is provided on an
- ; "AS IS" basis and THE AUTHOR AND THE SRFI EDITORS DISCLAIM ALL
- ; WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY
- ; WARRANTY THAT THE USE OF THE INFORMATION HEREIN WILL NOT INFRINGE
- ; ANY RIGHTS OR ANY IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS
- ; FOR A PARTICULAR PURPOSE.
- ;;; PROMISES A LA SRFI-45:
- ;;; A separate implementation is necessary to
- ;;; have promises that answer #t to stream?
- ;;; This requires lots of complicated type conversions.
- (define-record-type stream-promise :stream-promise
- (make-s:promise kind content)
- s:promise?
- (kind s:promise-kind set-s:promise-kind!)
- (content s:promise-content set-s:promise-content!))
- (define-syntax lazy
- (syntax-rules ()
- ((lazy exp)
- (make-cell (make-s:promise 'lazy (lambda () exp))))))
- (define (eager x)
- (make-stream (make-cell (make-s:promise 'eager x))))
- (define-syntax delay
- (syntax-rules ()
- ((delay exp) (lazy (eager exp)))))
- (define (force promise)
- (let ((content (cell-ref promise)))
- (case (s:promise-kind content)
- ((eager) (s:promise-content content))
- ((lazy)
- (let* ((promise* (stream-promise ((s:promise-content content))))
- (content (cell-ref promise)))
- (if (not (eqv? 'eager (s:promise-kind content)))
- (begin
- (set-s:promise-kind! content (s:promise-kind (cell-ref promise*)))
- (set-s:promise-content! content (s:promise-content (cell-ref promise*)))
- (cell-set! promise* content)))
- (force promise))))))
- ;; STREAM-TYPE -- type of streams
- ;; STREAM? object -- #t if object is a stream, #f otherwise
- (define-record-type stream :stream
- (make-stream promise)
- stream?
- (promise stream-promise))
- ;;; UTILITY FUNCTIONS
- ;; STREAM-ERROR message -- print message then abort execution
- ; replace this with a call to the native error handler
- ; if stream-error returns, so will the stream library function that called it
- (define stream-error error)
- ;;; STREAM SYNTAX AND FUNCTIONS
- ;; STREAM-NULL -- the distinguished nil stream
- (define stream-null (make-stream (delay '())))
- ;; STREAM-CONS object stream -- primitive constructor of streams
- (define-syntax stream-cons
- (syntax-rules ()
- ((stream-cons obj strm)
- (make-stream
- (delay
- (if (not (stream? strm))
- (stream-error "attempt to stream-cons onto non-stream")
- (cons obj strm)))))))
- ;; STREAM-NULL? object -- #t if object is the null stream, #f otherwise
- (define (stream-null? obj)
- (and (stream? obj) (null? (force (stream-promise obj)))))
- ;; STREAM-PAIR? object -- #t if object is a non-null stream, #f otherwise
- (define (stream-pair? obj)
- (and (stream? obj) (not (null? (force (stream-promise obj))))))
- ;; STREAM-CAR stream -- first element of stream
- (define (stream-car strm)
- (cond ((not (stream? strm)) (stream-error "attempt to take stream-car of non-stream"))
- ((stream-null? strm) (stream-error "attempt to take stream-car of null stream"))
- (else (car (force (stream-promise strm))))))
- ;; STREAM-CDR stream -- remaining elements of stream after first
- (define (stream-cdr strm)
- (cond ((not (stream? strm)) (stream-error "attempt to take stream-cdr of non-stream"))
- ((stream-null? strm) (stream-error "attempt to take stream-cdr of null stream"))
- (else (cdr (force (stream-promise strm))))))
- ;; STREAM-DELAY object -- the essential stream mechanism
- (define-syntax stream-delay
- (syntax-rules ()
- ((stream-delay expr)
- (make-stream
- (lazy expr)))))
- ;; STREAM object ... -- new stream whose elements are object ...
- (define (stream . objs)
- (let loop ((objs objs))
- (stream-delay
- (if (null? objs)
- stream-null
- (stream-cons (car objs) (loop (cdr objs)))))))
- ;; STREAM-UNFOLDN generator seed n -- n+1 streams from (generator seed)
- (define (stream-unfoldn gen seed n)
- (define (unfold-result-stream gen seed)
- (let loop ((seed seed))
- (stream-delay
- (call-with-values
- (lambda () (gen seed))
- (lambda (next . results)
- (stream-cons results (loop next)))))))
- (define (result-stream->output-stream result-stream i)
- (stream-delay
- (let ((result (list-ref (stream-car result-stream) i)))
- (cond ((pair? result)
- (stream-cons (car result)
- (result-stream->output-stream
- (stream-cdr result-stream) i)))
- ((not result)
- (result-stream->output-stream (stream-cdr result-stream) i))
- ((null? result) stream-null)
- (else (stream-error "can't happen"))))))
- (define (result-stream->output-streams result-stream n)
- (let loop ((i 0) (outputs '()))
- (if (= i n)
- (apply values (reverse outputs))
- (loop (+ i 1)
- (cons (result-stream->output-stream result-stream i)
- outputs)))))
- (result-stream->output-streams (unfold-result-stream gen seed) n))
- ;; STREAM-MAP func stream ... -- stream produced by applying func element-wise
- (define (stream-map func . strms)
- (cond ((not (procedure? func)) (stream-error "non-functional argument to stream-map"))
- ((null? strms) (stream-error "no stream arguments to stream-map"))
- ((not (every stream? strms)) (stream-error "non-stream argument to stream-map"))
- (else (let loop ((strms strms))
- (stream-delay
- (if (any stream-null? strms)
- stream-null
- (stream-cons (apply func (map stream-car strms))
- (loop (map stream-cdr strms)))))))))
- ;; STREAM-FOR-EACH proc stream ... -- apply proc element-wise for side-effects
- (define (stream-for-each proc . strms)
- (cond ((not (procedure? proc)) (stream-error "non-functional argument to stream-for-each"))
- ((null? strms) (stream-error "no stream arguments to stream-for-each"))
- ((not (every stream? strms)) (stream-error "non-stream argument to stream-for-each"))
- (else (let loop ((strms strms))
- (if (not (any stream-null? strms))
- (begin (apply proc (map stream-car strms))
- (loop (map stream-cdr strms))))))))
-
- ;; STREAM-FILTER pred? stream -- new stream including only items passing pred?
- (define (stream-filter pred? strm)
- (cond ((not (procedure? pred?)) (stream-error "non-functional argument to stream-filter"))
- ((not (stream? strm)) (stream-error "attempt to apply stream-filter to non-stream"))
- (else (stream-unfoldn
- (lambda (s)
- (values
- (stream-cdr s)
- (cond ((stream-null? s) '())
- ((pred? (stream-car s)) (list (stream-car s)))
- (else #f))))
- strm
- 1))))
|