|
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 |
}
|