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
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
|
// Unit test for bit_cast template.
#include "tensorflow/core/lib/core/casts.h"
#include <gtest/gtest.h>
#include "tensorflow/core/platform/logging.h"
namespace tensorflow {
// Marshall and unmarshall.
// ISO spec C++ section 3.9 promises this will work.
template <int N>
struct marshall {
char buf[N];
};
template <class T>
void TestMarshall(const T values[], int num_values) {
for (int i = 0; i < num_values; ++i) {
T t0 = values[i];
marshall<sizeof(T)> m0 = bit_cast<marshall<sizeof(T)> >(t0);
T t1 = bit_cast<T>(m0);
marshall<sizeof(T)> m1 = bit_cast<marshall<sizeof(T)> >(t1);
ASSERT_EQ(0, memcmp(&t0, &t1, sizeof(T)));
ASSERT_EQ(0, memcmp(&m0, &m1, sizeof(T)));
}
}
// Convert back and forth to an integral type. The C++ standard does
// not guarantee this will work.
//
// There are implicit assumptions about sizeof(float) and
// sizeof(double). These assumptions are quite extant everywhere.
template <class T, class I>
void TestIntegral(const T values[], int num_values) {
for (int i = 0; i < num_values; ++i) {
T t0 = values[i];
I i0 = bit_cast<I>(t0);
T t1 = bit_cast<T>(i0);
I i1 = bit_cast<I>(t1);
ASSERT_EQ(0, memcmp(&t0, &t1, sizeof(T)));
ASSERT_EQ(i0, i1);
}
}
TEST(BitCast, Bool) {
LOG(INFO) << "Test bool";
static const bool bool_list[] = {false, true};
TestMarshall<bool>(bool_list, TF_ARRAYSIZE(bool_list));
}
TEST(BitCast, Int32) {
static const int32 int_list[] = {0, 1, 100, 2147483647,
-1, -100, -2147483647, -2147483647 - 1};
TestMarshall<int32>(int_list, TF_ARRAYSIZE(int_list));
}
TEST(BitCast, Int64) {
static const int64 int64_list[] = {0, 1, 1LL << 40, -1, -(1LL << 40)};
TestMarshall<int64>(int64_list, TF_ARRAYSIZE(int64_list));
}
TEST(BitCast, Uint64) {
static const uint64 uint64_list[] = {0, 1, 1LLU << 40, 1LLU << 63};
TestMarshall<uint64>(uint64_list, TF_ARRAYSIZE(uint64_list));
}
TEST(BitCast, Float) {
static const float float_list[] = {0.0, 1.0, -1.0, 10.0, -10.0, 1e10,
1e20, 1e-10, 1e-20, 2.71828, 3.14159};
TestMarshall<float>(float_list, TF_ARRAYSIZE(float_list));
TestIntegral<float, int32>(float_list, TF_ARRAYSIZE(float_list));
TestIntegral<float, uint32>(float_list, TF_ARRAYSIZE(float_list));
}
TEST(BitCast, Double) {
static const double double_list[] = {
0.0,
1.0,
-1.0,
10.0,
-10.0,
1e10,
1e100,
1e-10,
1e-100,
2.718281828459045,
3.141592653589793238462643383279502884197169399375105820974944};
TestMarshall<double>(double_list, TF_ARRAYSIZE(double_list));
TestIntegral<double, int64>(double_list, TF_ARRAYSIZE(double_list));
TestIntegral<double, uint64>(double_list, TF_ARRAYSIZE(double_list));
}
} // namespace tensorflow
|