self.assert_specialized(contains_op_set, "CONTAINS_OP_SET")
self.assert_no_opcode(contains_op_set, "CONTAINS_OP")
+ @cpython_only
+ @requires_specialization_ft
+ def test_send_with(self):
+ def run_async(coro):
+ while True:
+ try:
+ coro.send(None)
+ except StopIteration:
+ break
+
+ class CM:
+ async def __aenter__(self):
+ return self
+
+ async def __aexit__(self, *exc):
+ pass
+
+ async def send_with():
+ for i in range(100):
+ async with CM():
+ x = 1
+
+ run_async(send_with())
+ # Note there are still unspecialized "SEND" opcodes in the
+ # cleanup paths of the 'with' statement.
+ self.assert_specialized(send_with, "SEND_GEN")
+
+ @cpython_only
+ @requires_specialization_ft
+ def test_send_yield_from(self):
+ def g():
+ yield None
+
+ def send_yield_from():
+ yield from g()
+
+ for i in range(100):
+ list(send_yield_from())
+
+ self.assert_specialized(send_yield_from, "SEND_GEN")
+ self.assert_no_opcode(send_yield_from, "SEND")
+
@cpython_only
@requires_specialization_ft
def test_to_bool(self):
};
specializing op(_SPECIALIZE_SEND, (counter/1, receiver, unused -- receiver, unused)) {
- #if ENABLE_SPECIALIZATION
+ #if ENABLE_SPECIALIZATION_FT
if (ADAPTIVE_COUNTER_TRIGGERS(counter)) {
next_instr = this_instr;
_Py_Specialize_Send(receiver, next_instr);
}
OPCODE_DEFERRED_INC(SEND);
ADVANCE_ADAPTIVE_COUNTER(this_instr[1].counter);
- #endif /* ENABLE_SPECIALIZATION */
+ #endif /* ENABLE_SPECIALIZATION_FT */
}
op(_SEND, (receiver, v -- receiver, retval)) {
receiver = stack_pointer[-2];
uint16_t counter = read_u16(&this_instr[1].cache);
(void)counter;
- #if ENABLE_SPECIALIZATION
+ #if ENABLE_SPECIALIZATION_FT
if (ADAPTIVE_COUNTER_TRIGGERS(counter)) {
next_instr = this_instr;
_PyFrame_SetStackPointer(frame, stack_pointer);
}
OPCODE_DEFERRED_INC(SEND);
ADVANCE_ADAPTIVE_COUNTER(this_instr[1].counter);
- #endif /* ENABLE_SPECIALIZATION */
+ #endif /* ENABLE_SPECIALIZATION_FT */
}
// _SEND
{
{
PyObject *receiver = PyStackRef_AsPyObjectBorrow(receiver_st);
- assert(ENABLE_SPECIALIZATION);
+ assert(ENABLE_SPECIALIZATION_FT);
assert(_PyOpcode_Caches[SEND] == INLINE_CACHE_ENTRIES_SEND);
- _PySendCache *cache = (_PySendCache *)(instr + 1);
PyTypeObject *tp = Py_TYPE(receiver);
if (tp == &PyGen_Type || tp == &PyCoro_Type) {
if (_PyInterpreterState_GET()->eval_frame) {
SPECIALIZATION_FAIL(SEND, SPEC_FAIL_OTHER);
goto failure;
}
- instr->op.code = SEND_GEN;
- goto success;
+ specialize(instr, SEND_GEN);
+ return;
}
SPECIALIZATION_FAIL(SEND,
_PySpecialization_ClassifyIterator(receiver));
failure:
- STAT_INC(SEND, failure);
- instr->op.code = SEND;
- cache->counter = adaptive_counter_backoff(cache->counter);
- return;
-success:
- STAT_INC(SEND, success);
- cache->counter = adaptive_counter_cooldown();
+ unspecialize(instr);
}
#ifdef Py_STATS