|
|
|
// Copyright (c) 2018 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/inference/analysis/ir_pass_manager.h"
|
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <unordered_set>
|
|
|
|
#include <vector>
|
|
|
|
#include "paddle/fluid/framework/ir/fuse_pass_base.h"
|
|
|
|
#include "paddle/fluid/framework/ir/graph.h"
|
|
|
|
#include "paddle/fluid/framework/scope.h"
|
|
|
|
#include "paddle/fluid/inference/analysis/argument.h"
|
|
|
|
#include "paddle/fluid/inference/analysis/ir_passes/subgraph_detector.h"
|
|
|
|
#include "paddle/fluid/string/pretty_log.h"
|
|
|
|
|
|
|
|
namespace paddle {
|
|
|
|
namespace inference {
|
|
|
|
namespace analysis {
|
|
|
|
using string::PrettyLogEndl;
|
|
|
|
using string::PrettyLog;
|
|
|
|
using string::Style;
|
|
|
|
|
|
|
|
IRPassManager::IRPassManager(Argument *argument) {
|
|
|
|
ARGUMENT_CHECK_FIELD(argument, main_program);
|
|
|
|
graph_ = std::unique_ptr<Graph>(new Graph(argument->main_program()));
|
|
|
|
if (argument->Has("scope")) {
|
|
|
|
graph_->Set(framework::ir::kParamScopeAttr,
|
|
|
|
new framework::Scope *(
|
|
|
|
const_cast<framework::Scope *>(&argument->scope())));
|
|
|
|
}
|
|
|
|
|
|
|
|
ARGUMENT_CHECK_FIELD(argument, ir_analysis_passes);
|
|
|
|
CreatePasses(argument, argument->ir_analysis_passes());
|
|
|
|
}
|
|
|
|
|
|
|
|
void IRPassManager::CreatePasses(Argument *argument,
|
|
|
|
const std::vector<std::string> &passes) {
|
|
|
|
std::string pre_pass;
|
|
|
|
int pass_num = 0;
|
|
|
|
for (const std::string &pass_name : passes) {
|
|
|
|
auto pass = framework::ir::PassRegistry::Instance().Get(pass_name);
|
|
|
|
|
|
|
|
if (pass_name == "graph_viz_pass") {
|
|
|
|
std::string dot_file_path = std::to_string(pass_num) + "_ir_" +
|
|
|
|
(pre_pass.empty() ? "origin" : pre_pass) +
|
|
|
|
".dot";
|
|
|
|
pass->Set("graph_viz_path", new std::string(std::move(dot_file_path)));
|
|
|
|
pass_num++;
|
|
|
|
} else if (pass_name == "mkldnn_placement_pass") {
|
|
|
|
pass->Set("mkldnn_enabled_op_types",
|
|
|
|
new std::unordered_set<std::string>(
|
|
|
|
argument->mkldnn_enabled_op_types()));
|
|
|
|
} else if (pass_name == "cpu_quantize_placement_pass") {
|
|
|
|
pass->Set("quantize_enabled_op_types",
|
|
|
|
new std::unordered_set<std::string>(
|
|
|
|
argument->quantize_enabled_op_types()));
|
|
|
|
pass->Set(
|
|
|
|
"quantize_excluded_op_ids",
|
|
|
|
new std::unordered_set<int>(argument->quantize_excluded_op_ids()));
|
|
|
|
} else if (pass_name == "cpu_quantize_pass") {
|
|
|
|
pass->Set("quant_var_scales",
|
|
|
|
new VarQuantScale(argument->quant_var_scales()));
|
|
|
|
} else if (pass_name == "tensorrt_subgraph_pass") {
|
|
|
|
pass->Set("workspace_size", new int(argument->tensorrt_workspace_size()));
|
|
|
|
pass->Set("max_batch_size", new int(argument->tensorrt_max_batch_size()));
|
|
|
|
pass->Set("min_subgraph_size",
|
|
|
|
new int(argument->tensorrt_min_subgraph_size()));
|
|
|
|
pass->Set("program",
|
|
|
|
new framework::ProgramDesc *(&argument->main_program()));
|
|
|
|
|
|
|
|
bool enable_int8 = argument->tensorrt_precision_mode() ==
|
|
|
|
AnalysisConfig::Precision::kInt8;
|
|
|
|
|
|
|
|
pass->Set("enable_int8", new bool(enable_int8));
|
|
|
|
std::string model_opt_cache_dir =
|
|
|
|
argument->Has("model_dir")
|
|
|
|
? argument->model_dir()
|
|
|
|
: GetDirRoot(argument->model_program_path());
|
|
|
|
pass->Set(
|
|
|
|
"model_opt_cache_dir",
|
|
|
|
new std::string(GetOrCreateModelOptCacheDir(model_opt_cache_dir)));
|
|
|
|
pass->Set("gpu_device_id", new int(argument->gpu_device_id()));
|
|
|
|
pass->Set("use_static_engine",
|
|
|
|
new bool(argument->tensorrt_use_static_engine()));
|
|
|
|
}
|
|
|
|
|
|
|
|
pre_pass = pass_name;
|
|
|
|
|
|
|
|
passes_.emplace_back(std::move(pass));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<Graph> IRPassManager::Apply(std::unique_ptr<Graph> graph) {
|
|
|
|
if (passes_.empty()) {
|
|
|
|
return graph;
|
|
|
|
}
|
|
|
|
PADDLE_ENFORCE(graph.get());
|
|
|
|
// Apply all the passes
|
|
|
|
for (const auto &pass : passes_) {
|
|
|
|
if (pass->Type() != "graph_viz_pass") {
|
|
|
|
PrettyLogEndl(Style::H2(), "--- Running IR pass [%s]", pass->Type());
|
|
|
|
}
|
|
|
|
graph = pass->Apply(std::move(graph));
|
|
|
|
}
|
Clang build fixes (#15628)
* Remove some superfluous std::move calls
The std:move triggered a build error (with -Werror):
```
[ 9%] Building CXX object paddle/fluid/memory/allocation/CMakeFiles/allocator_facade.dir/allocator_facade.cc.o
/home/tej/code/gbuella_paddle/paddle/fluid/memory/allocation/allocator_facade.cc:86:29: error: moving a temporary object prevents copy elision [-Werror,-Wpessimizing-move]
[this] { return std::move(CreateAllocatorWithChunk()); }, capacity);
^
/home/tej/code/gbuella_paddle/paddle/fluid/memory/allocation/allocator_facade.cc:86:29: note: remove std::move call here
[this] { return std::move(CreateAllocatorWithChunk()); }, capacity);
^~~~~~~~~~ ~
1 error generated.
```
See: https://reviews.llvm.org/D7633
* Remove a superfluous lambda capture from framework/operator.h
```
[ 10%] Building CXX object paddle/fluid/platform/CMakeFiles/device_context.dir/init.cc.o
In file included from /home/tej/code/gbuella_paddle/paddle/fluid/platform/init.cc:19:
/home/tej/code/gbuella_paddle/paddle/fluid/framework/operator.h:229:21: error: lambda capture 'this' is not used [-Werror,-Wunused-lambda-capture]
[this](Variable* var) { return var; });
^~~~
1 error generated.
```
Changing it to `return it->second;`, as is in the function below.
* Rethrow an exception (instead of copying it)
```
[ 11%] Building CXX object paddle/fluid/framework/CMakeFiles/operator.dir/operator.cc.o
/home/tej/code/gbuella_paddle/paddle/fluid/framework/operator.cc:191:13: error: local variable 'exception' will be copied despite being thrown by name [-Werror,-Wreturn-std-move]
throw exception;
^~~~~~~~~
/home/tej/code/gbuella_paddle/paddle/fluid/framework/operator.cc:191:13: note: call 'std::move' explicitly to avoid copying
throw exception;
^~~~~~~~~
std::move(exception)
```
See https://reviews.llvm.org/D43322 for an explanation of this diagnostic message.
* Remove an unused variable
```
/home/tej/code/gbuella_paddle/paddle/fluid/framework/operator.cc:884:16: error: private field 'scope_' is not used [-Werror,-Wunused-private-field]
const Scope& scope_;
^
```
* struct ComputationOpHandle -> class ComputationOpHandle
```
[ 13%] Building CXX object paddle/fluid/framework/details/CMakeFiles/memory_early_delete_pass.dir/memory_early_delete_pass.cc.o
In file included from /home/tej/code/gbuella_paddle/paddle/fluid/framework/details/memory_early_delete_pass.cc:21:
/home/tej/code/gbuella_paddle/paddle/fluid/framework/details/reference_count_pass_helper.h:30:1: error: class 'ComputationOpHandle' was previously declared as a struct; this is valid, but may result in linker errors under the Microsoft C++ ABI [-Werror,-Wmismatched-tags]
class ComputationOpHandle;
^
/home/tej/code/gbuella_paddle/paddle/fluid/framework/details/computation_op_handle.h:29:8: note: previous use is here
struct ComputationOpHandle : public OpHandleBase {
^
/home/tej/code/gbuella_paddle/paddle/fluid/framework/details/reference_count_pass_helper.h:30:1: note: did you mean struct here?
class ComputationOpHandle;
^~~~~
struct
1 error generated.
```
* Fix name() methods under fluid/operators
```
In file included from /home/tej/code/gbuella_paddle/paddle/fluid/operators/jit/gen/act.cc:15:
In file included from /home/tej/code/gbuella_paddle/paddle/fluid/operators/jit/gen/act.h:19:
/home/tej/code/gbuella_paddle/paddle/fluid/operators/jit/gen/jitcode.h:71:23: error: 'name' overrides a member function but is not marked 'override' [-Werror,-Winconsistent-missing-override]
virtual const char* name() const = 0;
^
/home/tej/code/gbuella_paddle/paddle/fluid/operators/jit/gen_base.h:31:23: note: overridden virtual function is here
virtual const char* name() const = 0;
^
```
test=develop
6 years ago
|
|
|
return graph;
|
|
|
|
}
|
|
|
|
|
|
|
|
framework::proto::ProgramDesc IRPassManager::AcquireProgram(
|
|
|
|
std::unique_ptr<Graph> *graph, ProgramDesc *program) const {
|
|
|
|
auto pass =
|
|
|
|
framework::ir::PassRegistry::Instance().Get("graph_to_program_pass");
|
|
|
|
|
|
|
|
// Direct using ProgramDesc desc(argument->main_program()) may cause
|
|
|
|
// incomplete copies of information.
|
|
|
|
ProgramDesc desc;
|
|
|
|
desc.CopyFrom(*program->Proto());
|
|
|
|
pass->SetNotOwned("program", &desc);
|
|
|
|
auto *the_graph = graph->release();
|
|
|
|
*graph = pass->Apply(std::unique_ptr<Graph>(the_graph));
|
|
|
|
return *desc.Proto();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace analysis
|
|
|
|
} // namespace inference
|
|
|
|
} // namespace paddle
|