blob: 5873a05888356edce2f88312c7fb62dfa30e3ce7 [file] [log] [blame]
Wenchao Xia21cd70d2014-06-18 08:43:28 +02001#
2# QAPI event generator
3#
4# Copyright (c) 2014 Wenchao Xia
Markus Armbruster05f43a92015-09-16 13:06:14 +02005# Copyright (c) 2015 Red Hat Inc.
Wenchao Xia21cd70d2014-06-18 08:43:28 +02006#
7# Authors:
8# Wenchao Xia <wenchaoqemu@gmail.com>
Markus Armbruster05f43a92015-09-16 13:06:14 +02009# Markus Armbruster <armbru@redhat.com>
Wenchao Xia21cd70d2014-06-18 08:43:28 +020010#
11# This work is licensed under the terms of the GNU GPL, version 2.
12# See the COPYING file in the top-level directory.
13
Wenchao Xia21cd70d2014-06-18 08:43:28 +020014from qapi import *
Wenchao Xia21cd70d2014-06-18 08:43:28 +020015
Markus Armbrustere98859a2015-09-16 13:06:16 +020016
17def gen_event_send_proto(name, arg_type):
18 api_name = "void qapi_event_send_%s(" % c_name(name).lower()
Wenchao Xia21cd70d2014-06-18 08:43:28 +020019 l = len(api_name)
20
Markus Armbrustere98859a2015-09-16 13:06:16 +020021 if arg_type:
22 for m in arg_type.members:
Markus Armbruster05f43a92015-09-16 13:06:14 +020023 if m.optional:
24 api_name += "bool has_%s,\n" % c_name(m.name)
Wenchao Xia21cd70d2014-06-18 08:43:28 +020025 api_name += "".ljust(l)
26
Markus Armbruster05f43a92015-09-16 13:06:14 +020027 api_name += "%s %s,\n" % (m.type.c_type(is_param=True),
28 c_name(m.name))
Wenchao Xia21cd70d2014-06-18 08:43:28 +020029 api_name += "".ljust(l)
30
31 api_name += "Error **errp)"
Markus Armbrustere98859a2015-09-16 13:06:16 +020032 return api_name
Wenchao Xia21cd70d2014-06-18 08:43:28 +020033
34
Markus Armbrustere98859a2015-09-16 13:06:16 +020035def gen_event_send_decl(name, arg_type):
Wenchao Xia21cd70d2014-06-18 08:43:28 +020036 return mcgen('''
37
Markus Armbrustere98859a2015-09-16 13:06:16 +020038%(proto)s;
Wenchao Xia21cd70d2014-06-18 08:43:28 +020039''',
Markus Armbrustere98859a2015-09-16 13:06:16 +020040 proto=gen_event_send_proto(name, arg_type))
Wenchao Xia21cd70d2014-06-18 08:43:28 +020041
Wenchao Xia21cd70d2014-06-18 08:43:28 +020042
Markus Armbrustere98859a2015-09-16 13:06:16 +020043def gen_event_send(name, arg_type):
44 ret = mcgen('''
45
46%(proto)s
Wenchao Xia21cd70d2014-06-18 08:43:28 +020047{
48 QDict *qmp;
49 Error *local_err = NULL;
50 QMPEventFuncEmit emit;
Markus Armbrustere98859a2015-09-16 13:06:16 +020051''',
52 proto=gen_event_send_proto(name, arg_type))
Wenchao Xia21cd70d2014-06-18 08:43:28 +020053
Markus Armbrustere98859a2015-09-16 13:06:16 +020054 if arg_type and arg_type.members:
55 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +020056 QmpOutputVisitor *qov;
57 Visitor *v;
58 QObject *obj;
59
Markus Armbrustere98859a2015-09-16 13:06:16 +020060''')
Wenchao Xia21cd70d2014-06-18 08:43:28 +020061
Markus Armbrustere98859a2015-09-16 13:06:16 +020062 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +020063 emit = qmp_event_get_func_emit();
64 if (!emit) {
65 return;
66 }
67
Markus Armbrustere98859a2015-09-16 13:06:16 +020068 qmp = qmp_event_build_dict("%(name)s");
Wenchao Xia21cd70d2014-06-18 08:43:28 +020069
Markus Armbrustere98859a2015-09-16 13:06:16 +020070''',
71 name=name)
Wenchao Xia21cd70d2014-06-18 08:43:28 +020072
Markus Armbrustere98859a2015-09-16 13:06:16 +020073 if arg_type and arg_type.members:
74 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +020075 qov = qmp_output_visitor_new();
76 g_assert(qov);
77
78 v = qmp_output_get_visitor(qov);
79 g_assert(v);
80
81 /* Fake visit, as if all members are under a structure */
Markus Armbrustere98859a2015-09-16 13:06:16 +020082 visit_start_struct(v, NULL, "", "%(name)s", 0, &local_err);
Wenchao Xia21cd70d2014-06-18 08:43:28 +020083 if (local_err) {
84 goto clean;
85 }
86
Markus Armbrustere98859a2015-09-16 13:06:16 +020087''',
88 name=name)
Wenchao Xia21cd70d2014-06-18 08:43:28 +020089
Markus Armbrustere98859a2015-09-16 13:06:16 +020090 for memb in arg_type.members:
Markus Armbruster05f43a92015-09-16 13:06:14 +020091 if memb.optional:
Markus Armbrustere98859a2015-09-16 13:06:16 +020092 ret += mcgen('''
93 if (has_%(c_name)s) {
94''',
95 c_name=c_name(memb.name))
Wenchao Xia21cd70d2014-06-18 08:43:28 +020096 push_indent()
97
Markus Armbrustere98859a2015-09-16 13:06:16 +020098 # Ugly: need to cast away the const
Markus Armbruster05f43a92015-09-16 13:06:14 +020099 if memb.type.name == "str":
Markus Armbrustere98859a2015-09-16 13:06:16 +0200100 cast = '(char **)'
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200101 else:
Markus Armbrustere98859a2015-09-16 13:06:16 +0200102 cast = ''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200103
Markus Armbrustere98859a2015-09-16 13:06:16 +0200104 ret += mcgen('''
105 visit_type_%(c_type)s(v, %(cast)s&%(c_name)s, "%(name)s", &local_err);
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200106 if (local_err) {
107 goto clean;
108 }
Markus Armbrustere98859a2015-09-16 13:06:16 +0200109''',
110 cast=cast,
111 c_name=c_name(memb.name),
112 c_type=memb.type.c_name(),
Markus Armbruster05f43a92015-09-16 13:06:14 +0200113 name=memb.name)
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200114
Markus Armbruster05f43a92015-09-16 13:06:14 +0200115 if memb.optional:
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200116 pop_indent()
Markus Armbrustere98859a2015-09-16 13:06:16 +0200117 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200118 }
Markus Armbrustere98859a2015-09-16 13:06:16 +0200119''')
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200120
Markus Armbrustere98859a2015-09-16 13:06:16 +0200121 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200122
123 visit_end_struct(v, &local_err);
124 if (local_err) {
125 goto clean;
126 }
127
128 obj = qmp_output_get_qobject(qov);
129 g_assert(obj != NULL);
130
131 qdict_put_obj(qmp, "data", obj);
Markus Armbrustere98859a2015-09-16 13:06:16 +0200132''')
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200133
Markus Armbrustere98859a2015-09-16 13:06:16 +0200134 ret += mcgen('''
135 emit(%(c_enum)s, qmp, &local_err);
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200136
Markus Armbrustere98859a2015-09-16 13:06:16 +0200137''',
138 c_enum=c_enum_const(event_enum_name, name))
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200139
Markus Armbrustere98859a2015-09-16 13:06:16 +0200140 if arg_type and arg_type.members:
141 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200142 clean:
143 qmp_output_visitor_cleanup(qov);
Markus Armbrustere98859a2015-09-16 13:06:16 +0200144''')
145 ret += mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200146 error_propagate(errp, local_err);
147 QDECREF(qmp);
148}
Markus Armbrustere98859a2015-09-16 13:06:16 +0200149''')
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200150 return ret
151
Markus Armbruster05f43a92015-09-16 13:06:14 +0200152
153class QAPISchemaGenEventVisitor(QAPISchemaVisitor):
154 def __init__(self):
155 self.decl = None
156 self.defn = None
157 self._event_names = None
158
159 def visit_begin(self, schema):
160 self.decl = ''
161 self.defn = ''
162 self._event_names = []
163
164 def visit_end(self):
Markus Armbrustere98859a2015-09-16 13:06:16 +0200165 self.decl += gen_enum(event_enum_name, self._event_names)
166 self.defn += gen_enum_lookup(event_enum_name, self._event_names)
Markus Armbruster05f43a92015-09-16 13:06:14 +0200167 self._event_names = None
168
169 def visit_event(self, name, info, arg_type):
Markus Armbrustere98859a2015-09-16 13:06:16 +0200170 self.decl += gen_event_send_decl(name, arg_type)
171 self.defn += gen_event_send(name, arg_type)
Markus Armbruster05f43a92015-09-16 13:06:14 +0200172 self._event_names.append(name)
173
174
Markus Armbruster2114f5a2015-04-02 13:12:21 +0200175(input_file, output_dir, do_c, do_h, prefix, dummy) = parse_command_line()
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200176
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200177c_comment = '''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200178/*
179 * schema-defined QAPI event functions
180 *
181 * Copyright (c) 2014 Wenchao Xia
182 *
183 * Authors:
184 * Wenchao Xia <wenchaoqemu@gmail.com>
185 *
186 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
187 * See the COPYING.LIB file in the top-level directory.
188 *
189 */
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200190'''
191h_comment = '''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200192/*
193 * schema-defined QAPI event functions
194 *
195 * Copyright (c) 2014 Wenchao Xia
196 *
197 * Authors:
198 * Wenchao Xia <wenchaoqemu@gmail.com>
199 *
200 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
201 * See the COPYING.LIB file in the top-level directory.
202 *
203 */
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200204'''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200205
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200206(fdef, fdecl) = open_output(output_dir, do_c, do_h, prefix,
207 'qapi-event.c', 'qapi-event.h',
208 c_comment, h_comment)
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200209
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200210fdef.write(mcgen('''
211#include "qemu-common.h"
212#include "%(prefix)sqapi-event.h"
213#include "%(prefix)sqapi-visit.h"
214#include "qapi/qmp-output-visitor.h"
215#include "qapi/qmp-event.h"
216
217''',
218 prefix=prefix))
219
220fdecl.write(mcgen('''
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200221#include "qapi/error.h"
222#include "qapi/qmp/qdict.h"
223#include "%(prefix)sqapi-types.h"
224
225''',
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200226 prefix=prefix))
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200227
Markus Armbruster016a3352015-07-01 12:59:40 +0200228event_enum_name = c_name(prefix + "QAPIEvent", protect=False)
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200229
Markus Armbruster05f43a92015-09-16 13:06:14 +0200230schema = QAPISchema(input_file)
231gen = QAPISchemaGenEventVisitor()
232schema.visit(gen)
233fdef.write(gen.defn)
234fdecl.write(gen.decl)
Wenchao Xia21cd70d2014-06-18 08:43:28 +0200235
Markus Armbruster12f8e1b2015-04-02 14:46:39 +0200236close_output(fdef, fdecl)