📄 xtalk.c
字号:
//======================================================================
// XTalk - A simple application for Windows CE
//
// Written for the book Programming Windows CE
// Copyright (C) 1998 Douglas Boling
//======================================================================
#include <windows.h> // For all that Windows stuff
#include <commctrl.h> // Command bar includes
#include "xtalk.h" // Program-specific stuff
//----------------------------------------------------------------------
// 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
// Message dispatch table for MainWindowProc
const struct decodeUINT MainMessages[] = {
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.
rc = InitApp (hInstance);
if (rc) return rc;
// Initialize this instance.
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);
}
//----------------------------------------------------------------------
// InitApp - Application initialization
//
int InitApp (HINSTANCE hInstance) {
WNDCLASS wc;
// 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) GetStockObject (WHITE_BRUSH);
wc.lpszMenuName = NULL; // Menu name
wc.lpszClassName = szAppName; // Window class name
if (RegisterClass (&wc) == 0) return 1;
return 0;
}
//----------------------------------------------------------------------
// InitInstance - Instance initialization
//
HWND InitInstance (HINSTANCE hInstance, LPWSTR lpCmdLine, int nCmdShow){
HWND hWnd;
HANDLE hThread;
INT rc;
BOOL fFirstApp = TRUE;
// Save program instance handle in global variable.
hInst = hInstance;
// 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 the initialization is done.
// This is necessary since CreateMutex doesn't wait.
rc = WaitForSingleObject (g_hmWriteOkay, 2000);
if (rc != WAIT_OBJECT_0)
return 0;
// Create a file-mapping object.
g_hMMObj = CreateFileMapping ((HANDLE)-1, 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 mutext 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);
rc = GetLastError();
// Create secondary threads for interprocess communication.
hThread = CreateThread (NULL, 0, SenderThread, hWnd, 0, &rc);
if (hThread)
CloseHandle (hThread);
else {
DestroyWindow (hWnd);
return 0;
}
hThread = CreateThread (NULL, 0, ReaderThread, hWnd, 0, &rc);
if (hThread)
CloseHandle (hThread);
else {
DestroyWindow (hWnd);
return 0;
}
// Return fail code if window not created.
if (!IsWindow (hWnd)) return 0;
// Standard show and update calls
ShowWindow (hWnd, nCmdShow);
UpdateWindow (hWnd);
return hWnd;
}
//----------------------------------------------------------------------
// TermInstance - Program cleanup
//
int TermInstance (HINSTANCE hInstance, int nDefRC) {
// 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);
// Close event handles.
if (g_hReadEvent)
CloseHandle (g_hReadEvent);
if (g_hReadDoneEvent)
CloseHandle (g_hReadDoneEvent);
if (g_hSendEvent)
CloseHandle (g_hSendEvent);
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);
}
//----------------------------------------------------------------------
// 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
//
int 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) {
lstrcpy (g_pBuff->szText, 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
//
int 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 + -