/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserved. Licensed 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 "paddle/fluid/operators/beam_search_op.h" #include #include #include "paddle/fluid/framework/op_registry.h" namespace paddle { namespace operators { class BeamSearchOpMaker : public framework::OpProtoAndCheckerMaker { public: void Make() override { // inputs and outputs stored in proto AddInput("pre_ids", "(LoDTensor) The LoDTensor containing the selected ids at the " "previous step. It should be a tensor with shape (batch_size, 1) " "and lod `[[0, 1, ... , batch_size], [0, 1, ..., batch_size]]` at " "the first step."); AddInput("pre_scores", "(LoDTensor) The LoDTensor containing the accumulated " "scores corresponding to the selected ids at the previous step."); AddInput("ids", "(LoDTensor) The LoDTensor containing the candidates ids. Its " "shape should be (batch_size * beam_size, W). If not set, it will " "be calculated out according to Input(scores) in this operator.") .AsDispensable(); AddInput("scores", "(LoDTensor) The LoDTensor containing the current scores " "corresponding to Input(ids). If Input(ids) is not nullptr, its " "shape is the same as that of Input(ids)." "If is_accumulated is true, Input(scores) is accumulated scores " "and will be used derectedly. Else, each score will be " "transformed to the log field and accumulate Input(pre_sores) " "first."); AddOutput("selected_ids", "A LodTensor that stores the IDs selected by beam search."); AddOutput("selected_scores", "A LoDTensor containing the accumulated scores corresponding to " "Output(selected_ids)."); AddOutput("parent_idx", "A Tensor preserving the selected_ids' parent index in pre_ids.") .AsDispensable(); // Attributes stored in AttributeMap AddAttr("level", "the level of LoDTensor"); AddAttr("beam_size", "beam size for beam search"); AddAttr("end_id", "the token id which indicates the end of a sequence"); AddAttr("is_accumulated", "Whether the Input(scores) is accumulated scores.") .SetDefault(true); AddComment(R"DOC( This operator does the search in beams for one time step. Specifically, it selects the top-K candidate word ids of current step from Input(ids) according to their Input(scores) for all source sentences, where K is Attr(beam_size) and Input(ids), Input(scores) are predicted results from the computation cell. Additionally, Input(pre_ids) and Input(pre_scores) are the output of beam_search at previous step, they are needed for special use to handle ended candidate translations. The paths linking prefixes and selected candidates are organized and reserved in lod. Note that the Input(scores) passed in should be accumulated scores, and length penalty should be done with extra operators before calculating the accumulated scores if needed, also suggest finding top-K before it and using the top-K candidates following. )DOC"); } }; class BeamSearchOp : public framework::OperatorWithKernel { public: using framework::OperatorWithKernel::OperatorWithKernel; void InferShape(framework::InferShapeContext *ctx) const override { for (const std::string &arg : std::vector({"pre_ids", "scores"})) { OP_INOUT_CHECK(ctx->HasInput(arg), "Input", arg, "BeamSeach"); } for (const std::string &arg : std::vector({"selected_ids", "selected_scores"})) { OP_INOUT_CHECK(ctx->HasOutput(arg), "Output", arg, "BeamSeach"); } auto id_dims = ctx->GetInputDim("pre_ids"); ctx->SetOutputDim("selected_scores", ctx->GetInputDim("pre_scores")); ctx->SetOutputDim("selected_ids", id_dims); ctx->SetOutputDim("parent_idx", {id_dims[0]}); } protected: framework::OpKernelType GetExpectedKernelType( const framework::ExecutionContext &ctx) const override { auto *scores = ctx.Input("scores"); size_t level = ctx.Attr("level"); size_t batch_size = scores->lod()[level].size() - 1; // The current CUDA kernel only support cases with batch_size < 4. // Compute on CPU for cases with batch_size > 4. if (batch_size <= 4) { return framework::OpKernelType( OperatorWithKernel::IndicateVarDataType(ctx, "pre_ids"), ctx.GetPlace()); } else { return framework::OpKernelType( OperatorWithKernel::IndicateVarDataType(ctx, "pre_ids"), platform::CPUPlace()); } } }; class BeamSearchInferVarType : public framework::VarTypeInference { public: void operator()(framework::InferVarTypeContext *ctx) const override { ctx->SetOutputType("selected_ids", framework::proto::VarType::LOD_TENSOR, framework::ALL_ELEMENTS); ctx->SetOutputType("selected_scores", framework::proto::VarType::LOD_TENSOR, framework::ALL_ELEMENTS); } }; } // namespace operators } // namespace paddle namespace ops = paddle::operators; REGISTER_OPERATOR(beam_search, ops::BeamSearchOp, ops::BeamSearchOpMaker, ops::BeamSearchInferVarType); REGISTER_OP_CPU_KERNEL( beam_search, ops::BeamSearchOpKernel, ops::BeamSearchOpKernel, ops::BeamSearchOpKernel, ops::BeamSearchOpKernel);