Coverage Report

Created: 2026-03-13 14:44

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
be/src/exec/pipeline/task_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/pipeline/task_queue.h"
19
20
// IWYU pragma: no_include <bits/chrono.h>
21
#include <chrono> // IWYU pragma: keep
22
#include <memory>
23
#include <string>
24
25
#include "common/logging.h"
26
#include "exec/pipeline/pipeline_task.h"
27
#include "runtime/workload_group/workload_group.h"
28
29
namespace doris {
30
#include "common/compile_check_begin.h"
31
32
8.58M
PipelineTaskSPtr SubTaskQueue::try_take(bool is_steal) {
33
8.58M
    if (_queue.empty()) {
34
0
        return nullptr;
35
0
    }
36
8.58M
    auto task = _queue.front();
37
8.58M
    _queue.pop();
38
8.58M
    return task;
39
8.58M
}
40
41
////////////////////  PriorityTaskQueue ////////////////////
42
43
3.52k
PriorityTaskQueue::PriorityTaskQueue() : _closed(false) {
44
3.52k
    double factor = 1;
45
24.7k
    for (int i = SUB_QUEUE_LEVEL - 1; i >= 0; i--) {
46
21.1k
        _sub_queues[i].set_level_factor(factor);
47
21.1k
        factor *= LEVEL_QUEUE_TIME_FACTOR;
48
21.1k
    }
49
3.52k
}
50
51
1.51k
void PriorityTaskQueue::close() {
52
1.51k
    std::unique_lock<std::mutex> lock(_work_size_mutex);
53
1.51k
    _closed = true;
54
1.51k
    _wait_task.notify_all();
55
1.51k
    DorisMetrics::instance()->pipeline_task_queue_size->increment(-_total_task_size);
56
1.51k
}
57
58
1.67G
PipelineTaskSPtr PriorityTaskQueue::_try_take_unprotected(bool is_steal) {
59
1.67G
    if (_total_task_size == 0 || _closed) {
60
1.66G
        return nullptr;
61
1.66G
    }
62
63
17.7M
    double min_vruntime = 0;
64
17.7M
    int level = -1;
65
69.2M
    for (int i = 0; i < SUB_QUEUE_LEVEL; ++i) {
66
51.4M
        double cur_queue_vruntime = _sub_queues[i].get_vruntime();
67
51.4M
        if (!_sub_queues[i].empty()) {
68
8.58M
            if (level == -1 || cur_queue_vruntime < min_vruntime) {
69
8.58M
                level = i;
70
8.58M
                min_vruntime = cur_queue_vruntime;
71
8.58M
            }
72
8.58M
        }
73
51.4M
    }
74
17.7M
    DCHECK(level != -1);
75
17.7M
    _queue_level_min_vruntime = uint64_t(min_vruntime);
76
77
17.7M
    auto task = _sub_queues[level].try_take(is_steal);
78
17.7M
    if (task) {
79
8.58M
        task->update_queue_level(level);
80
8.58M
        _total_task_size--;
81
8.58M
        DorisMetrics::instance()->pipeline_task_queue_size->increment(-1);
82
8.58M
    }
83
17.7M
    return task;
84
1.67G
}
85
86
8.58M
int PriorityTaskQueue::_compute_level(uint64_t runtime) {
87
18.4E
    for (int i = 0; i < SUB_QUEUE_LEVEL - 1; ++i) {
88
8.58M
        if (runtime <= _queue_level_limit[i]) {
89
8.58M
            return i;
90
8.58M
        }
91
8.58M
    }
92
18.4E
    return SUB_QUEUE_LEVEL - 1;
93
8.58M
}
94
95
1.64G
PipelineTaskSPtr PriorityTaskQueue::try_take(bool is_steal) {
96
    // TODO other efficient lock? e.g. if get lock fail, return null_ptr
97
1.64G
    std::unique_lock<std::mutex> lock(_work_size_mutex);
98
1.64G
    return _try_take_unprotected(is_steal);
99
1.64G
}
100
101
34.3M
PipelineTaskSPtr PriorityTaskQueue::take(uint32_t timeout_ms) {
102
34.3M
    std::unique_lock<std::mutex> lock(_work_size_mutex);
103
34.3M
    auto task = _try_take_unprotected(false);
104
34.3M
    if (task) {
105
12.5k
        return task;
106
34.2M
    } else {
107
34.3M
        if (timeout_ms > 0) {
108
34.3M
            _wait_task.wait_for(lock, std::chrono::milliseconds(timeout_ms));
109
18.4E
        } else {
110
18.4E
            _wait_task.wait(lock);
111
18.4E
        }
112
34.2M
        return _try_take_unprotected(false);
113
34.2M
    }
114
34.3M
}
115
116
8.58M
Status PriorityTaskQueue::push(PipelineTaskSPtr task) {
117
8.58M
    if (_closed) {
118
0
        return Status::InternalError("WorkTaskQueue closed");
119
0
    }
120
8.58M
    auto level = _compute_level(task->get_runtime_ns());
121
8.58M
    std::unique_lock<std::mutex> lock(_work_size_mutex);
122
123
    // update empty queue's  runtime, to avoid too high priority
124
8.58M
    if (_sub_queues[level].empty() &&
125
8.58M
        double(_queue_level_min_vruntime) > _sub_queues[level].get_vruntime()) {
126
0
        _sub_queues[level].adjust_runtime(_queue_level_min_vruntime);
127
0
    }
128
129
8.58M
    _sub_queues[level].push_back(task);
130
8.58M
    _total_task_size++;
131
8.58M
    DorisMetrics::instance()->pipeline_task_queue_size->increment(1);
132
8.58M
    _wait_task.notify_one();
133
8.58M
    return Status::OK();
134
8.58M
}
135
136
101
MultiCoreTaskQueue::~MultiCoreTaskQueue() = default;
137
138
MultiCoreTaskQueue::MultiCoreTaskQueue(int core_size)
139
167
        : _prio_task_queues(core_size), _closed(false), _core_size(core_size) {}
140
141
76
void MultiCoreTaskQueue::close() {
142
76
    if (_closed) {
143
0
        return;
144
0
    }
145
76
    _closed = true;
146
    // close all priority task queue
147
76
    std::ranges::for_each(_prio_task_queues,
148
1.51k
                          [](auto& prio_task_queue) { prio_task_queue.close(); });
149
76
}
150
151
12.1M
PipelineTaskSPtr MultiCoreTaskQueue::take(int core_id) {
152
12.1M
    PipelineTaskSPtr task = nullptr;
153
43.3M
    while (!_closed) {
154
18.4E
        DCHECK(_prio_task_queues.size() > core_id)
155
18.4E
                << " list size: " << _prio_task_queues.size() << " core_id: " << core_id
156
18.4E
                << " _core_size: " << _core_size << " _next_core: " << _next_core.load();
157
39.7M
        task = _prio_task_queues[core_id].try_take(false);
158
39.7M
        if (task) {
159
2.10M
            break;
160
2.10M
        }
161
37.6M
        task = _steal_take(core_id);
162
37.6M
        if (task) {
163
3.28M
            break;
164
3.28M
        }
165
34.3M
        task = _prio_task_queues[core_id].take(WAIT_CORE_TASK_TIMEOUT_MS /* timeout_ms */);
166
34.3M
        if (task) {
167
3.20M
            break;
168
3.20M
        }
169
34.3M
    }
170
12.1M
    if (task) {
171
8.59M
        task->pop_out_runnable_queue();
172
8.59M
    }
173
12.1M
    return task;
174
12.1M
}
175
176
37.5M
PipelineTaskSPtr MultiCoreTaskQueue::_steal_take(int core_id) {
177
37.5M
    DCHECK(core_id < _core_size);
178
37.5M
    int next_id = core_id;
179
1.65G
    for (int i = 1; i < _core_size; ++i) {
180
1.62G
        ++next_id;
181
1.62G
        if (next_id == _core_size) {
182
34.5M
            next_id = 0;
183
34.5M
        }
184
1.62G
        DCHECK(next_id < _core_size);
185
1.62G
        auto task = _prio_task_queues[next_id].try_take(true);
186
1.62G
        if (task) {
187
3.28M
            return task;
188
3.28M
        }
189
1.62G
    }
190
34.2M
    return nullptr;
191
37.5M
}
192
193
7.66M
Status MultiCoreTaskQueue::push_back(PipelineTaskSPtr task) {
194
7.66M
    int thread_id = task->get_thread_id(_core_size);
195
7.66M
    if (thread_id < 0) {
196
2.17M
        thread_id = _next_core.fetch_add(1) % _core_size;
197
2.17M
    }
198
7.66M
    return push_back(task, thread_id);
199
7.66M
}
200
201
8.57M
Status MultiCoreTaskQueue::push_back(PipelineTaskSPtr task, int core_id) {
202
8.57M
    DCHECK(core_id < _core_size);
203
8.57M
    task->put_in_runnable_queue();
204
8.57M
    return _prio_task_queues[core_id].push(task);
205
8.57M
}
206
207
7.66M
void MultiCoreTaskQueue::update_statistics(PipelineTask* task, int64_t time_spent) {
208
    // if the task not execute but exception early close, core_id == -1
209
    // should not do update_statistics
210
7.67M
    if (auto core_id = task->get_thread_id(_core_size); core_id >= 0) {
211
7.67M
        task->inc_runtime_ns(time_spent);
212
7.67M
        _prio_task_queues[core_id].inc_sub_queue_runtime(task->get_queue_level(), time_spent);
213
7.67M
    }
214
7.66M
}
215
216
} // namespace doris