aboutsummaryrefslogtreecommitdiff
path: root/externals/grill/py
diff options
context:
space:
mode:
Diffstat (limited to 'externals/grill/py')
-rw-r--r--externals/grill/py/source/bound.cpp68
-rw-r--r--externals/grill/py/source/pyext.cpp2
-rw-r--r--externals/grill/py/source/pyext.h13
3 files changed, 74 insertions, 9 deletions
diff --git a/externals/grill/py/source/bound.cpp b/externals/grill/py/source/bound.cpp
index 58250830..e8e2599f 100644
--- a/externals/grill/py/source/bound.cpp
+++ b/externals/grill/py/source/bound.cpp
@@ -11,6 +11,7 @@ WARRANTIES, see the file, "license.txt," in this distribution.
#include "pyext.h"
#include "flinternal.h"
+#ifndef USEFLEXTBINDING
t_class *pyext::px_class;
pyext::py_proxy *pyext::px_head,*pyext::px_tail;
@@ -28,6 +29,26 @@ void pyext::py_proxy::px_method(py_proxy *obj,const t_symbol *s,int argc,const t
PY_UNLOCK
}
+#else
+struct bounddata { PyObject *self,*func; };
+
+bool pyext::boundmeth(flext_base *,const t_symbol *sym,int argc,const t_atom *argv,void *data)
+{
+ bounddata *obj = (bounddata *)data;
+
+ PY_LOCK
+
+ PyObject *args = MakePyArgs(sym,AtomList(argc,argv),-1,obj->self != NULL);
+ PyObject *ret = PyObject_CallObject(obj->func,args);
+ if(!ret) {
+ PyErr_Print();
+ }
+ Py_XDECREF(ret);
+
+ PY_UNLOCK
+ return true;
+}
+#endif
PyObject *pyext::pyext_bind(PyObject *,PyObject *args)
{
@@ -39,18 +60,20 @@ PyObject *pyext::pyext_bind(PyObject *,PyObject *args)
post("py/pyext - Wrong argument types!");
}
else {
- t_symbol *recv = gensym(name);
+ const t_symbol *recv = MakeSymbol(name);
/*
if(GetBound(recv))
post("py/pyext - Symbol \"%s\" is already hooked",GetString(recv));
*/
// make a proxy object
- py_proxy *px = (py_proxy *)object_new(px_class);
+
if(PyMethod_Check(meth)) {
PyObject *no = PyObject_GetAttrString(meth,"__name__");
meth = PyObject_GetAttr(self,no);
Py_DECREF(no);
}
+#ifndef USEFLEXTBINDING
+ py_proxy *px = (py_proxy *)object_new(px_class);
px->init(recv,self,meth);
// add it to the list
@@ -59,7 +82,13 @@ PyObject *pyext::pyext_bind(PyObject *,PyObject *args)
px_tail = px;
// Do bind
- pd_bind(&px->obj.ob_pd,recv);
+ pd_bind(&px->obj.ob_pd,(t_symbol *)recv);
+#else
+ bounddata *data = new bounddata;
+ data->self = self;
+ data->func = meth;
+ GetThis(self)->BindMethod(recv,boundmeth,data);
+#endif
Py_INCREF(self); // self is borrowed reference
}
@@ -78,13 +107,14 @@ PyObject *pyext::pyext_unbind(PyObject *,PyObject *args)
post("py/pyext - Wrong argument types!");
}
else {
- t_symbol *recv = gensym(name);
+ const t_symbol *recv = MakeSymbol(name);
if(PyMethod_Check(meth)) {
PyObject *no = PyObject_GetAttrString(meth,"__name__");
meth = PyObject_GetAttr(self,no); // meth is given a new reference!
Py_DECREF(no);
}
+#ifndef USEFLEXTBINDING
// search proxy object
py_proxy *pp = NULL,*px = px_head;
while(px) {
@@ -103,12 +133,23 @@ PyObject *pyext::pyext_unbind(PyObject *,PyObject *args)
// do unbind
if(px) {
- pd_unbind(&px->obj.ob_pd,recv);
+ pd_unbind(&px->obj.ob_pd,(t_symbol *)recv);
object_free(&px->obj);
Py_DECREF(self);
if(PyMethod_Check(meth)) Py_DECREF(meth);
}
+#else
+ void *data = NULL;
+ if(GetThis(self)->UnbindMethod(recv,boundmeth,&data)) {
+ bounddata *bdt = (bounddata *)data;
+ if(bdt) {
+ Py_DECREF(bdt->self);
+ if(PyMethod_Check(bdt->func)) Py_DECREF(bdt->func);
+ if(data) delete bdt;
+ }
+ }
+#endif
}
Py_INCREF(Py_None);
@@ -118,6 +159,7 @@ PyObject *pyext::pyext_unbind(PyObject *,PyObject *args)
V pyext::ClearBinding()
{
+#ifndef USEFLEXTBINDING
// search proxy object
py_proxy *pp = NULL,*px = px_head;
while(px) {
@@ -132,12 +174,26 @@ V pyext::ClearBinding()
Py_DECREF(px->self);
if(PyMethod_Check(px->func)) Py_DECREF(px->func);
- pd_unbind(&px->obj.ob_pd,px->name);
+ pd_unbind(&px->obj.ob_pd,(t_symbol *)px->name);
object_free(&px->obj);
}
else pp = px;
px = pn;
}
+#else
+ void *data = NULL;
+ const t_symbol *sym = NULL;
+
+ // unbind all
+ while(GetThis(pyobj)->UnbindMethod(sym,NULL,&data)) {
+ bounddata *bdt = (bounddata *)data;
+ if(bdt) {
+ Py_DECREF(bdt->self);
+ if(PyMethod_Check(bdt->func)) Py_DECREF(bdt->func);
+ if(data) delete bdt;
+ }
+ }
+#endif
}
diff --git a/externals/grill/py/source/pyext.cpp b/externals/grill/py/source/pyext.cpp
index 5f984b4b..ddd9fda6 100644
--- a/externals/grill/py/source/pyext.cpp
+++ b/externals/grill/py/source/pyext.cpp
@@ -16,10 +16,12 @@ FLEXT_LIB_V("pyext",pyext)
V pyext::Setup(t_classid c)
{
+#ifndef USEFLEXTBINDING
px_head = px_tail = NULL;
px_class = class_new(gensym("pyext proxy"),NULL,NULL,sizeof(py_proxy),CLASS_PD|CLASS_NOINLET, A_NULL);
::add_anything(px_class,py_proxy::px_method); // for other inlets
+#endif
FLEXT_CADDMETHOD_(c,0,"reload.",m_reload);
FLEXT_CADDMETHOD_(c,0,"reload",m_reload_);
diff --git a/externals/grill/py/source/pyext.h b/externals/grill/py/source/pyext.h
index 60601555..8f9a1b1b 100644
--- a/externals/grill/py/source/pyext.h
+++ b/externals/grill/py/source/pyext.h
@@ -70,6 +70,7 @@ private:
// -------- bound stuff ------------------
+#ifndef USEFLEXTBINDING
static t_class *px_class;
friend class py_proxy;
@@ -79,11 +80,11 @@ private:
public:
t_object obj; // MUST reside at memory offset 0
PyObject *self,*func;
- t_symbol *name;
+ const t_symbol *name;
py_proxy *nxt;
- void init(t_symbol *n,PyObject *s,PyObject *f) { name = n,self = s,func = f,nxt = NULL; }
+ void init(const t_symbol *n,PyObject *s,PyObject *f) { name = n,self = s,func = f,nxt = NULL; }
// bool cmp(PyObject *s,PyObject *f) const { return self == s && func == f; }
// void init(PyObject *s,char *f) { self = s,func = f,nxt = NULL; }
// bool cmp(PyObject *s,char *f) const { return self == s && func == f; }
@@ -93,6 +94,10 @@ private:
static PyObject *pyext_bind(PyObject *,PyObject *args);
static PyObject *pyext_unbind(PyObject *,PyObject *args);
+#else
+ static PyObject *pyext_bind(PyObject *,PyObject *args);
+ static PyObject *pyext_unbind(PyObject *,PyObject *args);
+#endif
// ---------------------------
@@ -118,7 +123,9 @@ private:
PyThreadState *pythr;
private:
- FLEXT_CALLBACK(m_reload)
+ static bool boundmeth(flext_base *,const t_symbol *sym,int argc,const t_atom *argv,void *data);
+
+ FLEXT_CALLBACK(m_reload)
FLEXT_CALLBACK_V(m_reload_)
FLEXT_CALLBACK(m_doc_)
};