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
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Boxiang Sun
cython
Commits
ede3eb92
Commit
ede3eb92
authored
Jul 14, 2014
by
Robert Bradshaw
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'cdef_closure'
parents
78a0f1fd
914dc35d
Changes
11
Hide whitespace changes
Inline
Side-by-side
Showing
11 changed files
with
148 additions
and
25 deletions
+148
-25
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+11
-1
Cython/Compiler/ParseTreeTransforms.py
Cython/Compiler/ParseTreeTransforms.py
+7
-4
runtests.py
runtests.py
+103
-13
tests/bugs.txt
tests/bugs.txt
+0
-1
tests/errors/e_cdef_closure.pyx
tests/errors/e_cdef_closure.pyx
+1
-6
tests/run/closure_self.pyx
tests/run/closure_self.pyx
+16
-0
tests/run/closure_tests_1.pyx
tests/run/closure_tests_1.pyx
+2
-0
tests/run/closure_tests_2.pyx
tests/run/closure_tests_2.pyx
+2
-0
tests/run/closure_tests_3.pyx
tests/run/closure_tests_3.pyx
+2
-0
tests/run/closure_tests_4.pyx
tests/run/closure_tests_4.pyx
+2
-0
tests/run/closures_T82.pyx
tests/run/closures_T82.pyx
+2
-0
No files found.
Cython/Compiler/Nodes.py
View file @
ede3eb92
...
@@ -2395,8 +2395,8 @@ class CFuncDefNode(FuncDefNode):
...
@@ -2395,8 +2395,8 @@ class CFuncDefNode(FuncDefNode):
def
generate_argument_parsing_code
(
self
,
env
,
code
):
def
generate_argument_parsing_code
(
self
,
env
,
code
):
i
=
0
i
=
0
used
=
0
used
=
0
scope
=
self
.
local_scope
if
self
.
type
.
optional_arg_count
:
if
self
.
type
.
optional_arg_count
:
scope
=
self
.
local_scope
code
.
putln
(
'if (%s) {'
%
Naming
.
optional_args_cname
)
code
.
putln
(
'if (%s) {'
%
Naming
.
optional_args_cname
)
for
arg
in
self
.
args
:
for
arg
in
self
.
args
:
if
arg
.
default
:
if
arg
.
default
:
...
@@ -2417,6 +2417,16 @@ class CFuncDefNode(FuncDefNode):
...
@@ -2417,6 +2417,16 @@ class CFuncDefNode(FuncDefNode):
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
code
.
putln
(
'}'
)
# Move arguments into closure if required
def
put_into_closure
(
entry
):
if
entry
.
in_closure
and
not
arg
.
default
:
code
.
putln
(
'%s = %s;'
%
(
entry
.
cname
,
entry
.
original_cname
))
code
.
put_var_incref
(
entry
)
code
.
put_var_giveref
(
entry
)
for
arg
in
self
.
args
:
put_into_closure
(
scope
.
lookup_here
(
arg
.
name
))
def
generate_argument_conversion_code
(
self
,
code
):
def
generate_argument_conversion_code
(
self
,
code
):
pass
pass
...
...
Cython/Compiler/ParseTreeTransforms.py
View file @
ede3eb92
...
@@ -2261,8 +2261,8 @@ class MarkClosureVisitor(CythonTransform):
...
@@ -2261,8 +2261,8 @@ class MarkClosureVisitor(CythonTransform):
def
visit_CFuncDefNode
(
self
,
node
):
def
visit_CFuncDefNode
(
self
,
node
):
self
.
visit_FuncDefNode
(
node
)
self
.
visit_FuncDefNode
(
node
)
if
node
.
needs_closure
:
if
node
.
needs_closure
and
node
.
overridable
:
error
(
node
.
pos
,
"closures inside cdef functions not yet supported"
)
error
(
node
.
pos
,
"closures inside c
p
def functions not yet supported"
)
return
node
return
node
def
visit_LambdaNode
(
self
,
node
):
def
visit_LambdaNode
(
self
,
node
):
...
@@ -2401,8 +2401,11 @@ class CreateClosureClasses(CythonTransform):
...
@@ -2401,8 +2401,11 @@ class CreateClosureClasses(CythonTransform):
return
node
return
node
def
visit_CFuncDefNode
(
self
,
node
):
def
visit_CFuncDefNode
(
self
,
node
):
self
.
visitchildren
(
node
)
if
not
node
.
overridable
:
return
node
return
self
.
visit_FuncDefNode
(
node
)
else
:
self
.
visitchildren
(
node
)
return
node
class
GilCheck
(
VisitorTransform
):
class
GilCheck
(
VisitorTransform
):
...
...
runtests.py
View file @
ede3eb92
...
@@ -115,6 +115,77 @@ def unpatch_inspect_isfunction():
...
@@ -115,6 +115,77 @@ def unpatch_inspect_isfunction():
else
:
else
:
inspect
.
isfunction
=
orig_isfunction
inspect
.
isfunction
=
orig_isfunction
def
def_to_cdef
(
source
):
'''
Converts the module-level def methods into cdef methods, i.e.
@decorator
def foo([args]):
"""
[tests]
"""
[body]
becomes
def foo([args]):
"""
[tests]
"""
return foo_c([args])
cdef foo_c([args]):
[body]
'''
output
=
[]
skip
=
False
def_node
=
re
.
compile
(
r'def (\
w+)
\(([^()*]*)\
):
').match
lines = iter(source.split('
\
n
'))
for line in lines:
if not line.strip():
output.append(line)
continue
if skip:
if line[0] != '
':
skip = False
else:
continue
if line[0] == '
@
':
skip = True
continue
m = def_node(line)
if m:
name = m.group(1)
args = m.group(2)
if args:
args_no_types = ", ".join(arg.split()[-1] for arg in args.split('
,
'))
else:
args_no_types = ""
output.append("def %s(%s):" % (name, args_no_types))
line = next(lines)
if '"""' in line:
has_docstring = True
output.append(line)
for line in lines:
output.append(line)
if '"""' in line:
break
else:
has_docstring = False
output.append(" return %s_c(%s)" % (name, args_no_types))
output.append('')
output.append("cdef %s_c(%s):" % (name, args))
if not has_docstring:
output.append(line)
else:
output.append(line)
return '
\
n
'.join(output)
def update_linetrace_extension(ext):
def update_linetrace_extension(ext):
ext.define_macros.append(('
CYTHON_TRACE
', 1))
ext.define_macros.append(('
CYTHON_TRACE
', 1))
return ext
return ext
...
@@ -331,7 +402,7 @@ def parse_tags(filepath):
...
@@ -331,7 +402,7 @@ def parse_tags(filepath):
if tag == 'tags':
if tag == 'tags':
tag = 'tag'
tag = 'tag'
print("WARNING: test tags use the 'tag' directive, not 'tags' (%s)" % filepath)
print("WARNING: test tags use the 'tag' directive, not 'tags' (%s)" % filepath)
if tag not in ('mode', 'tag', 'ticket', 'cython', 'distutils'):
if tag not in ('mode', 'tag', 'ticket', 'cython', 'distutils'
, 'preparse'
):
print("WARNING: unknown test directive '%s' found (%s)" % (tag, filepath))
print("WARNING: unknown test directive '%s' found (%s)" % (tag, filepath))
values = values.split(',')
values = values.split(',')
tags[tag].extend(filter(None, [value.strip() for value in values]))
tags[tag].extend(filter(None, [value.strip() for value in values]))
...
@@ -532,19 +603,25 @@ class TestBuilder(object):
...
@@ -532,19 +603,25 @@ class TestBuilder(object):
elif 'no-cpp' in tags['tag'] and 'cpp' in self.languages:
elif 'no-cpp' in tags['tag'] and 'cpp' in self.languages:
languages = list(languages)
languages = list(languages)
languages.remove('cpp')
languages.remove('cpp')
preparse_list = tags.get('preparse', ['id'])
tests = [ self.build_test(test_class, path, workdir, module, tags,
tests = [ self.build_test(test_class, path, workdir, module, tags,
language, expect_errors, warning_errors)
language, expect_errors, warning_errors, preparse)
for language in languages ]
for language in languages
for preparse in preparse_list ]
return tests
return tests
def build_test(self, test_class, path, workdir, module, tags,
def build_test(self, test_class, path, workdir, module, tags,
language, expect_errors, warning_errors):
language, expect_errors, warning_errors
, preparse
):
language_workdir = os.path.join(workdir, language)
language_workdir = os.path.join(workdir, language)
if not os.path.exists(language_workdir):
if not os.path.exists(language_workdir):
os.makedirs(language_workdir)
os.makedirs(language_workdir)
workdir = os.path.join(language_workdir, module)
workdir = os.path.join(language_workdir, module)
if preparse != 'id':
workdir += '_%s' % str(preparse)
return test_class(path, workdir, module, tags,
return test_class(path, workdir, module, tags,
language=language,
language=language,
preparse=preparse,
expect_errors=expect_errors,
expect_errors=expect_errors,
annotate=self.annotate,
annotate=self.annotate,
cleanup_workdir=self.cleanup_workdir,
cleanup_workdir=self.cleanup_workdir,
...
@@ -556,7 +633,7 @@ class TestBuilder(object):
...
@@ -556,7 +633,7 @@ class TestBuilder(object):
warning_errors=warning_errors)
warning_errors=warning_errors)
class CythonCompileTestCase(unittest.TestCase):
class CythonCompileTestCase(unittest.TestCase):
def __init__(self, test_directory, workdir, module, tags, language='c',
def __init__(self, test_directory, workdir, module, tags, language='c',
preparse='id',
expect_errors=False, annotate=False, cleanup_workdir=True,
expect_errors=False, annotate=False, cleanup_workdir=True,
cleanup_sharedlibs=True, cleanup_failures=True, cython_only=False,
cleanup_sharedlibs=True, cleanup_failures=True, cython_only=False,
fork=True, language_level=2, warning_errors=False):
fork=True, language_level=2, warning_errors=False):
...
@@ -565,6 +642,8 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -565,6 +642,8 @@ class CythonCompileTestCase(unittest.TestCase):
self.workdir = workdir
self.workdir = workdir
self.module = module
self.module = module
self.language = language
self.language = language
self.preparse = preparse
self.name = module if self.preparse == "id" else "%s_%s" % (module, preparse)
self.expect_errors = expect_errors
self.expect_errors = expect_errors
self.annotate = annotate
self.annotate = annotate
self.cleanup_workdir = cleanup_workdir
self.cleanup_workdir = cleanup_workdir
...
@@ -577,7 +656,7 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -577,7 +656,7 @@ class CythonCompileTestCase(unittest.TestCase):
unittest.TestCase.__init__(self)
unittest.TestCase.__init__(self)
def shortDescription(self):
def shortDescription(self):
return "
compiling
(
%
s
)
%
s
" % (self.language, self.
modul
e)
return "compiling (%s) %s" % (self.language, self.
nam
e)
def setUp(self):
def setUp(self):
from Cython.Compiler import Options
from Cython.Compiler import Options
...
@@ -660,11 +739,16 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -660,11 +739,16 @@ class CythonCompileTestCase(unittest.TestCase):
if is_related(filename)]
if is_related(filename)]
def copy_files(self, test_directory, target_directory, file_list):
def copy_files(self, test_directory, target_directory, file_list):
# use symlink on Unix, copy on Windows
if self.preparse and self.preparse != 'id':
try:
preparse_func = globals()[self.preparse]
copy = os.symlink
def copy(src, dest):
except AttributeError:
open(dest, 'w').write(preparse_func(open(src).read()))
copy = shutil.copy
else:
# use symlink on Unix, copy on Windows
try:
copy = os.symlink
except AttributeError:
copy = shutil.copy
join = os.path.join
join = os.path.join
for filename in file_list:
for filename in file_list:
...
@@ -707,6 +791,12 @@ class CythonCompileTestCase(unittest.TestCase):
...
@@ -707,6 +791,12 @@ class CythonCompileTestCase(unittest.TestCase):
include_dirs.append(incdir)
include_dirs.append(incdir)
source = self.find_module_source_file(
source = self.find_module_source_file(
os.path.join(test_directory, module + '.pyx'))
os.path.join(test_directory, module + '.pyx'))
if self.preparse == 'id':
source = self.find_module_source_file(
os.path.join(test_directory, module + '.pyx'))
else:
self.copy_files(test_directory, targetdir, [module + '.pyx'])
source = os.path.join(targetdir, module + '.pyx')
target = os.path.join(targetdir, self.build_target_filename(module))
target = os.path.join(targetdir, self.build_target_filename(module))
if extra_compile_options is None:
if extra_compile_options is None:
...
@@ -903,7 +993,7 @@ class CythonRunTestCase(CythonCompileTestCase):
...
@@ -903,7 +993,7 @@ class CythonRunTestCase(CythonCompileTestCase):
if self.cython_only:
if self.cython_only:
return CythonCompileTestCase.shortDescription(self)
return CythonCompileTestCase.shortDescription(self)
else:
else:
return "
compiling
(
%
s
)
and
running
%
s
" % (self.language, self.
modul
e)
return "compiling (%s) and running %s" % (self.language, self.
nam
e)
def run(self, result=None):
def run(self, result=None):
if result is None:
if result is None:
...
@@ -1105,7 +1195,7 @@ class PartialTestResult(_TextTestResult):
...
@@ -1105,7 +1195,7 @@ class PartialTestResult(_TextTestResult):
class CythonUnitTestCase(CythonRunTestCase):
class CythonUnitTestCase(CythonRunTestCase):
def shortDescription(self):
def shortDescription(self):
return "
compiling
(
%
s
)
tests
in
%
s
" % (self.language, self.
modul
e)
return "compiling (%s) tests in %s" % (self.language, self.
nam
e)
def run_tests(self, result, ext_so_path):
def run_tests(self, result, ext_so_path):
module = import_ext(self.module, ext_so_path)
module = import_ext(self.module, ext_so_path)
...
...
tests/bugs.txt
View file @
ede3eb92
...
@@ -6,7 +6,6 @@ unsignedbehaviour_T184
...
@@ -6,7 +6,6 @@ unsignedbehaviour_T184
missing_baseclass_in_predecl_T262
missing_baseclass_in_predecl_T262
cfunc_call_tuple_args_T408
cfunc_call_tuple_args_T408
cpp_structs
cpp_structs
closure_inside_cdef_T554
genexpr_iterable_lookup_T600
genexpr_iterable_lookup_T600
generator_expressions_in_class
generator_expressions_in_class
for_from_pyvar_loop_T601
for_from_pyvar_loop_T601
...
...
tests/errors/e_cdef_closure.pyx
View file @
ede3eb92
# mode: error
# mode: error
cdef
cdef
_yield
():
def
inner
():
pass
cpdef
cpdef
_yield
():
cpdef
cpdef
_yield
():
def
inner
():
def
inner
():
pass
pass
_ERRORS
=
u"""
_ERRORS
=
u"""
3:5: closures inside cdef functions not yet supported
3:6: closures inside cpdef functions not yet supported
7:6: closures inside cdef functions not yet supported
"""
"""
tests/run/closure_self.pyx
View file @
ede3eb92
...
@@ -38,3 +38,19 @@ cdef class SelfInClosure(object):
...
@@ -38,3 +38,19 @@ cdef class SelfInClosure(object):
def
nested
():
def
nested
():
return
self
.
x
,
t
.
x
return
self
.
x
,
t
.
x
return
nested
return
nested
def
call_closure_method_cdef_attr_c
(
self
,
Test
t
):
"""
>>> o = SelfInClosure()
>>> o.call_closure_method_cdef_attr_c(Test())()
(1, 2)
"""
return
self
.
closure_method_cdef_attr_c
(
t
)
cdef
closure_method_cdef_attr_c
(
self
,
Test
t
):
t
.
x
=
2
self
.
_t
=
t
self
.
x
=
1
def
nested
():
return
self
.
x
,
t
.
x
return
nested
tests/run/closure_tests_1.pyx
View file @
ede3eb92
# mode: run
# mode: run
# tag: closures
# tag: closures
# preparse: id
# preparse: def_to_cdef
#
#
# closure_tests_1.pyx
# closure_tests_1.pyx
#
#
...
...
tests/run/closure_tests_2.pyx
View file @
ede3eb92
# mode: run
# mode: run
# tag: closures
# tag: closures
# preparse: id
# preparse: def_to_cdef
#
#
# closure_tests_2.pyx
# closure_tests_2.pyx
#
#
...
...
tests/run/closure_tests_3.pyx
View file @
ede3eb92
# mode: run
# mode: run
# tag: closures
# tag: closures
# preparse: id
# preparse: def_to_cdef
#
#
# closure_tests_3.pyx
# closure_tests_3.pyx
#
#
...
...
tests/run/closure_tests_4.pyx
View file @
ede3eb92
# mode: run
# mode: run
# tag: closures
# tag: closures
# preparse: id
# preparse: def_to_cdef
#
#
# closure_tests_4.pyx
# closure_tests_4.pyx
#
#
...
...
tests/run/closures_T82.pyx
View file @
ede3eb92
# mode: run
# mode: run
# tag: closures
# tag: closures
# ticket: 82
# ticket: 82
# preparse: id
# preparse: def_to_cdef
cimport
cython
cimport
cython
...
...
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