filename | src/main.c |
changeset | 1109:700c5ab26a63 |
prev | 1108:305ef2082079 |
next | 1111:742c073f353f |
author | nkeynes |
date | Thu Jun 10 22:13:16 2010 +1000 (11 years ago) |
permissions | -rw-r--r-- |
last change | Integrate executable wrapping into the user interface - command-line now loads wrapped by default, -e <bin> to run binary - add support for .bin executables - Add useful (internal) error codes |
file | annotate | diff | log | raw |
nkeynes@30 | 1 | /** |
nkeynes@561 | 2 | * $Id$ |
nkeynes@30 | 3 | * |
nkeynes@30 | 4 | * Main program, initializes dreamcast and gui, then passes control off to |
nkeynes@537 | 5 | * the main loop. |
nkeynes@30 | 6 | * |
nkeynes@30 | 7 | * Copyright (c) 2005 Nathan Keynes. |
nkeynes@30 | 8 | * |
nkeynes@30 | 9 | * This program is free software; you can redistribute it and/or modify |
nkeynes@30 | 10 | * it under the terms of the GNU General Public License as published by |
nkeynes@30 | 11 | * the Free Software Foundation; either version 2 of the License, or |
nkeynes@30 | 12 | * (at your option) any later version. |
nkeynes@30 | 13 | * |
nkeynes@30 | 14 | * This program is distributed in the hope that it will be useful, |
nkeynes@30 | 15 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
nkeynes@30 | 16 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
nkeynes@30 | 17 | * GNU General Public License for more details. |
nkeynes@1 | 18 | */ |
nkeynes@1 | 19 | |
nkeynes@678 | 20 | #include <stdlib.h> |
nkeynes@68 | 21 | #include <unistd.h> |
nkeynes@94 | 22 | #include <getopt.h> |
nkeynes@1107 | 23 | #include <libisofs/libisofs.h> |
nkeynes@772 | 24 | #include "lxdream.h" |
nkeynes@1041 | 25 | #include "lxpaths.h" |
nkeynes@755 | 26 | #include "gettext.h" |
nkeynes@422 | 27 | #include "mem.h" |
nkeynes@27 | 28 | #include "dreamcast.h" |
nkeynes@759 | 29 | #include "dream.h" |
nkeynes@422 | 30 | #include "display.h" |
nkeynes@759 | 31 | #include "gui.h" |
nkeynes@759 | 32 | #include "gdlist.h" |
nkeynes@759 | 33 | #include "syscall.h" |
nkeynes@422 | 34 | #include "loader.h" |
nkeynes@106 | 35 | #include "aica/audio.h" |
nkeynes@422 | 36 | #include "gdrom/gdrom.h" |
nkeynes@144 | 37 | #include "maple/maple.h" |
nkeynes@564 | 38 | #include "sh4/sh4.h" |
nkeynes@998 | 39 | #include "aica/armdasm.h" |
nkeynes@1034 | 40 | #include "vmu/vmulist.h" |
nkeynes@1081 | 41 | #include "serial.h" |
nkeynes@1015 | 42 | #include "hotkeys.h" |
nkeynes@1024 | 43 | #include "plugin.h" |
nkeynes@11 | 44 | |
nkeynes@1109 | 45 | char *option_list = "a:A:bc:e:dfg:G:hHl:m:npt:T:uvV:x?"; |
nkeynes@700 | 46 | struct option longopts[] = { |
nkeynes@700 | 47 | { "aica", required_argument, NULL, 'a' }, |
nkeynes@700 | 48 | { "audio", required_argument, NULL, 'A' }, |
nkeynes@1100 | 49 | { "biosless", no_argument, NULL, 'b' }, |
nkeynes@700 | 50 | { "config", required_argument, NULL, 'c' }, |
nkeynes@1108 | 51 | { "debugger", no_argument, NULL, 'd' }, |
nkeynes@1109 | 52 | { "execute", required_argument, NULL, 'e' }, |
nkeynes@1015 | 53 | { "fullscreen", no_argument, NULL, 'f' }, |
nkeynes@998 | 54 | { "gdb-sh4", required_argument, NULL, 'g' }, |
nkeynes@998 | 55 | { "gdb-arm", required_argument, NULL, 'G' }, |
nkeynes@700 | 56 | { "help", no_argument, NULL, 'h' }, |
nkeynes@700 | 57 | { "headless", no_argument, NULL, 'H' }, |
nkeynes@700 | 58 | { "log", required_argument, NULL,'l' }, |
nkeynes@700 | 59 | { "multiplier", required_argument, NULL, 'm' }, |
nkeynes@700 | 60 | { "run-time", required_argument, NULL, 't' }, |
nkeynes@700 | 61 | { "trace", required_argument, NULL, 'T' }, |
nkeynes@700 | 62 | { "unsafe", no_argument, NULL, 'u' }, |
nkeynes@700 | 63 | { "video", no_argument, NULL, 'V' }, |
nkeynes@700 | 64 | { "version", no_argument, NULL, 'v' }, |
nkeynes@700 | 65 | { NULL, 0, 0, 0 } }; |
nkeynes@68 | 66 | char *aica_program = NULL; |
nkeynes@531 | 67 | char *display_driver_name = NULL; |
nkeynes@531 | 68 | char *audio_driver_name = NULL; |
nkeynes@562 | 69 | char *trace_regions = NULL; |
nkeynes@998 | 70 | char *sh4_gdb_port = NULL; |
nkeynes@998 | 71 | char *arm_gdb_port = NULL; |
nkeynes@68 | 72 | gboolean start_immediately = FALSE; |
nkeynes@575 | 73 | gboolean no_start = FALSE; |
nkeynes@77 | 74 | gboolean headless = FALSE; |
nkeynes@402 | 75 | gboolean use_xlat = TRUE; |
nkeynes@392 | 76 | gboolean show_debugger = FALSE; |
nkeynes@1015 | 77 | gboolean show_fullscreen = FALSE; |
nkeynes@1100 | 78 | gboolean use_bootrom = TRUE; |
nkeynes@414 | 79 | extern uint32_t sh4_cpu_multiplier; |
nkeynes@68 | 80 | |
nkeynes@968 | 81 | static void print_version() |
nkeynes@700 | 82 | { |
nkeynes@738 | 83 | printf( "lxdream %s\n", lxdream_full_version ); |
nkeynes@700 | 84 | } |
nkeynes@700 | 85 | |
nkeynes@968 | 86 | static void print_usage() |
nkeynes@700 | 87 | { |
nkeynes@700 | 88 | print_version(); |
nkeynes@1109 | 89 | printf( "Usage: lxdream %s [options] [disc-file] [save-state]\n\n", lxdream_full_version ); |
nkeynes@736 | 90 | |
nkeynes@700 | 91 | printf( "Options:\n" ); |
nkeynes@700 | 92 | printf( " -a, --aica=PROGFILE %s\n", _("Run the AICA SPU only, with the supplied program") ); |
nkeynes@700 | 93 | printf( " -A, --audio=DRIVER %s\n", _("Use the specified audio driver (? to list)") ); |
nkeynes@1100 | 94 | printf( " -b, --biosless %s\n", _("Run without the BIOS boot rom even if available") ); |
nkeynes@700 | 95 | printf( " -c, --config=CONFFILE %s\n", _("Load configuration from CONFFILE") ); |
nkeynes@1109 | 96 | printf( " -e, --execute=PROGRAM %s\n", _("Load and execute the given SH4 program") ); |
nkeynes@700 | 97 | printf( " -d, --debugger %s\n", _("Start in debugger mode") ); |
nkeynes@1015 | 98 | printf( " -f, --fullscreen %s\n", _("Start in fullscreen mode") ); |
nkeynes@998 | 99 | printf( " -g, --gdb-sh4=PORT %s\n", _("Start GDB remote server on PORT for SH4") ); |
nkeynes@998 | 100 | printf( " -G, --gdb-arm=PORT %s\n", _("Start GDB remote server on PORT for ARM") ); |
nkeynes@700 | 101 | printf( " -h, --help %s\n", _("Display this usage information") ); |
nkeynes@700 | 102 | printf( " -H, --headless %s\n", _("Run in headless (no video) mode") ); |
nkeynes@700 | 103 | printf( " -l, --log=LEVEL %s\n", _("Set the output log level") ); |
nkeynes@700 | 104 | printf( " -m, --multiplier=SCALE %s\n", _("Set the SH4 multiplier (1.0 = fullspeed)") ); |
nkeynes@700 | 105 | printf( " -n %s\n", _("Don't start running immediately") ); |
nkeynes@700 | 106 | printf( " -p %s\n", _("Start running immediately on startup") ); |
nkeynes@700 | 107 | printf( " -t, --run-time=SECONDS %s\n", _("Run for the specified number of seconds") ); |
nkeynes@700 | 108 | printf( " -T, --trace=REGIONS %s\n", _("Output trace information for the named regions") ); |
nkeynes@700 | 109 | printf( " -u, --unsafe %s\n", _("Allow unsafe dcload syscalls") ); |
nkeynes@700 | 110 | printf( " -v, --version %s\n", _("Print the lxdream version string") ); |
nkeynes@700 | 111 | printf( " -V, --video=DRIVER %s\n", _("Use the specified video driver (? to list)") ); |
nkeynes@700 | 112 | printf( " -x %s\n", _("Disable the SH4 translator") ); |
nkeynes@700 | 113 | } |
nkeynes@700 | 114 | |
nkeynes@968 | 115 | static void bind_gettext_domain() |
nkeynes@723 | 116 | { |
nkeynes@723 | 117 | #ifdef ENABLE_NLS |
nkeynes@866 | 118 | bindtextdomain( PACKAGE, get_locale_path() ); |
nkeynes@723 | 119 | textdomain(PACKAGE); |
nkeynes@723 | 120 | #endif |
nkeynes@723 | 121 | } |
nkeynes@723 | 122 | |
nkeynes@30 | 123 | int main (int argc, char *argv[]) |
nkeynes@1 | 124 | { |
nkeynes@669 | 125 | int opt; |
nkeynes@372 | 126 | double t; |
nkeynes@1109 | 127 | gboolean display_ok, have_disc = FALSE, have_save = FALSE, have_exec = FALSE; |
nkeynes@689 | 128 | uint32_t time_secs, time_nanos; |
nkeynes@1109 | 129 | const char *exec_name = NULL; |
nkeynes@495 | 130 | |
nkeynes@495 | 131 | install_crash_handler(); |
nkeynes@723 | 132 | bind_gettext_domain(); |
nkeynes@531 | 133 | display_ok = gui_parse_cmdline(&argc, &argv); |
nkeynes@464 | 134 | |
nkeynes@94 | 135 | while( (opt = getopt_long( argc, argv, option_list, longopts, NULL )) != -1 ) { |
nkeynes@736 | 136 | switch( opt ) { |
nkeynes@736 | 137 | case 'a': /* AICA only mode - argument is an AICA program */ |
nkeynes@736 | 138 | aica_program = optarg; |
nkeynes@77 | 139 | break; |
nkeynes@736 | 140 | case 'A': /* Audio driver */ |
nkeynes@736 | 141 | audio_driver_name = optarg; |
nkeynes@736 | 142 | break; |
nkeynes@1100 | 143 | case 'b': /* No Boot rom */ |
nkeynes@1100 | 144 | use_bootrom = FALSE; |
nkeynes@736 | 145 | case 'c': /* Config file */ |
nkeynes@736 | 146 | lxdream_set_config_filename(optarg); |
nkeynes@736 | 147 | break; |
nkeynes@736 | 148 | case 'd': /* Launch w/ debugger */ |
nkeynes@736 | 149 | show_debugger = TRUE; |
nkeynes@736 | 150 | break; |
nkeynes@1109 | 151 | case 'e': |
nkeynes@1109 | 152 | exec_name = optarg; |
nkeynes@1109 | 153 | break; |
nkeynes@1015 | 154 | case 'f': |
nkeynes@1015 | 155 | show_fullscreen = TRUE; |
nkeynes@1015 | 156 | break; |
nkeynes@998 | 157 | case 'g': |
nkeynes@998 | 158 | sh4_gdb_port = optarg; |
nkeynes@998 | 159 | break; |
nkeynes@998 | 160 | case 'G': |
nkeynes@998 | 161 | arm_gdb_port = optarg; |
nkeynes@998 | 162 | break; |
nkeynes@736 | 163 | case 'h': /* help */ |
nkeynes@736 | 164 | case '?': |
nkeynes@736 | 165 | print_usage(); |
nkeynes@736 | 166 | exit(0); |
nkeynes@736 | 167 | break; |
nkeynes@736 | 168 | case 'H': /* Headless - shorthand for -V null */ |
nkeynes@736 | 169 | display_driver_name = "null"; |
nkeynes@736 | 170 | break; |
nkeynes@736 | 171 | case 'l': /* Log verbosity */ |
nkeynes@736 | 172 | if( !set_global_log_level(optarg) ) { |
nkeynes@736 | 173 | ERROR( "Unrecognized log level '%s'", optarg ); |
nkeynes@736 | 174 | } |
nkeynes@736 | 175 | break; |
nkeynes@736 | 176 | case 'm': /* Set SH4 CPU clock multiplier (default 0.5) */ |
nkeynes@736 | 177 | t = strtod(optarg, NULL); |
nkeynes@736 | 178 | sh4_cpu_multiplier = (int)(1000.0/t); |
nkeynes@736 | 179 | break; |
nkeynes@736 | 180 | case 'n': /* Don't start immediately */ |
nkeynes@736 | 181 | no_start = TRUE; |
nkeynes@736 | 182 | start_immediately = FALSE; |
nkeynes@736 | 183 | break; |
nkeynes@736 | 184 | case 'p': /* Start immediately */ |
nkeynes@736 | 185 | start_immediately = TRUE; |
nkeynes@736 | 186 | no_start = FALSE; |
nkeynes@736 | 187 | break; |
nkeynes@736 | 188 | case 't': /* Time limit + auto quit */ |
nkeynes@736 | 189 | t = strtod(optarg, NULL); |
nkeynes@736 | 190 | time_secs = (uint32_t)t; |
nkeynes@736 | 191 | time_nanos = (int)((t - time_secs) * 1000000000); |
nkeynes@736 | 192 | dreamcast_set_run_time( time_secs, time_nanos ); |
nkeynes@736 | 193 | dreamcast_set_exit_on_stop( TRUE ); |
nkeynes@736 | 194 | break; |
nkeynes@736 | 195 | case 'T': /* trace regions */ |
nkeynes@736 | 196 | trace_regions = optarg; |
nkeynes@736 | 197 | set_global_log_level("trace"); |
nkeynes@736 | 198 | break; |
nkeynes@736 | 199 | case 'u': /* Allow unsafe dcload syscalls */ |
nkeynes@736 | 200 | dcload_set_allow_unsafe(TRUE); |
nkeynes@736 | 201 | break; |
nkeynes@736 | 202 | case 'v': |
nkeynes@700 | 203 | print_version(); |
nkeynes@700 | 204 | exit(0); |
nkeynes@736 | 205 | break; |
nkeynes@736 | 206 | case 'V': /* Video driver */ |
nkeynes@736 | 207 | display_driver_name = optarg; |
nkeynes@736 | 208 | break; |
nkeynes@736 | 209 | case 'x': /* Disable translator */ |
nkeynes@736 | 210 | use_xlat = FALSE; |
nkeynes@736 | 211 | break; |
nkeynes@700 | 212 | } |
nkeynes@68 | 213 | } |
nkeynes@30 | 214 | |
nkeynes@1024 | 215 | #ifdef ENABLE_SHARED |
nkeynes@1027 | 216 | plugin_init(); |
nkeynes@1024 | 217 | #endif |
nkeynes@1024 | 218 | |
nkeynes@1038 | 219 | lxdream_make_config_dir( ); |
nkeynes@450 | 220 | lxdream_load_config( ); |
nkeynes@144 | 221 | |
nkeynes@1024 | 222 | if( audio_driver_name != NULL && strcmp(audio_driver_name, "?") == 0 ) { |
nkeynes@1024 | 223 | print_version(); |
nkeynes@1024 | 224 | print_audio_drivers(stdout); |
nkeynes@1024 | 225 | exit(0); |
nkeynes@1024 | 226 | } |
nkeynes@1024 | 227 | |
nkeynes@1024 | 228 | if( display_driver_name != NULL && strcmp(display_driver_name,"?") == 0 ) { |
nkeynes@1024 | 229 | print_version(); |
nkeynes@1024 | 230 | print_display_drivers(stdout); |
nkeynes@1024 | 231 | exit(0); |
nkeynes@1024 | 232 | } |
nkeynes@1024 | 233 | |
nkeynes@1107 | 234 | iso_init(); |
nkeynes@691 | 235 | gdrom_list_init(); |
nkeynes@1034 | 236 | vmulist_init(); |
nkeynes@736 | 237 | |
nkeynes@68 | 238 | if( aica_program == NULL ) { |
nkeynes@1100 | 239 | dreamcast_init(use_bootrom); |
nkeynes@68 | 240 | } else { |
nkeynes@700 | 241 | dreamcast_configure_aica_only(); |
nkeynes@700 | 242 | mem_load_block( aica_program, 0x00800000, 2048*1024 ); |
nkeynes@68 | 243 | } |
nkeynes@562 | 244 | mem_set_trace( trace_regions, TRUE ); |
nkeynes@1 | 245 | |
nkeynes@759 | 246 | audio_init_driver( audio_driver_name ); |
nkeynes@736 | 247 | |
nkeynes@700 | 248 | headless = display_driver_name != NULL && strcasecmp( display_driver_name, "null" ) == 0; |
nkeynes@106 | 249 | if( headless ) { |
nkeynes@700 | 250 | display_set_driver( &display_null_driver ); |
nkeynes@106 | 251 | } else { |
nkeynes@1015 | 252 | gui_init(show_debugger, show_fullscreen); |
nkeynes@435 | 253 | |
nkeynes@700 | 254 | display_driver_t display_driver = get_display_driver_by_name(display_driver_name); |
nkeynes@700 | 255 | if( display_driver == NULL ) { |
nkeynes@700 | 256 | ERROR( "Video driver '%s' not found, aborting.", display_driver_name ); |
nkeynes@700 | 257 | exit(2); |
nkeynes@700 | 258 | } else if( display_set_driver( display_driver ) == FALSE ) { |
nkeynes@700 | 259 | ERROR( "Video driver '%s' failed to initialize (could not connect to display?)", |
nkeynes@700 | 260 | display_driver->name ); |
nkeynes@700 | 261 | exit(2); |
nkeynes@700 | 262 | } |
nkeynes@87 | 263 | } |
nkeynes@1015 | 264 | |
nkeynes@1015 | 265 | hotkeys_init(); |
nkeynes@1077 | 266 | serial_init(); |
nkeynes@106 | 267 | |
nkeynes@144 | 268 | maple_reattach_all(); |
nkeynes@180 | 269 | INFO( "%s! ready...", APP_NAME ); |
nkeynes@446 | 270 | |
nkeynes@1109 | 271 | for( ; optind < argc; optind++ ) { |
nkeynes@1108 | 272 | ERROR err; |
nkeynes@1109 | 273 | lxdream_file_type_t type = file_identify(argv[optind], -1, &err); |
nkeynes@1109 | 274 | if( type == FILE_SAVE_STATE ) { |
nkeynes@1109 | 275 | if( have_save ) { |
nkeynes@1109 | 276 | ERROR( "Multiple save states given on command-line, ignoring %s", argv[optind] ); |
nkeynes@1109 | 277 | } else { |
nkeynes@1109 | 278 | have_save = dreamcast_load_state(argv[optind]); |
nkeynes@1109 | 279 | if( !have_save ) |
nkeynes@1109 | 280 | no_start = TRUE; |
nkeynes@1109 | 281 | } |
nkeynes@1109 | 282 | } else { |
nkeynes@1109 | 283 | if( have_disc ) { |
nkeynes@1109 | 284 | ERROR( "Multiple GD-ROM discs given on command-line, ignoring %s", argv[optind] ); |
nkeynes@1109 | 285 | } else { |
nkeynes@1109 | 286 | have_disc = gdrom_mount_image(argv[optind], &err); |
nkeynes@1109 | 287 | if( !have_disc ) |
nkeynes@1109 | 288 | no_start = TRUE; |
nkeynes@1108 | 289 | } |
nkeynes@1108 | 290 | } |
nkeynes@1108 | 291 | } |
nkeynes@1108 | 292 | |
nkeynes@1109 | 293 | if( exec_name != NULL ) { |
nkeynes@1109 | 294 | ERROR err; |
nkeynes@1109 | 295 | if( have_save ) { |
nkeynes@1109 | 296 | ERROR( "Both a save state and an executable were specified, ignoring %s", exec_name ); |
nkeynes@1109 | 297 | } else { |
nkeynes@1109 | 298 | have_exec = file_load_exec( exec_name, &err ); |
nkeynes@1109 | 299 | if( !have_exec ) |
nkeynes@1109 | 300 | no_start = TRUE; |
nkeynes@700 | 301 | } |
nkeynes@144 | 302 | } |
nkeynes@144 | 303 | |
nkeynes@1109 | 304 | if( !no_start && (have_exec || have_disc || have_save) ) { |
nkeynes@1109 | 305 | start_immediately = TRUE; |
nkeynes@1109 | 306 | } |
nkeynes@1109 | 307 | |
nkeynes@464 | 308 | if( gdrom_get_current_disc() == NULL ) { |
nkeynes@1109 | 309 | ERROR err; |
nkeynes@1036 | 310 | gchar *disc_file = lxdream_get_global_config_path_value( CONFIG_GDROM ); |
nkeynes@678 | 311 | if( disc_file != NULL ) { |
nkeynes@1109 | 312 | gboolean ok = gdrom_mount_image( disc_file, &err ); |
nkeynes@1036 | 313 | g_free(disc_file); |
nkeynes@1109 | 314 | if( !ok ) { |
nkeynes@1109 | 315 | WARN( err.msg ); |
nkeynes@1109 | 316 | } |
nkeynes@678 | 317 | } |
nkeynes@464 | 318 | } |
nkeynes@464 | 319 | |
nkeynes@740 | 320 | sh4_translate_set_enabled( use_xlat ); |
nkeynes@379 | 321 | |
nkeynes@998 | 322 | /* If requested, start the gdb server immediately before we go into the main |
nkeynes@998 | 323 | * loop. |
nkeynes@998 | 324 | */ |
nkeynes@998 | 325 | if( sh4_gdb_port != NULL ) { |
nkeynes@998 | 326 | gdb_init_server( NULL, strtol(sh4_gdb_port,NULL,0), &sh4_cpu_desc, TRUE ); |
nkeynes@372 | 327 | } |
nkeynes@998 | 328 | if( arm_gdb_port != NULL ) { |
nkeynes@998 | 329 | gdb_init_server( NULL, strtol(arm_gdb_port,NULL,0), &arm_cpu_desc, TRUE ); |
nkeynes@77 | 330 | } |
nkeynes@998 | 331 | |
nkeynes@689 | 332 | if( headless ) { |
nkeynes@689 | 333 | dreamcast_run(); |
nkeynes@689 | 334 | } else { |
nkeynes@681 | 335 | gui_main_loop( start_immediately && dreamcast_can_run() ); |
nkeynes@77 | 336 | } |
nkeynes@671 | 337 | dreamcast_shutdown(); |
nkeynes@68 | 338 | return 0; |
nkeynes@1 | 339 | } |
nkeynes@1 | 340 |
.