]> git.proxmox.com Git - mirror_qemu.git/blob - util/qemu-config.c
Merge remote-tracking branch 'bonzini/hw-dirs' into staging
[mirror_qemu.git] / util / qemu-config.c
1 #include "qemu-common.h"
2 #include "qemu/error-report.h"
3 #include "qemu/option.h"
4 #include "qemu/config-file.h"
5 #include "qapi/qmp/qerror.h"
6 #include "hw/qdev.h"
7 #include "qapi/error.h"
8
9 static QemuOptsList *vm_config_groups[32];
10
11 static QemuOptsList *find_list(QemuOptsList **lists, const char *group,
12 Error **errp)
13 {
14 int i;
15
16 for (i = 0; lists[i] != NULL; i++) {
17 if (strcmp(lists[i]->name, group) == 0)
18 break;
19 }
20 if (lists[i] == NULL) {
21 error_set(errp, QERR_INVALID_OPTION_GROUP, group);
22 }
23 return lists[i];
24 }
25
26 QemuOptsList *qemu_find_opts(const char *group)
27 {
28 QemuOptsList *ret;
29 Error *local_err = NULL;
30
31 ret = find_list(vm_config_groups, group, &local_err);
32 if (error_is_set(&local_err)) {
33 error_report("%s", error_get_pretty(local_err));
34 error_free(local_err);
35 }
36
37 return ret;
38 }
39
40 QemuOptsList *qemu_find_opts_err(const char *group, Error **errp)
41 {
42 return find_list(vm_config_groups, group, errp);
43 }
44
45 void qemu_add_opts(QemuOptsList *list)
46 {
47 int entries, i;
48
49 entries = ARRAY_SIZE(vm_config_groups);
50 entries--; /* keep list NULL terminated */
51 for (i = 0; i < entries; i++) {
52 if (vm_config_groups[i] == NULL) {
53 vm_config_groups[i] = list;
54 return;
55 }
56 }
57 fprintf(stderr, "ran out of space in vm_config_groups");
58 abort();
59 }
60
61 int qemu_set_option(const char *str)
62 {
63 char group[64], id[64], arg[64];
64 QemuOptsList *list;
65 QemuOpts *opts;
66 int rc, offset;
67
68 rc = sscanf(str, "%63[^.].%63[^.].%63[^=]%n", group, id, arg, &offset);
69 if (rc < 3 || str[offset] != '=') {
70 error_report("can't parse: \"%s\"", str);
71 return -1;
72 }
73
74 list = qemu_find_opts(group);
75 if (list == NULL) {
76 return -1;
77 }
78
79 opts = qemu_opts_find(list, id);
80 if (!opts) {
81 error_report("there is no %s \"%s\" defined",
82 list->name, id);
83 return -1;
84 }
85
86 if (qemu_opt_set(opts, arg, str+offset+1) == -1) {
87 return -1;
88 }
89 return 0;
90 }
91
92 struct ConfigWriteData {
93 QemuOptsList *list;
94 FILE *fp;
95 };
96
97 static int config_write_opt(const char *name, const char *value, void *opaque)
98 {
99 struct ConfigWriteData *data = opaque;
100
101 fprintf(data->fp, " %s = \"%s\"\n", name, value);
102 return 0;
103 }
104
105 static int config_write_opts(QemuOpts *opts, void *opaque)
106 {
107 struct ConfigWriteData *data = opaque;
108 const char *id = qemu_opts_id(opts);
109
110 if (id) {
111 fprintf(data->fp, "[%s \"%s\"]\n", data->list->name, id);
112 } else {
113 fprintf(data->fp, "[%s]\n", data->list->name);
114 }
115 qemu_opt_foreach(opts, config_write_opt, data, 0);
116 fprintf(data->fp, "\n");
117 return 0;
118 }
119
120 void qemu_config_write(FILE *fp)
121 {
122 struct ConfigWriteData data = { .fp = fp };
123 QemuOptsList **lists = vm_config_groups;
124 int i;
125
126 fprintf(fp, "# qemu config file\n\n");
127 for (i = 0; lists[i] != NULL; i++) {
128 data.list = lists[i];
129 qemu_opts_foreach(data.list, config_write_opts, &data, 0);
130 }
131 }
132
133 int qemu_config_parse(FILE *fp, QemuOptsList **lists, const char *fname)
134 {
135 char line[1024], group[64], id[64], arg[64], value[1024];
136 Location loc;
137 QemuOptsList *list = NULL;
138 Error *local_err = NULL;
139 QemuOpts *opts = NULL;
140 int res = -1, lno = 0;
141
142 loc_push_none(&loc);
143 while (fgets(line, sizeof(line), fp) != NULL) {
144 loc_set_file(fname, ++lno);
145 if (line[0] == '\n') {
146 /* skip empty lines */
147 continue;
148 }
149 if (line[0] == '#') {
150 /* comment */
151 continue;
152 }
153 if (sscanf(line, "[%63s \"%63[^\"]\"]", group, id) == 2) {
154 /* group with id */
155 list = find_list(lists, group, &local_err);
156 if (error_is_set(&local_err)) {
157 error_report("%s", error_get_pretty(local_err));
158 error_free(local_err);
159 goto out;
160 }
161 opts = qemu_opts_create(list, id, 1, NULL);
162 continue;
163 }
164 if (sscanf(line, "[%63[^]]]", group) == 1) {
165 /* group without id */
166 list = find_list(lists, group, &local_err);
167 if (error_is_set(&local_err)) {
168 error_report("%s", error_get_pretty(local_err));
169 error_free(local_err);
170 goto out;
171 }
172 opts = qemu_opts_create_nofail(list);
173 continue;
174 }
175 if (sscanf(line, " %63s = \"%1023[^\"]\"", arg, value) == 2) {
176 /* arg = value */
177 if (opts == NULL) {
178 error_report("no group defined");
179 goto out;
180 }
181 if (qemu_opt_set(opts, arg, value) != 0) {
182 goto out;
183 }
184 continue;
185 }
186 error_report("parse error");
187 goto out;
188 }
189 if (ferror(fp)) {
190 error_report("error reading file");
191 goto out;
192 }
193 res = 0;
194 out:
195 loc_pop(&loc);
196 return res;
197 }
198
199 int qemu_read_config_file(const char *filename)
200 {
201 FILE *f = fopen(filename, "r");
202 int ret;
203
204 if (f == NULL) {
205 return -errno;
206 }
207
208 ret = qemu_config_parse(f, vm_config_groups, filename);
209 fclose(f);
210
211 if (ret == 0) {
212 return 0;
213 } else {
214 return -EINVAL;
215 }
216 }