Blame source/vdo/base/vdoSuspend.c

Packit Service 75d76b
/*
Packit Service 75d76b
 * Copyright (c) 2020 Red Hat, Inc.
Packit Service 75d76b
 *
Packit Service 75d76b
 * This program is free software; you can redistribute it and/or
Packit Service 75d76b
 * modify it under the terms of the GNU General Public License
Packit Service 75d76b
 * as published by the Free Software Foundation; either version 2
Packit Service 75d76b
 * of the License, or (at your option) any later version.
Packit Service 75d76b
 * 
Packit Service 75d76b
 * This program is distributed in the hope that it will be useful,
Packit Service 75d76b
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 75d76b
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 75d76b
 * GNU General Public License for more details.
Packit Service 75d76b
 * 
Packit Service 75d76b
 * You should have received a copy of the GNU General Public License
Packit Service 75d76b
 * along with this program; if not, write to the Free Software
Packit Service 75d76b
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
Packit Service 75d76b
 * 02110-1301, USA. 
Packit Service 75d76b
 *
Packit Service 75d76b
 * $Id: //eng/vdo-releases/aluminum/src/c++/vdo/base/vdoSuspend.c#4 $
Packit Service 75d76b
 */
Packit Service 75d76b
Packit Service 75d76b
#include "vdoSuspend.h"
Packit Service 75d76b
Packit Service 75d76b
#include "logger.h"
Packit Service 75d76b
Packit Service 75d76b
#include "adminCompletion.h"
Packit Service 75d76b
#include "blockMap.h"
Packit Service 75d76b
#include "completion.h"
Packit Service 75d76b
#include "logicalZone.h"
Packit Service 75d76b
#include "recoveryJournal.h"
Packit Service 75d76b
#include "slabDepot.h"
Packit Service 75d76b
#include "slabSummary.h"
Packit Service 75d76b
#include "threadConfig.h"
Packit Service 75d76b
#include "vdoInternal.h"
Packit Service 75d76b
Packit Service 75d76b
typedef enum {
Packit Service 75d76b
  SUSPEND_PHASE_START = 0,
Packit Service 75d76b
  SUSPEND_PHASE_PACKER,
Packit Service 75d76b
  SUSPEND_PHASE_LOGICAL_ZONES,
Packit Service 75d76b
  SUSPEND_PHASE_BLOCK_MAP,
Packit Service 75d76b
  SUSPEND_PHASE_JOURNAL,
Packit Service 75d76b
  SUSPEND_PHASE_DEPOT,
Packit Service 75d76b
  SUSPEND_PHASE_WRITE_SUPER_BLOCK,
Packit Service 75d76b
  SUSPEND_PHASE_END,
Packit Service 75d76b
} SuspendPhase;
Packit Service 75d76b
Packit Service 75d76b
static const char *SUSPEND_PHASE_NAMES[] = {
Packit Service 75d76b
  "SUSPEND_PHASE_START",
Packit Service 75d76b
  "SUSPEND_PHASE_PACKER",
Packit Service 75d76b
  "SUSPEND_PHASE_LOGICAL_ZONES",
Packit Service 75d76b
  "SUSPEND_PHASE_BLOCK_MAP",
Packit Service 75d76b
  "SUSPEND_PHASE_JOURNAL",
Packit Service 75d76b
  "SUSPEND_PHASE_DEPOT",
Packit Service 75d76b
  "SUSPEND_PHASE_WRITE_SUPER_BLOCK",
Packit Service 75d76b
  "SUSPEND_PHASE_END",
Packit Service 75d76b
};
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Implements ThreadIDGetterForPhase.
Packit Service 75d76b
 **/
