// Copyright (c) 2011 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/message_pump_glib_x.h" #include #if defined(HAVE_XINPUT2) #include #else #include #endif #include "base/message_pump_glib_x_dispatch.h" namespace { gboolean PlaceholderDispatch(GSource* source, GSourceFunc cb, gpointer data) { return TRUE; } #if defined(HAVE_XINPUT2) // Setup XInput2 select for the GtkWidget. gboolean GtkWidgetRealizeCallback(GSignalInvocationHint* hint, guint nparams, const GValue* pvalues, gpointer data) { GtkWidget* widget = GTK_WIDGET(g_value_get_object(pvalues)); GdkWindow* window = widget->window; base::MessagePumpGlibX* msgpump = static_cast(data); DCHECK(window); // TODO(sad): Remove once determined if necessary. if (GDK_WINDOW_TYPE(window) != GDK_WINDOW_TOPLEVEL && GDK_WINDOW_TYPE(window) != GDK_WINDOW_CHILD && GDK_WINDOW_TYPE(window) != GDK_WINDOW_DIALOG) return true; // TODO(sad): Do we need to set a flag on |window| to make sure we don't // select for the same GdkWindow multiple times? Does it matter? msgpump->SetupXInput2ForXWindow(GDK_WINDOW_XID(window)); return true; } // We need to capture all the GDK windows that get created, and start // listening for XInput2 events. So we setup a callback to the 'realize' // signal for GTK+ widgets, so that whenever the signal triggers for any // GtkWidget, which means the GtkWidget should now have a GdkWindow, we can // setup XInput2 events for the GdkWindow. static guint realize_signal_id = 0; static guint realize_hook_id = 0; void SetupGtkWidgetRealizeNotifier(base::MessagePumpGlibX* msgpump) { gpointer klass = g_type_class_ref(GTK_TYPE_WIDGET); g_signal_parse_name("realize", GTK_TYPE_WIDGET, &realize_signal_id, NULL, FALSE); realize_hook_id = g_signal_add_emission_hook(realize_signal_id, 0, GtkWidgetRealizeCallback, static_cast(msgpump), NULL); g_type_class_unref(klass); } void RemoveGtkWidgetRealizeNotifier() { if (realize_signal_id != 0) g_signal_remove_emission_hook(realize_signal_id, realize_hook_id); realize_signal_id = 0; realize_hook_id = 0; } #endif // HAVE_XINPUT2 } // namespace namespace base { MessagePumpGlibX::MessagePumpGlibX() : base::MessagePumpForUI(), #if defined(HAVE_XINPUT2) xiopcode_(-1), pointer_devices_(), #endif gdksource_(NULL), dispatching_event_(false), capture_x_events_(0), capture_gdk_events_(0) { gdk_event_handler_set(&EventDispatcherX, this, NULL); #if defined(HAVE_XINPUT2) InitializeXInput2(); #endif InitializeEventsToCapture(); } MessagePumpGlibX::~MessagePumpGlibX() { #if defined(HAVE_XINPUT2) RemoveGtkWidgetRealizeNotifier(); #endif } #if defined(HAVE_XINPUT2) void MessagePumpGlibX::SetupXInput2ForXWindow(Window xwindow) { Display* xdisplay = GDK_DISPLAY_XDISPLAY(gdk_display_get_default()); // Setup mask for mouse events. unsigned char mask[(XI_LASTEVENT + 7)/8]; memset(mask, 0, sizeof(mask)); XISetMask(mask, XI_ButtonPress); XISetMask(mask, XI_ButtonRelease); XISetMask(mask, XI_Motion); XIEventMask evmasks[pointer_devices_.size()]; int count = 0; for (std::set::const_iterator iter = pointer_devices_.begin(); iter != pointer_devices_.end(); ++iter, ++count) { evmasks[count].deviceid = *iter; evmasks[count].mask_len = sizeof(mask); evmasks[count].mask = mask; } XISelectEvents(xdisplay, xwindow, evmasks, pointer_devices_.size()); // TODO(sad): Setup masks for keyboard events. XFlush(xdisplay); } #endif // HAVE_XINPUT2 bool MessagePumpGlibX::RunOnce(GMainContext* context, bool block) { GdkDisplay* gdisp = gdk_display_get_default(); if (!gdisp || !GetDispatcher()) return MessagePumpForUI::RunOnce(context, block); Display* display = GDK_DISPLAY_XDISPLAY(gdisp); bool should_quit = false; if (XPending(display)) { XEvent xev; XPeekEvent(display, &xev); if (capture_x_events_[xev.type] #if defined(HAVE_XINPUT2) && (xev.type != GenericEvent || xev.xcookie.extension == xiopcode_) #endif ) { XNextEvent(display, &xev); #if defined(HAVE_XINPUT2) bool have_cookie = false; if (xev.type == GenericEvent && XGetEventData(xev.xgeneric.display, &xev.xcookie)) { have_cookie = true; } #endif MessagePumpGlibXDispatcher::DispatchStatus status = static_cast (GetDispatcher())->DispatchX(&xev); if (status == MessagePumpGlibXDispatcher::EVENT_QUIT) { should_quit = true; Quit(); } else if (status == MessagePumpGlibXDispatcher::EVENT_IGNORED) { DLOG(WARNING) << "Event (" << xev.type << ") not handled."; // TODO(sad): It is necessary to put back the event so that the default // GDK events handler can take care of it. Without this, it is // impossible to use the omnibox at the moment. However, this will // eventually be removed once the omnibox code is updated for touchui. XPutBackEvent(display, &xev); if (gdksource_) gdksource_->source_funcs->dispatch = gdkdispatcher_; g_main_context_iteration(context, FALSE); } #if defined(HAVE_XINPUT2) if (have_cookie) { XFreeEventData(xev.xgeneric.display, &xev.xcookie); } #endif } else { // TODO(sad): A couple of extra events can still sneak in during this. // Those should be sent back to the X queue from the dispatcher // EventDispatcherX. if (gdksource_) gdksource_->source_funcs->dispatch = gdkdispatcher_; g_main_context_iteration(context, FALSE); } } if (should_quit) return true; bool retvalue; if (gdksource_) { // Replace the dispatch callback of the GDK event source temporarily so that // it doesn't read events from X. gboolean (*cb)(GSource*, GSourceFunc, void*) = gdksource_->source_funcs->dispatch; gdksource_->source_funcs->dispatch = PlaceholderDispatch; dispatching_event_ = true; retvalue = g_main_context_iteration(context, block); dispatching_event_ = false; gdksource_->source_funcs->dispatch = cb; } else { retvalue = g_main_context_iteration(context, block); } return retvalue; } void MessagePumpGlibX::EventDispatcherX(GdkEvent* event, gpointer data) { MessagePumpGlibX* pump_x = reinterpret_cast(data); if (!pump_x->gdksource_) { pump_x->gdksource_ = g_main_current_source(); if (pump_x->gdksource_) pump_x->gdkdispatcher_ = pump_x->gdksource_->source_funcs->dispatch; } else if (!pump_x->IsDispatchingEvent()) { if (event->type != GDK_NOTHING && pump_x->capture_gdk_events_[event->type]) { // TODO(sad): An X event is caught by the GDK handler. Put it back in the // X queue so that we catch it in the next iteration. When done, the // following DLOG statement will be removed. DLOG(WARNING) << "GDK received an event it shouldn't have"; } } pump_x->DispatchEvents(event); } void MessagePumpGlibX::InitializeEventsToCapture(void) { // TODO(sad): Decide which events we want to capture and update the tables // accordingly. capture_x_events_[KeyPress] = true; capture_gdk_events_[GDK_KEY_PRESS] = true; capture_x_events_[KeyRelease] = true; capture_gdk_events_[GDK_KEY_RELEASE] = true; capture_x_events_[ButtonPress] = true; capture_gdk_events_[GDK_BUTTON_PRESS] = true; capture_x_events_[ButtonRelease] = true; capture_gdk_events_[GDK_BUTTON_RELEASE] = true; capture_x_events_[MotionNotify] = true; capture_gdk_events_[GDK_MOTION_NOTIFY] = true; #if defined(HAVE_XINPUT2) capture_x_events_[GenericEvent] = true; #endif } #if defined(HAVE_XINPUT2) void MessagePumpGlibX::InitializeXInput2(void) { GdkDisplay* display = gdk_display_get_default(); if (!display) return; Display* xdisplay = GDK_DISPLAY_XDISPLAY(display); int event, err; if (!XQueryExtension(xdisplay, "XInputExtension", &xiopcode_, &event, &err)) { DLOG(WARNING) << "X Input extension not available."; xiopcode_ = -1; return; } int major = 2, minor = 0; if (XIQueryVersion(xdisplay, &major, &minor) == BadRequest) { DLOG(WARNING) << "XInput2 not supported in the server."; xiopcode_ = -1; return; } // TODO(sad): Here, we only setup so that the X windows created by GTK+ are // setup for XInput2 events. We need a way to listen for XInput2 events for X // windows created by other means (e.g. for context menus). SetupGtkWidgetRealizeNotifier(this); // Instead of asking X for the list of devices all the time, let's maintain a // list of pointer devices we care about. // It is not necessary to select for slave devices. XInput2 provides enough // information to the event callback to decide which slave device triggered // the event, thus decide whether the 'pointer event' is a 'mouse event' or a // 'touch event'. // If the touch device has 'GrabDevice' set and 'SendCoreEvents' unset (which // is possible), then the device is detected as a floating device, and a // floating device is not connected to a master device. So it is necessary to // also select on the floating devices. int count = 0; XIDeviceInfo* devices = XIQueryDevice(xdisplay, XIAllDevices, &count); for (int i = 0; i < count; i++) { XIDeviceInfo* devinfo = devices + i; if (devinfo->use == XIFloatingSlave || devinfo->use == XIMasterPointer) pointer_devices_.insert(devinfo->deviceid); } XIFreeDeviceInfo(devices); // TODO(sad): Select on root for XI_HierarchyChanged so that floats_ and // masters_ can be kept up-to-date. This is a relatively rare event, so we can // put it off for a later time. // Note: It is not necessary to listen for XI_DeviceChanged events. } #endif // HAVE_XINPUT2 } // namespace base