Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[stdlib] Implement SE-0233 AdditiveArithmetic #20422

Merged
merged 9 commits into from
Nov 10, 2018
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
142 changes: 81 additions & 61 deletions stdlib/public/core/Integers.swift
Original file line number Diff line number Diff line change
Expand Up @@ -23,10 +23,89 @@ extension ExpressibleByIntegerLiteral
}
}

//===----------------------------------------------------------------------===//
//===--- AdditiveArithmetic -----------------------------------------------===//
//===----------------------------------------------------------------------===//

// FIXME: Add doc comment.
public protocol AdditiveArithmetic : Equatable {
/// The zero value.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we expand this comment with a note that zero is the name of the additive identity? Something like
/// zero is the identity element for addition; for any value x,x + .zero == x and .zero + x == x.

Copy link
Contributor Author

@rxwei rxwei Nov 8, 2018

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. There's still no doc comment on the protocol itself, but writing good doc comments would take time :) I added TODO's for that.

///
/// - Note: Zero is the identity element for addition; for any value,
/// `x + .zero == x` and `.zero + x == x`.
static var zero: Self { get }

/// Adds two values and produces their sum.
///
/// The addition operator (`+`) calculates the sum of its two arguments. For
/// example:
///
/// 1 + 2 // 3
/// -10 + 15 // 5
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Examples involving the unary negative operator should probably be removed from the documentation for this protocol.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, but it was originally on Numeric, and Numeric did not have - either. I'd like for this PR to focus on the implementation (to unblock @stephentyrone's SIMD PR, for example), and write a separate patch to refine documentation.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Right. @xwu, can you open a SR bug for this (since we would want to consider it on Numeric anyway) and CC me and @natecook1000?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

/// -15 + -5 // -20
/// 21.5 + 3.25 // 24.75
///
/// You cannot use `+` with arguments of different types. To add values of
/// different types, convert one of the values to the other value's type.
///
/// let x: Int8 = 21
/// let y: Int = 1000000
/// Int(x) + y // 1000021
///
/// - Parameters:
/// - lhs: The first value to add.
/// - rhs: The second value to add.
static func +(lhs: Self, rhs: Self) -> Self

/// Adds two values and stores the result in the left-hand-side variable.
///
/// - Parameters:
/// - lhs: The first value to add.
/// - rhs: The second value to add.
static func +=(lhs: inout Self, rhs: Self)

/// Subtracts one value from another and produces their difference.
///
/// The subtraction operator (`-`) calculates the difference of its two
/// arguments. For example:
///
/// 8 - 3 // 5
/// -10 - 5 // -15
/// 100 - -5 // 105
/// 10.5 - 100.0 // -89.5
///
/// You cannot use `-` with arguments of different types. To subtract values
/// of different types, convert one of the values to the other value's type.
///
/// let x: UInt8 = 21
/// let y: UInt = 1000000
/// y - UInt(x) // 999979
///
/// - Parameters:
/// - lhs: A numeric value.
/// - rhs: The value to subtract from `lhs`.
static func -(lhs: Self, rhs: Self) -> Self

/// Subtracts the second value from the first and stores the difference in the
/// left-hand-side variable.
///
/// - Parameters:
/// - lhs: A numeric value.
/// - rhs: The value to subtract from `lhs`.
static func -=(lhs: inout Self, rhs: Self)
}

public extension AdditiveArithmetic where Self : ExpressibleByIntegerLiteral {
static var zero: Self {
return 0
}
}

//===----------------------------------------------------------------------===//
//===--- Numeric ----------------------------------------------------------===//
//===----------------------------------------------------------------------===//

