Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1338 lines
37 KiB

  1. #pragma warning(disable:4786)
  2. #include <string>
  3. #include <map>
  4. #include <vector>
  5. #include <stdio.h>
  6. #include <algorithm>
  7. #include "windows.h"
  8. #include "io.h"
  9. #include <limits.h>
  10. #define NUMBER_OF(x) (sizeof(x)/sizeof((x)[0]))
  11. #include <assert.h>
  12. #if defined(_WIN64)
  13. typedef unsigned __int64 ULONG_PTR;
  14. #else
  15. typedef unsigned long ULONG_PTR;
  16. #endif
  17. struct tagACTCTXW;
  18. typedef const tagACTCTXW* PCACTCTXW;
  19. typedef std::string::size_type size_type;
  20. // we also wrap user and loadlibrary
  21. const char Namespace[] = "SideBySide";
  22. const char NamespaceComCtl[] = "SideBySideCommonControls";
  23. #define NAMESPACE "SideBySide"
  24. #define NAMESPACE_COMCTL "SideBySideCommonControls"
  25. /*
  26. enum ETokenType
  27. {
  28. ttComment,
  29. ttChar,
  30. ttIdentifer
  31. };
  32. class CToken
  33. {
  34. public:
  35. ETokenType m_etype;
  36. std::string m_string;
  37. };
  38. enum EType
  39. {
  40. tVoid,
  41. tBool,
  42. tInt,
  43. tHRESULT,
  44. tOther
  45. };
  46. */
  47. class CType
  48. {
  49. public:
  50. //EType m_etype;
  51. std::string m_string; // right from a CToken, we ignore the multiple token case for now
  52. };
  53. class CParameter
  54. {
  55. public:
  56. CType m_type;
  57. std::string m_name;
  58. };
  59. std::string BaseFunctions[] =
  60. {
  61. "LoadLibrary"
  62. };
  63. std::string UserFunctions[] =
  64. {
  65. "RegisterClass",
  66. "RegisterClassEx",
  67. "CreateWindow",
  68. "CreateWindowEx",
  69. "GetClassInfo",
  70. "GetClassInfoEx",
  71. "DialogBoxParam",
  72. "DialogBoxIndirectParam",
  73. "CreateDialogIndirect",
  74. "CreateDialogIndirectParam",
  75. "CreateDialogParam"
  76. };
  77. #define BEGIN(x) x
  78. #define END(x) ((x) + sizeof(x) / sizeof((x)[0]))
  79. std::string AlsoNoFusionFunctions[] =
  80. {
  81. //"CreateDialogIndirectT",
  82. //"CreateDialogParamT",
  83. "CreateWindowExT",
  84. //"CreateWindowT",
  85. "CreateDialogIndirectParamT",
  86. "CreatePropertySheetPageT"
  87. };
  88. const static char prefix[]=
  89. "\n\
  90. #include \"windows.h\"\n\
  91. \n\
  92. HMODULE g_" NAMESPACE_COMCTL "DllHandle;\n\
  93. \n\
  94. DWORD " NAMESPACE_COMCTL "GetLastError()\n\
  95. {\n\
  96. DWORD Error = GetLastError();\n\
  97. if (Error == NO_ERROR)\n\
  98. {\n\
  99. Error = -1;//ERROR_UNSUCCESSFUL;\n\
  100. }\n\
  101. return Error;\n\
  102. }\n\
  103. \n\
  104. \n\
  105. HANDLE g_" NAMESPACE_COMCTL "ActCtx;\n\
  106. \n\
  107. BOOL\n\
  108. WINAPI\n\
  109. " NAMESPACE_COMCTL "DeactivateActCtx_DownLevel(\n\
  110. DWORD dwFlags,\n\
  111. ULONG_PTR ulCookie\n\
  112. )\n\
  113. {\n\
  114. return TRUE;\n\
  115. }\n\
  116. \n\
  117. BOOL\n\
  118. WINAPI\n\
  119. " NAMESPACE_COMCTL "DeactivateActCtx(\n\
  120. DWORD dwFlags,\n\
  121. ULONG_PTR ulCookie\n\
  122. )\n\
  123. {\n\
  124. typedef BOOL (WINAPI* PFN)(DWORD dwFlags, ULONG_PTR ulCookie);\n\
  125. static PFN pfn;\n\
  126. BOOL Success = FALSE;\n\
  127. \n\
  128. if (pfn == NULL)\n\
  129. {\n\
  130. HMODULE Kernel32 = GetModuleHandleW(L\"Kernel32.dll\");\n\
  131. if (Kernel32 == NULL) // this is fatal, even downlevel\n\
  132. goto Exit;\n\
  133. pfn = (PFN)GetProcAddress(Kernel32, \"DeactivateActCtx\");\n\
  134. if (pfn == NULL)\n\
  135. pfn = " NAMESPACE_COMCTL "DeactivateActCtx_DownLevel;\n\
  136. }\n\
  137. Success = pfn(dwFlags, ulCookie);\n\
  138. Exit:\n\
  139. return Success;\n\
  140. }\n\
  141. \n\
  142. BOOL\n\
  143. WINAPI\n\
  144. " NAMESPACE_COMCTL "ActivateActCtx_DownLevel(\n\
  145. HANDLE hActCtx,\n\
  146. ULONG_PTR *lpCookie\n\
  147. )\n\
  148. {\n\
  149. return TRUE;\n\
  150. }\n\
  151. \n\
  152. BOOL\n\
  153. WINAPI\n\
  154. " NAMESPACE_COMCTL "ActivateActCtx(\n\
  155. HANDLE hActCtx,\n\
  156. ULONG_PTR *lpCookie\n\
  157. )\n\
  158. {\n\
  159. typedef BOOL (WINAPI* PFN)(HANDLE hActCtx, ULONG_PTR *lpCookie);\n\
  160. static PFN pfn;\n\
  161. BOOL Success = FALSE;\n\
  162. \n\
  163. if (pfn == NULL)\n\
  164. {\n\
  165. HMODULE Kernel32 = GetModuleHandleW(L\"Kernel32.dll\");\n\
  166. if (Kernel32 == NULL) // this is fatal, even downlevel\n\
  167. goto Exit;\n\
  168. pfn = (PFN)GetProcAddress(Kernel32, \"ActivateActCtx\");\n\
  169. if (pfn == NULL)\n\
  170. pfn = " NAMESPACE_COMCTL "ActivateActCtx_DownLevel;\n\
  171. }\n\
  172. Success = pfn(hActCtx, lpCookie);\n\
  173. Exit:\n\
  174. return Success;\n\
  175. }\n\
  176. \n\
  177. BOOL\n\
  178. WINAPI\n\
  179. " NAMESPACE_COMCTL "CreateActCtxW_DownLevel(\n\
  180. PCACTCTXW pActCtx\n\
  181. )\n\
  182. {\n\
  183. return TRUE;\n\
  184. }\n\
  185. \n\
  186. BOOL\n\
  187. WINAPI\n\
  188. " NAMESPACE_COMCTL "CreateActCtxW(\n\
  189. PCACTCTXW pActCtx\n\
  190. )\n\
  191. {\n""\
  192. typedef BOOL (WINAPI* PFN)(PCACTCTXW pActCtx);\n\
  193. static PFN pfn;\n\
  194. BOOL Success = FALSE;\n\
  195. \n\
  196. if (pfn == NULL)\n\
  197. {\n\
  198. HMODULE Kernel32 = GetModuleHandleW(L\"Kernel32.dll\");\n\
  199. if (Kernel32 == NULL) // this is fatal, even downlevel\n\
  200. goto Exit;\n\
  201. pfn = (PFN)GetProcAddress(Kernel32, \"CreateActCtxW\");\n\
  202. if (pfn == NULL)\n\
  203. pfn = " NAMESPACE_COMCTL "CreateActCtxW_DownLevel;\n\
  204. }\n\
  205. Success = pfn(pActCtx);\n\
  206. Exit:\n\
  207. return Success;\n\
  208. }\n\
  209. \n\
  210. /* Requires NT4.0, Win98. SideBySide functionality is currently only on Whistler,\n\
  211. and will fail before it tries this. */\n\
  212. PVOID "NAMESPACE_COMCTL"InterlockedCompareExchangePointer(PVOID* Dest, PVOID Exch, PVOID Comperand)\n\
  213. {\n\
  214. #if defined(_X86_)\n\
  215. typedef PVOID (WINAPI* PFN)(PVOID*, PVOID, PVOID);\n\
  216. static PFN pfn;\n\
  217. /* Note, the export has a different name; this is correct. */\n\
  218. pfn = (PFN)GetProcAddress(GetModuleHandle(TEXT(\"Kernel32.dll\"), \"InterlockedCompareExchange\"));\n\
  219. /* UNDONE Error handling here.. */\n\
  220. return pfn(Dest, Exch, Comperand);\n\
  221. #else\n\
  222. /* always available on IA64, and most other processors */\n\
  223. return InterlockedCompareExchangePointer(Dest, Exch, Comperand);\n\
  224. #endif\n\
  225. }\n\
  226. \n\
  227. HANDLE g_" NAMESPACE_COMCTL "ActCtxHandle = INVALID_HANDLE_VALUE;\n\
  228. \n\
  229. #if defined(__cplusplus)\n\
  230. extern \"C\"\n\
  231. #endif\n\
  232. int __ImageBase; /* requires VC6 linker */\n\
  233. \n\
  234. BOOL\n\
  235. WINAPI\n\
  236. " NAMESPACE_COMCTL "CreateMyActCtx(\n\
  237. )\n\
  238. {\n\
  239. ACTCTX ActCtx;\n\
  240. HANDLE LocalActCtxHandle = INVALID_HANDLE_VALUE;\n\
  241. static DWORD Error;\n\
  242. PTSTR ModuleFileName = NULL;\n\
  243. DWORD ModuleNameBufferSize = 32;\n\
  244. DWORD ModuleNameSize;\n\
  245. BOOL Success = FALSE;\n\
  246. BOOL First = TRUE;\n\
  247. \n\
  248. if (Error != 0)\n\
  249. {\n\
  250. SetLastError(Error);\n\
  251. goto Exit;\n\
  252. }\n\
  253. if (g_"NAMESPACE_COMCTL"ActCtxHandle == INVALID_HANDLE_VALUE)\n\
  254. {\n\
  255. do\n\
  256. {\n\
  257. if (ModuleName == NULL)\n\
  258. {\n\
  259. ModuleFileName = (PTSTR)HeapAlloc(GetProcessHeap(), 0, ModuleNameBufferSize * sizeof(*ModuleName));\n\
  260. if (ModuleName == NULL)\n\
  261. {\n\
  262. Error = ERROR_NOT_ENOUGH_MEMORY;\n\
  263. goto Exit;\n\
  264. }\n\
  265. }\n\
  266. else\n\
  267. {\n\
  268. PTSTR ModuleNameLonger\n\
  269. ModuleBufferSize *= 2;\n\
  270. ModuleNameLonger = (PSTR)HeapReAlloc(GetProcessHeap(), 0, ModuleFileName, ModuleBufferSize * sizeof(*ModuleNameLonger));\n\
  271. if (ModuleNameLonger == NULL)\n\
  272. {\n\
  273. Error = ERROR_NOT_ENOUGH_MEMORY;\n\
  274. goto Exit;\n\
  275. }\n\
  276. ModuleFileName = ModuleNameLonger;\n\
  277. }\n\
  278. ModuleFileName[ModuleNameBufferSize - 2] = 0;\n\
  279. if (!GetModuleFileName(&__ImageBase, ModuleFileName, ModuleNameBufferSize))\n\
  280. {\n\
  281. Error = " NAMESPACE_COMCTL "GetLastError();\n\
  282. goto Exit;\n\
  283. }\n\
  284. } while (ModuleName[ModuleNameBufferSize - 2] != 0)\n\
  285. ActCtx.cbSize = sizeof(ActCtx);\n\
  286. ActCtx.dwFlags = 0;\n\
  287. ActCtx.lpSource = ModuleName;\n\
  288. LocalActCtxHandle = NAMESPACE_COMCTL CreateActCtx(&ActCtx);\n\
  289. if (LocalActCtxHandle == INVALID_HANDLE_VALUE)\n\
  290. goto Exit;\n\
  291. NAMESPACE_COMCTL InterlockedCompareExchangePointer((PVOID*)&g_"NAMESPACE_COMCTL"ActCtxHandle, LocalActCtxHandle, INVALID_HANDLE_VALUE);\n\
  292. }\n\
  293. Success = TRUE;\n\
  294. Exit:\n\
  295. /* If out of memory, let it be retried. */\n\
  296. if (Error != ERROR_NOT_ENOUGH_MEMORY)\n\
  297. Initialized = TRUE;\n\
  298. if (ModuleFileName != NULL)\n\
  299. {\n\
  300. DWORD LastError = GetLastError();\n\
  301. HeapFree(GetProcessHeap(), 0, ModuleFileName);\n\
  302. SetLastError(LastError);\n\
  303. }\n\
  304. \n\
  305. return Success;\n\
  306. }\n\
  307. \n\
  308. BOOL " NAMESPACE_COMCTL "DelayLoad(HMODULE* Module, DWORD* Error)\n\
  309. {\n\
  310. BOOL Success = FALSE;\n\
  311. ULONG_PTR Cookie;\n\
  312. BOOL ActivateSuccess = FALSE;\n\
  313. \n\
  314. if (*Module != NULL)\n\
  315. {\n\
  316. Success = TRUE;\n\
  317. goto Exit;\n\
  318. }\n\
  319. if (*Error != 0)\n\
  320. {\n\
  321. SetLastError(*Error);\n\
  322. goto Exit;\n\
  323. }\n\
  324. Success = " NAMESPACE_COMCTL "ActivateActCtx(g_" NAMESPACE_COMCTL "ActCtxHandle, &Cookie);\n\
  325. if (!Success)\n\
  326. {\n\
  327. *Error = " NAMESPACE_COMCTL "GetLastError();\n\
  328. goto Exit;\n\
  329. }\n\
  330. __try\n\
  331. {\n\
  332. *Module = LoadLibrary(TEXT(\"comctl32.dll\"));\n\
  333. if (*Module == NULL)\n\
  334. {\n\
  335. *Error = " NAMESPACE_COMCTL "GetLastError();\n\
  336. goto Exit;\n\
  337. }\n\
  338. }\n\
  339. __finally\n\
  340. {\n\
  341. if (AbnormalTermination())\n\
  342. {\n\
  343. DWORD LastError = GetLastError();\n\
  344. " NAMESPACE_COMCTL "DeactivateActCtx(0, Cookie);\n\
  345. SetLastError(LastError);\n\
  346. }\n\
  347. else\n\
  348. {\n\
  349. if (!" NAMESPACE_COMCTL "DeactivateActCtx(0, Cookie))\n\
  350. {\n\
  351. *Error = " NAMESPACE_COMCTL "GetLastError();\n\
  352. goto Exit;\n\
  353. }\n\
  354. }\n\
  355. }\n\
  356. Success = TRUE;\n\
  357. Exit:\n\
  358. return Success;\n\
  359. }\n\
  360. \n" "\n\
  361. \n\
  362. /* TODO wrapping up multiple const-static parameters in a struct is a code size win */\n\
  363. #if 0\n\
  364. typedef "NAMESPACE_COMCTL"GetProcAddressParameters\n\
  365. {\n\
  366. PCSTR Name;\n\
  367. FARPROC* Address;\n\
  368. DWORD* Error;\n\
  369. } "NAMESPACE_COMCTL"GetProcAddressParameters\n\
  370. #endif\n\
  371. \n\
  372. BOOL " NAMESPACE_COMCTL "GetProcAddress(PCSTR ProcedureName, FARPROC* ppfn, DWORD* GetProcAddressError)\n\
  373. /*\n\
  374. We have an error per function, as well as one overriding error, because\n\
  375. we can error on LoadLibrary or we could succeed LoadLibrary but then\n\
  376. error on some GetProcAddresses. Error on LoadLibrary propatated to all GetProcAddresses\n\
  377. with no retries. Errors on GetProcAddresses are isolated to GetProcAddresses.\n\
  378. */\n\
  379. {\n\
  380. static HMODULE Module;\n\
  381. static DWORD LoadLibraryError;\n\
  382. BOOL Success = FALSE;\n\
  383. \n\
  384. if (*ppfn != NULL)\n\
  385. {\n\
  386. Success = TRUE;\n\
  387. goto Exit;\n\
  388. }\n\
  389. if (*GetProcAddressError != 0)\n\
  390. {\n\
  391. SetLastError(*GetProcAddressError);\n\
  392. goto Exit;\n\
  393. }\n\
  394. if (LoadLibraryError != 0)\n\
  395. {\n\
  396. SetLastError(LoadLibraryError);\n\
  397. *GetProcAddressError = LoadLibraryError;\n\
  398. goto Exit;\n\
  399. }\n\
  400. if (!" NAMESPACE_COMCTL "DelayLoad(&Module, &LoadLibraryError))\n\
  401. {\n\
  402. *GetProcAddressError = LoadLibraryError;\n\
  403. goto Exit;\n\
  404. }\n\
  405. if ((*ppfn = GetProcAddress(Module, ProcedureName)) == NULL)\n\
  406. {\n\
  407. *GetProcAddressError = " NAMESPACE_COMCTL "GetLastError();\n\
  408. /*\n\
  409. Don't touch LoadLibraryError for GetProcAddress.\n\
  410. */\n\
  411. goto Exit;\n\
  412. }\n\
  413. Success = TRUE;\n\
  414. Exit:\n\
  415. return Success;\n\
  416. }\n\
  417. ";
  418. #if 0 // FUTURE We should make this be multipass, it's easier.
  419. class CStreamBuffer : public IStream
  420. {
  421. public:
  422. CStreamBuffer(IStream* Stream) : m_Stream(Stream) { }
  423. STDMETHOD(Read)(void* pv, ULONG cb, ULONG* pcbRead)
  424. {
  425. ULONG cbRead = 0;
  426. HRESULT hr;
  427. if (pcbRead != NULL)
  428. *pcbRead = 0;
  429. if (cb == 0)
  430. {
  431. hr = S_OK;
  432. goto Exit;
  433. }
  434. if (FAILED(hr = FlushWriteBuffer()))
  435. goto Exit;
  436. if (m_BufferSize != 0)
  437. {
  438. cbRead = MIN(cb, m_BufferSize);
  439. CopyMemory(pv, m_Buffer + m_BufferPosition, cbRead);
  440. cb -= cbRead;
  441. m_BufferPosition += cbRead;
  442. m_BufferSize -= cbRead;
  443. }
  444. if (cb == 0)
  445. {
  446. hr = S_OK;
  447. goto Exit;
  448. }
  449. /*
  450. depending on the client, we could stop here with a partial read
  451. for more compatibility, we will push ahead and fill the buffer in order
  452. to give them all that they asked for
  453. */
  454. if (cb >= NUMBER_OF(m_Buffer))
  455. {
  456. hr = m_Stream->Read(pv, cb, pcbRead);
  457. if (pcbRead != NULL)
  458. cbRead += *pcbRead;
  459. goto Exit;
  460. }
  461. if (FAILED(hr = FillReadBuffer()))
  462. return hr;
  463. CopyMemory(pv, reinterpret_cast<PBYTE>(m_Buffer) + m_BufferPosition, cb);
  464. m_BufferPosition += cb;
  465. m_BufferSize -= cb;
  466. cbRead += cb;
  467. hr = S_OK;
  468. Exit:
  469. if (pcbRead != NULL)
  470. *pcbRead = cbRead;
  471. return hr;
  472. }
  473. HRESULT FillReadBuffer()
  474. {
  475. ULONG cb;
  476. m_fReadBuffer = true;
  477. HRESULT hr = m_Stream->Read(m_Buffer, NUMBER_OF(m_Buffer), &cb);
  478. m_BufferPosition = 0;
  479. m_BufferSize = cb;
  480. return hr;
  481. }
  482. HRESULT FlushWriteBuffer()
  483. {
  484. ULONG cb;
  485. if (m_fWriteBuffer && m_BufferSize != 0)
  486. {
  487. HRESULT hr = m_Stream->Write(m_Buffer, m_BufferSize, &cb);
  488. m_BufferPosition = 0;
  489. m_BufferSize = cb;
  490. return hr;
  491. }
  492. return S_OK;
  493. }
  494. IStream* m_Stream;
  495. BYTE m_Buffer[4096];
  496. ULONG m_BufferPosition;
  497. ULONG m_BufferSize;
  498. bool m_fReadBuffer;
  499. bool m_fWriteBuffer;
  500. };
  501. #endif // FUTURE
  502. class CFunction
  503. {
  504. public:
  505. CFunction()
  506. {
  507. Clear();
  508. }
  509. void PrinDefine(FILE* File, const char* FromPrefix, const char* ToPrefix) const
  510. {
  511. fprintf(File, "#define %s%s %s%s\n", FromPrefix, m_name.c_str(), ToPrefix, m_name.c_str());
  512. }
  513. void PrintStub(FILE* File) const
  514. {
  515. if (m_isComctl)
  516. {
  517. fprintf(
  518. File,
  519. "%s\n"
  520. "WINAPI\n"
  521. "SideBySide%s(%s)\n"
  522. "{\n"
  523. " static DWORD Error\n"
  524. " typedef %s (WINAPI* PFN)(%s);\n"
  525. " static PFN pfn;\n"
  526. " if (pfn == NULL)\n"
  527. " " NAMESPACE_COMCTL "GetProcAddress(\"%s\", (FARPROC*)&pfn, &Error);\n"
  528. " if (pfn != NULL)\n"
  529. " return pfn(%s);\n"
  530. " SetLastError(Error);\n"
  531. " return %s;\n"
  532. "}\n\n",
  533. m_returnType.m_string.c_str(),
  534. m_name.c_str(),
  535. m_parameterTypesNamesString.c_str(),
  536. m_returnType.m_string.c_str(),
  537. m_parameterTypesNamesString.c_str(),
  538. m_name.c_str(),
  539. m_parameterNamesString.c_str(),
  540. m_error.c_str()
  541. );
  542. }
  543. else
  544. {
  545. fprintf(
  546. File,
  547. "%s\n"
  548. "WINAPI\n"
  549. "SideBySide%s(%s)\n"
  550. "{\n"
  551. " ULONG_PTR ActCtxCookie;\n"
  552. " %s %s %s;
  553. " BOOL ActivateActCtxSuccess;
  554. "
  555. " ActivateActCtxSuccess
  556. " return %s(%s);\n"
  557. " SetLastError(Error);\n"
  558. " return %s;\n"
  559. "}\n\n",
  560. m_returnType.m_string.c_str(),
  561. m_name.c_str(),
  562. m_parameterTypesNamesString.c_str(),
  563. m_returnType.m_string.c_str(),
  564. m_parameterTypesNamesString.c_str(),
  565. m_name.c_str(),
  566. m_parameterNamesString.c_str(),
  567. m_error.c_str()
  568. );
  569. }
  570. }
  571. void FormParameterStrings()
  572. {
  573. m_parameterTypesNamesString.erase();
  574. m_parameterNamesString.erase();
  575. bool first = true;
  576. if (m_parameters.begin() != m_parameters.end())
  577. {
  578. for (CParameters::const_iterator i = m_parameters.begin() ; i != m_parameters.end() ; ++i)
  579. {
  580. if (!first)
  581. {
  582. m_parameterTypesNamesString += ", ";
  583. m_parameterNamesString += ", ";
  584. }
  585. first = false;
  586. m_parameterTypesNamesString += i->m_type.m_string;
  587. m_parameterTypesNamesString += " ";
  588. m_parameterTypesNamesString += i->m_name.c_str();
  589. m_parameterNamesString += i->m_name.c_str();
  590. }
  591. m_isVoid = false;
  592. }
  593. else
  594. {
  595. m_parameterTypesNamesString = "void";
  596. m_isVoid = true;
  597. }
  598. }
  599. void Clear()
  600. {
  601. m_isVoid = false;
  602. m_isBase = false;
  603. m_isUser = false;
  604. m_isComctl = false;
  605. m_parameters.clear();
  606. m_parameterTypesNamesString.erase();
  607. m_parameterNamesString.erase();
  608. m_name.erase();
  609. m_error.erase();
  610. m_returnType.m_string.erase();
  611. // temporary, until we finish annotating commctrl.h
  612. m_error = "0";
  613. }
  614. CType m_returnType;
  615. typedef std::vector<CParameter> CParameters;
  616. CParameters m_parameters;
  617. std::string m_parameterTypesNamesString;
  618. std::string m_parameterNamesString;
  619. std::string m_name;
  620. std::string m_nameGeneric;
  621. std::string m_error;
  622. bool m_isVoid;
  623. bool m_isBase; // WINBASEAPI (just LoadLibrary)
  624. bool m_isUser; // WINUSERAPI
  625. bool m_isComctl; // WINCOMMCTRLAPI
  626. };
  627. std::string GetEnvironmentVariable(const std::string& s)
  628. {
  629. char buffer[4000];
  630. buffer[0] = 0;
  631. GetEnvironmentVariableA(s.c_str(), buffer, NUMBER_OF(buffer));
  632. return buffer;
  633. }
  634. std::vector<std::string> SplitStringOnChar(std::string s, char c)
  635. {
  636. std::vector<std::string> result;
  637. #if 0
  638. std::string::const_iterator i;
  639. std::string::const_iterator j;
  640. for (i = s.begin() ; i != s.end() ; ++i)
  641. {
  642. i += findf_first
  643. }
  644. #endif
  645. return result;
  646. }
  647. void ThrowErrno()
  648. {
  649. throw errno;
  650. }
  651. class CStdioFile
  652. {
  653. public:
  654. CStdioFile(FILE* file = NULL) : m_file(file) { }
  655. operator FILE*()
  656. {
  657. return m_file;
  658. }
  659. void operator=(FILE* file)
  660. {
  661. Close();
  662. m_file = file;
  663. }
  664. FILE* operator->()
  665. {
  666. return m_file;
  667. }
  668. ~CStdioFile()
  669. {
  670. Close();
  671. }
  672. void Close()
  673. {
  674. FILE* file = m_file;
  675. m_file = NULL;
  676. if (file != NULL)
  677. {
  678. if (fclose(file) != 0)
  679. {
  680. ThrowErrno();
  681. }
  682. }
  683. }
  684. FILE* m_file;
  685. };
  686. class CMakeSideBySideCommonControls
  687. {
  688. public:
  689. CMakeSideBySideCommonControls() { }
  690. std::string m_path;
  691. std::string m_string;
  692. //std::vector<CToken> m_tokens;
  693. typedef std::vector<CFunction> CFunctions;
  694. CFunctions m_functions;
  695. void OpenAndRead(const std::string& filename)
  696. {
  697. std::string ntDrive = GetEnvironmentVariable("_NTDRIVE");
  698. std::string ntRoot = GetEnvironmentVariable("_NTROOT");
  699. std::string include = GetEnvironmentVariable("include"); // future
  700. std::vector<std::string> includes = SplitStringOnChar(include, ';'); // future
  701. #if 0
  702. if (ntDrive == "")
  703. {
  704. ntDrive = "Z:";
  705. }
  706. if (ntRoot == "")
  707. {
  708. ntRoot = "nt";
  709. }
  710. #else
  711. if (ntDrive == "")
  712. {
  713. ntDrive = "x:";
  714. }
  715. if (ntRoot == "")
  716. {
  717. ntRoot = "sh1";
  718. }
  719. #endif
  720. std::string directory = ntDrive + "\\" + ntRoot + "\\public\\sdk\\inc\\";
  721. m_path = directory + filename;
  722. CStdioFile file = fopen(m_path.c_str(), "rb");
  723. if (file == NULL)
  724. ThrowErrno();
  725. __int64 size = _filelengthi64(_fileno(file));
  726. if (size == -1)
  727. ThrowErrno();
  728. //size_type previousSize = m_string.size();
  729. size_type previousSize = 0;
  730. m_string.resize(previousSize + static_cast<size_type>(size));
  731. size_t elementsRead = fread(&m_string[previousSize], sizeof(char), size, file);
  732. if (elementsRead != size)
  733. ThrowErrno();
  734. m_string.append("\n\n");
  735. }
  736. static void FindAndReplaceChar(std::string& s, char from, char to)
  737. {
  738. for (std::string::iterator i = s.begin(); i != s.end() ; ++i)
  739. {
  740. if (*i == from)
  741. *i = to;
  742. }
  743. }
  744. #define STRIPCOMMENTS_SLASHSLASH 0x000001
  745. #define STRIPCOMMENTS_SLASHSTAR 0x000002
  746. #define STRIPCOMMENTS_SLASHSLASH_UNAWARE 0x000004
  747. #define STRIPCOMMENTS_SLASHSTAR_UNAWARE 0x000008
  748. static void StripComments(int flags, std::string& s)
  749. /*
  750. We generally want to be "aware" of both types so that we don't
  751. strip nested comments. Consider the comments that follow.
  752. */
  753. // /* slash star in slsh slash */
  754. /* // slashslash
  755. in slash star
  756. */
  757. {
  758. std::string t;
  759. std::string::const_iterator i;
  760. const std::string::const_iterator j = s.end();
  761. std::string::const_iterator k;
  762. bool closed = true;
  763. t.reserve(s.size());
  764. for (i = s.begin() ; closed && i != j && i + 1 != j; )
  765. {
  766. if (
  767. ((flags & STRIPCOMMENTS_SLASHSTAR) || (flags & STRIPCOMMENTS_SLASHSTAR_UNAWARE) == 0)
  768. && *i == '/'
  769. && *(i + 1) == '*'
  770. )
  771. {
  772. closed = false;
  773. for (k = i + 2 ; k != j && k + 1 != j && !(closed = (*k == '*' && *(k + 1) == '/')) ; ++k)
  774. {
  775. }
  776. if (flags & STRIPCOMMENTS_SLASHSTAR)
  777. t.append(1, ' ');
  778. else
  779. t.append(i, k + 2);
  780. i = k + 2;
  781. }
  782. else if (
  783. ((flags & STRIPCOMMENTS_SLASHSLASH) || (flags & STRIPCOMMENTS_SLASHSLASH_UNAWARE) == 0)
  784. && *i == '/'
  785. && *(i + 1) == '/'
  786. )
  787. {
  788. closed = false;
  789. for (k = i + 2 ; k != j && !(closed = (*k == '\r' || *k == '\n')) ; ++k)
  790. {
  791. }
  792. for ( ; k != j && *k == '\r' || *k == '\n' ; ++k)
  793. {
  794. }
  795. if (flags & STRIPCOMMENTS_SLASHSLASH)
  796. t.append(1, '\n');
  797. else
  798. t.append(i, k);
  799. i = k;
  800. }
  801. if (closed && i != j)
  802. t.append(1, *i++);
  803. }
  804. if (closed)
  805. {
  806. for ( ; i != j ; ++i )
  807. {
  808. t.append(1, *i);
  809. }
  810. }
  811. s = t;
  812. }
  813. void ProcessDeclaration(CFunction function, const std::string& comment, std::string declaration)
  814. {
  815. size_type i = 0;
  816. size_type j = 0;
  817. const static char whitespaceCommaLparen[] = ",) \t\r\n";
  818. const static char whitespace[] = " \t\r\n";
  819. const static char beforeParamsDelims[] = " \t\n\r(";
  820. const static char intracommaLparen[] = " \t\n\r*&"; // crude way to detect
  821. // void F(char*) vs. void F(char x)
  822. const static char commaLparen[] = ",)";
  823. const static char identifierCharacters[] = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_";
  824. bool rparen = false;
  825. bool end = false;
  826. bool comma = false;
  827. const size_type npos = std::string::npos;
  828. //
  829. // This way our ban on [] can be less strict.
  830. // We ban [] because aren't smart about multi token types.
  831. // We allow parameters like const char * foo, because commctrl does have parameters
  832. // whose types are multiple tokens.
  833. //
  834. StripComments(STRIPCOMMENTS_SLASHSLASH | STRIPCOMMENTS_SLASHSTAR, declaration);
  835. if (declaration.find_first_of("[]") != npos)
  836. {
  837. assert(declaration.find("WriteFileGather") != npos
  838. || declaration.find("ReadFileScatter") != npos
  839. );
  840. }
  841. i = declaration.find_first_not_of(beforeParamsDelims, j);
  842. j = declaration.find_first_of(beforeParamsDelims, i);
  843. assert(declaration.substr(i, j - i) == "WINBASEAPI"
  844. || declaration.substr(i, j - i) == "WINUSERAPI"
  845. || declaration.substr(i, j - i) == "WINCOMMCTRLAPI"
  846. );
  847. i = declaration.find_first_not_of(beforeParamsDelims, j);
  848. j = declaration.find_first_of(beforeParamsDelims, i);
  849. if (
  850. declaration.substr(i, j - i) == "DECLSPEC_NORETURN"
  851. )
  852. {
  853. i = declaration.find_first_not_of(beforeParamsDelims, j);
  854. j = declaration.find_first_of(beforeParamsDelims, i);
  855. }
  856. function.m_returnType.m_string.assign(declaration.substr(i, j != npos ? j - i : npos));
  857. i = declaration.find_first_not_of(beforeParamsDelims, j);
  858. j = declaration.find_first_of(beforeParamsDelims, i);
  859. if (
  860. declaration.substr(i, j - i) == "WINAPI"
  861. || declaration.substr(i, j - i) == "WINAPIV"
  862. || declaration.substr(i, j - i) == "NTAPI"
  863. || declaration.substr(i, j - i) == "_stcall"
  864. || declaration.substr(i, j - i) == "__stcall"
  865. || declaration.substr(i, j - i) == "_cdecl"
  866. || declaration.substr(i, j - i) == "__cdecl"
  867. || declaration.substr(i, j - i) == "APIENTRY"
  868. || declaration.substr(i, j - i) == "FASTCALL"
  869. || declaration.substr(i, j - i) == "_fastcall"
  870. || declaration.substr(i, j - i) == "__fastcall"
  871. )
  872. {
  873. i = declaration.find_first_not_of(beforeParamsDelims, j);
  874. j = declaration.find_first_of(beforeParamsDelims, i);
  875. }
  876. function.m_name.assign(declaration.substr(i, j != npos ? j - i : npos));
  877. if ( *(function.m_name.end() - 1) == 'A'
  878. || *(function.m_name.end() - 1) == 'W'
  879. )
  880. {
  881. function.m_nameGeneric.assign(function.m_name.begin(), function.m_name.end() - 1);
  882. }
  883. j += (j != npos);
  884. //
  885. // now split mainly on comma
  886. //
  887. while (
  888. i != declaration.size()
  889. && j != declaration.size()
  890. && i != npos
  891. && j != npos
  892. )
  893. {
  894. CParameter parameter;
  895. i = declaration.find_first_not_of(whitespaceCommaLparen, j);
  896. if (i == npos)
  897. break;
  898. j = declaration.find_first_of(commaLparen, i);
  899. //
  900. // temporary, we still might split off the name
  901. //
  902. parameter.m_type.m_string = declaration.substr(i, j != npos ? j - i : npos);
  903. if ( parameter.m_type.m_string == "void"
  904. || parameter.m_type.m_string == "VOID"
  905. || parameter.m_type.m_string == ""
  906. )
  907. {
  908. break;
  909. }
  910. #if 0
  911. size_type splitParamNameFromType = parameter.m_type.m_string.find_last_of(typeTokenDelims);
  912. if (splitParamNameFromType != npos)
  913. {
  914. // the parameter has a name
  915. parameter.m_name = parameter.m_type.m_string.substr(splitParamNameFromType);
  916. parameter.m_type.m_string.resize(splitParamNameFromType);
  917. }
  918. #else
  919. size_type splitParamNameFromType = parameter.m_type.m_string.find_last_not_of(identifierCharacters);
  920. if (splitParamNameFromType != npos)
  921. {
  922. // the parameter has a name
  923. parameter.m_name = parameter.m_type.m_string.substr(splitParamNameFromType + 1);
  924. parameter.m_type.m_string.resize(splitParamNameFromType);
  925. }
  926. #endif
  927. else
  928. {
  929. char buffer[sizeof(int) * CHAR_BIT];
  930. sprintf(buffer, "%d", static_cast<int>(function.m_parameters.size()));
  931. parameter.m_name = "UnnamedParameter";
  932. parameter.m_name += buffer;
  933. }
  934. j += (j != npos);
  935. if (!comment.empty())
  936. {
  937. char functionFromComment[100];
  938. char errorFromComment[100];
  939. functionFromComment[0] = 0;
  940. errorFromComment[0] = 0;
  941. if (2 == sscanf(comment.c_str(), " @Function %s @Error %s ", functionFromComment, errorFromComment))
  942. {
  943. if (function.m_name != functionFromComment)
  944. {
  945. printf(
  946. "comment /* %s */ has different function than expected %s in %s\n",
  947. comment.c_str(),
  948. function.m_name.c_str(),
  949. m_path.c_str()
  950. );
  951. throw -1;
  952. }
  953. function.m_error = errorFromComment;
  954. }
  955. }
  956. function.m_parameters.push_back(parameter);
  957. }
  958. if (function.m_name == "LoadLibraryA")
  959. {
  960. ;//DebugBreak();
  961. }
  962. if (function.m_name == "RegisterClassA")
  963. {
  964. ;//DebugBreak();
  965. }
  966. if (
  967. (function.m_isBase && (std::binary_search(BEGIN(BaseFunctions), END(BaseFunctions), function.m_name)
  968. || std::binary_search(BEGIN(BaseFunctions), END(BaseFunctions), function.m_nameGeneric)))
  969. || (function.m_isUser && (std::binary_search(BEGIN(UserFunctions), END(UserFunctions), function.m_name)
  970. || std::binary_search(BEGIN(UserFunctions), END(UserFunctions), function.m_nameGeneric)))
  971. || function.m_isComctl
  972. //true
  973. )
  974. {
  975. function.FormParameterStrings();
  976. m_functions.push_back(function);
  977. }
  978. else
  979. {
  980. // just ignore it
  981. }
  982. }
  983. void HackInsteadOfTokenizeAndParse()
  984. {
  985. /*
  986. This is not a proper C/C++ tokenizer.
  987. It works well enough for our purposes with commctrl.h.
  988. We are interested in function declarations that start
  989. WINCOMMCTRLAPI, and comments precending them that contain @Function.
  990. Declarations end with a semicolon.
  991. We ignore preprocessor directives.
  992. We ignore slash continuation.
  993. */
  994. std::string comment;
  995. std::string declaration;
  996. std::string::const_iterator i;
  997. std::string::const_iterator j;
  998. std::string::const_iterator startOfDeclaration = m_string.end();
  999. std::string::const_iterator endOfDeclaration;
  1000. std::string::const_iterator startOfComment;
  1001. std::string::const_iterator endOfComment;
  1002. std::string::const_iterator startOfLine = m_string.begin();
  1003. bool gotRparen = false;
  1004. bool gotLparen = false;
  1005. /* This might become useful.
  1006. std::vector<std::string> typeNames;
  1007. static const char* typenamesConstData[] =
  1008. {
  1009. "bool", "char", "short", "int", "long", "float", "double", "void",
  1010. "BOOL", "LRESULT", "HRESULT" "HIMAGELIST", "WORD", "DWORD", "ULONG"
  1011. "UINT", "INT", "BYTE", "COLORREF", "void *", "HDPA",
  1012. "HANDLE, "HWND", "POINT", "LPRECT", "LPINT", "LPSCROLLINFO",
  1013. "INT_PTR", "UINT_PTR", "LONG_PTR", "ULONG_PTR",
  1014. };
  1015. for (const char** pp = typenamesConstData ; pp != typenamesConstData + NUMBER_OF(typenamesConstData) ; ++pp)
  1016. {
  1017. typeNames.push_back(*pp);
  1018. }
  1019. */
  1020. CFunction function;
  1021. for (i = m_string.begin() ; i != m_string.end() ; )
  1022. {
  1023. switch (*i)
  1024. {
  1025. Ldefault:
  1026. default:
  1027. startOfLine = m_string.end();
  1028. ++i;
  1029. break;
  1030. case '#':
  1031. /*
  1032. Macros happen do not occur in the middle of declarations that we care about.
  1033. They do keep us away from declrations that might otherwise mess us up, lik
  1034. WinMain.
  1035. */
  1036. startOfDeclaration = m_string.end();
  1037. goto Ldefault;
  1038. case ';':
  1039. if (startOfDeclaration == m_string.end()
  1040. || !gotLparen
  1041. || !gotRparen
  1042. )
  1043. goto Ldefault;
  1044. endOfDeclaration = i;
  1045. declaration.assign(startOfDeclaration, endOfDeclaration);
  1046. startOfDeclaration = m_string.end();
  1047. ProcessDeclaration(function, comment, declaration);
  1048. comment.erase();
  1049. declaration.erase();
  1050. function.Clear();
  1051. gotLparen = false;
  1052. gotRparen = false;
  1053. goto Ldefault;
  1054. case 'W':
  1055. if (startOfLine == m_string.end()
  1056. || (i + 1) == m_string.end()
  1057. || (i + 2) == m_string.end()
  1058. || (i + 3) == m_string.end()
  1059. || (i + 4) == m_string.end()
  1060. || (i + 5) == m_string.end()
  1061. || *(i + 1) != 'I'
  1062. || *(i + 2) != 'N'
  1063. || *(i + 3) == 'A'
  1064. || *(i + 4) == 'P'
  1065. || *(i + 5) == 'I'
  1066. )
  1067. goto Ldefault;
  1068. const char* s;
  1069. startOfDeclaration = i;
  1070. gotLparen = false;
  1071. gotRparen = false;
  1072. ++i;
  1073. for (j = startOfDeclaration, s = "WINCOMMCTRLAPI"; j != m_string.end() && *s ; ++s, ++j)
  1074. {
  1075. if (*j != *s)
  1076. break;
  1077. }
  1078. if (*s == 0)
  1079. {
  1080. function.m_isComctl = true;
  1081. break;
  1082. }
  1083. for (j = startOfDeclaration, s = "WINUSERAPI"; j != m_string.end() && *s ; ++s, ++j)
  1084. {
  1085. if (*j != *s)
  1086. break;
  1087. }
  1088. if (*s == 0)
  1089. {
  1090. function.m_isUser = true;
  1091. break;
  1092. }
  1093. for (j = startOfDeclaration, s = "WINBASEAPI"; j != m_string.end() && *s ; ++s, ++j)
  1094. {
  1095. if (*j != *s)
  1096. break;
  1097. }
  1098. if (*s == 0)
  1099. {
  1100. function.m_isBase = true;
  1101. break;
  1102. }
  1103. startOfDeclaration = m_string.end();
  1104. break;
  1105. case '(':
  1106. gotLparen = (startOfDeclaration != m_string.end());
  1107. goto Ldefault;
  1108. case ')':
  1109. gotRparen = (gotLparen && startOfDeclaration != m_string.end());
  1110. goto Ldefault;
  1111. case '/':
  1112. if (startOfLine == m_string.end())
  1113. goto Ldefault;
  1114. ++i;
  1115. if (*i == '*')
  1116. {
  1117. ++i;
  1118. startOfComment = i;
  1119. comment.erase();
  1120. for (; i != m_string.end() && comment.empty(); ++i)
  1121. {
  1122. switch (*i)
  1123. {
  1124. default:
  1125. break;
  1126. case '*':
  1127. endOfComment = i;
  1128. i++;
  1129. /* allow for comments like this ***/
  1130. while (*i == '*')
  1131. {
  1132. endOfComment = i;
  1133. i++;
  1134. }
  1135. if (*i == '/')
  1136. {
  1137. ++i;
  1138. comment.assign(startOfComment, endOfComment);
  1139. //printf("/*\n%s\n*/\n", comment.c_str());
  1140. }
  1141. break;
  1142. }
  1143. }
  1144. }
  1145. break;
  1146. case '\n':
  1147. case '\r':
  1148. for ( ; i != m_string.end() && (*i == '\n' || *i == '\r'); ++i )
  1149. {
  1150. /* nothing */
  1151. }
  1152. startOfLine = i;
  1153. break;
  1154. case ' ':
  1155. case '\t':
  1156. for ( ; i != m_string.end() && (*i == ' ' || *i == '\t'); ++i )
  1157. {
  1158. /* nothing */
  1159. }
  1160. // leave startOfLine unchanged
  1161. break;
  1162. }
  1163. }
  1164. }
  1165. void PrintStubs()
  1166. {
  1167. for (CFunctions::const_iterator i = m_functions.begin() ; i != m_functions.end() ; ++i)
  1168. {
  1169. i->PrintStub(stdout);
  1170. }
  1171. }
  1172. /*
  1173. -in foo.c -out x.h
  1174. */
  1175. void System(const char* s)
  1176. {
  1177. printf("%s\n", s);
  1178. system(s);
  1179. }
  1180. std::string m_defines;
  1181. std::string m_includes;
  1182. void operator()(int argc, char** argv)
  1183. {
  1184. std::sort(BEGIN(BaseFunctions), END(BaseFunctions));
  1185. std::sort(BEGIN(UserFunctions), END(UserFunctions));
  1186. std::sort(BEGIN(AlsoNoFusionFunctions), END(AlsoNoFusionFunctions));
  1187. OpenAndRead("winbase.h");
  1188. StripComments(STRIPCOMMENTS_SLASHSLASH, m_string);
  1189. HackInsteadOfTokenizeAndParse();
  1190. OpenAndRead("winuser.h");
  1191. StripComments(STRIPCOMMENTS_SLASHSLASH, m_string);
  1192. HackInsteadOfTokenizeAndParse();
  1193. OpenAndRead("commctrl.h");
  1194. StripComments(STRIPCOMMENTS_SLASHSLASH, m_string);
  1195. HackInsteadOfTokenizeAndParse();
  1196. OpenAndRead("prsht.h");
  1197. StripComments(STRIPCOMMENTS_SLASHSLASH, m_string);
  1198. HackInsteadOfTokenizeAndParse();
  1199. printf("%s\n", prefix);
  1200. #if 0
  1201. while (*++argv)
  1202. {
  1203. std::string s = *argv;
  1204. FindAndReplaceChar(s, 'n', '\n');
  1205. FindAndReplaceChar(s, 'r', '\r');
  1206. StripComments(0, s);
  1207. printf("%s\n", s.c_str());
  1208. }
  1209. #endif
  1210. PrintStubs();
  1211. }
  1212. };
  1213. int main(int argc, char** argv)
  1214. {
  1215. CMakeSideBySideCommonControls makeSideBySideCommonControls;
  1216. makeSideBySideCommonControls(argc, argv);
  1217. return 0;
  1218. }