Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
cython
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Commits
Open sidebar
nexedi
cython
Commits
50a0470d
Commit
50a0470d
authored
Aug 12, 2011
by
scoder
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #38 from vitek/_bindings
CyFunction
parents
508ee78a
80579d55
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
331 additions
and
27 deletions
+331
-27
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+261
-23
Cython/Compiler/ModuleNode.py
Cython/Compiler/ModuleNode.py
+2
-2
Cython/Compiler/Symtab.py
Cython/Compiler/Symtab.py
+2
-2
runtests.py
runtests.py
+3
-0
tests/run/cyfunction.pyx
tests/run/cyfunction.pyx
+63
-0
No files found.
Cython/Compiler/ExprNodes.py
View file @
50a0470d
...
@@ -5192,7 +5192,7 @@ class PyCFunctionNode(ExprNode, ModuleNameMixin):
...
@@ -5192,7 +5192,7 @@ class PyCFunctionNode(ExprNode, ModuleNameMixin):
def
generate_result_code
(
self
,
code
):
def
generate_result_code
(
self
,
code
):
if
self
.
binding
:
if
self
.
binding
:
constructor
=
"
%s_NewEx"
%
Naming
.
binding_cfunc
constructor
=
"
__Pyx_CyFunction_NewEx"
else
:
else
:
constructor
=
"PyCFunction_NewEx"
constructor
=
"PyCFunction_NewEx"
py_mod_name
=
self
.
get_py_mod_name
(
code
)
py_mod_name
=
self
.
get_py_mod_name
(
code
)
...
@@ -8754,61 +8754,299 @@ proto="""
...
@@ -8754,61 +8754,299 @@ proto="""
binding_cfunc_utility_code
=
UtilityCode
(
binding_cfunc_utility_code
=
UtilityCode
(
proto
=
"""
proto
=
"""
#define %(binding_cfunc)s_USED 1
#define __Pyx_CyFunction_USED 1
#include <structmember.h>
typedef struct {
typedef struct {
PyCFunctionObject func;
PyCFunctionObject func;
} %(binding_cfunc)s_object;
PyObject *func_dict;
PyObject *func_weakreflist;
PyObject *func_name;
PyObject *func_doc;
} __pyx_CyFunctionObject;
static PyTypeObject %(binding_cfunc)s_type;
static PyTypeObject *__pyx_CyFunctionType = 0;
static PyTypeObject *%(binding_cfunc)s = NULL;
static PyObject *
%(binding_cfunc)s_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module); /* proto */
static PyObject *
__Pyx_CyFunction_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module);
#define
%(binding_cfunc)s_New(ml, self) %(binding_cfunc)s
_NewEx(ml, self, NULL)
#define
__Pyx_CyFunction_New(ml, self) __Pyx_CyFunction
_NewEx(ml, self, NULL)
static int
%(binding_cfunc)s_init(void); /* proto */
static int
__Pyx_CyFunction_init(void);
"""
%
Naming
.
__dict__
,
"""
%
Naming
.
__dict__
,
impl
=
"""
impl
=
"""
static PyObject *
__Pyx_CyFunction_get_doc(__pyx_CyFunctionObject *op, void *closure)
{
if (op->func_doc == NULL && op->func.m_ml->ml_doc) {
#if PY_MAJOR_VERSION >= 3
op->func_doc = PyUnicode_FromString(op->func.m_ml->ml_doc);
#else
op->func_doc = PyString_FromString(op->func.m_ml->ml_doc);
#endif
}
if (op->func_doc == 0) {
Py_INCREF(Py_None);
return Py_None;
}
Py_INCREF(op->func_doc);
return op->func_doc;
}
static int
__Pyx_CyFunction_set_doc(__pyx_CyFunctionObject *op, PyObject *value)
{
PyObject *tmp = op->func_doc;
if (value == NULL)
op->func_doc = Py_None; /* Mark as deleted */
else
op->func_doc = value;
Py_INCREF(op->func_doc);
Py_XDECREF(tmp);
return 0;
}
static PyObject *
__Pyx_CyFunction_get_name(__pyx_CyFunctionObject *op)
{
if (op->func_name == NULL) {
#if PY_MAJOR_VERSION >= 3
op->func_name = PyUnicode_InternFromString(op->func.m_ml->ml_name);
#else
op->func_name = PyString_InternFromString(op->func.m_ml->ml_name);
#endif
}
Py_INCREF(op->func_name);
return op->func_name;
}
static int
__Pyx_CyFunction_set_name(__pyx_CyFunctionObject *op, PyObject *value)
{
PyObject *tmp;
#if PY_MAJOR_VERSION >= 3
if (value == NULL || !PyUnicode_Check(value)) {
#else
if (value == NULL || !PyString_Check(value)) {
#endif
PyErr_SetString(PyExc_TypeError,
"__name__ must be set to a string object");
return -1;
}
tmp = op->func_name;
Py_INCREF(value);
op->func_name = value;
Py_XDECREF(tmp);
return 0;
}
static PyObject *
__Pyx_CyFunction_get_self(__pyx_CyFunctionObject *m, void *closure)
{
PyObject *self;
self = m->func.m_self;
if (self == NULL)
self = Py_None;
Py_INCREF(self);
return self;
}
static PyObject *
__Pyx_CyFunction_get_dict(__pyx_CyFunctionObject *op)
{
if (op->func_dict == NULL) {
op->func_dict = PyDict_New();
if (op->func_dict == NULL)
return NULL;
}
Py_INCREF(op->func_dict);
return op->func_dict;
}
static int
__Pyx_CyFunction_set_dict(__pyx_CyFunctionObject *op, PyObject *value)
{
PyObject *tmp;
static PyObject *%(binding_cfunc)s_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module) {
if (value == NULL) {
%(binding_cfunc)s_object *op = PyObject_GC_New(%(binding_cfunc)s_object, %(binding_cfunc)s);
PyErr_SetString(PyExc_TypeError,
"function's dictionary may not be deleted");
return -1;
}
if (!PyDict_Check(value)) {
PyErr_SetString(PyExc_TypeError,
"setting function's dictionary to a non-dict");
return -1;
}
tmp = op->func_dict;
Py_INCREF(value);
op->func_dict = value;
Py_XDECREF(tmp);
return 0;
}
static PyGetSetDef __pyx_CyFunction_getsets[] = {
{(char *) "func_doc", (getter)__Pyx_CyFunction_get_doc, (setter)__Pyx_CyFunction_set_doc, 0, 0},
{(char *) "__doc__", (getter)__Pyx_CyFunction_get_doc, (setter)__Pyx_CyFunction_set_doc, 0, 0},
{(char *) "func_name", (getter)__Pyx_CyFunction_get_name, (setter)__Pyx_CyFunction_set_name, 0, 0},
{(char *) "__name__", (getter)__Pyx_CyFunction_get_name, (setter)__Pyx_CyFunction_set_name, 0, 0},
{(char *) "__self__", (getter)__Pyx_CyFunction_get_self, 0, 0, 0},
{(char *) "func_dict", (getter)__Pyx_CyFunction_get_dict, (setter)__Pyx_CyFunction_set_dict, 0, 0},
{(char *) "__dict__", (getter)__Pyx_CyFunction_get_dict, (setter)__Pyx_CyFunction_set_dict, 0, 0},
{0, 0, 0, 0, 0}
};
#ifndef PY_WRITE_RESTRICTED /* < Py2.5 */
#define PY_WRITE_RESTRICTED WRITE_RESTRICTED
#endif
static PyMemberDef __pyx_CyFunction_members[] = {
{(char *) "__module__", T_OBJECT, offsetof(__pyx_CyFunctionObject, func.m_module), PY_WRITE_RESTRICTED, 0},
{0, 0, 0, 0, 0}
};
static PyObject *
__Pyx_CyFunction_reduce(__pyx_CyFunctionObject *m, CYTHON_UNUSED PyObject *args)
{
#if PY_MAJOR_VERSION >= 3
return PyUnicode_FromString(m->func.m_ml->ml_name);
#else
return PyString_FromString(m->func.m_ml->ml_name);
#endif
}
static PyMethodDef __pyx_CyFunction_methods[] = {
{__Pyx_NAMESTR("__reduce__"), (PyCFunction)__Pyx_CyFunction_reduce, METH_VARARGS, 0},
{0, 0, 0, 0}
};
static PyObject *__Pyx_CyFunction_NewEx(PyMethodDef *ml, PyObject *self, PyObject *module) {
__pyx_CyFunctionObject *op = PyObject_GC_New(__pyx_CyFunctionObject, __pyx_CyFunctionType);
if (op == NULL)
if (op == NULL)
return NULL;
return NULL;
op->func_weakreflist = NULL;
op->func.m_ml = ml;
op->func.m_ml = ml;
Py_XINCREF(self);
Py_XINCREF(self);
op->func.m_self = self;
op->func.m_self = self;
Py_XINCREF(module);
Py_XINCREF(module);
op->func.m_module = module;
op->func.m_module = module;
op->func_dict = NULL;
op->func_name = NULL;
op->func_doc = NULL;
PyObject_GC_Track(op);
PyObject_GC_Track(op);
return (PyObject *)op;
return (PyObject *)op;
}
}
static void %(binding_cfunc)s_dealloc(%(binding_cfunc)s_object *m) {
static void __Pyx_CyFunction_dealloc(__pyx_CyFunctionObject *m)
{
PyObject_GC_UnTrack(m);
PyObject_GC_UnTrack(m);
if (m->func_weakreflist != NULL)
PyObject_ClearWeakRefs((PyObject *) m);
Py_XDECREF(m->func.m_self);
Py_XDECREF(m->func.m_self);
Py_XDECREF(m->func.m_module);
Py_XDECREF(m->func.m_module);
Py_XDECREF(m->func_dict);
Py_XDECREF(m->func_name);
Py_XDECREF(m->func_doc);
PyObject_GC_Del(m);
PyObject_GC_Del(m);
}
}
static PyObject *%(binding_cfunc)s_descr_get(PyObject *func, PyObject *obj, PyObject *type) {
static int __Pyx_CyFunction_traverse(__pyx_CyFunctionObject *m, visitproc visit, void *arg)
{
Py_VISIT(m->func.m_self);
Py_VISIT(m->func.m_module);
Py_VISIT(m->func_dict);
Py_VISIT(m->func_name);
Py_VISIT(m->func_doc);
return 0;
}
static PyObject *__Pyx_CyFunction_descr_get(PyObject *func, PyObject *obj, PyObject *type)
{
if (obj == Py_None)
if (obj == Py_None)
obj = NULL;
obj = NULL;
return PyMethod_New(func, obj, type);
return PyMethod_New(func, obj, type);
}
}
static int %(binding_cfunc)s_init(void) {
static PyObject*
%(binding_cfunc)s_type = PyCFunction_Type;
__Pyx_CyFunction_repr(__pyx_CyFunctionObject *op)
%(binding_cfunc)s_type.tp_name = __Pyx_NAMESTR("cython_binding_builtin_function_or_method");
{
%(binding_cfunc)s_type.tp_dealloc = (destructor)%(binding_cfunc)s_dealloc;
PyObject *func_name = __Pyx_CyFunction_get_name(op);
%(binding_cfunc)s_type.tp_descr_get = %(binding_cfunc)s_descr_get;
if (PyType_Ready(&%(binding_cfunc)s_type) < 0) {
#if PY_MAJOR_VERSION >= 3
return PyUnicode_FromFormat("<cyfunction %U at %p>",
func_name, op);
#else
return PyString_FromFormat("<cyfunction %s at %p>",
PyString_AsString(func_name), op);
#endif
}
static PyTypeObject __pyx_CyFunctionType_type = {
PyVarObject_HEAD_INIT(0, 0)
__Pyx_NAMESTR("cython_function_or_method"), /*tp_name*/
sizeof(__pyx_CyFunctionObject), /*tp_basicsize*/
0, /*tp_itemsize*/
(destructor) __Pyx_CyFunction_dealloc, /*tp_dealloc*/
0, /*tp_print*/
0, /*tp_getattr*/
0, /*tp_setattr*/
#if PY_MAJOR_VERSION < 3
0, /*tp_compare*/
#else
0, /*reserved*/
#endif
(reprfunc) __Pyx_CyFunction_repr, /*tp_repr*/
0, /*tp_as_number*/
0, /*tp_as_sequence*/
0, /*tp_as_mapping*/
0, /*tp_hash*/
PyCFunction_Call, /*tp_call*/
0, /*tp_str*/
0, /*tp_getattro*/
0, /*tp_setattro*/
0, /*tp_as_buffer*/
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC, /* tp_flags*/
0, /*tp_doc*/
(traverseproc) __Pyx_CyFunction_traverse, /*tp_traverse*/
0, /*tp_clear*/
0, /*tp_richcompare*/
offsetof(__pyx_CyFunctionObject, func_weakreflist), /* tp_weaklistoffse */
0, /*tp_iter*/
0, /*tp_iternext*/
__pyx_CyFunction_methods, /*tp_methods*/
__pyx_CyFunction_members, /*tp_members*/
__pyx_CyFunction_getsets, /*tp_getset*/
0, /*tp_base*/
0, /*tp_dict*/
__Pyx_CyFunction_descr_get, /*tp_descr_get*/
0, /*tp_descr_set*/
offsetof(__pyx_CyFunctionObject, func_dict),/*tp_dictoffset*/
0, /*tp_init*/
0, /*tp_alloc*/
0, /*tp_new*/
0, /*tp_free*/
0, /*tp_is_gc*/
0, /*tp_bases*/
0, /*tp_mro*/
0, /*tp_cache*/
0, /*tp_subclasses*/
0, /*tp_weaklist*/
0, /*tp_del*/
#if PY_VERSION_HEX >= 0x02060000
0, /*tp_version_tag*/
#endif
};
static int __Pyx_CyFunction_init(void)
{
if (PyType_Ready(&__pyx_CyFunctionType_type) < 0)
return -1;
return -1;
}
__pyx_CyFunctionType = &__pyx_CyFunctionType_type;
%(binding_cfunc)s = &%(binding_cfunc)s_type;
return 0;
return 0;
}
}
"""
%
Naming
.
__dict__
)
"""
)
generator_utility_code
=
UtilityCode
(
generator_utility_code
=
UtilityCode
(
proto
=
"""
proto
=
"""
...
...
Cython/Compiler/ModuleNode.py
View file @
50a0470d
...
@@ -1869,8 +1869,8 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
...
@@ -1869,8 +1869,8 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code
.
putln
(
"%s = PyTuple_New(0); %s"
%
(
Naming
.
empty_tuple
,
code
.
error_goto_if_null
(
Naming
.
empty_tuple
,
self
.
pos
)));
code
.
putln
(
"%s = PyTuple_New(0); %s"
%
(
Naming
.
empty_tuple
,
code
.
error_goto_if_null
(
Naming
.
empty_tuple
,
self
.
pos
)));
code
.
putln
(
"%s = PyBytes_FromStringAndSize(
\
"
\
"
, 0); %s"
%
(
Naming
.
empty_bytes
,
code
.
error_goto_if_null
(
Naming
.
empty_bytes
,
self
.
pos
)));
code
.
putln
(
"%s = PyBytes_FromStringAndSize(
\
"
\
"
, 0); %s"
%
(
Naming
.
empty_bytes
,
code
.
error_goto_if_null
(
Naming
.
empty_bytes
,
self
.
pos
)));
code
.
putln
(
"#ifdef
%s_USED"
%
Naming
.
binding_cfunc
)
code
.
putln
(
"#ifdef
__Pyx_CyFunction_USED"
)
code
.
putln
(
"if (
%s_init() < 0) %s"
%
(
Naming
.
binding_cfunc
,
code
.
error_goto
(
self
.
pos
)
))
code
.
putln
(
"if (
__Pyx_CyFunction_init() < 0) %s"
%
code
.
error_goto
(
self
.
pos
))
code
.
putln
(
"#endif"
)
code
.
putln
(
"#endif"
)
code
.
putln
(
"/*--- Library function declarations ---*/"
)
code
.
putln
(
"/*--- Library function declarations ---*/"
)
...
...
Cython/Compiler/Symtab.py
View file @
50a0470d
...
@@ -2047,8 +2047,8 @@ static PyObject* __Pyx_Method_ClassMethod(PyObject *method) {
...
@@ -2047,8 +2047,8 @@ static PyObject* __Pyx_Method_ClassMethod(PyObject *method) {
else if (PyCFunction_Check(method)) {
else if (PyCFunction_Check(method)) {
return PyClassMethod_New(method);
return PyClassMethod_New(method);
}
}
#ifdef __
pyx_binding_PyCFunctionType
_USED
#ifdef __
Pyx_CyFunction
_USED
else if (PyObject_TypeCheck(method, __pyx_
binding_PyCFunctionType)) { /* binded CFunction */
else if (PyObject_TypeCheck(method, __pyx_
CyFunctionType)) {
return PyClassMethod_New(method);
return PyClassMethod_New(method);
}
}
#endif
#endif
...
...
runtests.py
View file @
50a0470d
...
@@ -433,6 +433,7 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -433,6 +433,7 @@ class CythonCompileTestCase(unittest.TestCase):
from Cython.Compiler import Options
from Cython.Compiler import Options
self._saved_options = [ (name, getattr(Options, name))
self._saved_options = [ (name, getattr(Options, name))
for name in ('warning_errors', 'error_on_unknown_names') ]
for name in ('warning_errors', 'error_on_unknown_names') ]
self._saved_default_directives = Options.directive_defaults.items()
Options.warning_errors = self.warning_errors
Options.warning_errors = self.warning_errors
if self.workdir not in sys.path:
if self.workdir not in sys.path:
...
@@ -442,6 +443,7 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -442,6 +443,7 @@ class CythonCompileTestCase(unittest.TestCase):
from Cython.Compiler import Options
from Cython.Compiler import Options
for name, value in self._saved_options:
for name, value in self._saved_options:
setattr(Options, name, value)
setattr(Options, name, value)
Options.directive_defaults = dict(self._saved_default_directives)
try:
try:
sys.path.remove(self.workdir)
sys.path.remove(self.workdir)
...
@@ -849,6 +851,7 @@ class CythonPyregrTestCase(CythonRunTestCase):
...
@@ -849,6 +851,7 @@ class CythonPyregrTestCase(CythonRunTestCase):
CythonRunTestCase.setUp(self)
CythonRunTestCase.setUp(self)
from Cython.Compiler import Options
from Cython.Compiler import Options
Options.error_on_unknown_names = False
Options.error_on_unknown_names = False
Options.directive_defaults['binding'] = True
def _run_unittest(self, result, *classes):
def _run_unittest(self, result, *classes):
"""Run tests from unittest.TestCase-derived classes."""
"""Run tests from unittest.TestCase-derived classes."""
...
...
tests/run/cyfunction.pyx
0 → 100644
View file @
50a0470d
# cython: binding=True
# mode: run
# tag: cyfunction
def
test_dict
():
"""
>>> test_dict.foo = 123
>>> test_dict.__dict__
{'foo': 123}
>>> test_dict.__dict__ = {'bar': 321}
>>> test_dict.__dict__
{'bar': 321}
>>> test_dict.func_dict
{'bar': 321}
"""
def
test_name
():
"""
>>> test_name.__name__
'test_name'
>>> test_name.func_name
'test_name'
>>> test_name.__name__ = 123 #doctest:+ELLIPSIS
Traceback (most recent call last):
TypeError: __name__ must be set to a ... object
>>> test_name.__name__ = 'foo'
>>> test_name.__name__
'foo'
"""
def
test_doc
():
"""
>>> del test_doc.__doc__
>>> test_doc.__doc__
>>> test_doc.__doc__ = 'docstring'
>>> test_doc.__doc__
'docstring'
>>> test_doc.func_doc
'docstring'
"""
def
test_reduce
():
"""
>>> import pickle
>>> pickle.loads(pickle.dumps(test_reduce))()
'Hello, world!'
"""
return
'Hello, world!'
def
test_method
(
self
):
return
self
class
BindingTest
:
"""
>>> BindingTest.test_method = test_method
>>> BindingTest.test_method() #doctest:+ELLIPSIS
Traceback (most recent call last):
TypeError: ...
>>> BindingTest().test_method()
<BindingTest instance>
"""
def
__repr__
(
self
):
return
'<BindingTest instance>'
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment