1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
// Copyright (C) 2017 ITAGE Corporation, author: <yusuke.binsaki@itage.co.jp>
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
#include "qwaylandivishellintegration.h"
#include <QtCore/qsize.h>
#include <QtCore/qdebug.h>
#include <QtWaylandClient/private/qwaylanddisplay_p.h>
#include <QtWaylandClient/private/qwaylandwindow_p.h>
#include <QtWaylandClient/private/qwaylandabstractdecoration_p.h>
#include "qwaylandivisurface_p.h"
#include <mutex>
#include <unistd.h>
QT_BEGIN_NAMESPACE
namespace QtWaylandClient {
QWaylandIviShellIntegration::QWaylandIviShellIntegration()
{
}
bool QWaylandIviShellIntegration::initialize(QWaylandDisplay *display)
{
for (QWaylandDisplay::RegistryGlobal global : display->globals()) {
if (global.interface == QLatin1String("ivi_application") && !m_iviApplication)
m_iviApplication.reset(new QtWayland::ivi_application(display->wl_registry(), global.id, global.version));
if (global.interface == QLatin1String("ivi_controller") && !m_iviController)
m_iviController.reset(new QtWayland::ivi_controller(display->wl_registry(), global.id, global.version));
}
if (!m_iviApplication) {
qCDebug(lcQpaWayland) << "Couldn't find global ivi_application for ivi-shell";
return false;
}
return true;
}
/* get unique id
* pattern1:
* When set QT_IVI_SURFACE_ID, We use it as ID.
* Next ID is increment.
* pattern2:
* When not set QT_IVI_SURFACE_ID, We use process ID and unused bit.
* process ID maximum is 2^22. Unused bit is 23 to 32 bit.
* Therefor, We use 23 to 32 bit. This do not overlap with other clients.
* Next ID is increment of 23 to 32 bit.
* +------------+---------------------------+
* |31 23|22 0|
* +------------+---------------------------+
* |0000 0000 00|00 0000 0000 0000 0000 0000|
* |<- ID ->|<- process ID ->|
* +------------+---------------------------+
*/
uint32_t QWaylandIviShellIntegration::getNextUniqueSurfaceId()
{
const uint32_t PID_MAX_EXPONENTIATION = 22; // 22 bit shift operation
const uint32_t ID_LIMIT = 1 << (32 - PID_MAX_EXPONENTIATION); // 10 bit is unique id
const std::lock_guard<QRecursiveMutex> locker(m_mutex);
if (m_lastSurfaceId == 0) {
QByteArray env = qgetenv("QT_IVI_SURFACE_ID");
bool ok;
m_lastSurfaceId = env.toUInt(&ok, 10);
if (ok)
m_useEnvSurfaceId = true;
else
m_lastSurfaceId = getpid();
return m_lastSurfaceId;
}
if (m_useEnvSurfaceId) {
m_lastSurfaceId++;
} else {
m_surfaceNumber++;
if (m_surfaceNumber >= ID_LIMIT) {
qWarning("IVI surface id counter overflow\n");
return 0;
}
m_lastSurfaceId += (m_surfaceNumber << PID_MAX_EXPONENTIATION);
}
return m_lastSurfaceId;
}
QWaylandShellSurface *QWaylandIviShellIntegration::createShellSurface(QWaylandWindow *window)
{
if (!m_iviApplication)
return nullptr;
uint32_t surfaceId = getNextUniqueSurfaceId();
if (surfaceId == 0)
return nullptr;
struct ivi_surface *surface = m_iviApplication->surface_create(surfaceId, window->wlSurface());
if (!m_iviController)
return new QWaylandIviSurface(surface, window);
struct ::ivi_controller_surface *controller = m_iviController->ivi_controller::surface_create(surfaceId);
QWaylandIviSurface *iviSurface = new QWaylandIviSurface(surface, window, controller);
if (window->window()->type() == Qt::Popup) {
QPoint transientPos = window->geometry().topLeft(); // this is absolute
QWaylandWindow *parent = window->transientParent();
if (parent && parent->decoration()) {
transientPos -= parent->geometry().topLeft();
transientPos.setX(transientPos.x() + parent->decoration()->margins().left());
transientPos.setY(transientPos.y() + parent->decoration()->margins().top());
}
QSize size = window->windowGeometry().size();
iviSurface->ivi_controller_surface::set_destination_rectangle(transientPos.x(),
transientPos.y(),
size.width(),
size.height());
}
return iviSurface;
}
}
QT_END_NAMESPACE
|