483 lines
15 KiB
C++
483 lines
15 KiB
C++
/* 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/. */
|
|
|
|
#ifndef MEDIAENGINE_H_
|
|
#define MEDIAENGINE_H_
|
|
|
|
#include "mozilla/RefPtr.h"
|
|
#include "DOMMediaStream.h"
|
|
#include "MediaStreamGraph.h"
|
|
#include "MediaTrackConstraints.h"
|
|
#include "mozilla/dom/MediaStreamTrackBinding.h"
|
|
#include "mozilla/dom/VideoStreamTrack.h"
|
|
#include "mozilla/media/DeviceChangeCallback.h"
|
|
|
|
namespace mozilla {
|
|
|
|
namespace dom {
|
|
class Blob;
|
|
} // namespace dom
|
|
|
|
enum {
|
|
kVideoTrack = 1,
|
|
kAudioTrack = 2,
|
|
kTrackCount
|
|
};
|
|
|
|
/**
|
|
* Abstract interface for managing audio and video devices. Each platform
|
|
* must implement a concrete class that will map these classes and methods
|
|
* to the appropriate backend. For example, on Desktop platforms, these will
|
|
* correspond to equivalent webrtc (GIPS) calls, and on B2G they will map to
|
|
* a Gonk interface.
|
|
*/
|
|
class MediaEngineVideoSource;
|
|
class MediaEngineAudioSource;
|
|
|
|
enum MediaEngineState {
|
|
kAllocated,
|
|
kStarted,
|
|
kStopped,
|
|
kReleased
|
|
};
|
|
|
|
class MediaEngine : public DeviceChangeCallback
|
|
{
|
|
public:
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(MediaEngine)
|
|
|
|
static const int DEFAULT_VIDEO_FPS = 30;
|
|
static const int DEFAULT_VIDEO_MIN_FPS = 10;
|
|
static const int DEFAULT_43_VIDEO_WIDTH = 640;
|
|
static const int DEFAULT_43_VIDEO_HEIGHT = 480;
|
|
static const int DEFAULT_169_VIDEO_WIDTH = 1280;
|
|
static const int DEFAULT_169_VIDEO_HEIGHT = 720;
|
|
|
|
static const int DEFAULT_SAMPLE_RATE = 32000;
|
|
|
|
// This allows using whatever rate the graph is using for the
|
|
// MediaStreamTrack. This is useful for microphone data, we know it's already
|
|
// at the correct rate for insertion in the MSG.
|
|
static const int USE_GRAPH_RATE = -1;
|
|
|
|
/* Populate an array of video sources in the nsTArray. Also include devices
|
|
* that are currently unavailable. */
|
|
virtual void EnumerateVideoDevices(dom::MediaSourceEnum,
|
|
nsTArray<RefPtr<MediaEngineVideoSource> >*) = 0;
|
|
|
|
/* Populate an array of audio sources in the nsTArray. Also include devices
|
|
* that are currently unavailable. */
|
|
virtual void EnumerateAudioDevices(dom::MediaSourceEnum,
|
|
nsTArray<RefPtr<MediaEngineAudioSource> >*) = 0;
|
|
|
|
virtual void Shutdown() = 0;
|
|
|
|
virtual void SetFakeDeviceChangeEvents() {}
|
|
|
|
protected:
|
|
virtual ~MediaEngine() {}
|
|
};
|
|
|
|
/**
|
|
* Video source and friends.
|
|
*/
|
|
class MediaEnginePrefs {
|
|
public:
|
|
MediaEnginePrefs()
|
|
: mWidth(0)
|
|
, mHeight(0)
|
|
, mFPS(0)
|
|
, mMinFPS(0)
|
|
, mFreq(0)
|
|
, mAecOn(false)
|
|
, mAgcOn(false)
|
|
, mNoiseOn(false)
|
|
, mAec(0)
|
|
, mAgc(0)
|
|
, mNoise(0)
|
|
, mPlayoutDelay(0)
|
|
, mFullDuplex(false)
|
|
, mExtendedFilter(false)
|
|
, mDelayAgnostic(false)
|
|
, mFakeDeviceChangeEventOn(false)
|
|
{}
|
|
|
|
int32_t mWidth;
|
|
int32_t mHeight;
|
|
int32_t mFPS;
|
|
int32_t mMinFPS;
|
|
int32_t mFreq; // for test tones (fake:true)
|
|
bool mAecOn;
|
|
bool mAgcOn;
|
|
bool mNoiseOn;
|
|
int32_t mAec;
|
|
int32_t mAgc;
|
|
int32_t mNoise;
|
|
int32_t mPlayoutDelay;
|
|
bool mFullDuplex;
|
|
bool mExtendedFilter;
|
|
bool mDelayAgnostic;
|
|
bool mFakeDeviceChangeEventOn;
|
|
|
|
// mWidth and/or mHeight may be zero (=adaptive default), so use functions.
|
|
|
|
int32_t GetWidth(bool aHD = false) const {
|
|
return mWidth? mWidth : (mHeight?
|
|
(mHeight * GetDefWidth(aHD)) / GetDefHeight(aHD) :
|
|
GetDefWidth(aHD));
|
|
}
|
|
|
|
int32_t GetHeight(bool aHD = false) const {
|
|
return mHeight? mHeight : (mWidth?
|
|
(mWidth * GetDefHeight(aHD)) / GetDefWidth(aHD) :
|
|
GetDefHeight(aHD));
|
|
}
|
|
private:
|
|
static int32_t GetDefWidth(bool aHD = false) {
|
|
// It'd be nice if we could use the ternary operator here, but we can't
|
|
// because of bug 1002729.
|
|
if (aHD) {
|
|
return MediaEngine::DEFAULT_169_VIDEO_WIDTH;
|
|
}
|
|
|
|
return MediaEngine::DEFAULT_43_VIDEO_WIDTH;
|
|
}
|
|
|
|
static int32_t GetDefHeight(bool aHD = false) {
|
|
// It'd be nice if we could use the ternary operator here, but we can't
|
|
// because of bug 1002729.
|
|
if (aHD) {
|
|
return MediaEngine::DEFAULT_169_VIDEO_HEIGHT;
|
|
}
|
|
|
|
return MediaEngine::DEFAULT_43_VIDEO_HEIGHT;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Callback interface for TakePhoto(). Either PhotoComplete() or PhotoError()
|
|
* should be called.
|
|
*/
|
|
class MediaEnginePhotoCallback {
|
|
public:
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(MediaEnginePhotoCallback)
|
|
|
|
// aBlob is the image captured by MediaEngineSource. It is
|
|
// called on main thread.
|
|
virtual nsresult PhotoComplete(already_AddRefed<dom::Blob> aBlob) = 0;
|
|
|
|
// It is called on main thread. aRv is the error code.
|
|
virtual nsresult PhotoError(nsresult aRv) = 0;
|
|
|
|
protected:
|
|
virtual ~MediaEnginePhotoCallback() {}
|
|
};
|
|
|
|
/**
|
|
* Common abstract base class for audio and video sources.
|
|
*
|
|
* By default, the base class implements Allocate and Deallocate using its
|
|
* UpdateSingleSource pattern, which manages allocation handles and calculates
|
|
* net constraints from competing allocations and updates a single shared device.
|
|
*
|
|
* Classes that don't operate as a single shared device can override Allocate
|
|
* and Deallocate and simply not pass the methods up.
|
|
*/
|
|
class MediaEngineSource : public nsISupports,
|
|
protected MediaConstraintsHelper
|
|
{
|
|
public:
|
|
// code inside webrtc.org assumes these sizes; don't use anything smaller
|
|
// without verifying it's ok
|
|
static const unsigned int kMaxDeviceNameLength = 128;
|
|
static const unsigned int kMaxUniqueIdLength = 256;
|
|
|
|
virtual ~MediaEngineSource()
|
|
{
|
|
if (!mInShutdown) {
|
|
Shutdown();
|
|
}
|
|
}
|
|
|
|
virtual void Shutdown()
|
|
{
|
|
mInShutdown = true;
|
|
};
|
|
|
|
/* Populate the human readable name of this device in the nsAString */
|
|
virtual void GetName(nsAString&) const = 0;
|
|
|
|
/* Populate the UUID of this device in the nsACString */
|
|
virtual void GetUUID(nsACString&) const = 0;
|
|
|
|
/* Override w/true if source does end-run around cross origin restrictions. */
|
|
virtual bool GetScary() const { return false; };
|
|
|
|
class AllocationHandle
|
|
{
|
|
public:
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(AllocationHandle);
|
|
protected:
|
|
~AllocationHandle() {}
|
|
public:
|
|
AllocationHandle(const dom::MediaTrackConstraints& aConstraints,
|
|
const nsACString& aOrigin,
|
|
const MediaEnginePrefs& aPrefs,
|
|
const nsString& aDeviceId)
|
|
: mConstraints(aConstraints),
|
|
mOrigin(aOrigin),
|
|
mPrefs(aPrefs),
|
|
mDeviceId(aDeviceId) {}
|
|
public:
|
|
NormalizedConstraints mConstraints;
|
|
nsCString mOrigin;
|
|
MediaEnginePrefs mPrefs;
|
|
nsString mDeviceId;
|
|
};
|
|
|
|
/* Release the device back to the system. */
|
|
virtual nsresult Deallocate(AllocationHandle* aHandle)
|
|
{
|
|
MOZ_ASSERT(aHandle);
|
|
RefPtr<AllocationHandle> handle = aHandle;
|
|
|
|
class Comparator {
|
|
public:
|
|
static bool Equals(const RefPtr<AllocationHandle>& a,
|
|
const RefPtr<AllocationHandle>& b) {
|
|
return a.get() == b.get();
|
|
}
|
|
};
|
|
|
|
auto ix = mRegisteredHandles.IndexOf(handle, 0, Comparator());
|
|
if (ix == mRegisteredHandles.NoIndex) {
|
|
MOZ_ASSERT(false);
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
mRegisteredHandles.RemoveElementAt(ix);
|
|
if (mRegisteredHandles.Length() && !mInShutdown) {
|
|
// Whenever constraints are removed, other parties may get closer to ideal.
|
|
auto& first = mRegisteredHandles[0];
|
|
const char* badConstraint = nullptr;
|
|
return ReevaluateAllocation(nullptr, nullptr, first->mPrefs,
|
|
first->mDeviceId, &badConstraint);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
/* Start the device and add the track to the provided SourceMediaStream, with
|
|
* the provided TrackID. You may start appending data to the track
|
|
* immediately after. */
|
|
virtual nsresult Start(SourceMediaStream*, TrackID, const PrincipalHandle&) = 0;
|
|
|
|
/* tell the source if there are any direct listeners attached */
|
|
virtual void SetDirectListeners(bool) = 0;
|
|
|
|
/* Called when the stream wants more data */
|
|
virtual void NotifyPull(MediaStreamGraph* aGraph,
|
|
SourceMediaStream *aSource,
|
|
TrackID aId,
|
|
StreamTime aDesiredTime,
|
|
const PrincipalHandle& aPrincipalHandle) = 0;
|
|
|
|
/* Stop the device and release the corresponding MediaStream */
|
|
virtual nsresult Stop(SourceMediaStream *aSource, TrackID aID) = 0;
|
|
|
|
/* Restart with new capability */
|
|
virtual nsresult Restart(AllocationHandle* aHandle,
|
|
const dom::MediaTrackConstraints& aConstraints,
|
|
const MediaEnginePrefs &aPrefs,
|
|
const nsString& aDeviceId,
|
|
const char** aOutBadConstraint) = 0;
|
|
|
|
/* Returns true if a source represents a fake capture device and
|
|
* false otherwise
|
|
*/
|
|
virtual bool IsFake() = 0;
|
|
|
|
/* Returns the type of media source (camera, microphone, screen, window, etc) */
|
|
virtual dom::MediaSourceEnum GetMediaSource() const = 0;
|
|
|
|
/* If implementation of MediaEngineSource supports TakePhoto(), the picture
|
|
* should be return via aCallback object. Otherwise, it returns NS_ERROR_NOT_IMPLEMENTED.
|
|
* Currently, only Gonk MediaEngineSource implementation supports it.
|
|
*/
|
|
virtual nsresult TakePhoto(MediaEnginePhotoCallback* aCallback) = 0;
|
|
|
|
/* Return false if device is currently allocated or started */
|
|
bool IsAvailable() {
|
|
if (mState == kAllocated || mState == kStarted) {
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
/* It is an error to call Start() before an Allocate(), and Stop() before
|
|
* a Start(). Only Allocate() may be called after a Deallocate(). */
|
|
|
|
/* This call reserves but does not start the device. */
|
|
virtual nsresult Allocate(const dom::MediaTrackConstraints &aConstraints,
|
|
const MediaEnginePrefs &aPrefs,
|
|
const nsString& aDeviceId,
|
|
const nsACString& aOrigin,
|
|
AllocationHandle** aOutHandle,
|
|
const char** aOutBadConstraint)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aOutHandle);
|
|
RefPtr<AllocationHandle> handle = new AllocationHandle(aConstraints, aOrigin,
|
|
aPrefs, aDeviceId);
|
|
nsresult rv = ReevaluateAllocation(handle, nullptr, aPrefs, aDeviceId,
|
|
aOutBadConstraint);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
mRegisteredHandles.AppendElement(handle);
|
|
handle.forget(aOutHandle);
|
|
return NS_OK;
|
|
}
|
|
|
|
virtual uint32_t GetBestFitnessDistance(
|
|
const nsTArray<const NormalizedConstraintSet*>& aConstraintSets,
|
|
const nsString& aDeviceId) const = 0;
|
|
|
|
void GetSettings(dom::MediaTrackSettings& aOutSettings)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
aOutSettings = mSettings;
|
|
}
|
|
|
|
protected:
|
|
// Only class' own members can be initialized in constructor initializer list.
|
|
explicit MediaEngineSource(MediaEngineState aState)
|
|
: mState(aState)
|
|
#ifdef DEBUG
|
|
, mOwningThread(PR_GetCurrentThread())
|
|
#endif
|
|
, mInShutdown(false)
|
|
{}
|
|
|
|
/* UpdateSingleSource - Centralized abstract function to implement in those
|
|
* cases where a single device is being shared between users. Should apply net
|
|
* constraints and restart the device as needed.
|
|
*
|
|
* aHandle - New or existing handle, or null to update after removal.
|
|
* aNetConstraints - Net constraints to be applied to the single device.
|
|
* aPrefs - As passed in (in case of changes in about:config).
|
|
* aDeviceId - As passed in (origin dependent).
|
|
* aOutBadConstraint - Result: nonzero if failed to apply. Name of culprit.
|
|
*/
|
|
|
|
virtual nsresult
|
|
UpdateSingleSource(const AllocationHandle* aHandle,
|
|
const NormalizedConstraints& aNetConstraints,
|
|
const MediaEnginePrefs& aPrefs,
|
|
const nsString& aDeviceId,
|
|
const char** aOutBadConstraint) {
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
};
|
|
|
|
/* ReevaluateAllocation - Call to change constraints for an allocation of
|
|
* a single device. Manages allocation handles, calculates net constraints
|
|
* from all competing allocations, and calls UpdateSingleSource with the net
|
|
* result, to restart the single device as needed.
|
|
*
|
|
* aHandle - New or existing handle, or null to update after removal.
|
|
* aConstraintsUpdate - Constraints to be applied to existing handle, or null.
|
|
* aPrefs - As passed in (in case of changes from about:config).
|
|
* aDeviceId - As passed in (origin-dependent id).
|
|
* aOutBadConstraint - Result: nonzero if failed to apply. Name of culprit.
|
|
*/
|
|
|
|
nsresult
|
|
ReevaluateAllocation(AllocationHandle* aHandle,
|
|
NormalizedConstraints* aConstraintsUpdate,
|
|
const MediaEnginePrefs& aPrefs,
|
|
const nsString& aDeviceId,
|
|
const char** aOutBadConstraint)
|
|
{
|
|
// aHandle and/or aConstraintsUpdate may be nullptr (see below)
|
|
|
|
AutoTArray<const NormalizedConstraints*, 10> allConstraints;
|
|
for (auto& registered : mRegisteredHandles) {
|
|
if (aConstraintsUpdate && registered.get() == aHandle) {
|
|
continue; // Don't count old constraints
|
|
}
|
|
allConstraints.AppendElement(®istered->mConstraints);
|
|
}
|
|
if (aConstraintsUpdate) {
|
|
allConstraints.AppendElement(aConstraintsUpdate);
|
|
} else if (aHandle) {
|
|
// In the case of AddShareOfSingleSource, the handle isn't registered yet.
|
|
allConstraints.AppendElement(&aHandle->mConstraints);
|
|
}
|
|
|
|
NormalizedConstraints netConstraints(allConstraints);
|
|
if (netConstraints.mBadConstraint) {
|
|
*aOutBadConstraint = netConstraints.mBadConstraint;
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsresult rv = UpdateSingleSource(aHandle, netConstraints, aPrefs, aDeviceId,
|
|
aOutBadConstraint);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
if (aHandle && aConstraintsUpdate) {
|
|
aHandle->mConstraints = *aConstraintsUpdate;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void AssertIsOnOwningThread()
|
|
{
|
|
MOZ_ASSERT(PR_GetCurrentThread() == mOwningThread);
|
|
}
|
|
|
|
MediaEngineState mState;
|
|
#ifdef DEBUG
|
|
PRThread* mOwningThread;
|
|
#endif
|
|
nsTArray<RefPtr<AllocationHandle>> mRegisteredHandles;
|
|
bool mInShutdown;
|
|
|
|
// Main-thread only:
|
|
dom::MediaTrackSettings mSettings;
|
|
};
|
|
|
|
class MediaEngineVideoSource : public MediaEngineSource
|
|
{
|
|
public:
|
|
virtual ~MediaEngineVideoSource() {}
|
|
|
|
protected:
|
|
explicit MediaEngineVideoSource(MediaEngineState aState)
|
|
: MediaEngineSource(aState) {}
|
|
MediaEngineVideoSource()
|
|
: MediaEngineSource(kReleased) {}
|
|
};
|
|
|
|
/**
|
|
* Audio source and friends.
|
|
*/
|
|
class MediaEngineAudioSource : public MediaEngineSource,
|
|
public AudioDataListenerInterface
|
|
{
|
|
public:
|
|
virtual ~MediaEngineAudioSource() {}
|
|
|
|
protected:
|
|
explicit MediaEngineAudioSource(MediaEngineState aState)
|
|
: MediaEngineSource(aState) {}
|
|
MediaEngineAudioSource()
|
|
: MediaEngineSource(kReleased) {}
|
|
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif /* MEDIAENGINE_H_ */
|