/* SPDX-License-Identifier: GPL-2.0-only */ /* * Copyright 2023 Red Hat */ #ifndef VDO_COMPLETION_H #define VDO_COMPLETION_H #include "permassert.h" #include "status-codes.h" #include "types.h" /** * vdo_run_completion() - Run a completion's callback or error handler on the current thread. * * Context: This function must be called from the correct callback thread. */ static inline void vdo_run_completion(struct vdo_completion *completion) { if ((completion->result != VDO_SUCCESS) && (completion->error_handler != NULL)) { completion->error_handler(completion); return; } completion->callback(completion); } void vdo_set_completion_result(struct vdo_completion *completion, int result); void vdo_initialize_completion(struct vdo_completion *completion, struct vdo *vdo, enum vdo_completion_type type); /** * vdo_reset_completion() - Reset a completion to a clean state, while keeping the type, vdo and * parent information. */ static inline void vdo_reset_completion(struct vdo_completion *completion) { completion->result = VDO_SUCCESS; completion->complete = false; } void vdo_launch_completion_with_priority(struct vdo_completion *completion, enum vdo_completion_priority priority); /** * vdo_launch_completion() - Launch a completion with default priority. */ static inline void vdo_launch_completion(struct vdo_completion *completion) { vdo_launch_completion_with_priority(completion, VDO_WORK_Q_DEFAULT_PRIORITY); } /** * vdo_continue_completion() - Continue processing a completion. * @result: The current result (will not mask older errors). * * Continue processing a completion by setting the current result and calling * vdo_launch_completion(). */ static inline void vdo_continue_completion(struct vdo_completion *completion, int result) { vdo_set_completion_result(completion, result); vdo_launch_completion(completion); } void vdo_finish_completion(struct vdo_completion *completion); /** * vdo_fail_completion() - Set the result of a completion if it does not already have an error, * then finish it. */ static inline void vdo_fail_completion(struct vdo_completion *completion, int result) { vdo_set_completion_result(completion, result); vdo_finish_completion(completion); } /** * vdo_assert_completion_type() - Assert that a completion is of the correct type. * * Return: VDO_SUCCESS or an error */ static inline int vdo_assert_completion_type(struct vdo_completion *completion, enum vdo_completion_type expected) { return VDO_ASSERT(expected == completion->type, "completion type should be %u, not %u", expected, completion->type); } static inline void vdo_set_completion_callback(struct vdo_completion *completion, vdo_action_fn callback, thread_id_t callback_thread_id) { completion->callback = callback; completion->callback_thread_id = callback_thread_id; } /** * vdo_launch_completion_callback() - Set the callback for a completion and launch it immediately. */ static inline void vdo_launch_completion_callback(struct vdo_completion *completion, vdo_action_fn callback, thread_id_t callback_thread_id) { vdo_set_completion_callback(completion, callback, callback_thread_id); vdo_launch_completion(completion); } /** * vdo_prepare_completion() - Prepare a completion for launch. * * Resets the completion, and then sets its callback, error handler, callback thread, and parent. */ static inline void vdo_prepare_completion(struct vdo_completion *completion, vdo_action_fn callback, vdo_action_fn error_handler, thread_id_t callback_thread_id, void *parent) { vdo_reset_completion(completion); vdo_set_completion_callback(completion, callback, callback_thread_id); completion->error_handler = error_handler; completion->parent = parent; } /** * vdo_prepare_completion_for_requeue() - Prepare a completion for launch ensuring that it will * always be requeued. * * Resets the completion, and then sets its callback, error handler, callback thread, and parent. */ static inline void vdo_prepare_completion_for_requeue(struct vdo_completion *completion, vdo_action_fn callback, vdo_action_fn error_handler, thread_id_t callback_thread_id, void *parent) { vdo_prepare_completion(completion, callback, error_handler, callback_thread_id, parent); completion->requeue = true; } void vdo_enqueue_completion(struct vdo_completion *completion, enum vdo_completion_priority priority); bool vdo_requeue_completion_if_needed(struct vdo_completion *completion, thread_id_t callback_thread_id); #endif /* VDO_COMPLETION_H */