// Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// e32\euser\epoc\arm\uc_realx.cia
//
//
#include <e32cia.h>
#include <u32std.h>
#include <e32math.h>
#ifdef __USE_VFP_MATH
#include <arm_vfp.h>
#endif
#if defined(__USE_VFP_MATH) && !defined(__CPU_HAS_VFP)
#error __USE_VFP_MATH was defined but not __CPU_HAS_VFP - impossible combination, check variant.mmh
#endif
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX()
/**
Constructs a default extended precision object.
This sets the value to zero.
*/
{
asm("mov r1, #0 ");
asm("str r1, [r0] ");
asm("str r1, [r0, #4] ");
asm("str r1, [r0, #8] ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::TRealX(TUint /*anExp*/, TUint /*aMantHi*/, TUint /*aMantLo*/)
/**
Constructs an extended precision object from an explicit exponent and
a 64 bit mantissa.
@param anExp The exponent
@param aMantHi The high order 32 bits of the 64 bit mantissa
@param aMantLo The low order 32 bits of the 64 bit mantissa
*/
{
asm("str r1, [r0, #8] ");
asm("str r2, [r0, #4] ");
asm("str r3, [r0, #0] ");
__JUMP(,lr);
}
#endif
__NAKED__ EXPORT_C TInt TRealX::Set(TInt /*anInt*/)
/**
Gives this extended precision object a new value taken
from a signed integer.
@param anInt The signed integer value.
@return KErrNone, always.
*/
{
asm("stmfd sp!, {lr} ");
asm("mov r2, r1 ");
asm("bl ConvertIntToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, #0 "); // return KErrNone
__POPRET("");
}
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX(TInt /*anInt*/)
/**
Constructs an extended precision object from a signed integer value.
@param anInt The signed integer value.
*/
{
// fall through
}
#endif
__NAKED__ EXPORT_C TRealX& TRealX::operator=(TInt /*anInt*/)
/**
Assigns the specified signed integer value to this extended precision object.
@param anInt The signed integer value.
@return A reference to this extended precision object.
*/
{
asm("stmfd sp!, {lr} ");
asm("mov r2, r1 ");
asm("bl ConvertIntToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
__POPRET("");
asm("ConvertIntToTRealX: ");
asm("cmp r2, #0 ");
asm("movpl r3, #0 "); // if int>0, r3=0
asm("beq ConvertIntToTRealX0 "); // if int=0, return 0
asm("movmi r3, #1 "); // if int<0, r3=1
asm("rsbmi r2, r2, #0 "); // if int -ve, negate it
asm("orr r3, r3, #0x001E0000 ");
asm("orr r3, r3, #0x80000000 "); // r3=exponent 801E + sign bit
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,2);
asm("mov r2, r2, lsl r12 ");
asm("sub r3, r3, r12, lsl #16 ");
#else
asm("cmp r2, #0x10000 "); // normalise mantissa, decrementing exponent as needed
asm("movcc r2, r2, lsl #16 ");
asm("subcc r3, r3, #0x100000 ");
asm("cmp r2, #0x1000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r3, r3, #0x080000 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r3, r3, #0x040000 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r3, r3, #0x020000 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("subcc r3, r3, #0x010000 ");
#endif
asm("ConvertIntToTRealX0: ");
asm("mov r1, #0 "); // low order word of mantissa = 0
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::Set(const TInt64& /*anInt*/)
/**
Gives this extended precision object a new value taken from
a 64 bit integer.
@param anInt The 64 bit integer value.
@return KErrNone, always.
*/
{
asm("stmfd sp!, {lr} ");
asm("ldmia r1, {r1,r2} ");
asm("bl ConvertInt64ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, #0 "); // return KErrNone
__POPRET("");
}
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX(const TInt64& /*anInt*/)
/**
Constructs an extended precision object from a 64 bit integer.
@param anInt A reference to a 64 bit integer.
*/
{
// fall through
}
#endif
__NAKED__ EXPORT_C TRealX& TRealX::operator=(const TInt64& /*anInt*/)
/**
Assigns the specified 64 bit integer value to this extended precision object.
@param anInt A reference to a 64 bit integer.
@return A reference to this extended precision object.
*/
{
asm("stmfd sp!, {lr} ");
asm("ldmia r1, {r1,r2} ");
asm("bl ConvertInt64ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
__POPRET("");
asm("ConvertInt64ToTRealX: ");
asm("movs r3, r2, lsr #31 "); // sign bit into r3 bit 0
asm("beq ConvertInt64ToTRealX1 "); // skip if plus
asm("rsbs r1, r1, #0 "); // take absolute value
asm("rsc r2, r2, #0 ");
asm("ConvertInt64ToTRealX1: ");
asm("cmp r2, #0 "); // does it fit into 32 bits?
asm("moveq r2, r1 "); // if it does, do 32 bit conversion
asm("beq ConvertUintToTRealX1 ");
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,2);
asm("mov r2, r2, lsl r12 ");
asm("rsb r12, r12, #32 ");
asm("orr r2, r2, r1, lsr r12 ");
asm("rsb r12, r12, #32 ");
#else
asm("mov r12, #32 "); // 32-number of left-shifts needed to normalise
asm("cmp r2, #0x10000 "); // calculate number required
asm("movcc r2, r2, lsl #16 ");
asm("subcc r12, r12, #16 ");
asm("cmp r2, #0x1000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r12, r12, #8 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r12, r12, #4 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r12, r12, #2 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("subcc r12, r12, #1 "); // r2 is now normalised
asm("orr r2, r2, r1, lsr r12 "); // shift r1 left into r2
asm("rsb r12, r12, #32 ");
#endif
asm("mov r1, r1, lsl r12 ");
asm("add r3, r3, #0x80000000 "); // exponent = 803E-r12
asm("add r3, r3, #0x003E0000 ");
asm("sub r3, r3, r12, lsl #16 ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::Set(TUint /*anInt*/)
/**
Gives this extended precision object a new value taken from
an unsigned integer.
@param The unsigned integer value.
@return KErrNone, always.
*/
{
asm("stmfd sp!, {lr} ");
asm("mov r2, r1 ");
asm("bl ConvertUintToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, #0 "); // return KErrNone
__POPRET("");
}
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX(TUint /*anInt*/)
/**
Constructs an extended precision object from an unsigned integer value.
@param anInt The unsigned integer value.
*/
{
// fall through
}
#endif
__NAKED__ EXPORT_C TRealX& TRealX::operator=(TUint /*anInt*/)
/**
Assigns the specified unsigned integer value to this extended precision object.
@param anInt The unsigned integer value.
@return A reference to this extended precision object.
*/
{
asm("stmfd sp!, {lr} ");
asm("mov r2, r1 ");
asm("bl ConvertUintToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
__POPRET("");
asm("ConvertUintToTRealX: ");
asm("mov r3, #0 ");
asm("ConvertUintToTRealX1: ");
asm("cmp r2, #0 "); // check for zero
asm("beq ConvertUintToTRealX0 ");
asm("orr r3, r3, #0x001E0000 ");
asm("orr r3, r3, #0x80000000 "); // r3=exponent 801E
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,2);
asm("mov r2, r2, lsl r12 ");
asm("sub r3, r3, r12, lsl #16 ");
#else
asm("cmp r2, #0x10000 "); // normalise mantissa, decrementing exponent as needed
asm("movcc r2, r2, lsl #16 ");
asm("subcc r3, r3, #0x100000 ");
asm("cmp r2, #0x1000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r3, r3, #0x080000 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r3, r3, #0x040000 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r3, r3, #0x020000 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("subcc r3, r3, #0x010000 ");
#endif
asm("ConvertUintToTRealX0: ");
asm("mov r1, #0 "); // low order word of mantissa = 0
__JUMP(,lr);
}
__NAKED__ EXPORT_C void TRealX::SetZero(TBool /*aNegative*/)
/**
Sets the value of this extended precision object to zero.
@param aNegative ETrue, the value is a negative zero;
EFalse, the value is a positive zero, this is the default.
*/
{
asm("mov r3, #0 ");
asm("cmp r1, #0 ");
asm("movne r3, #1 ");
asm("mov r2, #0 ");
asm("mov r1, #0 ");
asm("stmia r0, {r1,r2,r3} ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C void TRealX::SetNaN()
/**
Sets the value of this extended precision object to 'not a number'.
*/
{
asm("ldr r3, [pc, #__RealIndefiniteExponent-.-8] ");
asm("mov r2, #0xC0000000 ");
asm("mov r1, #0 ");
asm("stmia r0, {r1,r2,r3} ");
__JUMP(,lr);
asm("__RealIndefiniteExponent: ");
asm(".word 0xFFFF0001 ");
}
__NAKED__ EXPORT_C void TRealX::SetInfinite(TBool /*aNegative*/)
/**
Sets the value of this extended precision object to infinity.
@param aNegative ETrue, the value is a negative zero;
EFalse, the value is a positive zero.
*/
{
asm("ldr r3, [pc, #__InfiniteExponent-.-8] ");
asm("cmp r1, #0 ");
asm("orrne r3, r3, #1 ");
asm("mov r2, #0x80000000 ");
asm("mov r1, #0 ");
asm("stmia r0, {r1,r2,r3} ");
__JUMP(,lr);
asm("__InfiniteExponent: ");
asm(".word 0xFFFF0000 ");
}
__NAKED__ EXPORT_C TBool TRealX::IsZero() const
/**
Determines whether the extended precision value is zero.
@return True, if the extended precision value is zero, false, otherwise.
*/
{
asm("ldr r1, [r0, #8] "); // get exponent word
asm("mov r0, #0 "); // default return value is 0
asm("cmp r1, #0x10000 "); // is exponent=0 ?
asm("movcc r0, #1 "); // if so return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool TRealX::IsNaN() const
/**
Determines whether the extended precision value is 'not a number'.
@return True, if the extended precision value is 'not a number',
false, otherwise.
*/
{
asm("ldmia r0, {r1,r2,r3} ");
asm("mov r0, #0 "); // default return value is 0
asm("cmn r3, #0x10000 "); // check for exponent 65535
asm("bcc 1f "); // branch if not
asm("cmp r2, #0x80000000 "); // check if infinity
asm("cmpeq r1, #0 ");
asm("movne r0, #1 "); // if not, return 1
asm("1: ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool TRealX::IsInfinite() const
/**
Determines whether the extended precision value has a finite value.
@return True, if the extended precision value is finite,
false, if the value is 'not a number' or is infinite,
*/
{
asm("ldmia r0, {r1,r2,r3} ");
asm("mov r0, #0 "); // default return value is 0
asm("cmn r3, #0x10000 "); // check for exponent 65535
asm("bcc 1f "); // branch if not
asm("cmp r2, #0x80000000 "); // check if infinity
asm("cmpeq r1, #0 ");
asm("moveq r0, #1 "); // if it is, return 1
asm("1: ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool TRealX::IsFinite() const
/**
Determines whether the extended precision value has a finite value.
@return True, if the extended precision value is finite,
false, if the value is 'not a number' or is infinite,
*/
{
asm("ldr r1, [r0, #8] "); // get exponent word
asm("mov r0, #0 "); // default return value is 0
asm("cmn r1, #0x10000 "); // is exponent=65535 (infinity or NaN) ?
asm("movcc r0, #1 "); // if not return 1
__JUMP(,lr);
}
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX(TReal32 /*aReal*/) __SOFTFP
/**
Constructs an extended precision object from
a single precision floating point number.
@param aReal The single precision floating point value.
*/
{
// fall through
}
#endif
__NAKED__ EXPORT_C TRealX& TRealX::operator=(TReal32 /*aReal*/) __SOFTFP
/**
Assigns the specified single precision floating point number to
this extended precision object.
@param aReal The single precision floating point value.
@return A reference to this extended precision object.
*/
{
asm("stmfd sp!, {lr} ");
asm("bl ConvertTReal32ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
__POPRET("");
}
__NAKED__ EXPORT_C TInt TRealX::Set(TReal32 /*aReal*/) __SOFTFP
/**
Gives this extended precision object a new value taken from
a single precision floating point number.
@param aReal The single precision floating point value.
@return KErrNone, if a valid number;
KErrOverflow, if the number is infinite;
KErrArgument, if not a number.
*/
{
// aReal is in r1 on entry
// sign in bit 31, exponent in 30-23, mantissa (non-integer bits) in 22-0
asm("stmfd sp!, {lr} ");
asm("bl ConvertTReal32ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r0, #0 "); // if neither, return KErrNone
asm("bcc trealx_set_treal32_0 ");
asm("cmp r2, #0x80000000 "); // check for infinity
asm("mvneq r0, #8 "); // if so, return KErrOverflow
asm("mvnne r0, #5 "); // else return KErrArgument
asm("trealx_set_treal32_0: ");
__POPRET("");
// Convert 32-bit real in r1 to TRealX in r1,r2,r3
// r0 unmodified, r1,r2,r3,r12 modified
asm("ConvertTReal32ToTRealX: ");
asm("mov r3, r1, lsr #7 "); // r3 bits 16-31 = TReal32 exponent
asm("ands r3, r3, #0x00FF0000 ");
asm("mov r2, r1, lsl #8 "); // r2 = TReal32 mantissa << 8, bit 31 not yet in
asm("orrne r2, r2, #0x80000000 "); // if not zero/denormal, put in implied integer bit
asm("orr r3, r3, r1, lsr #31 "); // r3 bit 0 = sign bit
asm("mov r1, #0 "); // low word of mantissa = 0
asm("beq ConvertTReal32ToTRealX0 "); // branch if zero/denormal
asm("cmp r3, #0x00FF0000 "); // check for infinity or NaN
asm("orrcs r3, r3, #0xFF000000 "); // if infinity or NaN, exponent = FFFF
asm("addcc r3, r3, #0x7F000000 "); // else exponent = TReal32 exponent + 7F80
asm("addcc r3, r3, #0x00800000 ");
__JUMP(,lr);
asm("ConvertTReal32ToTRealX0: "); // come here if zero or denormal
asm("adds r2, r2, r2 "); // shift mantissa left one more and check if zero
__JUMP(eq,lr);
asm("add r3, r3, #0x7F000000 "); // else exponent = 7F80 (highest denormal exponent)
asm("add r3, r3, #0x00800000 ");
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,2);
asm("mov r2, r2, lsl r12 ");
asm("sub r3, r3, r12, lsl #16 ");
#else
asm("cmp r2, #0x10000 "); // normalise mantissa, decrementing exponent as needed
asm("movcc r2, r2, lsl #16 ");
asm("subcc r3, r3, #0x100000 ");
asm("cmp r2, #0x1000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r3, r3, #0x080000 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r3, r3, #0x040000 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r3, r3, #0x020000 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("subcc r3, r3, #0x010000 ");
#endif
__JUMP(,lr);
}
#ifndef __EABI_CTORS__
__NAKED__ EXPORT_C TRealX::TRealX(TReal64 /*aReal*/) __SOFTFP
/**
Constructs an extended precision object from
a double precision floating point number.
@param aReal The double precision floating point value.
*/
{
// fall through
}
#endif
__NAKED__ EXPORT_C TRealX& TRealX::operator=(TReal64 /*aReal*/) __SOFTFP
/**
Assigns the specified double precision floating point number to
this extended precision object.
@param aReal The double precision floating point value.
@return A reference to this extended precision object.
*/
{
asm("stmfd sp!, {lr} ");
asm("bl ConvertTReal64ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
__POPRET("");
}
__NAKED__ EXPORT_C TInt TRealX::Set(TReal64 /*aReal*/) __SOFTFP
/**
Gives this extended precision object a new value taken from
a double precision floating point number.
@param aReal The double precision floating point value.
@return KErrNone, if a valid number;
KErrOverflow, if the number is infinite;
KErrArgument, if not a number.
*/
{
// aReal is in r1,r2 on entry
// sign in bit 31 of r1, exponent in 30-20 of r1
// mantissa (non-integer bits) in 19-0 of r1 (high) and r2 (low)
asm("stmfd sp!, {lr} ");
asm("bl ConvertTReal64ToTRealX ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r0, #0 "); // if neither, return KErrNone
asm("bcc trealx_set_treal64_0 ");
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("mvneq r0, #8 "); // if so, return KErrOverflow
asm("mvnne r0, #5 "); // else return KErrArgument
asm("trealx_set_treal64_0: ");
__POPRET("");
// convert TReal64 in r1,r2 in GCC and r2 and r3 in RVCT
// if __DOUBLE_WORDS_SWAPPED__ r1=sign,exp,high mant, r2=low mant
// else r1 unused , r2=low mant, r3=sign,exp,high mant (as a result of EABI alignment reqs)
// into TRealX in r1,r2,r3 (r2,r1=mant high,low r3=exp,flag,sign)
// r0 unmodified, r1,r2,r3,r12 modified
asm("ConvertTReal64ToTRealX: ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("mov r12, r2 "); // ls word of mantissa into r12
#else
asm("mov r12, r2 "); // ls word of mantissa into r12
asm("mov r1, r3 ");
#endif
asm("mov r3, r1, lsr #20 "); // sign and exp into bottom 12 bits of r3
asm("mov r2, r1, lsl #11 "); // left justify mantissa in r2,r1
asm("mov r3, r3, lsl #16 "); // and into bits 16-27
asm("bics r3, r3, #0x08000000 "); // remove sign, leaving exponent in bits 16-26
asm("orr r2, r2, r12, lsr #21 ");
asm("orrne r2, r2, #0x80000000 "); // if not zero/denormal, put in implied integer bit
asm("orr r3, r3, r1, lsr #31 "); // sign bit into bit 0 of r3
asm("mov r1, r12, lsl #11 ");
asm("beq ConvertTReal64ToTRealX0 "); // branch if zero or denormal
asm("mov r12, r3, lsl #5 "); // exponent into bits 21-31 of r12
asm("cmn r12, #0x00200000 "); // check if exponent=7FF (infinity or NaN)
asm("addcs r3, r3, #0xF8000000 "); // if so, result exponent=FFFF
asm("addcc r3, r3, #0x7C000000 "); // else result exponent = TReal64 exponent + 7C00
__JUMP(,lr);
asm("ConvertTReal64ToTRealX0: "); // come here if zero or denormal
asm("adds r1, r1, r1 "); // shift mantissa left one more bit
asm("adcs r2, r2, r2 ");
asm("cmpeq r1, #0 "); // and test for zero
__JUMP(eq,lr);
asm("add r3, r3, #0x7C000000 "); // else exponent=7C00 (highest denormal exponent)
asm("cmp r2, #0 "); // normalise - first check if r2=0
asm("moveq r2, r1 "); // if so, shift up by 32
asm("moveq r1, #0 ");
asm("subeq r3, r3, #0x200000 "); // and subtract 32 from exponent
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,2);
asm("mov r2, r2, lsl r12 ");
asm("rsb r12, r12, #32 ");
asm("orr r2, r2, r1, lsr r12 ");
asm("rsb r12, r12, #32 ");
#else
asm("mov r12, #32 "); // 32-number of left-shifts needed to normalise
asm("cmp r2, #0x10000 "); // calculate number required
asm("movcc r2, r2, lsl #16 ");
asm("subcc r12, r12, #16 ");
asm("cmp r2, #0x1000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r12, r12, #8 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r12, r12, #4 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r12, r12, #2 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("subcc r12, r12, #1 "); // r2 is now normalised
asm("orr r2, r2, r1, lsr r12 "); // shift r1 left into r2
asm("rsb r12, r12, #32 ");
#endif
asm("mov r1, r1, lsl r12 ");
asm("sub r3, r3, r12, lsl #16 "); // exponent -= number of left shifts
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::operator TInt() const
/**
Gets the extended precision value as a signed integer value.
The operator returns:
1. zero , if the extended precision value is not a number
2. 0x7FFFFFFF, if the value is positive and too big to fit into a TInt.
3. 0x80000000, if the value is negative and too big to fit into a TInt.
*/
{
asm("ldmia r0, {r1,r2,r3} "); // get value into r1,r2,r3
asm("ConvertTRealXToInt: ");
asm("mov r12, #0x8000 "); // r12=0x801E
asm("orr r12, r12, #0x001E ");
asm("subs r12, r12, r3, lsr #16 "); // r12=801E-exponent
asm("bls ConvertTRealXToInt1 "); // branch if exponent>=801E
asm("cmp r12, #31 "); // test if exponent<7FFF
asm("movhi r0, #0 "); // if so, underflow result to zero
__JUMP(hi,lr);
asm("mov r0, r2, lsr r12 "); // shift mantissa right to form integer
asm("tst r3, #1 "); // check sign bit
asm("rsbne r0, r0, #0 "); // if negative, r0=-r0
__JUMP(,lr);
asm("ConvertTRealXToInt1: ");
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("bcc ConvertTRealXToInt2 "); // branch if neither
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("movne r0, #0 "); // if NaN, return 0
__JUMP(ne,lr);
asm("ConvertTRealXToInt2: ");
asm("mov r0, #0x80000000 "); // return 0x80000000 if -ve overflow, 0x7FFFFFFF if +ve
asm("movs r3, r3, lsr #1 ");
asm("sbc r0, r0, #0 ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::operator TUint() const
/**
Returns the extended precision value as an unsigned signed integer value.
The operator returns:
1. zero, if the extended precision value is not a number
2. 0xFFFFFFFF, if the value is positive and too big to fit into a TUint.
3. zero, if the value is negative and too big to fit into a TUint.
*/
{
asm("ldmia r0, {r1,r2,r3} "); // get value into r1,r2,r3
asm("ConvertTRealXToUint: ");
asm("mov r12, #0x8000 "); // r12=0x801E
asm("orr r12, r12, #0x001E ");
asm("subs r12, r12, r3, lsr #16 "); // r12=801E-exponent
asm("bcc ConvertTRealXToUint1 "); // branch if exponent>801E
asm("cmp r12, #31 "); // test if exponent<7FFF
asm("movhi r0, #0 "); // if so, underflow result to zero
__JUMP(hi,lr);
asm("tst r3, #1 "); // check sign bit
asm("moveq r0, r2, lsr r12 "); // if +ve, shift mantissa right to form integer
asm("movne r0, #0 "); // if negative, r0=0
__JUMP(,lr);
asm("ConvertTRealXToUint1: ");
asm("mov r0, #0 "); // r0=0 initially
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("bcc ConvertTRealXToUint2 "); // branch if neither
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
__JUMP(ne,lr);
asm("ConvertTRealXToUint2: ");
asm("movs r3, r3, lsr #1 "); // sign bit into carry
asm("sbc r0, r0, #0 "); // r0=0 if -ve, 0xFFFFFFFF if +ve
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::operator TInt64() const
/**
Returns the extended precision value as a 64 bit integer value.
The operator returns:
1. zero, if the extended precision value is not a number
2. 0x7FFFFFFF FFFFFFFF, if the value is positive and too big to fit
into a TInt64
3. 0x80000000 00000000, if the value is negative and too big to fit
into a TInt.
*/
{
// r0 = this, result in r1:r0
asm("ldmia r0, {r0,r1,r2} "); // get value into r0,r1,r2
asm("ConvertTRealXToInt64: ");
asm("mov r3, #0x8000 "); // r3=0x803E
asm("orr r3, r3, #0x003E ");
asm("subs r3, r3, r2, lsr #16 "); // r3=803E-exponent
asm("bls ConvertTRealXToInt64a "); // branch if exponent>=803E
asm("cmp r3, #63 "); // test if exponent<7FFF
asm("movhi r1, #0 "); // if so, underflow result to zero
asm("movhi r0, #0 ");
__JUMP(hi,lr);
asm("cmp r3, #32 "); // >=32 shifts required?
asm("subcs r3, r3, #32 "); // if so, r3-=32
asm("movcs r0, r1, lsr r3 "); // r1:r0 >>= (r3+32)
asm("movcs r1, #0 ");
asm("movcc r0, r0, lsr r3 "); // else r1:r0>>=r3
asm("rsbcc r3, r3, #32 ");
asm("orrcc r0, r0, r1, lsl r3 ");
asm("rsbcc r3, r3, #32 ");
asm("movcc r1, r1, lsr r3 "); // r1:r0 = absolute integer
asm("tst r2, #1 "); // check sign bit
__JUMP(eq,lr);
asm("rsbs r0, r0, #0 "); // else negate answer
asm("rsc r1, r1, #0 ");
__JUMP(,lr);
asm("ConvertTRealXToInt64a: ");
asm("cmn r2, #0x10000 "); // check for infinity or NaN
asm("bcc ConvertTRealXToInt64b "); // branch if neither
asm("cmp r1, #0x80000000 "); // check for infinity
asm("cmpeq r0, #0 ");
asm("movne r1, #0 "); // if NaN, return 0
asm("movne r0, #0 ");
__JUMP(ne,lr);
asm("ConvertTRealXToInt64b: ");
asm("mov r1, #0x80000000 "); // return KMaxTInt64/KMinTInt64 depending on sign
asm("mov r0, #0 ");
asm("movs r2, r2, lsr #1 ");
asm("sbcs r0, r0, #0 ");
asm("sbc r1, r1, #0 ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::operator TReal32() const __SOFTFP
/**
Returns the extended precision value as
a single precision floating point value.
*/
{
asm("ldmia r0, {r1,r2,r3} "); // r1,r2,r3=input value
// Convert TRealX in r1,r2,r3 to TReal32 in r0
asm("ConvertTRealXToTReal32: ");
asm("mov r12, #0x8000 ");
asm("orr r12, r12, #0x007F "); // r12=0x807F
asm("cmp r3, r12, lsl #16 "); // check if exponent>=807F
asm("bcs ConvertTRealXToTReal32a "); // branch if it is
asm("sub r12, r12, #0x00FF "); // r12=0x7F80
asm("rsbs r12, r12, r3, lsr #16 "); // r12=exp in - 7F80 = result exponent if in range
asm("bgt ConvertTRealXToTReal32b "); // branch if normalised result
asm("cmn r12, #23 "); // check for total underflow or zero
asm("movlt r0, r3, lsl #31 "); // in this case, return zero with appropriate sign
__JUMP(lt,lr);
asm("add r12, r12, #31 "); // r12=32-mantissa shift required = 32-(1-r12)
asm("movs r0, r1, lsl r12 "); // r0=lost bits when r2:r1 is shifted
asm("bicne r3, r3, #0x300 "); // if these are not zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 ");
asm("mov r1, r1, lsr r0 ");
asm("orr r1, r1, r2, lsl r12 ");
asm("mov r2, r2, lsr r0 "); // r2 top 24 bits now give unrounded result mantissa
asm("mov r12, #0 "); // result exponent will be zero
asm("ConvertTRealXToTReal32b: ");
asm("movs r0, r2, lsl #24 "); // top 8 truncated bits into top byte of r0
asm("bpl ConvertTRealXToTReal32c "); // if top bit clear, truncate
asm("cmp r0, #0x80000000 ");
asm("cmpeq r1, #0 "); // compare rounding bits to 1000...
asm("bhi ConvertTRealXToTReal32d "); // if >, round up
asm("movs r0, r3, lsl #23 "); // round up flag into C, round down flag into N
asm("bcs ConvertTRealXToTReal32c "); // if rounded up, truncate
asm("bmi ConvertTRealXToTReal32d "); // if rounded down, round up
asm("tst r2, #0x100 "); // else round to even - test LSB of result mantissa
asm("beq ConvertTRealXToTReal32c "); // if zero, truncate, else round up
asm("ConvertTRealXToTReal32d: "); // come here to round up
asm("adds r2, r2, #0x100 "); // increment the mantissa
asm("movcs r2, #0x80000000 "); // if carry, mantissa=800000
asm("addcs r12, r12, #1 "); // and increment exponent
asm("cmpmi r12, #1 "); // if mantissa normalised, check exponent>0
asm("movmi r12, #1 "); // if normalised and exponent=0, set exponent to 1
asm("ConvertTRealXToTReal32c: "); // come here to truncate
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, r12, lsl #23 "); // exponent into r0 bits 23-30
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("orr r0, r0, r2, lsr #8 "); // non-integer mantissa bits into r0 bits 0-22
__JUMP(,lr);
asm("ConvertTRealXToTReal32a: "); // come here if overflow, infinity or NaN
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r2, #0 "); // if not, set mantissa to 0 for infinity result
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, #0x7F000000 "); // r0 bits 23-30 = FF = exponent
asm("orr r0, r0, #0x00800000 ");
asm("orr r0, r0, r2, lsr #8 "); // r0 bits 0-22 = result mantissa
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::operator TReal64() const __SOFTFP
/**
Returns the extended precision value as
a double precision floating point value.
*/
{
asm("ldmia r0, {r1,r2,r3} "); // r1,r2,r3=input value
// Convert TRealX in r1,r2,r3 to TReal64 in r0,r1
// if __DOUBLE_WORDS_SWAPPED__ r0=sign,exp,high mant, r1=low mant
// else r0, r1 reversed
asm("ConvertTRealXToTReal64: ");
asm("mov r12, #0x8300 ");
asm("orr r12, r12, #0x00FF "); // r12=0x83FF
asm("cmp r3, r12, lsl #16 "); // check if exponent>=83FF
asm("bcs ConvertTRealXToTReal64a "); // branch if it is
asm("mov r12, #0x7C00 ");
asm("rsbs r12, r12, r3, lsr #16 "); // r12=exp in - 7C00 = result exponent if in range
asm("bgt ConvertTRealXToTReal64b "); // branch if normalised result
asm("cmn r12, #52 "); // check for total underflow or zero
asm("movlt r0, r3, lsl #31 "); // in this case, return zero with appropriate sign
asm("movlt r1, #0 ");
asm("blt ConvertTRealXToTReal64_end ");
asm("adds r12, r12, #31 "); // check if >=32 shifts needed, r12=32-shift count
asm("ble ConvertTRealXToTReal64e "); // branch if >=32 shifts needed
asm("movs r0, r1, lsl r12 "); // r0=lost bits when r2:r1 is shifted
asm("bicne r3, r3, #0x300 "); // if these are not zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 "); // r0=shift count
asm("mov r1, r1, lsr r0 ");
asm("orr r1, r1, r2, lsl r12 ");
asm("mov r2, r2, lsr r0 "); // r2:r1 top 53 bits = unrounded result mantissa
asm("b ConvertTRealXToTReal64f ");
asm("ConvertTRealXToTReal64e: ");
asm("add r12, r12, #32 "); // r12=64-shift count
asm("cmp r1, #0 "); // r1 bits are all lost - test them
asm("moveqs r0, r2, lsl r12 "); // if zero, test lost bits from r2
asm("bicne r3, r3, #0x300 "); // if lost bits not all zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 "); // r0=shift count-32
asm("mov r1, r2, lsr r0 "); // shift r2:r1 right
asm("mov r2, #0 ");
asm("ConvertTRealXToTReal64f: ");
asm("mov r12, #0 "); // result exponent will be zero for denormals
asm("ConvertTRealXToTReal64b: ");
asm("movs r0, r1, lsl #21 "); // 11 rounding bits to top of r0
asm("bpl ConvertTRealXToTReal64c "); // if top bit clear, truncate
asm("cmp r0, #0x80000000 "); // compare rounding bits to 10000000000
asm("bhi ConvertTRealXToTReal64d "); // if >, round up
asm("movs r0, r3, lsl #23 "); // round up flag into C, round down flag into N
asm("bcs ConvertTRealXToTReal64c "); // if rounded up, truncate
asm("bmi ConvertTRealXToTReal64d "); // if rounded down, round up
asm("tst r1, #0x800 "); // else round to even - test LSB of result mantissa
asm("beq ConvertTRealXToTReal64c "); // if zero, truncate, else round up
asm("ConvertTRealXToTReal64d: "); // come here to round up
asm("adds r1, r1, #0x800 "); // increment the mantissa
asm("adcs r2, r2, #0 ");
asm("movcs r2, #0x80000000 "); // if carry, mantissa=10000...0
asm("addcs r12, r12, #1 "); // and increment exponent
asm("cmpmi r12, #1 "); // if mantissa normalised, check exponent>0
asm("movmi r12, #1 "); // if normalised and exponent=0, set exponent to 1
asm("ConvertTRealXToTReal64c: "); // come here to truncate
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, r12, lsl #20 "); // exponent into r0 bits 20-30
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("orr r0, r0, r2, lsr #11 "); // non-integer mantissa bits into r0 bits 0-19
asm("mov r1, r1, lsr #11 "); // and r1
asm("orr r1, r1, r2, lsl #21 ");
asm("b ConvertTRealXToTReal64_end ");
asm("ConvertTRealXToTReal64a: "); // come here if overflow, infinity or NaN
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r2, #0 "); // if not, set mantissa to 0 for infinity result
asm("movcc r1, #0 ");
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, #0x7F000000 "); // r0 bits 20-30 = 7FF = exponent
asm("orr r0, r0, #0x00F00000 ");
asm("orr r0, r0, r2, lsr #11 "); // r0 bits 0-19 = result mantissa high bits
asm("mov r1, r1, lsr #11 "); // and r1=result mantissa low bits
asm("orr r1, r1, r2, lsl #21 ");
asm("ConvertTRealXToTReal64_end: ");
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r2, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r2 ");
#endif
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::GetTReal(TReal32& /*aVal*/) const
/**
Extracts the extended precision value as
a single precision floating point value.
@param aVal A reference to a single precision object which contains
the result of the operation.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
asm("stmfd sp!, {r4,lr} ");
asm("mov r4, r1 ");
asm("ldmia r0, {r1,r2,r3} "); // r1,r2,r3=input value
asm("bl TRealXGetTReal32 ");
asm("str r0, [r4] "); // store converted TReal32
asm("mov r0, r12 "); // return value into r0
__POPRET("r4,");
// Convert TRealX in r1,r2,r3 to TReal32 in r0
// Return error code in r12
// r0-r3, r12 modified
asm("TRealXGetTReal32: ");
asm("mov r12, #0x8000 ");
asm("orr r12, r12, #0x007F "); // r12=0x807F
asm("cmp r3, r12, lsl #16 "); // check if exponent>=807F
asm("bcs TRealXGetTReal32a "); // branch if it is
asm("sub r12, r12, #0x00FF "); // r12=0x7F80
asm("rsbs r12, r12, r3, lsr #16 "); // r12=exp in - 7F80 = result exponent if in range
asm("bgt TRealXGetTReal32b "); // branch if normalised result
asm("cmn r12, #23 "); // check for total underflow or zero
asm("bge TRealXGetTReal32e "); // skip if not
asm("mov r0, r3, lsl #31 "); // else return zero with appropriate sign
asm("mov r1, #0 ");
asm("cmp r3, #0x10000 "); // check for zero
asm("movcc r12, #0 "); // if zero return KErrNone
asm("mvncs r12, #9 "); // else return KErrUnderflow
__JUMP(,lr);
asm("TRealXGetTReal32e: ");
asm("add r12, r12, #31 "); // r12=32-mantissa shift required = 32-(1-r12)
asm("movs r0, r1, lsl r12 "); // r0=lost bits when r2:r1 is shifted
asm("bicne r3, r3, #0x300 "); // if these are not zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 ");
asm("mov r1, r1, lsr r0 ");
asm("orr r1, r1, r2, lsl r12 ");
asm("mov r2, r2, lsr r0 "); // r2 top 24 bits now give unrounded result mantissa
asm("mov r12, #0 "); // result exponent will be zero
asm("TRealXGetTReal32b: ");
asm("movs r0, r2, lsl #24 "); // top 8 truncated bits into top byte of r0
asm("bpl TRealXGetTReal32c "); // if top bit clear, truncate
asm("cmp r0, #0x80000000 ");
asm("cmpeq r1, #0 "); // compare rounding bits to 1000...
asm("bhi TRealXGetTReal32d "); // if >, round up
asm("movs r0, r3, lsl #23 "); // round up flag into C, round down flag into N
asm("bcs TRealXGetTReal32c "); // if rounded up, truncate
asm("bmi TRealXGetTReal32d "); // if rounded down, round up
asm("tst r2, #0x100 "); // else round to even - test LSB of result mantissa
asm("beq TRealXGetTReal32c "); // if zero, truncate, else round up
asm("TRealXGetTReal32d: "); // come here to round up
asm("adds r2, r2, #0x100 "); // increment the mantissa
asm("movcs r2, #0x80000000 "); // if carry, mantissa=800000
asm("addcs r12, r12, #1 "); // and increment exponent
asm("cmpmi r12, #1 "); // if mantissa normalised, check exponent>0
asm("movmi r12, #1 "); // if normalised and exponent=0, set exponent to 1
asm("TRealXGetTReal32c: "); // come here to truncate
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, r12, lsl #23 "); // exponent into r0 bits 23-30
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("orr r0, r0, r2, lsr #8 "); // non-integer mantissa bits into r0 bits 0-22
asm("cmp r12, #0xFF "); // check for overflow
asm("mvneq r12, #8 "); // if overflow, return KErrOverflow
__JUMP(eq,lr);
asm("bics r1, r0, #0x80000000 "); // check for underflow
asm("mvneq r12, #9 "); // if underflow return KErrUnderflow
asm("movne r12, #0 "); // else return KErrNone
__JUMP(,lr);
asm("TRealXGetTReal32a: "); // come here if overflow, infinity or NaN
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r2, #0 "); // if not, set mantissa to 0 for infinity result
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, #0x7F000000 "); // r0 bits 23-30 = FF = exponent
asm("orr r0, r0, #0x00800000 ");
asm("orr r0, r0, r2, lsr #8 "); // r0 bits 0-22 = result mantissa
asm("movs r12, r0, lsl #9 "); // check if result is infinity or NaN
asm("mvneq r12, #8 "); // if infinity return KErrOverflow
asm("mvnne r12, #5 "); // else return KErrArgument
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::GetTReal(TReal64& /*aVal*/) const
/**
Extracts the extended precision value as
a double precision floating point value.
@param aVal A reference to a double precision object which
contains the result of the operation.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
asm("stmfd sp!, {r4,lr} ");
asm("mov r4, r1 ");
asm("ldmia r0, {r1,r2,r3} "); // r1,r2,r3=input value
asm("bl TRealXGetTReal64 ");
asm("stmia r4, {r0,r1} "); // store converted TReal64
asm("mov r0, r12 "); // return value into r0
__POPRET("r4,");
// Convert TRealX in r1,r2,r3 to TReal64 in r0,r1
// Return error code in r12
// r0-r3, r12 modified
asm("TRealXGetTReal64: ");
asm("mov r12, #0x8300 ");
asm("orr r12, r12, #0x00FF "); // r12=0x83FF
asm("cmp r3, r12, lsl #16 "); // check if exponent>=83FF
asm("bcs TRealXGetTReal64a "); // branch if it is
asm("mov r12, #0x7C00 ");
asm("rsbs r12, r12, r3, lsr #16 "); // r12=exp in - 7C00 = result exponent if in range
asm("bgt TRealXGetTReal64b "); // branch if normalised result
asm("cmn r12, #52 "); // check for total underflow or zero
asm("bge TRealXGetTReal64g "); // skip if not
asm("mov r0, r3, lsl #31 "); // else return zero with appropriate sign
asm("mov r1, #0 ");
asm("cmp r3, #0x10000 "); // check for zero
asm("movcc r12, #0 "); // if zero return KErrNone
asm("mvncs r12, #9 "); // else return KErrUnderflow
asm("b TRealXGetTReal64_end ");
asm("TRealXGetTReal64g: ");
asm("adds r12, r12, #31 "); // check if >=32 shifts needed, r12=32-shift count
asm("ble TRealXGetTReal64e "); // branch if >=32 shifts needed
asm("movs r0, r1, lsl r12 "); // r0=lost bits when r2:r1 is shifted
asm("bicne r3, r3, #0x300 "); // if these are not zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 "); // r0=shift count
asm("mov r1, r1, lsr r0 ");
asm("orr r1, r1, r2, lsl r12 ");
asm("mov r2, r2, lsr r0 "); // r2:r1 top 53 bits = unrounded result mantissa
asm("b TRealXGetTReal64f ");
asm("TRealXGetTReal64e: ");
asm("add r12, r12, #32 "); // r12=64-shift count
asm("cmp r1, #0 "); // r1 bits are all lost - test them
asm("moveqs r0, r2, lsl r12 "); // if zero, test lost bits from r2
asm("bicne r3, r3, #0x300 "); // if lost bits not all zero, set rounded down flag
asm("orrne r3, r3, #0x100 ");
asm("rsb r0, r12, #32 "); // r0=shift count-32
asm("mov r1, r2, lsr r0 "); // shift r2:r1 right
asm("mov r2, #0 ");
asm("TRealXGetTReal64f: ");
asm("mov r12, #0 "); // result exponent will be zero for denormals
asm("TRealXGetTReal64b: ");
asm("movs r0, r1, lsl #21 "); // 11 rounding bits to top of r0
asm("bpl TRealXGetTReal64c "); // if top bit clear, truncate
asm("cmp r0, #0x80000000 "); // compare rounding bits to 10000000000
asm("bhi TRealXGetTReal64d "); // if >, round up
asm("movs r0, r3, lsl #23 "); // round up flag into C, round down flag into N
asm("bcs TRealXGetTReal64c "); // if rounded up, truncate
asm("bmi TRealXGetTReal64d "); // if rounded down, round up
asm("tst r1, #0x800 "); // else round to even - test LSB of result mantissa
asm("beq TRealXGetTReal64c "); // if zero, truncate, else round up
asm("TRealXGetTReal64d: "); // come here to round up
asm("adds r1, r1, #0x800 "); // increment the mantissa
asm("adcs r2, r2, #0 ");
asm("movcs r2, #0x80000000 "); // if carry, mantissa=10000...0
asm("addcs r12, r12, #1 "); // and increment exponent
asm("cmpmi r12, #1 "); // if mantissa normalised, check exponent>0
asm("movmi r12, #1 "); // if normalised and exponent=0, set exponent to 1
asm("TRealXGetTReal64c: "); // come here to truncate
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, r12, lsl #20 "); // exponent into r0 bits 20-30
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("orr r0, r0, r2, lsr #11 "); // non-integer mantissa bits into r0 bits 0-19
asm("mov r1, r1, lsr #11 "); // and r1
asm("orr r1, r1, r2, lsl #21 ");
asm("add r12, r12, #1 ");
asm("cmp r12, #0x800 "); // check for overflow
asm("mvneq r12, #8 "); // if overflow, return KErrOverflow
asm("beq TRealXGetTReal64_end ");
asm("bics r12, r0, #0x80000000 "); // check for underflow
asm("cmpeq r1, #0 ");
asm("mvneq r12, #9 "); // if underflow return KErrUnderflow
asm("movne r12, #0 "); // else return KErrNone
asm("b TRealXGetTReal64_end ");
asm("TRealXGetTReal64a: "); // come here if overflow, infinity or NaN
asm("cmn r3, #0x10000 "); // check for infinity or NaN
asm("movcc r2, #0 "); // if not, set mantissa to 0 for infinity result
asm("movcc r1, #0 ");
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("mov r0, r3, lsl #31 "); // r0 bit 31 = sign bit
asm("orr r0, r0, #0x7F000000 "); // r0 bits 20-30 = 7FF = exponent
asm("orr r0, r0, #0x00F00000 ");
asm("orr r0, r0, r2, lsr #11 "); // r0 bits 0-19 = result mantissa high bits
asm("mov r1, r1, lsr #11 "); // and r1=result mantissa low bits
asm("orr r1, r1, r2, lsl #21 ");
asm("movs r12, r0, lsl #12 "); // check if result is infinity or NaN
asm("cmpeq r1, #0 ");
asm("mvneq r12, #8 "); // if infinity return KErrOverflow
asm("mvnne r12, #5 "); // else return KErrArgument
asm("TRealXGetTReal64_end: ");
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r2, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r2 ");
#endif
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX TRealX::operator+() const
/**
Returns this extended precision number unchanged.
Note that this may also be referred to as a unary plus operator.
@return The extended precision number.
*/
{
asm("ldmia r1, {r2,r3,r12} ");
asm("stmia r0, {r2,r3,r12} ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX TRealX::operator-() const
/**
Negates this extended precision number.
This may also be referred to as a unary minus operator.
@return The negative of the extended precision number.
*/
{
asm("ldmia r1, {r2,r3,r12} ");
asm("eor r12, r12, #1 "); // unary - changes sign bit
asm("stmia r0, {r2,r3,r12} ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TRealX::TRealXOrder TRealX::Compare(const TRealX& /*aVal*/) const
/**
*/
{
asm("stmfd sp!, {r4,r5,r6,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXCompare ");
__POPRET("r4-r6,");
// Compare TRealX in r1,r2,r3 to TRealX in r4,r5,r6
// Return TRealXOrder result in r0
asm("TRealXCompare: ");
asm("cmn r3, #0x10000 "); // check for NaNs/infinity
asm("bcs TRealXCompare1 ");
asm("TRealXCompare6: "); // will come back here if infinity
asm("cmn r6, #0x10000 ");
asm("bcs TRealXCompare2 ");
asm("TRealXCompare7: "); // will come back here if infinity
asm("cmp r3, #0x10000 "); // check for zeros
asm("bcc TRealXCompare3 ");
asm("cmp r6, #0x10000 ");
asm("bcc TRealXCompare4 ");
asm("mov r12, r6, lsl #31 ");
asm("cmp r12, r3, lsl #31 "); // compare signs
asm("movne r0, #4 ");
asm("bne TRealXCompare5 "); // branch if signs different
asm("mov r12, r3, lsr #16 "); // r12=first exponent
asm("cmp r12, r6, lsr #16 "); // compare exponents
asm("cmpeq r2, r5 "); // if equal compare high words of mantissa
asm("cmpeq r1, r4 "); // if equal compare low words of mantissa
asm("moveq r0, #2 "); // if equal return 2
__JUMP(eq,lr);
asm("movhi r0, #4 "); // r0=4 if first exp bigger
asm("movcc r0, #1 "); // else r0=1
asm("TRealXCompare5: ");
asm("tst r3, #1 "); // if signs negative
asm("eorne r0, r0, #5 "); // then switch 1 and 4
__JUMP(,lr);
asm("TRealXCompare3: "); // first operand zero
asm("cmp r6, #0x10000 "); // check if second also zero
asm("movcc r0, #2 "); // if so, return 2
__JUMP(cc,lr);
asm("tst r6, #1 "); // else check sign of operand 2
asm("moveq r0, #1 "); // if +, return 1
asm("movne r0, #4 "); // else return 4
__JUMP(,lr);
asm("TRealXCompare4: "); // second operand zero, first nonzero
asm("tst r3, #1 "); // check sign of operand 1
asm("moveq r0, #4 "); // if +, return 4
asm("movne r0, #1 "); // else return 1
__JUMP(,lr);
asm("TRealXCompare1: "); // first operand NaN or infinity
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("beq TRealXCompare6 "); // if infinity, can handle normally
asm("mov r0, #8 "); // if NaN, return 8 (unordered)
__JUMP(,lr);
asm("TRealXCompare2: "); // second operand NaN or infinity
asm("cmp r5, #0x80000000 "); // check for infinity
asm("cmpeq r4, #0 ");
asm("beq TRealXCompare7 "); // if infinity, can handle normally
asm("mov r0, #8 "); // if NaN, return 8 (unordered)
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::SubEq(const TRealX& /*aVal*/)
/**
Subtracts an extended precision value from this extended precision number.
@param aVal The extended precision value to be subtracted.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, r12 ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::AddEq(const TRealX& /*aVal*/)
/**
Adds an extended precision value to this extended precision number.
@param aVal The extended precision value to be added.
@return KErrNone, if the operation is successful;
KErrOverflow,if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXAdd ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, r12 ");
__JUMP(,lr);
// TRealX subtraction r1,r2,r3 - r4,r5,r6 result in r1,r2,r3
// Error code returned in r12
// Registers r0-r8,r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
asm("TRealXSubtract: ");
asm("eor r6, r6, #1 "); // negate second operand and add
// TRealX addition r1,r2,r3 + r4,r5,r6 result in r1,r2,r3
// Error code returned in r12
// Registers r0-r8,r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
// Note: +0 + +0 = +0, -0 + -0 = -0, +0 + -0 = -0 + +0 = +0,
// +/-0 + X = X + +/-0 = X, X + -X = -X + X = +0
asm("TRealXAdd: ");
asm("mov r12, #0 "); // initialise return value to KErrNone
asm("bic r3, r3, #0x300 "); // clear rounding flags
asm("bic r6, r6, #0x300 "); // clear rounding flags
asm("cmn r3, #0x10000 "); // check if first operand is NaN or infinity
asm("bcs TRealXAdd1 "); // branch if it is
asm("cmn r6, #0x10000 "); // check if second operand is NaN or infinity
asm("bcs TRealXAdd2 "); // branch if it is
asm("cmp r6, #0x10000 "); // check if second operand zero
asm("bcc TRealXAdd3a "); // branch if it is
asm("cmp r3, #0x10000 "); // check if first operand zero
asm("bcc TRealXAdd3 "); // branch if it is
asm("mov r7, #0 "); // r7 will be rounding word
asm("mov r0, r3, lsr #16 "); // r0 = first operand exponent
asm("subs r0, r0, r6, lsr #16 "); // r0 = first exponent - second exponent
asm("beq TRealXAdd8 "); // if equal, no mantissa shifting needed
asm("bhi TRealXAdd4 "); // skip if first exponent bigger
asm("rsb r0, r0, #0 "); // need to shift first mantissa right by r0 to align
asm("mov r8, r1 "); // swap the numbers to the one to be shifted is 2nd
asm("mov r1, r4 ");
asm("mov r4, r8 ");
asm("mov r8, r2 ");
asm("mov r2, r5 ");
asm("mov r5, r8 ");
asm("mov r8, r3 ");
asm("mov r3, r6 ");
asm("mov r6, r8 ");
asm("TRealXAdd4: "); // need to shift 2nd mantissa right by r0 to align
asm("cmp r0, #64 "); // more than 64 shifts needed?
asm("bhi TRealXAdd6 "); // if so, smaller number cannot affect larger
asm("cmp r0, #32 ");
asm("bhi TRealXAdd7 "); // branch if shift count>32
asm("rsb r8, r0, #32 ");
asm("mov r7, r4, lsl r8 "); // shift r5:r4 right into r7
asm("mov r4, r4, lsr r0 ");
asm("orr r4, r4, r5, lsl r8 ");
asm("mov r5, r5, lsr r0 ");
asm("b TRealXAdd8 ");
asm("TRealXAdd7: "); // 64 >= shift count > 32
asm("sub r0, r0, #32 ");
asm("rsb r8, r0, #32 ");
asm("movs r7, r4, lsl r8 "); // test bits lost in shift
asm("orrne r6, r6, #0x100 "); // if not all zero, flag 2nd mantissa rounded down
asm("mov r7, r4, lsr r0 "); // shift r5:r4 right into r7 by 32+r0
asm("orr r7, r7, r5, lsl r8 ");
asm("mov r4, r5, lsr r0 ");
asm("mov r5, #0 ");
asm("TRealXAdd8: "); // mantissas are now aligned
asm("mov r8, r3, lsl #31 "); // r8=sign of first operand
asm("cmp r8, r6, lsl #31 "); // compare signs
asm("bne TRealXSub1 "); // if different, need to do a subtraction
asm("adds r1, r1, r4 "); // signs the same - add mantissas
asm("adcs r2, r2, r5 ");
asm("bcc TRealXAdd9 "); // skip if no carry
asm(".word 0xE1B02062 "); // movs r2, r2, rrx shift carry into mantissa
asm(".word 0xE1B01061 "); // movs r1, r1, rrx
asm(".word 0xE1B07067 "); // movs r7, r7, rrx
asm("orrcs r6, r6, #0x100 "); // if 1 shifted out, flag 2nd mantissa rounded down
asm("add r3, r3, #0x10000 "); // increment exponent
asm("TRealXAdd9: ");
asm("cmp r7, #0x80000000 "); // check rounding word
asm("bcc TRealXAdd10 "); // if <0x80000000 round down
asm("bhi TRealXAdd11 "); // if >0x80000000 round up
asm("tst r6, #0x100 "); // if =0x80000000 check if 2nd mantissa rounded down
asm("bne TRealXAdd11 "); // if so, round up
asm("tst r6, #0x200 "); // if =0x80000000 check if 2nd mantissa rounded up
asm("bne TRealXAdd10 "); // if so, round down
asm("tst r1, #1 "); // else round to even - check LSB
asm("beq TRealXAdd10 "); // if zero, round down
asm("TRealXAdd11: "); // come here to round up
asm("adds r1, r1, #1 "); // increment mantissa
asm("adcs r2, r2, #0 ");
asm("movcs r2, #0x80000000 "); // if carry, mantissa = 80000000 00000000
asm("addcs r3, r3, #0x10000 "); // and increment exponent
asm("cmn r3, #0x10000 "); // check overflow
asm("orrcc r3, r3, #0x200 "); // if no overflow, set rounded-up flag ...
__JUMP(cc,lr);
asm("b TRealXAdd12 "); // if overflow, return infinity
asm("TRealXAdd10: "); // come here to round down
asm("cmn r3, #0x10000 "); // check overflow
asm("bcs TRealXAdd12 "); // if overflow, return infinity
asm("cmp r7, #0 "); // if no overflow check if rounding word is zero
asm("orrne r3, r3, #0x100 "); // if not, set rounded-down flag ...
__JUMP(ne,lr);
asm("and r6, r6, #0x300 "); // else transfer 2nd mantissa rounding flags
asm("orr r3, r3, r6 "); // to result
__JUMP(,lr);
asm("TRealXAdd12: "); // come here if overflow - return infinity
asm("mov r2, #0x80000000 ");
asm("mov r1, #0 ");
asm("mvn r12, #8 "); // and return KErrOverflow
__JUMP(,lr);
asm("TRealXSub1: "); // come here if operand signs differ
asm("tst r6, #0x300 "); // check if 2nd mantissa rounded
asm("eorne r6, r6, #0x300 "); // if so, change rounding
asm("rsbs r7, r7, #0 "); // subtract mantissas r2:r1:0 -= r5:r4:r7
asm("sbcs r1, r1, r4 ");
asm("sbcs r2, r2, r5 ");
asm("bcs TRealXSub2 "); // skip if no borrow
asm("tst r6, #0x300 "); // check if 2nd mantissa rounded
asm("eorne r6, r6, #0x300 "); // if so, change rounding
asm("rsbs r7, r7, #0 "); // negate result
asm("rscs r1, r1, #0 ");
asm("rscs r2, r2, #0 ");
asm("eor r3, r3, #1 "); // and change result sign
asm("TRealXSub2: ");
asm("bne TRealXSub3 "); // skip if mantissa top word is not zero
asm("movs r2, r1 "); // else shift up by 32
asm("mov r1, r7 ");
asm("mov r7, #0 ");
asm("bne TRealXSub3a "); // skip if mantissa top word is not zero now
asm("movs r2, r1 "); // else shift up by 32 again
asm("mov r1, #0 ");
asm("moveq r3, #0 "); // if r2 still zero, result is zero - return +0
__JUMP(eq,lr);
asm("subs r3, r3, #0x00400000 "); // else, decrement exponent by 64
asm("bcs TRealXSub3 "); // if no borrow, proceed
asm("b TRealXSub4 "); // if borrow, underflow
asm("TRealXSub3a: "); // needed one 32-bit shift
asm("subs r3, r3, #0x00200000 "); // so decrement exponent by 32
asm("bcc TRealXSub4 "); // if borrow, underflow
asm("TRealXSub3: "); // r2 is now non-zero; still may need up to 31 shifts
#ifdef __CPU_ARM_HAS_CLZ
CLZ(0,2);
asm("mov r2, r2, lsl r0 ");
#else
asm("mov r0, #0 "); // r0 will be shift count
asm("cmp r2, #0x00010000 ");
asm("movcc r2, r2, lsl #16 ");
asm("addcc r0, r0, #16 ");
asm("cmp r2, #0x01000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("addcc r0, r0, #8 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("addcc r0, r0, #4 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("addcc r0, r0, #2 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 ");
asm("addcc r0, r0, #1 ");
#endif
asm("rsb r8, r0, #32 ");
asm("subs r3, r3, r0, lsl #16 "); // subtract shift count from exponent
asm("bcc TRealXSub4 "); // if borrow, underflow
asm("orr r2, r2, r1, lsr r8 "); // else shift mantissa up
asm("mov r1, r1, lsl r0 ");
asm("orr r1, r1, r7, lsr r8 ");
asm("mov r7, r7, lsl r0 ");
asm("cmp r3, #0x10000 "); // check for underflow
asm("bcs TRealXAdd9 "); // if no underflow, branch to round result
asm("TRealXSub4: "); // come here if underflow
asm("and r3, r3, #1 "); // set exponent to zero, leave sign
asm("mov r2, #0 ");
asm("mov r1, #0 ");
asm("mvn r12, #9 "); // return KErrUnderflow
__JUMP(,lr);
asm("TRealXAdd6: "); // come here if exponents differ by more than 64
asm("mov r8, r3, lsl #31 "); // r8=sign of first operand
asm("cmp r8, r6, lsl #31 "); // compare signs
asm("orreq r3, r3, #0x100 "); // if same, result has been rounded down
asm("orrne r3, r3, #0x200 "); // else result has been rounded up
__JUMP(,lr);
asm("TRealXAdd3a: "); // come here if second operand zero
asm("cmp r3, #0x10000 "); // check if first operand also zero
asm("andcc r3, r3, r6 "); // if so, result is negative iff both zeros negative
asm("andcc r3, r3, #1 ");
__JUMP(,lr);
asm("TRealXAdd3: "); // come here if first operand zero, second nonzero
asm("mov r1, r4 "); // return second operand unchanged
asm("mov r2, r5 ");
asm("mov r3, r6 ");
__JUMP(,lr);
asm("TRealXAdd1: "); // come here if first operand NaN or infinity
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("cmn r6, #0x10000 "); // check 2nd operand for NaN/infinity
asm("mvncc r12, #8 "); // if neither, return KErrOverflow
__JUMP(cc,lr);
asm("cmp r5, #0x80000000 "); // check 2nd operand for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("mov r0, r3, lsl #31 "); // both operands are infinity - check signs
asm("cmp r0, r6, lsl #31 ");
asm("mvneq r12, #8 "); // if same, return KErrOverflow
__JUMP(eq,lr);
// Return 'real indefinite'
asm("TRealXRealIndefinite: ");
asm("ldr r3, [pc, #__RealIndefiniteExponent-.-8] ");
asm("mov r2, #0xC0000000 ");
asm("mov r1, #0 ");
asm("mvn r12, #5 "); // return KErrArgument
__JUMP(,lr);
asm("TRealXAdd2: "); // come here if 2nd operand NaN/infinity, first finite
asm("cmp r5, #0x80000000 "); // check for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("mov r1, r4 "); // else return 2nd operand (infinity)
asm("mov r2, r5 ");
asm("mov r3, r6 ");
asm("mvn r12, #8 "); // return KErrOverflow
__JUMP(,lr);
asm("TRealXBinOpNan: "); // generic routine to process NaNs in binary
// operations
asm("cmn r3, #0x10000 "); // check if first operand is NaN
asm("movcc r0, r1 "); // if not, swap the operands
asm("movcc r1, r4 ");
asm("movcc r4, r0 ");
asm("movcc r0, r2 ");
asm("movcc r2, r5 ");
asm("movcc r5, r0 ");
asm("movcc r0, r3 ");
asm("movcc r3, r6 ");
asm("movcc r6, r0 ");
asm("cmn r6, #0x10000 "); // both operands NaNs?
asm("bcc TRealXBinOpNan1 "); // skip if not
asm("cmp r2, r5 "); // if so, compare the significands
asm("cmpeq r1, r4 ");
asm("movcc r1, r4 "); // r1,r2,r3 will get NaN with larger significand
asm("movcc r2, r5 ");
asm("movcc r3, r6 ");
asm("TRealXBinOpNan1: ");
asm("orr r2, r2, #0x40000000 "); // convert an SNaN to a QNaN
asm("mvn r12, #5 "); // return KErrArgument
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::MultEq(const TRealX& /*aVal*/)
/**
Multiplies this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the multiplier.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow
*/
{
// Version for ARM 3M or later
// Uses umull/umlal
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXMultiply ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, r12 ");
__JUMP(,lr);
// TRealX multiplication r1,r2,r3 * r4,r5,r6 result in r1,r2,r3
// Error code returned in r12
// Registers r0-r7,r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
asm("TRealXMultiply: ");
asm("mov r12, #0 "); // initialise return value to KErrNone
asm("bic r3, r3, #0x300 "); // clear rounding flags
asm("tst r6, #1 ");
asm("eorne r3, r3, #1 "); // Exclusive-OR signs
asm("cmn r3, #0x10000 "); // check if first operand is NaN or infinity
asm("bcs TRealXMultiply1 "); // branch if it is
asm("cmn r6, #0x10000 "); // check if second operand is NaN or infinity
asm("bcs TRealXMultiply2 "); // branch if it is
asm("cmp r3, #0x10000 "); // check if first operand zero
__JUMP(cc,lr); // if so, exit
// Multiply mantissas in r2:r1 and r5:r4, result in r2:r1:r12:r7
asm("umull r7, r12, r1, r4 "); // r7:r12=m1.low*m2.low
asm("movs r0, r6, lsr #16 "); // r0=2nd operand exponent
asm("beq TRealXMultiply3 "); // if zero, return zero
asm("mov r6, #0 "); // clear r6 initially
asm("umlal r12, r6, r1, r5 "); // r6:r12:r7=m1.low*m2, r1 no longer needed
asm("add r0, r0, r3, lsr #16 "); // r0=sum of exponents
asm("tst r3, #1 ");
asm("mov r3, #0 "); // clear r3 initially
asm("umlal r6, r3, r2, r5 "); // r3:r6:r12:r7=m2.low*m1+m2.high*m1.high<<64
// r1,r5 no longer required
asm("orrne lr, lr, #1 "); // save sign in bottom bit of lr
asm("sub r0, r0, #0x7F00 ");
asm("sub r0, r0, #0x00FE "); // r0 now contains result exponent
asm("umull r1, r5, r2, r4 "); // r5:r1=m2.high*m1.low
asm("adds r12, r12, r1 "); // shift left by 32 and add to give final result
asm("adcs r1, r6, r5 ");
asm("adcs r2, r3, #0 "); // final result now in r2:r1:r12:r7
// set flags on final value of r2 (ms word of result)
// normalise the result mantissa
asm("bmi TRealXMultiply4 "); // skip if already normalised
asm("adds r7, r7, r7 "); // else shift left (will only ever need one shift)
asm("adcs r12, r12, r12 ");
asm("adcs r1, r1, r1 ");
asm("adcs r2, r2, r2 ");
asm("sub r0, r0, #1 "); // and decrement exponent by one
// round the result mantissa
asm("TRealXMultiply4: ");
asm("and r3, lr, #1 "); // result sign bit back into r3
asm("orrs r4, r7, r12 "); // check for exact result
asm("beq TRealXMultiply5 "); // skip if exact
asm("cmp r12, #0x80000000 "); // compare bottom 64 bits to 80000000 00000000
asm("cmpeq r7, #0 ");
asm("moveqs r4, r1, lsr #1 "); // if exactly equal, set carry=lsb of result
// so we round up if lsb=1
asm("orrcc r3, r3, #0x100 "); // if rounding down, set rounded-down flag
asm("orrcs r3, r3, #0x200 "); // if rounding up, set rounded-up flag
asm("adcs r1, r1, #0 "); // increment mantissa if necessary
asm("adcs r2, r2, #0 ");
asm("movcs r2, #0x80000000 "); // if carry, set mantissa to 80000000 00000000
asm("addcs r0, r0, #1 "); // and increment result exponent
// check for overflow or underflow and assemble final result
asm("TRealXMultiply5: ");
asm("add r4, r0, #1 "); // need to add 1 to get usable threshold
asm("cmp r4, #0x10000 "); // check if exponent >= 0xFFFF
asm("bge TRealXMultiply6 "); // if so, overflow
asm("cmp r0, #0 "); // check for underflow
asm("orrgt r3, r3, r0, lsl #16 "); // if no underflow, result exponent into r3, ...
asm("movgt r12, #0 "); // ... return KErrNone ...
asm("bicgt pc, lr, #3 ");
// underflow
asm("mvn r12, #9 "); // return KErrUnderflow
asm("bic pc, lr, #3 ");
// overflow
asm("TRealXMultiply6: ");
asm("bic r3, r3, #0x0000FF00 "); // clear rounding flags
asm("orr r3, r3, #0xFF000000 "); // make exponent FFFF for infinity
asm("orr r3, r3, #0x00FF0000 ");
asm("mov r2, #0x80000000 "); // mantissa = 80000000 00000000
asm("mov r1, #0 ");
asm("mvn r12, #8 "); // return KErrOverflow
asm("bic pc, lr, #3 ");
// come here if second operand zero
asm("TRealXMultiply3: ");
asm("mov r1, #0 ");
asm("mov r2, #0 ");
asm("and r3, r3, #1 "); // zero exponent, keep xor sign
asm("mov r12, #0 "); // return KErrNone
asm("bic pc, lr, #3 ");
// First operand NaN or infinity
asm("TRealXMultiply1: ");
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("cmn r6, #0x10000 "); // check 2nd operand for NaN/infinity
asm("bcs TRealXMultiply1a "); // branch if it is
asm("cmp r6, #0x10000 "); // else check if second operand zero
asm("mvncs r12, #8 "); // if not, return infinity and KErrOverflow
asm("biccs pc, lr, #3 ");
asm("b TRealXRealIndefinite "); // else return 'real indefinite'
asm("TRealXMultiply1a: ");
asm("cmp r5, #0x80000000 "); // check 2nd operand for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("mvn r12, #8 "); // else (infinity), return KErrOverflow
asm("bic pc, lr, #3 ");
// Second operand NaN or infinity, first operand finite
asm("TRealXMultiply2: ");
asm("cmp r5, #0x80000000 "); // check for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("cmp r3, #0x10000 "); // if infinity, check if first operand zero
asm("bcc TRealXRealIndefinite "); // if it is, return 'real indefinite'
asm("orr r3, r3, #0xFF000000 "); // else return infinity with xor sign
asm("orr r3, r3, #0x00FF0000 ");
asm("mov r2, #0x80000000 ");
asm("mov r1, #0 ");
asm("mvn r12, #8 "); // return KErrOverflow
asm("bic pc, lr, #3 ");
}
__NAKED__ EXPORT_C TInt TRealX::DivEq(const TRealX& /*aVal*/)
/**
Divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow;
KErrDivideByZero, if the divisor is zero.
*/
{
asm("stmfd sp!, {r0,r4-r9,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXDivide ");
asm("ldmfd sp!, {r0,r4-r9,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, r12 ");
__JUMP(,lr);
// TRealX division r1,r2,r3 / r4,r5,r6 result in r1,r2,r3
// Error code returned in r12
// Registers r0-r9,r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
asm("TRealXDivide: ");
asm("mov r12, #0 "); // initialise return value to KErrNone
asm("bic r3, r3, #0x300 "); // clear rounding flags
asm("tst r6, #1 ");
asm("eorne r3, r3, #1 "); // Exclusive-OR signs
asm("cmn r3, #0x10000 "); // check if dividend is NaN or infinity
asm("bcs TRealXDivide1 "); // branch if it is
asm("cmn r6, #0x10000 "); // check if divisor is NaN or infinity
asm("bcs TRealXDivide2 "); // branch if it is
asm("cmp r6, #0x10000 "); // check if divisor zero
asm("bcc TRealXDivide3 "); // branch if it is
asm("cmp r3, #0x10000 "); // check if dividend zero
__JUMP(cc,lr); // if zero, exit
asm("tst r3, #1 ");
asm("orrne lr, lr, #1 "); // save sign in bottom bit of lr
// calculate result exponent
asm("mov r0, r3, lsr #16 "); // r0=dividend exponent
asm("sub r0, r0, r6, lsr #16 "); // r0=dividend exponent - divisor exponent
asm("add r0, r0, #0x7F00 ");
asm("add r0, r0, #0x00FF "); // r0 now contains result exponent
asm("mov r6, r1 "); // move dividend into r6,r7,r8
asm("mov r7, r2 ");
asm("mov r8, #0 "); // use r8 to hold extra bit shifted up
// r2:r1 will hold result mantissa
asm("mov r2, #1 "); // we will make sure first bit is 1
asm("cmp r7, r5 "); // compare dividend mantissa to divisor mantissa
asm("cmpeq r6, r4 ");
asm("bcs TRealXDivide4 "); // branch if dividend >= divisor
asm("adds r6, r6, r6 "); // else shift dividend left one
asm("adcs r7, r7, r7 "); // ignore carry here
asm("sub r0, r0, #1 "); // decrement result exponent by one
asm("TRealXDivide4: ");
asm("subs r6, r6, r4 "); // subtract divisor from dividend
asm("sbcs r7, r7, r5 ");
// Main mantissa division code
// First calculate the top 32 bits of the result
// Top bit is 1, do 10 lots of 3 bits the one more bit
asm("mov r12, #10 ");
asm("TRealXDivide5: ");
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r2, r2, r2 "); // shift in new result bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r2, r2, r2 "); // shift in new result bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r2, r2, r2 "); // shift in new result bit
asm("subs r12, r12, #1 ");
asm("bne TRealXDivide5 "); // iterate the loop
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r2, r2, r2 "); // shift in new result bit - now have 32 bits
// Now calculate the bottom 32 bits of the result
// Do 8 lots of 4 bits
asm("mov r12, #8 ");
asm("TRealXDivide5a: ");
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r1, r1, r1 "); // shift in new result bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r1, r1, r1 "); // shift in new result bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r1, r1, r1 "); // shift in new result bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r9, r6, r4 "); // subtract divisor from accumulator, result in r9,r3
asm("sbcs r3, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("movcs r6, r9 "); // if no borrow, replace accumulator with result
asm("movcs r7, r3 ");
asm("adcs r1, r1, r1 "); // shift in new result bit
asm("subs r12, r12, #1 ");
asm("bne TRealXDivide5a "); // iterate the loop
// r2:r1 now contains a 64-bit normalised mantissa
// need to do rounding now
asm("and r3, lr, #1 "); // result sign back into r3
asm("orrs r9, r6, r7 "); // check if accumulator zero
asm("beq TRealXDivide6 "); // if it is, result is exact, else generate next bit
asm("adds r6, r6, r6 "); // shift accumulator left by one
asm("adcs r7, r7, r7 ");
asm("adcs r8, r8, r8 ");
asm("subs r6, r6, r4 "); // subtract divisor from accumulator
asm("sbcs r7, r7, r5 ");
asm("movccs r8, r8, lsr #1 "); // if borrow, check for carry from shift
asm("orrcc r3, r3, #0x100 "); // if borrow, round down and set round-down flag
asm("bcc TRealXDivide6 ");
asm("orrs r9, r6, r7 "); // if no borrow, check if exactly half-way
asm("moveqs r9, r1, lsr #1 "); // if exactly half-way, round to even
asm("orrcc r3, r3, #0x100 "); // if C=0, round result down and set round-down flag
asm("bcc TRealXDivide6 ");
asm("orr r3, r3, #0x200 "); // else set round-up flag
asm("adds r1, r1, #1 "); // and round mantissa up
asm("adcs r2, r2, #0 ");
asm("movcs r2, #0x80000000 "); // if carry, mantissa = 80000000 00000000
asm("addcs r0, r0, #1 "); // and increment exponent
// check for overflow or underflow and assemble final result
asm("TRealXDivide6: ");
asm("add r4, r0, #1 "); // need to add 1 to get usable threshold
asm("cmp r4, #0x10000 "); // check if exponent >= 0xFFFF
asm("bge TRealXMultiply6 "); // if so, overflow
asm("cmp r0, #0 "); // check for underflow
asm("orrgt r3, r3, r0, lsl #16 "); // if no underflow, result exponent into r3, ...
asm("movgt r12, #0 "); // ... return KErrNone ...
asm("bicgt pc, lr, #3 ");
// underflow
asm("and r3, r3, #1 "); // set exponent=0, keep sign
asm("mvn r12, #9 "); // return KErrUnderflow
asm("bic pc, lr, #3 ");
// come here if divisor is zero, dividend finite
asm("TRealXDivide3: ");
asm("cmp r3, #0x10000 "); // check if dividend also zero
asm("bcc TRealXRealIndefinite "); // if so, return 'real indefinite'
asm("orr r3, r3, #0xFF000000 "); // else return infinity with xor sign
asm("orr r3, r3, #0x00FF0000 ");
asm("mov r2, #0x80000000 ");
asm("mov r1, #0 ");
asm("mvn r12, #40 "); // return KErrDivideByZero
asm("bic pc, lr, #3 ");
// Dividend is NaN or infinity
asm("TRealXDivide1: ");
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("cmn r6, #0x10000 "); // check 2nd operand for NaN/infinity
asm("mvncc r12, #8 "); // if not, return KErrOverflow
asm("biccc pc, lr, #3 ");
// Dividend=infinity, divisor=NaN or infinity
asm("cmp r5, #0x80000000 "); // check 2nd operand for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("b TRealXRealIndefinite "); // else return 'real indefinite'
// Divisor is NaN or infinity, dividend finite
asm("TRealXDivide2: ");
asm("cmp r5, #0x80000000 "); // check for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("and r3, r3, #1 "); // else return zero with xor sign
asm("bic pc, lr, #3 ");
}
__NAKED__ EXPORT_C TInt TRealX::ModEq(const TRealX& /*aVal*/)
/**
Modulo-divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return KErrNone, if the operation is successful;
KErrTotalLossOfPrecision, if precision is lost;
KErrUnderflow, if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXModulo ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("mov r0, r12 ");
__JUMP(,lr);
// TRealX remainder r1,r2,r3 % r4,r5,r6 result in r1,r2,r3
// Error code returned in r12
// Registers r0-r7,r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
asm("TRealXModulo: ");
asm("mov r12, #0 "); // initialise return value to KErrNone
asm("cmn r3, #0x10000 "); // check if dividend is NaN or infinity
asm("bcs TRealXModulo1 "); // branch if it is
asm("cmn r6, #0x10000 "); // check if divisor is NaN or infinity
asm("bcs TRealXModulo2 "); // branch if it is
asm("cmp r6, #0x10000 "); // check if divisor zero
asm("bcc TRealXRealIndefinite "); // if it is, return 'real indefinite'
asm("mov r0, r3, lsr #16 "); // r0=dividend exponent
asm("subs r0, r0, r6, lsr #16 "); // r0=dividend exponent-divisor exponent
__JUMP(lt,lr);
asm("cmp r0, #64 "); // check if difference >= 64 bits
asm("bcs TRealXModuloLp "); // if so, underflow
asm("b TRealXModulo4 "); // skip left shift on first iteration
asm("TRealXModulo3: ");
asm("adds r1, r1, r1 "); // shift dividend mantissa left one bit
asm("adcs r2, r2, r2 ");
asm("bcs TRealXModulo5 "); // if one shifted out, override comparison
asm("TRealXModulo4: ");
asm("cmp r2, r5 "); // compare dividend to divisor
asm("cmpeq r1, r4 ");
asm("bcc TRealXModulo6 "); // if dividend<divisor, skip
asm("TRealXModulo5: ");
asm("subs r1, r1, r4 "); // if dividend>=divisor, dividend-=divisor
asm("sbcs r2, r2, r5 ");
asm("TRealXModulo6: ");
asm("subs r0, r0, #1 "); // decrement loop count
asm("bpl TRealXModulo3 "); // if more bits to do, loop
asm("orrs r0, r1, r2 "); // test for exact zero result
asm("andeq r3, r3, #1 "); // if so, return zero with same sign as dividend
__JUMP(eq,lr);
asm("and r7, r3, #1 "); // dividend sign bit into r7
asm("mov r3, r6, lsr #16 "); // r3 lower 16 bits=result exponent=divisor exponent
asm("cmp r2, #0 "); // test if upper 32 bits zero
asm("moveq r2, r1 "); // if so, shift left by 32
asm("moveq r1, #0 ");
asm("subeqs r3, r3, #32 "); // and subtract 32 from exponent
asm("bls TRealXModuloUnderflow "); // if borrow from exponent or exponent 0, underflow
asm("mov r0, #32 "); // r0 will hold 32-number of shifts to normalise
asm("cmp r2, #0x00010000 "); // normalise
asm("movcc r2, r2, lsl #16 ");
asm("subcc r0, r0, #16 ");
asm("cmp r2, #0x01000000 ");
asm("movcc r2, r2, lsl #8 ");
asm("subcc r0, r0, #8 ");
asm("cmp r2, #0x10000000 ");
asm("movcc r2, r2, lsl #4 ");
asm("subcc r0, r0, #4 ");
asm("cmp r2, #0x40000000 ");
asm("movcc r2, r2, lsl #2 ");
asm("subcc r0, r0, #2 ");
asm("cmp r2, #0x80000000 ");
asm("movcc r2, r2, lsl #1 "); // top bit of r2 is now set
asm("subcc r0, r0, #1 ");
asm("orr r2, r2, r1, lsr r0 "); // top bits of r1 into bottom bits of r2
asm("rsb r0, r0, #32 "); // r0=number of shifts to normalise
asm("mov r1, r1, lsl r0 "); // shift r1 left - mantissa now normalised
asm("subs r3, r3, r0 "); // subtract r0 from exponent
asm("bls TRealXModuloUnderflow "); // if borrow from exponent or exponent 0, underflow
asm("orr r3, r7, r3, lsl #16 "); // else r3=result exponent and sign
__JUMP(,lr);
// dividend=NaN or infinity
asm("TRealXModulo1: ");
asm("cmp r2, #0x80000000 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("cmn r6, #0x10000 "); // check 2nd operand for NaN/infinity
asm("bcc TRealXRealIndefinite "); // infinity%finite - return 'real indefinite'
asm("cmp r5, #0x80000000 "); // check if divisor=infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
asm("b TRealXRealIndefinite "); // else infinity%infinity - return 'real indefinite'
// divisor=NaN or infinity, dividend finite
asm("TRealXModulo2: ");
asm("cmp r5, #0x80000000 "); // check for infinity
asm("cmpeq r4, #0 ");
asm("bne TRealXBinOpNan "); // branch if NaN
__JUMP(,lr);
asm("TRealXModuloLp: ");
asm("mvn r12, #%a0" : : "i" ((TInt)~KErrTotalLossOfPrecision));
asm("mov r1, #0 ");
asm("mov r2, #0 ");
asm("and r3, r3, #1 ");
__JUMP(,lr);
asm("TRealXModuloUnderflow: ");
asm("mvn r12, #%a0" : : "i" ((TInt)~KErrUnderflow));
asm("mov r1, #0 ");
asm("mov r2, #0 ");
asm("and r3, r3, #1 ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt TRealX::Add(TRealX& /*aResult*/,const TRealX& /*aVal*/) const
/**
Adds an extended precision value to this extended precision number.
@param aResult On return, a reference to an extended precision object
containing the result of the operation.
@param aVal The extended precision value to be added.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
// r0=this, r1=&aResult, r2=&aVal
asm("stmfd sp!, {r1,r4-r8,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXAdd ");
asm("ldmfd sp!, {lr} "); // lr=&aResult
asm("stmia lr, {r1,r2,r3} ");
asm("mov r0, r12 "); // return value into r0
__POPRET("r4-r8,");
}
__NAKED__ EXPORT_C TInt TRealX::Sub(TRealX& /*aResult*/,const TRealX& /*aVal*/) const
/**
Subtracts an extended precision value from this extended precision number.
@param aResult On return, a reference to an extended precision object
containing the result of the operation.
@param aVal The extended precision value to be subtracted.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
// r0=this, r1=&aResult, r2=&aVal
asm("stmfd sp!, {r1,r4-r8,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {lr} "); // lr=&aResult
asm("stmia lr, {r1,r2,r3} ");
asm("mov r0, r12 "); // return value into r0
__POPRET("r4-r8,");
}
__NAKED__ EXPORT_C TInt TRealX::Mult(TRealX& /*aResult*/,const TRealX& /*aVal*/) const
/**
Multiplies this extended precision number by an extended precision value.
@param aResult On return, a reference to an extended precision object
containing the result of the operation.
@param aVal The extended precision value to be used as the multiplier.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow.
*/
{
// r0=this, r1=&aResult, r2=&aVal
asm("stmfd sp!, {r1,r4-r7,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXMultiply ");
asm("ldmfd sp!, {lr} "); // lr=&aResult
asm("stmia lr, {r1,r2,r3} ");
asm("mov r0, r12 "); // return value into r0
__POPRET("r4-r7,");
}
__NAKED__ EXPORT_C TInt TRealX::Div(TRealX& /*aResult*/,const TRealX& /*aVal*/) const
/**
Divides this extended precision number by an extended precision value.
@param aResult On return, a reference to an extended precision object
containing the result of the operation.
@param aVal The extended precision value to be used as the divisor.
@return KErrNone, if the operation is successful;
KErrOverflow, if the operation results in overflow;
KErrUnderflow, if the operation results in underflow;
KErrDivideByZero, if the divisor is zero.
*/
{
// r0=this, r1=&aResult, r2=&aVal
asm("stmfd sp!, {r1,r4-r9,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXDivide ");
asm("ldmfd sp!, {lr} "); // lr=&aResult
asm("stmia lr, {r1,r2,r3} ");
asm("mov r0, r12 "); // return value into r0
__POPRET("r4-r9,");
}
__NAKED__ EXPORT_C TInt TRealX::Mod(TRealX& /*aResult*/,const TRealX& /*aVal*/) const
/**
Modulo-divides this extended precision number by an extended precision value.
@param aResult On return, a reference to an extended precision object
containing the result of the operation.
@param aVal The extended precision value to be used as the divisor.
@return KErrNone, if the operation is successful;
KErrTotalLossOfPrecision, if precision is lost;
KErrUnderflow, if the operation results in underflow.
*/
{
// r0=this, r1=&aResult, r2=&aVal
asm("stmfd sp!, {r1,r4-r7,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXModulo ");
asm("ldmfd sp!, {lr} "); // lr=&aResult
asm("stmia lr, {r1,r2,r3} ");
asm("mov r0, r12 "); // return value into r0
__POPRET("r4-r7,");
}
extern void PanicOverUnderflowDividebyZero(const TInt aErr);
__NAKED__ EXPORT_C const TRealX& TRealX::operator+=(const TRealX& /*aVal*/)
/**
Adds an extended precision value to this extended precision number.
@param aVal The extended precision value to be added.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXAdd ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C const TRealX& TRealX::operator-=(const TRealX& /*aVal*/)
/**
Subtracts an extended precision value from this extended precision number.
@param aVal The extended precision value to be subtracted.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C const TRealX& TRealX::operator*=(const TRealX& /*aVal*/)
/**
Multiplies this extended precision number by an extended precision value.
@param aVal The extended precision value to be subtracted.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXMultiply ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C const TRealX& TRealX::operator/=(const TRealX& /*aVal*/)
/**
Divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
@panic MATHX KErrDivideByZero if the divisor is zero.
*/
{
asm("stmfd sp!, {r0,r4-r9,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXDivide ");
asm("ldmfd sp!, {r0,r4-r9,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C const TRealX& TRealX::operator%=(const TRealX& /*aVal*/)
/**
Modulo-divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return A reference to this object.
@panic MATHX KErrTotalLossOfPrecision panic if precision is lost.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r1, {r4,r5,r6} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("bl TRealXModulo ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
asm("cmpne r12, #%a0" : : "i" ((TInt)KErrTotalLossOfPrecision));
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX& TRealX::operator++()
/**
Increments this extended precision number by one,
and then returns a reference to it.
This is also referred to as a prefix operator.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// pre-increment
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("add r4, pc, #__TRealXOne-.-8 ");
asm("ldmia r4, {r4,r5,r6} "); // r4,r5,r6=1.0
asm("bl TRealXAdd ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
asm("__TRealXOne: ");
asm(".word 0x00000000 ");
asm(".word 0x80000000 ");
asm(".word 0x7FFF0000 ");
}
__NAKED__ EXPORT_C TRealX TRealX::operator++(TInt)
/**
Returns this extended precision number before incrementing it by one.
This is also referred to as a postfix operator.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// post-increment
// r0=address of return value, r1=this
asm("stmfd sp!, {r0,r1,r4-r8,lr} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("stmia r0, {r1,r2,r3} "); // store old value
asm("add r4, pc, #__TRealXOne-.-8 ");
asm("ldmia r4, {r4,r5,r6} "); // r4,r5,r6=1.0
asm("bl TRealXAdd ");
asm("ldmfd sp!, {r0,lr} "); // restore r0, lr=this
asm("stmia lr, {r1,r2,r3} "); // store incremented value
asm("ldmfd sp!, {r4-r8,lr} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX& TRealX::operator--()
/**
Decrements this extended precision number by one,
and then returns a reference to it.
This is also referred to as a prefix operator.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// pre-decrement
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r0, {r1,r2,r3} ");
asm("add r4, pc, #__TRealXOne-.-8 ");
asm("ldmia r4, {r4,r5,r6} "); // r4,r5,r6=1.0
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator--(TInt)
/**
Returns this extended precision number before decrementing it by one.
This is also referred to as a postfix operator.
@return A reference to this object.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// post-decrement
// r0=address of return value, r1=this
asm("stmfd sp!, {r0,r1,r4-r8,lr} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("stmia r0, {r1,r2,r3} "); // store old value
asm("add r4, pc, #__TRealXOne-.-8 ");
asm("ldmia r4, {r4,r5,r6} "); // r4,r5,r6=1.0
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {r0,lr} "); // restore r0, lr=this
asm("stmia lr, {r1,r2,r3} "); // store decremented value
asm("ldmfd sp!, {r4-r8,lr} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator+(const TRealX& /*aVal*/) const
/**
Adds an extended precision value to this extended precision number.
@param aVal The extended precision value to be added.
@return An extended precision object containing the result.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// r0=address of return value, r1=this, r2=&aVal
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("bl TRealXAdd ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator-(const TRealX& /*aVal*/) const
/**
Subtracts an extended precision value from this extended precision number.
@param aVal The extended precision value to be subtracted.
@return An extended precision object containing the result.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// r0=address of return value, r1=this, r2=&aVal
asm("stmfd sp!, {r0,r4-r8,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("bl TRealXSubtract ");
asm("ldmfd sp!, {r0,r4-r8,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator*(const TRealX& /*aVal*/) const
/**
Multiplies this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the multiplier.
@return An extended precision object containing the result.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// r0=address of return value, r1=this, r2=&aVal
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("bl TRealXMultiply ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator/(const TRealX& /*aVal*/) const
/**
Divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return An extended precision object containing the result.
@panic MATHX KErrOverflow if the operation results in overflow.
@panic MATHX KErrUnderflow if the operation results in underflow.
@panic MATHX KErrDivideByZero if the divisor is zero.
*/
{
// r0=address of return value, r1=this, r2=&aVal
asm("stmfd sp!, {r0,r4-r9,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("bl TRealXDivide ");
asm("ldmfd sp!, {r0,r4-r9,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
__NAKED__ EXPORT_C TRealX TRealX::operator%(const TRealX& /*aVal*/) const
/**
Modulo-divides this extended precision number by an extended precision value.
@param aVal The extended precision value to be used as the divisor.
@return An extended precision object containing the result.
@panic MATHX KErrTotalLossOfPrecision if precision is lost.
@panic MATHX KErrUnderflow if the operation results in underflow.
*/
{
// r0=address of return value, r1=this, r2=&aVal
asm("stmfd sp!, {r0,r4-r7,lr} ");
asm("ldmia r2, {r4,r5,r6} ");
asm("ldmia r1, {r1,r2,r3} ");
asm("bl TRealXModulo ");
asm("ldmfd sp!, {r0,r4-r7,lr} ");
asm("stmia r0, {r1,r2,r3} ");
asm("cmp r12, #0 "); // check the error code
asm("cmpne r12, #%a0" : : "i" ((TInt)KErrTotalLossOfPrecision));
__JUMP(eq,lr);
asm("mov r0, r12 ");
asm("b " CSM_Z30PanicOverUnderflowDividebyZeroi); // else panic
}
#ifdef __REALS_MACHINE_CODED__
__NAKED__ EXPORT_C TInt Math::Sqrt( TReal &/*aDest*/, const TReal &/*aSrc*/ )
/**
Calculates the square root of a number.
@param aDest A reference containing the result.
@param aSrc The number whose square-root is required.
@return KErrNone if successful, otherwise another of
the system-wide error codes.
*/
{
// r0=address of aDest, r1=address of aSrc
#ifdef __USE_VFP_MATH
VFP_FLDD(CC_AL,0,1,0);
VFP_FSQRTD(,0,0);
VFP_FMRRD(CC_AL,3,2,0);
asm("bic r1, r2, #0x80000000 "); // remove sign bit
asm("cmn r1, #0x00100000 "); // check if exp=7FF
asm("movpl r1, #0 "); // if not return KErrNone
asm("bpl donesqrt ");
asm("movs r1, r1, lsl #12 "); // if exp=7FF, check mantissa
asm("cmpeq r3, #0 ");
asm("moveq r1, #-9 "); // if exp=7FF, mant=0, return KErrOverflow
asm("mvnne r2, #0x80000000 "); // else set NaN
asm("mvnne r3, #0 ");
asm("movne r1, #-6 "); // and return KErrArgument
asm("donesqrt: ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r2,r3} "); // store the result
#else
asm("str r2, [r0, #4] ");
asm("str r3, [r0, #0] ");
#endif
asm("mov r0, r1 ");
__JUMP(,lr);
#else // __USE_VFP_MATH
asm("stmfd sp!, {r4-r10,lr} ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r1, {r3,r4} "); // low mant into r4, sign:exp:high mant into r3
#else
asm("ldr r3, [r1, #4] ");
asm("ldr r4, [r1, #0] ");
#endif
asm("bic r5, r3, #0xFF000000 ");
asm("bic r5, r5, #0x00F00000 "); // high word of mantissa into r5
asm("mov r2, r3, lsr #20 ");
asm("bics r2, r2, #0x800 "); // exponent now in r2
asm("beq fastsqrt1 "); // branch if exponent zero (zero or denormal)
asm("mov r6, #0xFF ");
asm("orr r6, r6, #0x700 ");
asm("cmp r2, r6 "); // check for infinity or NaN
asm("beq fastsqrt2 "); // branch if infinity or NaN
asm("movs r3, r3 "); // test sign
asm("bmi fastsqrtn "); // branch if negative
asm("sub r2, r2, #0xFF "); // unbias the exponent
asm("sub r2, r2, #0x300 "); //
asm("fastsqrtd1: ");
asm("mov r1, #0x40000000 "); // value for comparison
asm("mov r3, #27 "); // loop counter (number of bits/2)
asm("movs r2, r2, asr #1 "); // divide exponent by 2, LSB into CF
asm("movcs r7, r5, lsl #11 "); // mantissa into r6,r7 with MSB in MSB of r7
asm("orrcs r7, r7, r4, lsr #21 ");
asm("movcs r6, r4, lsl #11 ");
asm("movcs r4, #0 "); // r4, r5 will hold result mantissa
asm("orrcs r7, r7, #0x80000000 "); // if exponent odd, restore MSB of mantissa
asm("movcc r7, r5, lsl #12 "); // mantissa into r6,r7 with MSB in MSB of r7
asm("orrcc r7, r7, r4, lsr #20 "); // if exponent even, shift mantissa left an extra
asm("movcc r6, r4, lsl #12 "); // place, lose top bit, and
asm("movcc r4, #1 "); // set MSB of result, and
asm("mov r5, #0 "); // r4, r5 will hold result mantissa
asm("mov r8, #0 "); // r8, r9 will be comparison accumulator
asm("mov r9, #0 ");
asm("bcc fastsqrt4 "); // if exponent even, calculate one less bit
// as result MSB already known
// Main mantissa square-root loop
asm("fastsqrt3: "); // START OF MAIN LOOP
asm("subs r10, r7, r1 "); // subtract result:01 from acc:mant
asm("sbcs r12, r8, r4 "); // result into r14:r12:r10
asm("sbcs r14, r9, r5 ");
asm("movcs r7, r10 "); // if no borrow replace accumulator with result
asm("movcs r8, r12 ");
asm("movcs r9, r14 ");
asm("adcs r4, r4, r4 "); // shift result left one, putting in next bit
asm("adcs r5, r5, r5 ");
asm("mov r9, r9, lsl #2 "); // shift acc:mant left by 2 bits
asm("orr r9, r9, r8, lsr #30 ");
asm("mov r8, r8, lsl #2 ");
asm("orr r8, r8, r7, lsr #30 ");
asm("mov r7, r7, lsl #2 ");
asm("orr r7, r7, r6, lsr #30 ");
asm("mov r6, r6, lsl #2 ");
asm("fastsqrt4: "); // Come in here if we need to do one less iteration
asm("subs r10, r7, r1 "); // subtract result:01 from acc:mant
asm("sbcs r12, r8, r4 "); // result into r14:r12:r10
asm("sbcs r14, r9, r5 ");
asm("movcs r7, r10 "); // if no borrow replace accumulator with result
asm("movcs r8, r12 ");
asm("movcs r9, r14 ");
asm("adcs r4, r4, r4 "); // shift result left one, putting in next bit
asm("adcs r5, r5, r5 ");
asm("mov r9, r9, lsl #2 "); // shift acc:mant left by 2 bits
asm("orr r9, r9, r8, lsr #30 ");
asm("mov r8, r8, lsl #2 ");
asm("orr r8, r8, r7, lsr #30 ");
asm("mov r7, r7, lsl #2 ");
asm("orr r7, r7, r6, lsr #30 ");
asm("mov r6, r6, lsl #2 ");
asm("subs r3, r3, #1 "); // decrement loop counter
asm("bne fastsqrt3 "); // do necessary number of iterations
asm("movs r4, r4, lsr #1 "); // shift result mantissa right 1 place
asm("orr r4, r4, r5, lsl #31 "); // LSB (=rounding bit) into carry
asm("mov r5, r5, lsr #1 ");
asm("adcs r4, r4, #0 "); // round the mantissa to 53 bits
asm("adcs r5, r5, #0 ");
asm("cmp r5, #0x00200000 "); // check for mantissa overflow
asm("addeq r2, r2, #1 "); // if so, increment exponent - can never overflow
asm("bic r5, r5, #0x00300000 "); // remove top bit of mantissa - it is implicit
asm("add r2, r2, #0xFF "); // re-bias the exponent
asm("add r3, r2, #0x300 "); // and move into r3
asm("orr r3, r5, r3, lsl #20 "); // r3 now contains exponent + top of mantissa
asm("fastsqrt_ok: ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r3,r4} "); // store the result
#else
asm("str r3, [r0, #4] ");
asm("str r4, [r0, #0] ");
#endif
asm("mov r0, #0 "); // error code KErrNone
__POPRET("r4-r10,");
asm("fastsqrt1: ");
asm("orrs r6, r5, r4 "); // exponent zero - test mantissa
asm("beq fastsqrt_ok "); // if zero, return 0
asm("movs r3, r3 "); // denormal - test sign
asm("bmi fastsqrtn "); // branch out if negative
asm("sub r2, r2, #0xFE "); // unbias the exponent
asm("sub r2, r2, #0x300 "); //
asm("fastsqrtd: ");
asm("adds r4, r4, r4 "); // shift mantissa left
asm("adcs r5, r5, r5 ");
asm("sub r2, r2, #1 "); // and decrement exponent
asm("tst r5, #0x00100000 "); // test if normalised
asm("beq fastsqrtd "); // loop until normalised
asm("b fastsqrtd1 "); // now treat as a normalised number
asm("fastsqrt2: "); // get here if infinity or NaN
asm("orrs r6, r5, r4 "); // if mantissa zero, infinity
asm("bne fastsqrtnan "); // branch if not - must be NaN
asm("movs r3, r3 "); // test sign of infinity
asm("bmi fastsqrtn "); // branch if -ve
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r3,r4} "); // store the result
#else
asm("str r3, [r0, #4] ");
asm("str r4, [r0, #0] ");
#endif
asm("mov r0, #-9 "); // return KErrOverflow
asm("b fastsqrt_end ");
asm("fastsqrtn: "); // get here if negative or QNaN operand
asm("mov r3, #0xFF000000 "); // generate "real indefinite" QNaN
asm("orr r3, r3, #0x00F80000 "); // sign=1, exp=7FF, mantissa = 1000...0
asm("mov r4, #0 ");
asm("fastsqrtxa: ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r3,r4} "); // store the result
#else
asm("str r3, [r0, #4] ");
asm("str r4, [r0, #0] ");
#endif
asm("mov r0, #-6 "); // return KErrArgument
asm("fastsqrt_end: ");
__POPRET("r4-r10,");
asm("fastsqrtnan: "); // operand is a NaN
asm("tst r5, #0x00080000 "); // test MSB of mantissa
asm("bne fastsqrtn "); // if set it is a QNaN - so return "real indefinite"
asm("bic r3, r3, #0x00080000 "); // else convert SNaN to QNaN
asm("b fastsqrtxa "); // and return KErrArgument
#endif // __USE_VFP_MATH
}
__NAKED__ EXPORT_C TReal Math::Poly(TReal /*aX*/,const SPoly* /*aPoly*/) __SOFTFP
/**
Evaluates the polynomial:
{a[n]X^n + a[n-1]X^(n-1) + ... + a[2]X^2 + a[1]X^1 + a[0]}.
@param aX The value of the x-variable
@param aPoly A pointer to the structure containing the set of coefficients
in the order: a[0], a[1], ..., a[n-1], a[n].
@return The result of the evaluation.
*/
//
// Evaluate a power series in x for a P_POLY coefficient table.
// Changed to use TRealX throughout the calculation
//
{
// On entry r0,r1=aX, r2=aPoly
asm("stmfd sp!, {r4-r11,lr} ");
asm("mov r11, r2 ");
asm("ldr r10, [r11], #4 "); // r10=number of coefficients, r11=first coeff addr
asm("add r11, r11, r10, lsl #3 "); // r11=address of last coefficient+8
asm("mov r2, r1 "); // aX into r1,r2
asm("mov r1, r0 ");
asm("bl ConvertTReal64ToTRealX "); // convert to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("ldmdb r11!, {r1,r2} "); // last coefficient into r1,r2
asm("bl ConvertTReal64ToTRealX "); // convert to TRealX in r1,r2,r3
asm("subs r10, r10, #1 ");
asm("beq polynomial0 "); // if no more coefficients, exit
asm("polynomial1: ");
asm("stmfd sp!, {r4,r5,r6} "); // save value of aX
asm("bl TRealXMultiply "); // r *= aX
asm("mov r4, r1 "); // move result into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("ldmdb r11!, {r1,r2} "); // next coefficient into r1,r2
asm("bl ConvertTReal64ToTRealX "); // convert to TRealX in r1,r2,r3
asm("bl TRealXAdd "); // r += *--pR
asm("ldmfd sp!, {r4,r5,r6} "); // aX back into r4,r5,r6
asm("subs r10, r10, #1 "); // iterate until all coefficients processed
asm("bne polynomial1 ");
asm("polynomial0: "); // result now in r1,r2,r3
asm("bl ConvertTRealXToTReal64 "); // convert back to TReal64
__POPRET("r4-r11,");
}
__NAKED__ EXPORT_C void Math::PolyX(TRealX& /*aY*/,const TRealX& /*aX*/,TInt /*aDeg*/,const TRealX* /*aCoef*/)
/**
Evaluates the polynomial:
{a[n]X^n + a[n-1]X^(n-1) + ... + a[2]X^2 + a[1]X^1 + a[0]}.
@param aY A reference containing the result.
@param aX The value of the x-variable.
@param aDeg The degree of the polynomial (the highest power of x
which is present).
@param aCoef A pointer to a contiguous set of TRealX values containing
the coefficients.
They must be in the order: a[0], a[1], ..., a[n-1], a[n].
*/
//
// Evaluate a polynomial with TRealX argument, coefficients and result
//
{
// On entry r0=&aY, r1=&aX, r2=aDeg, r3=aCoef
asm("stmfd sp!, {r0,r4-r11,lr} ");
asm("add r11, r3, r2, lsl #3 "); // r11=address of last coefficient
asm("add r11, r11, r2, lsl #2 ");
asm("mov r9, r1 "); // r9=address of argument
asm("movs r10, r2 "); // r10=number of coefficients-1
asm("ldmia r11, {r1,r2,r3} "); // last coefficient into r1,r2,r3
asm("beq polyx0 "); // if no more coefficients, exit
asm("polyx1: ");
asm("ldmia r9, {r4,r5,r6} "); // aX into r4,r5,r6
asm("bl TRealXMultiply "); // result *= aX
asm("ldmdb r11!, {r4,r5,r6} "); // next coefficient into r4,r5,r6
asm("bl TRealXAdd "); // result += next coeff
asm("subs r10, r10, #1 "); // iterate until all coefficients processed
asm("bne polyx1 ");
asm("polyx0: "); // result now in r1,r2,r3
asm("ldmfd sp!, {r0,r4-r11,lr} "); // restore registers, including destination address in r0
asm("stmia r0, {r1,r2,r3} "); // store result
__JUMP(,lr);
}
#ifndef __USE_VFP_MATH
__NAKED__ EXPORT_C TInt Math::Int(TReal& /*aTrg*/, const TReal& /*aSrc*/)
/**
Calculates the integer part of a number.
The integer part is that before a decimal point.
Truncation is toward zero, so that
int(2.4)=2, int(2)=2, int(-1)=-1, int(-1.4)=-1, int(-1.999)=-1.
@param aTrg A reference containing the result.
@param aSrc The number whose integer part is required.
@return KErrNone if successful, otherwise another of
the system-wide error codes.
*/
//
// Write the integer part of aSrc to the TReal at aTrg
// Negative numbers are rounded towards zero.
//
{
// r0=&aTrg, r1=&aSrc, return value in r0
asm("stmfd sp!, {lr} ");
asm("mov r12, r0 "); // r12=&aTrg
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r1, {r0,r1} "); // input value into r0,r1
#else
asm("ldr r0, [r1, #4] ");
asm("ldr r1, [r1, #0] ");
#endif
asm("bl TReal64Int ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r12, {r0,r1} "); // store result
#else
asm("str r0, [r12, #4] ");
asm("str r1, [r12, #0] ");
#endif
asm("bic r0, r0, #0x80000000 "); // remove sign bit
asm("cmn r0, #0x00100000 "); // check for NaN or infinity
asm("movpl r0, #0 "); // if neither, return KErrNone
asm("bpl math_int_0 ");
asm("movs r0, r0, lsl #12 "); // check for infinity
asm("cmpeq r1, #0 ");
asm("mvneq r0, #8 "); // if infinity return KErrOverflow
asm("mvnne r0, #5 "); // else return KErrArgument
asm("math_int_0: ");
__POPRET("");
// Take integer part of TReal64 in r0,r1
// Infinity and NaNs are unaffected
// r0-r3 modified
asm("TReal64Int: ");
asm("mov r2, r0, lsr #20 ");
asm("bic r2, r2, #0x800 "); // r2=exponent
asm("mov r3, #0x300 ");
asm("orr r3, r3, #0xFF "); // r3=0x3FF
asm("subs r2, r2, r3 "); // r2=exponent-3FF=number of integer bits-1
asm("ble TReal64Int1 "); // branch if <=1 integer bits
asm("cmp r2, #52 ");
__JUMP(ge,lr);
asm("cmp r2, #20 ");
asm("bgt TReal64Int2 "); // jump if >21 integer bits (r0 will be unaffected)
asm("rsb r2, r2, #20 "); // r2=number of bits to clear at bottom end of r0
asm("mov r0, r0, lsr r2 "); // clear them
asm("mov r0, r0, lsl r2 ");
asm("mov r1, #0 "); // clear r1
__JUMP(,lr);
asm("TReal64Int2: ");
asm("rsb r2, r2, #52 "); // r2=number of bits to clear at bottom end of r1
asm("mov r1, r1, lsr r2 "); // clear them
asm("mov r1, r1, lsl r2 ");
__JUMP(,lr);
asm("TReal64Int1: "); // result is either 0 or 1
asm("mov r1, #0 "); // lower mantissa bits of result will be zero
asm("moveq r0, r0, lsr #20 "); // if result is 1, clear mantissa but leave exponent
asm("moveq r0, r0, lsl #20 ");
asm("andlt r0, r0, #0x80000000 "); // if result is 0, clear mantissa and exponent
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt Math::Int(TInt16& /*aTrg*/, const TReal& /*aSrc*/)
/**
Calculates the integer part of a number.
The integer part is that before a decimal point.
Truncation is toward zero, so that:
int(2.4)=2, int(2)=2, int(-1)=-1, int(-1.4)=-1, int(-1.999)=-1.
This function is suitable when the result is known to be small enough
for a 16-bit signed integer.
@param aTrg A reference containing the result.
@param aSrc The number whose integer part is required.
@return KErrNone if successful, otherwise another of
the system-wide error codes.
*/
//
// If the integer part of aSrc is in the range -32768 to +32767
// inclusive, write the integer part to the TInt16 at aTrg
// Negative numbers are rounded towards zero.
// If an overflow or underflow occurs, aTrg is set to the max/min value
//
{
// r0=&aTrg, r1=&aSrc
asm("stmfd sp!, {lr} ");
asm("mov r3, r0 "); // r3=&aTrg
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r1, {r0,r1} "); // input value into r0,r1
#else
asm("ldr r0, [r1, #4] ");
asm("ldr r1, [r1, #0] ");
#endif
asm("bl TReal64GetTInt "); // do the conversion
asm("cmp r0, #0x8000 "); // limit answer to TInt16 range
asm("movge r0, #0x7F00 ");
asm("orrge r0, r0, #0xFF ");
asm("mvnge r12, #8 "); // set error code if limiting occurred
asm("cmn r0, #0x8000 ");
asm("movlt r0, #0x8000 ");
asm("mvnlt r12, #9 "); // set error code if limiting occurred
asm("mov r1, r0, lsr #8 "); // top byte of answer into r1
asm("strb r0, [r3] "); // store result in aTrg
asm("strb r1, [r3, #1] ");
asm("mov r0, r12 "); // return error code in r0
__POPRET("");
}
__NAKED__ EXPORT_C TInt Math::Int(TInt32& /*aTrg*/, const TReal& /*aSrc*/)
/**
Calculates the integer part of a number.
The integer part is that before a decimal point.
Truncation is toward zero, so that
int(2.4)=2, int(2)=2, int(-1)=-1, int(-1.4)=-1, int(-1.999)=-1.
This function is suitable when the result is known to be small enough
for a 32-bit signed integer.
@param aTrg A reference containing the result.
@param aSrc The number whose integer part is required.
@return KErrNone if successful, otherwise another of
the system-wide error codes.
*/
//
// If the integer part of the float is in the range -2147483648 to +2147483647
// inclusive, write the integer part to the TInt32 at aTrg
// Negative numbers are rounded towards zero.
// If an overflow or underflow occurs, aTrg is set to the max/min value
//
{
// r0=&aTrg, r1=&aSrc
asm("stmfd sp!, {lr} ");
asm("mov r3, r0 "); // r3=&aTrg
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r1, {r0,r1} "); // input value into r0,r1
#else
asm("ldr r0, [r1, #4] ");
asm("ldr r1, [r1, #0] ");
#endif
asm("bl TReal64GetTInt "); // do the conversion
asm("str r0, [r3] "); // store result in aTrg
asm("mov r0, r12 "); // return error code in r0
__POPRET("");
// Convert double in r0,r1 to int in r0
// Return error code in r12
// Registers r0,r1,r2,r12 modified
asm("TReal64GetTInt: ");
asm("mov r2, r0, lsr #20 ");
asm("bic r2, r2, #0x800 "); // r1=exponent
asm("add r12, r2, #1 ");
asm("cmp r12, #0x800 "); // check for NaN
asm("bne TReal64GetTInt1 ");
asm("movs r12, r0, lsl #12 "); // exponent=FF, check mantissa
asm("cmpeq r1, #0 ");
asm("movne r0, #0 "); // if non-zero, input is a NaN so return 0
asm("mvnne r12, #5 "); // and return KErrArgument
__JUMP(ne,lr);
asm("TReal64GetTInt1: ");
asm("mov r12, #0x400 ");
asm("orr r12, r12, #0x1E "); // r12=0x41E (exponent of 2^31)
asm("subs r2, r12, r2 "); // r2=number of shifts to produce integer
asm("mov r12, #0 "); // set return code to KErrNone
asm("ble TReal64GetTInt2 "); // if <=0, saturate result
asm("cmp r2, #31 "); // check if more than 31 shifts needed
asm("movhi r0, #0 "); // if so, underflow result to 0
__JUMP(hi,lr);
asm("cmp r0, #0 "); // check sign bit
asm("orr r0, r0, #0x00100000 "); // set implicit integer bit
asm("mov r0, r0, lsl #11 "); // shift mantissa up so MSB is in MSB of r0
asm("orr r0, r0, r1, lsr #21 "); // put in bits from r1
asm("mov r0, r0, lsr r2 "); // r0=absolute integer
asm("rsbmi r0, r0, #0 "); // if negative, negate
__JUMP(,lr);
asm("TReal64GetTInt2: ");
asm("blt TReal64GetTInt3 "); // if exponent>0x41E, definitely an overflow
asm("cmp r0, #0 "); // check sign bit
asm("bpl TReal64GetTInt3 "); // if positive, definitely an overflow
asm("orr r0, r0, #0x00100000 "); // set implicit integer bit
asm("mov r0, r0, lsl #11 "); // shift mantissa up so MSB is in MSB of r0
asm("orr r0, r0, r1, lsr #21 "); // put in bits from r1
asm("cmp r0, #0x80000000 "); // check if value is = -2^31
__JUMP(eq,lr);
asm("TReal64GetTInt3: ");
asm("cmp r0, #0 "); // check sign
asm("mov r0, #0x80000000 ");
asm("subpl r0, r0, #1 "); // if -ve return 80000000, if +ve return 7FFFFFFF
asm("mvnpl r12, #8 "); // if +ve return KErrOverflow
asm("mvnmi r12, #9 "); // if -ve return KErrUnderflow
__JUMP(,lr);
}
#endif // __USE_VFP_MATH
__NAKED__ EXPORT_C TBool Math::IsZero(const TReal& /*aVal*/)
/**
Determines whether a value is zero.
@param aVal A reference to the value to be checked.
@return True, if aVal is zero; false, otherwise.
*/
{
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r0, {r1,r2} "); // input value into r0,r1
#else
asm("ldr r2, [r0, #0] ");
asm("ldr r1, [r0, #4] ");
#endif
asm("TReal64IsZero: ");
asm("mov r0, #0 "); // default return value is 0
asm("bics r1, r1, #0x80000000 "); // remove sign bit
asm("cmpeq r2, #0 "); // and check both exponent and mantissa are zero
asm("moveq r0, #1 "); // return 1 if zero
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool Math::IsNaN(const TReal& /*aVal*/)
/**
Determines whether a value is not a number.
@param aVal A reference to the value to be checked.
@return True, if aVal is not a number; false, otherwise.
*/
{
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r0, {r1,r2} "); // input value into r0,r1
#else
asm("ldr r2, [r0, #0] ");
asm("ldr r1, [r0, #4] ");
#endif
asm("TReal64IsNaN: ");
asm("mov r0, #0 "); // default return value is 0
asm("bic r1, r1, #0x80000000 "); // remove sign bit
asm("cmn r1, #0x00100000 "); // check if exponent=7FF
__JUMP(pl,lr);
asm("movs r1, r1, lsl #12 "); // exponent=7FF, check mantissa
asm("cmpeq r2, #0 ");
asm("movne r0, #1 "); // if mantissa nonzero, return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool Math::IsInfinite(const TReal& /*aVal*/)
/**
Determines whether a value is infinite.
@param aVal A reference to the value to be checked.
@return True, if aVal is infinite; false, otherwise.
*/
{
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r0, {r1,r2} "); // input value into r0,r1
#else
asm("ldr r2, [r0, #0] ");
asm("ldr r1, [r0, #4] ");
#endif
asm("TReal64IsInfinite: ");
asm("mov r0, #0 "); // default return value is 0
asm("mov r3, #0x00200000 "); // r3 == - (0x7ff00000 << 1)
asm("cmp r2, #0 ");
asm("cmneq r3, r1, lsl #1 "); // check exp=7FF && mant=0
asm("moveq r0, #1 "); // if so, return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C TBool Math::IsFinite(const TReal& /*aVal*/)
/**
Determines whether a value is finite.
In this context, a value is finite if it is a valid number and
is not infinite.
@param aVal A reference to the value to be checked.
@return True, if aVal is finite; false, otherwise.
*/
{
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldr r1, [r0, #0] "); // only need exponent - get it into r0
#else
asm("ldr r1, [r0, #4] "); // only need exponent - get it into r0
#endif
asm("TReal64IsFinite: ");
asm("mov r0, #0 "); // default return value is 0
asm("bic r1, r1, #0x80000000 "); // remove sign bit
asm("cmn r1, #0x00100000 "); // check if exponent=7FF
asm("movpl r0, #1 "); // else return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C void Math::SetZero(TReal& /*aVal*/, TInt /*aSign*/)
//
// Constructs zeros, assuming default sign is positive
//
{
asm("cmp r1, #0 "); // test aSign
asm("movne r1, #0x80000000 "); // if nonzero, set sign bit
asm("mov r2, #0 "); // mantissa=0
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r1,r2} ");
#else
asm("str r2, [r0, #0] ");
asm("str r1, [r0, #4] ");
#endif
__JUMP(,lr);
}
__NAKED__ EXPORT_C void Math::SetNaN(TReal& /*aVal*/)
//
// Constructs NaN (+ve sign for Java)
//
{
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("mvn r1, #0x80000000 "); // r1=7FFFFFFF
asm("mvn r2, #0 "); // r2=FFFFFFFF
#else
asm("mvn r2, #0x80000000 "); // r2=7FFFFFFF
asm("mvn r1, #0 "); // r1=FFFFFFFF
#endif
asm("stmia r0, {r1,r2} ");
__JUMP(,lr);
}
__NAKED__ EXPORT_C void Math::SetInfinite(TReal& /*aVal*/, TInt /*aSign*/)
//
// Constructs infinities
//
{
asm("cmp r1, #0 "); // test aSign
asm("movne r1, #0x80000000 "); // if nonzero, set sign bit
asm("orr r1, r1, #0x70000000 "); // set exponent to 7FF
asm("orr r1, r1, #0x0FF00000 ");
asm("mov r2, #0 "); // mantissa=0
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r1,r2} ");
#else
asm("str r2, [r0, #0] ");
asm("str r1, [r0, #4] ");
#endif
__JUMP(,lr);
}
#ifndef __USE_VFP_MATH
__NAKED__ EXPORT_C TInt Math::Frac(TReal& /*aTrg*/, const TReal& /*aSrc*/)
/**
Calculates the fractional part of a number.
The fractional part is that after a decimal point.
Truncation is toward zero, so that
Frac(2.4)=0.4, Frac(2)=0, Frac(-1)=0, Frac(-1.4)=0.4.
@param aTrg A reference containing the result.
@param aSrc The number whose fractional part is required.
@return KErrNone if successful, otherwise another of
the system-wide error codes.
*/
{
// on entry r0=aTrg, r1=&Src
// on exit r0=return code
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("ldmia r1, {r1,r2} "); // r1,r2=aSrc
#else
asm("ldr r2, [r1, #0] ");
asm("ldr r1, [r1, #4] ");
#endif
asm("and r3, r1, #0x80000000 ");
asm("str r3, [sp, #-4]! "); // save sign
asm("mov r3, r1, lsr #20 ");
asm("bic r3, r3, #0x800 "); // r3=exponent of aSrc
asm("mov r12, #0x300 ");
asm("orr r12, r12, #0xFE "); // r12=0x3FE
asm("subs r3, r3, r12 "); // r3=exponent of aSrc-0x3FE=number of integer bits
asm("ble MathFrac0 "); // if <=0, return aSrc unaltered
asm("cmp r3, #53 ");
asm("bge MathFrac1 "); // if >=53 integer bits, there is no fractional part
asm("mov r1, r1, lsl #11 "); // left-justify mantissa in r1,r2
asm("orr r1, r1, r2, lsr #21 ");
asm("mov r2, r2, lsl #11 ");
asm("cmp r3, #32 "); // check for >=32 integer bits
asm("bge MathFrac2 ");
asm("rsb r12, r3, #32 ");
asm("mov r1, r1, lsl r3 "); // shift mantissa left by number of integer bits
asm("orrs r1, r1, r2, lsr r12 ");
asm("mov r2, r2, lsl r3 ");
asm("mov r3, #0x300 "); // r3 holds exponent = 0x3FE initially
asm("orr r3, r3, #0xFE ");
asm("beq MathFrac3 "); // branch if >=32 shifts to normalise
#ifdef __CPU_ARM_HAS_CLZ
CLZ(12,1);
asm("mov r1, r1, lsl r12 ");
asm("rsb r12, r12, #32 ");
asm("orr r1, r1, r2, lsr r12 ");
asm("rsb r12, r12, #32 ");
#else
asm("mov r12, #32 "); // else r12=32-number of shifts needed
asm("cmp r1, #0x10000 "); // calculate shift count
asm("movcc r1, r1, lsl #16 ");
asm("subcc r12, r12, #16 ");
asm("cmp r1, #0x1000000 ");
asm("movcc r1, r1, lsl #8 ");
asm("subcc r12, r12, #8 ");
asm("cmp r1, #0x10000000 ");
asm("movcc r1, r1, lsl #4 ");
asm("subcc r12, r12, #4 ");
asm("cmp r1, #0x40000000 ");
asm("movcc r1, r1, lsl #2 ");
asm("subcc r12, r12, #2 ");
asm("cmp r1, #0x80000000 ");
asm("movcc r1, r1, lsl #1 ");
asm("subcc r12, r12, #1 ");
asm("orr r1, r1, r2, lsr r12 "); // normalise
asm("rsb r12, r12, #32 "); // r12=shift count
#endif
asm("mov r2, r2, lsl r12 ");
asm("sub r3, r3, r12 "); // exponent-=shift count
asm("b MathFrac4 "); // branch to assemble and store result
// come here if >=32 shifts to normalise
asm("MathFrac3: ");
asm("sub r3, r3, #32 "); // decrement exponent by 32
asm("movs r1, r2 "); // shift left by 32, set Z if result zero
asm("mov r2, #0 ");
asm("bne MathFrac6 "); // if result nonzero, normalise
asm("beq MathFrac5 "); // branch if result zero
// come here if >=32 integer bits
asm("MathFrac2: ");
asm("sub r3, r3, #32 ");
asm("movs r1, r2, lsl r3 "); // shift left by number of integer bits, set Z if result zero
asm("mov r2, #0 ");
asm("mov r3, #0x300 "); // r3 holds exponent = 0x3FE initially
asm("orr r3, r3, #0xFE ");
asm("beq MathFrac5 "); // branch if result zero
asm("MathFrac6: ");
asm("cmp r1, #0x10000 "); // else normalise
asm("movcc r1, r1, lsl #16 ");
asm("subcc r3, r3, #16 ");
asm("cmp r1, #0x1000000 ");
asm("movcc r1, r1, lsl #8 ");
asm("subcc r3, r3, #8 ");
asm("cmp r1, #0x10000000 ");
asm("movcc r1, r1, lsl #4 ");
asm("subcc r3, r3, #4 ");
asm("cmp r1, #0x40000000 ");
asm("movcc r1, r1, lsl #2 ");
asm("subcc r3, r3, #2 ");
asm("cmp r1, #0x80000000 ");
asm("movcc r1, r1, lsl #1 ");
asm("subcc r3, r3, #1 ");
// come here to assemble and store result
asm("MathFrac4: ");
asm("bic r1, r1, #0x80000000 "); // remove integer bit
asm("mov r2, r2, lsr #11 "); // shift mantissa right by 11
asm("orr r2, r2, r1, lsl #21 ");
asm("mov r1, r1, lsr #11 ");
asm("ldr r12, [sp] ");
asm("orr r1, r1, r3, lsl #20 "); // exponent into r1 bits 20-30
asm("orr r1, r1, r12 "); // sign bit into r1 bit 31
// come here to return source unaltered
asm("MathFrac0: ");
asm("add sp, sp, #4 ");
asm("MathFrac_ok: ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r1,r2} "); // store result
#else
asm("str r2, [r0, #0] ");
asm("str r1, [r0, #4] ");
#endif
asm("mov r0, #0 "); // return KErrNone
__JUMP(,lr);
// come here if infinity, NaN or >=53 integer bits
asm("MathFrac1: ");
asm("cmp r3, #0x400 "); // check for infinity/NaN
asm("bhi MathFrac7 "); // branch if so
// come here to return zero
asm("MathFrac5: ");
asm("ldr r1, [sp], #4 "); // r1 bit 31=sign, rest zero
asm("mov r2, #0 ");
asm("b MathFrac_ok ");
// come here if infinity/NaN
asm("MathFrac7: ");
asm("movs r12, r1, lsl #12 "); // check for infinity
asm("cmpeq r2, #0 ");
asm("bne MathFrac8 "); // branch if NaN
asm("ldr r1, [sp], #4 "); // r1 bit 31=sign, rest zero
asm("mov r2, #0 ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r1,r2} "); // store zero result
#else
asm("str r2, [r0, #0] ");
asm("str r1, [r0, #4] ");
#endif
asm("mvn r0, #8 "); // return KErrOverflow
__JUMP(,lr);
asm("MathFrac8: "); // NaN
asm("add sp, sp, #4 ");
#ifdef __DOUBLE_WORDS_SWAPPED__
asm("stmia r0, {r1,r2} "); // store NaN unchanged
#else
asm("str r2, [r0, #0] ");
asm("str r1, [r0, #4] ");
#endif
asm("mvn r0, #5 "); // return KErrArgument
__JUMP(,lr);
}
#endif // __USE_VFP_MATH
#endif
#ifdef __REALS_MACHINE_CODED__
#ifndef __ARMCC__
extern "C" {
extern "C" void __math_exception(TInt aErrType);
__NAKED__ EXPORT_C TReal32 __addsf3(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Add two floats
//
{
// a1 is in r0, a2 in r1 on entry; return with answer in r0
asm("stmfd sp!, {r4-r8,lr} ");
asm("bl ConvertTReal32ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r0 "); // a1 into r1
asm("bl ConvertTReal32ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXAdd "); // add a1+a2, result in r1,r2,r3
asm("bl TRealXGetTReal32 "); // convert result to TReal32 in r0, error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r8,");
asm("stmfd sp!, {r0} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r4-r8,");
}
__NAKED__ EXPORT_C TReal64 __adddf3(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Add two doubles
//
{
// a1 is in r0,r1 a2 in r2,r3 on entry; return with answer in r0,r1
asm("stmfd sp!, {r4-r8,lr} ");
asm("mov r7, r2 "); // save a2
asm("mov r8, r3 ");
asm("mov r2, r1 "); // a1 into r1,r2
asm("mov r1, r0 ");
asm("bl ConvertTReal64ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r7 "); // a2 into r1,r2
asm("mov r2, r8 ");
asm("bl ConvertTReal64ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("bl TRealXAdd "); // add a1+a2, result in r1,r2,r3
asm("bl TRealXGetTReal64 "); // convert result to TReal64 in r0,r1 error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r8,");
asm("stmfd sp!, {r0,r1} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r1,r4-r8,");
}
__NAKED__ EXPORT_C TReal32 __subsf3(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Subtract two floats
//
{
// a1 is in r0, a2 in r1 on entry; return with answer in r0
asm("stmfd sp!, {r4-r8,lr} ");
asm("bl ConvertTReal32ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r0 "); // a1 into r1
asm("bl ConvertTReal32ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXSubtract "); // subtract a1-a2, result in r1,r2,r3
asm("bl TRealXGetTReal32 "); // convert result to TReal32 in r0, error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r8,");
asm("stmfd sp!, {r0} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r4-r8,");
}
__NAKED__ EXPORT_C TReal64 __subdf3(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Subtract two doubles
//
{
// a1 is in r0,r1 a2 in r2,r3 on entry; return with answer in r0,r1
asm("stmfd sp!, {r4-r8,lr} ");
asm("mov r7, r0 "); // save a1
asm("mov r8, r1 ");
asm("mov r1, r2 "); // a2 into r1,r2
asm("mov r2, r3 ");
asm("bl ConvertTReal64ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r7 "); // a1 into r1,r2
asm("mov r2, r8 ");
asm("bl ConvertTReal64ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXSubtract "); // subtract a1-a2, result in r1,r2,r3
asm("bl TRealXGetTReal64 "); // convert result to TReal64 in r0,r1 error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r8,");
asm("stmfd sp!, {r0,r1} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r1,r4-r8,");
}
__NAKED__ EXPORT_C TInt __cmpsf3(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare two floats
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("mov r0, r0, lsl #28 ");
asm("msr cpsr_flg, r0 "); // N=unordered, Z=(a1>a2), C=(a1=a2), V=(a1<a2)
asm("mov r0, #0 ");
asm("mvnvs r0, #0 "); // if a1<a2 r0=-1
asm("moveq r0, #1 "); // if a1>a2 r0=+1
__POPRET("");
// Compare two TReal32s in r0, r1.
// Return 1 if r0<r1, 2 if r0=r1, 4 if r0>r1, 8 if unordered
// Registers r0,r1,r12 modified
asm("CompareTReal32: ");
asm("mov r12, r0, lsr #23 ");
asm("and r12, r12, #0xFF "); // r12=r0 exponent
asm("cmp r12, #0xFF "); // check if r0 is a NaN
asm("bne CompareTReal32a ");
asm("movs r12, r0, lsl #9 "); // exponent=FF, check mantissa
asm("movne r0, #8 "); // if not zero, r0 is a NaN so result is unordered
__JUMP(ne,lr);
asm("CompareTReal32a: ");
asm("mov r12, r1, lsr #23 ");
asm("and r12, r12, #0xFF "); // r12=r1 exponent
asm("cmp r12, #0xFF "); // check if r1 is a NaN
asm("bne CompareTReal32b ");
asm("movs r12, r1, lsl #9 "); // exponent=FF, check mantissa
asm("movne r0, #8 "); // if not zero, r1 is a NaN so result is unordered
__JUMP(ne,lr);
asm("CompareTReal32b: ");
asm("bics r12, r0, #0x80000000 "); // check if r0=0 (can be +0 or -0)
asm("moveq r0, #0 "); // if it is, make it +0
asm("bics r12, r1, #0x80000000 "); // check if r1=0 (can be +0 or -0)
asm("moveq r1, #0 "); // if it is, make it +0
asm("teq r0, r1 "); // test if signs different
asm("bmi CompareTReal32c "); // branch if different
asm("cmp r0, r1 "); // if same, check exponents + mantissas
asm("moveq r0, #2 "); // if equal, return 2
__JUMP(eq,lr);
asm("movhi r0, #4 "); // if r0>r1, r0=4
asm("movcc r0, #1 "); // if r0<r1, r0=1
asm("cmp r1, #0 "); // check signs
asm("eormi r0, r0, #5 "); // if negative, switch 1 and 4
__JUMP(,lr);
asm("CompareTReal32c: "); // come here if signs different
asm("cmp r0, #0 "); // check sign of r0
asm("movpl r0, #4 "); // if r0 nonnegative, then r0 is greater so return 4
asm("movmi r0, #1 "); // if r0 negative, return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt __cmpdf3(TReal64 /*a1*/,TReal64 /*a2*/)
//
// Compare two doubles
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("mov r0, r0, lsl #28 ");
asm("msr cpsr_flg, r0 "); // N=unordered, Z=(a1>a2), C=(a1=a2), V=(a1<a2)
asm("mov r0, #0 ");
asm("mvnvs r0, #0 "); // if a1<a2 r0=-1
asm("moveq r0, #1 "); // if a1>a2 r0=+1
__POPRET("");
// Compare two TReal64s in r0,r1 and r2,r3.
// Return 1 if r0,r1<r2,r3
// Return 2 if r0,r1=r2,r3
// Return 4 if r0,r1>r2,r3
// Return 8 if unordered
// Registers r0,r1,r12 modified
asm("CompareTReal64: ");
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r12, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r12 ");
asm("mov r12, r2 ");
asm("mov r2, r3 ");
asm("mov r3, r12 ");
#endif
asm("mov r12, r0, lsr #20 ");
asm("bic r12, r12, #0x800 "); // r12=first operand exponent
asm("add r12, r12, #1 "); // add 1 to get usable compare value
asm("cmp r12, #0x800 "); // check if first operand is a NaN
asm("bne CompareTReal64a ");
asm("movs r12, r0, lsl #12 "); // exponent=7FF, check mantissa
asm("cmpeq r1, #0 ");
asm("movne r0, #8 "); // if not zero, 1st op is a NaN so result is unordered
__JUMP(ne,lr);
asm("CompareTReal64a: ");
asm("mov r12, r2, lsr #20 ");
asm("bic r12, r12, #0x800 "); // r12=second operand exponent
asm("add r12, r12, #1 "); // add 1 to get usable compare value
asm("cmp r12, #0x800 "); // check if second operand is a NaN
asm("bne CompareTReal64b ");
asm("movs r12, r2, lsl #12 "); // exponent=7FF, check mantissa
asm("cmpeq r3, #0 ");
asm("movne r0, #8 "); // if not zero, 2nd op is a NaN so result is unordered
__JUMP(ne,lr);
asm("CompareTReal64b: ");
asm("bics r12, r0, #0x80000000 "); // check if first operand is zero (can be +0 or -0)
asm("cmpeq r1, #0 ");
asm("moveq r0, #0 "); // if it is, make it +0
asm("bics r12, r2, #0x80000000 "); // check if second operand is zero (can be +0 or -0)
asm("cmpeq r3, #0 ");
asm("moveq r2, #0 "); // if it is, make it +0
asm("teq r0, r2 "); // test if signs different
asm("bmi CompareTReal64c "); // branch if different
asm("cmp r0, r2 "); // if same, check exponents + mantissas
asm("cmpeq r1, r3 ");
asm("moveq r0, #2 "); // if equal, return 2
__JUMP(eq,lr);
asm("movhi r0, #4 "); // if 1st operand > 2nd operand, r0=4
asm("movcc r0, #1 "); // if 1st operand < 2nd operand, r0=1
asm("cmp r2, #0 "); // check signs
asm("eormi r0, r0, #5 "); // if negative, switch 1 and 4
__JUMP(,lr);
asm("CompareTReal64c: "); // come here if signs different
asm("cmp r0, #0 "); // check sign of r0
asm("movpl r0, #4 "); // if first operand nonnegative, return 4
asm("movmi r0, #1 "); // if first operand negative, return 1
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt __eqsf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if two floats are equal
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #2 ");
asm("movne r0, #0 "); // if ordered and equal return 0
asm("moveq r0, #1 "); // else return 1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __eqdf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if two doubles are equal
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #2 ");
asm("movne r0, #0 "); // if ordered and equal return 0
asm("moveq r0, #1 "); // else return 1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __nesf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if two floats are not equal
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #5 "); // test if ordered and unequal
asm("moveq r0, #0 "); // if equal or unordered return 0
asm("movne r0, #1 "); // if ordered and unequal return 1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __nedf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if two doubles are not equal
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #5 "); // test if ordered and unequal
asm("moveq r0, #0 "); // if equal or unordered return 0
asm("movne r0, #1 "); // if ordered and unequal return 1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __gtsf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if one float is greater than another
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #4 "); // test if ordered and a1>a2
asm("movne r0, #1 "); // if ordered and a1>a2 return +1
asm("mvneq r0, #0 "); // else return -1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __gtdf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if one double is greater than another
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #4 "); // test if ordered and a1>a2
asm("movne r0, #1 "); // if ordered and a1>a2 return +1
asm("mvneq r0, #0 "); // else return -1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __gesf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if one float is greater than or equal to another
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #6 "); // test if ordered and a1>=a2
asm("movne r0, #1 "); // if ordered and a1>=a2 return +1
asm("mvneq r0, #0 "); // else return -1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __gedf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if one double is greater than or equal to another
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #6 "); // test if ordered and a1>=a2
asm("movne r0, #1 "); // if ordered and a1>=a2 return +1
asm("mvneq r0, #0 "); // else return -1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __ltsf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if one float is less than another
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #1 "); // test if ordered and a1<a2
asm("mvnne r0, #0 "); // if ordered and a1<a2 return -1
asm("moveq r0, #1 "); // else return +1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __ltdf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if one double is less than another
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #1 "); // test if ordered and a1<a2
asm("mvnne r0, #0 "); // if ordered and a1<a2 return -1
asm("moveq r0, #1 "); // else return +1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __lesf2(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Compare if one float is less than or equal to another
//
{
// a1 in r0, a2 in r1 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal32 "); // compare the two numbers
asm("tst r0, #3 "); // test if ordered and a1<=a2
asm("mvnne r0, #0 "); // if ordered and a1<=a2 return -1
asm("moveq r0, #1 "); // else return +1
__POPRET("");
}
__NAKED__ EXPORT_C TInt __ledf2(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Compare if one double is less than or equal to another
//
{
// a1 in r0,r1, a2 in r2,r3 on entry
asm("stmfd sp!, {lr} ");
asm("bl CompareTReal64 "); // compare the two numbers
asm("tst r0, #3 "); // test if ordered and a1<=a2
asm("mvnne r0, #0 "); // if ordered and a1<=a2 return -1
asm("moveq r0, #1 "); // else return +1
__POPRET("");
}
__NAKED__ EXPORT_C TReal32 __mulsf3(TReal32 /*a1*/,TReal32 /*a2*/)
//
// Multiply two floats
//
{
// a1 is in r0, a2 in r1 on entry; return with answer in r0
asm("stmfd sp!, {r4-r7,lr} ");
asm("bl ConvertTReal32ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r0 "); // a1 into r1
asm("bl ConvertTReal32ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXMultiply "); // multiply a1*a2, result in r1,r2,r3
asm("bl TRealXGetTReal32 "); // convert result to TReal32 in r0, error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r7,");
asm("stmfd sp!, {r0} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r4-r7,");
}
__NAKED__ EXPORT_C TReal64 __muldf3(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Multiply two doubles
//
{
// a1 is in r0,r1 a2 in r2,r3 on entry; return with answer in r0,r1
asm("stmfd sp!, {r4-r8,lr} ");
asm("mov r7, r2 "); // save a2
asm("mov r8, r3 ");
asm("mov r2, r1 "); // a1 into r1,r2
asm("mov r1, r0 ");
asm("bl ConvertTReal64ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r7 "); // a2 into r1,r2
asm("mov r2, r8 ");
asm("bl ConvertTReal64ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("bl TRealXMultiply "); // multiply a1*a2, result in r1,r2,r3
asm("bl TRealXGetTReal64 "); // convert result to TReal64 in r0,r1 error code in r12
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r8,");
asm("stmfd sp!, {r0,r1} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r1,r4-r8,");
}
__NAKED__ EXPORT_C TReal32 __divsf3(TReal32 /*a1*/, TReal32 /*a2*/)
//
// Divide two floats
//
{
// a1 is in r0, a2 in r1 on entry; return with answer in r0
asm("stmfd sp!, {r4-r9,lr} ");
asm("bl ConvertTReal32ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r0 "); // a1 into r1
asm("bl ConvertTReal32ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXDivide "); // divide a1/a2, result in r1,r2,r3 error code in r12
asm("mov r9, r12 "); // save error code in case it's division by zero
asm("bl TRealXGetTReal32 "); // convert result to TReal32 in r0, error code in r12
asm("cmn r9, #41 "); // check for KErrDivideByZero
asm("moveq r12, r9 ");
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r9,");
asm("stmfd sp!, {r0} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r4-r9,");
}
__NAKED__ EXPORT_C TReal64 __divdf3(TReal64 /*a1*/, TReal64 /*a2*/)
//
// Divide two doubles
//
{
// a1 is in r0,r1 a2 in r2,r3 on entry; return with answer in r0,r1
asm("stmfd sp!, {r4-r9,lr} ");
asm("mov r7, r0 "); // save a1
asm("mov r8, r1 ");
asm("mov r1, r2 "); // a2 into r1,r2
asm("mov r2, r3 ");
asm("bl ConvertTReal64ToTRealX "); // convert a2 to TRealX in r1,r2,r3
asm("mov r4, r1 "); // move into r4,r5,r6
asm("mov r5, r2 ");
asm("mov r6, r3 ");
asm("mov r1, r7 "); // a1 into r1,r2
asm("mov r2, r8 ");
asm("bl ConvertTReal64ToTRealX "); // convert a1 to TRealX in r1,r2,r3
asm("bl TRealXDivide "); // divide a1/a2, result in r1,r2,r3
asm("mov r9, r12 "); // save error code in case it's division by zero
asm("bl TRealXGetTReal64 "); // convert result to TReal64 in r0,r1 error code in r12
asm("cmn r9, #41 "); // check for KErrDivideByZero
asm("moveq r12, r9 ");
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"r4-r9,");
asm("stmfd sp!, {r0,r1} "); // save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,r1,r4-r9,");
}
__NAKED__ EXPORT_C TReal32 __negsf2(TReal32 /*a1*/)
//
// Negate a float
//
{
// a1 in r0 on entry, return value in r0
asm("eor r0, r0, #0x80000000 "); // change sign bit
__JUMP(,lr);
}
__NAKED__ EXPORT_C TReal64 __negdf2(TReal64 /*a1*/)
//
// Negate a double
//
{
// a1 in r0,r1 on entry, return value in r0,r1
asm("eor r0, r0, #0x80000000 "); // change sign bit
__JUMP(,lr);
}
__NAKED__ EXPORT_C TReal32 __floatsisf(TInt /*a1*/)
//
// Convert int to float
//
{
// a1 in r0 on entry, return value in r0
asm("cmp r0, #0 "); // test for zero or negative
__JUMP(eq,lr);
asm("and ip, r0, #0x80000000 "); // ip=bit 31 of r0 (sign bit)
asm("rsbmi r0, r0, #0 "); // if negative, negate it
asm("mov r2, #0x9E "); // r2=0x9E=exponent of 2^31
asm("cmp r0, #0x00010000 "); // normalise integer, adjusting exponent
asm("movcc r0, r0, lsl #16 ");
asm("subcc r2, r2, #16 ");
asm("cmp r0, #0x01000000 ");
asm("movcc r0, r0, lsl #8 ");
asm("subcc r2, r2, #8 ");
asm("cmp r0, #0x10000000 ");
asm("movcc r0, r0, lsl #4 ");
asm("subcc r2, r2, #4 ");
asm("cmp r0, #0x40000000 ");
asm("movcc r0, r0, lsl #2 ");
asm("subcc r2, r2, #2 ");
asm("cmp r0, #0x80000000 ");
asm("movcc r0, r0, lsl #1 ");
asm("subcc r2, r2, #1 ");
asm("and r1, r0, #0xFF "); // r1=bottom 8 bits=rounding bits
asm("cmp r1, #0x80 "); // check if we need to round up (carry=1 if we do)
asm("moveqs r1, r0, lsr #9 "); // if bottom 8 bits=0x80, set carry=LSB of mantissa
asm("addcss r0, r0, #0x100 "); // round up if necessary
asm("addcs r2, r2, #1 "); // if carry, increment exponent
asm("bic r0, r0, #0x80000000 "); // remove top bit (integer bit of mantissa implicit)
asm("mov r0, r0, lsr #8 "); // mantissa into r0 bits 0-22
asm("orr r0, r0, r2, lsl #23 "); // exponent into r0 bits 23-30
asm("orr r0, r0, ip "); // sign bit into r0 bit 31
__JUMP(,lr);
}
__NAKED__ EXPORT_C TReal64 __floatsidf(TInt /*a1*/)
//
// Convert int to double
//
{
// a1 in r0 on entry, return value in r0,r1
asm("cmp r0, #0 "); // test for zero or negative
asm("moveq r1, #0 "); // if zero, return 0
__JUMP(eq,lr);
asm("and ip, r0, #0x80000000 "); // ip=bit 31 of r0 (sign bit)
asm("rsbmi r0, r0, #0 "); // if negative, negate it
asm("mov r2, #0x400 "); // r2=0x41E=exponent of 2^31
asm("orr r2, r2, #0x1E ");
asm("cmp r0, #0x00010000 "); // normalise integer, adjusting exponent
asm("movcc r0, r0, lsl #16 ");
asm("subcc r2, r2, #16 ");
asm("cmp r0, #0x01000000 ");
asm("movcc r0, r0, lsl #8 ");
asm("subcc r2, r2, #8 ");
asm("cmp r0, #0x10000000 ");
asm("movcc r0, r0, lsl #4 ");
asm("subcc r2, r2, #4 ");
asm("cmp r0, #0x40000000 ");
asm("movcc r0, r0, lsl #2 ");
asm("subcc r2, r2, #2 ");
asm("cmp r0, #0x80000000 ");
asm("movcc r0, r0, lsl #1 ");
asm("subcc r2, r2, #1 ");
asm("bic r0, r0, #0x80000000 "); // remove top bit (integer bit of mantissa implicit)
asm("mov r1, r0, lsl #21 "); // low 11 bits of mantissa into r1
asm("mov r0, r0, lsr #11 "); // high 20 bits of mantissa into r0 bits 0-19
asm("orr r0, r0, r2, lsl #20 "); // exponent into r0 bits 20-30
asm("orr r0, r0, ip "); // sign bit into r0 bit 31
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov ip, r0 ");
asm("mov r0, r1 ");
asm("mov r1, ip ");
#endif
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt __fixsfsi(TReal32 /*a1*/)
//
// Convert float to int
//
{
// a1 in r0 on entry, return value in r0
asm("mov r1, r0, lsr #23 ");
asm("and r1, r1, #0xFF "); // r1=exponent of a1
asm("cmp r1, #0xFF "); // check for NaN
asm("bne fixsfsi1 ");
asm("movs r2, r0, lsl #9 "); // exponent=FF, check mantissa
asm("movne r0, #0 "); // if non-zero, a1 is a NaN so return 0
__JUMP(ne,lr);
asm("fixsfsi1: ");
asm("rsbs r1, r1, #0x9E "); // r1=number of shifts to produce integer
asm("ble fixsfsi2 "); // if <=0, saturate result
asm("cmp r0, #0 "); // check sign bit
asm("orr r0, r0, #0x00800000 "); // set implicit integer bit
asm("mov r0, r0, lsl #8 "); // shift mantissa up so MSB is in MSB of r0
asm("mov r0, r0, lsr r1 "); // r0=absolute integer
asm("rsbmi r0, r0, #0 "); // if negative, negate
__JUMP(,lr);
asm("fixsfsi2: ");
asm("cmp r0, #0 "); // check sign
asm("mov r0, #0x80000000 ");
asm("subpl r0, r0, #1 "); // if -ve return 80000000, if +ve return 7FFFFFFF
__JUMP(,lr);
}
__NAKED__ EXPORT_C TInt __fixdfsi(TReal64 /*a1*/)
//
// Convert double to int
//
{
// a1 in r0,r1 on entry, return value in r0
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r2, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r2 ");
#endif
asm("mov r2, r0, lsr #20 ");
asm("bic r2, r2, #0x800 "); // r1=exponent of a1
asm("add r3, r2, #1 ");
asm("cmp r3, #0x800 "); // check for NaN
asm("bne fixdfsi1 ");
asm("movs r3, r0, lsl #12 "); // exponent=FF, check mantissa
asm("cmpeq r1, #0 ");
asm("movne r0, #0 "); // if non-zero, a1 is a NaN so return 0
__JUMP(ne,lr);
asm("fixdfsi1: ");
asm("mov r3, #0x400 ");
asm("orr r3, r3, #0x1E "); // r3=0x41E (exponent of 2^31)
asm("subs r2, r3, r2 "); // r2=number of shifts to produce integer
asm("ble fixdfsi2 "); // if <=0, saturate result
asm("cmp r2, #31 "); // check if more than 31 shifts needed
asm("movhi r0, #0 "); // if so, underflow result to 0
__JUMP(hi,lr);
asm("cmp r0, #0 "); // check sign bit
asm("orr r0, r0, #0x00100000 "); // set implicit integer bit
asm("mov r0, r0, lsl #11 "); // shift mantissa up so MSB is in MSB of r0
asm("orr r0, r0, r1, lsr #21 "); // put in bits from r1
asm("mov r0, r0, lsr r2 "); // r0=absolute integer
asm("rsbmi r0, r0, #0 "); // if negative, negate
__JUMP(,lr);
asm("fixdfsi2: ");
asm("cmp r0, #0 "); // check sign
asm("mov r0, #0x80000000 ");
asm("subpl r0, r0, #1 "); // if -ve return 80000000, if +ve return 7FFFFFFF
__JUMP(,lr);
}
__NAKED__ EXPORT_C TReal64 __extendsfdf2(TReal32 /*a1*/)
//
// Convert a float to a double
//
{
// a1 in r0, return in r0,r1
asm("mov r3, r0, lsr #3 ");
asm("ands r3, r3, #0x0FF00000 "); // r3 bits 20-27 hold exponent, Z=1 if zero/denormal
asm("mov r1, r0, lsl #9 "); // r1 = TReal32 mantissa << 9
asm("and r0, r0, #0x80000000 "); // leave only sign bit in r0
asm("beq extendsfdf2a "); // branch if zero/denormal
asm("cmp r3, #0x0FF00000 "); // check for infinity or NaN
asm("orrcs r3, r3, #0x70000000 "); // if infinity or NaN, exponent = 7FF
asm("addcc r3, r3, #0x38000000 "); // else exponent = TReal32 exponent + 380
asm("orr r0, r0, r1, lsr #12 "); // top 20 mantissa bits into r0 bits 0-19
asm("mov r1, r1, lsl #20 "); // remaining mantissa bits in r1 bits 29-31
asm("orr r0, r0, r3 "); // exponent into r0 bits 20-30
asm("b 0f ");
asm("extendsfdf2a: "); // come here if zero or denormal
asm("cmp r1, #0 "); // check for zero
asm("beq 0f ");
asm("mov r3, #0x38000000 "); // else exponent = 380 (highest denormal exponent)
asm("cmp r1, #0x10000 "); // normalise mantissa, decrementing exponent as needed
asm("movcc r1, r1, lsl #16 ");
asm("subcc r3, r3, #0x01000000 ");
asm("cmp r1, #0x1000000 ");
asm("movcc r1, r1, lsl #8 ");
asm("subcc r3, r3, #0x00800000 ");
asm("cmp r1, #0x10000000 ");
asm("movcc r1, r1, lsl #4 ");
asm("subcc r3, r3, #0x00400000 ");
asm("cmp r1, #0x40000000 ");
asm("movcc r1, r1, lsl #2 ");
asm("subcc r3, r3, #0x00200000 ");
asm("cmp r1, #0x80000000 ");
asm("movcc r1, r1, lsl #1 ");
asm("subcc r3, r3, #0x00100000 ");
asm("add r1, r1, r1 "); // shift mantissa left one more to remove integer bit
asm("orr r0, r0, r1, lsr #12 "); // top 20 mantissa bits into r0 bits 0-19
asm("mov r1, r1, lsl #20 "); // remaining mantissa bits in r1 bits 29-31
asm("orr r0, r0, r3 "); // exponent into r0 bits 20-30
asm("0: ");
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r3, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r3 ");
#endif
__JUMP(,lr);
}
__NAKED__ EXPORT_C TReal32 __truncdfsf2(TReal64 /*a1*/)
//
// Convert a double to a float
// Raises an exception if conversion results in an error
//
{
asm("stmfd sp!, {lr} ");
asm("bl TReal64GetTReal32 "); // do the conversion
asm("cmp r12, #0 "); // check error code
__CPOPRET(eq,"");
asm("stmfd sp!, {r0} "); // else save result
asm("mov r0, r12 "); // error code into r0
asm("bl __math_exception "); // raise exception
__POPRET("r0,");
// Convert TReal64 in r0,r1 to TReal32 in r0
// Return error code in r12
// r0-r3, r12 modified
// NB This function is purely internal to EUSER and therefore IS ONLY EVER CALLED IN ARM MODE.
asm("TReal64GetTReal32: ");
#ifndef __DOUBLE_WORDS_SWAPPED__
asm("mov r2, r0 ");
asm("mov r0, r1 ");
asm("mov r1, r2 ");
#endif
asm("mov r12, r0, lsr #20 ");
asm("bic r12, r12, #0x800 "); // r12=a1 exponent
asm("sub r12, r12, #0x380 "); // r12=exp in - 380 = result exponent if in range
asm("cmp r12, #0xFF "); // check if input exponent too big for TReal32
asm("bge TReal64GetTReal32a "); // branch if it is
asm("mov r2, r0, lsl #11 "); // left justify mantissa in r2:r1
asm("orr r2, r2, r1, lsr #21 ");
asm("mov r1, r1, lsl #11 ");
asm("orr r2, r2, #0x80000000 "); // set implied integer bit in mantissa
asm("cmp r12, #0 ");
asm("bgt TReal64GetTReal32b "); // branch if normalised result
asm("cmn r12, #23 "); // check for total underflow or zero
asm("bge TReal64GetTReal32e "); // skip if not
asm("bics r2, r0, #0x80000000 "); // check if input value zero
asm("cmpeq r1, #0 ");
asm("moveq r12, #0 "); // if zero return KErrNone
asm("mvnne r12, #9 "); // else return KErrUnderflow
asm("and r0, r0, #0x80000000 "); // return zero of appropriate sign
asm("mov r1, #0 ");
__JUMP(,lr);
asm("TReal64GetTReal32e: "); // result will be a denormal
asm("add r12, r12, #31 "); // r12=32-mantissa shift required = 32-(1-r12)
asm("movs r3, r1, lsl r12 "); // r3=lost bits when r2:r1 is shifted
asm("orrne lr, lr, #1 "); // if these are not zero, set rounded down flag
asm("rsb r3, r12, #32 ");
asm("mov r1, r1, lsr r3 ");
asm("orr r1, r1, r2, lsl r12 ");
asm("mov r2, r2, lsr r3 "); // r2 top 24 bits now give unrounded result mantissa
asm("mov r12, #0 "); // result exponent will be zero
asm("TReal64GetTReal32b: ");
asm("movs r3, r2, lsl #24 "); // top 8 truncated bits into top byte of r3
asm("bpl TReal64GetTReal32c "); // if top bit clear, truncate
asm("cmp r3, #0x80000000 ");
asm("cmpeq r1, #0 "); // compare rounding bits to 1000...
asm("bhi TReal64GetTReal32d "); // if >, round up
asm("tst lr, #1 "); // check rounded-down flag
asm("bne TReal64GetTReal32d "); // if rounded down, round up
asm("tst r2, #0x100 "); // else round to even - test LSB of result mantissa
asm("beq TReal64GetTReal32c "); // if zero, truncate, else round up
asm("TReal64GetTReal32d: "); // come here to round up
asm("adds r2, r2, #0x100 "); // increment the mantissa
asm("movcs r2, #0x80000000 "); // if carry, mantissa=800000
asm("addcs r12, r12, #1 "); // and increment exponent
asm("cmpmi r12, #1 "); // if mantissa normalised, check exponent>0
asm("movmi r12, #1 "); // if normalised and exponent=0, set exponent to 1
asm("TReal64GetTReal32c: "); // come here to truncate
asm("and r0, r0, #0x80000000 "); // leave only sign bit in r0
asm("orr r0, r0, r12, lsl #23 "); // exponent into r0 bits 23-30
asm("bic r2, r2, #0x80000000 "); // remove integer bit from mantissa
asm("orr r0, r0, r2, lsr #8 "); // non-integer mantissa bits into r0 bits 0-22
asm("cmp r12, #0xFF "); // check for overflow
asm("mvneq r12, #8 "); // if overflow, return KErrOverflow
asm("biceq pc, lr, #3 ");
asm("bics r1, r0, #0x80000000 "); // check for underflow
asm("mvneq r12, #9 "); // if underflow return KErrUnderflow
asm("movne r12, #0 "); // else return KErrNone
asm("bic pc, lr, #3 ");
asm("TReal64GetTReal32a: "); // come here if overflow, infinity or NaN
asm("add r3, r12, #1 ");
asm("cmp r3, #0x480 "); // check for infinity or NaN
asm("movne r1, #0 "); // if not, set mantissa to 0 for infinity result
asm("movne r0, r0, lsr #20 ");
asm("movne r0, r0, lsl #20 ");
asm("mov r1, r1, lsr #29 "); // assemble 23 bit mantissa in r1
asm("orr r1, r1, r0, lsl #3 ");
asm("bic r1, r1, #0xFF000000 ");
asm("and r0, r0, #0x80000000 "); // leave only sign in r0
asm("orr r0, r0, #0x7F000000 "); // r0 bits 23-30 = FF = exponent
asm("orr r0, r0, #0x00800000 ");
asm("orr r0, r0, r1 "); // r0 bits 0-22 = result mantissa
asm("movs r12, r0, lsl #9 "); // check if result is infinity or NaN
asm("mvneq r12, #8 "); // if infinity return KErrOverflow
asm("mvnne r12, #5 "); // else return KErrArgument
asm("bic pc, lr, #3 ");
}
} // end of extern "C" declaration
#endif
#endif