// distribution boxbackup-0.10 (svn version: 494)
//
// Copyright (c) 2003 - 2006
// Ben Summers and contributors. All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions
// are met:
// 1. Redistributions of source code must retain the above copyright
// notice, this list of conditions and the following disclaimer.
// 2. Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
// 3. All use of this software and associated advertising materials must
// display the following acknowledgment:
// This product includes software developed by Ben Summers.
// 4. The names of the Authors may not be used to endorse or promote
// products derived from this software without specific prior written
// permission.
//
// [Where legally impermissible the Authors do not disclaim liability for
// direct physical injury or death caused solely by defects in the software
// unless it is modified by a third party.]
//
// THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR
// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
// WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT,
// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
// (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
// HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
// ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
//
//
// --------------------------------------------------------------------------
//
// File
// Name: RaidFileUtil.cpp
// Purpose: Utilities for raid files
// Created: 2003/07/11
//
// --------------------------------------------------------------------------
#include "Box.h"
#include <sys/types.h>
#include <sys/stat.h>
#include "RaidFileUtil.h"
#include "FileModificationTime.h"
#include "RaidFileRead.h" // for type definition
#include "MemLeakFindOn.h"
// --------------------------------------------------------------------------
//
// Function
// Name: RaidFileUtil::RaidFileExists(RaidFileDiscSet &, const std::string &)
// Purpose: Check to see the state of a RaidFile on disc (doesn't look at contents,
// just at existense of files)
// Created: 2003/07/11
//
// --------------------------------------------------------------------------
RaidFileUtil::ExistType RaidFileUtil::RaidFileExists(RaidFileDiscSet &rDiscSet, const std::string &rFilename, int *pStartDisc, int *pExisitingFiles, int64_t *pRevisionID)
{
if(pExisitingFiles)
{
*pExisitingFiles = 0;
}
// For stat call, although the results are not examined
struct stat st;
// check various files
int startDisc = 0;
{
std::string writeFile(RaidFileUtil::MakeWriteFileName(rDiscSet, rFilename, &startDisc));
if(pStartDisc)
{
*pStartDisc = startDisc;
}
if(::stat(writeFile.c_str(), &st) == 0)
{
// write file exists, use that
// Get unique ID
if(pRevisionID != 0)
{
(*pRevisionID) = FileModificationTime(st);
#ifndef HAVE_STRUCT_STAT_ST_MTIMESPEC
// On linux, the time resolution is very low for modification times.
// So add the size to it to give a bit more chance of it changing.
// TODO: Make this better.
(*pRevisionID) += st.st_size;
#endif
}
// return non-raid file
return NonRaid;
}
}
// Now see how many of the raid components exist
int64_t revisionID = 0;
int setSize = rDiscSet.size();
int rfCount = 0;
#ifndef HAVE_STRUCT_STAT_ST_MTIMESPEC
// TODO: replace this with better linux revision ID detection
int64_t revisionIDplus = 0;
#endif
for(int f = 0; f < setSize; ++f)
{
std::string componentFile(RaidFileUtil::MakeRaidComponentName(rDiscSet, rFilename, (f + startDisc) % setSize));
if(::stat(componentFile.c_str(), &st) == 0)
{
// Component file exists, add to count
rfCount++;
// Set flags for existance?
if(pExisitingFiles)
{
(*pExisitingFiles) |= (1 << f);
}
// Revision ID
if(pRevisionID != 0)
{
int64_t rid = FileModificationTime(st);
if(rid > revisionID) revisionID = rid;
#ifndef HAVE_STRUCT_STAT_ST_MTIMESPEC
revisionIDplus += st.st_size;
#endif
}
}
}
if(pRevisionID != 0)
{
(*pRevisionID) = revisionID;
#ifndef HAVE_STRUCT_STAT_ST_MTIMESPEC
(*pRevisionID) += revisionIDplus;
#endif
}
// Return a status based on how many parts are available
if(rfCount == setSize)
{
return AsRaid;
}
else if((setSize > 1) && rfCount == (setSize - 1))
{
return AsRaidWithMissingReadable;
}
else if(rfCount > 0)
{
return AsRaidWithMissingNotRecoverable;
}
return NoFile; // Obviously doesn't exist
}
// --------------------------------------------------------------------------
//
// Function
// Name: RaidFileUtil::DiscUsageInBlocks(int64_t, const RaidFileDiscSet &)
// Purpose: Returns the size of the file in blocks, given the file size and disc set
// Created: 2003/09/03
//
// --------------------------------------------------------------------------
int64_t RaidFileUtil::DiscUsageInBlocks(int64_t FileSize, const RaidFileDiscSet &rDiscSet)
{
// Get block size
int blockSize = rDiscSet.GetBlockSize();
// OK... so as the size of the file is always sizes of stripe1 + stripe2, we can
// do a very simple calculation for the main data.
int64_t blocks = (FileSize + (((int64_t)blockSize) - 1)) / ((int64_t)blockSize);
// It's just that simple calculation for non-RAID disc sets
if(rDiscSet.IsNonRaidSet())
{
return blocks;
}
// It's the parity which is mildly complex.
// First of all, add in size for all but the last two blocks.
int64_t parityblocks = (FileSize / ((int64_t)blockSize)) / 2;
blocks += parityblocks;
// Work out how many bytes are left
int bytesOver = (int)(FileSize - (parityblocks * ((int64_t)(blockSize*2))));
// Then... (let compiler optimise this out)
if(bytesOver == 0)
{
// Extra block for the size info
blocks++;
}
else if(bytesOver == sizeof(RaidFileRead::FileSizeType))
{
// For last block of parity, plus the size info
blocks += 2;
}
else if(bytesOver < blockSize)
{
// Just want the parity block
blocks += 1;
}
else if(bytesOver == blockSize || bytesOver >= ((blockSize*2)-((int)sizeof(RaidFileRead::FileSizeType))))
{
// Last block, plus size info
blocks += 2;
}
else
{
// Just want parity block
blocks += 1;
}
return blocks;
}
syntax highlighted by Code2HTML, v. 0.9.1