home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
OS/2 Shareware BBS: 10 Tools
/
10-Tools.zip
/
mitsch75.zip
/
scheme-7_5_17-src.zip
/
scheme-7.5.17
/
src
/
runtime
/
udata.scm
< prev
next >
Wrap
Text File
|
1999-03-24
|
11KB
|
296 lines
#| -*-Scheme-*-
$Id: udata.scm,v 14.20 1999/03/25 03:44:20 cph Exp $
Copyright (c) 1988-1999 Massachusetts Institute of Technology
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at
your option) any later version.
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, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|#
;;;; Simple Microcode Data Structures
;;; package: ()
(declare (usual-integrations))
(define (return-address? object)
(or (interpreter-return-address? object)
(compiled-return-address? object)))
(define-integrable (interpreter-return-address? object)
(object-type? (ucode-type return-address) object))
(define-integrable (make-return-address code)
((ucode-primitive map-code-to-machine-address 2) (ucode-type return-address)
code))
(define-integrable (return-address/code return-address)
((ucode-primitive map-machine-address-to-code 2) (ucode-type return-address)
return-address))
(define (return-address/name return-address)
(microcode-return/code->name (return-address/code return-address)))
(define (microcode-error name)
(or (microcode-error/name->code name)
(error "MICROCODE-ERROR: Unknown name" name)))
(define (microcode-return name)
(or (microcode-return/name->code name)
(error "MICROCODE-RETURN: Unknown name" name)))
(define (microcode-termination name)
(or (microcode-termination/name->code name)
(error "MICROCODE-TERMINATION: Unknown name" name)))
(define (microcode-type name)
(or (microcode-type/name->code name)
(error "MICROCODE-TYPE: Unknown name" name)))
;;;; Compiled Code Entries
(define-integrable (compiled-code-address? object)
(object-type? (ucode-type compiled-entry) object))
(define-integrable (stack-address? object)
(object-type? (ucode-type stack-environment) object))
(define (compiled-expression? object)
(and (compiled-code-address? object)
(eq? (compiled-entry-type object) 'COMPILED-EXPRESSION)))
(define (compiled-return-address? object)
(and (compiled-code-address? object)
(eq? (compiled-entry-type object) 'COMPILED-RETURN-ADDRESS)))
(define-primitives
(stack-address-offset 1)
(compiled-code-address->block 1)
(compiled-code-address->offset 1))
(define (discriminate-compiled-entry entry
if-procedure
if-return-address
if-expression
if-other)
(case (system-hunk3-cxr0 ((ucode-primitive compiled-entry-kind 1) entry))
((0) (if-procedure))
((1) (if-return-address))
((2) (if-expression))
(else (if-other))))
(define (compiled-entry-type entry)
(case (system-hunk3-cxr0 ((ucode-primitive compiled-entry-kind 1) entry))
((0) 'COMPILED-PROCEDURE)
((1) 'COMPILED-RETURN-ADDRESS)
((2) 'COMPILED-EXPRESSION)
(else 'COMPILED-ENTRY)))
(define (compiled-continuation/next-continuation-offset entry)
(let ((offset
(system-hunk3-cxr2 ((ucode-primitive compiled-entry-kind 1) entry))))
(and (not (negative? offset))
offset)))
(define (compiled-continuation/return-to-interpreter? entry)
(let ((kind ((ucode-primitive compiled-entry-kind 1) entry)))
(and (fix:= (system-hunk3-cxr1 kind) 2)
(fix:= (system-hunk3-cxr2 kind) 0))))
(define (compiled-continuation/reflect-to-interface? entry)
(let ((kind ((ucode-primitive compiled-entry-kind 1) entry)))
(and (fix:= (system-hunk3-cxr1 kind) 2)
(not (fix:= (system-hunk3-cxr2 kind) 0)))))
(define (stack-address->index address start-offset)
(if (not (stack-address? address))
(error "Not a stack address" address))
(let ((index (- start-offset (stack-address-offset address))))
(if (negative? index)
(error "Stack address out of range" address start-offset))
index))
;;;; Compiled Code Blocks
#|
Compiled code blocks contain both nonmarked code and marked constants.
Code positions are referred to as OFFSETS, which start from the
beginning of the block and are measured in bytes. The positions of
constants are referred to as INDICES, and use the normal index
numbering for vectors. The conversion between offsets and indices is
specified by COMPILED-CODE-BLOCK/BYTES-PER-OBJECT, which should be set
to the correct value before these operations are used.
Note: This code needs to be changed somewhat. MANIFEST-CLOSURES are
compiled-code-blocks, but the format of them is completely different.
The constants block in a compiled-code-block often has a linkage
section that you cannot just vector-ref into as it contains raw
amchine addresses. COMPILED-CODE-BLOCK/MARKED-START returns the start
index of this area. COMPILED-CODE-BLOCK/CONSTANTS-START returns the
start index of the area following the linkage section, which usually
contains constants derived from the source program.
|#
(define compiled-code-block/bytes-per-object)
(define-integrable (compiled-code-block? object)
(object-type? (ucode-type compiled-code-block) object))
(define-integrable (compiled-code-block/read-file filename)
(compiled-code-address->block (fasload filename)))
(define (compiled-code-block/manifest-closure? block)
(object-type?
(ucode-type manifest-closure)
;; This combination returns an unsafe object, but since it
;; is used as an argument to a primitive, I can get away
;; with not turning off the garbage collector.
((ucode-primitive primitive-object-ref 2) block 0)))
(define (compiled-code-block/index->offset index)
(* (1+ index) compiled-code-block/bytes-per-object))
(define (compiled-code-block/offset->index offset)
(-1+ (quotient offset compiled-code-block/bytes-per-object)))
(define (compiled-code-block/code-length block)
(* compiled-code-block/bytes-per-object
(object-datum (system-vector-ref block 0))))
(define (compiled-code-block/code-start block)
block
(* compiled-code-block/bytes-per-object 2))
(define (compiled-code-block/code-end block)
(+ (compiled-code-block/code-start block)
(compiled-code-block/code-length block)))
(define (compiled-code-block/marked-start block)
;; The first offset that is a marked constant
(1+ (object-datum (system-vector-ref block 0))))
(define (compiled-code-block/constants-start block)
;; Skip over linkage sections and manifect vector templates to find an
;; index that can be used to extract constants.
(let ((marked-start (compiled-code-block/marked-start block))
(end (compiled-code-block/constants-end block)))
(let loop ((index marked-start))
(if (>= index end)
end
(let ((type (object-type (system-vector-ref block index)))
(datum (object-datum (system-vector-ref block index))))
(cond ((= type (ucode-type manifest-closure))
(loop (+ index 1 4)))
((or (= type (ucode-type linkage-section)) ;; linked or..
(= type (ucode-type positive-fixnum))) ;; before linking
;; [Before linking the execute caches are headed by fixnums
;; and contain symbols and fixnums]
(let ((kind (quotient datum #x10000))
(count (remainder datum #x10000)))
kind
(loop (+ index 1 count))))
(else
index)))))))
(define (compiled-code-block/constants-end block)
(- (system-vector-length block) 2))
(define (compiled-code-block/debugging-info? block)
(not (memq (compiled-code-block/debugging-info block) '(#F DEBUGGING-INFO))))
(define (compiled-code-block/debugging-info block)
(system-vector-ref block (- (system-vector-length block) 2)))
(define (set-compiled-code-block/debugging-info! block info)
(system-vector-set! block (- (system-vector-length block) 2) info))
(define (compiled-code-block/environment block)
(system-vector-ref block (-1+ (system-vector-length block))))
;;;; Environment Extensions
(define-integrable (environment-extension? object)
(vector? object))
(define-integrable (environment-extension-parent extension)
(vector-ref extension 0))
(define-integrable (set-environment-extension-parent! extension parent)
(vector-set! extension 0 parent))
(define-integrable (environment-extension-procedure extension)
(vector-ref extension 1))
(define (environment-extension-aux-list extension)
(let filter-potentially-dangerous
((aux-list
(let ((first-aux-slot 3))
(subvector->list
extension
first-aux-slot
(+ first-aux-slot (object-datum (vector-ref extension 2)))))))
(cond ((null? aux-list) '())
((unbound-reference-trap?
(map-reference-trap (lambda () (cdar aux-list))))
(filter-potentially-dangerous (cdr aux-list)))
(else
(cons (car aux-list)
(filter-potentially-dangerous (cdr aux-list)))))))
;;;; Promises
(define-integrable (promise? object)
(object-type? (ucode-type delayed) object))
(define-integrable (promise-forced? promise)
(eq? true (system-pair-car promise)))
(define-integrable (promise-non-expression? promise)
(eqv? 0 (system-pair-car promise)))
(define (promise-value promise)
(if (not (promise-forced? promise))
(error "Promise not yet forced" promise))
(system-pair-cdr promise))
(define (promise-expression promise)
(if (promise-forced? promise)
(error "Promise already forced" promise))
(if (promise-non-expression? promise)
(error "Promise has no expression" promise))
(system-pair-cdr promise))
(define (promise-environment promise)
(if (promise-forced? promise)
(error "Promise already forced" promise))
(if (promise-non-expression? promise)
(error "Promise has no environment" promise))
(system-pair-car promise))
(define (force promise)
(cond ((not (promise? promise))
(error:wrong-type-argument promise "promise" 'FORCE))
((eq? #T (system-pair-car promise))
(system-pair-cdr promise))
((eqv? 0 (system-pair-car promise)) ; compiled promise
(let ((result ((system-pair-cdr promise))))
(system-pair-set-cdr! promise result)
(system-pair-set-car! promise #T)
result))
(else ; losing old style
((ucode-primitive force 1) promise))))