Coverage Report

Created: 2026-03-13 03:47

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
be/src/exec/common/endian.h
Line
Count
Source
1
// Licensed to the Apache Software Foundation (ASF) under one
2
// or more contributor license agreements.  See the NOTICE file
3
// distributed with this work for additional information
4
// regarding copyright ownership.  The ASF licenses this file
5
// to you under the Apache License, Version 2.0 (the
6
// "License"); you may not use this file except in compliance
7
// with the License.  You may obtain a copy of the License at
8
//
9
//   http://www.apache.org/licenses/LICENSE-2.0
10
//
11
// Unless required by applicable law or agreed to in writing,
12
// software distributed under the License is distributed on an
13
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14
// KIND, either express or implied.  See the License for the
15
// specific language governing permissions and limitations
16
// under the License.
17
18
#pragma once
19
20
#include "core/extended_types.h"
21
#include "core/uint24.h"
22
#include "util/unaligned.h"
23
24
namespace doris {
25
#include "common/compile_check_begin.h"
26
8.09k
inline uint64_t gbswap_64(uint64_t host_int) {
27
8.09k
#if defined(__GNUC__) && defined(__x86_64__) && !defined(__APPLE__)
28
    // Adapted from /usr/include/byteswap.h.  Not available on Mac.
29
8.09k
    if (__builtin_constant_p(host_int)) {
30
0
        return __bswap_constant_64(host_int);
31
8.09k
    } else {
32
8.09k
        uint64_t result;
33
8.09k
        __asm__("bswap %0" : "=r"(result) : "0"(host_int));
34
8.09k
        return result;
35
8.09k
    }
36
#elif defined(bswap_64)
37
    return bswap_64(host_int);
38
#else
39
    return static_cast<uint64_t>(bswap_32(static_cast<uint32_t>(host_int >> 32))) |
40
           (static_cast<uint64_t>(bswap_32(static_cast<uint32_t>(host_int))) << 32);
41
#endif // bswap_64
42
8.09k
}
43
44
7.38M
inline unsigned __int128 gbswap_128(unsigned __int128 host_int) {
45
7.38M
    return static_cast<unsigned __int128>(bswap_64(static_cast<uint64_t>(host_int >> 64))) |
46
7.38M
           (static_cast<unsigned __int128>(bswap_64(static_cast<uint64_t>(host_int))) << 64);
47
7.38M
}
48
49
2.02k
inline wide::UInt256 gbswap_256(wide::UInt256 host_int) {
50
2.02k
    wide::UInt256 result {gbswap_64(host_int.items[3]), gbswap_64(host_int.items[2]),
51
2.02k
                          gbswap_64(host_int.items[1]), gbswap_64(host_int.items[0])};
52
2.02k
    return result;
53
2.02k
}
54
55
// Swap bytes of a 24-bit value.
56
401k
inline uint32_t bswap_24(uint32_t x) {
57
401k
    return uint32_t((x & 0x0000ffULL) << 16) | ((x & 0x00ff00ULL)) | ((x & 0xff0000ULL) >> 16);
58
401k
}
59
60
// use std::byteswap after doris enable cpp23
61
template <typename T>
62
49.6M
T byte_swap(T x) {
63
49.6M
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
2.02k
        return gbswap_256(x);
65
7.38M
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
7.38M
        return gbswap_128(x);
67
16.0M
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
16.0M
        return bswap_64(x);
69
23.8M
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
23.8M
        return bswap_32(x);
71
23.8M
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
402k
        return bswap_24(x);
73
614k
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
614k
        return bswap_16(x);
75
1.35M
    } else {
76
1.35M
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
1.35M
        return x; // No byte swap needed for unsupported types
78
1.35M
    }
79
49.6M
}
_ZN5doris9byte_swapItEET_S1_
Line
Count
Source
62
614k
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
614k
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
614k
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
614k
}
_ZN5doris9byte_swapIjEET_S1_
Line
Count
Source
62
23.8M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
23.8M
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
23.8M
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
23.8M
}
_ZN5doris9byte_swapImEET_S1_
Line
Count
Source
62
15.4M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
15.4M
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
15.4M
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
15.4M
}
_ZN5doris9byte_swapINS_8uint24_tEEET_S2_
Line
Count
Source
62
402k
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
402k
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
402k
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
402k
}
_ZN5doris9byte_swapIlEET_S1_
Line
Count
Source
62
533k
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
533k
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
533k
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
533k
}
_ZN5doris9byte_swapInEET_S1_
Line
Count
Source
62
5.17M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
5.17M
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
5.17M
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
5.17M
}
Unexecuted instantiation: _ZN5doris9byte_swapIN4wide7integerILm256EiEEEET_S4_
Unexecuted instantiation: _ZN5doris9byte_swapIiEET_S1_
_ZN5doris9byte_swapIhEET_S1_
Line
Count
Source
62
1.35M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
1.35M
    } else {
76
1.35M
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
1.35M
        return x; // No byte swap needed for unsupported types
78
1.35M
    }
79
1.35M
}
_ZN5doris9byte_swapIoEET_S1_
Line
Count
Source
62
2.20M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
2.20M
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
2.20M
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
2.20M
}
_ZN5doris9byte_swapIN4wide7integerILm256EjEEEET_S4_
Line
Count
Source
62
2.02k
T byte_swap(T x) {
63
2.02k
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
2.02k
        return gbswap_256(x);
65
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
        return gbswap_128(x);
67
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
        return bswap_64(x);
69
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
        return bswap_32(x);
71
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
        return bswap_24(x);
73
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
        return bswap_16(x);
75
    } else {
76
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
        return x; // No byte swap needed for unsupported types
78
    }
79
2.02k
}
80
81
template <std::endian target, typename T>
82
271M
T to_endian(T value) {
83
271M
    if constexpr (std::endian::native == target) {
84
222M
        return value; // No swap needed
85
222M
    } else {
86
49.6M
        static_assert(std::endian::native == std::endian::big ||
87
49.6M
                              std::endian::native == std::endian::little,
88
49.6M
                      "Unsupported endianness");
89
49.6M
        return byte_swap(value);
90
49.6M
    }
91
271M
}
_ZN5doris9to_endianILSt6endian1234EtEET0_S2_
Line
Count
Source
82
208k
T to_endian(T value) {
83
208k
    if constexpr (std::endian::native == target) {
84
208k
        return value; // No swap needed
85
    } else {
86
        static_assert(std::endian::native == std::endian::big ||
87
                              std::endian::native == std::endian::little,
88
                      "Unsupported endianness");
89
        return byte_swap(value);
90
    }
91
208k
}
_ZN5doris9to_endianILSt6endian1234EjEET0_S2_
Line
Count
Source
82
218M
T to_endian(T value) {
83
218M
    if constexpr (std::endian::native == target) {
84
218M
        return value; // No swap needed
85
    } else {
86
        static_assert(std::endian::native == std::endian::big ||
87
                              std::endian::native == std::endian::little,
88
                      "Unsupported endianness");
89
        return byte_swap(value);
90
    }
91
218M
}
_ZN5doris9to_endianILSt6endian1234EmEET0_S2_
Line
Count
Source
82
3.66M
T to_endian(T value) {
83
3.66M
    if constexpr (std::endian::native == target) {
84
3.66M
        return value; // No swap needed
85
    } else {
86
        static_assert(std::endian::native == std::endian::big ||
87
                              std::endian::native == std::endian::little,
88
                      "Unsupported endianness");
89
        return byte_swap(value);
90
    }
91
3.66M
}
_ZN5doris9to_endianILSt6endian4321EtEET0_S2_
Line
Count
Source
82
614k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
614k
    } else {
86
614k
        static_assert(std::endian::native == std::endian::big ||
87
614k
                              std::endian::native == std::endian::little,
88
614k
                      "Unsupported endianness");
89
614k
        return byte_swap(value);
90
614k
    }
91
614k
}
_ZN5doris9to_endianILSt6endian4321EjEET0_S2_
Line
Count
Source
82
23.8M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
23.8M
    } else {
86
23.8M
        static_assert(std::endian::native == std::endian::big ||
87
23.8M
                              std::endian::native == std::endian::little,
88
23.8M
                      "Unsupported endianness");
89
23.8M
        return byte_swap(value);
90
23.8M
    }
91
23.8M
}
_ZN5doris9to_endianILSt6endian4321EmEET0_S2_
Line
Count
Source
82
15.4M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
15.4M
    } else {
86
15.4M
        static_assert(std::endian::native == std::endian::big ||
87
15.4M
                              std::endian::native == std::endian::little,
88
15.4M
                      "Unsupported endianness");
89
15.4M
        return byte_swap(value);
90
15.4M
    }
91
15.4M
}
_ZN5doris9to_endianILSt6endian1234EoEET0_S2_
Line
Count
Source
82
48.8k
T to_endian(T value) {
83
48.8k
    if constexpr (std::endian::native == target) {
84
48.8k
        return value; // No swap needed
85
    } else {
86
        static_assert(std::endian::native == std::endian::big ||
87
                              std::endian::native == std::endian::little,
88
                      "Unsupported endianness");
89
        return byte_swap(value);
90
    }
91
48.8k
}
_ZN5doris9to_endianILSt6endian4321ENS_8uint24_tEEET0_S3_
Line
Count
Source
82
402k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
402k
    } else {
86
402k
        static_assert(std::endian::native == std::endian::big ||
87
402k
                              std::endian::native == std::endian::little,
88
402k
                      "Unsupported endianness");
89
402k
        return byte_swap(value);
90
402k
    }
91
402k
}
_ZN5doris9to_endianILSt6endian4321ElEET0_S2_
Line
Count
Source
82
533k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
533k
    } else {
86
533k
        static_assert(std::endian::native == std::endian::big ||
87
533k
                              std::endian::native == std::endian::little,
88
533k
                      "Unsupported endianness");
89
533k
        return byte_swap(value);
90
533k
    }
91
533k
}
_ZN5doris9to_endianILSt6endian4321EnEET0_S2_
Line
Count
Source
82
5.17M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
5.17M
    } else {
86
5.17M
        static_assert(std::endian::native == std::endian::big ||
87
5.17M
                              std::endian::native == std::endian::little,
88
5.17M
                      "Unsupported endianness");
89
5.17M
        return byte_swap(value);
90
5.17M
    }
91
5.17M
}
Unexecuted instantiation: _ZN5doris9to_endianILSt6endian4321EN4wide7integerILm256EiEEEET0_S5_
Unexecuted instantiation: _ZN5doris9to_endianILSt6endian4321EiEET0_S2_
_ZN5doris9to_endianILSt6endian4321EhEET0_S2_
Line
Count
Source
82
1.35M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
1.35M
    } else {
86
1.35M
        static_assert(std::endian::native == std::endian::big ||
87
1.35M
                              std::endian::native == std::endian::little,
88
1.35M
                      "Unsupported endianness");
89
1.35M
        return byte_swap(value);
90
1.35M
    }
91
1.35M
}
_ZN5doris9to_endianILSt6endian4321EoEET0_S2_
Line
Count
Source
82
2.21M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
2.21M
    } else {
86
2.21M
        static_assert(std::endian::native == std::endian::big ||
87
2.21M
                              std::endian::native == std::endian::little,
88
2.21M
                      "Unsupported endianness");
89
2.21M
        return byte_swap(value);
90
2.21M
    }
91
2.21M
}
_ZN5doris9to_endianILSt6endian4321EN4wide7integerILm256EjEEEET0_S5_
Line
Count
Source
82
2.02k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
2.02k
    } else {
86
2.02k
        static_assert(std::endian::native == std::endian::big ||
87
2.02k
                              std::endian::native == std::endian::little,
88
2.02k
                      "Unsupported endianness");
89
2.02k
        return byte_swap(value);
90
2.02k
    }
91
2.02k
}
92
93
// Utilities to convert numbers between the current hosts's native byte
94
// order and little-endian byte order
95
//
96
// Load/Store methods are alignment safe
97
class LittleEndian {
98
public:
99
    // Functions to do unaligned loads and stores in little-endian order.
100
0
    static uint16_t Load16(const void* p) {
101
0
        return to_endian<std::endian::little>(unaligned_load<uint16_t>(p));
102
0
    }
103
104
0
    static void Store16(void* p, uint16_t v) {
105
0
        unaligned_store<uint16_t>(p, to_endian<std::endian::little>(v));
106
0
    }
107
108
16.9M
    static uint32_t Load32(const void* p) {
109
16.9M
        return to_endian<std::endian::little>(unaligned_load<uint32_t>(p));
110
16.9M
    }
111
112
0
    static void Store32(void* p, uint32_t v) {
113
0
        unaligned_store<uint32_t>(p, to_endian<std::endian::little>(v));
114
0
    }
115
116
3.06M
    static uint64_t Load64(const void* p) {
117
3.06M
        return to_endian<std::endian::little>(unaligned_load<uint64_t>(p));
118
3.06M
    }
119
120
0
    static void Store64(void* p, uint64_t v) {
121
0
        unaligned_store<uint64_t>(p, to_endian<std::endian::little>(v));
122
0
    }
123
};
124
125
// Utilities to convert numbers between the current hosts's native byte
126
// order and big-endian byte order (same as network byte order)
127
//
128
// Load/Store methods are alignment safe
129
class BigEndian {
130
public:
131
    // Functions to do unaligned loads and stores in little-endian order.
132
0
    static uint16_t Load16(const void* p) {
133
0
        return to_endian<std::endian::big>(unaligned_load<uint16_t>(p));
134
0
    }
135
136
0
    static void Store16(void* p, uint16_t v) {
137
0
        unaligned_store<uint16_t>(p, to_endian<std::endian::big>(v));
138
0
    }
139
140
668
    static uint32_t Load32(const void* p) {
141
668
        return to_endian<std::endian::big>(unaligned_load<uint32_t>(p));
142
668
    }
143
144
132
    static void Store32(void* p, uint32_t v) {
145
132
        unaligned_store<uint32_t>(p, to_endian<std::endian::big>(v));
146
132
    }
147
148
0
    static uint64_t Load64(const void* p) {
149
0
        return to_endian<std::endian::big>(unaligned_load<uint64_t>(p));
150
0
    }
151
152
0
    static void Store64(void* p, uint64_t v) {
153
0
        unaligned_store<uint64_t>(p, to_endian<std::endian::big>(v));
154
0
    }
155
}; // BigEndian
156
#include "common/compile_check_end.h"
157
} // namespace doris