mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-08 12:22:34 +00:00
343 lines
11 KiB
C++
343 lines
11 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "RtspOmxReader.h"
|
|
|
|
#include "AbstractMediaDecoder.h"
|
|
#include "MediaDecoderStateMachine.h"
|
|
#include "MPAPI.h"
|
|
#include "mozilla/dom/TimeRanges.h"
|
|
#include "mozilla/Mutex.h"
|
|
#include "mozilla/TimeStamp.h"
|
|
#include "OmxDecoder.h"
|
|
#include "RtspMediaResource.h"
|
|
#include "RtspOmxDecoder.h"
|
|
#include "VideoUtils.h"
|
|
|
|
#include <stagefright/MediaExtractor.h>
|
|
#include <stagefright/MediaBufferGroup.h>
|
|
#include <stagefright/MetaData.h>
|
|
|
|
#define FRAME_DEFAULT_SIZE 1024
|
|
|
|
using namespace android;
|
|
|
|
namespace mozilla {
|
|
|
|
/* class RtspMediaSource : implements MediaSource for OMX.
|
|
* The decoder thread will trigger the MediaDecodeStateMachine to read a/v frame.
|
|
* Then RtspOmxReader calls OMX decoder to decode a/v frame. Finally the code
|
|
* path run into the read() here, it reads un-decoded frame data from mResource
|
|
* and construct a MediaBuffer for output to OMX decoder.
|
|
* */
|
|
class RtspMediaSource : public android::MediaSource {
|
|
public:
|
|
RtspMediaSource(RtspMediaResource *aRtspMediaResource,
|
|
ssize_t aTrackIdx,
|
|
uint32_t aFrameMaxSize,
|
|
const sp<MetaData>& aMeta)
|
|
: mRtspResource(aRtspMediaResource)
|
|
, mFormat(aMeta)
|
|
, mTrackIdx(aTrackIdx)
|
|
, mMonitor("RtspMediaSource.mMonitor")
|
|
, mIsStarted(false)
|
|
, mGroup(nullptr)
|
|
, mBuffer(nullptr)
|
|
, mFrameMaxSize(aFrameMaxSize) {
|
|
MOZ_COUNT_CTOR(RtspMediaSource);
|
|
};
|
|
virtual ~RtspMediaSource() {
|
|
MOZ_COUNT_DTOR(RtspMediaSource);
|
|
}
|
|
virtual status_t start(MetaData *params = nullptr) MOZ_FINAL MOZ_OVERRIDE;
|
|
virtual status_t stop() MOZ_FINAL MOZ_OVERRIDE;
|
|
virtual sp<MetaData> getFormat() MOZ_FINAL MOZ_OVERRIDE {
|
|
ReentrantMonitorAutoEnter mon(mMonitor);
|
|
return mFormat;
|
|
};
|
|
virtual status_t read(MediaBuffer **buffer,
|
|
const ReadOptions *options = nullptr) MOZ_FINAL MOZ_OVERRIDE ;
|
|
private:
|
|
nsRefPtr<RtspMediaResource> mRtspResource;
|
|
sp<MetaData> mFormat;
|
|
uint32_t mTrackIdx;
|
|
ReentrantMonitor mMonitor;
|
|
bool mIsStarted;
|
|
|
|
// mGroup owns the mBuffer. mFrameMaxSize is the mBuffer size.
|
|
// mBuffer is the input buffer for omx decoder.
|
|
nsAutoPtr<MediaBufferGroup> mGroup;
|
|
MediaBuffer* mBuffer;
|
|
uint32_t mFrameMaxSize;
|
|
};
|
|
|
|
status_t RtspMediaSource::start(MetaData *params)
|
|
{
|
|
ReentrantMonitorAutoEnter mon(mMonitor);
|
|
if (!mIsStarted) {
|
|
// RtspMediaSource relinquish the ownership of MediaBuffer |buf| to mGroup.
|
|
mGroup = new MediaBufferGroup();
|
|
MediaBuffer* buf = new MediaBuffer(mFrameMaxSize);
|
|
mGroup->add_buffer(buf);
|
|
mIsStarted = true;
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
status_t RtspMediaSource::stop()
|
|
{
|
|
ReentrantMonitorAutoEnter mon(mMonitor);
|
|
if (mIsStarted) {
|
|
if (mBuffer) {
|
|
mBuffer->release();
|
|
mBuffer = nullptr;
|
|
}
|
|
mGroup = nullptr;
|
|
mIsStarted = false;
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
status_t RtspMediaSource::read(MediaBuffer **out, const ReadOptions *options)
|
|
{
|
|
ReentrantMonitorAutoEnter mon(mMonitor);
|
|
NS_ENSURE_TRUE(mIsStarted, MEDIA_ERROR_BASE);
|
|
NS_ENSURE_TRUE(out, MEDIA_ERROR_BASE);
|
|
*out = nullptr;
|
|
|
|
// Video/audio track's initial frame size is FRAME_DEFAULT_SIZE.
|
|
// We need to realloc the mBuffer if the mBuffer doesn't have enough space
|
|
// for next ReadFrameFromTrack function. (actualFrameSize > mFrameMaxSize)
|
|
status_t err;
|
|
uint32_t readCount;
|
|
uint32_t actualFrameSize;
|
|
uint64_t time;
|
|
nsresult rv;
|
|
|
|
while (1) {
|
|
err = mGroup->acquire_buffer(&mBuffer);
|
|
NS_ENSURE_TRUE(err == OK, err);
|
|
|
|
rv = mRtspResource->ReadFrameFromTrack((uint8_t *)mBuffer->data(),
|
|
mFrameMaxSize, mTrackIdx, readCount,
|
|
time, actualFrameSize);
|
|
if (NS_FAILED(rv)) {
|
|
// Release mGroup and mBuffer.
|
|
stop();
|
|
// Since RtspMediaSource is an implementation of Android media source,
|
|
// it's held by OMXCodec and isn't released yet. So we have to re-construct
|
|
// mGroup and mBuffer.
|
|
start();
|
|
NS_WARNING("ReadFrameFromTrack failed; releasing buffers and returning.");
|
|
return ERROR_CONNECTION_LOST;
|
|
}
|
|
if (actualFrameSize > mFrameMaxSize) {
|
|
// release mGroup and mBuffer
|
|
stop();
|
|
// re-construct mGroup and mBuffer
|
|
mFrameMaxSize = actualFrameSize;
|
|
err = start();
|
|
NS_ENSURE_TRUE(err == OK, err);
|
|
} else {
|
|
// ReadFrameFromTrack success, break the while loop.
|
|
break;
|
|
}
|
|
}
|
|
mBuffer->set_range(0, readCount);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
mBuffer->meta_data()->clear();
|
|
// fill the meta data
|
|
mBuffer->meta_data()->setInt64(kKeyTime, time);
|
|
*out = mBuffer;
|
|
mBuffer = nullptr;
|
|
return OK;
|
|
}
|
|
|
|
return ERROR_END_OF_STREAM;
|
|
}
|
|
|
|
|
|
// RtspExtractor is a custom extractor for Rtsp stream, whereas the other
|
|
// XXXExtractors are made for container media content.
|
|
// The extractor is used for |OmxDecoder::Init|, it provides the essential
|
|
// information for creating OMXCodec instance.
|
|
// For example, the |getTrackMetaData| returns metadata that includes the
|
|
// codec type.
|
|
class RtspExtractor: public MediaExtractor
|
|
{
|
|
public:
|
|
virtual size_t countTracks() MOZ_FINAL MOZ_OVERRIDE;
|
|
virtual sp<android::MediaSource> getTrack(size_t index)
|
|
MOZ_FINAL MOZ_OVERRIDE;
|
|
virtual sp<MetaData> getTrackMetaData(
|
|
size_t index, uint32_t flag = 0) MOZ_FINAL MOZ_OVERRIDE;
|
|
virtual uint32_t flags() const MOZ_FINAL MOZ_OVERRIDE;
|
|
|
|
RtspExtractor(RtspMediaResource *aResource)
|
|
: mRtspResource(aResource) {
|
|
MOZ_COUNT_CTOR(RtspExtractor);
|
|
MOZ_ASSERT(aResource);
|
|
mController = mRtspResource->GetMediaStreamController();
|
|
MOZ_ASSERT(mController);
|
|
}
|
|
virtual ~RtspExtractor() MOZ_OVERRIDE {
|
|
MOZ_COUNT_DTOR(RtspExtractor);
|
|
}
|
|
private:
|
|
// mRtspResource is a pointer to RtspMediaResource. When |getTrack| is called
|
|
// we use mRtspResource to construct a RtspMediaSource.
|
|
RtspMediaResource* mRtspResource;
|
|
// Through the mController in mRtspResource, we can get the essential
|
|
// information for the extractor.
|
|
nsRefPtr<nsIStreamingProtocolController> mController;
|
|
};
|
|
|
|
size_t RtspExtractor::countTracks()
|
|
{
|
|
uint8_t tracks = 0;
|
|
if (mController) {
|
|
mController->GetTotalTracks(&tracks);
|
|
}
|
|
return size_t(tracks);
|
|
}
|
|
|
|
sp<android::MediaSource> RtspExtractor::getTrack(size_t index)
|
|
{
|
|
NS_ENSURE_TRUE(index < countTracks(), nullptr);
|
|
|
|
sp<MetaData> meta = getTrackMetaData(index);
|
|
sp<android::MediaSource> source = new RtspMediaSource(mRtspResource,
|
|
index,
|
|
FRAME_DEFAULT_SIZE,
|
|
meta);
|
|
return source;
|
|
}
|
|
|
|
sp<MetaData> RtspExtractor::getTrackMetaData(size_t index, uint32_t flag)
|
|
{
|
|
NS_ENSURE_TRUE(index < countTracks(), nullptr);;
|
|
|
|
sp<MetaData> meta = new MetaData();
|
|
nsCOMPtr<nsIStreamingProtocolMetaData> rtspMetadata;
|
|
mController->GetTrackMetaData(index, getter_AddRefs(rtspMetadata));
|
|
|
|
if (rtspMetadata) {
|
|
// Convert msMeta into meta.
|
|
// The getter function of nsIStreamingProtocolMetaData will initialize the
|
|
// metadata values to 0 before setting them.
|
|
nsCString mime;
|
|
rtspMetadata->GetMimeType(mime);
|
|
meta->setCString(kKeyMIMEType, mime.get());
|
|
uint32_t temp32;
|
|
rtspMetadata->GetWidth(&temp32);
|
|
meta->setInt32(kKeyWidth, temp32);
|
|
rtspMetadata->GetHeight(&temp32);
|
|
meta->setInt32(kKeyHeight, temp32);
|
|
rtspMetadata->GetSampleRate(&temp32);
|
|
meta->setInt32(kKeySampleRate, temp32);
|
|
rtspMetadata->GetChannelCount(&temp32);
|
|
meta->setInt32(kKeyChannelCount, temp32);
|
|
uint64_t temp64;
|
|
rtspMetadata->GetDuration(&temp64);
|
|
meta->setInt64(kKeyDuration, temp64);
|
|
|
|
nsCString tempCString;
|
|
rtspMetadata->GetEsdsData(tempCString);
|
|
if (tempCString.Length()) {
|
|
meta->setData(kKeyESDS, 0, tempCString.get(), tempCString.Length());
|
|
}
|
|
rtspMetadata->GetAvccData(tempCString);
|
|
if (tempCString.Length()) {
|
|
meta->setData(kKeyAVCC, 0, tempCString.get(), tempCString.Length());
|
|
}
|
|
}
|
|
return meta;
|
|
}
|
|
|
|
uint32_t RtspExtractor::flags() const
|
|
{
|
|
if (mRtspResource->IsRealTime()) {
|
|
return 0;
|
|
} else {
|
|
return MediaExtractor::CAN_SEEK;
|
|
}
|
|
}
|
|
|
|
nsresult RtspOmxReader::InitOmxDecoder()
|
|
{
|
|
if (!mOmxDecoder.get()) {
|
|
NS_ASSERTION(mDecoder, "RtspOmxReader mDecoder is null.");
|
|
NS_ASSERTION(mDecoder->GetResource(),
|
|
"RtspOmxReader mDecoder->GetResource() is null.");
|
|
mExtractor = new RtspExtractor(mRtspResource);
|
|
mOmxDecoder = new OmxDecoder(mDecoder->GetResource(), mDecoder);
|
|
if (!mOmxDecoder->Init(mExtractor)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult RtspOmxReader::Seek(int64_t aTime, int64_t aStartTime,
|
|
int64_t aEndTime, int64_t aCurrentTime)
|
|
{
|
|
// The seek function of Rtsp is time-based, we call the SeekTime function in
|
|
// RtspMediaResource. The SeekTime function finally send a seek command to
|
|
// Rtsp stream server through network and also clear the buffer data in
|
|
// RtspMediaResource.
|
|
if (mRtspResource) {
|
|
mRtspResource->SeekTime(aTime);
|
|
}
|
|
|
|
// Call |MediaOmxReader::Seek| to notify the OMX decoder we are performing a
|
|
// seek operation. The function will clear the |mVideoQueue| and |mAudioQueue|
|
|
// that store the decoded data and also call the |DecodeToTarget| to pass
|
|
// the seek time to OMX a/v decoders.
|
|
return MediaOmxReader::Seek(aTime, aStartTime, aEndTime, aCurrentTime);
|
|
}
|
|
|
|
void RtspOmxReader::OnDecodeThreadStart() {
|
|
// Start RTSP streaming right after starting the decoding thread in
|
|
// MediaDecoderStateMachine and before starting OMXCodec decoding.
|
|
if (mRtspResource) {
|
|
nsIStreamingProtocolController* controller =
|
|
mRtspResource->GetMediaStreamController();
|
|
if (controller) {
|
|
controller->Play();
|
|
}
|
|
}
|
|
|
|
// Call parent class to start OMXCodec decoding.
|
|
MediaOmxReader::OnDecodeThreadStart();
|
|
}
|
|
|
|
void RtspOmxReader::OnDecodeThreadFinish() {
|
|
// Call parent class to pause OMXCodec decoding.
|
|
MediaOmxReader::OnDecodeThreadFinish();
|
|
|
|
// Stop RTSP streaming right before destroying the decoding thread in
|
|
// MediaDecoderStateMachine and after pausing OMXCodec decoding.
|
|
// RTSP streaming should not be paused until OMXCodec has been paused and
|
|
// until the decoding thread in MediaDecoderStateMachine is about to be
|
|
// destroyed. Otherwise, RtspMediaSource::read() would block the binder
|
|
// thread of OMXCodecObserver::onMessage() --> OMXCodec::on_message() -->
|
|
// OMXCodec::drainInputBuffer() due to network data starvation. Because
|
|
// OMXCodec::mLock is held by the binder thread in this case, all other
|
|
// threads would be blocked when they try to lock this mutex. As a result, the
|
|
// decoding thread in MediaDecoderStateMachine would be blocked forever in
|
|
// OMXCodec::read() if there is no enough data for RtspMediaSource::read() to
|
|
// return.
|
|
if (mRtspResource) {
|
|
nsIStreamingProtocolController* controller =
|
|
mRtspResource->GetMediaStreamController();
|
|
if (controller) {
|
|
controller->Pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace mozilla
|