blob: b7d86642ff06257f300dcda4713445fd10517846 [file] [log] [blame]
Phil Burke1ce4912016-11-21 10:40:25 -08001/*
2 * Copyright 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "AudioStreamTrack"
18//#define LOG_NDEBUG 0
19#include <utils/Log.h>
20
21#include <stdint.h>
22#include <media/AudioTrack.h>
23
Phil Burk5ed503c2017-02-01 09:38:15 -080024#include <aaudio/AAudio.h>
Phil Burke1ce4912016-11-21 10:40:25 -080025#include "AudioClock.h"
26#include "AudioStreamTrack.h"
27
28
29using namespace android;
Phil Burk5ed503c2017-02-01 09:38:15 -080030using namespace aaudio;
Phil Burke1ce4912016-11-21 10:40:25 -080031
32/*
33 * Create a stream that uses the AudioTrack.
34 */
35AudioStreamTrack::AudioStreamTrack()
36 : AudioStream()
37{
38}
39
40AudioStreamTrack::~AudioStreamTrack()
41{
Phil Burk5ed503c2017-02-01 09:38:15 -080042 const aaudio_stream_state_t state = getState();
43 bool bad = !(state == AAUDIO_STREAM_STATE_UNINITIALIZED || state == AAUDIO_STREAM_STATE_CLOSED);
Phil Burke1ce4912016-11-21 10:40:25 -080044 ALOGE_IF(bad, "stream not closed, in state %d", state);
45}
46
Phil Burk5ed503c2017-02-01 09:38:15 -080047aaudio_result_t AudioStreamTrack::open(const AudioStreamBuilder& builder)
Phil Burke1ce4912016-11-21 10:40:25 -080048{
Phil Burk5ed503c2017-02-01 09:38:15 -080049 aaudio_result_t result = AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -080050
51 result = AudioStream::open(builder);
52 if (result != OK) {
53 return result;
54 }
55
56 // Try to create an AudioTrack
57 // TODO Support UNSPECIFIED in AudioTrack. For now, use stereo if unspecified.
Phil Burk5ed503c2017-02-01 09:38:15 -080058 int32_t samplesPerFrame = (getSamplesPerFrame() == AAUDIO_UNSPECIFIED)
Phil Burke1ce4912016-11-21 10:40:25 -080059 ? 2 : getSamplesPerFrame();
60 audio_channel_mask_t channelMask = audio_channel_out_mask_from_count(samplesPerFrame);
Phil Burkd8bdcab2017-01-03 17:20:30 -080061 ALOGD("AudioStreamTrack::open(), samplesPerFrame = %d, channelMask = 0x%08x",
Phil Burke1ce4912016-11-21 10:40:25 -080062 samplesPerFrame, channelMask);
63
Phil Burkd8bdcab2017-01-03 17:20:30 -080064 AudioTrack::callback_t callback = nullptr;
Phil Burke1ce4912016-11-21 10:40:25 -080065 // TODO add more performance options
66 audio_output_flags_t flags = (audio_output_flags_t) AUDIO_OUTPUT_FLAG_FAST;
Phil Burk3df348f2017-02-08 11:41:55 -080067 size_t frameCount = (builder.getBufferCapacity() == AAUDIO_UNSPECIFIED) ? 0
68 : builder.getBufferCapacity();
Phil Burke1ce4912016-11-21 10:40:25 -080069 // TODO implement an unspecified AudioTrack format then use that.
Phil Burk5ed503c2017-02-01 09:38:15 -080070 audio_format_t format = (getFormat() == AAUDIO_UNSPECIFIED)
Phil Burke1ce4912016-11-21 10:40:25 -080071 ? AUDIO_FORMAT_PCM_FLOAT
Phil Burk5ed503c2017-02-01 09:38:15 -080072 : AAudioConvert_aaudioToAndroidDataFormat(getFormat());
Phil Burke1ce4912016-11-21 10:40:25 -080073
74 mAudioTrack = new AudioTrack(
75 (audio_stream_type_t) AUDIO_STREAM_MUSIC,
76 getSampleRate(),
77 format,
78 channelMask,
79 frameCount,
80 flags,
81 callback,
Phil Burkd8bdcab2017-01-03 17:20:30 -080082 nullptr, // user callback data
83 0, // notificationFrames
Phil Burke1ce4912016-11-21 10:40:25 -080084 AUDIO_SESSION_ALLOCATE,
85 AudioTrack::transfer_type::TRANSFER_SYNC // TODO - this does not allow FAST
86 );
87
88 // Did we get a valid track?
89 status_t status = mAudioTrack->initCheck();
Phil Burkd8bdcab2017-01-03 17:20:30 -080090 ALOGD("AudioStreamTrack::open(), initCheck() returned %d", status);
Phil Burkdec33ab2017-01-17 14:48:16 -080091 if (status != NO_ERROR) {
Phil Burke1ce4912016-11-21 10:40:25 -080092 close();
93 ALOGE("AudioStreamTrack::open(), initCheck() returned %d", status);
Phil Burk5ed503c2017-02-01 09:38:15 -080094 return AAudioConvert_androidToAAudioResult(status);
Phil Burke1ce4912016-11-21 10:40:25 -080095 }
96
97 // Get the actual values from the AudioTrack.
98 setSamplesPerFrame(mAudioTrack->channelCount());
99 setSampleRate(mAudioTrack->getSampleRate());
Phil Burk5ed503c2017-02-01 09:38:15 -0800100 setFormat(AAudioConvert_androidToAAudioDataFormat(mAudioTrack->format()));
Phil Burke1ce4912016-11-21 10:40:25 -0800101
Phil Burk5ed503c2017-02-01 09:38:15 -0800102 setState(AAUDIO_STREAM_STATE_OPEN);
Phil Burke1ce4912016-11-21 10:40:25 -0800103
Phil Burk5ed503c2017-02-01 09:38:15 -0800104 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800105}
106
Phil Burk5ed503c2017-02-01 09:38:15 -0800107aaudio_result_t AudioStreamTrack::close()
Phil Burke1ce4912016-11-21 10:40:25 -0800108{
109 // TODO maybe add close() or release() to AudioTrack API then call it from here
Phil Burk5ed503c2017-02-01 09:38:15 -0800110 if (getState() != AAUDIO_STREAM_STATE_CLOSED) {
Phil Burke1ce4912016-11-21 10:40:25 -0800111 mAudioTrack.clear(); // TODO is this right?
Phil Burk5ed503c2017-02-01 09:38:15 -0800112 setState(AAUDIO_STREAM_STATE_CLOSED);
Phil Burke1ce4912016-11-21 10:40:25 -0800113 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800114 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800115}
116
Phil Burk5ed503c2017-02-01 09:38:15 -0800117aaudio_result_t AudioStreamTrack::requestStart()
Phil Burke1ce4912016-11-21 10:40:25 -0800118{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800119 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800120 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800121 }
122 // Get current position so we can detect when the track is playing.
123 status_t err = mAudioTrack->getPosition(&mPositionWhenStarting);
124 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800125 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800126 }
127 err = mAudioTrack->start();
128 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800129 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800130 } else {
Phil Burk5ed503c2017-02-01 09:38:15 -0800131 setState(AAUDIO_STREAM_STATE_STARTING);
Phil Burke1ce4912016-11-21 10:40:25 -0800132 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800133 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800134}
135
Phil Burk5ed503c2017-02-01 09:38:15 -0800136aaudio_result_t AudioStreamTrack::requestPause()
Phil Burke1ce4912016-11-21 10:40:25 -0800137{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800138 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800139 return AAUDIO_ERROR_INVALID_STATE;
140 } else if (getState() != AAUDIO_STREAM_STATE_STARTING
141 && getState() != AAUDIO_STREAM_STATE_STARTED) {
142 ALOGE("requestPause(), called when state is %s", AAudio_convertStreamStateToText(getState()));
143 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800144 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800145 setState(AAUDIO_STREAM_STATE_PAUSING);
Phil Burke1ce4912016-11-21 10:40:25 -0800146 mAudioTrack->pause();
147 status_t err = mAudioTrack->getPosition(&mPositionWhenPausing);
148 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800149 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800150 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800151 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800152}
153
Phil Burk5ed503c2017-02-01 09:38:15 -0800154aaudio_result_t AudioStreamTrack::requestFlush() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800155 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800156 return AAUDIO_ERROR_INVALID_STATE;
157 } else if (getState() != AAUDIO_STREAM_STATE_PAUSED) {
158 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800159 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800160 setState(AAUDIO_STREAM_STATE_FLUSHING);
Phil Burke1ce4912016-11-21 10:40:25 -0800161 incrementFramesRead(getFramesWritten() - getFramesRead());
162 mAudioTrack->flush();
163 mFramesWritten.reset32();
Phil Burk5ed503c2017-02-01 09:38:15 -0800164 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800165}
166
Phil Burk5ed503c2017-02-01 09:38:15 -0800167aaudio_result_t AudioStreamTrack::requestStop() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800168 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800169 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800170 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800171 setState(AAUDIO_STREAM_STATE_STOPPING);
Phil Burke1ce4912016-11-21 10:40:25 -0800172 incrementFramesRead(getFramesWritten() - getFramesRead()); // TODO review
173 mAudioTrack->stop();
174 mFramesWritten.reset32();
Phil Burk5ed503c2017-02-01 09:38:15 -0800175 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800176}
177
Phil Burk5ed503c2017-02-01 09:38:15 -0800178aaudio_result_t AudioStreamTrack::updateState()
Phil Burke1ce4912016-11-21 10:40:25 -0800179{
180 status_t err;
Phil Burk5ed503c2017-02-01 09:38:15 -0800181 aaudio_wrapping_frames_t position;
Phil Burke1ce4912016-11-21 10:40:25 -0800182 switch (getState()) {
183 // TODO add better state visibility to AudioTrack
Phil Burk5ed503c2017-02-01 09:38:15 -0800184 case AAUDIO_STREAM_STATE_STARTING:
Phil Burke1ce4912016-11-21 10:40:25 -0800185 if (mAudioTrack->hasStarted()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800186 setState(AAUDIO_STREAM_STATE_STARTED);
Phil Burke1ce4912016-11-21 10:40:25 -0800187 }
188 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800189 case AAUDIO_STREAM_STATE_PAUSING:
Phil Burke1ce4912016-11-21 10:40:25 -0800190 if (mAudioTrack->stopped()) {
191 err = mAudioTrack->getPosition(&position);
192 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800193 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800194 } else if (position == mPositionWhenPausing) {
195 // Has stream really stopped advancing?
Phil Burk5ed503c2017-02-01 09:38:15 -0800196 setState(AAUDIO_STREAM_STATE_PAUSED);
Phil Burke1ce4912016-11-21 10:40:25 -0800197 }
198 mPositionWhenPausing = position;
199 }
200 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800201 case AAUDIO_STREAM_STATE_FLUSHING:
Phil Burke1ce4912016-11-21 10:40:25 -0800202 {
203 err = mAudioTrack->getPosition(&position);
204 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800205 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800206 } else if (position == 0) {
207 // Advance frames read to match written.
Phil Burk5ed503c2017-02-01 09:38:15 -0800208 setState(AAUDIO_STREAM_STATE_FLUSHED);
Phil Burke1ce4912016-11-21 10:40:25 -0800209 }
210 }
211 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800212 case AAUDIO_STREAM_STATE_STOPPING:
Phil Burke1ce4912016-11-21 10:40:25 -0800213 if (mAudioTrack->stopped()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800214 setState(AAUDIO_STREAM_STATE_STOPPED);
Phil Burke1ce4912016-11-21 10:40:25 -0800215 }
216 break;
217 default:
218 break;
219 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800220 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800221}
222
Phil Burk5ed503c2017-02-01 09:38:15 -0800223aaudio_result_t AudioStreamTrack::write(const void *buffer,
224 aaudio_size_frames_t numFrames,
225 aaudio_nanoseconds_t timeoutNanoseconds)
Phil Burke1ce4912016-11-21 10:40:25 -0800226{
Phil Burk5ed503c2017-02-01 09:38:15 -0800227 aaudio_size_frames_t bytesPerFrame = getBytesPerFrame();
228 aaudio_size_bytes_t numBytes;
229 aaudio_result_t result = AAudioConvert_framesToBytes(numFrames, bytesPerFrame, &numBytes);
230 if (result != AAUDIO_OK) {
Phil Burke1ce4912016-11-21 10:40:25 -0800231 return result;
232 }
233
234 // TODO add timeout to AudioTrack
235 bool blocking = timeoutNanoseconds > 0;
236 ssize_t bytesWritten = mAudioTrack->write(buffer, numBytes, blocking);
237 if (bytesWritten == WOULD_BLOCK) {
238 return 0;
239 } else if (bytesWritten < 0) {
240 ALOGE("invalid write, returned %d", (int)bytesWritten);
Phil Burk5ed503c2017-02-01 09:38:15 -0800241 return AAudioConvert_androidToAAudioResult(bytesWritten);
Phil Burke1ce4912016-11-21 10:40:25 -0800242 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800243 aaudio_size_frames_t framesWritten = (aaudio_size_frames_t)(bytesWritten / bytesPerFrame);
Phil Burke1ce4912016-11-21 10:40:25 -0800244 incrementFramesWritten(framesWritten);
245 return framesWritten;
246}
247
Phil Burk5ed503c2017-02-01 09:38:15 -0800248aaudio_result_t AudioStreamTrack::setBufferSize(aaudio_size_frames_t requestedFrames,
249 aaudio_size_frames_t *actualFrames)
Phil Burke1ce4912016-11-21 10:40:25 -0800250{
251 ssize_t result = mAudioTrack->setBufferSizeInFrames(requestedFrames);
252 if (result != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800253 return AAudioConvert_androidToAAudioResult(result);
Phil Burke1ce4912016-11-21 10:40:25 -0800254 } else {
255 *actualFrames = result;
Phil Burk5ed503c2017-02-01 09:38:15 -0800256 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800257 }
258}
259
Phil Burk5ed503c2017-02-01 09:38:15 -0800260aaudio_size_frames_t AudioStreamTrack::getBufferSize() const
Phil Burke1ce4912016-11-21 10:40:25 -0800261{
Phil Burk5ed503c2017-02-01 09:38:15 -0800262 return static_cast<aaudio_size_frames_t>(mAudioTrack->getBufferSizeInFrames());
Phil Burke1ce4912016-11-21 10:40:25 -0800263}
264
Phil Burk5ed503c2017-02-01 09:38:15 -0800265aaudio_size_frames_t AudioStreamTrack::getBufferCapacity() const
Phil Burke1ce4912016-11-21 10:40:25 -0800266{
Phil Burk5ed503c2017-02-01 09:38:15 -0800267 return static_cast<aaudio_size_frames_t>(mAudioTrack->frameCount());
Phil Burke1ce4912016-11-21 10:40:25 -0800268}
269
270int32_t AudioStreamTrack::getXRunCount() const
271{
272 return static_cast<int32_t>(mAudioTrack->getUnderrunCount());
273}
274
275int32_t AudioStreamTrack::getFramesPerBurst() const
276{
277 return 192; // TODO add query to AudioTrack.cpp
278}
279
Phil Burk5ed503c2017-02-01 09:38:15 -0800280aaudio_position_frames_t AudioStreamTrack::getFramesRead() {
281 aaudio_wrapping_frames_t position;
Phil Burke1ce4912016-11-21 10:40:25 -0800282 status_t result;
283 switch (getState()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800284 case AAUDIO_STREAM_STATE_STARTING:
285 case AAUDIO_STREAM_STATE_STARTED:
286 case AAUDIO_STREAM_STATE_STOPPING:
Phil Burke1ce4912016-11-21 10:40:25 -0800287 result = mAudioTrack->getPosition(&position);
288 if (result == OK) {
289 mFramesRead.update32(position);
290 }
291 break;
292 default:
293 break;
294 }
295 return AudioStream::getFramesRead();
296}