nkeynes@30: /** nkeynes@586: * $Id$ nkeynes@30: * nkeynes@30: * Main program, initializes dreamcast and gui, then passes control off to nkeynes@537: * the main loop. nkeynes@30: * nkeynes@30: * Copyright (c) 2005 Nathan Keynes. nkeynes@30: * nkeynes@30: * This program is free software; you can redistribute it and/or modify nkeynes@30: * it under the terms of the GNU General Public License as published by nkeynes@30: * the Free Software Foundation; either version 2 of the License, or nkeynes@30: * (at your option) any later version. nkeynes@30: * nkeynes@30: * This program is distributed in the hope that it will be useful, nkeynes@30: * but WITHOUT ANY WARRANTY; without even the implied warranty of nkeynes@30: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the nkeynes@30: * GNU General Public License for more details. nkeynes@1: */ nkeynes@1: nkeynes@678: #include nkeynes@68: #include nkeynes@94: #include nkeynes@772: #include "lxdream.h" nkeynes@755: #include "gettext.h" nkeynes@422: #include "mem.h" nkeynes@27: #include "dreamcast.h" nkeynes@759: #include "dream.h" nkeynes@422: #include "display.h" nkeynes@759: #include "gui.h" nkeynes@759: #include "gdlist.h" nkeynes@759: #include "syscall.h" nkeynes@422: #include "loader.h" nkeynes@106: #include "aica/audio.h" nkeynes@422: #include "gdrom/gdrom.h" nkeynes@144: #include "maple/maple.h" nkeynes@586: #include "sh4/sh4.h" nkeynes@11: nkeynes@723: nkeynes@700: char *option_list = "a:A:c:dhHl:m:npt:T:uvV:x?"; nkeynes@700: struct option longopts[] = { nkeynes@700: { "aica", required_argument, NULL, 'a' }, nkeynes@700: { "audio", required_argument, NULL, 'A' }, nkeynes@700: { "config", required_argument, NULL, 'c' }, nkeynes@700: { "debugger", no_argument, NULL, 'D' }, nkeynes@700: { "help", no_argument, NULL, 'h' }, nkeynes@700: { "headless", no_argument, NULL, 'H' }, nkeynes@700: { "log", required_argument, NULL,'l' }, nkeynes@700: { "multiplier", required_argument, NULL, 'm' }, nkeynes@700: { "run-time", required_argument, NULL, 't' }, nkeynes@700: { "trace", required_argument, NULL, 'T' }, nkeynes@700: { "unsafe", no_argument, NULL, 'u' }, nkeynes@700: { "video", no_argument, NULL, 'V' }, nkeynes@700: { "version", no_argument, NULL, 'v' }, nkeynes@700: { NULL, 0, 0, 0 } }; nkeynes@68: char *aica_program = NULL; nkeynes@531: char *display_driver_name = NULL; nkeynes@531: char *audio_driver_name = NULL; nkeynes@586: char *trace_regions = NULL; nkeynes@68: gboolean start_immediately = FALSE; nkeynes@586: gboolean no_start = FALSE; nkeynes@77: gboolean headless = FALSE; nkeynes@402: gboolean use_xlat = TRUE; nkeynes@392: gboolean show_debugger = FALSE; nkeynes@414: extern uint32_t sh4_cpu_multiplier; nkeynes@68: nkeynes@968: static void print_version() nkeynes@700: { nkeynes@738: printf( "lxdream %s\n", lxdream_full_version ); nkeynes@700: } nkeynes@700: nkeynes@968: static void print_usage() nkeynes@700: { nkeynes@700: print_version(); nkeynes@738: printf( "Usage: lxdream %s [options] [disc-file] [program-file]\n\n", lxdream_full_version ); nkeynes@736: nkeynes@700: printf( "Options:\n" ); nkeynes@700: printf( " -a, --aica=PROGFILE %s\n", _("Run the AICA SPU only, with the supplied program") ); nkeynes@700: printf( " -A, --audio=DRIVER %s\n", _("Use the specified audio driver (? to list)") ); nkeynes@700: printf( " -c, --config=CONFFILE %s\n", _("Load configuration from CONFFILE") ); nkeynes@700: printf( " -d, --debugger %s\n", _("Start in debugger mode") ); nkeynes@700: printf( " -h, --help %s\n", _("Display this usage information") ); nkeynes@700: printf( " -H, --headless %s\n", _("Run in headless (no video) mode") ); nkeynes@700: printf( " -l, --log=LEVEL %s\n", _("Set the output log level") ); nkeynes@700: printf( " -m, --multiplier=SCALE %s\n", _("Set the SH4 multiplier (1.0 = fullspeed)") ); nkeynes@700: printf( " -n %s\n", _("Don't start running immediately") ); nkeynes@700: printf( " -p %s\n", _("Start running immediately on startup") ); nkeynes@700: printf( " -t, --run-time=SECONDS %s\n", _("Run for the specified number of seconds") ); nkeynes@700: printf( " -T, --trace=REGIONS %s\n", _("Output trace information for the named regions") ); nkeynes@700: printf( " -u, --unsafe %s\n", _("Allow unsafe dcload syscalls") ); nkeynes@700: printf( " -v, --version %s\n", _("Print the lxdream version string") ); nkeynes@700: printf( " -V, --video=DRIVER %s\n", _("Use the specified video driver (? to list)") ); nkeynes@700: printf( " -x %s\n", _("Disable the SH4 translator") ); nkeynes@700: } nkeynes@700: nkeynes@968: static void bind_gettext_domain() nkeynes@723: { nkeynes@723: #ifdef ENABLE_NLS nkeynes@866: bindtextdomain( PACKAGE, get_locale_path() ); nkeynes@723: textdomain(PACKAGE); nkeynes@723: #endif nkeynes@723: } nkeynes@723: nkeynes@30: int main (int argc, char *argv[]) nkeynes@1: { nkeynes@669: int opt; nkeynes@372: double t; nkeynes@531: gboolean display_ok; nkeynes@689: uint32_t time_secs, time_nanos; nkeynes@495: nkeynes@495: install_crash_handler(); nkeynes@723: bind_gettext_domain(); nkeynes@531: display_ok = gui_parse_cmdline(&argc, &argv); nkeynes@464: nkeynes@94: while( (opt = getopt_long( argc, argv, option_list, longopts, NULL )) != -1 ) { nkeynes@736: switch( opt ) { nkeynes@736: case 'a': /* AICA only mode - argument is an AICA program */ nkeynes@736: aica_program = optarg; nkeynes@736: break; nkeynes@736: case 'A': /* Audio driver */ nkeynes@736: audio_driver_name = optarg; nkeynes@736: if( strcmp(audio_driver_name, "?") == 0 ) { nkeynes@736: print_version(); nkeynes@736: print_audio_drivers(stdout); nkeynes@736: exit(0); nkeynes@736: } nkeynes@736: break; nkeynes@736: case 'c': /* Config file */ nkeynes@736: lxdream_set_config_filename(optarg); nkeynes@736: break; nkeynes@736: case 'd': /* Launch w/ debugger */ nkeynes@736: show_debugger = TRUE; nkeynes@736: break; nkeynes@736: case 'h': /* help */ nkeynes@736: case '?': nkeynes@736: print_usage(); nkeynes@736: exit(0); nkeynes@736: break; nkeynes@736: case 'H': /* Headless - shorthand for -V null */ nkeynes@736: display_driver_name = "null"; nkeynes@736: break; nkeynes@736: case 'l': /* Log verbosity */ nkeynes@736: if( !set_global_log_level(optarg) ) { nkeynes@736: ERROR( "Unrecognized log level '%s'", optarg ); nkeynes@736: } nkeynes@736: break; nkeynes@736: case 'm': /* Set SH4 CPU clock multiplier (default 0.5) */ nkeynes@736: t = strtod(optarg, NULL); nkeynes@736: sh4_cpu_multiplier = (int)(1000.0/t); nkeynes@736: break; nkeynes@736: case 'n': /* Don't start immediately */ nkeynes@736: no_start = TRUE; nkeynes@736: start_immediately = FALSE; nkeynes@736: break; nkeynes@736: case 'p': /* Start immediately */ nkeynes@736: start_immediately = TRUE; nkeynes@736: no_start = FALSE; nkeynes@736: break; nkeynes@736: case 't': /* Time limit + auto quit */ nkeynes@736: t = strtod(optarg, NULL); nkeynes@736: time_secs = (uint32_t)t; nkeynes@736: time_nanos = (int)((t - time_secs) * 1000000000); nkeynes@736: dreamcast_set_run_time( time_secs, time_nanos ); nkeynes@736: dreamcast_set_exit_on_stop( TRUE ); nkeynes@736: break; nkeynes@736: case 'T': /* trace regions */ nkeynes@736: trace_regions = optarg; nkeynes@736: set_global_log_level("trace"); nkeynes@736: break; nkeynes@736: case 'u': /* Allow unsafe dcload syscalls */ nkeynes@736: dcload_set_allow_unsafe(TRUE); nkeynes@736: break; nkeynes@736: case 'v': nkeynes@700: print_version(); nkeynes@700: exit(0); nkeynes@736: break; nkeynes@736: case 'V': /* Video driver */ nkeynes@736: display_driver_name = optarg; nkeynes@736: if( strcmp(display_driver_name,"?") == 0 ) { nkeynes@736: print_version(); nkeynes@736: print_display_drivers(stdout); nkeynes@736: exit(0); nkeynes@736: } nkeynes@736: break; nkeynes@736: case 'x': /* Disable translator */ nkeynes@736: use_xlat = FALSE; nkeynes@736: break; nkeynes@700: } nkeynes@68: } nkeynes@30: nkeynes@450: lxdream_load_config( ); nkeynes@691: gdrom_list_init(); nkeynes@736: nkeynes@68: if( aica_program == NULL ) { nkeynes@700: dreamcast_init(); nkeynes@68: } else { nkeynes@700: dreamcast_configure_aica_only(); nkeynes@700: mem_load_block( aica_program, 0x00800000, 2048*1024 ); nkeynes@68: } nkeynes@586: mem_set_trace( trace_regions, TRUE ); nkeynes@1: nkeynes@759: audio_init_driver( audio_driver_name ); nkeynes@736: nkeynes@700: headless = display_driver_name != NULL && strcasecmp( display_driver_name, "null" ) == 0; nkeynes@106: if( headless ) { nkeynes@700: display_set_driver( &display_null_driver ); nkeynes@106: } else { nkeynes@700: gui_init(show_debugger); nkeynes@435: nkeynes@700: display_driver_t display_driver = get_display_driver_by_name(display_driver_name); nkeynes@700: if( display_driver == NULL ) { nkeynes@700: ERROR( "Video driver '%s' not found, aborting.", display_driver_name ); nkeynes@700: exit(2); nkeynes@700: } else if( display_set_driver( display_driver ) == FALSE ) { nkeynes@700: ERROR( "Video driver '%s' failed to initialize (could not connect to display?)", nkeynes@700: display_driver->name ); nkeynes@700: exit(2); nkeynes@700: } nkeynes@106: } nkeynes@106: nkeynes@144: maple_reattach_all(); nkeynes@180: INFO( "%s! ready...", APP_NAME ); nkeynes@446: nkeynes@446: for( ; optind < argc; optind++ ) { nkeynes@700: gboolean ok = gdrom_mount_image(argv[optind]); nkeynes@700: if( !ok ) { nkeynes@700: ok = file_load_magic( argv[optind] ); nkeynes@700: } nkeynes@700: if( !ok ) { nkeynes@700: ERROR( "Unrecognized file '%s'", argv[optind] ); nkeynes@700: } nkeynes@700: if( !no_start ) { nkeynes@700: start_immediately = ok; nkeynes@700: } nkeynes@144: } nkeynes@144: nkeynes@464: if( gdrom_get_current_disc() == NULL ) { nkeynes@700: const gchar *disc_file = lxdream_get_config_value( CONFIG_GDROM ); nkeynes@678: if( disc_file != NULL ) { nkeynes@678: gdrom_mount_image( disc_file ); nkeynes@678: } nkeynes@464: } nkeynes@464: nkeynes@740: sh4_translate_set_enabled( use_xlat ); nkeynes@379: nkeynes@689: if( headless ) { nkeynes@689: dreamcast_run(); nkeynes@689: } else { nkeynes@681: gui_main_loop( start_immediately && dreamcast_can_run() ); nkeynes@77: } nkeynes@671: dreamcast_shutdown(); nkeynes@68: return 0; nkeynes@1: } nkeynes@1: