2010-03-17 03:42:27 +00:00
|
|
|
#include <algorithm>
|
2010-03-15 03:05:48 +00:00
|
|
|
#include <cerrno>
|
2010-03-17 03:42:27 +00:00
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
#include <sys/stat.h>
|
|
|
|
|
2010-03-17 03:42:27 +00:00
|
|
|
#include <File/MappedFile.h>
|
|
|
|
#include <ProFUSE/Exception.h>
|
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
|
2010-03-17 03:42:27 +00:00
|
|
|
using ProFUSE::POSIXException;
|
2010-03-15 03:05:48 +00:00
|
|
|
|
|
|
|
MappedFile::MappedFile()
|
|
|
|
{
|
|
|
|
_length = -1;
|
|
|
|
_address = MAP_FAILED;
|
2010-05-18 02:58:26 +00:00
|
|
|
_readOnly = true;
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MappedFile::MappedFile(MappedFile &mf)
|
|
|
|
{
|
|
|
|
_address = mf._address;
|
|
|
|
_length = mf._length;
|
2010-05-18 02:58:26 +00:00
|
|
|
_readOnly = mf._readOnly;
|
2010-03-15 03:05:48 +00:00
|
|
|
|
|
|
|
mf._address = MAP_FAILED;
|
|
|
|
mf._length = -1;
|
2010-05-18 02:58:26 +00:00
|
|
|
mf._readOnly = true;
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|
|
|
|
|
2010-05-18 02:58:26 +00:00
|
|
|
MappedFile::MappedFile(const File &f, bool readOnly, size_t size)
|
2010-03-15 03:05:48 +00:00
|
|
|
{
|
2010-05-18 02:58:26 +00:00
|
|
|
_length = -1;
|
|
|
|
_address = MAP_FAILED;
|
|
|
|
_readOnly = readOnly;
|
|
|
|
|
|
|
|
init(f, readOnly, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MappedFile::MappedFile(const char *name, bool readOnly)
|
|
|
|
{
|
|
|
|
File f(name, readOnly);
|
|
|
|
|
|
|
|
_length = -1;
|
|
|
|
_address = MAP_FAILED;
|
|
|
|
_readOnly = readOnly;
|
|
|
|
|
2010-05-19 16:06:42 +00:00
|
|
|
init(f, readOnly, 0);
|
2010-05-18 02:58:26 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-03-17 03:42:27 +00:00
|
|
|
|
2010-05-18 02:58:26 +00:00
|
|
|
|
|
|
|
MappedFile::~MappedFile()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void MappedFile::init(const File &f, bool readOnly, size_t size)
|
|
|
|
{
|
|
|
|
#undef __METHOD__
|
|
|
|
#define __METHOD__ "MappedFile::init"
|
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
struct stat st;
|
2010-05-19 16:06:42 +00:00
|
|
|
int prot = readOnly ? PROT_READ : PROT_READ | PROT_WRITE;
|
|
|
|
int flags = MAP_FILE | MAP_SHARED;
|
2010-03-15 03:05:48 +00:00
|
|
|
|
|
|
|
// close enough
|
|
|
|
if (f.fd() < 0)
|
2010-03-17 03:42:27 +00:00
|
|
|
throw POSIXException( __METHOD__, EBADF);
|
2010-03-15 03:05:48 +00:00
|
|
|
|
2010-05-18 02:58:26 +00:00
|
|
|
|
2010-05-19 16:06:42 +00:00
|
|
|
if (!size)
|
2010-05-18 02:58:26 +00:00
|
|
|
{
|
|
|
|
if (::fstat(f.fd(), &st) != 0)
|
|
|
|
throw POSIXException(__METHOD__ ": fstat", errno);
|
|
|
|
|
|
|
|
if (!S_ISREG(st.st_mode))
|
|
|
|
throw POSIXException(__METHOD__, ENODEV);
|
2010-03-15 03:05:48 +00:00
|
|
|
|
2010-05-18 02:58:26 +00:00
|
|
|
size = st.st_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
_length = size;
|
2010-05-19 16:06:42 +00:00
|
|
|
_address = ::mmap(0, _length, prot, flags, f.fd(), 0);
|
2010-05-18 02:58:26 +00:00
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
if (_address == MAP_FAILED)
|
2010-03-17 03:42:27 +00:00
|
|
|
throw POSIXException(__METHOD__ ": mmap", errno);
|
2010-05-18 02:58:26 +00:00
|
|
|
|
|
|
|
_readOnly = readOnly;
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-03-17 03:42:27 +00:00
|
|
|
void MappedFile::close()
|
2010-03-15 03:05:48 +00:00
|
|
|
{
|
2010-03-17 03:42:27 +00:00
|
|
|
#undef __METHOD__
|
|
|
|
#define __METHOD__ "MappedFile::close"
|
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
if (_address != MAP_FAILED)
|
|
|
|
{
|
|
|
|
void *address = _address;
|
|
|
|
size_t length = _length;
|
|
|
|
|
|
|
|
_address = MAP_FAILED;
|
|
|
|
_length = -1;
|
2010-05-18 02:58:26 +00:00
|
|
|
_readOnly = true;
|
2010-03-15 03:05:48 +00:00
|
|
|
|
|
|
|
if (::munmap(address, length) != 0)
|
2010-03-17 03:42:27 +00:00
|
|
|
throw POSIXException(__METHOD__ ": munmap", errno);
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-17 03:42:27 +00:00
|
|
|
void MappedFile::sync()
|
2010-03-15 03:05:48 +00:00
|
|
|
{
|
2010-03-17 03:42:27 +00:00
|
|
|
#undef __METHOD__
|
|
|
|
#define __METHOD__ "MappedFile::sync"
|
|
|
|
|
2010-03-15 03:05:48 +00:00
|
|
|
if (_address != MAP_FAILED)
|
|
|
|
{
|
|
|
|
if (::msync(_address, _length, MS_SYNC) != 0)
|
2010-03-17 03:42:27 +00:00
|
|
|
throw POSIXException(__METHOD__ ": msync", errno);
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|
2010-03-17 03:42:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MappedFile::adopt(MappedFile &mf)
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
_address = mf._address;
|
|
|
|
_length = mf._length;
|
2010-05-18 02:58:26 +00:00
|
|
|
_readOnly = mf._readOnly;
|
2010-03-17 03:42:27 +00:00
|
|
|
|
|
|
|
mf._address = MAP_FAILED;
|
|
|
|
mf._length = -1;
|
2010-05-18 02:58:26 +00:00
|
|
|
mf._readOnly = true;
|
2010-03-17 03:42:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MappedFile::swap(MappedFile &mf)
|
|
|
|
{
|
|
|
|
std::swap(_address, mf._address);
|
|
|
|
std::swap(_length, mf._length);
|
2010-05-18 02:58:26 +00:00
|
|
|
std::swap(_readOnly, mf._readOnly);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-19 01:07:47 +00:00
|
|
|
MappedFile *MappedFile::Create(const char *name, size_t size)
|
2010-05-18 02:58:26 +00:00
|
|
|
{
|
|
|
|
#undef __METHOD__
|
|
|
|
#define __METHOD__ "MappedFile::Create"
|
|
|
|
|
|
|
|
File fd(::open(name, O_CREAT | O_TRUNC | O_RDWR, 0644));
|
|
|
|
|
|
|
|
if (!fd.isValid())
|
|
|
|
{
|
|
|
|
throw POSIXException(__METHOD__ ": Unable to create file.", errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO -- is ftruncate portable?
|
|
|
|
if (::ftruncate(fd.fd(), size) < 0)
|
|
|
|
{
|
|
|
|
// TODO -- unlink?
|
|
|
|
throw POSIXException(__METHOD__ ": Unable to truncate file.", errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new MappedFile(fd, false, size);
|
2010-03-15 03:05:48 +00:00
|
|
|
}
|