/* * Copyright (C) 2020 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. */ // #define LOG_NDEBUG 0 #define LOG_TAG "MediaTrackTranscoder" #include #include #include #include namespace android { media_status_t MediaTrackTranscoder::configure( const std::shared_ptr& mediaSampleReader, int trackIndex, const std::shared_ptr& destinationFormat) { std::scoped_lock lock{mStateMutex}; if (mState != UNINITIALIZED) { LOG(ERROR) << "Configure can only be called once"; return AMEDIA_ERROR_UNSUPPORTED; } if (mediaSampleReader == nullptr) { LOG(ERROR) << "MediaSampleReader is null"; return AMEDIA_ERROR_INVALID_PARAMETER; } if (trackIndex < 0 || trackIndex >= mediaSampleReader->getTrackCount()) { LOG(ERROR) << "TrackIndex is invalid " << trackIndex; return AMEDIA_ERROR_INVALID_PARAMETER; } mMediaSampleReader = mediaSampleReader; mTrackIndex = trackIndex; mSourceFormat = std::shared_ptr(mMediaSampleReader->getTrackFormat(mTrackIndex), &AMediaFormat_delete); if (mSourceFormat == nullptr) { LOG(ERROR) << "Unable to get format for track #" << mTrackIndex; return AMEDIA_ERROR_MALFORMED; } media_status_t status = configureDestinationFormat(destinationFormat); if (status != AMEDIA_OK) { LOG(ERROR) << "configure failed with error " << status; return status; } mState = CONFIGURED; return AMEDIA_OK; } bool MediaTrackTranscoder::start() { std::scoped_lock lock{mStateMutex}; if (mState != CONFIGURED) { LOG(ERROR) << "TrackTranscoder must be configured before started"; return false; } mState = STARTED; std::thread([this] { androidSetThreadPriority(0 /* tid (0 = current) */, ANDROID_PRIORITY_BACKGROUND); bool stopped = false; media_status_t status = runTranscodeLoop(&stopped); // Output an EOS sample if the transcoder was stopped. if (stopped) { auto sample = std::make_shared(); sample->info.flags = SAMPLE_FLAG_END_OF_STREAM; onOutputSampleAvailable(sample); } // Notify the client. if (auto callbacks = mTranscoderCallback.lock()) { if (stopped) { callbacks->onTrackStopped(this); } else if (status == AMEDIA_OK) { callbacks->onTrackFinished(this); } else { callbacks->onTrackError(this, status); } } }).detach(); return true; } void MediaTrackTranscoder::stop(bool stopOnSyncSample) { std::scoped_lock lock{mStateMutex}; if (mState == STARTED || (mStopRequest == STOP_ON_SYNC && !stopOnSyncSample)) { mStopRequest = stopOnSyncSample ? STOP_ON_SYNC : STOP_NOW; abortTranscodeLoop(); mState = STOPPED; } else { LOG(WARNING) << "TrackTranscoder must be started before stopped"; } } void MediaTrackTranscoder::notifyTrackFormatAvailable() { if (auto callbacks = mTranscoderCallback.lock()) { callbacks->onTrackFormatAvailable(this); } } void MediaTrackTranscoder::onOutputSampleAvailable(const std::shared_ptr& sample) { std::scoped_lock lock{mSampleMutex}; if (mSampleConsumer == nullptr) { mSampleQueue.enqueue(sample); } else { mSampleConsumer(sample); } } void MediaTrackTranscoder::setSampleConsumer( const MediaSampleWriter::MediaSampleConsumerFunction& sampleConsumer) { std::scoped_lock lock{mSampleMutex}; mSampleConsumer = sampleConsumer; std::shared_ptr sample; while (!mSampleQueue.isEmpty() && !mSampleQueue.dequeue(&sample)) { mSampleConsumer(sample); } } } // namespace android