Commit d1f76b27 authored by Stefan Behnel's avatar Stefan Behnel

implement min(a,b,...) and max(a,b,...) in unrolled conditional code

parent a2e62213
...@@ -1721,6 +1721,63 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform): ...@@ -1721,6 +1721,63 @@ class OptimizeBuiltinCalls(Visitor.EnvTransform):
is_temp = False) is_temp = False)
return ExprNodes.CastNode(node, PyrexTypes.py_object_type) return ExprNodes.CastNode(node, PyrexTypes.py_object_type)
def _handle_simple_function_min(self, node, pos_args):
return self._optimise_min_max(node, pos_args, '<')
def _handle_simple_function_max(self, node, pos_args):
return self._optimise_min_max(node, pos_args, '>')
def _optimise_min_max(self, node, args, operator):
"""Replace min(a,b,...) and max(a,b,...) by explicit conditional code.
"""
if len(args) <= 1:
# leave this to Python
return node
unpacked_args = []
for arg in args:
if isinstance(arg, ExprNodes.CoerceToPyTypeNode):
arg = arg.arg
unpacked_args.append(arg)
spanning_type = reduce(PyrexTypes.spanning_type,
[ arg.type for arg in unpacked_args ])
is_pycompare = spanning_type.is_pyobject
result_ref = UtilNodes.ResultRefNode(pos=node.pos, type=spanning_type)
stats = [
Nodes.SingleAssignmentNode(
node.pos,
lhs = UtilNodes.ResultRefNode(pos=node.pos, expression=result_ref),
rhs = unpacked_args[0].coerce_to(spanning_type, self.current_env()),
first = True)
]
for arg in unpacked_args[1:]:
stats.append(Nodes.IfStatNode(
arg.pos,
else_clause = None,
if_clauses = [ Nodes.IfClauseNode(
arg.pos,
condition = ExprNodes.PrimaryCmpNode(
arg.pos,
operand1 = arg.coerce_to(spanning_type, self.current_env()),
operator = operator,
operand2 = result_ref,
is_pycmp = is_pycompare,
is_temp = is_pycompare,
type = is_pycompare and PyrexTypes.py_object_type or PyrexTypes.c_bint_type
).coerce_to_boolean(self.current_env()),
body = Nodes.SingleAssignmentNode(
arg.pos,
lhs = result_ref,
rhs = arg)
)]
))
return UtilNodes.TempResultFromStatNode(
result_ref, Nodes.StatListNode(node.pos, stats = stats)
).coerce_to(node.type, self.current_env())
### special methods ### special methods
Pyx_tp_new_func_type = PyrexTypes.CFuncType( Pyx_tp_new_func_type = PyrexTypes.CFuncType(
......
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