blob: a60b5b437961a383295a0f5bc312f9a96d2468b0 [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;
67 size_t frameCount = 0;
68 // TODO implement an unspecified AudioTrack format then use that.
Phil Burk5ed503c2017-02-01 09:38:15 -080069 audio_format_t format = (getFormat() == AAUDIO_UNSPECIFIED)
Phil Burke1ce4912016-11-21 10:40:25 -080070 ? AUDIO_FORMAT_PCM_FLOAT
Phil Burk5ed503c2017-02-01 09:38:15 -080071 : AAudioConvert_aaudioToAndroidDataFormat(getFormat());
Phil Burke1ce4912016-11-21 10:40:25 -080072
73 mAudioTrack = new AudioTrack(
74 (audio_stream_type_t) AUDIO_STREAM_MUSIC,
75 getSampleRate(),
76 format,
77 channelMask,
78 frameCount,
79 flags,
80 callback,
Phil Burkd8bdcab2017-01-03 17:20:30 -080081 nullptr, // user callback data
82 0, // notificationFrames
Phil Burke1ce4912016-11-21 10:40:25 -080083 AUDIO_SESSION_ALLOCATE,
84 AudioTrack::transfer_type::TRANSFER_SYNC // TODO - this does not allow FAST
85 );
86
87 // Did we get a valid track?
88 status_t status = mAudioTrack->initCheck();
Phil Burkd8bdcab2017-01-03 17:20:30 -080089 ALOGD("AudioStreamTrack::open(), initCheck() returned %d", status);
Phil Burkdec33ab2017-01-17 14:48:16 -080090 if (status != NO_ERROR) {
Phil Burke1ce4912016-11-21 10:40:25 -080091 close();
92 ALOGE("AudioStreamTrack::open(), initCheck() returned %d", status);
Phil Burk5ed503c2017-02-01 09:38:15 -080093 return AAudioConvert_androidToAAudioResult(status);
Phil Burke1ce4912016-11-21 10:40:25 -080094 }
95
96 // Get the actual values from the AudioTrack.
97 setSamplesPerFrame(mAudioTrack->channelCount());
98 setSampleRate(mAudioTrack->getSampleRate());
Phil Burk5ed503c2017-02-01 09:38:15 -080099 setFormat(AAudioConvert_androidToAAudioDataFormat(mAudioTrack->format()));
Phil Burke1ce4912016-11-21 10:40:25 -0800100
Phil Burk5ed503c2017-02-01 09:38:15 -0800101 setState(AAUDIO_STREAM_STATE_OPEN);
Phil Burke1ce4912016-11-21 10:40:25 -0800102
Phil Burk5ed503c2017-02-01 09:38:15 -0800103 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800104}
105
Phil Burk5ed503c2017-02-01 09:38:15 -0800106aaudio_result_t AudioStreamTrack::close()
Phil Burke1ce4912016-11-21 10:40:25 -0800107{
108 // TODO maybe add close() or release() to AudioTrack API then call it from here
Phil Burk5ed503c2017-02-01 09:38:15 -0800109 if (getState() != AAUDIO_STREAM_STATE_CLOSED) {
Phil Burke1ce4912016-11-21 10:40:25 -0800110 mAudioTrack.clear(); // TODO is this right?
Phil Burk5ed503c2017-02-01 09:38:15 -0800111 setState(AAUDIO_STREAM_STATE_CLOSED);
Phil Burke1ce4912016-11-21 10:40:25 -0800112 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800113 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800114}
115
Phil Burk5ed503c2017-02-01 09:38:15 -0800116aaudio_result_t AudioStreamTrack::requestStart()
Phil Burke1ce4912016-11-21 10:40:25 -0800117{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800118 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800119 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800120 }
121 // Get current position so we can detect when the track is playing.
122 status_t err = mAudioTrack->getPosition(&mPositionWhenStarting);
123 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800124 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800125 }
126 err = mAudioTrack->start();
127 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800128 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800129 } else {
Phil Burk5ed503c2017-02-01 09:38:15 -0800130 setState(AAUDIO_STREAM_STATE_STARTING);
Phil Burke1ce4912016-11-21 10:40:25 -0800131 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800132 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800133}
134
Phil Burk5ed503c2017-02-01 09:38:15 -0800135aaudio_result_t AudioStreamTrack::requestPause()
Phil Burke1ce4912016-11-21 10:40:25 -0800136{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800137 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800138 return AAUDIO_ERROR_INVALID_STATE;
139 } else if (getState() != AAUDIO_STREAM_STATE_STARTING
140 && getState() != AAUDIO_STREAM_STATE_STARTED) {
141 ALOGE("requestPause(), called when state is %s", AAudio_convertStreamStateToText(getState()));
142 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800143 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800144 setState(AAUDIO_STREAM_STATE_PAUSING);
Phil Burke1ce4912016-11-21 10:40:25 -0800145 mAudioTrack->pause();
146 status_t err = mAudioTrack->getPosition(&mPositionWhenPausing);
147 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800148 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800149 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800150 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800151}
152
Phil Burk5ed503c2017-02-01 09:38:15 -0800153aaudio_result_t AudioStreamTrack::requestFlush() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800154 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800155 return AAUDIO_ERROR_INVALID_STATE;
156 } else if (getState() != AAUDIO_STREAM_STATE_PAUSED) {
157 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800158 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800159 setState(AAUDIO_STREAM_STATE_FLUSHING);
Phil Burke1ce4912016-11-21 10:40:25 -0800160 incrementFramesRead(getFramesWritten() - getFramesRead());
161 mAudioTrack->flush();
162 mFramesWritten.reset32();
Phil Burk5ed503c2017-02-01 09:38:15 -0800163 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800164}
165
Phil Burk5ed503c2017-02-01 09:38:15 -0800166aaudio_result_t AudioStreamTrack::requestStop() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800167 if (mAudioTrack.get() == nullptr) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800168 return AAUDIO_ERROR_INVALID_STATE;
Phil Burke1ce4912016-11-21 10:40:25 -0800169 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800170 setState(AAUDIO_STREAM_STATE_STOPPING);
Phil Burke1ce4912016-11-21 10:40:25 -0800171 incrementFramesRead(getFramesWritten() - getFramesRead()); // TODO review
172 mAudioTrack->stop();
173 mFramesWritten.reset32();
Phil Burk5ed503c2017-02-01 09:38:15 -0800174 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800175}
176
Phil Burk5ed503c2017-02-01 09:38:15 -0800177aaudio_result_t AudioStreamTrack::updateState()
Phil Burke1ce4912016-11-21 10:40:25 -0800178{
179 status_t err;
Phil Burk5ed503c2017-02-01 09:38:15 -0800180 aaudio_wrapping_frames_t position;
Phil Burke1ce4912016-11-21 10:40:25 -0800181 switch (getState()) {
182 // TODO add better state visibility to AudioTrack
Phil Burk5ed503c2017-02-01 09:38:15 -0800183 case AAUDIO_STREAM_STATE_STARTING:
Phil Burke1ce4912016-11-21 10:40:25 -0800184 if (mAudioTrack->hasStarted()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800185 setState(AAUDIO_STREAM_STATE_STARTED);
Phil Burke1ce4912016-11-21 10:40:25 -0800186 }
187 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800188 case AAUDIO_STREAM_STATE_PAUSING:
Phil Burke1ce4912016-11-21 10:40:25 -0800189 if (mAudioTrack->stopped()) {
190 err = mAudioTrack->getPosition(&position);
191 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800192 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800193 } else if (position == mPositionWhenPausing) {
194 // Has stream really stopped advancing?
Phil Burk5ed503c2017-02-01 09:38:15 -0800195 setState(AAUDIO_STREAM_STATE_PAUSED);
Phil Burke1ce4912016-11-21 10:40:25 -0800196 }
197 mPositionWhenPausing = position;
198 }
199 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800200 case AAUDIO_STREAM_STATE_FLUSHING:
Phil Burke1ce4912016-11-21 10:40:25 -0800201 {
202 err = mAudioTrack->getPosition(&position);
203 if (err != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800204 return AAudioConvert_androidToAAudioResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800205 } else if (position == 0) {
206 // Advance frames read to match written.
Phil Burk5ed503c2017-02-01 09:38:15 -0800207 setState(AAUDIO_STREAM_STATE_FLUSHED);
Phil Burke1ce4912016-11-21 10:40:25 -0800208 }
209 }
210 break;
Phil Burk5ed503c2017-02-01 09:38:15 -0800211 case AAUDIO_STREAM_STATE_STOPPING:
Phil Burke1ce4912016-11-21 10:40:25 -0800212 if (mAudioTrack->stopped()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800213 setState(AAUDIO_STREAM_STATE_STOPPED);
Phil Burke1ce4912016-11-21 10:40:25 -0800214 }
215 break;
216 default:
217 break;
218 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800219 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800220}
221
Phil Burk5ed503c2017-02-01 09:38:15 -0800222aaudio_result_t AudioStreamTrack::write(const void *buffer,
223 aaudio_size_frames_t numFrames,
224 aaudio_nanoseconds_t timeoutNanoseconds)
Phil Burke1ce4912016-11-21 10:40:25 -0800225{
Phil Burk5ed503c2017-02-01 09:38:15 -0800226 aaudio_size_frames_t bytesPerFrame = getBytesPerFrame();
227 aaudio_size_bytes_t numBytes;
228 aaudio_result_t result = AAudioConvert_framesToBytes(numFrames, bytesPerFrame, &numBytes);
229 if (result != AAUDIO_OK) {
Phil Burke1ce4912016-11-21 10:40:25 -0800230 return result;
231 }
232
233 // TODO add timeout to AudioTrack
234 bool blocking = timeoutNanoseconds > 0;
235 ssize_t bytesWritten = mAudioTrack->write(buffer, numBytes, blocking);
236 if (bytesWritten == WOULD_BLOCK) {
237 return 0;
238 } else if (bytesWritten < 0) {
239 ALOGE("invalid write, returned %d", (int)bytesWritten);
Phil Burk5ed503c2017-02-01 09:38:15 -0800240 return AAudioConvert_androidToAAudioResult(bytesWritten);
Phil Burke1ce4912016-11-21 10:40:25 -0800241 }
Phil Burk5ed503c2017-02-01 09:38:15 -0800242 aaudio_size_frames_t framesWritten = (aaudio_size_frames_t)(bytesWritten / bytesPerFrame);
Phil Burke1ce4912016-11-21 10:40:25 -0800243 incrementFramesWritten(framesWritten);
244 return framesWritten;
245}
246
Phil Burk5ed503c2017-02-01 09:38:15 -0800247aaudio_result_t AudioStreamTrack::setBufferSize(aaudio_size_frames_t requestedFrames,
248 aaudio_size_frames_t *actualFrames)
Phil Burke1ce4912016-11-21 10:40:25 -0800249{
250 ssize_t result = mAudioTrack->setBufferSizeInFrames(requestedFrames);
251 if (result != OK) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800252 return AAudioConvert_androidToAAudioResult(result);
Phil Burke1ce4912016-11-21 10:40:25 -0800253 } else {
254 *actualFrames = result;
Phil Burk5ed503c2017-02-01 09:38:15 -0800255 return AAUDIO_OK;
Phil Burke1ce4912016-11-21 10:40:25 -0800256 }
257}
258
Phil Burk5ed503c2017-02-01 09:38:15 -0800259aaudio_size_frames_t AudioStreamTrack::getBufferSize() const
Phil Burke1ce4912016-11-21 10:40:25 -0800260{
Phil Burk5ed503c2017-02-01 09:38:15 -0800261 return static_cast<aaudio_size_frames_t>(mAudioTrack->getBufferSizeInFrames());
Phil Burke1ce4912016-11-21 10:40:25 -0800262}
263
Phil Burk5ed503c2017-02-01 09:38:15 -0800264aaudio_size_frames_t AudioStreamTrack::getBufferCapacity() const
Phil Burke1ce4912016-11-21 10:40:25 -0800265{
Phil Burk5ed503c2017-02-01 09:38:15 -0800266 return static_cast<aaudio_size_frames_t>(mAudioTrack->frameCount());
Phil Burke1ce4912016-11-21 10:40:25 -0800267}
268
269int32_t AudioStreamTrack::getXRunCount() const
270{
271 return static_cast<int32_t>(mAudioTrack->getUnderrunCount());
272}
273
274int32_t AudioStreamTrack::getFramesPerBurst() const
275{
276 return 192; // TODO add query to AudioTrack.cpp
277}
278
Phil Burk5ed503c2017-02-01 09:38:15 -0800279aaudio_position_frames_t AudioStreamTrack::getFramesRead() {
280 aaudio_wrapping_frames_t position;
Phil Burke1ce4912016-11-21 10:40:25 -0800281 status_t result;
282 switch (getState()) {
Phil Burk5ed503c2017-02-01 09:38:15 -0800283 case AAUDIO_STREAM_STATE_STARTING:
284 case AAUDIO_STREAM_STATE_STARTED:
285 case AAUDIO_STREAM_STATE_STOPPING:
Phil Burke1ce4912016-11-21 10:40:25 -0800286 result = mAudioTrack->getPosition(&position);
287 if (result == OK) {
288 mFramesRead.update32(position);
289 }
290 break;
291 default:
292 break;
293 }
294 return AudioStream::getFramesRead();
295}