Skip to content

Commit 78a530a

Browse files
authored
gh-115999: Add free-threaded specialization for TO_BOOL (gh-126616)
1 parent 09c240f commit 78a530a

File tree

7 files changed

+168
-69
lines changed

7 files changed

+168
-69
lines changed

Include/internal/pycore_typeobject.h

+10
Original file line numberDiff line numberDiff line change
@@ -269,6 +269,16 @@ extern unsigned int _PyType_GetVersionForCurrentState(PyTypeObject *tp);
269269
PyAPI_FUNC(void) _PyType_SetVersion(PyTypeObject *tp, unsigned int version);
270270
PyTypeObject *_PyType_LookupByVersion(unsigned int version);
271271

272+
// Function pointer type for user-defined validation function that will be
273+
// called by _PyType_Validate().
274+
// It should return 0 if the validation is passed, otherwise it will return -1.
275+
typedef int (*_py_validate_type)(PyTypeObject *);
276+
277+
// It will verify the ``ty`` through user-defined validation function ``validate``,
278+
// and if the validation is passed, it will set the ``tp_version`` as valid
279+
// tp_version_tag from the ``ty``.
280+
extern int _PyType_Validate(PyTypeObject *ty, _py_validate_type validate, unsigned int *tp_version);
281+
272282
#ifdef __cplusplus
273283
}
274284
#endif

Lib/test/test_opcache.py

+66
Original file line numberDiff line numberDiff line change
@@ -1272,6 +1272,72 @@ def g():
12721272
self.assert_specialized(g, "CONTAINS_OP_SET")
12731273
self.assert_no_opcode(g, "CONTAINS_OP")
12741274

1275+
@cpython_only
1276+
@requires_specialization_ft
1277+
def test_to_bool(self):
1278+
def to_bool_bool():
1279+
true_cnt, false_cnt = 0, 0
1280+
elems = [e % 2 == 0 for e in range(100)]
1281+
for e in elems:
1282+
if e:
1283+
true_cnt += 1
1284+
else:
1285+
false_cnt += 1
1286+
self.assertEqual(true_cnt, 50)
1287+
self.assertEqual(false_cnt, 50)
1288+
1289+
to_bool_bool()
1290+
self.assert_specialized(to_bool_bool, "TO_BOOL_BOOL")
1291+
self.assert_no_opcode(to_bool_bool, "TO_BOOL")
1292+
1293+
def to_bool_int():
1294+
count = 0
1295+
for i in range(100):
1296+
if i:
1297+
count += 1
1298+
else:
1299+
count -= 1
1300+
self.assertEqual(count, 98)
1301+
1302+
to_bool_int()
1303+
self.assert_specialized(to_bool_int, "TO_BOOL_INT")
1304+
self.assert_no_opcode(to_bool_int, "TO_BOOL")
1305+
1306+
def to_bool_list():
1307+
count = 0
1308+
elems = [1, 2, 3]
1309+
while elems:
1310+
count += elems.pop()
1311+
self.assertEqual(elems, [])
1312+
self.assertEqual(count, 6)
1313+
1314+
to_bool_list()
1315+
self.assert_specialized(to_bool_list, "TO_BOOL_LIST")
1316+
self.assert_no_opcode(to_bool_list, "TO_BOOL")
1317+
1318+
def to_bool_none():
1319+
count = 0
1320+
elems = [None, None, None, None]
1321+
for e in elems:
1322+
if not e:
1323+
count += 1
1324+
self.assertEqual(count, len(elems))
1325+
1326+
to_bool_none()
1327+
self.assert_specialized(to_bool_none, "TO_BOOL_NONE")
1328+
self.assert_no_opcode(to_bool_none, "TO_BOOL")
1329+
1330+
def to_bool_str():
1331+
count = 0
1332+
elems = ["", "foo", ""]
1333+
for e in elems:
1334+
if e:
1335+
count += 1
1336+
self.assertEqual(count, 1)
1337+
1338+
to_bool_str()
1339+
self.assert_specialized(to_bool_str, "TO_BOOL_STR")
1340+
self.assert_no_opcode(to_bool_str, "TO_BOOL")
12751341

12761342

12771343
if __name__ == "__main__":

Objects/typeobject.c

+18
Original file line numberDiff line numberDiff line change
@@ -5645,6 +5645,24 @@ _PyType_SetFlags(PyTypeObject *self, unsigned long mask, unsigned long flags)
56455645
END_TYPE_LOCK();
56465646
}
56475647