// FIXME: Update comment based on the `AdditiveArithmetic` change.
/// Declares methods backing binary arithmetic operators--such as `+`, `-` and
/// `*`--and their mutating counterparts.
///
Expand Down Expand Up @@ -61,7 +140,7 @@ extension ExpressibleByIntegerLiteral
/// the required mutating methods. Extensions to `Numeric` provide default
/// implementations for the protocol's nonmutating methods based on the
/// mutating variants.
public protocol Numeric : Equatable, ExpressibleByIntegerLiteral {
public protocol Numeric : AdditiveArithmetic, ExpressibleByIntegerLiteral {
/// Creates a new instance from the given integer, if it can be represented
/// exactly.
///
Expand Down Expand Up @@ -100,65 +179,6 @@ public protocol Numeric : Equatable, ExpressibleByIntegerLiteral {
/// instead of the `magnitude` property is encouraged.
var magnitude: Magnitude { get }

/// Adds two values and produces their sum.
///
/// The addition operator (`+`) calculates the sum of its two arguments. For
/// example:
///
/// 1 + 2 // 3
/// -10 + 15 // 5
/// -15 + -5 // -20
/// 21.5 + 3.25 // 24.75
///
/// You cannot use `+` with arguments of different types. To add values of
/// different types, convert one of the values to the other value's type.
///
/// let x: Int8 = 21
/// let y: Int = 1000000
/// Int(x) + y // 1000021
///
/// - Parameters:
/// - lhs: The first value to add.
/// - rhs: The second value to add.
static func +(lhs: Self, rhs: Self) -> Self

/// Adds two values and stores the result in the left-hand-side variable.
///
/// - Parameters:
/// - lhs: The first value to add.
/// - rhs: The second value to add.
static func +=(lhs: inout Self, rhs: Self)

/// Subtracts one value from another and produces their difference.
///
/// The subtraction operator (`-`) calculates the difference of its two
/// arguments. For example:
///
/// 8 - 3 // 5
/// -10 - 5 // -15
/// 100 - -5 // 105
/// 10.5 - 100.0 // -89.5
///
/// You cannot use `-` with arguments of different types. To subtract values
/// of different types, convert one of the values to the other value's type.
///
/// let x: UInt8 = 21
/// let y: UInt = 1000000
/// y - UInt(x) // 999979
///
/// - Parameters:
/// - lhs: A numeric value.
/// - rhs: The value to subtract from `lhs`.
static func -(lhs: Self, rhs: Self) -> Self

/// Subtracts the second value from the first and stores the difference in the
/// left-hand-side variable.
///
/// - Parameters:
/// - lhs: A numeric value.
/// - rhs: The value to subtract from `lhs`.
static func -=(lhs: inout Self, rhs: Self)

/// Multiplies two values and produces their product.
///
/// The multiplication operator (`*`) calculates the product of its two
Expand Down Expand Up @@ -327,7 +347,7 @@ public func abs<T : SignedNumeric & Comparable>(_ x: T) -> T {
return x < (0 as T) ? -x : x
}

extension Numeric {
extension AdditiveArithmetic {
/// Returns the given number unchanged.
///
/// You can use the unary plus operator (`+`) to provide symmetry in your
Expand Down
10 changes: 5 additions & 5 deletions stdlib/public/core/Policy.swift
Original file line number Diff line number Diff line change
Expand Up @@ -396,7 +396,7 @@ prefix operator ++
prefix operator --
prefix operator !
prefix operator ~ : BinaryInteger
prefix operator + : Numeric
prefix operator + : AdditiveArithmetic
prefix operator - : SignedNumeric
prefix operator ...
prefix operator ..<
Expand All @@ -420,9 +420,9 @@ infix operator & : MultiplicationPrecedence, BinaryInteger

// "Additive"

infix operator + : AdditionPrecedence, Numeric, String, Strideable
infix operator + : AdditionPrecedence, AdditiveArithmetic, String, Strideable
infix operator &+ : AdditionPrecedence, FixedWidthInteger
infix operator - : AdditionPrecedence, Numeric, Strideable
infix operator - : AdditionPrecedence, AdditiveArithmetic, Strideable
infix operator &- : AdditionPrecedence, FixedWidthInteger
infix operator | : AdditionPrecedence, BinaryInteger
infix operator ^ : AdditionPrecedence, BinaryInteger
Expand Down Expand Up @@ -474,9 +474,9 @@ infix operator *= : AssignmentPrecedence, Numeric
infix operator &*= : AssignmentPrecedence, FixedWidthInteger
infix operator /= : AssignmentPrecedence, BinaryInteger
infix operator %= : AssignmentPrecedence, BinaryInteger
infix operator += : AssignmentPrecedence, Numeric, String, Strideable
infix operator += : AssignmentPrecedence, AdditiveArithmetic, String, Strideable
infix operator &+= : AssignmentPrecedence, FixedWidthInteger
infix operator -= : AssignmentPrecedence, Numeric, Strideable
infix operator -= : AssignmentPrecedence, AdditiveArithmetic, Strideable
infix operator &-= : AssignmentPrecedence, FixedWidthInteger
infix operator <<= : AssignmentPrecedence, BinaryInteger
infix operator &<<= : AssignmentPrecedence, FixedWidthInteger
Expand Down
5 changes: 5 additions & 0 deletions test/api-digester/Outputs/cake-abi.json
Original file line number Diff line number Diff line change
Expand Up @@ -1342,6 +1342,11 @@
}
]
},
{
"kind": "Conformance",
"name": "AdditiveArithmetic",
"printedName": "AdditiveArithmetic"
},
{
"kind": "Conformance",
"name": "ExpressibleByIntegerLiteral",
Expand Down
5 changes: 5 additions & 0 deletions test/api-digester/Outputs/cake.json
Original file line number Diff line number Diff line change
Expand Up @@ -1236,6 +1236,11 @@
}
]
},
{
"kind": "Conformance",
"name": "AdditiveArithmetic",
"printedName": "AdditiveArithmetic"
},
{
"kind": "Conformance",
"name": "ExpressibleByIntegerLiteral",
Expand Down
17 changes: 17 additions & 0 deletions test/api-digester/Outputs/stability-stdlib-abi.swift.expected
Original file line number Diff line number Diff line change
@@ -1,11 +1,17 @@
Func MutableCollection._partition(within:by:) has generic signature change from <τ_0_0 where τ_0_0 : MutableCollection, τ_0_0 : RandomAccessCollection> to <τ_0_0 where τ_0_0 : BidirectionalCollection, τ_0_0 : MutableCollection>
Protocol Numeric has generic signature change from <τ_0_0 : Equatable, τ_0_0 : ExpressibleByIntegerLiteral, τ_0_0.Magnitude : Comparable, τ_0_0.Magnitude : Numeric> to <τ_0_0 : AdditiveArithmetic, τ_0_0 : ExpressibleByIntegerLiteral, τ_0_0.Magnitude : Comparable, τ_0_0.Magnitude : Numeric>
Func MutableCollection._partition(within:by:) has been renamed to Func MutableCollection._partitionImpl(by:)
Func MutableCollection._heapSort(within:by:) has been removed
Func MutableCollection._heapify(within:by:) has been removed
Func MutableCollection._introSort(within:by:) has been removed
Func MutableCollection._introSortImpl(within:by:depthLimit:) has been removed
Func MutableCollection._siftDown(_:within:by:) has been removed
Func MutableCollection._sort3(_:_:_:by:) has been removed
Func Numeric.+(_:) has been removed
Func Numeric.+(_:_:) has been removed
Func Numeric.+=(_:_:) has been removed
Func Numeric.-(_:_:) has been removed
Func Numeric.-=(_:_:) has been removed
Func MutableCollection._partition(within:by:) has parameter 0 type change from Range<τ_0_0.Index> to (τ_0_0.Element) throws -> Bool

Func BinaryInteger._description(radix:uppercase:) has been removed
Expand Down Expand Up @@ -60,4 +66,15 @@ Struct _HasherTailBuffer has been removed
Var Hasher._core has declared type change from _BufferingHasher<Hasher._Core> to Hasher._Core
Var Hasher._Core._buffer is added to a non-resilient type
Var Hasher._Core._state in a non-resilient type changes position from 0 to 1
Protocol BinaryFloatingPoint has added inherited protocol AdditiveArithmetic
Protocol BinaryInteger has added inherited protocol AdditiveArithmetic
Protocol FixedWidthInteger has added inherited protocol AdditiveArithmetic
Protocol FloatingPoint has added inherited protocol AdditiveArithmetic
Protocol Numeric has added inherited protocol AdditiveArithmetic
Protocol SignedInteger has added inherited protocol AdditiveArithmetic
Protocol SignedNumeric has added inherited protocol AdditiveArithmetic
Protocol UnsignedInteger has added inherited protocol AdditiveArithmetic
Struct Hasher._Core has removed conformance to _HasherCore

/* Decl Attribute changes */

14 changes: 14 additions & 0 deletions test/api-digester/Outputs/stability-stdlib-source.swift.expected
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@

/* Generic Signature Changes */
Protocol Numeric has generic signature change from <Self : Equatable, Self : ExpressibleByIntegerLiteral, Self.Magnitude : Comparable, Self.Magnitude : Numeric> to <Self : AdditiveArithmetic, Self : ExpressibleByIntegerLiteral, Self.Magnitude : Comparable, Self.Magnitude : Numeric>
Protocol StringProtocol has generic signature change from <Self : BidirectionalCollection, Self : Comparable, Self : ExpressibleByStringLiteral, Self : Hashable, Self : LosslessStringConvertible, Self : TextOutputStream, Self : TextOutputStreamable, Self.Element == Character, Self.Index == String.Index, Self.SubSequence : StringProtocol, Self.UTF16View : BidirectionalCollection, Self.UTF8View : Collection, Self.UnicodeScalarView : BidirectionalCollection, Self.UTF16View.Element == UInt16, Self.UTF16View.Index == String.Index, Self.UTF8View.Element == UInt8, Self.UTF8View.Index == String.Index, Self.UnicodeScalarView.Element == Unicode.Scalar, Self.UnicodeScalarView.Index == String.Index, Self.SubSequence.UTF16View.Index == String.Index, Self.SubSequence.UTF8View.Index == String.Index, Self.SubSequence.UnicodeScalarView.Index == String.Index> to <Self : BidirectionalCollection, Self : Comparable, Self : ExpressibleByStringInterpolation, Self : Hashable, Self : LosslessStringConvertible, Self : TextOutputStream, Self : TextOutputStreamable, Self.Element == Character, Self.Index == String.Index, Self.StringInterpolation == DefaultStringInterpolation, Self.SubSequence : StringProtocol, Self.UTF16View : BidirectionalCollection, Self.UTF8View : Collection, Self.UnicodeScalarView : BidirectionalCollection, Self.UTF16View.Element == UInt16, Self.UTF16View.Index == String.Index, Self.UTF8View.Element == UInt8, Self.UTF8View.Index == String.Index, Self.UnicodeScalarView.Element == Unicode.Scalar, Self.UnicodeScalarView.Index == String.Index, Self.SubSequence.UTF16View.Index == String.Index, Self.SubSequence.UTF8View.Index == String.Index, Self.SubSequence.UnicodeScalarView.Index == String.Index>

/* RawRepresentable Changes */
Expand All @@ -9,6 +10,11 @@ Constructor String.init(stringInterpolationSegment:) has been removed
Func Collection.prefix(through:) has been removed
Func Collection.prefix(upTo:) has been removed
Func Collection.suffix(from:) has been removed
Func Numeric.+(_:) has been removed
Func Numeric.+(_:_:) has been removed
Func Numeric.+=(_:_:) has been removed
Func Numeric.-(_:_:) has been removed
Func Numeric.-=(_:_:) has been removed
Func Sequence.filter(_:) has been removed
Func Sequence.forEach(_:) has been removed
Func Sequence.map(_:) has been removed
Expand Down Expand Up @@ -36,8 +42,16 @@ Var Set.first has been removed

/* Type Changes */
Constructor String.init(stringInterpolation:) has parameter 0 type change from [String] to DefaultStringInterpolation
Protocol BinaryFloatingPoint has added inherited protocol AdditiveArithmetic
Protocol BinaryInteger has added inherited protocol AdditiveArithmetic
Protocol FixedWidthInteger has added inherited protocol AdditiveArithmetic
Protocol FloatingPoint has added inherited protocol AdditiveArithmetic
Protocol Numeric has added inherited protocol AdditiveArithmetic
Protocol SignedInteger has added inherited protocol AdditiveArithmetic
Protocol SignedNumeric has added inherited protocol AdditiveArithmetic

/* Decl Attribute changes */

/* Protocol Requirement Changes */
Protocol StringProtocol has added inherited protocol ExpressibleByStringInterpolation
Protocol UnsignedInteger has added inherited protocol AdditiveArithmetic