// bootstrap.cxx -- bootstrap routines: main() // // Written by Curtis Olson, started May 1997. // // Copyright (C) 1997 - 2002 Curtis L. Olson - http://www.flightgear.org/~curt // // This program is free software; you can redistribute it and/or // modify it under the terms of the GNU General Public License as // published by the Free Software Foundation; either version 2 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. // // $Id$ #ifdef HAVE_CONFIG_H # include #endif #if defined(__linux__) && defined(__i386__) # include # include #endif #include #include #include #include #include #include #include STL_IOSTREAM SG_USING_STD(cerr); SG_USING_STD(endl); #include "main.hxx" #include "globals.hxx" #ifdef HAVE_WINDOWS_H # include # include #endif #include "fg_os.hxx" #ifdef macintosh # include // -dw- for command line dialog #endif char *homedir = ::getenv( "HOME" ); char *hostname = ::getenv( "HOSTNAME" ); bool free_hostname = false; // foreward declaration. void fgExitCleanup(); #if defined(__linux__) && defined(__i386__) static void handleFPE (int); static void initFPE () { fpu_control_t fpe_flags = 0; _FPU_GETCW(fpe_flags); // fpe_flags &= ~_FPU_MASK_IM; // invalid operation // fpe_flags &= ~_FPU_MASK_DM; // denormalized operand // fpe_flags &= ~_FPU_MASK_ZM; // zero-divide // fpe_flags &= ~_FPU_MASK_OM; // overflow // fpe_flags &= ~_FPU_MASK_UM; // underflow // fpe_flags &= ~_FPU_MASK_PM; // precision (inexact result) _FPU_SETCW(fpe_flags); signal(SIGFPE, handleFPE); } static void handleFPE (int num) { initFPE(); SG_LOG(SG_GENERAL, SG_ALERT, "Floating point interrupt (SIGFPE)"); } #endif #ifdef __APPLE__ typedef struct { int lo; int hi; } PSN; extern "C" { short CPSGetCurrentProcess(PSN *psn); short CPSSetProcessName (PSN *psn, char *processname); short CPSEnableForegroundOperation(PSN *psn, int _arg2, int _arg3, int _arg4, int _arg5); short CPSSetFrontProcess(PSN *psn); }; #define CPSEnableFG(psn) CPSEnableForegroundOperation(psn,0x03,0x3C,0x2C,0x1103) #endif #ifdef _MSC_VER int main ( int argc, char **argv ); int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { main( __argc, __argv ); } #endif #if defined( sgi ) #include #include #include /* * set the special "flush zero" bit (FS, bit 24) in the Control Status * Register of the FPU of R4k and beyond so that the result of any * underflowing operation will be clamped to zero, and no exception of * any kind will be generated on the CPU. This has no effect on an * R3000. */ void flush_fpe(void) { union fpc_csr f; f.fc_word = get_fpc_csr(); f.fc_struct.flush = 1; set_fpc_csr(f.fc_word); } #endif int _bootstrap_OSInit; // Main entry point; catch any exceptions that have made it this far. int main ( int argc, char **argv ) { _bootstrap_OSInit = 0; // Enable floating-point exceptions for Linux/x86 #if defined(__linux__) && defined(__i386__) initFPE(); #endif #if defined(sgi) flush_fpe(); // Bind all non-rendering threads to CPU1 // This will reduce the jitter caused by them to an absolute minimum, // but it will only work with superuser authority. if ( geteuid() == 0 ) { sysmp(MP_CLOCK, 0); // bind the timer only to CPU0 sysmp(MP_ISOLATE, 1 ); // Isolate CPU1 sysmp(MP_NONPREEMPTIVE, 1 ); // disable process time slicing on CPU1 } #endif // Enable floating-point exceptions for Windows #if defined( _MSC_VER ) && defined( DEBUG ) // Christian, we should document what this does _control87( _EM_INEXACT, _MCW_EM ); #endif #if defined( HAVE_BC5PLUS ) _control87(MCW_EM, MCW_EM); /* defined in float.h */ #endif // Keyboard focus hack #if defined(__APPLE__) && !defined(OSX_BUNDLE) { PSN psn; fgOSInit (&argc, argv); _bootstrap_OSInit++; CPSGetCurrentProcess(&psn); CPSSetProcessName(&psn, "FlightGear"); CPSEnableFG(&psn); CPSSetFrontProcess(&psn); } #endif // FIXME: add other, more specific // exceptions. try { atexit(fgExitCleanup); fgMainInit(argc, argv); } catch (sg_throwable &t) { // We must use cerr rather than // logging, since logging may be // disabled. cerr << "Fatal error: " << t.getFormattedMessage() << endl; if (!t.getOrigin().empty()) cerr << " (received from " << t.getOrigin() << ')' << endl; } catch (...) { cerr << "Unknown exception in the main loop. Aborting..." << endl; if (errno) perror("Possible cause"); } return 0; } // do some clean up on exit. Specifically we want to call alutExit() // which happens in the sound manager destructor. void fgExitCleanup() { if (_bootstrap_OSInit != 0) fgSetMouseCursor(MOUSE_CURSOR_POINTER); delete globals; if (free_hostname && hostname != NULL) free(hostname); }