123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439 |
- /*
- * Backend to run a Windows console session using ConPTY.
- */
- #include <stdio.h>
- #include <stdlib.h>
- #include <limits.h>
- #include "putty.h"
- #include <windows.h>
- #include <consoleapi.h>
- typedef struct ConPTY ConPTY;
- struct ConPTY {
- HPCON pseudoconsole;
- HANDLE outpipe, inpipe, hprocess;
- struct handle *out, *in;
- HandleWait *subprocess;
- bool exited;
- DWORD exitstatus;
- Seat *seat;
- LogContext *logctx;
- int bufsize;
- Backend backend;
- };
- DECL_WINDOWS_FUNCTION(static, HRESULT, CreatePseudoConsole,
- (COORD, HANDLE, HANDLE, DWORD, HPCON *));
- DECL_WINDOWS_FUNCTION(static, void, ClosePseudoConsole, (HPCON));
- DECL_WINDOWS_FUNCTION(static, HRESULT, ResizePseudoConsole, (HPCON, COORD));
- static bool init_conpty_api(void)
- {
- static bool tried = false;
- if (!tried) {
- tried = true;
- HMODULE kernel32_module = load_system32_dll("kernel32.dll");
- GET_WINDOWS_FUNCTION(kernel32_module, CreatePseudoConsole);
- GET_WINDOWS_FUNCTION(kernel32_module, ClosePseudoConsole);
- GET_WINDOWS_FUNCTION(kernel32_module, ResizePseudoConsole);
- }
- return (p_CreatePseudoConsole != NULL &&
- p_ClosePseudoConsole != NULL &&
- p_ResizePseudoConsole != NULL);
- }
- static void conpty_terminate(ConPTY *conpty)
- {
- if (conpty->out) {
- handle_free(conpty->out);
- conpty->out = NULL;
- }
- if (conpty->outpipe != INVALID_HANDLE_VALUE) {
- CloseHandle(conpty->outpipe);
- conpty->outpipe = INVALID_HANDLE_VALUE;
- }
- if (conpty->in) {
- handle_free(conpty->in);
- conpty->in = NULL;
- }
- if (conpty->inpipe != INVALID_HANDLE_VALUE) {
- CloseHandle(conpty->inpipe);
- conpty->inpipe = INVALID_HANDLE_VALUE;
- }
- if (conpty->subprocess) {
- delete_handle_wait(conpty->subprocess);
- conpty->subprocess = NULL;
- conpty->hprocess = INVALID_HANDLE_VALUE;
- }
- if (conpty->pseudoconsole != INVALID_HANDLE_VALUE) {
- p_ClosePseudoConsole(conpty->pseudoconsole);
- conpty->pseudoconsole = INVALID_HANDLE_VALUE;
- }
- }
- static void conpty_process_wait_callback(void *vctx)
- {
- ConPTY *conpty = (ConPTY *)vctx;
- if (!GetExitCodeProcess(conpty->hprocess, &conpty->exitstatus))
- return;
- conpty->exited = true;
- /*
- * We can stop waiting for the process now.
- */
- if (conpty->subprocess) {
- delete_handle_wait(conpty->subprocess);
- conpty->subprocess = NULL;
- conpty->hprocess = INVALID_HANDLE_VALUE;
- }
- /*
- * Once the contained process exits, close the pseudo-console as
- * well. But don't close the pipes yet, since apparently
- * ClosePseudoConsole can trigger a final bout of terminal output
- * as things clean themselves up.
- */
- if (conpty->pseudoconsole != INVALID_HANDLE_VALUE) {
- p_ClosePseudoConsole(conpty->pseudoconsole);
- conpty->pseudoconsole = INVALID_HANDLE_VALUE;
- }
- }
- static size_t conpty_gotdata(
- struct handle *h, const void *data, size_t len, int err)
- {
- ConPTY *conpty = (ConPTY *)handle_get_privdata(h);
- if (err || len == 0) {
- char *error_msg;
- conpty_terminate(conpty);
- seat_notify_remote_exit(conpty->seat);
- if (!err && conpty->exited) {
- /*
- * The clean-exit case: our subprocess terminated, we
- * deleted the PseudoConsole ourself, and now we got the
- * expected EOF on the pipe.
- */
- return 0;
- }
- if (err)
- error_msg = dupprintf("Error reading from console pty: %s",
- win_strerror(err));
- else
- error_msg = dupprintf(
- "Unexpected end of file reading from console pty");
- logevent(conpty->logctx, error_msg);
- seat_connection_fatal(conpty->seat, "%s", error_msg);
- sfree(error_msg);
- return 0;
- } else {
- return seat_stdout(conpty->seat, data, len);
- }
- }
- static void conpty_sentdata(struct handle *h, size_t new_backlog, int err,
- bool close)
- {
- ConPTY *conpty = (ConPTY *)handle_get_privdata(h);
- if (err) {
- const char *error_msg = "Error writing to conpty device";
- conpty_terminate(conpty);
- seat_notify_remote_exit(conpty->seat);
- logevent(conpty->logctx, error_msg);
- seat_connection_fatal(conpty->seat, "%s", error_msg);
- } else {
- conpty->bufsize = new_backlog;
- }
- }
- static char *conpty_init(const BackendVtable *vt, Seat *seat,
- Backend **backend_handle, LogContext *logctx,
- Conf *conf, const char *host, int port,
- char **realhost, bool nodelay, bool keepalive)
- {
- ConPTY *conpty;
- char *err = NULL;
- HANDLE in_r = INVALID_HANDLE_VALUE;
- HANDLE in_w = INVALID_HANDLE_VALUE;
- HANDLE out_r = INVALID_HANDLE_VALUE;
- HANDLE out_w = INVALID_HANDLE_VALUE;
- HPCON pcon;
- bool pcon_needs_cleanup = false;
- STARTUPINFOEXW si;
- memset(&si, 0, sizeof(si));
- if (!init_conpty_api()) {
- err = dupprintf("Pseudo-console API is not available on this "
- "Windows system");
- goto out;
- }
- if (!CreatePipe(&in_r, &in_w, NULL, 0)) {
- err = dupprintf("CreatePipe: %s", win_strerror(GetLastError()));
- goto out;
- }
- if (!CreatePipe(&out_r, &out_w, NULL, 0)) {
- err = dupprintf("CreatePipe: %s", win_strerror(GetLastError()));
- goto out;
- }
- COORD size;
- size.X = conf_get_int(conf, CONF_width);
- size.Y = conf_get_int(conf, CONF_height);
- HRESULT result = p_CreatePseudoConsole(size, in_r, out_w, 0, &pcon);
- if (FAILED(result)) {
- if (HRESULT_FACILITY(result) == FACILITY_WIN32)
- err = dupprintf("CreatePseudoConsole: %s",
- win_strerror(HRESULT_CODE(result)));
- else
- err = dupprintf("CreatePseudoConsole failed: HRESULT=0x%08x",
- (unsigned)result);
- goto out;
- }
- pcon_needs_cleanup = true;
- CloseHandle(in_r);
- in_r = INVALID_HANDLE_VALUE;
- CloseHandle(out_w);
- out_w = INVALID_HANDLE_VALUE;
- si.StartupInfo.cb = sizeof(si);
- SIZE_T attrsize = 0;
- InitializeProcThreadAttributeList(NULL, 1, 0, &attrsize);
- si.lpAttributeList = smalloc(attrsize);
- if (!InitializeProcThreadAttributeList(
- si.lpAttributeList, 1, 0, &attrsize)) {
- err = dupprintf("InitializeProcThreadAttributeList: %s",
- win_strerror(GetLastError()));
- goto out;
- }
- if (!UpdateProcThreadAttribute(
- si.lpAttributeList,
- 0, PROC_THREAD_ATTRIBUTE_PSEUDOCONSOLE,
- pcon, sizeof(pcon), NULL, NULL)) {
- err = dupprintf("UpdateProcThreadAttribute: %s",
- win_strerror(GetLastError()));
- goto out;
- }
- PROCESS_INFORMATION pi;
- memset(&pi, 0, sizeof(pi));
- wchar_t *command;
- {
- bool utf8;
- const char *conf_cmd = conf_get_str_ambi(conf, CONF_remote_cmd, &utf8);
- if (*conf_cmd) {
- command = dup_mb_to_wc(utf8 ? CP_UTF8 : CP_ACP, conf_cmd);
- } else {
- char *cmd = dupcat(get_system_dir(), "\\cmd.exe");
- command = dup_mb_to_wc(CP_ACP, cmd);
- sfree(cmd);
- }
- }
- bool created_ok = CreateProcessW(NULL, command, NULL, NULL,
- false, EXTENDED_STARTUPINFO_PRESENT,
- NULL, NULL, &si.StartupInfo, &pi);
- sfree(command);
- if (!created_ok) {
- err = dupprintf("CreateProcess: %s",
- win_strerror(GetLastError()));
- goto out;
- }
- /* No local authentication phase in this protocol */
- seat_set_trust_status(seat, false);
- conpty = snew(ConPTY);
- memset(conpty, 0, sizeof(ConPTY));
- conpty->pseudoconsole = pcon;
- pcon_needs_cleanup = false;
- conpty->outpipe = in_w;
- conpty->out = handle_output_new(in_w, conpty_sentdata, conpty, 0);
- in_w = INVALID_HANDLE_VALUE;
- conpty->inpipe = out_r;
- conpty->in = handle_input_new(out_r, conpty_gotdata, conpty, 0);
- out_r = INVALID_HANDLE_VALUE;
- conpty->subprocess = add_handle_wait(
- pi.hProcess, conpty_process_wait_callback, conpty);
- conpty->hprocess = pi.hProcess;
- CloseHandle(pi.hThread);
- conpty->exited = false;
- conpty->exitstatus = 0;
- conpty->bufsize = 0;
- conpty->backend.vt = vt;
- *backend_handle = &conpty->backend;
- conpty->seat = seat;
- conpty->logctx = logctx;
- *realhost = dupstr("");
- /*
- * Specials are always available.
- */
- seat_update_specials_menu(conpty->seat);
- out:
- if (in_r != INVALID_HANDLE_VALUE)
- CloseHandle(in_r);
- if (in_w != INVALID_HANDLE_VALUE)
- CloseHandle(in_w);
- if (out_r != INVALID_HANDLE_VALUE)
- CloseHandle(out_r);
- if (out_w != INVALID_HANDLE_VALUE)
- CloseHandle(out_w);
- if (pcon_needs_cleanup)
- p_ClosePseudoConsole(pcon);
- sfree(si.lpAttributeList);
- return err;
- }
- static void conpty_free(Backend *be)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- conpty_terminate(conpty);
- expire_timer_context(conpty);
- sfree(conpty);
- }
- static void conpty_reconfig(Backend *be, Conf *conf)
- {
- }
- static void conpty_send(Backend *be, const char *buf, size_t len)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- if (conpty->out == NULL)
- return;
- conpty->bufsize = handle_write(conpty->out, buf, len);
- }
- static size_t conpty_sendbuffer(Backend *be)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- return conpty->bufsize;
- }
- static void conpty_size(Backend *be, int width, int height)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- COORD size;
- size.X = width;
- size.Y = height;
- p_ResizePseudoConsole(conpty->pseudoconsole, size);
- }
- static void conpty_special(Backend *be, SessionSpecialCode code, int arg)
- {
- }
- static const SessionSpecial *conpty_get_specials(Backend *be)
- {
- static const SessionSpecial specials[] = {
- {NULL, SS_EXITMENU}
- };
- return specials;
- }
- static bool conpty_connected(Backend *be)
- {
- return true; /* always connected */
- }
- static bool conpty_sendok(Backend *be)
- {
- return true;
- }
- static void conpty_unthrottle(Backend *be, size_t backlog)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- if (conpty->in)
- handle_unthrottle(conpty->in, backlog);
- }
- static bool conpty_ldisc(Backend *be, int option)
- {
- return false;
- }
- static void conpty_provide_ldisc(Backend *be, Ldisc *ldisc)
- {
- }
- static int conpty_exitcode(Backend *be)
- {
- ConPTY *conpty = container_of(be, ConPTY, backend);
- if (conpty->exited) {
- /*
- * PuTTY's representation of exit statuses expects them to be
- * non-negative 'int' values. But Windows exit statuses can
- * include all those exception codes like 0xC000001D which
- * convert to negative 32-bit ints.
- *
- * I don't think there's a great deal of use for returning
- * those in full detail, right now. (Though if we ever
- * connected this system up to a Windows version of psusan or
- * Uppity, perhaps there might be?)
- *
- * So we clip them at INT_MAX-1, since INT_MAX is reserved for
- * 'exit so unclean as to inhibit Close On Clean Exit'.
- */
- return (0 <= conpty->exitstatus && conpty->exitstatus < INT_MAX) ?
- conpty->exitstatus : INT_MAX-1;
- } else {
- return -1;
- }
- }
- static int conpty_cfg_info(Backend *be)
- {
- return 0;
- }
- const BackendVtable conpty_backend = {
- .init = conpty_init,
- .free = conpty_free,
- .reconfig = conpty_reconfig,
- .send = conpty_send,
- .sendbuffer = conpty_sendbuffer,
- .size = conpty_size,
- .special = conpty_special,
- .get_specials = conpty_get_specials,
- .connected = conpty_connected,
- .exitcode = conpty_exitcode,
- .sendok = conpty_sendok,
- .ldisc_option_state = conpty_ldisc,
- .provide_ldisc = conpty_provide_ldisc,
- .unthrottle = conpty_unthrottle,
- .cfg_info = conpty_cfg_info,
- .id = "conpty",
- .displayname_tc = "ConPTY",
- .displayname_lc = "ConPTY", /* proper name, so capitalise it anyway */
- .protocol = -1,
- };
|