// 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: BackupClientInodeToIDMap.cpp
// Purpose: Map of inode numbers to file IDs on the store
// Created: 11/11/03
//
// --------------------------------------------------------------------------
#include "Box.h"
#ifdef HAVE_DB
// Include db headers and other OS files if they're needed for the disc implementation
#include <sys/types.h>
#include <fcntl.h>
#include <limits.h>
#include <db.h>
#include <sys/stat.h>
#endif
#define BACKIPCLIENTINODETOIDMAP_IMPLEMENTATION
#include "BackupClientInodeToIDMap.h"
#include "BackupStoreException.h"
#include "MemLeakFindOn.h"
// What type of Berkeley DB shall we use?
#define TABLE_DATABASE_TYPE DB_HASH
typedef struct
{
int64_t mObjectID;
int64_t mInDirectory;
} IDBRecord;
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::BackupClientInodeToIDMap()
// Purpose: Constructor
// Created: 11/11/03
//
// --------------------------------------------------------------------------
BackupClientInodeToIDMap::BackupClientInodeToIDMap()
#ifndef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
: mReadOnly(true),
mEmpty(false),
dbp(0)
#endif
{
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::~BackupClientInodeToIDMap()
// Purpose: Destructor
// Created: 11/11/03
//
// --------------------------------------------------------------------------
BackupClientInodeToIDMap::~BackupClientInodeToIDMap()
{
#ifndef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
if(dbp != 0)
{
#if BDB_VERSION_MAJOR >= 3
dbp->close(0);
#else
dbp->close(dbp);
#endif
}
#endif
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::Open(const char *, bool, bool)
// Purpose: Open the database map, creating a file on disc to store everything
// Created: 20/11/03
//
// --------------------------------------------------------------------------
void BackupClientInodeToIDMap::Open(const char *Filename, bool ReadOnly, bool CreateNew)
{
#ifndef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
// Correct arguments?
ASSERT(!(CreateNew && ReadOnly));
// Correct usage?
ASSERT(dbp == 0);
ASSERT(!mEmpty);
// Open the database file
#if BDB_VERSION_MAJOR >= 3
dbp = new Db(0,0);
dbp->set_pagesize(1024); /* Page size: 1K. */
dbp->set_cachesize(0, 32 * 1024, 0);
dbp->open(NULL, Filename, NULL, DB_HASH, DB_CREATE, 0664);
#else
dbp = dbopen(Filename, (CreateNew?O_CREAT:0) | (ReadOnly?O_RDONLY:O_RDWR), S_IRUSR | S_IWUSR | S_IRGRP, TABLE_DATABASE_TYPE, NULL);
#endif
if(dbp == NULL)
{
THROW_EXCEPTION(BackupStoreException, BerkelyDBFailure);
}
// Read only flag
mReadOnly = ReadOnly;
#endif
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::OpenEmpty()
// Purpose: 'Open' this map. Not associated with a disc file. Useful for when a map
// is required, but is against an empty file on disc which shouldn't be created.
// Implies read only.
// Created: 20/11/03
//
// --------------------------------------------------------------------------
void BackupClientInodeToIDMap::OpenEmpty()
{
#ifndef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
ASSERT(dbp == 0);
mEmpty = true;
mReadOnly = true;
#endif
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::Close()
// Purpose: Close the database file
// Created: 20/11/03
//
// --------------------------------------------------------------------------
void BackupClientInodeToIDMap::Close()
{
#ifndef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
if(dbp != 0)
{
#if BDB_VERSION_MAJOR >= 3
if(dbp->close(0) != 0)
#else
if(dbp->close(dbp) != 0)
#endif
{
THROW_EXCEPTION(BackupStoreException, BerkelyDBFailure);
}
dbp = 0;
}
#endif
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::AddToMap(InodeRefType, int64_t, int64_t)
// Purpose: Adds an entry to the map. Overwrites any existing entry.
// Created: 11/11/03
//
// --------------------------------------------------------------------------
void BackupClientInodeToIDMap::AddToMap(InodeRefType InodeRef, int64_t ObjectID, int64_t InDirectory)
{
#ifdef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
mMap[InodeRef] = std::pair<int64_t, int64_t>(ObjectID, InDirectory);
#else
if(mReadOnly)
{
THROW_EXCEPTION(BackupStoreException, InodeMapIsReadOnly);
}
if(dbp == 0)
{
THROW_EXCEPTION(BackupStoreException, InodeMapNotOpen);
}
// Setup structures
IDBRecord rec;
rec.mObjectID = ObjectID;
rec.mInDirectory = InDirectory;
#if BDB_VERSION_MAJOR >= 3
Dbt key(&InodeRef, sizeof(InodeRef));
Dbt data(&rec, sizeof(rec));
if (dbp->put(0, &key, &data, 0) != 0) {
THROW_EXCEPTION(BackupStoreException, BerkelyDBFailure);
}
#else
DBT key;
key.data = &InodeRef;
key.size = sizeof(InodeRef);
DBT data;
data.data = &rec;
data.size = sizeof(rec);
// Add to map (or replace existing entry)
if(dbp->put(dbp, &key, &data, 0) != 0)
{
THROW_EXCEPTION(BackupStoreException, BerkelyDBFailure);
}
#endif
#endif
}
// --------------------------------------------------------------------------
//
// Function
// Name: BackupClientInodeToIDMap::Lookup(InodeRefType, int64_t &, int64_t &) const
// Purpose: Looks up an inode in the map, returning true if it exists, and the object
// ids of it and the directory it's in the reference arguments.
// Created: 11/11/03
//
// --------------------------------------------------------------------------
bool BackupClientInodeToIDMap::Lookup(InodeRefType InodeRef, int64_t &rObjectIDOut, int64_t &rInDirectoryOut) const
{
#ifdef BACKIPCLIENTINODETOIDMAP_IN_MEMORY_IMPLEMENTATION
std::map<InodeRefType, std::pair<int64_t, int64_t> >::const_iterator i(mMap.find(InodeRef));
// Found?
if(i == mMap.end())
{
return false;
}
// Yes. Return the details
rObjectIDOut = i->second.first;
rInDirectoryOut = i->second.second;
return true;
#else
if(mEmpty)
{
// Map is empty
return false;
}
if(dbp == 0)
{
THROW_EXCEPTION(BackupStoreException, InodeMapNotOpen);
}
#if BDB_VERSION_MAJOR >= 3
Dbt key(&InodeRef, sizeof(InodeRef));
Dbt data(0, 0);
switch(dbp->get(NULL, &key, &data, 0))
#else
DBT key;
key.data = &InodeRef;
key.size = sizeof(InodeRef);
DBT data;
data.data = 0;
data.size = 0;
switch(dbp->get(dbp, &key, &data, 0))
#endif
{
case 1: // key not in file
return false;
case -1: // error
default: // not specified in docs
THROW_EXCEPTION(BackupStoreException, BerkelyDBFailure);
return false;
case 0: // success, found it
break;
}
// Check for sensible return
#if BDB_VERSION_MAJOR >= 3
if(key.get_data() == 0 || data.get_size() != sizeof(IDBRecord))
{
// Assert in debug version
ASSERT(key.get_data() == 0 || data.get_size() != sizeof(IDBRecord));
// Invalid entries mean it wasn't found
return false;
}
// Data alignment isn't guaranteed to be on a suitable boundary
IDBRecord rec;
::memcpy(&rec, data.get_data(), sizeof(rec));
#else
if(key.data == 0 || data.size != sizeof(IDBRecord))
{
// Assert in debug version
ASSERT(key.data == 0 || data.size != sizeof(IDBRecord));
// Invalid entries mean it wasn't found
return false;
}
// Data alignment isn't guaranteed to be on a suitable boundary
IDBRecord rec;
::memcpy(&rec, data.data, sizeof(rec));
#endif
// Return data
rObjectIDOut = rec.mObjectID;
rInDirectoryOut = rec.mInDirectory;
// Don't have to worry about freeing the returned data
// Found
return true;
#endif
}
syntax highlighted by Code2HTML, v. 0.9.1