Commit e8d61806 authored by gsamain's avatar gsamain

Support of python-style self argument for cpp classes

parent b2cfdd32
...@@ -5549,6 +5549,7 @@ class SimpleCallNode(CallNode): ...@@ -5549,6 +5549,7 @@ class SimpleCallNode(CallNode):
nogil = False nogil = False
analysed = False analysed = False
overflowcheck = False overflowcheck = False
explicit_cpp_self = None
def compile_time_value(self, denv): def compile_time_value(self, denv):
function = self.function.compile_time_value(denv) function = self.function.compile_time_value(denv)
...@@ -5651,6 +5652,14 @@ class SimpleCallNode(CallNode): ...@@ -5651,6 +5652,14 @@ class SimpleCallNode(CallNode):
args = self.args args = self.args
elif self.self: elif self.self:
args = [self.self] + self.args args = [self.self] + self.args
elif (func_type.is_cfunction and len(self.args) > 0
and not func_type.is_static_method
and self.function.is_name and self.function.entry
and hasattr(self.function.entry, 'scope')
and self.function.entry.scope.is_cpp_class_scope
and self.function.entry.scope.type.is_cyp_class):
self.explicit_cpp_self = self.args[0].result()
args = self.args[1:]
else: else:
args = self.args args = self.args
...@@ -5881,6 +5890,12 @@ class SimpleCallNode(CallNode): ...@@ -5881,6 +5890,12 @@ class SimpleCallNode(CallNode):
for actual_arg in self.args[len(formal_args):]: for actual_arg in self.args[len(formal_args):]:
arg_list_code.append(actual_arg.result()) arg_list_code.append(actual_arg.result())
result = None
# Checking for self argument in cppclasses given as an explicit first
# argument. This is the opposite of self.self purpose.
if self.explicit_cpp_self:
result = "%s->%s(%s)" % (self.explicit_cpp_self, self.function.result(), ', '.join(arg_list_code))
else:
result = "%s(%s)" % (self.function.result(), ', '.join(arg_list_code)) result = "%s(%s)" % (self.function.result(), ', '.join(arg_list_code))
return result return result
......
...@@ -648,6 +648,7 @@ class CFuncDeclaratorNode(CDeclaratorNode): ...@@ -648,6 +648,7 @@ class CFuncDeclaratorNode(CDeclaratorNode):
optional_arg_count = 0 optional_arg_count = 0
is_const_method = 0 is_const_method = 0
templates = None templates = None
skipped_self = None
def analyse_templates(self): def analyse_templates(self):
if isinstance(self.base, CArrayDeclaratorNode): if isinstance(self.base, CArrayDeclaratorNode):
...@@ -680,7 +681,7 @@ class CFuncDeclaratorNode(CDeclaratorNode): ...@@ -680,7 +681,7 @@ class CFuncDeclaratorNode(CDeclaratorNode):
for i, arg_node in enumerate(self.args): for i, arg_node in enumerate(self.args):
name_declarator, type = arg_node.analyse( name_declarator, type = arg_node.analyse(
env, nonempty=nonempty, env, nonempty=nonempty,
is_self_arg=(i == 0 and env.is_c_class_scope and 'staticmethod' not in env.directives)) is_self_arg=(i == 0 and (env.is_c_class_scope or env.is_cpp_class_scope and env.parent_type.is_cyp_class) and 'staticmethod' not in env.directives))
name = name_declarator.name name = name_declarator.name
if name in directive_locals: if name in directive_locals:
type_node = directive_locals[name] type_node = directive_locals[name]
...@@ -695,7 +696,11 @@ class CFuncDeclaratorNode(CDeclaratorNode): ...@@ -695,7 +696,11 @@ class CFuncDeclaratorNode(CDeclaratorNode):
type = other_type type = other_type
if name_declarator.cname: if name_declarator.cname:
error(self.pos, "Function argument cannot have C name specification") error(self.pos, "Function argument cannot have C name specification")
if i == 0 and env.is_c_class_scope and type.is_unspecified: if i == 0 and (env.is_c_class_scope or env.is_cpp_class_scope and env.parent_type.is_cyp_class) and type.is_unspecified:
if env.is_cpp_class_scope and env.parent_type.is_cyp_class:
type = env.lookup_here("this").type
self.skipped_self = (name, type, arg_node.pos, arg_node)
continue
# fix the type of self # fix the type of self
type = env.parent_type type = env.parent_type
# Turn *[] argument into ** # Turn *[] argument into **
...@@ -711,6 +716,8 @@ class CFuncDeclaratorNode(CDeclaratorNode): ...@@ -711,6 +716,8 @@ class CFuncDeclaratorNode(CDeclaratorNode):
elif self.optional_arg_count: elif self.optional_arg_count:
error(self.pos, "Non-default argument follows default argument") error(self.pos, "Non-default argument follows default argument")
if self.skipped_self:
self.args = self.args[1:] if len(self.args) > 1 else []
exc_val = None exc_val = None
exc_check = 0 exc_check = 0
if self.exception_check == '+': if self.exception_check == '+':
...@@ -1017,9 +1024,9 @@ class CSimpleBaseTypeNode(CBaseTypeNode): ...@@ -1017,9 +1024,9 @@ class CSimpleBaseTypeNode(CBaseTypeNode):
type = cy_object_type type = cy_object_type
self.arg_name = EncodedString(self.name) self.arg_name = EncodedString(self.name)
elif self.name is None: elif self.name is None:
if self.is_self_arg and env.is_c_class_scope: if self.is_self_arg and (env.is_c_class_scope or env.is_cpp_class_scope and env.parent_type.is_cyp_class):
#print "CSimpleBaseTypeNode.analyse: defaulting to parent type" ### #print "CSimpleBaseTypeNode.analyse: defaulting to parent type" ###
type = env.parent_type type = env.parent_type if env.is_c_class_scope else PyrexTypes.unspecified_type
## elif self.is_type_arg and env.is_c_class_scope: ## elif self.is_type_arg and env.is_c_class_scope:
## type = Builtin.type_type ## type = Builtin.type_type
else: else:
...@@ -1052,8 +1059,8 @@ class CSimpleBaseTypeNode(CBaseTypeNode): ...@@ -1052,8 +1059,8 @@ class CSimpleBaseTypeNode(CBaseTypeNode):
if type is not None: if type is not None:
pass pass
elif could_be_name: elif could_be_name:
if self.is_self_arg and env.is_c_class_scope: if self.is_self_arg and (env.is_c_class_scope or env.is_cpp_class_scope and env.parent_type.is_cyp_class):
type = env.parent_type type = env.parent_type if env.is_c_class_scope else PyrexTypes.unspecified_type
## elif self.is_type_arg and env.is_c_class_scope: ## elif self.is_type_arg and env.is_c_class_scope:
## type = Builtin.type_type ## type = Builtin.type_type
else: else:
...@@ -2501,8 +2508,23 @@ class CFuncDefNode(FuncDefNode): ...@@ -2501,8 +2508,23 @@ class CFuncDefNode(FuncDefNode):
# An error will be produced in the cdef function # An error will be produced in the cdef function
self.overridable = False self.overridable = False
func_declarator = self.declarator.base if isinstance(self.declarator, CPtrDeclaratorNode)\
else self.declarator
if env.is_cpp_class_scope and env.parent_type.is_cyp_class\
and not declarator.skipped_self and not self.is_static_method:
# It means we have a cypclass method without the self argument
# => shout
error(self.pos, "Cypclass methods must have a self argument")
self.declare_cpdef_wrapper(env) self.declare_cpdef_wrapper(env)
self.create_local_scope(env) self.create_local_scope(env)
# HACK: in case we had a skipped self in arguments (functions in cpp classes),
# declare it here as a valid local variable mapped to 'this'
if declarator.skipped_self:
_name, _type, _pos, _ = declarator.skipped_self
_cname = "this"
entry = self.local_scope.declare(_name, _cname, _type, _pos, 'private')
entry.is_variable = 1
def declare_cpdef_wrapper(self, env): def declare_cpdef_wrapper(self, env):
if self.overridable: if self.overridable:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment