📄 xtalk.cpp
字号:
//======================================================================
// XTalk - A simple interprocess communication application for Windows CE
//
// Written for the book Programming Windows CE
// Copyright (C) 2007 Douglas Boling
//======================================================================
#include <windows.h> // For all that Windows stuff
#include <commctrl.h> // Command bar includes
#include "xtalk.h" // Program-specific stuff
// The include and lib files for the Windows Mobile are conditionally
// included so that this example can share the same project file. This
// is necessary because this example must have a menu bar on the
// Windows Mobile device to have a SIP button.
#if defined(WIN32_PLATFORM_PSPC)
#include <aygshell.h> // Add Pocket PC includes
#pragma comment( lib, "aygshell" ) // Link Pocket PC lib for menu bar
#endif
//----------------------------------------------------------------------
// Global data
//
const TCHAR szAppName[] = TEXT ("xtalk");
HINSTANCE hInst; // Program instance handle
HANDLE g_hMMObj = 0; // Memory-mapped object
PSHAREBUFF g_pBuff = 0; // Pointer to mm object
HANDLE g_hmWriteOkay = 0; // Write mutex
HANDLE g_hSendEvent = 0; // Local send event
HANDLE g_hReadEvent = 0; // Shared read data event
HANDLE g_hReadDoneEvent = 0; // Shared data read event
HANDLE g_hSendThread = 0; // Sender thread handle
HANDLE g_hReadThread = 0; // Sender thread handle
// Message dispatch table for MainWindowProc
const struct decodeUINT MainMessages[] = {
WM_CREATE, DoCreateMain,
WM_SETFOCUS, DoSetFocusMain,
WM_COMMAND, DoCommandMain,
WM_DESTROY, DoDestroyMain,
};
// Command Message dispatch for MainWindowProc
const struct decodeCMD MainCommandItems[] = {
IDOK, DoMainCommandExit,
IDCANCEL, DoMainCommandExit,
IDD_SENDTEXT, DoMainCommandSend,
};
//======================================================================
// Program entry point
//
int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
LPWSTR lpCmdLine, int nCmdShow) {
MSG msg;
int rc = 0;
HWND hwndMain;
// Initialize application.
hwndMain = InitInstance (hInstance, lpCmdLine, nCmdShow);
if (hwndMain == 0)
return TermInstance (hInstance, 0x10);
// Application message loop
while (GetMessage (&msg, NULL, 0, 0)) {
if ((hwndMain == 0) || !IsDialogMessage (hwndMain, &msg)) {
TranslateMessage (&msg);
DispatchMessage (&msg);
}
}
// Instance cleanup
return TermInstance (hInstance, msg.wParam);
}
//----------------------------------------------------------------------
// InitInstance - Instance initialization
//
HWND InitInstance (HINSTANCE hInstance, LPWSTR lpCmdLine, int nCmdShow){
HWND hWnd;
RECT rect;
int rc;
BOOL fFirstApp = TRUE;
WNDCLASS wc;
// Save program instance handle in global variable.
hInst = hInstance;
// Register application main window class.
wc.style = 0; // Window style
wc.lpfnWndProc = MainWndProc; // Callback function
wc.cbClsExtra = 0; // Extra class data
wc.cbWndExtra = DLGWINDOWEXTRA; // Extra window data
wc.hInstance = hInstance; // Owner handle
wc.hIcon = NULL, // Application icon
wc.hCursor = NULL; // Default cursor
wc.hbrBackground = (HBRUSH) (COLOR_BTNFACE + 1);
wc.lpszMenuName = NULL; // Menu name
wc.lpszClassName = szAppName; // Window class name
if (RegisterClass (&wc) == 0) return 0;
// Create mutex used to share memory-mapped structure.
g_hmWriteOkay = CreateMutex (NULL, TRUE, TEXT ("XTALKWRT"));
rc = GetLastError();
if (rc == ERROR_ALREADY_EXISTS)
fFirstApp = FALSE;
else if (rc) return 0;
// Wait here for ownership to ensure that the initialization is done.
// This is necessary since CreateMutex doesn抰 wait.
rc = WaitForSingleObject (g_hmWriteOkay, 2000);
if (rc != WAIT_OBJECT_0)
return 0;
// Create a file-mapping object.
g_hMMObj = CreateFileMapping (INVALID_HANDLE_VALUE, NULL,
PAGE_READWRITE, 0,
MMBUFFSIZE, TEXT ("XTALKBLK"));
if (g_hMMObj == 0) return 0;
// Map into memory the file-mapping object.
g_pBuff = (PSHAREBUFF)MapViewOfFile (g_hMMObj, FILE_MAP_WRITE,
0, 0, 0);
if (!g_pBuff)
CloseHandle (g_hMMObj);
// Initialize structure if first application started.
if (fFirstApp)
memset (g_pBuff, 0, sizeof (SHAREBUFF));
// Increment app running count. Interlock not needed due to mutex.
g_pBuff->nAppCnt++;
// Release the mutex. We need to release the mutex twice
// if we owned it when we entered the wait above.
ReleaseMutex (g_hmWriteOkay);
if (fFirstApp)
ReleaseMutex (g_hmWriteOkay);
// Now create events for read, and send notification.
g_hSendEvent = CreateEvent (NULL, FALSE, FALSE, NULL);
g_hReadEvent = CreateEvent (NULL, TRUE, FALSE, TEXT ("XTALKREAD"));
g_hReadDoneEvent = CreateEvent (NULL, FALSE, FALSE,
TEXT ("XTALKDONE"));
if (!g_hReadEvent || !g_hSendEvent || !g_hReadDoneEvent)
return 0;
// Create main window.
hWnd = CreateDialog (hInst, szAppName, NULL, NULL);
// Return fail code if window not created.
if (!IsWindow (hWnd)) return 0;
if (!fFirstApp) {
GetWindowRect (hWnd, &rect);
MoveWindow (hWnd, rect.left+10, rect.top+10,
rect.right-rect.left, rect.bottom-rect.top, FALSE);
}
// Create secondary threads for interprocess communication.
g_hSendThread = CreateThread (NULL, 0, SenderThread, hWnd, 0, NULL);
g_hReadThread = CreateThread (NULL, 0, ReaderThread, hWnd, 0, NULL);
if ((g_hSendThread == 0) || (g_hReadThread == 0)) {
DestroyWindow (hWnd);
return 0;
}
// Standard show and update calls
ShowWindow (hWnd, nCmdShow);
UpdateWindow (hWnd);
return hWnd;
}
//----------------------------------------------------------------------
// TermInstance - Program cleanup
//
int TermInstance (HINSTANCE hInstance, int nDefRC) {
// Close event handles.
if (g_hReadEvent)
CloseHandle (g_hReadEvent);
if (g_hReadDoneEvent)
CloseHandle (g_hReadDoneEvent);
if (g_hSendEvent)
CloseHandle (g_hSendEvent);
// Wait for the threads to terminate. They'll do so when the
// handles above are closed since the waits will fail.
if (g_hSendThread) {
WaitForSingleObject (g_hSendThread, 1000);
CloseHandle (g_hSendThread);
}
if (g_hReadThread) {
WaitForSingleObject (g_hReadThread, 1000);
CloseHandle (g_hReadThread);
}
// Free memory-mapped object.
if (g_pBuff) {
// Decrement app running count.
InterlockedDecrement (&g_pBuff->nAppCnt);
UnmapViewOfFile (g_pBuff);
}
if (g_hMMObj)
CloseHandle (g_hMMObj);
// Free mutex.
if (g_hmWriteOkay)
CloseHandle (g_hmWriteOkay);
return nDefRC;
}
//======================================================================
// Message handling procedures for main window
//----------------------------------------------------------------------
// MainWndProc - Callback function for application window
//
LRESULT CALLBACK MainWndProc (HWND hWnd, UINT wMsg, WPARAM wParam,
LPARAM lParam) {
int i;
//
// Search message list to see if we need to handle this
// message. If in list, call procedure.
//
for (i = 0; i < dim(MainMessages); i++) {
if (wMsg == MainMessages[i].Code)
return (*MainMessages[i].Fxn)(hWnd, wMsg, wParam, lParam);
}
return DefWindowProc (hWnd, wMsg, wParam, lParam);
}
//----------------------------------------------------------------------
// DoCreateMain - Process WM_CREATE message for window.
//
LRESULT DoCreateMain (HWND hWnd, UINT wMsg, WPARAM wParam,
LPARAM lParam) {
#if defined(WIN32_PLATFORM_PSPC) || defined(WIN32_PLATFORM_WFSP)
SHMENUBARINFO mbi; // For Pocket PC, create
memset(&mbi, 0, sizeof(SHMENUBARINFO)); // menu bar so that we
mbi.cbSize = sizeof(SHMENUBARINFO); // have a sip button.
mbi.hwndParent = hWnd;
mbi.dwFlags = SHCMBF_EMPTYBAR; // No menu
SHCreateMenuBar(&mbi);
#endif
return 0;
}
//----------------------------------------------------------------------
// DoSetFocusMain - Process WM_SETFOCUS message for window.
//
LRESULT DoSetFocusMain (HWND hWnd, UINT wMsg, WPARAM wParam,
LPARAM lParam) {
SetFocus (GetDlgItem (hWnd, IDD_OUTTEXT));
return 0;
}
//----------------------------------------------------------------------
// DoCommandMain - Process WM_COMMAND message for window.
//
LRESULT DoCommandMain (HWND hWnd, UINT wMsg, WPARAM wParam,
LPARAM lParam) {
WORD idItem, wNotifyCode;
HWND hwndCtl;
int i;
// Parse the parameters.
idItem = (WORD) LOWORD (wParam);
wNotifyCode = (WORD) HIWORD (wParam);
hwndCtl = (HWND) lParam;
// Call routine to handle control message.
for(i = 0; i < dim(MainCommandItems); i++) {
if(idItem == MainCommandItems[i].Code)
return (*MainCommandItems[i].Fxn)(hWnd, idItem, hwndCtl,
wNotifyCode);
}
return 0;
}
//----------------------------------------------------------------------
// DoDestroyMain - Process WM_DESTROY message for window.
//
LRESULT DoDestroyMain (HWND hWnd, UINT wMsg, WPARAM wParam,
LPARAM lParam) {
PostQuitMessage (0);
return 0;
}
//======================================================================
// Command handler routines
//----------------------------------------------------------------------
// DoMainCommandExit - Process Program Exit command.
//
LPARAM DoMainCommandExit (HWND hWnd, WORD idItem, HWND hwndCtl,
WORD wNotifyCode) {
SendMessage (hWnd, WM_CLOSE, 0, 0);
return 0;
}
//----------------------------------------------------------------------
// DoMainCommandSend - Process Program Send command.
//
LPARAM DoMainCommandSend (HWND hWnd, WORD idItem, HWND hwndCtl,
WORD wNotifyCode) {
SetEvent (g_hSendEvent);
return 0;
}
//======================================================================
// SenderThread - Performs the interprocess communication
//
DWORD WINAPI SenderThread (PVOID pArg) {
HWND hWnd;
int nGoCode, rc;
TCHAR szText[TEXTSIZE];
hWnd = (HWND)pArg;
while (1) {
nGoCode = WaitForSingleObject (g_hSendEvent, INFINITE);
if (nGoCode == WAIT_OBJECT_0) {
SendDlgItemMessage (hWnd, IDD_OUTTEXT, WM_GETTEXT,
sizeof (szText), (LPARAM)szText);
rc = WaitForSingleObject (g_hmWriteOkay, 2000);
if (rc == WAIT_OBJECT_0) {
StringCchCopy (g_pBuff->szText, TEXTSIZE, szText);
g_pBuff->nReadCnt = g_pBuff->nAppCnt;
PulseEvent (g_hReadEvent);
// Wait while reader threads get data.
while (g_pBuff->nReadCnt)
rc = WaitForSingleObject (g_hReadDoneEvent,
INFINITE);
ReleaseMutex (g_hmWriteOkay);
}
} else
return -1;
}
return 0;
}
//======================================================================
// ReaderThread - Performs the interprocess communication
//
DWORD WINAPI ReaderThread (PVOID pArg) {
HWND hWnd;
int nGoCode, rc, i;
TCHAR szText[TEXTSIZE];
hWnd = (HWND)pArg;
while (1) {
nGoCode = WaitForSingleObject (g_hReadEvent, INFINITE);
if (nGoCode == WAIT_OBJECT_0) {
i = SendDlgItemMessage (hWnd, IDD_INTEXT, LB_ADDSTRING, 0,
(LPARAM)g_pBuff->szText);
SendDlgItemMessage (hWnd, IDD_INTEXT, LB_SETTOPINDEX, i, 0);
InterlockedDecrement (&g_pBuff->nReadCnt);
SetEvent (g_hReadDoneEvent);
} else {
rc = GetLastError();
wsprintf (szText, TEXT ("rc:%d"), rc);
MessageBox (hWnd, szText, TEXT ("ReadThread Err"), MB_OK);
}
}
return 0;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -