blob: 454e13ba97341b2bb1401f82e9bc1ffec1ee2399 [file] [log] [blame]
kma@webrtc.org0221b782012-09-08 00:09:26 +00001/*
2 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11/* The global function contained in this file initializes SPL function
12 * pointers, currently only for ARM platforms.
13 *
14 * Some code came from common/rtcd.c in the WebM project.
15 */
16
pbos@webrtc.orgaa30bb72013-05-27 09:49:58 +000017#include "webrtc/common_audio/signal_processing/include/real_fft.h"
18#include "webrtc/common_audio/signal_processing/include/signal_processing_library.h"
19#include "webrtc/system_wrappers/interface/cpu_features_wrapper.h"
kma@webrtc.org0221b782012-09-08 00:09:26 +000020
21/* Declare function pointers. */
22MaxAbsValueW16 WebRtcSpl_MaxAbsValueW16;
23MaxAbsValueW32 WebRtcSpl_MaxAbsValueW32;
24MaxValueW16 WebRtcSpl_MaxValueW16;
25MaxValueW32 WebRtcSpl_MaxValueW32;
26MinValueW16 WebRtcSpl_MinValueW16;
27MinValueW32 WebRtcSpl_MinValueW32;
28CrossCorrelation WebRtcSpl_CrossCorrelation;
29DownsampleFast WebRtcSpl_DownsampleFast;
30ScaleAndAddVectorsWithRound WebRtcSpl_ScaleAndAddVectorsWithRound;
kma@webrtc.orgfc8aaf02013-07-24 17:38:23 +000031CreateRealFFT WebRtcSpl_CreateRealFFT;
32FreeRealFFT WebRtcSpl_FreeRealFFT;
kma@webrtc.orgf9e6cc22012-09-21 18:51:12 +000033RealForwardFFT WebRtcSpl_RealForwardFFT;
34RealInverseFFT WebRtcSpl_RealInverseFFT;
kma@webrtc.org0221b782012-09-08 00:09:26 +000035
andrew@webrtc.org5140e242013-02-21 20:12:21 +000036#if (defined(WEBRTC_DETECT_ARM_NEON) || !defined(WEBRTC_ARCH_ARM_NEON)) && \
37 !defined(MIPS32_LE)
kma@webrtc.org0221b782012-09-08 00:09:26 +000038/* Initialize function pointers to the generic C version. */
39static void InitPointersToC() {
40 WebRtcSpl_MaxAbsValueW16 = WebRtcSpl_MaxAbsValueW16C;
41 WebRtcSpl_MaxAbsValueW32 = WebRtcSpl_MaxAbsValueW32C;
42 WebRtcSpl_MaxValueW16 = WebRtcSpl_MaxValueW16C;
43 WebRtcSpl_MaxValueW32 = WebRtcSpl_MaxValueW32C;
44 WebRtcSpl_MinValueW16 = WebRtcSpl_MinValueW16C;
45 WebRtcSpl_MinValueW32 = WebRtcSpl_MinValueW32C;
46 WebRtcSpl_CrossCorrelation = WebRtcSpl_CrossCorrelationC;
47 WebRtcSpl_DownsampleFast = WebRtcSpl_DownsampleFastC;
48 WebRtcSpl_ScaleAndAddVectorsWithRound =
49 WebRtcSpl_ScaleAndAddVectorsWithRoundC;
kma@webrtc.orgfc8aaf02013-07-24 17:38:23 +000050 WebRtcSpl_CreateRealFFT = WebRtcSpl_CreateRealFFTC;
51 WebRtcSpl_FreeRealFFT = WebRtcSpl_FreeRealFFTC;
kma@webrtc.orgf9e6cc22012-09-21 18:51:12 +000052 WebRtcSpl_RealForwardFFT = WebRtcSpl_RealForwardFFTC;
53 WebRtcSpl_RealInverseFFT = WebRtcSpl_RealInverseFFTC;
kma@webrtc.org0221b782012-09-08 00:09:26 +000054}
wu@webrtc.org7aa11892012-10-10 20:59:35 +000055#endif
kma@webrtc.org0221b782012-09-08 00:09:26 +000056
57#if defined(WEBRTC_DETECT_ARM_NEON) || defined(WEBRTC_ARCH_ARM_NEON)
58/* Initialize function pointers to the Neon version. */
59static void InitPointersToNeon() {
60 WebRtcSpl_MaxAbsValueW16 = WebRtcSpl_MaxAbsValueW16Neon;
61 WebRtcSpl_MaxAbsValueW32 = WebRtcSpl_MaxAbsValueW32Neon;
62 WebRtcSpl_MaxValueW16 = WebRtcSpl_MaxValueW16Neon;
63 WebRtcSpl_MaxValueW32 = WebRtcSpl_MaxValueW32Neon;
64 WebRtcSpl_MinValueW16 = WebRtcSpl_MinValueW16Neon;
65 WebRtcSpl_MinValueW32 = WebRtcSpl_MinValueW32Neon;
66 WebRtcSpl_CrossCorrelation = WebRtcSpl_CrossCorrelationNeon;
67 WebRtcSpl_DownsampleFast = WebRtcSpl_DownsampleFastNeon;
68 WebRtcSpl_ScaleAndAddVectorsWithRound =
69 WebRtcSpl_ScaleAndAddVectorsWithRoundNeon;
kma@webrtc.orgfc8aaf02013-07-24 17:38:23 +000070 WebRtcSpl_CreateRealFFT = WebRtcSpl_CreateRealFFTNeon;
71 WebRtcSpl_FreeRealFFT = WebRtcSpl_FreeRealFFTNeon;
kma@webrtc.orgf9e6cc22012-09-21 18:51:12 +000072 WebRtcSpl_RealForwardFFT = WebRtcSpl_RealForwardFFTNeon;
73 WebRtcSpl_RealInverseFFT = WebRtcSpl_RealInverseFFTNeon;
kma@webrtc.org0221b782012-09-08 00:09:26 +000074}
75#endif
76
andrew@webrtc.org5140e242013-02-21 20:12:21 +000077#if defined(MIPS32_LE)
78/* Initialize function pointers to the MIPS version. */
79static void InitPointersToMIPS() {
80 WebRtcSpl_MaxAbsValueW16 = WebRtcSpl_MaxAbsValueW16_mips;
81 WebRtcSpl_MaxValueW16 = WebRtcSpl_MaxValueW16_mips;
82 WebRtcSpl_MaxValueW32 = WebRtcSpl_MaxValueW32_mips;
83 WebRtcSpl_MinValueW16 = WebRtcSpl_MinValueW16_mips;
84 WebRtcSpl_MinValueW32 = WebRtcSpl_MinValueW32_mips;
andrew@webrtc.org8bf755d2013-09-18 17:40:46 +000085 WebRtcSpl_CrossCorrelation = WebRtcSpl_CrossCorrelation_mips;
andrew@webrtc.orgeed919d2013-05-30 16:38:36 +000086 WebRtcSpl_DownsampleFast = WebRtcSpl_DownsampleFast_mips;
kma@webrtc.orgfc8aaf02013-07-24 17:38:23 +000087 WebRtcSpl_CreateRealFFT = WebRtcSpl_CreateRealFFTC;
88 WebRtcSpl_FreeRealFFT = WebRtcSpl_FreeRealFFTC;
andrew@webrtc.org5140e242013-02-21 20:12:21 +000089 WebRtcSpl_RealForwardFFT = WebRtcSpl_RealForwardFFTC;
90 WebRtcSpl_RealInverseFFT = WebRtcSpl_RealInverseFFTC;
91#if defined(MIPS_DSP_R1_LE)
92 WebRtcSpl_MaxAbsValueW32 = WebRtcSpl_MaxAbsValueW32_mips;
andrew@webrtc.org8bf755d2013-09-18 17:40:46 +000093 WebRtcSpl_ScaleAndAddVectorsWithRound =
94 WebRtcSpl_ScaleAndAddVectorsWithRound_mips;
andrew@webrtc.org5140e242013-02-21 20:12:21 +000095#else
96 WebRtcSpl_MaxAbsValueW32 = WebRtcSpl_MaxAbsValueW32C;
andrew@webrtc.org8bf755d2013-09-18 17:40:46 +000097 WebRtcSpl_ScaleAndAddVectorsWithRound =
98 WebRtcSpl_ScaleAndAddVectorsWithRoundC;
andrew@webrtc.org5140e242013-02-21 20:12:21 +000099#endif
100}
101#endif
102
kma@webrtc.org0221b782012-09-08 00:09:26 +0000103static void InitFunctionPointers(void) {
104#if defined(WEBRTC_DETECT_ARM_NEON)
105 if ((WebRtc_GetCPUFeaturesARM() & kCPUFeatureNEON) != 0) {
106 InitPointersToNeon();
107 } else {
108 InitPointersToC();
109 }
110#elif defined(WEBRTC_ARCH_ARM_NEON)
111 InitPointersToNeon();
andrew@webrtc.org5140e242013-02-21 20:12:21 +0000112#elif defined(MIPS32_LE)
113 InitPointersToMIPS();
kma@webrtc.org0221b782012-09-08 00:09:26 +0000114#else
115 InitPointersToC();
116#endif /* WEBRTC_DETECT_ARM_NEON */
117}
118
kma@webrtc.org0221b782012-09-08 00:09:26 +0000119#if defined(WEBRTC_POSIX)
120#include <pthread.h>
121
122static void once(void (*func)(void)) {
123 static pthread_once_t lock = PTHREAD_ONCE_INIT;
124 pthread_once(&lock, func);
125}
126
127#elif defined(_WIN32)
128#include <windows.h>
129
130static void once(void (*func)(void)) {
131 /* Didn't use InitializeCriticalSection() since there's no race-free context
132 * in which to execute it.
133 *
134 * TODO(kma): Change to different implementation (e.g.
135 * InterlockedCompareExchangePointer) to avoid issues similar to
136 * http://code.google.com/p/webm/issues/detail?id=467.
137 */
andrew@webrtc.org63e09642013-01-29 06:45:22 +0000138 static CRITICAL_SECTION lock = {(void *)((size_t)-1), -1, 0, 0, 0, 0};
kma@webrtc.org0221b782012-09-08 00:09:26 +0000139 static int done = 0;
140
141 EnterCriticalSection(&lock);
142 if (!done) {
143 func();
144 done = 1;
145 }
146 LeaveCriticalSection(&lock);
147}
148
149/* There's no fallback version as an #else block here to ensure thread safety.
150 * In case of neither pthread for WEBRTC_POSIX nor _WIN32 is present, build
151 * system should pick it up.
152 */
153#endif /* WEBRTC_POSIX */
154
155void WebRtcSpl_Init() {
156 once(InitFunctionPointers);
157}