vtgl

terminal emulator implemented in OpenGL
git clone anongit@rnpnr.xyz:vtgl.git
Log | Files | Refs | Feed | LICENSE

platform_linux_x11.c (17180B)


      1 /* See LICENSE for copyright details */
      2 #define GL_GLEXT_PROTOTYPES
      3 #include <GLFW/glfw3.h>
      4 
      5 /* TODO: fix glfw */
      6 typedef void *RROutput;
      7 typedef void *RRCrtc;
      8 typedef void *Display;
      9 typedef void *Window;
     10 
     11 #define GLFW_EXPOSE_NATIVE_X11
     12 #define GLFW_NATIVE_INCLUDE_NONE
     13 #include <GLFW/glfw3native.h>
     14 
     15 #include "vtgl.h"
     16 
     17 i32 XConnectionNumber(void *display);
     18 i32 XPending(void *display);
     19 
     20 #ifndef _DEBUG
     21 #define do_debug(...)
     22 #include "vtgl.c"
     23 #else
     24 #include <dlfcn.h>
     25 
     26 #define DEBUG_LIB_NAME "./vtgl.so"
     27 
     28 #define LIB_FNS                  \
     29 	X(vtgl_active_selection) \
     30 	X(vtgl_initialize)       \
     31 	X(vtgl_render_frame)     \
     32 	X(vtgl_handle_keys)      \
     33 	X(vtgl_frame_step)
     34 
     35 #define X(name) static name ## _fn *name;
     36 LIB_FNS
     37 #undef X
     38 
     39 static PLATFORM_FILE_WATCH_CALLBACK_FN(debug_reload_library)
     40 {
     41 	PlatformCtx *ctx = user_ctx;
     42 
     43 	if (ctx->input.executable_reloaded)
     44 		return;
     45 
     46 	/* NOTE(rnp): spin until render thread finishes its work */
     47 	while (!ctx->render_stack->thread_asleep);
     48 
     49 	ctx->input.executable_reloaded = 1;
     50 	s8 nl = s8("\n");
     51 	/* NOTE: glibc sucks and will crash if this is NULL */
     52 	if (ctx->library_handle)
     53 		dlclose(ctx->library_handle);
     54 	ctx->library_handle = dlopen((c8 *)path, RTLD_NOW|RTLD_LOCAL);
     55 	if (!ctx->library_handle)
     56 		stream_push_s8s(&ctx->error_stream, 3,
     57 		                (s8 []){s8("dlopen: "), c_str_to_s8(dlerror()), nl});
     58 
     59 	#define X(name) \
     60 		name = dlsym(ctx->library_handle, #name); \
     61 		if (!name) stream_push_s8s(&ctx->error_stream, 3, (s8 []){s8("dlsym: "), \
     62 			                                                  c_str_to_s8(dlerror()), nl});
     63 	LIB_FNS
     64 	#undef X
     65 
     66 	stream_push_s8(&ctx->error_stream, s8("Reloaded Main Program\n"));
     67 
     68 	os_write_err_msg(stream_to_s8(&ctx->error_stream));
     69 	ctx->error_stream.widx = 0;
     70 }
     71 #endif /* _DEBUG */
     72 
     73 static void
     74 glfw_error_callback(int code, const char *desc)
     75 {
     76 	u8 buf[256];
     77 	Stream err = {.cap = sizeof(buf), .buf = buf};
     78 	stream_push_s8(&err, s8("GLFW Error (0x"));
     79 	stream_push_hex_u64(&err, code);
     80 	stream_push_s8(&err, s8("): "));
     81 	os_write_err_msg(stream_to_s8(&err));
     82 	os_write_err_msg(c_str_to_s8((char *)desc));
     83 	os_write_err_msg(s8("\n"));
     84 }
     85 
     86 static void
     87 char_callback(GLFWwindow *win, u32 codepoint)
     88 {
     89 	PlatformCtx *ctx = glfwGetWindowUserPointer(win);
     90 	stream_push_s8(&ctx->char_stream, utf8_encode(codepoint));
     91 }
     92 
     93 /* NOTE: called when the window was resized */
     94 static void
     95 fb_callback(GLFWwindow *win, i32 w, i32 h)
     96 {
     97 	PlatformCtx *ctx       = glfwGetWindowUserPointer(win);
     98 	ctx->input.window_size = (iv2){.w = w, .h = h};
     99 }
    100 
    101 static void
    102 scroll_callback(GLFWwindow *win, f64 xoff, f64 yoff)
    103 {
    104 	PlatformCtx *ctx          = glfwGetWindowUserPointer(win);
    105 	ctx->input.mouse_scroll.x = xoff;
    106 	ctx->input.mouse_scroll.y = yoff;
    107 }
    108 
    109 static void
    110 key_callback(GLFWwindow *win, i32 key, i32 scancode, i32 action, i32 modifiers)
    111 {
    112 	PlatformCtx *ctx     = glfwGetWindowUserPointer(win);
    113 	TerminalInput *input = &ctx->input;
    114 
    115 	/* TODO: base this on X11 keys directly */
    116 	switch (key) {
    117 	case GLFW_KEY_LEFT_SHIFT:
    118 		button_action(input->keys + KEY_LEFT_SHIFT, action == GLFW_PRESS);
    119 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_SHIFT;
    120 		else                        input->modifiers |=  MOD_SHIFT;
    121 		break;
    122 	case GLFW_KEY_LEFT_CONTROL:
    123 		button_action(input->keys + KEY_LEFT_CONTROL, action == GLFW_PRESS);
    124 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_CONTROL;
    125 		else                        input->modifiers |=  MOD_CONTROL;
    126 		break;
    127 	case GLFW_KEY_LEFT_ALT:
    128 		button_action(input->keys + KEY_LEFT_ALT, action == GLFW_PRESS);
    129 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_ALT;
    130 		else                        input->modifiers |=  MOD_ALT;
    131 		break;
    132 	case GLFW_KEY_LEFT_SUPER:
    133 		button_action(input->keys + KEY_LEFT_SUPER, action == GLFW_PRESS);
    134 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_SUPER;
    135 		else                        input->modifiers |=  MOD_SUPER;
    136 		break;
    137 	case GLFW_KEY_RIGHT_SHIFT:
    138 		button_action(input->keys + KEY_RIGHT_SHIFT, action == GLFW_PRESS);
    139 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_SHIFT;
    140 		else                        input->modifiers |=  MOD_SHIFT;
    141 		break;
    142 	case GLFW_KEY_RIGHT_CONTROL:
    143 		button_action(input->keys + KEY_RIGHT_CONTROL, action == GLFW_PRESS);
    144 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_CONTROL;
    145 		else                        input->modifiers |=  MOD_CONTROL;
    146 		break;
    147 	case GLFW_KEY_RIGHT_ALT:
    148 		button_action(input->keys + KEY_RIGHT_ALT, action == GLFW_PRESS);
    149 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_ALT;
    150 		else                        input->modifiers |=  MOD_ALT;
    151 		break;
    152 	case GLFW_KEY_RIGHT_SUPER:
    153 		button_action(input->keys + KEY_RIGHT_SUPER, action == GLFW_PRESS);
    154 		if (action == GLFW_RELEASE) input->modifiers &= ~MOD_SUPER;
    155 		else                        input->modifiers |=  MOD_SUPER;
    156 		break;
    157 	case GLFW_KEY_MENU:
    158 		button_action(input->keys + KEY_MENU, action == GLFW_PRESS);
    159 		break;
    160 	}
    161 	vtgl_handle_keys(&ctx->memory, &ctx->input, key, action, modifiers);
    162 }
    163 
    164 static void
    165 mouse_button_callback(GLFWwindow *win, i32 button, i32 action, i32 modifiers)
    166 {
    167 	PlatformCtx *ctx     = glfwGetWindowUserPointer(win);
    168 	TerminalInput *input = &ctx->input;
    169 
    170 	switch (button) {
    171 	case GLFW_MOUSE_BUTTON_LEFT:
    172 		button_action(input->keys + MOUSE_LEFT, action == GLFW_PRESS);
    173 		break;
    174 	case GLFW_MOUSE_BUTTON_RIGHT:
    175 		button_action(input->keys + MOUSE_RIGHT, action == GLFW_PRESS);
    176 		break;
    177 	case GLFW_MOUSE_BUTTON_MIDDLE:
    178 		button_action(input->keys + MOUSE_MIDDLE, action == GLFW_PRESS);
    179 		break;
    180 	case GLFW_MOUSE_BUTTON_4:
    181 		button_action(input->keys + MOUSE_EXTENDED_0, action == GLFW_PRESS);
    182 		break;
    183 	case GLFW_MOUSE_BUTTON_5:
    184 		button_action(input->keys + MOUSE_EXTENDED_1, action == GLFW_PRESS);
    185 		break;
    186 	}
    187 }
    188 
    189 static void
    190 focus_callback(GLFWwindow *win, i32 focused)
    191 {
    192 	PlatformCtx *ctx            = glfwGetWindowUserPointer(win);
    193 	ctx->input.window_focused   = focused;
    194 	/* NOTE: force a refresh as well when the focus changes */
    195 	ctx->input.window_refreshed = 1;
    196 }
    197 
    198 static void
    199 refresh_callback(GLFWwindow *win)
    200 {
    201 	PlatformCtx *ctx            = glfwGetWindowUserPointer(win);
    202 	ctx->input.window_refreshed = 1;
    203 }
    204 
    205 static GLFWwindow *
    206 init_window(PlatformCtx *ctx, iv2 window_size)
    207 {
    208 	glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4);
    209 	glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
    210 	glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GLFW_TRUE);
    211 	glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
    212 
    213 	#ifdef _DEBUG
    214 	glfwWindowHint(GLFW_CONTEXT_DEBUG, GLFW_TRUE);
    215 	#endif
    216 
    217 	/* NOTE: we initially hide the window so that it can be freely resized behind the
    218 	 * back of the window manager prior to showing */
    219 	glfwWindowHint(GLFW_VISIBLE,   GLFW_FALSE);
    220 	glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE);
    221 
    222 	GLFWwindow *window = glfwCreateWindow(window_size.w, window_size.h, "vtgl", NULL, NULL);
    223 	if (!window) {
    224 		glfwTerminate();
    225 		os_fatal(s8("Failed to spawn GLFW window\n"));
    226 	}
    227 	glfwMakeContextCurrent(window);
    228 	glfwSetWindowUserPointer(window, ctx);
    229 
    230 	/* TODO: swap interval is not needed because we will sleep on waiting for terminal input */
    231 	glfwSwapInterval(1);
    232 
    233 	glfwSetCharCallback(window,            char_callback);
    234 	glfwSetFramebufferSizeCallback(window, fb_callback);
    235 	glfwSetKeyCallback(window,             key_callback);
    236 	glfwSetMouseButtonCallback(window,     mouse_button_callback);
    237 	glfwSetScrollCallback(window,          scroll_callback);
    238 	glfwSetWindowFocusCallback(window,     focus_callback);
    239 	glfwSetWindowRefreshCallback(window,   refresh_callback);
    240 
    241 	ctx->win_fd = XConnectionNumber(glfwGetX11Display());
    242 
    243 	return window;
    244 }
    245 
    246 static void
    247 update_input(PlatformCtx *ctx)
    248 {
    249 	TerminalInput *input = &ctx->input;
    250 	ctx->input.executable_reloaded = 0;
    251 	ctx->input.window_refreshed    = 0;
    252 
    253 	/* NOTE: mouse */
    254 	input->last_mouse   = input->mouse;
    255 	input->mouse_scroll = (v2){0};
    256 
    257 	f64 mouse_x, mouse_y;
    258 	glfwGetCursorPos(ctx->window, &mouse_x, &mouse_y);
    259 	input->mouse.x = mouse_x;
    260 	input->mouse.y = input->window_size.h - mouse_y;
    261 
    262 	for (u32 i = 0; i < ARRAY_COUNT(input->keys); i++)
    263 		input->keys[i].transitions = 0;
    264 
    265 	ctx->char_stream.widx = 0;
    266 
    267 	i64 timeout[2] = {0, 25e6};
    268 	if (input->pending_updates) {
    269 		timeout[1]             = 0;
    270 		input->pending_updates = 0;
    271 	}
    272 
    273 	sys_fd_set rfd = {0};
    274 	FD_SET(ctx->child.handle, rfd);
    275 	FD_SET(ctx->inotify_fd,   rfd);
    276 	FD_SET(ctx->win_fd,       rfd);
    277 
    278 	i32 max_fd = MAX(ctx->inotify_fd, ctx->child.handle);
    279 	max_fd     = MAX(max_fd, ctx->win_fd);
    280 	syscall6(SYS_pselect6, max_fd + 1, (iptr)rfd, 0, 0, (iptr)timeout, 0);
    281 
    282 	input->data_available = FD_ISSET(ctx->child.handle, rfd) != 0;
    283 
    284 	try_deferred_file_loads(ctx);
    285 	if (FD_ISSET(ctx->inotify_fd, rfd))
    286 		dispatch_file_watch_events(ctx);
    287 
    288 	glfwPollEvents();
    289 
    290 	/* NOTE: char_stream was filled in char callback */
    291 	input->character_input = stream_to_s8(&ctx->char_stream);
    292 }
    293 
    294 static PLATFORM_CLIPBOARD_FN(x11_get_clipboard)
    295 {
    296 	/* NOTE: this does a bunch of extra copying and other garbage. both GLFW and X11 are
    297 	 * at fault. The API is designed to do what the terminal wants and  not be constrained
    298 	 * by GLFW and X11 garbage */
    299 	ASSERT(buffer);
    300 	char *text = 0;
    301 	switch (clipboard) {
    302 	case CLIPBOARD_0: text = (c8 *)glfwGetClipboardString(0);   break;
    303 	case CLIPBOARD_1: text = (c8 *)glfwGetX11SelectionString(); break;
    304 	}
    305 	if (text) {
    306 		/* TODO: we may need to replace '\n' with '\r\n' */
    307 		stream_push_s8(buffer, c_str_to_s8(text));
    308 	}
    309 
    310 	return !buffer->errors;
    311 }
    312 
    313 static PLATFORM_CLIPBOARD_FN(x11_set_clipboard)
    314 {
    315 	ASSERT(buffer);
    316 	stream_push_byte(buffer, 0);
    317 
    318 	if (!buffer->errors) {
    319 		switch (clipboard) {
    320 		case CLIPBOARD_0: glfwSetClipboardString(0, (c8 *)buffer->buf); break;
    321 		case CLIPBOARD_1: glfwSetX11SelectionString((c8 *)buffer->buf); break;
    322 		}
    323 	}
    324 	return !buffer->errors;
    325 }
    326 
    327 static PLATFORM_WINDOW_TITLE_FN(x11_get_window_title)
    328 {
    329 	ASSERT(buffer);
    330 	char *title = (c8 *)glfwGetWindowTitle(linux_ctx.window);
    331 	if (title) stream_push_s8(buffer, c_str_to_s8(title));
    332 }
    333 
    334 static PLATFORM_WINDOW_TITLE_FN(x11_set_window_title)
    335 {
    336 	ASSERT(buffer);
    337 	stream_push_byte(buffer, 0);
    338 	if (!buffer->errors)
    339 		glfwSetWindowTitle(linux_ctx.window, (c8 *)buffer->buf);
    340 }
    341 
    342 static void
    343 linux_render_thread_entry(struct stack_base *stack)
    344 {
    345 	{
    346 		/* NOTE: set thread name */
    347 		char name[16] = "[render]";
    348 		syscall2(SYS_prctl, PR_SET_NAME, (iptr)name);
    349 		/* NOTE: halt until main thread is ready to hand off gl context */
    350 		syscall4(SYS_futex, (iptr)&stack->work_futex, FUTEX_WAIT, 0, 0);
    351 
    352 		glfwMakeContextCurrent(stack->window);
    353 	}
    354 
    355 	for (;;) {
    356 		stack->thread_asleep = 1;
    357 		syscall4(SYS_futex, (iptr)&stack->work_futex, FUTEX_WAIT, 0, 0);
    358 		stack->thread_asleep = 0;
    359 		vtgl_render_frame(stack->terminal_memory, stack->input, stack->thread_arena);
    360 		glfwSwapBuffers(stack->window);
    361 	}
    362 
    363 	__builtin_unreachable();
    364 }
    365 
    366 i32
    367 main(i32 argc, char *argv[], char *envp[])
    368 {
    369 	linux_ctx.platform_memory = arena_from_memory_block(linux_block_alloc(MB(2)));
    370 	linux_ctx.error_stream    = stream_alloc(&linux_ctx.platform_memory, KB(256));
    371 
    372 	iv2 cells = {.x = -1, .y = -1};
    373 
    374 	char *argv0 = *argv++;
    375 	argc--;
    376 	struct conversion_result cres;
    377 	for (i32 i = 0; i < argc; i++) {
    378 		char *arg = argv[i];
    379 		if (!arg || !arg[0])
    380 			usage(argv0, &linux_ctx.error_stream);
    381 		if (arg[0] != '-')
    382 			break;
    383 		arg++;
    384 		switch (arg[0]) {
    385 		case 'g':
    386 			if (!argv[i + 1])
    387 				usage(argv0, &linux_ctx.error_stream);
    388 			cres = i32_from_cstr(argv[i + 1], 'x');
    389 			if (cres.status == CR_SUCCESS)
    390 				cells.w = cres.i;
    391 			cres = i32_from_cstr(cres.unparsed, 0);
    392 			if (cres.status == CR_SUCCESS)
    393 				cells.h = cres.i;
    394 			if (cells.w <= 0 || cells.h <= 0) {
    395 				stream_push_s8(&linux_ctx.error_stream, s8("ignoring malformed geometry: "));
    396 				stream_push_s8(&linux_ctx.error_stream, c_str_to_s8(argv[i + 1]));
    397 				stream_push_byte(&linux_ctx.error_stream, '\n');
    398 			}
    399 			argv++;
    400 			argc--;
    401 			break;
    402 		case 'v':
    403 			stream_push_s8s(&linux_ctx.error_stream, 2,
    404 			                (s8 []){c_str_to_s8(argv0), s8(" " VERSION "\n")});
    405 			os_fatal(stream_to_s8(&linux_ctx.error_stream));
    406 		default:
    407 			usage(argv0, &linux_ctx.error_stream);
    408 		}
    409 	}
    410 	if (linux_ctx.error_stream.widx) {
    411 		os_write_err_msg(stream_to_s8(&linux_ctx.error_stream));
    412 		linux_ctx.error_stream.widx = 0;
    413 	}
    414 
    415 	linux_ctx.render_stack                = new_stack(KB(256));
    416 	linux_ctx.render_stack->entry         = linux_render_thread_entry;
    417 	linux_ctx.render_stack->thread_asleep = 1;
    418 	new_thread(linux_ctx.render_stack);
    419 
    420 	{
    421 		Arena tmp = linux_ctx.platform_memory;
    422 		SLLVariableVector environment_block = parse_environment(&tmp, envp);
    423 
    424 		/* TODO: build up argv for the child as well */
    425 		c8 **child_envp = construct_c_str_array(&tmp, environment_block);
    426 		linux_ctx.child = os_fork_child(get_default_cmd(envp), child_envp);
    427 	}
    428 
    429 	{
    430 		MemoryBlock terminal_memory  = linux_block_alloc(MB(32));
    431 		linux_ctx.memory.memory      = terminal_memory.memory;
    432 		linux_ctx.memory.memory_size = terminal_memory.size;
    433 #ifdef _DEBUG
    434 		MemoryBlock debug_memory           = linux_block_alloc(MB(128));
    435 		linux_ctx.memory.debug_memory      = debug_memory.memory;
    436 		linux_ctx.memory.debug_memory_size = debug_memory.size;
    437 #endif
    438 	}
    439 
    440 	linux_ctx.inotify_fd = syscall1(SYS_inotify_init1, O_NONBLOCK|O_CLOEXEC);
    441 
    442 #ifdef _DEBUG
    443 	debug_reload_library((u8 *)DEBUG_LIB_NAME, &linux_ctx);
    444 	linux_add_file_watch((u8 *)DEBUG_LIB_NAME, debug_reload_library, &linux_ctx);
    445 #endif
    446 
    447 	linux_ctx.memory.platform_api.add_file_watch       = linux_add_file_watch;
    448 	linux_ctx.memory.platform_api.allocate_ring_buffer = os_allocate_ring_buffer;
    449 	linux_ctx.memory.platform_api.get_clipboard        = x11_get_clipboard;
    450 	linux_ctx.memory.platform_api.set_clipboard        = x11_set_clipboard;
    451 	linux_ctx.memory.platform_api.read_file            = os_read_file;
    452 	linux_ctx.memory.platform_api.read                 = os_read;
    453 	linux_ctx.memory.platform_api.set_terminal_size    = os_set_terminal_size;
    454 	linux_ctx.memory.platform_api.get_window_title     = x11_get_window_title;
    455 	linux_ctx.memory.platform_api.set_window_title     = x11_set_window_title;
    456 	linux_ctx.memory.platform_api.write                = os_write;
    457 	linux_ctx.memory.platform_api.path_separator       = '/';
    458 
    459 	if (!glfwInit())
    460 		os_fatal(s8("Failed to init GLFW\n"));
    461 	glfwSetErrorCallback(glfw_error_callback);
    462 
    463 	GLFWmonitor *mon = glfwGetPrimaryMonitor();
    464 	if (!mon) {
    465 		glfwTerminate();
    466 		os_fatal(s8("Failed to find any monitors!\n"));
    467 	}
    468 	iv2 monitor_size;
    469 	glfwGetMonitorWorkarea(mon, NULL, NULL, &monitor_size.w, &monitor_size.h);
    470 
    471 	linux_ctx.char_stream = arena_stream(linux_ctx.platform_memory);
    472 
    473 	iv2 window_size  = {.w = 1280, .h = 720};
    474 	linux_ctx.window = init_window(&linux_ctx, window_size);
    475 
    476 	iv2 requested_size = vtgl_initialize(&linux_ctx.memory, linux_ctx.child.handle, cells, monitor_size);
    477 	if (requested_size.w > 0 && requested_size.h > 0 &&
    478 	    (requested_size.w != window_size.w || requested_size.h != window_size.h))
    479 	{
    480 		glfwSetWindowAttrib(linux_ctx.window, GLFW_FLOATING, GLFW_TRUE);
    481 		i32 x       = ABS(window_size.w - requested_size.w) / 2;
    482 		i32 y       = ABS(window_size.h - requested_size.h) / 2;
    483 		window_size = requested_size;
    484 		glfwSetWindowMonitor(linux_ctx.window, 0, x, y, window_size.w, window_size.h, GLFW_DONT_CARE);
    485 		/* NOTE: resizable must be set after the window is shown; otherwise tiling window
    486 		 * managers will forcibly resize us even if we are supposed to be floating */
    487 		glfwShowWindow(linux_ctx.window);
    488 		glfwSetWindowAttrib(linux_ctx.window, GLFW_RESIZABLE, GLFW_TRUE);
    489 	} else {
    490 		/* NOTE: on the other hand we should let the window be resized if no size was
    491 		 * explicitly requested */
    492 		glfwSetWindowAttrib(linux_ctx.window, GLFW_RESIZABLE, GLFW_TRUE);
    493 		glfwSetWindowPos(linux_ctx.window,
    494 		                 ABS(monitor_size.w - window_size.w) / 2,
    495 		                 ABS(monitor_size.h - window_size.h) / 2);
    496 		glfwShowWindow(linux_ctx.window);
    497 	}
    498 	glfwMakeContextCurrent(0);
    499 
    500 	linux_ctx.input.window_size = window_size;
    501 
    502 	linux_ctx.render_stack->input           = &linux_ctx.input;
    503 	linux_ctx.render_stack->terminal_memory = &linux_ctx.memory;
    504 	linux_ctx.render_stack->thread_arena    = arena_from_memory_block(linux_block_alloc(MB(8)));
    505 	linux_ctx.render_stack->window          = linux_ctx.window;
    506 	syscall3(SYS_futex, (iptr)&linux_ctx.render_stack->work_futex, FUTEX_WAKE, 1);
    507 
    508 	Range last_sel   = {0};
    509 	f64   last_time  = os_get_time();
    510 	while (!glfwWindowShouldClose(linux_ctx.window)) {
    511 		if (os_child_exited(linux_ctx.child.process_id))
    512 			break;
    513 
    514 		/* TODO: cpu time excluding waiting for the vblank */
    515 		f64 current_time   = os_get_time();
    516 		linux_ctx.input.dt = current_time - last_time;
    517 		last_time          = current_time;
    518 
    519 		update_input(&linux_ctx);
    520 		if (vtgl_frame_step(&linux_ctx.memory, &linux_ctx.input))
    521 			syscall3(SYS_futex, (iptr)&linux_ctx.render_stack->work_futex, FUTEX_WAKE, 1);
    522 
    523 		Range current_sel = vtgl_active_selection(&linux_ctx.memory, 0);
    524 		if (is_valid_range(current_sel) && !equal_range(current_sel, last_sel)) {
    525 			Stream buf = arena_stream(linux_ctx.platform_memory);
    526 			vtgl_active_selection(&linux_ctx.memory, &buf);
    527 			stream_push_byte(&buf, 0);
    528 			if (!buf.errors)
    529 				glfwSetX11SelectionString((c8 *)buf.buf);
    530 			last_sel = current_sel;
    531 		}
    532 	}
    533 
    534 	syscall1(SYS_exit_group, 0);
    535 	__builtin_unreachable();
    536 
    537 	return 0;
    538 }