root / hw / qdev-properties.c @ 5a053d1f
History | View | Annotate | Download (7.8 kB)
1 |
#include "qdev.h" |
---|---|
2 |
|
3 |
void *qdev_get_prop_ptr(DeviceState *dev, Property *prop)
|
4 |
{ |
5 |
void *ptr = dev;
|
6 |
ptr += prop->offset; |
7 |
return ptr;
|
8 |
} |
9 |
|
10 |
/* --- 16bit integer --- */
|
11 |
|
12 |
static int parse_uint16(DeviceState *dev, Property *prop, const char *str) |
13 |
{ |
14 |
uint16_t *ptr = qdev_get_prop_ptr(dev, prop); |
15 |
const char *fmt; |
16 |
|
17 |
/* accept both hex and decimal */
|
18 |
fmt = strncasecmp(str, "0x",2) == 0 ? "%" PRIx16 : "%" PRIu16; |
19 |
if (sscanf(str, fmt, ptr) != 1) |
20 |
return -1; |
21 |
return 0; |
22 |
} |
23 |
|
24 |
static int print_uint16(DeviceState *dev, Property *prop, char *dest, size_t len) |
25 |
{ |
26 |
uint16_t *ptr = qdev_get_prop_ptr(dev, prop); |
27 |
return snprintf(dest, len, "%" PRIu16, *ptr); |
28 |
} |
29 |
|
30 |
PropertyInfo qdev_prop_uint16 = { |
31 |
.name = "uint16",
|
32 |
.type = PROP_TYPE_UINT16, |
33 |
.size = sizeof(uint16_t),
|
34 |
.parse = parse_uint16, |
35 |
.print = print_uint16, |
36 |
}; |
37 |
|
38 |
/* --- 32bit integer --- */
|
39 |
|
40 |
static int parse_uint32(DeviceState *dev, Property *prop, const char *str) |
41 |
{ |
42 |
uint32_t *ptr = qdev_get_prop_ptr(dev, prop); |
43 |
const char *fmt; |
44 |
|
45 |
/* accept both hex and decimal */
|
46 |
fmt = strncasecmp(str, "0x",2) == 0 ? "%" PRIx32 : "%" PRIu32; |
47 |
if (sscanf(str, fmt, ptr) != 1) |
48 |
return -1; |
49 |
return 0; |
50 |
} |
51 |
|
52 |
static int print_uint32(DeviceState *dev, Property *prop, char *dest, size_t len) |
53 |
{ |
54 |
uint32_t *ptr = qdev_get_prop_ptr(dev, prop); |
55 |
return snprintf(dest, len, "%" PRIu32, *ptr); |
56 |
} |
57 |
|
58 |
PropertyInfo qdev_prop_uint32 = { |
59 |
.name = "uint32",
|
60 |
.type = PROP_TYPE_UINT32, |
61 |
.size = sizeof(uint32_t),
|
62 |
.parse = parse_uint32, |
63 |
.print = print_uint32, |
64 |
}; |
65 |
|
66 |
/* --- 32bit hex value --- */
|
67 |
|
68 |
static int parse_hex32(DeviceState *dev, Property *prop, const char *str) |
69 |
{ |
70 |
uint32_t *ptr = qdev_get_prop_ptr(dev, prop); |
71 |
|
72 |
if (sscanf(str, "%" PRIx32, ptr) != 1) |
73 |
return -1; |
74 |
return 0; |
75 |
} |
76 |
|
77 |
static int print_hex32(DeviceState *dev, Property *prop, char *dest, size_t len) |
78 |
{ |
79 |
uint32_t *ptr = qdev_get_prop_ptr(dev, prop); |
80 |
return snprintf(dest, len, "0x%" PRIx32, *ptr); |
81 |
} |
82 |
|
83 |
PropertyInfo qdev_prop_hex32 = { |
84 |
.name = "hex32",
|
85 |
.type = PROP_TYPE_UINT32, |
86 |
.size = sizeof(uint32_t),
|
87 |
.parse = parse_hex32, |
88 |
.print = print_hex32, |
89 |
}; |
90 |
|
91 |
/* --- 64bit integer --- */
|
92 |
|
93 |
static int parse_uint64(DeviceState *dev, Property *prop, const char *str) |
94 |
{ |
95 |
uint64_t *ptr = qdev_get_prop_ptr(dev, prop); |
96 |
const char *fmt; |
97 |
|
98 |
/* accept both hex and decimal */
|
99 |
fmt = strncasecmp(str, "0x",2) == 0 ? "%" PRIx64 : "%" PRIu64; |
100 |
if (sscanf(str, fmt, ptr) != 1) |
101 |
return -1; |
102 |
return 0; |
103 |
} |
104 |
|
105 |
static int print_uint64(DeviceState *dev, Property *prop, char *dest, size_t len) |
106 |
{ |
107 |
uint64_t *ptr = qdev_get_prop_ptr(dev, prop); |
108 |
return snprintf(dest, len, "%" PRIu64, *ptr); |
109 |
} |
110 |
|
111 |
PropertyInfo qdev_prop_uint64 = { |
112 |
.name = "uint64",
|
113 |
.type = PROP_TYPE_UINT64, |
114 |
.size = sizeof(uint64_t),
|
115 |
.parse = parse_uint64, |
116 |
.print = print_uint64, |
117 |
}; |
118 |
|
119 |
/* --- 64bit hex value --- */
|
120 |
|
121 |
static int parse_hex64(DeviceState *dev, Property *prop, const char *str) |
122 |
{ |
123 |
uint64_t *ptr = qdev_get_prop_ptr(dev, prop); |
124 |
|
125 |
if (sscanf(str, "%" PRIx64, ptr) != 1) |
126 |
return -1; |
127 |
return 0; |
128 |
} |
129 |
|
130 |
static int print_hex64(DeviceState *dev, Property *prop, char *dest, size_t len) |
131 |
{ |
132 |
uint64_t *ptr = qdev_get_prop_ptr(dev, prop); |
133 |
return snprintf(dest, len, "0x%" PRIx64, *ptr); |
134 |
} |
135 |
|
136 |
PropertyInfo qdev_prop_hex64 = { |
137 |
.name = "hex64",
|
138 |
.type = PROP_TYPE_UINT64, |
139 |
.size = sizeof(uint64_t),
|
140 |
.parse = parse_hex64, |
141 |
.print = print_hex64, |
142 |
}; |
143 |
|
144 |
/* --- pointer --- */
|
145 |
|
146 |
static int print_ptr(DeviceState *dev, Property *prop, char *dest, size_t len) |
147 |
{ |
148 |
void **ptr = qdev_get_prop_ptr(dev, prop);
|
149 |
return snprintf(dest, len, "<%p>", *ptr); |
150 |
} |
151 |
|
152 |
PropertyInfo qdev_prop_ptr = { |
153 |
.name = "ptr",
|
154 |
.type = PROP_TYPE_PTR, |
155 |
.size = sizeof(void*), |
156 |
.print = print_ptr, |
157 |
}; |
158 |
|
159 |
/* --- mac address --- */
|
160 |
|
161 |
/*
|
162 |
* accepted syntax versions:
|
163 |
* 01:02:03:04:05:06
|
164 |
* 01-02-03-04-05-06
|
165 |
*/
|
166 |
static int parse_mac(DeviceState *dev, Property *prop, const char *str) |
167 |
{ |
168 |
uint8_t *mac = qdev_get_prop_ptr(dev, prop); |
169 |
int i, pos;
|
170 |
char *p;
|
171 |
|
172 |
for (i = 0, pos = 0; i < 6; i++, pos += 3) { |
173 |
if (!qemu_isxdigit(str[pos]))
|
174 |
return -1; |
175 |
if (!qemu_isxdigit(str[pos+1])) |
176 |
return -1; |
177 |
if (i == 5 && str[pos+2] != '\0') |
178 |
return -1; |
179 |
if (str[pos+2] != ':' && str[pos+2] != '-') |
180 |
return -1; |
181 |
mac[i] = strtol(str+pos, &p, 16);
|
182 |
} |
183 |
return 0; |
184 |
} |
185 |
|
186 |
static int print_mac(DeviceState *dev, Property *prop, char *dest, size_t len) |
187 |
{ |
188 |
uint8_t *mac = qdev_get_prop_ptr(dev, prop); |
189 |
return snprintf(dest, len, "%02x:%02x:%02x:%02x:%02x:%02x", |
190 |
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); |
191 |
} |
192 |
|
193 |
PropertyInfo qdev_prop_macaddr = { |
194 |
.name = "mac-addr",
|
195 |
.type = PROP_TYPE_MACADDR, |
196 |
.size = 6,
|
197 |
.parse = parse_mac, |
198 |
.print = print_mac, |
199 |
}; |
200 |
|
201 |
/* --- public helpers --- */
|
202 |
|
203 |
static Property *qdev_prop_walk(Property *props, const char *name) |
204 |
{ |
205 |
if (!props)
|
206 |
return NULL; |
207 |
while (props->name) {
|
208 |
if (strcmp(props->name, name) == 0) |
209 |
return props;
|
210 |
props++; |
211 |
} |
212 |
return NULL; |
213 |
} |
214 |
|
215 |
static Property *qdev_prop_find(DeviceState *dev, const char *name) |
216 |
{ |
217 |
Property *prop; |
218 |
|
219 |
/* device properties */
|
220 |
prop = qdev_prop_walk(dev->info->props, name); |
221 |
if (prop)
|
222 |
return prop;
|
223 |
|
224 |
/* bus properties */
|
225 |
prop = qdev_prop_walk(dev->parent_bus->info->props, name); |
226 |
if (prop)
|
227 |
return prop;
|
228 |
|
229 |
return NULL; |
230 |
} |
231 |
|
232 |
int qdev_prop_parse(DeviceState *dev, const char *name, const char *value) |
233 |
{ |
234 |
Property *prop; |
235 |
|
236 |
prop = qdev_prop_find(dev, name); |
237 |
if (!prop) {
|
238 |
fprintf(stderr, "property \"%s.%s\" not found\n",
|
239 |
dev->info->name, name); |
240 |
return -1; |
241 |
} |
242 |
if (!prop->info->parse) {
|
243 |
fprintf(stderr, "property \"%s.%s\" has no parser\n",
|
244 |
dev->info->name, name); |
245 |
return -1; |
246 |
} |
247 |
return prop->info->parse(dev, prop, value);
|
248 |
} |
249 |
|
250 |
void qdev_prop_set(DeviceState *dev, const char *name, void *src, enum PropertyType type) |
251 |
{ |
252 |
Property *prop; |
253 |
void *dst;
|
254 |
|
255 |
prop = qdev_prop_find(dev, name); |
256 |
if (!prop) {
|
257 |
fprintf(stderr, "%s: property \"%s.%s\" not found\n",
|
258 |
__FUNCTION__, dev->info->name, name); |
259 |
abort(); |
260 |
} |
261 |
if (prop->info->type != type) {
|
262 |
fprintf(stderr, "%s: property \"%s.%s\" type mismatch\n",
|
263 |
__FUNCTION__, dev->info->name, name); |
264 |
abort(); |
265 |
} |
266 |
dst = qdev_get_prop_ptr(dev, prop); |
267 |
memcpy(dst, src, prop->info->size); |
268 |
} |
269 |
|
270 |
void qdev_prop_set_uint16(DeviceState *dev, const char *name, uint16_t value) |
271 |
{ |
272 |
qdev_prop_set(dev, name, &value, PROP_TYPE_UINT16); |
273 |
} |
274 |
|
275 |
void qdev_prop_set_uint32(DeviceState *dev, const char *name, uint32_t value) |
276 |
{ |
277 |
qdev_prop_set(dev, name, &value, PROP_TYPE_UINT32); |
278 |
} |
279 |
|
280 |
void qdev_prop_set_uint64(DeviceState *dev, const char *name, uint64_t value) |
281 |
{ |
282 |
qdev_prop_set(dev, name, &value, PROP_TYPE_UINT64); |
283 |
} |
284 |
|
285 |
void qdev_prop_set_ptr(DeviceState *dev, const char *name, void *value) |
286 |
{ |
287 |
qdev_prop_set(dev, name, &value, PROP_TYPE_PTR); |
288 |
} |
289 |
|
290 |
void qdev_prop_set_defaults(DeviceState *dev, Property *props)
|
291 |
{ |
292 |
char *dst;
|
293 |
|
294 |
if (!props)
|
295 |
return;
|
296 |
while (props->name) {
|
297 |
if (props->defval) {
|
298 |
dst = qdev_get_prop_ptr(dev, props); |
299 |
memcpy(dst, props->defval, props->info->size); |
300 |
} |
301 |
props++; |
302 |
} |
303 |
} |
304 |
|
305 |
static CompatProperty *compat_props;
|
306 |
|
307 |
void qdev_prop_register_compat(CompatProperty *props)
|
308 |
{ |
309 |
compat_props = props; |
310 |
} |
311 |
|
312 |
void qdev_prop_set_compat(DeviceState *dev)
|
313 |
{ |
314 |
CompatProperty *prop; |
315 |
|
316 |
if (!compat_props) {
|
317 |
return;
|
318 |
} |
319 |
for (prop = compat_props; prop->driver != NULL; prop++) { |
320 |
if (strcmp(dev->info->name, prop->driver) != 0) { |
321 |
continue;
|
322 |
} |
323 |
if (qdev_prop_parse(dev, prop->property, prop->value) != 0) { |
324 |
abort(); |
325 |
} |
326 |
} |
327 |
} |