/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "ARTPAssembler.h" #include #include #include #include #include namespace android { ARTPAssembler::ARTPAssembler() : mFirstFailureTimeUs(-1) { } void ARTPAssembler::onPacketReceived(const sp &source) { AssemblyStatus status; for (;;) { status = assembleMore(source); if (status == WRONG_SEQUENCE_NUMBER) { if (mFirstFailureTimeUs >= 0) { if (ALooper::GetNowUs() - mFirstFailureTimeUs > 10000ll) { mFirstFailureTimeUs = -1; // LOG(VERBOSE) << "waited too long for packet."; packetLost(); continue; } } else { mFirstFailureTimeUs = ALooper::GetNowUs(); } break; } else { mFirstFailureTimeUs = -1; if (status == NOT_ENOUGH_DATA) { break; } } } } // static void ARTPAssembler::CopyTimes(const sp &to, const sp &from) { uint32_t rtpTime; CHECK(from->meta()->findInt32("rtp-time", (int32_t *)&rtpTime)); to->meta()->setInt32("rtp-time", rtpTime); // Copy the seq number. to->setInt32Data(from->int32Data()); } // static sp ARTPAssembler::MakeADTSCompoundFromAACFrames( unsigned profile, unsigned samplingFreqIndex, unsigned channelConfig, const List > &frames) { size_t totalSize = 0; for (List >::const_iterator it = frames.begin(); it != frames.end(); ++it) { // Each frame is prefixed by a 7 byte ADTS header totalSize += (*it)->size() + 7; } sp accessUnit = new ABuffer(totalSize); size_t offset = 0; for (List >::const_iterator it = frames.begin(); it != frames.end(); ++it) { sp nal = *it; uint8_t *dst = accessUnit->data() + offset; static const unsigned kADTSId = 0; static const unsigned kADTSLayer = 0; static const unsigned kADTSProtectionAbsent = 1; unsigned frameLength = nal->size() + 7; dst[0] = 0xff; dst[1] = 0xf0 | (kADTSId << 3) | (kADTSLayer << 1) | kADTSProtectionAbsent; dst[2] = (profile << 6) | (samplingFreqIndex << 2) | (channelConfig >> 2); dst[3] = ((channelConfig & 3) << 6) | (frameLength >> 11); dst[4] = (frameLength >> 3) & 0xff; dst[5] = (frameLength & 7) << 5; dst[6] = 0x00; memcpy(dst + 7, nal->data(), nal->size()); offset += nal->size() + 7; } CopyTimes(accessUnit, *frames.begin()); return accessUnit; } // static sp ARTPAssembler::MakeCompoundFromPackets( const List > &packets) { size_t totalSize = 0; for (List >::const_iterator it = packets.begin(); it != packets.end(); ++it) { totalSize += (*it)->size(); } sp accessUnit = new ABuffer(totalSize); size_t offset = 0; for (List >::const_iterator it = packets.begin(); it != packets.end(); ++it) { sp nal = *it; memcpy(accessUnit->data() + offset, nal->data(), nal->size()); offset += nal->size(); } CopyTimes(accessUnit, *packets.begin()); return accessUnit; } } // namespace android