Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
194d0710
Commit
194d0710
authored
Aug 03, 2005
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Plain Diff
Merge master.kernel.org:/home/rmk/linux-2.6-arm
parents
a68d2ebc
9bbd0375
Changes
19
Show whitespace changes
Inline
Side-by-side
Showing
19 changed files
with
337 additions
and
484 deletions
+337
-484
Documentation/dontdiff
Documentation/dontdiff
+1
-0
arch/arm/mach-ixp4xx/coyote-setup.c
arch/arm/mach-ixp4xx/coyote-setup.c
+1
-1
arch/arm/mach-ixp4xx/gtwx5715-setup.c
arch/arm/mach-ixp4xx/gtwx5715-setup.c
+1
-1
arch/arm/mach-ixp4xx/ixdp425-setup.c
arch/arm/mach-ixp4xx/ixdp425-setup.c
+2
-2
arch/arm/mach-s3c2410/mach-bast.c
arch/arm/mach-s3c2410/mach-bast.c
+15
-1
arch/arm/mm/proc-xscale.S
arch/arm/mm/proc-xscale.S
+0
-136
arch/arm/nwfpe/double_cpdo.c
arch/arm/nwfpe/double_cpdo.c
+12
-12
arch/arm/nwfpe/extended_cpdo.c
arch/arm/nwfpe/extended_cpdo.c
+12
-12
arch/arm/nwfpe/fpa11.c
arch/arm/nwfpe/fpa11.c
+11
-19
arch/arm/nwfpe/fpa11.h
arch/arm/nwfpe/fpa11.h
+9
-2
arch/arm/nwfpe/fpa11_cpdo.c
arch/arm/nwfpe/fpa11_cpdo.c
+17
-11
arch/arm/nwfpe/fpa11_cpdt.c
arch/arm/nwfpe/fpa11_cpdt.c
+14
-8
arch/arm/nwfpe/fpa11_cprt.c
arch/arm/nwfpe/fpa11_cprt.c
+19
-9
arch/arm/nwfpe/fpmodule.c
arch/arm/nwfpe/fpmodule.c
+6
-9
arch/arm/nwfpe/single_cpdo.c
arch/arm/nwfpe/single_cpdo.c
+12
-12
arch/arm/nwfpe/softfloat.c
arch/arm/nwfpe/softfloat.c
+163
-171
arch/arm/nwfpe/softfloat.h
arch/arm/nwfpe/softfloat.h
+29
-39
arch/arm/vfp/vfpdouble.c
arch/arm/vfp/vfpdouble.c
+3
-0
drivers/char/watchdog/sa1100_wdt.c
drivers/char/watchdog/sa1100_wdt.c
+10
-39
No files found.
Documentation/dontdiff
View file @
194d0710
...
...
@@ -104,6 +104,7 @@ logo_*.c
logo_*_clut224.c
logo_*_mono.c
lxdialog
mach-types
mach-types.h
make_times_h
map
...
...
arch/arm/mach-ixp4xx/coyote-setup.c
View file @
194d0710
...
...
@@ -61,7 +61,7 @@ static struct plat_serial8250_port coyote_uart_data[] = {
.
mapbase
=
IXP4XX_UART2_BASE_PHYS
,
.
membase
=
(
char
*
)
IXP4XX_UART2_BASE_VIRT
+
REG_OFFSET
,
.
irq
=
IRQ_IXP4XX_UART2
,
.
flags
=
UPF_BOOT_AUTOCONF
,
.
flags
=
UPF_BOOT_AUTOCONF
|
UPF_SKIP_TEST
,
.
iotype
=
UPIO_MEM
,
.
regshift
=
2
,
.
uartclk
=
IXP4XX_UART_XTAL
,
...
...
arch/arm/mach-ixp4xx/gtwx5715-setup.c
View file @
194d0710
...
...
@@ -83,7 +83,7 @@ static struct plat_serial8250_port gtwx5715_uart_platform_data[] = {
.
mapbase
=
IXP4XX_UART2_BASE_PHYS
,
.
membase
=
(
char
*
)
IXP4XX_UART2_BASE_VIRT
+
REG_OFFSET
,
.
irq
=
IRQ_IXP4XX_UART2
,
.
flags
=
UPF_BOOT_AUTOCONF
,
.
flags
=
UPF_BOOT_AUTOCONF
|
UPF_SKIP_TEST
,
.
iotype
=
UPIO_MEM
,
.
regshift
=
2
,
.
uartclk
=
IXP4XX_UART_XTAL
,
...
...
arch/arm/mach-ixp4xx/ixdp425-setup.c
View file @
194d0710
...
...
@@ -82,7 +82,7 @@ static struct plat_serial8250_port ixdp425_uart_data[] = {
.
mapbase
=
IXP4XX_UART1_BASE_PHYS
,
.
membase
=
(
char
*
)
IXP4XX_UART1_BASE_VIRT
+
REG_OFFSET
,
.
irq
=
IRQ_IXP4XX_UART1
,
.
flags
=
UPF_BOOT_AUTOCONF
,
.
flags
=
UPF_BOOT_AUTOCONF
|
UPF_SKIP_TEST
,
.
iotype
=
UPIO_MEM
,
.
regshift
=
2
,
.
uartclk
=
IXP4XX_UART_XTAL
,
...
...
@@ -91,7 +91,7 @@ static struct plat_serial8250_port ixdp425_uart_data[] = {
.
mapbase
=
IXP4XX_UART2_BASE_PHYS
,
.
membase
=
(
char
*
)
IXP4XX_UART2_BASE_VIRT
+
REG_OFFSET
,
.
irq
=
IRQ_IXP4XX_UART1
,
.
flags
=
UPF_BOOT_AUTOCONF
,
.
flags
=
UPF_BOOT_AUTOCONF
|
UPF_SKIP_TEST
,
.
iotype
=
UPIO_MEM
,
.
regshift
=
2
,
.
uartclk
=
IXP4XX_UART_XTAL
,
...
...
arch/arm/mach-s3c2410/mach-bast.c
View file @
194d0710
...
...
@@ -30,6 +30,7 @@
* 28-Jun-2005 BJD Moved pm functionality out to common code
* 17-Jul-2005 BJD Changed to platform device for SuperIO 16550s
* 25-Jul-2005 BJD Removed ASIX static mappings
* 27-Jul-2005 BJD Ensure maximum frequency of i2c bus
*/
#include <linux/kernel.h>
...
...
@@ -60,6 +61,7 @@
#include <asm/arch/regs-mem.h>
#include <asm/arch/regs-lcd.h>
#include <asm/arch/nand.h>
#include <asm/arch/iic.h>
#include <linux/mtd/mtd.h>
#include <linux/mtd/nand.h>
...
...
@@ -304,7 +306,7 @@ static void bast_nand_select(struct s3c2410_nand_set *set, int slot)
}
static
struct
s3c2410_platform_nand
bast_nand_info
=
{
.
tacls
=
8
0
,
.
tacls
=
4
0
,
.
twrph0
=
80
,
.
twrph1
=
80
,
.
nr_sets
=
ARRAY_SIZE
(
bast_nand_sets
),
...
...
@@ -385,6 +387,17 @@ static struct platform_device bast_sio = {
},
};
/* we have devices on the bus which cannot work much over the
* standard 100KHz i2c bus frequency
*/
static
struct
s3c2410_platform_i2c
bast_i2c_info
=
{
.
flags
=
0
,
.
slave_addr
=
0x10
,
.
bus_freq
=
100
*
1000
,
.
max_freq
=
130
*
1000
,
};
/* Standard BAST devices */
static
struct
platform_device
*
bast_devices
[]
__initdata
=
{
...
...
@@ -431,6 +444,7 @@ void __init bast_map_io(void)
s3c24xx_uclk
.
parent
=
&
s3c24xx_clkout1
;
s3c_device_nand
.
dev
.
platform_data
=
&
bast_nand_info
;
s3c_device_i2c
.
dev
.
platform_data
=
&
bast_i2c_info
;
s3c24xx_init_io
(
bast_iodesc
,
ARRAY_SIZE
(
bast_iodesc
));
s3c24xx_init_clocks
(
0
);
...
...
arch/arm/mm/proc-xscale.S
View file @
194d0710
...
...
@@ -370,142 +370,6 @@ ENTRY(cpu_xscale_dcache_clean_area)
bhi
1
b
mov
pc
,
lr
/*
===============================
=
CACHE
LOCKING
============================
*
*
The
XScale
MicroArchitecture
implements
support
for
locking
entries
into
*
the
data
and
instruction
cache
.
The
following
functions
implement
the
core
*
low
level
instructions
needed
to
accomplish
the
locking
.
The
developer
's
*
manual
states
that
the
code
that
performs
the
locking
must
be
in
non
-
cached
*
memory
.
To
accomplish
this
,
the
code
in
xscale
-
cache
-
lock
.
c
copies
the
*
following
functions
from
the
cache
into
a
non
-
cached
memory
region
that
*
is
allocated
through
consistent_alloc
()
.
*
*/
.
align
5
/*
*
xscale_icache_lock
*
*
r0
:
starting
address
to
lock
*
r1
:
end
address
to
lock
*/
ENTRY
(
xscale_icache_lock
)
iLockLoop
:
bic
r0
,
r0
,
#
CACHELINESIZE
-
1
mcr
p15
,
0
,
r0
,
c9
,
c1
,
0
@
lock
into
cache
cmp
r0
,
r1
@
are
we
done
?
add
r0
,
r0
,
#
CACHELINESIZE
@
advance
to
next
cache
line
bls
iLockLoop
mov
pc
,
lr
/*
*
xscale_icache_unlock
*/
ENTRY
(
xscale_icache_unlock
)
mcr
p15
,
0
,
r0
,
c9
,
c1
,
1
@
Unlock
icache
mov
pc
,
lr
/*
*
xscale_dcache_lock
*
*
r0
:
starting
address
to
lock
*
r1
:
end
address
to
lock
*/
ENTRY
(
xscale_dcache_lock
)
mcr
p15
,
0
,
ip
,
c7
,
c10
,
4
@
Drain
Write
(&
Fill
)
Buffer
mov
r2
,
#
1
mcr
p15
,
0
,
r2
,
c9
,
c2
,
0
@
Put
dcache
in
lock
mode
cpwait
ip
@
Wait
for
completion
mrs
r2
,
cpsr
orr
r3
,
r2
,
#
PSR_F_BIT
|
PSR_I_BIT
dLockLoop
:
msr
cpsr_c
,
r3
mcr
p15
,
0
,
r0
,
c7
,
c10
,
1
@
Write
back
line
if
it
is
dirty
mcr
p15
,
0
,
r0
,
c7
,
c6
,
1
@
Flush
/
invalidate
line
msr
cpsr_c
,
r2
ldr
ip
,
[
r0
],
#
CACHELINESIZE
@
Preload
32
bytes
into
cache
from
@
location
[
r0
]
.
Post
-
increment
@
r3
to
next
cache
line
cmp
r0
,
r1
@
Are
we
done
?
bls
dLockLoop
mcr
p15
,
0
,
ip
,
c7
,
c10
,
4
@
Drain
Write
(&
Fill
)
Buffer
mov
r2
,
#
0
mcr
p15
,
0
,
r2
,
c9
,
c2
,
0
@
Get
out
of
lock
mode
cpwait_ret
lr
,
ip
/*
*
xscale_dcache_unlock
*/
ENTRY
(
xscale_dcache_unlock
)
mcr
p15
,
0
,
ip
,
c7
,
c10
,
4
@
Drain
Write
(&
Fill
)
Buffer
mcr
p15
,
0
,
ip
,
c9
,
c2
,
1
@
Unlock
cache
mov
pc
,
lr
/*
*
Needed
to
determine
the
length
of
the
code
that
needs
to
be
copied
.
*/
.
align
5
ENTRY
(
xscale_cache_dummy
)
mov
pc
,
lr
/*
===============================
=
TLB
LOCKING
==============================
*
*
The
XScale
MicroArchitecture
implements
support
for
locking
entries
into
*
the
Instruction
and
Data
TLBs
.
The
following
functions
provide
the
*
low
level
support
for
supporting
these
under
Linux
.
xscale
-
lock
.
c
*
implements
some
higher
level
management
code
.
Most
of
the
following
*
is
taken
straight
out
of
the
Developer
's Manual.
*/
/*
*
Lock
I
-
TLB
entry
*
*
r0
:
Virtual
address
to
translate
and
lock
*/
.
align
5
ENTRY
(
xscale_itlb_lock
)
mrs
r2
,
cpsr
orr
r3
,
r2
,
#
PSR_F_BIT
|
PSR_I_BIT
msr
cpsr_c
,
r3
@
Disable
interrupts
mcr
p15
,
0
,
r0
,
c8
,
c5
,
1
@
Invalidate
I
-
TLB
entry
mcr
p15
,
0
,
r0
,
c10
,
c4
,
0
@
Translate
and
lock
msr
cpsr_c
,
r2
@
Restore
interrupts
cpwait_ret
lr
,
ip
/*
*
Lock
D
-
TLB
entry
*
*
r0
:
Virtual
address
to
translate
and
lock
*/
.
align
5
ENTRY
(
xscale_dtlb_lock
)
mrs
r2
,
cpsr
orr
r3
,
r2
,
#
PSR_F_BIT
|
PSR_I_BIT
msr
cpsr_c
,
r3
@
Disable
interrupts
mcr
p15
,
0
,
r0
,
c8
,
c6
,
1
@
Invalidate
D
-
TLB
entry
mcr
p15
,
0
,
r0
,
c10
,
c8
,
0
@
Translate
and
lock
msr
cpsr_c
,
r2
@
Restore
interrupts
cpwait_ret
lr
,
ip
/*
*
Unlock
all
I
-
TLB
entries
*/
.
align
5
ENTRY
(
xscale_itlb_unlock
)
mcr
p15
,
0
,
ip
,
c10
,
c4
,
1
@
Unlock
I
-
TLB
mcr
p15
,
0
,
ip
,
c8
,
c5
,
0
@
Invalidate
I
-
TLB
cpwait_ret
lr
,
ip
/*
*
Unlock
all
D
-
TLB
entries
*/
ENTRY
(
xscale_dtlb_unlock
)
mcr
p15
,
0
,
ip
,
c10
,
c8
,
1
@
Unlock
D
-
TBL
mcr
p15
,
0
,
ip
,
c8
,
c6
,
0
@
Invalidate
D
-
TLB
cpwait_ret
lr
,
ip
/*
==============================
=
PageTable
==============================
*/
#define PTE_CACHE_WRITE_ALLOCATE 0
...
...
arch/arm/nwfpe/double_cpdo.c
View file @
194d0710
...
...
@@ -40,17 +40,17 @@ float64 float64_arccos(float64 rFm);
float64
float64_pow
(
float64
rFn
,
float64
rFm
);
float64
float64_pol
(
float64
rFn
,
float64
rFm
);
static
float64
float64_rsf
(
float64
rFn
,
float64
rFm
)
static
float64
float64_rsf
(
struct
roundingData
*
roundData
,
float64
rFn
,
float64
rFm
)
{
return
float64_sub
(
rFm
,
rFn
);
return
float64_sub
(
r
oundData
,
r
Fm
,
rFn
);
}
static
float64
float64_rdv
(
float64
rFn
,
float64
rFm
)
static
float64
float64_rdv
(
struct
roundingData
*
roundData
,
float64
rFn
,
float64
rFm
)
{
return
float64_div
(
rFm
,
rFn
);
return
float64_div
(
r
oundData
,
r
Fm
,
rFn
);
}
static
float64
(
*
const
dyadic_double
[
16
])(
float64
rFn
,
float64
rFm
)
=
{
static
float64
(
*
const
dyadic_double
[
16
])(
struct
roundingData
*
,
float64
rFn
,
float64
rFm
)
=
{
[
ADF_CODE
>>
20
]
=
float64_add
,
[
MUF_CODE
>>
20
]
=
float64_mul
,
[
SUF_CODE
>>
20
]
=
float64_sub
,
...
...
@@ -65,12 +65,12 @@ static float64 (*const dyadic_double[16])(float64 rFn, float64 rFm) = {
[
FRD_CODE
>>
20
]
=
float64_rdv
,
};
static
float64
float64_mvf
(
float64
rFm
)
static
float64
float64_mvf
(
struct
roundingData
*
roundData
,
float64
rFm
)
{
return
rFm
;
}
static
float64
float64_mnf
(
float64
rFm
)
static
float64
float64_mnf
(
struct
roundingData
*
roundData
,
float64
rFm
)
{
union
float64_components
u
;
...
...
@@ -84,7 +84,7 @@ static float64 float64_mnf(float64 rFm)
return
u
.
f64
;
}
static
float64
float64_abs
(
float64
rFm
)
static
float64
float64_abs
(
struct
roundingData
*
roundData
,
float64
rFm
)
{
union
float64_components
u
;
...
...
@@ -98,7 +98,7 @@ static float64 float64_abs(float64 rFm)
return
u
.
f64
;
}
static
float64
(
*
const
monadic_double
[
16
])(
float64
rFm
)
=
{
static
float64
(
*
const
monadic_double
[
16
])(
struct
roundingData
*
,
float64
rFm
)
=
{
[
MVF_CODE
>>
20
]
=
float64_mvf
,
[
MNF_CODE
>>
20
]
=
float64_mnf
,
[
ABS_CODE
>>
20
]
=
float64_abs
,
...
...
@@ -108,7 +108,7 @@ static float64 (*const monadic_double[16])(float64 rFm) = {
[
NRM_CODE
>>
20
]
=
float64_mvf
,
};
unsigned
int
DoubleCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
)
unsigned
int
DoubleCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
)
{
FPA11
*
fpa11
=
GET_FPA11
();
float64
rFm
;
...
...
@@ -151,13 +151,13 @@ unsigned int DoubleCPDO(const unsigned int opcode, FPREG * rFd)
}
if
(
dyadic_double
[
opc_mask_shift
])
{
rFd
->
fDouble
=
dyadic_double
[
opc_mask_shift
](
rFn
,
rFm
);
rFd
->
fDouble
=
dyadic_double
[
opc_mask_shift
](
r
oundData
,
r
Fn
,
rFm
);
}
else
{
return
0
;
}
}
else
{
if
(
monadic_double
[
opc_mask_shift
])
{
rFd
->
fDouble
=
monadic_double
[
opc_mask_shift
](
rFm
);
rFd
->
fDouble
=
monadic_double
[
opc_mask_shift
](
r
oundData
,
r
Fm
);
}
else
{
return
0
;
}
...
...
arch/arm/nwfpe/extended_cpdo.c
View file @
194d0710
...
...
@@ -35,17 +35,17 @@ floatx80 floatx80_arccos(floatx80 rFm);
floatx80
floatx80_pow
(
floatx80
rFn
,
floatx80
rFm
);
floatx80
floatx80_pol
(
floatx80
rFn
,
floatx80
rFm
);
static
floatx80
floatx80_rsf
(
floatx80
rFn
,
floatx80
rFm
)
static
floatx80
floatx80_rsf
(
struct
roundingData
*
roundData
,
floatx80
rFn
,
floatx80
rFm
)
{
return
floatx80_sub
(
rFm
,
rFn
);
return
floatx80_sub
(
r
oundData
,
r
Fm
,
rFn
);
}
static
floatx80
floatx80_rdv
(
floatx80
rFn
,
floatx80
rFm
)
static
floatx80
floatx80_rdv
(
struct
roundingData
*
roundData
,
floatx80
rFn
,
floatx80
rFm
)
{
return
floatx80_div
(
rFm
,
rFn
);
return
floatx80_div
(
r
oundData
,
r
Fm
,
rFn
);
}
static
floatx80
(
*
const
dyadic_extended
[
16
])(
floatx80
rFn
,
floatx80
rFm
)
=
{
static
floatx80
(
*
const
dyadic_extended
[
16
])(
struct
roundingData
*
,
floatx80
rFn
,
floatx80
rFm
)
=
{
[
ADF_CODE
>>
20
]
=
floatx80_add
,
[
MUF_CODE
>>
20
]
=
floatx80_mul
,
[
SUF_CODE
>>
20
]
=
floatx80_sub
,
...
...
@@ -60,24 +60,24 @@ static floatx80 (*const dyadic_extended[16])(floatx80 rFn, floatx80 rFm) = {
[
FRD_CODE
>>
20
]
=
floatx80_rdv
,
};
static
floatx80
floatx80_mvf
(
floatx80
rFm
)
static
floatx80
floatx80_mvf
(
struct
roundingData
*
roundData
,
floatx80
rFm
)
{
return
rFm
;
}
static
floatx80
floatx80_mnf
(
floatx80
rFm
)
static
floatx80
floatx80_mnf
(
struct
roundingData
*
roundData
,
floatx80
rFm
)
{
rFm
.
high
^=
0x8000
;
return
rFm
;
}
static
floatx80
floatx80_abs
(
floatx80
rFm
)
static
floatx80
floatx80_abs
(
struct
roundingData
*
roundData
,
floatx80
rFm
)
{
rFm
.
high
&=
0x7fff
;
return
rFm
;
}
static
floatx80
(
*
const
monadic_extended
[
16
])(
floatx80
rFm
)
=
{
static
floatx80
(
*
const
monadic_extended
[
16
])(
struct
roundingData
*
,
floatx80
rFm
)
=
{
[
MVF_CODE
>>
20
]
=
floatx80_mvf
,
[
MNF_CODE
>>
20
]
=
floatx80_mnf
,
[
ABS_CODE
>>
20
]
=
floatx80_abs
,
...
...
@@ -87,7 +87,7 @@ static floatx80 (*const monadic_extended[16])(floatx80 rFm) = {
[
NRM_CODE
>>
20
]
=
floatx80_mvf
,
};
unsigned
int
ExtendedCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
)
unsigned
int
ExtendedCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
)
{
FPA11
*
fpa11
=
GET_FPA11
();
floatx80
rFm
;
...
...
@@ -138,13 +138,13 @@ unsigned int ExtendedCPDO(const unsigned int opcode, FPREG * rFd)
}
if
(
dyadic_extended
[
opc_mask_shift
])
{
rFd
->
fExtended
=
dyadic_extended
[
opc_mask_shift
](
rFn
,
rFm
);
rFd
->
fExtended
=
dyadic_extended
[
opc_mask_shift
](
r
oundData
,
r
Fn
,
rFm
);
}
else
{
return
0
;
}
}
else
{
if
(
monadic_extended
[
opc_mask_shift
])
{
rFd
->
fExtended
=
monadic_extended
[
opc_mask_shift
](
rFm
);
rFd
->
fExtended
=
monadic_extended
[
opc_mask_shift
](
r
oundData
,
r
Fm
);
}
else
{
return
0
;
}
...
...
arch/arm/nwfpe/fpa11.c
View file @
194d0710
...
...
@@ -51,48 +51,42 @@ static void resetFPA11(void)
fpa11
->
fpsr
=
FP_EMULATOR
|
BIT_AC
;
}
void
SetRoundingMode
(
const
unsigned
int
opcode
)
int8
SetRoundingMode
(
const
unsigned
int
opcode
)
{
switch
(
opcode
&
MASK_ROUNDING_MODE
)
{
default:
case
ROUND_TO_NEAREST
:
float_rounding_mode
=
float_round_nearest_even
;
break
;
return
float_round_nearest_even
;
case
ROUND_TO_PLUS_INFINITY
:
float_rounding_mode
=
float_round_up
;
break
;
return
float_round_up
;
case
ROUND_TO_MINUS_INFINITY
:
float_rounding_mode
=
float_round_down
;
break
;
return
float_round_down
;
case
ROUND_TO_ZERO
:
float_rounding_mode
=
float_round_to_zero
;
break
;
return
float_round_to_zero
;
}
}
void
SetRoundingPrecision
(
const
unsigned
int
opcode
)
int8
SetRoundingPrecision
(
const
unsigned
int
opcode
)
{
#ifdef CONFIG_FPE_NWFPE_XP
switch
(
opcode
&
MASK_ROUNDING_PRECISION
)
{
case
ROUND_SINGLE
:
floatx80_rounding_precision
=
32
;
break
;
return
32
;
case
ROUND_DOUBLE
:
floatx80_rounding_precision
=
64
;
break
;
return
64
;
case
ROUND_EXTENDED
:
floatx80_rounding_precision
=
80
;
break
;
return
80
;
default:
floatx80_rounding_precision
=
80
;
return
80
;
}
#endif
return
80
;
}
void
nwfpe_init_fpa
(
union
fp_state
*
fp
)
...
...
@@ -103,8 +97,6 @@ void nwfpe_init_fpa(union fp_state *fp)
#endif
memset
(
fpa11
,
0
,
sizeof
(
FPA11
));
resetFPA11
();
SetRoundingMode
(
ROUND_TO_NEAREST
);
SetRoundingPrecision
(
ROUND_EXTENDED
);
fpa11
->
initflag
=
1
;
}
...
...
arch/arm/nwfpe/fpa11.h
View file @
194d0710
...
...
@@ -37,6 +37,13 @@
/* includes */
#include "fpsr.h"
/* FP control and status register definitions */
#include "milieu.h"
struct
roundingData
{
int8
mode
;
int8
precision
;
signed
char
exception
;
};
#include "softfloat.h"
#define typeNone 0x00
...
...
@@ -84,8 +91,8 @@ typedef struct tagFPA11 {
initialised. */
}
FPA11
;
extern
void
SetRoundingMode
(
const
unsigned
int
);
extern
void
SetRoundingPrecision
(
const
unsigned
int
);
extern
int8
SetRoundingMode
(
const
unsigned
int
);
extern
int8
SetRoundingPrecision
(
const
unsigned
int
);
extern
void
nwfpe_init_fpa
(
union
fp_state
*
fp
);
#endif
arch/arm/nwfpe/fpa11_cpdo.c
View file @
194d0710
...
...
@@ -24,15 +24,16 @@
#include "fpa11.h"
#include "fpopcode.h"
unsigned
int
SingleCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
DoubleCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
ExtendedCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
SingleCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
DoubleCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
ExtendedCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
);
unsigned
int
EmulateCPDO
(
const
unsigned
int
opcode
)
{
FPA11
*
fpa11
=
GET_FPA11
();
FPREG
*
rFd
;
unsigned
int
nType
,
nDest
,
nRc
;
struct
roundingData
roundData
;
/* Get the destination size. If not valid let Linux perform
an invalid instruction trap. */
...
...
@@ -40,7 +41,9 @@ unsigned int EmulateCPDO(const unsigned int opcode)
if
(
typeNone
==
nDest
)
return
0
;
SetRoundingMode
(
opcode
);
roundData
.
mode
=
SetRoundingMode
(
opcode
);
roundData
.
precision
=
SetRoundingPrecision
(
opcode
);
roundData
.
exception
=
0
;
/* Compare the size of the operands in Fn and Fm.
Choose the largest size and perform operations in that size,
...
...
@@ -63,14 +66,14 @@ unsigned int EmulateCPDO(const unsigned int opcode)
switch
(
nType
)
{
case
typeSingle
:
nRc
=
SingleCPDO
(
opcode
,
rFd
);
nRc
=
SingleCPDO
(
&
roundData
,
opcode
,
rFd
);
break
;
case
typeDouble
:
nRc
=
DoubleCPDO
(
opcode
,
rFd
);
nRc
=
DoubleCPDO
(
&
roundData
,
opcode
,
rFd
);
break
;
#ifdef CONFIG_FPE_NWFPE_XP
case
typeExtended
:
nRc
=
ExtendedCPDO
(
opcode
,
rFd
);
nRc
=
ExtendedCPDO
(
&
roundData
,
opcode
,
rFd
);
break
;
#endif
default:
...
...
@@ -93,9 +96,9 @@ unsigned int EmulateCPDO(const unsigned int opcode)
case
typeSingle
:
{
if
(
typeDouble
==
nType
)
rFd
->
fSingle
=
float64_to_float32
(
rFd
->
fDouble
);
rFd
->
fSingle
=
float64_to_float32
(
&
roundData
,
rFd
->
fDouble
);
else
rFd
->
fSingle
=
floatx80_to_float32
(
rFd
->
fExtended
);
rFd
->
fSingle
=
floatx80_to_float32
(
&
roundData
,
rFd
->
fExtended
);
}
break
;
...
...
@@ -104,7 +107,7 @@ unsigned int EmulateCPDO(const unsigned int opcode)
if
(
typeSingle
==
nType
)
rFd
->
fDouble
=
float32_to_float64
(
rFd
->
fSingle
);
else
rFd
->
fDouble
=
floatx80_to_float64
(
rFd
->
fExtended
);
rFd
->
fDouble
=
floatx80_to_float64
(
&
roundData
,
rFd
->
fExtended
);
}
break
;
...
...
@@ -121,12 +124,15 @@ unsigned int EmulateCPDO(const unsigned int opcode)
#else
if
(
nDest
!=
nType
)
{
if
(
nDest
==
typeSingle
)
rFd
->
fSingle
=
float64_to_float32
(
rFd
->
fDouble
);
rFd
->
fSingle
=
float64_to_float32
(
&
roundData
,
rFd
->
fDouble
);
else
rFd
->
fDouble
=
float32_to_float64
(
rFd
->
fSingle
);
}
#endif
}
if
(
roundData
.
exception
)
float_raise
(
roundData
.
exception
);
return
nRc
;
}
arch/arm/nwfpe/fpa11_cpdt.c
View file @
194d0710
...
...
@@ -96,7 +96,7 @@ static inline void loadMultiple(const unsigned int Fn, const unsigned int __user
}
}
static
inline
void
storeSingle
(
const
unsigned
int
Fn
,
unsigned
int
__user
*
pMem
)
static
inline
void
storeSingle
(
struct
roundingData
*
roundData
,
const
unsigned
int
Fn
,
unsigned
int
__user
*
pMem
)
{
FPA11
*
fpa11
=
GET_FPA11
();
union
{
...
...
@@ -106,12 +106,12 @@ static inline void storeSingle(const unsigned int Fn, unsigned int __user *pMem)
switch
(
fpa11
->
fType
[
Fn
])
{
case
typeDouble
:
val
.
f
=
float64_to_float32
(
fpa11
->
fpreg
[
Fn
].
fDouble
);
val
.
f
=
float64_to_float32
(
roundData
,
fpa11
->
fpreg
[
Fn
].
fDouble
);
break
;
#ifdef CONFIG_FPE_NWFPE_XP
case
typeExtended
:
val
.
f
=
floatx80_to_float32
(
fpa11
->
fpreg
[
Fn
].
fExtended
);
val
.
f
=
floatx80_to_float32
(
roundData
,
fpa11
->
fpreg
[
Fn
].
fExtended
);
break
;
#endif
...
...
@@ -122,7 +122,7 @@ static inline void storeSingle(const unsigned int Fn, unsigned int __user *pMem)
put_user
(
val
.
i
[
0
],
pMem
);
}
static
inline
void
storeDouble
(
const
unsigned
int
Fn
,
unsigned
int
__user
*
pMem
)
static
inline
void
storeDouble
(
struct
roundingData
*
roundData
,
const
unsigned
int
Fn
,
unsigned
int
__user
*
pMem
)
{
FPA11
*
fpa11
=
GET_FPA11
();
union
{
...
...
@@ -137,7 +137,7 @@ static inline void storeDouble(const unsigned int Fn, unsigned int __user *pMem)
#ifdef CONFIG_FPE_NWFPE_XP
case
typeExtended
:
val
.
f
=
floatx80_to_float64
(
fpa11
->
fpreg
[
Fn
].
fExtended
);
val
.
f
=
floatx80_to_float64
(
roundData
,
fpa11
->
fpreg
[
Fn
].
fExtended
);
break
;
#endif
...
...
@@ -259,8 +259,11 @@ unsigned int PerformSTF(const unsigned int opcode)
{
unsigned
int
__user
*
pBase
,
*
pAddress
,
*
pFinal
;
unsigned
int
nRc
=
1
,
write_back
=
WRITE_BACK
(
opcode
);
struct
roundingData
roundData
;
SetRoundingMode
(
ROUND_TO_NEAREST
);
roundData
.
mode
=
SetRoundingMode
(
opcode
);
roundData
.
precision
=
SetRoundingPrecision
(
opcode
);
roundData
.
exception
=
0
;
pBase
=
(
unsigned
int
__user
*
)
readRegister
(
getRn
(
opcode
));
if
(
REG_PC
==
getRn
(
opcode
))
{
...
...
@@ -281,10 +284,10 @@ unsigned int PerformSTF(const unsigned int opcode)
switch
(
opcode
&
MASK_TRANSFER_LENGTH
)
{
case
TRANSFER_SINGLE
:
storeSingle
(
getFd
(
opcode
),
pAddress
);
storeSingle
(
&
roundData
,
getFd
(
opcode
),
pAddress
);
break
;
case
TRANSFER_DOUBLE
:
storeDouble
(
getFd
(
opcode
),
pAddress
);
storeDouble
(
&
roundData
,
getFd
(
opcode
),
pAddress
);
break
;
#ifdef CONFIG_FPE_NWFPE_XP
case
TRANSFER_EXTENDED
:
...
...
@@ -295,6 +298,9 @@ unsigned int PerformSTF(const unsigned int opcode)
nRc
=
0
;
}
if
(
roundData
.
exception
)
float_raise
(
roundData
.
exception
);
if
(
write_back
)
writeRegister
(
getRn
(
opcode
),
(
unsigned
long
)
pFinal
);
return
nRc
;
...
...
arch/arm/nwfpe/fpa11_cprt.c
View file @
194d0710
...
...
@@ -33,8 +33,6 @@ extern flag floatx80_is_nan(floatx80);
extern
flag
float64_is_nan
(
float64
);
extern
flag
float32_is_nan
(
float32
);
void
SetRoundingMode
(
const
unsigned
int
opcode
);
unsigned
int
PerformFLT
(
const
unsigned
int
opcode
);
unsigned
int
PerformFIX
(
const
unsigned
int
opcode
);
...
...
@@ -77,14 +75,17 @@ unsigned int EmulateCPRT(const unsigned int opcode)
unsigned
int
PerformFLT
(
const
unsigned
int
opcode
)
{
FPA11
*
fpa11
=
GET_FPA11
();
SetRoundingMode
(
opcode
);
SetRoundingPrecision
(
opcode
);
struct
roundingData
roundData
;
roundData
.
mode
=
SetRoundingMode
(
opcode
);
roundData
.
precision
=
SetRoundingPrecision
(
opcode
);
roundData
.
exception
=
0
;
switch
(
opcode
&
MASK_ROUNDING_PRECISION
)
{
case
ROUND_SINGLE
:
{
fpa11
->
fType
[
getFn
(
opcode
)]
=
typeSingle
;
fpa11
->
fpreg
[
getFn
(
opcode
)].
fSingle
=
int32_to_float32
(
readRegister
(
getRd
(
opcode
)));
fpa11
->
fpreg
[
getFn
(
opcode
)].
fSingle
=
int32_to_float32
(
&
roundData
,
readRegister
(
getRd
(
opcode
)));
}
break
;
...
...
@@ -108,6 +109,9 @@ unsigned int PerformFLT(const unsigned int opcode)
return
0
;
}
if
(
roundData
.
exception
)
float_raise
(
roundData
.
exception
);
return
1
;
}
...
...
@@ -115,26 +119,29 @@ unsigned int PerformFIX(const unsigned int opcode)
{
FPA11
*
fpa11
=
GET_FPA11
();
unsigned
int
Fn
=
getFm
(
opcode
);
struct
roundingData
roundData
;
SetRoundingMode
(
opcode
);
roundData
.
mode
=
SetRoundingMode
(
opcode
);
roundData
.
precision
=
SetRoundingPrecision
(
opcode
);
roundData
.
exception
=
0
;
switch
(
fpa11
->
fType
[
Fn
])
{
case
typeSingle
:
{
writeRegister
(
getRd
(
opcode
),
float32_to_int32
(
fpa11
->
fpreg
[
Fn
].
fSingle
));
writeRegister
(
getRd
(
opcode
),
float32_to_int32
(
&
roundData
,
fpa11
->
fpreg
[
Fn
].
fSingle
));
}
break
;
case
typeDouble
:
{
writeRegister
(
getRd
(
opcode
),
float64_to_int32
(
fpa11
->
fpreg
[
Fn
].
fDouble
));
writeRegister
(
getRd
(
opcode
),
float64_to_int32
(
&
roundData
,
fpa11
->
fpreg
[
Fn
].
fDouble
));
}
break
;
#ifdef CONFIG_FPE_NWFPE_XP
case
typeExtended
:
{
writeRegister
(
getRd
(
opcode
),
floatx80_to_int32
(
fpa11
->
fpreg
[
Fn
].
fExtended
));
writeRegister
(
getRd
(
opcode
),
floatx80_to_int32
(
&
roundData
,
fpa11
->
fpreg
[
Fn
].
fExtended
));
}
break
;
#endif
...
...
@@ -143,6 +150,9 @@ unsigned int PerformFIX(const unsigned int opcode)
return
0
;
}
if
(
roundData
.
exception
)
float_raise
(
roundData
.
exception
);
return
1
;
}
...
...
arch/arm/nwfpe/fpmodule.c
View file @
194d0710
...
...
@@ -116,8 +116,6 @@ fpmodule.c to integrate with the NetBSD kernel (I hope!).
code to access data in user space in some other source files at the
moment (grep for get_user / put_user calls). --philb]
float_exception_flags is a global variable in SoftFloat.
This function is called by the SoftFloat routines to raise a floating
point exception. We check the trap enable byte in the FPSR, and raise
a SIGFPE exception if necessary. If not the relevant bits in the
...
...
@@ -129,15 +127,14 @@ void float_raise(signed char flags)
register
unsigned
int
fpsr
,
cumulativeTraps
;
#ifdef CONFIG_DEBUG_USER
/* Ignore inexact errors as there are far too many of them to log */
if
(
flags
&
~
BIT_IXC
)
printk
(
KERN_DEBUG
"NWFPE: %s[%d] takes exception %08x at %p from %08lx
\n
"
,
current
->
comm
,
current
->
pid
,
flags
,
__builtin_return_address
(
0
),
GET_USERREG
()
->
ARM_pc
);
#endif
/* Keep SoftFloat exception flags up to date. */
float_exception_flags
|=
flags
;
/* Read fpsr and initialize the cumulativeTraps. */
fpsr
=
readFPSR
();
cumulativeTraps
=
0
;
...
...
arch/arm/nwfpe/single_cpdo.c
View file @
194d0710
...
...
@@ -36,17 +36,17 @@ float32 float32_arccos(float32 rFm);
float32
float32_pow
(
float32
rFn
,
float32
rFm
);
float32
float32_pol
(
float32
rFn
,
float32
rFm
);
static
float32
float32_rsf
(
float32
rFn
,
float32
rFm
)
static
float32
float32_rsf
(
struct
roundingData
*
roundData
,
float32
rFn
,
float32
rFm
)
{
return
float32_sub
(
rFm
,
rFn
);
return
float32_sub
(
r
oundData
,
r
Fm
,
rFn
);
}
static
float32
float32_rdv
(
float32
rFn
,
float32
rFm
)
static
float32
float32_rdv
(
struct
roundingData
*
roundData
,
float32
rFn
,
float32
rFm
)
{
return
float32_div
(
rFm
,
rFn
);
return
float32_div
(
r
oundData
,
r
Fm
,
rFn
);
}
static
float32
(
*
const
dyadic_single
[
16
])(
float32
rFn
,
float32
rFm
)
=
{
static
float32
(
*
const
dyadic_single
[
16
])(
struct
roundingData
*
,
float32
rFn
,
float32
rFm
)
=
{
[
ADF_CODE
>>
20
]
=
float32_add
,
[
MUF_CODE
>>
20
]
=
float32_mul
,
[
SUF_CODE
>>
20
]
=
float32_sub
,
...
...
@@ -60,22 +60,22 @@ static float32 (*const dyadic_single[16])(float32 rFn, float32 rFm) = {
[
FRD_CODE
>>
20
]
=
float32_rdv
,
};
static
float32
float32_mvf
(
float32
rFm
)
static
float32
float32_mvf
(
struct
roundingData
*
roundData
,
float32
rFm
)
{
return
rFm
;
}
static
float32
float32_mnf
(
float32
rFm
)
static
float32
float32_mnf
(
struct
roundingData
*
roundData
,
float32
rFm
)
{
return
rFm
^
0x80000000
;
}
static
float32
float32_abs
(
float32
rFm
)
static
float32
float32_abs
(
struct
roundingData
*
roundData
,
float32
rFm
)
{
return
rFm
&
0x7fffffff
;
}
static
float32
(
*
const
monadic_single
[
16
])(
float32
rFm
)
=
{
static
float32
(
*
const
monadic_single
[
16
])(
struct
roundingData
*
,
float32
rFm
)
=
{
[
MVF_CODE
>>
20
]
=
float32_mvf
,
[
MNF_CODE
>>
20
]
=
float32_mnf
,
[
ABS_CODE
>>
20
]
=
float32_abs
,
...
...
@@ -85,7 +85,7 @@ static float32 (*const monadic_single[16])(float32 rFm) = {
[
NRM_CODE
>>
20
]
=
float32_mvf
,
};
unsigned
int
SingleCPDO
(
const
unsigned
int
opcode
,
FPREG
*
rFd
)
unsigned
int
SingleCPDO
(
struct
roundingData
*
roundData
,
const
unsigned
int
opcode
,
FPREG
*
rFd
)
{
FPA11
*
fpa11
=
GET_FPA11
();
float32
rFm
;
...
...
@@ -108,13 +108,13 @@ unsigned int SingleCPDO(const unsigned int opcode, FPREG * rFd)
if
(
fpa11
->
fType
[
Fn
]
==
typeSingle
&&
dyadic_single
[
opc_mask_shift
])
{
rFn
=
fpa11
->
fpreg
[
Fn
].
fSingle
;
rFd
->
fSingle
=
dyadic_single
[
opc_mask_shift
](
rFn
,
rFm
);
rFd
->
fSingle
=
dyadic_single
[
opc_mask_shift
](
r
oundData
,
r
Fn
,
rFm
);
}
else
{
return
0
;
}
}
else
{
if
(
monadic_single
[
opc_mask_shift
])
{
rFd
->
fSingle
=
monadic_single
[
opc_mask_shift
](
rFm
);
rFd
->
fSingle
=
monadic_single
[
opc_mask_shift
](
r
oundData
,
r
Fm
);
}
else
{
return
0
;
}
...
...
arch/arm/nwfpe/softfloat.c
View file @
194d0710
...
...
@@ -34,16 +34,6 @@ this code that are retained.
//#include "milieu.h"
//#include "softfloat.h"
/*
-------------------------------------------------------------------------------
Floating-point rounding mode, extended double-precision rounding precision,
and exception flags.
-------------------------------------------------------------------------------
*/
int8
float_rounding_mode
=
float_round_nearest_even
;
int8
floatx80_rounding_precision
=
80
;
int8
float_exception_flags
;
/*
-------------------------------------------------------------------------------
Primitive arithmetic functions, including multi-word arithmetic, and
...
...
@@ -77,14 +67,14 @@ input is too large, however, the invalid exception is raised and the largest
positive or negative integer is returned.
-------------------------------------------------------------------------------
*/
static
int32
roundAndPackInt32
(
flag
zSign
,
bits64
absZ
)
static
int32
roundAndPackInt32
(
struct
roundingData
*
roundData
,
flag
zSign
,
bits64
absZ
)
{
int8
roundingMode
;
flag
roundNearestEven
;
int8
roundIncrement
,
roundBits
;
int32
z
;
roundingMode
=
float_rounding_
mode
;
roundingMode
=
roundData
->
mode
;
roundNearestEven
=
(
roundingMode
==
float_round_nearest_even
);
roundIncrement
=
0x40
;
if
(
!
roundNearestEven
)
{
...
...
@@ -107,10 +97,10 @@ static int32 roundAndPackInt32( flag zSign, bits64 absZ )
z
=
absZ
;
if
(
zSign
)
z
=
-
z
;
if
(
(
absZ
>>
32
)
||
(
z
&&
(
(
z
<
0
)
^
zSign
)
)
)
{
float_exception_flags
|=
float_flag_invalid
;
roundData
->
exception
|=
float_flag_invalid
;
return
zSign
?
0x80000000
:
0x7FFFFFFF
;
}
if
(
roundBits
)
float_exception_flags
|=
float_flag_inexact
;
if
(
roundBits
)
roundData
->
exception
|=
float_flag_inexact
;
return
z
;
}
...
...
@@ -224,14 +214,14 @@ The handling of underflow and overflow follows the IEC/IEEE Standard for
Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float32
roundAndPackFloat32
(
flag
zSign
,
int16
zExp
,
bits32
zSig
)
static
float32
roundAndPackFloat32
(
struct
roundingData
*
roundData
,
flag
zSign
,
int16
zExp
,
bits32
zSig
)
{
int8
roundingMode
;
flag
roundNearestEven
;
int8
roundIncrement
,
roundBits
;
flag
isTiny
;
roundingMode
=
float_rounding_
mode
;
roundingMode
=
roundData
->
mode
;
roundNearestEven
=
(
roundingMode
==
float_round_nearest_even
);
roundIncrement
=
0x40
;
if
(
!
roundNearestEven
)
{
...
...
@@ -254,7 +244,7 @@ static float32 roundAndPackFloat32( flag zSign, int16 zExp, bits32 zSig )
||
(
(
zExp
==
0xFD
)
&&
(
(
sbits32
)
(
zSig
+
roundIncrement
)
<
0
)
)
)
{
float_raise
(
float_flag_overflow
|
float_flag_inexact
)
;
roundData
->
exception
|=
float_flag_overflow
|
float_flag_inexact
;
return
packFloat32
(
zSign
,
0xFF
,
0
)
-
(
roundIncrement
==
0
);
}
if
(
zExp
<
0
)
{
...
...
@@ -265,10 +255,10 @@ static float32 roundAndPackFloat32( flag zSign, int16 zExp, bits32 zSig )
shift32RightJamming
(
zSig
,
-
zExp
,
&
zSig
);
zExp
=
0
;
roundBits
=
zSig
&
0x7F
;
if
(
isTiny
&&
roundBits
)
float_raise
(
float_flag_underflow
)
;
if
(
isTiny
&&
roundBits
)
roundData
->
exception
|=
float_flag_underflow
;
}
}
if
(
roundBits
)
float_exception_flags
|=
float_flag_inexact
;
if
(
roundBits
)
roundData
->
exception
|=
float_flag_inexact
;
zSig
=
(
zSig
+
roundIncrement
)
>>
7
;
zSig
&=
~
(
(
(
roundBits
^
0x40
)
==
0
)
&
roundNearestEven
);
if
(
zSig
==
0
)
zExp
=
0
;
...
...
@@ -287,12 +277,12 @@ point exponent.
-------------------------------------------------------------------------------
*/
static
float32
normalizeRoundAndPackFloat32
(
flag
zSign
,
int16
zExp
,
bits32
zSig
)
normalizeRoundAndPackFloat32
(
struct
roundingData
*
roundData
,
flag
zSign
,
int16
zExp
,
bits32
zSig
)
{
int8
shiftCount
;
shiftCount
=
countLeadingZeros32
(
zSig
)
-
1
;
return
roundAndPackFloat32
(
zSign
,
zExp
-
shiftCount
,
zSig
<<
shiftCount
);
return
roundAndPackFloat32
(
roundData
,
zSign
,
zExp
-
shiftCount
,
zSig
<<
shiftCount
);
}
...
...
@@ -395,14 +385,14 @@ The handling of underflow and overflow follows the IEC/IEEE Standard for
Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float64
roundAndPackFloat64
(
flag
zSign
,
int16
zExp
,
bits64
zSig
)
static
float64
roundAndPackFloat64
(
struct
roundingData
*
roundData
,
flag
zSign
,
int16
zExp
,
bits64
zSig
)
{
int8
roundingMode
;
flag
roundNearestEven
;
int16
roundIncrement
,
roundBits
;
flag
isTiny
;
roundingMode
=
float_rounding_
mode
;
roundingMode
=
roundData
->
mode
;
roundNearestEven
=
(
roundingMode
==
float_round_nearest_even
);
roundIncrement
=
0x200
;
if
(
!
roundNearestEven
)
{
...
...
@@ -427,7 +417,7 @@ static float64 roundAndPackFloat64( flag zSign, int16 zExp, bits64 zSig )
)
{
//register int lr = __builtin_return_address(0);
//printk("roundAndPackFloat64 called from 0x%08x\n",lr);
float_raise
(
float_flag_overflow
|
float_flag_inexact
)
;
roundData
->
exception
|=
float_flag_overflow
|
float_flag_inexact
;
return
packFloat64
(
zSign
,
0x7FF
,
0
)
-
(
roundIncrement
==
0
);
}
if
(
zExp
<
0
)
{
...
...
@@ -438,10 +428,10 @@ static float64 roundAndPackFloat64( flag zSign, int16 zExp, bits64 zSig )
shift64RightJamming
(
zSig
,
-
zExp
,
&
zSig
);
zExp
=
0
;
roundBits
=
zSig
&
0x3FF
;
if
(
isTiny
&&
roundBits
)
float_raise
(
float_flag_underflow
)
;
if
(
isTiny
&&
roundBits
)
roundData
->
exception
|=
float_flag_underflow
;
}
}
if
(
roundBits
)
float_exception_flags
|=
float_flag_inexact
;
if
(
roundBits
)
roundData
->
exception
|=
float_flag_inexact
;
zSig
=
(
zSig
+
roundIncrement
)
>>
10
;
zSig
&=
~
(
(
(
roundBits
^
0x200
)
==
0
)
&
roundNearestEven
);
if
(
zSig
==
0
)
zExp
=
0
;
...
...
@@ -460,12 +450,12 @@ point exponent.
-------------------------------------------------------------------------------
*/
static
float64
normalizeRoundAndPackFloat64
(
flag
zSign
,
int16
zExp
,
bits64
zSig
)
normalizeRoundAndPackFloat64
(
struct
roundingData
*
roundData
,
flag
zSign
,
int16
zExp
,
bits64
zSig
)
{
int8
shiftCount
;
shiftCount
=
countLeadingZeros64
(
zSig
)
-
1
;
return
roundAndPackFloat64
(
zSign
,
zExp
-
shiftCount
,
zSig
<<
shiftCount
);
return
roundAndPackFloat64
(
roundData
,
zSign
,
zExp
-
shiftCount
,
zSig
<<
shiftCount
);
}
...
...
@@ -572,14 +562,15 @@ Floating-point Arithmetic.
*/
static
floatx80
roundAndPackFloatx80
(
int8
roundingPrecision
,
flag
zSign
,
int32
zExp
,
bits64
zSig0
,
bits64
zSig1
struct
roundingData
*
roundData
,
flag
zSign
,
int32
zExp
,
bits64
zSig0
,
bits64
zSig1
)
{
int8
roundingMode
;
int8
roundingMode
,
roundingPrecision
;
flag
roundNearestEven
,
increment
,
isTiny
;
int64
roundIncrement
,
roundMask
,
roundBits
;
roundingMode
=
float_rounding_mode
;
roundingMode
=
roundData
->
mode
;
roundingPrecision
=
roundData
->
precision
;
roundNearestEven
=
(
roundingMode
==
float_round_nearest_even
);
if
(
roundingPrecision
==
80
)
goto
precision80
;
if
(
roundingPrecision
==
64
)
{
...
...
@@ -623,8 +614,8 @@ static floatx80
shift64RightJamming
(
zSig0
,
1
-
zExp
,
&
zSig0
);
zExp
=
0
;
roundBits
=
zSig0
&
roundMask
;
if
(
isTiny
&&
roundBits
)
float_raise
(
float_flag_underflow
)
;
if
(
roundBits
)
float_exception_flags
|=
float_flag_inexact
;
if
(
isTiny
&&
roundBits
)
roundData
->
exception
|=
float_flag_underflow
;
if
(
roundBits
)
roundData
->
exception
|=
float_flag_inexact
;
zSig0
+=
roundIncrement
;
if
(
(
sbits64
)
zSig0
<
0
)
zExp
=
1
;
roundIncrement
=
roundMask
+
1
;
...
...
@@ -635,7 +626,7 @@ static floatx80
return
packFloatx80
(
zSign
,
zExp
,
zSig0
);
}
}
if
(
roundBits
)
float_exception_flags
|=
float_flag_inexact
;
if
(
roundBits
)
roundData
->
exception
|=
float_flag_inexact
;
zSig0
+=
roundIncrement
;
if
(
zSig0
<
roundIncrement
)
{
++
zExp
;
...
...
@@ -672,7 +663,7 @@ static floatx80
)
{
roundMask
=
0
;
overflow:
float_raise
(
float_flag_overflow
|
float_flag_inexact
)
;
roundData
->
exception
|=
float_flag_overflow
|
float_flag_inexact
;
if
(
(
roundingMode
==
float_round_to_zero
)
||
(
zSign
&&
(
roundingMode
==
float_round_up
)
)
||
(
!
zSign
&&
(
roundingMode
==
float_round_down
)
)
...
...
@@ -689,8 +680,8 @@ static floatx80
||
(
zSig0
<
LIT64
(
0xFFFFFFFFFFFFFFFF
)
);
shift64ExtraRightJamming
(
zSig0
,
zSig1
,
1
-
zExp
,
&
zSig0
,
&
zSig1
);
zExp
=
0
;
if
(
isTiny
&&
zSig1
)
float_raise
(
float_flag_underflow
)
;
if
(
zSig1
)
float_exception_flags
|=
float_flag_inexact
;
if
(
isTiny
&&
zSig1
)
roundData
->
exception
|=
float_flag_underflow
;
if
(
zSig1
)
roundData
->
exception
|=
float_flag_inexact
;
if
(
roundNearestEven
)
{
increment
=
(
(
sbits64
)
zSig1
<
0
);
}
...
...
@@ -710,7 +701,7 @@ static floatx80
return
packFloatx80
(
zSign
,
zExp
,
zSig0
);
}
}
if
(
zSig1
)
float_exception_flags
|=
float_flag_inexact
;
if
(
zSig1
)
roundData
->
exception
|=
float_flag_inexact
;
if
(
increment
)
{
++
zSig0
;
if
(
zSig0
==
0
)
{
...
...
@@ -740,7 +731,7 @@ normalized.
*/
static
floatx80
normalizeRoundAndPackFloatx80
(
int8
roundingPrecision
,
flag
zSign
,
int32
zExp
,
bits64
zSig0
,
bits64
zSig1
struct
roundingData
*
roundData
,
flag
zSign
,
int32
zExp
,
bits64
zSig0
,
bits64
zSig1
)
{
int8
shiftCount
;
...
...
@@ -754,7 +745,7 @@ static floatx80
shortShift128Left
(
zSig0
,
zSig1
,
shiftCount
,
&
zSig0
,
&
zSig1
);
zExp
-=
shiftCount
;
return
roundAndPackFloatx80
(
round
ingPrecision
,
zSign
,
zExp
,
zSig0
,
zSig1
);
roundAndPackFloatx80
(
round
Data
,
zSign
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -767,14 +758,14 @@ the single-precision floating-point format. The conversion is performed
according to the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
int32_to_float32
(
int32
a
)
float32
int32_to_float32
(
struct
roundingData
*
roundData
,
int32
a
)
{
flag
zSign
;
if
(
a
==
0
)
return
0
;
if
(
a
==
0x80000000
)
return
packFloat32
(
1
,
0x9E
,
0
);
zSign
=
(
a
<
0
);
return
normalizeRoundAndPackFloat32
(
zSign
,
0x9C
,
zSign
?
-
a
:
a
);
return
normalizeRoundAndPackFloat32
(
roundData
,
zSign
,
0x9C
,
zSign
?
-
a
:
a
);
}
...
...
@@ -840,7 +831,7 @@ positive integer is returned. Otherwise, if the conversion overflows, the
largest integer with the same sign as `a' is returned.
-------------------------------------------------------------------------------
*/
int32
float32_to_int32
(
float32
a
)
int32
float32_to_int32
(
struct
roundingData
*
roundData
,
float32
a
)
{
flag
aSign
;
int16
aExp
,
shiftCount
;
...
...
@@ -856,7 +847,7 @@ int32 float32_to_int32( float32 a )
zSig
=
aSig
;
zSig
<<=
32
;
if
(
0
<
shiftCount
)
shift64RightJamming
(
zSig
,
shiftCount
,
&
zSig
);
return
roundAndPackInt32
(
aSign
,
zSig
);
return
roundAndPackInt32
(
roundData
,
aSign
,
zSig
);
}
...
...
@@ -889,13 +880,13 @@ int32 float32_to_int32_round_to_zero( float32 a )
return
0x80000000
;
}
else
if
(
aExp
<=
0x7E
)
{
if
(
aExp
|
aSig
)
float_
exception_flags
|=
float_flag_inexact
;
if
(
aExp
|
aSig
)
float_
raise
(
float_flag_inexact
)
;
return
0
;
}
aSig
=
(
aSig
|
0x00800000
)
<<
8
;
z
=
aSig
>>
(
-
shiftCount
);
if
(
(
bits32
)
(
aSig
<<
(
shiftCount
&
31
)
)
)
{
float_
exception_flags
|=
float_flag_inexact
;
float_
raise
(
float_flag_inexact
)
;
}
return
aSign
?
-
z
:
z
;
...
...
@@ -973,7 +964,7 @@ operation is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_round_to_int
(
float32
a
)
float32
float32_round_to_int
(
struct
roundingData
*
roundData
,
float32
a
)
{
flag
aSign
;
int16
aExp
;
...
...
@@ -988,11 +979,12 @@ float32 float32_round_to_int( float32 a )
}
return
a
;
}
roundingMode
=
roundData
->
mode
;
if
(
aExp
<=
0x7E
)
{
if
(
(
bits32
)
(
a
<<
1
)
==
0
)
return
a
;
float_exception_flags
|=
float_flag_inexact
;
roundData
->
exception
|=
float_flag_inexact
;
aSign
=
extractFloat32Sign
(
a
);
switch
(
float_rounding_m
ode
)
{
switch
(
roundingM
ode
)
{
case
float_round_nearest_even
:
if
(
(
aExp
==
0x7E
)
&&
extractFloat32Frac
(
a
)
)
{
return
packFloat32
(
aSign
,
0x7F
,
0
);
...
...
@@ -1009,7 +1001,6 @@ float32 float32_round_to_int( float32 a )
lastBitMask
<<=
0x96
-
aExp
;
roundBitsMask
=
lastBitMask
-
1
;
z
=
a
;
roundingMode
=
float_rounding_mode
;
if
(
roundingMode
==
float_round_nearest_even
)
{
z
+=
lastBitMask
>>
1
;
if
(
(
z
&
roundBitsMask
)
==
0
)
z
&=
~
lastBitMask
;
...
...
@@ -1020,7 +1011,7 @@ float32 float32_round_to_int( float32 a )
}
}
z
&=
~
roundBitsMask
;
if
(
z
!=
a
)
float_exception_flags
|=
float_flag_inexact
;
if
(
z
!=
a
)
roundData
->
exception
|=
float_flag_inexact
;
return
z
;
}
...
...
@@ -1034,7 +1025,7 @@ addition is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float32
addFloat32Sigs
(
float32
a
,
float32
b
,
flag
zSign
)
static
float32
addFloat32Sigs
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
,
flag
zSign
)
{
int16
aExp
,
bExp
,
zExp
;
bits32
aSig
,
bSig
,
zSig
;
...
...
@@ -1093,7 +1084,7 @@ static float32 addFloat32Sigs( float32 a, float32 b, flag zSign )
++
zExp
;
}
roundAndPack:
return
roundAndPackFloat32
(
zSign
,
zExp
,
zSig
);
return
roundAndPackFloat32
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -1106,7 +1097,7 @@ result is a NaN. The subtraction is performed according to the IEC/IEEE
Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float32
subFloat32Sigs
(
float32
a
,
float32
b
,
flag
zSign
)
static
float32
subFloat32Sigs
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
,
flag
zSign
)
{
int16
aExp
,
bExp
,
zExp
;
bits32
aSig
,
bSig
,
zSig
;
...
...
@@ -1123,7 +1114,7 @@ static float32 subFloat32Sigs( float32 a, float32 b, flag zSign )
if
(
expDiff
<
0
)
goto
bExpBigger
;
if
(
aExp
==
0xFF
)
{
if
(
aSig
|
bSig
)
return
propagateFloat32NaN
(
a
,
b
);
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
if
(
aExp
==
0
)
{
...
...
@@ -1132,7 +1123,7 @@ static float32 subFloat32Sigs( float32 a, float32 b, flag zSign )
}
if
(
bSig
<
aSig
)
goto
aBigger
;
if
(
aSig
<
bSig
)
goto
bBigger
;
return
packFloat32
(
float_rounding_
mode
==
float_round_down
,
0
,
0
);
return
packFloat32
(
roundData
->
mode
==
float_round_down
,
0
,
0
);
bExpBigger:
if
(
bExp
==
0xFF
)
{
if
(
bSig
)
return
propagateFloat32NaN
(
a
,
b
);
...
...
@@ -1169,7 +1160,7 @@ static float32 subFloat32Sigs( float32 a, float32 b, flag zSign )
zExp
=
aExp
;
normalizeRoundAndPack:
--
zExp
;
return
normalizeRoundAndPackFloat32
(
zSign
,
zExp
,
zSig
);
return
normalizeRoundAndPackFloat32
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -1180,17 +1171,17 @@ and `b'. The operation is performed according to the IEC/IEEE Standard for
Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_add
(
float32
a
,
float32
b
)
float32
float32_add
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloat32Sign
(
a
);
bSign
=
extractFloat32Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
addFloat32Sigs
(
a
,
b
,
aSign
);
return
addFloat32Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
subFloat32Sigs
(
a
,
b
,
aSign
);
return
subFloat32Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -1202,17 +1193,17 @@ Returns the result of subtracting the single-precision floating-point values
for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_sub
(
float32
a
,
float32
b
)
float32
float32_sub
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloat32Sign
(
a
);
bSign
=
extractFloat32Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
subFloat32Sigs
(
a
,
b
,
aSign
);
return
subFloat32Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
addFloat32Sigs
(
a
,
b
,
aSign
);
return
addFloat32Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -1224,7 +1215,7 @@ Returns the result of multiplying the single-precision floating-point values
for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_mul
(
float32
a
,
float32
b
)
float32
float32_mul
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
zExp
;
...
...
@@ -1244,7 +1235,7 @@ float32 float32_mul( float32 a, float32 b )
return
propagateFloat32NaN
(
a
,
b
);
}
if
(
(
bExp
|
bSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
return
packFloat32
(
zSign
,
0xFF
,
0
);
...
...
@@ -1252,7 +1243,7 @@ float32 float32_mul( float32 a, float32 b )
if
(
bExp
==
0xFF
)
{
if
(
bSig
)
return
propagateFloat32NaN
(
a
,
b
);
if
(
(
aExp
|
aSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
return
packFloat32
(
zSign
,
0xFF
,
0
);
...
...
@@ -1274,7 +1265,7 @@ float32 float32_mul( float32 a, float32 b )
zSig
<<=
1
;
--
zExp
;
}
return
roundAndPackFloat32
(
zSign
,
zExp
,
zSig
);
return
roundAndPackFloat32
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -1285,7 +1276,7 @@ by the corresponding value `b'. The operation is performed according to the
IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_div
(
float32
a
,
float32
b
)
float32
float32_div
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
zExp
;
...
...
@@ -1302,7 +1293,7 @@ float32 float32_div( float32 a, float32 b )
if
(
aSig
)
return
propagateFloat32NaN
(
a
,
b
);
if
(
bExp
==
0xFF
)
{
if
(
bSig
)
return
propagateFloat32NaN
(
a
,
b
);
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
return
packFloat32
(
zSign
,
0xFF
,
0
);
...
...
@@ -1314,10 +1305,10 @@ float32 float32_div( float32 a, float32 b )
if
(
bExp
==
0
)
{
if
(
bSig
==
0
)
{
if
(
(
aExp
|
aSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
float_raise
(
float_flag_divbyzero
)
;
roundData
->
exception
|=
float_flag_divbyzero
;
return
packFloat32
(
zSign
,
0xFF
,
0
);
}
normalizeFloat32Subnormal
(
bSig
,
&
bExp
,
&
bSig
);
...
...
@@ -1341,7 +1332,7 @@ float32 float32_div( float32 a, float32 b )
if
(
(
zSig
&
0x3F
)
==
0
)
{
zSig
|=
(
(
(
bits64
)
bSig
)
*
zSig
!=
(
(
bits64
)
aSig
)
<<
32
);
}
return
roundAndPackFloat32
(
zSign
,
zExp
,
zSig
);
return
roundAndPackFloat32
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -1352,7 +1343,7 @@ with respect to the corresponding value `b'. The operation is performed
according to the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_rem
(
float32
a
,
float32
b
)
float32
float32_rem
(
struct
roundingData
*
roundData
,
float32
a
,
float32
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
expDiff
;
...
...
@@ -1372,7 +1363,7 @@ float32 float32_rem( float32 a, float32 b )
if
(
aSig
||
(
(
bExp
==
0xFF
)
&&
bSig
)
)
{
return
propagateFloat32NaN
(
a
,
b
);
}
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
if
(
bExp
==
0xFF
)
{
...
...
@@ -1381,7 +1372,7 @@ float32 float32_rem( float32 a, float32 b )
}
if
(
bExp
==
0
)
{
if
(
bSig
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
normalizeFloat32Subnormal
(
bSig
,
&
bExp
,
&
bSig
);
...
...
@@ -1444,7 +1435,7 @@ float32 float32_rem( float32 a, float32 b )
}
zSign
=
(
(
sbits32
)
aSig
<
0
);
if
(
zSign
)
aSig
=
-
aSig
;
return
normalizeRoundAndPackFloat32
(
aSign
^
zSign
,
bExp
,
aSig
);
return
normalizeRoundAndPackFloat32
(
roundData
,
aSign
^
zSign
,
bExp
,
aSig
);
}
...
...
@@ -1455,7 +1446,7 @@ The operation is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float32_sqrt
(
float32
a
)
float32
float32_sqrt
(
struct
roundingData
*
roundData
,
float32
a
)
{
flag
aSign
;
int16
aExp
,
zExp
;
...
...
@@ -1468,12 +1459,12 @@ float32 float32_sqrt( float32 a )
if
(
aExp
==
0xFF
)
{
if
(
aSig
)
return
propagateFloat32NaN
(
a
,
0
);
if
(
!
aSign
)
return
a
;
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
if
(
aSign
)
{
if
(
(
aExp
|
aSig
)
==
0
)
return
a
;
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float32_default_nan
;
}
if
(
aExp
==
0
)
{
...
...
@@ -1499,7 +1490,7 @@ float32 float32_sqrt( float32 a )
}
}
shift32RightJamming
(
zSig
,
1
,
&
zSig
);
return
roundAndPackFloat32
(
0
,
zExp
,
zSig
);
return
roundAndPackFloat32
(
roundData
,
0
,
zExp
,
zSig
);
}
...
...
@@ -1661,7 +1652,7 @@ positive integer is returned. Otherwise, if the conversion overflows, the
largest integer with the same sign as `a' is returned.
-------------------------------------------------------------------------------
*/
int32
float64_to_int32
(
float64
a
)
int32
float64_to_int32
(
struct
roundingData
*
roundData
,
float64
a
)
{
flag
aSign
;
int16
aExp
,
shiftCount
;
...
...
@@ -1674,7 +1665,7 @@ int32 float64_to_int32( float64 a )
if
(
aExp
)
aSig
|=
LIT64
(
0x0010000000000000
);
shiftCount
=
0x42C
-
aExp
;
if
(
0
<
shiftCount
)
shift64RightJamming
(
aSig
,
shiftCount
,
&
aSig
);
return
roundAndPackInt32
(
aSign
,
aSig
);
return
roundAndPackInt32
(
roundData
,
aSign
,
aSig
);
}
...
...
@@ -1705,7 +1696,7 @@ int32 float64_to_int32_round_to_zero( float64 a )
goto
invalid
;
}
else
if
(
52
<
shiftCount
)
{
if
(
aExp
||
aSig
)
float_
exception_flags
|=
float_flag_inexact
;
if
(
aExp
||
aSig
)
float_
raise
(
float_flag_inexact
)
;
return
0
;
}
aSig
|=
LIT64
(
0x0010000000000000
);
...
...
@@ -1715,11 +1706,11 @@ int32 float64_to_int32_round_to_zero( float64 a )
if
(
aSign
)
z
=
-
z
;
if
(
(
z
<
0
)
^
aSign
)
{
invalid:
float_
exception_flags
|=
float_flag_invalid
;
float_
raise
(
float_flag_invalid
)
;
return
aSign
?
0x80000000
:
0x7FFFFFFF
;
}
if
(
(
aSig
<<
shiftCount
)
!=
savedASig
)
{
float_
exception_flags
|=
float_flag_inexact
;
float_
raise
(
float_flag_inexact
)
;
}
return
z
;
...
...
@@ -1736,7 +1727,7 @@ positive integer is returned. Otherwise, if the conversion overflows, the
largest positive integer is returned.
-------------------------------------------------------------------------------
*/
int32
float64_to_uint32
(
float64
a
)
int32
float64_to_uint32
(
struct
roundingData
*
roundData
,
float64
a
)
{
flag
aSign
;
int16
aExp
,
shiftCount
;
...
...
@@ -1749,7 +1740,7 @@ int32 float64_to_uint32( float64 a )
if
(
aExp
)
aSig
|=
LIT64
(
0x0010000000000000
);
shiftCount
=
0x42C
-
aExp
;
if
(
0
<
shiftCount
)
shift64RightJamming
(
aSig
,
shiftCount
,
&
aSig
);
return
roundAndPackInt32
(
aSign
,
aSig
);
return
roundAndPackInt32
(
roundData
,
aSign
,
aSig
);
}
/*
...
...
@@ -1778,7 +1769,7 @@ int32 float64_to_uint32_round_to_zero( float64 a )
goto
invalid
;
}
else
if
(
52
<
shiftCount
)
{
if
(
aExp
||
aSig
)
float_
exception_flags
|=
float_flag_inexact
;
if
(
aExp
||
aSig
)
float_
raise
(
float_flag_inexact
)
;
return
0
;
}
aSig
|=
LIT64
(
0x0010000000000000
);
...
...
@@ -1788,11 +1779,11 @@ int32 float64_to_uint32_round_to_zero( float64 a )
if
(
aSign
)
z
=
-
z
;
if
(
(
z
<
0
)
^
aSign
)
{
invalid:
float_
exception_flags
|=
float_flag_invalid
;
float_
raise
(
float_flag_invalid
)
;
return
aSign
?
0x80000000
:
0x7FFFFFFF
;
}
if
(
(
aSig
<<
shiftCount
)
!=
savedASig
)
{
float_
exception_flags
|=
float_flag_inexact
;
float_
raise
(
float_flag_inexact
)
;
}
return
z
;
}
...
...
@@ -1805,7 +1796,7 @@ performed according to the IEC/IEEE Standard for Binary Floating-point
Arithmetic.
-------------------------------------------------------------------------------
*/
float32
float64_to_float32
(
float64
a
)
float32
float64_to_float32
(
struct
roundingData
*
roundData
,
float64
a
)
{
flag
aSign
;
int16
aExp
;
...
...
@@ -1825,7 +1816,7 @@ float32 float64_to_float32( float64 a )
zSig
|=
0x40000000
;
aExp
-=
0x381
;
}
return
roundAndPackFloat32
(
aSign
,
aExp
,
zSig
);
return
roundAndPackFloat32
(
roundData
,
aSign
,
aExp
,
zSig
);
}
...
...
@@ -1872,7 +1863,7 @@ operation is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_round_to_int
(
float64
a
)
float64
float64_round_to_int
(
struct
roundingData
*
roundData
,
float64
a
)
{
flag
aSign
;
int16
aExp
;
...
...
@@ -1889,9 +1880,9 @@ float64 float64_round_to_int( float64 a )
}
if
(
aExp
<=
0x3FE
)
{
if
(
(
bits64
)
(
a
<<
1
)
==
0
)
return
a
;
float_exception_flags
|=
float_flag_inexact
;
roundData
->
exception
|=
float_flag_inexact
;
aSign
=
extractFloat64Sign
(
a
);
switch
(
float_rounding_
mode
)
{
switch
(
roundData
->
mode
)
{
case
float_round_nearest_even
:
if
(
(
aExp
==
0x3FE
)
&&
extractFloat64Frac
(
a
)
)
{
return
packFloat64
(
aSign
,
0x3FF
,
0
);
...
...
@@ -1909,7 +1900,7 @@ float64 float64_round_to_int( float64 a )
lastBitMask
<<=
0x433
-
aExp
;
roundBitsMask
=
lastBitMask
-
1
;
z
=
a
;
roundingMode
=
float_rounding_
mode
;
roundingMode
=
roundData
->
mode
;
if
(
roundingMode
==
float_round_nearest_even
)
{
z
+=
lastBitMask
>>
1
;
if
(
(
z
&
roundBitsMask
)
==
0
)
z
&=
~
lastBitMask
;
...
...
@@ -1920,7 +1911,7 @@ float64 float64_round_to_int( float64 a )
}
}
z
&=
~
roundBitsMask
;
if
(
z
!=
a
)
float_exception_flags
|=
float_flag_inexact
;
if
(
z
!=
a
)
roundData
->
exception
|=
float_flag_inexact
;
return
z
;
}
...
...
@@ -1934,7 +1925,7 @@ addition is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float64
addFloat64Sigs
(
float64
a
,
float64
b
,
flag
zSign
)
static
float64
addFloat64Sigs
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
,
flag
zSign
)
{
int16
aExp
,
bExp
,
zExp
;
bits64
aSig
,
bSig
,
zSig
;
...
...
@@ -1993,7 +1984,7 @@ static float64 addFloat64Sigs( float64 a, float64 b, flag zSign )
++
zExp
;
}
roundAndPack:
return
roundAndPackFloat64
(
zSign
,
zExp
,
zSig
);
return
roundAndPackFloat64
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -2006,7 +1997,7 @@ result is a NaN. The subtraction is performed according to the IEC/IEEE
Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
float64
subFloat64Sigs
(
float64
a
,
float64
b
,
flag
zSign
)
static
float64
subFloat64Sigs
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
,
flag
zSign
)
{
int16
aExp
,
bExp
,
zExp
;
bits64
aSig
,
bSig
,
zSig
;
...
...
@@ -2023,7 +2014,7 @@ static float64 subFloat64Sigs( float64 a, float64 b, flag zSign )
if
(
expDiff
<
0
)
goto
bExpBigger
;
if
(
aExp
==
0x7FF
)
{
if
(
aSig
|
bSig
)
return
propagateFloat64NaN
(
a
,
b
);
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
if
(
aExp
==
0
)
{
...
...
@@ -2032,7 +2023,7 @@ static float64 subFloat64Sigs( float64 a, float64 b, flag zSign )
}
if
(
bSig
<
aSig
)
goto
aBigger
;
if
(
aSig
<
bSig
)
goto
bBigger
;
return
packFloat64
(
float_rounding_
mode
==
float_round_down
,
0
,
0
);
return
packFloat64
(
roundData
->
mode
==
float_round_down
,
0
,
0
);
bExpBigger:
if
(
bExp
==
0x7FF
)
{
if
(
bSig
)
return
propagateFloat64NaN
(
a
,
b
);
...
...
@@ -2069,7 +2060,7 @@ static float64 subFloat64Sigs( float64 a, float64 b, flag zSign )
zExp
=
aExp
;
normalizeRoundAndPack:
--
zExp
;
return
normalizeRoundAndPackFloat64
(
zSign
,
zExp
,
zSig
);
return
normalizeRoundAndPackFloat64
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -2080,17 +2071,17 @@ and `b'. The operation is performed according to the IEC/IEEE Standard for
Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_add
(
float64
a
,
float64
b
)
float64
float64_add
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloat64Sign
(
a
);
bSign
=
extractFloat64Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
addFloat64Sigs
(
a
,
b
,
aSign
);
return
addFloat64Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
subFloat64Sigs
(
a
,
b
,
aSign
);
return
subFloat64Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -2102,17 +2093,17 @@ Returns the result of subtracting the double-precision floating-point values
for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_sub
(
float64
a
,
float64
b
)
float64
float64_sub
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloat64Sign
(
a
);
bSign
=
extractFloat64Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
subFloat64Sigs
(
a
,
b
,
aSign
);
return
subFloat64Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
addFloat64Sigs
(
a
,
b
,
aSign
);
return
addFloat64Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -2124,7 +2115,7 @@ Returns the result of multiplying the double-precision floating-point values
for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_mul
(
float64
a
,
float64
b
)
float64
float64_mul
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
zExp
;
...
...
@@ -2142,7 +2133,7 @@ float64 float64_mul( float64 a, float64 b )
return
propagateFloat64NaN
(
a
,
b
);
}
if
(
(
bExp
|
bSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
return
packFloat64
(
zSign
,
0x7FF
,
0
);
...
...
@@ -2150,7 +2141,7 @@ float64 float64_mul( float64 a, float64 b )
if
(
bExp
==
0x7FF
)
{
if
(
bSig
)
return
propagateFloat64NaN
(
a
,
b
);
if
(
(
aExp
|
aSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
return
packFloat64
(
zSign
,
0x7FF
,
0
);
...
...
@@ -2172,7 +2163,7 @@ float64 float64_mul( float64 a, float64 b )
zSig0
<<=
1
;
--
zExp
;
}
return
roundAndPackFloat64
(
zSign
,
zExp
,
zSig0
);
return
roundAndPackFloat64
(
roundData
,
zSign
,
zExp
,
zSig0
);
}
...
...
@@ -2183,7 +2174,7 @@ by the corresponding value `b'. The operation is performed according to
the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_div
(
float64
a
,
float64
b
)
float64
float64_div
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
zExp
;
...
...
@@ -2202,7 +2193,7 @@ float64 float64_div( float64 a, float64 b )
if
(
aSig
)
return
propagateFloat64NaN
(
a
,
b
);
if
(
bExp
==
0x7FF
)
{
if
(
bSig
)
return
propagateFloat64NaN
(
a
,
b
);
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
return
packFloat64
(
zSign
,
0x7FF
,
0
);
...
...
@@ -2214,10 +2205,10 @@ float64 float64_div( float64 a, float64 b )
if
(
bExp
==
0
)
{
if
(
bSig
==
0
)
{
if
(
(
aExp
|
aSig
)
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
float_raise
(
float_flag_divbyzero
)
;
roundData
->
exception
|=
float_flag_divbyzero
;
return
packFloat64
(
zSign
,
0x7FF
,
0
);
}
normalizeFloat64Subnormal
(
bSig
,
&
bExp
,
&
bSig
);
...
...
@@ -2243,7 +2234,7 @@ float64 float64_div( float64 a, float64 b )
}
zSig
|=
(
rem1
!=
0
);
}
return
roundAndPackFloat64
(
zSign
,
zExp
,
zSig
);
return
roundAndPackFloat64
(
roundData
,
zSign
,
zExp
,
zSig
);
}
...
...
@@ -2254,7 +2245,7 @@ with respect to the corresponding value `b'. The operation is performed
according to the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_rem
(
float64
a
,
float64
b
)
float64
float64_rem
(
struct
roundingData
*
roundData
,
float64
a
,
float64
b
)
{
flag
aSign
,
bSign
,
zSign
;
int16
aExp
,
bExp
,
expDiff
;
...
...
@@ -2272,7 +2263,7 @@ float64 float64_rem( float64 a, float64 b )
if
(
aSig
||
(
(
bExp
==
0x7FF
)
&&
bSig
)
)
{
return
propagateFloat64NaN
(
a
,
b
);
}
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
if
(
bExp
==
0x7FF
)
{
...
...
@@ -2281,7 +2272,7 @@ float64 float64_rem( float64 a, float64 b )
}
if
(
bExp
==
0
)
{
if
(
bSig
==
0
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
normalizeFloat64Subnormal
(
bSig
,
&
bExp
,
&
bSig
);
...
...
@@ -2329,7 +2320,7 @@ float64 float64_rem( float64 a, float64 b )
}
zSign
=
(
(
sbits64
)
aSig
<
0
);
if
(
zSign
)
aSig
=
-
aSig
;
return
normalizeRoundAndPackFloat64
(
aSign
^
zSign
,
bExp
,
aSig
);
return
normalizeRoundAndPackFloat64
(
roundData
,
aSign
^
zSign
,
bExp
,
aSig
);
}
...
...
@@ -2340,7 +2331,7 @@ The operation is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
float64_sqrt
(
float64
a
)
float64
float64_sqrt
(
struct
roundingData
*
roundData
,
float64
a
)
{
flag
aSign
;
int16
aExp
,
zExp
;
...
...
@@ -2354,12 +2345,12 @@ float64 float64_sqrt( float64 a )
if
(
aExp
==
0x7FF
)
{
if
(
aSig
)
return
propagateFloat64NaN
(
a
,
a
);
if
(
!
aSign
)
return
a
;
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
if
(
aSign
)
{
if
(
(
aExp
|
aSig
)
==
0
)
return
a
;
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
float64_default_nan
;
}
if
(
aExp
==
0
)
{
...
...
@@ -2390,7 +2381,7 @@ float64 float64_sqrt( float64 a )
}
}
shift64RightJamming
(
zSig
,
1
,
&
zSig
);
return
roundAndPackFloat64
(
0
,
zExp
,
zSig
);
return
roundAndPackFloat64
(
roundData
,
0
,
zExp
,
zSig
);
}
...
...
@@ -2554,7 +2545,7 @@ largest positive integer is returned. Otherwise, if the conversion
overflows, the largest integer with the same sign as `a' is returned.
-------------------------------------------------------------------------------
*/
int32
floatx80_to_int32
(
floatx80
a
)
int32
floatx80_to_int32
(
struct
roundingData
*
roundData
,
floatx80
a
)
{
flag
aSign
;
int32
aExp
,
shiftCount
;
...
...
@@ -2567,7 +2558,7 @@ int32 floatx80_to_int32( floatx80 a )
shiftCount
=
0x4037
-
aExp
;
if
(
shiftCount
<=
0
)
shiftCount
=
1
;
shift64RightJamming
(
aSig
,
shiftCount
,
&
aSig
);
return
roundAndPackInt32
(
aSign
,
aSig
);
return
roundAndPackInt32
(
roundData
,
aSign
,
aSig
);
}
...
...
@@ -2598,7 +2589,7 @@ int32 floatx80_to_int32_round_to_zero( floatx80 a )
goto
invalid
;
}
else
if
(
63
<
shiftCount
)
{
if
(
aExp
||
aSig
)
float_
exception_flags
|=
float_flag_inexact
;
if
(
aExp
||
aSig
)
float_
raise
(
float_flag_inexact
)
;
return
0
;
}
savedASig
=
aSig
;
...
...
@@ -2607,11 +2598,11 @@ int32 floatx80_to_int32_round_to_zero( floatx80 a )
if
(
aSign
)
z
=
-
z
;
if
(
(
z
<
0
)
^
aSign
)
{
invalid:
float_
exception_flags
|=
float_flag_invalid
;
float_
raise
(
float_flag_invalid
)
;
return
aSign
?
0x80000000
:
0x7FFFFFFF
;
}
if
(
(
aSig
<<
shiftCount
)
!=
savedASig
)
{
float_
exception_flags
|=
float_flag_inexact
;
float_
raise
(
float_flag_inexact
)
;
}
return
z
;
...
...
@@ -2625,7 +2616,7 @@ conversion is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float32
floatx80_to_float32
(
floatx80
a
)
float32
floatx80_to_float32
(
struct
roundingData
*
roundData
,
floatx80
a
)
{
flag
aSign
;
int32
aExp
;
...
...
@@ -2642,7 +2633,7 @@ float32 floatx80_to_float32( floatx80 a )
}
shift64RightJamming
(
aSig
,
33
,
&
aSig
);
if
(
aExp
||
aSig
)
aExp
-=
0x3F81
;
return
roundAndPackFloat32
(
aSign
,
aExp
,
aSig
);
return
roundAndPackFloat32
(
roundData
,
aSign
,
aExp
,
aSig
);
}
...
...
@@ -2654,7 +2645,7 @@ conversion is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
float64
floatx80_to_float64
(
floatx80
a
)
float64
floatx80_to_float64
(
struct
roundingData
*
roundData
,
floatx80
a
)
{
flag
aSign
;
int32
aExp
;
...
...
@@ -2671,7 +2662,7 @@ float64 floatx80_to_float64( floatx80 a )
}
shift64RightJamming
(
aSig
,
1
,
&
zSig
);
if
(
aExp
||
aSig
)
aExp
-=
0x3C01
;
return
roundAndPackFloat64
(
aSign
,
aExp
,
zSig
);
return
roundAndPackFloat64
(
roundData
,
aSign
,
aExp
,
zSig
);
}
...
...
@@ -2683,7 +2674,7 @@ value. The operation is performed according to the IEC/IEEE Standard for
Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_round_to_int
(
floatx80
a
)
floatx80
floatx80_round_to_int
(
struct
roundingData
*
roundData
,
floatx80
a
)
{
flag
aSign
;
int32
aExp
;
...
...
@@ -2703,9 +2694,9 @@ floatx80 floatx80_round_to_int( floatx80 a )
&&
(
(
bits64
)
(
extractFloatx80Frac
(
a
)
<<
1
)
==
0
)
)
{
return
a
;
}
float_exception_flags
|=
float_flag_inexact
;
roundData
->
exception
|=
float_flag_inexact
;
aSign
=
extractFloatx80Sign
(
a
);
switch
(
float_rounding_
mode
)
{
switch
(
roundData
->
mode
)
{
case
float_round_nearest_even
:
if
(
(
aExp
==
0x3FFE
)
&&
(
bits64
)
(
extractFloatx80Frac
(
a
)
<<
1
)
)
{
...
...
@@ -2729,7 +2720,7 @@ floatx80 floatx80_round_to_int( floatx80 a )
lastBitMask
<<=
0x403E
-
aExp
;
roundBitsMask
=
lastBitMask
-
1
;
z
=
a
;
roundingMode
=
float_rounding_
mode
;
roundingMode
=
roundData
->
mode
;
if
(
roundingMode
==
float_round_nearest_even
)
{
z
.
low
+=
lastBitMask
>>
1
;
if
(
(
z
.
low
&
roundBitsMask
)
==
0
)
z
.
low
&=
~
lastBitMask
;
...
...
@@ -2744,7 +2735,7 @@ floatx80 floatx80_round_to_int( floatx80 a )
++
z
.
high
;
z
.
low
=
LIT64
(
0x8000000000000000
);
}
if
(
z
.
low
!=
a
.
low
)
float_exception_flags
|=
float_flag_inexact
;
if
(
z
.
low
!=
a
.
low
)
roundData
->
exception
|=
float_flag_inexact
;
return
z
;
}
...
...
@@ -2758,7 +2749,7 @@ The addition is performed according to the IEC/IEEE Standard for Binary
Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
floatx80
addFloatx80Sigs
(
floatx80
a
,
floatx80
b
,
flag
zSign
)
static
floatx80
addFloatx80Sigs
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
,
flag
zSign
)
{
int32
aExp
,
bExp
,
zExp
;
bits64
aSig
,
bSig
,
zSig0
,
zSig1
;
...
...
@@ -2814,7 +2805,7 @@ static floatx80 addFloatx80Sigs( floatx80 a, floatx80 b, flag zSign )
roundAndPack:
return
roundAndPackFloatx80
(
floatx80_rounding_precision
,
zSign
,
zExp
,
zSig0
,
zSig1
);
roundData
,
zSign
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -2827,7 +2818,7 @@ result is a NaN. The subtraction is performed according to the IEC/IEEE
Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
static
floatx80
subFloatx80Sigs
(
floatx80
a
,
floatx80
b
,
flag
zSign
)
static
floatx80
subFloatx80Sigs
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
,
flag
zSign
)
{
int32
aExp
,
bExp
,
zExp
;
bits64
aSig
,
bSig
,
zSig0
,
zSig1
;
...
...
@@ -2845,7 +2836,7 @@ static floatx80 subFloatx80Sigs( floatx80 a, floatx80 b, flag zSign )
if
(
(
bits64
)
(
(
aSig
|
bSig
)
<<
1
)
)
{
return
propagateFloatx80NaN
(
a
,
b
);
}
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
z
.
low
=
floatx80_default_nan_low
;
z
.
high
=
floatx80_default_nan_high
;
return
z
;
...
...
@@ -2857,7 +2848,7 @@ static floatx80 subFloatx80Sigs( floatx80 a, floatx80 b, flag zSign )
zSig1
=
0
;
if
(
bSig
<
aSig
)
goto
aBigger
;
if
(
aSig
<
bSig
)
goto
bBigger
;
return
packFloatx80
(
float_rounding_
mode
==
float_round_down
,
0
,
0
);
return
packFloatx80
(
roundData
->
mode
==
float_round_down
,
0
,
0
);
bExpBigger:
if
(
bExp
==
0x7FFF
)
{
if
(
(
bits64
)
(
bSig
<<
1
)
)
return
propagateFloatx80NaN
(
a
,
b
);
...
...
@@ -2883,7 +2874,7 @@ static floatx80 subFloatx80Sigs( floatx80 a, floatx80 b, flag zSign )
normalizeRoundAndPack:
return
normalizeRoundAndPackFloatx80
(
floatx80_rounding_precision
,
zSign
,
zExp
,
zSig0
,
zSig1
);
roundData
,
zSign
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -2894,17 +2885,17 @@ values `a' and `b'. The operation is performed according to the IEC/IEEE
Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_add
(
floatx80
a
,
floatx80
b
)
floatx80
floatx80_add
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloatx80Sign
(
a
);
bSign
=
extractFloatx80Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
addFloatx80Sigs
(
a
,
b
,
aSign
);
return
addFloatx80Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
subFloatx80Sigs
(
a
,
b
,
aSign
);
return
subFloatx80Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -2916,17 +2907,17 @@ point values `a' and `b'. The operation is performed according to the
IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_sub
(
floatx80
a
,
floatx80
b
)
floatx80
floatx80_sub
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
)
{
flag
aSign
,
bSign
;
aSign
=
extractFloatx80Sign
(
a
);
bSign
=
extractFloatx80Sign
(
b
);
if
(
aSign
==
bSign
)
{
return
subFloatx80Sigs
(
a
,
b
,
aSign
);
return
subFloatx80Sigs
(
roundData
,
a
,
b
,
aSign
);
}
else
{
return
addFloatx80Sigs
(
a
,
b
,
aSign
);
return
addFloatx80Sigs
(
roundData
,
a
,
b
,
aSign
);
}
}
...
...
@@ -2938,7 +2929,7 @@ point values `a' and `b'. The operation is performed according to the
IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_mul
(
floatx80
a
,
floatx80
b
)
floatx80
floatx80_mul
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
)
{
flag
aSign
,
bSign
,
zSign
;
int32
aExp
,
bExp
,
zExp
;
...
...
@@ -2964,7 +2955,7 @@ floatx80 floatx80_mul( floatx80 a, floatx80 b )
if
(
(
bits64
)
(
bSig
<<
1
)
)
return
propagateFloatx80NaN
(
a
,
b
);
if
(
(
aExp
|
aSig
)
==
0
)
{
invalid:
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
z
.
low
=
floatx80_default_nan_low
;
z
.
high
=
floatx80_default_nan_high
;
return
z
;
...
...
@@ -2987,7 +2978,7 @@ floatx80 floatx80_mul( floatx80 a, floatx80 b )
}
return
roundAndPackFloatx80
(
floatx80_rounding_precision
,
zSign
,
zExp
,
zSig0
,
zSig1
);
roundData
,
zSign
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -2998,7 +2989,7 @@ value `a' by the corresponding value `b'. The operation is performed
according to the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_div
(
floatx80
a
,
floatx80
b
)
floatx80
floatx80_div
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
)
{
flag
aSign
,
bSign
,
zSign
;
int32
aExp
,
bExp
,
zExp
;
...
...
@@ -3029,12 +3020,12 @@ floatx80 floatx80_div( floatx80 a, floatx80 b )
if
(
bSig
==
0
)
{
if
(
(
aExp
|
aSig
)
==
0
)
{
invalid:
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
z
.
low
=
floatx80_default_nan_low
;
z
.
high
=
floatx80_default_nan_high
;
return
z
;
}
float_raise
(
float_flag_divbyzero
)
;
roundData
->
exception
|=
float_flag_divbyzero
;
return
packFloatx80
(
zSign
,
0x7FFF
,
LIT64
(
0x8000000000000000
)
);
}
normalizeFloatx80Subnormal
(
bSig
,
&
bExp
,
&
bSig
);
...
...
@@ -3068,7 +3059,7 @@ floatx80 floatx80_div( floatx80 a, floatx80 b )
}
return
roundAndPackFloatx80
(
floatx80_rounding_precision
,
zSign
,
zExp
,
zSig0
,
zSig1
);
roundData
,
zSign
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -3079,7 +3070,7 @@ Returns the remainder of the extended double-precision floating-point value
according to the IEC/IEEE Standard for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_rem
(
floatx80
a
,
floatx80
b
)
floatx80
floatx80_rem
(
struct
roundingData
*
roundData
,
floatx80
a
,
floatx80
b
)
{
flag
aSign
,
bSign
,
zSign
;
int32
aExp
,
bExp
,
expDiff
;
...
...
@@ -3107,7 +3098,7 @@ floatx80 floatx80_rem( floatx80 a, floatx80 b )
if
(
bExp
==
0
)
{
if
(
bSig
==
0
)
{
invalid:
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
z
.
low
=
floatx80_default_nan_low
;
z
.
high
=
floatx80_default_nan_high
;
return
z
;
...
...
@@ -3164,9 +3155,10 @@ floatx80 floatx80_rem( floatx80 a, floatx80 b )
aSig1
=
alternateASig1
;
zSign
=
!
zSign
;
}
return
normalizeRoundAndPackFloatx80
(
80
,
zSign
,
bExp
+
expDiff
,
aSig0
,
aSig1
);
roundData
,
zSign
,
bExp
+
expDiff
,
aSig0
,
aSig1
);
}
...
...
@@ -3177,7 +3169,7 @@ value `a'. The operation is performed according to the IEC/IEEE Standard
for Binary Floating-point Arithmetic.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_sqrt
(
floatx80
a
)
floatx80
floatx80_sqrt
(
struct
roundingData
*
roundData
,
floatx80
a
)
{
flag
aSign
;
int32
aExp
,
zExp
;
...
...
@@ -3197,7 +3189,7 @@ floatx80 floatx80_sqrt( floatx80 a )
if
(
aSign
)
{
if
(
(
aExp
|
aSig0
)
==
0
)
return
a
;
invalid:
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
z
.
low
=
floatx80_default_nan_low
;
z
.
high
=
floatx80_default_nan_high
;
return
z
;
...
...
@@ -3242,7 +3234,7 @@ floatx80 floatx80_sqrt( floatx80 a )
}
return
roundAndPackFloatx80
(
floatx80_rounding_precision
,
0
,
zExp
,
zSig0
,
zSig1
);
roundData
,
0
,
zExp
,
zSig0
,
zSig1
);
}
...
...
@@ -3264,7 +3256,7 @@ flag floatx80_eq( floatx80 a, floatx80 b )
)
{
if
(
floatx80_is_signaling_nan
(
a
)
||
floatx80_is_signaling_nan
(
b
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
}
return
0
;
}
...
...
@@ -3294,7 +3286,7 @@ flag floatx80_le( floatx80 a, floatx80 b )
||
(
(
extractFloatx80Exp
(
b
)
==
0x7FFF
)
&&
(
bits64
)
(
extractFloatx80Frac
(
b
)
<<
1
)
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
0
;
}
aSign
=
extractFloatx80Sign
(
a
);
...
...
@@ -3328,7 +3320,7 @@ flag floatx80_lt( floatx80 a, floatx80 b )
||
(
(
extractFloatx80Exp
(
b
)
==
0x7FFF
)
&&
(
bits64
)
(
extractFloatx80Frac
(
b
)
<<
1
)
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
0
;
}
aSign
=
extractFloatx80Sign
(
a
);
...
...
@@ -3361,7 +3353,7 @@ flag floatx80_eq_signaling( floatx80 a, floatx80 b )
||
(
(
extractFloatx80Exp
(
b
)
==
0x7FFF
)
&&
(
bits64
)
(
extractFloatx80Frac
(
b
)
<<
1
)
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
return
0
;
}
return
...
...
@@ -3392,7 +3384,7 @@ flag floatx80_le_quiet( floatx80 a, floatx80 b )
)
{
if
(
floatx80_is_signaling_nan
(
a
)
||
floatx80_is_signaling_nan
(
b
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
}
return
0
;
}
...
...
@@ -3429,7 +3421,7 @@ flag floatx80_lt_quiet( floatx80 a, floatx80 b )
)
{
if
(
floatx80_is_signaling_nan
(
a
)
||
floatx80_is_signaling_nan
(
b
)
)
{
float_raise
(
float_flag_invalid
)
;
roundData
->
exception
|=
float_flag_invalid
;
}
return
0
;
}
...
...
arch/arm/nwfpe/softfloat.h
View file @
194d0710
...
...
@@ -74,7 +74,7 @@ enum {
Software IEC/IEEE floating-point rounding mode.
-------------------------------------------------------------------------------
*/
extern
signed
char
float_rounding_mode
;
//extern int8
float_rounding_mode;
enum
{
float_round_nearest_even
=
0
,
float_round_to_zero
=
1
,
...
...
@@ -86,7 +86,6 @@ enum {
-------------------------------------------------------------------------------
Software IEC/IEEE floating-point exception flags.
-------------------------------------------------------------------------------
extern signed char float_exception_flags;
enum {
float_flag_inexact = 1,
float_flag_underflow = 2,
...
...
@@ -99,7 +98,6 @@ ScottB: November 4, 1998
Changed the enumeration to match the bit order in the FPA11.
*/
extern
signed
char
float_exception_flags
;
enum
{
float_flag_invalid
=
1
,
float_flag_divbyzero
=
2
,
...
...
@@ -121,7 +119,7 @@ void float_raise( signed char );
Software IEC/IEEE integer-to-floating-point conversion routines.
-------------------------------------------------------------------------------
*/
float32
int32_to_float32
(
signed
int
);
float32
int32_to_float32
(
s
truct
roundingData
*
,
s
igned
int
);
float64
int32_to_float64
(
signed
int
);
#ifdef FLOATX80
floatx80
int32_to_floatx80
(
signed
int
);
...
...
@@ -132,7 +130,7 @@ floatx80 int32_to_floatx80( signed int );
Software IEC/IEEE single-precision conversion routines.
-------------------------------------------------------------------------------
*/
signed
int
float32_to_int32
(
float32
);
signed
int
float32_to_int32
(
struct
roundingData
*
,
float32
);
signed
int
float32_to_int32_round_to_zero
(
float32
);
float64
float32_to_float64
(
float32
);
#ifdef FLOATX80
...
...
@@ -144,13 +142,13 @@ floatx80 float32_to_floatx80( float32 );
Software IEC/IEEE single-precision operations.
-------------------------------------------------------------------------------
*/
float32
float32_round_to_int
(
float32
);
float32
float32_add
(
float32
,
float32
);
float32
float32_sub
(
float32
,
float32
);
float32
float32_mul
(
float32
,
float32
);
float32
float32_div
(
float32
,
float32
);
float32
float32_rem
(
float32
,
float32
);
float32
float32_sqrt
(
float32
);
float32
float32_round_to_int
(
struct
roundingData
*
,
float32
);
float32
float32_add
(
struct
roundingData
*
,
float32
,
float32
);
float32
float32_sub
(
struct
roundingData
*
,
float32
,
float32
);
float32
float32_mul
(
struct
roundingData
*
,
float32
,
float32
);
float32
float32_div
(
struct
roundingData
*
,
float32
,
float32
);
float32
float32_rem
(
struct
roundingData
*
,
float32
,
float32
);
float32
float32_sqrt
(
struct
roundingData
*
,
float32
);
char
float32_eq
(
float32
,
float32
);
char
float32_le
(
float32
,
float32
);
char
float32_lt
(
float32
,
float32
);
...
...
@@ -164,9 +162,9 @@ char float32_is_signaling_nan( float32 );
Software IEC/IEEE double-precision conversion routines.
-------------------------------------------------------------------------------
*/
signed
int
float64_to_int32
(
float64
);
signed
int
float64_to_int32
(
struct
roundingData
*
,
float64
);
signed
int
float64_to_int32_round_to_zero
(
float64
);
float32
float64_to_float32
(
float64
);
float32
float64_to_float32
(
struct
roundingData
*
,
float64
);
#ifdef FLOATX80
floatx80
float64_to_floatx80
(
float64
);
#endif
...
...
@@ -176,13 +174,13 @@ floatx80 float64_to_floatx80( float64 );
Software IEC/IEEE double-precision operations.
-------------------------------------------------------------------------------
*/
float64
float64_round_to_int
(
float64
);
float64
float64_add
(
float64
,
float64
);
float64
float64_sub
(
float64
,
float64
);
float64
float64_mul
(
float64
,
float64
);
float64
float64_div
(
float64
,
float64
);
float64
float64_rem
(
float64
,
float64
);
float64
float64_sqrt
(
float64
);
float64
float64_round_to_int
(
struct
roundingData
*
,
float64
);
float64
float64_add
(
struct
roundingData
*
,
float64
,
float64
);
float64
float64_sub
(
struct
roundingData
*
,
float64
,
float64
);
float64
float64_mul
(
struct
roundingData
*
,
float64
,
float64
);
float64
float64_div
(
struct
roundingData
*
,
float64
,
float64
);
float64
float64_rem
(
struct
roundingData
*
,
float64
,
float64
);
float64
float64_sqrt
(
struct
roundingData
*
,
float64
);
char
float64_eq
(
float64
,
float64
);
char
float64_le
(
float64
,
float64
);
char
float64_lt
(
float64
,
float64
);
...
...
@@ -198,31 +196,23 @@ char float64_is_signaling_nan( float64 );
Software IEC/IEEE extended double-precision conversion routines.
-------------------------------------------------------------------------------
*/
signed
int
floatx80_to_int32
(
floatx80
);
signed
int
floatx80_to_int32
(
struct
roundingData
*
,
floatx80
);
signed
int
floatx80_to_int32_round_to_zero
(
floatx80
);
float32
floatx80_to_float32
(
floatx80
);
float64
floatx80_to_float64
(
floatx80
);
/*
-------------------------------------------------------------------------------
Software IEC/IEEE extended double-precision rounding precision. Valid
values are 32, 64, and 80.
-------------------------------------------------------------------------------
*/
extern
signed
char
floatx80_rounding_precision
;
float32
floatx80_to_float32
(
struct
roundingData
*
,
floatx80
);
float64
floatx80_to_float64
(
struct
roundingData
*
,
floatx80
);
/*
-------------------------------------------------------------------------------
Software IEC/IEEE extended double-precision operations.
-------------------------------------------------------------------------------
*/
floatx80
floatx80_round_to_int
(
floatx80
);
floatx80
floatx80_add
(
floatx80
,
floatx80
);
floatx80
floatx80_sub
(
floatx80
,
floatx80
);
floatx80
floatx80_mul
(
floatx80
,
floatx80
);
floatx80
floatx80_div
(
floatx80
,
floatx80
);
floatx80
floatx80_rem
(
floatx80
,
floatx80
);
floatx80
floatx80_sqrt
(
floatx80
);
floatx80
floatx80_round_to_int
(
struct
roundingData
*
,
floatx80
);
floatx80
floatx80_add
(
struct
roundingData
*
,
floatx80
,
floatx80
);
floatx80
floatx80_sub
(
struct
roundingData
*
,
floatx80
,
floatx80
);
floatx80
floatx80_mul
(
struct
roundingData
*
,
floatx80
,
floatx80
);
floatx80
floatx80_div
(
struct
roundingData
*
,
floatx80
,
floatx80
);
floatx80
floatx80_rem
(
struct
roundingData
*
,
floatx80
,
floatx80
);
floatx80
floatx80_sqrt
(
struct
roundingData
*
,
floatx80
);
char
floatx80_eq
(
floatx80
,
floatx80
);
char
floatx80_le
(
floatx80
,
floatx80
);
char
floatx80_lt
(
floatx80
,
floatx80
);
...
...
arch/arm/vfp/vfpdouble.c
View file @
194d0710
...
...
@@ -770,6 +770,9 @@ vfp_double_add(struct vfp_double *vdd, struct vfp_double *vdn,
if
((
s64
)
m_sig
<
0
)
{
vdd
->
sign
=
vfp_sign_negate
(
vdd
->
sign
);
m_sig
=
-
m_sig
;
}
else
if
(
m_sig
==
0
)
{
vdd
->
sign
=
(
fpscr
&
FPSCR_RMODE_MASK
)
==
FPSCR_ROUND_MINUSINF
?
0x8000
:
0
;
}
}
else
{
m_sig
+=
vdn
->
significand
;
...
...
drivers/char/watchdog/sa1100_wdt.c
View file @
194d0710
...
...
@@ -36,13 +36,10 @@
#include <asm/uaccess.h>
#define OSCR_FREQ CLOCK_TICK_RATE
#define SA1100_CLOSE_MAGIC (0x5afc4453)
static
unsigned
long
sa1100wdt_users
;
static
int
expect_close
;
static
int
pre_margin
;
static
int
boot_status
;
static
int
nowayout
=
WATCHDOG_NOWAYOUT
;
/*
* Allow only one person to hold it open
...
...
@@ -62,55 +59,33 @@ static int sa1100dog_open(struct inode *inode, struct file *file)
}
/*
* Shut off the timer.
* Lock it in if it's a module and we defined ...NOWAYOUT
* Oddly, the watchdog can only be enabled, but we can turn off
* the interrupt, which appears to prevent the watchdog timing out.
* The watchdog cannot be disabled.
*
* Previous comments suggested that turning off the interrupt by
* clearing OIER[E3] would prevent the watchdog timing out but this
* does not appear to be true (at least on the PXA255).
*/
static
int
sa1100dog_release
(
struct
inode
*
inode
,
struct
file
*
file
)
{
OSMR3
=
OSCR
+
pre_margin
;
if
(
expect_close
==
SA1100_CLOSE_MAGIC
)
{
OIER
&=
~
OIER_E3
;
}
else
{
printk
(
KERN_CRIT
"WATCHDOG: WDT device closed unexpectedly. WDT will not stop!
\n
"
);
}
printk
(
KERN_CRIT
"WATCHDOG: Device closed - timer will not stop
\n
"
);
clear_bit
(
1
,
&
sa1100wdt_users
);
expect_close
=
0
;
return
0
;
}
static
ssize_t
sa1100dog_write
(
struct
file
*
file
,
const
char
*
data
,
size_t
len
,
loff_t
*
ppos
)
{
if
(
len
)
{
if
(
!
nowayout
)
{
size_t
i
;
expect_close
=
0
;
for
(
i
=
0
;
i
!=
len
;
i
++
)
{
char
c
;
if
(
get_user
(
c
,
data
+
i
))
return
-
EFAULT
;
if
(
c
==
'V'
)
expect_close
=
SA1100_CLOSE_MAGIC
;
}
}
if
(
len
)
/* Refresh OSMR3 timer. */
OSMR3
=
OSCR
+
pre_margin
;
}
return
len
;
}
static
struct
watchdog_info
ident
=
{
.
options
=
WDIOF_CARDRESET
|
WDIOF_MAGICCLOSE
|
WDIOF_SETTIMEOUT
|
WDIOF_KEEPALIVEPING
,
.
identity
=
"SA1100 Watchdog"
,
.
options
=
WDIOF_CARDRESET
|
WDIOF_SETTIMEOUT
|
WDIOF_KEEPALIVEPING
,
.
identity
=
"SA1100/PXA255 Watchdog"
,
};
static
int
sa1100dog_ioctl
(
struct
inode
*
inode
,
struct
file
*
file
,
...
...
@@ -172,7 +147,7 @@ static struct file_operations sa1100dog_fops =
static
struct
miscdevice
sa1100dog_miscdev
=
{
.
minor
=
WATCHDOG_MINOR
,
.
name
=
"
SA1100/PXA2xx
watchdog"
,
.
name
=
"watchdog"
,
.
fops
=
&
sa1100dog_fops
,
};
...
...
@@ -194,7 +169,6 @@ static int __init sa1100dog_init(void)
if
(
ret
==
0
)
printk
(
"SA1100/PXA2xx Watchdog Timer: timer margin %d sec
\n
"
,
margin
);
return
ret
;
}
...
...
@@ -212,8 +186,5 @@ MODULE_DESCRIPTION("SA1100/PXA2xx Watchdog");
module_param
(
margin
,
int
,
0
);
MODULE_PARM_DESC
(
margin
,
"Watchdog margin in seconds (default 60s)"
);
module_param
(
nowayout
,
int
,
0
);
MODULE_PARM_DESC
(
nowayout
,
"Watchdog cannot be stopped once started"
);
MODULE_LICENSE
(
"GPL"
);
MODULE_ALIAS_MISCDEV
(
WATCHDOG_MINOR
);
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment