476 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			476 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  *Copyright (C) 2003-2004 Harold L Hunt II All Rights Reserved.
 | |
|  *Copyright (C) Colin Harrison 2005-2008
 | |
|  *
 | |
|  *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 HAROLD L HUNT II 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 copyright holder(s)
 | |
|  *and author(s) 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 copyright holder(s) and author(s).
 | |
|  *
 | |
|  * Authors:	Harold L Hunt II
 | |
|  *              Colin Harrison
 | |
|  */
 | |
| 
 | |
| #ifdef HAVE_XWIN_CONFIG_H
 | |
| #include <xwin-config.h>
 | |
| #else
 | |
| #define HAS_WINSOCK 1
 | |
| #endif
 | |
| #include <sys/types.h>
 | |
| #include <signal.h>
 | |
| #include "winclipboard.h"
 | |
| #ifdef __CYGWIN__
 | |
| #include <errno.h>
 | |
| #endif
 | |
| #include "misc.h"
 | |
| 
 | |
| /*
 | |
|  * References to external symbols
 | |
|  */
 | |
| 
 | |
| extern Bool g_fUnicodeClipboard;
 | |
| extern Bool g_fClipboardStarted;
 | |
| extern Bool g_fClipboardLaunched;
 | |
| extern Bool g_fClipboard;
 | |
| extern HWND g_hwndClipboard;
 | |
| extern void *g_pClipboardDisplay;
 | |
| extern Window g_iClipboardWindow;
 | |
| 
 | |
| /*
 | |
|  * Global variables
 | |
|  */
 | |
| 
 | |
| static jmp_buf g_jmpEntry;
 | |
| static int clipboardRestarts = 0;
 | |
| static XIOErrorHandler g_winClipboardOldIOErrorHandler;
 | |
| static pthread_t g_winClipboardProcThread;
 | |
| 
 | |
| Bool g_fUseUnicode = FALSE;
 | |
| 
 | |
| /*
 | |
|  * Local function prototypes
 | |
|  */
 | |
| 
 | |
| static int
 | |
|  winClipboardErrorHandler(Display * pDisplay, XErrorEvent * pErr);
 | |
| 
 | |
| static int
 | |
|  winClipboardIOErrorHandler(Display * pDisplay);
 | |
| 
 | |
| /*
 | |
|  * Main thread function
 | |
|  */
 | |
| 
 | |
| void *
 | |
| winClipboardProc(void *pvNotUsed)
 | |
| {
 | |
|     Atom atomClipboard, atomClipboardManager;
 | |
|     int iReturn;
 | |
|     HWND hwnd = NULL;
 | |
|     int iConnectionNumber = 0;
 | |
| 
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|     int fdMessageQueue = 0;
 | |
| #else
 | |
|     struct timeval tvTimeout;
 | |
| #endif
 | |
|     fd_set fdsRead;
 | |
|     int iMaxDescriptor;
 | |
|     Display *pDisplay = NULL;
 | |
|     Window iWindow = None;
 | |
|     int iRetries;
 | |
|     Bool fUseUnicode;
 | |
|     char szDisplay[512];
 | |
|     int iSelectError;
 | |
| 
 | |
|     winDebug("winClipboardProc - Hello\n");
 | |
|     ++clipboardRestarts;
 | |
| 
 | |
|     /* Do we use Unicode clipboard? */
 | |
|     fUseUnicode = g_fUnicodeClipboard;
 | |
| 
 | |
|     /* Save the Unicode support flag in a global */
 | |
|     g_fUseUnicode = fUseUnicode;
 | |
| 
 | |
|     /* Allow multiple threads to access Xlib */
 | |
|     if (XInitThreads() == 0) {
 | |
|         ErrorF("winClipboardProc - XInitThreads failed.\n");
 | |
|         goto winClipboardProc_Exit;
 | |
|     }
 | |
| 
 | |
|     /* See if X supports the current locale */
 | |
|     if (XSupportsLocale() == False) {
 | |
|         ErrorF("winClipboardProc - Warning: Locale not supported by X.\n");
 | |
|     }
 | |
| 
 | |
|     /* Set error handler */
 | |
|     XSetErrorHandler(winClipboardErrorHandler);
 | |
|     g_winClipboardProcThread = pthread_self();
 | |
|     g_winClipboardOldIOErrorHandler =
 | |
|         XSetIOErrorHandler(winClipboardIOErrorHandler);
 | |
| 
 | |
|     /* 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);
 | |
|         goto winClipboardProc_Exit;
 | |
|     }
 | |
|     else if (iReturn == WIN_JMP_ERROR_IO) {
 | |
|         /* TODO: Cleanup the Win32 window and free any allocated memory */
 | |
|         ErrorF("winClipboardProc - setjmp returned for IO Error Handler.\n");
 | |
|         pthread_exit(NULL);
 | |
|     }
 | |
| 
 | |
|     /* Use our generated cookie for authentication */
 | |
|     winSetAuthorization();
 | |
| 
 | |
|     /* Initialize retry count */
 | |
|     iRetries = 0;
 | |
| 
 | |
|     /* Setup the display connection string x */
 | |
|     /*
 | |
|      * NOTE: Always connect to screen 0 since we require that screen
 | |
|      * numbers start at 0 and increase without gaps.  We only need
 | |
|      * to connect to one screen on the display to get events
 | |
|      * for all screens on the display.  That is why there is only
 | |
|      * one clipboard client thread.
 | |
|      */
 | |
|     snprintf(szDisplay, 512, "127.0.0.1:%s.0", display);
 | |
| 
 | |
|     /* 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");
 | |
|         goto winClipboardProc_Done;
 | |
|     }
 | |
| 
 | |
|     /* Save the display in the screen privates */
 | |
|     g_pClipboardDisplay = pDisplay;
 | |
| 
 | |
|     ErrorF("winClipboardProc - XOpenDisplay () returned and "
 | |
|            "successfully opened the display.\n");
 | |
| 
 | |
|     /* Get our connection number */
 | |
|     iConnectionNumber = ConnectionNumber(pDisplay);
 | |
| 
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|     /* 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);
 | |
|         goto winClipboardProc_Done;
 | |
|     }
 | |
| 
 | |
|     /* Find max of our file descriptors */
 | |
|     iMaxDescriptor = max(fdMessageQueue, iConnectionNumber) + 1;
 | |
| #else
 | |
|     iMaxDescriptor = iConnectionNumber + 1;
 | |
| #endif
 | |
| 
 | |
|     /* Create atoms */
 | |
|     atomClipboard = XInternAtom(pDisplay, "CLIPBOARD", False);
 | |
|     atomClipboardManager = XInternAtom(pDisplay, "CLIPBOARD_MANAGER", False);
 | |
| 
 | |
|     /* 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 an X window.\n");
 | |
|         goto winClipboardProc_Done;
 | |
|     }
 | |
| 
 | |
|     XStoreName(pDisplay, iWindow, "xwinclip");
 | |
| 
 | |
|     /* Select event types to watch */
 | |
|     if (XSelectInput(pDisplay, iWindow, PropertyChangeMask) == BadWindow)
 | |
|         ErrorF("winClipboardProc - XSelectInput generated BadWindow "
 | |
|                "on messaging window\n");
 | |
| 
 | |
|     /* Save the window in the screen privates */
 | |
|     g_iClipboardWindow = iWindow;
 | |
| 
 | |
|     /* Create Windows messaging window */
 | |
|     hwnd = winClipboardCreateMessagingWindow();
 | |
| 
 | |
|     /* Save copy of HWND in screen privates */
 | |
|     g_hwndClipboard = hwnd;
 | |
| 
 | |
|     /* Assert ownership of selections if Win32 clipboard is owned */
 | |
