paddlex.cpp 29.6 KB
Newer Older
C
Channingss 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
//   Copyright (c) 2020 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.
S
syyxsxx 已提交
14 15

#include <math.h>
J
jack 已提交
16
#include <omp.h>
J
jack 已提交
17
#include <algorithm>
J
jack 已提交
18
#include <fstream>
J
jack 已提交
19
#include <cstring>
J
jack 已提交
20
#include "include/paddlex/paddlex.h"
C
Channingss 已提交
21 22 23 24
namespace PaddleX {

void Model::create_predictor(const std::string& model_dir,
                             bool use_gpu,
C
Channingss 已提交
25
                             bool use_trt,
S
syyxsxx 已提交
26
                             bool use_mkl,
C
Channingss 已提交
27
                             int gpu_id,
J
jack 已提交
28
                             std::string key,
S
syyxsxx 已提交
29
                             int thread_num,
J
jack 已提交
30
                             bool use_ir_optim) {
C
Channingss 已提交
31 32 33
  paddle::AnalysisConfig config;
  std::string model_file = model_dir + OS_PATH_SEP + "__model__";
  std::string params_file = model_dir + OS_PATH_SEP + "__params__";
J
jack 已提交
34
  std::string yaml_file = model_dir + OS_PATH_SEP + "model.yml";
J
jack 已提交
35
  std::string yaml_input = "";
C
Channingss 已提交
36
#ifdef WITH_ENCRYPTION
J
jack 已提交
37
  if (key != "") {
F
FlyingQianMM 已提交
38 39
    model_file = model_dir + OS_PATH_SEP + "__model__.encrypted";
    params_file = model_dir + OS_PATH_SEP + "__params__.encrypted";
J
jack 已提交
40
    yaml_file = model_dir + OS_PATH_SEP + "model.yml.encrypted";
J
jack 已提交
41 42
    paddle_security_load_model(
        &config, key.c_str(), model_file.c_str(), params_file.c_str());
J
jack 已提交
43
    yaml_input = decrypt_file(yaml_file.c_str(), key.c_str());
C
Channingss 已提交
44 45
  }
#endif
J
jack 已提交
46
  if (yaml_input == "") {
S
syyxsxx 已提交
47
    // read yaml file
J
jack 已提交
48 49 50 51 52 53 54
    std::ifstream yaml_fin(yaml_file);
    yaml_fin.seekg(0, std::ios::end);
    size_t yaml_file_size = yaml_fin.tellg();
    yaml_input.assign(yaml_file_size, ' ');
    yaml_fin.seekg(0);
    yaml_fin.read(&yaml_input[0], yaml_file_size);
  }
S
syyxsxx 已提交
55
  // load yaml file
J
jack 已提交
56
  if (!load_config(yaml_input)) {
J
jack 已提交
57 58 59 60
    std::cerr << "Parse file 'model.yml' failed!" << std::endl;
    exit(-1);
  }

J
jack 已提交
61
  if (key == "") {
C
Channingss 已提交
62 63
    config.SetModel(model_file, params_file);
  }
S
syyxsxx 已提交
64 65 66 67
  if (use_mkl && name != "HRNet" && name != "DeepLabv3p") {
    config.EnableMKLDNN();
    config.SetCpuMathLibraryNumThreads(12);
  }
C
Channingss 已提交
68 69 70 71 72 73 74
  if (use_gpu) {
    config.EnableUseGpu(100, gpu_id);
  } else {
    config.DisableGpu();
  }
  config.SwitchUseFeedFetchOps(false);
  config.SwitchSpecifyInputNames(true);
S
syyxsxx 已提交
75
  // enable graph Optim
F
FlyingQianMM 已提交
76 77 78
#if defined(__arm__) || defined(__aarch64__)
  config.SwitchIrOptim(false);
#else
J
jack 已提交
79
  config.SwitchIrOptim(use_ir_optim);
F
FlyingQianMM 已提交
80
#endif
S
syyxsxx 已提交
81
  // enable Memory Optim
C
Channingss 已提交
82
  config.EnableMemoryOptim();
C
Channingss 已提交
83 84 85 86 87 88 89 90
  if (use_trt) {
    config.EnableTensorRtEngine(
        1 << 20 /* workspace_size*/,
        32 /* max_batch_size*/,
        20 /* min_subgraph_size*/,
        paddle::AnalysisConfig::Precision::kFloat32 /* precision*/,
        true /* use_static*/,
        false /* use_calib_mode*/);
C
Channingss 已提交
91
  }
C
Channingss 已提交
92 93 94
  predictor_ = std::move(CreatePaddlePredictor(config));
}

J
jack 已提交
95 96
bool Model::load_config(const std::string& yaml_input) {
  YAML::Node config = YAML::Load(yaml_input);
C
Channingss 已提交
97 98
  type = config["_Attributes"]["model_type"].as<std::string>();
  name = config["Model"].as<std::string>();
F
FlyingQianMM 已提交
99 100
  std::string version = config["version"].as<std::string>();
  if (version[0] == '0') {
J
jack 已提交
101 102 103 104 105
    std::cerr << "[Init] Version of the loaded model is lower than 1.0.0, "
              << "deployment cannot be done, please refer to "
              << "https://github.com/PaddlePaddle/PaddleX/blob/develop/docs"
              << "/tutorials/deploy/upgrade_version.md "
              << "to transfer version." << std::endl;
F
FlyingQianMM 已提交
106 107
    return false;
  }
C
Channingss 已提交
108 109 110 111 112 113 114 115 116 117 118
  bool to_rgb = true;
  if (config["TransformsMode"].IsDefined()) {
    std::string mode = config["TransformsMode"].as<std::string>();
    if (mode == "BGR") {
      to_rgb = false;
    } else if (mode != "RGB") {
      std::cerr << "[Init] Only 'RGB' or 'BGR' is supported for TransformsMode"
                << std::endl;
      return false;
    }
  }
S
syyxsxx 已提交
119
  // build data preprocess stream
C
Channingss 已提交
120
  transforms_.Init(config["Transforms"], to_rgb);
S
syyxsxx 已提交
121
  // read label list
C
Channingss 已提交
122 123 124 125 126 127 128 129 130 131
  labels.clear();
  for (const auto& item : config["_Attributes"]["labels"]) {
    int index = labels.size();
    labels[index] = item.as<std::string>();
  }
  return true;
}

bool Model::preprocess(const cv::Mat& input_im, ImageBlob* blob) {
  cv::Mat im = input_im.clone();
132
  if (!transforms_.Run(&im, blob)) {
C
Channingss 已提交
133 134 135 136 137
    return false;
  }
  return true;
}

J
jack 已提交
138
// use openmp
J
jack 已提交
139 140 141
bool Model::preprocess(const std::vector<cv::Mat>& input_im_batch,
                       std::vector<ImageBlob>* blob_batch,
                       int thread_num) {
J
jack 已提交
142
  int batch_size = input_im_batch.size();
J
jack 已提交
143
  bool success = true;
J
jack 已提交
144 145
  thread_num = std::min(thread_num, batch_size);
  #pragma omp parallel for num_threads(thread_num)
J
jack 已提交
146
  for (int i = 0; i < input_im_batch.size(); ++i) {
J
jack 已提交
147
    cv::Mat im = input_im_batch[i].clone();
J
jack 已提交
148
    if (!transforms_.Run(&im, &(*blob_batch)[i])) {
J
jack 已提交
149 150 151 152 153 154
      success = false;
    }
  }
  return success;
}

C
Channingss 已提交
155 156 157 158 159
bool Model::predict(const cv::Mat& im, ClsResult* result) {
  inputs_.clear();
  if (type == "detector") {
    std::cerr << "Loading model is a 'detector', DetResult should be passed to "
                 "function predict()!"
J
jack 已提交
160
                 "to function predict()!" << std::endl;
C
Channingss 已提交
161 162
    return false;
  }
S
syyxsxx 已提交
163
  // im preprocess
C
Channingss 已提交
164 165 166 167
  if (!preprocess(im, &inputs_)) {
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }
S
syyxsxx 已提交
168
  // predict
C
Channingss 已提交
169 170 171 172 173 174
  auto in_tensor = predictor_->GetInputTensor("image");
  int h = inputs_.new_im_size_[0];
  int w = inputs_.new_im_size_[1];
  in_tensor->Reshape({1, 3, h, w});
  in_tensor->copy_from_cpu(inputs_.im_data_.data());
  predictor_->ZeroCopyRun();
S
syyxsxx 已提交
175
  // get result
C
Channingss 已提交
176 177 178 179 180 181 182 183 184
  auto output_names = predictor_->GetOutputNames();
  auto output_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_shape = output_tensor->shape();
  int size = 1;
  for (const auto& i : output_shape) {
    size *= i;
  }
  outputs_.resize(size);
  output_tensor->copy_to_cpu(outputs_.data());
S
syyxsxx 已提交
185
  // postprocess
C
Channingss 已提交
186 187 188 189
  auto ptr = std::max_element(std::begin(outputs_), std::end(outputs_));
  result->category_id = std::distance(std::begin(outputs_), ptr);
  result->score = *ptr;
  result->category = labels[result->category_id];
J
jack 已提交
190
  return true;
C
Channingss 已提交
191 192
}

J
jack 已提交
193 194 195 196
bool Model::predict(const std::vector<cv::Mat>& im_batch,
                    std::vector<ClsResult>* results,
                    int thread_num) {
  for (auto& inputs : inputs_batch_) {
J
jack 已提交
197 198 199 200
    inputs.clear();
  }
  if (type == "detector") {
    std::cerr << "Loading model is a 'detector', DetResult should be passed to "
J
jack 已提交
201
                 "function predict()!" << std::endl;
J
jack 已提交
202 203 204
    return false;
  } else if (type == "segmenter") {
    std::cerr << "Loading model is a 'segmenter', SegResult should be passed "
J
jack 已提交
205
                 "to function predict()!" << std::endl;
J
jack 已提交
206 207
    return false;
  }
J
jack 已提交
208
  inputs_batch_.assign(im_batch.size(), ImageBlob());
S
syyxsxx 已提交
209
  // preprocess
J
jack 已提交
210
  if (!preprocess(im_batch, &inputs_batch_, thread_num)) {
J
jack 已提交
211 212 213
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }
S
syyxsxx 已提交
214
  // predict
J
jack 已提交
215 216 217 218 219 220
  int batch_size = im_batch.size();
  auto in_tensor = predictor_->GetInputTensor("image");
  int h = inputs_batch_[0].new_im_size_[0];
  int w = inputs_batch_[0].new_im_size_[1];
  in_tensor->Reshape({batch_size, 3, h, w});
  std::vector<float> inputs_data(batch_size * 3 * h * w);
J
jack 已提交
221 222 223 224
  for (int i = 0; i < batch_size; ++i) {
    std::copy(inputs_batch_[i].im_data_.begin(),
              inputs_batch_[i].im_data_.end(),
              inputs_data.begin() + i * 3 * h * w);
J
jack 已提交
225 226
  }
  in_tensor->copy_from_cpu(inputs_data.data());
J
jack 已提交
227
  // in_tensor->copy_from_cpu(inputs_.im_data_.data());
J
jack 已提交
228
  predictor_->ZeroCopyRun();
S
syyxsxx 已提交
229
  // get result
J
jack 已提交
230 231 232 233 234 235 236 237 238
  auto output_names = predictor_->GetOutputNames();
  auto output_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_shape = output_tensor->shape();
  int size = 1;
  for (const auto& i : output_shape) {
    size *= i;
  }
  outputs_.resize(size);
  output_tensor->copy_to_cpu(outputs_.data());
S
syyxsxx 已提交
239
  // postprocess
240 241
  (*results).clear();
  (*results).resize(batch_size);
J
jack 已提交
242
  int single_batch_size = size / batch_size;
J
jack 已提交
243
  for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
244 245 246 247 248
    auto start_ptr = std::begin(outputs_);
    auto end_ptr = std::begin(outputs_);
    std::advance(start_ptr, i * single_batch_size);
    std::advance(end_ptr, (i + 1) * single_batch_size);
    auto ptr = std::max_element(start_ptr, end_ptr);
J
jack 已提交
249 250 251
    (*results)[i].category_id = std::distance(start_ptr, ptr);
    (*results)[i].score = *ptr;
    (*results)[i].category = labels[(*results)[i].category_id];
J
jack 已提交
252 253 254 255
  }
  return true;
}

