17c58bb76SPaolo Bonzini /* 27c58bb76SPaolo Bonzini * QEMU storage daemon 37c58bb76SPaolo Bonzini * 47c58bb76SPaolo Bonzini * Copyright (c) 2003-2008 Fabrice Bellard 57c58bb76SPaolo Bonzini * Copyright (c) 2019 Kevin Wolf <kwolf@redhat.com> 67c58bb76SPaolo Bonzini * 77c58bb76SPaolo Bonzini * Permission is hereby granted, free of charge, to any person obtaining a copy 87c58bb76SPaolo Bonzini * of this software and associated documentation files (the "Software"), to deal 97c58bb76SPaolo Bonzini * in the Software without restriction, including without limitation the rights 107c58bb76SPaolo Bonzini * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 117c58bb76SPaolo Bonzini * copies of the Software, and to permit persons to whom the Software is 127c58bb76SPaolo Bonzini * furnished to do so, subject to the following conditions: 137c58bb76SPaolo Bonzini * 147c58bb76SPaolo Bonzini * The above copyright notice and this permission notice shall be included in 157c58bb76SPaolo Bonzini * all copies or substantial portions of the Software. 167c58bb76SPaolo Bonzini * 177c58bb76SPaolo Bonzini * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 187c58bb76SPaolo Bonzini * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 197c58bb76SPaolo Bonzini * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 207c58bb76SPaolo Bonzini * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 217c58bb76SPaolo Bonzini * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 227c58bb76SPaolo Bonzini * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 237c58bb76SPaolo Bonzini * THE SOFTWARE. 247c58bb76SPaolo Bonzini */ 257c58bb76SPaolo Bonzini 267c58bb76SPaolo Bonzini #include "qemu/osdep.h" 277c58bb76SPaolo Bonzini 287c58bb76SPaolo Bonzini #include <getopt.h> 297c58bb76SPaolo Bonzini 307c58bb76SPaolo Bonzini #include "block/block.h" 317c58bb76SPaolo Bonzini #include "block/nbd.h" 327c58bb76SPaolo Bonzini #include "chardev/char.h" 337c58bb76SPaolo Bonzini #include "crypto/init.h" 347c58bb76SPaolo Bonzini #include "monitor/monitor.h" 357c58bb76SPaolo Bonzini #include "monitor/monitor-internal.h" 367c58bb76SPaolo Bonzini 377c58bb76SPaolo Bonzini #include "qapi/error.h" 387c58bb76SPaolo Bonzini #include "qapi/qapi-visit-block-core.h" 395daa6bfdSKevin Wolf #include "qapi/qapi-visit-block-export.h" 407c58bb76SPaolo Bonzini #include "qapi/qapi-visit-control.h" 417c58bb76SPaolo Bonzini #include "qapi/qmp/qdict.h" 427c58bb76SPaolo Bonzini #include "qapi/qmp/qstring.h" 437c58bb76SPaolo Bonzini #include "qapi/qobject-input-visitor.h" 447c58bb76SPaolo Bonzini 457c58bb76SPaolo Bonzini #include "qemu-common.h" 467c58bb76SPaolo Bonzini #include "qemu-version.h" 477c58bb76SPaolo Bonzini #include "qemu/config-file.h" 487c58bb76SPaolo Bonzini #include "qemu/error-report.h" 497c58bb76SPaolo Bonzini #include "qemu/help_option.h" 507c58bb76SPaolo Bonzini #include "qemu/log.h" 517c58bb76SPaolo Bonzini #include "qemu/main-loop.h" 527c58bb76SPaolo Bonzini #include "qemu/module.h" 537c58bb76SPaolo Bonzini #include "qemu/option.h" 547c58bb76SPaolo Bonzini #include "qom/object_interfaces.h" 557c58bb76SPaolo Bonzini 567c58bb76SPaolo Bonzini #include "storage-daemon/qapi/qapi-commands.h" 577c58bb76SPaolo Bonzini #include "storage-daemon/qapi/qapi-init-commands.h" 587c58bb76SPaolo Bonzini 597c58bb76SPaolo Bonzini #include "sysemu/runstate.h" 607c58bb76SPaolo Bonzini #include "trace/control.h" 617c58bb76SPaolo Bonzini 627c58bb76SPaolo Bonzini static volatile bool exit_requested = false; 637c58bb76SPaolo Bonzini 647c58bb76SPaolo Bonzini void qemu_system_killed(int signal, pid_t pid) 657c58bb76SPaolo Bonzini { 667c58bb76SPaolo Bonzini exit_requested = true; 677c58bb76SPaolo Bonzini } 687c58bb76SPaolo Bonzini 697c58bb76SPaolo Bonzini void qmp_quit(Error **errp) 707c58bb76SPaolo Bonzini { 717c58bb76SPaolo Bonzini exit_requested = true; 727c58bb76SPaolo Bonzini } 737c58bb76SPaolo Bonzini 747c58bb76SPaolo Bonzini static void help(void) 757c58bb76SPaolo Bonzini { 767c58bb76SPaolo Bonzini printf( 777c58bb76SPaolo Bonzini "Usage: %s [options]\n" 787c58bb76SPaolo Bonzini "QEMU storage daemon\n" 797c58bb76SPaolo Bonzini "\n" 807c58bb76SPaolo Bonzini " -h, --help display this help and exit\n" 817c58bb76SPaolo Bonzini " -T, --trace [[enable=]<pattern>][,events=<file>][,file=<file>]\n" 827c58bb76SPaolo Bonzini " specify tracing options\n" 837c58bb76SPaolo Bonzini " -V, --version output version information and exit\n" 847c58bb76SPaolo Bonzini "\n" 857c58bb76SPaolo Bonzini " --blockdev [driver=]<driver>[,node-name=<N>][,discard=ignore|unmap]\n" 867c58bb76SPaolo Bonzini " [,cache.direct=on|off][,cache.no-flush=on|off]\n" 877c58bb76SPaolo Bonzini " [,read-only=on|off][,auto-read-only=on|off]\n" 887c58bb76SPaolo Bonzini " [,force-share=on|off][,detect-zeroes=on|off|unmap]\n" 897c58bb76SPaolo Bonzini " [,driver specific parameters...]\n" 907c58bb76SPaolo Bonzini " configure a block backend\n" 917c58bb76SPaolo Bonzini "\n" 927c58bb76SPaolo Bonzini " --chardev <options> configure a character device backend\n" 937c58bb76SPaolo Bonzini " (see the qemu(1) man page for possible options)\n" 947c58bb76SPaolo Bonzini "\n" 957c58bb76SPaolo Bonzini " --export [type=]nbd,device=<node-name>[,name=<export-name>]\n" 967c58bb76SPaolo Bonzini " [,writable=on|off][,bitmap=<name>]\n" 977c58bb76SPaolo Bonzini " export the specified block node over NBD\n" 987c58bb76SPaolo Bonzini " (requires --nbd-server)\n" 997c58bb76SPaolo Bonzini "\n" 1007c58bb76SPaolo Bonzini " --monitor [chardev=]name[,mode=control][,pretty[=on|off]]\n" 1017c58bb76SPaolo Bonzini " configure a QMP monitor\n" 1027c58bb76SPaolo Bonzini "\n" 1037c58bb76SPaolo Bonzini " --nbd-server addr.type=inet,addr.host=<host>,addr.port=<port>\n" 1047c58bb76SPaolo Bonzini " [,tls-creds=<id>][,tls-authz=<id>]\n" 1057c58bb76SPaolo Bonzini " --nbd-server addr.type=unix,addr.path=<path>\n" 1067c58bb76SPaolo Bonzini " [,tls-creds=<id>][,tls-authz=<id>]\n" 1077c58bb76SPaolo Bonzini " start an NBD server for exporting block nodes\n" 1087c58bb76SPaolo Bonzini "\n" 1097c58bb76SPaolo Bonzini " --object help list object types that can be added\n" 1107c58bb76SPaolo Bonzini " --object <type>,help list properties for the given object type\n" 1117c58bb76SPaolo Bonzini " --object <type>[,<property>=<value>...]\n" 1127c58bb76SPaolo Bonzini " create a new object of type <type>, setting\n" 1137c58bb76SPaolo Bonzini " properties in the order they are specified. Note\n" 1147c58bb76SPaolo Bonzini " that the 'id' property must be set.\n" 1157c58bb76SPaolo Bonzini " See the qemu(1) man page for documentation of the\n" 1167c58bb76SPaolo Bonzini " objects that can be added.\n" 1177c58bb76SPaolo Bonzini "\n" 1187c58bb76SPaolo Bonzini QEMU_HELP_BOTTOM "\n", 1197c58bb76SPaolo Bonzini error_get_progname()); 1207c58bb76SPaolo Bonzini } 1217c58bb76SPaolo Bonzini 1227c58bb76SPaolo Bonzini enum { 1237c58bb76SPaolo Bonzini OPTION_BLOCKDEV = 256, 1247c58bb76SPaolo Bonzini OPTION_CHARDEV, 1257c58bb76SPaolo Bonzini OPTION_EXPORT, 1267c58bb76SPaolo Bonzini OPTION_MONITOR, 1277c58bb76SPaolo Bonzini OPTION_NBD_SERVER, 1287c58bb76SPaolo Bonzini OPTION_OBJECT, 1297c58bb76SPaolo Bonzini }; 1307c58bb76SPaolo Bonzini 1317c58bb76SPaolo Bonzini extern QemuOptsList qemu_chardev_opts; 1327c58bb76SPaolo Bonzini 1337c58bb76SPaolo Bonzini static QemuOptsList qemu_object_opts = { 1347c58bb76SPaolo Bonzini .name = "object", 1357c58bb76SPaolo Bonzini .implied_opt_name = "qom-type", 1367c58bb76SPaolo Bonzini .head = QTAILQ_HEAD_INITIALIZER(qemu_object_opts.head), 1377c58bb76SPaolo Bonzini .desc = { 1387c58bb76SPaolo Bonzini { } 1397c58bb76SPaolo Bonzini }, 1407c58bb76SPaolo Bonzini }; 1417c58bb76SPaolo Bonzini 1427c58bb76SPaolo Bonzini static void init_qmp_commands(void) 1437c58bb76SPaolo Bonzini { 1447c58bb76SPaolo Bonzini qmp_init_marshal(&qmp_commands); 1457c58bb76SPaolo Bonzini qmp_register_command(&qmp_commands, "query-qmp-schema", 1467c58bb76SPaolo Bonzini qmp_query_qmp_schema, QCO_ALLOW_PRECONFIG); 1477c58bb76SPaolo Bonzini 1487c58bb76SPaolo Bonzini QTAILQ_INIT(&qmp_cap_negotiation_commands); 1497c58bb76SPaolo Bonzini qmp_register_command(&qmp_cap_negotiation_commands, "qmp_capabilities", 1507c58bb76SPaolo Bonzini qmp_marshal_qmp_capabilities, QCO_ALLOW_PRECONFIG); 1517c58bb76SPaolo Bonzini } 1527c58bb76SPaolo Bonzini 153*143ea767SKevin Wolf static void init_export(BlockExportOptions *export, Error **errp) 1547c58bb76SPaolo Bonzini { 1557c58bb76SPaolo Bonzini switch (export->type) { 1567c58bb76SPaolo Bonzini case BLOCK_EXPORT_TYPE_NBD: 1577c58bb76SPaolo Bonzini qmp_nbd_server_add(&export->u.nbd, errp); 1587c58bb76SPaolo Bonzini break; 1597c58bb76SPaolo Bonzini default: 1607c58bb76SPaolo Bonzini g_assert_not_reached(); 1617c58bb76SPaolo Bonzini } 1627c58bb76SPaolo Bonzini } 1637c58bb76SPaolo Bonzini 1647c58bb76SPaolo Bonzini static void process_options(int argc, char *argv[]) 1657c58bb76SPaolo Bonzini { 1667c58bb76SPaolo Bonzini int c; 1677c58bb76SPaolo Bonzini 1687c58bb76SPaolo Bonzini static const struct option long_options[] = { 1697c58bb76SPaolo Bonzini {"blockdev", required_argument, NULL, OPTION_BLOCKDEV}, 1707c58bb76SPaolo Bonzini {"chardev", required_argument, NULL, OPTION_CHARDEV}, 1717c58bb76SPaolo Bonzini {"export", required_argument, NULL, OPTION_EXPORT}, 1727c58bb76SPaolo Bonzini {"help", no_argument, NULL, 'h'}, 1737c58bb76SPaolo Bonzini {"monitor", required_argument, NULL, OPTION_MONITOR}, 1747c58bb76SPaolo Bonzini {"nbd-server", required_argument, NULL, OPTION_NBD_SERVER}, 1757c58bb76SPaolo Bonzini {"object", required_argument, NULL, OPTION_OBJECT}, 1767c58bb76SPaolo Bonzini {"trace", required_argument, NULL, 'T'}, 1777c58bb76SPaolo Bonzini {"version", no_argument, NULL, 'V'}, 1787c58bb76SPaolo Bonzini {0, 0, 0, 0} 1797c58bb76SPaolo Bonzini }; 1807c58bb76SPaolo Bonzini 1817c58bb76SPaolo Bonzini /* 1827c58bb76SPaolo Bonzini * In contrast to the system emulator, options are processed in the order 1837c58bb76SPaolo Bonzini * they are given on the command lines. This means that things must be 1847c58bb76SPaolo Bonzini * defined first before they can be referenced in another option. 1857c58bb76SPaolo Bonzini */ 1867c58bb76SPaolo Bonzini while ((c = getopt_long(argc, argv, "hT:V", long_options, NULL)) != -1) { 1877c58bb76SPaolo Bonzini switch (c) { 1887c58bb76SPaolo Bonzini case '?': 1897c58bb76SPaolo Bonzini exit(EXIT_FAILURE); 1907c58bb76SPaolo Bonzini case 'h': 1917c58bb76SPaolo Bonzini help(); 1927c58bb76SPaolo Bonzini exit(EXIT_SUCCESS); 1937c58bb76SPaolo Bonzini case 'T': 1947c58bb76SPaolo Bonzini { 1957c58bb76SPaolo Bonzini char *trace_file = trace_opt_parse(optarg); 1967c58bb76SPaolo Bonzini trace_init_file(trace_file); 1977c58bb76SPaolo Bonzini g_free(trace_file); 1987c58bb76SPaolo Bonzini break; 1997c58bb76SPaolo Bonzini } 2007c58bb76SPaolo Bonzini case 'V': 2017c58bb76SPaolo Bonzini printf("qemu-storage-daemon version " 2027c58bb76SPaolo Bonzini QEMU_FULL_VERSION "\n" QEMU_COPYRIGHT "\n"); 2037c58bb76SPaolo Bonzini exit(EXIT_SUCCESS); 2047c58bb76SPaolo Bonzini case OPTION_BLOCKDEV: 2057c58bb76SPaolo Bonzini { 2067c58bb76SPaolo Bonzini Visitor *v; 2077c58bb76SPaolo Bonzini BlockdevOptions *options; 2087c58bb76SPaolo Bonzini 2097c58bb76SPaolo Bonzini v = qobject_input_visitor_new_str(optarg, "driver", 2107c58bb76SPaolo Bonzini &error_fatal); 2117c58bb76SPaolo Bonzini 2127c58bb76SPaolo Bonzini visit_type_BlockdevOptions(v, NULL, &options, &error_fatal); 2137c58bb76SPaolo Bonzini visit_free(v); 2147c58bb76SPaolo Bonzini 2157c58bb76SPaolo Bonzini qmp_blockdev_add(options, &error_fatal); 2167c58bb76SPaolo Bonzini qapi_free_BlockdevOptions(options); 2177c58bb76SPaolo Bonzini break; 2187c58bb76SPaolo Bonzini } 2197c58bb76SPaolo Bonzini case OPTION_CHARDEV: 2207c58bb76SPaolo Bonzini { 2217c58bb76SPaolo Bonzini /* TODO This interface is not stable until we QAPIfy it */ 2227c58bb76SPaolo Bonzini QemuOpts *opts = qemu_opts_parse_noisily(&qemu_chardev_opts, 2237c58bb76SPaolo Bonzini optarg, true); 2247c58bb76SPaolo Bonzini if (opts == NULL) { 2257c58bb76SPaolo Bonzini exit(EXIT_FAILURE); 2267c58bb76SPaolo Bonzini } 2277c58bb76SPaolo Bonzini 2287c58bb76SPaolo Bonzini if (!qemu_chr_new_from_opts(opts, NULL, &error_fatal)) { 2297c58bb76SPaolo Bonzini /* No error, but NULL returned means help was printed */ 2307c58bb76SPaolo Bonzini exit(EXIT_SUCCESS); 2317c58bb76SPaolo Bonzini } 2327c58bb76SPaolo Bonzini qemu_opts_del(opts); 2337c58bb76SPaolo Bonzini break; 2347c58bb76SPaolo Bonzini } 2357c58bb76SPaolo Bonzini case OPTION_EXPORT: 2367c58bb76SPaolo Bonzini { 2377c58bb76SPaolo Bonzini Visitor *v; 238*143ea767SKevin Wolf BlockExportOptions *export; 2397c58bb76SPaolo Bonzini 2407c58bb76SPaolo Bonzini v = qobject_input_visitor_new_str(optarg, "type", &error_fatal); 241*143ea767SKevin Wolf visit_type_BlockExportOptions(v, NULL, &export, &error_fatal); 2427c58bb76SPaolo Bonzini visit_free(v); 2437c58bb76SPaolo Bonzini 2447c58bb76SPaolo Bonzini init_export(export, &error_fatal); 245*143ea767SKevin Wolf qapi_free_BlockExportOptions(export); 2467c58bb76SPaolo Bonzini break; 2477c58bb76SPaolo Bonzini } 2487c58bb76SPaolo Bonzini case OPTION_MONITOR: 2497c58bb76SPaolo Bonzini { 2507c58bb76SPaolo Bonzini Visitor *v; 2517c58bb76SPaolo Bonzini MonitorOptions *monitor; 2527c58bb76SPaolo Bonzini 2537c58bb76SPaolo Bonzini v = qobject_input_visitor_new_str(optarg, "chardev", 2547c58bb76SPaolo Bonzini &error_fatal); 2557c58bb76SPaolo Bonzini visit_type_MonitorOptions(v, NULL, &monitor, &error_fatal); 2567c58bb76SPaolo Bonzini visit_free(v); 2577c58bb76SPaolo Bonzini 2587c58bb76SPaolo Bonzini /* TODO Catch duplicate monitor IDs */ 2597c58bb76SPaolo Bonzini monitor_init(monitor, false, &error_fatal); 2607c58bb76SPaolo Bonzini qapi_free_MonitorOptions(monitor); 2617c58bb76SPaolo Bonzini break; 2627c58bb76SPaolo Bonzini } 2637c58bb76SPaolo Bonzini case OPTION_NBD_SERVER: 2647c58bb76SPaolo Bonzini { 2657c58bb76SPaolo Bonzini Visitor *v; 2667c58bb76SPaolo Bonzini NbdServerOptions *options; 2677c58bb76SPaolo Bonzini 2687c58bb76SPaolo Bonzini v = qobject_input_visitor_new_str(optarg, NULL, &error_fatal); 2697c58bb76SPaolo Bonzini visit_type_NbdServerOptions(v, NULL, &options, &error_fatal); 2707c58bb76SPaolo Bonzini visit_free(v); 2717c58bb76SPaolo Bonzini 2727c58bb76SPaolo Bonzini nbd_server_start_options(options, &error_fatal); 2737c58bb76SPaolo Bonzini qapi_free_NbdServerOptions(options); 2747c58bb76SPaolo Bonzini break; 2757c58bb76SPaolo Bonzini } 2767c58bb76SPaolo Bonzini case OPTION_OBJECT: 2777c58bb76SPaolo Bonzini { 2787c58bb76SPaolo Bonzini QemuOpts *opts; 2797c58bb76SPaolo Bonzini const char *type; 2807c58bb76SPaolo Bonzini QDict *args; 2817c58bb76SPaolo Bonzini 2827c58bb76SPaolo Bonzini /* FIXME The keyval parser rejects 'help' arguments, so we must 2837c58bb76SPaolo Bonzini * unconditionall try QemuOpts first. */ 2847c58bb76SPaolo Bonzini opts = qemu_opts_parse(&qemu_object_opts, 2857c58bb76SPaolo Bonzini optarg, true, &error_fatal); 2867c58bb76SPaolo Bonzini type = qemu_opt_get(opts, "qom-type"); 2877c58bb76SPaolo Bonzini if (type && user_creatable_print_help(type, opts)) { 2887c58bb76SPaolo Bonzini exit(EXIT_SUCCESS); 2897c58bb76SPaolo Bonzini } 2907c58bb76SPaolo Bonzini qemu_opts_del(opts); 2917c58bb76SPaolo Bonzini 2927c58bb76SPaolo Bonzini args = keyval_parse(optarg, "qom-type", &error_fatal); 2937c58bb76SPaolo Bonzini user_creatable_add_dict(args, true, &error_fatal); 2947c58bb76SPaolo Bonzini qobject_unref(args); 2957c58bb76SPaolo Bonzini break; 2967c58bb76SPaolo Bonzini } 2977c58bb76SPaolo Bonzini default: 2987c58bb76SPaolo Bonzini g_assert_not_reached(); 2997c58bb76SPaolo Bonzini } 3007c58bb76SPaolo Bonzini } 3017c58bb76SPaolo Bonzini if (optind != argc) { 3027c58bb76SPaolo Bonzini error_report("Unexpected argument: %s", argv[optind]); 3037c58bb76SPaolo Bonzini exit(EXIT_FAILURE); 3047c58bb76SPaolo Bonzini } 3057c58bb76SPaolo Bonzini } 3067c58bb76SPaolo Bonzini 3077c58bb76SPaolo Bonzini int main(int argc, char *argv[]) 3087c58bb76SPaolo Bonzini { 3097c58bb76SPaolo Bonzini #ifdef CONFIG_POSIX 3107c58bb76SPaolo Bonzini signal(SIGPIPE, SIG_IGN); 3117c58bb76SPaolo Bonzini #endif 3127c58bb76SPaolo Bonzini 3137c58bb76SPaolo Bonzini error_init(argv[0]); 3147c58bb76SPaolo Bonzini qemu_init_exec_dir(argv[0]); 3157c58bb76SPaolo Bonzini os_setup_signal_handling(); 3167c58bb76SPaolo Bonzini 3177c58bb76SPaolo Bonzini module_call_init(MODULE_INIT_QOM); 3187c58bb76SPaolo Bonzini module_call_init(MODULE_INIT_TRACE); 3197c58bb76SPaolo Bonzini qemu_add_opts(&qemu_object_opts); 3207c58bb76SPaolo Bonzini qemu_add_opts(&qemu_trace_opts); 3217c58bb76SPaolo Bonzini qcrypto_init(&error_fatal); 3227c58bb76SPaolo Bonzini bdrv_init(); 3237c58bb76SPaolo Bonzini monitor_init_globals_core(); 3247c58bb76SPaolo Bonzini init_qmp_commands(); 3257c58bb76SPaolo Bonzini 3267c58bb76SPaolo Bonzini if (!trace_init_backends()) { 3277c58bb76SPaolo Bonzini return EXIT_FAILURE; 3287c58bb76SPaolo Bonzini } 3297c58bb76SPaolo Bonzini qemu_set_log(LOG_TRACE); 3307c58bb76SPaolo Bonzini 3317c58bb76SPaolo Bonzini qemu_init_main_loop(&error_fatal); 3327c58bb76SPaolo Bonzini process_options(argc, argv); 3337c58bb76SPaolo Bonzini 3347c58bb76SPaolo Bonzini while (!exit_requested) { 3357c58bb76SPaolo Bonzini main_loop_wait(false); 3367c58bb76SPaolo Bonzini } 3377c58bb76SPaolo Bonzini 3387c58bb76SPaolo Bonzini monitor_cleanup(); 3397c58bb76SPaolo Bonzini qemu_chr_cleanup(); 3407c58bb76SPaolo Bonzini user_creatable_cleanup(); 3417c58bb76SPaolo Bonzini 3427c58bb76SPaolo Bonzini return EXIT_SUCCESS; 3437c58bb76SPaolo Bonzini } 344