Blame source/vdo/base/vioRead.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/vioRead.c#1 $
Packit Service 75d76b
 */
Packit Service 75d76b
Packit Service 75d76b
#include "vioRead.h"
Packit Service 75d76b
Packit Service 75d76b
#include "logger.h"
Packit Service 75d76b
Packit Service 75d76b
#include "blockMap.h"
Packit Service 75d76b
#include "dataVIO.h"
Packit Service 75d76b
#include "vdoInternal.h"
Packit Service 75d76b
#include "vioWrite.h"
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Do the modify-write part of a read-modify-write cycle. This callback is
Packit Service 75d76b
 * registered in readBlock().
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The DataVIO which has just finished its read
Packit Service 75d76b
 **/
Packit Service 75d76b
static void modifyForPartialWrite(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  DataVIO *dataVIO = asDataVIO(completion);
Packit Service 75d76b
  assertInLogicalZone(dataVIO);
Packit Service 75d76b
Packit Service 75d76b
  if (completion->result != VDO_SUCCESS) {
Packit Service 75d76b
    completeDataVIO(completion);
Packit Service 75d76b
    return;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  completion->layer->applyPartialWrite(dataVIO);
Packit Service 75d76b
  VIO *vio = dataVIOAsVIO(dataVIO);
Packit Service 75d76b
  vio->operation = VIO_WRITE | (vio->operation & ~VIO_READ_WRITE_MASK);
Packit Service 75d76b
  dataVIO->isPartialWrite  = true;
Packit Service 75d76b
  launchWriteDataVIO(dataVIO);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Read a block asynchronously. This is the callback registered in
Packit Service 75d76b
 * readBlockMapping().
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The DataVIO to read
Packit Service 75d76b
 **/
Packit Service 75d76b
static void readBlock(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  if (completion->result != VDO_SUCCESS) {
Packit Service 75d76b
    completeDataVIO(completion);
Packit Service 75d76b
    return;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  DataVIO *dataVIO = asDataVIO(completion);
Packit Service 75d76b
  VIO     *vio     = asVIO(completion);
Packit Service 75d76b
  completion->callback
Packit Service 75d76b
    = (isReadVIO(vio) ? completeDataVIO : modifyForPartialWrite);
Packit Service 75d76b
Packit Service 75d76b
  if (dataVIO->mapped.pbn == ZERO_BLOCK) {
Packit Service 75d76b
    completion->layer->zeroDataVIO(dataVIO);
Packit Service 75d76b
    invokeCallback(completion);
Packit Service 75d76b
    return;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  vio->physical = dataVIO->mapped.pbn;
Packit Service 75d76b
  dataVIO->lastAsyncOperation = READ_DATA;
Packit Service 75d76b
  completion->layer->readData(dataVIO);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Read the DataVIO's mapping from the block map. This callback is registered
Packit Service 75d76b
 * in launchReadDataVIO().
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The DataVIO to be read
Packit Service 75d76b
 **/
Packit Service 75d76b
static void readBlockMapping(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  if (completion->result != VDO_SUCCESS) {
Packit Service 75d76b
    completeDataVIO(completion);
Packit Service 75d76b
    return;
Packit Service 75d76b
  }
Packit Service 75d76b
Packit Service 75d76b
  DataVIO *dataVIO = asDataVIO(completion);
Packit Service 75d76b
  assertInLogicalZone(dataVIO);
Packit Service 75d76b
  setLogicalCallback(dataVIO, readBlock, THIS_LOCATION("$F;cb=readBlock"));
Packit Service 75d76b
  dataVIO->lastAsyncOperation = GET_MAPPED_BLOCK;
Packit Service 75d76b
  getMappedBlockAsync(dataVIO);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**********************************************************************/
Packit Service 75d76b
void launchReadDataVIO(DataVIO *dataVIO)
Packit Service 75d76b
{
Packit Service 75d76b
  assertInLogicalZone(dataVIO);
Packit Service 75d76b
  dataVIO->lastAsyncOperation = FIND_BLOCK_MAP_SLOT;
Packit Service 75d76b
  // Go find the block map slot for the LBN mapping.
Packit Service 75d76b
  findBlockMapSlotAsync(dataVIO, readBlockMapping,
Packit Service 75d76b
                        getLogicalZoneThreadID(dataVIO->logical.zone));
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Release the logical block lock which a read DataVIO obtained now that it
Packit Service 75d76b
 * is done.
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param completion  The DataVIO
Packit Service 75d76b
 **/
Packit Service 75d76b
static void releaseLogicalLock(VDOCompletion *completion)
Packit Service 75d76b
{
Packit Service 75d76b
  DataVIO *dataVIO = asDataVIO(completion);
Packit Service 75d76b
  assertInLogicalZone(dataVIO);
Packit Service 75d76b
  releaseLogicalBlockLock(dataVIO);
Packit Service 75d76b
  vioDoneCallback(completion);
Packit Service 75d76b
}
Packit Service 75d76b
Packit Service 75d76b
/**
Packit Service 75d76b
 * Clean up a DataVIO which has finished processing a read.
Packit Service 75d76b
 *
Packit Service 75d76b
 * @param dataVIO  The DataVIO to clean up
Packit Service 75d76b
 **/
Packit Service 75d76b
void cleanupReadDataVIO(DataVIO *dataVIO)
Packit Service 75d76b
{
Packit Service 75d76b
  launchLogicalCallback(dataVIO, releaseLogicalLock,
Packit Service 75d76b
                        THIS_LOCATION("$F;cb=releaseLL"));
Packit Service 75d76b
}