|     if (NULL != GetClipboardOwner()) {
 | |
|         /* PRIMARY */
 | |
|         iReturn = XSetSelectionOwner(pDisplay, XA_PRIMARY,
 | |
|                                      iWindow, CurrentTime);
 | |
|         if (iReturn == BadAtom || iReturn == BadWindow ||
 | |
|             XGetSelectionOwner(pDisplay, XA_PRIMARY) != iWindow) {
 | |
|             ErrorF("winClipboardProc - Could not set PRIMARY owner\n");
 | |
|             goto winClipboardProc_Done;
 | |
|         }
 | |
| 
 | |
|         /* CLIPBOARD */
 | |
|         iReturn = XSetSelectionOwner(pDisplay, atomClipboard,
 | |
|                                      iWindow, CurrentTime);
 | |
|         if (iReturn == BadAtom || iReturn == BadWindow ||
 | |
|             XGetSelectionOwner(pDisplay, atomClipboard) != iWindow) {
 | |
|             ErrorF("winClipboardProc - Could not set CLIPBOARD owner\n");
 | |
|             goto winClipboardProc_Done;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* 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, iWindow, pDisplay, fUseUnicode);
 | |
| 
 | |
|     /* Pre-flush Windows messages */
 | |
|     if (!winClipboardFlushWindowsMessageQueue(hwnd))
 | |
|         return 0;
 | |
| 
 | |
|     /* Signal that the clipboard client has started */
 | |
|     g_fClipboardStarted = TRUE;
 | |
| 
 | |
|     /* 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(iConnectionNumber, &fdsRead);
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|         FD_SET(fdMessageQueue, &fdsRead);
 | |
| #else
 | |
|         tvTimeout.tv_sec = 0;
 | |
|         tvTimeout.tv_usec = 100;
 | |
| #endif
 | |
| 
 | |
|         /* 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 */
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|                          NULL   /* No timeout */
 | |
| #else
 | |
|                          &tvTimeout     /* Set timeout */
 | |
| #endif
 | |
|             );
 | |
| 
 | |
| #ifndef HAS_WINSOCK
 | |
|         iSelectError = errno;
 | |
| #else
 | |
|         iSelectError = WSAGetLastError();
 | |
| #endif
 | |
| 
 | |
