ikarus/scheme/ikarus.io.input-strings.ss

107 lines
3.6 KiB
Scheme

;;; Ikarus Scheme -- A compiler for R6RS Scheme.
;;; Copyright (C) 2006,2007 Abdulaziz Ghuloum
;;;
;;; This program is free software: you can redistribute it and/or modify
;;; it under the terms of the GNU General Public License version 3 as
;;; published by the Free Software Foundation.
;;;
;;; This program is distributed in the hope that it will be useful, but
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
;;; General Public License for more details.
;;;
;;; You should have received a copy of the GNU General Public License
;;; along with this program. If not, see <http://www.gnu.org/licenses/>.
(library (ikarus io input-strings)
(export open-string-input-port)
(import
(ikarus system $strings)
(ikarus system $bytevectors)
(ikarus system $fx)
(ikarus system $pairs)
(ikarus system $ports)
(ikarus system $io)
(except (ikarus) open-string-input-port ))
(define-syntax message-case
(syntax-rules (else)
[(_ msg args
[(msg-name msg-arg* ...) b b* ...] ...
[else else1 else2 ...])
(let ([tmsg msg] [targs args])
(define-syntax match-and-bind
(syntax-rules ()
[(__ y () body)
(if (null? y)
body
(error 'message-case "unmatched" (cons tmsg targs)))]
[(__ y (a a* (... ...)) body)
(if (pair? y)
(let ([a (car y)] [d (cdr y)])
(match-and-bind d (a* (... ...)) body))
(error 'message-case "unmatched" (cons tmsg targs)))]))
(case tmsg
[(msg-name)
(match-and-bind targs (msg-arg* ...) (begin b b* ...))] ...
[else else1 else2 ...]))]))
(define make-input-string-handler
(lambda (str)
(let ((open? #t) (idx 0) (n (string-length str)))
(lambda (msg . args)
(message-case msg args
[(read-char p)
(if ($fx< idx n)
(let ([c ($string-ref str idx)])
(set! idx ($fxadd1 idx))
c)
(if open?
(eof-object)
(error 'read-char "port is closed" p)))]
[(peek-char p)
(if ($fx< idx n)
($string-ref str idx)
(if open?
(eof-object)
(error 'peek-char "port is closed" p)))]
[(port-name p) '*string-port*]
[(close-port p)
(when open?
(set! open? #f))]
[else
(error 'input-string-handler
"message not handled" (cons msg args))])))))
(define ($open-input-string str)
(let ([port (make-input-port
(make-input-string-handler str)
'#vu8())])
port))
;(define open-input-string
; (lambda (str)
; (unless (string? str)
; (error 'open-input-string "not a string" str))
; ($open-input-string str)))
(define open-string-input-port
(lambda (str)
(unless (string? str)
(error 'open-string-input-port "not a string" str))
($open-input-string str)))
;(define with-input-from-string
; (lambda (str proc)
; (unless (string? str)
; (error 'with-input-from-string "not a string" str))
; (unless (procedure? proc)
; (error 'with-input-from-string "not a procedure" proc))
; (let ([p ($open-input-string str)])
; (parameterize ([current-input-port p])
; (proc)))))
)