diff options
Diffstat (limited to 'x11vnc/xevents.c')
-rw-r--r-- | x11vnc/xevents.c | 132 |
1 files changed, 126 insertions, 6 deletions
diff --git a/x11vnc/xevents.c b/x11vnc/xevents.c index b4b7f7d..32bea29 100644 --- a/x11vnc/xevents.c +++ b/x11vnc/xevents.c @@ -1494,27 +1494,137 @@ void set_server_input(rfbClientPtr cl, int grab) { #endif } +static int wsock_timeout_sock = -1; + +static void wsock_timeout (int sig) { + rfbLog("sig: %d, wsock_timeout.\n", sig); + if (wsock_timeout_sock >= 0) { + close(wsock_timeout_sock); + wsock_timeout_sock = -1; + } +} + +static void try_local_chat_window(void) { + int i, port, lsock; + char cmd[100]; + struct sockaddr_in addr; +#ifdef __hpux + int addrlen = sizeof(addr); +#else + socklen_t addrlen = sizeof(addr); +#endif + + for (i = 0; i < 90; i++) { + /* find an open port */ + port = 7300 + i; + lsock = rfbListenOnTCPPort(port, htonl(INADDR_LOOPBACK)); + if (lsock >= 0) { + break; + } + port = 0; + } + + if (port == 0) { + return; + } + + /* have ssvncvncviewer connect back to us (n.b. sockpair fails) */ + + sprintf(cmd, "ssvnc -cmd VNC://localhost:%d -chatonly", port); + + pid_t pid = fork(); + + if (pid == -1) { + perror("fork"); + return; + } else if (pid == 0) { + char *args[4]; + int d; + args[0] = "/bin/sh"; + args[1] = "-c"; + /* "ssvnc -cmd VNC://fd=0 -chatonly"; */ + args[2] = cmd; + args[3] = NULL; + + for (d = 3; d < 256; d++) { + close(d); + } + set_env("VNCVIEWER_PASSWORD", "moo"); + + execvp(args[0], args); + perror("exec"); + exit(1); + } else { + int i, sock = -1; + rfbNewClientHookPtr new_save; + + signal(SIGALRM, wsock_timeout); + wsock_timeout_sock = lsock; + + alarm(10); + sock = accept(lsock, (struct sockaddr *)&addr, &addrlen); + alarm(0); + + signal(SIGALRM, SIG_DFL); + close(lsock); + + if (sock < 0) { + return; + } + + new_save = screen->newClientHook; + screen->newClientHook = new_client_chat_helper; + + chat_window_client = rfbNewClient(screen, sock); + + screen->newClientHook = new_save; + + if (chat_window_client != NULL) { + rfbPasswordCheckProcPtr pwchk_save = screen->passwordCheck; + rfbBool save_shared1 = screen->alwaysShared; + rfbBool save_shared2 = screen->neverShared; + + screen->alwaysShared = TRUE; + screen->neverShared = FALSE; + + screen->passwordCheck = password_check_chat_helper; + for (i=0; i<30; i++) { + rfbPE(-1); + if (!chat_window_client) { + break; + } + if (chat_window_client->state == RFB_NORMAL) { + break; + } + } + + screen->passwordCheck = pwchk_save; + screen->alwaysShared = save_shared1; + screen->neverShared = save_shared2; + } + } +} + void set_text_chat(rfbClientPtr cl, int len, char *txt) { int dochat = 1; rfbClientIteratorPtr iter; rfbClientPtr cl2; + unsigned int ulen = (unsigned int) len; if (no_ultra_ext || ! dochat) { return; } -#if 0 - rfbLog("set_text_chat: len=%d\n", len); - rfbLog("set_text_chat: len=0x%x txt='", len); - if (0 < len && len < 10000) write(2, txt, len); - fprintf(stderr, "'\n"); -#endif if (unixpw_in_progress) { rfbLog("set_text_chat: unixpw_in_progress, dropping client.\n"); rfbCloseClient(cl); return; } + if (chat_window && chat_window_client == NULL && ulen == rfbTextChatOpen) { + try_local_chat_window(); + } + saw_ultra_chat = 1; iter = rfbGetClientIterator(screen); @@ -1523,10 +1633,15 @@ void set_text_chat(rfbClientPtr cl, int len, char *txt) { if (cl2 == cl) { continue; } + if (cl2->state != RFB_NORMAL) { + continue; + } if (ulen == rfbTextChatOpen) { rfbSendTextChatMessage(cl2, rfbTextChatOpen, ""); } else if (ulen == rfbTextChatClose) { rfbSendTextChatMessage(cl2, rfbTextChatClose, ""); + /* not clear what is going on WRT close and finished... */ + rfbSendTextChatMessage(cl2, rfbTextChatFinished, ""); } else if (ulen == rfbTextChatFinished) { rfbSendTextChatMessage(cl2, rfbTextChatFinished, ""); } else if (len <= rfbTextMaxSize) { @@ -1534,6 +1649,11 @@ void set_text_chat(rfbClientPtr cl, int len, char *txt) { } } rfbReleaseClientIterator(iter); + + if (ulen == rfbTextChatClose && cl != NULL) { + /* not clear what is going on WRT close and finished... */ + rfbSendTextChatMessage(cl, rfbTextChatFinished, ""); + } } int get_keyboard_led_state_hook(rfbScreenInfoPtr s) { |