众所周知:WIN32子动态库user32.dll 封装了系统定义的全局标准控件类和WNDPROCESS,然后共享给各个Application,系统启动会加载入user32,然后注册它们的.仅仅是感兴趣没有其他目的
俺是一个小菜鸟,结合2k代码只分析应用层。 机器环境双核 xp sp2。
user32.dll中的函数命名一般有统一性,很有意思.呵呵.算是它的native风格定义吧。它们会进一步校验上层传的参数,并且适当设置GetLastError等细节,然后继续分发到ntdll Native层,或经过系统中断服务层进入win32.sys驱动层完成调用等.
比如:DispatchMessageA 调用DispatchMessageWorker,又而会调用 UserCallWinProcCheckWow,然后最终调用到InternalCallWinProc,把消息分发到我们的窗口的WinProc过程. 这个过程完全是在应用层进行的。
再如:SendMessageA -->SendMessageWorker---> UserCallWinProcCheckWow---> InternalCallWinProc-->MyDiyWinProc。也是在应用层。
这个:DefWindowProc-->DefWindowProcWorker--->InternalCallWinProc--->RealDefWindowProcWorker-->消息表---事件表调度>NtUserMessageCall--->进入内核和win32.sys-->处理完毕,返回应用层。有内核辅助处理。
先看看user32注册控件类的地方:
.text:77D2EC54 __stdcall RW_RegisterControls() proc near
.text:77D2EC54 ; CODE XREF: ClientThreadSetup()+11D p
.text:77D2EC54 ; CtxInitUser32():loc_77D4A7AD p
.text:77D2EC54
.text:77D2EC54 var_30 = dword ptr -30h
.text:77D2EC54 var_2C = dword ptr -2Ch
.text:77D2EC54 var_28 = dword ptr -28h
.text:77D2EC54 var_20 = dword ptr -20h
.text:77D2EC54 var_1C = dword ptr -1Ch
.text:77D2EC54 var_14 = dword ptr -14h
.text:77D2EC54 var_10 = dword ptr -10h
.text:77D2EC54 var_8 = dword ptr -8
.text:77D2EC54
.text:77D2EC54 mov edi, edi
.text:77D2EC56 push ebp
.text:77D2EC57 mov ebp, esp
.text:77D2EC59 sub esp, 30h
.text:77D2EC5C push ebx
.text:77D2EC5D push esi
.text:77D2EC5E push edi
.text:77D2EC5F push 0Ch
.text:77D2EC61 xor eax, eax
.text:77D2EC63 pop ecx
.text:77D2EC64 lea edi, [ebp+var_30]
.text:77D2EC67 rep stosd
.text:77D2EC69 mov eax, _hmodUser
.text:77D2EC6E xor ebx, ebx
.text:77D2EC70 inc ebx
.text:77D2EC71 xor edi, edi
.text:77D2EC73 mov [ebp+var_30], 30h
.text:77D2EC7A mov [ebp+var_1C], eax
.text:77D2EC7D xor esi, esi
.text:77D2EC7D
.text:77D2EC7F
.text:77D2EC7F loc_77D2EC7F:
.text:77D2EC7F mov eax, ds:dword_77D12C58[esi]---------->Win标准控件类表,看下边。
.text:77D2EC85 push ds:dword_77D12C64[esi] ; lpCursorName
.text:77D2EC8B mov [ebp+var_2C], eax
.text:77D2EC8E mov eax, ds:off_77D12C5C[esi]
.text:77D2EC94 mov [ebp+var_28], eax
.text:77D2EC97 mov eax, ds:dword_77D12C60[esi]
.text:77D2EC9D push edi ; hInstance
.text:77D2EC9E mov [ebp+var_20], eax
.text:77D2ECA1 call LoadCursorW(x,x)
.text:77D2ECA1
.text:77D2ECA6 mov [ebp+var_14], eax
.text:77D2ECA9 mov eax, ds:dword_77D12C68[esi]
.text:77D2ECAF mov [ebp+var_10], eax
.text:77D2ECB2 mov eax, ds:off_77D12C6C[esi]
.text:77D2ECB8 mov [ebp+var_8], eax
.text:77D2ECBB xor eax, eax
.text:77D2ECBD mov ax, ds:word_77D12C70[esi]
.text:77D2ECC4 push edi
.text:77D2ECC5 push eax
.text:77D2ECC6 push edi
.text:77D2ECC7 lea eax, [ebp+var_30]
.text:77D2ECCA push eax
.text:77D2ECCB call RegisterClassExWOWW(x,x,x,x)---->RegisterClassEx底层版,参数几乎都差不多,WNDCLASS或WNDCLASSEX,
最终于调到navtive api NtUserRegisterClassExWOW 完成应用层的注册.
.text:77D2ECCB
.text:77D2ECD0 neg ax
.text:77D2ECD3 sbb eax, eax
.text:77D2ECD5 neg eax
.text:77D2ECD7 add esi, 1Ch
.text:77D2ECDA and ebx, eax
.text:77D2ECDC cmp esi, 0FCh
.text:77D2ECE2 jb short loc_77D2EC7F------>循环完毕,注册定义的系统控件类。
.text:77D2ECE2
.text:77D2ECE4 pop edi
.text:77D2ECE5 pop esi
.text:77D2ECE6 mov eax, ebx
.text:77D2ECE8 pop ebx
.text:77D2ECE9 leave
.text:77D2ECEA retn
.text:77D2ECEA
.text:77D2ECEA __stdcall RW_RegisterControls() endp
.text:77D2ECEA
use32.dll库中Win标准控件类表依次是:
Button类和其消息回调函数ButtonWndProcW. (UNICODE版)
ComboLBox类和其消息回调函数ComboBoxWndProcW和ComboListBoxWndProcW (UNICODE版)
Edit类 对应EditWndProcW
ListBox类对应 ComboListBoxWndProcW
MDIClient类 对应 MDIClientWndProcW
IME类对应 ImeWndProcW
Static类StaticWndProcW
此表如下:
.text:77D12C58 dword_77D12C58 dd 408Bh ; DATA XREF: RW_RegisterControls():loc_77D2EC7F r
.text:77D12C5C off_77D12C5C dd offset ButtonWndProcW(x,x,x,x)
.text:77D12C5C ; DATA XREF: RW_RegisterControls()+3A r
.text:77D12C60 dword_77D12C60 dd 4 ; DATA XREF: RW_RegisterControls()+43 r
.text:77D12C64 dword_77D12C64 dd 7F00h ; DATA XREF: RW_RegisterControls()+31 r
.text:77D12C68 dword_77D12C68 dd 0 ; DATA XREF: RW_RegisterControls()+55 r
.text:77D12C6C off_77D12C6C dd offset s_Button ; DATA XREF: RW_RegisterControls()+5E r
.text:77D12C6C ; "Button"
.text:77D12C70 word_77D12C70 dw 2A1h, 0, 408Bh, 0 ; DATA XREF: RW_RegisterControls()+69 r
.text:77D12C78 dd offset ComboBoxWndProcW(x,x,x,x)
.text:77D12C7C dd 4, 7F00h, 0
.text:77D12C88 dd offset s_Combobox ; "ComboBox"
.text:77D12C8C dd 2A2h, 4808h
.text:77D12C94 dd offset ComboListBoxWndProcW(x,x,x,x)
.text:77D12C98 dd 4, 7F00h, 0
.text:77D12CA4 dd offset s_Combolbox ; "ComboLBox"
.text:77D12CA8 dd 2A3h, 4808h
.text:77D12CB0 dd offset DefDlgProcW(x,x,x,x)
.text:77D12CB4 dd 1Eh, 7F00h, 0
.text:77D12CC0 dd 8002h, 2A4h, 4088h
.text:77D12CCC dd offset EditWndProcW(x,x,x,x)
.text:77D12CD0 dd 6, 7F01h, 0
.text:77D12CDC dd offset s_Edit ; "Edit"
.text:77D12CE0 dd 2A5h, 4088h
.text:77D12CE8 dd offset ComboListBoxWndProcW(x,x,x,x)
.text:77D12CEC dd 4, 7F00h, 0
.text:77D12CF8 dd offset s_Listbox ; "ListBox"
.text:77D12CFC dd 2A6h, 4000h
.text:77D12D04 dd offset MDIClientWndProcW(x,x,x,x)
.text:77D12D08 dd 8, 7F00h, 0Dh
.text:77D12D14 dd offset s_Mdiclient ; "MDIClient"
.text:77D12D18 dd 2A7h, 4000h
.text:77D12D20 dd offset ImeWndProcW(x,x,x,x)
.text:77D12D24 dd 4, 7F00h, 0
.text:77D12D30 dd offset s_Ime ; "IME"
.text:77D12D34 db 0A9h, 2, 2 dup(0), 88h, 40h, 2 dup(0)
.text:77D12D3C dd offset StaticWndProcW(x,x,x,x)
.text:77D12D40 dd 4, 7F00h, 0
.text:77D12D4C dd offset s_Static ; "Static"
.text:77D12D50 dd 2A8h, 0
呵呵,顺便把win2k代码复制出来一份给大家参考:
BOOL RW_RegisterControls(VOID)
{
int i;
WNDCLASSEX wndcls;
BOOL fSuccess = TRUE;
static CONST struct {
UINT style;
WNDPROC lpfnWndProcW;
int cbWndExtra;
LPCTSTR lpszCursor;
HBRUSH hbrBackground;
LPCTSTR lpszClassName;
WORD fnid;----------------------------------->这个函数ID很有意思.我想这是微软定义区分系统控件的WNDPROC函数ID,
} rc[] = { //并且在MessageTable中也多次用这个fnid号来区分是哪个控件WNDPROC
// ,然后调用正确的WNDPROC地址。还有可能这个MessageTable表可以给窗体 //回调函数进程Hook行为。还有这个定义的结构体与WNDCLASS和WNDCLASSEX有 // 区别 !
{CS_GLOBALCLASS | CS_PARENTDC | CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW,
ButtonWndProcW,
sizeof(BUTNWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"Button",
FNID_BUTTON
},
{CS_GLOBALCLASS | CS_DBLCLKS | CS_PARENTDC | CS_VREDRAW | CS_HREDRAW,
ComboBoxWndProcW,
sizeof(COMBOWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"ComboBox",
FNID_COMBOBOX
},
{CS_GLOBALCLASS | CS_DBLCLKS | CS_SAVEBITS,
ComboListBoxWndProcW,
sizeof(LBWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"ComboLBox",
FNID_COMBOLISTBOX
},
{CS_GLOBALCLASS | CS_DBLCLKS | CS_SAVEBITS,
DefDlgProcW,
DLGWINDOWEXTRA,
IDC_ARROW,
NULL,
DIALOGCLASS,
FNID_DIALOG
},
{CS_GLOBALCLASS | CS_PARENTDC | CS_DBLCLKS,
EditWndProcW,
max((sizeof(EDITWND) - sizeof(WND)), CBEDITEXTRA),
IDC_IBEAM,
NULL,
L"Edit",
FNID_EDIT
},
{CS_GLOBALCLASS | CS_PARENTDC | CS_DBLCLKS,
ListBoxWndProcW,
sizeof(LBWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"ListBox",
FNID_LISTBOX
},
{CS_GLOBALCLASS,
MDIClientWndProcW,
sizeof(MDIWND) - sizeof(WND),
IDC_ARROW,
(HBRUSH)(COLOR_APPWORKSPACE + 1),
L"MDIClient",
FNID_MDICLIENT
},
{CS_GLOBALCLASS,
ImeWndProcW,
sizeof(IMEWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"IME",
FNID_IME
},
{CS_GLOBALCLASS | CS_PARENTDC | CS_DBLCLKS,
StaticWndProcW,
sizeof(STATWND) - sizeof(WND),
IDC_ARROW,
NULL,
L"Static",
FNID_STATIC
}
};
/*
* Classes are registered via the table.
*/
RtlZeroMemory(&wndcls, sizeof(wndcls));
wndcls.cbSize = sizeof(wndcls);
wndcls.hInstance = hmodUser;
for (i = 0; i < (sizeof(rc)/sizeof(rc[0])); i++) {
wndcls.style = rc[i].style;
wndcls.lpfnWndProc = rc[i].lpfnWndProcW;
wndcls.cbWndExtra = rc[i].cbWndExtra;
wndcls.hCursor = LoadCursor(NULL, rc[i].lpszCursor);
wndcls.hbrBackground= rc[i].hbrBackground;
wndcls.lpszClassName= rc[i].lpszClassName;
fSuccess &= !!RegisterClassExWOWW(&wndcls, NULL, rc[i].fnid);
}
if (!fSuccess) {
RIPMSG0(RIP_WARNING, "RW_RegisterControls failed to register classes");
}
return fSuccess;
}
Win2k头文件 user.h中找了fnid的说明,fnid更类似(窗口类的)索引,有起始范围。有兴趣的可以看看.
/*
* Server side address constants. When we want to call a server side proc,
* we pass an index indentifying the function, rather than the server side
* address itself. More robust. The functions between WNDPROCSTART/END
* have client side sutbs which map to this routines.
*
* Adding a new FNID (This is just what I figured out...so fix it if wrong or incomplete)
* -Decide what range it should be in:
* FNID_WNDPROCSTART to FNID_WNDPROCEND: Server side proc with client
* stub
* FIND_CONTROLSTART to FNID_CONTROLEND: Client side controls with no
* server side proc
* After FNID_CONTROLEND: other, like server side only procs or client
* side only....
* -Make sure to adjust FNID_*START and FNID_*END appropriately.
* -If the ID is to be associated with a window class, and it is for all
* windows of the class, make sure that the InternalRegisterClassEx call
* receives the id as a parameter.
* -If in FNID_WNDPROCSTART-END range, make the proper STOCID call in InitFunctionTables.
* -Add proper FNID call in InitFunctionTables.
* -If the class has a client side worker function (pcls->lpfnWorker) or you expect
* apps to send messages to it or call its window proc directly, define
* a message table in kernel\server.c and initialize it in InitMessageTables.
* -If there is a client side for this proc, you probably need to add it to
* PFNCLIENT.
* -Add the debug-only text description of this FNID to in gapszFNID in globals.c
* -See if you need to modify aiClassWow in client\client.c
* -Modify the gaFNIDtoICLS table in kernel\ntstubs.c
*/
#define FNID_START 0x0000029A
#define FNID_WNDPROCSTART 0x0000029A
#define FNID_SCROLLBAR 0x0000029A // xxxSBWndProc;
#define FNID_ICONTITLE 0x0000029B // xxxDefWindowProc;
#define FNID_MENU 0x0000029C // xxxMenuWindowProc;
#define FNID_DESKTOP 0x0000029D // xxxDesktopWndProc;
#define FNID_DEFWINDOWPROC 0x0000029E // xxxDefWindowProc;
#define FNID_WNDPROCEND 0x0000029E // see PatchThreadWindows
#define FNID_CONTROLSTART 0x0000029F
#define FNID_BUTTON 0x0000029F // No server side proc
#define FNID_COMBOBOX 0x000002A0 // No server side proc
#define FNID_COMBOLISTBOX 0x000002A1 // No server side proc
#define FNID_DIALOG 0x000002A2 // No server side proc
#define FNID_EDIT 0x000002A3 // No server side proc
#define FNID_LISTBOX 0x000002A4 // No server side proc
#define FNID_MDICLIENT 0x000002A5 // No server side proc
#define FNID_STATIC 0x000002A6 // No server side proc
#define FNID_IME 0x000002A7 // No server side proc
#define FNID_CONTROLEND 0x000002A7
#define FNID_HKINLPCWPEXSTRUCT 0x000002A8
#define FNID_HKINLPCWPRETEXSTRUCT 0x000002A9
#define FNID_DEFFRAMEPROC 0x000002AA // No server side proc
#define FNID_DEFMDICHILDPROC 0x000002AB // No server side proc
#define FNID_MB_DLGPROC 0x000002AC // No server side proc
#define FNID_MDIACTIVATEDLGPROC 0x000002AD // No server side proc
#define FNID_SENDMESSAGE 0x000002AE
#define FNID_SENDMESSAGEFF 0x000002AF
#define FNID_SENDMESSAGEEX 0x000002B0
#define FNID_CALLWINDOWPROC 0x000002B1
#define FNID_SENDMESSAGEBSM 0x000002B2
#define FNID_SWITCH 0x000002B3 // Just used by GetTopMostInserAfter
#define FNID_TOOLTIP 0x000002B4
#define FNID_END 0x000002B4
注册完了系统控件类,然后随便拉个控件Button类的回调看看,只是感兴趣:
text:77D2614C ; int __stdcall ButtonWndProcW(int,UINT Msg,int wParam,int lParam)
.text:77D2614C __stdcall ButtonWndProcW(x, x, x, x) proc near
.text:77D2614C
.text:77D2614C
.text:77D2614C arg_0 = dword ptr 8
.text:77D2614C Msg = dword ptr 0Ch
.text:77D2614C wParam = dword ptr 10h
.text:77D2614C lParam = dword ptr 14h
.text:77D2614C
.text:77D2614C mov edi, edi
.text:77D2614E push ebp
.text:77D2614F mov ebp, esp
.text:77D26151 mov ecx, [ebp+arg_0]
.text:77D26154 push esi
.text:77D26155 call ValidateHwnd(x)
.text:77D26155
.text:77D2615A mov esi, eax
.text:77D2615C test esi, esi
.text:77D2615E jz short loc_77D26198
.text:77D2615E
.text:77D26160 mov edx, [ebp+Msg]
.text:77D26163 cmp edx, dword_77D700E8
.text:77D26169 ja short loc_77D26189
.text:77D26169
.text:77D2616B xor eax, eax
.text:77D2616D mov ecx, edx
.text:77D2616F and ecx, 7
.text:77D26172 inc eax
.text:77D26173 shl eax, cl
.text:77D26175 push edi
.text:77D26176 mov edi, dword_77D700EC
.text:77D2617C mov ecx, edx
.text:77D2617E shr ecx, 3
.text:77D26181 mov cl, [ecx+edi]
.text:77D26184 test al, cl
.text:77D26186 pop edi
.text:77D26187 jnz short loc_77D2619D
.text:77D26187
.text:77D26189
.text:77D26189 loc_77D26189:
.text:77D26189 push 0 ; int
.text:77D2618B push [ebp+lParam] ; lParam
.text:77D2618E push [ebp+wParam] ; wParam
.text:77D26191 push edx ; Msg
.text:77D26192 push esi ; int
.text:77D26193 call DefWindowProcWorker(x,x,x,x,x)
.text:77D26193
.text:77D26198
.text:77D26198 loc_77D26198:
.text:77D26198
.text:77D26198 pop esi
.text:77D26199 pop ebp
.text:77D2619A retn 10h
.text:77D2619A
.text:77D2619D ; ---------------------------------------------------------------------------
.text:77D2619D
.text:77D2619D loc_77D2619D:
.text:77D2619D push 0 ; int
.text:77D2619F push [ebp+lParam] ; int
.text:77D261A2 push [ebp+wParam] ; int
.text:77D261A5 push edx ; Msg
.text:77D261A6 push esi ; int
.text:77D261A7 call ButtonWndProcWorker(x,x,x,x,x)------>
.text:77D261A7
.text:77D261AC jmp short loc_77D26198
.text:77D261AC
.text:77D261AC __stdcall ButtonWndProcW(x, x, x, x) endp
参考它的win32k代码如下:
\win2k\private\ntos\w32\ntuser\client\btnctl.c
LRESULT WINAPI ButtonWndProcW(
HWND hwnd,
UINT message,
WPARAM wParam,
LPARAM lParam)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return (0L);
}
/*
* If the control is not interested in this message,
* pass it to DefWindowProc.
*/
if (!FWINDOWMSG(message, FNID_BUTTON))------------------>这里识别FNID_BUTTON ID了.
return DefWindowProcWorker(pwnd, message, wParam, lParam, FALSE);
return ButtonWndProcWorker(pwnd, message, wParam, lParam, FALSE);
}
ButtonWndProcWorker 这个方法处理控件Button的所关心Window消息,本身的控件消息,硬件消息等,
精简了一下,目的是看看它处理的消息。有兴趣的朋友查看\win2k\private\ntos\w32\ntuser\client\btnctl.c。
RESULT APIENTRY ButtonWndProcWorker(
PWND pwnd,
UINT message,
WPARAM wParam,
LPARAM lParam,
DWORD fAnsi)
{
switch (message) {
case WM_NCHITTEST:
...
break;
case WM_ERASEBKGND:
...
return (LONG)TRUE;
case WM_PRINTCLIENT:
xxxBNPaint(pbutn, (HDC)wParam);
break;
case WM_PAINT:
....
break;
case WM_SETFOCUS:
....
break;
case WM_GETDLGCODE:
case WM_CAPTURECHANGED:
....
break;
case WM_KILLFOCUS:
......
break;
case WM_LBUTTONDBLCLK:
....
case WM_LBUTTONUP:
if (BUTTONSTATE(pbutn) & BST_MOUSE) {
xxxBNReleaseCapture(pbutn, TRUE);
}
break;
case WM_MOUSEMOVE:
......
break;
case WM_LBUTTONDOWN:
......
break;
case WM_CHAR:
......
break;
case BM_CLICK:
......
break;
case WM_KEYDOWN:
......
break;
case WM_KEYUP:
case WM_SYSKEYUP:
......
break;
case BM_GETSTATE:
......
break;
case BM_SETSTATE:
......
break;
case BM_GETCHECK:
case BM_SETCHECK:
......
break;
case BM_SETSTYLE:
......
break;
case WM_SETTEXT:
case WM_ENABLE:
......
break;
case WM_SETFONT:
......
break;
case WM_GETFONT:
......
break;
case BM_GETIMAGE:
case BM_SETIMAGE:
......
break;
case WM_NCDESTROY:
case WM_FINALDESTROY:
....
break;
case WM_NCCREATE:
....
break;
case WM_INPUTLANGCHANGEREQUEST:
...
break;
case WM_UPDATEUISTATE:
...
break;
default:
CallDWP:
return DefWindowProcWorker(pwnd, message, wParam, lParam, fAnsi);
}
return 0L;
}
另外user32有两个辅助表用于继续分发消息,比较有意思!
一个是消息表:MessageTable。记的笨笨雄也以前提出来过这个东东!
一个是消息所对应跳转的事件表:gapfnScSendMessage 。
两个表在下面给出:
首先根据消息的ID在MessageTable表取出消息ID所对应的sig,然后与3F相加,
得出的结果是它所对应事件表gapfnScSendMessage的索引然后跳到到它的事件地址中。
随便找个消息分发函数比如:DefWindowProc--->RealDefWindowProc--> RealDefWindowProcWorker,
真正用到MessageTable 索引是到了RealDefWindowProcWorker这个函数中.win2k代码好象没有找到个函数!
这个过程用od跟不错,od跟踪到这里:
77D1B3FD 81FF 00040000 cmp edi, 400------------>如果消息ID超出WM_USER(0x0400),不用MessageTalbe
77D1B403 FF75 18 push dword ptr [ebp+18]----WParam
77D1B406 68 9E020000 push 29E
77D1B40B 6A 00 push 0
77D1B40D FF75 14 push dword ptr [ebp+14]--->LPRAM
77D1B410 FF75 10 push dword ptr [ebp+10]
77D1B413 57 push edi------->msg
77D1B414 50 push eax
77D1B415 0F83 4AF70200 jnb 77D4AB65----->上面的判断跳转
77D1B41B 33C9 xor ecx, ecx
77D1B41D 8A8F E814D177 mov cl, byte ptr [edi+77D114E8]---->7D114E8就是MessageTable表,edi是消息ID号!
77D1B423 83E1 3F and ecx, 3F------>运算一下
77D1B426 FF148D E818D177 call dword ptr [ecx*4+77D118E8] ------->77D118E8就是gapfnScSendMessage的地址,ecx*4它的索引!
77D1B42D ^ E9 FAFEFFFF jmp 77D1B32C
上面汇编对应的win2k代码:
pmsg->message >= WM_USER ? (ULONG_PTR)SfnDWORD :
(ULONG_PTR)gapfnScSendMessage[MessageTable[pmsg->message].iFunction];
win2k定义MessageTable表部分如下(message.h):
CONST MSG_TABLE_ENTRY MessageTable[] = {
{IMSG_DWORD, FALSE, FALSE}, // WM_NULL 0x0000
{IMSG_INLPCREATESTRUCT, TRUE, TRUE}, // WM_CREATE 0x0001
{IMSG_DWORD, FALSE, FALSE}, // WM_DESTROY 0x0002
{IMSG_DWORD, FALSE, FALSE}, // WM_MOVE 0x0003
{IMSG_DWORD, FALSE, FALSE}, // WM_SIZEWAIT 0x0004
{IMSG_DWORD, FALSE, FALSE}, // WM_SIZE 0x0005
{IMSG_DWORD, FALSE, FALSE}, // WM_ACTIVATE 0x0006
{IMSG_DWORD, FALSE, FALSE}, // WM_SETFOCUS 0x0007
{IMSG_DWORD, FALSE, FALSE}, // WM_KILLFOCUS 0x0008
{IMSG_DWORD, FALSE, FALSE}, // WM_SETVISIBLE 0x0009
{IMSG_DWORD, FALSE, FALSE}, // WM_ENABLE 0x000A
{IMSG_DWORD, FALSE, FALSE}, // WM_SETREDRAW 0x000B
{IMSG_INSTRINGNULL, TRUE, TRUE}, // WM_SETTEXT 0x000C
{IMSG_OUTSTRING, TRUE, TRUE}, // WM_GETTEXT 0x000D
{IMSG_GETDBCSTEXTLENGTHS, TRUE, TRUE}, // WM_GETTEXTLENGTH 0x000E
{IMSG_DWORD, FALSE, FALSE}, // WM_PAINT 0x000F
{IMSG_DWORD, FALSE, FALSE}, // WM_CLOSE 0x0010
{IMSG_DWORD, FALSE, FALSE}, // WM_QUERYENDSESSION 0x0011
{IMSG_DWORD, FALSE, FALSE}, // WM_QUIT 0x0012
{IMSG_DWORD, FALSE, FALSE}, // WM_QUERYOPEN 0x0013
{IMSG_DWORD, FALSE, TRUE}, // WM_ERASEBKGND 0x0014
{IMSG_DWORD, FALSE, FALSE}, // WM_SYSCOLORCHANGE 0x0015
{IMSG_DWORD, FALSE, FALSE}, // WM_ENDSESSION 0x0016
{IMSG_DWORD, FALSE, FALSE}, // WM_SYSTEMERROR 0x0017
{IMSG_DWORD, FALSE, FALSE}, // WM_SHOWWINDOW 0x0018
{IMSG_RESERVED, FALSE, FALSE}, // WM_CTLCOLOR 0x0019
{IMSG_INSTRINGNULL, TRUE, TRUE}, // WM_WININICHANGE 0x001A
{IMSG_INSTRING, TRUE, TRUE}, // WM_DEVMODECHANGE 0x001B
{IMSG_DWORD, FALSE, FALSE}, // WM_ACTIVATEAPP 0x001C
{IMSG_DWORD, FALSE, FALSE}, // WM_FONTCHANGE 0x001D
{IMSG_DWORD, FALSE, FALSE}, // WM_TIMECHANGE 0x001E
{IMSG_DWORD, FALSE, FALSE}, // WM_CANCELMODE 0x001F
{IMSG_DWORD, FALSE, FALSE}, // WM_SETCURSOR 0x0020
{IMSG_DWORD, FALSE, FALSE}, // WM_MOUSEACTIVATE 0x0021
{IMSG_DWORD, FALSE, FALSE}, // WM_CHILDACTIVATE 0x0022
{IMSG_DWORD, FALSE, FALSE}, // WM_QUEUESYNC 0x0023
{IMSG_INOUTLPPOINT5, FALSE, TRUE}, // WM_GETMINMAXINFO 0x0024
{IMSG_EMPTY, FALSE, FALSE}, // empty 0x0025
{IMSG_DWORD, FALSE, FALSE}, // WM_PAINTICON 0x0026
{IMSG_DWORD, FALSE, TRUE}, // WM_ICONERASEBKGND 0x0027
{IMSG_DWORD, FALSE, FALSE}, // WM_NEXTDLGCTL 0x0028
{IMSG_DWORD, FALSE, FALSE}, // WM_ALTTABACTIVE 0x0029
{IMSG_DWORD, FALSE, FALSE}, // WM_SPOOLERSTATUS 0x002A
{IMSG_INLPDRAWITEMSTRUCT, FALSE, TRUE}, // WM_DRAWITEM 0x002B
{IMSG_INOUTLPMEASUREITEMSTRUCT, FALSE, TRUE},// WM_MEASUREITEM 0x002C
{IMSG_INLPDELETEITEMSTRUCT, FALSE, TRUE}, // WM_DELETEITEM 0x002D
{IMSG_DWORD, FALSE, FALSE}, // WM_VKEYTOITEM 0x002E
{IMSG_INWPARAMCHAR, TRUE, FALSE}, // WM_CHARTOITEM 0x002F
{IMSG_DWORD, FALSE, FALSE}, // WM_SETFONT 0x0030
{IMSG_DWORD, FALSE, TRUE}, // WM_GETFONT 0x0031
{IMSG_DWORD, FALSE, FALSE}, // WM_SETHOTKEY 0x0032
{IMSG_DWORD, FALSE, FALSE}, // WM_GETHOTKEY 0x0033
{IMSG_DWORD, FALSE, FALSE}, // WM_FILESYSCHANGE 0x0034
{IMSG_DWORD, FALSE, FALSE}, // WM_ISACTIVEICON 0x0035
{IMSG_DWORD, FALSE, FALSE}, // WM_QUERYPARKICON 0x0036
{IMSG_DWORD, FALSE, FALSE}, // WM_QUERYDRAGICON 0x0037
{IMSG_INLPHLPSTRUCT, FALSE, TRUE}, // WM_WINHELP 0x0038
{IMSG_INLPCOMPAREITEMSTRUCT, FALSE, TRUE}, // WM_COMPAREITEM 0x0039
{IMSG_DWORD, FALSE, FALSE}, // WM_FULLSCREEN 0x003A
{IMSG_DWORD, FALSE, FALSE}, // WM_CLIENTSHUTDOWN 0x003B
{IMSG_KERNELONLY, FALSE, TRUE}, // WM_DDEMLEVENT 0x003C
{IMSG_EMPTY, FALSE, FALSE}, // empty 0x003D
{IMSG_EMPTY, FALSE, FALSE}, // empty 0x003E
{IMSG_DWORD, FALSE, FALSE}, // MM_CALCSCROLL 0x003F
{IMSG_RESERVED, FALSE, FALSE}, // WM_TESTING 0x0040
{IMSG_DWORD, FALSE, FALSE}, // WM_COMPACTING 0x0041
{IMSG_RESERVED, FALSE, FALSE}, // WM_OTHERWINDOWCREATED 0x0042
{IMSG_RESERVED, FALSE, FALSE}, // WM_OTHERWINDOWDESTROYED 0x0043
{IMSG_RESERVED, FALSE, FALSE}, // WM_COMMNOTIFY 0x0044
{IMSG_RESERVED, FALSE, FALSE}, // WM_MEDIASTATUSCHANGE 0x0045
{IMSG_INOUTLPWINDOWPOS, FALSE, TRUE}, // WM_WINDOWPOSCHANGING 0x0046
{IMSG_INLPWINDOWPOS, FALSE, TRUE}, // WM_WINDOWPOSCHANGED 0x0047
{IMSG_RESERVED, FALSE, FALSE}, // WM_POWER 0x0048
{IMSG_COPYGLOBALDATA, TRUE, TRUE}, // WM_COPYGLOBALDATA 0x0049
{IMSG_COPYDATA, FALSE, TRUE}, // WM_COPYDATA 0x004A
{IMSG_RESERVED, FALSE, FALSE}, // WM_CANCELJOURNAL 0x004B
{IMSG_LOGONNOTIFY, FALSE, FALSE}, // WM_LOGONNOTIFY 0x004C
{IMSG_DWORD, FALSE, FALSE}, // WM_KEYF1 0x004D
{IMSG_DWORD, FALSE, FALSE}, // WM_NOTIFY 0x004E
{IMSG_RESERVED, FALSE, FALSE}, // WM_ACCESS_WINDOW 0x004f
...
...
...
}
对应跳转的事件表:gapfnScSendMessage
/*
* Message thunks.
*/
typedef LRESULT (APIENTRY *SFNSCSENDMESSAGE)(PWND, UINT, WPARAM, LPARAM, //定义在userk.h
ULONG_PTR, PROC, DWORD, PSMS);
extern CONST SFNSCSENDMESSAGE gapfnScSendMessage[];//定义在globals.h
gapfnScSendMessage 表如下:
.text:77D118E8 gapfnScSendMessage dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D118EC dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D118F0 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D118F4 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D118F8 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D118FC dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11900 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11904 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11908 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1190C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11910 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11914 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11918 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1191C off_77D1191C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1191C ; DATA XREF: RealDefWindowProcWorker(x,x,x,x,x)+1DF0 r
.text:77D11920 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11924 dd offset fnCOPYGLOBALDATA(x,x,x,x,x,x,x)
.text:77D11928 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1192C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11930 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11934 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11938 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1193C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11940 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11944 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11948 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1194C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11950 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11954 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11958 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1195C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11960 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11964 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11968 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1196C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11970 dd offset fnINDEVICECHANGE(x,x,x,x,x,x,x)
.text:77D11974 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11978 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1197C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11980 dd offset fnINSIZECLIPBRD(x,x,x,x,x,x,x)
.text:77D11984 dd offset fnINSIZECLIPBRD(x,x,x,x,x,x,x)
.text:77D11988 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1198C dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11990 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11994 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D11998 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D1199C dd offset fnCBGETEDITSEL(x,x,x,x,x,x,x)
.text:77D119A0 dd offset fnEMSETSEL(x,x,x,x,x,x,x)
.text:77D119A4 dd offset fnINWPARAMDBCSCHAR(x,x,x,x,x,x,x)
.text:77D119A8 dd offset fnCBGETEDITSEL(x,x,x,x,x,x,x)
.text:77D119AC dd offset fnIMECONTROL(x,x,x,x,x,x,x)
.text:77D119B0 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D119B4 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D119B8 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D119BC dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D119C0 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
.text:77D119C4 dd offset NtUserMessageCall(x,x,x,x,x,x,x)
NtUserMessageCall是个native api,它又跑到系统核心层去了。其他的fnCBGETEDITSEL,fnIMECONTROL等感兴趣的自己跟或者查看2k代码。
最后给出DefWindowProcWorker关系处理的win消息,感兴趣的可以看看精简了一下:
LRESULT DefWindowProcWorker( //定义在clmsg.c中
PWND pwnd,
UINT message,
WPARAM wParam,
LPARAM lParam,
DWORD fAnsi)
{
switch (message) {
case WM_HELP:
return(0L);
case WM_MOUSEWHEEL:
break;
......
case WM_CONTEXTMENU:
break;
......
case WM_RBUTTONUP:
case WM_APPCOMMAND:
......
break;
case WM_NCXBUTTONUP:
case WM_XBUTTONUP:
break;
case WM_WINDOWPOSCHANGED: {
PWINDOWPOS ppos = (PWINDOWPOS)lParam;
return 0;
}
case WM_MOUSEACTIVATE: {
......
return ((LOWORD(lParam) == HTCAPTION) && (HIWORD(lParam) == WM_LBUTTONDOWN )) ?
(LONG)MA_NOACTIVATE : (LONG)MA_ACTIVATE;
}
case WM_CTLCOLORSCROLLBAR:
return((LRESULT)gpsi->hbrGray);
}
case WM_CTLCOLORBTN:
goto SetColor;
case WM_CTLCOLORSTATIC:
case WM_CTLCOLORDLG:
case WM_CTLCOLORMSGBOX:
......
case WM_CTLCOLOR: // here for WOW only
case WM_CTLCOLORLISTBOX:
case WM_CTLCOLOREDIT:
......
case WM_NCHITTEST:
return FindNCHit(pwnd, (LONG)lParam);
case WM_GETTEXT:
......
return cchSrc;
case WM_GETTEXTLENGTH:
......
return cch;
}
return 0L;
case WM_QUERYDRAGICON:
......
return (LRESULT)LoadIconW(pwnd->hModule, MAKEINTRESOURCE(1));
case WM_QUERYOPEN:
case WM_QUERYENDSESSION:
case WM_DEVICECHANGE:
case WM_POWERBROADCAST:
return TRUE;
case WM_KEYDOWN:
if (wParam == VK_F10) {
return CsSendMessage(hwnd, message, wParam, lParam, 0L,
FNID_DEFWINDOWPROC, fAnsi);
}
break;
case WM_SYSKEYDOWN:
if ((HIWORD(lParam) & SYS_ALTERNATE) || (wParam == VK_F10) ||
(wParam == VK_ESCAPE))
return CsSendMessage(hwnd, message, wParam, lParam, 0L,
FNID_DEFWINDOWPROC, fAnsi);
break;
case WM_CHARTOITEM:
case WM_VKEYTOITEM:
/*
* Do default processing for keystrokes into owner draw listboxes.
*/
return -1;
case WM_ACTIVATE:
if (LOWORD(wParam))
return CsSendMessage(hwnd, message, wParam, lParam, 0L,
FNID_DEFWINDOWPROC, fAnsi);
break;
case WM_SHOWWINDOW:
if (lParam != 0)
return CsSendMessage(hwnd, message, wParam, lParam, 0L,
FNID_DEFWINDOWPROC, fAnsi);
break;
case WM_DROPOBJECT:
return DO_DROPFILE;
case WM_WINDOWPOSCHANGING:
/*
* If the window's size is changing, adjust the passed-in size
*/
#define ppos ((WINDOWPOS *)lParam)
if (!(ppos->flags & SWP_NOSIZE))
return CsSendMessage(hwnd, message, wParam, lParam, 0L,
FNID_DEFWINDOWPROC, fAnsi);
#undef ppos
break;
case WM_KLUDGEMINRECT:
......
break;
...............................
}
其他的公共控件见comdlg32等库.敢兴趣的自己鼓捣吧,提供一个小思路,虽然2k代码有点老,不过参考还是很不错!
基本完毕了,没有哈目的,纯熟兴趣!
俺是一只小菜鸟,多谢大家指教!!!