Blame libfreerdp/primitives/prim_YUV.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Generic YUV/RGB conversion operations
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2015-2017 Armin Novak <armin.novak@thincast.com>
Packit 1fb8d4
 * Copyright 2015-2017 Norbert Federa <norbert.federa@thincast.com>
Packit 1fb8d4
 * Copyright 2015-2017 Vic Lee
Packit 1fb8d4
 * Copyright 2015-2017 Thincast Technologies GmbH
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/types.h>
Packit 1fb8d4
#include <freerdp/primitives.h>
Packit 1fb8d4
#include <freerdp/codec/color.h>
Packit 1fb8d4
#include "prim_internal.h"
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_LumaToYUV444(const BYTE* const pSrcRaw[3], const UINT32 srcStep[3],
Packit 1fb8d4
                                      BYTE* pDstRaw[3], const UINT32 dstStep[3],
Packit 1fb8d4
                                      const RECTANGLE_16* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	const UINT32 nWidth = roi->right - roi->left;
Packit 1fb8d4
	const UINT32 nHeight = roi->bottom - roi->top;
Packit 1fb8d4
	const UINT32 halfWidth = (nWidth + 1) / 2;
Packit 1fb8d4
	const UINT32 halfHeight = (nHeight + 1) / 2;
Packit 1fb8d4
	const UINT32 oddY = 1;
Packit 1fb8d4
	const UINT32 evenY = 0;
Packit 1fb8d4
	const UINT32 oddX = 1;
Packit 1fb8d4
	const UINT32 evenX = 0;
Packit Service 5a9772
	const BYTE* pSrc[3] = { pSrcRaw[0] + roi->top * srcStep[0] + roi->left,
Packit Service 5a9772
		                    pSrcRaw[1] + roi->top / 2 * srcStep[1] + roi->left / 2,
Packit Service 5a9772
		                    pSrcRaw[2] + roi->top / 2 * srcStep[2] + roi->left / 2 };
Packit Service 5a9772
	BYTE* pDst[3] = { pDstRaw[0] + roi->top * dstStep[0] + roi->left,
Packit Service 5a9772
		              pDstRaw[1] + roi->top * dstStep[1] + roi->left,
Packit Service 5a9772
		              pDstRaw[2] + roi->top * dstStep[2] + roi->left };
Packit 1fb8d4
Packit 1fb8d4
	/* Y data is already here... */
Packit 1fb8d4
	/* B1 */
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* Ym = pSrc[0] + srcStep[0] * y;
Packit 1fb8d4
		BYTE* pY = pDst[0] + dstStep[0] * y;
Packit 1fb8d4
		memcpy(pY, Ym, nWidth);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* The first half of U, V are already here part of this frame. */
Packit 1fb8d4
	/* B2 and B3 */
Packit 1fb8d4
	for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const UINT32 val2y = (2 * y + evenY);
Packit 1fb8d4
		const UINT32 val2y1 = val2y + oddY;
Packit 1fb8d4
		const BYTE* Um = pSrc[1] + srcStep[1] * y;
Packit 1fb8d4
		const BYTE* Vm = pSrc[2] + srcStep[2] * y;
Packit 1fb8d4
		BYTE* pU = pDst[1] + dstStep[1] * val2y;
Packit 1fb8d4
		BYTE* pV = pDst[2] + dstStep[2] * val2y;
Packit 1fb8d4
		BYTE* pU1 = pDst[1] + dstStep[1] * val2y1;
Packit 1fb8d4
		BYTE* pV1 = pDst[2] + dstStep[2] * val2y1;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 val2x = 2 * x + evenX;
Packit 1fb8d4
			const UINT32 val2x1 = val2x + oddX;
Packit 1fb8d4
			pU[val2x] = Um[x];
Packit 1fb8d4
			pV[val2x] = Vm[x];
Packit 1fb8d4
			pU[val2x1] = Um[x];
Packit 1fb8d4
			pV[val2x1] = Vm[x];
Packit 1fb8d4
			pU1[val2x] = Um[x];
Packit 1fb8d4
			pV1[val2x] = Vm[x];
Packit 1fb8d4
			pU1[val2x1] = Um[x];
Packit 1fb8d4
			pV1[val2x1] = Vm[x];
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static pstatus_t general_ChromaFilter(BYTE* pDst[3], const UINT32 dstStep[3],
Packit 1fb8d4
                                      const RECTANGLE_16* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 oddY = 1;
Packit 1fb8d4
	const UINT32 evenY = 0;
Packit 1fb8d4
	const UINT32 nWidth = roi->right - roi->left;
Packit 1fb8d4
	const UINT32 nHeight = roi->bottom - roi->top;
Packit 1fb8d4
	const UINT32 halfHeight = (nHeight + 1) / 2;
Packit 1fb8d4
	const UINT32 halfWidth = (nWidth + 1) / 2;
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
Packit 1fb8d4
	/* Filter */
Packit 1fb8d4
	for (y = roi->top; y < halfHeight + roi->top; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const UINT32 val2y = (y * 2 + evenY);
Packit 1fb8d4
		const UINT32 val2y1 = val2y + oddY;
Packit 1fb8d4
		BYTE* pU1 = pDst[1] + dstStep[1] * val2y1;
Packit 1fb8d4
		BYTE* pV1 = pDst[2] + dstStep[2] * val2y1;
Packit 1fb8d4
		BYTE* pU = pDst[1] + dstStep[1] * val2y;
Packit 1fb8d4
		BYTE* pV = pDst[2] + dstStep[2] * val2y;
Packit 1fb8d4
Packit 1fb8d4
		if (val2y1 > nHeight)
Packit 1fb8d4
			continue;
Packit 1fb8d4
Packit 1fb8d4
		for (x = roi->left; x < halfWidth + roi->left; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 val2x = (x * 2);
Packit 1fb8d4
			const UINT32 val2x1 = val2x + 1;
Packit 1fb8d4
			const INT32 up = pU[val2x] * 4;
Packit 1fb8d4
			const INT32 vp = pV[val2x] * 4;
Packit 1fb8d4
			INT32 u2020;
Packit 1fb8d4
			INT32 v2020;
Packit 1fb8d4
Packit 1fb8d4
			if (val2x1 > nWidth)
Packit 1fb8d4
				continue;
Packit 1fb8d4
Packit 1fb8d4
			u2020 = up - pU[val2x1] - pU1[val2x] - pU1[val2x1];
Packit 1fb8d4
			v2020 = vp - pV[val2x1] - pV1[val2x] - pV1[val2x1];
Packit 1fb8d4
			pU[val2x] = CLIP(u2020);
Packit 1fb8d4
			pV[val2x] = CLIP(v2020);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_ChromaV1ToYUV444(const BYTE* const pSrcRaw[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                          BYTE* pDstRaw[3], const UINT32 dstStep[3],
Packit Service 5a9772
                                          const RECTANGLE_16* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 mod = 16;
Packit 1fb8d4
	UINT32 uY = 0;
Packit 1fb8d4
	UINT32 vY = 0;
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	const UINT32 nWidth = roi->right - roi->left;
Packit 1fb8d4
	const UINT32 nHeight = roi->bottom - roi->top;
Packit 1fb8d4
	const UINT32 halfWidth = (nWidth) / 2;
Packit 1fb8d4
	const UINT32 halfHeight = (nHeight) / 2;
Packit 1fb8d4
	const UINT32 oddY = 1;
Packit 1fb8d4
	const UINT32 evenY = 0;
Packit 1fb8d4
	const UINT32 oddX = 1;
Packit 1fb8d4
	/* The auxilary frame is aligned to multiples of 16x16.
Packit 1fb8d4
	 * We need the padded height for B4 and B5 conversion. */
Packit 1fb8d4
	const UINT32 padHeigth = nHeight + 16 - nHeight % 16;
Packit Service 5a9772
	const BYTE* pSrc[3] = { pSrcRaw[0] + roi->top * srcStep[0] + roi->left,
Packit Service 5a9772
		                    pSrcRaw[1] + roi->top / 2 * srcStep[1] + roi->left / 2,
Packit Service 5a9772
		                    pSrcRaw[2] + roi->top / 2 * srcStep[2] + roi->left / 2 };
Packit Service 5a9772
	BYTE* pDst[3] = { pDstRaw[0] + roi->top * dstStep[0] + roi->left,
Packit Service 5a9772
		              pDstRaw[1] + roi->top * dstStep[1] + roi->left,
Packit Service 5a9772
		              pDstRaw[2] + roi->top * dstStep[2] + roi->left };
Packit 1fb8d4
Packit 1fb8d4
	/* The second half of U and V is a bit more tricky... */
Packit 1fb8d4
	/* B4 and B5 */
Packit 1fb8d4
	for (y = 0; y < padHeigth; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* Ya = pSrc[0] + srcStep[0] * y;
Packit 1fb8d4
		BYTE* pX;
Packit 1fb8d4
Packit 1fb8d4
		if ((y) % mod < (mod + 1) / 2)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 pos = (2 * uY++ + oddY);
Packit 1fb8d4
Packit 1fb8d4
			if (pos >= nHeight)
Packit 1fb8d4
				continue;
Packit 1fb8d4
Packit 1fb8d4
			pX = pDst[1] + dstStep[1] * pos;
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 pos = (2 * vY++ + oddY);
Packit 1fb8d4
Packit 1fb8d4
			if (pos >= nHeight)
Packit 1fb8d4
				continue;
Packit 1fb8d4
Packit 1fb8d4
			pX = pDst[2] + dstStep[2] * pos;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		memcpy(pX, Ya, nWidth);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* B6 and B7 */
Packit 1fb8d4
	for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const UINT32 val2y = (y * 2 + evenY);
Packit 1fb8d4
		const BYTE* Ua = pSrc[1] + srcStep[1] * y;
Packit 1fb8d4
		const BYTE* Va = pSrc[2] + srcStep[2] * y;
Packit 1fb8d4
		BYTE* pU = pDst[1] + dstStep[1] * val2y;
Packit 1fb8d4
		BYTE* pV = pDst[2] + dstStep[2] * val2y;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 val2x1 = (x * 2 + oddX);
Packit 1fb8d4
			pU[val2x1] = Ua[x];
Packit 1fb8d4
			pV[val2x1] = Va[x];
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* Filter */
Packit 1fb8d4
	return general_ChromaFilter(pDst, dstStep, roi);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_ChromaV2ToYUV444(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                          UINT32 nTotalWidth, UINT32 nTotalHeight, BYTE* pDst[3],
Packit Service 5a9772
                                          const UINT32 dstStep[3], const RECTANGLE_16* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	const UINT32 nWidth = roi->right - roi->left;
Packit 1fb8d4
	const UINT32 nHeight = roi->bottom - roi->top;
Packit 1fb8d4
	const UINT32 halfWidth = (nWidth + 1) / 2;
Packit 1fb8d4
	const UINT32 halfHeight = (nHeight + 1) / 2;
Packit 1fb8d4
	const UINT32 quaterWidth = (nWidth + 3) / 4;
Packit 1fb8d4
Packit 1fb8d4
	/* B4 and B5: odd UV values for width/2, height */
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const UINT32 yTop = y + roi->top;
Packit 1fb8d4
		const BYTE* pYaU = pSrc[0] + srcStep[0] * yTop + roi->left / 2;
Packit 1fb8d4
		const BYTE* pYaV = pYaU + nTotalWidth / 2;
Packit 1fb8d4
		BYTE* pU = pDst[1] + dstStep[1] * yTop + roi->left;
Packit 1fb8d4
		BYTE* pV = pDst[2] + dstStep[2] * yTop + roi->left;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 odd = 2 * x + 1;
Packit 1fb8d4
			pU[odd] = *pYaU++;
Packit 1fb8d4
			pV[odd] = *pYaV++;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* B6 - B9 */
Packit 1fb8d4
	for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pUaU = pSrc[1] + srcStep[1] * (y + roi->top / 2) + roi->left / 4;
Packit 1fb8d4
		const BYTE* pUaV = pUaU + nTotalWidth / 4;
Packit 1fb8d4
		const BYTE* pVaU = pSrc[2] + srcStep[2] * (y + roi->top / 2) + roi->left / 4;
Packit 1fb8d4
		const BYTE* pVaV = pVaU + nTotalWidth / 4;
Packit 1fb8d4
		BYTE* pU = pDst[1] + dstStep[1] * (2 * y + 1 + roi->top) + roi->left;
Packit 1fb8d4
		BYTE* pV = pDst[2] + dstStep[2] * (2 * y + 1 + roi->top) + roi->left;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < quaterWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			pU[4 * x + 0] = *pUaU++;
Packit 1fb8d4
			pV[4 * x + 0] = *pUaV++;
Packit 1fb8d4
			pU[4 * x + 2] = *pVaU++;
Packit 1fb8d4
			pV[4 * x + 2] = *pVaV++;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return general_ChromaFilter(pDst, dstStep, roi);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_YUV420CombineToYUV444(avc444_frame_type type, const BYTE* const pSrc[3],
Packit Service 5a9772
                                               const UINT32 srcStep[3], UINT32 nWidth,
Packit Service 5a9772
                                               UINT32 nHeight, BYTE* pDst[3],
Packit Service 5a9772
                                               const UINT32 dstStep[3], const RECTANGLE_16* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!pSrc || !pSrc[0] || !pSrc[1] || !pSrc[2])
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	if (!pDst || !pDst[0] || !pDst[1] || !pDst[2])
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	if (!roi)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	switch (type)
Packit 1fb8d4
	{
Packit 1fb8d4
		case AVC444_LUMA:
Packit 1fb8d4
			return general_LumaToYUV444(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		case AVC444_CHROMAv1:
Packit 1fb8d4
			return general_ChromaV1ToYUV444(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		case AVC444_CHROMAv2:
Packit 1fb8d4
			return general_ChromaV2ToYUV444(pSrc, srcStep, nWidth, nHeight, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_YUV444SplitToYUV420(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                             BYTE* pMainDst[3], const UINT32 dstMainStep[3],
Packit Service 5a9772
                                             BYTE* pAuxDst[3], const UINT32 dstAuxStep[3],
Packit Service 5a9772
                                             const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y, uY = 0, vY = 0;
Packit 1fb8d4
	UINT32 halfWidth, halfHeight;
Packit 1fb8d4
	/* The auxilary frame is aligned to multiples of 16x16.
Packit 1fb8d4
	 * We need the padded height for B4 and B5 conversion. */
Packit 1fb8d4
	const UINT32 padHeigth = roi->height + 16 - roi->height % 16;
Packit 1fb8d4
	halfWidth = (roi->width + 1) / 2;
Packit 1fb8d4
	halfHeight = (roi->height + 1) / 2;
Packit 1fb8d4
Packit 1fb8d4
	/* B1 */
Packit 1fb8d4
	for (y = 0; y < roi->height; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pSrcY = pSrc[0] + y * srcStep[0];
Packit 1fb8d4
		BYTE* pY = pMainDst[0] + y * dstMainStep[0];
Packit 1fb8d4
		memcpy(pY, pSrcY, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* B2 and B3 */
Packit 1fb8d4
	for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pSrcU = pSrc[1] + 2 * y * srcStep[1];
Packit 1fb8d4
		const BYTE* pSrcV = pSrc[2] + 2 * y * srcStep[2];
Packit 1fb8d4
		const BYTE* pSrcU1 = pSrc[1] + (2 * y + 1) * srcStep[1];
Packit 1fb8d4
		const BYTE* pSrcV1 = pSrc[2] + (2 * y + 1) * srcStep[2];
Packit 1fb8d4
		BYTE* pU = pMainDst[1] + y * dstMainStep[1];
Packit 1fb8d4
		BYTE* pV = pMainDst[2] + y * dstMainStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* Filter */
Packit Service 5a9772
			const INT32 u = pSrcU[2 * x] + pSrcU[2 * x + 1] + pSrcU1[2 * x] + pSrcU1[2 * x + 1];
Packit Service 5a9772
			const INT32 v = pSrcV[2 * x] + pSrcV[2 * x + 1] + pSrcV1[2 * x] + pSrcV1[2 * x + 1];
Packit 1fb8d4
			pU[x] = CLIP(u / 4L);
Packit 1fb8d4
			pV[x] = CLIP(v / 4L);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* B4 and B5 */
Packit 1fb8d4
	for (y = 0; y < padHeigth; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		BYTE* pY = pAuxDst[0] + y * dstAuxStep[0];
Packit 1fb8d4
Packit 1fb8d4
		if (y % 16 < 8)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 pos = (2 * uY++ + 1);
Packit 1fb8d4
			const BYTE* pSrcU = pSrc[1] + pos * srcStep[1];
Packit 1fb8d4
Packit 1fb8d4
			if (pos >= roi->height)
Packit 1fb8d4
				continue;
Packit 1fb8d4
Packit 1fb8d4
			memcpy(pY, pSrcU, roi->width);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 pos = (2 * vY++ + 1);
Packit 1fb8d4
			const BYTE* pSrcV = pSrc[2] + pos * srcStep[2];
Packit 1fb8d4
Packit 1fb8d4
			if (pos >= roi->height)
Packit 1fb8d4
				continue;
Packit 1fb8d4
Packit 1fb8d4
			memcpy(pY, pSrcV, roi->width);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	/* B6 and B7 */
Packit 1fb8d4
	for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pSrcU = pSrc[1] + 2 * y * srcStep[1];
Packit 1fb8d4
		const BYTE* pSrcV = pSrc[2] + 2 * y * srcStep[2];
Packit 1fb8d4
		BYTE* pU = pAuxDst[1] + y * dstAuxStep[1];
Packit 1fb8d4
		BYTE* pV = pAuxDst[2] + y * dstAuxStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			pU[x] = pSrcU[2 * x + 1];
Packit 1fb8d4
			pV[x] = pSrcV[2 * x + 1];
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_YUV444ToRGB_8u_P3AC4R_general(const BYTE* const pSrc[3],
Packit Service 5a9772
                                                       const UINT32 srcStep[3], BYTE* pDst,
Packit Service 5a9772
                                                       UINT32 dstStep, UINT32 DstFormat,
Packit Service 5a9772
                                                       const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	UINT32 nWidth, nHeight;
Packit 1fb8d4
	const DWORD formatSize = GetBytesPerPixel(DstFormat);
Packit 1fb8d4
	fkt_writePixel writePixel = getPixelWriteFunction(DstFormat);
Packit 1fb8d4
	nWidth = roi->width;
Packit 1fb8d4
	nHeight = roi->height;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pY = pSrc[0] + y * srcStep[0];
Packit 1fb8d4
		const BYTE* pU = pSrc[1] + y * srcStep[1];
Packit 1fb8d4
		const BYTE* pV = pSrc[2] + y * srcStep[2];
Packit 1fb8d4
		BYTE* pRGB = pDst + y * dstStep;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < nWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Y = pY[x];
Packit 1fb8d4
			const BYTE U = pU[x];
Packit 1fb8d4
			const BYTE V = pV[x];
Packit 1fb8d4
			const BYTE r = YUV2R(Y, U, V);
Packit 1fb8d4
			const BYTE g = YUV2G(Y, U, V);
Packit 1fb8d4
			const BYTE b = YUV2B(Y, U, V);
Packit 1fb8d4
			pRGB = (*writePixel)(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_YUV444ToRGB_8u_P3AC4R_BGRX(const BYTE* const pSrc[3],
Packit Service 5a9772
                                                    const UINT32 srcStep[3], BYTE* pDst,
Packit Service 5a9772
                                                    UINT32 dstStep, UINT32 DstFormat,
Packit Service 5a9772
                                                    const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	UINT32 nWidth, nHeight;
Packit 1fb8d4
	const DWORD formatSize = GetBytesPerPixel(DstFormat);
Packit 1fb8d4
	nWidth = roi->width;
Packit 1fb8d4
	nHeight = roi->height;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pY = pSrc[0] + y * srcStep[0];
Packit 1fb8d4
		const BYTE* pU = pSrc[1] + y * srcStep[1];
Packit 1fb8d4
		const BYTE* pV = pSrc[2] + y * srcStep[2];
Packit 1fb8d4
		BYTE* pRGB = pDst + y * dstStep;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < nWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Y = pY[x];
Packit 1fb8d4
			const BYTE U = pU[x];
Packit 1fb8d4
			const BYTE V = pV[x];
Packit 1fb8d4
			const BYTE r = YUV2R(Y, U, V);
Packit 1fb8d4
			const BYTE g = YUV2G(Y, U, V);
Packit 1fb8d4
			const BYTE b = YUV2B(Y, U, V);
Packit 1fb8d4
			pRGB = writePixelBGRX(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_YUV444ToRGB_8u_P3AC4R(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                               BYTE* pDst, UINT32 dstStep, UINT32 DstFormat,
Packit Service 5a9772
                                               const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (DstFormat)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PIXEL_FORMAT_BGRA32:
Packit 1fb8d4
		case PIXEL_FORMAT_BGRX32:
Packit 1fb8d4
			return general_YUV444ToRGB_8u_P3AC4R_BGRX(pSrc, srcStep, pDst, dstStep, DstFormat, roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit Service 5a9772
			return general_YUV444ToRGB_8u_P3AC4R_general(pSrc, srcStep, pDst, dstStep, DstFormat,
Packit Service 5a9772
			                                             roi);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
/**
Packit Service 5a9772
 * | R |   ( | 256     0    403 | |    Y    | )
Packit Service 5a9772
 * | G | = ( | 256   -48   -120 | | U - 128 | ) >> 8
Packit Service 5a9772
 * | B |   ( | 256   475      0 | | V - 128 | )
Packit 1fb8d4
 */
Packit Service 5a9772
static pstatus_t general_YUV420ToRGB_8u_P3AC4R(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                               BYTE* pDst, UINT32 dstStep, UINT32 DstFormat,
Packit Service 5a9772
                                               const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	UINT32 dstPad;
Packit 1fb8d4
	UINT32 srcPad[3];
Packit 1fb8d4
	BYTE Y, U, V;
Packit 1fb8d4
	UINT32 halfWidth;
Packit 1fb8d4
	UINT32 halfHeight;
Packit 1fb8d4
	const BYTE* pY;
Packit 1fb8d4
	const BYTE* pU;
Packit 1fb8d4
	const BYTE* pV;
Packit 1fb8d4
	BYTE* pRGB = pDst;
Packit 1fb8d4
	UINT32 nWidth, nHeight;
Packit 1fb8d4
	UINT32 lastRow, lastCol;
Packit 1fb8d4
	const DWORD formatSize = GetBytesPerPixel(DstFormat);
Packit 1fb8d4
	fkt_writePixel writePixel = getPixelWriteFunction(DstFormat);
Packit 1fb8d4
	pY = pSrc[0];
Packit 1fb8d4
	pU = pSrc[1];
Packit 1fb8d4
	pV = pSrc[2];
Packit 1fb8d4
	lastCol = roi->width & 0x01;
Packit 1fb8d4
	lastRow = roi->height & 0x01;
Packit 1fb8d4
	nWidth = (roi->width + 1) & ~0x0001;
Packit 1fb8d4
	nHeight = (roi->height + 1) & ~0x0001;
Packit 1fb8d4
	halfWidth = nWidth / 2;
Packit 1fb8d4
	halfHeight = nHeight / 2;
Packit 1fb8d4
	srcPad[0] = (srcStep[0] - nWidth);
Packit 1fb8d4
	srcPad[1] = (srcStep[1] - halfWidth);
Packit 1fb8d4
	srcPad[2] = (srcStep[2] - halfWidth);
Packit 1fb8d4
	dstPad = (dstStep - (nWidth * 4));
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < halfHeight;)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (++y == halfHeight)
Packit 1fb8d4
			lastRow <<= 1;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth;)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r;
Packit 1fb8d4
			BYTE g;
Packit 1fb8d4
			BYTE b;
Packit 1fb8d4
Packit 1fb8d4
			if (++x == halfWidth)
Packit 1fb8d4
				lastCol <<= 1;
Packit 1fb8d4
Packit 1fb8d4
			U = *pU++;
Packit 1fb8d4
			V = *pV++;
Packit 1fb8d4
			/* 1st pixel */
Packit 1fb8d4
			Y = *pY++;
Packit 1fb8d4
			r = YUV2R(Y, U, V);
Packit 1fb8d4
			g = YUV2G(Y, U, V);
Packit 1fb8d4
			b = YUV2B(Y, U, V);
Packit 1fb8d4
			pRGB = (*writePixel)(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
Packit 1fb8d4
			/* 2nd pixel */
Packit 1fb8d4
			if (!(lastCol & 0x02))
Packit 1fb8d4
			{
Packit 1fb8d4
				Y = *pY++;
Packit 1fb8d4
				r = YUV2R(Y, U, V);
Packit 1fb8d4
				g = YUV2G(Y, U, V);
Packit 1fb8d4
				b = YUV2B(Y, U, V);
Packit 1fb8d4
				pRGB = (*writePixel)(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
			}
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				pY++;
Packit 1fb8d4
				pRGB += formatSize;
Packit 1fb8d4
				lastCol >>= 1;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pY += srcPad[0];
Packit 1fb8d4
		pU -= halfWidth;
Packit 1fb8d4
		pV -= halfWidth;
Packit 1fb8d4
		pRGB += dstPad;
Packit 1fb8d4
Packit 1fb8d4
		if (lastRow & 0x02)
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < halfWidth;)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r;
Packit 1fb8d4
			BYTE g;
Packit 1fb8d4
			BYTE b;
Packit 1fb8d4
Packit 1fb8d4
			if (++x == halfWidth)
Packit 1fb8d4
				lastCol <<= 1;
Packit 1fb8d4
Packit 1fb8d4
			U = *pU++;
Packit 1fb8d4
			V = *pV++;
Packit 1fb8d4
			/* 3rd pixel */
Packit 1fb8d4
			Y = *pY++;
Packit 1fb8d4
			r = YUV2R(Y, U, V);
Packit 1fb8d4
			g = YUV2G(Y, U, V);
Packit 1fb8d4
			b = YUV2B(Y, U, V);
Packit 1fb8d4
			pRGB = (*writePixel)(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
Packit 1fb8d4
			/* 4th pixel */
Packit 1fb8d4
			if (!(lastCol & 0x02))
Packit 1fb8d4
			{
Packit 1fb8d4
				Y = *pY++;
Packit 1fb8d4
				r = YUV2R(Y, U, V);
Packit 1fb8d4
				g = YUV2G(Y, U, V);
Packit 1fb8d4
				b = YUV2B(Y, U, V);
Packit 1fb8d4
				pRGB = (*writePixel)(pRGB, formatSize, DstFormat, r, g, b, 0xFF);
Packit 1fb8d4
			}
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				pY++;
Packit 1fb8d4
				pRGB += formatSize;
Packit 1fb8d4
				lastCol >>= 1;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pY += srcPad[0];
Packit 1fb8d4
		pU += srcPad[1];
Packit 1fb8d4
		pV += srcPad[2];
Packit 1fb8d4
		pRGB += dstPad;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * | Y |    ( |  54   183     18 | | R | )        |  0  |
Packit 1fb8d4
 * | U | =  ( | -29   -99    128 | | G | ) >> 8 + | 128 |
Packit 1fb8d4
 * | V |    ( | 128  -116    -12 | | B | )        | 128 |
Packit 1fb8d4
 */
Packit 1fb8d4
static INLINE BYTE RGB2Y(BYTE R, BYTE G, BYTE B)
Packit 1fb8d4
{
Packit 1fb8d4
	return (54 * R + 183 * G + 18 * B) >> 8;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BYTE RGB2U(BYTE R, BYTE G, BYTE B)
Packit 1fb8d4
{
Packit 1fb8d4
	return ((-29 * R - 99 * G + 128 * B) >> 8) + 128;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BYTE RGB2V(INT32 R, INT32 G, INT32 B)
Packit 1fb8d4
{
Packit 1fb8d4
	return ((128L * R - 116 * G - 12 * B) >> 8) + 128;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_RGBToYUV444_8u_P3AC4R(const BYTE* pSrc, UINT32 SrcFormat,
Packit Service 5a9772
                                               const UINT32 srcStep, BYTE* pDst[3],
Packit Service 5a9772
                                               UINT32 dstStep[3], const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 bpp = GetBytesPerPixel(SrcFormat);
Packit 1fb8d4
	UINT32 x, y;
Packit 1fb8d4
	UINT32 nWidth, nHeight;
Packit 1fb8d4
	nWidth = roi->width;
Packit 1fb8d4
	nHeight = roi->height;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* pRGB = pSrc + y * srcStep;
Packit 1fb8d4
		BYTE* pY = pDst[0] + y * dstStep[0];
Packit 1fb8d4
		BYTE* pU = pDst[1] + y * dstStep[1];
Packit 1fb8d4
		BYTE* pV = pDst[2] + y * dstStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < nWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE B, G, R;
Packit 1fb8d4
			const UINT32 color = ReadColor(&pRGB[x * bpp], SrcFormat);
Packit 1fb8d4
			SplitColor(color, SrcFormat, &R, &G, &B, NULL, NULL);
Packit 1fb8d4
			pY[x] = RGB2Y(R, G, B);
Packit 1fb8d4
			pU[x] = RGB2U(R, G, B);
Packit 1fb8d4
			pV[x] = RGB2V(R, G, B);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToYUV420_BGRX(const BYTE* pSrc, UINT32 srcStep, BYTE* pDst[3],
Packit Service 5a9772
                                                 UINT32 dstStep[3], const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y, i;
Packit 1fb8d4
	size_t x1 = 0, x2 = 4, x3 = srcStep, x4 = srcStep + 4;
Packit 1fb8d4
	size_t y1 = 0, y2 = 1, y3 = dstStep[0], y4 = dstStep[0] + 1;
Packit 1fb8d4
	UINT32 max_x = roi->width - 1;
Packit 1fb8d4
	UINT32 max_y = roi->height - 1;
Packit 1fb8d4
Packit 1fb8d4
	for (y = i = 0; y < roi->height; y += 2, i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* src = pSrc + y * srcStep;
Packit 1fb8d4
		BYTE* ydst = pDst[0] + y * dstStep[0];
Packit 1fb8d4
		BYTE* udst = pDst[1] + i * dstStep[1];
Packit 1fb8d4
		BYTE* vdst = pDst[2] + i * dstStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < roi->width; x += 2)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE R, G, B;
Packit 1fb8d4
			INT32 Ra, Ga, Ba;
Packit 1fb8d4
			/* row 1, pixel 1 */
Packit 1fb8d4
			Ba = B = *(src + x1 + 0);
Packit 1fb8d4
			Ga = G = *(src + x1 + 1);
Packit 1fb8d4
			Ra = R = *(src + x1 + 2);
Packit 1fb8d4
			ydst[y1] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
			if (x < max_x)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 1, pixel 2 */
Packit 1fb8d4
				Ba += B = *(src + x2 + 0);
Packit 1fb8d4
				Ga += G = *(src + x2 + 1);
Packit 1fb8d4
				Ra += R = *(src + x2 + 2);
Packit 1fb8d4
				ydst[y2] = RGB2Y(R, G, B);
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (y < max_y)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 2, pixel 1 */
Packit 1fb8d4
				Ba += B = *(src + x3 + 0);
Packit 1fb8d4
				Ga += G = *(src + x3 + 1);
Packit 1fb8d4
				Ra += R = *(src + x3 + 2);
Packit 1fb8d4
				ydst[y3] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
				if (x < max_x)
Packit 1fb8d4
				{
Packit 1fb8d4
					/* row 2, pixel 2 */
Packit 1fb8d4
					Ba += B = *(src + x4 + 0);
Packit 1fb8d4
					Ga += G = *(src + x4 + 1);
Packit 1fb8d4
					Ra += R = *(src + x4 + 2);
Packit 1fb8d4
					ydst[y4] = RGB2Y(R, G, B);
Packit 1fb8d4
				}
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			Ba >>= 2;
Packit 1fb8d4
			Ga >>= 2;
Packit 1fb8d4
			Ra >>= 2;
Packit 1fb8d4
			*udst++ = RGB2U(Ra, Ga, Ba);
Packit 1fb8d4
			*vdst++ = RGB2V(Ra, Ga, Ba);
Packit 1fb8d4
			ydst += 2;
Packit 1fb8d4
			src += 8;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToYUV420_RGBX(const BYTE* pSrc, UINT32 srcStep, BYTE* pDst[3],
Packit Service 5a9772
                                                 UINT32 dstStep[3], const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x, y, i;
Packit 1fb8d4
	size_t x1 = 0, x2 = 4, x3 = srcStep, x4 = srcStep + 4;
Packit 1fb8d4
	size_t y1 = 0, y2 = 1, y3 = dstStep[0], y4 = dstStep[0] + 1;
Packit 1fb8d4
	UINT32 max_x = roi->width - 1;
Packit 1fb8d4
	UINT32 max_y = roi->height - 1;
Packit 1fb8d4
Packit 1fb8d4
	for (y = i = 0; y < roi->height; y += 2, i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* src = pSrc + y * srcStep;
Packit 1fb8d4
		BYTE* ydst = pDst[0] + y * dstStep[0];
Packit 1fb8d4
		BYTE* udst = pDst[1] + i * dstStep[1];
Packit 1fb8d4
		BYTE* vdst = pDst[2] + i * dstStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < roi->width; x += 2)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE R, G, B;
Packit 1fb8d4
			INT32 Ra, Ga, Ba;
Packit 1fb8d4
			/* row 1, pixel 1 */
Packit 1fb8d4
			Ra = R = *(src + x1 + 0);
Packit 1fb8d4
			Ga = G = *(src + x1 + 1);
Packit 1fb8d4
			Ba = B = *(src + x1 + 2);
Packit 1fb8d4
			ydst[y1] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
			if (x < max_x)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 1, pixel 2 */
Packit 1fb8d4
				Ra += R = *(src + x2 + 0);
Packit 1fb8d4
				Ga += G = *(src + x2 + 1);
Packit 1fb8d4
				Ba += B = *(src + x2 + 2);
Packit 1fb8d4
				ydst[y2] = RGB2Y(R, G, B);
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (y < max_y)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 2, pixel 1 */
Packit 1fb8d4
				Ra += R = *(src + x3 + 0);
Packit 1fb8d4
				Ga += G = *(src + x3 + 1);
Packit 1fb8d4
				Ba += B = *(src + x3 + 2);
Packit 1fb8d4
				ydst[y3] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
				if (x < max_x)
Packit 1fb8d4
				{
Packit 1fb8d4
					/* row 2, pixel 2 */
Packit 1fb8d4
					Ra += R = *(src + x4 + 0);
Packit 1fb8d4
					Ga += G = *(src + x4 + 1);
Packit 1fb8d4
					Ba += B = *(src + x4 + 2);
Packit 1fb8d4
					ydst[y4] = RGB2Y(R, G, B);
Packit 1fb8d4
				}
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			Ba >>= 2;
Packit 1fb8d4
			Ga >>= 2;
Packit 1fb8d4
			Ra >>= 2;
Packit 1fb8d4
			*udst++ = RGB2U(Ra, Ga, Ba);
Packit 1fb8d4
			*vdst++ = RGB2V(Ra, Ga, Ba);
Packit 1fb8d4
			ydst += 2;
Packit 1fb8d4
			src += 8;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToYUV420_ANY(const BYTE* pSrc, UINT32 srcFormat, UINT32 srcStep,
Packit Service 5a9772
                                                BYTE* pDst[3], UINT32 dstStep[3],
Packit Service 5a9772
                                                const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 bpp = GetBytesPerPixel(srcFormat);
Packit 1fb8d4
	UINT32 x, y, i;
Packit 1fb8d4
	size_t x1 = 0, x2 = bpp, x3 = srcStep, x4 = srcStep + bpp;
Packit 1fb8d4
	size_t y1 = 0, y2 = 1, y3 = dstStep[0], y4 = dstStep[0] + 1;
Packit 1fb8d4
	UINT32 max_x = roi->width - 1;
Packit 1fb8d4
	UINT32 max_y = roi->height - 1;
Packit 1fb8d4
Packit 1fb8d4
	for (y = i = 0; y < roi->height; y += 2, i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* src = pSrc + y * srcStep;
Packit 1fb8d4
		BYTE* ydst = pDst[0] + y * dstStep[0];
Packit 1fb8d4
		BYTE* udst = pDst[1] + i * dstStep[1];
Packit 1fb8d4
		BYTE* vdst = pDst[2] + i * dstStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < roi->width; x += 2)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE R, G, B;
Packit 1fb8d4
			INT32 Ra, Ga, Ba;
Packit 1fb8d4
			UINT32 color;
Packit 1fb8d4
			/* row 1, pixel 1 */
Packit 1fb8d4
			color = ReadColor(src + x1, srcFormat);
Packit 1fb8d4
			SplitColor(color, srcFormat, &R, &G, &B, NULL, NULL);
Packit 1fb8d4
			Ra = R;
Packit 1fb8d4
			Ga = G;
Packit 1fb8d4
			Ba = B;
Packit 1fb8d4
			ydst[y1] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
			if (x < max_x)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 1, pixel 2 */
Packit 1fb8d4
				color = ReadColor(src + x2, srcFormat);
Packit 1fb8d4
				SplitColor(color, srcFormat, &R, &G, &B, NULL, NULL);
Packit 1fb8d4
				Ra += R;
Packit 1fb8d4
				Ga += G;
Packit 1fb8d4
				Ba += B;
Packit 1fb8d4
				ydst[y2] = RGB2Y(R, G, B);
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (y < max_y)
Packit 1fb8d4
			{
Packit 1fb8d4
				/* row 2, pixel 1 */
Packit 1fb8d4
				color = ReadColor(src + x3, srcFormat);
Packit 1fb8d4
				SplitColor(color, srcFormat, &R, &G, &B, NULL, NULL);
Packit 1fb8d4
				Ra += R;
Packit 1fb8d4
				Ga += G;
Packit 1fb8d4
				Ba += B;
Packit 1fb8d4
				ydst[y3] = RGB2Y(R, G, B);
Packit 1fb8d4
Packit 1fb8d4
				if (x < max_x)
Packit 1fb8d4
				{
Packit 1fb8d4
					/* row 2, pixel 2 */
Packit 1fb8d4
					color = ReadColor(src + x4, srcFormat);
Packit 1fb8d4
					SplitColor(color, srcFormat, &R, &G, &B, NULL, NULL);
Packit 1fb8d4
					Ra += R;
Packit 1fb8d4
					Ga += G;
Packit 1fb8d4
					Ba += B;
Packit 1fb8d4
					ydst[y4] = RGB2Y(R, G, B);
Packit 1fb8d4
				}
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			Ra >>= 2;
Packit 1fb8d4
			Ga >>= 2;
Packit 1fb8d4
			Ba >>= 2;
Packit 1fb8d4
			*udst++ = RGB2U(Ra, Ga, Ba);
Packit 1fb8d4
			*vdst++ = RGB2V(Ra, Ga, Ba);
Packit 1fb8d4
			ydst += 2;
Packit 1fb8d4
			src += 2 * bpp;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t general_RGBToYUV420_8u_P3AC4R(const BYTE* pSrc, UINT32 srcFormat, UINT32 srcStep,
Packit Service 5a9772
                                               BYTE* pDst[3], UINT32 dstStep[3],
Packit Service 5a9772
                                               const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (srcFormat)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PIXEL_FORMAT_BGRA32:
Packit 1fb8d4
		case PIXEL_FORMAT_BGRX32:
Packit 1fb8d4
			return general_RGBToYUV420_BGRX(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_RGBA32:
Packit 1fb8d4
		case PIXEL_FORMAT_RGBX32:
Packit 1fb8d4
			return general_RGBToYUV420_RGBX(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return general_RGBToYUV420_ANY(pSrc, srcFormat, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE void general_RGBToAVC444YUV_BGRX_DOUBLE_ROW(const BYTE* srcEven, const BYTE* srcOdd,
Packit Service 5a9772
                                                          BYTE* b1Even, BYTE* b1Odd, BYTE* b2,
Packit Service 5a9772
                                                          BYTE* b3, BYTE* b4, BYTE* b5, BYTE* b6,
Packit Service 5a9772
                                                          BYTE* b7, UINT32 width)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < width; x += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL lastX = (x + 1) >= width;
Packit 1fb8d4
		BYTE Y1e, Y2e, U1e, V1e, U2e, V2e;
Packit 1fb8d4
		BYTE Y1o, Y2o, U1o, V1o, U2o, V2o;
Packit 1fb8d4
		/* Read 4 pixels, 2 from even, 2 from odd lines */
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Y1e = Y2e = Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1e = U2e = U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1e = V2e = V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Y2e = RGB2Y(r, g, b);
Packit 1fb8d4
			U2e = RGB2U(r, g, b);
Packit 1fb8d4
			V2e = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd && !lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* We have 4 Y pixels, so store them. */
Packit 1fb8d4
		*b1Even++ = Y1e;
Packit 1fb8d4
		*b1Even++ = Y2e;
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b1Odd++ = Y1o;
Packit 1fb8d4
			*b1Odd++ = Y2o;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* 2x 2y pixel in luma UV plane use averaging
Packit 1fb8d4
		 */
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Uavg = ((UINT16)U1e + (UINT16)U2e + (UINT16)U1o + (UINT16)U2o) / 4;
Packit 1fb8d4
			const BYTE Vavg = ((UINT16)V1e + (UINT16)V2e + (UINT16)V1o + (UINT16)V2o) / 4;
Packit 1fb8d4
			*b2++ = Uavg;
Packit 1fb8d4
			*b3++ = Vavg;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x, 2y+1 */
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b4++ = U1o;
Packit 1fb8d4
			*b5++ = V1o;
Packit 1fb8d4
Packit 1fb8d4
			if (!lastX)
Packit 1fb8d4
			{
Packit 1fb8d4
				*b4++ = U2o;
Packit 1fb8d4
				*b5++ = V2o;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x+1, 2y */
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b6++ = U2e;
Packit 1fb8d4
			*b7++ = V2e;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUV_BGRX(const BYTE* pSrc, UINT32 srcStep,
Packit Service 5a9772
                                                    BYTE* pDst1[3], const UINT32 dst1Step[3],
Packit Service 5a9772
                                                    BYTE* pDst2[3], const UINT32 dst2Step[3],
Packit Service 5a9772
                                                    const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	/**
Packit 1fb8d4
	 * Note:
Packit 1fb8d4
	 * Read information in function general_RGBToAVC444YUV_ANY below !
Packit 1fb8d4
	 */
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
	const BYTE* pMaxSrc = pSrc + (roi->height - 1) * srcStep;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < roi->height; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL last = (y >= (roi->height - 1));
Packit 1fb8d4
		const BYTE* srcEven = y < roi->height ? pSrc + y * srcStep : pMaxSrc;
Packit 1fb8d4
		const BYTE* srcOdd = !last ? pSrc + (y + 1) * srcStep : pMaxSrc;
Packit 1fb8d4
		const UINT32 i = y >> 1;
Packit 1fb8d4
		const UINT32 n = (i & ~7) + i;
Packit 1fb8d4
		BYTE* b1Even = pDst1[0] + y * dst1Step[0];
Packit 1fb8d4
		BYTE* b1Odd = !last ? (b1Even + dst1Step[0]) : NULL;
Packit 1fb8d4
		BYTE* b2 = pDst1[1] + (y / 2) * dst1Step[1];
Packit 1fb8d4
		BYTE* b3 = pDst1[2] + (y / 2) * dst1Step[2];
Packit 1fb8d4
		BYTE* b4 = pDst2[0] + dst2Step[0] * n;
Packit 1fb8d4
		BYTE* b5 = b4 + 8 * dst2Step[0];
Packit 1fb8d4
		BYTE* b6 = pDst2[1] + (y / 2) * dst2Step[1];
Packit 1fb8d4
		BYTE* b7 = pDst2[2] + (y / 2) * dst2Step[2];
Packit Service 5a9772
		general_RGBToAVC444YUV_BGRX_DOUBLE_ROW(srcEven, srcOdd, b1Even, b1Odd, b2, b3, b4, b5, b6,
Packit Service 5a9772
		                                       b7, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE void general_RGBToAVC444YUV_RGBX_DOUBLE_ROW(const BYTE* srcEven, const BYTE* srcOdd,
Packit Service 5a9772
                                                          BYTE* b1Even, BYTE* b1Odd, BYTE* b2,
Packit Service 5a9772
                                                          BYTE* b3, BYTE* b4, BYTE* b5, BYTE* b6,
Packit Service 5a9772
                                                          BYTE* b7, UINT32 width)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < width; x += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL lastX = (x + 1) >= width;
Packit 1fb8d4
		BYTE Y1e, Y2e, U1e, V1e, U2e, V2e;
Packit 1fb8d4
		BYTE Y1o, Y2o, U1o, V1o, U2o, V2o;
Packit 1fb8d4
		/* Read 4 pixels, 2 from even, 2 from odd lines */
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Y1e = Y2e = Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1e = U2e = U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1e = V2e = V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Y2e = RGB2Y(r, g, b);
Packit 1fb8d4
			U2e = RGB2U(r, g, b);
Packit 1fb8d4
			V2e = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd && !lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* We have 4 Y pixels, so store them. */
Packit 1fb8d4
		*b1Even++ = Y1e;
Packit 1fb8d4
		*b1Even++ = Y2e;
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b1Odd++ = Y1o;
Packit 1fb8d4
			*b1Odd++ = Y2o;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* 2x 2y pixel in luma UV plane use averaging
Packit 1fb8d4
		 */
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Uavg = ((UINT16)U1e + (UINT16)U2e + (UINT16)U1o + (UINT16)U2o) / 4;
Packit 1fb8d4
			const BYTE Vavg = ((UINT16)V1e + (UINT16)V2e + (UINT16)V1o + (UINT16)V2o) / 4;
Packit 1fb8d4
			*b2++ = Uavg;
Packit 1fb8d4
			*b3++ = Vavg;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x, 2y+1 */
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b4++ = U1o;
Packit 1fb8d4
			*b5++ = V1o;
Packit 1fb8d4
Packit 1fb8d4
			if (!lastX)
Packit 1fb8d4
			{
Packit 1fb8d4
				*b4++ = U2o;
Packit 1fb8d4
				*b5++ = V2o;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x+1, 2y */
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b6++ = U2e;
Packit 1fb8d4
			*b7++ = V2e;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUV_RGBX(const BYTE* pSrc, UINT32 srcStep,
Packit Service 5a9772
                                                    BYTE* pDst1[3], const UINT32 dst1Step[3],
Packit Service 5a9772
                                                    BYTE* pDst2[3], const UINT32 dst2Step[3],
Packit Service 5a9772
                                                    const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	/**
Packit 1fb8d4
	 * Note:
Packit 1fb8d4
	 * Read information in function general_RGBToAVC444YUV_ANY below !
Packit 1fb8d4
	 */
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
	const BYTE* pMaxSrc = pSrc + (roi->height - 1) * srcStep;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < roi->height; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL last = (y >= (roi->height - 1));
Packit 1fb8d4
		const BYTE* srcEven = y < roi->height ? pSrc + y * srcStep : pMaxSrc;
Packit 1fb8d4
		const BYTE* srcOdd = !last ? pSrc + (y + 1) * srcStep : pMaxSrc;
Packit 1fb8d4
		const UINT32 i = y >> 1;
Packit 1fb8d4
		const UINT32 n = (i & ~7) + i;
Packit 1fb8d4
		BYTE* b1Even = pDst1[0] + y * dst1Step[0];
Packit 1fb8d4
		BYTE* b1Odd = !last ? (b1Even + dst1Step[0]) : NULL;
Packit 1fb8d4
		BYTE* b2 = pDst1[1] + (y / 2) * dst1Step[1];
Packit 1fb8d4
		BYTE* b3 = pDst1[2] + (y / 2) * dst1Step[2];
Packit 1fb8d4
		BYTE* b4 = pDst2[0] + dst2Step[0] * n;
Packit 1fb8d4
		BYTE* b5 = b4 + 8 * dst2Step[0];
Packit 1fb8d4
		BYTE* b6 = pDst2[1] + (y / 2) * dst2Step[1];
Packit 1fb8d4
		BYTE* b7 = pDst2[2] + (y / 2) * dst2Step[2];
Packit Service 5a9772
		general_RGBToAVC444YUV_RGBX_DOUBLE_ROW(srcEven, srcOdd, b1Even, b1Odd, b2, b3, b4, b5, b6,
Packit Service 5a9772
		                                       b7, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE void general_RGBToAVC444YUV_ANY_DOUBLE_ROW(const BYTE* srcEven, const BYTE* srcOdd,
Packit Service 5a9772
                                                         UINT32 srcFormat, BYTE* b1Even,
Packit Service 5a9772
                                                         BYTE* b1Odd, BYTE* b2, BYTE* b3, BYTE* b4,
Packit Service 5a9772
                                                         BYTE* b5, BYTE* b6, BYTE* b7, UINT32 width)
Packit 1fb8d4
{
Packit 1fb8d4
	const UINT32 bpp = GetBytesPerPixel(srcFormat);
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < width; x += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL lastX = (x + 1) >= width;
Packit 1fb8d4
		BYTE Y1e, Y2e, U1e, V1e, U2e, V2e;
Packit 1fb8d4
		BYTE Y1o, Y2o, U1o, V1o, U2o, V2o;
Packit 1fb8d4
		/* Read 4 pixels, 2 from even, 2 from odd lines */
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r, g, b;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcEven, srcFormat);
Packit 1fb8d4
			srcEven += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Y1e = Y2e = Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1e = U2e = U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1e = V2e = V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r, g, b;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcEven, srcFormat);
Packit 1fb8d4
			srcEven += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Y2e = RGB2Y(r, g, b);
Packit 1fb8d4
			U2e = RGB2U(r, g, b);
Packit 1fb8d4
			V2e = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r, g, b;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcOdd, srcFormat);
Packit 1fb8d4
			srcOdd += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Y1o = Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U1o = U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V1o = V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd && !lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE r, g, b;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcOdd, srcFormat);
Packit 1fb8d4
			srcOdd += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Y2o = RGB2Y(r, g, b);
Packit 1fb8d4
			U2o = RGB2U(r, g, b);
Packit 1fb8d4
			V2o = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* We have 4 Y pixels, so store them. */
Packit 1fb8d4
		*b1Even++ = Y1e;
Packit 1fb8d4
		*b1Even++ = Y2e;
Packit 1fb8d4
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b1Odd++ = Y1o;
Packit 1fb8d4
			*b1Odd++ = Y2o;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* 2x 2y pixel in luma UV plane use averaging
Packit 1fb8d4
		 */
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Uavg = ((UINT16)U1e + (UINT16)U2e + (UINT16)U1o + (UINT16)U2o) / 4;
Packit 1fb8d4
			const BYTE Vavg = ((UINT16)V1e + (UINT16)V2e + (UINT16)V1o + (UINT16)V2o) / 4;
Packit 1fb8d4
			*b2++ = Uavg;
Packit 1fb8d4
			*b3++ = Vavg;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x, 2y+1 */
Packit 1fb8d4
		if (b1Odd)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b4++ = U1o;
Packit 1fb8d4
			*b5++ = V1o;
Packit 1fb8d4
Packit 1fb8d4
			if (!lastX)
Packit 1fb8d4
			{
Packit 1fb8d4
				*b4++ = U2o;
Packit 1fb8d4
				*b5++ = V2o;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* UV from 2x+1, 2y */
Packit 1fb8d4
		if (!lastX)
Packit 1fb8d4
		{
Packit 1fb8d4
			*b6++ = U2e;
Packit 1fb8d4
			*b7++ = V2e;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUV_ANY(const BYTE* pSrc, UINT32 srcFormat,
Packit Service 5a9772
                                                   UINT32 srcStep, BYTE* pDst1[3],
Packit Service 5a9772
                                                   const UINT32 dst1Step[3], BYTE* pDst2[3],
Packit Service 5a9772
                                                   const UINT32 dst2Step[3], const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	/**
Packit 1fb8d4
	 * Note: According to [MS-RDPEGFX 2.2.4.4 RFX_AVC420_BITMAP_STREAM] the
Packit 1fb8d4
	 * width and height of the MPEG-4 AVC/H.264 codec bitstream MUST be aligned
Packit 1fb8d4
	 * to a multiple of 16.
Packit 1fb8d4
	 * Hence the passed destination YUV420/CHROMA420 buffers must have been
Packit 1fb8d4
	 * allocated accordingly !!
Packit 1fb8d4
	 */
Packit 1fb8d4
	/**
Packit 1fb8d4
	 * [MS-RDPEGFX 3.3.8.3.2 YUV420p Stream Combination] defines the following "Bx areas":
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * YUV420 frame (main view):
Packit 1fb8d4
	 * B1:  From Y444 all pixels
Packit 1fb8d4
	 * B2:  From U444 all pixels in even rows with even columns
Packit 1fb8d4
	 * B3:  From V444 all pixels in even rows with even columns
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * Chroma420 frame (auxillary view):
Packit 1fb8d4
	 * B45: From U444 and V444 all pixels from all odd rows
Packit 1fb8d4
	 *      (The odd U444 and V444 rows must be interleaved in 8-line blocks in B45 !!!)
Packit 1fb8d4
	 * B6:  From U444 all pixels in even rows with odd columns
Packit 1fb8d4
	 * B7:  From V444 all pixels in even rows with odd columns
Packit 1fb8d4
	 *
Packit Service 5a9772
	 * Microsoft's horrible unclear description in MS-RDPEGFX translated to pseudo code looks like
Packit Service 5a9772
	 * this:
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * for (y = 0; y < fullHeight; y++)
Packit 1fb8d4
	 * {
Packit 1fb8d4
	 *     for (x = 0; x < fullWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B1[x,y] = Y444[x,y];
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *  }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	 * {
Packit 1fb8d4
	 *     for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B2[x,y] = U444[2 * x,     2 * y];
Packit 1fb8d4
	 *         B3[x,y] = V444[2 * x,     2 * y];
Packit 1fb8d4
	 *         B6[x,y] = U444[2 * x + 1, 2 * y];
Packit 1fb8d4
	 *     	   B7[x,y] = V444[2 * x + 1, 2 * y];
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *  }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	 * {
Packit 1fb8d4
	 *     yU  = (y / 8) * 16;   // identify first row of correct 8-line U block in B45
Packit 1fb8d4
	 *     yU += (y % 8);        // add offset rows in destination block
Packit 1fb8d4
	 *     yV  = yU + 8;         // the corresponding v line is always 8 rows ahead
Packit 1fb8d4
	 *
Packit 1fb8d4
	 *     for (x = 0; x < fullWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B45[x,yU] = U444[x, 2 * y + 1];
Packit 1fb8d4
	 *         B45[x,yV] = V444[x, 2 * y + 1];
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *  }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 */
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
	const BYTE* pMaxSrc = pSrc + (roi->height - 1) * srcStep;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < roi->height; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BOOL last = (y >= (roi->height - 1));
Packit 1fb8d4
		const BYTE* srcEven = y < roi->height ? pSrc + y * srcStep : pMaxSrc;
Packit 1fb8d4
		const BYTE* srcOdd = !last ? pSrc + (y + 1) * srcStep : pMaxSrc;
Packit 1fb8d4
		const UINT32 i = y >> 1;
Packit 1fb8d4
		const UINT32 n = (i & ~7) + i;
Packit 1fb8d4
		BYTE* b1Even = pDst1[0] + y * dst1Step[0];
Packit 1fb8d4
		BYTE* b1Odd = !last ? (b1Even + dst1Step[0]) : NULL;
Packit 1fb8d4
		BYTE* b2 = pDst1[1] + (y / 2) * dst1Step[1];
Packit 1fb8d4
		BYTE* b3 = pDst1[2] + (y / 2) * dst1Step[2];
Packit 1fb8d4
		BYTE* b4 = pDst2[0] + dst2Step[0] * n;
Packit 1fb8d4
		BYTE* b5 = b4 + 8 * dst2Step[0];
Packit 1fb8d4
		BYTE* b6 = pDst2[1] + (y / 2) * dst2Step[1];
Packit 1fb8d4
		BYTE* b7 = pDst2[2] + (y / 2) * dst2Step[2];
Packit Service 5a9772
		general_RGBToAVC444YUV_ANY_DOUBLE_ROW(srcEven, srcOdd, srcFormat, b1Even, b1Odd, b2, b3, b4,
Packit Service 5a9772
		                                      b5, b6, b7, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUV(const BYTE* pSrc, UINT32 srcFormat, UINT32 srcStep,
Packit Service 5a9772
                                               BYTE* pDst1[3], const UINT32 dst1Step[3],
Packit Service 5a9772
                                               BYTE* pDst2[3], const UINT32 dst2Step[3],
Packit Service 5a9772
                                               const prim_size_t* roi)
Packit 1fb8d4
{
Packit Service 5a9772
	if (!pSrc || !pDst1 || !dst1Step || !pDst2 || !dst2Step)
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if (!pDst1[0] || !pDst1[1] || !pDst1[2])
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if (!dst1Step[0] || !dst1Step[1] || !dst1Step[2])
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if (!pDst2[0] || !pDst2[1] || !pDst2[2])
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if (!dst2Step[0] || !dst2Step[1] || !dst2Step[2])
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit 1fb8d4
	switch (srcFormat)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PIXEL_FORMAT_BGRA32:
Packit 1fb8d4
		case PIXEL_FORMAT_BGRX32:
Packit Service 5a9772
			return general_RGBToAVC444YUV_BGRX(pSrc, srcStep, pDst1, dst1Step, pDst2, dst2Step,
Packit Service 5a9772
			                                   roi);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_RGBA32:
Packit 1fb8d4
		case PIXEL_FORMAT_RGBX32:
Packit Service 5a9772
			return general_RGBToAVC444YUV_RGBX(pSrc, srcStep, pDst1, dst1Step, pDst2, dst2Step,
Packit Service 5a9772
			                                   roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit Service 5a9772
			return general_RGBToAVC444YUV_ANY(pSrc, srcFormat, srcStep, pDst1, dst1Step, pDst2,
Packit Service 5a9772
			                                  dst2Step, roi);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return !PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void general_RGBToAVC444YUVv2_ANY_DOUBLE_ROW(
Packit Service 5a9772
    const BYTE* srcEven, const BYTE* srcOdd, UINT32 srcFormat, BYTE* yLumaDstEven,
Packit Service 5a9772
    BYTE* yLumaDstOdd, BYTE* uLumaDst, BYTE* vLumaDst, BYTE* yEvenChromaDst1, BYTE* yEvenChromaDst2,
Packit Service 5a9772
    BYTE* yOddChromaDst1, BYTE* yOddChromaDst2, BYTE* uChromaDst1, BYTE* uChromaDst2,
Packit Service 5a9772
    BYTE* vChromaDst1, BYTE* vChromaDst2, UINT32 width)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
	const UINT32 bpp = GetBytesPerPixel(srcFormat);
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < width; x += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		BYTE Ya, Ua, Va;
Packit 1fb8d4
		BYTE Yb, Ub, Vb;
Packit 1fb8d4
		BYTE Yc, Uc, Vc;
Packit 1fb8d4
		BYTE Yd, Ud, Vd;
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE b, g, r;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcEven, srcFormat);
Packit 1fb8d4
			srcEven += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Ya = RGB2Y(r, g, b);
Packit 1fb8d4
			Ua = RGB2U(r, g, b);
Packit 1fb8d4
			Va = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE b, g, r;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcEven, srcFormat);
Packit 1fb8d4
			srcEven += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Yb = RGB2Y(r, g, b);
Packit 1fb8d4
			Ub = RGB2U(r, g, b);
Packit 1fb8d4
			Vb = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yb = Ya;
Packit 1fb8d4
			Ub = Ua;
Packit 1fb8d4
			Vb = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE b, g, r;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcOdd, srcFormat);
Packit 1fb8d4
			srcOdd += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Yc = RGB2Y(r, g, b);
Packit 1fb8d4
			Uc = RGB2U(r, g, b);
Packit 1fb8d4
			Vc = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yc = Ya;
Packit 1fb8d4
			Uc = Ua;
Packit 1fb8d4
			Vc = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd && (x < width - 1))
Packit 1fb8d4
		{
Packit 1fb8d4
			BYTE b, g, r;
Packit 1fb8d4
			const UINT32 color = ReadColor(srcOdd, srcFormat);
Packit 1fb8d4
			srcOdd += bpp;
Packit 1fb8d4
			SplitColor(color, srcFormat, &r, &g, &b, NULL, NULL);
Packit 1fb8d4
			Yd = RGB2Y(r, g, b);
Packit 1fb8d4
			Ud = RGB2U(r, g, b);
Packit 1fb8d4
			Vd = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yd = Ya;
Packit 1fb8d4
			Ud = Ua;
Packit 1fb8d4
			Vd = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* Y [b1] */
Packit 1fb8d4
		*yLumaDstEven++ = Ya;
Packit 1fb8d4
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
			*yLumaDstEven++ = Yb;
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
			*yLumaDstOdd++ = Yc;
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd && (x < width - 1))
Packit 1fb8d4
			*yLumaDstOdd++ = Yd;
Packit 1fb8d4
Packit 1fb8d4
		/* 2x 2y [b2,b3] */
Packit 1fb8d4
		*uLumaDst++ = (Ua + Ub + Uc + Ud) / 4;
Packit 1fb8d4
		*vLumaDst++ = (Va + Vb + Vc + Vd) / 4;
Packit 1fb8d4
Packit 1fb8d4
		/* 2x+1, y [b4,b5] even */
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
		{
Packit 1fb8d4
			*yEvenChromaDst1++ = Ub;
Packit 1fb8d4
			*yEvenChromaDst2++ = Vb;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* 2x+1, y [b4,b5] odd */
Packit 1fb8d4
			if (x < width - 1)
Packit 1fb8d4
			{
Packit 1fb8d4
				*yOddChromaDst1++ = Ud;
Packit 1fb8d4
				*yOddChromaDst2++ = Vd;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			/* 4x 2y+1 [b6, b7] */
Packit 1fb8d4
			if (x % 4 == 0)
Packit 1fb8d4
			{
Packit 1fb8d4
				*uChromaDst1++ = Uc;
Packit 1fb8d4
				*uChromaDst2++ = Vc;
Packit 1fb8d4
			}
Packit 1fb8d4
			/* 4x+2 2y+1 [b8, b9] */
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				*vChromaDst1++ = Uc;
Packit 1fb8d4
				*vChromaDst2++ = Vc;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUVv2_ANY(const BYTE* pSrc, UINT32 srcFormat,
Packit Service 5a9772
                                                     UINT32 srcStep, BYTE* pDst1[3],
Packit Service 5a9772
                                                     const UINT32 dst1Step[3], BYTE* pDst2[3],
Packit Service 5a9772
                                                     const UINT32 dst2Step[3],
Packit Service 5a9772
                                                     const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	/**
Packit 1fb8d4
	 * Note: According to [MS-RDPEGFX 2.2.4.4 RFX_AVC420_BITMAP_STREAM] the
Packit 1fb8d4
	 * width and height of the MPEG-4 AVC/H.264 codec bitstream MUST be aligned
Packit 1fb8d4
	 * to a multiple of 16.
Packit 1fb8d4
	 * Hence the passed destination YUV420/CHROMA420 buffers must have been
Packit 1fb8d4
	 * allocated accordingly !!
Packit 1fb8d4
	 */
Packit 1fb8d4
	/**
Packit Service 5a9772
	 * [MS-RDPEGFX 3.3.8.3.3 YUV420p Stream Combination for YUV444v2 mode] defines the following "Bx
Packit Service 5a9772
	 * areas":
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * YUV420 frame (main view):
Packit 1fb8d4
	 * B1:  From Y444 all pixels
Packit 1fb8d4
	 * B2:  From U444 all pixels in even rows with even rows and columns
Packit 1fb8d4
	 * B3:  From V444 all pixels in even rows with even rows and columns
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * Chroma420 frame (auxillary view):
Packit 1fb8d4
	 * B45: From U444 and V444 all pixels from all odd columns
Packit 1fb8d4
	 * B67: From U444 and V444 every 4th pixel in odd rows
Packit 1fb8d4
	 * B89:  From U444 and V444 every 4th pixel (initial offset of 2) in odd rows
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * Chroma Bxy areas correspond to the left and right half of the YUV420 plane.
Packit 1fb8d4
	 * for (y = 0; y < fullHeight; y++)
Packit 1fb8d4
	 * {
Packit 1fb8d4
	 *     for (x = 0; x < fullWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B1[x,y] = Y444[x,y];
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 *     for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B4[x,y] = U444[2 * x, 2 * y];
Packit 1fb8d4
	 *         B5[x,y] = V444[2 * x, 2 * y];
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *  }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * for (y = 0; y < halfHeight; y++)
Packit 1fb8d4
	 * {
Packit 1fb8d4
	 *     for (x = 0; x < halfWidth; x++)
Packit 1fb8d4
	 *     {
Packit 1fb8d4
	 *         B2[x,y] = U444[2 * x,     2 * y];
Packit 1fb8d4
	 *         B3[x,y] = V444[2 * x,     2 * y];
Packit 1fb8d4
	 *         B6[x,y] = U444[4 * x,     2 * y + 1];
Packit 1fb8d4
	 *         B7[x,y] = V444[4 * x,     2 * y + 1];
Packit 1fb8d4
	 *         B8[x,y] = V444[4 * x + 2, 2 * y + 1];
Packit 1fb8d4
	 *         B9[x,y] = V444[4 * x + 2, 2 * y] + 1;
Packit 1fb8d4
	 *     }
Packit 1fb8d4
	 *  }
Packit 1fb8d4
	 *
Packit 1fb8d4
	 */
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
Packit 1fb8d4
	if (roi->height < 1 || roi->width < 1)
Packit 1fb8d4
		return !PRIMITIVES_SUCCESS;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < roi->height; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* srcEven = (pSrc + y * srcStep);
Packit 1fb8d4
		const BYTE* srcOdd = (y < roi->height - 1) ? (srcEven + srcStep) : NULL;
Packit 1fb8d4
		BYTE* dstLumaYEven = (pDst1[0] + y * dst1Step[0]);
Packit 1fb8d4
		BYTE* dstLumaYOdd = (dstLumaYEven + dst1Step[0]);
Packit 1fb8d4
		BYTE* dstLumaU = (pDst1[1] + (y / 2) * dst1Step[1]);
Packit 1fb8d4
		BYTE* dstLumaV = (pDst1[2] + (y / 2) * dst1Step[2]);
Packit 1fb8d4
		BYTE* dstEvenChromaY1 = (pDst2[0] + y * dst2Step[0]);
Packit 1fb8d4
		BYTE* dstEvenChromaY2 = dstEvenChromaY1 + roi->width / 2;
Packit 1fb8d4
		BYTE* dstOddChromaY1 = dstEvenChromaY1 + dst2Step[0];
Packit 1fb8d4
		BYTE* dstOddChromaY2 = dstEvenChromaY2 + dst2Step[0];
Packit 1fb8d4
		BYTE* dstChromaU1 = (pDst2[1] + (y / 2) * dst2Step[1]);
Packit 1fb8d4
		BYTE* dstChromaV1 = (pDst2[2] + (y / 2) * dst2Step[2]);
Packit 1fb8d4
		BYTE* dstChromaU2 = dstChromaU1 + roi->width / 4;
Packit 1fb8d4
		BYTE* dstChromaV2 = dstChromaV1 + roi->width / 4;
Packit Service 5a9772
		general_RGBToAVC444YUVv2_ANY_DOUBLE_ROW(
Packit Service 5a9772
		    srcEven, srcOdd, srcFormat, dstLumaYEven, dstLumaYOdd, dstLumaU, dstLumaV,
Packit Service 5a9772
		    dstEvenChromaY1, dstEvenChromaY2, dstOddChromaY1, dstOddChromaY2, dstChromaU1,
Packit Service 5a9772
		    dstChromaU2, dstChromaV1, dstChromaV2, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void general_RGBToAVC444YUVv2_BGRX_DOUBLE_ROW(
Packit Service 5a9772
    const BYTE* srcEven, const BYTE* srcOdd, BYTE* yLumaDstEven, BYTE* yLumaDstOdd, BYTE* uLumaDst,
Packit Service 5a9772
    BYTE* vLumaDst, BYTE* yEvenChromaDst1, BYTE* yEvenChromaDst2, BYTE* yOddChromaDst1,
Packit Service 5a9772
    BYTE* yOddChromaDst2, BYTE* uChromaDst1, BYTE* uChromaDst2, BYTE* vChromaDst1,
Packit Service 5a9772
    BYTE* vChromaDst2, UINT32 width)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < width; x += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		BYTE Ya, Ua, Va;
Packit 1fb8d4
		BYTE Yb, Ub, Vb;
Packit 1fb8d4
		BYTE Yc, Uc, Vc;
Packit 1fb8d4
		BYTE Yd, Ud, Vd;
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Ya = RGB2Y(r, g, b);
Packit 1fb8d4
			Ua = RGB2U(r, g, b);
Packit 1fb8d4
			Va = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcEven++;
Packit 1fb8d4
			const BYTE g = *srcEven++;
Packit 1fb8d4
			const BYTE r = *srcEven++;
Packit 1fb8d4
			srcEven++;
Packit 1fb8d4
			Yb = RGB2Y(r, g, b);
Packit 1fb8d4
			Ub = RGB2U(r, g, b);
Packit 1fb8d4
			Vb = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yb = Ya;
Packit 1fb8d4
			Ub = Ua;
Packit 1fb8d4
			Vb = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Yc = RGB2Y(r, g, b);
Packit 1fb8d4
			Uc = RGB2U(r, g, b);
Packit 1fb8d4
			Vc = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yc = Ya;
Packit 1fb8d4
			Uc = Ua;
Packit 1fb8d4
			Vc = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd && (x < width - 1))
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE b = *srcOdd++;
Packit 1fb8d4
			const BYTE g = *srcOdd++;
Packit 1fb8d4
			const BYTE r = *srcOdd++;
Packit 1fb8d4
			srcOdd++;
Packit 1fb8d4
			Yd = RGB2Y(r, g, b);
Packit 1fb8d4
			Ud = RGB2U(r, g, b);
Packit 1fb8d4
			Vd = RGB2V(r, g, b);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			Yd = Ya;
Packit 1fb8d4
			Ud = Ua;
Packit 1fb8d4
			Vd = Va;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		/* Y [b1] */
Packit 1fb8d4
		*yLumaDstEven++ = Ya;
Packit 1fb8d4
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
			*yLumaDstEven++ = Yb;
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
			*yLumaDstOdd++ = Yc;
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd && (x < width - 1))
Packit 1fb8d4
			*yLumaDstOdd++ = Yd;
Packit 1fb8d4
Packit 1fb8d4
		/* 2x 2y [b2,b3] */
Packit 1fb8d4
		*uLumaDst++ = (Ua + Ub + Uc + Ud) / 4;
Packit 1fb8d4
		*vLumaDst++ = (Va + Vb + Vc + Vd) / 4;
Packit 1fb8d4
Packit 1fb8d4
		/* 2x+1, y [b4,b5] even */
Packit 1fb8d4
		if (x < width - 1)
Packit 1fb8d4
		{
Packit 1fb8d4
			*yEvenChromaDst1++ = Ub;
Packit 1fb8d4
			*yEvenChromaDst2++ = Vb;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (srcOdd)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* 2x+1, y [b4,b5] odd */
Packit 1fb8d4
			if (x < width - 1)
Packit 1fb8d4
			{
Packit 1fb8d4
				*yOddChromaDst1++ = Ud;
Packit 1fb8d4
				*yOddChromaDst2++ = Vd;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			/* 4x 2y+1 [b6, b7] */
Packit 1fb8d4
			if (x % 4 == 0)
Packit 1fb8d4
			{
Packit 1fb8d4
				*uChromaDst1++ = Uc;
Packit 1fb8d4
				*uChromaDst2++ = Vc;
Packit 1fb8d4
			}
Packit 1fb8d4
			/* 4x+2 2y+1 [b8, b9] */
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				*vChromaDst1++ = Uc;
Packit 1fb8d4
				*vChromaDst2++ = Vc;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUVv2_BGRX(const BYTE* pSrc, UINT32 srcStep,
Packit Service 5a9772
                                                      BYTE* pDst1[3], const UINT32 dst1Step[3],
Packit Service 5a9772
                                                      BYTE* pDst2[3], const UINT32 dst2Step[3],
Packit Service 5a9772
                                                      const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
Packit 1fb8d4
	if (roi->height < 1 || roi->width < 1)
Packit 1fb8d4
		return !PRIMITIVES_SUCCESS;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < roi->height; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const BYTE* srcEven = (pSrc + y * srcStep);
Packit 1fb8d4
		const BYTE* srcOdd = (y < roi->height - 1) ? (srcEven + srcStep) : NULL;
Packit 1fb8d4
		BYTE* dstLumaYEven = (pDst1[0] + y * dst1Step[0]);
Packit 1fb8d4
		BYTE* dstLumaYOdd = (dstLumaYEven + dst1Step[0]);
Packit 1fb8d4
		BYTE* dstLumaU = (pDst1[1] + (y / 2) * dst1Step[1]);
Packit 1fb8d4
		BYTE* dstLumaV = (pDst1[2] + (y / 2) * dst1Step[2]);
Packit 1fb8d4
		BYTE* dstEvenChromaY1 = (pDst2[0] + y * dst2Step[0]);
Packit 1fb8d4
		BYTE* dstEvenChromaY2 = dstEvenChromaY1 + roi->width / 2;
Packit 1fb8d4
		BYTE* dstOddChromaY1 = dstEvenChromaY1 + dst2Step[0];
Packit 1fb8d4
		BYTE* dstOddChromaY2 = dstEvenChromaY2 + dst2Step[0];
Packit 1fb8d4
		BYTE* dstChromaU1 = (pDst2[1] + (y / 2) * dst2Step[1]);
Packit 1fb8d4
		BYTE* dstChromaV1 = (pDst2[2] + (y / 2) * dst2Step[2]);
Packit 1fb8d4
		BYTE* dstChromaU2 = dstChromaU1 + roi->width / 4;
Packit 1fb8d4
		BYTE* dstChromaV2 = dstChromaV1 + roi->width / 4;
Packit Service 5a9772
		general_RGBToAVC444YUVv2_BGRX_DOUBLE_ROW(
Packit Service 5a9772
		    srcEven, srcOdd, dstLumaYEven, dstLumaYOdd, dstLumaU, dstLumaV, dstEvenChromaY1,
Packit Service 5a9772
		    dstEvenChromaY2, dstOddChromaY1, dstOddChromaY2, dstChromaU1, dstChromaU2, dstChromaV1,
Packit Service 5a9772
		    dstChromaV2, roi->width);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t general_RGBToAVC444YUVv2(const BYTE* pSrc, UINT32 srcFormat, UINT32 srcStep,
Packit Service 5a9772
                                                 BYTE* pDst1[3], const UINT32 dst1Step[3],
Packit Service 5a9772
                                                 BYTE* pDst2[3], const UINT32 dst2Step[3],
Packit Service 5a9772
                                                 const prim_size_t* roi)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (srcFormat)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PIXEL_FORMAT_BGRA32:
Packit 1fb8d4
		case PIXEL_FORMAT_BGRX32:
Packit Service 5a9772
			return general_RGBToAVC444YUVv2_BGRX(pSrc, srcStep, pDst1, dst1Step, pDst2, dst2Step,
Packit Service 5a9772
			                                     roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit Service 5a9772
			return general_RGBToAVC444YUVv2_ANY(pSrc, srcFormat, srcStep, pDst1, dst1Step, pDst2,
Packit Service 5a9772
			                                    dst2Step, roi);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return !PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void primitives_init_YUV(primitives_t* prims)
Packit 1fb8d4
{
Packit 1fb8d4
	prims->YUV420ToRGB_8u_P3AC4R = general_YUV420ToRGB_8u_P3AC4R;
Packit 1fb8d4
	prims->YUV444ToRGB_8u_P3AC4R = general_YUV444ToRGB_8u_P3AC4R;
Packit 1fb8d4
	prims->RGBToYUV420_8u_P3AC4R = general_RGBToYUV420_8u_P3AC4R;
Packit 1fb8d4
	prims->RGBToYUV444_8u_P3AC4R = general_RGBToYUV444_8u_P3AC4R;
Packit 1fb8d4
	prims->YUV420CombineToYUV444 = general_YUV420CombineToYUV444;
Packit 1fb8d4
	prims->YUV444SplitToYUV420 = general_YUV444SplitToYUV420;
Packit 1fb8d4
	prims->RGBToAVC444YUV = general_RGBToAVC444YUV;
Packit 1fb8d4
	prims->RGBToAVC444YUVv2 = general_RGBToAVC444YUVv2;
Packit 1fb8d4
}