Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 1 | # |
| 2 | # QAPI event generator |
| 3 | # |
| 4 | # Copyright (c) 2014 Wenchao Xia |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 5 | # Copyright (c) 2015 Red Hat Inc. |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 6 | # |
| 7 | # Authors: |
| 8 | # Wenchao Xia <wenchaoqemu@gmail.com> |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 9 | # Markus Armbruster <armbru@redhat.com> |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 10 | # |
| 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 Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 14 | from qapi import * |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 15 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 16 | |
| 17 | def gen_event_send_proto(name, arg_type): |
Markus Armbruster | 03b4367 | 2015-09-16 13:06:20 +0200 | [diff] [blame] | 18 | return 'void qapi_event_send_%(c_name)s(%(param)s)' % { |
| 19 | 'c_name': c_name(name.lower()), |
| 20 | 'param': gen_params(arg_type, 'Error **errp')} |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 21 | |
| 22 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 23 | def gen_event_send_decl(name, arg_type): |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 24 | return mcgen(''' |
| 25 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 26 | %(proto)s; |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 27 | ''', |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 28 | proto=gen_event_send_proto(name, arg_type)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 29 | |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 30 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 31 | def gen_event_send(name, arg_type): |
| 32 | ret = mcgen(''' |
| 33 | |
| 34 | %(proto)s |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 35 | { |
| 36 | QDict *qmp; |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 37 | Error *err = NULL; |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 38 | QMPEventFuncEmit emit; |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 39 | ''', |
| 40 | proto=gen_event_send_proto(name, arg_type)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 41 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 42 | if arg_type and arg_type.members: |
| 43 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 44 | QmpOutputVisitor *qov; |
| 45 | Visitor *v; |
| 46 | QObject *obj; |
| 47 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 48 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 49 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 50 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 51 | emit = qmp_event_get_func_emit(); |
| 52 | if (!emit) { |
| 53 | return; |
| 54 | } |
| 55 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 56 | qmp = qmp_event_build_dict("%(name)s"); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 57 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 58 | ''', |
| 59 | name=name) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 60 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 61 | if arg_type and arg_type.members: |
| 62 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 63 | qov = qmp_output_visitor_new(); |
| 64 | g_assert(qov); |
| 65 | |
| 66 | v = qmp_output_get_visitor(qov); |
| 67 | g_assert(v); |
| 68 | |
| 69 | /* Fake visit, as if all members are under a structure */ |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 70 | visit_start_struct(v, NULL, "", "%(name)s", 0, &err); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 71 | ''', |
| 72 | name=name) |
Eric Blake | 1f35334 | 2015-09-29 16:21:13 -0600 | [diff] [blame^] | 73 | ret += gen_err_check() |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 74 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 75 | for memb in arg_type.members: |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 76 | if memb.optional: |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 77 | ret += mcgen(''' |
| 78 | if (has_%(c_name)s) { |
| 79 | ''', |
| 80 | c_name=c_name(memb.name)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 81 | push_indent() |
| 82 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 83 | # Ugly: need to cast away the const |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 84 | if memb.type.name == "str": |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 85 | cast = '(char **)' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 86 | else: |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 87 | cast = '' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 88 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 89 | ret += mcgen(''' |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 90 | visit_type_%(c_type)s(v, %(cast)s&%(c_name)s, "%(name)s", &err); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 91 | ''', |
| 92 | cast=cast, |
| 93 | c_name=c_name(memb.name), |
| 94 | c_type=memb.type.c_name(), |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 95 | name=memb.name) |
Eric Blake | 1f35334 | 2015-09-29 16:21:13 -0600 | [diff] [blame^] | 96 | ret += gen_err_check() |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 97 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 98 | if memb.optional: |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 99 | pop_indent() |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 100 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 101 | } |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 102 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 103 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 104 | ret += mcgen(''' |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 105 | visit_end_struct(v, &err); |
| 106 | if (err) { |
Eric Blake | f782399 | 2015-09-29 16:21:10 -0600 | [diff] [blame] | 107 | goto out; |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 108 | } |
| 109 | |
| 110 | obj = qmp_output_get_qobject(qov); |
| 111 | g_assert(obj != NULL); |
| 112 | |
| 113 | qdict_put_obj(qmp, "data", obj); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 114 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 115 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 116 | ret += mcgen(''' |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 117 | emit(%(c_enum)s, qmp, &err); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 118 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 119 | ''', |
| 120 | c_enum=c_enum_const(event_enum_name, name)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 121 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 122 | if arg_type and arg_type.members: |
| 123 | ret += mcgen(''' |
Eric Blake | f782399 | 2015-09-29 16:21:10 -0600 | [diff] [blame] | 124 | out: |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 125 | qmp_output_visitor_cleanup(qov); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 126 | ''') |
| 127 | ret += mcgen(''' |
Eric Blake | 2a0f50e | 2015-09-29 16:21:08 -0600 | [diff] [blame] | 128 | error_propagate(errp, err); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 129 | QDECREF(qmp); |
| 130 | } |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 131 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 132 | return ret |
| 133 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 134 | |
| 135 | class QAPISchemaGenEventVisitor(QAPISchemaVisitor): |
| 136 | def __init__(self): |
| 137 | self.decl = None |
| 138 | self.defn = None |
| 139 | self._event_names = None |
| 140 | |
| 141 | def visit_begin(self, schema): |
| 142 | self.decl = '' |
| 143 | self.defn = '' |
| 144 | self._event_names = [] |
| 145 | |
| 146 | def visit_end(self): |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 147 | self.decl += gen_enum(event_enum_name, self._event_names) |
| 148 | self.defn += gen_enum_lookup(event_enum_name, self._event_names) |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 149 | self._event_names = None |
| 150 | |
| 151 | def visit_event(self, name, info, arg_type): |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 152 | self.decl += gen_event_send_decl(name, arg_type) |
| 153 | self.defn += gen_event_send(name, arg_type) |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 154 | self._event_names.append(name) |
| 155 | |
| 156 | |
Markus Armbruster | 2114f5a | 2015-04-02 13:12:21 +0200 | [diff] [blame] | 157 | (input_file, output_dir, do_c, do_h, prefix, dummy) = parse_command_line() |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 158 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 159 | c_comment = ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 160 | /* |
| 161 | * schema-defined QAPI event functions |
| 162 | * |
| 163 | * Copyright (c) 2014 Wenchao Xia |
| 164 | * |
| 165 | * Authors: |
| 166 | * Wenchao Xia <wenchaoqemu@gmail.com> |
| 167 | * |
| 168 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| 169 | * See the COPYING.LIB file in the top-level directory. |
| 170 | * |
| 171 | */ |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 172 | ''' |
| 173 | h_comment = ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 174 | /* |
| 175 | * schema-defined QAPI event functions |
| 176 | * |
| 177 | * Copyright (c) 2014 Wenchao Xia |
| 178 | * |
| 179 | * Authors: |
| 180 | * Wenchao Xia <wenchaoqemu@gmail.com> |
| 181 | * |
| 182 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| 183 | * See the COPYING.LIB file in the top-level directory. |
| 184 | * |
| 185 | */ |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 186 | ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 187 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 188 | (fdef, fdecl) = open_output(output_dir, do_c, do_h, prefix, |
| 189 | 'qapi-event.c', 'qapi-event.h', |
| 190 | c_comment, h_comment) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 191 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 192 | fdef.write(mcgen(''' |
| 193 | #include "qemu-common.h" |
| 194 | #include "%(prefix)sqapi-event.h" |
| 195 | #include "%(prefix)sqapi-visit.h" |
| 196 | #include "qapi/qmp-output-visitor.h" |
| 197 | #include "qapi/qmp-event.h" |
| 198 | |
| 199 | ''', |
| 200 | prefix=prefix)) |
| 201 | |
| 202 | fdecl.write(mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 203 | #include "qapi/error.h" |
| 204 | #include "qapi/qmp/qdict.h" |
| 205 | #include "%(prefix)sqapi-types.h" |
| 206 | |
| 207 | ''', |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 208 | prefix=prefix)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 209 | |
Markus Armbruster | 016a335 | 2015-07-01 12:59:40 +0200 | [diff] [blame] | 210 | event_enum_name = c_name(prefix + "QAPIEvent", protect=False) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 211 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 212 | schema = QAPISchema(input_file) |
| 213 | gen = QAPISchemaGenEventVisitor() |
| 214 | schema.visit(gen) |
| 215 | fdef.write(gen.defn) |
| 216 | fdecl.write(gen.decl) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 217 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 218 | close_output(fdef, fdecl) |