Search
lxdream.org :: lxdream/src/main.c
lxdream 0.9.1
released Jun 29
Download Now
filename src/main.c
changeset 708:0b8b5eaa3b52
prev700:4650d0c7f6f9
next723:a13e5529c52a
author nkeynes
date Wed Jun 25 22:50:41 2008 +0000 (15 years ago)
permissions -rw-r--r--
last change Fix size test before end-of-block extension, and add assert after sh4_translate_end_block() for consistency
view annotate diff log raw
     1 /**
     2  * $Id$
     3  *
     4  * Main program, initializes dreamcast and gui, then passes control off to
     5  * the main loop. 
     6  *
     7  * Copyright (c) 2005 Nathan Keynes.
     8  *
     9  * This program is free software; you can redistribute it and/or modify
    10  * it under the terms of the GNU General Public License as published by
    11  * the Free Software Foundation; either version 2 of the License, or
    12  * (at your option) any later version.
    13  *
    14  * This program is distributed in the hope that it will be useful,
    15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    17  * GNU General Public License for more details.
    18  */
    20 #include <stdlib.h>
    21 #include <unistd.h>
    22 #include <getopt.h>
    23 #include <glib/gi18n.h>
    24 #include "lxdream.h"
    25 #include "syscall.h"
    26 #include "mem.h"
    27 #include "dreamcast.h"
    28 #include "display.h"
    29 #include "loader.h"
    30 #include "gui.h"
    31 #include "aica/audio.h"
    32 #include "gdrom/gdrom.h"
    33 #include "maple/maple.h"
    34 #include "sh4/sh4.h"
    36 char *option_list = "a:A:c:dhHl:m:npt:T:uvV:x?";
    37 struct option longopts[] = {
    38         { "aica", required_argument, NULL, 'a' },
    39         { "audio", required_argument, NULL, 'A' },
    40         { "config", required_argument, NULL, 'c' },
    41         { "debugger", no_argument, NULL, 'D' },
    42         { "help", no_argument, NULL, 'h' },
    43         { "headless", no_argument, NULL, 'H' },
    44         { "log", required_argument, NULL,'l' }, 
    45         { "multiplier", required_argument, NULL, 'm' },
    46         { "run-time", required_argument, NULL, 't' },
    47         { "trace", required_argument, NULL, 'T' },
    48         { "unsafe", no_argument, NULL, 'u' },
    49         { "video", no_argument, NULL, 'V' },
    50         { "version", no_argument, NULL, 'v' }, 
    51         { NULL, 0, 0, 0 } };
    52 char *aica_program = NULL;
    53 char *display_driver_name = NULL;
    54 char *audio_driver_name = NULL;
    55 char *trace_regions = NULL;
    56 gboolean start_immediately = FALSE;
    57 gboolean no_start = FALSE;
    58 gboolean headless = FALSE;
    59 gboolean use_xlat = TRUE;
    60 gboolean show_debugger = FALSE;
    61 extern uint32_t sh4_cpu_multiplier;
    63 void print_version()
    64 {
    65     printf( "lxdream " APP_VERSION "\n" );
    66 }
    68 void print_usage()
    69 {
    70     print_version();
    71     printf( "Usage: lxdream [options] [disc-file] [program-file]\n\n" );
    73     printf( "Options:\n" );
    74     printf( "   -a, --aica=PROGFILE    %s\n", _("Run the AICA SPU only, with the supplied program") );
    75     printf( "   -A, --audio=DRIVER     %s\n", _("Use the specified audio driver (? to list)") );
    76     printf( "   -c, --config=CONFFILE  %s\n", _("Load configuration from CONFFILE") );
    77     printf( "   -d, --debugger         %s\n", _("Start in debugger mode") );
    78     printf( "   -h, --help             %s\n", _("Display this usage information") );
    79     printf( "   -H, --headless         %s\n", _("Run in headless (no video) mode") );
    80     printf( "   -l, --log=LEVEL        %s\n", _("Set the output log level") );
    81     printf( "   -m, --multiplier=SCALE %s\n", _("Set the SH4 multiplier (1.0 = fullspeed)") );
    82     printf( "   -n                     %s\n", _("Don't start running immediately") );
    83     printf( "   -p                     %s\n", _("Start running immediately on startup") );
    84     printf( "   -t, --run-time=SECONDS %s\n", _("Run for the specified number of seconds") );
    85     printf( "   -T, --trace=REGIONS    %s\n", _("Output trace information for the named regions") );
    86     printf( "   -u, --unsafe           %s\n", _("Allow unsafe dcload syscalls") );
    87     printf( "   -v, --version          %s\n", _("Print the lxdream version string") );
    88     printf( "   -V, --video=DRIVER     %s\n", _("Use the specified video driver (? to list)") );
    89     printf( "   -x                     %s\n", _("Disable the SH4 translator") );
    90 }
    92 int main (int argc, char *argv[])
    93 {
    94     int opt;
    95     double t;
    96     gboolean display_ok;
    97     uint32_t time_secs, time_nanos;
    99     install_crash_handler();
   100 #ifdef ENABLE_NLS
   101     bindtextdomain (PACKAGE, PACKAGE_LOCALE_DIR);
   102     textdomain (PACKAGE);
   103 #endif
   104     display_ok = gui_parse_cmdline(&argc, &argv);
   106     while( (opt = getopt_long( argc, argv, option_list, longopts, NULL )) != -1 ) {
   107 	switch( opt ) {
   108 	case 'a': /* AICA only mode - argument is an AICA program */
   109 	    aica_program = optarg;
   110 	    break;
   111     case 'A': /* Audio driver */
   112         audio_driver_name = optarg;
   113         if( strcmp(audio_driver_name, "?") == 0 ) {
   114             print_version();
   115             print_audio_drivers(stdout);
   116             exit(0);
   117         }
   118         break;
   119 	case 'c': /* Config file */
   120 	    lxdream_set_config_filename(optarg);
   121 	    break;
   122 	case 'd': /* Launch w/ debugger */
   123 	    show_debugger = TRUE;
   124 	    break;
   125     case 'h': /* help */
   126     case '?':
   127         print_usage();
   128         exit(0);
   129         break;
   130     case 'H': /* Headless - shorthand for -V null */
   131         display_driver_name = "null";
   132         break;
   133     case 'l': /* Log verbosity */
   134         if( !set_global_log_level(optarg) ) {
   135             ERROR( "Unrecognized log level '%s'", optarg );
   136         }
   137         break;
   138 	case 'm': /* Set SH4 CPU clock multiplier (default 0.5) */
   139 	    t = strtod(optarg, NULL);
   140 	    sh4_cpu_multiplier = (int)(1000.0/t);
   141 	    break;
   142 	case 'n': /* Don't start immediately */
   143 	    no_start = TRUE;
   144 	    start_immediately = FALSE;
   145 	    break;
   146 	case 'p': /* Start immediately */
   147 	    start_immediately = TRUE;
   148 	    no_start = FALSE;
   149 	    break;
   150 	case 't': /* Time limit + auto quit */
   151 	    t = strtod(optarg, NULL);
   152 	    time_secs = (uint32_t)t;
   153 	    time_nanos = (int)((t - time_secs) * 1000000000);
   154 	    dreamcast_set_run_time( time_secs, time_nanos );
   155 	    dreamcast_set_exit_on_stop( TRUE );
   156 	    break;
   157 	case 'T': /* trace regions */
   158 	    trace_regions = optarg;
   159 	    set_global_log_level("trace");
   160 	    break;
   161     case 'u': /* Allow unsafe dcload syscalls */
   162         dcload_set_allow_unsafe(TRUE);
   163         break;
   164 	case 'v': 
   165 	    print_version();
   166 	    exit(0);
   167 	    break;
   168     case 'V': /* Video driver */
   169         display_driver_name = optarg;
   170         if( strcmp(display_driver_name,"?") == 0 ) {
   171             print_version();
   172             print_display_drivers(stdout);
   173             exit(0);
   174         }
   175         break;
   176 	case 'x': /* Disable translator */
   177 	    use_xlat = FALSE;
   178 	    break;
   179 	}
   180     }
   182     lxdream_load_config( );
   183     gdrom_list_init();
   185     if( aica_program == NULL ) {
   186         dreamcast_init();
   187     } else {
   188         dreamcast_configure_aica_only();
   189         mem_load_block( aica_program, 0x00800000, 2048*1024 );
   190     }
   191     mem_set_trace( trace_regions, TRUE );
   193     audio_init_driver( audio_driver_name, 44100, AUDIO_FMT_16ST );
   195     headless = display_driver_name != NULL && strcasecmp( display_driver_name, "null" ) == 0;
   196     if( headless ) {
   197         display_set_driver( &display_null_driver );
   198     } else {
   199         gui_init(show_debugger);
   201         display_driver_t display_driver = get_display_driver_by_name(display_driver_name);
   202         if( display_driver == NULL ) {
   203             ERROR( "Video driver '%s' not found, aborting.", display_driver_name );
   204             exit(2);
   205         } else if( display_set_driver( display_driver ) == FALSE ) {
   206             ERROR( "Video driver '%s' failed to initialize (could not connect to display?)", 
   207                     display_driver->name );
   208             exit(2);
   209         }
   210     }
   212     maple_reattach_all();
   213     INFO( "%s! ready...", APP_NAME );
   215     for( ; optind < argc; optind++ ) {
   216         gboolean ok = gdrom_mount_image(argv[optind]);
   217         if( !ok ) {
   218             ok = file_load_magic( argv[optind] );
   219         }
   220         if( !ok ) {
   221             ERROR( "Unrecognized file '%s'", argv[optind] );
   222         }
   223         if( !no_start ) {
   224             start_immediately = ok;
   225         }
   226     }
   228     if( gdrom_get_current_disc() == NULL ) {
   229         const gchar *disc_file = lxdream_get_config_value( CONFIG_GDROM );
   230         if( disc_file != NULL ) {
   231             gdrom_mount_image( disc_file );
   232         }
   233     }
   235     sh4_set_use_xlat( use_xlat );
   237     if( headless ) {
   238         dreamcast_run();
   239     } else {
   240         gui_main_loop( start_immediately && dreamcast_can_run() );
   241     }
   242     dreamcast_shutdown();
   243     return 0;
   244 }
.