qapi-types.py 10.9 KB
Newer Older
1 2 3 4 5 6 7 8
#
# QAPI types generator
#
# Copyright IBM, Corp. 2011
#
# Authors:
#  Anthony Liguori <aliguori@us.ibm.com>
#
9 10
# This work is licensed under the terms of the GNU GPL, version 2.
# See the COPYING file in the top-level directory.
11 12 13 14 15 16

from ordereddict import OrderedDict
from qapi import *
import os
import errno

17 18 19 20 21 22
def generate_fwd_struct(name, members, builtin_type=False):
    if builtin_type:
        return mcgen('''

typedef struct %(name)sList
{
23 24 25 26
    union {
        %(type)s value;
        uint64_t padding;
    };
27 28 29 30 31 32
    struct %(name)sList *next;
} %(name)sList;
''',
                     type=c_type(name),
                     name=name)

33
    return mcgen('''
34

35 36 37 38
typedef struct %(name)s %(name)s;

typedef struct %(name)sList
{
39 40 41 42
    union {
        %(name)s *value;
        uint64_t padding;
    };
43 44 45
    struct %(name)sList *next;
} %(name)sList;
''',
E
Eric Blake 已提交
46
                 name=c_name(name))
47

48 49 50 51
def generate_fwd_enum_struct(name, members):
    return mcgen('''
typedef struct %(name)sList
{
52 53 54 55
    union {
        %(name)s value;
        uint64_t padding;
    };
56 57 58
    struct %(name)sList *next;
} %(name)sList;
''',
E
Eric Blake 已提交
59
                 name=c_name(name))
60

61 62
def generate_struct_fields(members):
    ret = ''
63

64
    for argname, argentry, optional in parse_args(members):
65 66 67 68
        if optional:
            ret += mcgen('''
    bool has_%(c_name)s;
''',
69
                         c_name=c_name(argname))
70
        ret += mcgen('''
71 72
    %(c_type)s %(c_name)s;
''',
73
                     c_type=c_type(argentry), c_name=c_name(argname))
74

75 76
    return ret

77 78
def generate_struct(expr):

79
    structname = expr.get('struct', "")
80 81
    fieldname = expr.get('field', "")
    members = expr['data']
82
    base = expr.get('base')
83

84 85 86 87
    ret = mcgen('''
struct %(name)s
{
''',
E
Eric Blake 已提交
88
          name=c_name(structname))
89

90 91 92
    if base:
        ret += generate_struct_fields({'base': base})

93 94
    ret += generate_struct_fields(members)

95 96 97 98 99 100 101 102
    # Make sure that all structs have at least one field; this avoids
    # potential issues with attempting to malloc space for zero-length structs
    # in C, and also incompatibility with C++ (where an empty struct is size 1).
    if not base and not members:
            ret += mcgen('''
    char qapi_dummy_field_for_empty_struct;
''')

103 104 105 106 107 108 109 110 111 112 113 114 115
    if len(fieldname):
        fieldname = " " + fieldname
    ret += mcgen('''
}%(field)s;
''',
            field=fieldname)

    return ret

def generate_enum_lookup(name, values):
    ret = mcgen('''
const char *%(name)s_lookup[] = {
''',
E
Eric Blake 已提交
116
                name=c_name(name))
117 118
    i = 0
    for value in values:
119
        index = c_enum_const(name, value)
120
        ret += mcgen('''
121
    [%(index)s] = "%(value)s",
122
''',
123
                     index = index, value = value)
124

125
    max_index = c_enum_const(name, 'MAX')
126
    ret += mcgen('''
127
    [%(max_index)s] = NULL,
128 129
};

130 131
''',
        max_index=max_index)
132 133 134
    return ret

def generate_enum(name, values):
E
Eric Blake 已提交
135
    name = c_name(name)
136 137 138 139 140 141 142 143 144 145 146
    lookup_decl = mcgen('''
extern const char *%(name)s_lookup[];
''',
                name=name)

    enum_decl = mcgen('''
typedef enum %(name)s
{
''',
                name=name)

147 148 149
    # append automatically generated _MAX value
    enum_values = values + [ 'MAX' ]

150
    i = 0
151
    for value in enum_values:
152
        enum_full_value = c_enum_const(name, value)
153
        enum_decl += mcgen('''
154
    %(enum_full_value)s = %(i)d,
155
''',
156
                     enum_full_value = enum_full_value,
157 158 159 160 161 162 163 164 165 166
                     i=i)
        i += 1

    enum_decl += mcgen('''
} %(name)s;
''',
                 name=name)

    return lookup_decl + enum_decl

167
def generate_alternate_qtypes(expr):
K
Kevin Wolf 已提交
168

169
    name = expr['alternate']
K
Kevin Wolf 已提交
170 171 172 173 174
    members = expr['data']

    ret = mcgen('''
const int %(name)s_qtypes[QTYPE_MAX] = {
''',
E
Eric Blake 已提交
175
                name=c_name(name))
