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