Skip to content

Async function generates potentially panicking code #147041

@ZhekaS

Description

@ZhekaS

It appears that async fn() {...} desugars into Future state machine with a possible panic!() path. I understand that this path is taken if poll() is executed on a finished task. This is a problem when trying to write a panic-free code (that is with no panic-related code linked to the final artifact).
I was not able to write a minimal reproducing example, so I will put here the excerpts from my full code (no_std, attempting to implement custom executor):

pub async fn async_number() -> u32 {
    42
}

pub async fn example_task() {
    let num = async_number().await;
    let _ = writeln!(dbg_print::DebugPrint, "Async number: {}", num);
}

This produces the following llvm-ir:

; test_futures::example_task::{{closure}}
; Function Attrs: inlinehint minsize nounwind optsize
define internal noundef zeroext i1 @"_ZN12test_futures12example_task28_$u7b$$u7b$closure$u7d$$u7d$17h69ea2006b28af955E"(ptr nocapture noundef nonnull align 1 %_1, ptr noalias nocapture readnone align 4 %_2) unnamed_addr #4 !dbg !1446 {
start:
  %_14 = alloca [0 x i8], align 1
  %args = alloca [8 x i8], align 4
  %_15 = alloca [24 x i8], align 4
  %num = alloca [4 x i8], align 4
  // ...................
  %0 = load i8, ptr %_1, align 1, !dbg !1477, !range !1478, !noundef !17
  switch i8 %0, label %bb7 [
    i8 0, label %bb4.thread
    i8 1, label %panic
    i8 3, label %bb4
  ], !dbg !1477

// ........

panic.i:                                          ; preds = %bb4
; call core::panicking::panic_const::panic_const_async_fn_resumed
  tail call fastcc void @_ZN4core9panicking11panic_const28panic_const_async_fn_resumed17h3b19b0997d8ed2ccE() #12, !dbg !1505
  unreachable, !dbg !1505

In the final disassembly the call to panic_const_async_fn_resumed() appears as well.

I am not sure what would be the right way to solve this. Perhaps introducing Future::try_poll() ? Or adding a "Finished" variant to the Poll enum.

UPDATE:
I believe the code responsible for generating the panic block is this:

if can_return {
let block = match transform.coroutine_kind {
CoroutineKind::Desugared(CoroutineDesugaring::Async, _)
| CoroutineKind::Coroutine(_) => {
// For `async_drop_in_place<T>::{closure}` we just keep return Poll::Ready,
// because async drop of such coroutine keeps polling original coroutine
if tcx.is_async_drop_in_place_coroutine(body.source.def_id()) {
insert_poll_ready_block(tcx, body)
} else {
insert_panic_block(tcx, body, ResumedAfterReturn(transform.coroutine_kind))
}

It is also documented:

//! One of them is the implementation of `Coroutine::resume` / `Future::poll`.
//! For coroutines with state 0 (unresumed) it starts the execution of the coroutine.
//! For coroutines with state 1 (returned) and state 2 (poisoned) it panics.
//! Otherwise it continues the execution from the last suspension point.

However it does not change the fact that async/await is not panic-free-friendly.

Metadata

Metadata

Assignees

No one assigned

    Labels

    A-async-awaitArea: Async & AwaitC-discussionCategory: Discussion or questions that doesn't represent real issues.T-langRelevant to the language team

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions