#!/usr/bin/python3 # # Copyright (C) 2017 Mentor Graphics Development (Deutschland) GmbH # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # import sys OUT = sys.stdout def set_output(f): global OUT OUT = f def p(*args): OUT.write('\n'.join(args)) OUT.write('\n') def emit_func_impl(api, f): args = f.get('args', []) if len(args) > 0: p(' json_object *jreq = afb_req_json(req);', '') for arg in args: arg['jtype'] = arg.get('jtype', arg['type']) # add jtype default p(' json_object *j_%(name)s = nullptr;' % arg, ' if (! json_object_object_get_ex(jreq, "%(name)s", &j_%(name)s)) {' % arg, ' afb_req_fail(req, "failed", "Need %(type)s argument %(name)s");' % arg, ' return;', ' }', ' %(type)s a_%(name)s = json_object_get_%(jtype)s(j_%(name)s);' % arg, '') p(' auto ret = %(api)s' % api + '%(name)s(' % f + ', '.join(map(lambda x: 'a_' + x['name'], args)) + ');') p(' if (ret.is_err()) {', ' afb_req_fail(req, "failed", ret.unwrap_err());', ' return;', ' }', '') p(' afb_req_success(req, ret.unwrap(), "success");') def emit_func(api, f): p('void %(impl_name)s(afb_req req) noexcept {' % f) p(' std::lock_guard guard(binding_m);') p(' #ifdef ST') p(' ST();') p(' #endif') p(' if (g_afb_instance == nullptr) {', ' afb_req_fail(req, "failed", "Binding not initialized, did the compositor die?");', ' return;', ' }', '', ' try {', ' // BEGIN impl') emit_func_impl(api, f) p(' // END impl', ' } catch (std::exception &e) {', ' afb_req_fail_f(req, "failed", "Uncaught exception while calling %(name)s: %%s", e.what());' % f, ' return;', ' }', '') p('}', '') def emit_afb_verbs(api): p('const struct afb_verb_v2 %(name)s_verbs[] = {' % api) for f in api['functions']: p(' { "%(name)s", %(impl_name)s, nullptr, nullptr, AFB_SESSION_NONE },' % f) p(' {}', '};') def emit_binding(api): p('namespace {') p('std::mutex binding_m;', '') for func in api['functions']: emit_func(api, func) p('} // namespace', '') emit_afb_verbs(api) def generate_names(api): for f in api['functions']: f['impl_name'] = '%s_%s_thunk' % (api['name'], f['name']) def emit_afb_api(api): p('#include "result.hpp"', '') p('#include ', '') p('namespace wm {', '') p('struct App;', '') p('struct binding_api {') p(' typedef wm::result result_type;') p(' struct wm::App *app;') p(' void send_event(char const *evname, char const *label);') for f in api['functions']: p(' result_type %(name)s(' % f + ', '.join(map(lambda x: '%(type)s %(name)s' % x, f.get('args', []))) + ');') p('};', '') p('} // namespace wm', '') # names must always be valid in c and unique for each function (that is its arguments) # arguments will be looked up from json request, range checking needs to be implemented # by the actual API call API = { 'name': 'winman', 'api': 'g_afb_instance->app.api.', # where are our API functions 'functions': [ { 'name': 'requestsurface', #'return_type': 'int', # Or do they return all just some json? 'args': [ # describes the functions arguments, and their names as found in the json request { 'name': 'drawing_name', 'type': 'char const*', 'jtype': 'string' }, ], }, { 'name': 'activatesurface', 'args': [ { 'name': 'drawing_name', 'type': 'char const*', 'jtype': 'string' }, ], }, { 'name': 'deactivatesurface', 'args': [ { 'name': 'drawing_name', 'type': 'char const*', 'jtype': 'string' }, ], }, { 'name': 'enddraw', 'args': [ { 'name': 'drawing_name', 'type': 'char const*', 'jtype': 'string' }, ], }, { 'name': 'list_drawing_names', }, { 'name': 'ping' }, { 'name': 'debug_status', }, { 'name': 'debug_layers', }, { 'name': 'debug_surfaces', }, { 'name': 'debug_terminate' }, ] } def main(): with open('afb_binding_glue.inl', 'w') as out: set_output(out) p('// This file was generated, do not edit', '') generate_names(API) emit_binding(API) with open('afb_binding_api.hpp', 'w') as out: set_output(out) p('// This file was generated, do not edit', '') emit_afb_api(API) __name__ == '__main__' and main()