Packit Service 75d76b
__attribute__((warn_unused_result))
Packit Service 75d76b
static ThreadID getThreadIDForPhase(AdminCompletion *adminCompletion)
Packit Service 75d76b
{
Packit Service 75d76b
  const ThreadConfig *threadConfig
Packit Service 75d76b
    = getThreadConfig(adminCompletion->completion.parent);
Packit Service 75d76b
  switch (adminCompletion->phase) {
Packit Service 75d76b
  case SUSPEND_PHASE_PACKER:
Packit Service 75d76b
    return getPackerZoneThread(threadConfig);
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_JOURNAL:
Packit Service 75d76b
    return getJournalZoneThread(threadConfig);
Packit Service 75d76b
Packit Service 75d76b
  default:
Packit Service 75d76b
    return getAdminThread(threadConfig);
Packit Service 75d76b
  }
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Update the VDO state and save the super block.
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param vdo         The VDO being suspended
Packit Service 75d76b
 * @param completion  The AdminCompletion's sub-task completion
Packit Service 75d76b
 **/
Packit Service 75d76b
static void writeSuperBlock(VDO *vdo, VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  switch (vdo->state) {
Packit Service 75d76b
  case VDO_DIRTY:
Packit Service 75d76b
  case VDO_NEW:
Packit Service 75d76b
    vdo->state = VDO_CLEAN;
Packit Service 75d76b
    break;
Packit Service 75d76b
Packit Service 75d76b
  case VDO_CLEAN:
Packit Service 75d76b
  case VDO_READ_ONLY_MODE:
Packit Service 75d76b
  case VDO_FORCE_REBUILD:
Packit Service 75d76b
  case VDO_RECOVERING:
Packit Service 75d76b
  case VDO_REBUILD_FOR_UPGRADE:
Packit Service 75d76b
    break;
Packit Service 75d76b
Packit Service 75d76b
  case VDO_REPLAYING:
Packit Service 75d76b
  default:
Packit Service 75d76b
    finishCompletion(completion, UDS_BAD_STATE);
Packit Service 75d76b
    return;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  saveVDOComponentsAsync(vdo, completion);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Callback to initiate a suspend, registered in performVDOSuspend().
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The sub-task completion
Packit Service 75d76b
 **/
Packit Service 75d76b
static void suspendCallback(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  AdminCompletion *adminCompletion = adminCompletionFromSubTask(completion);
Packit Service 75d76b
  ASSERT_LOG_ONLY(((adminCompletion->type == ADMIN_OPERATION_SUSPEND)
Packit Service 75d76b
                   || (adminCompletion->type == ADMIN_OPERATION_SAVE)),
Packit Service 75d76b
                  "unexpected admin operation type %u is neither "
Packit Service 75d76b
                  "suspend nor save", adminCompletion->type);
Packit Service 75d76b
  assertAdminPhaseThread(adminCompletion, __func__, SUSPEND_PHASE_NAMES);
Packit Service 75d76b
Packit Service 75d76b
  VDO *vdo = adminCompletion->completion.parent;
Packit Service 75d76b
  switch (adminCompletion->phase++) {
Packit Service 75d76b
  case SUSPEND_PHASE_START:
Packit Service 75d76b
    if (!startDraining(&vdo->adminState,
Packit Service 75d76b
                       ((adminCompletion->type == ADMIN_OPERATION_SUSPEND)
Packit Service 75d76b
                        ? ADMIN_STATE_SUSPENDING : ADMIN_STATE_SAVING),
Packit Service 75d76b
                       &adminCompletion->completion, NULL)) {
Packit Service 75d76b
      return;
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    if (!vdo->closeRequired) {
Packit Service 75d76b
      // There's nothing to do.
Packit Service 75d76b
      break;
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    waitUntilNotEnteringReadOnlyMode(vdo->readOnlyNotifier,
Packit Service 75d76b
                                     resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_PACKER:
Packit Service 75d76b
    /*
Packit Service 75d76b
     * If the VDO was already resumed from a prior suspend while read-only,
Packit Service 75d76b
     * some of the components may not have been resumed. By setting a read-only
Packit Service 75d76b
     * error here, we guarantee that the result of this suspend will be
Packit Service 75d76b
     * VDO_READ_ONLY and not VDO_INVALID_ADMIN_STATE in that case.
Packit Service 75d76b
     */
Packit Service 75d76b
    if (inReadOnlyMode(vdo)) {
Packit Service 75d76b
      setCompletionResult(&adminCompletion->completion, VDO_READ_ONLY);
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    drainPacker(vdo->packer, resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_LOGICAL_ZONES:
Packit Service 75d76b
    drainLogicalZones(vdo->logicalZones, vdo->adminState.state,
Packit Service 75d76b
                      resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_BLOCK_MAP:
Packit Service 75d76b
    drainBlockMap(vdo->blockMap, vdo->adminState.state,
Packit Service 75d76b
                  resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_JOURNAL:
Packit Service 75d76b
    drainRecoveryJournal(vdo->recoveryJournal, vdo->adminState.state,
Packit Service 75d76b
                         resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_DEPOT:
Packit Service 75d76b
    drainSlabDepot(vdo->depot, vdo->adminState.state,
Packit Service 75d76b
                   resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_WRITE_SUPER_BLOCK:
Packit Service 75d76b
    if (isSuspending(&vdo->adminState)
Packit Service 75d76b
        || (adminCompletion->completion.result != VDO_SUCCESS)) {
Packit Service 75d76b
      // If we didn't save the VDO or there was an error, we're done.
Packit Service 75d76b
      break;
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    writeSuperBlock(vdo, resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case SUSPEND_PHASE_END:
Packit Service 75d76b
    break;
Packit Service 75d76b
Packit Service 75d76b
  default:
Packit Service 75d76b
    setCompletionResult(completion, UDS_BAD_STATE);
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  finishDrainingWithResult(&vdo->adminState, completion->result);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**********************************************************************/
Packit Service 75d76b
int performVDOSuspend(VDO *vdo, bool save)
Packit Service 75d76b
{
Packit Service 75d76b
  return performAdminOperation(vdo, (save
Packit Service 75d76b
                                     ? ADMIN_OPERATION_SAVE
Packit Service 75d76b
                                     : ADMIN_OPERATION_SUSPEND),
Packit Service 75d76b
                               getThreadIDForPhase, suspendCallback,
Packit Service 75d76b
                               preserveErrorAndContinue);
Packit Service 75d76b
}