/* * Mixer Interface - python binding simple abstact module * Copyright (c) 2007 by Jaroslav Kysela * * * This library is free software; you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as * published by the Free Software Foundation; either version 2.1 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * */ #include "Python.h" #include #include "config.h" #include "asoundlib.h" #include "mixer_abst.h" #if (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 2)) #pragma GCC diagnostic ignored "-Wstrict-aliasing" #endif struct python_priv { int py_initialized; PyObject *py_event_func; PyObject *py_mdict; PyObject *py_mixer; }; #define SCRIPT ALSA_PLUGIN_DIR "/smixer/python/main.py" struct pymelem { PyObject_HEAD sm_selem_t selem; PyObject *py_mixer; snd_mixer_elem_t *melem; }; struct pymixer { PyObject_HEAD snd_mixer_class_t *class; snd_mixer_t *mixer; PyObject *mdict; int hctl_count; void **hctl; int helem_count; void **helem; int melem_count; void **melem; }; static PyInterpreterState *main_interpreter; #if PY_MAJOR_VERSION >= 3 #define PyInt_FromLong PyLong_FromLong #endif static inline int get_long(PyObject *o, long *val) { #if PY_MAJOR_VERSION < 3 if (PyInt_Check(o)) { *val = PyInt_AsLong(o); return 0; } #endif if (PyLong_Check(o)) { *val = PyLong_AsLong(o); return 0; } return 1; } static inline PyObject *InternFromString(const char *name) { #if PY_MAJOR_VERSION < 3 return PyString_InternFromString(name); #else return PyUnicode_InternFromString(name); #endif } static void *get_C_ptr(PyObject *obj, const char *attr) { PyObject *o; long val; o = PyObject_GetAttr(obj, InternFromString(attr)); if (!o) { PyErr_Format(PyExc_TypeError, "missing '%s' attribute", attr); return NULL; } if (get_long(o, &val)) { PyErr_Format(PyExc_TypeError, "'%s' attribute is not Int or Long", attr); return NULL; } return (void *)val; } static struct pymelem *melem_to_pymelem(snd_mixer_elem_t *elem) { return (struct pymelem *)((char *)snd_mixer_elem_get_private(elem) - offsetof(struct pymelem, selem)); } static int pcall(struct pymelem *pymelem, const char *attr, PyObject *args, PyObject **_res) { PyObject *obj = (PyObject *)pymelem, *res; long xres = 0; if (_res) *_res = NULL; obj = PyObject_GetAttr(obj, InternFromString(attr)); if (!obj) { PyErr_Format(PyExc_TypeError, "missing '%s' attribute", attr); PyErr_Print(); PyErr_Clear(); Py_DECREF(args); return -EIO; } res = PyObject_CallObject(obj, args); Py_XDECREF(args); if (res == NULL) { PyErr_Print(); PyErr_Clear(); return -EIO; } if (_res && PyTuple_Check(res)) { *_res = res; res = PyTuple_GetItem(res, 0); } if (PyLong_Check(res)) { xres = PyLong_AsLong(res); #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(res)) { xres = PyInt_AsLong(res); #endif } else if (res == Py_None) { xres = 0; } else if (PyBool_Check(res)) { xres = res == Py_True; } else { PyErr_Format(PyExc_TypeError, "wrong result from '%s'!", attr); PyErr_Print(); PyErr_Clear(); Py_DECREF(res); if (_res) *_res = NULL; return -EIO; } if (_res && *_res) return xres; Py_DECREF(res); return xres; } static int is_ops(snd_mixer_elem_t *elem, int dir, int cmd, int val) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); char *s, fcn[32] = "opsIs"; int res, xdir = 1, xval = 0; switch (cmd) { case SM_OPS_IS_ACTIVE: s = "Active"; xdir = 0; break; case SM_OPS_IS_MONO: s = "Mono"; break; case SM_OPS_IS_CHANNEL: s = "Channel"; xval = 1; break; case SM_OPS_IS_ENUMERATED: s = "Enumerated"; xdir = val == 1; break; case SM_OPS_IS_ENUMCNT: s = "EnumCnt"; break; default: return 1; } strcat(fcn, s); obj1 = PyTuple_New(xdir + xval); if (xdir) { PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); if (xval) PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(val)); } res = pcall(pymelem, fcn, obj1, NULL); return res < 0 ? 0 : res; } static int get_x_range_ops(snd_mixer_elem_t *elem, int dir, long *min, long *max, const char *attr) { PyObject *obj1, *t1, *t2, *res; struct pymelem *pymelem = melem_to_pymelem(elem); int err; obj1 = PyTuple_New(1); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); err = pcall(pymelem, attr, obj1, &res); if (err >= 0) { t1 = PyTuple_GetItem(res, 1); t2 = PyTuple_GetItem(res, 2); if (PyLong_Check(t1) && PyLong_Check(t2)) { *min = PyLong_AsLong(PyTuple_GetItem(res, 1)); *max = PyLong_AsLong(PyTuple_GetItem(res, 2)); err = 0; #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(t1) && PyInt_Check(t2)) { *min = PyInt_AsLong(PyTuple_GetItem(res, 1)); *max = PyInt_AsLong(PyTuple_GetItem(res, 2)); err = 0; #endif } else { PyErr_Format(PyExc_TypeError, "wrong result (invalid tuple)"); PyErr_Print(); PyErr_Clear(); err = -EIO; } } Py_XDECREF(res); return err; } static int get_range_ops(snd_mixer_elem_t *elem, int dir, long *min, long *max) { return get_x_range_ops(elem, dir, min, max, "opsGetRange"); } static int set_range_ops(snd_mixer_elem_t *elem, int dir, long min, long max) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); obj1 = PyTuple_New(3); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(min)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(max)); return pcall(pymelem, "opsGetRange", obj1, NULL); } static int get_x_ops(snd_mixer_elem_t *elem, int dir, long channel, long *value, const char *attr) { PyObject *obj1, *t1, *res; struct pymelem *pymelem = melem_to_pymelem(elem); int err; obj1 = PyTuple_New(2); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(channel)); err = pcall(pymelem, attr, obj1, &res); if (err >= 0) { t1 = PyTuple_GetItem(res, 1); if (PyLong_Check(t1)) { *value = PyLong_AsLong(t1); err = 0; #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(t1)) { *value = PyInt_AsLong(t1); err = 0; #endif } else { PyErr_Format(PyExc_TypeError, "wrong result (invalid tuple)"); PyErr_Print(); PyErr_Clear(); err = -EIO; } } Py_XDECREF(res); return err; } static int get_volume_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, long *value) { return get_x_ops(elem, dir, channel, value, "opsGetVolume"); } static int get_switch_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, int *value) { long value1; int res; res = get_x_ops(elem, dir, channel, &value1, "opsGetSwitch"); *value = value1; return res; } static int ask_vol_dB_ops(snd_mixer_elem_t *elem, int dir, long value, long *dbValue) { return get_x_ops(elem, dir, value, dbValue, "opsGetVolDB"); } static int ask_dB_vol_ops(snd_mixer_elem_t *elem, int dir, long value, long *dbValue, int xdir) { PyObject *obj1, *t1, *res; struct pymelem *pymelem = melem_to_pymelem(elem); int err; obj1 = PyTuple_New(3); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(value)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(xdir)); err = pcall(pymelem, "opsGetDBVol", obj1, &res); if (err >= 0) { t1 = PyTuple_GetItem(res, 1); if (PyLong_Check(t1)) { *dbValue = PyLong_AsLong(t1); err = 0; #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(t1)) { *dbValue = PyInt_AsLong(t1); err = 0; #endif } else { PyErr_Format(PyExc_TypeError, "wrong result (invalid tuple)"); PyErr_Print(); PyErr_Clear(); err = -EIO; } } Py_XDECREF(res); return err; } static int get_dB_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, long *value) { return get_x_ops(elem, dir, channel, value, "opsGetDB"); } static int get_dB_range_ops(snd_mixer_elem_t *elem, int dir, long *min, long *max) { return get_x_range_ops(elem, dir, min, max, "opsGetDBRange"); } static int set_volume_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, long value) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); obj1 = PyTuple_New(3); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(channel)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(value)); return pcall(pymelem, "opsSetVolume", obj1, NULL); } static int set_switch_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, int value) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); obj1 = PyTuple_New(3); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(channel)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(value)); return pcall(pymelem, "opsSetSwitch", obj1, NULL); } static int set_dB_ops(snd_mixer_elem_t *elem, int dir, snd_mixer_selem_channel_id_t channel, long db_gain, int xdir) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); obj1 = PyTuple_New(4); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(dir)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(channel)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(db_gain)); PyTuple_SET_ITEM(obj1, 3, PyInt_FromLong(xdir)); return pcall(pymelem, "opsSetDB", obj1, NULL); } static int enum_item_name_ops(snd_mixer_elem_t *elem, unsigned int item, size_t maxlen, char *buf) { PyObject *obj1, *obj2, *t1, *res; struct pymelem *pymelem = melem_to_pymelem(elem); int err; unsigned int len; char *s; obj1 = PyTuple_New(1); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(item)); err = pcall(pymelem, "opsGetEnumItemName", obj1, &res); if (err >= 0) { t1 = PyTuple_GetItem(res, 1); if (PyUnicode_Check(t1)) { obj2 = PyUnicode_AsEncodedString(t1, "utf-8", "strict"); if (obj2) { s = PyBytes_AsString(obj2); len = strlen(s); if (maxlen - 1 > len) len = maxlen - 1; memcpy(buf, s, len); buf[len] = '\0'; Py_DECREF(obj2); } else { goto errlbl; } #if PY_MAJOR_VERSION < 3 } else if (PyString_Check(t1)) { s = PyString_AsString(t1); len = strlen(s); if (maxlen - 1 > len) len = maxlen - 1; memcpy(buf, s, len); buf[len] = '\0'; #endif } else { errlbl: PyErr_Format(PyExc_TypeError, "wrong result (invalid tuple)"); PyErr_Print(); PyErr_Clear(); err = -EIO; } } Py_XDECREF(res); return err; } static int get_enum_item_ops(snd_mixer_elem_t *elem, snd_mixer_selem_channel_id_t channel, unsigned int *itemp) { PyObject *obj1, *t1, *res; struct pymelem *pymelem = melem_to_pymelem(elem); int err; obj1 = PyTuple_New(1); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(channel)); err = pcall(pymelem, "opsGetEnumItem", obj1, &res); if (err >= 0) { t1 = PyTuple_GetItem(res, 1); if (PyLong_Check(t1)) { *itemp = PyLong_AsLong(t1); err = 0; #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(t1)) { *itemp = PyInt_AsLong(t1); err = 0; #endif } else { PyErr_Format(PyExc_TypeError, "wrong result (invalid tuple)"); PyErr_Print(); PyErr_Clear(); err = -EIO; } } Py_XDECREF(res); return err; } static int set_enum_item_ops(snd_mixer_elem_t *elem, snd_mixer_selem_channel_id_t channel, unsigned int item) { PyObject *obj1; struct pymelem *pymelem = melem_to_pymelem(elem); obj1 = PyTuple_New(2); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong(channel)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong(item)); return pcall(pymelem, "opsSetEnumItem", obj1, NULL); } static struct sm_elem_ops simple_python_ops = { .is = is_ops, .get_range = get_range_ops, .get_dB_range = get_dB_range_ops, .set_range = set_range_ops, .ask_vol_dB = ask_vol_dB_ops, .ask_dB_vol = ask_dB_vol_ops, .get_volume = get_volume_ops, .get_dB = get_dB_ops, .set_volume = set_volume_ops, .set_dB = set_dB_ops, .get_switch = get_switch_ops, .set_switch = set_switch_ops, .enum_item_name = enum_item_name_ops, .get_enum_item = get_enum_item_ops, .set_enum_item = set_enum_item_ops }; static void selem_free(snd_mixer_elem_t *elem) { sm_selem_t *simple = snd_mixer_elem_get_private(elem); if (simple->id) { snd_mixer_selem_id_free(simple->id); simple->id = NULL; } } static PyObject * pymelem_cap(struct pymelem *pymelem ATTRIBUTE_UNUSED, void *priv) { return PyInt_FromLong((long)priv); } static PyObject * pymelem_get_caps(struct pymelem *pymelem, void *priv ATTRIBUTE_UNUSED) { return PyInt_FromLong(pymelem->selem.caps); } static PyObject * pymelem_get_name(struct pymelem *pymelem, void *priv ATTRIBUTE_UNUSED) { return PyUnicode_FromString(snd_mixer_selem_id_get_name(pymelem->selem.id)); } static PyObject * pymelem_get_index(struct pymelem *pymelem, void *priv ATTRIBUTE_UNUSED) { return PyInt_FromLong(snd_mixer_selem_id_get_index(pymelem->selem.id)); } static int pymelem_set_caps(struct pymelem *pymelem, PyObject *val, void *priv ATTRIBUTE_UNUSED) { if (PyLong_Check(val)) { pymelem->selem.caps = PyLong_AsLong(val); return 0; } #if PY_MAJOR_VERSION < 3 if (PyInt_Check(val)) { pymelem->selem.caps = PyInt_AsLong(val); return 0; } #endif PyErr_SetString(PyExc_TypeError, "The last attribute value must be an integer"); return -1; } static PyObject * pymelem_ignore(struct pymelem *pymelem ATTRIBUTE_UNUSED, PyObject *args ATTRIBUTE_UNUSED) { Py_RETURN_NONE; } static PyObject * pymelem_ignore1(struct pymelem *pymelem ATTRIBUTE_UNUSED, PyObject *args ATTRIBUTE_UNUSED) { Py_RETURN_TRUE; } static PyObject * pymelem_error(struct pymelem *pymelem ATTRIBUTE_UNUSED, PyObject *args ATTRIBUTE_UNUSED) { return PyInt_FromLong(-EIO); } static PyObject * pymelem_attach(struct pymelem *pymelem, PyObject *args) { PyObject *obj; snd_hctl_elem_t *helem; int err; if (!PyArg_ParseTuple(args, "O", &obj)) return NULL; helem = (snd_hctl_elem_t *)get_C_ptr(obj, "get_C_helem"); if (helem == NULL) return NULL; err = snd_mixer_elem_attach(pymelem->melem, helem); if (err < 0) { PyErr_Format(PyExc_RuntimeError, "Cannot attach hcontrol element to mixer element: %s", snd_strerror(err)); return NULL; } Py_RETURN_NONE; } static PyObject * pymelem_detach(struct pymelem *pymelem, PyObject *args) { PyObject *obj; snd_hctl_elem_t *helem; int err; if (!PyArg_ParseTuple(args, "O", &obj)) return NULL; helem = (snd_hctl_elem_t *)get_C_ptr(obj, "get_C_helem"); if (helem == NULL) return NULL; err = snd_mixer_elem_detach(pymelem->melem, helem); if (err < 0) { PyErr_Format(PyExc_RuntimeError, "Cannot detach hcontrol element to mixer element: %s", snd_strerror(err)); return NULL; } Py_RETURN_NONE; } static PyObject * pymelem_event_info(struct pymelem *pymelem, PyObject *args) { if (!PyArg_ParseTuple(args, "")) return NULL; return PyInt_FromLong(snd_mixer_elem_info(pymelem->melem)); } static PyObject * pymelem_event_value(struct pymelem *pymelem, PyObject *args) { if (!PyArg_ParseTuple(args, "")) return NULL; return PyInt_FromLong(snd_mixer_elem_value(pymelem->melem)); } static int pymelem_init(struct pymelem *pymelem, PyObject *args, PyObject *kwds ATTRIBUTE_UNUSED) { char *name; long index, weight; snd_mixer_selem_id_t *id; int err; if (!PyArg_ParseTuple(args, "Osii", &pymelem->py_mixer, &name, &index, &weight)) return -1; memset(&pymelem->selem, 0, sizeof(pymelem->selem)); if (snd_mixer_selem_id_malloc(&id)) return -1; snd_mixer_selem_id_set_name(id, name); snd_mixer_selem_id_set_index(id, index); pymelem->selem.id = id; pymelem->selem.ops = &simple_python_ops; err = snd_mixer_elem_new(&pymelem->melem, SND_MIXER_ELEM_SIMPLE, weight, &pymelem->selem, selem_free); if (err < 0) { snd_mixer_selem_id_free(id); return -1; } return 0; } static void pymelem_dealloc(struct pymelem *self) { selem_free(self->melem); } static PyGetSetDef pymelem_getseters[] = { {"CAP_GVOLUME", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_GVOLUME}, {"CAP_GSWITCH", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_GSWITCH}, {"CAP_PVOLUME", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_PVOLUME}, {"CAP_PVOLUME_JOIN", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_PVOLUME_JOIN}, {"CAP_PSWITCH", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_PSWITCH}, {"CAP_PSWITCH_JOIN", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_PSWITCH_JOIN}, {"CAP_CVOLUME", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CVOLUME}, {"CAP_CVOLUME_JOIN", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CVOLUME_JOIN}, {"CAP_CSWITCH", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CSWITCH}, {"CAP_CSWITCH_JOIN", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CSWITCH_JOIN}, {"CAP_CSWITCH_EXCL", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CSWITCH_EXCL}, {"CAP_PENUM", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_PENUM}, {"CAP_CENUM", (getter)pymelem_cap, NULL, NULL, (void *)SM_CAP_CENUM}, {"caps", (getter)pymelem_get_caps, (setter)pymelem_set_caps, NULL, NULL}, {"name", (getter)pymelem_get_name, NULL, NULL, NULL}, {"index", (getter)pymelem_get_index, NULL, NULL, NULL}, {NULL,NULL,NULL,NULL,NULL} }; static PyMethodDef pymelem_methods[] = { {"attach", (PyCFunction)pymelem_attach, METH_VARARGS, NULL}, {"detach", (PyCFunction)pymelem_detach, METH_VARARGS, NULL}, /* "default" functions - no functionality */ {"opsIsActive", (PyCFunction)pymelem_ignore1, METH_VARARGS, NULL}, {"opsIsMono", (PyCFunction)pymelem_ignore, METH_VARARGS, NULL}, {"opsIsChannel", (PyCFunction)pymelem_ignore, METH_VARARGS, NULL}, {"opsIsEnumerated", (PyCFunction)pymelem_ignore, METH_VARARGS, NULL}, {"opsIsEnumCnt", (PyCFunction)pymelem_ignore, METH_VARARGS, NULL}, {"opsGetDB", (PyCFunction)pymelem_error, METH_VARARGS, NULL}, {"eventInfo", (PyCFunction)pymelem_event_info, METH_VARARGS, NULL}, {"eventValue", (PyCFunction)pymelem_event_value, METH_VARARGS, NULL}, {NULL,NULL,0,NULL} }; static PyTypeObject pymelem_type = { PyVarObject_HEAD_INIT(NULL, 0) tp_name: "smixer_python.InternalMElement", tp_basicsize: sizeof(struct pymelem), tp_dealloc: (destructor)pymelem_dealloc, tp_flags: Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, tp_doc: NULL /* mixerinit__doc__ */, tp_getset: pymelem_getseters, tp_init: (initproc)pymelem_init, tp_alloc: PyType_GenericAlloc, tp_new: PyType_GenericNew, tp_free: PyObject_Del, tp_methods: pymelem_methods, }; static PyObject * pymixer_attach_hctl(struct pymixer *pymixer, PyObject *args) { PyObject *obj; snd_hctl_t *hctl; void **hctls; int err; if (!PyArg_ParseTuple(args, "O", &obj)) return NULL; hctl = (snd_hctl_t *)get_C_ptr(obj, "get_C_hctl"); if (hctl == NULL) return NULL; err = snd_mixer_attach_hctl(pymixer->mixer, hctl); if (err < 0) { PyErr_Format(PyExc_RuntimeError, "Cannot attach hctl: %s", snd_strerror(err)); return NULL; } hctls = realloc(pymixer->hctl, sizeof(void *) * (pymixer->hctl_count+1) * 2); if (hctls == NULL) { PyErr_SetString(PyExc_RuntimeError, "No enough memory"); return NULL; } pymixer->hctl = hctls; pymixer->hctl[pymixer->hctl_count*2] = (void *)hctl; pymixer->hctl[pymixer->hctl_count*2+1] = (void *)obj; pymixer->hctl_count++; Py_INCREF(obj); Py_RETURN_NONE; } static PyObject * pymixer_register(struct pymixer *pymixer, PyObject *args) { int err; if (!PyArg_ParseTuple(args, "")) return NULL; err = snd_mixer_class_register(pymixer->class, pymixer->mixer); if (err < 0) { PyErr_Format(PyExc_RuntimeError, "Cannot register mixer: %s", snd_strerror(err)); return NULL; } Py_RETURN_NONE; } static PyObject * pymixer_melement_new(struct pymixer *pymixer, PyObject *args) { PyObject *obj, *obj1, *obj2; char *class, *name; long index, weight; if (!PyArg_ParseTuple(args, "ssii", &class, &name, &index, &weight)) return NULL; obj = PyDict_GetItemString(pymixer->mdict, class); if (obj) { obj1 = PyTuple_New(4); if (PyTuple_SET_ITEM(obj1, 0, (PyObject *)pymixer)) Py_INCREF((PyObject *)pymixer); PyTuple_SET_ITEM(obj1, 1, PyUnicode_FromString(name)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong(index)); PyTuple_SET_ITEM(obj1, 3, PyInt_FromLong(weight)); obj2 = PyObject_CallObject(obj, obj1); Py_XDECREF(obj1); if (obj2) { struct pymelem *pymelem = (struct pymelem *)obj2; void **melems = realloc(pymixer->melem, sizeof(void *) * (pymixer->melem_count + 1) * 2); if (melems == NULL) { Py_DECREF(obj2); return NULL; } melems[pymixer->melem_count*2] = pymelem->melem; melems[pymixer->melem_count*2+1] = obj2; Py_INCREF(obj2); pymixer->melem = melems; pymixer->melem_count++; } } else { PyErr_Format(PyExc_RuntimeError, "Cannot find class '%s'", class); return NULL; } return obj2; } static PyObject * pymixer_melement_add(struct pymixer *pymixer, PyObject *args) { PyObject *obj; struct pymelem *pymelem; int err; if (!PyArg_ParseTuple(args, "O", &obj)) return NULL; pymelem = (struct pymelem *)obj; err = snd_mixer_elem_add(pymelem->melem, pymixer->class); if (err < 0) { PyErr_Format(PyExc_RuntimeError, "Cannot add mixer element: %s", snd_strerror(err)); return NULL; } Py_RETURN_NONE; } static int pymixer_init(struct pymixer *pymixer, PyObject *args, PyObject *kwds ATTRIBUTE_UNUSED) { long class, mixer; if (!PyArg_ParseTuple(args, "iiO", &class, &mixer, &pymixer->mdict)) return -1; pymixer->class = (snd_mixer_class_t *)class; pymixer->mixer = (snd_mixer_t *)mixer; pymixer->hctl_count = 0; pymixer->hctl = NULL; pymixer->helem_count = 0; pymixer->helem = NULL; pymixer->melem_count = 0; pymixer->melem = NULL; return 0; } static void pymixer_free(struct pymixer *self) { int idx; for (idx = 0; idx < self->hctl_count; idx++) { snd_mixer_detach_hctl(self->mixer, self->hctl[idx*2]); Py_DECREF((PyObject *)self->hctl[idx*2+1]); } if (self->hctl) free(self->hctl); self->hctl_count = 0; self->hctl = NULL; for (idx = 0; idx < self->helem_count; idx++) Py_DECREF((PyObject *)self->helem[idx*2+1]); if (self->helem) free(self->helem); self->helem_count = 0; self->helem = NULL; for (idx = 0; idx < self->melem_count; idx++) Py_DECREF((PyObject *)self->melem[idx*2+1]); if (self->melem) free(self->melem); self->melem_count = 0; self->melem = NULL; } static void pymixer_dealloc(struct pymixer *self) { pymixer_free(self); } static PyGetSetDef pymixer_getseters[] = { {NULL,NULL,NULL,NULL,NULL} }; static PyMethodDef pymixer_methods[] = { {"attachHCtl", (PyCFunction)pymixer_attach_hctl, METH_VARARGS, NULL}, {"register", (PyCFunction)pymixer_register, METH_VARARGS, NULL}, {"newMElement", (PyCFunction)pymixer_melement_new, METH_VARARGS, NULL}, {"addMElement", (PyCFunction)pymixer_melement_add, METH_VARARGS, NULL}, {NULL,NULL,0,NULL} }; static PyTypeObject pymixer_type = { PyVarObject_HEAD_INIT(NULL, 0) tp_name: "smixer_python.InternalMixer", tp_basicsize: sizeof(struct pymixer), tp_dealloc: (destructor)pymixer_dealloc, tp_flags: Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, tp_doc: NULL /* mixerinit__doc__ */, tp_getset: pymixer_getseters, tp_init: (initproc)pymixer_init, tp_alloc: PyType_GenericAlloc, tp_new: PyType_GenericNew, tp_free: PyObject_Del, tp_methods: pymixer_methods, }; static PyMethodDef python_methods[] = { {NULL, NULL, 0, NULL} }; static PyObject *new_helem(struct python_priv *priv, snd_hctl_elem_t *helem) { PyObject *obj, *py_hctl = NULL, *obj1, *obj2; snd_hctl_t *hctl = snd_hctl_elem_get_hctl(helem); struct pymixer *pymixer = (struct pymixer *)priv->py_mixer; int idx; for (idx = 0; idx < pymixer->hctl_count; idx++) { if (pymixer->hctl[idx] == hctl) { py_hctl = pymixer->hctl[idx*2+1]; break; } } if (py_hctl == NULL) return NULL; obj = PyDict_GetItemString(priv->py_mdict, "HElement"); if (obj) { obj1 = PyTuple_New(3); if (PyTuple_SET_ITEM(obj1, 0, py_hctl)) Py_INCREF(py_hctl); PyTuple_SET_ITEM(obj1, 1, PyFloat_FromDouble(1)); PyTuple_SET_ITEM(obj1, 2, PyInt_FromLong((long)helem)); obj2 = PyObject_CallObject(obj, obj1); if (obj2 == NULL) { PyErr_Print(); PyErr_Clear(); } Py_XDECREF(obj1); } else { SNDERR("Unable to create InternalMixer object"); return NULL; } if (obj2) { struct pymixer *pymixer = (struct pymixer *)priv->py_mixer; void **helems = realloc(pymixer->helem, sizeof(void *) * (pymixer->helem_count + 1) * 2); if (helems == NULL) { Py_DECREF(obj2); return NULL; } helems[pymixer->helem_count*2] = helem; helems[pymixer->helem_count*2+1] = obj2; Py_INCREF(obj2); pymixer->helem = helems; pymixer->helem_count++; } return obj2; } static PyObject *find_helem(struct python_priv *priv, snd_hctl_elem_t *helem) { struct pymixer *pymixer = (struct pymixer *)priv->py_mixer; int idx; for (idx = 0; idx < pymixer->helem_count; idx++) { if (pymixer->helem[idx*2] == helem) return (PyObject *)pymixer->helem[idx*2+1]; } return NULL; } static PyObject *find_melem(struct python_priv *priv, snd_mixer_elem_t *melem) { struct pymixer *pymixer = (struct pymixer *)priv->py_mixer; int idx; for (idx = 0; idx < pymixer->melem_count; idx++) { if (pymixer->melem[idx*2] == melem) return (PyObject *)pymixer->melem[idx*2+1]; } return NULL; } int alsa_mixer_simple_event(snd_mixer_class_t *class, unsigned int mask, snd_hctl_elem_t *helem, snd_mixer_elem_t *melem) { struct python_priv *priv = snd_mixer_sbasic_get_private(class); PyThreadState *tstate; PyObject *t, *o, *r; int res = -ENOMEM; tstate = PyThreadState_New(main_interpreter); PyThreadState_Swap(tstate); t = PyTuple_New(3); if (t) { PyTuple_SET_ITEM(t, 0, (PyObject *)PyInt_FromLong(mask)); o = find_helem(priv, helem); if (mask & SND_CTL_EVENT_MASK_ADD) { if (o == NULL) o = new_helem(priv, helem); } if (o == NULL) return 0; if (PyTuple_SET_ITEM(t, 1, o)) Py_INCREF(o); o = melem ? find_melem(priv, melem) : Py_None; if (PyTuple_SET_ITEM(t, 2, o)) Py_INCREF(o); r = PyObject_CallObject(priv->py_event_func, t); Py_DECREF(t); if (r) { if (PyLong_Check(r)) { res = PyLong_AsLong(r); #if PY_MAJOR_VERSION < 3 } else if (PyInt_Check(r)) { res = PyInt_AsLong(r); #endif } else if (r == Py_None) { res = 0; } Py_DECREF(r); } else { PyErr_Print(); PyErr_Clear(); res = -EIO; } } return res; } static void alsa_mixer_simple_free(snd_mixer_class_t *class) { struct python_priv *priv = snd_mixer_sbasic_get_private(class); if (priv->py_mixer) { pymixer_free((struct pymixer *)priv->py_mixer); Py_DECREF(priv->py_mixer); } if (priv->py_initialized) { Py_XDECREF(priv->py_event_func); Py_Finalize(); } free(priv); } static int alsa_mixer_simple_pyinit(struct python_priv *priv, PyObject *py_mod, FILE *fp, const char *file, snd_mixer_class_t *class, snd_mixer_t *mixer, const char *device) { PyObject *obj, *obj1, *obj2, *mdict; mdict = priv->py_mdict = PyModule_GetDict(py_mod); obj = PyUnicode_FromString(file); if (obj) PyDict_SetItemString(mdict, "__file__", obj); Py_XDECREF(obj); obj = PyUnicode_FromString(device); if (obj) PyDict_SetItemString(mdict, "device", obj); Py_XDECREF(obj); Py_INCREF(&pymelem_type); Py_INCREF(&pymixer_type); PyModule_AddObject(py_mod, "InternalMElement", (PyObject *)&pymelem_type); PyModule_AddObject(py_mod, "InternalMixer", (PyObject *)&pymixer_type); obj = PyDict_GetItemString(mdict, "InternalMixer"); if (obj) { obj1 = PyTuple_New(3); PyTuple_SET_ITEM(obj1, 0, PyInt_FromLong((long)class)); PyTuple_SET_ITEM(obj1, 1, PyInt_FromLong((long)mixer)); if (PyTuple_SET_ITEM(obj1, 2, mdict)) Py_INCREF(mdict); obj2 = PyObject_CallObject(obj, obj1); Py_XDECREF(obj1); PyDict_SetItemString(mdict, "mixer", obj2); priv->py_mixer = obj2; } else { SNDERR("Unable to create InternalMixer object"); return -EIO; } obj = PyRun_FileEx(fp, file, Py_file_input, mdict, mdict, 1); if (obj == NULL) PyErr_Print(); Py_XDECREF(obj); priv->py_event_func = PyDict_GetItemString(mdict, "event"); if (priv->py_event_func == NULL) { SNDERR("Unable to find python function 'event'"); return -EIO; } return 0; } #if PY_MAJOR_VERSION >= 3 static struct PyModuleDef smixer_python_module = { PyModuleDef_HEAD_INIT, "smixer_python", NULL, 0, python_methods, NULL, NULL, NULL, NULL }; #endif int alsa_mixer_simple_finit(snd_mixer_class_t *class, snd_mixer_t *mixer, const char *device) { struct python_priv *priv; FILE *fp; const char *file; PyObject *obj, *py_mod; priv = calloc(1, sizeof(*priv)); if (priv == NULL) return -ENOMEM; snd_mixer_sbasic_set_private(class, priv); snd_mixer_sbasic_set_private_free(class, alsa_mixer_simple_free); file = getenv("ALSA_MIXER_SIMPLE_MPYTHON"); if (file == NULL) file = SCRIPT; fp = fopen(file, "r"); if (fp == NULL) { SNDERR("Unable to find python module '%s'", file); return -ENODEV; } Py_Initialize(); if (PyType_Ready(&pymelem_type) < 0 || PyType_Ready(&pymixer_type) < 0) { fclose(fp); return -EIO; } #if PY_MAJOR_VERSION < 3 Py_InitModule("smixer_python", python_methods); #else PyModule_Create(&smixer_python_module); #endif priv->py_initialized = 1; main_interpreter = PyThreadState_Get()->interp; obj = PyImport_GetModuleDict(); py_mod = PyDict_GetItemString(obj, "__main__"); if (py_mod) alsa_mixer_simple_pyinit(priv, py_mod, fp, file, class, mixer, device); return 0; }