paddlex.cpp 29.4 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.
J
jack 已提交
14
#include <omp.h>
J
jack 已提交
15
#include <algorithm>
J
jack 已提交
16
#include <fstream>
J
jack 已提交
17
#include <cstring>
J
jack 已提交
18
#include "include/paddlex/paddlex.h"
C
Channingss 已提交
19 20 21 22
namespace PaddleX {

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

J
jack 已提交
59
  if (key == "") {
C
Channingss 已提交
60 61
    config.SetModel(model_file, params_file);
  }
S
syyxsxx 已提交
62 63 64 65
  if (use_mkl && name != "HRNet" && name != "DeepLabv3p") {
    config.EnableMKLDNN();
    config.SetCpuMathLibraryNumThreads(12);
  }
C
Channingss 已提交
66 67 68 69 70 71 72
  if (use_gpu) {
    config.EnableUseGpu(100, gpu_id);
  } else {
    config.DisableGpu();
  }
  config.SwitchUseFeedFetchOps(false);
  config.SwitchSpecifyInputNames(true);
S
syyxsxx 已提交
73
  // enable graph Optim
F
FlyingQianMM 已提交
74 75 76
#if defined(__arm__) || defined(__aarch64__)
  config.SwitchIrOptim(false);
#else
J
jack 已提交
77
  config.SwitchIrOptim(use_ir_optim);
F
FlyingQianMM 已提交
78
#endif
S
syyxsxx 已提交
79
  // enable Memory Optim
C
Channingss 已提交
80
  config.EnableMemoryOptim();
C
Channingss 已提交
81 82 83 84 85 86 87 88
  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 已提交
89
  }
C
Channingss 已提交
90 91 92
  predictor_ = std::move(CreatePaddlePredictor(config));
}

J
jack 已提交
93 94
bool Model::load_config(const std::string& yaml_input) {
  YAML::Node config = YAML::Load(yaml_input);
C
Channingss 已提交
95 96
  type = config["_Attributes"]["model_type"].as<std::string>();
  name = config["Model"].as<std::string>();
F
FlyingQianMM 已提交
97 98
  std::string version = config["version"].as<std::string>();
  if (version[0] == '0') {
J
jack 已提交
99 100 101 102 103
    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 已提交
104 105
    return false;
  }
C
Channingss 已提交
106 107 108 109 110 111 112 113 114 115 116
  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 已提交
117
  // build data preprocess stream
C
Channingss 已提交
118
  transforms_.Init(config["Transforms"], to_rgb);
S
syyxsxx 已提交
119
  // read label list
C
Channingss 已提交
120 121 122 123 124 125 126 127 128 129
  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();
130
  if (!transforms_.Run(&im, blob)) {
C
Channingss 已提交
131 132 133 134 135
    return false;
  }
  return true;
}

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

C
Channingss 已提交
153 154 155 156 157
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 已提交
158
                 "to function predict()!" << std::endl;
C
Channingss 已提交
159 160
    return false;
  }
S
syyxsxx 已提交
161
  // im preprocess
C
Channingss 已提交
162 163 164 165
  if (!preprocess(im, &inputs_)) {
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }
S
syyxsxx 已提交
166
  // predict
C
Channingss 已提交
167 168 169 170 171 172
  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 已提交
173
  // get result
C
Channingss 已提交
174 175 176 177 178 179 180 181 182
  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 已提交
183
  // postprocess
C
Channingss 已提交
184 185 186 187
  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 已提交
188
  return true;
C
Channingss 已提交
189 190
}

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

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

S
syyxsxx 已提交
267
  // preprocess
C
Channingss 已提交
268 269 270 271 272 273 274 275 276 277
  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 已提交
278

C
Channingss 已提交
279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296
  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 已提交
297
  // predict
C
Channingss 已提交
298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314
  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 已提交
315
  // box postprocess
C
Channingss 已提交
316 317 318 319 320 321 322 323 324 325 326 327 328 329
  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 已提交
330
  // mask postprocess
C
Channingss 已提交
331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
  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;
      box->mask.data.assign(begin_mask, end_mask);
      box->mask.shape = {static_cast<int>(box->coordinate[2]),
                         static_cast<int>(box->coordinate[3])};
    }
  }
J
jack 已提交
354
  return true;
C
Channingss 已提交
355 356
}

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

J
jack 已提交
373
  inputs_batch_.assign(im_batch.size(), ImageBlob());
J
jack 已提交
374
  int batch_size = im_batch.size();
S
syyxsxx 已提交
375
  // preprocess
J
jack 已提交
376
  if (!preprocess(im_batch, &inputs_batch_, thread_num)) {
J
jack 已提交
377 378 379
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }
S
syyxsxx 已提交
380
  // RCNN model padding
J
jack 已提交
381 382 383 384
  if (batch_size > 1) {
    if (name == "FasterRCNN" || name == "MaskRCNN") {
      int max_h = -1;
      int max_w = -1;
J
jack 已提交
385
      for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
386 387
        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 已提交
388 389
        // std::cout << "(" << inputs_batch_[i].new_im_size_[0]
        //          << ", " << inputs_batch_[i].new_im_size_[1]
J
jack 已提交
390
        //          <<  ")" << std::endl;
J
jack 已提交
391
      }
J
jack 已提交
392 393
      thread_num = std::min(thread_num, batch_size);
      #pragma omp parallel for num_threads(thread_num)
J
jack 已提交
394
      for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
395 396 397
        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 已提交
398
        if (max_h != h || max_w != w) {
J
jack 已提交
399
          std::vector<float> temp_buffer(c * max_h * max_w);
J
jack 已提交
400 401 402
          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 已提交
403 404
            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 已提交
405 406 407
            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 已提交
408 409 410 411
            }
          }
          inputs_batch_[i].im_data_.swap(temp_buffer);
          inputs_batch_[i].new_im_size_[0] = max_h;
J
jack 已提交
412
          inputs_batch_[i].new_im_size_[1] = max_w;
J
jack 已提交
413 414 415 416
        }
      }
    }
  }
J
jack 已提交
417 418 419 420 421
  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 已提交
422 423 424 425
  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 已提交
426 427 428 429 430
  }
  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 已提交
431 432 433 434 435
    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 已提交
436 437 438 439 440 441 442
    }
    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 已提交
443

J
jack 已提交
444 445
    std::vector<float> im_info(3 * batch_size);
    std::vector<float> im_shape(3 * batch_size);
J
jack 已提交
446
    for (int i = 0; i < batch_size; ++i) {
J
jack 已提交
447 448 449 450 451 452 453 454 455 456 457 458 459 460
      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 已提交
461
  // predict
J
jack 已提交
462 463
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
464
  // get all box
J
jack 已提交
465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480
  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 已提交
481
  // box postprocess
482 483
  (*results).clear();
  (*results).resize(batch_size);
J
jack 已提交
484
  for (int i = 0; i < lod_vector[0].size() - 1; ++i) {
J
jack 已提交
485
    for (int j = lod_vector[0][i]; j < lod_vector[0][i + 1]; ++j) {
J
jack 已提交
486
      Box box;
J
jack 已提交
487
      box.category_id = static_cast<int>(round(output_box[j * 6]));
J
jack 已提交
488 489 490 491 492 493 494 495 496
      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};
497
      (*results)[i].boxes.push_back(std::move(box));
J
jack 已提交
498 499 500
    }
  }

S
syyxsxx 已提交
501
  // mask postprocess
J
jack 已提交
502 503 504 505 506 507 508 509 510 511 512 513 514
  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 已提交
515
    for (int i = 0; i < lod_vector[0].size() - 1; ++i) {
516 517 518
      (*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 已提交
519 520 521
        int category_id = box->category_id;
        auto begin_mask = output_mask.begin() +
                          (mask_idx * classes + category_id) * mask_pixels;
J
jack 已提交
522 523 524 525 526 527 528 529
        auto end_mask = begin_mask + mask_pixels;
        box->mask.data.assign(begin_mask, end_mask);
        box->mask.shape = {static_cast<int>(box->coordinate[2]),
                           static_cast<int>(box->coordinate[3])};
        mask_idx++;
      }
    }
  }
J
jack 已提交
530
  return true;
J
jack 已提交
531 532
}

C
Channingss 已提交
533 534 535 536 537
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 已提交
538
                 "to function predict()!" << std::endl;
C
Channingss 已提交
539 540 541
    return false;
  } else if (type == "detector") {
    std::cerr << "Loading model is a 'detector', DetResult should be passed to "
J
jack 已提交
542
                 "function predict()!" << std::endl;
C
Channingss 已提交
543 544 545
    return false;
  }

S
syyxsxx 已提交
546
  // preprocess
C
Channingss 已提交
547 548 549 550 551 552 553 554 555 556 557
  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 已提交
558
  // predict
C
Channingss 已提交
559 560
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
561
  // get labelmap
C
Channingss 已提交
562 563 564 565 566 567 568 569
  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 已提交
570

C
Channingss 已提交
571 572 573
  result->label_map.data.resize(size);
  output_label_tensor->copy_to_cpu(result->label_map.data.data());

S
syyxsxx 已提交
574
  // get scoremap
C
Channingss 已提交
575 576 577 578 579 580 581
  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 已提交
582

C
Channingss 已提交
583 584 585
  result->score_map.data.resize(size);
  output_score_tensor->copy_to_cpu(result->score_map.data.data());

S
syyxsxx 已提交
586
  // get origin image result
C
Channingss 已提交
587 588 589 590 591 592 593 594 595 596 597
  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 已提交
598
  int idx = 1;
C
Channingss 已提交
599
  int len_postprocess = inputs_.im_size_before_resize_.size();
