ir_pass_manager.cc 14.0 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
// 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"
16
#include <map>
17
#include <memory>
18
#include <string>
19
#include <unordered_map>
20 21
#include <unordered_set>
#include <utility>
L
luotao1 已提交
22
#include <vector>
Y
Yan Chunwei 已提交
23
#include "paddle/fluid/framework/ir/fuse_pass_base.h"
24 25
#include "paddle/fluid/framework/ir/graph.h"
#include "paddle/fluid/framework/scope.h"
26
#include "paddle/fluid/inference/analysis/argument.h"
Y
Yan Chunwei 已提交
27
#include "paddle/fluid/string/pretty_log.h"
28 29 30 31

namespace paddle {
namespace inference {
namespace analysis {
Y
Yan Chunwei 已提交
32 33 34
using string::PrettyLogEndl;
using string::PrettyLog;
using string::Style;
35

36 37 38 39
IRPassManager::IRPassManager(Argument *argument) {
  ARGUMENT_CHECK_FIELD(argument, main_program);
  graph_ = std::unique_ptr<Graph>(new Graph(argument->main_program()));
  if (argument->Has("scope")) {
40
    auto *scope_ptr = argument->scope_ptr();
41 42 43
    PADDLE_ENFORCE_NOT_NULL(scope_ptr,
                            platform::errors::PreconditionNotMet(
                                "The scope ptr should not be nullptr."));
44
    graph_->SetNotOwned(framework::ir::kParamScopeAttr, scope_ptr);
45
  }
46
  disable_logs_ = argument->disable_logs();
47 48 49

  ARGUMENT_CHECK_FIELD(argument, ir_analysis_passes);
  CreatePasses(argument, argument->ir_analysis_passes());
50 51
}

52 53
void IRPassManager::CreatePasses(Argument *argument,
                                 const std::vector<std::string> &passes) {
54
  std::string pre_pass;
L
luotao1 已提交
55
  int pass_num = 0;
56
  for (const std::string &pass_name : passes) {
57
    auto pass = framework::ir::PassRegistry::Instance().Get(pass_name);
58
    pass->Set("use_varseqlen", new bool(argument->tensorrt_use_varseqlen()));
59 60
    pass->Set("with_interleaved",
              new bool(argument->tensorrt_with_interleaved()));
61 62 63 64
    pass->Set("tensorrt_transformer_posid",
              new std::string(argument->tensorrt_transformer_posid()));
    pass->Set("tensorrt_transformer_maskid",
              new std::string(argument->tensorrt_transformer_maskid()));
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
    pass->Set("disable_logs", new bool(argument->disable_logs()));
    auto precision_mode = argument->tensorrt_precision_mode();
    bool enable_int8 = precision_mode == AnalysisConfig::Precision::kInt8;
    pass->Set("enable_int8", new bool(enable_int8));
    pass->Set("max_input_shape", new std::map<std::string, std::vector<int>>(
                                     argument->max_input_shape()));
    pass->Set("min_input_shape", new std::map<std::string, std::vector<int>>(
                                     argument->min_input_shape()));
    pass->Set("optim_input_shape", new std::map<std::string, std::vector<int>>(
                                       argument->optim_input_shape()));
    // tuned trt dynamic_shape
    pass->Set("trt_tuned_dynamic_shape",
              new bool(argument->tensorrt_tuned_dynamic_shape()));
    bool with_dynamic_shape = (argument->max_input_shape().size() > 0 &&
                               argument->min_input_shape().size() > 0 &&
                               argument->optim_input_shape().size() > 0) ||
                              argument->tensorrt_tuned_dynamic_shape();
    pass->Set("with_dynamic_shape", new bool(with_dynamic_shape));
83

84
    if (pass_name == "graph_viz_pass") {
85 86 87 88 89 90 91 92 93 94
      std::string optim_cache_dir = argument->optim_cache_dir();
      std::string dot_file_path;
      if (optim_cache_dir.empty()) {
        dot_file_path = std::to_string(pass_num) + "_ir_" +
                        (pre_pass.empty() ? "origin" : pre_pass) + ".dot";
      } else {
        dot_file_path = optim_cache_dir + "/" + std::to_string(pass_num) +
                        "_ir_" + (pre_pass.empty() ? "origin" : pre_pass) +
                        ".dot";
      }
95
      pass->Set("graph_viz_path", new std::string(std::move(dot_file_path)));
96
      pass->Set("optim_cache_dir", new std::string(std::move(optim_cache_dir)));
L
luotao1 已提交
97
      pass_num++;
98
    } else if (pass_name == "mkldnn_placement_pass") {
99 100 101
      pass->Set("mkldnn_enabled_op_types",
                new std::unordered_set<std::string>(
                    argument->mkldnn_enabled_op_types()));
102 103 104
    } else if (pass_name == "cudnn_placement_pass") {
      pass->Set("cudnn_enabled_op_types",
                new std::unordered_set<std::string>());
105
#ifdef PADDLE_WITH_MKLDNN
106 107 108 109 110 111 112
    } 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()));
113
    } else if (pass_name == "cpu_quantize_pass") {
B
baoachun 已提交
114 115 116 117
      if (argument->quantize_enabled_op_types().count("conv2d") ||
          argument->quantize_enabled_op_types().count("depthwise_conv2d")) {
        pass->Set("data_layout", new std::string("NHWC"));
      }
118 119
      pass->Set("quant_var_scales",
                new VarQuantScale(argument->quant_var_scales()));
120 121 122 123
    } else if (pass_name == "cpu_bfloat16_placement_pass") {
      pass->Set("bfloat16_enabled_op_types",
                new std::unordered_set<std::string>(
                    argument->bfloat16_enabled_op_types()));
124
#endif
125
    } else if (pass_name == "tensorrt_subgraph_pass") {
126 127
      pass->Set("workspace_size", new int(argument->tensorrt_workspace_size()));
      pass->Set("max_batch_size", new int(argument->tensorrt_max_batch_size()));
128 129
      pass->Set("min_subgraph_size",
                new int(argument->tensorrt_min_subgraph_size()));
N
nhzlx 已提交
130 131
      pass->Set("program",
                new framework::ProgramDesc *(&argument->main_program()));
132
      pass->Set("predictor_id", new int(argument->predictor_id()));
133 134
      bool use_calib_mode = argument->tensorrt_use_calib_mode();
      pass->Set("use_calib_mode", new bool(use_calib_mode));
Z
Zhaolong Xing 已提交
135 136
      pass->Set("precision_mode",
                new AnalysisConfig::Precision(precision_mode));
137 138 139

      bool use_static_engine = argument->tensorrt_use_static_engine();
      bool model_from_memory = argument->model_from_memory();
140
      std::string optim_cache_dir = argument->optim_cache_dir();
141 142
      bool int8_valid = !(model_from_memory && optim_cache_dir.empty() &&
                          enable_int8 && use_calib_mode);
143 144 145 146 147 148
      PADDLE_ENFORCE_EQ(
          int8_valid, true,
          platform::errors::PreconditionNotMet(
              "When you are in TRT INT8 mode, and load model from "
              "memory, you should set optim_cache_dir using "
              "config.SetOptimCacheDir()"));
149 150 151 152 153 154 155 156 157
      if (model_from_memory && use_static_engine) {
        PADDLE_ENFORCE_EQ(
            optim_cache_dir.empty(), false,
            platform::errors::PreconditionNotMet(
                "When you are using Paddle-TRT, and using load model "
                "from memory, and also set the use_static to true. "
                "you must set optim_cache_dir using "
                "config.SetOptimCacheDir()."));
      }
N
nhzlx 已提交
158

159
      if (!optim_cache_dir.empty()) {
160 161 162 163 164 165 166 167
        if (!PathExists(optim_cache_dir)) {
          PADDLE_ENFORCE_NE(
              MKDIR(optim_cache_dir.c_str()), -1,
              platform::errors::PreconditionNotMet(
                  "Can not create optimize cache directory: %s, Make sure you "
                  "have permission to write",
                  optim_cache_dir));
        }
168 169
        pass->Set("model_opt_cache_dir", new std::string(optim_cache_dir));
      } else if (use_static_engine || enable_int8) {
170 171 172 173 174 175 176 177
        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)));
      }
N
nhzlx 已提交
178
      pass->Set("gpu_device_id", new int(argument->gpu_device_id()));
179
      pass->Set("use_static_engine", new bool(use_static_engine));
180
      pass->Set("model_from_memory", new bool(argument->model_from_memory()));
181
      pass->Set("use_inspector", new bool(argument->tensorrt_use_inspector()));
182 183 184 185 186 187

      // tuned trt dynamic_shape
      pass->Set("trt_shape_range_info_path",
                new std::string(argument->tensorrt_shape_range_info_path()));
      pass->Set("trt_allow_build_at_runtime",
                new bool(argument->tensorrt_allow_build_at_runtime()));
188 189
      pass->Set("trt_disabled_ops", new std::vector<std::string>(
                                        argument->tensorrt_disabled_ops()));
190 191
      pass->Set("trt_use_dla", new bool(argument->tensorrt_use_dla()));
      pass->Set("trt_dla_core", new int(argument->tensorrt_dla_core()));
192
      // Setting the disable_trt_plugin_fp16 to true means that TRT plugin will
193
      // not run fp16.
194 195
      pass->Set("disable_trt_plugin_fp16",
                new bool(argument->disable_trt_plugin_fp16()));
D
denglin-github 已提交
196 197 198 199 200
    } else if (pass_name == "dlnne_subgraph_pass") {
      pass->Set("min_subgraph_size",
                new int(argument->dlnne_min_subgraph_size()));
      pass->Set("program",
                new framework::ProgramDesc *(&argument->main_program()));
201 202 203 204
    } else if (pass_name == "mixed_precision_configure_pass") {
      pass->Set("gpu_fp16_disabled_op_types",
                new std::unordered_set<std::string>(
                    argument->gpu_fp16_disabled_op_types()));
205
    }
