diff --git a/src/Surface.zig b/src/Surface.zig index 95bc4d765..7ee05d642 100644 --- a/src/Surface.zig +++ b/src/Surface.zig @@ -101,7 +101,7 @@ color_scheme: apprt.ColorScheme = .light, last_binding_trigger: u64 = 0, /// The terminal IO handler. -io: termio.Impl, +io: termio.Termio, io_thread: termio.Thread, io_thr: std.Thread, @@ -396,33 +396,8 @@ pub fn init( ); errdefer render_thread.deinit(); - // Start our IO implementation - var io = try termio.Impl.init(alloc, .{ - .grid_size = grid_size, - .screen_size = screen_size, - .padding = padding, - .full_config = config, - .config = try termio.Impl.DerivedConfig.init(alloc, config), - .resources_dir = main.state.resources_dir, - .renderer_state = &self.renderer_state, - .renderer_wakeup = render_thread.wakeup, - .renderer_mailbox = render_thread.mailbox, - .surface_mailbox = .{ .surface = self, .app = app_mailbox }, - - // Get the cgroup if we're on linux and have the decl. I'd love - // to change this from a decl to a surface options struct because - // then we can do memory management better (don't need to retain - // the string around). - .linux_cgroup = if (comptime builtin.os.tag == .linux and - @hasDecl(apprt.runtime.Surface, "cgroup")) - rt_surface.cgroup() - else - Command.linux_cgroup_default, - }); - errdefer io.deinit(); - // Create the IO thread - var io_thread = try termio.Thread.init(alloc, &self.io); + var io_thread = try termio.Thread.init(alloc); errdefer io_thread.deinit(); self.* = .{ @@ -440,7 +415,7 @@ pub fn init( }, .renderer_thr = undefined, .mouse = .{}, - .io = io, + .io = undefined, .io_thread = io_thread, .io_thr = undefined, .screen_size = .{ .width = 0, .height = 0 }, @@ -450,6 +425,53 @@ pub fn init( .config = derived_config, }; + // Start our IO implementation + // This separate block ({}) is important because our errdefers must + // be scoped here to be valid. + { + // Initialize our IO backend + var io_exec = try termio.Exec.init(alloc, .{ + .command = config.command, + .shell_integration = config.@"shell-integration", + .shell_integration_features = config.@"shell-integration-features", + .working_directory = config.@"working-directory", + .resources_dir = main.state.resources_dir, + .term = config.term, + + // Get the cgroup if we're on linux and have the decl. I'd love + // to change this from a decl to a surface options struct because + // then we can do memory management better (don't need to retain + // the string around). + .linux_cgroup = if (comptime builtin.os.tag == .linux and + @hasDecl(apprt.runtime.Surface, "cgroup")) + rt_surface.cgroup() + else + Command.linux_cgroup_default, + }); + errdefer io_exec.deinit(); + + // Initialize our IO mailbox + var io_mailbox = try termio.Mailbox.initSPSC(alloc); + errdefer io_mailbox.deinit(alloc); + + try termio.Termio.init(&self.io, alloc, .{ + .grid_size = grid_size, + .screen_size = screen_size, + .padding = padding, + .full_config = config, + .config = try termio.Termio.DerivedConfig.init(alloc, config), + .backend = .{ .exec = io_exec }, + .mailbox = io_mailbox, + .renderer_state = &self.renderer_state, + .renderer_wakeup = render_thread.wakeup, + .renderer_mailbox = render_thread.mailbox, + .surface_mailbox = .{ .surface = self, .app = app_mailbox }, + }); + } + // Outside the block, IO has now taken ownership of our temporary state + // so we can just defer this and not the subcomponents. + errdefer self.io.deinit(); + // Report initial cell size on surface creation try rt_surface.setCellSize(cell_size.width, cell_size.height); @@ -483,7 +505,7 @@ pub fn init( self.io_thr = try std.Thread.spawn( .{}, termio.Thread.threadMain, - .{&self.io_thread}, + .{ &self.io_thread, &self.io }, ); self.io_thr.setName("io") catch {}; @@ -616,7 +638,7 @@ pub fn activateInspector(self: *Surface) !void { // Notify our components we have an inspector active _ = self.renderer_thread.mailbox.push(.{ .inspector = true }, .{ .forever = {} }); - _ = self.io_thread.mailbox.push(.{ .inspector = true }, .{ .forever = {} }); + self.io.queueMessage(.{ .inspector = true }, .unlocked); } /// Deactivate the inspector and stop collecting any information. @@ -633,7 +655,7 @@ pub fn deactivateInspector(self: *Surface) void { // Notify our components we have deactivated inspector _ = self.renderer_thread.mailbox.push(.{ .inspector = false }, .{ .forever = {} }); - _ = self.io_thread.mailbox.push(.{ .inspector = false }, .{ .forever = {} }); + self.io.queueMessage(.{ .inspector = false }, .unlocked); // Deinit the inspector insp.deinit(); @@ -733,8 +755,7 @@ fn reportColorScheme(self: *Surface) !void { .dark => "\x1B[?997;1n", }; - _ = self.io_thread.mailbox.push(.{ .write_stable = output }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + self.io.queueMessage(.{ .write_stable = output }, .unlocked); } /// Call this when modifiers change. This is safe to call even if modifiers @@ -809,26 +830,23 @@ fn changeConfig(self: *Surface, config: *const configpkg.Config) !void { // our messages aren't huge. var renderer_message = try renderer.Message.initChangeConfig(self.alloc, config); errdefer renderer_message.deinit(); - var termio_config_ptr = try self.alloc.create(termio.Impl.DerivedConfig); + var termio_config_ptr = try self.alloc.create(termio.Termio.DerivedConfig); errdefer self.alloc.destroy(termio_config_ptr); - termio_config_ptr.* = try termio.Impl.DerivedConfig.init(self.alloc, config); + termio_config_ptr.* = try termio.Termio.DerivedConfig.init(self.alloc, config); errdefer termio_config_ptr.deinit(); _ = self.renderer_thread.mailbox.push(renderer_message, .{ .forever = {} }); - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .change_config = .{ .alloc = self.alloc, .ptr = termio_config_ptr, }, - }, .{ .forever = {} }); + }, .unlocked); // With mailbox messages sent, we have to wake them up so they process it. self.queueRender() catch |err| { log.warn("failed to notify renderer of config change err={}", .{err}); }; - self.io_thread.wakeup.notify() catch |err| { - log.warn("failed to notify io thread of config change err={}", .{err}); - }; } /// Returns true if the terminal has a selection. @@ -1066,14 +1084,13 @@ fn setCellSize(self: *Surface, size: renderer.CellSize) !void { ); // Notify the terminal - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .resize = .{ .grid_size = self.grid_size, .screen_size = self.screen_size, .padding = self.padding, }, - }, .{ .forever = {} }); - self.io_thread.wakeup.notify() catch {}; + }, .unlocked); // Notify the window try self.rt_surface.setCellSize(size.width, size.height); @@ -1169,14 +1186,13 @@ fn resize(self: *Surface, size: renderer.ScreenSize) !void { } // Mail the IO thread - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .resize = .{ .grid_size = self.grid_size, .screen_size = self.screen_size, .padding = self.padding, }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); } /// Called to set the preedit state for character input. Preedit is used @@ -1542,12 +1558,11 @@ pub fn keyCallback( ev.pty = copy; } - _ = self.io_thread.mailbox.push(switch (write_req) { + self.io.queueMessage(switch (write_req) { .small => |v| .{ .write_small = v }, .stable => |v| .{ .write_stable = v }, .alloc => |v| .{ .write_alloc = v }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); // If our event is any keypress that isn't a modifier and we generated // some data to send to the pty, then we move the viewport down to the @@ -1647,11 +1662,7 @@ pub fn focusCallback(self: *Surface, focused: bool) !void { if (focus_event) { const seq = if (focused) "\x1b[I" else "\x1b[O"; - _ = self.io_thread.mailbox.push(.{ - .write_stable = seq, - }, .{ .forever = {} }); - - try self.io_thread.wakeup.notify(); + self.io.queueMessage(.{ .write_stable = seq }, .unlocked); } } } @@ -1786,14 +1797,10 @@ pub fn scrollCallback( break :seq if (y.delta < 0) "\x1b[A" else "\x1b[B"; }; for (0..y.delta_unsigned) |_| { - _ = self.io_thread.mailbox.push(.{ - .write_stable = seq, - }, .{ .instant = {} }); + self.io.queueMessage(.{ .write_stable = seq }, .locked); } } - // After sending all our messages we have to notify our IO thread - try self.io_thread.wakeup.notify(); return; } @@ -1995,12 +2002,10 @@ fn mouseReport( data[5] = 32 + @as(u8, @intCast(viewport_point.y)) + 1; // Ask our IO thread to write the data - _ = self.io_thread.mailbox.push(.{ - .write_small = .{ - .data = data, - .len = 6, - }, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_small = .{ + .data = data, + .len = 6, + } }, .locked); }, .utf8 => { @@ -2020,12 +2025,10 @@ fn mouseReport( i += try std.unicode.utf8Encode(@intCast(32 + viewport_point.y + 1), data[i..]); // Ask our IO thread to write the data - _ = self.io_thread.mailbox.push(.{ - .write_small = .{ - .data = data, - .len = @intCast(i), - }, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_small = .{ + .data = data, + .len = @intCast(i), + } }, .locked); }, .sgr => { @@ -2043,12 +2046,10 @@ fn mouseReport( }); // Ask our IO thread to write the data - _ = self.io_thread.mailbox.push(.{ - .write_small = .{ - .data = data, - .len = @intCast(resp.len), - }, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_small = .{ + .data = data, + .len = @intCast(resp.len), + } }, .locked); }, .urxvt => { @@ -2062,12 +2063,10 @@ fn mouseReport( }); // Ask our IO thread to write the data - _ = self.io_thread.mailbox.push(.{ - .write_small = .{ - .data = data, - .len = @intCast(resp.len), - }, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_small = .{ + .data = data, + .len = @intCast(resp.len), + } }, .locked); }, .sgr_pixels => { @@ -2085,17 +2084,12 @@ fn mouseReport( }); // Ask our IO thread to write the data - _ = self.io_thread.mailbox.push(.{ - .write_small = .{ - .data = data, - .len = @intCast(resp.len), - }, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_small = .{ + .data = data, + .len = @intCast(resp.len), + } }, .locked); }, } - - // After sending all our messages we have to notify our IO thread - try self.io_thread.wakeup.notify(); } /// Returns true if the shift modifier is allowed to be captured by modifier @@ -2496,9 +2490,7 @@ fn clickMoveCursor(self: *Surface, to: terminal.Pin) !void { break :arrow if (t.modes.get(.cursor_keys)) "\x1bOB" else "\x1b[B"; }; for (0..@abs(path.y)) |_| { - _ = self.io_thread.mailbox.push(.{ - .write_stable = arrow, - }, .{ .instant = {} }); + self.io.queueMessage(.{ .write_stable = arrow }, .locked); } } if (path.x != 0) { @@ -2508,13 +2500,9 @@ fn clickMoveCursor(self: *Surface, to: terminal.Pin) !void { break :arrow if (t.modes.get(.cursor_keys)) "\x1bOC" else "\x1b[C"; }; for (0..@abs(path.x)) |_| { - _ = self.io_thread.mailbox.push(.{ - .write_stable = arrow, - }, .{ .instant = {} }); + self.io.queueMessage(.{ .write_stable = arrow }, .locked); } } - - try self.io_thread.wakeup.notify(); } /// Returns the link at the given cursor position, if any. @@ -3188,11 +3176,10 @@ pub fn performBindingAction(self: *Surface, action: input.Binding.Action) !bool .esc => try std.fmt.bufPrint(&buf, "\x1b{s}", .{data}), else => unreachable, }; - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + self.io.queueMessage(try termio.Message.writeReq( self.alloc, full_data, - ), .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + ), .unlocked); // CSI/ESC triggers a scroll. { @@ -3216,11 +3203,10 @@ pub fn performBindingAction(self: *Surface, action: input.Binding.Action) !bool ); return true; }; - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + self.io.queueMessage(try termio.Message.writeReq( self.alloc, text, - ), .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + ), .unlocked); // Text triggers a scroll. { @@ -3250,16 +3236,10 @@ pub fn performBindingAction(self: *Surface, action: input.Binding.Action) !bool }; if (normal) { - _ = self.io_thread.mailbox.push(.{ - .write_stable = ck.normal, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_stable = ck.normal }, .unlocked); } else { - _ = self.io_thread.mailbox.push(.{ - .write_stable = ck.application, - }, .{ .forever = {} }); + self.io.queueMessage(.{ .write_stable = ck.application }, .unlocked); } - - try self.io_thread.wakeup.notify(); }, .reset => { @@ -3341,63 +3321,55 @@ pub fn performBindingAction(self: *Surface, action: input.Binding.Action) !bool if (self.io.terminal.active_screen == .alternate) return false; } - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .clear_screen = .{ .history = true }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_to_top => { - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .top = {} }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_to_bottom => { - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .bottom = {} }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_page_up => { const rows: isize = @intCast(self.grid_size.rows); - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .delta = -1 * rows }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_page_down => { const rows: isize = @intCast(self.grid_size.rows); - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .delta = rows }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_page_fractional => |fraction| { const rows: f32 = @floatFromInt(self.grid_size.rows); const delta: isize = @intFromFloat(@floor(fraction * rows)); - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .delta = delta }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .scroll_page_lines => |lines| { - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .scroll_viewport = .{ .delta = lines }, - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .jump_to_prompt => |delta| { - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .jump_to_prompt = @intCast(delta), - }, .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + }, .unlocked); }, .write_scrollback_file => write_scrollback_file: { @@ -3441,11 +3413,10 @@ pub fn performBindingAction(self: *Surface, action: input.Binding.Action) !bool var path_buf: [std.fs.MAX_PATH_BYTES]u8 = undefined; const path = try tmp_dir.dir.realpath("scrollback", &path_buf); - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + self.io.queueMessage(try termio.Message.writeReq( self.alloc, path, - ), .{ .forever = {} }); - try self.io_thread.wakeup.notify(); + ), .unlocked); }, .new_window => try self.app.newWindow(self.rt_app, .{ .parent = self }), @@ -3700,16 +3671,16 @@ fn completeClipboardPaste( if (critical.bracketed) { // If we're bracketd we write the data as-is to the terminal with // the bracketed paste escape codes around it. - _ = self.io_thread.mailbox.push(.{ + self.io.queueMessage(.{ .write_stable = "\x1B[200~", - }, .{ .forever = {} }); - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + }, .unlocked); + self.io.queueMessage(try termio.Message.writeReq( self.alloc, data, - ), .{ .forever = {} }); - _ = self.io_thread.mailbox.push(.{ + ), .unlocked); + self.io.queueMessage(.{ .write_stable = "\x1B[201~", - }, .{ .forever = {} }); + }, .unlocked); } else { // If its not bracketed the input bytes are indistinguishable from // keystrokes, so we must be careful. For example, we must replace @@ -3736,13 +3707,11 @@ fn completeClipboardPaste( len += 1; } - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + self.io.queueMessage(try termio.Message.writeReq( self.alloc, buf[0..len], - ), .{ .forever = {} }); + ), .unlocked); } - - try self.io_thread.wakeup.notify(); } fn completeClipboardReadOSC52( @@ -3784,11 +3753,10 @@ fn completeClipboardReadOSC52( const encoded = enc.encode(buf[prefix.len..], data); assert(encoded.len == size); - _ = self.io_thread.mailbox.push(try termio.Message.writeReq( + self.io.queueMessage(try termio.Message.writeReq( self.alloc, buf, - ), .{ .forever = {} }); - self.io_thread.wakeup.notify() catch {}; + ), .unlocked); } fn showDesktopNotification(self: *Surface, title: [:0]const u8, body: [:0]const u8) !void { diff --git a/src/termio.zig b/src/termio.zig index 45382cda1..b4943cca9 100644 --- a/src/termio.zig +++ b/src/termio.zig @@ -1,18 +1,35 @@ -//! IO implementation and utilities. The IO implementation is responsible -//! for taking the config, spinning up a child process, and handling IO -//! with the terminal. +//! Termio is responsible for "terminal IO." Specifically, this is the +//! reading and writing of bytes for the underlying pty or pty-like device. +//! +//! Termio is constructed of a few components: +//! - Termio - The main shared struct that has common logic across all +//! backends and mailboxes (defined below). +//! - Backend - Responsible for the actual physical IO. For example, one +//! implementation creates a subprocess, allocates and assigns a pty, +//! and sets up a read thread on the pty. +//! - Mailbox - Responsible for storing/dispensing event messages to +//! the backend. This exists separately from backends because termio +//! is built to be both single and multi-threaded. +//! +//! Termio supports (and recommends) multi-threaded operation. Multi-threading +//! enables the read/writes to generally happen on separate threads and +//! almost always improves throughput and latency under heavy IO load. To +//! enable threading, use the Thread struct. This wraps a Termio, requires +//! specific backend/mailbox capabilities, and sets up the necessary threads. + +const stream_handler = @import("termio/stream_handler.zig"); pub usingnamespace @import("termio/message.zig"); +pub const backend = @import("termio/backend.zig"); +pub const mailbox = @import("termio/mailbox.zig"); pub const Exec = @import("termio/Exec.zig"); pub const Options = @import("termio/Options.zig"); +pub const Termio = @import("termio/Termio.zig"); pub const Thread = @import("termio/Thread.zig"); -pub const Mailbox = Thread.Mailbox; - -/// The implementation to use for the IO. This is just "exec" for now but -/// this is somewhat pluggable so that in the future we can introduce other -/// options for other platforms (i.e. wasm) or even potentially a vtable -/// implementation for runtime polymorphism. -pub const Impl = Exec; +pub const Backend = backend.Backend; +pub const DerivedConfig = Termio.DerivedConfig; +pub const Mailbox = mailbox.Mailbox; +pub const StreamHandler = stream_handler.StreamHandler; test { @import("std").testing.refAllDecls(@This()); diff --git a/src/termio/Exec.zig b/src/termio/Exec.zig index b0f973b1e..65607c67f 100644 --- a/src/termio/Exec.zig +++ b/src/termio/Exec.zig @@ -1,230 +1,102 @@ -//! Implementation of IO that uses child exec to talk to the child process. -pub const Exec = @This(); +//! Exec implements the logic for starting and stopping a subprocess with a +//! pty as well as spinning up the necessary read thread to read from the +//! pty and forward it to the Termio instance. +const Exec = @This(); const std = @import("std"); const builtin = @import("builtin"); -const build_config = @import("../build_config.zig"); const assert = std.debug.assert; const Allocator = std.mem.Allocator; const ArenaAllocator = std.heap.ArenaAllocator; -const EnvMap = std.process.EnvMap; const posix = std.posix; -const termio = @import("../termio.zig"); -const Command = @import("../Command.zig"); -const Pty = @import("../pty.zig").Pty; -const SegmentedPool = @import("../segmented_pool.zig").SegmentedPool; -const terminal = @import("../terminal/main.zig"); -const terminfo = @import("../terminfo/main.zig"); const xev = @import("xev"); -const renderer = @import("../renderer.zig"); -const apprt = @import("../apprt.zig"); +const build_config = @import("../build_config.zig"); +const configpkg = @import("../config.zig"); const fastmem = @import("../fastmem.zig"); const internal_os = @import("../os/main.zig"); -const windows = internal_os.windows; -const configpkg = @import("../config.zig"); +const renderer = @import("../renderer.zig"); const shell_integration = @import("shell_integration.zig"); +const terminal = @import("../terminal/main.zig"); +const termio = @import("../termio.zig"); +const Command = @import("../Command.zig"); +const SegmentedPool = @import("../segmented_pool.zig").SegmentedPool; +const Pty = @import("../pty.zig").Pty; +const EnvMap = std.process.EnvMap; +const windows = internal_os.windows; const log = std.log.scoped(.io_exec); -const c = @cImport({ - @cInclude("errno.h"); - @cInclude("signal.h"); - @cInclude("unistd.h"); -}); - -/// True if we should disable the kitty keyboard protocol. We have to -/// disable this on GLFW because GLFW input events don't support the -/// correct granularity of events. -const disable_kitty_keyboard_protocol = apprt.runtime == apprt.glfw; - -/// Allocator -alloc: Allocator, - -/// This is the pty fd created for the subcommand. +/// The subprocess state for our exec backend. subprocess: Subprocess, -/// The derived configuration for this termio implementation. -config: DerivedConfig, - -/// The terminal emulator internal state. This is the abstract "terminal" -/// that manages input, grid updating, etc. and is renderer-agnostic. It -/// just stores internal state about a grid. -terminal: terminal.Terminal, - -/// The shared render state -renderer_state: *renderer.State, - -/// A handle to wake up the renderer. This hints to the renderer that that -/// a repaint should happen. -renderer_wakeup: xev.Async, - -/// The mailbox for notifying the renderer of things. -renderer_mailbox: *renderer.Thread.Mailbox, - -/// The mailbox for communicating with the surface. -surface_mailbox: apprt.surface.Mailbox, - -/// The cached grid size whenever a resize is called. -grid_size: renderer.GridSize, - -/// The data associated with the currently running thread. -data: ?*EventData, - -/// The configuration for this IO that is derived from the main -/// configuration. This must be exported so that we don't need to -/// pass around Config pointers which makes memory management a pain. -pub const DerivedConfig = struct { - arena: ArenaAllocator, - - palette: terminal.color.Palette, - image_storage_limit: usize, - cursor_style: terminal.CursorStyle, - cursor_blink: ?bool, - cursor_color: ?configpkg.Config.Color, - foreground: configpkg.Config.Color, - background: configpkg.Config.Color, - osc_color_report_format: configpkg.Config.OSCColorReportFormat, - term: []const u8, - grapheme_width_method: configpkg.Config.GraphemeWidthMethod, - abnormal_runtime_threshold_ms: u32, - wait_after_command: bool, - enquiry_response: []const u8, - - pub fn init( - alloc_gpa: Allocator, - config: *const configpkg.Config, - ) !DerivedConfig { - var arena = ArenaAllocator.init(alloc_gpa); - errdefer arena.deinit(); - const alloc = arena.allocator(); - - return .{ - .palette = config.palette.value, - .image_storage_limit = config.@"image-storage-limit", - .cursor_style = config.@"cursor-style", - .cursor_blink = config.@"cursor-style-blink", - .cursor_color = config.@"cursor-color", - .foreground = config.foreground, - .background = config.background, - .osc_color_report_format = config.@"osc-color-report-format", - .term = try alloc.dupe(u8, config.term), - .grapheme_width_method = config.@"grapheme-width-method", - .abnormal_runtime_threshold_ms = config.@"abnormal-command-exit-runtime", - .wait_after_command = config.@"wait-after-command", - .enquiry_response = try alloc.dupe(u8, config.@"enquiry-response"), - - // This has to be last so that we copy AFTER the arena allocations - // above happen (Zig assigns in order). - .arena = arena, - }; - } - - pub fn deinit(self: *DerivedConfig) void { - self.arena.deinit(); - } -}; - -/// Initialize the exec implementation. This will also start the child -/// process. -pub fn init(alloc: Allocator, opts: termio.Options) !Exec { - // Create our terminal - var term = try terminal.Terminal.init(alloc, .{ - .cols = opts.grid_size.columns, - .rows = opts.grid_size.rows, - .max_scrollback = opts.full_config.@"scrollback-limit", - }); - errdefer term.deinit(alloc); - term.default_palette = opts.config.palette; - term.color_palette.colors = opts.config.palette; - - // Setup our initial grapheme cluster support if enabled. We use a - // switch to ensure we get a compiler error if more cases are added. - switch (opts.config.grapheme_width_method) { - .unicode => term.modes.set(.grapheme_cluster, true), - .legacy => {}, - } - - // Set the image size limits - try term.screen.kitty_images.setLimit( - alloc, - &term.screen, - opts.config.image_storage_limit, - ); - try term.secondary_screen.kitty_images.setLimit( - alloc, - &term.secondary_screen, - opts.config.image_storage_limit, - ); - - // Set default cursor blink settings - term.modes.set( - .cursor_blinking, - opts.config.cursor_blink orelse true, - ); - - // Set our default cursor style - term.screen.cursor.cursor_style = opts.config.cursor_style; - - var subprocess = try Subprocess.init(alloc, opts); +/// Initialize the exec state. This will NOT start it, this only sets +/// up the internal state necessary to start it later. +pub fn init( + alloc: Allocator, + cfg: Config, +) !Exec { + var subprocess = try Subprocess.init(alloc, cfg); errdefer subprocess.deinit(); - // If we have an initial pwd requested by the subprocess, then we - // set that on the terminal now. This allows rapidly initializing - // new surfaces to use the proper pwd. - if (subprocess.cwd) |cwd| term.setPwd(cwd) catch |err| { - log.warn("error setting initial pwd err={}", .{err}); - }; - - // Initial width/height based on subprocess - term.width_px = subprocess.screen_size.width; - term.height_px = subprocess.screen_size.height; - - return .{ - .alloc = alloc, - .terminal = term, - .subprocess = subprocess, - .config = opts.config, - .renderer_state = opts.renderer_state, - .renderer_wakeup = opts.renderer_wakeup, - .renderer_mailbox = opts.renderer_mailbox, - .surface_mailbox = opts.surface_mailbox, - .grid_size = opts.grid_size, - .data = null, - }; + return .{ .subprocess = subprocess }; } pub fn deinit(self: *Exec) void { self.subprocess.deinit(); - self.terminal.deinit(self.alloc); - self.config.deinit(); } -pub fn threadEnter(self: *Exec, thread: *termio.Thread) !ThreadData { - assert(self.data == null); - const alloc = self.alloc; +/// Call to initialize the terminal state as necessary for this backend. +/// This is called before any termio begins. This should not be called +/// after termio begins because it may put the internal terminal state +/// into a bad state. +pub fn initTerminal(self: *Exec, term: *terminal.Terminal) void { + // If we have an initial pwd requested by the subprocess, then we + // set that on the terminal now. This allows rapidly initializing + // new surfaces to use the proper pwd. + if (self.subprocess.cwd) |cwd| term.setPwd(cwd) catch |err| { + log.warn("error setting initial pwd err={}", .{err}); + }; + // Setup our initial grid/screen size from the terminal. This + // can't fail because the pty should not exist at this point. + self.resize(.{ + .columns = term.cols, + .rows = term.rows, + }, .{ + .width = term.width_px, + .height = term.height_px, + }) catch unreachable; +} + +pub fn threadEnter( + self: *Exec, + alloc: Allocator, + io: *termio.Termio, + td: *termio.Termio.ThreadData, +) !void { // Start our subprocess const pty_fds = self.subprocess.start(alloc) catch |err| { // If we specifically got this error then we are in the forked // process and our child failed to execute. In that case - if (err != error.ExecFailedInChild) return err; + if (err != error.Termio) return err; // Output an error message about the exec faililng and exit. // This generally should NOT happen because we always wrap // our command execution either in login (macOS) or /bin/sh // (Linux) which are usually guaranteed to exist. Still, we // want to handle this scenario. - self.execFailedInChild() catch {}; + execFailedInChild() catch {}; posix.exit(1); }; errdefer self.subprocess.stop(); + + // Get the pid from the subprocess const pid = pid: { const command = self.subprocess.command orelse return error.ProcessNotStarted; break :pid command.pid orelse return error.ProcessNoPid; }; - // Track our process start time so we know how long it was - // running for. + // Track our process start time for abnormal exits const process_start = try std.time.Instant.now(); // Create our pipe that we'll use to kill our read thread. @@ -233,123 +105,60 @@ pub fn threadEnter(self: *Exec, thread: *termio.Thread) !ThreadData { errdefer posix.close(pipe[0]); errdefer posix.close(pipe[1]); - // Setup our data that is used for callbacks - var ev_data_ptr = try alloc.create(EventData); - errdefer alloc.destroy(ev_data_ptr); - // Setup our stream so that we can write. var stream = xev.Stream.initFd(pty_fds.write); errdefer stream.deinit(); - // Wakeup watcher for the writer thread. - var wakeup = try xev.Async.init(); - errdefer wakeup.deinit(); - // Watcher to detect subprocess exit var process = try xev.Process.init(pid); errdefer process.deinit(); - // Setup our event data before we start - ev_data_ptr.* = .{ - .writer_mailbox = thread.mailbox, - .writer_wakeup = thread.wakeup, - .surface_mailbox = self.surface_mailbox, - .renderer_state = self.renderer_state, - .renderer_wakeup = self.renderer_wakeup, - .renderer_mailbox = self.renderer_mailbox, - .process = process, - .process_start = process_start, - .data_stream = stream, - .loop = &thread.loop, - .terminal_stream = .{ - .handler = StreamHandler.init( - self.alloc, - ev_data_ptr, - &self.grid_size, - &self.terminal, - &self.config, - ), - .parser = .{ - .osc_parser = .{ - // Populate the OSC parser allocator (optional) because - // we want to support large OSC payloads such as OSC 52. - .alloc = self.alloc, - }, - }, - }, - .abnormal_runtime_threshold_ms = self.config.abnormal_runtime_threshold_ms, - .wait_after_command = self.config.wait_after_command, - }; - errdefer ev_data_ptr.deinit(self.alloc); - - // Store our data so our callbacks can access it - self.data = ev_data_ptr; - errdefer self.data = null; - - // Start our process watcher - process.wait( - ev_data_ptr.loop, - &ev_data_ptr.process_wait_c, - EventData, - ev_data_ptr, - processExit, - ); - - // Start our reader thread + // Start our read thread const read_thread = try std.Thread.spawn( .{}, if (builtin.os.tag == .windows) ReadThread.threadMainWindows else ReadThread.threadMainPosix, - .{ pty_fds.read, ev_data_ptr, pipe[0] }, + .{ pty_fds.read, io, pipe[0] }, ); read_thread.setName("io-reader") catch {}; - // Return our thread data - return ThreadData{ - .alloc = alloc, - .ev = ev_data_ptr, + // Setup our threadata backend state to be our own + td.backend = .{ .exec = .{ + .start = process_start, + .abnormal_runtime_threshold_ms = io.config.abnormal_runtime_threshold_ms, + .wait_after_command = io.config.wait_after_command, + .write_stream = stream, + .process = process, .read_thread = read_thread, .read_thread_pipe = pipe[1], .read_thread_fd = if (builtin.os.tag == .windows) pty_fds.read else {}, - }; + } }; + + // Start our process watcher + process.wait( + td.loop, + &td.backend.exec.process_wait_c, + termio.Termio.ThreadData, + td, + processExit, + ); } -/// This outputs an error message when exec failed and we are the -/// child process. This returns so the caller should probably exit -/// after calling this. -/// -/// Note that this usually is only called under very very rare -/// circumstances because we wrap our command execution in login -/// (macOS) or /bin/sh (Linux). So this output can be pretty crude -/// because it should never happen. Notably, this is not the error -/// users see when `command` is invalid. -fn execFailedInChild(self: *Exec) !void { - _ = self; - const stderr = std.io.getStdErr().writer(); - try stderr.writeAll("exec failed\n"); - try stderr.writeAll("press any key to exit\n"); +pub fn threadExit(self: *Exec, td: *termio.Termio.ThreadData) void { + assert(td.backend == .exec); + const exec = &td.backend.exec; - var buf: [1]u8 = undefined; - var reader = std.io.getStdIn().reader(); - _ = try reader.read(&buf); -} - -pub fn threadExit(self: *Exec, data: ThreadData) void { - // Clear out our data since we're not active anymore. - self.data = null; - - // Stop our subprocess - if (data.ev.process_exited) self.subprocess.externalExit(); + if (exec.exited) self.subprocess.externalExit(); self.subprocess.stop(); // Quit our read thread after exiting the subprocess so that // we don't get stuck waiting for data to stop flowing if it is // a particularly noisy process. - _ = posix.write(data.read_thread_pipe, "x") catch |err| + _ = posix.write(exec.read_thread_pipe, "x") catch |err| log.warn("error writing to read thread quit pipe err={}", .{err}); if (comptime builtin.os.tag == .windows) { // Interrupt the blocking read so the thread can see the quit message - if (windows.kernel32.CancelIoEx(data.read_thread_fd, null) == 0) { + if (windows.kernel32.CancelIoEx(exec.read_thread_fd, null) == 0) { switch (windows.kernel32.GetLastError()) { .NOT_FOUND => {}, else => |err| log.warn("error interrupting read thread err={}", .{err}), @@ -357,173 +166,27 @@ pub fn threadExit(self: *Exec, data: ThreadData) void { } } - data.read_thread.join(); + exec.read_thread.join(); } -/// Update the configuration. -pub fn changeConfig(self: *Exec, config: *DerivedConfig) !void { - // The remainder of this function is modifying terminal state or - // the read thread data, all of which requires holding the renderer - // state lock. - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - - // Deinit our old config. We do this in the lock because the - // stream handler may be referencing the old config (i.e. enquiry resp) - self.config.deinit(); - self.config = config.*; - - // Update our stream handler. The stream handler uses the same - // renderer mutex so this is safe to do despite being executed - // from another thread. - if (self.data) |data| { - data.abnormal_runtime_threshold_ms = config.abnormal_runtime_threshold_ms; - data.wait_after_command = config.wait_after_command; - data.terminal_stream.handler.changeConfig(&self.config); - } - - // Update the configuration that we know about. - // - // Specific things we don't update: - // - command, working-directory: we never restart the underlying - // process so we don't care or need to know about these. - - // Update the default palette. Note this will only apply to new colors drawn - // since we decode all palette colors to RGB on usage. - self.terminal.default_palette = config.palette; - - // Update the active palette, except for any colors that were modified with - // OSC 4 - for (0..config.palette.len) |i| { - if (!self.terminal.color_palette.mask.isSet(i)) { - self.terminal.color_palette.colors[i] = config.palette[i]; - self.terminal.flags.dirty.palette = true; - } - } - - // Set the image size limits - try self.terminal.screen.kitty_images.setLimit( - self.alloc, - &self.terminal.screen, - config.image_storage_limit, - ); - try self.terminal.secondary_screen.kitty_images.setLimit( - self.alloc, - &self.terminal.secondary_screen, - config.image_storage_limit, - ); -} - -/// Resize the terminal. pub fn resize( self: *Exec, grid_size: renderer.GridSize, screen_size: renderer.ScreenSize, - padding: renderer.Padding, ) !void { - // Update the size of our pty. - const padded_size = screen_size.subPadding(padding); - try self.subprocess.resize(grid_size, padded_size); - - // Update our cached grid size - self.grid_size = grid_size; - - // Enter the critical area that we want to keep small - { - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - - // Update the size of our terminal state - try self.terminal.resize( - self.alloc, - grid_size.columns, - grid_size.rows, - ); - - // Update our pixel sizes - self.terminal.width_px = padded_size.width; - self.terminal.height_px = padded_size.height; - - // Disable synchronized output mode so that we show changes - // immediately for a resize. This is allowed by the spec. - self.terminal.modes.set(.synchronized_output, false); - - // Wake up our renderer so any changes will be shown asap - self.renderer_wakeup.notify() catch {}; - } + return try self.subprocess.resize(grid_size, screen_size); } -/// Reset the synchronized output mode. This is usually called by timer -/// expiration from the termio thread. -pub fn resetSynchronizedOutput(self: *Exec) void { - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - self.terminal.modes.set(.synchronized_output, false); - self.renderer_wakeup.notify() catch {}; -} - -/// Clear the screen. -pub fn clearScreen(self: *Exec, history: bool) !void { - { - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - - // If we're on the alternate screen, we do not clear. Since this is an - // emulator-level screen clear, this messes up the running programs - // knowledge of where the cursor is and causes rendering issues. So, - // for alt screen, we do nothing. - if (self.terminal.active_screen == .alternate) return; - - // Clear our scrollback - if (history) self.terminal.eraseDisplay(.scrollback, false); - - // If we're not at a prompt, we just delete above the cursor. - if (!self.terminal.cursorIsAtPrompt()) { - if (self.terminal.screen.cursor.y > 0) { - self.terminal.screen.eraseRows( - .{ .active = .{ .y = 0 } }, - .{ .active = .{ .y = self.terminal.screen.cursor.y - 1 } }, - ); - } - - return; - } - - // At a prompt, we want to first fully clear the screen, and then after - // send a FF (0x0C) to the shell so that it can repaint the screen. - // Mark the current row as a not a prompt so we can properly - // clear the full screen in the next eraseDisplay call. - self.terminal.markSemanticPrompt(.command); - assert(!self.terminal.cursorIsAtPrompt()); - self.terminal.eraseDisplay(.complete, false); - } - - // If we reached here it means we're at a prompt, so we send a form-feed. - try self.queueWrite(&[_]u8{0x0C}, false); -} - -/// Scroll the viewport -pub fn scrollViewport(self: *Exec, scroll: terminal.Terminal.ScrollViewport) !void { - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - try self.terminal.scrollViewport(scroll); -} - -/// Jump the viewport to the prompt. -pub fn jumpToPrompt(self: *Exec, delta: isize) !void { - { - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - self.terminal.screen.scroll(.{ .delta_prompt = delta }); - } - - try self.renderer_wakeup.notify(); -} - -/// Called when the child process exited abnormally but before -/// the surface is notified. -pub fn childExitedAbnormally(self: *Exec, exit_code: u32, runtime_ms: u64) !void { - var arena = ArenaAllocator.init(self.alloc); +/// Called when the child process exited abnormally but before the surface +/// is notified. +pub fn childExitedAbnormally( + self: *Exec, + gpa: Allocator, + t: *terminal.Terminal, + exit_code: u32, + runtime_ms: u64, +) !void { + var arena = ArenaAllocator.init(gpa); defer arena.deinit(); const alloc = arena.allocator(); @@ -531,12 +194,6 @@ pub fn childExitedAbnormally(self: *Exec, exit_code: u32, runtime_ms: u64) !void const command = try std.mem.join(alloc, " ", self.subprocess.args); const runtime_str = try std.fmt.allocPrint(alloc, "{d} ms", .{runtime_ms}); - // Modify the terminal to show our error message. This - // requires grabbing the renderer state lock. - self.renderer_state.mutex.lock(); - defer self.renderer_state.mutex.unlock(); - const t = self.renderer_state.terminal; - // No matter what move the cursor back to the column 0. t.carriageReturn(); @@ -597,13 +254,124 @@ pub fn childExitedAbnormally(self: *Exec, exit_code: u32, runtime_ms: u64) !void t.modes.set(.cursor_visible, false); } -pub inline fn queueWrite(self: *Exec, data: []const u8, linefeed: bool) !void { - const ev = self.data.?; +/// This outputs an error message when exec failed and we are the +/// child process. This returns so the caller should probably exit +/// after calling this. +/// +/// Note that this usually is only called under very very rare +/// circumstances because we wrap our command execution in login +/// (macOS) or /bin/sh (Linux). So this output can be pretty crude +/// because it should never happen. Notably, this is not the error +/// users see when `command` is invalid. +fn execFailedInChild() !void { + const stderr = std.io.getStdErr().writer(); + try stderr.writeAll("exec failed\n"); + try stderr.writeAll("press any key to exit\n"); + + var buf: [1]u8 = undefined; + var reader = std.io.getStdIn().reader(); + _ = try reader.read(&buf); +} + +fn processExit( + td_: ?*termio.Termio.ThreadData, + _: *xev.Loop, + _: *xev.Completion, + r: xev.Process.WaitError!u32, +) xev.CallbackAction { + const exit_code = r catch unreachable; + + const td = td_.?; + assert(td.backend == .exec); + const execdata = &td.backend.exec; + execdata.exited = true; + + // Determine how long the process was running for. + const runtime_ms: ?u64 = runtime: { + const process_end = std.time.Instant.now() catch break :runtime null; + const runtime_ns = process_end.since(execdata.start); + const runtime_ms = runtime_ns / std.time.ns_per_ms; + break :runtime runtime_ms; + }; + log.debug( + "child process exited status={} runtime={}ms", + .{ exit_code, runtime_ms orelse 0 }, + ); + + // If our runtime was below some threshold then we assume that this + // was an abnormal exit and we show an error message. + if (runtime_ms) |runtime| runtime: { + // On macOS, our exit code detection doesn't work, possibly + // because of our `login` wrapper. More investigation required. + if (comptime !builtin.target.isDarwin()) { + // If our exit code is zero, then the command was successful + // and we don't ever consider it abnormal. + if (exit_code == 0) break :runtime; + } + + // Our runtime always has to be under the threshold to be + // considered abnormal. This is because a user can always + // manually do something like `exit 1` in their shell to + // force the exit code to be non-zero. We only want to detect + // abnormal exits that happen so quickly the user can't react. + if (runtime > execdata.abnormal_runtime_threshold_ms) break :runtime; + log.warn("abnormal process exit detected, showing error message", .{}); + + // Notify our main writer thread which has access to more + // information so it can show a better error message. + td.mailbox.send(.{ + .child_exited_abnormally = .{ + .exit_code = exit_code, + .runtime_ms = runtime, + }, + }, null); + td.mailbox.notify(); + + return .disarm; + } + + // If we're purposely waiting then we just return since the process + // exited flag is set to true. This allows the terminal window to remain + // open. + if (execdata.wait_after_command) { + // We output a message so that the user knows whats going on and + // doesn't think their terminal just froze. + terminal: { + td.renderer_state.mutex.lock(); + defer td.renderer_state.mutex.unlock(); + const t = td.renderer_state.terminal; + t.carriageReturn(); + t.linefeed() catch break :terminal; + t.printString("Process exited. Press any key to close the terminal.") catch + break :terminal; + t.modes.set(.cursor_visible, false); + } + + return .disarm; + } + + // Notify our surface we want to close + _ = td.surface_mailbox.push(.{ + .child_exited = {}, + }, .{ .forever = {} }); + + return .disarm; +} + +pub fn queueWrite( + self: *Exec, + alloc: Allocator, + td: *termio.Termio.ThreadData, + data: []const u8, + linefeed: bool, +) !void { + _ = self; + const exec = &td.backend.exec; // If our process is exited then we send our surface a message // about it but we don't queue any more writes. - if (ev.process_exited) { - _ = ev.surface_mailbox.push(.{ + if (exec.exited) { + _ = td.surface_mailbox.push(.{ .child_exited = {}, }, .{ .forever = {} }); return; @@ -613,8 +381,8 @@ pub inline fn queueWrite(self: *Exec, data: []const u8, linefeed: bool) !void { // our cached buffers that we can queue to the stream. var i: usize = 0; while (i < data.len) { - const req = try ev.write_req_pool.getGrow(self.alloc); - const buf = try ev.write_buf_pool.getGrow(self.alloc); + const req = try exec.write_req_pool.getGrow(alloc); + const buf = try exec.write_buf_pool.getGrow(alloc); const slice = slice: { // The maximum end index is either the end of our data or // the end of our buffer, whichever is smaller. @@ -651,226 +419,29 @@ pub inline fn queueWrite(self: *Exec, data: []const u8, linefeed: bool) !void { //for (slice) |b| log.warn("write: {x}", .{b}); - ev.data_stream.queueWrite( - ev.loop, - &ev.write_queue, + exec.write_stream.queueWrite( + td.loop, + &exec.write_queue, req, .{ .slice = slice }, - EventData, - ev, + termio.Exec.ThreadData, + exec, ttyWrite, ); } } -const ThreadData = struct { - /// Allocator used for the event data - alloc: Allocator, - - /// The data that is attached to the callbacks. - ev: *EventData, - - /// Our read thread - read_thread: std.Thread, - read_thread_pipe: posix.fd_t, - read_thread_fd: if (builtin.os.tag == .windows) posix.fd_t else void, - - pub fn deinit(self: *ThreadData) void { - posix.close(self.read_thread_pipe); - self.ev.deinit(self.alloc); - self.alloc.destroy(self.ev); - self.* = undefined; - } -}; - -const EventData = struct { - // The preallocation size for the write request pool. This should be big - // enough to satisfy most write requests. It must be a power of 2. - const WRITE_REQ_PREALLOC = std.math.pow(usize, 2, 5); - - /// Mailbox for data to the writer thread. - writer_mailbox: *termio.Mailbox, - writer_wakeup: xev.Async, - - /// Mailbox for the surface. - surface_mailbox: apprt.surface.Mailbox, - - /// The stream parser. This parses the stream of escape codes and so on - /// from the child process and calls callbacks in the stream handler. - terminal_stream: terminal.Stream(StreamHandler), - - /// The shared render state - renderer_state: *renderer.State, - - /// A handle to wake up the renderer. This hints to the renderer that that - /// a repaint should happen. - renderer_wakeup: xev.Async, - - /// The mailbox for notifying the renderer of things. - renderer_mailbox: *renderer.Thread.Mailbox, - - /// The process watcher - process: xev.Process, - process_start: std.time.Instant, - process_exited: bool = false, - - /// This is used for both waiting for the process to exit and then - /// subsequently to wait for the data_stream to close. - process_wait_c: xev.Completion = .{}, - - /// The data stream is the main IO for the pty. - data_stream: xev.Stream, - - /// The event loop, - loop: *xev.Loop, - - /// The write queue for the data stream. - write_queue: xev.Stream.WriteQueue = .{}, - - /// This is the pool of available (unused) write requests. If you grab - /// one from the pool, you must put it back when you're done! - write_req_pool: SegmentedPool(xev.Stream.WriteRequest, WRITE_REQ_PREALLOC) = .{}, - - /// The pool of available buffers for writing to the pty. - write_buf_pool: SegmentedPool([64]u8, WRITE_REQ_PREALLOC) = .{}, - - /// Last time the cursor was reset. This is used to prevent message - /// flooding with cursor resets. - last_cursor_reset: i64 = 0, - - /// This is set to true when we've seen a title escape sequence. We use - /// this to determine if we need to default the window title. - seen_title: bool = false, - - /// The number of milliseconds below which we consider a process - /// exit to be abnormal. This is used to show an error message - /// when the process exits too quickly. - abnormal_runtime_threshold_ms: u32, - - /// If true, do not immediately send a child exited message to the - /// surface to close the surface when the command exits. - wait_after_command: bool, - - pub fn deinit(self: *EventData, alloc: Allocator) void { - // Clear our write pools. We know we aren't ever going to do - // any more IO since we stop our data stream below so we can just - // drop this. - self.write_req_pool.deinit(alloc); - self.write_buf_pool.deinit(alloc); - - // Stop our data stream - self.data_stream.deinit(); - - // Stop our process watcher - self.process.deinit(); - - // Clear any StreamHandler state - self.terminal_stream.handler.deinit(); - self.terminal_stream.deinit(); - } - - /// This queues a render operation with the renderer thread. The render - /// isn't guaranteed to happen immediately but it will happen as soon as - /// practical. - inline fn queueRender(self: *EventData) !void { - try self.renderer_wakeup.notify(); - } -}; - -fn processExit( - ev_: ?*EventData, - _: *xev.Loop, - _: *xev.Completion, - r: xev.Process.WaitError!u32, -) xev.CallbackAction { - const exit_code = r catch unreachable; - - const ev = ev_.?; - ev.process_exited = true; - - // Determine how long the process was running for. - const runtime_ms: ?u64 = runtime: { - const process_end = std.time.Instant.now() catch break :runtime null; - const runtime_ns = process_end.since(ev.process_start); - const runtime_ms = runtime_ns / std.time.ns_per_ms; - break :runtime runtime_ms; - }; - log.debug( - "child process exited status={} runtime={}ms", - .{ exit_code, runtime_ms orelse 0 }, - ); - - // If our runtime was below some threshold then we assume that this - // was an abnormal exit and we show an error message. - if (runtime_ms) |runtime| runtime: { - // On macOS, our exit code detection doesn't work, possibly - // because of our `login` wrapper. More investigation required. - if (comptime !builtin.target.isDarwin()) { - // If our exit code is zero, then the command was successful - // and we don't ever consider it abnormal. - if (exit_code == 0) break :runtime; - } - - // Our runtime always has to be under the threshold to be - // considered abnormal. This is because a user can always - // manually do something like `exit 1` in their shell to - // force the exit code to be non-zero. We only want to detect - // abnormal exits that happen so quickly the user can't react. - if (runtime > ev.abnormal_runtime_threshold_ms) break :runtime; - log.warn("abnormal process exit detected, showing error message", .{}); - - // Notify our main writer thread which has access to more - // information so it can show a better error message. - _ = ev.writer_mailbox.push(.{ - .child_exited_abnormally = .{ - .exit_code = exit_code, - .runtime_ms = runtime, - }, - }, .{ .forever = {} }); - ev.writer_wakeup.notify() catch break :runtime; - - return .disarm; - } - - // If we're purposely waiting then we just return since the process - // exited flag is set to true. This allows the terminal window to remain - // open. - if (ev.wait_after_command) { - // We output a message so that the user knows whats going on and - // doesn't think their terminal just froze. - terminal: { - ev.renderer_state.mutex.lock(); - defer ev.renderer_state.mutex.unlock(); - const t = ev.renderer_state.terminal; - t.carriageReturn(); - t.linefeed() catch break :terminal; - t.printString("Process exited. Press any key to close the terminal.") catch - break :terminal; - t.modes.set(.cursor_visible, false); - } - - return .disarm; - } - - // Notify our surface we want to close - _ = ev.surface_mailbox.push(.{ - .child_exited = {}, - }, .{ .forever = {} }); - - return .disarm; -} - fn ttyWrite( - ev_: ?*EventData, + td_: ?*ThreadData, _: *xev.Loop, _: *xev.Completion, _: xev.Stream, _: xev.WriteBuffer, r: xev.Stream.WriteError!usize, ) xev.CallbackAction { - const ev = ev_.?; - ev.write_req_pool.put(); - ev.write_buf_pool.put(); + const td = td_.?; + td.write_req_pool.put(); + td.write_buf_pool.put(); const d = r catch |err| { log.err("write error: {}", .{err}); @@ -882,12 +453,90 @@ fn ttyWrite( return .disarm; } -/// Subprocess manages the lifecycle of the shell subprocess. +/// The thread local data for the exec implementation. +pub const ThreadData = struct { + // The preallocation size for the write request pool. This should be big + // enough to satisfy most write requests. It must be a power of 2. + const WRITE_REQ_PREALLOC = std.math.pow(usize, 2, 5); + + /// Process start time and boolean of whether its already exited. + start: std.time.Instant, + exited: bool = false, + + /// The number of milliseconds below which we consider a process + /// exit to be abnormal. This is used to show an error message + /// when the process exits too quickly. + abnormal_runtime_threshold_ms: u32, + + /// If true, do not immediately send a child exited message to the + /// surface to close the surface when the command exits. If this is + /// false we'll show a process exited message and wait for user input + /// to close the surface. + wait_after_command: bool, + + /// The data stream is the main IO for the pty. + write_stream: xev.Stream, + + /// The process watcher + process: xev.Process, + + /// This is the pool of available (unused) write requests. If you grab + /// one from the pool, you must put it back when you're done! + write_req_pool: SegmentedPool(xev.Stream.WriteRequest, WRITE_REQ_PREALLOC) = .{}, + + /// The pool of available buffers for writing to the pty. + write_buf_pool: SegmentedPool([64]u8, WRITE_REQ_PREALLOC) = .{}, + + /// The write queue for the data stream. + write_queue: xev.Stream.WriteQueue = .{}, + + /// This is used for both waiting for the process to exit and then + /// subsequently to wait for the data_stream to close. + process_wait_c: xev.Completion = .{}, + + /// Reader thread state + read_thread: std.Thread, + read_thread_pipe: posix.fd_t, + read_thread_fd: if (builtin.os.tag == .windows) posix.fd_t else void, + + pub fn deinit(self: *ThreadData, alloc: Allocator) void { + posix.close(self.read_thread_pipe); + + // Clear our write pools. We know we aren't ever going to do + // any more IO since we stop our data stream below so we can just + // drop this. + self.write_req_pool.deinit(alloc); + self.write_buf_pool.deinit(alloc); + + // Stop our process watcher + self.process.deinit(); + + // Stop our write stream + self.write_stream.deinit(); + } +}; + +pub const Config = struct { + command: ?[]const u8 = null, + shell_integration: configpkg.Config.ShellIntegration = .detect, + shell_integration_features: configpkg.Config.ShellIntegrationFeatures = .{}, + working_directory: ?[]const u8 = null, + resources_dir: ?[]const u8, + term: []const u8, + linux_cgroup: Command.LinuxCgroup = Command.linux_cgroup_default, +}; + const Subprocess = struct { /// If we build with flatpak support then we have to keep track of /// a potential execution on the host. const FlatpakHostCommand = if (build_config.flatpak) internal_os.FlatpakHostCommand else void; + const c = @cImport({ + @cInclude("errno.h"); + @cInclude("signal.h"); + @cInclude("unistd.h"); + }); + arena: std.heap.ArenaAllocator, cwd: ?[]const u8, env: EnvMap, @@ -901,7 +550,7 @@ const Subprocess = struct { /// Initialize the subprocess. This will NOT start it, this only sets /// up the internal state necessary to start it later. - pub fn init(gpa: Allocator, opts: termio.Options) !Subprocess { + pub fn init(gpa: Allocator, cfg: Config) !Subprocess { // We have a lot of maybe-allocations that all share the same lifetime // so use an arena so we don't end up in an accounting nightmare. var arena = std.heap.ArenaAllocator.init(gpa); @@ -923,7 +572,7 @@ const Subprocess = struct { errdefer env.deinit(); // If we have a resources dir then set our env var - if (opts.resources_dir) |dir| { + if (cfg.resources_dir) |dir| { log.info("found Ghostty resources dir: {s}", .{dir}); try env.put("GHOSTTY_RESOURCES_DIR", dir); } @@ -934,8 +583,8 @@ const Subprocess = struct { // // For now, we just look up a bundled dir but in the future we should // also load the terminfo database and look for it. - if (opts.resources_dir) |base| { - try env.put("TERM", opts.config.term); + if (cfg.resources_dir) |base| { + try env.put("TERM", cfg.term); try env.put("COLORTERM", "truecolor"); // Assume that the resources directory is adjacent to the terminfo @@ -992,7 +641,7 @@ const Subprocess = struct { // Add the man pages from our application bundle to MANPATH. if (comptime builtin.target.isDarwin()) { - if (opts.resources_dir) |resources_dir| man: { + if (cfg.resources_dir) |resources_dir| man: { var buf: [std.fs.MAX_PATH_BYTES]u8 = undefined; const dir = std.fmt.bufPrint(&buf, "{s}/../man", .{resources_dir}) catch |err| { log.warn("error building manpath, man pages may not be available err={}", .{err}); @@ -1044,12 +693,12 @@ const Subprocess = struct { // Setup our shell integration, if we can. const integrated_shell: ?shell_integration.Shell, const shell_command: []const u8 = shell: { - const default_shell_command = opts.full_config.command orelse switch (builtin.os.tag) { + const default_shell_command = cfg.command orelse switch (builtin.os.tag) { .windows => "cmd.exe", else => "sh", }; - const force: ?shell_integration.Shell = switch (opts.full_config.@"shell-integration") { + const force: ?shell_integration.Shell = switch (cfg.shell_integration) { .none => break :shell .{ null, default_shell_command }, .detect => null, .bash => .bash, @@ -1058,7 +707,7 @@ const Subprocess = struct { .zsh => .zsh, }; - const dir = opts.resources_dir orelse break :shell .{ + const dir = cfg.resources_dir orelse break :shell .{ null, default_shell_command, }; @@ -1069,7 +718,7 @@ const Subprocess = struct { default_shell_command, &env, force, - opts.full_config.@"shell-integration-features", + cfg.shell_integration_features, ) orelse break :shell .{ null, default_shell_command }; break :shell .{ integration.shell, integration.command }; @@ -1080,7 +729,7 @@ const Subprocess = struct { "shell integration automatically injected shell={}", .{shell}, ); - } else if (opts.full_config.@"shell-integration" != .none) { + } else if (cfg.shell_integration != .none) { log.warn("shell could not be detected, no automatic shell integration will be injected", .{}); } @@ -1217,7 +866,7 @@ const Subprocess = struct { // We have to copy the cwd because there is no guarantee that // pointers in full_config remain valid. - const cwd: ?[]u8 = if (opts.full_config.@"working-directory") |cwd| + const cwd: ?[]u8 = if (cfg.working_directory) |cwd| try alloc.dupe(u8, cwd) else null; @@ -1227,21 +876,20 @@ const Subprocess = struct { const linux_cgroup: Command.LinuxCgroup = cgroup: { const default = Command.linux_cgroup_default; if (comptime builtin.os.tag != .linux) break :cgroup default; - const path = opts.linux_cgroup orelse break :cgroup default; + const path = cfg.linux_cgroup orelse break :cgroup default; break :cgroup try alloc.dupe(u8, path); }; - // Our screen size should be our padded size - const padded_size = opts.screen_size.subPadding(opts.padding); - return .{ .arena = arena, .env = env, .cwd = cwd, .args = args, - .grid_size = opts.grid_size, - .screen_size = padded_size, .linux_cgroup = linux_cgroup, + + // Should be initialized with initTerminal call. + .grid_size = .{}, + .screen_size = .{ .width = 1, .height = 1 }, }; } @@ -1527,8 +1175,8 @@ const Subprocess = struct { /// We use a basic poll syscall here because we are only monitoring two /// fds and this is still much faster and lower overhead than any async /// mechanism. -const ReadThread = struct { - fn threadMainPosix(fd: posix.fd_t, ev: *EventData, quit: posix.fd_t) void { +pub const ReadThread = struct { + fn threadMainPosix(fd: posix.fd_t, io: *termio.Termio, quit: posix.fd_t) void { // Always close our end of the pipe when we exit. defer posix.close(quit); @@ -1592,7 +1240,7 @@ const ReadThread = struct { if (n == 0) break; // log.info("DATA: {d}", .{n}); - @call(.always_inline, process, .{ ev, buf[0..n] }); + @call(.always_inline, termio.Termio.processOutput, .{ io, buf[0..n] }); } // Wait for data. @@ -1609,7 +1257,7 @@ const ReadThread = struct { } } - fn threadMainWindows(fd: posix.fd_t, ev: *EventData, quit: posix.fd_t) void { + fn threadMainWindows(fd: posix.fd_t, io: *termio.Termio, quit: posix.fd_t) void { // Always close our end of the pipe when we exit. defer posix.close(quit); @@ -1630,7 +1278,7 @@ const ReadThread = struct { } } - @call(.always_inline, process, .{ ev, buf[0..n] }); + @call(.always_inline, termio.Termio.processOutput, .{ io, buf[0..n] }); } var quit_bytes: windows.DWORD = 0; @@ -1646,1328 +1294,4 @@ const ReadThread = struct { } } } - - fn process( - ev: *EventData, - buf: []const u8, - ) void { - // log.info("DATA: {d}", .{n}); - // log.info("DATA: {any}", .{buf[0..@intCast(usize, n)]}); - - // Whenever a character is typed, we ensure the cursor is in the - // non-blink state so it is rendered if visible. If we're under - // HEAVY read load, we don't want to send a ton of these so we - // use a timer under the covers - const now = ev.loop.now(); - if (now - ev.last_cursor_reset > 500) { - ev.last_cursor_reset = now; - _ = ev.renderer_mailbox.push(.{ - .reset_cursor_blink = {}, - }, .{ .forever = {} }); - } - - // We are modifying terminal state from here on out - ev.renderer_state.mutex.lock(); - defer ev.renderer_state.mutex.unlock(); - - // Schedule a render - ev.queueRender() catch unreachable; - - // If we have an inspector, we enter SLOW MODE because we need to - // process a byte at a time alternating between the inspector handler - // and the termio handler. This is very slow compared to our optimizations - // below but at least users only pay for it if they're using the inspector. - if (ev.renderer_state.inspector) |insp| { - for (buf, 0..) |byte, i| { - insp.recordPtyRead(buf[i .. i + 1]) catch |err| { - log.err("error recording pty read in inspector err={}", .{err}); - }; - - ev.terminal_stream.next(byte) catch |err| - log.err("error processing terminal data: {}", .{err}); - } - } else { - ev.terminal_stream.nextSlice(buf) catch |err| - log.err("error processing terminal data: {}", .{err}); - } - - // If our stream handling caused messages to be sent to the writer - // thread, then we need to wake it up so that it processes them. - if (ev.terminal_stream.handler.writer_messaged) { - ev.terminal_stream.handler.writer_messaged = false; - ev.writer_wakeup.notify() catch |err| { - log.warn("failed to wake up writer thread err={}", .{err}); - }; - } - } -}; - -/// This is used as the handler for the terminal.Stream type. This is -/// stateful and is expected to live for the entire lifetime of the terminal. -/// It is NOT VALID to stop a stream handler, create a new one, and use that -/// unless all of the member fields are copied. -const StreamHandler = struct { - ev: *EventData, - alloc: Allocator, - grid_size: *renderer.GridSize, - terminal: *terminal.Terminal, - - /// The APC command handler maintains the APC state. APC is like - /// CSI or OSC, but it is a private escape sequence that is used - /// to send commands to the terminal emulator. This is used by - /// the kitty graphics protocol. - apc: terminal.apc.Handler = .{}, - - /// The DCS handler maintains DCS state. DCS is like CSI or OSC, - /// but requires more stateful parsing. This is used by functionality - /// such as XTGETTCAP. - dcs: terminal.dcs.Handler = .{}, - - /// This is set to true when a message was written to the writer - /// mailbox. This can be used by callers to determine if they need - /// to wake up the writer. - writer_messaged: bool = false, - - /// The default cursor state. This is used with CSI q. This is - /// set to true when we're currently in the default cursor state. - default_cursor: bool = true, - default_cursor_style: terminal.CursorStyle, - default_cursor_blink: ?bool, - default_cursor_color: ?terminal.color.RGB, - - /// Actual cursor color. This can be changed with OSC 12. - cursor_color: ?terminal.color.RGB, - - /// The default foreground and background color are those set by the user's - /// config file. These can be overridden by terminal applications using OSC - /// 10 and OSC 11, respectively. - default_foreground_color: terminal.color.RGB, - default_background_color: terminal.color.RGB, - - /// The actual foreground and background color. Normally this will be the - /// same as the default foreground and background color, unless changed by a - /// terminal application. - foreground_color: terminal.color.RGB, - background_color: terminal.color.RGB, - - /// The response to use for ENQ requests. The memory is owned by - /// whoever owns StreamHandler. - enquiry_response: []const u8, - - osc_color_report_format: configpkg.Config.OSCColorReportFormat, - - pub fn init( - alloc: Allocator, - ev: *EventData, - grid_size: *renderer.GridSize, - t: *terminal.Terminal, - config: *const DerivedConfig, - ) StreamHandler { - const default_cursor_color = if (config.cursor_color) |col| - col.toTerminalRGB() - else - null; - - return .{ - .alloc = alloc, - .ev = ev, - .grid_size = grid_size, - .terminal = t, - .osc_color_report_format = config.osc_color_report_format, - .enquiry_response = config.enquiry_response, - .default_foreground_color = config.foreground.toTerminalRGB(), - .default_background_color = config.background.toTerminalRGB(), - .default_cursor_style = config.cursor_style, - .default_cursor_blink = config.cursor_blink, - .default_cursor_color = default_cursor_color, - .cursor_color = default_cursor_color, - .foreground_color = config.foreground.toTerminalRGB(), - .background_color = config.background.toTerminalRGB(), - }; - } - - pub fn deinit(self: *StreamHandler) void { - self.apc.deinit(); - self.dcs.deinit(); - } - - /// Change the configuration for this handler. - pub fn changeConfig(self: *StreamHandler, config: *DerivedConfig) void { - self.osc_color_report_format = config.osc_color_report_format; - self.enquiry_response = config.enquiry_response; - self.default_foreground_color = config.foreground.toTerminalRGB(); - self.default_background_color = config.background.toTerminalRGB(); - self.default_cursor_style = config.cursor_style; - self.default_cursor_blink = config.cursor_blink; - self.default_cursor_color = if (config.cursor_color) |col| - col.toTerminalRGB() - else - null; - - // If our cursor is the default, then we update it immediately. - if (self.default_cursor) self.setCursorStyle(.default) catch |err| { - log.warn("failed to set default cursor style: {}", .{err}); - }; - } - - inline fn queueRender(self: *StreamHandler) !void { - try self.ev.queueRender(); - } - - inline fn surfaceMessageWriter( - self: *StreamHandler, - msg: apprt.surface.Message, - ) void { - // See messageWriter which has similar logic and explains why - // we may have to do this. - if (self.ev.surface_mailbox.push(msg, .{ .instant = {} }) == 0) { - self.ev.renderer_state.mutex.unlock(); - defer self.ev.renderer_state.mutex.lock(); - _ = self.ev.surface_mailbox.push(msg, .{ .forever = {} }); - } - } - - inline fn messageWriter(self: *StreamHandler, msg: termio.Message) void { - // Try to write to the mailbox with an instant timeout. This is the - // fast path because we can queue without a lock. - if (self.ev.writer_mailbox.push(msg, .{ .instant = {} }) == 0) { - // If we enter this conditional, the mailbox is full. We wake up - // the writer thread so that it can process messages to clear up - // space. However, the writer thread may require the renderer - // lock so we need to unlock. - self.ev.writer_wakeup.notify() catch |err| { - log.warn("failed to wake up writer, data will be dropped err={}", .{err}); - return; - }; - - // Unlock the renderer state so the writer thread can acquire it. - // Then try to queue our message before continuing. This is a very - // slow path because we are having a lot of contention for data. - // But this only gets triggered in certain pathological cases. - // - // Note that writes themselves don't require a lock, but there - // are other messages in the writer mailbox (resize, focus) that - // could acquire the lock. This is why we have to release our lock - // here. - self.ev.renderer_state.mutex.unlock(); - defer self.ev.renderer_state.mutex.lock(); - _ = self.ev.writer_mailbox.push(msg, .{ .forever = {} }); - } - - // Normally, we just flag this true to wake up the writer thread - // once per batch of data. - self.writer_messaged = true; - } - - pub fn dcsHook(self: *StreamHandler, dcs: terminal.DCS) !void { - var cmd = self.dcs.hook(self.alloc, dcs) orelse return; - defer cmd.deinit(); - try self.dcsCommand(&cmd); - } - - pub fn dcsPut(self: *StreamHandler, byte: u8) !void { - var cmd = self.dcs.put(byte) orelse return; - defer cmd.deinit(); - try self.dcsCommand(&cmd); - } - - pub fn dcsUnhook(self: *StreamHandler) !void { - var cmd = self.dcs.unhook() orelse return; - defer cmd.deinit(); - try self.dcsCommand(&cmd); - } - - fn dcsCommand(self: *StreamHandler, cmd: *terminal.dcs.Command) !void { - // log.warn("DCS command: {}", .{cmd}); - switch (cmd.*) { - .tmux => |tmux| { - // TODO: process it - log.warn("tmux control mode event unimplemented cmd={}", .{tmux}); - }, - - .xtgettcap => |*gettcap| { - const map = comptime terminfo.ghostty.xtgettcapMap(); - while (gettcap.next()) |key| { - const response = map.get(key) orelse continue; - self.messageWriter(.{ .write_stable = response }); - } - }, - - .decrqss => |decrqss| { - var response: [128]u8 = undefined; - var stream = std.io.fixedBufferStream(&response); - const writer = stream.writer(); - - // Offset the stream position to just past the response prefix. - // We will write the "payload" (if any) below. If no payload is - // written then we send an invalid DECRPSS response. - const prefix_fmt = "\x1bP{d}$r"; - const prefix_len = std.fmt.comptimePrint(prefix_fmt, .{0}).len; - stream.pos = prefix_len; - - switch (decrqss) { - // Invalid or unhandled request - .none => {}, - - .sgr => { - const buf = try self.terminal.printAttributes(stream.buffer[stream.pos..]); - - // printAttributes wrote into our buffer, so adjust the stream - // position - stream.pos += buf.len; - - try writer.writeByte('m'); - }, - - .decscusr => { - const blink = self.terminal.modes.get(.cursor_blinking); - const style: u8 = switch (self.terminal.screen.cursor.cursor_style) { - .block => if (blink) 1 else 2, - .underline => if (blink) 3 else 4, - .bar => if (blink) 5 else 6, - }; - try writer.print("{d} q", .{style}); - }, - - .decstbm => { - try writer.print("{d};{d}r", .{ - self.terminal.scrolling_region.top + 1, - self.terminal.scrolling_region.bottom + 1, - }); - }, - - .decslrm => { - // We only send a valid response when left and right - // margin mode (DECLRMM) is enabled. - if (self.terminal.modes.get(.enable_left_and_right_margin)) { - try writer.print("{d};{d}s", .{ - self.terminal.scrolling_region.left + 1, - self.terminal.scrolling_region.right + 1, - }); - } - }, - } - - // Our response is valid if we have a response payload - const valid = stream.pos > prefix_len; - - // Write the terminator - try writer.writeAll("\x1b\\"); - - // Write the response prefix into the buffer - _ = try std.fmt.bufPrint(response[0..prefix_len], prefix_fmt, .{@intFromBool(valid)}); - const msg = try termio.Message.writeReq(self.alloc, response[0..stream.pos]); - self.messageWriter(msg); - }, - } - } - - pub fn apcStart(self: *StreamHandler) !void { - self.apc.start(); - } - - pub fn apcPut(self: *StreamHandler, byte: u8) !void { - self.apc.feed(self.alloc, byte); - } - - pub fn apcEnd(self: *StreamHandler) !void { - var cmd = self.apc.end() orelse return; - defer cmd.deinit(self.alloc); - - // log.warn("APC command: {}", .{cmd}); - switch (cmd) { - .kitty => |*kitty_cmd| { - if (self.terminal.kittyGraphics(self.alloc, kitty_cmd)) |resp| { - var buf: [1024]u8 = undefined; - var buf_stream = std.io.fixedBufferStream(&buf); - try resp.encode(buf_stream.writer()); - const final = buf_stream.getWritten(); - if (final.len > 2) { - // log.warn("kitty graphics response: {s}", .{std.fmt.fmtSliceHexLower(final)}); - self.messageWriter(try termio.Message.writeReq(self.alloc, final)); - } - } - }, - } - } - - pub fn print(self: *StreamHandler, ch: u21) !void { - try self.terminal.print(ch); - } - - pub fn printRepeat(self: *StreamHandler, count: usize) !void { - try self.terminal.printRepeat(count); - } - - pub fn bell(self: StreamHandler) !void { - _ = self; - log.info("BELL", .{}); - } - - pub fn backspace(self: *StreamHandler) !void { - self.terminal.backspace(); - } - - pub fn horizontalTab(self: *StreamHandler, count: u16) !void { - for (0..count) |_| { - const x = self.terminal.screen.cursor.x; - try self.terminal.horizontalTab(); - if (x == self.terminal.screen.cursor.x) break; - } - } - - pub fn horizontalTabBack(self: *StreamHandler, count: u16) !void { - for (0..count) |_| { - const x = self.terminal.screen.cursor.x; - try self.terminal.horizontalTabBack(); - if (x == self.terminal.screen.cursor.x) break; - } - } - - pub fn linefeed(self: *StreamHandler) !void { - // Small optimization: call index instead of linefeed because they're - // identical and this avoids one layer of function call overhead. - try self.terminal.index(); - } - - pub fn carriageReturn(self: *StreamHandler) !void { - self.terminal.carriageReturn(); - } - - pub fn setCursorLeft(self: *StreamHandler, amount: u16) !void { - self.terminal.cursorLeft(amount); - } - - pub fn setCursorRight(self: *StreamHandler, amount: u16) !void { - self.terminal.cursorRight(amount); - } - - pub fn setCursorDown(self: *StreamHandler, amount: u16, carriage: bool) !void { - self.terminal.cursorDown(amount); - if (carriage) self.terminal.carriageReturn(); - } - - pub fn setCursorUp(self: *StreamHandler, amount: u16, carriage: bool) !void { - self.terminal.cursorUp(amount); - if (carriage) self.terminal.carriageReturn(); - } - - pub fn setCursorCol(self: *StreamHandler, col: u16) !void { - self.terminal.setCursorPos(self.terminal.screen.cursor.y + 1, col); - } - - pub fn setCursorColRelative(self: *StreamHandler, offset: u16) !void { - self.terminal.setCursorPos( - self.terminal.screen.cursor.y + 1, - self.terminal.screen.cursor.x + 1 +| offset, - ); - } - - pub fn setCursorRow(self: *StreamHandler, row: u16) !void { - self.terminal.setCursorPos(row, self.terminal.screen.cursor.x + 1); - } - - pub fn setCursorRowRelative(self: *StreamHandler, offset: u16) !void { - self.terminal.setCursorPos( - self.terminal.screen.cursor.y + 1 +| offset, - self.terminal.screen.cursor.x + 1, - ); - } - - pub fn setCursorPos(self: *StreamHandler, row: u16, col: u16) !void { - self.terminal.setCursorPos(row, col); - } - - pub fn eraseDisplay(self: *StreamHandler, mode: terminal.EraseDisplay, protected: bool) !void { - if (mode == .complete) { - // Whenever we erase the full display, scroll to bottom. - try self.terminal.scrollViewport(.{ .bottom = {} }); - try self.queueRender(); - } - - self.terminal.eraseDisplay(mode, protected); - } - - pub fn eraseLine(self: *StreamHandler, mode: terminal.EraseLine, protected: bool) !void { - self.terminal.eraseLine(mode, protected); - } - - pub fn deleteChars(self: *StreamHandler, count: usize) !void { - self.terminal.deleteChars(count); - } - - pub fn eraseChars(self: *StreamHandler, count: usize) !void { - self.terminal.eraseChars(count); - } - - pub fn insertLines(self: *StreamHandler, count: usize) !void { - self.terminal.insertLines(count); - } - - pub fn insertBlanks(self: *StreamHandler, count: usize) !void { - self.terminal.insertBlanks(count); - } - - pub fn deleteLines(self: *StreamHandler, count: usize) !void { - self.terminal.deleteLines(count); - } - - pub fn reverseIndex(self: *StreamHandler) !void { - self.terminal.reverseIndex(); - } - - pub fn index(self: *StreamHandler) !void { - try self.terminal.index(); - } - - pub fn nextLine(self: *StreamHandler) !void { - try self.terminal.index(); - self.terminal.carriageReturn(); - } - - pub fn setTopAndBottomMargin(self: *StreamHandler, top: u16, bot: u16) !void { - self.terminal.setTopAndBottomMargin(top, bot); - } - - pub fn setLeftAndRightMarginAmbiguous(self: *StreamHandler) !void { - if (self.terminal.modes.get(.enable_left_and_right_margin)) { - try self.setLeftAndRightMargin(0, 0); - } else { - try self.saveCursor(); - } - } - - pub fn setLeftAndRightMargin(self: *StreamHandler, left: u16, right: u16) !void { - self.terminal.setLeftAndRightMargin(left, right); - } - - pub fn setModifyKeyFormat(self: *StreamHandler, format: terminal.ModifyKeyFormat) !void { - self.terminal.flags.modify_other_keys_2 = false; - switch (format) { - .other_keys => |v| switch (v) { - .numeric => self.terminal.flags.modify_other_keys_2 = true, - else => {}, - }, - else => {}, - } - } - - pub fn requestMode(self: *StreamHandler, mode_raw: u16, ansi: bool) !void { - // Get the mode value and respond. - const code: u8 = code: { - const mode = terminal.modes.modeFromInt(mode_raw, ansi) orelse break :code 0; - if (self.terminal.modes.get(mode)) break :code 1; - break :code 2; - }; - - var msg: termio.Message = .{ .write_small = .{} }; - const resp = try std.fmt.bufPrint( - &msg.write_small.data, - "\x1B[{s}{};{}$y", - .{ - if (ansi) "" else "?", - mode_raw, - code, - }, - ); - msg.write_small.len = @intCast(resp.len); - self.messageWriter(msg); - } - - pub fn saveMode(self: *StreamHandler, mode: terminal.Mode) !void { - // log.debug("save mode={}", .{mode}); - self.terminal.modes.save(mode); - } - - pub fn restoreMode(self: *StreamHandler, mode: terminal.Mode) !void { - // For restore mode we have to restore but if we set it, we - // always have to call setMode because setting some modes have - // side effects and we want to make sure we process those. - const v = self.terminal.modes.restore(mode); - // log.debug("restore mode={} v={}", .{ mode, v }); - try self.setMode(mode, v); - } - - pub fn setMode(self: *StreamHandler, mode: terminal.Mode, enabled: bool) !void { - // Note: this function doesn't need to grab the render state or - // terminal locks because it is only called from process() which - // grabs the lock. - - // If we are setting cursor blinking, we ignore it if we have - // a default cursor blink setting set. This is a really weird - // behavior so this comment will go deep into trying to explain it. - // - // There are two ways to set cursor blinks: DECSCUSR (CSI _ q) - // and DEC mode 12. DECSCUSR is the modern approach and has a - // way to revert to the "default" (as defined by the terminal) - // cursor style and blink by doing "CSI 0 q". DEC mode 12 controls - // blinking and is either on or off and has no way to set a - // default. DEC mode 12 is also the more antiquated approach. - // - // The problem is that if the user specifies a desired default - // cursor blink with `cursor-style-blink`, the moment a running - // program uses DEC mode 12, the cursor blink can never be reset - // to the default without an explicit DECSCUSR. But if a program - // is using mode 12, it is by definition not using DECSCUSR. - // This makes for somewhat annoying interactions where a poorly - // (or legacy) behaved program will stop blinking, and it simply - // never restarts. - // - // To get around this, we have a special case where if the user - // specifies some explicit default cursor blink desire, we ignore - // DEC mode 12. We allow DECSCUSR to still set the cursor blink - // because programs using DECSCUSR usually are well behaved and - // reset the cursor blink to the default when they exit. - // - // To be extra safe, users can also add a manual `CSI 0 q` to - // their shell config when they render prompts to ensure the - // cursor is exactly as they request. - if (mode == .cursor_blinking and - self.default_cursor_blink != null) - { - return; - } - - // We first always set the raw mode on our mode state. - self.terminal.modes.set(mode, enabled); - - // And then some modes require additional processing. - switch (mode) { - // Just noting here that autorepeat has no effect on - // the terminal. xterm ignores this mode and so do we. - // We know about just so that we don't log that it is - // an unknown mode. - .autorepeat => {}, - - // Schedule a render since we changed colors - .reverse_colors => { - self.terminal.flags.dirty.reverse_colors = true; - try self.queueRender(); - }, - - // Origin resets cursor pos. This is called whether or not - // we're enabling or disabling origin mode and whether or - // not the value changed. - .origin => self.terminal.setCursorPos(1, 1), - - .enable_left_and_right_margin => if (!enabled) { - // When we disable left/right margin mode we need to - // reset the left/right margins. - self.terminal.scrolling_region.left = 0; - self.terminal.scrolling_region.right = self.terminal.cols - 1; - }, - - .alt_screen => { - const opts: terminal.Terminal.AlternateScreenOptions = .{ - .cursor_save = false, - .clear_on_enter = false, - }; - - if (enabled) - self.terminal.alternateScreen(opts) - else - self.terminal.primaryScreen(opts); - - // Schedule a render since we changed screens - try self.queueRender(); - }, - - .alt_screen_save_cursor_clear_enter => { - const opts: terminal.Terminal.AlternateScreenOptions = .{ - .cursor_save = true, - .clear_on_enter = true, - }; - - if (enabled) - self.terminal.alternateScreen(opts) - else - self.terminal.primaryScreen(opts); - - // Schedule a render since we changed screens - try self.queueRender(); - }, - - // Force resize back to the window size - .enable_mode_3 => self.terminal.resize( - self.alloc, - self.grid_size.columns, - self.grid_size.rows, - ) catch |err| { - log.err("error updating terminal size: {}", .{err}); - }, - - .@"132_column" => try self.terminal.deccolm( - self.alloc, - if (enabled) .@"132_cols" else .@"80_cols", - ), - - // We need to start a timer to prevent the emulator being hung - // forever. - .synchronized_output => { - if (enabled) self.messageWriter(.{ .start_synchronized_output = {} }); - try self.queueRender(); - }, - - .linefeed => { - self.messageWriter(.{ .linefeed_mode = enabled }); - }, - - .mouse_event_x10 => { - if (enabled) { - self.terminal.flags.mouse_event = .x10; - try self.setMouseShape(.default); - } else { - self.terminal.flags.mouse_event = .none; - try self.setMouseShape(.text); - } - }, - .mouse_event_normal => { - if (enabled) { - self.terminal.flags.mouse_event = .normal; - try self.setMouseShape(.default); - } else { - self.terminal.flags.mouse_event = .none; - try self.setMouseShape(.text); - } - }, - .mouse_event_button => { - if (enabled) { - self.terminal.flags.mouse_event = .button; - try self.setMouseShape(.default); - } else { - self.terminal.flags.mouse_event = .none; - try self.setMouseShape(.text); - } - }, - .mouse_event_any => { - if (enabled) { - self.terminal.flags.mouse_event = .any; - try self.setMouseShape(.default); - } else { - self.terminal.flags.mouse_event = .none; - try self.setMouseShape(.text); - } - }, - - .mouse_format_utf8 => self.terminal.flags.mouse_format = if (enabled) .utf8 else .x10, - .mouse_format_sgr => self.terminal.flags.mouse_format = if (enabled) .sgr else .x10, - .mouse_format_urxvt => self.terminal.flags.mouse_format = if (enabled) .urxvt else .x10, - .mouse_format_sgr_pixels => self.terminal.flags.mouse_format = if (enabled) .sgr_pixels else .x10, - - else => {}, - } - } - - pub fn setMouseShiftCapture(self: *StreamHandler, v: bool) !void { - self.terminal.flags.mouse_shift_capture = if (v) .true else .false; - } - - pub fn setAttribute(self: *StreamHandler, attr: terminal.Attribute) !void { - switch (attr) { - .unknown => |unk| log.warn("unimplemented or unknown SGR attribute: {any}", .{unk}), - - else => self.terminal.setAttribute(attr) catch |err| - log.warn("error setting attribute {}: {}", .{ attr, err }), - } - } - - pub fn startHyperlink(self: *StreamHandler, uri: []const u8, id: ?[]const u8) !void { - try self.terminal.screen.startHyperlink(uri, id); - } - - pub fn endHyperlink(self: *StreamHandler) !void { - self.terminal.screen.endHyperlink(); - } - - pub fn deviceAttributes( - self: *StreamHandler, - req: terminal.DeviceAttributeReq, - params: []const u16, - ) !void { - _ = params; - - // For the below, we quack as a VT220. We don't quack as - // a 420 because we don't support DCS sequences. - switch (req) { - .primary => self.messageWriter(.{ - .write_stable = "\x1B[?62;22c", - }), - - .secondary => self.messageWriter(.{ - .write_stable = "\x1B[>1;10;0c", - }), - - else => log.warn("unimplemented device attributes req: {}", .{req}), - } - } - - pub fn deviceStatusReport( - self: *StreamHandler, - req: terminal.device_status.Request, - ) !void { - switch (req) { - .operating_status => self.messageWriter(.{ .write_stable = "\x1B[0n" }), - - .cursor_position => { - const pos: struct { - x: usize, - y: usize, - } = if (self.terminal.modes.get(.origin)) .{ - .x = self.terminal.screen.cursor.x -| self.terminal.scrolling_region.left, - .y = self.terminal.screen.cursor.y -| self.terminal.scrolling_region.top, - } else .{ - .x = self.terminal.screen.cursor.x, - .y = self.terminal.screen.cursor.y, - }; - - // Response always is at least 4 chars, so this leaves the - // remainder for the row/column as base-10 numbers. This - // will support a very large terminal. - var msg: termio.Message = .{ .write_small = .{} }; - const resp = try std.fmt.bufPrint(&msg.write_small.data, "\x1B[{};{}R", .{ - pos.y + 1, - pos.x + 1, - }); - msg.write_small.len = @intCast(resp.len); - - self.messageWriter(msg); - }, - - .color_scheme => self.surfaceMessageWriter(.{ .report_color_scheme = {} }), - } - } - - pub fn setCursorStyle( - self: *StreamHandler, - style: terminal.CursorStyleReq, - ) !void { - // Assume we're setting to a non-default. - self.default_cursor = false; - - switch (style) { - .default => { - self.default_cursor = true; - self.terminal.screen.cursor.cursor_style = self.default_cursor_style; - self.terminal.modes.set( - .cursor_blinking, - self.default_cursor_blink orelse true, - ); - }, - - .blinking_block => { - self.terminal.screen.cursor.cursor_style = .block; - self.terminal.modes.set(.cursor_blinking, true); - }, - - .steady_block => { - self.terminal.screen.cursor.cursor_style = .block; - self.terminal.modes.set(.cursor_blinking, false); - }, - - .blinking_underline => { - self.terminal.screen.cursor.cursor_style = .underline; - self.terminal.modes.set(.cursor_blinking, true); - }, - - .steady_underline => { - self.terminal.screen.cursor.cursor_style = .underline; - self.terminal.modes.set(.cursor_blinking, false); - }, - - .blinking_bar => { - self.terminal.screen.cursor.cursor_style = .bar; - self.terminal.modes.set(.cursor_blinking, true); - }, - - .steady_bar => { - self.terminal.screen.cursor.cursor_style = .bar; - self.terminal.modes.set(.cursor_blinking, false); - }, - - else => log.warn("unimplemented cursor style: {}", .{style}), - } - } - - pub fn setProtectedMode(self: *StreamHandler, mode: terminal.ProtectedMode) !void { - self.terminal.setProtectedMode(mode); - } - - pub fn decaln(self: *StreamHandler) !void { - try self.terminal.decaln(); - } - - pub fn tabClear(self: *StreamHandler, cmd: terminal.TabClear) !void { - self.terminal.tabClear(cmd); - } - - pub fn tabSet(self: *StreamHandler) !void { - self.terminal.tabSet(); - } - - pub fn tabReset(self: *StreamHandler) !void { - self.terminal.tabReset(); - } - - pub fn saveCursor(self: *StreamHandler) !void { - self.terminal.saveCursor(); - } - - pub fn restoreCursor(self: *StreamHandler) !void { - try self.terminal.restoreCursor(); - } - - pub fn enquiry(self: *StreamHandler) !void { - log.debug("sending enquiry response={s}", .{self.enquiry_response}); - self.messageWriter(try termio.Message.writeReq(self.alloc, self.enquiry_response)); - } - - pub fn scrollDown(self: *StreamHandler, count: usize) !void { - self.terminal.scrollDown(count); - } - - pub fn scrollUp(self: *StreamHandler, count: usize) !void { - self.terminal.scrollUp(count); - } - - pub fn setActiveStatusDisplay( - self: *StreamHandler, - req: terminal.StatusDisplay, - ) !void { - self.terminal.status_display = req; - } - - pub fn configureCharset( - self: *StreamHandler, - slot: terminal.CharsetSlot, - set: terminal.Charset, - ) !void { - self.terminal.configureCharset(slot, set); - } - - pub fn invokeCharset( - self: *StreamHandler, - active: terminal.CharsetActiveSlot, - slot: terminal.CharsetSlot, - single: bool, - ) !void { - self.terminal.invokeCharset(active, slot, single); - } - - pub fn fullReset( - self: *StreamHandler, - ) !void { - self.terminal.fullReset(); - try self.setMouseShape(.text); - } - - pub fn queryKittyKeyboard(self: *StreamHandler) !void { - if (comptime disable_kitty_keyboard_protocol) return; - - log.debug("querying kitty keyboard mode", .{}); - var data: termio.Message.WriteReq.Small.Array = undefined; - const resp = try std.fmt.bufPrint(&data, "\x1b[?{}u", .{ - self.terminal.screen.kitty_keyboard.current().int(), - }); - - self.messageWriter(.{ - .write_small = .{ - .data = data, - .len = @intCast(resp.len), - }, - }); - } - - pub fn pushKittyKeyboard( - self: *StreamHandler, - flags: terminal.kitty.KeyFlags, - ) !void { - if (comptime disable_kitty_keyboard_protocol) return; - - log.debug("pushing kitty keyboard mode: {}", .{flags}); - self.terminal.screen.kitty_keyboard.push(flags); - } - - pub fn popKittyKeyboard(self: *StreamHandler, n: u16) !void { - if (comptime disable_kitty_keyboard_protocol) return; - - log.debug("popping kitty keyboard mode n={}", .{n}); - self.terminal.screen.kitty_keyboard.pop(@intCast(n)); - } - - pub fn setKittyKeyboard( - self: *StreamHandler, - mode: terminal.kitty.KeySetMode, - flags: terminal.kitty.KeyFlags, - ) !void { - if (comptime disable_kitty_keyboard_protocol) return; - - log.debug("setting kitty keyboard mode: {} {}", .{ mode, flags }); - self.terminal.screen.kitty_keyboard.set(mode, flags); - } - - pub fn reportXtversion( - self: *StreamHandler, - ) !void { - log.debug("reporting XTVERSION: ghostty {s}", .{build_config.version_string}); - var buf: [288]u8 = undefined; - const resp = try std.fmt.bufPrint( - &buf, - "\x1BP>|{s} {s}\x1B\\", - .{ - "ghostty", - build_config.version_string, - }, - ); - const msg = try termio.Message.writeReq(self.alloc, resp); - self.messageWriter(msg); - } - - //------------------------------------------------------------------------- - // OSC - - pub fn changeWindowTitle(self: *StreamHandler, title: []const u8) !void { - var buf: [256]u8 = undefined; - if (title.len >= buf.len) { - log.warn("change title requested larger than our buffer size, ignoring", .{}); - return; - } - - @memcpy(buf[0..title.len], title); - buf[title.len] = 0; - - // Mark that we've seen a title - self.ev.seen_title = true; - self.surfaceMessageWriter(.{ .set_title = buf }); - } - - pub fn setMouseShape( - self: *StreamHandler, - shape: terminal.MouseShape, - ) !void { - // Avoid changing the shape it it is already set to avoid excess - // cross-thread messaging. - if (self.terminal.mouse_shape == shape) return; - - self.terminal.mouse_shape = shape; - self.surfaceMessageWriter(.{ .set_mouse_shape = shape }); - } - - pub fn clipboardContents(self: *StreamHandler, kind: u8, data: []const u8) !void { - // Note: we ignore the "kind" field and always use the standard clipboard. - // iTerm also appears to do this but other terminals seem to only allow - // certain. Let's investigate more. - - const clipboard_type: apprt.Clipboard = switch (kind) { - 'c' => .standard, - 's' => .selection, - 'p' => .primary, - else => .standard, - }; - - // Get clipboard contents - if (data.len == 1 and data[0] == '?') { - self.surfaceMessageWriter(.{ .clipboard_read = clipboard_type }); - return; - } - - // Write clipboard contents - self.surfaceMessageWriter(.{ - .clipboard_write = .{ - .req = try apprt.surface.Message.WriteReq.init( - self.alloc, - data, - ), - .clipboard_type = clipboard_type, - }, - }); - } - - pub fn promptStart(self: *StreamHandler, aid: ?[]const u8, redraw: bool) !void { - _ = aid; - self.terminal.markSemanticPrompt(.prompt); - self.terminal.flags.shell_redraws_prompt = redraw; - } - - pub fn promptContinuation(self: *StreamHandler, aid: ?[]const u8) !void { - _ = aid; - self.terminal.markSemanticPrompt(.prompt_continuation); - } - - pub fn promptEnd(self: *StreamHandler) !void { - self.terminal.markSemanticPrompt(.input); - } - - pub fn endOfInput(self: *StreamHandler) !void { - self.terminal.markSemanticPrompt(.command); - } - - pub fn reportPwd(self: *StreamHandler, url: []const u8) !void { - if (builtin.os.tag == .windows) { - log.warn("reportPwd unimplemented on windows", .{}); - return; - } - - const uri = std.Uri.parse(url) catch |e| { - log.warn("invalid url in OSC 7: {}", .{e}); - return; - }; - - if (!std.mem.eql(u8, "file", uri.scheme) and - !std.mem.eql(u8, "kitty-shell-cwd", uri.scheme)) - { - log.warn("OSC 7 scheme must be file, got: {s}", .{uri.scheme}); - return; - } - - // OSC 7 is a little sketchy because anyone can send any value from - // any host (such an SSH session). The best practice terminals follow - // is to valid the hostname to be local. - const host_valid = host_valid: { - const host_component = uri.host orelse break :host_valid false; - - // Get the raw string of the URI. Its unclear to me if the various - // tags of this enum guarantee no percent-encoding so we just - // check all of it. This isn't a performance critical path. - const host = switch (host_component) { - .raw => |v| v, - .percent_encoded => |v| v, - }; - if (host.len == 0 or std.mem.eql(u8, "localhost", host)) { - break :host_valid true; - } - - // Otherwise, it must match our hostname. - var buf: [posix.HOST_NAME_MAX]u8 = undefined; - const hostname = posix.gethostname(&buf) catch |err| { - log.warn("failed to get hostname for OSC 7 validation: {}", .{err}); - break :host_valid false; - }; - - break :host_valid std.mem.eql(u8, host, hostname); - }; - if (!host_valid) { - log.warn("OSC 7 host must be local", .{}); - return; - } - - // We need to unescape the path. We first try to unescape onto - // the stack and fall back to heap allocation if we have to. - var pathBuf: [1024]u8 = undefined; - const path, const heap = path: { - // Get the raw string of the URI. Its unclear to me if the various - // tags of this enum guarantee no percent-encoding so we just - // check all of it. This isn't a performance critical path. - const path = switch (uri.path) { - .raw => |v| v, - .percent_encoded => |v| v, - }; - - // If the path doesn't have any escapes, we can use it directly. - if (std.mem.indexOfScalar(u8, path, '%') == null) - break :path .{ path, false }; - - // First try to stack-allocate - var fba = std.heap.FixedBufferAllocator.init(&pathBuf); - if (std.fmt.allocPrint(fba.allocator(), "{raw}", .{uri.path})) |v| - break :path .{ v, false } - else |_| {} - - // Fall back to heap - if (std.fmt.allocPrint(self.alloc, "{raw}", .{uri.path})) |v| - break :path .{ v, true } - else |_| {} - - // Fall back to using it directly... - log.warn("failed to unescape OSC 7 path, using it directly path={s}", .{path}); - break :path .{ path, false }; - }; - defer if (heap) self.alloc.free(path); - - log.debug("terminal pwd: {s}", .{path}); - try self.terminal.setPwd(path); - - // If we haven't seen a title, use our pwd as the title. - if (!self.ev.seen_title) { - try self.changeWindowTitle(path); - self.ev.seen_title = false; - } - } - - /// Implements OSC 4, OSC 10, and OSC 11, which reports palette color, - /// default foreground color, and background color respectively. - pub fn reportColor( - self: *StreamHandler, - kind: terminal.osc.Command.ColorKind, - terminator: terminal.osc.Terminator, - ) !void { - if (self.osc_color_report_format == .none) return; - - const color = switch (kind) { - .palette => |i| self.terminal.color_palette.colors[i], - .foreground => self.foreground_color, - .background => self.background_color, - .cursor => self.cursor_color orelse self.foreground_color, - }; - - var msg: termio.Message = .{ .write_small = .{} }; - const resp = switch (self.osc_color_report_format) { - .@"16-bit" => switch (kind) { - .palette => |i| try std.fmt.bufPrint( - &msg.write_small.data, - "\x1B]{s};{d};rgb:{x:0>4}/{x:0>4}/{x:0>4}{s}", - .{ - kind.code(), - i, - @as(u16, color.r) * 257, - @as(u16, color.g) * 257, - @as(u16, color.b) * 257, - terminator.string(), - }, - ), - else => try std.fmt.bufPrint( - &msg.write_small.data, - "\x1B]{s};rgb:{x:0>4}/{x:0>4}/{x:0>4}{s}", - .{ - kind.code(), - @as(u16, color.r) * 257, - @as(u16, color.g) * 257, - @as(u16, color.b) * 257, - terminator.string(), - }, - ), - }, - - .@"8-bit" => switch (kind) { - .palette => |i| try std.fmt.bufPrint( - &msg.write_small.data, - "\x1B]{s};{d};rgb:{x:0>2}/{x:0>2}/{x:0>2}{s}", - .{ - kind.code(), - i, - @as(u16, color.r), - @as(u16, color.g), - @as(u16, color.b), - terminator.string(), - }, - ), - else => try std.fmt.bufPrint( - &msg.write_small.data, - "\x1B]{s};rgb:{x:0>2}/{x:0>2}/{x:0>2}{s}", - .{ - kind.code(), - @as(u16, color.r), - @as(u16, color.g), - @as(u16, color.b), - terminator.string(), - }, - ), - }, - .none => unreachable, // early return above - }; - msg.write_small.len = @intCast(resp.len); - self.messageWriter(msg); - } - - pub fn setColor( - self: *StreamHandler, - kind: terminal.osc.Command.ColorKind, - value: []const u8, - ) !void { - const color = try terminal.color.RGB.parse(value); - - switch (kind) { - .palette => |i| { - self.terminal.flags.dirty.palette = true; - self.terminal.color_palette.colors[i] = color; - self.terminal.color_palette.mask.set(i); - }, - .foreground => { - self.foreground_color = color; - _ = self.ev.renderer_mailbox.push(.{ - .foreground_color = color, - }, .{ .forever = {} }); - }, - .background => { - self.background_color = color; - _ = self.ev.renderer_mailbox.push(.{ - .background_color = color, - }, .{ .forever = {} }); - }, - .cursor => { - self.cursor_color = color; - _ = self.ev.renderer_mailbox.push(.{ - .cursor_color = color, - }, .{ .forever = {} }); - }, - } - } - - pub fn resetColor( - self: *StreamHandler, - kind: terminal.osc.Command.ColorKind, - value: []const u8, - ) !void { - switch (kind) { - .palette => { - const mask = &self.terminal.color_palette.mask; - if (value.len == 0) { - // Find all bit positions in the mask which are set and - // reset those indices to the default palette - var it = mask.iterator(.{}); - while (it.next()) |i| { - self.terminal.flags.dirty.palette = true; - self.terminal.color_palette.colors[i] = self.terminal.default_palette[i]; - mask.unset(i); - } - } else { - var it = std.mem.tokenizeScalar(u8, value, ';'); - while (it.next()) |param| { - // Skip invalid parameters - const i = std.fmt.parseUnsigned(u8, param, 10) catch continue; - if (mask.isSet(i)) { - self.terminal.flags.dirty.palette = true; - self.terminal.color_palette.colors[i] = self.terminal.default_palette[i]; - mask.unset(i); - } - } - } - }, - .foreground => { - self.foreground_color = self.default_foreground_color; - _ = self.ev.renderer_mailbox.push(.{ - .foreground_color = self.foreground_color, - }, .{ .forever = {} }); - }, - .background => { - self.background_color = self.default_background_color; - _ = self.ev.renderer_mailbox.push(.{ - .background_color = self.background_color, - }, .{ .forever = {} }); - }, - .cursor => { - self.cursor_color = self.default_cursor_color; - _ = self.ev.renderer_mailbox.push(.{ - .cursor_color = self.cursor_color, - }, .{ .forever = {} }); - }, - } - } - - pub fn showDesktopNotification( - self: *StreamHandler, - title: []const u8, - body: []const u8, - ) !void { - var message = apprt.surface.Message{ .desktop_notification = undefined }; - - const title_len = @min(title.len, message.desktop_notification.title.len); - @memcpy(message.desktop_notification.title[0..title_len], title[0..title_len]); - message.desktop_notification.title[title_len] = 0; - - const body_len = @min(body.len, message.desktop_notification.body.len); - @memcpy(message.desktop_notification.body[0..body_len], body[0..body_len]); - message.desktop_notification.body[body_len] = 0; - - self.surfaceMessageWriter(message); - } }; diff --git a/src/termio/Options.zig b/src/termio/Options.zig index b4edb473a..fe862a503 100644 --- a/src/termio/Options.zig +++ b/src/termio/Options.zig @@ -23,10 +23,14 @@ padding: renderer.Padding, full_config: *const Config, /// The derived configuration for this termio implementation. -config: termio.Impl.DerivedConfig, +config: termio.Termio.DerivedConfig, -/// The application resources directory. -resources_dir: ?[]const u8, +/// The backend for termio that implements where reads/writes are sourced. +backend: termio.Backend, + +/// The mailbox for the terminal. This is how messages are delivered. +/// If you're using termio.Thread this MUST be "mailbox". +mailbox: termio.Mailbox, /// The render state. The IO implementation can modify anything here. The /// surface thread will setup the initial "terminal" pointer but the IO impl @@ -43,7 +47,3 @@ renderer_mailbox: *renderer.Thread.Mailbox, /// The mailbox for sending the surface messages. surface_mailbox: apprt.surface.Mailbox, - -/// The cgroup to apply to the started termio process, if able by -/// the termio implementation. This only applies to Linux. -linux_cgroup: Command.LinuxCgroup = Command.linux_cgroup_default, diff --git a/src/termio/Termio.zig b/src/termio/Termio.zig new file mode 100644 index 000000000..9459f9152 --- /dev/null +++ b/src/termio/Termio.zig @@ -0,0 +1,548 @@ +//! Primary terminal IO ("termio") state. This maintains the terminal state, +//! pty, subprocess, etc. This is flexible enough to be used in environments +//! that don't have a pty and simply provides the input/output using raw +//! bytes. +pub const Termio = @This(); + +const std = @import("std"); +const builtin = @import("builtin"); +const build_config = @import("../build_config.zig"); +const assert = std.debug.assert; +const Allocator = std.mem.Allocator; +const ArenaAllocator = std.heap.ArenaAllocator; +const EnvMap = std.process.EnvMap; +const posix = std.posix; +const termio = @import("../termio.zig"); +const Command = @import("../Command.zig"); +const Pty = @import("../pty.zig").Pty; +const SegmentedPool = @import("../segmented_pool.zig").SegmentedPool; +const StreamHandler = @import("stream_handler.zig").StreamHandler; +const terminal = @import("../terminal/main.zig"); +const terminfo = @import("../terminfo/main.zig"); +const xev = @import("xev"); +const renderer = @import("../renderer.zig"); +const apprt = @import("../apprt.zig"); +const fastmem = @import("../fastmem.zig"); +const internal_os = @import("../os/main.zig"); +const windows = internal_os.windows; +const configpkg = @import("../config.zig"); +const shell_integration = @import("shell_integration.zig"); + +const log = std.log.scoped(.io_exec); + +/// Allocator +alloc: Allocator, + +/// This is the implementation responsible for io. +backend: termio.Backend, + +/// The derived configuration for this termio implementation. +config: DerivedConfig, + +/// The terminal emulator internal state. This is the abstract "terminal" +/// that manages input, grid updating, etc. and is renderer-agnostic. It +/// just stores internal state about a grid. +terminal: terminal.Terminal, + +/// The shared render state +renderer_state: *renderer.State, + +/// A handle to wake up the renderer. This hints to the renderer that that +/// a repaint should happen. +renderer_wakeup: xev.Async, + +/// The mailbox for notifying the renderer of things. +renderer_mailbox: *renderer.Thread.Mailbox, + +/// The mailbox for communicating with the surface. +surface_mailbox: apprt.surface.Mailbox, + +/// The cached grid size whenever a resize is called. +grid_size: renderer.GridSize, + +/// The mailbox implementation to use. +mailbox: termio.Mailbox, + +/// The stream parser. This parses the stream of escape codes and so on +/// from the child process and calls callbacks in the stream handler. +terminal_stream: terminal.Stream(StreamHandler), + +/// Last time the cursor was reset. This is used to prevent message +/// flooding with cursor resets. +last_cursor_reset: ?std.time.Instant = null, + +/// The configuration for this IO that is derived from the main +/// configuration. This must be exported so that we don't need to +/// pass around Config pointers which makes memory management a pain. +pub const DerivedConfig = struct { + arena: ArenaAllocator, + + palette: terminal.color.Palette, + image_storage_limit: usize, + cursor_style: terminal.CursorStyle, + cursor_blink: ?bool, + cursor_color: ?configpkg.Config.Color, + foreground: configpkg.Config.Color, + background: configpkg.Config.Color, + osc_color_report_format: configpkg.Config.OSCColorReportFormat, + abnormal_runtime_threshold_ms: u32, + wait_after_command: bool, + enquiry_response: []const u8, + + pub fn init( + alloc_gpa: Allocator, + config: *const configpkg.Config, + ) !DerivedConfig { + var arena = ArenaAllocator.init(alloc_gpa); + errdefer arena.deinit(); + const alloc = arena.allocator(); + + return .{ + .palette = config.palette.value, + .image_storage_limit = config.@"image-storage-limit", + .cursor_style = config.@"cursor-style", + .cursor_blink = config.@"cursor-style-blink", + .cursor_color = config.@"cursor-color", + .foreground = config.foreground, + .background = config.background, + .osc_color_report_format = config.@"osc-color-report-format", + .abnormal_runtime_threshold_ms = config.@"abnormal-command-exit-runtime", + .wait_after_command = config.@"wait-after-command", + .enquiry_response = try alloc.dupe(u8, config.@"enquiry-response"), + + // This has to be last so that we copy AFTER the arena allocations + // above happen (Zig assigns in order). + .arena = arena, + }; + } + + pub fn deinit(self: *DerivedConfig) void { + self.arena.deinit(); + } +}; + +/// Initialize the termio state. +/// +/// This will also start the child process if the termio is configured +/// to run a child process. +pub fn init(self: *Termio, alloc: Allocator, opts: termio.Options) !void { + // Create our terminal + var term = try terminal.Terminal.init(alloc, .{ + .cols = opts.grid_size.columns, + .rows = opts.grid_size.rows, + .max_scrollback = opts.full_config.@"scrollback-limit", + }); + errdefer term.deinit(alloc); + term.default_palette = opts.config.palette; + term.color_palette.colors = opts.config.palette; + + // Setup our initial grapheme cluster support if enabled. We use a + // switch to ensure we get a compiler error if more cases are added. + switch (opts.full_config.@"grapheme-width-method") { + .unicode => term.modes.set(.grapheme_cluster, true), + .legacy => {}, + } + + // Set the image size limits + try term.screen.kitty_images.setLimit( + alloc, + &term.screen, + opts.config.image_storage_limit, + ); + try term.secondary_screen.kitty_images.setLimit( + alloc, + &term.secondary_screen, + opts.config.image_storage_limit, + ); + + // Set default cursor blink settings + term.modes.set( + .cursor_blinking, + opts.config.cursor_blink orelse true, + ); + + // Set our default cursor style + term.screen.cursor.cursor_style = opts.config.cursor_style; + + // Setup our backend. + var backend = opts.backend; + backend.initTerminal(&term); + + // Setup our terminal size in pixels for certain requests. + const screen_size = opts.screen_size.subPadding(opts.padding); + term.width_px = screen_size.width; + term.height_px = screen_size.height; + + // Create our stream handler. This points to memory in self so it + // isn't safe to use until self.* is set. + const handler: StreamHandler = handler: { + const default_cursor_color = if (opts.config.cursor_color) |col| + col.toTerminalRGB() + else + null; + + break :handler .{ + .alloc = alloc, + .termio_mailbox = &self.mailbox, + .surface_mailbox = opts.surface_mailbox, + .renderer_state = opts.renderer_state, + .renderer_wakeup = opts.renderer_wakeup, + .renderer_mailbox = opts.renderer_mailbox, + .grid_size = &self.grid_size, + .terminal = &self.terminal, + .osc_color_report_format = opts.config.osc_color_report_format, + .enquiry_response = opts.config.enquiry_response, + .default_foreground_color = opts.config.foreground.toTerminalRGB(), + .default_background_color = opts.config.background.toTerminalRGB(), + .default_cursor_style = opts.config.cursor_style, + .default_cursor_blink = opts.config.cursor_blink, + .default_cursor_color = default_cursor_color, + .cursor_color = default_cursor_color, + .foreground_color = opts.config.foreground.toTerminalRGB(), + .background_color = opts.config.background.toTerminalRGB(), + }; + }; + + self.* = .{ + .alloc = alloc, + .terminal = term, + .config = opts.config, + .renderer_state = opts.renderer_state, + .renderer_wakeup = opts.renderer_wakeup, + .renderer_mailbox = opts.renderer_mailbox, + .surface_mailbox = opts.surface_mailbox, + .grid_size = opts.grid_size, + .backend = opts.backend, + .mailbox = opts.mailbox, + .terminal_stream = .{ + .handler = handler, + .parser = .{ + .osc_parser = .{ + // Populate the OSC parser allocator (optional) because + // we want to support large OSC payloads such as OSC 52. + .alloc = alloc, + }, + }, + }, + }; +} + +pub fn deinit(self: *Termio) void { + self.backend.deinit(); + self.terminal.deinit(self.alloc); + self.config.deinit(); + self.mailbox.deinit(self.alloc); + + // Clear any StreamHandler state + self.terminal_stream.handler.deinit(); + self.terminal_stream.deinit(); +} + +pub fn threadEnter(self: *Termio, thread: *termio.Thread, data: *ThreadData) !void { + data.* = .{ + .alloc = self.alloc, + .loop = &thread.loop, + .renderer_state = self.renderer_state, + .surface_mailbox = self.surface_mailbox, + .mailbox = &self.mailbox, + .backend = undefined, // Backend must replace this on threadEnter + }; + + // Setup our backend + try self.backend.threadEnter(self.alloc, self, data); +} + +pub fn threadExit(self: *Termio, data: *ThreadData) void { + self.backend.threadExit(data); +} + +/// Send a message to the the mailbox. Depending on the mailbox type in +/// use this may process now or it may just enqueue and process later. +/// +/// This will also notify the mailbox thread to process the message. If +/// you're sending a lot of messages, it may be more efficient to use +/// the mailbox directly and then call notify separately. +pub fn queueMessage( + self: *Termio, + msg: termio.Message, + mutex: enum { locked, unlocked }, +) void { + self.mailbox.send(msg, switch (mutex) { + .locked => self.renderer_state.mutex, + .unlocked => null, + }); + self.mailbox.notify(); +} + +/// Queue a write directly to the pty. +/// +/// If you're using termio.Thread, this must ONLY be called from the +/// mailbox thread. If you're not on the thread, use queueMessage with +/// mailbox messages instead. +/// +/// If you're not using termio.Thread, this is not threadsafe. +pub inline fn queueWrite( + self: *Termio, + td: *ThreadData, + data: []const u8, + linefeed: bool, +) !void { + try self.backend.queueWrite(self.alloc, td, data, linefeed); +} + +/// Update the configuration. +pub fn changeConfig(self: *Termio, td: *ThreadData, config: *DerivedConfig) !void { + // The remainder of this function is modifying terminal state or + // the read thread data, all of which requires holding the renderer + // state lock. + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + + // Deinit our old config. We do this in the lock because the + // stream handler may be referencing the old config (i.e. enquiry resp) + self.config.deinit(); + self.config = config.*; + + // Update our stream handler. The stream handler uses the same + // renderer mutex so this is safe to do despite being executed + // from another thread. + self.terminal_stream.handler.changeConfig(&self.config); + td.backend.changeConfig(&self.config); + + // Update the configuration that we know about. + // + // Specific things we don't update: + // - command, working-directory: we never restart the underlying + // process so we don't care or need to know about these. + + // Update the default palette. Note this will only apply to new colors drawn + // since we decode all palette colors to RGB on usage. + self.terminal.default_palette = config.palette; + + // Update the active palette, except for any colors that were modified with + // OSC 4 + for (0..config.palette.len) |i| { + if (!self.terminal.color_palette.mask.isSet(i)) { + self.terminal.color_palette.colors[i] = config.palette[i]; + self.terminal.flags.dirty.palette = true; + } + } + + // Set the image size limits + try self.terminal.screen.kitty_images.setLimit( + self.alloc, + &self.terminal.screen, + config.image_storage_limit, + ); + try self.terminal.secondary_screen.kitty_images.setLimit( + self.alloc, + &self.terminal.secondary_screen, + config.image_storage_limit, + ); +} + +/// Resize the terminal. +pub fn resize( + self: *Termio, + grid_size: renderer.GridSize, + screen_size: renderer.ScreenSize, + padding: renderer.Padding, +) !void { + // Update the size of our pty. + const padded_size = screen_size.subPadding(padding); + try self.backend.resize(grid_size, padded_size); + + // Update our cached grid size + self.grid_size = grid_size; + + // Enter the critical area that we want to keep small + { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + + // Update the size of our terminal state + try self.terminal.resize( + self.alloc, + grid_size.columns, + grid_size.rows, + ); + + // Update our pixel sizes + self.terminal.width_px = padded_size.width; + self.terminal.height_px = padded_size.height; + + // Disable synchronized output mode so that we show changes + // immediately for a resize. This is allowed by the spec. + self.terminal.modes.set(.synchronized_output, false); + + // Wake up our renderer so any changes will be shown asap + self.renderer_wakeup.notify() catch {}; + } +} + +/// Reset the synchronized output mode. This is usually called by timer +/// expiration from the termio thread. +pub fn resetSynchronizedOutput(self: *Termio) void { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + self.terminal.modes.set(.synchronized_output, false); + self.renderer_wakeup.notify() catch {}; +} + +/// Clear the screen. +pub fn clearScreen(self: *Termio, td: *ThreadData, history: bool) !void { + { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + + // If we're on the alternate screen, we do not clear. Since this is an + // emulator-level screen clear, this messes up the running programs + // knowledge of where the cursor is and causes rendering issues. So, + // for alt screen, we do nothing. + if (self.terminal.active_screen == .alternate) return; + + // Clear our scrollback + if (history) self.terminal.eraseDisplay(.scrollback, false); + + // If we're not at a prompt, we just delete above the cursor. + if (!self.terminal.cursorIsAtPrompt()) { + if (self.terminal.screen.cursor.y > 0) { + self.terminal.screen.eraseRows( + .{ .active = .{ .y = 0 } }, + .{ .active = .{ .y = self.terminal.screen.cursor.y - 1 } }, + ); + } + + return; + } + + // At a prompt, we want to first fully clear the screen, and then after + // send a FF (0x0C) to the shell so that it can repaint the screen. + // Mark the current row as a not a prompt so we can properly + // clear the full screen in the next eraseDisplay call. + self.terminal.markSemanticPrompt(.command); + assert(!self.terminal.cursorIsAtPrompt()); + self.terminal.eraseDisplay(.complete, false); + } + + // If we reached here it means we're at a prompt, so we send a form-feed. + try self.queueWrite(td, &[_]u8{0x0C}, false); +} + +/// Scroll the viewport +pub fn scrollViewport(self: *Termio, scroll: terminal.Terminal.ScrollViewport) !void { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + try self.terminal.scrollViewport(scroll); +} + +/// Jump the viewport to the prompt. +pub fn jumpToPrompt(self: *Termio, delta: isize) !void { + { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + self.terminal.screen.scroll(.{ .delta_prompt = delta }); + } + + try self.renderer_wakeup.notify(); +} + +/// Called when the child process exited abnormally but before +/// the surface is notified. +pub fn childExitedAbnormally(self: *Termio, exit_code: u32, runtime_ms: u64) !void { + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + const t = self.renderer_state.terminal; + try self.backend.childExitedAbnormally(self.alloc, t, exit_code, runtime_ms); +} + +/// Process output from the pty. This is the manual API that users can +/// call with pty data but it is also called by the read thread when using +/// an exec subprocess. +pub fn processOutput(self: *Termio, buf: []const u8) void { + // We are modifying terminal state from here on out and we need + // the lock to grab our read data. + self.renderer_state.mutex.lock(); + defer self.renderer_state.mutex.unlock(); + self.processOutputLocked(buf); +} + +/// Process output from readdata but the lock is already held. +fn processOutputLocked(self: *Termio, buf: []const u8) void { + // Schedule a render. We can call this first because we have the lock. + self.terminal_stream.handler.queueRender() catch unreachable; + + // Whenever a character is typed, we ensure the cursor is in the + // non-blink state so it is rendered if visible. If we're under + // HEAVY read load, we don't want to send a ton of these so we + // use a timer under the covers + if (std.time.Instant.now()) |now| cursor_reset: { + if (self.last_cursor_reset) |last| { + if (now.since(last) <= (500 * std.time.ns_per_ms)) { + break :cursor_reset; + } + } + + self.last_cursor_reset = now; + _ = self.renderer_mailbox.push(.{ + .reset_cursor_blink = {}, + }, .{ .instant = {} }); + } else |err| { + log.warn("failed to get current time err={}", .{err}); + } + + // If we have an inspector, we enter SLOW MODE because we need to + // process a byte at a time alternating between the inspector handler + // and the termio handler. This is very slow compared to our optimizations + // below but at least users only pay for it if they're using the inspector. + if (self.renderer_state.inspector) |insp| { + for (buf, 0..) |byte, i| { + insp.recordPtyRead(buf[i .. i + 1]) catch |err| { + log.err("error recording pty read in inspector err={}", .{err}); + }; + + self.terminal_stream.next(byte) catch |err| + log.err("error processing terminal data: {}", .{err}); + } + } else { + self.terminal_stream.nextSlice(buf) catch |err| + log.err("error processing terminal data: {}", .{err}); + } + + // If our stream handling caused messages to be sent to the mailbox + // thread, then we need to wake it up so that it processes them. + if (self.terminal_stream.handler.termio_messaged) { + self.terminal_stream.handler.termio_messaged = false; + self.mailbox.notify(); + } +} + +/// ThreadData is the data created and stored in the termio thread +/// when the thread is started and destroyed when the thread is +/// stopped. +/// +/// All of the fields in this struct should only be read/written by +/// the termio thread. As such, a lock is not necessary. +pub const ThreadData = struct { + /// Allocator used for the event data + alloc: Allocator, + + /// The event loop associated with this thread. This is owned by + /// the Thread but we have a pointer so we can queue new work to it. + loop: *xev.Loop, + + /// The shared render state + renderer_state: *renderer.State, + + /// Mailboxes for different threads + surface_mailbox: apprt.surface.Mailbox, + + /// Data associated with the backend implementation (i.e. pty/exec state) + backend: termio.backend.ThreadData, + mailbox: *termio.Mailbox, + + pub fn deinit(self: *ThreadData) void { + self.backend.deinit(self.alloc); + self.* = undefined; + } +}; diff --git a/src/termio/Thread.zig b/src/termio/Thread.zig index 45b9fd1ae..73e384f51 100644 --- a/src/termio/Thread.zig +++ b/src/termio/Thread.zig @@ -1,5 +1,14 @@ -//! Represents the IO thread logic. The IO thread is responsible for -//! the child process and pty management. +//! Represents the "writer" thread for terminal IO. The reader side is +//! handled by the Termio struct itself and dependent on the underlying +//! implementation (i.e. if its a pty, manual, etc.). +//! +//! The writer thread does handle writing bytes to the pty but also handles +//! different events such as starting synchronized output, changing some +//! modes (like linefeed), etc. The goal is to offload as much from the +//! reader thread as possible since it is the hot path in parsing VT +//! sequences and updating terminal state. +//! +//! This thread state can only be used by one thread at a time. pub const Thread = @This(); const std = @import("std"); @@ -12,11 +21,6 @@ const BlockingQueue = @import("../blocking_queue.zig").BlockingQueue; const Allocator = std.mem.Allocator; const log = std.log.scoped(.io_thread); -/// The type used for sending messages to the IO thread. For now this is -/// hardcoded with a capacity. We can make this a comptime parameter in -/// the future if we want it configurable. -pub const Mailbox = BlockingQueue(termio.Message, 64); - /// This stores the information that is coalesced. const Coalesce = struct { /// The number of milliseconds to coalesce certain messages like resize for. @@ -38,8 +42,8 @@ alloc: std.mem.Allocator, /// so that users of the loop always have an allocator. loop: xev.Loop, -/// This can be used to wake up the thread. -wakeup: xev.Async, +/// The completion to use for the wakeup async handle that is present +/// on the termio.Writer. wakeup_c: xev.Completion = .{}, /// This can be used to stop the thread on the next loop iteration. @@ -58,13 +62,6 @@ sync_reset: xev.Timer, sync_reset_c: xev.Completion = .{}, sync_reset_cancel_c: xev.Completion = .{}, -/// The underlying IO implementation. -impl: *termio.Impl, - -/// The mailbox that can be used to send this thread messages. Note -/// this is a blocking queue so if it is full you will get errors (or block). -mailbox: *Mailbox, - flags: packed struct { /// This is set to true only when an abnormal exit is detected. It /// tells our mailbox system to drain and ignore all messages. @@ -83,16 +80,11 @@ flags: packed struct { /// is up to the caller to start the thread with the threadMain entrypoint. pub fn init( alloc: Allocator, - impl: *termio.Impl, ) !Thread { // Create our event loop. var loop = try xev.Loop.init(.{}); errdefer loop.deinit(); - // This async handle is used to "wake up" the renderer and force a render. - var wakeup_h = try xev.Async.init(); - errdefer wakeup_h.deinit(); - // This async handle is used to stop the loop and force the thread to end. var stop_h = try xev.Async.init(); errdefer stop_h.deinit(); @@ -105,19 +97,12 @@ pub fn init( var sync_reset_h = try xev.Timer.init(); errdefer sync_reset_h.deinit(); - // The mailbox for messaging this thread - var mailbox = try Mailbox.create(alloc); - errdefer mailbox.destroy(alloc); - return Thread{ .alloc = alloc, .loop = loop, - .wakeup = wakeup_h, .stop = stop_h, .coalesce = coalesce_h, .sync_reset = sync_reset_h, - .impl = impl, - .mailbox = mailbox, }; } @@ -127,17 +112,13 @@ pub fn deinit(self: *Thread) void { self.coalesce.deinit(); self.sync_reset.deinit(); self.stop.deinit(); - self.wakeup.deinit(); self.loop.deinit(); - - // Nothing can possibly access the mailbox anymore, destroy it. - self.mailbox.destroy(self.alloc); } /// The main entrypoint for the thread. -pub fn threadMain(self: *Thread) void { +pub fn threadMain(self: *Thread, io: *termio.Termio) void { // Call child function so we can use errors... - self.threadMain_() catch |err| { + self.threadMain_(io) catch |err| { log.warn("error in io thread err={}", .{err}); // Use an arena to simplify memory management below @@ -150,9 +131,9 @@ pub fn threadMain(self: *Thread) void { // the error to the surface thread and let the apprt deal with it // in some way but this works for now. Without this, the user would // just see a blank terminal window. - self.impl.renderer_state.mutex.lock(); - defer self.impl.renderer_state.mutex.unlock(); - const t = self.impl.renderer_state.terminal; + io.renderer_state.mutex.lock(); + defer io.renderer_state.mutex.unlock(); + const t = io.renderer_state.terminal; // Hide the cursor t.modes.set(.cursor_visible, false); @@ -216,19 +197,30 @@ pub fn threadMain(self: *Thread) void { } } -fn threadMain_(self: *Thread) !void { +fn threadMain_(self: *Thread, io: *termio.Termio) !void { defer log.debug("IO thread exited", .{}); - // Start the async handlers. We start these first so that they're - // registered even if anything below fails so we can drain the mailbox. - self.wakeup.wait(&self.loop, &self.wakeup_c, Thread, self, wakeupCallback); - self.stop.wait(&self.loop, &self.stop_c, Thread, self, stopCallback); + // Get the mailbox. This must be an SPSC mailbox for threading. + const mailbox = switch (io.mailbox) { + .spsc => |*v| v, + // else => return error.TermioUnsupportedMailbox, + }; + + // This is the data sent to xev callbacks. We want a pointer to both + // ourselves and the thread data so we can thread that through (pun intended). + var cb: CallbackData = .{ .self = self, .io = io }; // Run our thread start/end callbacks. This allows the implementation - // to hook into the event loop as needed. - var data = try self.impl.threadEnter(self); - defer data.deinit(); - defer self.impl.threadExit(data); + // to hook into the event loop as needed. The thread data is created + // on the stack here so that it has a stable pointer throughout the + // lifetime of the thread. + try io.threadEnter(self, &cb.data); + defer cb.data.deinit(); + defer io.threadExit(&cb.data); + + // Start the async handlers. + mailbox.wakeup.wait(&self.loop, &self.wakeup_c, CallbackData, &cb, wakeupCallback); + self.stop.wait(&self.loop, &self.stop_c, CallbackData, &cb, stopCallback); // Run log.debug("starting IO thread", .{}); @@ -236,11 +228,26 @@ fn threadMain_(self: *Thread) !void { try self.loop.run(.until_done); } +/// This is the data passed to xev callbacks on the thread. +const CallbackData = struct { + self: *Thread, + io: *termio.Termio, + data: termio.Termio.ThreadData = undefined, +}; + /// Drain the mailbox, handling all the messages in our terminal implementation. -fn drainMailbox(self: *Thread) !void { +fn drainMailbox( + self: *Thread, + cb: *CallbackData, +) !void { + // We assert when starting the thread that this is the state + const mailbox = cb.io.mailbox.spsc.queue; + const io = cb.io; + const data = &cb.data; + // If we're draining, we just drain the mailbox and return. if (self.flags.drain) { - while (self.mailbox.pop()) |_| {} + while (mailbox.pop()) |_| {} return; } @@ -248,7 +255,7 @@ fn drainMailbox(self: *Thread) !void { // expectation is that all our message handlers will be non-blocking // ENOUGH to not mess up throughput on producers. var redraw: bool = false; - while (self.mailbox.pop()) |message| { + while (mailbox.pop()) |message| { // If we have a message we always redraw redraw = true; @@ -256,21 +263,33 @@ fn drainMailbox(self: *Thread) !void { switch (message) { .change_config => |config| { defer config.alloc.destroy(config.ptr); - try self.impl.changeConfig(config.ptr); + try io.changeConfig(data, config.ptr); }, .inspector => |v| self.flags.has_inspector = v, - .resize => |v| self.handleResize(v), - .clear_screen => |v| try self.impl.clearScreen(v.history), - .scroll_viewport => |v| try self.impl.scrollViewport(v), - .jump_to_prompt => |v| try self.impl.jumpToPrompt(v), - .start_synchronized_output => self.startSynchronizedOutput(), + .resize => |v| self.handleResize(cb, v), + .clear_screen => |v| try io.clearScreen(data, v.history), + .scroll_viewport => |v| try io.scrollViewport(v), + .jump_to_prompt => |v| try io.jumpToPrompt(v), + .start_synchronized_output => self.startSynchronizedOutput(cb), .linefeed_mode => |v| self.flags.linefeed_mode = v, - .child_exited_abnormally => |v| try self.impl.childExitedAbnormally(v.exit_code, v.runtime_ms), - .write_small => |v| try self.impl.queueWrite(v.data[0..v.len], self.flags.linefeed_mode), - .write_stable => |v| try self.impl.queueWrite(v, self.flags.linefeed_mode), + .child_exited_abnormally => |v| try io.childExitedAbnormally(v.exit_code, v.runtime_ms), + .write_small => |v| try io.queueWrite( + data, + v.data[0..v.len], + self.flags.linefeed_mode, + ), + .write_stable => |v| try io.queueWrite( + data, + v, + self.flags.linefeed_mode, + ), .write_alloc => |v| { defer v.alloc.free(v.data); - try self.impl.queueWrite(v.data, self.flags.linefeed_mode); + try io.queueWrite( + data, + v.data, + self.flags.linefeed_mode, + ); }, } } @@ -278,23 +297,23 @@ fn drainMailbox(self: *Thread) !void { // Trigger a redraw after we've drained so we don't waste cyces // messaging a redraw. if (redraw) { - try self.impl.renderer_wakeup.notify(); + try io.renderer_wakeup.notify(); } } -fn startSynchronizedOutput(self: *Thread) void { +fn startSynchronizedOutput(self: *Thread, cb: *CallbackData) void { self.sync_reset.reset( &self.loop, &self.sync_reset_c, &self.sync_reset_cancel_c, sync_reset_ms, - Thread, - self, + CallbackData, + cb, syncResetCallback, ); } -fn handleResize(self: *Thread, resize: termio.Message.Resize) void { +fn handleResize(self: *Thread, cb: *CallbackData, resize: termio.Message.Resize) void { self.coalesce_data.resize = resize; // If the timer is already active we just return. In the future we want @@ -307,14 +326,14 @@ fn handleResize(self: *Thread, resize: termio.Message.Resize) void { &self.coalesce_c, &self.coalesce_cancel_c, Coalesce.min_ms, - Thread, - self, + CallbackData, + cb, coalesceCallback, ); } fn syncResetCallback( - self_: ?*Thread, + cb_: ?*CallbackData, _: *xev.Loop, _: *xev.Completion, r: xev.Timer.RunError!void, @@ -327,13 +346,13 @@ fn syncResetCallback( }, }; - const self = self_ orelse return .disarm; - self.impl.resetSynchronizedOutput(); + const cb = cb_ orelse return .disarm; + cb.io.resetSynchronizedOutput(); return .disarm; } fn coalesceCallback( - self_: ?*Thread, + cb_: ?*CallbackData, _: *xev.Loop, _: *xev.Completion, r: xev.Timer.RunError!void, @@ -346,11 +365,11 @@ fn coalesceCallback( }, }; - const self = self_ orelse return .disarm; + const cb = cb_ orelse return .disarm; - if (self.coalesce_data.resize) |v| { - self.coalesce_data.resize = null; - self.impl.resize(v.grid_size, v.screen_size, v.padding) catch |err| { + if (cb.self.coalesce_data.resize) |v| { + cb.self.coalesce_data.resize = null; + cb.io.resize(v.grid_size, v.screen_size, v.padding) catch |err| { log.warn("error during resize err={}", .{err}); }; } @@ -359,7 +378,7 @@ fn coalesceCallback( } fn wakeupCallback( - self_: ?*Thread, + cb_: ?*CallbackData, _: *xev.Loop, _: *xev.Completion, r: xev.Async.WaitError!void, @@ -369,23 +388,22 @@ fn wakeupCallback( return .rearm; }; - const t = self_.?; - // When we wake up, we check the mailbox. Mailbox producers should // wake up our thread after publishing. - t.drainMailbox() catch |err| + const cb = cb_ orelse return .rearm; + cb.self.drainMailbox(cb) catch |err| log.err("error draining mailbox err={}", .{err}); return .rearm; } fn stopCallback( - self_: ?*Thread, + cb_: ?*CallbackData, _: *xev.Loop, _: *xev.Completion, r: xev.Async.WaitError!void, ) xev.CallbackAction { _ = r catch unreachable; - self_.?.loop.stop(); + cb_.?.self.loop.stop(); return .disarm; } diff --git a/src/termio/backend.zig b/src/termio/backend.zig new file mode 100644 index 000000000..b29df89c6 --- /dev/null +++ b/src/termio/backend.zig @@ -0,0 +1,123 @@ +const std = @import("std"); +const builtin = @import("builtin"); +const assert = std.debug.assert; +const Allocator = std.mem.Allocator; +const posix = std.posix; +const xev = @import("xev"); +const build_config = @import("../build_config.zig"); +const configpkg = @import("../config.zig"); +const internal_os = @import("../os/main.zig"); +const renderer = @import("../renderer.zig"); +const shell_integration = @import("shell_integration.zig"); +const terminal = @import("../terminal/main.zig"); +const termio = @import("../termio.zig"); +const Command = @import("../Command.zig"); +const SegmentedPool = @import("../segmented_pool.zig").SegmentedPool; +const Pty = @import("../pty.zig").Pty; + +// The preallocation size for the write request pool. This should be big +// enough to satisfy most write requests. It must be a power of 2. +const WRITE_REQ_PREALLOC = std.math.pow(usize, 2, 5); + +/// The kinds of backends. +pub const Kind = enum { exec }; + +/// Configuration for the various backend types. +pub const Config = union(Kind) { + /// Exec uses posix exec to run a command with a pty. + exec: termio.Exec.Config, +}; + +/// Backend implementations. A backend is responsible for owning the pty +/// behavior and providing read/write capabilities. +pub const Backend = union(Kind) { + exec: termio.Exec, + + pub fn deinit(self: *Backend) void { + switch (self.*) { + .exec => |*exec| exec.deinit(), + } + } + + pub fn initTerminal(self: *Backend, t: *terminal.Terminal) void { + switch (self.*) { + .exec => |*exec| exec.initTerminal(t), + } + } + + pub fn threadEnter( + self: *Backend, + alloc: Allocator, + io: *termio.Termio, + td: *termio.Termio.ThreadData, + ) !void { + switch (self.*) { + .exec => |*exec| try exec.threadEnter(alloc, io, td), + } + } + + pub fn threadExit(self: *Backend, td: *termio.Termio.ThreadData) void { + switch (self.*) { + .exec => |*exec| exec.threadExit(td), + } + } + + pub fn resize( + self: *Backend, + grid_size: renderer.GridSize, + screen_size: renderer.ScreenSize, + ) !void { + switch (self.*) { + .exec => |*exec| try exec.resize(grid_size, screen_size), + } + } + + pub fn queueWrite( + self: *Backend, + alloc: Allocator, + td: *termio.Termio.ThreadData, + data: []const u8, + linefeed: bool, + ) !void { + switch (self.*) { + .exec => |*exec| try exec.queueWrite(alloc, td, data, linefeed), + } + } + + pub fn childExitedAbnormally( + self: *Backend, + gpa: Allocator, + t: *terminal.Terminal, + exit_code: u32, + runtime_ms: u64, + ) !void { + switch (self.*) { + .exec => |*exec| try exec.childExitedAbnormally( + gpa, + t, + exit_code, + runtime_ms, + ), + } + } +}; + +/// Termio thread data. See termio.ThreadData for docs. +pub const ThreadData = union(Kind) { + exec: termio.Exec.ThreadData, + + pub fn deinit(self: *ThreadData, alloc: Allocator) void { + switch (self.*) { + .exec => |*exec| exec.deinit(alloc), + } + } + + pub fn changeConfig(self: *ThreadData, config: *termio.DerivedConfig) void { + switch (self.*) { + .exec => |*exec| { + exec.abnormal_runtime_threshold_ms = config.abnormal_runtime_threshold_ms; + exec.wait_after_command = config.wait_after_command; + }, + } + } +}; diff --git a/src/termio/mailbox.zig b/src/termio/mailbox.zig new file mode 100644 index 000000000..85471009d --- /dev/null +++ b/src/termio/mailbox.zig @@ -0,0 +1,108 @@ +const std = @import("std"); +const builtin = @import("builtin"); +const assert = std.debug.assert; +const Allocator = std.mem.Allocator; +const xev = @import("xev"); +const renderer = @import("../renderer.zig"); +const termio = @import("../termio.zig"); +const BlockingQueue = @import("../blocking_queue.zig").BlockingQueue; + +const log = std.log.scoped(.io_writer); + +/// A queue used for storing messages that is periodically drained. +/// Typically used by a multi-threaded application. The capacity is +/// hardcoded to a value that empirically has made sense for Ghostty usage +/// but I'm open to changing it with good arguments. +const Queue = BlockingQueue(termio.Message, 64); + +/// The location to where write-related messages are sent. +pub const Mailbox = union(enum) { + // /// Write messages to an unbounded list backed by an allocator. + // /// This is useful for single-threaded applications where you're not + // /// afraid of running out of memory. You should be careful that you're + // /// processing this in a timely manner though since some heavy workloads + // /// will produce a LOT of messages. + // /// + // /// At the time of authoring this, the primary use case for this is + // /// testing more than anything, but it probably will have a use case + // /// in libghostty eventually. + // unbounded: std.ArrayList(termio.Message), + + /// Write messages to a SPSC queue for multi-threaded applications. + spsc: struct { + queue: *Queue, + wakeup: xev.Async, + }, + + /// Init the SPSC writer. + pub fn initSPSC(alloc: Allocator) !Mailbox { + var queue = try Queue.create(alloc); + errdefer queue.destroy(alloc); + + var wakeup = try xev.Async.init(); + errdefer wakeup.deinit(); + + return .{ .spsc = .{ .queue = queue, .wakeup = wakeup } }; + } + + pub fn deinit(self: *Mailbox, alloc: Allocator) void { + switch (self.*) { + .spsc => |*v| { + v.queue.destroy(alloc); + v.wakeup.deinit(); + }, + } + } + + /// Sends the given message without notifying there are messages. + /// + /// If the optional mutex is given, it must already be LOCKED. If the + /// send would block, we'll unlock this mutex, resend the message, and + /// lock it again. This handles an edge case where queues are full. + /// This may not apply to all writer types. + pub fn send( + self: *Mailbox, + msg: termio.Message, + mutex: ?*std.Thread.Mutex, + ) void { + switch (self.*) { + .spsc => |*mb| send: { + // Try to write to the queue with an instant timeout. This is the + // fast path because we can queue without a lock. + if (mb.queue.push(msg, .{ .instant = {} }) > 0) break :send; + + // If we enter this conditional, the queue is full. We wake up + // the writer thread so that it can process messages to clear up + // space. However, the writer thread may require the renderer + // lock so we need to unlock. + mb.wakeup.notify() catch |err| { + log.warn("failed to wake up writer, data will be dropped err={}", .{err}); + return; + }; + + // Unlock the renderer state so the writer thread can acquire it. + // Then try to queue our message before continuing. This is a very + // slow path because we are having a lot of contention for data. + // But this only gets triggered in certain pathological cases. + // + // Note that writes themselves don't require a lock, but there + // are other messages in the writer queue (resize, focus) that + // could acquire the lock. This is why we have to release our lock + // here. + if (mutex) |m| m.unlock(); + defer if (mutex) |m| m.lock(); + _ = mb.queue.push(msg, .{ .forever = {} }); + }, + } + } + + /// Notify that there are new messages. This may be a noop depending + /// on the writer type. + pub fn notify(self: *Mailbox) void { + switch (self.*) { + .spsc => |*v| v.wakeup.notify() catch |err| { + log.warn("failed to notify writer, data will be dropped err={}", .{err}); + }, + } + } +}; diff --git a/src/termio/message.zig b/src/termio/message.zig index c91f3b5d6..31b203f05 100644 --- a/src/termio/message.zig +++ b/src/termio/message.zig @@ -33,7 +33,7 @@ pub const Message = union(enum) { /// is allocated via the allocator and is expected to be freed when done. change_config: struct { alloc: Allocator, - ptr: *termio.Impl.DerivedConfig, + ptr: *termio.Termio.DerivedConfig, }, /// Activate or deactivate the inspector. diff --git a/src/termio/stream_handler.zig b/src/termio/stream_handler.zig new file mode 100644 index 000000000..bb45cd480 --- /dev/null +++ b/src/termio/stream_handler.zig @@ -0,0 +1,1258 @@ +const std = @import("std"); +const builtin = @import("builtin"); +const Allocator = std.mem.Allocator; +const xev = @import("xev"); +const apprt = @import("../apprt.zig"); +const build_config = @import("../build_config.zig"); +const configpkg = @import("../config.zig"); +const renderer = @import("../renderer.zig"); +const termio = @import("../termio.zig"); +const terminal = @import("../terminal/main.zig"); +const terminfo = @import("../terminfo/main.zig"); +const posix = std.posix; + +const log = std.log.scoped(.io_handler); + +/// True if we should disable the kitty keyboard protocol. We have to +/// disable this on GLFW because GLFW input events don't support the +/// correct granularity of events. +const disable_kitty_keyboard_protocol = apprt.runtime == apprt.glfw; + +/// This is used as the handler for the terminal.Stream type. This is +/// stateful and is expected to live for the entire lifetime of the terminal. +/// It is NOT VALID to stop a stream handler, create a new one, and use that +/// unless all of the member fields are copied. +pub const StreamHandler = struct { + alloc: Allocator, + grid_size: *renderer.GridSize, + terminal: *terminal.Terminal, + + /// Mailbox for data to the termio thread. + termio_mailbox: *termio.Mailbox, + + /// Mailbox for the surface. + surface_mailbox: apprt.surface.Mailbox, + + /// The shared render state + renderer_state: *renderer.State, + + /// The mailbox for notifying the renderer of things. + renderer_mailbox: *renderer.Thread.Mailbox, + + /// A handle to wake up the renderer. This hints to the renderer that that + /// a repaint should happen. + renderer_wakeup: xev.Async, + + /// The default cursor state. This is used with CSI q. This is + /// set to true when we're currently in the default cursor state. + default_cursor: bool = true, + default_cursor_style: terminal.CursorStyle, + default_cursor_blink: ?bool, + default_cursor_color: ?terminal.color.RGB, + + /// Actual cursor color. This can be changed with OSC 12. + cursor_color: ?terminal.color.RGB, + + /// The default foreground and background color are those set by the user's + /// config file. These can be overridden by terminal applications using OSC + /// 10 and OSC 11, respectively. + default_foreground_color: terminal.color.RGB, + default_background_color: terminal.color.RGB, + + /// The actual foreground and background color. Normally this will be the + /// same as the default foreground and background color, unless changed by a + /// terminal application. + foreground_color: terminal.color.RGB, + background_color: terminal.color.RGB, + + /// The response to use for ENQ requests. The memory is owned by + /// whoever owns StreamHandler. + enquiry_response: []const u8, + + /// The color reporting format for OSC requests. + osc_color_report_format: configpkg.Config.OSCColorReportFormat, + + //--------------------------------------------------------------- + // Internal state + + /// The APC command handler maintains the APC state. APC is like + /// CSI or OSC, but it is a private escape sequence that is used + /// to send commands to the terminal emulator. This is used by + /// the kitty graphics protocol. + apc: terminal.apc.Handler = .{}, + + /// The DCS handler maintains DCS state. DCS is like CSI or OSC, + /// but requires more stateful parsing. This is used by functionality + /// such as XTGETTCAP. + dcs: terminal.dcs.Handler = .{}, + + /// This is set to true when a message was written to the termio + /// mailbox. This can be used by callers to determine if they need + /// to wake up the termio thread. + termio_messaged: bool = false, + + /// This is set to true when we've seen a title escape sequence. We use + /// this to determine if we need to default the window title. + seen_title: bool = false, + + pub fn deinit(self: *StreamHandler) void { + self.apc.deinit(); + self.dcs.deinit(); + } + + /// This queues a render operation with the renderer thread. The render + /// isn't guaranteed to happen immediately but it will happen as soon as + /// practical. + pub inline fn queueRender(self: *StreamHandler) !void { + try self.renderer_wakeup.notify(); + } + + /// Change the configuration for this handler. + pub fn changeConfig(self: *StreamHandler, config: *termio.DerivedConfig) void { + self.osc_color_report_format = config.osc_color_report_format; + self.enquiry_response = config.enquiry_response; + self.default_foreground_color = config.foreground.toTerminalRGB(); + self.default_background_color = config.background.toTerminalRGB(); + self.default_cursor_style = config.cursor_style; + self.default_cursor_blink = config.cursor_blink; + self.default_cursor_color = if (config.cursor_color) |col| + col.toTerminalRGB() + else + null; + + // If our cursor is the default, then we update it immediately. + if (self.default_cursor) self.setCursorStyle(.default) catch |err| { + log.warn("failed to set default cursor style: {}", .{err}); + }; + } + + inline fn surfaceMessageWriter( + self: *StreamHandler, + msg: apprt.surface.Message, + ) void { + // See messageWriter which has similar logic and explains why + // we may have to do this. + if (self.surface_mailbox.push(msg, .{ .instant = {} }) == 0) { + self.renderer_state.mutex.unlock(); + defer self.renderer_state.mutex.lock(); + _ = self.surface_mailbox.push(msg, .{ .forever = {} }); + } + } + + inline fn messageWriter(self: *StreamHandler, msg: termio.Message) void { + self.termio_mailbox.send(msg, self.renderer_state.mutex); + self.termio_messaged = true; + } + + pub fn dcsHook(self: *StreamHandler, dcs: terminal.DCS) !void { + var cmd = self.dcs.hook(self.alloc, dcs) orelse return; + defer cmd.deinit(); + try self.dcsCommand(&cmd); + } + + pub fn dcsPut(self: *StreamHandler, byte: u8) !void { + var cmd = self.dcs.put(byte) orelse return; + defer cmd.deinit(); + try self.dcsCommand(&cmd); + } + + pub fn dcsUnhook(self: *StreamHandler) !void { + var cmd = self.dcs.unhook() orelse return; + defer cmd.deinit(); + try self.dcsCommand(&cmd); + } + + fn dcsCommand(self: *StreamHandler, cmd: *terminal.dcs.Command) !void { + // log.warn("DCS command: {}", .{cmd}); + switch (cmd.*) { + .tmux => |tmux| { + // TODO: process it + log.warn("tmux control mode event unimplemented cmd={}", .{tmux}); + }, + + .xtgettcap => |*gettcap| { + const map = comptime terminfo.ghostty.xtgettcapMap(); + while (gettcap.next()) |key| { + const response = map.get(key) orelse continue; + self.messageWriter(.{ .write_stable = response }); + } + }, + + .decrqss => |decrqss| { + var response: [128]u8 = undefined; + var stream = std.io.fixedBufferStream(&response); + const writer = stream.writer(); + + // Offset the stream position to just past the response prefix. + // We will write the "payload" (if any) below. If no payload is + // written then we send an invalid DECRPSS response. + const prefix_fmt = "\x1bP{d}$r"; + const prefix_len = std.fmt.comptimePrint(prefix_fmt, .{0}).len; + stream.pos = prefix_len; + + switch (decrqss) { + // Invalid or unhandled request + .none => {}, + + .sgr => { + const buf = try self.terminal.printAttributes(stream.buffer[stream.pos..]); + + // printAttributes wrote into our buffer, so adjust the stream + // position + stream.pos += buf.len; + + try writer.writeByte('m'); + }, + + .decscusr => { + const blink = self.terminal.modes.get(.cursor_blinking); + const style: u8 = switch (self.terminal.screen.cursor.cursor_style) { + .block => if (blink) 1 else 2, + .underline => if (blink) 3 else 4, + .bar => if (blink) 5 else 6, + }; + try writer.print("{d} q", .{style}); + }, + + .decstbm => { + try writer.print("{d};{d}r", .{ + self.terminal.scrolling_region.top + 1, + self.terminal.scrolling_region.bottom + 1, + }); + }, + + .decslrm => { + // We only send a valid response when left and right + // margin mode (DECLRMM) is enabled. + if (self.terminal.modes.get(.enable_left_and_right_margin)) { + try writer.print("{d};{d}s", .{ + self.terminal.scrolling_region.left + 1, + self.terminal.scrolling_region.right + 1, + }); + } + }, + } + + // Our response is valid if we have a response payload + const valid = stream.pos > prefix_len; + + // Write the terminator + try writer.writeAll("\x1b\\"); + + // Write the response prefix into the buffer + _ = try std.fmt.bufPrint(response[0..prefix_len], prefix_fmt, .{@intFromBool(valid)}); + const msg = try termio.Message.writeReq(self.alloc, response[0..stream.pos]); + self.messageWriter(msg); + }, + } + } + + pub fn apcStart(self: *StreamHandler) !void { + self.apc.start(); + } + + pub fn apcPut(self: *StreamHandler, byte: u8) !void { + self.apc.feed(self.alloc, byte); + } + + pub fn apcEnd(self: *StreamHandler) !void { + var cmd = self.apc.end() orelse return; + defer cmd.deinit(self.alloc); + + // log.warn("APC command: {}", .{cmd}); + switch (cmd) { + .kitty => |*kitty_cmd| { + if (self.terminal.kittyGraphics(self.alloc, kitty_cmd)) |resp| { + var buf: [1024]u8 = undefined; + var buf_stream = std.io.fixedBufferStream(&buf); + try resp.encode(buf_stream.writer()); + const final = buf_stream.getWritten(); + if (final.len > 2) { + // log.warn("kitty graphics response: {s}", .{std.fmt.fmtSliceHexLower(final)}); + self.messageWriter(try termio.Message.writeReq(self.alloc, final)); + } + } + }, + } + } + + pub fn print(self: *StreamHandler, ch: u21) !void { + try self.terminal.print(ch); + } + + pub fn printRepeat(self: *StreamHandler, count: usize) !void { + try self.terminal.printRepeat(count); + } + + pub fn bell(self: StreamHandler) !void { + _ = self; + log.info("BELL", .{}); + } + + pub fn backspace(self: *StreamHandler) !void { + self.terminal.backspace(); + } + + pub fn horizontalTab(self: *StreamHandler, count: u16) !void { + for (0..count) |_| { + const x = self.terminal.screen.cursor.x; + try self.terminal.horizontalTab(); + if (x == self.terminal.screen.cursor.x) break; + } + } + + pub fn horizontalTabBack(self: *StreamHandler, count: u16) !void { + for (0..count) |_| { + const x = self.terminal.screen.cursor.x; + try self.terminal.horizontalTabBack(); + if (x == self.terminal.screen.cursor.x) break; + } + } + + pub fn linefeed(self: *StreamHandler) !void { + // Small optimization: call index instead of linefeed because they're + // identical and this avoids one layer of function call overhead. + try self.terminal.index(); + } + + pub fn carriageReturn(self: *StreamHandler) !void { + self.terminal.carriageReturn(); + } + + pub fn setCursorLeft(self: *StreamHandler, amount: u16) !void { + self.terminal.cursorLeft(amount); + } + + pub fn setCursorRight(self: *StreamHandler, amount: u16) !void { + self.terminal.cursorRight(amount); + } + + pub fn setCursorDown(self: *StreamHandler, amount: u16, carriage: bool) !void { + self.terminal.cursorDown(amount); + if (carriage) self.terminal.carriageReturn(); + } + + pub fn setCursorUp(self: *StreamHandler, amount: u16, carriage: bool) !void { + self.terminal.cursorUp(amount); + if (carriage) self.terminal.carriageReturn(); + } + + pub fn setCursorCol(self: *StreamHandler, col: u16) !void { + self.terminal.setCursorPos(self.terminal.screen.cursor.y + 1, col); + } + + pub fn setCursorColRelative(self: *StreamHandler, offset: u16) !void { + self.terminal.setCursorPos( + self.terminal.screen.cursor.y + 1, + self.terminal.screen.cursor.x + 1 +| offset, + ); + } + + pub fn setCursorRow(self: *StreamHandler, row: u16) !void { + self.terminal.setCursorPos(row, self.terminal.screen.cursor.x + 1); + } + + pub fn setCursorRowRelative(self: *StreamHandler, offset: u16) !void { + self.terminal.setCursorPos( + self.terminal.screen.cursor.y + 1 +| offset, + self.terminal.screen.cursor.x + 1, + ); + } + + pub fn setCursorPos(self: *StreamHandler, row: u16, col: u16) !void { + self.terminal.setCursorPos(row, col); + } + + pub fn eraseDisplay(self: *StreamHandler, mode: terminal.EraseDisplay, protected: bool) !void { + if (mode == .complete) { + // Whenever we erase the full display, scroll to bottom. + try self.terminal.scrollViewport(.{ .bottom = {} }); + try self.queueRender(); + } + + self.terminal.eraseDisplay(mode, protected); + } + + pub fn eraseLine(self: *StreamHandler, mode: terminal.EraseLine, protected: bool) !void { + self.terminal.eraseLine(mode, protected); + } + + pub fn deleteChars(self: *StreamHandler, count: usize) !void { + self.terminal.deleteChars(count); + } + + pub fn eraseChars(self: *StreamHandler, count: usize) !void { + self.terminal.eraseChars(count); + } + + pub fn insertLines(self: *StreamHandler, count: usize) !void { + self.terminal.insertLines(count); + } + + pub fn insertBlanks(self: *StreamHandler, count: usize) !void { + self.terminal.insertBlanks(count); + } + + pub fn deleteLines(self: *StreamHandler, count: usize) !void { + self.terminal.deleteLines(count); + } + + pub fn reverseIndex(self: *StreamHandler) !void { + self.terminal.reverseIndex(); + } + + pub fn index(self: *StreamHandler) !void { + try self.terminal.index(); + } + + pub fn nextLine(self: *StreamHandler) !void { + try self.terminal.index(); + self.terminal.carriageReturn(); + } + + pub fn setTopAndBottomMargin(self: *StreamHandler, top: u16, bot: u16) !void { + self.terminal.setTopAndBottomMargin(top, bot); + } + + pub fn setLeftAndRightMarginAmbiguous(self: *StreamHandler) !void { + if (self.terminal.modes.get(.enable_left_and_right_margin)) { + try self.setLeftAndRightMargin(0, 0); + } else { + try self.saveCursor(); + } + } + + pub fn setLeftAndRightMargin(self: *StreamHandler, left: u16, right: u16) !void { + self.terminal.setLeftAndRightMargin(left, right); + } + + pub fn setModifyKeyFormat(self: *StreamHandler, format: terminal.ModifyKeyFormat) !void { + self.terminal.flags.modify_other_keys_2 = false; + switch (format) { + .other_keys => |v| switch (v) { + .numeric => self.terminal.flags.modify_other_keys_2 = true, + else => {}, + }, + else => {}, + } + } + + pub fn requestMode(self: *StreamHandler, mode_raw: u16, ansi: bool) !void { + // Get the mode value and respond. + const code: u8 = code: { + const mode = terminal.modes.modeFromInt(mode_raw, ansi) orelse break :code 0; + if (self.terminal.modes.get(mode)) break :code 1; + break :code 2; + }; + + var msg: termio.Message = .{ .write_small = .{} }; + const resp = try std.fmt.bufPrint( + &msg.write_small.data, + "\x1B[{s}{};{}$y", + .{ + if (ansi) "" else "?", + mode_raw, + code, + }, + ); + msg.write_small.len = @intCast(resp.len); + self.messageWriter(msg); + } + + pub fn saveMode(self: *StreamHandler, mode: terminal.Mode) !void { + // log.debug("save mode={}", .{mode}); + self.terminal.modes.save(mode); + } + + pub fn restoreMode(self: *StreamHandler, mode: terminal.Mode) !void { + // For restore mode we have to restore but if we set it, we + // always have to call setMode because setting some modes have + // side effects and we want to make sure we process those. + const v = self.terminal.modes.restore(mode); + // log.debug("restore mode={} v={}", .{ mode, v }); + try self.setMode(mode, v); + } + + pub fn setMode(self: *StreamHandler, mode: terminal.Mode, enabled: bool) !void { + // Note: this function doesn't need to grab the render state or + // terminal locks because it is only called from process() which + // grabs the lock. + + // If we are setting cursor blinking, we ignore it if we have + // a default cursor blink setting set. This is a really weird + // behavior so this comment will go deep into trying to explain it. + // + // There are two ways to set cursor blinks: DECSCUSR (CSI _ q) + // and DEC mode 12. DECSCUSR is the modern approach and has a + // way to revert to the "default" (as defined by the terminal) + // cursor style and blink by doing "CSI 0 q". DEC mode 12 controls + // blinking and is either on or off and has no way to set a + // default. DEC mode 12 is also the more antiquated approach. + // + // The problem is that if the user specifies a desired default + // cursor blink with `cursor-style-blink`, the moment a running + // program uses DEC mode 12, the cursor blink can never be reset + // to the default without an explicit DECSCUSR. But if a program + // is using mode 12, it is by definition not using DECSCUSR. + // This makes for somewhat annoying interactions where a poorly + // (or legacy) behaved program will stop blinking, and it simply + // never restarts. + // + // To get around this, we have a special case where if the user + // specifies some explicit default cursor blink desire, we ignore + // DEC mode 12. We allow DECSCUSR to still set the cursor blink + // because programs using DECSCUSR usually are well behaved and + // reset the cursor blink to the default when they exit. + // + // To be extra safe, users can also add a manual `CSI 0 q` to + // their shell config when they render prompts to ensure the + // cursor is exactly as they request. + if (mode == .cursor_blinking and + self.default_cursor_blink != null) + { + return; + } + + // We first always set the raw mode on our mode state. + self.terminal.modes.set(mode, enabled); + + // And then some modes require additional processing. + switch (mode) { + // Just noting here that autorepeat has no effect on + // the terminal. xterm ignores this mode and so do we. + // We know about just so that we don't log that it is + // an unknown mode. + .autorepeat => {}, + + // Schedule a render since we changed colors + .reverse_colors => { + self.terminal.flags.dirty.reverse_colors = true; + try self.queueRender(); + }, + + // Origin resets cursor pos. This is called whether or not + // we're enabling or disabling origin mode and whether or + // not the value changed. + .origin => self.terminal.setCursorPos(1, 1), + + .enable_left_and_right_margin => if (!enabled) { + // When we disable left/right margin mode we need to + // reset the left/right margins. + self.terminal.scrolling_region.left = 0; + self.terminal.scrolling_region.right = self.terminal.cols - 1; + }, + + .alt_screen => { + const opts: terminal.Terminal.AlternateScreenOptions = .{ + .cursor_save = false, + .clear_on_enter = false, + }; + + if (enabled) + self.terminal.alternateScreen(opts) + else + self.terminal.primaryScreen(opts); + + // Schedule a render since we changed screens + try self.queueRender(); + }, + + .alt_screen_save_cursor_clear_enter => { + const opts: terminal.Terminal.AlternateScreenOptions = .{ + .cursor_save = true, + .clear_on_enter = true, + }; + + if (enabled) + self.terminal.alternateScreen(opts) + else + self.terminal.primaryScreen(opts); + + // Schedule a render since we changed screens + try self.queueRender(); + }, + + // Force resize back to the window size + .enable_mode_3 => self.terminal.resize( + self.alloc, + self.grid_size.columns, + self.grid_size.rows, + ) catch |err| { + log.err("error updating terminal size: {}", .{err}); + }, + + .@"132_column" => try self.terminal.deccolm( + self.alloc, + if (enabled) .@"132_cols" else .@"80_cols", + ), + + // We need to start a timer to prevent the emulator being hung + // forever. + .synchronized_output => { + if (enabled) self.messageWriter(.{ .start_synchronized_output = {} }); + try self.queueRender(); + }, + + .linefeed => { + self.messageWriter(.{ .linefeed_mode = enabled }); + }, + + .mouse_event_x10 => { + if (enabled) { + self.terminal.flags.mouse_event = .x10; + try self.setMouseShape(.default); + } else { + self.terminal.flags.mouse_event = .none; + try self.setMouseShape(.text); + } + }, + .mouse_event_normal => { + if (enabled) { + self.terminal.flags.mouse_event = .normal; + try self.setMouseShape(.default); + } else { + self.terminal.flags.mouse_event = .none; + try self.setMouseShape(.text); + } + }, + .mouse_event_button => { + if (enabled) { + self.terminal.flags.mouse_event = .button; + try self.setMouseShape(.default); + } else { + self.terminal.flags.mouse_event = .none; + try self.setMouseShape(.text); + } + }, + .mouse_event_any => { + if (enabled) { + self.terminal.flags.mouse_event = .any; + try self.setMouseShape(.default); + } else { + self.terminal.flags.mouse_event = .none; + try self.setMouseShape(.text); + } + }, + + .mouse_format_utf8 => self.terminal.flags.mouse_format = if (enabled) .utf8 else .x10, + .mouse_format_sgr => self.terminal.flags.mouse_format = if (enabled) .sgr else .x10, + .mouse_format_urxvt => self.terminal.flags.mouse_format = if (enabled) .urxvt else .x10, + .mouse_format_sgr_pixels => self.terminal.flags.mouse_format = if (enabled) .sgr_pixels else .x10, + + else => {}, + } + } + + pub fn setMouseShiftCapture(self: *StreamHandler, v: bool) !void { + self.terminal.flags.mouse_shift_capture = if (v) .true else .false; + } + + pub fn setAttribute(self: *StreamHandler, attr: terminal.Attribute) !void { + switch (attr) { + .unknown => |unk| log.warn("unimplemented or unknown SGR attribute: {any}", .{unk}), + + else => self.terminal.setAttribute(attr) catch |err| + log.warn("error setting attribute {}: {}", .{ attr, err }), + } + } + + pub fn startHyperlink(self: *StreamHandler, uri: []const u8, id: ?[]const u8) !void { + try self.terminal.screen.startHyperlink(uri, id); + } + + pub fn endHyperlink(self: *StreamHandler) !void { + self.terminal.screen.endHyperlink(); + } + + pub fn deviceAttributes( + self: *StreamHandler, + req: terminal.DeviceAttributeReq, + params: []const u16, + ) !void { + _ = params; + + // For the below, we quack as a VT220. We don't quack as + // a 420 because we don't support DCS sequences. + switch (req) { + .primary => self.messageWriter(.{ + .write_stable = "\x1B[?62;22c", + }), + + .secondary => self.messageWriter(.{ + .write_stable = "\x1B[>1;10;0c", + }), + + else => log.warn("unimplemented device attributes req: {}", .{req}), + } + } + + pub fn deviceStatusReport( + self: *StreamHandler, + req: terminal.device_status.Request, + ) !void { + switch (req) { + .operating_status => self.messageWriter(.{ .write_stable = "\x1B[0n" }), + + .cursor_position => { + const pos: struct { + x: usize, + y: usize, + } = if (self.terminal.modes.get(.origin)) .{ + .x = self.terminal.screen.cursor.x -| self.terminal.scrolling_region.left, + .y = self.terminal.screen.cursor.y -| self.terminal.scrolling_region.top, + } else .{ + .x = self.terminal.screen.cursor.x, + .y = self.terminal.screen.cursor.y, + }; + + // Response always is at least 4 chars, so this leaves the + // remainder for the row/column as base-10 numbers. This + // will support a very large terminal. + var msg: termio.Message = .{ .write_small = .{} }; + const resp = try std.fmt.bufPrint(&msg.write_small.data, "\x1B[{};{}R", .{ + pos.y + 1, + pos.x + 1, + }); + msg.write_small.len = @intCast(resp.len); + + self.messageWriter(msg); + }, + + .color_scheme => self.surfaceMessageWriter(.{ .report_color_scheme = {} }), + } + } + + pub fn setCursorStyle( + self: *StreamHandler, + style: terminal.CursorStyleReq, + ) !void { + // Assume we're setting to a non-default. + self.default_cursor = false; + + switch (style) { + .default => { + self.default_cursor = true; + self.terminal.screen.cursor.cursor_style = self.default_cursor_style; + self.terminal.modes.set( + .cursor_blinking, + self.default_cursor_blink orelse true, + ); + }, + + .blinking_block => { + self.terminal.screen.cursor.cursor_style = .block; + self.terminal.modes.set(.cursor_blinking, true); + }, + + .steady_block => { + self.terminal.screen.cursor.cursor_style = .block; + self.terminal.modes.set(.cursor_blinking, false); + }, + + .blinking_underline => { + self.terminal.screen.cursor.cursor_style = .underline; + self.terminal.modes.set(.cursor_blinking, true); + }, + + .steady_underline => { + self.terminal.screen.cursor.cursor_style = .underline; + self.terminal.modes.set(.cursor_blinking, false); + }, + + .blinking_bar => { + self.terminal.screen.cursor.cursor_style = .bar; + self.terminal.modes.set(.cursor_blinking, true); + }, + + .steady_bar => { + self.terminal.screen.cursor.cursor_style = .bar; + self.terminal.modes.set(.cursor_blinking, false); + }, + + else => log.warn("unimplemented cursor style: {}", .{style}), + } + } + + pub fn setProtectedMode(self: *StreamHandler, mode: terminal.ProtectedMode) !void { + self.terminal.setProtectedMode(mode); + } + + pub fn decaln(self: *StreamHandler) !void { + try self.terminal.decaln(); + } + + pub fn tabClear(self: *StreamHandler, cmd: terminal.TabClear) !void { + self.terminal.tabClear(cmd); + } + + pub fn tabSet(self: *StreamHandler) !void { + self.terminal.tabSet(); + } + + pub fn tabReset(self: *StreamHandler) !void { + self.terminal.tabReset(); + } + + pub fn saveCursor(self: *StreamHandler) !void { + self.terminal.saveCursor(); + } + + pub fn restoreCursor(self: *StreamHandler) !void { + try self.terminal.restoreCursor(); + } + + pub fn enquiry(self: *StreamHandler) !void { + log.debug("sending enquiry response={s}", .{self.enquiry_response}); + self.messageWriter(try termio.Message.writeReq(self.alloc, self.enquiry_response)); + } + + pub fn scrollDown(self: *StreamHandler, count: usize) !void { + self.terminal.scrollDown(count); + } + + pub fn scrollUp(self: *StreamHandler, count: usize) !void { + self.terminal.scrollUp(count); + } + + pub fn setActiveStatusDisplay( + self: *StreamHandler, + req: terminal.StatusDisplay, + ) !void { + self.terminal.status_display = req; + } + + pub fn configureCharset( + self: *StreamHandler, + slot: terminal.CharsetSlot, + set: terminal.Charset, + ) !void { + self.terminal.configureCharset(slot, set); + } + + pub fn invokeCharset( + self: *StreamHandler, + active: terminal.CharsetActiveSlot, + slot: terminal.CharsetSlot, + single: bool, + ) !void { + self.terminal.invokeCharset(active, slot, single); + } + + pub fn fullReset( + self: *StreamHandler, + ) !void { + self.terminal.fullReset(); + try self.setMouseShape(.text); + } + + pub fn queryKittyKeyboard(self: *StreamHandler) !void { + if (comptime disable_kitty_keyboard_protocol) return; + + log.debug("querying kitty keyboard mode", .{}); + var data: termio.Message.WriteReq.Small.Array = undefined; + const resp = try std.fmt.bufPrint(&data, "\x1b[?{}u", .{ + self.terminal.screen.kitty_keyboard.current().int(), + }); + + self.messageWriter(.{ + .write_small = .{ + .data = data, + .len = @intCast(resp.len), + }, + }); + } + + pub fn pushKittyKeyboard( + self: *StreamHandler, + flags: terminal.kitty.KeyFlags, + ) !void { + if (comptime disable_kitty_keyboard_protocol) return; + + log.debug("pushing kitty keyboard mode: {}", .{flags}); + self.terminal.screen.kitty_keyboard.push(flags); + } + + pub fn popKittyKeyboard(self: *StreamHandler, n: u16) !void { + if (comptime disable_kitty_keyboard_protocol) return; + + log.debug("popping kitty keyboard mode n={}", .{n}); + self.terminal.screen.kitty_keyboard.pop(@intCast(n)); + } + + pub fn setKittyKeyboard( + self: *StreamHandler, + mode: terminal.kitty.KeySetMode, + flags: terminal.kitty.KeyFlags, + ) !void { + if (comptime disable_kitty_keyboard_protocol) return; + + log.debug("setting kitty keyboard mode: {} {}", .{ mode, flags }); + self.terminal.screen.kitty_keyboard.set(mode, flags); + } + + pub fn reportXtversion( + self: *StreamHandler, + ) !void { + log.debug("reporting XTVERSION: ghostty {s}", .{build_config.version_string}); + var buf: [288]u8 = undefined; + const resp = try std.fmt.bufPrint( + &buf, + "\x1BP>|{s} {s}\x1B\\", + .{ + "ghostty", + build_config.version_string, + }, + ); + const msg = try termio.Message.writeReq(self.alloc, resp); + self.messageWriter(msg); + } + + //------------------------------------------------------------------------- + // OSC + + pub fn changeWindowTitle(self: *StreamHandler, title: []const u8) !void { + var buf: [256]u8 = undefined; + if (title.len >= buf.len) { + log.warn("change title requested larger than our buffer size, ignoring", .{}); + return; + } + + @memcpy(buf[0..title.len], title); + buf[title.len] = 0; + + // Mark that we've seen a title + self.seen_title = true; + self.surfaceMessageWriter(.{ .set_title = buf }); + } + + pub fn setMouseShape( + self: *StreamHandler, + shape: terminal.MouseShape, + ) !void { + // Avoid changing the shape it it is already set to avoid excess + // cross-thread messaging. + if (self.terminal.mouse_shape == shape) return; + + self.terminal.mouse_shape = shape; + self.surfaceMessageWriter(.{ .set_mouse_shape = shape }); + } + + pub fn clipboardContents(self: *StreamHandler, kind: u8, data: []const u8) !void { + // Note: we ignore the "kind" field and always use the standard clipboard. + // iTerm also appears to do this but other terminals seem to only allow + // certain. Let's investigate more. + + const clipboard_type: apprt.Clipboard = switch (kind) { + 'c' => .standard, + 's' => .selection, + 'p' => .primary, + else => .standard, + }; + + // Get clipboard contents + if (data.len == 1 and data[0] == '?') { + self.surfaceMessageWriter(.{ .clipboard_read = clipboard_type }); + return; + } + + // Write clipboard contents + self.surfaceMessageWriter(.{ + .clipboard_write = .{ + .req = try apprt.surface.Message.WriteReq.init( + self.alloc, + data, + ), + .clipboard_type = clipboard_type, + }, + }); + } + + pub fn promptStart(self: *StreamHandler, aid: ?[]const u8, redraw: bool) !void { + _ = aid; + self.terminal.markSemanticPrompt(.prompt); + self.terminal.flags.shell_redraws_prompt = redraw; + } + + pub fn promptContinuation(self: *StreamHandler, aid: ?[]const u8) !void { + _ = aid; + self.terminal.markSemanticPrompt(.prompt_continuation); + } + + pub fn promptEnd(self: *StreamHandler) !void { + self.terminal.markSemanticPrompt(.input); + } + + pub fn endOfInput(self: *StreamHandler) !void { + self.terminal.markSemanticPrompt(.command); + } + + pub fn reportPwd(self: *StreamHandler, url: []const u8) !void { + if (builtin.os.tag == .windows) { + log.warn("reportPwd unimplemented on windows", .{}); + return; + } + + const uri = std.Uri.parse(url) catch |e| { + log.warn("invalid url in OSC 7: {}", .{e}); + return; + }; + + if (!std.mem.eql(u8, "file", uri.scheme) and + !std.mem.eql(u8, "kitty-shell-cwd", uri.scheme)) + { + log.warn("OSC 7 scheme must be file, got: {s}", .{uri.scheme}); + return; + } + + // OSC 7 is a little sketchy because anyone can send any value from + // any host (such an SSH session). The best practice terminals follow + // is to valid the hostname to be local. + const host_valid = host_valid: { + const host_component = uri.host orelse break :host_valid false; + + // Get the raw string of the URI. Its unclear to me if the various + // tags of this enum guarantee no percent-encoding so we just + // check all of it. This isn't a performance critical path. + const host = switch (host_component) { + .raw => |v| v, + .percent_encoded => |v| v, + }; + if (host.len == 0 or std.mem.eql(u8, "localhost", host)) { + break :host_valid true; + } + + // Otherwise, it must match our hostname. + var buf: [posix.HOST_NAME_MAX]u8 = undefined; + const hostname = posix.gethostname(&buf) catch |err| { + log.warn("failed to get hostname for OSC 7 validation: {}", .{err}); + break :host_valid false; + }; + + break :host_valid std.mem.eql(u8, host, hostname); + }; + if (!host_valid) { + log.warn("OSC 7 host must be local", .{}); + return; + } + + // We need to unescape the path. We first try to unescape onto + // the stack and fall back to heap allocation if we have to. + var pathBuf: [1024]u8 = undefined; + const path, const heap = path: { + // Get the raw string of the URI. Its unclear to me if the various + // tags of this enum guarantee no percent-encoding so we just + // check all of it. This isn't a performance critical path. + const path = switch (uri.path) { + .raw => |v| v, + .percent_encoded => |v| v, + }; + + // If the path doesn't have any escapes, we can use it directly. + if (std.mem.indexOfScalar(u8, path, '%') == null) + break :path .{ path, false }; + + // First try to stack-allocate + var fba = std.heap.FixedBufferAllocator.init(&pathBuf); + if (std.fmt.allocPrint(fba.allocator(), "{raw}", .{uri.path})) |v| + break :path .{ v, false } + else |_| {} + + // Fall back to heap + if (std.fmt.allocPrint(self.alloc, "{raw}", .{uri.path})) |v| + break :path .{ v, true } + else |_| {} + + // Fall back to using it directly... + log.warn("failed to unescape OSC 7 path, using it directly path={s}", .{path}); + break :path .{ path, false }; + }; + defer if (heap) self.alloc.free(path); + + log.debug("terminal pwd: {s}", .{path}); + try self.terminal.setPwd(path); + + // If we haven't seen a title, use our pwd as the title. + if (!self.seen_title) { + try self.changeWindowTitle(path); + self.seen_title = false; + } + } + + /// Implements OSC 4, OSC 10, and OSC 11, which reports palette color, + /// default foreground color, and background color respectively. + pub fn reportColor( + self: *StreamHandler, + kind: terminal.osc.Command.ColorKind, + terminator: terminal.osc.Terminator, + ) !void { + if (self.osc_color_report_format == .none) return; + + const color = switch (kind) { + .palette => |i| self.terminal.color_palette.colors[i], + .foreground => self.foreground_color, + .background => self.background_color, + .cursor => self.cursor_color orelse self.foreground_color, + }; + + var msg: termio.Message = .{ .write_small = .{} }; + const resp = switch (self.osc_color_report_format) { + .@"16-bit" => switch (kind) { + .palette => |i| try std.fmt.bufPrint( + &msg.write_small.data, + "\x1B]{s};{d};rgb:{x:0>4}/{x:0>4}/{x:0>4}{s}", + .{ + kind.code(), + i, + @as(u16, color.r) * 257, + @as(u16, color.g) * 257, + @as(u16, color.b) * 257, + terminator.string(), + }, + ), + else => try std.fmt.bufPrint( + &msg.write_small.data, + "\x1B]{s};rgb:{x:0>4}/{x:0>4}/{x:0>4}{s}", + .{ + kind.code(), + @as(u16, color.r) * 257, + @as(u16, color.g) * 257, + @as(u16, color.b) * 257, + terminator.string(), + }, + ), + }, + + .@"8-bit" => switch (kind) { + .palette => |i| try std.fmt.bufPrint( + &msg.write_small.data, + "\x1B]{s};{d};rgb:{x:0>2}/{x:0>2}/{x:0>2}{s}", + .{ + kind.code(), + i, + @as(u16, color.r), + @as(u16, color.g), + @as(u16, color.b), + terminator.string(), + }, + ), + else => try std.fmt.bufPrint( + &msg.write_small.data, + "\x1B]{s};rgb:{x:0>2}/{x:0>2}/{x:0>2}{s}", + .{ + kind.code(), + @as(u16, color.r), + @as(u16, color.g), + @as(u16, color.b), + terminator.string(), + }, + ), + }, + .none => unreachable, // early return above + }; + msg.write_small.len = @intCast(resp.len); + self.messageWriter(msg); + } + + pub fn setColor( + self: *StreamHandler, + kind: terminal.osc.Command.ColorKind, + value: []const u8, + ) !void { + const color = try terminal.color.RGB.parse(value); + + switch (kind) { + .palette => |i| { + self.terminal.flags.dirty.palette = true; + self.terminal.color_palette.colors[i] = color; + self.terminal.color_palette.mask.set(i); + }, + .foreground => { + self.foreground_color = color; + _ = self.renderer_mailbox.push(.{ + .foreground_color = color, + }, .{ .forever = {} }); + }, + .background => { + self.background_color = color; + _ = self.renderer_mailbox.push(.{ + .background_color = color, + }, .{ .forever = {} }); + }, + .cursor => { + self.cursor_color = color; + _ = self.renderer_mailbox.push(.{ + .cursor_color = color, + }, .{ .forever = {} }); + }, + } + } + + pub fn resetColor( + self: *StreamHandler, + kind: terminal.osc.Command.ColorKind, + value: []const u8, + ) !void { + switch (kind) { + .palette => { + const mask = &self.terminal.color_palette.mask; + if (value.len == 0) { + // Find all bit positions in the mask which are set and + // reset those indices to the default palette + var it = mask.iterator(.{}); + while (it.next()) |i| { + self.terminal.flags.dirty.palette = true; + self.terminal.color_palette.colors[i] = self.terminal.default_palette[i]; + mask.unset(i); + } + } else { + var it = std.mem.tokenizeScalar(u8, value, ';'); + while (it.next()) |param| { + // Skip invalid parameters + const i = std.fmt.parseUnsigned(u8, param, 10) catch continue; + if (mask.isSet(i)) { + self.terminal.flags.dirty.palette = true; + self.terminal.color_palette.colors[i] = self.terminal.default_palette[i]; + mask.unset(i); + } + } + } + }, + .foreground => { + self.foreground_color = self.default_foreground_color; + _ = self.renderer_mailbox.push(.{ + .foreground_color = self.foreground_color, + }, .{ .forever = {} }); + }, + .background => { + self.background_color = self.default_background_color; + _ = self.renderer_mailbox.push(.{ + .background_color = self.background_color, + }, .{ .forever = {} }); + }, + .cursor => { + self.cursor_color = self.default_cursor_color; + _ = self.renderer_mailbox.push(.{ + .cursor_color = self.cursor_color, + }, .{ .forever = {} }); + }, + } + } + + pub fn showDesktopNotification( + self: *StreamHandler, + title: []const u8, + body: []const u8, + ) !void { + var message = apprt.surface.Message{ .desktop_notification = undefined }; + + const title_len = @min(title.len, message.desktop_notification.title.len); + @memcpy(message.desktop_notification.title[0..title_len], title[0..title_len]); + message.desktop_notification.title[title_len] = 0; + + const body_len = @min(body.len, message.desktop_notification.body.len); + @memcpy(message.desktop_notification.body[0..body_len], body[0..body_len]); + message.desktop_notification.body[body_len] = 0; + + self.surfaceMessageWriter(message); + } +};