Source code of Windows XP (NT5)
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.

433 lines
13 KiB

  1. //+---------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 1992 - 1995.
  5. //
  6. // File: debug.c
  7. //
  8. // Contents:
  9. //
  10. // Classes:
  11. //
  12. // Functions:
  13. //
  14. // History:
  15. //
  16. //----------------------------------------------------------------------------
  17. #include <spbase.h>
  18. #include <alloca.h>
  19. HANDLE g_hfLogFile = NULL;
  20. #if DBG /* NOTE: This file not compiled for retail builds */
  21. #include <stdio.h>
  22. #include <stdarg.h>
  23. #define WINDEBUG
  24. #ifndef min
  25. #define min(x,y) ((x)<(y)?(x):(y))
  26. #endif
  27. DWORD g_dwInfoLevel = 0;
  28. DWORD g_dwDebugBreak = 0;
  29. DWORD PctTraceIndent = 0;
  30. #define MAX_DEBUG_BUFFER 2048
  31. void
  32. BuildDebugHeader(
  33. DWORD Mask,
  34. PSTR pszHeader,
  35. PDWORD pcbHeader);
  36. void
  37. SPDebugOutput(char *szOutString)
  38. {
  39. DWORD dwWritten;
  40. if (NULL != g_hfLogFile)
  41. {
  42. WriteFile(
  43. g_hfLogFile,
  44. szOutString,
  45. lstrlen(szOutString),
  46. &dwWritten,
  47. NULL);
  48. }
  49. OutputDebugStringA(szOutString);
  50. }
  51. void
  52. DbgDumpHexString(const unsigned char *String, DWORD cbString)
  53. {
  54. DWORD i,count;
  55. CHAR digits[]="0123456789abcdef";
  56. CHAR pbLine[MAX_PATH];
  57. DWORD cbLine, cbHeader;
  58. DWORD_PTR address;
  59. BuildDebugHeader(DEB_BUFFERS, pbLine, &cbHeader);
  60. if(String == NULL && cbString != 0)
  61. {
  62. strcat(pbLine, "<null> buffer!!!\n");
  63. SPDebugOutput(pbLine);
  64. return;
  65. }
  66. for(; cbString ; cbString -= count, String += count)
  67. {
  68. count = (cbString > 16) ? 16:cbString;
  69. cbLine = cbHeader;
  70. address = (DWORD_PTR)String;
  71. #if defined(_WIN64)
  72. pbLine[cbLine++] = digits[(address >> 0x3c) & 0x0f];
  73. pbLine[cbLine++] = digits[(address >> 0x38) & 0x0f];
  74. pbLine[cbLine++] = digits[(address >> 0x34) & 0x0f];
  75. pbLine[cbLine++] = digits[(address >> 0x30) & 0x0f];
  76. pbLine[cbLine++] = digits[(address >> 0x2c) & 0x0f];
  77. pbLine[cbLine++] = digits[(address >> 0x28) & 0x0f];
  78. pbLine[cbLine++] = digits[(address >> 0x24) & 0x0f];
  79. pbLine[cbLine++] = digits[(address >> 0x20) & 0x0f];
  80. #endif
  81. pbLine[cbLine++] = digits[(address >> 0x1c) & 0x0f];
  82. pbLine[cbLine++] = digits[(address >> 0x18) & 0x0f];
  83. pbLine[cbLine++] = digits[(address >> 0x14) & 0x0f];
  84. pbLine[cbLine++] = digits[(address >> 0x10) & 0x0f];
  85. pbLine[cbLine++] = digits[(address >> 0x0c) & 0x0f];
  86. pbLine[cbLine++] = digits[(address >> 0x08) & 0x0f];
  87. pbLine[cbLine++] = digits[(address >> 0x04) & 0x0f];
  88. pbLine[cbLine++] = digits[(address ) & 0x0f];
  89. pbLine[cbLine++] = ' ';
  90. pbLine[cbLine++] = ' ';
  91. for(i = 0; i < count; i++)
  92. {
  93. pbLine[cbLine++] = digits[String[i]>>4];
  94. pbLine[cbLine++] = digits[String[i]&0x0f];
  95. if(i == 7)
  96. {
  97. pbLine[cbLine++] = ':';
  98. }
  99. else
  100. {
  101. pbLine[cbLine++] = ' ';
  102. }
  103. }
  104. #if 1
  105. for(; i < 16; i++)
  106. {
  107. pbLine[cbLine++] = ' ';
  108. pbLine[cbLine++] = ' ';
  109. pbLine[cbLine++] = ' ';
  110. }
  111. pbLine[cbLine++] = ' ';
  112. for(i = 0; i < count; i++)
  113. {
  114. if(String[i] < 32 || String[i] > 126)
  115. {
  116. pbLine[cbLine++] = '.';
  117. }
  118. else
  119. {
  120. pbLine[cbLine++] = String[i];
  121. }
  122. }
  123. #endif
  124. pbLine[cbLine++] = '\n';
  125. pbLine[cbLine++] = 0;
  126. SPDebugOutput(pbLine);
  127. }
  128. }
  129. char *aszSPDebugLevel[] = {
  130. "Error ",
  131. "Warning",
  132. "Trace ",
  133. "Mem ",
  134. "Result "
  135. };
  136. void
  137. BuildDebugHeader(
  138. DWORD Mask,
  139. PSTR pszHeader,
  140. PDWORD pcbHeader)
  141. {
  142. SYSTEMTIME stTime;
  143. DWORD Level = 0;
  144. ULONG ClientProcess;
  145. ULONG ClientThread;
  146. GetLocalTime(&stTime);
  147. Level = 0;
  148. while (!(Mask & 1))
  149. {
  150. Level++;
  151. Mask >>= 1;
  152. }
  153. if (Level >= sizeof(aszSPDebugLevel) / sizeof(char *))
  154. {
  155. Level = sizeof(aszSPDebugLevel) / sizeof(char *) - 1;
  156. }
  157. SslGetClientProcess(&ClientProcess);
  158. SslGetClientThread(&ClientThread);
  159. *pcbHeader = 0;
  160. if(g_dwInfoLevel & SP_LOG_TIMESTAMP)
  161. {
  162. *pcbHeader = wsprintf(
  163. pszHeader,
  164. "[%2d/%2d %02d:%02d:%02d.%03d] %d.%d> %s: ",
  165. stTime.wMonth, stTime.wDay,
  166. stTime.wHour, stTime.wMinute, stTime.wSecond,
  167. stTime.wMilliseconds,
  168. ClientProcess, ClientThread,
  169. aszSPDebugLevel[Level]);
  170. }
  171. else
  172. {
  173. *pcbHeader = wsprintf(
  174. pszHeader,
  175. "%d.%d> %s: ",
  176. ClientProcess, ClientThread,
  177. aszSPDebugLevel[Level]);
  178. }
  179. }
  180. void
  181. SPDebugLog(long Mask, const char *Format, ...)
  182. {
  183. va_list ArgList;
  184. int PrefixSize = 0;
  185. int iOut;
  186. char szOutString[MAX_DEBUG_BUFFER];
  187. long OriginalMask = Mask;
  188. if (Mask & g_dwInfoLevel)
  189. {
  190. BuildDebugHeader(Mask, szOutString, &iOut);
  191. PrefixSize = min(60, PctTraceIndent * 3);
  192. FillMemory(szOutString+iOut, PrefixSize, ' ');
  193. PrefixSize += iOut;
  194. szOutString[PrefixSize] = '\0';
  195. va_start(ArgList, Format);
  196. if (wvsprintf(&szOutString[PrefixSize], Format, ArgList) < 0)
  197. {
  198. static char szOverFlow[] = "\n<256 byte OVERFLOW!>\n";
  199. // Less than zero indicates that the string would not fit into the
  200. // buffer. Output a special message indicating overflow.
  201. lstrcpy(
  202. &szOutString[sizeof(szOutString) - sizeof(szOverFlow)],
  203. szOverFlow);
  204. }
  205. va_end(ArgList);
  206. SPDebugOutput(szOutString);
  207. }
  208. }
  209. void
  210. SPLogDistinguishedName(
  211. DWORD LogLevel,
  212. LPSTR pszLabel,
  213. PBYTE pbName,
  214. DWORD cbName)
  215. {
  216. CERT_NAME_BLOB Name;
  217. LPSTR pszName;
  218. DWORD cchName;
  219. Name.pbData = pbName;
  220. Name.cbData = cbName;
  221. cchName = CertNameToStr(CRYPT_ASN_ENCODING,
  222. &Name,
  223. CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG,
  224. NULL,
  225. 0);
  226. if(cchName == 0)
  227. {
  228. return;
  229. }
  230. SafeAllocaAllocate(pszName, cchName);
  231. if(pszName == NULL)
  232. {
  233. return;
  234. }
  235. cchName = CertNameToStr(CRYPT_ASN_ENCODING,
  236. &Name,
  237. CERT_X500_NAME_STR | CERT_NAME_STR_NO_PLUS_FLAG,
  238. pszName,
  239. cchName);
  240. if(cchName == 0)
  241. {
  242. SafeAllocaFree(pszName);
  243. return;
  244. }
  245. DebugLog((LogLevel, pszLabel, pszName));
  246. SafeAllocaFree(pszName);
  247. }
  248. long
  249. SPLogErrorCode(
  250. long err,
  251. const char *szFile,
  252. long lLine)
  253. {
  254. char *szName = "Unknown";
  255. switch(err)
  256. {
  257. case PCT_ERR_OK: szName = "PCT_ERR_OK"; break;
  258. case PCT_ERR_BAD_CERTIFICATE: szName = "PCT_ERR_BAD_CERTIFICATE"; break;
  259. case PCT_ERR_CLIENT_AUTH_FAILED: szName = "PCT_ERR_CLIENT_AUTH_FAILED"; break;
  260. case PCT_ERR_ILLEGAL_MESSAGE: szName = "PCT_ERR_ILLEGAL_MESSAGE"; break;
  261. case PCT_ERR_INTEGRITY_CHECK_FAILED: szName = "PCT_ERR_INTEGRITY_CHECK_FAILED"; break;
  262. case PCT_ERR_SERVER_AUTH_FAILED: szName = "PCT_ERR_SERVER_AUTH_FAILED"; break;
  263. case PCT_ERR_SPECS_MISMATCH: szName = "PCT_ERR_SPECS_MISMATCH"; break;
  264. case PCT_ERR_SSL_STYLE_MSG: szName = "PCT_ERR_SSL_STYLE_MSG"; break;
  265. case PCT_ERR_RENEGOTIATE: szName = "PCT_ERR_RENEGOTIATE"; break;
  266. case PCT_ERR_UNKNOWN_CREDENTIAL: szName = "PCT_ERR_UNKNOWN_CREDENTIAL"; break;
  267. case PCT_INT_BUFF_TOO_SMALL: szName = "PCT_INT_BUFF_TOO_SMALL"; break;
  268. case PCT_INT_INCOMPLETE_MSG: szName = "PCT_INT_INCOMPLETE_MSG"; break;
  269. case PCT_INT_DROP_CONNECTION: szName = "PCT_INT_DROP_CONNECTION"; break;
  270. case PCT_INT_BAD_CERT: szName = "PCT_INT_BAD_CERT"; break;
  271. case PCT_INT_CLI_AUTH: szName = "PCT_INT_CLI_AUTH"; break;
  272. case PCT_INT_ILLEGAL_MSG: szName = "PCT_INT_ILLEGAL_MSG"; break;
  273. case PCT_INT_MSG_ALTERED: szName = "PCT_INT_MSG_ALTERED"; break;
  274. case PCT_INT_INTERNAL_ERROR: szName = "PCT_INT_INTERNAL_ERROR"; break;
  275. case PCT_INT_DATA_OVERFLOW: szName = "PCT_INT_DATA_OVERFLOW"; break;
  276. case PCT_INT_SPECS_MISMATCH: szName = "PCT_INT_SPECS_MISMATCH"; break;
  277. case PCT_INT_RENEGOTIATE: szName = "PCT_INT_RENEGOTIATE"; break;
  278. case PCT_INT_UNKNOWN_CREDENTIAL: szName = "PCT_INT_UNKNOWN_CREDENTIAL"; break;
  279. case SEC_E_INSUFFICIENT_MEMORY: szName = "SEC_E_INSUFFICIENT_MEMORY"; break;
  280. case SEC_E_INVALID_HANDLE: szName = "SEC_E_INVALID_HANDLE"; break;
  281. case SEC_E_UNSUPPORTED_FUNCTION: szName = "SEC_E_UNSUPPORTED_FUNCTION"; break;
  282. case SEC_E_TARGET_UNKNOWN: szName = "SEC_E_TARGET_UNKNOWN"; break;
  283. case SEC_E_INTERNAL_ERROR: szName = "SEC_E_INTERNAL_ERROR"; break;
  284. case SEC_E_SECPKG_NOT_FOUND: szName = "SEC_E_SECPKG_NOT_FOUND"; break;
  285. case SEC_E_NOT_OWNER: szName = "SEC_E_NOT_OWNER"; break;
  286. case SEC_E_CANNOT_INSTALL: szName = "SEC_E_CANNOT_INSTALL"; break;
  287. case SEC_E_INVALID_TOKEN: szName = "SEC_E_INVALID_TOKEN"; break;
  288. case SEC_E_CANNOT_PACK: szName = "SEC_E_CANNOT_PACK"; break;
  289. case SEC_E_QOP_NOT_SUPPORTED: szName = "SEC_E_QOP_NOT_SUPPORTED"; break;
  290. case SEC_E_NO_IMPERSONATION: szName = "SEC_E_NO_IMPERSONATION"; break;
  291. case SEC_E_LOGON_DENIED: szName = "SEC_E_LOGON_DENIED"; break;
  292. case SEC_E_UNKNOWN_CREDENTIALS: szName = "SEC_E_UNKNOWN_CREDENTIALS"; break;
  293. case SEC_E_NO_CREDENTIALS: szName = "SEC_E_NO_CREDENTIALS"; break;
  294. case SEC_E_MESSAGE_ALTERED: szName = "SEC_E_MESSAGE_ALTERED"; break;
  295. case SEC_E_OUT_OF_SEQUENCE: szName = "SEC_E_OUT_OF_SEQUENCE"; break;
  296. case SEC_E_NO_AUTHENTICATING_AUTHORITY: szName = "SEC_E_NO_AUTHENTICATING_AUTHORITY"; break;
  297. case SEC_I_CONTINUE_NEEDED: szName = "SEC_I_CONTINUE_NEEDED"; break;
  298. case SEC_I_COMPLETE_NEEDED: szName = "SEC_I_COMPLETE_NEEDED"; break;
  299. case SEC_I_COMPLETE_AND_CONTINUE: szName = "SEC_I_COMPLETE_AND_CONTINUE"; break;
  300. case SEC_I_LOCAL_LOGON: szName = "SEC_I_LOCAL_LOGON"; break;
  301. case SEC_E_BAD_PKGID: szName = "SEC_E_BAD_PKGID"; break;
  302. case SEC_E_CONTEXT_EXPIRED: szName = "SEC_E_CONTEXT_EXPIRED"; break;
  303. case SEC_E_INCOMPLETE_MESSAGE: szName = "SEC_E_INCOMPLETE_MESSAGE"; break;
  304. case SEC_E_INCOMPLETE_CREDENTIALS: szName = "SEC_E_INCOMPLETE_CREDENTIALS"; break;
  305. case SEC_E_BUFFER_TOO_SMALL: szName = "SEC_E_BUFFER_TOO_SMALL"; break;
  306. case SEC_I_INCOMPLETE_CREDENTIALS: szName = "SEC_I_INCOMPLETE_CREDENTIALS"; break;
  307. case SEC_I_RENEGOTIATE: szName = "SEC_I_RENEGOTIATE"; break;
  308. case SEC_E_WRONG_PRINCIPAL: szName = "SEC_E_WRONG_PRINCIPAL"; break;
  309. case SEC_I_NO_LSA_CONTEXT: szName = "SEC_I_NO_LSA_CONTEXT"; break;
  310. case CERT_E_EXPIRED: szName = "CERT_E_EXPIRED"; break;
  311. case CERT_E_UNTRUSTEDROOT: szName = "CERT_E_UNTRUSTEDROOT"; break;
  312. case CRYPT_E_REVOKED: szName = "CRYPT_E_REVOKED"; break;
  313. case CRYPT_E_NO_REVOCATION_CHECK: szName = "CRYPT_E_NO_REVOCATION_CHECK"; break;
  314. case CRYPT_E_REVOCATION_OFFLINE: szName = "CRYPT_E_REVOCATION_OFFLINE"; break;
  315. }
  316. SPDebugLog(SP_LOG_RES, "Result: %s (0x%lx) - %s, Line %d\n", szName, err, szFile, lLine);
  317. return err;
  318. }
  319. #pragma warning(disable:4206) /* Disable the empty translation unit */
  320. /* warning/error */
  321. void
  322. SPAssert(
  323. void * FailedAssertion,
  324. void * FileName,
  325. unsigned long LineNumber,
  326. char * Message)
  327. {
  328. SPDebugLog(SP_LOG_ERROR,
  329. "Assertion FAILED, %s, %s : %d\n",
  330. FailedAssertion,
  331. FileName,
  332. LineNumber);
  333. DebugBreak();
  334. }
  335. VOID *
  336. DBGAllocMem(DWORD cb, LPSTR szFile, DWORD iLine)
  337. {
  338. VOID *pv;
  339. DWORD cbTotal = cb;
  340. pv = LocalAlloc(LMEM_FIXED | LMEM_ZEROINIT, cbTotal);
  341. if(pv == NULL)
  342. {
  343. SPDebugLog(SP_LOG_ERROR,
  344. "Local Alloc Failed: %s, %d\n",
  345. szFile,
  346. iLine);
  347. DebugBreak();
  348. }
  349. return(pv);
  350. }
  351. VOID
  352. DBGFreeMem(VOID *pv, LPSTR szFile, DWORD iLine)
  353. {
  354. PVOID pvLocal = pv;
  355. LocalFree(pvLocal);
  356. }
  357. #endif /* DBG */ /* NOTE: This file not compiled for retail builds */