libcruft-util/posix/fd.cpp

243 lines
5.3 KiB
C++

/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* Copyright 2016-2019 Danny Robson <danny@nerdcruft.net>
*/
#include "fd.hpp"
#include "except.hpp"
#include "cast.hpp"
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
using cruft::posix::fd;
///////////////////////////////////////////////////////////////////////////////
fd::fd (const std::filesystem::path &path, int flags):
fd (path, flags, 0666)
{ ; }
//-----------------------------------------------------------------------------
fd::fd (const std::filesystem::path &path, int flags, mode_t mode):
m_fd (error::try_value (::open (path.string<char> ().c_str (), flags, mode)))
{
// You always want binary mode. Always.
//
// But we want the user to have considered this platform issue regardless
// so we won't forcibly set the flag, but we will abort the program when
// debugging if they've failed to do so.
CHECK_EQ (flags & O_BINARY, O_BINARY);
}
///////////////////////////////////////////////////////////////////////////////
fd::fd (fd &&rhs) noexcept:
m_fd (-1)
{
std::swap (m_fd, rhs.m_fd);
}
//-----------------------------------------------------------------------------
fd&
fd::operator= (fd &&rhs) noexcept
{
close ();
std::swap (m_fd, rhs.m_fd);
return *this;
}
//-----------------------------------------------------------------------------
fd&
fd::operator= (int rhs)
{
reset (rhs);
return *this;
}
///////////////////////////////////////////////////////////////////////////////
fd::fd (int _fd):
m_fd (_fd)
{ ; }
///////////////////////////////////////////////////////////////////////////////
fd
fd::dup (void) const
{
return dup (m_fd);
}
//-----------------------------------------------------------------------------
fd
fd::dup (int _fd)
{
return fd {
error::try_value (::dup (_fd))
};
}
///////////////////////////////////////////////////////////////////////////////
fd::~fd ()
{
if (m_fd < 0)
return;
close ();
}
///////////////////////////////////////////////////////////////////////////////
struct ::stat
fd::stat (void) const
{
struct stat buf;
error::try_value (fstat (m_fd, &buf));
return buf;
}
///////////////////////////////////////////////////////////////////////////////
#if defined(PLATFORM_LINUX)
void
fd::allocate (int mode, off_t offset, off_t len)
{
error::try_call (fallocate, m_fd, mode, offset, len);
}
#endif
///////////////////////////////////////////////////////////////////////////////
void
fd::close (void)
{
error::try_value (::close (m_fd));
m_fd = -1;
}
//-----------------------------------------------------------------------------
void
fd::reset (void)
{
if (m_fd >= 0) {
close ();
m_fd = -1;
}
}
//-----------------------------------------------------------------------------
void
fd::reset (int rhs)
{
if (m_fd >= 0)
close ();
m_fd = rhs;
}
//-----------------------------------------------------------------------------
int
fd::release (void)
{
int tmp = m_fd;
m_fd = -1;
return tmp;
}
///////////////////////////////////////////////////////////////////////////////
ssize_t
fd::read (void *buffer, std::size_t count)
{
return error::try_value (
::read (m_fd, buffer, cruft::cast::narrow<unsigned> (count))
);
}
//-----------------------------------------------------------------------------
ssize_t
fd::write (const void *buffer, size_t count)
{
return error::try_value (
::write (m_fd, buffer, cruft::cast::narrow<unsigned> (count))
);
}
#if defined(PLATFORM_WIN32)
#include "../win32/windows.hpp"
#endif
//-----------------------------------------------------------------------------
ssize_t
fd::pwrite (const void *buf, size_t count, size_t offset)
{
#if !defined(PLATFORM_WIN32)
return error::try_value (
::pwrite (native (), buf, count, offset)
);
#else
// Ideally we'd to use something like WriteFile with an OVERLAPPED
// structure applied to the handle extracted with _get_osfhandle (m_fd).
//
// But this modifies the file cursor, and if we need to seek anyway then
// we may as well just emulate the entire with a query and two seeks.
//
// NOTE: It is thus possible to observe the fd with an 'incorrect' file
// pointer. Do not rely on POSIX behaviour across multiple threads.
auto const initial = lseek (0, SEEK_CUR);
ssize_t written = -1;
try {
(void)lseek (offset, SEEK_SET);
written = write (buf, count);
} catch (...) {
(void)lseek (initial, SEEK_SET);
throw;
}
(void)lseek (initial, SEEK_SET);
return written;
#endif
}
///////////////////////////////////////////////////////////////////////////////
off_t
fd::lseek (off_t offset, int whence)
{
return error::try_value (
::lseek (m_fd, offset, whence)
);
}
///////////////////////////////////////////////////////////////////////////////
void
fd::truncate (off_t length)
{
error::try_call (
::ftruncate, m_fd, length
);
}
///////////////////////////////////////////////////////////////////////////////
fd::operator int (void) const
{
return m_fd;
}