1 /*
2 **
3 ** Copyright 2010, The Android Open Source Project
4 **
5 ** Licensed under the Apache License, Version 2.0 (the "License");
6 ** you may not use this file except in compliance with the License.
7 ** You may obtain a copy of the License at
8 **
9 **     http://www.apache.org/licenses/LICENSE-2.0
10 **
11 ** Unless required by applicable law or agreed to in writing, software
12 ** distributed under the License is distributed on an "AS IS" BASIS,
13 ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 ** See the License for the specific language governing permissions and
15 ** limitations under the License.
16 */
17 
18 
19 //#define LOG_NDEBUG 0
20 #define LOG_TAG "Visualizer"
21 #include <utils/Log.h>
22 
23 #include <stdint.h>
24 #include <sys/types.h>
25 #include <limits.h>
26 
27 #include <audio_utils/fixedfft.h>
28 #include <utils/Thread.h>
29 
30 #include <android/content/AttributionSourceState.h>
31 
32 #include "Visualizer.h"
33 
34 namespace android {
35 
36 // ---------------------------------------------------------------------------
37 
Visualizer(const android::content::AttributionSourceState & attributionSource)38 Visualizer::Visualizer (const android::content::AttributionSourceState& attributionSource)
39         :   AudioEffect(attributionSource)
40 {
41 }
42 
~Visualizer()43 Visualizer::~Visualizer()
44 {
45     ALOGV("Visualizer::~Visualizer()");
46     setEnabled(false);
47     setCaptureCallBack(NULL, NULL, 0, 0);
48 }
49 
set(int32_t priority,legacy_callback_t cbf,void * user,audio_session_t sessionId,audio_io_handle_t io,const AudioDeviceTypeAddr & device,bool probe)50 status_t Visualizer::set(int32_t priority,
51                          legacy_callback_t cbf,
52                          void* user,
53                          audio_session_t sessionId,
54                          audio_io_handle_t io,
55                          const AudioDeviceTypeAddr& device,
56                          bool probe)
57 {
58     status_t status = AudioEffect::set(
59             SL_IID_VISUALIZATION, nullptr, priority, cbf, user, sessionId, io, device, probe);
60     if (status == NO_ERROR || status == ALREADY_EXISTS) {
61         status = initCaptureSize();
62         if (status == NO_ERROR) initSampleRate();
63     }
64     return status;
65 }
66 
67 
release()68 void Visualizer::release()
69 {
70     ALOGV("Visualizer::release()");
71     setEnabled(false);
72     Mutex::Autolock _l(mCaptureLock);
73 
74     mCaptureThread.clear();
75     mCaptureCallBack = NULL;
76     mCaptureCbkUser = NULL;
77     mCaptureFlags = 0;
78     mCaptureRate = 0;
79 }
80 
setEnabled(bool enabled)81 status_t Visualizer::setEnabled(bool enabled)
82 {
83     Mutex::Autolock _l(mCaptureLock);
84 
85     sp<CaptureThread> t = mCaptureThread;
86     if (t != 0) {
87         if (enabled) {
88             if (t->exitPending()) {
89                 mCaptureLock.unlock();
90                 if (t->requestExitAndWait() == WOULD_BLOCK) {
91                     mCaptureLock.lock();
92                     ALOGE("Visualizer::enable() called from thread");
93                     return INVALID_OPERATION;
94                 }
95                 mCaptureLock.lock();
96             }
97         }
98         t->mLock.lock();
99     }
100 
101     status_t status = AudioEffect::setEnabled(enabled);
102 
103     if (t != 0) {
104         if (enabled && status == NO_ERROR) {
105             t->run("Visualizer");
106         } else {
107             t->requestExit();
108         }
109     }
110 
111     if (t != 0) {
112         t->mLock.unlock();
113     }
114 
115     return status;
116 }
117 
setCaptureCallBack(capture_cbk_t cbk,void * user,uint32_t flags,uint32_t rate)118 status_t Visualizer::setCaptureCallBack(capture_cbk_t cbk, void* user, uint32_t flags,
119         uint32_t rate)
120 {
121     if (rate > CAPTURE_RATE_MAX) {
122         return BAD_VALUE;
123     }
124     Mutex::Autolock _l(mCaptureLock);
125 
126     if (mEnabled) {
127         return INVALID_OPERATION;
128     }
129 
130     if (mCaptureThread != 0) {
131         sp<CaptureThread> t = mCaptureThread;
132         mCaptureLock.unlock();
133         t->requestExitAndWait();
134         mCaptureLock.lock();
135     }
136 
137     mCaptureThread.clear();
138     mCaptureCallBack = cbk;
139     mCaptureCbkUser = user;
140     mCaptureFlags = flags;
141     mCaptureRate = rate;
142 
143     if (cbk != NULL) {
144         mCaptureThread = sp<CaptureThread>::make(
145                 sp<Visualizer>::fromExisting(this), rate, ((flags & CAPTURE_CALL_JAVA) != 0));
146     }
147     ALOGV("setCaptureCallBack() rate: %d thread %p flags 0x%08x",
148             rate, mCaptureThread.get(), mCaptureFlags);
149     return NO_ERROR;
150 }
151 
setCaptureSize(uint32_t size)152 status_t Visualizer::setCaptureSize(uint32_t size)
153 {
154     if (!isCaptureSizeValid(size)) {
155         ALOGE("%s with invalid capture size %u from HAL", __func__, size);
156         return BAD_VALUE;
157     }
158 
159     Mutex::Autolock _l(mCaptureLock);
160     if (mEnabled) {
161         return INVALID_OPERATION;
162     }
163 
164     uint32_t buf32[sizeof(effect_param_t) / sizeof(uint32_t) + 2];
165     effect_param_t *p = (effect_param_t *)buf32;
166 
167     p->psize = sizeof(uint32_t);
168     p->vsize = sizeof(uint32_t);
169     *(int32_t *)p->data = VISUALIZER_PARAM_CAPTURE_SIZE;
170     *((int32_t *)p->data + 1)= size;
171     status_t status = setParameter(p);
172 
173     ALOGV("setCaptureSize size %u status %d p->status %d", size, status, p->status);
174 
175     if (status == NO_ERROR) {
176         status = p->status;
177         if (status == NO_ERROR) {
178             mCaptureSize = size;
179         }
180     }
181 
182     return status;
183 }
184 
setScalingMode(uint32_t mode)185 status_t Visualizer::setScalingMode(uint32_t mode) {
186     if ((mode != VISUALIZER_SCALING_MODE_NORMALIZED)
187             && (mode != VISUALIZER_SCALING_MODE_AS_PLAYED)) {
188         return BAD_VALUE;
189     }
190 
191     Mutex::Autolock _l(mCaptureLock);
192 
193     uint32_t buf32[sizeof(effect_param_t) / sizeof(uint32_t) + 2];
194     effect_param_t *p = (effect_param_t *)buf32;
195 
196     p->psize = sizeof(uint32_t);
197     p->vsize = sizeof(uint32_t);
198     *(int32_t *)p->data = VISUALIZER_PARAM_SCALING_MODE;
199     *((int32_t *)p->data + 1)= mode;
200     status_t status = setParameter(p);
201 
202     ALOGV("setScalingMode mode %d  status %d p->status %d", mode, status, p->status);
203 
204     if (status == NO_ERROR) {
205         status = p->status;
206         if (status == NO_ERROR) {
207             mScalingMode = mode;
208         }
209     }
210 
211     return status;
212 }
213 
setMeasurementMode(uint32_t mode)214 status_t Visualizer::setMeasurementMode(uint32_t mode) {
215     if ((mode != MEASUREMENT_MODE_NONE)
216             //Note: needs to be handled as a mask when more measurement modes are added
217             && ((mode & MEASUREMENT_MODE_PEAK_RMS) != mode)) {
218         return BAD_VALUE;
219     }
220 
221     Mutex::Autolock _l(mCaptureLock);
222 
223     uint32_t buf32[sizeof(effect_param_t) / sizeof(uint32_t) + 2];
224     effect_param_t *p = (effect_param_t *)buf32;
225 
226     p->psize = sizeof(uint32_t);
227     p->vsize = sizeof(uint32_t);
228     *(int32_t *)p->data = VISUALIZER_PARAM_MEASUREMENT_MODE;
229     *((int32_t *)p->data + 1)= mode;
230     status_t status = setParameter(p);
231 
232     ALOGV("setMeasurementMode mode %d  status %d p->status %d", mode, status, p->status);
233 
234     if (status == NO_ERROR) {
235         status = p->status;
236         if (status == NO_ERROR) {
237             mMeasurementMode = mode;
238         }
239     }
240     return status;
241 }
242 
getIntMeasurements(uint32_t type,uint32_t number,int32_t * measurements)243 status_t Visualizer::getIntMeasurements(uint32_t type, uint32_t number, int32_t *measurements) {
244     if (mMeasurementMode == MEASUREMENT_MODE_NONE) {
245         ALOGE("Cannot retrieve int measurements, no measurement mode set");
246         return INVALID_OPERATION;
247     }
248     if (!(mMeasurementMode & type)) {
249         // measurement type has not been set on this Visualizer
250         ALOGE("Cannot retrieve int measurements, requested measurement mode 0x%x not set(0x%x)",
251                 type, mMeasurementMode);
252         return INVALID_OPERATION;
253     }
254     // only peak+RMS measurement supported
255     if ((type != MEASUREMENT_MODE_PEAK_RMS)
256             // for peak+RMS measurement, the results are 2 int32_t values
257             || (number != 2)) {
258         ALOGE("Cannot retrieve int measurements, MEASUREMENT_MODE_PEAK_RMS returns 2 ints, not %u",
259               number);
260         return BAD_VALUE;
261     }
262 
263     status_t status = NO_ERROR;
264     if (mEnabled) {
265         uint32_t replySize = number * sizeof(int32_t);
266         status = command(VISUALIZER_CMD_MEASURE,
267                 sizeof(uint32_t)  /*cmdSize*/,
268                 &type /*cmdData*/,
269                 &replySize, measurements);
270         ALOGV("getMeasurements() command returned %d", status);
271         if ((status == NO_ERROR) && (replySize == 0)) {
272             status = NOT_ENOUGH_DATA;
273         }
274     } else {
275         ALOGV("getMeasurements() disabled");
276         return INVALID_OPERATION;
277     }
278     return status;
279 }
280 
getWaveForm(uint8_t * waveform)281 status_t Visualizer::getWaveForm(uint8_t *waveform)
282 {
283     if (waveform == NULL) {
284         return BAD_VALUE;
285     }
286     if (mCaptureSize == 0) {
287         return NO_INIT;
288     }
289 
290     status_t status = NO_ERROR;
291     if (mEnabled) {
292         uint32_t replySize = mCaptureSize;
293         status = command(VISUALIZER_CMD_CAPTURE, 0, NULL, &replySize, waveform);
294         ALOGV("getWaveForm() command returned %d", status);
295         if ((status == NO_ERROR) && (replySize == 0)) {
296             status = NOT_ENOUGH_DATA;
297         }
298     } else {
299         ALOGV("getWaveForm() disabled");
300         memset(waveform, 0x80, mCaptureSize);
301     }
302     return status;
303 }
304 
getFft(uint8_t * fft)305 status_t Visualizer::getFft(uint8_t *fft)
306 {
307     if (fft == NULL) {
308         return BAD_VALUE;
309     }
310     if (mCaptureSize == 0) {
311         return NO_INIT;
312     }
313 
314     status_t status = NO_ERROR;
315     if (mEnabled) {
316         uint8_t buf[mCaptureSize];
317         status = getWaveForm(buf);
318         if (status == NO_ERROR) {
319             status = doFft(fft, buf);
320         }
321     } else {
322         memset(fft, 0, mCaptureSize);
323     }
324     return status;
325 }
326 
doFft(uint8_t * fft,uint8_t * waveform)327 status_t Visualizer::doFft(uint8_t *fft, uint8_t *waveform)
328 {
329     int32_t workspace[mCaptureSize >> 1];
330     int32_t nonzero = 0;
331 
332     for (uint32_t i = 0; i < mCaptureSize; i += 2) {
333         workspace[i >> 1] =
334                 ((waveform[i] ^ 0x80) << 24) | ((waveform[i + 1] ^ 0x80) << 8);
335         nonzero |= workspace[i >> 1];
336     }
337 
338     if (nonzero) {
339         fixed_fft_real(mCaptureSize >> 1, workspace);
340     }
341 
342     for (uint32_t i = 0; i < mCaptureSize; i += 2) {
343         short tmp = workspace[i >> 1] >> 21;
344         while (tmp > 127 || tmp < -128) tmp >>= 1;
345         fft[i] = tmp;
346         tmp = workspace[i >> 1];
347         tmp >>= 5;
348         while (tmp > 127 || tmp < -128) tmp >>= 1;
349         fft[i + 1] = tmp;
350     }
351 
352     return NO_ERROR;
353 }
354 
periodicCapture()355 void Visualizer::periodicCapture()
356 {
357     Mutex::Autolock _l(mCaptureLock);
358     ALOGV("periodicCapture() %p mCaptureCallBack %p mCaptureFlags 0x%08x",
359             this, mCaptureCallBack, mCaptureFlags);
360     if (mCaptureCallBack != NULL &&
361         (mCaptureFlags & (CAPTURE_WAVEFORM|CAPTURE_FFT)) &&
362         mCaptureSize != 0) {
363         uint8_t waveform[mCaptureSize];
364         status_t status = getWaveForm(waveform);
365         if (status != NO_ERROR) {
366             return;
367         }
368         uint8_t fft[mCaptureSize];
369         if (mCaptureFlags & CAPTURE_FFT) {
370             status = doFft(fft, waveform);
371         }
372         if (status != NO_ERROR) {
373             return;
374         }
375         uint8_t *wavePtr = NULL;
376         uint8_t *fftPtr = NULL;
377         uint32_t waveSize = 0;
378         uint32_t fftSize = 0;
379         if (mCaptureFlags & CAPTURE_WAVEFORM) {
380             wavePtr = waveform;
381             waveSize = mCaptureSize;
382         }
383         if (mCaptureFlags & CAPTURE_FFT) {
384             fftPtr = fft;
385             fftSize = mCaptureSize;
386         }
387         mCaptureCallBack(mCaptureCbkUser, waveSize, wavePtr, fftSize, fftPtr, mSampleRate);
388     }
389 }
390 
initCaptureSize()391 status_t Visualizer::initCaptureSize()
392 {
393     uint32_t buf32[sizeof(effect_param_t) / sizeof(uint32_t) + 2];
394     effect_param_t *p = (effect_param_t *)buf32;
395 
396     p->psize = sizeof(uint32_t);
397     p->vsize = sizeof(uint32_t);
398     *(int32_t *)p->data = VISUALIZER_PARAM_CAPTURE_SIZE;
399     status_t status = getParameter(p);
400 
401     if (status == NO_ERROR) {
402         status = p->status;
403     }
404 
405     uint32_t size = 0;
406     if (status != NO_ERROR) {
407         ALOGE("%s getParameter failed status %d", __func__, status);
408         return status;
409     }
410 
411     size = *((int32_t *)p->data + 1);
412     if (!isCaptureSizeValid(size)) {
413         ALOGE("%s with invalid capture size %u from HAL", __func__, size);
414         return BAD_VALUE;
415     }
416 
417     mCaptureSize = size;
418     ALOGV("%s size %u status %d", __func__, mCaptureSize, status);
419     return NO_ERROR;
420 }
421 
initSampleRate()422 void Visualizer::initSampleRate()
423 {
424     audio_config_base_t inputConfig, outputConfig;
425     status_t status = getConfigs(&inputConfig, &outputConfig);
426     if (status == NO_ERROR) {
427         mSampleRate = outputConfig.sample_rate * 1000;
428     }
429     ALOGV("%s sample rate %d status %d", __func__, mSampleRate, status);
430 }
431 
controlStatusChanged(bool controlGranted)432 void Visualizer::controlStatusChanged(bool controlGranted) {
433     if (controlGranted) {
434         // this Visualizer instance regained control of the effect, reset the scaling mode
435         //   and capture size as has been cached through it.
436         ALOGV("controlStatusChanged(true) causes effect parameter reset:");
437         ALOGV("    scaling mode reset to %d", mScalingMode);
438         setScalingMode(mScalingMode);
439         ALOGV("    capture size reset to %d", mCaptureSize);
440         setCaptureSize(mCaptureSize);
441     }
442     AudioEffect::controlStatusChanged(controlGranted);
443 }
444 
445 //-------------------------------------------------------------------------
446 
CaptureThread(const sp<Visualizer> & receiver,uint32_t captureRate,bool bCanCallJava)447 Visualizer::CaptureThread::CaptureThread(const sp<Visualizer>& receiver, uint32_t captureRate,
448         bool bCanCallJava)
449     : Thread(bCanCallJava), mReceiver(receiver)
450 {
451     mSleepTimeUs = 1000000000 / captureRate;
452     ALOGV("CaptureThread cstor %p captureRate %d mSleepTimeUs %d", this, captureRate, mSleepTimeUs);
453 }
454 
threadLoop()455 bool Visualizer::CaptureThread::threadLoop()
456 {
457     ALOGV("CaptureThread %p enter", this);
458     sp<Visualizer> receiver = mReceiver.promote();
459     if (receiver == NULL) {
460         return false;
461     }
462     while (!exitPending())
463     {
464         usleep(mSleepTimeUs);
465         receiver->periodicCapture();
466     }
467     ALOGV("CaptureThread %p exiting", this);
468     return false;
469 }
470 
471 } // namespace android
472