1 /* 2 * Copyright 2020 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 package org.hyphonate.megaaudio.player; 17 18 import android.media.AudioDeviceInfo; 19 import android.media.AudioFormat; 20 import android.media.AudioTimestamp; 21 import android.media.AudioTrack; 22 import android.util.Log; 23 24 import org.hyphonate.megaaudio.common.StreamBase; 25 26 /** 27 * Implementation of abstract Player class implemented for the Android Java-based audio playback 28 * API, i.e. AudioTrack. 29 */ 30 public class JavaPlayer extends Player { 31 @SuppressWarnings("unused") private static String TAG = JavaPlayer.class.getSimpleName(); 32 @SuppressWarnings("unused") private static final boolean LOG = false; 33 34 /* 35 * Player infrastructure 36 */ 37 /* The AudioTrack for playing the audio stream */ 38 private AudioTrack mAudioTrack; 39 40 private AudioSource mAudioSource; 41 42 // Playback state 43 /** <code>true</code> if currently playing audio data */ 44 private boolean mPlaying; 45 46 /* 47 * Data buffers 48 */ 49 /** Number of FRAMES of audio data in a burst buffer */ 50 private int mNumBufferFrames; 51 52 /** The Burst Buffer. This is the buffer we fill with audio and feed into the AudioTrack. */ 53 private float[] mAudioBuffer; 54 55 // Player-specific extension 56 public AudioTrack getAudioTrack() { return mAudioTrack; } 57 58 public JavaPlayer(AudioSourceProvider sourceProvider) { 59 super(sourceProvider); 60 mNumBufferFrames = -1; // TODO need error defines 61 } 62 63 @Override 64 public AudioSource getAudioSource() { 65 return mAudioSource; 66 } 67 68 // 69 // Status 70 // 71 @Override 72 public boolean isPlaying() { 73 return mPlaying; 74 } 75 76 /** 77 * Allocates the array for the burst buffer. 78 */ 79 private void allocBurstBuffer() { 80 // pad it by 1 frame. This allows some sources to not have to worry about 81 // handling the end-of-buffer edge case. i.e. a "Guard Point" for interpolation. 82 mAudioBuffer = new float[(mNumBufferFrames + 1) * mChannelCount]; 83 } 84 85 // 86 // Attributes 87 // 88 /** 89 * @return The number of frames of audio data contained in the internal buffer. 90 */ 91 @Override 92 public int getNumBufferFrames() { 93 return mNumBufferFrames; 94 } 95 96 @Override 97 public int getRoutedDeviceId() { 98 if (mAudioTrack != null) { 99 AudioDeviceInfo routedDevice = mAudioTrack.getRoutedDevice(); 100 return routedDevice != null ? routedDevice.getId() : ROUTED_DEVICE_ID_INVALID; 101 } else { 102 return ROUTED_DEVICE_ID_INVALID; 103 } 104 } 105 106 /* 107 * State 108 */ 109 @Override 110 public int setupStream(int channelCount, int sampleRate, int numBufferFrames) { 111 if (LOG) { 112 Log.i(TAG, "setupStream(chans:" + channelCount + ", rate:" + sampleRate + 113 ", frames:" + numBufferFrames); 114 } 115 116 mChannelCount = channelCount; 117 mSampleRate = sampleRate; 118 mNumBufferFrames = numBufferFrames; 119 120 mAudioSource = mSourceProvider.getJavaSource(); 121 mAudioSource.init(mNumBufferFrames, mChannelCount); 122 123 try { 124 int bufferSizeInBytes = mNumBufferFrames * mChannelCount 125 * sampleSizeInBytes(AudioFormat.ENCODING_PCM_FLOAT); 126 mAudioTrack = new AudioTrack.Builder() 127 .setPerformanceMode(AudioTrack.PERFORMANCE_MODE_LOW_LATENCY) 128 .setAudioFormat(new AudioFormat.Builder() 129 .setEncoding(AudioFormat.ENCODING_PCM_FLOAT) 130 .setSampleRate(mSampleRate) 131 .setChannelIndexMask(StreamBase.channelCountToIndexMask(mChannelCount)) 132 // .setChannelMask(channelMask) 133 .build()) 134 .setBufferSizeInBytes(bufferSizeInBytes) 135 .build(); 136 137 allocBurstBuffer(); 138 mAudioTrack.setPreferredDevice(mRouteDevice); 139 } catch (UnsupportedOperationException ex) { 140 if (LOG) { 141 Log.i(TAG, "Couldn't open AudioTrack: " + ex); 142 } 143 mAudioTrack = null; 144 return ERROR_UNSUPPORTED; 145 } 146 147 return OK; 148 } 149 150 @Override 151 public int teardownStream() { 152 stopStream(); 153 154 waitForStreamThreadToExit(); 155 156 if (mAudioTrack != null) { 157 mAudioTrack.release(); 158 mAudioTrack = null; 159 } 160 161 mChannelCount = 0; 162 mSampleRate = 0; 163 164 //TODO - Retrieve errors from above 165 return OK; 166 } 167 168 /** 169 * Allocates the underlying AudioTrack and begins Playback. 170 * @return True if the stream is successfully started. 171 * 172 * This method returns when the start operation is complete, but before the first 173 * call to the AudioSource.pull() method. 174 */ 175 @Override 176 public int startStream() { 177 if (mAudioTrack == null) { 178 return ERROR_INVALID_STATE; 179 } 180 waitForStreamThreadToExit(); // just to be sure. 181 182 mStreamThread = new Thread(new StreamPlayerRunnable(), "StreamPlayer Thread"); 183 mPlaying = true; 184 mStreamThread.start(); 185 186 return OK; 187 } 188 189 /** 190 * Marks the stream for stopping on the next callback from the underlying system. 191 * 192 * Returns immediately, though a call to AudioSource.pull() may be in progress. 193 */ 194 @Override 195 public int stopStream() { 196 mPlaying = false; 197 return OK; 198 } 199 200 /** 201 * Gets a timestamp from the audio stream 202 * @param timestamp 203 * @return 204 */ 205 public boolean getTimestamp(AudioTimestamp timestamp) { 206 return mPlaying ? mAudioTrack.getTimestamp(timestamp) : false; 207 } 208 209 // 210 // StreamPlayerRunnable 211 // 212 /** 213 * Implements the <code>run</code> method for the playback thread. 214 * Gets initial audio data and starts the AudioTrack. Then continuously provides audio data 215 * until the flag <code>mPlaying</code> is set to false (in the stop() method). 216 */ 217 private class StreamPlayerRunnable implements Runnable { 218 @Override 219 public void run() { 220 final int numBufferSamples = mNumBufferFrames * mChannelCount; 221 222 mAudioTrack.play(); 223 while (mPlaying) { 224 mAudioSource.pull(mAudioBuffer, mNumBufferFrames, mChannelCount); 225 226 onPull(); 227 228 int numSamplesWritten = mAudioTrack.write( 229 mAudioBuffer,0, numBufferSamples, AudioTrack.WRITE_BLOCKING); 230 if (numSamplesWritten < 0) { 231 // error 232 if (LOG) { 233 Log.i(TAG, "AudioTrack write error: " + numSamplesWritten); 234 } 235 stopStream(); 236 } else if (numSamplesWritten < numBufferSamples) { 237 // end of stream 238 if (LOG) { 239 Log.i(TAG, "Stream Complete."); 240 } 241 stopStream(); 242 } 243 } 244 } 245 } 246 } 247