/* C API for _dbus_bindings, used by _dbus_glib_bindings and any third-party * main loop integration which might happen in future. * * This file is currently Python-version-independent - please keep it that way. * * Copyright (C) 2006 Collabora Ltd. * * Permission is hereby granted, free of charge, to any person * obtaining a copy of this software and associated documentation * files (the "Software"), to deal in the Software without * restriction, including without limitation the rights to use, copy, * modify, merge, publish, distribute, sublicense, and/or sell copies * of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */ #ifndef DBUS_PYTHON_H #define DBUS_PYTHON_H #include #include #if PY_MAJOR_VERSION >= 3 #define PY3 #define PYDBUS_CAPSULE_NAME "_dbus_bindings._C_API" #endif DBUS_BEGIN_DECLS typedef void (*_dbus_py_func_ptr)(void); typedef dbus_bool_t (*_dbus_py_conn_setup_func)(DBusConnection *, void *); typedef dbus_bool_t (*_dbus_py_srv_setup_func)(DBusServer *, void *); typedef void (*_dbus_py_free_func)(void *); #define DBUS_BINDINGS_API_COUNT 3 #ifdef INSIDE_DBUS_PYTHON_BINDINGS extern DBusConnection *DBusPyConnection_BorrowDBusConnection(PyObject *); extern PyObject *DBusPyNativeMainLoop_New4(_dbus_py_conn_setup_func, _dbus_py_srv_setup_func, _dbus_py_free_func, void *); #else static PyObject *_dbus_bindings_module = NULL; static _dbus_py_func_ptr *dbus_bindings_API; #define DBusPyConnection_BorrowDBusConnection \ (*(DBusConnection *(*)(PyObject *))dbus_bindings_API[1]) #define DBusPyNativeMainLoop_New4 \ ((PyObject *(*)(_dbus_py_conn_setup_func, _dbus_py_srv_setup_func, \ _dbus_py_free_func, void *))dbus_bindings_API[2]) static int import_dbus_bindings(const char *this_module_name) { PyObject *c_api; int count; _dbus_bindings_module = PyImport_ImportModule("_dbus_bindings"); if (!_dbus_bindings_module) { return -1; } c_api = PyObject_GetAttrString(_dbus_bindings_module, "_C_API"); if (c_api == NULL) return -1; #ifdef PY3 dbus_bindings_API = NULL; if (PyCapsule_IsValid(c_api, PYDBUS_CAPSULE_NAME)) { dbus_bindings_API = (_dbus_py_func_ptr *)PyCapsule_GetPointer( c_api, PYDBUS_CAPSULE_NAME); } Py_CLEAR(c_api); if (!dbus_bindings_API) { PyErr_SetString(PyExc_RuntimeError, "C API is not a PyCapsule"); return -1; } #else if (PyCObject_Check(c_api)) { dbus_bindings_API = (_dbus_py_func_ptr *)PyCObject_AsVoidPtr(c_api); } else { Py_DECREF(c_api); PyErr_SetString(PyExc_RuntimeError, "C API is not a PyCObject"); return -1; } Py_DECREF (c_api); #endif count = *(int *)dbus_bindings_API[0]; if (count < DBUS_BINDINGS_API_COUNT) { PyErr_Format(PyExc_RuntimeError, "_dbus_bindings has API version %d but %s needs " "_dbus_bindings API version at least %d", count, this_module_name, DBUS_BINDINGS_API_COUNT); return -1; } return 0; } #endif DBUS_END_DECLS #endif