blob: a59568d07077c711d3218733d44a1605e8ded2ab [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"
Paolo Bonzinib4a42f82013-02-04 11:37:52 +01005#include "qapi/qmp/qerror.h"
Gerd Hoffmannd0fef6f2009-12-08 13:11:34 +01006#include "hw/qdev.h"
Paolo Bonzini7b1b5d12012-12-17 18:19:43 +01007#include "qapi/error.h"
Amos Kong1f8f9872013-04-25 17:50:35 +08008#include "qmp-commands.h"
Gerd Hoffmann7282a032009-07-31 12:25:35 +02009
Paolo Bonzini4d454572012-11-26 16:03:42 +010010static QemuOptsList *vm_config_groups[32];
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020011
Luiz Capitulino2ac20612012-03-28 14:14:17 -030012static QemuOptsList *find_list(QemuOptsList **lists, const char *group,
13 Error **errp)
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020014{
Gerd Hoffmannddc97852009-10-14 10:39:25 +020015 int i;
Gerd Hoffmannd058fe02009-07-31 12:25:36 +020016
17 for (i = 0; lists[i] != NULL; i++) {
18 if (strcmp(lists[i]->name, group) == 0)
19 break;
20 }
21 if (lists[i] == NULL) {
Luiz Capitulino2ac20612012-03-28 14:14:17 -030022 error_set(errp, QERR_INVALID_OPTION_GROUP, group);
Gerd Hoffmannddc97852009-10-14 10:39:25 +020023 }
24 return lists[i];
25}
26
Kevin Wolf490b6482010-03-05 18:21:56 +010027QemuOptsList *qemu_find_opts(const char *group)
28{
Luiz Capitulino2ac20612012-03-28 14:14:17 -030029 QemuOptsList *ret;
30 Error *local_err = NULL;
31
32 ret = find_list(vm_config_groups, group, &local_err);
33 if (error_is_set(&local_err)) {
Markus Armbruster312fd5f2013-02-08 21:22:16 +010034 error_report("%s", error_get_pretty(local_err));
Luiz Capitulino2ac20612012-03-28 14:14:17 -030035 error_free(local_err);
36 }
37
38 return ret;
Kevin Wolf490b6482010-03-05 18:21:56 +010039}
40
Amos Kong1f8f9872013-04-25 17:50:35 +080041static CommandLineParameterInfoList *query_option_descs(const QemuOptDesc *desc)
42{
43 CommandLineParameterInfoList *param_list = NULL, *entry;
44 CommandLineParameterInfo *info;
45 int i;
46
47 for (i = 0; desc[i].name != NULL; i++) {
48 info = g_malloc0(sizeof(*info));
49 info->name = g_strdup(desc[i].name);
50
51 switch (desc[i].type) {
52 case QEMU_OPT_STRING:
53 info->type = COMMAND_LINE_PARAMETER_TYPE_STRING;
54 break;
55 case QEMU_OPT_BOOL:
56 info->type = COMMAND_LINE_PARAMETER_TYPE_BOOLEAN;
57 break;
58 case QEMU_OPT_NUMBER:
59 info->type = COMMAND_LINE_PARAMETER_TYPE_NUMBER;
60 break;
61 case QEMU_OPT_SIZE:
62 info->type = COMMAND_LINE_PARAMETER_TYPE_SIZE;
63 break;
64 }
65
66 if (desc[i].help) {
67 info->has_help = true;
68 info->help = g_strdup(desc[i].help);
69 }
70
71 entry = g_malloc0(sizeof(*entry));
72 entry->value = info;
73 entry->next = param_list;
74 param_list = entry;
75 }
76
77 return param_list;
78}
79
80CommandLineOptionInfoList *qmp_query_command_line_options(bool has_option,
81 const char *option,
82 Error **errp)
83{
84 CommandLineOptionInfoList *conf_list = NULL, *entry;
85 CommandLineOptionInfo *info;
86 int i;
87
88 for (i = 0; vm_config_groups[i] != NULL; i++) {
89 if (!has_option || !strcmp(option, vm_config_groups[i]->name)) {
90 info = g_malloc0(sizeof(*info));
91 info->option = g_strdup(vm_config_groups[i]->name);
92 info->parameters = query_option_descs(vm_config_groups[i]->desc);
93 entry = g_malloc0(sizeof(*entry));
94 entry->value = info;
95 entry->next = conf_list;
96 conf_list = entry;
97 }
98 }
99
100 if (conf_list == NULL) {
101 error_setg(errp, "invalid option name: %s", option);
102 }
103
104 return conf_list;
105}
106
Luiz Capitulino60d56662012-03-28 14:16:37 -0300107QemuOptsList *qemu_find_opts_err(const char *group, Error **errp)
108{
109 return find_list(vm_config_groups, group, errp);
110}
111
Gerd Hoffmanndfe795e2010-08-20 13:52:00 +0200112void qemu_add_opts(QemuOptsList *list)
113{
114 int entries, i;
115
116 entries = ARRAY_SIZE(vm_config_groups);
117 entries--; /* keep list NULL terminated */
118 for (i = 0; i < entries; i++) {
119 if (vm_config_groups[i] == NULL) {
120 vm_config_groups[i] = list;
121 return;
122 }
123 }
124 fprintf(stderr, "ran out of space in vm_config_groups");
125 abort();
126}
127
Gerd Hoffmannddc97852009-10-14 10:39:25 +0200128int qemu_set_option(const char *str)
129{
130 char group[64], id[64], arg[64];
131 QemuOptsList *list;
132 QemuOpts *opts;
133 int rc, offset;
134
135 rc = sscanf(str, "%63[^.].%63[^.].%63[^=]%n", group, id, arg, &offset);
136 if (rc < 3 || str[offset] != '=') {
Markus Armbruster1ecda022010-02-18 17:25:24 +0100137 error_report("can't parse: \"%s\"", str);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +0200138 return -1;
139 }
140
Markus Armbruster304329e2010-02-10 20:09:14 +0100141 list = qemu_find_opts(group);
Gerd Hoffmannddc97852009-10-14 10:39:25 +0200142 if (list == NULL) {
143 return -1;
144 }
145
146 opts = qemu_opts_find(list, id);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +0200147 if (!opts) {
Markus Armbruster1ecda022010-02-18 17:25:24 +0100148 error_report("there is no %s \"%s\" defined",
149 list->name, id);
Gerd Hoffmannd058fe02009-07-31 12:25:36 +0200150 return -1;
151 }
152
Mark McLoughlin3df04ac2009-09-23 11:24:05 +0100153 if (qemu_opt_set(opts, arg, str+offset+1) == -1) {
Gerd Hoffmannd058fe02009-07-31 12:25:36 +0200154 return -1;
155 }
156 return 0;
157}
158
Gerd Hoffmann9d993392009-10-14 10:39:26 +0200159struct ConfigWriteData {
160 QemuOptsList *list;
161 FILE *fp;
162};
163
164static int config_write_opt(const char *name, const char *value, void *opaque)
165{
166 struct ConfigWriteData *data = opaque;
167
168 fprintf(data->fp, " %s = \"%s\"\n", name, value);
169 return 0;
170}
171
172static int config_write_opts(QemuOpts *opts, void *opaque)
173{
174 struct ConfigWriteData *data = opaque;
175 const char *id = qemu_opts_id(opts);
176
177 if (id) {
178 fprintf(data->fp, "[%s \"%s\"]\n", data->list->name, id);
179 } else {
180 fprintf(data->fp, "[%s]\n", data->list->name);
181 }
182 qemu_opt_foreach(opts, config_write_opt, data, 0);
183 fprintf(data->fp, "\n");
184 return 0;
185}
186
187void qemu_config_write(FILE *fp)
188{
189 struct ConfigWriteData data = { .fp = fp };
Kevin Wolf490b6482010-03-05 18:21:56 +0100190 QemuOptsList **lists = vm_config_groups;
Gerd Hoffmann9d993392009-10-14 10:39:26 +0200191 int i;
192
193 fprintf(fp, "# qemu config file\n\n");
194 for (i = 0; lists[i] != NULL; i++) {
195 data.list = lists[i];
196 qemu_opts_foreach(data.list, config_write_opts, &data, 0);
197 }
198}
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200199
Kevin Wolf490b6482010-03-05 18:21:56 +0100200int qemu_config_parse(FILE *fp, QemuOptsList **lists, const char *fname)
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200201{
202 char line[1024], group[64], id[64], arg[64], value[1024];
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100203 Location loc;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200204 QemuOptsList *list = NULL;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300205 Error *local_err = NULL;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200206 QemuOpts *opts = NULL;
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100207 int res = -1, lno = 0;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200208
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100209 loc_push_none(&loc);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200210 while (fgets(line, sizeof(line), fp) != NULL) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100211 loc_set_file(fname, ++lno);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200212 if (line[0] == '\n') {
213 /* skip empty lines */
214 continue;
215 }
216 if (line[0] == '#') {
217 /* comment */
218 continue;
219 }
220 if (sscanf(line, "[%63s \"%63[^\"]\"]", group, id) == 2) {
221 /* group with id */
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300222 list = find_list(lists, group, &local_err);
223 if (error_is_set(&local_err)) {
Markus Armbruster312fd5f2013-02-08 21:22:16 +0100224 error_report("%s", error_get_pretty(local_err));
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300225 error_free(local_err);
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100226 goto out;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300227 }
Luiz Capitulino8be7e7e2012-03-20 15:51:57 -0300228 opts = qemu_opts_create(list, id, 1, NULL);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200229 continue;
230 }
231 if (sscanf(line, "[%63[^]]]", group) == 1) {
232 /* group without id */
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300233 list = find_list(lists, group, &local_err);
234 if (error_is_set(&local_err)) {
Markus Armbruster312fd5f2013-02-08 21:22:16 +0100235 error_report("%s", error_get_pretty(local_err));
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300236 error_free(local_err);
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100237 goto out;
Luiz Capitulino2ac20612012-03-28 14:14:17 -0300238 }
Dong Xu Wange478b442012-12-06 14:47:22 +0800239 opts = qemu_opts_create_nofail(list);
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200240 continue;
241 }
242 if (sscanf(line, " %63s = \"%1023[^\"]\"", arg, value) == 2) {
243 /* arg = value */
244 if (opts == NULL) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100245 error_report("no group defined");
246 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200247 }
248 if (qemu_opt_set(opts, arg, value) != 0) {
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100249 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200250 }
251 continue;
252 }
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100253 error_report("parse error");
254 goto out;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200255 }
Markus Armbrusteref825162010-02-18 19:56:01 +0100256 if (ferror(fp)) {
257 error_report("error reading file");
258 goto out;
259 }
Markus Armbrustercf5a65a2010-02-18 19:48:33 +0100260 res = 0;
261out:
262 loc_pop(&loc);
263 return res;
Gerd Hoffmann42262ba2009-10-14 10:39:27 +0200264}
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100265
266int qemu_read_config_file(const char *filename)
267{
268 FILE *f = fopen(filename, "r");
Kevin Wolf019e78b2010-05-17 10:36:47 +0200269 int ret;
270
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100271 if (f == NULL) {
272 return -errno;
273 }
274
Kevin Wolf019e78b2010-05-17 10:36:47 +0200275 ret = qemu_config_parse(f, vm_config_groups, filename);
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100276 fclose(f);
277
Kevin Wolf019e78b2010-05-17 10:36:47 +0200278 if (ret == 0) {
279 return 0;
280 } else {
281 return -EINVAL;
282 }
Kevin Wolfdcfb0932010-03-05 17:25:55 +0100283}