📄 yuvalpha.c
字号:
/////////////////////////////////////////////////////////////
//
// I420andYUVAtoUYVY
//
// This function alpha-blends two I420 buffers into a third
// UYVY buffer using the alpha info tacked to the
// end of the second I420 buffer
//
/////////////////////////////////////////////////////////////
int I420andYUVAtoUYVY (
unsigned char *src1_ptr, int src1_pels, int src1_lines, int src1_pitch,
int src1_startx, int src1_starty,
unsigned char *src2_ptr, int src2_pels, int src2_lines, int src2_pitch,
int src2_startx, int src2_starty,
unsigned char *dest_ptr, int dest_pels, int dest_lines, int dest_pitch,
int dest_startx, int dest_starty,
int width, int height)
{
int i, j;
unsigned char *sy11, *sy12, *su1, *sv1;
unsigned char *sy21, *sy22, *su2, *sv2, *sa21, *sa22;
unsigned int *dst1, *dst2;
// Initialize color plane pointers
sy11 = src1_ptr;
su1 = src1_ptr + src1_lines * src1_pitch;
sv1 = su1 + src1_lines * src1_pitch / 4;
sy21 = src2_ptr;
su2 = src2_ptr + src2_lines * src2_pitch;
sv2 = su2 + src2_lines * src2_pitch / 4;
sa21 = sv2 + src2_lines * src2_pitch / 4;
dst1 = (unsigned int *)(dest_ptr);
// Move color planes to start point
sy11 += src1_starty * src1_pitch + src1_startx;
sy12 = sy11 + src1_pitch;
su1 += src1_starty * src1_pitch / 4 + src1_startx / 2;
sv1 += src1_starty * src1_pitch / 4 + src1_startx / 2;
sy21 += src2_starty * src2_pitch + src2_startx;
sy22 = sy21 + src2_pitch;
su2 += src2_starty * src2_pitch / 4 + src2_startx / 2;
sv2 += src2_starty * src2_pitch / 4 + src2_startx / 2;
sa21 += src2_starty * src2_pitch + src2_startx;
sa22 = sa21 + src2_pitch;
dst1 += dest_starty * dest_pitch / 4 + dest_startx / 2;
dst2 = dst1 + dest_pitch / 4;
for (i = 0; i < height / 2; i++)
{
for (j = 0; j < width / 2; j++)
{
int x1, x2, a1, a2, d;
int u1, u2, v1, v2;
// First line
// Y0
x1 = *(sy11 + 0);
x2 = *(sy21 + 0);
x1 -= x2;
a1 = *(sa21 + 0);
x1 *= a1;
x1 >>= 8;
x1 += x2;
d = (x1 << UYVY_Y0_SHIFT);
// Y1
x1 = *(sy11 + 1);
x2 = *(sy21 + 1);
x1 -= x2;
a2 = *(sa21 + 1);
x1 *= a2;
x1 >>= 8;
x1 += x2;
d |= (x1 << UYVY_Y1_SHIFT);
sy11 += 2;
sy21 += 2;
sa21 += 2;
// Average Alphas
a1 += a2;
a1 >>= 1;
// U
x1 = u1 = *su1;
x2 = u2 = *su2;
x1 -= x2;
x1 *= a1;
x1 >>= 8;
x1 += x2;
d |= (x1 << UYVY_U_SHIFT);
// V
x1 = v1 = *sv1;
x2 = v2 = *sv2;
x1 -= x2;
x1 *= a1;
x1 >>= 8;
x1 += x2;
d |= (x1 << UYVY_V_SHIFT);
su1++;
su2++;
sv1++;
sv2++;
// Store YUY2 pel
*dst1++ = d;
// Second line
// Y0
x1 = *(sy12 + 0);
x2 = *(sy22 + 0);
x1 -= x2;
a1 = *(sa22 + 0);
x1 *= a1;
x1 >>= 8;
x1 += x2;
d = (x1 << UYVY_Y0_SHIFT);
// Y1
x1 = *(sy12 + 1);
x2 = *(sy22 + 1);
x1 -= x2;
a2 = *(sa22 + 1);
x1 *= a2;
x1 >>= 8;
x1 += x2;
d |= (x1 << UYVY_Y1_SHIFT);
sy12 += 2;
sy22 += 2;
sa22 += 2;
// Average Alphas
a1 += a2;
a1 >>= 1;
// U
u1 -= u2;
u1 *= a1;
u1 >>= 8;
u1 += u2;
d |= (u1 << UYVY_U_SHIFT);
// V
v1 -= v2;
v1 *= a1;
v1 >>= 8;
v1 += v2;
d |= (v1 << UYVY_V_SHIFT);
// Store YUY2 pel
*dst2++ = d;
}
// move down two lines
sy11 += 2 * src1_pitch - width;
sy12 += 2 * src1_pitch - width;
su1 += (src1_pitch - width) / 2;
sv1 += (src1_pitch - width) / 2;
sy21 += 2 * src2_pitch - width;
sy22 += 2 * src2_pitch - width;
su2 += (src2_pitch - width) / 2;
sv2 += (src2_pitch - width) / 2;
sa21 += 2 * src2_pitch - width;
sa22 += 2 * src2_pitch - width;
dst1 += (dest_pitch / 2) - (width / 2);
dst2 += (dest_pitch / 2) - (width / 2);
}
return 0;
}
/////////////////////////////////////////////////////////////
//
// I420andYUVAtoI420
//
// This function alpha-blends two I420 buffers into a third
// I420 buffer using the alpha info tacked to the
// end of the second I420 buffer
//
/////////////////////////////////////////////////////////////
int I420andYUVAtoI420 (
unsigned char *src1_ptr, int src1_pels, int src1_lines, int src1_pitch,
int src1_startx, int src1_starty,
unsigned char *src2_ptr, int src2_pels, int src2_lines, int src2_pitch,
int src2_startx, int src2_starty,
unsigned char *dest_ptr, int dest_pels, int dest_lines, int dest_pitch,
int dest_startx, int dest_starty,
int width, int height)
{
return I420andYUVAtoI420orYV12 (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height, CID_I420);
}
/////////////////////////////////////////////////////////////
//
// I420andYUVAtoYV12
//
// This function alpha-blends two I420 buffers into a third
// YV12 buffer using the alpha info tacked to the
// end of the second I420 buffer
//
/////////////////////////////////////////////////////////////
int I420andYUVAtoYV12 (
unsigned char *src1_ptr, int src1_pels, int src1_lines, int src1_pitch,
int src1_startx, int src1_starty,
unsigned char *src2_ptr, int src2_pels, int src2_lines, int src2_pitch,
int src2_startx, int src2_starty,
unsigned char *dest_ptr, int dest_pels, int dest_lines, int dest_pitch,
int dest_startx, int dest_starty,
int width, int height)
{
return I420andYUVAtoI420orYV12 (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height, CID_YV12);
}
/////////////////////////////////////////////////////////////
//
// I420andYUVA
//
// This function alpha-blends two I420 buffers into a third
// YV12 buffer using the alpha info tacked to the
// end of the second I420 buffer
//
/////////////////////////////////////////////////////////////
int HXEXPORT ENTRYPOINT(I420andYUVA) (
unsigned char *src1_ptr, int src1_pels, int src1_lines, int src1_pitch,
int src1_startx, int src1_starty,
unsigned char *src2_ptr, int src2_pels, int src2_lines, int src2_pitch,
int src2_startx, int src2_starty,
unsigned char *dest_ptr, int dest_pels, int dest_lines, int dest_pitch,
int dest_startx, int dest_starty,
int width, int height, int color_format)
{
#ifdef _USE_MMX_BLENDERS
if( !z_bCheckedForMMX )
{
z_bMMXAvailable = (checkMmxAvailablity()&CPU_HAS_MMX)?1:0;
z_bCheckedForMMX = TRUE;
}
if( z_bMMXAvailable )
{
switch (color_format)
{
case CID_I420:
return I420andYUVAtoI420_MMX (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
case CID_YV12:
return I420andYUVAtoYV12_MMX (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
case CID_YUY2:
return I420andYUVAtoYUY2_MMX (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
case CID_UYVY:
return I420andYUVAtoUYVY_MMX (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
default:
return -1;
}
}
#endif
switch (color_format)
{
case CID_I420:
case CID_YV12:
return I420andYUVAtoI420orYV12 (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height, color_format);
case CID_YUY2:
return I420andYUVAtoYUY2 (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
case CID_UYVY:
return I420andYUVAtoUYVY (
src1_ptr, src1_pels, src1_lines, src1_pitch, src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch, src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch, dest_startx, dest_starty,
width, height);
default:
return -1;
}
}
/////////////////////////////////////////////////////////////
//
// I420andI420toI420
//
// This function alpha-blends two I420 buffers into a third
// I420 buffer using the constant alpha given in the params.
//
/////////////////////////////////////////////////////////////
int HXEXPORT ENTRYPOINT(I420andI420toI420) (
unsigned char *src1_ptr, int src1_pels, int src1_lines, int src1_pitch,
int src1_startx, int src1_starty,
unsigned char *src2_ptr, int src2_pels, int src2_lines, int src2_pitch,
int src2_startx, int src2_starty,
unsigned char *dest_ptr, int dest_pels, int dest_lines, int dest_pitch,
int dest_startx, int dest_starty,
int width, int height, int alpha )
{
int ALPHA_tab[511];
int i, j, c;
unsigned char *s1, *s2, *d;
int s1_lineskip, s2_lineskip, d_lineskip;
#ifdef _USE_MMX_BLENDERS
if( !z_bCheckedForMMX )
{
z_bMMXAvailable = (checkMmxAvailablity()&CPU_HAS_MMX)?1:0;
z_bCheckedForMMX = TRUE;
}
if( z_bMMXAvailable )
{
return I420andI420toI420_MMX_sub (
src1_ptr, src1_pels, src1_lines, src1_pitch,
src1_startx, src1_starty,
src2_ptr, src2_pels, src2_lines, src2_pitch,
src2_startx, src2_starty,
dest_ptr, dest_pels, dest_lines, dest_pitch,
dest_startx, dest_starty,
width, height, alpha );
}
#endif
for (i = 0; i < 511; i++)
{
ALPHA_tab[i] = (i - 255) * alpha;
}
for (c = 0; c < 3; c++)
{
switch (c)
{
case 0:
// Y
s1 = src1_ptr + src1_starty * src1_pitch + src1_startx;
s2 = src2_ptr + src2_starty * src2_pitch + src2_startx;
d = dest_ptr + dest_starty * dest_pitch + dest_startx;
s1_lineskip = src1_pitch - width;
s2_lineskip = src2_pitch - width;
d_lineskip = dest_pitch - width;
break;
case 1:
// U
s1 = src1_ptr + src1_pitch * src1_lines;
s1 += (src1_starty * src1_pitch / 4) + src1_startx / 2;
s2 = src2_ptr + src2_pitch * src2_lines;
s2 += (src2_starty * src2_pitch / 4) + src2_startx / 2;
d = dest_ptr + dest_pitch * dest_lines;
d += (dest_starty * dest_pitch / 4) + dest_startx / 2;
s1_lineskip = (src1_pitch - width) / 2;
s2_lineskip = (src2_pitch - width) / 2;
d_lineskip = (dest_pitch - width) / 2;
width >>= 1;
height >>= 1;
break;
case 2:
// V
s1 = src1_ptr + 5 * src1_pitch * src1_lines / 4;
s1 += (src1_starty * src1_pitch / 4) + src1_startx / 2;
s2 = src2_ptr + 5 * src2_pitch * src2_lines / 4;
s2 += (src2_starty * src2_pitch / 4) + src2_startx / 2;
d = dest_ptr + 5 * dest_pitch * dest_lines / 4;
d += (dest_starty * dest_pitch / 4) + dest_startx / 2;
break;
}
for (i = 0; i < height; i++)
{
for (j = 0; j < (width-3); j += 4)
{
int x1, x2;
x1 = *(s1 + 0);
x2 = *(s2 + 0);
x1 -= x2;
x1 = *(ALPHA_tab + x1 + 255);
x1 >>= 8;
x1 += x2;
*(d + 0) = (unsigned char)x1;
x1 = *(s1 + 1);
x2 = *(s2 + 1);
x1 -= x2;
x1 = *(ALPHA_tab + x1 + 255);
x1 >>= 8;
x1 += x2;
*(d + 1) = (unsigned char)x1;
x1 = *(s1 + 2);
x2 = *(s2 + 2);
x1 -= x2;
x1 = *(ALPHA_tab + x1 + 255);
x1 >>= 8;
x1 += x2;
*(d + 2) = (unsigned char)x1;
x1 = *(s1 + 3);
x2 = *(s2 + 3);
x1 -= x2;
x1 = *(ALPHA_tab + x1 + 255);
x1 >>= 8;
x1 += x2;
*(d + 3) = (unsigned char)x1;
s1 += 4;
s2 += 4;
d += 4;
}
while (j < width)
{
int x1, x2;
x1 = *s1;
x2 = *s2;
x1 -= x2;
x1 = *(ALPHA_tab + x1 + 255);
x1 >>= 8;
x1 += x2;
*d++ = (unsigned char)x1;
s1++;
s2++;
j++;
}
s1 += s1_lineskip;
s2 += s2_lineskip;
d += d_lineskip;
}
}
return 0;
}
#if 0
void main (void)
{
unsigned char *s1, *s2, *dst;
int pels = 64;
int lines = 32;
int pitch = 64;
s1 = malloc (lines*pitch*3/2);
s2 = malloc (lines*pitch*5/2);
dst = malloc (lines*pitch*3/2);
memset (s1, 64, lines*pitch*3/2);
memset (s2, 192, lines*pitch*3/2);
memset (s2 + lines*pitch*3/2, 64, lines*pitch);
I420andYUVAtoI420orYV12 (
s1, pels, lines, pitch, pels, lines,
s2, pels, lines, pitch, pels, lines,
dst, pels, lines, pitch, pels, lines,
pels, lines, CID_I420);
free (s1);
free (s2);
free (dst);
}
#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -