blob: e98d8ea90004718cfc9044a09836af0b90e0fe01 [file] [log] [blame]
/*
* Copyright (C) 2011-2018 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#pragma once
#if ENABLE(VIDEO) && USE(AVFOUNDATION)
#include "MediaPlaybackTarget.h"
#include "MediaPlayerPrivateAVFoundation.h"
#include <wtf/Function.h>
#include <wtf/HashMap.h>
#include <wtf/threads/BinarySemaphore.h>
OBJC_CLASS AVAssetImageGenerator;
OBJC_CLASS AVAssetResourceLoadingRequest;
OBJC_CLASS AVMediaSelectionGroup;
OBJC_CLASS AVOutputContext;
OBJC_CLASS AVPlayerItem;
OBJC_CLASS AVPlayerItemLegibleOutput;
OBJC_CLASS AVPlayerItemMetadataCollector;
OBJC_CLASS AVPlayerItemMetadataOutput;
OBJC_CLASS AVPlayerItemVideoOutput;
OBJC_CLASS AVPlayerLayer;
OBJC_CLASS AVURLAsset;
OBJC_CLASS NSArray;
OBJC_CLASS WebCoreAVFLoaderDelegate;
OBJC_CLASS WebCoreAVFMovieObserver;
OBJC_CLASS WebCoreAVFPullDelegate;
typedef struct objc_object* id;
typedef struct CGImage *CGImageRef;
typedef struct __CVBuffer *CVPixelBufferRef;
namespace WebCore {
class AudioSourceProviderAVFObjC;
class AudioTrackPrivateAVFObjC;
class CDMInstanceFairPlayStreamingAVFObjC;
class CDMSessionAVFoundationObjC;
class ImageRotationSessionVT;
class InbandMetadataTextTrackPrivateAVF;
class MediaSelectionGroupAVFObjC;
class PixelBufferConformerCV;
class SharedBuffer;
class VideoFullscreenLayerManagerObjC;
class VideoTextureCopierCV;
class VideoTrackPrivateAVFObjC;
class WebCoreAVFResourceLoader;
class MediaPlayerPrivateAVFoundationObjC : public MediaPlayerPrivateAVFoundation {
public:
explicit MediaPlayerPrivateAVFoundationObjC(MediaPlayer*);
virtual ~MediaPlayerPrivateAVFoundationObjC();
static void registerMediaEngine(MediaEngineRegistrar);
static HashSet<RefPtr<SecurityOrigin>> originsInMediaCache(const String&);
static void clearMediaCache(const String&, WallTime modifiedSince);
static void clearMediaCacheForOrigins(const String&, const HashSet<RefPtr<SecurityOrigin>>&);
void setAsset(RetainPtr<id>&&);
void tracksChanged() override;
void didEnd() override;
#if HAVE(AVFOUNDATION_MEDIA_SELECTION_GROUP)
RetainPtr<AVPlayerItem> playerItem() const { return m_avPlayerItem; }
void processCue(NSArray *, NSArray *, const MediaTime&);
void flushCues();
#endif
AVPlayer *avPlayer() const { return m_avPlayer.get(); }
#if HAVE(AVFOUNDATION_LOADER_DELEGATE)
bool shouldWaitForLoadingOfResource(AVAssetResourceLoadingRequest*);
void didCancelLoadingRequest(AVAssetResourceLoadingRequest*);
void didStopLoadingRequest(AVAssetResourceLoadingRequest *);
#endif
#if ENABLE(LEGACY_ENCRYPTED_MEDIA)
RetainPtr<AVAssetResourceLoadingRequest> takeRequestForKeyURI(const String&);
void keyAdded() override;
#endif
void playerItemStatusDidChange(int);
void playbackLikelyToKeepUpWillChange();
void playbackLikelyToKeepUpDidChange(bool);
void playbackBufferEmptyWillChange();
void playbackBufferEmptyDidChange(bool);
void playbackBufferFullWillChange();
void playbackBufferFullDidChange(bool);
void loadedTimeRangesDidChange(RetainPtr<NSArray>&&);
void seekableTimeRangesDidChange(RetainPtr<NSArray>&&);
void tracksDidChange(const RetainPtr<NSArray>&);
void hasEnabledAudioDidChange(bool);
void presentationSizeDidChange(FloatSize);
void durationDidChange(const MediaTime&);
void rateDidChange(double);
void timeControlStatusDidChange(int);
void metadataGroupDidArrive(const RetainPtr<NSArray>&, const MediaTime&);
void metadataDidArrive(const RetainPtr<NSArray>&, const MediaTime&);
void firstFrameAvailableDidChange(bool);
void trackEnabledDidChange(bool);
void canPlayFastReverseDidChange(bool);
void canPlayFastForwardDidChange(bool);
void setBufferingPolicy(MediaPlayerEnums::BufferingPolicy) override;
#if HAVE(AVFOUNDATION_VIDEO_OUTPUT)
void outputMediaDataWillChange(AVPlayerItemVideoOutput*);
#endif
#if ENABLE(WIRELESS_PLAYBACK_TARGET)
void playbackTargetIsWirelessDidChange();
#endif
#if ENABLE(LEGACY_ENCRYPTED_MEDIA) || (ENABLE(ENCRYPTED_MEDIA) && HAVE(AVCONTENTKEYSESSION))
void outputObscuredDueToInsufficientExternalProtectionChanged(bool);
void beginSimulatedHDCPError() override { outputObscuredDueToInsufficientExternalProtectionChanged(true); }
void endSimulatedHDCPError() override { outputObscuredDueToInsufficientExternalProtectionChanged(false); }
#endif
#if ENABLE(AVF_CAPTIONS)
void notifyTrackModeChanged() override;
void synchronizeTextTrackState() override;
#endif
#if ENABLE(LEGACY_ENCRYPTED_MEDIA)
void removeSession(LegacyCDMSession&);
#endif
#if ENABLE(ENCRYPTED_MEDIA)
void cdmInstanceAttached(CDMInstance&) final;
void cdmInstanceDetached(CDMInstance&) final;
void attemptToDecryptWithInstance(CDMInstance&) final;
void setWaitingForKey(bool);
bool waitingForKey() const final { return m_waitingForKey; }
#endif
MediaTime currentMediaTime() const override;
private:
// engine support
static void getSupportedTypes(HashSet<String, ASCIICaseInsensitiveHash>& types);
static MediaPlayer::SupportsType supportsType(const MediaEngineSupportParameters&);
static bool supportsKeySystem(const String& keySystem, const String& mimeType);
static bool isAvailable();
void cancelLoad() override;
void platformSetVisible(bool) override;
void platformPlay() override;
void platformPause() override;
bool platformPaused() const override;
void setVolume(float) override;
void setMuted(bool) override;
void setClosedCaptionsVisible(bool) override;
void paint(GraphicsContext&, const FloatRect&) override;
void paintCurrentFrameInContext(GraphicsContext&, const FloatRect&) override;
PlatformLayer* platformLayer() const override;
void setVideoFullscreenLayer(PlatformLayer*, Function<void()>&& completionHandler) override;
void updateVideoFullscreenInlineImage() final;
void setVideoFullscreenFrame(FloatRect) override;
void setVideoFullscreenGravity(MediaPlayer::VideoGravity) override;
void setVideoFullscreenMode(MediaPlayer::VideoFullscreenMode) override;
void videoFullscreenStandbyChanged() override;
void setPlayerRate(double);
#if PLATFORM(IOS_FAMILY)
NSArray *timedMetadata() const override;
String accessLog() const override;
String errorLog() const override;
#endif
bool supportsAcceleratedRendering() const override { return true; }
MediaTime mediaTimeForTimeValue(const MediaTime&) const override;
double maximumDurationToCacheMediaTime() const override;
void createAVPlayer() override;
void createAVPlayerItem() override;
virtual void createAVPlayerLayer();
void createAVAssetForURL(const URL&) override;
MediaPlayerPrivateAVFoundation::ItemStatus playerItemStatus() const override;
MediaPlayerPrivateAVFoundation::AssetStatus assetStatus() const override;
long assetErrorCode() const override;
double seekableTimeRangesLastModifiedTime() const override;
double liveUpdateInterval() const override;
void checkPlayability() override;
void setRateDouble(double) override;
double rate() const override;
void setPreservesPitch(bool) override;
void seekToTime(const MediaTime&, const MediaTime& negativeTolerance, const MediaTime& positiveTolerance) override;
unsigned long long totalBytes() const override;
std::unique_ptr<PlatformTimeRanges> platformBufferedTimeRanges() const override;
MediaTime platformMinTimeSeekable() const override;
MediaTime platformMaxTimeSeekable() const override;
MediaTime platformDuration() const override;
MediaTime platformMaxTimeLoaded() const override;
void beginLoadingMetadata() override;
void sizeChanged() override;
void resolvedURLChanged() override;
bool hasAvailableVideoFrame() const override;
void createContextVideoRenderer() override;
void destroyContextVideoRenderer() override;
void createVideoLayer() override;
void destroyVideoLayer() override;
bool hasContextRenderer() const override;
bool hasLayerRenderer() const override;
void updateVideoLayerGravity() override;
bool didPassCORSAccessCheck() const override;
Optional<bool> wouldTaintOrigin(const SecurityOrigin&) const final;
MediaTime getStartDate() const override;
#if ENABLE(VIDEO_TRACK)
bool requiresTextTrackRepresentation() const override;
void setTextTrackRepresentation(TextTrackRepresentation*) override;
void syncTextTrackBounds() override;
#endif
void setAVPlayerItem(AVPlayerItem *);
#if ENABLE(WEB_AUDIO) && USE(MEDIATOOLBOX)
AudioSourceProvider* audioSourceProvider() override;
#endif
void createImageGenerator();
void destroyImageGenerator();
RetainPtr<CGImageRef> createImageForTimeInRect(float, const FloatRect&);
void paintWithImageGenerator(GraphicsContext&, const FloatRect&);
#if HAVE(AVFOUNDATION_VIDEO_OUTPUT)
enum class UpdateType { UpdateSynchronously, UpdateAsynchronously };
void updateLastImage(UpdateType type = UpdateType::UpdateAsynchronously);
void createVideoOutput();
void destroyVideoOutput();
bool updateLastPixelBuffer();
bool videoOutputHasAvailableFrame();
void paintWithVideoOutput(GraphicsContext&, const FloatRect&);
NativeImagePtr nativeImageForCurrentTime() override;
void waitForVideoOutputMediaDataWillChange();
bool copyVideoTextureToPlatformTexture(GraphicsContext3D*, Platform3DObject, GC3Denum target, GC3Dint level, GC3Denum internalFormat, GC3Denum format, GC3Denum type, bool premultiplyAlpha, bool flipY) override;
#endif
#if ENABLE(LEGACY_ENCRYPTED_MEDIA)
std::unique_ptr<LegacyCDMSession> createSession(const String& keySystem, LegacyCDMSessionClient*) override;
#endif
String languageOfPrimaryAudioTrack() const override;
#if HAVE(AVFOUNDATION_MEDIA_SELECTION_GROUP)
void processMediaSelectionOptions();
bool hasLoadedMediaSelectionGroups();
AVMediaSelectionGroup* safeMediaSelectionGroupForLegibleMedia();
AVMediaSelectionGroup* safeMediaSelectionGroupForAudibleMedia();
AVMediaSelectionGroup* safeMediaSelectionGroupForVisualMedia();
#endif
NSArray* safeAVAssetTracksForAudibleMedia();
#if ENABLE(DATACUE_VALUE)
void processMetadataTrack();
#endif
void setCurrentTextTrack(InbandTextTrackPrivateAVF*) override;
InbandTextTrackPrivateAVF* currentTextTrack() const override { return m_currentTextTrack; }
#if !HAVE(AVFOUNDATION_LEGIBLE_OUTPUT_SUPPORT)
void processLegacyClosedCaptionsTracks();
#endif
#if ENABLE(VIDEO_TRACK)
void updateAudioTracks();
void updateVideoTracks();
#endif
#if ENABLE(WIRELESS_PLAYBACK_TARGET)
bool isCurrentPlaybackTargetWireless() const override;
String wirelessPlaybackTargetName() const override;
MediaPlayer::WirelessPlaybackTargetType wirelessPlaybackTargetType() const override;
bool wirelessVideoPlaybackDisabled() const override;
void setWirelessVideoPlaybackDisabled(bool) override;
bool canPlayToWirelessPlaybackTarget() const override { return true; }
void updateDisableExternalPlayback();
#endif
#if ENABLE(WIRELESS_PLAYBACK_TARGET) && !PLATFORM(IOS_FAMILY)
void setWirelessPlaybackTarget(Ref<MediaPlaybackTarget>&&) override;
void setShouldPlayToPlaybackTarget(bool) override;
#endif
double maxFastForwardRate() const override { return m_cachedCanPlayFastForward ? std::numeric_limits<double>::infinity() : 2.0; }
double minFastReverseRate() const override { return m_cachedCanPlayFastReverse ? -std::numeric_limits<double>::infinity() : 0.0; }
Vector<String> preferredAudioCharacteristics() const;
void setShouldDisableSleep(bool) override;
void updateRotationSession();
Optional<VideoPlaybackQualityMetrics> videoPlaybackQualityMetrics() final;
#if !RELEASE_LOG_DISABLED
const char* logClassName() const final { return "MediaPlayerPrivateAVFoundationObjC"; }
#endif
AVPlayer *objCAVFoundationAVPlayer() const final { return m_avPlayer.get(); }
bool performTaskAtMediaTime(WTF::Function<void()>&&, MediaTime) final;
void setShouldObserveTimeControlStatus(bool);
RetainPtr<AVURLAsset> m_avAsset;
RetainPtr<AVPlayer> m_avPlayer;
RetainPtr<AVPlayerItem> m_avPlayerItem;
RetainPtr<AVPlayerLayer> m_videoLayer;
std::unique_ptr<VideoFullscreenLayerManagerObjC> m_videoFullscreenLayerManager;
MediaPlayer::VideoGravity m_videoFullscreenGravity;
RetainPtr<WebCoreAVFMovieObserver> m_objcObserver;
RetainPtr<id> m_timeObserver;
mutable String m_languageOfPrimaryAudioTrack;
bool m_videoFrameHasDrawn;
bool m_haveCheckedPlayability;
#if ENABLE(WEB_AUDIO) && USE(MEDIATOOLBOX)
RefPtr<AudioSourceProviderAVFObjC> m_provider;
#endif
RetainPtr<AVAssetImageGenerator> m_imageGenerator;
#if HAVE(AVFOUNDATION_VIDEO_OUTPUT)
RetainPtr<AVPlayerItemVideoOutput> m_videoOutput;
RetainPtr<WebCoreAVFPullDelegate> m_videoOutputDelegate;
RetainPtr<CVPixelBufferRef> m_lastPixelBuffer;
RetainPtr<CGImageRef> m_lastImage;
BinarySemaphore m_videoOutputSemaphore;
std::unique_ptr<ImageRotationSessionVT> m_imageRotationSession;
std::unique_ptr<VideoTextureCopierCV> m_videoTextureCopier;
#endif
#if HAVE(CORE_VIDEO)
std::unique_ptr<PixelBufferConformerCV> m_pixelBufferConformer;
#endif
#if HAVE(AVFOUNDATION_LOADER_DELEGATE)
friend class WebCoreAVFResourceLoader;
HashMap<RetainPtr<CFTypeRef>, RefPtr<WebCoreAVFResourceLoader>> m_resourceLoaderMap;
RetainPtr<WebCoreAVFLoaderDelegate> m_loaderDelegate;
HashMap<String, RetainPtr<AVAssetResourceLoadingRequest>> m_keyURIToRequestMap;
HashMap<String, RetainPtr<AVAssetResourceLoadingRequest>> m_sessionIDToRequestMap;
#endif
#if HAVE(AVFOUNDATION_MEDIA_SELECTION_GROUP) && HAVE(AVFOUNDATION_LEGIBLE_OUTPUT_SUPPORT)
RetainPtr<AVPlayerItemLegibleOutput> m_legibleOutput;
#endif
#if ENABLE(VIDEO_TRACK)
Vector<RefPtr<AudioTrackPrivateAVFObjC>> m_audioTracks;
Vector<RefPtr<VideoTrackPrivateAVFObjC>> m_videoTracks;
#if HAVE(AVFOUNDATION_MEDIA_SELECTION_GROUP)
RefPtr<MediaSelectionGroupAVFObjC> m_audibleGroup;
RefPtr<MediaSelectionGroupAVFObjC> m_visualGroup;
#endif
#endif
InbandTextTrackPrivateAVF* m_currentTextTrack;
#if ENABLE(DATACUE_VALUE)
RefPtr<InbandMetadataTextTrackPrivateAVF> m_metadataTrack;
#endif
#if PLATFORM(MAC) && ENABLE(WIRELESS_PLAYBACK_TARGET)
RetainPtr<AVOutputContext> m_outputContext;
RefPtr<MediaPlaybackTarget> m_playbackTarget { nullptr };
#endif
#if ENABLE(LEGACY_ENCRYPTED_MEDIA)
WeakPtr<CDMSessionAVFoundationObjC> m_session;
#endif
#if ENABLE(ENCRYPTED_MEDIA)
bool m_waitingForKey { false };
#if HAVE(AVCONTENTKEYSESSION)
RefPtr<CDMInstanceFairPlayStreamingAVFObjC> m_cdmInstance;
#endif
#endif
RetainPtr<AVPlayerItemMetadataCollector> m_metadataCollector;
RetainPtr<AVPlayerItemMetadataOutput> m_metadataOutput;
mutable RetainPtr<NSArray> m_cachedSeekableRanges;
mutable RetainPtr<NSArray> m_cachedLoadedRanges;
RetainPtr<NSArray> m_cachedTracks;
RetainPtr<NSArray> m_currentMetaData;
FloatSize m_cachedPresentationSize;
MediaTime m_cachedDuration;
RefPtr<SharedBuffer> m_keyID;
double m_cachedRate;
bool m_requestedPlaying { false };
double m_requestedRate { 1.0 };
int m_cachedTimeControlStatus { 0 };
mutable long long m_cachedTotalBytes;
unsigned m_pendingStatusChanges;
int m_cachedItemStatus;
MediaPlayer::BufferingPolicy m_bufferingPolicy { MediaPlayer::BufferingPolicy::Default };
bool m_cachedLikelyToKeepUp;
bool m_cachedBufferEmpty;
bool m_cachedBufferFull;
bool m_cachedHasEnabledAudio;
bool m_cachedIsReadyForDisplay;
bool m_haveBeenAskedToCreateLayer;
bool m_cachedCanPlayFastForward;
bool m_cachedCanPlayFastReverse;
bool m_muted { false };
bool m_shouldObserveTimeControlStatus { false };
mutable Optional<bool> m_tracksArePlayable;
#if ENABLE(WIRELESS_PLAYBACK_TARGET)
mutable bool m_allowsWirelessVideoPlayback;
bool m_shouldPlayToPlaybackTarget { false };
#endif
};
}
#endif