1 #include "qemu-common.h"
2 #include "qemu-error.h"
3 #include "qemu-option.h"
4 #include "qemu-config.h"
8 static QemuOptsList qemu_drive_opts
= {
10 .head
= QTAILQ_HEAD_INITIALIZER(qemu_drive_opts
.head
),
14 .type
= QEMU_OPT_NUMBER
,
18 .type
= QEMU_OPT_NUMBER
,
19 .help
= "unit number (i.e. lun for scsi)",
22 .type
= QEMU_OPT_STRING
,
23 .help
= "interface (ide, scsi, sd, mtd, floppy, pflash, virtio)",
26 .type
= QEMU_OPT_NUMBER
,
27 .help
= "index number",
30 .type
= QEMU_OPT_NUMBER
,
31 .help
= "number of cylinders (ide disk geometry)",
34 .type
= QEMU_OPT_NUMBER
,
35 .help
= "number of heads (ide disk geometry)",
38 .type
= QEMU_OPT_NUMBER
,
39 .help
= "number of sectors (ide disk geometry)",
42 .type
= QEMU_OPT_STRING
,
43 .help
= "chs translation (auto, lba. none)",
46 .type
= QEMU_OPT_STRING
,
47 .help
= "media type (disk, cdrom)",
50 .type
= QEMU_OPT_BOOL
,
51 .help
= "enable/disable snapshot mode",
54 .type
= QEMU_OPT_STRING
,
58 .type
= QEMU_OPT_STRING
,
59 .help
= "host cache usage (none, writeback, writethrough, "
60 "directsync, unsafe)",
63 .type
= QEMU_OPT_STRING
,
64 .help
= "host AIO implementation (threads, native)",
67 .type
= QEMU_OPT_STRING
,
68 .help
= "disk format (raw, qcow2, ...)",
71 .type
= QEMU_OPT_STRING
,
72 .help
= "disk serial number",
75 .type
= QEMU_OPT_STRING
,
76 .help
= "read error action",
79 .type
= QEMU_OPT_STRING
,
80 .help
= "write error action",
83 .type
= QEMU_OPT_STRING
,
84 .help
= "pci address (virtio only)",
87 .type
= QEMU_OPT_BOOL
,
88 .help
= "open drive file as read-only",
91 .type
= QEMU_OPT_NUMBER
,
92 .help
= "limit total I/O operations per second",
95 .type
= QEMU_OPT_NUMBER
,
96 .help
= "limit read operations per second",
99 .type
= QEMU_OPT_NUMBER
,
100 .help
= "limit write operations per second",
103 .type
= QEMU_OPT_NUMBER
,
104 .help
= "limit total bytes per second",
107 .type
= QEMU_OPT_NUMBER
,
108 .help
= "limit read bytes per second",
111 .type
= QEMU_OPT_NUMBER
,
112 .help
= "limit write bytes per second",
114 .name
= "copy-on-read",
115 .type
= QEMU_OPT_BOOL
,
116 .help
= "copy read data from backing file into image file",
118 { /* end of list */ }
122 static QemuOptsList qemu_iscsi_opts
= {
124 .head
= QTAILQ_HEAD_INITIALIZER(qemu_iscsi_opts
.head
),
128 .type
= QEMU_OPT_STRING
,
129 .help
= "username for CHAP authentication to target",
132 .type
= QEMU_OPT_STRING
,
133 .help
= "password for CHAP authentication to target",
135 .name
= "header-digest",
136 .type
= QEMU_OPT_STRING
,
137 .help
= "HeaderDigest setting. "
138 "{CRC32C|CRC32C-NONE|NONE-CRC32C|NONE}",
140 .name
= "initiator-name",
141 .type
= QEMU_OPT_STRING
,
142 .help
= "Initiator iqn name to use when connecting",
144 { /* end of list */ }
148 static QemuOptsList qemu_chardev_opts
= {
150 .implied_opt_name
= "backend",
151 .head
= QTAILQ_HEAD_INITIALIZER(qemu_chardev_opts
.head
),
155 .type
= QEMU_OPT_STRING
,
158 .type
= QEMU_OPT_STRING
,
161 .type
= QEMU_OPT_STRING
,
164 .type
= QEMU_OPT_STRING
,
167 .type
= QEMU_OPT_STRING
,
170 .type
= QEMU_OPT_STRING
,
173 .type
= QEMU_OPT_NUMBER
,
176 .type
= QEMU_OPT_BOOL
,
179 .type
= QEMU_OPT_BOOL
,
182 .type
= QEMU_OPT_BOOL
,
185 .type
= QEMU_OPT_BOOL
,
188 .type
= QEMU_OPT_BOOL
,
191 .type
= QEMU_OPT_BOOL
,
194 .type
= QEMU_OPT_NUMBER
,
197 .type
= QEMU_OPT_NUMBER
,
200 .type
= QEMU_OPT_NUMBER
,
203 .type
= QEMU_OPT_NUMBER
,
206 .type
= QEMU_OPT_BOOL
,
209 .type
= QEMU_OPT_BOOL
,
212 .type
= QEMU_OPT_STRING
,
215 .type
= QEMU_OPT_NUMBER
,
217 { /* end of list */ }
221 QemuOptsList qemu_fsdev_opts
= {
223 .implied_opt_name
= "fsdriver",
224 .head
= QTAILQ_HEAD_INITIALIZER(qemu_fsdev_opts
.head
),
228 .type
= QEMU_OPT_STRING
,
231 .type
= QEMU_OPT_STRING
,
233 .name
= "security_model",
234 .type
= QEMU_OPT_STRING
,
237 .type
= QEMU_OPT_STRING
,
240 .type
= QEMU_OPT_BOOL
,
244 .type
= QEMU_OPT_STRING
,
247 .type
= QEMU_OPT_NUMBER
,
254 QemuOptsList qemu_virtfs_opts
= {
256 .implied_opt_name
= "fsdriver",
257 .head
= QTAILQ_HEAD_INITIALIZER(qemu_virtfs_opts
.head
),
261 .type
= QEMU_OPT_STRING
,
264 .type
= QEMU_OPT_STRING
,
267 .type
= QEMU_OPT_STRING
,
269 .name
= "security_model",
270 .type
= QEMU_OPT_STRING
,
273 .type
= QEMU_OPT_STRING
,
276 .type
= QEMU_OPT_BOOL
,
279 .type
= QEMU_OPT_STRING
,
282 .type
= QEMU_OPT_NUMBER
,
289 static QemuOptsList qemu_device_opts
= {
291 .implied_opt_name
= "driver",
292 .head
= QTAILQ_HEAD_INITIALIZER(qemu_device_opts
.head
),
295 * no elements => accept any
296 * sanity checking will happen later
297 * when setting device properties
299 { /* end of list */ }
303 static QemuOptsList qemu_netdev_opts
= {
305 .implied_opt_name
= "type",
306 .head
= QTAILQ_HEAD_INITIALIZER(qemu_netdev_opts
.head
),
309 * no elements => accept any params
310 * validation will happen later
312 { /* end of list */ }
316 static QemuOptsList qemu_net_opts
= {
318 .implied_opt_name
= "type",
319 .head
= QTAILQ_HEAD_INITIALIZER(qemu_net_opts
.head
),
322 * no elements => accept any params
323 * validation will happen later
325 { /* end of list */ }
329 static QemuOptsList qemu_rtc_opts
= {
331 .head
= QTAILQ_HEAD_INITIALIZER(qemu_rtc_opts
.head
),
335 .type
= QEMU_OPT_STRING
,
338 .type
= QEMU_OPT_STRING
,
341 .type
= QEMU_OPT_STRING
,
343 { /* end of list */ }
347 static QemuOptsList qemu_global_opts
= {
349 .head
= QTAILQ_HEAD_INITIALIZER(qemu_global_opts
.head
),
353 .type
= QEMU_OPT_STRING
,
356 .type
= QEMU_OPT_STRING
,
359 .type
= QEMU_OPT_STRING
,
361 { /* end of list */ }
365 QemuOptsList qemu_sandbox_opts
= {
367 .implied_opt_name
= "enable",
368 .head
= QTAILQ_HEAD_INITIALIZER(qemu_sandbox_opts
.head
),
372 .type
= QEMU_OPT_BOOL
,
374 { /* end of list */ }
378 static QemuOptsList qemu_mon_opts
= {
380 .implied_opt_name
= "chardev",
381 .head
= QTAILQ_HEAD_INITIALIZER(qemu_mon_opts
.head
),
385 .type
= QEMU_OPT_STRING
,
388 .type
= QEMU_OPT_STRING
,
391 .type
= QEMU_OPT_BOOL
,
394 .type
= QEMU_OPT_BOOL
,
396 { /* end of list */ }
400 static QemuOptsList qemu_trace_opts
= {
402 .implied_opt_name
= "trace",
403 .head
= QTAILQ_HEAD_INITIALIZER(qemu_trace_opts
.head
),
407 .type
= QEMU_OPT_STRING
,
410 .type
= QEMU_OPT_STRING
,
412 { /* end of list */ }
416 static QemuOptsList qemu_cpudef_opts
= {
418 .head
= QTAILQ_HEAD_INITIALIZER(qemu_cpudef_opts
.head
),
422 .type
= QEMU_OPT_STRING
,
425 .type
= QEMU_OPT_NUMBER
,
428 .type
= QEMU_OPT_STRING
,
431 .type
= QEMU_OPT_NUMBER
,
434 .type
= QEMU_OPT_NUMBER
,
437 .type
= QEMU_OPT_NUMBER
,
439 .name
= "feature_edx", /* cpuid 0000_0001.edx */
440 .type
= QEMU_OPT_STRING
,
442 .name
= "feature_ecx", /* cpuid 0000_0001.ecx */
443 .type
= QEMU_OPT_STRING
,
445 .name
= "extfeature_edx", /* cpuid 8000_0001.edx */
446 .type
= QEMU_OPT_STRING
,
448 .name
= "extfeature_ecx", /* cpuid 8000_0001.ecx */
449 .type
= QEMU_OPT_STRING
,
452 .type
= QEMU_OPT_NUMBER
,
455 .type
= QEMU_OPT_STRING
,
457 .name
= "vendor_override",
458 .type
= QEMU_OPT_NUMBER
,
460 { /* end of list */ }
464 QemuOptsList qemu_spice_opts
= {
466 .head
= QTAILQ_HEAD_INITIALIZER(qemu_spice_opts
.head
),
470 .type
= QEMU_OPT_NUMBER
,
473 .type
= QEMU_OPT_NUMBER
,
476 .type
= QEMU_OPT_STRING
,
479 .type
= QEMU_OPT_BOOL
,
482 .type
= QEMU_OPT_BOOL
,
485 .type
= QEMU_OPT_STRING
,
487 .name
= "disable-ticketing",
488 .type
= QEMU_OPT_BOOL
,
490 .name
= "disable-copy-paste",
491 .type
= QEMU_OPT_BOOL
,
494 .type
= QEMU_OPT_BOOL
,
497 .type
= QEMU_OPT_STRING
,
499 .name
= "x509-key-file",
500 .type
= QEMU_OPT_STRING
,
502 .name
= "x509-key-password",
503 .type
= QEMU_OPT_STRING
,
505 .name
= "x509-cert-file",
506 .type
= QEMU_OPT_STRING
,
508 .name
= "x509-cacert-file",
509 .type
= QEMU_OPT_STRING
,
511 .name
= "x509-dh-key-file",
512 .type
= QEMU_OPT_STRING
,
514 .name
= "tls-ciphers",
515 .type
= QEMU_OPT_STRING
,
517 .name
= "tls-channel",
518 .type
= QEMU_OPT_STRING
,
520 .name
= "plaintext-channel",
521 .type
= QEMU_OPT_STRING
,
523 .name
= "image-compression",
524 .type
= QEMU_OPT_STRING
,
526 .name
= "jpeg-wan-compression",
527 .type
= QEMU_OPT_STRING
,
529 .name
= "zlib-glz-wan-compression",
530 .type
= QEMU_OPT_STRING
,
532 .name
= "streaming-video",
533 .type
= QEMU_OPT_STRING
,
535 .name
= "agent-mouse",
536 .type
= QEMU_OPT_BOOL
,
538 .name
= "playback-compression",
539 .type
= QEMU_OPT_BOOL
,
541 { /* end of list */ }
545 QemuOptsList qemu_option_rom_opts
= {
546 .name
= "option-rom",
547 .implied_opt_name
= "romfile",
548 .head
= QTAILQ_HEAD_INITIALIZER(qemu_option_rom_opts
.head
),
552 .type
= QEMU_OPT_NUMBER
,
555 .type
= QEMU_OPT_STRING
,
557 { /* end of list */ }
561 static QemuOptsList qemu_machine_opts
= {
563 .implied_opt_name
= "type",
565 .head
= QTAILQ_HEAD_INITIALIZER(qemu_machine_opts
.head
),
569 .type
= QEMU_OPT_STRING
,
570 .help
= "emulated machine"
573 .type
= QEMU_OPT_STRING
,
574 .help
= "accelerator list",
576 .name
= "kernel_irqchip",
577 .type
= QEMU_OPT_BOOL
,
578 .help
= "use KVM in-kernel irqchip",
580 .name
= "kvm_shadow_mem",
581 .type
= QEMU_OPT_SIZE
,
582 .help
= "KVM shadow MMU size",
585 .type
= QEMU_OPT_STRING
,
586 .help
= "Linux kernel image file",
589 .type
= QEMU_OPT_STRING
,
590 .help
= "Linux initial ramdisk file",
593 .type
= QEMU_OPT_STRING
,
594 .help
= "Linux kernel command line",
597 .type
= QEMU_OPT_STRING
,
598 .help
= "Linux kernel device tree file",
601 .type
= QEMU_OPT_STRING
,
602 .help
= "Dump current dtb to a file and quit",
604 .name
= "phandle_start",
605 .type
= QEMU_OPT_STRING
,
606 .help
= "The first phandle ID we may generate dynamically",
608 .name
= "dt_compatible",
609 .type
= QEMU_OPT_STRING
,
610 .help
= "Overrides the \"compatible\" property of the dt root node",
612 .name
= "dump-guest-core",
613 .type
= QEMU_OPT_BOOL
,
614 .help
= "Include guest memory in a core dump",
616 { /* End of list */ }
620 QemuOptsList qemu_boot_opts
= {
622 .head
= QTAILQ_HEAD_INITIALIZER(qemu_boot_opts
.head
),
624 /* the three names below are not used now */
627 .type
= QEMU_OPT_STRING
,
630 .type
= QEMU_OPT_STRING
,
633 .type
= QEMU_OPT_STRING
,
634 /* following are really used */
637 .type
= QEMU_OPT_STRING
,
639 .name
= "splash-time",
640 .type
= QEMU_OPT_STRING
,
646 static QemuOptsList
*vm_config_groups
[32] = {
657 &qemu_option_rom_opts
,
665 static QemuOptsList
*find_list(QemuOptsList
**lists
, const char *group
,
670 for (i
= 0; lists
[i
] != NULL
; i
++) {
671 if (strcmp(lists
[i
]->name
, group
) == 0)
674 if (lists
[i
] == NULL
) {
675 error_set(errp
, QERR_INVALID_OPTION_GROUP
, group
);
680 QemuOptsList
*qemu_find_opts(const char *group
)
683 Error
*local_err
= NULL
;
685 ret
= find_list(vm_config_groups
, group
, &local_err
);
686 if (error_is_set(&local_err
)) {
687 error_report("%s\n", error_get_pretty(local_err
));
688 error_free(local_err
);
694 QemuOptsList
*qemu_find_opts_err(const char *group
, Error
**errp
)
696 return find_list(vm_config_groups
, group
, errp
);
699 void qemu_add_opts(QemuOptsList
*list
)
703 entries
= ARRAY_SIZE(vm_config_groups
);
704 entries
--; /* keep list NULL terminated */
705 for (i
= 0; i
< entries
; i
++) {
706 if (vm_config_groups
[i
] == NULL
) {
707 vm_config_groups
[i
] = list
;
711 fprintf(stderr
, "ran out of space in vm_config_groups");
715 int qemu_set_option(const char *str
)
717 char group
[64], id
[64], arg
[64];
722 rc
= sscanf(str
, "%63[^.].%63[^.].%63[^=]%n", group
, id
, arg
, &offset
);
723 if (rc
< 3 || str
[offset
] != '=') {
724 error_report("can't parse: \"%s\"", str
);
728 list
= qemu_find_opts(group
);
733 opts
= qemu_opts_find(list
, id
);
735 error_report("there is no %s \"%s\" defined",
740 if (qemu_opt_set(opts
, arg
, str
+offset
+1) == -1) {
746 int qemu_global_option(const char *str
)
748 char driver
[64], property
[64];
752 rc
= sscanf(str
, "%63[^.].%63[^=]%n", driver
, property
, &offset
);
753 if (rc
< 2 || str
[offset
] != '=') {
754 error_report("can't parse: \"%s\"", str
);
758 opts
= qemu_opts_create(&qemu_global_opts
, NULL
, 0, NULL
);
759 qemu_opt_set(opts
, "driver", driver
);
760 qemu_opt_set(opts
, "property", property
);
761 qemu_opt_set(opts
, "value", str
+offset
+1);
765 struct ConfigWriteData
{
770 static int config_write_opt(const char *name
, const char *value
, void *opaque
)
772 struct ConfigWriteData
*data
= opaque
;
774 fprintf(data
->fp
, " %s = \"%s\"\n", name
, value
);
778 static int config_write_opts(QemuOpts
*opts
, void *opaque
)
780 struct ConfigWriteData
*data
= opaque
;
781 const char *id
= qemu_opts_id(opts
);
784 fprintf(data
->fp
, "[%s \"%s\"]\n", data
->list
->name
, id
);
786 fprintf(data
->fp
, "[%s]\n", data
->list
->name
);
788 qemu_opt_foreach(opts
, config_write_opt
, data
, 0);
789 fprintf(data
->fp
, "\n");
793 void qemu_config_write(FILE *fp
)
795 struct ConfigWriteData data
= { .fp
= fp
};
796 QemuOptsList
**lists
= vm_config_groups
;
799 fprintf(fp
, "# qemu config file\n\n");
800 for (i
= 0; lists
[i
] != NULL
; i
++) {
801 data
.list
= lists
[i
];
802 qemu_opts_foreach(data
.list
, config_write_opts
, &data
, 0);
806 int qemu_config_parse(FILE *fp
, QemuOptsList
**lists
, const char *fname
)
808 char line
[1024], group
[64], id
[64], arg
[64], value
[1024];
810 QemuOptsList
*list
= NULL
;
811 Error
*local_err
= NULL
;
812 QemuOpts
*opts
= NULL
;
813 int res
= -1, lno
= 0;
816 while (fgets(line
, sizeof(line
), fp
) != NULL
) {
817 loc_set_file(fname
, ++lno
);
818 if (line
[0] == '\n') {
819 /* skip empty lines */
822 if (line
[0] == '#') {
826 if (sscanf(line
, "[%63s \"%63[^\"]\"]", group
, id
) == 2) {
828 list
= find_list(lists
, group
, &local_err
);
829 if (error_is_set(&local_err
)) {
830 error_report("%s\n", error_get_pretty(local_err
));
831 error_free(local_err
);
834 opts
= qemu_opts_create(list
, id
, 1, NULL
);
837 if (sscanf(line
, "[%63[^]]]", group
) == 1) {
838 /* group without id */
839 list
= find_list(lists
, group
, &local_err
);
840 if (error_is_set(&local_err
)) {
841 error_report("%s\n", error_get_pretty(local_err
));
842 error_free(local_err
);
845 opts
= qemu_opts_create(list
, NULL
, 0, NULL
);
848 if (sscanf(line
, " %63s = \"%1023[^\"]\"", arg
, value
) == 2) {
851 error_report("no group defined");
854 if (qemu_opt_set(opts
, arg
, value
) != 0) {
859 error_report("parse error");
863 error_report("error reading file");
872 int qemu_read_config_file(const char *filename
)
874 FILE *f
= fopen(filename
, "r");
881 ret
= qemu_config_parse(f
, vm_config_groups
, filename
);