From db12067a8adec1317b88f9d984810ca437085cb5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Oscar=20Bystr=C3=B6m=20Ericsson?= Date: Tue, 17 Dec 2024 05:04:05 +0100 Subject: [PATCH 1/5] This patch adds **unavailable** `I128` and `U128` models (#143). The underlying `Swift.Int128` and `Swift.UInt128` models fail some tests (crash). I have submitted some fixes that got merged into the `Swift 6.1` branch. I'll try again later and make the `I128` and `U128` available when all tests pass successfully. --- Sources/CoreIop/BinaryInteger.swift | 28 ++++-- Sources/CoreKit/Models/CoreInt.swift | 92 +++++++++++++++++++ Sources/CoreKit/Stdlib/Swift+Int.swift | 21 +++++ Sources/CoreKit/Stdlib/Swift+UInt.swift | 8 ++ .../UltimathiopTests/Utilities/Globals.swift | 48 ++++++++-- .../UltimathnumTests/Utilities/Globals.swift | 50 +++++++--- 6 files changed, 216 insertions(+), 31 deletions(-) diff --git a/Sources/CoreIop/BinaryInteger.swift b/Sources/CoreIop/BinaryInteger.swift index 207b151b..3f740cdb 100644 --- a/Sources/CoreIop/BinaryInteger.swift +++ b/Sources/CoreIop/BinaryInteger.swift @@ -13,14 +13,22 @@ import CoreKit // MARK: * Binary Integer //*============================================================================* -extension IX: CompactIntegerInteroperable { } -extension I8: CompactIntegerInteroperable { } -extension I16: CompactIntegerInteroperable { } -extension I32: CompactIntegerInteroperable { } -extension I64: CompactIntegerInteroperable { } +extension IX: CompactIntegerInteroperable { } +extension I8: CompactIntegerInteroperable { } +extension I16: CompactIntegerInteroperable { } +extension I32: CompactIntegerInteroperable { } +extension I64: CompactIntegerInteroperable { } -extension UX: NaturalIntegerInteroperable { } -extension U8: NaturalIntegerInteroperable { } -extension U16: NaturalIntegerInteroperable { } -extension U32: NaturalIntegerInteroperable { } -extension U64: NaturalIntegerInteroperable { } +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +extension I128: CompactIntegerInteroperable { } + +extension UX: NaturalIntegerInteroperable { } +extension U8: NaturalIntegerInteroperable { } +extension U16: NaturalIntegerInteroperable { } +extension U32: NaturalIntegerInteroperable { } +extension U64: NaturalIntegerInteroperable { } + +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +extension U128: NaturalIntegerInteroperable { } diff --git a/Sources/CoreKit/Models/CoreInt.swift b/Sources/CoreKit/Models/CoreInt.swift index 59eacdc5..e5f46aca 100644 --- a/Sources/CoreKit/Models/CoreInt.swift +++ b/Sources/CoreKit/Models/CoreInt.swift @@ -234,6 +234,52 @@ } } +//*============================================================================* +// MARK: * Core Int x I128 +//*============================================================================* + +/// A 128-bit signed binary integer. +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +@frozen public struct I128: CoreInteger, SignedInteger, CoreIntegerWhereIsNotByte, CoreIntegerWhereIsNotToken { + + public typealias Stdlib = Swift.Int128 + + public typealias BitPattern = Stdlib.BitPattern + + public typealias Element = Self + + public typealias Signitude = Self + + public typealias Magnitude = U128 + + //=------------------------------------------------------------------------= + // MARK: State + //=------------------------------------------------------------------------= + + @usableFromInline var base: Stdlib + + //=------------------------------------------------------------------------= + // MARK: Initializers + //=------------------------------------------------------------------------= + + @inlinable public init(_ source: Stdlib) { + self.base = source + } + + @inlinable public init(raw source: BitPattern) { + self.base = Stdlib(bitPattern: source) + } + + @inlinable public init(integerLiteral source: Stdlib) { + self.base = source + } + + @inlinable public consuming func stdlib() -> Stdlib { + self.base + } +} + //*============================================================================* // MARK: * Core Int x UX //*============================================================================* @@ -453,3 +499,49 @@ self.base } } + +//*============================================================================* +// MARK: * Core Int x U128 +//*============================================================================* + +/// A 128-bit unsigned binary integer. +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +@frozen public struct U128: CoreInteger, UnsignedInteger, CoreIntegerWhereIsNotByte, CoreIntegerWhereIsNotToken { + + public typealias Stdlib = Swift.UInt128 + + public typealias BitPattern = Stdlib.BitPattern + + public typealias Element = Self + + public typealias Signitude = I128 + + public typealias Magnitude = Self + + //=------------------------------------------------------------------------= + // MARK: State + //=------------------------------------------------------------------------= + + @usableFromInline var base: Stdlib + + //=------------------------------------------------------------------------= + // MARK: Initializers + //=------------------------------------------------------------------------= + + @inlinable public init(_ source: Stdlib) { + self.base = source + } + + @inlinable public init(raw source: BitPattern) { + self.base = source + } + + @inlinable public init(integerLiteral source: Stdlib) { + self.base = source + } + + @inlinable public consuming func stdlib() -> Stdlib { + self.base + } +} diff --git a/Sources/CoreKit/Stdlib/Swift+Int.swift b/Sources/CoreKit/Stdlib/Swift+Int.swift index 0b531498..b15cf13c 100644 --- a/Sources/CoreKit/Stdlib/Swift+Int.swift +++ b/Sources/CoreKit/Stdlib/Swift+Int.swift @@ -101,3 +101,24 @@ extension Int64: BitCastable { Magnitude(bitPattern: self) } } + +//*============================================================================* +// MARK: * Swift x Int128 +//*============================================================================* + +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +extension Int128: BitCastable { + + //=------------------------------------------------------------------------= + // MARK: Initializers + //=------------------------------------------------------------------------= + + @inlinable public init(raw source: consuming Magnitude) { + self.init(bitPattern: source) + } + + @inlinable public consuming func load(as type: Magnitude.Type) -> Magnitude { + Magnitude(bitPattern: self) + } +} diff --git a/Sources/CoreKit/Stdlib/Swift+UInt.swift b/Sources/CoreKit/Stdlib/Swift+UInt.swift index a016ea70..5af9b4d4 100644 --- a/Sources/CoreKit/Stdlib/Swift+UInt.swift +++ b/Sources/CoreKit/Stdlib/Swift+UInt.swift @@ -36,3 +36,11 @@ extension UInt32: BitCastable { public typealias BitPattern = Magnitude } //*============================================================================* extension UInt64: BitCastable { public typealias BitPattern = Magnitude } + +//*============================================================================* +// MARK: * Swift x UInt128 +//*============================================================================* + +@available(*, unavailable) +@available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) +extension UInt128: BitCastable { public typealias BitPattern = Magnitude } diff --git a/Tests/UltimathiopTests/Utilities/Globals.swift b/Tests/UltimathiopTests/Utilities/Globals.swift index d962b847..fdc402e7 100644 --- a/Tests/UltimathiopTests/Utilities/Globals.swift +++ b/Tests/UltimathiopTests/Utilities/Globals.swift @@ -42,14 +42,42 @@ let typesAsLenientIntegerInteroperable: [any LenientIntegerInteroperable.Type] = InfiniInt.self, ] -let typesAsCompactIntegerInteroperable: [any CompactIntegerInteroperable.Type] = [ - IX.self, I8 .self, I16.self, I32.self, I64 .self, - DoubleInt.self, DoubleInt>.self, - DoubleInt.self, DoubleInt>.self, -] +let typesAsCompactIntegerInteroperable: [any CompactIntegerInteroperable.Type] = reduce([]) { + $0.append(IX .self) + $0.append(I8 .self) + $0.append(I16.self) + $0.append(I32.self) + $0.append(I64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(I128.self) + } + #endif + + $0.append(DoubleInt.self) + $0.append(DoubleInt.self) + + $0.append(DoubleInt>.self) + $0.append(DoubleInt>.self) +} -let typesAsNaturalIntegerInteroperable: [any NaturalIntegerInteroperable.Type] = [ - UX.self, U8 .self, U16.self, U32.self, U64 .self, - DoubleInt.self, DoubleInt>.self, - DoubleInt.self, DoubleInt>.self, -] +let typesAsNaturalIntegerInteroperable: [any NaturalIntegerInteroperable.Type] = reduce([]) { + $0.append(UX .self) + $0.append(U8 .self) + $0.append(U16.self) + $0.append(U32.self) + $0.append(U64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(CoreKit.U128.self) + } + #endif + + $0.append(DoubleInt.self) + $0.append(DoubleInt.self) + + $0.append(DoubleInt>.self) + $0.append(DoubleInt>.self) +} diff --git a/Tests/UltimathnumTests/Utilities/Globals.swift b/Tests/UltimathnumTests/Utilities/Globals.swift index 65dd9413..6d8a0595 100644 --- a/Tests/UltimathnumTests/Utilities/Globals.swift +++ b/Tests/UltimathnumTests/Utilities/Globals.swift @@ -102,14 +102,42 @@ let typesAsSystemsInteger: [any SystemsInteger.Type] = { typesAsSystemsIntegerAsUnsigned }() -let typesAsSystemsIntegerAsSigned: [any SystemsIntegerAsSigned.Type] = [ - IX.self, I8 .self, I16.self, I32.self, I64 .self, - DoubleInt.self, DoubleInt>.self, - DoubleInt.self, DoubleInt>.self, -] - -let typesAsSystemsIntegerAsUnsigned: [any SystemsIntegerAsUnsigned.Type] = [ - UX.self, U8 .self, U16.self, U32.self, U64 .self, - DoubleInt.self, DoubleInt>.self, - DoubleInt.self, DoubleInt>.self, -] +let typesAsSystemsIntegerAsSigned: [any SystemsIntegerAsSigned.Type] = reduce([]) { + $0.append(IX .self) + $0.append(I8 .self) + $0.append(I16.self) + $0.append(I32.self) + $0.append(I64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(I128.self) + } + #endif + + $0.append(DoubleInt.self) + $0.append(DoubleInt.self) + + $0.append(DoubleInt>.self) + $0.append(DoubleInt>.self) +} + +let typesAsSystemsIntegerAsUnsigned: [any SystemsIntegerAsUnsigned.Type] = reduce([]) { + $0.append(UX .self) + $0.append(U8 .self) + $0.append(U16.self) + $0.append(U32.self) + $0.append(U64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(CoreKit.U128.self) + } + #endif + + $0.append(DoubleInt.self) + $0.append(DoubleInt.self) + + $0.append(DoubleInt>.self) + $0.append(DoubleInt>.self) +} From 7c42294a3b0632401e6aa79de00d35ce6507bef1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Oscar=20Bystr=C3=B6m=20Ericsson?= Date: Tue, 17 Dec 2024 07:53:42 +0100 Subject: [PATCH 2/5] Conditionally add I128 and U128 to testing lists. --- Sources/TestKit/Global+Types.swift | 32 +++++++++++++++---- .../UltimathiopTests/Utilities/Globals.swift | 4 +-- .../UltimathnumTests/Utilities/Globals.swift | 3 +- 3 files changed, 28 insertions(+), 11 deletions(-) diff --git a/Sources/TestKit/Global+Types.swift b/Sources/TestKit/Global+Types.swift index 05e451c2..7e62076b 100644 --- a/Sources/TestKit/Global+Types.swift +++ b/Sources/TestKit/Global+Types.swift @@ -26,13 +26,33 @@ public let typesAsCoreIntegerAsByte: [any CoreInteger.Type] = [ U8.self, ] -public let typesAsCoreIntegerAsSigned: [any CoreIntegerAsSigned.Type] = [ - IX.self, I8.self, I16.self, I32.self, I64.self, -] +public let typesAsCoreIntegerAsSigned: [any CoreIntegerAsSigned.Type] = reduce([]) { + $0.append(IX .self) + $0.append(I8 .self) + $0.append(I16.self) + $0.append(I32.self) + $0.append(I64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(CoreKit.I128.self) + } + #endif +} -public let typesAsCoreIntegerAsUnsigned: [any CoreIntegerAsUnsigned.Type] = [ - UX.self, U8.self, U16.self, U32.self, U64.self, -] +public let typesAsCoreIntegerAsUnsigned: [any CoreIntegerAsUnsigned.Type] = reduce([]) { + $0.append(UX .self) + $0.append(U8 .self) + $0.append(U16.self) + $0.append(U32.self) + $0.append(U64.self) + + #if false + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + $0.append(CoreKit.U128.self) + } + #endif +} //=----------------------------------------------------------------------------= // MARK: + Floats diff --git a/Tests/UltimathiopTests/Utilities/Globals.swift b/Tests/UltimathiopTests/Utilities/Globals.swift index fdc402e7..7482109f 100644 --- a/Tests/UltimathiopTests/Utilities/Globals.swift +++ b/Tests/UltimathiopTests/Utilities/Globals.swift @@ -51,13 +51,12 @@ let typesAsCompactIntegerInteroperable: [any CompactIntegerInteroperable.Type] = #if false if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { - $0.append(I128.self) + $0.append(CoreKit.I128.self) } #endif $0.append(DoubleInt.self) $0.append(DoubleInt.self) - $0.append(DoubleInt>.self) $0.append(DoubleInt>.self) } @@ -77,7 +76,6 @@ let typesAsNaturalIntegerInteroperable: [any NaturalIntegerInteroperable.Type] = $0.append(DoubleInt.self) $0.append(DoubleInt.self) - $0.append(DoubleInt>.self) $0.append(DoubleInt>.self) } diff --git a/Tests/UltimathnumTests/Utilities/Globals.swift b/Tests/UltimathnumTests/Utilities/Globals.swift index 6d8a0595..358529b9 100644 --- a/Tests/UltimathnumTests/Utilities/Globals.swift +++ b/Tests/UltimathnumTests/Utilities/Globals.swift @@ -111,7 +111,7 @@ let typesAsSystemsIntegerAsSigned: [any SystemsIntegerAsSigned.Type] = reduce([] #if false if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { - $0.append(I128.self) + $0.append(CoreKit.I128.self) } #endif @@ -137,7 +137,6 @@ let typesAsSystemsIntegerAsUnsigned: [any SystemsIntegerAsUnsigned.Type] = reduc $0.append(DoubleInt.self) $0.append(DoubleInt.self) - $0.append(DoubleInt>.self) $0.append(DoubleInt>.self) } From 2cc4f77c13b23c889dca1c806d9175bf3fe6e2cd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Oscar=20Bystr=C3=B6m=20Ericsson?= Date: Tue, 17 Dec 2024 07:58:14 +0100 Subject: [PATCH 3/5] Todos about 128-bit DoubleInt typee aliases. --- Sources/CoreKit/Models/CoreInt.swift | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/Sources/CoreKit/Models/CoreInt.swift b/Sources/CoreKit/Models/CoreInt.swift index e5f46aca..99588e35 100644 --- a/Sources/CoreKit/Models/CoreInt.swift +++ b/Sources/CoreKit/Models/CoreInt.swift @@ -239,6 +239,11 @@ //*============================================================================* /// A 128-bit signed binary integer. +/// +/// ### Development +/// +/// - Todo: Remove `typealias I128 = DoubleInt` in `DoubleIntKit`. +/// @available(*, unavailable) @available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) @frozen public struct I128: CoreInteger, SignedInteger, CoreIntegerWhereIsNotByte, CoreIntegerWhereIsNotToken { @@ -505,6 +510,11 @@ //*============================================================================* /// A 128-bit unsigned binary integer. +/// +/// ### Development +/// +/// - Todo: Remove `typealias U128 = DoubleInt` in `DoubleIntKit`. +/// @available(*, unavailable) @available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) @frozen public struct U128: CoreInteger, UnsignedInteger, CoreIntegerWhereIsNotByte, CoreIntegerWhereIsNotToken { From 699ff86ff5eae9500bdd57e5af248926ef12c59f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Oscar=20Bystr=C3=B6m=20Ericsson?= Date: Tue, 17 Dec 2024 08:09:15 +0100 Subject: [PATCH 4/5] Add heterogeneous binary integer element alignment tests. --- .../BinaryInteger+Metadata.swift | 44 +++++++++++++++++-- 1 file changed, 40 insertions(+), 4 deletions(-) diff --git a/Tests/UltimathnumTests/BinaryInteger+Metadata.swift b/Tests/UltimathnumTests/BinaryInteger+Metadata.swift index c66e90ee..a56ddb5e 100644 --- a/Tests/UltimathnumTests/BinaryInteger+Metadata.swift +++ b/Tests/UltimathnumTests/BinaryInteger+Metadata.swift @@ -16,7 +16,7 @@ import TestKit // MARK: * Binary Integer x Metadata //*============================================================================* -@Suite(.tags(.documentation)) struct BinaryIntegerTestsOnMetadata { +@Suite struct BinaryIntegerTestsOnMetadata { //=------------------------------------------------------------------------= // MARK: Tests @@ -24,7 +24,7 @@ import TestKit @Test( "BinaryInteger/metadata: mode", - Tag.List.tags(.generic), + Tag.List.tags(.documentation, .generic), arguments: typesAsBinaryInteger ) func modes(type: any BinaryInteger.Type) throws { @@ -37,7 +37,7 @@ import TestKit @Test( "BinaryInteger/metadata: size", - Tag.List.tags(.generic), + Tag.List.tags(.documentation, .generic), arguments: typesAsBinaryInteger ) func sizes(type: any BinaryInteger.Type) throws { @@ -66,7 +66,7 @@ import TestKit @Test( "BinaryInteger/metadata: quadrants", - Tag.List.tags(.generic), + Tag.List.tags(.documentation, .generic), arguments: typesAsBinaryInteger ) func quadrants(type: any BinaryInteger.Type) throws { @@ -94,3 +94,39 @@ import TestKit } } } + +//*============================================================================* +// MARK: * Binary Integer x Metadata x Elements +//*============================================================================* + +@Suite struct BinaryIntegerTestsOnMetadataAboutElements { + + //=------------------------------------------------------------------------= + // MARK: Tests + //=------------------------------------------------------------------------= + + /// - Seealso: https://github.com/oscbyspro/Ultimathnum/pull/152 + @Test( + "BinaryInteger/metadata/elements: A.alignment vs B.alignment", + Tag.List.tags(.documentation, .generic), + arguments: CollectionOfOne(typesAsCoreInteger) + ) func heterogeneousAlignmentComparisons(list: [any CoreInteger.Type]) throws { + + for type in list { + for other in list { + try whereIs(type, versus: other) + } + } + + func whereIs(_ type: A.Type, versus other: B.Type) throws where A: CoreInteger, B: CoreInteger { + let a = MemoryLayout.self + let b = MemoryLayout.self + + switch A.size.compared(to: B.size) { + case Signum.negative: try #require(a.alignment <= b.alignment) + case Signum.zero: try #require(a.alignment == b.alignment) + case Signum.positive: try #require(a.alignment >= b.alignment) + } + } + } +} From 54ca192495f6b41e2f88b5d9b13cfc6d07bb998b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Oscar=20Bystr=C3=B6m=20Ericsson?= Date: Tue, 17 Dec 2024 08:22:34 +0100 Subject: [PATCH 5/5] Cleanup. --- Sources/TestKit/Global+Types.swift | 2 +- Tests/UltimathiopTests/Utilities/Globals.swift | 2 +- Tests/UltimathnumTests/Utilities/Globals.swift | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/Sources/TestKit/Global+Types.swift b/Sources/TestKit/Global+Types.swift index 7e62076b..d90e80d9 100644 --- a/Sources/TestKit/Global+Types.swift +++ b/Sources/TestKit/Global+Types.swift @@ -34,7 +34,7 @@ public let typesAsCoreIntegerAsSigned: [any CoreIntegerAsSigned.Type] = reduce([ $0.append(I64.self) #if false - if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { $0.append(CoreKit.I128.self) } #endif diff --git a/Tests/UltimathiopTests/Utilities/Globals.swift b/Tests/UltimathiopTests/Utilities/Globals.swift index 7482109f..c0828e46 100644 --- a/Tests/UltimathiopTests/Utilities/Globals.swift +++ b/Tests/UltimathiopTests/Utilities/Globals.swift @@ -50,7 +50,7 @@ let typesAsCompactIntegerInteroperable: [any CompactIntegerInteroperable.Type] = $0.append(I64.self) #if false - if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { $0.append(CoreKit.I128.self) } #endif diff --git a/Tests/UltimathnumTests/Utilities/Globals.swift b/Tests/UltimathnumTests/Utilities/Globals.swift index 358529b9..e32f09be 100644 --- a/Tests/UltimathnumTests/Utilities/Globals.swift +++ b/Tests/UltimathnumTests/Utilities/Globals.swift @@ -110,7 +110,7 @@ let typesAsSystemsIntegerAsSigned: [any SystemsIntegerAsSigned.Type] = reduce([] $0.append(I64.self) #if false - if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { + if #available(iOS 18.0, macOS 15.0, tvOS 18.0, visionOS 2.0, watchOS 11.0, *) { $0.append(CoreKit.I128.self) } #endif