K
Kevin Wolf 已提交
176 177

    for key in members:
178
        qtype = find_alternate_member_qtype(members[key])
179
        assert qtype, "Invalid alternate member"
K
Kevin Wolf 已提交
180 181

        ret += mcgen('''
E
Eric Blake 已提交
182
    [%(qtype)s] = %(enum_const)s,
K
Kevin Wolf 已提交
183
''',
E
Eric Blake 已提交
184 185
                     qtype = qtype,
                     enum_const = c_enum_const(name + 'Kind', key))
K
Kevin Wolf 已提交
186 187 188 189 190 191 192

    ret += mcgen('''
};
''')
    return ret


193
def generate_union(expr, meta):
194

195
    name = c_name(expr[meta])
196
    typeinfo = expr['data']
197

198
    base = expr.get('base')
199
    discriminator = expr.get('discriminator')
200

201 202 203 204 205 206
    enum_define = discriminator_find_enum_define(expr)
    if enum_define:
        discriminator_type_name = enum_define['enum_name']
    else:
        discriminator_type_name = '%sKind' % (name)

207 208 209
    ret = mcgen('''
struct %(name)s
{
210
    %(discriminator_type_name)s kind;
211
    union {
212
        void *data;
213
''',
214
                name=name,
215
                discriminator_type_name=c_name(discriminator_type_name))
216 217 218 219 220 221

    for key in typeinfo:
        ret += mcgen('''
        %(c_type)s %(c_name)s;
''',
                     c_type=c_type(typeinfo[key]),
222
                     c_name=c_name(key))
223 224 225

    ret += mcgen('''
    };
226 227 228
''')

    if base:
229 230 231
        assert discriminator
        base_fields = find_struct(base)['data'].copy()
        del base_fields[discriminator]
232 233 234
        ret += generate_struct_fields(base_fields)
    else:
        assert not discriminator
235 236

    ret += mcgen('''
237 238
};
''')
239
    if meta == 'alternate':
K
Kevin Wolf 已提交
240 241 242 243 244
        ret += mcgen('''
extern const int %(name)s_qtypes[];
''',
            name=name)

245 246 247 248 249

    return ret

def generate_type_cleanup_decl(name):
    ret = mcgen('''
E
Eric Blake 已提交
250
void qapi_free_%(name)s(%(c_type)s obj);
251
''',
E
Eric Blake 已提交
252
                c_type=c_type(name), name=c_name(name))
253 254 255 256
    return ret

def generate_type_cleanup(name):
    ret = mcgen('''
257

E
Eric Blake 已提交
258
void qapi_free_%(name)s(%(c_type)s obj)
259 260 261 262 263 264 265 266 267 268
{
    QapiDeallocVisitor *md;
    Visitor *v;

    if (!obj) {
        return;
    }

    md = qapi_dealloc_visitor_new();
    v = qapi_dealloc_get_visitor(md);
E
Eric Blake 已提交
269
    visit_type_%(name)s(v, &obj, NULL, NULL);
270 271 272
    qapi_dealloc_visitor_cleanup(md);
}
''',
E
Eric Blake 已提交
273
                c_type=c_type(name), name=c_name(name))
274 275 276 277
    return ret

c_file = 'qapi-types.c'
h_file = 'qapi-types.h'
278
do_builtins = False
279

280 281 282
(input_file, output_dir, do_c, do_h, prefix, opts) = \
    parse_command_line("b", ["builtins"])

283
for o, a in opts:
284
    if o in ("-b", "--builtins"):
285
        do_builtins = True
286

287 288 289 290 291 292 293 294 295
c_file = output_dir + prefix + c_file
h_file = output_dir + prefix + h_file

try:
    os.makedirs(output_dir)
except os.error, e:
    if e.errno != errno.EEXIST:
        raise

296 297 298
def maybe_open(really, name, opt):
    if really:
        return open(name, opt)
A
Avi Kivity 已提交
299 300 301
    else:
        import StringIO
        return StringIO.StringIO()
302 303 304

fdef = maybe_open(do_c, c_file, 'w')
fdecl = maybe_open(do_h, h_file, 'w')
305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322

fdef.write(mcgen('''
/* AUTOMATICALLY GENERATED, DO NOT MODIFY */

/*
 * deallocation functions for schema-defined QAPI types
 *
 * Copyright IBM, Corp. 2011
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *  Michael Roth      <mdroth@linux.vnet.ibm.com>
 *
 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
 * See the COPYING.LIB file in the top-level directory.
 *
 */

323
#include "qapi/dealloc-visitor.h"
324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347
#include "%(prefix)sqapi-types.h"
#include "%(prefix)sqapi-visit.h"

''',             prefix=prefix))

