/* * Copyright (C) 2016 Google, Inc. * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */ #include <cassert> #include <sstream> #include <dlfcn.h> #include <time.h> #include "Helpers.h" #include "Game.h" #include "ShellXcb.h" namespace { class PosixTimer { public: PosixTimer() { reset(); } void reset() { clock_gettime(CLOCK_MONOTONIC, &start_); } double get() const { struct timespec now; clock_gettime(CLOCK_MONOTONIC, &now); constexpr long one_s_in_ns = 1000 * 1000 * 1000; constexpr double one_s_in_ns_d = static_cast<double>(one_s_in_ns); time_t s = now.tv_sec - start_.tv_sec; long ns; if (now.tv_nsec > start_.tv_nsec) { ns = now.tv_nsec - start_.tv_nsec; } else { assert(s > 0); s--; ns = one_s_in_ns - (start_.tv_nsec - now.tv_nsec); } return static_cast<double>(s) + static_cast<double>(ns) / one_s_in_ns_d; } private: struct timespec start_; }; xcb_intern_atom_cookie_t intern_atom_cookie(xcb_connection_t *c, const std::string &s) { return xcb_intern_atom(c, false, s.size(), s.c_str()); } xcb_atom_t intern_atom(xcb_connection_t *c, xcb_intern_atom_cookie_t cookie) { xcb_atom_t atom = XCB_ATOM_NONE; xcb_intern_atom_reply_t *reply = xcb_intern_atom_reply(c, cookie, nullptr); if (reply) { atom = reply->atom; free(reply); } return atom; } } // namespace ShellXcb::ShellXcb(Game &game) : Shell(game) { instance_extensions_.push_back(VK_KHR_XCB_SURFACE_EXTENSION_NAME); init_connection(); init_vk(); } ShellXcb::~ShellXcb() { cleanup_vk(); dlclose(lib_handle_); xcb_disconnect(c_); } void ShellXcb::init_connection() { int scr; c_ = xcb_connect(nullptr, &scr); if (!c_ || xcb_connection_has_error(c_)) { xcb_disconnect(c_); throw std::runtime_error("failed to connect to the display server"); } const xcb_setup_t *setup = xcb_get_setup(c_); xcb_screen_iterator_t iter = xcb_setup_roots_iterator(setup); while (scr-- > 0) xcb_screen_next(&iter); scr_ = iter.data; } void ShellXcb::create_window() { win_ = xcb_generate_id(c_); uint32_t value_mask, value_list[32]; value_mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK; value_list[0] = scr_->black_pixel; value_list[1] = XCB_EVENT_MASK_KEY_PRESS | XCB_EVENT_MASK_STRUCTURE_NOTIFY; xcb_create_window(c_, XCB_COPY_FROM_PARENT, win_, scr_->root, 0, 0, settings_.initial_width, settings_.initial_height, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT, scr_->root_visual, value_mask, value_list); xcb_intern_atom_cookie_t utf8_string_cookie = intern_atom_cookie(c_, "UTF8_STRING"); xcb_intern_atom_cookie_t _net_wm_name_cookie = intern_atom_cookie(c_, "_NET_WM_NAME"); xcb_intern_atom_cookie_t wm_protocols_cookie = intern_atom_cookie(c_, "WM_PROTOCOLS"); xcb_intern_atom_cookie_t wm_delete_window_cookie = intern_atom_cookie(c_, "WM_DELETE_WINDOW"); // set title xcb_atom_t utf8_string = intern_atom(c_, utf8_string_cookie); xcb_atom_t _net_wm_name = intern_atom(c_, _net_wm_name_cookie); xcb_change_property(c_, XCB_PROP_MODE_REPLACE, win_, _net_wm_name, utf8_string, 8, settings_.name.size(), settings_.name.c_str()); // advertise WM_DELETE_WINDOW wm_protocols_ = intern_atom(c_, wm_protocols_cookie); wm_delete_window_ = intern_atom(c_, wm_delete_window_cookie); xcb_change_property(c_, XCB_PROP_MODE_REPLACE, win_, wm_protocols_, XCB_ATOM_ATOM, 32, 1, &wm_delete_window_); } PFN_vkGetInstanceProcAddr ShellXcb::load_vk() { const char filename[] = "libvulkan.so"; void *handle, *symbol; #ifdef UNINSTALLED_LOADER handle = dlopen(UNINSTALLED_LOADER, RTLD_LAZY); if (!handle) handle = dlopen(filename, RTLD_LAZY); #else handle = dlopen(filename, RTLD_LAZY); #endif if (handle) symbol = dlsym(handle, "vkGetInstanceProcAddr"); if (!handle || !symbol) { std::stringstream ss; ss << "failed to load " << dlerror(); if (handle) dlclose(handle); throw std::runtime_error(ss.str()); } lib_handle_ = handle; return reinterpret_cast<PFN_vkGetInstanceProcAddr>(symbol); } bool ShellXcb::can_present(VkPhysicalDevice phy, uint32_t queue_family) { return vk::GetPhysicalDeviceXcbPresentationSupportKHR(phy, queue_family, c_, scr_->root_visual); } VkSurfaceKHR ShellXcb::create_surface(VkInstance instance) { VkXcbSurfaceCreateInfoKHR surface_info = {}; surface_info.sType = VK_STRUCTURE_TYPE_XCB_SURFACE_CREATE_INFO_KHR; surface_info.connection = c_; surface_info.window = win_; VkSurfaceKHR surface; vk::assert_success(vk::CreateXcbSurfaceKHR(instance, &surface_info, nullptr, &surface)); return surface; } void ShellXcb::handle_event(const xcb_generic_event_t *ev) { switch (ev->response_type & 0x7f) { case XCB_CONFIGURE_NOTIFY: { const xcb_configure_notify_event_t *notify = reinterpret_cast<const xcb_configure_notify_event_t *>(ev); resize_swapchain(notify->width, notify->height); } break; case XCB_KEY_PRESS: { const xcb_key_press_event_t *press = reinterpret_cast<const xcb_key_press_event_t *>(ev); Game::Key key; // TODO translate xcb_keycode_t switch (press->detail) { case 9: key = Game::KEY_ESC; break; case 111: key = Game::KEY_UP; break; case 116: key = Game::KEY_DOWN; break; case 65: key = Game::KEY_SPACE; break; default: key = Game::KEY_UNKNOWN; break; } game_.on_key(key); } break; case XCB_CLIENT_MESSAGE: { const xcb_client_message_event_t *msg = reinterpret_cast<const xcb_client_message_event_t *>(ev); if (msg->type == wm_protocols_ && msg->data.data32[0] == wm_delete_window_) game_.on_key(Game::KEY_SHUTDOWN); } break; default: break; } } void ShellXcb::loop_wait() { while (true) { xcb_generic_event_t *ev = xcb_wait_for_event(c_); if (!ev) continue; handle_event(ev); free(ev); if (quit_) break; acquire_back_buffer(); present_back_buffer(); } } void ShellXcb::loop_poll() { PosixTimer timer; double current_time = timer.get(); double profile_start_time = current_time; int profile_present_count = 0; while (true) { // handle pending events while (true) { xcb_generic_event_t *ev = xcb_poll_for_event(c_); if (!ev) break; handle_event(ev); free(ev); } if (quit_) break; acquire_back_buffer(); double t = timer.get(); add_game_time(static_cast<float>(t - current_time)); present_back_buffer(); current_time = t; profile_present_count++; if (current_time - profile_start_time >= 5.0) { const double fps = profile_present_count / (current_time - profile_start_time); std::stringstream ss; ss << profile_present_count << " presents in " << current_time - profile_start_time << " seconds " << "(FPS: " << fps << ")"; log(LOG_INFO, ss.str().c_str()); profile_start_time = current_time; profile_present_count = 0; } } } void ShellXcb::run() { create_window(); xcb_map_window(c_, win_); xcb_flush(c_); create_context(); resize_swapchain(settings_.initial_width, settings_.initial_height); quit_ = false; if (settings_.animate) loop_poll(); else loop_wait(); destroy_context(); xcb_destroy_window(c_, win_); xcb_flush(c_); }