|
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/recoveryUtils.c#4 $
|
|
Packit Service |
75d76b |
*/
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
#include "recoveryUtils.h"
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
#include "logger.h"
|
|
Packit Service |
75d76b |
#include "memoryAlloc.h"
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
#include "completion.h"
|
|
Packit Service |
75d76b |
#include "extent.h"
|
|
Packit Service |
75d76b |
#include "packedRecoveryJournalBlock.h"
|
|
Packit Service |
75d76b |
#include "recoveryJournalEntry.h"
|
|
Packit Service |
75d76b |
#include "recoveryJournalInternals.h"
|
|
Packit Service |
75d76b |
#include "slabDepot.h"
|
|
Packit Service |
75d76b |
#include "vdoInternal.h"
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
/**
|
|
Packit Service |
75d76b |
* Finish loading the journal by freeing the extent and notifying the parent.
|
|
Packit Service |
75d76b |
* This callback is registered in loadJournalAsync().
|
|
Packit Service |
75d76b |
*
|
|
Packit Service |
75d76b |
* @param completion The load extent
|
|
Packit Service |
75d76b |
**/
|
|
Packit Service |
75d76b |
static void finishJournalLoad(VDOCompletion *completion)
|
|
Packit Service |
75d76b |
{
|
|
Packit Service |
75d76b |
int result = completion->result;
|
|
Packit Service |
75d76b |
VDOCompletion *parent = completion->parent;
|
|
Packit Service |
75d76b |
VDOExtent *extent = asVDOExtent(completion);
|
|
Packit Service |
75d76b |
freeExtent(&extent);
|
|
Packit Service |
75d76b |
finishCompletion(parent, result);
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
/**********************************************************************/
|
|
Packit Service |
75d76b |
void loadJournalAsync(RecoveryJournal *journal,
|
|
Packit Service |
75d76b |
VDOCompletion *parent,
|
|
Packit Service |
75d76b |
char **journalDataPtr)
|
|
Packit Service |
75d76b |
{
|
|
Packit Service |
75d76b |
int result = ALLOCATE(journal->size * VDO_BLOCK_SIZE, char, __func__,
|
|
Packit Service |
75d76b |
journalDataPtr);
|
|
Packit Service |
75d76b |
if (result != VDO_SUCCESS) {
|
|
Packit Service |
75d76b |
finishCompletion(parent, result);
|
|
Packit Service |
75d76b |
return;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
VDOExtent *extent;
|
|
Packit Service |
75d76b |
result = createExtent(parent->layer, VIO_TYPE_RECOVERY_JOURNAL,
|
|
Packit Service |
75d76b |
VIO_PRIORITY_METADATA, journal->size,
|
|
Packit Service |
75d76b |
*journalDataPtr, &extent);
|
|
Packit Service |
75d76b |
if (result != VDO_SUCCESS) {
|
|
Packit Service |
75d76b |
finishCompletion(parent, result);
|
|
Packit Service |
75d76b |
return;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
prepareCompletion(&extent->completion, finishJournalLoad, finishJournalLoad,
|
|
Packit Service |
75d76b |
parent->callbackThreadID, parent);
|
|
Packit Service |
75d76b |
readMetadataExtent(extent,
|
|
Packit Service |
75d76b |
getFixedLayoutPartitionOffset(journal->partition));
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
/**
|
|
Packit Service |
75d76b |
* Determine whether the given header describe a valid block for the
|
|
Packit Service |
75d76b |
* given journal that could appear at the given offset in the journal.
|
|
Packit Service |
75d76b |
*
|
|
Packit Service |
75d76b |
* @param journal The journal to use
|
|
Packit Service |
75d76b |
* @param header The unpacked block header to check
|
|
Packit Service |
75d76b |
* @param offset An offset indicating where the block was in the journal
|
|
Packit Service |
75d76b |
*
|
|
Packit Service |
75d76b |
* @return True if the header matches
|
|
Packit Service |
75d76b |
**/
|
|
Packit Service |
75d76b |
__attribute__((warn_unused_result))
|
|
Packit Service |
75d76b |
static bool isCongruentRecoveryJournalBlock(RecoveryJournal *journal,
|
|
Packit Service |
75d76b |
const RecoveryBlockHeader *header,
|
|
Packit Service |
75d76b |
PhysicalBlockNumber offset)
|
|
Packit Service |
75d76b |
{
|
|
Packit Service |
75d76b |
PhysicalBlockNumber expectedOffset
|
|
Packit Service |
75d76b |
= getRecoveryJournalBlockNumber(journal, header->sequenceNumber);
|
|
Packit Service |
75d76b |
return ((expectedOffset == offset)
|
|
Packit Service |
75d76b |
&& isValidRecoveryJournalBlock(journal, header));
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
/**********************************************************************/
|
|
Packit Service |
75d76b |
bool findHeadAndTail(RecoveryJournal *journal,
|
|
Packit Service |
75d76b |
char *journalData,
|
|
Packit Service |
75d76b |
SequenceNumber *tailPtr,
|
|
Packit Service |
75d76b |
SequenceNumber *blockMapHeadPtr,
|
|
Packit Service |
75d76b |
SequenceNumber *slabJournalHeadPtr)
|
|
Packit Service |
75d76b |
{
|
|
Packit Service |
75d76b |
SequenceNumber highestTail = journal->tail;
|
|
Packit Service |
75d76b |
SequenceNumber blockMapHeadMax = 0;
|
|
Packit Service |
75d76b |
SequenceNumber slabJournalHeadMax = 0;
|
|
Packit Service |
75d76b |
bool foundEntries = false;
|
|
Packit Service |
75d76b |
for (PhysicalBlockNumber i = 0; i < journal->size; i++) {
|
|
Packit Service |
75d76b |
PackedJournalHeader *packedHeader
|
|
Packit Service |
75d76b |
= getJournalBlockHeader(journal, journalData, i);
|
|
Packit Service |
75d76b |
RecoveryBlockHeader header;
|
|
Packit Service |
75d76b |
unpackRecoveryBlockHeader(packedHeader, &header);
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
if (!isCongruentRecoveryJournalBlock(journal, &header, i)) {
|
|
Packit Service |
75d76b |
// This block is old, unformatted, or doesn't belong at this location.
|
|
Packit Service |
75d76b |
continue;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
if (header.sequenceNumber >= highestTail) {
|
|
Packit Service |
75d76b |
foundEntries = true;
|
|
Packit Service |
75d76b |
highestTail = header.sequenceNumber;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
if (header.blockMapHead > blockMapHeadMax) {
|
|
Packit Service |
75d76b |
blockMapHeadMax = header.blockMapHead;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
if (header.slabJournalHead > slabJournalHeadMax) {
|
|
Packit Service |
75d76b |
slabJournalHeadMax = header.slabJournalHead;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
*tailPtr = highestTail;
|
|
Packit Service |
75d76b |
if (!foundEntries) {
|
|
Packit Service |
75d76b |
return false;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
*blockMapHeadPtr = blockMapHeadMax;
|
|
Packit Service |
75d76b |
if (slabJournalHeadPtr != NULL) {
|
|
Packit Service |
75d76b |
*slabJournalHeadPtr = slabJournalHeadMax;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
return true;
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
/**********************************************************************/
|
|
Packit Service |
75d76b |
int validateRecoveryJournalEntry(const VDO *vdo,
|
|
Packit Service |
75d76b |
const RecoveryJournalEntry *entry)
|
|
Packit Service |
75d76b |
{
|
|
Packit Service |
75d76b |
if ((entry->slot.pbn >= vdo->config.physicalBlocks)
|
|
Packit Service |
75d76b |
|| (entry->slot.slot >= BLOCK_MAP_ENTRIES_PER_PAGE)
|
|
Packit Service |
75d76b |
|| !isValidLocation(&entry->mapping)
|
|
Packit Service |
75d76b |
|| !isPhysicalDataBlock(vdo->depot, entry->mapping.pbn)) {
|
|
Packit Service |
75d76b |
return logErrorWithStringError(VDO_CORRUPT_JOURNAL, "Invalid entry:"
|
|
Packit Service |
75d76b |
" (%llu, %" PRIu16 ") to %" PRIu64
|
|
Packit Service |
75d76b |
" (%s) is not within bounds",
|
|
Packit Service |
75d76b |
entry->slot.pbn, entry->slot.slot,
|
|
Packit Service |
75d76b |
entry->mapping.pbn,
|
|
Packit Service |
75d76b |
getJournalOperationName(entry->operation));
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
if ((entry->operation == BLOCK_MAP_INCREMENT)
|
|
Packit Service |
75d76b |
&& (isCompressed(entry->mapping.state)
|
|
Packit Service |
75d76b |
|| (entry->mapping.pbn == ZERO_BLOCK))) {
|
|
Packit Service |
75d76b |
return logErrorWithStringError(VDO_CORRUPT_JOURNAL, "Invalid entry:"
|
|
Packit Service |
75d76b |
" (%llu, %" PRIu16 ") to %" PRIu64
|
|
Packit Service |
75d76b |
" (%s) is not a valid tree mapping",
|
|
Packit Service |
75d76b |
entry->slot.pbn, entry->slot.slot,
|
|
Packit Service |
75d76b |
entry->mapping.pbn,
|
|
Packit Service |
75d76b |
getJournalOperationName(entry->operation));
|
|
Packit Service |
75d76b |
}
|
|
Packit Service |
75d76b |
|
|
Packit Service |
75d76b |
return VDO_SUCCESS;
|
|
Packit Service |
75d76b |
}
|