-
Notifications
You must be signed in to change notification settings - Fork 10.5k
/
Copy pathNumeric.cpp
59 lines (49 loc) · 1.93 KB
/
Numeric.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
//===--- Numeric.cpp - Swift Language ABI numerics support ----------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// Implementations of the numeric-support ABI functions.
//
//===----------------------------------------------------------------------===//
#include "swift/Runtime/Numeric.h"
using namespace swift;
/// Convert an integer literal to the floating-point type T.
template <class T>
static T convert(IntegerLiteral value) {
using SignedChunk = IntegerLiteral::SignedChunk;
using UnsignedChunk = IntegerLiteral::UnsignedChunk;
auto data = value.getData();
assert(!data.empty() && "always require at least one chunk");
// The single-word case is the easiest.
if (data.size() == 1) {
return T(SignedChunk(data[0]));
}
// In two's complement, only the topmost chunk is really signed;
// everything else is added to that as an unsigned value.
static_assert(IntegerLiteral::BitsPerChunk == 32 ||
IntegerLiteral::BitsPerChunk == 64,
"expected either 32-bit or 64-bit chunking");
T chunkFactor = (IntegerLiteral::BitsPerChunk == 32 ? 0x1p32 : 0x1p64);
T result = UnsignedChunk(data[0]);
T scale = chunkFactor;
for (size_t i = 1, e = data.size() - 1; i != e; ++i) {
result += UnsignedChunk(data[i]) * scale;
scale *= chunkFactor;
}
result += SignedChunk(data.back()) * scale;
return result;
}
float swift::swift_intToFloat32(IntegerLiteral value) {
return convert<float>(value);
}
double swift::swift_intToFloat64(IntegerLiteral value) {
return convert<double>(value);
}