diff --git a/Sources/Data Conversion.swift b/Sources/Data Conversion.swift index b74e683..3b35fe8 100644 --- a/Sources/Data Conversion.swift +++ b/Sources/Data Conversion.swift @@ -68,6 +68,8 @@ extension BigUInt { i -= 1 } } + let zeroOut = UnsafeMutableBufferPointer(start: buffer.baseAddress, count: i) + zeroOut.initialize(repeating: 0) return UnsafeRawBufferPointer(start: buffer.baseAddress, count: byteCount) } diff --git a/Sources/Floating Point Conversion.swift b/Sources/Floating Point Conversion.swift index 8c4b767..5a5e8b2 100644 --- a/Sources/Floating Point Conversion.swift +++ b/Sources/Floating Point Conversion.swift @@ -29,38 +29,43 @@ extension BigUInt { #if canImport(Foundation) public init?(exactly source: Decimal) { - guard source.isFinite else { return nil } - guard !source.isZero else { self = 0; return } - guard source.sign == .plus else { return nil } - assert(source.floatingPointClass == .positiveNormal) guard source.exponent >= 0 else { return nil } - let intMaxD = Decimal(UInt.max) - let intMaxB = BigUInt(UInt.max) - var start = BigUInt() - var value = source - while value >= intMaxD { - start += intMaxB - value -= intMaxD - } - start += BigUInt((value as NSNumber).uintValue) - self = start + self.init(commonDecimal: source) } public init?(truncating source: Decimal) { - guard source.isFinite else { return nil } - guard !source.isZero else { self = 0; return } - guard source.sign == .plus else { return nil } - assert(source.floatingPointClass == .positiveNormal) - let intMaxD = Decimal(UInt.max) - let intMaxB = BigUInt(UInt.max) - var start = BigUInt() - var value = source - while value >= intMaxD { - start += intMaxB - value -= intMaxD + self.init(commonDecimal: source) + } + + private init?(commonDecimal source: Decimal) { + var integer = source + if source.exponent < 0 { + var source = source + NSDecimalRound(&integer, &source, 0, .down) } - start += BigUInt((value as NSNumber).uintValue) - self = start + + guard !integer.isZero else { self = 0; return } + guard integer.isFinite else { return nil } + guard integer.sign == .plus else { return nil } + assert(integer.floatingPointClass == .positiveNormal) + + #if os(Linux) + // `Decimal._mantissa` has an internal access level on linux, and it might get + // deprecated in the future, so keeping the string implementation around for now. + let significand = BigUInt("\(integer.significand)")! + #else + let significand = { + var start = BigUInt(0) + for (place, value) in integer.significand.mantissaParts.enumerated() { + guard value > 0 else { continue } + start += (1 << (place * 16)) * BigUInt(value) + } + return start + }() + #endif + let exponent = BigUInt(10).power(integer.exponent) + + self = significand * exponent } #endif } @@ -132,3 +137,45 @@ extension BigInt.Sign { } } } + +#if canImport(Foundation) +public extension Decimal { + init(_ value: BigUInt) { + guard + value < BigUInt(exactly: Decimal.greatestFiniteMagnitude)! + else { + self = .greatestFiniteMagnitude + return + } + guard !value.isZero else { self = 0; return } + + self.init(string: "\(value)")! + } + + init(_ value: BigInt) { + if value >= 0 { + self.init(BigUInt(value)) + } else { + self.init(value.magnitude) + self *= -1 + } + } +} +#endif + +#if canImport(Foundation) && !os(Linux) +private extension Decimal { + var mantissaParts: [UInt16] { + [ + _mantissa.0, + _mantissa.1, + _mantissa.2, + _mantissa.3, + _mantissa.4, + _mantissa.5, + _mantissa.6, + _mantissa.7, + ] + } +} +#endif diff --git a/Tests/BigIntTests/BigIntTests.swift b/Tests/BigIntTests/BigIntTests.swift index c46c022..76da4a5 100644 --- a/Tests/BigIntTests/BigIntTests.swift +++ b/Tests/BigIntTests/BigIntTests.swift @@ -102,6 +102,8 @@ class BigIntTests: XCTestCase { XCTAssertEqual(BigInt(exactly: Decimal(1001.5)), nil) XCTAssertEqual(BigInt(exactly: Decimal(UInt.max) + 5), "18446744073709551620") XCTAssertEqual(BigInt(exactly: (Decimal(UInt.max) + 5.5)), nil) + XCTAssertEqual(BigInt(exactly: Decimal.greatestFiniteMagnitude), + "3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000") XCTAssertEqual(BigInt(truncating: Decimal(0)), 0) XCTAssertEqual(BigInt(truncating: Decimal(Double.nan)), nil) XCTAssertEqual(BigInt(truncating: Decimal(10)), 10) @@ -119,6 +121,8 @@ class BigIntTests: XCTestCase { XCTAssertEqual(BigInt(exactly: -Decimal(1001.5)), nil) XCTAssertEqual(BigInt(exactly: -(Decimal(UInt.max) + 5)), "-18446744073709551620") XCTAssertEqual(BigInt(exactly: -(Decimal(UInt.max) + 5.5)), nil) + XCTAssertEqual(BigInt(exactly: Decimal.leastFiniteMagnitude), + "-3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000") XCTAssertEqual(BigInt(truncating: -Decimal(10)), -10) XCTAssertEqual(BigInt(truncating: -Decimal(1000)), -1000) XCTAssertEqual(BigInt(truncating: -Decimal(1000.1)), -1000) @@ -181,6 +185,18 @@ class BigIntTests: XCTestCase { test(BigInt(1) << 1024, Float.infinity) test(BigInt(words: convertWords([0, 0xFFFFFF0000000000, 0])), Float.greatestFiniteMagnitude) + + XCTAssertEqual(Decimal(BigInt(0)), 0) + XCTAssertEqual(Decimal(BigInt(20)), 20) + XCTAssertEqual(Decimal(BigInt(123456789)), 123456789) + XCTAssertEqual(Decimal(BigInt(exactly: Decimal.greatestFiniteMagnitude)!), .greatestFiniteMagnitude) + XCTAssertEqual(Decimal(BigInt(exactly: Decimal.greatestFiniteMagnitude)! * 2), .greatestFiniteMagnitude) + XCTAssertEqual(Decimal(-BigInt(0)), 0) + XCTAssertEqual(Decimal(-BigInt(20)), -20) + XCTAssertEqual(Decimal(-BigInt(123456789)), -123456789) + XCTAssertEqual(Decimal(-BigInt(exactly: Decimal.greatestFiniteMagnitude)!), -.greatestFiniteMagnitude) + XCTAssertEqual(Decimal(-BigInt(exactly: Decimal.greatestFiniteMagnitude)! * 2), -.greatestFiniteMagnitude) + } func testTwosComplement() { diff --git a/Tests/BigIntTests/BigUIntTests.swift b/Tests/BigIntTests/BigUIntTests.swift index 08412ac..244901c 100644 --- a/Tests/BigIntTests/BigUIntTests.swift +++ b/Tests/BigIntTests/BigUIntTests.swift @@ -167,6 +167,8 @@ class BigUIntTests: XCTestCase { XCTAssertEqual(BigUInt(exactly: Decimal(1001.5)), nil) XCTAssertEqual(BigUInt(exactly: Decimal(UInt.max) + 5), "18446744073709551620") XCTAssertEqual(BigUInt(exactly: (Decimal(UInt.max) + 5.5)), nil) + XCTAssertEqual(BigUInt(exactly: Decimal.greatestFiniteMagnitude), + "3402823669209384634633746074317682114550000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000") XCTAssertEqual(BigUInt(truncating: Decimal(0)), 0) XCTAssertEqual(BigUInt(truncating: Decimal(Double.nan)), nil) XCTAssertEqual(BigUInt(truncating: Decimal(10)), 10) @@ -184,6 +186,7 @@ class BigUIntTests: XCTestCase { XCTAssertEqual(BigUInt(exactly: -Decimal(1001.5)), nil) XCTAssertEqual(BigUInt(exactly: -Decimal(UInt.max) + 5), nil) XCTAssertEqual(BigUInt(exactly: -(Decimal(UInt.max) + 5.5)), nil) + XCTAssertEqual(BigUInt(exactly: Decimal.leastFiniteMagnitude), nil) XCTAssertEqual(BigUInt(truncating: -Decimal(10)), nil) XCTAssertEqual(BigUInt(truncating: -Decimal(1000)), nil) XCTAssertEqual(BigUInt(truncating: -Decimal(1000.1)), nil) @@ -261,6 +264,12 @@ class BigUIntTests: XCTestCase { test(BigUInt(0x8000027FFFFFFFFF as UInt64), 0x800002p40 as Float) test(BigUInt(0x8000028000000000 as UInt64), 0x800002p40 as Float) test(BigUInt(0x800002FFFFFFFFFF as UInt64), 0x800002p40 as Float) + + XCTAssertEqual(Decimal(BigUInt(0)), 0) + XCTAssertEqual(Decimal(BigUInt(20)), 20) + XCTAssertEqual(Decimal(BigUInt(123456789)), 123456789) + XCTAssertEqual(Decimal(BigUInt(exactly: Decimal.greatestFiniteMagnitude)!), .greatestFiniteMagnitude) + XCTAssertEqual(Decimal(BigUInt(exactly: Decimal.greatestFiniteMagnitude)! * 2), .greatestFiniteMagnitude) } func testInit_Misc() { @@ -685,6 +694,7 @@ class BigUIntTests: XCTestCase { test(BigUInt(), []) test(BigUInt(1), [0x01]) test(BigUInt(2), [0x02]) + test(BigUInt(0x010203), [0x1, 0x2, 0x3]) test(BigUInt(0x0102030405060708), [0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08]) test(BigUInt(0x01) << 64 + BigUInt(0x0203040506070809), [0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09]) } @@ -1513,5 +1523,4 @@ class BigUIntTests: XCTestCase { let limit = BigUInt(UInt64.max) * BigUInt(UInt64.max) * BigUInt(UInt64.max) check { BigUInt.randomInteger(lessThan: limit, using: &$0) } } - }