instr.offset > 0)
if new_source_line:
print(file=file)
- is_current_instr = instr.offset == lasti
+ if show_caches:
+ is_current_instr = instr.offset == lasti
+ else:
+ # Each CACHE takes 2 bytes
+ is_current_instr = instr.offset <= lasti \
+ <= instr.offset + 2 * _inline_cache_entries[_deoptop(instr.opcode)]
print(instr._disassemble(lineno_width, is_current_instr, offset_width),
file=file)
if exception_entries:
self.assertEqual(caches.count(""), empty_caches)
self.assertEqual(len(caches), total_caches)
+ @cpython_only
+ def test_show_currinstr_with_cache(self):
+ """
+ Make sure that with lasti pointing to CACHE, it still shows the current
+ line correctly
+ """
+ def f():
+ print(a)
+ # The code above should generate a LOAD_GLOBAL which has CACHE instr after
+ # However, this might change in the future. So we explicitly try to find
+ # a CACHE entry in the instructions. If we can't do that, fail the test
+
+ for inst in dis.get_instructions(f, show_caches=True):
+ if inst.opname == "CACHE":
+ op_offset = inst.offset - 2
+ cache_offset = inst.offset
+ break
+ else:
+ self.fail("Can't find a CACHE entry in the function provided to do the test")
+
+ assem_op = self.get_disassembly(f.__code__, lasti=op_offset, wrapper=False)
+ assem_cache = self.get_disassembly(f.__code__, lasti=cache_offset, wrapper=False)
+
+ # Make sure --> exists and points to the correct offset
+ self.assertRegex(assem_op, fr"-->\s+{op_offset}")
+ # Make sure when lasti points to cache, it shows the same disassembly
+ self.assertEqual(assem_op, assem_cache)
+
+
class DisWithFileTests(DisTests):
# Run the tests again, using the file arg instead of print