blob: 291e56c28d0d3b07acb44396c1014c8ee2e54293 [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
24#include <oboe/OboeAudio.h>
25#include "AudioClock.h"
26#include "AudioStreamTrack.h"
27
28
29using namespace android;
30using namespace oboe;
31
32/*
33 * Create a stream that uses the AudioTrack.
34 */
35AudioStreamTrack::AudioStreamTrack()
36 : AudioStream()
37{
38}
39
40AudioStreamTrack::~AudioStreamTrack()
41{
42 const oboe_stream_state_t state = getState();
43 bool bad = !(state == OBOE_STREAM_STATE_UNINITIALIZED || state == OBOE_STREAM_STATE_CLOSED);
44 ALOGE_IF(bad, "stream not closed, in state %d", state);
45}
46
47oboe_result_t AudioStreamTrack::open(const AudioStreamBuilder& builder)
48{
49 oboe_result_t result = OBOE_OK;
50
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.
58 int32_t samplesPerFrame = (getSamplesPerFrame() == OBOE_UNSPECIFIED)
59 ? 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.
69 audio_format_t format = (getFormat() == OBOE_UNSPECIFIED)
70 ? AUDIO_FORMAT_PCM_FLOAT
71 : OboeConvert_oboeToAndroidDataFormat(getFormat());
72
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 Burkdec33ab2017-01-17 14:48:16 -080093 return OboeConvert_androidToOboeResult(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());
99 setFormat(OboeConvert_androidToOboeDataFormat(mAudioTrack->format()));
100
101 setState(OBOE_STREAM_STATE_OPEN);
102
103 return OBOE_OK;
104}
105
106oboe_result_t AudioStreamTrack::close()
107{
108 // TODO maybe add close() or release() to AudioTrack API then call it from here
109 if (getState() != OBOE_STREAM_STATE_CLOSED) {
110 mAudioTrack.clear(); // TODO is this right?
111 setState(OBOE_STREAM_STATE_CLOSED);
112 }
113 return OBOE_OK;
114}
115
116oboe_result_t AudioStreamTrack::requestStart()
117{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800118 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800119 return OBOE_ERROR_INVALID_STATE;
120 }
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 Burkdec33ab2017-01-17 14:48:16 -0800124 return OboeConvert_androidToOboeResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800125 }
126 err = mAudioTrack->start();
127 if (err != OK) {
Phil Burkdec33ab2017-01-17 14:48:16 -0800128 return OboeConvert_androidToOboeResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800129 } else {
130 setState(OBOE_STREAM_STATE_STARTING);
131 }
132 return OBOE_OK;
133}
134
135oboe_result_t AudioStreamTrack::requestPause()
136{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800137 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800138 return OBOE_ERROR_INVALID_STATE;
139 } else if (getState() != OBOE_STREAM_STATE_STARTING
140 && getState() != OBOE_STREAM_STATE_STARTED) {
141 ALOGE("requestPause(), called when state is %s", Oboe_convertStreamStateToText(getState()));
142 return OBOE_ERROR_INVALID_STATE;
143 }
144 setState(OBOE_STREAM_STATE_PAUSING);
145 mAudioTrack->pause();
146 status_t err = mAudioTrack->getPosition(&mPositionWhenPausing);
147 if (err != OK) {
Phil Burkdec33ab2017-01-17 14:48:16 -0800148 return OboeConvert_androidToOboeResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800149 }
150 return OBOE_OK;
151}
152
153oboe_result_t AudioStreamTrack::requestFlush() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800154 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800155 return OBOE_ERROR_INVALID_STATE;
156 } else if (getState() != OBOE_STREAM_STATE_PAUSED) {
157 return OBOE_ERROR_INVALID_STATE;
158 }
159 setState(OBOE_STREAM_STATE_FLUSHING);
160 incrementFramesRead(getFramesWritten() - getFramesRead());
161 mAudioTrack->flush();
162 mFramesWritten.reset32();
163 return OBOE_OK;
164}
165
166oboe_result_t AudioStreamTrack::requestStop() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800167 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800168 return OBOE_ERROR_INVALID_STATE;
169 }
170 setState(OBOE_STREAM_STATE_STOPPING);
171 incrementFramesRead(getFramesWritten() - getFramesRead()); // TODO review
172 mAudioTrack->stop();
173 mFramesWritten.reset32();
174 return OBOE_OK;
175}
176
177oboe_result_t AudioStreamTrack::updateState()
178{
179 status_t err;
180 oboe_wrapping_frames_t position;
181 switch (getState()) {
182 // TODO add better state visibility to AudioTrack
183 case OBOE_STREAM_STATE_STARTING:
184 if (mAudioTrack->hasStarted()) {
185 setState(OBOE_STREAM_STATE_STARTED);
186 }
187 break;
188 case OBOE_STREAM_STATE_PAUSING:
189 if (mAudioTrack->stopped()) {
190 err = mAudioTrack->getPosition(&position);
191 if (err != OK) {
Phil Burkdec33ab2017-01-17 14:48:16 -0800192 return OboeConvert_androidToOboeResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800193 } else if (position == mPositionWhenPausing) {
194 // Has stream really stopped advancing?
195 setState(OBOE_STREAM_STATE_PAUSED);
196 }
197 mPositionWhenPausing = position;
198 }
199 break;
200 case OBOE_STREAM_STATE_FLUSHING:
201 {
202 err = mAudioTrack->getPosition(&position);
203 if (err != OK) {
Phil Burkdec33ab2017-01-17 14:48:16 -0800204 return OboeConvert_androidToOboeResult(err);
Phil Burke1ce4912016-11-21 10:40:25 -0800205 } else if (position == 0) {
206 // Advance frames read to match written.
207 setState(OBOE_STREAM_STATE_FLUSHED);
208 }
209 }
210 break;
211 case OBOE_STREAM_STATE_STOPPING:
212 if (mAudioTrack->stopped()) {
213 setState(OBOE_STREAM_STATE_STOPPED);
214 }
215 break;
216 default:
217 break;
218 }
219 return OBOE_OK;
220}
221
222oboe_result_t AudioStreamTrack::write(const void *buffer,
223 oboe_size_frames_t numFrames,
224 oboe_nanoseconds_t timeoutNanoseconds)
225{
226 oboe_size_frames_t bytesPerFrame = getBytesPerFrame();
227 oboe_size_bytes_t numBytes;
228 oboe_result_t result = OboeConvert_framesToBytes(numFrames, bytesPerFrame, &numBytes);
229 if (result != OBOE_OK) {
230 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 Burkdec33ab2017-01-17 14:48:16 -0800240 return OboeConvert_androidToOboeResult(bytesWritten);
Phil Burke1ce4912016-11-21 10:40:25 -0800241 }
242 oboe_size_frames_t framesWritten = (oboe_size_frames_t)(bytesWritten / bytesPerFrame);
243 incrementFramesWritten(framesWritten);
244 return framesWritten;
245}
246
247oboe_result_t AudioStreamTrack::setBufferSize(oboe_size_frames_t requestedFrames,
248 oboe_size_frames_t *actualFrames)
249{
250 ssize_t result = mAudioTrack->setBufferSizeInFrames(requestedFrames);
251 if (result != OK) {
Phil Burkdec33ab2017-01-17 14:48:16 -0800252 return OboeConvert_androidToOboeResult(result);
Phil Burke1ce4912016-11-21 10:40:25 -0800253 } else {
254 *actualFrames = result;
255 return OBOE_OK;
256 }
257}
258
259oboe_size_frames_t AudioStreamTrack::getBufferSize() const
260{
261 return static_cast<oboe_size_frames_t>(mAudioTrack->getBufferSizeInFrames());
262}
263
264oboe_size_frames_t AudioStreamTrack::getBufferCapacity() const
265{
266 return static_cast<oboe_size_frames_t>(mAudioTrack->frameCount());
267}
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
279oboe_position_frames_t AudioStreamTrack::getFramesRead() {
280 oboe_wrapping_frames_t position;
281 status_t result;
282 switch (getState()) {
283 case OBOE_STREAM_STATE_STARTING:
284 case OBOE_STREAM_STATE_STARTED:
285 case OBOE_STREAM_STATE_STOPPING:
286 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}