# HG changeset patch # User Paper # Date 1705436549 18000 # Node ID 593108b3d55557566e4bb9c5ef4f939aff523877 # Parent 8ccf0302afb1ab9a5e02907ca0413fbfda7f9aa7 dep/animia: x11: finalize xcb conversion diff -r 8ccf0302afb1 -r 593108b3d555 dep/animia/src/win/x11.cc --- a/dep/animia/src/win/x11.cc Tue Jan 16 08:08:42 2024 -0500 +++ b/dep/animia/src/win/x11.cc Tue Jan 16 15:22:29 2024 -0500 @@ -1,3 +1,11 @@ +/* + * win/x11.cc: provides support for X11 clients via XCB + * + * This code is fairly fast (I think...). As a result, + * a lot of it is hard to read if you're unfamiliar with + * asynchronous programming, but it works much better than + * Xlib (which can take much longer). +*/ #include "animia/win/x11.h" #include "animia/win.h" #include "animia.h" @@ -11,8 +19,6 @@ #include #include -#include - static size_t str_nlen(const char* s, size_t len) { size_t i = 0; for (; i < len && s[i]; i++); @@ -21,32 +27,6 @@ namespace animia::internal::x11 { -static void GetWindowPID(xcb_connection_t* connection, const std::vector& windows, std::unordered_map& result) { - std::vector cookies; - cookies.reserve(windows.size()); - - for (const auto& window : windows) { - xcb_res_client_id_spec_t spec = { - .client = window, - .mask = XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID - }; - - cookies.push_back(::xcb_res_query_client_ids(connection, 1, &spec)); - } - - for (size_t i = 0; i < cookies.size(); i++) { - xcb_res_query_client_ids_reply_t* reply = ::xcb_res_query_client_ids_reply(connection, cookies.at(i), NULL); - - xcb_res_client_id_value_iterator_t it = xcb_res_query_client_ids_ids_iterator(reply); - for (; it.rem; xcb_res_client_id_value_next(&it)) { - if (it.data->spec.mask & XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID) { - result[windows.at(i)] = *::xcb_res_client_id_value_value(it.data); - continue; - } - } - } -} - static bool GetAllTopLevelWindowsEWMH(xcb_connection_t* connection, const std::vector& roots, std::set& result) { const xcb_atom_t Atom__NET_CLIENT_LIST = [connection]{ static constexpr std::string_view name = "_NET_CLIENT_LIST"; @@ -70,23 +50,25 @@ for (const auto& cookie : cookies) { xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, cookie, NULL); - if (reply) { + + if (reply && reply->length == 32) { xcb_window_t* value = reinterpret_cast(::xcb_get_property_value(reply)); int len = ::xcb_get_property_value_length(reply); for (size_t i = 0; i < len; i++) result.insert(value[i]); + success = true; } + free(reply); } return success; } -/* I have no idea why this works. */ static bool WalkWindows(xcb_connection_t* connection, const std::vector& roots, std::set& result) { - /* move this somewhere */ + /* move this somewhere else pl0x */ xcb_atom_t Atom_WM_STATE = [connection]{ static constexpr std::string_view name = "WM_STATE"; xcb_intern_atom_cookie_t cookie = ::xcb_intern_atom(connection, true, name.size(), name.data()); @@ -99,49 +81,98 @@ if (Atom_WM_STATE == XCB_ATOM_NONE) return false; + /* for each toplevel: search recursively for */ + std::function find_wm_state_window = [&](const xcb_window_t* wins, const int wins_len, xcb_window_t& out) { + /* Check for wm_state */ + { + std::vector property_cookies; + property_cookies.reserve(wins_len); + + for (size_t j = 0; j < wins_len; j++) + property_cookies.push_back(::xcb_get_property(connection, 0, wins[j], Atom_WM_STATE, Atom_WM_STATE, 0, 0)); + + for (size_t j = 0; j < property_cookies.size(); j++) { + xcb_generic_error_t* err = NULL; + xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, property_cookies.at(j), &err); + + if (reply->format || reply->type || reply->length) { + out = wins[j]; + free(reply); + return true; + } + + free(reply); + } + } + + /* Query tree for recursion */ + { + std::vector cookies; + cookies.reserve(wins_len); + + for (size_t j = 0; j < wins_len; j++) + cookies.push_back(::xcb_query_tree(connection, wins[j])); + + for (const auto& cookie : cookies) { + xcb_query_tree_reply_t* reply = ::xcb_query_tree_reply(connection, cookie, NULL); + + xcb_window_t* windows = ::xcb_query_tree_children(reply); + int len = ::xcb_query_tree_children_length(reply); + + if (find_wm_state_window(windows, len, out)) { + free(reply); + return true; + } + + free(reply); + } + } + + return false; + }; + + /* Get the tree for each root */ std::vector cookies; cookies.reserve(roots.size()); for (const auto& root : roots) cookies.push_back(::xcb_query_tree(connection, root)); - for (const auto& cookie : cookies) { - xcb_query_tree_reply_t* reply = ::xcb_query_tree_reply(connection, cookie, NULL); + for (size_t i = 0; i < cookies.size(); i++) { + xcb_query_tree_reply_t* reply = ::xcb_query_tree_reply(connection, cookies.at(i), NULL); - std::vector windows = [reply]{ - xcb_window_t* windows = ::xcb_query_tree_children(reply); - int len = ::xcb_query_tree_children_length(reply); + xcb_window_t* windows = ::xcb_query_tree_children(reply); + int len = ::xcb_query_tree_children_length(reply); + if (len < 1) + continue; - std::vector w; - w.reserve(len); - - for (int i = 0; i < len; i++) - w.push_back(windows[i]); + /* Then get the tree of each child window. */ + std::vector cookies; + cookies.reserve(len); - return w; - }(); - - std::vector state_property_cookies; - state_property_cookies.reserve(windows.size()); + for (size_t j = 0; j < len; j++) + cookies.push_back(::xcb_query_tree(connection, windows[j])); - for (int i = 0; i < windows.size(); i++) - state_property_cookies.push_back(::xcb_get_property(connection, 0, windows[i], Atom_WM_STATE, Atom_WM_STATE, 0, 0)); + /* For each child window... */ + for (size_t j = 0; j < cookies.size(); j++) { + xcb_query_tree_reply_t* reply = ::xcb_query_tree_reply(connection, cookies.at(j), NULL); - for (size_t i = 0; i < state_property_cookies.size(); i++) { - xcb_generic_error_t* err = NULL; - xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, state_property_cookies.at(i), &err); - - if (reply->format || reply->type || reply->length) { - result.insert(windows[i]); - free(reply); + xcb_window_t* children = ::xcb_query_tree_children(reply); + int children_len = ::xcb_query_tree_children_length(reply); + if (children_len < 1) { + result.insert(windows[j]); continue; } - free(reply); + xcb_window_t out = windows[j]; + /* Search recursively for a window with WM_STATE. If we don't, + * just add the toplevel. + */ + find_wm_state_window(children, children_len, out); + result.insert(out); } - if (WalkWindows(connection, windows, result)) - continue; + free(reply); } return false; @@ -170,12 +201,12 @@ roots.push_back(screen.root); std::set windows; - //if (!GetAllTopLevelWindowsEWMH(connection, roots, windows)) + if (!GetAllTopLevelWindowsEWMH(connection, roots, windows)) WalkWindows(connection, roots, windows); std::vector class_property_cookies; std::vector name_property_cookies; - std::vector pid_property_cookies; + std::vector pid_property_cookies; class_property_cookies.reserve(windows.size()); name_property_cookies.reserve(windows.size()); pid_property_cookies.reserve(windows.size()); @@ -183,7 +214,13 @@ for (const auto& window : windows) { class_property_cookies.push_back(::xcb_get_property(connection, 0, window, XCB_ATOM_WM_CLASS, XCB_ATOM_STRING, 0L, 2048L)); name_property_cookies.push_back(::xcb_get_property(connection, 0, window, XCB_ATOM_WM_NAME, XCB_ATOM_STRING, 0L, UINT_MAX)); - pid_property_cookies.push_back(::xcb_get_property(connection, 0, window, XCB_ATOM_WM_NAME, XCB_ATOM_CARDINAL, 0L, 1L)); + + xcb_res_client_id_spec_t spec = { + .client = window, + .mask = XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID + }; + + pid_property_cookies.push_back(::xcb_res_query_client_ids(connection, 1, &spec)); } size_t i = 0; @@ -192,6 +229,7 @@ win.id = window; { xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, class_property_cookies.at(i), NULL); + if (reply && reply->format == 8) { const char* data = reinterpret_cast(::xcb_get_property_value(reply)); const int data_len = ::xcb_get_property_value_length(reply); @@ -201,27 +239,38 @@ win.class_name = std::string(class_name, str_nlen(class_name, data_len - (instance_len + 1))); } + free(reply); } { xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, name_property_cookies.at(i), NULL); - if (reply) { + + if (reply && reply->format == 8) { const char* data = reinterpret_cast(::xcb_get_property_value(reply)); int len = ::xcb_get_property_value_length(reply); - win.text = std::string((char*)data, len); + win.text = std::string(data, len); } - free(reply); - } - std::cout << win.class_name << ": " << win.text << std::endl; - Process proc = {0}; - { - xcb_get_property_reply_t* reply = ::xcb_get_property_reply(connection, pid_property_cookies.at(i), NULL); - if (reply) - proc.pid = *reinterpret_cast(::xcb_get_property_value(reply)); free(reply); } + Process proc = {0}; + { + xcb_res_query_client_ids_reply_t* reply = ::xcb_res_query_client_ids_reply(connection, pid_property_cookies.at(i), NULL); + + if (reply->length) { + xcb_res_client_id_value_iterator_t it = ::xcb_res_query_client_ids_ids_iterator(reply); + for (; it.rem; ::xcb_res_client_id_value_next(&it)) { + if (it.data->spec.mask & XCB_RES_CLIENT_ID_MASK_LOCAL_CLIENT_PID) { + proc.pid = *::xcb_res_client_id_value_value(it.data); + break; + } + } + } + + free(reply); + } + if (!window_proc(proc, win)) { ::xcb_disconnect(connection); return false;