Commit fa831883 authored by fbarchard@google.com's avatar fbarchard@google.com

scale port to c. completes all scaling functions.

BUG=303
TESTED=cl /c /TC /Iinclude source/scale.cc
R=tpsiaki@google.com

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

git-svn-id: http://libyuv.googlecode.com/svn/trunk@974 16f28f9a-4ce2-e073-06de-1de4eb20be90
parent f5f5bbf8
Name: libyuv Name: libyuv
URL: http://code.google.com/p/libyuv/ URL: http://code.google.com/p/libyuv/
Version: 973 Version: 974
License: BSD License: BSD
License File: LICENSE License File: LICENSE
......
...@@ -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 973 #define LIBYUV_VERSION 974
#endif // INCLUDE_LIBYUV_VERSION_H_ NOLINT #endif // INCLUDE_LIBYUV_VERSION_H_ NOLINT
...@@ -41,6 +41,7 @@ static void ScalePlaneDown2(int src_width, int src_height, ...@@ -41,6 +41,7 @@ static void ScalePlaneDown2(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_ptr, uint8* dst_ptr, const uint8* src_ptr, uint8* dst_ptr,
enum FilterMode filtering) { enum FilterMode filtering) {
int y;
void (*ScaleRowDown2)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown2)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width) = uint8* dst_ptr, int dst_width) =
filtering == kFilterNone ? ScaleRowDown2_C : filtering == kFilterNone ? ScaleRowDown2_C :
...@@ -82,7 +83,6 @@ static void ScalePlaneDown2(int src_width, int src_height, ...@@ -82,7 +83,6 @@ static void ScalePlaneDown2(int src_width, int src_height,
src_stride = 0; src_stride = 0;
} }
// TODO(fbarchard): Loop through source height to allow odd height. // TODO(fbarchard): Loop through source height to allow odd height.
int y;
for (y = 0; y < dst_height; ++y) { for (y = 0; y < dst_height; ++y) {
ScaleRowDown2(src_ptr, src_stride, dst_ptr, dst_width); ScaleRowDown2(src_ptr, src_stride, dst_ptr, dst_width);
src_ptr += row_stride; src_ptr += row_stride;
...@@ -99,6 +99,7 @@ static void ScalePlaneDown4(int src_width, int src_height, ...@@ -99,6 +99,7 @@ static void ScalePlaneDown4(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_ptr, uint8* dst_ptr, const uint8* src_ptr, uint8* dst_ptr,
enum FilterMode filtering) { enum FilterMode filtering) {
int y;
void (*ScaleRowDown4)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown4)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width) = uint8* dst_ptr, int dst_width) =
filtering ? ScaleRowDown4Box_C : ScaleRowDown4_C; filtering ? ScaleRowDown4Box_C : ScaleRowDown4_C;
...@@ -129,7 +130,6 @@ static void ScalePlaneDown4(int src_width, int src_height, ...@@ -129,7 +130,6 @@ static void ScalePlaneDown4(int src_width, int src_height,
if (filtering == kFilterLinear) { if (filtering == kFilterLinear) {
src_stride = 0; src_stride = 0;
} }
int y;
for (y = 0; y < dst_height; ++y) { for (y = 0; y < dst_height; ++y) {
ScaleRowDown4(src_ptr, src_stride, dst_ptr, dst_width); ScaleRowDown4(src_ptr, src_stride, dst_ptr, dst_width);
src_ptr += row_stride; src_ptr += row_stride;
...@@ -144,11 +144,13 @@ static void ScalePlaneDown34(int src_width, int src_height, ...@@ -144,11 +144,13 @@ static void ScalePlaneDown34(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_ptr, uint8* dst_ptr, const uint8* src_ptr, uint8* dst_ptr,
enum FilterMode filtering) { enum FilterMode filtering) {
assert(dst_width % 3 == 0); int y;
void (*ScaleRowDown34_0)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown34_0)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width); uint8* dst_ptr, int dst_width);
void (*ScaleRowDown34_1)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown34_1)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width); uint8* dst_ptr, int dst_width);
const int filter_stride = (filtering == kFilterLinear) ? 0 : src_stride;
assert(dst_width % 3 == 0);
if (!filtering) { if (!filtering) {
ScaleRowDown34_0 = ScaleRowDown34_C; ScaleRowDown34_0 = ScaleRowDown34_C;
ScaleRowDown34_1 = ScaleRowDown34_C; ScaleRowDown34_1 = ScaleRowDown34_C;
...@@ -193,8 +195,6 @@ static void ScalePlaneDown34(int src_width, int src_height, ...@@ -193,8 +195,6 @@ static void ScalePlaneDown34(int src_width, int src_height,
} }
#endif #endif
const int filter_stride = (filtering == kFilterLinear) ? 0 : src_stride;
int y;
for (y = 0; y < dst_height - 2; y += 3) { for (y = 0; y < dst_height - 2; y += 3) {
ScaleRowDown34_0(src_ptr, filter_stride, dst_ptr, dst_width); ScaleRowDown34_0(src_ptr, filter_stride, dst_ptr, dst_width);
src_ptr += src_stride; src_ptr += src_stride;
...@@ -240,11 +240,13 @@ static void ScalePlaneDown38(int src_width, int src_height, ...@@ -240,11 +240,13 @@ static void ScalePlaneDown38(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_ptr, uint8* dst_ptr, const uint8* src_ptr, uint8* dst_ptr,
enum FilterMode filtering) { enum FilterMode filtering) {
assert(dst_width % 3 == 0); int y;
void (*ScaleRowDown38_3)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown38_3)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width); uint8* dst_ptr, int dst_width);
void (*ScaleRowDown38_2)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleRowDown38_2)(const uint8* src_ptr, ptrdiff_t src_stride,
uint8* dst_ptr, int dst_width); uint8* dst_ptr, int dst_width);
const int filter_stride = (filtering == kFilterLinear) ? 0 : src_stride;
assert(dst_width % 3 == 0);
if (!filtering) { if (!filtering) {
ScaleRowDown38_3 = ScaleRowDown38_C; ScaleRowDown38_3 = ScaleRowDown38_C;
ScaleRowDown38_2 = ScaleRowDown38_C; ScaleRowDown38_2 = ScaleRowDown38_C;
...@@ -287,8 +289,6 @@ static void ScalePlaneDown38(int src_width, int src_height, ...@@ -287,8 +289,6 @@ static void ScalePlaneDown38(int src_width, int src_height,
} }
#endif #endif
const int filter_stride = (filtering == kFilterLinear) ? 0 : src_stride;
int y;
for (y = 0; y < dst_height - 2; y += 3) { for (y = 0; y < dst_height - 2; y += 3) {
ScaleRowDown38_3(src_ptr, filter_stride, dst_ptr, dst_width); ScaleRowDown38_3(src_ptr, filter_stride, dst_ptr, dst_width);
src_ptr += src_stride * 3; src_ptr += src_stride * 3;
...@@ -314,10 +314,10 @@ static void ScalePlaneDown38(int src_width, int src_height, ...@@ -314,10 +314,10 @@ static void ScalePlaneDown38(int src_width, int src_height,
static __inline uint32 SumBox(int iboxwidth, int iboxheight, static __inline uint32 SumBox(int iboxwidth, int iboxheight,
ptrdiff_t src_stride, const uint8* src_ptr) { ptrdiff_t src_stride, const uint8* src_ptr) {
assert(iboxwidth > 0);
assert(iboxheight > 0);
uint32 sum = 0u; uint32 sum = 0u;
int y; int y;
assert(iboxwidth > 0);
assert(iboxheight > 0);
for (y = 0; y < iboxheight; ++y) { for (y = 0; y < iboxheight; ++y) {
int x; int x;
for (x = 0; x < iboxwidth; ++x) { for (x = 0; x < iboxwidth; ++x) {
...@@ -332,19 +332,20 @@ static void ScalePlaneBoxRow_C(int dst_width, int boxheight, ...@@ -332,19 +332,20 @@ static void ScalePlaneBoxRow_C(int dst_width, int boxheight,
int x, int dx, ptrdiff_t src_stride, int x, int dx, ptrdiff_t src_stride,
const uint8* src_ptr, uint8* dst_ptr) { const uint8* src_ptr, uint8* dst_ptr) {
int i; int i;
int boxwidth;
for (i = 0; i < dst_width; ++i) { for (i = 0; i < dst_width; ++i) {
int ix = x >> 16; int ix = x >> 16;
x += dx; x += dx;
int boxwidth = (x >> 16) - ix; boxwidth = (x >> 16) - ix;
*dst_ptr++ = SumBox(boxwidth, boxheight, src_stride, src_ptr + ix) / *dst_ptr++ = SumBox(boxwidth, boxheight, src_stride, src_ptr + ix) /
(boxwidth * boxheight); (boxwidth * boxheight);
} }
} }
static __inline uint32 SumPixels(int iboxwidth, const uint16* src_ptr) { static __inline uint32 SumPixels(int iboxwidth, const uint16* src_ptr) {
assert(iboxwidth > 0);
uint32 sum = 0u; uint32 sum = 0u;
int x; int x;
assert(iboxwidth > 0);
for (x = 0; x < iboxwidth; ++x) { for (x = 0; x < iboxwidth; ++x) {
sum += src_ptr[x]; sum += src_ptr[x];
} }
...@@ -353,16 +354,17 @@ static __inline uint32 SumPixels(int iboxwidth, const uint16* src_ptr) { ...@@ -353,16 +354,17 @@ static __inline uint32 SumPixels(int iboxwidth, const uint16* src_ptr) {
static void ScaleAddCols2_C(int dst_width, int boxheight, int x, int dx, static void ScaleAddCols2_C(int dst_width, int boxheight, int x, int dx,
const uint16* src_ptr, uint8* dst_ptr) { const uint16* src_ptr, uint8* dst_ptr) {
int i;
int scaletbl[2]; int scaletbl[2];
int minboxwidth = (dx >> 16); int minboxwidth = (dx >> 16);
int* scaleptr = scaletbl - minboxwidth;
int boxwidth;
scaletbl[0] = 65536 / (minboxwidth * boxheight); scaletbl[0] = 65536 / (minboxwidth * boxheight);
scaletbl[1] = 65536 / ((minboxwidth + 1) * boxheight); scaletbl[1] = 65536 / ((minboxwidth + 1) * boxheight);
int* scaleptr = scaletbl - minboxwidth;
int i;
for (i = 0; i < dst_width; ++i) { for (i = 0; i < dst_width; ++i) {
int ix = x >> 16; int ix = x >> 16;
x += dx; x += dx;
int boxwidth = (x >> 16) - ix; boxwidth = (x >> 16) - ix;
*dst_ptr++ = SumPixels(boxwidth, src_ptr + ix) * scaleptr[boxwidth] >> 16; *dst_ptr++ = SumPixels(boxwidth, src_ptr + ix) * scaleptr[boxwidth] >> 16;
} }
} }
...@@ -389,27 +391,29 @@ static void ScalePlaneBox(int src_width, int src_height, ...@@ -389,27 +391,29 @@ static void ScalePlaneBox(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_ptr, uint8* dst_ptr) { const uint8* src_ptr, uint8* dst_ptr) {
int j;
// Initial source x/y coordinate and step values as 16.16 fixed point. // Initial source x/y coordinate and step values as 16.16 fixed point.
int x = 0; int x = 0;
int y = 0; int y = 0;
int dx = 0; int dx = 0;
int dy = 0; int dy = 0;
const int max_y = (src_height << 16);
ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox, ScaleSlope(src_width, src_height, dst_width, dst_height, kFilterBox,
&x, &y, &dx, &dy); &x, &y, &dx, &dy);
src_width = Abs(src_width); src_width = Abs(src_width);
const int max_y = (src_height << 16);
// TODO(fbarchard): Remove this and make AddRows handle boxheight 1. // TODO(fbarchard): Remove this and make AddRows handle boxheight 1.
if (!IS_ALIGNED(src_width, 16) || dst_height * 2 > src_height) { if (!IS_ALIGNED(src_width, 16) || dst_height * 2 > src_height) {
uint8* dst = dst_ptr; uint8* dst = dst_ptr;
int j; int j;
for (j = 0; j < dst_height; ++j) { for (j = 0; j < dst_height; ++j) {
int boxheight;
int iy = y >> 16; int iy = y >> 16;
const uint8* src = src_ptr + iy * src_stride; const uint8* src = src_ptr + iy * src_stride;
y += dy; y += dy;
if (y > max_y) { if (y > max_y) {
y = max_y; y = max_y;
} }
int boxheight = (y >> 16) - iy; boxheight = (y >> 16) - iy;
ScalePlaneBoxRow_C(dst_width, boxheight, ScalePlaneBoxRow_C(dst_width, boxheight,
x, dx, src_stride, x, dx, src_stride,
src, dst); src, dst);
...@@ -417,14 +421,15 @@ static void ScalePlaneBox(int src_width, int src_height, ...@@ -417,14 +421,15 @@ static void ScalePlaneBox(int src_width, int src_height,
} }
return; return;
} }
{
// Allocate a row buffer of uint16. // Allocate a row buffer of uint16.
align_buffer_64(row16, src_width * 2); align_buffer_64(row16, src_width * 2);
void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx, void (*ScaleAddCols)(int dst_width, int boxheight, int x, int dx,
const uint16* src_ptr, uint8* dst_ptr) = const uint16* src_ptr, uint8* dst_ptr) =
(dx & 0xffff) ? ScaleAddCols2_C: ScaleAddCols1_C; (dx & 0xffff) ? ScaleAddCols2_C: ScaleAddCols1_C;
void (*ScaleAddRows)(const uint8* src_ptr, ptrdiff_t src_stride, void (*ScaleAddRows)(const uint8* src_ptr, ptrdiff_t src_stride,
uint16* dst_ptr, int src_width, int src_height) = ScaleAddRows_C; uint16* dst_ptr, int src_width, int src_height) = ScaleAddRows_C;
#if defined(HAS_SCALEADDROWS_SSE2) #if defined(HAS_SCALEADDROWS_SSE2)
if (TestCpuFlag(kCpuHasSSE2) && if (TestCpuFlag(kCpuHasSSE2) &&
#ifdef AVOID_OVERREAD #ifdef AVOID_OVERREAD
...@@ -435,15 +440,15 @@ static void ScalePlaneBox(int src_width, int src_height, ...@@ -435,15 +440,15 @@ static void ScalePlaneBox(int src_width, int src_height,
} }
#endif #endif
int j;
for (j = 0; j < dst_height; ++j) { for (j = 0; j < dst_height; ++j) {
int boxheight;
int iy = y >> 16; int iy = y >> 16;
const uint8* src = src_ptr + iy * src_stride; const uint8* src = src_ptr + iy * src_stride;
y += dy; y += dy;
if (y > (src_height << 16)) { if (y > (src_height << 16)) {
y = (src_height << 16); y = (src_height << 16);
} }
int boxheight = (y >> 16) - iy; boxheight = (y >> 16) - iy;
ScaleAddRows(src, src_stride, (uint16*)(row16), ScaleAddRows(src, src_stride, (uint16*)(row16),
src_width, boxheight); src_width, boxheight);
ScaleAddCols(dst_width, boxheight, x, dx, (uint16*)(row16), ScaleAddCols(dst_width, boxheight, x, dx, (uint16*)(row16),
...@@ -451,6 +456,7 @@ static void ScalePlaneBox(int src_width, int src_height, ...@@ -451,6 +456,7 @@ static void ScalePlaneBox(int src_width, int src_height,
dst_ptr += dst_stride; dst_ptr += dst_stride;
} }
free_aligned_buffer_64(row16); free_aligned_buffer_64(row16);
}
} }
// Scale plane down with bilinear interpolation. // Scale plane down with bilinear interpolation.
...@@ -464,13 +470,22 @@ void ScalePlaneBilinearDown(int src_width, int src_height, ...@@ -464,13 +470,22 @@ void ScalePlaneBilinearDown(int src_width, int src_height,
int y = 0; int y = 0;
int dx = 0; int dx = 0;
int dy = 0; int dy = 0;
ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, // TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
&x, &y, &dx, &dy); // Allocate a row buffer.
src_width = Abs(src_width); align_buffer_64(row, src_width);
const int max_y = (src_height - 1) << 16;
int j;
void (*ScaleFilterCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) =
(src_width >= 32768) ? ScaleFilterCols64_C : ScaleFilterCols_C;
void (*InterpolateRow)(uint8* dst_ptr, const uint8* src_ptr, void (*InterpolateRow)(uint8* dst_ptr, const uint8* src_ptr,
ptrdiff_t src_stride, int dst_width, int source_y_fraction) = ptrdiff_t src_stride, int dst_width, int source_y_fraction) =
InterpolateRow_C; InterpolateRow_C;
ScaleSlope(src_width, src_height, dst_width, dst_height, filtering,
&x, &y, &dx, &dy);
src_width = Abs(src_width);
#if defined(HAS_INTERPOLATEROW_SSE2) #if defined(HAS_INTERPOLATEROW_SSE2)
if (TestCpuFlag(kCpuHasSSE2) && src_width >= 16) { if (TestCpuFlag(kCpuHasSSE2) && src_width >= 16) {
InterpolateRow = InterpolateRow_Any_SSE2; InterpolateRow = InterpolateRow_Any_SSE2;
...@@ -518,26 +533,17 @@ void ScalePlaneBilinearDown(int src_width, int src_height, ...@@ -518,26 +533,17 @@ void ScalePlaneBilinearDown(int src_width, int src_height,
} }
#endif #endif
void (*ScaleFilterCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) =
(src_width >= 32768) ? ScaleFilterCols64_C : ScaleFilterCols_C;
#if defined(HAS_SCALEFILTERCOLS_SSSE3) #if defined(HAS_SCALEFILTERCOLS_SSSE3)
if (TestCpuFlag(kCpuHasSSSE3) && src_width < 32768) { if (TestCpuFlag(kCpuHasSSSE3) && src_width < 32768) {
ScaleFilterCols = ScaleFilterCols_SSSE3; ScaleFilterCols = ScaleFilterCols_SSSE3;
} }
#endif #endif
// TODO(fbarchard): Consider not allocating row buffer for kFilterLinear.
// Allocate a row buffer.
align_buffer_64(row, src_width);
const int max_y = (src_height - 1) << 16;
int j;
for (j = 0; j < dst_height; ++j) {
if (y > max_y) { if (y > max_y) {
y = max_y; y = max_y;
} }
for (j = 0; j < dst_height; ++j) {
int yi = y >> 16; int yi = y >> 16;
const uint8* src = src_ptr + yi * src_stride; const uint8* src = src_ptr + yi * src_stride;
if (filtering == kFilterLinear) { if (filtering == kFilterLinear) {
...@@ -549,6 +555,9 @@ void ScalePlaneBilinearDown(int src_width, int src_height, ...@@ -549,6 +555,9 @@ void ScalePlaneBilinearDown(int src_width, int src_height,
} }
dst_ptr += dst_stride; dst_ptr += dst_stride;
y += dy; y += dy;
if (y > max_y) {
y = max_y;
}
} }
free_aligned_buffer_64(row); free_aligned_buffer_64(row);
} }
...@@ -559,18 +568,23 @@ void ScalePlaneBilinearUp(int src_width, int src_height, ...@@ -559,18 +568,23 @@ void ScalePlaneBilinearUp(int src_width, int src_height,
int src_stride, int dst_stride, int src_stride, int dst_stride,
const uint8* src_ptr, uint8* dst_ptr, const uint8* src_ptr, uint8* dst_ptr,
enum FilterMode filtering) { enum FilterMode filtering) {
int j;
// Initial source x/y coordinate and step values as 16.16 fixed point. // Initial source x/y coordinate and step values as 16.16 fixed point.
int x = 0; int x = 0;
int y = 0; int y = 0;
int dx = 0; int dx = 0;
int dy = 0; int dy = 0;
const int max_y = (src_height - 1) << 16;
void (*InterpolateRow)(uint8* dst_ptr, const uint8* src_ptr,
ptrdiff_t src_stride, int dst_width, int source_y_fraction) =
InterpolateRow_C;
void (*ScaleFilterCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) =
filtering ? ScaleFilterCols_C : ScaleCols_C;
ScaleSlope(src_width, src_height, dst_width, dst_height, filtering, ScaleSlope(src_width, src_height, dst_width, dst_height, filtering,
&x, &y, &dx, &dy); &x, &y, &dx, &dy);
src_width = Abs(src_width); src_width = Abs(src_width);
void (*InterpolateRow)(uint8* dst_ptr, const uint8* src_ptr,
ptrdiff_t src_stride, int dst_width, int source_y_fraction) =
InterpolateRow_C;
#if defined(HAS_INTERPOLATEROW_SSE2) #if defined(HAS_INTERPOLATEROW_SSE2)
if (TestCpuFlag(kCpuHasSSE2) && dst_width >= 16) { if (TestCpuFlag(kCpuHasSSE2) && dst_width >= 16) {
InterpolateRow = InterpolateRow_Any_SSE2; InterpolateRow = InterpolateRow_Any_SSE2;
...@@ -618,9 +632,6 @@ void ScalePlaneBilinearUp(int src_width, int src_height, ...@@ -618,9 +632,6 @@ void ScalePlaneBilinearUp(int src_width, int src_height,
} }
#endif #endif
void (*ScaleFilterCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) =
filtering ? ScaleFilterCols_C : ScaleCols_C;
if (filtering && src_width >= 32768) { if (filtering && src_width >= 32768) {
ScaleFilterCols = ScaleFilterCols64_C; ScaleFilterCols = ScaleFilterCols64_C;
} }
...@@ -640,10 +651,10 @@ void ScalePlaneBilinearUp(int src_width, int src_height, ...@@ -640,10 +651,10 @@ void ScalePlaneBilinearUp(int src_width, int src_height,
#endif #endif
} }
const int max_y = (src_height - 1) << 16;
if (y > max_y) { if (y > max_y) {
y = max_y; y = max_y;
} }
{
int yi = y >> 16; int yi = y >> 16;
const uint8* src = src_ptr + yi * src_stride; const uint8* src = src_ptr + yi * src_stride;
...@@ -662,7 +673,6 @@ void ScalePlaneBilinearUp(int src_width, int src_height, ...@@ -662,7 +673,6 @@ void ScalePlaneBilinearUp(int src_width, int src_height,
ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx); ScaleFilterCols(rowptr + rowstride, src, dst_width, x, dx);
src += src_stride; src += src_stride;
int j;
for (j = 0; j < dst_height; ++j) { for (j = 0; j < dst_height; ++j) {
yi = y >> 16; yi = y >> 16;
if (yi != lasty) { if (yi != lasty) {
...@@ -689,6 +699,7 @@ void ScalePlaneBilinearUp(int src_width, int src_height, ...@@ -689,6 +699,7 @@ void ScalePlaneBilinearUp(int src_width, int src_height,
y += dy; y += dy;
} }
free_aligned_buffer_64(row); free_aligned_buffer_64(row);
}
} }
// Scale Plane to/from any dimensions, without interpolation. // Scale Plane to/from any dimensions, without interpolation.
...@@ -700,6 +711,9 @@ static void ScalePlaneSimple(int src_width, int src_height, ...@@ -700,6 +711,9 @@ static void ScalePlaneSimple(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_ptr, uint8* dst_ptr) { const uint8* src_ptr, uint8* dst_ptr) {
int i;
void (*ScaleCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) = ScaleCols_C;
// Initial source x/y coordinate and step values as 16.16 fixed point. // Initial source x/y coordinate and step values as 16.16 fixed point.
int x = 0; int x = 0;
int y = 0; int y = 0;
...@@ -709,8 +723,6 @@ static void ScalePlaneSimple(int src_width, int src_height, ...@@ -709,8 +723,6 @@ static void ScalePlaneSimple(int src_width, int src_height,
&x, &y, &dx, &dy); &x, &y, &dx, &dy);
src_width = Abs(src_width); src_width = Abs(src_width);
void (*ScaleCols)(uint8* dst_ptr, const uint8* src_ptr,
int dst_width, int x, int dx) = ScaleCols_C;
if (src_width * 2 == dst_width && x < 0x8000) { if (src_width * 2 == dst_width && x < 0x8000) {
ScaleCols = ScaleColsUp2_C; ScaleCols = ScaleColsUp2_C;
#if defined(HAS_SCALECOLS_SSE2) #if defined(HAS_SCALECOLS_SSE2)
...@@ -722,7 +734,6 @@ static void ScalePlaneSimple(int src_width, int src_height, ...@@ -722,7 +734,6 @@ static void ScalePlaneSimple(int src_width, int src_height,
#endif #endif
} }
int i;
for (i = 0; i < dst_height; ++i) { for (i = 0; i < dst_height; ++i) {
ScaleCols(dst_ptr, src_ptr + (y >> 16) * src_stride, ScaleCols(dst_ptr, src_ptr + (y >> 16) * src_stride,
dst_width, x, dx); dst_width, x, dx);
...@@ -831,14 +842,14 @@ int I420Scale(const uint8* src_y, int src_stride_y, ...@@ -831,14 +842,14 @@ int I420Scale(const uint8* src_y, int src_stride_y,
uint8* dst_v, int dst_stride_v, uint8* dst_v, int dst_stride_v,
int dst_width, int dst_height, int dst_width, int dst_height,
enum FilterMode filtering) { enum FilterMode filtering) {
if (!src_y || !src_u || !src_v || src_width == 0 || src_height == 0 ||
!dst_y || !dst_u || !dst_v || dst_width <= 0 || dst_height <= 0) {
return -1;
}
int src_halfwidth = SUBSAMPLE(src_width, 1, 1); int src_halfwidth = SUBSAMPLE(src_width, 1, 1);
int src_halfheight = SUBSAMPLE(src_height, 1, 1); int src_halfheight = SUBSAMPLE(src_height, 1, 1);
int dst_halfwidth = SUBSAMPLE(dst_width, 1, 1); int dst_halfwidth = SUBSAMPLE(dst_width, 1, 1);
int dst_halfheight = SUBSAMPLE(dst_height, 1, 1); int dst_halfheight = SUBSAMPLE(dst_height, 1, 1);
if (!src_y || !src_u || !src_v || src_width == 0 || src_height == 0 ||
!dst_y || !dst_u || !dst_v || dst_width <= 0 || dst_height <= 0) {
return -1;
}
ScalePlane(src_y, src_stride_y, src_width, src_height, ScalePlane(src_y, src_stride_y, src_width, src_height,
dst_y, dst_stride_y, dst_width, dst_height, dst_y, dst_stride_y, dst_width, dst_height,
...@@ -877,26 +888,27 @@ LIBYUV_API ...@@ -877,26 +888,27 @@ LIBYUV_API
int ScaleOffset(const uint8* src, int src_width, int src_height, int ScaleOffset(const uint8* src, int src_width, int src_height,
uint8* dst, int dst_width, int dst_height, int dst_yoffset, uint8* dst, int dst_width, int dst_height, int dst_yoffset,
LIBYUV_BOOL interpolate) { LIBYUV_BOOL interpolate) {
if (!src || src_width <= 0 || src_height <= 0 || // Chroma requires offset to multiple of 2.
!dst || dst_width <= 0 || dst_height <= 0 || dst_yoffset < 0 || int dst_yoffset_even = dst_yoffset & ~1;
dst_yoffset >= dst_height) {
return -1;
}
dst_yoffset = dst_yoffset & ~1; // chroma requires offset to multiple of 2.
int src_halfwidth = SUBSAMPLE(src_width, 1, 1); int src_halfwidth = SUBSAMPLE(src_width, 1, 1);
int src_halfheight = SUBSAMPLE(src_height, 1, 1); int src_halfheight = SUBSAMPLE(src_height, 1, 1);
int dst_halfwidth = SUBSAMPLE(dst_width, 1, 1); int dst_halfwidth = SUBSAMPLE(dst_width, 1, 1);
int dst_halfheight = SUBSAMPLE(dst_height, 1, 1); int dst_halfheight = SUBSAMPLE(dst_height, 1, 1);
int aheight = dst_height - dst_yoffset * 2; // actual output height int aheight = dst_height - dst_yoffset_even * 2; // actual output height
const uint8* src_y = src; const uint8* src_y = src;
const uint8* src_u = src + src_width * src_height; const uint8* src_u = src + src_width * src_height;
const uint8* src_v = src + src_width * src_height + const uint8* src_v = src + src_width * src_height +
src_halfwidth * src_halfheight; src_halfwidth * src_halfheight;
uint8* dst_y = dst + dst_yoffset * dst_width; uint8* dst_y = dst + dst_yoffset_even * dst_width;
uint8* dst_u = dst + dst_width * dst_height + uint8* dst_u = dst + dst_width * dst_height +
(dst_yoffset >> 1) * dst_halfwidth; (dst_yoffset_even >> 1) * dst_halfwidth;
uint8* dst_v = dst + dst_width * dst_height + dst_halfwidth * dst_halfheight + uint8* dst_v = dst + dst_width * dst_height + dst_halfwidth * dst_halfheight +
(dst_yoffset >> 1) * dst_halfwidth; (dst_yoffset_even >> 1) * dst_halfwidth;
if (!src || src_width <= 0 || src_height <= 0 ||
!dst || dst_width <= 0 || dst_height <= 0 || dst_yoffset_even < 0 ||
dst_yoffset_even >= dst_height) {
return -1;
}
return I420Scale(src_y, src_width, return I420Scale(src_y, src_width,
src_u, src_halfwidth, src_u, src_halfwidth,
src_v, src_halfwidth, src_v, src_halfwidth,
......
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