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.

1575 lines
47 KiB

  1. /* sherlock.c - Help deduce cause of Unrecoverable Application Errors
  2. (C) Copyright 1991, Microsoft Corp
  3. Written by Don Corbitt, based upon work of Cameron Stevens and others.
  4. Features -
  5. Help Script
  6. Dialog box to change options
  7. Option to trap Ctrl-Alt-SysRq to break endless loops
  8. Disassembler should look up symbols for CALL instructions
  9. Button could call up editor - file extension associations
  10. Toggle Icon when message occurs
  11. Write formal spec - (Program is done, so write the spec)
  12. Disable operation in Real Mode
  13. Dump stack bytes
  14. If all is blown, dump message to text monitor
  15. Data Symbols in disassembly
  16. Bugs -
  17. Doesn't buffer file output - this could be slow (but how many GP/hour?)
  18. Need to watch for invalid memory
  19. Need to handle Jump to bad address
  20. What if there aren't any file handles left at fault time???
  21. Need to handle no file handles available for .SYM file reader
  22. Should open files (.sym, .log) with proper share flags
  23. Could dump config.sys and autoexec.bat also
  24. Can't handle fault in Sherlock - locks up machine - very ugly
  25. Need to check InDOS flag
  26. Some errors not detected
  27. Jump/Call to invalid address
  28. Load invalid selector
  29. Run twice in Real Mode causes system hang
  30. GP Continue doesn't update 32 bit registers for string moves
  31. */
  32. #define DRWATSON_C
  33. #define STRICT
  34. #include <windows.h>
  35. #include <string.h> /* strcpy() */
  36. #include <stdarg.h> /* va_stuff() */
  37. #include <io.h> /* dup() - why is this spread over 3 files ??? */
  38. #include "toolhelp.h" /* all the good stuff */
  39. #include "disasm.h" /* DisAsm86(), memXxxx vars */
  40. #include "drwatson.h"
  41. #include "str.h" /* Support for string resources */
  42. #define STATIC /*static */
  43. char far foo[1]; /* force far data seg, make single instance */
  44. // Does this make sense considering we have code and strings to
  45. // tell the user they're in error to run two copies of Dr. Watson?
  46. /******************/
  47. /***** Macros *****/
  48. /******************/
  49. #define version "1.00b"
  50. /* This string is concatenated with other strings in various places */
  51. /* so it can't be an array variable. It must stay a #define. */
  52. /* These strings are not localized. */
  53. #define szAppNameMacro "Dr. Watson"
  54. #define szAppNameShortMacro "drwatson"
  55. STATIC char szAppName[] = szAppNameMacro;
  56. STATIC char szAppNameShort[] = szAppNameShortMacro;
  57. STATIC char szAppNameShortLog[] = szAppNameShortMacro ".log";
  58. static char szAppNameVers[] = szAppNameMacro " " version;
  59. #define YOO_HOO (WM_USER+22) /* user activated Dr. Watson */
  60. #define HEAP_BIG_FILE (WM_USER+23) /* log file is getting large */
  61. #define JUST_THE_FACTS (WM_USER+24) /* tell me about your problem */
  62. #define BIG_FILE 100000L
  63. /* Don't like MSC-style FP macros, use my own */
  64. #undef MK_FP
  65. #undef FP_SEG
  66. #undef FP_OFF
  67. #define MK_FP(seg, off) (void far *)(((long)(seg) << 16) | (unsigned short)(off))
  68. #define FP_SEG(fp) (unsigned)((long)(fp) >> 16)
  69. #define FP_OFF(fp) (unsigned)(long)fp
  70. /***************************/
  71. /***** Data Structures *****/
  72. /***************************/
  73. LPSTR aszStrings[STRING_COUNT];
  74. /* This points to the stack the GP fault handler should use */
  75. char *newsp;
  76. /* These structures are used by Watson.asm and Disasm.c - don't change */
  77. /* Also, they can't be static. They contain the CPU register contents */
  78. /* at the time the fault occurred. */
  79. struct {
  80. word ax, cx, dx, bx, sp, bp, si, di, ip, flags;
  81. word es, cs, ss, ds, fs, gs, intNum;
  82. } regs;
  83. /* If we have a 32 bit CPU, the full 32 bit values will be stored here. */
  84. /* The lower 16 bits will still be in the generic regs above. */
  85. struct {
  86. DWORD eax, ecx, edx, ebx, esp, ebp, esi, edi, eip, eflags;
  87. } regs32;
  88. /* Each of these flags disables a part of the error output report */
  89. /* The error report itself indicates how each section is named. */
  90. /* The word in () can be added to the [Dr. Watson] section of WIN.INI */
  91. /* to disable that section of the report. */
  92. /* clu Clues dialog box */
  93. /* deb OutputDebugString trapping */
  94. /* dis Simple disassembly */
  95. /* err Error logging */
  96. /* inf System info */
  97. /* loc Local vars on stack dump */
  98. /* mod Module dump */
  99. /* par Parameter error logging */
  100. /* reg Register dump */
  101. /* sum 3 line summary */
  102. /* seg not visible to users, but available */
  103. /* sou But I _like_ the sound effects! */
  104. /* sta Stack trace */
  105. /* tas Task dump */
  106. /* tim Time start/stop */
  107. /* 32b 32 bit register dump */
  108. STATIC char syms[] =
  109. "clu deb dis err inf lin loc mod par reg sum seg sou sta tas tim 32b ";
  110. #define cntFlag (sizeof(syms)/4)
  111. /* This array is used to decode the flags in WIN.INI. I only check */
  112. /* the first 3 chars of an entry. Each entry must be separated by a */
  113. /* space from the previous one. */
  114. unsigned long ddFlag;
  115. int retflag; /* used in watson.asm */
  116. struct {
  117. char bit, name;
  118. } flBit[] = {
  119. 11, 'O',
  120. 10, 'D',
  121. 9, 'I',
  122. 7, 'S',
  123. 6, 'Z',
  124. 4, 'A',
  125. 2, 'P',
  126. 0, 'C',
  127. };
  128. #define cntFlBit (sizeof(flBit)/sizeof(flBit[0]))
  129. STATIC int disLen = 8; /* Number of instructions to disassemble */
  130. STATIC int trapZero = 0; /* Should I trap divide by 0 faults */
  131. STATIC int iFeelLucky = 1; /* Should we restart after GP fault? */
  132. /* 1 = allow continue
  133. 2 = skip report
  134. 4 = continue in Kernel
  135. 8 = continue in User
  136. 16 = allow sound
  137. */
  138. STATIC int imTrying; /* trying to continue operation */
  139. STATIC struct {
  140. FARPROC adr;
  141. WORD code;
  142. HTASK task;
  143. DWORD parm;
  144. } lastErr;
  145. STATIC int disStack = 2; /* Disassemble 2 levels of stack trace */
  146. int cpu32; /* True if cpu has 32 bit regs */
  147. STATIC int fh = -1; /* Handle of open log file */
  148. STATIC int level; /* if >0, in nested FileOpen() call */
  149. STATIC int bugCnt, sound;
  150. STATIC int pending; /* If a pending Clues dialog */
  151. STATIC int whined; /* If already warned about a large file */
  152. STATIC long pitch, deltaPitch = 250L << 16;
  153. STATIC HINSTANCE hInst;
  154. STATIC char logFile[80]; /* Default log file is "drwatson.log" */
  155. /* and is stored in the windows dir */
  156. STATIC struct { /* Help print out value of CPU flags */
  157. WORD mask;
  158. LPSTR name;
  159. } wf[] = {
  160. WF_80x87, (LPSTR) IDSTRCoprocessor, // IDSTRs are fixed up to pointers
  161. WF_CPU086, (LPSTR) IDSTR8086, // by LoadStringResources
  162. WF_CPU186, (LPSTR) IDSTR80186,
  163. WF_CPU286, (LPSTR) IDSTR80286,
  164. WF_CPU386, (LPSTR) IDSTR80386,
  165. WF_CPU486, (LPSTR) IDSTR80486,
  166. WF_ENHANCED, (LPSTR) IDSTREnhancedMode,
  167. WF_PMODE, (LPSTR) IDSTRProtectMode,
  168. WF_STANDARD, (LPSTR) IDSTRStandardMode,
  169. WF_WINNT, (LPSTR) IDSTRWindowsNT,
  170. };
  171. #define wfCnt (sizeof(wf)/sizeof(wf[0]))
  172. HWND hWnd; /* Handle to main window */
  173. HANDLE hTask; /* current task (me) */
  174. /***********************/
  175. /***** Extern Defs *****/
  176. /***********************/
  177. /* Get base 32 bit linear address of a memory segment - calls DPMI */
  178. extern DWORD SegBase(WORD segVal);
  179. /* Get segment flags - 0 if error */
  180. extern WORD SegRights(WORD segVal);
  181. /* Get (segment length -1) */
  182. extern DWORD SegLimit(WORD segVal);
  183. /* Fills in regs32 structure with value from regs struct and current high */
  184. /* word of registers - don't do any 32 bit ops before calling this func */
  185. extern void GetRegs32(void);
  186. /* Fills in non-standard time/date structure using DOS calls. The C */
  187. /* run-time has a similar function (asctime()), but it pulls in over */
  188. /* 6K of other functions. This is much smaller and faster, and */
  189. /* doesn't depend on environment variables, etc. */
  190. extern void GetTimeDate(void *tdstruc);
  191. /* Called by ToolHelp as a notify hook */
  192. extern BOOL far /*pascal*/ CallMe(WORD, DWORD);
  193. char *LogParamErrorStr(WORD err, FARPROC lpfn, DWORD param);
  194. extern int FindFile(void *ffstruct, char *name);
  195. /* This routine is called by ToolHelp when a GP fault occurs. It */
  196. /* switches stacks and calls Sherlock() to handle the fault. */
  197. extern void CALLBACK GPFault(void);
  198. /* Return name of nearest symbol in file, or 0 */
  199. extern char *NearestSym(int segIndex, unsigned offset, char *exeName);
  200. STATIC void cdecl Show(const LPSTR format, ...);
  201. /************************************/
  202. /***** Segment Helper Functions *****/
  203. /************************************/
  204. /************************************
  205. Name: LPSTR SegFlags(WORD segVal)
  206. Desc: Given a selector, SegFlags checks for validity and then returns
  207. an ascii string indicating whether it is a code or data selector,
  208. and read or writeable.
  209. Bugs: Should check other flags (accessed), and call gates.
  210. Returns pointer to static array, overwritten on each new call.
  211. *************************************/
  212. STATIC LPSTR SegFlags(WORD segVal) {
  213. static char flag[10];
  214. if (segVal == 0) return STR(NullPtr);
  215. segVal = SegRights(segVal);
  216. if (segVal == 0) return STR(Invalid);
  217. segVal >>= 8;
  218. if (!(0x80 & segVal)) return STR(NotPresent);
  219. if (segVal & 8) {
  220. lstrcpy(flag, STR(Code));
  221. lstrcat(flag, segVal & 2 ? STR(ExR) : STR(ExO));
  222. } else {
  223. lstrcpy(flag, STR(Data));
  224. lstrcat(flag, segVal&2 ? STR(RW) : STR(RO));
  225. }
  226. return flag;
  227. } /* SegFlags */
  228. /************************************
  229. Name: char *SegInfo(WORD seg)
  230. Desc: Given a selector, SegInfo returns an ascii string indicating the
  231. linear base address, limit, and attribute flags of the selector.
  232. Bugs: Returns pointer to static array, overwritten on each new call.
  233. *************************************/
  234. STATIC char *SegInfo(WORD seg) {
  235. static char info[30];
  236. if (noSeg) return "";
  237. wsprintf(info, "%8lx:%04lx %-9s",
  238. SegBase(seg), SegLimit(seg), (FP)SegFlags(seg));
  239. return info;
  240. } /* SegInfo */
  241. /************************************
  242. Name: WORD SegNum(WORD segVal)
  243. Desc: Returns the index of this segment in the module table. Used to
  244. translate between a physical segment number and the index as
  245. seen in e.g. the map file.
  246. Bugs: Don't know what ToolHelp returns for data or GlobalAlloc segments.
  247. This is mainly useful for converting a code segment value.
  248. Check for GT_DATA - will also be valid index.
  249. *************************************/
  250. STATIC WORD SegNum(HGLOBAL segVal) {
  251. GLOBALENTRY ge;
  252. ge.dwSize = sizeof(ge);
  253. if (GlobalEntryHandle(&ge, segVal) && (ge.wType == GT_CODE)) {
  254. return ge.wData; /* defined to be 'file segment index' */
  255. }
  256. return (WORD)-1;
  257. } /* SegNum */
  258. /************************************
  259. Name: LPSTR ModuleName(WORD segVal)
  260. Desc: Returns name of this code segment's module
  261. Bugs:
  262. *************************************/
  263. STATIC LPSTR ModuleName(WORD segVal) {
  264. static char name[12];
  265. GLOBALENTRY ge;
  266. MODULEENTRY me;
  267. ge.dwSize = sizeof(ge);
  268. me.dwSize = sizeof(me);
  269. if (GlobalEntryHandle(&ge, (HGLOBAL)segVal) && (ge.wType == GT_CODE)) {
  270. if (ModuleFindHandle(&me, ge.hOwner)) {
  271. strcpy(name, me.szModule);
  272. return name;
  273. } /* else Show("ModuleFindHandle() failed\n"); */
  274. } /* else Show("GlobalEntryHandle() failed\n"); */
  275. return STR(Unknown);
  276. } /* ModuleName */
  277. /**********************************/
  278. /***** Other Helper Functions *****/
  279. /**********************************/
  280. /************************************
  281. Name: char *FaultType(void)
  282. Desc: Returns ascii string indicating what kind of fault caused ToolHelp
  283. to call our GPFault handler.
  284. Bugs: May not handle Ctrl-Alt-SysR nicely (we shouldn't trap it)
  285. *************************************/
  286. /* static char *FaultType(void) {
  287. switch (regs.intNum) {
  288. case 0: return STR(DivideByZero);
  289. case 6: return STR(InvalidOpcode);
  290. case 13: return STR(GeneralProtection);
  291. default: return STR(Unknown);
  292. }
  293. } /* FaultType */
  294. /************************************
  295. Name: char *DecodeFault(int op, word seg, dword offset, word size)
  296. Desc: Pokes at memory address passed in, trying to determine fault cause
  297. Segment wrap-around
  298. Null selector
  299. Write to read only data
  300. Write to code segment
  301. Read from execute only code segment
  302. Exceed segment limit
  303. Invalid selector
  304. Bugs: Jump, string, call, and stack memory adr's aren't set by DisAsm
  305. *************************************/
  306. STATIC LPSTR DecodeFault(int op, word seg, dword offset, word size) {
  307. int v;
  308. dword lim;
  309. switch (op) {
  310. case memNOP:
  311. break; /* since no mem access, no fault */
  312. case memSegMem: /* load seg reg from memory */
  313. seg = *(short far *)MK_FP(seg, offset);
  314. /* fall through */
  315. case memSegReg: /* load seg reg with value */
  316. v = SegRights(seg); /* lets see if this is a selector */
  317. if (!v) return STR(InvalidSelector);
  318. break; /* See no evil... */
  319. case memRead:
  320. case memRMW:
  321. case memWrite:
  322. if (seg == 0) return STR(NullSelector);
  323. v = SegRights(seg);
  324. if (!v) return STR(InvalidSelector);
  325. v >>= 8;
  326. if (!(0x80 & v)) return STR(SegmentNotPresent);
  327. lim = SegLimit(seg);
  328. if (lim < (offset+size)) return STR(ExceedSegmentBounds);
  329. if (v & 8) { /* code segment */
  330. if ((op == memRMW) || (op == memWrite))
  331. return /* Write to */ STR(CodeSegment);
  332. else if (!(v&2)) return /* Read */ STR(ExecuteOnlySegment);
  333. } else { /* data segment */
  334. if (((op == memRMW) || (op == memWrite)) && !(v&2))
  335. return /* Write to */ STR(ReadOnlySegment);
  336. }
  337. break;
  338. default:
  339. return 0; /* obviously unknown condition */
  340. }
  341. return 0;
  342. } /* DecodeFault */
  343. LPSTR SafeDisAsm86(void far *code, int *len) {
  344. unsigned long limit = SegLimit(FP_SEG(code));
  345. if ((unsigned long)(FP_OFF(code)+10) > limit) {
  346. *len = 1;
  347. return STR(SegNotPresentOrPastEnd);
  348. }
  349. return DisAsm86(code, (int *)len);
  350. } /* SafeDisAsm86 */
  351. /************************************
  352. Name: LPSTR FaultCause(void)
  353. Desc: Decodes the actual cause of the fault. This is trivial for Div0
  354. and Invalid Opcode, but much trickier for GP Faults. I need to
  355. try to detect at least the following:
  356. Segment wrap-around
  357. Null selector
  358. Write to read only data
  359. Write to code segment
  360. Read from execute only code segment
  361. Exceed segment limit
  362. Invalid selector
  363. Bugs:
  364. *************************************/
  365. STATIC LPSTR FaultCause(void) {
  366. int foo;
  367. LPSTR s, s1;
  368. static char cause[54];
  369. switch (regs.intNum) {
  370. case 0: return STR(DivideByZero);
  371. case 6: return STR(InvalidOpcode);
  372. case 20: return STR(ErrorLog);
  373. case 21: return STR(ParameterErrorLog);
  374. case 13:
  375. SafeDisAsm86(MK_FP(regs.cs, regs.ip), &foo); /* Set global memXxxx vars */
  376. /* See if first memory access caused fault */
  377. s = DecodeFault(memOp, memSeg, memLinear, memSize);
  378. s1 = memName[memOp];
  379. /* no, see if second memory access caused fault */
  380. if (!s && memDouble) {
  381. s = DecodeFault(memOp2, memSeg2, memLinear2, memSize2);
  382. s1 = memName[memOp2];
  383. }
  384. if (s) {
  385. wsprintf(cause, "%s (%s)", s, s1);
  386. return cause;
  387. }
  388. }
  389. return STR(Unknown);
  390. } /* FaultCause */
  391. /************************************
  392. Name: LPSTR CurModuleName(hTask task)
  393. Desc: Call ToolHelp to find name of faulting module
  394. Bugs:
  395. *************************************/
  396. STATIC LPSTR CurModuleName(HTASK hTask) {
  397. TASKENTRY te;
  398. static char name[10];
  399. te.dwSize = sizeof(te);
  400. if (!TaskFindHandle(&te, hTask)) /* Thanks, ToolHelp */
  401. return STR(Unknown);
  402. strcpy(name, te.szModule);
  403. return name;
  404. } /* ModuleName */
  405. /************************************
  406. Name: LPSTR FileInfo(char *name)
  407. Desc: Find file time, date, and size
  408. Bugs:
  409. *************************************/
  410. STATIC LPSTR FileInfo(char *name) {
  411. struct {
  412. char resv[21];
  413. char attr;
  414. unsigned time;
  415. unsigned date;
  416. long len;
  417. char name[13];
  418. char resv1[10];
  419. } f;
  420. static char buf[30];
  421. if (FindFile(&f, name)) return STR(FileNotFound);
  422. wsprintf(buf, "%7ld %02d-%02d-%02d %2d:%02d",
  423. f.len,
  424. (f.date >> 5) & 15, f.date & 31, (f.date >> 9) + 80,
  425. f.time >> 11, (f.time >> 5) & 63);
  426. return buf;
  427. } /* FileInfo */
  428. /************************************
  429. Name: char *CurFileName(void)
  430. Desc: Call ToolHelp to find filename and path of faulting module
  431. Bugs:
  432. *************************************/
  433. /* STATIC char *CurFileName(void) {
  434. TASKENTRY te;
  435. MODULEENTRY me;
  436. static char name[80];
  437. te.dwSize = sizeof(te);
  438. me.dwSize = sizeof(me);
  439. if (!TaskFindHandle(&te, GetCurrentTask()) ||
  440. !ModuleFindName(&me, te.szModule))
  441. return STR(Unknown);
  442. strcpy(name, me.szExePath);
  443. return name;
  444. } /* FileName */
  445. /************************************
  446. Name: char *CurTime(void)
  447. Desc: Generates string with current time and date. Similar to asctime(),
  448. except it doesn't pull in another 6K of run-time library code :-)
  449. Bugs: Magic structure passed to asm routine
  450. *************************************/
  451. STATIC char *CurTime(void) {
  452. static char t[48];
  453. struct { /* This magic struct is hard-coded to */
  454. char week, resv; /* match the assembly language in */
  455. short year; /* watson.asm GetTimeDate() */
  456. char day, month; /* This means I recommend you don't */
  457. char minute, hour; /* change the size or order of the */
  458. char hund, second; /* fields! */
  459. } td;
  460. GetTimeDate(&td);
  461. wsprintf(t, "%s %s %2d %02d:%02d:%02d %d",
  462. aszStrings[IDSTRSun + td.week], aszStrings[IDSTRJan + td.month - 1],
  463. td.day, td.hour, td.minute, td.second, td.year);
  464. return t;
  465. } /* CurTime */
  466. /************************************
  467. Name: LPSTR Tab2Spc(LPSTR temp)
  468. Desc: Converts tabs found in string 'temp' into the proper number of
  469. spaces. I need this since DisAsm86() returns a string with tabs
  470. in it, and TextOut() didn't like them. This was easier than
  471. getting TabbedTextOut() set up to work. Since I'm no longer dumping
  472. to the screen, this routine may be superfluous.
  473. Bugs:
  474. *************************************/
  475. STATIC LPSTR Tab2Spc(LPSTR temp) {
  476. char newbuf[80];
  477. LPSTR s1, s2;
  478. s1 = temp;
  479. s2 = newbuf;
  480. while ((*s2 = *s1++) != 0) {
  481. if (*s2++ == 9) {
  482. s2[-1] = ' ';
  483. while ((s2-(LPSTR)newbuf) & 7) *s2++ = ' ';
  484. }
  485. }
  486. lstrcpy(temp, newbuf);
  487. return temp;
  488. } /* Tab2Spc */
  489. /************************************
  490. Name: void Show(const LPSTR format, ...)
  491. Desc: Think of this as (minor) shortcut fprintf(). I originally had this
  492. dumping info to a Windows window, and then changed it to write to
  493. the file we want. All output goes through this func, so if you
  494. want to change something, this is the place.
  495. Bugs: Now writing to a file handle, opened in text mode so it does the
  496. LF->CR/LF translation for me.
  497. No buffering performed on writes, except for what DOS might do.
  498. Blows up if stuff passed in expands to longer than 200 chars.
  499. *************************************/
  500. STATIC void cdecl Show(const LPSTR format, ...) {
  501. char line[CCH_MAX_STRING_RESOURCE];
  502. char *prev, *cur;
  503. wvsprintf(line, format, (LPSTR)(&format + 1));
  504. if (fh != -1) {
  505. prev = cur = line;
  506. while (*cur) { /* expand LF to CR/LF */
  507. if (cur[0] == '\n' && /* at LF */
  508. ((prev == cur) || /* and first of line */
  509. (cur[-1] != '\r'))) { /* or previous wasn't CR */
  510. cur[0] = '\r'; /* append CR to text up to LF */
  511. _lwrite(fh, prev, cur-prev+1);
  512. cur[0] = '\n'; /* leave LF for next write */
  513. prev = cur;
  514. }
  515. cur++;
  516. }
  517. if (prev != cur) /* write trailing part */
  518. _lwrite(fh, prev, cur-prev);
  519. }
  520. } /* Show */
  521. /************************************
  522. Name: void MyFlush(void)
  523. Desc: Any routine named MyXxxx() had better be a private hack, and this
  524. one is. It just appends an extra CRLF to the output file, and makes
  525. sure that the info written so far makes it to disk. This way, if
  526. a later part of the program blows up, at least you will know this
  527. much.
  528. Bugs:
  529. *************************************/
  530. STATIC void MyFlush(void) {
  531. int h;
  532. Show("\n");
  533. if (fh != -1) {
  534. h = dup(fh);
  535. if (h != -1) _lclose(h);
  536. }
  537. if (sound) {
  538. StopSound();
  539. SetVoiceSound(1, pitch, 20);
  540. pitch += deltaPitch;
  541. StartSound();
  542. }
  543. } /* MyFlush */
  544. /************************************
  545. Name: void DisAsmAround(char far *cp, int count)
  546. Desc: The 'cp' parameter is a pointer to a code segment in memory. This
  547. routine backs up a few instructions from the current point, and
  548. dumps a disassembly showing the context of the selected instruction.
  549. Bugs: Needs to check for segmentation problems, such as invalid selector.
  550. *************************************/
  551. STATIC void DisAsmAround(byte far *cp, int count) {
  552. int len, back;
  553. byte far *oldcp = cp;
  554. byte far *cp1;
  555. GLOBALENTRY ge;
  556. MODULEENTRY me;
  557. char *szSym = 0;
  558. long limit;
  559. unsigned segLim;
  560. char symBuf[40];
  561. ge.dwSize = sizeof(ge);
  562. me.dwSize = sizeof(me);
  563. if (GlobalEntryHandle(&ge, (HGLOBAL)FP_SEG(cp)) && (ge.wType == GT_CODE)) {
  564. if (ModuleFindHandle(&me, ge.hOwner)) {
  565. szSym = NearestSym(ge.wData, FP_OFF(cp), me.szExePath);
  566. if (!szSym) { /* if we know module name, but no syms */
  567. sprintf(symBuf, "%d:%04x", ge.wData, FP_OFF(cp));
  568. szSym = symBuf;
  569. }
  570. }
  571. }
  572. cp -= count*2 + 10; /* back up */
  573. if ((FP_OFF(cp) & 0xff00) == 0xff00) /* if wrapped around, trunc to 0 */
  574. cp = MK_FP(FP_SEG(cp), 0);
  575. cp1 = cp;
  576. limit = SegLimit(FP_SEG(cp));
  577. segLim = limit > 0xffffL ? 0xffff : (int)limit;
  578. if (segLim == 0) {
  579. Show(STR(CodeSegmentNPOrInvalid));
  580. return;
  581. }
  582. back = 0;
  583. while (cp < oldcp) { /* count how many instructions to point */
  584. SafeDisAsm86(cp, &len);
  585. cp += len;
  586. back++;
  587. }
  588. cp = cp1;
  589. back -= (count >> 1);
  590. while (back>0) { /* step forward until (len/2) remain */
  591. SafeDisAsm86(cp, &len); /* before desired instruction point */
  592. cp += len;
  593. back--;
  594. }
  595. while (count--) { /* display desired instructions */
  596. if (cp == oldcp) {
  597. if (szSym) Show("(%s:%s)\n", (FP)me.szModule, (FP)szSym);
  598. else Show(STR(NoSymbolsFound));
  599. }
  600. Show("%04x:%04x %-22s %s\n",
  601. FP_SEG(cp), FP_OFF(cp), /* address */
  602. (FP)hexData, /* opcodes in hex */
  603. (FP)/*Tab2Spc*/(SafeDisAsm86(cp, &len)));/* actual disassembly */
  604. cp += len;
  605. }
  606. } /* DisAsmAround */
  607. /************************************
  608. Name: int MyOpen(void)
  609. Desc: Tries to open logFile for append. If this fails, tries to
  610. create it.
  611. Bugs: Should set sharing flags?
  612. *************************************/
  613. STATIC int MyOpen(void) {
  614. if (fh != -1) return fh; /* Already open */
  615. fh = _lopen(logFile, OF_WRITE | OF_SHARE_DENY_WRITE);
  616. if (fh == -1) {
  617. fh = _lcreat(logFile, 0);
  618. } else _llseek(fh, 0L, 2);
  619. if (fh != -1) level++;
  620. return fh != -1;
  621. } /* MyOpen */
  622. /************************************
  623. Name: void MyClose(void)
  624. Desc: close output file, clear handle to -1
  625. Bugs: Should set sharing flags?
  626. *************************************/
  627. STATIC void MyClose(void) {
  628. if (--level == 0) {
  629. if (fh != -1) _lclose(fh);
  630. fh = -1;
  631. }
  632. } /* MyClose */
  633. void PutDate(LPSTR msg) {
  634. MyOpen();
  635. if (fh == -1) return;
  636. Show("%s %s - %s\n", (FP)msg, (FP)szAppNameVers, (FP)CurTime());
  637. MyClose();
  638. } /* PutDate */
  639. int far pascal SherlockDialog(HWND hDlg, WORD wMsg, WPARAM wParam, LPARAM lParam) {
  640. char line[255];
  641. int i, len, count;
  642. HWND hItem;
  643. lParam = lParam;
  644. if (wMsg == WM_INITDIALOG) return 1;
  645. if ((wMsg != WM_COMMAND) ||
  646. (wParam != IDOK && wParam != IDCANCEL))
  647. return 0;
  648. if (wParam == IDOK) {
  649. MyOpen();
  650. if (fh != -1) {
  651. hItem = GetDlgItem(hDlg, 102);
  652. if (hItem) {
  653. count = (int)SendMessage(hItem, EM_GETLINECOUNT, 0, 0L);
  654. for (i=0; i<count; i++) {
  655. *(int *)line = sizeof(line) - sizeof(int) -1;
  656. len = (int)SendMessage(hItem, EM_GETLINE, i, (long)((void far *)line));
  657. line[len] = 0;
  658. Show("%d> %s\n", i+1, (FP)line);
  659. }
  660. }
  661. MyClose();
  662. }
  663. }
  664. EndDialog(hDlg, 0);
  665. return 1;
  666. } /* SherlockDialog */
  667. extern int far pascal SysErrorBox(char far *text, char far *caption,
  668. int b1, int b2, int b3);
  669. #define SEB_OK 1 /* Button with "OK". */
  670. #define SEB_CANCEL 2 /* Button with "Cancel" */
  671. #define SEB_YES 3 /* Button with "&Yes" */
  672. #define SEB_NO 4 /* Button with "&No" */
  673. #define SEB_RETRY 5 /* Button with "&Retry" */
  674. #define SEB_ABORT 6 /* Button with "&Abort" */
  675. #define SEB_IGNORE 7 /* Button with "&Ignore" */
  676. #define SEB_CLOSE 8 /* Button with "Close" */
  677. #define SEB_DEFBUTTON 0x8000 /* Mask to make this button default */
  678. #define SEB_BTN1 1 /* Button 1 was selected */
  679. #define SEB_BTN2 2 /* Button 1 was selected */
  680. #define SEB_BTN3 3 /* Button 1 was selected */
  681. /************************************
  682. Name: int PrepareToParty(LPSTR modName, LPSTR appName)
  683. Desc: Checks whether we can continue the current app by skipping an
  684. instruction. If so, it performs the side effects of the
  685. instruction. This must be called after a call to DisAsm86() has
  686. set the gpXxxx global vars.
  687. Checks value of iFeelLucky, bit 0 must be set to continue a fault.
  688. Bugs: Should do more checking, should check for within a device driver,
  689. shouldn't require that DisAsm86() be called for the failing
  690. instruction immediately before call.
  691. *************************************/
  692. int PrepareToParty(LPSTR modName, LPSTR appName) {
  693. if (!(iFeelLucky&1)) return 0;
  694. if (!gpSafe) return 0;
  695. /* compare module to KERNEL */
  696. if (!(iFeelLucky&4) && !lstrcmp(modName, "KERNEL")) return 0;
  697. /* compare module to USER */
  698. if (!(iFeelLucky&8) && !lstrcmp(modName, "USER")) return 0;
  699. /* already asked, trying to continue, skip this fault */
  700. if (imTrying>0) return 1;
  701. if (3 != SysErrorBox(STR(GPText), appName, SEB_CLOSE|SEB_DEFBUTTON, 0, SEB_IGNORE))
  702. return 0;
  703. imTrying = 100;
  704. return 1;
  705. } /* PrepareToParty */
  706. STATIC void DumpInfo(void) {
  707. WORD w = (int)GetVersion();
  708. DWORD lw = GetWinFlags();
  709. SYSHEAPINFO si;
  710. int i;
  711. MEMMANINFO mm;
  712. Show(STR(SystemInfoInfo));
  713. Show(STR(WindowsVersion), w&0xff, w>>8);
  714. if (GetSystemMetrics(SM_DEBUG)) Show(STR(DebugBuild));
  715. else Show(STR(RetailBuild));
  716. {
  717. HANDLE hUser = GetModuleHandle("USER");
  718. char szBuffer[80];
  719. if (LoadString(hUser, 516, szBuffer, sizeof(szBuffer)))
  720. Show(STR(WindowsBuild), (FP)szBuffer);
  721. if (LoadString(hUser, 514, szBuffer, sizeof(szBuffer)))
  722. Show(STR(Username), (FP)szBuffer);
  723. if (LoadString(hUser, 515, szBuffer, sizeof(szBuffer)))
  724. Show(STR(Organization), (FP)szBuffer);
  725. }
  726. Show(STR(SystemFreeSpace), GetFreeSpace(0));
  727. if (SegLimit(regs.ss) > 0x10) {
  728. int far *ip = MK_FP(regs.ss, 0);
  729. Show(STR(StackBaseTopLowestSize),
  730. ip[5], ip[7], ip[6], ip[7]-ip[5]);
  731. }
  732. si.dwSize = sizeof(si);
  733. if (SystemHeapInfo(&si))
  734. Show(STR(SystemResourcesUserGDI),
  735. si.wUserFreePercent, si.hUserSegment,
  736. si.wGDIFreePercent, si.hGDISegment);
  737. mm.dwSize = sizeof(mm);
  738. if (MemManInfo(&mm)) {
  739. Show(STR(MemManInfo1),
  740. mm.dwLargestFreeBlock, mm.dwMaxPagesAvailable, mm.dwMaxPagesLockable);
  741. Show(STR(MemManInfo2),
  742. mm.dwTotalLinearSpace, mm.dwTotalUnlockedPages, mm.dwFreePages);
  743. Show(STR(MemManInfo3),
  744. mm.dwTotalPages, mm.dwFreeLinearSpace, mm.dwSwapFilePages);
  745. Show(STR(MemManInfo4), mm.wPageSize);
  746. }
  747. Show(STR(TasksExecuting), GetNumTasks());
  748. Show(STR(WinFlags));
  749. for (i=0; i<wfCnt; i++) if (lw & wf[i].mask)
  750. Show(" %s\n", (FP)wf[i].name);
  751. MyFlush();
  752. } /* DumpInfo */
  753. LPSTR GetProcName(FARPROC fn) {
  754. GLOBALENTRY ge;
  755. MODULEENTRY me;
  756. LPSTR szSym = STR(UnknownAddress);
  757. static char symBuf[80];
  758. ge.dwSize = sizeof(ge);
  759. me.dwSize = sizeof(me);
  760. if (GlobalEntryHandle(&ge, (HGLOBAL)FP_SEG(fn)) && (ge.wType == GT_CODE)) {
  761. if (ModuleFindHandle(&me, ge.hOwner)) {
  762. szSym = NearestSym(ge.wData, FP_OFF(fn), me.szExePath);
  763. if (!szSym) { /* if we know module name, but no syms */
  764. sprintf(symBuf, "%s %d:%04x", (FP)me.szModule, ge.wData, FP_OFF(fn));
  765. } else sprintf(symBuf, "%s %s", (FP)me.szModule, szSym);
  766. szSym = symBuf;
  767. }
  768. }
  769. return szSym;
  770. } /* GetProcName */
  771. STATIC void DumpStack(int disCnt, int parmCnt, int cnt, int first) {
  772. STACKTRACEENTRY ste;
  773. MODULEENTRY me;
  774. int frame = 0;
  775. unsigned oldsp = regs.sp+16;
  776. ste.dwSize = sizeof(ste);
  777. me.dwSize = sizeof(me);
  778. Show(STR(StackDumpStack));
  779. if (StackTraceCSIPFirst(&ste, regs.ss, regs.cs, regs.ip, regs.bp)) do {
  780. if (frame >= first--) {
  781. me.szModule[0] = 0;
  782. ModuleFindHandle(&me, ste.hModule);
  783. Show(STR(StackFrameInfo),
  784. frame++,
  785. (FP)GetProcName((FARPROC)MK_FP(ste.wCS, ste.wIP)),
  786. ste.wSS, ste.wBP);
  787. if (!noLocal && (parmCnt-- > 0)) {
  788. if (oldsp & 15) {
  789. int i;
  790. Show("ss:%04x ", oldsp & ~15);
  791. for (i=0; i < (int)(oldsp & 15); i++) Show(" ");
  792. }
  793. while (oldsp < ste.wBP) {
  794. if (!(oldsp & 15)) Show("\nss:%04x ", oldsp);
  795. Show("%02x ", *(byte far *)MK_FP(regs.ss, oldsp++));
  796. }
  797. Show("\n");
  798. }
  799. if (frame <= disStack && (disCnt-- >0)) {
  800. Show("\n");
  801. DisAsmAround(MK_FP(ste.wCS, ste.wIP), 8);
  802. }
  803. MyFlush();
  804. } /* if after first to show */
  805. } while (StackTraceNext(&ste) && (cnt-- > 0));
  806. } /* DumpStack */
  807. int BeginReport(LPSTR time) {
  808. int i;
  809. MyOpen();
  810. if (fh == -1) { /* maybe we're out of handles */
  811. _lclose(4); /* trash one at random */
  812. MyOpen(); /* and try again */
  813. }
  814. if (fh == -1) return 0;
  815. for (i=0; i<4; i++) Show("*******************");
  816. Show(STR(FailureReport), (FP)szAppNameVers, (FP)time);
  817. MyFlush();
  818. if (!noSound) {
  819. sound = OpenSound();
  820. pitch = 1000L << 16;
  821. } else sound = 0;
  822. return 1;
  823. } /* BeginReport */
  824. void EndReport(void) {
  825. if (fh != -1) {
  826. if (!whined && _llseek(fh, 0L, 2) > BIG_FILE) {
  827. PostMessage(hWnd, HEAP_BIG_FILE, 0, 0);
  828. whined = 1;
  829. }
  830. MyClose();
  831. }
  832. if (sound) {
  833. StopSound();
  834. CloseSound();
  835. sound = 0;
  836. }
  837. } /* EndReport */
  838. void ShowParamError(int sync) {
  839. if (GetCurrentTask() == lastErr.task)
  840. Show("$param$, %s %s\n",
  841. sync ? (FP)"" : (FP)STR(LastParamErrorWas),
  842. (FP)LogParamErrorStr(lastErr.code, lastErr.adr, lastErr.parm));
  843. lastErr.task = 0;
  844. } /* ShowParamError */
  845. /************************************
  846. Name: void Sherlock(void)
  847. Desc: Handles GP faults in applications by dumping as much system
  848. information as I can think of to a log file.
  849. This is the big routine.
  850. Bugs:
  851. *************************************/
  852. enum {s_prog, s_fault, s_name, s_instr, s_time, s_last};
  853. int Sherlock(void) {
  854. int i, faultlen, party;
  855. LPSTR s[s_last];
  856. if ((!trapZero || regs.intNum != 0) &&
  857. regs.intNum != 6 &&
  858. regs.intNum != 13)
  859. return 0;
  860. if (imTrying>0) {
  861. s[s_prog] = CurModuleName(GetCurrentTask());
  862. SafeDisAsm86(MK_FP(regs.cs, regs.ip), &faultlen);
  863. party = PrepareToParty(ModuleName(regs.cs), s[s_prog]);
  864. imTrying--;
  865. if (party) goto SkipReport;
  866. }
  867. if (++bugCnt > 20) return 0;
  868. if (!BeginReport(s[s_time] = CurTime()))
  869. return 0;
  870. s[s_prog] = CurModuleName(GetCurrentTask());
  871. s[s_fault] = FaultCause();
  872. s[s_name] = GetProcName((FARPROC)MK_FP(regs.cs, regs.ip));
  873. Show(STR(HadAFaultAt),
  874. (FP)s[s_prog],
  875. (FP)s[s_fault],
  876. (FP)s[s_name]);
  877. if (!noSummary) Show("$tag$%s$%s$%s$",
  878. (FP)s[s_prog],
  879. (FP)s[s_fault],
  880. (FP)s[s_name]);
  881. s[s_instr] = Tab2Spc(SafeDisAsm86(MK_FP(regs.cs, regs.ip), &faultlen));
  882. Show("%s$%s\n", (FP)s[s_instr], (FP)s[s_time]);
  883. ShowParamError(0);
  884. MyFlush();
  885. party = PrepareToParty(ModuleName(regs.cs), s[s_prog]);
  886. if ((bugCnt > 3) || ((party>0) && (iFeelLucky & 2))) {
  887. goto SkipReport;
  888. }
  889. if (!noReg) {
  890. Show(STR(CPURegistersRegs));
  891. Show("ax=%04x bx=%04x cx=%04x dx=%04x si=%04x di=%04x\n",
  892. regs.ax, regs.bx, regs.cx, regs.dx, regs.si, regs.di);
  893. Show("ip=%04x sp=%04x bp=%04x ", regs.ip, regs.sp+16, regs.bp);
  894. for (i=0; i<cntFlBit; i++)
  895. Show("%c%c ", flBit[i].name, regs.flags & (1 << flBit[i].bit) ? '+' : '-');
  896. Show("\n");
  897. Show("cs = %04x %s\n", regs.cs, (FP)SegInfo(regs.cs));
  898. Show("ss = %04x %s\n", regs.ss, (FP)SegInfo(regs.ss));
  899. Show("ds = %04x %s\n", regs.ds, (FP)SegInfo(regs.ds));
  900. Show("es = %04x %s\n", regs.es, (FP)SegInfo(regs.es));
  901. MyFlush();
  902. }
  903. if (cpu32 && !noReg32) {
  904. Show(STR(CPU32bitRegisters32bit));
  905. Show("eax = %08lx ebx = %08lx ecx = %08lx edx = %08lx\n",
  906. regs32.eax, regs32.ebx, regs32.ecx, regs32.edx);
  907. Show("esi = %08lx edi = %08lx ebp = %08lx esp = %08lx\n",
  908. regs32.esi, regs32.edi, regs32.ebp, regs32.esp);
  909. Show("fs = %04x %s\n", regs.fs, (FP)SegInfo(regs.fs));
  910. Show("gs = %04x %s\n", regs.gs, (FP)SegInfo(regs.gs));
  911. Show("eflag = %08lx\n", regs32.eflags);
  912. MyFlush();
  913. }
  914. if (!noDisasm) {
  915. Show(STR(InstructionDisasm));
  916. DisAsmAround(MK_FP(regs.cs, regs.ip), disLen);
  917. MyFlush();
  918. }
  919. if (!noInfo)
  920. DumpInfo();
  921. if (!noStack)
  922. DumpStack(disStack, 0x7fff, 0x7fff, 0);
  923. if (!noTasks) {
  924. TASKENTRY te;
  925. MODULEENTRY me;
  926. te.dwSize = sizeof(te);
  927. me.dwSize = sizeof(me);
  928. Show(STR(SystemTasksTasks));
  929. if (TaskFirst(&te)) do {
  930. ModuleFindName(&me, te.szModule);
  931. Show(STR(TaskHandleFlagsInfo),
  932. (FP)te.szModule, te.hTask, me.wcUsage,
  933. (FP)FileInfo(me.szExePath));
  934. Show(STR(Filename), (FP)me.szExePath); /* */
  935. } while (TaskNext(&te));
  936. MyFlush();
  937. }
  938. if (!noModules) {
  939. MODULEENTRY me;
  940. Show(STR(SystemModulesModules));
  941. me.dwSize = sizeof(me);
  942. if (ModuleFirst(&me)) do {
  943. Show(STR(ModuleHandleFlagsInfo),
  944. (FP)me.szModule, me.hModule, me.wcUsage,
  945. (FP)FileInfo(me.szExePath));
  946. Show(STR(File), (FP)me.szExePath); /* */
  947. } while (ModuleNext(&me));
  948. MyFlush();
  949. }
  950. SkipReport:
  951. if (party>0) {
  952. int len;
  953. word far * stack = MK_FP(regs.ss, regs.sp);
  954. Show(STR(ContinuingExecution), (FP)CurTime());
  955. MyFlush();
  956. /* fix up regs */
  957. if (gpRegs & segDS) regs.ds = 0;
  958. if (gpRegs & segES) regs.es = 0;
  959. if (gpRegs & segFS) regs.fs = 0;
  960. if (gpRegs & segGS) regs.gs = 0;
  961. regs.ip += faultlen; /* set at top of func - don't reuse */
  962. if ((int)gpStack < 0) {
  963. for (i=0; i<8; i++) stack[i+gpStack] = stack[i];
  964. } else if (gpStack) {
  965. for (i=7; i>=0; i--) stack[i+gpStack] = stack[i];
  966. }
  967. regs.sp += gpStack << 1;
  968. if (gpRegs & strCX) {
  969. len = regs.cx * memSize;
  970. regs.cx = 0;
  971. } else len = memSize;
  972. if (gpRegs & strSI) { /* doesn't handle 32 bit regs */
  973. regs.si += len;
  974. if (regs.si < (word)len) /* if overflow, set to big value */
  975. regs.si = 0xfff0; /* so global vars in heap don't get */
  976. } /* trashed when we continue */
  977. if (gpRegs & strDI) {
  978. regs.di += len;
  979. if (regs.di < (word)len) regs.di = 0xfff0;
  980. }
  981. }
  982. EndReport();
  983. if (!noClues && /* if we want clues */
  984. !pending && /* no clues waited for */
  985. (!party || !(iFeelLucky & 2))) { /* and we aren't quiet partiers */
  986. PostMessage(hWnd, JUST_THE_FACTS, (WPARAM)GetCurrentTask(), party);
  987. pending++;
  988. }
  989. if (party < 0) TerminateApp(GetCurrentTask(), NO_UAE_BOX);
  990. return party;
  991. } /* Sherlock */
  992. void far *bogus;
  993. int CallMeToo(WORD wID, DWORD dwData) {
  994. NFYLOGPARAMERROR far *lpep;
  995. LPSTR s[s_last];
  996. if (wID == NFY_OUTSTR) {
  997. if (noDebStr)
  998. return FALSE;
  999. MyOpen();
  1000. if (fh == -1) return FALSE;
  1001. Show(STR(DebugString), dwData);
  1002. MyClose();
  1003. return TRUE;
  1004. }
  1005. if (wID == NFY_LOGERROR && noErr)
  1006. return FALSE;
  1007. lpep = (void far *)dwData; /* Get the data for next log entry */
  1008. lastErr.adr = lpep->lpfnErrorAddr;
  1009. lastErr.code = lpep->wErrCode;
  1010. lastErr.parm = (DWORD)(lpep->lpBadParam);
  1011. lastErr.task = GetCurrentTask();
  1012. if ((lastErr.code & 0x3000) == 0x1000)
  1013. lastErr.parm = (WORD)lastErr.parm;
  1014. else if ((lastErr.code & 0x3000) == 0)
  1015. lastErr.parm = (BYTE)lastErr.parm;
  1016. if (wID == NFY_LOGPARAMERROR && noParam) {
  1017. return FALSE;
  1018. }
  1019. if (bugCnt++ > 60)
  1020. return FALSE;
  1021. if (!BeginReport(s[s_time] = CurTime())) /* Can't open file */
  1022. return FALSE;
  1023. switch (wID) {
  1024. case NFY_LOGERROR:
  1025. #if 0
  1026. lep = (void far *)dwData;
  1027. cs = ip = 0;
  1028. parm = 0;
  1029. code = lep->wErrCode;
  1030. s[s_fault] = STR(ApplicationError);
  1031. #endif
  1032. break;
  1033. case NFY_LOGPARAMERROR:
  1034. s[s_fault] = STR(InvalidParameter);
  1035. break;
  1036. default:
  1037. return FALSE;
  1038. }
  1039. s[s_prog] = CurModuleName(lastErr.task);
  1040. s[s_name] = GetProcName(lastErr.adr);
  1041. s[s_instr] = STR(NA); /* not interesting */
  1042. Show(STR(HadAFaultAt2),
  1043. (FP)s[s_prog],
  1044. (FP)s[s_fault], lastErr.code,
  1045. (FP)s[s_name]);
  1046. if (!noSummary) Show("$tag$%s$%s (%x)$%s$",
  1047. (FP)s[s_prog],
  1048. (FP)s[s_fault], lastErr.code,
  1049. (FP)s[s_name]);
  1050. Show(STR(ParamIs), lastErr.parm, (FP)s[s_time]);
  1051. ShowParamError(1);
  1052. MyFlush();
  1053. if (!noInfo && bugCnt < 2)
  1054. DumpInfo();
  1055. if (!noStack)
  1056. DumpStack(0, 0, 0x7fff, 4);
  1057. EndReport();
  1058. return TRUE;
  1059. } /* CallMe */
  1060. /* Parse SkipInfo= and ShowInfo= lines into flags array */
  1061. void ParseInfo(char *s, int val) {
  1062. int i;
  1063. strlwr(s);
  1064. while (*s) {
  1065. for (i=0; i<cntFlag; i++) if (0 == strncmp(s, syms+(i<<2), 3)) {
  1066. if (val) SetFlag(i);
  1067. else ClrFlag(i);
  1068. break;
  1069. }
  1070. while (*s && *s++ != ' ')
  1071. if (s[-1] == ',') break;
  1072. while (*s && *s == ' ') s++;
  1073. }
  1074. } /* ParseInfo */
  1075. /************************************
  1076. Name: BOOL LoadStringResources(void)
  1077. Desc: Load all string resources into GlobalAlloc'd buffer and
  1078. initialize aszStrings array with pointers to each string.
  1079. Also fixes up string IDs in wf (winflags) array to pointers.
  1080. Note that we don't free the memory allocated, we count on
  1081. kernel to clean up for us on termination.
  1082. Bugs:
  1083. *************************************/
  1084. BOOL LoadStringResources(void)
  1085. {
  1086. int n;
  1087. HANDLE h;
  1088. LPSTR lp;
  1089. WORD cbTotal;
  1090. WORD cbUsed;
  1091. WORD cbStrLen;
  1092. //
  1093. // Allocate too much memory for strings (maximum possible) at first,
  1094. // reallocate to the real size when we're done loading strings.
  1095. //
  1096. #if (STRING_COUNT * CCH_MAX_STRING_RESOURCE > 65536 - 64)
  1097. #error Need to use HUGE pointer for lp and DWORD for cb in LoadStringResources
  1098. #endif
  1099. cbTotal = STRING_COUNT;
  1100. cbTotal *= CCH_MAX_STRING_RESOURCE;
  1101. h = GlobalAlloc(GMEM_FIXED, cbTotal);
  1102. if ( ! h ) {
  1103. return FALSE;
  1104. }
  1105. lp = GlobalLock(h);
  1106. cbUsed = 0;
  1107. for ( n = 0; n < STRING_COUNT; n++ ) {
  1108. cbStrLen = LoadString(hInst, n, lp, CCH_MAX_STRING_RESOURCE);
  1109. if ( ! cbStrLen ) {
  1110. return FALSE;
  1111. }
  1112. aszStrings[n] = lp;
  1113. lp += cbStrLen + 1; // LoadString return doesn't count null terminator
  1114. cbUsed += cbStrLen + 1;
  1115. }
  1116. GlobalReAlloc(h, cbUsed, 0);
  1117. //
  1118. // Fix up winflags array elements from string resource IDs to pointers
  1119. //
  1120. for ( n = 0; n < wfCnt; n++ ) {
  1121. wf[n].name = aszStrings[ (int)(DWORD)wf[n].name ];
  1122. }
  1123. return TRUE;
  1124. }
  1125. /************************************
  1126. Name: void DumpIni(void)
  1127. Desc: Write profile strings to log file
  1128. Bugs:
  1129. *************************************/
  1130. #if 0
  1131. void DumpIni() {
  1132. int i;
  1133. char buf[4];
  1134. buf[3] = 0;
  1135. MyOpen();
  1136. Show("Re-read win.ini\nshowinfo="); // move to resource file if ever used
  1137. for (i=0; i<cntFlag; i++) {
  1138. if (!flag(i)) {
  1139. memcpy(buf, syms+(i<<2), 3);
  1140. Show("%s ", (FP)buf);
  1141. }
  1142. }
  1143. Show("\nskipinfo=");
  1144. for (i=0; i<cntFlag; i++) {
  1145. if (flag(i)) {
  1146. memcpy(buf, syms+(i<<2), 3);
  1147. Show("%s ", (FP)buf);
  1148. }
  1149. }
  1150. Show("\n");
  1151. MyClose();
  1152. } /* DumpIni */
  1153. #endif
  1154. /************************************
  1155. Name: int ReadWinIni(void)
  1156. Desc: Read profile strings from WIN.INI.
  1157. Return 0 if failure.
  1158. Bugs:
  1159. *************************************/
  1160. STATIC int ReadWinIni(void) {
  1161. char line[80];
  1162. int len;
  1163. /* how many instructions should I disassemble by default? */
  1164. disLen = GetProfileInt(szAppName, "dislen", 8);
  1165. /* should I trap divide by 0 faults? */
  1166. trapZero = GetProfileInt(szAppName, "trapzero", 0);
  1167. /* should we allow restarting apps? */
  1168. iFeelLucky = GetProfileInt(szAppName, "GPContinue", 1);
  1169. /* if (!(iFeelLucky & 16)) noSound = 1; */
  1170. /* how many stack frames should be disassembled? */
  1171. disStack = GetProfileInt(szAppName, "DisStack", 2);
  1172. /* where should I write the log file to? */
  1173. GetProfileString(szAppName, "logfile", szAppNameShortLog, logFile, sizeof(logFile));
  1174. len = strlen(logFile);
  1175. if ((len == 0) || // logfile=
  1176. (logFile[len-1] == '\\') || // directory only (boo, hiss)
  1177. (logFile[len-1] == '/') ||
  1178. (logFile[len-1] == ':')) { // drive only
  1179. if (len && (logFile[len-1] == ':')) { // drive only, put in root
  1180. strcat(logFile, "\\");
  1181. }
  1182. strcat(logFile, szAppNameShortLog); // append a file name
  1183. }
  1184. if (!(strchr(logFile, '\\') // if no path specified, put in WinDir
  1185. || strchr(logFile, ':')
  1186. || strchr(logFile, '/'))) {
  1187. char logname[80];
  1188. int n;
  1189. GetWindowsDirectory(logname, sizeof(logname));
  1190. n = strlen(logname);
  1191. if (n && logname[n-1] != '\\')
  1192. strcat(logname, "\\");
  1193. strcat(logname, logFile);
  1194. strcpy(logFile, logname);
  1195. }
  1196. /* Set default flag values - see DrWatson.h for default values */
  1197. ddFlag = DefFlag;
  1198. /* do I really have to print out all this information? */
  1199. if (GetProfileString(szAppName, "skipinfo", "", line, sizeof(line)))
  1200. ParseInfo(line, 1);
  1201. if (GetProfileString(szAppName, "showinfo", "", line, sizeof(line)))
  1202. ParseInfo(line, 0);
  1203. #if 0
  1204. DumpIni();
  1205. #endif
  1206. return 1;
  1207. } /* ReadWinIni */
  1208. /************************************
  1209. Name: int InitSherlock(void)
  1210. Desc: Initialize Sherlock processing. Install GP fault handler.
  1211. Return 0 if failure.
  1212. Bugs:
  1213. *************************************/
  1214. STATIC int InitSherlock(void) {
  1215. /* do I have 32 bit registers? */
  1216. cpu32 = (GetWinFlags() & (WF_CPU386|WF_CPU486)) != 0;
  1217. /* see what WIN.INI [drwatson] has to say */
  1218. if (!ReadWinIni()) return 0;
  1219. NotifyRegister(hTask, (LPFNNOTIFYCALLBACK)CallMe, NF_NORMAL);
  1220. /* Now get ToolHelp to do the dirty work */
  1221. return InterruptRegister(hTask, GPFault);
  1222. } /* InitSherlock */
  1223. /************************************
  1224. Name: void Moriarty
  1225. Desc: Destroy any evidence Sherlock was loaded.
  1226. Bugs: Am I freeing all resources I used?
  1227. *************************************/
  1228. int init;
  1229. STATIC void Moriarty(void) {
  1230. if (init) {
  1231. if (!noTime) PutDate(STR(Stop));
  1232. InterruptUnRegister(hTask);
  1233. NotifyUnRegister(hTask);
  1234. init = 0;
  1235. }
  1236. } /* Moriary */
  1237. /************************************
  1238. Name: WINAPI SherlockWndProc(hWnd, wMessage, wParam, lParam)
  1239. Desc: Handle sherlock icon, close processing
  1240. Bugs: Should pull up dialog boxes for About and GetInfo
  1241. *************************************/
  1242. LRESULT CALLBACK SherlockWndProc (HWND hWnd, UINT iMessage,
  1243. WPARAM wParam, LPARAM lParam) {
  1244. char msg[200];
  1245. /* int (FAR PASCAL *dfp)(HWND, WORD, WORD, DWORD); */
  1246. FARPROC dfp;
  1247. switch (iMessage) {
  1248. case WM_ENDSESSION:
  1249. if (wParam) Moriarty();
  1250. break;
  1251. case WM_DESTROY: /* Quit Sherlock */
  1252. PostQuitMessage (0);
  1253. break;
  1254. case WM_QUERYOPEN: /* never open a window??? */
  1255. PostMessage(hWnd, YOO_HOO, 0, 1);
  1256. ReadWinIni();
  1257. break;
  1258. case WM_WININICHANGE: /* Re-read WIN.INI parameters */
  1259. ReadWinIni();
  1260. break;
  1261. case YOO_HOO:
  1262. if (bugCnt) {
  1263. wsprintf(msg, STR(Faulty), bugCnt, (FP)logFile);
  1264. MessageBox(hWnd, msg, szAppNameVers,
  1265. MB_ICONINFORMATION | MB_OK | MB_TASKMODAL);
  1266. } else {
  1267. MessageBox(hWnd, STR(NoFault), szAppNameVers,
  1268. MB_ICONINFORMATION | MB_OK | MB_TASKMODAL);
  1269. }
  1270. break;
  1271. case HEAP_BIG_FILE:
  1272. wsprintf(msg, STR(LogFileGettingLarge),
  1273. (FP)logFile);
  1274. MessageBox(hWnd, msg, szAppNameVers,
  1275. MB_ICONEXCLAMATION | MB_OK | MB_TASKMODAL);
  1276. break;
  1277. case JUST_THE_FACTS:
  1278. dfp = MakeProcInstance((FARPROC)SherlockDialog, hInst);
  1279. DialogBox(hInst, "SherDiag", hWnd, (DLGPROC)dfp);
  1280. FreeProcInstance(dfp);
  1281. pending = 0; /* finished all old business */
  1282. break;
  1283. default:
  1284. return DefWindowProc (hWnd,iMessage,wParam,lParam);
  1285. }
  1286. return 0L;
  1287. }
  1288. /************************************
  1289. Name: WinMain(hInst, hPrevInst, cmdLine, cmdShow)
  1290. Desc: Init Sherlock - this is where it all begins
  1291. Bugs:
  1292. *************************************/
  1293. int PASCAL WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
  1294. LPSTR lpszCmdLine, int nCmdShow) {
  1295. MSG msg; /* Message returned from message loop */
  1296. WNDCLASS wndclass; /* Sherlock window class */
  1297. char watsonStack[4096];
  1298. nCmdShow = nCmdShow;
  1299. lpszCmdLine = lpszCmdLine;
  1300. newsp = watsonStack + sizeof(watsonStack);
  1301. hInst = hInstance;
  1302. hTask = GetCurrentTask();
  1303. /* Check if Sherlock is already running */
  1304. if (!hPrevInstance) {
  1305. if (!LoadStringResources()) {
  1306. MessageBox(NULL, "Dr. Watson could not load all string resources",
  1307. szAppNameVers, MB_ICONEXCLAMATION | MB_OK | MB_SYSTEMMODAL);
  1308. return 1;
  1309. }
  1310. /* Define a new window class */
  1311. wndclass.style = CS_HREDRAW | CS_VREDRAW | CS_DBLCLKS;
  1312. wndclass.lpfnWndProc = SherlockWndProc;
  1313. wndclass.cbClsExtra = 0;
  1314. wndclass.cbWndExtra = 0;
  1315. wndclass.hInstance = hInstance;
  1316. wndclass.hIcon = LoadIcon (hInstance, szAppNameShortMacro "Icon");
  1317. wndclass.hCursor = LoadCursor (NULL,IDC_ARROW);
  1318. wndclass.hbrBackground = GetStockObject (WHITE_BRUSH);
  1319. wndclass.lpszMenuName = NULL;
  1320. wndclass.lpszClassName = szAppName;
  1321. if (!RegisterClass (&wndclass)) {
  1322. MessageBox(NULL, STR(ClassMsg), szAppNameVers, MB_ICONEXCLAMATION | MB_OK |
  1323. MB_SYSTEMMODAL);
  1324. return 1;
  1325. }
  1326. } else {
  1327. /* Instance is already running, issue warning and terminate */
  1328. MessageBox (NULL, STR(ErrMsg), szAppNameVers, MB_ICONEXCLAMATION | MB_OK |
  1329. MB_SYSTEMMODAL);
  1330. return 1;
  1331. }
  1332. /* Create window and display in iconic form */
  1333. hWnd = CreateWindow (szAppName, szAppName, WS_OVERLAPPEDWINDOW, CW_USEDEFAULT,
  1334. 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
  1335. ShowWindow (hWnd, SW_SHOWMINNOACTIVE);
  1336. UpdateWindow (hWnd);
  1337. if (!InitSherlock()) {
  1338. MessageBox (/*NULL*/hWnd, STR(Vers), szAppNameVers, MB_ICONEXCLAMATION | MB_OK |
  1339. MB_SYSTEMMODAL);
  1340. DestroyWindow(hWnd);
  1341. return 1;
  1342. }
  1343. if (!noTime) PutDate(STR(Start));
  1344. init = 1;
  1345. while (GetMessage (&msg, NULL, 0, 0)) {/* Enter message loop */
  1346. TranslateMessage (&msg);
  1347. DispatchMessage (&msg);
  1348. imTrying = 0;
  1349. }
  1350. Moriarty(); /* Remove Sherlock GP Handler from GP Handler chain */
  1351. return msg.wParam;
  1352. } /* WinMain */