2017-09-08 20:30:18 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2006 Lubos Lunak <l.lunak@kde.org>
|
|
|
|
Copyright (C) 2009, 2010, 2011 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*********************************************************************/
|
|
|
|
#ifndef KWIN_SCENE_OPENGL_BACKEND_H
|
|
|
|
#define KWIN_SCENE_OPENGL_BACKEND_H
|
|
|
|
|
|
|
|
#include <QElapsedTimer>
|
|
|
|
#include <QRegion>
|
|
|
|
|
|
|
|
#include <kwin_export.h>
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
class OpenGLBackend;
|
|
|
|
class OverlayWindow;
|
|
|
|
class SceneOpenGL;
|
|
|
|
class SceneOpenGLTexture;
|
|
|
|
class SceneOpenGLTexturePrivate;
|
|
|
|
class WindowPixmap;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief The OpenGLBackend creates and holds the OpenGL context and is responsible for Texture from Pixmap.
|
|
|
|
*
|
|
|
|
* The OpenGLBackend is an abstract base class used by the SceneOpenGL to abstract away the differences
|
|
|
|
* between various OpenGL windowing systems such as GLX and EGL.
|
|
|
|
*
|
|
|
|
* A concrete implementation has to create and release the OpenGL context in a way so that the
|
|
|
|
* SceneOpenGL does not have to care about it.
|
|
|
|
*
|
|
|
|
* In addition a major task for this class is to generate the SceneOpenGLTexturePrivate which is
|
|
|
|
* able to perform the texture from pixmap operation in the given backend.
|
|
|
|
*
|
|
|
|
* @author Martin Gräßlin <mgraesslin@kde.org>
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
class KWIN_EXPORT OpenGLBackend
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
OpenGLBackend();
|
|
|
|
virtual ~OpenGLBackend();
|
|
|
|
|
|
|
|
virtual void init() = 0;
|
|
|
|
/**
|
|
|
|
* @return Time passes since start of rendering current frame.
|
|
|
|
* @see startRenderTimer
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
qint64 renderTime() {
|
|
|
|
return m_renderTimer.nsecsElapsed();
|
|
|
|
}
|
|
|
|
virtual void screenGeometryChanged(const QSize &size) = 0;
|
|
|
|
virtual SceneOpenGLTexturePrivate *createBackendTexture(SceneOpenGLTexture *texture) = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Backend specific code to prepare the rendering of a frame including flushing the
|
|
|
|
* previously rendered frame to the screen if the backend works this way.
|
|
|
|
*
|
|
|
|
* @return A region that if not empty will be repainted in addition to the damaged region
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
virtual QRegion prepareRenderingFrame() = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Backend specific code to handle the end of rendering a frame.
|
|
|
|
*
|
|
|
|
* @param renderedRegion The possibly larger region that has been rendered
|
|
|
|
* @param damagedRegion The damaged region that should be posted
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
virtual void endRenderingFrame(const QRegion &damage, const QRegion &damagedRegion) = 0;
|
|
|
|
virtual void endRenderingFrameForScreen(int screenId, const QRegion &damage, const QRegion &damagedRegion);
|
|
|
|
virtual bool makeCurrent() = 0;
|
|
|
|
virtual void doneCurrent() = 0;
|
|
|
|
virtual bool usesOverlayWindow() const = 0;
|
|
|
|
/**
|
|
|
|
* Whether the rendering needs to be split per screen.
|
|
|
|
* Default implementation returns @c false.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
virtual bool perScreenRendering() const;
|
|
|
|
virtual QRegion prepareRenderingForScreen(int screenId);
|
|
|
|
/**
|
|
|
|
* @brief Compositor is going into idle mode, flushes any pending paints.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void idle();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return bool Whether the scene needs to flush a frame.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool hasPendingFlush() const {
|
|
|
|
return !m_lastDamage.isEmpty();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Returns the OverlayWindow used by the backend.
|
|
|
|
*
|
|
|
|
* A backend does not have to use an OverlayWindow, this is mostly for the X world.
|
|
|
|
* In case the backend does not use an OverlayWindow it is allowed to return @c null.
|
|
|
|
* It's the task of the caller to check whether it is @c null.
|
|
|
|
*
|
|
|
|
* @return :OverlayWindow*
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
virtual OverlayWindow *overlayWindow();
|
|
|
|
/**
|
|
|
|
* @brief Whether the creation of the Backend failed.
|
|
|
|
*
|
|
|
|
* The SceneOpenGL should test whether the Backend got constructed correctly. If this method
|
|
|
|
* returns @c true, the SceneOpenGL should not try to start the rendering.
|
|
|
|
*
|
|
|
|
* @return bool @c true if the creation of the Backend failed, @c false otherwise.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool isFailed() const {
|
|
|
|
return m_failed;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Whether the Backend provides VSync.
|
|
|
|
*
|
|
|
|
* Currently only the GLX backend can provide VSync.
|
|
|
|
*
|
|
|
|
* @return bool @c true if VSync support is available, @c false otherwise
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool syncsToVBlank() const {
|
|
|
|
return m_syncsToVBlank;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Whether VSync blocks execution until the screen is in the retrace
|
|
|
|
*
|
|
|
|
* Case for waitVideoSync and non triple buffering buffer swaps
|
|
|
|
*
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool blocksForRetrace() const {
|
|
|
|
return m_blocksForRetrace;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Whether the backend uses direct rendering.
|
|
|
|
*
|
|
|
|
* Some OpenGLScene modes require direct rendering. E.g. the OpenGL 2 should not be used
|
|
|
|
* if direct rendering is not supported by the Scene.
|
|
|
|
*
|
|
|
|
* @return bool @c true if the GL context is direct, @c false if indirect
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool isDirectRendering() const {
|
|
|
|
return m_directRendering;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool supportsBufferAge() const {
|
|
|
|
return m_haveBufferAge;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns whether the context is surfaceless
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool isSurfaceLessContext() const {
|
|
|
|
return m_surfaceLessContext;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the damage that has accumulated since a buffer of the given age was presented.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
QRegion accumulatedDamageHistory(int bufferAge) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Saves the given region to damage history.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void addToDamageHistory(const QRegion ®ion);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The backend specific extensions (e.g. EGL/GLX extensions).
|
|
|
|
*
|
|
|
|
* Not the OpenGL (ES) extension!
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
QList<QByteArray> extensions() const {
|
|
|
|
return m_extensions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns whether the backend specific extensions contains @p extension.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool hasExtension(const QByteArray &extension) const {
|
|
|
|
return m_extensions.contains(extension);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy a region of pixels from the current read to the current draw buffer
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void copyPixels(const QRegion ®ion);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
/**
|
|
|
|
* @brief Backend specific flushing of frame to screen.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
virtual void present() = 0;
|
|
|
|
/**
|
|
|
|
* @brief Sets the backend initialization to failed.
|
|
|
|
*
|
|
|
|
* This method should be called by the concrete subclass in case the initialization failed.
|
|
|
|
* The given @p reason is logged as a warning.
|
|
|
|
*
|
|
|
|
* @param reason The reason why the initialization failed.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setFailed(const QString &reason);
|
|
|
|
/**
|
|
|
|
* @brief Sets whether the backend provides VSync.
|
|
|
|
*
|
|
|
|
* Should be called by the concrete subclass once it is determined whether VSync is supported.
|
|
|
|
* If the subclass does not call this method, the backend defaults to @c false.
|
|
|
|
* @param enabled @c true if VSync support available, @c false otherwise.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setSyncsToVBlank(bool enabled) {
|
|
|
|
m_syncsToVBlank = enabled;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Sets whether the VSync iplementation blocks
|
|
|
|
*
|
|
|
|
* Should be called by the concrete subclass once it is determined how VSync works.
|
|
|
|
* If the subclass does not call this method, the backend defaults to @c false.
|
|
|
|
* @param enabled @c true if VSync blocks, @c false otherwise.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setBlocksForRetrace(bool enabled) {
|
|
|
|
m_blocksForRetrace = enabled;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Sets whether the OpenGL context is direct.
|
|
|
|
*
|
|
|
|
* Should be called by the concrete subclass once it is determined whether the OpenGL context is
|
|
|
|
* direct or indirect.
|
|
|
|
* If the subclass does not call this method, the backend defaults to @c false.
|
|
|
|
*
|
|
|
|
* @param direct @c true if the OpenGL context is direct, @c false if indirect
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setIsDirectRendering(bool direct) {
|
|
|
|
m_directRendering = direct;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSupportsBufferAge(bool value) {
|
|
|
|
m_haveBufferAge = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return const QRegion& Damage of previously rendered frame
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
const QRegion &lastDamage() const {
|
|
|
|
return m_lastDamage;
|
|
|
|
}
|
|
|
|
void setLastDamage(const QRegion &damage) {
|
|
|
|
m_lastDamage = damage;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @brief Starts the timer for how long it takes to render the frame.
|
|
|
|
*
|
|
|
|
* @see renderTime
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void startRenderTimer() {
|
|
|
|
m_renderTimer.start();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param set whether the context is surface less
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setSurfaceLessContext(bool set) {
|
|
|
|
m_surfaceLessContext = set;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the platform-specific @p extensions.
|
|
|
|
*
|
|
|
|
* These are the EGL/GLX extensions, not the OpenGL extensions
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
void setExtensions(const QList<QByteArray> &extensions) {
|
|
|
|
m_extensions = extensions;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
/**
|
|
|
|
* @brief Whether VSync is available and used, defaults to @c false.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool m_syncsToVBlank;
|
|
|
|
/**
|
|
|
|
* @brief Whether present() will block execution until the next vertical retrace @c false.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool m_blocksForRetrace;
|
|
|
|
/**
|
|
|
|
* @brief Whether direct rendering is used, defaults to @c false.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool m_directRendering;
|
|
|
|
/**
|
|
|
|
* @brief Whether the backend supports GLX_EXT_buffer_age / EGL_EXT_buffer_age.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool m_haveBufferAge;
|
|
|
|
/**
|
|
|
|
* @brief Whether the initialization failed, of course default to @c false.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
bool m_failed;
|
|
|
|
/**
|
|
|
|
* @brief Damaged region of previously rendered frame.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
QRegion m_lastDamage;
|
|
|
|
/**
|
|
|
|
* @brief The damage history for the past 10 frames.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
QList<QRegion> m_damageHistory;
|
|
|
|
/**
|
|
|
|
* @brief Timer to measure how long a frame renders.
|
2019-07-29 18:58:33 +00:00
|
|
|
*/
|
2017-09-08 20:30:18 +00:00
|
|
|
QElapsedTimer m_renderTimer;
|
|
|
|
bool m_surfaceLessContext = false;
|
|
|
|
|
|
|
|
QList<QByteArray> m_extensions;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|