C
Channingss 已提交
256
bool Model::predict(const cv::Mat& im, DetResult* result) {
J
jack 已提交
257
  inputs_.clear();
C
Channingss 已提交
258 259 260
  result->clear();
  if (type == "classifier") {
    std::cerr << "Loading model is a 'classifier', ClsResult should be passed "
J
jack 已提交
261
                 "to function predict()!" << std::endl;
C
Channingss 已提交
262 263 264
    return false;
  } else if (type == "segmenter") {
    std::cerr << "Loading model is a 'segmenter', SegResult should be passed "
J
jack 已提交
265
                 "to function predict()!" << std::endl;
C
Channingss 已提交
266 267 268
    return false;
  }

S
syyxsxx 已提交
269
  // preprocess
C
Channingss 已提交
270 271 272 273 274 275 276 277 278 279
  if (!preprocess(im, &inputs_)) {
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }

  int h = inputs_.new_im_size_[0];
  int w = inputs_.new_im_size_[1];
  auto im_tensor = predictor_->GetInputTensor("image");
  im_tensor->Reshape({1, 3, h, w});
  im_tensor->copy_from_cpu(inputs_.im_data_.data());
J
jack 已提交
280

C
Channingss 已提交
281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298
  if (name == "YOLOv3") {
    auto im_size_tensor = predictor_->GetInputTensor("im_size");
    im_size_tensor->Reshape({1, 2});
    im_size_tensor->copy_from_cpu(inputs_.ori_im_size_.data());
  } else if (name == "FasterRCNN" || name == "MaskRCNN") {
    auto im_info_tensor = predictor_->GetInputTensor("im_info");
    auto im_shape_tensor = predictor_->GetInputTensor("im_shape");
    im_info_tensor->Reshape({1, 3});
    im_shape_tensor->Reshape({1, 3});
    float ori_h = static_cast<float>(inputs_.ori_im_size_[0]);
    float ori_w = static_cast<float>(inputs_.ori_im_size_[1]);
    float new_h = static_cast<float>(inputs_.new_im_size_[0]);
    float new_w = static_cast<float>(inputs_.new_im_size_[1]);
    float im_info[] = {new_h, new_w, inputs_.scale};
    float im_shape[] = {ori_h, ori_w, 1.0};
    im_info_tensor->copy_from_cpu(im_info);
    im_shape_tensor->copy_from_cpu(im_shape);
  }
S
syyxsxx 已提交
299
  // predict
C
Channingss 已提交
300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316
  predictor_->ZeroCopyRun();

  std::vector<float> output_box;
  auto output_names = predictor_->GetOutputNames();
  auto output_box_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_box_shape = output_box_tensor->shape();
  int size = 1;
  for (const auto& i : output_box_shape) {
    size *= i;
  }
  output_box.resize(size);
  output_box_tensor->copy_to_cpu(output_box.data());
  if (size < 6) {
    std::cerr << "[WARNING] There's no object detected." << std::endl;
    return true;
  }
  int num_boxes = size / 6;
S
syyxsxx 已提交
317
  // box postprocess
C
Channingss 已提交
318 319 320 321 322 323 324 325 326 327 328 329 330 331
  for (int i = 0; i < num_boxes; ++i) {
    Box box;
    box.category_id = static_cast<int>(round(output_box[i * 6]));
    box.category = labels[box.category_id];
    box.score = output_box[i * 6 + 1];
    float xmin = output_box[i * 6 + 2];
    float ymin = output_box[i * 6 + 3];
    float xmax = output_box[i * 6 + 4];
    float ymax = output_box[i * 6 + 5];
    float w = xmax - xmin + 1;
    float h = ymax - ymin + 1;
    box.coordinate = {xmin, ymin, w, h};
    result->boxes.push_back(std::move(box));
  }
S
syyxsxx 已提交
332
  // mask postprocess
C
Channingss 已提交
333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350
  if (name == "MaskRCNN") {
    std::vector<float> output_mask;
    auto output_mask_tensor = predictor_->GetOutputTensor(output_names[1]);
    std::vector<int> output_mask_shape = output_mask_tensor->shape();
    int masks_size = 1;
    for (const auto& i : output_mask_shape) {
      masks_size *= i;
    }
    int mask_pixels = output_mask_shape[2] * output_mask_shape[3];
    int classes = output_mask_shape[1];
    output_mask.resize(masks_size);
    output_mask_tensor->copy_to_cpu(output_mask.data());
    result->mask_resolution = output_mask_shape[2];
    for (int i = 0; i < result->boxes.size(); ++i) {
      Box* box = &result->boxes[i];
      auto begin_mask =
          output_mask.begin() + (i * classes + box->category_id) * mask_pixels;
      auto end_mask = begin_mask + mask_pixels;
S
syyxsxx 已提交
351 352 353 354
      for (auto iter = begin_mask; iter != end_mask; iter++) {
        int mask_int = floor((*iter) + 0.5);
        box->mask.push_back(mask_int);
      }
C
Channingss 已提交
355 356 357 358
      box->mask.shape = {static_cast<int>(box->coordinate[2]),
                         static_cast<int>(box->coordinate[3])};
    }
  }
J
jack 已提交
359
  return true;
C
Channingss 已提交
360 361
}

J
jack 已提交
362
bool Model::predict(const std::vector<cv::Mat>& im_batch,
363
                    std::vector<DetResult>* results,
J
jack 已提交
364 365
                    int thread_num) {
  for (auto& inputs : inputs_batch_) {
J
jack 已提交
366 367
    inputs.clear();
  }
J
jack 已提交
368 369
  if (type == "classifier") {
    std::cerr << "Loading model is a 'classifier', ClsResult should be passed "
J
jack 已提交
370
                 "to function predict()!" << std::endl;
J
jack 已提交
371 372 373
    return false;
  } else if (type == "segmenter") {
    std::cerr << "Loading model is a 'segmenter', SegResult should be passed "
J
jack 已提交
374
                 "to function predict()!" << std::endl;
J
jack 已提交
375 376 377
    return false;
  }

J
jack 已提交
378
  inputs_batch_.assign(im_batch.size(), ImageBlob());
J
jack 已提交
379
  int batch_size = im_batch.size();
S
syyxsxx 已提交
380
  // preprocess
J
jack 已提交
381
  if (!preprocess(im_batch, &inputs_batch_, thread_num)) {
J
jack 已提交
382 383 384
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }
S
syyxsxx 已提交
385
  // RCNN model padding
J
jack 已提交
386 387 388 389
  if (batch_size > 1) {
    if (name == "FasterRCNN" || name == "MaskRCNN") {
      int max_h = -1;
      int max_w = -1;
J
jack 已提交
390
      for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
391 392
        max_h = std::max(max_h, inputs_batch_[i].new_im_size_[0]);
        max_w = std::max(max_w, inputs_batch_[i].new_im_size_[1]);
J
jack 已提交
393 394
        // std::cout << "(" << inputs_batch_[i].new_im_size_[0]
        //          << ", " << inputs_batch_[i].new_im_size_[1]
J
jack 已提交
395
        //          <<  ")" << std::endl;
J
jack 已提交
396
      }
J
jack 已提交
397 398
      thread_num = std::min(thread_num, batch_size);
      #pragma omp parallel for num_threads(thread_num)
J
jack 已提交
399
      for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
400 401 402
        int h = inputs_batch_[i].new_im_size_[0];
        int w = inputs_batch_[i].new_im_size_[1];
        int c = im_batch[i].channels();
J
jack 已提交
403
        if (max_h != h || max_w != w) {
J
jack 已提交
404
          std::vector<float> temp_buffer(c * max_h * max_w);
J
jack 已提交
405 406 407
          float* temp_ptr = temp_buffer.data();
          float* ptr = inputs_batch_[i].im_data_.data();
          for (int cur_channel = c - 1; cur_channel >= 0; --cur_channel) {
J
jack 已提交
408 409
            int ori_pos = cur_channel * h * w + (h - 1) * w;
            int des_pos = cur_channel * max_h * max_w + (h - 1) * max_w;
J
jack 已提交
410 411 412
            int last_pos = cur_channel * h * w;
            for (; ori_pos >= last_pos; ori_pos -= w, des_pos -= max_w) {
              memcpy(temp_ptr + des_pos, ptr + ori_pos, w * sizeof(float));
J
jack 已提交
413 414 415 416
            }
          }
          inputs_batch_[i].im_data_.swap(temp_buffer);
          inputs_batch_[i].new_im_size_[0] = max_h;
J
jack 已提交
417
          inputs_batch_[i].new_im_size_[1] = max_w;
J
jack 已提交
418 419 420 421
        }
      }
    }
  }
