Commit 53324e72 authored by Xavier Thompson's avatar Xavier Thompson

Detect const-incorrect calls of non-const method from const cypclass object

parent 1a1d49d7
...@@ -779,6 +779,7 @@ class CFuncDeclaratorNode(CDeclaratorNode): ...@@ -779,6 +779,7 @@ class CFuncDeclaratorNode(CDeclaratorNode):
calling_convention=self.base.calling_convention, calling_convention=self.base.calling_convention,
nogil=self.nogil, with_gil=self.with_gil, is_overridable=self.overridable, nogil=self.nogil, with_gil=self.with_gil, is_overridable=self.overridable,
is_const_method=self.is_const_method, is_const_method=self.is_const_method,
is_cyp_class_method = env.is_cyp_class_scope,
templates=self.templates) templates=self.templates)
if self.optional_arg_count: if self.optional_arg_count:
......
...@@ -2941,19 +2941,21 @@ class CFuncType(CType): ...@@ -2941,19 +2941,21 @@ class CFuncType(CType):
# (used for optimisation overrides) # (used for optimisation overrides)
# is_const_method boolean # is_const_method boolean
# is_static_method boolean # is_static_method boolean
# is_cyp_class_method boolean
is_cfunction = 1 is_cfunction = 1
original_sig = None original_sig = None
cached_specialized_types = None cached_specialized_types = None
from_fused = False from_fused = False
is_const_method = False is_const_method = False
is_cyp_class_method = False
subtypes = ['return_type', 'args'] subtypes = ['return_type', 'args']
def __init__(self, return_type, args, has_varargs = 0, def __init__(self, return_type, args, has_varargs = 0,
exception_value = None, exception_check = 0, calling_convention = "", exception_value = None, exception_check = 0, calling_convention = "",
nogil = 0, with_gil = 0, is_overridable = 0, optional_arg_count = 0, nogil = 0, with_gil = 0, is_overridable = 0, optional_arg_count = 0,
is_const_method = False, is_static_method=False, is_const_method = False, is_static_method=False, is_cyp_class_method=False,
templates = None, is_strict_signature = False): templates = None, is_strict_signature = False):
self.return_type = return_type self.return_type = return_type
self.args = args self.args = args
...@@ -2967,6 +2969,7 @@ class CFuncType(CType): ...@@ -2967,6 +2969,7 @@ class CFuncType(CType):
self.is_overridable = is_overridable self.is_overridable = is_overridable
self.is_const_method = is_const_method self.is_const_method = is_const_method
self.is_static_method = is_static_method self.is_static_method = is_static_method
self.is_cyp_class_method = is_cyp_class_method
self.templates = templates self.templates = templates
self.is_strict_signature = is_strict_signature self.is_strict_signature = is_strict_signature
...@@ -2997,6 +3000,7 @@ class CFuncType(CType): ...@@ -2997,6 +3000,7 @@ class CFuncType(CType):
with_gil, with_gil,
self.is_overridable, self.optional_arg_count, self.is_overridable, self.optional_arg_count,
self.is_const_method, self.is_static_method, self.is_const_method, self.is_static_method,
self.is_cyp_class_method,
self.templates, self.is_strict_signature) self.templates, self.is_strict_signature)
def calling_convention_prefix(self): def calling_convention_prefix(self):
...@@ -3348,6 +3352,7 @@ class CFuncType(CType): ...@@ -3348,6 +3352,7 @@ class CFuncType(CType):
optional_arg_count = self.optional_arg_count, optional_arg_count = self.optional_arg_count,
is_const_method = self.is_const_method, is_const_method = self.is_const_method,
is_static_method = self.is_static_method, is_static_method = self.is_static_method,
is_cyp_class_method = self.is_cyp_class_method,
templates = self.templates) templates = self.templates)
result.from_fused = self.is_fused result.from_fused = self.is_fused
...@@ -5157,6 +5162,13 @@ def best_match(arg_types, functions, pos=None, env=None, args=None, throw=False) ...@@ -5157,6 +5162,13 @@ def best_match(arg_types, functions, pos=None, env=None, args=None, throw=False)
% (expectation, actual_nargs) % (expectation, actual_nargs)
errors.append((func, error_mesg)) errors.append((func, error_mesg))
continue continue
# Skip non_const methods called on const object
if func_type.is_const and not func_type.is_const_method:
# Impose const-correctness only on cypclass methods for now
if func_type.is_cyp_class_method:
error_mesg = "Cannot call non-const method on const object"
errors.append((func, error_mesg))
continue
if func_type.templates: if func_type.templates:
deductions = reduce( deductions = reduce(
merge_template_deductions, merge_template_deductions,
......
...@@ -3253,10 +3253,21 @@ class CConstOrVolatileScope(Scope): ...@@ -3253,10 +3253,21 @@ class CConstOrVolatileScope(Scope):
return self.cached_const_or_volatile_entries[name] return self.cached_const_or_volatile_entries[name]
except KeyError: except KeyError:
entry = self.base_type_scope.lookup_here(name) entry = self.base_type_scope.lookup_here(name)
if entry is not None: if entry is not None and not entry.is_type:
entry = copy.copy(entry) entry = copy.copy(entry)
entry.type = PyrexTypes.c_const_or_volatile_type( entry.type = PyrexTypes.c_const_or_volatile_type(
entry.type, self.is_const, self.is_volatile) entry.type, self.is_const, self.is_volatile)
# Copy and adapt all the overloaded alternatives
if entry.is_cfunction:
# the overloaded entry at index 0 is always the one in the scope dict
overloaded_alternatives = [entry]
for alt in entry.overloaded_alternatives[1:]:
qual_alt = copy.copy(alt)
qual_alt.type = PyrexTypes.c_const_or_volatile_type(
alt.type, self.is_const, self.is_volatile)
qual_alt.overloaded_alternatives = overloaded_alternatives
overloaded_alternatives.append(qual_alt)
entry.overloaded_alternatives = overloaded_alternatives
self.cached_const_or_volatile_entries[name] = entry self.cached_const_or_volatile_entries[name] = entry
return entry return entry
......
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