Blame source/vdo/base/vdoResizeLogical.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/vdoResizeLogical.c#6 $
Packit Service 75d76b
 */
Packit Service 75d76b
Packit Service 75d76b
#include "vdoResizeLogical.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 "vdoInternal.h"
Packit Service 75d76b
Packit Service 75d76b
typedef enum {
Packit Service 75d76b
  GROW_LOGICAL_PHASE_START = 0,
Packit Service 75d76b
  GROW_LOGICAL_PHASE_GROW_BLOCK_MAP,
Packit Service 75d76b
  GROW_LOGICAL_PHASE_END,
Packit Service 75d76b
  GROW_LOGICAL_PHASE_ERROR,
Packit Service 75d76b
} GrowLogicalPhase;
Packit Service 75d76b
Packit Service 75d76b
static const char *GROW_LOGICAL_PHASE_NAMES[] = {
Packit Service 75d76b
  "GROW_LOGICAL_PHASE_START",
Packit Service 75d76b
  "GROW_LOGICAL_PHASE_GROW_BLOCK_MAP",
Packit Service 75d76b
  "GROW_LOGICAL_PHASE_END",
Packit Service 75d76b
  "GROW_LOGICAL_PHASE_ERROR",
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
  return getAdminThread(getThreadConfig(adminCompletion->completion.parent));
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Callback to initiate a grow logical, registered in performGrowLogical().
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The sub-task completion
Packit Service 75d76b
 **/
Packit Service 75d76b
static void growLogicalCallback(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  AdminCompletion *adminCompletion = adminCompletionFromSubTask(completion);
Packit Service 75d76b
  assertAdminOperationType(adminCompletion, ADMIN_OPERATION_GROW_LOGICAL);
Packit Service 75d76b
  assertAdminPhaseThread(adminCompletion, __func__, GROW_LOGICAL_PHASE_NAMES);
Packit Service 75d76b
Packit Service 75d76b
  VDO *vdo = adminCompletion->completion.parent;
Packit Service 75d76b
  switch (adminCompletion->phase++) {
Packit Service 75d76b
  case GROW_LOGICAL_PHASE_START:
Packit Service 75d76b
    if (isReadOnly(vdo->readOnlyNotifier)) {
Packit Service 75d76b
      logErrorWithStringError(VDO_READ_ONLY,
Packit Service 75d76b
                              "Can't grow logical size of a read-only VDO");
Packit Service 75d76b
      finishCompletion(resetAdminSubTask(completion), VDO_READ_ONLY);
Packit Service 75d76b
      return;
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    if (startOperationWithWaiter(&vdo->adminState,
Packit Service 75d76b
                                 ADMIN_STATE_SUSPENDED_OPERATION,
Packit Service 75d76b
                                 &adminCompletion->completion, NULL)) {
Packit Service 75d76b
Packit Service 75d76b
      vdo->config.logicalBlocks = getNewEntryCount(getBlockMap(vdo));
Packit Service 75d76b
      saveVDOComponentsAsync(vdo, resetAdminSubTask(completion));
Packit Service 75d76b
    }
Packit Service 75d76b
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case GROW_LOGICAL_PHASE_GROW_BLOCK_MAP:
Packit Service 75d76b
    growBlockMap(getBlockMap(vdo), resetAdminSubTask(completion));
Packit Service 75d76b
    return;
Packit Service 75d76b
Packit Service 75d76b
  case GROW_LOGICAL_PHASE_END:
Packit Service 75d76b
    break;
Packit Service 75d76b
Packit Service 75d76b
  case GROW_LOGICAL_PHASE_ERROR:
Packit Service 75d76b
    enterReadOnlyMode(vdo->readOnlyNotifier, completion->result);
Packit Service 75d76b
    break;
Packit Service 75d76b
Packit Service 75d76b
  default:
Packit Service 75d76b
    setCompletionResult(resetAdminSubTask(completion), UDS_BAD_STATE);
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  finishOperationWithResult(&vdo->adminState, completion->result);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Handle an error during the grow physical process.
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The sub-task completion
Packit Service 75d76b
 **/
Packit Service 75d76b
static void handleGrowthError(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  AdminCompletion *adminCompletion = adminCompletionFromSubTask(completion);
Packit Service 75d76b
  if (adminCompletion->phase == GROW_LOGICAL_PHASE_GROW_BLOCK_MAP) {
Packit Service 75d76b
    // We've failed to write the new size in the super block, so set our
Packit Service 75d76b
    // in memory config back to the old size.
Packit Service 75d76b
    VDO      *vdo = adminCompletion->completion.parent;
Packit Service 75d76b
    BlockMap *map = getBlockMap(vdo);
Packit Service 75d76b
    vdo->config.logicalBlocks = getNumberOfBlockMapEntries(map);
Packit Service 75d76b
    abandonBlockMapGrowth(map);
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  adminCompletion->phase = GROW_LOGICAL_PHASE_ERROR;
Packit Service 75d76b
  growLogicalCallback(completion);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**********************************************************************/
Packit Service 75d76b
int performGrowLogical(VDO *vdo, BlockCount newLogicalBlocks)
Packit Service 75d76b
{
Packit Service 75d76b
  if (getNewEntryCount(getBlockMap(vdo)) != newLogicalBlocks) {
Packit Service 75d76b
    return VDO_PARAMETER_MISMATCH;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  return performAdminOperation(vdo, ADMIN_OPERATION_GROW_LOGICAL,
Packit Service 75d76b
                               getThreadIDForPhase, growLogicalCallback,
Packit Service 75d76b
                               handleGrowthError);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**********************************************************************/
Packit Service 75d76b
int prepareToGrowLogical(VDO *vdo, BlockCount newLogicalBlocks)
Packit Service 75d76b
{
Packit Service 75d76b
  if (newLogicalBlocks < vdo->config.logicalBlocks) {
Packit Service 75d76b
    return logErrorWithStringError(VDO_PARAMETER_MISMATCH,
Packit Service 75d76b
                                   "Can't shrink VDO logical size from its "
Packit Service 75d76b
                                   "current value of %llu",
Packit Service 75d76b
                                   vdo->config.logicalBlocks);
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  if (newLogicalBlocks == vdo->config.logicalBlocks) {
Packit Service 75d76b
    return logErrorWithStringError(VDO_PARAMETER_MISMATCH,
Packit Service 75d76b
                                   "Can't grow VDO logical size to its "
Packit Service 75d76b
                                   "current value of %llu",
Packit Service 75d76b
                                   vdo->config.logicalBlocks);
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  return prepareToGrowBlockMap(getBlockMap(vdo), newLogicalBlocks);
Packit Service 75d76b
}