aboutsummaryrefslogtreecommitdiff
path: root/libbb/lineedit.c
diff options
context:
space:
mode:
Diffstat (limited to 'libbb/lineedit.c')
-rw-r--r--libbb/lineedit.c45
1 files changed, 24 insertions, 21 deletions
diff --git a/libbb/lineedit.c b/libbb/lineedit.c
index 8e339da53..7c0eef90d 100644
--- a/libbb/lineedit.c
+++ b/libbb/lineedit.c
@@ -1658,27 +1658,28 @@ static void win_changed(int nsig)
1658static int lineedit_read_key(char *read_key_buffer) 1658static int lineedit_read_key(char *read_key_buffer)
1659{ 1659{
1660 int64_t ic; 1660 int64_t ic;
1661 struct pollfd pfd; 1661 int timeout = -1;
1662 int delay = -1;
1663#if ENABLE_FEATURE_ASSUME_UNICODE 1662#if ENABLE_FEATURE_ASSUME_UNICODE
1664 char unicode_buf[MB_CUR_MAX + 1]; 1663 char unicode_buf[MB_CUR_MAX + 1];
1665 int unicode_idx = 0; 1664 int unicode_idx = 0;
1666#endif 1665#endif
1667 1666
1668 pfd.fd = STDIN_FILENO; 1667 while (1) {
1669 pfd.events = POLLIN; 1668 /* Wait for input. TIMEOUT = -1 makes read_key wait even
1670 do { 1669 * on nonblocking stdin, TIMEOUT = 50 makes sure we won't
1671#if ENABLE_FEATURE_EDITING_ASK_TERMINAL || ENABLE_FEATURE_ASSUME_UNICODE 1670 * insist on full MB_CUR_MAX buffer to declare input like
1672 poll_again: 1671 * "\xff\n",pause,"ls\n" invalid and thus won't lose "ls".
1672 *
1673 * Note: read_key sets errno to 0 on success.
1674 */
1675 ic = read_key(STDIN_FILENO, read_key_buffer, timeout);
1676 if (errno) {
1677#if ENABLE_FEATURE_ASSUME_UNICODE
1678 if (errno == EAGAIN && unicode_idx != 0)
1679 goto pushback;
1673#endif 1680#endif
1674 if (read_key_buffer[0] == 0) { 1681 break;
1675 /* Wait for input. Can't just call read_key,
1676 * it returns at once if stdin
1677 * is in non-blocking mode. */
1678 safe_poll(&pfd, 1, delay);
1679 } 1682 }
1680 /* Note: read_key sets errno to 0 on success: */
1681 ic = read_key(STDIN_FILENO, read_key_buffer);
1682 1683
1683#if ENABLE_FEATURE_EDITING_ASK_TERMINAL 1684#if ENABLE_FEATURE_EDITING_ASK_TERMINAL
1684 if ((int32_t)ic == KEYCODE_CURSOR_POS 1685 if ((int32_t)ic == KEYCODE_CURSOR_POS
@@ -1695,7 +1696,7 @@ static int lineedit_read_key(char *read_key_buffer)
1695 } 1696 }
1696 } 1697 }
1697 } 1698 }
1698 goto poll_again; 1699 continue;
1699 } 1700 }
1700#endif 1701#endif
1701 1702
@@ -1704,19 +1705,20 @@ static int lineedit_read_key(char *read_key_buffer)
1704 wchar_t wc; 1705 wchar_t wc;
1705 1706
1706 if ((int32_t)ic < 0) /* KEYCODE_xxx */ 1707 if ((int32_t)ic < 0) /* KEYCODE_xxx */
1707 return ic; 1708 break;
1708 // TODO: imagine sequence like: 0xff, <left-arrow>: we are currently losing 0xff... 1709 // TODO: imagine sequence like: 0xff,<left-arrow>: we are currently losing 0xff...
1709 1710
1710 unicode_buf[unicode_idx++] = ic; 1711 unicode_buf[unicode_idx++] = ic;
1711 unicode_buf[unicode_idx] = '\0'; 1712 unicode_buf[unicode_idx] = '\0';
1712 if (mbstowcs(&wc, unicode_buf, 1) != 1) { 1713 if (mbstowcs(&wc, unicode_buf, 1) != 1) {
1713 /* Not (yet?) a valid unicode char */ 1714 /* Not (yet?) a valid unicode char */
1714 if (unicode_idx < MB_CUR_MAX) { 1715 if (unicode_idx < MB_CUR_MAX) {
1715 delay = 50; 1716 timeout = 50;
1716 goto poll_again; 1717 continue;
1717 } 1718 }
1719 pushback:
1718 /* Invalid sequence. Save all "bad bytes" except first */ 1720 /* Invalid sequence. Save all "bad bytes" except first */
1719 read_key_ungets(read_key_buffer, unicode_buf + 1, MB_CUR_MAX - 1); 1721 read_key_ungets(read_key_buffer, unicode_buf + 1, unicode_idx - 1);
1720 /* 1722 /*
1721 * ic = unicode_buf[0] sounds even better, but currently 1723 * ic = unicode_buf[0] sounds even better, but currently
1722 * this does not work: wchar_t[] -> char[] conversion 1724 * this does not work: wchar_t[] -> char[] conversion
@@ -1730,7 +1732,8 @@ static int lineedit_read_key(char *read_key_buffer)
1730 } 1732 }
1731 } 1733 }
1732#endif 1734#endif
1733 } while (errno == EAGAIN); 1735 break;
1736 }
1734 1737
1735 return ic; 1738 return ic;
1736} 1739}