diff options
Diffstat (limited to 'linden/indra/newview/llappviewerwin32.cpp')
-rw-r--r-- | linden/indra/newview/llappviewerwin32.cpp | 461 |
1 files changed, 461 insertions, 0 deletions
diff --git a/linden/indra/newview/llappviewerwin32.cpp b/linden/indra/newview/llappviewerwin32.cpp new file mode 100644 index 0000000..814c209 --- /dev/null +++ b/linden/indra/newview/llappviewerwin32.cpp | |||
@@ -0,0 +1,461 @@ | |||
1 | /** | ||
2 | * @file llappviewerwin32.cpp | ||
3 | * @brief The LLAppViewerWin32 class definitions | ||
4 | * | ||
5 | * $LicenseInfo:firstyear=2007&license=viewergpl$ | ||
6 | * | ||
7 | * Copyright (c) 2007, Linden Research, Inc. | ||
8 | * | ||
9 | * Second Life Viewer Source Code | ||
10 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
11 | * to you under the terms of the GNU General Public License, version 2.0 | ||
12 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
13 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
14 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
15 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
16 | * | ||
17 | * There are special exceptions to the terms and conditions of the GPL as | ||
18 | * it is applied to this Source Code. View the full text of the exception | ||
19 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
20 | * online at http://secondlife.com/developers/opensource/flossexception | ||
21 | * | ||
22 | * By copying, modifying or distributing this software, you acknowledge | ||
23 | * that you have read and understood your obligations described above, | ||
24 | * and agree to abide by those obligations. | ||
25 | * | ||
26 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
27 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
28 | * COMPLETENESS OR PERFORMANCE. | ||
29 | * $/LicenseInfo$ | ||
30 | */ | ||
31 | |||
32 | #include "llviewerprecompiledheaders.h" | ||
33 | |||
34 | #include "llappviewerwin32.h" | ||
35 | |||
36 | #include "llmemtype.h" | ||
37 | |||
38 | #include "llwindowwin32.cpp" // *FIX: for setting gIconResource. | ||
39 | #include "res/resource.h" // *FIX: for setting gIconResource. | ||
40 | |||
41 | #include <fcntl.h> //_O_APPEND | ||
42 | #include <io.h> //_open_osfhandle() | ||
43 | #include <errorrep.h> // for AddERExcludedApplicationA() | ||
44 | #include <process.h> // _spawnl() | ||
45 | #include <tchar.h> // For TCHAR support | ||
46 | |||
47 | #include "llviewercontrol.h" | ||
48 | #include "lldxhardware.h" | ||
49 | |||
50 | #include "llweb.h" | ||
51 | #include "llsecondlifeurls.h" | ||
52 | |||
53 | #include "llwindebug.h" | ||
54 | |||
55 | #include "llviewernetwork.h" | ||
56 | #include "llmd5.h" | ||
57 | |||
58 | void fill_args(int& argc, char** argv, const S32 max_args, LPSTR cmd_line) | ||
59 | { | ||
60 | char *token = NULL; | ||
61 | if( cmd_line[0] == '\"' ) | ||
62 | { | ||
63 | // Exe name is enclosed in quotes | ||
64 | token = strtok( cmd_line, "\"" ); | ||
65 | argv[argc++] = token; | ||
66 | token = strtok( NULL, " \t," ); | ||
67 | } | ||
68 | else | ||
69 | { | ||
70 | // Exe name is not enclosed in quotes | ||
71 | token = strtok( cmd_line, " \t," ); | ||
72 | } | ||
73 | |||
74 | while( (token != NULL) && (argc < max_args) ) | ||
75 | { | ||
76 | argv[argc++] = token; | ||
77 | /* Get next token: */ | ||
78 | if (*(token + strlen(token) + 1) == '\"') /* Flawfinder: ignore*/ | ||
79 | { | ||
80 | token = strtok( NULL, "\""); | ||
81 | } | ||
82 | else | ||
83 | { | ||
84 | token = strtok( NULL, " \t," ); | ||
85 | } | ||
86 | } | ||
87 | } | ||
88 | |||
89 | // *NOTE:Mani - this code is stolen from LLApp, where its never actually used. | ||
90 | LONG WINAPI viewer_windows_exception_handler(struct _EXCEPTION_POINTERS *exception_infop) | ||
91 | { | ||
92 | // Translate the signals/exceptions into cross-platform stuff | ||
93 | // Windows implementation | ||
94 | llinfos << "Entering Windows Exception Handler..." << llendl; | ||
95 | |||
96 | // Make sure the user sees something to indicate that the app crashed. | ||
97 | LONG retval; | ||
98 | |||
99 | if (LLApp::isError()) | ||
100 | { | ||
101 | llwarns << "Got another fatal signal while in the error handler, die now!" << llendl; | ||
102 | retval = EXCEPTION_EXECUTE_HANDLER; | ||
103 | return retval; | ||
104 | } | ||
105 | |||
106 | // Generate a minidump if we can. | ||
107 | // Before we wake the error thread... | ||
108 | // Which will start the crash reporting. | ||
109 | LLWinDebug::handleException(exception_infop); | ||
110 | |||
111 | // Flag status to error, so thread_error starts its work | ||
112 | LLApp::setError(); | ||
113 | |||
114 | // Block in the exception handler until the app has stopped | ||
115 | // This is pretty sketchy, but appears to work just fine | ||
116 | while (!LLApp::isStopped()) | ||
117 | { | ||
118 | ms_sleep(10); | ||
119 | } | ||
120 | |||
121 | // | ||
122 | // At this point, we always want to exit the app. There's no graceful | ||
123 | // recovery for an unhandled exception. | ||
124 | // | ||
125 | // Just kill the process. | ||
126 | retval = EXCEPTION_EXECUTE_HANDLER; | ||
127 | return retval; | ||
128 | } | ||
129 | |||
130 | int APIENTRY WinMain(HINSTANCE hInstance, | ||
131 | HINSTANCE hPrevInstance, | ||
132 | LPSTR lpCmdLine, | ||
133 | int nCmdShow) | ||
134 | { | ||
135 | LLMemType mt1(LLMemType::MTYPE_STARTUP); | ||
136 | |||
137 | // *FIX: global | ||
138 | gIconResource = MAKEINTRESOURCE(IDI_LL_ICON); | ||
139 | |||
140 | // In Win32, we need to generate argc and argv ourselves... | ||
141 | // Note: GetCommandLine() returns a potentially return a LPTSTR | ||
142 | // which can resolve to a LPWSTR (unicode string). | ||
143 | // (That's why it's different from lpCmdLine which is a LPSTR.) | ||
144 | // We don't currently do unicode, so call the non-unicode version | ||
145 | // directly. | ||
146 | LPSTR cmd_line_including_exe_name = GetCommandLineA(); | ||
147 | |||
148 | const S32 MAX_ARGS = 100; | ||
149 | int argc = 0; | ||
150 | char* argv[MAX_ARGS]; /* Flawfinder: ignore */ | ||
151 | |||
152 | fill_args(argc, argv, MAX_ARGS, cmd_line_including_exe_name); | ||
153 | |||
154 | LLAppViewerWin32* viewer_app_ptr = new LLAppViewerWin32(); | ||
155 | |||
156 | // *FIX:Mani This method is poorly named, since the exception | ||
157 | // is now handled by LLApp. | ||
158 | bool ok = LLWinDebug::setupExceptionHandler(); | ||
159 | |||
160 | // Actually here's the exception setup. | ||
161 | LPTOP_LEVEL_EXCEPTION_FILTER prev_filter; | ||
162 | prev_filter = SetUnhandledExceptionFilter(viewer_windows_exception_handler); | ||
163 | if (!prev_filter) | ||
164 | { | ||
165 | llwarns << "Our exception handler (" << (void *)LLWinDebug::handleException << ") replaced with NULL!" << llendl; | ||
166 | ok = FALSE; | ||
167 | } | ||
168 | if (prev_filter != LLWinDebug::handleException) | ||
169 | { | ||
170 | llwarns << "Our exception handler (" << (void *)LLWinDebug::handleException << ") replaced with " << prev_filter << "!" << llendl; | ||
171 | ok = FALSE; | ||
172 | } | ||
173 | |||
174 | viewer_app_ptr->setErrorHandler(LLAppViewer::handleViewerCrash); | ||
175 | |||
176 | ok = viewer_app_ptr->tempStoreCommandOptions(argc, argv); | ||
177 | if(!ok) | ||
178 | { | ||
179 | llwarns << "Unable to parse command line." << llendl; | ||
180 | return -1; | ||
181 | } | ||
182 | |||
183 | ok = viewer_app_ptr->init(); | ||
184 | if(!ok) | ||
185 | { | ||
186 | llwarns << "Application init failed." << llendl; | ||
187 | return -1; | ||
188 | } | ||
189 | |||
190 | // Run the application main loop | ||
191 | if(!LLApp::isQuitting()) | ||
192 | { | ||
193 | viewer_app_ptr->mainLoop(); | ||
194 | } | ||
195 | |||
196 | if (!LLApp::isError()) | ||
197 | { | ||
198 | // | ||
199 | // We don't want to do cleanup here if the error handler got called - | ||
200 | // the assumption is that the error handler is responsible for doing | ||
201 | // app cleanup if there was a problem. | ||
202 | // | ||
203 | viewer_app_ptr->cleanup(); | ||
204 | } | ||
205 | delete viewer_app_ptr; | ||
206 | viewer_app_ptr = NULL; | ||
207 | return 0; | ||
208 | } | ||
209 | |||
210 | void LLAppViewerWin32::disableWinErrorReporting() | ||
211 | { | ||
212 | const char win_xp_string[] = "Microsoft Windows XP"; | ||
213 | BOOL is_win_xp = ( getOSInfo().getOSString().substr(0, strlen(win_xp_string) ) == win_xp_string ); /* Flawfinder: ignore*/ | ||
214 | if( is_win_xp ) | ||
215 | { | ||
216 | // Note: we need to use run-time dynamic linking, because load-time dynamic linking will fail | ||
217 | // on systems that don't have the library installed (all non-Windows XP systems) | ||
218 | HINSTANCE fault_rep_dll_handle = LoadLibrary(L"faultrep.dll"); /* Flawfinder: ignore */ | ||
219 | if( fault_rep_dll_handle ) | ||
220 | { | ||
221 | pfn_ADDEREXCLUDEDAPPLICATIONA pAddERExcludedApplicationA = (pfn_ADDEREXCLUDEDAPPLICATIONA) GetProcAddress(fault_rep_dll_handle, "AddERExcludedApplicationA"); | ||
222 | if( pAddERExcludedApplicationA ) | ||
223 | { | ||
224 | |||
225 | // Strip the path off the name | ||
226 | const char* executable_name = gDirUtilp->getExecutableFilename().c_str(); | ||
227 | |||
228 | if( 0 == pAddERExcludedApplicationA( executable_name ) ) | ||
229 | { | ||
230 | U32 error_code = GetLastError(); | ||
231 | llinfos << "AddERExcludedApplication() failed with error code " << error_code << llendl; | ||
232 | } | ||
233 | else | ||
234 | { | ||
235 | llinfos << "AddERExcludedApplication() success for " << executable_name << llendl; | ||
236 | } | ||
237 | } | ||
238 | FreeLibrary( fault_rep_dll_handle ); | ||
239 | } | ||
240 | } | ||
241 | } | ||
242 | |||
243 | const S32 MAX_CONSOLE_LINES = 500; | ||
244 | |||
245 | void create_console() | ||
246 | { | ||
247 | int h_con_handle; | ||
248 | long l_std_handle; | ||
249 | |||
250 | CONSOLE_SCREEN_BUFFER_INFO coninfo; | ||
251 | FILE *fp; | ||
252 | |||
253 | // allocate a console for this app | ||
254 | AllocConsole(); | ||
255 | |||
256 | // set the screen buffer to be big enough to let us scroll text | ||
257 | GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), &coninfo); | ||
258 | coninfo.dwSize.Y = MAX_CONSOLE_LINES; | ||
259 | SetConsoleScreenBufferSize(GetStdHandle(STD_OUTPUT_HANDLE), coninfo.dwSize); | ||
260 | |||
261 | // redirect unbuffered STDOUT to the console | ||
262 | l_std_handle = (long)GetStdHandle(STD_OUTPUT_HANDLE); | ||
263 | h_con_handle = _open_osfhandle(l_std_handle, _O_TEXT); | ||
264 | fp = _fdopen( h_con_handle, "w" ); | ||
265 | *stdout = *fp; | ||
266 | setvbuf( stdout, NULL, _IONBF, 0 ); | ||
267 | |||
268 | // redirect unbuffered STDIN to the console | ||
269 | l_std_handle = (long)GetStdHandle(STD_INPUT_HANDLE); | ||
270 | h_con_handle = _open_osfhandle(l_std_handle, _O_TEXT); | ||
271 | fp = _fdopen( h_con_handle, "r" ); | ||
272 | *stdin = *fp; | ||
273 | setvbuf( stdin, NULL, _IONBF, 0 ); | ||
274 | |||
275 | // redirect unbuffered STDERR to the console | ||
276 | l_std_handle = (long)GetStdHandle(STD_ERROR_HANDLE); | ||
277 | h_con_handle = _open_osfhandle(l_std_handle, _O_TEXT); | ||
278 | fp = _fdopen( h_con_handle, "w" ); | ||
279 | *stderr = *fp; | ||
280 | setvbuf( stderr, NULL, _IONBF, 0 ); | ||
281 | } | ||
282 | |||
283 | LLAppViewerWin32::LLAppViewerWin32() | ||
284 | { | ||
285 | } | ||
286 | |||
287 | LLAppViewerWin32::~LLAppViewerWin32() | ||
288 | { | ||
289 | } | ||
290 | |||
291 | bool LLAppViewerWin32::init() | ||
292 | { | ||
293 | // Platform specific initialization. | ||
294 | |||
295 | // Turn off Windows XP Error Reporting | ||
296 | // (Don't send our data to Microsoft--at least until we are Logo approved and have a way | ||
297 | // of getting the data back from them.) | ||
298 | // | ||
299 | llinfos << "Turning off Windows error reporting." << llendl; | ||
300 | disableWinErrorReporting(); | ||
301 | |||
302 | return LLAppViewer::init(); | ||
303 | } | ||
304 | |||
305 | bool LLAppViewerWin32::cleanup() | ||
306 | { | ||
307 | bool result = LLAppViewer::cleanup(); | ||
308 | |||
309 | gDXHardware.cleanup(); | ||
310 | |||
311 | return result; | ||
312 | } | ||
313 | |||
314 | bool LLAppViewerWin32::initWindow() | ||
315 | { | ||
316 | // pop up debug console if necessary | ||
317 | if (gUseConsole && gSavedSettings.getBOOL("ShowConsoleWindow")) | ||
318 | { | ||
319 | create_console(); | ||
320 | } | ||
321 | |||
322 | return LLAppViewer::initWindow(); | ||
323 | } | ||
324 | |||
325 | void write_debug_dx(const char* str) | ||
326 | { | ||
327 | LLString value = gDebugInfo["DXInfo"].asString(); | ||
328 | value += str; | ||
329 | gDebugInfo["DXInfo"] = value; | ||
330 | } | ||
331 | |||
332 | void write_debug_dx(const std::string& str) | ||
333 | { | ||
334 | write_debug_dx(str.c_str()); | ||
335 | } | ||
336 | |||
337 | bool LLAppViewerWin32::initHardwareTest() | ||
338 | { | ||
339 | // | ||
340 | // Do driver verification and initialization based on DirectX | ||
341 | // hardware polling and driver versions | ||
342 | // | ||
343 | if (gProbeHardware) | ||
344 | { | ||
345 | BOOL vram_only = !gSavedSettings.getBOOL("ProbeHardwareOnStartup"); | ||
346 | |||
347 | LLSplashScreen::update("Detecting hardware..."); | ||
348 | |||
349 | llinfos << "Attempting to poll DirectX for hardware info" << llendl; | ||
350 | gDXHardware.setWriteDebugFunc(write_debug_dx); | ||
351 | BOOL probe_ok = gDXHardware.getInfo(vram_only); | ||
352 | |||
353 | if (!probe_ok | ||
354 | && gSavedSettings.getWarning("AboutDirectX9")) | ||
355 | { | ||
356 | llinfos << "DirectX probe failed, alerting user." << llendl; | ||
357 | |||
358 | // Warn them that runnin without DirectX 9 will | ||
359 | // not allow us to tell them about driver issues | ||
360 | std::ostringstream msg; | ||
361 | msg << | ||
362 | LLAppViewer::instance()->getSecondLifeTitle() << " is unable to detect DirectX 9.0b or greater.\n" | ||
363 | "\n" << | ||
364 | LLAppViewer::instance()->getSecondLifeTitle() << " uses DirectX to detect hardware and/or\n" | ||
365 | "outdated drivers that can cause stability problems,\n" | ||
366 | "poor performance and crashes. While you can run\n" << | ||
367 | LLAppViewer::instance()->getSecondLifeTitle() << " without it, we highly recommend running\n" | ||
368 | "with DirectX 9.0b\n" | ||
369 | "\n" | ||
370 | "Do you wish to continue?\n"; | ||
371 | S32 button = OSMessageBox( | ||
372 | msg.str().c_str(), | ||
373 | "Warning", | ||
374 | OSMB_YESNO); | ||
375 | if (OSBTN_NO== button) | ||
376 | { | ||
377 | llinfos << "User quitting after failed DirectX 9 detection" << llendl; | ||
378 | LLWeb::loadURLExternal(DIRECTX_9_URL); | ||
379 | return false; | ||
380 | } | ||
381 | gSavedSettings.setWarning("AboutDirectX9", FALSE); | ||
382 | } | ||
383 | llinfos << "Done polling DirectX for hardware info" << llendl; | ||
384 | |||
385 | // Only probe once after installation | ||
386 | gSavedSettings.setBOOL("ProbeHardwareOnStartup", FALSE); | ||
387 | |||
388 | // Disable so debugger can work | ||
389 | std::ostringstream splash_msg; | ||
390 | splash_msg << "Loading " << LLAppViewer::instance()->getSecondLifeTitle() << "..."; | ||
391 | |||
392 | LLSplashScreen::update(splash_msg.str().c_str()); | ||
393 | } | ||
394 | |||
395 | if (!LLWinDebug::setupExceptionHandler()) | ||
396 | { | ||
397 | llwarns << " Someone took over my exception handler (post hardware probe)!" << llendl; | ||
398 | } | ||
399 | |||
400 | gGLManager.mVRAM = gDXHardware.getVRAM(); | ||
401 | llinfos << "Detected VRAM: " << gGLManager.mVRAM << llendl; | ||
402 | |||
403 | return true; | ||
404 | } | ||
405 | |||
406 | void LLAppViewerWin32::handleCrashReporting() | ||
407 | { | ||
408 | // Windows only behaivor. Spawn win crash reporter. | ||
409 | std::string exe_path = gDirUtilp->getAppRODataDir(); | ||
410 | exe_path += gDirUtilp->getDirDelimiter(); | ||
411 | exe_path += "win_crash_logger.exe"; | ||
412 | |||
413 | std::string arg_string = "-user "; | ||
414 | arg_string += gGridName; | ||
415 | |||
416 | switch(getCrashBehavior()) | ||
417 | { | ||
418 | case CRASH_BEHAVIOR_ASK: | ||
419 | default: | ||
420 | arg_string += " -dialog "; | ||
421 | _spawnl(_P_NOWAIT, exe_path.c_str(), exe_path.c_str(), arg_string.c_str(), NULL); | ||
422 | break; | ||
423 | |||
424 | case CRASH_BEHAVIOR_ALWAYS_SEND: | ||
425 | _spawnl(_P_NOWAIT, exe_path.c_str(), exe_path.c_str(), arg_string.c_str(), NULL); | ||
426 | break; | ||
427 | |||
428 | case CRASH_BEHAVIOR_NEVER_SEND: | ||
429 | break; | ||
430 | } | ||
431 | } | ||
432 | |||
433 | std::string LLAppViewerWin32::generateSerialNumber() | ||
434 | { | ||
435 | char serial_md5[MD5HEX_STR_SIZE]; // Flawfinder: ignore | ||
436 | serial_md5[0] = 0; | ||
437 | |||
438 | DWORD serial = 0; | ||
439 | DWORD flags = 0; | ||
440 | BOOL success = GetVolumeInformation( | ||
441 | L"C:\\", | ||
442 | NULL, // volume name buffer | ||
443 | 0, // volume name buffer size | ||
444 | &serial, // volume serial | ||
445 | NULL, // max component length | ||
446 | &flags, // file system flags | ||
447 | NULL, // file system name buffer | ||
448 | 0); // file system name buffer size | ||
449 | if (success) | ||
450 | { | ||
451 | LLMD5 md5; | ||
452 | md5.update( (unsigned char*)&serial, sizeof(DWORD)); | ||
453 | md5.finalize(); | ||
454 | md5.hex_digest(serial_md5); | ||
455 | } | ||
456 | else | ||
457 | { | ||
458 | llwarns << "GetVolumeInformation failed" << llendl; | ||
459 | } | ||
460 | return serial_md5; | ||
461 | } \ No newline at end of file | ||