dotemacs

My Emacs configuration
git clone git://git.entf.net/dotemacs
Log | Files | Refs | LICENSE

vterm-module.c (51117B)


      1 #include "vterm-module.h"
      2 #include "elisp.h"
      3 #include "utf8.h"
      4 #include <assert.h>
      5 #include <fcntl.h>
      6 #include <limits.h>
      7 #include <stdio.h>
      8 #include <string.h>
      9 #include <termios.h>
     10 #include <unistd.h>
     11 #include <vterm.h>
     12 
     13 static LineInfo *alloc_lineinfo() {
     14   LineInfo *info = malloc(sizeof(LineInfo));
     15   info->directory = NULL;
     16   info->prompt_col = -1;
     17   return info;
     18 }
     19 void free_lineinfo(LineInfo *line) {
     20   if (line == NULL) {
     21     return;
     22   }
     23   if (line->directory != NULL) {
     24     free(line->directory);
     25     line->directory = NULL;
     26   }
     27   free(line);
     28 }
     29 static int term_sb_push(int cols, const VTermScreenCell *cells, void *data) {
     30   Term *term = (Term *)data;
     31 
     32   if (!term->sb_size) {
     33     return 0;
     34   }
     35 
     36   // copy vterm cells into sb_buffer
     37   size_t c = (size_t)cols;
     38   ScrollbackLine *sbrow = NULL;
     39   if (term->sb_current == term->sb_size) {
     40     if (term->sb_buffer[term->sb_current - 1]->cols == c) {
     41       // Recycle old row if it's the right size
     42       sbrow = term->sb_buffer[term->sb_current - 1];
     43     } else {
     44       if (term->sb_buffer[term->sb_current - 1]->info != NULL) {
     45         free_lineinfo(term->sb_buffer[term->sb_current - 1]->info);
     46         term->sb_buffer[term->sb_current - 1]->info = NULL;
     47       }
     48       free(term->sb_buffer[term->sb_current - 1]);
     49     }
     50 
     51     // Make room at the start by shifting to the right.
     52     memmove(term->sb_buffer + 1, term->sb_buffer,
     53             sizeof(term->sb_buffer[0]) * (term->sb_current - 1));
     54 
     55   } else if (term->sb_current > 0) {
     56     // Make room at the start by shifting to the right.
     57     memmove(term->sb_buffer + 1, term->sb_buffer,
     58             sizeof(term->sb_buffer[0]) * term->sb_current);
     59   }
     60 
     61   if (!sbrow) {
     62     sbrow = malloc(sizeof(ScrollbackLine) + c * sizeof(sbrow->cells[0]));
     63     sbrow->cols = c;
     64     sbrow->info = NULL;
     65   }
     66   if (sbrow->info != NULL) {
     67     free_lineinfo(sbrow->info);
     68   }
     69   sbrow->info = term->lines[0];
     70   memmove(term->lines, term->lines + 1,
     71           sizeof(term->lines[0]) * (term->lines_len - 1));
     72   if (term->resizing) {
     73     /* pushed by window height decr */
     74     if (term->lines[term->lines_len - 1] != NULL) {
     75       /* do not need free here ,it is reused ,we just need set null */
     76       term->lines[term->lines_len - 1] = NULL;
     77     }
     78     term->lines_len--;
     79   } else {
     80     LineInfo *lastline = term->lines[term->lines_len - 1];
     81     if (lastline != NULL) {
     82       LineInfo *line = alloc_lineinfo();
     83       if (lastline->directory != NULL) {
     84         line->directory = malloc(1 + strlen(lastline->directory));
     85         strcpy(line->directory, lastline->directory);
     86       }
     87       term->lines[term->lines_len - 1] = line;
     88     }
     89   }
     90 
     91   // New row is added at the start of the storage buffer.
     92   term->sb_buffer[0] = sbrow;
     93   if (term->sb_current < term->sb_size) {
     94     term->sb_current++;
     95   }
     96 
     97   if (term->sb_pending < term->sb_size) {
     98     term->sb_pending++;
     99     /* when window height decreased */
    100     if (term->height_resize < 0 &&
    101         term->sb_pending_by_height_decr < -term->height_resize) {
    102       term->sb_pending_by_height_decr++;
    103     }
    104   }
    105 
    106   memcpy(sbrow->cells, cells, c * sizeof(cells[0]));
    107 
    108   return 1;
    109 }
    110 /// Scrollback pop handler (from pangoterm).
    111 ///
    112 /// @param cols
    113 /// @param cells  VTerm state to update.
    114 /// @param data   Term
    115 static int term_sb_pop(int cols, VTermScreenCell *cells, void *data) {
    116   Term *term = (Term *)data;
    117 
    118   if (!term->sb_current) {
    119     return 0;
    120   }
    121 
    122   if (term->sb_pending) {
    123     term->sb_pending--;
    124   }
    125 
    126   ScrollbackLine *sbrow = term->sb_buffer[0];
    127   term->sb_current--;
    128   // Forget the "popped" row by shifting the rest onto it.
    129   memmove(term->sb_buffer, term->sb_buffer + 1,
    130           sizeof(term->sb_buffer[0]) * (term->sb_current));
    131 
    132   size_t cols_to_copy = (size_t)cols;
    133   if (cols_to_copy > sbrow->cols) {
    134     cols_to_copy = sbrow->cols;
    135   }
    136 
    137   // copy to vterm state
    138   memcpy(cells, sbrow->cells, sizeof(cells[0]) * cols_to_copy);
    139   size_t col;
    140   for (col = cols_to_copy; col < (size_t)cols; col++) {
    141     cells[col].chars[0] = 0;
    142     cells[col].width = 1;
    143   }
    144 
    145   LineInfo **lines = malloc(sizeof(LineInfo *) * (term->lines_len + 1));
    146 
    147   memmove(lines + 1, term->lines, sizeof(term->lines[0]) * term->lines_len);
    148   lines[0] = sbrow->info;
    149   free(sbrow);
    150   term->lines_len += 1;
    151   free(term->lines);
    152   term->lines = lines;
    153 
    154   return 1;
    155 }
    156 
    157 static int term_sb_clear(void *data) {
    158   Term *term = (Term *)data;
    159 
    160   if (term->sb_clear_pending) {
    161     // Another scrollback clear is already pending, so skip this one.
    162     return 0;
    163   }
    164 
    165   for (int i = 0; i < term->sb_current; i++) {
    166     if (term->sb_buffer[i]->info != NULL) {
    167       free_lineinfo(term->sb_buffer[i]->info);
    168       term->sb_buffer[i]->info = NULL;
    169     }
    170     free(term->sb_buffer[i]);
    171   }
    172   free(term->sb_buffer);
    173   term->sb_buffer = malloc(sizeof(ScrollbackLine *) * term->sb_size);
    174   term->sb_clear_pending = true;
    175   term->sb_current = 0;
    176   term->sb_pending = 0;
    177   term->sb_pending_by_height_decr = 0;
    178   invalidate_terminal(term, -1, -1);
    179 
    180   return 0;
    181 }
    182 
    183 static int row_to_linenr(Term *term, int row) {
    184   return row != INT_MAX ? row + (int)term->sb_current + 1 : INT_MAX;
    185 }
    186 
    187 static int linenr_to_row(Term *term, int linenr) {
    188   return linenr - (int)term->sb_current - 1;
    189 }
    190 
    191 static void fetch_cell(Term *term, int row, int col, VTermScreenCell *cell) {
    192   if (row < 0) {
    193     ScrollbackLine *sbrow = term->sb_buffer[-row - 1];
    194     if ((size_t)col < sbrow->cols) {
    195       *cell = sbrow->cells[col];
    196     } else {
    197       // fill the pointer with an empty cell
    198       VTermColor fg, bg;
    199       VTermState *state = vterm_obtain_state(term->vt);
    200       vterm_state_get_default_colors(state, &fg, &bg);
    201 
    202       *cell = (VTermScreenCell){.chars = {0}, .width = 1, .bg = bg};
    203     }
    204   } else {
    205     vterm_screen_get_cell(term->vts, (VTermPos){.row = row, .col = col}, cell);
    206   }
    207 }
    208 
    209 static char *get_row_directory(Term *term, int row) {
    210   if (row < 0) {
    211     ScrollbackLine *sbrow = term->sb_buffer[-row - 1];
    212     return sbrow->info->directory;
    213     /* return term->dirs[0]; */
    214   } else {
    215     LineInfo *line = term->lines[row];
    216     return line ? line->directory : NULL;
    217   }
    218 }
    219 static LineInfo *get_lineinfo(Term *term, int row) {
    220   if (row < 0) {
    221     ScrollbackLine *sbrow = term->sb_buffer[-row - 1];
    222     return sbrow->info;
    223     /* return term->dirs[0]; */
    224   } else {
    225     return term->lines[row];
    226   }
    227 }
    228 static bool is_eol(Term *term, int end_col, int row, int col) {
    229   /* This cell is EOL if this and every cell to the right is black */
    230   if (row >= 0) {
    231     VTermPos pos = {.row = row, .col = col};
    232     return vterm_screen_is_eol(term->vts, pos);
    233   }
    234 
    235   ScrollbackLine *sbrow = term->sb_buffer[-row - 1];
    236   int c;
    237   for (c = col; c < end_col && c < sbrow->cols;) {
    238     if (sbrow->cells[c].chars[0]) {
    239       return 0;
    240     }
    241     c += sbrow->cells[c].width;
    242   }
    243   return 1;
    244 }
    245 static int is_end_of_prompt(Term *term, int end_col, int row, int col) {
    246   LineInfo *info = get_lineinfo(term, row);
    247   if (info == NULL) {
    248     return 0;
    249   }
    250   if (info->prompt_col < 0) {
    251     return 0;
    252   }
    253   if (info->prompt_col == col) {
    254     return 1;
    255   }
    256   if (is_eol(term, end_col, row, col) && info->prompt_col >= col) {
    257     return 1;
    258   }
    259   return 0;
    260 }
    261 
    262 static void goto_col(Term *term, emacs_env *env, int row, int end_col) {
    263   int col = 0;
    264   size_t offset = 0;
    265   size_t beyond_eol = 0;
    266 
    267   int height;
    268   int width;
    269   vterm_get_size(term->vt, &height, &width);
    270 
    271   while (col < end_col) {
    272     VTermScreenCell cell;
    273     fetch_cell(term, row, col, &cell);
    274     if (cell.chars[0]) {
    275       if (cell.width > 1) {
    276         offset += cell.width - 1;
    277       }
    278     } else {
    279       if (is_eol(term, term->width, row, col)) {
    280         offset += cell.width;
    281         beyond_eol += cell.width;
    282       }
    283     }
    284     col += cell.width;
    285   }
    286 
    287   forward_char(env, env->make_integer(env, end_col - offset));
    288   emacs_value space = env->make_string(env, " ", 1);
    289   for (int i = 0; i < beyond_eol; i += 1)
    290     insert(env, space);
    291 }
    292 
    293 static void refresh_lines(Term *term, emacs_env *env, int start_row,
    294                           int end_row, int end_col) {
    295   if (end_row < start_row) {
    296     return;
    297   }
    298   int i, j;
    299 
    300 #define PUSH_BUFFER(c)                                                         \
    301   do {                                                                         \
    302     if (length == capacity) {                                                  \
    303       capacity += end_col * 4;                                                 \
    304       buffer = realloc(buffer, capacity * sizeof(char));                       \
    305     }                                                                          \
    306     buffer[length] = (c);                                                      \
    307     length++;                                                                  \
    308   } while (0)
    309 
    310   int capacity = ((end_row - start_row + 1) * end_col) * 4;
    311   int length = 0;
    312   char *buffer = malloc(capacity * sizeof(char));
    313   VTermScreenCell cell;
    314   VTermScreenCell lastCell;
    315   fetch_cell(term, start_row, 0, &lastCell);
    316 
    317   for (i = start_row; i < end_row; i++) {
    318 
    319     int newline = 0;
    320     int isprompt = 0;
    321     for (j = 0; j < end_col; j++) {
    322       fetch_cell(term, i, j, &cell);
    323       if (isprompt && length > 0) {
    324         emacs_value text = render_text(env, term, buffer, length, &lastCell);
    325         insert(env, render_prompt(env, text));
    326         length = 0;
    327       }
    328 
    329       isprompt = is_end_of_prompt(term, end_col, i, j);
    330       if (isprompt && length > 0) {
    331         insert(env, render_text(env, term, buffer, length, &lastCell));
    332         length = 0;
    333       }
    334 
    335       if (!compare_cells(&cell, &lastCell)) {
    336         emacs_value text = render_text(env, term, buffer, length, &lastCell);
    337         insert(env, text);
    338         length = 0;
    339       }
    340 
    341       lastCell = cell;
    342       if (cell.chars[0] == 0) {
    343         if (is_eol(term, end_col, i, j)) {
    344           /* This cell is EOL if this and every cell to the right is black */
    345           PUSH_BUFFER('\n');
    346           newline = 1;
    347           break;
    348         }
    349         PUSH_BUFFER(' ');
    350       } else {
    351         for (int k = 0; k < VTERM_MAX_CHARS_PER_CELL && cell.chars[k]; ++k) {
    352           unsigned char bytes[4];
    353           size_t count = codepoint_to_utf8(cell.chars[k], bytes);
    354           for (int l = 0; l < count; l++) {
    355             PUSH_BUFFER(bytes[l]);
    356           }
    357         }
    358       }
    359 
    360       if (cell.width > 1) {
    361         int w = cell.width - 1;
    362         j = j + w;
    363       }
    364     }
    365     if (isprompt && length > 0) {
    366       emacs_value text = render_text(env, term, buffer, length, &lastCell);
    367       insert(env, render_prompt(env, text));
    368       length = 0;
    369       isprompt = 0;
    370     }
    371 
    372     if (!newline) {
    373       emacs_value text = render_text(env, term, buffer, length, &lastCell);
    374       insert(env, text);
    375       length = 0;
    376       text = render_fake_newline(env, term);
    377       insert(env, text);
    378     }
    379   }
    380   emacs_value text = render_text(env, term, buffer, length, &lastCell);
    381   insert(env, text);
    382 
    383 #undef PUSH_BUFFER
    384   free(buffer);
    385 
    386   return;
    387 }
    388 // Refresh the screen (visible part of the buffer when the terminal is
    389 // focused) of a invalidated terminal
    390 static void refresh_screen(Term *term, emacs_env *env) {
    391   // Term height may have decreased before `invalid_end` reflects it.
    392   term->invalid_end = MIN(term->invalid_end, term->height);
    393 
    394   if (term->invalid_end >= term->invalid_start) {
    395     int startrow = -(term->height - term->invalid_start - term->linenum_added);
    396     /* startrow is negative,so we backward  -startrow lines from end of buffer
    397        then delete lines there.
    398      */
    399     goto_line(env, startrow);
    400     delete_lines(env, startrow, term->invalid_end - term->invalid_start, true);
    401     refresh_lines(term, env, term->invalid_start, term->invalid_end,
    402                   term->width);
    403 
    404     /* term->linenum_added is lines added  by window height increased */
    405     term->linenum += term->linenum_added;
    406     term->linenum_added = 0;
    407   }
    408 
    409   term->invalid_start = INT_MAX;
    410   term->invalid_end = -1;
    411 }
    412 
    413 static int term_resize(int rows, int cols, void *user_data) {
    414   /* can not use invalidate_terminal here */
    415   /* when the window height decreased, */
    416   /*  the value of term->invalid_end can't bigger than window height */
    417   Term *term = (Term *)user_data;
    418   term->invalid_start = 0;
    419   term->invalid_end = rows;
    420 
    421   /* if rows=term->lines_len, that means term_sb_pop already resize term->lines
    422    */
    423   /* if rows<term->lines_len, term_sb_push would resize term->lines there */
    424   /* we noly need to take care of rows>term->height */
    425 
    426   if (rows > term->height) {
    427     if (rows > term->lines_len) {
    428       LineInfo **infos = term->lines;
    429       term->lines = malloc(sizeof(LineInfo *) * rows);
    430       memmove(term->lines, infos, sizeof(infos[0]) * term->lines_len);
    431 
    432       LineInfo *lastline = term->lines[term->lines_len - 1];
    433       for (int i = term->lines_len; i < rows; i++) {
    434         if (lastline != NULL) {
    435           LineInfo *line = alloc_lineinfo();
    436           if (lastline->directory != NULL) {
    437             line->directory =
    438                 malloc(1 + strlen(term->lines[term->lines_len - 1]->directory));
    439             strcpy(line->directory,
    440                    term->lines[term->lines_len - 1]->directory);
    441           }
    442           term->lines[i] = line;
    443         } else {
    444           term->lines[i] = NULL;
    445         }
    446       }
    447       term->lines_len = rows;
    448       free(infos);
    449     }
    450   }
    451 
    452   term->width = cols;
    453   term->height = rows;
    454 
    455   invalidate_terminal(term, -1, -1);
    456   term->resizing = false;
    457 
    458   return 1;
    459 }
    460 
    461 // Refresh the scrollback of an invalidated terminal.
    462 static void refresh_scrollback(Term *term, emacs_env *env) {
    463   int max_line_count = (int)term->sb_current + term->height;
    464   int del_cnt = 0;
    465   if (term->sb_clear_pending) {
    466     del_cnt = term->linenum - term->height;
    467     if (del_cnt > 0) {
    468       delete_lines(env, 1, del_cnt, true);
    469       term->linenum -= del_cnt;
    470     }
    471     term->sb_clear_pending = false;
    472   }
    473   if (term->sb_pending > 0) {
    474     // This means that either the window height has decreased or the screen
    475     // became full and libvterm had to push all rows up. Convert the first
    476     // pending scrollback row into a string and append it just above the visible
    477     // section of the buffer
    478 
    479     del_cnt = term->linenum - term->height - (int)term->sb_size +
    480               term->sb_pending - term->sb_pending_by_height_decr;
    481     if (del_cnt > 0) {
    482       delete_lines(env, 1, del_cnt, true);
    483       term->linenum -= del_cnt;
    484     }
    485 
    486     term->linenum += term->sb_pending;
    487     del_cnt = term->linenum - max_line_count; /* extra lines at the bottom */
    488     /* buf_index is negative,so we move to end of buffer,then backward
    489        -buf_index lines. goto lines backward is effectively when
    490        vterm-max-scrollback is a large number.
    491      */
    492     int buf_index = -(term->height + del_cnt);
    493     goto_line(env, buf_index);
    494     refresh_lines(term, env, -term->sb_pending, 0, term->width);
    495 
    496     term->sb_pending = 0;
    497   }
    498 
    499   // Remove extra lines at the bottom
    500   del_cnt = term->linenum - max_line_count;
    501   if (del_cnt > 0) {
    502     term->linenum -= del_cnt;
    503     /* -del_cnt is negative,so we delete_lines from end of buffer.
    504        this line means: delete del_cnt count of lines at end of buffer.
    505      */
    506     delete_lines(env, -del_cnt, del_cnt, true);
    507   }
    508 
    509   term->sb_pending_by_height_decr = 0;
    510   term->height_resize = 0;
    511 }
    512 
    513 static void adjust_topline(Term *term, emacs_env *env) {
    514   VTermState *state = vterm_obtain_state(term->vt);
    515   VTermPos pos;
    516   vterm_state_get_cursorpos(state, &pos);
    517 
    518   /* pos.row-term->height is negative,so we backward term->height-pos.row
    519    * lines from end of buffer
    520    */
    521 
    522   goto_line(env, pos.row - term->height);
    523   goto_col(term, env, pos.row, pos.col);
    524 
    525   emacs_value windows = get_buffer_window_list(env);
    526   emacs_value swindow = selected_window(env);
    527   int winnum = env->extract_integer(env, length(env, windows));
    528   for (int i = 0; i < winnum; i++) {
    529     emacs_value window = nth(env, i, windows);
    530     if (eq(env, window, swindow)) {
    531       int win_body_height =
    532           env->extract_integer(env, window_body_height(env, window));
    533 
    534       /* recenter:If ARG is negative, it counts up from the bottom of the
    535        * window.  (ARG should be less than the height of the window ) */
    536       if (term->height - pos.row <= win_body_height) {
    537         recenter(env, env->make_integer(env, pos.row - term->height));
    538       } else {
    539         recenter(env, env->make_integer(env, pos.row));
    540       }
    541     } else {
    542       if (env->is_not_nil(env, window)) {
    543         set_window_point(env, window, point(env));
    544       }
    545     }
    546   }
    547 }
    548 
    549 static void invalidate_terminal(Term *term, int start_row, int end_row) {
    550   if (start_row != -1 && end_row != -1) {
    551     term->invalid_start = MIN(term->invalid_start, start_row);
    552     term->invalid_end = MAX(term->invalid_end, end_row);
    553   }
    554   term->is_invalidated = true;
    555 }
    556 
    557 static int term_damage(VTermRect rect, void *data) {
    558   invalidate_terminal(data, rect.start_row, rect.end_row);
    559   return 1;
    560 }
    561 
    562 static int term_moverect(VTermRect dest, VTermRect src, void *data) {
    563   invalidate_terminal(data, MIN(dest.start_row, src.start_row),
    564                       MAX(dest.end_row, src.end_row));
    565   return 1;
    566 }
    567 
    568 static int term_movecursor(VTermPos new, VTermPos old, int visible,
    569                            void *data) {
    570   Term *term = data;
    571   term->cursor.row = new.row;
    572   term->cursor.col = new.col;
    573   invalidate_terminal(term, old.row, old.row + 1);
    574   invalidate_terminal(term, new.row, new.row + 1);
    575 
    576   return 1;
    577 }
    578 
    579 static void term_redraw_cursor(Term *term, emacs_env *env) {
    580   if (term->cursor.cursor_blink_changed) {
    581     term->cursor.cursor_blink_changed = false;
    582     set_cursor_blink(env, term->cursor.cursor_blink);
    583   }
    584 
    585   if (term->cursor.cursor_type_changed) {
    586     term->cursor.cursor_type_changed = false;
    587 
    588     if (!term->cursor.cursor_visible) {
    589       set_cursor_type(env, Qnil);
    590       return;
    591     }
    592 
    593     switch (term->cursor.cursor_type) {
    594     case VTERM_PROP_CURSORSHAPE_BLOCK:
    595       set_cursor_type(env, Qbox);
    596       break;
    597     case VTERM_PROP_CURSORSHAPE_UNDERLINE:
    598       set_cursor_type(env, Qhbar);
    599       break;
    600     case VTERM_PROP_CURSORSHAPE_BAR_LEFT:
    601       set_cursor_type(env, Qbar);
    602       break;
    603     default:
    604       set_cursor_type(env, Qt);
    605       break;
    606     }
    607   }
    608 }
    609 
    610 static void term_redraw(Term *term, emacs_env *env) {
    611   term_redraw_cursor(term, env);
    612 
    613   if (term->is_invalidated) {
    614     int oldlinenum = term->linenum;
    615     refresh_scrollback(term, env);
    616     refresh_screen(term, env);
    617     term->linenum_added = term->linenum - oldlinenum;
    618     adjust_topline(term, env);
    619     term->linenum_added = 0;
    620   }
    621 
    622   if (term->title_changed) {
    623     set_title(env, env->make_string(env, term->title, strlen(term->title)));
    624     term->title_changed = false;
    625   }
    626 
    627   if (term->directory_changed) {
    628     set_directory(
    629         env, env->make_string(env, term->directory, strlen(term->directory)));
    630     term->directory_changed = false;
    631   }
    632 
    633   while (term->elisp_code_first) {
    634     ElispCodeListNode *node = term->elisp_code_first;
    635     term->elisp_code_first = node->next;
    636     emacs_value elisp_code = env->make_string(env, node->code, node->code_len);
    637     vterm_eval(env, elisp_code);
    638 
    639     free(node->code);
    640     free(node);
    641   }
    642   term->elisp_code_p_insert = &term->elisp_code_first;
    643 
    644   if (term->selection_data) {
    645     emacs_value selection_mask = env->make_integer(env, term->selection_mask);
    646     emacs_value selection_data = env->make_string(env, term->selection_data,
    647                                                   strlen(term->selection_data));
    648     vterm_set_selection(env, selection_mask, selection_data);
    649     free(term->selection_data);
    650     term->selection_data = NULL;
    651     term->selection_mask = 0;
    652   }
    653 
    654   term->is_invalidated = false;
    655 }
    656 
    657 static VTermScreenCallbacks vterm_screen_callbacks = {
    658     .damage = term_damage,
    659     .moverect = term_moverect,
    660     .movecursor = term_movecursor,
    661     .settermprop = term_settermprop,
    662     .resize = term_resize,
    663     .sb_pushline = term_sb_push,
    664     .sb_popline = term_sb_pop,
    665 #if !defined(VTermSBClearNotExists)
    666     .sb_clear = term_sb_clear,
    667 #endif
    668 };
    669 
    670 static bool compare_cells(VTermScreenCell *a, VTermScreenCell *b) {
    671   bool equal = true;
    672   equal = equal && vterm_color_is_equal(&a->fg, &b->fg);
    673   equal = equal && vterm_color_is_equal(&a->bg, &b->bg);
    674   equal = equal && (a->attrs.bold == b->attrs.bold);
    675   equal = equal && (a->attrs.underline == b->attrs.underline);
    676   equal = equal && (a->attrs.italic == b->attrs.italic);
    677   equal = equal && (a->attrs.reverse == b->attrs.reverse);
    678   equal = equal && (a->attrs.strike == b->attrs.strike);
    679   return equal;
    680 }
    681 
    682 static bool is_key(unsigned char *key, size_t len, char *key_description) {
    683   return (len == strlen(key_description) &&
    684           memcmp(key, key_description, len) == 0);
    685 }
    686 
    687 /* str1=concat(str1,str2,str2_len,true); */
    688 /* str1 can be NULL */
    689 static char *concat(char *str1, const char *str2, size_t str2_len,
    690                     bool free_str1) {
    691   if (str1 == NULL) {
    692     str1 = malloc(str2_len + 1);
    693     memcpy(str1, str2, str2_len);
    694     str1[str2_len] = '\0';
    695     return str1;
    696   }
    697   size_t str1_len = strlen(str1);
    698   char *buf = malloc(str1_len + str2_len + 1);
    699   memcpy(buf, str1, str1_len);
    700   memcpy(&buf[str1_len], str2, str2_len);
    701   buf[str1_len + str2_len] = '\0';
    702   if (free_str1) {
    703     free(str1);
    704   }
    705   return buf;
    706 }
    707 static void term_set_title(Term *term, const char *title, size_t len,
    708                            bool initial, bool final) {
    709   if (term->title && initial) {
    710     free(term->title);
    711     term->title = NULL;
    712     term->title_changed = false;
    713   }
    714   term->title = concat(term->title, title, len, true);
    715   if (final) {
    716     term->title_changed = true;
    717   }
    718   return;
    719 }
    720 
    721 static int term_settermprop(VTermProp prop, VTermValue *val, void *user_data) {
    722   Term *term = (Term *)user_data;
    723   switch (prop) {
    724   case VTERM_PROP_CURSORVISIBLE:
    725     invalidate_terminal(term, term->cursor.row, term->cursor.row + 1);
    726     term->cursor.cursor_visible = val->boolean;
    727     term->cursor.cursor_type_changed = true;
    728     break;
    729   case VTERM_PROP_CURSORBLINK:
    730     if (term->ignore_blink_cursor)
    731       break;
    732     invalidate_terminal(term, term->cursor.row, term->cursor.row + 1);
    733     term->cursor.cursor_blink = val->boolean;
    734     term->cursor.cursor_blink_changed = true;
    735     break;
    736   case VTERM_PROP_CURSORSHAPE:
    737     invalidate_terminal(term, term->cursor.row, term->cursor.row + 1);
    738     term->cursor.cursor_type = val->number;
    739     term->cursor.cursor_type_changed = true;
    740     break;
    741   case VTERM_PROP_TITLE:
    742 #ifdef VTermStringFragmentNotExists
    743     term_set_title(term, val->string, strlen(val->string), true, true);
    744 #else
    745     term_set_title(term, val->string.str, val->string.len, val->string.initial,
    746                    val->string.final);
    747 #endif
    748     break;
    749   case VTERM_PROP_ALTSCREEN:
    750     invalidate_terminal(term, 0, term->height);
    751     break;
    752   default:
    753     return 0;
    754   }
    755 
    756   return 1;
    757 }
    758 
    759 static emacs_value render_text(emacs_env *env, Term *term, char *buffer,
    760                                int len, VTermScreenCell *cell) {
    761   emacs_value text;
    762   if (len == 0) {
    763     text = env->make_string(env, "", 0);
    764     return text;
    765   } else {
    766     text = env->make_string(env, buffer, len);
    767   }
    768 
    769   emacs_value fg = cell_rgb_color(env, term, cell, true);
    770   emacs_value bg = cell_rgb_color(env, term, cell, false);
    771   /* With vterm-disable-bold-font, vterm-disable-underline,
    772    * vterm-disable-inverse-video, users can disable some text properties.
    773    * Here, we check whether the text would require adding such properties.
    774    * In case it does, and the user does not disable the attribute, we later
    775    * append the property to the list props.  If the text does not require
    776    * such property, or the user disable it, we set the variable to nil.
    777    * Properties that are marked as nil are not added to the text. */
    778   emacs_value bold =
    779       cell->attrs.bold && !term->disable_bold_font ? Qbold : Qnil;
    780   emacs_value underline =
    781       cell->attrs.underline && !term->disable_underline ? Qt : Qnil;
    782   emacs_value italic = cell->attrs.italic ? Qitalic : Qnil;
    783   emacs_value reverse =
    784       cell->attrs.reverse && !term->disable_inverse_video ? Qt : Qnil;
    785   emacs_value strike = cell->attrs.strike ? Qt : Qnil;
    786 
    787   // TODO: Blink, font, dwl, dhl is missing
    788   int emacs_major_version =
    789       env->extract_integer(env, symbol_value(env, Qemacs_major_version));
    790   emacs_value properties;
    791   emacs_value props[64];
    792   int props_len = 0;
    793   if (env->is_not_nil(env, fg))
    794     props[props_len++] = Qforeground, props[props_len++] = fg;
    795   if (env->is_not_nil(env, bg))
    796     props[props_len++] = Qbackground, props[props_len++] = bg;
    797   if (bold != Qnil)
    798     props[props_len++] = Qweight, props[props_len++] = bold;
    799   if (underline != Qnil)
    800     props[props_len++] = Qunderline, props[props_len++] = underline;
    801   if (italic != Qnil)
    802     props[props_len++] = Qslant, props[props_len++] = italic;
    803   if (reverse != Qnil)
    804     props[props_len++] = Qreverse, props[props_len++] = reverse;
    805   if (strike != Qnil)
    806     props[props_len++] = Qstrike, props[props_len++] = strike;
    807   if (emacs_major_version >= 27)
    808     props[props_len++] = Qextend, props[props_len++] = Qt;
    809 
    810   properties = list(env, props, props_len);
    811 
    812   if (props_len)
    813     put_text_property(env, text, Qface, properties);
    814 
    815   return text;
    816 }
    817 static emacs_value render_prompt(emacs_env *env, emacs_value text) {
    818 
    819   emacs_value properties;
    820 
    821   properties =
    822       list(env, (emacs_value[]){Qvterm_prompt, Qt, Qrear_nonsticky, Qt}, 4);
    823 
    824   add_text_properties(env, text, properties);
    825 
    826   return text;
    827 }
    828 
    829 static emacs_value render_fake_newline(emacs_env *env, Term *term) {
    830 
    831   emacs_value text;
    832   text = env->make_string(env, "\n", 1);
    833 
    834   emacs_value properties;
    835 
    836   properties =
    837       list(env, (emacs_value[]){Qvterm_line_wrap, Qt, Qrear_nonsticky, Qt}, 4);
    838 
    839   add_text_properties(env, text, properties);
    840 
    841   return text;
    842 }
    843 
    844 static emacs_value cell_rgb_color(emacs_env *env, Term *term,
    845                                   VTermScreenCell *cell, bool is_foreground) {
    846   VTermColor *color = is_foreground ? &cell->fg : &cell->bg;
    847 
    848   /** NOTE: -10 is used as index offset for special indexes,
    849    * see C-h f vterm--get-color RET
    850    */
    851   if (VTERM_COLOR_IS_DEFAULT_FG(color)) {
    852     return vterm_get_color(env, -1 + (cell->attrs.underline ? -10 : 0));
    853   }
    854   if (VTERM_COLOR_IS_DEFAULT_BG(color)) {
    855     return vterm_get_color(env, -2 + (cell->attrs.reverse ? -10 : 0));
    856   }
    857   if (VTERM_COLOR_IS_INDEXED(color)) {
    858     if (color->indexed.idx < 16) {
    859       return vterm_get_color(env, color->indexed.idx);
    860     } else {
    861       VTermState *state = vterm_obtain_state(term->vt);
    862       vterm_state_get_palette_color(state, color->indexed.idx, color);
    863     }
    864   } else if (VTERM_COLOR_IS_RGB(color)) {
    865     /* do nothing just use the argument color directly */
    866   }
    867 
    868   char buffer[8];
    869   snprintf(buffer, 8, "#%02X%02X%02X", color->rgb.red, color->rgb.green,
    870            color->rgb.blue);
    871   return env->make_string(env, buffer, 7);
    872 }
    873 
    874 static void term_flush_output(Term *term, emacs_env *env) {
    875   size_t len = vterm_output_get_buffer_current(term->vt);
    876   if (len) {
    877     char buffer[len];
    878     len = vterm_output_read(term->vt, buffer, len);
    879 
    880     emacs_value output = env->make_string(env, buffer, len);
    881     env->funcall(env, Fvterm_flush_output, 1, (emacs_value[]){output});
    882   }
    883 }
    884 
    885 static void term_clear_scrollback(Term *term, emacs_env *env) {
    886   term_sb_clear(term);
    887   vterm_screen_flush_damage(term->vts);
    888   term_redraw(term, env);
    889 }
    890 
    891 static void term_process_key(Term *term, emacs_env *env, unsigned char *key,
    892                              size_t len, VTermModifier modifier) {
    893   if (is_key(key, len, "<clear_scrollback>")) {
    894     term_clear_scrollback(term, env);
    895   } else if (is_key(key, len, "<start>")) {
    896     tcflow(term->pty_fd, TCOON);
    897   } else if (is_key(key, len, "<stop>")) {
    898     tcflow(term->pty_fd, TCOOFF);
    899   } else if (is_key(key, len, "<start_paste>")) {
    900     vterm_keyboard_start_paste(term->vt);
    901   } else if (is_key(key, len, "<end_paste>")) {
    902     vterm_keyboard_end_paste(term->vt);
    903   } else if (is_key(key, len, "<tab>")) {
    904     vterm_keyboard_key(term->vt, VTERM_KEY_TAB, modifier);
    905   } else if (is_key(key, len, "<backtab>") ||
    906              is_key(key, len, "<iso-lefttab>")) {
    907     vterm_keyboard_key(term->vt, VTERM_KEY_TAB, VTERM_MOD_SHIFT);
    908   } else if (is_key(key, len, "<backspace>")) {
    909     vterm_keyboard_key(term->vt, VTERM_KEY_BACKSPACE, modifier);
    910   } else if (is_key(key, len, "<escape>")) {
    911     vterm_keyboard_key(term->vt, VTERM_KEY_ESCAPE, modifier);
    912   } else if (is_key(key, len, "<up>")) {
    913     vterm_keyboard_key(term->vt, VTERM_KEY_UP, modifier);
    914   } else if (is_key(key, len, "<down>")) {
    915     vterm_keyboard_key(term->vt, VTERM_KEY_DOWN, modifier);
    916   } else if (is_key(key, len, "<left>")) {
    917     vterm_keyboard_key(term->vt, VTERM_KEY_LEFT, modifier);
    918   } else if (is_key(key, len, "<right>")) {
    919     vterm_keyboard_key(term->vt, VTERM_KEY_RIGHT, modifier);
    920   } else if (is_key(key, len, "<insert>")) {
    921     vterm_keyboard_key(term->vt, VTERM_KEY_INS, modifier);
    922   } else if (is_key(key, len, "<delete>")) {
    923     vterm_keyboard_key(term->vt, VTERM_KEY_DEL, modifier);
    924   } else if (is_key(key, len, "<home>")) {
    925     vterm_keyboard_key(term->vt, VTERM_KEY_HOME, modifier);
    926   } else if (is_key(key, len, "<end>")) {
    927     vterm_keyboard_key(term->vt, VTERM_KEY_END, modifier);
    928   } else if (is_key(key, len, "<prior>")) {
    929     vterm_keyboard_key(term->vt, VTERM_KEY_PAGEUP, modifier);
    930   } else if (is_key(key, len, "<next>")) {
    931     vterm_keyboard_key(term->vt, VTERM_KEY_PAGEDOWN, modifier);
    932   } else if (is_key(key, len, "<f0>")) {
    933     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(0), modifier);
    934   } else if (is_key(key, len, "<f1>")) {
    935     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(1), modifier);
    936   } else if (is_key(key, len, "<f2>")) {
    937     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(2), modifier);
    938   } else if (is_key(key, len, "<f3>")) {
    939     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(3), modifier);
    940   } else if (is_key(key, len, "<f4>")) {
    941     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(4), modifier);
    942   } else if (is_key(key, len, "<f5>")) {
    943     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(5), modifier);
    944   } else if (is_key(key, len, "<f6>")) {
    945     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(6), modifier);
    946   } else if (is_key(key, len, "<f7>")) {
    947     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(7), modifier);
    948   } else if (is_key(key, len, "<f8>")) {
    949     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(8), modifier);
    950   } else if (is_key(key, len, "<f9>")) {
    951     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(9), modifier);
    952   } else if (is_key(key, len, "<f10>")) {
    953     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(10), modifier);
    954   } else if (is_key(key, len, "<f11>")) {
    955     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(11), modifier);
    956   } else if (is_key(key, len, "<f12>")) {
    957     vterm_keyboard_key(term->vt, VTERM_KEY_FUNCTION(12), modifier);
    958   } else if (is_key(key, len, "<kp-0>")) {
    959     vterm_keyboard_key(term->vt, VTERM_KEY_KP_0, modifier);
    960   } else if (is_key(key, len, "<kp-1>")) {
    961     vterm_keyboard_key(term->vt, VTERM_KEY_KP_1, modifier);
    962   } else if (is_key(key, len, "<kp-2>")) {
    963     vterm_keyboard_key(term->vt, VTERM_KEY_KP_2, modifier);
    964   } else if (is_key(key, len, "<kp-3>")) {
    965     vterm_keyboard_key(term->vt, VTERM_KEY_KP_3, modifier);
    966   } else if (is_key(key, len, "<kp-4>")) {
    967     vterm_keyboard_key(term->vt, VTERM_KEY_KP_4, modifier);
    968   } else if (is_key(key, len, "<kp-5>")) {
    969     vterm_keyboard_key(term->vt, VTERM_KEY_KP_5, modifier);
    970   } else if (is_key(key, len, "<kp-6>")) {
    971     vterm_keyboard_key(term->vt, VTERM_KEY_KP_6, modifier);
    972   } else if (is_key(key, len, "<kp-7>")) {
    973     vterm_keyboard_key(term->vt, VTERM_KEY_KP_7, modifier);
    974   } else if (is_key(key, len, "<kp-8>")) {
    975     vterm_keyboard_key(term->vt, VTERM_KEY_KP_8, modifier);
    976   } else if (is_key(key, len, "<kp-9>")) {
    977     vterm_keyboard_key(term->vt, VTERM_KEY_KP_9, modifier);
    978   } else if (is_key(key, len, "<kp-add>")) {
    979     vterm_keyboard_key(term->vt, VTERM_KEY_KP_PLUS, modifier);
    980   } else if (is_key(key, len, "<kp-subtract>")) {
    981     vterm_keyboard_key(term->vt, VTERM_KEY_KP_MINUS, modifier);
    982   } else if (is_key(key, len, "<kp-multiply>")) {
    983     vterm_keyboard_key(term->vt, VTERM_KEY_KP_MULT, modifier);
    984   } else if (is_key(key, len, "<kp-divide>")) {
    985     vterm_keyboard_key(term->vt, VTERM_KEY_KP_DIVIDE, modifier);
    986   } else if (is_key(key, len, "<kp-equal>")) {
    987     vterm_keyboard_key(term->vt, VTERM_KEY_KP_EQUAL, modifier);
    988   } else if (is_key(key, len, "<kp-decimal>")) {
    989     vterm_keyboard_key(term->vt, VTERM_KEY_KP_PERIOD, modifier);
    990   } else if (is_key(key, len, "<kp-separator>")) {
    991     vterm_keyboard_key(term->vt, VTERM_KEY_KP_COMMA, modifier);
    992   } else if (is_key(key, len, "<kp-enter>")) {
    993     vterm_keyboard_key(term->vt, VTERM_KEY_KP_ENTER, modifier);
    994   } else if (is_key(key, len, "j") && (modifier == VTERM_MOD_CTRL)) {
    995     vterm_keyboard_unichar(term->vt, '\n', 0);
    996   } else if (is_key(key, len, "SPC")) {
    997     vterm_keyboard_unichar(term->vt, ' ', modifier);
    998   } else if (len <= 4) {
    999     uint32_t codepoint;
   1000     if (utf8_to_codepoint(key, len, &codepoint)) {
   1001       vterm_keyboard_unichar(term->vt, codepoint, modifier);
   1002     }
   1003   }
   1004 }
   1005 
   1006 void term_finalize(void *object) {
   1007   Term *term = (Term *)object;
   1008   for (int i = 0; i < term->sb_current; i++) {
   1009     if (term->sb_buffer[i]->info != NULL) {
   1010       free_lineinfo(term->sb_buffer[i]->info);
   1011       term->sb_buffer[i]->info = NULL;
   1012     }
   1013     free(term->sb_buffer[i]);
   1014   }
   1015   if (term->title) {
   1016     free(term->title);
   1017     term->title = NULL;
   1018   }
   1019 
   1020   if (term->directory) {
   1021     free(term->directory);
   1022     term->directory = NULL;
   1023   }
   1024 
   1025   while (term->elisp_code_first) {
   1026     ElispCodeListNode *node = term->elisp_code_first;
   1027     term->elisp_code_first = node->next;
   1028     free(node->code);
   1029     free(node);
   1030   }
   1031   term->elisp_code_p_insert = &term->elisp_code_first;
   1032 
   1033   if (term->cmd_buffer) {
   1034     free(term->cmd_buffer);
   1035     term->cmd_buffer = NULL;
   1036   }
   1037   if (term->selection_data) {
   1038     free(term->selection_data);
   1039     term->selection_data = NULL;
   1040   }
   1041 
   1042   for (int i = 0; i < term->lines_len; i++) {
   1043     if (term->lines[i] != NULL) {
   1044       free_lineinfo(term->lines[i]);
   1045       term->lines[i] = NULL;
   1046     }
   1047   }
   1048 
   1049   if (term->pty_fd > 0) {
   1050     close(term->pty_fd);
   1051   }
   1052 
   1053   free(term->sb_buffer);
   1054   free(term->lines);
   1055   vterm_free(term->vt);
   1056   free(term);
   1057 }
   1058 
   1059 static int handle_osc_cmd_51(Term *term, char subCmd, char *buffer) {
   1060   if (subCmd == 'A') {
   1061     /* "51;A" sets the current directory */
   1062     /* "51;A" has also the role of identifying the end of the prompt */
   1063     if (term->directory != NULL) {
   1064       free(term->directory);
   1065       term->directory = NULL;
   1066     }
   1067     term->directory = malloc(strlen(buffer) + 1);
   1068     strcpy(term->directory, buffer);
   1069     term->directory_changed = true;
   1070 
   1071     for (int i = term->cursor.row; i < term->lines_len; i++) {
   1072       if (term->lines[i] == NULL) {
   1073         term->lines[i] = alloc_lineinfo();
   1074       }
   1075 
   1076       if (term->lines[i]->directory != NULL) {
   1077         free(term->lines[i]->directory);
   1078       }
   1079       term->lines[i]->directory = malloc(strlen(buffer) + 1);
   1080       strcpy(term->lines[i]->directory, buffer);
   1081       if (i == term->cursor.row) {
   1082         term->lines[i]->prompt_col = term->cursor.col;
   1083       } else {
   1084         term->lines[i]->prompt_col = -1;
   1085       }
   1086     }
   1087     return 1;
   1088   } else if (subCmd == 'E') {
   1089     /* "51;E" executes elisp code */
   1090     /* The elisp code is executed in term_redraw */
   1091     ElispCodeListNode *node = malloc(sizeof(ElispCodeListNode));
   1092     node->code_len = strlen(buffer);
   1093     node->code = malloc(node->code_len + 1);
   1094     strcpy(node->code, buffer);
   1095     node->next = NULL;
   1096 
   1097     *(term->elisp_code_p_insert) = node;
   1098     term->elisp_code_p_insert = &(node->next);
   1099     return 1;
   1100   }
   1101   return 0;
   1102 }
   1103 
   1104 static int handle_osc_cmd(Term *term, int cmd, char *buffer) {
   1105   if (cmd == 51) {
   1106     char subCmd = '0';
   1107     if (strlen(buffer) == 0) {
   1108       return 0;
   1109     }
   1110     subCmd = buffer[0];
   1111     /* ++ skip the subcmd char */
   1112     return handle_osc_cmd_51(term, subCmd, ++buffer);
   1113   }
   1114   return 0;
   1115 }
   1116 /* maybe we should drop support of libvterm < v0.2 */
   1117 /* VTermStringFragmentNotExists was introduced when libvterm is not released */
   1118 #ifdef VTermStringFragmentNotExists
   1119 static int osc_callback(const char *command, size_t cmdlen, void *user) {
   1120   Term *term = (Term *)user;
   1121   char buffer[cmdlen + 1];
   1122   buffer[cmdlen] = '\0';
   1123   memcpy(buffer, command, cmdlen);
   1124 
   1125   if (cmdlen > 4 && buffer[0] == '5' && buffer[1] == '1' && buffer[2] == ';' &&
   1126       buffer[3] == 'A') {
   1127     return handle_osc_cmd_51(term, 'A', &buffer[4]);
   1128   } else if (cmdlen > 4 && buffer[0] == '5' && buffer[1] == '1' &&
   1129              buffer[2] == ';' && buffer[3] == 'E') {
   1130     return handle_osc_cmd_51(term, 'E', &buffer[4]);
   1131   }
   1132   return 0;
   1133 }
   1134 static VTermParserCallbacks parser_callbacks = {
   1135     .text = NULL,
   1136     .control = NULL,
   1137     .escape = NULL,
   1138     .csi = NULL,
   1139     .osc = &osc_callback,
   1140     .dcs = NULL,
   1141 };
   1142 #else
   1143 
   1144 static int osc_callback(int cmd, VTermStringFragment frag, void *user) {
   1145   /* osc_callback (OSC = Operating System Command) */
   1146 
   1147   /* We interpret escape codes that start with "51;" */
   1148   /* "51;A" sets the current directory */
   1149   /* "51;A" has also the role of identifying the end of the prompt */
   1150   /* "51;E" executes elisp code */
   1151   /* The elisp code is executed in term_redraw */
   1152   Term *term = (Term *)user;
   1153 
   1154   if (frag.initial) {
   1155     /* drop old fragment,because this is a initial fragment */
   1156     if (term->cmd_buffer) {
   1157       free(term->cmd_buffer);
   1158       term->cmd_buffer = NULL;
   1159     }
   1160   }
   1161 
   1162   if (!frag.initial && !frag.final && frag.len == 0) {
   1163     return 0;
   1164   }
   1165 
   1166   term->cmd_buffer = concat(term->cmd_buffer, frag.str, frag.len, true);
   1167   if (frag.final) {
   1168     handle_osc_cmd(term, cmd, term->cmd_buffer);
   1169     free(term->cmd_buffer);
   1170     term->cmd_buffer = NULL;
   1171   }
   1172   return 0;
   1173 }
   1174 static VTermStateFallbacks parser_callbacks = {
   1175     .control = NULL,
   1176     .csi = NULL,
   1177     .osc = &osc_callback,
   1178     .dcs = NULL,
   1179 };
   1180 #ifndef VTermSelectionMaskNotExists
   1181 static int set_selection(VTermSelectionMask mask, VTermStringFragment frag,
   1182                          void *user) {
   1183   Term *term = (Term *)user;
   1184 
   1185   if (frag.initial) {
   1186     term->selection_mask = mask;
   1187     if (term->selection_data) {
   1188       free(term->selection_data);
   1189     }
   1190     term->selection_data = NULL;
   1191   }
   1192 
   1193   if (frag.len) {
   1194     term->selection_data =
   1195         concat(term->selection_data, frag.str, frag.len, true);
   1196   }
   1197   return 1;
   1198 }
   1199 /* OSC 52 ; Pc ; Pd BEL */
   1200 /* Manipulate Selection Data */
   1201 /* https://invisible-island.net/xterm/ctlseqs/ctlseqs.html */
   1202 /* test by printf "\033]52;c;$(printf "%s" "blabla" | base64)\a" */
   1203 /* c , p , q , s , 0 , 1 , 2 , 3 , 4 , 5 , 6 , and 7 */
   1204 /* for clipboard, primary, secondary, select, or cut buffers 0 through 7 */
   1205 /* respectively */
   1206 static VTermSelectionCallbacks selection_callbacks = {
   1207     .set = &set_selection,
   1208     .query = NULL,
   1209 };
   1210 #endif /* VTermSelectionMaskNotExists */
   1211 
   1212 #endif
   1213 
   1214 emacs_value Fvterm_new(emacs_env *env, ptrdiff_t nargs, emacs_value args[],
   1215                        void *data) {
   1216   Term *term = malloc(sizeof(Term));
   1217 
   1218   int rows = env->extract_integer(env, args[0]);
   1219   int cols = env->extract_integer(env, args[1]);
   1220   int sb_size = env->extract_integer(env, args[2]);
   1221   int disable_bold_font = env->is_not_nil(env, args[3]);
   1222   int disable_underline = env->is_not_nil(env, args[4]);
   1223   int disable_inverse_video = env->is_not_nil(env, args[5]);
   1224   int ignore_blink_cursor = env->is_not_nil(env, args[6]);
   1225   int set_bold_hightbright = env->is_not_nil(env, args[7]);
   1226 
   1227   term->vt = vterm_new(rows, cols);
   1228   vterm_set_utf8(term->vt, 1);
   1229 
   1230   term->vts = vterm_obtain_screen(term->vt);
   1231 
   1232   VTermState *state = vterm_obtain_state(term->vt);
   1233   vterm_state_set_unrecognised_fallbacks(state, &parser_callbacks, term);
   1234 
   1235 #ifndef VTermSelectionMaskNotExists
   1236   vterm_state_set_selection_callbacks(state, &selection_callbacks, term,
   1237                                       term->selection_buf, SELECTION_BUF_LEN);
   1238 #endif
   1239   vterm_state_set_bold_highbright(state, set_bold_hightbright);
   1240 
   1241   vterm_screen_reset(term->vts, 1);
   1242   vterm_screen_set_callbacks(term->vts, &vterm_screen_callbacks, term);
   1243   vterm_screen_set_damage_merge(term->vts, VTERM_DAMAGE_SCROLL);
   1244   vterm_screen_enable_altscreen(term->vts, true);
   1245   term->sb_size = MIN(SB_MAX, sb_size);
   1246   term->sb_current = 0;
   1247   term->sb_pending = 0;
   1248   term->sb_clear_pending = false;
   1249   term->sb_pending_by_height_decr = 0;
   1250   term->sb_buffer = malloc(sizeof(ScrollbackLine *) * term->sb_size);
   1251   term->invalid_start = 0;
   1252   term->invalid_end = rows;
   1253   term->is_invalidated = false;
   1254   term->width = cols;
   1255   term->height = rows;
   1256   term->height_resize = 0;
   1257   term->disable_bold_font = disable_bold_font;
   1258   term->disable_underline = disable_underline;
   1259   term->disable_inverse_video = disable_inverse_video;
   1260   term->ignore_blink_cursor = ignore_blink_cursor;
   1261   emacs_value newline = env->make_string(env, "\n", 1);
   1262   for (int i = 0; i < term->height; i++) {
   1263     insert(env, newline);
   1264   }
   1265   term->linenum = term->height;
   1266   term->linenum_added = 0;
   1267   term->resizing = false;
   1268 
   1269   term->pty_fd = -1;
   1270 
   1271   term->title = NULL;
   1272   term->title_changed = false;
   1273 
   1274   term->cursor.row = 0;
   1275   term->cursor.col = 0;
   1276   term->cursor.cursor_type = -1;
   1277   term->cursor.cursor_visible = true;
   1278   term->cursor.cursor_type_changed = false;
   1279   term->cursor.cursor_blink = false;
   1280   term->cursor.cursor_blink_changed = false;
   1281   term->directory = NULL;
   1282   term->directory_changed = false;
   1283   term->elisp_code_first = NULL;
   1284   term->elisp_code_p_insert = &term->elisp_code_first;
   1285   term->selection_data = NULL;
   1286   term->selection_mask = 0;
   1287 
   1288   term->cmd_buffer = NULL;
   1289 
   1290   term->lines = malloc(sizeof(LineInfo *) * rows);
   1291   term->lines_len = rows;
   1292   for (int i = 0; i < rows; i++) {
   1293     term->lines[i] = NULL;
   1294   }
   1295 
   1296   return env->make_user_ptr(env, term_finalize, term);
   1297 }
   1298 
   1299 emacs_value Fvterm_update(emacs_env *env, ptrdiff_t nargs, emacs_value args[],
   1300                           void *data) {
   1301   Term *term = env->get_user_ptr(env, args[0]);
   1302 
   1303   // Process keys
   1304   if (nargs > 1) {
   1305     ptrdiff_t len = string_bytes(env, args[1]);
   1306     unsigned char key[len];
   1307     env->copy_string_contents(env, args[1], (char *)key, &len);
   1308     VTermModifier modifier = VTERM_MOD_NONE;
   1309     if (nargs > 2 && env->is_not_nil(env, args[2]))
   1310       modifier = modifier | VTERM_MOD_SHIFT;
   1311     if (nargs > 3 && env->is_not_nil(env, args[3]))
   1312       modifier = modifier | VTERM_MOD_ALT;
   1313     if (nargs > 4 && env->is_not_nil(env, args[4]))
   1314       modifier = modifier | VTERM_MOD_CTRL;
   1315 
   1316     // Ignore the final zero byte
   1317     term_process_key(term, env, key, len - 1, modifier);
   1318   }
   1319 
   1320   // Flush output
   1321   term_flush_output(term, env);
   1322   if (term->is_invalidated) {
   1323     vterm_invalidate(env);
   1324   }
   1325 
   1326   return env->make_integer(env, 0);
   1327 }
   1328 
   1329 emacs_value Fvterm_redraw(emacs_env *env, ptrdiff_t nargs, emacs_value args[],
   1330                           void *data) {
   1331   Term *term = env->get_user_ptr(env, args[0]);
   1332   term_redraw(term, env);
   1333   return env->make_integer(env, 0);
   1334 }
   1335 
   1336 emacs_value Fvterm_write_input(emacs_env *env, ptrdiff_t nargs,
   1337                                emacs_value args[], void *data) {
   1338   Term *term = env->get_user_ptr(env, args[0]);
   1339   ptrdiff_t len = string_bytes(env, args[1]);
   1340   char bytes[len];
   1341 
   1342   env->copy_string_contents(env, args[1], bytes, &len);
   1343 
   1344   vterm_input_write(term->vt, bytes, len);
   1345   vterm_screen_flush_damage(term->vts);
   1346 
   1347   return env->make_integer(env, 0);
   1348 }
   1349 
   1350 emacs_value Fvterm_set_size(emacs_env *env, ptrdiff_t nargs, emacs_value args[],
   1351                             void *data) {
   1352   Term *term = env->get_user_ptr(env, args[0]);
   1353   int rows = env->extract_integer(env, args[1]);
   1354   int cols = env->extract_integer(env, args[2]);
   1355 
   1356   if (cols != term->width || rows != term->height) {
   1357     term->height_resize = rows - term->height;
   1358     if (rows > term->height) {
   1359       if (rows - term->height > term->sb_current) {
   1360         term->linenum_added = rows - term->height - term->sb_current;
   1361       }
   1362     }
   1363     term->resizing = true;
   1364     vterm_set_size(term->vt, rows, cols);
   1365     vterm_screen_flush_damage(term->vts);
   1366 
   1367     term_redraw(term, env);
   1368   }
   1369 
   1370   return Qnil;
   1371 }
   1372 
   1373 emacs_value Fvterm_set_pty_name(emacs_env *env, ptrdiff_t nargs,
   1374                                 emacs_value args[], void *data) {
   1375   Term *term = env->get_user_ptr(env, args[0]);
   1376 
   1377   if (nargs > 1) {
   1378     ptrdiff_t len = string_bytes(env, args[1]);
   1379     char filename[len];
   1380 
   1381     env->copy_string_contents(env, args[1], filename, &len);
   1382 
   1383     term->pty_fd = open(filename, O_RDONLY);
   1384   }
   1385   return Qnil;
   1386 }
   1387 emacs_value Fvterm_get_pwd(emacs_env *env, ptrdiff_t nargs, emacs_value args[],
   1388                            void *data) {
   1389   Term *term = env->get_user_ptr(env, args[0]);
   1390   int linenum = env->extract_integer(env, args[1]);
   1391   int row = linenr_to_row(term, linenum);
   1392   char *dir = get_row_directory(term, row);
   1393 
   1394   return dir ? env->make_string(env, dir, strlen(dir)) : Qnil;
   1395 }
   1396 
   1397 emacs_value Fvterm_get_icrnl(emacs_env *env, ptrdiff_t nargs,
   1398                              emacs_value args[], void *data) {
   1399   Term *term = env->get_user_ptr(env, args[0]);
   1400 
   1401   if (term->pty_fd > 0) {
   1402     struct termios keys;
   1403     tcgetattr(term->pty_fd, &keys);
   1404 
   1405     if (keys.c_iflag & ICRNL)
   1406       return Qt;
   1407     else
   1408       return Qnil;
   1409   }
   1410   return Qnil;
   1411 }
   1412 
   1413 emacs_value Fvterm_reset_cursor_point(emacs_env *env, ptrdiff_t nargs,
   1414                                       emacs_value args[], void *data) {
   1415   Term *term = env->get_user_ptr(env, args[0]);
   1416   int line = row_to_linenr(term, term->cursor.row);
   1417   goto_line(env, line);
   1418   goto_col(term, env, term->cursor.row, term->cursor.col);
   1419   return point(env);
   1420 }
   1421 
   1422 int emacs_module_init(struct emacs_runtime *ert) {
   1423   emacs_env *env = ert->get_environment(ert);
   1424 
   1425   // Symbols;
   1426   Qt = env->make_global_ref(env, env->intern(env, "t"));
   1427   Qnil = env->make_global_ref(env, env->intern(env, "nil"));
   1428   Qnormal = env->make_global_ref(env, env->intern(env, "normal"));
   1429   Qbold = env->make_global_ref(env, env->intern(env, "bold"));
   1430   Qitalic = env->make_global_ref(env, env->intern(env, "italic"));
   1431   Qforeground = env->make_global_ref(env, env->intern(env, ":foreground"));
   1432   Qbackground = env->make_global_ref(env, env->intern(env, ":background"));
   1433   Qweight = env->make_global_ref(env, env->intern(env, ":weight"));
   1434   Qunderline = env->make_global_ref(env, env->intern(env, ":underline"));
   1435   Qslant = env->make_global_ref(env, env->intern(env, ":slant"));
   1436   Qreverse = env->make_global_ref(env, env->intern(env, ":inverse-video"));
   1437   Qstrike = env->make_global_ref(env, env->intern(env, ":strike-through"));
   1438   Qextend = env->make_global_ref(env, env->intern(env, ":extend"));
   1439   Qemacs_major_version =
   1440       env->make_global_ref(env, env->intern(env, "emacs-major-version"));
   1441   Qvterm_line_wrap =
   1442       env->make_global_ref(env, env->intern(env, "vterm-line-wrap"));
   1443   Qrear_nonsticky =
   1444       env->make_global_ref(env, env->intern(env, "rear-nonsticky"));
   1445   Qvterm_prompt = env->make_global_ref(env, env->intern(env, "vterm-prompt"));
   1446 
   1447   Qface = env->make_global_ref(env, env->intern(env, "font-lock-face"));
   1448   Qbox = env->make_global_ref(env, env->intern(env, "box"));
   1449   Qbar = env->make_global_ref(env, env->intern(env, "bar"));
   1450   Qhbar = env->make_global_ref(env, env->intern(env, "hbar"));
   1451   Qcursor_type = env->make_global_ref(env, env->intern(env, "cursor-type"));
   1452 
   1453   // Functions
   1454   Fblink_cursor_mode =
   1455       env->make_global_ref(env, env->intern(env, "blink-cursor-mode"));
   1456   Fsymbol_value = env->make_global_ref(env, env->intern(env, "symbol-value"));
   1457   Flength = env->make_global_ref(env, env->intern(env, "length"));
   1458   Flist = env->make_global_ref(env, env->intern(env, "list"));
   1459   Fnth = env->make_global_ref(env, env->intern(env, "nth"));
   1460   Ferase_buffer = env->make_global_ref(env, env->intern(env, "erase-buffer"));
   1461   Finsert = env->make_global_ref(env, env->intern(env, "vterm--insert"));
   1462   Fgoto_char = env->make_global_ref(env, env->intern(env, "goto-char"));
   1463   Fput_text_property =
   1464       env->make_global_ref(env, env->intern(env, "put-text-property"));
   1465   Fadd_text_properties =
   1466       env->make_global_ref(env, env->intern(env, "add-text-properties"));
   1467   Fset = env->make_global_ref(env, env->intern(env, "set"));
   1468   Fvterm_flush_output =
   1469       env->make_global_ref(env, env->intern(env, "vterm--flush-output"));
   1470   Fforward_line = env->make_global_ref(env, env->intern(env, "forward-line"));
   1471   Fgoto_line = env->make_global_ref(env, env->intern(env, "vterm--goto-line"));
   1472   Fdelete_lines =
   1473       env->make_global_ref(env, env->intern(env, "vterm--delete-lines"));
   1474   Frecenter = env->make_global_ref(env, env->intern(env, "recenter"));
   1475   Fset_window_point =
   1476       env->make_global_ref(env, env->intern(env, "set-window-point"));
   1477   Fwindow_body_height =
   1478       env->make_global_ref(env, env->intern(env, "window-body-height"));
   1479 
   1480   Fpoint = env->make_global_ref(env, env->intern(env, "point"));
   1481   Fforward_char = env->make_global_ref(env, env->intern(env, "forward-char"));
   1482   Fget_buffer_window_list =
   1483       env->make_global_ref(env, env->intern(env, "get-buffer-window-list"));
   1484   Fselected_window =
   1485       env->make_global_ref(env, env->intern(env, "selected-window"));
   1486 
   1487   Fvterm_set_title =
   1488       env->make_global_ref(env, env->intern(env, "vterm--set-title"));
   1489   Fvterm_set_directory =
   1490       env->make_global_ref(env, env->intern(env, "vterm--set-directory"));
   1491   Fvterm_invalidate =
   1492       env->make_global_ref(env, env->intern(env, "vterm--invalidate"));
   1493   Feq = env->make_global_ref(env, env->intern(env, "eq"));
   1494   Fvterm_get_color =
   1495       env->make_global_ref(env, env->intern(env, "vterm--get-color"));
   1496   Fvterm_eval = env->make_global_ref(env, env->intern(env, "vterm--eval"));
   1497   Fvterm_set_selection =
   1498       env->make_global_ref(env, env->intern(env, "vterm--set-selection"));
   1499 
   1500   // Exported functions
   1501   emacs_value fun;
   1502   fun =
   1503       env->make_function(env, 4, 8, Fvterm_new, "Allocate a new vterm.", NULL);
   1504   bind_function(env, "vterm--new", fun);
   1505 
   1506   fun = env->make_function(env, 1, 5, Fvterm_update,
   1507                            "Process io and update the screen.", NULL);
   1508   bind_function(env, "vterm--update", fun);
   1509 
   1510   fun =
   1511       env->make_function(env, 1, 1, Fvterm_redraw, "Redraw the screen.", NULL);
   1512   bind_function(env, "vterm--redraw", fun);
   1513 
   1514   fun = env->make_function(env, 2, 2, Fvterm_write_input,
   1515                            "Write input to vterm.", NULL);
   1516   bind_function(env, "vterm--write-input", fun);
   1517 
   1518   fun = env->make_function(env, 3, 3, Fvterm_set_size,
   1519                            "Set the size of the terminal.", NULL);
   1520   bind_function(env, "vterm--set-size", fun);
   1521 
   1522   fun = env->make_function(env, 2, 2, Fvterm_set_pty_name,
   1523                            "Set the name of the pty.", NULL);
   1524   bind_function(env, "vterm--set-pty-name", fun);
   1525   fun = env->make_function(env, 2, 2, Fvterm_get_pwd,
   1526                            "Get the working directory of at line n.", NULL);
   1527   bind_function(env, "vterm--get-pwd-raw", fun);
   1528   fun = env->make_function(env, 1, 1, Fvterm_reset_cursor_point,
   1529                            "Reset cursor postion.", NULL);
   1530   bind_function(env, "vterm--reset-point", fun);
   1531 
   1532   fun = env->make_function(env, 1, 1, Fvterm_get_icrnl,
   1533                            "Get the icrnl state of the pty", NULL);
   1534   bind_function(env, "vterm--get-icrnl", fun);
   1535 
   1536   provide(env, "vterm-module");
   1537 
   1538   return 0;
   1539 }