blob: b2c4ee1a294e358fe79717a61963578a34070a68 [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 Burke1ce4912016-11-21 10:40:25 -080090 // FIXME - this should work - if (status != NO_ERROR) {
91 // But initCheck() is returning 1 !
92 if (status < 0) {
93 close();
94 ALOGE("AudioStreamTrack::open(), initCheck() returned %d", status);
95 return OboeConvert_androidToOboeError(status);
96 }
97
98 // Get the actual values from the AudioTrack.
99 setSamplesPerFrame(mAudioTrack->channelCount());
100 setSampleRate(mAudioTrack->getSampleRate());
101 setFormat(OboeConvert_androidToOboeDataFormat(mAudioTrack->format()));
102
103 setState(OBOE_STREAM_STATE_OPEN);
104
105 return OBOE_OK;
106}
107
108oboe_result_t AudioStreamTrack::close()
109{
110 // TODO maybe add close() or release() to AudioTrack API then call it from here
111 if (getState() != OBOE_STREAM_STATE_CLOSED) {
112 mAudioTrack.clear(); // TODO is this right?
113 setState(OBOE_STREAM_STATE_CLOSED);
114 }
115 return OBOE_OK;
116}
117
118oboe_result_t AudioStreamTrack::requestStart()
119{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800120 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800121 return OBOE_ERROR_INVALID_STATE;
122 }
123 // Get current position so we can detect when the track is playing.
124 status_t err = mAudioTrack->getPosition(&mPositionWhenStarting);
125 if (err != OK) {
126 return OboeConvert_androidToOboeError(err);
127 }
128 err = mAudioTrack->start();
129 if (err != OK) {
130 return OboeConvert_androidToOboeError(err);
131 } else {
132 setState(OBOE_STREAM_STATE_STARTING);
133 }
134 return OBOE_OK;
135}
136
137oboe_result_t AudioStreamTrack::requestPause()
138{
Phil Burkd8bdcab2017-01-03 17:20:30 -0800139 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800140 return OBOE_ERROR_INVALID_STATE;
141 } else if (getState() != OBOE_STREAM_STATE_STARTING
142 && getState() != OBOE_STREAM_STATE_STARTED) {
143 ALOGE("requestPause(), called when state is %s", Oboe_convertStreamStateToText(getState()));
144 return OBOE_ERROR_INVALID_STATE;
145 }
146 setState(OBOE_STREAM_STATE_PAUSING);
147 mAudioTrack->pause();
148 status_t err = mAudioTrack->getPosition(&mPositionWhenPausing);
149 if (err != OK) {
150 return OboeConvert_androidToOboeError(err);
151 }
152 return OBOE_OK;
153}
154
155oboe_result_t AudioStreamTrack::requestFlush() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800156 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800157 return OBOE_ERROR_INVALID_STATE;
158 } else if (getState() != OBOE_STREAM_STATE_PAUSED) {
159 return OBOE_ERROR_INVALID_STATE;
160 }
161 setState(OBOE_STREAM_STATE_FLUSHING);
162 incrementFramesRead(getFramesWritten() - getFramesRead());
163 mAudioTrack->flush();
164 mFramesWritten.reset32();
165 return OBOE_OK;
166}
167
168oboe_result_t AudioStreamTrack::requestStop() {
Phil Burkd8bdcab2017-01-03 17:20:30 -0800169 if (mAudioTrack.get() == nullptr) {
Phil Burke1ce4912016-11-21 10:40:25 -0800170 return OBOE_ERROR_INVALID_STATE;
171 }
172 setState(OBOE_STREAM_STATE_STOPPING);
173 incrementFramesRead(getFramesWritten() - getFramesRead()); // TODO review
174 mAudioTrack->stop();
175 mFramesWritten.reset32();
176 return OBOE_OK;
177}
178
179oboe_result_t AudioStreamTrack::updateState()
180{
181 status_t err;
182 oboe_wrapping_frames_t position;
183 switch (getState()) {
184 // TODO add better state visibility to AudioTrack
185 case OBOE_STREAM_STATE_STARTING:
186 if (mAudioTrack->hasStarted()) {
187 setState(OBOE_STREAM_STATE_STARTED);
188 }
189 break;
190 case OBOE_STREAM_STATE_PAUSING:
191 if (mAudioTrack->stopped()) {
192 err = mAudioTrack->getPosition(&position);
193 if (err != OK) {
194 return OboeConvert_androidToOboeError(err);
195 } else if (position == mPositionWhenPausing) {
196 // Has stream really stopped advancing?
197 setState(OBOE_STREAM_STATE_PAUSED);
198 }
199 mPositionWhenPausing = position;
200 }
201 break;
202 case OBOE_STREAM_STATE_FLUSHING:
203 {
204 err = mAudioTrack->getPosition(&position);
205 if (err != OK) {
206 return OboeConvert_androidToOboeError(err);
207 } else if (position == 0) {
208 // Advance frames read to match written.
209 setState(OBOE_STREAM_STATE_FLUSHED);
210 }
211 }
212 break;
213 case OBOE_STREAM_STATE_STOPPING:
214 if (mAudioTrack->stopped()) {
215 setState(OBOE_STREAM_STATE_STOPPED);
216 }
217 break;
218 default:
219 break;
220 }
221 return OBOE_OK;
222}
223
224oboe_result_t AudioStreamTrack::write(const void *buffer,
225 oboe_size_frames_t numFrames,
226 oboe_nanoseconds_t timeoutNanoseconds)
227{
228 oboe_size_frames_t bytesPerFrame = getBytesPerFrame();
229 oboe_size_bytes_t numBytes;
230 oboe_result_t result = OboeConvert_framesToBytes(numFrames, bytesPerFrame, &numBytes);
231 if (result != OBOE_OK) {
232 return result;
233 }
234
235 // TODO add timeout to AudioTrack
236 bool blocking = timeoutNanoseconds > 0;
237 ssize_t bytesWritten = mAudioTrack->write(buffer, numBytes, blocking);
238 if (bytesWritten == WOULD_BLOCK) {
239 return 0;
240 } else if (bytesWritten < 0) {
241 ALOGE("invalid write, returned %d", (int)bytesWritten);
242 return OboeConvert_androidToOboeError(bytesWritten);
243 }
244 oboe_size_frames_t framesWritten = (oboe_size_frames_t)(bytesWritten / bytesPerFrame);
245 incrementFramesWritten(framesWritten);
246 return framesWritten;
247}
248
249oboe_result_t AudioStreamTrack::setBufferSize(oboe_size_frames_t requestedFrames,
250 oboe_size_frames_t *actualFrames)
251{
252 ssize_t result = mAudioTrack->setBufferSizeInFrames(requestedFrames);
253 if (result != OK) {
254 return OboeConvert_androidToOboeError(result);
255 } else {
256 *actualFrames = result;
257 return OBOE_OK;
258 }
259}
260
261oboe_size_frames_t AudioStreamTrack::getBufferSize() const
262{
263 return static_cast<oboe_size_frames_t>(mAudioTrack->getBufferSizeInFrames());
264}
265
266oboe_size_frames_t AudioStreamTrack::getBufferCapacity() const
267{
268 return static_cast<oboe_size_frames_t>(mAudioTrack->frameCount());
269}
270
271int32_t AudioStreamTrack::getXRunCount() const
272{
273 return static_cast<int32_t>(mAudioTrack->getUnderrunCount());
274}
275
276int32_t AudioStreamTrack::getFramesPerBurst() const
277{
278 return 192; // TODO add query to AudioTrack.cpp
279}
280
281oboe_position_frames_t AudioStreamTrack::getFramesRead() {
282 oboe_wrapping_frames_t position;
283 status_t result;
284 switch (getState()) {
285 case OBOE_STREAM_STATE_STARTING:
286 case OBOE_STREAM_STATE_STARTED:
287 case OBOE_STREAM_STATE_STOPPING:
288 result = mAudioTrack->getPosition(&position);
289 if (result == OK) {
290 mFramesRead.update32(position);
291 }
292 break;
293 default:
294 break;
295 }
296 return AudioStream::getFramesRead();
297}