1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 01:02:24 +00:00

Move the code that reads the Windows clipboard into a trivial

subthread, so that it won't deadlock if fetching the content of the
clipboard turns out to depend on a network connection forwarded
through PuTTY.

[originally from svn r8338]
This commit is contained in:
Simon Tatham 2008-11-28 18:28:23 +00:00
parent b892d451ee
commit 54835a9838

View File

@ -61,6 +61,7 @@
#define WM_IGNORE_CLIP (WM_APP + 2)
#define WM_FULLSCR_ON_MAX (WM_APP + 3)
#define WM_AGENT_CALLBACK (WM_APP + 4)
#define WM_GOT_CLIPDATA (WM_APP + 6)
/* Needed for Chinese support and apparently not always defined. */
#ifndef VK_PROCESSKEY
@ -93,6 +94,7 @@ static int is_full_screen(void);
static void make_full_screen(void);
static void clear_full_screen(void);
static void flip_full_screen(void);
static int process_clipdata(HGLOBAL clipdata, int unicode);
/* Window layout information */
static void reset_window(int);
@ -127,6 +129,9 @@ static const struct telnet_special *specials = NULL;
static HMENU specials_menu = NULL;
static int n_specials = 0;
static wchar_t *clipboard_contents;
static size_t clipboard_length;
#define TIMING_TIMER_ID 1234
static long timing_next_time;
@ -3004,6 +3009,10 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
sfree(c);
}
return 0;
case WM_GOT_CLIPDATA:
if (process_clipdata((HGLOBAL)lParam, wParam))
term_do_paste(term);
return 0;
default:
if (message == wm_mousewheel || message == WM_MOUSEWHEEL) {
int shift_pressed=0, control_pressed=0;
@ -4413,16 +4422,6 @@ static int TranslateKey(UINT message, WPARAM wParam, LPARAM lParam,
return -1;
}
void request_paste(void *frontend)
{
/*
* In Windows, pasting is synchronous: we can read the
* clipboard with no difficulty, so request_paste() can just go
* ahead and paste.
*/
term_do_paste(term);
}
void set_title(void *frontend, char *title)
{
sfree(window_name);
@ -4896,46 +4895,91 @@ void write_clip(void *frontend, wchar_t * data, int *attr, int len, int must_des
SendMessage(hwnd, WM_IGNORE_CLIP, FALSE, 0);
}
void get_clip(void *frontend, wchar_t ** p, int *len)
static DWORD WINAPI clipboard_read_threadfunc(void *param)
{
static HGLOBAL clipdata = NULL;
static wchar_t *converted = 0;
wchar_t *p2;
HWND hwnd = (HWND)param;
HGLOBAL clipdata;
if (converted) {
sfree(converted);
converted = 0;
}
if (!p) {
if (clipdata)
GlobalUnlock(clipdata);
clipdata = NULL;
return;
} else if (OpenClipboard(NULL)) {
if (OpenClipboard(NULL)) {
if ((clipdata = GetClipboardData(CF_UNICODETEXT))) {
CloseClipboard();
*p = GlobalLock(clipdata);
if (*p) {
for (p2 = *p; *p2; p2++);
*len = p2 - *p;
return;
}
} else if ( (clipdata = GetClipboardData(CF_TEXT)) ) {
char *s;
int i;
CloseClipboard();
s = GlobalLock(clipdata);
i = MultiByteToWideChar(CP_ACP, 0, s, strlen(s) + 1, 0, 0);
*p = converted = snewn(i, wchar_t);
MultiByteToWideChar(CP_ACP, 0, s, strlen(s) + 1, converted, i);
*len = i - 1;
return;
} else
CloseClipboard();
SendMessage(hwnd, WM_GOT_CLIPDATA, (WPARAM)1, (LPARAM)clipdata);
} else if ((clipdata = GetClipboardData(CF_TEXT))) {
SendMessage(hwnd, WM_GOT_CLIPDATA, (WPARAM)0, (LPARAM)clipdata);
}
CloseClipboard();
}
*p = NULL;
*len = 0;
return 0;
}
static int process_clipdata(HGLOBAL clipdata, int unicode)
{
static wchar_t *converted = 0;
sfree(clipboard_contents);
clipboard_contents = NULL;
clipboard_length = 0;
if (unicode) {
wchar_t *p = GlobalLock(clipdata);
wchar_t *p2;
if (p) {
/* Unwilling to rely on Windows having wcslen() */
for (p2 = p; *p2; p2++);
clipboard_length = p2 - p;
clipboard_contents = snewn(clipboard_length + 1, wchar_t);
memcpy(clipboard_contents, p, clipboard_length * sizeof(wchar_t));
clipboard_contents[clipboard_length] = L'\0';
return TRUE;
}
} else {
char *s = GlobalLock(clipdata);
int i;
if (s) {
i = MultiByteToWideChar(CP_ACP, 0, s, strlen(s) + 1, 0, 0);
clipboard_contents = snewn(i, wchar_t);
MultiByteToWideChar(CP_ACP, 0, s, strlen(s) + 1,
clipboard_contents, i);
clipboard_length = i - 1;
clipboard_contents[clipboard_length] = L'\0';
return TRUE;
}
}
return FALSE;
}
void request_paste(void *frontend)
{
/*
* I always thought pasting was synchronous in Windows; the
* clipboard access functions certainly _look_ synchronous,
* unlike the X ones. But in fact it seems that in some
* situations the contents of the clipboard might not be
* immediately available, and the clipboard-reading functions
* may block. This leads to trouble if the application
* delivering the clipboard data has to get hold of it by -
* for example - talking over a network connection which is
* forwarded through this very PuTTY.
*
* Hence, we spawn a subthread to read the clipboard, and do
* our paste when it's finished. The thread will send a
* message back to our main window when it terminates, and
* that tells us it's OK to paste.
*/
DWORD in_threadid; /* required for Win9x */
CreateThread(NULL, 0, clipboard_read_threadfunc,
hwnd, 0, &in_threadid);
}
void get_clip(void *frontend, wchar_t **p, int *len)
{
if (p) {
*p = clipboard_contents;
*len = clipboard_length;
}
}
#if 0