summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGuido van Rossum <guido@python.org>2023-07-06 23:46:06 (GMT)
committerGitHub <noreply@github.com>2023-07-06 23:46:06 (GMT)
commite1d45b8ed43e1590862319fec33539f8adbc0849 (patch)
tree2a6a80fda614860c65bb8e4e3f8bc91f4e5b92d5
parentc60df361ce2d734148d503f4a711e67c110fe223 (diff)
downloadcpython-e1d45b8ed43e1590862319fec33539f8adbc0849.zip
cpython-e1d45b8ed43e1590862319fec33539f8adbc0849.tar.gz
cpython-e1d45b8ed43e1590862319fec33539f8adbc0849.tar.bz2
gh-104584: Handle EXTENDED_ARG in superblock creation (#106489)
With test.
-rw-r--r--Lib/test/test_capi/test_misc.py56
-rw-r--r--Python/optimizer.c16
2 files changed, 72 insertions, 0 deletions
diff --git a/Lib/test/test_capi/test_misc.py b/Lib/test/test_capi/test_misc.py
index 8d597e7..181e6b8 100644
--- a/Lib/test/test_capi/test_misc.py
+++ b/Lib/test/test_capi/test_misc.py
@@ -7,6 +7,7 @@ import contextlib
import importlib.machinery
import importlib.util
import json
+import opcode
import os
import pickle
import queue
@@ -2352,6 +2353,7 @@ def temporary_optimizer(opt):
finally:
_testinternalcapi.set_optimizer(None)
+
@contextlib.contextmanager
def clear_executors(func):
# Clear executors in func before and after running a block
@@ -2418,6 +2420,19 @@ class TestOptimizerAPI(unittest.TestCase):
self.assertEqual(opt.get_count(), 10)
+
+def get_first_executor(code):
+ co_code = code.co_code
+ JUMP_BACKWARD = opcode.opmap["JUMP_BACKWARD"]
+ for i in range(0, len(co_code), 2):
+ if co_code[i] == JUMP_BACKWARD or 1:
+ try:
+ return _testinternalcapi.get_executor(code, i)
+ except ValueError:
+ pass
+ return None
+
+
class TestUops(unittest.TestCase):
def test_basic_loop(self):
@@ -2443,6 +2458,47 @@ class TestUops(unittest.TestCase):
self.assertIn("SAVE_IP", uops)
self.assertIn("LOAD_FAST", uops)
+ def test_extended_arg(self):
+ "Check EXTENDED_ARG handling in superblock creation"
+ def many_vars():
+ # 260 vars, so z9 should have index 259
+ a0 = a1 = a2 = a3 = a4 = a5 = a6 = a7 = a8 = a9 = 42
+ b0 = b1 = b2 = b3 = b4 = b5 = b6 = b7 = b8 = b9 = 42
+ c0 = c1 = c2 = c3 = c4 = c5 = c6 = c7 = c8 = c9 = 42
+ d0 = d1 = d2 = d3 = d4 = d5 = d6 = d7 = d8 = d9 = 42
+ e0 = e1 = e2 = e3 = e4 = e5 = e6 = e7 = e8 = e9 = 42
+ f0 = f1 = f2 = f3 = f4 = f5 = f6 = f7 = f8 = f9 = 42
+ g0 = g1 = g2 = g3 = g4 = g5 = g6 = g7 = g8 = g9 = 42
+ h0 = h1 = h2 = h3 = h4 = h5 = h6 = h7 = h8 = h9 = 42
+ i0 = i1 = i2 = i3 = i4 = i5 = i6 = i7 = i8 = i9 = 42
+ j0 = j1 = j2 = j3 = j4 = j5 = j6 = j7 = j8 = j9 = 42
+ k0 = k1 = k2 = k3 = k4 = k5 = k6 = k7 = k8 = k9 = 42
+ l0 = l1 = l2 = l3 = l4 = l5 = l6 = l7 = l8 = l9 = 42
+ m0 = m1 = m2 = m3 = m4 = m5 = m6 = m7 = m8 = m9 = 42
+ n0 = n1 = n2 = n3 = n4 = n5 = n6 = n7 = n8 = n9 = 42
+ o0 = o1 = o2 = o3 = o4 = o5 = o6 = o7 = o8 = o9 = 42
+ p0 = p1 = p2 = p3 = p4 = p5 = p6 = p7 = p8 = p9 = 42
+ q0 = q1 = q2 = q3 = q4 = q5 = q6 = q7 = q8 = q9 = 42
+ r0 = r1 = r2 = r3 = r4 = r5 = r6 = r7 = r8 = r9 = 42
+ s0 = s1 = s2 = s3 = s4 = s5 = s6 = s7 = s8 = s9 = 42
+ t0 = t1 = t2 = t3 = t4 = t5 = t6 = t7 = t8 = t9 = 42
+ u0 = u1 = u2 = u3 = u4 = u5 = u6 = u7 = u8 = u9 = 42
+ v0 = v1 = v2 = v3 = v4 = v5 = v6 = v7 = v8 = v9 = 42
+ w0 = w1 = w2 = w3 = w4 = w5 = w6 = w7 = w8 = w9 = 42
+ x0 = x1 = x2 = x3 = x4 = x5 = x6 = x7 = x8 = x9 = 42
+ y0 = y1 = y2 = y3 = y4 = y5 = y6 = y7 = y8 = y9 = 42
+ z0 = z1 = z2 = z3 = z4 = z5 = z6 = z7 = z8 = z9 = 42
+ while z9 > 0:
+ z9 = z9 - 1
+
+ opt = _testinternalcapi.get_uop_optimizer()
+ with temporary_optimizer(opt):
+ ex = get_first_executor(many_vars.__code__)
+ self.assertIsNone(ex)
+ many_vars()
+ ex = get_first_executor(many_vars.__code__)
+ self.assertIn(("LOAD_FAST", 259), list(ex))
+
if __name__ == "__main__":
unittest.main()
diff --git a/Python/optimizer.c b/Python/optimizer.c
index d2fdca5..db117bb 100644
--- a/Python/optimizer.c
+++ b/Python/optimizer.c
@@ -412,6 +412,13 @@ translate_bytecode_to_trace(
ADD_TO_TRACE(SAVE_IP, (int)(instr - (_Py_CODEUNIT *)code->co_code_adaptive));
int opcode = instr->op.code;
uint64_t operand = instr->op.arg;
+ int extras = 0;
+ while (opcode == EXTENDED_ARG) {
+ instr++;
+ extras += 1;
+ opcode = instr->op.code;
+ operand = (operand << 8) | instr->op.arg;
+ }
switch (opcode) {
case LOAD_FAST_LOAD_FAST:
case STORE_FAST_LOAD_FAST:
@@ -458,6 +465,15 @@ translate_bytecode_to_trace(
int offset = expansion->uops[i].offset;
switch (expansion->uops[i].size) {
case 0:
+ if (extras && OPCODE_HAS_JUMP(opcode)) {
+ if (opcode == JUMP_BACKWARD_NO_INTERRUPT) {
+ operand -= extras;
+ }
+ else {
+ assert(opcode != JUMP_BACKWARD);
+ operand += extras;
+ }
+ }
break;
case 1:
operand = read_u16(&instr[offset].cache);