; Type system. ; This provides the low level classes for describing data, except for ; the actual type (confusingly enough) which is described in mode.scm. ; Copyright (C) 2000 Red Hat, Inc. ; This file is part of CGEN. ; See file COPYING.CGEN for details. ; Array type. ; DIMENSIONS has a suitable initial value so (new ) to works. (define (class-make ' nil '(mode (dimensions . ())) nil)) ; Return number of elements in array. (method-make! 'get-num-elms (lambda (self) (apply * (elm-get self 'dimensions))) ) ; Return mode of the array. (method-make! 'get-mode (lambda (self) (elm-get self 'mode))) ; Return the rank of the array (number of dimensions). (method-make! 'get-rank (lambda (self) (length (elm-get self 'dimensions))) ) ; Return shape of array (method-make! 'get-shape (lambda (self) (elm-get self 'dimensions)) ) ; Return #t if X is an array. (define (array? x) (class-instance? x)) ; Scalar type. (define (class-make ' '() nil nil)) (method-make-make! '(mode)) ; Return #t if X is a scalar. (define (scalar? x) (and (array? x) (= (send x 'get-rank) 0))) ; Return number of bits in an element of TYPE. (define (type-bits type) (mode:bits (send type 'get-mode)) ) ; Integers. ; These are like scalars but are specified in bits. ; BITS is the size in bits. ; ATTRS contains !UNSIGNED [or nothing] or UNSIGNED. ; ; A mode is needed so we know how big a field is needed to record the value. ; It might be more appropriate to use a host mode though. ; ; FIXME: Need to separate rank from type. scalar/array are not types. ; ;(define (class-make ' nil '(attrs bits) nil)) ; ;(method-make! 'get-atlist (lambda (self) (elm-get self 'attrs))) ; ;(method-make! ; 'get-mode ; (lambda (self) ; (mode-find (elm-get self 'bits) ; (if (has-attr? self 'UNSIGNED) 'UINT 'INT)) ; ) ;) ; ; FIXME: Quick hack. Revisit. ; ;(method-make! 'get-rank (lambda (self) 0)) ; Structures. ; FIXME: Unfinished. (define (class-make ' nil '(members) nil)) ; Parse a type spec. ; TYPE-SPEC is: (mode [(dimensions ...)]) ; or: ((mode bits) [(dimensions ...)]) (define (parse-type errtxt type-spec) ; Preliminary error checking. (if (and (list? (car type-spec)) (not (= (length (car type-spec)) 2))) (parse-error errtxt "invalid type spec" type-spec)) ; Pick out the arguments. (let ((mode (if (list? (car type-spec)) (caar type-spec) (car type-spec))) (bits (if (list? (car type-spec)) (cadar type-spec) #f)) (dims (if (> (length type-spec) 1) (cadr type-spec) nil))) ; FIXME: Need more error checking here. ; Validate the mode and bits. (let ((mode-obj (case mode ((INT) (if (integer? bits) (mode-make-int bits) (parse-error errtxt "invalid number of bits" bits))) ((UINT) (if (integer? bits) (mode-make-uint bits) (parse-error errtxt "invalid number of bits" bits))) ((BI QI HI SI DI WI UQI UHI USI UDI UWI SF DF XF TF) (let ((x (parse-mode-name mode errtxt))) (if (and bits (not (= bits (mode:bits x)))) (parse-error errtxt "wrong number of bits for mode" bits)) x)) (else (parse-error errtxt "unknown/unsupported mode" mode))))) ; Validate the dimension spec. (if (or (not (list? dims)) (not (all-true? (map integer? dims)))) (parse-error errtxt "invalid dimension spec" dims)) ; All done, create the object. ; ??? Special casing scalars is a concession for apps that think ; scalars aren't arrays. Not sure it should stay. (if (null? dims) (make mode-obj) (make mode-obj dims)))) ) ; Bit ranges. ; ??? Perhaps this should live in a different source file, but for now ; it's here. ; ; Endianness is not recorded with the bitrange. ; Values are operated on a "word" at a time. ; This is to handle bi-endian systems: we don't want two copies of ; every bitrange. ; ; Instruction word sizes are based on the "base insn length" which is the ; number of bytes the cpu first looks at to decode an insn. In cases where ; the total length is longer than the base insn length, the word length ; for the rest of the insn is the base insn length replicated as many times ; as necessary. The trailing part [last few bytes] of the insn may not fill ; the entire word, in which case the numbering is adjusted for it. ; ??? Might need to have an insn-base-length and an insn-word-length. ; ; Instructions that have words of one endianness and sub-words of a different ; endianness are handled at a higher level. ; ; Bit numbering examples: ; [each byte is represented MSB to LSB, low address to high address] ; ; lsb0? = #f ; insn-word-length = 2 ; endian = little ; | 8 ... 15 | 0 ... 7 | 24 ... 31 | 16 ... 23 | 40 ... 47 | 32 ... 39 | ; ; lsb0? = #t ; insn-word-length = 2 ; endian = little ; [note that this is the little endian canonical form (*) ; - word length is irrelevant] ; | 7 ... 0 | 15 ... 8 | 23 ... 16 | 31 ... 24 | 39 ... 32 | 47 ... 40 | ; ; lsb0? = #f ; insn-word-length = 2 ; endian = big ; [note that this is the big endian canonical form (*) ; - word length is irrelevant] ; | 0 ... 7 | 8 ... 15 | 16 ... 23 | 24 ... 31 | 32 ... 39 | 40 ... 47 | ; ; lsb0? = #t ; insn-word-length = 2 ; endian = big ; | 15 ... 8 | 7 ... 0 | 31 ... 24 | 23 ... 16 | 47 ... 40 | 39 ... 32 | ; ; (*) NOTE: This canonical form should not be confused with what might be ; called the canonical form when writing .cpu ifield descriptions: lsb0? = #f. ; The ifield canonical form is lsb0? = #f because the starting bit number of ; ifields is defined to be the MSB. ; --- ; At the bitrange level, insns with different sized words is supported. ; This is because each contains the specs of the word it resides in. ; For example a 48 bit insn with a 16 bit opcode and a 32 bit immediate value ; might [but not necessarily] consist of one 16 bit "word" and one 32 bit ; "word". ; ; Examples: ; ; lsb0? = #f ; insn-word-length = 2, 4 ; endian = little ; | 8 ... 15 | 0 ... 7 | 40 ... 47 | 32 ... 39 | 24 ... 31 | 16 ... 23 | ; ; lsb0? = #t ; insn-word-length = 2, 4 ; endian = little ; | 7 ... 0 | 15 ... 8 | 23 ... 16 | 31 ... 24 | 39 ... 32 | 47 ... 40 | ; ; lsb0? = #f ; insn-word-length = 2, 4 ; endian = big ; | 0 ... 7 | 8 ... 15 | 16 ... 23 | 24 ... 31 | 32 ... 39 | 40 ... 47 | ; ; lsb0? = #t ; insn-word-length = 2, 4 ; endian = big ; | 15 ... 8 | 7 ... 0 | 47 ... 40 | 39 ... 32 | 31 ... 24 | 23 ... 16 | (define (class-make ' nil '( ; offset in bits from the start of the insn of the word ; in which the value resides [must be divisible by 8] ; [this allows the bitrange to be independent of the lengths ; of words preceding this one] word-offset ; starting bit number within the word, ; this is the MSB of the bitrange within the word ; [externally, = word-offset + start] start ; number of bits in the value length ; length of word in which the value resides word-length ; lsb = bit number 0? lsb0? ) nil) ) ; Accessor fns. (define-getters bitrange (word-offset start length word-length lsb0?) ) (define-setters bitrange ; lsb0? left out on purpose: not sure changing it should be allowed (word-offset start length word-length) ) ; Return a boolean indicating if two bitranges overlap. ; ; lsb0? = #t: 31 ... 0 ; lsb0? = #f: 0 ... 31 (define (bitrange-overlap? start1 length1 start2 length2 lsb0?) (if lsb0? (let ((end1 (- start1 length1)) (end2 (- start2 length2))) (and (< end1 start2) (> start1 end2))) (let ((end1 (+ start1 length1)) (end2 (+ start2 length2))) (and (> end1 start2) (< start1 end2)))) ) ; Return a boolean indicating if BITPOS is beyond bitrange START,LEN. ; ??? This needs more thought. (define (bitpos-beyond? bitpos start length word-length lsb0?) (>= bitpos (+ start length)) ) ; Return the offset of the word after br. (define (bitrange-next-word br) (let ((word-offset (bitrange-word-offset br)) (start (bitrange-start br)) (length (bitrange-length br)) (word-length (bitrange-word-length br)) (lsb0? (bitrange-lsb0? br))) ; ??? revisit (+ word-offset word-length)) ) ; Initialize/finalize support. (define (types-init!) *UNSPECIFIED* ) (define (types-finish!) *UNSPECIFIED* )