石晓伟 已提交
206
    if (pass_name == "lite_subgraph_pass") {
207
      bool lite_enable_int8 =
石晓伟 已提交
208 209 210 211 212 213
          argument->lite_precision_mode() == AnalysisConfig::Precision::kInt8;
      pass->Set("program",
                new framework::ProgramDesc *(&argument->main_program()));
      pass->Set("lite_ops_filter",
                new std::vector<std::string>(argument->lite_ops_filter()));
      pass->Set("predictor_id", new int(argument->predictor_id()));
214 215
      pass->Erase("enable_int8");
      pass->Set("enable_int8", new bool(lite_enable_int8));
石晓伟 已提交
216
      pass->Set("use_gpu", new bool(argument->use_gpu()));
217 218 219 220
      pass->Set("zero_copy", new bool(argument->lite_zero_copy()));
      pass->Set("use_xpu", new bool(argument->use_xpu()));
      pass->Set("xpu_l3_workspace_size",
                new int(argument->xpu_l3_workspace_size()));
W
Wilber 已提交
221 222
      pass->Set("cpu_math_library_num_threads",
                new int(argument->cpu_math_library_num_threads()));
W
Wilber 已提交
223 224 225 226 227 228
      pass->Set("locked", new bool(argument->xpu_locked()));
      pass->Set("autotune", new bool(argument->xpu_autotune()));
      pass->Set("autotune_file",
                new std::string(argument->xpu_autotune_file()));
      pass->Set("precision", new std::string(argument->xpu_precision()));
      pass->Set("adaptive_seqlen", new bool(argument->xpu_adaptive_seqlen()));
229
      pass->Set("xpu_device_id", new int(argument->xpu_device_id()));
230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250
      // NNAdapter Related
      pass->Set("use_nnadapter", new bool(argument->use_nnadapter()));
      pass->Set("nnadapter_model_cache_dir",
                new std::string(argument->nnadapter_model_cache_dir()));
      pass->Set(
          "nnadapter_device_names",
          new std::vector<std::string>(argument->nnadapter_device_names()));
      pass->Set("nnadapter_context_properties",
                new std::string(argument->nnadapter_context_properties()));
      pass->Set("nnadapter_subgraph_partition_config_buffer",
                new std::string(
                    argument->nnadapter_subgraph_partition_config_buffer()));
      pass->Set("nnadapter_subgraph_partition_config_path",
                new std::string(
                    argument->nnadapter_subgraph_partition_config_path()));
      pass->Set("nnadapter_model_cache_buffer",
                new std::vector<std::vector<char>>(
                    argument->nnadapter_model_cache_buffer()));
      pass->Set("nnadapter_model_cache_token",
                new std::vector<std::string>(
                    argument->nnadapter_model_cache_token()));
石晓伟 已提交
251
    }
