;;; (in-package "SB!VM") ;;;; Define the registers (eval-when (:compile-toplevel :load-toplevel :execute) (defvar *register-names* (make-array 32 :initial-element nil))) (macrolet ((defreg (name offset) (let ((offset-sym (symbolicate name "-OFFSET"))) `(eval-when (:compile-toplevel :load-toplevel :execute) (def!constant ,offset-sym ,offset) (setf (svref *register-names* ,offset-sym) ,(symbol-name name))))) (defregset (name &rest regs) `(eval-when (:compile-toplevel :load-toplevel :execute) (defparameter ,name (list ,@(mapcar #'(lambda (name) (symbolicate name "-OFFSET")) regs)))))) (defreg zero 0) (defreg nsp 1) (defreg rtoc 2) ; May be "NULL" someday. (defreg nl0 3) (defreg nl1 4) (defreg nl2 5) (defreg nl3 6) (defreg nl4 7) (defreg nl5 8) (defreg nl6 9) (defreg fdefn 10) ; was nl7 (defreg nargs 11) (defreg nfp 12) (defreg cfunc 13) (defreg bsp 14) (defreg cfp 15) (defreg csp 16) (defreg alloc 17) (defreg null 18) (defreg code 19) (defreg cname 20) (defreg lexenv 21) (defreg ocfp 22) (defreg lra 23) (defreg a0 24) (defreg a1 25) (defreg a2 26) (defreg a3 27) (defreg l0 28) (defreg l1 29) (defreg l2 30) (defreg lip 31) (defregset non-descriptor-regs nl0 nl1 nl2 nl3 nl4 nl5 nl6 #+nil nl7 cfunc nargs nfp) (defregset descriptor-regs fdefn a0 a1 a2 a3 ocfp lra cname lexenv l0 l1 l2 ) (defregset *register-arg-offsets* a0 a1 a2 a3) (defparameter register-arg-names '(a0 a1 a2 a3))) ;;;; SB and SC definition: (define-storage-base registers :finite :size 32) (define-storage-base float-registers :finite :size 32) (define-storage-base control-stack :unbounded :size 8) (define-storage-base non-descriptor-stack :unbounded :size 0) (define-storage-base constant :non-packed) (define-storage-base immediate-constant :non-packed) ;;; ;;; Handy macro so we don't have to keep changing all the numbers whenever ;;; we insert a new storage class. ;;; (defmacro define-storage-classes (&rest classes) (do ((forms (list 'progn) (let* ((class (car classes)) (sc-name (car class)) (constant-name (intern (concatenate 'simple-string (string sc-name) "-SC-NUMBER")))) (list* `(define-storage-class ,sc-name ,index ,@(cdr class)) `(def!constant ,constant-name ,index) forms))) (index 0 (1+ index)) (classes classes (cdr classes))) ((null classes) (nreverse forms)))) ;; XXX this is most likely wrong. Check with Eric Marsden next time you ;; see him (def!constant sb!vm::kludge-nondeterministic-catch-block-size 7) (define-storage-classes ;; Non-immediate contstants in the constant pool (constant constant) ;; ZERO and NULL are in registers. (zero immediate-constant) (null immediate-constant) ;; Anything else that can be an immediate. (immediate immediate-constant) ;; **** The stacks. ;; The control stack. (Scanned by GC) (control-stack control-stack) ;; The non-descriptor stacks. (signed-stack non-descriptor-stack) ; (signed-byte 32) (unsigned-stack non-descriptor-stack) ; (unsigned-byte 32) (base-char-stack non-descriptor-stack) ; non-descriptor characters. (sap-stack non-descriptor-stack) ; System area pointers. (single-stack non-descriptor-stack) ; single-floats (double-stack non-descriptor-stack :element-size 2 :alignment 2) ; double floats. (complex-single-stack non-descriptor-stack :element-size 2) (complex-double-stack non-descriptor-stack :element-size 4 :alignment 2) ;; **** Things that can go in the integer registers. ;; Immediate descriptor objects. Don't have to be seen by GC, but nothing ;; bad will happen if they are. (fixnums, characters, header values, etc). (any-reg registers :locations #.(append non-descriptor-regs descriptor-regs) :constant-scs (zero immediate) :save-p t :alternate-scs (control-stack)) ;; Pointer descriptor objects. Must be seen by GC. (descriptor-reg registers :locations #.descriptor-regs :constant-scs (constant null immediate) :save-p t :alternate-scs (control-stack)) ;; Non-Descriptor characters (base-char-reg registers :locations #.non-descriptor-regs :constant-scs (immediate) :save-p t :alternate-scs (base-char-stack)) ;; Non-Descriptor SAP's (arbitrary pointers into address space) (sap-reg registers :locations #.non-descriptor-regs :constant-scs (immediate) :save-p t :alternate-scs (sap-stack)) ;; Non-Descriptor (signed or unsigned) numbers. (signed-reg registers :locations #.non-descriptor-regs :constant-scs (zero immediate) :save-p t :alternate-scs (signed-stack)) (unsigned-reg registers :locations #.non-descriptor-regs :constant-scs (zero immediate) :save-p t :alternate-scs (unsigned-stack)) ;; Random objects that must not be seen by GC. Used only as temporaries. (non-descriptor-reg registers :locations #.non-descriptor-regs) ;; Pointers to the interior of objects. Used only as a temporary. (interior-reg registers :locations (#.lip-offset)) ;; **** Things that can go in the floating point registers. ;; Non-Descriptor single-floats. (single-reg float-registers :locations #.(loop for i from 0 to 31 collect i) ;; ### Note: We really should have every location listed, but then we ;; would have to make load-tns work with element-sizes other than 1. :constant-scs () :save-p t :alternate-scs (single-stack)) ;; Non-Descriptor double-floats. (double-reg float-registers :locations #.(loop for i from 0 to 31 collect i) ;; ### Note: load-tns don't work with an element-size other than 1. ;; :element-size 2 :alignment 2 :constant-scs () :save-p t :alternate-scs (double-stack)) (complex-single-reg float-registers :locations #.(loop for i from 0 to 30 by 2 collect i) :element-size 2 :constant-scs () :save-p t :alternate-scs (complex-single-stack)) (complex-double-reg float-registers :locations #.(loop for i from 0 to 30 by 2 collect i) :element-size 2 :constant-scs () :save-p t :alternate-scs (complex-double-stack)) ;; A catch or unwind block. (catch-block control-stack :element-size sb!vm::kludge-nondeterministic-catch-block-size)) ;;;; Make some random tns for important registers. (macrolet ((defregtn (name sc) (let ((offset-sym (symbolicate name "-OFFSET")) (tn-sym (symbolicate name "-TN"))) `(defparameter ,tn-sym (make-random-tn :kind :normal :sc (sc-or-lose ',sc) :offset ,offset-sym))))) (defregtn zero any-reg) (defregtn lip interior-reg) (defregtn null descriptor-reg) (defregtn code descriptor-reg) (defregtn alloc any-reg) (defregtn nargs any-reg) (defregtn bsp any-reg) (defregtn csp any-reg) (defregtn cfp any-reg) (defregtn ocfp any-reg) (defregtn nsp any-reg)) ;;; Immediate-Constant-SC -- Interface ;;; ;;; If value can be represented as an immediate constant, then return the ;;; appropriate SC number, otherwise return NIL. ;;; (!def-vm-support-routine immediate-constant-sc (value) (typecase value ((integer 0 0) (sc-number-or-lose 'zero)) (null (sc-number-or-lose 'null)) ((or fixnum system-area-pointer character) (sc-number-or-lose 'immediate)) (symbol (if (static-symbol-p value) (sc-number-or-lose 'immediate) nil)))) ;;;; Function Call Parameters ;;; The SC numbers for register and stack arguments/return values. ;;; (def!constant register-arg-scn (meta-sc-number-or-lose 'descriptor-reg)) (def!constant immediate-arg-scn (meta-sc-number-or-lose 'any-reg)) (def!constant control-stack-arg-scn (meta-sc-number-or-lose 'control-stack)) (eval-when (:compile-toplevel :load-toplevel :execute) ;;; Offsets of special stack frame locations (def!constant ocfp-save-offset 0) (def!constant lra-save-offset 1) (def!constant nfp-save-offset 2) ;;; The number of arguments/return values passed in registers. ;;; (def!constant register-arg-count 4) ;;; Names to use for the argument registers. ;;; ); Eval-When (:compile-toplevel :load-toplevel :execute) ;;; A list of TN's describing the register arguments. ;;; (defparameter *register-arg-tns* (mapcar #'(lambda (n) (make-random-tn :kind :normal :sc (sc-or-lose 'descriptor-reg) :offset n)) *register-arg-offsets*)) (export 'single-value-return-byte-offset) ;;; SINGLE-VALUE-RETURN-BYTE-OFFSET ;;; ;;; This is used by the debugger. ;;; (def!constant single-value-return-byte-offset 8) ;;; LOCATION-PRINT-NAME -- Interface ;;; ;;; This function is called by debug output routines that want a pretty name ;;; for a TN's location. It returns a thing that can be printed with PRINC. ;;; (!def-vm-support-routine location-print-name (tn) (declare (type tn tn)) (let ((sb (sb-name (sc-sb (tn-sc tn)))) (offset (tn-offset tn))) (ecase sb (registers (or (svref *register-names* offset) (format nil "R~D" offset))) (float-registers (format nil "F~D" offset)) (control-stack (format nil "CS~D" offset)) (non-descriptor-stack (format nil "NS~D" offset)) (constant (format nil "Const~D" offset)) (immediate-constant "Immed")))) ;;; The loader uses this to convert alien names to the form they ;;; occur in the symbol table. This is ELF, so do nothing (defun extern-alien-name (name) (declare (type simple-base-string name)) name)