Commit d29a8293 authored by Xavier Thompson's avatar Xavier Thompson

Enforce 'no overloads with implicit conversions' rule for cypclass methods

parent 5e7e5047
...@@ -2958,6 +2958,25 @@ class CFuncType(CType): ...@@ -2958,6 +2958,25 @@ class CFuncType(CType):
return 0 return 0
return 1 return 1
def convertible_arguments_with(self, other_type):
return self.convertible_arguments_with_resolved_type(other_type.resolve())
def convertible_arguments_with_resolved_type(self, other_type):
if other_type is error_type:
return 1
if not other_type.is_cfunction:
return 0
if len(self.args) - self.optional_arg_count != len(other_type.args) - other_type.optional_arg_count:
return 0
if self.has_varargs != other_type.has_varargs:
return 0
arg_type_pairs = ((arg.type, other_arg.type) for arg, other_arg in zip(self.args, other_type.args))
if (all((arg.assignable_from(other) for arg, other in arg_type_pairs))
or
all((other.assignable_from(arg) for arg, other in arg_type_pairs))):
return 1
return 0
def compatible_arguments_with(self, other_type, as_cmethod=0): def compatible_arguments_with(self, other_type, as_cmethod=0):
return self.compatible_arguments_with_resolved_type(other_type.resolve(), as_cmethod) return self.compatible_arguments_with_resolved_type(other_type.resolve(), as_cmethod)
......
...@@ -526,7 +526,7 @@ class Scope(object): ...@@ -526,7 +526,7 @@ class Scope(object):
entries = self.entries entries = self.entries
# The indices of an overridable overloaded alternatives that this declaration will hide. # The indices of all previous overloaded alternatives that this declaration will hide.
previous_alternative_indices = [] previous_alternative_indices = []
if name and name in entries and not shadow: if name and name in entries and not shadow:
...@@ -535,44 +535,41 @@ class Scope(object): ...@@ -535,44 +535,41 @@ class Scope(object):
cpp_override_allowed = False cpp_override_allowed = False
if type.is_cfunction and old_entry.type.is_cfunction and self.is_cpp(): if type.is_cfunction and old_entry.type.is_cfunction and self.is_cpp():
cpp_override_allowed = True cpp_override_allowed = True
for index, alt_entry in enumerate(old_entry.all_alternatives()):
# in a cypclass, a method can hide a method inherited from a different class
# based only on their argument types
if self.is_cyp_class_scope: if self.is_cyp_class_scope:
might_redeclare_or_override = type.compatible_arguments_with
else:
might_redeclare_or_override = type.compatible_signature_with
if might_redeclare_or_override(alt_entry.type): for index, alt_entry in enumerate(old_entry.all_alternatives()):
alt_type = alt_entry.type
if type.convertible_arguments_with(alt_type):
cpp_override_allowed = False cpp_override_allowed = False
if self.is_cyp_class_scope:
# allow default constructor or __alloc__ to be redeclared by user # allow default constructor or __alloc__ to be redeclared by user
if alt_entry.is_default: if alt_entry.is_default:
previous_alternative_indices.append(index) previous_alternative_indices.append(index)
cpp_override_allowed = True cpp_override_allowed = True
continue continue
# Any inherited method is visible # enforce cypclass overloading rules
# until overloaded by a method with the same signature alt_declarator_str = alt_type.declarator_code(name, for_display = 1).strip()
if alt_entry.is_inherited: new_declarator_str = type.declarator_code(name, for_display = 1).strip()
if new_declarator_str != alt_declarator_str:
if self.is_cyp_class_scope: error(pos, ("Cypclass methods have conflicting signatures:\n"
"Cypclass method\n"
# in a cypclass, if the arguments are compatible, then the new method must actually ">> %s\n"
# override the inherited method: the whole signature must be identical "has implicitly convertible arguments from or to:\n"
old_declarator = alt_entry.type.declarator_code(name, for_display = 1).strip() ">> %s\n"
new_declarator = type.declarator_code(name, for_display = 1).strip() "but their signatures are not exactly the same"
if not new_declarator == old_declarator: % (type.declaration_code(name, for_display = 1).strip(),
comparison_message = " ---> %s\nvs -> %s" % (new_declarator, old_declarator) alt_type.declaration_code(name, for_display = 1).strip()))
error(pos, "Cypclass method with compatible arguments but incompatible signature:\n%s" )
% comparison_message)
if alt_entry.pos is not None: if alt_entry.pos is not None:
error(alt_entry.pos, "Conflicting method is defined here") error(alt_entry.pos, "Conflicting method is defined here")
# also, the return type must be covariant elif alt_entry.is_inherited:
elif not type.return_type.subtype_of_resolved_type(alt_entry.type.return_type):
# the return type must be covariant
if not type.return_type.subtype_of_resolved_type(alt_type.return_type):
error(pos, "Cypclass method overrides another with incompatible return type") error(pos, "Cypclass method overrides another with incompatible return type")
if alt_entry.pos is not None: if alt_entry.pos is not None:
error(alt_entry.pos, "Conflicting method is defined here") error(alt_entry.pos, "Conflicting method is defined here")
...@@ -581,12 +578,23 @@ class Scope(object): ...@@ -581,12 +578,23 @@ class Scope(object):
cpp_override_allowed = True cpp_override_allowed = True
continue continue
elif self.is_cyp_class_scope: # stop if cpp_override_allowed is False for the current alternative
if (type.narrower_arguments_than(alt_entry.type) break
or alt_entry.type.narrower_arguments_than(type)):
error(pos, "Cypclass overloaded method with narrower arguments") # normal cpp case
if alt_entry.pos is not None: else:
error(alt_entry.pos, "Conflicting method is defined here") for index, alt_entry in enumerate(old_entry.all_alternatives()):
if type.compatible_signature_with(alt_entry.type):
cpp_override_allowed = False
if alt_entry.is_inherited:
previous_alternative_indices.append(index)
cpp_override_allowed = True
continue
# stop if cpp_override_allowed is False for the current alternative
break
if cpp_override_allowed: if cpp_override_allowed:
# C++ function/method overrides with different signatures are ok. # C++ function/method overrides with different signatures are ok.
......
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