Coverage Report

Created: 2026-04-16 09:07

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
be/src/util/rle_encoding.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
#pragma once
18
19
#include <glog/logging.h>
20
21
#include <limits> // IWYU pragma: keep
22
23
#include "common/cast_set.h"
24
#include "util/bit_stream_utils.inline.h"
25
#include "util/bit_util.h"
26
27
namespace doris {
28
29
// Utility classes to do run length encoding (RLE) for fixed bit width values.  If runs
30
// are sufficiently long, RLE is used, otherwise, the values are just bit-packed
31
// (literal encoding).
32
// For both types of runs, there is a byte-aligned indicator which encodes the length
33
// of the run and the type of the run.
34
// This encoding has the benefit that when there aren't any long enough runs, values
35
// are always decoded at fixed (can be precomputed) bit offsets OR both the value and
36
// the run length are byte aligned. This allows for very efficient decoding
37
// implementations.
38
// The encoding is:
39
//    encoded-block := run*
40
//    run := literal-run | repeated-run
41
//    literal-run := literal-indicator < literal bytes >
42
//    repeated-run := repeated-indicator < repeated value. padded to byte boundary >
43
//    literal-indicator := varint_encode( number_of_groups << 1 | 1)
44
//    repeated-indicator := varint_encode( number_of_repetitions << 1 )
45
//
46
// Each run is preceded by a varint. The varint's least significant bit is
47
// used to indicate whether the run is a literal run or a repeated run. The rest
48
// of the varint is used to determine the length of the run (eg how many times the
49
// value repeats).
50
//
51
// In the case of literal runs, the run length is always a multiple of 8 (i.e. encode
52
// in groups of 8), so that no matter the bit-width of the value, the sequence will end
53
// on a byte boundary without padding.
54
// Given that we know it is a multiple of 8, we store the number of 8-groups rather than
55
// the actual number of encoded ints. (This means that the total number of encoded values
56
// can not be determined from the encoded data, since the number of values in the last
57
// group may not be a multiple of 8).
58
// There is a break-even point when it is more storage efficient to do run length
59
// encoding.  For 1 bit-width values, that point is 8 values.  They require 2 bytes
60
// for both the repeated encoding or the literal encoding.  This value can always
61
// be computed based on the bit-width.
62
// TODO: think about how to use this for strings.  The bit packing isn't quite the same.
63
//
64
// Examples with bit-width 1 (eg encoding booleans):
65
// ----------------------------------------
66
// 100 1s followed by 100 0s:
67
// <varint(100 << 1)> <1, padded to 1 byte> <varint(100 << 1)> <0, padded to 1 byte>
68
//  - (total 4 bytes)
69
//
70
// alternating 1s and 0s (200 total):
71
// 200 ints = 25 groups of 8
72
// <varint((25 << 1) | 1)> <25 bytes of values, bitpacked>
73
// (total 26 bytes, 1 byte overhead)
74
//
75
76
// Decoder class for RLE encoded data.
77
//
78
// NOTE: the encoded format does not have any length prefix or any other way of
79
// indicating that the encoded sequence ends at a certain point, so the Decoder
80
// methods may return some extra bits at the end before the read methods start
81
// to return 0/false.
82
template <typename T>
83
class RleDecoder {
84
public:
85
    // Create a decoder object. buffer/buffer_len is the decoded data.
86
    // bit_width is the width of each value (before encoding).
87
    RleDecoder(const uint8_t* buffer, int buffer_len, int bit_width)
88
1.18M
            : bit_reader_(buffer, buffer_len),
89
1.18M
              bit_width_(bit_width),
90
1.18M
              current_value_(0),
91
1.18M
              repeat_count_(0),
92
1.18M
              literal_count_(0),
93
1.18M
              rewind_state_(CANT_REWIND) {
94
1.18M
        DCHECK_GE(bit_width_, 1);
95
1.18M
        DCHECK_LE(bit_width_, 64);
96
1.18M
    }
_ZN5doris10RleDecoderIbEC2EPKhii
Line
Count
Source
88
427k
            : bit_reader_(buffer, buffer_len),
89
427k
              bit_width_(bit_width),
90
427k
              current_value_(0),
91
427k
              repeat_count_(0),
92
427k
              literal_count_(0),
93
427k
              rewind_state_(CANT_REWIND) {
94
427k
        DCHECK_GE(bit_width_, 1);
95
        DCHECK_LE(bit_width_, 64);
96
427k
    }
_ZN5doris10RleDecoderIhEC2EPKhii
Line
Count
Source
88
24.1k
            : bit_reader_(buffer, buffer_len),
89
24.1k
              bit_width_(bit_width),
90
24.1k
              current_value_(0),
91
24.1k
              repeat_count_(0),
92
24.1k
              literal_count_(0),
93
24.1k
              rewind_state_(CANT_REWIND) {
94
24.1k
        DCHECK_GE(bit_width_, 1);
95
        DCHECK_LE(bit_width_, 64);
96
24.1k
    }
_ZN5doris10RleDecoderIsEC2EPKhii
Line
Count
Source
88
731k
            : bit_reader_(buffer, buffer_len),
89
731k
              bit_width_(bit_width),
90
731k
              current_value_(0),
91
731k
              repeat_count_(0),
92
731k
              literal_count_(0),
93
731k
              rewind_state_(CANT_REWIND) {
94
731k
        DCHECK_GE(bit_width_, 1);
95
        DCHECK_LE(bit_width_, 64);
96
731k
    }
97
98
37.5M
    RleDecoder() {}
_ZN5doris10RleDecoderIbEC2Ev
Line
Count
Source
98
37.1M
    RleDecoder() {}
_ZN5doris10RleDecoderIhEC2Ev
Line
Count
Source
98
21.6k
    RleDecoder() {}
_ZN5doris10RleDecoderIsEC2Ev
Line
Count
Source
98
354k
    RleDecoder() {}
99
100
    // Skip n values, and returns the number of non-zero entries skipped.
101
    size_t Skip(size_t to_skip);
102
103
    // Gets the next value.  Returns false if there are no more.
104
    bool Get(T* val);
105
106
    // Seek to the previous value.
107
    void RewindOne();
108
109
    // Gets the next run of the same 'val'. Returns 0 if there is no
110
    // more data to be decoded. Will return a run of at most 'max_run'
111
    // values. If there are more values than this, the next call to
112
    // GetNextRun will return more from the same run.
113
    size_t GetNextRun(T* val, size_t max_run);
114
115
    size_t get_values(T* values, size_t num_values);
116
117
    // Get the count of current repeated value
118
    size_t repeated_count();
119
120
    // Get current repeated value, make sure that count equals repeated_count()
121
    T get_repeated_value(size_t count);
122
123
0
    const BitReader& bit_reader() const { return bit_reader_; }
124
125
private:
126
    bool ReadHeader();
127
128
    enum RewindState { REWIND_LITERAL, REWIND_RUN, CANT_REWIND };
129
130
    BitReader bit_reader_;
131
    int bit_width_;
132
    uint64_t current_value_;
133
    uint32_t repeat_count_;
134
    uint32_t literal_count_;
135
    RewindState rewind_state_;
136
};
137
138
// Class to incrementally build the rle data.
139
// The encoding has two modes: encoding repeated runs and literal runs.
140
// If the run is sufficiently short, it is more efficient to encode as a literal run.
141
// This class does so by buffering 8 values at a time.  If they are not all the same
142
// they are added to the literal run.  If they are the same, they are added to the
143
// repeated run.  When we switch modes, the previous run is flushed out.
144
template <typename T>
145
class RleEncoder {
146
public:
147
    // buffer: buffer to write bits to.
148
    // bit_width: max number of bits for value.
149
    // TODO: consider adding a min_repeated_run_length so the caller can control
150
    // when values should be encoded as repeated runs.  Currently this is derived
151
    // based on the bit_width, which can determine a storage optimal choice.
152
    explicit RleEncoder(faststring* buffer, int bit_width)
153
549k
            : bit_width_(bit_width), bit_writer_(buffer) {
154
549k
        DCHECK_GE(bit_width_, 1);
155
549k
        DCHECK_LE(bit_width_, 64);
156
549k
        Clear();
157
549k
    }
_ZN5doris10RleEncoderIhEC2EPNS_10faststringEi
Line
Count
Source
153
14.7k
            : bit_width_(bit_width), bit_writer_(buffer) {
154
14.7k
        DCHECK_GE(bit_width_, 1);
155
        DCHECK_LE(bit_width_, 64);
156
14.7k
        Clear();
157
14.7k
    }
_ZN5doris10RleEncoderIbEC2EPNS_10faststringEi
Line
Count
Source
153
534k
            : bit_width_(bit_width), bit_writer_(buffer) {
154
534k
        DCHECK_GE(bit_width_, 1);
155
        DCHECK_LE(bit_width_, 64);
156
534k
        Clear();
157
534k
    }
158
159
    // Reserve 'num_bytes' bytes for a plain encoded header, set each
160
    // byte with 'val': this is used for the RLE-encoded data blocks in
161
    // order to be able to able to store the initial ordinal position
162
    // and number of elements. This is a part of RleEncoder in order to
163
    // maintain the correct offset in 'buffer'.
164
    void Reserve(int num_bytes, uint8_t val);
165
166
    // Encode value. This value must be representable with bit_width_ bits.
167
    void Put(T value, size_t run_length = 1);
168
169
    // Flushes any pending values to the underlying buffer.
170
    // Returns the total number of bytes written
171
    int Flush();
172
173
    // Resets all the state in the encoder.
174
    void Clear();
175
176
226k
    int32_t len() const { return bit_writer_.bytes_written(); }
177
178
private:
179
    // Flushes any buffered values.  If this is part of a repeated run, this is largely
180
    // a no-op.
181
    // If it is part of a literal run, this will call FlushLiteralRun, which writes
182
    // out the buffered literal values.
183
    // If 'done' is true, the current run would be written even if it would normally
184
    // have been buffered more.  This should only be called at the end, when the
185
    // encoder has received all values even if it would normally continue to be
186
    // buffered.
187
    void FlushBufferedValues(bool done);
188
189
    // Flushes literal values to the underlying buffer.  If update_indicator_byte,
190
    // then the current literal run is complete and the indicator byte is updated.
191
    void FlushLiteralRun(bool update_indicator_byte);
192
193
    // Flushes a repeated run to the underlying buffer.
194
    void FlushRepeatedRun();
195
196
    // Number of bits needed to encode the value.
197
    const int bit_width_;
198
199
    // Underlying buffer.
200
    BitWriter bit_writer_;
201
202
    // We need to buffer at most 8 values for literals.  This happens when the
203
    // bit_width is 1 (so 8 values fit in one byte).
204
    // TODO: generalize this to other bit widths
205
    uint64_t buffered_values_[8];
206
207
    // Number of values in buffered_values_
208
    int num_buffered_values_;
209
210
    // The current (also last) value that was written and the count of how
211
    // many times in a row that value has been seen.  This is maintained even
212
    // if we are in a literal run.  If the repeat_count_ get high enough, we switch
213
    // to encoding repeated runs.
214
    uint64_t current_value_;
215
    int repeat_count_;
216
217
    // Number of literals in the current run.  This does not include the literals
218
    // that might be in buffered_values_.  Only after we've got a group big enough
219
    // can we decide if they should part of the literal_count_ or repeat_count_
220
    int literal_count_;
221
222
    // Index of a byte in the underlying buffer that stores the indicator byte.
223
    // This is reserved as soon as we need a literal run but the value is written
224
    // when the literal run is complete. We maintain an index rather than a pointer
225
    // into the underlying buffer because the pointer value may become invalid if
226
    // the underlying buffer is resized.
227
    int literal_indicator_byte_idx_;
228
};
229
230
template <typename T>
231
137M
bool RleDecoder<T>::ReadHeader() {
232
137M
    DCHECK(bit_reader_.is_initialized());
233
137M
    if (literal_count_ == 0 && repeat_count_ == 0) [[unlikely]] {
234
        // Read the next run's indicator int, it could be a literal or repeated run
235
        // The int is encoded as a vlq-encoded value.
236
7.73M
        uint32_t indicator_value = 0;
237
7.73M
        bool result = bit_reader_.GetVlqInt(&indicator_value);
238
7.73M
        if (!result) [[unlikely]] {
239
12.0k
            return false;
240
12.0k
        }
241
242
        // lsb indicates if it is a literal run or repeated run
243
7.71M
        bool is_literal = indicator_value & 1;
244
7.71M
        if (is_literal) {
245
3.73M
            literal_count_ = (indicator_value >> 1) * 8;
246
3.73M
            DCHECK_GT(literal_count_, 0);
247
3.98M
        } else {
248
3.98M
            repeat_count_ = indicator_value >> 1;
249
3.98M
            DCHECK_GT(repeat_count_, 0);
250
3.98M
            bool result1 = bit_reader_.GetAligned<T>(BitUtil::Ceil(bit_width_, 8),
251
3.98M
                                                     reinterpret_cast<T*>(&current_value_));
252
3.98M
            DCHECK(result1);
253
3.98M
        }
254
7.71M
    }
255
137M
    return true;
256
137M
}
_ZN5doris10RleDecoderIsE10ReadHeaderEv
Line
Count
Source
231
123M
bool RleDecoder<T>::ReadHeader() {
232
123M
    DCHECK(bit_reader_.is_initialized());
233
123M
    if (literal_count_ == 0 && repeat_count_ == 0) [[unlikely]] {
234
        // Read the next run's indicator int, it could be a literal or repeated run
235
        // The int is encoded as a vlq-encoded value.
236
5.84M
        uint32_t indicator_value = 0;
237
5.84M
        bool result = bit_reader_.GetVlqInt(&indicator_value);
238
5.84M
        if (!result) [[unlikely]] {
239
55
            return false;
240
55
        }
241
242
        // lsb indicates if it is a literal run or repeated run
243
5.84M
        bool is_literal = indicator_value & 1;
244
5.84M
        if (is_literal) {
245
2.78M
            literal_count_ = (indicator_value >> 1) * 8;
246
2.78M
            DCHECK_GT(literal_count_, 0);
247
3.05M
        } else {
248
3.05M
            repeat_count_ = indicator_value >> 1;
249
3.05M
            DCHECK_GT(repeat_count_, 0);
250
3.05M
            bool result1 = bit_reader_.GetAligned<T>(BitUtil::Ceil(bit_width_, 8),
251
3.05M
                                                     reinterpret_cast<T*>(&current_value_));
252
3.05M
            DCHECK(result1);
253
3.05M
        }
254
5.84M
    }
255
123M
    return true;
256
123M
}
_ZN5doris10RleDecoderIbE10ReadHeaderEv
Line
Count
Source
231
5.33M
bool RleDecoder<T>::ReadHeader() {
232
5.33M
    DCHECK(bit_reader_.is_initialized());
233
5.33M
    if (literal_count_ == 0 && repeat_count_ == 0) [[unlikely]] {
234
        // Read the next run's indicator int, it could be a literal or repeated run
235
        // The int is encoded as a vlq-encoded value.
236
1.73M
        uint32_t indicator_value = 0;
237
1.73M
        bool result = bit_reader_.GetVlqInt(&indicator_value);
238
1.73M
        if (!result) [[unlikely]] {
239
12.0k
            return false;
240
12.0k
        }
241
242
        // lsb indicates if it is a literal run or repeated run
243
1.72M
        bool is_literal = indicator_value & 1;
244
1.72M
        if (is_literal) {
245
874k
            literal_count_ = (indicator_value >> 1) * 8;
246
874k
            DCHECK_GT(literal_count_, 0);
247
874k
        } else {
248
852k
            repeat_count_ = indicator_value >> 1;
249
852k
            DCHECK_GT(repeat_count_, 0);
250
852k
            bool result1 = bit_reader_.GetAligned<T>(BitUtil::Ceil(bit_width_, 8),
251
852k
                                                     reinterpret_cast<T*>(&current_value_));
252
852k
            DCHECK(result1);
253
852k
        }
254
1.72M
    }
255
5.32M
    return true;
256
5.33M
}
_ZN5doris10RleDecoderIhE10ReadHeaderEv
Line
Count
Source
231
8.27M
bool RleDecoder<T>::ReadHeader() {
232
8.27M
    DCHECK(bit_reader_.is_initialized());
233
8.27M
    if (literal_count_ == 0 && repeat_count_ == 0) [[unlikely]] {
234
        // Read the next run's indicator int, it could be a literal or repeated run
235
        // The int is encoded as a vlq-encoded value.
236
149k
        uint32_t indicator_value = 0;
237
149k
        bool result = bit_reader_.GetVlqInt(&indicator_value);
238
149k
        if (!result) [[unlikely]] {
239
0
            return false;
240
0
        }
241
242
        // lsb indicates if it is a literal run or repeated run
243
149k
        bool is_literal = indicator_value & 1;
244
149k
        if (is_literal) {
245
73.3k
            literal_count_ = (indicator_value >> 1) * 8;
246
73.3k
            DCHECK_GT(literal_count_, 0);
247
75.6k
        } else {
248
75.6k
            repeat_count_ = indicator_value >> 1;
249
75.6k
            DCHECK_GT(repeat_count_, 0);
250
75.6k
            bool result1 = bit_reader_.GetAligned<T>(BitUtil::Ceil(bit_width_, 8),
251
75.6k
                                                     reinterpret_cast<T*>(&current_value_));
252
75.6k
            DCHECK(result1);
253
75.6k
        }
254
149k
    }
255
8.27M
    return true;
256
8.27M
}
257
258
template <typename T>
259
127M
bool RleDecoder<T>::Get(T* val) {
260
127M
    DCHECK(bit_reader_.is_initialized());
261
127M
    if (!ReadHeader()) [[unlikely]] {
262
0
        return false;
263
0
    }
264
265
127M
    if (repeat_count_ > 0) [[likely]] {
266
62.9M
        *val = cast_set<T>(current_value_);
267
62.9M
        --repeat_count_;
268
62.9M
        rewind_state_ = REWIND_RUN;
269
64.1M
    } else {
270
64.1M
        DCHECK(literal_count_ > 0);
271
64.1M
        bool result = bit_reader_.GetValue(bit_width_, val);
272
64.1M
        DCHECK(result);
273
64.1M
        --literal_count_;
274
64.1M
        rewind_state_ = REWIND_LITERAL;
275
64.1M
    }
276
277
127M
    return true;
278
127M
}
_ZN5doris10RleDecoderIsE3GetEPs
Line
Count
Source
259
118M
bool RleDecoder<T>::Get(T* val) {
260
118M
    DCHECK(bit_reader_.is_initialized());
261
118M
    if (!ReadHeader()) [[unlikely]] {
262
0
        return false;
263
0
    }
264
265
118M
    if (repeat_count_ > 0) [[likely]] {
266
62.6M
        *val = cast_set<T>(current_value_);
267
62.6M
        --repeat_count_;
268
62.6M
        rewind_state_ = REWIND_RUN;
269
62.6M
    } else {
270
56.2M
        DCHECK(literal_count_ > 0);
271
56.2M
        bool result = bit_reader_.GetValue(bit_width_, val);
272
56.2M
        DCHECK(result);
273
56.2M
        --literal_count_;
274
56.2M
        rewind_state_ = REWIND_LITERAL;
275
56.2M
    }
276
277
118M
    return true;
278
118M
}
_ZN5doris10RleDecoderIhE3GetEPh
Line
Count
Source
259
8.12M
bool RleDecoder<T>::Get(T* val) {
260
8.12M
    DCHECK(bit_reader_.is_initialized());
261
8.12M
    if (!ReadHeader()) [[unlikely]] {
262
0
        return false;
263
0
    }
264
265
8.12M
    if (repeat_count_ > 0) [[likely]] {
266
254k
        *val = cast_set<T>(current_value_);
267
254k
        --repeat_count_;
268
254k
        rewind_state_ = REWIND_RUN;
269
7.86M
    } else {
270
7.86M
        DCHECK(literal_count_ > 0);
271
7.86M
        bool result = bit_reader_.GetValue(bit_width_, val);
272
7.86M
        DCHECK(result);
273
7.86M
        --literal_count_;
274
7.86M
        rewind_state_ = REWIND_LITERAL;
275
7.86M
    }
276
277
8.12M
    return true;
278
8.12M
}
279
280
template <typename T>
281
33.3k
void RleDecoder<T>::RewindOne() {
282
33.3k
    DCHECK(bit_reader_.is_initialized());
283
284
33.3k
    switch (rewind_state_) {
285
0
    case CANT_REWIND:
286
0
        throw Exception(Status::FatalError("Can't rewind more than once after each read!"));
287
0
        break;
288
14.5k
    case REWIND_RUN:
289
14.5k
        ++repeat_count_;
290
14.5k
        break;
291
18.8k
    case REWIND_LITERAL: {
292
18.8k
        bit_reader_.Rewind(bit_width_);
293
18.8k
        ++literal_count_;
294
18.8k
        break;
295
0
    }
296
33.3k
    }
297
298
33.3k
    rewind_state_ = CANT_REWIND;
299
33.3k
}
300
301
template <typename T>
302
5.60M
size_t RleDecoder<T>::GetNextRun(T* val, size_t max_run) {
303
5.60M
    DCHECK(bit_reader_.is_initialized());
304
5.60M
    DCHECK_GT(max_run, 0);
305
5.60M
    size_t ret = 0;
306
5.60M
    size_t rem = max_run;
307
6.90M
    while (ReadHeader()) {
308
6.90M
        if (repeat_count_ > 0) [[likely]] {
309
1.86M
            if (ret > 0 && *val != current_value_) [[unlikely]] {
310
76.7k
                return ret;
311
76.7k
            }
312
1.78M
            *val = cast_set<T>(current_value_);
313
1.78M
            if (repeat_count_ >= rem) {
314
                // The next run is longer than the amount of remaining data
315
                // that the caller wants to read. Only consume it partially.
316
1.12M
                repeat_count_ -= rem;
317
1.12M
                ret += rem;
318
1.12M
                return ret;
319
1.12M
            }
320
661k
            ret += repeat_count_;
321
661k
            rem -= repeat_count_;
322
661k
            repeat_count_ = 0;
323
5.04M
        } else {
324
5.04M
            DCHECK(literal_count_ > 0);
325
5.04M
            if (ret == 0) {
326
4.41M
                bool has_more = bit_reader_.GetValue(bit_width_, val);
327
4.41M
                DCHECK(has_more);
328
4.41M
                literal_count_--;
329
4.41M
                ret++;
330
4.41M
                rem--;
331
4.41M
            }
332
333
13.0M
            while (literal_count_ > 0) {
334
12.4M
                bool result = bit_reader_.GetValue(bit_width_, &current_value_);
335
12.4M
                DCHECK(result);
336
12.4M
                if (current_value_ != *val || rem == 0) {
337
4.40M
                    bit_reader_.Rewind(bit_width_);
338
4.40M
                    return ret;
339
4.40M
                }
340
8.00M
                ret++;
341
8.00M
                rem--;
342
8.00M
                literal_count_--;
343
8.00M
            }
344
5.04M
        }
345
6.90M
    }
346
18.4E
    return ret;
347
5.60M
}
_ZN5doris10RleDecoderIsE10GetNextRunEPsm
Line
Count
Source
302
1.86M
size_t RleDecoder<T>::GetNextRun(T* val, size_t max_run) {
303
1.86M
    DCHECK(bit_reader_.is_initialized());
304
1.86M
    DCHECK_GT(max_run, 0);
305
1.86M
    size_t ret = 0;
306
1.86M
    size_t rem = max_run;
307
2.26M
    while (ReadHeader()) {
308
2.26M
        if (repeat_count_ > 0) [[likely]] {
309
697k
            if (ret > 0 && *val != current_value_) [[unlikely]] {
310
15.3k
                return ret;
311
15.3k
            }
312
681k
            *val = cast_set<T>(current_value_);
313
681k
            if (repeat_count_ >= rem) {
314
                // The next run is longer than the amount of remaining data
315
                // that the caller wants to read. Only consume it partially.
316
485k
                repeat_count_ -= rem;
317
485k
                ret += rem;
318
485k
                return ret;
319
485k
            }
320
196k
            ret += repeat_count_;
321
196k
            rem -= repeat_count_;
322
196k
            repeat_count_ = 0;
323
1.56M
        } else {
324
1.56M
            DCHECK(literal_count_ > 0);
325
1.56M
            if (ret == 0) {
326
1.36M
                bool has_more = bit_reader_.GetValue(bit_width_, val);
327
1.36M
                DCHECK(has_more);
328
1.36M
                literal_count_--;
329
1.36M
                ret++;
330
1.36M
                rem--;
331
1.36M
            }
332
333
4.44M
            while (literal_count_ > 0) {
334
4.24M
                bool result = bit_reader_.GetValue(bit_width_, &current_value_);
335
4.24M
                DCHECK(result);
336
4.24M
                if (current_value_ != *val || rem == 0) {
337
1.36M
                    bit_reader_.Rewind(bit_width_);
338
1.36M
                    return ret;
339
1.36M
                }
340
2.87M
                ret++;
341
2.87M
                rem--;
342
2.87M
                literal_count_--;
343
2.87M
            }
344
1.56M
        }
345
2.26M
    }
346
37
    return ret;
347
1.86M
}
_ZN5doris10RleDecoderIbE10GetNextRunEPbm
Line
Count
Source
302
3.73M
size_t RleDecoder<T>::GetNextRun(T* val, size_t max_run) {
303
3.73M
    DCHECK(bit_reader_.is_initialized());
304
3.73M
    DCHECK_GT(max_run, 0);
305
3.73M
    size_t ret = 0;
306
3.73M
    size_t rem = max_run;
307
4.64M
    while (ReadHeader()) {
308
4.64M
        if (repeat_count_ > 0) [[likely]] {
309
1.16M
            if (ret > 0 && *val != current_value_) [[unlikely]] {
310
61.4k
                return ret;
311
61.4k
            }
312
1.10M
            *val = cast_set<T>(current_value_);
313
1.10M
            if (repeat_count_ >= rem) {
314
                // The next run is longer than the amount of remaining data
315
                // that the caller wants to read. Only consume it partially.
316
640k
                repeat_count_ -= rem;
317
640k
                ret += rem;
318
640k
                return ret;
319
640k
            }
320
465k
            ret += repeat_count_;
321
465k
            rem -= repeat_count_;
322
465k
            repeat_count_ = 0;
323
3.47M
        } else {
324
3.47M
            DCHECK(literal_count_ > 0);
325
3.47M
            if (ret == 0) {
326
3.04M
                bool has_more = bit_reader_.GetValue(bit_width_, val);
327
3.04M
                DCHECK(has_more);
328
3.04M
                literal_count_--;
329
3.04M
                ret++;
330
3.04M
                rem--;
331
3.04M
            }
332
333
8.60M
            while (literal_count_ > 0) {
334
8.17M
                bool result = bit_reader_.GetValue(bit_width_, &current_value_);
335
8.17M
                DCHECK(result);
336
8.17M
                if (current_value_ != *val || rem == 0) {
337
3.03M
                    bit_reader_.Rewind(bit_width_);
338
3.03M
                    return ret;
339
3.03M
                }
340
5.13M
                ret++;
341
5.13M
                rem--;
342
5.13M
                literal_count_--;
343
5.13M
            }
344
3.47M
        }
345
4.64M
    }
346
18.4E
    return ret;
347
3.73M
}
348
349
template <typename T>
350
136k
size_t RleDecoder<T>::get_values(T* values, size_t num_values) {
351
136k
    size_t read_num = 0;
352
5.27M
    while (read_num < num_values) {
353
5.14M
        size_t read_this_time = num_values - read_num;
354
355
5.14M
        if (LIKELY(repeat_count_ > 0)) {
356
1.30M
            read_this_time = std::min((size_t)repeat_count_, read_this_time);
357
1.30M
            std::fill(values, values + read_this_time, current_value_);
358
1.30M
            values += read_this_time;
359
1.30M
            repeat_count_ -= read_this_time;
360
1.30M
            read_num += read_this_time;
361
3.83M
        } else if (literal_count_ > 0) {
362
1.26M
            read_this_time = std::min((size_t)literal_count_, read_this_time);
363
20.7M
            for (int i = 0; i < read_this_time; ++i) {
364
19.4M
                bool result = bit_reader_.GetValue(bit_width_, values);
365
19.4M
                DCHECK(result);
366
19.4M
                values++;
367
19.4M
            }
368
1.26M
            literal_count_ -= read_this_time;
369
1.26M
            read_num += read_this_time;
370
2.56M
        } else {
371
2.56M
            if (!ReadHeader()) {
372
0
                return read_num;
373
0
            }
374
2.56M
        }
375
5.14M
    }
376
136k
    return read_num;
377
136k
}
_ZN5doris10RleDecoderIsE10get_valuesEPsm
Line
Count
Source
350
136k
size_t RleDecoder<T>::get_values(T* values, size_t num_values) {
351
136k
    size_t read_num = 0;
352
5.27M
    while (read_num < num_values) {
353
5.13M
        size_t read_this_time = num_values - read_num;
354
355
5.13M
        if (LIKELY(repeat_count_ > 0)) {
356
1.30M
            read_this_time = std::min((size_t)repeat_count_, read_this_time);
357
1.30M
            std::fill(values, values + read_this_time, current_value_);
358
1.30M
            values += read_this_time;
359
1.30M
            repeat_count_ -= read_this_time;
360
1.30M
            read_num += read_this_time;
361
3.83M
        } else if (literal_count_ > 0) {
362
1.26M
            read_this_time = std::min((size_t)literal_count_, read_this_time);
363
20.7M
            for (int i = 0; i < read_this_time; ++i) {
364
19.4M
                bool result = bit_reader_.GetValue(bit_width_, values);
365
19.4M
                DCHECK(result);
366
19.4M
                values++;
367
19.4M
            }
368
1.26M
            literal_count_ -= read_this_time;
369
1.26M
            read_num += read_this_time;
370
2.56M
        } else {
371
2.56M
            if (!ReadHeader()) {
372
0
                return read_num;
373
0
            }
374
2.56M
        }
375
5.13M
    }
376
136k
    return read_num;
377
136k
}
_ZN5doris10RleDecoderIhE10get_valuesEPhm
Line
Count
Source
350
377
size_t RleDecoder<T>::get_values(T* values, size_t num_values) {
351
377
    size_t read_num = 0;
352
1.13k
    while (read_num < num_values) {
353
753
        size_t read_this_time = num_values - read_num;
354
355
753
        if (LIKELY(repeat_count_ > 0)) {
356
36
            read_this_time = std::min((size_t)repeat_count_, read_this_time);
357
36
            std::fill(values, values + read_this_time, current_value_);
358
36
            values += read_this_time;
359
36
            repeat_count_ -= read_this_time;
360
36
            read_num += read_this_time;
361
717
        } else if (literal_count_ > 0) {
362
341
            read_this_time = std::min((size_t)literal_count_, read_this_time);
363
1.50k
            for (int i = 0; i < read_this_time; ++i) {
364
1.16k
                bool result = bit_reader_.GetValue(bit_width_, values);
365
1.16k
                DCHECK(result);
366
1.16k
                values++;
367
1.16k
            }
368
341
            literal_count_ -= read_this_time;
369
341
            read_num += read_this_time;
370
376
        } else {
371
376
            if (!ReadHeader()) {
372
0
                return read_num;
373
0
            }
374
376
        }
375
753
    }
376
377
    return read_num;
377
377
}
378
379
template <typename T>
380
size_t RleDecoder<T>::repeated_count() {
381
    if (repeat_count_ > 0) {
382
        return repeat_count_;
383
    }
384
    if (literal_count_ == 0) {
385
        ReadHeader();
386
    }
387
    return repeat_count_;
388
}
389
390
template <typename T>
391
T RleDecoder<T>::get_repeated_value(size_t count) {
392
    DCHECK_GE(repeat_count_, count);
393
    repeat_count_ -= count;
394
    return current_value_;
395
}
396
397
template <typename T>
398
1.42M
size_t RleDecoder<T>::Skip(size_t to_skip) {
399
1.42M
    DCHECK(bit_reader_.is_initialized());
400
401
1.42M
    size_t set_count = 0;
402
2.25M
    while (to_skip > 0) {
403
829k
        bool result = ReadHeader();
404
829k
        DCHECK(result);
405
406
829k
        if (repeat_count_ > 0) [[likely]] {
407
393k
            size_t nskip = (repeat_count_ < to_skip) ? repeat_count_ : to_skip;
408
393k
            repeat_count_ -= nskip;
409
393k
            to_skip -= nskip;
410
393k
            if (current_value_ != 0) {
411
295k
                set_count += nskip;
412
295k
            }
413
435k
        } else {
414
435k
            DCHECK(literal_count_ > 0);
415
435k
            size_t nskip = (literal_count_ < to_skip) ? literal_count_ : to_skip;
416
435k
            literal_count_ -= nskip;
417
435k
            to_skip -= nskip;
418
24.7M
            for (; nskip > 0; nskip--) {
419
24.3M
                T value = 0;
420
24.3M
                bool result1 = bit_reader_.GetValue(bit_width_, &value);
421
24.3M
                DCHECK(result1);
422
24.3M
                if (value != 0) {
423
14.4M
                    set_count++;
424
14.4M
                }
425
24.3M
            }
426
435k
        }
427
829k
    }
428
1.42M
    return set_count;
429
1.42M
}
_ZN5doris10RleDecoderIbE4SkipEm
Line
Count
Source
398
308k
size_t RleDecoder<T>::Skip(size_t to_skip) {
399
308k
    DCHECK(bit_reader_.is_initialized());
400
401
308k
    size_t set_count = 0;
402
988k
    while (to_skip > 0) {
403
679k
        bool result = ReadHeader();
404
679k
        DCHECK(result);
405
406
679k
        if (repeat_count_ > 0) [[likely]] {
407
317k
            size_t nskip = (repeat_count_ < to_skip) ? repeat_count_ : to_skip;
408
317k
            repeat_count_ -= nskip;
409
317k
            to_skip -= nskip;
410
317k
            if (current_value_ != 0) {
411
260k
                set_count += nskip;
412
260k
            }
413
362k
        } else {
414
362k
            DCHECK(literal_count_ > 0);
415
362k
            size_t nskip = (literal_count_ < to_skip) ? literal_count_ : to_skip;
416
362k
            literal_count_ -= nskip;
417
362k
            to_skip -= nskip;
418
7.58M
            for (; nskip > 0; nskip--) {
419
7.22M
                T value = 0;
420
7.22M
                bool result1 = bit_reader_.GetValue(bit_width_, &value);
421
7.22M
                DCHECK(result1);
422
7.22M
                if (value != 0) {
423
5.76M
                    set_count++;
424
5.76M
                }
425
7.22M
            }
426
362k
        }
427
679k
    }
428
308k
    return set_count;
429
308k
}
_ZN5doris10RleDecoderIhE4SkipEm
Line
Count
Source
398
1.11M
size_t RleDecoder<T>::Skip(size_t to_skip) {
399
1.11M
    DCHECK(bit_reader_.is_initialized());
400
401
1.11M
    size_t set_count = 0;
402
1.26M
    while (to_skip > 0) {
403
149k
        bool result = ReadHeader();
404
149k
        DCHECK(result);
405
406
149k
        if (repeat_count_ > 0) [[likely]] {
407
76.1k
            size_t nskip = (repeat_count_ < to_skip) ? repeat_count_ : to_skip;
408
76.1k
            repeat_count_ -= nskip;
409
76.1k
            to_skip -= nskip;
410
76.1k
            if (current_value_ != 0) {
411
34.8k
                set_count += nskip;
412
34.8k
            }
413
76.1k
        } else {
414
73.4k
            DCHECK(literal_count_ > 0);
415
73.4k
            size_t nskip = (literal_count_ < to_skip) ? literal_count_ : to_skip;
416
73.4k
            literal_count_ -= nskip;
417
73.4k
            to_skip -= nskip;
418
17.1M
            for (; nskip > 0; nskip--) {
419
17.0M
                T value = 0;
420
17.0M
                bool result1 = bit_reader_.GetValue(bit_width_, &value);
421
17.0M
                DCHECK(result1);
422
17.0M
                if (value != 0) {
423
8.73M
                    set_count++;
424
8.73M
                }
425
17.0M
            }
426
73.4k
        }
427
149k
    }
428
1.11M
    return set_count;
429
1.11M
}
430
431
// This function buffers input values 8 at a time.  After seeing all 8 values,
432
// it decides whether they should be encoded as a literal or repeated run.
433
template <typename T>
434
12.9M
void RleEncoder<T>::Put(T value, size_t run_length) {
435
12.9M
    DCHECK(bit_width_ == 64 || value < (1LL << bit_width_));
436
437
    // Fast path: if this is a continuation of the current repeated run and
438
    // we've already buffered enough values, just increment repeat_count_
439
12.9M
    if (current_value_ == value && repeat_count_ >= 8 && run_length > 0) [[likely]] {
440
1.14M
        repeat_count_ += run_length;
441
1.14M
        return;
442
1.14M
    }
443
444
    // Handle run_length > 1 more efficiently
445
27.9M
    while (run_length > 0) {
446
17.6M
        if (current_value_ == value) [[likely]] {
447
            // Need to buffer values until we reach 8
448
8.12M
            size_t to_buffer = std::min(run_length, size_t(8 - num_buffered_values_));
449
37.4M
            for (size_t i = 0; i < to_buffer; ++i) {
450
29.3M
                buffered_values_[num_buffered_values_++] = value;
451
29.3M
                ++repeat_count_;
452
29.3M
            }
453
8.12M
            run_length -= to_buffer;
454
8.12M
            if (num_buffered_values_ == 8) {
455
3.89M
                DCHECK_EQ(literal_count_ % 8, 0);
456
3.89M
                FlushBufferedValues(false);
457
                // After flushing, if we still have a repeated run and more values,
458
                // we can add them directly to repeat_count_
459
3.89M
                if (repeat_count_ >= 8 && run_length > 0) {
460
1.44M
                    repeat_count_ += run_length;
461
1.44M
                    return;
462
1.44M
                }
463
3.89M
            }
464
9.49M
        } else {
465
            // Value changed
466
9.49M
            if (repeat_count_ >= 8) {
467
                // We had a run that was long enough but it has ended.  Flush the
468
                // current repeated run.
469
1.39M
                DCHECK_EQ(literal_count_, 0);
470
1.39M
                FlushRepeatedRun();
471
1.39M
            }
472
9.49M
            repeat_count_ = 1;
473
9.49M
            current_value_ = value;
474
475
9.49M
            buffered_values_[num_buffered_values_++] = value;
476
9.49M
            --run_length;
477
9.49M
            if (num_buffered_values_ == 8) {
478
859k
                DCHECK_EQ(literal_count_ % 8, 0);
479
859k
                FlushBufferedValues(false);
480
859k
            }
481
9.49M
        }
482
17.6M
    }
483
11.7M
}
_ZN5doris10RleEncoderIhE3PutEhm
Line
Count
Source
434
4.19M
void RleEncoder<T>::Put(T value, size_t run_length) {
435
4.19M
    DCHECK(bit_width_ == 64 || value < (1LL << bit_width_));
436
437
    // Fast path: if this is a continuation of the current repeated run and
438
    // we've already buffered enough values, just increment repeat_count_
439
4.19M
    if (current_value_ == value && repeat_count_ >= 8 && run_length > 0) [[likely]] {
440
495k
        repeat_count_ += run_length;
441
495k
        return;
442
495k
    }
443
444
    // Handle run_length > 1 more efficiently
445
7.39M
    while (run_length > 0) {
446
3.70M
        if (current_value_ == value) [[likely]] {
447
            // Need to buffer values until we reach 8
448
1.85M
            size_t to_buffer = std::min(run_length, size_t(8 - num_buffered_values_));
449
3.71M
            for (size_t i = 0; i < to_buffer; ++i) {
450
1.85M
                buffered_values_[num_buffered_values_++] = value;
451
1.85M
                ++repeat_count_;
452
1.85M
            }
453
1.85M
            run_length -= to_buffer;
454
1.85M
            if (num_buffered_values_ == 8) {
455
233k
                DCHECK_EQ(literal_count_ % 8, 0);
456
233k
                FlushBufferedValues(false);
457
                // After flushing, if we still have a repeated run and more values,
458
                // we can add them directly to repeat_count_
459
233k
                if (repeat_count_ >= 8 && run_length > 0) {
460
3
                    repeat_count_ += run_length;
461
3
                    return;
462
3
                }
463
233k
            }
464
1.85M
        } else {
465
            // Value changed
466
1.84M
            if (repeat_count_ >= 8) {
467
                // We had a run that was long enough but it has ended.  Flush the
468
                // current repeated run.
469
6.24k
                DCHECK_EQ(literal_count_, 0);
470
6.24k
                FlushRepeatedRun();
471
6.24k
            }
472
1.84M
            repeat_count_ = 1;
473
1.84M
            current_value_ = value;
474
475
1.84M
            buffered_values_[num_buffered_values_++] = value;
476
1.84M
            --run_length;
477
1.84M
            if (num_buffered_values_ == 8) {
478
                DCHECK_EQ(literal_count_ % 8, 0);
479
227k
                FlushBufferedValues(false);
480
227k
            }
481
1.84M
        }
482
3.70M
    }
483
3.69M
}
_ZN5doris10RleEncoderIbE3PutEbm
Line
Count
Source
434
8.73M
void RleEncoder<T>::Put(T value, size_t run_length) {
435
8.73M
    DCHECK(bit_width_ == 64 || value < (1LL << bit_width_));
436
437
    // Fast path: if this is a continuation of the current repeated run and
438
    // we've already buffered enough values, just increment repeat_count_
439
8.73M
    if (current_value_ == value && repeat_count_ >= 8 && run_length > 0) [[likely]] {
440
652k
        repeat_count_ += run_length;
441
652k
        return;
442
652k
    }
443
444
    // Handle run_length > 1 more efficiently
445
20.5M
    while (run_length > 0) {
446
13.9M
        if (current_value_ == value) [[likely]] {
447
            // Need to buffer values until we reach 8
448
6.26M
            size_t to_buffer = std::min(run_length, size_t(8 - num_buffered_values_));
449
33.7M
            for (size_t i = 0; i < to_buffer; ++i) {
450
27.5M
                buffered_values_[num_buffered_values_++] = value;
451
27.5M
                ++repeat_count_;
452
27.5M
            }
453
6.26M
            run_length -= to_buffer;
454
6.26M
            if (num_buffered_values_ == 8) {
455
3.66M
                DCHECK_EQ(literal_count_ % 8, 0);
456
3.66M
                FlushBufferedValues(false);
457
                // After flushing, if we still have a repeated run and more values,
458
                // we can add them directly to repeat_count_
459
3.66M
                if (repeat_count_ >= 8 && run_length > 0) {
460
1.44M
                    repeat_count_ += run_length;
461
1.44M
                    return;
462
1.44M
                }
463
3.66M
            }
464
7.65M
        } else {
465
            // Value changed
466
7.65M
            if (repeat_count_ >= 8) {
467
                // We had a run that was long enough but it has ended.  Flush the
468
                // current repeated run.
469
1.39M
                DCHECK_EQ(literal_count_, 0);
470
1.39M
                FlushRepeatedRun();
471
1.39M
            }
472
7.65M
            repeat_count_ = 1;
473
7.65M
            current_value_ = value;
474
475
7.65M
            buffered_values_[num_buffered_values_++] = value;
476
7.65M
            --run_length;
477
7.65M
            if (num_buffered_values_ == 8) {
478
                DCHECK_EQ(literal_count_ % 8, 0);
479
632k
                FlushBufferedValues(false);
480
632k
            }
481
7.65M
        }
482
13.9M
    }
483
8.08M
}
484
485
template <typename T>
486
4.61M
void RleEncoder<T>::FlushLiteralRun(bool update_indicator_byte) {
487
4.61M
    if (literal_indicator_byte_idx_ < 0) {
488
        // The literal indicator byte has not been reserved yet, get one now.
489
1.42M
        literal_indicator_byte_idx_ = cast_set<int>(bit_writer_.GetByteIndexAndAdvance(1));
490
1.42M
        DCHECK_GE(literal_indicator_byte_idx_, 0);
491
1.42M
    }
492
493
    // Write all the buffered values as bit packed literals
494
30.2M
    for (int i = 0; i < num_buffered_values_; ++i) {
495
25.6M
        bit_writer_.PutValue(buffered_values_[i], bit_width_);
496
25.6M
    }
497
4.61M
    num_buffered_values_ = 0;
498
499
4.61M
    if (update_indicator_byte) {
500
        // At this point we need to write the indicator byte for the literal run.
501
        // We only reserve one byte, to allow for streaming writes of literal values.
502
        // The logic makes sure we flush literal runs often enough to not overrun
503
        // the 1 byte.
504
1.42M
        int num_groups = BitUtil::Ceil(literal_count_, 8);
505
1.42M
        int32_t indicator_value = (num_groups << 1) | 1;
506
1.42M
        DCHECK_EQ(indicator_value & 0xFFFFFF00, 0);
507
1.42M
        bit_writer_.buffer()->data()[literal_indicator_byte_idx_] =
508
1.42M
                cast_set<uint8_t>(indicator_value);
509
1.42M
        literal_indicator_byte_idx_ = -1;
510
1.42M
        literal_count_ = 0;
511
1.42M
    }
512
4.61M
}
_ZN5doris10RleEncoderIhE15FlushLiteralRunEb
Line
Count
Source
486
460k
void RleEncoder<T>::FlushLiteralRun(bool update_indicator_byte) {
487
460k
    if (literal_indicator_byte_idx_ < 0) {
488
        // The literal indicator byte has not been reserved yet, get one now.
489
13.7k
        literal_indicator_byte_idx_ = cast_set<int>(bit_writer_.GetByteIndexAndAdvance(1));
490
13.7k
        DCHECK_GE(literal_indicator_byte_idx_, 0);
491
13.7k
    }
492
493
    // Write all the buffered values as bit packed literals
494
4.08M
    for (int i = 0; i < num_buffered_values_; ++i) {
495
3.62M
        bit_writer_.PutValue(buffered_values_[i], bit_width_);
496
3.62M
    }
497
460k
    num_buffered_values_ = 0;
498
499
460k
    if (update_indicator_byte) {
500
        // At this point we need to write the indicator byte for the literal run.
501
        // We only reserve one byte, to allow for streaming writes of literal values.
502
        // The logic makes sure we flush literal runs often enough to not overrun
503
        // the 1 byte.
504
13.7k
        int num_groups = BitUtil::Ceil(literal_count_, 8);
505
13.7k
        int32_t indicator_value = (num_groups << 1) | 1;
506
        DCHECK_EQ(indicator_value & 0xFFFFFF00, 0);
507
13.7k
        bit_writer_.buffer()->data()[literal_indicator_byte_idx_] =
508
13.7k
                cast_set<uint8_t>(indicator_value);
509
13.7k
        literal_indicator_byte_idx_ = -1;
510
13.7k
        literal_count_ = 0;
511
13.7k
    }
512
460k
}
_ZN5doris10RleEncoderIbE15FlushLiteralRunEb
Line
Count
Source
486
4.15M
void RleEncoder<T>::FlushLiteralRun(bool update_indicator_byte) {
487
4.15M
    if (literal_indicator_byte_idx_ < 0) {
488
        // The literal indicator byte has not been reserved yet, get one now.
489
1.41M
        literal_indicator_byte_idx_ = cast_set<int>(bit_writer_.GetByteIndexAndAdvance(1));
490
1.41M
        DCHECK_GE(literal_indicator_byte_idx_, 0);
491
1.41M
    }
492
493
    // Write all the buffered values as bit packed literals
494
26.1M
    for (int i = 0; i < num_buffered_values_; ++i) {
495
22.0M
        bit_writer_.PutValue(buffered_values_[i], bit_width_);
496
22.0M
    }
497
4.15M
    num_buffered_values_ = 0;
498
499
4.15M
    if (update_indicator_byte) {
500
        // At this point we need to write the indicator byte for the literal run.
501
        // We only reserve one byte, to allow for streaming writes of literal values.
502
        // The logic makes sure we flush literal runs often enough to not overrun
503
        // the 1 byte.
504
1.41M
        int num_groups = BitUtil::Ceil(literal_count_, 8);
505
1.41M
        int32_t indicator_value = (num_groups << 1) | 1;
506
        DCHECK_EQ(indicator_value & 0xFFFFFF00, 0);
507
1.41M
        bit_writer_.buffer()->data()[literal_indicator_byte_idx_] =
508
1.41M
                cast_set<uint8_t>(indicator_value);
509
1.41M
        literal_indicator_byte_idx_ = -1;
510
1.41M
        literal_count_ = 0;
511
1.41M
    }
512
4.15M
}
513
514
template <typename T>
515
1.51M
void RleEncoder<T>::FlushRepeatedRun() {
516
1.51M
    DCHECK_GT(repeat_count_, 0);
517
    // The lsb of 0 indicates this is a repeated run
518
1.51M
    int32_t indicator_value = repeat_count_ << 1 | 0;
519
1.51M
    bit_writer_.PutVlqInt(indicator_value);
520
1.51M
    bit_writer_.PutAligned(current_value_, BitUtil::Ceil(bit_width_, 8));
521
1.51M
    num_buffered_values_ = 0;
522
1.51M
    repeat_count_ = 0;
523
1.51M
}
_ZN5doris10RleEncoderIhE16FlushRepeatedRunEv
Line
Count
Source
515
15.9k
void RleEncoder<T>::FlushRepeatedRun() {
516
15.9k
    DCHECK_GT(repeat_count_, 0);
517
    // The lsb of 0 indicates this is a repeated run
518
15.9k
    int32_t indicator_value = repeat_count_ << 1 | 0;
519
15.9k
    bit_writer_.PutVlqInt(indicator_value);
520
15.9k
    bit_writer_.PutAligned(current_value_, BitUtil::Ceil(bit_width_, 8));
521
15.9k
    num_buffered_values_ = 0;
522
15.9k
    repeat_count_ = 0;
523
15.9k
}
_ZN5doris10RleEncoderIbE16FlushRepeatedRunEv
Line
Count
Source
515
1.50M
void RleEncoder<T>::FlushRepeatedRun() {
516
1.50M
    DCHECK_GT(repeat_count_, 0);
517
    // The lsb of 0 indicates this is a repeated run
518
1.50M
    int32_t indicator_value = repeat_count_ << 1 | 0;
519
1.50M
    bit_writer_.PutVlqInt(indicator_value);
520
1.50M
    bit_writer_.PutAligned(current_value_, BitUtil::Ceil(bit_width_, 8));
521
1.50M
    num_buffered_values_ = 0;
522
1.50M
    repeat_count_ = 0;
523
1.50M
}
524
525
// Flush the values that have been buffered.  At this point we decide whether
526
// we need to switch between the run types or continue the current one.
527
template <typename T>
528
4.74M
void RleEncoder<T>::FlushBufferedValues(bool done) {
529
4.74M
    if (repeat_count_ >= 8) {
530
        // Clear the buffered values.  They are part of the repeated run now and we
531
        // don't want to flush them out as literals.
532
1.55M
        num_buffered_values_ = 0;
533
1.55M
        if (literal_count_ != 0) {
534
            // There was a current literal run.  All the values in it have been flushed
535
            // but we still need to update the indicator byte.
536
1.36M
            DCHECK_EQ(literal_count_ % 8, 0);
537
1.36M
            DCHECK_EQ(repeat_count_, 8);
538
1.36M
            FlushLiteralRun(true);
539
1.36M
        }
540
1.55M
        DCHECK_EQ(literal_count_, 0);
541
1.55M
        return;
542
1.55M
    }
543
544
3.19M
    literal_count_ += num_buffered_values_;
545
3.19M
    int num_groups = BitUtil::Ceil(literal_count_, 8);
546
3.19M
    if (num_groups + 1 >= (1 << 6)) {
547
        // We need to start a new literal run because the indicator byte we've reserved
548
        // cannot store more values.
549
8.83k
        DCHECK_GE(literal_indicator_byte_idx_, 0);
550
8.83k
        FlushLiteralRun(true);
551
3.18M
    } else {
552
3.18M
        FlushLiteralRun(done);
553
3.18M
    }
554
3.19M
    repeat_count_ = 0;
555
3.19M
}
_ZN5doris10RleEncoderIhE19FlushBufferedValuesEb
Line
Count
Source
528
460k
void RleEncoder<T>::FlushBufferedValues(bool done) {
529
460k
    if (repeat_count_ >= 8) {
530
        // Clear the buffered values.  They are part of the repeated run now and we
531
        // don't want to flush them out as literals.
532
7.90k
        num_buffered_values_ = 0;
533
7.90k
        if (literal_count_ != 0) {
534
            // There was a current literal run.  All the values in it have been flushed
535
            // but we still need to update the indicator byte.
536
4.99k
            DCHECK_EQ(literal_count_ % 8, 0);
537
4.99k
            DCHECK_EQ(repeat_count_, 8);
538
4.99k
            FlushLiteralRun(true);
539
4.99k
        }
540
7.90k
        DCHECK_EQ(literal_count_, 0);
541
7.90k
        return;
542
7.90k
    }
543
544
452k
    literal_count_ += num_buffered_values_;
545
452k
    int num_groups = BitUtil::Ceil(literal_count_, 8);
546
452k
    if (num_groups + 1 >= (1 << 6)) {
547
        // We need to start a new literal run because the indicator byte we've reserved
548
        // cannot store more values.
549
5.31k
        DCHECK_GE(literal_indicator_byte_idx_, 0);
550
5.31k
        FlushLiteralRun(true);
551
447k
    } else {
552
447k
        FlushLiteralRun(done);
553
447k
    }
554
452k
    repeat_count_ = 0;
555
452k
}
_ZN5doris10RleEncoderIbE19FlushBufferedValuesEb
Line
Count
Source
528
4.28M
void RleEncoder<T>::FlushBufferedValues(bool done) {
529
4.28M
    if (repeat_count_ >= 8) {
530
        // Clear the buffered values.  They are part of the repeated run now and we
531
        // don't want to flush them out as literals.
532
1.54M
        num_buffered_values_ = 0;
533
1.54M
        if (literal_count_ != 0) {
534
            // There was a current literal run.  All the values in it have been flushed
535
            // but we still need to update the indicator byte.
536
1.36M
            DCHECK_EQ(literal_count_ % 8, 0);
537
1.36M
            DCHECK_EQ(repeat_count_, 8);
538
1.36M
            FlushLiteralRun(true);
539
1.36M
        }
540
1.54M
        DCHECK_EQ(literal_count_, 0);
541
1.54M
        return;
542
1.54M
    }
543
544
2.74M
    literal_count_ += num_buffered_values_;
545
2.74M
    int num_groups = BitUtil::Ceil(literal_count_, 8);
546
2.74M
    if (num_groups + 1 >= (1 << 6)) {
547
        // We need to start a new literal run because the indicator byte we've reserved
548
        // cannot store more values.
549
3.51k
        DCHECK_GE(literal_indicator_byte_idx_, 0);
550
3.51k
        FlushLiteralRun(true);
551
2.73M
    } else {
552
2.73M
        FlushLiteralRun(done);
553
2.73M
    }
554
2.74M
    repeat_count_ = 0;
555
2.74M
}
556
557
template <typename T>
558
28.5k
void RleEncoder<T>::Reserve(int num_bytes, uint8_t val) {
559
142k
    for (int i = 0; i < num_bytes; ++i) {
560
114k
        bit_writer_.PutValue(val, 8);
561
114k
    }
562
28.5k
}
563
564
template <typename T>
565
174k
int RleEncoder<T>::Flush() {
566
174k
    if (literal_count_ > 0 || repeat_count_ > 0 || num_buffered_values_ > 0) {
567
173k
        bool all_repeat = literal_count_ == 0 &&
568
173k
                          (repeat_count_ == num_buffered_values_ || num_buffered_values_ == 0);
569
        // There is something pending, figure out if it's a repeated or literal run
570
173k
        if (repeat_count_ > 0 && all_repeat) {
571
121k
            FlushRepeatedRun();
572
121k
        } else {
573
51.5k
            literal_count_ += num_buffered_values_;
574
51.5k
            FlushLiteralRun(true);
575
51.5k
            repeat_count_ = 0;
576
51.5k
        }
577
173k
    }
578
174k
    bit_writer_.Flush();
579
174k
    DCHECK_EQ(num_buffered_values_, 0);
580
174k
    DCHECK_EQ(literal_count_, 0);
581
174k
    DCHECK_EQ(repeat_count_, 0);
582
174k
    return bit_writer_.bytes_written();
583
174k
}
_ZN5doris10RleEncoderIhE5FlushEv
Line
Count
Source
565
13.7k
int RleEncoder<T>::Flush() {
566
13.7k
    if (literal_count_ > 0 || repeat_count_ > 0 || num_buffered_values_ > 0) {
567
13.1k
        bool all_repeat = literal_count_ == 0 &&
568
13.1k
                          (repeat_count_ == num_buffered_values_ || num_buffered_values_ == 0);
569
        // There is something pending, figure out if it's a repeated or literal run
570
13.1k
        if (repeat_count_ > 0 && all_repeat) {
571
9.72k
            FlushRepeatedRun();
572
9.72k
        } else {
573
3.44k
            literal_count_ += num_buffered_values_;
574
3.44k
            FlushLiteralRun(true);
575
3.44k
            repeat_count_ = 0;
576
3.44k
        }
577
13.1k
    }
578
13.7k
    bit_writer_.Flush();
579
13.7k
    DCHECK_EQ(num_buffered_values_, 0);
580
13.7k
    DCHECK_EQ(literal_count_, 0);
581
    DCHECK_EQ(repeat_count_, 0);
582
13.7k
    return bit_writer_.bytes_written();
583
13.7k
}
_ZN5doris10RleEncoderIbE5FlushEv
Line
Count
Source
565
160k
int RleEncoder<T>::Flush() {
566
160k
    if (literal_count_ > 0 || repeat_count_ > 0 || num_buffered_values_ > 0) {
567
160k
        bool all_repeat = literal_count_ == 0 &&
568
160k
                          (repeat_count_ == num_buffered_values_ || num_buffered_values_ == 0);
569
        // There is something pending, figure out if it's a repeated or literal run
570
160k
        if (repeat_count_ > 0 && all_repeat) {
571
112k
            FlushRepeatedRun();
572
112k
        } else {
573
48.1k
            literal_count_ += num_buffered_values_;
574
48.1k
            FlushLiteralRun(true);
575
48.1k
            repeat_count_ = 0;
576
48.1k
        }
577
160k
    }
578
160k
    bit_writer_.Flush();
579
160k
    DCHECK_EQ(num_buffered_values_, 0);
580
160k
    DCHECK_EQ(literal_count_, 0);
581
    DCHECK_EQ(repeat_count_, 0);
582
160k
    return bit_writer_.bytes_written();
583
160k
}
584
585
template <typename T>
586
1.11M
void RleEncoder<T>::Clear() {
587
1.11M
    current_value_ = 0;
588
1.11M
    repeat_count_ = 0;
589
1.11M
    num_buffered_values_ = 0;
590
1.11M
    literal_count_ = 0;
591
1.11M
    literal_indicator_byte_idx_ = -1;
592
1.11M
    bit_writer_.Clear();
593
1.11M
}
_ZN5doris10RleEncoderIhE5ClearEv
Line
Count
Source
586
43.3k
void RleEncoder<T>::Clear() {
587
43.3k
    current_value_ = 0;
588
43.3k
    repeat_count_ = 0;
589
43.3k
    num_buffered_values_ = 0;
590
43.3k
    literal_count_ = 0;
591
43.3k
    literal_indicator_byte_idx_ = -1;
592
43.3k
    bit_writer_.Clear();
593
43.3k
}
_ZN5doris10RleEncoderIbE5ClearEv
Line
Count
Source
586
1.06M
void RleEncoder<T>::Clear() {
587
1.06M
    current_value_ = 0;
588
1.06M
    repeat_count_ = 0;
589
1.06M
    num_buffered_values_ = 0;
590
1.06M
    literal_count_ = 0;
591
1.06M
    literal_indicator_byte_idx_ = -1;
592
1.06M
    bit_writer_.Clear();
593
1.06M
}
594
595
// Copy from https://github.com/apache/impala/blob/master/be/src/util/rle-encoding.h
596
// Utility classes to do run length encoding (RLE) for fixed bit width values.  If runs
597
// are sufficiently long, RLE is used, otherwise, the values are just bit-packed
598
// (literal encoding).
599
//
600
// For both types of runs, there is a byte-aligned indicator which encodes the length
601
// of the run and the type of the run.
602
//
603
// This encoding has the benefit that when there aren't any long enough runs, values
604
// are always decoded at fixed (can be precomputed) bit offsets OR both the value and
605
// the run length are byte aligned. This allows for very efficient decoding
606
// implementations.
607
// The encoding is:
608
//    encoded-block := run*
609
//    run := literal-run | repeated-run
610
//    literal-run := literal-indicator < literal bytes >
611
//    repeated-run := repeated-indicator < repeated value. padded to byte boundary >
612
//    literal-indicator := varint_encode( number_of_groups << 1 | 1)
613
//    repeated-indicator := varint_encode( number_of_repetitions << 1 )
614
//
615
// Each run is preceded by a varint. The varint's least significant bit is
616
// used to indicate whether the run is a literal run or a repeated run. The rest
617
// of the varint is used to determine the length of the run (eg how many times the
618
// value repeats).
619
//
620
// In the case of literal runs, the run length is always a multiple of 8 (i.e. encode
621
// in groups of 8), so that no matter the bit-width of the value, the sequence will end
622
// on a byte boundary without padding.
623
// Given that we know it is a multiple of 8, we store the number of 8-groups rather than
624
// the actual number of encoded ints. (This means that the total number of encoded values
625
// can not be determined from the encoded data, since the number of values in the last
626
// group may not be a multiple of 8). For the last group of literal runs, we pad
627
// the group to 8 with zeros. This allows for 8 at a time decoding on the read side
628
// without the need for additional checks.
629
//
630
// There is a break-even point when it is more storage efficient to do run length
631
// encoding.  For 1 bit-width values, that point is 8 values.  They require 2 bytes
632
// for both the repeated encoding or the literal encoding.  This value can always
633
// be computed based on the bit-width.
634
// TODO: For 1 bit-width values it can be optimal to use 16 or 24 values, but more
635
// investigation is needed to do this efficiently, see the reverted IMPALA-6658.
636
// TODO: think about how to use this for strings.  The bit packing isn't quite the same.
637
//
638
// Examples with bit-width 1 (eg encoding booleans):
639
// ----------------------------------------
640
// 100 1s followed by 100 0s:
641
// <varint(100 << 1)> <1, padded to 1 byte> <varint(100 << 1)> <0, padded to 1 byte>
642
//  - (total 4 bytes)
643
//
644
// alternating 1s and 0s (200 total):
645
// 200 ints = 25 groups of 8
646
// <varint((25 << 1) | 1)> <25 bytes of values, bitpacked>
647
// (total 26 bytes, 1 byte overhead)
648
649
// RLE decoder with a batch-oriented interface that enables fast decoding.
650
// Users of this class must first initialize the class to point to a buffer of
651
// RLE-encoded data, passed into the constructor or Reset(). The provided
652
// bit_width must be at most min(sizeof(T) * 8, BatchedBitReader::MAX_BITWIDTH).
653
// Then they can decode data by checking NextNumRepeats()/NextNumLiterals() to
654
// see if the next run is a repeated or literal run, then calling
655
// GetRepeatedValue() or GetLiteralValues() respectively to read the values.
656
//
657
// End-of-input is signalled by NextNumRepeats() == NextNumLiterals() == 0.
658
// Other decoding errors are signalled by functions returning false. If an
659
// error is encountered then it is not valid to read any more data until
660
// Reset() is called.
661
662
//bit-packed-run-len and rle-run-len must be in the range [1, 2^31 - 1].
663
// This means that a Parquet implementation can always store the run length in a signed 32-bit integer.
664
template <typename T>
665
class RleBatchDecoder {
666
public:
667
339k
    RleBatchDecoder(uint8_t* buffer, int buffer_len, int bit_width) {
668
339k
        Reset(buffer, buffer_len, bit_width);
669
339k
    }
670
671
    RleBatchDecoder() = default;
672
673
    // Reset the decoder to read from a new buffer.
674
    void Reset(uint8_t* buffer, int buffer_len, int bit_width);
675
676
    // Return the size of the current repeated run. Returns zero if the current run is
677
    // a literal run or if no more runs can be read from the input.
678
    int32_t NextNumRepeats();
679
680
    // Get the value of the current repeated run and consume the given number of repeats.
681
    // Only valid to call when NextNumRepeats() > 0. The given number of repeats cannot
682
    // be greater than the remaining number of repeats in the run. 'num_repeats_to_consume'
683
    // can be set to 0 to peek at the value without consuming repeats.
684
    T GetRepeatedValue(int32_t num_repeats_to_consume);
685
686
    // Return the size of the current literal run. Returns zero if the current run is
687
    // a repeated run or if no more runs can be read from the input.
688
    int32_t NextNumLiterals();
689
690
    // Consume 'num_literals_to_consume' literals from the current literal run,
691
    // copying the values to 'values'. 'num_literals_to_consume' must be <=
692
    // NextNumLiterals(). Returns true if the requested number of literals were
693
    // successfully read or false if an error was encountered, e.g. the input was
694
    // truncated.
695
    bool GetLiteralValues(int32_t num_literals_to_consume, T* values) WARN_UNUSED_RESULT;
696
697
    // Consume 'num_values_to_consume' values and copy them to 'values'.
698
    // Returns the number of consumed values or 0 if an error occurred.
699
    uint32_t GetBatch(T* values, uint32_t batch_num);
700
701
private:
702
    // Called when both 'literal_count_' and 'repeat_count_' have been exhausted.
703
    // Sets either 'literal_count_' or 'repeat_count_' to the size of the next literal
704
    // or repeated run, or leaves both at 0 if no more values can be read (either because
705
    // the end of the input was reached or an error was encountered decoding).
706
    void NextCounts();
707
708
    /// Fill the literal buffer. Invalid to call if there are already buffered literals.
709
    /// Return false if the input was truncated. This does not advance 'literal_count_'.
710
    bool FillLiteralBuffer() WARN_UNUSED_RESULT;
711
712
4.04M
    bool HaveBufferedLiterals() const { return literal_buffer_pos_ < num_buffered_literals_; }
713
714
    /// Output buffered literals, advancing 'literal_buffer_pos_' and decrementing
715
    /// 'literal_count_'. Returns the number of literals outputted.
716
    int32_t OutputBufferedLiterals(int32_t max_to_output, T* values);
717
718
    BatchedBitReader bit_reader_;
719
720
    // Number of bits needed to encode the value. Must be between 0 and 64 after
721
    // the decoder is initialized with a buffer. -1 indicates the decoder was not
722
    // initialized.
723
    int bit_width_ = -1;
724
725
    // If a repeated run, the number of repeats remaining in the current run to be read.
726
    // If the current run is a literal run, this is 0.
727
    int32_t repeat_count_ = 0;
728
729
    // If a literal run, the number of literals remaining in the current run to be read.
730
    // If the current run is a repeated run, this is 0.
731
    int32_t literal_count_ = 0;
732
733
    // If a repeated run, the current repeated value.
734
    T repeated_value_;
735
736
    // Size of buffer for literal values. Large enough to decode a full batch of 32
737
    // literals. The buffer is needed to allow clients to read in batches that are not
738
    // multiples of 32.
739
    static constexpr int LITERAL_BUFFER_LEN = 32;
740
741
    // Buffer containing 'num_buffered_literals_' values. 'literal_buffer_pos_' is the
742
    // position of the next literal to be read from the buffer.
743
    T literal_buffer_[LITERAL_BUFFER_LEN];
744
    int num_buffered_literals_ = 0;
745
    int literal_buffer_pos_ = 0;
746
};
747
748
template <typename T>
749
3.72M
int32_t RleBatchDecoder<T>::OutputBufferedLiterals(int32_t max_to_output, T* values) {
750
3.72M
    int32_t num_to_output =
751
3.72M
            std::min<int32_t>(max_to_output, num_buffered_literals_ - literal_buffer_pos_);
752
3.72M
    memcpy(values, &literal_buffer_[literal_buffer_pos_], sizeof(T) * num_to_output);
753
3.72M
    literal_buffer_pos_ += num_to_output;
754
3.72M
    literal_count_ -= num_to_output;
755
3.72M
    return num_to_output;
756
3.72M
}
757
758
template <typename T>
759
339k
void RleBatchDecoder<T>::Reset(uint8_t* buffer, int buffer_len, int bit_width) {
760
339k
    bit_reader_.Reset(buffer, buffer_len);
761
339k
    bit_width_ = bit_width;
762
339k
    repeat_count_ = 0;
763
339k
    literal_count_ = 0;
764
339k
    num_buffered_literals_ = 0;
765
339k
    literal_buffer_pos_ = 0;
766
339k
}
767
768
template <typename T>
769
7.40M
int32_t RleBatchDecoder<T>::NextNumRepeats() {
770
7.40M
    if (repeat_count_ > 0) return repeat_count_;
771
7.40M
    if (literal_count_ == 0) NextCounts();
772
7.40M
    return repeat_count_;
773
7.40M
}
774
775
template <typename T>
776
7.33M
void RleBatchDecoder<T>::NextCounts() {
777
    // Read the next run's indicator int, it could be a literal or repeated run.
778
    // The int is encoded as a ULEB128-encoded value.
779
7.33M
    uint32_t indicator_value = 0;
780
7.33M
    if (UNLIKELY(!bit_reader_.GetUleb128<uint32_t>(&indicator_value))) {
781
0
        return;
782
0
    }
783
784
    // lsb indicates if it is a literal run or repeated run
785
7.33M
    bool is_literal = indicator_value & 1;
786
787
    // Don't try to handle run lengths that don't fit in an int32_t - just fail gracefully.
788
    // The Parquet standard does not allow longer runs - see PARQUET-1290.
789
7.33M
    uint32_t run_len = indicator_value >> 1;
790
7.33M
    if (is_literal) {
791
        // Use int64_t to avoid overflowing multiplication.
792
3.98M
        int64_t literal_count = static_cast<int64_t>(run_len) * 8;
793
3.98M
        if (UNLIKELY(literal_count > std::numeric_limits<int32_t>::max())) return;
794
3.98M
        literal_count_ = cast_set<int32_t>(literal_count);
795
3.98M
    } else {
796
3.35M
        if (UNLIKELY(run_len == 0)) return;
797
3.35M
        bool result = bit_reader_.GetBytes<T>(BitUtil::Ceil(bit_width_, 8), &repeated_value_);
798
3.35M
        if (UNLIKELY(!result)) return;
799
3.35M
        repeat_count_ = run_len;
800
3.35M
    }
801
7.33M
}
802
803
template <typename T>
804
3.36M
T RleBatchDecoder<T>::GetRepeatedValue(int32_t num_repeats_to_consume) {
805
3.36M
    repeat_count_ -= num_repeats_to_consume;
806
3.36M
    return repeated_value_;
807
3.36M
}
808
809
template <typename T>
810
4.04M
int32_t RleBatchDecoder<T>::NextNumLiterals() {
811
4.04M
    if (literal_count_ > 0) return literal_count_;
812
18.4E
    if (repeat_count_ == 0) NextCounts();
813
18.4E
    return literal_count_;
814
4.04M
}
815
816
template <typename T>
817
4.04M
bool RleBatchDecoder<T>::GetLiteralValues(int32_t num_literals_to_consume, T* values) {
818
4.04M
    int32_t num_consumed = 0;
819
    // Copy any buffered literals left over from previous calls.
820
4.04M
    if (HaveBufferedLiterals()) {
821
48.7k
        num_consumed = OutputBufferedLiterals(num_literals_to_consume, values);
822
48.7k
    }
823
824
4.04M
    int32_t num_remaining = num_literals_to_consume - num_consumed;
825
    // Copy literals directly to the output, bypassing 'literal_buffer_' when possible.
826
    // Need to round to a batch of 32 if the caller is consuming only part of the current
827
    // run avoid ending on a non-byte boundary.
828
4.04M
    int32_t num_to_bypass =
829
4.04M
            std::min<int32_t>(literal_count_, BitUtil::RoundDownToPowerOf2(num_remaining, 32));
830
4.04M
    if (num_to_bypass > 0) {
831
2.50M
        int num_read = bit_reader_.UnpackBatch(bit_width_, num_to_bypass, values + num_consumed);
832
        // If we couldn't read the expected number, that means the input was truncated.
833
2.50M
        if (num_read < num_to_bypass) return false;
834
2.50M
        literal_count_ -= num_to_bypass;
835
2.50M
        num_consumed += num_to_bypass;
836
2.50M
        num_remaining = num_literals_to_consume - num_consumed;
837
2.50M
    }
838
839
4.04M
    if (num_remaining > 0) {
840
        // We weren't able to copy all the literals requested directly from the input.
841
        // Buffer literals and copy over the requested number.
842
3.67M
        if (UNLIKELY(!FillLiteralBuffer())) return false;
843
3.67M
        OutputBufferedLiterals(num_remaining, values + num_consumed);
844
3.67M
    }
845
4.04M
    return true;
846
4.04M
}
847
848
template <typename T>
849
3.67M
bool RleBatchDecoder<T>::FillLiteralBuffer() {
850
3.67M
    int32_t num_to_buffer = std::min<int32_t>(LITERAL_BUFFER_LEN, literal_count_);
851
3.67M
    num_buffered_literals_ = bit_reader_.UnpackBatch(bit_width_, num_to_buffer, literal_buffer_);
852
    // If we couldn't read the expected number, that means the input was truncated.
853
3.67M
    if (UNLIKELY(num_buffered_literals_ < num_to_buffer)) return false;
854
3.67M
    literal_buffer_pos_ = 0;
855
3.67M
    return true;
856
3.67M
}
857
858
template <typename T>
859
424k
uint32_t RleBatchDecoder<T>::GetBatch(T* values, uint32_t batch_num) {
860
424k
    uint32_t num_consumed = 0;
861
7.83M
    while (num_consumed < batch_num) {
862
        // Add RLE encoded values by repeating the current value this number of times.
863
7.40M
        uint32_t num_repeats = NextNumRepeats();
864
7.40M
        if (num_repeats > 0) {
865
3.36M
            int32_t num_repeats_to_set = std::min(num_repeats, batch_num - num_consumed);
866
3.36M
            T repeated_value = GetRepeatedValue(num_repeats_to_set);
867
61.2M
            for (int i = 0; i < num_repeats_to_set; ++i) {
868
57.9M
                values[num_consumed + i] = repeated_value;
869
57.9M
            }
870
3.36M
            num_consumed += num_repeats_to_set;
871
3.36M
            continue;
872
3.36M
        }
873
874
        // Add remaining literal values, if any.
875
4.03M
        uint32_t num_literals = NextNumLiterals();
876
4.03M
        if (num_literals == 0) {
877
0
            break;
878
0
        }
879
4.03M
        uint32_t num_literals_to_set = std::min(num_literals, batch_num - num_consumed);
880
4.03M
        if (!GetLiteralValues(num_literals_to_set, values + num_consumed)) {
881
0
            return 0;
882
0
        }
883
4.03M
        num_consumed += num_literals_to_set;
884
4.03M
    }
885
424k
    return num_consumed;
886
424k
}
887
} // namespace doris