J
jack 已提交
422 423 424 425 426
  int h = inputs_batch_[0].new_im_size_[0];
  int w = inputs_batch_[0].new_im_size_[1];
  auto im_tensor = predictor_->GetInputTensor("image");
  im_tensor->Reshape({batch_size, 3, h, w});
  std::vector<float> inputs_data(batch_size * 3 * h * w);
J
jack 已提交
427 428 429 430
  for (int i = 0; i < batch_size; ++i) {
    std::copy(inputs_batch_[i].im_data_.begin(),
              inputs_batch_[i].im_data_.end(),
              inputs_data.begin() + i * 3 * h * w);
J
jack 已提交
431 432 433 434 435
  }
  im_tensor->copy_from_cpu(inputs_data.data());
  if (name == "YOLOv3") {
    auto im_size_tensor = predictor_->GetInputTensor("im_size");
    im_size_tensor->Reshape({batch_size, 2});
J
jack 已提交
436 437 438 439 440
    std::vector<int> inputs_data_size(batch_size * 2);
    for (int i = 0; i < batch_size; ++i) {
      std::copy(inputs_batch_[i].ori_im_size_.begin(),
                inputs_batch_[i].ori_im_size_.end(),
                inputs_data_size.begin() + 2 * i);
J
jack 已提交
441 442 443 444 445 446 447
    }
    im_size_tensor->copy_from_cpu(inputs_data_size.data());
  } else if (name == "FasterRCNN" || name == "MaskRCNN") {
    auto im_info_tensor = predictor_->GetInputTensor("im_info");
    auto im_shape_tensor = predictor_->GetInputTensor("im_shape");
    im_info_tensor->Reshape({batch_size, 3});
    im_shape_tensor->Reshape({batch_size, 3});
J
jack 已提交
448

J
jack 已提交
449 450
    std::vector<float> im_info(3 * batch_size);
    std::vector<float> im_shape(3 * batch_size);
J
jack 已提交
451
    for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
452 453 454 455 456 457 458 459 460 461 462 463 464 465
      float ori_h = static_cast<float>(inputs_batch_[i].ori_im_size_[0]);
      float ori_w = static_cast<float>(inputs_batch_[i].ori_im_size_[1]);
      float new_h = static_cast<float>(inputs_batch_[i].new_im_size_[0]);
      float new_w = static_cast<float>(inputs_batch_[i].new_im_size_[1]);
      im_info[i * 3] = new_h;
      im_info[i * 3 + 1] = new_w;
      im_info[i * 3 + 2] = inputs_batch_[i].scale;
      im_shape[i * 3] = ori_h;
      im_shape[i * 3 + 1] = ori_w;
      im_shape[i * 3 + 2] = 1.0;
    }
    im_info_tensor->copy_from_cpu(im_info.data());
    im_shape_tensor->copy_from_cpu(im_shape.data());
  }
