diff options
Diffstat (limited to 'linden/indra/newview/llwindebug.cpp')
-rw-r--r-- | linden/indra/newview/llwindebug.cpp | 356 |
1 files changed, 264 insertions, 92 deletions
diff --git a/linden/indra/newview/llwindebug.cpp b/linden/indra/newview/llwindebug.cpp index e48b7bf..129d925 100644 --- a/linden/indra/newview/llwindebug.cpp +++ b/linden/indra/newview/llwindebug.cpp | |||
@@ -4,7 +4,7 @@ | |||
4 | * | 4 | * |
5 | * $LicenseInfo:firstyear=2004&license=viewergpl$ | 5 | * $LicenseInfo:firstyear=2004&license=viewergpl$ |
6 | * | 6 | * |
7 | * Copyright (c) 2004-2008, Linden Research, Inc. | 7 | * Copyright (c) 2004-2009, Linden Research, Inc. |
8 | * | 8 | * |
9 | * Second Life Viewer Source Code | 9 | * Second Life Viewer Source Code |
10 | * The source code in this file ("Source Code") is provided by Linden Lab | 10 | * The source code in this file ("Source Code") is provided by Linden Lab |
@@ -42,6 +42,7 @@ | |||
42 | #pragma warning(disable: 4200) //nonstandard extension used : zero-sized array in struct/union | 42 | #pragma warning(disable: 4200) //nonstandard extension used : zero-sized array in struct/union |
43 | #pragma warning(disable: 4100) //unreferenced formal parameter | 43 | #pragma warning(disable: 4100) //unreferenced formal parameter |
44 | 44 | ||
45 | |||
45 | /* | 46 | /* |
46 | LLSD Block for Windows Dump Information | 47 | LLSD Block for Windows Dump Information |
47 | <llsd> | 48 | <llsd> |
@@ -120,8 +121,18 @@ MODULE32_NEST Module32Next_; | |||
120 | #define CALL_TRACE_MAX ((DUMP_SIZE_MAX - 2000) / (MAX_PATH + 40)) //max number of traced calls | 121 | #define CALL_TRACE_MAX ((DUMP_SIZE_MAX - 2000) / (MAX_PATH + 40)) //max number of traced calls |
121 | #define NL L"\r\n" //new line | 122 | #define NL L"\r\n" //new line |
122 | 123 | ||
123 | BOOL WINAPI Get_Module_By_Ret_Addr(PBYTE Ret_Addr, LPWSTR Module_Name, PBYTE & Module_Addr); | ||
124 | 124 | ||
125 | typedef struct STACK | ||
126 | { | ||
127 | STACK * Ebp; | ||
128 | PBYTE Ret_Addr; | ||
129 | DWORD Param[0]; | ||
130 | } STACK, * PSTACK; | ||
131 | |||
132 | BOOL WINAPI Get_Module_By_Ret_Addr(PBYTE Ret_Addr, LPWSTR Module_Name, PBYTE & Module_Addr); | ||
133 | void WINAPI Get_Call_Stack(const EXCEPTION_RECORD* exception_record, | ||
134 | const CONTEXT* context_record, | ||
135 | LLSD& info); | ||
125 | 136 | ||
126 | void printError( CHAR* msg ) | 137 | void printError( CHAR* msg ) |
127 | { | 138 | { |
@@ -184,69 +195,6 @@ BOOL GetProcessThreadIDs(DWORD process_id, std::vector<DWORD>& thread_ids) | |||
184 | return( TRUE ); | 195 | return( TRUE ); |
185 | } | 196 | } |
186 | 197 | ||
187 | void WINAPI GetCallStackData(const CONTEXT* context_struct, LLSD& info) | ||
188 | { | ||
189 | // Fill Str with call stack info. | ||
190 | // pException can be either GetExceptionInformation() or NULL. | ||
191 | // If pException = NULL - get current call stack. | ||
192 | |||
193 | LPWSTR Module_Name = new WCHAR[MAX_PATH]; | ||
194 | PBYTE Module_Addr = 0; | ||
195 | |||
196 | typedef struct STACK | ||
197 | { | ||
198 | STACK * Ebp; | ||
199 | PBYTE Ret_Addr; | ||
200 | DWORD Param[0]; | ||
201 | } STACK, * PSTACK; | ||
202 | |||
203 | PSTACK Ebp; | ||
204 | |||
205 | if(context_struct) | ||
206 | { | ||
207 | Ebp = (PSTACK)context_struct->Ebp; | ||
208 | } | ||
209 | else | ||
210 | { | ||
211 | // The context struct is NULL, | ||
212 | // so we will use the current stack. | ||
213 | Ebp = (PSTACK)&context_struct - 1; | ||
214 | |||
215 | // Skip frame of GetCallStackData(). | ||
216 | if (!IsBadReadPtr(Ebp, sizeof(PSTACK))) | ||
217 | Ebp = Ebp->Ebp; //caller ebp | ||
218 | } | ||
219 | |||
220 | // Trace CALL_TRACE_MAX calls maximum - not to exceed DUMP_SIZE_MAX. | ||
221 | // Break trace on wrong stack frame. | ||
222 | for (int Ret_Addr_I = 0, i = 0; | ||
223 | (Ret_Addr_I < CALL_TRACE_MAX) && !IsBadReadPtr(Ebp, sizeof(PSTACK)) && !IsBadCodePtr(FARPROC(Ebp->Ret_Addr)); | ||
224 | Ret_Addr_I++, Ebp = Ebp->Ebp, ++i) | ||
225 | { | ||
226 | // If module with Ebp->Ret_Addr found. | ||
227 | |||
228 | if (Get_Module_By_Ret_Addr(Ebp->Ret_Addr, Module_Name, Module_Addr)) | ||
229 | { | ||
230 | // Save module's address and full path. | ||
231 | info["CallStack"][i]["ModuleName"] = ll_convert_wide_to_string(Module_Name); | ||
232 | info["CallStack"][i]["ModuleAddress"] = (int)Module_Addr; | ||
233 | info["CallStack"][i]["CallOffset"] = (int)(Ebp->Ret_Addr - Module_Addr); | ||
234 | |||
235 | LLSD params; | ||
236 | // Save 5 params of the call. We don't know the real number of params. | ||
237 | if (!IsBadReadPtr(Ebp, sizeof(PSTACK) + 5 * sizeof(DWORD))) | ||
238 | { | ||
239 | for(int j = 0; j < 5; ++j) | ||
240 | { | ||
241 | params[j] = (int)Ebp->Param[j]; | ||
242 | } | ||
243 | } | ||
244 | info["CallStack"][i]["Parameters"] = params; | ||
245 | } | ||
246 | info["CallStack"][i]["ReturnAddress"] = (int)Ebp->Ret_Addr; | ||
247 | } | ||
248 | } | ||
249 | |||
250 | BOOL GetThreadCallStack(DWORD thread_id, LLSD& info) | 198 | BOOL GetThreadCallStack(DWORD thread_id, LLSD& info) |
251 | { | 199 | { |
252 | if(GetCurrentThreadId() == thread_id) | 200 | if(GetCurrentThreadId() == thread_id) |
@@ -271,7 +219,7 @@ BOOL GetThreadCallStack(DWORD thread_id, LLSD& info) | |||
271 | context_struct.ContextFlags = CONTEXT_FULL; | 219 | context_struct.ContextFlags = CONTEXT_FULL; |
272 | if(GetThreadContext(thread_handle, &context_struct)) | 220 | if(GetThreadContext(thread_handle, &context_struct)) |
273 | { | 221 | { |
274 | GetCallStackData(&context_struct, info); | 222 | Get_Call_Stack(NULL, &context_struct, info); |
275 | result = true; | 223 | result = true; |
276 | } | 224 | } |
277 | ResumeThread(thread_handle); | 225 | ResumeThread(thread_handle); |
@@ -326,35 +274,143 @@ BOOL WINAPI Get_Module_By_Ret_Addr(PBYTE Ret_Addr, LPWSTR Module_Name, PBYTE & M | |||
326 | return found; | 274 | return found; |
327 | } //Get_Module_By_Ret_Addr | 275 | } //Get_Module_By_Ret_Addr |
328 | 276 | ||
277 | bool has_valid_call_before(PDWORD cur_stack_loc) | ||
278 | { | ||
279 | PBYTE p_first_byte = (PBYTE)(*cur_stack_loc - 1); | ||
280 | PBYTE p_second_byte = (PBYTE)(*cur_stack_loc -2); | ||
281 | PBYTE p_fifth_byte = (PBYTE)(*cur_stack_loc - 5); | ||
282 | PBYTE p_sixth_byte = (PBYTE)(*cur_stack_loc - 6); | ||
283 | |||
284 | // make sure we can read it | ||
285 | if(IsBadReadPtr(p_sixth_byte, 6 * sizeof(BYTE))) | ||
286 | { | ||
287 | return false; | ||
288 | } | ||
289 | |||
290 | // check for 9a + 4 bytes | ||
291 | if(*p_fifth_byte == 0x9A) | ||
292 | { | ||
293 | return true; | ||
294 | } | ||
295 | |||
296 | // Check for E8 + 4 bytes and last byte is 00 or FF | ||
297 | if(*p_fifth_byte == 0xE8 && (*p_first_byte == 0x00 || *p_first_byte == 0xFF)) | ||
298 | { | ||
299 | return true; | ||
300 | } | ||
301 | |||
302 | // the other is six bytes | ||
303 | if(*p_sixth_byte == 0xFF || *p_second_byte == 0xFF) | ||
304 | { | ||
305 | return true; | ||
306 | } | ||
307 | |||
308 | return false; | ||
309 | } | ||
310 | |||
311 | PBYTE get_valid_frame(PBYTE esp) | ||
312 | { | ||
313 | PDWORD cur_stack_loc = NULL; | ||
314 | const int max_search = 400; | ||
315 | WCHAR module_name[MAX_PATH]; | ||
316 | PBYTE module_addr = 0; | ||
317 | |||
318 | // round to highest multiple of four | ||
319 | esp = (esp + (4 - ((int)esp % 4)) % 4); | ||
320 | |||
321 | // scroll through stack a few hundred places. | ||
322 | for (cur_stack_loc = (PDWORD) esp; cur_stack_loc < (PDWORD)esp + max_search; cur_stack_loc += 1) | ||
323 | { | ||
324 | // if you can read the pointer, | ||
325 | if (IsBadReadPtr(cur_stack_loc, sizeof(PDWORD))) | ||
326 | { | ||
327 | continue; | ||
328 | } | ||
329 | |||
330 | // check if it's in a module | ||
331 | if (!Get_Module_By_Ret_Addr((PBYTE)*cur_stack_loc, module_name, module_addr)) | ||
332 | { | ||
333 | continue; | ||
334 | } | ||
335 | |||
336 | // check if the code before the instruction ptr is a call | ||
337 | if(!has_valid_call_before(cur_stack_loc)) | ||
338 | { | ||
339 | continue; | ||
340 | } | ||
341 | |||
342 | // if these all pass, return that ebp, otherwise continue till we're dead | ||
343 | return (PBYTE)(cur_stack_loc - 1); | ||
344 | } | ||
345 | |||
346 | return NULL; | ||
347 | } | ||
348 | |||
349 | bool shouldUseStackWalker(PSTACK Ebp, int max_depth) | ||
350 | { | ||
351 | WCHAR Module_Name[MAX_PATH]; | ||
352 | PBYTE Module_Addr = 0; | ||
353 | int depth = 0; | ||
354 | |||
355 | while (depth < max_depth) | ||
356 | { | ||
357 | if (IsBadReadPtr(Ebp, sizeof(PSTACK)) || | ||
358 | IsBadReadPtr(Ebp->Ebp, sizeof(PSTACK)) || | ||
359 | Ebp->Ebp < Ebp || | ||
360 | Ebp->Ebp - Ebp > 0xFFFFFF || | ||
361 | IsBadCodePtr(FARPROC(Ebp->Ebp->Ret_Addr)) || | ||
362 | !Get_Module_By_Ret_Addr(Ebp->Ebp->Ret_Addr, Module_Name, Module_Addr)) | ||
363 | { | ||
364 | return true; | ||
365 | } | ||
366 | depth++; | ||
367 | Ebp = Ebp->Ebp; | ||
368 | } | ||
369 | |||
370 | return false; | ||
371 | } | ||
372 | |||
329 | //****************************************************************** | 373 | //****************************************************************** |
330 | void WINAPI Get_Call_Stack(PEXCEPTION_POINTERS pException, LLSD& info) | 374 | void WINAPI Get_Call_Stack(const EXCEPTION_RECORD* exception_record, |
375 | const CONTEXT* context_record, | ||
376 | LLSD& info) | ||
331 | //****************************************************************** | 377 | //****************************************************************** |
332 | // Fill Str with call stack info. | 378 | // Fill Str with call stack info. |
333 | // pException can be either GetExceptionInformation() or NULL. | 379 | // pException can be either GetExceptionInformation() or NULL. |
334 | // If pException = NULL - get current call stack. | 380 | // If pException = NULL - get current call stack. |
335 | { | 381 | { |
336 | LPWSTR Module_Name = new WCHAR[MAX_PATH]; | 382 | LPWSTR Module_Name = new WCHAR[MAX_PATH]; |
337 | PBYTE Module_Addr = 0; | 383 | PBYTE Module_Addr = 0; |
338 | 384 | LLSD params; | |
339 | typedef struct STACK | 385 | PBYTE Esp = NULL; |
340 | { | 386 | LLSD tmp_info; |
341 | STACK * Ebp; | 387 | |
342 | PBYTE Ret_Addr; | 388 | bool fake_frame = false; |
343 | DWORD Param[0]; | 389 | bool ebp_used = false; |
344 | } STACK, * PSTACK; | 390 | const int HEURISTIC_MAX_WALK = 20; |
391 | int heuristic_walk_i = 0; | ||
392 | int Ret_Addr_I = 0; | ||
345 | 393 | ||
346 | STACK Stack = {0, 0}; | 394 | STACK Stack = {0, 0}; |
347 | PSTACK Ebp; | 395 | PSTACK Ebp; |
348 | 396 | ||
349 | if (pException) //fake frame for exception address | 397 | if (exception_record && context_record) //fake frame for exception address |
350 | { | 398 | { |
351 | Stack.Ebp = (PSTACK)pException->ContextRecord->Ebp; | 399 | Stack.Ebp = (PSTACK)(context_record->Ebp); |
352 | Stack.Ret_Addr = (PBYTE)pException->ExceptionRecord->ExceptionAddress; | 400 | Stack.Ret_Addr = (PBYTE)exception_record->ExceptionAddress; |
353 | Ebp = &Stack; | 401 | Ebp = &Stack; |
402 | Esp = (PBYTE) context_record->Esp; | ||
403 | fake_frame = true; | ||
404 | } | ||
405 | else if(context_record) | ||
406 | { | ||
407 | Ebp = (PSTACK)(context_record->Ebp); | ||
408 | Esp = (PBYTE)(context_record->Esp); | ||
354 | } | 409 | } |
355 | else | 410 | else |
356 | { | 411 | { |
357 | Ebp = (PSTACK)&pException - 1; //frame addr of Get_Call_Stack() | 412 | Ebp = (PSTACK)&exception_record - 1; //frame addr of Get_Call_Stack() |
413 | Esp = (PBYTE)&exception_record; | ||
358 | 414 | ||
359 | // Skip frame of Get_Call_Stack(). | 415 | // Skip frame of Get_Call_Stack(). |
360 | if (!IsBadReadPtr(Ebp, sizeof(PSTACK))) | 416 | if (!IsBadReadPtr(Ebp, sizeof(PSTACK))) |
@@ -363,22 +419,21 @@ void WINAPI Get_Call_Stack(PEXCEPTION_POINTERS pException, LLSD& info) | |||
363 | 419 | ||
364 | // Trace CALL_TRACE_MAX calls maximum - not to exceed DUMP_SIZE_MAX. | 420 | // Trace CALL_TRACE_MAX calls maximum - not to exceed DUMP_SIZE_MAX. |
365 | // Break trace on wrong stack frame. | 421 | // Break trace on wrong stack frame. |
366 | for (int Ret_Addr_I = 0, i = 0; | 422 | for (Ret_Addr_I = 0; |
367 | (Ret_Addr_I < CALL_TRACE_MAX) && !IsBadReadPtr(Ebp, sizeof(PSTACK)) && !IsBadCodePtr(FARPROC(Ebp->Ret_Addr)); | 423 | heuristic_walk_i < HEURISTIC_MAX_WALK && |
368 | Ret_Addr_I++, Ebp = Ebp->Ebp, ++i) | 424 | Ret_Addr_I < CALL_TRACE_MAX && !IsBadReadPtr(Ebp, sizeof(PSTACK)) && !IsBadCodePtr(FARPROC(Ebp->Ret_Addr)); |
425 | Ret_Addr_I++) | ||
369 | { | 426 | { |
370 | // If module with Ebp->Ret_Addr found. | 427 | // If module with Ebp->Ret_Addr found. |
371 | |||
372 | if (Get_Module_By_Ret_Addr(Ebp->Ret_Addr, Module_Name, Module_Addr)) | 428 | if (Get_Module_By_Ret_Addr(Ebp->Ret_Addr, Module_Name, Module_Addr)) |
373 | { | 429 | { |
374 | // Save module's address and full path. | 430 | // Save module's address and full path. |
375 | info["CallStack"][i]["ModuleName"] = ll_convert_wide_to_string(Module_Name); | 431 | tmp_info["CallStack"][Ret_Addr_I]["ModuleName"] = ll_convert_wide_to_string(Module_Name); |
376 | info["CallStack"][i]["ModuleAddress"] = (int)Module_Addr; | 432 | tmp_info["CallStack"][Ret_Addr_I]["ModuleAddress"] = (int)Module_Addr; |
377 | info["CallStack"][i]["CallOffset"] = (int)(Ebp->Ret_Addr - Module_Addr); | 433 | tmp_info["CallStack"][Ret_Addr_I]["CallOffset"] = (int)(Ebp->Ret_Addr - Module_Addr); |
378 | 434 | ||
379 | LLSD params; | ||
380 | // Save 5 params of the call. We don't know the real number of params. | 435 | // Save 5 params of the call. We don't know the real number of params. |
381 | if (pException && !Ret_Addr_I) //fake frame for exception address | 436 | if (fake_frame && !Ret_Addr_I) //fake frame for exception address |
382 | params[0] = "Exception Offset"; | 437 | params[0] = "Exception Offset"; |
383 | else if (!IsBadReadPtr(Ebp, sizeof(PSTACK) + 5 * sizeof(DWORD))) | 438 | else if (!IsBadReadPtr(Ebp, sizeof(PSTACK) + 5 * sizeof(DWORD))) |
384 | { | 439 | { |
@@ -387,10 +442,64 @@ void WINAPI Get_Call_Stack(PEXCEPTION_POINTERS pException, LLSD& info) | |||
387 | params[j] = (int)Ebp->Param[j]; | 442 | params[j] = (int)Ebp->Param[j]; |
388 | } | 443 | } |
389 | } | 444 | } |
390 | info["CallStack"][i]["Parameters"] = params; | 445 | tmp_info["CallStack"][Ret_Addr_I]["Parameters"] = params; |
391 | } | 446 | } |
392 | info["CallStack"][i]["ReturnAddress"] = (int)Ebp->Ret_Addr; | 447 | |
448 | tmp_info["CallStack"][Ret_Addr_I]["ReturnAddress"] = (int)Ebp->Ret_Addr; | ||
449 | |||
450 | // get ready for next frame | ||
451 | // Set ESP to just after return address. Not the real esp, but just enough after the return address | ||
452 | if(!fake_frame) { | ||
453 | Esp = (PBYTE)Ebp + 8; | ||
454 | } | ||
455 | else | ||
456 | { | ||
457 | fake_frame = false; | ||
458 | } | ||
459 | |||
460 | // is next ebp valid? | ||
461 | // only run if we've never found a good ebp | ||
462 | // and make sure the one after is valid as well | ||
463 | if( !ebp_used && | ||
464 | shouldUseStackWalker(Ebp, 2)) | ||
465 | { | ||
466 | heuristic_walk_i++; | ||
467 | PBYTE new_ebp = get_valid_frame(Esp); | ||
468 | if (new_ebp != NULL) | ||
469 | { | ||
470 | Ebp = (PSTACK)new_ebp; | ||
471 | } | ||
472 | } | ||
473 | else | ||
474 | { | ||
475 | ebp_used = true; | ||
476 | Ebp = Ebp->Ebp; | ||
477 | } | ||
478 | } | ||
479 | /* TODO remove or turn this code back on to edit the stack after i see a few raw ones. -Palmer | ||
480 | // Now go back through and edit out heuristic stacks that could very well be bogus. | ||
481 | // Leave the top and the last 3 stack chosen by the heuristic, however. | ||
482 | if(heuristic_walk_i > 2) | ||
483 | { | ||
484 | info["CallStack"][0] = tmp_info["CallStack"][0]; | ||
485 | std::string ttest = info["CallStack"][0]["ModuleName"]; | ||
486 | for(int cur_frame = 1; | ||
487 | (cur_frame + heuristic_walk_i - 2 < Ret_Addr_I); | ||
488 | ++cur_frame) | ||
489 | { | ||
490 | // edit out the middle heuristic found frames | ||
491 | info["CallStack"][cur_frame] = tmp_info["CallStack"][cur_frame + heuristic_walk_i - 2]; | ||
492 | } | ||
493 | } | ||
494 | else | ||
495 | { | ||
496 | info = tmp_info; | ||
393 | } | 497 | } |
498 | */ | ||
499 | info = tmp_info; | ||
500 | info["HeuristicWalkI"] = heuristic_walk_i; | ||
501 | info["EbpUsed"] = ebp_used; | ||
502 | |||
394 | } //Get_Call_Stack | 503 | } //Get_Call_Stack |
395 | 504 | ||
396 | //*********************************** | 505 | //*********************************** |
@@ -429,7 +538,7 @@ LLSD WINAPI Get_Exception_Info(PEXCEPTION_POINTERS pException) | |||
429 | FILETIME Last_Write_Time; | 538 | FILETIME Last_Write_Time; |
430 | FILETIME Local_File_Time; | 539 | FILETIME Local_File_Time; |
431 | SYSTEMTIME T; | 540 | SYSTEMTIME T; |
432 | 541 | ||
433 | Str = new WCHAR[DUMP_SIZE_MAX]; | 542 | Str = new WCHAR[DUMP_SIZE_MAX]; |
434 | Str_Len = 0; | 543 | Str_Len = 0; |
435 | if (!Str) | 544 | if (!Str) |
@@ -439,6 +548,7 @@ LLSD WINAPI Get_Exception_Info(PEXCEPTION_POINTERS pException) | |||
439 | 548 | ||
440 | GetModuleFileName(NULL, Str, MAX_PATH); | 549 | GetModuleFileName(NULL, Str, MAX_PATH); |
441 | info["Process"] = ll_convert_wide_to_string(Str); | 550 | info["Process"] = ll_convert_wide_to_string(Str); |
551 | info["ThreadID"] = (S32)GetCurrentThreadId(); | ||
442 | 552 | ||
443 | // If exception occurred. | 553 | // If exception occurred. |
444 | if (pException) | 554 | if (pException) |
@@ -506,7 +616,7 @@ LLSD WINAPI Get_Exception_Info(PEXCEPTION_POINTERS pException) | |||
506 | } //if (pException) | 616 | } //if (pException) |
507 | 617 | ||
508 | // Save call stack info. | 618 | // Save call stack info. |
509 | Get_Call_Stack(pException, info); | 619 | Get_Call_Stack(pException->ExceptionRecord, pException->ContextRecord, info); |
510 | 620 | ||
511 | return info; | 621 | return info; |
512 | } //Get_Exception_Info | 622 | } //Get_Exception_Info |
@@ -552,6 +662,58 @@ void LLMemoryReserve::release() | |||
552 | 662 | ||
553 | static LLMemoryReserve gEmergencyMemoryReserve; | 663 | static LLMemoryReserve gEmergencyMemoryReserve; |
554 | 664 | ||
665 | #ifndef _M_IX86 | ||
666 | #error "The following code only works for x86!" | ||
667 | #endif | ||
668 | LPTOP_LEVEL_EXCEPTION_FILTER WINAPI MyDummySetUnhandledExceptionFilter( | ||
669 | LPTOP_LEVEL_EXCEPTION_FILTER lpTopLevelExceptionFilter) | ||
670 | { | ||
671 | if(lpTopLevelExceptionFilter == gFilterFunc) | ||
672 | return gFilterFunc; | ||
673 | |||
674 | llinfos << "Someone tried to set the exception filter. Listing call stack modules" << llendl; | ||
675 | LLSD cs_info; | ||
676 | Get_Call_Stack(NULL, NULL, cs_info); | ||
677 | |||
678 | if(cs_info.has("CallStack") && cs_info["CallStack"].isArray()) | ||
679 | { | ||
680 | LLSD cs = cs_info["CallStack"]; | ||
681 | for(LLSD::array_iterator i = cs.beginArray(); | ||
682 | i != cs.endArray(); | ||
683 | ++i) | ||
684 | { | ||
685 | llinfos << "Module: " << (*i)["ModuleName"] << llendl; | ||
686 | } | ||
687 | } | ||
688 | |||
689 | return gFilterFunc; | ||
690 | } | ||
691 | |||
692 | BOOL PreventSetUnhandledExceptionFilter() | ||
693 | { | ||
694 | HMODULE hKernel32 = LoadLibrary(_T("kernel32.dll")); | ||
695 | if (hKernel32 == NULL) | ||
696 | return FALSE; | ||
697 | |||
698 | void *pOrgEntry = GetProcAddress(hKernel32, "SetUnhandledExceptionFilter"); | ||
699 | if(pOrgEntry == NULL) | ||
700 | return FALSE; | ||
701 | |||
702 | unsigned char newJump[ 100 ]; | ||
703 | DWORD dwOrgEntryAddr = (DWORD)pOrgEntry; | ||
704 | dwOrgEntryAddr += 5; // add 5 for 5 op-codes for jmp far | ||
705 | void *pNewFunc = &MyDummySetUnhandledExceptionFilter; | ||
706 | DWORD dwNewEntryAddr = (DWORD) pNewFunc; | ||
707 | DWORD dwRelativeAddr = dwNewEntryAddr - dwOrgEntryAddr; | ||
708 | |||
709 | newJump[ 0 ] = 0xE9; // JMP absolute | ||
710 | memcpy(&newJump[ 1 ], &dwRelativeAddr, sizeof(pNewFunc)); | ||
711 | SIZE_T bytesWritten; | ||
712 | BOOL bRet = WriteProcessMemory(GetCurrentProcess(), | ||
713 | pOrgEntry, newJump, sizeof(pNewFunc) + 1, &bytesWritten); | ||
714 | return bRet; | ||
715 | } | ||
716 | |||
555 | // static | 717 | // static |
556 | void LLWinDebug::initExceptionHandler(LPTOP_LEVEL_EXCEPTION_FILTER filter_func) | 718 | void LLWinDebug::initExceptionHandler(LPTOP_LEVEL_EXCEPTION_FILTER filter_func) |
557 | { | 719 | { |
@@ -602,6 +764,9 @@ void LLWinDebug::initExceptionHandler(LPTOP_LEVEL_EXCEPTION_FILTER filter_func) | |||
602 | LPTOP_LEVEL_EXCEPTION_FILTER prev_filter; | 764 | LPTOP_LEVEL_EXCEPTION_FILTER prev_filter; |
603 | prev_filter = SetUnhandledExceptionFilter(filter_func); | 765 | prev_filter = SetUnhandledExceptionFilter(filter_func); |
604 | 766 | ||
767 | // *REMOVE:Mani | ||
768 | //PreventSetUnhandledExceptionFilter(); | ||
769 | |||
605 | if(prev_filter != gFilterFunc) | 770 | if(prev_filter != gFilterFunc) |
606 | { | 771 | { |
607 | LL_WARNS("AppInit") | 772 | LL_WARNS("AppInit") |
@@ -737,3 +902,10 @@ void LLWinDebug::generateCrashStacks(struct _EXCEPTION_POINTERS *exception_infop | |||
737 | LLSDSerialize::toPrettyXML(info, out_file); | 902 | LLSDSerialize::toPrettyXML(info, out_file); |
738 | out_file.close(); | 903 | out_file.close(); |
739 | } | 904 | } |
905 | |||
906 | void LLWinDebug::clearCrashStacks() | ||
907 | { | ||
908 | LLSD info; | ||
909 | std::string dump_path = gDirUtilp->getExpandedFilename(LL_PATH_LOGS, "SecondLifeException.log"); | ||
910 | LLFile::remove(dump_path); | ||
911 | } | ||