µOS++ IIIe Reference 7.0.0
The third edition of µOS++, a POSIX inspired open source framework, written in C++
Loading...
Searching...
No Matches
file-descriptors-manager.cpp
Go to the documentation of this file.
1/*
2 * This file is part of the µOS++ project (https://micro-os-plus.github.io/).
3 * Copyright (c) 2015-2025 Liviu Ionescu. All rights reserved.
4 *
5 * Permission to use, copy, modify, and/or distribute this software
6 * for any purpose is hereby granted, under the terms of the MIT license.
7 *
8 * If a copy of the license was not distributed with this file, it can
9 * be obtained from https://opensource.org/licenses/mit.
10 */
11
12#if defined(OS_USE_OS_APP_CONFIG_H)
13#include <cmsis-plus/os-app-config.h>
14#endif
15
19
21
22#include <cerrno>
23#include <cassert>
24#include <cstddef>
25
26// ----------------------------------------------------------------------------
27
28#if defined(__clang__)
29#pragma clang diagnostic ignored "-Wc++98-compat"
30#endif
31
32// ----------------------------------------------------------------------------
33
34namespace os
35{
36 namespace posix
37 {
38 // ------------------------------------------------------------------------
39
44 std::size_t file_descriptors_manager::size__;
45
46#pragma GCC diagnostic push
47#if defined(__clang__)
48#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
49#endif
50 io** file_descriptors_manager::descriptors_array__;
51#pragma GCC diagnostic pop
52
57 // ========================================================================
59 {
60 trace::printf ("file_descriptors_manager::%s(%d)=%p\n", __func__, size,
61 this);
62
63 assert (size > 0);
64
65 size__ = size + reserved__; // Add space for standard files.
66 descriptors_array__ = new class io*[size__];
67
68 for (std::size_t i = 0; i < file_descriptors_manager::size (); ++i)
69 {
70#pragma GCC diagnostic push
71#if defined(__clang__)
72#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
73#endif
74 descriptors_array__[i] = nullptr;
75#pragma GCC diagnostic pop
76 }
77 }
78
80 {
81 trace::printf ("file_descriptors_manager::%s(%) @%p\n", __func__, this);
82
83 delete[] descriptors_array__;
84 size__ = 0;
85 }
86
87 // ------------------------------------------------------------------------
88
89 io*
91 {
92 // Check if valid descriptor or buffer not yet initialised
93 if ((fildes < 0) || (static_cast<std::size_t> (fildes) >= size__)
94 || (descriptors_array__ == nullptr))
95 {
96 return nullptr;
97 }
98#pragma GCC diagnostic push
99#if defined(__clang__)
100#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
101#endif
102 return descriptors_array__[fildes];
103#pragma GCC diagnostic pop
104 }
105
106 bool
108 {
109 if ((fildes < 0) || (static_cast<std::size_t> (fildes) >= size__))
110 {
111 return false;
112 }
113 return true;
114 }
115
116 int
118 {
119#if defined(OS_TRACE_POSIX_IO_FILE_DESCRIPTORS_MANAGER)
120 trace::printf ("file_descriptors_manager::%s(%p)\n", __func__, io);
121#endif
122
123 if (io->file_descriptor () >= 0)
124 {
125 // Already allocated
126 errno = EBUSY;
127 return -1;
128 }
129
130 for (std::size_t i = reserved__; i < size__; ++i)
131 {
132#pragma GCC diagnostic push
133#if defined(__clang__)
134#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
135#endif
136 if (descriptors_array__[i] == nullptr)
137 {
138 descriptors_array__[i] = io;
139 io->file_descriptor (static_cast<int> (i));
140#if defined(OS_TRACE_POSIX_IO_FILE_DESCRIPTORS_MANAGER)
141 trace::printf ("file_descriptors_manager::%s(%p) fd=%d\n",
142 __func__, io, i);
143#endif
144 return static_cast<int> (i);
145 }
146#pragma GCC diagnostic pop
147 }
148
149 // Too many files open in system.
150 errno = ENFILE;
151 return -1;
152 }
153
154 int
156 {
157 if ((fildes < 0) || (static_cast<std::size_t> (fildes) >= size__))
158 {
159 errno = EBADF;
160 return -1;
161 }
162
163 if (io->file_descriptor () >= 0)
164 {
165 // Already allocated
166 errno = EBUSY;
167 return -1;
168 }
169
170#pragma GCC diagnostic push
171#if defined(__clang__)
172#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
173#endif
174 descriptors_array__[fildes] = io;
175#pragma GCC diagnostic pop
176 io->file_descriptor (fildes);
177 return fildes;
178 }
179
180 int
182 {
183#if defined(OS_TRACE_POSIX_IO_FILE_DESCRIPTORS_MANAGER)
184 trace::printf ("file_descriptors_manager::%s(%d)\n", __func__, fildes);
185#endif
186
187 if ((fildes < 0) || (static_cast<std::size_t> (fildes) >= size__))
188 {
189 errno = EBADF;
190 return -1;
191 }
192
193#pragma GCC diagnostic push
194#if defined(__clang__)
195#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
196#endif
197 descriptors_array__[fildes]->clear_file_descriptor ();
198 descriptors_array__[fildes] = nullptr;
199#pragma GCC diagnostic pop
200 return 0;
201 }
202
203 /* class */ socket*
205 {
206 assert ((fildes >= 0) && (static_cast<std::size_t> (fildes) < size__));
207#pragma GCC diagnostic push
208#if defined(__clang__)
209#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
210#endif
211 auto* const io = descriptors_array__[fildes];
212#pragma GCC diagnostic pop
213 if (io->get_type () != static_cast<posix::io::type_t> (io::type::socket))
214 {
215 return nullptr;
216 }
217 return reinterpret_cast<class socket*> (io);
218 }
219
220 size_t
222 {
223 std::size_t count = reserved__;
224 for (std::size_t i = reserved__; i < file_descriptors_manager::size ();
225 ++i)
226 {
227#pragma GCC diagnostic push
228#if defined(__clang__)
229#pragma clang diagnostic ignored "-Wunsafe-buffer-usage"
230#endif
231 if (descriptors_array__[i] != nullptr)
232 {
233 ++count;
234 }
235#pragma GCC diagnostic pop
236 }
237 return count;
238 }
239
240 // ========================================================================
241 } /* namespace posix */
242} /* namespace os */
243
244// ----------------------------------------------------------------------------
static class socket * socket(int fildes)
static int deallocate(file_descriptor_t fildes)
static int assign(file_descriptor_t fildes, class io *io)
Base I/O class.
Definition io.h:86
file_descriptor_t file_descriptor(void) const
Definition io.h:452
unsigned int type_t
Definition io.h:125
type_t get_type(void) const
Definition io.h:434
Network socket.
int printf(const char *format,...)
Write a formatted string to the trace device.
Definition trace.cpp:59
int file_descriptor_t
Definition types.h:42
System namespace.