Issue 17277: incorrect line numbers in backtrace after removing a trace function (original) (raw)
Created on 2013-02-22 16:42 by xdegaye, last changed 2022-04-11 14:57 by admin.
Messages (8)
Author: Xavier de Gaye (xdegaye) *
Date: 2013-02-22 16:42
It seems that using f_trace in the f_lineno getter PyFrame_GetLineNumber(), as the condition to decide when tracing is active, is incorrect. See the following two examples.
In the backtrace printed by tracer.py running with python 3.3, the last entry should be line 12 instead of line 10:
$ python3 /tmp/tracer.py Traceback (most recent call last): File "/tmp/tracer.py", line 15, in foo() File "/tmp/tracer.py", line 10, in foo bar() ZeroDivisionError: division by zero
This simple case does not occur with pdb, because pdb takes care of deleting the f_trace attribute of all the frames in the call stack when removing the trace function (see set_continue() in bdb.py). But this is not good enough when a generator is involved as can be seen when running generator.py. In the backtrace the last entry should be line 6 instead of line 8:
$ python3 /tmp/generator.py
/tmp/generator.py(16)() -> foo() (Pdb) step --Call-- /tmp/generator.py(10)foo() -> def foo(): (Pdb) step /tmp/generator.py(11)foo() -> it = gen() (Pdb) step /tmp/generator.py(12)foo() -> next(it) (Pdb) step --Call-- /tmp/generator.py(3)gen() -> def gen(): (Pdb) return --Return-- /tmp/generator.py(8)gen()->0 -> yield i (Pdb) step /tmp/generator.py(13)foo() -> next(it) (Pdb) continue Traceback (most recent call last): File "/tmp/generator.py", line 16, in foo() File "/tmp/generator.py", line 13, in foo next(it) File "/tmp/generator.py", line 8, in gen yield i ZeroDivisionError: division by zero
It seems that it could be possible to fix this issue by replacing the test for f->f_trace in PyFrame_GetLineNumber, by a test for f->f_tstate->use_tracing, and updating accordingly the f_lineno and f_trace setters.
Author: Jesús Cea Avión (jcea) *
Date: 2013-02-23 03:36
Xavier, could you possibly provide a patch and a test?
Author: Xavier de Gaye (xdegaye) *
Date: 2013-02-23 14:34
The patch (on the default branch) reverts one of the changes made in r72488 to introduce the new PyFrame_GetLineNumber() function (issue 5954): tb_lineno is now back again the result of the call to PyCode_Addr2Line() instead of the call to PyFrame_GetLineNumber().
The other changes made by r72488 in _warnings.c and ceval.c should also probably be reverted as well.
The patch updates bdb set_continue() for consistency.
The patch adds a test to test_sys_settrace.
Author: Xavier de Gaye (xdegaye) *
Date: 2013-02-24 11:34
The proposed patch fixes the backtrace line numbers issue, but it does not fix PyFrame_GetLineNumber() which is the recommended way to get the frame line number.
As mentionned in the original message, testing for f->f_trace to implement the f_lineno getter is not correct. The f_lineno setter is also wrong in allowing to modify f_lineno when the frame is not the one that is being traced (pdb prevents that to happen though, in do_jump()).
I am working on another patch that should fix the issue by changing PyFrame_GetLineNumber() and the f_lineno accessors.
Author: Xavier de Gaye (xdegaye) *
Date: 2013-02-24 16:57
fix the issue by changing PyFrame_GetLineNumber() and the f_lineno accessors
The new patch named traced_frame.patch has been uploaded.
Also, now it is not allowed anymore to set the f_lineno attribute of a frame that is not the frame being traced, as f_lasti is invalidated anyway on returning to the evaluation of that frame.
Author: Xavier de Gaye (xdegaye) *
Date: 2013-02-25 16:05
The traced_frame.patch fixes also issue 7238 and issue 16482.
Author: Xavier de Gaye (xdegaye) *
Date: 2014-07-09 14:45
The previous patch changed a field in the PyThreadState structure. This new patch is simpler and does not prevent to change f_lineno when it is not the attribute of the frame being traced. The new patch fixes also issue 7238, issue 16482 and issue 17697.
Author: Xavier de Gaye (xdegaye) *
Date: 2015-07-04 20:31
The patch is wrong, the frame may not be run by the current PyThreadState.
History
Date
User
Action
Args
2022-04-11 14:57:42
admin
set
github: 61479
2017-12-20 08:56:12
ishimoto
set
nosy: + ishimoto
2015-07-04 20:31:42
xdegaye
set
messages: +
2014-07-09 14:45:55
xdegaye
set
files: + lineno_getter.patch
messages: +
2014-06-03 03:57:46
nikratio
set
nosy: + nikratio
2014-04-24 05:53:41
pconnell
set
nosy: + pconnell
2013-02-25 16:05:59
xdegaye
set
messages: +
2013-02-24 16:57:43
xdegaye
set
files: + traced_frame.patch
messages: +
2013-02-24 11:34:51
xdegaye
set
messages: +
2013-02-23 14:34:38
xdegaye
set
files: + backtrace_lno.patch
keywords: + patch
messages: +
2013-02-23 03:36:02
jcea
set
nosy: + jcea
messages: +
2013-02-22 20:31:19
terry.reedy
set
nosy: + belopolsky
2013-02-22 16:43:21
xdegaye
set
files: + generator.py
2013-02-22 16:42:49
xdegaye
create