Coverage Report

Created: 2026-03-15 15:59

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.13k
inline uint64_t gbswap_64(uint64_t host_int) {
27
8.13k
#if defined(__GNUC__) && defined(__x86_64__) && !defined(__APPLE__)
28
    // Adapted from /usr/include/byteswap.h.  Not available on Mac.
29
8.13k
    if (__builtin_constant_p(host_int)) {
30
0
        return __bswap_constant_64(host_int);
31
8.13k
    } else {
32
8.13k
        uint64_t result;
33
8.13k
        __asm__("bswap %0" : "=r"(result) : "0"(host_int));
34
8.13k
        return result;
35
8.13k
    }
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.13k
}
43
44
7.09M
inline unsigned __int128 gbswap_128(unsigned __int128 host_int) {
45
7.09M
    return static_cast<unsigned __int128>(bswap_64(static_cast<uint64_t>(host_int >> 64))) |
46
7.09M
           (static_cast<unsigned __int128>(bswap_64(static_cast<uint64_t>(host_int))) << 64);
47
7.09M
}
48
49
2.03k
inline wide::UInt256 gbswap_256(wide::UInt256 host_int) {
50
2.03k
    wide::UInt256 result {gbswap_64(host_int.items[3]), gbswap_64(host_int.items[2]),
51
2.03k
                          gbswap_64(host_int.items[1]), gbswap_64(host_int.items[0])};
52
2.03k
    return result;
53
2.03k
}
54
55
// Swap bytes of a 24-bit value.
56
400k
inline uint32_t bswap_24(uint32_t x) {
57
400k
    return uint32_t((x & 0x0000ffULL) << 16) | ((x & 0x00ff00ULL)) | ((x & 0xff0000ULL) >> 16);
58
400k
}
59
60
// use std::byteswap after doris enable cpp23
61
template <typename T>
62
51.4M
T byte_swap(T x) {
63
51.4M
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
2.03k
        return gbswap_256(x);
65
7.09M
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
7.09M
        return gbswap_128(x);
67
17.7M
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
17.7M
        return bswap_64(x);
69
24.3M
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
24.3M
        return bswap_32(x);
71
24.3M
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
401k
        return bswap_24(x);
73
619k
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
619k
        return bswap_16(x);
75
1.15M
    } else {
76
1.15M
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
1.15M
        return x; // No byte swap needed for unsupported types
78
1.15M
    }
79
51.4M
}
_ZN5doris9byte_swapItEET_S1_
Line
Count
Source
62
619k
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
619k
    } else if constexpr (sizeof(T) == sizeof(int16_t)) {
74
619k
        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
619k
}
_ZN5doris9byte_swapIjEET_S1_
Line
Count
Source
62
24.3M
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
24.3M
    } else if constexpr (sizeof(T) == sizeof(int32_t)) {
70
24.3M
        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
24.3M
}
_ZN5doris9byte_swapImEET_S1_
Line
Count
Source
62
17.2M
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
17.2M
    } else if constexpr (sizeof(T) == sizeof(int64_t)) {
68
17.2M
        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
17.2M
}
_ZN5doris9byte_swapINS_8uint24_tEEET_S2_
Line
Count
Source
62
401k
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
401k
    } else if constexpr (sizeof(T) == sizeof(doris::uint24_t)) {
72
401k
        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
401k
}
_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.15M
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.15M
    } else {
76
1.15M
        static_assert(sizeof(T) == 1, "Unsupported type size for byte_swap");
77
1.15M
        return x; // No byte swap needed for unsupported types
78
1.15M
    }