S
syyxsxx 已提交
466
  // predict
J
jack 已提交
467 468
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
469
  // get all box
J
jack 已提交
470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485
  std::vector<float> output_box;
  auto output_names = predictor_->GetOutputNames();
  auto output_box_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_box_shape = output_box_tensor->shape();
  int size = 1;
  for (const auto& i : output_box_shape) {
    size *= i;
  }
  output_box.resize(size);
  output_box_tensor->copy_to_cpu(output_box.data());
  if (size < 6) {
    std::cerr << "[WARNING] There's no object detected." << std::endl;
    return true;
  }
  auto lod_vector = output_box_tensor->lod();
  int num_boxes = size / 6;
S
syyxsxx 已提交
486
  // box postprocess
487 488
  (*results).clear();
  (*results).resize(batch_size);
J
jack 已提交
489
  for (int i = 0; i < lod_vector[0].size() - 1; ++i) {
J
jack 已提交
490
    for (int j = lod_vector[0][i]; j < lod_vector[0][i + 1]; ++j) {
J
jack 已提交
491
      Box box;
J
jack 已提交
492
      box.category_id = static_cast<int>(round(output_box[j * 6]));
J
jack 已提交
493 494 495 496 497 498 499 500 501
      box.category = labels[box.category_id];
      box.score = output_box[j * 6 + 1];
      float xmin = output_box[j * 6 + 2];
      float ymin = output_box[j * 6 + 3];
      float xmax = output_box[j * 6 + 4];
      float ymax = output_box[j * 6 + 5];
      float w = xmax - xmin + 1;
      float h = ymax - ymin + 1;
      box.coordinate = {xmin, ymin, w, h};
502
      (*results)[i].boxes.push_back(std::move(box));
J
jack 已提交
503 504 505
    }
  }

