blob: be19b6007c9c0fbfffb859e40fd34751493fe7d1 [file] [log] [blame]
/*
* Copyright (C) 2016 Igalia S.L.
*
* 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 COMPUTER, 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 COMPUTER, 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.
*/
#include "config.h"
#include "AcceleratedBackingStoreX11.h"
#if PLATFORM(X11)
#include "DrawingAreaProxyCoordinatedGraphics.h"
#include "LayerTreeContext.h"
#include "WebPageProxy.h"
#include <WebCore/CairoUtilities.h>
#include <WebCore/PlatformDisplayX11.h>
#include <WebCore/XErrorTrapper.h>
#include <X11/Xlib.h>
#include <X11/extensions/Xdamage.h>
#include <cairo-xlib.h>
#if USE(GTK4)
#include <gdk/x11/gdkx.h>
#else
#include <gdk/gdkx.h>
#endif
#include <gtk/gtk.h>
#include <wtf/HashMap.h>
#include <wtf/NeverDestroyed.h>
namespace WebKit {
using namespace WebCore;
static Optional<int> s_damageEventBase;
static Optional<int> s_damageErrorBase;
class XDamageNotifier {
WTF_MAKE_NONCOPYABLE(XDamageNotifier);
friend NeverDestroyed<XDamageNotifier>;
public:
static XDamageNotifier& singleton()
{
static NeverDestroyed<XDamageNotifier> notifier;
return notifier;
}
void add(Damage damage, WTF::Function<void()>&& notifyFunction)
{
if (m_notifyFunctions.isEmpty()) {
#if USE(GTK4)
g_signal_connect(gdk_display_get_default(), "xevent", G_CALLBACK(filterXDamageEvent), this);
#else
gdk_window_add_filter(nullptr, reinterpret_cast<GdkFilterFunc>(&filterXDamageEvent), this);
#endif
}
m_notifyFunctions.add(damage, WTFMove(notifyFunction));
}
void remove(Damage damage)
{
m_notifyFunctions.remove(damage);
if (m_notifyFunctions.isEmpty()) {
#if USE(GTK4)
g_signal_handlers_disconnect_by_data(gdk_display_get_default(), this);
#else
gdk_window_remove_filter(nullptr, reinterpret_cast<GdkFilterFunc>(&filterXDamageEvent), this);
#endif
}
}
private:
XDamageNotifier() = default;
#if USE(GTK4)
static gboolean filterXDamageEvent(GdkDisplay*, XEvent* xEvent, XDamageNotifier* notifier)
{
if (xEvent->type != s_damageEventBase.value() + XDamageNotify)
return GDK_EVENT_PROPAGATE;
auto* damageEvent = reinterpret_cast<XDamageNotifyEvent*>(xEvent);
if (notifier->notify(damageEvent->damage)) {
XDamageSubtract(xEvent->xany.display, damageEvent->damage, None, None);
return GDK_EVENT_STOP;
}
return GDK_EVENT_PROPAGATE;
}
#else
static GdkFilterReturn filterXDamageEvent(GdkXEvent* event, GdkEvent*, XDamageNotifier* notifier)
{
auto* xEvent = static_cast<XEvent*>(event);
if (xEvent->type != s_damageEventBase.value() + XDamageNotify)
return GDK_FILTER_CONTINUE;
auto* damageEvent = reinterpret_cast<XDamageNotifyEvent*>(xEvent);
if (notifier->notify(damageEvent->damage)) {
XDamageSubtract(xEvent->xany.display, damageEvent->damage, None, None);
return GDK_FILTER_REMOVE;
}
return GDK_FILTER_CONTINUE;
}
#endif
bool notify(Damage damage) const
{
auto it = m_notifyFunctions.find(damage);
if (it != m_notifyFunctions.end()) {
((*it).value)();
return true;
}
return false;
}
HashMap<Damage, WTF::Function<void()>> m_notifyFunctions;
};
bool AcceleratedBackingStoreX11::checkRequirements()
{
auto& display = downcast<PlatformDisplayX11>(PlatformDisplay::sharedDisplay());
return display.supportsXComposite() && display.supportsXDamage(s_damageEventBase, s_damageErrorBase);
}
std::unique_ptr<AcceleratedBackingStoreX11> AcceleratedBackingStoreX11::create(WebPageProxy& webPage)
{
ASSERT(checkRequirements());
return std::unique_ptr<AcceleratedBackingStoreX11>(new AcceleratedBackingStoreX11(webPage));
}
AcceleratedBackingStoreX11::AcceleratedBackingStoreX11(WebPageProxy& webPage)
: AcceleratedBackingStore(webPage)
{
}
static inline unsigned char xDamageErrorCode(unsigned char errorCode)
{
ASSERT(s_damageErrorBase);
return static_cast<unsigned>(s_damageErrorBase.value()) + errorCode;
}
AcceleratedBackingStoreX11::~AcceleratedBackingStoreX11()
{
if (!m_surface && !m_damage)
return;
Display* display = downcast<PlatformDisplayX11>(PlatformDisplay::sharedDisplay()).native();
XErrorTrapper trapper(display, XErrorTrapper::Policy::Crash, { BadDrawable, xDamageErrorCode(BadDamage) });
if (m_damage) {
XDamageNotifier::singleton().remove(m_damage.get());
m_damage.reset();
XSync(display, False);
}
}
void AcceleratedBackingStoreX11::update(const LayerTreeContext& layerTreeContext)
{
Pixmap pixmap = layerTreeContext.contextID;
if (m_surface && cairo_xlib_surface_get_drawable(m_surface.get()) == pixmap)
return;
Display* display = downcast<PlatformDisplayX11>(PlatformDisplay::sharedDisplay()).native();
if (m_surface) {
XErrorTrapper trapper(display, XErrorTrapper::Policy::Crash, { BadDrawable, xDamageErrorCode(BadDamage) });
if (m_damage) {
XDamageNotifier::singleton().remove(m_damage.get());
m_damage.reset();
XSync(display, False);
}
m_surface = nullptr;
}
if (!pixmap)
return;
auto* drawingArea = static_cast<DrawingAreaProxyCoordinatedGraphics*>(m_webPage.drawingArea());
if (!drawingArea)
return;
IntSize size = drawingArea->size();
float deviceScaleFactor = m_webPage.deviceScaleFactor();
size.scale(deviceScaleFactor);
XErrorTrapper trapper(display, XErrorTrapper::Policy::Crash, { BadDrawable, xDamageErrorCode(BadDamage) });
ASSERT(downcast<PlatformDisplayX11>(PlatformDisplay::sharedDisplay()).native() == gdk_x11_display_get_xdisplay(gdk_display_get_default()));
#if USE(GTK4)
auto* visual = WK_XVISUAL(downcast<PlatformDisplayX11>(PlatformDisplay::sharedDisplay()));
#else
GdkVisual* gdkVisual = gdk_screen_get_rgba_visual(gdk_screen_get_default());
if (!gdkVisual)
gdkVisual = gdk_screen_get_system_visual(gdk_screen_get_default());
auto* visual = GDK_VISUAL_XVISUAL(gdkVisual);
#endif
m_surface = adoptRef(cairo_xlib_surface_create(display, pixmap, visual, size.width(), size.height()));
cairoSurfaceSetDeviceScale(m_surface.get(), deviceScaleFactor, deviceScaleFactor);
m_damage = XDamageCreate(display, pixmap, XDamageReportNonEmpty);
XDamageNotifier::singleton().add(m_damage.get(), [this] {
if (m_webPage.isViewVisible())
gtk_widget_queue_draw(m_webPage.viewWidget());
});
XSync(display, False);
}
#if USE(GTK4)
void AcceleratedBackingStoreX11::snapshot(GtkSnapshot* gtkSnapshot)
{
if (!m_surface)
return;
// The surface can be modified by the web process at any time, so we mark it
// as dirty to ensure we always render the updated contents as soon as possible.
cairo_surface_mark_dirty(m_surface.get());
FloatSize viewSize(gtk_widget_get_width(m_webPage.viewWidget()), gtk_widget_get_height(m_webPage.viewWidget()));
graphene_rect_t rect = GRAPHENE_RECT_INIT(0, 0, viewSize.width(), viewSize.height());
RefPtr<cairo_t> cr = adoptRef(gtk_snapshot_append_cairo(gtkSnapshot, &rect));
cairo_set_source_surface(cr.get(), m_surface.get(), 0, 0);
cairo_set_operator(cr.get(), CAIRO_OPERATOR_OVER);
cairo_paint(cr.get());
cairo_surface_flush(m_surface.get());
}
#else
bool AcceleratedBackingStoreX11::paint(cairo_t* cr, const IntRect& clipRect)
{
if (!m_surface)
return false;
cairo_save(cr);
// The surface can be modified by the web process at any time, so we mark it
// as dirty to ensure we always render the updated contents as soon as possible.
cairo_surface_mark_dirty(m_surface.get());
cairo_rectangle(cr, clipRect.x(), clipRect.y(), clipRect.width(), clipRect.height());
cairo_set_source_surface(cr, m_surface.get(), 0, 0);
cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
cairo_fill(cr);
cairo_restore(cr);
cairo_surface_flush(m_surface.get());
return true;
}
#endif
} // namespace WebKit
#endif // PLATFORM(X11)