mirror of
https://github.com/elliotnunn/mac-rom.git
synced 2024-12-29 07:29:15 +00:00
400 lines
11 KiB
Plaintext
400 lines
11 KiB
Plaintext
|
;
|
|||
|
; File: XOpErr.a
|
|||
|
;
|
|||
|
; Contains: Routines to handle the Operand Error Exception
|
|||
|
;
|
|||
|
; Originally Written by: Motorola Inc.
|
|||
|
; Adapted to Apple/MPW: Jon Okada
|
|||
|
;
|
|||
|
; Copyright: <09> 1990, 1991 by Apple Computer, Inc., all rights reserved.
|
|||
|
;
|
|||
|
; This file is used in these builds: Mac32
|
|||
|
;
|
|||
|
; Change History (most recent first):
|
|||
|
;
|
|||
|
; <2> 3/30/91 BG Rolling in Jon Okada's latest changes.
|
|||
|
; <1> 12/14/90 BG First checked into TERROR/BBS.
|
|||
|
|
|||
|
; xoperr.a
|
|||
|
|
|||
|
; Based upon Motorola file 'x_operr.sa'.
|
|||
|
|
|||
|
; CHANGE LOG:
|
|||
|
; 09 Jan 91 JPO Inserted label "operr" at top of code. Deleted
|
|||
|
; unreferenced labels "ck_inex", "enabled", and
|
|||
|
; "take_inex". Replaced "bsr get_fline" with in-line
|
|||
|
; code. Modified code which branches to user handlers.
|
|||
|
; Changed "bra fpsp_done" to "rte".
|
|||
|
; 14 Jan 91 JPO Folded in Motorola update of 'x_operr.sa'.
|
|||
|
;
|
|||
|
|
|||
|
*
|
|||
|
* x_operr.sa 3.3 12/18/90
|
|||
|
*
|
|||
|
* fpsp_operr --- FPSP handler for operand error exception
|
|||
|
*
|
|||
|
* See 68040 User's Manual pp. 9-44f
|
|||
|
*
|
|||
|
* Note 1: For trap disabled 040 does the following:
|
|||
|
* If the dest is a fp reg, then an extended precision non_signaling
|
|||
|
* NAN is stored in the dest reg. If the dest format is b, w, or l and
|
|||
|
* the source op is a NAN, then garbage is stored as the result (actually
|
|||
|
* the upper 32 bits of the mantissa are sent to the integer unit). If
|
|||
|
* the dest format is integer (b, w, l) and the operr is caused by
|
|||
|
* integer overflow, or the source op is inf, then the result stored is
|
|||
|
* garbage.
|
|||
|
* There are three cases in which operr is incorrectly signaled on the
|
|||
|
* 040. This occurs for move_out of format b, w, or l for the largest
|
|||
|
* negative integer (-2^7 for b, -2^15 for w, -2^31 for l).
|
|||
|
*
|
|||
|
* On opclass = 011 fmove.(b,w,l) that causes a conversion
|
|||
|
* overflow -> OPERR, the exponent in wbte (and fpte) is:
|
|||
|
* byte 56 - (62 - exp)
|
|||
|
* word 48 - (62 - exp)
|
|||
|
* long 32 - (62 - exp)
|
|||
|
*
|
|||
|
* where exp = (true exp) - 1
|
|||
|
*
|
|||
|
* So, wbtemp and fptemp will contain the following on erroneoulsy
|
|||
|
* signalled operr:
|
|||
|
* fpts = 1
|
|||
|
* fpte = $4000 (15 bit externally)
|
|||
|
* byte fptm = $ffffffff ffffff80
|
|||
|
* word fptm = $ffffffff ffff8000
|
|||
|
* long fptm = $ffffffff 80000000
|
|||
|
*
|
|||
|
* Note 2: For trap enabled 040 does the following:
|
|||
|
* If the inst is move_out, then same as Note 1.
|
|||
|
* If the inst is not move_out, the dest is not modified.
|
|||
|
* The exceptional operand is not defined for integer overflow
|
|||
|
* during a move_out.
|
|||
|
*
|
|||
|
|
|||
|
* Copyright (C) Motorola, Inc. 1990
|
|||
|
* All Rights Reserved
|
|||
|
*
|
|||
|
* THIS IS UNPUBLISHED PROPRIETARY SOURCE CODE OF MOTOROLA
|
|||
|
* The copyright notice above does not evidence any
|
|||
|
* actual or intended publication of such source code.
|
|||
|
|
|||
|
* X_OPERR IDNT 2,1 Motorola 040 Floating Point Software Package
|
|||
|
|
|||
|
|
|||
|
operr: ; label added <1/9/91, JPO>
|
|||
|
fpsp_operr:
|
|||
|
*
|
|||
|
link a6,#-LOCAL_SIZE
|
|||
|
fsave -(a7)
|
|||
|
movem.l d0-d1/a0-a1,USER_DA(a6)
|
|||
|
fmovem.x fp0-fp3,USER_FP0(a6)
|
|||
|
fmovem.l fpcr/fpsr/fpiar,USER_FPCR(a6)
|
|||
|
*
|
|||
|
* Check if this is an opclass 3 instruction.
|
|||
|
* If so, fall through, else branch to operr_end
|
|||
|
*
|
|||
|
btst.b #TFLAG,T_BYTE(a6)
|
|||
|
beq.b operr_end
|
|||
|
|
|||
|
*
|
|||
|
* If the destination size is B,W,or L, the operr must be
|
|||
|
* handled here.
|
|||
|
*
|
|||
|
move.l CMDREG1B(a6),d0
|
|||
|
bfextu d0{3:3},d0 ;0=long, 4=word, 6=byte
|
|||
|
cmpi.b #0,d0 ;determine size; check long
|
|||
|
beq.b operr_long
|
|||
|
cmpi.b #4,d0 ;check word
|
|||
|
beq operr_word
|
|||
|
cmpi.b #6,d0 ;check byte
|
|||
|
beq.w operr_byte
|
|||
|
|
|||
|
*
|
|||
|
* The size is not B,W,or L, so the operr is handled by the
|
|||
|
* kernel handler. Set the operr bits and clean up, leaving
|
|||
|
* only the integer exception frame on the stack, and the
|
|||
|
* fpu in the original exceptional state.
|
|||
|
*
|
|||
|
operr_end:
|
|||
|
bset.b #operr_bit,FPSR_EXCEPT(a6)
|
|||
|
bset.b #aiop_bit,FPSR_AEXCEPT(a6)
|
|||
|
|
|||
|
movem.l USER_DA(a6),d0-d1/a0-a1
|
|||
|
fmovem.x USER_FP0(a6),fp0-fp3
|
|||
|
fmovem.l USER_FPCR(a6),fpcr/fpsr/fpiar
|
|||
|
frestore (a7)+
|
|||
|
unlk a6
|
|||
|
|
|||
|
; bra.l real_operr ; deleted <1/9/91, JPO>
|
|||
|
|
|||
|
move.l (FPOPERR_VEC040).W,-(sp) ; push vector to user's handler <1/9/91, JPO>
|
|||
|
rts ; execute user's handler <1/9/91, JPO>
|
|||
|
|
|||
|
operr_long:
|
|||
|
moveq.l #4,d1 ;write size to d1
|
|||
|
move.b STAG(a6),d0 ;test stag for nan
|
|||
|
andi.b #$e0,d0 ;clr all but tag
|
|||
|
cmpi.b #$60,d0 ;check for nan
|
|||
|
beq operr_nan
|
|||
|
cmpi.l #$80000000,FPTEMP_LO(a6) ;test if ls lword is special
|
|||
|
bne.b chklerr ;if not equal, check for incorrect operr
|
|||
|
bsr check_upper ;check if exp and ms mant are special
|
|||
|
tst.l d0
|
|||
|
bne.b chklerr ;if d0 is true, check for incorrect operr
|
|||
|
move.l #$80000000,d0 ;store special case result
|
|||
|
bsr operr_store
|
|||
|
bra.w not_enabled ;clean and exit
|
|||
|
|
|||
|
*
|
|||
|
* CHECK FOR INCORRECTLY GENERATED OPERR EXCEPTION HERE
|
|||
|
*
|
|||
|
chklerr:
|
|||
|
move.w FPTEMP_EX(a6),d0
|
|||
|
and.w #$7FFF,d0 ;ignore sign bit
|
|||
|
cmp.w #$3FFE,d0 ;this is the only possible exponent value
|
|||
|
bne.b chklerr2
|
|||
|
fixlong:
|
|||
|
move.l FPTEMP_LO(a6),d0
|
|||
|
bsr operr_store
|
|||
|
bra.w not_enabled
|
|||
|
chklerr2:
|
|||
|
move.w FPTEMP_EX(a6),d0
|
|||
|
and.w #$7FFF,d0 ;ignore sign bit
|
|||
|
cmp.w #$4000,d0
|
|||
|
bhs.w store_max ;exponent out of range
|
|||
|
|
|||
|
move.l FPTEMP_LO(a6),d0
|
|||
|
and.l #$7FFF0000,d0 ;look for all 1's on bits 30-16
|
|||
|
cmp.l #$7FFF0000,d0
|
|||
|
beq.b fixlong
|
|||
|
|
|||
|
tst.l FPTEMP_LO(a6)
|
|||
|
bpl.b chklepos
|
|||
|
cmp.l #$FFFFFFFF,FPTEMP_HI(a6)
|
|||
|
beq.b fixlong
|
|||
|
bra.w store_max
|
|||
|
chklepos:
|
|||
|
tst.l FPTEMP_HI(a6)
|
|||
|
beq.b fixlong
|
|||
|
bra.w store_max
|
|||
|
|
|||
|
operr_word:
|
|||
|
moveq.l #2,d1 ;write size to d1
|
|||
|
move.b STAG(a6),d0 ;test stag for nan
|
|||
|
andi.b #$e0,d0 ;clr all but tag
|
|||
|
cmpi.b #$60,d0 ;check for nan
|
|||
|
beq operr_nan
|
|||
|
cmpi.l #$ffff8000,FPTEMP_LO(a6) ;test if ls lword is special
|
|||
|
bne.b chkwerr ;if not equal, check for incorrect operr
|
|||
|
bsr check_upper ;check if exp and ms mant are special
|
|||
|
tst.l d0
|
|||
|
bne.b chkwerr ;if d0 is true, check for incorrect operr
|
|||
|
move.l #$80000000,d0 ;store special case result
|
|||
|
bsr operr_store
|
|||
|
bra.w not_enabled ;clean and exit
|
|||
|
|
|||
|
*
|
|||
|
* CHECK FOR INCORRECTLY GENERATED OPERR EXCEPTION HERE
|
|||
|
*
|
|||
|
chkwerr:
|
|||
|
move.w FPTEMP_EX(a6),d0
|
|||
|
and.w #$7FFF,d0 ;ignore sign bit
|
|||
|
cmp.w #$3FFE,d0 ;this is the only possible exponent value
|
|||
|
bne.b store_max
|
|||
|
move.l FPTEMP_LO(a6),d0
|
|||
|
swap d0
|
|||
|
bsr operr_store
|
|||
|
bra.w not_enabled
|
|||
|
|
|||
|
operr_byte:
|
|||
|
moveq.l #1,d1 ;write size to d1
|
|||
|
move.b STAG(a6),d0 ;test stag for nan
|
|||
|
andi.b #$e0,d0 ;clr all but tag
|
|||
|
cmpi.b #$60,d0 ;check for nan
|
|||
|
beq.b operr_nan
|
|||
|
cmpi.l #$ffffff80,FPTEMP_LO(a6) ;test if ls lword is special
|
|||
|
bne.b chkberr ;if not equal, check for incorrect operr
|
|||
|
bsr check_upper ;check if exp and ms mant are special
|
|||
|
tst.l d0
|
|||
|
bne.b chkberr ;if d0 is true, check for incorrect operr
|
|||
|
move.l #$80000000,d0 ;store special case result
|
|||
|
bsr.b operr_store
|
|||
|
bra.w not_enabled ;clean and exit
|
|||
|
*
|
|||
|
* CHECK FOR INCORRECTLY GENERATED OPERR EXCEPTION HERE
|
|||
|
*
|
|||
|
chkberr:
|
|||
|
move.w FPTEMP_EX(a6),d0
|
|||
|
and.w #$7FFF,d0 ;ignore sign bit
|
|||
|
cmp.w #$3FFE,d0 ;this is the only possible exponent value
|
|||
|
bne.b store_max
|
|||
|
move.l FPTEMP_LO(a6),d0
|
|||
|
asl.l #8,d0
|
|||
|
swap d0
|
|||
|
bsr.b operr_store
|
|||
|
bra.w not_enabled
|
|||
|
|
|||
|
*
|
|||
|
* This operr condition is not of the special case. Set operr
|
|||
|
* and aiop and write the portion of the nan to memory for the
|
|||
|
* given size.
|
|||
|
*
|
|||
|
operr_nan:
|
|||
|
or.l #opaop_mask,USER_FPSR(a6) ;set operr & aiop
|
|||
|
|
|||
|
move.l ETEMP_HI(a6),d0 ;output will be from upper 32 bits
|
|||
|
bsr.b operr_store
|
|||
|
bra end_operr
|
|||
|
*
|
|||
|
* Store_max loads the max pos or negative for the size, sets
|
|||
|
* the operr and aiop bits, and clears inex and ainex, incorrectly
|
|||
|
* set by the 040.
|
|||
|
*
|
|||
|
store_max:
|
|||
|
or.l #opaop_mask,USER_FPSR(a6) ;set operr & aiop
|
|||
|
bclr.b #inex2_bit,FPSR_EXCEPT(a6)
|
|||
|
bclr.b #ainex_bit,FPSR_AEXCEPT(a6)
|
|||
|
fmove.l #0,FPSR
|
|||
|
|
|||
|
tst.w FPTEMP_EX(a6) ;check sign
|
|||
|
blt.b load_neg
|
|||
|
move.l #$7fffffff,d0
|
|||
|
bsr.b operr_store
|
|||
|
bra.b end_operr
|
|||
|
load_neg:
|
|||
|
move.l #$80000000,d0
|
|||
|
bsr.b operr_store
|
|||
|
bra.b end_operr
|
|||
|
|
|||
|
*
|
|||
|
* This routine stores the data in d0, for the given size in d1,
|
|||
|
* to memory or data register as required. A read of the fline
|
|||
|
* is required to determine the destination.
|
|||
|
*
|
|||
|
operr_store:
|
|||
|
move.l d0,L_SCR1(a6) ;move write data to L_SCR1
|
|||
|
; move.l d1,-(a7) ;save register size - DELETED <1/9/91, JPO>
|
|||
|
; bsr.l get_fline ;fline returned in d0 - DELETED <1/9/91, JPO>
|
|||
|
; move.l (a7)+,d1 ;deleted <1/9/91, JPO>
|
|||
|
|
|||
|
movea.l USER_FPIAR(a6),a0 ; return opcode in d0.w <1/9/91, JPO>
|
|||
|
sub.l d0,d0 ; <1/9/91, JPO>
|
|||
|
move.w (a0),d0 ; <1/9/91, JPO>
|
|||
|
|
|||
|
bftst d0{26:3} ;if mode is zero, dest is Dn
|
|||
|
bne.b dest_mem
|
|||
|
*
|
|||
|
* Destination is Dn. Get register number from d0. Data is on
|
|||
|
* the stack at (a7). D1 has size: 1=byte,2=word,4=long/single
|
|||
|
*
|
|||
|
andi.l #7,d0 ;isolate register number
|
|||
|
cmpi.l #4,d1
|
|||
|
beq.b op_long ;the most frequent case
|
|||
|
cmpi.l #2,d1
|
|||
|
bne.b op_con
|
|||
|
or.l #8,d0
|
|||
|
bra.b op_con
|
|||
|
op_long:
|
|||
|
or.l #$10,d0
|
|||
|
op_con:
|
|||
|
move.l d0,d1 ;format size:reg for reg_dest
|
|||
|
bra reg_dest ;call to reg_dest returns to caller
|
|||
|
* ;of operr_store
|
|||
|
*
|
|||
|
* Destination is memory. Get <ea> from integer exception frame
|
|||
|
* and call mem_write.
|
|||
|
*
|
|||
|
dest_mem:
|
|||
|
lea.l L_SCR1(a6),a0 ;put ptr to write data in a0
|
|||
|
move.l EXC_EA(a6),a1 ;put user destination address in a1
|
|||
|
move.l d1,d0 ;put size in d0
|
|||
|
bsr mem_write
|
|||
|
rts
|
|||
|
*
|
|||
|
* Check the exponent for $c000 and the upper 32 bits of the
|
|||
|
* mantissa for $ffffffff. If both are true, return d0 clr
|
|||
|
* and store the lower n bits of the least lword of FPTEMP
|
|||
|
* to d0 for write out. If not, it is a real operr, and set d0.
|
|||
|
*
|
|||
|
check_upper:
|
|||
|
cmpi.l #$ffffffff,FPTEMP_HI(a6) ;check if first byte is all 1's
|
|||
|
bne.b true_operr ;if not all 1's then was true operr
|
|||
|
cmpi.w #$c000,FPTEMP_EX(a6) ;check if incorrectly signalled
|
|||
|
beq.b not_true_operr ;branch if not true operr
|
|||
|
cmpi.w #$bfff,FPTEMP_EX(a6) ;check if incorrectly signalled
|
|||
|
beq.b not_true_operr ;branch if not true operr
|
|||
|
true_operr:
|
|||
|
move.l #1,d0 ;signal real operr
|
|||
|
rts
|
|||
|
not_true_operr:
|
|||
|
clr.l d0 ;signal no real operr
|
|||
|
rts
|
|||
|
|
|||
|
*
|
|||
|
* End_operr tests for operr enabled. If not, it cleans up the stack
|
|||
|
* and does an rte. If enabled, it cleans up the stack and branches
|
|||
|
* to the kernel operr handler with only the integer exception
|
|||
|
* frame on the stack and the fpu in the original exceptional state
|
|||
|
* with correct data written to the destination.
|
|||
|
*
|
|||
|
end_operr:
|
|||
|
btst.b #operr_bit,FPCR_ENABLE(a6)
|
|||
|
beq.b not_enabled
|
|||
|
;enabled: ; label deleted <1/9/91, JPO>
|
|||
|
movem.l USER_DA(a6),d0-d1/a0-a1
|
|||
|
fmovem.x USER_FP0(a6),fp0-fp3
|
|||
|
fmovem.l USER_FPCR(a6),fpcr/fpsr/fpiar
|
|||
|
frestore (a7)+
|
|||
|
unlk a6
|
|||
|
|
|||
|
; bra.l real_operr ; deleted <1/9/91, JPO>
|
|||
|
|
|||
|
move.l (FPOPERR_VEC040).W,-(sp) ; push vector to user handler <1/9/91, JPO>
|
|||
|
rts ; execute user handler <1/9/91, JPO>
|
|||
|
|
|||
|
not_enabled:
|
|||
|
*
|
|||
|
* It is possible to have either inex2 or inex1 exceptions with the
|
|||
|
* operr. If the inex enable bit is set in the FPCR, and either
|
|||
|
* inex2 or inex1 occured, we must clean up and branch to the
|
|||
|
* real inex handler.
|
|||
|
*
|
|||
|
;ck_inex: ; label DELETED <1/9/91, JPO>
|
|||
|
move.b FPCR_ENABLE(a6),d0
|
|||
|
and.b FPSR_EXCEPT(a6),d0
|
|||
|
andi.b #$3,d0
|
|||
|
beq.b operr_exit
|
|||
|
*
|
|||
|
* Inexact enabled and reported, and we must take an inexact exception.
|
|||
|
*
|
|||
|
;take_inex: ; label deleted <1/9/91, JPO>
|
|||
|
move.b #INEX_VEC,EXC_VEC+1(a6)
|
|||
|
move.l USER_FPSR(a6),FPSR_SHADOW(a6)
|
|||
|
or.l #sx_mask,E_BYTE(a6)
|
|||
|
movem.l USER_DA(a6),d0-d1/a0-a1
|
|||
|
fmovem.x USER_FP0(a6),fp0-fp3
|
|||
|
fmovem.l USER_FPCR(a6),fpcr/fpsr/fpiar
|
|||
|
frestore (a7)+
|
|||
|
unlk a6
|
|||
|
|
|||
|
; bra.l real_inex ; deleted <1/9/91, JPO>
|
|||
|
|
|||
|
move.l ($00C4).W,-(sp) ; push vector to user INEX handler <1/9/91, JPO>
|
|||
|
rts ; execute user handler <1/9/91, JPO>
|
|||
|
|
|||
|
*
|
|||
|
* Since operr is only an E1 exception, there is no need to frestore
|
|||
|
* any state back to the fpu.
|
|||
|
*
|
|||
|
operr_exit:
|
|||
|
movem.l USER_DA(a6),d0-d1/a0-a1
|
|||
|
fmovem.x USER_FP0(a6),fp0-fp3
|
|||
|
fmovem.l USER_FPCR(a6),fpcr/fpsr/fpiar
|
|||
|
unlk a6
|
|||
|
|
|||
|
; bra.l fpsp_done ; deleted <1/9/91, JPO>
|
|||
|
|
|||
|
rte ; <1/9/91, JPO>
|
|||
|
|
|||
|
|
|||
|
|