;;; Ports ;;; Copyright (C) 2016,2019,2021 Free Software Foundation, Inc. ;;; ;;; This library is free software: you can redistribute it and/or modify ;;; it under the terms of the GNU Lesser General Public License as ;;; published by the Free Software Foundation, either version 3 of the ;;; License, or (at your option) any later version. ;;; ;;; This library 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 ;;; Lesser General Public License for more details. ;;; ;;; You should have received a copy of the GNU Lesser General Public ;;; License along with this program. If not, see ;;; . ;;; Commentary: ;;; ;;; Implementation of input/output routines over ports. ;;; ;;; Note that loading this module overrides some core bindings; see the ;;; `replace-bootstrap-bindings' invocation below for details. ;;; ;;; Code: (define-module (ice-9 ports) #:export (;; Definitions from ports.c. %port-property %set-port-property! current-input-port current-output-port current-error-port current-warning-port current-load-port set-current-input-port set-current-output-port set-current-error-port port-mode port? input-port? output-port? port-closed? eof-object? close-port close-input-port close-output-port ;; These two are currently defined by scm_init_ports; fix? ;; %default-port-encoding ;; %default-port-conversion-strategy port-encoding set-port-encoding! port-conversion-strategy set-port-conversion-strategy! read-char peek-char unread-char unread-string setvbuf drain-input force-output char-ready? seek SEEK_SET SEEK_CUR SEEK_END truncate-file port-line set-port-line! port-column set-port-column! port-filename set-port-filename! port-for-each flush-all-ports %make-void-port ;; Definitions from fports.c. open-file file-port? port-revealed set-port-revealed! adjust-port-revealed! ;; note: %file-port-name-canonicalization is used in boot-9 ;; Definitions from ioext.c. ftell redirect-port dup->fdes dup2 fileno isatty? fdopen primitive-move->fdes fdes->ports ;; Definitions in Scheme file-position file-set-position move->fdes release-port-handle dup->port dup->inport dup->outport dup duplicate-port fdes->inport fdes->outport port->fdes OPEN_READ OPEN_WRITE OPEN_BOTH *null-device* open-input-file open-output-file open-io-file call-with-port call-with-input-file call-with-output-file with-input-from-port with-output-to-port with-error-to-port with-input-from-file with-output-to-file with-error-to-file call-with-input-string with-input-from-string call-with-output-string with-output-to-string with-error-to-string the-eof-object inherit-print-state)) (define (replace-bootstrap-bindings syms) (for-each (lambda (sym) (let* ((var (module-variable the-scm-module sym)) (mod (current-module)) (iface (module-public-interface mod))) (unless var (error "unbound in root module" sym)) (module-add! mod sym var) (when (module-local-variable iface sym) (module-add! iface sym var)))) syms)) (replace-bootstrap-bindings '(open-file open-input-file set-port-encoding! eof-object? force-output call-with-output-string close-port current-error-port current-warning-port)) (load-extension (string-append "libguile-" (effective-version)) "scm_init_ice_9_ports") (load-extension (string-append "libguile-" (effective-version)) "scm_init_ice_9_fports") (load-extension (string-append "libguile-" (effective-version)) "scm_init_ice_9_ioext") (define (port-encoding port) "Return, as a string, the character encoding that @var{port} uses to interpret its input and output." (symbol->string (%port-encoding port))) (define-module (ice-9 ports internal) #:use-module (ice-9 ports) #:export (port-read-buffer port-write-buffer port-auxiliary-write-buffer port-line-buffered? expand-port-read-buffer! port-buffer-bytevector port-buffer-cur port-buffer-end port-buffer-has-eof? port-buffer-position set-port-buffer-cur! set-port-buffer-end! set-port-buffer-has-eof?! port-position-line port-position-column set-port-position-line! set-port-position-column! port-read port-write port-clear-stream-start-for-bom-read port-clear-stream-start-for-bom-write %port-encoding specialize-port-encoding! port-random-access? port-decode-char port-encode-char port-encode-chars port-read-buffering port-poll port-read-wait-fd port-write-wait-fd put-char put-string)) (define-syntax-rule (port-buffer-bytevector buf) (vector-ref buf 0)) (define-syntax-rule (port-buffer-cur buf) (vector-ref buf 1)) (define-syntax-rule (port-buffer-end buf) (vector-ref buf 2)) (define-syntax-rule (port-buffer-has-eof? buf) (vector-ref buf 3)) (define-syntax-rule (port-buffer-position buf) (vector-ref buf 4)) (define-syntax-rule (set-port-buffer-cur! buf cur) (vector-set! buf 1 cur)) (define-syntax-rule (set-port-buffer-end! buf end) (vector-set! buf 2 end)) (define-syntax-rule (set-port-buffer-has-eof?! buf has-eof?) (vector-set! buf 3 has-eof?)) (define-syntax-rule (port-position-line position) (car position)) (define-syntax-rule (port-position-column position) (cdr position)) (define-syntax-rule (set-port-position-line! position line) (set-car! position line)) (define-syntax-rule (set-port-position-column! position column) (set-cdr! position column)) (eval-when (expand) (define-syntax-rule (private-port-bindings binding ...) (begin (define binding (@@ (ice-9 ports) binding)) ...))) (private-port-bindings port-read-buffer port-write-buffer port-auxiliary-write-buffer port-line-buffered? expand-port-read-buffer! port-read port-write port-clear-stream-start-for-bom-read port-clear-stream-start-for-bom-write %port-encoding specialize-port-encoding! port-decode-char port-encode-char port-encode-chars port-random-access? port-read-buffering port-poll port-read-wait-fd port-write-wait-fd put-char put-string) ;; And we're back. (define-module (ice-9 ports)) ;;; Current ports as parameters. ;;; (define current-input-port (fluid->parameter %current-input-port-fluid (lambda (x) (unless (input-port? x) (error "expected an input port" x)) x))) (define current-output-port (fluid->parameter %current-output-port-fluid (lambda (x) (unless (output-port? x) (error "expected an output port" x)) x))) (define current-error-port (fluid->parameter %current-error-port-fluid (lambda (x) (unless (output-port? x) (error "expected an output port" x)) x))) (define current-warning-port (fluid->parameter %current-warning-port-fluid (lambda (x) (unless (output-port? x) (error "expected an output port" x)) x))) ;;; {File Descriptors and Ports} ;;; (define file-position ftell) (define* (file-set-position port offset #:optional (whence SEEK_SET)) (seek port offset whence)) (define (move->fdes fd/port fd) (cond ((integer? fd/port) (dup->fdes fd/port fd) (close fd/port) fd) (else (primitive-move->fdes fd/port fd) (set-port-revealed! fd/port 1) fd/port))) (define (release-port-handle port) (let ((revealed (port-revealed port))) (if (> revealed 0) (set-port-revealed! port (- revealed 1))))) (define dup->port (case-lambda ((port/fd mode) (fdopen (dup->fdes port/fd) mode)) ((port/fd mode new-fd) (let ((port (fdopen (dup->fdes port/fd new-fd) mode))) (set-port-revealed! port 1) port)))) (define dup->inport (case-lambda ((port/fd) (dup->port port/fd "r")) ((port/fd new-fd) (dup->port port/fd "r" new-fd)))) (define dup->outport (case-lambda ((port/fd) (dup->port port/fd "w")) ((port/fd new-fd) (dup->port port/fd "w" new-fd)))) (define dup (case-lambda ((port/fd) (if (integer? port/fd) (dup->fdes port/fd) (dup->port port/fd (port-mode port/fd)))) ((port/fd new-fd) (if (integer? port/fd) (dup->fdes port/fd new-fd) (dup->port port/fd (port-mode port/fd) new-fd))))) (define (duplicate-port port modes) (dup->port port modes)) (define (fdes->inport fdes) (let loop ((rest-ports (fdes->ports fdes))) (cond ((null? rest-ports) (let ((result (fdopen fdes "r"))) (set-port-revealed! result 1) result)) ((input-port? (car rest-ports)) (set-port-revealed! (car rest-ports) (+ (port-revealed (car rest-ports)) 1)) (car rest-ports)) (else (loop (cdr rest-ports)))))) (define (fdes->outport fdes) (let loop ((rest-ports (fdes->ports fdes))) (cond ((null? rest-ports) (let ((result (fdopen fdes "w"))) (set-port-revealed! result 1) result)) ((output-port? (car rest-ports)) (set-port-revealed! (car rest-ports) (+ (port-revealed (car rest-ports)) 1)) (car rest-ports)) (else (loop (cdr rest-ports)))))) (define (port->fdes port) (set-port-revealed! port (+ (port-revealed port) 1)) (fileno port)) ;; Legacy interfaces. (define (set-current-input-port port) "Set the current default input port to @var{port}." (current-input-port port)) (define (set-current-output-port port) "Set the current default output port to @var{port}." (current-output-port port)) (define (set-current-error-port port) "Set the current default error port to @var{port}." (current-error-port port)) ;;;; high level routines ;;; {High-Level Port Routines} ;;; ;; These are used to request the proper mode to open files in. ;; (define OPEN_READ "r") (define OPEN_WRITE "w") (define OPEN_BOTH "r+") (define *null-device* "/dev/null") (define* (open-input-file file #:key (binary #f) (encoding #f) (guess-encoding #f)) "Takes a string naming an existing file and returns an input port capable of delivering characters from the file. If the file cannot be opened, an error is signalled." (open-file file (if binary "rb" "r") #:encoding encoding #:guess-encoding guess-encoding)) (define* (open-output-file file #:key (binary #f) (encoding #f)) "Takes a string naming an output file to be created and returns an output port capable of writing characters to a new file by that name. If the file cannot be opened, an error is signalled. If a file with the given name already exists, the effect is unspecified." (open-file file (if binary "wb" "w") #:encoding encoding)) (define (open-io-file str) "Open file with name STR for both input and output." (open-file str OPEN_BOTH)) (define (call-with-port port proc) "Call @var{proc}, passing it @var{port} and closing @var{port} upon exit of @var{proc}. Return the return values of @var{proc}." (call-with-values (lambda () (proc port)) (lambda vals (close-port port) (apply values vals)))) (define* (call-with-input-file file proc #:key (binary #f) (encoding #f) (guess-encoding #f)) "PROC should be a procedure of one argument, and FILE should be a string naming a file. The file must already exist. These procedures call PROC with one argument: the port obtained by opening the named file for input or output. If the file cannot be opened, an error is signalled. If the procedure returns, then the port is closed automatically and the values yielded by the procedure are returned. If the procedure does not return, then the port will not be closed automatically unless it is possible to prove that the port will never again be used for a read or write operation." (let ((p (open-input-file file #:binary binary #:encoding encoding #:guess-encoding guess-encoding))) (call-with-port p proc))) (define* (call-with-output-file file proc #:key (binary #f) (encoding #f)) "PROC should be a procedure of one argument, and FILE should be a string naming a file. The behaviour is unspecified if the file already exists. These procedures call PROC with one argument: the port obtained by opening the named file for input or output. If the file cannot be opened, an error is signalled. If the procedure returns, then the port is closed automatically and the values yielded by the procedure are returned. If the procedure does not return, then the port will not be closed automatically unless it is possible to prove that the port will never again be used for a read or write operation." (let ((p (open-output-file file #:binary binary #:encoding encoding))) (call-with-port p proc))) (define (with-input-from-port port thunk) (parameterize ((current-input-port port)) (thunk))) (define (with-output-to-port port thunk) (parameterize ((current-output-port port)) (thunk))) (define (with-error-to-port port thunk) (parameterize ((current-error-port port)) (thunk))) (define* (with-input-from-file file thunk #:key (binary #f) (encoding #f) (guess-encoding #f)) "THUNK must be a procedure of no arguments, and FILE must be a string naming a file. The file must already exist. The file is opened for input, an input port connected to it is made the default value returned by `current-input-port', and the THUNK is called with no arguments. When the THUNK returns, the port is closed and the previous default is restored. Returns the values yielded by THUNK. If an escape procedure is used to escape from the continuation of these procedures, their behavior is implementation dependent." (call-with-input-file file (lambda (p) (with-input-from-port p thunk)) #:binary binary #:encoding encoding #:guess-encoding guess-encoding)) (define* (with-output-to-file file thunk #:key (binary #f) (encoding #f)) "THUNK must be a procedure of no arguments, and FILE must be a string naming a file. The effect is unspecified if the file already exists. The file is opened for output, an output port connected to it is made the default value returned by `current-output-port', and the THUNK is called with no arguments. When the THUNK returns, the port is closed and the previous default is restored. Returns the values yielded by THUNK. If an escape procedure is used to escape from the continuation of these procedures, their behavior is implementation dependent." (call-with-output-file file (lambda (p) (with-output-to-port p thunk)) #:binary binary #:encoding encoding)) (define* (with-error-to-file file thunk #:key (binary #f) (encoding #f)) "THUNK must be a procedure of no arguments, and FILE must be a string naming a file. The effect is unspecified if the file already exists. The file is opened for output, an output port connected to it is made the default value returned by `current-error-port', and the THUNK is called with no arguments. When the THUNK returns, the port is closed and the previous default is restored. Returns the values yielded by THUNK. If an escape procedure is used to escape from the continuation of these procedures, their behavior is implementation dependent." (call-with-output-file file (lambda (p) (with-error-to-port p thunk)) #:binary binary #:encoding encoding)) (define (call-with-input-string string proc) "Call the one-argument procedure @var{proc} with a newly created input port from which @var{string}'s contents may be read. All values yielded by the @var{proc} are returned." (proc (open-input-string string))) (define (with-input-from-string string thunk) "THUNK must be a procedure of no arguments. The test of STRING is opened for input, an input port connected to it is made, and the THUNK is called with no arguments. When the THUNK returns, the port is closed. Returns the values yielded by THUNK. If an escape procedure is used to escape from the continuation of these procedures, their behavior is implementation dependent." (call-with-input-string string (lambda (p) (with-input-from-port p thunk)))) (define (call-with-output-string proc) "Call the one-argument procedure @var{proc} with a newly created output port. When the function returns, port is closed and the string composed of the characters written into the port is returned." (let ((port (open-output-string))) (proc port) (let ((res (get-output-string port))) (close-port port) res))) (define (with-output-to-string thunk) "Calls THUNK and returns its output as a string." (call-with-output-string (lambda (p) (with-output-to-port p thunk)))) (define (with-error-to-string thunk) "Calls THUNK and returns its error output as a string." (call-with-output-string (lambda (p) (with-error-to-port p thunk)))) (define (inherit-print-state old-port new-port) (if (get-print-state old-port) (port-with-print-state new-port (get-print-state old-port)) new-port))