分析代码

实战部分

DebugView Not Connected

打印机删掉
添加串口

新建MFC动态库

image.png
image.png

MFCLEI.dll

  1. // MFCLEI.cpp: 定义 DLL 的初始化例程。
  2. //
  3. #include "stdafx.h"
  4. #include "MFCLEI.h"
  5. #ifdef _DEBUG
  6. #define new DEBUG_NEW
  7. #endif
  8. //
  9. //TODO: 如果此 DLL 相对于 MFC DLL 是动态链接的,
  10. // 则从此 DLL 导出的任何调入
  11. // MFC 的函数必须将 AFX_MANAGE_STATE 宏添加到
  12. // 该函数的最前面。
  13. //
  14. // 例如:
  15. //
  16. // extern "C" BOOL PASCAL EXPORT ExportedFunction()
  17. // {
  18. // AFX_MANAGE_STATE(AfxGetStaticModuleState());
  19. // // 此处为普通函数体
  20. // }
  21. //
  22. // 此宏先于任何 MFC 调用
  23. // 出现在每个函数中十分重要。 这意味着
  24. // 它必须作为以下项中的第一个语句:
  25. // 出现,甚至先于所有对象变量声明,
  26. // 这是因为它们的构造函数可能生成 MFC
  27. // DLL 调用。
  28. //
  29. // 有关其他详细信息,
  30. // 请参阅 MFC 技术说明 33 和 58。
  31. //
  32. // CMFCLEIApp
  33. BEGIN_MESSAGE_MAP(CMFCLEIApp, CWinApp)
  34. END_MESSAGE_MAP()
  35. // CMFCLEIApp 构造
  36. CMFCLEIApp::CMFCLEIApp()
  37. {
  38. // TODO: 在此处添加构造代码,
  39. // 将所有重要的初始化放置在 InitInstance 中
  40. }
  41. // 唯一的 CMFCLEIApp 对象
  42. CMFCLEIApp theApp;
  43. HWND g_Wnd;
  44. WNDPROC g_OldhProc;
  45. PWORD g_pHeight = (PWORD)0x1005338;
  46. PWORD g_pWidth =(PWORD)0x1005334;
  47. PWORD g_pMineCount = (PWORD)0x1005330;
  48. #define MINE 0x8F
  49. PBYTE g_pBase = (PBYTE)0x1005340;
  50. // CMFCLEIApp 初始化
  51. LRESULT CALLBACK WindowProc(
  52. _In_ HWND hWnd,
  53. _In_ UINT Msg,
  54. _In_ WPARAM wParam,
  55. _In_ LPARAM lParam)
  56. {
  57. if (Msg == WM_KEYDOWN && wParam == VK_F5)//如果键盘按下并且是f5
  58. {
  59. //一件秒杀
  60. OutputDebugString(L"F5");
  61. int nHeight = *g_pHeight;
  62. int nWidth = *g_pWidth;
  63. int nMineCount = *g_pMineCount;
  64. CString strString;
  65. strString.Format(L"宽度:%d, 高度:%d, 雷数:%d", nHeight, nWidth, nMineCount);
  66. OutputDebugString(strString.GetBuffer());
  67. //雷区基地址
  68. //雷区中的元素标识
  69. //01002EE4 | . 8B0D 34530001 MOV ECX, DWORD PTR DS : [<宽度x>]; 宽度
  70. //01002EEA | . 8B15 38530001 MOV EDX, DWORD PTR DS : [<高度y>]; 高度
  71. //01002EF0 | . 8D41 02 LEA EAX, DWORD PTR DS : [ECX + 0x2]
  72. //01002EF3 | . 85C0 TEST EAX, EAX
  73. //01002EF5 | . 56 PUSH ESI
  74. //01002EF6 | . 74 19 JE SHORT winmine.01002F11
  75. //01002EF8 | . 8BF2 MOV ESI, EDX
  76. //01002EFA | .C1E6 05 SHL ESI, 0x5
  77. //01002EFD | . 8DB6 60530001 LEA ESI, DWORD PTR DS : [ESI + 0x1005360]
  78. //01002F03 | > 48 / DEC EAX
  79. //01002F04 | .C680 40530001 > | MOV BYTE PTR DS : [EAX + <数组基地址>], 0x10
  80. //01002F0B | .C60406 10 | MOV BYTE PTR DS : [ESI + EAX], 0x10
  81. //01002F0F | . ^ 75 F2 \JNZ SHORT winmine.01002F03
  82. int nFindCount = 0;
  83. int y = 1;
  84. for (int y=1;y<nHeight;y++)
  85. {
  86. CString strLine;
  87. for (int x = 1; x < nWidth + 1; x++)
  88. {
  89. BYTE byCode = *(PBYTE)((DWORD)g_pBase + x + y * 32);
  90. if (byCode == MINE)
  91. {
  92. nFindCount++;
  93. int xPos, yPos;
  94. xPos = (x << 4) - 4;
  95. yPos = (y << 4) + 0x27;
  96. SendMessage(hWnd, WM_RBUTTONDOWN, 0, MAKELPARAM(xPos, yPos));
  97. SendMessage(hWnd, WM_RBUTTONUP, 0, MAKELPARAM(xPos, yPos));
  98. SendMessage(hWnd, WM_RBUTTONDOWN, 0, MAKELPARAM(xPos, yPos));
  99. SendMessage(hWnd, WM_RBUTTONUP, 0, MAKELPARAM(xPos, yPos));
  100. }
  101. else
  102. {
  103. int xPos, yPos;
  104. xPos = (x << 4) - 4;
  105. yPos = (y << 4) + 0x27;
  106. SendMessage(hWnd, WM_LBUTTONDOWN, 0, MAKELPARAM(xPos, yPos));
  107. SendMessage(hWnd, WM_LBUTTONUP, 0, MAKELPARAM(xPos, yPos));
  108. }
  109. CString strCode;
  110. strCode.Format(L"%02x", byCode);
  111. strLine += strCode;
  112. }
  113. OutputDebugString(strLine.GetBuffer());
  114. }
  115. CString strCode;
  116. strCode.Format(L"找到的雷数%d", nFindCount);
  117. OutputDebugString(strCode.GetBuffer());
  118. }
  119. else if(Msg==WM_MOUSEMOVE)
  120. {
  121. //鼠标移动
  122. int x, y;
  123. x = LOWORD(lParam);
  124. y = HIWORD(lParam);
  125. x = (x + 4) >> 4;
  126. y = (y - 0x27) >> 4;
  127. BYTE byCode = *(PBYTE)((DWORD)g_pBase + x + y * 32);
  128. if (byCode==MINE)
  129. {
  130. SetWindowText(hWnd, L"此处有雷!!");
  131. }
  132. else
  133. {
  134. CString strPos;
  135. strPos.Format(L"(%d,%d", y, x);
  136. SetWindowText(hWnd, strPos);
  137. }
  138. }
  139. return CallWindowProc(g_OldhProc, hWnd, Msg, wParam, lParam);
  140. }
  141. BOOL CMFCLEIApp::InitInstance()
  142. {
  143. CWinApp::InitInstance();
  144. //1.通过查找窗口,获取窗口句柄
  145. g_Wnd = ::FindWindow(L"扫雷", L"扫雷");
  146. if (NULL== g_Wnd)
  147. {
  148. OutputDebugString(L"无法找到 扫雷窗口");
  149. return false;
  150. }
  151. //2.设置窗口回调函数
  152. g_OldhProc=(WNDPROC)SetWindowLong(g_Wnd,GWL_WNDPROC,(LONG)WindowProc);//设置窗口句柄(窗口句柄,索引值,)
  153. if (NULL == g_OldhProc)
  154. {
  155. OutputDebugString(L"设置窗口回调函数 失败");
  156. return false;
  157. }
  158. return TRUE;
  159. }

image.png