Commit cae65475 authored by Stefan Behnel's avatar Stefan Behnel

fix and normalise None index handling in SliceIndexNode

parent aefc177d
...@@ -3358,8 +3358,15 @@ class SliceIndexNode(ExprNode): ...@@ -3358,8 +3358,15 @@ class SliceIndexNode(ExprNode):
return py_object_type return py_object_type
def calculate_constant_result(self): def calculate_constant_result(self):
self.constant_result = self.base.constant_result[ if self.start is None:
self.start.constant_result : self.stop.constant_result] start = None
else:
start = self.start.constant_result
if self.stop is None:
stop = None
else:
stop = self.stop.constant_result
self.constant_result = self.base.constant_result[start:stop]
def compile_time_value(self, denv): def compile_time_value(self, denv):
base = self.base.compile_time_value(denv) base = self.base.compile_time_value(denv)
......
...@@ -51,6 +51,11 @@ def is_common_value(a, b): ...@@ -51,6 +51,11 @@ def is_common_value(a, b):
return not a.is_py_attr and is_common_value(a.obj, b.obj) and a.attribute == b.attribute return not a.is_py_attr and is_common_value(a.obj, b.obj) and a.attribute == b.attribute
return False return False
def filter_none_node(node):
if node is not None and node.constant_result is None:
return None
return node
class IterationTransform(Visitor.EnvTransform): class IterationTransform(Visitor.EnvTransform):
"""Transform some common for-in loop patterns into efficient C loops: """Transform some common for-in loop patterns into efficient C loops:
...@@ -349,8 +354,8 @@ class IterationTransform(Visitor.EnvTransform): ...@@ -349,8 +354,8 @@ class IterationTransform(Visitor.EnvTransform):
neg_step = False neg_step = False
if isinstance(slice_node, ExprNodes.SliceIndexNode): if isinstance(slice_node, ExprNodes.SliceIndexNode):
slice_base = slice_node.base slice_base = slice_node.base
start = slice_node.start start = filter_none_node(slice_node.start)
stop = slice_node.stop stop = filter_none_node(slice_node.stop)
step = None step = None
if not stop: if not stop:
if not slice_base.type.is_pyobject: if not slice_base.type.is_pyobject:
...@@ -361,13 +366,11 @@ class IterationTransform(Visitor.EnvTransform): ...@@ -361,13 +366,11 @@ class IterationTransform(Visitor.EnvTransform):
assert isinstance(slice_node.index, ExprNodes.SliceNode) assert isinstance(slice_node.index, ExprNodes.SliceNode)
slice_base = slice_node.base slice_base = slice_node.base
index = slice_node.index index = slice_node.index
start = index.start start = filter_none_node(index.start)
stop = index.stop stop = filter_none_node(index.stop)
step = index.step step = filter_none_node(index.step)
if step: if step:
if step.constant_result is None: if not isinstance(step.constant_result, (int,long)) \
step = None
elif not isinstance(step.constant_result, (int,long)) \
or step.constant_result == 0 \ or step.constant_result == 0 \
or step.constant_result > 0 and not stop \ or step.constant_result > 0 and not stop \
or step.constant_result < 0 and not start: or step.constant_result < 0 and not start:
...@@ -401,15 +404,9 @@ class IterationTransform(Visitor.EnvTransform): ...@@ -401,15 +404,9 @@ class IterationTransform(Visitor.EnvTransform):
return node return node
if start: if start:
if start.constant_result is None: start = start.coerce_to(PyrexTypes.c_py_ssize_t_type, self.current_env())
start = None
else:
start = start.coerce_to(PyrexTypes.c_py_ssize_t_type, self.current_env())
if stop: if stop:
if stop.constant_result is None: stop = stop.coerce_to(PyrexTypes.c_py_ssize_t_type, self.current_env())
stop = None
else:
stop = stop.coerce_to(PyrexTypes.c_py_ssize_t_type, self.current_env())
if stop is None: if stop is None:
if neg_step: if neg_step:
stop = ExprNodes.IntNode( stop = ExprNodes.IntNode(
...@@ -3170,6 +3167,15 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations): ...@@ -3170,6 +3167,15 @@ class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
else: else:
return Nodes.StatListNode(node.pos, stats=[]) return Nodes.StatListNode(node.pos, stats=[])
def visit_SliceIndexNode(self, node):
self._calculate_const(node)
# normalise start/stop values
if node.start and node.start.constant_result is None:
node.start = None
if node.stop and node.stop.constant_result is None:
node.stop = None
return node
def visit_ForInStatNode(self, node): def visit_ForInStatNode(self, node):
self.visitchildren(node) self.visitchildren(node)
# iterating over a list literal? => tuples are more efficient # iterating over a list literal? => tuples are more efficient
......
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