ldisc.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581
  1. /*
  2. * ldisc.c: PuTTY line discipline. Sits between the input coming
  3. * from keypresses in the window, and the output channel leading to
  4. * the back end. Implements echo and/or local line editing,
  5. * depending on what's currently configured.
  6. */
  7. #include <stdio.h>
  8. #include <ctype.h>
  9. #include <assert.h>
  10. #include "putty.h"
  11. #include "terminal.h"
  12. struct Ldisc_tag {
  13. Terminal *term;
  14. Backend *backend;
  15. Seat *seat;
  16. /*
  17. * When the backend is not reporting true from sendok(), terminal
  18. * input that comes here is stored in this bufchain instead. When
  19. * the backend later decides it wants session input, we empty the
  20. * queue in ldisc_check_sendok_callback(), passing its contents on
  21. * to the backend. Before then, we also provide data from this
  22. * queue to term_get_userpass_input() via ldisc_get_input_token(),
  23. * to be interpreted as user responses to username and password
  24. * prompts during authentication.
  25. *
  26. * Unfortunately, the data stored in this queue is not all of the
  27. * same type: our output to the backend consists of both raw bytes
  28. * sent to backend_send(), and also session specials such as
  29. * SS_EOL and SS_EC. So we have to encode our queued data in a way
  30. * that can represent both.
  31. *
  32. * The encoding is private to this source file, so we can change
  33. * it if necessary and only have to worry about the encode and
  34. * decode functions here. Currently, it is:
  35. *
  36. * - Bytes other than 0xFF are stored literally.
  37. * - The byte 0xFF itself is stored as 0xFF 0xFF.
  38. * - A session special (code, arg) is stored as 0xFF, followed by
  39. * a big-endian 4-byte integer containing code, followed by
  40. * another big-endian 4-byte integer containing arg.
  41. *
  42. * (This representation relies on session special codes being at
  43. * most 0xFEFFFFFF when represented in 32 bits, so that the first
  44. * byte of the 'code' integer can't be confused with the 0xFF
  45. * followup byte indicating a literal 0xFF, But since session
  46. * special codes are defined by an enum counting up from zero, and
  47. * there are only a couple of dozen of them, that shouldn't be a
  48. * problem! Even so, just in case, an assertion checks that at
  49. * encode time.)
  50. */
  51. bufchain input_queue;
  52. IdempotentCallback input_queue_callback;
  53. prompts_t *prompts;
  54. /*
  55. * Values cached out of conf.
  56. */
  57. bool telnet_keyboard, telnet_newline;
  58. int protocol, localecho, localedit;
  59. char *buf;
  60. size_t buflen, bufsiz;
  61. bool quotenext;
  62. };
  63. #define ECHOING (ldisc->localecho == FORCE_ON || \
  64. (ldisc->localecho == AUTO && \
  65. (backend_ldisc_option_state(ldisc->backend, LD_ECHO))))
  66. #define EDITING (ldisc->localedit == FORCE_ON || \
  67. (ldisc->localedit == AUTO && \
  68. (backend_ldisc_option_state(ldisc->backend, LD_EDIT))))
  69. static void c_write(Ldisc *ldisc, const void *buf, int len)
  70. {
  71. seat_stdout(ldisc->seat, buf, len);
  72. }
  73. static int plen(Ldisc *ldisc, unsigned char c)
  74. {
  75. if ((c >= 32 && c <= 126) || (c >= 160 && !in_utf(ldisc->term)))
  76. return 1;
  77. else if (c < 128)
  78. return 2; /* ^x for some x */
  79. else if (in_utf(ldisc->term) && c >= 0xC0)
  80. return 1; /* UTF-8 introducer character
  81. * (FIXME: combining / wide chars) */
  82. else if (in_utf(ldisc->term) && c >= 0x80 && c < 0xC0)
  83. return 0; /* UTF-8 followup character */
  84. else
  85. return 4; /* <XY> hex representation */
  86. }
  87. static void pwrite(Ldisc *ldisc, unsigned char c)
  88. {
  89. if ((c >= 32 && c <= 126) ||
  90. (!in_utf(ldisc->term) && c >= 0xA0) ||
  91. (in_utf(ldisc->term) && c >= 0x80)) {
  92. c_write(ldisc, &c, 1);
  93. } else if (c < 128) {
  94. char cc[2];
  95. cc[1] = (c == 127 ? '?' : c + 0x40);
  96. cc[0] = '^';
  97. c_write(ldisc, cc, 2);
  98. } else {
  99. char cc[5];
  100. sprintf(cc, "<%02X>", c);
  101. c_write(ldisc, cc, 4);
  102. }
  103. }
  104. static bool char_start(Ldisc *ldisc, unsigned char c)
  105. {
  106. if (in_utf(ldisc->term))
  107. return (c < 0x80 || c >= 0xC0);
  108. else
  109. return true;
  110. }
  111. static void bsb(Ldisc *ldisc, int n)
  112. {
  113. while (n--)
  114. c_write(ldisc, "\010 \010", 3);
  115. }
  116. static void ldisc_input_queue_callback(void *ctx);
  117. #define CTRL(x) (x^'@')
  118. #define KCTRL(x) ((x^'@') | 0x100)
  119. Ldisc *ldisc_create(Conf *conf, Terminal *term, Backend *backend, Seat *seat)
  120. {
  121. Ldisc *ldisc = snew(Ldisc);
  122. ldisc->buf = NULL;
  123. ldisc->buflen = 0;
  124. ldisc->bufsiz = 0;
  125. ldisc->quotenext = false;
  126. ldisc->backend = backend;
  127. ldisc->term = term;
  128. ldisc->seat = seat;
  129. bufchain_init(&ldisc->input_queue);
  130. ldisc->prompts = NULL;
  131. ldisc->input_queue_callback.fn = ldisc_input_queue_callback;
  132. ldisc->input_queue_callback.ctx = ldisc;
  133. ldisc->input_queue_callback.queued = false;
  134. bufchain_set_callback(&ldisc->input_queue, &ldisc->input_queue_callback);
  135. ldisc_configure(ldisc, conf);
  136. /* Link ourselves into the backend and the terminal */
  137. if (term)
  138. term->ldisc = ldisc;
  139. if (backend)
  140. backend_provide_ldisc(backend, ldisc);
  141. return ldisc;
  142. }
  143. void ldisc_configure(Ldisc *ldisc, Conf *conf)
  144. {
  145. ldisc->telnet_keyboard = conf_get_bool(conf, CONF_telnet_keyboard);
  146. ldisc->telnet_newline = conf_get_bool(conf, CONF_telnet_newline);
  147. ldisc->protocol = conf_get_int(conf, CONF_protocol);
  148. ldisc->localecho = conf_get_int(conf, CONF_localecho);
  149. ldisc->localedit = conf_get_int(conf, CONF_localedit);
  150. }
  151. void ldisc_free(Ldisc *ldisc)
  152. {
  153. bufchain_clear(&ldisc->input_queue);
  154. if (ldisc->term)
  155. ldisc->term->ldisc = NULL;
  156. if (ldisc->backend)
  157. backend_provide_ldisc(ldisc->backend, NULL);
  158. if (ldisc->buf)
  159. sfree(ldisc->buf);
  160. if (ldisc->prompts && ldisc->prompts->ldisc_ptr_to_us == &ldisc->prompts)
  161. ldisc->prompts->ldisc_ptr_to_us = NULL;
  162. delete_callbacks_for_context(ldisc);
  163. sfree(ldisc);
  164. }
  165. void ldisc_echoedit_update(Ldisc *ldisc)
  166. {
  167. seat_echoedit_update(ldisc->seat, ECHOING, EDITING);
  168. }
  169. void ldisc_enable_prompt_callback(Ldisc *ldisc, prompts_t *prompts)
  170. {
  171. /*
  172. * Called by the terminal to indicate that there's a prompts_t
  173. * currently in flight, or to indicate that one has just finished
  174. * (by passing NULL). When ldisc->prompts is not null, we notify
  175. * the terminal whenever new data arrives in our input queue, so
  176. * that it can continue the interactive prompting process.
  177. */
  178. ldisc->prompts = prompts;
  179. if (prompts)
  180. ldisc->prompts->ldisc_ptr_to_us = &ldisc->prompts;
  181. }
  182. static void ldisc_input_queue_callback(void *ctx)
  183. {
  184. /*
  185. * Toplevel callback that is triggered whenever the input queue
  186. * lengthens. If we're currently processing an interactive prompt,
  187. * we call back the Terminal to tell it to do some more stuff with
  188. * that prompt based on the new input.
  189. */
  190. Ldisc *ldisc = (Ldisc *)ctx;
  191. if (ldisc->term && ldisc->prompts) {
  192. /*
  193. * The integer return value from this call is discarded,
  194. * because we have no channel to pass it on to the backend
  195. * that originally wanted it. But that's OK, because if the
  196. * return value is >= 0 (that is, the prompts are either
  197. * completely filled in, or aborted by the user), then the
  198. * terminal will notify the callback in the prompts_t, and
  199. * when that calls term_get_userpass_input again, it will
  200. * return the same answer again.
  201. */
  202. term_get_userpass_input(ldisc->term, ldisc->prompts);
  203. }
  204. }
  205. static void ldisc_to_backend_raw(
  206. Ldisc *ldisc, const void *vbuf, size_t len)
  207. {
  208. if (backend_sendok(ldisc->backend)) {
  209. backend_send(ldisc->backend, vbuf, len);
  210. } else {
  211. const char *buf = (const char *)vbuf;
  212. while (len > 0) {
  213. /*
  214. * Encode raw data in input_queue, by storing large chunks
  215. * as long as they don't include 0xFF, and pausing every
  216. * time they do to escape it.
  217. */
  218. const char *ff = memchr(buf, '\xFF', len);
  219. size_t this_len = ff ? ff - buf : len;
  220. if (this_len > 0) {
  221. bufchain_add(&ldisc->input_queue, buf, len);
  222. } else {
  223. bufchain_add(&ldisc->input_queue, "\xFF\xFF", 2);
  224. this_len = 1;
  225. }
  226. buf += this_len;
  227. len -= this_len;
  228. }
  229. }
  230. }
  231. static void ldisc_to_backend_special(
  232. Ldisc *ldisc, SessionSpecialCode code, int arg)
  233. {
  234. if (backend_sendok(ldisc->backend)) {
  235. backend_special(ldisc->backend, code, arg);
  236. } else {
  237. /*
  238. * Encode a session special in input_queue.
  239. */
  240. unsigned char data[9];
  241. data[0] = 0xFF;
  242. PUT_32BIT_MSB_FIRST(data+1, code);
  243. PUT_32BIT_MSB_FIRST(data+5, arg);
  244. assert(data[1] != 0xFF &&
  245. "SessionSpecialCode encoding collides with FF FF escape");
  246. bufchain_add(&ldisc->input_queue, data, 9);
  247. }
  248. }
  249. bool ldisc_has_input_buffered(Ldisc *ldisc)
  250. {
  251. return bufchain_size(&ldisc->input_queue) > 0;
  252. }
  253. LdiscInputToken ldisc_get_input_token(Ldisc *ldisc)
  254. {
  255. assert(bufchain_size(&ldisc->input_queue) > 0 &&
  256. "You're not supposed to call this unless there is buffered input!");
  257. LdiscInputToken tok;
  258. char c;
  259. bufchain_fetch_consume(&ldisc->input_queue, &c, 1);
  260. if (c != '\xFF') {
  261. /* A literal non-FF byte */
  262. tok.is_special = false;
  263. tok.chr = c;
  264. return tok;
  265. } else {
  266. char data[8];
  267. /* See if the byte after the FF is also FF, indicating a literal FF */
  268. bufchain_fetch_consume(&ldisc->input_queue, data, 1);
  269. if (data[0] == '\xFF') {
  270. tok.is_special = false;
  271. tok.chr = '\xFF';
  272. return tok;
  273. }
  274. /* If not, get the rest of an 8-byte chunk and decode a special */
  275. bufchain_fetch_consume(&ldisc->input_queue, data+1, 7);
  276. tok.is_special = true;
  277. tok.code = GET_32BIT_MSB_FIRST(data);
  278. tok.arg = toint(GET_32BIT_MSB_FIRST(data+4));
  279. return tok;
  280. }
  281. }
  282. static void ldisc_check_sendok_callback(void *ctx)
  283. {
  284. Ldisc *ldisc = (Ldisc *)ctx;
  285. if (!(ldisc->backend && backend_sendok(ldisc->backend)))
  286. return;
  287. /*
  288. * Flush the ldisc input queue into the backend, which is now
  289. * willing to receive the data.
  290. */
  291. while (bufchain_size(&ldisc->input_queue) > 0) {
  292. /*
  293. * Process either a chunk of non-special data, or an FF
  294. * escape, depending on whether the first thing we see is an
  295. * FF byte.
  296. */
  297. ptrlen data = bufchain_prefix(&ldisc->input_queue);
  298. const char *ff = memchr(data.ptr, '\xFF', data.len);
  299. if (ff != data.ptr) {
  300. /* Send a maximal block of data not containing any
  301. * difficult bytes. */
  302. if (ff)
  303. data.len = ff - (const char *)data.ptr;
  304. backend_send(ldisc->backend, data.ptr, data.len);
  305. bufchain_consume(&ldisc->input_queue, data.len);
  306. } else {
  307. /* Decode either a special or an escaped FF byte. The
  308. * easiest way to do this is to reuse the decoding code
  309. * already in ldisc_get_input_token. */
  310. LdiscInputToken tok = ldisc_get_input_token(ldisc);
  311. if (tok.is_special)
  312. backend_special(ldisc->backend, tok.code, tok.arg);
  313. else
  314. backend_send(ldisc->backend, &tok.chr, 1);
  315. }
  316. }
  317. }
  318. void ldisc_check_sendok(Ldisc *ldisc)
  319. {
  320. queue_toplevel_callback(ldisc_check_sendok_callback, ldisc);
  321. }
  322. void ldisc_send(Ldisc *ldisc, const void *vbuf, int len, bool interactive)
  323. {
  324. const char *buf = (const char *)vbuf;
  325. int keyflag = 0;
  326. assert(ldisc->term);
  327. if (interactive) {
  328. /*
  329. * Interrupt a paste from the clipboard, if one was in
  330. * progress when the user pressed a key. This is easier than
  331. * buffering the current piece of data and saving it until the
  332. * terminal has finished pasting, and has the potential side
  333. * benefit of permitting a user to cancel an accidental huge
  334. * paste.
  335. */
  336. term_nopaste(ldisc->term);
  337. }
  338. /*
  339. * Less than zero means null terminated special string.
  340. */
  341. if (len < 0) {
  342. len = strlen(buf);
  343. keyflag = KCTRL('@');
  344. }
  345. /*
  346. * Either perform local editing, or just send characters.
  347. */
  348. if (EDITING) {
  349. while (len--) {
  350. int c;
  351. c = (unsigned char)(*buf++) + keyflag;
  352. if (!interactive && c == '\r')
  353. c += KCTRL('@');
  354. switch (ldisc->quotenext ? ' ' : c) {
  355. /*
  356. * ^h/^?: delete, and output BSBs, to return to
  357. * last character boundary (in UTF-8 mode this may
  358. * be more than one byte)
  359. * ^w: delete, and output BSBs, to return to last
  360. * space/nonspace boundary
  361. * ^u: delete, and output BSBs, to return to BOL
  362. * ^c: Do a ^u then send a telnet IP
  363. * ^z: Do a ^u then send a telnet SUSP
  364. * ^\: Do a ^u then send a telnet ABORT
  365. * ^r: echo "^R\n" and redraw line
  366. * ^v: quote next char
  367. * ^d: if at BOL, end of file and close connection,
  368. * else send line and reset to BOL
  369. * ^m: send line-plus-\r\n and reset to BOL
  370. */
  371. case KCTRL('H'):
  372. case KCTRL('?'): /* backspace/delete */
  373. if (ldisc->buflen > 0) {
  374. do {
  375. if (ECHOING)
  376. bsb(ldisc, plen(ldisc, ldisc->buf[ldisc->buflen - 1]));
  377. ldisc->buflen--;
  378. } while (!char_start(ldisc, ldisc->buf[ldisc->buflen]));
  379. }
  380. break;
  381. case CTRL('W'): /* delete word */
  382. while (ldisc->buflen > 0) {
  383. if (ECHOING)
  384. bsb(ldisc, plen(ldisc, ldisc->buf[ldisc->buflen - 1]));
  385. ldisc->buflen--;
  386. if (ldisc->buflen > 0 &&
  387. isspace((unsigned char)ldisc->buf[ldisc->buflen-1]) &&
  388. !isspace((unsigned char)ldisc->buf[ldisc->buflen]))
  389. break;
  390. }
  391. break;
  392. case CTRL('U'): /* delete line */
  393. case CTRL('C'): /* Send IP */
  394. case CTRL('\\'): /* Quit */
  395. case CTRL('Z'): /* Suspend */
  396. while (ldisc->buflen > 0) {
  397. if (ECHOING)
  398. bsb(ldisc, plen(ldisc, ldisc->buf[ldisc->buflen - 1]));
  399. ldisc->buflen--;
  400. }
  401. if (c == CTRL('U'))
  402. break; /* ^U *just* erases a line */
  403. ldisc_to_backend_special(ldisc, SS_EL, 0);
  404. /*
  405. * We don't send IP, SUSP or ABORT if the user has
  406. * configured telnet specials off! This breaks
  407. * talkers otherwise.
  408. */
  409. if (!ldisc->telnet_keyboard)
  410. goto default_case;
  411. if (c == CTRL('C'))
  412. ldisc_to_backend_special(ldisc, SS_IP, 0);
  413. if (c == CTRL('Z'))
  414. ldisc_to_backend_special(ldisc, SS_SUSP, 0);
  415. if (c == CTRL('\\'))
  416. ldisc_to_backend_special(ldisc, SS_ABORT, 0);
  417. break;
  418. case CTRL('R'): /* redraw line */
  419. if (ECHOING) {
  420. int i;
  421. c_write(ldisc, "^R\r\n", 4);
  422. for (i = 0; i < ldisc->buflen; i++)
  423. pwrite(ldisc, ldisc->buf[i]);
  424. }
  425. break;
  426. case CTRL('V'): /* quote next char */
  427. ldisc->quotenext = true;
  428. break;
  429. case CTRL('D'): /* logout or send */
  430. if (ldisc->buflen == 0) {
  431. ldisc_to_backend_special(ldisc, SS_EOF, 0);
  432. } else {
  433. ldisc_to_backend_raw(ldisc, ldisc->buf, ldisc->buflen);
  434. ldisc->buflen = 0;
  435. }
  436. break;
  437. /*
  438. * This particularly hideous bit of code from RDB
  439. * allows ordinary ^M^J to do the same thing as
  440. * magic-^M when in Raw protocol. The line `case
  441. * KCTRL('M'):' is _inside_ the if block. Thus:
  442. *
  443. * - receiving regular ^M goes straight to the
  444. * default clause and inserts as a literal ^M.
  445. * - receiving regular ^J _not_ directly after a
  446. * literal ^M (or not in Raw protocol) fails the
  447. * if condition, leaps to the bottom of the if,
  448. * and falls through into the default clause
  449. * again.
  450. * - receiving regular ^J just after a literal ^M
  451. * in Raw protocol passes the if condition,
  452. * deletes the literal ^M, and falls through
  453. * into the magic-^M code
  454. * - receiving a magic-^M empties the line buffer,
  455. * signals end-of-line in one of the various
  456. * entertaining ways, and _doesn't_ fall out of
  457. * the bottom of the if and through to the
  458. * default clause because of the break.
  459. */
  460. case CTRL('J'):
  461. if (ldisc->protocol == PROT_RAW &&
  462. ldisc->buflen > 0 && ldisc->buf[ldisc->buflen - 1] == '\r') {
  463. if (ECHOING)
  464. bsb(ldisc, plen(ldisc, ldisc->buf[ldisc->buflen - 1]));
  465. ldisc->buflen--;
  466. /* FALLTHROUGH */
  467. case KCTRL('M'): /* send with newline */
  468. if (ldisc->buflen > 0)
  469. ldisc_to_backend_raw(ldisc, ldisc->buf, ldisc->buflen);
  470. if (ldisc->protocol == PROT_RAW)
  471. ldisc_to_backend_raw(ldisc, "\r\n", 2);
  472. else if (ldisc->protocol == PROT_TELNET && ldisc->telnet_newline)
  473. ldisc_to_backend_special(ldisc, SS_EOL, 0);
  474. else
  475. ldisc_to_backend_raw(ldisc, "\r", 1);
  476. if (ECHOING)
  477. c_write(ldisc, "\r\n", 2);
  478. ldisc->buflen = 0;
  479. break;
  480. }
  481. /* FALLTHROUGH */
  482. default: /* get to this label from ^V handler */
  483. default_case:
  484. sgrowarray(ldisc->buf, ldisc->bufsiz, ldisc->buflen);
  485. ldisc->buf[ldisc->buflen++] = c;
  486. if (ECHOING)
  487. pwrite(ldisc, (unsigned char) c);
  488. ldisc->quotenext = false;
  489. break;
  490. }
  491. }
  492. } else {
  493. if (ldisc->buflen != 0) {
  494. ldisc_to_backend_raw(ldisc, ldisc->buf, ldisc->buflen);
  495. while (ldisc->buflen > 0) {
  496. bsb(ldisc, plen(ldisc, ldisc->buf[ldisc->buflen - 1]));
  497. ldisc->buflen--;
  498. }
  499. }
  500. if (len > 0) {
  501. if (ECHOING)
  502. c_write(ldisc, buf, len);
  503. if (keyflag && ldisc->protocol == PROT_TELNET && len == 1) {
  504. switch (buf[0]) {
  505. case CTRL('M'):
  506. if (ldisc->protocol == PROT_TELNET && ldisc->telnet_newline)
  507. ldisc_to_backend_special(ldisc, SS_EOL, 0);
  508. else
  509. ldisc_to_backend_raw(ldisc, "\r", 1);
  510. break;
  511. case CTRL('?'):
  512. case CTRL('H'):
  513. if (ldisc->telnet_keyboard) {
  514. ldisc_to_backend_special(ldisc, SS_EC, 0);
  515. break;
  516. }
  517. case CTRL('C'):
  518. if (ldisc->telnet_keyboard) {
  519. ldisc_to_backend_special(ldisc, SS_IP, 0);
  520. break;
  521. }
  522. case CTRL('Z'):
  523. if (ldisc->telnet_keyboard) {
  524. ldisc_to_backend_special(ldisc, SS_SUSP, 0);
  525. break;
  526. }
  527. default:
  528. ldisc_to_backend_raw(ldisc, buf, len);
  529. break;
  530. }
  531. } else
  532. ldisc_to_backend_raw(ldisc, buf, len);
  533. }
  534. }
  535. }