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