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
d61b5318
Commit
d61b5318
authored
May 03, 2009
by
Stefan Behnel
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
lambda expressions
parent
7bcb1102
Changes
9
Show whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
236 additions
and
46 deletions
+236
-46
Cython/Compiler/ExprNodes.py
Cython/Compiler/ExprNodes.py
+35
-4
Cython/Compiler/ModuleNode.py
Cython/Compiler/ModuleNode.py
+4
-0
Cython/Compiler/Naming.py
Cython/Compiler/Naming.py
+1
-0
Cython/Compiler/Nodes.py
Cython/Compiler/Nodes.py
+28
-10
Cython/Compiler/Optimize.py
Cython/Compiler/Optimize.py
+2
-2
Cython/Compiler/ParseTreeTransforms.py
Cython/Compiler/ParseTreeTransforms.py
+34
-2
Cython/Compiler/Parsing.py
Cython/Compiler/Parsing.py
+69
-27
Cython/Compiler/Symtab.py
Cython/Compiler/Symtab.py
+15
-1
tests/run/lambda_T195.pyx
tests/run/lambda_T195.pyx
+48
-0
No files found.
Cython/Compiler/ExprNodes.py
View file @
d61b5318
...
@@ -3674,7 +3674,7 @@ class UnboundMethodNode(ExprNode):
...
@@ -3674,7 +3674,7 @@ class UnboundMethodNode(ExprNode):
code
.
error_goto_if_null
(
self
.
result
(),
self
.
pos
)))
code
.
error_goto_if_null
(
self
.
result
(),
self
.
pos
)))
code
.
put_gotref
(
self
.
py_result
())
code
.
put_gotref
(
self
.
py_result
())
class
PyCFunctionNode
(
Atomic
ExprNode
):
class
PyCFunctionNode
(
ExprNode
):
# Helper class used in the implementation of Python
# Helper class used in the implementation of Python
# class definitions. Constructs a PyCFunction object
# class definitions. Constructs a PyCFunction object
# from a PyMethodDef struct.
# from a PyMethodDef struct.
...
@@ -3682,6 +3682,7 @@ class PyCFunctionNode(AtomicExprNode):
...
@@ -3682,6 +3682,7 @@ class PyCFunctionNode(AtomicExprNode):
# pymethdef_cname string PyMethodDef structure
# pymethdef_cname string PyMethodDef structure
# self_object ExprNode or None
# self_object ExprNode or None
subexprs
=
[]
self_object
=
None
self_object
=
None
def
analyse_types
(
self
,
env
):
def
analyse_types
(
self
,
env
):
...
@@ -3690,19 +3691,49 @@ class PyCFunctionNode(AtomicExprNode):
...
@@ -3690,19 +3691,49 @@ class PyCFunctionNode(AtomicExprNode):
gil_message
=
"Constructing Python function"
gil_message
=
"Constructing Python function"
def
generate_result_code
(
self
,
code
):
def
self_result_code
(
self
):
if
self
.
self_object
is
None
:
if
self
.
self_object
is
None
:
self_result
=
"NULL"
self_result
=
"NULL"
else
:
else
:
self_result
=
self
.
self_object
.
py_result
()
self_result
=
self
.
self_object
.
py_result
()
return
self_result
def
generate_result_code
(
self
,
code
):
code
.
putln
(
code
.
putln
(
"%s = PyCFunction_New(&%s, %s); %s"
%
(
"%s = PyCFunction_New(&%s, %s); %s"
%
(
self
.
result
(),
self
.
result
(),
self
.
pymethdef_cname
,
self
.
pymethdef_cname
,
self
_result
,
self
.
self_result_code
()
,
code
.
error_goto_if_null
(
self
.
result
(),
self
.
pos
)))
code
.
error_goto_if_null
(
self
.
result
(),
self
.
pos
)))
code
.
put_gotref
(
self
.
py_result
())
code
.
put_gotref
(
self
.
py_result
())
class
InnerFunctionNode
(
PyCFunctionNode
):
# Special PyCFunctionNode that depends on a closure class
#
def
self_result_code
(
self
):
return
"((PyObject*)%s)"
%
Naming
.
cur_scope_cname
class
LambdaNode
(
InnerFunctionNode
):
# Lambda expression node (only used as a function reference)
#
# args [CArgDeclNode] formal arguments
# star_arg PyArgDeclNode or None * argument
# starstar_arg PyArgDeclNode or None ** argument
# lambda_name string a module-globally unique lambda name
# result_expr ExprNode
# def_node DefNode the underlying function 'def' node
child_attrs
=
[
'def_node'
]
def_node
=
None
name
=
StringEncoding
.
EncodedString
(
'<lambda>'
)
def
analyse_declarations
(
self
,
env
):
#self.def_node.needs_closure = self.needs_closure
self
.
def_node
.
analyse_declarations
(
env
)
self
.
pymethdef_cname
=
self
.
def_node
.
entry
.
pymethdef_cname
env
.
add_lambda_def
(
self
.
def_node
)
#-------------------------------------------------------------------
#-------------------------------------------------------------------
#
#
# Unary operator nodes
# Unary operator nodes
...
...
Cython/Compiler/ModuleNode.py
View file @
d61b5318
...
@@ -258,6 +258,10 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
...
@@ -258,6 +258,10 @@ class ModuleNode(Nodes.Node, Nodes.BlockNode):
code
.
globalstate
.
insert_global_var_declarations_into
(
code
)
code
.
globalstate
.
insert_global_var_declarations_into
(
code
)
self
.
generate_cached_builtins_decls
(
env
,
code
)
self
.
generate_cached_builtins_decls
(
env
,
code
)
# generate lambda function definitions
for
node
in
env
.
lambda_defs
:
node
.
generate_function_definitions
(
env
,
code
)
# generate normal function definitions
self
.
body
.
generate_function_definitions
(
env
,
code
)
self
.
body
.
generate_function_definitions
(
env
,
code
)
code
.
mark_pos
(
None
)
code
.
mark_pos
(
None
)
self
.
generate_typeobj_definitions
(
env
,
code
)
self
.
generate_typeobj_definitions
(
env
,
code
)
...
...
Cython/Compiler/Naming.py
View file @
d61b5318
...
@@ -46,6 +46,7 @@ opt_arg_prefix = pyrex_prefix + "opt_args_"
...
@@ -46,6 +46,7 @@ opt_arg_prefix = pyrex_prefix + "opt_args_"
convert_func_prefix
=
pyrex_prefix
+
"convert_"
convert_func_prefix
=
pyrex_prefix
+
"convert_"
closure_scope_prefix
=
pyrex_prefix
+
"scope_"
closure_scope_prefix
=
pyrex_prefix
+
"scope_"
closure_class_prefix
=
pyrex_prefix
+
"scope_struct_"
closure_class_prefix
=
pyrex_prefix
+
"scope_struct_"
lambda_func_prefix
=
pyrex_prefix
+
"lambda_"
args_cname
=
pyrex_prefix
+
"args"
args_cname
=
pyrex_prefix
+
"args"
pykwdlist_cname
=
pyrex_prefix
+
"pyargnames"
pykwdlist_cname
=
pyrex_prefix
+
"pyargnames"
...
...
Cython/Compiler/Nodes.py
View file @
d61b5318
...
@@ -1000,6 +1000,9 @@ class FuncDefNode(StatNode, BlockNode):
...
@@ -1000,6 +1000,9 @@ class FuncDefNode(StatNode, BlockNode):
lenv
.
mangle_closure_cnames
(
outer_scope_cname
)
lenv
.
mangle_closure_cnames
(
outer_scope_cname
)
# Generate closure function definitions
# Generate closure function definitions
self
.
body
.
generate_function_definitions
(
lenv
,
code
)
self
.
body
.
generate_function_definitions
(
lenv
,
code
)
# generate lambda function definitions
for
node
in
lenv
.
lambda_defs
:
node
.
generate_function_definitions
(
lenv
,
code
)
is_getbuffer_slot
=
(
self
.
entry
.
name
==
"__getbuffer__"
and
is_getbuffer_slot
=
(
self
.
entry
.
name
==
"__getbuffer__"
and
self
.
entry
.
scope
.
is_c_class_scope
)
self
.
entry
.
scope
.
is_c_class_scope
)
...
@@ -1013,12 +1016,13 @@ class FuncDefNode(StatNode, BlockNode):
...
@@ -1013,12 +1016,13 @@ class FuncDefNode(StatNode, BlockNode):
self
.
generate_cached_builtins_decls
(
lenv
,
code
)
self
.
generate_cached_builtins_decls
(
lenv
,
code
)
# ----- Function header
# ----- Function header
code
.
putln
(
""
)
code
.
putln
(
""
)
with_pymethdef
=
env
.
is_py_class_scope
or
env
.
is_closure_scope
if
self
.
py_func
:
if
self
.
py_func
:
self
.
py_func
.
generate_function_header
(
code
,
self
.
py_func
.
generate_function_header
(
code
,
with_pymethdef
=
env
.
is_py_class_scope
or
env
.
is_closure_scope
,
with_pymethdef
=
with_pymethdef
,
proto_only
=
True
)
proto_only
=
True
)
self
.
generate_function_header
(
code
,
self
.
generate_function_header
(
code
,
with_pymethdef
=
env
.
is_py_class_scope
or
env
.
is_closure_scope
)
with_pymethdef
=
with_pymethdef
)
# ----- Local variable declarations
# ----- Local variable declarations
if
lenv
.
is_closure_scope
:
if
lenv
.
is_closure_scope
:
code
.
put
(
lenv
.
scope_class
.
type
.
declaration_code
(
Naming
.
cur_scope_cname
))
code
.
put
(
lenv
.
scope_class
.
type
.
declaration_code
(
Naming
.
cur_scope_cname
))
...
@@ -1546,6 +1550,7 @@ class DefNode(FuncDefNode):
...
@@ -1546,6 +1550,7 @@ class DefNode(FuncDefNode):
# A Python function definition.
# A Python function definition.
#
#
# name string the Python name of the function
# name string the Python name of the function
# lambda_name string the internal name of a lambda 'function'
# decorators [DecoratorNode] list of decorators
# decorators [DecoratorNode] list of decorators
# args [CArgDeclNode] formal arguments
# args [CArgDeclNode] formal arguments
# star_arg PyArgDeclNode or None * argument
# star_arg PyArgDeclNode or None * argument
...
@@ -1560,6 +1565,7 @@ class DefNode(FuncDefNode):
...
@@ -1560,6 +1565,7 @@ class DefNode(FuncDefNode):
child_attrs
=
[
"args"
,
"star_arg"
,
"starstar_arg"
,
"body"
,
"decorators"
]
child_attrs
=
[
"args"
,
"star_arg"
,
"starstar_arg"
,
"body"
,
"decorators"
]
lambda_name
=
None
assmt
=
None
assmt
=
None
num_kwonly_args
=
0
num_kwonly_args
=
0
num_required_kw_args
=
0
num_required_kw_args
=
0
...
@@ -1675,6 +1681,9 @@ class DefNode(FuncDefNode):
...
@@ -1675,6 +1681,9 @@ class DefNode(FuncDefNode):
if
arg
.
not_none
and
not
arg
.
type
.
is_extension_type
:
if
arg
.
not_none
and
not
arg
.
type
.
is_extension_type
:
error
(
self
.
pos
,
error
(
self
.
pos
,
"Only extension type arguments can have 'not None'"
)
"Only extension type arguments can have 'not None'"
)
if
self
.
name
==
'<lambda>'
:
self
.
declare_lambda_function
(
env
)
else
:
self
.
declare_pyfunction
(
env
)
self
.
declare_pyfunction
(
env
)
self
.
analyse_signature
(
env
)
self
.
analyse_signature
(
env
)
self
.
return_type
=
self
.
entry
.
signature
.
return_type
()
self
.
return_type
=
self
.
entry
.
signature
.
return_type
()
...
@@ -1760,10 +1769,10 @@ class DefNode(FuncDefNode):
...
@@ -1760,10 +1769,10 @@ class DefNode(FuncDefNode):
def
declare_pyfunction
(
self
,
env
):
def
declare_pyfunction
(
self
,
env
):
#print "DefNode.declare_pyfunction:", self.name, "in", env ###
#print "DefNode.declare_pyfunction:", self.name, "in", env ###
name
=
self
.
name
name
=
self
.
name
entry
=
env
.
lookup_here
(
self
.
name
)
entry
=
env
.
lookup_here
(
name
)
if
entry
and
entry
.
type
.
is_cfunction
and
not
self
.
is_wrapper
:
if
entry
and
entry
.
type
.
is_cfunction
and
not
self
.
is_wrapper
:
warning
(
self
.
pos
,
"Overriding cdef method with def method."
,
5
)
warning
(
self
.
pos
,
"Overriding cdef method with def method."
,
5
)
entry
=
env
.
declare_pyfunction
(
self
.
name
,
self
.
pos
)
entry
=
env
.
declare_pyfunction
(
name
,
self
.
pos
)
self
.
entry
=
entry
self
.
entry
=
entry
prefix
=
env
.
scope_prefix
prefix
=
env
.
scope_prefix
entry
.
func_cname
=
\
entry
.
func_cname
=
\
...
@@ -1777,6 +1786,18 @@ class DefNode(FuncDefNode):
...
@@ -1777,6 +1786,18 @@ class DefNode(FuncDefNode):
else
:
else
:
entry
.
doc
=
None
entry
.
doc
=
None
def
declare_lambda_function
(
self
,
env
):
name
=
self
.
name
prefix
=
env
.
scope_prefix
func_cname
=
\
Naming
.
lambda_func_prefix
+
u'funcdef'
+
prefix
+
self
.
lambda_name
entry
=
env
.
declare_lambda_function
(
func_cname
,
self
.
pos
)
entry
.
pymethdef_cname
=
\
Naming
.
lambda_func_prefix
+
u'methdef'
+
prefix
+
self
.
lambda_name
entry
.
qualified_name
=
env
.
qualify_name
(
self
.
lambda_name
)
entry
.
doc
=
None
self
.
entry
=
entry
def
declare_arguments
(
self
,
env
):
def
declare_arguments
(
self
,
env
):
for
arg
in
self
.
args
:
for
arg
in
self
.
args
:
if
not
arg
.
name
:
if
not
arg
.
name
:
...
@@ -1824,11 +1845,8 @@ class DefNode(FuncDefNode):
...
@@ -1824,11 +1845,8 @@ class DefNode(FuncDefNode):
function
=
ExprNodes
.
PyCFunctionNode
(
self
.
pos
,
function
=
ExprNodes
.
PyCFunctionNode
(
self
.
pos
,
pymethdef_cname
=
self
.
entry
.
pymethdef_cname
))
pymethdef_cname
=
self
.
entry
.
pymethdef_cname
))
elif
env
.
is_closure_scope
:
elif
env
.
is_closure_scope
:
self_object
=
ExprNodes
.
TempNode
(
self
.
pos
,
env
.
scope_class
.
type
,
env
)
rhs
=
ExprNodes
.
InnerFunctionNode
(
self_object
.
temp_cname
=
"((PyObject*)%s)"
%
Naming
.
cur_scope_cname
self
.
pos
,
pymethdef_cname
=
self
.
entry
.
pymethdef_cname
)
rhs
=
ExprNodes
.
PyCFunctionNode
(
self
.
pos
,
self_object
=
self_object
,
pymethdef_cname
=
self
.
entry
.
pymethdef_cname
)
self
.
assmt
=
SingleAssignmentNode
(
self
.
pos
,
self
.
assmt
=
SingleAssignmentNode
(
self
.
pos
,
lhs
=
ExprNodes
.
NameNode
(
self
.
pos
,
name
=
self
.
name
),
lhs
=
ExprNodes
.
NameNode
(
self
.
pos
,
name
=
self
.
name
),
rhs
=
rhs
)
rhs
=
rhs
)
...
...
Cython/Compiler/Optimize.py
View file @
d61b5318
...
@@ -748,9 +748,9 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
...
@@ -748,9 +748,9 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
for
child_result
in
children
.
itervalues
():
for
child_result
in
children
.
itervalues
():
if
type
(
child_result
)
is
list
:
if
type
(
child_result
)
is
list
:
for
child
in
child_result
:
for
child
in
child_result
:
if
child
.
constant_result
is
not_a_constant
:
if
getattr
(
child
,
'constant_result'
,
not_a_constant
)
is
not_a_constant
:
return
return
elif
child_result
.
constant_result
is
not_a_constant
:
elif
getattr
(
child_result
,
'constant_result'
,
not_a_constant
)
is
not_a_constant
:
return
return
# now try to calculate the real constant value
# now try to calculate the real constant value
...
...
Cython/Compiler/ParseTreeTransforms.py
View file @
d61b5318
...
@@ -177,6 +177,7 @@ class PostParse(CythonTransform):
...
@@ -177,6 +177,7 @@ class PostParse(CythonTransform):
def
visit_ModuleNode
(
self
,
node
):
def
visit_ModuleNode
(
self
,
node
):
self
.
scope_type
=
'module'
self
.
scope_type
=
'module'
self
.
scope_node
=
node
self
.
scope_node
=
node
self
.
lambda_counter
=
1
self
.
visitchildren
(
node
)
self
.
visitchildren
(
node
)
return
node
return
node
...
@@ -197,6 +198,25 @@ class PostParse(CythonTransform):
...
@@ -197,6 +198,25 @@ class PostParse(CythonTransform):
def
visit_CStructOrUnionDefNode
(
self
,
node
):
def
visit_CStructOrUnionDefNode
(
self
,
node
):
return
self
.
visit_scope
(
node
,
'struct'
)
return
self
.
visit_scope
(
node
,
'struct'
)
def
visit_LambdaNode
(
self
,
node
):
# unpack a lambda expression into the corresponding DefNode
if
self
.
scope_type
!=
'function'
:
error
(
node
.
pos
,
"lambda functions are currently only supported in functions"
)
lambda_id
=
self
.
lambda_counter
self
.
lambda_counter
+=
1
node
.
lambda_name
=
EncodedString
(
u'lambda%d'
%
lambda_id
)
body
=
Nodes
.
ReturnStatNode
(
node
.
result_expr
.
pos
,
value
=
node
.
result_expr
)
node
.
def_node
=
Nodes
.
DefNode
(
node
.
pos
,
name
=
node
.
name
,
lambda_name
=
node
.
lambda_name
,
args
=
node
.
args
,
star_arg
=
node
.
star_arg
,
starstar_arg
=
node
.
starstar_arg
,
body
=
body
)
self
.
visitchildren
(
node
)
return
node
# cdef variables
# cdef variables
def
handle_bufferdefaults
(
self
,
decl
):
def
handle_bufferdefaults
(
self
,
decl
):
if
not
isinstance
(
decl
.
default
,
DictNode
):
if
not
isinstance
(
decl
.
default
,
DictNode
):
...
@@ -693,6 +713,11 @@ property NAME:
...
@@ -693,6 +713,11 @@ property NAME:
self
.
seen_vars_stack
.
pop
()
self
.
seen_vars_stack
.
pop
()
return
node
return
node
def
visit_LambdaNode
(
self
,
node
):
node
.
analyse_declarations
(
self
.
env_stack
[
-
1
])
self
.
visitchildren
(
node
)
return
node
def
visit_FuncDefNode
(
self
,
node
):
def
visit_FuncDefNode
(
self
,
node
):
self
.
seen_vars_stack
.
append
(
set
())
self
.
seen_vars_stack
.
append
(
set
())
lenv
=
node
.
create_local_scope
(
self
.
env_stack
[
-
1
])
lenv
=
node
.
create_local_scope
(
self
.
env_stack
[
-
1
])
...
@@ -846,6 +871,13 @@ class MarkClosureVisitor(CythonTransform):
...
@@ -846,6 +871,13 @@ class MarkClosureVisitor(CythonTransform):
self
.
needs_closure
=
True
self
.
needs_closure
=
True
return
node
return
node
def
visit_LambdaNode
(
self
,
node
):
self
.
needs_closure
=
False
self
.
visitchildren
(
node
)
node
.
needs_closure
=
self
.
needs_closure
self
.
needs_closure
=
True
return
node
def
visit_ClassDefNode
(
self
,
node
):
def
visit_ClassDefNode
(
self
,
node
):
self
.
visitchildren
(
node
)
self
.
visitchildren
(
node
)
self
.
needs_closure
=
True
self
.
needs_closure
=
True
...
...
Cython/Compiler/Parsing.py
View file @
d61b5318
...
@@ -78,7 +78,12 @@ def p_binop_expr(s, ops, p_sub_expr):
...
@@ -78,7 +78,12 @@ def p_binop_expr(s, ops, p_sub_expr):
#expression: or_test [if or_test else test] | lambda_form
#expression: or_test [if or_test else test] | lambda_form
# actually:
#test: or_test ['if' or_test 'else' test] | lambdef
def
p_simple_expr
(
s
):
def
p_simple_expr
(
s
):
if
s
.
sy
==
'lambda'
:
return
p_lambdef
(
s
)
pos
=
s
.
position
()
pos
=
s
.
position
()
expr
=
p_or_test
(
s
)
expr
=
p_or_test
(
s
)
if
s
.
sy
==
'if'
:
if
s
.
sy
==
'if'
:
...
@@ -90,11 +95,45 @@ def p_simple_expr(s):
...
@@ -90,11 +95,45 @@ def p_simple_expr(s):
else
:
else
:
return
expr
return
expr
#test: or_test | lambda_form
#lambdef: 'lambda' [varargslist] ':' test
def
p_lambdef
(
s
,
allow_conditional
=
True
):
# s.sy == 'lambda'
pos
=
s
.
position
()
s
.
next
()
if
s
.
sy
==
':'
:
args
=
[]
star_arg
=
starstar_arg
=
None
else
:
args
,
star_arg
,
starstar_arg
=
p_varargslist
(
s
,
terminator
=
':'
)
s
.
expect
(
':'
)
if
allow_conditional
:
expr
=
p_test
(
s
)
else
:
expr
=
p_test_nocond
(
s
)
return
ExprNodes
.
LambdaNode
(
pos
,
args
=
args
,
star_arg
=
star_arg
,
starstar_arg
=
starstar_arg
,
result_expr
=
expr
)
#lambdef_nocond: 'lambda' [varargslist] ':' test_nocond
def
p_lambdef_nocond
(
s
):
return
p_lambdef
(
s
,
allow_conditional
=
False
)
#test: or_test | lambdef
def
p_test
(
s
):
def
p_test
(
s
):
return
p_or_test
(
s
)
return
p_or_test
(
s
)
#test_nocond: or_test | lambdef_nocond
def
p_test_nocond
(
s
):
if
s
.
sy
==
'lambda'
:
return
p_lambdef_nocond
(
s
)
else
:
return
p_or_test
(
s
)
#or_test: and_test ('or' and_test)*
#or_test: and_test ('or' and_test)*
def
p_or_test
(
s
):
def
p_or_test
(
s
):
...
@@ -694,10 +733,10 @@ def p_string_literal(s):
...
@@ -694,10 +733,10 @@ def p_string_literal(s):
return
kind
,
value
return
kind
,
value
# list_display ::= "[" [listmaker] "]"
# list_display ::= "[" [listmaker] "]"
# listmaker ::= expression (
list
_for | ( "," expression )* [","] )
# listmaker ::= expression (
comp
_for | ( "," expression )* [","] )
#
list_iter ::= list_for | list
_if
#
comp_iter ::= comp_for | comp
_if
#
list_for ::= "for" expression_list "in" testlist [list
_iter]
#
comp_for ::= "for" expression_list "in" testlist [comp
_iter]
#
list_if ::= "if" test [list
_iter]
#
comp_if ::= "if" test [comp
_iter]
def
p_list_maker
(
s
):
def
p_list_maker
(
s
):
# s.sy == '['
# s.sy == '['
...
@@ -711,7 +750,7 @@ def p_list_maker(s):
...
@@ -711,7 +750,7 @@ def p_list_maker(s):
target
=
ExprNodes
.
ListNode
(
pos
,
args
=
[])
target
=
ExprNodes
.
ListNode
(
pos
,
args
=
[])
append
=
ExprNodes
.
ComprehensionAppendNode
(
append
=
ExprNodes
.
ComprehensionAppendNode
(
pos
,
expr
=
expr
,
target
=
ExprNodes
.
CloneNode
(
target
))
pos
,
expr
=
expr
,
target
=
ExprNodes
.
CloneNode
(
target
))
loop
=
p_
list
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
loop
=
p_
comp
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
s
.
expect
(
']'
)
s
.
expect
(
']'
)
return
ExprNodes
.
ComprehensionNode
(
return
ExprNodes
.
ComprehensionNode
(
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
...
@@ -723,32 +762,32 @@ def p_list_maker(s):
...
@@ -723,32 +762,32 @@ def p_list_maker(s):
s
.
expect
(
']'
)
s
.
expect
(
']'
)
return
ExprNodes
.
ListNode
(
pos
,
args
=
exprs
)
return
ExprNodes
.
ListNode
(
pos
,
args
=
exprs
)
def
p_
list
_iter
(
s
,
body
):
def
p_
comp
_iter
(
s
,
body
):
if
s
.
sy
==
'for'
:
if
s
.
sy
==
'for'
:
return
p_
list
_for
(
s
,
body
)
return
p_
comp
_for
(
s
,
body
)
elif
s
.
sy
==
'if'
:
elif
s
.
sy
==
'if'
:
return
p_
list
_if
(
s
,
body
)
return
p_
comp
_if
(
s
,
body
)
else
:
else
:
# insert the 'append' operation into the loop
# insert the 'append' operation into the loop
return
body
return
body
def
p_
list
_for
(
s
,
body
):
def
p_
comp
_for
(
s
,
body
):
# s.sy == 'for'
# s.sy == 'for'
pos
=
s
.
position
()
pos
=
s
.
position
()
s
.
next
()
s
.
next
()
kw
=
p_for_bounds
(
s
)
kw
=
p_for_bounds
(
s
)
kw
[
'else_clause'
]
=
None
kw
[
'else_clause'
]
=
None
kw
[
'body'
]
=
p_
list
_iter
(
s
,
body
)
kw
[
'body'
]
=
p_
comp
_iter
(
s
,
body
)
return
Nodes
.
ForStatNode
(
pos
,
**
kw
)
return
Nodes
.
ForStatNode
(
pos
,
**
kw
)
def
p_
list
_if
(
s
,
body
):
def
p_
comp
_if
(
s
,
body
):
# s.sy == 'if'
# s.sy == 'if'
pos
=
s
.
position
()
pos
=
s
.
position
()
s
.
next
()
s
.
next
()
test
=
p_test
(
s
)
test
=
p_test
_nocond
(
s
)
return
Nodes
.
IfStatNode
(
pos
,
return
Nodes
.
IfStatNode
(
pos
,
if_clauses
=
[
Nodes
.
IfClauseNode
(
pos
,
condition
=
test
,
if_clauses
=
[
Nodes
.
IfClauseNode
(
pos
,
condition
=
test
,
body
=
p_
list
_iter
(
s
,
body
))],
body
=
p_
comp
_iter
(
s
,
body
))],
else_clause
=
None
)
else_clause
=
None
)
#dictmaker: test ':' test (',' test ':' test)* [',']
#dictmaker: test ':' test (',' test ':' test)* [',']
...
@@ -776,7 +815,7 @@ def p_dict_or_set_maker(s):
...
@@ -776,7 +815,7 @@ def p_dict_or_set_maker(s):
target
=
ExprNodes
.
SetNode
(
pos
,
args
=
[])
target
=
ExprNodes
.
SetNode
(
pos
,
args
=
[])
append
=
ExprNodes
.
ComprehensionAppendNode
(
append
=
ExprNodes
.
ComprehensionAppendNode
(
item
.
pos
,
expr
=
item
,
target
=
ExprNodes
.
CloneNode
(
target
))
item
.
pos
,
expr
=
item
,
target
=
ExprNodes
.
CloneNode
(
target
))
loop
=
p_
list
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
loop
=
p_
comp
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
s
.
expect
(
'}'
)
s
.
expect
(
'}'
)
return
ExprNodes
.
ComprehensionNode
(
return
ExprNodes
.
ComprehensionNode
(
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
...
@@ -791,7 +830,7 @@ def p_dict_or_set_maker(s):
...
@@ -791,7 +830,7 @@ def p_dict_or_set_maker(s):
append
=
ExprNodes
.
DictComprehensionAppendNode
(
append
=
ExprNodes
.
DictComprehensionAppendNode
(
item
.
pos
,
key_expr
=
key
,
value_expr
=
value
,
item
.
pos
,
key_expr
=
key
,
value_expr
=
value
,
target
=
ExprNodes
.
CloneNode
(
target
))
target
=
ExprNodes
.
CloneNode
(
target
))
loop
=
p_
list
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
loop
=
p_
comp
_for
(
s
,
Nodes
.
ExprStatNode
(
append
.
pos
,
expr
=
append
))
s
.
expect
(
'}'
)
s
.
expect
(
'}'
)
return
ExprNodes
.
ComprehensionNode
(
return
ExprNodes
.
ComprehensionNode
(
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
pos
,
loop
=
loop
,
append
=
append
,
target
=
target
)
...
@@ -2382,8 +2421,17 @@ def p_def_statement(s, decorators=None):
...
@@ -2382,8 +2421,17 @@ def p_def_statement(s, decorators=None):
pos
=
s
.
position
()
pos
=
s
.
position
()
s
.
next
()
s
.
next
()
name
=
EncodedString
(
p_ident
(
s
)
)
name
=
EncodedString
(
p_ident
(
s
)
)
#args = []
s
.
expect
(
'('
);
s
.
expect
(
'('
);
args
,
star_arg
,
starstar_arg
=
p_varargslist
(
s
,
terminator
=
')'
)
s
.
expect
(
')'
)
if
p_nogil
(
s
):
error
(
s
.
pos
,
"Python function cannot be declared nogil"
)
doc
,
body
=
p_suite
(
s
,
Ctx
(
level
=
'function'
),
with_doc
=
1
)
return
Nodes
.
DefNode
(
pos
,
name
=
name
,
args
=
args
,
star_arg
=
star_arg
,
starstar_arg
=
starstar_arg
,
doc
=
doc
,
body
=
body
,
decorators
=
decorators
)
def
p_varargslist
(
s
,
terminator
=
')'
):
args
=
p_c_arg_list
(
s
,
in_pyfunc
=
1
,
nonempty_declarators
=
1
)
args
=
p_c_arg_list
(
s
,
in_pyfunc
=
1
,
nonempty_declarators
=
1
)
star_arg
=
None
star_arg
=
None
starstar_arg
=
None
starstar_arg
=
None
...
@@ -2395,18 +2443,12 @@ def p_def_statement(s, decorators=None):
...
@@ -2395,18 +2443,12 @@ def p_def_statement(s, decorators=None):
s
.
next
()
s
.
next
()
args
.
extend
(
p_c_arg_list
(
s
,
in_pyfunc
=
1
,
args
.
extend
(
p_c_arg_list
(
s
,
in_pyfunc
=
1
,
nonempty_declarators
=
1
,
kw_only
=
1
))
nonempty_declarators
=
1
,
kw_only
=
1
))
elif
s
.
sy
!=
')'
:
elif
s
.
sy
!=
terminator
:
s
.
error
(
"Syntax error in Python function argument list"
)
s
.
error
(
"Syntax error in Python function argument list"
)
if
s
.
sy
==
'**'
:
if
s
.
sy
==
'**'
:
s
.
next
()
s
.
next
()
starstar_arg
=
p_py_arg_decl
(
s
)
starstar_arg
=
p_py_arg_decl
(
s
)
s
.
expect
(
')'
)
return
(
args
,
star_arg
,
starstar_arg
)
if
p_nogil
(
s
):
error
(
s
.
pos
,
"Python function cannot be declared nogil"
)
doc
,
body
=
p_suite
(
s
,
Ctx
(
level
=
'function'
),
with_doc
=
1
)
return
Nodes
.
DefNode
(
pos
,
name
=
name
,
args
=
args
,
star_arg
=
star_arg
,
starstar_arg
=
starstar_arg
,
doc
=
doc
,
body
=
body
,
decorators
=
decorators
)
def
p_py_arg_decl
(
s
):
def
p_py_arg_decl
(
s
):
pos
=
s
.
position
()
pos
=
s
.
position
()
...
...
Cython/Compiler/Symtab.py
View file @
d61b5318
...
@@ -247,6 +247,7 @@ class Scope(object):
...
@@ -247,6 +247,7 @@ class Scope(object):
self.obj_to_entry = {}
self.obj_to_entry = {}
self.pystring_entries = []
self.pystring_entries = []
self.buffer_entries = []
self.buffer_entries = []
self.lambda_defs = []
self.control_flow = ControlFlow.LinearControlFlow()
self.control_flow = ControlFlow.LinearControlFlow()
def start_branching(self, pos):
def start_branching(self, pos):
...
@@ -431,6 +432,19 @@ class Scope(object):
...
@@ -431,6 +432,19 @@ class Scope(object):
self.pyfunc_entries.append(entry)
self.pyfunc_entries.append(entry)
return entry
return entry
def declare_lambda_function(self, func_cname, pos):
# Add an entry for an anonymous Python function.
entry = self.declare_var(None, py_object_type, pos,
cname=func_cname, visibility='private')
entry.name = EncodedString(func_cname)
entry.func_cname = func_cname
entry.signature = pyfunction_signature
self.pyfunc_entries.append(entry)
return entry
def add_lambda_def(self, def_node):
self.lambda_defs.append(def_node)
def register_pyfunction(self, entry):
def register_pyfunction(self, entry):
self.pyfunc_entries.append(entry)
self.pyfunc_entries.append(entry)
...
...
tests/run/lambda_T195.pyx
0 → 100644
View file @
d61b5318
__doc__
=
u"""
#>>> py_identity = lambda x:x
#>>> py_identity(1) == cy_identity(1)
#True
>>> idcall = make_identity()
>>> idcall(1)
1
>>> idcall(2)
2
>>> make_const0(1)()
1
>>> make_const1(1)(2)
1
>>> make_const1(1)(2)
1
>>> make_const_calc0()()
11
>>> make_const_calc1()(2)
11
>>> make_const_calc1_xy(8)(2)
27
"""
#cy_identity = lambda x:x
def
make_identity
():
return
lambda
x
:
x
def
make_const0
(
x
):
return
lambda
:
x
def
make_const1
(
x
):
return
lambda
_
:
x
def
make_const_calc0
():
return
lambda
:
1
*
2
*
3
+
5
def
make_const_calc1
():
return
lambda
_
:
1
*
2
*
3
+
5
def
make_const_calc1_xy
(
x
):
return
lambda
y
:
x
*
y
+
(
1
*
2
*
3
+
5
)
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