summaryrefslogtreecommitdiffstats
path: root/Lib/test
diff options
context:
space:
mode:
authormpage <mpage@meta.com>2024-12-13 18:17:16 (GMT)
committerGitHub <noreply@github.com>2024-12-13 18:17:16 (GMT)
commit2de048ce79e621f5ae0574095b9600fe8595f607 (patch)
treede3116284fc2016192787297de7a67ba348e5825 /Lib/test
parent292067fbc9db81896c16ff12d51c21d2b0f233e2 (diff)
downloadcpython-2de048ce79e621f5ae0574095b9600fe8595f607.zip
cpython-2de048ce79e621f5ae0574095b9600fe8595f607.tar.gz
cpython-2de048ce79e621f5ae0574095b9600fe8595f607.tar.bz2
gh-115999: Specialize loading attributes from modules in free-threaded builds (#127711)
We use the same approach that was used for specialization of LOAD_GLOBAL in free-threaded builds: _CHECK_ATTR_MODULE is renamed to _CHECK_ATTR_MODULE_PUSH_KEYS; it pushes the keys object for the following _LOAD_ATTR_MODULE_FROM_KEYS (nee _LOAD_ATTR_MODULE). This arrangement avoids having to recheck the keys version. _LOAD_ATTR_MODULE is renamed to _LOAD_ATTR_MODULE_FROM_KEYS; it loads the value from the keys object pushed by the preceding _CHECK_ATTR_MODULE_PUSH_KEYS at the cached index.
Diffstat (limited to 'Lib/test')
-rw-r--r--Lib/test/test_capi/test_misc.py47
-rw-r--r--Lib/test/test_generated_cases.py74
-rw-r--r--Lib/test/test_opcache.py2
3 files changed, 113 insertions, 10 deletions
diff --git a/Lib/test/test_capi/test_misc.py b/Lib/test/test_capi/test_misc.py
index 61512e6..ada3018 100644
--- a/Lib/test/test_capi/test_misc.py
+++ b/Lib/test/test_capi/test_misc.py
@@ -48,6 +48,8 @@ except ModuleNotFoundError:
# Skip this test if the _testcapi module isn't available.
_testcapi = import_helper.import_module('_testcapi')
+from _testcapi import HeapCTypeSubclass, HeapCTypeSubclassWithFinalizer
+
import _testlimitedcapi
import _testinternalcapi
@@ -653,9 +655,9 @@ class CAPITest(unittest.TestCase):
self.assertEqual(type_refcnt - 1, sys.getrefcount(_testcapi.HeapCTypeSubclass))
def test_c_subclass_of_heap_ctype_with_del_modifying_dunder_class_only_decrefs_once(self):
- subclass_instance = _testcapi.HeapCTypeSubclassWithFinalizer()
- type_refcnt = sys.getrefcount(_testcapi.HeapCTypeSubclassWithFinalizer)
- new_type_refcnt = sys.getrefcount(_testcapi.HeapCTypeSubclass)
+ subclass_instance = HeapCTypeSubclassWithFinalizer()
+ type_refcnt = sys.getrefcount(HeapCTypeSubclassWithFinalizer)
+ new_type_refcnt = sys.getrefcount(HeapCTypeSubclass)
# Test that subclass instance was fully created
self.assertEqual(subclass_instance.value, 10)
@@ -665,19 +667,46 @@ class CAPITest(unittest.TestCase):
del subclass_instance
# Test that setting __class__ modified the reference counts of the types
+ #
+ # This is highly sensitive to implementation details and may break in the future.
+ #
+ # We expect the refcount on the old type, HeapCTypeSubclassWithFinalizer, to
+ # remain the same: the finalizer gets a strong reference (+1) when it gets the
+ # type from the module and setting __class__ decrements the refcount (-1).
+ #
+ # We expect the refcount on the new type, HeapCTypeSubclass, to increase by 2:
+ # the finalizer get a strong reference (+1) when it gets the type from the
+ # module and setting __class__ increments the refcount (+1).
+ expected_type_refcnt = type_refcnt
+ expected_new_type_refcnt = new_type_refcnt + 2
+
+ if not Py_GIL_DISABLED:
+ # In default builds the result returned from sys.getrefcount
+ # includes a temporary reference that is created by the interpreter
+ # when it pushes its argument on the operand stack. This temporary
+ # reference is not included in the result returned by Py_REFCNT, which
+ # is used in the finalizer.
+ #
+ # In free-threaded builds the result returned from sys.getrefcount
+ # does not include the temporary reference. Types use deferred
+ # refcounting and the interpreter will not create a new reference
+ # for deferred values on the operand stack.
+ expected_type_refcnt -= 1
+ expected_new_type_refcnt -= 1
+
if support.Py_DEBUG:
# gh-89373: In debug mode, _Py_Dealloc() keeps a strong reference
# to the type while calling tp_dealloc()
- self.assertEqual(type_refcnt, _testcapi.HeapCTypeSubclassWithFinalizer.refcnt_in_del)
- else:
- self.assertEqual(type_refcnt - 1, _testcapi.HeapCTypeSubclassWithFinalizer.refcnt_in_del)
- self.assertEqual(new_type_refcnt + 1, _testcapi.HeapCTypeSubclass.refcnt_in_del)
+ expected_type_refcnt += 1
+
+ self.assertEqual(expected_type_refcnt, HeapCTypeSubclassWithFinalizer.refcnt_in_del)
+ self.assertEqual(expected_new_type_refcnt, HeapCTypeSubclass.refcnt_in_del)
# Test that the original type already has decreased its refcnt
- self.assertEqual(type_refcnt - 1, sys.getrefcount(_testcapi.HeapCTypeSubclassWithFinalizer))
+ self.assertEqual(type_refcnt - 1, sys.getrefcount(HeapCTypeSubclassWithFinalizer))
# Test that subtype_dealloc decref the newly assigned __class__ only once
- self.assertEqual(new_type_refcnt, sys.getrefcount(_testcapi.HeapCTypeSubclass))
+ self.assertEqual(new_type_refcnt, sys.getrefcount(HeapCTypeSubclass))
def test_heaptype_with_setattro(self):
obj = _testcapi.HeapCTypeSetattr()
diff --git a/Lib/test/test_generated_cases.py b/Lib/test/test_generated_cases.py
index 66862ec..9c65e81 100644
--- a/Lib/test/test_generated_cases.py
+++ b/Lib/test/test_generated_cases.py
@@ -1639,6 +1639,80 @@ class TestGeneratedCases(unittest.TestCase):
"""
self.run_cases_test(input, output)
+ def test_pop_dead_inputs_all_live(self):
+ input = """
+ inst(OP, (a, b --)) {
+ POP_DEAD_INPUTS();
+ HAM(a, b);
+ INPUTS_DEAD();
+ }
+ """
+ output = """
+ TARGET(OP) {
+ frame->instr_ptr = next_instr;
+ next_instr += 1;
+ INSTRUCTION_STATS(OP);
+ _PyStackRef a;
+ _PyStackRef b;
+ b = stack_pointer[-1];
+ a = stack_pointer[-2];
+ HAM(a, b);
+ stack_pointer += -2;
+ assert(WITHIN_STACK_BOUNDS());
+ DISPATCH();
+ }
+ """
+ self.run_cases_test(input, output)
+
+ def test_pop_dead_inputs_some_live(self):
+ input = """
+ inst(OP, (a, b, c --)) {
+ POP_DEAD_INPUTS();
+ HAM(a);
+ INPUTS_DEAD();
+ }
+ """
+ output = """
+ TARGET(OP) {
+ frame->instr_ptr = next_instr;
+ next_instr += 1;
+ INSTRUCTION_STATS(OP);
+ _PyStackRef a;
+ a = stack_pointer[-3];
+ stack_pointer += -2;
+ assert(WITHIN_STACK_BOUNDS());
+ HAM(a);
+ stack_pointer += -1;
+ assert(WITHIN_STACK_BOUNDS());
+ DISPATCH();
+ }
+ """
+ self.run_cases_test(input, output)
+
+ def test_pop_dead_inputs_with_output(self):
+ input = """
+ inst(OP, (a, b -- c)) {
+ POP_DEAD_INPUTS();
+ c = SPAM();
+ }
+ """
+ output = """
+ TARGET(OP) {
+ frame->instr_ptr = next_instr;
+ next_instr += 1;
+ INSTRUCTION_STATS(OP);
+ _PyStackRef c;
+ stack_pointer += -2;
+ assert(WITHIN_STACK_BOUNDS());
+ c = SPAM();
+ stack_pointer[0] = c;
+ stack_pointer += 1;
+ assert(WITHIN_STACK_BOUNDS());
+ DISPATCH();
+ }
+ """
+ self.run_cases_test(input, output)
+
class TestGeneratedAbstractCases(unittest.TestCase):
def setUp(self) -> None:
diff --git a/Lib/test/test_opcache.py b/Lib/test/test_opcache.py
index 50b5f36..0a7557a 100644
--- a/Lib/test/test_opcache.py
+++ b/Lib/test/test_opcache.py
@@ -892,7 +892,7 @@ class TestRacesDoNotCrash(TestBase):
opname = "LOAD_ATTR_METHOD_WITH_VALUES"
self.assert_races_do_not_crash(opname, get_items, read, write)
- @requires_specialization
+ @requires_specialization_ft
def test_load_attr_module(self):
def get_items():
items = []