1/*** 2 This file is part of PulseAudio. 3 4 PulseAudio is free software; you can redistribute it and/or modify 5 it under the terms of the GNU Lesser General Public License as published 6 by the Free Software Foundation; either version 2.1 of the License, 7 or (at your option) any later version. 8 9 PulseAudio is distributed in the hope that it will be useful, but 10 WITHOUT ANY WARRANTY; without even the implied warranty of 11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 12 General Public License for more details. 13 14 You should have received a copy of the GNU Lesser General Public License 15 along with PulseAudio; if not, see <http://www.gnu.org/licenses/>. 16***/ 17 18#ifdef HAVE_CONFIG_H 19#include <config.h> 20#endif 21 22#include <check.h> 23 24#include <pulsecore/cpu.h> 25#include <pulsecore/cpu-arm.h> 26#include <pulsecore/random.h> 27#include <pulsecore/macro.h> 28#include <pulsecore/mix.h> 29 30#include "runtime-test-util.h" 31 32#define SAMPLES 1028 33#define TIMES 1000 34#define TIMES2 100 35 36static void acquire_mix_streams(pa_mix_info streams[], unsigned nstreams) { 37 unsigned i; 38 39 for (i = 0; i < nstreams; i++) 40 streams[i].ptr = pa_memblock_acquire_chunk(&streams[i].chunk); 41} 42 43static void release_mix_streams(pa_mix_info streams[], unsigned nstreams) { 44 unsigned i; 45 46 for (i = 0; i < nstreams; i++) 47 pa_memblock_release(streams[i].chunk.memblock); 48} 49 50static void run_mix_test( 51 pa_do_mix_func_t func, 52 pa_do_mix_func_t orig_func, 53 int align, 54 int channels, 55 bool correct, 56 bool perf) { 57 58 PA_DECLARE_ALIGNED(8, int16_t, in0[SAMPLES * 4]) = { 0 }; 59 PA_DECLARE_ALIGNED(8, int16_t, in1[SAMPLES * 4]) = { 0 }; 60 PA_DECLARE_ALIGNED(8, int16_t, out[SAMPLES * 4]) = { 0 }; 61 PA_DECLARE_ALIGNED(8, int16_t, out_ref[SAMPLES * 4]) = { 0 }; 62 int16_t *samples0, *samples1; 63 int16_t *samples, *samples_ref; 64 int nsamples; 65 pa_mempool *pool; 66 pa_memchunk c0, c1; 67 pa_mix_info m[2]; 68 int i; 69 70 pa_assert(channels == 1 || channels == 2 || channels == 4); 71 72 /* Force sample alignment as requested */ 73 samples0 = in0 + (8 - align); 74 samples1 = in1 + (8 - align); 75 samples = out + (8 - align); 76 samples_ref = out_ref + (8 - align); 77 nsamples = channels * (SAMPLES - (8 - align)); 78 79 fail_unless((pool = pa_mempool_new(PA_MEM_TYPE_PRIVATE, 0, true)) != NULL); 80 81 pa_random(samples0, nsamples * sizeof(int16_t)); 82 c0.memblock = pa_memblock_new_fixed(pool, samples0, nsamples * sizeof(int16_t), false); 83 c0.length = pa_memblock_get_length(c0.memblock); 84 c0.index = 0; 85 86 pa_random(samples1, nsamples * sizeof(int16_t)); 87 c1.memblock = pa_memblock_new_fixed(pool, samples1, nsamples * sizeof(int16_t), false); 88 c1.length = pa_memblock_get_length(c1.memblock); 89 c1.index = 0; 90 91 m[0].chunk = c0; 92 m[0].volume.channels = channels; 93 for (i = 0; i < channels; i++) { 94 m[0].volume.values[i] = PA_VOLUME_NORM; 95 m[0].linear[i].i = 0x5555; 96 } 97 98 m[1].chunk = c1; 99 m[1].volume.channels = channels; 100 for (i = 0; i < channels; i++) { 101 m[1].volume.values[i] = PA_VOLUME_NORM; 102 m[1].linear[i].i = 0x6789; 103 } 104 105 if (correct) { 106 acquire_mix_streams(m, 2); 107 orig_func(m, 2, channels, samples_ref, nsamples * sizeof(int16_t)); 108 release_mix_streams(m, 2); 109 110 acquire_mix_streams(m, 2); 111 func(m, 2, channels, samples, nsamples * sizeof(int16_t)); 112 release_mix_streams(m, 2); 113 114 for (i = 0; i < nsamples; i++) { 115 if (samples[i] != samples_ref[i]) { 116 pa_log_debug("Correctness test failed: align=%d, channels=%d", align, channels); 117 pa_log_debug("%d: %hd != %04hd (%hd + %hd)", 118 i, 119 samples[i], samples_ref[i], 120 samples0[i], samples1[i]); 121 ck_abort(); 122 } 123 } 124 } 125 126 if (perf) { 127 pa_log_debug("Testing %d-channel mixing performance with %d sample alignment", channels, align); 128 129 PA_RUNTIME_TEST_RUN_START("func", TIMES, TIMES2) { 130 acquire_mix_streams(m, 2); 131 func(m, 2, channels, samples, nsamples * sizeof(int16_t)); 132 release_mix_streams(m, 2); 133 } PA_RUNTIME_TEST_RUN_STOP 134 135 PA_RUNTIME_TEST_RUN_START("orig", TIMES, TIMES2) { 136 acquire_mix_streams(m, 2); 137 orig_func(m, 2, channels, samples_ref, nsamples * sizeof(int16_t)); 138 release_mix_streams(m, 2); 139 } PA_RUNTIME_TEST_RUN_STOP 140 } 141 142 pa_memblock_unref(c0.memblock); 143 pa_memblock_unref(c1.memblock); 144 145 pa_mempool_unref(pool); 146} 147 148START_TEST (mix_special_test) { 149 pa_cpu_info cpu_info = { PA_CPU_UNDEFINED, {}, false }; 150 pa_do_mix_func_t orig_func, special_func; 151 152 cpu_info.force_generic_code = true; 153 pa_mix_func_init(&cpu_info); 154 orig_func = pa_get_mix_func(PA_SAMPLE_S16NE); 155 156 cpu_info.force_generic_code = false; 157 pa_mix_func_init(&cpu_info); 158 special_func = pa_get_mix_func(PA_SAMPLE_S16NE); 159 160 pa_log_debug("Checking special mix (s16, stereo)"); 161 run_mix_test(special_func, orig_func, 7, 2, true, true); 162 163 pa_log_debug("Checking special mix (s16, 4-channel)"); 164 run_mix_test(special_func, orig_func, 7, 4, true, true); 165 166 pa_log_debug("Checking special mix (s16, mono)"); 167 run_mix_test(special_func, orig_func, 7, 1, true, true); 168} 169END_TEST 170 171#if defined (__arm__) && defined (__linux__) && defined (HAVE_NEON) 172START_TEST (mix_neon_test) { 173 pa_do_mix_func_t orig_func, neon_func; 174 pa_cpu_arm_flag_t flags = 0; 175 176 pa_cpu_get_arm_flags(&flags); 177 178 if (!(flags & PA_CPU_ARM_NEON)) { 179 pa_log_info("NEON not supported. Skipping"); 180 return; 181 } 182 183 orig_func = pa_get_mix_func(PA_SAMPLE_S16NE); 184 pa_mix_func_init_neon(flags); 185 neon_func = pa_get_mix_func(PA_SAMPLE_S16NE); 186 187 pa_log_debug("Checking NEON mix (s16, stereo)"); 188 run_mix_test(neon_func, orig_func, 7, 2, true, true); 189 190 pa_log_debug("Checking NEON mix (s16, 4-channel)"); 191 run_mix_test(neon_func, orig_func, 7, 4, true, true); 192 193 pa_log_debug("Checking NEON mix (s16, mono)"); 194 run_mix_test(neon_func, orig_func, 7, 1, true, true); 195} 196END_TEST 197#endif /* defined (__arm__) && defined (__linux__) && defined (HAVE_NEON) */ 198 199int main(int argc, char *argv[]) { 200 int failed = 0; 201 Suite *s; 202 TCase *tc; 203 SRunner *sr; 204 205 if (!getenv("MAKE_CHECK")) 206 pa_log_set_level(PA_LOG_DEBUG); 207 208 s = suite_create("CPU"); 209 210 tc = tcase_create("mix"); 211 tcase_add_test(tc, mix_special_test); 212#if defined (__arm__) && defined (__linux__) && defined (HAVE_NEON) 213 tcase_add_test(tc, mix_neon_test); 214#endif 215 tcase_set_timeout(tc, 120); 216 suite_add_tcase(s, tc); 217 218 sr = srunner_create(s); 219 srunner_run_all(sr, CK_NORMAL); 220 failed = srunner_ntests_failed(sr); 221 srunner_free(sr); 222 223 return (failed == 0) ? EXIT_SUCCESS : EXIT_FAILURE; 224} 225