23#include <freerdp/config.h> 
   25#include <winpr/sysinfo.h> 
   27#include <freerdp/types.h> 
   28#include <freerdp/primitives.h> 
   30#include "prim_internal.h" 
   33#if defined(NEON_INTRINSICS_ENABLED) 
   38static inline uint8x8_t neon_YUV2R_single(uint16x8_t C, int16x8_t D, int16x8_t E)
 
   41  const int32x4_t Ch = vreinterpretq_s32_u32(vmovl_u16(vget_high_u16(C)));
 
   42  const int32x4_t e403h = vmull_n_s16(vget_high_s16(E), 403);
 
   43  const int32x4_t cehm = vaddq_s32(Ch, e403h);
 
   44  const int32x4_t ceh = vshrq_n_s32(cehm, 8);
 
   46  const int32x4_t Cl = vreinterpretq_s32_u32(vmovl_u16(vget_low_u16(C)));
 
   47  const int32x4_t e403l = vmull_n_s16(vget_low_s16(E), 403);
 
   48  const int32x4_t celm = vaddq_s32(Cl, e403l);
 
   49  const int32x4_t cel = vshrq_n_s32(celm, 8);
 
   50  const int16x8_t ce = vcombine_s16(vqmovn_s32(cel), vqmovn_s32(ceh));
 
   51  return vqmovun_s16(ce);
 
   54static inline uint8x8x2_t neon_YUV2R(uint16x8x2_t C, int16x8x2_t D, int16x8x2_t E)
 
   56  uint8x8x2_t res = { { neon_YUV2R_single(C.val[0], D.val[0], E.val[0]),
 
   57                      neon_YUV2R_single(C.val[1], D.val[1], E.val[1]) } };
 
   61static inline uint8x8_t neon_YUV2G_single(uint16x8_t C, int16x8_t D, int16x8_t E)
 
   64  const int16x8_t d48 = vmulq_n_s16(D, 48);
 
   65  const int16x8_t e120 = vmulq_n_s16(E, 120);
 
   66  const int32x4_t deh = vaddl_s16(vget_high_s16(d48), vget_high_s16(e120));
 
   67  const int32x4_t Ch = vreinterpretq_s32_u32(vmovl_u16(vget_high_u16(C)));
 
   68  const int32x4_t cdeh32m = vsubq_s32(Ch, deh);
 
   69  const int32x4_t cdeh32 = vshrq_n_s32(cdeh32m, 8);
 
   70  const int16x4_t cdeh = vqmovn_s32(cdeh32);
 
   72  const int32x4_t del = vaddl_s16(vget_low_s16(d48), vget_low_s16(e120));
 
   73  const int32x4_t Cl = vreinterpretq_s32_u32(vmovl_u16(vget_low_u16(C)));
 
   74  const int32x4_t cdel32m = vsubq_s32(Cl, del);
 
   75  const int32x4_t cdel32 = vshrq_n_s32(cdel32m, 8);
 
   76  const int16x4_t cdel = vqmovn_s32(cdel32);
 
   77  const int16x8_t cde = vcombine_s16(cdel, cdeh);
 
   78  return vqmovun_s16(cde);
 
   81static inline uint8x8x2_t neon_YUV2G(uint16x8x2_t C, int16x8x2_t D, int16x8x2_t E)
 
   83  uint8x8x2_t res = { { neon_YUV2G_single(C.val[0], D.val[0], E.val[0]),
 
   84                      neon_YUV2G_single(C.val[1], D.val[1], E.val[1]) } };
 
   88static inline uint8x8_t neon_YUV2B_single(uint16x8_t C, int16x8_t D, int16x8_t E)
 
   91  const int32x4_t Ch = vreinterpretq_s32_u32(vmovl_u16(vget_high_u16(C)));
 
   92  const int32x4_t d475h = vmull_n_s16(vget_high_s16(D), 475);
 
   93  const int32x4_t cdhm = vaddq_s32(Ch, d475h);
 
   94  const int32x4_t cdh = vshrq_n_s32(cdhm, 8);
 
   96  const int32x4_t Cl = vreinterpretq_s32_u32(vmovl_u16(vget_low_u16(C)));
 
   97  const int32x4_t d475l = vmull_n_s16(vget_low_s16(D), 475);
 
   98  const int32x4_t cdlm = vaddq_s32(Cl, d475l);
 
   99  const int32x4_t cdl = vshrq_n_s32(cdlm, 8);
 
  100  const int16x8_t cd = vcombine_s16(vqmovn_s32(cdl), vqmovn_s32(cdh));
 
  101  return vqmovun_s16(cd);
 
  104static inline uint8x8x2_t neon_YUV2B(uint16x8x2_t C, int16x8x2_t D, int16x8x2_t E)
 
  106  uint8x8x2_t res = { { neon_YUV2B_single(C.val[0], D.val[0], E.val[0]),
 
  107                      neon_YUV2B_single(C.val[1], D.val[1], E.val[1]) } };
 
  111static inline void neon_store_bgrx(BYTE* WINPR_RESTRICT pRGB, uint8x8_t r, uint8x8_t g, uint8x8_t b,
 
  112                                   uint8_t rPos, uint8_t gPos, uint8_t bPos, uint8_t aPos)
 
  114  uint8x8x4_t bgrx = vld4_u8(pRGB);
 
  121static inline void neon_YuvToRgbPixel(BYTE* pRGB, uint8x8x2_t Y, int16x8x2_t D, int16x8x2_t E,
 
  122                                      const uint8_t rPos, 
const uint8_t gPos, 
const uint8_t bPos,
 
  126  const uint16x8x2_t C = { { vshlq_n_u16(vmovl_u8(Y.val[0]), 8),
 
  127                           vshlq_n_u16(vmovl_u8(Y.val[1]), 8) } };
 
  129  const uint8x8x2_t r = neon_YUV2R(C, D, E);
 
  130  const uint8x8x2_t g = neon_YUV2G(C, D, E);
 
  131  const uint8x8x2_t b = neon_YUV2B(C, D, E);
 
  133  neon_store_bgrx(pRGB, r.val[0], g.val[0], b.val[0], rPos, gPos, bPos, aPos);
 
  134  neon_store_bgrx(pRGB + 
sizeof(uint8x8x4_t), r.val[1], g.val[1], b.val[1], rPos, gPos, bPos,
 
  138static inline int16x8x2_t loadUV(
const BYTE* WINPR_RESTRICT pV, 
size_t x)
 
  140  const uint8x8_t Vraw = vld1_u8(&pV[x / 2]);
 
  141  const int16x8_t V = vreinterpretq_s16_u16(vmovl_u8(Vraw));
 
  142  const int16x8_t c128 = vdupq_n_s16(128);
 
  143  const int16x8_t E = vsubq_s16(V, c128);
 
  144  return vzipq_s16(E, E);
 
  147static inline void neon_write_pixel(BYTE* pRGB, BYTE Y, BYTE U, BYTE V, 
const uint8_t rPos,
 
  148                                    const uint8_t gPos, 
const uint8_t bPos, 
const uint8_t aPos)
 
  150  const BYTE r = YUV2R(Y, U, V);
 
  151  const BYTE g = YUV2G(Y, U, V);
 
  152  const BYTE b = YUV2B(Y, U, V);
 
  159static inline void neon_YUV420ToX_DOUBLE_ROW(
const BYTE* WINPR_RESTRICT pY[2],
 
  160                                             const BYTE* WINPR_RESTRICT pU,
 
  161                                             const BYTE* WINPR_RESTRICT pV,
 
  162                                             BYTE* WINPR_RESTRICT pRGB[2], 
size_t width,
 
  163                                             const uint8_t rPos, 
const uint8_t gPos,
 
  164                                             const uint8_t bPos, 
const uint8_t aPos)
 
  168  for (; x < width - width % 16; x += 16)
 
  170    const uint8x16_t Y0raw = vld1q_u8(&pY[0][x]);
 
  171    const uint8x8x2_t Y0 = { { vget_low_u8(Y0raw), vget_high_u8(Y0raw) } };
 
  172    const int16x8x2_t D = loadUV(pU, x);
 
  173    const int16x8x2_t E = loadUV(pV, x);
 
  174    neon_YuvToRgbPixel(&pRGB[0][4ULL * x], Y0, D, E, rPos, gPos, bPos, aPos);
 
  176    const uint8x16_t Y1raw = vld1q_u8(&pY[1][x]);
 
  177    const uint8x8x2_t Y1 = { { vget_low_u8(Y1raw), vget_high_u8(Y1raw) } };
 
  178    neon_YuvToRgbPixel(&pRGB[1][4ULL * x], Y1, D, E, rPos, gPos, bPos, aPos);
 
  181  for (; x < width - width % 2; x += 2)
 
  183    const BYTE U = pU[x / 2];
 
  184    const BYTE V = pV[x / 2];
 
  186    neon_write_pixel(&pRGB[0][4 * x], pY[0][x], U, V, rPos, gPos, bPos, aPos);
 
  187    neon_write_pixel(&pRGB[0][4 * (1ULL + x)], pY[0][1ULL + x], U, V, rPos, gPos, bPos, aPos);
 
  188    neon_write_pixel(&pRGB[1][4 * x], pY[1][x], U, V, rPos, gPos, bPos, aPos);
 
  189    neon_write_pixel(&pRGB[1][4 * (1ULL + x)], pY[1][1ULL + x], U, V, rPos, gPos, bPos, aPos);
 
  192  for (; x < width; x++)
 
  194    const BYTE U = pU[x / 2];
 
  195    const BYTE V = pV[x / 2];
 
  197    neon_write_pixel(&pRGB[0][4 * x], pY[0][x], U, V, rPos, gPos, bPos, aPos);
 
  198    neon_write_pixel(&pRGB[1][4 * x], pY[1][x], U, V, rPos, gPos, bPos, aPos);
 
  202static inline void neon_YUV420ToX_SINGLE_ROW(
const BYTE* WINPR_RESTRICT pY,
 
  203                                             const BYTE* WINPR_RESTRICT pU,
 
  204                                             const BYTE* WINPR_RESTRICT pV,
 
  205                                             BYTE* WINPR_RESTRICT pRGB, 
size_t width,
 
  206                                             const uint8_t rPos, 
const uint8_t gPos,
 
  207                                             const uint8_t bPos, 
const uint8_t aPos)
 
  211  for (; x < width - width % 16; x += 16)
 
  213    const uint8x16_t Y0raw = vld1q_u8(&pY[x]);
 
  214    const uint8x8x2_t Y0 = { { vget_low_u8(Y0raw), vget_high_u8(Y0raw) } };
 
  215    const int16x8x2_t D = loadUV(pU, x);
 
  216    const int16x8x2_t E = loadUV(pV, x);
 
  217    neon_YuvToRgbPixel(&pRGB[4ULL * x], Y0, D, E, rPos, gPos, bPos, aPos);
 
  220  for (; x < width - width % 2; x += 2)
 
  222    const BYTE U = pU[x / 2];
 
  223    const BYTE V = pV[x / 2];
 
  225    neon_write_pixel(&pRGB[4 * x], pY[x], U, V, rPos, gPos, bPos, aPos);
 
  226    neon_write_pixel(&pRGB[4 * (1ULL + x)], pY[1ULL + x], U, V, rPos, gPos, bPos, aPos);
 
  228  for (; x < width; x++)
 
  230    const BYTE U = pU[x / 2];
 
  231    const BYTE V = pV[x / 2];
 
  233    neon_write_pixel(&pRGB[4 * x], pY[x], U, V, rPos, gPos, bPos, aPos);
 
  237static inline pstatus_t neon_YUV420ToX(
const BYTE* WINPR_RESTRICT pSrc[3], 
const UINT32 srcStep[3],
 
  238                                       BYTE* WINPR_RESTRICT pDst, UINT32 dstStep,
 
  239                                       const prim_size_t* WINPR_RESTRICT roi, 
const uint8_t rPos,
 
  240                                       const uint8_t gPos, 
const uint8_t bPos, 
const uint8_t aPos)
 
  242  const UINT32 nWidth = roi->width;
 
  243  const UINT32 nHeight = roi->height;
 
  245  WINPR_ASSERT(nHeight > 0);
 
  247  for (; y < (nHeight - 1); y += 2)
 
  249    const uint8_t* pY[2] = { pSrc[0] + y * srcStep[0], pSrc[0] + (1ULL + y) * srcStep[0] };
 
  250    const uint8_t* pU = pSrc[1] + (y / 2) * srcStep[1];
 
  251    const uint8_t* pV = pSrc[2] + (y / 2) * srcStep[2];
 
  252    uint8_t* pRGB[2] = { pDst + y * dstStep, pDst + (1ULL + y) * dstStep };
 
  254    neon_YUV420ToX_DOUBLE_ROW(pY, pU, pV, pRGB, nWidth, rPos, gPos, bPos, aPos);
 
  256  for (; y < nHeight; y++)
 
  258    const uint8_t* pY = pSrc[0] + y * srcStep[0];
 
  259    const uint8_t* pU = pSrc[1] + (y / 2) * srcStep[1];
 
  260    const uint8_t* pV = pSrc[2] + (y / 2) * srcStep[2];
 
  261    uint8_t* pRGB = pDst + y * dstStep;
 
  263    neon_YUV420ToX_SINGLE_ROW(pY, pU, pV, pRGB, nWidth, rPos, gPos, bPos, aPos);
 
  265  return PRIMITIVES_SUCCESS;
 
  268static pstatus_t neon_YUV420ToRGB_8u_P3AC4R(
const BYTE* WINPR_RESTRICT pSrc[3],
 
  269                                            const UINT32 srcStep[3], BYTE* WINPR_RESTRICT pDst,
 
  270                                            UINT32 dstStep, UINT32 DstFormat,
 
  275    case PIXEL_FORMAT_BGRA32:
 
  276    case PIXEL_FORMAT_BGRX32:
 
  277      return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 2, 1, 0, 3);
 
  279    case PIXEL_FORMAT_RGBA32:
 
  280    case PIXEL_FORMAT_RGBX32:
 
  281      return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 0, 1, 2, 3);
 
  283    case PIXEL_FORMAT_ARGB32:
 
  284    case PIXEL_FORMAT_XRGB32:
 
  285      return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 1, 2, 3, 0);
 
  287    case PIXEL_FORMAT_ABGR32:
 
  288    case PIXEL_FORMAT_XBGR32:
 
  289      return neon_YUV420ToX(pSrc, srcStep, pDst, dstStep, roi, 3, 2, 1, 0);
 
  292      return generic->YUV420ToRGB_8u_P3AC4R(pSrc, srcStep, pDst, dstStep, DstFormat, roi);
 
  296static inline int16x8_t loadUVreg(uint8x8_t Vraw)
 
  298  const int16x8_t V = vreinterpretq_s16_u16(vmovl_u8(Vraw));
 
  299  const int16x8_t c128 = vdupq_n_s16(128);
 
  300  const int16x8_t E = vsubq_s16(V, c128);
 
  304static inline int16x8x2_t loadUV444(uint8x16_t Vld)
 
  306  const uint8x8x2_t V = { { vget_low_u8(Vld), vget_high_u8(Vld) } };
 
  307  const int16x8x2_t res = { {
 
  314static inline void avgUV(BYTE U[2][2])
 
  316  const BYTE u00 = U[0][0];
 
  317  const INT16 umul = (INT16)u00 << 2;
 
  318  const INT16 sum = (INT16)U[0][1] + U[1][0] + U[1][1];
 
  319  const INT16 wavg = umul - sum;
 
  320  const BYTE val = CONDITIONAL_CLIP(wavg, u00);
 
  324static inline void neon_avgUV(uint8x16_t pU[2])
 
  328  const uint8x16x2_t usplit = vuzpq_u8(pU[0], pU[1]);
 
  329  const uint8x16_t ueven = usplit.val[0];
 
  330  const uint8x16_t uodd = usplit.val[1];
 
  332  const uint8x8_t u00 = vget_low_u8(ueven);
 
  333  const uint8x8_t u01 = vget_low_u8(uodd);
 
  334  const uint8x8_t u10 = vget_high_u8(ueven);
 
  335  const uint8x8_t u11 = vget_high_u8(uodd);
 
  338  const uint16x8_t uoddsum = vaddl_u8(u01, u10);
 
  339  const uint16x8_t usum = vaddq_u16(uoddsum, vmovl_u8(u11));
 
  342  const uint16x8_t umul = vshll_n_u8(u00, 2);
 
  345  const int16x8_t wavg = vsubq_s16(vreinterpretq_s16_u16(umul), vreinterpretq_s16_u16(usum));
 
  346  const uint8x8_t avg = vqmovun_s16(wavg);
 
  349  const uint8x8_t absdiff = vabd_u8(avg, u00);
 
  352  const uint8x8_t mask = vclt_u8(absdiff, vdup_n_u8(30));
 
  355  const uint8x8_t out1 = vand_u8(u00, mask);
 
  358  const uint8x8_t notmask = vmvn_u8(mask);
 
  361  const uint8x8_t out2 = vand_u8(avg, notmask);
 
  364  const uint8x8_t out = vorr_u8(out1, out2);
 
  366  const uint8x8x2_t ua = vzip_u8(out, u01);
 
  367  const uint8x16_t u = vcombine_u8(ua.val[0], ua.val[1]);
 
  371static inline pstatus_t neon_YUV444ToX_SINGLE_ROW(
const BYTE* WINPR_RESTRICT pY,
 
  372                                                  const BYTE* WINPR_RESTRICT pU,
 
  373                                                  const BYTE* WINPR_RESTRICT pV,
 
  374                                                  BYTE* WINPR_RESTRICT pRGB, 
size_t width,
 
  375                                                  const uint8_t rPos, 
const uint8_t gPos,
 
  376                                                  const uint8_t bPos, 
const uint8_t aPos)
 
  378  WINPR_ASSERT(width % 2 == 0);
 
  382  for (; x < width - width % 16; x += 16)
 
  384    uint8x16_t U = vld1q_u8(&pU[x]);
 
  385    uint8x16_t V = vld1q_u8(&pV[x]);
 
  386    const uint8x16_t Y0raw = vld1q_u8(&pY[x]);
 
  387    const uint8x8x2_t Y0 = { { vget_low_u8(Y0raw), vget_high_u8(Y0raw) } };
 
  388    const int16x8x2_t D0 = loadUV444(U);
 
  389    const int16x8x2_t E0 = loadUV444(V);
 
  390    neon_YuvToRgbPixel(&pRGB[4ULL * x], Y0, D0, E0, rPos, gPos, bPos, aPos);
 
  393  for (; x < width; x += 2)
 
  395    BYTE* rgb = &pRGB[x * 4];
 
  397    for (
size_t j = 0; j < 2; j++)
 
  399      const BYTE y = pY[x + j];
 
  400      const BYTE u = pU[x + j];
 
  401      const BYTE v = pV[x + j];
 
  403      neon_write_pixel(&rgb[4 * (j)], y, u, v, rPos, gPos, bPos, aPos);
 
  407  return PRIMITIVES_SUCCESS;
 
  410static inline pstatus_t neon_YUV444ToX_DOUBLE_ROW(
const BYTE* WINPR_RESTRICT pY[2],
 
  411                                                  const BYTE* WINPR_RESTRICT pU[2],
 
  412                                                  const BYTE* WINPR_RESTRICT pV[2],
 
  413                                                  BYTE* WINPR_RESTRICT pRGB[2], 
size_t width,
 
  414                                                  const uint8_t rPos, 
const uint8_t gPos,
 
  415                                                  const uint8_t bPos, 
const uint8_t aPos)
 
  417  WINPR_ASSERT(width % 2 == 0);
 
  421  for (; x < width - width % 16; x += 16)
 
  423    uint8x16_t U[2] = { vld1q_u8(&pU[0][x]), vld1q_u8(&pU[1][x]) };
 
  426    uint8x16_t V[2] = { vld1q_u8(&pV[0][x]), vld1q_u8(&pV[1][x]) };
 
  429    const uint8x16_t Y0raw = vld1q_u8(&pY[0][x]);
 
  430    const uint8x8x2_t Y0 = { { vget_low_u8(Y0raw), vget_high_u8(Y0raw) } };
 
  431    const int16x8x2_t D0 = loadUV444(U[0]);
 
  432    const int16x8x2_t E0 = loadUV444(V[0]);
 
  433    neon_YuvToRgbPixel(&pRGB[0][4ULL * x], Y0, D0, E0, rPos, gPos, bPos, aPos);
 
  435    const uint8x16_t Y1raw = vld1q_u8(&pY[1][x]);
 
  436    const uint8x8x2_t Y1 = { { vget_low_u8(Y1raw), vget_high_u8(Y1raw) } };
 
  437    const int16x8x2_t D1 = loadUV444(U[1]);
 
  438    const int16x8x2_t E1 = loadUV444(V[1]);
 
  439    neon_YuvToRgbPixel(&pRGB[1][4ULL * x], Y1, D1, E1, rPos, gPos, bPos, aPos);
 
  442  for (; x < width; x += 2)
 
  444    BYTE* rgb[2] = { &pRGB[0][x * 4], &pRGB[1][x * 4] };
 
  445    BYTE U[2][2] = { { pU[0][x], pU[0][x + 1] }, { pU[1][x], pU[1][x + 1] } };
 
  448    BYTE V[2][2] = { { pV[0][x], pV[0][x + 1] }, { pV[1][x], pV[1][x + 1] } };
 
  451    for (
size_t i = 0; i < 2; i++)
 
  453      for (
size_t j = 0; j < 2; j++)
 
  455        const BYTE y = pY[i][x + j];
 
  456        const BYTE u = U[i][j];
 
  457        const BYTE v = V[i][j];
 
  459        neon_write_pixel(&rgb[i][4 * (j)], y, u, v, rPos, gPos, bPos, aPos);
 
  464  return PRIMITIVES_SUCCESS;
 
  467static inline pstatus_t neon_YUV444ToX(
const BYTE* WINPR_RESTRICT pSrc[3], 
const UINT32 srcStep[3],
 
  468                                       BYTE* WINPR_RESTRICT pDst, UINT32 dstStep,
 
  469                                       const prim_size_t* WINPR_RESTRICT roi, 
const uint8_t rPos,
 
  470                                       const uint8_t gPos, 
const uint8_t bPos, 
const uint8_t aPos)
 
  473  const UINT32 nWidth = roi->width;
 
  474  const UINT32 nHeight = roi->height;
 
  477  for (; y < nHeight - nHeight % 2; y += 2)
 
  479    const uint8_t* WINPR_RESTRICT pY[2] = { pSrc[0] + y * srcStep[0],
 
  480                                          pSrc[0] + (y + 1) * srcStep[0] };
 
  481    const uint8_t* WINPR_RESTRICT pU[2] = { pSrc[1] + y * srcStep[1],
 
  482                                          pSrc[1] + (y + 1) * srcStep[1] };
 
  483    const uint8_t* WINPR_RESTRICT pV[2] = { pSrc[2] + y * srcStep[2],
 
  484                                          pSrc[2] + (y + 1) * srcStep[2] };
 
  486    uint8_t* WINPR_RESTRICT pRGB[2] = { &pDst[y * dstStep], &pDst[(y + 1) * dstStep] };
 
  489        neon_YUV444ToX_DOUBLE_ROW(pY, pU, pV, pRGB, nWidth, rPos, gPos, bPos, aPos);
 
  490    if (rc != PRIMITIVES_SUCCESS)
 
  493  for (; y < nHeight; y++)
 
  495    const uint8_t* WINPR_RESTRICT pY = pSrc[0] + y * srcStep[0];
 
  496    const uint8_t* WINPR_RESTRICT pU = pSrc[1] + y * srcStep[1];
 
  497    const uint8_t* WINPR_RESTRICT pV = pSrc[2] + y * srcStep[2];
 
  498    uint8_t* WINPR_RESTRICT pRGB = &pDst[y * dstStep];
 
  501        neon_YUV444ToX_SINGLE_ROW(pY, pU, pV, pRGB, nWidth, rPos, gPos, bPos, aPos);
 
  502    if (rc != PRIMITIVES_SUCCESS)
 
  506  return PRIMITIVES_SUCCESS;
 
  509static pstatus_t neon_YUV444ToRGB_8u_P3AC4R(
const BYTE* WINPR_RESTRICT pSrc[3],
 
  510                                            const UINT32 srcStep[3], BYTE* WINPR_RESTRICT pDst,
 
  511                                            UINT32 dstStep, UINT32 DstFormat,
 
  516    case PIXEL_FORMAT_BGRA32:
 
  517    case PIXEL_FORMAT_BGRX32:
 
  518      return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 2, 1, 0, 3);
 
  520    case PIXEL_FORMAT_RGBA32:
 
  521    case PIXEL_FORMAT_RGBX32:
 
  522      return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 0, 1, 2, 3);
 
  524    case PIXEL_FORMAT_ARGB32:
 
  525    case PIXEL_FORMAT_XRGB32:
 
  526      return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 1, 2, 3, 0);
 
  528    case PIXEL_FORMAT_ABGR32:
 
  529    case PIXEL_FORMAT_XBGR32:
 
  530      return neon_YUV444ToX(pSrc, srcStep, pDst, dstStep, roi, 3, 2, 1, 0);
 
  533      return generic->YUV444ToRGB_8u_P3AC4R(pSrc, srcStep, pDst, dstStep, DstFormat, roi);
 
  537static pstatus_t neon_LumaToYUV444(
const BYTE* WINPR_RESTRICT pSrcRaw[3], 
const UINT32 srcStep[3],
 
  538                                   BYTE* WINPR_RESTRICT pDstRaw[3], 
const UINT32 dstStep[3],
 
  541  const UINT32 nWidth = roi->right - roi->left;
 
  542  const UINT32 nHeight = roi->bottom - roi->top;
 
  543  const UINT32 halfWidth = (nWidth + 1) / 2;
 
  544  const UINT32 halfHeight = (nHeight + 1) / 2;
 
  545  const UINT32 evenY = 0;
 
  546  const BYTE* pSrc[3] = { pSrcRaw[0] + roi->top * srcStep[0] + roi->left,
 
  547                        pSrcRaw[1] + roi->top / 2 * srcStep[1] + roi->left / 2,
 
  548                        pSrcRaw[2] + roi->top / 2 * srcStep[2] + roi->left / 2 };
 
  549  BYTE* pDst[3] = { pDstRaw[0] + roi->top * dstStep[0] + roi->left,
 
  550                  pDstRaw[1] + roi->top * dstStep[1] + roi->left,
 
  551                  pDstRaw[2] + roi->top * dstStep[2] + roi->left };
 
  555  for (UINT32 y = 0; y < nHeight; y++)
 
  557    const BYTE* Ym = pSrc[0] + srcStep[0] * y;
 
  558    BYTE* pY = pDst[0] + dstStep[0] * y;
 
  559    memcpy(pY, Ym, nWidth);
 
  564  for (UINT32 y = 0; y < halfHeight; y++)
 
  566    const UINT32 val2y = (2 * y + evenY);
 
  567    const BYTE* Um = pSrc[1] + srcStep[1] * y;
 
  568    const BYTE* Vm = pSrc[2] + srcStep[2] * y;
 
  569    BYTE* pU = pDst[1] + dstStep[1] * val2y;
 
  570    BYTE* pV = pDst[2] + dstStep[2] * val2y;
 
  571    BYTE* pU1 = pU + dstStep[1];
 
  572    BYTE* pV1 = pV + dstStep[2];
 
  575    for (; x + 16 < halfWidth; x += 16)
 
  578        const uint8x16_t u = vld1q_u8(Um);
 
  589        const uint8x16_t v = vld1q_u8(Vm);
 
  601    for (; x < halfWidth; x++)
 
  603      const BYTE u = *Um++;
 
  604      const BYTE v = *Vm++;
 
  616  return PRIMITIVES_SUCCESS;
 
  619static pstatus_t neon_ChromaV1ToYUV444(
const BYTE* WINPR_RESTRICT pSrcRaw[3],
 
  620                                       const UINT32 srcStep[3], BYTE* WINPR_RESTRICT pDstRaw[3],
 
  621                                       const UINT32 dstStep[3],
 
  624  const UINT32 mod = 16;
 
  627  const UINT32 nWidth = roi->right - roi->left;
 
  628  const UINT32 nHeight = roi->bottom - roi->top;
 
  629  const UINT32 halfWidth = (nWidth) / 2;
 
  630  const UINT32 halfHeight = (nHeight) / 2;
 
  631  const UINT32 oddY = 1;
 
  632  const UINT32 evenY = 0;
 
  633  const UINT32 oddX = 1;
 
  636  const UINT32 padHeigth = nHeight + 16 - nHeight % 16;
 
  637  const UINT32 halfPad = halfWidth % 16;
 
  638  const BYTE* pSrc[3] = { pSrcRaw[0] + roi->top * srcStep[0] + roi->left,
 
  639                        pSrcRaw[1] + roi->top / 2 * srcStep[1] + roi->left / 2,
 
  640                        pSrcRaw[2] + roi->top / 2 * srcStep[2] + roi->left / 2 };
 
  641  BYTE* pDst[3] = { pDstRaw[0] + roi->top * dstStep[0] + roi->left,
 
  642                  pDstRaw[1] + roi->top * dstStep[1] + roi->left,
 
  643                  pDstRaw[2] + roi->top * dstStep[2] + roi->left };
 
  647  for (UINT32 y = 0; y < padHeigth; y++)
 
  649    const BYTE* Ya = pSrc[0] + srcStep[0] * y;
 
  652    if ((y) % mod < (mod + 1) / 2)
 
  654      const UINT32 pos = (2 * uY++ + oddY);
 
  659      pX = pDst[1] + dstStep[1] * pos;
 
  663      const UINT32 pos = (2 * vY++ + oddY);
 
  668      pX = pDst[2] + dstStep[2] * pos;
 
  671    memcpy(pX, Ya, nWidth);
 
  675  for (UINT32 y = 0; y < halfHeight; y++)
 
  677    const UINT32 val2y = (y * 2 + evenY);
 
  678    const BYTE* Ua = pSrc[1] + srcStep[1] * y;
 
  679    const BYTE* Va = pSrc[2] + srcStep[2] * y;
 
  680    BYTE* pU = pDst[1] + dstStep[1] * val2y;
 
  681    BYTE* pV = pDst[2] + dstStep[2] * val2y;
 
  684    for (; x < halfWidth - halfPad; x += 16)
 
  687        uint8x16x2_t u = vld2q_u8(&pU[2 * x]);
 
  688        u.val[1] = vld1q_u8(&Ua[x]);
 
  689        vst2q_u8(&pU[2 * x], u);
 
  692        uint8x16x2_t v = vld2q_u8(&pV[2 * x]);
 
  693        v.val[1] = vld1q_u8(&Va[x]);
 
  694        vst2q_u8(&pV[2 * x], v);
 
  698    for (; x < halfWidth; x++)
 
  700      const UINT32 val2x1 = (x * 2 + oddX);
 
  706  return PRIMITIVES_SUCCESS;
 
  709static pstatus_t neon_ChromaV2ToYUV444(
const BYTE* WINPR_RESTRICT pSrc[3], 
const UINT32 srcStep[3],
 
  710                                       UINT32 nTotalWidth, UINT32 nTotalHeight,
 
  711                                       BYTE* WINPR_RESTRICT pDst[3], 
const UINT32 dstStep[3],
 
  714  const UINT32 nWidth = roi->right - roi->left;
 
  715  const UINT32 nHeight = roi->bottom - roi->top;
 
  716  const UINT32 halfWidth = (nWidth + 1) / 2;
 
  717  const UINT32 halfPad = halfWidth % 16;
 
  718  const UINT32 halfHeight = (nHeight + 1) / 2;
 
  719  const UINT32 quaterWidth = (nWidth + 3) / 4;
 
  720  const UINT32 quaterPad = quaterWidth % 16;
 
  723  for (UINT32 y = 0; y < nHeight; y++)
 
  725    const UINT32 yTop = y + roi->top;
 
  726    const BYTE* pYaU = pSrc[0] + srcStep[0] * yTop + roi->left / 2;
 
  727    const BYTE* pYaV = pYaU + nTotalWidth / 2;
 
  728    BYTE* pU = pDst[1] + dstStep[1] * yTop + roi->left;
 
  729    BYTE* pV = pDst[2] + dstStep[2] * yTop + roi->left;
 
  732    for (; x < halfWidth - halfPad; x += 16)
 
  735        uint8x16x2_t u = vld2q_u8(&pU[2 * x]);
 
  736        u.val[1] = vld1q_u8(&pYaU[x]);
 
  737        vst2q_u8(&pU[2 * x], u);
 
  740        uint8x16x2_t v = vld2q_u8(&pV[2 * x]);
 
  741        v.val[1] = vld1q_u8(&pYaV[x]);
 
  742        vst2q_u8(&pV[2 * x], v);
 
  746    for (; x < halfWidth; x++)
 
  748      const UINT32 odd = 2 * x + 1;
 
  755  for (UINT32 y = 0; y < halfHeight; y++)
 
  757    const BYTE* pUaU = pSrc[1] + srcStep[1] * (y + roi->top / 2) + roi->left / 4;
 
  758    const BYTE* pUaV = pUaU + nTotalWidth / 4;
 
  759    const BYTE* pVaU = pSrc[2] + srcStep[2] * (y + roi->top / 2) + roi->left / 4;
 
  760    const BYTE* pVaV = pVaU + nTotalWidth / 4;
 
  761    BYTE* pU = pDst[1] + dstStep[1] * (2 * y + 1 + roi->top) + roi->left;
 
  762    BYTE* pV = pDst[2] + dstStep[2] * (2 * y + 1 + roi->top) + roi->left;
 
  765    for (; x < quaterWidth - quaterPad; x += 16)
 
  768        uint8x16x4_t u = vld4q_u8(&pU[4 * x]);
 
  769        u.val[0] = vld1q_u8(&pUaU[x]);
 
  770        u.val[2] = vld1q_u8(&pVaU[x]);
 
  771        vst4q_u8(&pU[4 * x], u);
 
  774        uint8x16x4_t v = vld4q_u8(&pV[4 * x]);
 
  775        v.val[0] = vld1q_u8(&pUaV[x]);
 
  776        v.val[2] = vld1q_u8(&pVaV[x]);
 
  777        vst4q_u8(&pV[4 * x], v);
 
  781    for (; x < quaterWidth; x++)
 
  783      pU[4 * x + 0] = pUaU[x];
 
  784      pV[4 * x + 0] = pUaV[x];
 
  785      pU[4 * x + 2] = pVaU[x];
 
  786      pV[4 * x + 2] = pVaV[x];
 
  790  return PRIMITIVES_SUCCESS;
 
  793static pstatus_t neon_YUV420CombineToYUV444(avc444_frame_type type,
 
  794                                            const BYTE* WINPR_RESTRICT pSrc[3],
 
  795                                            const UINT32 srcStep[3], UINT32 nWidth, UINT32 nHeight,
 
  796                                            BYTE* WINPR_RESTRICT pDst[3], 
const UINT32 dstStep[3],
 
  799  if (!pSrc || !pSrc[0] || !pSrc[1] || !pSrc[2])
 
  802  if (!pDst || !pDst[0] || !pDst[1] || !pDst[2])
 
  811      return neon_LumaToYUV444(pSrc, srcStep, pDst, dstStep, roi);
 
  813    case AVC444_CHROMAv1:
 
  814      return neon_ChromaV1ToYUV444(pSrc, srcStep, pDst, dstStep, roi);
 
  816    case AVC444_CHROMAv2:
 
  817      return neon_ChromaV2ToYUV444(pSrc, srcStep, nWidth, nHeight, pDst, dstStep, roi);
 
  825void primitives_init_YUV_neon_int(
primitives_t* WINPR_RESTRICT prims)
 
  827#if defined(NEON_INTRINSICS_ENABLED) 
  828  generic = primitives_get_generic();
 
  829  WLog_VRB(PRIM_TAG, 
"NEON optimizations");
 
  830  prims->YUV420ToRGB_8u_P3AC4R = neon_YUV420ToRGB_8u_P3AC4R;
 
  831  prims->YUV444ToRGB_8u_P3AC4R = neon_YUV444ToRGB_8u_P3AC4R;
 
  832  prims->YUV420CombineToYUV444 = neon_YUV420CombineToYUV444;
 
  834  WLog_VRB(PRIM_TAG, 
"undefined WITH_SIMD or neon intrinsics not available");