Commit 461767d6 authored by fbarchard@google.com's avatar fbarchard@google.com

Bilinear upsample

BUG=208
TEST=out\release\libyuv_unittest --gtest_filter=*ARGBScale*640*

Review URL: https://webrtc-codereview.appspot.com/1303006

git-svn-id: http://libyuv.googlecode.com/svn/trunk@664 16f28f9a-4ce2-e073-06de-1de4eb20be90
parent 4127a263
Name: libyuv Name: libyuv
URL: http://code.google.com/p/libyuv/ URL: http://code.google.com/p/libyuv/
Version: 663 Version: 664
License: BSD
License File: LICENSE License File: LICENSE
Description: Description:
......
...@@ -19,12 +19,11 @@ extern "C" { ...@@ -19,12 +19,11 @@ extern "C" {
#endif #endif
// TODO(fbarchard): Remove kMaxStride. // TODO(fbarchard): Remove kMaxStride.
// Functions should allocate a single row buffer of this size on the stack. #ifdef __arm__
// Functions that allocate more than one row buffer may fail or cause stack #define kMaxStride (1920 * 4)
// probe. #else
// This size is a retina Mac pixels of 32 bit ARGB. #define kMaxStride (4096 * 4)
// Functions may want less for 8 or 16 bit row buffers. #endif
#define kMaxStride (2880 * 4)
#define IS_ALIGNED(p, a) (!((uintptr_t)(p) & ((a) - 1))) #define IS_ALIGNED(p, a) (!((uintptr_t)(p) & ((a) - 1)))
#if defined(__CLR_VER) || defined(COVERAGE_ENABLED) || \ #if defined(__CLR_VER) || defined(COVERAGE_ENABLED) || \
......
...@@ -11,6 +11,6 @@ ...@@ -11,6 +11,6 @@
#ifndef INCLUDE_LIBYUV_VERSION_H_ // NOLINT #ifndef INCLUDE_LIBYUV_VERSION_H_ // NOLINT
#define INCLUDE_LIBYUV_VERSION_H_ #define INCLUDE_LIBYUV_VERSION_H_
#define LIBYUV_VERSION 663 #define LIBYUV_VERSION 664
#endif // INCLUDE_LIBYUV_VERSION_H_ NOLINT #endif // INCLUDE_LIBYUV_VERSION_H_ NOLINT
...@@ -24,7 +24,6 @@ extern "C" { ...@@ -24,7 +24,6 @@ extern "C" {
#endif #endif
// ARGB scaling uses bilinear or point, but not box filter. // ARGB scaling uses bilinear or point, but not box filter.
#if !defined(LIBYUV_DISABLE_NEON) && \ #if !defined(LIBYUV_DISABLE_NEON) && \
(defined(__ARM_NEON__) || defined(LIBYUV_NEON)) (defined(__ARM_NEON__) || defined(LIBYUV_NEON))
#define HAS_SCALEARGBROWDOWNEVEN_NEON #define HAS_SCALEARGBROWDOWNEVEN_NEON
...@@ -1297,8 +1296,7 @@ static void ScaleARGBDownEven(int src_width, int src_height, ...@@ -1297,8 +1296,7 @@ static void ScaleARGBDownEven(int src_width, int src_height,
// ScaleARGB ARGB to/from any dimensions, with bilinear // ScaleARGB ARGB to/from any dimensions, with bilinear
// interpolation. // interpolation.
static void ScaleARGBBilinearDown(int src_width, int src_height,
static void ScaleARGBBilinear(int src_width, int src_height,
int dst_width, int dst_height, int dst_width, int dst_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_argb, uint8* dst_argb) { const uint8* src_argb, uint8* dst_argb) {
...@@ -1308,17 +1306,35 @@ static void ScaleARGBBilinear(int src_width, int src_height, ...@@ -1308,17 +1306,35 @@ static void ScaleARGBBilinear(int src_width, int src_height,
SIMD_ALIGNED(uint8 row[kMaxStride + 16]); SIMD_ALIGNED(uint8 row[kMaxStride + 16]);
void (*ScaleARGBFilterRows)(uint8* dst_argb, const uint8* src_argb, void (*ScaleARGBFilterRows)(uint8* dst_argb, const uint8* src_argb,
ptrdiff_t src_stride, int dst_width, int source_y_fraction) = ptrdiff_t src_stride, int dst_width, int source_y_fraction) =
ScaleARGBFilterRows_C; ARGBInterpolateRow_C;
#if defined(HAS_SCALEARGBFILTERROWS_SSE2) #if defined(HAS_ARGBINTERPOLATEROW_SSE2)
if (TestCpuFlag(kCpuHasSSE2) && IS_ALIGNED(src_width, 4) && if (TestCpuFlag(kCpuHasSSE2) && src_width >= 4) {
IS_ALIGNED(src_argb, 16) && IS_ALIGNED(src_stride, 16)) { ScaleARGBFilterRows = ARGBInterpolateRow_Any_SSE2;
ScaleARGBFilterRows = ScaleARGBFilterRows_SSE2; if (IS_ALIGNED(src_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_Unaligned_SSE2;
if (IS_ALIGNED(src_argb, 16) && IS_ALIGNED(src_stride, 16)) {
ScaleARGBFilterRows = ARGBInterpolateRow_SSE2;
}
}
} }
#endif #endif
#if defined(HAS_SCALEARGBFILTERROWS_SSSE3) #if defined(HAS_ARGBINTERPOLATEROW_SSSE3)
if (TestCpuFlag(kCpuHasSSSE3) && IS_ALIGNED(src_width, 4) && if (TestCpuFlag(kCpuHasSSSE3) && src_width >= 4) {
IS_ALIGNED(src_argb, 16) && IS_ALIGNED(src_stride, 16)) { ScaleARGBFilterRows = ARGBInterpolateRow_Any_SSSE3;
ScaleARGBFilterRows = ScaleARGBFilterRows_SSSE3; if (IS_ALIGNED(src_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_Unaligned_SSSE3;
if (IS_ALIGNED(src_argb, 16) && IS_ALIGNED(src_stride, 16)) {
ScaleARGBFilterRows = ARGBInterpolateRow_SSSE3;
}
}
}
#endif
#if defined(HAS_ARGBINTERPOLATEROW_NEON)
if (TestCpuFlag(kCpuHasNEON) && src_width >= 4) {
ScaleARGBFilterRows = ARGBInterpolateRow_Any_NEON;
if (IS_ALIGNED(src_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_NEON;
}
} }
#endif #endif
void (*ScaleARGBFilterCols)(uint8* dst_argb, const uint8* src_argb, void (*ScaleARGBFilterCols)(uint8* dst_argb, const uint8* src_argb,
...@@ -1328,15 +1344,22 @@ static void ScaleARGBBilinear(int src_width, int src_height, ...@@ -1328,15 +1344,22 @@ static void ScaleARGBBilinear(int src_width, int src_height,
ScaleARGBFilterCols = ScaleARGBFilterCols_SSSE3; ScaleARGBFilterCols = ScaleARGBFilterCols_SSSE3;
} }
#endif #endif
#if defined(HAS_SCALEARGBFILTERROWS_NEON) int dx = 0;
if (TestCpuFlag(kCpuHasNEON) && IS_ALIGNED(src_width, 4)) { int dy = 0;
ScaleARGBFilterRows = ScaleARGBFilterRows_NEON; int x = 0;
int y = 0;
if (dst_width <= src_width) {
dx = (src_width << 16) / dst_width;
x = (dx >> 1) - 32768;
} else if (dst_width > 1) {
dx = ((src_width - 1) << 16) / (dst_width - 1);
}
if (dst_height <= src_height) {
dy = (src_height << 16) / dst_height;
y = (dy >> 1) - 32768;
} else if (dst_height > 1) {
dy = ((src_height - 1) << 16) / (dst_height - 1);
} }
#endif
int dx = (src_width << 16) / dst_width;
int dy = (src_height << 16) / dst_height;
int x = (dx >= 65536) ? ((dx >> 1) - 32768) : (dx >> 1);
int y = (dy >= 65536) ? ((dy >> 1) - 32768) : (dy >> 1);
int maxy = (src_height > 1) ? ((src_height - 1) << 16) - 1 : 0; int maxy = (src_height > 1) ? ((src_height - 1) << 16) - 1 : 0;
for (int j = 0; j < dst_height; ++j) { for (int j = 0; j < dst_height; ++j) {
if (y > maxy) { if (y > maxy) {
...@@ -1352,6 +1375,112 @@ static void ScaleARGBBilinear(int src_width, int src_height, ...@@ -1352,6 +1375,112 @@ static void ScaleARGBBilinear(int src_width, int src_height,
} }
} }
// ScaleARGB ARGB to/from any dimensions, with bilinear
// interpolation.
static void ScaleARGBBilinearUp(int src_width, int src_height,
int dst_width, int dst_height,
int src_stride, int dst_stride,
const uint8* src_argb, uint8* dst_argb) {
assert(dst_width > 0);
assert(dst_height > 0);
assert(dst_width * 4 <= kMaxStride);
void (*ScaleARGBFilterRows)(uint8* dst_argb, const uint8* src_argb,
ptrdiff_t src_stride, int dst_width, int source_y_fraction) =
ARGBInterpolateRow_C;
#if defined(HAS_ARGBINTERPOLATEROW_SSE2)
if (TestCpuFlag(kCpuHasSSE2) && dst_width >= 4) {
ScaleARGBFilterRows = ARGBInterpolateRow_Any_SSE2;
if (IS_ALIGNED(dst_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_Unaligned_SSE2;
if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride, 16)) {
ScaleARGBFilterRows = ARGBInterpolateRow_SSE2;
}
}
}
#endif
#if defined(HAS_ARGBINTERPOLATEROW_SSSE3)
if (TestCpuFlag(kCpuHasSSSE3) && dst_width >= 4) {
ScaleARGBFilterRows = ARGBInterpolateRow_Any_SSSE3;
if (IS_ALIGNED(dst_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_Unaligned_SSSE3;
if (IS_ALIGNED(dst_argb, 16) && IS_ALIGNED(dst_stride, 16)) {
ScaleARGBFilterRows = ARGBInterpolateRow_SSSE3;
}
}
}
#endif
#if defined(HAS_ARGBINTERPOLATEROW_NEON)
if (TestCpuFlag(kCpuHasNEON) && dst_width >= 4) {
ScaleARGBFilterRows = ARGBInterpolateRow_Any_NEON;
if (IS_ALIGNED(dst_width, 4)) {
ScaleARGBFilterRows = ARGBInterpolateRow_NEON;
}
}
#endif
void (*ScaleARGBFilterCols)(uint8* dst_argb, const uint8* src_argb,
int dst_width, int x, int dx) = ScaleARGBFilterCols_C;
#if defined(HAS_SCALEARGBFILTERCOLS_SSSE3)
if (TestCpuFlag(kCpuHasSSSE3)) {
ScaleARGBFilterCols = ScaleARGBFilterCols_SSSE3;
}
#endif
int dx = 0;
int dy = 0;
int x = 0;
int y = 0;
if (dst_width <= src_width) {
dx = (src_width << 16) / dst_width;
x = (dx >> 1) - 32768;
} else if (dst_width > 1) {
dx = ((src_width - 1) << 16) / (dst_width - 1);
}
if (dst_height <= src_height) {
dy = (src_height << 16) / dst_height;
y = (dy >> 1) - 32768;
} else if (dst_height > 1) {
dy = ((src_height - 1) << 16) / (dst_height - 1);
}
int maxy = (src_height > 1) ? ((src_height - 1) << 16) - 1 : 0;
if (y > maxy) {
y = maxy;
}
int yi = y >> 16;
int yf = (y >> 8) & 255;
const uint8* src = src_argb + yi * src_stride;
SIMD_ALIGNED(uint8 row[2 * kMaxStride]);
uint8* rowptr = row;
int rowstride = kMaxStride;
int lasty = 0;
ScaleARGBFilterCols(rowptr, src, dst_width, x, dx);
if (src_height > 1) {
src += src_stride;
}
ScaleARGBFilterCols(rowptr + rowstride, src, dst_width, x, dx);
src += src_stride;
for (int j = 0; j < dst_height; ++j) {
yi = y >> 16;
yf = (y >> 8) & 255;
if (yi != lasty) {
if (y <= maxy) {
y = maxy;
yi = y >> 16;
yf = (y >> 8) & 255;
} else {
ScaleARGBFilterCols(rowptr, src, dst_width, x, dx);
rowptr += rowstride;
rowstride = -rowstride;
lasty = yi;
src += src_stride;
}
}
ScaleARGBFilterRows(dst_argb, rowptr, rowstride, dst_width, yf);
dst_argb += dst_stride;
y += dy;
}
}
// Scales a single row of pixels using point sampling. // Scales a single row of pixels using point sampling.
// Code is adapted from libyuv bilinear yuv scaling, but with bilinear // Code is adapted from libyuv bilinear yuv scaling, but with bilinear
// interpolation off, and argb pixels instead of yuv. // interpolation off, and argb pixels instead of yuv.
...@@ -1406,11 +1535,17 @@ static void ScaleARGBAnySize(int src_width, int src_height, ...@@ -1406,11 +1535,17 @@ static void ScaleARGBAnySize(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_argb, uint8* dst_argb, const uint8* src_argb, uint8* dst_argb,
FilterMode filtering) { FilterMode filtering) {
if (!filtering || (src_width * 4 > kMaxStride)) { if (!filtering ||
(src_width * 4 > kMaxStride && dst_width * 4 > kMaxStride)) {
ScaleARGBSimple(src_width, src_height, dst_width, dst_height, ScaleARGBSimple(src_width, src_height, dst_width, dst_height,
src_stride, dst_stride, src_argb, dst_argb); src_stride, dst_stride, src_argb, dst_argb);
return;
}
if (dst_height <= src_height || dst_width * 4 > kMaxStride) {
ScaleARGBBilinearDown(src_width, src_height, dst_width, dst_height,
src_stride, dst_stride, src_argb, dst_argb);
} else { } else {
ScaleARGBBilinear(src_width, src_height, dst_width, dst_height, ScaleARGBBilinearUp(src_width, src_height, dst_width, dst_height,
src_stride, dst_stride, src_argb, dst_argb); src_stride, dst_stride, src_argb, dst_argb);
} }
} }
......
...@@ -123,20 +123,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy2_Bilinear) { ...@@ -123,20 +123,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy2_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
}
TEST_F(libyuvTest, ARGBScaleDownBy2_Box) {
const int src_width = benchmark_width_;
const int src_height = benchmark_height_;
const int dst_width = src_width / 2;
const int dst_height = src_height / 2;
int max_diff = ARGBTestFilter(src_width, src_height,
dst_width, dst_height,
kFilterBox,
benchmark_iterations_);
EXPECT_LE(max_diff, 1);
} }
TEST_F(libyuvTest, ARGBScaleDownBy4_None) { TEST_F(libyuvTest, ARGBScaleDownBy4_None) {
...@@ -162,20 +149,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy4_Bilinear) { ...@@ -162,20 +149,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy4_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
}
TEST_F(libyuvTest, ARGBScaleDownBy4_Box) {
const int src_width = benchmark_width_;
const int src_height = benchmark_height_;
const int dst_width = src_width / 4;
const int dst_height = src_height / 4;
int max_diff = ARGBTestFilter(src_width, src_height,
dst_width, dst_height,
kFilterBox,
benchmark_iterations_);
EXPECT_LE(max_diff, 1);
} }
TEST_F(libyuvTest, ARGBScaleDownBy5_None) { TEST_F(libyuvTest, ARGBScaleDownBy5_None) {
...@@ -201,20 +175,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy5_Bilinear) { ...@@ -201,20 +175,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy5_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
}
TEST_F(libyuvTest, ARGBScaleDownBy5_Box) {
const int src_width = benchmark_width_;
const int src_height = benchmark_height_;
const int dst_width = src_width / 5;
const int dst_height = src_height / 5;
int max_diff = ARGBTestFilter(src_width, src_height,
dst_width, dst_height,
kFilterBox,
benchmark_iterations_);
EXPECT_LE(max_diff, 1);
} }
TEST_F(libyuvTest, ARGBScaleDownBy8_None) { TEST_F(libyuvTest, ARGBScaleDownBy8_None) {
...@@ -240,20 +201,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy8_Bilinear) { ...@@ -240,20 +201,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy8_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
}
TEST_F(libyuvTest, ARGBScaleDownBy8_Box) {
const int src_width = benchmark_width_;
const int src_height = benchmark_height_;
const int dst_width = src_width / 8;
const int dst_height = src_height / 8;
int max_diff = ARGBTestFilter(src_width, src_height,
dst_width, dst_height,
kFilterBox,
benchmark_iterations_);
EXPECT_LE(max_diff, 1);
} }
TEST_F(libyuvTest, ARGBScaleDownBy16_None) { TEST_F(libyuvTest, ARGBScaleDownBy16_None) {
...@@ -279,7 +227,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy16_Bilinear) { ...@@ -279,7 +227,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy16_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
TEST_F(libyuvTest, ARGBScaleDownBy34_None) { TEST_F(libyuvTest, ARGBScaleDownBy34_None) {
...@@ -305,7 +253,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy34_Bilinear) { ...@@ -305,7 +253,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy34_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
TEST_F(libyuvTest, ARGBScaleDownBy38_None) { TEST_F(libyuvTest, ARGBScaleDownBy38_None) {
...@@ -331,7 +279,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy38_Bilinear) { ...@@ -331,7 +279,7 @@ TEST_F(libyuvTest, ARGBScaleDownBy38_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
TEST_F(libyuvTest, ARGBScaleTo1366x768_None) { TEST_F(libyuvTest, ARGBScaleTo1366x768_None) {
...@@ -357,7 +305,7 @@ TEST_F(libyuvTest, ARGBScaleTo1366x768_Bilinear) { ...@@ -357,7 +305,7 @@ TEST_F(libyuvTest, ARGBScaleTo1366x768_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
...@@ -384,7 +332,7 @@ TEST_F(libyuvTest, ARGBScaleTo1280x720_Bilinear) { ...@@ -384,7 +332,7 @@ TEST_F(libyuvTest, ARGBScaleTo1280x720_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
TEST_F(libyuvTest, ARGBScaleTo853x480_None) { TEST_F(libyuvTest, ARGBScaleTo853x480_None) {
...@@ -410,7 +358,7 @@ TEST_F(libyuvTest, ARGBScaleTo853x480_Bilinear) { ...@@ -410,7 +358,7 @@ TEST_F(libyuvTest, ARGBScaleTo853x480_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
TEST_F(libyuvTest, ARGBScaleFrom640x360_None) { TEST_F(libyuvTest, ARGBScaleFrom640x360_None) {
...@@ -436,7 +384,7 @@ TEST_F(libyuvTest, ARGBScaleFrom640x360_Bilinear) { ...@@ -436,7 +384,7 @@ TEST_F(libyuvTest, ARGBScaleFrom640x360_Bilinear) {
dst_width, dst_height, dst_width, dst_height,
kFilterBilinear, kFilterBilinear,
benchmark_iterations_); benchmark_iterations_);
EXPECT_LE(max_diff, 1); EXPECT_LE(max_diff, 2);
} }
} // namespace libyuv } // namespace libyuv
...@@ -32,6 +32,7 @@ int fileindex_rec = 0; // argv argument contains the reconstructed file name. ...@@ -32,6 +32,7 @@ int fileindex_rec = 0; // argv argument contains the reconstructed file name.
int num_rec = 0; // Number of reconstructed images. int num_rec = 0; // Number of reconstructed images.
int num_skip_org = 0; // Number of frames to skip in original. int num_skip_org = 0; // Number of frames to skip in original.
int num_frames = 0; // Number of frames to convert. int num_frames = 0; // Number of frames to convert.
int filter = 1; // Bilinear filter for scaling.
// Parse PYUV format. ie name.1920x800_24Hz_P420.yuv // Parse PYUV format. ie name.1920x800_24Hz_P420.yuv
bool ExtractResolutionFromFilename(const char* name, bool ExtractResolutionFromFilename(const char* name,
...@@ -58,6 +59,7 @@ void PrintHelp(const char * program) { ...@@ -58,6 +59,7 @@ void PrintHelp(const char * program) {
" resolution (ie. " " resolution (ie. "
"name.1920x800_24Hz_P420.yuv)\n"); "name.1920x800_24Hz_P420.yuv)\n");
printf(" -d <width> <height> .... specify destination resolution.\n"); printf(" -d <width> <height> .... specify destination resolution.\n");
printf(" -f <filter> ............ 0 = point, 1 = bilinear (default).\n");
printf(" -skip <src_argb> ....... Number of frame to skip of src_argb\n"); printf(" -skip <src_argb> ....... Number of frame to skip of src_argb\n");
printf(" -frames <num> .......... Number of frames to convert\n"); printf(" -frames <num> .......... Number of frames to convert\n");
printf(" -v ..................... verbose\n"); printf(" -v ..................... verbose\n");
...@@ -82,6 +84,8 @@ void ParseOptions(int argc, const char* argv[]) { ...@@ -82,6 +84,8 @@ void ParseOptions(int argc, const char* argv[]) {
num_skip_org = atoi(argv[++c]); // NOLINT num_skip_org = atoi(argv[++c]); // NOLINT
} else if (!strcmp(argv[c], "-frames") && c + 1 < argc) { } else if (!strcmp(argv[c], "-frames") && c + 1 < argc) {
num_frames = atoi(argv[++c]); // NOLINT num_frames = atoi(argv[++c]); // NOLINT
} else if (!strcmp(argv[c], "-f") && c + 1 < argc) {
filter = atoi(argv[++c]); // NOLINT
} else if (argv[c][0] == '-') { } else if (argv[c][0] == '-') {
fprintf(stderr, "Unknown option. %s\n", argv[c]); fprintf(stderr, "Unknown option. %s\n", argv[c]);
} else if (fileindex_org == 0) { } else if (fileindex_org == 0) {
...@@ -202,8 +206,9 @@ int main(int argc, const char* argv[]) { ...@@ -202,8 +206,9 @@ int main(int argc, const char* argv[]) {
if (num_frames && number_of_frames >= num_frames) if (num_frames && number_of_frames >= num_frames)
break; break;
size_t bytes_org = fread(ch_org, sizeof(uint8), org_size, file_org); size_t bytes_org = fread(ch_org, sizeof(uint8),
if (bytes_org < total_size) static_cast<size_t>(org_size), file_org);
if (bytes_org < static_cast<size_t>(org_size))
break; break;
for (int cur_rec = 0; cur_rec < num_rec; ++cur_rec) { for (int cur_rec = 0; cur_rec < num_rec; ++cur_rec) {
...@@ -211,8 +216,17 @@ int main(int argc, const char* argv[]) { ...@@ -211,8 +216,17 @@ int main(int argc, const char* argv[]) {
image_width, image_height, image_width, image_height,
ch_dst, dst_width * 4, ch_dst, dst_width * 4,
dst_width, dst_height, dst_width, dst_height,
libyuv::kFilterBilinear); static_cast<libyuv::FilterMode>(filter));
// Output scaled ARGB.
if (strstr(argv[fileindex_rec + cur_rec], "_ARGB.")) {
size_t bytes_rec = fwrite(ch_dst, sizeof(uint8),
static_cast<size_t>(dst_size),
file_rec[cur_rec]);
if (bytes_rec < static_cast<size_t>(dst_size))
break;
} else {
// Output YUV.
int half_width = (dst_width + 1) / 2; int half_width = (dst_width + 1) / 2;
int half_height = (dst_height + 1) / 2; int half_height = (dst_height + 1) / 2;
libyuv::ARGBToI420(ch_dst, dst_width * 4, libyuv::ARGBToI420(ch_dst, dst_width * 4,
...@@ -222,9 +236,11 @@ int main(int argc, const char* argv[]) { ...@@ -222,9 +236,11 @@ int main(int argc, const char* argv[]) {
half_width * half_height, half_width, half_width * half_height, half_width,
dst_width, dst_height); dst_width, dst_height);
size_t bytes_rec = fwrite(ch_rec, sizeof(uint8), size_t bytes_rec = fwrite(ch_rec, sizeof(uint8),
total_size, file_rec[cur_rec]); static_cast<size_t>(total_size),
if (bytes_rec < total_size) file_rec[cur_rec]);
if (bytes_rec < static_cast<size_t>(total_size))
break; break;
}
if (verbose) { if (verbose) {
printf("%5d", number_of_frames); printf("%5d", number_of_frames);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment