| | |
| | smovecr.sa 3.1 12/10/90 |
| | |
| | The entry point sMOVECR returns the constant at the |
| | offset given in the instruction field. |
| | |
| | Input: An offset in the instruction word. |
| | |
| | Output: The constant rounded to the user's rounding |
| | mode unchecked for overflow. |
| | |
| | Modified: fp0. |
| | |
| | |
| | Copyright (C) Motorola, Inc. 1990 |
| | All Rights Reserved |
| | |
| | For details on the license for this file, please see the |
| | file, README, in this same directory. |
| |
| |SMOVECR idnt 2,1 | Motorola 040 Floating Point Software Package |
| |
| |section 8 |
| |
| #include "fpsp.h" |
| |
| |xref nrm_set |
| |xref round |
| |xref PIRN |
| |xref PIRZRM |
| |xref PIRP |
| |xref SMALRN |
| |xref SMALRZRM |
| |xref SMALRP |
| |xref BIGRN |
| |xref BIGRZRM |
| |xref BIGRP |
| |
| FZERO: .long 00000000 |
| | |
| | FMOVECR |
| | |
| .global smovcr |
| smovcr: |
| bfextu CMDREG1B(%a6){#9:#7},%d0 |get offset |
| bfextu USER_FPCR(%a6){#26:#2},%d1 |get rmode |
| | |
| | check range of offset |
| | |
| tstb %d0 |if zero, offset is to pi |
| beqs PI_TBL |it is pi |
| cmpib #0x0a,%d0 |check range $01 - $0a |
| bles Z_VAL |if in this range, return zero |
| cmpib #0x0e,%d0 |check range $0b - $0e |
| bles SM_TBL |valid constants in this range |
| cmpib #0x2f,%d0 |check range $10 - $2f |
| bles Z_VAL |if in this range, return zero |
| cmpib #0x3f,%d0 |check range $30 - $3f |
| ble BG_TBL |valid constants in this range |
| Z_VAL: |
| fmoves FZERO,%fp0 |
| rts |
| PI_TBL: |
| tstb %d1 |offset is zero, check for rmode |
| beqs PI_RN |if zero, rn mode |
| cmpib #0x3,%d1 |check for rp |
| beqs PI_RP |if 3, rp mode |
| PI_RZRM: |
| leal PIRZRM,%a0 |rmode is rz or rm, load PIRZRM in a0 |
| bra set_finx |
| PI_RN: |
| leal PIRN,%a0 |rmode is rn, load PIRN in a0 |
| bra set_finx |
| PI_RP: |
| leal PIRP,%a0 |rmode is rp, load PIRP in a0 |
| bra set_finx |
| SM_TBL: |
| subil #0xb,%d0 |make offset in 0 - 4 range |
| tstb %d1 |check for rmode |
| beqs SM_RN |if zero, rn mode |
| cmpib #0x3,%d1 |check for rp |
| beqs SM_RP |if 3, rp mode |
| SM_RZRM: |
| leal SMALRZRM,%a0 |rmode is rz or rm, load SMRZRM in a0 |
| cmpib #0x2,%d0 |check if result is inex |
| ble set_finx |if 0 - 2, it is inexact |
| bra no_finx |if 3, it is exact |
| SM_RN: |
| leal SMALRN,%a0 |rmode is rn, load SMRN in a0 |
| cmpib #0x2,%d0 |check if result is inex |
| ble set_finx |if 0 - 2, it is inexact |
| bra no_finx |if 3, it is exact |
| SM_RP: |
| leal SMALRP,%a0 |rmode is rp, load SMRP in a0 |
| cmpib #0x2,%d0 |check if result is inex |
| ble set_finx |if 0 - 2, it is inexact |
| bra no_finx |if 3, it is exact |
| BG_TBL: |
| subil #0x30,%d0 |make offset in 0 - f range |
| tstb %d1 |check for rmode |
| beqs BG_RN |if zero, rn mode |
| cmpib #0x3,%d1 |check for rp |
| beqs BG_RP |if 3, rp mode |
| BG_RZRM: |
| leal BIGRZRM,%a0 |rmode is rz or rm, load BGRZRM in a0 |
| cmpib #0x1,%d0 |check if result is inex |
| ble set_finx |if 0 - 1, it is inexact |
| cmpib #0x7,%d0 |second check |
| ble no_finx |if 0 - 7, it is exact |
| bra set_finx |if 8 - f, it is inexact |
| BG_RN: |
| leal BIGRN,%a0 |rmode is rn, load BGRN in a0 |
| cmpib #0x1,%d0 |check if result is inex |
| ble set_finx |if 0 - 1, it is inexact |
| cmpib #0x7,%d0 |second check |
| ble no_finx |if 0 - 7, it is exact |
| bra set_finx |if 8 - f, it is inexact |
| BG_RP: |
| leal BIGRP,%a0 |rmode is rp, load SMRP in a0 |
| cmpib #0x1,%d0 |check if result is inex |
| ble set_finx |if 0 - 1, it is inexact |
| cmpib #0x7,%d0 |second check |
| ble no_finx |if 0 - 7, it is exact |
| | bra set_finx ;if 8 - f, it is inexact |
| set_finx: |
| orl #inx2a_mask,USER_FPSR(%a6) |set inex2/ainex |
| no_finx: |
| mulul #12,%d0 |use offset to point into tables |
| movel %d1,L_SCR1(%a6) |load mode for round call |
| bfextu USER_FPCR(%a6){#24:#2},%d1 |get precision |
| tstl %d1 |check if extended precision |
| | |
| | Precision is extended |
| | |
| bnes not_ext |if extended, do not call round |
| fmovemx (%a0,%d0),%fp0-%fp0 |return result in fp0 |
| rts |
| | |
| | Precision is single or double |
| | |
| not_ext: |
| swap %d1 |rnd prec in upper word of d1 |
| addl L_SCR1(%a6),%d1 |merge rmode in low word of d1 |
| movel (%a0,%d0),FP_SCR1(%a6) |load first word to temp storage |
| movel 4(%a0,%d0),FP_SCR1+4(%a6) |load second word |
| movel 8(%a0,%d0),FP_SCR1+8(%a6) |load third word |
| clrl %d0 |clear g,r,s |
| lea FP_SCR1(%a6),%a0 |
| btstb #sign_bit,LOCAL_EX(%a0) |
| sne LOCAL_SGN(%a0) |convert to internal ext. format |
| |
| bsr round |go round the mantissa |
| |
| bfclr LOCAL_SGN(%a0){#0:#8} |convert back to IEEE ext format |
| beqs fin_fcr |
| bsetb #sign_bit,LOCAL_EX(%a0) |
| fin_fcr: |
| fmovemx (%a0),%fp0-%fp0 |
| rts |
| |
| |end |