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; |
| 37 | Error *local_err = NULL; |
| 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 */ |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 70 | visit_start_struct(v, NULL, "", "%(name)s", 0, &local_err); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 71 | if (local_err) { |
| 72 | goto clean; |
| 73 | } |
| 74 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 75 | ''', |
| 76 | name=name) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 77 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 78 | for memb in arg_type.members: |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 79 | if memb.optional: |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 80 | ret += mcgen(''' |
| 81 | if (has_%(c_name)s) { |
| 82 | ''', |
| 83 | c_name=c_name(memb.name)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 84 | push_indent() |
| 85 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 86 | # Ugly: need to cast away the const |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 87 | if memb.type.name == "str": |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 88 | cast = '(char **)' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 89 | else: |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 90 | cast = '' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 91 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 92 | ret += mcgen(''' |
| 93 | visit_type_%(c_type)s(v, %(cast)s&%(c_name)s, "%(name)s", &local_err); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 94 | if (local_err) { |
| 95 | goto clean; |
| 96 | } |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 97 | ''', |
| 98 | cast=cast, |
| 99 | c_name=c_name(memb.name), |
| 100 | c_type=memb.type.c_name(), |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 101 | name=memb.name) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 102 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 103 | if memb.optional: |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 104 | pop_indent() |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 105 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 106 | } |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 107 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 108 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 109 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 110 | |
| 111 | visit_end_struct(v, &local_err); |
| 112 | if (local_err) { |
| 113 | goto clean; |
| 114 | } |
| 115 | |
| 116 | obj = qmp_output_get_qobject(qov); |
| 117 | g_assert(obj != NULL); |
| 118 | |
| 119 | qdict_put_obj(qmp, "data", obj); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 120 | ''') |
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 | ret += mcgen(''' |
| 123 | emit(%(c_enum)s, qmp, &local_err); |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 124 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 125 | ''', |
| 126 | c_enum=c_enum_const(event_enum_name, name)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 127 | |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 128 | if arg_type and arg_type.members: |
| 129 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 130 | clean: |
| 131 | qmp_output_visitor_cleanup(qov); |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 132 | ''') |
| 133 | ret += mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 134 | error_propagate(errp, local_err); |
| 135 | QDECREF(qmp); |
| 136 | } |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 137 | ''') |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 138 | return ret |
| 139 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 140 | |
| 141 | class QAPISchemaGenEventVisitor(QAPISchemaVisitor): |
| 142 | def __init__(self): |
| 143 | self.decl = None |
| 144 | self.defn = None |
| 145 | self._event_names = None |
| 146 | |
| 147 | def visit_begin(self, schema): |
| 148 | self.decl = '' |
| 149 | self.defn = '' |
| 150 | self._event_names = [] |
| 151 | |
| 152 | def visit_end(self): |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 153 | self.decl += gen_enum(event_enum_name, self._event_names) |
| 154 | self.defn += gen_enum_lookup(event_enum_name, self._event_names) |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 155 | self._event_names = None |
| 156 | |
| 157 | def visit_event(self, name, info, arg_type): |
Markus Armbruster | e98859a | 2015-09-16 13:06:16 +0200 | [diff] [blame] | 158 | self.decl += gen_event_send_decl(name, arg_type) |
| 159 | self.defn += gen_event_send(name, arg_type) |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 160 | self._event_names.append(name) |
| 161 | |
| 162 | |
Markus Armbruster | 2114f5a | 2015-04-02 13:12:21 +0200 | [diff] [blame] | 163 | (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] | 164 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 165 | c_comment = ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 166 | /* |
| 167 | * schema-defined QAPI event functions |
| 168 | * |
| 169 | * Copyright (c) 2014 Wenchao Xia |
| 170 | * |
| 171 | * Authors: |
| 172 | * Wenchao Xia <wenchaoqemu@gmail.com> |
| 173 | * |
| 174 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| 175 | * See the COPYING.LIB file in the top-level directory. |
| 176 | * |
| 177 | */ |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 178 | ''' |
| 179 | h_comment = ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 180 | /* |
| 181 | * schema-defined QAPI event functions |
| 182 | * |
| 183 | * Copyright (c) 2014 Wenchao Xia |
| 184 | * |
| 185 | * Authors: |
| 186 | * Wenchao Xia <wenchaoqemu@gmail.com> |
| 187 | * |
| 188 | * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| 189 | * See the COPYING.LIB file in the top-level directory. |
| 190 | * |
| 191 | */ |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 192 | ''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 193 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 194 | (fdef, fdecl) = open_output(output_dir, do_c, do_h, prefix, |
| 195 | 'qapi-event.c', 'qapi-event.h', |
| 196 | c_comment, h_comment) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 197 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 198 | fdef.write(mcgen(''' |
| 199 | #include "qemu-common.h" |
| 200 | #include "%(prefix)sqapi-event.h" |
| 201 | #include "%(prefix)sqapi-visit.h" |
| 202 | #include "qapi/qmp-output-visitor.h" |
| 203 | #include "qapi/qmp-event.h" |
| 204 | |
| 205 | ''', |
| 206 | prefix=prefix)) |
| 207 | |
| 208 | fdecl.write(mcgen(''' |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 209 | #include "qapi/error.h" |
| 210 | #include "qapi/qmp/qdict.h" |
| 211 | #include "%(prefix)sqapi-types.h" |
| 212 | |
| 213 | ''', |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 214 | prefix=prefix)) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 215 | |
Markus Armbruster | 016a335 | 2015-07-01 12:59:40 +0200 | [diff] [blame] | 216 | event_enum_name = c_name(prefix + "QAPIEvent", protect=False) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 217 | |
Markus Armbruster | 05f43a9 | 2015-09-16 13:06:14 +0200 | [diff] [blame] | 218 | schema = QAPISchema(input_file) |
| 219 | gen = QAPISchemaGenEventVisitor() |
| 220 | schema.visit(gen) |
| 221 | fdef.write(gen.defn) |
| 222 | fdecl.write(gen.decl) |
Wenchao Xia | 21cd70d | 2014-06-18 08:43:28 +0200 | [diff] [blame] | 223 | |
Markus Armbruster | 12f8e1b | 2015-04-02 14:46:39 +0200 | [diff] [blame] | 224 | close_output(fdef, fdecl) |