#include "action-manager.h"
#include "memory-alloc.h"
#include "permassert.h"
#include "admin-state.h"
#include "completion.h"
#include "status-codes.h"
#include "types.h"
#include "vdo.h"
struct action {
bool in_use;
const struct admin_state_code *operation;
vdo_action_preamble_fn preamble;
vdo_zone_action_fn zone_action;
vdo_action_conclusion_fn conclusion;
struct vdo_completion *parent;
void *context;
struct action *next;
};
struct action_manager {
struct vdo_completion completion;
struct admin_state state;
struct action actions[2];
struct action *current_action;
zone_count_t zones;
vdo_action_scheduler_fn scheduler;
vdo_zone_thread_getter_fn get_zone_thread_id;
thread_id_t initiator_thread_id;
void *context;
zone_count_t acting_zone;
};
static inline struct action_manager *as_action_manager(struct vdo_completion *completion)
{
vdo_assert_completion_type(completion, VDO_ACTION_COMPLETION);
return container_of(completion, struct action_manager, completion);
}
static bool no_default_action(void *context __always_unused)
{
return false;
}
static void no_preamble(void *context __always_unused, struct vdo_completion *completion)
{
vdo_finish_completion(completion);
}
static int no_conclusion(void *context __always_unused)
{
return VDO_SUCCESS;
}
int vdo_make_action_manager(zone_count_t zones,
vdo_zone_thread_getter_fn get_zone_thread_id,
thread_id_t initiator_thread_id, void *context,
vdo_action_scheduler_fn scheduler, struct vdo *vdo,
struct action_manager **manager_ptr)
{
struct action_manager *manager;
int result = vdo_allocate(1, struct action_manager, __func__, &manager);
if (result != VDO_SUCCESS)
return result;
*manager = (struct action_manager) {
.zones = zones,
.scheduler =
((scheduler == NULL) ? no_default_action : scheduler),
.get_zone_thread_id = get_zone_thread_id,
.initiator_thread_id = initiator_thread_id,
.context = context,
};
manager->actions[0].next = &manager->actions[1];
manager->current_action = manager->actions[1].next =
&manager->actions[0];
vdo_set_admin_state_code(&manager->state, VDO_ADMIN_STATE_NORMAL_OPERATION);
vdo_initialize_completion(&manager->completion, vdo, VDO_ACTION_COMPLETION);
*manager_ptr = manager;
return VDO_SUCCESS;
}
const struct admin_state_code *vdo_get_current_manager_operation(struct action_manager *manager)
{
return vdo_get_admin_state_code(&manager->state);
}
void *vdo_get_current_action_context(struct action_manager *manager)
{
return manager->current_action->in_use ? manager->current_action->context : NULL;
}
static void finish_action_callback(struct vdo_completion *completion);
static void apply_to_zone(struct vdo_completion *completion);
static thread_id_t get_acting_zone_thread_id(struct action_manager *manager)
{
return manager->get_zone_thread_id(manager->context, manager->acting_zone);
}
static void preserve_error(struct vdo_completion *completion)
{
if (completion->parent != NULL)
vdo_set_completion_result(completion->parent, completion->result);
vdo_reset_completion(completion);
vdo_run_completion(completion);
}
static void prepare_for_next_zone(struct action_manager *manager)
{
vdo_prepare_completion_for_requeue(&manager->completion, apply_to_zone,
preserve_error,
get_acting_zone_thread_id(manager),
manager->current_action->parent);
}
static void prepare_for_conclusion(struct action_manager *manager)
{
vdo_prepare_completion_for_requeue(&manager->completion, finish_action_callback,
preserve_error, manager->initiator_thread_id,
manager->current_action->parent);
}
static void apply_to_zone(struct vdo_completion *completion)
{
zone_count_t zone;
struct action_manager *manager = as_action_manager(completion);
VDO_ASSERT_LOG_ONLY((vdo_get_callback_thread_id() == get_acting_zone_thread_id(manager)),
"%s() called on acting zones's thread", __func__);
zone = manager->acting_zone++;
if (manager->acting_zone == manager->zones) {
prepare_for_conclusion(manager);
} else {
prepare_for_next_zone(manager);
}
manager->current_action->zone_action(manager->context, zone, completion);
}
static void handle_preamble_error(struct vdo_completion *completion)
{
completion->callback = finish_action_callback;
preserve_error(completion);
}
static void launch_current_action(struct action_manager *manager)
{
struct action *action = manager->current_action;
int result = vdo_start_operation(&manager->state, action->operation);
if (result != VDO_SUCCESS) {
if (action->parent != NULL)
vdo_set_completion_result(action->parent, result);
action->conclusion = no_conclusion;
finish_action_callback(&manager->completion);
return;
}
if (action->zone_action == NULL) {
prepare_for_conclusion(manager);
} else {
manager->acting_zone = 0;
vdo_prepare_completion_for_requeue(&manager->completion, apply_to_zone,
handle_preamble_error,
get_acting_zone_thread_id(manager),
manager->current_action->parent);
}
action->preamble(manager->context, &manager->completion);
}
bool vdo_schedule_default_action(struct action_manager *manager)
{
const struct admin_state_code *code = vdo_get_current_manager_operation(manager);
return ((code == VDO_ADMIN_STATE_NORMAL_OPERATION) &&
manager->scheduler(manager->context));
}
static void finish_action_callback(struct vdo_completion *completion)
{
bool has_next_action;
int result;
struct action_manager *manager = as_action_manager(completion);
struct action action = *(manager->current_action);
manager->current_action->in_use = false;
manager->current_action = manager->current_action->next;
has_next_action =
(manager->current_action->in_use || vdo_schedule_default_action(manager));
result = action.conclusion(manager->context);
vdo_finish_operation(&manager->state, VDO_SUCCESS);
if (action.parent != NULL)
vdo_continue_completion(action.parent, result);
if (has_next_action)
launch_current_action(manager);
}
bool vdo_schedule_action(struct action_manager *manager, vdo_action_preamble_fn preamble,
vdo_zone_action_fn action, vdo_action_conclusion_fn conclusion,
struct vdo_completion *parent)
{
return vdo_schedule_operation(manager, VDO_ADMIN_STATE_OPERATING, preamble,
action, conclusion, parent);
}
bool vdo_schedule_operation(struct action_manager *manager,
const struct admin_state_code *operation,
vdo_action_preamble_fn preamble, vdo_zone_action_fn action,
vdo_action_conclusion_fn conclusion,
struct vdo_completion *parent)
{
return vdo_schedule_operation_with_context(manager, operation, preamble, action,
conclusion, NULL, parent);
}
bool vdo_schedule_operation_with_context(struct action_manager *manager,
const struct admin_state_code *operation,
vdo_action_preamble_fn preamble,
vdo_zone_action_fn action,
vdo_action_conclusion_fn conclusion,
void *context, struct vdo_completion *parent)
{
struct action *current_action;
VDO_ASSERT_LOG_ONLY((vdo_get_callback_thread_id() == manager->initiator_thread_id),
"action initiated from correct thread");
if (!manager->current_action->in_use) {
current_action = manager->current_action;
} else if (!manager->current_action->next->in_use) {
current_action = manager->current_action->next;
} else {
if (parent != NULL)
vdo_continue_completion(parent, VDO_COMPONENT_BUSY);
return false;
}
*current_action = (struct action) {
.in_use = true,
.operation = operation,
.preamble = (preamble == NULL) ? no_preamble : preamble,
.zone_action = action,
.conclusion = (conclusion == NULL) ? no_conclusion : conclusion,
.context = context,
.parent = parent,
.next = current_action->next,
};
if (current_action == manager->current_action)
launch_current_action(manager);
return true;
}