blob: b70639c426ac6fbd4bd66003dd652733fff6e77e [file] [log] [blame]
/*
* Copyright (C) 2014 Igalia S.L
* Copyright (C) 2016-2025 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. AND ITS CONTRIBUTORS ``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 ITS 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)
#include <WebCore/CachedRawResourceClient.h>
#include <WebCore/CachedResourceHandle.h>
#include <WebCore/ContextDestructionObserver.h>
#include <WebCore/FetchOptions.h>
#include <WebCore/PlatformMediaResourceLoader.h>
#include <WebCore/ResourceResponse.h>
#include <wtf/Atomics.h>
#include <wtf/HashSet.h>
#include <wtf/Ref.h>
#include <wtf/TZoneMalloc.h>
#include <wtf/WeakHashSet.h>
#include <wtf/text/WTFString.h>
namespace WebCore {
class CachedRawResource;
class Document;
class Element;
class MediaResource;
class WeakPtrImplWithEventTargetData;
enum class LoadedFromOpaqueSource : bool;
class MediaResourceLoader final
: public ThreadSafeRefCountedAndCanMakeThreadSafeWeakPtr<MediaResourceLoader, WTF::DestructionThread::Main>
, public PlatformMediaResourceLoader
, public ContextDestructionObserver {
WTF_MAKE_TZONE_ALLOCATED_EXPORT(MediaResourceLoader, WEBCORE_EXPORT);
public:
static Ref<MediaResourceLoader> create(Document& document, Element& element, const String& crossOriginMode, FetchOptions::Destination destination) { return adoptRef(*new MediaResourceLoader(document, element, crossOriginMode, destination)); }
WEBCORE_EXPORT virtual ~MediaResourceLoader();
// ContextDestructionObserver.
void ref() const final { ThreadSafeRefCountedAndCanMakeThreadSafeWeakPtr::ref(); }
void deref() const final { ThreadSafeRefCountedAndCanMakeThreadSafeWeakPtr::deref(); }
ThreadSafeWeakPtrControlBlock& controlBlock() const final { return ThreadSafeRefCountedAndCanMakeThreadSafeWeakPtr::controlBlock(); }
uint32_t weakRefCount() const final { return ThreadSafeRefCountedAndCanMakeThreadSafeWeakPtr::weakRefCount(); }
RefPtr<PlatformMediaResource> requestResource(ResourceRequest&&, LoadOptions) final;
void sendH2Ping(const URL&, CompletionHandler<void(Expected<Seconds, ResourceError>&&)>&&) final;
void removeResource(MediaResource&);
Document* document();
RefPtr<Document> protectedDocument();
const String& crossOriginMode() const;
WEBCORE_EXPORT static void recordResponsesForTesting();
WEBCORE_EXPORT Vector<ResourceResponse> responsesForTesting() const;
void addResponseForTesting(const ResourceResponse&);
bool verifyMediaResponse(const URL& requestURL, const ResourceResponse&, const SecurityOrigin*);
void redirectReceived(const URL&);
private:
WEBCORE_EXPORT MediaResourceLoader(Document&, Element&, const String& crossOriginMode, FetchOptions::Destination);
void contextDestroyed() final;
WeakPtr<Document, WeakPtrImplWithEventTargetData> m_document WTF_GUARDED_BY_CAPABILITY(mainThread);
WeakPtr<Element, WeakPtrImplWithEventTargetData> m_element WTF_GUARDED_BY_CAPABILITY(mainThread);
String m_crossOriginMode WTF_GUARDED_BY_CAPABILITY(mainThread);
SingleThreadWeakHashSet<MediaResource> m_resources WTF_GUARDED_BY_CAPABILITY(mainThread);
Vector<ResourceResponse> m_responsesForTesting WTF_GUARDED_BY_CAPABILITY(mainThread);
FetchOptions::Destination m_destination WTF_GUARDED_BY_CAPABILITY(mainThread);
struct ValidationInformation {
RefPtr<const SecurityOrigin> origin;
bool usedOpaqueResponse { false };
bool usedServiceWorker { false };
};
HashMap<URL, ValidationInformation> m_validationLoadInformations WTF_GUARDED_BY_CAPABILITY(mainThread);
HashSet<URL> m_nonOpaqueLoadURLs;
std::optional<LoadedFromOpaqueSource> m_loadedFromOpaqueSource;
};
class MediaResource : public PlatformMediaResource, public CachedRawResourceClient {
WTF_MAKE_TZONE_ALLOCATED(MediaResource);
public:
static Ref<MediaResource> create(MediaResourceLoader&, CachedResourceHandle<CachedRawResource>&&);
virtual ~MediaResource();
// PlatformMediaResource
void shutdown() override;
bool didPassAccessControlCheck() const override { return m_didPassAccessControlCheck.load(); }
// CachedRawResourceClient
void ref() const final { PlatformMediaResource::ref(); }
void deref() const final { PlatformMediaResource::deref(); }
void responseReceived(const CachedResource&, const ResourceResponse&, CompletionHandler<void()>&&) override;
void redirectReceived(CachedResource&, ResourceRequest&&, const ResourceResponse&, CompletionHandler<void(ResourceRequest&&)>&&) override;
bool shouldCacheResponse(CachedResource&, const ResourceResponse&) override;
void dataSent(CachedResource&, unsigned long long, unsigned long long) override;
void dataReceived(CachedResource&, const SharedBuffer&) override;
void notifyFinished(CachedResource&, const NetworkLoadMetrics&, LoadWillContinueInAnotherProcess) override;
private:
CachedResourceHandle<CachedRawResource> protectedResource() const;
MediaResource(MediaResourceLoader&, CachedResourceHandle<CachedRawResource>&&);
void ensureShutdown();
const Ref<MediaResourceLoader> m_loader;
Atomic<bool> m_didPassAccessControlCheck { false };
CachedResourceHandle<CachedRawResource> m_resource;
};
} // namespace WebCore
#endif // ENABLE(VIDEO)