xref: /vim-8.2.3635/src/libvterm/t/harness.c (revision e5886ccb)
1 #include "vterm.h"
2 #include "../src/vterm_internal.h" // We pull in some internal bits too
3 
4 #include <stdio.h>
5 #include <string.h>
6 
7 #define streq(a,b) (!strcmp(a,b))
8 #define strstartswith(a,b) (!strncmp(a,b,strlen(b)))
9 
inplace_hex2bytes(char * s)10 static size_t inplace_hex2bytes(char *s)
11 {
12   char *inpos = s, *outpos = s;
13 
14   while(*inpos) {
15     unsigned int ch;
16     sscanf(inpos, "%2x", &ch);
17     *outpos = ch;
18     outpos += 1; inpos += 2;
19   }
20 
21   return outpos - s;
22 }
23 
strpe_modifiers(char ** strp)24 static VTermModifier strpe_modifiers(char **strp)
25 {
26   VTermModifier state = 0;
27 
28   while((*strp)[0]) {
29     switch(((*strp)++)[0]) {
30       case 'S': state |= VTERM_MOD_SHIFT; break;
31       case 'C': state |= VTERM_MOD_CTRL;  break;
32       case 'A': state |= VTERM_MOD_ALT;   break;
33       default: return state;
34     }
35   }
36 
37   return state;
38 }
39 
strp_key(char * str)40 static VTermKey strp_key(char *str)
41 {
42   static struct {
43     char *name;
44     VTermKey key;
45   } keys[] = {
46     { "Up",    VTERM_KEY_UP },
47     { "Tab",   VTERM_KEY_TAB },
48     { "Enter", VTERM_KEY_ENTER },
49     { "KP0",   VTERM_KEY_KP_0 },
50     { "F1",    VTERM_KEY_FUNCTION(1) },
51     { NULL,    VTERM_KEY_NONE },
52   };
53   int i;
54 
55   for(i = 0; keys[i].name; i++) {
56     if(streq(str, keys[i].name))
57       return keys[i].key;
58   }
59 
60   return VTERM_KEY_NONE;
61 }
62 
print_color(const VTermColor * col)63 static void print_color(const VTermColor *col)
64 {
65   if (VTERM_COLOR_IS_RGB(col)) {
66     printf("rgb(%d,%d,%d", col->red, col->green, col->blue);
67   }
68   else if (VTERM_COLOR_IS_INDEXED(col)) {
69     printf("idx(%d", col->index);
70   }
71   else {
72     printf("invalid(%d", col->type);
73   }
74   if (VTERM_COLOR_IS_DEFAULT_FG(col)) {
75     printf(",is_default_fg");
76   }
77   if (VTERM_COLOR_IS_DEFAULT_BG(col)) {
78     printf(",is_default_bg");
79   }
80   printf(")");
81 }
82 
83 static VTerm *vt;
84 static VTermState *state;
85 static VTermScreen *screen;
86 
87 static VTermEncodingInstance encoding;
88 
term_output(const char * s,size_t len,void * user UNUSED)89 static void term_output(const char *s, size_t len, void *user UNUSED)
90 {
91   size_t i;
92 
93   printf("output ");
94   for(i = 0; i < len; i++)
95     printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
96 }
97 
printhex(const char * s,size_t len)98 static void printhex(const char *s, size_t len)
99 {
100   while(len--)
101     printf("%02x", (s++)[0]);
102 }
103 
parser_text(const char bytes[],size_t len,void * user UNUSED)104 static int parser_text(const char bytes[], size_t len, void *user UNUSED)
105 {
106   size_t i;
107 
108   printf("text ");
109   for(i = 0; i < len; i++) {
110     unsigned char b = bytes[i];
111     if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
112       break;
113     printf(i ? ",%x" : "%x", b);
114   }
115   printf("\n");
116 
117   return i;
118 }
119 
parser_control(unsigned char control,void * user UNUSED)120 static int parser_control(unsigned char control, void *user UNUSED)
121 {
122   printf("control %02x\n", control);
123 
124   return 1;
125 }
126 
parser_escape(const char bytes[],size_t len,void * user UNUSED)127 static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
128 {
129   if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
130     if(len < 2)
131       return -1;
132     len = 2;
133   }
134   else {
135     len = 1;
136   }
137 
138   printf("escape ");
139   printhex(bytes, len);
140   printf("\n");
141 
142   return len;
143 }
144 
parser_csi(const char * leader,const long args[],int argcount,const char * intermed,char command,void * user UNUSED)145 static int parser_csi(const char *leader, const long args[], int argcount, const char *intermed, char command, void *user UNUSED)
146 {
147   int i;
148   printf("csi %02x", command);
149 
150   if(leader && leader[0]) {
151     printf(" L=");
152     for(i = 0; leader[i]; i++)
153       printf("%02x", leader[i]);
154   }
155 
156   for(i = 0; i < argcount; i++) {
157     char sep = i ? ',' : ' ';
158 
159     if(args[i] == CSI_ARG_MISSING)
160       printf("%c*", sep);
161     else
162       printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
163   }
164 
165   if(intermed && intermed[0]) {
166     printf(" I=");
167     for(i = 0; intermed[i]; i++)
168       printf("%02x", intermed[i]);
169   }
170 
171   printf("\n");
172 
173   return 1;
174 }
175 
parser_osc(int command,VTermStringFragment frag,void * user UNUSED)176 static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
177 {
178 
179   printf("osc ");
180 
181   if(frag.initial) {
182     if(command == -1)
183       printf("[");
184     else
185       printf("[%d;", command);
186   }
187 
188   printhex(frag.str, frag.len);
189 
190   if(frag.final)
191     printf("]");
192 
193   printf("\n");
194 
195   return 1;
196 }
197 
parser_dcs(const char * command,size_t commandlen,VTermStringFragment frag,void * user UNUSED)198 static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
199 {
200   printf("dcs ");
201 
202   if(frag.initial) {
203     size_t i;
204     printf("[");
205     for(i = 0; i < commandlen; i++)
206       printf("%02x", command[i]);
207   }
208 
209   printhex(frag.str, frag.len);
210 
211   if(frag.final)
212     printf("]");
213 
214   printf("\n");
215 
216   return 1;
217 }
218 
219 static VTermParserCallbacks parser_cbs = {
220   parser_text, // text
221   parser_control, // control
222   parser_escape, // escape
223   parser_csi, // csi
224   parser_osc, // osc
225   parser_dcs, // dcs
226   NULL // resize
227 };
228 
229 static VTermStateFallbacks fallbacks = {
230   parser_control, // control
231   parser_csi, // csi
232   parser_osc, // osc
233   parser_dcs // dcs
234 };
235 
236 /* These callbacks are shared by State and Screen */
237 
238 static int want_movecursor = 0;
239 static VTermPos state_pos;
movecursor(VTermPos pos,VTermPos oldpos UNUSED,int visible UNUSED,void * user UNUSED)240 static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
241 {
242   state_pos = pos;
243 
244   if(want_movecursor)
245     printf("movecursor %d,%d\n", pos.row, pos.col);
246 
247   return 1;
248 }
249 
250 static int want_scrollrect = 0;
scrollrect(VTermRect rect,int downward,int rightward,void * user UNUSED)251 static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
252 {
253   if(!want_scrollrect)
254     return 0;
255 
256   printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
257       rect.start_row, rect.end_row, rect.start_col, rect.end_col,
258       downward, rightward);
259 
260   return 1;
261 }
262 
263 static int want_moverect = 0;
moverect(VTermRect dest,VTermRect src,void * user UNUSED)264 static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
265 {
266   if(!want_moverect)
267     return 0;
268 
269   printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
270       src.start_row,  src.end_row,  src.start_col,  src.end_col,
271       dest.start_row, dest.end_row, dest.start_col, dest.end_col);
272 
273   return 1;
274 }
275 
276 static int want_settermprop = 0;
settermprop(VTermProp prop,VTermValue * val,void * user UNUSED)277 static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
278 {
279   VTermValueType type;
280   if(!want_settermprop)
281     return 1;
282 
283   type = vterm_get_prop_type(prop);
284   switch(type) {
285   case VTERM_VALUETYPE_BOOL:
286     printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
287     return 1;
288   case VTERM_VALUETYPE_INT:
289     printf("settermprop %d %d\n", prop, val->number);
290     return 1;
291   case VTERM_VALUETYPE_STRING:
292     printf("settermprop %d %s\"%.*s\"%s\n", prop,
293         val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
294     return 1;
295   case VTERM_VALUETYPE_COLOR:
296     printf("settermprop %d ", prop);
297     print_color(&val->color);
298     printf("\n");
299     return 1;
300 
301   case VTERM_N_VALUETYPES:
302     return 0;
303   }
304 
305   return 0;
306 }
307 
308 // These callbacks are for State
309 
310 static int want_state_putglyph = 0;
state_putglyph(VTermGlyphInfo * info,VTermPos pos,void * user UNUSED)311 static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
312 {
313   int i;
314   if(!want_state_putglyph)
315     return 1;
316 
317   printf("putglyph ");
318   for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
319     printf(i ? ",%x" : "%x", info->chars[i]);
320   printf(" %d %d,%d", info->width, pos.row, pos.col);
321   if(info->protected_cell)
322     printf(" prot");
323   if(info->dwl)
324     printf(" dwl");
325   if(info->dhl)
326     printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
327   printf("\n");
328 
329   return 1;
330 }
331 
332 static int want_state_erase = 0;
state_erase(VTermRect rect,int selective,void * user UNUSED)333 static int state_erase(VTermRect rect, int selective, void *user UNUSED)
334 {
335   if(!want_state_erase)
336     return 1;
337 
338   printf("erase %d..%d,%d..%d%s\n",
339       rect.start_row, rect.end_row, rect.start_col, rect.end_col,
340       selective ? " selective" : "");
341 
342   return 1;
343 }
344 
345 static struct {
346   int bold;
347   int underline;
348   int italic;
349   int blink;
350   int reverse;
351   int conceal;
352   int strike;
353   int font;
354   VTermColor foreground;
355   VTermColor background;
356 } state_pen;
state_setpenattr(VTermAttr attr,VTermValue * val,void * user UNUSED)357 static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
358 {
359   switch(attr) {
360   case VTERM_ATTR_BOLD:
361     state_pen.bold = val->boolean;
362     break;
363   case VTERM_ATTR_UNDERLINE:
364     state_pen.underline = val->number;
365     break;
366   case VTERM_ATTR_ITALIC:
367     state_pen.italic = val->boolean;
368     break;
369   case VTERM_ATTR_BLINK:
370     state_pen.blink = val->boolean;
371     break;
372   case VTERM_ATTR_REVERSE:
373     state_pen.reverse = val->boolean;
374     break;
375   case VTERM_ATTR_CONCEAL:
376     state_pen.conceal = val->boolean;
377     break;
378   case VTERM_ATTR_STRIKE:
379     state_pen.strike = val->boolean;
380     break;
381   case VTERM_ATTR_FONT:
382     state_pen.font = val->number;
383     break;
384   case VTERM_ATTR_FOREGROUND:
385     state_pen.foreground = val->color;
386     break;
387   case VTERM_ATTR_BACKGROUND:
388     state_pen.background = val->color;
389     break;
390 
391   case VTERM_N_ATTRS:
392     return 0;
393   }
394 
395   return 1;
396 }
397 
state_setlineinfo(int row UNUSED,const VTermLineInfo * newinfo UNUSED,const VTermLineInfo * oldinfo UNUSED,void * user UNUSED)398 static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
399 {
400   return 1;
401 }
402 
403 VTermStateCallbacks state_cbs = {
404   state_putglyph, // putglyph
405   movecursor, // movecursor
406   scrollrect, // scrollrect
407   moverect, // moverect
408   state_erase, // erase
409   NULL, // initpen
410   state_setpenattr, // setpenattr
411   settermprop, // settermprop
412   NULL, // bell
413   NULL, // resize
414   state_setlineinfo, // setlineinfo
415 };
416 
417 static int want_screen_damage = 0;
418 static int want_screen_damage_cells = 0;
screen_damage(VTermRect rect,void * user UNUSED)419 static int screen_damage(VTermRect rect, void *user UNUSED)
420 {
421   if(!want_screen_damage)
422     return 1;
423 
424   printf("damage %d..%d,%d..%d",
425       rect.start_row, rect.end_row, rect.start_col, rect.end_col);
426 
427   if(want_screen_damage_cells) {
428     int equals = FALSE;
429     int row;
430     int col;
431 
432     for(row = rect.start_row; row < rect.end_row; row++) {
433       int eol = rect.end_col;
434       while(eol > rect.start_col) {
435         VTermScreenCell cell;
436 	VTermPos pos;
437 	pos.row = row;
438 	pos.col = eol-1;
439         vterm_screen_get_cell(screen, pos, &cell);
440         if(cell.chars[0])
441           break;
442 
443         eol--;
444       }
445 
446       if(eol == rect.start_col)
447         break;
448 
449       if(!equals)
450         printf(" ="), equals = TRUE;
451 
452       printf(" %d<", row);
453       for(col = rect.start_col; col < eol; col++) {
454         VTermScreenCell cell;
455 	VTermPos pos;
456 	pos.row = row;
457 	pos.col = col;
458         vterm_screen_get_cell(screen, pos, &cell);
459         printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
460       }
461       printf(">");
462     }
463   }
464 
465   printf("\n");
466 
467   return 1;
468 }
469 
470 static int want_screen_scrollback = 0;
screen_sb_pushline(int cols,const VTermScreenCell * cells,void * user UNUSED)471 static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
472 {
473   int eol;
474   int c;
475 
476   if(!want_screen_scrollback)
477     return 1;
478 
479   eol = cols;
480   while(eol && !cells[eol-1].chars[0])
481     eol--;
482 
483   printf("sb_pushline %d =", cols);
484   for(c = 0; c < eol; c++)
485     printf(" %02X", cells[c].chars[0]);
486   printf("\n");
487 
488   return 1;
489 }
490 
screen_sb_popline(int cols,VTermScreenCell * cells,void * user UNUSED)491 static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
492 {
493   int col;
494 
495   if(!want_screen_scrollback)
496     return 0;
497 
498   // All lines of scrollback contain "ABCDE"
499   for(col = 0; col < cols; col++) {
500     if(col < 5)
501       cells[col].chars[0] = 'A' + col;
502     else
503       cells[col].chars[0] = 0;
504 
505     cells[col].width = 1;
506   }
507 
508   printf("sb_popline %d\n", cols);
509   return 1;
510 }
511 
512 VTermScreenCallbacks screen_cbs = {
513   screen_damage, // damage
514   moverect, // moverect
515   movecursor, // movecursor
516   settermprop, // settermprop
517   NULL, // bell
518   NULL, // resize
519   screen_sb_pushline, // sb_pushline
520   screen_sb_popline // sb_popline
521 };
522 
main(int argc UNUSED,char ** argv UNUSED)523 int main(int argc UNUSED, char **argv UNUSED)
524 {
525   char line[1024] = {0};
526   int flag;
527 
528   int err;
529 
530   setvbuf(stdout, NULL, _IONBF, 0);
531 
532   while(fgets(line, sizeof line, stdin)) {
533     char *nl;
534     size_t outlen;
535     err = 0;
536 
537     if((nl = strchr(line, '\n')))
538       *nl = '\0';
539 
540     if(streq(line, "INIT")) {
541       if(!vt)
542         vt = vterm_new(25, 80);
543 
544       // Somehow this makes tests fail
545       // vterm_output_set_callback(vt, term_output, NULL);
546     }
547 
548     else if(streq(line, "WANTPARSER")) {
549       vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
550     }
551 
552     else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
553       int i = 9;
554       int sense = 1;
555       if(!state) {
556         state = vterm_obtain_state(vt);
557         vterm_state_set_callbacks(state, &state_cbs, NULL);
558         vterm_state_set_bold_highbright(state, 1);
559         vterm_state_reset(state, 1);
560       }
561 
562       while(line[i] == ' ')
563         i++;
564       for( ; line[i]; i++)
565         switch(line[i]) {
566         case '+':
567           sense = 1;
568           break;
569         case '-':
570           sense = 0;
571           break;
572         case 'g':
573           want_state_putglyph = sense;
574           break;
575         case 's':
576           want_scrollrect = sense;
577           break;
578         case 'm':
579           want_moverect = sense;
580           break;
581         case 'e':
582           want_state_erase = sense;
583           break;
584         case 'p':
585           want_settermprop = sense;
586           break;
587         case 'f':
588           vterm_state_set_unrecognised_fallbacks(state, sense ? &fallbacks : NULL, NULL);
589           break;
590         default:
591           fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
592         }
593     }
594 
595     else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
596       int i = 10;
597       int sense = 1;
598       if(!screen)
599         screen = vterm_obtain_screen(vt);
600       vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
601 
602       while(line[i] == ' ')
603         i++;
604       for( ; line[i]; i++)
605         switch(line[i]) {
606         case '-':
607           sense = 0;
608           break;
609         case 'a':
610           vterm_screen_enable_altscreen(screen, 1);
611           break;
612         case 'd':
613           want_screen_damage = sense;
614           break;
615         case 'D':
616           want_screen_damage = sense;
617           want_screen_damage_cells = sense;
618           break;
619         case 'm':
620           want_moverect = sense;
621           break;
622         case 'c':
623           want_movecursor = sense;
624           break;
625         case 'p':
626           want_settermprop = 1;
627           break;
628         case 'b':
629           want_screen_scrollback = sense;
630           break;
631         default:
632           fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
633         }
634     }
635 
636     else if(sscanf(line, "UTF8 %d", &flag)) {
637       vterm_set_utf8(vt, flag);
638     }
639 
640     else if(streq(line, "RESET")) {
641       if(state) {
642         vterm_state_reset(state, 1);
643         vterm_state_get_cursorpos(state, &state_pos);
644       }
645       if(screen) {
646         vterm_screen_reset(screen, 1);
647       }
648     }
649 
650     else if(strstartswith(line, "RESIZE ")) {
651       int rows, cols;
652       char *linep = line + 7;
653       while(linep[0] == ' ')
654         linep++;
655       sscanf(linep, "%d, %d", &rows, &cols);
656       vterm_set_size(vt, rows, cols);
657     }
658 
659     else if(strstartswith(line, "PUSH ")) {
660       char *bytes = line + 5;
661       size_t len = inplace_hex2bytes(bytes);
662       size_t written = vterm_input_write(vt, bytes, len);
663       if(written < len)
664         fprintf(stderr, "! short write\n");
665     }
666 
667     else if(streq(line, "WANTENCODING")) {
668       // This isn't really external API but it's hard to get this out any
669       // other way
670       encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
671       if(encoding.enc->init)
672         (*encoding.enc->init)(encoding.enc, encoding.data);
673     }
674 
675     else if(strstartswith(line, "ENCIN ")) {
676       char *bytes = line + 6;
677       size_t len = inplace_hex2bytes(bytes);
678 
679       uint32_t cp[1024];
680       int cpi = 0;
681       size_t pos = 0;
682 
683       (*encoding.enc->decode)(encoding.enc, encoding.data,
684           cp, &cpi, len, bytes, &pos, len);
685 
686       if(cpi > 0) {
687 	int i;
688         printf("encout ");
689         for(i = 0; i < cpi; i++) {
690           printf(i ? ",%x" : "%x", cp[i]);
691         }
692         printf("\n");
693       }
694     }
695 
696     else if(strstartswith(line, "INCHAR ")) {
697       char *linep = line + 7;
698       unsigned int c = 0;
699       VTermModifier mod;
700       while(linep[0] == ' ')
701         linep++;
702       mod = strpe_modifiers(&linep);
703       sscanf(linep, " %x", &c);
704 
705       vterm_keyboard_unichar(vt, c, mod);
706     }
707 
708     else if(strstartswith(line, "INKEY ")) {
709       VTermModifier mod;
710       VTermKey key;
711       char *linep = line + 6;
712       while(linep[0] == ' ')
713         linep++;
714       mod = strpe_modifiers(&linep);
715       while(linep[0] == ' ')
716         linep++;
717       key = strp_key(linep);
718 
719       vterm_keyboard_key(vt, key, mod);
720     }
721 
722     else if(strstartswith(line, "PASTE ")) {
723       char *linep = line + 6;
724       if(streq(linep, "START"))
725         vterm_keyboard_start_paste(vt);
726       else if(streq(linep, "END"))
727         vterm_keyboard_end_paste(vt);
728       else
729         goto abort_line;
730     }
731 
732     else if(strstartswith(line, "FOCUS ")) {
733       char *linep = line + 6;
734       if(streq(linep, "IN"))
735         vterm_state_focus_in(state);
736       else if(streq(linep, "OUT"))
737         vterm_state_focus_out(state);
738       else
739         goto abort_line;
740     }
741 
742     else if(strstartswith(line, "MOUSEMOVE ")) {
743       char *linep = line + 10;
744       int row, col, len;
745       VTermModifier mod;
746       while(linep[0] == ' ')
747         linep++;
748       sscanf(linep, "%d,%d%n", &row, &col, &len);
749       linep += len;
750       while(linep[0] == ' ')
751         linep++;
752       mod = strpe_modifiers(&linep);
753       vterm_mouse_move(vt, row, col, mod);
754     }
755 
756     else if(strstartswith(line, "MOUSEBTN ")) {
757       char *linep = line + 9;
758       char press;
759       int button, len;
760       VTermModifier mod;
761       while(linep[0] == ' ')
762         linep++;
763       sscanf(linep, "%c %d%n", &press, &button, &len);
764       linep += len;
765       while(linep[0] == ' ')
766         linep++;
767       mod = strpe_modifiers(&linep);
768       vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
769     }
770 
771     else if(strstartswith(line, "DAMAGEMERGE ")) {
772       char *linep = line + 12;
773       while(linep[0] == ' ')
774         linep++;
775       if(streq(linep, "CELL"))
776         vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
777       else if(streq(linep, "ROW"))
778         vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
779       else if(streq(linep, "SCREEN"))
780         vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
781       else if(streq(linep, "SCROLL"))
782         vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
783     }
784 
785     else if(strstartswith(line, "DAMAGEFLUSH")) {
786       vterm_screen_flush_damage(screen);
787     }
788 
789     else if(line[0] == '?') {
790       if(streq(line, "?cursor")) {
791         VTermPos pos;
792         vterm_state_get_cursorpos(state, &pos);
793         if(pos.row != state_pos.row)
794           printf("! row mismatch: state=%d,%d event=%d,%d\n",
795               pos.row, pos.col, state_pos.row, state_pos.col);
796         else if(pos.col != state_pos.col)
797           printf("! col mismatch: state=%d,%d event=%d,%d\n",
798               pos.row, pos.col, state_pos.row, state_pos.col);
799         else
800           printf("%d,%d\n", state_pos.row, state_pos.col);
801       }
802       else if(strstartswith(line, "?pen ")) {
803         VTermValue val;
804         char *linep = line + 5;
805         while(linep[0] == ' ')
806           linep++;
807 
808 #define BOOLSTR(v) ((v) ? "on" : "off")
809 
810         if(streq(linep, "bold")) {
811           vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
812           if(val.boolean != state_pen.bold)
813             printf("! pen bold mismatch; state=%s, event=%s\n",
814                 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
815           else
816             printf("%s\n", BOOLSTR(state_pen.bold));
817         }
818         else if(streq(linep, "underline")) {
819           vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
820           if(val.boolean != state_pen.underline)
821             printf("! pen underline mismatch; state=%d, event=%d\n",
822                 val.boolean, state_pen.underline);
823           else
824             printf("%d\n", state_pen.underline);
825         }
826         else if(streq(linep, "italic")) {
827           vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
828           if(val.boolean != state_pen.italic)
829             printf("! pen italic mismatch; state=%s, event=%s\n",
830                 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
831           else
832             printf("%s\n", BOOLSTR(state_pen.italic));
833         }
834         else if(streq(linep, "blink")) {
835           vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
836           if(val.boolean != state_pen.blink)
837             printf("! pen blink mismatch; state=%s, event=%s\n",
838                 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
839           else
840             printf("%s\n", BOOLSTR(state_pen.blink));
841         }
842         else if(streq(linep, "reverse")) {
843           vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
844           if(val.boolean != state_pen.reverse)
845             printf("! pen reverse mismatch; state=%s, event=%s\n",
846                 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
847           else
848             printf("%s\n", BOOLSTR(state_pen.reverse));
849         }
850         else if(streq(linep, "font")) {
851           vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
852           if(val.boolean != state_pen.font)
853             printf("! pen font mismatch; state=%d, event=%d\n",
854                 val.boolean, state_pen.font);
855           else
856             printf("%d\n", state_pen.font);
857         }
858         else if(streq(linep, "foreground")) {
859           print_color(&state_pen.foreground);
860           printf("\n");
861         }
862         else if(streq(linep, "background")) {
863           print_color(&state_pen.background);
864           printf("\n");
865         }
866         else
867           printf("?\n");
868       }
869       else if(strstartswith(line, "?lineinfo ")) {
870         char *linep = line + 10;
871         int row;
872         const VTermLineInfo *info;
873         while(linep[0] == ' ')
874           linep++;
875         if(sscanf(linep, "%d", &row) < 1) {
876           printf("! lineinfo unrecognised input\n");
877           goto abort_line;
878         }
879         info = vterm_state_get_lineinfo(state, row);
880         if(info->doublewidth)
881           printf("dwl ");
882         if(info->doubleheight)
883           printf("dhl ");
884         if(info->continuation)
885           printf("cont ");
886         printf("\n");
887       }
888       else if(strstartswith(line, "?screen_chars ")) {
889         char *linep = line + 13;
890         VTermRect rect;
891         size_t len;
892         while(linep[0] == ' ')
893           linep++;
894         if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
895           printf("! screen_chars unrecognised input\n");
896           goto abort_line;
897         }
898         len = vterm_screen_get_chars(screen, NULL, 0, rect);
899         if(len == (size_t)-1)
900           printf("! screen_chars error\n");
901         else if(len == 0)
902           printf("\n");
903         else {
904           uint32_t *chars = malloc(sizeof(uint32_t) * len);
905           size_t i;
906           vterm_screen_get_chars(screen, chars, len, rect);
907           for(i = 0; i < len; i++) {
908             printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
909           }
910           free(chars);
911         }
912       }
913       else if(strstartswith(line, "?screen_text ")) {
914         char *linep = line + 12;
915         VTermRect rect;
916         size_t len;
917         while(linep[0] == ' ')
918           linep++;
919         if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
920           printf("! screen_text unrecognised input\n");
921           goto abort_line;
922         }
923         len = vterm_screen_get_text(screen, NULL, 0, rect);
924         if(len == (size_t)-1)
925           printf("! screen_text error\n");
926         else if(len == 0)
927           printf("\n");
928         else {
929           // Put an overwrite guard at both ends of the buffer
930           unsigned char *buffer = malloc(len + 4);
931           unsigned char *text = buffer + 2;
932           text[-2] = 0x55; text[-1] = 0xAA;
933           text[len] = 0x55; text[len+1] = 0xAA;
934 
935           vterm_screen_get_text(screen, (char *)text, len, rect);
936 
937           if(text[-2] != 0x55 || text[-1] != 0xAA)
938             printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
939           else if(text[len] != 0x55 || text[len+1] != 0xAA)
940             printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
941           else
942 	  {
943 	    size_t i;
944             for(i = 0; i < len; i++) {
945               printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
946             }
947 	  }
948 
949           free(buffer);
950         }
951       }
952       else if(strstartswith(line, "?screen_cell ")) {
953         char *linep = line + 12;
954 	int i;
955         VTermPos pos;
956         VTermScreenCell cell;
957         while(linep[0] == ' ')
958           linep++;
959         if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
960           printf("! screen_cell unrecognised input\n");
961           goto abort_line;
962         }
963         if(!vterm_screen_get_cell(screen, pos, &cell))
964           goto abort_line;
965         printf("{");
966         for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
967           printf("%s0x%x", i ? "," : "", cell.chars[i]);
968         }
969         printf("} width=%d attrs={", cell.width);
970         if(cell.attrs.bold)      printf("B");
971         if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
972         if(cell.attrs.italic)    printf("I");
973         if(cell.attrs.blink)     printf("K");
974         if(cell.attrs.reverse)   printf("R");
975         if(cell.attrs.font)      printf("F%d", cell.attrs.font);
976         printf("} ");
977         if(cell.attrs.dwl)       printf("dwl ");
978         if(cell.attrs.dhl)       printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
979         printf("fg=");
980         vterm_screen_convert_color_to_rgb(screen, &cell.fg);
981         print_color(&cell.fg);
982         printf(" bg=");
983         vterm_screen_convert_color_to_rgb(screen, &cell.bg);
984         print_color(&cell.bg);
985         printf("\n");
986       }
987       else if(strstartswith(line, "?screen_eol ")) {
988         VTermPos pos;
989         char *linep = line + 12;
990         while(linep[0] == ' ')
991           linep++;
992         if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
993           printf("! screen_eol unrecognised input\n");
994           goto abort_line;
995         }
996         printf("%d\n", vterm_screen_is_eol(screen, pos));
997       }
998       else if(strstartswith(line, "?screen_attrs_extent ")) {
999         VTermPos pos;
1000         VTermRect rect;
1001         char *linep = line + 21;
1002         while(linep[0] == ' ')
1003           linep++;
1004         if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1005           printf("! screen_attrs_extent unrecognised input\n");
1006           goto abort_line;
1007         }
1008 	rect.start_col = 0;
1009 	rect.end_col   = -1;
1010         if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
1011           printf("! screen_attrs_extent failed\n");
1012           goto abort_line;
1013         }
1014         printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
1015       }
1016       else
1017         printf("?\n");
1018 
1019       memset(line, 0, sizeof line);
1020       continue;
1021     }
1022 
1023     else
1024       abort_line: err = 1;
1025 
1026     outlen = vterm_output_get_buffer_current(vt);
1027     if(outlen > 0) {
1028       char outbuff[1024];
1029       vterm_output_read(vt, outbuff, outlen);
1030 
1031       term_output(outbuff, outlen, NULL);
1032     }
1033 
1034     printf(err ? "?\n" : "DONE\n");
1035   }
1036 
1037   vterm_free(vt);
1038 
1039   return 0;
1040 }
1041