forked from frkd-dev/yuvit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathyuvit.cpp
508 lines (425 loc) · 12.4 KB
/
yuvit.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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
#include <sstream>
#include <string>
#include <vector>
#if defined(__MINGW__)
#undef _WIN32
#undef __WIN32__
#endif
#define FREEIMAGE_LIB
#include "FreeImage.h"
#include "getopt_pp.h"
using namespace std;
#define LOG_MESSAGE(...) {printf(__VA_ARGS__); printf("\n");}
#define LOG_ERROR(...) {printf("ERROR: "); printf(__VA_ARGS__); printf("\n");}
enum YUVFormat
{
YUV_YUV,
YUV_UYVY,
YUV_YUYV,
YUV_YYUV
};
enum YUVScale
{
SCALE_H2V2,
SCALE_H2V1,
SCALE_H1V2,
SCALE_H1V1
};
class Config
{
public:
bool appendMode; /* if not zero append YUV image(s) to output file */
bool uvInterleave; /* if not zero, UV rows in planar images are interleaved*/
bool uvOrderSwap; /**/
uint32_t yuvFormat; /* YUV output mode. Default: h2v2 */
uint32_t uvScale; /* Defines how UV components are scaled in planar mode */
uint32_t seqStart; /* Sequence start for multiple files */
uint32_t seqEnd; /* Sequence end for multiple files */
string inFileNamePattern;
string outFileNamePattern;
Config();
// Returns "false" on errors and "true" if no errors
bool ParseArgs(char* args[], int count);
private:
bool ParseSequenceRange(string range);
};
void PrintHelp();
string ExpandPattern(string pattern, int counter);
int main(int argc, char* argv[])
{
Config cfg;
string inFileName, outFileName;
FILE* hOutFile;
uint8_t errorFlag = 1; // By default we will exiting with error
FIBITMAP *inImage = 0;
uint32_t lumaWidth, lumaHeight;
uint32_t chromaWidth, chromaHeight;
uint32_t x, y, xMask, yMask;
uint8_t Rc, Gc, Bc;
uint8_t *rgbPixels;
uint8_t *yPixels, *uPixels, *vPixels;
uint8_t *yPtr, *uPtr, *vPtr;
bool warned = false;
if(!cfg.ParseArgs(argv, argc))
exit(1);
FreeImage_Initialise();
LOG_MESSAGE("Processing:");
/* First expanding of output filename. If append mode - this is only one place
where name is expanded */
outFileName = ExpandPattern(cfg.outFileNamePattern, cfg.seqStart);
/* Main loop of passing through all images */
while(cfg.seqStart <= cfg.seqEnd)
{
inFileName = ExpandPattern(cfg.inFileNamePattern, cfg.seqStart);
FREE_IMAGE_FORMAT format = FreeImage_GetFileType(inFileName.c_str());
if(format == FIF_UNKNOWN)
{
LOG_ERROR("Input image format is unknown or unsupported...");
goto HandleError;
}
inImage = FreeImage_Load(format, inFileName.c_str());
if(inImage == 0)
{
LOG_ERROR("Some problem occurred with reading of input image...");
goto HandleError;
}
if( !cfg.appendMode )
outFileName = ExpandPattern(cfg.outFileNamePattern, cfg.seqStart);
hOutFile = fopen( outFileName.c_str(), cfg.appendMode ? "ab" : "wb" );
if(hOutFile == 0)
{
LOG_ERROR("Can not open output file...");
goto HandleError;
}
lumaWidth = FreeImage_GetWidth(inImage);
lumaHeight = FreeImage_GetHeight(inImage);
LOG_MESSAGE("\t%s [%dx%d]", inFileName.c_str(), lumaWidth, lumaHeight);
if( (lumaWidth & 1) && cfg.yuvFormat != YUV_YUV)
{
LOG_ERROR("Width of source image is odd - this is incompatible with packed format...");
goto HandleError;
}
if( !warned && ((lumaWidth & 1) || (lumaHeight & 1)) && (cfg.yuvFormat == YUV_YUV || cfg.yuvFormat == YUV_YYUV))
{
LOG_MESSAGE("Warning! Dimensions of the source image are odd. This may cause incompatibility with some YUV viewers.");
warned = true; // Show warning only once
}
FreeImage_FlipVertical(inImage);
/* Calculate dimensions of destination UV components */
switch(cfg.uvScale)
{
default: /* Default scale h1v1 */
case SCALE_H1V1:
chromaWidth = lumaWidth;
chromaHeight = lumaHeight;
xMask = 0;
yMask = 0;
break;
case SCALE_H2V2:
chromaWidth = lumaWidth / 2;
chromaHeight = lumaHeight / 2;
yMask = xMask = 1;
break;
case SCALE_H1V2:
chromaWidth = lumaWidth;
chromaHeight = lumaHeight / 2;
xMask = 0;
yMask = 1;
break;
case SCALE_H2V1:
chromaWidth = lumaWidth / 2;
chromaHeight = lumaHeight;
xMask = 1;
yMask = 0;
break;
}
// Pointers that are always pointing on buffers
yPixels = new uint8_t[lumaHeight * lumaWidth];
uPixels = new uint8_t[chromaHeight * chromaWidth];
vPixels = new uint8_t[chromaHeight * chromaWidth];
// Pointers we are working with
yPtr = yPixels;
uPtr = uPixels;
vPtr = vPixels;
/* Main converting cycle */
for(y = 0; y < lumaHeight; y++)
{
rgbPixels = FreeImage_GetScanLine(inImage, y);
for(x = 0; x < lumaWidth; x++)
{
Rc = *rgbPixels++;
Gc = *rgbPixels++;
Bc = *rgbPixels++;
*yPtr++ = (0.257f * Rc) + (0.504f * Gc) + (0.098f * Bc) + 16;
if((y & yMask) == 0 && (x & xMask) == 0 && (y / 2) < chromaHeight && (x / 2) < chromaWidth)
{
*uPtr++ = -(0.148f * Rc) - (0.291f * Gc) + (0.439f * Bc) + 128;
*vPtr++ = (0.439f * Rc) - (0.368f * Gc) - (0.071f * Bc) + 128;
}
}
}
if(cfg.uvOrderSwap)
{ // UV components should be swapped, so just swap pointers
uint8_t* tmp = uPixels;
uPixels = vPixels;
vPixels = tmp;
}
yPtr = yPixels;
uPtr = uPixels;
vPtr = vPixels;
if(cfg.yuvFormat == YUV_YUV)
{ // Writing planar image
fwrite(yPixels, 1, lumaWidth * lumaHeight, hOutFile);
if(cfg.uvInterleave)
{ // U and V rows should be interleaved after each other
while(chromaHeight--)
{
fwrite(uPtr, 1, chromaWidth, hOutFile); // Write U line
fwrite(vPtr, 1, chromaWidth, hOutFile); // Write V line
uPtr += chromaWidth;
vPtr += chromaWidth;
}
}else{
// Simply write U and V planes
fwrite(uPixels, 1, chromaWidth * chromaHeight, hOutFile);
fwrite(vPixels, 1, chromaWidth * chromaHeight, hOutFile);
}
}else if(cfg.yuvFormat == YUV_YYUV)
{ // Writing planar image
fwrite(yPixels, 1, lumaWidth * lumaHeight, hOutFile);
// U and V columns should be interleaved after each other
for(uint32_t row = 0; row < chromaHeight; row++)
{
for(uint32_t col = 0; col < chromaWidth; col++)
{ // Write in following order Y, U, Y, V
fwrite(uPtr++, 1, 1, hOutFile);
fwrite(vPtr++, 1, 1, hOutFile);
}
}
}else{
// Writing packed image
if(cfg.yuvFormat == YUV_YUYV)
{
for(uint32_t row = 0; row < lumaHeight; row++)
{
for(uint32_t col = 0; col < lumaWidth; col += 2)
{ // Write in following order Y, U, Y, V
fwrite(yPtr++, 1, 1, hOutFile);
fwrite(uPtr++, 1, 1, hOutFile);
fwrite(yPtr++, 1, 1, hOutFile);
fwrite(vPtr++, 1, 1, hOutFile);
}
}
}else{
for(uint32_t row = 0; row < lumaHeight; row++)
{
for(uint32_t col = 0; col < lumaWidth; col += 2)
{ // Write in following order U, Y, V, Y
fwrite(uPtr++, 1, 1, hOutFile);
fwrite(yPtr++, 1, 1, hOutFile);
fwrite(vPtr++, 1, 1, hOutFile);
fwrite(yPtr++, 1, 1, hOutFile);
}
}
}
}
FreeImage_Unload(inImage);
inImage = 0;
fclose(hOutFile);
hOutFile = 0;
delete yPixels;
delete uPixels;
delete vPixels;
cfg.seqStart++;
}
LOG_MESSAGE("Done!");
errorFlag = 0; // We successful passed all stages, so set flag to zero which means - OK
HandleError:
if(inImage)
{
FreeImage_Unload(inImage);
inImage = 0;
}
if(hOutFile)
{
fclose(hOutFile);
hOutFile = 0;
}
FreeImage_DeInitialise();
return errorFlag;
}
string toString(int value)
{
ostringstream oss;
oss << value;
return oss.str();
}
/*
Description: Function searches for '#' symbols and replaces them by integer value with leading zeros
Returns: Formed string
*/
string ExpandPattern(string pattern, int counter)
{
string result;
string::iterator it = pattern.begin();
uint32_t cntr = 0;
string counterStr = toString(counter);
// Copy from input pattern to resulting string until we meet '#' sign
while( it != pattern.end() && *it != '#') result += *it++;
// Calculate number of '#' signs in input pattern
while( it != pattern.end() && *it == '#')
{
it++;
cntr++;
}
// If we not found any patterns - just return unmodified input pattern
if( !cntr )
return pattern;
// Determine leading zeros
if(cntr > counterStr.length())
{
// We have leading zeros
cntr -= counterStr.length();
}else{
// Resulting counter bigger than pattern has defined,
// therefore there are no leading zeros
cntr = 0;
}
// cntr from now contains number of leading zeros
// Append to result leading zeros
if(cntr)
result.append(cntr, '0');
// Append integer value
result.append(counterStr);
// Copy from input pattern to resulting string rest of input pattern
while( it != pattern.end() ) result += *it++;
return result;
}
Config::Config()
{
appendMode = false; /* if not zero append YUV image(s) to output file */
yuvFormat = YUV_YUV; /* YUV output mode. Default: h2v2 */
uvScale = SCALE_H1V1; /* UV scaling for planar mode. Default: h1v1 */
seqStart = 0; /* Sequence start for multiple files */
seqEnd = 0; /* Sequence end for multiple files */
}
bool Config::ParseArgs(char* args[], int count)
{
using namespace GetOpt;
vector<string> files;
string seqRangeOption;
string yuvFormatOption;
string uvScaleOption;
bool error = true;
GetOpt_pp opt(count, args);
opt >> OptionPresent('a', appendMode);
opt >> OptionPresent('i', uvInterleave);
opt >> OptionPresent('w', uvOrderSwap);
opt >> Option('m', seqRangeOption);
opt >> Option('f', yuvFormatOption);
opt >> Option('s', uvScaleOption);
opt >> GlobalOption(files);
if(files.size() > 0)
{
inFileNamePattern = files[0];
outFileNamePattern = files[0] + ".yuv"; // If output didn't been specified, we will use original name + .yuv
} else {
LOG_ERROR("You've not specified files to process...");
goto HandleError;
}
if(files.size() > 1)
outFileNamePattern = files[1]; // Output file was specified, use its name
// Scaling could be overridden by format, then select scale first
if( uvScaleOption == "h2v2")
uvScale = SCALE_H2V2;
else if(uvScaleOption == "h1v1")
uvScale = SCALE_H1V1;
else if(uvScaleOption == "h1v2")
uvScale = SCALE_H1V2;
else if(uvScaleOption == "h2v1")
uvScale = SCALE_H2V1;
else if( !uvScaleOption.empty())
{
LOG_ERROR("Unknown UV scaling...");
goto HandleError;
}
if( yuvFormatOption == "yuv")
yuvFormat = YUV_YUV;
else if(yuvFormatOption == "yyuv")
yuvFormat = YUV_YYUV;
else if(yuvFormatOption == "yuyv")
yuvFormat = YUV_YUYV, uvScale = SCALE_H2V1; // Packed format always h2v1
else if(yuvFormatOption == "uyvy")
yuvFormat = YUV_UYVY, uvScale = SCALE_H2V1; // Packed format always h2v1
else if( !yuvFormatOption.empty())
{
LOG_ERROR("Unknown YUV format...");
goto HandleError;
}
if(!seqRangeOption.empty())
if(!ParseSequenceRange(seqRangeOption))
{
LOG_ERROR("You've specified bad sequence range...");
goto HandleError;
}
error = false;
HandleError:
if(error)
PrintHelp();
return !error;
}
bool Config::ParseSequenceRange(string range)
{
string::iterator it = range.begin();
string seqStartOpt;
string seqEndOpt;
// Copy from input to seqStartOpt until we will find ':' character
while(it != range.end() && *it != ':')
seqStartOpt += *it++;
if(it == range.end() || *it++ != ':')
return false;
// Copy from input to seqEndOpt till the end
while(it != range.end())
seqEndOpt += *it++;
if(seqStartOpt.empty() || seqEndOpt.empty())
return false;
seqStart = atoi(seqStartOpt.c_str());
seqEnd = atoi(seqEndOpt.c_str());
return true;
}
void PrintHelp()
{
printf(
"\nUsage: yuvit [options] [-f format] [-s uvscale] <InFile> [OutFile]\n\n"
"Options:\n"
" -a : Add new image to the end of output file. Don't truncate output file.\n"
" -m <start>:<end> : Multiple file input. Where:\n"
" start : Sequence start\n"
" end : Sequence end\n"
" -i : Interleave UV rows for planar formats\n"
" -w : Swap UV components order\n"
"\nFormats (-f option):\n"
" yuv : Planar format [DEFAULT]\n"
" yuyv : Packed format\n"
" uyvy : Packed format\n"
" yyuv : Planar packed chroma format\n"
"\nUV scales (-s option. Used only with -f and planar formats):\n"
" h1v1 : UV not scaled down [DEFAULT]\n"
" h2v2 : UV scaled down by 2x horizontally and vertically\n"
" h2v1 : UV scaled down by 2x horizontally\n"
" h1v2 : UV scaled down by 2x vertically\n"
"\nNote: Use symbol '#' in file names for enumerators.\n"
"\nExamples:\n"
"yuvit -a -m 0:100 test###.bmp out.yuv\n"
" Convert images from 'test000.bmp' to 'test100.bmp' into single 'out.yuv' file\n"
"\nyuvit -m 10:200 test######.jpg out###.yuv\n"
" Convert images 'test000010.jpg'...'test000200.jpg' into files 'out010.yuv'...'out200.yuv'\n"
"\n"
"Converter to YUV. v" APPVERSION "\n"
"(c) 2011-2012 Alexander Shashkevych\n"
"This tool licensed under LGPL\n"
);
}