~ chicken-core (chicken-5) /read-syntax.scm


  1;;;; read-syntax.scm - CHICKEN reader extensions
  2;
  3; Copyright (c) 2008-2022, The CHICKEN Team
  4; Copyright (c) 2000-2007, Felix L. Winkelmann
  5; All rights reserved.
  6;
  7; Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following
  8; conditions are met:
  9;
 10;   Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 11;     disclaimer.
 12;   Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following
 13;     disclaimer in the documentation and/or other materials provided with the distribution.
 14;   Neither the name of the author nor the names of its contributors may be used to endorse or promote
 15;     products derived from this software without specific prior written permission.
 16;
 17; THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS
 18; OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
 19; AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR
 20; CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 21; CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 22; SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 23; THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
 24; OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 25; POSSIBILITY OF SUCH DAMAGE.
 26
 27(declare
 28  (unit read-syntax)
 29  (uses internal)
 30  (disable-interrupts))
 31
 32(module chicken.read-syntax
 33  (copy-read-table current-read-table define-reader-ctor set-read-syntax!
 34   set-sharp-read-syntax! set-parameterized-read-syntax!)
 35
 36(import scheme chicken.base chicken.internal chicken.platform)
 37
 38(include "common-declarations.scm")
 39
 40(define (set-read-mark! sym proc)
 41  (let ((a (assq sym ##sys#read-marks)))
 42    (if a
 43	(##sys#setslot a 1 proc)
 44	(set! ##sys#read-marks (cons (cons sym proc) ##sys#read-marks)))))
 45
 46(define current-read-table ##sys#current-read-table)
 47
 48(define ((syntax-setter loc slot wrap) chr proc)
 49  (if (symbol? chr)
 50      (set-read-mark! chr proc)
 51      (let ((crt (current-read-table)))
 52	(unless (##sys#slot crt slot)
 53	  (##sys#setslot crt slot (##sys#make-vector 256 #f)))
 54	(##sys#check-char chr loc)
 55	(let ((i (char->integer chr)))
 56	  (##sys#check-range i 0 256 loc)
 57	  (cond (proc
 58		 (##sys#check-closure proc loc)
 59		 (##sys#setslot (##sys#slot crt slot) i (wrap proc)))
 60		(else
 61		 (##sys#setslot (##sys#slot crt slot) i #f)))))))
 62
 63(define set-read-syntax!
 64  (syntax-setter
 65   'set-read-syntax! 1
 66   (lambda (proc)
 67     (lambda (_ port)
 68       (##sys#read-char-0 port)
 69       (proc port)))))
 70
 71(define set-sharp-read-syntax!
 72  (syntax-setter
 73   'set-sharp-read-syntax! 2
 74   (lambda (proc)
 75     (lambda (_ port)
 76       (##sys#read-char-0 port)
 77       (proc port)))))
 78
 79(define set-parameterized-read-syntax!
 80  (syntax-setter
 81   'set-parameterized-read-syntax! 3
 82   (lambda (proc)
 83     (lambda (_ port num)
 84       (##sys#read-char-0 port)
 85       (proc port num)))))
 86
 87;;; Read-table operations:
 88
 89(define (copy-read-table rt)
 90  (##sys#check-structure rt 'read-table 'copy-read-table)
 91  (##sys#make-structure
 92   'read-table
 93   (let ((t1 (##sys#slot rt 1)))
 94     (and t1 (##sys#vector-resize t1 (##sys#size t1) #f)))
 95   (let ((t2 (##sys#slot rt 2)))
 96     (and t2 (##sys#vector-resize t2 (##sys#size t2) #f)))
 97   (let ((t3 (##sys#slot rt 3)))
 98     (and t3 (##sys#vector-resize t3 (##sys#size t3) #f)))))
 99
100;;; SRFI-10:
101
102(define sharp-comma-reader-ctors (make-vector 301 '()))
103
104(define (define-reader-ctor spec proc)
105  (##sys#check-symbol spec 'define-reader-ctor)
106  (hash-table-set! sharp-comma-reader-ctors spec proc))
107
108(set! ##sys#user-read-hook
109  (let ((old ##sys#user-read-hook)
110	(read-char read-char)
111	(read read))
112    (lambda (char port)
113      (cond ((char=? char #\,)
114	     (read-char port)
115	     (let* ((exp (read port))
116		    (err (lambda () (##sys#read-error port "invalid sharp-comma external form" exp))))
117	       (if (or (null? exp) (not (list? exp)))
118		   (err)
119		   (let ([spec (##sys#slot exp 0)])
120		     (if (not (symbol? spec))
121			 (err)
122			 (let ((ctor (hash-table-ref sharp-comma-reader-ctors spec)))
123			   (if ctor
124			       (apply ctor (##sys#slot exp 1))
125			       (##sys#read-error port "undefined sharp-comma constructor" spec))))))))
126	    (else (old char port))))))
127
128(register-feature! 'srfi-10))
Trap