blob: b72c0c54acd4d64afffaf1c0b6af7f929fbb1116 [file] [log] [blame]
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001/*
Florin Coras5e062572019-03-14 19:07:51 -07002 * Copyright (c) 2016-2019 Cisco and/or its affiliates.
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07003 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15#include <unistd.h>
16#include <stdio.h>
17#include <signal.h>
18#include <dlfcn.h>
19#include <pthread.h>
20#include <time.h>
21#include <stdarg.h>
shrinivasan ganapathy1d359632017-10-15 15:46:09 -070022#include <sys/resource.h>
Dave Wallace048b1d62018-01-03 22:24:41 -050023#include <netinet/tcp.h>
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -070024
Dave Wallace2a865272018-02-07 21:00:42 -050025#include <vcl/ldp_socket_wrapper.h>
26#include <vcl/ldp.h>
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -070027#include <sys/time.h>
28
Florin Coras7baeb712019-01-04 17:05:43 -080029#include <vcl/vcl_locked.h>
Dave Wallace048b1d62018-01-03 22:24:41 -050030#include <vppinfra/time.h>
31#include <vppinfra/bitmap.h>
Florin Coras30e273b2018-11-27 00:04:59 -080032#include <vppinfra/lock.h>
33#include <vppinfra/pool.h>
34#include <vppinfra/hash.h>
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -070035
36#define HAVE_CONSTRUCTOR_ATTRIBUTE
37#ifdef HAVE_CONSTRUCTOR_ATTRIBUTE
38#define CONSTRUCTOR_ATTRIBUTE \
39 __attribute__ ((constructor))
40#else
41#define CONSTRUCTOR_ATTRIBUTE
42#endif /* HAVE_CONSTRUCTOR_ATTRIBUTE */
43
44#define HAVE_DESTRUCTOR_ATTRIBUTE
45#ifdef HAVE_DESTRUCTOR_ATTRIBUTE
46#define DESTRUCTOR_ATTRIBUTE \
47 __attribute__ ((destructor))
48#else
49#define DESTRUCTOR_ATTRIBUTE
50#endif
51
Florin Corasdfe4cf42018-11-28 22:13:45 -080052#define LDP_MAX_NWORKERS 32
53
Florin Corasdfe4cf42018-11-28 22:13:45 -080054typedef struct ldp_worker_ctx_
Dave Wallace048b1d62018-01-03 22:24:41 -050055{
Dave Wallace048b1d62018-01-03 22:24:41 -050056 u8 *io_buffer;
57 clib_time_t clib_time;
Florin Corasdfe4cf42018-11-28 22:13:45 -080058
59 /*
60 * Select state
61 */
Dave Wallace048b1d62018-01-03 22:24:41 -050062 clib_bitmap_t *rd_bitmap;
63 clib_bitmap_t *wr_bitmap;
64 clib_bitmap_t *ex_bitmap;
Florin Coras294afe22019-01-07 17:49:17 -080065 clib_bitmap_t *si_rd_bitmap;
66 clib_bitmap_t *si_wr_bitmap;
67 clib_bitmap_t *si_ex_bitmap;
Dave Wallace048b1d62018-01-03 22:24:41 -050068 clib_bitmap_t *libc_rd_bitmap;
69 clib_bitmap_t *libc_wr_bitmap;
70 clib_bitmap_t *libc_ex_bitmap;
Florin Corasdfe4cf42018-11-28 22:13:45 -080071
72 /*
73 * Poll state
74 */
Dave Wallace048b1d62018-01-03 22:24:41 -050075 vcl_poll_t *vcl_poll;
Florin Coras6917b942018-11-13 22:44:54 -080076 struct pollfd *libc_poll;
77 u16 *libc_poll_idxs;
Florin Corasdfe4cf42018-11-28 22:13:45 -080078
79 /*
80 * Epoll state
81 */
Dave Wallace048b1d62018-01-03 22:24:41 -050082 u8 epoll_wait_vcl;
hanlin4266d4d2020-05-19 17:34:17 +080083 u8 mq_epfd_added;
Florin Coras99368312018-08-02 10:45:44 -070084 int vcl_mq_epfd;
Florin Corasdfe4cf42018-11-28 22:13:45 -080085
86} ldp_worker_ctx_t;
87
Florin Coras294afe22019-01-07 17:49:17 -080088/* clib_bitmap_t, fd_mask and vcl_si_set are used interchangeably. Make sure
89 * they are the same size */
90STATIC_ASSERT (sizeof (clib_bitmap_t) == sizeof (fd_mask),
91 "ldp bitmap size mismatch");
92STATIC_ASSERT (sizeof (vcl_si_set) == sizeof (fd_mask),
93 "ldp bitmap size mismatch");
94
Florin Corasdfe4cf42018-11-28 22:13:45 -080095typedef struct
96{
97 ldp_worker_ctx_t *workers;
98 int init;
99 char app_name[LDP_APP_NAME_MAX];
Florin Coras7baeb712019-01-04 17:05:43 -0800100 u32 vlsh_bit_val;
101 u32 vlsh_bit_mask;
Florin Corasdfe4cf42018-11-28 22:13:45 -0800102 u32 debug;
Florin Corasdfe4cf42018-11-28 22:13:45 -0800103
104 /** vcl needs next epoll_create to go to libc_epoll */
105 u8 vcl_needs_real_epoll;
Florin Corasa5a9efd2021-01-05 17:03:29 -0800106
107 /**
108 * crypto state used only for testing
109 */
110 u8 transparent_tls;
111 u32 ckpair_index;
Dave Wallace2a865272018-02-07 21:00:42 -0500112} ldp_main_t;
Florin Corasdfe4cf42018-11-28 22:13:45 -0800113
Dave Wallace2a865272018-02-07 21:00:42 -0500114#define LDP_DEBUG ldp->debug
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700115
Florin Coras99368312018-08-02 10:45:44 -0700116#define LDBG(_lvl, _fmt, _args...) \
117 if (ldp->debug > _lvl) \
hanlin9f3f18f2019-12-30 16:25:20 +0800118 { \
119 int errno_saved = errno; \
Florin Coras585c86a2020-10-16 17:57:36 -0700120 fprintf (stderr, "ldp<%d>: " _fmt "\n", getpid(), ##_args); \
hanlin9f3f18f2019-12-30 16:25:20 +0800121 errno = errno_saved; \
122 }
Florin Coras99368312018-08-02 10:45:44 -0700123
Dave Wallace2a865272018-02-07 21:00:42 -0500124static ldp_main_t ldp_main = {
Florin Coras7baeb712019-01-04 17:05:43 -0800125 .vlsh_bit_val = (1 << LDP_SID_BIT_MIN),
126 .vlsh_bit_mask = (1 << LDP_SID_BIT_MIN) - 1,
Dave Wallace2a865272018-02-07 21:00:42 -0500127 .debug = LDP_DEBUG_INIT,
Yu Ping7b74b072019-05-08 00:40:24 +0800128 .transparent_tls = 0,
Florin Corasa5a9efd2021-01-05 17:03:29 -0800129 .ckpair_index = ~0,
Dave Wallace048b1d62018-01-03 22:24:41 -0500130};
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700131
Dave Wallace2a865272018-02-07 21:00:42 -0500132static ldp_main_t *ldp = &ldp_main;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700133
Florin Corasdfe4cf42018-11-28 22:13:45 -0800134static inline ldp_worker_ctx_t *
135ldp_worker_get_current (void)
136{
137 return (ldp->workers + vppcom_worker_index ());
138}
139
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700140/*
141 * RETURN: 0 on success or -1 on error.
142 * */
Dave Wallace048b1d62018-01-03 22:24:41 -0500143static inline void
Dave Wallace2a865272018-02-07 21:00:42 -0500144ldp_set_app_name (char *app_name)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700145{
Benoît Ganne747b3d82019-08-21 18:27:23 +0200146 snprintf (ldp->app_name, LDP_APP_NAME_MAX,
147 "ldp-%d-%s", getpid (), app_name);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700148}
149
Dave Wallace048b1d62018-01-03 22:24:41 -0500150static inline char *
Dave Wallace2a865272018-02-07 21:00:42 -0500151ldp_get_app_name ()
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700152{
Dave Wallace2a865272018-02-07 21:00:42 -0500153 if (ldp->app_name[0] == '\0')
154 ldp_set_app_name ("app");
Dave Wallace048b1d62018-01-03 22:24:41 -0500155
Dave Wallace2a865272018-02-07 21:00:42 -0500156 return ldp->app_name;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700157}
158
Dave Wallace048b1d62018-01-03 22:24:41 -0500159static inline int
Florin Coras7baeb712019-01-04 17:05:43 -0800160ldp_vlsh_to_fd (vls_handle_t vlsh)
Dave Wallace048b1d62018-01-03 22:24:41 -0500161{
Florin Coras7baeb712019-01-04 17:05:43 -0800162 return (vlsh + ldp->vlsh_bit_val);
Dave Wallace048b1d62018-01-03 22:24:41 -0500163}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700164
Florin Coras7baeb712019-01-04 17:05:43 -0800165static inline vls_handle_t
166ldp_fd_to_vlsh (int fd)
Dave Wallace048b1d62018-01-03 22:24:41 -0500167{
Florin Coras7baeb712019-01-04 17:05:43 -0800168 if (fd < ldp->vlsh_bit_val)
169 return VLS_INVALID_HANDLE;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700170
Florin Coras7baeb712019-01-04 17:05:43 -0800171 return (fd - ldp->vlsh_bit_val);
Dave Wallace048b1d62018-01-03 22:24:41 -0500172}
173
Florin Coras2d9b4272019-03-11 10:14:37 -0700174static void
175ldp_alloc_workers (void)
176{
177 if (ldp->workers)
178 return;
179 pool_alloc (ldp->workers, LDP_MAX_NWORKERS);
180}
181
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700182static inline int
Dave Wallace2a865272018-02-07 21:00:42 -0500183ldp_init (void)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700184{
Florin Corasdfe4cf42018-11-28 22:13:45 -0800185 ldp_worker_ctx_t *ldpw;
Florin Coras99368312018-08-02 10:45:44 -0700186 int rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700187
Florin Coras99368312018-08-02 10:45:44 -0700188 if (PREDICT_TRUE (ldp->init))
189 return 0;
190
191 ldp->init = 1;
192 ldp->vcl_needs_real_epoll = 1;
Florin Coras7baeb712019-01-04 17:05:43 -0800193 rv = vls_app_create (ldp_get_app_name ());
Florin Coras99368312018-08-02 10:45:44 -0700194 if (rv != VPPCOM_OK)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700195 {
Florin Coras955bfbb2018-12-04 13:43:45 -0800196 ldp->vcl_needs_real_epoll = 0;
197 if (rv == VPPCOM_EEXIST)
198 return 0;
Florin Coras05ecfcc2018-12-12 18:19:39 -0800199 LDBG (2, "\nERROR: ldp_init: vppcom_app_create()"
200 " failed! rv = %d (%s)\n", rv, vppcom_retval_str (rv));
Florin Coras99368312018-08-02 10:45:44 -0700201 ldp->init = 0;
202 return rv;
203 }
204 ldp->vcl_needs_real_epoll = 0;
Florin Coras2d9b4272019-03-11 10:14:37 -0700205 ldp_alloc_workers ();
Florin Corasdfe4cf42018-11-28 22:13:45 -0800206 ldpw = ldp_worker_get_current ();
Florin Coras99368312018-08-02 10:45:44 -0700207
208 char *env_var_str = getenv (LDP_ENV_DEBUG);
209 if (env_var_str)
210 {
211 u32 tmp;
212 if (sscanf (env_var_str, "%u", &tmp) != 1)
213 clib_warning ("LDP<%d>: WARNING: Invalid LDP debug level specified in"
214 " the env var " LDP_ENV_DEBUG " (%s)!", getpid (),
215 env_var_str);
216 else
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700217 {
Florin Coras99368312018-08-02 10:45:44 -0700218 ldp->debug = tmp;
Florin Coras05ecfcc2018-12-12 18:19:39 -0800219 LDBG (0, "configured LDP debug level (%u) from env var "
220 LDP_ENV_DEBUG "!", ldp->debug);
Florin Coras99368312018-08-02 10:45:44 -0700221 }
222 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500223
Florin Coras99368312018-08-02 10:45:44 -0700224 env_var_str = getenv (LDP_ENV_APP_NAME);
225 if (env_var_str)
226 {
227 ldp_set_app_name (env_var_str);
Florin Coras05ecfcc2018-12-12 18:19:39 -0800228 LDBG (0, "configured LDP app name (%s) from the env var "
229 LDP_ENV_APP_NAME "!", ldp->app_name);
Florin Coras99368312018-08-02 10:45:44 -0700230 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500231
Florin Coras99368312018-08-02 10:45:44 -0700232 env_var_str = getenv (LDP_ENV_SID_BIT);
233 if (env_var_str)
234 {
235 u32 sb;
236 if (sscanf (env_var_str, "%u", &sb) != 1)
237 {
Florin Coras294afe22019-01-07 17:49:17 -0800238 LDBG (0, "WARNING: Invalid LDP sid bit specified in the env var "
239 LDP_ENV_SID_BIT " (%s)! sid bit value %d (0x%x)", env_var_str,
240 ldp->vlsh_bit_val, ldp->vlsh_bit_val);
Florin Coras99368312018-08-02 10:45:44 -0700241 }
242 else if (sb < LDP_SID_BIT_MIN)
243 {
Florin Coras7baeb712019-01-04 17:05:43 -0800244 ldp->vlsh_bit_val = (1 << LDP_SID_BIT_MIN);
245 ldp->vlsh_bit_mask = ldp->vlsh_bit_val - 1;
Dave Wallace048b1d62018-01-03 22:24:41 -0500246
Florin Coras294afe22019-01-07 17:49:17 -0800247 LDBG (0, "WARNING: LDP sid bit (%u) specified in the env var "
248 LDP_ENV_SID_BIT " (%s) is too small. Using LDP_SID_BIT_MIN"
249 " (%d)! sid bit value %d (0x%x)", sb, env_var_str,
250 LDP_SID_BIT_MIN, ldp->vlsh_bit_val, ldp->vlsh_bit_val);
Florin Coras99368312018-08-02 10:45:44 -0700251 }
252 else if (sb > LDP_SID_BIT_MAX)
253 {
Florin Coras7baeb712019-01-04 17:05:43 -0800254 ldp->vlsh_bit_val = (1 << LDP_SID_BIT_MAX);
255 ldp->vlsh_bit_mask = ldp->vlsh_bit_val - 1;
Dave Wallace048b1d62018-01-03 22:24:41 -0500256
Florin Coras294afe22019-01-07 17:49:17 -0800257 LDBG (0, "WARNING: LDP sid bit (%u) specified in the env var "
258 LDP_ENV_SID_BIT " (%s) is too big. Using LDP_SID_BIT_MAX"
259 " (%d)! sid bit value %d (0x%x)", sb, env_var_str,
260 LDP_SID_BIT_MAX, ldp->vlsh_bit_val, ldp->vlsh_bit_val);
Dave Wallace048b1d62018-01-03 22:24:41 -0500261 }
262 else
263 {
Florin Coras7baeb712019-01-04 17:05:43 -0800264 ldp->vlsh_bit_val = (1 << sb);
265 ldp->vlsh_bit_mask = ldp->vlsh_bit_val - 1;
Florin Coras99368312018-08-02 10:45:44 -0700266
Florin Coras05ecfcc2018-12-12 18:19:39 -0800267 LDBG (0, "configured LDP sid bit (%u) from "
268 LDP_ENV_SID_BIT "! sid bit value %d (0x%x)", sb,
Florin Coras7baeb712019-01-04 17:05:43 -0800269 ldp->vlsh_bit_val, ldp->vlsh_bit_val);
Dave Wallace048b1d62018-01-03 22:24:41 -0500270 }
Florin Coras294afe22019-01-07 17:49:17 -0800271
272 /* Make sure there are enough bits in the fd set for vcl sessions */
273 if (ldp->vlsh_bit_val > FD_SETSIZE / 2)
274 {
275 LDBG (0, "ERROR: LDP vlsh bit value %d > FD_SETSIZE/2 %d!",
276 ldp->vlsh_bit_val, FD_SETSIZE / 2);
277 ldp->init = 0;
278 return -1;
279 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500280 }
Yu Ping7b74b072019-05-08 00:40:24 +0800281 env_var_str = getenv (LDP_ENV_TLS_TRANS);
282 if (env_var_str)
283 {
284 ldp->transparent_tls = 1;
285 }
Florin Coras99368312018-08-02 10:45:44 -0700286
Florin Coras4dee8cd2019-01-29 21:28:16 -0800287 /* *INDENT-OFF* */
Damjan Marionb2c31b62020-12-13 21:47:40 +0100288 pool_foreach (ldpw, ldp->workers) {
Florin Coras4dee8cd2019-01-29 21:28:16 -0800289 clib_memset (&ldpw->clib_time, 0, sizeof (ldpw->clib_time));
Damjan Marionb2c31b62020-12-13 21:47:40 +0100290 }
Florin Coras4dee8cd2019-01-29 21:28:16 -0800291 /* *INDENT-ON* */
292
Florin Coras05ecfcc2018-12-12 18:19:39 -0800293 LDBG (0, "LDP initialization: done!");
Florin Coras99368312018-08-02 10:45:44 -0700294
295 return 0;
Dave Wallace048b1d62018-01-03 22:24:41 -0500296}
297
298int
299close (int fd)
300{
Florin Coras7baeb712019-01-04 17:05:43 -0800301 vls_handle_t vlsh;
302 int rv, epfd;
Dave Wallace048b1d62018-01-03 22:24:41 -0500303
Dave Wallace2a865272018-02-07 21:00:42 -0500304 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500305 return -1;
306
Florin Coras7baeb712019-01-04 17:05:43 -0800307 vlsh = ldp_fd_to_vlsh (fd);
308 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -0500309 {
Florin Coras7baeb712019-01-04 17:05:43 -0800310 epfd = vls_attr (vlsh, VPPCOM_ATTR_GET_LIBC_EPFD, 0, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -0500311 if (epfd > 0)
312 {
Florin Coras7baeb712019-01-04 17:05:43 -0800313 LDBG (0, "fd %d: calling libc_close: epfd %u", fd, epfd);
Dave Wallace048b1d62018-01-03 22:24:41 -0500314
315 rv = libc_close (epfd);
316 if (rv < 0)
317 {
318 u32 size = sizeof (epfd);
319 epfd = 0;
320
Florin Coras7baeb712019-01-04 17:05:43 -0800321 (void) vls_attr (vlsh, VPPCOM_ATTR_SET_LIBC_EPFD, &epfd, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -0500322 }
323 }
324 else if (PREDICT_FALSE (epfd < 0))
325 {
326 errno = -epfd;
327 rv = -1;
328 goto done;
329 }
330
Florin Coras7baeb712019-01-04 17:05:43 -0800331 LDBG (0, "fd %d: calling vls_close: vlsh %u", fd, vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -0500332
Florin Coras7baeb712019-01-04 17:05:43 -0800333 rv = vls_close (vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -0500334 if (rv != VPPCOM_OK)
335 {
336 errno = -rv;
337 rv = -1;
338 }
339 }
340 else
341 {
Florin Coras7baeb712019-01-04 17:05:43 -0800342 LDBG (0, "fd %d: calling libc_close", fd);
Dave Wallace048b1d62018-01-03 22:24:41 -0500343 rv = libc_close (fd);
344 }
345
346done:
Dave Wallace048b1d62018-01-03 22:24:41 -0500347 return rv;
348}
349
350ssize_t
351read (int fd, void *buf, size_t nbytes)
352{
Florin Coras7baeb712019-01-04 17:05:43 -0800353 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -0500354 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -0500355
Dave Wallace2a865272018-02-07 21:00:42 -0500356 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500357 return -1;
358
Florin Coras7baeb712019-01-04 17:05:43 -0800359 vlsh = ldp_fd_to_vlsh (fd);
360 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -0500361 {
Florin Coras7baeb712019-01-04 17:05:43 -0800362 size = vls_read (vlsh, buf, nbytes);
Dave Wallace048b1d62018-01-03 22:24:41 -0500363 if (size < 0)
364 {
365 errno = -size;
366 size = -1;
367 }
368 }
369 else
370 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500371 size = libc_read (fd, buf, nbytes);
372 }
373
Dave Wallace048b1d62018-01-03 22:24:41 -0500374 return size;
375}
376
377ssize_t
378readv (int fd, const struct iovec * iov, int iovcnt)
379{
Dave Wallace8aaba562018-01-18 17:21:19 -0500380 int rv = 0, i, total = 0;
Florin Coras7baeb712019-01-04 17:05:43 -0800381 vls_handle_t vlsh;
382 ssize_t size = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -0500383
Dave Wallace2a865272018-02-07 21:00:42 -0500384 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500385 return -1;
386
Florin Coras7baeb712019-01-04 17:05:43 -0800387 vlsh = ldp_fd_to_vlsh (fd);
388 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -0500389 {
Florin Coras067f9542020-02-14 05:33:46 +0000390 for (i = 0; i < iovcnt; ++i)
Dave Wallace048b1d62018-01-03 22:24:41 -0500391 {
Florin Coras067f9542020-02-14 05:33:46 +0000392 rv = vls_read (vlsh, iov[i].iov_base, iov[i].iov_len);
393 if (rv <= 0)
394 break;
395 else
Dave Wallace048b1d62018-01-03 22:24:41 -0500396 {
Florin Coras067f9542020-02-14 05:33:46 +0000397 total += rv;
398 if (rv < iov[i].iov_len)
Dave Wallace048b1d62018-01-03 22:24:41 -0500399 break;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700400 }
401 }
Florin Coras067f9542020-02-14 05:33:46 +0000402 if (rv < 0 && total == 0)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700403 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500404 errno = -rv;
405 size = -1;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700406 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500407 else
408 size = total;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700409 }
410 else
411 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500412 size = libc_readv (fd, iov, iovcnt);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700413 }
414
Dave Wallace048b1d62018-01-03 22:24:41 -0500415 return size;
416}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700417
Dave Wallace048b1d62018-01-03 22:24:41 -0500418ssize_t
419write (int fd, const void *buf, size_t nbytes)
420{
Florin Coras7baeb712019-01-04 17:05:43 -0800421 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -0500422 ssize_t size = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -0500423
Dave Wallace2a865272018-02-07 21:00:42 -0500424 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500425 return -1;
426
Florin Coras7baeb712019-01-04 17:05:43 -0800427 vlsh = ldp_fd_to_vlsh (fd);
428 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -0500429 {
Florin Coras7baeb712019-01-04 17:05:43 -0800430 size = vls_write_msg (vlsh, (void *) buf, nbytes);
Dave Wallace048b1d62018-01-03 22:24:41 -0500431 if (size < 0)
432 {
433 errno = -size;
434 size = -1;
435 }
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700436 }
437 else
438 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500439 size = libc_write (fd, buf, nbytes);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700440 }
441
Dave Wallace048b1d62018-01-03 22:24:41 -0500442 return size;
443}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700444
Dave Wallace048b1d62018-01-03 22:24:41 -0500445ssize_t
446writev (int fd, const struct iovec * iov, int iovcnt)
447{
Dave Wallace048b1d62018-01-03 22:24:41 -0500448 ssize_t size = 0, total = 0;
Florin Coras7baeb712019-01-04 17:05:43 -0800449 vls_handle_t vlsh;
Dave Wallace8aaba562018-01-18 17:21:19 -0500450 int i, rv = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -0500451
Dave Wallace2a865272018-02-07 21:00:42 -0500452 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500453 return -1;
454
Florin Coras7baeb712019-01-04 17:05:43 -0800455 vlsh = ldp_fd_to_vlsh (fd);
456 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -0500457 {
Florin Coraseda1b8c2020-03-23 16:00:35 +0000458 for (i = 0; i < iovcnt; ++i)
Dave Wallace048b1d62018-01-03 22:24:41 -0500459 {
Florin Coraseda1b8c2020-03-23 16:00:35 +0000460 rv = vls_write_msg (vlsh, iov[i].iov_base, iov[i].iov_len);
461 if (rv < 0)
462 break;
463 else
Dave Wallace048b1d62018-01-03 22:24:41 -0500464 {
Florin Coraseda1b8c2020-03-23 16:00:35 +0000465 total += rv;
466 if (rv < iov[i].iov_len)
Dave Wallace048b1d62018-01-03 22:24:41 -0500467 break;
Dave Wallace048b1d62018-01-03 22:24:41 -0500468 }
469 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500470
Florin Coraseda1b8c2020-03-23 16:00:35 +0000471 if (rv < 0 && total == 0)
Dave Wallace048b1d62018-01-03 22:24:41 -0500472 {
473 errno = -rv;
474 size = -1;
475 }
476 else
477 size = total;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700478 }
479 else
480 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500481 size = libc_writev (fd, iov, iovcnt);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700482 }
483
Dave Wallace048b1d62018-01-03 22:24:41 -0500484 return size;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700485}
486
Florin Coras0ab36f52020-05-26 19:45:45 +0000487static int
488fcntl_internal (int fd, int cmd, va_list ap)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700489{
Florin Coras7baeb712019-01-04 17:05:43 -0800490 vls_handle_t vlsh;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700491 int rv = 0;
Florin Coras7baeb712019-01-04 17:05:43 -0800492
493 vlsh = ldp_fd_to_vlsh (fd);
494 LDBG (0, "fd %u vlsh %d, cmd %u", fd, vlsh, cmd);
495 if (vlsh != VLS_INVALID_HANDLE)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700496 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500497 int flags = va_arg (ap, int);
498 u32 size;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700499
Dave Wallace048b1d62018-01-03 22:24:41 -0500500 size = sizeof (flags);
501 rv = -EOPNOTSUPP;
502 switch (cmd)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700503 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500504 case F_SETFL:
Florin Coras7baeb712019-01-04 17:05:43 -0800505 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_FLAGS, &flags, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -0500506 break;
507
508 case F_GETFL:
Florin Coras7baeb712019-01-04 17:05:43 -0800509 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_FLAGS, &flags, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -0500510 if (rv == VPPCOM_OK)
Florin Coras7baeb712019-01-04 17:05:43 -0800511 rv = flags;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700512 break;
Florin Coras173bae32018-11-16 18:56:28 -0800513 case F_SETFD:
514 /* TODO handle this */
515 LDBG (0, "F_SETFD ignored flags %u", flags);
516 rv = 0;
517 break;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700518 default:
Dave Wallace048b1d62018-01-03 22:24:41 -0500519 rv = -EOPNOTSUPP;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700520 break;
521 }
Dave Wallace048b1d62018-01-03 22:24:41 -0500522 if (rv < 0)
523 {
524 errno = -rv;
525 rv = -1;
526 }
527 }
528 else
529 {
Carl Smithe16707b2019-11-13 14:37:39 +1300530#ifdef HAVE_FCNTL64
531 rv = libc_vfcntl64 (fd, cmd, ap);
532#else
Dave Wallace048b1d62018-01-03 22:24:41 -0500533 rv = libc_vfcntl (fd, cmd, ap);
Carl Smithe16707b2019-11-13 14:37:39 +1300534#endif
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -0700535 }
536
Florin Coras0ab36f52020-05-26 19:45:45 +0000537 return rv;
538}
539
540int
541fcntl (int fd, int cmd, ...)
542{
543 va_list ap;
544 int rv;
545
546 if ((errno = -ldp_init ()))
547 return -1;
548
549 va_start (ap, cmd);
550 rv = fcntl_internal (fd, cmd, ap);
Dave Wallace048b1d62018-01-03 22:24:41 -0500551 va_end (ap);
552
Dave Wallace048b1d62018-01-03 22:24:41 -0500553 return rv;
554}
555
556int
Florin Corasd7586d52020-04-29 02:19:51 +0000557fcntl64 (int fd, int cmd, ...)
558{
559 va_list ap;
560 int rv;
561
Florin Coras0ab36f52020-05-26 19:45:45 +0000562 if ((errno = -ldp_init ()))
563 return -1;
564
Florin Corasd7586d52020-04-29 02:19:51 +0000565 va_start (ap, cmd);
Florin Coras0ab36f52020-05-26 19:45:45 +0000566 rv = fcntl_internal (fd, cmd, ap);
Florin Corasd7586d52020-04-29 02:19:51 +0000567 va_end (ap);
568 return rv;
569}
570
571int
Dave Wallace048b1d62018-01-03 22:24:41 -0500572ioctl (int fd, unsigned long int cmd, ...)
573{
Florin Coras7baeb712019-01-04 17:05:43 -0800574 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -0500575 va_list ap;
Florin Coras7baeb712019-01-04 17:05:43 -0800576 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -0500577
Dave Wallace2a865272018-02-07 21:00:42 -0500578 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500579 return -1;
580
581 va_start (ap, cmd);
Dave Wallace048b1d62018-01-03 22:24:41 -0500582
Florin Coras7baeb712019-01-04 17:05:43 -0800583 vlsh = ldp_fd_to_vlsh (fd);
584 if (vlsh != VLS_INVALID_HANDLE)
585 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500586 switch (cmd)
587 {
588 case FIONREAD:
Florin Coras7baeb712019-01-04 17:05:43 -0800589 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_NREAD, 0, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -0500590 break;
591
592 case FIONBIO:
593 {
594 u32 flags = va_arg (ap, int) ? O_NONBLOCK : 0;
595 u32 size = sizeof (flags);
596
597 /* TBD: When VPPCOM_ATTR_[GS]ET_FLAGS supports flags other than
598 * non-blocking, the flags should be read here and merged
599 * with O_NONBLOCK.
600 */
Florin Coras7baeb712019-01-04 17:05:43 -0800601 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_FLAGS, &flags, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -0500602 }
603 break;
604
605 default:
606 rv = -EOPNOTSUPP;
607 break;
608 }
609 if (rv < 0)
610 {
611 errno = -rv;
612 rv = -1;
613 }
614 }
615 else
616 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500617 rv = libc_vioctl (fd, cmd, ap);
618 }
619
Dave Wallace048b1d62018-01-03 22:24:41 -0500620 va_end (ap);
621 return rv;
622}
623
Florin Coras294afe22019-01-07 17:49:17 -0800624always_inline void
625ldp_select_init_maps (fd_set * __restrict original,
626 clib_bitmap_t ** resultb, clib_bitmap_t ** libcb,
627 clib_bitmap_t ** vclb, int nfds, u32 minbits,
628 u32 n_bytes, uword * si_bits, uword * libc_bits)
629{
630 uword si_bits_set, libc_bits_set;
631 vls_handle_t vlsh;
632 int fd;
633
634 clib_bitmap_validate (*vclb, minbits);
635 clib_bitmap_validate (*libcb, minbits);
636 clib_bitmap_validate (*resultb, minbits);
637 clib_memcpy_fast (*resultb, original, n_bytes);
638 memset (original, 0, n_bytes);
639
640 /* *INDENT-OFF* */
Damjan Marionf0ca1e82020-12-13 23:26:56 +0100641 clib_bitmap_foreach (fd, *resultb) {
Florin Coras294afe22019-01-07 17:49:17 -0800642 if (fd > nfds)
643 break;
644 vlsh = ldp_fd_to_vlsh (fd);
645 if (vlsh == VLS_INVALID_HANDLE)
646 clib_bitmap_set_no_check (*libcb, fd, 1);
647 else
Florin Corascbce80a2020-04-20 01:32:38 +0000648 *vclb = clib_bitmap_set (*vclb, vlsh_to_session_index (vlsh), 1);
Damjan Marionf0ca1e82020-12-13 23:26:56 +0100649 }
Florin Coras294afe22019-01-07 17:49:17 -0800650 /* *INDENT-ON* */
651
652 si_bits_set = clib_bitmap_last_set (*vclb) + 1;
653 *si_bits = (si_bits_set > *si_bits) ? si_bits_set : *si_bits;
Florin Corascbce80a2020-04-20 01:32:38 +0000654 clib_bitmap_validate (*resultb, *si_bits);
Florin Coras294afe22019-01-07 17:49:17 -0800655
656 libc_bits_set = clib_bitmap_last_set (*libcb) + 1;
657 *libc_bits = (libc_bits_set > *libc_bits) ? libc_bits_set : *libc_bits;
658}
659
660always_inline int
661ldp_select_vcl_map_to_libc (clib_bitmap_t * vclb, fd_set * __restrict libcb)
662{
663 vls_handle_t vlsh;
664 uword si;
665 int fd;
666
667 if (!libcb)
668 return 0;
669
670 /* *INDENT-OFF* */
Damjan Marionf0ca1e82020-12-13 23:26:56 +0100671 clib_bitmap_foreach (si, vclb) {
Florin Coras294afe22019-01-07 17:49:17 -0800672 vlsh = vls_session_index_to_vlsh (si);
Florin Coras54140622020-02-04 19:04:34 +0000673 ASSERT (vlsh != VLS_INVALID_HANDLE);
Florin Coras294afe22019-01-07 17:49:17 -0800674 fd = ldp_vlsh_to_fd (vlsh);
675 if (PREDICT_FALSE (fd < 0))
676 {
677 errno = EBADFD;
678 return -1;
679 }
680 FD_SET (fd, libcb);
Damjan Marionf0ca1e82020-12-13 23:26:56 +0100681 }
Florin Coras294afe22019-01-07 17:49:17 -0800682 /* *INDENT-ON* */
683
684 return 0;
685}
686
687always_inline void
688ldp_select_libc_map_merge (clib_bitmap_t * result, fd_set * __restrict libcb)
689{
690 uword fd;
691
Florin Coras78b5fa62019-02-21 20:04:15 -0800692 if (!libcb)
693 return;
694
Florin Coras294afe22019-01-07 17:49:17 -0800695 /* *INDENT-OFF* */
Damjan Marionf0ca1e82020-12-13 23:26:56 +0100696 clib_bitmap_foreach (fd, result)
Florin Coras294afe22019-01-07 17:49:17 -0800697 FD_SET ((int)fd, libcb);
Florin Coras294afe22019-01-07 17:49:17 -0800698 /* *INDENT-ON* */
699}
700
Dave Wallace048b1d62018-01-03 22:24:41 -0500701int
Dave Wallace2a865272018-02-07 21:00:42 -0500702ldp_pselect (int nfds, fd_set * __restrict readfds,
703 fd_set * __restrict writefds,
704 fd_set * __restrict exceptfds,
705 const struct timespec *__restrict timeout,
706 const __sigset_t * __restrict sigmask)
Dave Wallace048b1d62018-01-03 22:24:41 -0500707{
Florin Coras294afe22019-01-07 17:49:17 -0800708 u32 minbits = clib_max (nfds, BITS (uword)), n_bytes;
Florin Corasdfe4cf42018-11-28 22:13:45 -0800709 ldp_worker_ctx_t *ldpw = ldp_worker_get_current ();
Florin Coras294afe22019-01-07 17:49:17 -0800710 struct timespec libc_tspec = { 0 };
711 f64 time_out, vcl_timeout = 0;
712 uword si_bits, libc_bits;
713 int rv, bits_set = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -0500714
715 if (nfds < 0)
716 {
717 errno = EINVAL;
718 return -1;
719 }
720
Florin Coras4dee8cd2019-01-29 21:28:16 -0800721 if (PREDICT_FALSE (ldpw->clib_time.init_cpu_time == 0))
722 clib_time_init (&ldpw->clib_time);
723
Dave Wallace3ee1fe12018-02-23 01:09:11 -0500724 if (timeout)
725 {
726 time_out = (timeout->tv_sec == 0 && timeout->tv_nsec == 0) ?
Florin Coras7baeb712019-01-04 17:05:43 -0800727 (f64) 0 : (f64) timeout->tv_sec + (f64) timeout->tv_nsec / (f64) 1e9;
Dave Wallace3ee1fe12018-02-23 01:09:11 -0500728
729 /* select as fine grained sleep */
730 if (!nfds)
731 {
Florin Corasdfe4cf42018-11-28 22:13:45 -0800732 time_out += clib_time_now (&ldpw->clib_time);
733 while (clib_time_now (&ldpw->clib_time) < time_out)
Dave Wallace3ee1fe12018-02-23 01:09:11 -0500734 ;
735 return 0;
736 }
737 }
738 else if (!nfds)
739 {
740 errno = EINVAL;
741 return -1;
742 }
743 else
744 time_out = -1;
745
Florin Coras7baeb712019-01-04 17:05:43 -0800746 if (nfds <= ldp->vlsh_bit_val)
Dave Wallace048b1d62018-01-03 22:24:41 -0500747 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500748 rv = libc_pselect (nfds, readfds, writefds, exceptfds,
749 timeout, sigmask);
750 goto done;
751 }
752
Florin Coras294afe22019-01-07 17:49:17 -0800753 si_bits = libc_bits = 0;
754 n_bytes = nfds / 8 + ((nfds % 8) ? 1 : 0);
Florin Coras7baeb712019-01-04 17:05:43 -0800755
Dave Wallace048b1d62018-01-03 22:24:41 -0500756 if (readfds)
Florin Coras294afe22019-01-07 17:49:17 -0800757 ldp_select_init_maps (readfds, &ldpw->rd_bitmap, &ldpw->libc_rd_bitmap,
758 &ldpw->si_rd_bitmap, nfds, minbits, n_bytes,
759 &si_bits, &libc_bits);
Dave Wallace048b1d62018-01-03 22:24:41 -0500760 if (writefds)
Florin Coras294afe22019-01-07 17:49:17 -0800761 ldp_select_init_maps (writefds, &ldpw->wr_bitmap,
762 &ldpw->libc_wr_bitmap, &ldpw->si_wr_bitmap, nfds,
763 minbits, n_bytes, &si_bits, &libc_bits);
Dave Wallace048b1d62018-01-03 22:24:41 -0500764 if (exceptfds)
Florin Coras294afe22019-01-07 17:49:17 -0800765 ldp_select_init_maps (exceptfds, &ldpw->ex_bitmap,
766 &ldpw->libc_ex_bitmap, &ldpw->si_ex_bitmap, nfds,
767 minbits, n_bytes, &si_bits, &libc_bits);
Dave Wallace048b1d62018-01-03 22:24:41 -0500768
Florin Coras294afe22019-01-07 17:49:17 -0800769 if (PREDICT_FALSE (!si_bits && !libc_bits))
Dave Wallace048b1d62018-01-03 22:24:41 -0500770 {
771 errno = EINVAL;
772 rv = -1;
773 goto done;
774 }
775
Florin Coras78b5fa62019-02-21 20:04:15 -0800776 if (!si_bits)
777 libc_tspec = timeout ? *timeout : libc_tspec;
Florin Coras294afe22019-01-07 17:49:17 -0800778
Dave Wallace048b1d62018-01-03 22:24:41 -0500779 do
780 {
Florin Coras294afe22019-01-07 17:49:17 -0800781 if (si_bits)
Dave Wallace048b1d62018-01-03 22:24:41 -0500782 {
Dave Wallace048b1d62018-01-03 22:24:41 -0500783 if (readfds)
Florin Coras294afe22019-01-07 17:49:17 -0800784 clib_memcpy_fast (ldpw->rd_bitmap, ldpw->si_rd_bitmap,
Florin Corascbce80a2020-04-20 01:32:38 +0000785 vec_len (ldpw->si_rd_bitmap) *
Dave Barach178cf492018-11-13 16:34:13 -0500786 sizeof (clib_bitmap_t));
Dave Wallace048b1d62018-01-03 22:24:41 -0500787 if (writefds)
Florin Coras294afe22019-01-07 17:49:17 -0800788 clib_memcpy_fast (ldpw->wr_bitmap, ldpw->si_wr_bitmap,
Florin Corascbce80a2020-04-20 01:32:38 +0000789 vec_len (ldpw->si_wr_bitmap) *
Dave Barach178cf492018-11-13 16:34:13 -0500790 sizeof (clib_bitmap_t));
Dave Wallace048b1d62018-01-03 22:24:41 -0500791 if (exceptfds)
Florin Coras294afe22019-01-07 17:49:17 -0800792 clib_memcpy_fast (ldpw->ex_bitmap, ldpw->si_ex_bitmap,
Florin Corascbce80a2020-04-20 01:32:38 +0000793 vec_len (ldpw->si_ex_bitmap) *
Dave Barach178cf492018-11-13 16:34:13 -0500794 sizeof (clib_bitmap_t));
Florin Coras294afe22019-01-07 17:49:17 -0800795
Florin Coras0ef8ef22019-01-18 08:37:13 -0800796 rv = vls_select (si_bits, readfds ? ldpw->rd_bitmap : NULL,
797 writefds ? ldpw->wr_bitmap : NULL,
798 exceptfds ? ldpw->ex_bitmap : NULL, vcl_timeout);
Florin Coras294afe22019-01-07 17:49:17 -0800799 if (rv < 0)
800 {
801 errno = -rv;
802 rv = -1;
Florin Coras5e6222a2020-04-24 17:09:25 +0000803 goto done;
Florin Coras294afe22019-01-07 17:49:17 -0800804 }
805 else if (rv > 0)
806 {
807 if (ldp_select_vcl_map_to_libc (ldpw->rd_bitmap, readfds))
808 {
809 rv = -1;
810 goto done;
811 }
812
813 if (ldp_select_vcl_map_to_libc (ldpw->wr_bitmap, writefds))
814 {
815 rv = -1;
816 goto done;
817 }
818
819 if (ldp_select_vcl_map_to_libc (ldpw->ex_bitmap, exceptfds))
820 {
821 rv = -1;
822 goto done;
823 }
824 bits_set = rv;
825 }
826 }
827 if (libc_bits)
828 {
829 if (readfds)
830 clib_memcpy_fast (ldpw->rd_bitmap, ldpw->libc_rd_bitmap,
831 vec_len (ldpw->libc_rd_bitmap) *
832 sizeof (clib_bitmap_t));
833 if (writefds)
834 clib_memcpy_fast (ldpw->wr_bitmap, ldpw->libc_wr_bitmap,
835 vec_len (ldpw->libc_wr_bitmap) *
836 sizeof (clib_bitmap_t));
837 if (exceptfds)
838 clib_memcpy_fast (ldpw->ex_bitmap, ldpw->libc_ex_bitmap,
839 vec_len (ldpw->libc_ex_bitmap) *
840 sizeof (clib_bitmap_t));
841
Dave Wallace048b1d62018-01-03 22:24:41 -0500842 rv = libc_pselect (libc_bits,
Florin Coras294afe22019-01-07 17:49:17 -0800843 readfds ? (fd_set *) ldpw->rd_bitmap : NULL,
844 writefds ? (fd_set *) ldpw->wr_bitmap : NULL,
845 exceptfds ? (fd_set *) ldpw->ex_bitmap : NULL,
846 &libc_tspec, sigmask);
847 if (rv > 0)
848 {
849 ldp_select_libc_map_merge (ldpw->rd_bitmap, readfds);
850 ldp_select_libc_map_merge (ldpw->wr_bitmap, writefds);
851 ldp_select_libc_map_merge (ldpw->ex_bitmap, exceptfds);
852 bits_set += rv;
853 }
854 }
855
856 if (bits_set)
857 {
858 rv = bits_set;
859 goto done;
Dave Wallace048b1d62018-01-03 22:24:41 -0500860 }
861 }
Florin Corasdfe4cf42018-11-28 22:13:45 -0800862 while ((time_out == -1) || (clib_time_now (&ldpw->clib_time) < time_out));
Dave Wallace048b1d62018-01-03 22:24:41 -0500863 rv = 0;
864
865done:
866 /* TBD: set timeout to amount of time left */
Florin Corasdfe4cf42018-11-28 22:13:45 -0800867 clib_bitmap_zero (ldpw->rd_bitmap);
Florin Coras294afe22019-01-07 17:49:17 -0800868 clib_bitmap_zero (ldpw->si_rd_bitmap);
Florin Corasdfe4cf42018-11-28 22:13:45 -0800869 clib_bitmap_zero (ldpw->libc_rd_bitmap);
870 clib_bitmap_zero (ldpw->wr_bitmap);
Florin Coras294afe22019-01-07 17:49:17 -0800871 clib_bitmap_zero (ldpw->si_wr_bitmap);
Florin Corasdfe4cf42018-11-28 22:13:45 -0800872 clib_bitmap_zero (ldpw->libc_wr_bitmap);
873 clib_bitmap_zero (ldpw->ex_bitmap);
Florin Coras294afe22019-01-07 17:49:17 -0800874 clib_bitmap_zero (ldpw->si_ex_bitmap);
Florin Corasdfe4cf42018-11-28 22:13:45 -0800875 clib_bitmap_zero (ldpw->libc_ex_bitmap);
Dave Wallace048b1d62018-01-03 22:24:41 -0500876
Dave Wallace048b1d62018-01-03 22:24:41 -0500877 return rv;
878}
879
880int
881select (int nfds, fd_set * __restrict readfds,
882 fd_set * __restrict writefds,
883 fd_set * __restrict exceptfds, struct timeval *__restrict timeout)
884{
885 struct timespec tspec;
886
887 if (timeout)
888 {
889 tspec.tv_sec = timeout->tv_sec;
890 tspec.tv_nsec = timeout->tv_usec * 1000;
891 }
Dave Wallace2a865272018-02-07 21:00:42 -0500892 return ldp_pselect (nfds, readfds, writefds, exceptfds,
893 timeout ? &tspec : NULL, NULL);
Dave Wallace048b1d62018-01-03 22:24:41 -0500894}
895
896#ifdef __USE_XOPEN2K
897int
898pselect (int nfds, fd_set * __restrict readfds,
899 fd_set * __restrict writefds,
900 fd_set * __restrict exceptfds,
901 const struct timespec *__restrict timeout,
902 const __sigset_t * __restrict sigmask)
903{
Dave Wallace2a865272018-02-07 21:00:42 -0500904 return ldp_pselect (nfds, readfds, writefds, exceptfds, timeout, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -0500905}
906#endif
907
Yu Ping7b74b072019-05-08 00:40:24 +0800908/* If transparent TLS mode is turned on, then ldp will load key and cert.
909 */
910static int
Florin Corasa5a9efd2021-01-05 17:03:29 -0800911load_cert_key_pair (void)
Yu Ping7b74b072019-05-08 00:40:24 +0800912{
Florin Corasa5a9efd2021-01-05 17:03:29 -0800913 char *cert_str = getenv (LDP_ENV_TLS_CERT);
914 char *key_str = getenv (LDP_ENV_TLS_KEY);
915 char cert_buf[4096], key_buf[4096];
916 int cert_size, key_size;
917 vppcom_cert_key_pair_t crypto;
918 int ckp_index;
Yu Ping7b74b072019-05-08 00:40:24 +0800919 FILE *fp;
920
Florin Corasa5a9efd2021-01-05 17:03:29 -0800921 if (!cert_str || !key_str)
Yu Ping7b74b072019-05-08 00:40:24 +0800922 {
923 LDBG (0, "ERROR: failed to read LDP environment %s\n",
924 LDP_ENV_TLS_CERT);
925 return -1;
926 }
Florin Corasa5a9efd2021-01-05 17:03:29 -0800927
928 fp = fopen (cert_str, "r");
929 if (fp == NULL)
930 {
931 LDBG (0, "ERROR: failed to open cert file %s \n", cert_str);
932 return -1;
933 }
934 cert_size = fread (cert_buf, sizeof (char), sizeof (cert_buf), fp);
935 fclose (fp);
936
937 fp = fopen (key_str, "r");
938 if (fp == NULL)
939 {
940 LDBG (0, "ERROR: failed to open key file %s \n", key_str);
941 return -1;
942 }
943 key_size = fread (key_buf, sizeof (char), sizeof (key_buf), fp);
944 fclose (fp);
945
946 crypto.cert = cert_buf;
947 crypto.key = key_buf;
948 crypto.cert_len = cert_size;
949 crypto.key_len = key_size;
950 ckp_index = vppcom_add_cert_key_pair (&crypto);
951 if (ckp_index < 0)
952 {
953 LDBG (0, "ERROR: failed to add cert key pair\n");
954 return -1;
955 }
956
957 ldp->ckpair_index = ckp_index;
958
Yu Ping7b74b072019-05-08 00:40:24 +0800959 return 0;
960}
961
962static int
Florin Corasa5a9efd2021-01-05 17:03:29 -0800963assign_cert_key_pair (vls_handle_t vlsh)
Yu Ping7b74b072019-05-08 00:40:24 +0800964{
Florin Corasa5a9efd2021-01-05 17:03:29 -0800965 uint32_t ckp_len;
Yu Ping7b74b072019-05-08 00:40:24 +0800966
Florin Corasa5a9efd2021-01-05 17:03:29 -0800967 if (ldp->ckpair_index == ~0 && load_cert_key_pair () < 0)
968 return -1;
969
970 ckp_len = sizeof (ldp->ckpair_index);
971 return vppcom_session_attr (vlsh_to_session_index (vlsh),
972 VPPCOM_ATTR_SET_CKPAIR, &ldp->ckpair_index,
973 &ckp_len);
Yu Ping7b74b072019-05-08 00:40:24 +0800974}
975
Dave Wallace048b1d62018-01-03 22:24:41 -0500976int
977socket (int domain, int type, int protocol)
978{
Florin Coras7baeb712019-01-04 17:05:43 -0800979 int rv, sock_type = type & ~(SOCK_CLOEXEC | SOCK_NONBLOCK);
Dave Wallace048b1d62018-01-03 22:24:41 -0500980 u8 is_nonblocking = type & SOCK_NONBLOCK ? 1 : 0;
Florin Coras7baeb712019-01-04 17:05:43 -0800981 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -0500982
Dave Wallace2a865272018-02-07 21:00:42 -0500983 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -0500984 return -1;
985
986 if (((domain == AF_INET) || (domain == AF_INET6)) &&
987 ((sock_type == SOCK_STREAM) || (sock_type == SOCK_DGRAM)))
988 {
Yu Ping7b74b072019-05-08 00:40:24 +0800989 u8 proto;
990 if (ldp->transparent_tls)
991 {
992 proto = VPPCOM_PROTO_TLS;
993 }
994 else
995 proto = ((sock_type == SOCK_DGRAM) ?
996 VPPCOM_PROTO_UDP : VPPCOM_PROTO_TCP);
Dave Wallace048b1d62018-01-03 22:24:41 -0500997
Florin Coras7baeb712019-01-04 17:05:43 -0800998 LDBG (0, "calling vls_create: proto %u (%s), is_nonblocking %u",
999 proto, vppcom_proto_str (proto), is_nonblocking);
Dave Wallace048b1d62018-01-03 22:24:41 -05001000
Florin Coras7baeb712019-01-04 17:05:43 -08001001 vlsh = vls_create (proto, is_nonblocking);
1002 if (vlsh < 0)
Dave Wallace048b1d62018-01-03 22:24:41 -05001003 {
Florin Coras7baeb712019-01-04 17:05:43 -08001004 errno = -vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001005 rv = -1;
1006 }
1007 else
1008 {
Yu Ping7b74b072019-05-08 00:40:24 +08001009 if (ldp->transparent_tls)
1010 {
Florin Corasa5a9efd2021-01-05 17:03:29 -08001011 if (assign_cert_key_pair (vlsh) < 0)
1012 return -1;
Yu Ping7b74b072019-05-08 00:40:24 +08001013 }
Florin Coras7baeb712019-01-04 17:05:43 -08001014 rv = ldp_vlsh_to_fd (vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -05001015 }
1016 }
1017 else
1018 {
Florin Coras7baeb712019-01-04 17:05:43 -08001019 LDBG (0, "calling libc_socket");
Dave Wallace048b1d62018-01-03 22:24:41 -05001020 rv = libc_socket (domain, type, protocol);
1021 }
1022
Dave Wallace048b1d62018-01-03 22:24:41 -05001023 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001024}
1025
1026/*
1027 * Create two new sockets, of type TYPE in domain DOMAIN and using
1028 * protocol PROTOCOL, which are connected to each other, and put file
1029 * descriptors for them in FDS[0] and FDS[1]. If PROTOCOL is zero,
1030 * one will be chosen automatically.
1031 * Returns 0 on success, -1 for errors.
1032 * */
1033int
Dave Wallace048b1d62018-01-03 22:24:41 -05001034socketpair (int domain, int type, int protocol, int fds[2])
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001035{
Florin Coras7baeb712019-01-04 17:05:43 -08001036 int rv, sock_type = type & ~(SOCK_CLOEXEC | SOCK_NONBLOCK);
Dave Wallace048b1d62018-01-03 22:24:41 -05001037
Dave Wallace2a865272018-02-07 21:00:42 -05001038 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001039 return -1;
1040
1041 if (((domain == AF_INET) || (domain == AF_INET6)) &&
1042 ((sock_type == SOCK_STREAM) || (sock_type == SOCK_DGRAM)))
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001043 {
Florin Coras7baeb712019-01-04 17:05:43 -08001044 LDBG (0, "LDP-TBD");
Dave Wallace048b1d62018-01-03 22:24:41 -05001045 errno = ENOSYS;
1046 rv = -1;
1047 }
1048 else
1049 {
Florin Coras7baeb712019-01-04 17:05:43 -08001050 LDBG (1, "calling libc_socketpair");
Florin Coras173bae32018-11-16 18:56:28 -08001051 rv = libc_socketpair (domain, type, protocol, fds);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001052 }
1053
Dave Wallace048b1d62018-01-03 22:24:41 -05001054 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001055}
1056
1057int
Dave Wallace048b1d62018-01-03 22:24:41 -05001058bind (int fd, __CONST_SOCKADDR_ARG addr, socklen_t len)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001059{
Florin Coras7baeb712019-01-04 17:05:43 -08001060 vls_handle_t vlsh;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001061 int rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001062
Dave Wallace2a865272018-02-07 21:00:42 -05001063 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001064 return -1;
1065
Florin Coras7baeb712019-01-04 17:05:43 -08001066 vlsh = ldp_fd_to_vlsh (fd);
1067 if (vlsh != VLS_INVALID_HANDLE)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001068 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001069 vppcom_endpt_t ep;
1070
Dave Wallace048b1d62018-01-03 22:24:41 -05001071 switch (addr->sa_family)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001072 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001073 case AF_INET:
1074 if (len != sizeof (struct sockaddr_in))
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001075 {
Florin Coras7baeb712019-01-04 17:05:43 -08001076 LDBG (0, "ERROR: fd %d: vlsh %u: Invalid AF_INET addr len %u!",
1077 fd, vlsh, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001078 errno = EINVAL;
1079 rv = -1;
1080 goto done;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001081 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001082 ep.is_ip4 = VPPCOM_IS_IP4;
1083 ep.ip = (u8 *) & ((const struct sockaddr_in *) addr)->sin_addr;
1084 ep.port = (u16) ((const struct sockaddr_in *) addr)->sin_port;
1085 break;
1086
1087 case AF_INET6:
1088 if (len != sizeof (struct sockaddr_in6))
1089 {
Florin Coras7baeb712019-01-04 17:05:43 -08001090 LDBG (0, "ERROR: fd %d: vlsh %u: Invalid AF_INET6 addr len %u!",
1091 fd, vlsh, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001092 errno = EINVAL;
1093 rv = -1;
1094 goto done;
1095 }
1096 ep.is_ip4 = VPPCOM_IS_IP6;
1097 ep.ip = (u8 *) & ((const struct sockaddr_in6 *) addr)->sin6_addr;
1098 ep.port = (u16) ((const struct sockaddr_in6 *) addr)->sin6_port;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001099 break;
1100
1101 default:
Florin Coras7baeb712019-01-04 17:05:43 -08001102 LDBG (0, "ERROR: fd %d: vlsh %u: Unsupported address family %u!",
1103 fd, vlsh, addr->sa_family);
Dave Wallace048b1d62018-01-03 22:24:41 -05001104 errno = EAFNOSUPPORT;
1105 rv = -1;
1106 goto done;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001107 }
Florin Coras7baeb712019-01-04 17:05:43 -08001108 LDBG (0, "fd %d: calling vls_bind: vlsh %u, addr %p, len %u", fd, vlsh,
1109 addr, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001110
Florin Coras7baeb712019-01-04 17:05:43 -08001111 rv = vls_bind (vlsh, &ep);
Dave Wallace048b1d62018-01-03 22:24:41 -05001112 if (rv != VPPCOM_OK)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001113 {
1114 errno = -rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05001115 rv = -1;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001116 }
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001117 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001118 else
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001119 {
Florin Coras7baeb712019-01-04 17:05:43 -08001120 LDBG (0, "fd %d: calling libc_bind: addr %p, len %u", fd, addr, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001121 rv = libc_bind (fd, addr, len);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001122 }
1123
Dave Wallace048b1d62018-01-03 22:24:41 -05001124done:
Florin Coras7baeb712019-01-04 17:05:43 -08001125 LDBG (1, "fd %d: returning %d", fd, rv);
Florin Coras05ecfcc2018-12-12 18:19:39 -08001126
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001127 return rv;
1128}
1129
1130static inline int
Dave Wallace2a865272018-02-07 21:00:42 -05001131ldp_copy_ep_to_sockaddr (__SOCKADDR_ARG addr, socklen_t * __restrict len,
1132 vppcom_endpt_t * ep)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001133{
Dave Wallace048b1d62018-01-03 22:24:41 -05001134 int rv = 0;
1135 int sa_len, copy_len;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001136
Dave Wallace2a865272018-02-07 21:00:42 -05001137 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001138 return -1;
1139
1140 if (addr && len && ep)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001141 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001142 addr->sa_family = (ep->is_ip4 == VPPCOM_IS_IP4) ? AF_INET : AF_INET6;
1143 switch (addr->sa_family)
1144 {
1145 case AF_INET:
1146 ((struct sockaddr_in *) addr)->sin_port = ep->port;
1147 if (*len > sizeof (struct sockaddr_in))
1148 *len = sizeof (struct sockaddr_in);
1149 sa_len = sizeof (struct sockaddr_in) - sizeof (struct in_addr);
1150 copy_len = *len - sa_len;
1151 if (copy_len > 0)
1152 memcpy (&((struct sockaddr_in *) addr)->sin_addr, ep->ip,
1153 copy_len);
1154 break;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001155
Dave Wallace048b1d62018-01-03 22:24:41 -05001156 case AF_INET6:
1157 ((struct sockaddr_in6 *) addr)->sin6_port = ep->port;
1158 if (*len > sizeof (struct sockaddr_in6))
1159 *len = sizeof (struct sockaddr_in6);
1160 sa_len = sizeof (struct sockaddr_in6) - sizeof (struct in6_addr);
1161 copy_len = *len - sa_len;
1162 if (copy_len > 0)
1163 memcpy (((struct sockaddr_in6 *) addr)->sin6_addr.
1164 __in6_u.__u6_addr8, ep->ip, copy_len);
1165 break;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001166
Dave Wallace048b1d62018-01-03 22:24:41 -05001167 default:
1168 /* Not possible */
1169 rv = -EAFNOSUPPORT;
1170 break;
1171 }
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001172 }
Dave Wallacee695cb42017-11-02 22:04:42 -04001173 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001174}
1175
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001176int
Dave Wallace048b1d62018-01-03 22:24:41 -05001177getsockname (int fd, __SOCKADDR_ARG addr, socklen_t * __restrict len)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001178{
Florin Coras7baeb712019-01-04 17:05:43 -08001179 vls_handle_t vlsh;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001180 int rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001181
Dave Wallace2a865272018-02-07 21:00:42 -05001182 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001183 return -1;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001184
Florin Coras7baeb712019-01-04 17:05:43 -08001185 vlsh = ldp_fd_to_vlsh (fd);
1186 if (vlsh != VLS_INVALID_HANDLE)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001187 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001188 vppcom_endpt_t ep;
1189 u8 addr_buf[sizeof (struct in6_addr)];
1190 u32 size = sizeof (ep);
1191
1192 ep.ip = addr_buf;
Dave Wallace048b1d62018-01-03 22:24:41 -05001193
Florin Coras7baeb712019-01-04 17:05:43 -08001194 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_LCL_ADDR, &ep, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -05001195 if (rv != VPPCOM_OK)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001196 {
1197 errno = -rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05001198 rv = -1;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001199 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001200 else
1201 {
Dave Wallace2a865272018-02-07 21:00:42 -05001202 rv = ldp_copy_ep_to_sockaddr (addr, len, &ep);
Dave Wallace048b1d62018-01-03 22:24:41 -05001203 if (rv != VPPCOM_OK)
1204 {
1205 errno = -rv;
1206 rv = -1;
1207 }
1208 }
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001209 }
1210 else
1211 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001212 rv = libc_getsockname (fd, addr, len);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001213 }
1214
Dave Wallace048b1d62018-01-03 22:24:41 -05001215 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001216}
1217
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001218int
Dave Wallace048b1d62018-01-03 22:24:41 -05001219connect (int fd, __CONST_SOCKADDR_ARG addr, socklen_t len)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001220{
Florin Coras7baeb712019-01-04 17:05:43 -08001221 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001222 int rv;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001223
Dave Wallace2a865272018-02-07 21:00:42 -05001224 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001225 return -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001226
Dave Wallace048b1d62018-01-03 22:24:41 -05001227 if (!addr)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001228 {
Florin Coras7baeb712019-01-04 17:05:43 -08001229 LDBG (0, "ERROR: fd %d: NULL addr, len %u", fd, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001230 errno = EINVAL;
1231 rv = -1;
1232 goto done;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07001233 }
1234
Florin Coras7baeb712019-01-04 17:05:43 -08001235 vlsh = ldp_fd_to_vlsh (fd);
1236 if (vlsh != VLS_INVALID_HANDLE)
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001237 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001238 vppcom_endpt_t ep;
1239
Dave Wallace048b1d62018-01-03 22:24:41 -05001240 switch (addr->sa_family)
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001241 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001242 case AF_INET:
1243 if (len != sizeof (struct sockaddr_in))
1244 {
Florin Coras7baeb712019-01-04 17:05:43 -08001245 LDBG (0, "fd %d: ERROR vlsh %u: Invalid AF_INET addr len %u!",
1246 fd, vlsh, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001247 errno = EINVAL;
1248 rv = -1;
1249 goto done;
1250 }
1251 ep.is_ip4 = VPPCOM_IS_IP4;
1252 ep.ip = (u8 *) & ((const struct sockaddr_in *) addr)->sin_addr;
1253 ep.port = (u16) ((const struct sockaddr_in *) addr)->sin_port;
1254 break;
1255
1256 case AF_INET6:
1257 if (len != sizeof (struct sockaddr_in6))
1258 {
Florin Coras7baeb712019-01-04 17:05:43 -08001259 LDBG (0, "fd %d: ERROR vlsh %u: Invalid AF_INET6 addr len %u!",
1260 fd, vlsh, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001261 errno = EINVAL;
1262 rv = -1;
1263 goto done;
1264 }
1265 ep.is_ip4 = VPPCOM_IS_IP6;
1266 ep.ip = (u8 *) & ((const struct sockaddr_in6 *) addr)->sin6_addr;
1267 ep.port = (u16) ((const struct sockaddr_in6 *) addr)->sin6_port;
1268 break;
1269
1270 default:
Florin Coras7baeb712019-01-04 17:05:43 -08001271 LDBG (0, "fd %d: ERROR vlsh %u: Unsupported address family %u!",
1272 fd, vlsh, addr->sa_family);
Dave Wallace048b1d62018-01-03 22:24:41 -05001273 errno = EAFNOSUPPORT;
1274 rv = -1;
1275 goto done;
1276 }
Florin Coras7baeb712019-01-04 17:05:43 -08001277 LDBG (0, "fd %d: calling vls_connect(): vlsh %u addr %p len %u", fd,
1278 vlsh, addr, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001279
Florin Coras7baeb712019-01-04 17:05:43 -08001280 rv = vls_connect (vlsh, &ep);
Dave Wallace048b1d62018-01-03 22:24:41 -05001281 if (rv != VPPCOM_OK)
1282 {
1283 errno = -rv;
1284 rv = -1;
1285 }
1286 }
1287 else
1288 {
Florin Coras7baeb712019-01-04 17:05:43 -08001289 LDBG (0, "fd %d: calling libc_connect(): addr %p, len %u",
1290 fd, addr, len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001291
1292 rv = libc_connect (fd, addr, len);
1293 }
1294
1295done:
Florin Coras7baeb712019-01-04 17:05:43 -08001296 LDBG (1, "fd %d: returning %d (0x%x)", fd, rv, rv);
Dave Wallace048b1d62018-01-03 22:24:41 -05001297 return rv;
1298}
1299
1300int
1301getpeername (int fd, __SOCKADDR_ARG addr, socklen_t * __restrict len)
1302{
Florin Coras7baeb712019-01-04 17:05:43 -08001303 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001304 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05001305
Dave Wallace2a865272018-02-07 21:00:42 -05001306 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001307 return -1;
1308
Florin Coras7baeb712019-01-04 17:05:43 -08001309 vlsh = ldp_fd_to_vlsh (fd);
1310 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001311 {
1312 vppcom_endpt_t ep;
1313 u8 addr_buf[sizeof (struct in6_addr)];
1314 u32 size = sizeof (ep);
1315
1316 ep.ip = addr_buf;
Florin Coras7baeb712019-01-04 17:05:43 -08001317 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_PEER_ADDR, &ep, &size);
Dave Wallace048b1d62018-01-03 22:24:41 -05001318 if (rv != VPPCOM_OK)
1319 {
1320 errno = -rv;
1321 rv = -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001322 }
1323 else
1324 {
Dave Wallace2a865272018-02-07 21:00:42 -05001325 rv = ldp_copy_ep_to_sockaddr (addr, len, &ep);
Dave Wallace048b1d62018-01-03 22:24:41 -05001326 if (rv != VPPCOM_OK)
1327 {
1328 errno = -rv;
1329 rv = -1;
1330 }
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001331 }
1332 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001333 else
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001334 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001335 rv = libc_getpeername (fd, addr, len);
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001336 }
1337
Dave Wallace048b1d62018-01-03 22:24:41 -05001338 return rv;
1339}
1340
1341ssize_t
1342send (int fd, const void *buf, size_t n, int flags)
1343{
Florin Coras7baeb712019-01-04 17:05:43 -08001344 vls_handle_t vlsh = ldp_fd_to_vlsh (fd);
Dave Wallace048b1d62018-01-03 22:24:41 -05001345 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001346
Dave Wallace2a865272018-02-07 21:00:42 -05001347 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001348 return -1;
1349
Florin Coras7baeb712019-01-04 17:05:43 -08001350 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001351 {
Florin Coras7baeb712019-01-04 17:05:43 -08001352 size = vls_sendto (vlsh, (void *) buf, n, flags, NULL);
qchangaa8f63c2018-05-30 11:44:18 -07001353 if (size < VPPCOM_OK)
Dave Wallace048b1d62018-01-03 22:24:41 -05001354 {
1355 errno = -size;
1356 size = -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001357 }
1358 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001359 else
1360 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001361 size = libc_send (fd, buf, n, flags);
1362 }
1363
Dave Wallace048b1d62018-01-03 22:24:41 -05001364 return size;
1365}
1366
1367ssize_t
1368sendfile (int out_fd, int in_fd, off_t * offset, size_t len)
1369{
Florin Corasdfe4cf42018-11-28 22:13:45 -08001370 ldp_worker_ctx_t *ldpw = ldp_worker_get_current ();
Florin Coras7baeb712019-01-04 17:05:43 -08001371 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001372 ssize_t size = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -05001373
Dave Wallace2a865272018-02-07 21:00:42 -05001374 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001375 return -1;
1376
Florin Coras7baeb712019-01-04 17:05:43 -08001377 vlsh = ldp_fd_to_vlsh (out_fd);
1378 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001379 {
1380 int rv;
1381 ssize_t results = 0;
1382 size_t n_bytes_left = len;
1383 size_t bytes_to_read;
1384 int nbytes;
Dave Wallace048b1d62018-01-03 22:24:41 -05001385 u8 eagain = 0;
1386 u32 flags, flags_len = sizeof (flags);
1387
Florin Coras7baeb712019-01-04 17:05:43 -08001388 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_FLAGS, &flags, &flags_len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001389 if (PREDICT_FALSE (rv != VPPCOM_OK))
1390 {
Florin Coras7baeb712019-01-04 17:05:43 -08001391 LDBG (0, "ERROR: out fd %d: vls_attr: vlsh %u, returned %d (%s)!",
1392 out_fd, vlsh, rv, vppcom_retval_str (rv));
Dave Wallace048b1d62018-01-03 22:24:41 -05001393
Florin Corasdfe4cf42018-11-28 22:13:45 -08001394 vec_reset_length (ldpw->io_buffer);
Dave Wallace048b1d62018-01-03 22:24:41 -05001395 errno = -rv;
1396 size = -1;
1397 goto done;
1398 }
1399
1400 if (offset)
1401 {
1402 off_t off = lseek (in_fd, *offset, SEEK_SET);
1403 if (PREDICT_FALSE (off == -1))
1404 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001405 size = -1;
1406 goto done;
1407 }
1408
1409 ASSERT (off == *offset);
1410 }
1411
1412 do
1413 {
Florin Coras7baeb712019-01-04 17:05:43 -08001414 size = vls_attr (vlsh, VPPCOM_ATTR_GET_NWRITE, 0, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05001415 if (size < 0)
1416 {
Florin Coraseb801d02020-09-16 17:44:58 -07001417 LDBG (0, "ERROR: fd %d: vls_attr: vlsh %u returned %ld (%s)!",
Florin Coras7baeb712019-01-04 17:05:43 -08001418 out_fd, vlsh, size, vppcom_retval_str (size));
Florin Corasdfe4cf42018-11-28 22:13:45 -08001419 vec_reset_length (ldpw->io_buffer);
Dave Wallace048b1d62018-01-03 22:24:41 -05001420 errno = -size;
1421 size = -1;
1422 goto done;
1423 }
1424
1425 bytes_to_read = size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001426 if (bytes_to_read == 0)
1427 {
1428 if (flags & O_NONBLOCK)
1429 {
1430 if (!results)
Florin Coras7baeb712019-01-04 17:05:43 -08001431 eagain = 1;
Dave Wallace048b1d62018-01-03 22:24:41 -05001432 goto update_offset;
1433 }
1434 else
1435 continue;
1436 }
1437 bytes_to_read = clib_min (n_bytes_left, bytes_to_read);
Florin Corasdfe4cf42018-11-28 22:13:45 -08001438 vec_validate (ldpw->io_buffer, bytes_to_read);
1439 nbytes = libc_read (in_fd, ldpw->io_buffer, bytes_to_read);
Dave Wallace048b1d62018-01-03 22:24:41 -05001440 if (nbytes < 0)
1441 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001442 if (results == 0)
1443 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08001444 vec_reset_length (ldpw->io_buffer);
Dave Wallace048b1d62018-01-03 22:24:41 -05001445 size = -1;
1446 goto done;
1447 }
1448 goto update_offset;
1449 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001450
Florin Coras7baeb712019-01-04 17:05:43 -08001451 size = vls_write (vlsh, ldpw->io_buffer, nbytes);
Dave Wallace048b1d62018-01-03 22:24:41 -05001452 if (size < 0)
1453 {
1454 if (size == VPPCOM_EAGAIN)
1455 {
1456 if (flags & O_NONBLOCK)
1457 {
1458 if (!results)
Florin Coras7baeb712019-01-04 17:05:43 -08001459 eagain = 1;
Dave Wallace048b1d62018-01-03 22:24:41 -05001460 goto update_offset;
1461 }
1462 else
1463 continue;
1464 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001465 if (results == 0)
1466 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08001467 vec_reset_length (ldpw->io_buffer);
Dave Wallace048b1d62018-01-03 22:24:41 -05001468 errno = -size;
1469 size = -1;
1470 goto done;
1471 }
1472 goto update_offset;
1473 }
1474
1475 results += nbytes;
1476 ASSERT (n_bytes_left >= nbytes);
1477 n_bytes_left = n_bytes_left - nbytes;
1478 }
1479 while (n_bytes_left > 0);
1480
1481 update_offset:
Florin Corasdfe4cf42018-11-28 22:13:45 -08001482 vec_reset_length (ldpw->io_buffer);
Dave Wallace048b1d62018-01-03 22:24:41 -05001483 if (offset)
1484 {
1485 off_t off = lseek (in_fd, *offset, SEEK_SET);
1486 if (PREDICT_FALSE (off == -1))
1487 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001488 size = -1;
1489 goto done;
1490 }
1491
1492 ASSERT (off == *offset);
1493 *offset += results + 1;
1494 }
1495 if (eagain)
1496 {
1497 errno = EAGAIN;
1498 size = -1;
1499 }
1500 else
1501 size = results;
1502 }
1503 else
1504 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001505 size = libc_sendfile (out_fd, in_fd, offset, len);
1506 }
1507
1508done:
Dave Wallace048b1d62018-01-03 22:24:41 -05001509 return size;
1510}
1511
1512ssize_t
1513sendfile64 (int out_fd, int in_fd, off_t * offset, size_t len)
1514{
1515 return sendfile (out_fd, in_fd, offset, len);
1516}
1517
1518ssize_t
1519recv (int fd, void *buf, size_t n, int flags)
1520{
Florin Coras7baeb712019-01-04 17:05:43 -08001521 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001522 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001523
Dave Wallace2a865272018-02-07 21:00:42 -05001524 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001525 return -1;
1526
Florin Coras7baeb712019-01-04 17:05:43 -08001527 vlsh = ldp_fd_to_vlsh (fd);
1528 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001529 {
Florin Coras7baeb712019-01-04 17:05:43 -08001530 size = vls_recvfrom (vlsh, buf, n, flags, NULL);
Dave Wallace048b1d62018-01-03 22:24:41 -05001531 if (size < 0)
Florin Coras2a6642e2020-03-24 15:24:29 +00001532 {
1533 errno = -size;
1534 size = -1;
1535 }
Dave Wallace048b1d62018-01-03 22:24:41 -05001536 }
1537 else
1538 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001539 size = libc_recv (fd, buf, n, flags);
1540 }
1541
Dave Wallace048b1d62018-01-03 22:24:41 -05001542 return size;
1543}
1544
Florin Corasce17f462020-05-22 20:36:29 +00001545static int
1546ldp_vls_sendo (vls_handle_t vlsh, const void *buf, size_t n, int flags,
1547 __CONST_SOCKADDR_ARG addr, socklen_t addr_len)
1548{
1549 vppcom_endpt_t *ep = 0;
1550 vppcom_endpt_t _ep;
1551
1552 if (addr)
1553 {
1554 ep = &_ep;
1555 switch (addr->sa_family)
1556 {
1557 case AF_INET:
1558 ep->is_ip4 = VPPCOM_IS_IP4;
1559 ep->ip =
1560 (uint8_t *) & ((const struct sockaddr_in *) addr)->sin_addr;
1561 ep->port = (uint16_t) ((const struct sockaddr_in *) addr)->sin_port;
1562 break;
1563
1564 case AF_INET6:
1565 ep->is_ip4 = VPPCOM_IS_IP6;
1566 ep->ip =
1567 (uint8_t *) & ((const struct sockaddr_in6 *) addr)->sin6_addr;
1568 ep->port =
1569 (uint16_t) ((const struct sockaddr_in6 *) addr)->sin6_port;
1570 break;
1571
1572 default:
1573 return EAFNOSUPPORT;
1574 }
1575 }
1576
1577 return vls_sendto (vlsh, (void *) buf, n, flags, ep);
1578}
1579
1580static int
1581ldp_vls_recvfrom (vls_handle_t vlsh, void *__restrict buf, size_t n,
1582 int flags, __SOCKADDR_ARG addr,
1583 socklen_t * __restrict addr_len)
1584{
1585 u8 src_addr[sizeof (struct sockaddr_in6)];
1586 vppcom_endpt_t ep;
1587 ssize_t size;
1588 int rv;
1589
1590 if (addr)
1591 {
1592 ep.ip = src_addr;
1593 size = vls_recvfrom (vlsh, buf, n, flags, &ep);
1594
1595 if (size > 0)
1596 {
1597 rv = ldp_copy_ep_to_sockaddr (addr, addr_len, &ep);
1598 if (rv < 0)
1599 size = rv;
1600 }
1601 }
1602 else
1603 size = vls_recvfrom (vlsh, buf, n, flags, NULL);
1604
1605 return size;
1606}
1607
Dave Wallace048b1d62018-01-03 22:24:41 -05001608ssize_t
1609sendto (int fd, const void *buf, size_t n, int flags,
1610 __CONST_SOCKADDR_ARG addr, socklen_t addr_len)
1611{
Florin Coras7baeb712019-01-04 17:05:43 -08001612 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001613 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001614
Dave Wallace2a865272018-02-07 21:00:42 -05001615 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001616 return -1;
1617
Florin Coras7baeb712019-01-04 17:05:43 -08001618 vlsh = ldp_fd_to_vlsh (fd);
1619 if (vlsh != INVALID_SESSION_ID)
Dave Wallace048b1d62018-01-03 22:24:41 -05001620 {
Florin Corasce17f462020-05-22 20:36:29 +00001621 size = ldp_vls_sendo (vlsh, buf, n, flags, addr, addr_len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001622 if (size < 0)
1623 {
1624 errno = -size;
1625 size = -1;
1626 }
1627 }
1628 else
1629 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001630 size = libc_sendto (fd, buf, n, flags, addr, addr_len);
1631 }
1632
Dave Wallace048b1d62018-01-03 22:24:41 -05001633 return size;
1634}
1635
1636ssize_t
1637recvfrom (int fd, void *__restrict buf, size_t n, int flags,
1638 __SOCKADDR_ARG addr, socklen_t * __restrict addr_len)
1639{
Florin Corasce17f462020-05-22 20:36:29 +00001640 vls_handle_t vlsh;
1641 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001642
Dave Wallace2a865272018-02-07 21:00:42 -05001643 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001644 return -1;
1645
Florin Corasce17f462020-05-22 20:36:29 +00001646 vlsh = ldp_fd_to_vlsh (fd);
1647 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001648 {
Florin Corasce17f462020-05-22 20:36:29 +00001649 size = ldp_vls_recvfrom (vlsh, buf, n, flags, addr, addr_len);
Dave Wallace048b1d62018-01-03 22:24:41 -05001650 if (size < 0)
1651 {
1652 errno = -size;
1653 size = -1;
1654 }
1655 }
1656 else
1657 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001658 size = libc_recvfrom (fd, buf, n, flags, addr, addr_len);
1659 }
1660
Dave Wallace048b1d62018-01-03 22:24:41 -05001661 return size;
1662}
1663
1664ssize_t
Florin Corasce17f462020-05-22 20:36:29 +00001665sendmsg (int fd, const struct msghdr * msg, int flags)
Dave Wallace048b1d62018-01-03 22:24:41 -05001666{
Florin Coras7baeb712019-01-04 17:05:43 -08001667 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001668 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001669
Dave Wallace2a865272018-02-07 21:00:42 -05001670 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001671 return -1;
1672
Florin Coras7baeb712019-01-04 17:05:43 -08001673 vlsh = ldp_fd_to_vlsh (fd);
1674 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001675 {
Florin Corasce17f462020-05-22 20:36:29 +00001676 struct iovec *iov = msg->msg_iov;
1677 ssize_t total = 0;
1678 int i, rv;
1679
1680 for (i = 0; i < msg->msg_iovlen; ++i)
1681 {
1682 rv = ldp_vls_sendo (vlsh, iov[i].iov_base, iov[i].iov_len, flags,
1683 msg->msg_name, msg->msg_namelen);
1684 if (rv < 0)
1685 break;
1686 else
1687 {
1688 total += rv;
1689 if (rv < iov[i].iov_len)
1690 break;
1691 }
1692 }
1693
1694 if (rv < 0 && total == 0)
1695 {
1696 errno = -rv;
1697 size = -1;
1698 }
1699 else
1700 size = total;
Dave Wallace048b1d62018-01-03 22:24:41 -05001701 }
1702 else
1703 {
Florin Corasce17f462020-05-22 20:36:29 +00001704 size = libc_sendmsg (fd, msg, flags);
Dave Wallace048b1d62018-01-03 22:24:41 -05001705 }
1706
Dave Wallace048b1d62018-01-03 22:24:41 -05001707 return size;
1708}
1709
1710#ifdef USE_GNU
1711int
1712sendmmsg (int fd, struct mmsghdr *vmessages, unsigned int vlen, int flags)
1713{
1714 ssize_t size;
1715 const char *func_str;
Florin Coras7baeb712019-01-04 17:05:43 -08001716 u32 sh = ldp_fd_to_vlsh (fd);
Dave Wallace048b1d62018-01-03 22:24:41 -05001717
Dave Wallace2a865272018-02-07 21:00:42 -05001718 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001719 return -1;
1720
Florin Coras7baeb712019-01-04 17:05:43 -08001721 if (sh != INVALID_SESSION_ID)
Dave Wallace048b1d62018-01-03 22:24:41 -05001722 {
1723 clib_warning ("LDP<%d>: LDP-TBD", getpid ());
1724 errno = ENOSYS;
1725 size = -1;
1726 }
1727 else
1728 {
1729 func_str = "libc_sendmmsg";
1730
Dave Wallace2a865272018-02-07 21:00:42 -05001731 if (LDP_DEBUG > 2)
Dave Wallace048b1d62018-01-03 22:24:41 -05001732 clib_warning ("LDP<%d>: fd %d (0x%x): calling %s(): "
1733 "vmessages %p, vlen %u, flags 0x%x",
1734 getpid (), fd, fd, func_str, vmessages, vlen, flags);
1735
1736 size = libc_sendmmsg (fd, vmessages, vlen, flags);
1737 }
1738
Dave Wallace2a865272018-02-07 21:00:42 -05001739 if (LDP_DEBUG > 2)
Dave Wallace048b1d62018-01-03 22:24:41 -05001740 {
1741 if (size < 0)
1742 {
1743 int errno_val = errno;
1744 perror (func_str);
1745 clib_warning ("LDP<%d>: ERROR: fd %d (0x%x): %s() failed! "
1746 "rv %d, errno = %d", getpid (), fd, fd,
1747 func_str, size, errno_val);
1748 errno = errno_val;
1749 }
1750 else
1751 clib_warning ("LDP<%d>: fd %d (0x%x): returning %d (0x%x)",
1752 getpid (), fd, fd, size, size);
1753 }
1754 return size;
1755}
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001756#endif
1757
Dave Wallace048b1d62018-01-03 22:24:41 -05001758ssize_t
Florin Corasce17f462020-05-22 20:36:29 +00001759recvmsg (int fd, struct msghdr * msg, int flags)
Dave Wallace048b1d62018-01-03 22:24:41 -05001760{
Florin Coras7baeb712019-01-04 17:05:43 -08001761 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001762 ssize_t size;
Dave Wallace048b1d62018-01-03 22:24:41 -05001763
Dave Wallace2a865272018-02-07 21:00:42 -05001764 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001765 return -1;
1766
Florin Coras7baeb712019-01-04 17:05:43 -08001767 vlsh = ldp_fd_to_vlsh (fd);
1768 if (vlsh != VLS_INVALID_HANDLE)
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001769 {
Florin Corasce17f462020-05-22 20:36:29 +00001770 struct iovec *iov = msg->msg_iov;
1771 ssize_t max_deq, total = 0;
1772 int i, rv;
1773
1774 max_deq = vls_attr (vlsh, VPPCOM_ATTR_GET_NREAD, 0, 0);
1775 if (!max_deq)
1776 return 0;
1777
1778 for (i = 0; i < msg->msg_iovlen; i++)
1779 {
1780 rv = ldp_vls_recvfrom (vlsh, iov[i].iov_base, iov[i].iov_len, flags,
1781 (i == 0 ? msg->msg_name : NULL),
1782 (i == 0 ? &msg->msg_namelen : NULL));
1783 if (rv <= 0)
1784 break;
1785 else
1786 {
1787 total += rv;
1788 if (rv < iov[i].iov_len)
1789 break;
1790 }
1791 if (total >= max_deq)
1792 break;
1793 }
1794
1795 if (rv < 0 && total == 0)
1796 {
1797 errno = -rv;
1798 size = -1;
1799 }
1800 else
1801 size = total;
Dave Wallace048b1d62018-01-03 22:24:41 -05001802 }
1803 else
1804 {
Florin Corasce17f462020-05-22 20:36:29 +00001805 size = libc_recvmsg (fd, msg, flags);
Dave Wallace048b1d62018-01-03 22:24:41 -05001806 }
1807
Dave Wallace048b1d62018-01-03 22:24:41 -05001808 return size;
1809}
1810
1811#ifdef USE_GNU
1812int
1813recvmmsg (int fd, struct mmsghdr *vmessages,
1814 unsigned int vlen, int flags, struct timespec *tmo)
1815{
1816 ssize_t size;
1817 const char *func_str;
Florin Coras7baeb712019-01-04 17:05:43 -08001818 u32 sh = ldp_fd_to_vlsh (fd);
Dave Wallace048b1d62018-01-03 22:24:41 -05001819
Dave Wallace2a865272018-02-07 21:00:42 -05001820 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001821 return -1;
1822
Florin Coras7baeb712019-01-04 17:05:43 -08001823 if (sh != INVALID_SESSION_ID)
Dave Wallace048b1d62018-01-03 22:24:41 -05001824 {
1825 clib_warning ("LDP<%d>: LDP-TBD", getpid ());
1826 errno = ENOSYS;
1827 size = -1;
1828 }
1829 else
1830 {
1831 func_str = "libc_recvmmsg";
1832
Dave Wallace2a865272018-02-07 21:00:42 -05001833 if (LDP_DEBUG > 2)
Dave Wallace048b1d62018-01-03 22:24:41 -05001834 clib_warning ("LDP<%d>: fd %d (0x%x): calling %s(): "
1835 "vmessages %p, vlen %u, flags 0x%x, tmo %p",
1836 getpid (), fd, fd, func_str, vmessages, vlen,
1837 flags, tmo);
1838
1839 size = libc_recvmmsg (fd, vmessages, vlen, flags, tmo);
1840 }
1841
Dave Wallace2a865272018-02-07 21:00:42 -05001842 if (LDP_DEBUG > 2)
Dave Wallace048b1d62018-01-03 22:24:41 -05001843 {
1844 if (size < 0)
1845 {
1846 int errno_val = errno;
1847 perror (func_str);
1848 clib_warning ("LDP<%d>: ERROR: fd %d (0x%x): %s() failed! "
1849 "rv %d, errno = %d", getpid (), fd, fd,
1850 func_str, size, errno_val);
1851 errno = errno_val;
1852 }
1853 else
1854 clib_warning ("LDP<%d>: fd %d (0x%x): returning %d (0x%x)",
1855 getpid (), fd, fd, size, size);
1856 }
1857 return size;
1858}
1859#endif
1860
1861int
1862getsockopt (int fd, int level, int optname,
1863 void *__restrict optval, socklen_t * __restrict optlen)
1864{
Florin Coras7baeb712019-01-04 17:05:43 -08001865 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001866 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05001867
Dave Wallace2a865272018-02-07 21:00:42 -05001868 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001869 return -1;
1870
Florin Coras7baeb712019-01-04 17:05:43 -08001871 vlsh = ldp_fd_to_vlsh (fd);
1872 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001873 {
1874 rv = -EOPNOTSUPP;
1875
1876 switch (level)
1877 {
1878 case SOL_TCP:
1879 switch (optname)
1880 {
1881 case TCP_NODELAY:
Florin Coras7baeb712019-01-04 17:05:43 -08001882 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_TCP_NODELAY,
1883 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001884 break;
1885 case TCP_MAXSEG:
Florin Coras7baeb712019-01-04 17:05:43 -08001886 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_TCP_USER_MSS,
1887 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001888 break;
1889 case TCP_KEEPIDLE:
Florin Coras7baeb712019-01-04 17:05:43 -08001890 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_TCP_KEEPIDLE,
1891 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001892 break;
1893 case TCP_KEEPINTVL:
Florin Coras7baeb712019-01-04 17:05:43 -08001894 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_TCP_KEEPINTVL,
1895 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001896 break;
1897 case TCP_INFO:
1898 if (optval && optlen && (*optlen == sizeof (struct tcp_info)))
1899 {
Florin Coras7baeb712019-01-04 17:05:43 -08001900 LDBG (1, "fd %d: vlsh %u SOL_TCP, TCP_INFO, optval %p, "
1901 "optlen %d: #LDP-NOP#", fd, vlsh, optval, *optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001902 memset (optval, 0, *optlen);
1903 rv = VPPCOM_OK;
1904 }
1905 else
1906 rv = -EFAULT;
1907 break;
Florin Coras0ed24e92019-01-21 09:03:10 -08001908 case TCP_CONGESTION:
Florin Coras0ed24e92019-01-21 09:03:10 -08001909 *optlen = strlen ("cubic");
Dave Barach02500902020-04-04 18:34:41 -04001910 strncpy (optval, "cubic", *optlen + 1);
Florin Coras0ed24e92019-01-21 09:03:10 -08001911 rv = 0;
1912 break;
Dave Wallace048b1d62018-01-03 22:24:41 -05001913 default:
Florin Coras7baeb712019-01-04 17:05:43 -08001914 LDBG (0, "ERROR: fd %d: getsockopt SOL_TCP: sid %u, "
1915 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05001916 break;
1917 }
1918 break;
1919 case SOL_IPV6:
1920 switch (optname)
1921 {
1922 case IPV6_V6ONLY:
Florin Coras7baeb712019-01-04 17:05:43 -08001923 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_V6ONLY, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001924 break;
1925 default:
Florin Coras7baeb712019-01-04 17:05:43 -08001926 LDBG (0, "ERROR: fd %d: getsockopt SOL_IPV6: vlsh %u "
1927 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05001928 break;
1929 }
1930 break;
1931 case SOL_SOCKET:
1932 switch (optname)
1933 {
1934 case SO_ACCEPTCONN:
Florin Coras7baeb712019-01-04 17:05:43 -08001935 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_LISTEN, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001936 break;
1937 case SO_KEEPALIVE:
Florin Coras7baeb712019-01-04 17:05:43 -08001938 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_KEEPALIVE, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001939 break;
1940 case SO_PROTOCOL:
Florin Coras7baeb712019-01-04 17:05:43 -08001941 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_PROTOCOL, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001942 *(int *) optval = *(int *) optval ? SOCK_DGRAM : SOCK_STREAM;
1943 break;
1944 case SO_SNDBUF:
Florin Coras7baeb712019-01-04 17:05:43 -08001945 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_TX_FIFO_LEN,
1946 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001947 break;
1948 case SO_RCVBUF:
Florin Coras7baeb712019-01-04 17:05:43 -08001949 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_RX_FIFO_LEN,
1950 optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001951 break;
1952 case SO_REUSEADDR:
Florin Coras7baeb712019-01-04 17:05:43 -08001953 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_REUSEADDR, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001954 break;
1955 case SO_BROADCAST:
Florin Coras7baeb712019-01-04 17:05:43 -08001956 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_BROADCAST, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001957 break;
1958 case SO_ERROR:
Florin Coras7baeb712019-01-04 17:05:43 -08001959 rv = vls_attr (vlsh, VPPCOM_ATTR_GET_ERROR, optval, optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05001960 break;
1961 default:
Florin Coras7baeb712019-01-04 17:05:43 -08001962 LDBG (0, "ERROR: fd %d: getsockopt SOL_SOCKET: vlsh %u "
1963 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05001964 break;
1965 }
1966 break;
1967 default:
1968 break;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001969 }
1970
Dave Wallace048b1d62018-01-03 22:24:41 -05001971 if (rv != VPPCOM_OK)
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07001972 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001973 errno = -rv;
1974 rv = -1;
1975 }
1976 }
1977 else
1978 {
Dave Wallace048b1d62018-01-03 22:24:41 -05001979 rv = libc_getsockopt (fd, level, optname, optval, optlen);
1980 }
1981
Dave Wallace048b1d62018-01-03 22:24:41 -05001982 return rv;
1983}
1984
1985int
1986setsockopt (int fd, int level, int optname,
1987 const void *optval, socklen_t optlen)
1988{
Florin Coras7baeb712019-01-04 17:05:43 -08001989 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05001990 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05001991
Dave Wallace2a865272018-02-07 21:00:42 -05001992 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05001993 return -1;
1994
Florin Coras7baeb712019-01-04 17:05:43 -08001995 vlsh = ldp_fd_to_vlsh (fd);
1996 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05001997 {
1998 rv = -EOPNOTSUPP;
1999
2000 switch (level)
2001 {
2002 case SOL_TCP:
2003 switch (optname)
2004 {
2005 case TCP_NODELAY:
Florin Coras7baeb712019-01-04 17:05:43 -08002006 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_TCP_NODELAY,
2007 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002008 break;
2009 case TCP_MAXSEG:
Florin Coras7baeb712019-01-04 17:05:43 -08002010 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_TCP_USER_MSS,
2011 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002012 break;
2013 case TCP_KEEPIDLE:
Florin Coras7baeb712019-01-04 17:05:43 -08002014 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_TCP_KEEPIDLE,
2015 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002016 break;
2017 case TCP_KEEPINTVL:
Florin Coras7baeb712019-01-04 17:05:43 -08002018 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_TCP_KEEPINTVL,
2019 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002020 break;
Florin Coras0ed24e92019-01-21 09:03:10 -08002021 case TCP_CONGESTION:
Florin Coras8509aa22019-04-04 12:55:30 -07002022 case TCP_CORK:
Florin Coras0ed24e92019-01-21 09:03:10 -08002023 /* Ignore */
2024 rv = 0;
2025 break;
Dave Wallace048b1d62018-01-03 22:24:41 -05002026 default:
Florin Coras7baeb712019-01-04 17:05:43 -08002027 LDBG (0, "ERROR: fd %d: setsockopt() SOL_TCP: vlsh %u"
2028 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05002029 break;
2030 }
2031 break;
2032 case SOL_IPV6:
2033 switch (optname)
2034 {
2035 case IPV6_V6ONLY:
Florin Coras7baeb712019-01-04 17:05:43 -08002036 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_V6ONLY,
2037 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002038 break;
2039 default:
Florin Coras7baeb712019-01-04 17:05:43 -08002040 LDBG (0, "ERROR: fd %d: setsockopt SOL_IPV6: vlsh %u"
2041 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05002042 break;
2043 }
2044 break;
2045 case SOL_SOCKET:
2046 switch (optname)
2047 {
2048 case SO_KEEPALIVE:
Florin Coras7baeb712019-01-04 17:05:43 -08002049 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_KEEPALIVE,
2050 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002051 break;
2052 case SO_REUSEADDR:
Florin Coras7baeb712019-01-04 17:05:43 -08002053 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_REUSEADDR,
2054 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002055 break;
2056 case SO_BROADCAST:
Florin Coras7baeb712019-01-04 17:05:43 -08002057 rv = vls_attr (vlsh, VPPCOM_ATTR_SET_BROADCAST,
2058 (void *) optval, &optlen);
Dave Wallace048b1d62018-01-03 22:24:41 -05002059 break;
2060 default:
Florin Coras7baeb712019-01-04 17:05:43 -08002061 LDBG (0, "ERROR: fd %d: setsockopt SOL_SOCKET: vlsh %u "
2062 "optname %d unsupported!", fd, vlsh, optname);
Dave Wallace048b1d62018-01-03 22:24:41 -05002063 break;
2064 }
2065 break;
2066 default:
2067 break;
2068 }
2069
2070 if (rv != VPPCOM_OK)
2071 {
2072 errno = -rv;
2073 rv = -1;
2074 }
2075 }
2076 else
2077 {
Dave Wallace048b1d62018-01-03 22:24:41 -05002078 rv = libc_setsockopt (fd, level, optname, optval, optlen);
2079 }
2080
Dave Wallace048b1d62018-01-03 22:24:41 -05002081 return rv;
2082}
2083
2084int
2085listen (int fd, int n)
2086{
Florin Coras7baeb712019-01-04 17:05:43 -08002087 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002088 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05002089
Dave Wallace2a865272018-02-07 21:00:42 -05002090 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002091 return -1;
2092
Florin Coras7baeb712019-01-04 17:05:43 -08002093 vlsh = ldp_fd_to_vlsh (fd);
2094 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05002095 {
Florin Coras7baeb712019-01-04 17:05:43 -08002096 LDBG (0, "fd %d: calling vls_listen: vlsh %u, n %d", fd, vlsh, n);
Dave Wallace048b1d62018-01-03 22:24:41 -05002097
Florin Coras7baeb712019-01-04 17:05:43 -08002098 rv = vls_listen (vlsh, n);
Dave Wallace048b1d62018-01-03 22:24:41 -05002099 if (rv != VPPCOM_OK)
2100 {
2101 errno = -rv;
2102 rv = -1;
2103 }
2104 }
2105 else
2106 {
Florin Coras7baeb712019-01-04 17:05:43 -08002107 LDBG (0, "fd %d: calling libc_listen(): n %d", fd, n);
Dave Wallace048b1d62018-01-03 22:24:41 -05002108 rv = libc_listen (fd, n);
2109 }
2110
Florin Coras7baeb712019-01-04 17:05:43 -08002111 LDBG (1, "fd %d: returning %d", fd, rv);
Dave Wallace048b1d62018-01-03 22:24:41 -05002112 return rv;
2113}
2114
2115static inline int
Dave Wallace2a865272018-02-07 21:00:42 -05002116ldp_accept4 (int listen_fd, __SOCKADDR_ARG addr,
2117 socklen_t * __restrict addr_len, int flags)
Dave Wallace048b1d62018-01-03 22:24:41 -05002118{
Florin Coras7baeb712019-01-04 17:05:43 -08002119 vls_handle_t listen_vlsh, accept_vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002120 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05002121
Dave Wallace2a865272018-02-07 21:00:42 -05002122 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002123 return -1;
2124
Florin Coras7baeb712019-01-04 17:05:43 -08002125 listen_vlsh = ldp_fd_to_vlsh (listen_fd);
2126 if (listen_vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05002127 {
2128 vppcom_endpt_t ep;
2129 u8 src_addr[sizeof (struct sockaddr_in6)];
Dave Wallace8aaba562018-01-18 17:21:19 -05002130 memset (&ep, 0, sizeof (ep));
Dave Wallace048b1d62018-01-03 22:24:41 -05002131 ep.ip = src_addr;
2132
Florin Coras7baeb712019-01-04 17:05:43 -08002133 LDBG (0, "listen fd %d: calling vppcom_session_accept: listen sid %u,"
Florin Coraseb801d02020-09-16 17:44:58 -07002134 " ep %p, flags 0x%x", listen_fd, listen_vlsh, &ep, flags);
Dave Wallace048b1d62018-01-03 22:24:41 -05002135
Florin Coras7baeb712019-01-04 17:05:43 -08002136 accept_vlsh = vls_accept (listen_vlsh, &ep, flags);
2137 if (accept_vlsh < 0)
Dave Wallace048b1d62018-01-03 22:24:41 -05002138 {
Florin Coras7baeb712019-01-04 17:05:43 -08002139 errno = -accept_vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002140 rv = -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002141 }
2142 else
2143 {
Dave Wallace2a865272018-02-07 21:00:42 -05002144 rv = ldp_copy_ep_to_sockaddr (addr, addr_len, &ep);
Dave Wallace048b1d62018-01-03 22:24:41 -05002145 if (rv != VPPCOM_OK)
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002146 {
Florin Coras7baeb712019-01-04 17:05:43 -08002147 (void) vls_close (accept_vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -05002148 errno = -rv;
2149 rv = -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002150 }
Dave Wallace048b1d62018-01-03 22:24:41 -05002151 else
2152 {
Florin Coras7baeb712019-01-04 17:05:43 -08002153 rv = ldp_vlsh_to_fd (accept_vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -05002154 }
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002155 }
2156 }
Dave Wallace048b1d62018-01-03 22:24:41 -05002157 else
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002158 {
Florin Coras7baeb712019-01-04 17:05:43 -08002159 LDBG (0, "listen fd %d: calling libc_accept4(): addr %p, addr_len %p,"
2160 " flags 0x%x", listen_fd, addr, addr_len, flags);
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002161
Dave Wallace048b1d62018-01-03 22:24:41 -05002162 rv = libc_accept4 (listen_fd, addr, addr_len, flags);
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002163 }
2164
Florin Coras7baeb712019-01-04 17:05:43 -08002165 LDBG (1, "listen fd %d: accept returning %d", listen_fd, rv);
Florin Coras05ecfcc2018-12-12 18:19:39 -08002166
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002167 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002168}
2169
Dave Wallace048b1d62018-01-03 22:24:41 -05002170int
2171accept4 (int fd, __SOCKADDR_ARG addr, socklen_t * __restrict addr_len,
2172 int flags)
2173{
Dave Wallace2a865272018-02-07 21:00:42 -05002174 return ldp_accept4 (fd, addr, addr_len, flags);
Dave Wallace048b1d62018-01-03 22:24:41 -05002175}
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002176
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002177int
Dave Wallace048b1d62018-01-03 22:24:41 -05002178accept (int fd, __SOCKADDR_ARG addr, socklen_t * __restrict addr_len)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002179{
Dave Wallace2a865272018-02-07 21:00:42 -05002180 return ldp_accept4 (fd, addr, addr_len, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05002181}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002182
Dave Wallace048b1d62018-01-03 22:24:41 -05002183int
2184shutdown (int fd, int how)
2185{
Florin Coras7baeb712019-01-04 17:05:43 -08002186 vls_handle_t vlsh;
2187 int rv = 0, flags;
2188 u32 flags_len = sizeof (flags);
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002189
Dave Wallace2a865272018-02-07 21:00:42 -05002190 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002191 return -1;
2192
Florin Coras7baeb712019-01-04 17:05:43 -08002193 vlsh = ldp_fd_to_vlsh (fd);
2194 if (vlsh != VLS_INVALID_HANDLE)
Dave Wallace048b1d62018-01-03 22:24:41 -05002195 {
Florin Coras7baeb712019-01-04 17:05:43 -08002196 LDBG (0, "called shutdown: fd %u vlsh %u how %d", fd, vlsh, how);
Florin Corasa7a1a222018-12-30 17:11:31 -08002197
Florin Coras7baeb712019-01-04 17:05:43 -08002198 if (vls_attr (vlsh, VPPCOM_ATTR_SET_SHUT, &how, &flags_len))
Florin Corasa7a1a222018-12-30 17:11:31 -08002199 {
Florin Coras7baeb712019-01-04 17:05:43 -08002200 close (fd);
Florin Corasa7a1a222018-12-30 17:11:31 -08002201 return -1;
2202 }
2203
Florin Coras7baeb712019-01-04 17:05:43 -08002204 if (vls_attr (vlsh, VPPCOM_ATTR_GET_SHUT, &flags, &flags_len))
2205 {
2206 close (fd);
2207 return -1;
2208 }
Florin Corasa7a1a222018-12-30 17:11:31 -08002209
Florin Coras7baeb712019-01-04 17:05:43 -08002210 if (flags == SHUT_RDWR)
Florin Corasa7a1a222018-12-30 17:11:31 -08002211 rv = close (fd);
Dave Wallace048b1d62018-01-03 22:24:41 -05002212 }
2213 else
2214 {
Florin Coras7baeb712019-01-04 17:05:43 -08002215 LDBG (0, "fd %d: calling libc_shutdown: how %d", fd, how);
Dave Wallace048b1d62018-01-03 22:24:41 -05002216 rv = libc_shutdown (fd, how);
2217 }
2218
Dave Wallace048b1d62018-01-03 22:24:41 -05002219 return rv;
2220}
2221
2222int
2223epoll_create1 (int flags)
2224{
Florin Corasdfe4cf42018-11-28 22:13:45 -08002225 ldp_worker_ctx_t *ldpw = ldp_worker_get_current ();
Florin Coras7baeb712019-01-04 17:05:43 -08002226 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002227 int rv;
2228
Dave Wallace2a865272018-02-07 21:00:42 -05002229 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002230 return -1;
2231
hanlina3a48962020-07-13 11:09:15 +08002232 if (ldp->vcl_needs_real_epoll || vls_use_real_epoll ())
Florin Coras99368312018-08-02 10:45:44 -07002233 {
Florin Coras2d9b4272019-03-11 10:14:37 -07002234 /* Make sure workers have been allocated */
2235 if (!ldp->workers)
2236 {
2237 ldp_alloc_workers ();
2238 ldpw = ldp_worker_get_current ();
2239 }
Florin Coras99368312018-08-02 10:45:44 -07002240 rv = libc_epoll_create1 (flags);
2241 ldp->vcl_needs_real_epoll = 0;
Florin Corasdfe4cf42018-11-28 22:13:45 -08002242 ldpw->vcl_mq_epfd = rv;
Florin Coras05ecfcc2018-12-12 18:19:39 -08002243 LDBG (0, "created vcl epfd %u", rv);
Florin Coras99368312018-08-02 10:45:44 -07002244 return rv;
2245 }
Dave Wallace048b1d62018-01-03 22:24:41 -05002246
Florin Coras7baeb712019-01-04 17:05:43 -08002247 vlsh = vls_epoll_create ();
2248 if (PREDICT_FALSE (vlsh == VLS_INVALID_HANDLE))
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002249 {
Florin Coras7baeb712019-01-04 17:05:43 -08002250 errno = -vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002251 rv = -1;
shrinivasan ganapathy1d359632017-10-15 15:46:09 -07002252 }
Dave Wallace048b1d62018-01-03 22:24:41 -05002253 else
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002254 {
Florin Coras7baeb712019-01-04 17:05:43 -08002255 rv = ldp_vlsh_to_fd (vlsh);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002256 }
Florin Coras7baeb712019-01-04 17:05:43 -08002257 LDBG (0, "epoll_create epfd %u vlsh %u", rv, vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -05002258 return rv;
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002259}
2260
2261int
Dave Wallace048b1d62018-01-03 22:24:41 -05002262epoll_create (int size)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002263{
Dave Wallace048b1d62018-01-03 22:24:41 -05002264 return epoll_create1 (0);
2265}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002266
Dave Wallace048b1d62018-01-03 22:24:41 -05002267int
2268epoll_ctl (int epfd, int op, int fd, struct epoll_event *event)
2269{
Florin Coras7baeb712019-01-04 17:05:43 -08002270 vls_handle_t vep_vlsh, vlsh;
Florin Coras99368312018-08-02 10:45:44 -07002271 int rv;
Dave Wallace048b1d62018-01-03 22:24:41 -05002272
Dave Wallace2a865272018-02-07 21:00:42 -05002273 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002274 return -1;
2275
Florin Coras7baeb712019-01-04 17:05:43 -08002276 vep_vlsh = ldp_fd_to_vlsh (epfd);
2277 if (PREDICT_FALSE (vep_vlsh == VLS_INVALID_HANDLE))
Dave Wallace048b1d62018-01-03 22:24:41 -05002278 {
Dave Wallace3ee1fe12018-02-23 01:09:11 -05002279 /* The LDP epoll_create1 always creates VCL epfd's.
2280 * The app should never have a kernel base epoll fd unless it
2281 * was acquired outside of the LD_PRELOAD process context.
2282 * In any case, if we get one, punt it to libc_epoll_ctl.
2283 */
Florin Coras7baeb712019-01-04 17:05:43 -08002284 LDBG (1, "epfd %d: calling libc_epoll_ctl: op %d, fd %d"
2285 " event %p", epfd, op, fd, event);
Dave Wallace048b1d62018-01-03 22:24:41 -05002286
2287 rv = libc_epoll_ctl (epfd, op, fd, event);
Florin Coras99368312018-08-02 10:45:44 -07002288 goto done;
2289 }
2290
Florin Coras7baeb712019-01-04 17:05:43 -08002291 vlsh = ldp_fd_to_vlsh (fd);
Florin Coras99368312018-08-02 10:45:44 -07002292
Florin Coras7baeb712019-01-04 17:05:43 -08002293 LDBG (0, "epfd %d ep_vlsh %d, fd %u vlsh %d, op %u", epfd, vep_vlsh, fd,
2294 vlsh, op);
Florin Coras99368312018-08-02 10:45:44 -07002295
Florin Coras7baeb712019-01-04 17:05:43 -08002296 if (vlsh != VLS_INVALID_HANDLE)
Florin Coras99368312018-08-02 10:45:44 -07002297 {
Florin Coras7baeb712019-01-04 17:05:43 -08002298 LDBG (1, "epfd %d: calling vls_epoll_ctl: ep_vlsh %d op %d, vlsh %u,"
Florin Coraseb801d02020-09-16 17:44:58 -07002299 " event %p", epfd, vep_vlsh, op, vlsh, event);
Florin Coras99368312018-08-02 10:45:44 -07002300
Florin Coras7baeb712019-01-04 17:05:43 -08002301 rv = vls_epoll_ctl (vep_vlsh, op, vlsh, event);
Florin Coras99368312018-08-02 10:45:44 -07002302 if (rv != VPPCOM_OK)
2303 {
2304 errno = -rv;
2305 rv = -1;
2306 }
2307 }
2308 else
2309 {
2310 int libc_epfd;
2311 u32 size = sizeof (epfd);
2312
Florin Coras7baeb712019-01-04 17:05:43 -08002313 libc_epfd = vls_attr (vep_vlsh, VPPCOM_ATTR_GET_LIBC_EPFD, 0, 0);
Florin Coras99368312018-08-02 10:45:44 -07002314 if (!libc_epfd)
2315 {
Florin Coras7baeb712019-01-04 17:05:43 -08002316 LDBG (1, "epfd %d, vep_vlsh %d calling libc_epoll_create1: "
2317 "EPOLL_CLOEXEC", epfd, vep_vlsh);
Florin Coras99368312018-08-02 10:45:44 -07002318
2319 libc_epfd = libc_epoll_create1 (EPOLL_CLOEXEC);
2320 if (libc_epfd < 0)
2321 {
2322 rv = libc_epfd;
2323 goto done;
2324 }
2325
Florin Coras7baeb712019-01-04 17:05:43 -08002326 rv = vls_attr (vep_vlsh, VPPCOM_ATTR_SET_LIBC_EPFD, &libc_epfd,
2327 &size);
Florin Coras99368312018-08-02 10:45:44 -07002328 if (rv < 0)
2329 {
2330 errno = -rv;
2331 rv = -1;
2332 goto done;
2333 }
2334 }
2335 else if (PREDICT_FALSE (libc_epfd < 0))
2336 {
2337 errno = -epfd;
2338 rv = -1;
2339 goto done;
2340 }
2341
Florin Coras7baeb712019-01-04 17:05:43 -08002342 LDBG (1, "epfd %d: calling libc_epoll_ctl: libc_epfd %d, op %d, fd %d,"
2343 " event %p", epfd, libc_epfd, op, fd, event);
Florin Coras99368312018-08-02 10:45:44 -07002344
2345 rv = libc_epoll_ctl (libc_epfd, op, fd, event);
Dave Wallace048b1d62018-01-03 22:24:41 -05002346 }
2347
2348done:
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002349 return rv;
2350}
Dave Wallace048b1d62018-01-03 22:24:41 -05002351
2352static inline int
Florin Coras99368312018-08-02 10:45:44 -07002353ldp_epoll_pwait (int epfd, struct epoll_event *events, int maxevents,
2354 int timeout, const sigset_t * sigmask)
Dave Wallace048b1d62018-01-03 22:24:41 -05002355{
Florin Corasdfe4cf42018-11-28 22:13:45 -08002356 ldp_worker_ctx_t *ldpw = ldp_worker_get_current ();
Florin Coras72f77822019-01-22 19:05:52 -08002357 double time_to_wait = (double) 0, max_time;
Florin Coras99368312018-08-02 10:45:44 -07002358 int libc_epfd, rv = 0;
Florin Coras7baeb712019-01-04 17:05:43 -08002359 vls_handle_t ep_vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002360
Dave Wallace2a865272018-02-07 21:00:42 -05002361 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002362 return -1;
2363
2364 if (PREDICT_FALSE (!events || (timeout < -1)))
2365 {
2366 errno = EFAULT;
2367 return -1;
2368 }
2369
Florin Corasdfe4cf42018-11-28 22:13:45 -08002370 if (epfd == ldpw->vcl_mq_epfd)
Florin Coras99368312018-08-02 10:45:44 -07002371 return libc_epoll_pwait (epfd, events, maxevents, timeout, sigmask);
2372
Florin Coras7baeb712019-01-04 17:05:43 -08002373 ep_vlsh = ldp_fd_to_vlsh (epfd);
2374 if (PREDICT_FALSE (ep_vlsh == VLS_INVALID_HANDLE))
Dave Wallace048b1d62018-01-03 22:24:41 -05002375 {
Florin Coras7baeb712019-01-04 17:05:43 -08002376 LDBG (0, "epfd %d: bad ep_vlsh %d!", epfd, ep_vlsh);
Dave Wallace048b1d62018-01-03 22:24:41 -05002377 errno = EBADFD;
2378 return -1;
2379 }
2380
Florin Coras4dee8cd2019-01-29 21:28:16 -08002381 if (PREDICT_FALSE (ldpw->clib_time.init_cpu_time == 0))
2382 clib_time_init (&ldpw->clib_time);
Florin Corasb0f662f2018-12-27 14:51:46 -08002383 time_to_wait = ((timeout >= 0) ? (double) timeout / 1000 : 0);
Florin Coras72f77822019-01-22 19:05:52 -08002384 max_time = clib_time_now (&ldpw->clib_time) + time_to_wait;
Dave Wallace048b1d62018-01-03 22:24:41 -05002385
Florin Coras7baeb712019-01-04 17:05:43 -08002386 libc_epfd = vls_attr (ep_vlsh, VPPCOM_ATTR_GET_LIBC_EPFD, 0, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05002387 if (PREDICT_FALSE (libc_epfd < 0))
2388 {
2389 errno = -libc_epfd;
2390 rv = -1;
2391 goto done;
2392 }
2393
Florin Coras7baeb712019-01-04 17:05:43 -08002394 LDBG (2, "epfd %d: vep_idx %d, libc_epfd %d, events %p, maxevents %d, "
2395 "timeout %d, sigmask %p: time_to_wait %.02f", epfd, ep_vlsh,
Florin Coras72f77822019-01-22 19:05:52 -08002396 libc_epfd, events, maxevents, timeout, sigmask, time_to_wait);
Dave Wallace048b1d62018-01-03 22:24:41 -05002397 do
2398 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002399 if (!ldpw->epoll_wait_vcl)
Dave Wallace048b1d62018-01-03 22:24:41 -05002400 {
Florin Coras7baeb712019-01-04 17:05:43 -08002401 rv = vls_epoll_wait (ep_vlsh, events, maxevents, 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05002402 if (rv > 0)
2403 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002404 ldpw->epoll_wait_vcl = 1;
Dave Wallace048b1d62018-01-03 22:24:41 -05002405 goto done;
2406 }
2407 else if (rv < 0)
2408 {
2409 errno = -rv;
2410 rv = -1;
2411 goto done;
2412 }
2413 }
2414 else
Florin Corasdfe4cf42018-11-28 22:13:45 -08002415 ldpw->epoll_wait_vcl = 0;
Dave Wallace048b1d62018-01-03 22:24:41 -05002416
2417 if (libc_epfd > 0)
2418 {
Florin Corasb0f662f2018-12-27 14:51:46 -08002419 rv = libc_epoll_pwait (libc_epfd, events, maxevents, 0, sigmask);
Dave Wallace048b1d62018-01-03 22:24:41 -05002420 if (rv != 0)
2421 goto done;
2422 }
Dave Wallace048b1d62018-01-03 22:24:41 -05002423 }
Florin Coras72f77822019-01-22 19:05:52 -08002424 while ((timeout == -1) || (clib_time_now (&ldpw->clib_time) < max_time));
Dave Wallace048b1d62018-01-03 22:24:41 -05002425
2426done:
Dave Wallace048b1d62018-01-03 22:24:41 -05002427 return rv;
2428}
2429
hanlin4266d4d2020-05-19 17:34:17 +08002430static inline int
2431ldp_epoll_pwait_eventfd (int epfd, struct epoll_event *events,
2432 int maxevents, int timeout, const sigset_t * sigmask)
2433{
hanlina3a48962020-07-13 11:09:15 +08002434 ldp_worker_ctx_t *ldpw;
hanlin4266d4d2020-05-19 17:34:17 +08002435 int libc_epfd, rv = 0, num_ev;
2436 vls_handle_t ep_vlsh;
2437
2438 if ((errno = -ldp_init ()))
2439 return -1;
2440
2441 if (PREDICT_FALSE (!events || (timeout < -1)))
2442 {
2443 errno = EFAULT;
2444 return -1;
2445 }
2446
Florin Corasff40d8f2020-08-11 22:05:28 -07002447 /* Make sure the vcl worker is valid. Could be that epoll fd was created on
2448 * one thread but it is now used on another */
2449 if (PREDICT_FALSE (vppcom_worker_index () == ~0))
2450 vls_register_vcl_worker ();
hanlina3a48962020-07-13 11:09:15 +08002451
2452 ldpw = ldp_worker_get_current ();
hanlin4266d4d2020-05-19 17:34:17 +08002453 if (epfd == ldpw->vcl_mq_epfd)
2454 return libc_epoll_pwait (epfd, events, maxevents, timeout, sigmask);
2455
2456 ep_vlsh = ldp_fd_to_vlsh (epfd);
2457 if (PREDICT_FALSE (ep_vlsh == VLS_INVALID_HANDLE))
2458 {
2459 LDBG (0, "epfd %d: bad ep_vlsh %d!", epfd, ep_vlsh);
2460 errno = EBADFD;
2461 return -1;
2462 }
2463
2464 libc_epfd = vls_attr (ep_vlsh, VPPCOM_ATTR_GET_LIBC_EPFD, 0, 0);
2465 if (PREDICT_FALSE (!libc_epfd))
2466 {
2467 u32 size = sizeof (epfd);
2468
2469 LDBG (1, "epfd %d, vep_vlsh %d calling libc_epoll_create1: "
2470 "EPOLL_CLOEXEC", epfd, ep_vlsh);
2471 libc_epfd = libc_epoll_create1 (EPOLL_CLOEXEC);
2472 if (libc_epfd < 0)
2473 {
2474 rv = libc_epfd;
2475 goto done;
2476 }
2477
2478 rv = vls_attr (ep_vlsh, VPPCOM_ATTR_SET_LIBC_EPFD, &libc_epfd, &size);
2479 if (rv < 0)
2480 {
2481 errno = -rv;
2482 rv = -1;
2483 goto done;
2484 }
2485 }
2486 if (PREDICT_FALSE (libc_epfd <= 0))
2487 {
2488 errno = -libc_epfd;
2489 rv = -1;
2490 goto done;
2491 }
2492
2493 if (PREDICT_FALSE (!ldpw->mq_epfd_added))
2494 {
2495 struct epoll_event e = { 0 };
2496 e.events = EPOLLIN;
2497 e.data.fd = ldpw->vcl_mq_epfd;
2498 if (libc_epoll_ctl (libc_epfd, EPOLL_CTL_ADD, ldpw->vcl_mq_epfd, &e) <
2499 0)
2500 {
2501 LDBG (0, "epfd %d, add libc mq epoll fd %d to libc epoll fd %d",
2502 epfd, ldpw->vcl_mq_epfd, libc_epfd);
2503 rv = -1;
2504 goto done;
2505 }
2506 ldpw->mq_epfd_added = 1;
2507 }
2508
2509 rv = vls_epoll_wait (ep_vlsh, events, maxevents, 0);
2510 if (rv > 0)
2511 goto done;
hanlina3a48962020-07-13 11:09:15 +08002512 else if (PREDICT_FALSE (rv < 0))
hanlin4266d4d2020-05-19 17:34:17 +08002513 {
2514 errno = -rv;
2515 rv = -1;
2516 goto done;
2517 }
2518
2519 rv = libc_epoll_pwait (libc_epfd, events, maxevents, timeout, sigmask);
2520 if (rv <= 0)
2521 goto done;
2522 for (int i = 0; i < rv; i++)
2523 {
2524 if (events[i].data.fd == ldpw->vcl_mq_epfd)
2525 {
2526 /* We should remove mq epoll fd from events. */
2527 rv--;
2528 if (i != rv)
2529 {
2530 events[i].events = events[rv].events;
2531 events[i].data.u64 = events[rv].data.u64;
2532 }
2533 num_ev = vls_epoll_wait (ep_vlsh, &events[rv], maxevents - rv, 0);
2534 if (PREDICT_TRUE (num_ev > 0))
2535 rv += num_ev;
2536 break;
2537 }
2538 }
2539
2540done:
2541 return rv;
2542}
2543
Dave Wallace048b1d62018-01-03 22:24:41 -05002544int
2545epoll_pwait (int epfd, struct epoll_event *events,
2546 int maxevents, int timeout, const sigset_t * sigmask)
2547{
hanlin4266d4d2020-05-19 17:34:17 +08002548 if (vls_use_eventfd ())
2549 return ldp_epoll_pwait_eventfd (epfd, events, maxevents, timeout,
2550 sigmask);
2551 else
2552 return ldp_epoll_pwait (epfd, events, maxevents, timeout, sigmask);
Dave Wallace048b1d62018-01-03 22:24:41 -05002553}
2554
2555int
2556epoll_wait (int epfd, struct epoll_event *events, int maxevents, int timeout)
2557{
hanlin4266d4d2020-05-19 17:34:17 +08002558 if (vls_use_eventfd ())
2559 return ldp_epoll_pwait_eventfd (epfd, events, maxevents, timeout, NULL);
2560 else
2561 return ldp_epoll_pwait (epfd, events, maxevents, timeout, NULL);
Dave Wallace048b1d62018-01-03 22:24:41 -05002562}
2563
2564int
2565poll (struct pollfd *fds, nfds_t nfds, int timeout)
2566{
Florin Corasdfe4cf42018-11-28 22:13:45 -08002567 ldp_worker_ctx_t *ldpw = ldp_worker_get_current ();
Florin Coras6917b942018-11-13 22:44:54 -08002568 int rv, i, n_revents = 0;
Florin Coras7baeb712019-01-04 17:05:43 -08002569 vls_handle_t vlsh;
Dave Wallace048b1d62018-01-03 22:24:41 -05002570 vcl_poll_t *vp;
Florin Coras4dee8cd2019-01-29 21:28:16 -08002571 double max_time;
Dave Wallace048b1d62018-01-03 22:24:41 -05002572
Florin Coraseb801d02020-09-16 17:44:58 -07002573 LDBG (3, "fds %p, nfds %ld, timeout %d", fds, nfds, timeout);
Dave Wallace048b1d62018-01-03 22:24:41 -05002574
Florin Coras4dee8cd2019-01-29 21:28:16 -08002575 if (PREDICT_FALSE (ldpw->clib_time.init_cpu_time == 0))
2576 clib_time_init (&ldpw->clib_time);
2577
2578 max_time = (timeout >= 0) ? (f64) timeout / 1000 : 0;
2579 max_time += clib_time_now (&ldpw->clib_time);
Dave Wallace048b1d62018-01-03 22:24:41 -05002580
Dave Wallace048b1d62018-01-03 22:24:41 -05002581 for (i = 0; i < nfds; i++)
2582 {
Florin Coras6917b942018-11-13 22:44:54 -08002583 if (fds[i].fd < 0)
2584 continue;
Dave Wallace048b1d62018-01-03 22:24:41 -05002585
Florin Coras7baeb712019-01-04 17:05:43 -08002586 vlsh = ldp_fd_to_vlsh (fds[i].fd);
2587 if (vlsh != VLS_INVALID_HANDLE)
Florin Coras6917b942018-11-13 22:44:54 -08002588 {
2589 fds[i].fd = -fds[i].fd;
Florin Corasdfe4cf42018-11-28 22:13:45 -08002590 vec_add2 (ldpw->vcl_poll, vp, 1);
Florin Coras6917b942018-11-13 22:44:54 -08002591 vp->fds_ndx = i;
Florin Coras7baeb712019-01-04 17:05:43 -08002592 vp->sh = vlsh_to_sh (vlsh);
Florin Coras6917b942018-11-13 22:44:54 -08002593 vp->events = fds[i].events;
Dave Wallace048b1d62018-01-03 22:24:41 -05002594#ifdef __USE_XOPEN2K
Florin Coras6917b942018-11-13 22:44:54 -08002595 if (fds[i].events & POLLRDNORM)
2596 vp->events |= POLLIN;
2597 if (fds[i].events & POLLWRNORM)
2598 vp->events |= POLLOUT;
Dave Wallace048b1d62018-01-03 22:24:41 -05002599#endif
Florin Coras6917b942018-11-13 22:44:54 -08002600 vp->revents = fds[i].revents;
2601 }
2602 else
2603 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002604 vec_add1 (ldpw->libc_poll, fds[i]);
2605 vec_add1 (ldpw->libc_poll_idxs, i);
Dave Wallace048b1d62018-01-03 22:24:41 -05002606 }
2607 }
2608
Dave Wallace048b1d62018-01-03 22:24:41 -05002609 do
2610 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002611 if (vec_len (ldpw->vcl_poll))
Dave Wallace048b1d62018-01-03 22:24:41 -05002612 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002613 rv = vppcom_poll (ldpw->vcl_poll, vec_len (ldpw->vcl_poll), 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05002614 if (rv < 0)
2615 {
2616 errno = -rv;
2617 rv = -1;
2618 goto done;
2619 }
2620 else
2621 n_revents += rv;
2622 }
2623
Florin Corasdfe4cf42018-11-28 22:13:45 -08002624 if (vec_len (ldpw->libc_poll))
Dave Wallace048b1d62018-01-03 22:24:41 -05002625 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002626 rv = libc_poll (ldpw->libc_poll, vec_len (ldpw->libc_poll), 0);
Dave Wallace048b1d62018-01-03 22:24:41 -05002627 if (rv < 0)
2628 goto done;
2629 else
2630 n_revents += rv;
2631 }
2632
2633 if (n_revents)
2634 {
2635 rv = n_revents;
2636 goto done;
2637 }
2638 }
Florin Coras4dee8cd2019-01-29 21:28:16 -08002639 while ((timeout < 0) || (clib_time_now (&ldpw->clib_time) < max_time));
Dave Wallace048b1d62018-01-03 22:24:41 -05002640 rv = 0;
2641
2642done:
Florin Corasdfe4cf42018-11-28 22:13:45 -08002643 vec_foreach (vp, ldpw->vcl_poll)
Dave Wallace048b1d62018-01-03 22:24:41 -05002644 {
2645 fds[vp->fds_ndx].fd = -fds[vp->fds_ndx].fd;
Florin Coras6917b942018-11-13 22:44:54 -08002646 fds[vp->fds_ndx].revents = vp->revents;
Dave Wallace048b1d62018-01-03 22:24:41 -05002647#ifdef __USE_XOPEN2K
2648 if ((fds[vp->fds_ndx].revents & POLLIN) &&
2649 (fds[vp->fds_ndx].events & POLLRDNORM))
2650 fds[vp->fds_ndx].revents |= POLLRDNORM;
2651 if ((fds[vp->fds_ndx].revents & POLLOUT) &&
2652 (fds[vp->fds_ndx].events & POLLWRNORM))
2653 fds[vp->fds_ndx].revents |= POLLWRNORM;
2654#endif
2655 }
Florin Corasdfe4cf42018-11-28 22:13:45 -08002656 vec_reset_length (ldpw->vcl_poll);
Dave Wallace048b1d62018-01-03 22:24:41 -05002657
Florin Corasdfe4cf42018-11-28 22:13:45 -08002658 for (i = 0; i < vec_len (ldpw->libc_poll); i++)
Florin Coras6917b942018-11-13 22:44:54 -08002659 {
Florin Corasdfe4cf42018-11-28 22:13:45 -08002660 fds[ldpw->libc_poll_idxs[i]].revents = ldpw->libc_poll[i].revents;
Florin Coras6917b942018-11-13 22:44:54 -08002661 }
Florin Corasdfe4cf42018-11-28 22:13:45 -08002662 vec_reset_length (ldpw->libc_poll_idxs);
2663 vec_reset_length (ldpw->libc_poll);
Florin Coras6917b942018-11-13 22:44:54 -08002664
Dave Wallace048b1d62018-01-03 22:24:41 -05002665 return rv;
2666}
2667
2668#ifdef USE_GNU
2669int
2670ppoll (struct pollfd *fds, nfds_t nfds,
2671 const struct timespec *timeout, const sigset_t * sigmask)
2672{
Dave Wallace2a865272018-02-07 21:00:42 -05002673 if ((errno = -ldp_init ()))
Dave Wallace048b1d62018-01-03 22:24:41 -05002674 return -1;
2675
2676 clib_warning ("LDP<%d>: LDP-TBD", getpid ());
2677 errno = ENOSYS;
2678
2679
2680 return -1;
2681}
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002682#endif
2683
Dave Wallace2a865272018-02-07 21:00:42 -05002684void CONSTRUCTOR_ATTRIBUTE ldp_constructor (void);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002685
Dave Wallace2a865272018-02-07 21:00:42 -05002686void DESTRUCTOR_ATTRIBUTE ldp_destructor (void);
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002687
Dave Wallace048b1d62018-01-03 22:24:41 -05002688/*
2689 * This function is called when the library is loaded
2690 */
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002691void
Dave Wallace2a865272018-02-07 21:00:42 -05002692ldp_constructor (void)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002693{
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002694 swrap_constructor ();
Dave Wallace2a865272018-02-07 21:00:42 -05002695 if (ldp_init () != 0)
Florin Corasd89411e2019-03-19 19:44:51 -07002696 {
2697 fprintf (stderr, "\nLDP<%d>: ERROR: ldp_constructor: failed!\n",
2698 getpid ());
2699 _exit (1);
2700 }
Dave Wallace69d01192018-02-22 16:22:09 -05002701 else if (LDP_DEBUG > 0)
Dave Wallace2a865272018-02-07 21:00:42 -05002702 clib_warning ("LDP<%d>: LDP constructor: done!\n", getpid ());
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002703}
2704
2705/*
2706 * This function is called when the library is unloaded
2707 */
2708void
Dave Wallace2a865272018-02-07 21:00:42 -05002709ldp_destructor (void)
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002710{
Florin Coras0ef8ef22019-01-18 08:37:13 -08002711 /*
2712 swrap_destructor ();
2713 if (ldp->init)
2714 ldp->init = 0;
2715 */
Dave Wallace048b1d62018-01-03 22:24:41 -05002716
2717 /* Don't use clib_warning() here because that calls writev()
Dave Wallace2a865272018-02-07 21:00:42 -05002718 * which will call ldp_init().
Dave Wallace048b1d62018-01-03 22:24:41 -05002719 */
Dave Wallace69d01192018-02-22 16:22:09 -05002720 if (LDP_DEBUG > 0)
Florin Coras0ef8ef22019-01-18 08:37:13 -08002721 fprintf (stderr, "%s:%d: LDP<%d>: LDP destructor: done!\n",
2722 __func__, __LINE__, getpid ());
Keith Burns (alagalah)b327c2b2017-10-09 08:52:59 -07002723}
2724
2725
2726/*
2727 * fd.io coding-style-patch-verification: ON
2728 *
2729 * Local Variables:
2730 * eval: (c-set-style "gnu")
2731 * End:
2732 */