Blame src/dsp/neon.h

Packit 9c6abc
// Copyright 2014 Google Inc. All Rights Reserved.
Packit 9c6abc
//
Packit 9c6abc
// Use of this source code is governed by a BSD-style license
Packit 9c6abc
// that can be found in the COPYING file in the root of the source
Packit 9c6abc
// tree. An additional intellectual property rights grant can be found
Packit 9c6abc
// in the file PATENTS. All contributing project authors may
Packit 9c6abc
// be found in the AUTHORS file in the root of the source tree.
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
//
Packit 9c6abc
//  NEON common code.
Packit 9c6abc
Packit 9c6abc
#ifndef WEBP_DSP_NEON_H_
Packit 9c6abc
#define WEBP_DSP_NEON_H_
Packit 9c6abc
Packit 9c6abc
#include <arm_neon.h>
Packit 9c6abc
Packit 9c6abc
#include "src/dsp/dsp.h"
Packit 9c6abc
Packit 9c6abc
// Right now, some intrinsics functions seem slower, so we disable them
Packit 9c6abc
// everywhere except newer clang/gcc or aarch64 where the inline assembly is
Packit 9c6abc
// incompatible.
Packit 9c6abc
#if LOCAL_CLANG_PREREQ(3,8) || LOCAL_GCC_PREREQ(4,9) || defined(__aarch64__)
Packit 9c6abc
#define WEBP_USE_INTRINSICS   // use intrinsics when possible
Packit 9c6abc
#endif
Packit 9c6abc
Packit 9c6abc
#define INIT_VECTOR2(v, a, b) do {  \
Packit 9c6abc
  v.val[0] = a;                     \
Packit 9c6abc
  v.val[1] = b;                     \
Packit 9c6abc
} while (0)
Packit 9c6abc
Packit 9c6abc
#define INIT_VECTOR3(v, a, b, c) do {  \
Packit 9c6abc
  v.val[0] = a;                        \
Packit 9c6abc
  v.val[1] = b;                        \
Packit 9c6abc
  v.val[2] = c;                        \
Packit 9c6abc
} while (0)
Packit 9c6abc
Packit 9c6abc
#define INIT_VECTOR4(v, a, b, c, d) do {  \
Packit 9c6abc
  v.val[0] = a;                           \
Packit 9c6abc
  v.val[1] = b;                           \
Packit 9c6abc
  v.val[2] = c;                           \
Packit 9c6abc
  v.val[3] = d;                           \
Packit 9c6abc
} while (0)
Packit 9c6abc
Packit 9c6abc
// if using intrinsics, this flag avoids some functions that make gcc-4.6.3
Packit 9c6abc
// crash ("internal compiler error: in immed_double_const, at emit-rtl.").
Packit 9c6abc
// (probably similar to gcc.gnu.org/bugzilla/show_bug.cgi?id=48183)
Packit 9c6abc
#if !(LOCAL_CLANG_PREREQ(3,8) || LOCAL_GCC_PREREQ(4,8) || defined(__aarch64__))
Packit 9c6abc
#define WORK_AROUND_GCC
Packit 9c6abc
#endif
Packit 9c6abc
Packit 9c6abc
static WEBP_INLINE int32x4x4_t Transpose4x4_NEON(const int32x4x4_t rows) {
Packit 9c6abc
  uint64x2x2_t row01, row23;
Packit 9c6abc
Packit 9c6abc
  row01.val[0] = vreinterpretq_u64_s32(rows.val[0]);
Packit 9c6abc
  row01.val[1] = vreinterpretq_u64_s32(rows.val[1]);
Packit 9c6abc
  row23.val[0] = vreinterpretq_u64_s32(rows.val[2]);
Packit 9c6abc
  row23.val[1] = vreinterpretq_u64_s32(rows.val[3]);
Packit 9c6abc
  // Transpose 64-bit values (there's no vswp equivalent)
Packit 9c6abc
  {
Packit 9c6abc
    const uint64x1_t row0h = vget_high_u64(row01.val[0]);
Packit 9c6abc
    const uint64x1_t row2l = vget_low_u64(row23.val[0]);
Packit 9c6abc
    const uint64x1_t row1h = vget_high_u64(row01.val[1]);
Packit 9c6abc
    const uint64x1_t row3l = vget_low_u64(row23.val[1]);
Packit 9c6abc
    row01.val[0] = vcombine_u64(vget_low_u64(row01.val[0]), row2l);
Packit 9c6abc
    row23.val[0] = vcombine_u64(row0h, vget_high_u64(row23.val[0]));
Packit 9c6abc
    row01.val[1] = vcombine_u64(vget_low_u64(row01.val[1]), row3l);
Packit 9c6abc
    row23.val[1] = vcombine_u64(row1h, vget_high_u64(row23.val[1]));
Packit 9c6abc
  }
Packit 9c6abc
  {
Packit 9c6abc
    const int32x4x2_t out01 = vtrnq_s32(vreinterpretq_s32_u64(row01.val[0]),
Packit 9c6abc
                                        vreinterpretq_s32_u64(row01.val[1]));
Packit 9c6abc
    const int32x4x2_t out23 = vtrnq_s32(vreinterpretq_s32_u64(row23.val[0]),
Packit 9c6abc
                                        vreinterpretq_s32_u64(row23.val[1]));
Packit 9c6abc
    int32x4x4_t out;
Packit 9c6abc
    out.val[0] = out01.val[0];
Packit 9c6abc
    out.val[1] = out01.val[1];
Packit 9c6abc
    out.val[2] = out23.val[0];
Packit 9c6abc
    out.val[3] = out23.val[1];
Packit 9c6abc
    return out;
Packit 9c6abc
  }
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
#if 0     // Useful debug macro.
Packit 9c6abc
#include <stdio.h>
Packit 9c6abc
#define PRINT_REG(REG, SIZE) do {                       \
Packit 9c6abc
  int i;                                                \
Packit 9c6abc
  printf("%s \t[%d]: 0x", #REG, SIZE);                  \
Packit 9c6abc
  if (SIZE == 8) {                                      \
Packit 9c6abc
    uint8_t _tmp[8];                                    \
Packit 9c6abc
    vst1_u8(_tmp, (REG));                               \
Packit 9c6abc
    for (i = 0; i < 8; ++i) printf("%.2x ", _tmp[i]);   \
Packit 9c6abc
  } else if (SIZE == 16) {                              \
Packit 9c6abc
    uint16_t _tmp[4];                                   \
Packit 9c6abc
    vst1_u16(_tmp, (REG));                              \
Packit 9c6abc
    for (i = 0; i < 4; ++i) printf("%.4x ", _tmp[i]);   \
Packit 9c6abc
  }                                                     \
Packit 9c6abc
  printf("\n");                                         \
Packit 9c6abc
} while (0)
Packit 9c6abc
#endif
Packit 9c6abc
Packit 9c6abc
#endif  // WEBP_DSP_NEON_H_