c2943c63a5
Code section had been in an #ifndef NDEBUG which is the reason why I did not find the usage of Extensions there and why it always compiled successfully. Some data elements which got dropped needed to be added again like a name for the extension and errorBase, etc. Sorry for the inconvenience of a semi-broken master.
252 lines
8.1 KiB
C++
252 lines
8.1 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2006 Lubos Lunak <l.lunak@kde.org>
|
|
Copyright (C) 2012 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/>.
|
|
*********************************************************************/
|
|
#include "xcbutils.h"
|
|
#include "utils.h"
|
|
// KDE
|
|
#include <KDE/KDebug>
|
|
// xcb
|
|
#include <xcb/composite.h>
|
|
#include <xcb/damage.h>
|
|
#include <xcb/randr.h>
|
|
#include <xcb/render.h>
|
|
#include <xcb/shape.h>
|
|
#include <xcb/sync.h>
|
|
#include <xcb/xfixes.h>
|
|
|
|
namespace KWin {
|
|
|
|
namespace Xcb {
|
|
|
|
static const int COMPOSITE_MAX_MAJOR = 0;
|
|
static const int COMPOSITE_MAX_MINOR = 4;
|
|
static const int DAMAGE_MAX_MAJOR = 1;
|
|
static const int DAMAGE_MIN_MAJOR = 1;
|
|
static const int SYNC_MAX_MAJOR = 3;
|
|
static const int SYNC_MAX_MINOR = 0;
|
|
static const int RANDR_MAX_MAJOR = 1;
|
|
static const int RANDR_MAX_MINOR = 4;
|
|
static const int RENDER_MAX_MAJOR = 0;
|
|
static const int RENDER_MAX_MINOR = 11;
|
|
static const int XFIXES_MAX_MAJOR = 5;
|
|
static const int XFIXES_MAX_MINOR = 0;
|
|
|
|
ExtensionData::ExtensionData()
|
|
: version(0)
|
|
, eventBase(0)
|
|
, errorBase(0)
|
|
, majorOpcode(0)
|
|
, present(0)
|
|
{
|
|
}
|
|
|
|
template<typename reply, typename T, typename F>
|
|
void Extensions::initVersion(T cookie, F f, ExtensionData *dataToFill)
|
|
{
|
|
ScopedCPointer<reply> version = f(connection(), cookie, NULL);
|
|
dataToFill->version = version->major_version * 0x10 + version->minor_version;
|
|
}
|
|
|
|
Extensions *Extensions::s_self = NULL;
|
|
|
|
Extensions *Extensions::self()
|
|
{
|
|
if (!s_self) {
|
|
s_self = new Extensions();
|
|
}
|
|
return s_self;
|
|
}
|
|
|
|
void Extensions::destroy()
|
|
{
|
|
delete s_self;
|
|
s_self = NULL;
|
|
}
|
|
|
|
Extensions::Extensions()
|
|
{
|
|
init();
|
|
}
|
|
|
|
Extensions::~Extensions()
|
|
{
|
|
}
|
|
|
|
void Extensions::init()
|
|
{
|
|
xcb_connection_t *c = connection();
|
|
xcb_prefetch_extension_data(c, &xcb_shape_id);
|
|
xcb_prefetch_extension_data(c, &xcb_randr_id);
|
|
xcb_prefetch_extension_data(c, &xcb_damage_id);
|
|
xcb_prefetch_extension_data(c, &xcb_composite_id);
|
|
xcb_prefetch_extension_data(c, &xcb_xfixes_id);
|
|
xcb_prefetch_extension_data(c, &xcb_render_id);
|
|
xcb_prefetch_extension_data(c, &xcb_sync_id);
|
|
|
|
m_shape.name = QByteArray("SHAPE");
|
|
m_randr.name = QByteArray("RANDR");
|
|
m_damage.name = QByteArray("DAMAGE");
|
|
m_composite.name = QByteArray("Composite");
|
|
m_fixes.name = QByteArray("XFIXES");
|
|
m_render.name = QByteArray("RENDER");
|
|
m_sync.name = QByteArray("SYNC");
|
|
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_shape_id), &m_shape);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_randr_id), &m_randr);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_damage_id), &m_damage);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_composite_id), &m_composite);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_xfixes_id), &m_fixes);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_render_id), &m_render);
|
|
extensionQueryReply(xcb_get_extension_data(c, &xcb_sync_id), &m_sync);
|
|
|
|
// extension specific queries
|
|
xcb_shape_query_version_cookie_t shapeVersion;
|
|
xcb_randr_query_version_cookie_t randrVersion;
|
|
xcb_damage_query_version_cookie_t damageVersion;
|
|
xcb_composite_query_version_cookie_t compositeVersion;
|
|
xcb_xfixes_query_version_cookie_t xfixesVersion;
|
|
xcb_render_query_version_cookie_t renderVersion;
|
|
xcb_sync_initialize_cookie_t syncVersion;
|
|
if (m_shape.present) {
|
|
shapeVersion = xcb_shape_query_version_unchecked(c);
|
|
}
|
|
if (m_randr.present) {
|
|
randrVersion = xcb_randr_query_version_unchecked(c, RANDR_MAX_MAJOR, RANDR_MAX_MINOR);
|
|
}
|
|
if (m_damage.present) {
|
|
damageVersion = xcb_damage_query_version_unchecked(c, DAMAGE_MAX_MAJOR, DAMAGE_MIN_MAJOR);
|
|
}
|
|
if (m_composite.present) {
|
|
compositeVersion = xcb_composite_query_version_unchecked(c, COMPOSITE_MAX_MAJOR, COMPOSITE_MAX_MINOR);
|
|
}
|
|
if (m_fixes.present) {
|
|
xfixesVersion = xcb_xfixes_query_version_unchecked(c, XFIXES_MAX_MAJOR, XFIXES_MAX_MINOR);
|
|
}
|
|
if (m_render.present) {
|
|
renderVersion = xcb_render_query_version_unchecked(c, RENDER_MAX_MAJOR, RENDER_MAX_MINOR);
|
|
}
|
|
if (m_sync.present) {
|
|
syncVersion = xcb_sync_initialize(c, SYNC_MAX_MAJOR, SYNC_MAX_MINOR);
|
|
}
|
|
// handle replies
|
|
if (m_shape.present) {
|
|
initVersion<xcb_shape_query_version_reply_t>(shapeVersion, &xcb_shape_query_version_reply, &m_shape);
|
|
}
|
|
if (m_randr.present) {
|
|
initVersion<xcb_randr_query_version_reply_t>(randrVersion, &xcb_randr_query_version_reply, &m_randr);
|
|
}
|
|
if (m_damage.present) {
|
|
initVersion<xcb_damage_query_version_reply_t>(damageVersion, &xcb_damage_query_version_reply, &m_damage);
|
|
}
|
|
if (m_composite.present) {
|
|
initVersion<xcb_composite_query_version_reply_t>(compositeVersion, &xcb_composite_query_version_reply, &m_composite);
|
|
}
|
|
if (m_fixes.present) {
|
|
initVersion<xcb_xfixes_query_version_reply_t>(xfixesVersion, &xcb_xfixes_query_version_reply, &m_fixes);
|
|
}
|
|
if (m_render.present) {
|
|
initVersion<xcb_render_query_version_reply_t>(renderVersion, &xcb_render_query_version_reply, &m_render);
|
|
}
|
|
if (m_sync.present) {
|
|
initVersion<xcb_sync_initialize_reply_t>(syncVersion, &xcb_sync_initialize_reply, &m_sync);
|
|
}
|
|
kDebug(1212) << "Extensions: shape: 0x" << QString::number(m_shape.version, 16)
|
|
<< " composite: 0x" << QString::number(m_composite.version, 16)
|
|
<< " render: 0x" << QString::number(m_render.version, 16)
|
|
<< " fixes: 0x" << QString::number(m_fixes.version, 16)
|
|
<< " randr: 0x" << QString::number(m_randr.version, 16)
|
|
<< " sync: 0x" << QString::number(m_sync.version, 16)
|
|
<< " damage: 0x " << QString::number(m_damage.version, 16) << endl;
|
|
}
|
|
|
|
void Extensions::extensionQueryReply(const xcb_query_extension_reply_t *extension, ExtensionData *dataToFill)
|
|
{
|
|
if (!extension) {
|
|
return;
|
|
}
|
|
dataToFill->present = extension->present;
|
|
dataToFill->eventBase = extension->first_event;
|
|
dataToFill->errorBase = extension->first_error;
|
|
dataToFill->majorOpcode = extension->major_opcode;
|
|
}
|
|
|
|
int Extensions::damageNotifyEvent() const
|
|
{
|
|
return m_damage.eventBase + XCB_DAMAGE_NOTIFY;
|
|
}
|
|
|
|
bool Extensions::hasShape(xcb_window_t w) const
|
|
{
|
|
if (!isShapeAvailable()) {
|
|
return false;
|
|
}
|
|
ScopedCPointer<xcb_shape_query_extents_reply_t> extents = xcb_shape_query_extents_reply(
|
|
connection(), xcb_shape_query_extents_unchecked(connection(), w), NULL);
|
|
if (extents.isNull()) {
|
|
return false;
|
|
}
|
|
return extents->bounding_shaped > 0;
|
|
}
|
|
|
|
bool Extensions::isCompositeOverlayAvailable() const
|
|
{
|
|
return m_composite.version >= 0x03; // 0.3
|
|
}
|
|
|
|
bool Extensions::isFixesRegionAvailable() const
|
|
{
|
|
return m_fixes.version >= 0x30; // 3
|
|
}
|
|
|
|
bool Extensions::isShapeInputAvailable() const
|
|
{
|
|
return m_shape.version >= 0x11; // 1.1
|
|
}
|
|
|
|
int Extensions::randrNotifyEvent() const
|
|
{
|
|
return m_randr.eventBase + XCB_RANDR_SCREEN_CHANGE_NOTIFY;
|
|
}
|
|
|
|
int Extensions::shapeNotifyEvent() const
|
|
{
|
|
return m_shape.eventBase + XCB_SHAPE_NOTIFY;
|
|
}
|
|
|
|
int Extensions::syncAlarmNotifyEvent() const
|
|
{
|
|
return m_sync.eventBase + XCB_SYNC_ALARM_NOTIFY;
|
|
}
|
|
|
|
QVector<ExtensionData> Extensions::extensions() const
|
|
{
|
|
QVector<ExtensionData> extensions;
|
|
extensions << m_shape
|
|
<< m_randr
|
|
<< m_damage
|
|
<< m_composite
|
|
<< m_render
|
|
<< m_fixes
|
|
<< m_sync;
|
|
return extensions;
|
|
}
|
|
|
|
} // namespace Xcb
|
|
} // namespace KWin
|