system/FileDescriptor: new wrapper class for a file descriptor
This commit is contained in:
parent
818d729d8b
commit
40a587bbaf
@ -57,9 +57,9 @@ src_mpd_LDADD = \
|
|||||||
libevent.a \
|
libevent.a \
|
||||||
libthread.a \
|
libthread.a \
|
||||||
libnet.a \
|
libnet.a \
|
||||||
|
$(FS_LIBS) \
|
||||||
libsystem.a \
|
libsystem.a \
|
||||||
libutil.a \
|
libutil.a \
|
||||||
$(FS_LIBS) \
|
|
||||||
$(ICU_LDADD) \
|
$(ICU_LDADD) \
|
||||||
$(SYSTEMD_DAEMON_LIBS) \
|
$(SYSTEMD_DAEMON_LIBS) \
|
||||||
$(GLIB_LIBS)
|
$(GLIB_LIBS)
|
||||||
@ -431,6 +431,7 @@ libnet_a_SOURCES = \
|
|||||||
libsystem_a_SOURCES = \
|
libsystem_a_SOURCES = \
|
||||||
src/system/ByteOrder.hxx \
|
src/system/ByteOrder.hxx \
|
||||||
src/system/FatalError.cxx src/system/FatalError.hxx \
|
src/system/FatalError.cxx src/system/FatalError.hxx \
|
||||||
|
src/system/FileDescriptor.cxx src/system/FileDescriptor.hxx \
|
||||||
src/system/fd_util.c src/system/fd_util.h \
|
src/system/fd_util.c src/system/fd_util.h \
|
||||||
src/system/EventPipe.cxx src/system/EventPipe.hxx \
|
src/system/EventPipe.cxx src/system/EventPipe.hxx \
|
||||||
src/system/EventFD.cxx src/system/EventFD.hxx \
|
src/system/EventFD.cxx src/system/EventFD.hxx \
|
||||||
|
@ -21,7 +21,7 @@
|
|||||||
#include "InotifySource.hxx"
|
#include "InotifySource.hxx"
|
||||||
#include "InotifyDomain.hxx"
|
#include "InotifyDomain.hxx"
|
||||||
#include "util/Error.hxx"
|
#include "util/Error.hxx"
|
||||||
#include "system/fd_util.h"
|
#include "system/FileDescriptor.hxx"
|
||||||
#include "system/FatalError.hxx"
|
#include "system/FatalError.hxx"
|
||||||
#include "Log.hxx"
|
#include "Log.hxx"
|
||||||
|
|
||||||
@ -72,8 +72,8 @@ InotifySource::OnSocketReady(gcc_unused unsigned flags)
|
|||||||
inline
|
inline
|
||||||
InotifySource::InotifySource(EventLoop &_loop,
|
InotifySource::InotifySource(EventLoop &_loop,
|
||||||
mpd_inotify_callback_t _callback, void *_ctx,
|
mpd_inotify_callback_t _callback, void *_ctx,
|
||||||
int _fd)
|
FileDescriptor _fd)
|
||||||
:SocketMonitor(_fd, _loop),
|
:SocketMonitor(_fd.Get(), _loop),
|
||||||
callback(_callback), callback_ctx(_ctx)
|
callback(_callback), callback_ctx(_ctx)
|
||||||
{
|
{
|
||||||
ScheduleRead();
|
ScheduleRead();
|
||||||
@ -85,8 +85,8 @@ InotifySource::Create(EventLoop &loop,
|
|||||||
mpd_inotify_callback_t callback, void *callback_ctx,
|
mpd_inotify_callback_t callback, void *callback_ctx,
|
||||||
Error &error)
|
Error &error)
|
||||||
{
|
{
|
||||||
int fd = inotify_init_cloexec();
|
FileDescriptor fd;
|
||||||
if (fd < 0) {
|
if (!fd.CreateInotify()) {
|
||||||
error.SetErrno("inotify_init() has failed");
|
error.SetErrno("inotify_init() has failed");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
#include "Compiler.h"
|
#include "Compiler.h"
|
||||||
|
|
||||||
class Error;
|
class Error;
|
||||||
|
class FileDescriptor;
|
||||||
|
|
||||||
typedef void (*mpd_inotify_callback_t)(int wd, unsigned mask,
|
typedef void (*mpd_inotify_callback_t)(int wd, unsigned mask,
|
||||||
const char *name, void *ctx);
|
const char *name, void *ctx);
|
||||||
@ -33,7 +34,8 @@ class InotifySource final : private SocketMonitor {
|
|||||||
void *callback_ctx;
|
void *callback_ctx;
|
||||||
|
|
||||||
InotifySource(EventLoop &_loop,
|
InotifySource(EventLoop &_loop,
|
||||||
mpd_inotify_callback_t callback, void *ctx, int fd);
|
mpd_inotify_callback_t callback, void *ctx,
|
||||||
|
FileDescriptor fd);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
~InotifySource() {
|
~InotifySource() {
|
||||||
|
@ -75,17 +75,11 @@ FileReader::Close()
|
|||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#include <errno.h>
|
|
||||||
|
|
||||||
FileReader::FileReader(Path _path, Error &error)
|
FileReader::FileReader(Path _path, Error &error)
|
||||||
:path(_path),
|
:path(_path)
|
||||||
fd(OpenFile(path,
|
|
||||||
O_RDONLY,
|
|
||||||
0))
|
|
||||||
{
|
{
|
||||||
if (fd < 0)
|
fd.OpenReadOnly(path.c_str());
|
||||||
|
if (!fd.IsDefined())
|
||||||
error.FormatErrno("Failed to open %s", path.c_str());
|
error.FormatErrno("Failed to open %s", path.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -94,7 +88,7 @@ FileReader::Read(void *data, size_t size, Error &error)
|
|||||||
{
|
{
|
||||||
assert(IsDefined());
|
assert(IsDefined());
|
||||||
|
|
||||||
ssize_t nbytes = read(fd, data, size);
|
ssize_t nbytes = fd.Read(data, size);
|
||||||
if (nbytes < 0) {
|
if (nbytes < 0) {
|
||||||
error.FormatErrno("Failed to read from %s", path.c_str());
|
error.FormatErrno("Failed to read from %s", path.c_str());
|
||||||
nbytes = 0;
|
nbytes = 0;
|
||||||
@ -108,7 +102,7 @@ FileReader::Seek(off_t offset, Error &error)
|
|||||||
{
|
{
|
||||||
assert(IsDefined());
|
assert(IsDefined());
|
||||||
|
|
||||||
auto result = lseek(fd, offset, SEEK_SET);
|
auto result = fd.Seek(offset);
|
||||||
const bool success = result >= 0;
|
const bool success = result >= 0;
|
||||||
if (!success)
|
if (!success)
|
||||||
error.SetErrno("Failed to seek");
|
error.SetErrno("Failed to seek");
|
||||||
@ -121,8 +115,7 @@ FileReader::Close()
|
|||||||
{
|
{
|
||||||
assert(IsDefined());
|
assert(IsDefined());
|
||||||
|
|
||||||
close(fd);
|
fd.Close();
|
||||||
fd = -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -25,6 +25,10 @@
|
|||||||
#include "fs/AllocatedPath.hxx"
|
#include "fs/AllocatedPath.hxx"
|
||||||
#include "Compiler.h"
|
#include "Compiler.h"
|
||||||
|
|
||||||
|
#ifndef WIN32
|
||||||
|
#include "system/FileDescriptor.hxx"
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
@ -39,7 +43,7 @@ class FileReader final : public Reader {
|
|||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
HANDLE handle;
|
HANDLE handle;
|
||||||
#else
|
#else
|
||||||
int fd;
|
FileDescriptor fd;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -55,7 +59,7 @@ public:
|
|||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
return handle != INVALID_HANDLE_VALUE;
|
return handle != INVALID_HANDLE_VALUE;
|
||||||
#else
|
#else
|
||||||
return fd >= 0;
|
return fd.IsDefined();
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -23,10 +23,8 @@
|
|||||||
#include "../InputPlugin.hxx"
|
#include "../InputPlugin.hxx"
|
||||||
#include "util/Error.hxx"
|
#include "util/Error.hxx"
|
||||||
#include "util/Domain.hxx"
|
#include "util/Domain.hxx"
|
||||||
#include "fs/FileSystem.hxx"
|
|
||||||
#include "fs/Path.hxx"
|
#include "fs/Path.hxx"
|
||||||
#include "system/fd_util.h"
|
#include "system/FileDescriptor.hxx"
|
||||||
#include "open.h"
|
|
||||||
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
@ -35,10 +33,10 @@
|
|||||||
static constexpr Domain file_domain("file");
|
static constexpr Domain file_domain("file");
|
||||||
|
|
||||||
class FileInputStream final : public InputStream {
|
class FileInputStream final : public InputStream {
|
||||||
const int fd;
|
FileDescriptor fd;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
FileInputStream(const char *path, int _fd, off_t _size,
|
FileInputStream(const char *path, FileDescriptor _fd, off_t _size,
|
||||||
Mutex &_mutex, Cond &_cond)
|
Mutex &_mutex, Cond &_cond)
|
||||||
:InputStream(path, _mutex, _cond),
|
:InputStream(path, _mutex, _cond),
|
||||||
fd(_fd) {
|
fd(_fd) {
|
||||||
@ -48,7 +46,7 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
~FileInputStream() {
|
~FileInputStream() {
|
||||||
close(fd);
|
fd.Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* virtual methods from InputStream */
|
/* virtual methods from InputStream */
|
||||||
@ -66,29 +64,29 @@ OpenFileInputStream(Path path,
|
|||||||
Mutex &mutex, Cond &cond,
|
Mutex &mutex, Cond &cond,
|
||||||
Error &error)
|
Error &error)
|
||||||
{
|
{
|
||||||
const int fd = OpenFile(path, O_RDONLY|O_BINARY, 0);
|
FileDescriptor fd;
|
||||||
if (fd < 0) {
|
if (!fd.OpenReadOnly(path.c_str())) {
|
||||||
error.FormatErrno("Failed to open \"%s\"",
|
error.FormatErrno("Failed to open \"%s\"",
|
||||||
path.c_str());
|
path.c_str());
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct stat st;
|
struct stat st;
|
||||||
if (fstat(fd, &st) < 0) {
|
if (fstat(fd.Get(), &st) < 0) {
|
||||||
error.FormatErrno("Failed to stat \"%s\"", path.c_str());
|
error.FormatErrno("Failed to stat \"%s\"", path.c_str());
|
||||||
close(fd);
|
fd.Close();
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!S_ISREG(st.st_mode)) {
|
if (!S_ISREG(st.st_mode)) {
|
||||||
error.Format(file_domain, "Not a regular file: %s",
|
error.Format(file_domain, "Not a regular file: %s",
|
||||||
path.c_str());
|
path.c_str());
|
||||||
close(fd);
|
fd.Close();
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef POSIX_FADV_SEQUENTIAL
|
#ifdef POSIX_FADV_SEQUENTIAL
|
||||||
posix_fadvise(fd, (off_t)0, st.st_size, POSIX_FADV_SEQUENTIAL);
|
posix_fadvise(fd.Get(), (off_t)0, st.st_size, POSIX_FADV_SEQUENTIAL);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return new FileInputStream(path.c_str(), fd, st.st_size, mutex, cond);
|
return new FileInputStream(path.c_str(), fd, st.st_size, mutex, cond);
|
||||||
@ -107,7 +105,7 @@ input_file_open(gcc_unused const char *filename,
|
|||||||
bool
|
bool
|
||||||
FileInputStream::Seek(offset_type new_offset, Error &error)
|
FileInputStream::Seek(offset_type new_offset, Error &error)
|
||||||
{
|
{
|
||||||
auto result = lseek(fd, (off_t)new_offset, SEEK_SET);
|
auto result = fd.Seek((off_t)new_offset);
|
||||||
if (result < 0) {
|
if (result < 0) {
|
||||||
error.SetErrno("Failed to seek");
|
error.SetErrno("Failed to seek");
|
||||||
return false;
|
return false;
|
||||||
@ -120,7 +118,7 @@ FileInputStream::Seek(offset_type new_offset, Error &error)
|
|||||||
size_t
|
size_t
|
||||||
FileInputStream::Read(void *ptr, size_t read_size, Error &error)
|
FileInputStream::Read(void *ptr, size_t read_size, Error &error)
|
||||||
{
|
{
|
||||||
ssize_t nbytes = read(fd, ptr, read_size);
|
ssize_t nbytes = fd.Read(ptr, read_size);
|
||||||
if (nbytes < 0) {
|
if (nbytes < 0) {
|
||||||
error.SetErrno("Failed to read");
|
error.SetErrno("Failed to read");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -20,46 +20,35 @@
|
|||||||
#include "config.h"
|
#include "config.h"
|
||||||
#ifdef USE_EVENTFD
|
#ifdef USE_EVENTFD
|
||||||
#include "EventFD.hxx"
|
#include "EventFD.hxx"
|
||||||
#include "system/fd_util.h"
|
|
||||||
#include "system/FatalError.hxx"
|
#include "system/FatalError.hxx"
|
||||||
#include "Compiler.h"
|
#include "Compiler.h"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <unistd.h>
|
|
||||||
|
|
||||||
#include <sys/eventfd.h>
|
#include <sys/eventfd.h>
|
||||||
|
|
||||||
EventFD::EventFD()
|
EventFD::EventFD()
|
||||||
:fd(eventfd_cloexec_nonblock(0, 0))
|
|
||||||
{
|
{
|
||||||
if (fd < 0)
|
if (!fd.CreateEventFD(0))
|
||||||
FatalSystemError("eventfd() failed");
|
FatalSystemError("eventfd() failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
EventFD::~EventFD()
|
|
||||||
{
|
|
||||||
assert(fd >= 0);
|
|
||||||
|
|
||||||
close(fd);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
EventFD::Read()
|
EventFD::Read()
|
||||||
{
|
{
|
||||||
assert(fd >= 0);
|
assert(fd.IsDefined());
|
||||||
|
|
||||||
eventfd_t value;
|
eventfd_t value;
|
||||||
return read(fd, &value, sizeof(value)) == (ssize_t)sizeof(value);
|
return fd.Read(&value, sizeof(value)) == (ssize_t)sizeof(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
EventFD::Write()
|
EventFD::Write()
|
||||||
{
|
{
|
||||||
assert(fd >= 0);
|
assert(fd.IsDefined());
|
||||||
|
|
||||||
static constexpr eventfd_t value = 1;
|
static constexpr eventfd_t value = 1;
|
||||||
gcc_unused ssize_t nbytes =
|
gcc_unused ssize_t nbytes =
|
||||||
write(fd, &value, sizeof(value));
|
fd.Write(&value, sizeof(value));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif /* USE_EVENTFD */
|
#endif /* USE_EVENTFD */
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
#define MPD_EVENT_FD_HXX
|
#define MPD_EVENT_FD_HXX
|
||||||
|
|
||||||
#include "check.h"
|
#include "check.h"
|
||||||
|
#include "FileDescriptor.hxx"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A class that wraps eventfd().
|
* A class that wraps eventfd().
|
||||||
@ -28,17 +29,19 @@
|
|||||||
* Errors in the constructor are fatal.
|
* Errors in the constructor are fatal.
|
||||||
*/
|
*/
|
||||||
class EventFD {
|
class EventFD {
|
||||||
int fd;
|
FileDescriptor fd;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
EventFD();
|
EventFD();
|
||||||
~EventFD();
|
~EventFD() {
|
||||||
|
fd.Close();
|
||||||
|
}
|
||||||
|
|
||||||
EventFD(const EventFD &other) = delete;
|
EventFD(const EventFD &other) = delete;
|
||||||
EventFD &operator=(const EventFD &other) = delete;
|
EventFD &operator=(const EventFD &other) = delete;
|
||||||
|
|
||||||
int Get() const {
|
int Get() const {
|
||||||
return fd;
|
return fd.Get();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
219
src/system/FileDescriptor.cxx
Normal file
219
src/system/FileDescriptor.cxx
Normal file
@ -0,0 +1,219 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2012-2015 Max Kellermann <max@duempel.org>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* - 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.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* ``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
|
||||||
|
* FOUNDATION OR CONTRIBUTORS 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "FileDescriptor.hxx"
|
||||||
|
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX
|
||||||
|
#include <poll.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_EVENTFD
|
||||||
|
#include <sys/eventfd.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_SIGNALFD
|
||||||
|
#include <sys/signalfd.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_INOTIFY_INIT
|
||||||
|
#include <sys/inotify.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef O_NOCTTY
|
||||||
|
#define O_NOCTTY 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef O_CLOEXEC
|
||||||
|
#define O_CLOEXEC 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::Open(const char *pathname, int flags)
|
||||||
|
{
|
||||||
|
assert(!IsDefined());
|
||||||
|
|
||||||
|
fd = ::open(pathname, flags);
|
||||||
|
return IsDefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::OpenReadOnly(const char *pathname)
|
||||||
|
{
|
||||||
|
return Open(pathname, O_RDONLY | O_NOCTTY | O_CLOEXEC);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::OpenNonBlocking(const char *pathname)
|
||||||
|
{
|
||||||
|
return Open(pathname, O_RDWR | O_NOCTTY | O_CLOEXEC | O_NONBLOCK);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::CreatePipe(FileDescriptor &r, FileDescriptor &w)
|
||||||
|
{
|
||||||
|
int fds[2];
|
||||||
|
|
||||||
|
#ifdef HAVE_PIPE2
|
||||||
|
const int flags = O_CLOEXEC;
|
||||||
|
const int result = pipe2(fds, flags);
|
||||||
|
#else
|
||||||
|
const int result = pipe(fds);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (result < 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
r = FileDescriptor(fds[0]);
|
||||||
|
w = FileDescriptor(fds[1]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
FileDescriptor::SetNonBlocking()
|
||||||
|
{
|
||||||
|
assert(IsDefined());
|
||||||
|
|
||||||
|
int flags = fcntl(fd, F_GETFL);
|
||||||
|
fcntl(fd, F_SETFL, flags | O_NONBLOCK);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
FileDescriptor::SetBlocking()
|
||||||
|
{
|
||||||
|
assert(IsDefined());
|
||||||
|
|
||||||
|
int flags = fcntl(fd, F_GETFL);
|
||||||
|
fcntl(fd, F_SETFL, flags & ~O_NONBLOCK);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_EVENTFD
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::CreateEventFD(unsigned initval)
|
||||||
|
{
|
||||||
|
assert(!IsDefined());
|
||||||
|
|
||||||
|
fd = ::eventfd(initval, EFD_NONBLOCK|EFD_CLOEXEC);
|
||||||
|
return fd >= 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_SIGNALFD
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::CreateSignalFD(const sigset_t *mask)
|
||||||
|
{
|
||||||
|
int new_fd = ::signalfd(fd, mask, SFD_NONBLOCK|SFD_CLOEXEC);
|
||||||
|
if (new_fd < 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
fd = new_fd;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_INOTIFY_INIT
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::CreateInotify()
|
||||||
|
{
|
||||||
|
#ifdef HAVE_INOTIFY_INIT1
|
||||||
|
int new_fd = inotify_init1(IN_CLOEXEC|IN_NONBLOCK);
|
||||||
|
#else
|
||||||
|
int new_fd = inotify_init();
|
||||||
|
#endif
|
||||||
|
if (new_fd < 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
#ifndef HAVE_INOTIFY_INIT1
|
||||||
|
SetNonBlocking();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
fd = new_fd;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
bool
|
||||||
|
FileDescriptor::Rewind()
|
||||||
|
{
|
||||||
|
assert(IsDefined());
|
||||||
|
|
||||||
|
return lseek(fd, 0, SEEK_SET) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
off_t
|
||||||
|
FileDescriptor::GetSize() const
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
return ::fstat(fd, &st) >= 0
|
||||||
|
? (long)st.st_size
|
||||||
|
: -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX
|
||||||
|
|
||||||
|
int
|
||||||
|
FileDescriptor::Poll(short events, int timeout) const
|
||||||
|
{
|
||||||
|
assert(IsDefined());
|
||||||
|
|
||||||
|
struct pollfd pfd;
|
||||||
|
pfd.fd = fd;
|
||||||
|
pfd.events = events;
|
||||||
|
int result = poll(&pfd, 1, timeout);
|
||||||
|
return result > 0
|
||||||
|
? pfd.revents
|
||||||
|
: result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
FileDescriptor::WaitReadable(int timeout) const
|
||||||
|
{
|
||||||
|
return Poll(POLLIN, timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
FileDescriptor::WaitWritable(int timeout) const
|
||||||
|
{
|
||||||
|
return Poll(POLLOUT, timeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
171
src/system/FileDescriptor.hxx
Normal file
171
src/system/FileDescriptor.hxx
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2012-2015 Max Kellermann <max@duempel.org>
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions
|
||||||
|
* are met:
|
||||||
|
*
|
||||||
|
* - Redistributions of source code must retain the above copyright
|
||||||
|
* notice, this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* - 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.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* ``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
|
||||||
|
* FOUNDATION OR CONTRIBUTORS 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef FILE_DESCRIPTOR_HXX
|
||||||
|
#define FILE_DESCRIPTOR_HXX
|
||||||
|
|
||||||
|
#include "check.h"
|
||||||
|
#include "Compiler.h"
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#ifdef USE_SIGNALFD
|
||||||
|
#include <signal.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An OO wrapper for a UNIX file descriptor.
|
||||||
|
*
|
||||||
|
* This class is unmanaged and trivial.
|
||||||
|
*/
|
||||||
|
class FileDescriptor {
|
||||||
|
protected:
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FileDescriptor() = default;
|
||||||
|
explicit constexpr FileDescriptor(int _fd):fd(_fd) {}
|
||||||
|
|
||||||
|
constexpr bool operator==(FileDescriptor other) const {
|
||||||
|
return fd == other.fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool IsDefined() const {
|
||||||
|
return fd >= 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the file descriptor. This may only be called if
|
||||||
|
* IsDefined() returns true.
|
||||||
|
*/
|
||||||
|
constexpr int Get() const {
|
||||||
|
return fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Set(int _fd) {
|
||||||
|
fd = _fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
int Steal() {
|
||||||
|
assert(IsDefined());
|
||||||
|
|
||||||
|
int _fd = fd;
|
||||||
|
fd = -1;
|
||||||
|
return _fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetUndefined() {
|
||||||
|
fd = -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static constexpr FileDescriptor Undefined() {
|
||||||
|
return FileDescriptor(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Open(const char *pathname, int flags);
|
||||||
|
bool OpenReadOnly(const char *pathname);
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX
|
||||||
|
bool OpenNonBlocking(const char *pathname);
|
||||||
|
|
||||||
|
static bool CreatePipe(FileDescriptor &r, FileDescriptor &w);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enable non-blocking mode on this file descriptor.
|
||||||
|
*/
|
||||||
|
void SetNonBlocking();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Enable blocking mode on this file descriptor.
|
||||||
|
*/
|
||||||
|
void SetBlocking();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Duplicate the file descriptor onto the given file descriptor.
|
||||||
|
*/
|
||||||
|
bool Duplicate(int new_fd) const {
|
||||||
|
return ::dup2(Get(), new_fd) == 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_EVENTFD
|
||||||
|
bool CreateEventFD(unsigned initval=0);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_SIGNALFD
|
||||||
|
bool CreateSignalFD(const sigset_t *mask);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HAVE_INOTIFY_INIT
|
||||||
|
bool CreateInotify();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Close the file descriptor. It is legal to call it on an
|
||||||
|
* "undefined" object. After this call, IsDefined() is guaranteed
|
||||||
|
* to return false, and this object may be reused.
|
||||||
|
*/
|
||||||
|
void Close() {
|
||||||
|
::close(Steal());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Rewind the pointer to the beginning of the file.
|
||||||
|
*/
|
||||||
|
bool Rewind();
|
||||||
|
|
||||||
|
off_t Seek(off_t offset) {
|
||||||
|
return lseek(Get(), offset, SEEK_SET);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the size of the file in bytes, or -1 on error.
|
||||||
|
*/
|
||||||
|
gcc_pure
|
||||||
|
off_t GetSize() const;
|
||||||
|
|
||||||
|
ssize_t Read(void *buffer, size_t length) {
|
||||||
|
return ::read(fd, buffer, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t Write(const void *buffer, size_t length) {
|
||||||
|
return ::write(fd, buffer, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef HAVE_POSIX
|
||||||
|
int Poll(short events, int timeout) const;
|
||||||
|
|
||||||
|
int WaitReadable(int timeout) const;
|
||||||
|
int WaitWritable(int timeout) const;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@ -23,33 +23,29 @@
|
|||||||
#include "FatalError.hxx"
|
#include "FatalError.hxx"
|
||||||
|
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
#include <unistd.h>
|
|
||||||
#include <sys/signalfd.h>
|
#include <sys/signalfd.h>
|
||||||
|
|
||||||
void
|
void
|
||||||
SignalFD::Create(const sigset_t &mask)
|
SignalFD::Create(const sigset_t &mask)
|
||||||
{
|
{
|
||||||
fd = ::signalfd(fd, &mask, SFD_NONBLOCK|SFD_CLOEXEC);
|
if (!fd.CreateSignalFD(&mask))
|
||||||
if (fd < 0)
|
|
||||||
FatalSystemError("signalfd() failed");
|
FatalSystemError("signalfd() failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
SignalFD::Close()
|
SignalFD::Close()
|
||||||
{
|
{
|
||||||
if (fd >= 0) {
|
if (fd.IsDefined())
|
||||||
::close(fd);
|
fd.Close();
|
||||||
fd = -1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
SignalFD::Read()
|
SignalFD::Read()
|
||||||
{
|
{
|
||||||
assert(fd >= 0);
|
assert(fd.IsDefined());
|
||||||
|
|
||||||
signalfd_siginfo info;
|
signalfd_siginfo info;
|
||||||
return read(fd, &info, sizeof(info)) > 0
|
return fd.Read(&info, sizeof(info)) > 0
|
||||||
? info.ssi_signo
|
? info.ssi_signo
|
||||||
: -1;
|
: -1;
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,7 @@
|
|||||||
#define MPD_SIGNAL_FD_HXX
|
#define MPD_SIGNAL_FD_HXX
|
||||||
|
|
||||||
#include "check.h"
|
#include "check.h"
|
||||||
|
#include "FileDescriptor.hxx"
|
||||||
|
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
|
||||||
@ -28,7 +29,7 @@
|
|||||||
* A class that wraps signalfd().
|
* A class that wraps signalfd().
|
||||||
*/
|
*/
|
||||||
class SignalFD {
|
class SignalFD {
|
||||||
int fd;
|
FileDescriptor fd;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SignalFD():fd(-1) {}
|
SignalFD():fd(-1) {}
|
||||||
@ -48,7 +49,7 @@ public:
|
|||||||
void Close();
|
void Close();
|
||||||
|
|
||||||
int Get() const {
|
int Get() const {
|
||||||
return fd;
|
return fd.Get();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -41,10 +41,6 @@
|
|||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_INOTIFY_INIT
|
|
||||||
#include <sys/inotify.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_EVENTFD
|
#ifdef USE_EVENTFD
|
||||||
#include <sys/eventfd.h>
|
#include <sys/eventfd.h>
|
||||||
#endif
|
#endif
|
||||||
@ -200,38 +196,6 @@ accept_cloexec_nonblock(int fd, struct sockaddr *address,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef HAVE_INOTIFY_INIT
|
|
||||||
|
|
||||||
int
|
|
||||||
inotify_init_cloexec(void)
|
|
||||||
{
|
|
||||||
int fd;
|
|
||||||
|
|
||||||
#ifdef HAVE_INOTIFY_INIT1
|
|
||||||
fd = inotify_init1(IN_CLOEXEC);
|
|
||||||
if (fd >= 0 || errno != ENOSYS)
|
|
||||||
return fd;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
fd = inotify_init();
|
|
||||||
if (fd >= 0)
|
|
||||||
fd_set_cloexec(fd, true);
|
|
||||||
|
|
||||||
return fd;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_EVENTFD
|
|
||||||
|
|
||||||
int
|
|
||||||
eventfd_cloexec_nonblock(unsigned initval, int flags)
|
|
||||||
{
|
|
||||||
return eventfd(initval, flags | EFD_CLOEXEC | EFD_NONBLOCK);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
int
|
int
|
||||||
close_socket(int fd)
|
close_socket(int fd)
|
||||||
{
|
{
|
||||||
|
@ -91,28 +91,6 @@ int
|
|||||||
accept_cloexec_nonblock(int fd, struct sockaddr *address,
|
accept_cloexec_nonblock(int fd, struct sockaddr *address,
|
||||||
size_t *address_length_r);
|
size_t *address_length_r);
|
||||||
|
|
||||||
#ifdef HAVE_INOTIFY_INIT
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper for inotify_init(), which sets the CLOEXEC flag (atomically
|
|
||||||
* if supported by the OS).
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
inotify_init_cloexec(void);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_EVENTFD
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Wrapper for eventfd() which sets the flags CLOEXEC and NONBLOCK
|
|
||||||
* flag (atomically if supported by the OS).
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
eventfd_cloexec_nonblock(unsigned initval, int flags);
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Portable wrapper for close(); use closesocket() on WIN32/WinSock.
|
* Portable wrapper for close(); use closesocket() on WIN32/WinSock.
|
||||||
*/
|
*/
|
||||||
|
Loading…
Reference in New Issue
Block a user