485 lines
13 KiB
C
485 lines
13 KiB
C
/*
|
|
*Copyright (C) 1994-2000 The XFree86 Project, Inc. All Rights Reserved.
|
|
*
|
|
*Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
*"Software"), to deal in the Software without restriction, including
|
|
*without limitation the rights to use, copy, modify, merge, publish,
|
|
*distribute, sublicense, and/or sell copies of the Software, and to
|
|
*permit persons to whom the Software is furnished to do so, subject to
|
|
*the following conditions:
|
|
*
|
|
*The above copyright notice and this permission notice shall be
|
|
*included in all copies or substantial portions of the Software.
|
|
*
|
|
*THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
*EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
*MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
*NONINFRINGEMENT. IN NO EVENT SHALL THE XFREE86 PROJECT BE LIABLE FOR
|
|
*ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
|
|
*CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
*WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*Except as contained in this notice, the name of the XFree86 Project
|
|
*shall not be used in advertising or otherwise to promote the sale, use
|
|
*or other dealings in this Software without prior written authorization
|
|
*from the XFree86 Project.
|
|
*
|
|
* Authors: Harold L Hunt II
|
|
*/
|
|
/* $XFree86: xc/programs/Xserver/hw/xwin/winclipboardthread.c,v 1.3 2003/10/02 13:30:10 eich Exp $ */
|
|
|
|
#include "winclipboard.h"
|
|
|
|
/*
|
|
* References to external symbols
|
|
*/
|
|
|
|
extern Bool g_fCalledSetLocale;
|
|
|
|
|
|
/*
|
|
* Global variables
|
|
*/
|
|
|
|
static jmp_buf g_jmpEntry;
|
|
static Bool g_shutdown = FALSE;
|
|
|
|
|
|
/*
|
|
* Local function prototypes
|
|
*/
|
|
|
|
static int
|
|
winClipboardErrorHandler (Display *pDisplay, XErrorEvent *pErr);
|
|
|
|
static int
|
|
winClipboardIOErrorHandler (Display *pDisplay);
|
|
|
|
|
|
/*
|
|
* Main thread function
|
|
*/
|
|
|
|
void *
|
|
winClipboardProc (void *pArg)
|
|
{
|
|
Atom atomClipboard, atomClipboardManager;
|
|
Atom atomLocalProperty, atomCompoundText;
|
|
Atom atomUTF8String, atomTargets;
|
|
int iReturn;
|
|
HWND hwnd = NULL;
|
|
int iConnectionNumber;
|
|
int fdMessageQueue;
|
|
fd_set fdsRead;
|
|
int iMaxDescriptor;
|
|
Display *pDisplay;
|
|
Window iWindow;
|
|
Atom atomDeleteWindow;
|
|
Bool fReturn;
|
|
int iRetries;
|
|
Bool fUnicodeSupport;
|
|
char szDisplay[512];
|
|
ClipboardProcArgPtr pProcArg = (ClipboardProcArgPtr) pArg;
|
|
|
|
ErrorF ("winClipboardProc - Hello\n");
|
|
|
|
/* Check that argument pointer is not invalid */
|
|
if (pArg == NULL)
|
|
{
|
|
ErrorF ("winClipboardProc - pArg is NULL, bailing.\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
ErrorF ("winClipboardProc - Calling pthread_mutex_lock ()\n");
|
|
|
|
/* Grab the server started mutex - pause until we get it */
|
|
iReturn = pthread_mutex_lock (pProcArg->ppmServerStarted);
|
|
if (iReturn != 0)
|
|
{
|
|
ErrorF ("winClipboardProc - pthread_mutex_lock () failed: %d\n",
|
|
iReturn);
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
ErrorF ("winClipboardProc - pthread_mutex_lock () returned.\n");
|
|
|
|
/* Do we have Unicode support? */
|
|
fUnicodeSupport = winClipboardDetectUnicodeSupport ();
|
|
|
|
/* Set the current locale? What does this do? */
|
|
if (fUnicodeSupport && !g_fCalledSetLocale)
|
|
{
|
|
ErrorF ("winClipboardProc - Calling setlocale ()\n");
|
|
if (!setlocale (LC_ALL, ""))
|
|
{
|
|
ErrorF ("winClipboardProc - setlocale () error\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
ErrorF ("winClipboardProc - setlocale () returned\n");
|
|
|
|
/* See if X supports the current locale */
|
|
if (XSupportsLocale () == False)
|
|
{
|
|
ErrorF ("winClipboardProc - Locale not supported by X\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
}
|
|
|
|
/* Flag that we have called setlocale */
|
|
g_fCalledSetLocale = TRUE;
|
|
|
|
/* Allow multiple threads to access Xlib */
|
|
if (XInitThreads () == 0)
|
|
{
|
|
ErrorF ("winClipboardProc - XInitThreads failed.\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
ErrorF ("winClipboardProc - XInitThreads () returned.\n");
|
|
|
|
/* Release the server started mutex */
|
|
pthread_mutex_unlock (pProcArg->ppmServerStarted);
|
|
|
|
ErrorF ("winClipboardProc - pthread_mutex_unlock () returned.\n");
|
|
|
|
/* Set jump point for Error exits */
|
|
iReturn = setjmp (g_jmpEntry);
|
|
|
|
/* Check if we should continue operations */
|
|
if (iReturn != WIN_JMP_ERROR_IO
|
|
&& iReturn != WIN_JMP_OKAY)
|
|
{
|
|
/* setjmp returned an unknown value, exit */
|
|
ErrorF ("winClipboardProc - setjmp returned: %d exiting\n",
|
|
iReturn);
|
|
pthread_exit (NULL);
|
|
}
|
|
else if (g_shutdown)
|
|
{
|
|
/* Shutting down, the X server severed out connection! */
|
|
ErrorF ("winClipboardProc - Detected shutdown in progress\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
else if (iReturn == WIN_JMP_ERROR_IO)
|
|
{
|
|
ErrorF ("winClipboardProc - setjmp returned and hwnd: %08x\n", hwnd);
|
|
}
|
|
|
|
/* Initialize retry count */
|
|
iRetries = 0;
|
|
|
|
/* Setup the display connection string x */
|
|
snprintf (szDisplay,
|
|
512,
|
|
"127.0.0.1:%s.%d",
|
|
display,
|
|
(int) pProcArg->dwScreen);
|
|
|
|
/* Print the display connection string */
|
|
ErrorF ("winClipboardProc - DISPLAY=%s\n", szDisplay);
|
|
|
|
/* Open the X display */
|
|
do
|
|
{
|
|
pDisplay = XOpenDisplay (szDisplay);
|
|
if (pDisplay == NULL)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not open display, "
|
|
"try: %d, sleeping: %d\n",
|
|
iRetries + 1, WIN_CONNECT_DELAY);
|
|
++iRetries;
|
|
sleep (WIN_CONNECT_DELAY);
|
|
continue;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
while (pDisplay == NULL && iRetries < WIN_CONNECT_RETRIES);
|
|
|
|
/* Make sure that the display opened */
|
|
if (pDisplay == NULL)
|
|
{
|
|
ErrorF ("winClipboardProc - Failed opening the display, giving up\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
ErrorF ("winClipboardProc - XOpenDisplay () returned and "
|
|
"successfully opened the display.\n");
|
|
|
|
/* Create Windows messaging window */
|
|
hwnd = winClipboardCreateMessagingWindow ();
|
|
|
|
/* Get our connection number */
|
|
iConnectionNumber = ConnectionNumber (pDisplay);
|
|
|
|
/* Open a file descriptor for the windows message queue */
|
|
fdMessageQueue = open (WIN_MSG_QUEUE_FNAME, O_RDONLY);
|
|
if (fdMessageQueue == -1)
|
|
{
|
|
ErrorF ("winClipboardProc - Failed opening %s\n", WIN_MSG_QUEUE_FNAME);
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Find max of our file descriptors */
|
|
iMaxDescriptor = max (fdMessageQueue, iConnectionNumber) + 1;
|
|
|
|
/* Select event types to watch */
|
|
if (XSelectInput (pDisplay,
|
|
DefaultRootWindow (pDisplay),
|
|
SubstructureNotifyMask |
|
|
StructureNotifyMask |
|
|
PropertyChangeMask) == BadWindow)
|
|
ErrorF ("winClipboardProc - XSelectInput generated BadWindow "
|
|
"on RootWindow\n\n");
|
|
|
|
/* Create a messaging window */
|
|
iWindow = XCreateSimpleWindow (pDisplay,
|
|
DefaultRootWindow (pDisplay),
|
|
1, 1,
|
|
500, 500,
|
|
0,
|
|
BlackPixel (pDisplay, 0),
|
|
BlackPixel (pDisplay, 0));
|
|
if (iWindow == 0)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create a window\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* This looks like our only hope for getting a message before shutdown */
|
|
/* Register for WM_DELETE_WINDOW message from window manager */
|
|
atomDeleteWindow = XInternAtom (pDisplay, "WM_DELETE_WINDOW", False);
|
|
XSetWMProtocols (pDisplay, iWindow, &atomDeleteWindow, 1);
|
|
|
|
/* Set error handler */
|
|
XSetErrorHandler (winClipboardErrorHandler);
|
|
XSetIOErrorHandler (winClipboardIOErrorHandler);
|
|
|
|
/* Create an atom for CLIPBOARD_MANAGER */
|
|
atomClipboardManager = XInternAtom (pDisplay, "CLIPBOARD_MANAGER", False);
|
|
if (atomClipboardManager == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create CLIPBOARD_MANAGER atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Assert ownership of CLIPBOARD_MANAGER */
|
|
iReturn = XSetSelectionOwner (pDisplay, atomClipboardManager,
|
|
iWindow, CurrentTime);
|
|
if (iReturn == BadAtom || iReturn == BadWindow)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not set CLIPBOARD_MANAGER owner\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Create an atom for CLIPBOARD */
|
|
atomClipboard = XInternAtom (pDisplay, "CLIPBOARD", False);
|
|
if (atomClipboard == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create CLIPBOARD atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Assert ownership of CLIPBOARD */
|
|
iReturn = XSetSelectionOwner (pDisplay, atomClipboard,
|
|
iWindow, CurrentTime);
|
|
if (iReturn == BadAtom || iReturn == BadWindow)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not set CLIPBOARD owner\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Assert ownership of PRIMARY */
|
|
iReturn = XSetSelectionOwner (pDisplay, XA_PRIMARY,
|
|
iWindow, CurrentTime);
|
|
if (iReturn == BadAtom || iReturn == BadWindow)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not set PRIMARY owner\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Local property to hold pasted data */
|
|
atomLocalProperty = XInternAtom (pDisplay, "CYGX_CUT_BUFFER", False);
|
|
if (atomLocalProperty == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create CYGX_CUT_BUFFER atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Create an atom for UTF8_STRING */
|
|
atomUTF8String = XInternAtom (pDisplay, "UTF8_STRING", False);
|
|
if (atomUTF8String == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create UTF8_STRING atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Create an atom for COMPOUND_TEXT */
|
|
atomCompoundText = XInternAtom (pDisplay, "COMPOUND_TEXT", False);
|
|
if (atomCompoundText == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create COMPOUND_TEXT atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Create an atom for TARGETS */
|
|
atomTargets = XInternAtom (pDisplay, "TARGETS", False);
|
|
if (atomTargets == None)
|
|
{
|
|
ErrorF ("winClipboardProc - Could not create TARGETS atom\n");
|
|
pthread_exit (NULL);
|
|
}
|
|
|
|
/* Pre-flush X events */
|
|
/*
|
|
* NOTE: Apparently you'll freeze if you don't do this,
|
|
* because there may be events in local data structures
|
|
* already.
|
|
*/
|
|
winClipboardFlushXEvents (hwnd,
|
|
atomClipboard,
|
|
atomLocalProperty,
|
|
atomUTF8String,
|
|
atomCompoundText,
|
|
atomTargets,
|
|
atomDeleteWindow,
|
|
iWindow,
|
|
pDisplay,
|
|
fUnicodeSupport);
|
|
|
|
/* Pre-flush Windows messages */
|
|
if (!winClipboardFlushWindowsMessageQueue (hwnd))
|
|
return 0;
|
|
|
|
/* Loop for X events */
|
|
while (1)
|
|
{
|
|
/* Setup the file descriptor set */
|
|
/*
|
|
* NOTE: You have to do this before every call to select
|
|
* because select modifies the mask to indicate
|
|
* which descriptors are ready.
|
|
*/
|
|
FD_ZERO (&fdsRead);
|
|
FD_SET (fdMessageQueue, &fdsRead);
|
|
FD_SET (iConnectionNumber, &fdsRead);
|
|
|
|
/* Wait for a Windows event or an X event */
|
|
iReturn = select (iMaxDescriptor, /* Highest fds number */
|
|
&fdsRead, /* Read mask */
|
|
NULL, /* No write mask */
|
|
NULL, /* No exception mask */
|
|
NULL); /* No timeout */
|
|
if (iReturn <= 0)
|
|
{
|
|
ErrorF ("winClipboardProc - Call to select () failed: %d. "
|
|
"Bailing.\n", iReturn);
|
|
break;
|
|
}
|
|
|
|
/* Branch on which descriptor became active */
|
|
if (FD_ISSET (iConnectionNumber, &fdsRead))
|
|
{
|
|
/* X event ready */
|
|
#if 0
|
|
ErrorF ("winClipboardProc - X event ready\n");
|
|
#endif
|
|
|
|
/* Process X events */
|
|
/* Exit when we see that server is shutting down */
|
|
fReturn = winClipboardFlushXEvents (hwnd,
|
|
atomClipboard,
|
|
atomLocalProperty,
|
|
atomUTF8String,
|
|
atomCompoundText,
|
|
atomTargets,
|
|
atomDeleteWindow,
|
|
iWindow,
|
|
pDisplay,
|
|
fUnicodeSupport);
|
|
if (!fReturn)
|
|
{
|
|
ErrorF ("winClipboardProc - Caught WM_DELETE_WINDOW - "
|
|
"shutting down\n");
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Check for Windows event ready */
|
|
if (FD_ISSET (fdMessageQueue, &fdsRead))
|
|
{
|
|
/* Windows event ready */
|
|
#if 0
|
|
ErrorF ("winClipboardProc - Windows event ready\n");
|
|
#endif
|
|
|
|
/* Process Windows messages */
|
|
if (!winClipboardFlushWindowsMessageQueue (hwnd))
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* winClipboardErrorHandler - Our application specific error handler
|
|
*/
|
|
|
|
static int
|
|
winClipboardErrorHandler (Display *pDisplay, XErrorEvent *pErr)
|
|
{
|
|
char pszErrorMsg[100];
|
|
|
|
XGetErrorText (pDisplay,
|
|
pErr->error_code,
|
|
pszErrorMsg,
|
|
sizeof (pszErrorMsg));
|
|
ErrorF ("winClipboardErrorHandler - ERROR: \n\t%s\n", pszErrorMsg);
|
|
|
|
if (pErr->error_code == BadWindow
|
|
|| pErr->error_code == BadMatch
|
|
|| pErr->error_code == BadDrawable)
|
|
{
|
|
#if 0
|
|
pthread_exit (NULL);
|
|
#endif
|
|
}
|
|
|
|
#if 0
|
|
pthread_exit (NULL);
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* winClipboardIOErrorHandler - Our application specific IO error handler
|
|
*/
|
|
|
|
static int
|
|
winClipboardIOErrorHandler (Display *pDisplay)
|
|
{
|
|
printf ("\nwinClipboardIOErrorHandler!\n\n");
|
|
|
|
/* Restart at the main entry point */
|
|
longjmp (g_jmpEntry, WIN_JMP_ERROR_IO);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Notify the clipboard thread we're exiting and not to reconnect
|
|
*/
|
|
|
|
void
|
|
winDeinitClipboard ()
|
|
{
|
|
ErrorF ("winDeinitClipboard - Noting shutdown in progress\n");
|
|
g_shutdown = TRUE;
|
|
}
|