]> granicus.if.org Git - python/commitdiff
An object with __call__ as an attribute, when called, will have that attribute checke...
authorBrett Cannon <bcannon@gmail.com>
Fri, 9 Jun 2006 22:31:23 +0000 (22:31 +0000)
committerBrett Cannon <bcannon@gmail.com>
Fri, 9 Jun 2006 22:31:23 +0000 (22:31 +0000)
Closes bug #532646, again.  Will be backported.

Lib/test/crashers/infinite_rec_3.py [deleted file]
Lib/test/test_descr.py
Misc/NEWS
Objects/abstract.c

diff --git a/Lib/test/crashers/infinite_rec_3.py b/Lib/test/crashers/infinite_rec_3.py
deleted file mode 100644 (file)
index 0b04e4c..0000000
+++ /dev/null
@@ -1,9 +0,0 @@
-
-# http://python.org/sf/1202533
-
-class A(object):
-    pass
-A.__call__ = A()
-
-if __name__ == '__main__':
-    A()()   # segfault: infinite recursion in C
index ca91042ce670d11d2ca979e7825dd246ecd7b9ac..d9249b6cd2cf055a57f2abbc86d62872e9eec65c 100644 (file)
@@ -3171,6 +3171,21 @@ def kwdargs():
     list.__init__(a, sequence=[0, 1, 2])
     vereq(a, [0, 1, 2])
 
+def recursive__call__():
+    if verbose: print ("Testing recursive __call__() by setting to instance of "
+                        "class ...")
+    class A(object):
+        pass
+
+    A.__call__ = A()
+    try:
+        A()()
+    except RuntimeError:
+        pass
+    else:
+        raise TestFailed("Recursion limit should have been reached for "
+                         "__call__()")
+
 def delhook():
     if verbose: print "Testing __del__ hook..."
     log = []
@@ -4164,6 +4179,7 @@ def test_main():
     buffer_inherit()
     str_of_str_subclass()
     kwdargs()
+    recursive__call__()
     delhook()
     hashinherit()
     strops()
index 3bd732eafb2620c8d5f93b5a6a0121ed49ea96bb..3ec43af294825c5f5a2413298d2745b84f019edc 100644 (file)
--- a/Misc/NEWS
+++ b/Misc/NEWS
@@ -12,6 +12,12 @@ What's New in Python 2.5 beta 1?
 Core and builtins
 -----------------
 
+- Bug #532646: object.__call__() will continue looking for the __call__
+  attribute on objects until one without one is found.  This leads to recursion
+  when you take a class and set its __call__ attribute to an instance of the
+  class.  Originally fixed for classic classes, but this fix is for new-style.
+  Removes the infinite_rec_3 crasher.
+
 - The string and unicode methods startswith() and endswith() now accept
   a tuple of prefixes/suffixes to look for. Implements RFE #1491485.
 
index 83c48c21cb04db001575b9d9f482474d807dfce7..53898c59900c004bb7b8724cdc237ed3f2e6124c 100644 (file)
@@ -1790,7 +1790,15 @@ PyObject_Call(PyObject *func, PyObject *arg, PyObject *kw)
         ternaryfunc call;
 
        if ((call = func->ob_type->tp_call) != NULL) {
+               /* slot_tp_call() will be called and ends up calling
+                  PyObject_Call() if the object returned for __call__ has
+                  __call__ itself defined upon it.  This can be an infinite
+                  recursion if you set __call__ in a class to an instance of
+                  it. */
+               if (Py_EnterRecursiveCall(" in __call__"))
+                   return NULL;
                PyObject *result = (*call)(func, arg, kw);
+               Py_LeaveRecursiveCall();
                if (result == NULL && !PyErr_Occurred())
                        PyErr_SetString(
                                PyExc_SystemError,