fdecl.write(mcgen('''
/* AUTOMATICALLY GENERATED, DO NOT MODIFY */

/*
 * schema-defined QAPI types
 *
 * Copyright IBM, Corp. 2011
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *
 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
 * See the COPYING.LIB file in the top-level directory.
 *
 */

#ifndef %(guard)s
#define %(guard)s

348 349
#include <stdbool.h>
#include <stdint.h>
350

351 352 353
''',
                  guard=guardname(h_file)))

354
exprs = parse_schema(input_file)
355
exprs = filter(lambda expr: not expr.has_key('gen'), exprs)
356

357
fdecl.write(guardstart("QAPI_TYPES_BUILTIN_STRUCT_DECL"))
358
for typename in builtin_types.keys():
359 360 361
    fdecl.write(generate_fwd_struct(typename, None, builtin_type=True))
fdecl.write(guardend("QAPI_TYPES_BUILTIN_STRUCT_DECL"))

362 363
for expr in exprs:
    ret = "\n"
364 365
    if expr.has_key('struct'):
        ret += generate_fwd_struct(expr['struct'], expr['data'])
366
    elif expr.has_key('enum'):
367 368
        ret += generate_enum(expr['enum'], expr['data']) + "\n"
        ret += generate_fwd_enum_struct(expr['enum'], expr['data'])
369 370 371
        fdef.write(generate_enum_lookup(expr['enum'], expr['data']))
    elif expr.has_key('union'):
        ret += generate_fwd_struct(expr['union'], expr['data']) + "\n"
372 373 374 375 376
        enum_define = discriminator_find_enum_define(expr)
        if not enum_define:
            ret += generate_enum('%sKind' % expr['union'], expr['data'].keys())
            fdef.write(generate_enum_lookup('%sKind' % expr['union'],
                                            expr['data'].keys()))
377 378 379 380 381 382
    elif expr.has_key('alternate'):
        ret += generate_fwd_struct(expr['alternate'], expr['data']) + "\n"
        ret += generate_enum('%sKind' % expr['alternate'], expr['data'].keys())
        fdef.write(generate_enum_lookup('%sKind' % expr['alternate'],
                                        expr['data'].keys()))
        fdef.write(generate_alternate_qtypes(expr))
383 384 385 386
    else:
        continue
    fdecl.write(ret)

387 388 389
# to avoid header dependency hell, we always generate declarations
# for built-in types in our header files and simply guard them
fdecl.write(guardstart("QAPI_TYPES_BUILTIN_CLEANUP_DECL"))
390
for typename in builtin_types.keys():
391 392 393 394 395 396 397 398
    fdecl.write(generate_type_cleanup_decl(typename + "List"))
fdecl.write(guardend("QAPI_TYPES_BUILTIN_CLEANUP_DECL"))

# ...this doesn't work for cases where we link in multiple objects that
# have the functions defined, so we use -b option to provide control
# over these cases
if do_builtins:
    fdef.write(guardstart("QAPI_TYPES_BUILTIN_CLEANUP_DEF"))
399
    for typename in builtin_types.keys():
400 401 402
        fdef.write(generate_type_cleanup(typename + "List"))
    fdef.write(guardend("QAPI_TYPES_BUILTIN_CLEANUP_DEF"))

403 404
for expr in exprs:
    ret = "\n"
405
    if expr.has_key('struct'):
406
        ret += generate_struct(expr) + "\n"
407 408 409 410
        ret += generate_type_cleanup_decl(expr['struct'] + "List")
        fdef.write(generate_type_cleanup(expr['struct'] + "List") + "\n")
        ret += generate_type_cleanup_decl(expr['struct'])
        fdef.write(generate_type_cleanup(expr['struct']) + "\n")
411
    elif expr.has_key('union'):
412
        ret += generate_union(expr, 'union')
413 414 415 416
        ret += generate_type_cleanup_decl(expr['union'] + "List")
        fdef.write(generate_type_cleanup(expr['union'] + "List") + "\n")
        ret += generate_type_cleanup_decl(expr['union'])
        fdef.write(generate_type_cleanup(expr['union']) + "\n")
417 418 419 420 421 422
    elif expr.has_key('alternate'):
        ret += generate_union(expr, 'alternate')
        ret += generate_type_cleanup_decl(expr['alternate'] + "List")
        fdef.write(generate_type_cleanup(expr['alternate'] + "List") + "\n")
        ret += generate_type_cleanup_decl(expr['alternate'])
        fdef.write(generate_type_cleanup(expr['alternate']) + "\n")
423 424 425
    elif expr.has_key('enum'):
        ret += generate_type_cleanup_decl(expr['enum'] + "List")
        fdef.write(generate_type_cleanup(expr['enum'] + "List") + "\n")
426 427 428 429 430 431 432 433 434 435
    else:
        continue
    fdecl.write(ret)

fdecl.write('''
#endif
''')

fdecl.flush()
fdecl.close()
436 437 438

fdef.flush()
fdef.close()