Skip to content

Commit

Permalink
Fix delivering response promises in actor state destructors
Browse files Browse the repository at this point in the history
  • Loading branch information
dominiklohmann committed Jan 16, 2024
1 parent 261a6b3 commit 9770387
Showing 1 changed file with 19 additions and 10 deletions.
29 changes: 19 additions & 10 deletions libcaf_core/src/response_promise.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -166,22 +166,31 @@ void response_promise::state::cancel() {

void response_promise::state::deliver_impl(message msg) {
CAF_LOG_TRACE(CAF_ARG(msg));
// Even though we are holding a weak pointer, we can access the pointer
// without any additional check here because only the actor itself is allowed
// to call this function.
auto self = static_cast<local_actor*>(weak_self.get()->get());
auto cancel_guard = detail::make_scope_guard([this] {
cancel();
});
if (msg.empty() && id.is_async()) {
CAF_LOG_DEBUG("drop response: empty response to asynchronous input");
} else if (!stages.empty()) {
return;
}
auto self = weak_self.lock();
if (self == nullptr) {
auto element = make_mailbox_element(self, id.response_id(),
std::move(stages),
std::move(msg));
source->enqueue(std::move(element), nullptr);
return;
}
if (!stages.empty()) {
auto next = std::move(stages.back());
stages.pop_back();
detail::profiled_send(self, std::move(source), next, id, std::move(stages),
self->context(), std::move(msg));
} else if (source != nullptr) {
detail::profiled_send(self, self->ctrl(), source, id.response_id(),
forwarding_stack{}, self->context(), std::move(msg));
}
cancel();
return;
}
CAF_ASSERT(source != nullptr);
detail::profiled_send(self, self->ctrl(), source, id.response_id(),
forwarding_stack{}, self->context(), std::move(msg));
}

void response_promise::state::delegate_impl(abstract_actor* receiver,
Expand Down

0 comments on commit 9770387

Please sign in to comment.