C
Channingss 已提交
600 601
  for (std::vector<std::string>::reverse_iterator iter =
           inputs_.reshape_order_.rbegin();
C
Channingss 已提交
602 603
       iter != inputs_.reshape_order_.rend();
       ++iter) {
C
Channingss 已提交
604
    if (*iter == "padding") {
C
Channingss 已提交
605
      auto before_shape = inputs_.im_size_before_resize_[len_postprocess - idx];
C
Channingss 已提交
606 607 608
      inputs_.im_size_before_resize_.pop_back();
      auto padding_w = before_shape[0];
      auto padding_h = before_shape[1];
J
jack 已提交
609 610
      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 已提交
611
    } else if (*iter == "resize") {
C
Channingss 已提交
612
      auto before_shape = inputs_.im_size_before_resize_[len_postprocess - idx];
C
Channingss 已提交
613 614 615
      inputs_.im_size_before_resize_.pop_back();
      auto resize_w = before_shape[0];
      auto resize_h = before_shape[1];
C
Channingss 已提交
616 617 618 619 620 621 622 623 624 625 626
      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 已提交
627
                 cv::INTER_LINEAR);
C
Channingss 已提交
628
    }
C
Channingss 已提交
629
    ++idx;
C
Channingss 已提交
630 631 632 633 634 635 636
  }
  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 已提交
637 638 639
  return true;
}

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

S
syyxsxx 已提交
656
  // preprocess
J
jack 已提交
657
  inputs_batch_.assign(im_batch.size(), ImageBlob());
J
jack 已提交
658
  if (!preprocess(im_batch, &inputs_batch_, thread_num)) {
J
jack 已提交
659 660 661 662 663
    std::cerr << "Preprocess failed!" << std::endl;
    return false;
  }

  int batch_size = im_batch.size();
664 665
  (*results).clear();
  (*results).resize(batch_size);
J
jack 已提交
666 667 668 669 670
  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 已提交
671 672 673 674
  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 已提交
675 676
  }
  im_tensor->copy_from_cpu(inputs_data.data());
J
jack 已提交
677
  // im_tensor->copy_from_cpu(inputs_.im_data_.data());
J
jack 已提交
678

S
syyxsxx 已提交
679
  // predict
J
jack 已提交
680 681
  predictor_->ZeroCopyRun();

S
syyxsxx 已提交
682
  // get labelmap
J
jack 已提交
683 684 685 686 687 688 689 690 691 692 693 694 695
  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 已提交
696
  for (int i = 0; i < batch_size; ++i) {
697 698
    (*results)[i].label_map.data.resize(single_batch_size);
    (*results)[i].label_map.shape.push_back(1);
J
jack 已提交
699
    for (int j = 1; j < output_label_shape.size(); ++j) {
700
      (*results)[i].label_map.shape.push_back(output_label_shape[j]);
J
jack 已提交
701
    }
J
jack 已提交
702 703
    std::copy(output_labels_iter + i * single_batch_size,
              output_labels_iter + (i + 1) * single_batch_size,
704
              (*results)[i].label_map.data.data());
J
jack 已提交
705 706
  }

S
syyxsxx 已提交
707
  // get scoremap
J
jack 已提交
708 709 710 711 712 713 714 715 716 717 718 719
  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 已提交
720
  for (int i = 0; i < batch_size; ++i) {
721 722
    (*results)[i].score_map.data.resize(single_batch_score_size);
    (*results)[i].score_map.shape.push_back(1);
J
jack 已提交
723
    for (int j = 1; j < output_score_shape.size(); ++j) {
724
      (*results)[i].score_map.shape.push_back(output_score_shape[j]);
J
jack 已提交
725
    }
J
jack 已提交
726 727
    std::copy(output_scores_iter + i * single_batch_score_size,
              output_scores_iter + (i + 1) * single_batch_score_size,
728
              (*results)[i].score_map.data.data());
J
jack 已提交
729 730
  }

S
syyxsxx 已提交
731
  // get origin image result
J
jack 已提交
732
  for (int i = 0; i < batch_size; ++i) {
733 734 735 736
    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 已提交
737 738
                       CV_8UC1,
                       label_map.data());
J
jack 已提交
739

740 741
    cv::Mat mask_score((*results)[i].score_map.shape[2],
                       (*results)[i].score_map.shape[3],
J
jack 已提交
742
                       CV_32FC1,
743
                       (*results)[i].score_map.data.data());
J
jack 已提交
744 745 746 747 748 749 750
    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 已提交
751 752
        auto before_shape =
            inputs_batch_[i].im_size_before_resize_[len_postprocess - idx];
J
jack 已提交
753 754 755 756 757 758
        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 已提交
759 760
        auto before_shape =
            inputs_batch_[i].im_size_before_resize_[len_postprocess - idx];
J
jack 已提交
761 762 763 764 765 766 767 768 769 770 771 772 773 774
        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 已提交
775
                   cv::INTER_LINEAR);
J
jack 已提交
776 777 778
      }
      ++idx;
    }
779
    (*results)[i].label_map.data.assign(mask_label.begin<uint8_t>(),
J
jack 已提交
780
                                       mask_label.end<uint8_t>());
781 782
    (*results)[i].label_map.shape = {mask_label.rows, mask_label.cols};
    (*results)[i].score_map.data.assign(mask_score.begin<float>(),
J
jack 已提交
783
                                       mask_score.end<float>());
784
    (*results)[i].score_map.shape = {mask_score.rows, mask_score.cols};
J
jack 已提交
785 786
  }
  return true;
C
Channingss 已提交
787 788
}

J
jack 已提交
789
}  // namespace PaddleX