S
syyxsxx 已提交
506
  // mask postprocess
J
jack 已提交
507 508 509 510 511 512 513 514 515 516 517 518 519
  if (name == "MaskRCNN") {
    std::vector<float> output_mask;
    auto output_mask_tensor = predictor_->GetOutputTensor(output_names[1]);
    std::vector<int> output_mask_shape = output_mask_tensor->shape();
    int masks_size = 1;
    for (const auto& i : output_mask_shape) {
      masks_size *= i;
    }
    int mask_pixels = output_mask_shape[2] * output_mask_shape[3];
    int classes = output_mask_shape[1];
    output_mask.resize(masks_size);
    output_mask_tensor->copy_to_cpu(output_mask.data());
    int mask_idx = 0;
J
jack 已提交
520
    for (int i = 0; i < lod_vector[0].size() - 1; ++i) {
521 522 523
      (*results)[i].mask_resolution = output_mask_shape[2];
      for (int j = 0; j < (*results)[i].boxes.size(); ++j) {
        Box* box = &(*results)[i].boxes[j];
J
jack 已提交
524 525 526
        int category_id = box->category_id;
        auto begin_mask = output_mask.begin() +
                          (mask_idx * classes + category_id) * mask_pixels;
J
jack 已提交
527
        auto end_mask = begin_mask + mask_pixels;
S
syyxsxx 已提交
528 529 530 531
        for (auto iter = begin_mask; iter != end_mask; iter++) {
          int mask_int = floor((*iter) + 0.5);
          box->mask.push_back(mask_int);
        }
J
jack 已提交
532 533 534 535 536 537
        box->mask.shape = {static_cast<int>(box->coordinate[2]),
                           static_cast<int>(box->coordinate[3])};
        mask_idx++;
      }
    }
  }
J
jack 已提交
538
  return true;
J
jack 已提交
539 540
}

C
Channingss 已提交
541 542 543 544 545
bool Model::predict(const cv::Mat& im, SegResult* result) {
  result->clear();
  inputs_.clear();
  if (type == "classifier") {
    std::cerr << "Loading model is a 'classifier', ClsResult should be passed "
J
jack 已提交
546
                 "to function predict()!" << std::endl;
C
Channingss 已提交
547 548 549
    return false;
  } else if (type == "detector") {
    std::cerr << "Loading model is a 'detector', DetResult should be passed to "
J
jack 已提交
550
                 "function predict()!" << std::endl;
C
Channingss 已提交
551 552 553
    return false;
  }

S
syyxsxx 已提交
554
  // preprocess
C
Channingss 已提交
555 556 557 558 559 560 561 562 563 564 565
  if (!preprocess(im, &inputs_)) {
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }

  int h = inputs_.new_im_size_[0];
  int w = inputs_.new_im_size_[1];
  auto im_tensor = predictor_->GetInputTensor("image");
  im_tensor->Reshape({1, 3, h, w});
  im_tensor->copy_from_cpu(inputs_.im_data_.data());

S
syyxsxx 已提交
566
  // predict
C
Channingss 已提交
567 568
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
569
  // get labelmap
C
Channingss 已提交
570 571 572 573 574 575 576 577
  auto output_names = predictor_->GetOutputNames();
  auto output_label_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_label_shape = output_label_tensor->shape();
  int size = 1;
  for (const auto& i : output_label_shape) {
    size *= i;
    result->label_map.shape.push_back(i);
  }
J
jack 已提交
578

C
Channingss 已提交
579 580 581
  result->label_map.data.resize(size);
  output_label_tensor->copy_to_cpu(result->label_map.data.data());

S
syyxsxx 已提交
582
  // get scoremap
C
Channingss 已提交
583 584 585 586 587 588 589
  auto output_score_tensor = predictor_->GetOutputTensor(output_names[1]);
  std::vector<int> output_score_shape = output_score_tensor->shape();
  size = 1;
  for (const auto& i : output_score_shape) {
    size *= i;
    result->score_map.shape.push_back(i);
  }
J
jack 已提交
590

C
Channingss 已提交
591 592 593
  result->score_map.data.resize(size);
  output_score_tensor->copy_to_cpu(result->score_map.data.data());

S
syyxsxx 已提交
594
  // get origin image result
C
Channingss 已提交
595 596 597 598 599 600 601 602 603 604 605
  std::vector<uint8_t> label_map(result->label_map.data.begin(),
                                 result->label_map.data.end());
  cv::Mat mask_label(result->label_map.shape[1],
                     result->label_map.shape[2],
                     CV_8UC1,
                     label_map.data());

  cv::Mat mask_score(result->score_map.shape[2],
                     result->score_map.shape[3],
                     CV_32FC1,
                     result->score_map.data.data());
C
Channingss 已提交
606
  int idx = 1;
C
Channingss 已提交
607
  int len_postprocess = inputs_.im_size_before_resize_.size();
C
Channingss 已提交
608 609
  for (std::vector<std::string>::reverse_iterator iter =
           inputs_.reshape_order_.rbegin();
C
Channingss 已提交
610 611
       iter != inputs_.reshape_order_.rend();
       ++iter) {
C
Channingss 已提交
612
    if (*iter == "padding") {
C
Channingss 已提交
613
      auto before_shape = inputs_.im_size_before_resize_[len_postprocess - idx];
C
Channingss 已提交
614 615 616
      inputs_.im_size_before_resize_.pop_back();
      auto padding_w = before_shape[0];
      auto padding_h = before_shape[1];
J
jack 已提交
617 618
      mask_label = mask_label(cv::Rect(0, 0, padding_h, padding_w));
      mask_score = mask_score(cv::Rect(0, 0, padding_h, padding_w));
C
Channingss 已提交
619
    } else if (*iter == "resize") {
C
Channingss 已提交
620
      auto before_shape = inputs_.im_size_before_resize_[len_postprocess - idx];
C
Channingss 已提交
621 622 623
      inputs_.im_size_before_resize_.pop_back();
      auto resize_w = before_shape[0];
      auto resize_h = before_shape[1];
C
Channingss 已提交
624 625 626 627 628 629 630 631 632 633 634
      cv::resize(mask_label,
                 mask_label,
                 cv::Size(resize_h, resize_w),
                 0,
                 0,
                 cv::INTER_NEAREST);
      cv::resize(mask_score,
                 mask_score,
                 cv::Size(resize_h, resize_w),
                 0,
                 0,
J
jack 已提交
635
                 cv::INTER_LINEAR);
C
Channingss 已提交
636
    }
C
Channingss 已提交
637
    ++idx;
C
Channingss 已提交
638 639 640 641 642 643 644
  }
  result->label_map.data.assign(mask_label.begin<uint8_t>(),
                                mask_label.end<uint8_t>());
  result->label_map.shape = {mask_label.rows, mask_label.cols};
  result->score_map.data.assign(mask_score.begin<float>(),
                                mask_score.end<float>());
  result->score_map.shape = {mask_score.rows, mask_score.cols};
J
jack 已提交
645 646 647
  return true;
}

