// SPDX-License-Identifier: GPL-2.0-or-later /* Call state changing functions. * * Copyright (C) 2022 Red Hat, Inc. All Rights Reserved. * Written by David Howells (dhowells@redhat.com) */ #include "ar-internal.h" /* * Transition a call to the complete state. */ bool rxrpc_set_call_completion(struct rxrpc_call *call, enum rxrpc_call_completion compl, u32 abort_code, int error) { if (__rxrpc_call_state(call) == RXRPC_CALL_COMPLETE) return false; call->abort_code = abort_code; call->error = error; call->completion = compl; /* Allow reader of completion state to operate locklessly */ rxrpc_set_call_state(call, RXRPC_CALL_COMPLETE); trace_rxrpc_call_complete(call); wake_up(&call->waitq); rxrpc_notify_socket(call); return true; } /* * Record that a call successfully completed. */ bool rxrpc_call_completed(struct rxrpc_call *call) { return rxrpc_set_call_completion(call, RXRPC_CALL_SUCCEEDED, 0, 0); } /* * Record that a call is locally aborted. */ bool rxrpc_abort_call(struct rxrpc_call *call, rxrpc_seq_t seq, u32 abort_code, int error, enum rxrpc_abort_reason why) { trace_rxrpc_abort(call->debug_id, why, call->cid, call->call_id, seq, abort_code, error); if (!rxrpc_set_call_completion(call, RXRPC_CALL_LOCALLY_ABORTED, abort_code, error)) return false; if (test_bit(RXRPC_CALL_EXPOSED, &call->flags)) rxrpc_send_abort_packet(call); return true; } /* * Record that a call errored out before even getting off the ground, thereby * setting the state to allow it to be destroyed. */ void rxrpc_prefail_call(struct rxrpc_call *call, enum rxrpc_call_completion compl, int error) { call->abort_code = RX_CALL_DEAD; call->error = error; call->completion = compl; call->_state = RXRPC_CALL_COMPLETE; trace_rxrpc_call_complete(call); WARN_ON_ONCE(__test_and_set_bit(RXRPC_CALL_RELEASED, &call->flags)); }