libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
FileDescriptor.cxx
1// POSIX
2#include <unistd.h>
3
4// C++
5#include <string>
6
7// cosmos
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>
14
15namespace cosmos {
16
18 if(!valid())
19 return;
20 const auto fd = m_fd;
21 m_fd = FileNum::INVALID;
22 if (::close(to_integral(fd)) == 0) {
23 return;
24 }
25
26 cosmos_throw (ApiError("close()"));
27}
28
29int FileDescriptor::fcntl(int cmd) const {
30 return ::fcntl(to_integral(m_fd), cmd);
31}
32
33template <typename T>
34int FileDescriptor::fcntl(int cmd, T val) const {
35 return ::fcntl(to_integral(m_fd), cmd, val);
36}
37
38void FileDescriptor::duplicate(const FileDescriptor new_fd, const CloseOnExec cloexec) const {
39 const auto res = ::dup3(to_integral(m_fd), to_integral(new_fd.raw()), cloexec ? O_CLOEXEC : 0);
40
41 if (res == -1) {
42 cosmos_throw (ApiError("dup3()"));
43 }
44}
45
46FileDescriptor FileDescriptor::duplicate(const FileNum lowest, const CloseOnExec cloexec) const {
47 const auto fd = this->fcntl(cloexec ? F_DUPFD_CLOEXEC : F_DUPFD, to_integral(lowest));
48
49 if (fd == -1) {
50 cosmos_throw (ApiError("fcntl(F_DUPFD)"));
51 }
52
53 return FileDescriptor{FileNum{fd}};
54}
55
57 const auto flags = this->fcntl(F_GETFD);
58
59 if (flags == -1) {
60 cosmos_throw (ApiError("fcntl(F_GETFD)"));
61 }
62
63 return DescFlags{flags};
64}
65
67 const auto res = this->fcntl(F_SETFD, flags.raw());
68
69 if (res != 0) {
70 cosmos_throw (ApiError("fcntl(F_SETFD)"));
71 }
72}
73
74std::tuple<OpenMode, OpenFlags> FileDescriptor::getStatusFlags() const {
75 const auto flags = this->fcntl(F_GETFL);
76
77 if (flags == -1) {
78 cosmos_throw (ApiError("fcntl(F_GETFL)"));
79 }
80
81 OpenMode mode{flags & (O_RDONLY|O_WRONLY|O_RDWR)};
82 OpenFlags settings{flags & ~to_integral(mode)};
83
84 return {mode, settings};
85}
86
88 const auto res = this->fcntl(F_SETFL, flags.raw());
89
90 if (res != 0) {
91 cosmos_throw (ApiError("fcntl(F_SETFL)"));
92 }
93}
94
95namespace {
96
97 template <typename SyncFunc>
98 void sync_helper(FileDescriptor &fd, SyncFunc sync_func, const char *errlabel) {
99 while (true) {
100 if (sync_func(to_integral(fd.raw())) == 0) {
101 return;
102 }
103 else {
104 switch(get_errno()) {
105 default: break;
106 case Errno::INTERRUPTED: {
107 if (auto_restart_syscalls)
108 continue;
109 }
110 }
111
112 cosmos_throw (ApiError(errlabel));
113 }
114 }
115 }
116
117} // end anon ns
118
120 sync_helper(*this, fsync, "fsync()");
121}
122
124 sync_helper(*this, fdatasync, "fdatasync()");
125}
126
128 const auto res = this->fcntl(F_ADD_SEALS, flags.raw());
129
130 if (res != 0) {
131 cosmos_throw (ApiError("fcntl(F_ADD_SEALS)"));
132 }
133}
134
136 const auto res = this->fcntl(F_GET_SEALS);
137
138 if (res != 0) {
139 cosmos_throw (ApiError("fcntl(F_GET_SEALS)"));
140 }
141
142 return SealFlags{static_cast<SealFlag>(res)};
143}
144
146 const auto res = this->fcntl(F_GETPIPE_SZ);
147
148 if (res != 0) {
149 cosmos_throw (ApiError("fcntl(F_GETPIP_SZ)"));
150 }
151
152 return res;
153}
154
155int FileDescriptor::setPipeSize(const int new_size) {
156 const auto res = this->fcntl(F_SETPIPE_SZ, new_size);
157
158 if (res != 0) {
159 cosmos_throw (ApiError("fcntl(F_SETPIP_SZ)"));
160 }
161
162 return res;
163}
164
166 const auto res = this->fcntl(F_GETLK, &lock);
167
168 if (res != 0) {
169 cosmos_throw (ApiError("fcntl(F_GETLK)"));
170 }
171
172 return lock.type() == FileLock::Type::UNLOCK;
173}
174
175bool FileDescriptor::setLock(const FileLock &lock) const {
176 const auto res = this->fcntl(F_SETLK, &lock);
177
178 if (res != 0) {
179 switch (get_errno()) {
180 case Errno::AGAIN:
181 case Errno::ACCESS:
182 return false;
183 default:
184 cosmos_throw (ApiError("fcntl(F_SETLK)"));
185 }
186 }
187
188 return true;
189}
190
191void FileDescriptor::setLockWait(const FileLock &lock) const {
192 const auto res = this->fcntl(F_SETLKW, &lock);
193
194 if (res != 0) {
195 cosmos_throw (ApiError("fcntl(F_SETLKW)"));
196 }
197}
198
200 const auto res = this->fcntl(F_OFD_GETLK, &lock);
201
202 if (res != 0) {
203 cosmos_throw (ApiError("fcntl(F_OFD_GETLK)"));
204 }
205
206 return lock.type() == FileLock::Type::UNLOCK;
207}
208
209bool FileDescriptor::setOFDLock(const FileLock &lock) const {
210 if (lock.pid() != ProcessID{0}) {
211 // provide better diagnostics, the kernel would just return EINVAL in this case
212 cosmos_throw (UsageError("attempt to set OFD lock with l_pid != 0"));
213 }
214 const auto res = this->fcntl(F_OFD_SETLK, &lock);
215
216 if (res != 0) {
217 switch (get_errno()) {
218 case Errno::AGAIN:
219 case Errno::ACCESS:
220 return false;
221 default:
222 cosmos_throw (ApiError("fcntl(F_OFD_SETLK)"));
223 }
224 }
225
226 return true;
227}
228
230 if (lock.pid() != ProcessID{0}) {
231 // provide better diagnostics, the kernel would just return EINVAL in this case
232 cosmos_throw (UsageError("attempt to set OFD lock with l_pid != 0"));
233 }
234 const auto res = this->fcntl(F_OFD_SETLKW, &lock);
235
236 if (res != 0) {
237 cosmos_throw (ApiError("fcntl(F_OFD_SETLKW)"));
238 }
239}
240
241void FileDescriptor::getOwner(Owner &owner) const {
242 const auto res = this->fcntl(F_GETOWN_EX, owner.raw());
243
244 if (res != 0) {
245 cosmos_throw (ApiError("fcntl(F_GETOWN_EX)"));
246 }
247}
248
250 const auto res = this->fcntl(F_SETOWN_EX, owner.raw());
251
252 if (res != 0) {
253 cosmos_throw (ApiError("fcntl(F_SETOWN_EX)"));
254 }
255}
256
257std::optional<Signal> FileDescriptor::getSignal() const {
258 const auto res = this->fcntl(F_GETSIG);
259
260 if (res == -1) {
261 cosmos_throw (ApiError("fcntl(F_GETSIG)"));
262 }
263
264 if (res == 0)
265 return {};
266
267 return Signal{SignalNr{res}};
268}
269
270void FileDescriptor::setSignal(std::optional<Signal> sig) {
271 const auto res = this->fcntl(F_SETSIG, sig ? to_integral(sig->raw()) : 0);
272
273 if (res != 0) {
274 cosmos_throw (ApiError("fcntl(F_SETSIG)"));
275 }
276}
277
279 const auto res = this->fcntl(F_GETLEASE);
280
281 if (res == -1) {
282 cosmos_throw (ApiError("fcntl(F_GETLEASE)"));
283 }
284
285 return LeaseType{res};
286}
287
289 const auto res = this->fcntl(F_SETLEASE, to_integral(lease));
290
291 if (res != 0) {
292 cosmos_throw (ApiError("fcntl(F_SETLEASE)"));
293 }
294}
295
296FileDescriptor stdout(FileNum::STDOUT);
297FileDescriptor stderr(FileNum::STDERR);
298FileDescriptor stdin(FileNum::STDIN);
299
300} // end ns
Specialized exception type used when system APIs fail.
Definition ApiError.hxx:18
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
EnumBaseType raw() const
Returns the raw bitfield integer.
Definition BitMask.hxx:56
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.
Definition FileLock.hxx:60
Strong template type to wrap boolean values in a named type.
Definition utils.hxx:50
Represents a POSIX signal number and offers a minimal API around it.
Definition types.hxx:96
Exception type for logical usage errors within the application.
Errno get_errno()
Wrapper that returns the Errno strongly typed representation of the current errno
Definition errno.hxx:111
FileNum
Primitive file descriptor.
Definition types.hxx:32
OpenMode
Strong enum type wrapper for the basic open() mode flag.
Definition types.hxx:52
ProcessID
Definition types.hxx:25
SignalNr
A primitive signal number specification.
Definition types.hxx:50
Information about file owner settings.