blob: 47c81f72d3711130f7ac2b08cddaf9145af51e4f [file] [log] [blame]
Gerd Hoffmann7282a032009-07-31 12:25:35 +02001#include "qemu-common.h"
Paolo Bonzini1de7afc2012-12-17 18:20:00 +01002#include "qemu/error-report.h"
3#include "qemu/option.h"
4#include "qemu/config-file.h"
Gerd Hoffmannd0fef6f2009-12-08 13:11:34 +01005#include "hw/qdev.h"
Paolo Bonzini7b1b5d12012-12-17 18:19:43 +01006#include "qapi/error.h"
Gerd Hoffmann7282a032009-07-31 12:25:35 +02007
Paolo Bonzini4d454572012-11-26 16:03:42 +01008static QemuOptsList *vm_config_groups[32];
Gerd Hoffmannd058fe02009-07-31 12:25:36 +02009
Luiz Capitulino2ac20612012-03-28 14:14:17 -030010static QemuOptsList *find_list(QemuOptsList **lists, const char *group,
11 Error **errp)
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020012{
Gerd Hoffmannddc97852009-10-14 10:39:25 +020013 int i;
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020014
15 for (i = 0; lists[i] != NULL; i++) {
16 if (strcmp(lists[i]->name, group) == 0)
17 break;
18 }
19 if (lists[i] == NULL) {
Luiz Capitulino2ac20612012-03-28 14:14:17 -030020 error_set(errp, QERR_INVALID_OPTION_GROUP, group);
Gerd Hoffmannddc97852009-10-14 10:39:25 +020021 }
22 return lists[i];
23}
24
Kevin Wolf490b6482010-03-05 18:21:56 +010025QemuOptsList *qemu_find_opts(const char *group)
26{
Luiz Capitulino2ac20612012-03-28 14:14:17 -030027 QemuOptsList *ret;
28 Error *local_err = NULL;
29
30 ret = find_list(vm_config_groups, group, &local_err);
31 if (error_is_set(&local_err)) {
32 error_report("%s\n", error_get_pretty(local_err));
33 error_free(local_err);
34 }
35
36 return ret;
Kevin Wolf490b6482010-03-05 18:21:56 +010037}
38
Luiz Capitulino60d56662012-03-28 14:16:37 -030039QemuOptsList *qemu_find_opts_err(const char *group, Error **errp)
40{
41 return find_list(vm_config_groups, group, errp);
42}
43
Gerd Hoffmanndfe795e2010-08-20 13:52:00 +020044void qemu_add_opts(QemuOptsList *list)
45{
46 int entries, i;
47
48 entries = ARRAY_SIZE(vm_config_groups);
49 entries--; /* keep list NULL terminated */
50 for (i = 0; i < entries; i++) {
51 if (vm_config_groups[i] == NULL) {
52 vm_config_groups[i] = list;
53 return;
54 }
55 }
56 fprintf(stderr, "ran out of space in vm_config_groups");
57 abort();
58}
59
Gerd Hoffmannddc97852009-10-14 10:39:25 +020060int qemu_set_option(const char *str)
61{
62 char group[64], id[64], arg[64];
63 QemuOptsList *list;
64 QemuOpts *opts;
65 int rc, offset;
66
67 rc = sscanf(str, "%63[^.].%63[^.].%63[^=]%n", group, id, arg, &offset);
68 if (rc < 3 || str[offset] != '=') {
Markus Armbruster1ecda022010-02-18 17:25:24 +010069 error_report("can't parse: \"%s\"", str);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020070 return -1;
71 }
72
Markus Armbruster304329e2010-02-10 20:09:14 +010073 list = qemu_find_opts(group);
Gerd Hoffmannddc97852009-10-14 10:39:25 +020074 if (list == NULL) {
75 return -1;
76 }
77
78 opts = qemu_opts_find(list, id);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020079 if (!opts) {
Markus Armbruster1ecda022010-02-18 17:25:24 +010080 error_report("there is no %s \"%s\" defined",
81 list->name, id);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020082 return -1;
83 }
84
Mark McLoughlin3df04ac2009-09-23 11:24:05 +010085 if (qemu_opt_set(opts, arg, str+offset+1) == -1) {
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020086 return -1;
87 }
88 return 0;
89}
90
Gerd Hoffmann9d993392009-10-14 10:39:26 +020091struct ConfigWriteData {
92 QemuOptsList *list;
93 FILE *fp;
94};
95
96static int config_write_opt(const char *name, const char *value, void *opaque)
97{
98 struct ConfigWriteData *data = opaque;
99
100 fprintf(data->fp, " %s = \"%s\"\n", name, value);
101 return 0;
102}
103
104static int config_write_opts(QemuOpts *opts, void *opaque)
105{
106 struct ConfigWriteData *data = opaque;
107 const char *id = qemu_opts_id(opts);
108
109 if (id) {
110 fprintf(data->fp, "[%s \"%s\"]\n", data->list->name, id);
111 } else {
112 fprintf(data->fp, "[%s]\n", data->list->name);
113 }
114 qemu_opt_foreach(opts, config_write_opt, data, 0);
115 fprintf(data->fp, "\n");
116 return 0;
117}
118
119void qemu_config_write(FILE *fp)
120{
121 struct ConfigWriteData data = { .fp = fp };
Kevin Wolf490b6482010-03-05 18:21:56 +0100122 QemuOptsList **lists = vm_config_groups;
Gerd Hoffmann9d993392009-10-14 10:39:26 +0200123 int i;
124
125 fprintf(fp, "# qemu config file\n\n");
126 for (i = 0; lists[i] != NULL; i++) {
127 data.list = lists[i];
128 qemu_opts_foreach(data.list, config_write_opts, &data, 0);
129 }
130}
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200131
Kevin Wolf490b6482010-03-05 18:21:56 +0100132int qemu_config_parse(FILE *fp, QemuOptsList **lists, const char *fname)
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200133{
134 char line[1024], group[64], id[64], arg[64], value[1024];
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100135 Location loc;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200136 QemuOptsList *list = NULL;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300137 Error *local_err = NULL;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200138 QemuOpts *opts = NULL;
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100139 int res = -1, lno = 0;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200140
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100141 loc_push_none(&loc);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200142 while (fgets(line, sizeof(line), fp) != NULL) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100143 loc_set_file(fname, ++lno);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200144 if (line[0] == '\n') {
145 /* skip empty lines */
146 continue;
147 }
148 if (line[0] == '#') {
149 /* comment */
150 continue;
151 }
152 if (sscanf(line, "[%63s \"%63[^\"]\"]", group, id) == 2) {
153 /* group with id */
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300154 list = find_list(lists, group, &local_err);
155 if (error_is_set(&local_err)) {
156 error_report("%s\n", error_get_pretty(local_err));
157 error_free(local_err);
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100158 goto out;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300159 }
Luiz Capitulino8be7e7e2012-03-20 15:51:57 -0300160 opts = qemu_opts_create(list, id, 1, NULL);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200161 continue;
162 }
163 if (sscanf(line, "[%63[^]]]", group) == 1) {
164 /* group without id */
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300165 list = find_list(lists, group, &local_err);
166 if (error_is_set(&local_err)) {
167 error_report("%s\n", error_get_pretty(local_err));
168 error_free(local_err);
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100169 goto out;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300170 }
Dong Xu Wange478b442012-12-06 14:47:22 +0800171 opts = qemu_opts_create_nofail(list);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200172 continue;
173 }
174 if (sscanf(line, " %63s = \"%1023[^\"]\"", arg, value) == 2) {
175 /* arg = value */
176 if (opts == NULL) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100177 error_report("no group defined");
178 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200179 }
180 if (qemu_opt_set(opts, arg, value) != 0) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100181 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200182 }
183 continue;
184 }
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100185 error_report("parse error");
186 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200187 }
Markus Armbrusteref825162010-02-18 19:56:01 +0100188 if (ferror(fp)) {
189 error_report("error reading file");
190 goto out;
191 }
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100192 res = 0;
193out:
194 loc_pop(&loc);
195 return res;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200196}
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100197
198int qemu_read_config_file(const char *filename)
199{
200 FILE *f = fopen(filename, "r");
Kevin Wolf019e78b2010-05-17 10:36:47 +0200201 int ret;
202
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100203 if (f == NULL) {
204 return -errno;
205 }
206
Kevin Wolf019e78b2010-05-17 10:36:47 +0200207 ret = qemu_config_parse(f, vm_config_groups, filename);
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100208 fclose(f);
209
Kevin Wolf019e78b2010-05-17 10:36:47 +0200210 if (ret == 0) {
211 return 0;
212 } else {
213 return -EINVAL;
214 }
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100215}