From c3040a04a33a0662af1850de06b3c2f7645c7397 Mon Sep 17 00:00:00 2001 From: Bananymous Date: Sun, 15 Sep 2024 03:09:10 +0300 Subject: [PATCH] Terminal: Optimize printing a lot Terminal used to run `yes` at around 400 lines per second This patch pumps that to over 100'000 lines per second! There are 2 optimizations done: 1. only invalidate window once after rendering is done 2. if printing more than `rows()` newlines skip prior data --- userspace/programs/Terminal/Terminal.cpp | 95 ++++++++++++++++++------ userspace/programs/Terminal/Terminal.h | 24 +++++- 2 files changed, 97 insertions(+), 22 deletions(-) diff --git a/userspace/programs/Terminal/Terminal.cpp b/userspace/programs/Terminal/Terminal.cpp index 4bac57f9d9..6f5e313148 100644 --- a/userspace/programs/Terminal/Terminal.cpp +++ b/userspace/programs/Terminal/Terminal.cpp @@ -196,14 +196,69 @@ void Terminal::show_cursor() bool Terminal::read_shell() { - char buffer[128]; - ssize_t nread = read(m_shell_info.pts_master, buffer, sizeof(buffer) - 1); + char buffer[512]; + ssize_t nread = read(m_shell_info.pts_master, buffer, sizeof(buffer)); if (nread < 0) dwarnln("read: {}", strerror(errno)); if (nread <= 0) return false; - for (ssize_t i = 0; i < nread; i++) - putchar(buffer[i]); + + Rectangle should_invalidate; + + ssize_t i = 0; + while (i < nread) + { + // all ansi escape codes must be handled + if (buffer[i] == '\e') + { + while (i < nread) + { + char ch = buffer[i++]; + should_invalidate = should_invalidate.get_bounding_box(putchar(ch)); + if (isalpha(ch)) + break; + } + continue; + } + + // find the next ansi escape code or end of buffer + size_t non_ansi_end = i; + while (non_ansi_end < nread && buffer[non_ansi_end] != '\e') + non_ansi_end++; + + // we only need to process maximum of `rows()` newlines. + // anything before that would get overwritten anyway + size_t start = non_ansi_end; + size_t newline_count = 0; + while (start > i && newline_count < rows()) + newline_count += (buffer[--start] == '\n'); + + // do possible scrolling already in here, so `putchar()` doesnt + // have to scroll up to `rows()` times + if (m_cursor.y + newline_count >= rows()) + { + const uint32_t scroll = m_cursor.y + newline_count - rows() + 1; + m_cursor.y -= scroll; + m_window->shift_vertical(-scroll * (int32_t)m_font.height()); + m_window->fill_rect(0, m_window->height() - scroll * m_font.height(), m_window->width(), scroll * m_font.height(), m_bg_color); + should_invalidate = { 0, 0, m_window->width(), m_window->height() }; + } + + i = start; + for (i = start; i < non_ansi_end; i++) + should_invalidate = should_invalidate.get_bounding_box(putchar(buffer[i])); + } + + if (should_invalidate.height && should_invalidate.width) + { + m_window->invalidate( + should_invalidate.x, + should_invalidate.y, + should_invalidate.width, + should_invalidate.height + ); + } + return true; } @@ -361,7 +416,7 @@ void Terminal::handle_csi(char ch) m_state = State::Normal; } -void Terminal::putchar(uint8_t ch) +Rectangle Terminal::putchar(uint8_t ch) { if (m_state == State::ESC) { @@ -369,14 +424,14 @@ void Terminal::putchar(uint8_t ch) { dprintln("unknown escape character 0x{2H}", ch); m_state = State::Normal; - return; + return {}; } m_state = State::CSI; m_csi_info.index = 0; m_csi_info.fields[0] = -1; m_csi_info.fields[1] = -1; m_csi_info.question = false; - return; + return {}; } if (m_state == State::CSI) @@ -385,10 +440,10 @@ void Terminal::putchar(uint8_t ch) { dprintln("invalid CSI 0x{2H}", ch); m_state = State::Normal; - return; + return {}; } handle_csi(ch); - return; + return {}; } m_utf8_bytes[m_utf8_index++] = ch; @@ -398,10 +453,10 @@ void Terminal::putchar(uint8_t ch) { dwarnln("invalid utf8 leading byte 0x{2H}", ch); m_utf8_index = 0; - return; + return {}; } if (m_utf8_index < utf8_len) - return; + return {}; const uint32_t codepoint = BAN::UTF8::to_codepoint(m_utf8_bytes); m_utf8_index = 0; @@ -421,9 +476,11 @@ void Terminal::putchar(uint8_t ch) *--ptr = '\0'; dwarnln("invalid utf8 {}", utf8_hex); - return; + return {}; } + Rectangle should_invalidate; + switch (codepoint) { case '\e': @@ -449,7 +506,7 @@ void Terminal::putchar(uint8_t ch) m_window->fill_rect(cell_x, cell_y, cell_w, cell_h, m_bg_color); m_window->draw_character(codepoint, m_font, cell_x, cell_y, m_fg_color); - m_window->invalidate(cell_x, cell_y, cell_w, cell_h); + should_invalidate = { cell_x, cell_y, cell_w, cell_h }; m_cursor.x++; break; } @@ -461,14 +518,10 @@ void Terminal::putchar(uint8_t ch) m_cursor.y++; } - if (m_cursor.y >= rows()) - { - uint32_t scroll = m_cursor.y - rows() + 1; - m_cursor.y -= scroll; - m_window->shift_vertical(-scroll * (int32_t)m_font.height()); - m_window->fill_rect(0, m_window->height() - scroll * m_font.height(), m_window->width(), scroll * m_font.height(), m_bg_color); - m_window->invalidate(); - } + // scrolling is already handled in `read_shell()` + ASSERT(m_cursor.y < rows()); + + return should_invalidate; } void Terminal::on_key_event(LibGUI::EventPacket::KeyEvent event) diff --git a/userspace/programs/Terminal/Terminal.h b/userspace/programs/Terminal/Terminal.h index 22b5a7702d..d39983bebc 100644 --- a/userspace/programs/Terminal/Terminal.h +++ b/userspace/programs/Terminal/Terminal.h @@ -4,6 +4,28 @@ #include +struct Rectangle +{ + uint32_t x { 0 }; + uint32_t y { 0 }; + uint32_t width { 0 }; + uint32_t height { 0 }; + + Rectangle get_bounding_box(Rectangle other) const + { + const auto min_x = BAN::Math::min(x, other.x); + const auto min_y = BAN::Math::min(y, other.y); + const auto max_x = BAN::Math::max(x + width, other.x + other.width); + const auto max_y = BAN::Math::max(y + height, other.y + other.height); + return Rectangle { + .x = min_x, + .y = min_y, + .width = max_x - min_x, + .height = max_y - min_y, + }; + } +}; + class Terminal { public: @@ -15,7 +37,7 @@ public: private: void handle_csi(char ch); void handle_sgr(); - void putchar(uint8_t ch); + Rectangle putchar(uint8_t ch); bool read_shell(); void hide_cursor();