AacBqToPcmCbRenderer.cpp revision a6585530d8cf92658042c001b6a27fc397c62b9f
1/* 2 * Copyright (C) 2011 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 USE_LOG SLAndroidLogLevel_Debug 18 19#include "sles_allinclusive.h" 20#include "android/include/AacBqToPcmCbRenderer.h" 21#include "android/channels.h" 22#include <media/stagefright/foundation/ADebug.h> 23 24namespace android { 25 26// ADTS header size is 7, but frame size information ends on byte 6 (when counting from byte 1) 27#define ADTS_HEADER_SIZE_UP_TO_FRAMESIZE 6 28 29/** 30 * Returns the size of an AAC ADTS frame. 31 * Note that if the returned value + offset > size, it means that a partial frame starts at that 32 * offset, but this function will still return the size of the full frame. 33 * @param data pointer to the compressed audio data 34 * @param offset offset in bytes relative to data of where the frame is supposed to start 35 * @param size the size in bytes of the data block starting at data 36 * @return the size in bytes of the AAC ADTS frame starting at the given offset of the given 37 * memory address, 0 if the frame couldn't be parsed. 38 */ 39static size_t getAdtsFrameSize(const uint8_t *data, off64_t offset, size_t size) { 40 size_t frameSize = 0; 41 42 if (!(offset + ADTS_HEADER_SIZE_UP_TO_FRAMESIZE < (off64_t) size)) { 43 SL_LOGE("AacBqToPcmCbRenderer::getAdtsFrameSize() returns 0 (can't read syncword or header)" 44 ); 45 return 0; 46 } 47 48 const uint8_t *syncword = data + offset; 49 if ((syncword[0] != 0xff) || ((syncword[1] & 0xf6) != 0xf0)) { 50 SL_LOGE("AacBqToPcmCbRenderer::getAdtsFrameSize() returns 0 (wrong syncword)"); 51 return 0; 52 } 53 54 const uint8_t protectionAbsent = data[offset+1] & 0x1; 55 56 const uint8_t* header = data + offset + 3; 57 frameSize = (header[0] & 0x3) << 11 | header[1] << 3 | header[2] >> 5; 58 // the frame size read already contains the size of the header, so no need to add it here 59 60 // protectionAbsent is 0 if there is CRC 61 static const size_t kAdtsHeaderLengthNoCrc = 7; 62 static const size_t kAdtsHeaderLengthWithCrc = 9; 63 size_t headSize = protectionAbsent ? kAdtsHeaderLengthNoCrc : kAdtsHeaderLengthWithCrc; 64 if (headSize > frameSize) { 65 SL_LOGE("AacBqToPcmCbRenderer::getAdtsFrameSize() returns 0 (frameSize %zu < headSize %zu)", 66 frameSize, headSize); 67 return 0; 68 } 69 70 SL_LOGV("AacBqToPcmCbRenderer::getAdtsFrameSize() returns %u", frameSize); 71 72 return frameSize; 73} 74 75/** 76 * Returns whether a block of memory starts and ends on AAC ADTS frame boundaries 77 * @param data pointer to the compressed audio data 78 * @param size the size in bytes of the data block to validate 79 * @return SL_RESULT_SUCCESS if there is AAC ADTS data, and it starts and ends on frame boundaries, 80 * or an appropriate error code otherwise: 81 * SL_RESULT_PARAMETER_INVALID if not possible to attempt validation of even one frame 82 * SL_RESULT_CONTENT_CORRUPTED if the frame contents are otherwise invalid 83 */ 84SLresult AacBqToPcmCbRenderer::validateBufferStartEndOnFrameBoundaries(void* data, size_t size) 85{ 86 off64_t offset = 0; 87 size_t frameSize = 0; 88 89 if ((NULL == data) || (size == 0)) { 90 SL_LOGE("No ADTS to validate"); 91 return SL_RESULT_PARAMETER_INVALID; 92 } 93 94 while (offset < (off64_t) size) { 95 if ((frameSize = getAdtsFrameSize((uint8_t *)data, offset, size)) == 0) { 96 SL_LOGE("found ADTS frame of size 0 at offset %lld", (long long) offset); 97 return SL_RESULT_CONTENT_CORRUPTED; 98 } 99 //SL_LOGV("last good offset %llu", offset); 100 offset += frameSize; 101 if (offset > (off64_t) size) { 102 SL_LOGE("found incomplete ADTS frame at end of data"); 103 return SL_RESULT_CONTENT_CORRUPTED; 104 } 105 } 106 if (offset != (off64_t) size) { 107 SL_LOGE("ADTS parsing error: reached end of incomplete frame"); 108 } 109 assert(offset == (off64_t) size); 110 return SL_RESULT_SUCCESS; 111} 112 113//-------------------------------------------------------------------------------------------------- 114AacBqToPcmCbRenderer::AacBqToPcmCbRenderer(const AudioPlayback_Parameters* params, 115 IAndroidBufferQueue *androidBufferQueue) : 116 AudioToCbRenderer(params), 117 mBqSource(new BufferQueueSource(androidBufferQueue)) 118{ 119 SL_LOGD("AacBqToPcmCbRenderer::AacBqToPcmCbRenderer()"); 120} 121 122 123AacBqToPcmCbRenderer::~AacBqToPcmCbRenderer() { 124 SL_LOGD("AacBqToPcmCbRenderer::~AacBqToPcmCbRenderer()"); 125 126} 127 128 129//-------------------------------------------------- 130// Event handlers 131void AacBqToPcmCbRenderer::onPrepare() { 132 SL_LOGD("AacBqToPcmCbRenderer::onPrepare()"); 133 Mutex::Autolock _l(mBufferSourceLock); 134 135 // Initialize the PCM format info with the known parameters before the start of the decode 136 { 137 android::Mutex::Autolock autoLock(mPcmFormatLock); 138 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_BITSPERSAMPLE] = SL_PCMSAMPLEFORMAT_FIXED_16; 139 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_CONTAINERSIZE] = 16; 140 //FIXME not true on all platforms 141 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_ENDIANNESS] = SL_BYTEORDER_LITTLEENDIAN; 142 // initialization with the default values: they will be replaced by the actual values 143 // once the decoder has figured them out 144 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_NUMCHANNELS] = UNKNOWN_NUMCHANNELS; 145 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_SAMPLERATE] = UNKNOWN_SAMPLERATE; 146 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_CHANNELMASK] = UNKNOWN_CHANNELMASK; 147 } 148 149 sp<MediaExtractor> extractor = new AacAdtsExtractor(mBqSource); 150 151 // only decoding a single track of data 152 const size_t kTrackToDecode = 0; 153 154 sp<MediaSource> source = extractor->getTrack(kTrackToDecode); 155 if (source == 0) { 156 SL_LOGE("AacBqToPcmCbRenderer::onPrepare: error getting source from extractor"); 157 notifyPrepared(ERROR_UNSUPPORTED); 158 return; 159 } 160 sp<MetaData> meta = extractor->getTrackMetaData(kTrackToDecode); 161 162 // the audio content is not raw PCM, so we need a decoder 163 OMXClient client; 164 CHECK_EQ(client.connect(), (status_t)OK); 165 166 source = OMXCodec::Create( 167 client.interface(), meta, false /* createEncoder */, 168 source); 169 170 if (source == NULL) { 171 SL_LOGE("AacBqToPcmCbRenderer::onPrepare: Could not instantiate decoder."); 172 notifyPrepared(ERROR_UNSUPPORTED); 173 return; 174 } 175 176 meta = source->getFormat(); 177 178 SL_LOGD("AacBqToPcmCbRenderer::onPrepare() after instantiating decoder"); 179 180 if (source->start() != OK) { 181 SL_LOGE("AacBqToPcmCbRenderer::onPrepare() Failed to start source/decoder."); 182 notifyPrepared(MEDIA_ERROR_BASE); 183 return; 184 } 185 186 //--------------------------------- 187 int32_t channelCount; 188 CHECK(meta->findInt32(kKeyChannelCount, &channelCount)); 189 int32_t sr; 190 CHECK(meta->findInt32(kKeySampleRate, &sr)); 191 // FIXME similar to AudioSfDecoder::onPrepare() 192 193 // already "good to go" (compare to AudioSfDecoder::onPrepare) 194 mCacheStatus = kStatusHigh; 195 mCacheFill = 1000; 196 notifyStatus(); 197 notifyCacheFill(); 198 199 { 200 android::Mutex::Autolock autoLock(mPcmFormatLock); 201 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_SAMPLERATE] = sr; 202 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_NUMCHANNELS] = channelCount; 203 mPcmFormatValues[ANDROID_KEY_INDEX_PCMFORMAT_CHANNELMASK] = 204 channelCountToMask(channelCount); 205 } 206 SL_LOGV("AacBqToPcmCbRenderer::onPrepare() channel count=%d SR=%d", 207 channelCount, sr); 208 209 //--------------------------------- 210 // The data source, and audio source (a decoder) are ready to be used 211 mDataSource = mBqSource; 212 mAudioSource = source; 213 mAudioSourceStarted = true; 214 215 //------------------------------------- 216 // signal successful completion of prepare 217 mStateFlags |= kFlagPrepared; 218 219 // skipping past AudioToCbRenderer and AudioSfDecoder 220 GenericPlayer::onPrepare(); 221 222 SL_LOGD("AacBqToPcmCbRenderer::onPrepare() done, mStateFlags=0x%x", mStateFlags); 223} 224 225} // namespace android 226