Blame source/vdo/base/recoveryUtils.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/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
}