/* KWin - the KDE window manager This file is part of the KDE project. SPDX-FileCopyrightText: 1999, 2000 Matthias Ettrich SPDX-FileCopyrightText: 2003 Lubos Lunak SPDX-License-Identifier: GPL-2.0-or-later */ #include "main.h" #include #include "atoms.h" #include "colormanager.h" #include "composite.h" #include "core/outputbackend.h" #include "core/session.h" #include "cursor.h" #include "effects.h" #include "input.h" #include "inputmethod.h" #include "options.h" #include "outline.h" #include "pluginmanager.h" #include "pointer_input.h" #include "screenedge.h" #include "sm.h" #include "tabletmodemanager.h" #include "utils/xcbutils.h" #include "wayland/surface_interface.h" #include "workspace.h" #include "x11eventfilter.h" #if KWIN_BUILD_SCREENLOCKER #include "screenlockerwatcher.h" #endif #include // KDE #include #include // Qt #include #include #include #include #include #include #include #include #if __has_include() #include #endif #include // xcb #include #ifndef XCB_GE_GENERIC #define XCB_GE_GENERIC 35 #endif Q_DECLARE_METATYPE(KSharedConfigPtr) namespace KWin { Options *options; Atoms *atoms; int Application::crashes = 0; Application::Application(Application::OperationMode mode, int &argc, char **argv) : QApplication(argc, argv) , m_eventFilter(new XcbEventFilter()) , m_configLock(false) , m_config(KSharedConfig::openConfig(QStringLiteral("kwinrc"))) , m_kxkbConfig() , m_operationMode(mode) { qRegisterMetaType("Options::WindowOperation"); qRegisterMetaType(); qRegisterMetaType("KWaylandServer::SurfaceInterface *"); qRegisterMetaType(); qRegisterMetaType(); } void Application::setConfigLock(bool lock) { m_configLock = lock; } Application::OperationMode Application::operationMode() const { return m_operationMode; } void Application::setOperationMode(OperationMode mode) { m_operationMode = mode; } bool Application::shouldUseWaylandForCompositing() const { return m_operationMode == OperationModeWaylandOnly || m_operationMode == OperationModeXwayland; } void Application::start() { // Prevent KWin from synchronously autostarting kactivitymanagerd // Indeed, kactivitymanagerd being a QApplication it will depend // on KWin startup... this is unsatisfactory dependency wise, // and it turns out that it leads to a deadlock in the Wayland case setProperty("org.kde.KActivities.core.disableAutostart", true); setQuitOnLastWindowClosed(false); if (!m_config->isImmutable() && m_configLock) { // TODO: This shouldn't be necessary // config->setReadOnly( true ); m_config->reparseConfiguration(); } if (!m_kxkbConfig) { m_kxkbConfig = KSharedConfig::openConfig(QStringLiteral("kxkbrc"), KConfig::NoGlobals); } performStartup(); } Application::~Application() { delete options; destroyAtoms(); destroyPlatform(); m_session.reset(); } void Application::notifyStarted() { Q_EMIT started(); } void Application::destroyAtoms() { delete atoms; atoms = nullptr; } void Application::destroyPlatform() { m_outputBackend.reset(); } void Application::resetCrashesCount() { crashes = 0; } void Application::setCrashCount(int count) { crashes = count; } bool Application::wasCrash() { return crashes > 0; } void Application::createAboutData() { KAboutData aboutData(QStringLiteral("kwin"), // The program name used internally i18n("KWin"), // A displayable program name string QStringLiteral(KWIN_VERSION_STRING), // The program version string i18n("KDE window manager"), // Short description of what the app does KAboutLicense::GPL, // The license this code is released under i18n("(c) 1999-2019, The KDE Developers")); // Copyright Statement aboutData.addAuthor(i18n("Matthias Ettrich"), QString(), QStringLiteral("ettrich@kde.org")); aboutData.addAuthor(i18n("Cristian Tibirna"), QString(), QStringLiteral("tibirna@kde.org")); aboutData.addAuthor(i18n("Daniel M. Duley"), QString(), QStringLiteral("mosfet@kde.org")); aboutData.addAuthor(i18n("Luboš Luňák"), QString(), QStringLiteral("l.lunak@kde.org")); aboutData.addAuthor(i18n("Martin Flöser"), QString(), QStringLiteral("mgraesslin@kde.org")); aboutData.addAuthor(i18n("David Edmundson"), QStringLiteral("Maintainer"), QStringLiteral("davidedmundson@kde.org")); aboutData.addAuthor(i18n("Roman Gilg"), QStringLiteral("Maintainer"), QStringLiteral("subdiff@gmail.com")); aboutData.addAuthor(i18n("Vlad Zahorodnii"), QStringLiteral("Maintainer"), QStringLiteral("vlad.zahorodnii@kde.org")); KAboutData::setApplicationData(aboutData); } static const QString s_lockOption = QStringLiteral("lock"); static const QString s_crashesOption = QStringLiteral("crashes"); void Application::setupCommandLine(QCommandLineParser *parser) { QCommandLineOption lockOption(s_lockOption, i18n("Disable configuration options")); QCommandLineOption crashesOption(s_crashesOption, i18n("Indicate that KWin has recently crashed n times"), QStringLiteral("n")); parser->setApplicationDescription(i18n("KDE window manager")); parser->addOption(lockOption); parser->addOption(crashesOption); KAboutData::applicationData().setupCommandLine(parser); } void Application::processCommandLine(QCommandLineParser *parser) { KAboutData aboutData = KAboutData::applicationData(); aboutData.processCommandLine(parser); setConfigLock(parser->isSet(s_lockOption)); Application::setCrashCount(parser->value(s_crashesOption).toInt()); } void Application::setupTranslator() { QTranslator *qtTranslator = new QTranslator(qApp); qtTranslator->load("qt_" + QLocale::system().name(), QLibraryInfo::location(QLibraryInfo::TranslationsPath)); installTranslator(qtTranslator); } void Application::setupMalloc() { #ifdef M_TRIM_THRESHOLD // Prevent fragmentation of the heap by malloc (glibc). // // The default threshold is 128*1024, which can result in a large memory usage // due to fragmentation especially if we use the raster graphicssystem. On the // otherside if the threshold is too low, free() starts to permanently ask the kernel // about shrinking the heap. const int pagesize = sysconf(_SC_PAGESIZE); mallopt(M_TRIM_THRESHOLD, 5 * pagesize); #endif // M_TRIM_THRESHOLD } void Application::setupLocalizedString() { KLocalizedString::setApplicationDomain("kwin"); } void Application::createWorkspace() { // we want all QQuickWindows with an alpha buffer, do here as Workspace might create QQuickWindows QQuickWindow::setDefaultAlphaBuffer(true); // This tries to detect compositing options and can use GLX. GLX problems // (X errors) shouldn't cause kwin to abort, so this is out of the // critical startup section where x errors cause kwin to abort. // create workspace. (void)new Workspace(); Q_EMIT workspaceCreated(); } void Application::createInput() { #if KWIN_BUILD_SCREENLOCKER m_screenLockerWatcher = std::make_unique(); #endif auto input = InputRedirection::create(this); input->init(); createPlatformCursor(this); } void Application::createAtoms() { atoms = new Atoms; } void Application::createOptions() { options = new Options; } void Application::createPlugins() { m_pluginManager = std::make_unique(); } void Application::createColorManager() { m_colorManager = std::make_unique(); } void Application::createInputMethod() { m_inputMethod = std::make_unique(); } void Application::createTabletModeManager() { m_tabletModeManager = std::make_unique(); } void Application::installNativeX11EventFilter() { installNativeEventFilter(m_eventFilter.get()); } void Application::removeNativeX11EventFilter() { removeNativeEventFilter(m_eventFilter.get()); } void Application::destroyInput() { delete InputRedirection::self(); } void Application::destroyWorkspace() { delete Workspace::self(); } void Application::destroyCompositor() { delete Compositor::self(); } void Application::destroyPlugins() { m_pluginManager.reset(); } void Application::destroyColorManager() { m_colorManager.reset(); } void Application::destroyInputMethod() { m_inputMethod.reset(); } std::unique_ptr Application::createScreenEdge(ScreenEdges *edges) { return std::make_unique(edges); } void Application::createPlatformCursor(QObject *parent) { new InputRedirectionCursor(parent); } std::unique_ptr Application::createOutline(Outline *outline) { if (Compositor::compositing()) { return std::make_unique(outline); } return nullptr; } void Application::createEffectsHandler(Compositor *compositor, WorkspaceScene *scene) { new EffectsHandlerImpl(compositor, scene); } void Application::registerEventFilter(X11EventFilter *filter) { if (filter->isGenericEvent()) { m_genericEventFilters.append(new X11EventFilterContainer(filter)); } else { m_eventFilters.append(new X11EventFilterContainer(filter)); } } static X11EventFilterContainer *takeEventFilter(X11EventFilter *eventFilter, QList> &list) { for (int i = 0; i < list.count(); ++i) { X11EventFilterContainer *container = list.at(i); if (container->filter() == eventFilter) { return list.takeAt(i); } } return nullptr; } void Application::setXwaylandScale(qreal scale) { if (scale != m_xwaylandScale) { m_xwaylandScale = scale; Q_EMIT xwaylandScaleChanged(); } } void Application::unregisterEventFilter(X11EventFilter *filter) { X11EventFilterContainer *container = nullptr; if (filter->isGenericEvent()) { container = takeEventFilter(filter, m_genericEventFilters); } else { container = takeEventFilter(filter, m_eventFilters); } delete container; } bool Application::dispatchEvent(xcb_generic_event_t *event) { static const QVector s_xcbEerrors({QByteArrayLiteral("Success"), QByteArrayLiteral("BadRequest"), QByteArrayLiteral("BadValue"), QByteArrayLiteral("BadWindow"), QByteArrayLiteral("BadPixmap"), QByteArrayLiteral("BadAtom"), QByteArrayLiteral("BadCursor"), QByteArrayLiteral("BadFont"), QByteArrayLiteral("BadMatch"), QByteArrayLiteral("BadDrawable"), QByteArrayLiteral("BadAccess"), QByteArrayLiteral("BadAlloc"), QByteArrayLiteral("BadColor"), QByteArrayLiteral("BadGC"), QByteArrayLiteral("BadIDChoice"), QByteArrayLiteral("BadName"), QByteArrayLiteral("BadLength"), QByteArrayLiteral("BadImplementation"), QByteArrayLiteral("Unknown")}); kwinApp()->updateX11Time(event); const uint8_t x11EventType = event->response_type & ~0x80; if (!x11EventType) { // let's check whether it's an error from one of the extensions KWin uses xcb_generic_error_t *error = reinterpret_cast(event); const QVector extensions = Xcb::Extensions::self()->extensions(); for (const auto &extension : extensions) { if (error->major_code == extension.majorOpcode) { QByteArray errorName; if (error->error_code < s_xcbEerrors.size()) { errorName = s_xcbEerrors.at(error->error_code); } else if (error->error_code >= extension.errorBase) { const int index = error->error_code - extension.errorBase; if (index >= 0 && index < extension.errorCodes.size()) { errorName = extension.errorCodes.at(index); } } if (errorName.isEmpty()) { errorName = QByteArrayLiteral("Unknown"); } qCWarning(KWIN_CORE, "XCB error: %d (%s), sequence: %d, resource id: %d, major code: %d (%s), minor code: %d (%s)", int(error->error_code), errorName.constData(), int(error->sequence), int(error->resource_id), int(error->major_code), extension.name.constData(), int(error->minor_code), extension.opCodes.size() > error->minor_code ? extension.opCodes.at(error->minor_code).constData() : "Unknown"); return true; } } return false; } if (x11EventType == XCB_GE_GENERIC) { xcb_ge_generic_event_t *ge = reinterpret_cast(event); // We need to make a shadow copy of the event filter list because an activated event // filter may mutate it by removing or installing another event filter. const auto eventFilters = m_genericEventFilters; for (X11EventFilterContainer *container : eventFilters) { if (!container) { continue; } X11EventFilter *filter = container->filter(); if (filter->extension() == ge->extension && filter->genericEventTypes().contains(ge->event_type) && filter->event(event)) { return true; } } } else { // We need to make a shadow copy of the event filter list because an activated event // filter may mutate it by removing or installing another event filter. const auto eventFilters = m_eventFilters; for (X11EventFilterContainer *container : eventFilters) { if (!container) { continue; } X11EventFilter *filter = container->filter(); if (filter->eventTypes().contains(x11EventType) && filter->event(event)) { return true; } } } if (workspace()) { return workspace()->workspaceEvent(event); } return false; } static quint32 monotonicTime() { timespec ts; const int result = clock_gettime(CLOCK_MONOTONIC, &ts); if (result) { qCWarning(KWIN_CORE, "Failed to query monotonic time: %s", strerror(errno)); } return ts.tv_sec * 1000 + ts.tv_nsec / 1000000L; } void Application::updateXTime() { switch (operationMode()) { case Application::OperationModeX11: setX11Time(QX11Info::getTimestamp(), TimestampUpdate::Always); break; case Application::OperationModeXwayland: setX11Time(monotonicTime(), TimestampUpdate::Always); break; default: // Do not update the current X11 time stamp if it's the Wayland only session. break; } } void Application::updateX11Time(xcb_generic_event_t *event) { xcb_timestamp_t time = XCB_TIME_CURRENT_TIME; const uint8_t eventType = event->response_type & ~0x80; switch (eventType) { case XCB_KEY_PRESS: case XCB_KEY_RELEASE: time = reinterpret_cast(event)->time; break; case XCB_BUTTON_PRESS: case XCB_BUTTON_RELEASE: time = reinterpret_cast(event)->time; break; case XCB_MOTION_NOTIFY: time = reinterpret_cast(event)->time; break; case XCB_ENTER_NOTIFY: case XCB_LEAVE_NOTIFY: time = reinterpret_cast(event)->time; break; case XCB_FOCUS_IN: case XCB_FOCUS_OUT: case XCB_KEYMAP_NOTIFY: case XCB_EXPOSE: case XCB_GRAPHICS_EXPOSURE: case XCB_NO_EXPOSURE: case XCB_VISIBILITY_NOTIFY: case XCB_CREATE_NOTIFY: case XCB_DESTROY_NOTIFY: case XCB_UNMAP_NOTIFY: case XCB_MAP_NOTIFY: case XCB_MAP_REQUEST: case XCB_REPARENT_NOTIFY: case XCB_CONFIGURE_NOTIFY: case XCB_CONFIGURE_REQUEST: case XCB_GRAVITY_NOTIFY: case XCB_RESIZE_REQUEST: case XCB_CIRCULATE_NOTIFY: case XCB_CIRCULATE_REQUEST: // no timestamp return; case XCB_PROPERTY_NOTIFY: time = reinterpret_cast(event)->time; break; case XCB_SELECTION_CLEAR: time = reinterpret_cast(event)->time; break; case XCB_SELECTION_REQUEST: time = reinterpret_cast(event)->time; break; case XCB_SELECTION_NOTIFY: time = reinterpret_cast(event)->time; break; case XCB_COLORMAP_NOTIFY: case XCB_CLIENT_MESSAGE: case XCB_MAPPING_NOTIFY: case XCB_GE_GENERIC: // no timestamp return; default: // extension handling if (Xcb::Extensions::self()) { if (eventType == Xcb::Extensions::self()->shapeNotifyEvent()) { time = reinterpret_cast(event)->server_time; } if (eventType == Xcb::Extensions::self()->damageNotifyEvent()) { time = reinterpret_cast(event)->timestamp; } } break; } setX11Time(time); } bool XcbEventFilter::nativeEventFilter(const QByteArray &eventType, void *message, qintptr *result) { if (eventType == "xcb_generic_event_t") { return kwinApp()->dispatchEvent(static_cast(message)); } return false; } QProcessEnvironment Application::processStartupEnvironment() const { return m_processEnvironment; } void Application::setProcessStartupEnvironment(const QProcessEnvironment &environment) { m_processEnvironment = environment; } void Application::setOutputBackend(std::unique_ptr &&backend) { Q_ASSERT(!m_outputBackend); m_outputBackend = std::move(backend); } void Application::setSession(std::unique_ptr &&session) { Q_ASSERT(!m_session); m_session = std::move(session); } PluginManager *Application::pluginManager() const { return m_pluginManager.get(); } InputMethod *Application::inputMethod() const { return m_inputMethod.get(); } ColorManager *Application::colorManager() const { return m_colorManager.get(); } XwaylandInterface *Application::xwayland() const { return nullptr; } #if KWIN_BUILD_SCREENLOCKER ScreenLockerWatcher *Application::screenLockerWatcher() const { return m_screenLockerWatcher.get(); } #endif PlatformCursorImage Application::cursorImage() const { Cursor *cursor = Cursors::self()->currentCursor(); return PlatformCursorImage(cursor->image(), cursor->hotspot()); } void Application::startInteractiveWindowSelection(std::function callback, const QByteArray &cursorName) { if (!input()) { callback(nullptr); return; } input()->startInteractiveWindowSelection(callback, cursorName); } void Application::startInteractivePositionSelection(std::function callback) { if (!input()) { callback(QPointF(-1, -1)); return; } input()->startInteractivePositionSelection(callback); } } // namespace