You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
915 lines
35 KiB
915 lines
35 KiB
/*
|
|
* @DEC_COPYRIGHT@
|
|
*/
|
|
/*
|
|
* HISTORY
|
|
* $Log: scon_yuv_to_rgb.c,v $
|
|
* $EndLog$
|
|
*/
|
|
/*****************************************************************************
|
|
** Copyright (c) Digital Equipment Corporation, 1997 **
|
|
** **
|
|
** All Rights Reserved. Unpublished rights reserved under the copyright **
|
|
** laws of the United States. **
|
|
** **
|
|
** The software contained on this media is proprietary to and embodies **
|
|
** the confidential technology of Digital Equipment Corporation. **
|
|
** Possession, use, duplication or dissemination of the software and **
|
|
** media is authorized only pursuant to a valid written license from **
|
|
** Digital Equipment Corporation. **
|
|
** **
|
|
** RESTRICTED RIGHTS LEGEND Use, duplication, or disclosure by the U.S. **
|
|
** Government is subject to restrictions as set forth in Subparagraph **
|
|
** (c)(1)(ii) of DFARS 252.227-7013, or in FAR 52.227-19, as applicable. **
|
|
******************************************************************************/
|
|
|
|
/**********************************************************************/
|
|
/**********************************************************************/
|
|
/* */
|
|
/* Function: YUV_To_RGB_422_Init */
|
|
/* Author: Bill Hallahan */
|
|
/* Date: July 29, 1994 */
|
|
/* */
|
|
/* Abstract: */
|
|
/* */
|
|
/* This function produces a table that is used by the */
|
|
/* yuv_to_rgb_422 conversion routines. This table is required */
|
|
/* by the following routines. */
|
|
/* */
|
|
/* YUV_To_RGB_422_24_Bit */
|
|
/* YUV_To_RGB_422_555 */
|
|
/* */
|
|
/* */
|
|
/* YUV to RGB conversion can be described by a 3x3 matrix */
|
|
/* multiplication: */
|
|
/* */
|
|
/* R | 1 0 VR | Y */
|
|
/* G = | 1 UG VG | U */
|
|
/* B | 1 UB 0 | V */
|
|
/* */
|
|
/* where: */
|
|
/* */
|
|
/* 0 <= Y <= 255 */
|
|
/* -128 <= U <= 127 UG = -0.3455 UB = 1.7790 */
|
|
/* -128 <= V <= 127 VR = 1.4075 VG = -0.7169 */
|
|
/* */
|
|
/* The Red, Green, and Blue output values are obtained in */
|
|
/* parallel by summing three 64 bit words as shown. Each of the */
|
|
/* quadwords is obtained from either the Y_Table, the U_Table, */
|
|
/* or the V_table using the corresponding 8 bit Y, U, or V value */
|
|
/* as an index. Thus all multiplications are performed by table */
|
|
/* lookup. Note that the matrix output is ordered as B, R, G */
|
|
/* and then B again (starting at the LSB). */
|
|
/* This is to allow an efficient conversion to the output format. */
|
|
/* */
|
|
/* For 32-bit RGB, the Red and Blue bits are already in the */
|
|
/* correct position, the conversion routine only has to shift */
|
|
/* the Green bits. For General BI-BITFIELDS however, the Red, */
|
|
/* Green and Blue bits could be anywhere in a 16-bit or 32-bit */
|
|
/* word (we only support 16-bit for now). To avoid a costly */
|
|
/* decicion in the inner loop whether to shift the Blue bits left */
|
|
/* or right, we maintain a copy of the Blue bits in position 48 */
|
|
/* so a right shift will always work. Each conversion routine can */
|
|
/* choose whichever set of Blue bits that are fastest to use, */
|
|
/* they are identical. */
|
|
/* */
|
|
/* */
|
|
/* MSW LSW */
|
|
/* */
|
|
/* 63 48 47 32 31 16 15 0 Index */
|
|
/* ----------------------------------------------- */
|
|
/* | Y| Y| Y| Y| Y */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/* ----------------------------------------------- */
|
|
/* | ub| ug| 0| ub| U */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/* ----------------------------------------------- */
|
|
/* + | 0| vg| vr| 0| V */
|
|
/* ----------------------------------------------- */
|
|
/* __________________________________________________________ */
|
|
/* */
|
|
/* ----------------------------------------------- */
|
|
/* Total | 0| 0| x| G| x| R| x| B| */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/* */
|
|
/* where: */
|
|
/* */
|
|
/* ub = UB * U */
|
|
/* ug = UG * U */
|
|
/* vg = VG * V */
|
|
/* vr = VR * V */
|
|
/* */
|
|
/* */
|
|
/* The maximum absolute value for Y is 255 and the maximum */
|
|
/* for U or V is 128, so 9 bits is the minimum size to represent */
|
|
/* them together as two's complement values. The maximum */
|
|
/* chrominance (U or V) magnitude is 128. This is 0.5 as a Q9 */
|
|
/* two's complement fraction. 255 is 1 - 2^-8 in Q9 fraction form. */
|
|
/* */
|
|
/* The maximum possible bit growth is determined as follows. */
|
|
/* */
|
|
/* R_Max = 1 - 2^-8 + 0.5 * fabs(VR) = 1.6998 */
|
|
/* G_Max = 1 - 2^-8 + 0.5 * fabs(UG) + 0.5 * fabs(VG) = 1.5273 */
|
|
/* B_Max = 1 - 2^-8 + 0.5 * fabs(UB) = 1.8856 */
|
|
/* */
|
|
/* */
|
|
/* Since B_Max = 1.8856 then the next highest integer */
|
|
/* greater than or equal to log base 2 of 1.8856 is 1. So 1 bit */
|
|
/* is required for bit growth. The minimum accumulator size */
|
|
/* required is 9 + 1 = 10 bits. This code uses 12 bit accumulators */
|
|
/* since there are bits to spare. */
|
|
/* */
|
|
/* The 11'th bit (starting at bit 0) of each accumulator */
|
|
/* is the sign bit. This may be tested to determine if there is */
|
|
/* a negative result. Accumulator overflows are discarded as is */
|
|
/* normal for two's complement arithmetic. Each R, G, or B result */
|
|
/* that is over 255 is set to 255. Each R, G, or B result that is */
|
|
/* less than zero is set to zero. */
|
|
/* */
|
|
/* */
|
|
/* Input: */
|
|
/* */
|
|
/* */
|
|
/* bSign Contains a 32 bit boolean that if non-zero, changes */
|
|
/* the interpretation of the chrominance (U and V) data */
|
|
/* from an offset binary format, where the values range */
|
|
/* from 0 to 255 with 128 representing 0 chrominance, */
|
|
/* to a signed two's complement format, where the values */
|
|
/* range from -128 to 127. */
|
|
/* */
|
|
/* */
|
|
/* bBGR Contains a 32 bit boolean that if non-zero, changes */
|
|
/* the order of the conversion from RGB to BGR. */
|
|
/* */
|
|
/* */
|
|
/* pTable The address of the RGB (or BGR) conversion table */
|
|
/* that is filled in by this function. The table */
|
|
/* address must be quadword aligned. The table size */
|
|
/* is 6244 bytes 3 * 256 quadwords. */
|
|
/* */
|
|
/* */
|
|
/* Output: */
|
|
/* */
|
|
/* This function has no return value. */
|
|
/* */
|
|
/* */
|
|
/**********************************************************************/
|
|
/**********************************************************************/
|
|
|
|
/*
|
|
#define _SLIBDEBUG_
|
|
*/
|
|
|
|
#include "scon_int.h"
|
|
#include "SC_err.h"
|
|
#include "SC_conv.h"
|
|
|
|
#ifdef _SLIBDEBUG_
|
|
#include "sc_debug.h"
|
|
|
|
#define _DEBUG_ 1 /* detailed debuging statements */
|
|
#define _VERBOSE_ 1 /* show progress */
|
|
#define _VERIFY_ 1 /* verify correct operation */
|
|
#define _WARN_ 1 /* warnings about strange behavior */
|
|
#endif
|
|
|
|
/*
|
|
* Define NEW_YCBCR to use new YCbCr conversion values.
|
|
*/
|
|
#define NEW_YCBCR
|
|
|
|
#define GetRGB555(in16, r, g, b) b = (in16>>7)&0xF8; \
|
|
g = (in16>>2)&0xF8; \
|
|
r = (in16<<3)&0xF8
|
|
|
|
#define AddRGB555(in16, r, g, b) b += (in16>>7)&0xF8; \
|
|
g += (in16>>2)&0xF8; \
|
|
r += (in16<<3)&0xF8
|
|
|
|
#define PutRGB565(r, g, b, out16) out16 = ((r&0xf8)<<8)|((g&0xfC)<<3)|((b&0xf8)>>3)
|
|
|
|
#define PutRGB555(r, g, b, out16) out16 = ((r&0xf8)<<7)|((g&0xf8)<<2)|((b&0xf8)>>3)
|
|
|
|
#ifdef NEW_YCBCR
|
|
/*
|
|
** y = 16.000 + 0.257 * r + 0.504 * g + 0.098 * b ;
|
|
** u = 16.055 + 0.148 * (255-r) + 0.291 * (255-g) + 0.439 * b ;
|
|
** v = 16.055 + 0.439 * r + 0.368 * (255-g) + 0.071 * (255-b) ;
|
|
*/
|
|
#define YC 16.000
|
|
#define UC 16.055
|
|
#define VC 16.055
|
|
#define YR 0.257
|
|
#define UR 0.148
|
|
#define VR 0.439
|
|
#define YG 0.504
|
|
#define UG 0.291
|
|
#define VG 0.368
|
|
#define YB 0.098
|
|
#define UB 0.439
|
|
#define VB 0.071
|
|
|
|
#else /* !NEW_YCBCR */
|
|
/*
|
|
** ( y = 0.0 0.299 * r + 0.587 * g + 0.1140 * b ; )
|
|
** ( u = 0.245 + 0.169 * (255-r) + 0.332 * (255-g) + 0.5000 * b ; )
|
|
** ( v = 0.4235 + 0.500 * r + 0.419 * (255-g) + 0.0813 * (255-b) ; )
|
|
*/
|
|
#define YC 0.0
|
|
#define UC 0.245
|
|
#define VC 0.4235
|
|
#define YR 0.299
|
|
#define UR 0.169
|
|
#define VR 0.500
|
|
#define YG 0.587
|
|
#define UG 0.332
|
|
#define VG 0.419
|
|
#define YB 0.1140
|
|
#define UB 0.5000
|
|
#define VB 0.0813
|
|
|
|
#endif /* !NEW_YCBCR */
|
|
|
|
/********************************** YUV to RGB ***********************************/
|
|
/*
|
|
* The YUV to RGB conversion routines
|
|
* generates RGB values in a 64-bit
|
|
* word thus:
|
|
*
|
|
* 63 56 55 48 47 40 39 32 31 24 23 16 15 8 7 0
|
|
* +-------+-------+-------+-------+-------+-------+-------+-------+
|
|
* | 0 | Blue | 0 | Green | 0 | Red | 0 | Blue |
|
|
* +-------+-------+-------+-------+-------+-------+-------+-------+
|
|
*
|
|
* Figure out how many steps to the right are needed to
|
|
* shift the red, green and blue into the correct position.
|
|
*/
|
|
#define ArrangRGB565(inrgb, outrgb) \
|
|
outrgb=((inrgb>>8)&0xf800)|((inrgb>>29)&0x07e0)|((inrgb>>51)&0x001f)
|
|
#define ArrangRGB888(inrgb, outrgb) \
|
|
outrgb=(inrgb&0xFF0000)|((inrgb>>24)&0x00FF00)|(inrgb&0x0000FF)
|
|
|
|
|
|
SconStatus_t sconInitYUVtoRGB(SconInfo_t *Info)
|
|
{
|
|
qword i, qY, qUV, qSigned;
|
|
SconBoolean_t bBGR=Info->Output.vinfo.Rmask&1; /* BGR vs RGB ordering */
|
|
SconBoolean_t bSign=FALSE; /* U and V are signed values */
|
|
unsigned qword qRed;
|
|
unsigned qword qGreen;
|
|
unsigned qword qBlue;
|
|
unsigned qword qTemp;
|
|
unsigned qword qAccMask = 0xFFF;
|
|
unsigned qword *pTable, *pU_Table, *pV_Table;
|
|
double Chrominance;
|
|
#ifdef NEW_YCBCR
|
|
double CF_UB = 2.018;
|
|
double CF_UG = -0.391;
|
|
double CF_VG = -0.813;
|
|
double CF_VR = 1.596;
|
|
#else /* !NEW_YCBCR */
|
|
double CF_UB = 1.7790;
|
|
double CF_UG = -0.3455;
|
|
double CF_VG = -0.7169;
|
|
double CF_VR = 1.4075;
|
|
#endif /* !NEW_YCBCR */
|
|
|
|
/* allocate memory to hold the lookup table */
|
|
if (Info->Table && Info->TableSize<256*3*8)
|
|
{
|
|
ScPaFree(Info->Table);
|
|
Info->Table=NULL;
|
|
}
|
|
if (Info->Table==NULL)
|
|
{
|
|
if ((Info->Table = ScPaMalloc(256*3*8)) == NULL)
|
|
return(SconErrorMemory);
|
|
Info->TableSize=256*3*8;
|
|
}
|
|
/*
|
|
* Set constant that determines if the U and V chrominance values
|
|
* are interpreted as signed or unsigned values.
|
|
*/
|
|
if ( !bSign )
|
|
qSigned = 0;
|
|
else
|
|
qSigned = 0xFFFFFFFFFFFFFF80;
|
|
|
|
/* Get the U, and V table pointers. */
|
|
pTable = (unsigned qword *)Info->Table;
|
|
pU_Table = pTable + 256;
|
|
pV_Table = pU_Table + 256;
|
|
|
|
/* Initialize the Y_Table, the U_Table, and the V_Table. */
|
|
for ( i = 0; i < 256; i++ )
|
|
{
|
|
/******************************************************************/
|
|
/* Construct the Y array value for the current index value. */
|
|
/* */
|
|
/* 63 48 47 32 31 16 15 0 Index */
|
|
/* ----------------------------------------------- */
|
|
/* | Y| Y| Y| Y| Y = i */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/******************************************************************/
|
|
|
|
#ifdef NEW_YCBCR
|
|
qY = (qword) ((i-16)*1.164) ;
|
|
qY = (qY < 0) ? 0 : (qY > 255) ? 255 : qY ;
|
|
#else /* !NEW_YCBCR */
|
|
qY = i ;
|
|
#endif /* !NEW_YCBCR */
|
|
qY |= qY << 16 ;
|
|
*pTable++ = qY | ( qY << 32 ) ;
|
|
/******************************************************************/
|
|
/* Construct the U array value for the current index value. */
|
|
/* */
|
|
/* 63 48 47 32 31 16 15 0 Index */
|
|
/* ----------------------------------------------- */
|
|
/* | ub| ug| 0| ub| U = i */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/******************************************************************/
|
|
|
|
#ifdef NEW_YCBCR
|
|
qUV = (i< 16) ? 16
|
|
: (i<240) ? i
|
|
: 240 ;
|
|
#else /* !NEW_YCBCR */
|
|
qUV = i ;
|
|
#endif /* !NEW_YCBCR */
|
|
|
|
Chrominance = (double) (( qUV - 128 ) ^ qSigned );
|
|
|
|
qBlue = ((qword)( CF_UB * Chrominance )) & qAccMask;
|
|
qGreen = ((qword)( CF_UG * Chrominance )) & qAccMask;
|
|
qRed = 0;
|
|
if ( bBGR )
|
|
{
|
|
qTemp = qBlue;
|
|
qBlue = qRed;
|
|
qRed = qTemp;
|
|
}
|
|
*pU_Table++ = qBlue | ( qRed << 16 ) | ( qGreen << 32 ) | ( qBlue << 48 );
|
|
/******************************************************************/
|
|
/* Construct the V array value for the current index value. */
|
|
/* */
|
|
/* 63 48 47 32 31 16 15 0 Index */
|
|
/* ----------------------------------------------- */
|
|
/* | 0| vg| vr| 0| V = i */
|
|
/* ----------------------------------------------- */
|
|
/* */
|
|
/******************************************************************/
|
|
qBlue = 0;
|
|
qGreen = ((qword)( CF_VG * Chrominance )) & qAccMask;
|
|
qRed = ((qword)( CF_VR * Chrominance )) & qAccMask;
|
|
if ( bBGR )
|
|
{
|
|
qTemp = qBlue;
|
|
qBlue = qRed;
|
|
qRed = qTemp;
|
|
}
|
|
*pV_Table++ = qBlue | ( qRed << 16 ) | ( qGreen << 32 );
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
SconStatus_t scon422ToRGB565(unsigned char *inimage, unsigned char *outimage,
|
|
unsigned dword width, unsigned dword height,
|
|
dword stride, unsigned qword *pTable)
|
|
{
|
|
unsigned qword y, u, v, mask ;
|
|
unsigned qword y0, y1, y2, y3 ;
|
|
unsigned qword y4, y5, y6, y7 ;
|
|
unsigned qword u01, v01, u23, v23 ;
|
|
unsigned qword *yData=(unsigned qword *)inimage;
|
|
unsigned int *uData=(unsigned int *)(inimage+width*height);
|
|
unsigned int *vData=(unsigned int *)(inimage+(width*height*3)/2);
|
|
unsigned qword *rgbData=(unsigned qword *)outimage;
|
|
unsigned dword x, line;
|
|
if (stride<0) /* flip */
|
|
outimage=outimage+((height-1)*(-stride));
|
|
for (line=height; line>0; line--, outimage+=stride)
|
|
{
|
|
rgbData=(unsigned qword *)outimage;
|
|
for (x=width>>3; x>0; x--)
|
|
{
|
|
y = *yData++ ;
|
|
y0 = y & 255 ; y >>= 8 ;
|
|
y1 = y & 255 ; y >>= 8 ;
|
|
y2 = y & 255 ; y >>= 8 ;
|
|
y3 = y & 255 ; y >>= 8 ;
|
|
y4 = y & 255 ; y >>= 8 ;
|
|
y5 = y & 255 ; y >>= 8 ;
|
|
y6 = y & 255 ; y >>= 8 ;
|
|
y7 = y & 255 ; y >>= 8 ;
|
|
|
|
u = *uData++ ;
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ; u >>= 8 ;
|
|
|
|
v = *vData++ ;
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ; v >>= 8 ;
|
|
|
|
y0 = pTable[y0] ;
|
|
y1 = pTable[y1] ;
|
|
y2 = pTable[y2] ;
|
|
y3 = pTable[y3] ;
|
|
y4 = pTable[y4] ;
|
|
y5 = pTable[y5] ;
|
|
y6 = pTable[y6] ;
|
|
y7 = pTable[y7] ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
/* Now, convert to RGB */
|
|
y0 += u01 + v01 ;
|
|
y1 += u01 + v01 ;
|
|
y2 += u23 + v23 ;
|
|
y3 += u23 + v23 ;
|
|
/*
|
|
* Same thing for more pixels.
|
|
* Use u01 for u45 and u23 for u67
|
|
*/
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ;
|
|
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
/* Convert to RGB. */
|
|
y4 += u01 + v01 ;
|
|
y5 += u01 + v01 ;
|
|
y6 += u23 + v23 ;
|
|
y7 += u23 + v23 ;
|
|
/* See if any value is out of range. */
|
|
mask = (unsigned qword)0x0F000F000F000F00L;
|
|
if( (y0 | y1 | y2 | y3 | y4 | y5 | y6 | y7) & mask )
|
|
{
|
|
/* Zero values that are negative */
|
|
mask = (unsigned qword)0x0800080008000800L ;
|
|
y = y0 & mask ; y0 &= ~(y - (y>>11)) ;
|
|
y = y1 & mask ; y1 &= ~(y - (y>>11)) ;
|
|
y = y2 & mask ; y2 &= ~(y - (y>>11)) ;
|
|
y = y3 & mask ; y3 &= ~(y - (y>>11)) ;
|
|
y = y4 & mask ; y4 &= ~(y - (y>>11)) ;
|
|
y = y5 & mask ; y5 &= ~(y - (y>>11)) ;
|
|
y = y6 & mask ; y6 &= ~(y - (y>>11)) ;
|
|
y = y7 & mask ; y7 &= ~(y - (y>>11)) ;
|
|
/* Clamp values that are > 255 to 255. */
|
|
mask = (unsigned qword)0x0100010001000100L ;
|
|
y = y0 & mask ; y0 |= (y - (y >> 8)) ;
|
|
y = y1 & mask ; y1 |= (y - (y >> 8)) ;
|
|
y = y2 & mask ; y2 |= (y - (y >> 8)) ;
|
|
y = y3 & mask ; y3 |= (y - (y >> 8)) ;
|
|
y = y4 & mask ; y4 |= (y - (y >> 8)) ;
|
|
y = y5 & mask ; y5 |= (y - (y >> 8)) ;
|
|
y = y6 & mask ; y6 |= (y - (y >> 8)) ;
|
|
y = y7 & mask ; y7 |= (y - (y >> 8)) ;
|
|
/* Stray bits left over will be masked below */
|
|
}
|
|
ArrangRGB565(y0, y0);
|
|
ArrangRGB565(y1, y1);
|
|
ArrangRGB565(y2, y2);
|
|
ArrangRGB565(y3, y3);
|
|
ArrangRGB565(y4, y4);
|
|
ArrangRGB565(y5, y5);
|
|
ArrangRGB565(y6, y6);
|
|
ArrangRGB565(y7, y7);
|
|
*rgbData++ = y0 | (y1 << 16) | (y2 << 32) | (y3 << 48) ;
|
|
*rgbData++ = y4 | (y5 << 16) | (y6 << 32) | (y7 << 48) ;
|
|
}
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
SconStatus_t scon422ToRGB888(unsigned char *inimage, unsigned char *outimage,
|
|
unsigned dword width, unsigned dword height,
|
|
dword stride, unsigned qword *pTable)
|
|
{
|
|
unsigned qword y, u, v, mask ;
|
|
unsigned qword y0, y1, y2, y3 ;
|
|
unsigned qword u01, v01, u23, v23 ;
|
|
unsigned dword *yData=(unsigned dword *)inimage;
|
|
unsigned word *uData=(unsigned word *)(inimage+width*height);
|
|
unsigned word *vData=(unsigned word *)(inimage+(width*height*3)/2);
|
|
unsigned dword *rgbData=(unsigned dword *)outimage;
|
|
unsigned dword x, line;
|
|
if (stride<0) /* flip */
|
|
outimage=outimage+((height-1)*(-stride));
|
|
for (line=height; line>0; line--, outimage+=stride)
|
|
{
|
|
rgbData=(unsigned dword *)outimage;
|
|
for (x=width>>2; x>0; x--)
|
|
{
|
|
y = *yData++ ;
|
|
y0 = y & 255 ; y >>= 8 ;
|
|
y1 = y & 255 ; y >>= 8 ;
|
|
y2 = y & 255 ; y >>= 8 ;
|
|
y3 = y & 255 ;
|
|
|
|
u = *uData++ ;
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ;
|
|
|
|
v = *vData++ ;
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ;
|
|
|
|
y0 = pTable[y0] ;
|
|
y1 = pTable[y1] ;
|
|
y2 = pTable[y2] ;
|
|
y3 = pTable[y3] ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
/* Now, convert to RGB */
|
|
y0 += u01 + v01 ;
|
|
y1 += u01 + v01 ;
|
|
y2 += u23 + v23 ;
|
|
y3 += u23 + v23 ;
|
|
/* See if any value is out of range. */
|
|
mask = (unsigned qword)0x0F000F000F000F00L;
|
|
if( (y0 | y1 | y2 | y3) & mask )
|
|
{
|
|
/* Zero values that are negative */
|
|
mask = (unsigned qword)0x0800080008000800L ;
|
|
y = y0 & mask ; y0 &= ~(y - (y>>11)) ;
|
|
y = y1 & mask ; y1 &= ~(y - (y>>11)) ;
|
|
y = y2 & mask ; y2 &= ~(y - (y>>11)) ;
|
|
y = y3 & mask ; y3 &= ~(y - (y>>11)) ;
|
|
/* Clamp values that are > 255 to 255. */
|
|
mask = (unsigned qword)0x0100010001000100L ;
|
|
y = y0 & mask ; y0 |= (y - (y >> 8)) ;
|
|
y = y1 & mask ; y1 |= (y - (y >> 8)) ;
|
|
y = y2 & mask ; y2 |= (y - (y >> 8)) ;
|
|
y = y3 & mask ; y3 |= (y - (y >> 8)) ;
|
|
/* Stray bits left over will be masked below */
|
|
}
|
|
ArrangRGB888(y0, y0);
|
|
ArrangRGB888(y1, y1);
|
|
ArrangRGB888(y2, y2);
|
|
ArrangRGB888(y3, y3);
|
|
*rgbData++ = (unsigned dword)(y0 | (y1 << 24));
|
|
*rgbData++ = (unsigned dword)((y1 & 0xFFFF) | (y2 << 16));
|
|
*rgbData++ = (unsigned dword)((y2 & 0xFF) | (y3 << 8));
|
|
}
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
SconStatus_t scon420ToRGB565(unsigned char *inimage, unsigned char *outimage,
|
|
unsigned dword width, unsigned dword height,
|
|
dword stride, unsigned qword *pTable)
|
|
{
|
|
unsigned qword y, u, v, mask ;
|
|
unsigned qword y0, y1, y2, y3 ;
|
|
unsigned qword y4, y5, y6, y7 ;
|
|
unsigned qword u01, v01, u23, v23;
|
|
unsigned qword *yData=(unsigned qword *)inimage;
|
|
unsigned int *uData=(unsigned int *)(inimage+width*height);
|
|
unsigned int *vData=(unsigned int *)(inimage+(width*height*5)/4);
|
|
unsigned int *puData, *pvData;
|
|
unsigned qword *rgbData=(unsigned qword *)outimage;
|
|
unsigned dword x, line;
|
|
if (stride<0) /* flip */
|
|
outimage=outimage+((height-1)*(-stride));
|
|
puData=uData;
|
|
pvData=vData;
|
|
for (line=(height>>1)<<1; line>0; line--, outimage+=stride)
|
|
{
|
|
rgbData=(unsigned qword *)outimage;
|
|
if (line&1) /* odd line, reuse U and V */
|
|
{
|
|
puData=uData;
|
|
pvData=vData;
|
|
}
|
|
else
|
|
{
|
|
uData=puData;
|
|
vData=pvData;
|
|
}
|
|
for (x=width>>3; x>0; x--)
|
|
{
|
|
y = *yData++ ;
|
|
y0 = y & 255 ; y >>= 8 ;
|
|
y1 = y & 255 ; y >>= 8 ;
|
|
y2 = y & 255 ; y >>= 8 ;
|
|
y3 = y & 255 ; y >>= 8 ;
|
|
y4 = y & 255 ; y >>= 8 ;
|
|
y5 = y & 255 ; y >>= 8 ;
|
|
y6 = y & 255 ; y >>= 8 ;
|
|
y7 = y & 255 ; y >>= 8 ;
|
|
|
|
u = *puData++ ;
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ; u >>= 8 ;
|
|
|
|
v = *pvData++ ;
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ; v >>= 8 ;
|
|
|
|
y0 = pTable[y0] ;
|
|
y1 = pTable[y1] ;
|
|
y2 = pTable[y2] ;
|
|
y3 = pTable[y3] ;
|
|
y4 = pTable[y4] ;
|
|
y5 = pTable[y5] ;
|
|
y6 = pTable[y6] ;
|
|
y7 = pTable[y7] ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
|
|
/* Now, convert to RGB */
|
|
y0 += u01 + v01 ;
|
|
y1 += u01 + v01 ;
|
|
y2 += u23 + v23 ;
|
|
y3 += u23 + v23 ;
|
|
/*
|
|
* Same thing for more pixels.
|
|
* Use u01 for u45 and u23 for u67
|
|
*/
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ;
|
|
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
/* Convert to RGB. */
|
|
y4 += u01 + v01 ;
|
|
y5 += u01 + v01 ;
|
|
y6 += u23 + v23 ;
|
|
y7 += u23 + v23 ;
|
|
/* See if any value is out of range. */
|
|
mask = (unsigned qword)0x0F000F000F000F00L;
|
|
if( (y0 | y1 | y2 | y3 | y4 | y5 | y6 | y7) & mask )
|
|
{
|
|
/* Zero values that are negative */
|
|
mask = (unsigned qword)0x0800080008000800L ;
|
|
y = y0 & mask ; y0 &= ~(y - (y>>11)) ;
|
|
y = y1 & mask ; y1 &= ~(y - (y>>11)) ;
|
|
y = y2 & mask ; y2 &= ~(y - (y>>11)) ;
|
|
y = y3 & mask ; y3 &= ~(y - (y>>11)) ;
|
|
y = y4 & mask ; y4 &= ~(y - (y>>11)) ;
|
|
y = y5 & mask ; y5 &= ~(y - (y>>11)) ;
|
|
y = y6 & mask ; y6 &= ~(y - (y>>11)) ;
|
|
y = y7 & mask ; y7 &= ~(y - (y>>11)) ;
|
|
/* Clamp values that are > 255 to 255. */
|
|
mask = (unsigned qword)0x0100010001000100L ;
|
|
y = y0 & mask ; y0 |= (y - (y >> 8)) ;
|
|
y = y1 & mask ; y1 |= (y - (y >> 8)) ;
|
|
y = y2 & mask ; y2 |= (y - (y >> 8)) ;
|
|
y = y3 & mask ; y3 |= (y - (y >> 8)) ;
|
|
y = y4 & mask ; y4 |= (y - (y >> 8)) ;
|
|
y = y5 & mask ; y5 |= (y - (y >> 8)) ;
|
|
y = y6 & mask ; y6 |= (y - (y >> 8)) ;
|
|
y = y7 & mask ; y7 |= (y - (y >> 8)) ;
|
|
/* Stray bits left over will be masked below */
|
|
}
|
|
ArrangRGB565(y0, y0);
|
|
ArrangRGB565(y1, y1);
|
|
ArrangRGB565(y2, y2);
|
|
ArrangRGB565(y3, y3);
|
|
ArrangRGB565(y4, y4);
|
|
ArrangRGB565(y5, y5);
|
|
ArrangRGB565(y6, y6);
|
|
ArrangRGB565(y7, y7);
|
|
*rgbData++ = y0 | (y1 << 16) | (y2 << 32) | (y3 << 48) ;
|
|
*rgbData++ = y4 | (y5 << 16) | (y6 << 32) | (y7 << 48) ;
|
|
}
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
SconStatus_t scon420ToRGB888(unsigned char *inimage, unsigned char *outimage,
|
|
unsigned dword width, unsigned dword height,
|
|
dword stride, unsigned qword *pTable)
|
|
{
|
|
unsigned qword y, u, v, mask ;
|
|
unsigned qword y0, y1, y2, y3 ;
|
|
unsigned qword u01, v01, u23, v23;
|
|
unsigned dword *yData=(unsigned dword *)inimage;
|
|
unsigned word *uData=(unsigned word *)(inimage+width*height);
|
|
unsigned word *vData=(unsigned word *)(inimage+(width*height*5)/4);
|
|
unsigned word *puData, *pvData;
|
|
unsigned dword *rgbData=(unsigned dword *)outimage;
|
|
unsigned dword x, line;
|
|
if (stride<0) /* flip */
|
|
outimage=outimage+((height-1)*(-stride));
|
|
puData=uData;
|
|
pvData=vData;
|
|
for (line=(height>>1)<<1; line>0; line--, outimage+=stride)
|
|
{
|
|
rgbData=(unsigned dword *)outimage;
|
|
if (line&1) /* odd line, reuse U and V */
|
|
{
|
|
puData=uData;
|
|
pvData=vData;
|
|
}
|
|
else
|
|
{
|
|
uData=puData;
|
|
vData=pvData;
|
|
}
|
|
for (x=width>>2; x>0; x--)
|
|
{
|
|
y = *yData++ ;
|
|
y0 = y & 255 ; y >>= 8 ;
|
|
y1 = y & 255 ; y >>= 8 ;
|
|
y2 = y & 255 ; y >>= 8 ;
|
|
y3 = y & 255 ;
|
|
|
|
u = *puData++ ;
|
|
u01 = u & 255 ; u >>= 8 ;
|
|
u23 = u & 255 ;
|
|
|
|
v = *pvData++ ;
|
|
v01 = v & 255 ; v >>= 8 ;
|
|
v23 = v & 255 ;
|
|
|
|
y0 = pTable[y0] ;
|
|
y1 = pTable[y1] ;
|
|
y2 = pTable[y2] ;
|
|
y3 = pTable[y3] ;
|
|
|
|
u01 = pTable[u01+256] ;
|
|
u23 = pTable[u23+256] ;
|
|
|
|
v01 = pTable[v01+512] ;
|
|
v23 = pTable[v23+512] ;
|
|
|
|
/* Now, convert to RGB */
|
|
y0 += u01 + v01 ;
|
|
y1 += u01 + v01 ;
|
|
y2 += u23 + v23 ;
|
|
y3 += u23 + v23 ;
|
|
/* See if any value is out of range. */
|
|
mask = (unsigned qword)0x0F000F000F000F00L;
|
|
if( (y0 | y1 | y2 | y3) & mask )
|
|
{
|
|
/* Zero values that are negative */
|
|
mask = (unsigned qword)0x0800080008000800L ;
|
|
y = y0 & mask ; y0 &= ~(y - (y>>11)) ;
|
|
y = y1 & mask ; y1 &= ~(y - (y>>11)) ;
|
|
y = y2 & mask ; y2 &= ~(y - (y>>11)) ;
|
|
y = y3 & mask ; y3 &= ~(y - (y>>11)) ;
|
|
/* Clamp values that are > 255 to 255. */
|
|
mask = (unsigned qword)0x0100010001000100L ;
|
|
y = y0 & mask ; y0 |= (y - (y >> 8)) ;
|
|
y = y1 & mask ; y1 |= (y - (y >> 8)) ;
|
|
y = y2 & mask ; y2 |= (y - (y >> 8)) ;
|
|
y = y3 & mask ; y3 |= (y - (y >> 8)) ;
|
|
/* Stray bits left over will be masked below */
|
|
}
|
|
ArrangRGB888(y0, y0);
|
|
ArrangRGB888(y1, y1);
|
|
ArrangRGB888(y2, y2);
|
|
ArrangRGB888(y3, y3);
|
|
*rgbData++ = (unsigned dword)(y0 | (y1 << 24));
|
|
*rgbData++ = (unsigned dword)((y1 >> 8) | (y2 << 16));
|
|
*rgbData++ = (unsigned dword)((y2 >> 16) | (y3 << 8));
|
|
}
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
/********************************** YUV to RGB ***********************************/
|
|
SconStatus_t sconInitRGBtoYUV(SconInfo_t *Info)
|
|
{
|
|
unsigned dword i, y, u, v ;
|
|
qword *RedToYuyv, *GreenToYuyv, *BlueToYuyv;
|
|
|
|
/* allocate memory to hold the lookup table */
|
|
if (Info->Table && Info->TableSize<256*3*8)
|
|
{
|
|
ScPaFree(Info->Table);
|
|
Info->Table=NULL;
|
|
}
|
|
if (Info->Table==NULL)
|
|
{
|
|
if ((Info->Table = ScPaMalloc(256*3*8)) == NULL)
|
|
return(SconErrorMemory);
|
|
Info->TableSize=256*3*8;
|
|
}
|
|
RedToYuyv=(unsigned qword *)Info->Table;
|
|
GreenToYuyv=RedToYuyv+256;
|
|
BlueToYuyv=RedToYuyv+512;
|
|
|
|
for( i=0 ; i<256 ; i++ )
|
|
{
|
|
/*
|
|
* Calculate contribution from red.
|
|
* We will also add in the constant here.
|
|
* Pack it into the tables thus: lsb->YUYV<-msb
|
|
*/
|
|
y = (unsigned dword) ((float)YC + (float)YR * (float)i) ;
|
|
u = (unsigned dword) ((float)UC + (float)UR * (float)(255-i)) ;
|
|
v = (unsigned dword) ((float)VC + (float)VR * (float)i) ;
|
|
RedToYuyv[i] = (y | (u<<8) | (y<<16) | (v<<24)) ;
|
|
/*
|
|
* Calculate contribution from green.
|
|
*/
|
|
y = (unsigned dword) ((float)YG * (float)i) ;
|
|
u = (unsigned dword) ((float)UG * (float)(255-i)) ;
|
|
v = (unsigned dword) ((float)VG * (float)(255-i)) ;
|
|
GreenToYuyv[i] = (y | (u<<8) | (y<<16) | (v<<24)) ;
|
|
/*
|
|
* Calculate contribution from blue.
|
|
*/
|
|
y = (unsigned dword) ((float)YB * (float)i) ;
|
|
u = (unsigned dword) ((float)UB * (float)i) ;
|
|
v = (unsigned dword) ((float)VB * (float)(255-i)) ;
|
|
BlueToYuyv[i] = (y | (u<<8) | (y<<16) | (v<<24)) ;
|
|
}
|
|
return(SconErrorNone);
|
|
}
|
|
|
|
/*
|
|
** Name: sconRGB888To420
|
|
** Purpose: convert 24-bit RGB (8:8:8 format) to 16-bit YCrCb (4:1:1 format)
|
|
*/
|
|
SconStatus_t sconRGB888To420(unsigned char *inimage, unsigned char *outimage,
|
|
unsigned dword width, unsigned dword height,
|
|
dword stride, unsigned qword *pTable)
|
|
{
|
|
unsigned char *yData=(unsigned char *)outimage;
|
|
unsigned char *uData=(unsigned char *)(outimage+width*height);
|
|
unsigned char *vData=(unsigned char *)(outimage+(width*height*5)/4);
|
|
register unsigned dword row, col;
|
|
unsigned dword yuyv, r, g, b;
|
|
unsigned char *tmp, *evl, *odl;
|
|
if (stride<0)
|
|
inimage=inimage+(-stride*(height-1));
|
|
for (row=height; row>0; row--)
|
|
{
|
|
if (row&1)
|
|
{
|
|
tmp=inimage;
|
|
for (col = 0; col < width; col++)
|
|
{
|
|
r = *tmp++;
|
|
g = *tmp++;
|
|
b = *tmp++;
|
|
yuyv = (unsigned dword)(pTable[r] + pTable[g+256] + pTable[b+512]);
|
|
*yData++ = (yuyv&0xff);
|
|
}
|
|
inimage+=stride;
|
|
}
|
|
else
|
|
{
|
|
tmp = evl = inimage;
|
|
inimage+=stride;
|
|
odl = inimage;
|
|
for (col = 0; col < width; col++)
|
|
{
|
|
r = *tmp++;
|
|
g = *tmp++;
|
|
b = *tmp++;
|
|
yuyv = (unsigned dword)(pTable[r] + pTable[g+256] + pTable[b+512]);
|
|
*yData++ = (yuyv&0xff);
|
|
/* We only store every fourth value of u and v components */
|
|
if (col & 1)
|
|
{
|
|
/* Compute average r, g and b values */
|
|
r = (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
g = (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
b = (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
r += (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
g += (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
b += (unsigned dword)*evl++ + (unsigned dword)*odl++;
|
|
r = r >> 2;
|
|
g = g >> 2;
|
|
b = b >> 2;
|
|
yuyv = (unsigned dword)(pTable[r] + pTable[g+256] + pTable[b+512]);
|
|
*uData++ = (yuyv>>24)& 0xff; // V
|
|
*vData++ = (yuyv>>8) & 0xff; // U
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return(SconErrorNone);
|
|
}
|