//======================================================================== // GLFW 3.3 Mir - www.glfw.org //------------------------------------------------------------------------ // Copyright (c) 2014-2017 Brandon Schaefer // // This software is provided 'as-is', without any express or implied // warranty. In no event will the authors be held liable for any damages // arising from the use of this software. // // Permission is granted to anyone to use this software for any purpose, // including commercial applications, and to alter it and redistribute it // freely, subject to the following restrictions: // // 1. The origin of this software must not be misrepresented; you must not // claim that you wrote the original software. If you use this software // in a product, an acknowledgment in the product documentation would // be appreciated but is not required. // // 2. Altered source versions must be plainly marked as such, and must not // be misrepresented as being the original software. // // 3. This notice may not be removed or altered from any source // distribution. // //======================================================================== #include "internal.h" #include #include #include typedef struct EventNode { TAILQ_ENTRY(EventNode) entries; const MirEvent* event; _GLFWwindow* window; } EventNode; static void deleteNode(EventQueue* queue, EventNode* node) { mir_event_unref(node->event); free(node); } static GLFWbool emptyEventQueue(EventQueue* queue) { return queue->head.tqh_first == NULL; } // TODO The mir_event_ref is not supposed to be used but ... its needed // in this case. Need to wait until we can read from an FD set up by mir // for single threaded event handling. static EventNode* newEventNode(const MirEvent* event, _GLFWwindow* context) { EventNode* newNode = calloc(1, sizeof(EventNode)); newNode->event = mir_event_ref(event); newNode->window = context; return newNode; } static void enqueueEvent(const MirEvent* event, _GLFWwindow* context) { pthread_mutex_lock(&_glfw.mir.eventMutex); EventNode* newNode = newEventNode(event, context); TAILQ_INSERT_TAIL(&_glfw.mir.eventQueue->head, newNode, entries); pthread_cond_signal(&_glfw.mir.eventCond); pthread_mutex_unlock(&_glfw.mir.eventMutex); } static EventNode* dequeueEvent(EventQueue* queue) { EventNode* node = NULL; pthread_mutex_lock(&_glfw.mir.eventMutex); node = queue->head.tqh_first; if (node) TAILQ_REMOVE(&queue->head, node, entries); pthread_mutex_unlock(&_glfw.mir.eventMutex); return node; } static MirPixelFormat findValidPixelFormat(void) { unsigned int i, validFormats, mirPixelFormats = 32; MirPixelFormat formats[mir_pixel_formats]; mir_connection_get_available_surface_formats(_glfw.mir.connection, formats, mirPixelFormats, &validFormats); for (i = 0; i < validFormats; i++) { if (formats[i] == mir_pixel_format_abgr_8888 || formats[i] == mir_pixel_format_xbgr_8888 || formats[i] == mir_pixel_format_argb_8888 || formats[i] == mir_pixel_format_xrgb_8888) { return formats[i]; } } return mir_pixel_format_invalid; } static int mirModToGLFWMod(uint32_t mods) { int publicMods = 0x0; if (mods & mir_input_event_modifier_alt) publicMods |= GLFW_MOD_ALT; else if (mods & mir_input_event_modifier_shift) publicMods |= GLFW_MOD_SHIFT; else if (mods & mir_input_event_modifier_ctrl) publicMods |= GLFW_MOD_CONTROL; else if (mods & mir_input_event_modifier_meta) publicMods |= GLFW_MOD_SUPER; return publicMods; } static int toGLFWKeyCode(uint32_t key) { if (key < sizeof(_glfw.mir.keycodes) / sizeof(_glfw.mir.keycodes[0])) return _glfw.mir.keycodes[key]; return GLFW_KEY_UNKNOWN; } static void handleKeyEvent(const MirKeyboardEvent* key_event, _GLFWwindow* window) { const int action = mir_keyboard_event_action (key_event); const int scan_code = mir_keyboard_event_scan_code(key_event); const int key_code = mir_keyboard_event_key_code (key_event); const int modifiers = mir_keyboard_event_modifiers(key_event); const int pressed = action == mir_keyboard_action_up ? GLFW_RELEASE : GLFW_PRESS; const int mods = mirModToGLFWMod(modifiers); const long text = _glfwKeySym2Unicode(key_code); const int plain = !(mods & (GLFW_MOD_CONTROL | GLFW_MOD_ALT)); _glfwInputKey(window, toGLFWKeyCode(scan_code), scan_code, pressed, mods); if (text != -1) _glfwInputChar(window, text, mods, plain); } static void handlePointerButton(_GLFWwindow* window, int pressed, const MirPointerEvent* pointer_event) { int mods = mir_pointer_event_modifiers(pointer_event); const int publicMods = mirModToGLFWMod(mods); MirPointerButton button = mir_pointer_button_primary; static uint32_t oldButtonStates = 0; uint32_t newButtonStates = mir_pointer_event_buttons(pointer_event); int publicButton = GLFW_MOUSE_BUTTON_LEFT; // XOR our old button states our new states to figure out what was added or removed button = newButtonStates ^ oldButtonStates; switch (button) { case mir_pointer_button_primary: publicButton = GLFW_MOUSE_BUTTON_LEFT; break; case mir_pointer_button_secondary: publicButton = GLFW_MOUSE_BUTTON_RIGHT; break; case mir_pointer_button_tertiary: publicButton = GLFW_MOUSE_BUTTON_MIDDLE; break; case mir_pointer_button_forward: // FIXME What is the forward button? publicButton = GLFW_MOUSE_BUTTON_4; break; case mir_pointer_button_back: // FIXME What is the back button? publicButton = GLFW_MOUSE_BUTTON_5; break; default: break; } oldButtonStates = newButtonStates; _glfwInputMouseClick(window, publicButton, pressed, publicMods); } static void handlePointerMotion(_GLFWwindow* window, const MirPointerEvent* pointer_event) { const int hscroll = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_hscroll); const int vscroll = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_vscroll); if (window->cursorMode == GLFW_CURSOR_DISABLED) { if (_glfw.mir.disabledCursorWindow != window) return; const int dx = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_relative_x); const int dy = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_relative_y); const int current_x = window->virtualCursorPosX; const int current_y = window->virtualCursorPosY; _glfwInputCursorPos(window, dx + current_x, dy + current_y); } else { const int x = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_x); const int y = mir_pointer_event_axis_value(pointer_event, mir_pointer_axis_y); _glfwInputCursorPos(window, x, y); } if (hscroll != 0 || vscroll != 0) _glfwInputScroll(window, hscroll, vscroll); } static void handlePointerEvent(const MirPointerEvent* pointer_event, _GLFWwindow* window) { int action = mir_pointer_event_action(pointer_event); switch (action) { case mir_pointer_action_button_down: handlePointerButton(window, GLFW_PRESS, pointer_event); break; case mir_pointer_action_button_up: handlePointerButton(window, GLFW_RELEASE, pointer_event); break; case mir_pointer_action_motion: handlePointerMotion(window, pointer_event); break; case mir_pointer_action_enter: case mir_pointer_action_leave: break; default: break; } } static void handleInput(const MirInputEvent* input_event, _GLFWwindow* window) { int type = mir_input_event_get_type(input_event); switch (type) { case mir_input_event_type_key: handleKeyEvent(mir_input_event_get_keyboard_event(input_event), window); break; case mir_input_event_type_pointer: handlePointerEvent(mir_input_event_get_pointer_event(input_event), window); break; default: break; } } static void handleEvent(const MirEvent* event, _GLFWwindow* window) { int type = mir_event_get_type(event); switch (type) { case mir_event_type_input: handleInput(mir_event_get_input_event(event), window); break; default: break; } } static void addNewEvent(MirWindow* window, const MirEvent* event, void* context) { enqueueEvent(event, context); } static GLFWbool createWindow(_GLFWwindow* window) { MirWindowSpec* spec; MirBufferUsage buffer_usage = mir_buffer_usage_hardware; MirPixelFormat pixel_format = findValidPixelFormat(); if (pixel_format == mir_pixel_format_invalid) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unable to find a correct pixel format"); return GLFW_FALSE; } spec = mir_create_normal_window_spec(_glfw.mir.connection, window->mir.width, window->mir.height); mir_window_spec_set_pixel_format(spec, pixel_format); mir_window_spec_set_buffer_usage(spec, buffer_usage); window->mir.window = mir_create_window_sync(spec); mir_window_spec_release(spec); if (!mir_window_is_valid(window->mir.window)) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unable to create window: %s", mir_window_get_error_message(window->mir.window)); return GLFW_FALSE; } mir_window_set_event_handler(window->mir.window, addNewEvent, window); return GLFW_TRUE; } static void setWindowConfinement(_GLFWwindow* window, MirPointerConfinementState state) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_pointer_confinement(spec, state); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } ////////////////////////////////////////////////////////////////////////// ////// GLFW internal API ////// ////////////////////////////////////////////////////////////////////////// void _glfwInitEventQueueMir(EventQueue* queue) { TAILQ_INIT(&queue->head); } void _glfwDeleteEventQueueMir(EventQueue* queue) { if (queue) { EventNode* node, *node_next; node = queue->head.tqh_first; while (node != NULL) { node_next = node->entries.tqe_next; TAILQ_REMOVE(&queue->head, node, entries); deleteNode(queue, node); node = node_next; } free(queue); } } ////////////////////////////////////////////////////////////////////////// ////// GLFW platform API ////// ////////////////////////////////////////////////////////////////////////// int _glfwPlatformCreateWindow(_GLFWwindow* window, const _GLFWwndconfig* wndconfig, const _GLFWctxconfig* ctxconfig, const _GLFWfbconfig* fbconfig) { if (window->monitor) { GLFWvidmode mode; _glfwPlatformGetVideoMode(window->monitor, &mode); mir_window_set_state(window->mir.window, mir_window_state_fullscreen); if (wndconfig->width > mode.width || wndconfig->height > mode.height) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Requested window size too large: %ix%i", wndconfig->width, wndconfig->height); return GLFW_FALSE; } } window->mir.width = wndconfig->width; window->mir.height = wndconfig->height; window->mir.currentCursor = NULL; if (!createWindow(window)) return GLFW_FALSE; window->mir.nativeWindow = mir_buffer_stream_get_egl_native_window( mir_window_get_buffer_stream(window->mir.window)); if (ctxconfig->client != GLFW_NO_API) { if (ctxconfig->source == GLFW_EGL_CONTEXT_API || ctxconfig->source == GLFW_NATIVE_CONTEXT_API) { if (!_glfwInitEGL()) return GLFW_FALSE; if (!_glfwCreateContextEGL(window, ctxconfig, fbconfig)) return GLFW_FALSE; } else if (ctxconfig->source == GLFW_OSMESA_CONTEXT_API) { if (!_glfwInitOSMesa()) return GLFW_FALSE; if (!_glfwCreateContextOSMesa(window, ctxconfig, fbconfig)) return GLFW_FALSE; } } return GLFW_TRUE; } void _glfwPlatformDestroyWindow(_GLFWwindow* window) { if (_glfw.mir.disabledCursorWindow == window) _glfw.mir.disabledCursorWindow = NULL; if (mir_window_is_valid(window->mir.window)) { mir_window_release_sync(window->mir.window); window->mir.window= NULL; } if (window->context.destroy) window->context.destroy(window); } void _glfwPlatformSetWindowTitle(_GLFWwindow* window, const char* title) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_name(spec, title); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformSetWindowIcon(_GLFWwindow* window, int count, const GLFWimage* images) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetWindowSize(_GLFWwindow* window, int width, int height) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_width (spec, width); mir_window_spec_set_height(spec, height); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformSetWindowSizeLimits(_GLFWwindow* window, int minwidth, int minheight, int maxwidth, int maxheight) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_max_width (spec, maxwidth); mir_window_spec_set_max_height(spec, maxheight); mir_window_spec_set_min_width (spec, minwidth); mir_window_spec_set_min_height(spec, minheight); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformSetWindowAspectRatio(_GLFWwindow* window, int numer, int denom) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetWindowPos(_GLFWwindow* window, int xpos, int ypos) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformGetWindowFrameSize(_GLFWwindow* window, int* left, int* top, int* right, int* bottom) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformGetWindowPos(_GLFWwindow* window, int* xpos, int* ypos) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformGetWindowSize(_GLFWwindow* window, int* width, int* height) { if (width) *width = window->mir.width; if (height) *height = window->mir.height; } void _glfwPlatformIconifyWindow(_GLFWwindow* window) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_state(spec, mir_window_state_minimized); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformRestoreWindow(_GLFWwindow* window) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_state(spec, mir_window_state_restored); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformMaximizeWindow(_GLFWwindow* window) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_state(spec, mir_window_state_maximized); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformHideWindow(_GLFWwindow* window) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_state(spec, mir_window_state_hidden); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformShowWindow(_GLFWwindow* window) { MirWindowSpec* spec; spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_state(spec, mir_window_state_restored); mir_window_apply_spec(window->mir.window, spec); mir_window_spec_release(spec); } void _glfwPlatformRequestWindowAttention(_GLFWwindow* window) { } void _glfwPlatformFocusWindow(_GLFWwindow* window) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetWindowMonitor(_GLFWwindow* window, _GLFWmonitor* monitor, int xpos, int ypos, int width, int height, int refreshRate) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } int _glfwPlatformWindowFocused(_GLFWwindow* window) { return mir_window_get_focus_state(window->mir.window) == mir_window_focus_state_focused; } int _glfwPlatformWindowIconified(_GLFWwindow* window) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); return GLFW_FALSE; } int _glfwPlatformWindowVisible(_GLFWwindow* window) { return mir_window_get_visibility(window->mir.window) == mir_window_visibility_exposed; } int _glfwPlatformWindowMaximized(_GLFWwindow* window) { return mir_window_get_state(window->mir.window) == mir_window_state_maximized; } void _glfwPlatformSetWindowResizable(_GLFWwindow* window, GLFWbool enabled) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetWindowDecorated(_GLFWwindow* window, GLFWbool enabled) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetWindowFloating(_GLFWwindow* window, GLFWbool enabled) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformPollEvents(void) { EventNode* node = NULL; while ((node = dequeueEvent(_glfw.mir.eventQueue))) { handleEvent(node->event, node->window); deleteNode(_glfw.mir.eventQueue, node); } } void _glfwPlatformWaitEvents(void) { pthread_mutex_lock(&_glfw.mir.eventMutex); while (emptyEventQueue(_glfw.mir.eventQueue)) pthread_cond_wait(&_glfw.mir.eventCond, &_glfw.mir.eventMutex); pthread_mutex_unlock(&_glfw.mir.eventMutex); _glfwPlatformPollEvents(); } void _glfwPlatformWaitEventsTimeout(double timeout) { pthread_mutex_lock(&_glfw.mir.eventMutex); if (emptyEventQueue(_glfw.mir.eventQueue)) { struct timespec time; clock_gettime(CLOCK_REALTIME, &time); time.tv_sec += (long) timeout; time.tv_nsec += (long) ((timeout - (long) timeout) * 1e9); pthread_cond_timedwait(&_glfw.mir.eventCond, &_glfw.mir.eventMutex, &time); } pthread_mutex_unlock(&_glfw.mir.eventMutex); _glfwPlatformPollEvents(); } void _glfwPlatformPostEmptyEvent(void) { } void _glfwPlatformGetFramebufferSize(_GLFWwindow* window, int* width, int* height) { if (width) *width = window->mir.width; if (height) *height = window->mir.height; } int _glfwPlatformCreateCursor(_GLFWcursor* cursor, const GLFWimage* image, int xhot, int yhot) { MirBufferStream* stream; int i_w = image->width; int i_h = image->height; stream = mir_connection_create_buffer_stream_sync(_glfw.mir.connection, i_w, i_h, mir_pixel_format_argb_8888, mir_buffer_usage_software); cursor->mir.conf = mir_cursor_configuration_from_buffer_stream(stream, xhot, yhot); MirGraphicsRegion region; mir_buffer_stream_get_graphics_region(stream, ®ion); unsigned char* pixels = image->pixels; char* dest = region.vaddr; int i; for (i = 0; i < i_w * i_h; i++, pixels += 4) { unsigned int alpha = pixels[3]; *dest++ = (char)(pixels[2] * alpha / 255); *dest++ = (char)(pixels[1] * alpha / 255); *dest++ = (char)(pixels[0] * alpha / 255); *dest++ = (char)alpha; } mir_buffer_stream_swap_buffers_sync(stream); cursor->mir.customCursor = stream; return GLFW_TRUE; } static const char* getSystemCursorName(int shape) { switch (shape) { case GLFW_ARROW_CURSOR: return mir_arrow_cursor_name; case GLFW_IBEAM_CURSOR: return mir_caret_cursor_name; case GLFW_CROSSHAIR_CURSOR: return mir_crosshair_cursor_name; case GLFW_HAND_CURSOR: return mir_open_hand_cursor_name; case GLFW_HRESIZE_CURSOR: return mir_horizontal_resize_cursor_name; case GLFW_VRESIZE_CURSOR: return mir_vertical_resize_cursor_name; } return NULL; } int _glfwPlatformCreateStandardCursor(_GLFWcursor* cursor, int shape) { cursor->mir.conf = NULL; cursor->mir.customCursor = NULL; cursor->mir.cursorName = getSystemCursorName(shape); return cursor->mir.cursorName != NULL; } void _glfwPlatformDestroyCursor(_GLFWcursor* cursor) { if (cursor->mir.conf) mir_cursor_configuration_destroy(cursor->mir.conf); if (cursor->mir.customCursor) mir_buffer_stream_release_sync(cursor->mir.customCursor); } static void setCursorNameForWindow(MirWindow* window, char const* name) { MirWindowSpec* spec = mir_create_window_spec(_glfw.mir.connection); mir_window_spec_set_cursor_name(spec, name); mir_window_apply_spec(window, spec); mir_window_spec_release(spec); } void _glfwPlatformSetCursor(_GLFWwindow* window, _GLFWcursor* cursor) { if (cursor) { window->mir.currentCursor = cursor; if (cursor->mir.cursorName) { setCursorNameForWindow(window->mir.window, cursor->mir.cursorName); } else if (cursor->mir.conf) { mir_window_configure_cursor(window->mir.window, cursor->mir.conf); } } else { setCursorNameForWindow(window->mir.window, mir_default_cursor_name); } } void _glfwPlatformGetCursorPos(_GLFWwindow* window, double* xpos, double* ypos) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetCursorPos(_GLFWwindow* window, double xpos, double ypos) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } void _glfwPlatformSetCursorMode(_GLFWwindow* window, int mode) { if (mode == GLFW_CURSOR_DISABLED) { _glfw.mir.disabledCursorWindow = window; setWindowConfinement(window, mir_pointer_confined_to_window); setCursorNameForWindow(window->mir.window, mir_disabled_cursor_name); } else { // If we were disabled before lets undo that! if (_glfw.mir.disabledCursorWindow == window) { _glfw.mir.disabledCursorWindow = NULL; setWindowConfinement(window, mir_pointer_unconfined); } if (window->cursorMode == GLFW_CURSOR_NORMAL) { _glfwPlatformSetCursor(window, window->mir.currentCursor); } else if (window->cursorMode == GLFW_CURSOR_HIDDEN) { setCursorNameForWindow(window->mir.window, mir_disabled_cursor_name); } } } const char* _glfwPlatformGetKeyName(int key, int scancode) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); return NULL; } int _glfwPlatformGetKeyScancode(int key) { return _glfw.mir.scancodes[key]; } void _glfwPlatformSetClipboardString(_GLFWwindow* window, const char* string) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); } const char* _glfwPlatformGetClipboardString(_GLFWwindow* window) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Unsupported function %s", __PRETTY_FUNCTION__); return NULL; } void _glfwPlatformGetRequiredInstanceExtensions(char** extensions) { if (!_glfw.vk.KHR_surface || !_glfw.vk.KHR_mir_surface) return; extensions[0] = "VK_KHR_surface"; extensions[1] = "VK_KHR_mir_surface"; } int _glfwPlatformGetPhysicalDevicePresentationSupport(VkInstance instance, VkPhysicalDevice device, uint32_t queuefamily) { PFN_vkGetPhysicalDeviceMirPresentationSupportKHR vkGetPhysicalDeviceMirPresentationSupportKHR = (PFN_vkGetPhysicalDeviceMirPresentationSupportKHR) vkGetInstanceProcAddr(instance, "vkGetPhysicalDeviceMirPresentationSupportKHR"); if (!vkGetPhysicalDeviceMirPresentationSupportKHR) { _glfwInputError(GLFW_API_UNAVAILABLE, "Mir: Vulkan instance missing VK_KHR_mir_surface extension"); return GLFW_FALSE; } return vkGetPhysicalDeviceMirPresentationSupportKHR(device, queuefamily, _glfw.mir.connection); } VkResult _glfwPlatformCreateWindowSurface(VkInstance instance, _GLFWwindow* window, const VkAllocationCallbacks* allocator, VkSurfaceKHR* surface) { VkResult err; VkMirWindowCreateInfoKHR sci; PFN_vkCreateMirWindowKHR vkCreateMirWindowKHR; vkCreateMirWindowKHR = (PFN_vkCreateMirWindowKHR) vkGetInstanceProcAddr(instance, "vkCreateMirWindowKHR"); if (!vkCreateMirWindowKHR) { _glfwInputError(GLFW_API_UNAVAILABLE, "Mir: Vulkan instance missing VK_KHR_mir_surface extension"); return VK_ERROR_EXTENSION_NOT_PRESENT; } memset(&sci, 0, sizeof(sci)); sci.sType = VK_STRUCTURE_TYPE_MIR_SURFACE_CREATE_INFO_KHR; sci.connection = _glfw.mir.connection; sci.mirWindow = window->mir.window; err = vkCreateMirWindowKHR(instance, &sci, allocator, surface); if (err) { _glfwInputError(GLFW_PLATFORM_ERROR, "Mir: Failed to create Vulkan surface: %s", _glfwGetVulkanResultString(err)); } return err; } ////////////////////////////////////////////////////////////////////////// ////// GLFW native API ////// ////////////////////////////////////////////////////////////////////////// GLFWAPI MirConnection* glfwGetMirDisplay(void) { _GLFW_REQUIRE_INIT_OR_RETURN(NULL); return _glfw.mir.connection; } GLFWAPI MirWindow* glfwGetMirWindow(GLFWwindow* handle) { _GLFWwindow* window = (_GLFWwindow*) handle; _GLFW_REQUIRE_INIT_OR_RETURN(NULL); return window->mir.window; }