mirror of
https://github.com/pytorch/pytorch.git
synced 2025-10-20 21:14:14 +08:00
Summary: It is sometimes beneficial to run multiple batches in one benchmark and check the aggregated results. This PR enables this functionality. Pull Request resolved: https://github.com/pytorch/pytorch/pull/15443 Reviewed By: llyfacebook Differential Revision: D13531129 Pulled By: sf-wind fbshipit-source-id: 553a762a5cbadf5a3d9fd6af767ae34899bc1aa2
451 lines
13 KiB
C++
Executable File
451 lines
13 KiB
C++
Executable File
/**
|
|
* Copyright (c) 2016-present, Facebook, Inc.
|
|
*
|
|
* 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 <opencv2/opencv.hpp>
|
|
#include <cmath>
|
|
#include <fstream>
|
|
|
|
#include "caffe2/core/common.h"
|
|
#include "caffe2/core/db.h"
|
|
#include "caffe2/core/init.h"
|
|
#include "caffe2/core/logging.h"
|
|
#include "caffe2/core/timer.h"
|
|
#include "caffe2/proto/caffe2_pb.h"
|
|
#include "caffe2/utils/proto_utils.h"
|
|
#include "caffe2/utils/string_utils.h"
|
|
|
|
|
|
C10_DEFINE_int(
|
|
batch_size,
|
|
-1,
|
|
"Specify the batch size of the input. The number of items in the "
|
|
"input needs to be multiples of the batch size. If the batch size "
|
|
"is less than 0, all inputs are in one batch.")
|
|
C10_DEFINE_bool(color, true, "If set, load images in color.");
|
|
C10_DEFINE_string(
|
|
crop,
|
|
"-1,-1",
|
|
"The center cropped hight and width. If the value is less than zero, "
|
|
"it is not cropped.");
|
|
C10_DEFINE_string(input_images, "", "Comma separated images");
|
|
C10_DEFINE_string(input_image_file, "", "The file containing imput images");
|
|
C10_DEFINE_string(input_text_file, "", "the text file to be written to blobs");
|
|
C10_DEFINE_string(
|
|
output_tensor,
|
|
"",
|
|
"The output tensor file in NCHW for input images");
|
|
C10_DEFINE_string(
|
|
output_text_tensor,
|
|
"",
|
|
"The output tensor file for the text input specified in input_text_file");
|
|
C10_DEFINE_string(
|
|
preprocess,
|
|
"",
|
|
"Options to specify the preprocess routines. The available options are "
|
|
"subtract128, normalize, mean, std, bgrtorgb. If multiple steps are provided, they "
|
|
"are separated by comma (,) in sequence.");
|
|
C10_DEFINE_string(
|
|
report_time,
|
|
"",
|
|
"Report the conversion stage time to screen. "
|
|
"The format of the string is <type>|<identifier>. "
|
|
"The valid type is 'json'. "
|
|
"The valid identifier is nothing or an identifer that prefix every line");
|
|
C10_DEFINE_string(
|
|
scale,
|
|
"-1,-1",
|
|
"Scale the images to be within the min,max box. The shorter edge is "
|
|
"min pixels. But if the other edge is more than the max pixels, the "
|
|
"other edge and scaled to max pixels (and the shorter edge can be less "
|
|
"than the min pixels");
|
|
C10_DEFINE_bool(text_output, false, "Write the output in text format.");
|
|
C10_DEFINE_bool(warp, false, "If warp is set, warp the images to square.");
|
|
|
|
namespace caffe2 {
|
|
|
|
void reportTime(
|
|
std::string type,
|
|
double ts,
|
|
std::string metric,
|
|
std::string unit) {
|
|
if (FLAGS_report_time == "") {
|
|
return;
|
|
}
|
|
vector<string> s = caffe2::split('|', FLAGS_report_time);
|
|
assert(s[0] == "json");
|
|
std::string identifier = "";
|
|
if (s.size() > 1) {
|
|
identifier = s[1];
|
|
}
|
|
std::cout << identifier << "{\"type\": \"" << type << "\", \"value\": " << ts
|
|
<< ", \"metric\": \"" << metric << "\", \"unit\": \"" << unit
|
|
<< "\"}" << std::endl;
|
|
}
|
|
|
|
void splitSizes(const std::string& arg, int* ptr0, int* ptr1) {
|
|
vector<string> sizes = caffe2::split(',', arg);
|
|
if (sizes.size() == 2) {
|
|
*ptr0 = std::stoi(sizes[0]);
|
|
*ptr1 = std::stoi(sizes[1]);
|
|
} else if (sizes.size() == 1) {
|
|
*ptr0 = std::stoi(sizes[0]);
|
|
*ptr1 = std::stoi(sizes[0]);
|
|
} else {
|
|
assert(false);
|
|
}
|
|
}
|
|
|
|
|
|
cv::Mat resizeImage(cv::Mat& img) {
|
|
int min_size, max_size;
|
|
splitSizes(FLAGS_scale, &min_size, &max_size);
|
|
if ((min_size <= 0) && (max_size <= 0)) {
|
|
return img;
|
|
}
|
|
if (max_size < 0) {
|
|
max_size = INT_MAX;
|
|
}
|
|
assert(min_size <= max_size);
|
|
|
|
int im_min_size = img.rows > img.cols ? img.cols : img.rows;
|
|
int im_max_size = img.rows > img.cols ? img.rows : img.cols;
|
|
|
|
double im_scale = 1.0 * min_size / im_min_size;
|
|
if (im_scale * im_max_size > max_size) {
|
|
im_scale = 1.0 * max_size / im_max_size;
|
|
}
|
|
int scaled_width = int(round(img.cols * im_scale));
|
|
int scaled_height = int(round(img.rows * im_scale));
|
|
assert((scaled_width <= max_size) && (scaled_height <= max_size));
|
|
if ((scaled_width < min_size) || (scaled_height < min_size)) {
|
|
assert((scaled_width == max_size) || (scaled_height == max_size));
|
|
} else {
|
|
assert((scaled_width == min_size) || (scaled_height == min_size));
|
|
}
|
|
cv::Mat resized_img;
|
|
cv::resize(
|
|
img,
|
|
resized_img,
|
|
cv::Size(),
|
|
im_scale,
|
|
im_scale,
|
|
cv::INTER_LINEAR);
|
|
return resized_img;
|
|
}
|
|
|
|
cv::Mat cropToRec(cv::Mat& img, int* height_ptr, int* width_ptr) {
|
|
int height = *height_ptr;
|
|
int width = *width_ptr;
|
|
if ((height > 0) && (width > 0) &&
|
|
((img.rows != height) || (img.cols != width))) {
|
|
cv::Mat cropped_img, cimg;
|
|
cv::Rect roi;
|
|
roi.x = int((img.cols - width) / 2);
|
|
roi.y = int((img.rows - height) / 2);
|
|
roi.x = roi.x < 0 ? 0 : roi.x;
|
|
roi.y = roi.y < 0 ? 0 : roi.y;
|
|
width = width > img.cols ? img.cols : width;
|
|
height = height > img.rows ? img.rows : height;
|
|
roi.width = width;
|
|
roi.height = height;
|
|
assert(
|
|
0 <= roi.x && 0 <= roi.width && roi.x + roi.width <= img.cols &&
|
|
0 <= roi.y && 0 <= roi.height && roi.y + roi.height <= img.rows);
|
|
cropped_img = img(roi);
|
|
// Make the image in continuous space in memory
|
|
cimg = cropped_img.clone();
|
|
*height_ptr = height;
|
|
*width_ptr = width;
|
|
return cimg;
|
|
} else {
|
|
return img;
|
|
}
|
|
}
|
|
|
|
std::vector<float> convertToVector(cv::Mat& img) {
|
|
std::vector<float> normalize(3, 1);
|
|
std::vector<float> mean(3, 0);
|
|
std::vector<float> std(3, 1);
|
|
bool bgrtorgb = false;
|
|
int size = img.cols * img.rows;
|
|
vector<string> steps = caffe2::split(',', FLAGS_preprocess);
|
|
for (int i = 0; i < steps.size(); i++) {
|
|
auto step = steps[i];
|
|
if (step == "subtract128") {
|
|
mean = {128, 128, 128};
|
|
std = {1, 1, 1};
|
|
normalize = {1, 1, 1};
|
|
} else if (step == "normalize") {
|
|
normalize = {255, 255, 255};
|
|
} else if (step == "mean") {
|
|
mean = {0.406f, 0.456f, 0.485f};
|
|
} else if (step == "std") {
|
|
std = {0.225f, 0.224f, 0.229f};
|
|
} else if (step == "bgrtorgb") {
|
|
bgrtorgb = true;
|
|
} else {
|
|
CAFFE_ENFORCE(
|
|
false,
|
|
"Unsupported preprocess step. The supported steps are: subtract128, "
|
|
"normalize,mean, std, swaprb.");
|
|
}
|
|
}
|
|
|
|
int C = FLAGS_color ? 3 : 1;
|
|
int total_size = C * size;
|
|
std::vector<float> values(total_size);
|
|
if (C == 1) {
|
|
cv::MatIterator_<float> it, end;
|
|
int idx = 0;
|
|
for (it = img.begin<float>(), end = img.end<float>(); it != end; ++it) {
|
|
values[idx++] = (*it / normalize[0] - mean[0]) / std[0];
|
|
}
|
|
} else {
|
|
int i = 0;
|
|
cv::MatIterator_<cv::Vec3f> it, end;
|
|
int b = bgrtorgb ? 2 : 0;
|
|
int g = 1;
|
|
int r = bgrtorgb ? 0 : 2;
|
|
for (it = img.begin<cv::Vec3f>(), end = img.end<cv::Vec3f>(); it != end;
|
|
++it, i++) {
|
|
values[i] = (((*it)[b] / normalize[0] - mean[0]) / std[0]);
|
|
int offset = size + i;
|
|
values[offset] = (((*it)[g] / normalize[1] - mean[1]) / std[1]);
|
|
offset = size + offset;
|
|
values[offset] = (((*it)[r] / normalize[2] - mean[2]) / std[2]);
|
|
}
|
|
}
|
|
return values;
|
|
}
|
|
|
|
std::vector<float> convertOneImage(
|
|
std::string& filename,
|
|
int* height_ptr,
|
|
int* width_ptr) {
|
|
assert(filename[0] != '~');
|
|
|
|
std::cout << "Converting " << filename << std::endl;
|
|
|
|
// Load image
|
|
cv::Mat img_uint8 = cv::imread(
|
|
#if CV_MAJOR_VERSION <= 3
|
|
filename, FLAGS_color ? CV_LOAD_IMAGE_COLOR : CV_LOAD_IMAGE_GRAYSCALE);
|
|
#else
|
|
filename, FLAGS_color ? cv::IMREAD_COLOR : cv::IMREAD_GRAYSCALE);
|
|
#endif
|
|
caffe2::Timer timer;
|
|
timer.Start();
|
|
cv::Mat img;
|
|
// Convert image to floating point values
|
|
img_uint8.convertTo(img, CV_32F);
|
|
// Resize image
|
|
cv::Mat resized_img = resizeImage(img);
|
|
|
|
int height, width;
|
|
splitSizes(FLAGS_crop, &height, &width);
|
|
if ((height <= 0) || (width <= 0)) {
|
|
height = resized_img.rows;
|
|
width = resized_img.cols;
|
|
}
|
|
cv::Mat crop = cropToRec(resized_img, &height, &width);
|
|
|
|
// Assert we don't have to deal with alignment
|
|
DCHECK(crop.isContinuous());
|
|
assert(crop.rows == height);
|
|
assert(crop.cols == width);
|
|
std::vector<float> one_image_values = convertToVector(crop);
|
|
*height_ptr = height;
|
|
*width_ptr = width;
|
|
double ts = timer.MicroSeconds();
|
|
reportTime("image_preprocess", ts, "convert", "us");
|
|
return one_image_values;
|
|
}
|
|
|
|
int getBatchSize(int num_items) {
|
|
int batch_size = FLAGS_batch_size;
|
|
if (batch_size < 0) {
|
|
batch_size = num_items;
|
|
} else {
|
|
assert(num_items % batch_size == 0);
|
|
}
|
|
return batch_size;
|
|
}
|
|
|
|
void writeValues(
|
|
std::vector<std::vector<std::vector<float>>>& values,
|
|
std::vector<std::vector<int>>& dims,
|
|
std::string output_file) {
|
|
|
|
caffe2::Timer timer;
|
|
timer.Start();
|
|
|
|
assert(dims.size() == values.size());
|
|
int num_batches = dims.size();
|
|
|
|
TensorProtos protos;
|
|
for (int k = 0; k < num_batches; k++) {
|
|
TensorProto* data;
|
|
data = protos.add_protos();
|
|
data->set_data_type(TensorProto::FLOAT);
|
|
auto one_dim = dims[k];
|
|
for (int dim : one_dim) {
|
|
data->add_dims(dim);
|
|
}
|
|
int batch_size = one_dim[0];
|
|
long long int entry_size = 1;
|
|
for (int i = 1; i < one_dim.size(); i++) {
|
|
entry_size *= one_dim[i];
|
|
}
|
|
|
|
// Not optimized
|
|
for (int i = 0; i < batch_size; i++) {
|
|
assert(values[k][i].size() == entry_size);
|
|
for (int j = 0; j < values[k][i].size(); j++) {
|
|
data->add_float_data(values[k][i][j]);
|
|
}
|
|
}
|
|
}
|
|
double ts = timer.MicroSeconds();
|
|
reportTime("preprocess", ts, "data_pack", "us");
|
|
|
|
if (FLAGS_text_output) {
|
|
caffe2::WriteProtoToTextFile(protos, output_file);
|
|
} else {
|
|
caffe2::WriteProtoToBinaryFile(protos, output_file);
|
|
}
|
|
}
|
|
|
|
void convertImages() {
|
|
vector<string> file_names;
|
|
if (FLAGS_input_images != "") {
|
|
file_names = caffe2::split(',', FLAGS_input_images);
|
|
} else if (FLAGS_input_image_file != "") {
|
|
std::ifstream infile(FLAGS_input_image_file);
|
|
std::string line;
|
|
while (std::getline(infile, line)) {
|
|
vector<string> file_name = caffe2::split(',', line);
|
|
string name;
|
|
if (file_name.size() == 3) {
|
|
name = file_name[2];
|
|
} else {
|
|
name = line;
|
|
}
|
|
file_names.push_back(name);
|
|
}
|
|
} else {
|
|
return;
|
|
}
|
|
int batch_size = getBatchSize(file_names.size());
|
|
int num_batches = file_names.size() / batch_size;
|
|
assert(file_names.size() == batch_size * num_batches);
|
|
std::vector<std::vector<std::vector<float>>> values;
|
|
std::vector<std::vector<int>> dims;
|
|
int C = FLAGS_color ? 3 : 1;
|
|
for (int k = 0; k < num_batches; k++) {
|
|
std::vector<std::vector<float>> one_value;
|
|
int height = -1;
|
|
int width = -1;
|
|
for (int i = 0; i < batch_size; i++) {
|
|
int idx = k * batch_size + i;
|
|
int one_height, one_width;
|
|
std::vector<float> one_image_values =
|
|
convertOneImage(file_names[idx], &one_height, &one_width);
|
|
if (height < 0 && width < 0) {
|
|
height = one_height;
|
|
width = one_width;
|
|
} else {
|
|
assert(height == one_height);
|
|
assert(width == one_width);
|
|
}
|
|
one_value.push_back(one_image_values);
|
|
}
|
|
vector<int> one_dim = {batch_size, C, height, width};
|
|
dims.push_back(one_dim);
|
|
values.push_back(one_value);
|
|
}
|
|
writeValues(values, dims, FLAGS_output_tensor);
|
|
}
|
|
|
|
template <class TYPE>
|
|
vector<TYPE> splitString(std::string& line) {
|
|
vector<string> vector_str = caffe2::split(',', line);
|
|
vector<TYPE> vector_int;
|
|
for (string str : vector_str) {
|
|
vector_int.push_back((TYPE)std::stod(str));
|
|
}
|
|
return vector_int;
|
|
}
|
|
|
|
/* Convert the values in a json file to blobs
|
|
The format of the json file should be:
|
|
<number of items>, <dim2>.... (dimensions of items)
|
|
<entry>, <entry>, <entry>... (all entries in one item)
|
|
<entry>, <entry>, <entry>...
|
|
....
|
|
*/
|
|
void convertValues() {
|
|
if (FLAGS_input_text_file == "") {
|
|
return;
|
|
}
|
|
std::ifstream infile(FLAGS_input_text_file);
|
|
std::string line;
|
|
std::getline(infile, line);
|
|
vector<int> file_dims = splitString <int>(line);
|
|
assert(file_dims.size() >= 2);
|
|
|
|
int num_items = file_dims[0];
|
|
int batch_size = getBatchSize(num_items);
|
|
int num_batches = num_items / batch_size;
|
|
assert(num_items == batch_size * num_batches);
|
|
vector<string> lines;
|
|
while (std::getline(infile, line)) {
|
|
lines.push_back(line);
|
|
}
|
|
assert(lines.size() == num_items);
|
|
std::vector<std::vector<std::vector<float>>> values;
|
|
std::vector<std::vector<int>> dims;
|
|
for (int i = 0; i < num_batches; i++) {
|
|
std::vector<std::vector<float>> one_value;
|
|
int num = -1;
|
|
for (int j = 0; j < batch_size; j++) {
|
|
int idx = i * batch_size + j;
|
|
std::string line = lines[idx];
|
|
vector<float> item = splitString<float>(line);
|
|
if (num < 0) {
|
|
num = item.size();
|
|
} else {
|
|
assert(num == item.size());
|
|
}
|
|
one_value.push_back(item);
|
|
}
|
|
vector<int> batch_dims = file_dims;
|
|
batch_dims[0] = batch_size;
|
|
dims.push_back(batch_dims);
|
|
values.push_back(one_value);
|
|
}
|
|
|
|
writeValues(values, dims, FLAGS_output_text_tensor);
|
|
}
|
|
|
|
} // namespace caffe2
|
|
|
|
int main(int argc, char** argv) {
|
|
caffe2::GlobalInit(&argc, &argv);
|
|
caffe2::convertImages();
|
|
caffe2::convertValues();
|
|
return 0;
|
|
}
|