/*
* Copyright (c) 2003, 2012, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* FUNCTION
* mlib_ImageZoom - image scaling with edge condition
*
* SYNOPSIS
* mlib_status mlib_ImageZoom(mlib_image *dst,
* const mlib_image *src,
* mlib_f32 zoomx,
* mlib_f32 zoomy,
* mlib_filter filter,
* mlib_edge edge)
*
* ARGUMENTS
* dst Pointer to destination image
* src Pointer to source image
* zoomx X zoom factor.
* zoomy Y zoom factor.
* filter Type of resampling filter.
* edge Type of edge condition.
*
* DESCRIPTION
* The center of the source image is mapped to the center of the
* destination image.
*
* The upper-left corner pixel of an image is located at (0.5, 0.5).
*
* The resampling filter can be one of the following:
* MLIB_NEAREST
* MLIB_BILINEAR
* MLIB_BICUBIC
* MLIB_BICUBIC2
*
* The edge condition can be one of the following:
* MLIB_EDGE_DST_NO_WRITE (default)
* MLIB_EDGE_DST_FILL_ZERO
* MLIB_EDGE_OP_NEAREST
* MLIB_EDGE_SRC_EXTEND
* MLIB_EDGE_SRC_PADDED
*/
#ifdef MACOSX
#include <machine/endian.h>
#endif
#include <mlib_image.h>
#include <mlib_ImageZoom.h>
#define MLIB_COPY_FUNC mlib_ImageCopy_na
/***************************************************************/
#ifdef i386 /* do not perform the coping by mlib_d64 data type for x86 */
typedef struct {
mlib_s32 int0, int1;
} two_int;
#define TYPE_64 two_int
#else /* i386 ( do not perform the coping by mlib_d64 data type for x86 ) */
#define TYPE_64 mlib_d64
#endif /* i386 ( do not perform the coping by mlib_d64 data type for x86 ) */
/***************************************************************/
typedef union {
TYPE_64 d64;
struct {
mlib_f32 f0, f1;
} f32s;
} d64_2_f32;
/***************************************************************/
#define CLAMP_U8(X) ((X >= 256) ? (255) : ((X) &~ ((X) >> 31)))
/***************************************************************/
#ifdef _LITTLE_ENDIAN
static const mlib_u32 mlib_bit_mask4[16] = {
0x00000000u, 0xFF000000u, 0x00FF0000u, 0xFFFF0000u,
0x0000FF00u, 0xFF00FF00u, 0x00FFFF00u, 0xFFFFFF00u,
0x000000FFu, 0xFF0000FFu, 0x00FF00FFu, 0xFFFF00FFu,
0x0000FFFFu, 0xFF00FFFFu, 0x00FFFFFFu, 0xFFFFFFFFu
};
#else /* _LITTLE_ENDIAN */
static const mlib_u32 mlib_bit_mask4[16] = {
0x00000000u, 0x000000FFu, 0x0000FF00u, 0x0000FFFFu,
0x00FF0000u, 0x00FF00FFu, 0x00FFFF00u, 0x00FFFFFFu,
0xFF000000u, 0xFF0000FFu, 0xFF00FF00u, 0xFF00FFFFu,
0xFFFF0000u, 0xFFFF00FFu, 0xFFFFFF00u, 0xFFFFFFFFu
};
#endif /* _LITTLE_ENDIAN */
/***************************************************************/
#define VARIABLE(FORMAT) \
mlib_s32 j, \
dx = GetElemStruct(DX), \
dy = GetElemStruct(DY), \
x = GetElemSubStruct(current, srcX), \
y = GetElemSubStruct(current, srcY), \
src_stride = GetElemStruct(src_stride), \
dst_stride = GetElemStruct(dst_stride), \
width = GetElemSubStruct(current, width), \
height = GetElemSubStruct(current, height); \
FORMAT *sp = (FORMAT *)GetElemSubStruct(current, sp), \
*dp = (FORMAT *)GetElemSubStruct(current, dp)
/***************************************************************/
#define BUFF_SIZE 256
#define BYTE_POS_MASK ((1 << (MLIB_SHIFT + 3)) - 1)
/***************************************************************/
mlib_status mlib_ImageZoom_BIT_1_Nearest(mlib_work_image *param,
mlib_s32 s_bitoff,
mlib_s32 d_bitoff)
{
VARIABLE(mlib_u8);
mlib_s32 i;
mlib_s32 buff_loc[BUFF_SIZE], *buff = buff_loc;
mlib_s32 srcX = GetElemSubStruct(current, srcX);
mlib_s32 dstX = GetElemSubStruct(current, dstX);
mlib_u8 *sl = sp - (srcX >> MLIB_SHIFT), *dl = dp - dstX, *dt;
mlib_s32 bit, res, k, y_step = -1;
mlib_s32 num0, n_al, mask0, mask1;
srcX += (s_bitoff << MLIB_SHIFT);
dstX += d_bitoff;
num0 = 8 - (dstX & 7);
if (num0 > width)
num0 = width;
num0 &= 7;
mask0 = ((0xFF00 >> num0) & 0xFF) >> (dstX & 7);
n_al = width - num0;
mask1 = ((0xFF00 >> (n_al & 7)) & 0xFF);
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
if (n_al > BUFF_SIZE) {
buff = mlib_malloc(sizeof(mlib_s32) * n_al);
if (buff == NULL)
return MLIB_FAILURE;
}
/* save shifts for bit extracting */
x = srcX + num0 * dx;
#if 0
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (n_al >> 3); i++) {
buff[8 * i] = (((x >> MLIB_SHIFT)) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 1] = (((x >> MLIB_SHIFT) - 1) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 2] = (((x >> MLIB_SHIFT) - 2) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 3] = (((x >> MLIB_SHIFT) - 3) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 4] = (((x >> MLIB_SHIFT) - 4) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 5] = (((x >> MLIB_SHIFT) - 5) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 6] = (((x >> MLIB_SHIFT) - 6) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
buff[8 * i + 7] = (((x >> MLIB_SHIFT) - 7) & 7) | (x & ~BYTE_POS_MASK);
x += dx;
}
x_last = x;
#else /* 0 */
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (n_al >> 3); i++) {
buff[8 * i] = (((x >> MLIB_SHIFT)) & 7);
x += dx;
buff[8 * i + 1] = (((x >> MLIB_SHIFT) - 1) & 7);
x += dx;
buff[8 * i + 2] = (((x >> MLIB_SHIFT) - 2) & 7);
x += dx;
buff[8 * i + 3] = (((x >> MLIB_SHIFT) - 3) & 7);
x += dx;
buff[8 * i + 4] = (((x >> MLIB_SHIFT) - 4) & 7);
x += dx;
buff[8 * i + 5] = (((x >> MLIB_SHIFT) - 5) & 7);
x += dx;
buff[8 * i + 6] = (((x >> MLIB_SHIFT) - 6) & 7);
x += dx;
buff[8 * i + 7] = (((x >> MLIB_SHIFT) - 7) & 7);
x += dx;
}
#endif /* 0 */
for (j = 0; j < height; j++) {
if (!y_step) {
dp = dl + (dstX >> 3);
dt = dp - dst_stride;
if (num0) {
dp[0] = (dp[0] & ~mask0) | (*dt++ & mask0);
dp++;
}
#if 0
MLIB_COPY_FUNC(dt, dp, n_al >> 3);
if (n_al & 7) {
dp[n_al >> 3] = (dp[n_al >> 3] & ~mask1) | (dt[n_al >> 3] & mask1);
}
#else /* 0 */
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (n_al >> 3); i++) {
dp[i] = dt[i];
}
if (n_al & 7) {
dp[i] = (dp[i] & ~mask1) | (dt[i] & mask1);
}
#endif /* 0 */
}
else {
x = srcX;
dp = dl + (dstX >> 3);
if (num0) {
mlib_s32 res = dp[0] & ~mask0;
for (k = dstX; k < (dstX + num0); k++) {
bit = 7 - (k & 7);
res |=
(((sl[x >> (MLIB_SHIFT + 3)] >> (7 - (x >> MLIB_SHIFT) & 7)) & 1) << bit);
x += dx;
}
*dp++ = res;
}
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (n_al >> 3); i++) {
#if 0
res = ((sl[buff[8 * i] >> (MLIB_SHIFT + 3)] << buff[8 * i]) & 0x8080);
res |= ((sl[buff[8 * i + 1] >> (MLIB_SHIFT + 3)] << buff[8 * i + 1]) & 0x4040);
res |= ((sl[buff[8 * i + 2] >> (MLIB_SHIFT + 3)] << buff[8 * i + 2]) & 0x2020);
res |= ((sl[buff[8 * i + 3] >> (MLIB_SHIFT + 3)] << buff[8 * i + 3]) & 0x1010);
res |= ((sl[buff[8 * i + 4] >> (MLIB_SHIFT + 3)] << buff[8 * i + 4]) & 0x0808);
res |= ((sl[buff[8 * i + 5] >> (MLIB_SHIFT + 3)] << buff[8 * i + 5]) & 0x0404);
res |= ((sl[buff[8 * i + 6] >> (MLIB_SHIFT + 3)] << buff[8 * i + 6]) & 0x0202);
res |= ((sl[buff[8 * i + 7] >> (MLIB_SHIFT + 3)] << buff[8 * i + 7]) & 0x0101);
#else /* 0 */
res = ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i]) & 0x8080);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 1]) & 0x4040);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 2]) & 0x2020);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 3]) & 0x1010);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 4]) & 0x0808);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 5]) & 0x0404);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 6]) & 0x0202);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 7]) & 0x0101);
x += dx;
#endif /* 0 */
dp[i] = res | (res >> 8);
}
if (mask1) {
mlib_s32 res = dp[i] & ~mask1;
for (k = 0; k < (n_al & 7); k++) {
bit = 7 - (k & 7);
res |=
(((sl[x >> (MLIB_SHIFT + 3)] >> (7 - (x >> MLIB_SHIFT) & 7)) & 1) << bit);
x += dx;
}
dp[i] = res;
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dl = (void *)((mlib_u8 *) dl + dst_stride);
sl = (void *)((mlib_u8 *) sl + y_step * src_stride);
}
if (buff != buff_loc)
mlib_free(buff);
return MLIB_SUCCESS;
}
/***************************************************************/
#ifdef _NO_LONGLONG
typedef struct {
#ifdef _LITTLE_ENDIAN
mlib_u32 uint1, uint0;
#else /* _LITTLE_ENDIAN */
mlib_u32 uint0, uint1;
#endif /* _LITTLE_ENDIAN */
} two_uint;
/***************************************************************/
#define DTYPE two_uint
#define MASK(dst) (dst).uint0 = (dst).uint1 = -1
/***************************************************************/
#define RSHIFT(dst, src, rshift) { \
DTYPE tmp = (src); \
if ((rshift) >= 32) { \
tmp.uint1 = tmp.uint0 >> ((rshift) - 32); \
tmp.uint0 = 0; \
} \
else { \
tmp.uint1 = (tmp.uint1 >> (rshift)) | (tmp.uint0 << (32 - (rshift))); \
tmp.uint0 = tmp.uint0 >> (rshift); \
} \
(dst) = tmp; \
}
/***************************************************************/
#define LSHIFT(dst, src, lshift) { \
DTYPE tmp = (src); \
if ((lshift) >= 32) { \
tmp.uint0 = tmp.uint1 << ((lshift) - 32); \
tmp.uint1 = 0; \
} \
else { \
tmp.uint0 = (tmp.uint0 << (lshift)) | (tmp.uint1 >> (32 - (lshift))); \
tmp.uint1 = tmp.uint1 << (lshift); \
} \
(dst) = tmp; \
}
/***************************************************************/
#define LOGIC(dst, src1, src2, OPERATION) { \
DTYPE tmp; \
((tmp).uint0 = (src1).uint0 OPERATION (src2).uint0); \
((tmp).uint1 = (src1).uint1 OPERATION (src2).uint1); \
(dst) = tmp; \
}
#else /* _NO_LONGLONG */
/***************************************************************/
#define DTYPE mlib_u64
#define MASK(dst) (dst) = ((mlib_u64)((mlib_s64) -1))
#define RSHIFT(dst, src, rshift) (dst) = ((src) >> (rshift))
#define LSHIFT(dst, src, lshift) (dst) = ((src) << (lshift))
#define LOGIC(dst, src1, src2, OPERATION) (dst) = ((src1) OPERATION (src2))
#endif /* _NO_LONGLONG */
/***************************************************************/
mlib_status mlib_ImageZoom_BitToGray_1_Nearest(mlib_work_image *param,
mlib_s32 s_bitoff,
const mlib_s32 *ghigh,
const mlib_s32 *glow)
{
VARIABLE(mlib_u8);
mlib_s32 i;
DTYPE gray_mask[256], dd, dd_old, *da, dtmp, dtmp1;
mlib_u32 *pgray = (mlib_u32 *) gray_mask;
mlib_u8 buff_loc[2 * BUFF_SIZE], *buff = buff_loc;
mlib_u8 *sl, *dl, gray_val[2];
mlib_s32 srcX = GetElemSubStruct(current, srcX);
mlib_u32 gray_val0, gray_val1;
mlib_s32 width8, res, y_step = -1;
mlib_s32 k;
sl = sp - (srcX >> MLIB_SHIFT);
dl = dp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
srcX += (s_bitoff << MLIB_SHIFT);
width8 = width / 8;
if (width8 > 2 * BUFF_SIZE) {
buff = mlib_malloc(width8 * sizeof(mlib_u8));
if (buff == NULL)
return MLIB_FAILURE;
}
/* save shifts for bit extracting */
x = srcX;
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width8; i++) {
buff[8 * i] = (((x >> MLIB_SHIFT)) & 7);
x += dx;
buff[8 * i + 1] = (((x >> MLIB_SHIFT) - 1) & 7);
x += dx;
buff[8 * i + 2] = (((x >> MLIB_SHIFT) - 2) & 7);
x += dx;
buff[8 * i + 3] = (((x >> MLIB_SHIFT) - 3) & 7);
x += dx;
buff[8 * i + 4] = (((x >> MLIB_SHIFT) - 4) & 7);
x += dx;
buff[8 * i + 5] = (((x >> MLIB_SHIFT) - 5) & 7);
x += dx;
buff[8 * i + 6] = (((x >> MLIB_SHIFT) - 6) & 7);
x += dx;
buff[8 * i + 7] = (((x >> MLIB_SHIFT) - 7) & 7);
x += dx;
}
/* calculate lookup table */
gray_val0 = CLAMP_U8(glow[0]);
gray_val1 = CLAMP_U8(ghigh[0]);
gray_val[0] = gray_val0;
gray_val[1] = gray_val1;
gray_val0 |= (gray_val0 << 8);
gray_val0 |= (gray_val0 << 16);
gray_val1 |= (gray_val1 << 8);
gray_val1 |= (gray_val1 << 16);
for (i = 0; i < 16; i++) {
mlib_u32 v, mask = mlib_bit_mask4[i];
v = (gray_val0 & ~mask) | (gray_val1 & mask);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (j = 0; j < 16; j++) {
pgray[2 * (16 * i + j)] = v;
}
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (j = 0; j < 16; j++) {
pgray[2 * (i + 16 * j) + 1] = v;
}
}
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dl - dst_stride, dl, width);
}
else {
mlib_s32 off = (mlib_addr) dl & 7;
da = (DTYPE *) (dl - off);
x = srcX;
if (off) { /* not aligned */
DTYPE mask;
MASK(mask);
off *= 8;
#ifdef _LITTLE_ENDIAN
LSHIFT(dd_old, da[0], 64 - off);
#else /* _LITTLE_ENDIAN */
RSHIFT(dd_old, da[0], 64 - off);
#endif /* _LITTLE_ENDIAN */
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (width / 8); i++) {
res = ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i]) & 0x8080);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 1]) & 0x4040);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 2]) & 0x2020);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 3]) & 0x1010);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 4]) & 0x0808);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 5]) & 0x0404);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 6]) & 0x0202);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 7]) & 0x0101);
x += dx;
res = (res & 0xff) | (res >> 8);
dd = gray_mask[res];
#ifdef _LITTLE_ENDIAN
/* *da++ = (dd_old >> (64 - off)) | (dd << off);*/
RSHIFT(dd_old, dd_old, 64 - off);
LSHIFT(dtmp, dd, off);
#else /* _LITTLE_ENDIAN */
/* *da++ = (dd_old << (64 - off)) | (dd >> off);*/
LSHIFT(dd_old, dd_old, 64 - off);
RSHIFT(dtmp, dd, off);
#endif /* _LITTLE_ENDIAN */
LOGIC(*da++, dd_old, dtmp, |);
dd_old = dd;
}
#ifdef _LITTLE_ENDIAN
/* da[0] = (dd_old >> (64 - off)) | (da[0] & ((mlib_u64)((mlib_s64) -1) << off));*/
LSHIFT(dtmp, mask, off);
LOGIC(dtmp, da[0], dtmp, &);
RSHIFT(dtmp1, dd_old, 64 - off);
#else /* _LITTLE_ENDIAN */
/* da[0] = (dd_old << (64 - off)) | (da[0] & ((mlib_u64)((mlib_s64) -1) >> off));*/
RSHIFT(dtmp, mask, off);
LOGIC(dtmp, da[0], dtmp, &);
LSHIFT(dtmp1, dd_old, 64 - off);
#endif /* _LITTLE_ENDIAN */
LOGIC(da[0], dtmp, dtmp1, |);
}
else { /* aligned */
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < (width / 8); i++) {
res = ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i]) & 0x8080);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 1]) & 0x4040);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 2]) & 0x2020);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 3]) & 0x1010);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 4]) & 0x0808);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 5]) & 0x0404);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 6]) & 0x0202);
x += dx;
res |= ((sl[x >> (MLIB_SHIFT + 3)] << buff[8 * i + 7]) & 0x0101);
x += dx;
res = (res & 0xff) | (res >> 8);
*da++ = gray_mask[res];
}
}
if (width & 7) {
dp = dl + (width & ~7);
for (k = 0; k < (width & 7); k++) {
dp[k] =
gray_val[(sl[x >> (MLIB_SHIFT + 3)] >> (7 - (x >> MLIB_SHIFT) & 7)) & 1];
x += dx;
}
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dl += dst_stride;
sl += y_step * src_stride;
}
if (buff != buff_loc)
mlib_free(buff);
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_U8_2_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_u8);
mlib_s32 i;
mlib_u8 *tdp, *tsp, tmp0, tmp1;
mlib_s32 cx, y_step = -1;
tsp = sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, dp, 2 * width);
}
else {
tdp = dp;
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
cx = (x >> (MLIB_SHIFT - 1)) & ~1;
tmp0 = tsp[cx];
tmp1 = tsp[cx + 1];
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++, tdp += 2) {
tdp[0] = tmp0;
tdp[1] = tmp1;
x += dx;
cx = (x >> (MLIB_SHIFT - 1)) & ~1;
tmp0 = tsp[cx];
tmp1 = tsp[cx + 1];
}
tdp[0] = tmp0;
tdp[1] = tmp1;
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_U8_4_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_u8);
mlib_s32 i;
mlib_u8 *tdp, *tsp, tmp0, tmp1, tmp2, tmp3;
mlib_u16 utmp0, utmp1;
mlib_s32 cx, y_step = -1;
tsp = sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, dp, 4 * width);
}
else {
tdp = dp;
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if (((mlib_addr) tdp | (mlib_addr) tsp) & 1) { /* sp & dp pointers not aligned */
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
tmp0 = tsp[cx];
tmp1 = tsp[cx + 1];
tmp2 = tsp[cx + 2];
tmp3 = tsp[cx + 3];
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
tdp[0] = tmp0;
tdp[1] = tmp1;
tdp[2] = tmp2;
tdp[3] = tmp3;
x += dx;
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
tmp0 = tsp[cx];
tmp1 = tsp[cx + 1];
tmp2 = tsp[cx + 2];
tmp3 = tsp[cx + 3];
tdp += 4;
}
tdp[0] = tmp0;
tdp[1] = tmp1;
tdp[2] = tmp2;
tdp[3] = tmp3;
}
else {
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
utmp0 = *(mlib_u16 *) (tsp + cx);
utmp1 = *(mlib_u16 *) (tsp + cx + 2);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
*(mlib_u16 *) tdp = utmp0;
*(mlib_u16 *) (tdp + 2) = utmp1;
x += dx;
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
utmp0 = *(mlib_u16 *) (tsp + cx);
utmp1 = *(mlib_u16 *) (tsp + cx + 2);
tdp += 4;
}
*(mlib_u16 *) tdp = utmp0;
*(mlib_u16 *) (tdp + 2) = utmp1;
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S16_2_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_u16);
mlib_s32 i;
mlib_u8 *tsp, *tdp;
mlib_u16 tmp0, tmp1;
mlib_s32 cx, y_step = -1;
mlib_u32 utmp;
tsp = (mlib_u8 *) sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, (void *)dp, 4 * width);
}
else {
tdp = (mlib_u8 *) dp;
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if (((mlib_addr) tdp | (mlib_addr) tsp) & 3) { /* sp & dp pointers not aligned */
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
tmp0 = *(mlib_u16 *) (tsp + cx);
tmp1 = *(mlib_u16 *) (tsp + cx + 2);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++, tdp += 4) {
*(mlib_u16 *) tdp = tmp0;
*(mlib_u16 *) (tdp + 2) = tmp1;
x += dx;
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
tmp0 = *(mlib_u16 *) (tsp + cx);
tmp1 = *(mlib_u16 *) (tsp + cx + 2);
}
*(mlib_u16 *) tdp = tmp0;
*(mlib_u16 *) (tdp + 2) = tmp1;
}
else {
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
utmp = *(mlib_u32 *) (tsp + cx);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++, tdp += 4) {
*(mlib_u32 *) tdp = utmp;
x += dx;
cx = (x >> (MLIB_SHIFT - 2)) & ~3;
utmp = *(mlib_u32 *) (tsp + cx);
}
*(mlib_u32 *) tdp = utmp;
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S16_4_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_u16);
mlib_s32 i;
mlib_u8 *tsp, *tdp;
mlib_s32 cx, y_step = -1;
mlib_u16 tmp0, tmp1, tmp2, tmp3;
TYPE_64 dtmp;
mlib_f32 ftmp0, ftmp1;
tsp = (mlib_u8 *) sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, (void *)dp, 8 * width);
}
else {
tdp = (mlib_u8 *) dp;
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if (((mlib_addr) tdp | (mlib_addr) tsp) & 7) {
if (((mlib_addr) tdp | (mlib_addr) tsp) & 3) {
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
tmp0 = *(mlib_u16 *) (tsp + cx);
tmp1 = *(mlib_u16 *) (tsp + cx + 2);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
tmp2 = *(mlib_u16 *) (tsp + cx + 4);
tmp3 = *(mlib_u16 *) (tsp + cx + 6);
*(mlib_u16 *) tdp = tmp0;
*(mlib_u16 *) (tdp + 2) = tmp1;
*(mlib_u16 *) (tdp + 4) = tmp2;
*(mlib_u16 *) (tdp + 6) = tmp3;
x += dx;
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
tmp0 = *(mlib_u16 *) (tsp + cx);
tmp1 = *(mlib_u16 *) (tsp + cx + 2);
tdp += 8;
}
tmp2 = *(mlib_u16 *) (tsp + cx + 4);
tmp3 = *(mlib_u16 *) (tsp + cx + 6);
*(mlib_u16 *) tdp = tmp0;
*(mlib_u16 *) (tdp + 2) = tmp1;
*(mlib_u16 *) (tdp + 4) = tmp2;
*(mlib_u16 *) (tdp + 6) = tmp3;
}
else { /* align to word */
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
ftmp0 = *(mlib_f32 *) (tsp + cx);
ftmp1 = *(mlib_f32 *) (tsp + cx + 4);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
*(mlib_f32 *) tdp = ftmp0;
*(mlib_f32 *) (tdp + 4) = ftmp1;
x += dx;
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
ftmp0 = *(mlib_f32 *) (tsp + cx);
ftmp1 = *(mlib_f32 *) (tsp + cx + 4);
tdp += 8;
}
*(mlib_f32 *) tdp = ftmp0;
*(mlib_f32 *) (tdp + 4) = ftmp1;
}
}
else { /* align to mlib_d64 word */
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
dtmp = *(TYPE_64 *) (tsp + cx);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
*(TYPE_64 *) tdp = dtmp;
x += dx;
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
dtmp = *(TYPE_64 *) (tsp + cx);
tdp += 8;
}
*(TYPE_64 *) tdp = dtmp;
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S32_1_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_s32);
mlib_s32 *dl = dp, *tsp;
mlib_s32 y_step = -1;
tsp = sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dl - dst_stride, (void *)dl, 4 * width);
}
else {
mlib_s32 *dp = dl, *dend = dl + width;
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if ((mlib_addr) dp & 7) {
*dp++ = tsp[x >> MLIB_SHIFT];
x += dx;
}
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (; dp <= dend - 2; dp += 2) {
d64_2_f32 dd;
dd.f32s.f0 = *(mlib_f32 *) ((mlib_u8 *) tsp + ((x >> (MLIB_SHIFT - 2)) & ~3));
x += dx;
dd.f32s.f1 = *(mlib_f32 *) ((mlib_u8 *) tsp + ((x >> (MLIB_SHIFT - 2)) & ~3));
x += dx;
*(TYPE_64 *) dp = dd.d64;
}
if (dp < dend) {
*dp++ = tsp[x >> MLIB_SHIFT];
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dl = (void *)((mlib_u8 *) dl + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S32_2_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_s32);
mlib_s32 i;
mlib_u8 *tsp;
mlib_s32 cx, y_step = -1, tmp0, tmp1, tmp2, tmp3, x_max;
TYPE_64 dtmp0;
tsp = (mlib_u8 *) sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
x_max = (param->sline_size) << MLIB_SHIFT;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, (void *)dp, 8 * width);
}
else {
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if (!(((mlib_addr) dp | (mlib_addr) tsp) & 7)) {
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width; i++) {
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
x += dx;
dtmp0 = *(TYPE_64 *) (tsp + cx);
((TYPE_64 *) dp)[i] = dtmp0;
}
}
else {
cx = (x >> (MLIB_SHIFT - 3)) & ~7;
x += dx;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
cx = ((x >> (MLIB_SHIFT - 3)) & ~7) & ((x - x_max) >> 31);
x += dx;
tmp2 = *(mlib_s32 *) (tsp + cx);
tmp3 = *(mlib_s32 *) (tsp + cx + 4);
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i <= width - 2; i += 2) {
dp[2 * i] = tmp0;
dp[2 * i + 1] = tmp1;
dp[2 * i + 2] = tmp2;
dp[2 * i + 3] = tmp3;
cx = ((x >> (MLIB_SHIFT - 3)) & ~7) & ((x - x_max) >> 31);
x += dx;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
cx = ((x >> (MLIB_SHIFT - 3)) & ~7) & ((x - x_max) >> 31);
x += dx;
tmp2 = *(mlib_s32 *) (tsp + cx);
tmp3 = *(mlib_s32 *) (tsp + cx + 4);
}
if (width & 1) {
dp[2 * i] = tmp0;
dp[2 * i + 1] = tmp1;
}
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S32_3_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_s32);
mlib_s32 i;
mlib_u8 *tsp;
mlib_s32 cx, y_step = -1, tmp0, tmp1, tmp2;
tsp = (mlib_u8 *) sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, (void *)dp, 12 * width);
}
else {
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
cx = (x >> MLIB_SHIFT) * 12;
x += dx;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
tmp2 = *(mlib_s32 *) (tsp + cx + 8);
cx = (x >> MLIB_SHIFT) * 12;
x += dx;
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
dp[3 * i + 0] = tmp0;
dp[3 * i + 1] = tmp1;
dp[3 * i + 2] = tmp2;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
tmp2 = *(mlib_s32 *) (tsp + cx + 8);
cx = (x >> MLIB_SHIFT) * 12;
x += dx;
}
dp[3 * i + 0] = tmp0;
dp[3 * i + 1] = tmp1;
dp[3 * i + 2] = tmp2;
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_S32_4_Nearest(mlib_work_image *param)
{
VARIABLE(mlib_s32);
mlib_s32 i;
mlib_u8 *tsp;
mlib_s32 cx, y_step = -1, tmp0, tmp1, tmp2, tmp3;
TYPE_64 dtmp0, dtmp1;
tsp = (mlib_u8 *) sp;
y = GetElemSubStruct(current, srcY) & MLIB_MASK;
for (j = 0; j < height; j++) {
if (!y_step) {
MLIB_COPY_FUNC((mlib_u8 *) dp - dst_stride, (void *)dp, 16 * width);
}
else {
x = GetElemSubStruct(current, srcX) & MLIB_MASK;
if (((mlib_addr) dp | (mlib_addr) tsp) & 7) {
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
tmp2 = *(mlib_s32 *) (tsp + cx + 8);
tmp3 = *(mlib_s32 *) (tsp + cx + 12);
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
dp[4 * i + 0] = tmp0;
dp[4 * i + 1] = tmp1;
dp[4 * i + 2] = tmp2;
dp[4 * i + 3] = tmp3;
tmp0 = *(mlib_s32 *) (tsp + cx);
tmp1 = *(mlib_s32 *) (tsp + cx + 4);
tmp2 = *(mlib_s32 *) (tsp + cx + 8);
tmp3 = *(mlib_s32 *) (tsp + cx + 12);
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
}
dp[4 * i + 0] = tmp0;
dp[4 * i + 1] = tmp1;
dp[4 * i + 2] = tmp2;
dp[4 * i + 3] = tmp3;
}
else {
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
dtmp0 = *(TYPE_64 *) (tsp + cx);
dtmp1 = *(TYPE_64 *) (tsp + cx + 8);
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
#ifdef __SUNPRO_C
#pragma pipeloop(0)
#endif /* __SUNPRO_C */
for (i = 0; i < width - 1; i++) {
*(TYPE_64 *) (dp + 4 * i) = dtmp0;
*(TYPE_64 *) (dp + 4 * i + 2) = dtmp1;
dtmp0 = *(TYPE_64 *) (tsp + cx);
dtmp1 = *(TYPE_64 *) (tsp + cx + 8);
cx = (x >> (MLIB_SHIFT - 4)) & ~15;
x += dx;
}
*(TYPE_64 *) (dp + 4 * i) = dtmp0;
*(TYPE_64 *) (dp + 4 * i + 2) = dtmp1;
}
}
y_step = ((y + dy) - (y & ~MLIB_MASK)) >> MLIB_SHIFT;
y += dy;
dp = (void *)((mlib_u8 *) dp + dst_stride);
tsp = (void *)((mlib_u8 *) tsp + y_step * src_stride);
}
return MLIB_SUCCESS;
}
/***************************************************************/
mlib_status mlib_ImageZoom_D64_1_Nearest(mlib_work_image *param)
{
/**代码未完, 请加载全部代码(NowJava.com).**/