Commit fdd6d694 authored by Stefan Behnel's avatar Stefan Behnel

general optimisation support for calls to builtin types and their methods

currently providing optimisations for
- getattr(o,a)
- getattr(o,a,d)
- X.append(o)
- L.append(o)
- list.append(L,x)
parent 6a7cd28b
...@@ -21,7 +21,7 @@ builtin_function_table = [ ...@@ -21,7 +21,7 @@ builtin_function_table = [
#('eval', "", "", ""), #('eval', "", "", ""),
#('execfile', "", "", ""), #('execfile', "", "", ""),
#('filter', "", "", ""), #('filter', "", "", ""),
('getattr', "OO", "O", "PyObject_GetAttr"), #('getattr', "OO", "O", "PyObject_GetAttr"), # optimised later on
('getattr3', "OOO", "O", "__Pyx_GetAttr3", "getattr"), ('getattr3', "OOO", "O", "__Pyx_GetAttr3", "getattr"),
('hasattr', "OO", "b", "PyObject_HasAttr"), ('hasattr', "OO", "b", "PyObject_HasAttr"),
('hash', "O", "l", "PyObject_Hash"), ('hash', "O", "l", "PyObject_Hash"),
......
...@@ -2315,15 +2315,6 @@ class SimpleCallNode(CallNode): ...@@ -2315,15 +2315,6 @@ class SimpleCallNode(CallNode):
function = self.function function = self.function
function.is_called = 1 function.is_called = 1
self.function.analyse_types(env) self.function.analyse_types(env)
if function.is_attribute and function.is_py_attr and \
function.attribute == "append" and len(self.args) == 1:
# L.append(x) is almost always applied to a list
self.py_func = self.function
self.function = NameNode(pos=self.function.pos, name="__Pyx_PyObject_Append")
self.function.analyse_types(env)
self.self = self.py_func.obj
function.obj = CloneNode(self.self)
env.use_utility_code(append_utility_code)
if function.is_attribute and function.entry and function.entry.is_cmethod: if function.is_attribute and function.entry and function.entry.is_cmethod:
# Take ownership of the object from which the attribute # Take ownership of the object from which the attribute
# was obtained, because we need to pass it as 'self'. # was obtained, because we need to pass it as 'self'.
...@@ -2518,6 +2509,36 @@ class SimpleCallNode(CallNode): ...@@ -2518,6 +2509,36 @@ class SimpleCallNode(CallNode):
code.put_gotref(self.py_result()) code.put_gotref(self.py_result())
class PythonCapiFunctionNode(ExprNode):
subexprs = []
def __init__(self, pos, name, func_type, utility_code = None):
self.pos = pos
self.name = name
self.type = func_type
self.utility_code = utility_code
def generate_result_code(self, code):
if self.utility_code:
code.globalstate.use_utility_code(self.utility_code)
def calculate_result_code(self):
return self.name
class PythonCapiCallNode(SimpleCallNode):
# Python C-API Function call (only created in transforms)
def __init__(self, pos, function_name, func_type,
utility_code = None, **kwargs):
self.type = func_type.return_type
self.result_ctype = self.type
self.function = PythonCapiFunctionNode(
pos, function_name, func_type,
utility_code = utility_code)
# call this last so that we can override the constructed
# attributes above with explicit keyword arguments if required
SimpleCallNode.__init__(self, pos, **kwargs)
class GeneralCallNode(CallNode): class GeneralCallNode(CallNode):
# General Python function call, including keyword, # General Python function call, including keyword,
# * and ** arguments. # * and ** arguments.
...@@ -5419,29 +5440,6 @@ impl = "" ...@@ -5419,29 +5440,6 @@ impl = ""
#------------------------------------------------------------------------------------ #------------------------------------------------------------------------------------
append_utility_code = UtilityCode(
proto = """
static INLINE PyObject* __Pyx_PyObject_Append(PyObject* L, PyObject* x) {
if (likely(PyList_CheckExact(L))) {
if (PyList_Append(L, x) < 0) return NULL;
Py_INCREF(Py_None);
return Py_None; /* this is just to have an accurate signature */
}
else {
PyObject *r, *m;
m = __Pyx_GetAttrString(L, "append");
if (!m) return NULL;
r = PyObject_CallFunctionObjArgs(m, x, NULL);
Py_DECREF(m);
return r;
}
}
""",
impl = ""
)
#------------------------------------------------------------------------------------
# If the is_unsigned flag is set, we need to do some extra work to make # If the is_unsigned flag is set, we need to do some extra work to make
# sure the index doesn't become negative. # sure the index doesn't become negative.
......
...@@ -84,7 +84,7 @@ class Context(object): ...@@ -84,7 +84,7 @@ class Context(object):
from ParseTreeTransforms import GilCheck from ParseTreeTransforms import GilCheck
from AutoDocTransforms import EmbedSignature from AutoDocTransforms import EmbedSignature
from Optimize import FlattenInListTransform, SwitchTransform, IterationTransform from Optimize import FlattenInListTransform, SwitchTransform, IterationTransform
from Optimize import FlattenBuiltinTypeCreation, ConstantFolding, FinalOptimizePhase from Optimize import OptimiseBuiltinCalls, ConstantFolding, FinalOptimizePhase
from Buffer import IntroduceBufferAuxiliaryVars from Buffer import IntroduceBufferAuxiliaryVars
from ModuleNode import check_c_declarations from ModuleNode import check_c_declarations
...@@ -125,7 +125,7 @@ class Context(object): ...@@ -125,7 +125,7 @@ class Context(object):
IntroduceBufferAuxiliaryVars(self), IntroduceBufferAuxiliaryVars(self),
_check_c_declarations, _check_c_declarations,
AnalyseExpressionsTransform(self), AnalyseExpressionsTransform(self),
FlattenBuiltinTypeCreation(), OptimiseBuiltinCalls(),
# ComprehensionTransform(), # ComprehensionTransform(),
IterationTransform(), IterationTransform(),
SwitchTransform(), SwitchTransform(),
......
...@@ -7,8 +7,10 @@ import UtilNodes ...@@ -7,8 +7,10 @@ import UtilNodes
import TypeSlots import TypeSlots
import Symtab import Symtab
import Options import Options
from StringEncoding import EncodedString
from Cython.Utils import UtilityCode
from StringEncoding import EncodedString
from Errors import error
from ParseTreeTransforms import SkipDeclarations from ParseTreeTransforms import SkipDeclarations
#def unwrap_node(node): #def unwrap_node(node):
...@@ -414,18 +416,12 @@ class FlattenInListTransform(Visitor.VisitorTransform, SkipDeclarations): ...@@ -414,18 +416,12 @@ class FlattenInListTransform(Visitor.VisitorTransform, SkipDeclarations):
visit_Node = Visitor.VisitorTransform.recurse_to_children visit_Node = Visitor.VisitorTransform.recurse_to_children
class FlattenBuiltinTypeCreation(Visitor.VisitorTransform): class OptimiseBuiltinCalls(Visitor.VisitorTransform):
"""Optimise some common instantiation patterns for builtin types. """Optimise some common methods calls and instantiation patterns
for builtin types.
""" """
PyList_AsTuple_func_type = PyrexTypes.CFuncType( # only intercept on call nodes
PyrexTypes.py_object_type, [ visit_Node = Visitor.VisitorTransform.recurse_to_children
PyrexTypes.CFuncTypeArg("list", Builtin.list_type, None)
])
PyList_AsTuple_name = EncodedString("PyList_AsTuple")
PyList_AsTuple_entry = Symtab.Entry(
PyList_AsTuple_name, PyList_AsTuple_name, PyList_AsTuple_func_type)
def visit_GeneralCallNode(self, node): def visit_GeneralCallNode(self, node):
self.visitchildren(node) self.visitchildren(node)
...@@ -441,16 +437,38 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform): ...@@ -441,16 +437,38 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform):
node = handler(node, node.arg_tuple) node = handler(node, node.arg_tuple)
return node return node
def visit_PyTypeTestNode(self, node):
"""Flatten redundant type checks after tree changes.
"""
old_arg = node.arg
self.visitchildren(node)
if old_arg is node.arg or node.arg.type != node.type:
return node
return node.arg
def _find_handler(self, call_type, function): def _find_handler(self, call_type, function):
if not function.type.is_builtin_type: if not function.type.is_pyobject:
return None
if function.is_name:
if not function.type.is_builtin_type and '_' in function.name:
# not interesting anyway, so let's play safe here
return None return None
if not isinstance(function, ExprNodes.NameNode): match_name = function.name
elif isinstance(function, ExprNodes.AttributeNode):
if not function.obj.type.is_builtin_type:
type_name = "object" # safety measure
else:
type_name = function.obj.type.name
match_name = "%s_%s" % (type_name, function.attribute)
else:
return None return None
handler = getattr(self, '_handle_%s_%s' % (call_type, function.name), None) handler = getattr(self, '_handle_%s_%s' % (call_type, match_name), None)
if handler is None: if handler is None:
handler = getattr(self, '_handle_any_%s' % function.name, None) handler = getattr(self, '_handle_any_%s' % match_name, None)
return handler return handler
### builtin types
def _handle_general_dict(self, node, pos_args, kwargs): def _handle_general_dict(self, node, pos_args, kwargs):
"""Replace dict(a=b,c=d,...) by the underlying keyword dict """Replace dict(a=b,c=d,...) by the underlying keyword dict
construction which is done anyway. construction which is done anyway.
...@@ -491,6 +509,11 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform): ...@@ -491,6 +509,11 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform):
else: else:
return node return node
PyList_AsTuple_func_type = PyrexTypes.CFuncType(
Builtin.tuple_type, [
PyrexTypes.CFuncTypeArg("list", Builtin.list_type, None)
])
def _handle_simple_tuple(self, node, pos_args): def _handle_simple_tuple(self, node, pos_args):
"""Replace tuple([...]) by a call to PyList_AsTuple. """Replace tuple([...]) by a call to PyList_AsTuple.
""" """
...@@ -506,27 +529,139 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform): ...@@ -506,27 +529,139 @@ class FlattenBuiltinTypeCreation(Visitor.VisitorTransform):
# everything else may be None => take the safe path # everything else may be None => take the safe path
return node return node
node.args = pos_args.args return ExprNodes.PythonCapiCallNode(
node.arg_tuple = None node.pos, "PyList_AsTuple", self.PyList_AsTuple_func_type,
node.type = Builtin.tuple_type args = pos_args.args,
node.result_ctype = Builtin.tuple_type is_temp = node.is_temp
node.function = ExprNodes.NameNode( )
pos = node.pos,
name = self.PyList_AsTuple_name, ### builtin functions
type = self.PyList_AsTuple_func_type,
entry = self.PyList_AsTuple_entry) PyObject_GetAttr2_func_type = PyrexTypes.CFuncType(
PyrexTypes.py_object_type, [
PyrexTypes.CFuncTypeArg("object", PyrexTypes.py_object_type, None),
PyrexTypes.CFuncTypeArg("attr_name", PyrexTypes.py_object_type, None),
])
PyObject_GetAttr3_func_type = PyrexTypes.CFuncType(
PyrexTypes.py_object_type, [
PyrexTypes.CFuncTypeArg("object", PyrexTypes.py_object_type, None),
PyrexTypes.CFuncTypeArg("attr_name", PyrexTypes.py_object_type, None),
PyrexTypes.CFuncTypeArg("default", PyrexTypes.py_object_type, None),
])
def _handle_simple_getattr(self, node, pos_args):
# not really a builtin *type*, but worth optimising anyway
if not isinstance(pos_args, ExprNodes.TupleNode):
return node
args = pos_args.args
if len(args) == 2:
node = ExprNodes.PythonCapiCallNode(
node.pos, "PyObject_GetAttr", self.PyObject_GetAttr2_func_type,
args = args,
is_temp = node.is_temp
)
elif len(args) == 3:
node = ExprNodes.PythonCapiCallNode(
node.pos, "__Pyx_GetAttr3", self.PyObject_GetAttr3_func_type,
utility_code = Builtin.getattr3_utility_code,
args = args,
is_temp = node.is_temp
)
else:
error(node.pos, "getattr() called with wrong number of args, "
"expected 2 or 3, found %d" %
len(pos_args.args))
return node return node
def visit_PyTypeTestNode(self, node): ### methods of builtin types
"""Flatten redundant type checks after tree changes.
""" PyObject_Append_func_type = PyrexTypes.CFuncType(
old_arg = node.arg PyrexTypes.py_object_type, [
self.visitchildren(node) PyrexTypes.CFuncTypeArg("list", PyrexTypes.py_object_type, None),
if old_arg is node.arg or node.arg.type != node.type: PyrexTypes.CFuncTypeArg("item", PyrexTypes.py_object_type, None),
])
def _handle_simple_object_append(self, node, pos_args):
# X.append() is almost always referring to a list
if not isinstance(pos_args, ExprNodes.TupleNode):
return node
if len(pos_args.args) != 1:
return node return node
return node.arg
visit_Node = Visitor.VisitorTransform.recurse_to_children args = [node.function.obj] + pos_args.args
return ExprNodes.PythonCapiCallNode(
node.pos, "__Pyx_PyObject_Append", self.PyObject_Append_func_type,
args = args,
is_temp = node.is_temp,
utility_code = append_utility_code # FIXME: move to Builtin.py
)
PyList_Append_func_type = PyrexTypes.CFuncType(
PyrexTypes.c_int_type, [
PyrexTypes.CFuncTypeArg("list", PyrexTypes.py_object_type, None),
PyrexTypes.CFuncTypeArg("item", PyrexTypes.py_object_type, None),
],
exception_value = "-1")
def _handle_simple_list_append(self, node, pos_args):
if not isinstance(pos_args, ExprNodes.TupleNode):
return node
if len(pos_args.args) != 1:
error(node.pos, "list.append(x) called with wrong number of args, found %d" %
len(pos_args.args))
return node
obj = node.function.obj
# FIXME: obj may need a None check (ticket #166)
args = [obj] + pos_args.args
return ExprNodes.PythonCapiCallNode(
node.pos, "PyList_Append", self.PyList_Append_func_type,
args = args,
is_temp = node.is_temp
)
def _handle_simple_type_append(self, node, pos_args):
# unbound method call to list.append(L, x) ?
if node.function.obj.name != 'list':
return node
if not isinstance(pos_args, ExprNodes.TupleNode):
return node
args = pos_args.args
if len(args) != 2:
error(node.pos, "list.append(x) called with wrong number of args, found %d" %
len(pos_args.args))
return node
# FIXME: this may need a type check on the first operand
return ExprNodes.PythonCapiCallNode(
node.pos, "PyList_Append", self.PyList_Append_func_type,
args = args,
is_temp = node.is_temp
)
append_utility_code = UtilityCode(
proto = """
static INLINE PyObject* __Pyx_PyObject_Append(PyObject* L, PyObject* x) {
if (likely(PyList_CheckExact(L))) {
if (PyList_Append(L, x) < 0) return NULL;
Py_INCREF(Py_None);
return Py_None; /* this is just to have an accurate signature */
}
else {
PyObject *r, *m;
m = __Pyx_GetAttrString(L, "append");
if (!m) return NULL;
r = PyObject_CallFunctionObjArgs(m, x, NULL);
Py_DECREF(m);
return r;
}
}
""",
impl = ""
)
class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations): class ConstantFolding(Visitor.VisitorTransform, SkipDeclarations):
......
...@@ -235,6 +235,7 @@ class PyObjectType(PyrexType): ...@@ -235,6 +235,7 @@ class PyObjectType(PyrexType):
# #
# buffer_defaults dict or None Default options for bu # buffer_defaults dict or None Default options for bu
name = "object"
is_pyobject = 1 is_pyobject = 1
default_value = "0" default_value = "0"
pymemberdef_typecode = "T_OBJECT" pymemberdef_typecode = "T_OBJECT"
......
...@@ -12,9 +12,7 @@ __doc__ = u""" ...@@ -12,9 +12,7 @@ __doc__ = u"""
1 1
>>> g(t, 'b', 2) >>> g(t, 'b', 2)
2 2
"""
BROKEN = """
>>> h(t, 'a', 2) >>> h(t, 'a', 2)
1 1
>>> h(t, 'b', 2) >>> h(t, 'b', 2)
...@@ -27,5 +25,5 @@ def f(a, b): ...@@ -27,5 +25,5 @@ def f(a, b):
def g(a, b, c): def g(a, b, c):
return getattr3(a, b, c) return getattr3(a, b, c)
#def h(a, b, c): def h(a, b, c):
# return getattr(a, b, c) return getattr(a, b, c)
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