libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
FileDescriptor.hxx
1#pragma once
2
3// C++
4#include <optional>
5#include <tuple>
6
7// Linux
8#include <fcntl.h>
9
10// cosmos
11#include <cosmos/BitMask.hxx>
12#include <cosmos/fs/types.hxx>
13#include <cosmos/proc/types.hxx>
14#include <cosmos/thread/thread.hxx>
15#include <cosmos/utils.hxx>
16
17namespace cosmos {
18
19class FileLock;
20
22using CloseOnExec = NamedBool<struct cloexec_t, true>;
23
25
35class COSMOS_API FileDescriptor {
36public: // types
37
39 enum class DescFlag : int {
40 NONE = 0,
41 CLOEXEC = FD_CLOEXEC
42 };
43
46
48 enum class SealFlag : unsigned int {
49 SEAL = F_SEAL_SEAL,
50 SHRINK = F_SEAL_SHRINK,
51 GROW = F_SEAL_GROW,
52 WRITE = F_SEAL_WRITE,
53 FUTURE_WRITE = F_SEAL_FUTURE_WRITE
54 };
55
58
60
65 enum class LeaseType : int {
66 READ = F_RDLCK,
67 WRITE = F_WRLCK,
68 UNLOCK = F_UNLCK
69 };
70
72
81 struct Owner {
82
83 /*
84 * Contrary to what the man page says this is not an int, but
85 * an enum in the system headers.
86 *
87 * The type field being an old school enum complicates things
88 * for us here, we cannot use to_integral() to assign. Instead
89 * perform a static cast to the declaration type to make this
90 * work.
91 */
92 using RawType = decltype(f_owner_ex::type);
93
94 public: // types
95
97 enum class Type : std::underlying_type<RawType>::type {
98 THREAD = F_OWNER_TID,
99 PROCESS = F_OWNER_PID,
100 GROUP = F_OWNER_PGRP
101 };
102
103 public: // functions
104
105 Owner() {
106 invalidate();
107 }
108
109 explicit Owner(ProcessID pid) {
110 set(pid);
111 }
112
113 explicit Owner(ProcessGroupID pgid) {
114 set(pgid);
115 }
116
117 explicit Owner(ThreadID tid) {
118 set(tid);
119 }
120
121 Type type() const {
122 return Type{m_raw.type};
123 }
124
125 bool isTID() const {
126 return type() == Type::THREAD;
127 }
128
129 bool isPID() const {
130 return type() == Type::PROCESS;
131 }
132
133 bool isPGID() const {
134 return type() == Type::GROUP;
135 }
136
137 bool valid() const {
138 // if not owner was ever set then the PID will be zero
139 return m_raw.pid != 0 && (isTID() || isPID() || isPGID());
140 }
141
142 void invalidate() {
143 m_raw.type = (RawType)-1;
144 m_raw.pid = -1;
145 }
146
147 void set(ProcessID pid) {
148 m_raw.type = static_cast<RawType>(Type::PROCESS);
149 m_raw.pid = to_integral(pid);
150 }
151
152 void set(ProcessGroupID pgid) {
153 m_raw.type = static_cast<RawType>(Type::GROUP);
154 m_raw.pid = to_integral(pgid);
155 }
156
157 void set(ThreadID tid) {
158 m_raw.type = static_cast<RawType>(Type::THREAD);
159 m_raw.pid = to_integral(tid);
160 }
161
162 std::optional<ProcessID> asPID() const {
163 return isPID() ? std::make_optional(ProcessID{m_raw.pid}) : std::nullopt;
164 }
165
166 std::optional<ProcessGroupID> asPGID() const {
167 return isPGID() ? std::make_optional(ProcessGroupID{m_raw.pid}) : std::nullopt;
168 }
169
170 std::optional<ThreadID> asTID() const {
171 return isTID() ? std::make_optional(ThreadID{m_raw.pid}) : std::nullopt;
172 }
173
174 auto raw() {
175 return &m_raw;
176 }
177
178 auto raw() const {
179 return &m_raw;
180 }
181
182 protected: // data
183
184 f_owner_ex m_raw;
185 };
186
187public: // functions
188
189 constexpr FileDescriptor() = default;
190
191 explicit constexpr FileDescriptor(FileNum fd) :
192 m_fd{fd} {}
193
195 bool valid() const { return m_fd != FileNum::INVALID; }
196 bool invalid() const { return !valid(); }
197
199
203 void setFD(const FileNum fd) { m_fd = fd; }
204
206
212 void reset() { m_fd = FileNum::INVALID; }
213
215
228 void close();
229
231
240 void duplicate(const FileDescriptor new_fd, const CloseOnExec cloexec = CloseOnExec{true}) const;
241
243
255 FileDescriptor duplicate(
256 const FileNum lowest = FileNum{0},
257 const CloseOnExec cloexec = CloseOnExec{true}) const;
258
260 DescFlags getFlags() const;
261
263 void setFlags(const DescFlags flags);
264
266 void setCloseOnExec(bool on_off) {
267 setFlags({on_off ? DescFlag::CLOEXEC : DescFlag::NONE});
268 }
269
271 std::tuple<OpenMode, OpenFlags> getStatusFlags() const;
272
274
279 void setStatusFlags(const OpenFlags flags);
280
282
306 void sync();
307
309
315 void dataSync();
316
318
324 void addSeals(const SealFlags flags);
325
327 SealFlags getSeals() const;
328
330 int getPipeSize() const;
331
333
338 int setPipeSize(const int new_size);
339
341 FileNum raw() const { return m_fd; }
342
343 bool operator==(const FileDescriptor &other) const {
344 return m_fd == other.m_fd;
345 }
346
347 bool operator!=(const FileDescriptor &other) const {
348 return !(*this == other);
349 }
350
352
366 bool getLock(FileLock &lock) const;
367
369
380 bool setLock(const FileLock &lock) const;
381
383
390 void setLockWait(const FileLock &lock) const;
391
393 bool getOFDLock(FileLock &lock) const;
394
396 bool setOFDLock(const FileLock &lock) const;
397
399 void setOFDLockWait(const FileLock &lock) const;
400
402
409 void getOwner(Owner &owner) const;
410
412
417 void setOwner(const Owner owner);
418
420
436 std::optional<Signal> getSignal() const;
437
439
448 void setSignal(std::optional<Signal> sig);
449
451
466 LeaseType getLease() const;
467
469
524 void setLease(const LeaseType lease);
525
526protected: // functions
527
528 int fcntl(int cmd) const;
529
530 template <typename T>
531 int fcntl(int cmd, T val) const;
532
533protected: // data
534
535 FileNum m_fd = FileNum::INVALID;
536};
537
539extern COSMOS_API FileDescriptor stdout;
541extern COSMOS_API FileDescriptor stderr;
543extern COSMOS_API FileDescriptor stdin;
544
545} // end ns
A typesafe bit mask representation using class enums.
Definition BitMask.hxx:19
Thin Wrapper around OS file descriptors.
LeaseType
Different request types for managing file leases.
DescFlag
Configurable per file-descriptors flags.
FileNum raw() const
Returns the primitive file descriptor contained in the object.
void setFD(const FileNum fd)
Assigns a new primitive file descriptor to the object.
SealFlag
Flags used in addSeals().
bool valid() const
Returns whether currently a valid file descriptor number is assigned.
void reset()
Invalidates the stored file descriptor.
void setCloseOnExec(bool on_off)
convenience wrapper around setFlags to change CLOEXEC setting
Strong template type to wrap boolean values in a named type.
Definition utils.hxx:50
@ THREAD
The child shares the same thread group as the parent. Thread groups are used to implement thread sema...
FileNum
Primitive file descriptor.
Definition types.hxx:32
ProcessID
Definition types.hxx:25
Information about file owner settings.
Type
Strong owner type differentation.