252 253
    if (pass_name == "fc_fuse_pass") {
      pass->Set("use_gpu", new bool(argument->use_gpu()));
254 255 256 257 258 259 260 261
      bool fc_mkldnn_pass = 0;
      for (const std::string &pass_n : passes) {
        if (pass_n == "fc_mkldnn_pass") {
          fc_mkldnn_pass = 1;
        }
      }
      bool use_fc_padding = !fc_mkldnn_pass && argument->use_fc_padding();
      pass->Set("use_fc_padding", new bool(use_fc_padding));
262
    }
263
    pre_pass = pass_name;
264 265

    passes_.emplace_back(std::move(pass));
266 267 268
  }
}

269 270 271 272
std::unique_ptr<Graph> IRPassManager::Apply(std::unique_ptr<Graph> graph) {
  if (passes_.empty()) {
    return graph;
  }
273 274
  PADDLE_ENFORCE_NOT_NULL(graph.get(), platform::errors::PreconditionNotMet(
                                           "Graph cannot be NULL."));
275 276
  // Apply all the passes
  for (const auto &pass : passes_) {
277
    if (pass->Type() != "graph_viz_pass" && !disable_logs_) {
Y
Yan Chunwei 已提交
278 279
      PrettyLogEndl(Style::H2(), "--- Running IR pass [%s]", pass->Type());
    }
280 281 282 283 284
    // delete_fill_constant_op_pass is not apply under trt dynamic shape
    if (pass->Type() == "delete_fill_constant_op_pass") {
      bool use_dynamic = pass->Get<bool>("with_dynamic_shape");
      if (use_dynamic) continue;
    }
285
    graph.reset(pass->Apply(graph.release()));
286
  }
G
Gabor Buella 已提交
287
  return graph;
288 289 290
}

framework::proto::ProgramDesc IRPassManager::AcquireProgram(
N
nhzlx 已提交
291
    std::unique_ptr<Graph> *graph, ProgramDesc *program) const {
292 293 294
  auto pass =
      framework::ir::PassRegistry::Instance().Get("graph_to_program_pass");

N
nhzlx 已提交
295 296
  // Direct using ProgramDesc desc(argument->main_program()) may cause
  // incomplete copies of information.
N
nhzlx 已提交
297
  ProgramDesc desc;
N
nhzlx 已提交
298
  desc.CopyFrom(*program->Proto());
299 300
  pass->SetNotOwned("program", &desc);
  auto *the_graph = graph->release();
301
  graph->reset(pass->Apply(the_graph));
302 303 304
  return *desc.Proto();
}

305 306 307
}  // namespace analysis
}  // namespace inference
}  // namespace paddle