|         if (iReturn < 0) {
 | |
| #ifndef HAS_WINSOCK
 | |
|             if (iSelectError == EINTR)
 | |
| #else
 | |
|             if (iSelectError == WSAEINTR)
 | |
| #endif
 | |
|                 continue;
 | |
| 
 | |
|             ErrorF("winClipboardProc - Call to select () failed: %d.  "
 | |
|                    "Bailing.\n", iReturn);
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         /* Branch on which descriptor became active */
 | |
|         if (FD_ISSET(iConnectionNumber, &fdsRead)) {
 | |
|             /* Process X events */
 | |
|             /* Exit when we see that server is shutting down */
 | |
|             iReturn = winClipboardFlushXEvents(hwnd,
 | |
|                                                iWindow, pDisplay, fUseUnicode);
 | |
|             if (WIN_XEVENTS_SHUTDOWN == iReturn) {
 | |
|                 ErrorF("winClipboardProc - winClipboardFlushXEvents "
 | |
|                        "trapped shutdown event, exiting main loop.\n");
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|         /* Check for Windows event ready */
 | |
|         if (FD_ISSET(fdMessageQueue, &fdsRead))
 | |
| #else
 | |
|         if (1)
 | |
| #endif
 | |
|         {
 | |
|             /* Process Windows messages */
 | |
|             if (!winClipboardFlushWindowsMessageQueue(hwnd)) {
 | |
|                 ErrorF("winClipboardProc - "
 | |
|                        "winClipboardFlushWindowsMessageQueue trapped "
 | |
|                        "WM_QUIT message, exiting main loop.\n");
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|  winClipboardProc_Exit:
 | |
|     /* disable the clipboard, which means the thread will die */
 | |
|     g_fClipboard = FALSE;
 | |
| 
 | |
|  winClipboardProc_Done:
 | |
|     /* Close our Windows window */
 | |
|     if (g_hwndClipboard) {
 | |
|         /* Destroy the Window window (hwnd) */
 | |
|         winDebug("winClipboardProc - Destroy Windows window\n");
 | |
|         PostMessage(g_hwndClipboard, WM_DESTROY, 0, 0);
 | |
|         winClipboardFlushWindowsMessageQueue(g_hwndClipboard);
 | |
|     }
 | |
| 
 | |
|     /* Close our X window */
 | |
|     if (pDisplay && iWindow) {
 | |
|         iReturn = XDestroyWindow(pDisplay, iWindow);
 | |
|         if (iReturn == BadWindow)
 | |
|             ErrorF("winClipboardProc - XDestroyWindow returned BadWindow.\n");
 | |
|         else
 | |
|             ErrorF("winClipboardProc - XDestroyWindow succeeded.\n");
 | |
|     }
 | |
| 
 | |
| #ifdef HAS_DEVWINDOWS
 | |
|     /* Close our Win32 message handle */
 | |
|     if (fdMessageQueue)
 | |
|         close(fdMessageQueue);
 | |
| #endif
 | |
| 
 | |
| #if 0
 | |
|     /*
 | |
|      * FIXME: XCloseDisplay hangs if we call it, as of 2004/03/26.  The
 | |
|      * XSync and XSelectInput calls did not help.
 | |
|      */
 | |
| 
 | |
|     /* Discard any remaining events */
 | |
|     XSync(pDisplay, TRUE);
 | |
| 
 | |
|     /* Select event types to watch */
 | |
|     XSelectInput(pDisplay, DefaultRootWindow(pDisplay), None);
 | |
| 
 | |
|     /* Close our X display */
 | |
|     if (pDisplay) {
 | |
|         XCloseDisplay(pDisplay);
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     /* global clipboard variable reset */
 | |
|     g_fClipboardLaunched = FALSE;
 | |
|     g_fClipboardStarted = FALSE;
 | |
|     g_iClipboardWindow = None;
 | |
|     g_pClipboardDisplay = NULL;
 | |
|     g_hwndClipboard = NULL;
 | |
| 
 | |
|     /* checking if we need to restart */
 | |
|     if (clipboardRestarts >= WIN_CLIPBOARD_RETRIES) {
 | |
|         /* terminates clipboard thread but the main server still lives */
 | |
|         ErrorF
 | |
|             ("winClipboardProc - the clipboard thread has restarted %d times and seems to be unstable, disabling clipboard integration\n",
 | |
|              clipboardRestarts);
 | |
|         g_fClipboard = FALSE;
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (g_fClipboard) {
 | |
|         sleep(WIN_CLIPBOARD_DELAY);
 | |
|         ErrorF("winClipboardProc - trying to restart clipboard thread \n");
 | |
|         /* Create the clipboard client thread */
 | |
|         if (!winInitClipboard()) {
 | |
|             ErrorF("winClipboardProc - winClipboardInit failed.\n");
 | |
|             return NULL;
 | |
|         }
 | |
| 
 | |
|         winDebug("winClipboardProc - winInitClipboard returned.\n");
 | |
|         /* Flag that clipboard client has been launched */
 | |
|         g_fClipboardLaunched = TRUE;
 | |
|     }
 | |
|     else {
 | |
|         ErrorF("winClipboardProc - Clipboard disabled  - Exit from server \n");
 | |
|         /* clipboard thread has exited, stop server as well */
 | |
|         raise(SIGTERM);
 | |
|     }
 | |
| 
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * 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"
 | |
|            "\tSerial: %lu, Request Code: %d, Minor Code: %d\n",
 | |
|            pszErrorMsg, pErr->serial, pErr->request_code, pErr->minor_code);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * winClipboardIOErrorHandler - Our application specific IO error handler
 | |
|  */
 | |
| 
 | |
| static int
 | |
| winClipboardIOErrorHandler(Display * pDisplay)
 | |
| {
 | |
|     ErrorF("winClipboardIOErrorHandler!\n\n");
 | |
| 
 | |
|     if (pthread_equal(pthread_self(), g_winClipboardProcThread)) {
 | |
|         /* Restart at the main entry point */
 | |
|         longjmp(g_jmpEntry, WIN_JMP_ERROR_IO);
 | |
|     }
 | |
| 
 | |
|     if (g_winClipboardOldIOErrorHandler)
 | |
|         g_winClipboardOldIOErrorHandler(pDisplay);
 | |
| 
 | |
|     return 0;
 | |
| }
 |