xref: /aoo42x/main/crashrep/source/win32/soreport.cpp (revision 910823ae)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 #define UNICODE
25 #define WIN32_LEAN_AND_MEAN
26 #if defined _MSC_VER
27 #pragma warning(push, 1)
28 #pragma warning(disable:4917)
29 #endif
30 #include <windows.h>
31 #include <windowsx.h>
32 
33 #include <mapi.h>
34 #include <commctrl.h>
35 #include <commdlg.h>
36 #include <psapi.h>
37 
38 #include <shellapi.h>
39 #include <shlobj.h>
40 
41 #define _UNICODE
42 #include <tchar.h>
43 
44 #define _RICHEDIT_VER	0x0200
45 #include <richedit.h>
46 
47 #if defined _MSC_VER
48 #pragma warning(pop)
49 #endif
50 
51 #if _RICHEDIT_VER >= 0x0200
52 #define RICHEDIT	TEXT("riched20.dll")
53 #else
54 #define RICHEDIT	TEXT("riched32.dll")
55 #endif
56 
57 #include <systools/win32/uwinapi.h>
58 #include <rtl/digest.h>
59 #include <rtl/bootstrap.hxx>
60 #include <osl/file.hxx>
61 #include <osl/process.h>
62 
63 #include <stdlib.h>
64 #include <stdio.h>
65 #include <io.h>
66 #include <fcntl.h>
67 #include <string>
68 #include <hash_map>
69 #include <winsock.h>
70 #include <malloc.h>
71 #include <process.h>
72 
73 #include <_version.h>
74 
75 #include "resource.h"
76 #include "base64.h"
77 
78 #define FORMATBUFSIZE	(8*1024)
79 #define MAX_TEXT_BUFFER (32*1024-1)
80 #define MAX_HOSTNAME	(1024)
81 
82 #ifdef __MINGW32__
83 #include <imagehlp.h>
84 #else
85 #include <dbghelp.h>
86 #endif
87 
88 #ifdef _UNICODE
89 #define tstring	wstring
90 #else
91 #define tstring string
92 #endif
93 
94 using namespace ::std;
95 
96 
97 wstring  g_wstrProductKey;
98 string  g_strDefaultLanguage;
99 FILE	*g_fpStackFile = NULL;
100 FILE	*g_fpChecksumFile = NULL;
101 DWORD	g_dwExceptionCode = 0;
102 
103 CHAR	g_szReportServerA[MAX_HOSTNAME] = "";
104 USHORT	g_uReportPort = 80;
105 
106 TCHAR	g_szBuildId[256] = TEXT("");
107 
108 TCHAR	g_szDumpFileName[MAX_PATH] = TEXT("");
109 
110 CHAR	g_szDumpFileNameA[MAX_PATH] = "";
111 CHAR	g_szCommentFileNameA[MAX_PATH] = "";
112 CHAR	g_szReportFileNameA[MAX_PATH] = "";
113 
114 
115 bool	g_bNoUserInterface = false;
116 bool	g_bSendReport = false;
117 bool	g_bLoadReport = false;
118 
119 #define REPORT_SERVER	g_szReportServerA
120 #define REPORT_PORT		g_uReportPort
121 
122 
123 //***************************************************************************
124 // tmpfile from msvcrt creates the temporary file in the root of the current
125 // volume and can fail.
126 
127 static FILE *_xfopen( const _TCHAR *file, const _TCHAR *mode )
128 {
129 #ifdef UNICODE
130 	if ( (LONG)GetVersion() < 0 )
131 	{
132 		char	afile[MAX_PATH];
133 		char	amode[16];
134 
135 		WideCharToMultiByte( CP_ACP, 0, file, -1, afile, MAX_PATH, NULL, NULL );
136 		WideCharToMultiByte( CP_ACP, 0, mode, -1, amode, 16, NULL, NULL );
137 
138 
139 		return fopen( afile, amode );
140 	}
141 	else
142 #endif
143 		return _tfopen( file, mode );
144 }
145 
146 
147 static FILE *_tmpfile(void)
148 {
149 	FILE *fp = NULL;
150 
151 	TCHAR	szTempPath[MAX_PATH];
152 
153 	if ( GetTempPath( elementsof(szTempPath), szTempPath ) )
154 	{
155 		TCHAR	szFileName[MAX_PATH];
156 
157 		if ( GetTempFileName( szTempPath, TEXT("CRT"), 0, szFileName ) )
158 		{
159 			HANDLE	hFile =  CreateFile(
160 				szFileName,
161 				GENERIC_READ | GENERIC_WRITE,
162 				0, NULL,
163 				OPEN_EXISTING,
164 				FILE_FLAG_DELETE_ON_CLOSE | FILE_ATTRIBUTE_NORMAL,
165 				NULL );
166 
167 			if ( IsValidHandle( hFile ) )
168 			{
169 				int fd = _open_osfhandle( (int)hFile, 0 );
170 
171 				fp = _fdopen( fd, "w+b" );
172 			}
173 		}
174 	}
175 
176 	return fp;
177 }
178 //***************************************************************************
179 
180 static BOOL GetCrashDataPath( LPTSTR szBuffer )
181 {
182 	::rtl::OUString	ustrValue = ::rtl::OUString::createFromAscii("${$OOO_BASE_DIR/program/bootstrap.ini:UserInstallation}");
183 	::rtl::Bootstrap::expandMacros( ustrValue );
184 
185 	if ( ustrValue.getLength() )
186 	{
187 		ustrValue += ::rtl::OUString::createFromAscii("/user/crashdata");
188 
189 		::osl::FileBase::RC	result = ::osl::Directory::createPath( ustrValue );
190 
191 		if ( ::osl::FileBase::E_None == result || ::osl::FileBase::E_EXIST == result )
192 		{
193 			::rtl::OUString	ustrPath;
194 
195 			result = ::osl::FileBase::getSystemPathFromFileURL( ustrValue, ustrPath );
196 			if (  ::osl::FileBase::E_None == result  )
197 			{
198 				_tcsncpy( szBuffer, reinterpret_cast<LPCWSTR>(ustrPath.getStr()), MAX_PATH );
199 				return TRUE;
200 			}
201 		}
202 	}
203 
204 	return FALSE;
205 }
206 
207 
208 static FILE *_open_reportfile( LPCTSTR lpExt, LPCTSTR lpMode )
209 {
210 	FILE	*fp = NULL;
211 	TCHAR	szAppDataPath[MAX_PATH] = _T("");
212 
213 	if ( GetCrashDataPath( szAppDataPath ) )
214 	{
215 		_tcscat( szAppDataPath, _T("\\crashdat") );
216 		_tcscat( szAppDataPath, lpExt );
217 
218 		fp = _xfopen( szAppDataPath, lpMode );
219 	}
220 
221 	return fp;
222 }
223 
224 //***************************************************************************
225 
226 struct CrashReportParams
227 {
228 	BOOL				fAllowContact;
229 	tstring				sEmail;
230 	tstring				sTitle;
231 	tstring				sComment;
232 	ULONG				uInternetConnection;
233 	tstring				sProxyServer;
234 	tstring				sProxyPort;
235 
236 	CrashReportParams();
237 
238 	void WriteToRegistry();
239 	void ReadFromRegistry();
240 	void ReadFromEnvironment();
241 };
242 
243 bool SendCrashReport( HWND hwndParent, const CrashReportParams &rParams );
244 BOOL WriteCommentFile( LPCTSTR lpComment );
245 
246 //***************************************************************************
247 
248 LONG RegReadValue( HKEY hBaseKey, LPCTSTR lpSubKey, LPCTSTR lpValueName, LPVOID lpData, DWORD cbData )
249 {
250 	HKEY	hKey = NULL;
251 	LONG	lResult;
252 
253 	lResult = RegOpenKeyEx( hBaseKey, lpSubKey, 0, KEY_QUERY_VALUE, &hKey );
254 
255 	if ( ERROR_SUCCESS == lResult )
256 	{
257 		lResult = RegQueryValueEx( hKey, lpValueName, NULL, NULL, (LPBYTE)lpData, &cbData );
258 		RegCloseKey( hKey );
259 	}
260 
261 	return lResult;
262 }
263 
264 //***************************************************************************
265 
266 LONG RegWriteValue( HKEY hBaseKey, LPCTSTR lpSubKey, LPCTSTR lpValueName, DWORD dwType, LPCVOID lpData, DWORD cbData )
267 {
268 	HKEY	hKey = NULL;
269 	LONG	lResult;
270 
271 	lResult = RegCreateKeyEx( hBaseKey, lpSubKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL );
272 
273 	if ( ERROR_SUCCESS == lResult )
274 	{
275 		lResult = RegSetValueEx( hKey, lpValueName, NULL, dwType, (CONST BYTE *)lpData, cbData );
276 		RegCloseKey( hKey );
277 	}
278 
279 	return lResult;
280 }
281 
282 //***************************************************************************
283 
284 CrashReportParams::CrashReportParams()
285 {
286 	fAllowContact = FALSE;
287 	sTitle = TEXT("");
288 	sComment = TEXT("");
289 	sEmail = TEXT("");
290 	uInternetConnection = 0;
291 	sProxyServer = TEXT("");
292 	sProxyPort = TEXT("");
293 }
294 
295 //***************************************************************************
296 
297 void CrashReportParams::ReadFromRegistry()
298 {
299 	TCHAR	szBuffer[2048];
300 
301 	if ( ERROR_SUCCESS == RegReadValue(
302 		HKEY_CURRENT_USER,
303 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
304 		TEXT("HTTPProxyServer"),
305 		szBuffer,
306 		sizeof(szBuffer) ) )
307 		sProxyServer = szBuffer;
308 
309 	DWORD	dwProxyPort;
310 
311 	if ( ERROR_SUCCESS == RegReadValue(
312 		HKEY_CURRENT_USER,
313 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
314 		TEXT("HTTPProxyPort"),
315 		&dwProxyPort,
316 		sizeof(dwProxyPort) ) )
317 	{
318 		_stprintf( szBuffer, TEXT("%d"), dwProxyPort );
319 		sProxyPort = szBuffer;
320 	}
321 
322 	if ( ERROR_SUCCESS == RegReadValue(
323 		HKEY_CURRENT_USER,
324 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
325 		TEXT("ReturnAddress"),
326 		szBuffer,
327 		sizeof(szBuffer) ) )
328 		sEmail = szBuffer;
329 
330 	RegReadValue(
331 		HKEY_CURRENT_USER,
332 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
333 		TEXT("AllowContact"),
334 		&fAllowContact,
335 		sizeof(fAllowContact) );
336 
337 	RegReadValue(
338 		HKEY_CURRENT_USER,
339 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
340 		TEXT("HTTPConnection"),
341 		&uInternetConnection,
342 		sizeof(uInternetConnection) );
343 }
344 
345 //***************************************************************************
346 
347 void CrashReportParams::WriteToRegistry()
348 {
349 	RegWriteValue(
350 		HKEY_CURRENT_USER,
351 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
352 		TEXT("HTTPProxyServer"), REG_SZ,
353 		sProxyServer.c_str(),
354 		sizeof(TCHAR) * (sProxyServer.length() + 1) );
355 
356 	LPTSTR endptr = NULL;
357 	DWORD dwProxyPort = _tcstoul( sProxyPort.c_str(), &endptr, 10 );
358 
359 	RegWriteValue(
360 		HKEY_CURRENT_USER,
361 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
362 		TEXT("HTTPProxyPort"), REG_DWORD,
363 		&dwProxyPort,
364 		sizeof(DWORD) );
365 
366 	RegWriteValue(
367 		HKEY_CURRENT_USER,
368 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
369 		TEXT("AllowContact"), REG_DWORD,
370 		&fAllowContact,
371 		sizeof(DWORD) );
372 
373 
374 	RegWriteValue(
375 		HKEY_CURRENT_USER,
376 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
377 		TEXT("HTTPConnection"), REG_DWORD,
378 		&uInternetConnection,
379 		sizeof(DWORD) );
380 
381 	RegWriteValue(
382 		HKEY_CURRENT_USER,
383 		TEXT("SOFTWARE\\OpenOffice.org\\CrashReport"),
384 		TEXT("ReturnAddress"), REG_SZ,
385 		sEmail.c_str(),
386 		sizeof(TCHAR) * (sEmail.length() + 1) );
387 }
388 
389 //***************************************************************************
390 
391 void CrashReportParams::ReadFromEnvironment()
392 {
393 	TCHAR	szBuffer[2048];
394 
395 	DWORD dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_HTTPPROXYSERVER"), szBuffer, elementsof(szBuffer) );
396 
397 	if ( dwResult && dwResult < elementsof(szBuffer) )
398 		sProxyServer = szBuffer;
399 
400 	dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_HTTPPROXYPORT"), szBuffer, elementsof(szBuffer) );
401 
402 	if ( dwResult && dwResult < elementsof(szBuffer) )
403 		sProxyPort = szBuffer;
404 
405 	dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_RETURNADDRESS"), szBuffer, elementsof(szBuffer) );
406 
407 	if ( dwResult && dwResult < elementsof(szBuffer) )
408 	{
409 		sEmail = szBuffer;
410 		// fAllowContact = TRUE;
411 	}
412 
413 	dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_HTTPCONNECTIONTYPE"), szBuffer, elementsof(szBuffer) );
414 
415 	if ( dwResult && dwResult < elementsof(szBuffer) )
416 	{
417 		if ( 0 == _tcsicmp( szBuffer, _T("DIRECT") ) )
418 			uInternetConnection = 1;
419 		else if ( 0 == _tcsicmp( szBuffer, _T("MANUALPROXY") ) )
420 			uInternetConnection = 2;
421 		else if ( 0 == _tcsicmp( szBuffer, _T("SYSTEMDEFAULT") ) )
422 			uInternetConnection = 0;
423 	}
424 
425 	dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_SUBJECT"), szBuffer, elementsof(szBuffer) );
426 
427 	if ( dwResult && dwResult < elementsof(szBuffer) )
428 		sTitle = szBuffer;
429 
430 
431 	dwResult = GetEnvironmentVariable( TEXT("ERRORREPORT_BODYFILE"), szBuffer, elementsof(szBuffer) );
432 
433 	if ( dwResult && dwResult < elementsof(szBuffer) )
434 	{
435 		FILE *fp = _xfopen( szBuffer, _T("rb") );
436 
437 		if ( fp )
438 		{
439 			CHAR	aUTF8Buffer[256];
440 			size_t	nBytesRead;
441 
442 			sComment = TEXT("");
443 
444 			while ( 0 != (nBytesRead = fread( aUTF8Buffer, sizeof(aUTF8Buffer[0]), elementsof(aUTF8Buffer), fp )) )
445 			{
446 				TCHAR	aBuffer[256+1];
447 
448 				DWORD	dwCharacters = MultiByteToWideChar( CP_UTF8, 0, aUTF8Buffer, nBytesRead, aBuffer, elementsof(aBuffer) - 1 );
449 				aBuffer[dwCharacters] = 0;
450 				sComment += aBuffer;
451 			}
452 
453 			fclose( fp );
454 		}
455 	}
456 }
457 
458 //***************************************************************************
459 
460 typedef BOOL (WINAPI *MiniDumpWriteDump_PROC)(
461     IN HANDLE hProcess,
462     IN DWORD ProcessId,
463     IN HANDLE hFile,
464     IN MINIDUMP_TYPE DumpType,
465     IN CONST PMINIDUMP_EXCEPTION_INFORMATION ExceptionParam, OPTIONAL
466     IN CONST PMINIDUMP_USER_STREAM_INFORMATION UserStreamParam, OPTIONAL
467     IN CONST PMINIDUMP_CALLBACK_INFORMATION CallbackParam OPTIONAL
468     );
469 
470 //***************************************************************************
471 
472 static BOOL WINAPI InitRichEdit()
473 {
474 	return (NULL != LoadLibrary( RICHEDIT ));
475 }
476 
477 //***************************************************************************
478 
479 static BOOL WINAPI DeinitRichEdit()
480 {
481 	return FreeLibrary( GetModuleHandle( RICHEDIT ) );
482 }
483 
484 //***************************************************************************
485 
486 static string trim_string( const string& rString )
487 {
488 	string temp = rString;
489 
490 	while ( temp.length() && temp[0] == ' ' || temp[0] == '\t' )
491 		temp.erase( 0, 1 );
492 
493 	string::size_type	len = temp.length();
494 
495 	while ( len && temp[len-1] == ' ' || temp[len-1] == '\t' )
496 	{
497 		temp.erase( len - 1, 1 );
498 		len = temp.length();
499 	}
500 
501 	return temp;
502 }
503 
504 //***************************************************************************
505 
506 static int LoadAndFormatString( HINSTANCE hInstance, UINT uID, LPTSTR lpBuffer, int nBufferMax )
507 {
508 	TCHAR	szBuffer[FORMATBUFSIZE];
509 	TCHAR	szBuffer2[FORMATBUFSIZE];
510 
511 	LoadString( hInstance, uID, szBuffer, elementsof(szBuffer) );
512 
513 	LPCTSTR	src;
514 	LPTSTR	dest;
515 	for ( dest = szBuffer2, src = szBuffer; *src; src++, dest++ )
516 	{
517 		switch ( *src )
518 		{
519 		case '~':
520 			*dest = '&';
521 			break;
522 		case '\\':
523 			switch ( *(++src) )
524 			{
525 			case 'n':
526 				*dest = '\n';
527 				break;
528 			case 'r':
529 				*dest = '\r';
530 				break;
531 			default:
532 				*dest = *src;
533 				break;
534 			}
535 			break;
536 		default:
537 			*dest = *src;
538 			break;
539 		}
540 	}
541 
542 	*dest = *src;
543 
544 	return ExpandEnvironmentStrings( szBuffer2, lpBuffer, nBufferMax );
545 }
546 
547 
548 //***************************************************************************
549 
550 static string wstring2utf8( const wstring &rString )
551 {
552 	int nBufSize = WideCharToMultiByte( CP_UTF8, 0, rString.c_str(), -1, NULL, 0, NULL, FALSE );
553 
554 	LPSTR	pBuffer = (LPSTR)alloca( nBufSize );
555 
556 	WideCharToMultiByte(  CP_UTF8, 0, rString.c_str(), -1, pBuffer, nBufSize, NULL, FALSE );
557 
558 	return string( pBuffer );
559 }
560 
561 //***************************************************************************
562 
563 static string xml_encode( const string &rString )
564 {
565 	string temp = rString;
566     string::size_type pos = 0;
567 
568 	// First replace all occurences of '&' because it may occur in further
569 	// encoded chardters too
570 
571     for( pos = 0; (pos = temp.find( '&', pos )) != string::npos; pos += 4 )
572         temp.replace( pos, 1, "&amp;" );
573 
574 	for( pos = 0; (pos = temp.find( '<', pos )) != string::npos; pos += 4 )
575         temp.replace( pos, 1, "&lt;" );
576 
577     for( pos = 0; (pos = temp.find( '>', pos )) != string::npos; pos += 4 )
578         temp.replace( pos, 1, "&gt;" );
579 
580 	return temp;
581 }
582 
583 //***************************************************************************
584 
585 static size_t fcopy( FILE *fpin, FILE *fpout )
586 {
587 	char buffer[1024];
588 	size_t nBytes;
589 	size_t nBytesWritten = 0;
590 
591 	if ( fpin && fpout )
592 	{
593 		while ( 0 != (nBytes = fread( buffer, 1, sizeof(buffer), fpin )) )
594 		{
595 			nBytesWritten += fwrite( buffer, 1, nBytes, fpout );
596 		}
597 	}
598 
599 	return nBytesWritten;
600 }
601 
602 //***************************************************************************
603 
604 static string GetModuleDirectory( HMODULE hModule )
605 {
606 	TCHAR	szModuleName[MAX_PATH] = TEXT("");
607 	TCHAR	szDrive[_MAX_DRIVE];
608 	TCHAR	szDir[_MAX_DIR];
609 	TCHAR	szFName[_MAX_FNAME];
610 	TCHAR	szExt[_MAX_EXT];
611 
612 	if ( GetModuleFileName( hModule, szModuleName, MAX_PATH ) )
613 	{
614 		_tsplitpath( szModuleName, szDrive, szDir, szFName, szExt );
615 		_tmakepath( szModuleName, szDrive, szDir, _T(""), _T("") );
616 	}
617 
618 	CHAR	szModuleNameUTF8[MAX_PATH] = "";
619 
620 	WideCharToMultiByte( CP_UTF8, 0, szModuleName, -1, szModuleNameUTF8, elementsof(szModuleNameUTF8), NULL, NULL );
621 	return string( szModuleNameUTF8 );
622 }
623 
624 //***************************************************************************
625 
626 string GetFileDirectory( const string& rFilePath )
627 {
628 	string aDir = rFilePath;
629     size_t pos = aDir.rfind( '\\' );
630 
631 	if ( string::npos != pos )
632 		aDir.erase( pos + 1 );
633 	else
634 		aDir = "";
635 
636 	return aDir;
637 }
638 
639 //***************************************************************************
640 
641 string GetFileName( const string& rFilePath )
642 {
643 	string aName = rFilePath;
644     size_t pos = aName.rfind( '\\' );
645 
646 	if ( string::npos != pos )
647 		return aName.substr( pos + 1 );
648 	else
649 		return aName;
650 }
651 
652 //***************************************************************************
653 
654 BOOL WriteReportFile( CrashReportParams *pParams )
655 {
656 	BOOL	fSuccess = FALSE;
657 	TCHAR	szTempPath[MAX_PATH];
658 
659 	if ( GetTempPath( elementsof(szTempPath), szTempPath ) )
660 	{
661 		TCHAR	szFileName[MAX_PATH];
662 
663 		if ( GetTempFileName( szTempPath, TEXT("RPM"), 0, szFileName ) )
664 		{
665 			HANDLE	hFile =  CreateFile( szFileName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
666 
667 			if ( hFile )
668 			{
669 				int	fd = _open_osfhandle( (LONG)hFile, _O_TEXT );
670 				FILE	*fp = _fdopen( fd, "w+t" );
671 				CHAR	szTitle[1024] = "";
672 				CHAR	szBuildId[1024] = "";
673 				CHAR	szEmail[1024] = "";
674 				const char *pszUserType = getenv( "STAROFFICE_USERTYPE" );
675 
676 				WideCharToMultiByte( CP_UTF8, 0, pParams->sTitle.c_str(), -1, szTitle, sizeof(szTitle), NULL, NULL );
677 				WideCharToMultiByte( CP_UTF8, 0, g_szBuildId, -1, szBuildId, sizeof(szBuildId), NULL, NULL );
678 				WideCharToMultiByte( CP_UTF8, 0, pParams->sEmail.c_str(), -1, szEmail, sizeof(szEmail), NULL, NULL );
679 
680 				fprintf( fp,
681 					"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
682 					"<!DOCTYPE errormail:errormail PUBLIC \"-//OpenOffice.org//DTD ErrorMail 1.0//EN\" \"errormail.dtd\">\n"
683 					"<errormail:errormail xmlns:errormail=\"http://openoffice.org/2002/errormail\" usertype=\"%s\">\n"
684 					"<reportmail:mail xmlns:reportmail=\"http://openoffice.org/2002/reportmail\" version=\"1.1\" feedback=\"%s\" email=\"%s\">\n",
685 					pszUserType ? pszUserType : "",
686 					pParams->fAllowContact ? "true" : "false",
687 					pParams->fAllowContact ? xml_encode(szEmail).c_str() : ""
688 					);
689 
690 				fprintf( fp,
691 					"<reportmail:title>%s</reportmail:title>\n",
692 					xml_encode(szTitle).c_str() );
693 
694 				fprintf( fp,
695 					"<reportmail:attachment name=\"description.txt\" media-type=\"text/plain;charset=UTF-8\" class=\"UserComment\"/>\n"
696 					"<reportmail:attachment name=\"user.dmp\" media-type=\"application/octet-stream\" class=\"UserDump\"/>\n"
697 					"</reportmail:mail>\n"
698 
699 					"<officeinfo:officeinfo xmlns:officeinfo=\"http://openoffice.org/2002/officeinfo\" build=\"%s\" platform=\"%s\" language=\"%s\" procpath=\"%s\" exceptiontype=\"0x%08X\" product=\"%s\"/>\n",
700 					szBuildId,
701 					_INPATH,
702 					xml_encode(g_strDefaultLanguage).c_str(),
703 					xml_encode(GetModuleDirectory( NULL )).c_str(),
704 					g_dwExceptionCode,
705 					xml_encode(wstring2utf8(g_wstrProductKey)).c_str()
706 					);
707 
708 				OSVERSIONINFO	VersionInfo;
709 
710 				ZeroMemory( &VersionInfo, sizeof(VersionInfo) );
711 				VersionInfo.dwOSVersionInfoSize = sizeof(VersionInfo );
712 
713 				GetVersionEx( &VersionInfo );
714 
715 				fprintf( fp,
716 					"<systeminfo:systeminfo xmlns:systeminfo=\"http://openoffice.org/2002/systeminfo\">\n"
717 					"<systeminfo:System name=\"%s\" version=\"%d.%d\" build=\"%d\" locale=\"0x%08x\"/>\n"
718 					,
719 					VER_PLATFORM_WIN32_NT == VersionInfo.dwPlatformId ? "Windows NT" : "Windows",
720 					VersionInfo.dwMajorVersion, VersionInfo.dwMinorVersion,
721 					VersionInfo.dwBuildNumber,
722 					GetUserDefaultLangID()
723 
724 					);
725 				fprintf( fp, "<systeminfo:CPU type=\"x86\"/>\n" );
726 				fprintf( fp, "</systeminfo:systeminfo>\n" );
727 
728 				fseek( g_fpStackFile, 0, SEEK_SET );
729 				fcopy( g_fpStackFile, fp );
730 
731 				fseek( g_fpChecksumFile, 0, SEEK_SET );
732 				fcopy( g_fpChecksumFile, fp );
733 
734 				fprintf( fp, "</errormail:errormail>\n" );
735 
736 				fclose( fp );
737 
738 				fSuccess = TRUE;
739 
740 				WideCharToMultiByte( CP_ACP, 0, szFileName, -1, g_szReportFileNameA, MAX_PATH, NULL, NULL );
741 			}
742 
743 			if ( !fSuccess )
744 				DeleteFile( szFileName );
745 		}
746 	}
747 
748 	return fSuccess;
749 }
750 
751 //***************************************************************************
752 
753 static BOOL SaveDumpFile( HWND hwndOwner )
754 {
755 	OPENFILENAME	ofn;
756 	TCHAR	szFileName[MAX_PATH] = TEXT("");
757 
758 	ZeroMemory( &ofn, sizeof(ofn) );
759 	ofn.lStructSize = sizeof(ofn);
760 
761 	ofn.hwndOwner = hwndOwner;
762 	ofn.lpstrFilter = TEXT("*.dmp\0*.dmp\0*.*\0*.*\0");
763 	ofn.lpstrFile = szFileName;
764 	ofn.nMaxFile = MAX_PATH;
765 	ofn.Flags = OFN_ENABLESIZING | OFN_LONGNAMES | OFN_OVERWRITEPROMPT;
766 	ofn.lpstrDefExt = TEXT("dmp");
767 
768 	if ( GetSaveFileName( &ofn ) )
769 	{
770 		return CopyFile( g_szDumpFileName, szFileName, FALSE );
771 	}
772 
773 
774 	return FALSE;
775 }
776 
777 //***************************************************************************
778 
779 static BOOL ScreenToClientRect( HWND hwnd, LPRECT lprc )
780 {
781 	return ScreenToClient( hwnd, (LPPOINT)&lprc->left ) && ScreenToClient( hwnd, (LPPOINT)&lprc->right );
782 }
783 
784 static BOOL SetWindowRect( HWND hwnd, const RECT *lprc, BOOL fRepaint )
785 {
786 	return MoveWindow( hwnd, lprc->left, lprc->top, lprc->right - lprc->left, lprc->bottom - lprc->top, fRepaint );
787 }
788 
789 #define GM_LOX	0x01
790 #define GM_HIX	0x02
791 #define GM_LOY	0x04
792 #define GM_HIY	0x08
793 
794 static BOOL SetGrowMode( HWND hwnd, DWORD dwGrowMode )
795 {
796 	return SetProp( hwnd, TEXT("GrowMode"), (HANDLE)dwGrowMode );
797 }
798 
799 static DWORD GetGrowMode( HWND hwnd )
800 {
801 	return (DWORD)GetProp( hwnd, TEXT("GrowMode") );
802 }
803 
804 static BOOL GrowWindow( HWND hwnd, LONG dxClient, LONG dyClient, BOOL fRepaint )
805 {
806 	DWORD	dwGrowMode = GetGrowMode( hwnd );
807 	RECT	rc;
808 
809 	GetWindowRect( hwnd, &rc );
810 
811 	if ( dwGrowMode & GM_LOX )
812 		rc.left += dxClient;
813 	if ( dwGrowMode & GM_HIX )
814 		rc.right += dxClient;
815 	if ( dwGrowMode & GM_LOY )
816 		rc.top += dyClient;
817 	if ( dwGrowMode & GM_HIY )
818 		rc.bottom += dyClient;
819 
820 	ScreenToClientRect( GetParent( hwnd ), &rc );
821 	SetWindowRect( hwnd, &rc, fRepaint );
822 
823 	return TRUE;
824 }
825 
826 BOOL CALLBACK GrowChildWindows(
827   HWND hwnd,      // handle to child window
828   LPARAM lParam   // application-defined value
829 )
830 {
831 	LONG	cx = (SHORT)LOWORD( lParam );
832 	LONG	cy = (SHORT)HIWORD( lParam );
833 
834 	GrowWindow( hwnd, cx, cy, TRUE );
835 
836 	return TRUE;
837 }
838 
839 /*
840 BOOL CALLBACK EnumChildProc(HWND hwndChild, LPARAM lParam)
841 {
842     HFONT aFont = *((HFONT*) lParam);
843     HDC hDC = GetDC( hwndChild );
844     SelectObject( hDC, aFont );
845     ReleaseDC( hwndChild, hDC );
846     return TRUE;
847 }
848 
849 void ApplySystemFont( HWND hwndDlg )
850 {
851     NONCLIENTMETRICSA aNonClientMetrics;
852     aNonClientMetrics.cbSize = sizeof( aNonClientMetrics );
853     if ( SystemParametersInfoA( SPI_GETNONCLIENTMETRICS, sizeof( aNonClientMetrics ), &aNonClientMetrics, 0 ) )
854     {
855         HFONT aSysFont = CreateFontIndirectA( &aNonClientMetrics.lfMessageFont );
856         EnumChildWindows(hwndDlg, EnumChildProc, (LPARAM) &aSysFont);
857     }
858 }
859 */
860 
861 BOOL CALLBACK PreviewDialogProc(
862 	HWND hwndDlg,
863 	UINT uMsg,
864 	WPARAM wParam,
865 	LPARAM lParam
866 	)
867 {
868 	static RECT	rcClient;
869 
870 	switch ( uMsg )
871 	{
872 	case WM_SIZE:
873 		{
874 		LONG	cx = LOWORD( lParam );
875 		LONG	cy = HIWORD( lParam );
876 		LONG	dxClient, dyClient;
877 
878 		dxClient = cx - rcClient.right;
879 		dyClient = cy - rcClient.bottom;
880 
881 		EnumChildWindows( hwndDlg, GrowChildWindows, MAKELONG( (SHORT)dxClient, (SHORT)dyClient) );
882 
883 		GetClientRect( hwndDlg, &rcClient );
884 		}
885 		break;
886 	case WM_INITDIALOG:
887 		{
888 			GetClientRect( hwndDlg, &rcClient );
889 			SetGrowMode( GetDlgItem(hwndDlg, IDC_EDIT_PREVIEW), GM_HIX | GM_HIY );
890 			SetGrowMode( GetDlgItem(hwndDlg, IDOK), GM_LOX | GM_HIX | GM_LOY | GM_HIY );
891 
892 			CrashReportParams *pParams = (CrashReportParams *)lParam;
893 
894 			TCHAR	szBuffer[256] = TEXT("");
895 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong( hwndDlg, GWL_HINSTANCE );
896 			HWND	hwndParent = (HWND)GetWindowLong( hwndDlg, GWL_HWNDPARENT );
897 
898 			GetWindowText( hwndParent, szBuffer, elementsof(szBuffer) );
899 			SetWindowText( hwndDlg, szBuffer );
900 
901 			LoadAndFormatString( hInstance, IDS_OK_BUTTON, szBuffer, elementsof(szBuffer) );
902 			Button_SetText( GetDlgItem(hwndDlg, IDOK), szBuffer );
903 
904 			basic_string<TCHAR>	aString;
905 
906 			aString.append( pParams->sTitle );
907 			aString.append( _T("\r\n\r\n") );
908 			aString.append( pParams->sComment );
909 			aString.append( _T("\r\n---------- report ----------\r\n") );
910 
911 			FILE	*fp = fopen( g_szReportFileNameA, "r" );
912 
913 			if ( fp )
914 			{
915 				char	buf[1024];
916 
917 				while ( fgets( buf, elementsof(buf), fp ) != NULL )
918 				{
919 					WCHAR	bufW[1024];
920 
921 					MultiByteToWideChar( CP_UTF8, 0, buf, -1, bufW, elementsof(bufW) );
922 
923 					aString.append( bufW );
924 				}
925 
926 				fclose( fp );
927 			}
928 
929 			aString.append( _T("\r\n---------- stack ----------\r\n") );
930 
931 			fp = fopen( g_szDumpFileNameA, "rb" );
932 
933 			if ( fp )
934 			{
935 				unsigned char	buf[16];
936 				int		count;
937 
938 				do
939 				{
940 					int i;
941 
942 					count = fread( buf, sizeof(buf[0]), sizeof(buf)/sizeof(buf[0]), fp );
943 
944 					for ( i = 0; i < count; i++ )
945 					{
946 						TCHAR	output[16];
947 
948 						_sntprintf( output, elementsof(output), _T("%02X\x20"), buf[i] );
949 						aString.append( output );
950 					}
951 					for ( ; i < elementsof(buf); i++ )
952 					{
953 						aString.append( _T("\x20\x20\x20") );
954 					}
955 
956 					for ( i = 0; i < count; i++ )
957 					{
958 						TCHAR	output[2];
959 
960 						if ( (int)buf[i] >= 0x20 && (int)buf[i] <= 0x7F )
961 							output[0] = (TCHAR)buf[i];
962 						else
963 							output[0] = '.';
964 						output[1] = 0;
965 						aString.append( output );
966 					}
967 
968 					aString.append( _T("\r\n") );
969 
970 				} while ( count );
971 
972 				fclose( fp );
973 			}
974 
975 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_PREVIEW), aString.c_str() );
976 
977 
978 			SetWindowFont( GetDlgItem(hwndDlg, IDC_EDIT_PREVIEW), GetStockObject( SYSTEM_FIXED_FONT ), TRUE );
979 		}
980 		return TRUE;
981 	case WM_COMMAND:
982 		switch ( LOWORD(wParam) )
983 		{
984 		case IDOK:
985 		case IDCANCEL:
986 			EndDialog( hwndDlg, wParam );
987 			return TRUE;
988 		}
989 		break;
990 	default:
991 		break;
992 	}
993 
994 	return FALSE;
995 }
996 //***************************************************************************
997 
998 static void PreviewReport( HWND hwndParent, CrashReportParams *pParams )
999 {
1000 	HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndParent, GWL_HINSTANCE );
1001 
1002 	WriteReportFile( pParams );
1003 
1004 	DialogBoxParam(
1005 		hInstance,
1006 		MAKEINTRESOURCE(IDD_PREVIEW_FRAME),
1007 		hwndParent,
1008 		PreviewDialogProc,
1009 		(LPARAM)pParams
1010 		);
1011 
1012 	DeleteFileA( g_szReportFileNameA );
1013 }
1014 //***************************************************************************
1015 void UpdateOptionsDialogControls( HWND hwndDlg )
1016 {
1017 	if ( Button_GetCheck( GetDlgItem(hwndDlg, IDC_RADIO_MANUAL) ) & BST_CHECKED )
1018 	{
1019 		EnableWindow( GetDlgItem(hwndDlg, IDC_EDIT_PROXYSERVER), TRUE );
1020 		EnableWindow( GetDlgItem(hwndDlg, IDC_EDIT_PROXYPORT), TRUE );
1021 	}
1022 	else
1023 	{
1024 		EnableWindow( GetDlgItem(hwndDlg, IDC_EDIT_PROXYSERVER), FALSE );
1025 		EnableWindow( GetDlgItem(hwndDlg, IDC_EDIT_PROXYPORT), FALSE );
1026 	}
1027 }
1028 
1029 //***************************************************************************
1030 
1031 BOOL CALLBACK OptionsDialogProc(
1032 	HWND hwndDlg,
1033 	UINT uMsg,
1034 	WPARAM wParam,
1035 	LPARAM lParam
1036 	)
1037 {
1038 	static CrashReportParams *pParams;
1039 
1040 	switch ( uMsg )
1041 	{
1042 	case WM_INITDIALOG:
1043 		{
1044 			TCHAR	szBuffer[1024] = TEXT("");
1045 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong( hwndDlg, GWL_HINSTANCE );
1046 			//HWND	hwndParent = (HWND)GetWindowLong( hwndDlg, GWL_HWNDPARENT );
1047 
1048 			pParams = (CrashReportParams *)lParam;
1049 
1050 			LoadAndFormatString( hInstance, IDS_OPTIONS_CAPTION, szBuffer, elementsof(szBuffer) );
1051 			SetWindowText( hwndDlg, szBuffer );
1052 
1053 			LoadAndFormatString( hInstance, IDS_PROXY_SETTINGS_HEADER, szBuffer, elementsof(szBuffer) );
1054 			Static_SetText( GetDlgItem(hwndDlg, IDC_PROXY_SETTINGS), szBuffer );
1055 
1056 			LoadAndFormatString( hInstance, IDS_PROXY_SYSTEM, szBuffer, elementsof(szBuffer) );
1057 			Button_SetText( GetDlgItem(hwndDlg, IDC_RADIO_SYSTEM), szBuffer );
1058 
1059 			LoadAndFormatString( hInstance, IDS_PROXY_DIRECT, szBuffer, elementsof(szBuffer) );
1060 			Button_SetText( GetDlgItem(hwndDlg, IDC_RADIO_DIRECT), szBuffer );
1061 
1062 			LoadAndFormatString( hInstance, IDS_PROXY_MANUAL, szBuffer, elementsof(szBuffer) );
1063 			Button_SetText( GetDlgItem(hwndDlg, IDC_RADIO_MANUAL), szBuffer );
1064 
1065 			LoadAndFormatString( hInstance, IDS_LABEL_PROXYSERVER, szBuffer, elementsof(szBuffer) );
1066 			Static_SetText( GetDlgItem(hwndDlg, IDC_LABEL_PROXYSERVER), szBuffer );
1067 
1068 			LoadAndFormatString( hInstance, IDS_LABEL_PROXYPORT, szBuffer, elementsof(szBuffer) );
1069 			Static_SetText( GetDlgItem(hwndDlg, IDC_LABEL_PROXYPORT), szBuffer );
1070 
1071 			LoadAndFormatString( hInstance, IDS_OK_BUTTON, szBuffer, elementsof(szBuffer) );
1072 			Button_SetText( GetDlgItem(hwndDlg, IDOK), szBuffer );
1073 
1074 			LoadAndFormatString( hInstance, IDS_CANCEL_BUTTON, szBuffer, elementsof(szBuffer) );
1075 			Button_SetText( GetDlgItem(hwndDlg, IDCANCEL), szBuffer );
1076 
1077 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_PROXYSERVER), pParams->sProxyServer.c_str() );
1078 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_PROXYPORT), pParams->sProxyPort.c_str() );
1079 
1080 			Button_SetCheck( GetDlgItem(hwndDlg, IDC_RADIO_SYSTEM + pParams->uInternetConnection), BST_CHECKED );
1081 
1082 			SendMessage(
1083 				GetDlgItem(hwndDlg, IDC_PROXY_DESCRIPTION),
1084 				EM_SETBKGNDCOLOR,
1085 				(WPARAM)FALSE,
1086 				GetSysColor( COLOR_3DFACE ) );
1087 			LoadAndFormatString( hInstance, IDS_PROXY_DESCRIPTION, szBuffer, elementsof(szBuffer) );
1088 			Edit_SetText( GetDlgItem(hwndDlg, IDC_PROXY_DESCRIPTION), szBuffer );
1089 
1090 			UpdateOptionsDialogControls( hwndDlg );
1091 		}
1092 		return TRUE;
1093 	case WM_COMMAND:
1094 		switch ( LOWORD(wParam) )
1095 		{
1096 		case IDC_RADIO_SYSTEM:
1097 		case IDC_RADIO_DIRECT:
1098 		case IDC_RADIO_MANUAL:
1099 			if ( BN_CLICKED == HIWORD(wParam) )
1100 				UpdateOptionsDialogControls( hwndDlg );
1101 			break;
1102 		case IDOK:
1103 			{
1104 			TCHAR szBuffer[1024];
1105 
1106 			Edit_GetText( GetDlgItem(hwndDlg, IDC_EDIT_PROXYSERVER), szBuffer, elementsof(szBuffer) );
1107 			pParams->sProxyServer = szBuffer;
1108 
1109 			Edit_GetText( GetDlgItem(hwndDlg, IDC_EDIT_PROXYPORT), szBuffer, elementsof(szBuffer) );
1110 			pParams->sProxyPort = szBuffer;
1111 
1112 			if ( Button_GetCheck( GetDlgItem(hwndDlg, IDC_RADIO_DIRECT) ) & BST_CHECKED )
1113 				pParams->uInternetConnection = 1;
1114 			else if ( Button_GetCheck( GetDlgItem(hwndDlg, IDC_RADIO_MANUAL) ) & BST_CHECKED )
1115 				pParams->uInternetConnection = 2;
1116 			else
1117 				pParams->uInternetConnection = 0;
1118 			}
1119 		case IDCANCEL:
1120 			EndDialog( hwndDlg, wParam );
1121 			return TRUE;
1122 		}
1123 		break;
1124 	default:
1125 		break;
1126 	}
1127 
1128 	return FALSE;
1129 }
1130 
1131 //***************************************************************************
1132 
1133 static void OptionsDialog( HWND hwndParent, CrashReportParams *pParams )
1134 {
1135 	HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndParent, GWL_HINSTANCE );
1136 
1137 	if ( IDOK == DialogBoxParam(
1138 		hInstance,
1139 		MAKEINTRESOURCE(IDD_OPTIONS_FRAME),
1140 		hwndParent,
1141 		OptionsDialogProc,
1142 		(LPARAM)pParams
1143 		) )
1144 		pParams->WriteToRegistry();
1145 
1146 }
1147 //***************************************************************************
1148 
1149 void UpdateReportDialogControls( HWND hwndDlg )
1150 {
1151 	EnableWindow(
1152 		GetDlgItem(hwndDlg, IDC_EDIT_EMAIL),
1153 		Button_GetCheck(GetDlgItem(hwndDlg, IDC_ALLOW_CONTACT)) & BST_CHECKED ? TRUE : FALSE );
1154 	EnableWindow(
1155 		GetDlgItem(hwndDlg, IDC_LABEL_EMAIL),
1156 		Button_GetCheck(GetDlgItem(hwndDlg, IDC_ALLOW_CONTACT)) & BST_CHECKED ? TRUE : FALSE );
1157 }
1158 
1159 //***************************************************************************
1160 
1161 BOOL CALLBACK ReportDialogProc(
1162 	HWND hwndDlg,
1163 	UINT uMsg,
1164 	WPARAM wParam,
1165 	LPARAM
1166 	)
1167 {
1168 	switch ( uMsg )
1169 	{
1170 	case WM_INITDIALOG:
1171 		{
1172 		    CrashReportParams	*pParams = (CrashReportParams*)GetWindowLong( GetParent(hwndDlg), GWL_USERDATA );
1173 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndDlg, GWL_HINSTANCE );
1174 			TCHAR		szBuffer[FORMATBUFSIZE];
1175 
1176 			LoadAndFormatString( hInstance, IDS_REPORT_INTRO, szBuffer, elementsof(szBuffer) );
1177 			Static_SetText( GetDlgItem(hwndDlg, IDC_REPORT_INTRO), szBuffer );
1178 
1179 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT3), szBuffer );
1180 
1181 			LoadAndFormatString( hInstance, IDS_ENTER_TITLE, szBuffer, elementsof(szBuffer) );
1182 			Static_SetText( GetDlgItem(hwndDlg, IDC_ENTER_TITLE), szBuffer );
1183 
1184 			LoadAndFormatString( hInstance, IDS_ENTER_DESCRIPTION, szBuffer, elementsof(szBuffer) );
1185 			Static_SetText( GetDlgItem(hwndDlg, IDC_ENTER_DESCRIPTION), szBuffer );
1186 
1187 			LoadAndFormatString( hInstance, IDS_SHOW_REPORT_BUTTON, szBuffer, elementsof(szBuffer) );
1188 			Button_SetText( GetDlgItem(hwndDlg, IDC_SHOW_REPORT), szBuffer );
1189 
1190 			LoadAndFormatString( hInstance, IDS_SAVE_REPORT_BUTTON, szBuffer, elementsof(szBuffer) );
1191 			Button_SetText( GetDlgItem(hwndDlg, IDC_SAVE_REPORT), szBuffer );
1192 
1193 			const char *pszUserType = getenv( "STAROFFICE_USERTYPE" );
1194 			if ( pszUserType )
1195 				ShowWindow( GetDlgItem(hwndDlg, IDC_SAVE_REPORT), SW_SHOW );
1196 			else
1197 				ShowWindow( GetDlgItem(hwndDlg, IDC_SAVE_REPORT), SW_HIDE );
1198 
1199 			LoadAndFormatString( hInstance, IDS_OPTIONS_BUTTON, szBuffer, elementsof(szBuffer) );
1200 			Button_SetText( GetDlgItem(hwndDlg, IDC_OPTIONS), szBuffer );
1201 
1202 			LoadAndFormatString( hInstance, IDS_ALLOW_CONTACT, szBuffer, elementsof(szBuffer) );
1203 			Button_SetText( GetDlgItem(hwndDlg, IDC_ALLOW_CONTACT), szBuffer );
1204 			Button_SetCheck( GetDlgItem(hwndDlg, IDC_ALLOW_CONTACT), pParams->fAllowContact ? BST_CHECKED : BST_UNCHECKED );
1205 
1206 			LoadAndFormatString( hInstance, IDS_LABEL_EMAIL, szBuffer, elementsof(szBuffer) );
1207 			Button_SetText( GetDlgItem(hwndDlg, IDC_LABEL_EMAIL), szBuffer );
1208 
1209 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_EMAIL), pParams->sEmail.c_str() );
1210 
1211 			UpdateReportDialogControls( hwndDlg );
1212 		}
1213 		return TRUE;
1214 	case WM_SHOWWINDOW:
1215 		if ( (BOOL)wParam )
1216 		{
1217 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndDlg, GWL_HINSTANCE );
1218 			CrashReportParams	*pParams = (CrashReportParams*)GetWindowLong( GetParent(hwndDlg), GWL_USERDATA );
1219 			TCHAR		szBuffer[FORMATBUFSIZE];
1220 
1221 			LoadAndFormatString( hInstance, IDS_REPORT_CAPTION, szBuffer, elementsof(szBuffer) );
1222 			SetWindowText( GetParent(hwndDlg), szBuffer );
1223 
1224 			LoadAndFormatString( hInstance, IDS_REPORT_HEADER, szBuffer, elementsof(szBuffer) );
1225 			SetWindowText( GetDlgItem(GetParent(hwndDlg), IDC_HEADER), szBuffer );
1226 
1227 			LoadAndFormatString( hInstance, IDS_DONOT_SEND_BUTTON, szBuffer, elementsof(szBuffer) );
1228 			Button_SetText( GetDlgItem(GetParent(hwndDlg), IDCANCEL), szBuffer );
1229 
1230 
1231 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDBACK), TRUE );
1232 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDFINISH), TRUE );
1233 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDNEXT), FALSE );
1234 
1235 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_TITLE), pParams->sTitle.c_str() );
1236 			Edit_SetText( GetDlgItem(hwndDlg, IDC_EDIT_DESCRIPTION), pParams->sComment.c_str() );
1237 
1238             /*
1239 			SetWindowLong( GetDlgItem(GetParent(hwndDlg),IDFINISH), GWL_STYLE,
1240                 GetWindowLong( GetDlgItem(GetParent(hwndDlg),IDFINISH), GWL_STYLE) | BS_DEFPUSHBUTTON );
1241 			SetWindowLong( GetDlgItem(GetParent(hwndDlg),IDBACK), GWL_STYLE,
1242                 GetWindowLong( GetDlgItem(GetParent(hwndDlg),IDBACK), GWL_STYLE) &~ BS_DEFPUSHBUTTON );
1243                 */
1244 			SetFocus( GetDlgItem(hwndDlg,IDC_EDIT_TITLE) );
1245 		}
1246 		break;
1247 	case WM_COMMAND:
1248 		switch ( LOWORD(wParam) )
1249 		{
1250 		case IDC_SHOW_REPORT:
1251 			{
1252 				TCHAR	szBuffer[32767];
1253 
1254 				CrashReportParams	*pParams = (CrashReportParams*)GetWindowLong( GetParent(hwndDlg), GWL_USERDATA );
1255 
1256 				pParams->fAllowContact = Button_GetCheck( GetDlgItem(hwndDlg, IDC_ALLOW_CONTACT) ) ? TRUE : FALSE;
1257 
1258 				Edit_GetText( GetDlgItem(hwndDlg, IDC_EDIT_TITLE), szBuffer, elementsof(szBuffer) );
1259 				pParams->sTitle = szBuffer;
1260 
1261 				Edit_GetText( GetDlgItem(hwndDlg, IDC_EDIT_DESCRIPTION), szBuffer, elementsof(szBuffer) );
1262 				pParams->sComment = szBuffer;
1263 
1264 				Edit_GetText( GetDlgItem(hwndDlg, IDC_EDIT_EMAIL), szBuffer, elementsof(szBuffer) );
1265 				pParams->sEmail = szBuffer;
1266 
1267 				PreviewReport( GetParent(hwndDlg), pParams );
1268 			}
1269 			return TRUE;
1270 		case IDC_SAVE_REPORT:
1271 			SaveDumpFile( GetParent(hwndDlg) );
1272 			return TRUE;
1273 		case IDC_OPTIONS:
1274 			{
1275 				CrashReportParams	*pParams = (CrashReportParams*)GetWindowLong( GetParent(hwndDlg), GWL_USERDATA );
1276 				OptionsDialog( GetParent(hwndDlg), pParams );
1277 			}
1278 			return TRUE;
1279 		case IDC_ALLOW_CONTACT:
1280 			if ( BN_CLICKED == HIWORD(wParam) )
1281 				UpdateReportDialogControls( hwndDlg );
1282 			return TRUE;
1283 		}
1284 		break;
1285 	default:
1286 		break;
1287 	}
1288 
1289 	return FALSE;
1290 }
1291 //***************************************************************************
1292 
1293 BOOL CALLBACK WelcomeDialogProc( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam )
1294 {
1295 	switch ( uMsg )
1296 	{
1297 	case WM_INITDIALOG:
1298 		{
1299 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndDlg, GWL_HINSTANCE );
1300 			HWND	hwndRichEdit = GetDlgItem(hwndDlg, IDC_RICHEDIT21);
1301 			TCHAR	szBuffer[FORMATBUFSIZE];
1302 			TCHAR	szBuffer2[FORMATBUFSIZE];
1303 			TCHAR	szURL[256];
1304 			TCHAR	szCaption[256];
1305 
1306 			SendMessage(
1307 				hwndRichEdit,
1308 				EM_SETBKGNDCOLOR,
1309 				(WPARAM)FALSE,
1310 				GetSysColor( COLOR_3DFACE ) );
1311 
1312 			SendMessage( hwndRichEdit, EM_SETEVENTMASK, 0, ENM_LINK );
1313 			SendMessage( hwndRichEdit, EM_AUTOURLDETECT, TRUE, 0 );
1314 
1315 			LoadAndFormatString( hInstance, IDS_WELCOME_BODY1, szBuffer, elementsof(szBuffer) );
1316 			LoadAndFormatString( hInstance, IDS_WELCOME_BODY2, szBuffer2, elementsof(szBuffer2) );
1317 			_tcsncat( szBuffer, szBuffer2, elementsof(szBuffer) );
1318 			LoadAndFormatString( hInstance, IDS_WELCOME_BODY3, szBuffer2, elementsof(szBuffer2) );
1319 			_tcsncat( szBuffer, szBuffer2, elementsof(szBuffer) );
1320 			LoadString( hInstance, IDS_PRIVACY_URL, szURL, elementsof(szURL) );
1321 			_tcsncat( szBuffer, szURL, elementsof(szBuffer) );
1322 			SetWindowText( hwndRichEdit, szBuffer );
1323 
1324 			LoadAndFormatString( hInstance, IDS_WELCOME_CAPTION, szCaption, elementsof(szCaption) );
1325 			SetWindowText( GetParent(hwndDlg), szCaption );
1326 
1327 		}
1328 		return TRUE;
1329 	case WM_SHOWWINDOW:
1330 		if ( (BOOL)wParam )
1331 		{
1332 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndDlg, GWL_HINSTANCE );
1333 			TCHAR		szBuffer[FORMATBUFSIZE];
1334 
1335 			LoadAndFormatString( hInstance, IDS_WELCOME_CAPTION, szBuffer, elementsof(szBuffer) );
1336 			SetWindowText( GetParent(hwndDlg), szBuffer );
1337 
1338 			LoadAndFormatString( hInstance, IDS_WELCOME_HEADER, szBuffer, elementsof(szBuffer) );
1339 			SetWindowText( GetDlgItem(GetParent(hwndDlg), IDC_HEADER), szBuffer );
1340 
1341 			LoadAndFormatString( hInstance, IDS_CANCEL_BUTTON, szBuffer, elementsof(szBuffer) );
1342 			Button_SetText( GetDlgItem(GetParent(hwndDlg), IDCANCEL), szBuffer );
1343 
1344 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDBACK), FALSE );
1345 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDFINISH), FALSE );
1346 			ShowWindow( GetDlgItem(GetParent(hwndDlg),IDNEXT), TRUE );
1347 
1348 			SetFocus( GetDlgItem(GetParent(hwndDlg),IDNEXT) );
1349 		}
1350 		break;
1351 	case WM_NOTIFY:
1352 		{
1353 			LPNMHDR	pnmh = (LPNMHDR)lParam;
1354 
1355 			if ( pnmh->idFrom == IDC_RICHEDIT21 && pnmh->code == EN_LINK )
1356 			{
1357 				ENLINK	*plink = (ENLINK*)lParam;
1358 
1359 				if ( plink->msg == WM_LBUTTONUP )
1360 				{
1361 					TCHAR	szBuffer[256];
1362 					TEXTRANGE	range;
1363 
1364 					range.chrg = plink->chrg;
1365 					range.lpstrText = szBuffer;
1366 
1367 					SendMessage( pnmh->hwndFrom, EM_GETTEXTRANGE, 0, (LPARAM)&range );
1368 
1369 					ShellExecute( hwndDlg, NULL, szBuffer, NULL, NULL, SW_SHOWDEFAULT );
1370 				}
1371 
1372 			}
1373 		}
1374 		break;
1375 	default:
1376 		break;
1377 	}
1378 
1379 	return FALSE;
1380 }
1381 //***************************************************************************
1382 
1383 BOOL CALLBACK DialogProc( HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam )
1384 {
1385 	static	HWND	hwndPages[2] = { NULL };
1386 	static	int		iActualPage = 0;
1387 
1388 	switch ( uMsg )
1389 	{
1390 	case WM_INITDIALOG:
1391 		{
1392 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong(hwndDlg, GWL_HINSTANCE );
1393 			TCHAR		szBuffer[FORMATBUFSIZE];
1394 
1395 			SetWindowLong( hwndDlg, GWL_USERDATA, (LONG)lParam );
1396 			hwndPages[0] = CreateDialog(
1397 				hInstance,
1398 				MAKEINTRESOURCE(IDD_WELCOME_PAGE),
1399 				hwndDlg,
1400 				WelcomeDialogProc );
1401 
1402 			hwndPages[1] = CreateDialog(
1403 				hInstance,
1404 				MAKEINTRESOURCE(IDD_REPORT_PAGE),
1405 				hwndDlg,
1406 				ReportDialogProc );
1407 
1408 			CHARFORMAT	chfmt;
1409 
1410 			chfmt.cbSize = sizeof(chfmt);
1411 			chfmt.dwMask = CFM_BOLD;
1412 			chfmt.dwEffects = CFE_BOLD;
1413 
1414 			SendMessage(
1415 				GetDlgItem(hwndDlg, IDC_HEADER),
1416 				EM_SETCHARFORMAT,
1417 				SCF_ALL,
1418 				(LPARAM)&chfmt );
1419 
1420 			LoadAndFormatString( hInstance, IDS_CANCEL_BUTTON, szBuffer, elementsof(szBuffer) );
1421 			Button_SetText( GetDlgItem(hwndDlg, IDCANCEL), szBuffer );
1422 
1423 			LoadAndFormatString( hInstance, IDS_NEXT_BUTTON, szBuffer, elementsof(szBuffer) );
1424 			Button_SetText( GetDlgItem(hwndDlg, IDNEXT), szBuffer );
1425 
1426 			LoadAndFormatString( hInstance, IDS_SEND_BUTTON, szBuffer, elementsof(szBuffer) );
1427 			Button_SetText( GetDlgItem(hwndDlg, IDFINISH), szBuffer );
1428 
1429 			LoadAndFormatString( hInstance, IDS_BACK_BUTTON, szBuffer, elementsof(szBuffer) );
1430 			Button_SetText( GetDlgItem(hwndDlg, IDBACK), szBuffer );
1431 
1432 			ShowWindow( hwndPages[1], SW_HIDE );
1433 			ShowWindow( hwndPages[0], SW_SHOW );
1434 
1435 			// Let Crash Reporter window stay on top of all other windows
1436 			SetWindowPos( hwndDlg, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE );
1437 		}
1438 		return FALSE;
1439 	case WM_CTLCOLORSTATIC:
1440 		return (BOOL)CreateSolidBrush(GetSysColor(COLOR_WINDOW));
1441 	case WM_COMMAND:
1442 		switch ( LOWORD(wParam) )
1443 		{
1444 		case IDBACK:
1445 			if ( iActualPage > 0 )
1446 			{
1447 				ShowWindow( hwndPages[iActualPage], SW_HIDE );
1448 				ShowWindow( hwndPages[--iActualPage], SW_SHOW );
1449 			}
1450 			return TRUE;
1451 		case IDNEXT:
1452 			if ( iActualPage < elementsof(hwndPages) - 1 )
1453 			{
1454 				ShowWindow( hwndPages[iActualPage], SW_HIDE );
1455 				ShowWindow( hwndPages[++iActualPage], SW_SHOW );
1456 			}
1457 			return TRUE;
1458 		case IDFINISH:
1459 			{
1460 				TCHAR	szBuffer[32767];
1461 				CrashReportParams	*pParams = (CrashReportParams*)GetWindowLong( hwndDlg, GWL_USERDATA );
1462 
1463 				pParams->fAllowContact = Button_GetCheck( GetDlgItem(hwndPages[1], IDC_ALLOW_CONTACT) ) ? TRUE : FALSE;
1464 
1465 				Edit_GetText( GetDlgItem(hwndPages[1], IDC_EDIT_TITLE), szBuffer, elementsof(szBuffer) );
1466 				pParams->sTitle = szBuffer;
1467 
1468 				Edit_GetText( GetDlgItem(hwndPages[1], IDC_EDIT_DESCRIPTION), szBuffer, elementsof(szBuffer) );
1469 				pParams->sComment = szBuffer;
1470 
1471 				Edit_GetText( GetDlgItem(hwndPages[1], IDC_EDIT_EMAIL), szBuffer, elementsof(szBuffer) );
1472 				pParams->sEmail = szBuffer;
1473 
1474 				if ( pParams->fAllowContact && !pParams->sEmail.length() )
1475 				{
1476 					TCHAR	szMessage[MAX_TEXT_BUFFER];
1477 
1478 					LoadAndFormatString( GetModuleHandle(NULL), IDS_ERROR_MSG_NOEMAILADDRESS, szMessage, elementsof(szMessage) );
1479 
1480 					MessageBox( hwndDlg, szMessage, NULL, MB_ICONERROR | MB_OK );
1481 					break;  // Don't end the dialog
1482 				}
1483 				else
1484 				{
1485 					pParams->WriteToRegistry();
1486 
1487 					WriteCommentFile( pParams->sComment.c_str() );
1488 					WriteReportFile( pParams );
1489 
1490 					if ( !SendCrashReport( hwndDlg, *pParams ) )
1491 						break; // Don't end the dialog
1492 				}
1493 			}
1494 			// Fallthrough !!!
1495 		case IDCANCEL:
1496 			EndDialog( hwndDlg, wParam );
1497 			return TRUE;
1498 		}
1499 		break;
1500 	default:
1501 		break;
1502 	}
1503 
1504 	return FALSE;
1505 }
1506 
1507 
1508 
1509 //*****************************************************************************
1510 //* Generate MD5 checksum
1511 //*****************************************************************************
1512 
1513 #define MAGIC_DESCRIPTION_FILLER	'x'
1514 #define MAGIC_DESCRIPTION_COUNT		80
1515 
1516 static void repatch_soffice_exe( void *pBuffer, size_t nBufSize )
1517 {
1518 	wchar_t DescriptionBuffer[MAGIC_DESCRIPTION_COUNT];
1519 
1520 	memset( DescriptionBuffer, 0, sizeof(DescriptionBuffer) );
1521 	wcsncpy( DescriptionBuffer, g_wstrProductKey.c_str(), elementsof(DescriptionBuffer) - 1 );
1522 
1523 	bool bPatched = false;
1524 
1525 	do
1526 	{
1527 		void *pFound = memchr( pBuffer, ((char *)DescriptionBuffer)[0], nBufSize );
1528 
1529 		if ( pFound )
1530 		{
1531 			size_t distance = (char *)pFound - (char *)pBuffer;
1532 
1533 			if ( nBufSize >= distance )
1534 			{
1535 				nBufSize -= distance;
1536 
1537 				if ( nBufSize >= sizeof(DescriptionBuffer) &&
1538 					0 == memcmp( pFound, DescriptionBuffer, sizeof(DescriptionBuffer) ) )
1539 				{
1540 					for ( int i = 0; i < 80; i++ )
1541 					{
1542 						((wchar_t *)pFound)[i] = MAGIC_DESCRIPTION_FILLER;
1543 					}
1544 					bPatched = true;
1545 				}
1546 				else
1547 				{
1548 					pBuffer = (void *)(((char *)pFound) + 1);
1549 					nBufSize--;
1550 				}
1551 			}
1552 			else
1553 				nBufSize = 0;
1554 		}
1555 		else
1556 			nBufSize = 0;
1557 	} while ( !bPatched && nBufSize );
1558 }
1559 
1560 // Normalize executable/library images to prevent different MD5 checksums due
1561 // to a different PE header date/checksum (this doesn't affect the code/data
1562 // sections of a executable/library. Please see tools/source/bootstrp/md5.cxx
1563 // where the same method is also used. The tool so_checksum creates the MD5
1564 // checksums during build time. You have to make sure that both methods use the
1565 // same algorithm otherwise there could be problems with stack reports.
1566 static void normalize_pe_image(sal_uInt8* buffer, size_t nBufferSize)
1567 {
1568 	const int OFFSET_PE_OFFSET                  = 0x3c;
1569 	const int OFFSET_COFF_TIMEDATESTAMP         = 4;
1570 	const int PE_SIGNATURE_SIZE                 = 4;
1571 	const int COFFHEADER_SIZE                   = 20;
1572 	const int OFFSET_PE_OPTIONALHEADER_CHECKSUM = 64;
1573 
1574 	// Check the header part of the file buffer
1575 	if (buffer[0] == 'M' && buffer[1] == 'Z')
1576 	{
1577 		unsigned long PEHeaderOffset = (long)buffer[OFFSET_PE_OFFSET];
1578 		if (PEHeaderOffset < nBufferSize-4)
1579 		{
1580 			if ( buffer[PEHeaderOffset] == 'P' &&
1581 				 buffer[PEHeaderOffset+1] == 'E' &&
1582 				 buffer[PEHeaderOffset+2] == 0 &&
1583 				 buffer[PEHeaderOffset+3] == 0 )
1584 			{
1585 				PEHeaderOffset += PE_SIGNATURE_SIZE;
1586 				if (PEHeaderOffset+OFFSET_COFF_TIMEDATESTAMP < nBufferSize-4)
1587 				{
1588 					// Set timedatestamp and checksum fields to a normalized
1589 					// value to enforce the same MD5 checksum for identical
1590 					// Windows	executables/libraries.
1591 					buffer[PEHeaderOffset+OFFSET_COFF_TIMEDATESTAMP] = 0;
1592 					buffer[PEHeaderOffset+OFFSET_COFF_TIMEDATESTAMP+1] = 0;
1593 					buffer[PEHeaderOffset+OFFSET_COFF_TIMEDATESTAMP+2] = 0;
1594 					buffer[PEHeaderOffset+OFFSET_COFF_TIMEDATESTAMP+3] = 0;
1595 				}
1596 
1597 				if (PEHeaderOffset+COFFHEADER_SIZE+OFFSET_PE_OPTIONALHEADER_CHECKSUM < nBufferSize-4)
1598 				{
1599 					// Set checksum to a normalized value
1600 					buffer[PEHeaderOffset+COFFHEADER_SIZE+OFFSET_PE_OPTIONALHEADER_CHECKSUM] = 0;
1601 					buffer[PEHeaderOffset+COFFHEADER_SIZE+OFFSET_PE_OPTIONALHEADER_CHECKSUM+1] = 0;
1602 					buffer[PEHeaderOffset+COFFHEADER_SIZE+OFFSET_PE_OPTIONALHEADER_CHECKSUM+2] = 0;
1603 					buffer[PEHeaderOffset+COFFHEADER_SIZE+OFFSET_PE_OPTIONALHEADER_CHECKSUM+3] = 0;
1604 				}
1605 			}
1606 		}
1607 	}
1608 }
1609 
1610 static sal_uInt32 calc_md5_checksum(  const char *filename, sal_uInt8 *pChecksum, sal_uInt32 nChecksumLen )
1611 {
1612 	const int MINIMAL_FILESIZE = 512;
1613 
1614 	sal_uInt32	nBytesProcessed = 0;
1615 
1616 	FILE *fp = fopen( filename, "rb" );
1617 
1618 	if ( fp )
1619 	{
1620 		long	nFileSize;
1621 
1622 		if ( 0 == fseek( fp, 0, SEEK_END ) && -1 != (nFileSize = ftell(fp)) )
1623 		{
1624 			rewind( fp );
1625 
1626 			sal_uInt8 *pBuffer = new sal_uInt8[nFileSize];
1627 			size_t nBytesRead = fread( pBuffer, 1, nFileSize, fp );
1628 
1629 			if ( sal::static_int_cast<long>(nBytesRead) == nFileSize )
1630 			{
1631 				if ( 0 == stricmp( GetFileName(filename).c_str(), "soffice.bin" ) )
1632 					repatch_soffice_exe( pBuffer, nBytesRead );
1633 				else if ( nFileSize > MINIMAL_FILESIZE )
1634 					normalize_pe_image( pBuffer, nBytesRead );
1635 
1636 				rtlDigestError error = rtl_digest_MD5 (
1637 					pBuffer,   nBytesRead,
1638 					pChecksum, nChecksumLen	);
1639 
1640 				if ( rtl_Digest_E_None == error )
1641 					nBytesProcessed = nBytesRead;
1642 			}
1643 
1644 			delete[] pBuffer;
1645 		}
1646 
1647 		fclose( fp );
1648 
1649 	}
1650 
1651 	return nBytesProcessed;
1652 }
1653 
1654 #if 0
1655 static sal_uInt32 calc_md5_checksum( const char *filename, sal_uInt8 *pChecksum, sal_uInt32 nChecksumLen )
1656 {
1657 	sal_uInt32	nBytesProcessed = 0;
1658 
1659 	FILE *fp = fopen( filename, "rb" );
1660 
1661 	if ( fp )
1662 	{
1663 		rtlDigest digest = rtl_digest_createMD5();
1664 
1665 		if ( digest )
1666 		{
1667 			size_t			nBytesRead;
1668 			sal_uInt8		buffer[4096];
1669 			rtlDigestError	error = rtl_Digest_E_None;
1670 
1671 			while ( rtl_Digest_E_None == error &&
1672 				0 != (nBytesRead = fread( buffer, 1, sizeof(buffer), fp )) )
1673 			{
1674 				error = rtl_digest_updateMD5( digest, buffer, nBytesRead );
1675 				nBytesProcessed += nBytesRead;
1676 			}
1677 
1678 			if ( rtl_Digest_E_None == error )
1679 			{
1680 				error = rtl_digest_getMD5( digest, pChecksum, nChecksumLen );
1681 			}
1682 
1683 			if ( rtl_Digest_E_None != error )
1684 				nBytesProcessed = 0;
1685 
1686 			rtl_digest_destroyMD5( digest );
1687 		}
1688 
1689 		fclose( fp );
1690 	}
1691 
1692 	return nBytesProcessed;
1693 }
1694 
1695 #endif
1696 //***************************************************************************
1697 
1698 static bool WriteStackFile( FILE *fout, hash_map< string, string >& rLibraries, DWORD dwProcessId, PEXCEPTION_POINTERS pExceptionPointers )
1699 {
1700 	bool	fSuccess = false;
1701 
1702 	if ( fout && dwProcessId && pExceptionPointers )
1703 	{
1704 		HANDLE	hProcess = OpenProcess( PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, dwProcessId );
1705 
1706 		if ( IsValidHandle(hProcess) )
1707 		{
1708 			EXCEPTION_POINTERS	aExceptionPointers;
1709 			CONTEXT				aContextRecord;
1710 
1711 			ReadProcessMemory(
1712 				hProcess,
1713 				pExceptionPointers,
1714 				&aExceptionPointers,
1715 				sizeof(aExceptionPointers),
1716 				NULL );
1717 
1718 			ReadProcessMemory(
1719 				hProcess,
1720 				aExceptionPointers.ContextRecord,
1721 				&aContextRecord,
1722 				sizeof(aContextRecord),
1723 				NULL );
1724 
1725 			STACKFRAME	frame;
1726 
1727 			ZeroMemory( &frame, sizeof(frame) );
1728 			frame.AddrPC.Offset = aContextRecord.Eip;
1729 			frame.AddrPC.Mode = AddrModeFlat;
1730 			frame.AddrFrame.Offset = aContextRecord.Ebp;
1731 			frame.AddrFrame.Mode = AddrModeFlat;
1732 
1733 			BOOL bSuccess;
1734 			int frameNum = 0;
1735 
1736 			SymInitialize( hProcess, NULL, TRUE );
1737 
1738 			fprintf( fout, "<errormail:Stack type=\"Win32\">\n" );
1739 
1740 			do
1741 			{
1742 				fSuccess = true;
1743 
1744 				bSuccess = StackWalk( IMAGE_FILE_MACHINE_I386,
1745 					hProcess,
1746 					NULL,
1747 					&frame,
1748 					&aContextRecord,
1749 					(PREAD_PROCESS_MEMORY_ROUTINE)ReadProcessMemory,
1750 					SymFunctionTableAccess,
1751 					SymGetModuleBase,
1752 					NULL );
1753 
1754 				if ( bSuccess )
1755 				{
1756 					// Note: ImageHelp ANSI functions do not have an A postfix while
1757 					//       Unicode versions have a W postfix. There's no macro
1758 					//       that depends on define UNICODE
1759 
1760 					IMAGEHLP_MODULE	moduleInfo;
1761 
1762 					ZeroMemory( &moduleInfo, sizeof(moduleInfo) );
1763 					moduleInfo.SizeOfStruct = sizeof(moduleInfo);
1764 
1765 					if ( SymGetModuleInfo( hProcess, frame.AddrPC.Offset, &moduleInfo ) )
1766 					{
1767 						rLibraries[ GetFileName( moduleInfo.LoadedImageName ).c_str() ] = moduleInfo.LoadedImageName;
1768 
1769 						DWORD	dwRelOffset = 0;
1770 						BYTE	symbolBuffer[sizeof(IMAGEHLP_SYMBOL) + 256 ];
1771 						PIMAGEHLP_SYMBOL	pSymbol = (PIMAGEHLP_SYMBOL)symbolBuffer;
1772 
1773 						ZeroMemory( symbolBuffer, sizeof(symbolBuffer) );
1774 						pSymbol->SizeOfStruct = sizeof(IMAGEHLP_SYMBOL);
1775 						pSymbol->MaxNameLength = 256;
1776 
1777 						if ( SymGetSymFromAddr( hProcess, frame.AddrPC.Offset, &dwRelOffset, pSymbol ) )
1778 							fprintf( fout, "<errormail:StackInfo " \
1779 								"pos=\"%d\" ip=\"0x%p\" rel=\"0x%p\" ordinal=\"%s+0x%p\" name=\"%s\" path=\"%s\"/>\n",
1780 								frameNum,
1781 								frame.AddrPC.Offset,
1782 								frame.AddrPC.Offset - moduleInfo.BaseOfImage,
1783 								xml_encode(pSymbol->Name).c_str(),
1784 								frame.AddrPC.Offset - pSymbol->Address,
1785 								xml_encode(GetFileName( moduleInfo.LoadedImageName )).c_str(),
1786 								xml_encode( GetFileDirectory( moduleInfo.LoadedImageName )).c_str()
1787 								);
1788 						else
1789 							fprintf( fout, "<errormail:StackInfo " \
1790 								"pos=\"%d\" ip=\"0x%p\" rel=\"0x%p\" name=\"%s\" path=\"%s\"/>\n",
1791 								frameNum,
1792 								frame.AddrPC.Offset,
1793 								frame.AddrPC.Offset - moduleInfo.BaseOfImage,
1794 								xml_encode(GetFileName( moduleInfo.LoadedImageName )).c_str(),
1795 								xml_encode(GetFileDirectory( moduleInfo.LoadedImageName )).c_str()
1796 								);
1797 					}
1798 					else
1799 						fprintf( fout, "<errormail:StackInfo pos=\"%d\" ip=\"0x%p\"/>\n",
1800 							frameNum,
1801 							frame.AddrPC.Offset
1802 							);
1803 
1804 					frameNum++;
1805 				}
1806 
1807 			} while ( bSuccess );
1808 
1809 			fprintf( fout, "</errormail:Stack>\n" );
1810 
1811 			SymCleanup( hProcess );
1812 
1813 			CloseHandle( hProcess );
1814 		}
1815 
1816 	}
1817 
1818 	return fSuccess;
1819 }
1820 
1821 bool WriteChecksumFile( FILE *fchksum, const hash_map< string, string >& rLibraries )
1822 {
1823 	bool success = false;
1824 
1825 	if ( fchksum && rLibraries.size() )
1826 	{
1827 		fprintf( fchksum, "<errormail:Checksums type=\"MD5\">\n" );
1828 
1829 		hash_map< string, string >::const_iterator iter;
1830 
1831 		for ( iter = rLibraries.begin();
1832 			iter != rLibraries.end();
1833 			iter++ )
1834 		{
1835 			sal_uInt8 checksum[RTL_DIGEST_LENGTH_MD5];
1836 			sal_uInt32 nBytesProcessed = calc_md5_checksum(
1837 				iter->second.c_str(),
1838 				checksum, sizeof(checksum) );
1839 
1840 			if ( nBytesProcessed )
1841 			{
1842 				fprintf( fchksum, "<errormail:Checksum sum=\"0x" );
1843 				for ( int i = 0; i < sizeof(checksum); fprintf( fchksum, "%02X", checksum[i++] ) );
1844 				fprintf( fchksum, "\" bytes=\"%d\" file=\"%s\"/>\n",
1845 					nBytesProcessed,
1846 					GetFileName( iter->first ).c_str() );
1847 			}
1848 		}
1849 
1850 		fprintf( fchksum, "</errormail:Checksums>\n" );
1851 
1852 		success = true;
1853 	}
1854 
1855 	return success;
1856 }
1857 
1858 //***************************************************************************
1859 
1860 BOOL FindDumpFile()
1861 {
1862 	TCHAR	szFileName[MAX_PATH];
1863 
1864 	if ( GetCrashDataPath( szFileName ) )
1865 	{
1866 		_tcscat( szFileName, _T("\\crashdat.dmp") );
1867 
1868 		HANDLE	hFile =  CreateFile(
1869 			szFileName,
1870 			GENERIC_READ,
1871 			0, NULL,
1872 			OPEN_EXISTING,
1873 			FILE_ATTRIBUTE_NORMAL, NULL );
1874 
1875 		if ( hFile )
1876 		{
1877 			CloseHandle( hFile );
1878 
1879 			WideCharToMultiByte( CP_ACP, 0, szFileName, -1, g_szDumpFileNameA, MAX_PATH, NULL, NULL );
1880 			_tcscpy( g_szDumpFileName, szFileName );
1881 
1882 			return TRUE;
1883 		}
1884 	}
1885 
1886 	return FALSE;
1887 }
1888 
1889 BOOL WriteDumpFile( DWORD dwProcessId, PEXCEPTION_POINTERS pExceptionPointers, DWORD dwThreadId )
1890 {
1891 	BOOL	fSuccess = FALSE;
1892 	PMINIDUMP_EXCEPTION_INFORMATION	lpExceptionParam = NULL;
1893 	MINIDUMP_EXCEPTION_INFORMATION	ExceptionParam;
1894 
1895 	HMODULE	hDbgHelp = LoadLibrary( _T("DBGHELP.DLL" ) );
1896 	MiniDumpWriteDump_PROC	pMiniDumpWriteDump = NULL;
1897 
1898 	if ( hDbgHelp )
1899 	{
1900 		pMiniDumpWriteDump = (MiniDumpWriteDump_PROC)GetProcAddress( hDbgHelp, "MiniDumpWriteDump" );
1901 
1902 		if ( !pMiniDumpWriteDump )
1903 		{
1904 			FreeLibrary( hDbgHelp );
1905 			return false;
1906 		}
1907 	}
1908 
1909 	if ( !pMiniDumpWriteDump )
1910 		return false;
1911 
1912 	HANDLE	hProcess = OpenProcess( PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, dwProcessId );
1913 
1914 	if ( IsValidHandle(hProcess) )
1915 	{
1916 		TCHAR	szTempPath[MAX_PATH];
1917 
1918 //		if ( GetTempPath( elementsof(szTempPath), szTempPath ) )
1919 		if ( GetCrashDataPath( szTempPath ) )
1920 		{
1921 			TCHAR	szFileName[MAX_PATH];
1922 
1923 //			if ( GetTempFileName( szTempPath, TEXT("DMP"), 0, szFileName ) )
1924 			_tcscpy( szFileName, szTempPath );
1925 			_tcscat( szFileName, _T("\\crashdat.dmp") );
1926 			{
1927 				HANDLE	hFile =  CreateFile(
1928 					szFileName,
1929 					GENERIC_READ | GENERIC_WRITE,
1930 					0, NULL,
1931 //					OPEN_EXISTING,
1932 					CREATE_ALWAYS,
1933 					FILE_ATTRIBUTE_NORMAL, NULL );
1934 
1935 				if ( hFile )
1936 				{
1937 					if ( pExceptionPointers && dwThreadId )
1938 					{
1939 						ExceptionParam.ThreadId = dwThreadId;
1940 						ExceptionParam.ExceptionPointers = pExceptionPointers;
1941 						ExceptionParam.ClientPointers = TRUE;
1942 
1943 						EXCEPTION_POINTERS	aExceptionPointers;
1944 						EXCEPTION_RECORD	aExceptionRecord;
1945 
1946 						ReadProcessMemory(
1947 							hProcess,
1948 							pExceptionPointers,
1949 							&aExceptionPointers,
1950 							sizeof(aExceptionPointers),
1951 							NULL );
1952 
1953 
1954 						ReadProcessMemory(
1955 							hProcess,
1956 							aExceptionPointers.ExceptionRecord,
1957 							&aExceptionRecord,
1958 							sizeof(aExceptionRecord),
1959 							NULL );
1960 
1961 						g_dwExceptionCode = aExceptionRecord.ExceptionCode;
1962 
1963 						lpExceptionParam = &ExceptionParam;
1964 					}
1965 
1966 					fSuccess = pMiniDumpWriteDump( hProcess, dwProcessId, hFile, MiniDumpNormal, lpExceptionParam, NULL, NULL );
1967 
1968 					CloseHandle( hFile );
1969 
1970 					WideCharToMultiByte( CP_ACP, 0, szFileName, -1, g_szDumpFileNameA, MAX_PATH, NULL, NULL );
1971 					_tcscpy( g_szDumpFileName, szFileName );
1972 				}
1973 
1974 				if ( !fSuccess )
1975 					DeleteFile( szFileName );
1976 			}
1977 		}
1978 
1979 		CloseHandle( hProcess );
1980 	}
1981 
1982 	FreeLibrary( hDbgHelp );
1983 
1984 	return fSuccess;
1985 }
1986 
1987 //***************************************************************************
1988 
1989 static DWORD FindProcessForImage( LPCTSTR lpImagePath )
1990 {
1991 	DWORD	dwProcessId = 0;
1992 	DWORD	aProcesses[1024];
1993 	DWORD	dwSize = 0;
1994 	TCHAR	szShortImagePath[MAX_PATH];
1995 
1996 	if ( GetShortPathName( lpImagePath, szShortImagePath, elementsof(szShortImagePath) ) &&
1997 		EnumProcesses( aProcesses, sizeof(aProcesses), &dwSize ) )
1998 	{
1999 		unsigned nProcesses = dwSize / sizeof(aProcesses[0]);
2000 
2001 		for ( unsigned i = 0; !dwProcessId && i < nProcesses; i++ )
2002 		{
2003 			HANDLE	hProcess = OpenProcess( PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, aProcesses[i] );
2004 
2005 			if ( IsValidHandle(hProcess) )
2006 			{
2007 				TCHAR	szModulePath[MAX_PATH+1];
2008 
2009 				if ( GetModuleFileNameEx( hProcess, NULL, szModulePath, MAX_PATH ) )
2010 				{
2011 					TCHAR	szShortModulePath[MAX_PATH];
2012 
2013 					if ( GetShortPathName( szModulePath, szShortModulePath, elementsof(szShortModulePath) ) )
2014 					{
2015 						if ( 0 == _tcsicmp( szShortModulePath, szShortImagePath ) )
2016 							dwProcessId = aProcesses[i];
2017 					}
2018 				}
2019 
2020 				CloseHandle( hProcess );
2021 			}
2022 		}
2023 	}
2024 
2025 	return dwProcessId;
2026 }
2027 //***************************************************************************
2028 
2029 static bool ParseCommandArgs( LPDWORD pdwProcessId, PEXCEPTION_POINTERS* ppException, LPDWORD pdwThreadId )
2030 {
2031 	int		argc = __argc;
2032 #ifdef __MINGW32__
2033 #ifdef _UNICODE
2034 	TCHAR	**argv = reinterpret_cast<TCHAR **>(alloca((argc+1)*sizeof(WCHAR*)));
2035 	int *sizes = reinterpret_cast<int *>(alloca(argc*sizeof(int)));
2036 	int argsize=0;
2037 	char **ptr;
2038 	int i;
2039 	ptr=__argv;
2040 	for (i = 0; i < argc; ++i)
2041 	{
2042 		sizes[i]=MultiByteToWideChar(CP_ACP, 0, *ptr, -1, NULL, 0);
2043 		argsize+=sizes[i]+1;
2044 		++ptr;
2045 	}
2046 	++argsize;
2047 	TCHAR	*args = reinterpret_cast<TCHAR *>(alloca(argsize*sizeof(WCHAR)));
2048 	ptr=__argv;
2049 	TCHAR *cptr=args;
2050 	for (i = 0; i < argc; ++i)
2051 	{
2052 		argv[i]=cptr;
2053 		MultiByteToWideChar( CP_ACP, 0, *ptr, -1, cptr, sizes[i] );
2054 		++ptr;
2055 		cptr+=sizes[i];
2056 		*cptr=0;
2057 		++cptr;
2058 	}
2059 	argv[i]=cptr;
2060 	*cptr=0;
2061 #else
2062 	TCHAR	**argv = __argv;
2063 #endif
2064 #else
2065 	TCHAR	**argv = __targv;
2066 #endif
2067 	bool	bSuccess = true;
2068 
2069 	for ( int argn = 1; bSuccess && argn < argc; argn++ )
2070 	{
2071 		if ( 0 == _tcsicmp( argv[argn], _T("-h") ) ||
2072 			 0 == _tcsicmp( argv[argn], _T("/h") ) ||
2073 			 0 == _tcsicmp( argv[argn], _T("-?") ) ||
2074 			 0 == _tcsicmp( argv[argn], _T("/?") ) ||
2075 			 0 == _tcsicmp( argv[argn], _T("/help") ) ||
2076 			 0 == _tcsicmp( argv[argn], _T("-help") ) ||
2077 			 0 == _tcsicmp( argv[argn], _T("--help") )
2078 			 )
2079 		{
2080 			HINSTANCE	hInstance = GetModuleHandle(NULL);
2081 			TCHAR	szUsage[FORMATBUFSIZE];
2082 			TCHAR	szProcess[FORMATBUFSIZE];
2083 			TCHAR	szProcessDescription[FORMATBUFSIZE];
2084 			TCHAR	szHelpDescription[FORMATBUFSIZE];
2085 
2086 			LoadAndFormatString( hInstance, IDS_MSG_CMDLINE_USAGE, szUsage, elementsof(szUsage) );
2087 			LoadAndFormatString( hInstance, IDS_MSG_PARAM_PROCESSID, szProcess, elementsof(szProcess) );
2088 			LoadAndFormatString( hInstance, IDS_MSG_PARAM_PROCESSID_DESCRIPTION, szProcessDescription, elementsof(szProcessDescription) );
2089 			LoadAndFormatString( hInstance, IDS_MSG_PARAM_HELP_DESCRIPTION, szHelpDescription, elementsof(szHelpDescription) );
2090 
2091 			_tprintf(
2092 				TEXT("\n%s: crashrep %s\n\n")
2093 				TEXT("/?, -h[elp]          %s\n\n")
2094 				TEXT("%-20s %s\n\n"),
2095 				szUsage, szProcess, szHelpDescription, szProcess, szProcessDescription
2096 				);
2097 
2098 			return true;
2099 		}
2100 		else if ( 0 == _tcsicmp( argv[argn], _T("-p") ) ||
2101 			 0 == _tcsicmp( argv[argn], _T("/p") ) )
2102 		{
2103 			if ( ++argn < argc )
2104 				*pdwProcessId = _tcstoul( argv[argn], NULL, 0 );
2105 			else
2106 				bSuccess = false;
2107 		}
2108 		else if ( 0 == _tcsicmp( argv[argn], _T("-excp") ) ||
2109 			      0 == _tcsicmp( argv[argn], _T("/excp") ) )
2110 		{
2111 			if ( ++argn < argc )
2112 				*ppException = (PEXCEPTION_POINTERS)_tcstoul( argv[argn], NULL, 0 );
2113 			else
2114 				bSuccess = false;
2115 		}
2116 		else if ( 0 == _tcsicmp( argv[argn], _T("-t") ) ||
2117 			      0 == _tcsicmp( argv[argn], _T("/t") ) )
2118 		{
2119 			if ( ++argn < argc )
2120 				*pdwThreadId = _tcstoul( argv[argn], NULL, 0 );
2121 			else
2122 				bSuccess = false;
2123 		}
2124 		else if ( 0 == _tcsicmp( argv[argn], _T("-noui") ) ||
2125 			      0 == _tcsicmp( argv[argn], _T("/noui") ) )
2126 		{
2127 			g_bNoUserInterface = true;
2128 		}
2129 		else if ( 0 == _tcsicmp( argv[argn], _T("-send") ) ||
2130 			      0 == _tcsicmp( argv[argn], _T("/send") ) )
2131 		{
2132 			g_bSendReport = true;
2133 		}
2134 		else if ( 0 == _tcsicmp( argv[argn], _T("-load") ) ||
2135 			      0 == _tcsicmp( argv[argn], _T("/load") ) )
2136 		{
2137 			g_bLoadReport = true;
2138 		}
2139 		else // treat parameter as image path
2140 		{
2141 			TCHAR	szImagePath[MAX_PATH];
2142 			LPTSTR	lpImageName;
2143 
2144 			if ( GetFullPathName( argv[argn], MAX_PATH, szImagePath, &lpImageName ) )
2145 			{
2146 				DWORD	dwProcessId = FindProcessForImage( szImagePath );
2147 
2148 				if ( dwProcessId )
2149 					*pdwProcessId = dwProcessId;
2150 				else
2151 					bSuccess = false;
2152 			}
2153 		}
2154 	}
2155 
2156 	if ( !*pdwProcessId && !g_bLoadReport )
2157 	{
2158 		TCHAR	szImagePath[MAX_PATH];
2159 		LPTSTR	lpImageName;
2160 
2161 		if ( GetFullPathName( TEXT("soffice.exe"), MAX_PATH, szImagePath, &lpImageName ) )
2162 		{
2163 			DWORD	dwProcessId = FindProcessForImage( szImagePath );
2164 
2165 			if ( dwProcessId )
2166 				*pdwProcessId = dwProcessId;
2167 			else
2168 				bSuccess = false;
2169 		}
2170 	}
2171 
2172 	return bSuccess;
2173 }
2174 
2175 //***************************************************************************
2176 
2177 BOOL WriteCommentFile( LPCTSTR lpComment )
2178 {
2179 	BOOL	fSuccess = FALSE;
2180 	TCHAR	szTempPath[MAX_PATH];
2181 
2182 	if ( GetTempPath( elementsof(szTempPath), szTempPath ) )
2183 	{
2184 		TCHAR	szFileName[MAX_PATH];
2185 
2186 		if ( GetTempFileName( szTempPath, TEXT("CMT"), 0, szFileName ) )
2187 		{
2188 			HANDLE	hFile =  CreateFile( szFileName, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
2189 
2190 			if ( hFile )
2191 			{
2192 				DWORD	dwBytesWritten;
2193 
2194 				int needed = WideCharToMultiByte( CP_UTF8, 0, lpComment, -1, NULL, 0, NULL, NULL );
2195 				if ( needed )
2196 				{
2197 					char *lpCommentUTF8 = (char *)alloca( needed );
2198 					WideCharToMultiByte( CP_UTF8, 0, lpComment, -1, lpCommentUTF8, needed, NULL, NULL );
2199 					fSuccess = WriteFile( hFile, lpCommentUTF8, strlen(lpCommentUTF8), &dwBytesWritten, NULL );
2200 				}
2201 				else
2202 					fSuccess = TRUE;
2203 
2204 
2205 				CloseHandle( hFile );
2206 
2207 				WideCharToMultiByte( CP_ACP, 0, szFileName, -1, g_szCommentFileNameA, MAX_PATH, NULL, NULL );
2208 			}
2209 
2210 			if ( !fSuccess )
2211 				DeleteFile( szFileName );
2212 		}
2213 	}
2214 
2215 	return fSuccess;
2216 }
2217 
2218 //***************************************************************************
2219 
2220 static int _tsetenv( const _TCHAR *lpVar, const _TCHAR *lpValue )
2221 {
2222 	if ( !lpValue )
2223 		lpValue = _T("");
2224 
2225 	_TCHAR	*envstr = (TCHAR *)alloca( (_tcslen( lpVar ) + _tcslen( lpValue ) + 2) * sizeof(_TCHAR) );
2226 
2227 	_tcscpy( envstr, lpVar );
2228 	_tcscat( envstr, _T("=") );
2229 	_tcscat( envstr, lpValue );
2230 
2231 	return _tputenv( envstr );
2232 }
2233 
2234 static bool read_line( FILE *fp, string& rLine )
2235 {
2236 	char szBuffer[1024];
2237 	bool bSuccess = false;
2238 	bool bEOL = false;
2239 	string	line;
2240 
2241 
2242 	while ( !bEOL && fgets( szBuffer, sizeof(szBuffer), fp ) )
2243 	{
2244 		int	len = strlen(szBuffer);
2245 
2246 		bSuccess = true;
2247 
2248 		while ( len && szBuffer[len - 1] == '\n' )
2249 		{
2250 			szBuffer[--len] = 0;
2251 			bEOL = true;
2252 		}
2253 
2254 		line.append( szBuffer );
2255 	}
2256 
2257 	rLine = line;
2258 	return bSuccess;
2259 }
2260 
2261 static string get_script_string( const char *pFileName, const char *pKeyName )
2262 {
2263 	FILE	*fp = fopen( pFileName, "rt" );
2264 	string	retValue;
2265 
2266 	if ( fp )
2267 	{
2268 		string line;
2269 		string section;
2270 
2271 		while ( read_line( fp, line ) )
2272 		{
2273 			line = trim_string( line );
2274 
2275 
2276 			string::size_type iEqualSign = line.find( '=', 0 );
2277 
2278 			if ( iEqualSign != string::npos )
2279 			{
2280 				string	keyname = line.substr( 0, iEqualSign );
2281 				keyname = trim_string( keyname );
2282 
2283 				string	value = line.substr( sal::static_int_cast<string::size_type>(iEqualSign + 1) );
2284 				value = trim_string( value );
2285 
2286 				if ( value.length() && '\"' == value[0] )
2287 				{
2288 					value.erase( 0, 1 );
2289 
2290 					string::size_type iQuotes = value.find( '"', 0 );
2291 
2292 					if ( iQuotes != string::npos )
2293 						value.erase( iQuotes );
2294 				}
2295 
2296 				if ( 0 == stricmp( keyname.c_str(), pKeyName ) )
2297 				{
2298 					retValue = value;
2299 					break;
2300 				}
2301 			}
2302 		}
2303 
2304 		fclose( fp );
2305 	}
2306 
2307 	return retValue;
2308 }
2309 
2310 static bool ReadBootstrapParams( CrashReportParams &rParams )
2311 {
2312 	TCHAR	szBuffer[256] = TEXT("");
2313 	TCHAR	szModuleName[MAX_PATH];
2314 	TCHAR	szModuleVersionName[MAX_PATH];
2315 	TCHAR	szDrive[_MAX_DRIVE];
2316 	TCHAR	szDir[_MAX_DIR];
2317 	TCHAR	szFName[_MAX_FNAME];
2318 	TCHAR	szExt[_MAX_EXT];
2319 	TCHAR	szReportServer[MAX_HOSTNAME];
2320 	TCHAR	szReportPort[256];
2321 	bool	bSuccess = false;
2322 
2323 	GetModuleFileName( NULL, szModuleName, MAX_PATH );
2324 	_tsplitpath( szModuleName, szDrive, szDir, szFName, szExt );
2325 	_tmakepath( szModuleName, szDrive, szDir, _T("bootstrap"), _T(".ini") );
2326 	_tmakepath( szModuleVersionName, szDrive, szDir, _T("version"), _T(".ini") );
2327 
2328 	if (
2329 		GetPrivateProfileString(
2330 		TEXT("Bootstrap"),
2331 		TEXT("ProductKey"),
2332 		TEXT("OpenOffice.org"),
2333 		szBuffer,
2334 		elementsof(szBuffer),
2335 		szModuleName )
2336 		)
2337 	{
2338 		TCHAR	*pVersion = _tcschr( szBuffer, ' ' );
2339 
2340 		g_wstrProductKey = szBuffer;
2341 
2342 		if ( pVersion )
2343 		{
2344 			*pVersion = 0;
2345 			pVersion++;
2346 		}
2347 		else
2348 			pVersion = TEXT("");
2349 
2350 		if ( !_tgetenv( _T("PRODUCTNAME") ) )
2351 		{
2352 			_tsetenv( TEXT("PRODUCTNAME"), szBuffer );
2353 		}
2354 		if ( !_tgetenv( _T("PRODUCTVERSION") ) )
2355 			_tsetenv( TEXT("PRODUCTVERSION"), pVersion );
2356 	}
2357 
2358 	GetPrivateProfileString(
2359 		TEXT("Version"),
2360 		TEXT("buildid"),
2361 		TEXT("unknown"),
2362 		g_szBuildId, elementsof(g_szBuildId),
2363 		szModuleVersionName );
2364 
2365 	g_strDefaultLanguage = get_script_string( "instdb.inf", "DefaultLanguage" );
2366 
2367 	if ( GetPrivateProfileString(
2368 		TEXT("ErrorReport"),
2369 		TEXT("ErrorReportPort"),
2370 		TEXT("80"),
2371 		szReportPort, elementsof(szReportPort),
2372 		szModuleName
2373 		) )
2374 	{
2375 		TCHAR *endptr = NULL;
2376 
2377 		unsigned short uReportPort = (unsigned short)_tcstoul( szReportPort, &endptr, 10 );
2378 		if ( uReportPort )
2379 			g_uReportPort = uReportPort;
2380 	}
2381 
2382 	if ( GetPrivateProfileString(
2383 		TEXT("ErrorReport"),
2384 		TEXT("ErrorReportServer"),
2385 		TEXT(""),
2386 		szReportServer, elementsof(szReportServer),
2387 		szModuleName
2388 		) )
2389 	{
2390 		bSuccess = 0 != WideCharToMultiByte( CP_ACP, 0, szReportServer, -1, g_szReportServerA, elementsof(g_szReportServerA), NULL, NULL );
2391 	}
2392 
2393 	LPCTSTR	lpEnvString;
2394 
2395 	if ( 0 != (lpEnvString = _tgetenv( _T("ERRORREPORT_PROXYSERVER") )) )
2396 		rParams.sProxyServer = lpEnvString;
2397 
2398 	if ( 0 != (lpEnvString = _tgetenv( _T("ERRORREPORT_PROXYPORT") )) )
2399 		rParams.sProxyPort = lpEnvString;
2400 
2401 	if ( 0 != (lpEnvString = _tgetenv( _T("ERRORREPORT_SENDERADDRESS") )) )
2402 		rParams.sEmail = lpEnvString;
2403 
2404 	return bSuccess;
2405 }
2406 
2407 //***************************************************************************
2408 
2409 bool SendHTTPRequest(
2410 				FILE *fp,
2411 				const char *pszServer,
2412 				unsigned short uPort = 80,
2413 				const char *pszProxyServer = NULL,
2414 				unsigned short uProxyPort = 8080 )
2415 {
2416 	bool success = false;
2417 
2418 	struct hostent *hp;
2419 
2420 	if ( pszProxyServer )
2421 		hp = gethostbyname( pszProxyServer );
2422 	else
2423 		hp = gethostbyname( pszServer );
2424 
2425 	if ( hp )
2426 	{
2427 		SOCKET	s = socket( AF_INET, SOCK_STREAM, 0 );
2428 
2429 		if ( s )
2430 		{
2431 			struct sockaddr_in address;
2432 
2433 			memcpy(&(address.sin_addr.s_addr), *(hp->h_addr_list),sizeof(struct in_addr));
2434 			address.sin_family = AF_INET;
2435 
2436 			if ( pszProxyServer )
2437 				address.sin_port = ntohs( uProxyPort );
2438 			else
2439 				address.sin_port = ntohs( uPort );
2440 
2441 			if ( 0 == connect( s, (struct sockaddr *)&address, sizeof(struct sockaddr_in)) )
2442 			{
2443 				fseek( fp, 0, SEEK_END );
2444 				size_t length = ftell( fp );
2445 				fseek( fp, 0, SEEK_SET );
2446 
2447 				char buffer[2048];
2448 
2449 				if ( pszProxyServer )
2450 					sprintf( buffer,
2451 					"POST http://%s:%d/soap/servlet/rpcrouter HTTP/1.0\r\n"
2452 						"Content-Type: text/xml; charset=\"utf-8\"\r\n"
2453 						"Content-Length: %d\r\n"
2454 						"SOAPAction: \"\"\r\n\r\n",
2455 						pszServer,
2456 						uPort,
2457 						length
2458 						);
2459 				else
2460 					sprintf( buffer,
2461 						"POST /soap/servlet/rpcrouter HTTP/1.0\r\n"
2462 						"Content-Type: text/xml; charset=\"utf-8\"\r\n"
2463 						"Content-Length: %d\r\n"
2464 						"SOAPAction: \"\"\r\n\r\n",
2465 						length
2466 						);
2467 
2468 				if ( SOCKET_ERROR != send( s, buffer, strlen(buffer), 0 ) )
2469 				{
2470 					size_t nBytes;
2471 
2472 					do
2473 					{
2474 						nBytes = fread( buffer, 1, sizeof(buffer), fp );
2475 
2476 						if ( nBytes )
2477 							success = SOCKET_ERROR != send( s, buffer, nBytes, 0 );
2478 					} while( nBytes && success );
2479 
2480 					if ( success )
2481 					{
2482 						memset( buffer, 0, sizeof(buffer) );
2483 						success = SOCKET_ERROR != recv( s, buffer, sizeof(buffer), 0 );
2484 						if ( success )
2485 						{
2486 							char szHTTPSignature[sizeof(buffer)] = "";
2487 							unsigned uHTTPReturnCode = 0;
2488 
2489 							sscanf( buffer, "%s %d ", szHTTPSignature, &uHTTPReturnCode );
2490 							success = uHTTPReturnCode == 200;
2491 						}
2492 					}
2493 				}
2494 
2495 			}
2496 
2497 			closesocket( s );
2498 		}
2499 	}
2500 
2501 	return success;
2502 }
2503 
2504 //***************************************************************************
2505 
2506 static void WriteSOAPRequest( FILE *fp )
2507 {
2508 	fprintf( fp,
2509 		"<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
2510 		"<SOAP-ENV:Envelope xmlns:SOAP-ENV=\"http://schemas.xmlsoap.org/soap/envelope/\"\n"
2511 		"xmlns:SOAP-ENC=\"http://schemas.xmlsoap.org/soap/encoding/\"\n"
2512 		"xmlns:xsi=\"http://www.w3.org/1999/XMLSchema-instance\"\n"
2513 		"xmlns:xsd=\"http://www.w3.org/1999/XMLSchema\"\n"
2514 		"xmlns:rds=\"urn:ReportDataService\"\n"
2515 		"xmlns:apache=\"http://xml.apache.org/xml-soap\"\n"
2516 		"SOAP-ENV:encodingStyle=\"http://schemas.xmlsoap.org/soap/encoding/\">\n"
2517 		"<SOAP-ENV:Body>\n"
2518 		);
2519 
2520 	fprintf( fp, "<rds:submitReport>\n"  );
2521 	fprintf( fp, "<body xsi:type=\"xsd:string\">This is an autogenerated crash report mail.</body>\n" );
2522 	fprintf( fp, "<hash xsi:type=\"apache:Map\">\n" );
2523 
2524 	FILE	*fpin = fopen( g_szReportFileNameA, "r" );
2525 	if ( fpin )
2526 	{
2527 		fprintf( fp,
2528 			"<item>\n"
2529 			"<key xsi:type=\"xsd:string\">reportmail.xml</key>\n"
2530 			"<value xsi:type=\"xsd:string\"><![CDATA[" );
2531 		fcopy( fpin, fp );
2532 		fprintf( fp, "]]></value></item>\n" );
2533 		fclose( fpin );
2534 	}
2535 
2536 	fpin = fopen( g_szCommentFileNameA, "r" );
2537 	if ( fpin )
2538 	{
2539 		fprintf( fp,
2540 			"<item>\n"
2541 			"<key xsi:type=\"xsd:string\">description.txt</key>\n"
2542 			"<value xsi:type=\"xsd:string\"><![CDATA[" );
2543 		fcopy( fpin, fp );
2544 		fprintf( fp, "]]></value></item>\n" );
2545 		fclose( fpin );
2546 	};
2547 
2548 
2549 	fpin = fopen( g_szDumpFileNameA, "rb" );
2550 	if ( fpin )
2551 	{
2552 		FILE *fptemp = _tmpfile();
2553 
2554 		if ( fptemp )
2555 		{
2556 			if ( base64_encode( fpin, fptemp ) )
2557 			{
2558 				fseek( fptemp, 0, SEEK_SET );
2559 				fprintf( fp,
2560 					"<item>\n"
2561 					"<key xsi:type=\"xsd:string\">user.dmp</key>\n"
2562 					"<value xsi:type=\"xsd:string\">" );
2563 				fcopy( fptemp, fp );
2564 				fprintf( fp, "</value></item>\n" );
2565 			}
2566 			fclose( fptemp );
2567 		}
2568 		fclose( fpin );
2569 	}
2570 
2571 	fprintf( fp,
2572 		"</hash>\n"
2573 		"</rds:submitReport>\n"
2574 		"</SOAP-ENV:Body>\n"
2575 		"</SOAP-ENV:Envelope>\n"
2576 		);
2577 }
2578 
2579 //***************************************************************************
2580 
2581 struct RequestParams
2582 {
2583 	bool	success;
2584 	FILE	*fpin;
2585 	const char *lpServer;
2586 	unsigned short uPort;
2587 	const char *lpProxyServer;
2588 	unsigned short uProxyPort;
2589 	HWND	hwndStatus;
2590 };
2591 
2592 void _cdecl SendingThread( void *lpArgs )
2593 {
2594 	RequestParams *pParams = (RequestParams *)lpArgs;
2595 
2596     pParams->success = SendHTTPRequest( pParams->fpin, pParams->lpServer, pParams->uPort, pParams->lpProxyServer, pParams->uProxyPort );
2597 
2598 	PostMessage( pParams->hwndStatus, WM_COMMAND, IDOK, 0 );
2599 }
2600 
2601 //***************************************************************************
2602 
2603 BOOL CALLBACK SendingStatusDialogProc(
2604 	HWND hwndDlg,
2605 	UINT uMsg,
2606 	WPARAM wParam,
2607 	LPARAM lParam
2608 	)
2609 {
2610 	static RequestParams *pRequest = NULL;
2611 	static HANDLE hSendingThread = NULL;
2612 
2613 	switch ( uMsg )
2614 	{
2615 	case WM_INITDIALOG:
2616 		{
2617 			TCHAR	szBuffer[1024] = TEXT("");
2618 			HINSTANCE	hInstance = (HINSTANCE)GetWindowLong( hwndDlg, GWL_HINSTANCE );
2619 			//HWND	hwndParent = (HWND)GetWindowLong( hwndDlg, GWL_HWNDPARENT );
2620 
2621 			pRequest = (RequestParams *)lParam;
2622 
2623 			LoadAndFormatString( hInstance, IDS_SENDING_REPORT_HEADER, szBuffer, elementsof(szBuffer) );
2624 			SetWindowText( hwndDlg, szBuffer );
2625 
2626 			LoadAndFormatString( hInstance, IDS_SENDING_REPORT_STATUS, szBuffer, elementsof(szBuffer) );
2627 			Static_SetText( GetDlgItem(hwndDlg, IDC_SENDING_REPORT_STATUS), szBuffer );
2628 
2629 			LoadAndFormatString( hInstance, IDS_CANCEL_BUTTON, szBuffer, elementsof(szBuffer) );
2630 			Button_SetText( GetDlgItem(hwndDlg, IDCANCEL), szBuffer );
2631 
2632 			pRequest->hwndStatus = hwndDlg;
2633 
2634 			hSendingThread = (HANDLE)_beginthread( SendingThread, 0, pRequest );
2635 		}
2636 		return TRUE;
2637 	case WM_COMMAND:
2638 		switch ( LOWORD(wParam) )
2639 		{
2640 		case IDCANCEL:
2641 			TerminateThread( hSendingThread, 0 );
2642 		case IDOK:
2643 			WaitForSingleObject( hSendingThread, INFINITE );
2644 			CloseHandle( hSendingThread );
2645 			EndDialog( hwndDlg, wParam );
2646 			return TRUE;
2647 		}
2648 		break;
2649 	default:
2650 		break;
2651 	}
2652 
2653 	return FALSE;
2654 }
2655 
2656 //***************************************************************************
2657 
2658 bool SendCrashReport( HWND hwndParent, const CrashReportParams &rParams )
2659 {
2660 	bool success = false;
2661 	char szProxyServer[1024] = "";
2662 	unsigned short uProxyPort = 8080;
2663 	TCHAR *endptr = NULL;
2664 
2665 	switch ( rParams.uInternetConnection )
2666 	{
2667 	case 2:
2668 		{
2669 			WideCharToMultiByte(
2670 				CP_ACP, 0, rParams.sProxyServer.c_str(), -1,
2671 				szProxyServer, sizeof(szProxyServer), NULL, NULL );
2672 			uProxyPort = (unsigned short)_tcstoul( rParams.sProxyPort.c_str(), &endptr, 10 );
2673 		}
2674 		break;
2675 	case 0:
2676 		{
2677 			DWORD	dwProxyEnable = 0;
2678 
2679 			RegReadValue( HKEY_CURRENT_USER,
2680 				TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings"),
2681 				TEXT("ProxyEnable"),
2682 				&dwProxyEnable,
2683 				sizeof(dwProxyEnable) );
2684 
2685 			if ( dwProxyEnable )
2686 			{
2687 				TCHAR	tszProxyServers[1024] = TEXT("");
2688 
2689 				if ( ERROR_SUCCESS == RegReadValue( HKEY_CURRENT_USER,
2690 					TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings"),				TEXT("ProxyServer"),
2691 					tszProxyServers,
2692 					sizeof(tszProxyServers) ) )
2693 				{
2694 					TCHAR *lpHttpStart = _tcsstr( tszProxyServers, TEXT("http=") );
2695 
2696 					if ( lpHttpStart )
2697 						lpHttpStart += 5;
2698 					else
2699 						lpHttpStart = tszProxyServers;
2700 
2701 					TCHAR *lpHttpEnd = _tcschr( lpHttpStart, ';' );
2702 
2703 					if ( lpHttpEnd )
2704 						*lpHttpEnd = 0;
2705 
2706 					char	szHTTPProxyServer[1024] = "";
2707 					WideCharToMultiByte( CP_ACP, 0, lpHttpStart, -1, szHTTPProxyServer, sizeof(szHTTPProxyServer), NULL, NULL );
2708 
2709 					char *lpColon = strchr( szHTTPProxyServer, ':' );
2710 
2711 					if ( lpColon )
2712 					{
2713 						char *endptr = NULL;
2714 
2715 						*lpColon = 0;
2716 						uProxyPort = (unsigned short)strtoul( lpColon + 1, &endptr, 10 );
2717 					}
2718 					else
2719 						uProxyPort = 8080;
2720 
2721 					strcpy( szProxyServer, szHTTPProxyServer );
2722 
2723 				}
2724 			}
2725 		}
2726 		break;
2727 	default:
2728 	case 1:
2729 		break;
2730 	}
2731 
2732 	FILE	*fptemp = _tmpfile();
2733 	if ( fptemp )
2734 	{
2735 		RequestParams	request;
2736 
2737 		request.success = false;
2738 		request.fpin = fptemp;
2739 		request.lpServer = REPORT_SERVER;
2740 		request.uPort = REPORT_PORT;
2741 		request.lpProxyServer = szProxyServer[0] ? szProxyServer : NULL;
2742 		request.uProxyPort = uProxyPort;
2743 		request.hwndStatus = NULL;
2744 
2745 		WriteSOAPRequest( fptemp );
2746 		fseek( fptemp, 0, SEEK_SET );
2747 
2748 		if ( hwndParent )
2749 		{
2750 			int retid = DialogBoxParam(
2751 				GetModuleHandle(NULL),
2752 				MAKEINTRESOURCE(IDD_SENDING_STATUS),
2753 				hwndParent,
2754 				SendingStatusDialogProc,
2755 				(LPARAM)&request
2756 				);
2757 
2758 			success = request.success;
2759 
2760 			if ( IDOK == retid )
2761 			{
2762 				if ( !success )
2763 				{
2764 					TCHAR	szMessage[1024];
2765 
2766 					LoadAndFormatString( GetModuleHandle(NULL), IDS_ERROR_MSG_PROXY, szMessage, elementsof(szMessage) );
2767 
2768 					MessageBox( hwndParent, szMessage, NULL, MB_ICONERROR | MB_OK );
2769 				}
2770 				else
2771 				{
2772 					TCHAR	szMessage[1024];
2773 					TCHAR	szTitle[1024];
2774 
2775 					LoadAndFormatString( GetModuleHandle(NULL), IDS_SENDING_REPORT_STATUS_FINISHED, szMessage, elementsof(szMessage) );
2776 					LoadAndFormatString( GetModuleHandle(NULL), IDS_SENDING_REPORT_HEADER, szTitle, elementsof(szTitle) );
2777 
2778 					MessageBox( hwndParent, szMessage, szTitle, MB_ICONINFORMATION | MB_OK );
2779 				}
2780 			}
2781 
2782 		}
2783 		else
2784 		{
2785 			HANDLE hSendingThread = (HANDLE)_beginthread( SendingThread, 0, (void *)&request );
2786 
2787 			WaitForSingleObject( hSendingThread, INFINITE );
2788 
2789 			success = request.success;
2790 			if ( !success )
2791 			{
2792 				TCHAR	szMessage[1024];
2793 
2794 				LoadAndFormatString( GetModuleHandle(NULL), IDS_ERROR_MSG_PROXY, szMessage, elementsof(szMessage) );
2795 				_ftprintf( stderr, _T("ERROR: %s\n"), szMessage );
2796 			}
2797 			else
2798 			{
2799 				TCHAR	szMessage[1024];
2800 
2801 				LoadAndFormatString( GetModuleHandle(NULL), IDS_SENDING_REPORT_STATUS_FINISHED, szMessage, elementsof(szMessage) );
2802 
2803 				_ftprintf( stderr, _T("SUCCESS: %s\n"), szMessage );
2804 			}
2805 		}
2806 		fclose( fptemp );
2807 	}
2808 	else
2809 	{
2810 		TCHAR	szMessage[1024];
2811 
2812 		LoadAndFormatString( GetModuleHandle(NULL), IDS_ERROR_MSG_DISK_FULL, szMessage, elementsof(szMessage) );
2813 
2814 		if ( hwndParent )
2815 			MessageBox( hwndParent, szMessage, NULL, MB_ICONERROR | MB_OK );
2816 		else
2817 			_ftprintf( stderr, _T("ERROR: %s\n"), szMessage );
2818 	}
2819 
2820 	return success;
2821 }
2822 
2823 //***************************************************************************
2824 
2825 #ifdef __MINGW32__
2826 int WINAPI WinMain( HINSTANCE hInstance, HINSTANCE, LPSTR /*lpCmdLine*/, int )
2827 #else
2828 int WINAPI _tWinMain( HINSTANCE hInstance, HINSTANCE, LPTSTR /*lpCmdLine*/, int )
2829 #endif
2830 {
2831 	int exitcode = -1;
2832 	int argc = __argc;
2833 
2834 #ifdef __MINGW32__
2835 	char **argv = __argv;
2836 #else
2837 #ifdef _UNICODE
2838 	char **argv = new char *[argc + 1];
2839 
2840 	for ( int argn = 0; argn < argc; argn++ )
2841 	{
2842 		int nBytes = WideCharToMultiByte( CP_ACP, 0, __targv[argn], -1, NULL, 0, NULL, NULL );
2843 		argv[argn] = new char[nBytes];
2844 		WideCharToMultiByte( CP_ACP, 0, __targv[argn], -1, argv[argn], nBytes, NULL, NULL );
2845 	}
2846 	argv[argc] = NULL;
2847 #else
2848 	char **argv = __targv;
2849 #endif
2850 #endif
2851 
2852 	osl_setCommandArgs( argc, argv );
2853 
2854 	PEXCEPTION_POINTERS	pExceptionPointers = NULL;
2855 	DWORD				dwProcessId = 0;
2856 	DWORD				dwThreadId = 0;
2857 
2858 	WSADATA wsaData;
2859 	WORD    wVersionRequested;
2860 
2861 	wVersionRequested = MAKEWORD(1, 1);
2862 	WSAStartup(wVersionRequested, &wsaData);
2863 
2864 	CrashReportParams	Params;
2865 
2866 	Params.ReadFromRegistry();
2867 	Params.ReadFromEnvironment();
2868 
2869 	if ( ReadBootstrapParams( Params ) &&
2870 		ParseCommandArgs( &dwProcessId, &pExceptionPointers, &dwThreadId ) )
2871 	{
2872 		bool	bGotDumpFile;
2873 
2874 		if ( g_bLoadReport )
2875 			bGotDumpFile = FindDumpFile();
2876 		else
2877 			bGotDumpFile = WriteDumpFile( dwProcessId, pExceptionPointers, dwThreadId );
2878 
2879 		if( bGotDumpFile )
2880 		{
2881 			hash_map< string, string > aLibraries;
2882 
2883 			if ( g_bLoadReport )
2884 			{
2885 				g_fpStackFile = _open_reportfile( _T(".stk"), _T("rb") );
2886 				g_fpChecksumFile = _open_reportfile( _T(".chk"), _T("rb") );
2887 			}
2888 			else
2889 			{
2890 				if ( g_bSendReport )
2891 				{
2892 					g_fpStackFile = _tmpfile();
2893 					g_fpChecksumFile = _tmpfile();
2894 				}
2895 				else
2896 				{
2897 					g_fpStackFile = _open_reportfile( _T(".stk"), _T("w+b") );
2898 					g_fpChecksumFile = _open_reportfile( _T(".chk"), _T("w+b") );
2899 
2900 					FILE	*fpUnsent = _open_reportfile( _T(".lck"), _T("w+b") );
2901 					if ( fpUnsent )
2902 					{
2903 						fprintf( fpUnsent, "Unsent\r\n" );
2904 						fclose( fpUnsent );
2905 					}
2906 				}
2907 
2908 				WriteStackFile( g_fpStackFile, aLibraries, dwProcessId, pExceptionPointers );
2909 				WriteChecksumFile( g_fpChecksumFile, aLibraries );
2910 				WriteReportFile( &Params );
2911 
2912 				FILE	*fpPreview = _open_reportfile( _T(".prv"), _T("w+b") );
2913 
2914 				if ( fpPreview )
2915 				{
2916 					FILE	 *fp = fopen( g_szReportFileNameA, "rb" );
2917 					if ( fp )
2918 					{
2919 						fcopy( fp, fpPreview );
2920 						fclose( fp );
2921 					}
2922 					fclose( fpPreview );
2923 				}
2924 			}
2925 
2926 			if ( g_bSendReport )
2927 			{
2928 				InitCommonControls();
2929 
2930 				// Actually this should never be true anymore
2931 				if ( !g_bNoUserInterface && InitRichEdit() )
2932 				{
2933 
2934 					INT_PTR result = DialogBoxParam( hInstance, MAKEINTRESOURCE(IDD_DIALOG_FRAME), NULL, DialogProc, (LPARAM)&Params );
2935 
2936 					if ( result > 0 )
2937 					{
2938 						exitcode = 0;
2939 					}
2940 					DeinitRichEdit();
2941 				}
2942 				else
2943 				{
2944 					WriteCommentFile( Params.sComment.c_str() );
2945 					WriteReportFile( &Params );
2946 					if ( SendCrashReport( NULL, Params ) )
2947 						exitcode = 0;
2948 				}
2949 
2950 
2951 				if ( g_szReportFileNameA[0] )
2952 					DeleteFileA( g_szReportFileNameA );
2953 
2954 				if ( g_szCommentFileNameA[0] )
2955 					DeleteFileA( g_szCommentFileNameA );
2956 			}
2957 			else
2958 			{
2959 				if ( g_szReportFileNameA[0] )
2960 					DeleteFileA( g_szReportFileNameA );
2961 				exitcode = 0;
2962 			}
2963 
2964 			if ( g_szDumpFileNameA[0] && g_bSendReport )
2965 					DeleteFileA( g_szDumpFileNameA );
2966 
2967 			if ( g_fpStackFile )
2968 				fclose( g_fpStackFile );
2969 
2970 			if ( g_fpChecksumFile )
2971 				fclose( g_fpChecksumFile );
2972 		}
2973 	}
2974 
2975 
2976 	return exitcode;
2977 }
2978 
2979