/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #ifndef INCLUDED_TOOLS_BIGINT_HXX #define INCLUDED_TOOLS_BIGINT_HXX #include #include #include #include #include #include #define MAX_DIGITS 4 class SAL_WARN_UNUSED TOOLS_DLLPUBLIC BigInt { private: // we only use one of these two fields at a time union { sal_Int32 nVal; sal_uInt32 nNum[MAX_DIGITS]; }; sal_uInt8 nLen; // current length, if 0, data is in nVal, otherwise data is in nNum bool bIsNeg; // Is Sign negative? TOOLS_DLLPRIVATE BigInt MakeBig() const; // Create a BigInt with minimal non-0 nLen TOOLS_DLLPRIVATE void Normalize(); // Use nVal if possible, otherwise use minimal nLen TOOLS_DLLPRIVATE bool ABS_IsLessBig(BigInt const& rVal) const; TOOLS_DLLPRIVATE void AddBig(BigInt& rB, BigInt& rRes); TOOLS_DLLPRIVATE void SubBig(BigInt& rB, BigInt& rRes); TOOLS_DLLPRIVATE void MultBig(BigInt const& rB, BigInt& rRes) const; TOOLS_DLLPRIVATE void DivModBig(BigInt const& rB, BigInt* pDiv, BigInt* pMod) const; public: BigInt() : nVal(0) , nLen(0) , bIsNeg(false) { } BigInt(sal_Int32 nValue) : nVal(nValue) , nLen(0) , bIsNeg(false) { } BigInt( double nVal ); BigInt( sal_uInt32 nVal ); BigInt( sal_Int64 nVal ); BigInt(const BigInt& rBigInt) = default; BigInt( std::u16string_view rString ); template requires(std::is_integral_v && std::is_signed_v && sizeof(N) <= sizeof(sal_Int32)) BigInt(N val) : BigInt(sal_Int32(val)) { } template requires(std::is_integral_v && std::is_unsigned_v && sizeof(N) <= sizeof(sal_uInt32)) BigInt(N val) : BigInt(sal_uInt32(val)) { } template requires(std::is_integral_v && std::is_signed_v && sizeof(N) == sizeof(sal_Int64)) BigInt(N val) : BigInt(sal_Int64(val)) { } operator sal_Int16() const; operator sal_uInt16() const; operator sal_Int32() const; operator sal_uInt32() const; operator double() const; operator sal_Int64() const; bool IsNeg() const { return IsBig() ? bIsNeg : nVal < 0; } bool IsZero() const { return !IsBig() && nVal == 0; } bool IsBig() const { return nLen != 0; } BigInt Abs() const; BigInt& operator =( const BigInt& rVal ); BigInt& operator +=( const BigInt& rVal ); BigInt& operator -=( const BigInt& rVal ); BigInt& operator *=( const BigInt& rVal ); BigInt& operator /=( const BigInt& rVal ); BigInt& operator %=( const BigInt& rVal ); BigInt& operator =( sal_Int32 nValue ); void DivMod(BigInt const& rVal, BigInt* pDiv, BigInt* pMod) const; /* Scale and round value */ static tools::Long Scale(tools::Long nVal, tools::Long nMult, tools::Long nDiv); TOOLS_DLLPUBLIC friend bool operator==( const BigInt& rVal1, const BigInt& rVal2 ); TOOLS_DLLPUBLIC friend std::strong_ordering operator<=> ( const BigInt& rVal1, const BigInt& rVal2 ); }; inline BigInt::operator sal_Int16() const { if (!IsBig() && nVal >= SAL_MIN_INT16 && nVal <= SAL_MAX_INT16) return static_cast(nVal); assert(false && "out of range"); return 0; } inline BigInt::operator sal_uInt16() const { if (!IsBig() && nVal >= 0 && nVal <= SAL_MAX_UINT16) return static_cast(nVal); assert(false && "out of range"); return 0; } inline BigInt::operator sal_Int32() const { if (!IsBig()) return nVal; assert(false && "out of range"); return 0; } inline BigInt::operator sal_uInt32() const { if (!IsBig() && nVal >= 0) return static_cast(nVal); assert(false && "out of range"); return 0; } inline BigInt::operator sal_Int64() const { constexpr sal_uInt64 maxForPosInt64 = std::numeric_limits::max(); constexpr sal_uInt64 maxForNegInt64 = std::numeric_limits::min(); switch (nLen) { case 0: return nVal; case 1: return bIsNeg ? -sal_Int64(nNum[0]) : nNum[0]; case 2: if (sal_uInt64 n = (sal_uInt64(nNum[1]) << 32) + nNum[0]; bIsNeg && n < maxForNegInt64) return -sal_Int64(n); else if ((bIsNeg && n == maxForNegInt64) || (!bIsNeg && n <= maxForPosInt64)) return n; // maxForNegInt64 will convert correctly } assert(false && "out of range"); return 0; } inline BigInt& BigInt::operator =( sal_Int32 nValue ) { nLen = 0; nVal = nValue; return *this; } inline BigInt BigInt::Abs() const { BigInt aRes(*this); if (IsBig()) aRes.bIsNeg = false; else if ( nVal < 0 ) { if (nVal == std::numeric_limits::min()) { aRes.nNum[0] = -sal_Int64(std::numeric_limits::min()); aRes.nLen = 1; aRes.bIsNeg = false; } else aRes.nVal = -nVal; } return aRes; } inline BigInt operator+( const BigInt &rVal1, const BigInt &rVal2 ) { BigInt aRes( rVal1 ); aRes += rVal2; return aRes; } inline BigInt operator-( const BigInt &rVal1, const BigInt &rVal2 ) { BigInt aRes( rVal1 ); aRes -= rVal2; return aRes; } inline BigInt operator*( const BigInt &rVal1, const BigInt &rVal2 ) { BigInt aRes( rVal1 ); aRes *= rVal2; return aRes; } inline BigInt operator/( const BigInt &rVal1, const BigInt &rVal2 ) { BigInt aRes( rVal1 ); aRes /= rVal2; return aRes; } inline BigInt operator%( const BigInt &rVal1, const BigInt &rVal2 ) { BigInt aRes( rVal1 ); aRes %= rVal2; return aRes; } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */