homepage

This issue tracker has been migrated to GitHub , and is currently read-only.
For more information, see the GitHub FAQs in the Python's Developer Guide.

classification
Title: Recursive class instance "error"
Type: Stage:
Components: Interpreter Core Versions: Python 2.3
process
Status: closed Resolution: fixed
Dependencies: Superseder:
Assigned To: brett.cannon Nosy List: brett.cannon, gregmi, gvanrossum, nnorwitz, nobody
Priority: normal Keywords:

Created on 2002年03月20日 18:56 by gregmi, last changed 2022年04月10日 16:05 by admin. This issue is now closed.

Files
File name Uploaded Description Edit
c2.diff nnorwitz, 2002年06月07日 22:52 updated patch 1
Messages (18)
msg9820 - (view) Author: Gregor Mirai (gregmi) Date: 2002年03月20日 18:56
If one writes the following code (tested on Python 
2.1, 2.2 on platforms MacOS X Server, MacOS X, Windows 
98, NT, 2000) one can easily produce several "errors".
MacOS X, MacOS X Server (Python 2.1, 2.2)
------------------------------------------
class A:
 def __getattr__(self, name):
 print name,
 return A()
------------------------------------------
>>> a=A()
>>> a.foo
Segmentation fault
Win98, NT, 2000 (Python 2.1, 2.2)
------------------------------------------
class A:
 def __getattr__(self, name):
 print name
 return A()
------------------------------------------
>>> a=A()
>>> a.foo
foo
__repr__ __call__ __call__ __call__ ... ad inf
msg9821 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年03月20日 22:10
Logged In: YES 
user_id=6380
This is buggy user code.
But the segfault should be fixed if possible.
msg9822 - (view) Author: Nobody/Anonymous (nobody) Date: 2002年03月21日 09:47
Logged In: NO 
>>> a=A() 
>>> a.foo 
foo 
__repr__ __call__ __call__ __call__ ... ad inf
This is normal behavior. The code at the top is buggy. The 
correct one is:
class A:
 def __getattr__(self, name):
 if name == '__repr__':
 return self.__repr__()
 print name
 return A()
msg9823 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2002年06月07日 19:21
Logged In: YES 
user_id=33168
The attached patch stops the segfault and raises an exception:
TypeError: 'A' max recursion limit (100000) exceeded
I'm not sure if this is genarally applicable, or what a
reasonable # is.
100,000 was a guess and should probably be made a #def
w/comment.
msg9824 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年06月07日 20:08
Logged In: YES 
user_id=6380
Is this fix enough? Aren't there lots of other ways to
generate the same error? E.g. new-style classes (where this
particular example doesn't apply, but others might).
Or is this specific to instance_call?
msg9825 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2002年06月07日 20:23
Logged In: YES 
user_id=33168
I don't know. I tried to come up with other test cases and
failed.
Tried __str__ and __getitem__. Also tried new-style classes.
The problem was that the code bounced back and forth
between PyObject_Call() & instance_call() (mutual recursion).
msg9826 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年06月07日 20:27
Logged In: YES 
user_id=6380
Fair enough. Ideally this should share the recursion_limit
variable from ceval.c and the recursion_depth field of the
stack frame.
msg9827 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2002年06月07日 20:35
Logged In: YES 
user_id=33168
That sounds more reasonable.
I'll take a look and see if they can be integrated.
msg9828 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2002年06月07日 22:52
Logged In: YES 
user_id=33168
The recursion fields are shared in the new patch.
msg9829 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年06月10日 16:08
Logged In: YES 
user_id=6380
Can you explain how the fix works? Why does the first
comment say
/* this shouldn't be reached, but just in case */
and why is this test necesary
if (tstate->recursion_depth < Py_GetRecursionLimit()) {
???
And who *does* report the recursion error?
msg9830 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2002年06月12日 22:38
Logged In: YES 
user_id=33168
The problem is that there is mutual recursion
between instance_call() and PyObject_Call().
The recursion apparently goes through the eval_frame() loop.
This patch increases the recursion depth so the mutual recursion
will eventually end (otherwise, the stack grows without bounds).
ISTM the first check SHOULD be reached, but it is not.
I don't understand why. The recursion error must be set
in eval_frame().
The first block in the patch could be just this line:
 ++tstate->recursion_depth;
I didn't do it that way, since I didn't expect returning to
eval_frame(). I'm not sure if it's guaranteed to return
to eval_frame() which is why I left the first condition
in with the comment. I suppose the comment should say
something like: /* this condition doesn't seem to be triggered,
the recursion depth limit is exceeded in eval_frame */
The test for recursion_depth is necessary to ensure
that the recursion error isn't overwritten. If this check
is removed, the follow exception is raised:
 AttributeError: A instance has no __call__ method
Hopefully this makes sense.
msg9831 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年06月13日 20:41
Logged In: YES 
user_id=6380
Hm. I'm tracing this in the debugger now, and it appears
that the problem is when trying to *print* an A instance.
The statement a.foo causes the problem simply because it
returns an A instance. (Proof: "a=A().foo; print a" fails in
the print.)
I think that instance_call may not be the real cause of the
problem...
msg9832 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2002年06月13日 21:50
Logged In: YES 
user_id=6380
It was very specific to __call__ after all, and I found an
example that didn't involve __getattr__. See the comments I
checked in as part of the fix in classobject.c.
msg9833 - (view) Author: Brett Cannon (brett.cannon) * (Python committer) Date: 2006年04月18日 00:00
Logged In: YES 
user_id=357491
This was not fixed for new-style classes and still segfaults
the interpreter at least back to 2.4.
Reopening.
msg9834 - (view) Author: Neal Norwitz (nnorwitz) * (Python committer) Date: 2006年04月18日 04:37
Logged In: YES 
user_id=33168
Please add a test case to Lib/test/crashers.
msg9835 - (view) Author: Guido van Rossum (gvanrossum) * (Python committer) Date: 2006年04月18日 08:09
Logged In: YES 
user_id=6380
Ironically, I just found the same (?) crash in the p3yk
branch, in test_class.py (which is now testing new-style
classes for compatibility with the behavior of classic classes).
I can't fix this myself, so assigning to Brett.
msg9836 - (view) Author: Brett Cannon (brett.cannon) * (Python committer) Date: 2006年04月18日 19:08
Logged In: YES 
user_id=357491
The test is already in crashers: infinite_rec_3 . And it is
the same crash as in py3k; that's how I found it.
msg9837 - (view) Author: Brett Cannon (brett.cannon) * (Python committer) Date: 2006年06月09日 22:39
Logged In: YES 
user_id=357491
Rev. 46806 for 2.5 and rev. 46807 for 2.4 have the fix for
new-style classes.
History
Date User Action Args
2022年04月10日 16:05:07adminsetgithub: 36294
2002年03月20日 18:56:23gregmicreate

AltStyle によって変換されたページ (->オリジナル) /