5648+
int
5649+
_PyType_Validate(PyTypeObject *ty, _py_validate_type validate, unsigned int *tp_version)
5650+
{
5651+
int err;
5652+
BEGIN_TYPE_LOCK();
5653+
err = validate(ty);
5654+
if (!err) {
5655+
if(assign_version_tag(_PyInterpreterState_GET(), ty)) {
5656+
*tp_version = ty->tp_version_tag;
5657+
}
5658+
else {
5659+
err = -1;
5660+
}
5661+
}
5662+
END_TYPE_LOCK();
5663+
return err;
5664+
}
5665+
56485666
static void
56495667
set_flags_recursive(PyTypeObject *self, unsigned long mask, unsigned long flags)
56505668
{

Python/bytecodes.c

+3-3
Original file line numberDiff line numberDiff line change
@@ -391,15 +391,15 @@ dummy_func(
391391
};
392392

393393
specializing op(_SPECIALIZE_TO_BOOL, (counter/1, value -- value)) {
394-
#if ENABLE_SPECIALIZATION
394+
#if ENABLE_SPECIALIZATION_FT
395395
if (ADAPTIVE_COUNTER_TRIGGERS(counter)) {
396396
next_instr = this_instr;
397397
_Py_Specialize_ToBool(value, next_instr);
398398
DISPATCH_SAME_OPARG();
399399
}
400400
OPCODE_DEFERRED_INC(TO_BOOL);
401401
ADVANCE_ADAPTIVE_COUNTER(this_instr[1].counter);
402-
#endif /* ENABLE_SPECIALIZATION */
402+
#endif /* ENABLE_SPECIALIZATION_FT */
403403
}
404404

405405
op(_TO_BOOL, (value -- res)) {
@@ -435,7 +435,7 @@ dummy_func(
435435
PyObject *value_o = PyStackRef_AsPyObjectBorrow(value);
436436
EXIT_IF(!PyList_CheckExact(value_o));
437437
STAT_INC(TO_BOOL, hit);
438-
res = Py_SIZE(value_o) ? PyStackRef_True : PyStackRef_False;
438+
res = PyList_GET_SIZE(value_o) ? PyStackRef_True : PyStackRef_False;
439439
DECREF_INPUTS();
440440
}
441441

Python/executor_cases.c.h

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/generated_cases.c.h

+3-3
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Python/specialize.c

+67-62
Original file line numberDiff line numberDiff line change
@@ -2667,101 +2667,106 @@ _Py_Specialize_Send(_PyStackRef receiver_st, _Py_CODEUNIT *instr)
26672667
cache->counter = adaptive_counter_cooldown();
26682668
}
26692669

2670+
#ifdef Py_STATS
2671+
static int
2672+
to_bool_fail_kind(PyObject *value)
2673+
{
2674+
if (PyByteArray_CheckExact(value)) {
2675+
return SPEC_FAIL_TO_BOOL_BYTEARRAY;
2676+
}
2677+
if (PyBytes_CheckExact(value)) {
2678+
return SPEC_FAIL_TO_BOOL_BYTES;
2679+
}
2680+
if (PyDict_CheckExact(value)) {
2681+
return SPEC_FAIL_TO_BOOL_DICT;
2682+
}
2683+
if (PyFloat_CheckExact(value)) {
2684+
return SPEC_FAIL_TO_BOOL_FLOAT;
2685+
}
2686+
if (PyMemoryView_Check(value)) {
2687+
return SPEC_FAIL_TO_BOOL_MEMORY_VIEW;
2688+
}
2689+
if (PyAnySet_CheckExact(value)) {
2690+
return SPEC_FAIL_TO_BOOL_SET;
2691+
}
2692+
if (PyTuple_CheckExact(value)) {
2693+
return SPEC_FAIL_TO_BOOL_TUPLE;
2694+
}
2695+
return SPEC_FAIL_OTHER;
2696+
}
2697+
#endif // Py_STATS
2698+
2699+
static int
2700+
check_type_always_true(PyTypeObject *ty)
2701+
{
2702+
PyNumberMethods *nb = ty->tp_as_number;
2703+
if (nb && nb->nb_bool) {
2704+
return SPEC_FAIL_TO_BOOL_NUMBER;
2705+
}
2706+
PyMappingMethods *mp = ty->tp_as_mapping;
2707+
if (mp && mp->mp_length) {
2708+
return SPEC_FAIL_TO_BOOL_MAPPING;
2709+
}
2710+
PySequenceMethods *sq = ty->tp_as_sequence;
2711+
if (sq && sq->sq_length) {
2712+
return SPEC_FAIL_TO_BOOL_SEQUENCE;
2713+
}
2714+
return 0;
2715+
}
2716+
26702717
void
26712718
_Py_Specialize_ToBool(_PyStackRef value_o, _Py_CODEUNIT *instr)
26722719
{
2673-
assert(ENABLE_SPECIALIZATION);
2720+
assert(ENABLE_SPECIALIZATION_FT);
26742721
assert(_PyOpcode_Caches[TO_BOOL] == INLINE_CACHE_ENTRIES_TO_BOOL);
26752722
_PyToBoolCache *cache = (_PyToBoolCache *)(instr + 1);
26762723
PyObject *value = PyStackRef_AsPyObjectBorrow(value_o);
2724+
uint8_t specialized_op;
26772725
if (PyBool_Check(value)) {
2678-
instr->op.code = TO_BOOL_BOOL;
2726+
specialized_op = TO_BOOL_BOOL;
26792727
goto success;
26802728
}
26812729
if (PyLong_CheckExact(value)) {
2682-
instr->op.code = TO_BOOL_INT;
2730+
specialized_op = TO_BOOL_INT;
26832731
goto success;
26842732
}
26852733
if (PyList_CheckExact(value)) {
2686-
instr->op.code = TO_BOOL_LIST;
2734+
specialized_op = TO_BOOL_LIST;
26872735
goto success;
26882736
}
26892737
if (Py_IsNone(value)) {
2690-
instr->op.code = TO_BOOL_NONE;
2738+
specialized_op = TO_BOOL_NONE;
26912739
goto success;
26922740
}
26932741
if (PyUnicode_CheckExact(value)) {
2694-
instr->op.code = TO_BOOL_STR;
2742+
specialized_op = TO_BOOL_STR;
26952743
goto success;
26962744
}
26972745
if (PyType_HasFeature(Py_TYPE(value), Py_TPFLAGS_HEAPTYPE)) {
2698-
PyNumberMethods *nb = Py_TYPE(value)->tp_as_number;
2699-
if (nb && nb->nb_bool) {
2700-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_NUMBER);
2701-
goto failure;
2702-
}
2703-
PyMappingMethods *mp = Py_TYPE(value)->tp_as_mapping;
2704-
if (mp && mp->mp_length) {
2705-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_MAPPING);
2706-
goto failure;
2707-
}
2708-
PySequenceMethods *sq = Py_TYPE(value)->tp_as_sequence;
2709-
if (sq && sq->sq_length) {
2710-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_SEQUENCE);
2711-
goto failure;
2712-
}
2713-
if (!PyUnstable_Type_AssignVersionTag(Py_TYPE(value))) {
2746+
unsigned int version = 0;
2747+
int err = _PyType_Validate(Py_TYPE(value), check_type_always_true, &version);
2748+
if (err < 0) {
27142749
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_OUT_OF_VERSIONS);
27152750
goto failure;
27162751
}
2717-
uint32_t version = type_get_version(Py_TYPE(value), TO_BOOL);
2718-
if (version == 0) {
2752+
else if (err > 0) {
2753+
SPECIALIZATION_FAIL(TO_BOOL, err);
27192754
goto failure;
27202755
}
2721-
instr->op.code = TO_BOOL_ALWAYS_TRUE;
2722-
write_u32(cache->version, version);
2756+
2757+
assert(err == 0);
27232758
assert(version);
2759+
write_u32(cache->version, version);
2760+
specialized_op = TO_BOOL_ALWAYS_TRUE;
27242761
goto success;
27252762
}
2726-
#ifdef Py_STATS
2727-
if (PyByteArray_CheckExact(value)) {
2728-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_BYTEARRAY);
2729-
goto failure;
2730-
}
2731-
if (PyBytes_CheckExact(value)) {
2732-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_BYTES);
2733-
goto failure;
2734-
}
2735-
if (PyDict_CheckExact(value)) {
2736-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_DICT);
2737-
goto failure;
2738-
}
2739-
if (PyFloat_CheckExact(value)) {
2740-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_FLOAT);
2741-
goto failure;
2742-
}
2743-
if (PyMemoryView_Check(value)) {
2744-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_MEMORY_VIEW);
2745-
goto failure;
2746-
}
2747-
if (PyAnySet_CheckExact(value)) {
2748-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_SET);
2749-
goto failure;
2750-
}
2751-
if (PyTuple_CheckExact(value)) {
2752-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_TO_BOOL_TUPLE);
2753-
goto failure;
2754-
}
2755-
SPECIALIZATION_FAIL(TO_BOOL, SPEC_FAIL_OTHER);
2756-
#endif // Py_STATS
2763+
2764+
SPECIALIZATION_FAIL(TO_BOOL, to_bool_fail_kind(value));
27572765
failure:
2758-
STAT_INC(TO_BOOL, failure);
2759-
instr->op.code = TO_BOOL;
2760-
cache->counter = adaptive_counter_backoff(cache->counter);
2766+
unspecialize(instr);
27612767
return;
27622768
success:
2763-
STAT_INC(TO_BOOL, success);
2764-
cache->counter = adaptive_counter_cooldown();
2769+
specialize(instr, specialized_op);
27652770
}
27662771

27672772
#ifdef Py_STATS

0 commit comments

Comments
 (0)