Coverage Report

Created: 2026-05-15 17:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
be/src/exec/operator/data_queue.cpp
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
#include "exec/operator/data_queue.h"
19
20
#include <glog/logging.h>
21
22
#include <algorithm>
23
#include <utility>
24
25
#include "common/thread_safety_annotations.h"
26
#include "core/block/block.h"
27
#include "exec/pipeline/dependency.h"
28
29
namespace doris {
30
31
13.7k
void SubQueue::try_pop(std::unique_ptr<Block>* output_block) {
32
13.7k
    LockGuard l(queue_lock);
33
13.7k
    if (!blocks.empty()) {
34
6.22k
        *output_block = std::move(blocks.front());
35
6.22k
        blocks.pop_front();
36
6.22k
        bytes_in_queue -= (*output_block)->allocated_bytes();
37
6.22k
        blocks_in_queue -= 1;
38
6.22k
        if (blocks.empty()) {
39
6.16k
            sink_dependency->set_ready();
40
6.16k
        }
41
6.22k
    }
42
13.7k
}
43
44
6.25k
bool SubQueue::try_push(std::unique_ptr<Block> block, std::atomic_uint32_t& total_counter) {
45
6.25k
    LockGuard l(queue_lock);
46
6.25k
    if (is_finished) {
47
8
        return false;
48
8
    }
49
6.24k
    total_counter++;
50
6.24k
    bytes_in_queue += block->allocated_bytes();
51
6.24k
    blocks.emplace_back(std::move(block));
52
6.24k
    blocks_in_queue += 1;
53
6.24k
    if (static_cast<int64_t>(blocks.size()) > max_blocks_in_queue.load()) {
54
71
        sink_dependency->block();
55
71
    }
56
6.24k
    return true;
57
6.25k
}
58
59
bool SubQueue::mark_finished(std::atomic_uint32_t& unfinished_counter,
60
12.1k
                             std::atomic_bool& all_finished) {
61
12.1k
    LockGuard l(queue_lock);
62
12.1k
    if (is_finished) {
63
6.06k
        return false;
64
6.06k
    }
65
6.10k
    is_finished = true;
66
6.10k
    if (unfinished_counter.fetch_sub(1) == 1) {
67
2.89k
        all_finished = true;
68
2.89k
    }
69
6.10k
    return true;
70
12.1k
}
71
72
6.07k
void SubQueue::clear_blocks() {
73
6.07k
    bool need_set_always_ready = false;
74
6.07k
    {
75
6.07k
        LockGuard l(queue_lock);
76
6.07k
        if (!blocks.empty()) {
77
13
            blocks.clear();
78
13
            bytes_in_queue = 0;
79
13
            blocks_in_queue = 0;
80
13
            need_set_always_ready = true;
81
13
        }
82
6.07k
    }
83
    // Notify outside of queue_lock to keep lock ordering simple.
84
6.07k
    if (need_set_always_ready) {
85
13
        sink_dependency->set_always_ready();
86
13
    }
87
6.07k
}
88
89
2.89k
DataQueue::DataQueue(int child_count) : _sub_queues(child_count), _child_count(child_count) {
90
6.12k
    for (auto& sub : _sub_queues) {
91
6.12k
        sub = std::make_unique<SubQueue>();
92
6.12k
    }
93
2.89k
    _un_finished_counter = child_count;
94
2.89k
}
95
96
3
bool DataQueue::has_more_data() const {
97
3
    return _cur_blocks_total_nums.load() > 0;
98
3
}
99
100
void DataQueue::set_source_dependency(std::shared_ptr<Dependency> source_dependency)
101
2.89k
        NO_THREAD_SAFETY_ANALYSIS {
102
2.89k
    _source_dependency = std::move(source_dependency);
103
2.89k
}
104
105
6.10k
void DataQueue::set_sink_dependency(Dependency* sink_dependency, int child_idx) {
106
6.10k
    _sub_queues[child_idx]->sink_dependency = sink_dependency;
107
6.10k
}
108
109
6.09k
void DataQueue::set_max_blocks_in_sub_queue(int64_t max_blocks) {
110
15.4k
    for (auto& sub : _sub_queues) {
111
15.4k
        sub->max_blocks_in_queue = max_blocks;
112
15.4k
    }
113
6.09k
}
114
115
1
void DataQueue::set_low_memory_mode() {
116
1
    _is_low_memory_mode = true;
117
3
    for (auto& sub : _sub_queues) {
118
3
        sub->max_blocks_in_queue = 1;
119
3
    }
120
1
    clear_free_blocks();
121
1
}
122
123
6.08k
std::unique_ptr<Block> DataQueue::get_free_block(int child_idx) {
124
6.08k
    auto& sub = *_sub_queues[child_idx];
125
6.08k
    {
126
6.08k
        LockGuard l(sub.free_lock);
127
6.08k
        if (!sub.free_blocks.empty()) {
128
5
            auto block = std::move(sub.free_blocks.front());
129
5
            sub.free_blocks.pop_front();
130
5
            return block;
131
5
        }
132
6.08k
    }
133
134
6.08k
    return Block::create_unique();
135
6.08k
}
136
137
6.06k
void DataQueue::push_free_block(std::unique_ptr<Block> block, int child_idx) {
138
6.06k
    DCHECK(block->rows() == 0);
139
140
6.06k
    if (!_is_low_memory_mode) {
141
6.06k
        auto& sub = *_sub_queues[child_idx];
142
6.06k
        LockGuard l(sub.free_lock);
143
6.06k
        sub.free_blocks.emplace_back(std::move(block));
144
6.06k
    }
145
6.06k
}
146
147
2.87k
void DataQueue::clear_free_blocks() {
148
6.08k
    for (auto& sub : _sub_queues) {
149
6.08k
        LockGuard l(sub->free_lock);
150
6.08k
        std::deque<std::unique_ptr<Block>> tmp_queue;
151
6.08k
        sub->free_blocks.swap(tmp_queue);
152
6.08k
    }
153
2.87k
}
154
155
2.87k
void DataQueue::terminate() {
156
8.95k
    for (int i = 0; i < _child_count; ++i) {
157
6.08k
        set_finish(i);
158
6.08k
        _sub_queues[i]->clear_blocks();
159
6.08k
    }
160
2.87k
    clear_free_blocks();
161
2.87k
}
162
163
//check which queue have data, and save the idx in _flag_queue_idx,
164
//so next loop, will check the record idx + 1 first
165
//maybe it's useful with many queue, others maybe always 0
166
19.3k
bool DataQueue::remaining_has_data() {
167
19.3k
    int count = _child_count;
168
55.3k
    while (--count >= 0) {
169
42.1k
        _flag_queue_idx++;
170
42.1k
        if (_flag_queue_idx == _child_count) {
171
16.3k
            _flag_queue_idx = 0;
172
16.3k
        }
173
42.1k
        if (_sub_queues[_flag_queue_idx]->blocks_in_queue.load() > 0) {
174
6.21k
            return true;
175
6.21k
        }
176
42.1k
    }
177
13.1k
    return false;
178
19.3k
}
179
180
//the _flag_queue_idx indicate which queue has data, and in check can_read
181
//will be set idx in remaining_has_data function
182
13.6k
Status DataQueue::get_block_from_queue(std::unique_ptr<Block>* output_block, int* child_idx) {
183
13.6k
    const int idx = _flag_queue_idx;
184
13.6k
    auto& sub = *_sub_queues[idx];
185
186
13.6k
    sub.try_pop(output_block);
187
13.6k
    if (*output_block) {
188
6.22k
        if (child_idx) {
189
6.21k
            *child_idx = idx;
190
6.21k
        }
191
6.22k
        auto old_total = _cur_blocks_total_nums.fetch_sub(1);
192
6.22k
        if (old_total == 1) {
193
5.74k
            set_source_block();
194
5.74k
        }
195
6.22k
    }
196
13.6k
    return Status::OK();
197
13.6k
}
198
199
6.23k
Status DataQueue::push_block(std::unique_ptr<Block> block, int child_idx) {
200
6.23k
    if (!block) {
201
0
        return Status::OK();
202
0
    }
203
6.23k
    auto& sub = *_sub_queues[child_idx];
204
    // total_counter is incremented inside try_push under queue_lock, only when the
205
    // block is actually enqueued. This ensures get_block_from_queue() always observes
206
    // _cur_blocks_total_nums >= 1 when it successfully pops a block, with no risk of
207
    // underflow or the need for a rollback on failure.
208
6.23k
    if (!sub.try_push(std::move(block), _cur_blocks_total_nums)) {
209
7
        return Status::EndOfFile("SubQueue already finished");
210
7
    }
211
6.23k
    set_source_ready();
212
6.23k
    return Status::OK();
213
6.23k
}
214
215
12.1k
void DataQueue::set_finish(int child_idx) {
216
12.1k
    auto& sub = *_sub_queues[child_idx];
217
12.1k
    if (!sub.mark_finished(_un_finished_counter, _is_all_finished)) {
218
6.06k
        return;
219
6.06k
    }
220
6.10k
    set_source_ready();
221
6.10k
}
222
223
16.0k
bool DataQueue::is_all_finish() {
224
16.0k
    return _is_all_finished;
225
16.0k
}
226
227
12.3k
void DataQueue::set_source_ready() {
228
12.3k
    LockGuard lc(_source_lock);
229
12.3k
    if (_source_dependency) {
230
12.3k
        _source_dependency->set_ready();
231
12.3k
    }
232
12.3k
}
233
234
5.74k
void DataQueue::set_source_block() {
235
    // Re-check under _source_lock to avoid blocking the source when a concurrent push
236
    // has already added new blocks (or all children have finished) since we observed
237
    // the counter drop to zero.
238
5.74k
    LockGuard lc(_source_lock);
239
5.74k
    if (_source_dependency && _cur_blocks_total_nums == 0 && !is_all_finish()) {
240
2.89k
        _source_dependency->block();
241
2.89k
    }
242
5.74k
}
243
244
} // namespace doris