blob: 8bb03c77dee3a205cfbf8195b2bcabb94106887b [file] [log] [blame]
// Copyright (c) 2012 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.
// This file defines utility functions for X11 (Linux only). This code has been
// ported from XCB since we can't use XCB on Ubuntu while its 32-bit support
// remains woefully incomplete.
#include "ui/base/x/x11_util.h"
#include <ctype.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <bitset>
#include <limits>
#include <list>
#include <map>
#include <utility>
#include <vector>
#include "base/bind.h"
#include "base/command_line.h"
#include "base/compiler_specific.h"
#include "base/location.h"
#include "base/logging.h"
#include "base/memory/ref_counted_memory.h"
#include "base/memory/singleton.h"
#include "base/metrics/histogram_macros.h"
#include "base/no_destructor.h"
#include "base/notreached.h"
#include "base/single_thread_task_runner.h"
#include "base/stl_util.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
#include "base/sys_byteorder.h"
#include "base/task/current_thread.h"
#include "base/threading/thread.h"
#include "base/threading/thread_local_storage.h"
#include "base/threading/thread_task_runner_handle.h"
#include "base/trace_event/trace_event.h"
#include "build/build_config.h"
#include "skia/ext/image_operations.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "third_party/skia/include/core/SkImageInfo.h"
#include "third_party/skia/include/core/SkTypes.h"
#include "ui/base/cursor/mojom/cursor_type.mojom-shared.h"
#include "ui/base/x/x11_cursor.h"
#include "ui/base/x/x11_cursor_loader.h"
#include "ui/base/x/x11_menu_list.h"
#include "ui/events/devices/x11/device_data_manager_x11.h"
#include "ui/events/devices/x11/touch_factory_x11.h"
#include "ui/events/event_utils.h"
#include "ui/events/keycodes/keyboard_code_conversion_x.h"
#include "ui/gfx/canvas.h"
#include "ui/gfx/geometry/insets.h"
#include "ui/gfx/geometry/point.h"
#include "ui/gfx/geometry/point_conversions.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/gfx/geometry/size.h"
#include "ui/gfx/image/image_skia.h"
#include "ui/gfx/image/image_skia_rep.h"
#include "ui/gfx/skia_util.h"
#include "ui/gfx/switches.h"
#include "ui/gfx/x/connection.h"
#include "ui/gfx/x/shm.h"
#include "ui/gfx/x/sync.h"
#include "ui/gfx/x/x11.h"
#include "ui/gfx/x/x11_atom_cache.h"
#include "ui/gfx/x/x11_error_tracker.h"
#include "ui/gfx/x/xproto.h"
#include "ui/gfx/x/xproto_util.h"
#if defined(OS_FREEBSD)
#include <sys/sysctl.h>
#include <sys/types.h>
#endif
namespace ui {
class TLSDestructionCheckerForX11 {
public:
static bool HasBeenDestroyed() {
return base::ThreadLocalStorage::HasBeenDestroyed();
}
};
namespace {
// Constants that are part of EWMH.
constexpr int kNetWMStateAdd = 1;
constexpr int kNetWMStateRemove = 0;
int DefaultX11ErrorHandler(XDisplay* d, XErrorEvent* e) {
// This callback can be invoked by drivers very late in thread destruction,
// when Chrome TLS is no longer usable. https://crbug.com/849225.
if (TLSDestructionCheckerForX11::HasBeenDestroyed())
return 0;
if (base::CurrentThread::Get()) {
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE,
base::BindOnce(&x11::LogErrorEventDescription, e->serial, e->error_code,
e->request_code, e->minor_code));
} else {
LOG(ERROR) << "X error received: "
<< "serial " << e->serial << ", "
<< "error_code " << static_cast<int>(e->error_code) << ", "
<< "request_code " << static_cast<int>(e->request_code) << ", "
<< "minor_code " << static_cast<int>(e->minor_code);
}
return 0;
}
int DefaultX11IOErrorHandler(XDisplay* d) {
// If there's an IO error it likely means the X server has gone away
LOG(ERROR) << "X IO error received (X server probably went away)";
_exit(1);
}
bool SupportsEWMH() {
static bool supports_ewmh = false;
static bool supports_ewmh_cached = false;
if (!supports_ewmh_cached) {
supports_ewmh_cached = true;
x11::Window wm_window = x11::Window::None;
if (!GetProperty(GetX11RootWindow(),
gfx::GetAtom("_NET_SUPPORTING_WM_CHECK"), &wm_window)) {
supports_ewmh = false;
return false;
}
// It's possible that a window manager started earlier in this X session
// left a stale _NET_SUPPORTING_WM_CHECK property when it was replaced by a
// non-EWMH window manager, so we trap errors in the following requests to
// avoid crashes (issue 23860).
// EWMH requires the supporting-WM window to also have a
// _NET_SUPPORTING_WM_CHECK property pointing to itself (to avoid a stale
// property referencing an ID that's been recycled for another window), so
// we check that too.
gfx::X11ErrorTracker err_tracker;
x11::Window wm_window_property = x11::Window::None;
bool result =
GetProperty(wm_window, gfx::GetAtom("_NET_SUPPORTING_WM_CHECK"),
&wm_window_property);
supports_ewmh = !err_tracker.FoundNewError() && result &&
wm_window_property == wm_window;
}
return supports_ewmh;
}
bool GetWindowManagerName(std::string* wm_name) {
DCHECK(wm_name);
if (!SupportsEWMH())
return false;
x11::Window wm_window = x11::Window::None;
if (!GetProperty(GetX11RootWindow(), gfx::GetAtom("_NET_SUPPORTING_WM_CHECK"),
&wm_window)) {
return false;
}
gfx::X11ErrorTracker err_tracker;
bool result = GetStringProperty(wm_window, "_NET_WM_NAME", wm_name);
return !err_tracker.FoundNewError() && result;
}
} // namespace
void DeleteProperty(x11::Window window, x11::Atom name) {
x11::Connection::Get()->DeleteProperty({
.window = static_cast<x11::Window>(window),
.property = name,
});
}
bool GetWmNormalHints(x11::Window window, SizeHints* hints) {
std::vector<uint32_t> hints32;
if (!GetArrayProperty(window, gfx::GetAtom("WM_NORMAL_HINTS"), &hints32))
return false;
if (hints32.size() != sizeof(SizeHints) / 4)
return false;
memcpy(hints, hints32.data(), sizeof(*hints));
return true;
}
void SetWmNormalHints(x11::Window window, const SizeHints& hints) {
std::vector<uint32_t> hints32(sizeof(SizeHints) / 4);
memcpy(hints32.data(), &hints, sizeof(SizeHints));
ui::SetArrayProperty(window, gfx::GetAtom("WM_NORMAL_HINTS"),
gfx::GetAtom("WM_SIZE_HINTS"), hints32);
}
bool GetWmHints(x11::Window window, WmHints* hints) {
std::vector<uint32_t> hints32;
if (!GetArrayProperty(window, gfx::GetAtom("WM_HINTS"), &hints32))
return false;
if (hints32.size() != sizeof(WmHints) / 4)
return false;
memcpy(hints, hints32.data(), sizeof(*hints));
return true;
}
void SetWmHints(x11::Window window, const WmHints& hints) {
std::vector<uint32_t> hints32(sizeof(WmHints) / 4);
memcpy(hints32.data(), &hints, sizeof(WmHints));
ui::SetArrayProperty(window, gfx::GetAtom("WM_HINTS"),
gfx::GetAtom("WM_HINTS"), hints32);
}
void WithdrawWindow(x11::Window window) {
auto* connection = x11::Connection::Get();
connection->UnmapWindow({window});
auto root = connection->default_root();
x11::UnmapNotifyEvent event{.event = root, .window = window};
auto mask =
x11::EventMask::SubstructureNotify | x11::EventMask::SubstructureRedirect;
SendEvent(event, root, mask);
}
void RaiseWindow(x11::Window window) {
x11::Connection::Get()->ConfigureWindow(
{.window = window, .stack_mode = x11::StackMode::Above});
}
void LowerWindow(x11::Window window) {
x11::Connection::Get()->ConfigureWindow(
{.window = window, .stack_mode = x11::StackMode::Below});
}
void DefineCursor(x11::Window window, x11::Cursor cursor) {
// TODO(https://crbug.com/1066670): Sync() should be removed. It's added for
// now because Xlib's XDefineCursor() sync'ed and removing it perturbs the
// timing on BookmarkBarViewTest8.DNDBackToOriginatingMenu on
// linux-chromeos-rel, causing it to flake.
x11::Connection::Get()
->ChangeWindowAttributes({.window = window, .cursor = cursor})
.Sync();
}
x11::Window CreateDummyWindow(const std::string& name) {
auto* connection = x11::Connection::Get();
auto window = connection->GenerateId<x11::Window>();
connection->CreateWindow({
.wid = window,
.parent = connection->default_root(),
.x = -100,
.y = -100,
.width = 10,
.height = 10,
.c_class = x11::WindowClass::InputOnly,
.override_redirect = x11::Bool32(true),
});
if (!name.empty())
SetStringProperty(window, x11::Atom::WM_NAME, x11::Atom::STRING, name);
return window;
}
void DrawPixmap(x11::Connection* connection,
x11::VisualId visual,
x11::Drawable drawable,
x11::GraphicsContext gc,
const SkPixmap& skia_pixmap,
int src_x,
int src_y,
int dst_x,
int dst_y,
int width,
int height) {
const auto* visual_info = connection->GetVisualInfoFromId(visual);
if (!visual_info)
return;
auto bpp = visual_info->format->bits_per_pixel;
auto align = visual_info->format->scanline_pad;
size_t row_bits = bpp * width;
row_bits += (align - (row_bits % align)) % align;
size_t row_bytes = (row_bits + 7) / 8;
auto color_type = ColorTypeForVisual(visual);
if (color_type == kUnknown_SkColorType) {
// TODO(https://crbug.com/1066670): Add a fallback path in case any users
// are running a server that uses visual types for which Skia doesn't have
// a corresponding color format.
return;
}
SkImageInfo image_info =
SkImageInfo::Make(width, height, color_type, kPremul_SkAlphaType);
std::vector<uint8_t> vec(row_bytes * height);
SkPixmap pixmap(image_info, vec.data(), row_bytes);
skia_pixmap.readPixels(pixmap, src_x, src_y);
x11::PutImageRequest put_image_request{
.format = x11::ImageFormat::ZPixmap,
.drawable = drawable,
.gc = gc,
.width = width,
.height = height,
.dst_x = dst_x,
.dst_y = dst_y,
.left_pad = 0,
.depth = visual_info->format->depth,
.data = base::RefCountedBytes::TakeVector(&vec),
};
connection->PutImage(put_image_request);
}
bool IsXInput2Available() {
return DeviceDataManagerX11::GetInstance()->IsXInput2Available();
}
bool QueryShmSupport() {
static bool supported = x11::Connection::Get()->shm().QueryVersion({}).Sync();
return supported;
}
int CoalescePendingMotionEvents(const x11::Event* x11_event,
x11::Event* last_event) {
const auto* motion = x11_event->As<x11::MotionNotifyEvent>();
const auto* device = x11_event->As<x11::Input::DeviceEvent>();
DCHECK(motion || device);
auto* conn = x11::Connection::Get();
int num_coalesced = 0;
conn->ReadResponses();
if (motion) {
for (auto it = conn->events().begin(); it != conn->events().end();) {
const auto& next_event = *it;
// Discard all but the most recent motion event that targets the same
// window with unchanged state.
const auto* next_motion = next_event.As<x11::MotionNotifyEvent>();
if (next_motion && next_motion->event == motion->event &&
next_motion->child == motion->child &&
next_motion->state == motion->state) {
*last_event = std::move(*it);
it = conn->events().erase(it);
} else {
break;
}
}
} else {
DCHECK(device->opcode == x11::Input::DeviceEvent::Motion ||
device->opcode == x11::Input::DeviceEvent::TouchUpdate);
auto* ddmx11 = ui::DeviceDataManagerX11::GetInstance();
for (auto it = conn->events().begin(); it != conn->events().end();) {
auto* next_device = it->As<x11::Input::DeviceEvent>();
if (!next_device)
break;
// If this isn't from a valid device, throw the event away, as
// that's what the message pump would do. Device events come in pairs
// with one from the master and one from the slave so there will
// always be at least one pending.
if (!ui::TouchFactory::GetInstance()->ShouldProcessDeviceEvent(
*next_device)) {
it = conn->events().erase(it);
continue;
}
if (next_device->opcode == device->opcode &&
!ddmx11->IsCMTGestureEvent(*it) &&
ddmx11->GetScrollClassEventDetail(*it) == SCROLL_TYPE_NO_SCROLL) {
// Confirm that the motion event is targeted at the same window
// and that no buttons or modifiers have changed.
if (device->event == next_device->event &&
device->child == next_device->child &&
device->detail == next_device->detail &&
device->button_mask == next_device->button_mask &&
device->mods.base == next_device->mods.base &&
device->mods.latched == next_device->mods.latched &&
device->mods.locked == next_device->mods.locked &&
device->mods.effective == next_device->mods.effective) {
*last_event = std::move(*it);
it = conn->events().erase(it);
num_coalesced++;
continue;
}
}
break;
}
}
return num_coalesced;
}
void SetUseOSWindowFrame(x11::Window window, bool use_os_window_frame) {
// This data structure represents additional hints that we send to the window
// manager and has a direct lineage back to Motif, which defined this de facto
// standard. We define this struct to match the wire-format (32-bit fields)
// rather than the Xlib API (XChangeProperty) format (long fields).
typedef struct {
uint32_t flags;
uint32_t functions;
uint32_t decorations;
int32_t input_mode;
uint32_t status;
} MotifWmHints;
MotifWmHints motif_hints;
memset(&motif_hints, 0, sizeof(motif_hints));
// Signals that the reader of the _MOTIF_WM_HINTS property should pay
// attention to the value of |decorations|.
motif_hints.flags = (1u << 1);
motif_hints.decorations = use_os_window_frame ? 1 : 0;
std::vector<uint32_t> hints(sizeof(MotifWmHints) / sizeof(uint32_t));
memcpy(hints.data(), &motif_hints, sizeof(MotifWmHints));
x11::Atom hint_atom = gfx::GetAtom("_MOTIF_WM_HINTS");
SetArrayProperty(window, hint_atom, hint_atom, hints);
}
bool IsShapeExtensionAvailable() {
return x11::Connection::Get()->shape().present();
}
x11::Window GetX11RootWindow() {
return x11::Connection::Get()->default_screen().root;
}
bool GetCurrentDesktop(int* desktop) {
return GetIntProperty(GetX11RootWindow(), "_NET_CURRENT_DESKTOP", desktop);
}
void SetHideTitlebarWhenMaximizedProperty(x11::Window window,
HideTitlebarWhenMaximized property) {
SetProperty(window, gfx::GetAtom("_GTK_HIDE_TITLEBAR_WHEN_MAXIMIZED"),
x11::Atom::CARDINAL, static_cast<uint32_t>(property));
}
bool IsWindowVisible(x11::Window window) {
TRACE_EVENT0("ui", "IsWindowVisible");
auto x11_window = static_cast<x11::Window>(window);
auto* connection = x11::Connection::Get();
auto response = connection->GetWindowAttributes({x11_window}).Sync();
if (!response || response->map_state != x11::MapState::Viewable)
return false;
// Minimized windows are not visible.
std::vector<x11::Atom> wm_states;
if (GetAtomArrayProperty(window, "_NET_WM_STATE", &wm_states)) {
x11::Atom hidden_atom = gfx::GetAtom("_NET_WM_STATE_HIDDEN");
if (base::Contains(wm_states, hidden_atom))
return false;
}
// Some compositing window managers (notably kwin) do not actually unmap
// windows on desktop switch, so we also must check the current desktop.
int window_desktop, current_desktop;
return (!GetWindowDesktop(window, &window_desktop) ||
!GetCurrentDesktop(&current_desktop) ||
window_desktop == kAllDesktops || window_desktop == current_desktop);
}
bool GetInnerWindowBounds(x11::Window window, gfx::Rect* rect) {
auto x11_window = static_cast<x11::Window>(window);
auto root = static_cast<x11::Window>(GetX11RootWindow());
x11::Connection* connection = x11::Connection::Get();
auto get_geometry = connection->GetGeometry({x11_window});
auto translate_coords = connection->TranslateCoordinates({x11_window, root});
// Sync after making both requests so only one round-trip is made.
auto geometry = get_geometry.Sync();
auto coords = translate_coords.Sync();
if (!geometry || !coords)
return false;
*rect = gfx::Rect(coords->dst_x, coords->dst_y, geometry->width,
geometry->height);
return true;
}
bool GetWindowExtents(x11::Window window, gfx::Insets* extents) {
std::vector<int> insets;
if (!GetIntArrayProperty(window, "_NET_FRAME_EXTENTS", &insets))
return false;
if (insets.size() != 4)
return false;
int left = insets[0];
int right = insets[1];
int top = insets[2];
int bottom = insets[3];
extents->Set(-top, -left, -bottom, -right);
return true;
}
bool GetOuterWindowBounds(x11::Window window, gfx::Rect* rect) {
if (!GetInnerWindowBounds(window, rect))
return false;
gfx::Insets extents;
if (GetWindowExtents(window, &extents))
rect->Inset(extents);
// Not all window managers support _NET_FRAME_EXTENTS so return true even if
// requesting the property fails.
return true;
}
bool WindowContainsPoint(x11::Window window, gfx::Point screen_loc) {
TRACE_EVENT0("ui", "WindowContainsPoint");
gfx::Rect window_rect;
if (!GetOuterWindowBounds(window, &window_rect))
return false;
if (!window_rect.Contains(screen_loc))
return false;
if (!IsShapeExtensionAvailable())
return true;
// According to http://www.x.org/releases/X11R7.6/doc/libXext/shapelib.html,
// if an X display supports the shape extension the bounds of a window are
// defined as the intersection of the window bounds and the interior
// rectangles. This means to determine if a point is inside a window for the
// purpose of input handling we have to check the rectangles in the ShapeInput
// list.
// According to http://www.x.org/releases/current/doc/xextproto/shape.html,
// we need to also respect the ShapeBounding rectangles.
// The effective input region of a window is defined to be the intersection
// of the client input region with both the default input region and the
// client bounding region. Any portion of the client input region that is not
// included in both the default input region and the client bounding region
// will not be included in the effective input region on the screen.
x11::Shape::Sk rectangle_kind[] = {x11::Shape::Sk::Input,
x11::Shape::Sk::Bounding};
for (auto kind : rectangle_kind) {
auto shape =
x11::Connection::Get()->shape().GetRectangles({window, kind}).Sync();
if (!shape)
return true;
if (shape->rectangles.empty()) {
// The shape can be empty when |window| is minimized.
return false;
}
bool is_in_shape_rects = false;
for (const auto& rect : shape->rectangles) {
// The ShapeInput and ShapeBounding rects are to be in window space, so we
// have to translate by the window_rect's offset to map to screen space.
gfx::Rect shape_rect =
gfx::Rect(rect.x + window_rect.x(), rect.y + window_rect.y(),
rect.width, rect.height);
if (shape_rect.Contains(screen_loc)) {
is_in_shape_rects = true;
break;
}
}
if (!is_in_shape_rects)
return false;
}
return true;
}
bool PropertyExists(x11::Window window, const std::string& property_name) {
auto response = x11::Connection::Get()
->GetProperty({
.window = static_cast<x11::Window>(window),
.property = gfx::GetAtom(property_name),
.long_length = 1,
})
.Sync();
return response && response->format;
}
bool GetRawBytesOfProperty(x11::Window window,
x11::Atom property,
scoped_refptr<base::RefCountedMemory>* out_data,
x11::Atom* out_type) {
auto future = x11::Connection::Get()->GetProperty({
.window = static_cast<x11::Window>(window),
.property = property,
// Don't limit the amount of returned data.
.long_length = std::numeric_limits<uint32_t>::max(),
});
auto response = future.Sync();
if (!response || !response->format)
return false;
*out_data = response->value;
if (out_type)
*out_type = response->type;
return true;
}
bool GetIntProperty(x11::Window window,
const std::string& property_name,
int* value) {
return GetProperty(window, gfx::GetAtom(property_name), value);
}
bool GetIntArrayProperty(x11::Window window,
const std::string& property_name,
std::vector<int32_t>* value) {
return GetArrayProperty(window, gfx::GetAtom(property_name), value);
}
bool GetAtomArrayProperty(x11::Window window,
const std::string& property_name,
std::vector<x11::Atom>* value) {
return GetArrayProperty(window, gfx::GetAtom(property_name), value);
}
bool GetStringProperty(x11::Window window,
const std::string& property_name,
std::string* value) {
std::vector<char> str;
if (!GetArrayProperty(window, gfx::GetAtom(property_name), &str))
return false;
value->assign(str.data(), str.size());
return true;
}
void SetIntProperty(x11::Window window,
const std::string& name,
const std::string& type,
int32_t value) {
std::vector<int> values(1, value);
return SetIntArrayProperty(window, name, type, values);
}
void SetIntArrayProperty(x11::Window window,
const std::string& name,
const std::string& type,
const std::vector<int32_t>& value) {
SetArrayProperty(window, gfx::GetAtom(name), gfx::GetAtom(type), value);
}
void SetAtomProperty(x11::Window window,
const std::string& name,
const std::string& type,
x11::Atom value) {
std::vector<x11::Atom> values(1, value);
return SetAtomArrayProperty(window, name, type, values);
}
void SetAtomArrayProperty(x11::Window window,
const std::string& name,
const std::string& type,
const std::vector<x11::Atom>& value) {
SetArrayProperty(window, gfx::GetAtom(name), gfx::GetAtom(type), value);
}
void SetStringProperty(x11::Window window,
x11::Atom property,
x11::Atom type,
const std::string& value) {
std::vector<char> str(value.begin(), value.end());
SetArrayProperty(window, property, type, str);
}
void SetWindowClassHint(x11::Connection* connection,
x11::Window window,
const std::string& res_name,
const std::string& res_class) {
auto str =
base::StringPrintf("%s%c%s", res_name.c_str(), '\0', res_class.c_str());
std::vector<char> data(str.data(), str.data() + str.size() + 1);
SetArrayProperty(window, x11::Atom::WM_CLASS, x11::Atom::STRING, data);
}
void SetWindowRole(x11::Window window, const std::string& role) {
x11::Atom prop = gfx::GetAtom("WM_WINDOW_ROLE");
if (role.empty())
DeleteProperty(window, prop);
else
SetStringProperty(window, prop, x11::Atom::STRING, role);
}
void SetWMSpecState(x11::Window window,
bool enabled,
x11::Atom state1,
x11::Atom state2) {
SendClientMessage(
window, GetX11RootWindow(), gfx::GetAtom("_NET_WM_STATE"),
{enabled ? kNetWMStateAdd : kNetWMStateRemove,
static_cast<uint32_t>(state1), static_cast<uint32_t>(state2), 1, 0});
}
void DoWMMoveResize(x11::Connection* connection,
x11::Window root_window,
x11::Window window,
const gfx::Point& location_px,
int direction) {
// This handler is usually sent when the window has the implicit grab. We
// need to dump it because what we're about to do is tell the window manager
// that it's now responsible for moving the window around; it immediately
// grabs when it receives the event below.
connection->UngrabPointer({x11::Time::CurrentTime});
SendClientMessage(window, root_window, gfx::GetAtom("_NET_WM_MOVERESIZE"),
{location_px.x(), location_px.y(), direction, 0, 0});
}
bool HasWMSpecProperty(const base::flat_set<x11::Atom>& properties,
x11::Atom atom) {
return properties.find(atom) != properties.end();
}
bool GetCustomFramePrefDefault() {
// If the window manager doesn't support enough of EWMH to tell us its name,
// assume that it doesn't want custom frames. For example, _NET_WM_MOVERESIZE
// is needed for frame-drag-initiated window movement.
std::string wm_name;
if (!GetWindowManagerName(&wm_name))
return false;
// Also disable custom frames for (at-least-partially-)EWMH-supporting tiling
// window managers.
ui::WindowManagerName wm = GuessWindowManager();
if (wm == WM_AWESOME || wm == WM_I3 || wm == WM_ION3 || wm == WM_MATCHBOX ||
wm == WM_NOTION || wm == WM_QTILE || wm == WM_RATPOISON ||
wm == WM_STUMPWM || wm == WM_WMII)
return false;
// Handle a few more window managers that don't get along well with custom
// frames.
if (wm == WM_ICE_WM || wm == WM_KWIN)
return false;
// For everything else, use custom frames.
return true;
}
bool IsWmTiling(WindowManagerName window_manager) {
switch (window_manager) {
case WM_BLACKBOX:
case WM_COMPIZ:
case WM_ENLIGHTENMENT:
case WM_FLUXBOX:
case WM_ICE_WM:
case WM_KWIN:
case WM_MATCHBOX:
case WM_METACITY:
case WM_MUFFIN:
case WM_MUTTER:
case WM_OPENBOX:
case WM_XFWM4:
// Stacking window managers.
return false;
case WM_I3:
case WM_ION3:
case WM_NOTION:
case WM_RATPOISON:
case WM_STUMPWM:
// Tiling window managers.
return true;
case WM_AWESOME:
case WM_QTILE:
case WM_XMONAD:
case WM_WMII:
// Dynamic (tiling and stacking) window managers. Assume tiling.
return true;
case WM_OTHER:
case WM_UNNAMED:
// Unknown. Assume stacking.
return false;
}
}
bool GetWindowDesktop(x11::Window window, int* desktop) {
return GetIntProperty(window, "_NET_WM_DESKTOP", desktop);
}
std::string GetX11ErrorString(XDisplay* display, int err) {
char buffer[256];
XGetErrorText(display, err, buffer, base::size(buffer));
return buffer;
}
// Returns true if |window| is a named window.
bool IsWindowNamed(x11::Window window) {
return PropertyExists(window, "WM_NAME");
}
bool EnumerateChildren(EnumerateWindowsDelegate* delegate,
x11::Window window,
const int max_depth,
int depth) {
if (depth > max_depth)
return false;
std::vector<x11::Window> windows;
std::vector<x11::Window>::iterator iter;
if (depth == 0) {
XMenuList::GetInstance()->InsertMenuWindows(&windows);
// Enumerate the menus first.
for (iter = windows.begin(); iter != windows.end(); iter++) {
if (delegate->ShouldStopIterating(*iter))
return true;
}
windows.clear();
}
auto query_tree = x11::Connection::Get()->QueryTree({window}).Sync();
if (!query_tree)
return false;
windows = std::move(query_tree->children);
// XQueryTree returns the children of |window| in bottom-to-top order, so
// reverse-iterate the list to check the windows from top-to-bottom.
for (iter = windows.begin(); iter != windows.end(); iter++) {
if (IsWindowNamed(*iter) && delegate->ShouldStopIterating(*iter))
return true;
}
// If we're at this point, we didn't find the window we're looking for at the
// current level, so we need to recurse to the next level. We use a second
// loop because the recursion and call to XQueryTree are expensive and is only
// needed for a small number of cases.
if (++depth <= max_depth) {
for (iter = windows.begin(); iter != windows.end(); iter++) {
if (EnumerateChildren(delegate, *iter, max_depth, depth))
return true;
}
}
return false;
}
bool EnumerateAllWindows(EnumerateWindowsDelegate* delegate, int max_depth) {
x11::Window root = GetX11RootWindow();
return EnumerateChildren(delegate, root, max_depth, 0);
}
void EnumerateTopLevelWindows(ui::EnumerateWindowsDelegate* delegate) {
std::vector<x11::Window> stack;
if (!ui::GetXWindowStack(ui::GetX11RootWindow(), &stack)) {
// Window Manager doesn't support _NET_CLIENT_LIST_STACKING, so fall back
// to old school enumeration of all X windows. Some WMs parent 'top-level'
// windows in unnamed actual top-level windows (ion WM), so extend the
// search depth to all children of top-level windows.
const int kMaxSearchDepth = 1;
ui::EnumerateAllWindows(delegate, kMaxSearchDepth);
return;
}
XMenuList::GetInstance()->InsertMenuWindows(&stack);
std::vector<x11::Window>::iterator iter;
for (iter = stack.begin(); iter != stack.end(); iter++) {
if (delegate->ShouldStopIterating(*iter))
return;
}
}
bool GetXWindowStack(x11::Window window, std::vector<x11::Window>* windows) {
if (!GetArrayProperty(window, gfx::GetAtom("_NET_CLIENT_LIST_STACKING"),
windows)) {
return false;
}
// It's more common to iterate from lowest window to highest,
// so reverse the vector.
std::reverse(windows->begin(), windows->end());
return true;
}
WindowManagerName GuessWindowManager() {
std::string name;
if (!GetWindowManagerName(&name))
return WM_UNNAMED;
// These names are taken from the WMs' source code.
if (name == "awesome")
return WM_AWESOME;
if (name == "Blackbox")
return WM_BLACKBOX;
if (name == "Compiz" || name == "compiz")
return WM_COMPIZ;
if (name == "e16" || name == "Enlightenment")
return WM_ENLIGHTENMENT;
if (name == "Fluxbox")
return WM_FLUXBOX;
if (name == "i3")
return WM_I3;
if (base::StartsWith(name, "IceWM", base::CompareCase::SENSITIVE))
return WM_ICE_WM;
if (name == "ion3")
return WM_ION3;
if (name == "KWin")
return WM_KWIN;
if (name == "matchbox")
return WM_MATCHBOX;
if (name == "Metacity")
return WM_METACITY;
if (name == "Mutter (Muffin)")
return WM_MUFFIN;
if (name == "GNOME Shell")
return WM_MUTTER; // GNOME Shell uses Mutter
if (name == "Mutter")
return WM_MUTTER;
if (name == "notion")
return WM_NOTION;
if (name == "Openbox")
return WM_OPENBOX;
if (name == "qtile")
return WM_QTILE;
if (name == "ratpoison")
return WM_RATPOISON;
if (name == "stumpwm")
return WM_STUMPWM;
if (name == "wmii")
return WM_WMII;
if (name == "Xfwm4")
return WM_XFWM4;
if (name == "xmonad")
return WM_XMONAD;
return WM_OTHER;
}
std::string GuessWindowManagerName() {
std::string name;
if (GetWindowManagerName(&name))
return name;
return "Unknown";
}
UMALinuxWindowManager GetWindowManagerUMA() {
switch (GuessWindowManager()) {
case WM_OTHER:
return UMALinuxWindowManager::kOther;
case WM_UNNAMED:
return UMALinuxWindowManager::kUnnamed;
case WM_AWESOME:
return UMALinuxWindowManager::kAwesome;
case WM_BLACKBOX:
return UMALinuxWindowManager::kBlackbox;
case WM_COMPIZ:
return UMALinuxWindowManager::kCompiz;
case WM_ENLIGHTENMENT:
return UMALinuxWindowManager::kEnlightenment;
case WM_FLUXBOX:
return UMALinuxWindowManager::kFluxbox;
case WM_I3:
return UMALinuxWindowManager::kI3;
case WM_ICE_WM:
return UMALinuxWindowManager::kIceWM;
case WM_ION3:
return UMALinuxWindowManager::kIon3;
case WM_KWIN:
return UMALinuxWindowManager::kKWin;
case WM_MATCHBOX:
return UMALinuxWindowManager::kMatchbox;
case WM_METACITY:
return UMALinuxWindowManager::kMetacity;
case WM_MUFFIN:
return UMALinuxWindowManager::kMuffin;
case WM_MUTTER:
return UMALinuxWindowManager::kMutter;
case WM_NOTION:
return UMALinuxWindowManager::kNotion;
case WM_OPENBOX:
return UMALinuxWindowManager::kOpenbox;
case WM_QTILE:
return UMALinuxWindowManager::kQtile;
case WM_RATPOISON:
return UMALinuxWindowManager::kRatpoison;
case WM_STUMPWM:
return UMALinuxWindowManager::kStumpWM;
case WM_WMII:
return UMALinuxWindowManager::kWmii;
case WM_XFWM4:
return UMALinuxWindowManager::kXfwm4;
case WM_XMONAD:
return UMALinuxWindowManager::kXmonad;
}
NOTREACHED();
return UMALinuxWindowManager::kOther;
}
bool IsCompositingManagerPresent() {
auto is_compositing_manager_present_impl = []() {
auto response = x11::Connection::Get()
->GetSelectionOwner({gfx::GetAtom("_NET_WM_CM_S0")})
.Sync();
return response && response->owner != x11::Window::None;
};
static bool is_compositing_manager_present =
is_compositing_manager_present_impl();
return is_compositing_manager_present;
}
void SetDefaultX11ErrorHandlers() {
SetX11ErrorHandlers(nullptr, nullptr);
}
bool IsX11WindowFullScreen(x11::Window window) {
// If _NET_WM_STATE_FULLSCREEN is in _NET_SUPPORTED, use the presence or
// absence of _NET_WM_STATE_FULLSCREEN in _NET_WM_STATE to determine
// whether we're fullscreen.
x11::Atom fullscreen_atom = gfx::GetAtom("_NET_WM_STATE_FULLSCREEN");
if (WmSupportsHint(fullscreen_atom)) {
std::vector<x11::Atom> atom_properties;
if (GetAtomArrayProperty(window, "_NET_WM_STATE", &atom_properties)) {
return base::Contains(atom_properties, fullscreen_atom);
}
}
gfx::Rect window_rect;
if (!ui::GetOuterWindowBounds(window, &window_rect))
return false;
// TODO(thomasanderson): We should use
// display::Screen::GetDisplayNearestWindow() instead of using the
// connection screen size, which encompasses all displays.
auto* connection = x11::Connection::Get();
int width = connection->default_screen().width_in_pixels;
int height = connection->default_screen().height_in_pixels;
return window_rect.size() == gfx::Size(width, height);
}
bool WmSupportsHint(x11::Atom atom) {
if (!SupportsEWMH())
return false;
std::vector<x11::Atom> supported_atoms;
if (!GetAtomArrayProperty(GetX11RootWindow(), "_NET_SUPPORTED",
&supported_atoms)) {
return false;
}
return base::Contains(supported_atoms, atom);
}
gfx::ICCProfile GetICCProfileForMonitor(int monitor) {
gfx::ICCProfile icc_profile;
if (base::CommandLine::ForCurrentProcess()->HasSwitch(switches::kHeadless))
return icc_profile;
std::string atom_name = monitor == 0
? "_ICC_PROFILE"
: base::StringPrintf("_ICC_PROFILE_%d", monitor);
scoped_refptr<base::RefCountedMemory> data;
if (GetRawBytesOfProperty(GetX11RootWindow(), gfx::GetAtom(atom_name), &data,
nullptr)) {
icc_profile = gfx::ICCProfile::FromData(data->data(), data->size());
}
return icc_profile;
}
bool IsSyncExtensionAvailable() {
// Chrome for ChromeOS can be run with X11 on a Linux desktop. In this case,
// NotifySwapAfterResize is never called as the compositor does not notify about
// swaps after resize. Thus, simply disable usage of XSyncCounter on ChromeOS
// builds.
//
// TODO(https://crbug.com/1036285): Also, disable sync extension for all ozone
// builds as long as our EGL impl for Ozone/X11 is not mature enough and we do
// not receive swap completions on time, which results in weird resize behaviour
// as X Server waits for the XSyncCounter changes.
#if defined(OS_CHROMEOS) || defined(USE_OZONE)
return false;
#else
static bool result =
x11::Connection::Get()
->sync()
.Initialize({x11::Sync::major_version, x11::Sync::minor_version})
.Sync();
return result;
#endif
}
SkColorType ColorTypeForVisual(x11::VisualId visual) {
struct {
SkColorType color_type;
unsigned long red_mask;
unsigned long green_mask;
unsigned long blue_mask;
int bpp;
} color_infos[] = {
{kRGB_565_SkColorType, 0xf800, 0x7e0, 0x1f, 16},
{kARGB_4444_SkColorType, 0xf000, 0xf00, 0xf0, 16},
{kRGBA_8888_SkColorType, 0xff, 0xff00, 0xff0000, 32},
{kBGRA_8888_SkColorType, 0xff0000, 0xff00, 0xff, 32},
{kRGBA_1010102_SkColorType, 0x3ff, 0xffc00, 0x3ff00000, 32},
{kBGRA_1010102_SkColorType, 0x3ff00000, 0xffc00, 0x3ff, 32},
};
auto* connection = x11::Connection::Get();
const auto* vis = connection->GetVisualInfoFromId(visual);
if (!vis)
return kUnknown_SkColorType;
// We don't currently support anything other than TrueColor and DirectColor.
if (!vis->visual_type->red_mask || !vis->visual_type->green_mask ||
!vis->visual_type->blue_mask) {
return kUnknown_SkColorType;
}
for (const auto& color_info : color_infos) {
if (vis->visual_type->red_mask == color_info.red_mask &&
vis->visual_type->green_mask == color_info.green_mask &&
vis->visual_type->blue_mask == color_info.blue_mask &&
vis->format->bits_per_pixel == color_info.bpp) {
return color_info.color_type;
}
}
LOG(ERROR) << "Unsupported visual with rgb mask 0x" << std::hex
<< vis->visual_type->red_mask << ", 0x"
<< vis->visual_type->green_mask << ", 0x"
<< vis->visual_type->blue_mask
<< ". Please report this to https://crbug.com/1025266";
return kUnknown_SkColorType;
}
x11::Future<void> SendClientMessage(x11::Window window,
x11::Window target,
x11::Atom type,
const std::array<uint32_t, 5> data,
x11::EventMask event_mask) {
x11::ClientMessageEvent event{.format = 32, .window = window, .type = type};
event.data.data32 = data;
return SendEvent(event, target, event_mask);
}
XRefcountedMemory::XRefcountedMemory(unsigned char* x11_data, size_t length)
: x11_data_(length ? x11_data : nullptr), length_(length) {}
const unsigned char* XRefcountedMemory::front() const {
return x11_data_.get();
}
size_t XRefcountedMemory::size() const {
return length_;
}
XRefcountedMemory::~XRefcountedMemory() = default;
void XImageDeleter::operator()(XImage* image) const {
XDestroyImage(image);
}
void SetX11ErrorHandlers(XErrorHandler error_handler,
XIOErrorHandler io_error_handler) {
XSetErrorHandler(error_handler ? error_handler : DefaultX11ErrorHandler);
XSetIOErrorHandler(io_error_handler ? io_error_handler
: DefaultX11IOErrorHandler);
}
// static
XVisualManager* XVisualManager::GetInstance() {
return base::Singleton<XVisualManager>::get();
}
XVisualManager::XVisualManager() : connection_(x11::Connection::Get()) {
base::AutoLock lock(lock_);
for (const auto& depth : connection_->default_screen().allowed_depths) {
for (const auto& visual : depth.visuals) {
visuals_[visual.visual_id] =
std::make_unique<XVisualData>(connection_, depth.depth, &visual);
}
}
// Choose the opaque visual.
default_visual_id_ = connection_->default_screen().root_visual;
system_visual_id_ = default_visual_id_;
DCHECK_NE(system_visual_id_, x11::VisualId{});
DCHECK(visuals_.find(system_visual_id_) != visuals_.end());
// Choose the transparent visual.
for (const auto& pair : visuals_) {
// Why support only 8888 ARGB? Because it's all that GTK+ supports. In
// gdkvisual-x11.cc, they look for this specific visual and use it for
// all their alpha channel using needs.
const auto& data = *pair.second;
if (data.depth == 32 && data.info->red_mask == 0xff0000 &&
data.info->green_mask == 0x00ff00 && data.info->blue_mask == 0x0000ff) {
transparent_visual_id_ = pair.first;
break;
}
}
if (transparent_visual_id_ != x11::VisualId{})
DCHECK(visuals_.find(transparent_visual_id_) != visuals_.end());
}
XVisualManager::~XVisualManager() = default;
void XVisualManager::ChooseVisualForWindow(bool want_argb_visual,
x11::VisualId* visual_id,
uint8_t* depth,
x11::ColorMap* colormap,
bool* visual_has_alpha) {
base::AutoLock lock(lock_);
bool use_argb = want_argb_visual && IsCompositingManagerPresent() &&
(using_software_rendering_ || have_gpu_argb_visual_);
x11::VisualId visual = use_argb && transparent_visual_id_ != x11::VisualId{}
? transparent_visual_id_
: system_visual_id_;
if (visual_id)
*visual_id = visual;
bool success = GetVisualInfoImpl(visual, depth, colormap, visual_has_alpha);
DCHECK(success);
}
bool XVisualManager::GetVisualInfo(x11::VisualId visual_id,
uint8_t* depth,
x11::ColorMap* colormap,
bool* visual_has_alpha) {
base::AutoLock lock(lock_);
return GetVisualInfoImpl(visual_id, depth, colormap, visual_has_alpha);
}
bool XVisualManager::OnGPUInfoChanged(bool software_rendering,
x11::VisualId system_visual_id,
x11::VisualId transparent_visual_id) {
base::AutoLock lock(lock_);
// TODO(thomasanderson): Cache these visual IDs as a property of the root
// window so that newly created browser processes can get them immediately.
if ((system_visual_id != x11::VisualId{} &&
!visuals_.count(system_visual_id)) ||
(transparent_visual_id != x11::VisualId{} &&
!visuals_.count(transparent_visual_id)))
return false;
using_software_rendering_ = software_rendering;
have_gpu_argb_visual_ =
have_gpu_argb_visual_ || transparent_visual_id != x11::VisualId{};
if (system_visual_id != x11::VisualId{})
system_visual_id_ = system_visual_id;
if (transparent_visual_id != x11::VisualId{})
transparent_visual_id_ = transparent_visual_id;
return true;
}
bool XVisualManager::ArgbVisualAvailable() const {
base::AutoLock lock(lock_);
return IsCompositingManagerPresent() &&
(using_software_rendering_ || have_gpu_argb_visual_);
}
bool XVisualManager::GetVisualInfoImpl(x11::VisualId visual_id,
uint8_t* depth,
x11::ColorMap* colormap,
bool* visual_has_alpha) {
auto it = visuals_.find(visual_id);
if (it == visuals_.end())
return false;
XVisualData& data = *it->second;
const x11::VisualType& info = *data.info;
bool is_default_visual = visual_id == default_visual_id_;
if (depth)
*depth = data.depth;
if (colormap)
*colormap = is_default_visual ? x11::ColorMap{} : data.GetColormap();
if (visual_has_alpha) {
auto popcount = [](auto x) {
return std::bitset<8 * sizeof(decltype(x))>(x).count();
};
*visual_has_alpha = popcount(info.red_mask) + popcount(info.green_mask) +
popcount(info.blue_mask) <
static_cast<std::size_t>(data.depth);
}
return true;
}
XVisualManager::XVisualData::XVisualData(x11::Connection* connection,
uint8_t depth,
const x11::VisualType* info)
: depth(depth), info(info), connection_(connection) {}
// Do not free the colormap as this would uninstall the colormap even for
// non-Chromium clients.
XVisualManager::XVisualData::~XVisualData() = default;
x11::ColorMap XVisualManager::XVisualData::GetColormap() {
if (colormap_ == x11::ColorMap{}) {
colormap_ = connection_->GenerateId<x11::ColorMap>();
connection_->CreateColormap({x11::ColormapAlloc::None, colormap_,
connection_->default_root(), info->visual_id});
}
return colormap_;
}
} // namespace ui