nkeynes@30: /** nkeynes@561: * $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@1179: #include "lxdream.h" nkeynes@1111: #include nkeynes@1041: #include "lxpaths.h" nkeynes@755: #include "gettext.h" nkeynes@1129: #include "dream.h" nkeynes@27: #include "dreamcast.h" nkeynes@422: #include "display.h" nkeynes@759: #include "gui.h" nkeynes@759: #include "gdlist.h" nkeynes@1129: #include "hotkeys.h" nkeynes@1129: #include "loader.h" nkeynes@1129: #include "mem.h" nkeynes@1129: #include "plugin.h" nkeynes@1129: #include "serial.h" nkeynes@759: #include "syscall.h" nkeynes@106: #include "aica/audio.h" nkeynes@1129: #include "aica/armdasm.h" nkeynes@422: #include "gdrom/gdrom.h" nkeynes@144: #include "maple/maple.h" nkeynes@1129: #include "pvr2/glutil.h" nkeynes@564: #include "sh4/sh4.h" nkeynes@1034: #include "vmu/vmulist.h" nkeynes@1129: nkeynes@1129: #define GL_INFO_OPT 1 nkeynes@11: nkeynes@1184: char *option_list = "a:A:bc:e:dfg:G:hHl:m:npPt:T:uvV:xX?"; nkeynes@700: struct option longopts[] = { nkeynes@700: { "aica", required_argument, NULL, 'a' }, nkeynes@700: { "audio", required_argument, NULL, 'A' }, nkeynes@1100: { "biosless", no_argument, NULL, 'b' }, nkeynes@700: { "config", required_argument, NULL, 'c' }, nkeynes@1108: { "debugger", no_argument, NULL, 'd' }, nkeynes@1109: { "execute", required_argument, NULL, 'e' }, nkeynes@1015: { "fullscreen", no_argument, NULL, 'f' }, nkeynes@998: { "gdb-sh4", required_argument, NULL, 'g' }, nkeynes@1129: { "gdb-arm", required_argument, NULL, 'G' }, nkeynes@1129: { "gl-info", no_argument, NULL, GL_INFO_OPT }, 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@1125: { "shadow", no_argument, NULL, 'X' }, 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@1184: { "sh4-profile-blocks", no_argument, NULL, 'P' }, 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@562: char *trace_regions = NULL; nkeynes@998: char *sh4_gdb_port = NULL; nkeynes@998: char *arm_gdb_port = NULL; nkeynes@68: gboolean start_immediately = FALSE; nkeynes@575: gboolean no_start = FALSE; nkeynes@77: gboolean headless = FALSE; nkeynes@1125: sh4core_t sh4_core = SH4_TRANSLATE; nkeynes@392: gboolean show_debugger = FALSE; nkeynes@1015: gboolean show_fullscreen = FALSE; nkeynes@1100: gboolean use_bootrom = TRUE; 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@1109: printf( "Usage: lxdream %s [options] [disc-file] [save-state]\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@1100: printf( " -b, --biosless %s\n", _("Run without the BIOS boot rom even if available") ); nkeynes@700: printf( " -c, --config=CONFFILE %s\n", _("Load configuration from CONFFILE") ); nkeynes@1109: printf( " -e, --execute=PROGRAM %s\n", _("Load and execute the given SH4 program") ); nkeynes@700: printf( " -d, --debugger %s\n", _("Start in debugger mode") ); nkeynes@1015: printf( " -f, --fullscreen %s\n", _("Start in fullscreen mode") ); nkeynes@998: printf( " -g, --gdb-sh4=PORT %s\n", _("Start GDB remote server on PORT for SH4") ); nkeynes@998: printf( " -G, --gdb-arm=PORT %s\n", _("Start GDB remote server on PORT for ARM") ); 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@1125: printf( " -X %s\n", _("Run both SH4 interpreter and 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@1109: gboolean display_ok, have_disc = FALSE, have_save = FALSE, have_exec = FALSE; nkeynes@1184: gboolean print_glinfo = FALSE, sh4_profile_blocks = FALSE; nkeynes@689: uint32_t time_secs, time_nanos; nkeynes@1109: const char *exec_name = NULL; 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@77: break; nkeynes@736: case 'A': /* Audio driver */ nkeynes@736: audio_driver_name = optarg; nkeynes@736: break; nkeynes@1100: case 'b': /* No Boot rom */ nkeynes@1100: use_bootrom = FALSE; 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@1109: case 'e': nkeynes@1109: exec_name = optarg; nkeynes@1109: break; nkeynes@1015: case 'f': nkeynes@1015: show_fullscreen = TRUE; nkeynes@1015: break; nkeynes@998: case 'g': nkeynes@998: sh4_gdb_port = optarg; nkeynes@998: break; nkeynes@998: case 'G': nkeynes@998: arm_gdb_port = optarg; nkeynes@998: 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@1184: case 'P': nkeynes@1184: sh4_profile_blocks = TRUE; nkeynes@1184: 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: break; nkeynes@736: case 'x': /* Disable translator */ nkeynes@1125: sh4_core = SH4_INTERPRET; nkeynes@1125: break; nkeynes@1125: case 'X': /* Shadow translator */ nkeynes@1125: sh4_core = SH4_SHADOW; nkeynes@736: break; nkeynes@1129: case GL_INFO_OPT: nkeynes@1129: print_glinfo = TRUE; nkeynes@1129: break; nkeynes@700: } nkeynes@68: } nkeynes@30: nkeynes@1225: #ifdef BUILD_PLUGINS nkeynes@1027: plugin_init(); nkeynes@1024: #endif nkeynes@1024: nkeynes@1038: lxdream_make_config_dir( ); nkeynes@450: lxdream_load_config( ); nkeynes@144: nkeynes@1024: if( audio_driver_name != NULL && strcmp(audio_driver_name, "?") == 0 ) { nkeynes@1024: print_version(); nkeynes@1024: print_audio_drivers(stdout); nkeynes@1024: exit(0); nkeynes@1024: } nkeynes@1024: nkeynes@1024: if( display_driver_name != NULL && strcmp(display_driver_name,"?") == 0 ) { nkeynes@1024: print_version(); nkeynes@1024: print_display_drivers(stdout); nkeynes@1024: exit(0); nkeynes@1024: } nkeynes@1024: nkeynes@1129: if( print_glinfo ) { nkeynes@1129: gui_init(FALSE, FALSE); nkeynes@1129: display_driver_t display_driver = get_display_driver_by_name(display_driver_name); nkeynes@1129: if( display_driver == NULL ) { nkeynes@1129: ERROR( "Video driver '%s' not found, aborting.", display_driver_name ); nkeynes@1129: exit(2); nkeynes@1129: } else if( display_set_driver( display_driver ) == FALSE ) { nkeynes@1129: ERROR( "Video driver '%s' failed to initialize (could not connect to display?)", nkeynes@1129: display_driver->name ); nkeynes@1129: exit(2); nkeynes@1134: } else if( display_driver->capabilities.has_gl == FALSE ) { nkeynes@1134: ERROR( "Video driver '%s' has no GL capabilities.", display_driver_name ); nkeynes@1134: exit(2); nkeynes@1129: } nkeynes@1129: glPrintInfo(stdout); nkeynes@1129: exit(0); nkeynes@1129: nkeynes@1129: } nkeynes@1129: nkeynes@1129: nkeynes@1107: iso_init(); nkeynes@691: gdrom_list_init(); nkeynes@1034: vmulist_init(); nkeynes@736: nkeynes@68: if( aica_program == NULL ) { nkeynes@1100: dreamcast_init(use_bootrom); nkeynes@68: } else { nkeynes@700: dreamcast_configure_aica_only(); nkeynes@700: mem_load_block( aica_program, 0x00800000, 2048*1024 ); nkeynes@68: } nkeynes@562: 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@1015: gui_init(show_debugger, show_fullscreen); 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@87: } nkeynes@1015: nkeynes@1015: hotkeys_init(); nkeynes@1077: serial_init(); nkeynes@106: nkeynes@144: maple_reattach_all(); nkeynes@180: INFO( "%s! ready...", APP_NAME ); nkeynes@446: nkeynes@1109: for( ; optind < argc; optind++ ) { nkeynes@1108: ERROR err; nkeynes@1109: lxdream_file_type_t type = file_identify(argv[optind], -1, &err); nkeynes@1109: if( type == FILE_SAVE_STATE ) { nkeynes@1109: if( have_save ) { nkeynes@1109: ERROR( "Multiple save states given on command-line, ignoring %s", argv[optind] ); nkeynes@1109: } else { nkeynes@1109: have_save = dreamcast_load_state(argv[optind]); nkeynes@1109: if( !have_save ) nkeynes@1109: no_start = TRUE; nkeynes@1109: } nkeynes@1109: } else { nkeynes@1109: if( have_disc ) { nkeynes@1109: ERROR( "Multiple GD-ROM discs given on command-line, ignoring %s", argv[optind] ); nkeynes@1109: } else { nkeynes@1109: have_disc = gdrom_mount_image(argv[optind], &err); nkeynes@1116: if( !have_disc ) { nkeynes@1116: ERROR( err.msg ); nkeynes@1109: no_start = TRUE; nkeynes@1116: } nkeynes@1108: } nkeynes@1108: } nkeynes@1108: } nkeynes@1108: nkeynes@1109: if( exec_name != NULL ) { nkeynes@1109: ERROR err; nkeynes@1109: if( have_save ) { nkeynes@1109: ERROR( "Both a save state and an executable were specified, ignoring %s", exec_name ); nkeynes@1109: } else { nkeynes@1109: have_exec = file_load_exec( exec_name, &err ); nkeynes@1116: if( !have_exec ) { nkeynes@1116: ERROR( err.msg ); nkeynes@1109: no_start = TRUE; nkeynes@1116: } nkeynes@700: } nkeynes@144: } nkeynes@144: nkeynes@1109: if( !no_start && (have_exec || have_disc || have_save) ) { nkeynes@1109: start_immediately = TRUE; nkeynes@1109: } nkeynes@1109: nkeynes@464: if( gdrom_get_current_disc() == NULL ) { nkeynes@1109: ERROR err; nkeynes@1036: gchar *disc_file = lxdream_get_global_config_path_value( CONFIG_GDROM ); nkeynes@678: if( disc_file != NULL ) { nkeynes@1109: gboolean ok = gdrom_mount_image( disc_file, &err ); nkeynes@1036: g_free(disc_file); nkeynes@1109: if( !ok ) { nkeynes@1109: WARN( err.msg ); nkeynes@1109: } nkeynes@678: } nkeynes@464: } nkeynes@464: nkeynes@1125: sh4_set_core( sh4_core ); nkeynes@1218: sh4_set_profile_blocks( sh4_profile_blocks ); nkeynes@379: nkeynes@998: /* If requested, start the gdb server immediately before we go into the main nkeynes@998: * loop. nkeynes@998: */ nkeynes@998: if( sh4_gdb_port != NULL ) { nkeynes@998: gdb_init_server( NULL, strtol(sh4_gdb_port,NULL,0), &sh4_cpu_desc, TRUE ); nkeynes@372: } nkeynes@998: if( arm_gdb_port != NULL ) { nkeynes@998: gdb_init_server( NULL, strtol(arm_gdb_port,NULL,0), &arm_cpu_desc, TRUE ); nkeynes@77: } nkeynes@998: 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: