qapi-types.py 11.3 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 48 49
    struct %(name)sList *next;
} %(name)sList;
''',
                 name=name)

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 61 62
    struct %(name)sList *next;
} %(name)sList;
''',
                 name=name)

63 64
def generate_struct_fields(members):
    ret = ''
65 66 67 68 69 70 71 72 73

    for argname, argentry, optional, structured in parse_args(members):
        if optional:
            ret += mcgen('''
    bool has_%(c_name)s;
''',
                         c_name=c_var(argname))
        if structured:
            push_indent()
74
            ret += generate_struct({ "field": argname, "data": argentry})
75 76 77 78 79 80 81
            pop_indent()
        else:
            ret += mcgen('''
    %(c_type)s %(c_name)s;
''',
                     c_type=c_type(argentry), c_name=c_var(argname))

82 83
    return ret

84 85 86 87 88
def generate_struct(expr):

    structname = expr.get('type', "")
    fieldname = expr.get('field', "")
    members = expr['data']
89
    base = expr.get('base')
90

91 92 93 94 95 96
    ret = mcgen('''
struct %(name)s
{
''',
          name=structname)

97 98 99
    if base:
        ret += generate_struct_fields({'base': base})

100 101
    ret += generate_struct_fields(members)

102 103 104 105 106 107 108 109
    # 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;
''')

110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
    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[] = {
''',
                         name=name)
    i = 0
    for value in values:
126
        index = generate_enum_full_value(name, value)
127
        ret += mcgen('''
128
    [%(index)s] = "%(value)s",
129
''',
130
                     index = index, value = value)
131

132
    max_index = generate_enum_full_value(name, 'MAX')
133
    ret += mcgen('''
134
    [%(max_index)s] = NULL,
135 136
};

137 138
''',
        max_index=max_index)
139 140 141 142 143 144 145 146 147 148 149 150 151 152
    return ret

def generate_enum(name, values):
    lookup_decl = mcgen('''
extern const char *%(name)s_lookup[];
''',
                name=name)

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

153 154 155
    # append automatically generated _MAX value
    enum_values = values + [ 'MAX' ]

156
    i = 0
157
    for value in enum_values:
158
        enum_full_value = generate_enum_full_value(name, value)
159
        enum_decl += mcgen('''
160
    %(enum_full_value)s = %(i)d,
161
''',
162
                     enum_full_value = enum_full_value,
163 164 165 166 167 168 169 170 171 172
                     i=i)
        i += 1

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

    return lookup_decl + enum_decl

K
Kevin Wolf 已提交
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
def generate_anon_union_qtypes(expr):

    name = expr['union']
    members = expr['data']

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

    for key in members:
        qapi_type = members[key]
        if builtin_type_qtypes.has_key(qapi_type):
            qtype = builtin_type_qtypes[qapi_type]
        elif find_struct(qapi_type):
            qtype = "QTYPE_QDICT"
        elif find_union(qapi_type):
            qtype = "QTYPE_QDICT"
191 192
        elif find_enum(qapi_type):
            qtype = "QTYPE_QSTRING"
K
Kevin Wolf 已提交
193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
        else:
            assert False, "Invalid anonymous union member"

        ret += mcgen('''
    [ %(qtype)s ] = %(abbrev)s_KIND_%(enum)s,
''',
        qtype = qtype,
        abbrev = de_camel_case(name).upper(),
        enum = c_fun(de_camel_case(key),False).upper())

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


209 210 211 212
def generate_union(expr):

    name = expr['union']
    typeinfo = expr['data']
213

214
    base = expr.get('base')
215
    discriminator = expr.get('discriminator')
216

217 218 219 220 221 222
    enum_define = discriminator_find_enum_define(expr)
    if enum_define:
        discriminator_type_name = enum_define['enum_name']
    else:
        discriminator_type_name = '%sKind' % (name)

223 224 225
    ret = mcgen('''
struct %(name)s
{
226
    %(discriminator_type_name)s kind;
227
    union {
228
        void *data;
229
''',
230 231
                name=name,
                discriminator_type_name=discriminator_type_name)
232 233 234 235 236 237

    for key in typeinfo:
        ret += mcgen('''
        %(c_type)s %(c_name)s;
''',
                     c_type=c_type(typeinfo[key]),
238
                     c_name=c_fun(key))
239 240 241

    ret += mcgen('''
    };
242 243 244
''')

    if base:
245 246 247 248 249 250 251
        base_fields = find_struct(base)['data']
        if discriminator:
            base_fields = base_fields.copy()
            del base_fields[discriminator]
        ret += generate_struct_fields(base_fields)
    else:
        assert not discriminator
252 253

    ret += mcgen('''
254 255
};
''')
K
Kevin Wolf 已提交
256 257 258 259 260 261
    if discriminator == {}:
        ret += mcgen('''
extern const int %(name)s_qtypes[];
''',
            name=name)

262 263 264 265 266 267 268 269 270 271 272 273

    return ret

def generate_type_cleanup_decl(name):
    ret = mcgen('''
void qapi_free_%(type)s(%(c_type)s obj);
''',
                c_type=c_type(name),type=name)
    return ret

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

275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294
void qapi_free_%(type)s(%(c_type)s obj)
{
    QapiDeallocVisitor *md;
    Visitor *v;

    if (!obj) {
        return;
    }

    md = qapi_dealloc_visitor_new();
    v = qapi_dealloc_get_visitor(md);
    visit_type_%(type)s(v, &obj, NULL, NULL);
    qapi_dealloc_visitor_cleanup(md);
}
''',
                c_type=c_type(name),type=name)
    return ret


try:
295
    opts, args = getopt.gnu_getopt(sys.argv[1:], "chbp:i:o:",
296
                                   ["source", "header", "builtins",
297
                                    "prefix=", "input-file=", "output-dir="])
298 299 300 301 302
except getopt.GetoptError, err:
    print str(err)
    sys.exit(1)

output_dir = ""
303
input_file = ""
304 305 306 307
prefix = ""
c_file = 'qapi-types.c'
h_file = 'qapi-types.h'

308 309
do_c = False
do_h = False
310
do_builtins = False
311

312 313 314
for o, a in opts:
    if o in ("-p", "--prefix"):
        prefix = a
315 316
    elif o in ("-i", "--input-file"):
        input_file = a
317 318
    elif o in ("-o", "--output-dir"):
        output_dir = a + "/"
319 320
    elif o in ("-c", "--source"):
        do_c = True
A
Avi Kivity 已提交
321 322
    elif o in ("-h", "--header"):
        do_h = True
323 324
    elif o in ("-b", "--builtins"):
        do_builtins = True
325 326 327 328

if not do_c and not do_h:
    do_c = True
    do_h = True
329 330 331 332 333 334 335 336 337 338

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

339 340 341
def maybe_open(really, name, opt):
    if really:
        return open(name, opt)
A
Avi Kivity 已提交
342 343 344
    else:
        import StringIO
        return StringIO.StringIO()
345 346 347

fdef = maybe_open(do_c, c_file, 'w')
fdecl = maybe_open(do_h, h_file, 'w')
348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365

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

366
#include "qapi/dealloc-visitor.h"
367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390
#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

391 392
#include <stdbool.h>
#include <stdint.h>
393

394 395 396
''',
                  guard=guardname(h_file)))

397
exprs = parse_schema(input_file)
398
exprs = filter(lambda expr: not expr.has_key('gen'), exprs)
399

400 401 402 403 404
fdecl.write(guardstart("QAPI_TYPES_BUILTIN_STRUCT_DECL"))
for typename in builtin_types:
    fdecl.write(generate_fwd_struct(typename, None, builtin_type=True))
fdecl.write(guardend("QAPI_TYPES_BUILTIN_STRUCT_DECL"))

405 406 407 408 409
for expr in exprs:
    ret = "\n"
    if expr.has_key('type'):
        ret += generate_fwd_struct(expr['type'], expr['data'])
    elif expr.has_key('enum'):
410 411
        ret += generate_enum(expr['enum'], expr['data']) + "\n"
        ret += generate_fwd_enum_struct(expr['enum'], expr['data'])
412 413 414
        fdef.write(generate_enum_lookup(expr['enum'], expr['data']))
    elif expr.has_key('union'):
        ret += generate_fwd_struct(expr['union'], expr['data']) + "\n"
415 416 417 418 419
        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()))
K
Kevin Wolf 已提交
420 421
        if expr.get('discriminator') == {}:
            fdef.write(generate_anon_union_qtypes(expr))
422 423 424 425
    else:
        continue
    fdecl.write(ret)

426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441
# 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"))
for typename in builtin_types:
    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"))
    for typename in builtin_types:
        fdef.write(generate_type_cleanup(typename + "List"))
    fdef.write(guardend("QAPI_TYPES_BUILTIN_CLEANUP_DEF"))

442 443 444
for expr in exprs:
    ret = "\n"
    if expr.has_key('type'):
445
        ret += generate_struct(expr) + "\n"
446 447
        ret += generate_type_cleanup_decl(expr['type'] + "List")
        fdef.write(generate_type_cleanup(expr['type'] + "List") + "\n")
448 449 450
        ret += generate_type_cleanup_decl(expr['type'])
        fdef.write(generate_type_cleanup(expr['type']) + "\n")
    elif expr.has_key('union'):
451
        ret += generate_union(expr)
452 453 454 455
        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")
456 457 458
    elif expr.has_key('enum'):
        ret += generate_type_cleanup_decl(expr['enum'] + "List")
        fdef.write(generate_type_cleanup(expr['enum'] + "List") + "\n")
459 460 461 462 463 464 465 466 467 468
    else:
        continue
    fdecl.write(ret)

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

fdecl.flush()
fdecl.close()
469 470 471

fdef.flush()
fdef.close()