AudioTrack.cpp revision 0ca3cf94c0dfc173ad7886ae162c4b67067539f6
199e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten/* 289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** 389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Copyright 2007, The Android Open Source Project 489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** 589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Licensed under the Apache License, Version 2.0 (the "License"); 689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** you may not use this file except in compliance with the License. 789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** You may obtain a copy of the License at 889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** 989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** http://www.apache.org/licenses/LICENSE-2.0 1089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** 1189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Unless required by applicable law or agreed to in writing, software 1289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** distributed under the License is distributed on an "AS IS" BASIS, 1389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** See the License for the specific language governing permissions and 1589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** limitations under the License. 1689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project*/ 1789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project//#define LOG_NDEBUG 0 2089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#define LOG_TAG "AudioTrack" 2189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 2289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <stdint.h> 2389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/types.h> 2489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <limits.h> 2589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 2689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sched.h> 2789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/resource.h> 2889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 2989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <private/media/AudioTrackShared.h> 3089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 3189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/AudioSystem.h> 3289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/AudioTrack.h> 3389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 3489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/Log.h> 357562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/Parcel.h> 367562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/IPCThreadState.h> 3789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/Timers.h> 3838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent#include <utils/Atomic.h> 3989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 40fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin#include <cutils/bitops.h> 41f6b1678f8f508b447155a81b44e214475ab634a8Glenn Kasten#include <cutils/compiler.h> 42fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin 4364760240f931714858a59c1579f07264d7182ba2Dima Zavin#include <system/audio.h> 447394a4f358fa9908a9f0a7c954b65c399f4268e6Dima Zavin#include <system/audio_policy.h> 45fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin 46511754b5839fd9b09fc56b89ae007fbc39084a33Glenn Kasten#include <audio_utils/primitives.h> 47511754b5839fd9b09fc56b89ae007fbc39084a33Glenn Kasten 4889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectnamespace android { 4933005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh// --------------------------------------------------------------------------- 5033005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh 5133005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh// static 5233005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yehstatus_t AudioTrack::getMinFrameCount( 5333005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh int* frameCount, 54fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten audio_stream_type_t streamType, 5533005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh uint32_t sampleRate) 5633005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh{ 5733005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh int afSampleRate; 5833005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh if (AudioSystem::getOutputSamplingRate(&afSampleRate, streamType) != NO_ERROR) { 5933005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh return NO_INIT; 6033005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh } 6133005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh int afFrameCount; 6233005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh if (AudioSystem::getOutputFrameCount(&afFrameCount, streamType) != NO_ERROR) { 6333005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh return NO_INIT; 6433005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh } 6533005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh uint32_t afLatency; 6633005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh if (AudioSystem::getOutputLatency(&afLatency, streamType) != NO_ERROR) { 6733005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh return NO_INIT; 6833005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh } 6933005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh 7033005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh // Ensure that buffer depth covers at least audio hardware latency 7133005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh uint32_t minBufCount = afLatency / ((1000 * afFrameCount) / afSampleRate); 7233005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh if (minBufCount < 2) minBufCount = 2; 7333005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh 7433005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh *frameCount = (sampleRate == 0) ? afFrameCount * minBufCount : 75e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten afFrameCount * minBufCount * sampleRate / afSampleRate; 7633005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh return NO_ERROR; 7733005a932c60a0780fe9b7307d5988df3d9f6c26Chia-chi Yeh} 7889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 7989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// --------------------------------------------------------------------------- 8089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 8189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAudioTrack::AudioTrack() 82879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten : mStatus(NO_INIT), 834ff14bae91075eb274eb1c2975982358946e7e63John Grossman mIsTimed(false), 844ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousPriority(ANDROID_PRIORITY_NORMAL), 854ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousSchedulingGroup(ANDROID_TGROUP_DEFAULT) 8689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 8789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 8889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 8989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAudioTrack::AudioTrack( 90fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten audio_stream_type_t streamType, 9189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t sampleRate, 92e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten audio_format_t format, 930d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi int channelMask, 9489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project int frameCount, 950ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent audio_output_flags_t flags, 9689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project callback_t cbf, 9789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project void* user, 98be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent int notificationFrames, 99be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent int sessionId) 100879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten : mStatus(NO_INIT), 1014ff14bae91075eb274eb1c2975982358946e7e63John Grossman mIsTimed(false), 1024ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousPriority(ANDROID_PRIORITY_NORMAL), 1034ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousSchedulingGroup(ANDROID_TGROUP_DEFAULT) 10489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 1050d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi mStatus = set(streamType, sampleRate, format, channelMask, 106a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent frameCount, flags, cbf, user, notificationFrames, 10717a736c3e1d062d7fc916329eb32aef8935614afGlenn Kasten 0 /*sharedBuffer*/, false /*threadCanCallJava*/, sessionId); 10889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 10989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 11017a736c3e1d062d7fc916329eb32aef8935614afGlenn Kasten// DEPRECATED 11189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAudioTrack::AudioTrack( 112c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int streamType, 113c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber uint32_t sampleRate, 114c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int format, 115c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int channelMask, 116c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int frameCount, 117c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber uint32_t flags, 118c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber callback_t cbf, 119c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber void* user, 120c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int notificationFrames, 121c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber int sessionId) 122c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber : mStatus(NO_INIT), 12318868c5db2f90309c6d11e5837822135e4a0c0faGlenn Kasten mIsTimed(false), 124c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber mPreviousPriority(ANDROID_PRIORITY_NORMAL), mPreviousSchedulingGroup(ANDROID_TGROUP_DEFAULT) 125c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber{ 126c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber mStatus = set((audio_stream_type_t)streamType, sampleRate, (audio_format_t)format, channelMask, 1270ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent frameCount, (audio_output_flags_t)flags, cbf, user, notificationFrames, 12817a736c3e1d062d7fc916329eb32aef8935614afGlenn Kasten 0 /*sharedBuffer*/, false /*threadCanCallJava*/, sessionId); 129c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber} 130c813985abd8ba61e999b3505f6a332574f87a1beAndreas Huber 131c813985abd8ba61e999b3505f6a332574f87a1beAndreas HuberAudioTrack::AudioTrack( 132fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten audio_stream_type_t streamType, 13389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t sampleRate, 134e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten audio_format_t format, 1350d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi int channelMask, 13689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project const sp<IMemory>& sharedBuffer, 1370ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent audio_output_flags_t flags, 13889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project callback_t cbf, 13989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project void* user, 140be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent int notificationFrames, 141be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent int sessionId) 142879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten : mStatus(NO_INIT), 1434ff14bae91075eb274eb1c2975982358946e7e63John Grossman mIsTimed(false), 1444ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousPriority(ANDROID_PRIORITY_NORMAL), 1454ff14bae91075eb274eb1c2975982358946e7e63John Grossman mPreviousSchedulingGroup(ANDROID_TGROUP_DEFAULT) 14689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 1470d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi mStatus = set(streamType, sampleRate, format, channelMask, 14817a736c3e1d062d7fc916329eb32aef8935614afGlenn Kasten 0 /*frameCount*/, flags, cbf, user, notificationFrames, 14917a736c3e1d062d7fc916329eb32aef8935614afGlenn Kasten sharedBuffer, false /*threadCanCallJava*/, sessionId); 15089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 15189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 15289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAudioTrack::~AudioTrack() 15389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 1543856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV_IF(mSharedBuffer != 0, "Destructor sharedBuffer: %p", mSharedBuffer->pointer()); 15589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 15689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mStatus == NO_ERROR) { 15789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Make sure that callback function exits in the case where 15889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // it is looping on buffer full condition in obtainBuffer(). 15989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Otherwise the callback thread will never exit. 16089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project stop(); 16189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mAudioTrackThread != 0) { 16289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrackThread->requestExitAndWait(); 16389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrackThread.clear(); 16489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 16589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrack.clear(); 16689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project IPCThreadState::self()->flushCommands(); 1673a34befc6fb04a4945a849e8bda8b84e4bf973feMarco Nelissen AudioSystem::releaseAudioSessionId(mSessionId); 16889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 16989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 17089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 17189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::set( 172fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten audio_stream_type_t streamType, 17389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t sampleRate, 174e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten audio_format_t format, 1750d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi int channelMask, 17689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project int frameCount, 1770ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent audio_output_flags_t flags, 17889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project callback_t cbf, 17989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project void* user, 18089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project int notificationFrames, 18189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project const sp<IMemory>& sharedBuffer, 182be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent bool threadCanCallJava, 183be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent int sessionId) 18489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 18589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1863856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV_IF(sharedBuffer != 0, "sharedBuffer: %p, size: %d", sharedBuffer->pointer(), sharedBuffer->size()); 18789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1881a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent ALOGV("set() streamType %d frameCount %d flags %04x", streamType, frameCount, flags); 1891a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent 1901703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 1911dd70b9f04961a06fcb73a97fca10a53b3245d3cEric Laurent if (mAudioTrack != 0) { 19229357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Track already in use"); 19389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return INVALID_OPERATION; 19489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 19589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 19689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // handle default values first. 197fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (streamType == AUDIO_STREAM_DEFAULT) { 198fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin streamType = AUDIO_STREAM_MUSIC; 19989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 200ea7939a079b3600cab955760839b021326f8cfc3Glenn Kasten 2011a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent int afSampleRate; 2021a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent if (AudioSystem::getOutputSamplingRate(&afSampleRate, streamType) != NO_ERROR) { 2031a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent return NO_INIT; 2041a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent } 20589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (sampleRate == 0) { 20689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project sampleRate = afSampleRate; 20789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 208ea7939a079b3600cab955760839b021326f8cfc3Glenn Kasten 20989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // these below should probably come from the audioFlinger too... 210e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten if (format == AUDIO_FORMAT_DEFAULT) { 211fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin format = AUDIO_FORMAT_PCM_16_BIT; 21289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 2130d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi if (channelMask == 0) { 2140d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi channelMask = AUDIO_CHANNEL_OUT_STEREO; 21589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 21689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 21789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // validate parameters 218fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (!audio_is_valid_format(format)) { 21929357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Invalid format"); 22089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return BAD_VALUE; 22189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 222c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 223c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent // force direct flag if format is not linear PCM 224fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (!audio_is_linear_pcm(format)) { 2250ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent flags = (audio_output_flags_t) 2260ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent ((flags | AUDIO_OUTPUT_FLAG_DIRECT) & ~AUDIO_OUTPUT_FLAG_FAST); 227c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } 228c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 2290d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi if (!audio_is_output_channel(channelMask)) { 23029357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Invalid channel mask"); 23189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return BAD_VALUE; 23289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 2330d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi uint32_t channelCount = popcount(channelMask); 23489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 235fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin audio_io_handle_t output = AudioSystem::getOutput( 236fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten streamType, 237e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten sampleRate, format, channelMask, 23818868c5db2f90309c6d11e5837822135e4a0c0faGlenn Kasten flags); 23989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 240c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent if (output == 0) { 24129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Could not get audio output for stream type %d", streamType); 242c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent return BAD_VALUE; 24389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 24489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 24534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mVolume[LEFT] = 1.0f; 24634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mVolume[RIGHT] = 1.0f; 24705632a5fa4b88ca474294887fc92a9fcdf0e2352Glenn Kasten mSendLevel = 0.0f; 248d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mFrameCount = frameCount; 249d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mNotificationFramesReq = notificationFrames; 250be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent mSessionId = sessionId; 2512beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent mAuxEffectId = 0; 2524a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten mCbf = cbf; 253be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 25434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent // create the IAudioTrack 2551703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent status_t status = createTrack_l(streamType, 2561703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent sampleRate, 257e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten format, 2580d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi (uint32_t)channelMask, 2591703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent frameCount, 2601703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent flags, 2611703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent sharedBuffer, 262291f4d505aff81969e6666049d3cc3446f836af2Glenn Kasten output); 26389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 26434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (status != NO_ERROR) { 26589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return status; 26689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 26734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent 268a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (cbf != NULL) { 26989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrackThread = new AudioTrackThread(*this, threadCanCallJava); 27089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 27189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 27289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mStatus = NO_ERROR; 27389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 27489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mStreamType = streamType; 275e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten mFormat = format; 2760d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi mChannelMask = (uint32_t)channelMask; 27789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mChannelCount = channelCount; 27889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mSharedBuffer = sharedBuffer; 27989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mMuted = false; 2809a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten mActive = false; 28189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mUserData = user; 28289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mLoopCount = 0; 28389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mMarkerPosition = 0; 2842c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerReached = false; 28589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mNewPosition = 0; 28689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mUpdatePeriod = 0; 287cd07594333cbe8b2c86c6609cce01a74d6cc33f8Jean-Michel Trivi mFlushed = false; 288c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent mFlags = flags; 2893a34befc6fb04a4945a849e8bda8b84e4bf973feMarco Nelissen AudioSystem::acquireAudioSessionId(mSessionId); 290cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent mRestoreStatus = NO_ERROR; 29189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 29289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 29389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 29489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::initCheck() const 29589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 29689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mStatus; 29789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 29889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 29989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 30089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 30189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t AudioTrack::latency() const 30289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 30389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mLatency; 30489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 30589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 306fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kastenaudio_stream_type_t AudioTrack::streamType() const 30789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 30889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mStreamType; 30989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 31089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 311e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kastenaudio_format_t AudioTrack::format() const 31289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 31389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mFormat; 31489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 31589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 31689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectint AudioTrack::channelCount() const 31789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 31889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mChannelCount; 31989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 32089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 32189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t AudioTrack::frameCount() const 32289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 323d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent return mCblk->frameCount; 32489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 32589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 326b9980659501d0428d65d8292f3c32da69d37fbd2Glenn Kastensize_t AudioTrack::frameSize() const 32789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 328fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (audio_is_linear_pcm(mFormat)) { 329671a636931295d9c33ffca74551a804479d01241Eric Laurent return channelCount()*audio_bytes_per_sample(mFormat); 330c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } else { 331c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent return sizeof(uint8_t); 332c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } 33389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 33489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 33589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<IMemory>& AudioTrack::sharedBuffer() 33689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 33789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mSharedBuffer; 33889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 33989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 34089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 34189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 34289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::start() 34389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 34489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project sp<AudioTrackThread> t = mAudioTrackThread; 345d0965dde97f2815ae0a15fe6b40946f8a741a81eGlenn Kasten status_t status = NO_ERROR; 34689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 3473856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("start %p", this); 34889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (t != 0) { 34989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (t->exitPending()) { 35089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (t->requestExitAndWait() == WOULD_BLOCK) { 35129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("AudioTrack::start called from thread"); 35289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return; 35389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 35489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 355e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten } 35689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 357f5aafb209d01ba2ab6cb55d1a12cfc653e2b4be0Eric Laurent AutoMutex lock(mLock); 3581703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // acquire a strong reference on the IMemory and IAudioTrack so that they cannot be destroyed 3591703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // while we are accessing the cblk 360e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IAudioTrack> audioTrack = mAudioTrack; 361e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IMemory> iMem = mCblkMemory; 3621703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent audio_track_cblk_t* cblk = mCblk; 3631703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 3649a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (!mActive) { 365cd07594333cbe8b2c86c6609cce01a74d6cc33f8Jean-Michel Trivi mFlushed = false; 3669a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten mActive = true; 3671703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mNewPosition = cblk->server + mUpdatePeriod; 36833797ea64d067dfeaacbfd7ebe7f3383b73961b5Eric Laurent cblk->lock.lock(); 3691703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->bufferTimeoutMs = MAX_STARTUP_TIMEOUT_MS; 3701703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->waitTimeMs = 0; 37138ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_and(~CBLK_DISABLED_ON, &cblk->flags); 3726dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten pid_t tid; 3732b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent if (t != 0) { 374480b46802bef1371d5caa16ad5454fce04769c57Glenn Kasten t->run("AudioTrack", ANDROID_PRIORITY_AUDIO); 3756dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten tid = t->getTid(); // pid_t is unknown until run() 3766dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten ALOGV("getTid=%d", tid); 3776dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten if (tid == -1) { 3786dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten tid = 0; 3796dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten } 3802b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent } else { 381879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten mPreviousPriority = getpriority(PRIO_PROCESS, 0); 382879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten mPreviousSchedulingGroup = androidGetThreadSchedulingGroup(0); 383879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten androidSetThreadPriority(0, ANDROID_PRIORITY_AUDIO); 3846dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten tid = 0; // not gettid() 3852b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent } 3862b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent 3873856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("start %p before lock cblk %p", this, mCblk); 3881703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (!(cblk->flags & CBLK_INVALID_MSK)) { 3891703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 3906dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten ALOGV("mAudioTrack->start(tid=%d)", tid); 3916dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten status = mAudioTrack->start(tid); 3921703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.lock(); 3931703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (status == DEAD_OBJECT) { 39438ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_or(CBLK_INVALID_ON, &cblk->flags); 3956100d2d60517ff33ed8eb35d0b7ea63cde0831c9Eric Laurent } 3962b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent } 3971703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (cblk->flags & CBLK_INVALID_MSK) { 3981703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent status = restoreTrack_l(cblk, true); 3991703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 4001703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 4012b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent if (status != NO_ERROR) { 4023856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("start() failed"); 4039a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten mActive = false; 40434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (t != 0) { 4052b584244930c9de0e3bc46898a801e9ccb731900Eric Laurent t->requestExit(); 40634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } else { 407879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten setpriority(PRIO_PROCESS, 0, mPreviousPriority); 408879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten androidSetThreadSchedulingGroup(0, mPreviousSchedulingGroup); 40934f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 41089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 41189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 41289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 41389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 41489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 41589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::stop() 41689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 41789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project sp<AudioTrackThread> t = mAudioTrackThread; 41889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 4193856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("stop %p", this); 42089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 421f5aafb209d01ba2ab6cb55d1a12cfc653e2b4be0Eric Laurent AutoMutex lock(mLock); 4229a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (mActive) { 4239a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten mActive = false; 4241dd70b9f04961a06fcb73a97fca10a53b3245d3cEric Laurent mCblk->cv.signal(); 42589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrack->stop(); 42689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Cancel loops (If we are in the middle of a loop, playback 42789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // would not stop until loopCount reaches 0). 4281703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent setLoop_l(0, 0, 0); 4292c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi // the playback head position will reset to 0, so if a marker is set, we need 4302c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi // to activate it again 4312c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerReached = false; 43289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Force flush if a shared buffer is used otherwise audioflinger 43389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // will not stop before end of buffer is reached. 43489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mSharedBuffer != 0) { 4351703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent flush_l(); 43689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 43789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (t != 0) { 43889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project t->requestExit(); 43989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 440879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten setpriority(PRIO_PROCESS, 0, mPreviousPriority); 441879135196fd1c97deefc538c888037c56c2879a7Glenn Kasten androidSetThreadSchedulingGroup(0, mPreviousSchedulingGroup); 44289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 44389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 44489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 44589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 44689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 44789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool AudioTrack::stopped() const 44889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 4499a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten AutoMutex lock(mLock); 4509a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten return stopped_l(); 45189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 45289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 45389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::flush() 45489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 4551703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 4561703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent flush_l(); 4571703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent} 4581703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 4591703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// must be called with mLock held 4601703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurentvoid AudioTrack::flush_l() 4611703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent{ 4623856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("flush"); 463c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 4642c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi // clear playback marker and periodic update counter 4652c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerPosition = 0; 4662c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerReached = false; 4672c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mUpdatePeriod = 0; 468c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 46989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (!mActive) { 470cd07594333cbe8b2c86c6609cce01a74d6cc33f8Jean-Michel Trivi mFlushed = true; 47189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrack->flush(); 47289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Release AudioTrack callback thread in case it was waiting for new buffers 47389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // in AudioTrack::obtainBuffer() 47489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCblk->cv.signal(); 47589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 47689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 47789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 47889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::pause() 47989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 4803856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("pause"); 481f5aafb209d01ba2ab6cb55d1a12cfc653e2b4be0Eric Laurent AutoMutex lock(mLock); 4829a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (mActive) { 4839a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten mActive = false; 48489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrack->pause(); 48589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 48689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 48789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 48889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::mute(bool e) 48989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 49089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mAudioTrack->mute(e); 49189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mMuted = e; 49289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 49389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 49489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool AudioTrack::muted() const 49589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 49689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mMuted; 49789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 49889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 499be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurentstatus_t AudioTrack::setVolume(float left, float right) 50089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 501f0c495012bad92230604a9a12a907812ec49ee8fGlenn Kasten if (left < 0.0f || left > 1.0f || right < 0.0f || right > 1.0f) { 502be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent return BAD_VALUE; 503be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent } 504be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 5051703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 50689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mVolume[LEFT] = left; 50789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mVolume[RIGHT] = right; 50889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 50983d86538c4c479a9225c75ab27938e8f05abb9c8Glenn Kasten mCblk->setVolumeLR((uint32_t(uint16_t(right * 0x1000)) << 16) | uint16_t(left * 0x1000)); 510be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 511be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent return NO_ERROR; 51289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 51389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 514a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenvoid AudioTrack::getVolume(float* left, float* right) const 51589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 516be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent if (left != NULL) { 517be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent *left = mVolume[LEFT]; 518be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent } 519be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent if (right != NULL) { 520be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent *right = mVolume[RIGHT]; 521be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent } 522be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent} 523be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 5242beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t AudioTrack::setAuxEffectSendLevel(float level) 525be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent{ 5263856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("setAuxEffectSendLevel(%f)", level); 52705632a5fa4b88ca474294887fc92a9fcdf0e2352Glenn Kasten if (level < 0.0f || level > 1.0f) { 528be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent return BAD_VALUE; 529be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent } 5301703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 531be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 532be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent mSendLevel = level; 533be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 53405632a5fa4b88ca474294887fc92a9fcdf0e2352Glenn Kasten mCblk->setSendLevel(level); 535be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 536be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent return NO_ERROR; 537be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent} 538be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 539a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenvoid AudioTrack::getAuxEffectSendLevel(float* level) const 540be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent{ 541be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent if (level != NULL) { 542be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent *level = mSendLevel; 543be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent } 54489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 54589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 546573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurentstatus_t AudioTrack::setSampleRate(int rate) 54789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 54889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project int afSamplingRate; 54989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 5504ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mIsTimed) { 5514ff14bae91075eb274eb1c2975982358946e7e63John Grossman return INVALID_OPERATION; 5524ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 5534ff14bae91075eb274eb1c2975982358946e7e63John Grossman 55489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (AudioSystem::getOutputSamplingRate(&afSamplingRate, mStreamType) != NO_ERROR) { 555573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent return NO_INIT; 55689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 55789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Resampler implementation limits input sampling rate to 2 x output sampling rate. 558573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent if (rate <= 0 || rate > afSamplingRate*2 ) return BAD_VALUE; 55989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 5601703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 561573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent mCblk->sampleRate = rate; 562573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent return NO_ERROR; 56389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 56489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 565a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenuint32_t AudioTrack::getSampleRate() const 56689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 5674ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mIsTimed) { 5684ff14bae91075eb274eb1c2975982358946e7e63John Grossman return INVALID_OPERATION; 5694ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 5704ff14bae91075eb274eb1c2975982358946e7e63John Grossman 5711703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 572573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent return mCblk->sampleRate; 57389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 57489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 57589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::setLoop(uint32_t loopStart, uint32_t loopEnd, int loopCount) 57689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 5771703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 5781703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent return setLoop_l(loopStart, loopEnd, loopCount); 5791703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent} 5801703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 5811703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// must be called with mLock held 5821703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurentstatus_t AudioTrack::setLoop_l(uint32_t loopStart, uint32_t loopEnd, int loopCount) 5831703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent{ 58489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audio_track_cblk_t* cblk = mCblk; 58589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 58689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project Mutex::Autolock _l(cblk->lock); 58789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 58889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (loopCount == 0) { 58989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopStart = UINT_MAX; 59089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopEnd = UINT_MAX; 59189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopCount = 0; 59289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mLoopCount = 0; 59389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 59489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 59589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 5964ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mIsTimed) { 5974ff14bae91075eb274eb1c2975982358946e7e63John Grossman return INVALID_OPERATION; 5984ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 5994ff14bae91075eb274eb1c2975982358946e7e63John Grossman 60089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (loopStart >= loopEnd || 6019b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent loopEnd - loopStart > cblk->frameCount || 6029b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent cblk->server > loopStart) { 60329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("setLoop invalid value: loopStart %d, loopEnd %d, loopCount %d, framecount %d, user %d", loopStart, loopEnd, loopCount, cblk->frameCount, cblk->user); 60489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return BAD_VALUE; 60589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 60689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 6079b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent if ((mSharedBuffer != 0) && (loopEnd > cblk->frameCount)) { 60829357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("setLoop invalid value: loop markers beyond data: loopStart %d, loopEnd %d, framecount %d", 609d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent loopStart, loopEnd, cblk->frameCount); 61089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return BAD_VALUE; 611c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } 61289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 61389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopStart = loopStart; 61489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopEnd = loopEnd; 61589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->loopCount = loopCount; 61689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mLoopCount = loopCount; 61789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 61889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 61989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 62089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 62189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::setMarkerPosition(uint32_t marker) 62289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 623a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (mCbf == NULL) return INVALID_OPERATION; 62489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 62589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mMarkerPosition = marker; 6262c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerReached = false; 62789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 62889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 62989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 63089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 631a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenstatus_t AudioTrack::getMarkerPosition(uint32_t *marker) const 63289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 633a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (marker == NULL) return BAD_VALUE; 63489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 63589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project *marker = mMarkerPosition; 63689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 63789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 63889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 63989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 64089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::setPositionUpdatePeriod(uint32_t updatePeriod) 64189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 642a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (mCbf == NULL) return INVALID_OPERATION; 64389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 64489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t curPosition; 64589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project getPosition(&curPosition); 64689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mNewPosition = curPosition + updatePeriod; 64789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mUpdatePeriod = updatePeriod; 64889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 64989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 65089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 65189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 652a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenstatus_t AudioTrack::getPositionUpdatePeriod(uint32_t *updatePeriod) const 65389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 654a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (updatePeriod == NULL) return BAD_VALUE; 65589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 65689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project *updatePeriod = mUpdatePeriod; 65789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 65889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 65989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 66089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 66189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::setPosition(uint32_t position) 66289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 6634ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mIsTimed) return INVALID_OPERATION; 6644ff14bae91075eb274eb1c2975982358946e7e63John Grossman 6651703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 66689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 6679a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (!stopped_l()) return INVALID_OPERATION; 6689a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten 6699a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten Mutex::Autolock _l(mCblk->lock); 67089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 67189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (position > mCblk->user) return BAD_VALUE; 67289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 67389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCblk->server = position; 67438ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_or(CBLK_FORCEREADY_ON, &mCblk->flags); 675c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 67689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 67789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 67889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 67989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::getPosition(uint32_t *position) 68089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 681a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten if (position == NULL) return BAD_VALUE; 6821703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 683cd07594333cbe8b2c86c6609cce01a74d6cc33f8Jean-Michel Trivi *position = mFlushed ? 0 : mCblk->server; 68489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 68589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 68689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 68789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 68889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::reload() 68989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 6901703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 6911703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 6929a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (!stopped_l()) return INVALID_OPERATION; 693c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 6941703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent flush_l(); 69589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 696d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mCblk->stepUser(mCblk->frameCount); 69789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 69889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 69989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 70089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 701c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurentaudio_io_handle_t AudioTrack::getOutput() 702c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent{ 7031703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 7041703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent return getOutput_l(); 7051703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent} 7061703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 7071703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// must be called with mLock held 7081703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurentaudio_io_handle_t AudioTrack::getOutput_l() 7091703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent{ 710fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten return AudioSystem::getOutput(mStreamType, 71118868c5db2f90309c6d11e5837822135e4a0c0faGlenn Kasten mCblk->sampleRate, mFormat, mChannelMask, mFlags); 712c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent} 713c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 714a5224f319e2ba4b51ddb4287705ccf8d4b8ecc51Glenn Kastenint AudioTrack::getSessionId() const 715be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent{ 716be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent return mSessionId; 717be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent} 718be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 719be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurentstatus_t AudioTrack::attachAuxEffect(int effectId) 720be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent{ 7213856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("attachAuxEffect(%d)", effectId); 7222beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent status_t status = mAudioTrack->attachAuxEffect(effectId); 7232beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent if (status == NO_ERROR) { 7242beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent mAuxEffectId = effectId; 7252beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent } 7262beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent return status; 727be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent} 728be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent 72989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 73089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 7311703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// must be called with mLock held 7321703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurentstatus_t AudioTrack::createTrack_l( 733fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kasten audio_stream_type_t streamType, 73434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent uint32_t sampleRate, 735e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten audio_format_t format, 7360d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi uint32_t channelMask, 73734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent int frameCount, 7380ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent audio_output_flags_t flags, 73934f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent const sp<IMemory>& sharedBuffer, 740291f4d505aff81969e6666049d3cc3446f836af2Glenn Kasten audio_io_handle_t output) 74134f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent{ 74234f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent status_t status; 74334f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent const sp<IAudioFlinger>& audioFlinger = AudioSystem::get_audio_flinger(); 74434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (audioFlinger == 0) { 745e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten ALOGE("Could not get audioflinger"); 746e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten return NO_INIT; 74734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 74834f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent 749d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent int afSampleRate; 7501a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent if (AudioSystem::getSamplingRate(output, streamType, &afSampleRate) != NO_ERROR) { 751d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent return NO_INIT; 752d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 753d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent int afFrameCount; 7541a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent if (AudioSystem::getFrameCount(output, streamType, &afFrameCount) != NO_ERROR) { 755d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent return NO_INIT; 756d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 757d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent uint32_t afLatency; 7581a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent if (AudioSystem::getLatency(output, streamType, &afLatency) != NO_ERROR) { 759d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent return NO_INIT; 760d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 761d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent 7624a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten // Client decides whether the track is TIMED (see below), but can only express a preference 7634a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten // for FAST. Server will perform additional tests. 7640ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if ((flags & AUDIO_OUTPUT_FLAG_FAST) && !( 7654a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten // either of these use cases: 7664a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten // use case 1: shared buffer 7674a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten (sharedBuffer != 0) || 7684a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten // use case 2: callback handler 7694a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten (mCbf != NULL))) { 7700ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent ALOGW("AUDIO_OUTPUT_FLAG_FAST denied"); 7710ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent flags = (audio_output_flags_t) (flags & ~AUDIO_OUTPUT_FLAG_FAST); 7724a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten } 7731a9ed11a472493cac7f6dfcbfac2064526a493edEric Laurent ALOGV("createTrack_l() output %d afFrameCount %d afLatency %d", output, afFrameCount, afLatency); 7744a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten 775d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mNotificationFramesAct = mNotificationFramesReq; 776fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (!audio_is_linear_pcm(format)) { 777d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (sharedBuffer != 0) { 778d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent frameCount = sharedBuffer->size(); 779d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 780d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } else { 781d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent // Ensure that buffer depth covers at least audio hardware latency 782d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent uint32_t minBufCount = afLatency / ((1000 * afFrameCount)/afSampleRate); 783d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (minBufCount < 2) minBufCount = 2; 784d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent 785d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent int minFrameCount = (afFrameCount*sampleRate*minBufCount)/afSampleRate; 786d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent 787d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (sharedBuffer == 0) { 788d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (frameCount == 0) { 789d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent frameCount = minFrameCount; 790d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 791d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (mNotificationFramesAct == 0) { 792d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mNotificationFramesAct = frameCount/2; 793d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 794d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent // Make sure that application is notified with sufficient margin 795d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent // before underrun 796d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (mNotificationFramesAct > (uint32_t)frameCount/2) { 797d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mNotificationFramesAct = frameCount/2; 798d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 7990ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if (frameCount < minFrameCount && !(flags & AUDIO_OUTPUT_FLAG_FAST)) { 800291f4d505aff81969e6666049d3cc3446f836af2Glenn Kasten // not ALOGW because it happens all the time when playing key clicks over A2DP 801291f4d505aff81969e6666049d3cc3446f836af2Glenn Kasten ALOGV("Minimum buffer size corrected from %d to %d", 802d8d6185c978c9b27ca69e7791785e0983ed9e8b8Eric Laurent frameCount, minFrameCount); 803d8d6185c978c9b27ca69e7791785e0983ed9e8b8Eric Laurent frameCount = minFrameCount; 804d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 805d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } else { 80699e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // Ensure that buffer alignment matches channelCount 8070d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi int channelCount = popcount(channelMask); 808d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (((uint32_t)sharedBuffer->pointer() & (channelCount | 1)) != 0) { 80929357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Invalid buffer alignement: address %p, channelCount %d", sharedBuffer->pointer(), channelCount); 810d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent return BAD_VALUE; 811d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 812d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent frameCount = sharedBuffer->size()/channelCount/sizeof(int16_t); 813d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 814d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 815d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent 816a075db4ff9b086ac2885df77bb6da0869293df92Glenn Kasten IAudioFlinger::track_flags_t trackFlags = IAudioFlinger::TRACK_DEFAULT; 817a075db4ff9b086ac2885df77bb6da0869293df92Glenn Kasten if (mIsTimed) { 818a075db4ff9b086ac2885df77bb6da0869293df92Glenn Kasten trackFlags |= IAudioFlinger::TRACK_TIMED; 819a075db4ff9b086ac2885df77bb6da0869293df92Glenn Kasten } 8200ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if (flags & AUDIO_OUTPUT_FLAG_FAST) { 8214a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten trackFlags |= IAudioFlinger::TRACK_FAST; 8224a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten } 8234a4a0959bca78e03e3c3f486ba17829c28314d8cGlenn Kasten 82434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent sp<IAudioTrack> track = audioFlinger->createTrack(getpid(), 82534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent streamType, 82634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent sampleRate, 82734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent format, 8280d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi channelMask, 82934f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent frameCount, 830a075db4ff9b086ac2885df77bb6da0869293df92Glenn Kasten trackFlags, 83134f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent sharedBuffer, 83234f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent output, 833be916aa1267e2e6b1c148f51d11bcbbc79cb864cEric Laurent &mSessionId, 83434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent &status); 83534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent 83634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (track == 0) { 83729357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("AudioFlinger could not create track, status: %d", status); 83834f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent return status; 83934f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 84034f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent sp<IMemory> cblk = track->getCblk(); 84134f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (cblk == 0) { 84229357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("Could not get control block"); 84334f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent return NO_INIT; 84434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 84534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mAudioTrack = track; 84634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mCblkMemory = cblk; 84734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mCblk = static_cast<audio_track_cblk_t*>(cblk->pointer()); 84838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_or(CBLK_DIRECTION_OUT, &mCblk->flags); 84934f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (sharedBuffer == 0) { 85034f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mCblk->buffers = (char*)mCblk + sizeof(audio_track_cblk_t); 85134f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } else { 85234f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent mCblk->buffers = sharedBuffer->pointer(); 853e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten // Force buffer full condition as data is already present in shared memory 854d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mCblk->stepUser(mCblk->frameCount); 85534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 85634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent 85783d86538c4c479a9225c75ab27938e8f05abb9c8Glenn Kasten mCblk->setVolumeLR((uint32_t(uint16_t(mVolume[RIGHT] * 0x1000)) << 16) | uint16_t(mVolume[LEFT] * 0x1000)); 85805632a5fa4b88ca474294887fc92a9fcdf0e2352Glenn Kasten mCblk->setSendLevel(mSendLevel); 8592beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent mAudioTrack->attachAuxEffect(mAuxEffectId); 8606100d2d60517ff33ed8eb35d0b7ea63cde0831c9Eric Laurent mCblk->bufferTimeoutMs = MAX_STARTUP_TIMEOUT_MS; 8616100d2d60517ff33ed8eb35d0b7ea63cde0831c9Eric Laurent mCblk->waitTimeMs = 0; 862d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mRemainingFrames = mNotificationFramesAct; 863d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mLatency = afLatency + (1000*mCblk->frameCount) / sampleRate; 86434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent return NO_ERROR; 86534f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent} 86634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent 86789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::obtainBuffer(Buffer* audioBuffer, int32_t waitCount) 86889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 8691703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 8709a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten bool active; 871d0965dde97f2815ae0a15fe6b40946f8a741a81eGlenn Kasten status_t result = NO_ERROR; 87289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audio_track_cblk_t* cblk = mCblk; 87389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t framesReq = audioBuffer->frameCount; 8741dd70b9f04961a06fcb73a97fca10a53b3245d3cEric Laurent uint32_t waitTimeMs = (waitCount < 0) ? cblk->bufferTimeoutMs : WAIT_PERIOD_MS; 87589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 87689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audioBuffer->frameCount = 0; 87789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audioBuffer->size = 0; 87889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 87989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t framesAvail = cblk->framesAvailable(); 88089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 8819b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent cblk->lock.lock(); 8829b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent if (cblk->flags & CBLK_INVALID_MSK) { 8839b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent goto create_new_track; 8849b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent } 8859b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent cblk->lock.unlock(); 8869b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent 88789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (framesAvail == 0) { 88834f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent cblk->lock.lock(); 88989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project goto start_loop_here; 89089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project while (framesAvail == 0) { 89189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project active = mActive; 892f6b1678f8f508b447155a81b44e214475ab634a8Glenn Kasten if (CC_UNLIKELY(!active)) { 8933856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("Not active and NO_MORE_BUFFERS"); 89434f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent cblk->lock.unlock(); 89589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_MORE_BUFFERS; 89689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 897f6b1678f8f508b447155a81b44e214475ab634a8Glenn Kasten if (CC_UNLIKELY(!waitCount)) { 89834f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent cblk->lock.unlock(); 89989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return WOULD_BLOCK; 90034f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 901d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (!(cblk->flags & CBLK_INVALID_MSK)) { 9021703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.unlock(); 903d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent result = cblk->cv.waitRelative(cblk->lock, milliseconds(waitTimeMs)); 9041703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 9051703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.lock(); 9069a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (!mActive) { 9071703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent return status_t(STOPPED); 9081703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 9091703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.lock(); 910d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 9111703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 912d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (cblk->flags & CBLK_INVALID_MSK) { 913d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent goto create_new_track; 914d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent } 915f6b1678f8f508b447155a81b44e214475ab634a8Glenn Kasten if (CC_UNLIKELY(result != NO_ERROR)) { 9161dd70b9f04961a06fcb73a97fca10a53b3245d3cEric Laurent cblk->waitTimeMs += waitTimeMs; 91789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (cblk->waitTimeMs >= cblk->bufferTimeoutMs) { 91889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // timing out when a loop has been set and we have already written upto loop end 91989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // is a normal condition: no need to wake AudioFlinger up. 92089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (cblk->user < cblk->loopEnd) { 9215ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW( "obtainBuffer timed out (is the CPU pegged?) %p " 92289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project "user=%08x, server=%08x", this, cblk->user, cblk->server); 923c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent //unlock cblk mutex before calling mAudioTrack->start() (see issue #1617140) 92489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->lock.unlock(); 9256dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten result = mAudioTrack->start(0); // callback thread hasn't changed 9261703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.lock(); 92734f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent if (result == DEAD_OBJECT) { 92838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_or(CBLK_INVALID_ON, &cblk->flags); 929d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurentcreate_new_track: 9301703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent result = restoreTrack_l(cblk, false); 9311703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 9321703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (result != NO_ERROR) { 9335ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("obtainBuffer create Track error %d", result); 9341703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 9351703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent return result; 93634f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent } 93789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 93889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->waitTimeMs = 0; 93989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 940c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 94189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (--waitCount == 0) { 94234f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent cblk->lock.unlock(); 94389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return TIMED_OUT; 94489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 94589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 94689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // read the server count again 94789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project start_loop_here: 94889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project framesAvail = cblk->framesAvailable_l(); 94989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 95034f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent cblk->lock.unlock(); 95189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 95289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 95344d9848d6656777a18019223e0d35f2fcc67719aEric Laurent // restart track if it was disabled by audioflinger due to previous underrun 9541703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (mActive && (cblk->flags & CBLK_DISABLED_MSK)) { 95538ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_and(~CBLK_DISABLED_ON, &cblk->flags); 9565ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("obtainBuffer() track %p disabled, restarting", this); 9576dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten mAudioTrack->start(0); // callback thread hasn't changed 95844d9848d6656777a18019223e0d35f2fcc67719aEric Laurent } 95944d9848d6656777a18019223e0d35f2fcc67719aEric Laurent 96089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project cblk->waitTimeMs = 0; 961c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 96289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (framesReq > framesAvail) { 96389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project framesReq = framesAvail; 96489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 96589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 96689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t u = cblk->user; 96789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t bufferEnd = cblk->userBase + cblk->frameCount; 96889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 969a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (framesReq > bufferEnd - u) { 97089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project framesReq = bufferEnd - u; 97189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 97289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 973c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->flags = mMuted ? Buffer::MUTE : 0; 974c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->channelCount = mChannelCount; 975c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->frameCount = framesReq; 976c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->size = framesReq * cblk->frameSize; 977fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin if (audio_is_linear_pcm(mFormat)) { 978fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin audioBuffer->format = AUDIO_FORMAT_PCM_16_BIT; 979c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } else { 980c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->format = mFormat; 981c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } 982c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer->raw = (int8_t *)cblk->buffer(u); 98389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project active = mActive; 98489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return active ? status_t(NO_ERROR) : status_t(STOPPED); 98589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 98689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 98789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::releaseBuffer(Buffer* audioBuffer) 98889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 9891703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent AutoMutex lock(mLock); 9901703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mCblk->stepUser(audioBuffer->frameCount); 99189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 99289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 99389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 99489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 99589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t AudioTrack::write(const void* buffer, size_t userSize) 99689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 99789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 99889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mSharedBuffer != 0) return INVALID_OPERATION; 9994ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mIsTimed) return INVALID_OPERATION; 100089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 100189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (ssize_t(userSize) < 0) { 100299e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // Sanity-check: user is most-likely passing an error code, and it would 100399e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // make the return value ambiguous (actualSize vs error). 100429357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE("AudioTrack::write(buffer=%p, size=%u (%d)", 100589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project buffer, userSize, userSize); 100689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return BAD_VALUE; 100789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 100889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 10093856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("write %p: %d bytes, mActive=%d", this, userSize, mActive); 101089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 10111703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // acquire a strong reference on the IMemory and IAudioTrack so that they cannot be destroyed 10121703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // while we are accessing the cblk 10131703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.lock(); 1014e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IAudioTrack> audioTrack = mAudioTrack; 1015e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IMemory> iMem = mCblkMemory; 10161703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.unlock(); 10171703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 101889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project ssize_t written = 0; 101989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project const int8_t *src = (const int8_t *)buffer; 102089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project Buffer audioBuffer; 1021b9980659501d0428d65d8292f3c32da69d37fbd2Glenn Kasten size_t frameSz = frameSize(); 102289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 102389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project do { 102433797ea64d067dfeaacbfd7ebe7f3383b73961b5Eric Laurent audioBuffer.frameCount = userSize/frameSz; 1025c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 102689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project status_t err = obtainBuffer(&audioBuffer, -1); 102789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (err < 0) { 102889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // out of buffers, return #bytes written 102989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (err == status_t(NO_MORE_BUFFERS)) 103089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project break; 103189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return ssize_t(err); 103289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 103389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 103489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project size_t toWrite; 1035c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 10360ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if (mFormat == AUDIO_FORMAT_PCM_8_BIT && !(mFlags & AUDIO_OUTPUT_FLAG_DIRECT)) { 103789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Divide capacity by 2 to take expansion into account 103889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project toWrite = audioBuffer.size>>1; 1039511754b5839fd9b09fc56b89ae007fbc39084a33Glenn Kasten memcpy_to_i16_from_u8(audioBuffer.i16, (const uint8_t *) src, toWrite); 10403302526f6276911b2dc40c731ea7fa0e7972d908Eric Laurent } else { 104189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project toWrite = audioBuffer.size; 104289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project memcpy(audioBuffer.i8, src, toWrite); 104389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project src += toWrite; 104489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 104589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project userSize -= toWrite; 104689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project written += toWrite; 104789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 104889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project releaseBuffer(&audioBuffer); 104933797ea64d067dfeaacbfd7ebe7f3383b73961b5Eric Laurent } while (userSize >= frameSz); 105089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 105189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return written; 105289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 105389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 105489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 105589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 10564ff14bae91075eb274eb1c2975982358946e7e63John GrossmanTimedAudioTrack::TimedAudioTrack() { 10574ff14bae91075eb274eb1c2975982358946e7e63John Grossman mIsTimed = true; 10584ff14bae91075eb274eb1c2975982358946e7e63John Grossman} 10594ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10604ff14bae91075eb274eb1c2975982358946e7e63John Grossmanstatus_t TimedAudioTrack::allocateTimedBuffer(size_t size, sp<IMemory>* buffer) 10614ff14bae91075eb274eb1c2975982358946e7e63John Grossman{ 10624ff14bae91075eb274eb1c2975982358946e7e63John Grossman status_t result = UNKNOWN_ERROR; 10634ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10644ff14bae91075eb274eb1c2975982358946e7e63John Grossman // If the track is not invalid already, try to allocate a buffer. alloc 10654ff14bae91075eb274eb1c2975982358946e7e63John Grossman // fails indicating that the server is dead, flag the track as invalid so 10664ff14bae91075eb274eb1c2975982358946e7e63John Grossman // we can attempt to restore in in just a bit. 10674ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (!(mCblk->flags & CBLK_INVALID_MSK)) { 10684ff14bae91075eb274eb1c2975982358946e7e63John Grossman result = mAudioTrack->allocateTimedBuffer(size, buffer); 10694ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (result == DEAD_OBJECT) { 10704ff14bae91075eb274eb1c2975982358946e7e63John Grossman android_atomic_or(CBLK_INVALID_ON, &mCblk->flags); 10714ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 10724ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 10734ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10744ff14bae91075eb274eb1c2975982358946e7e63John Grossman // If the track is invalid at this point, attempt to restore it. and try the 10754ff14bae91075eb274eb1c2975982358946e7e63John Grossman // allocation one more time. 10764ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mCblk->flags & CBLK_INVALID_MSK) { 10774ff14bae91075eb274eb1c2975982358946e7e63John Grossman mCblk->lock.lock(); 10784ff14bae91075eb274eb1c2975982358946e7e63John Grossman result = restoreTrack_l(mCblk, false); 10794ff14bae91075eb274eb1c2975982358946e7e63John Grossman mCblk->lock.unlock(); 10804ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10814ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (result == OK) 10824ff14bae91075eb274eb1c2975982358946e7e63John Grossman result = mAudioTrack->allocateTimedBuffer(size, buffer); 10834ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 10844ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10854ff14bae91075eb274eb1c2975982358946e7e63John Grossman return result; 10864ff14bae91075eb274eb1c2975982358946e7e63John Grossman} 10874ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10884ff14bae91075eb274eb1c2975982358946e7e63John Grossmanstatus_t TimedAudioTrack::queueTimedBuffer(const sp<IMemory>& buffer, 10894ff14bae91075eb274eb1c2975982358946e7e63John Grossman int64_t pts) 10904ff14bae91075eb274eb1c2975982358946e7e63John Grossman{ 10914ff14bae91075eb274eb1c2975982358946e7e63John Grossman // restart track if it was disabled by audioflinger due to previous underrun 10924ff14bae91075eb274eb1c2975982358946e7e63John Grossman if (mActive && (mCblk->flags & CBLK_DISABLED_MSK)) { 10934ff14bae91075eb274eb1c2975982358946e7e63John Grossman android_atomic_and(~CBLK_DISABLED_ON, &mCblk->flags); 10944ff14bae91075eb274eb1c2975982358946e7e63John Grossman ALOGW("queueTimedBuffer() track %p disabled, restarting", this); 10954ff14bae91075eb274eb1c2975982358946e7e63John Grossman mAudioTrack->start(0); 10964ff14bae91075eb274eb1c2975982358946e7e63John Grossman } 10974ff14bae91075eb274eb1c2975982358946e7e63John Grossman 10984ff14bae91075eb274eb1c2975982358946e7e63John Grossman return mAudioTrack->queueTimedBuffer(buffer, pts); 10994ff14bae91075eb274eb1c2975982358946e7e63John Grossman} 11004ff14bae91075eb274eb1c2975982358946e7e63John Grossman 11014ff14bae91075eb274eb1c2975982358946e7e63John Grossmanstatus_t TimedAudioTrack::setMediaTimeTransform(const LinearTransform& xform, 11024ff14bae91075eb274eb1c2975982358946e7e63John Grossman TargetTimeline target) 11034ff14bae91075eb274eb1c2975982358946e7e63John Grossman{ 11044ff14bae91075eb274eb1c2975982358946e7e63John Grossman return mAudioTrack->setMediaTimeTransform(xform, target); 11054ff14bae91075eb274eb1c2975982358946e7e63John Grossman} 11064ff14bae91075eb274eb1c2975982358946e7e63John Grossman 11074ff14bae91075eb274eb1c2975982358946e7e63John Grossman// ------------------------------------------------------------------------- 11084ff14bae91075eb274eb1c2975982358946e7e63John Grossman 110989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool AudioTrack::processAudioBuffer(const sp<AudioTrackThread>& thread) 111089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 111189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project Buffer audioBuffer; 111289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t frames; 111389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project size_t writtenSize; 111489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 11151703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.lock(); 11161703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // acquire a strong reference on the IMemory and IAudioTrack so that they cannot be destroyed 11171703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // while we are accessing the cblk 1118e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IAudioTrack> audioTrack = mAudioTrack; 1119e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten sp<IMemory> iMem = mCblkMemory; 11201703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent audio_track_cblk_t* cblk = mCblk; 11219a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten bool active = mActive; 11221703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.unlock(); 11231703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 112489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Manage underrun callback 11259a2aaf927e56a4b4acab23ef16b3f133a9f48a63Glenn Kasten if (active && (cblk->framesAvailable() == cblk->frameCount)) { 11263856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("Underrun user: %x, server: %x, flags %04x", cblk->user, cblk->server, cblk->flags); 112738ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (!(android_atomic_or(CBLK_UNDERRUN_ON, &cblk->flags) & CBLK_UNDERRUN_MSK)) { 112889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCbf(EVENT_UNDERRUN, mUserData, 0); 11291703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (cblk->server == cblk->frameCount) { 1130c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent mCbf(EVENT_BUFFER_END, mUserData, 0); 113189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 113289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mSharedBuffer != 0) return false; 113389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 113489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 1135c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 113689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Manage loop end callback 11371703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent while (mLoopCount > cblk->loopCount) { 113889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project int loopCount = -1; 113989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mLoopCount--; 114089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mLoopCount >= 0) loopCount = mLoopCount; 114189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 114289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCbf(EVENT_LOOP_END, mUserData, (void *)&loopCount); 114389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 114489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 114589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Manage marker callback 11462c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi if (!mMarkerReached && (mMarkerPosition > 0)) { 11471703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (cblk->server >= mMarkerPosition) { 114889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCbf(EVENT_MARKER, mUserData, (void *)&mMarkerPosition); 11492c22aeb65e801f663a754d043062f85e49f77739Jean-Michel Trivi mMarkerReached = true; 115089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 115189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 115289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 115389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Manage new position callback 1154c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent if (mUpdatePeriod > 0) { 11551703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent while (cblk->server >= mNewPosition) { 115689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCbf(EVENT_NEW_POS, mUserData, (void *)&mNewPosition); 115789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mNewPosition += mUpdatePeriod; 115889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 115989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 116089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 116189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // If Shared buffer is used, no data is requested from client. 116289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (mSharedBuffer != 0) { 116389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project frames = 0; 116489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 116589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project frames = mRemainingFrames; 116689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 116789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 116899e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // See description of waitCount parameter at declaration of obtainBuffer(). 116999e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // The logic below prevents us from being stuck below at obtainBuffer() 117099e53b86eebb605b70dd7591b89bf61a9414ed0eGlenn Kasten // not being able to handle timed events (position, markers, loops). 11712267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent int32_t waitCount = -1; 11722267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent if (mUpdatePeriod || (!mMarkerReached && mMarkerPosition) || mLoopCount) { 11732267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent waitCount = 1; 11742267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent } 11752267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent 117689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project do { 117789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 117889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audioBuffer.frameCount = frames; 1179c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 11802267ba18d0d2b2d4bd7f5411821ad89ac2659a88Eric Laurent status_t err = obtainBuffer(&audioBuffer, waitCount); 118189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (err < NO_ERROR) { 118289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (err != TIMED_OUT) { 118329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block ALOGE_IF(err != status_t(NO_MORE_BUFFERS), "Error obtaining an audio buffer, giving up."); 118489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return false; 118589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 118689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project break; 118789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 118889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (err == status_t(STOPPED)) return false; 118989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 119089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Divide buffer size by 2 to take into account the expansion 119189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // due to 8 to 16 bit conversion: the callback must fill only half 119289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // of the destination buffer 11930ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if (mFormat == AUDIO_FORMAT_PCM_8_BIT && !(mFlags & AUDIO_OUTPUT_FLAG_DIRECT)) { 119489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audioBuffer.size >>= 1; 119589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 119689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 119789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project size_t reqSize = audioBuffer.size; 119889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mCbf(EVENT_MORE_DATA, mUserData, &audioBuffer); 119989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project writtenSize = audioBuffer.size; 120089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 120189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Sanity check on returned size 12028555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project if (ssize_t(writtenSize) <= 0) { 12038555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project // The callback is done filling buffers 12048555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project // Keep this thread going to handle timed events and 12058555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project // still try to get more data in intervals of WAIT_PERIOD_MS 12068555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project // but don't just loop and block the CPU, so wait 12078555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project usleep(WAIT_PERIOD_MS*1000); 12088555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project break; 12098555d0867c3e8fe6cc5c7ad40af557fe6b92fa72The Android Open Source Project } 121089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (writtenSize > reqSize) writtenSize = reqSize; 121189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 12120ca3cf94c0dfc173ad7886ae162c4b67067539f6Eric Laurent if (mFormat == AUDIO_FORMAT_PCM_8_BIT && !(mFlags & AUDIO_OUTPUT_FLAG_DIRECT)) { 1213511754b5839fd9b09fc56b89ae007fbc39084a33Glenn Kasten // 8 to 16 bit conversion, note that source and destination are the same address 1214511754b5839fd9b09fc56b89ae007fbc39084a33Glenn Kasten memcpy_to_i16_from_u8(audioBuffer.i16, (const uint8_t *) audioBuffer.i8, writtenSize); 121589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project writtenSize <<= 1; 121689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 121789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 121889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project audioBuffer.size = writtenSize; 1219c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent // NOTE: mCblk->frameSize is not equal to AudioTrack::frameSize() for 1220b9980659501d0428d65d8292f3c32da69d37fbd2Glenn Kasten // 8 bit PCM data: in this case, mCblk->frameSize is based on a sample size of 1221c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent // 16 bit. 1222c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent audioBuffer.frameCount = writtenSize/mCblk->frameSize; 1223c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent 122489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project frames -= audioBuffer.frameCount; 122589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 122689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project releaseBuffer(&audioBuffer); 122789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 122889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project while (frames); 122989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 123089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (frames == 0) { 1231d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent mRemainingFrames = mNotificationFramesAct; 123289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 123389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project mRemainingFrames = frames; 123489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 123589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return true; 123689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 123789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 12381703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// must be called with mLock and cblk.lock held. Callers must also hold strong references on 12391703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// the IAudioTrack and IMemory in case they are recreated here. 12401703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent// If the IAudioTrack is successfully restored, the cblk pointer is updated 12411703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurentstatus_t AudioTrack::restoreTrack_l(audio_track_cblk_t*& cblk, bool fromStart) 12421703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent{ 12431703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent status_t result; 12441703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 124538ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (!(android_atomic_or(CBLK_RESTORING_ON, &cblk->flags) & CBLK_RESTORING_MSK)) { 12465ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("dead IAudioTrack, creating a new one from %s TID %d", 1247e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten fromStart ? "start()" : "obtainBuffer()", gettid()); 12481703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 12491703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // signal old cblk condition so that other threads waiting for available buffers stop 12501703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // waiting now 12511703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->cv.broadcast(); 12521703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 12531703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 12549f6530f53ae9eda43f4e7c1cb30d2379db00aa00Eric Laurent // refresh the audio configuration cache in this process to make sure we get new 12559f6530f53ae9eda43f4e7c1cb30d2379db00aa00Eric Laurent // output parameters in getOutput_l() and createTrack_l() 12569f6530f53ae9eda43f4e7c1cb30d2379db00aa00Eric Laurent AudioSystem::clearAudioConfigCache(); 12579f6530f53ae9eda43f4e7c1cb30d2379db00aa00Eric Laurent 12581703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // if the new IAudioTrack is created, createTrack_l() will modify the 12591703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // following member variables: mAudioTrack, mCblkMemory and mCblk. 12601703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // It will also delete the strong references on previous IAudioTrack and IMemory 12611703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent result = createTrack_l(mStreamType, 12621703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->sampleRate, 12631703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mFormat, 12640d255b2d9061ba31f13ada3fc0f7e51916407176Jean-Michel Trivi mChannelMask, 12651703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mFrameCount, 12661703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mFlags, 12671703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mSharedBuffer, 1268291f4d505aff81969e6666049d3cc3446f836af2Glenn Kasten getOutput_l()); 12691703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 12701703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (result == NO_ERROR) { 1271408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent uint32_t user = cblk->user; 1272408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent uint32_t server = cblk->server; 12739b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent // restore write index and set other indexes to reflect empty buffer status 1274408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->user = user; 1275408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->server = user; 1276408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->userBase = user; 1277408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->serverBase = user; 12789b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent // restore loop: this is not guaranteed to succeed if new frame count is not 12799b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent // compatible with loop length 12809b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent setLoop_l(cblk->loopStart, cblk->loopEnd, cblk->loopCount); 12811703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (!fromStart) { 12821703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mCblk->bufferTimeoutMs = MAX_RUN_TIMEOUT_MS; 1283408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent // Make sure that a client relying on callback events indicating underrun or 1284408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent // the actual amount of audio frames played (e.g SoundPool) receives them. 1285408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent if (mSharedBuffer == 0) { 1286408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent uint32_t frames = 0; 1287408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent if (user > server) { 1288408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent frames = ((user - server) > mCblk->frameCount) ? 1289408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->frameCount : (user - server); 1290408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent memset(mCblk->buffers, 0, frames * mCblk->frameSize); 1291408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent } 1292408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent // restart playback even if buffer is not completely filled. 1293408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent android_atomic_or(CBLK_FORCEREADY_ON, &mCblk->flags); 1294408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent // stepUser() clears CBLK_UNDERRUN_ON flag enabling underrun callbacks to 1295408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent // the client 1296408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent mCblk->stepUser(frames); 1297408b8dc3c0a364c6f6b4991d15da9e6bcc2b8008Eric Laurent } 12981703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 12999b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent if (mActive) { 13006dbc1359f778575d09d6da722b060a6d72c2e7c5Glenn Kasten result = mAudioTrack->start(0); // callback thread hasn't changed 13015ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW_IF(result != NO_ERROR, "restoreTrack_l() start() failed status %d", result); 13029b7d950f1f3b0c526712b713dbceb0e22762c015Eric Laurent } 13031703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (fromStart && result == NO_ERROR) { 13041703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mNewPosition = mCblk->server + mUpdatePeriod; 13051703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 13061703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 13071703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (result != NO_ERROR) { 1308cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent android_atomic_and(~CBLK_RESTORING_ON, &cblk->flags); 13095ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW_IF(result != NO_ERROR, "restoreTrack_l() failed status %d", result); 13101703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 1311cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent mRestoreStatus = result; 13121703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // signal old cblk condition for other threads waiting for restore completion 131338ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_or(CBLK_RESTORED_ON, &cblk->flags); 13141703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->cv.broadcast(); 13151703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } else { 13161703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (!(cblk->flags & CBLK_RESTORED_MSK)) { 13175ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("dead IAudioTrack, waiting for a new one TID %d", gettid()); 13181703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.unlock(); 13191703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent result = cblk->cv.waitRelative(cblk->lock, milliseconds(RESTORE_TIMEOUT_MS)); 1320cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent if (result == NO_ERROR) { 1321cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent result = mRestoreStatus; 1322cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent } 13231703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 13241703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent mLock.lock(); 13251703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } else { 13265ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("dead IAudioTrack, already restored TID %d", gettid()); 1327cfe2ba6b01a258e39f9c215ffc7b750e0b68f708Eric Laurent result = mRestoreStatus; 13281703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.unlock(); 13291703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 13301703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 13313856b090cd04ba5dd4a59a12430ed724d5995909Steve Block ALOGV("restoreTrack_l() status %d mActive %d cblk %p, old cblk %p flags %08x old flags %08x", 1332e53b9ead781c36e96d6b6f012ddffc93a3d80f0dGlenn Kasten result, mActive, mCblk, cblk, mCblk->flags, cblk->flags); 13331703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 13341703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (result == NO_ERROR) { 13351703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent // from now on we switch to the newly created cblk 13361703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk = mCblk; 13371703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 13381703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cblk->lock.lock(); 13391703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 13405ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW_IF(result != NO_ERROR, "restoreTrack_l() error %d TID %d", result, gettid()); 13411703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 13421703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent return result; 13431703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent} 13441703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent 134589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::dump(int fd, const Vector<String16>& args) const 134689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 134789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 134889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project const size_t SIZE = 256; 134989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project char buffer[SIZE]; 135089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project String8 result; 135189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 135289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project result.append(" AudioTrack::dump\n"); 135389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project snprintf(buffer, 255, " stream type(%d), left - right volume(%f, %f)\n", mStreamType, mVolume[0], mVolume[1]); 135489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project result.append(buffer); 1355d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent snprintf(buffer, 255, " format(%d), channel count(%d), frame count(%d)\n", mFormat, mChannelCount, mCblk->frameCount); 135689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project result.append(buffer); 1357573266210fb2b2e7d86fbd46d0dfe16763611d91Eric Laurent snprintf(buffer, 255, " sample rate(%d), status(%d), muted(%d)\n", (mCblk == 0) ? 0 : mCblk->sampleRate, mStatus, mMuted); 135889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project result.append(buffer); 135989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project snprintf(buffer, 255, " active(%d), latency (%d)\n", mActive, mLatency); 136089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project result.append(buffer); 136189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project ::write(fd, result.string(), result.size()); 136289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 136389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 136489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 136589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ========================================================================= 136689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 136789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAudioTrack::AudioTrackThread::AudioTrackThread(AudioTrack& receiver, bool bCanCallJava) 136889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project : Thread(bCanCallJava), mReceiver(receiver) 136989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 137089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 137189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 137289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool AudioTrack::AudioTrackThread::threadLoop() 137389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 137489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return mReceiver.processAudioBuffer(this); 137589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 137689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 137789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t AudioTrack::AudioTrackThread::readyToRun() 137889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 137989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return NO_ERROR; 138089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 138189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 138289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid AudioTrack::AudioTrackThread::onFirstRef() 138389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 138489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 138589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 138689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ========================================================================= 138789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 138838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent 138989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectaudio_track_cblk_t::audio_track_cblk_t() 139054b1a0550411c2fd2084d82d28934d505c37349aMathias Agopian : lock(Mutex::SHARED), cv(Condition::SHARED), user(0), server(0), 1391a0d68338a88c2ddb4502f95017b546d603ef1ec7Glenn Kasten userBase(0), serverBase(0), buffers(NULL), frameCount(0), 139283d86538c4c479a9225c75ab27938e8f05abb9c8Glenn Kasten loopStart(UINT_MAX), loopEnd(UINT_MAX), loopCount(0), mVolumeLR(0x10001000), 139305632a5fa4b88ca474294887fc92a9fcdf0e2352Glenn Kasten mSendLevel(0), flags(0) 139489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 139589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 139689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 139789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t audio_track_cblk_t::stepUser(uint32_t frameCount) 139889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 1399a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen ALOGV("stepuser %08x %08x %d", user, server, frameCount); 140089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1401a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen uint32_t u = user; 140289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project u += frameCount; 140389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Ensure that user is never ahead of server for AudioRecord 1404d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (flags & CBLK_DIRECTION_MSK) { 140589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // If stepServer() has been called once, switch to normal obtainBuffer() timeout period 140689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (bufferTimeoutMs == MAX_STARTUP_TIMEOUT_MS-1) { 140789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project bufferTimeoutMs = MAX_RUN_TIMEOUT_MS; 140889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 14099054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten } else if (u > server) { 1410a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen ALOGW("stepUser occurred after track reset"); 14119054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten u = server; 141289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 141389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1414a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen uint32_t fc = this->frameCount; 1415a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (u >= fc) { 1416a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen // common case, user didn't just wrap 1417a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (u - fc >= userBase ) { 1418a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen userBase += fc; 1419a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen } 1420a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen } else if (u >= userBase + fc) { 1421a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen // user just wrapped 1422a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen userBase += fc; 142389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 142489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 14259054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten user = u; 142689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 142789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Clear flow control error condition as new data has been written/read to/from buffer. 142833797ea64d067dfeaacbfd7ebe7f3383b73961b5Eric Laurent if (flags & CBLK_UNDERRUN_MSK) { 142938ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent android_atomic_and(~CBLK_UNDERRUN_MSK, &flags); 143033797ea64d067dfeaacbfd7ebe7f3383b73961b5Eric Laurent } 143189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 143289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return u; 143389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 143489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 143589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool audio_track_cblk_t::stepServer(uint32_t frameCount) 143689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 1437a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen ALOGV("stepserver %08x %08x %d", user, server, frameCount); 1438a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen 143938ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (!tryLock()) { 14405ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("stepServer() could not lock cblk"); 144189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return false; 144289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 144389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 14449054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten uint32_t s = server; 1445a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen bool flushed = (s == user); 144689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 144789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project s += frameCount; 1448d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (flags & CBLK_DIRECTION_MSK) { 144989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // Mark that we have read the first buffer so that next time stepUser() is called 145089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // we switch to normal obtainBuffer() timeout period 145189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (bufferTimeoutMs == MAX_STARTUP_TIMEOUT_MS) { 145234f1d8ecd23169a5f299937e3aaf1bd7937578a0Eric Laurent bufferTimeoutMs = MAX_STARTUP_TIMEOUT_MS - 1; 1453c2f1f07084818942352c6bbfb36af9b6b330eb4eEric Laurent } 145489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // It is possible that we receive a flush() 145589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // while the mixer is processing a block: in this case, 145689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // stepServer() is called After the flush() has reset u & s and 145789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project // we have s > u 1458a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (flushed) { 14595ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("stepServer occurred after track reset"); 14609054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten s = user; 146189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 146289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 146389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 146489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (s >= loopEnd) { 14655ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW_IF(s > loopEnd, "stepServer: s %u > loopEnd %u", s, loopEnd); 146689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project s = loopStart; 146789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (--loopCount == 0) { 146889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project loopEnd = UINT_MAX; 146989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project loopStart = UINT_MAX; 147089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 147189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 1472a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen 1473a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen uint32_t fc = this->frameCount; 1474a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (s >= fc) { 1475a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen // common case, server didn't just wrap 1476a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen if (s - fc >= serverBase ) { 1477a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen serverBase += fc; 1478a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen } 1479a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen } else if (s >= serverBase + fc) { 1480a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen // server just wrapped 1481a1472d9883e35edd280201c8be3191695007dfd4Marco Nelissen serverBase += fc; 148289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 148389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 14849054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten server = s; 148589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 14861703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent if (!(flags & CBLK_INVALID_MSK)) { 14871703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent cv.signal(); 14881703cdfee717b1b312bf8979816a9e2f16a82e5dEric Laurent } 148989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project lock.unlock(); 149089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return true; 149189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 149289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 149389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid* audio_track_cblk_t::buffer(uint32_t offset) const 149489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 14959054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten return (int8_t *)buffers + (offset - userBase) * frameSize; 149689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 149789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 149889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t audio_track_cblk_t::framesAvailable() 149989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 150089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project Mutex::Autolock _l(lock); 150189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return framesAvailable_l(); 150289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 150389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 150489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t audio_track_cblk_t::framesAvailable_l() 150589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 15069054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten uint32_t u = user; 15079054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten uint32_t s = server; 150889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1509d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (flags & CBLK_DIRECTION_MSK) { 151089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project uint32_t limit = (s < loopStart) ? s : loopStart; 151189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return limit + frameCount - u; 151289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 151389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return frameCount + u - s; 151489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 151589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 151689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 151789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t audio_track_cblk_t::framesReady() 151889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{ 15199054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten uint32_t u = user; 15209054897ab9ffb307fadae81b774a1fc61cb542e7Glenn Kasten uint32_t s = server; 152189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 1522d1b449aad6c087a69f5ec66b7facb2845b73f1cbEric Laurent if (flags & CBLK_DIRECTION_MSK) { 152389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (u < loopEnd) { 152489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return u - s; 152589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 152638ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent // do not block on mutex shared with client on AudioFlinger side 152738ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (!tryLock()) { 15285ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block ALOGW("framesReady() could not lock cblk"); 152938ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent return 0; 153038ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent } 153138ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent uint32_t frames = UINT_MAX; 153289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project if (loopCount >= 0) { 153338ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent frames = (loopEnd - loopStart)*loopCount + u - s; 153489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 153538ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent lock.unlock(); 153638ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent return frames; 153789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 153889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } else { 153989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project return s - u; 154089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project } 154189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} 154289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 154338ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurentbool audio_track_cblk_t::tryLock() 154438ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent{ 154538ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent // the code below simulates lock-with-timeout 154638ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent // we MUST do this to protect the AudioFlinger server 154738ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent // as this lock is shared with the client. 154838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent status_t err; 154938ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent 155038ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent err = lock.tryLock(); 155138ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (err == -EBUSY) { // just wait a bit 155238ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent usleep(1000); 155338ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent err = lock.tryLock(); 155438ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent } 155538ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent if (err != NO_ERROR) { 155638ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent // probably, the client just died. 155738ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent return false; 155838ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent } 155938ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent return true; 156038ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent} 156138ccae2c0324daa305f3fe77d25fdf5edec0b0e1Eric Laurent 156289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// ------------------------------------------------------------------------- 156389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project 156489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}; // namespace android 1565