Files
doris/be/src/exec/select_node.cpp
morningman 2419384e8a push 3.3.19 to github (#193)
* push 3.3.19 to github

* merge to 20ed420122a8283200aa37b0a6179b6a571d2837
2018-05-15 20:38:22 +08:00

147 lines
4.8 KiB
C++

// Modifications copyright (C) 2017, Baidu.com, Inc.
// Copyright 2017 The Apache Software Foundation
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
#include "exec/select_node.h"
#include "exprs/expr.h"
#include "gen_cpp/PlanNodes_types.h"
#include "runtime/row_batch.h"
#include "runtime/runtime_state.h"
#include "runtime/raw_value.h"
#include "util/debug_util.h"
namespace palo {
SelectNode::SelectNode(
ObjectPool* pool, const TPlanNode& tnode, const DescriptorTbl& descs)
: ExecNode(pool, tnode, descs),
_child_row_batch(NULL),
_child_row_idx(0),
_child_eos(false) {
}
Status SelectNode::prepare(RuntimeState* state) {
RETURN_IF_ERROR(ExecNode::prepare(state));
_child_row_batch.reset(
new RowBatch(child(0)->row_desc(), state->batch_size(), mem_tracker()));
return Status::OK;
}
Status SelectNode::open(RuntimeState* state) {
RETURN_IF_ERROR(exec_debug_action(TExecNodePhase::OPEN));
RETURN_IF_ERROR(ExecNode::open(state));
RETURN_IF_ERROR(child(0)->open(state));
return Status::OK;
}
Status SelectNode::get_next(RuntimeState* state, RowBatch* row_batch, bool* eos) {
RETURN_IF_ERROR(exec_debug_action(TExecNodePhase::GETNEXT));
RETURN_IF_CANCELLED(state);
SCOPED_TIMER(_runtime_profile->total_time_counter());
if (reached_limit() || (_child_row_idx == _child_row_batch->num_rows() && _child_eos)) {
// we're already done or we exhausted the last child batch and there won't be any
// new ones
_child_row_batch->transfer_resource_ownership(row_batch);
*eos = true;
return Status::OK;
}
*eos = false;
// start (or continue) consuming row batches from child
while (true) {
RETURN_IF_CANCELLED(state);
if (_child_row_idx == _child_row_batch->num_rows()) {
// fetch next batch
_child_row_idx = 0;
_child_row_batch->transfer_resource_ownership(row_batch);
_child_row_batch->reset();
if (row_batch->at_capacity()) {
return Status::OK;
}
RETURN_IF_ERROR(child(0)->get_next(state, _child_row_batch.get(), &_child_eos));
}
if (copy_rows(row_batch)) {
*eos = reached_limit()
|| (_child_row_idx == _child_row_batch->num_rows() && _child_eos);
if (*eos) {
_child_row_batch->transfer_resource_ownership(row_batch);
}
return Status::OK;
}
if (_child_eos) {
// finished w/ last child row batch, and child eos is true
_child_row_batch->transfer_resource_ownership(row_batch);
*eos = true;
return Status::OK;
}
}
return Status::OK;
}
bool SelectNode::copy_rows(RowBatch* output_batch) {
ExprContext** ctxs = &_conjunct_ctxs[0];
int num_ctxs = _conjunct_ctxs.size();
for (; _child_row_idx < _child_row_batch->num_rows(); ++_child_row_idx) {
// Add a new row to output_batch
int dst_row_idx = output_batch->add_row();
if (dst_row_idx == RowBatch::INVALID_ROW_INDEX) {
return true;
}
TupleRow* dst_row = output_batch->get_row(dst_row_idx);
TupleRow* src_row = _child_row_batch->get_row(_child_row_idx);
if (ExecNode::eval_conjuncts(ctxs, num_ctxs, src_row)) {
output_batch->copy_row(src_row, dst_row);
output_batch->commit_last_row();
++_num_rows_returned;
COUNTER_SET(_rows_returned_counter, _num_rows_returned);
if (reached_limit()) {
return true;
}
}
}
if (VLOG_ROW_IS_ON) {
for (int i = 0; i < output_batch->num_rows(); ++i) {
TupleRow* row = output_batch->get_row(i);
VLOG_ROW << "SelectNode input row: " << print_row(row, row_desc());
}
}
return output_batch->is_full() || output_batch->at_resource_limit();
}
Status SelectNode::close(RuntimeState* state) {
if (is_closed()) {
return Status::OK;
}
_child_row_batch.reset();
return ExecNode::close(state);
}
}