8#include <cosmos/error/ApiError.hxx>
9#include <cosmos/error/UsageError.hxx>
10#include <cosmos/fs/FileDescriptor.hxx>
11#include <cosmos/fs/FileLock.hxx>
12#include <cosmos/private/cosmos.hxx>
13#include <cosmos/utils.hxx>
21 m_fd = FileNum::INVALID;
22 if (
::close(to_integral(fd)) == 0) {
29int FileDescriptor::fcntl(
int cmd)
const {
30 return ::fcntl(to_integral(m_fd), cmd);
34int FileDescriptor::fcntl(
int cmd, T val)
const {
35 return ::fcntl(to_integral(m_fd), cmd, val);
39 const auto res = ::dup3(to_integral(m_fd), to_integral(new_fd.
raw()), cloexec ? O_CLOEXEC : 0);
47 const auto fd = this->fcntl(cloexec ? F_DUPFD_CLOEXEC : F_DUPFD, to_integral(lowest));
50 cosmos_throw (
ApiError(
"fcntl(F_DUPFD)"));
57 const auto flags = this->fcntl(F_GETFD);
60 cosmos_throw (
ApiError(
"fcntl(F_GETFD)"));
67 const auto res = this->fcntl(F_SETFD, flags.
raw());
70 cosmos_throw (
ApiError(
"fcntl(F_SETFD)"));
75 const auto flags = this->fcntl(F_GETFL);
78 cosmos_throw (
ApiError(
"fcntl(F_GETFL)"));
81 OpenMode mode{flags & (O_RDONLY|O_WRONLY|O_RDWR)};
82 OpenFlags settings{flags & ~to_integral(mode)};
84 return {mode, settings};
88 const auto res = this->fcntl(F_SETFL, flags.
raw());
91 cosmos_throw (
ApiError(
"fcntl(F_SETFL)"));
97 template <
typename SyncFunc>
98 void sync_helper(
FileDescriptor &fd, SyncFunc sync_func,
const char *errlabel) {
100 if (sync_func(to_integral(fd.
raw())) == 0) {
104 switch(get_errno()) {
106 case Errno::INTERRUPTED: {
107 if (auto_restart_syscalls)
112 cosmos_throw (ApiError(errlabel));
120 sync_helper(*
this, fsync,
"fsync()");
124 sync_helper(*
this, fdatasync,
"fdatasync()");
128 const auto res = this->fcntl(F_ADD_SEALS, flags.
raw());
131 cosmos_throw (
ApiError(
"fcntl(F_ADD_SEALS)"));
136 const auto res = this->fcntl(F_GET_SEALS);
139 cosmos_throw (
ApiError(
"fcntl(F_GET_SEALS)"));
146 const auto res = this->fcntl(F_GETPIPE_SZ);
149 cosmos_throw (
ApiError(
"fcntl(F_GETPIP_SZ)"));
156 const auto res = this->fcntl(F_SETPIPE_SZ, new_size);
159 cosmos_throw (
ApiError(
"fcntl(F_SETPIP_SZ)"));
166 const auto res = this->fcntl(F_GETLK, &lock);
169 cosmos_throw (
ApiError(
"fcntl(F_GETLK)"));
172 return lock.type() == FileLock::Type::UNLOCK;
176 const auto res = this->fcntl(F_SETLK, &lock);
184 cosmos_throw (
ApiError(
"fcntl(F_SETLK)"));
192 const auto res = this->fcntl(F_SETLKW, &lock);
195 cosmos_throw (
ApiError(
"fcntl(F_SETLKW)"));
200 const auto res = this->fcntl(F_OFD_GETLK, &lock);
203 cosmos_throw (
ApiError(
"fcntl(F_OFD_GETLK)"));
206 return lock.type() == FileLock::Type::UNLOCK;
212 cosmos_throw (
UsageError(
"attempt to set OFD lock with l_pid != 0"));
214 const auto res = this->fcntl(F_OFD_SETLK, &lock);
222 cosmos_throw (
ApiError(
"fcntl(F_OFD_SETLK)"));
232 cosmos_throw (
UsageError(
"attempt to set OFD lock with l_pid != 0"));
234 const auto res = this->fcntl(F_OFD_SETLKW, &lock);
237 cosmos_throw (
ApiError(
"fcntl(F_OFD_SETLKW)"));
242 const auto res = this->fcntl(F_GETOWN_EX, owner.raw());
245 cosmos_throw (
ApiError(
"fcntl(F_GETOWN_EX)"));
250 const auto res = this->fcntl(F_SETOWN_EX, owner.raw());
253 cosmos_throw (
ApiError(
"fcntl(F_SETOWN_EX)"));
258 const auto res = this->fcntl(F_GETSIG);
261 cosmos_throw (
ApiError(
"fcntl(F_GETSIG)"));
271 const auto res = this->fcntl(F_SETSIG, sig ? to_integral(sig->raw()) : 0);
274 cosmos_throw (
ApiError(
"fcntl(F_SETSIG)"));
279 const auto res = this->fcntl(F_GETLEASE);
282 cosmos_throw (
ApiError(
"fcntl(F_GETLEASE)"));
289 const auto res = this->fcntl(F_SETLEASE, to_integral(lease));
292 cosmos_throw (
ApiError(
"fcntl(F_SETLEASE)"));
Specialized exception type used when system APIs fail.
A typesafe bit mask representation using class enums.
EnumBaseType raw() const
Returns the raw bitfield integer.
Thin Wrapper around OS file descriptors.
bool setOFDLock(const FileLock &lock) const
Just like setLock() but using open-file-description locks.
DescFlags getFlags() const
Retrieves the current file descriptor flags.
bool getOFDLock(FileLock &lock) const
Just like getLock() but using open-file-description locks.
void setLease(const LeaseType lease)
Sets a new lease type on the file descriptor.
void dataSync()
Flush outstanding writes to disk except metadata.
void addSeals(const SealFlags flags)
Add a seal for memory file descriptors.
bool getLock(FileLock &lock) const
Check lock availability for traditional process-wide POSIX locks.
void setOFDLockWait(const FileLock &lock) const
Just like setLockWait() but using open-file-description locks.
LeaseType
Different request types for managing file leases.
LeaseType getLease() const
Gets the lease type currently set, or required to resolve a lease break.
std::tuple< OpenMode, OpenFlags > getStatusFlags() const
Retrieve the file's OpenMode and current OpenFlags.
void setLockWait(const FileLock &lock) const
Blocking version of setLock().
bool setLock(const FileLock &lock) const
Release, or attempt to obtain, a traditional process-wide POSIX lock.
void setStatusFlags(const OpenFlags flags)
Change certain file descriptor status flags.
void setFlags(const DescFlags flags)
Changes the current file descriptor flags.
int getPipeSize() const
For pipe file descriptors return the size of the pipe buffer in the kernel.
void close()
Explicitly close the contained FD.
void setSignal(std::optional< Signal > sig)
Configure the signal to be used for asynchronous I/O.
void sync()
Flush oustanding writes to disk.
SealFlags getSeals() const
Get the currently set SealFlags for the file descriptor.
FileNum raw() const
Returns the primitive file descriptor contained in the object.
void duplicate(const FileDescriptor new_fd, const CloseOnExec cloexec=CloseOnExec{true}) const
Get a duplicate file descriptor that will further be known as new_fd.
SealFlag
Flags used in addSeals().
void getOwner(Owner &owner) const
Returns the current file descriptor owner settings.
std::optional< Signal > getSignal() const
Returns the currently configured signal for asynchronous I/O.
void setOwner(const Owner owner)
Change the current file descriptor owner settings.
bool valid() const
Returns whether currently a valid file descriptor number is assigned.
int setPipeSize(const int new_size)
For pipe file descriptors this sets a new size for the pipe buffer in the kernel.
Wrapper around struct flock used for advisory file locking in FileDescriptor.
Strong template type to wrap boolean values in a named type.
Represents a POSIX signal number and offers a minimal API around it.
Exception type for logical usage errors within the application.
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
FileNum
Primitive file descriptor.
OpenMode
Strong enum type wrapper for the basic open() mode flag.
SignalNr
A primitive signal number specification.
Information about file owner settings.