79
1.15M
}
_ZN5doris9byte_swapIoEET_S1_
Line
Count
Source
62
1.92M
T byte_swap(T x) {
63
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
        return gbswap_256(x);
65
1.92M
    } else if constexpr (sizeof(T) == sizeof(__int128)) {
66
1.92M
        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
1.92M
}
_ZN5doris9byte_swapIN4wide7integerILm256EjEEEET_S4_
Line
Count
Source
62
2.03k
T byte_swap(T x) {
63
2.03k
    if constexpr (sizeof(T) == sizeof(wide::Int256)) {
64
2.03k
        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.03k
}
80
81
template <std::endian target, typename T>
82
397M
T to_endian(T value) {
83
397M
    if constexpr (std::endian::native == target) {
84
345M
        return value; // No swap needed
85
345M
    } else {
86
51.4M
        static_assert(std::endian::native == std::endian::big ||
87
51.4M
                              std::endian::native == std::endian::little,
88
51.4M
                      "Unsupported endianness");
89
51.4M
        return byte_swap(value);
90
51.4M
    }
91
397M
}
_ZN5doris9to_endianILSt6endian1234EtEET0_S2_
Line
Count
Source
82
205k
T to_endian(T value) {
83
205k
    if constexpr (std::endian::native == target) {
84
205k
        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
205k
}
_ZN5doris9to_endianILSt6endian1234EjEET0_S2_
Line
Count
Source
82
342M
T to_endian(T value) {
83
342M
    if constexpr (std::endian::native == target) {
84
342M
        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
342M
}
_ZN5doris9to_endianILSt6endian1234EmEET0_S2_
Line
Count
Source
82
3.32M
T to_endian(T value) {
83
3.32M
    if constexpr (std::endian::native == target) {
84
3.32M
        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.32M
}
_ZN5doris9to_endianILSt6endian4321EtEET0_S2_
Line
Count
Source
82
619k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
619k
    } else {
86
619k
        static_assert(std::endian::native == std::endian::big ||
87
619k
                              std::endian::native == std::endian::little,
88
619k
                      "Unsupported endianness");
89
619k
        return byte_swap(value);
90
619k
    }
91
619k
}
_ZN5doris9to_endianILSt6endian4321EjEET0_S2_
Line
Count
Source
82
24.3M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
24.3M
    } else {
86
24.3M
        static_assert(std::endian::native == std::endian::big ||
87
24.3M
                              std::endian::native == std::endian::little,
88
24.3M
                      "Unsupported endianness");
89
24.3M
        return byte_swap(value);
90
24.3M
    }
91
24.3M
}
_ZN5doris9to_endianILSt6endian4321EmEET0_S2_
Line
Count
Source
82
17.2M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
17.2M
    } else {
86
17.2M
        static_assert(std::endian::native == std::endian::big ||
87
17.2M
                              std::endian::native == std::endian::little,
88
17.2M
                      "Unsupported endianness");
89
17.2M
        return byte_swap(value);
90
17.2M
    }
91
17.2M
}
_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
401k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
401k
    } else {
86
401k
        static_assert(std::endian::native == std::endian::big ||
87
401k
                              std::endian::native == std::endian::little,
88
401k
                      "Unsupported endianness");
89
401k
        return byte_swap(value);
90
401k
    }
91
401k
}
_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.15M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
1.15M
    } else {
86
1.15M
        static_assert(std::endian::native == std::endian::big ||
87
1.15M
                              std::endian::native == std::endian::little,
88
1.15M
                      "Unsupported endianness");
89
1.15M
        return byte_swap(value);
90
1.15M
    }
91
1.15M
}
_ZN5doris9to_endianILSt6endian4321EoEET0_S2_
Line
Count
Source
82
1.92M
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
1.92M
    } else {
86
1.92M
        static_assert(std::endian::native == std::endian::big ||
87
1.92M
                              std::endian::native == std::endian::little,
88
1.92M
                      "Unsupported endianness");
89
1.92M
        return byte_swap(value);
90
1.92M
    }
91
1.92M
}
_ZN5doris9to_endianILSt6endian4321EN4wide7integerILm256EjEEEET0_S5_
Line
Count
Source
82
2.03k
T to_endian(T value) {
83
    if constexpr (std::endian::native == target) {
84
        return value; // No swap needed
85
2.03k
    } else {
86
2.03k
        static_assert(std::endian::native == std::endian::big ||
87
2.03k
                              std::endian::native == std::endian::little,
88
2.03k
                      "Unsupported endianness");
89
2.03k
        return byte_swap(value);
90
2.03k
    }
91
2.03k
}
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
18.1M
    static uint32_t Load32(const void* p) {
109
18.1M
        return to_endian<std::endian::little>(unaligned_load<uint32_t>(p));
110
18.1M
    }
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