blob: 7d310720aa3d940f9902d2a772e9e8d86a63c9c5 [file] [log] [blame]
/*
* Copyright (C) 2017 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
#include "CSSParserContext.h"
#include "CanvasNoiseInjection.h"
#include "FloatRect.h"
#include "IntSize.h"
#include "PixelBuffer.h"
#include "TaskSource.h"
#include <atomic>
#include <wtf/AbstractRefCountedAndCanMakeWeakPtr.h>
#include <wtf/HashSet.h>
#include <wtf/TypeCasts.h>
#include <wtf/WeakHashSet.h>
namespace WebCore {
class CanvasDisplayBufferObserver;
}
namespace WTF {
template<typename T> struct IsDeprecatedWeakRefSmartPointerException;
template<> struct IsDeprecatedWeakRefSmartPointerException<WebCore::CanvasDisplayBufferObserver> : std::true_type { };
}
namespace WebCore {
class AffineTransform;
class CanvasBase;
class CanvasObserver;
class CanvasRenderingContext;
class Element;
class Event;
class GraphicsContext;
class GraphicsContextStateSaver;
class Image;
class ImageBuffer;
class IntRect;
class ScriptExecutionContext;
class SecurityOrigin;
class WebCoreOpaqueRoot;
enum class ShouldApplyPostProcessingToDirtyRect : bool { No, Yes };
class CanvasDisplayBufferObserver : public CanMakeWeakPtr<CanvasDisplayBufferObserver> {
public:
virtual ~CanvasDisplayBufferObserver() = default;
virtual void canvasDisplayBufferPrepared(CanvasBase&) = 0;
};
class CanvasBase : public AbstractRefCountedAndCanMakeWeakPtr<CanvasBase> {
public:
virtual ~CanvasBase();
virtual bool isHTMLCanvasElement() const { return false; }
virtual bool isOffscreenCanvas() const { return false; }
virtual bool isCustomPaintCanvas() const { return false; }
virtual unsigned width() const { return m_size.width(); }
virtual unsigned height() const { return m_size.height(); }
const IntSize& size() const { return m_size; }
ImageBuffer* buffer() const;
virtual void setImageBufferAndMarkDirty(RefPtr<ImageBuffer>&&) { }
RefPtr<ImageBuffer> makeRenderingResultsAvailable(ShouldApplyPostProcessingToDirtyRect = ShouldApplyPostProcessingToDirtyRect::Yes);
size_t memoryCost() const;
#if ENABLE(RESOURCE_USAGE)
size_t externalMemoryCost() const;
#endif
void setOriginClean() { m_originClean = true; }
void setOriginTainted() { m_originClean = false; }
bool originClean() const { return m_originClean; }
virtual SecurityOrigin* securityOrigin() const { return nullptr; }
ScriptExecutionContext* scriptExecutionContext() const { return canvasBaseScriptExecutionContext(); }
virtual CanvasRenderingContext* renderingContext() const = 0;
const CSSParserContext& cssParserContext() const;
void addObserver(CanvasObserver&);
void removeObserver(CanvasObserver&);
bool hasObserver(CanvasObserver&) const;
void notifyObserversCanvasChanged(const FloatRect&);
void notifyObserversCanvasResized();
void notifyObserversCanvasDestroyed(); // Must be called in destruction before clearing m_context.
void addDisplayBufferObserver(CanvasDisplayBufferObserver&);
void removeDisplayBufferObserver(CanvasDisplayBufferObserver&);
void notifyObserversCanvasDisplayBufferPrepared();
bool hasDisplayBufferObservers() const { return !m_displayBufferObservers.isEmptyIgnoringNullReferences(); }
HashSet<Element*> cssCanvasClients() const;
// !rect means caller knows the full canvas is invalidated previously.
void didDraw(const std::optional<FloatRect>& rect) { return didDraw(rect, ShouldApplyPostProcessingToDirtyRect::Yes); }
virtual void didDraw(const std::optional<FloatRect>&, ShouldApplyPostProcessingToDirtyRect);
virtual Image* copiedImage() const = 0;
virtual void clearCopiedImage() const = 0;
bool hasActiveInspectorCanvasCallTracer() const;
bool shouldAccelerate(const IntSize&) const;
WEBCORE_EXPORT static void setMaxCanvasAreaForTesting(std::optional<size_t>);
virtual void queueTaskKeepingObjectAlive(TaskSource, Function<void(CanvasBase&)>&&) = 0;
virtual void dispatchEvent(Event&) = 0;
bool postProcessPixelBufferResults(Ref<PixelBuffer>&&) const;
void recordLastFillText(const String&);
void resetGraphicsContextState() const;
void setNoiseInjectionSalt(NoiseInjectionHashSalt salt) { m_canvasNoiseHashSalt = salt; }
bool havePendingCanvasNoiseInjection() const { return m_canvasNoiseInjection.haveDirtyRects(); }
// FIXME(https://bugs.webkit.org/show_bug.cgi?id=275100): The image buffer from CanvasBase should be moved to CanvasRenderingContext2DBase.
RefPtr<ImageBuffer> allocateImageBuffer() const;
void setHasCreatedImageBuffer(bool hasCreatedImageBuffer) { m_hasCreatedImageBuffer = hasCreatedImageBuffer; }
bool hasCreatedImageBuffer() const { return m_hasCreatedImageBuffer; }
RefPtr<ImageBuffer> createImageForNoiseInjection() const;
protected:
explicit CanvasBase(IntSize, ScriptExecutionContext&);
virtual ScriptExecutionContext* canvasBaseScriptExecutionContext() const = 0;
virtual std::unique_ptr<CSSParserContext> createCSSParserContext() const = 0;
virtual void setSize(const IntSize&);
RefPtr<ImageBuffer> setImageBuffer(RefPtr<ImageBuffer>&&) const;
String lastFillText() const { return m_lastFillText; }
void addCanvasNeedingPreparationForDisplayOrFlush();
void removeCanvasNeedingPreparationForDisplayOrFlush();
private:
bool shouldInjectNoiseBeforeReadback() const;
virtual void createImageBuffer() const { }
bool shouldAccelerate(uint64_t area) const;
mutable IntSize m_size;
mutable RefPtr<ImageBuffer> m_imageBuffer;
mutable std::atomic<size_t> m_imageBufferMemoryCost { 0 };
mutable std::unique_ptr<GraphicsContextStateSaver> m_contextStateSaver;
mutable std::unique_ptr<CSSParserContext> m_cssParserContext;
String m_lastFillText;
WeakHashSet<CanvasObserver> m_observers;
WeakHashSet<CanvasDisplayBufferObserver> m_displayBufferObservers;
CanvasNoiseInjection m_canvasNoiseInjection;
Markable<NoiseInjectionHashSalt, IntegralMarkableTraits<NoiseInjectionHashSalt, std::numeric_limits<int64_t>::max()>> m_canvasNoiseHashSalt;
bool m_originClean { true };
// m_hasCreatedImageBuffer means we tried to malloc the buffer. We didn't necessarily get it.
bool m_hasCreatedImageBuffer { false };
#if ASSERT_ENABLED
bool m_didNotifyObserversCanvasDestroyed { false };
#endif
};
WebCoreOpaqueRoot root(CanvasBase*);
inline const CSSParserContext& CanvasBase::cssParserContext() const
{
if (!m_cssParserContext) [[unlikely]]
m_cssParserContext = createCSSParserContext();
return *m_cssParserContext;
}
} // namespace WebCore
#define SPECIALIZE_TYPE_TRAITS_CANVAS(ToValueTypeName, predicate) \
SPECIALIZE_TYPE_TRAITS_BEGIN(ToValueTypeName) \
static bool isType(const WebCore::CanvasBase& canvas) { return canvas.predicate; } \
SPECIALIZE_TYPE_TRAITS_END()