| /* microcom.c - Simple serial console. |
| * |
| * Copyright 2017 The Android Open Source Project. |
| |
| USE_MICROCOM(NEWTOY(microcom, "<1>1s#=115200X", TOYFLAG_USR|TOYFLAG_BIN)) |
| |
| config MICROCOM |
| bool "microcom" |
| default y |
| help |
| usage: microcom [-s SPEED] [-X] DEVICE |
| |
| Simple serial console. Hit CTRL-] for menu. |
| |
| -s Set baud rate to SPEED (default 115200) |
| -X Ignore ^] menu escape |
| */ |
| |
| #define FOR_microcom |
| #include "toys.h" |
| |
| GLOBALS( |
| long s; |
| |
| int fd, stok; |
| struct termios old_stdin, old_fd; |
| ) |
| |
| // TODO: tty_sigreset outputs ansi escape sequences, how to disable? |
| static void restore_states(int i) |
| { |
| if (TT.stok) tcsetattr(0, TCSAFLUSH, &TT.old_stdin); |
| tcsetattr(TT.fd, TCSAFLUSH, &TT.old_fd); |
| } |
| |
| static void handle_esc(void) |
| { |
| char input; |
| |
| xputsn("\r\n[b]reak, [p]aste file, [q]uit: "); |
| if (read(0, &input, 1)<1 || input == CTRL('D') || input == 'q') { |
| xputs("exit\r"); |
| xexit(); |
| } |
| if (input == 'b') tcsendbreak(TT.fd, 0); |
| else if (input == 'p') { |
| long long written = 0, size; |
| char* filename; |
| int len = 0, fd; |
| |
| // TODO: share code with hexedit's prompt() and vi's ex mode. |
| // TODO: tab completion! |
| memset(toybuf, 0, sizeof(toybuf)); |
| while (1) { |
| xprintf("\r\e[2K\e[1mFilename: \e[0m%s", toybuf); |
| if (read(0, &input, 1) <= 0 || input == CTRL('[')) { |
| return; |
| } |
| if (input == '\r') break; |
| if (input == 0x7f && len > 0) toybuf[--len] = 0; |
| else if (input == CTRL('U')) while (len > 0) toybuf[--len] = 0; |
| else if (input >= ' ' && input <= 0x7f && len < sizeof(toybuf)) |
| toybuf[len++] = input; |
| } |
| toybuf[len] = 0; |
| if (!len) return; |
| filename = xstrdup(toybuf); |
| fd = xopen(filename, O_RDONLY); |
| size = fdlength(fd); |
| // The alternative would be to just feed this fd into the usual loop, |
| // so we're reading back these characters if they're being echoed, but |
| // for my specific use case of pasting into `base64 -d -i > foo`, this |
| // is a much more convenient UI. |
| while ((len = read(fd, toybuf, sizeof(toybuf))) > 0) { |
| written += len; |
| xprintf("\r\e[2KPasting '%s' %lld/%lld (%lld%%)...", filename, written, |
| size, written*100/size); |
| xwrite(TT.fd, toybuf, len); |
| } |
| free(filename); |
| close(fd); |
| } else { |
| xprintf("Ignoring unknown command."); |
| } |
| xprintf("\r\n"); |
| } |
| |
| void microcom_main(void) |
| { |
| struct termios tio; |
| struct pollfd fds[2]; |
| int i; |
| |
| // Open with O_NDELAY, but switch back to blocking for reads. |
| TT.fd = xopen(*toys.optargs, O_RDWR | O_NOCTTY | O_NDELAY); |
| if (-1==(i = fcntl(TT.fd, F_GETFL, 0)) || fcntl(TT.fd, F_SETFL, i&~O_NDELAY) |
| || tcgetattr(TT.fd, &TT.old_fd)) |
| perror_exit_raw(*toys.optargs); |
| |
| // Set both input and output to raw mode. |
| memcpy(&tio, &TT.old_fd, sizeof(struct termios)); |
| cfmakeraw(&tio); |
| xsetspeed(&tio, TT.s); |
| if (tcsetattr(TT.fd, TCSAFLUSH, &tio)) perror_exit("set speed"); |
| if (!set_terminal(0, 1, 0, &TT.old_stdin)) TT.stok++; |
| // ...and arrange to restore things, however we may exit. |
| sigatexit(restore_states); |
| |
| fds[0].fd = TT.fd; |
| fds[1].fd = 0; |
| fds[0].events = fds[1].events = POLLIN; |
| |
| if (!FLAG(X)) xputs("Escape character is '^]'.\r"); |
| while (poll(fds, 2, -1) > 0) { |
| |
| // Read from connection, write to stdout. |
| if (fds[0].revents) { |
| if (0 < (i = read(TT.fd, toybuf, sizeof(toybuf)))) xwrite(0, toybuf, i); |
| else break; |
| } |
| |
| // Read from stdin, write to connection. |
| if (fds[1].revents) { |
| if (read(0, toybuf, 1) != 1) break; |
| if (!FLAG(X) && *toybuf == CTRL(']')) handle_esc(); |
| else xwrite(TT.fd, toybuf, 1); |
| } |
| } |
| } |