J
jack 已提交
648
bool Model::predict(const std::vector<cv::Mat>& im_batch,
649
                    std::vector<SegResult>* results,
J
jack 已提交
650 651
                    int thread_num) {
  for (auto& inputs : inputs_batch_) {
J
jack 已提交
652 653 654 655
    inputs.clear();
  }
  if (type == "classifier") {
    std::cerr << "Loading model is a 'classifier', ClsResult should be passed "
J
jack 已提交
656
                 "to function predict()!" << std::endl;
J
jack 已提交
657 658 659
    return false;
  } else if (type == "detector") {
    std::cerr << "Loading model is a 'detector', DetResult should be passed to "
J
jack 已提交
660
                 "function predict()!" << std::endl;
J
jack 已提交
661 662 663
    return false;
  }

S
syyxsxx 已提交
664
  // preprocess
J
jack 已提交
665
  inputs_batch_.assign(im_batch.size(), ImageBlob());
J
jack 已提交
666
  if (!preprocess(im_batch, &inputs_batch_, thread_num)) {
J
jack 已提交
667 668 669 670 671
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }

  int batch_size = im_batch.size();
672 673
  (*results).clear();
  (*results).resize(batch_size);
J
jack 已提交
674 675 676 677 678
  int h = inputs_batch_[0].new_im_size_[0];
  int w = inputs_batch_[0].new_im_size_[1];
  auto im_tensor = predictor_->GetInputTensor("image");
  im_tensor->Reshape({batch_size, 3, h, w});
  std::vector<float> inputs_data(batch_size * 3 * h * w);
J
jack 已提交
679 680 681 682
  for (int i = 0; i < batch_size; ++i) {
    std::copy(inputs_batch_[i].im_data_.begin(),
              inputs_batch_[i].im_data_.end(),
              inputs_data.begin() + i * 3 * h * w);
J
jack 已提交
683 684
  }
  im_tensor->copy_from_cpu(inputs_data.data());
J
jack 已提交
685
  // im_tensor->copy_from_cpu(inputs_.im_data_.data());
J
jack 已提交
686

S
syyxsxx 已提交
687
  // predict
J
jack 已提交
688 689
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
690
  // get labelmap
J
jack 已提交
691 692 693 694 695 696 697 698 699 700 701 702 703
  auto output_names = predictor_->GetOutputNames();
  auto output_label_tensor = predictor_->GetOutputTensor(output_names[0]);
  std::vector<int> output_label_shape = output_label_tensor->shape();
  int size = 1;
  for (const auto& i : output_label_shape) {
    size *= i;
  }

  std::vector<int64_t> output_labels(size, 0);
  output_label_tensor->copy_to_cpu(output_labels.data());
  auto output_labels_iter = output_labels.begin();

  int single_batch_size = size / batch_size;
J
jack 已提交
704
  for (int i = 0; i < batch_size; ++i) {
705 706
    (*results)[i].label_map.data.resize(single_batch_size);
    (*results)[i].label_map.shape.push_back(1);
J
jack 已提交
707
    for (int j = 1; j < output_label_shape.size(); ++j) {
708
      (*results)[i].label_map.shape.push_back(output_label_shape[j]);
J
jack 已提交
709
    }
J
jack 已提交
710 711
    std::copy(output_labels_iter + i * single_batch_size,
              output_labels_iter + (i + 1) * single_batch_size,
712
              (*results)[i].label_map.data.data());
J
jack 已提交
713 714
  }

S
syyxsxx 已提交
715
  // get scoremap
J
jack 已提交
716 717 718 719 720 721 722 723 724 725 726 727
  auto output_score_tensor = predictor_->GetOutputTensor(output_names[1]);
  std::vector<int> output_score_shape = output_score_tensor->shape();
  size = 1;
  for (const auto& i : output_score_shape) {
    size *= i;
  }

  std::vector<float> output_scores(size, 0);
  output_score_tensor->copy_to_cpu(output_scores.data());
  auto output_scores_iter = output_scores.begin();

  int single_batch_score_size = size / batch_size;
J
jack 已提交
728
  for (int i = 0; i < batch_size; ++i) {
729 730
    (*results)[i].score_map.data.resize(single_batch_score_size);
    (*results)[i].score_map.shape.push_back(1);
J
jack 已提交
731
    for (int j = 1; j < output_score_shape.size(); ++j) {
732
      (*results)[i].score_map.shape.push_back(output_score_shape[j]);
J
jack 已提交
733
    }
J
jack 已提交
734 735
    std::copy(output_scores_iter + i * single_batch_score_size,
              output_scores_iter + (i + 1) * single_batch_score_size,
736
              (*results)[i].score_map.data.data());
J
jack 已提交
737 738
  }

S
syyxsxx 已提交
739
  // get origin image result
J
jack 已提交
740
  for (int i = 0; i < batch_size; ++i) {
741 742 743 744
    std::vector<uint8_t> label_map((*results)[i].label_map.data.begin(),
                                   (*results)[i].label_map.data.end());
    cv::Mat mask_label((*results)[i].label_map.shape[1],
                       (*results)[i].label_map.shape[2],
J
jack 已提交
745 746
                       CV_8UC1,
                       label_map.data());
J
jack 已提交
747

748 749
    cv::Mat mask_score((*results)[i].score_map.shape[2],
                       (*results)[i].score_map.shape[3],
J
jack 已提交
750
                       CV_32FC1,
751
                       (*results)[i].score_map.data.data());
J
jack 已提交
752 753 754 755 756 757 758
    int idx = 1;
    int len_postprocess = inputs_batch_[i].im_size_before_resize_.size();
    for (std::vector<std::string>::reverse_iterator iter =
             inputs_batch_[i].reshape_order_.rbegin();
         iter != inputs_batch_[i].reshape_order_.rend();
         ++iter) {
      if (*iter == "padding") {
J
jack 已提交
759 760
        auto before_shape =
            inputs_batch_[i].im_size_before_resize_[len_postprocess - idx];
J
jack 已提交
761 762 763 764 765 766
        inputs_batch_[i].im_size_before_resize_.pop_back();
        auto padding_w = before_shape[0];
        auto padding_h = before_shape[1];
        mask_label = mask_label(cv::Rect(0, 0, padding_h, padding_w));
        mask_score = mask_score(cv::Rect(0, 0, padding_h, padding_w));
      } else if (*iter == "resize") {
J
jack 已提交
767 768
        auto before_shape =
            inputs_batch_[i].im_size_before_resize_[len_postprocess - idx];
J
jack 已提交
769 770 771 772 773 774 775 776 777 778 779 780 781 782
        inputs_batch_[i].im_size_before_resize_.pop_back();
        auto resize_w = before_shape[0];
        auto resize_h = before_shape[1];
        cv::resize(mask_label,
                   mask_label,
                   cv::Size(resize_h, resize_w),
                   0,
                   0,
                   cv::INTER_NEAREST);
        cv::resize(mask_score,
                   mask_score,
                   cv::Size(resize_h, resize_w),
                   0,
                   0,
J
jack 已提交
783
                   cv::INTER_LINEAR);
J
jack 已提交
784 785 786
      }
      ++idx;
    }
787
    (*results)[i].label_map.data.assign(mask_label.begin<uint8_t>(),
J
jack 已提交
788
                                       mask_label.end<uint8_t>());
789 790
    (*results)[i].label_map.shape = {mask_label.rows, mask_label.cols};
    (*results)[i].score_map.data.assign(mask_score.begin<float>(),
J
jack 已提交
791
                                       mask_score.end<float>());
792
    (*results)[i].score_map.shape = {mask_score.rows, mask_score.cols};
J
jack 已提交
793 794
  }
  return true;
C
Channingss 已提交
795 796
}

J
jack 已提交
797
}  // namespace PaddleX