libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
DirStream.cxx
1// Linux
2#include <fcntl.h>
3#include <sys/stat.h>
4#include <sys/types.h>
5
6// Cosmos
7#include <cosmos/formatting.hxx>
8#include <cosmos/private/cosmos.hxx>
9#include <cosmos/fs/filesystem.hxx>
10#include <cosmos/fs/DirStream.hxx>
11
12namespace cosmos {
13
15 try {
16 close();
17 } catch (const std::exception &ex) {
18 noncritical_error(
19 sprintf("%s: failed to close directory stream", __FUNCTION__),
20 ex);
21 }
22}
23
25 if (!m_stream) {
26 return;
27 }
28
29 auto ret = closedir(m_stream);
30 m_stream = nullptr;
31
32 if (ret == -1) {
33 cosmos_throw (ApiError("closedir()"));
34 }
35}
36
37void DirStream::open(const SysString path, const FollowSymlinks follow_links) {
38 close();
39
40 /*
41 * reuse the open(DirFD) logic and only open the file
42 * descriptor before. This allows us to pass needful flags like
43 * O_CLOEXEC. This gives us more control than when using opendir().
44 */
45 auto res = ::open(
46 path.raw(),
47 O_RDONLY | O_CLOEXEC | O_DIRECTORY | (follow_links ? O_NOFOLLOW : 0)
48 );
49
50 DirFD fd{FileNum{res}};
51
52 if (fd.invalid()) {
53 cosmos_throw (ApiError("open(O_DIRECTORY)"));
54 }
55
56 try {
57 open(fd.raw());
58 } catch (...) {
59 // intentionally ignore error conditions here
60 try {
61 fd.close();
62 } catch(...) {}
63 throw;
64 }
65}
66
67void DirStream::open(const DirFD fd) {
68 close();
69
70 auto duplicate = fd.duplicate();
71 open(duplicate.raw());
72}
73
74void DirStream::open(const FileNum fd) {
75 m_stream = ::fdopendir(to_integral(fd));
76
77 if (!m_stream) {
78 cosmos_throw (ApiError("fdopendir()"));
79 }
80}
81
82void DirStream::open(const DirFD dir_fd, const SysString subpath) {
83
84 const OpenFlags flags{OpenFlag::DIRECTORY};
85 auto fd = fs::open_at(dir_fd, subpath, OpenMode::READ_ONLY, flags);
86
87 // ownership is transferred to the stream
88 m_stream = ::fdopendir(to_integral(fd.raw()));
89
90 if (!m_stream) {
91 cosmos_throw (ApiError("fdopendir()"));
92 }
93}
94
96 requireOpenStream(__FUNCTION__);
97 auto fd = dirfd(m_stream);
98 DirFD ret{FileNum{fd}};
99
100 if (ret.invalid()) {
101 cosmos_throw (ApiError("dirfd()"));
102 }
103
104 return ret;
105}
106
107std::optional<DirEntry> DirStream::nextEntry() {
108 requireOpenStream(__FUNCTION__);
109
110 /*
111 * there's a bit confusion between readdir and readdir_r(). Today on
112 * Linux readdir() is thread-safe between different directory streams
113 * but not thread safe when using the same directory stream in
114 * parallel. The latter is rather peculiar and should not be needed.
115 * Therefore use readdir().
116 */
117
118 // needed to differentiate between end-of-stream and error condition
119 reset_errno();
120 const auto entry = readdir(m_stream);
121
122 if (entry) {
123 return DirEntry{entry};
124 }
125
126 if (is_errno_set()) {
127 cosmos_throw(ApiError("readdir()"));
128 }
129
130 return {};
131}
132
133} // 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
A single directory entry as returned from DirStream::nextEntry().
Definition DirEntry.hxx:23
A specialized FileDescriptor for directory objects.
Definition DirFD.hxx:17
~DirStream()
Closes the underlying directory object, if currently open.
Definition DirStream.cxx:14
void open(const DirFD fd)
Associate with the directory represented by the given file descriptor.
Definition DirStream.cxx:67
DirFD fd() const
Return the file descriptor associated with the current DirStream object.
Definition DirStream.cxx:95
std::optional< DirEntry > nextEntry()
Returns the next entry in the associated directory.
void close()
Close the currently associated directory.
Definition DirStream.cxx:24
void close()
Explicitly close the contained FD.
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.
Strong template type to wrap boolean values in a named type.
Definition utils.hxx:50
void reset_errno()
Resets the currently set errno to indicate no error.
Definition errno.hxx:113
bool is_errno_set()
Checks whether currently an errno is set.
Definition errno.hxx:117
FileNum
Primitive file descriptor.
Definition types.hxx:32
Wrapper type around a C-style string for use with system APIs.
Definition SysString.hxx:33