Blame libfreerdp/primitives/prim_YUV_neon.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Optimized YUV/RGB conversion operations
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Thomas Erbesdobler
Packit 1fb8d4
 * Copyright 2016-2017 Armin Novak <armin.novak@thincast.com>
Packit 1fb8d4
 * Copyright 2016-2017 Norbert Federa <norbert.federa@thincast.com>
Packit 1fb8d4
 * Copyright 2016-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 <winpr/sysinfo.h>
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <freerdp/types.h>
Packit 1fb8d4
#include <freerdp/primitives.h>
Packit 1fb8d4
Packit 1fb8d4
#include "prim_internal.h"
Packit 1fb8d4
Packit 1fb8d4
#if !defined(WITH_NEON)
Packit 1fb8d4
#error "This file must only be included if WITH_NEON is active!"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <arm_neon.h>
Packit 1fb8d4
Packit 1fb8d4
static primitives_t* generic = NULL;
Packit 1fb8d4
Packit Service 5a9772
static INLINE uint8x8_t neon_YUV2R(int32x4_t Ch, int32x4_t Cl, int16x4_t Dh, int16x4_t Dl,
Packit 1fb8d4
                                   int16x4_t Eh, int16x4_t El)
Packit 1fb8d4
{
Packit 1fb8d4
	/* R = (256 * Y + 403 * (V - 128)) >> 8 */
Packit 1fb8d4
	const int16x4_t c403 = vdup_n_s16(403);
Packit 1fb8d4
	const int32x4_t CEh = vmlal_s16(Ch, Eh, c403);
Packit 1fb8d4
	const int32x4_t CEl = vmlal_s16(Cl, El, c403);
Packit 1fb8d4
	const int32x4_t Rh = vrshrq_n_s32(CEh, 8);
Packit 1fb8d4
	const int32x4_t Rl = vrshrq_n_s32(CEl, 8);
Packit 1fb8d4
	const int16x8_t R = vcombine_s16(vqmovn_s32(Rl), vqmovn_s32(Rh));
Packit 1fb8d4
	return vqmovun_s16(R);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE uint8x8_t neon_YUV2G(int32x4_t Ch, int32x4_t Cl, int16x4_t Dh, int16x4_t Dl,
Packit 1fb8d4
                                   int16x4_t Eh, int16x4_t El)
Packit 1fb8d4
{
Packit 1fb8d4
	/* G = (256L * Y -  48 * (U - 128) - 120 * (V - 128)) >> 8 */
Packit 1fb8d4
	const int16x4_t c48 = vdup_n_s16(48);
Packit 1fb8d4
	const int16x4_t c120 = vdup_n_s16(120);
Packit 1fb8d4
	const int32x4_t CDh = vmlsl_s16(Ch, Dh, c48);
Packit 1fb8d4
	const int32x4_t CDl = vmlsl_s16(Cl, Dl, c48);
Packit 1fb8d4
	const int32x4_t CDEh = vmlsl_s16(CDh, Eh, c120);
Packit 1fb8d4
	const int32x4_t CDEl = vmlsl_s16(CDl, El, c120);
Packit 1fb8d4
	const int32x4_t Gh = vrshrq_n_s32(CDEh, 8);
Packit 1fb8d4
	const int32x4_t Gl = vrshrq_n_s32(CDEl, 8);
Packit 1fb8d4
	const int16x8_t G = vcombine_s16(vqmovn_s32(Gl), vqmovn_s32(Gh));
Packit 1fb8d4
	return vqmovun_s16(G);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE uint8x8_t neon_YUV2B(int32x4_t Ch, int32x4_t Cl, int16x4_t Dh, int16x4_t Dl,
Packit 1fb8d4
                                   int16x4_t Eh, int16x4_t El)
Packit 1fb8d4
{
Packit 1fb8d4
	/* B = (256L * Y + 475 * (U - 128)) >> 8*/
Packit 1fb8d4
	const int16x4_t c475 = vdup_n_s16(475);
Packit 1fb8d4
	const int32x4_t CDh = vmlal_s16(Ch, Dh, c475);
Packit 1fb8d4
	const int32x4_t CDl = vmlal_s16(Ch, Dl, c475);
Packit 1fb8d4
	const int32x4_t Bh = vrshrq_n_s32(CDh, 8);
Packit 1fb8d4
	const int32x4_t Bl = vrshrq_n_s32(CDl, 8);
Packit 1fb8d4
	const int16x8_t B = vcombine_s16(vqmovn_s32(Bl), vqmovn_s32(Bh));
Packit 1fb8d4
	return vqmovun_s16(B);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BYTE* neon_YuvToRgbPixel(BYTE* pRGB, int16x8_t Y, int16x8_t D, int16x8_t E,
Packit Service 5a9772
                                       const uint8_t rPos, const uint8_t gPos, const uint8_t bPos,
Packit Service 5a9772
                                       const uint8_t aPos)
Packit 1fb8d4
{
Packit 1fb8d4
	uint8x8x4_t bgrx;
Packit Service 5a9772
	const int32x4_t Ch = vmulq_n_s32(vmovl_s16(vget_high_s16(Y)), 256); /* Y * 256 */
Packit 1fb8d4
	const int32x4_t Cl = vmulq_n_s32(vmovl_s16(vget_low_s16(Y)), 256);  /* Y * 256 */
Packit 1fb8d4
	const int16x4_t Dh = vget_high_s16(D);
Packit 1fb8d4
	const int16x4_t Dl = vget_low_s16(D);
Packit 1fb8d4
	const int16x4_t Eh = vget_high_s16(E);
Packit 1fb8d4
	const int16x4_t El = vget_low_s16(E);
Packit 1fb8d4
	{
Packit 1fb8d4
		/* B = (256L * Y + 475 * (U - 128)) >> 8*/
Packit 1fb8d4
		const int16x4_t c475 = vdup_n_s16(475);
Packit 1fb8d4
		const int32x4_t CDh = vmlal_s16(Ch, Dh, c475);
Packit 1fb8d4
		const int32x4_t CDl = vmlal_s16(Cl, Dl, c475);
Packit 1fb8d4
		const int32x4_t Bh = vrshrq_n_s32(CDh, 8);
Packit 1fb8d4
		const int32x4_t Bl = vrshrq_n_s32(CDl, 8);
Packit 1fb8d4
		const int16x8_t B = vcombine_s16(vqmovn_s32(Bl), vqmovn_s32(Bh));
Packit 1fb8d4
		bgrx.val[bPos] = vqmovun_s16(B);
Packit 1fb8d4
	}
Packit 1fb8d4
	{
Packit 1fb8d4
		/* G = (256L * Y -  48 * (U - 128) - 120 * (V - 128)) >> 8 */
Packit 1fb8d4
		const int16x4_t c48 = vdup_n_s16(48);
Packit 1fb8d4
		const int16x4_t c120 = vdup_n_s16(120);
Packit 1fb8d4
		const int32x4_t CDh = vmlsl_s16(Ch, Dh, c48);
Packit 1fb8d4
		const int32x4_t CDl = vmlsl_s16(Cl, Dl, c48);
Packit 1fb8d4
		const int32x4_t CDEh = vmlsl_s16(CDh, Eh, c120);
Packit 1fb8d4
		const int32x4_t CDEl = vmlsl_s16(CDl, El, c120);
Packit 1fb8d4
		const int32x4_t Gh = vrshrq_n_s32(CDEh, 8);
Packit 1fb8d4
		const int32x4_t Gl = vrshrq_n_s32(CDEl, 8);
Packit 1fb8d4
		const int16x8_t G = vcombine_s16(vqmovn_s32(Gl), vqmovn_s32(Gh));
Packit 1fb8d4
		bgrx.val[gPos] = vqmovun_s16(G);
Packit 1fb8d4
	}
Packit 1fb8d4
	{
Packit 1fb8d4
		/* R = (256 * Y + 403 * (V - 128)) >> 8 */
Packit 1fb8d4
		const int16x4_t c403 = vdup_n_s16(403);
Packit 1fb8d4
		const int32x4_t CEh = vmlal_s16(Ch, Eh, c403);
Packit 1fb8d4
		const int32x4_t CEl = vmlal_s16(Cl, El, c403);
Packit 1fb8d4
		const int32x4_t Rh = vrshrq_n_s32(CEh, 8);
Packit 1fb8d4
		const int32x4_t Rl = vrshrq_n_s32(CEl, 8);
Packit 1fb8d4
		const int16x8_t R = vcombine_s16(vqmovn_s32(Rl), vqmovn_s32(Rh));
Packit 1fb8d4
		bgrx.val[rPos] = vqmovun_s16(R);
Packit 1fb8d4
	}
Packit 1fb8d4
	{
Packit 1fb8d4
		/* A */
Packit 1fb8d4
		bgrx.val[aPos] = vdup_n_u8(0xFF);
Packit 1fb8d4
	}
Packit 1fb8d4
	vst4_u8(pRGB, bgrx);
Packit 1fb8d4
	pRGB += 32;
Packit 1fb8d4
	return pRGB;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t neon_YUV420ToX(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                       BYTE* pDst, UINT32 dstStep, const prim_size_t* roi,
Packit Service 5a9772
                                       const uint8_t rPos, const uint8_t gPos, const uint8_t bPos,
Packit Service 5a9772
                                       const uint8_t aPos)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
	const UINT32 nWidth = roi->width;
Packit 1fb8d4
	const UINT32 nHeight = roi->height;
Packit 1fb8d4
	const DWORD pad = nWidth % 16;
Packit 1fb8d4
	const UINT32 yPad = srcStep[0] - roi->width;
Packit 1fb8d4
	const UINT32 uPad = srcStep[1] - roi->width / 2;
Packit 1fb8d4
	const UINT32 vPad = srcStep[2] - roi->width / 2;
Packit 1fb8d4
	const UINT32 dPad = dstStep - roi->width * 4;
Packit 1fb8d4
	const int16x8_t c128 = vdupq_n_s16(128);
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < nHeight; y += 2)
Packit 1fb8d4
	{
Packit 1fb8d4
		const uint8_t* pY1 = pSrc[0] + y * srcStep[0];
Packit 1fb8d4
		const uint8_t* pY2 = pY1 + srcStep[0];
Packit 1fb8d4
		const uint8_t* pU = pSrc[1] + (y / 2) * srcStep[1];
Packit 1fb8d4
		const uint8_t* pV = pSrc[2] + (y / 2) * srcStep[2];
Packit 1fb8d4
		uint8_t* pRGB1 = pDst + y * dstStep;
Packit 1fb8d4
		uint8_t* pRGB2 = pRGB1 + dstStep;
Packit 1fb8d4
		UINT32 x;
Packit 1fb8d4
		const BOOL lastY = y >= nHeight - 1;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < nWidth - pad;)
Packit 1fb8d4
		{
Packit 1fb8d4
			const uint8x8_t Uraw = vld1_u8(pU);
Packit 1fb8d4
			const uint8x8x2_t Uu = vzip_u8(Uraw, Uraw);
Packit 1fb8d4
			const int16x8_t U1 = vreinterpretq_s16_u16(vmovl_u8(Uu.val[0]));
Packit 1fb8d4
			const int16x8_t U2 = vreinterpretq_s16_u16(vmovl_u8(Uu.val[1]));
Packit 1fb8d4
			const uint8x8_t Vraw = vld1_u8(pV);
Packit 1fb8d4
			const uint8x8x2_t Vu = vzip_u8(Vraw, Vraw);
Packit 1fb8d4
			const int16x8_t V1 = vreinterpretq_s16_u16(vmovl_u8(Vu.val[0]));
Packit 1fb8d4
			const int16x8_t V2 = vreinterpretq_s16_u16(vmovl_u8(Vu.val[1]));
Packit 1fb8d4
			const int16x8_t D1 = vsubq_s16(U1, c128);
Packit 1fb8d4
			const int16x8_t E1 = vsubq_s16(V1, c128);
Packit 1fb8d4
			const int16x8_t D2 = vsubq_s16(U2, c128);
Packit 1fb8d4
			const int16x8_t E2 = vsubq_s16(V2, c128);
Packit 1fb8d4
			{
Packit 1fb8d4
				const uint8x8_t Y1u = vld1_u8(pY1);
Packit 1fb8d4
				const int16x8_t Y1 = vreinterpretq_s16_u16(vmovl_u8(Y1u));
Packit 1fb8d4
				pRGB1 = neon_YuvToRgbPixel(pRGB1, Y1, D1, E1, rPos, gPos, bPos, aPos);
Packit 1fb8d4
				pY1 += 8;
Packit 1fb8d4
				x += 8;
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				const uint8x8_t Y1u = vld1_u8(pY1);
Packit 1fb8d4
				const int16x8_t Y1 = vreinterpretq_s16_u16(vmovl_u8(Y1u));
Packit 1fb8d4
				pRGB1 = neon_YuvToRgbPixel(pRGB1, Y1, D2, E2, rPos, gPos, bPos, aPos);
Packit 1fb8d4
				pY1 += 8;
Packit 1fb8d4
				x += 8;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (!lastY)
Packit 1fb8d4
			{
Packit 1fb8d4
				{
Packit 1fb8d4
					const uint8x8_t Y2u = vld1_u8(pY2);
Packit 1fb8d4
					const int16x8_t Y2 = vreinterpretq_s16_u16(vmovl_u8(Y2u));
Packit 1fb8d4
					pRGB2 = neon_YuvToRgbPixel(pRGB2, Y2, D1, E1, rPos, gPos, bPos, aPos);
Packit 1fb8d4
					pY2 += 8;
Packit 1fb8d4
				}
Packit 1fb8d4
				{
Packit 1fb8d4
					const uint8x8_t Y2u = vld1_u8(pY2);
Packit 1fb8d4
					const int16x8_t Y2 = vreinterpretq_s16_u16(vmovl_u8(Y2u));
Packit 1fb8d4
					pRGB2 = neon_YuvToRgbPixel(pRGB2, Y2, D2, E2, rPos, gPos, bPos, aPos);
Packit 1fb8d4
					pY2 += 8;
Packit 1fb8d4
				}
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			pU += 8;
Packit 1fb8d4
			pV += 8;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; x < nWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE U = *pU;
Packit 1fb8d4
			const BYTE V = *pV;
Packit 1fb8d4
			{
Packit 1fb8d4
				const BYTE Y = *pY1++;
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
				pRGB1[aPos] = 0xFF;
Packit 1fb8d4
				pRGB1[rPos] = r;
Packit 1fb8d4
				pRGB1[gPos] = g;
Packit 1fb8d4
				pRGB1[bPos] = b;
Packit 1fb8d4
				pRGB1 += 4;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (!lastY)
Packit 1fb8d4
			{
Packit 1fb8d4
				const BYTE Y = *pY2++;
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
				pRGB2[aPos] = 0xFF;
Packit 1fb8d4
				pRGB2[rPos] = r;
Packit 1fb8d4
				pRGB2[gPos] = g;
Packit 1fb8d4
				pRGB2[bPos] = b;
Packit 1fb8d4
				pRGB2 += 4;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (x % 2)
Packit 1fb8d4
			{
Packit 1fb8d4
				pU++;
Packit 1fb8d4
				pV++;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pRGB1 += dPad;
Packit 1fb8d4
		pRGB2 += dPad;
Packit 1fb8d4
		pY1 += yPad;
Packit 1fb8d4
		pY2 += yPad;
Packit 1fb8d4
		pU += uPad;
Packit 1fb8d4
		pV += vPad;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_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
	switch (DstFormat)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PIXEL_FORMAT_BGRA32:
Packit 1fb8d4
		case PIXEL_FORMAT_BGRX32:
Packit 1fb8d4
			return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 2, 1, 0, 3);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_RGBA32:
Packit 1fb8d4
		case PIXEL_FORMAT_RGBX32:
Packit 1fb8d4
			return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 0, 1, 2, 3);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_ARGB32:
Packit 1fb8d4
		case PIXEL_FORMAT_XRGB32:
Packit 1fb8d4
			return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 1, 2, 3, 0);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_ABGR32:
Packit 1fb8d4
		case PIXEL_FORMAT_XBGR32:
Packit 1fb8d4
			return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 3, 2, 1, 0);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return generic->YUV420ToRGB_8u_P3AC4R(pSrc, srcStep, pDst, dstStep, DstFormat, roi);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static INLINE pstatus_t neon_YUV444ToX(const BYTE* const pSrc[3], const UINT32 srcStep[3],
Packit Service 5a9772
                                       BYTE* pDst, UINT32 dstStep, const prim_size_t* roi,
Packit Service 5a9772
                                       const uint8_t rPos, const uint8_t gPos, const uint8_t bPos,
Packit Service 5a9772
                                       const uint8_t aPos)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 y;
Packit 1fb8d4
	const UINT32 nWidth = roi->width;
Packit 1fb8d4
	const UINT32 nHeight = roi->height;
Packit 1fb8d4
	const UINT32 yPad = srcStep[0] - roi->width;
Packit 1fb8d4
	const UINT32 uPad = srcStep[1] - roi->width;
Packit 1fb8d4
	const UINT32 vPad = srcStep[2] - roi->width;
Packit 1fb8d4
	const UINT32 dPad = dstStep - roi->width * 4;
Packit 1fb8d4
	const uint8_t* pY = pSrc[0];
Packit 1fb8d4
	const uint8_t* pU = pSrc[1];
Packit 1fb8d4
	const uint8_t* pV = pSrc[2];
Packit 1fb8d4
	uint8_t* pRGB = pDst;
Packit 1fb8d4
	const int16x8_t c128 = vdupq_n_s16(128);
Packit 1fb8d4
	const DWORD pad = nWidth % 8;
Packit 1fb8d4
Packit 1fb8d4
	for (y = 0; y < nHeight; y++)
Packit 1fb8d4
	{
Packit 1fb8d4
		UINT32 x;
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < nWidth - pad; x += 8)
Packit 1fb8d4
		{
Packit 1fb8d4
			const uint8x8_t Yu = vld1_u8(pY);
Packit 1fb8d4
			const int16x8_t Y = vreinterpretq_s16_u16(vmovl_u8(Yu));
Packit 1fb8d4
			const uint8x8_t Uu = vld1_u8(pU);
Packit 1fb8d4
			const int16x8_t U = vreinterpretq_s16_u16(vmovl_u8(Uu));
Packit 1fb8d4
			const uint8x8_t Vu = vld1_u8(pV);
Packit 1fb8d4
			const int16x8_t V = vreinterpretq_s16_u16(vmovl_u8(Vu));
Packit 1fb8d4
			/* Do the calculations on Y in 32bit width, the result of 255 * 256 does not fit
Packit 1fb8d4
			 * a signed 16 bit value. */
Packit 1fb8d4
			const int16x8_t D = vsubq_s16(U, c128);
Packit 1fb8d4
			const int16x8_t E = vsubq_s16(V, c128);
Packit 1fb8d4
			pRGB = neon_YuvToRgbPixel(pRGB, Y, D, E, rPos, gPos, bPos, aPos);
Packit 1fb8d4
			pY += 8;
Packit 1fb8d4
			pU += 8;
Packit 1fb8d4
			pV += 8;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < pad; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE Y = *pY++;
Packit 1fb8d4
			const BYTE U = *pU++;
Packit 1fb8d4
			const BYTE V = *pV++;
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[aPos] = 0xFF;
Packit 1fb8d4
			pRGB[rPos] = r;
Packit 1fb8d4
			pRGB[gPos] = g;
Packit 1fb8d4
			pRGB[bPos] = b;
Packit 1fb8d4
			pRGB += 4;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pRGB += dPad;
Packit 1fb8d4
		pY += yPad;
Packit 1fb8d4
		pU += uPad;
Packit 1fb8d4
		pV += vPad;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_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 neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 2, 1, 0, 3);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_RGBA32:
Packit 1fb8d4
		case PIXEL_FORMAT_RGBX32:
Packit 1fb8d4
			return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 0, 1, 2, 3);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_ARGB32:
Packit 1fb8d4
		case PIXEL_FORMAT_XRGB32:
Packit 1fb8d4
			return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 1, 2, 3, 0);
Packit 1fb8d4
Packit 1fb8d4
		case PIXEL_FORMAT_ABGR32:
Packit 1fb8d4
		case PIXEL_FORMAT_XBGR32:
Packit 1fb8d4
			return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 3, 2, 1, 0);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return generic->YUV444ToRGB_8u_P3AC4R(pSrc, srcStep, pDst, dstStep, DstFormat, roi);
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_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 evenY = 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 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 = pU + dstStep[1];
Packit 1fb8d4
		BYTE* pV1 = pV + dstStep[2];
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x + 16 < halfWidth; x += 16)
Packit 1fb8d4
		{
Packit 1fb8d4
			{
Packit 1fb8d4
				const uint8x16_t u = vld1q_u8(Um);
Packit 1fb8d4
				uint8x16x2_t u2x;
Packit 1fb8d4
				u2x.val[0] = u;
Packit 1fb8d4
				u2x.val[1] = u;
Packit 1fb8d4
				vst2q_u8(pU, u2x);
Packit 1fb8d4
				vst2q_u8(pU1, u2x);
Packit 1fb8d4
				Um += 16;
Packit 1fb8d4
				pU += 32;
Packit 1fb8d4
				pU1 += 32;
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				const uint8x16_t v = vld1q_u8(Vm);
Packit 1fb8d4
				uint8x16x2_t v2x;
Packit 1fb8d4
				v2x.val[0] = v;
Packit 1fb8d4
				v2x.val[1] = v;
Packit 1fb8d4
				vst2q_u8(pV, v2x);
Packit 1fb8d4
				vst2q_u8(pV1, v2x);
Packit 1fb8d4
				Vm += 16;
Packit 1fb8d4
				pV += 32;
Packit 1fb8d4
				pV1 += 32;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const BYTE u = *Um++;
Packit 1fb8d4
			const BYTE v = *Vm++;
Packit 1fb8d4
			*pU++ = u;
Packit 1fb8d4
			*pU++ = u;
Packit 1fb8d4
			*pU1++ = u;
Packit 1fb8d4
			*pU1++ = u;
Packit 1fb8d4
			*pV++ = v;
Packit 1fb8d4
			*pV++ = v;
Packit 1fb8d4
			*pV1++ = v;
Packit 1fb8d4
			*pV1++ = v;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return PRIMITIVES_SUCCESS;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_ChromaFilter(BYTE* pDst[3], const UINT32 dstStep[3], 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
	const UINT32 halfPad = halfWidth % 16;
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 / 2; x < halfWidth + roi->left / 2 - halfPad; x += 16)
Packit 1fb8d4
		{
Packit 1fb8d4
			{
Packit 1fb8d4
				/* U = (U2x,2y << 2) - U2x1,2y - U2x,2y1 - U2x1,2y1 */
Packit 1fb8d4
				uint8x8x2_t u = vld2_u8(&pU[2 * x]);
Packit Service 5a9772
				const int16x8_t up =
Packit Service 5a9772
				    vreinterpretq_s16_u16(vshll_n_u8(u.val[0], 2)); /* Ux2,2y << 2 */
Packit 1fb8d4
				const uint8x8x2_t u1 = vld2_u8(&pU1[2 * x]);
Packit 1fb8d4
				const uint16x8_t usub = vaddl_u8(u1.val[1], u1.val[0]); /* U2x,2y1 + U2x1,2y1 */
Packit Service 5a9772
				const int16x8_t us = vreinterpretq_s16_u16(
Packit Service 5a9772
				    vaddw_u8(usub, u.val[1])); /* U2x1,2y + U2x,2y1 + U2x1,2y1 */
Packit 1fb8d4
				const int16x8_t un = vsubq_s16(up, us);
Packit 1fb8d4
				const uint8x8_t u8 = vqmovun_s16(un); /* CLIP(un) */
Packit 1fb8d4
				u.val[0] = u8;
Packit 1fb8d4
				vst2_u8(&pU[2 * x], u);
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				/* V = (V2x,2y << 2) - V2x1,2y - V2x,2y1 - V2x1,2y1 */
Packit 1fb8d4
				uint8x8x2_t v = vld2_u8(&pV[2 * x]);
Packit Service 5a9772
				const int16x8_t vp =
Packit Service 5a9772
				    vreinterpretq_s16_u16(vshll_n_u8(v.val[0], 2)); /* Vx2,2y << 2 */
Packit 1fb8d4
				const uint8x8x2_t v1 = vld2_u8(&pV1[2 * x]);
Packit 1fb8d4
				const uint16x8_t vsub = vaddl_u8(v1.val[1], v1.val[0]); /* V2x,2y1 + V2x1,2y1 */
Packit Service 5a9772
				const int16x8_t vs = vreinterpretq_s16_u16(
Packit Service 5a9772
				    vaddw_u8(vsub, v.val[1])); /* V2x1,2y + V2x,2y1 + V2x1,2y1 */
Packit 1fb8d4
				const int16x8_t vn = vsubq_s16(vp, vs);
Packit 1fb8d4
				const uint8x8_t v8 = vqmovun_s16(vn); /* CLIP(vn) */
Packit 1fb8d4
				v.val[0] = v8;
Packit 1fb8d4
				vst2_u8(&pV[2 * x], v);
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; x < halfWidth + roi->left / 2; 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 neon_ChromaV1ToYUV444(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
	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 1fb8d4
	const UINT32 halfPad = halfWidth % 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 - halfPad; x += 16)
Packit 1fb8d4
		{
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x2_t u = vld2q_u8(&pU[2 * x]);
Packit 1fb8d4
				u.val[1] = vld1q_u8(&Ua[x]);
Packit 1fb8d4
				vst2q_u8(&pU[2 * x], u);
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x2_t v = vld2q_u8(&pV[2 * x]);
Packit 1fb8d4
				v.val[1] = vld1q_u8(&Va[x]);
Packit 1fb8d4
				vst2q_u8(&pV[2 * x], v);
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; 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 neon_ChromaFilter(pDst, dstStep, roi);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_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 halfPad = halfWidth % 16;
Packit 1fb8d4
	const UINT32 halfHeight = (nHeight + 1) / 2;
Packit 1fb8d4
	const UINT32 quaterWidth = (nWidth + 3) / 4;
Packit 1fb8d4
	const UINT32 quaterPad = quaterWidth % 16;
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 - halfPad; x += 16)
Packit 1fb8d4
		{
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x2_t u = vld2q_u8(&pU[2 * x]);
Packit 1fb8d4
				u.val[1] = vld1q_u8(&pYaU[x]);
Packit 1fb8d4
				vst2q_u8(&pU[2 * x], u);
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x2_t v = vld2q_u8(&pV[2 * x]);
Packit 1fb8d4
				v.val[1] = vld1q_u8(&pYaV[x]);
Packit 1fb8d4
				vst2q_u8(&pV[2 * x], v);
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; x < halfWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const UINT32 odd = 2 * x + 1;
Packit 1fb8d4
			pU[odd] = pYaU[x];
Packit 1fb8d4
			pV[odd] = pYaV[x];
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 - quaterPad; x += 16)
Packit 1fb8d4
		{
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x4_t u = vld4q_u8(&pU[4 * x]);
Packit 1fb8d4
				u.val[0] = vld1q_u8(&pUaU[x]);
Packit 1fb8d4
				u.val[2] = vld1q_u8(&pVaU[x]);
Packit 1fb8d4
				vst4q_u8(&pU[4 * x], u);
Packit 1fb8d4
			}
Packit 1fb8d4
			{
Packit 1fb8d4
				uint8x16x4_t v = vld4q_u8(&pV[4 * x]);
Packit 1fb8d4
				v.val[0] = vld1q_u8(&pUaV[x]);
Packit 1fb8d4
				v.val[2] = vld1q_u8(&pVaV[x]);
Packit 1fb8d4
				vst4q_u8(&pV[4 * x], v);
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (; x < quaterWidth; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			pU[4 * x + 0] = pUaU[x];
Packit 1fb8d4
			pV[4 * x + 0] = pUaV[x];
Packit 1fb8d4
			pU[4 * x + 2] = pVaU[x];
Packit 1fb8d4
			pV[4 * x + 2] = pVaV[x];
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return neon_ChromaFilter(pDst, dstStep, roi);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static pstatus_t neon_YUV420CombineToYUV444(avc444_frame_type type, const BYTE* const pSrc[3],
Packit Service 5a9772
                                            const UINT32 srcStep[3], UINT32 nWidth, UINT32 nHeight,
Packit Service 5a9772
                                            BYTE* pDst[3], const UINT32 dstStep[3],
Packit Service 5a9772
                                            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 neon_LumaToYUV444(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		case AVC444_CHROMAv1:
Packit 1fb8d4
			return neon_ChromaV1ToYUV444(pSrc, srcStep, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		case AVC444_CHROMAv2:
Packit 1fb8d4
			return neon_ChromaV2ToYUV444(pSrc, srcStep, nWidth, nHeight, pDst, dstStep, roi);
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void primitives_init_YUV_opt(primitives_t* prims)
Packit 1fb8d4
{
Packit 1fb8d4
	generic = primitives_get_generic();
Packit 1fb8d4
	primitives_init_YUV(prims);
Packit 1fb8d4
Packit 1fb8d4
	if (IsProcessorFeaturePresent(PF_ARM_NEON_INSTRUCTIONS_AVAILABLE))
Packit 1fb8d4
	{
Packit 1fb8d4
		prims->YUV420ToRGB_8u_P3AC4R = neon_YUV420ToRGB_8u_P3AC4R;
Packit 1fb8d4
		prims->YUV444ToRGB_8u_P3AC4R = neon_YUV444ToRGB_8u_P3AC4R;
Packit 1fb8d4
		prims->YUV420CombineToYUV444 = neon_YUV420CombineToYUV444;
Packit 1fb8d4
	}
Packit 1fb8d4
}