https://github.com/python/cpython/commit/08bc03ff2a5545a165ba57f4ca73b8ff705dc757
commit: 08bc03ff2a5545a165ba57f4ca73b8ff705dc757
branch: main
author: Sam Gross <[email protected]>
committer: colesbury <[email protected]>
date: 2025-12-19T19:10:37Z
summary:
gh-120321: Make gi_frame_state transitions atomic in FT build (gh-142599)
This makes generator frame state transitions atomic in the free
threading build, which avoids segfaults when trying to execute
a generator from multiple threads concurrently.
There are still a few operations that aren't thread-safe and may crash
if performed concurrently on the same generator/coroutine:
* Accessing gi_yieldfrom/cr_await/ag_await
* Accessing gi_frame/cr_frame/ag_frame
* Async generator operations
files:
M Include/cpython/pyatomic.h
M Include/cpython/pyatomic_gcc.h
M Include/cpython/pyatomic_msc.h
M Include/cpython/pyatomic_std.h
M Include/internal/pycore_pyatomic_ft_wrappers.h
M Include/internal/pycore_tstate.h
M Include/internal/pycore_uop_ids.h
M Include/internal/pycore_uop_metadata.h
M Lib/test/test_free_threading/test_generators.py
M Objects/genobject.c
M Python/bytecodes.c
M Python/ceval.c
M Python/ceval_macros.h
M Python/executor_cases.c.h
M Python/generated_cases.c.h
M Tools/cases_generator/analyzer.py
diff --git a/Include/cpython/pyatomic.h b/Include/cpython/pyatomic.h
index 790640309f1e03..ce907fd6a4c453 100644
--- a/Include/cpython/pyatomic.h
+++ b/Include/cpython/pyatomic.h
@@ -523,6 +523,9 @@ _Py_atomic_store_uintptr_release(uintptr_t *obj, uintptr_t
value);
static inline void
_Py_atomic_store_ssize_release(Py_ssize_t *obj, Py_ssize_t value);
+static inline void
+_Py_atomic_store_int8_release(int8_t *obj, int8_t value);
+
static inline void
_Py_atomic_store_int_release(int *obj, int value);
diff --git a/Include/cpython/pyatomic_gcc.h b/Include/cpython/pyatomic_gcc.h
index 1566b83b9f6a1b..c045213c898a03 100644
--- a/Include/cpython/pyatomic_gcc.h
+++ b/Include/cpython/pyatomic_gcc.h
@@ -572,6 +572,10 @@ static inline void
_Py_atomic_store_int_release(int *obj, int value)
{ __atomic_store_n(obj, value, __ATOMIC_RELEASE); }
+static inline void
+_Py_atomic_store_int8_release(int8_t *obj, int8_t value)
+{ __atomic_store_n(obj, value, __ATOMIC_RELEASE); }
+
static inline void
_Py_atomic_store_ssize_release(Py_ssize_t *obj, Py_ssize_t value)
{ __atomic_store_n(obj, value, __ATOMIC_RELEASE); }
diff --git a/Include/cpython/pyatomic_msc.h b/Include/cpython/pyatomic_msc.h
index d155955df0cddf..8b9dd3eb0f8e16 100644
--- a/Include/cpython/pyatomic_msc.h
+++ b/Include/cpython/pyatomic_msc.h
@@ -1066,6 +1066,19 @@ _Py_atomic_store_int_release(int *obj, int value)
#endif
}
+static inline void
+_Py_atomic_store_int8_release(int8_t *obj, int8_t value)
+{
+#if defined(_M_X64) || defined(_M_IX86)
+ *(int8_t volatile *)obj = value;
+#elif defined(_M_ARM64)
+ _Py_atomic_ASSERT_ARG_TYPE(unsigned __int8);
+ __stlr8((unsigned __int8 volatile *)obj, (unsigned __int8)value);
+#else
+# error "no implementation of _Py_atomic_store_int8_release"
+#endif
+}
+
static inline void
_Py_atomic_store_ssize_release(Py_ssize_t *obj, Py_ssize_t value)
{
diff --git a/Include/cpython/pyatomic_std.h b/Include/cpython/pyatomic_std.h
index 7176f667a4082c..cfc8dbefc63d09 100644
--- a/Include/cpython/pyatomic_std.h
+++ b/Include/cpython/pyatomic_std.h
@@ -1023,6 +1023,14 @@ _Py_atomic_store_int_release(int *obj, int value)
memory_order_release);
}
+static inline void
+_Py_atomic_store_int8_release(int8_t *obj, int8_t value)
+{
+ _Py_USING_STD;
+ atomic_store_explicit((_Atomic(int8_t)*)obj, value,
+ memory_order_release);
+}
+
static inline void
_Py_atomic_store_uint_release(unsigned int *obj, unsigned int value)
{
diff --git a/Include/internal/pycore_pyatomic_ft_wrappers.h
b/Include/internal/pycore_pyatomic_ft_wrappers.h
index 1a6d5075361f3c..70a32db663b293 100644
--- a/Include/internal/pycore_pyatomic_ft_wrappers.h
+++ b/Include/internal/pycore_pyatomic_ft_wrappers.h
@@ -41,6 +41,8 @@ extern "C" {
_Py_atomic_load_uint8(&value)
#define FT_ATOMIC_STORE_UINT8(value, new_value) \
_Py_atomic_store_uint8(&value, new_value)
+#define FT_ATOMIC_LOAD_INT8_RELAXED(value) \
+ _Py_atomic_load_int8_relaxed(&value)
#define FT_ATOMIC_LOAD_UINT8_RELAXED(value) \
_Py_atomic_load_uint8_relaxed(&value)
#define FT_ATOMIC_LOAD_UINT16_RELAXED(value) \
@@ -55,6 +57,10 @@ extern "C" {
_Py_atomic_store_ptr_release(&value, new_value)
#define FT_ATOMIC_STORE_UINTPTR_RELEASE(value, new_value) \
_Py_atomic_store_uintptr_release(&value, new_value)
+#define FT_ATOMIC_STORE_INT8_RELAXED(value, new_value) \
+ _Py_atomic_store_int8_relaxed(&value, new_value)
+#define FT_ATOMIC_STORE_INT8_RELEASE(value, new_value) \
+ _Py_atomic_store_int8_release(&value, new_value)
#define FT_ATOMIC_STORE_SSIZE_RELAXED(value, new_value) \
_Py_atomic_store_ssize_relaxed(&value, new_value)
#define FT_ATOMIC_STORE_SSIZE_RELEASE(value, new_value) \
@@ -134,6 +140,7 @@ extern "C" {
#define FT_ATOMIC_LOAD_PTR_RELAXED(value) value
#define FT_ATOMIC_LOAD_UINT8(value) value
#define FT_ATOMIC_STORE_UINT8(value, new_value) value = new_value
+#define FT_ATOMIC_LOAD_INT8_RELAXED(value) value
#define FT_ATOMIC_LOAD_UINT8_RELAXED(value) value
#define FT_ATOMIC_LOAD_UINT16_RELAXED(value) value
#define FT_ATOMIC_LOAD_UINT32_RELAXED(value) value
@@ -141,6 +148,8 @@ extern "C" {
#define FT_ATOMIC_STORE_PTR_RELAXED(value, new_value) value = new_value
#define FT_ATOMIC_STORE_PTR_RELEASE(value, new_value) value = new_value
#define FT_ATOMIC_STORE_UINTPTR_RELEASE(value, new_value) value = new_value
+#define FT_ATOMIC_STORE_INT8_RELAXED(value, new_value) value = new_value
+#define FT_ATOMIC_STORE_INT8_RELEASE(value, new_value) value = new_value
#define FT_ATOMIC_STORE_SSIZE_RELAXED(value, new_value) value = new_value
#define FT_ATOMIC_STORE_SSIZE_RELEASE(value, new_value) value = new_value
#define FT_ATOMIC_STORE_UINT8_RELAXED(value, new_value) value = new_value
diff --git a/Include/internal/pycore_tstate.h b/Include/internal/pycore_tstate.h
index c4f723ac8abbbe..a57f1f45c135a6 100644
--- a/Include/internal/pycore_tstate.h
+++ b/Include/internal/pycore_tstate.h
@@ -82,6 +82,13 @@ typedef struct _PyThreadStateImpl {
PyObject *asyncio_running_loop; // Strong reference
PyObject *asyncio_running_task; // Strong reference
+ // Distinguishes between yield and return from PyEval_EvalFrame().
+ // See gen_send_ex2() in Objects/genobject.c
+ enum {
+ GENERATOR_RETURN = 0,
+ GENERATOR_YIELD = 1,
+ } generator_return_kind;
+
/* Head of circular linked-list of all tasks which are instances of
`asyncio.Task`
or subclasses of it used in `asyncio.all_tasks`.
*/
diff --git a/Include/internal/pycore_uop_ids.h
b/Include/internal/pycore_uop_ids.h
index 6130e5cad21bf4..a11959fb4057df 100644
--- a/Include/internal/pycore_uop_ids.h
+++ b/Include/internal/pycore_uop_ids.h
@@ -535,572 +535,574 @@ extern "C" {
#define _FORMAT_SIMPLE_r11 728
#define _FORMAT_WITH_SPEC_r21 729
#define _FOR_ITER_r23 730
-#define _FOR_ITER_GEN_FRAME_r23 731
-#define _FOR_ITER_TIER_TWO_r23 732
-#define _GET_AITER_r11 733
-#define _GET_ANEXT_r12 734
-#define _GET_AWAITABLE_r11 735
-#define _GET_ITER_r12 736
-#define _GET_LEN_r12 737
-#define _GET_YIELD_FROM_ITER_r11 738
-#define _GUARD_BINARY_OP_EXTEND_r22 739
-#define _GUARD_CALLABLE_ISINSTANCE_r03 740
-#define _GUARD_CALLABLE_ISINSTANCE_r13 741
-#define _GUARD_CALLABLE_ISINSTANCE_r23 742
-#define _GUARD_CALLABLE_ISINSTANCE_r33 743
-#define _GUARD_CALLABLE_LEN_r03 744
-#define _GUARD_CALLABLE_LEN_r13 745
-#define _GUARD_CALLABLE_LEN_r23 746
-#define _GUARD_CALLABLE_LEN_r33 747
-#define _GUARD_CALLABLE_LIST_APPEND_r03 748
-#define _GUARD_CALLABLE_LIST_APPEND_r13 749
-#define _GUARD_CALLABLE_LIST_APPEND_r23 750
-#define _GUARD_CALLABLE_LIST_APPEND_r33 751
-#define _GUARD_CALLABLE_STR_1_r03 752
-#define _GUARD_CALLABLE_STR_1_r13 753
-#define _GUARD_CALLABLE_STR_1_r23 754
-#define _GUARD_CALLABLE_STR_1_r33 755
-#define _GUARD_CALLABLE_TUPLE_1_r03 756
-#define _GUARD_CALLABLE_TUPLE_1_r13 757
-#define _GUARD_CALLABLE_TUPLE_1_r23 758
-#define _GUARD_CALLABLE_TUPLE_1_r33 759
-#define _GUARD_CALLABLE_TYPE_1_r03 760
-#define _GUARD_CALLABLE_TYPE_1_r13 761
-#define _GUARD_CALLABLE_TYPE_1_r23 762
-#define _GUARD_CALLABLE_TYPE_1_r33 763
-#define _GUARD_DORV_NO_DICT_r01 764
-#define _GUARD_DORV_NO_DICT_r11 765
-#define _GUARD_DORV_NO_DICT_r22 766
-#define _GUARD_DORV_NO_DICT_r33 767
-#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r01 768
-#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r11 769
-#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r22 770
-#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r33 771
-#define _GUARD_GLOBALS_VERSION_r00 772
-#define _GUARD_GLOBALS_VERSION_r11 773
-#define _GUARD_GLOBALS_VERSION_r22 774
-#define _GUARD_GLOBALS_VERSION_r33 775
-#define _GUARD_IP_RETURN_GENERATOR_r00 776
-#define _GUARD_IP_RETURN_GENERATOR_r11 777
-#define _GUARD_IP_RETURN_GENERATOR_r22 778
-#define _GUARD_IP_RETURN_GENERATOR_r33 779
-#define _GUARD_IP_RETURN_VALUE_r00 780
-#define _GUARD_IP_RETURN_VALUE_r11 781
-#define _GUARD_IP_RETURN_VALUE_r22 782
-#define _GUARD_IP_RETURN_VALUE_r33 783
-#define _GUARD_IP_YIELD_VALUE_r00 784
-#define _GUARD_IP_YIELD_VALUE_r11 785
-#define _GUARD_IP_YIELD_VALUE_r22 786
-#define _GUARD_IP_YIELD_VALUE_r33 787
-#define _GUARD_IP__PUSH_FRAME_r00 788
-#define _GUARD_IP__PUSH_FRAME_r11 789
-#define _GUARD_IP__PUSH_FRAME_r22 790
-#define _GUARD_IP__PUSH_FRAME_r33 791
-#define _GUARD_IS_FALSE_POP_r00 792
-#define _GUARD_IS_FALSE_POP_r10 793
-#define _GUARD_IS_FALSE_POP_r21 794
-#define _GUARD_IS_FALSE_POP_r32 795
-#define _GUARD_IS_NONE_POP_r00 796
-#define _GUARD_IS_NONE_POP_r10 797
-#define _GUARD_IS_NONE_POP_r21 798
-#define _GUARD_IS_NONE_POP_r32 799
-#define _GUARD_IS_NOT_NONE_POP_r10 800
-#define _GUARD_IS_TRUE_POP_r00 801
-#define _GUARD_IS_TRUE_POP_r10 802
-#define _GUARD_IS_TRUE_POP_r21 803
-#define _GUARD_IS_TRUE_POP_r32 804
-#define _GUARD_KEYS_VERSION_r01 805
-#define _GUARD_KEYS_VERSION_r11 806
-#define _GUARD_KEYS_VERSION_r22 807
-#define _GUARD_KEYS_VERSION_r33 808
-#define _GUARD_NOS_DICT_r02 809
-#define _GUARD_NOS_DICT_r12 810
-#define _GUARD_NOS_DICT_r22 811
-#define _GUARD_NOS_DICT_r33 812
-#define _GUARD_NOS_FLOAT_r02 813
-#define _GUARD_NOS_FLOAT_r12 814
-#define _GUARD_NOS_FLOAT_r22 815
-#define _GUARD_NOS_FLOAT_r33 816
-#define _GUARD_NOS_INT_r02 817
-#define _GUARD_NOS_INT_r12 818
-#define _GUARD_NOS_INT_r22 819
-#define _GUARD_NOS_INT_r33 820
-#define _GUARD_NOS_LIST_r02 821
-#define _GUARD_NOS_LIST_r12 822
-#define _GUARD_NOS_LIST_r22 823
-#define _GUARD_NOS_LIST_r33 824
-#define _GUARD_NOS_NOT_NULL_r02 825
-#define _GUARD_NOS_NOT_NULL_r12 826
-#define _GUARD_NOS_NOT_NULL_r22 827
-#define _GUARD_NOS_NOT_NULL_r33 828
-#define _GUARD_NOS_NULL_r02 829
-#define _GUARD_NOS_NULL_r12 830
-#define _GUARD_NOS_NULL_r22 831
-#define _GUARD_NOS_NULL_r33 832
-#define _GUARD_NOS_OVERFLOWED_r02 833
-#define _GUARD_NOS_OVERFLOWED_r12 834
-#define _GUARD_NOS_OVERFLOWED_r22 835
-#define _GUARD_NOS_OVERFLOWED_r33 836
-#define _GUARD_NOS_TUPLE_r02 837
-#define _GUARD_NOS_TUPLE_r12 838
-#define _GUARD_NOS_TUPLE_r22 839
-#define _GUARD_NOS_TUPLE_r33 840
-#define _GUARD_NOS_UNICODE_r02 841
-#define _GUARD_NOS_UNICODE_r12 842
-#define _GUARD_NOS_UNICODE_r22 843
-#define _GUARD_NOS_UNICODE_r33 844
-#define _GUARD_NOT_EXHAUSTED_LIST_r02 845
-#define _GUARD_NOT_EXHAUSTED_LIST_r12 846
-#define _GUARD_NOT_EXHAUSTED_LIST_r22 847
-#define _GUARD_NOT_EXHAUSTED_LIST_r33 848
-#define _GUARD_NOT_EXHAUSTED_RANGE_r02 849
-#define _GUARD_NOT_EXHAUSTED_RANGE_r12 850
-#define _GUARD_NOT_EXHAUSTED_RANGE_r22 851
-#define _GUARD_NOT_EXHAUSTED_RANGE_r33 852
-#define _GUARD_NOT_EXHAUSTED_TUPLE_r02 853
-#define _GUARD_NOT_EXHAUSTED_TUPLE_r12 854
-#define _GUARD_NOT_EXHAUSTED_TUPLE_r22 855
-#define _GUARD_NOT_EXHAUSTED_TUPLE_r33 856
-#define _GUARD_THIRD_NULL_r03 857
-#define _GUARD_THIRD_NULL_r13 858
-#define _GUARD_THIRD_NULL_r23 859
-#define _GUARD_THIRD_NULL_r33 860
-#define _GUARD_TOS_ANY_SET_r01 861
-#define _GUARD_TOS_ANY_SET_r11 862
-#define _GUARD_TOS_ANY_SET_r22 863
-#define _GUARD_TOS_ANY_SET_r33 864
-#define _GUARD_TOS_DICT_r01 865
-#define _GUARD_TOS_DICT_r11 866
-#define _GUARD_TOS_DICT_r22 867
-#define _GUARD_TOS_DICT_r33 868
-#define _GUARD_TOS_FLOAT_r01 869
-#define _GUARD_TOS_FLOAT_r11 870
-#define _GUARD_TOS_FLOAT_r22 871
-#define _GUARD_TOS_FLOAT_r33 872
-#define _GUARD_TOS_INT_r01 873
-#define _GUARD_TOS_INT_r11 874
-#define _GUARD_TOS_INT_r22 875
-#define _GUARD_TOS_INT_r33 876
-#define _GUARD_TOS_LIST_r01 877
-#define _GUARD_TOS_LIST_r11 878
-#define _GUARD_TOS_LIST_r22 879
-#define _GUARD_TOS_LIST_r33 880
-#define _GUARD_TOS_OVERFLOWED_r01 881
-#define _GUARD_TOS_OVERFLOWED_r11 882
-#define _GUARD_TOS_OVERFLOWED_r22 883
-#define _GUARD_TOS_OVERFLOWED_r33 884
-#define _GUARD_TOS_SLICE_r01 885
-#define _GUARD_TOS_SLICE_r11 886
-#define _GUARD_TOS_SLICE_r22 887
-#define _GUARD_TOS_SLICE_r33 888
-#define _GUARD_TOS_TUPLE_r01 889
-#define _GUARD_TOS_TUPLE_r11 890
-#define _GUARD_TOS_TUPLE_r22 891
-#define _GUARD_TOS_TUPLE_r33 892
-#define _GUARD_TOS_UNICODE_r01 893
-#define _GUARD_TOS_UNICODE_r11 894
-#define _GUARD_TOS_UNICODE_r22 895
-#define _GUARD_TOS_UNICODE_r33 896
-#define _GUARD_TYPE_VERSION_r01 897
-#define _GUARD_TYPE_VERSION_r11 898
-#define _GUARD_TYPE_VERSION_r22 899
-#define _GUARD_TYPE_VERSION_r33 900
-#define _GUARD_TYPE_VERSION_AND_LOCK_r01 901
-#define _GUARD_TYPE_VERSION_AND_LOCK_r11 902
-#define _GUARD_TYPE_VERSION_AND_LOCK_r22 903
-#define _GUARD_TYPE_VERSION_AND_LOCK_r33 904
-#define _HANDLE_PENDING_AND_DEOPT_r00 905
-#define _HANDLE_PENDING_AND_DEOPT_r10 906
-#define _HANDLE_PENDING_AND_DEOPT_r20 907
-#define _HANDLE_PENDING_AND_DEOPT_r30 908
-#define _IMPORT_FROM_r12 909
-#define _IMPORT_NAME_r21 910
-#define _INIT_CALL_BOUND_METHOD_EXACT_ARGS_r00 911
-#define _INIT_CALL_PY_EXACT_ARGS_r01 912
-#define _INIT_CALL_PY_EXACT_ARGS_0_r01 913
-#define _INIT_CALL_PY_EXACT_ARGS_1_r01 914
-#define _INIT_CALL_PY_EXACT_ARGS_2_r01 915
-#define _INIT_CALL_PY_EXACT_ARGS_3_r01 916
-#define _INIT_CALL_PY_EXACT_ARGS_4_r01 917
-#define _INSERT_NULL_r10 918
-#define _INSTRUMENTED_FOR_ITER_r23 919
-#define _INSTRUMENTED_INSTRUCTION_r00 920
-#define _INSTRUMENTED_JUMP_FORWARD_r00 921
-#define _INSTRUMENTED_JUMP_FORWARD_r11 922
-#define _INSTRUMENTED_JUMP_FORWARD_r22 923
-#define _INSTRUMENTED_JUMP_FORWARD_r33 924
-#define _INSTRUMENTED_LINE_r00 925
-#define _INSTRUMENTED_NOT_TAKEN_r00 926
-#define _INSTRUMENTED_NOT_TAKEN_r11 927
-#define _INSTRUMENTED_NOT_TAKEN_r22 928
-#define _INSTRUMENTED_NOT_TAKEN_r33 929
-#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r00 930
-#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r10 931
-#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r21 932
-#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r32 933
-#define _INSTRUMENTED_POP_JUMP_IF_NONE_r10 934
-#define _INSTRUMENTED_POP_JUMP_IF_NOT_NONE_r10 935
-#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r00 936
-#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r10 937
-#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r21 938
-#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r32 939
-#define _IS_NONE_r11 940
-#define _IS_OP_r21 941
-#define _ITER_CHECK_LIST_r02 942
-#define _ITER_CHECK_LIST_r12 943
-#define _ITER_CHECK_LIST_r22 944
-#define _ITER_CHECK_LIST_r33 945
-#define _ITER_CHECK_RANGE_r02 946
-#define _ITER_CHECK_RANGE_r12 947
-#define _ITER_CHECK_RANGE_r22 948
-#define _ITER_CHECK_RANGE_r33 949
-#define _ITER_CHECK_TUPLE_r02 950
-#define _ITER_CHECK_TUPLE_r12 951
-#define _ITER_CHECK_TUPLE_r22 952
-#define _ITER_CHECK_TUPLE_r33 953
-#define _ITER_JUMP_LIST_r02 954
-#define _ITER_JUMP_LIST_r12 955
-#define _ITER_JUMP_LIST_r22 956
-#define _ITER_JUMP_LIST_r33 957
-#define _ITER_JUMP_RANGE_r02 958
-#define _ITER_JUMP_RANGE_r12 959
-#define _ITER_JUMP_RANGE_r22 960
-#define _ITER_JUMP_RANGE_r33 961
-#define _ITER_JUMP_TUPLE_r02 962
-#define _ITER_JUMP_TUPLE_r12 963
-#define _ITER_JUMP_TUPLE_r22 964
-#define _ITER_JUMP_TUPLE_r33 965
-#define _ITER_NEXT_LIST_r23 966
-#define _ITER_NEXT_LIST_TIER_TWO_r23 967
-#define _ITER_NEXT_RANGE_r03 968
-#define _ITER_NEXT_RANGE_r13 969
-#define _ITER_NEXT_RANGE_r23 970
-#define _ITER_NEXT_TUPLE_r03 971
-#define _ITER_NEXT_TUPLE_r13 972
-#define _ITER_NEXT_TUPLE_r23 973
-#define _JUMP_BACKWARD_NO_INTERRUPT_r00 974
-#define _JUMP_BACKWARD_NO_INTERRUPT_r11 975
-#define _JUMP_BACKWARD_NO_INTERRUPT_r22 976
-#define _JUMP_BACKWARD_NO_INTERRUPT_r33 977
-#define _JUMP_TO_TOP_r00 978
-#define _LIST_APPEND_r10 979
-#define _LIST_EXTEND_r10 980
-#define _LOAD_ATTR_r10 981
-#define _LOAD_ATTR_CLASS_r11 982
-#define _LOAD_ATTR_GETATTRIBUTE_OVERRIDDEN_r11 983
-#define _LOAD_ATTR_INSTANCE_VALUE_r02 984
-#define _LOAD_ATTR_INSTANCE_VALUE_r12 985
-#define _LOAD_ATTR_INSTANCE_VALUE_r23 986
-#define _LOAD_ATTR_METHOD_LAZY_DICT_r02 987
-#define _LOAD_ATTR_METHOD_LAZY_DICT_r12 988
-#define _LOAD_ATTR_METHOD_LAZY_DICT_r23 989
-#define _LOAD_ATTR_METHOD_NO_DICT_r02 990
-#define _LOAD_ATTR_METHOD_NO_DICT_r12 991
-#define _LOAD_ATTR_METHOD_NO_DICT_r23 992
-#define _LOAD_ATTR_METHOD_WITH_VALUES_r02 993
-#define _LOAD_ATTR_METHOD_WITH_VALUES_r12 994
-#define _LOAD_ATTR_METHOD_WITH_VALUES_r23 995
-#define _LOAD_ATTR_MODULE_r11 996
-#define _LOAD_ATTR_NONDESCRIPTOR_NO_DICT_r11 997
-#define _LOAD_ATTR_NONDESCRIPTOR_WITH_VALUES_r11 998
-#define _LOAD_ATTR_PROPERTY_FRAME_r11 999
-#define _LOAD_ATTR_SLOT_r11 1000
-#define _LOAD_ATTR_WITH_HINT_r11 1001
-#define _LOAD_BUILD_CLASS_r01 1002
-#define _LOAD_BYTECODE_r00 1003
-#define _LOAD_COMMON_CONSTANT_r01 1004
-#define _LOAD_COMMON_CONSTANT_r12 1005
-#define _LOAD_COMMON_CONSTANT_r23 1006
-#define _LOAD_CONST_r01 1007
-#define _LOAD_CONST_r12 1008
-#define _LOAD_CONST_r23 1009
-#define _LOAD_CONST_INLINE_r01 1010
-#define _LOAD_CONST_INLINE_r12 1011
-#define _LOAD_CONST_INLINE_r23 1012
-#define _LOAD_CONST_INLINE_BORROW_r01 1013
-#define _LOAD_CONST_INLINE_BORROW_r12 1014
-#define _LOAD_CONST_INLINE_BORROW_r23 1015
-#define _LOAD_CONST_UNDER_INLINE_r02 1016
-#define _LOAD_CONST_UNDER_INLINE_r12 1017
-#define _LOAD_CONST_UNDER_INLINE_r23 1018
-#define _LOAD_CONST_UNDER_INLINE_BORROW_r02 1019
-#define _LOAD_CONST_UNDER_INLINE_BORROW_r12 1020
-#define _LOAD_CONST_UNDER_INLINE_BORROW_r23 1021
-#define _LOAD_DEREF_r01 1022
-#define _LOAD_FAST_r01 1023
-#define _LOAD_FAST_r12 1024
-#define _LOAD_FAST_r23 1025
-#define _LOAD_FAST_0_r01 1026
-#define _LOAD_FAST_0_r12 1027
-#define _LOAD_FAST_0_r23 1028
-#define _LOAD_FAST_1_r01 1029
-#define _LOAD_FAST_1_r12 1030
-#define _LOAD_FAST_1_r23 1031
-#define _LOAD_FAST_2_r01 1032
-#define _LOAD_FAST_2_r12 1033
-#define _LOAD_FAST_2_r23 1034
-#define _LOAD_FAST_3_r01 1035
-#define _LOAD_FAST_3_r12 1036
-#define _LOAD_FAST_3_r23 1037
-#define _LOAD_FAST_4_r01 1038
-#define _LOAD_FAST_4_r12 1039
-#define _LOAD_FAST_4_r23 1040
-#define _LOAD_FAST_5_r01 1041
-#define _LOAD_FAST_5_r12 1042
-#define _LOAD_FAST_5_r23 1043
-#define _LOAD_FAST_6_r01 1044
-#define _LOAD_FAST_6_r12 1045
-#define _LOAD_FAST_6_r23 1046
-#define _LOAD_FAST_7_r01 1047
-#define _LOAD_FAST_7_r12 1048
-#define _LOAD_FAST_7_r23 1049
-#define _LOAD_FAST_AND_CLEAR_r01 1050
-#define _LOAD_FAST_AND_CLEAR_r12 1051
-#define _LOAD_FAST_AND_CLEAR_r23 1052
-#define _LOAD_FAST_BORROW_r01 1053
-#define _LOAD_FAST_BORROW_r12 1054
-#define _LOAD_FAST_BORROW_r23 1055
-#define _LOAD_FAST_BORROW_0_r01 1056
-#define _LOAD_FAST_BORROW_0_r12 1057
-#define _LOAD_FAST_BORROW_0_r23 1058
-#define _LOAD_FAST_BORROW_1_r01 1059
-#define _LOAD_FAST_BORROW_1_r12 1060
-#define _LOAD_FAST_BORROW_1_r23 1061
-#define _LOAD_FAST_BORROW_2_r01 1062
-#define _LOAD_FAST_BORROW_2_r12 1063
-#define _LOAD_FAST_BORROW_2_r23 1064
-#define _LOAD_FAST_BORROW_3_r01 1065
-#define _LOAD_FAST_BORROW_3_r12 1066
-#define _LOAD_FAST_BORROW_3_r23 1067
-#define _LOAD_FAST_BORROW_4_r01 1068
-#define _LOAD_FAST_BORROW_4_r12 1069
-#define _LOAD_FAST_BORROW_4_r23 1070
-#define _LOAD_FAST_BORROW_5_r01 1071
-#define _LOAD_FAST_BORROW_5_r12 1072
-#define _LOAD_FAST_BORROW_5_r23 1073
-#define _LOAD_FAST_BORROW_6_r01 1074
-#define _LOAD_FAST_BORROW_6_r12 1075
-#define _LOAD_FAST_BORROW_6_r23 1076
-#define _LOAD_FAST_BORROW_7_r01 1077
-#define _LOAD_FAST_BORROW_7_r12 1078
-#define _LOAD_FAST_BORROW_7_r23 1079
-#define _LOAD_FAST_BORROW_LOAD_FAST_BORROW_r02 1080
-#define _LOAD_FAST_BORROW_LOAD_FAST_BORROW_r13 1081
-#define _LOAD_FAST_CHECK_r01 1082
-#define _LOAD_FAST_CHECK_r12 1083
-#define _LOAD_FAST_CHECK_r23 1084
-#define _LOAD_FAST_LOAD_FAST_r02 1085
-#define _LOAD_FAST_LOAD_FAST_r13 1086
-#define _LOAD_FROM_DICT_OR_DEREF_r11 1087
-#define _LOAD_FROM_DICT_OR_GLOBALS_r11 1088
-#define _LOAD_GLOBAL_r00 1089
-#define _LOAD_GLOBAL_BUILTINS_r01 1090
-#define _LOAD_GLOBAL_MODULE_r01 1091
-#define _LOAD_LOCALS_r01 1092
-#define _LOAD_LOCALS_r12 1093
-#define _LOAD_LOCALS_r23 1094
-#define _LOAD_NAME_r01 1095
-#define _LOAD_SMALL_INT_r01 1096
-#define _LOAD_SMALL_INT_r12 1097
-#define _LOAD_SMALL_INT_r23 1098
-#define _LOAD_SMALL_INT_0_r01 1099
-#define _LOAD_SMALL_INT_0_r12 1100
-#define _LOAD_SMALL_INT_0_r23 1101
-#define _LOAD_SMALL_INT_1_r01 1102
-#define _LOAD_SMALL_INT_1_r12 1103
-#define _LOAD_SMALL_INT_1_r23 1104
-#define _LOAD_SMALL_INT_2_r01 1105
-#define _LOAD_SMALL_INT_2_r12 1106
-#define _LOAD_SMALL_INT_2_r23 1107
-#define _LOAD_SMALL_INT_3_r01 1108
-#define _LOAD_SMALL_INT_3_r12 1109
-#define _LOAD_SMALL_INT_3_r23 1110
-#define _LOAD_SPECIAL_r00 1111
-#define _LOAD_SUPER_ATTR_ATTR_r31 1112
-#define _LOAD_SUPER_ATTR_METHOD_r32 1113
-#define _MAKE_CALLARGS_A_TUPLE_r33 1114
-#define _MAKE_CELL_r00 1115
-#define _MAKE_FUNCTION_r11 1116
-#define _MAKE_WARM_r00 1117
-#define _MAKE_WARM_r11 1118
-#define _MAKE_WARM_r22 1119
-#define _MAKE_WARM_r33 1120
-#define _MAP_ADD_r20 1121
-#define _MATCH_CLASS_r31 1122
-#define _MATCH_KEYS_r23 1123
-#define _MATCH_MAPPING_r02 1124
-#define _MATCH_MAPPING_r12 1125
-#define _MATCH_MAPPING_r23 1126
-#define _MATCH_SEQUENCE_r02 1127
-#define _MATCH_SEQUENCE_r12 1128
-#define _MATCH_SEQUENCE_r23 1129
-#define _MAYBE_EXPAND_METHOD_r00 1130
-#define _MAYBE_EXPAND_METHOD_KW_r11 1131
-#define _MONITOR_CALL_r00 1132
-#define _MONITOR_CALL_KW_r11 1133
-#define _MONITOR_JUMP_BACKWARD_r00 1134
-#define _MONITOR_JUMP_BACKWARD_r11 1135
-#define _MONITOR_JUMP_BACKWARD_r22 1136
-#define _MONITOR_JUMP_BACKWARD_r33 1137
-#define _MONITOR_RESUME_r00 1138
-#define _NOP_r00 1139
-#define _NOP_r11 1140
-#define _NOP_r22 1141
-#define _NOP_r33 1142
-#define _POP_CALL_r20 1143
-#define _POP_CALL_LOAD_CONST_INLINE_BORROW_r21 1144
-#define _POP_CALL_ONE_r30 1145
-#define _POP_CALL_ONE_LOAD_CONST_INLINE_BORROW_r31 1146
-#define _POP_CALL_TWO_r30 1147
-#define _POP_CALL_TWO_LOAD_CONST_INLINE_BORROW_r31 1148
-#define _POP_EXCEPT_r10 1149
-#define _POP_ITER_r20 1150
-#define _POP_JUMP_IF_FALSE_r00 1151
-#define _POP_JUMP_IF_FALSE_r10 1152
-#define _POP_JUMP_IF_FALSE_r21 1153
-#define _POP_JUMP_IF_FALSE_r32 1154
-#define _POP_JUMP_IF_TRUE_r00 1155
-#define _POP_JUMP_IF_TRUE_r10 1156
-#define _POP_JUMP_IF_TRUE_r21 1157
-#define _POP_JUMP_IF_TRUE_r32 1158
-#define _POP_TOP_r10 1159
-#define _POP_TOP_FLOAT_r00 1160
-#define _POP_TOP_FLOAT_r10 1161
-#define _POP_TOP_FLOAT_r21 1162
-#define _POP_TOP_FLOAT_r32 1163
-#define _POP_TOP_INT_r00 1164
-#define _POP_TOP_INT_r10 1165
-#define _POP_TOP_INT_r21 1166
-#define _POP_TOP_INT_r32 1167
-#define _POP_TOP_LOAD_CONST_INLINE_r11 1168
-#define _POP_TOP_LOAD_CONST_INLINE_BORROW_r11 1169
-#define _POP_TOP_NOP_r00 1170
-#define _POP_TOP_NOP_r10 1171
-#define _POP_TOP_NOP_r21 1172
-#define _POP_TOP_NOP_r32 1173
-#define _POP_TOP_UNICODE_r00 1174
-#define _POP_TOP_UNICODE_r10 1175
-#define _POP_TOP_UNICODE_r21 1176
-#define _POP_TOP_UNICODE_r32 1177
-#define _POP_TWO_r20 1178
-#define _POP_TWO_LOAD_CONST_INLINE_BORROW_r21 1179
-#define _PUSH_EXC_INFO_r02 1180
-#define _PUSH_EXC_INFO_r12 1181
-#define _PUSH_EXC_INFO_r23 1182
-#define _PUSH_FRAME_r10 1183
-#define _PUSH_NULL_r01 1184
-#define _PUSH_NULL_r12 1185
-#define _PUSH_NULL_r23 1186
-#define _PUSH_NULL_CONDITIONAL_r00 1187
-#define _PY_FRAME_GENERAL_r01 1188
-#define _PY_FRAME_KW_r11 1189
-#define _QUICKEN_RESUME_r00 1190
-#define _QUICKEN_RESUME_r11 1191
-#define _QUICKEN_RESUME_r22 1192
-#define _QUICKEN_RESUME_r33 1193
-#define _REPLACE_WITH_TRUE_r11 1194
-#define _RESUME_CHECK_r00 1195
-#define _RESUME_CHECK_r11 1196
-#define _RESUME_CHECK_r22 1197
-#define _RESUME_CHECK_r33 1198
-#define _RETURN_GENERATOR_r01 1199
-#define _RETURN_VALUE_r11 1200
-#define _SAVE_RETURN_OFFSET_r00 1201
-#define _SAVE_RETURN_OFFSET_r11 1202
-#define _SAVE_RETURN_OFFSET_r22 1203
-#define _SAVE_RETURN_OFFSET_r33 1204
-#define _SEND_r22 1205
-#define _SEND_GEN_FRAME_r22 1206
-#define _SETUP_ANNOTATIONS_r00 1207
-#define _SET_ADD_r10 1208
-#define _SET_FUNCTION_ATTRIBUTE_r01 1209
-#define _SET_FUNCTION_ATTRIBUTE_r11 1210
-#define _SET_FUNCTION_ATTRIBUTE_r21 1211
-#define _SET_FUNCTION_ATTRIBUTE_r32 1212
-#define _SET_IP_r00 1213
-#define _SET_IP_r11 1214
-#define _SET_IP_r22 1215
-#define _SET_IP_r33 1216
-#define _SET_UPDATE_r10 1217
-#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r03 1218
-#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r13 1219
-#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r23 1220
-#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r33 1221
-#define _SPILL_OR_RELOAD_r01 1222
-#define _SPILL_OR_RELOAD_r02 1223
-#define _SPILL_OR_RELOAD_r03 1224
-#define _SPILL_OR_RELOAD_r10 1225
-#define _SPILL_OR_RELOAD_r12 1226
-#define _SPILL_OR_RELOAD_r13 1227
-#define _SPILL_OR_RELOAD_r20 1228
-#define _SPILL_OR_RELOAD_r21 1229
-#define _SPILL_OR_RELOAD_r23 1230
-#define _SPILL_OR_RELOAD_r30 1231
-#define _SPILL_OR_RELOAD_r31 1232
-#define _SPILL_OR_RELOAD_r32 1233
-#define _START_EXECUTOR_r00 1234
-#define _STORE_ATTR_r20 1235
-#define _STORE_ATTR_INSTANCE_VALUE_r21 1236
-#define _STORE_ATTR_SLOT_r21 1237
-#define _STORE_ATTR_WITH_HINT_r21 1238
-#define _STORE_DEREF_r10 1239
-#define _STORE_FAST_r10 1240
-#define _STORE_FAST_0_r10 1241
-#define _STORE_FAST_1_r10 1242
-#define _STORE_FAST_2_r10 1243
-#define _STORE_FAST_3_r10 1244
-#define _STORE_FAST_4_r10 1245
-#define _STORE_FAST_5_r10 1246
-#define _STORE_FAST_6_r10 1247
-#define _STORE_FAST_7_r10 1248
-#define _STORE_FAST_LOAD_FAST_r11 1249
-#define _STORE_FAST_STORE_FAST_r20 1250
-#define _STORE_GLOBAL_r10 1251
-#define _STORE_NAME_r10 1252
-#define _STORE_SLICE_r30 1253
-#define _STORE_SUBSCR_r30 1254
-#define _STORE_SUBSCR_DICT_r31 1255
-#define _STORE_SUBSCR_LIST_INT_r32 1256
-#define _SWAP_r11 1257
-#define _SWAP_2_r02 1258
-#define _SWAP_2_r12 1259
-#define _SWAP_2_r22 1260
-#define _SWAP_2_r33 1261
-#define _SWAP_3_r03 1262
-#define _SWAP_3_r13 1263
-#define _SWAP_3_r23 1264
-#define _SWAP_3_r33 1265
-#define _TIER2_RESUME_CHECK_r00 1266
-#define _TIER2_RESUME_CHECK_r11 1267
-#define _TIER2_RESUME_CHECK_r22 1268
-#define _TIER2_RESUME_CHECK_r33 1269
-#define _TO_BOOL_r11 1270
-#define _TO_BOOL_BOOL_r01 1271
-#define _TO_BOOL_BOOL_r11 1272
-#define _TO_BOOL_BOOL_r22 1273
-#define _TO_BOOL_BOOL_r33 1274
-#define _TO_BOOL_INT_r11 1275
-#define _TO_BOOL_LIST_r11 1276
-#define _TO_BOOL_NONE_r01 1277
-#define _TO_BOOL_NONE_r11 1278
-#define _TO_BOOL_NONE_r22 1279
-#define _TO_BOOL_NONE_r33 1280
-#define _TO_BOOL_STR_r11 1281
-#define _TRACE_RECORD_r00 1282
-#define _UNARY_INVERT_r11 1283
-#define _UNARY_NEGATIVE_r11 1284
-#define _UNARY_NOT_r01 1285
-#define _UNARY_NOT_r11 1286
-#define _UNARY_NOT_r22 1287
-#define _UNARY_NOT_r33 1288
-#define _UNPACK_EX_r10 1289
-#define _UNPACK_SEQUENCE_r10 1290
-#define _UNPACK_SEQUENCE_LIST_r10 1291
-#define _UNPACK_SEQUENCE_TUPLE_r10 1292
-#define _UNPACK_SEQUENCE_TWO_TUPLE_r12 1293
-#define _WITH_EXCEPT_START_r33 1294
-#define _YIELD_VALUE_r11 1295
-#define MAX_UOP_REGS_ID 1295
+#define _FOR_ITER_GEN_FRAME_r03 731
+#define _FOR_ITER_GEN_FRAME_r13 732
+#define _FOR_ITER_GEN_FRAME_r23 733
+#define _FOR_ITER_TIER_TWO_r23 734
+#define _GET_AITER_r11 735
+#define _GET_ANEXT_r12 736
+#define _GET_AWAITABLE_r11 737
+#define _GET_ITER_r12 738
+#define _GET_LEN_r12 739
+#define _GET_YIELD_FROM_ITER_r11 740
+#define _GUARD_BINARY_OP_EXTEND_r22 741
+#define _GUARD_CALLABLE_ISINSTANCE_r03 742
+#define _GUARD_CALLABLE_ISINSTANCE_r13 743
+#define _GUARD_CALLABLE_ISINSTANCE_r23 744
+#define _GUARD_CALLABLE_ISINSTANCE_r33 745
+#define _GUARD_CALLABLE_LEN_r03 746
+#define _GUARD_CALLABLE_LEN_r13 747
+#define _GUARD_CALLABLE_LEN_r23 748
+#define _GUARD_CALLABLE_LEN_r33 749
+#define _GUARD_CALLABLE_LIST_APPEND_r03 750
+#define _GUARD_CALLABLE_LIST_APPEND_r13 751
+#define _GUARD_CALLABLE_LIST_APPEND_r23 752
+#define _GUARD_CALLABLE_LIST_APPEND_r33 753
+#define _GUARD_CALLABLE_STR_1_r03 754
+#define _GUARD_CALLABLE_STR_1_r13 755
+#define _GUARD_CALLABLE_STR_1_r23 756
+#define _GUARD_CALLABLE_STR_1_r33 757
+#define _GUARD_CALLABLE_TUPLE_1_r03 758
+#define _GUARD_CALLABLE_TUPLE_1_r13 759
+#define _GUARD_CALLABLE_TUPLE_1_r23 760
+#define _GUARD_CALLABLE_TUPLE_1_r33 761
+#define _GUARD_CALLABLE_TYPE_1_r03 762
+#define _GUARD_CALLABLE_TYPE_1_r13 763
+#define _GUARD_CALLABLE_TYPE_1_r23 764
+#define _GUARD_CALLABLE_TYPE_1_r33 765
+#define _GUARD_DORV_NO_DICT_r01 766
+#define _GUARD_DORV_NO_DICT_r11 767
+#define _GUARD_DORV_NO_DICT_r22 768
+#define _GUARD_DORV_NO_DICT_r33 769
+#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r01 770
+#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r11 771
+#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r22 772
+#define _GUARD_DORV_VALUES_INST_ATTR_FROM_DICT_r33 773
+#define _GUARD_GLOBALS_VERSION_r00 774
+#define _GUARD_GLOBALS_VERSION_r11 775
+#define _GUARD_GLOBALS_VERSION_r22 776
+#define _GUARD_GLOBALS_VERSION_r33 777
+#define _GUARD_IP_RETURN_GENERATOR_r00 778
+#define _GUARD_IP_RETURN_GENERATOR_r11 779
+#define _GUARD_IP_RETURN_GENERATOR_r22 780
+#define _GUARD_IP_RETURN_GENERATOR_r33 781
+#define _GUARD_IP_RETURN_VALUE_r00 782
+#define _GUARD_IP_RETURN_VALUE_r11 783
+#define _GUARD_IP_RETURN_VALUE_r22 784
+#define _GUARD_IP_RETURN_VALUE_r33 785
+#define _GUARD_IP_YIELD_VALUE_r00 786
+#define _GUARD_IP_YIELD_VALUE_r11 787
+#define _GUARD_IP_YIELD_VALUE_r22 788
+#define _GUARD_IP_YIELD_VALUE_r33 789
+#define _GUARD_IP__PUSH_FRAME_r00 790
+#define _GUARD_IP__PUSH_FRAME_r11 791
+#define _GUARD_IP__PUSH_FRAME_r22 792
+#define _GUARD_IP__PUSH_FRAME_r33 793
+#define _GUARD_IS_FALSE_POP_r00 794
+#define _GUARD_IS_FALSE_POP_r10 795
+#define _GUARD_IS_FALSE_POP_r21 796
+#define _GUARD_IS_FALSE_POP_r32 797
+#define _GUARD_IS_NONE_POP_r00 798
+#define _GUARD_IS_NONE_POP_r10 799
+#define _GUARD_IS_NONE_POP_r21 800
+#define _GUARD_IS_NONE_POP_r32 801
+#define _GUARD_IS_NOT_NONE_POP_r10 802
+#define _GUARD_IS_TRUE_POP_r00 803
+#define _GUARD_IS_TRUE_POP_r10 804
+#define _GUARD_IS_TRUE_POP_r21 805
+#define _GUARD_IS_TRUE_POP_r32 806
+#define _GUARD_KEYS_VERSION_r01 807
+#define _GUARD_KEYS_VERSION_r11 808
+#define _GUARD_KEYS_VERSION_r22 809
+#define _GUARD_KEYS_VERSION_r33 810
+#define _GUARD_NOS_DICT_r02 811
+#define _GUARD_NOS_DICT_r12 812
+#define _GUARD_NOS_DICT_r22 813
+#define _GUARD_NOS_DICT_r33 814
+#define _GUARD_NOS_FLOAT_r02 815
+#define _GUARD_NOS_FLOAT_r12 816
+#define _GUARD_NOS_FLOAT_r22 817
+#define _GUARD_NOS_FLOAT_r33 818
+#define _GUARD_NOS_INT_r02 819
+#define _GUARD_NOS_INT_r12 820
+#define _GUARD_NOS_INT_r22 821
+#define _GUARD_NOS_INT_r33 822
+#define _GUARD_NOS_LIST_r02 823
+#define _GUARD_NOS_LIST_r12 824
+#define _GUARD_NOS_LIST_r22 825
+#define _GUARD_NOS_LIST_r33 826
+#define _GUARD_NOS_NOT_NULL_r02 827
+#define _GUARD_NOS_NOT_NULL_r12 828
+#define _GUARD_NOS_NOT_NULL_r22 829
+#define _GUARD_NOS_NOT_NULL_r33 830
+#define _GUARD_NOS_NULL_r02 831
+#define _GUARD_NOS_NULL_r12 832
+#define _GUARD_NOS_NULL_r22 833
+#define _GUARD_NOS_NULL_r33 834
+#define _GUARD_NOS_OVERFLOWED_r02 835
+#define _GUARD_NOS_OVERFLOWED_r12 836
+#define _GUARD_NOS_OVERFLOWED_r22 837
+#define _GUARD_NOS_OVERFLOWED_r33 838
+#define _GUARD_NOS_TUPLE_r02 839
+#define _GUARD_NOS_TUPLE_r12 840
+#define _GUARD_NOS_TUPLE_r22 841
+#define _GUARD_NOS_TUPLE_r33 842
+#define _GUARD_NOS_UNICODE_r02 843
+#define _GUARD_NOS_UNICODE_r12 844
+#define _GUARD_NOS_UNICODE_r22 845
+#define _GUARD_NOS_UNICODE_r33 846
+#define _GUARD_NOT_EXHAUSTED_LIST_r02 847
+#define _GUARD_NOT_EXHAUSTED_LIST_r12 848
+#define _GUARD_NOT_EXHAUSTED_LIST_r22 849
+#define _GUARD_NOT_EXHAUSTED_LIST_r33 850
+#define _GUARD_NOT_EXHAUSTED_RANGE_r02 851
+#define _GUARD_NOT_EXHAUSTED_RANGE_r12 852
+#define _GUARD_NOT_EXHAUSTED_RANGE_r22 853
+#define _GUARD_NOT_EXHAUSTED_RANGE_r33 854
+#define _GUARD_NOT_EXHAUSTED_TUPLE_r02 855
+#define _GUARD_NOT_EXHAUSTED_TUPLE_r12 856
+#define _GUARD_NOT_EXHAUSTED_TUPLE_r22 857
+#define _GUARD_NOT_EXHAUSTED_TUPLE_r33 858
+#define _GUARD_THIRD_NULL_r03 859
+#define _GUARD_THIRD_NULL_r13 860
+#define _GUARD_THIRD_NULL_r23 861
+#define _GUARD_THIRD_NULL_r33 862
+#define _GUARD_TOS_ANY_SET_r01 863
+#define _GUARD_TOS_ANY_SET_r11 864
+#define _GUARD_TOS_ANY_SET_r22 865
+#define _GUARD_TOS_ANY_SET_r33 866
+#define _GUARD_TOS_DICT_r01 867
+#define _GUARD_TOS_DICT_r11 868
+#define _GUARD_TOS_DICT_r22 869
+#define _GUARD_TOS_DICT_r33 870
+#define _GUARD_TOS_FLOAT_r01 871
+#define _GUARD_TOS_FLOAT_r11 872
+#define _GUARD_TOS_FLOAT_r22 873
+#define _GUARD_TOS_FLOAT_r33 874
+#define _GUARD_TOS_INT_r01 875
+#define _GUARD_TOS_INT_r11 876
+#define _GUARD_TOS_INT_r22 877
+#define _GUARD_TOS_INT_r33 878
+#define _GUARD_TOS_LIST_r01 879
+#define _GUARD_TOS_LIST_r11 880
+#define _GUARD_TOS_LIST_r22 881
+#define _GUARD_TOS_LIST_r33 882
+#define _GUARD_TOS_OVERFLOWED_r01 883
+#define _GUARD_TOS_OVERFLOWED_r11 884
+#define _GUARD_TOS_OVERFLOWED_r22 885
+#define _GUARD_TOS_OVERFLOWED_r33 886
+#define _GUARD_TOS_SLICE_r01 887
+#define _GUARD_TOS_SLICE_r11 888
+#define _GUARD_TOS_SLICE_r22 889
+#define _GUARD_TOS_SLICE_r33 890
+#define _GUARD_TOS_TUPLE_r01 891
+#define _GUARD_TOS_TUPLE_r11 892
+#define _GUARD_TOS_TUPLE_r22 893
+#define _GUARD_TOS_TUPLE_r33 894
+#define _GUARD_TOS_UNICODE_r01 895
+#define _GUARD_TOS_UNICODE_r11 896
+#define _GUARD_TOS_UNICODE_r22 897
+#define _GUARD_TOS_UNICODE_r33 898
+#define _GUARD_TYPE_VERSION_r01 899
+#define _GUARD_TYPE_VERSION_r11 900
+#define _GUARD_TYPE_VERSION_r22 901
+#define _GUARD_TYPE_VERSION_r33 902
+#define _GUARD_TYPE_VERSION_AND_LOCK_r01 903
+#define _GUARD_TYPE_VERSION_AND_LOCK_r11 904
+#define _GUARD_TYPE_VERSION_AND_LOCK_r22 905
+#define _GUARD_TYPE_VERSION_AND_LOCK_r33 906
+#define _HANDLE_PENDING_AND_DEOPT_r00 907
+#define _HANDLE_PENDING_AND_DEOPT_r10 908
+#define _HANDLE_PENDING_AND_DEOPT_r20 909
+#define _HANDLE_PENDING_AND_DEOPT_r30 910
+#define _IMPORT_FROM_r12 911
+#define _IMPORT_NAME_r21 912
+#define _INIT_CALL_BOUND_METHOD_EXACT_ARGS_r00 913
+#define _INIT_CALL_PY_EXACT_ARGS_r01 914
+#define _INIT_CALL_PY_EXACT_ARGS_0_r01 915
+#define _INIT_CALL_PY_EXACT_ARGS_1_r01 916
+#define _INIT_CALL_PY_EXACT_ARGS_2_r01 917
+#define _INIT_CALL_PY_EXACT_ARGS_3_r01 918
+#define _INIT_CALL_PY_EXACT_ARGS_4_r01 919
+#define _INSERT_NULL_r10 920
+#define _INSTRUMENTED_FOR_ITER_r23 921
+#define _INSTRUMENTED_INSTRUCTION_r00 922
+#define _INSTRUMENTED_JUMP_FORWARD_r00 923
+#define _INSTRUMENTED_JUMP_FORWARD_r11 924
+#define _INSTRUMENTED_JUMP_FORWARD_r22 925
+#define _INSTRUMENTED_JUMP_FORWARD_r33 926
+#define _INSTRUMENTED_LINE_r00 927
+#define _INSTRUMENTED_NOT_TAKEN_r00 928
+#define _INSTRUMENTED_NOT_TAKEN_r11 929
+#define _INSTRUMENTED_NOT_TAKEN_r22 930
+#define _INSTRUMENTED_NOT_TAKEN_r33 931
+#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r00 932
+#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r10 933
+#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r21 934
+#define _INSTRUMENTED_POP_JUMP_IF_FALSE_r32 935
+#define _INSTRUMENTED_POP_JUMP_IF_NONE_r10 936
+#define _INSTRUMENTED_POP_JUMP_IF_NOT_NONE_r10 937
+#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r00 938
+#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r10 939
+#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r21 940
+#define _INSTRUMENTED_POP_JUMP_IF_TRUE_r32 941
+#define _IS_NONE_r11 942
+#define _IS_OP_r21 943
+#define _ITER_CHECK_LIST_r02 944
+#define _ITER_CHECK_LIST_r12 945
+#define _ITER_CHECK_LIST_r22 946
+#define _ITER_CHECK_LIST_r33 947
+#define _ITER_CHECK_RANGE_r02 948
+#define _ITER_CHECK_RANGE_r12 949
+#define _ITER_CHECK_RANGE_r22 950
+#define _ITER_CHECK_RANGE_r33 951
+#define _ITER_CHECK_TUPLE_r02 952
+#define _ITER_CHECK_TUPLE_r12 953
+#define _ITER_CHECK_TUPLE_r22 954
+#define _ITER_CHECK_TUPLE_r33 955
+#define _ITER_JUMP_LIST_r02 956
+#define _ITER_JUMP_LIST_r12 957
+#define _ITER_JUMP_LIST_r22 958
+#define _ITER_JUMP_LIST_r33 959
+#define _ITER_JUMP_RANGE_r02 960
+#define _ITER_JUMP_RANGE_r12 961
+#define _ITER_JUMP_RANGE_r22 962
+#define _ITER_JUMP_RANGE_r33 963
+#define _ITER_JUMP_TUPLE_r02 964
+#define _ITER_JUMP_TUPLE_r12 965
+#define _ITER_JUMP_TUPLE_r22 966
+#define _ITER_JUMP_TUPLE_r33 967
+#define _ITER_NEXT_LIST_r23 968
+#define _ITER_NEXT_LIST_TIER_TWO_r23 969
+#define _ITER_NEXT_RANGE_r03 970
+#define _ITER_NEXT_RANGE_r13 971
+#define _ITER_NEXT_RANGE_r23 972
+#define _ITER_NEXT_TUPLE_r03 973
+#define _ITER_NEXT_TUPLE_r13 974
+#define _ITER_NEXT_TUPLE_r23 975
+#define _JUMP_BACKWARD_NO_INTERRUPT_r00 976
+#define _JUMP_BACKWARD_NO_INTERRUPT_r11 977
+#define _JUMP_BACKWARD_NO_INTERRUPT_r22 978
+#define _JUMP_BACKWARD_NO_INTERRUPT_r33 979
+#define _JUMP_TO_TOP_r00 980
+#define _LIST_APPEND_r10 981
+#define _LIST_EXTEND_r10 982
+#define _LOAD_ATTR_r10 983
+#define _LOAD_ATTR_CLASS_r11 984
+#define _LOAD_ATTR_GETATTRIBUTE_OVERRIDDEN_r11 985
+#define _LOAD_ATTR_INSTANCE_VALUE_r02 986
+#define _LOAD_ATTR_INSTANCE_VALUE_r12 987
+#define _LOAD_ATTR_INSTANCE_VALUE_r23 988
+#define _LOAD_ATTR_METHOD_LAZY_DICT_r02 989
+#define _LOAD_ATTR_METHOD_LAZY_DICT_r12 990
+#define _LOAD_ATTR_METHOD_LAZY_DICT_r23 991
+#define _LOAD_ATTR_METHOD_NO_DICT_r02 992
+#define _LOAD_ATTR_METHOD_NO_DICT_r12 993
+#define _LOAD_ATTR_METHOD_NO_DICT_r23 994
+#define _LOAD_ATTR_METHOD_WITH_VALUES_r02 995
+#define _LOAD_ATTR_METHOD_WITH_VALUES_r12 996
+#define _LOAD_ATTR_METHOD_WITH_VALUES_r23 997
+#define _LOAD_ATTR_MODULE_r11 998
+#define _LOAD_ATTR_NONDESCRIPTOR_NO_DICT_r11 999
+#define _LOAD_ATTR_NONDESCRIPTOR_WITH_VALUES_r11 1000
+#define _LOAD_ATTR_PROPERTY_FRAME_r11 1001
+#define _LOAD_ATTR_SLOT_r11 1002
+#define _LOAD_ATTR_WITH_HINT_r11 1003
+#define _LOAD_BUILD_CLASS_r01 1004
+#define _LOAD_BYTECODE_r00 1005
+#define _LOAD_COMMON_CONSTANT_r01 1006
+#define _LOAD_COMMON_CONSTANT_r12 1007
+#define _LOAD_COMMON_CONSTANT_r23 1008
+#define _LOAD_CONST_r01 1009
+#define _LOAD_CONST_r12 1010
+#define _LOAD_CONST_r23 1011
+#define _LOAD_CONST_INLINE_r01 1012
+#define _LOAD_CONST_INLINE_r12 1013
+#define _LOAD_CONST_INLINE_r23 1014
+#define _LOAD_CONST_INLINE_BORROW_r01 1015
+#define _LOAD_CONST_INLINE_BORROW_r12 1016
+#define _LOAD_CONST_INLINE_BORROW_r23 1017
+#define _LOAD_CONST_UNDER_INLINE_r02 1018
+#define _LOAD_CONST_UNDER_INLINE_r12 1019
+#define _LOAD_CONST_UNDER_INLINE_r23 1020
+#define _LOAD_CONST_UNDER_INLINE_BORROW_r02 1021
+#define _LOAD_CONST_UNDER_INLINE_BORROW_r12 1022
+#define _LOAD_CONST_UNDER_INLINE_BORROW_r23 1023
+#define _LOAD_DEREF_r01 1024
+#define _LOAD_FAST_r01 1025
+#define _LOAD_FAST_r12 1026
+#define _LOAD_FAST_r23 1027
+#define _LOAD_FAST_0_r01 1028
+#define _LOAD_FAST_0_r12 1029
+#define _LOAD_FAST_0_r23 1030
+#define _LOAD_FAST_1_r01 1031
+#define _LOAD_FAST_1_r12 1032
+#define _LOAD_FAST_1_r23 1033
+#define _LOAD_FAST_2_r01 1034
+#define _LOAD_FAST_2_r12 1035
+#define _LOAD_FAST_2_r23 1036
+#define _LOAD_FAST_3_r01 1037
+#define _LOAD_FAST_3_r12 1038
+#define _LOAD_FAST_3_r23 1039
+#define _LOAD_FAST_4_r01 1040
+#define _LOAD_FAST_4_r12 1041
+#define _LOAD_FAST_4_r23 1042
+#define _LOAD_FAST_5_r01 1043
+#define _LOAD_FAST_5_r12 1044
+#define _LOAD_FAST_5_r23 1045
+#define _LOAD_FAST_6_r01 1046
+#define _LOAD_FAST_6_r12 1047
+#define _LOAD_FAST_6_r23 1048
+#define _LOAD_FAST_7_r01 1049
+#define _LOAD_FAST_7_r12 1050
+#define _LOAD_FAST_7_r23 1051
+#define _LOAD_FAST_AND_CLEAR_r01 1052
+#define _LOAD_FAST_AND_CLEAR_r12 1053
+#define _LOAD_FAST_AND_CLEAR_r23 1054
+#define _LOAD_FAST_BORROW_r01 1055
+#define _LOAD_FAST_BORROW_r12 1056
+#define _LOAD_FAST_BORROW_r23 1057
+#define _LOAD_FAST_BORROW_0_r01 1058
+#define _LOAD_FAST_BORROW_0_r12 1059
+#define _LOAD_FAST_BORROW_0_r23 1060
+#define _LOAD_FAST_BORROW_1_r01 1061
+#define _LOAD_FAST_BORROW_1_r12 1062
+#define _LOAD_FAST_BORROW_1_r23 1063
+#define _LOAD_FAST_BORROW_2_r01 1064
+#define _LOAD_FAST_BORROW_2_r12 1065
+#define _LOAD_FAST_BORROW_2_r23 1066
+#define _LOAD_FAST_BORROW_3_r01 1067
+#define _LOAD_FAST_BORROW_3_r12 1068
+#define _LOAD_FAST_BORROW_3_r23 1069
+#define _LOAD_FAST_BORROW_4_r01 1070
+#define _LOAD_FAST_BORROW_4_r12 1071
+#define _LOAD_FAST_BORROW_4_r23 1072
+#define _LOAD_FAST_BORROW_5_r01 1073
+#define _LOAD_FAST_BORROW_5_r12 1074
+#define _LOAD_FAST_BORROW_5_r23 1075
+#define _LOAD_FAST_BORROW_6_r01 1076
+#define _LOAD_FAST_BORROW_6_r12 1077
+#define _LOAD_FAST_BORROW_6_r23 1078
+#define _LOAD_FAST_BORROW_7_r01 1079
+#define _LOAD_FAST_BORROW_7_r12 1080
+#define _LOAD_FAST_BORROW_7_r23 1081
+#define _LOAD_FAST_BORROW_LOAD_FAST_BORROW_r02 1082
+#define _LOAD_FAST_BORROW_LOAD_FAST_BORROW_r13 1083
+#define _LOAD_FAST_CHECK_r01 1084
+#define _LOAD_FAST_CHECK_r12 1085
+#define _LOAD_FAST_CHECK_r23 1086
+#define _LOAD_FAST_LOAD_FAST_r02 1087
+#define _LOAD_FAST_LOAD_FAST_r13 1088
+#define _LOAD_FROM_DICT_OR_DEREF_r11 1089
+#define _LOAD_FROM_DICT_OR_GLOBALS_r11 1090
+#define _LOAD_GLOBAL_r00 1091
+#define _LOAD_GLOBAL_BUILTINS_r01 1092
+#define _LOAD_GLOBAL_MODULE_r01 1093
+#define _LOAD_LOCALS_r01 1094
+#define _LOAD_LOCALS_r12 1095
+#define _LOAD_LOCALS_r23 1096
+#define _LOAD_NAME_r01 1097
+#define _LOAD_SMALL_INT_r01 1098
+#define _LOAD_SMALL_INT_r12 1099
+#define _LOAD_SMALL_INT_r23 1100
+#define _LOAD_SMALL_INT_0_r01 1101
+#define _LOAD_SMALL_INT_0_r12 1102
+#define _LOAD_SMALL_INT_0_r23 1103
+#define _LOAD_SMALL_INT_1_r01 1104
+#define _LOAD_SMALL_INT_1_r12 1105
+#define _LOAD_SMALL_INT_1_r23 1106
+#define _LOAD_SMALL_INT_2_r01 1107
+#define _LOAD_SMALL_INT_2_r12 1108
+#define _LOAD_SMALL_INT_2_r23 1109
+#define _LOAD_SMALL_INT_3_r01 1110
+#define _LOAD_SMALL_INT_3_r12 1111
+#define _LOAD_SMALL_INT_3_r23 1112
+#define _LOAD_SPECIAL_r00 1113
+#define _LOAD_SUPER_ATTR_ATTR_r31 1114
+#define _LOAD_SUPER_ATTR_METHOD_r32 1115
+#define _MAKE_CALLARGS_A_TUPLE_r33 1116
+#define _MAKE_CELL_r00 1117
+#define _MAKE_FUNCTION_r11 1118
+#define _MAKE_WARM_r00 1119
+#define _MAKE_WARM_r11 1120
+#define _MAKE_WARM_r22 1121
+#define _MAKE_WARM_r33 1122
+#define _MAP_ADD_r20 1123
+#define _MATCH_CLASS_r31 1124
+#define _MATCH_KEYS_r23 1125
+#define _MATCH_MAPPING_r02 1126
+#define _MATCH_MAPPING_r12 1127
+#define _MATCH_MAPPING_r23 1128
+#define _MATCH_SEQUENCE_r02 1129
+#define _MATCH_SEQUENCE_r12 1130
+#define _MATCH_SEQUENCE_r23 1131
+#define _MAYBE_EXPAND_METHOD_r00 1132
+#define _MAYBE_EXPAND_METHOD_KW_r11 1133
+#define _MONITOR_CALL_r00 1134
+#define _MONITOR_CALL_KW_r11 1135
+#define _MONITOR_JUMP_BACKWARD_r00 1136
+#define _MONITOR_JUMP_BACKWARD_r11 1137
+#define _MONITOR_JUMP_BACKWARD_r22 1138
+#define _MONITOR_JUMP_BACKWARD_r33 1139
+#define _MONITOR_RESUME_r00 1140
+#define _NOP_r00 1141
+#define _NOP_r11 1142
+#define _NOP_r22 1143
+#define _NOP_r33 1144
+#define _POP_CALL_r20 1145
+#define _POP_CALL_LOAD_CONST_INLINE_BORROW_r21 1146
+#define _POP_CALL_ONE_r30 1147
+#define _POP_CALL_ONE_LOAD_CONST_INLINE_BORROW_r31 1148
+#define _POP_CALL_TWO_r30 1149
+#define _POP_CALL_TWO_LOAD_CONST_INLINE_BORROW_r31 1150
+#define _POP_EXCEPT_r10 1151
+#define _POP_ITER_r20 1152
+#define _POP_JUMP_IF_FALSE_r00 1153
+#define _POP_JUMP_IF_FALSE_r10 1154
+#define _POP_JUMP_IF_FALSE_r21 1155
+#define _POP_JUMP_IF_FALSE_r32 1156
+#define _POP_JUMP_IF_TRUE_r00 1157
+#define _POP_JUMP_IF_TRUE_r10 1158
+#define _POP_JUMP_IF_TRUE_r21 1159
+#define _POP_JUMP_IF_TRUE_r32 1160
+#define _POP_TOP_r10 1161
+#define _POP_TOP_FLOAT_r00 1162
+#define _POP_TOP_FLOAT_r10 1163
+#define _POP_TOP_FLOAT_r21 1164
+#define _POP_TOP_FLOAT_r32 1165
+#define _POP_TOP_INT_r00 1166
+#define _POP_TOP_INT_r10 1167
+#define _POP_TOP_INT_r21 1168
+#define _POP_TOP_INT_r32 1169
+#define _POP_TOP_LOAD_CONST_INLINE_r11 1170
+#define _POP_TOP_LOAD_CONST_INLINE_BORROW_r11 1171
+#define _POP_TOP_NOP_r00 1172
+#define _POP_TOP_NOP_r10 1173
+#define _POP_TOP_NOP_r21 1174
+#define _POP_TOP_NOP_r32 1175
+#define _POP_TOP_UNICODE_r00 1176
+#define _POP_TOP_UNICODE_r10 1177
+#define _POP_TOP_UNICODE_r21 1178
+#define _POP_TOP_UNICODE_r32 1179
+#define _POP_TWO_r20 1180
+#define _POP_TWO_LOAD_CONST_INLINE_BORROW_r21 1181
+#define _PUSH_EXC_INFO_r02 1182
+#define _PUSH_EXC_INFO_r12 1183
+#define _PUSH_EXC_INFO_r23 1184
+#define _PUSH_FRAME_r10 1185
+#define _PUSH_NULL_r01 1186
+#define _PUSH_NULL_r12 1187
+#define _PUSH_NULL_r23 1188
+#define _PUSH_NULL_CONDITIONAL_r00 1189
+#define _PY_FRAME_GENERAL_r01 1190
+#define _PY_FRAME_KW_r11 1191
+#define _QUICKEN_RESUME_r00 1192
+#define _QUICKEN_RESUME_r11 1193
+#define _QUICKEN_RESUME_r22 1194
+#define _QUICKEN_RESUME_r33 1195
+#define _REPLACE_WITH_TRUE_r11 1196
+#define _RESUME_CHECK_r00 1197
+#define _RESUME_CHECK_r11 1198
+#define _RESUME_CHECK_r22 1199
+#define _RESUME_CHECK_r33 1200
+#define _RETURN_GENERATOR_r01 1201
+#define _RETURN_VALUE_r11 1202
+#define _SAVE_RETURN_OFFSET_r00 1203
+#define _SAVE_RETURN_OFFSET_r11 1204
+#define _SAVE_RETURN_OFFSET_r22 1205
+#define _SAVE_RETURN_OFFSET_r33 1206
+#define _SEND_r22 1207
+#define _SEND_GEN_FRAME_r22 1208
+#define _SETUP_ANNOTATIONS_r00 1209
+#define _SET_ADD_r10 1210
+#define _SET_FUNCTION_ATTRIBUTE_r01 1211
+#define _SET_FUNCTION_ATTRIBUTE_r11 1212
+#define _SET_FUNCTION_ATTRIBUTE_r21 1213
+#define _SET_FUNCTION_ATTRIBUTE_r32 1214
+#define _SET_IP_r00 1215
+#define _SET_IP_r11 1216
+#define _SET_IP_r22 1217
+#define _SET_IP_r33 1218
+#define _SET_UPDATE_r10 1219
+#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r03 1220
+#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r13 1221
+#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r23 1222
+#define _SHUFFLE_3_LOAD_CONST_INLINE_BORROW_r33 1223
+#define _SPILL_OR_RELOAD_r01 1224
+#define _SPILL_OR_RELOAD_r02 1225
+#define _SPILL_OR_RELOAD_r03 1226
+#define _SPILL_OR_RELOAD_r10 1227
+#define _SPILL_OR_RELOAD_r12 1228
+#define _SPILL_OR_RELOAD_r13 1229
+#define _SPILL_OR_RELOAD_r20 1230
+#define _SPILL_OR_RELOAD_r21 1231
+#define _SPILL_OR_RELOAD_r23 1232
+#define _SPILL_OR_RELOAD_r30 1233
+#define _SPILL_OR_RELOAD_r31 1234
+#define _SPILL_OR_RELOAD_r32 1235
+#define _START_EXECUTOR_r00 1236
+#define _STORE_ATTR_r20 1237
+#define _STORE_ATTR_INSTANCE_VALUE_r21 1238
+#define _STORE_ATTR_SLOT_r21 1239
+#define _STORE_ATTR_WITH_HINT_r21 1240
+#define _STORE_DEREF_r10 1241
+#define _STORE_FAST_r10 1242
+#define _STORE_FAST_0_r10 1243
+#define _STORE_FAST_1_r10 1244
+#define _STORE_FAST_2_r10 1245
+#define _STORE_FAST_3_r10 1246
+#define _STORE_FAST_4_r10 1247
+#define _STORE_FAST_5_r10 1248
+#define _STORE_FAST_6_r10 1249
+#define _STORE_FAST_7_r10 1250
+#define _STORE_FAST_LOAD_FAST_r11 1251
+#define _STORE_FAST_STORE_FAST_r20 1252
+#define _STORE_GLOBAL_r10 1253
+#define _STORE_NAME_r10 1254
+#define _STORE_SLICE_r30 1255
+#define _STORE_SUBSCR_r30 1256
+#define _STORE_SUBSCR_DICT_r31 1257
+#define _STORE_SUBSCR_LIST_INT_r32 1258
+#define _SWAP_r11 1259
+#define _SWAP_2_r02 1260
+#define _SWAP_2_r12 1261
+#define _SWAP_2_r22 1262
+#define _SWAP_2_r33 1263
+#define _SWAP_3_r03 1264
+#define _SWAP_3_r13 1265
+#define _SWAP_3_r23 1266
+#define _SWAP_3_r33 1267
+#define _TIER2_RESUME_CHECK_r00 1268
+#define _TIER2_RESUME_CHECK_r11 1269
+#define _TIER2_RESUME_CHECK_r22 1270
+#define _TIER2_RESUME_CHECK_r33 1271
+#define _TO_BOOL_r11 1272
+#define _TO_BOOL_BOOL_r01 1273
+#define _TO_BOOL_BOOL_r11 1274
+#define _TO_BOOL_BOOL_r22 1275
+#define _TO_BOOL_BOOL_r33 1276
+#define _TO_BOOL_INT_r11 1277
+#define _TO_BOOL_LIST_r11 1278
+#define _TO_BOOL_NONE_r01 1279
+#define _TO_BOOL_NONE_r11 1280
+#define _TO_BOOL_NONE_r22 1281
+#define _TO_BOOL_NONE_r33 1282
+#define _TO_BOOL_STR_r11 1283
+#define _TRACE_RECORD_r00 1284
+#define _UNARY_INVERT_r11 1285
+#define _UNARY_NEGATIVE_r11 1286
+#define _UNARY_NOT_r01 1287
+#define _UNARY_NOT_r11 1288
+#define _UNARY_NOT_r22 1289
+#define _UNARY_NOT_r33 1290
+#define _UNPACK_EX_r10 1291
+#define _UNPACK_SEQUENCE_r10 1292
+#define _UNPACK_SEQUENCE_LIST_r10 1293
+#define _UNPACK_SEQUENCE_TUPLE_r10 1294
+#define _UNPACK_SEQUENCE_TWO_TUPLE_r12 1295
+#define _WITH_EXCEPT_START_r33 1296
+#define _YIELD_VALUE_r11 1297
+#define MAX_UOP_REGS_ID 1297
#ifdef __cplusplus
}
diff --git a/Include/internal/pycore_uop_metadata.h
b/Include/internal/pycore_uop_metadata.h
index 1281eeb041daab..ec374dd5818432 100644
--- a/Include/internal/pycore_uop_metadata.h
+++ b/Include/internal/pycore_uop_metadata.h
@@ -2131,10 +2131,10 @@ const _PyUopCachingInfo _PyUop_Caching[MAX_UOP_ID+1] = {
},
},
[_FOR_ITER_GEN_FRAME] = {
- .best = { 2, 2, 2, 2 },
+ .best = { 0, 1, 2, 2 },
.entries = {
- { -1, -1, -1 },
- { -1, -1, -1 },
+ { 3, 0, _FOR_ITER_GEN_FRAME_r03 },
+ { 3, 1, _FOR_ITER_GEN_FRAME_r13 },
{ 3, 2, _FOR_ITER_GEN_FRAME_r23 },
{ -1, -1, -1 },
},
@@ -3620,6 +3620,8 @@ const uint16_t _PyUop_Uncached[MAX_UOP_REGS_ID+1] = {
[_ITER_NEXT_RANGE_r03] = _ITER_NEXT_RANGE,
[_ITER_NEXT_RANGE_r13] = _ITER_NEXT_RANGE,
[_ITER_NEXT_RANGE_r23] = _ITER_NEXT_RANGE,
+ [_FOR_ITER_GEN_FRAME_r03] = _FOR_ITER_GEN_FRAME,
+ [_FOR_ITER_GEN_FRAME_r13] = _FOR_ITER_GEN_FRAME,
[_FOR_ITER_GEN_FRAME_r23] = _FOR_ITER_GEN_FRAME,
[_INSERT_NULL_r10] = _INSERT_NULL,
[_LOAD_SPECIAL_r00] = _LOAD_SPECIAL,
@@ -4182,6 +4184,8 @@ const char *const _PyOpcode_uop_name[MAX_UOP_REGS_ID+1] =
{
[_FORMAT_WITH_SPEC] = "_FORMAT_WITH_SPEC",
[_FORMAT_WITH_SPEC_r21] = "_FORMAT_WITH_SPEC_r21",
[_FOR_ITER_GEN_FRAME] = "_FOR_ITER_GEN_FRAME",
+ [_FOR_ITER_GEN_FRAME_r03] = "_FOR_ITER_GEN_FRAME_r03",
+ [_FOR_ITER_GEN_FRAME_r13] = "_FOR_ITER_GEN_FRAME_r13",
[_FOR_ITER_GEN_FRAME_r23] = "_FOR_ITER_GEN_FRAME_r23",
[_FOR_ITER_TIER_TWO] = "_FOR_ITER_TIER_TWO",
[_FOR_ITER_TIER_TWO_r23] = "_FOR_ITER_TIER_TWO_r23",
diff --git a/Lib/test/test_free_threading/test_generators.py
b/Lib/test/test_free_threading/test_generators.py
index d01675eb38b370..11f59301bcd51d 100644
--- a/Lib/test/test_free_threading/test_generators.py
+++ b/Lib/test/test_free_threading/test_generators.py
@@ -49,3 +49,74 @@ def test_concurrent_write(self):
self.concurrent_write_with_func(func=set_gen_name)
with self.subTest(func=set_gen_qualname):
self.concurrent_write_with_func(func=set_gen_qualname)
+
+ def test_concurrent_send(self):
+ def gen():
+ yield 1
+ yield 2
+ yield 3
+ yield 4
+ yield 5
+
+ def run_test(drive_generator):
+ g = gen()
+ values = []
+ threading_helper.run_concurrently(drive_generator,
self.NUM_THREADS, args=(g, values,))
+ self.assertEqual(sorted(values), [1, 2, 3, 4, 5])
+
+ def call_next(g, values):
+ while True:
+ try:
+ values.append(next(g))
+ except ValueError:
+ continue
+ except StopIteration:
+ break
+
+ with self.subTest(method='next'):
+ run_test(call_next)
+
+ def call_send(g, values):
+ while True:
+ try:
+ values.append(g.send(None))
+ except ValueError:
+ continue
+ except StopIteration:
+ break
+
+ with self.subTest(method='send'):
+ run_test(call_send)
+
+ def for_iter_gen(g, values):
+ while True:
+ try:
+ for value in g:
+ values.append(value)
+ else:
+ break
+ except ValueError:
+ continue
+
+ with self.subTest(method='for'):
+ run_test(for_iter_gen)
+
+ def test_concurrent_close(self):
+ def gen():
+ for i in range(10):
+ yield i
+ time.sleep(0.001)
+
+ def drive_generator(g):
+ while True:
+ try:
+ for value in g:
+ if value == 5:
+ g.close()
+ else:
+ return
+ except ValueError as e:
+ self.assertEqual(e.args[0], "generator already executing")
+
+ g = gen()
+ threading_helper.run_concurrently(drive_generator, self.NUM_THREADS,
args=(g,))
diff --git a/Objects/genobject.c b/Objects/genobject.c
index 3694198289de8c..508d215a0cfe64 100644
--- a/Objects/genobject.c
+++ b/Objects/genobject.c
@@ -36,6 +36,14 @@ static PyObject* async_gen_athrow_new(PyAsyncGenObject *,
PyObject *);
#define _PyAsyncGenObject_CAST(op) \
_Py_CAST(PyAsyncGenObject*, (op))
+#ifdef Py_GIL_DISABLED
+# define _Py_GEN_TRY_SET_FRAME_STATE(gen, expected, state) \
+ _Py_atomic_compare_exchange_int8(&(gen)->gi_frame_state, &expected,
(state))
+#else
+# define _Py_GEN_TRY_SET_FRAME_STATE(gen, expected, state) \
+ ((gen)->gi_frame_state = (state), true)
+#endif
+
static const char *NON_INIT_CORO_MSG = "can't send non-None value to a "
"just-started coroutine";
@@ -145,10 +153,7 @@ _PyGen_Finalize(PyObject *self)
static void
gen_clear_frame(PyGenObject *gen)
{
- if (gen->gi_frame_state == FRAME_CLEARED)
- return;
-
- gen->gi_frame_state = FRAME_CLEARED;
+ assert(gen->gi_frame_state == FRAME_CLEARED);
_PyInterpreterFrame *frame = &gen->gi_iframe;
frame->previous = NULL;
_PyFrame_ClearExceptCode(frame);
@@ -179,7 +184,10 @@ gen_dealloc(PyObject *self)
if (PyCoro_CheckExact(gen)) {
Py_CLEAR(((PyCoroObject *)gen)->cr_origin_or_finalizer);
}
- gen_clear_frame(gen);
+ if (gen->gi_frame_state != FRAME_CLEARED) {
+ gen->gi_frame_state = FRAME_CLEARED;
+ gen_clear_frame(gen);
+ }
assert(gen->gi_exc_state.exc_value == NULL);
PyStackRef_CLEAR(gen->gi_iframe.f_executable);
Py_CLEAR(gen->gi_name);
@@ -188,58 +196,31 @@ gen_dealloc(PyObject *self)
PyObject_GC_Del(gen);
}
-static PySendResult
-gen_send_ex2(PyGenObject *gen, PyObject *arg, PyObject **presult,
- int exc, int closing)
+static void
+gen_raise_already_executing_error(PyGenObject *gen)
{
- PyThreadState *tstate = _PyThreadState_GET();
- _PyInterpreterFrame *frame = &gen->gi_iframe;
-
- *presult = NULL;
- if (gen->gi_frame_state == FRAME_CREATED && arg && arg != Py_None) {
- const char *msg = "can't send non-None value to a "
- "just-started generator";
- if (PyCoro_CheckExact(gen)) {
- msg = NON_INIT_CORO_MSG;
- }
- else if (PyAsyncGen_CheckExact(gen)) {
- msg = "can't send non-None value to a "
- "just-started async generator";
- }
- PyErr_SetString(PyExc_TypeError, msg);
- return PYGEN_ERROR;
- }
- if (gen->gi_frame_state == FRAME_EXECUTING) {
- const char *msg = "generator already executing";
- if (PyCoro_CheckExact(gen)) {
- msg = "coroutine already executing";
- }
- else if (PyAsyncGen_CheckExact(gen)) {
- msg = "async generator already executing";
- }
- PyErr_SetString(PyExc_ValueError, msg);
- return PYGEN_ERROR;
+ const char *msg = "generator already executing";
+ if (PyCoro_CheckExact(gen)) {
+ msg = "coroutine already executing";
}
- if (FRAME_STATE_FINISHED(gen->gi_frame_state)) {
- if (PyCoro_CheckExact(gen) && !closing) {
- /* `gen` is an exhausted coroutine: raise an error,
- except when called from gen_close(), which should
- always be a silent method. */
- PyErr_SetString(
- PyExc_RuntimeError,
- "cannot reuse already awaited coroutine");
- }
- else if (arg && !exc) {
- /* `gen` is an exhausted generator:
- only return value if called from send(). */
- *presult = Py_NewRef(Py_None);
- return PYGEN_RETURN;
- }
- return PYGEN_ERROR;
+ else if (PyAsyncGen_CheckExact(gen)) {
+ msg = "async generator already executing";
}
+ PyErr_SetString(PyExc_ValueError, msg);
+}
+
+// Send 'arg' into 'gen'. On success, return PYGEN_NEXT or PYGEN_RETURN.
+// Returns PYGEN_ERROR on failure. 'presult' is set to the yielded or
+// returned value.
+// The generator must be in the FRAME_EXECUTING state when this function
+// is called.
+static PySendResult
+gen_send_ex2(PyGenObject *gen, PyObject *arg, PyObject **presult, int exc)
+{
+ assert(FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state) ==
FRAME_EXECUTING);
- assert((gen->gi_frame_state == FRAME_CREATED) ||
- FRAME_STATE_SUSPENDED(gen->gi_frame_state));
+ PyThreadState *tstate = _PyThreadState_GET();
+ _PyInterpreterFrame *frame = &gen->gi_iframe;
/* Push arg onto the frame's value stack */
PyObject *arg_obj = arg ? arg : Py_None;
@@ -254,21 +235,34 @@ gen_send_ex2(PyGenObject *gen, PyObject *arg, PyObject
**presult,
_PyErr_ChainStackItem();
}
- gen->gi_frame_state = FRAME_EXECUTING;
EVAL_CALL_STAT_INC(EVAL_CALL_GENERATOR);
PyObject *result = _PyEval_EvalFrame(tstate, frame, exc);
assert(tstate->exc_info == prev_exc_info);
+#ifndef Py_GIL_DISABLED
assert(gen->gi_exc_state.previous_item == NULL);
- assert(gen->gi_frame_state != FRAME_EXECUTING);
assert(frame->previous == NULL);
+ assert(gen->gi_frame_state != FRAME_EXECUTING);
+#endif
+
+ // The generator_return_kind field is used to distinguish between a
+ // yield and a return from within _PyEval_EvalFrame(). Earlier versions
+ // of CPython (prior to 3.15) used gi_frame_state for this purpose, but
+ // that requires the GIL for thread-safety.
+ int return_kind = ((_PyThreadStateImpl *)tstate)->generator_return_kind;
+
+ if (return_kind == GENERATOR_YIELD) {
+ assert(result != NULL && !_PyErr_Occurred(tstate));
+ *presult = result;
+ return PYGEN_NEXT;
+ }
+
+ assert(return_kind == GENERATOR_RETURN);
+ assert(gen->gi_exc_state.exc_value == NULL);
+ assert(FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state) == FRAME_CLEARED);
/* If the generator just returned (as opposed to yielding), signal
* that the generator is exhausted. */
if (result) {
- if (FRAME_STATE_SUSPENDED(gen->gi_frame_state)) {
- *presult = result;
- return PYGEN_NEXT;
- }
assert(result == Py_None || !PyAsyncGen_CheckExact(gen));
if (result == Py_None && !PyAsyncGen_CheckExact(gen) && !arg) {
/* Return NULL if called by gen_iternext() */
@@ -281,37 +275,82 @@ gen_send_ex2(PyGenObject *gen, PyObject *arg, PyObject
**presult,
!PyErr_ExceptionMatches(PyExc_StopAsyncIteration));
}
- assert(gen->gi_exc_state.exc_value == NULL);
- assert(gen->gi_frame_state == FRAME_CLEARED);
*presult = result;
return result ? PYGEN_RETURN : PYGEN_ERROR;
}
+// Set the generator 'gen' to the executing state and send 'arg' into it.
+// See gen_send_ex2() for details.
+static PySendResult
+gen_send_ex(PyGenObject *gen, PyObject *arg, PyObject **presult)
+{
+ *presult = NULL;
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ do {
+ if (frame_state == FRAME_CREATED && arg && arg != Py_None) {
+ const char *msg = "can't send non-None value to a "
+ "just-started generator";
+ if (PyCoro_CheckExact(gen)) {
+ msg = NON_INIT_CORO_MSG;
+ }
+ else if (PyAsyncGen_CheckExact(gen)) {
+ msg = "can't send non-None value to a "
+ "just-started async generator";
+ }
+ PyErr_SetString(PyExc_TypeError, msg);
+ return PYGEN_ERROR;
+ }
+ if (frame_state == FRAME_EXECUTING) {
+ gen_raise_already_executing_error(gen);
+ return PYGEN_ERROR;
+ }
+ if (FRAME_STATE_FINISHED(frame_state)) {
+ if (PyCoro_CheckExact(gen)) {
+ /* `gen` is an exhausted coroutine: raise an error,
+ except when called from gen_close(), which should
+ always be a silent method. */
+ PyErr_SetString(
+ PyExc_RuntimeError,
+ "cannot reuse already awaited coroutine");
+ }
+ else if (arg) {
+ /* `gen` is an exhausted generator:
+ only return value if called from send(). */
+ *presult = Py_None;
+ return PYGEN_RETURN;
+ }
+ return PYGEN_ERROR;
+ }
+
+ assert((frame_state == FRAME_CREATED) ||
+ FRAME_STATE_SUSPENDED(frame_state));
+ } while (!_Py_GEN_TRY_SET_FRAME_STATE(gen, frame_state, FRAME_EXECUTING));
+
+ return gen_send_ex2(gen, arg, presult, 0);
+}
+
static PySendResult
PyGen_am_send(PyObject *self, PyObject *arg, PyObject **result)
{
PyGenObject *gen = _PyGen_CAST(self);
- return gen_send_ex2(gen, arg, result, 0, 0);
+ return gen_send_ex(gen, arg, result);
}
static PyObject *
-gen_send_ex(PyGenObject *gen, PyObject *arg, int exc, int closing)
+gen_set_stop_iteration(PyGenObject *gen, PyObject *result)
{
- PyObject *result;
- if (gen_send_ex2(gen, arg, &result, exc, closing) == PYGEN_RETURN) {
- if (PyAsyncGen_CheckExact(gen)) {
- assert(result == Py_None);
- PyErr_SetNone(PyExc_StopAsyncIteration);
- }
- else if (result == Py_None) {
- PyErr_SetNone(PyExc_StopIteration);
- }
- else {
- _PyGen_SetStopIterationValue(result);
- }
- Py_CLEAR(result);
+ if (PyAsyncGen_CheckExact(gen)) {
+ assert(result == Py_None);
+ PyErr_SetNone(PyExc_StopAsyncIteration);
}
- return result;
+ else if (result == Py_None) {
+ PyErr_SetNone(PyExc_StopIteration);
+ }
+ else {
+ _PyGen_SetStopIterationValue(result);
+ }
+ Py_DECREF(result);
+ return NULL;
}
PyDoc_STRVAR(send_doc,
@@ -319,9 +358,14 @@ PyDoc_STRVAR(send_doc,
return next yielded value or raise StopIteration.");
static PyObject *
-gen_send(PyObject *gen, PyObject *arg)
+gen_send(PyObject *op, PyObject *arg)
{
- return gen_send_ex((PyGenObject*)gen, arg, 0, 0);
+ PyObject *result;
+ PyGenObject *gen = _PyGen_CAST(op);
+ if (gen_send_ex(gen, arg, &result) == PYGEN_RETURN) {
+ return gen_set_stop_iteration(gen, result);
+ }
+ return result;
}
PyDoc_STRVAR(close_doc,
@@ -370,43 +414,43 @@ is_resume(_Py_CODEUNIT *instr)
);
}
-PyObject *
-_PyGen_yf(PyGenObject *gen)
-{
- if (gen->gi_frame_state == FRAME_SUSPENDED_YIELD_FROM) {
- _PyInterpreterFrame *frame = &gen->gi_iframe;
- // GH-122390: These asserts are wrong in the presence of
ENTER_EXECUTOR!
- // assert(is_resume(frame->instr_ptr));
- // assert((frame->instr_ptr->op.arg & RESUME_OPARG_LOCATION_MASK) >=
RESUME_AFTER_YIELD_FROM);
- return PyStackRef_AsPyObjectNew(_PyFrame_StackPeek(frame));
- }
- return NULL;
-}
-
static PyObject *
gen_close(PyObject *self, PyObject *args)
{
PyGenObject *gen = _PyGen_CAST(self);
- if (gen->gi_frame_state == FRAME_CREATED) {
- gen->gi_frame_state = FRAME_COMPLETED;
- gen_clear_frame(gen);
- Py_RETURN_NONE;
- }
- if (FRAME_STATE_FINISHED(gen->gi_frame_state)) {
- Py_RETURN_NONE;
- }
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ do {
+ if (frame_state == FRAME_CREATED) {
+ if (!_Py_GEN_TRY_SET_FRAME_STATE(gen, frame_state, FRAME_CLEARED))
{
+ continue;
+ }
+ gen_clear_frame(gen);
+ Py_RETURN_NONE;
+ }
+
+ if (FRAME_STATE_FINISHED(frame_state)) {
+ Py_RETURN_NONE;
+ }
+
+ if (frame_state == FRAME_EXECUTING) {
+ gen_raise_already_executing_error(gen);
+ return NULL;
+ }
+
+ assert(frame_state == FRAME_SUSPENDED_YIELD_FROM ||
+ frame_state == FRAME_SUSPENDED);
+
+ } while (!_Py_GEN_TRY_SET_FRAME_STATE(gen, frame_state, FRAME_EXECUTING));
- PyObject *yf = _PyGen_yf(gen);
int err = 0;
- if (yf) {
- PyFrameState state = gen->gi_frame_state;
- gen->gi_frame_state = FRAME_EXECUTING;
+ _PyInterpreterFrame *frame = &gen->gi_iframe;
+ if (frame_state == FRAME_SUSPENDED_YIELD_FROM) {
+ PyObject *yf = PyStackRef_AsPyObjectNew(_PyFrame_StackPeek(frame));
err = gen_close_iter(yf);
- gen->gi_frame_state = state;
Py_DECREF(yf);
}
- _PyInterpreterFrame *frame = &gen->gi_iframe;
+
if (is_resume(frame->instr_ptr)) {
bool no_unwind_tools = _PyEval_NoToolsForUnwind(_PyThreadState_GET());
/* We can safely ignore the outermost try block
@@ -416,7 +460,7 @@ gen_close(PyObject *self, PyObject *args)
if (oparg & RESUME_OPARG_DEPTH1_MASK && no_unwind_tools) {
// RESUME after YIELD_VALUE and exception depth is 1
assert((oparg & RESUME_OPARG_LOCATION_MASK) !=
RESUME_AT_FUNC_START);
- gen->gi_frame_state = FRAME_COMPLETED;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, FRAME_CLEARED);
gen_clear_frame(gen);
Py_RETURN_NONE;
}
@@ -425,8 +469,13 @@ gen_close(PyObject *self, PyObject *args)
PyErr_SetNone(PyExc_GeneratorExit);
}
- PyObject *retval = gen_send_ex(gen, Py_None, 1, 1);
- if (retval) {
+ PyObject *retval;
+ if (gen_send_ex2(gen, Py_None, &retval, 1) == PYGEN_RETURN) {
+ // the generator returned a value while closing, return the value here
+ assert(!PyErr_Occurred());
+ return retval;
+ }
+ else if (retval) {
const char *msg = "generator ignored GeneratorExit";
if (PyCoro_CheckExact(gen)) {
msg = "coroutine ignored GeneratorExit";
@@ -443,15 +492,80 @@ gen_close(PyObject *self, PyObject *args)
PyErr_Clear(); /* ignore this error */
Py_RETURN_NONE;
}
+ return NULL;
+}
- /* if the generator returned a value while closing, StopIteration was
- * raised in gen_send_ex() above; retrieve and return the value here */
- if (_PyGen_FetchStopIterationValue(&retval) == 0) {
- return retval;
+// Set an exception for a gen.throw() call.
+// Return 0 on success, -1 on failure.
+static int
+gen_set_exception(PyObject *typ, PyObject *val, PyObject *tb)
+{
+ /* First, check the traceback argument, replacing None with
+ NULL. */
+ if (tb == Py_None) {
+ tb = NULL;
}
- return NULL;
+ else if (tb != NULL && !PyTraceBack_Check(tb)) {
+ PyErr_SetString(PyExc_TypeError,
+ "throw() third argument must be a traceback object");
+ return -1;
+ }
+
+ Py_INCREF(typ);
+ Py_XINCREF(val);
+ Py_XINCREF(tb);
+
+ if (PyExceptionClass_Check(typ)) {
+ PyErr_NormalizeException(&typ, &val, &tb);
+ }
+ else if (PyExceptionInstance_Check(typ)) {
+ /* Raising an instance. The value should be a dummy. */
+ if (val && val != Py_None) {
+ PyErr_SetString(PyExc_TypeError,
+ "instance exception may not have a separate value");
+ goto failed_throw;
+ }
+ else {
+ /* Normalize to raise <class>, <instance> */
+ Py_XSETREF(val, typ);
+ typ = Py_NewRef(PyExceptionInstance_Class(typ));
+
+ if (tb == NULL)
+ /* Returns NULL if there's no traceback */
+ tb = PyException_GetTraceback(val);
+ }
+ }
+ else {
+ /* Not something you can raise. throw() fails. */
+ PyErr_Format(PyExc_TypeError,
+ "exceptions must be classes or instances "
+ "deriving from BaseException, not %s",
+ Py_TYPE(typ)->tp_name);
+ goto failed_throw;
+ }
+
+ PyErr_Restore(typ, val, tb);
+ return 0;
+
+failed_throw:
+ /* Didn't use our arguments, so restore their original refcounts */
+ Py_DECREF(typ);
+ Py_XDECREF(val);
+ Py_XDECREF(tb);
+ return -1;
}
+static PyObject *
+gen_throw_current_exception(PyGenObject *gen)
+{
+ assert(gen->gi_frame_state == FRAME_EXECUTING);
+
+ PyObject *result;
+ if (gen_send_ex2(gen, Py_None, &result, 1) == PYGEN_RETURN) {
+ return gen_set_stop_iteration(gen, result);
+ }
+ return result;
+}
PyDoc_STRVAR(throw_doc,
"throw(value)\n\
@@ -466,10 +580,32 @@ static PyObject *
_gen_throw(PyGenObject *gen, int close_on_genexit,
PyObject *typ, PyObject *val, PyObject *tb)
{
- PyObject *yf = _PyGen_yf(gen);
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ do {
+ if (frame_state == FRAME_EXECUTING) {
+ gen_raise_already_executing_error(gen);
+ return NULL;
+ }
- if (yf) {
+ if (FRAME_STATE_FINISHED(frame_state)) {
+ if (PyCoro_CheckExact(gen)) {
+ /* `gen` is an exhausted coroutine: raise an error */
+ PyErr_SetString(
+ PyExc_RuntimeError,
+ "cannot reuse already awaited coroutine");
+ return NULL;
+ }
+ gen_set_exception(typ, val, tb);
+ return NULL;
+ }
+
+ assert((frame_state == FRAME_CREATED) ||
+ FRAME_STATE_SUSPENDED(frame_state));
+ } while (!_Py_GEN_TRY_SET_FRAME_STATE(gen, frame_state, FRAME_EXECUTING));
+
+ if (frame_state == FRAME_SUSPENDED_YIELD_FROM) {
_PyInterpreterFrame *frame = &gen->gi_iframe;
+ PyObject *yf = PyStackRef_AsPyObjectNew(_PyFrame_StackPeek(frame));
PyObject *ret;
int err;
if (PyErr_GivenExceptionMatches(typ, PyExc_GeneratorExit) &&
@@ -479,13 +615,11 @@ _gen_throw(PyGenObject *gen, int close_on_genexit,
We have to allow some awaits to work it through, hence the
`close_on_genexit` parameter here.
*/
- PyFrameState state = gen->gi_frame_state;
- gen->gi_frame_state = FRAME_EXECUTING;
err = gen_close_iter(yf);
- gen->gi_frame_state = state;
Py_DECREF(yf);
- if (err < 0)
- return gen_send_ex(gen, Py_None, 1, 0);
+ if (err < 0) {
+ return gen_throw_current_exception(gen);
+ }
goto throw_here;
}
PyThreadState *tstate = _PyThreadState_GET();
@@ -501,18 +635,17 @@ _gen_throw(PyGenObject *gen, int close_on_genexit,
tstate->current_frame = frame;
/* Close the generator that we are currently iterating with
'yield from' or awaiting on with 'await'. */
- PyFrameState state = gen->gi_frame_state;
- gen->gi_frame_state = FRAME_EXECUTING;
ret = _gen_throw((PyGenObject *)yf, close_on_genexit,
typ, val, tb);
- gen->gi_frame_state = state;
tstate->current_frame = prev;
frame->previous = NULL;
- } else {
+ }
+ else {
/* `yf` is an iterator or a coroutine-like object. */
PyObject *meth;
if (PyObject_GetOptionalAttr(yf, &_Py_ID(throw), &meth) < 0) {
Py_DECREF(yf);
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, frame_state);
return NULL;
}
if (meth == NULL) {
@@ -523,75 +656,26 @@ _gen_throw(PyGenObject *gen, int close_on_genexit,
_PyInterpreterFrame *prev = tstate->current_frame;
frame->previous = prev;
tstate->current_frame = frame;
- PyFrameState state = gen->gi_frame_state;
- gen->gi_frame_state = FRAME_EXECUTING;
ret = PyObject_CallFunctionObjArgs(meth, typ, val, tb, NULL);
- gen->gi_frame_state = state;
tstate->current_frame = prev;
frame->previous = NULL;
Py_DECREF(meth);
}
Py_DECREF(yf);
if (!ret) {
- ret = gen_send_ex(gen, Py_None, 1, 0);
+ return gen_throw_current_exception(gen);
}
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, frame_state);
return ret;
}
throw_here:
- /* First, check the traceback argument, replacing None with
- NULL. */
- if (tb == Py_None) {
- tb = NULL;
- }
- else if (tb != NULL && !PyTraceBack_Check(tb)) {
- PyErr_SetString(PyExc_TypeError,
- "throw() third argument must be a traceback object");
+ assert(FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state) ==
FRAME_EXECUTING);
+ if (gen_set_exception(typ, val, tb) < 0) {
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, frame_state);
return NULL;
}
-
- Py_INCREF(typ);
- Py_XINCREF(val);
- Py_XINCREF(tb);
-
- if (PyExceptionClass_Check(typ))
- PyErr_NormalizeException(&typ, &val, &tb);
-
- else if (PyExceptionInstance_Check(typ)) {
- /* Raising an instance. The value should be a dummy. */
- if (val && val != Py_None) {
- PyErr_SetString(PyExc_TypeError,
- "instance exception may not have a separate value");
- goto failed_throw;
- }
- else {
- /* Normalize to raise <class>, <instance> */
- Py_XSETREF(val, typ);
- typ = Py_NewRef(PyExceptionInstance_Class(typ));
-
- if (tb == NULL)
- /* Returns NULL if there's no traceback */
- tb = PyException_GetTraceback(val);
- }
- }
- else {
- /* Not something you can raise. throw() fails. */
- PyErr_Format(PyExc_TypeError,
- "exceptions must be classes or instances "
- "deriving from BaseException, not %s",
- Py_TYPE(typ)->tp_name);
- goto failed_throw;
- }
-
- PyErr_Restore(typ, val, tb);
- return gen_send_ex(gen, Py_None, 1, 0);
-
-failed_throw:
- /* Didn't use our arguments, so restore their original refcounts */
- Py_DECREF(typ);
- Py_XDECREF(val);
- Py_XDECREF(tb);
- return NULL;
+ return gen_throw_current_exception(gen);
}
@@ -633,7 +717,7 @@ gen_iternext(PyObject *self)
PyGenObject *gen = _PyGen_CAST(self);
PyObject *result;
- if (gen_send_ex2(gen, NULL, &result, 0, 0) == PYGEN_RETURN) {
+ if (gen_send_ex(gen, NULL, &result) == PYGEN_RETURN) {
if (result != Py_None) {
_PyGen_SetStopIterationValue(result);
}
@@ -757,13 +841,15 @@ gen_set_qualname(PyObject *self, PyObject *value, void
*Py_UNUSED(ignored))
}
static PyObject *
-gen_getyieldfrom(PyObject *gen, void *Py_UNUSED(ignored))
+gen_getyieldfrom(PyObject *self, void *Py_UNUSED(ignored))
{
- PyObject *yf = _PyGen_yf(_PyGen_CAST(gen));
- if (yf == NULL) {
+ PyGenObject *gen = _PyGen_CAST(self);
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ if (frame_state != FRAME_SUSPENDED_YIELD_FROM) {
Py_RETURN_NONE;
}
- return yf;
+ // TODO: still not thread-safe with free threading
+ return PyStackRef_AsPyObjectNew(_PyFrame_StackPeek(&gen->gi_iframe));
}
@@ -771,17 +857,16 @@ static PyObject *
gen_getrunning(PyObject *self, void *Py_UNUSED(ignored))
{
PyGenObject *gen = _PyGen_CAST(self);
- if (gen->gi_frame_state == FRAME_EXECUTING) {
- Py_RETURN_TRUE;
- }
- Py_RETURN_FALSE;
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ return frame_state == FRAME_EXECUTING ? Py_True : Py_False;
}
static PyObject *
gen_getsuspended(PyObject *self, void *Py_UNUSED(ignored))
{
PyGenObject *gen = _PyGen_CAST(self);
- return PyBool_FromLong(FRAME_STATE_SUSPENDED(gen->gi_frame_state));
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ return FRAME_STATE_SUSPENDED(frame_state) ? Py_True : Py_False;
}
static PyObject *
@@ -790,9 +875,11 @@ _gen_getframe(PyGenObject *gen, const char *const name)
if (PySys_Audit("object.__getattr__", "Os", gen, name) < 0) {
return NULL;
}
- if (FRAME_STATE_FINISHED(gen->gi_frame_state)) {
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(gen->gi_frame_state);
+ if (FRAME_STATE_FINISHED(frame_state)) {
Py_RETURN_NONE;
}
+ // TODO: still not thread-safe with free threading
return _Py_XNewRef((PyObject *)_PyFrame_GetFrameObject(&gen->gi_iframe));
}
@@ -1135,35 +1222,6 @@ coro_await(PyObject *coro)
return (PyObject *)cw;
}
-static PyObject *
-coro_get_cr_await(PyObject *coro, void *Py_UNUSED(ignored))
-{
- PyObject *yf = _PyGen_yf((PyGenObject *) coro);
- if (yf == NULL)
- Py_RETURN_NONE;
- return yf;
-}
-
-static PyObject *
-cr_getsuspended(PyObject *self, void *Py_UNUSED(ignored))
-{
- PyCoroObject *coro = _PyCoroObject_CAST(self);
- if (FRAME_STATE_SUSPENDED(coro->cr_frame_state)) {
- Py_RETURN_TRUE;
- }
- Py_RETURN_FALSE;
-}
-
-static PyObject *
-cr_getrunning(PyObject *self, void *Py_UNUSED(ignored))
-{
- PyCoroObject *coro = _PyCoroObject_CAST(self);
- if (coro->cr_frame_state == FRAME_EXECUTING) {
- Py_RETURN_TRUE;
- }
- Py_RETURN_FALSE;
-}
-
static PyObject *
cr_getframe(PyObject *coro, void *Py_UNUSED(ignored))
{
@@ -1181,12 +1239,12 @@ static PyGetSetDef coro_getsetlist[] = {
PyDoc_STR("name of the coroutine")},
{"__qualname__", gen_get_qualname, gen_set_qualname,
PyDoc_STR("qualified name of the coroutine")},
- {"cr_await", coro_get_cr_await, NULL,
+ {"cr_await", gen_getyieldfrom, NULL,
PyDoc_STR("object being awaited on, or None")},
- {"cr_running", cr_getrunning, NULL, NULL},
+ {"cr_running", gen_getrunning, NULL, NULL},
{"cr_frame", cr_getframe, NULL, NULL},
{"cr_code", cr_getcode, NULL, NULL},
- {"cr_suspended", cr_getsuspended, NULL, NULL},
+ {"cr_suspended", gen_getsuspended, NULL, NULL},
{NULL} /* Sentinel */
};
@@ -1602,26 +1660,16 @@ ag_getcode(PyObject *gen, void *Py_UNUSED(ignored))
return _gen_getcode((PyGenObject*)gen, "ag_code");
}
-static PyObject *
-ag_getsuspended(PyObject *self, void *Py_UNUSED(ignored))
-{
- PyAsyncGenObject *ag = _PyAsyncGenObject_CAST(self);
- if (FRAME_STATE_SUSPENDED(ag->ag_frame_state)) {
- Py_RETURN_TRUE;
- }
- Py_RETURN_FALSE;
-}
-
static PyGetSetDef async_gen_getsetlist[] = {
{"__name__", gen_get_name, gen_set_name,
PyDoc_STR("name of the async generator")},
{"__qualname__", gen_get_qualname, gen_set_qualname,
PyDoc_STR("qualified name of the async generator")},
- {"ag_await", coro_get_cr_await, NULL,
+ {"ag_await", gen_getyieldfrom, NULL,
PyDoc_STR("object being awaited on, or None")},
{"ag_frame", ag_getframe, NULL, NULL},
{"ag_code", ag_getcode, NULL, NULL},
- {"ag_suspended", ag_getsuspended, NULL, NULL},
+ {"ag_suspended", gen_getsuspended, NULL, NULL},
{NULL} /* Sentinel */
};
diff --git a/Python/bytecodes.c b/Python/bytecodes.c
index 1291fe56a59d5b..f7eb006e686800 100644
--- a/Python/bytecodes.c
+++ b/Python/bytecodes.c
@@ -1312,14 +1312,13 @@ dummy_func(
assert(frame->owner != FRAME_OWNED_BY_INTERPRETER);
if ((tstate->interp->eval_frame == NULL) &&
(Py_TYPE(receiver_o) == &PyGen_Type || Py_TYPE(receiver_o) ==
&PyCoro_Type) &&
- ((PyGenObject *)receiver_o)->gi_frame_state < FRAME_EXECUTING)
+ gen_try_set_executing((PyGenObject *)receiver_o))
{
PyGenObject *gen = (PyGenObject *)receiver_o;
_PyInterpreterFrame *gen_frame = &gen->gi_iframe;
_PyFrame_StackPush(gen_frame, PyStackRef_MakeHeapSafe(v));
DEAD(v);
SYNC_SP();
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
assert(INSTRUCTION_SIZE + oparg <= UINT16_MAX);
@@ -1360,12 +1359,11 @@ dummy_func(
op(_SEND_GEN_FRAME, (receiver, v -- receiver, gen_frame)) {
PyGenObject *gen = (PyGenObject
*)PyStackRef_AsPyObjectBorrow(receiver);
DEOPT_IF(Py_TYPE(gen) != &PyGen_Type && Py_TYPE(gen) !=
&PyCoro_Type);
- DEOPT_IF(gen->gi_frame_state >= FRAME_EXECUTING);
+ DEOPT_IF(!gen_try_set_executing((PyGenObject *)gen));
STAT_INC(SEND, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_MakeHeapSafe(v));
DEAD(v);
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
assert(INSTRUCTION_SIZE + oparg <= UINT16_MAX);
@@ -1389,7 +1387,6 @@ dummy_func(
PyGenObject *gen = _PyGen_GetGeneratorFromFrame(frame);
assert(FRAME_SUSPENDED_YIELD_FROM == FRAME_SUSPENDED + 1);
assert(oparg == 0 || oparg == 1);
- gen->gi_frame_state = FRAME_SUSPENDED + oparg;
_PyStackRef temp = retval;
DEAD(retval);
SAVE_STACK();
@@ -1399,6 +1396,8 @@ dummy_func(
_PyInterpreterFrame *gen_frame = frame;
frame = tstate->current_frame = frame->previous;
gen_frame->previous = NULL;
+ ((_PyThreadStateImpl *)tstate)->generator_return_kind =
GENERATOR_YIELD;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, FRAME_SUSPENDED
+ oparg);
/* We don't know which of these is relevant here, so keep them
equal */
assert(INLINE_CACHE_ENTRIES_SEND == INLINE_CACHE_ENTRIES_FOR_ITER);
#if TIER_ONE
@@ -3405,18 +3404,10 @@ dummy_func(
op(_FOR_ITER_GEN_FRAME, (iter, null -- iter, null, gen_frame)) {
PyGenObject *gen = (PyGenObject
*)PyStackRef_AsPyObjectBorrow(iter);
DEOPT_IF(Py_TYPE(gen) != &PyGen_Type);
-#ifdef Py_GIL_DISABLED
- // Since generators can't be used by multiple threads anyway we
- // don't need to deopt here, but this lets us work on making
- // generators thread-safe without necessarily having to
- // specialize them thread-safely as well.
- DEOPT_IF(!_PyObject_IsUniquelyReferenced((PyObject *)gen));
-#endif
- DEOPT_IF(gen->gi_frame_state >= FRAME_EXECUTING);
+ DEOPT_IF(!gen_try_set_executing((PyGenObject *)gen));
STAT_INC(FOR_ITER, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_None);
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
pushed_frame->previous = frame;
diff --git a/Python/ceval.c b/Python/ceval.c
index cf86d5484f0d6e..ec21d6bc2b852c 100644
--- a/Python/ceval.c
+++ b/Python/ceval.c
@@ -2304,7 +2304,8 @@ clear_gen_frame(PyThreadState *tstate,
_PyInterpreterFrame * frame)
{
assert(frame->owner == FRAME_OWNED_BY_GENERATOR);
PyGenObject *gen = _PyGen_GetGeneratorFromFrame(frame);
- gen->gi_frame_state = FRAME_CLEARED;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, FRAME_CLEARED);
+ ((_PyThreadStateImpl *)tstate)->generator_return_kind = GENERATOR_RETURN;
assert(tstate->exc_info == &gen->gi_exc_state);
tstate->exc_info = gen->gi_exc_state.previous_item;
gen->gi_exc_state.previous_item = NULL;
@@ -3979,15 +3980,13 @@ _PyEval_GetAwaitable(PyObject *iterable, int oparg)
Py_TYPE(iterable), oparg);
}
else if (PyCoro_CheckExact(iter)) {
- PyObject *yf = _PyGen_yf((PyGenObject*)iter);
- if (yf != NULL) {
- /* `iter` is a coroutine object that is being
- awaited, `yf` is a pointer to the current awaitable
- being awaited on. */
- Py_DECREF(yf);
+ PyCoroObject *coro = (PyCoroObject *)iter;
+ int8_t frame_state = FT_ATOMIC_LOAD_INT8_RELAXED(coro->cr_frame_state);
+ if (frame_state == FRAME_SUSPENDED_YIELD_FROM) {
+ /* `iter` is a coroutine object that is being awaited. */
Py_CLEAR(iter);
_PyErr_SetString(PyThreadState_GET(), PyExc_RuntimeError,
- "coroutine is being awaited already");
+ "coroutine is being awaited already");
}
}
return iter;
diff --git a/Python/ceval_macros.h b/Python/ceval_macros.h
index a526a453dd918a..c9472ec16d423e 100644
--- a/Python/ceval_macros.h
+++ b/Python/ceval_macros.h
@@ -496,3 +496,28 @@ check_periodics(PyThreadState *tstate) {
return 0;
}
+// Mark the generator as executing. Returns true if the state was changed,
+// false if it was already executing or finished.
+static inline bool
+gen_try_set_executing(PyGenObject *gen)
+{
+#ifdef Py_GIL_DISABLED
+ if (!_PyObject_IsUniquelyReferenced((PyObject *)gen)) {
+ int8_t frame_state =
_Py_atomic_load_int8_relaxed(&gen->gi_frame_state);
+ while (frame_state < FRAME_EXECUTING) {
+ if (_Py_atomic_compare_exchange_int8(&gen->gi_frame_state,
+ &frame_state,
+ FRAME_EXECUTING)) {
+ return true;
+ }
+ }
+ }
+#endif
+ // Use faster non-atomic modifications in the GIL-enabled build and when
+ // the object is uniquely referenced in the free-threaded build.
+ if (gen->gi_frame_state < FRAME_EXECUTING) {
+ gen->gi_frame_state = FRAME_EXECUTING;
+ return true;
+ }
+ return false;
+}
diff --git a/Python/executor_cases.c.h b/Python/executor_cases.c.h
index 2305df6ad5aaf1..b3eff63b30ab55 100644
--- a/Python/executor_cases.c.h
+++ b/Python/executor_cases.c.h
@@ -5823,7 +5823,7 @@
SET_CURRENT_CACHED_VALUES(2);
JUMP_TO_JUMP_TARGET();
}
- if (gen->gi_frame_state >= FRAME_EXECUTING) {
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
UOP_STAT_INC(uopcode, miss);
_tos_cache1 = v;
_tos_cache0 = receiver;
@@ -5833,7 +5833,6 @@
STAT_INC(SEND, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_MakeHeapSafe(v));
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
assert( 2u + oparg <= UINT16_MAX);
@@ -5861,7 +5860,6 @@
PyGenObject *gen = _PyGen_GetGeneratorFromFrame(frame);
assert(FRAME_SUSPENDED_YIELD_FROM == FRAME_SUSPENDED + 1);
assert(oparg == 0 || oparg == 1);
- gen->gi_frame_state = FRAME_SUSPENDED + oparg;
_PyStackRef temp = retval;
_PyFrame_SetStackPointer(frame, stack_pointer);
tstate->exc_info = gen->gi_exc_state.previous_item;
@@ -5870,6 +5868,8 @@
_PyInterpreterFrame *gen_frame = frame;
frame = tstate->current_frame = frame->previous;
gen_frame->previous = NULL;
+ ((_PyThreadStateImpl *)tstate)->generator_return_kind =
GENERATOR_YIELD;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, FRAME_SUSPENDED
+ oparg);
assert(INLINE_CACHE_ENTRIES_SEND == INLINE_CACHE_ENTRIES_FOR_ITER);
#if TIER_ONE
assert(frame->instr_ptr->op.code == INSTRUMENTED_LINE ||
@@ -10859,34 +10859,99 @@
break;
}
- case _FOR_ITER_GEN_FRAME_r23: {
- CHECK_CURRENT_CACHED_VALUES(2);
+ case _FOR_ITER_GEN_FRAME_r03: {
+ CHECK_CURRENT_CACHED_VALUES(0);
+ assert(WITHIN_STACK_BOUNDS_IGNORING_CACHE());
+ _PyStackRef iter;
+ _PyStackRef gen_frame;
+ oparg = CURRENT_OPARG();
+ iter = stack_pointer[-2];
+ PyGenObject *gen = (PyGenObject
*)PyStackRef_AsPyObjectBorrow(iter);
+ if (Py_TYPE(gen) != &PyGen_Type) {
+ UOP_STAT_INC(uopcode, miss);
+ SET_CURRENT_CACHED_VALUES(0);
+ JUMP_TO_JUMP_TARGET();
+ }
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
+ UOP_STAT_INC(uopcode, miss);
+ SET_CURRENT_CACHED_VALUES(0);
+ JUMP_TO_JUMP_TARGET();
+ }
+ STAT_INC(FOR_ITER, hit);
+ _PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
+ _PyFrame_StackPush(pushed_frame, PyStackRef_None);
+ gen->gi_exc_state.previous_item = tstate->exc_info;
+ tstate->exc_info = &gen->gi_exc_state;
+ pushed_frame->previous = frame;
+ frame->return_offset = (uint16_t)( 2u + oparg);
+ gen_frame = PyStackRef_Wrap(pushed_frame);
+ _tos_cache2 = gen_frame;
+ _tos_cache1 = stack_pointer[-1];
+ _tos_cache0 = iter;
+ SET_CURRENT_CACHED_VALUES(3);
+ stack_pointer += -2;
+ ASSERT_WITHIN_STACK_BOUNDS(__FILE__, __LINE__);
+ assert(WITHIN_STACK_BOUNDS_IGNORING_CACHE());
+ break;
+ }
+
+ case _FOR_ITER_GEN_FRAME_r13: {
+ CHECK_CURRENT_CACHED_VALUES(1);
assert(WITHIN_STACK_BOUNDS_IGNORING_CACHE());
_PyStackRef iter;
_PyStackRef gen_frame;
_PyStackRef _stack_item_0 = _tos_cache0;
- _PyStackRef _stack_item_1 = _tos_cache1;
oparg = CURRENT_OPARG();
- iter = _stack_item_0;
+ iter = stack_pointer[-1];
PyGenObject *gen = (PyGenObject
*)PyStackRef_AsPyObjectBorrow(iter);
if (Py_TYPE(gen) != &PyGen_Type) {
UOP_STAT_INC(uopcode, miss);
- _tos_cache1 = _stack_item_1;
- _tos_cache0 = iter;
- SET_CURRENT_CACHED_VALUES(2);
+ _tos_cache0 = _stack_item_0;
+ SET_CURRENT_CACHED_VALUES(1);
JUMP_TO_JUMP_TARGET();
}
- #ifdef Py_GIL_DISABLED
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
+ UOP_STAT_INC(uopcode, miss);
+ _tos_cache0 = _stack_item_0;
+ SET_CURRENT_CACHED_VALUES(1);
+ JUMP_TO_JUMP_TARGET();
+ }
+ STAT_INC(FOR_ITER, hit);
+ _PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
+ _PyFrame_StackPush(pushed_frame, PyStackRef_None);
+ gen->gi_exc_state.previous_item = tstate->exc_info;
+ tstate->exc_info = &gen->gi_exc_state;
+ pushed_frame->previous = frame;
+ frame->return_offset = (uint16_t)( 2u + oparg);
+ gen_frame = PyStackRef_Wrap(pushed_frame);
+ _tos_cache2 = gen_frame;
+ _tos_cache1 = _stack_item_0;
+ _tos_cache0 = iter;
+ SET_CURRENT_CACHED_VALUES(3);
+ stack_pointer += -1;
+ ASSERT_WITHIN_STACK_BOUNDS(__FILE__, __LINE__);
+ assert(WITHIN_STACK_BOUNDS_IGNORING_CACHE());
+ break;
+ }
- if (!_PyObject_IsUniquelyReferenced((PyObject *)gen)) {
+ case _FOR_ITER_GEN_FRAME_r23: {
+ CHECK_CURRENT_CACHED_VALUES(2);
+ assert(WITHIN_STACK_BOUNDS_IGNORING_CACHE());
+ _PyStackRef iter;
+ _PyStackRef gen_frame;
+ _PyStackRef _stack_item_0 = _tos_cache0;
+ _PyStackRef _stack_item_1 = _tos_cache1;
+ oparg = CURRENT_OPARG();
+ iter = _stack_item_0;
+ PyGenObject *gen = (PyGenObject
*)PyStackRef_AsPyObjectBorrow(iter);
+ if (Py_TYPE(gen) != &PyGen_Type) {
UOP_STAT_INC(uopcode, miss);
_tos_cache1 = _stack_item_1;
_tos_cache0 = iter;
SET_CURRENT_CACHED_VALUES(2);
JUMP_TO_JUMP_TARGET();
}
- #endif
- if (gen->gi_frame_state >= FRAME_EXECUTING) {
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
UOP_STAT_INC(uopcode, miss);
_tos_cache1 = _stack_item_1;
_tos_cache0 = iter;
@@ -10896,7 +10961,6 @@
STAT_INC(FOR_ITER, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_None);
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
pushed_frame->previous = frame;
diff --git a/Python/generated_cases.c.h b/Python/generated_cases.c.h
index 4482bb08a132f9..eaaa5f3bb96abc 100644
--- a/Python/generated_cases.c.h
+++ b/Python/generated_cases.c.h
@@ -5548,14 +5548,7 @@
assert(_PyOpcode_Deopt[opcode] == (FOR_ITER));
JUMP_TO_PREDICTED(FOR_ITER);
}
- #ifdef Py_GIL_DISABLED
- if (!_PyObject_IsUniquelyReferenced((PyObject *)gen)) {
- UPDATE_MISS_STATS(FOR_ITER);
- assert(_PyOpcode_Deopt[opcode] == (FOR_ITER));
- JUMP_TO_PREDICTED(FOR_ITER);
- }
- #endif
- if (gen->gi_frame_state >= FRAME_EXECUTING) {
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
UPDATE_MISS_STATS(FOR_ITER);
assert(_PyOpcode_Deopt[opcode] == (FOR_ITER));
JUMP_TO_PREDICTED(FOR_ITER);
@@ -5563,7 +5556,6 @@
STAT_INC(FOR_ITER, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_None);
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
pushed_frame->previous = frame;
@@ -7327,7 +7319,6 @@
PyGenObject *gen = _PyGen_GetGeneratorFromFrame(frame);
assert(FRAME_SUSPENDED_YIELD_FROM == FRAME_SUSPENDED + 1);
assert(oparg == 0 || oparg == 1);
- gen->gi_frame_state = FRAME_SUSPENDED + oparg;
_PyStackRef temp = retval;
stack_pointer += -1;
ASSERT_WITHIN_STACK_BOUNDS(__FILE__, __LINE__);
@@ -7338,6 +7329,8 @@
_PyInterpreterFrame *gen_frame = frame;
frame = tstate->current_frame = frame->previous;
gen_frame->previous = NULL;
+ ((_PyThreadStateImpl *)tstate)->generator_return_kind =
GENERATOR_YIELD;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state,
FRAME_SUSPENDED + oparg);
assert(INLINE_CACHE_ENTRIES_SEND ==
INLINE_CACHE_ENTRIES_FOR_ITER);
#if TIER_ONE
assert(frame->instr_ptr->op.code == INSTRUMENTED_LINE ||
@@ -10301,14 +10294,13 @@
assert(frame->owner != FRAME_OWNED_BY_INTERPRETER);
if ((tstate->interp->eval_frame == NULL) &&
(Py_TYPE(receiver_o) == &PyGen_Type || Py_TYPE(receiver_o)
== &PyCoro_Type) &&
- ((PyGenObject *)receiver_o)->gi_frame_state <
FRAME_EXECUTING)
+ gen_try_set_executing((PyGenObject *)receiver_o))
{
PyGenObject *gen = (PyGenObject *)receiver_o;
_PyInterpreterFrame *gen_frame = &gen->gi_iframe;
_PyFrame_StackPush(gen_frame, PyStackRef_MakeHeapSafe(v));
stack_pointer += -1;
ASSERT_WITHIN_STACK_BOUNDS(__FILE__, __LINE__);
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
assert( 2u + oparg <= UINT16_MAX);
@@ -10401,7 +10393,7 @@
assert(_PyOpcode_Deopt[opcode] == (SEND));
JUMP_TO_PREDICTED(SEND);
}
- if (gen->gi_frame_state >= FRAME_EXECUTING) {
+ if (!gen_try_set_executing((PyGenObject *)gen)) {
UPDATE_MISS_STATS(SEND);
assert(_PyOpcode_Deopt[opcode] == (SEND));
JUMP_TO_PREDICTED(SEND);
@@ -10409,7 +10401,6 @@
STAT_INC(SEND, hit);
_PyInterpreterFrame *pushed_frame = &gen->gi_iframe;
_PyFrame_StackPush(pushed_frame, PyStackRef_MakeHeapSafe(v));
- gen->gi_frame_state = FRAME_EXECUTING;
gen->gi_exc_state.previous_item = tstate->exc_info;
tstate->exc_info = &gen->gi_exc_state;
assert( 2u + oparg <= UINT16_MAX);
@@ -12045,7 +12036,6 @@
PyGenObject *gen = _PyGen_GetGeneratorFromFrame(frame);
assert(FRAME_SUSPENDED_YIELD_FROM == FRAME_SUSPENDED + 1);
assert(oparg == 0 || oparg == 1);
- gen->gi_frame_state = FRAME_SUSPENDED + oparg;
_PyStackRef temp = retval;
stack_pointer += -1;
ASSERT_WITHIN_STACK_BOUNDS(__FILE__, __LINE__);
@@ -12056,6 +12046,8 @@
_PyInterpreterFrame *gen_frame = frame;
frame = tstate->current_frame = frame->previous;
gen_frame->previous = NULL;
+ ((_PyThreadStateImpl *)tstate)->generator_return_kind =
GENERATOR_YIELD;
+ FT_ATOMIC_STORE_INT8_RELEASE(gen->gi_frame_state, FRAME_SUSPENDED
+ oparg);
assert(INLINE_CACHE_ENTRIES_SEND == INLINE_CACHE_ENTRIES_FOR_ITER);
#if TIER_ONE
assert(frame->instr_ptr->op.code == INSTRUMENTED_LINE ||
diff --git a/Tools/cases_generator/analyzer.py
b/Tools/cases_generator/analyzer.py
index fcd0dcf12acb2b..659befe312afaf 100644
--- a/Tools/cases_generator/analyzer.py
+++ b/Tools/cases_generator/analyzer.py
@@ -642,6 +642,7 @@ def has_error_without_pop(op: parser.CodeDef) -> bool:
"_PyFrame_StackPush",
"_PyFunction_SetVersion",
"_PyGen_GetGeneratorFromFrame",
+ "gen_try_set_executing",
"_PyInterpreterState_GET",
"_PyList_AppendTakeRef",
"_PyList_ITEMS",
_______________________________________________
Python-checkins mailing list -- [email protected]
To unsubscribe send an email to [email protected]
https://mail.python.org/mailman3//lists/python-checkins.python.org
Member address: [email protected]