forked from superlich7/caffe
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconvertJpg2Yuv.cpp
185 lines (163 loc) · 5.25 KB
/
convertJpg2Yuv.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/highgui/highgui_c.h>
#include <opencv2/imgproc/imgproc.hpp>
#include <boost/shared_ptr.hpp>
#include <cstdlib>
#include <ctime>
#include <iostream>
#include <string>
#include <vector>
#include <fstream>
#include "glog/logging.h"
using std::string;
using std::stringstream;
using std::ofstream;
using std::ifstream;
using std::endl;
DEFINE_bool(yuv2jpg, false,
"The function will change to convert yuv to JPG, and the default is jpg2yuv");
// Specify the width and height to determine the size of the image to use.
// jpg2yuv: If JPG is converted to yuv,
// JPG will be resize before conversion after setting size.
// yuv2jpg: Convert yuv data to JPG, requires providing the size of yuvdata.
DEFINE_int32(width, 0, "The width of the picture.");
DEFINE_int32(height, 0, "The height of the picture.");
void readYUV(string name, cv::Mat img, int h, int w) {
std::ifstream fin(name);
unsigned char a;
for (int i = 0; i < h; i++)
for (int j = 0; j < w; j++) {
a = fin.get();
img.at<char>(i, j) = a;
fin.get();
}
fin.close();
}
cv::Mat yuv420sp2Bgr24(cv::Mat yuv_image) {
cv::Mat bgr_image(yuv_image.rows / 3 * 2, yuv_image.cols, CV_8UC3);
cvtColor(yuv_image, bgr_image, CV_YUV420sp2BGR);
return bgr_image;
}
cv::Mat convertYuv2Mat(string img_name, cv::Size inGeometry) {
cv::Mat img = cv::Mat(inGeometry, CV_8UC1);
readYUV(img_name, img, inGeometry.height, inGeometry.width);
return img;
}
cv::Mat convertYuv2Mat(string img_name, int width, int height) {
cv::Size inGeometry_(width, height);
return convertYuv2Mat(img_name, inGeometry_);
}
void jpg2yuv(string image_path) {
ifstream stream(image_path.c_str());
if (!stream.good())
LOG(FATAL) << "file does not exist.";
cv::Mat src = cv::imread(image_path);
cv::Mat srcImg;
if (FLAGS_width > 0 && FLAGS_height > 0) {
cv::resize(src, srcImg, cv::Size(FLAGS_width, FLAGS_height), CV_INTER_CUBIC);
LOG(INFO) << "The image has been reiszed: " <<
FLAGS_width << "x" << FLAGS_height;
} else {
srcImg = src;
}
LOG(INFO) << "srcImg";
LOG(INFO) << "height: " << srcImg.rows;
LOG(INFO) << "width: " << srcImg.cols;
int yuv_height = srcImg.rows * 3 / 2;
cv::Mat yuv_image(yuv_height, srcImg.rows, CV_8UC1);
cvtColor(srcImg, yuv_image, CV_BGR2YUV_I420);
LOG(INFO) << "yuvImg";
LOG(INFO) << "height: " << yuv_image.rows;
LOG(INFO) << "width: " << yuv_image.cols;
// yuvdata
char* yuvdata = new char[yuv_height*srcImg.cols];
int index = 0;
for (int i = 0; i < yuv_height; i++) {
for (int j = 0; j < srcImg.cols; j++) {
yuvdata[index] = yuv_image.at<char>(i, j);
index++;
}
}
// YUV_I420 -> yuv420sp
int width = srcImg.cols;
int height = srcImg.rows;
LOG(INFO) << "width: " << width << "," << "height: " << height;
int frameSize = width * height;
int qFrameSize = frameSize / 4;
int tempFrameSize = frameSize * 5 / 4;
char* output = new char[frameSize * 3 / 2];
LOG(INFO) << "size: " << frameSize << ", " <<
qFrameSize << ", " << tempFrameSize;
memcpy(output, yuvdata, frameSize);
for (int i = 0; i < qFrameSize; i++) {
output[frameSize + i * 2] = yuvdata[tempFrameSize + i];
output[frameSize + i * 2 + 1] = yuvdata[frameSize + i];
}
delete[] yuvdata;
delete[] output;
stringstream ss;
string file_name;
int position = image_path.find_last_of('/');
string fileName(image_path.substr(position + 1));
ss << image_path << ".yuv";
ss >> file_name;
ofstream fout(file_name);
for (int i = 0; i < yuv_height; i++) {
for (int j = 0; j < srcImg.cols; j++) {
fout << output[i*srcImg.cols+j] << endl;
}
}
fout.close();
LOG(INFO) << "JPG had been converted to yuv data: " << file_name;
// yuv2jpg
cv::Mat yuv = convertYuv2Mat(file_name, yuv_image.cols, yuv_image.rows);
cv::Mat image = yuv420sp2Bgr24(yuv);
cv::imwrite("output.jpg", image);
LOG(INFO) << "yuvData converted to JPG image: output.jgp, " <<
"verify whether correct.";
}
void yuv2jpg(string yuv_path) {
if (FLAGS_width == 0 || FLAGS_height == 0) {
LOG(ERROR) << "width = 0, or height = 0";
}
cv::Mat yuv = convertYuv2Mat(yuv_path, FLAGS_width, FLAGS_height);
cv::Mat image = yuv420sp2Bgr24(yuv);
stringstream ss;
string file_name;
int position = yuv_path.find_last_of('/');
string fileName(yuv_path.substr(position + 1));
ss << yuv_path << ".jpg";
ss >> file_name;
cv::imwrite(file_name, image);
LOG(INFO) << "yuvData had been converted to JPG: " << file_name;
}
int main(int argc, char** argv) {
#ifdef USE_OPENCV
::google::InitGoogleLogging(argv[0]);
// Print output to stderr (while still logging)
FLAGS_alsologtostderr = 1;
#ifndef GFLAGS_GFLAGS_H_
namespace gflags = google;
#endif
gflags::SetUsageMessage("jgp2yuv");
gflags::ParseCommandLineFlags(&argc, &argv, true);
if (argc != 2) {
gflags::ShowUsageWithFlagsRestrict(argv[0], "tools/jpg2yuv");
return 1;
}
string image_path = argv[1];
string type = image_path.substr(image_path.find_last_of('.') + 1);
// check
if (FLAGS_yuv2jpg)
CHECK(type == "yuv") << "yuv";
else
CHECK(type == "jpg") << "jpg";
// convert
if (FLAGS_yuv2jpg)
yuv2jpg(argv[1]);
else
jpg2yuv(argv[1]);
return 0;
}
#endif // USE_OPENCV