| /* |
| * This file is part of FFmpeg. |
| * |
| * FFmpeg is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * FFmpeg is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License along |
| * with FFmpeg; if not, write to the Free Software Foundation, Inc., |
| * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. |
| */ |
| |
| #include "libavcodec/aacpsdsp.h" |
| #include "libavutil/intfloat.h" |
| |
| #include "checkasm.h" |
| |
| #define N 32 |
| #define STRIDE 128 |
| #define BUF_SIZE (N * STRIDE) |
| |
| #define randomize(buf, len) do { \ |
| int i; \ |
| for (i = 0; i < len; i++) { \ |
| const INTFLOAT f = (INTFLOAT)rnd() / UINT_MAX; \ |
| (buf)[i] = f; \ |
| } \ |
| } while (0) |
| |
| #define EPS 0.005 |
| |
| static void clear_less_significant_bits(INTFLOAT *buf, int len, int bits) |
| { |
| int i; |
| for (i = 0; i < len; i++) { |
| union av_intfloat32 u = { .f = buf[i] }; |
| u.i &= (0xffffffff << bits); |
| buf[i] = u.f; |
| } |
| } |
| |
| static void test_add_squares(void) |
| { |
| LOCAL_ALIGNED_16(INTFLOAT, dst0, [BUF_SIZE]); |
| LOCAL_ALIGNED_16(INTFLOAT, dst1, [BUF_SIZE]); |
| LOCAL_ALIGNED_16(INTFLOAT, src, [BUF_SIZE], [2]); |
| |
| declare_func(void, INTFLOAT *dst, |
| const INTFLOAT (*src)[2], int n); |
| |
| randomize((INTFLOAT *)src, BUF_SIZE * 2); |
| randomize(dst0, BUF_SIZE); |
| memcpy(dst1, dst0, BUF_SIZE * sizeof(INTFLOAT)); |
| call_ref(dst0, src, BUF_SIZE); |
| call_new(dst1, src, BUF_SIZE); |
| if (!float_near_abs_eps_array(dst0, dst1, EPS, BUF_SIZE)) |
| fail(); |
| bench_new(dst1, src, BUF_SIZE); |
| } |
| |
| static void test_mul_pair_single(void) |
| { |
| LOCAL_ALIGNED_16(INTFLOAT, dst0, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, dst1, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, src0, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, src1, [BUF_SIZE]); |
| |
| declare_func(void, INTFLOAT (*dst)[2], |
| INTFLOAT (*src0)[2], INTFLOAT *src1, int n); |
| |
| randomize((INTFLOAT *)src0, BUF_SIZE * 2); |
| randomize(src1, BUF_SIZE); |
| call_ref(dst0, src0, src1, BUF_SIZE); |
| call_new(dst1, src0, src1, BUF_SIZE); |
| if (!float_near_abs_eps_array((float *)dst0, (float *)dst1, EPS, BUF_SIZE * 2)) |
| fail(); |
| bench_new(dst1, src0, src1, BUF_SIZE); |
| } |
| |
| static void test_hybrid_analysis(void) |
| { |
| LOCAL_ALIGNED_16(INTFLOAT, dst0, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, dst1, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, in, [13], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, filter, [N], [8][2]); |
| |
| declare_func(void, INTFLOAT (*out)[2], INTFLOAT (*in)[2], |
| const INTFLOAT (*filter)[8][2], |
| ptrdiff_t stride, int n); |
| |
| randomize((INTFLOAT *)in, 13 * 2); |
| randomize((INTFLOAT *)filter, N * 8 * 2); |
| |
| randomize((INTFLOAT *)dst0, BUF_SIZE * 2); |
| memcpy(dst1, dst0, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| |
| call_ref(dst0, in, filter, STRIDE, N); |
| call_new(dst1, in, filter, STRIDE, N); |
| |
| if (!float_near_abs_eps_array((float *)dst0, (float *)dst1, EPS, BUF_SIZE * 2)) |
| fail(); |
| bench_new(dst1, in, filter, STRIDE, N); |
| } |
| |
| static void test_hybrid_analysis_ileave(void) |
| { |
| LOCAL_ALIGNED_16(INTFLOAT, in, [2], [38][64]); |
| LOCAL_ALIGNED_16(INTFLOAT, out0, [91], [32][2]); |
| LOCAL_ALIGNED_16(INTFLOAT, out1, [91], [32][2]); |
| |
| declare_func(void, INTFLOAT (*out)[32][2], INTFLOAT L[2][38][64], |
| int i, int len); |
| |
| randomize((INTFLOAT *)out0, 91 * 32 * 2); |
| randomize((INTFLOAT *)in, 2 * 38 * 64); |
| memcpy(out1, out0, 91 * 32 * 2 * sizeof(INTFLOAT)); |
| |
| /* len is hardcoded to 32 as that's the only value used in |
| libavcodec. asm functions are likely to be optimized |
| hardcoding this value in their loops and could fail with |
| anything else. |
| i is hardcoded to the two values currently used by the |
| aac decoder because the arm neon implementation is |
| micro-optimized for them and will fail for almost every |
| other value. */ |
| call_ref(out0, in, 3, 32); |
| call_new(out1, in, 3, 32); |
| |
| /* the function just moves data around, so memcmp is enough */ |
| if (memcmp(out0, out1, 91 * 32 * 2 * sizeof(INTFLOAT))) |
| fail(); |
| |
| call_ref(out0, in, 5, 32); |
| call_new(out1, in, 5, 32); |
| |
| if (memcmp(out0, out1, 91 * 32 * 2 * sizeof(INTFLOAT))) |
| fail(); |
| |
| bench_new(out1, in, 3, 32); |
| } |
| |
| static void test_hybrid_synthesis_deint(void) |
| { |
| LOCAL_ALIGNED_16(INTFLOAT, out0, [2], [38][64]); |
| LOCAL_ALIGNED_16(INTFLOAT, out1, [2], [38][64]); |
| LOCAL_ALIGNED_16(INTFLOAT, in, [91], [32][2]); |
| |
| declare_func(void, INTFLOAT out[2][38][64], INTFLOAT (*in)[32][2], |
| int i, int len); |
| |
| randomize((INTFLOAT *)in, 91 * 32 * 2); |
| randomize((INTFLOAT *)out0, 2 * 38 * 64); |
| memcpy(out1, out0, 2 * 38 * 64 * sizeof(INTFLOAT)); |
| |
| /* len is hardcoded to 32 as that's the only value used in |
| libavcodec. asm functions are likely to be optimized |
| hardcoding this value in their loops and could fail with |
| anything else. |
| i is hardcoded to the two values currently used by the |
| aac decoder because the arm neon implementation is |
| micro-optimized for them and will fail for almost every |
| other value. */ |
| call_ref(out0, in, 3, 32); |
| call_new(out1, in, 3, 32); |
| |
| /* the function just moves data around, so memcmp is enough */ |
| if (memcmp(out0, out1, 2 * 38 * 64 * sizeof(INTFLOAT))) |
| fail(); |
| |
| call_ref(out0, in, 5, 32); |
| call_new(out1, in, 5, 32); |
| |
| if (memcmp(out0, out1, 2 * 38 * 64 * sizeof(INTFLOAT))) |
| fail(); |
| |
| bench_new(out1, in, 3, 32); |
| } |
| |
| static void test_stereo_interpolate(PSDSPContext *psdsp) |
| { |
| int i; |
| LOCAL_ALIGNED_16(INTFLOAT, l, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, r, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, l0, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, r0, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, l1, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, r1, [BUF_SIZE], [2]); |
| LOCAL_ALIGNED_16(INTFLOAT, h, [2], [4]); |
| LOCAL_ALIGNED_16(INTFLOAT, h_step, [2], [4]); |
| |
| declare_func(void, INTFLOAT (*l)[2], INTFLOAT (*r)[2], |
| INTFLOAT h[2][4], INTFLOAT h_step[2][4], int len); |
| |
| randomize((INTFLOAT *)l, BUF_SIZE * 2); |
| randomize((INTFLOAT *)r, BUF_SIZE * 2); |
| |
| for (i = 0; i < 2; i++) { |
| if (check_func(psdsp->stereo_interpolate[i], "ps_stereo_interpolate%s", i ? "_ipdopd" : "")) { |
| memcpy(l0, l, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| memcpy(l1, l, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| memcpy(r0, r, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| memcpy(r1, r, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| |
| randomize((INTFLOAT *)h, 2 * 4); |
| randomize((INTFLOAT *)h_step, 2 * 4); |
| // Clear the least significant 14 bits of h_step, to avoid |
| // divergence when accumulating h_step BUF_SIZE times into |
| // a float variable which may or may not have extra intermediate |
| // precision. Therefore clear roughly log2(BUF_SIZE) less |
| // significant bits, to get the same result regardless of any |
| // extra precision in the accumulator. |
| clear_less_significant_bits((INTFLOAT *)h_step, 2 * 4, 14); |
| |
| call_ref(l0, r0, h, h_step, BUF_SIZE); |
| call_new(l1, r1, h, h_step, BUF_SIZE); |
| if (!float_near_abs_eps_array((float *)l0, (float *)l1, EPS, BUF_SIZE * 2) || |
| !float_near_abs_eps_array((float *)r0, (float *)r1, EPS, BUF_SIZE * 2)) |
| fail(); |
| |
| memcpy(l1, l, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| memcpy(r1, r, BUF_SIZE * 2 * sizeof(INTFLOAT)); |
| bench_new(l1, r1, h, h_step, BUF_SIZE); |
| } |
| } |
| } |
| |
| void checkasm_check_aacpsdsp(void) |
| { |
| PSDSPContext psdsp; |
| |
| ff_psdsp_init(&psdsp); |
| |
| if (check_func(psdsp.add_squares, "ps_add_squares")) |
| test_add_squares(); |
| report("add_squares"); |
| |
| if (check_func(psdsp.mul_pair_single, "ps_mul_pair_single")) |
| test_mul_pair_single(); |
| report("mul_pair_single"); |
| |
| if (check_func(psdsp.hybrid_analysis, "ps_hybrid_analysis")) |
| test_hybrid_analysis(); |
| report("hybrid_analysis"); |
| |
| if (check_func(psdsp.hybrid_analysis_ileave, "ps_hybrid_analysis_ileave")) |
| test_hybrid_analysis_ileave(); |
| report("hybrid_analysis_ileave"); |
| |
| if (check_func(psdsp.hybrid_synthesis_deint, "ps_hybrid_synthesis_deint")) |
| test_hybrid_synthesis_deint(); |
| report("hybrid_synthesis_deint"); |
| |
| test_stereo_interpolate(&psdsp); |
| report("stereo_interpolate"); |
| } |