qapi-types.py 11.4 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 17 18

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

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

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

35
    return mcgen('''
36

37 38 39 40
typedef struct %(name)s %(name)s;

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

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

63 64
def generate_struct_fields(members):
    ret = ''
65

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

77 78
    return ret

79 80
def generate_struct(expr):

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

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

92 93 94
    if base:
        ret += generate_struct_fields({'base': base})

95 96
    ret += generate_struct_fields(members)

97 98 99 100 101 102 103 104
    # 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;
''')

105 106 107 108 109 110 111 112 113 114 115 116 117
    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 已提交
118
                name=c_name(name))
119 120
    i = 0
    for value in values:
121
        index = c_enum_const(name, value)
122
        ret += mcgen('''
123
    [%(index)s] = "%(value)s",
124
''',
125
                     index = index, value = value)
126

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

132 133
''',
        max_index=max_index)
134 135 136
    return ret

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

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

149 150 151
    # append automatically generated _MAX value
    enum_values = values + [ 'MAX' ]

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

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

    return lookup_decl + enum_decl

169
def generate_alternate_qtypes(expr):
K
Kevin Wolf 已提交
170

171
    name = expr['alternate']
K
Kevin Wolf 已提交
172 173 174 175 176 177 178 179
    members = expr['data']

    ret = mcgen('''
const int %(name)s_qtypes[QTYPE_MAX] = {
''',
    name=name)

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

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

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


194
def generate_union(expr, meta):
195

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

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

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

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

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

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

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

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

246 247 248 249 250

    return ret

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

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

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

    if (!obj) {
        return;
    }

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


try:
279
    opts, args = getopt.gnu_getopt(sys.argv[1:], "chbp:i:o:",
280
                                   ["source", "header", "builtins",
281
                                    "prefix=", "input-file=", "output-dir="])
282 283 284 285 286
except getopt.GetoptError, err:
    print str(err)
    sys.exit(1)

output_dir = ""
287
input_file = ""
288 289 290 291
prefix = ""
c_file = 'qapi-types.c'
h_file = 'qapi-types.h'

292 293
do_c = False
do_h = False
294
do_builtins = False
295

296 297 298
for o, a in opts:
    if o in ("-p", "--prefix"):
        prefix = a
299 300
    elif o in ("-i", "--input-file"):
        input_file = a
301 302
    elif o in ("-o", "--output-dir"):
        output_dir = a + "/"
303 304
    elif o in ("-c", "--source"):
        do_c = True
A
Avi Kivity 已提交
305 306
    elif o in ("-h", "--header"):
        do_h = True
307 308
    elif o in ("-b", "--builtins"):
        do_builtins = True
309 310 311 312

if not do_c and not do_h:
    do_c = True
    do_h = True
313 314 315 316 317 318 319 320 321 322

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

323 324 325
def maybe_open(really, name, opt):
    if really:
        return open(name, opt)
A
Avi Kivity 已提交
326 327 328
    else:
        import StringIO
        return StringIO.StringIO()
329 330 331

fdef = maybe_open(do_c, c_file, 'w')
fdecl = maybe_open(do_h, h_file, 'w')
332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349

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.
 *
 */

350
#include "qapi/dealloc-visitor.h"
351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374
#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

375 376
#include <stdbool.h>
#include <stdint.h>
377

378 379 380
''',
                  guard=guardname(h_file)))

381
exprs = parse_schema(input_file)
382
exprs = filter(lambda expr: not expr.has_key('gen'), exprs)
383

384
fdecl.write(guardstart("QAPI_TYPES_BUILTIN_STRUCT_DECL"))
385
for typename in builtin_types.keys():
386 387 388
    fdecl.write(generate_fwd_struct(typename, None, builtin_type=True))
fdecl.write(guardend("QAPI_TYPES_BUILTIN_STRUCT_DECL"))

389 390
for expr in exprs:
    ret = "\n"
391 392
    if expr.has_key('struct'):
        ret += generate_fwd_struct(expr['struct'], expr['data'])
393
    elif expr.has_key('enum'):
394 395
        ret += generate_enum(expr['enum'], expr['data']) + "\n"
        ret += generate_fwd_enum_struct(expr['enum'], expr['data'])
396 397 398
        fdef.write(generate_enum_lookup(expr['enum'], expr['data']))
    elif expr.has_key('union'):
        ret += generate_fwd_struct(expr['union'], expr['data']) + "\n"
399 400 401 402 403
        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()))
404 405 406 407 408 409
    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))
410 411 412 413
    else:
        continue
    fdecl.write(ret)

414 415 416
# 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"))
417
for typename in builtin_types.keys():
418 419 420 421 422 423 424 425
    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"))
426
    for typename in builtin_types.keys():
427 428 429
        fdef.write(generate_type_cleanup(typename + "List"))
    fdef.write(guardend("QAPI_TYPES_BUILTIN_CLEANUP_DEF"))

430 431
for expr in exprs:
    ret = "\n"
432
    if expr.has_key('struct'):
433
        ret += generate_struct(expr) + "\n"
434 435 436 437
        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")
438
    elif expr.has_key('union'):
439
        ret += generate_union(expr, 'union')
440 441 442 443
        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")
444 445 446 447 448 449
    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")
450 451 452
    elif expr.has_key('enum'):
        ret += generate_type_cleanup_decl(expr['enum'] + "List")
        fdef.write(generate_type_cleanup(expr['enum'] + "List") + "\n")
453 454 455 456 457 458 459 460 461 462
    else:
        continue
    fdecl.write(ret)

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

fdecl.flush()
fdecl.close()
463 464 465

fdef.flush()
fdef.close()