blob: a9aaea4e8ddb45ef78de27c307790e88e7b4265f [file] [log] [blame]
Harald Welte8857f3b2022-11-18 13:54:44 +01001/*! \file osmo_io_poll.c
2 * New osmocom async I/O API.
3 *
4 * (C) 2022 by Harald Welte <laforge@osmocom.org>
5 * (C) 2022-2023 by sysmocom - s.f.m.c. GmbH <info@sysmocom.de>
6 * Author: Daniel Willmann <dwillmann@sysmocom.de>
7 *
8 * All Rights Reserved.
9 *
10 * SPDX-License-Identifier: GPL-2.0+
11 *
12 * This program is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
16 *
17 * This program is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
21 */
22
23#include "../config.h"
24#if defined(__linux__)
25
26#include <errno.h>
27#include <stdio.h>
28#include <talloc.h>
29#include <unistd.h>
30#include <stdbool.h>
31#include <sys/socket.h>
32
33#include <osmocom/core/osmo_io.h>
34#include <osmocom/core/linuxlist.h>
35#include <osmocom/core/logging.h>
36#include <osmocom/core/msgb.h>
37#include <osmocom/core/select.h>
38#include <osmocom/core/socket.h>
39#include <osmocom/core/talloc.h>
40#include <osmocom/core/utils.h>
41
42#include "osmo_io_internal.h"
43
44static void iofd_poll_ofd_cb_recvmsg_sendmsg(struct osmo_fd *ofd, unsigned int what)
45{
46 struct osmo_io_fd *iofd = ofd->data;
47 struct msgb *msg;
48 int rc, flags = 0;
49
50 if (what & OSMO_FD_READ) {
51 struct iofd_msghdr hdr;
52 msg = iofd_msgb_pending_or_alloc(iofd);
53 if (!msg) {
Daniel Willmanna8a27c22023-06-26 19:26:16 +020054 LOGPIO(iofd, LOGL_ERROR, "Could not allocate msgb for reading\n");
Harald Welte8857f3b2022-11-18 13:54:44 +010055 OSMO_ASSERT(0);
56 }
57
58 hdr.msg = msg;
arehbein5099d992023-06-16 22:37:44 +020059 hdr.iov[0].iov_base = msg->tail;
Harald Welte8857f3b2022-11-18 13:54:44 +010060 hdr.iov[0].iov_len = msgb_tailroom(msg);
Daniel Willmann435856b2023-09-05 14:11:55 +020061 hdr.hdr = (struct msghdr) {
62 .msg_iov = &hdr.iov[0],
63 .msg_iovlen = 1,
64 .msg_name = &hdr.osa.u.sa,
65 .msg_namelen = sizeof(struct osmo_sockaddr),
66 };
Harald Welte8857f3b2022-11-18 13:54:44 +010067
68 rc = recvmsg(ofd->fd, &hdr.hdr, flags);
69 if (rc > 0)
70 msgb_put(msg, rc);
71
Daniel Willmann2b34e922023-08-23 18:02:13 +020072 iofd_handle_recv(iofd, msg, rc, &hdr);
Harald Welte8857f3b2022-11-18 13:54:44 +010073 }
74
Daniel Willmannf89162f2023-06-26 19:24:46 +020075 if (IOFD_FLAG_ISSET(iofd, IOFD_FLAG_CLOSED))
Harald Welte8857f3b2022-11-18 13:54:44 +010076 return;
77
78 if (what & OSMO_FD_WRITE) {
79 struct iofd_msghdr *msghdr = iofd_txqueue_dequeue(iofd);
80 if (msghdr) {
81 msg = msghdr->msg;
82
83 rc = sendmsg(ofd->fd, &msghdr->hdr, msghdr->flags);
84 if (rc > 0 && rc < msgb_length(msg)) {
85 msgb_pull(msg, rc);
86 iofd_txqueue_enqueue_front(iofd, msghdr);
87 return;
88 }
arehbein2e2a0a12023-10-19 18:00:04 +020089 if (rc == -EAGAIN) {
90 iofd_txqueue_enqueue_front(iofd, msghdr);
91 return;
92 }
Harald Welte8857f3b2022-11-18 13:54:44 +010093
94 switch (iofd->mode) {
95 case OSMO_IO_FD_MODE_READ_WRITE:
96 iofd->io_ops.write_cb(iofd, rc, msg);
97 break;
98 case OSMO_IO_FD_MODE_RECVFROM_SENDTO:
99 iofd->io_ops.sendto_cb(iofd, rc, msg, &msghdr->osa);
100 break;
101 case OSMO_IO_FD_MODE_SCTP_RECVMSG_SENDMSG:
102 OSMO_ASSERT(false);
103 break;
104 }
105
106 talloc_free(msghdr);
107 msgb_free(msg);
Daniel Willmanncbbd17e2023-05-17 17:08:10 +0200108 } else {
109 if (iofd->mode == OSMO_IO_FD_MODE_READ_WRITE)
110 /* Socket is writable, but we have no data to send. A non-blocking/async
111 connect() is signalled this way. */
112 iofd->io_ops.write_cb(iofd, 0, NULL);
113 if (osmo_iofd_txqueue_len(iofd) == 0)
114 iofd_poll_ops.write_disable(iofd);
Harald Welte8857f3b2022-11-18 13:54:44 +0100115 }
Daniel Willmanncbbd17e2023-05-17 17:08:10 +0200116
Harald Welte8857f3b2022-11-18 13:54:44 +0100117 }
118}
119
120static int iofd_poll_ofd_cb_dispatch(struct osmo_fd *ofd, unsigned int what)
121{
122 struct osmo_io_fd *iofd = ofd->data;
123
Daniel Willmannf89162f2023-06-26 19:24:46 +0200124 IOFD_FLAG_SET(iofd, IOFD_FLAG_IN_CALLBACK);
Harald Welte8857f3b2022-11-18 13:54:44 +0100125 iofd_poll_ofd_cb_recvmsg_sendmsg(ofd, what);
Daniel Willmannf89162f2023-06-26 19:24:46 +0200126 IOFD_FLAG_UNSET(iofd, IOFD_FLAG_IN_CALLBACK);
Harald Welte8857f3b2022-11-18 13:54:44 +0100127
Daniel Willmannf89162f2023-06-26 19:24:46 +0200128 if (IOFD_FLAG_ISSET(iofd, IOFD_FLAG_TO_FREE)) {
Harald Welte8857f3b2022-11-18 13:54:44 +0100129 talloc_free(iofd);
130 return 0;
131 }
132
133 return 0;
134}
135
136int iofd_poll_register(struct osmo_io_fd *iofd)
137{
138 struct osmo_fd *ofd = &iofd->u.poll.ofd;
139 osmo_fd_setup(ofd, iofd->fd, 0, &iofd_poll_ofd_cb_dispatch, iofd, 0);
140 return osmo_fd_register(ofd);
141}
142
143int iofd_poll_unregister(struct osmo_io_fd *iofd)
144{
145 struct osmo_fd *ofd = &iofd->u.poll.ofd;
146 osmo_fd_unregister(ofd);
147
148 return 0;
149}
150
151int iofd_poll_close(struct osmo_io_fd *iofd)
152{
153 osmo_fd_close(&iofd->u.poll.ofd);
154
155 return 0;
156}
157
158void iofd_poll_read_enable(struct osmo_io_fd *iofd)
159{
160 osmo_fd_read_enable(&iofd->u.poll.ofd);
161}
162
163void iofd_poll_read_disable(struct osmo_io_fd *iofd)
164{
165 osmo_fd_read_disable(&iofd->u.poll.ofd);
166}
167
168void iofd_poll_write_enable(struct osmo_io_fd *iofd)
169{
170 osmo_fd_write_enable(&iofd->u.poll.ofd);
171}
172
173void iofd_poll_write_disable(struct osmo_io_fd *iofd)
174{
175 osmo_fd_write_disable(&iofd->u.poll.ofd);
176}
177
178const struct iofd_backend_ops iofd_poll_ops = {
179 .register_fd = iofd_poll_register,
180 .unregister_fd = iofd_poll_unregister,
181 .close = iofd_poll_close,
182 .write_enable = iofd_poll_write_enable,
183 .write_disable = iofd_poll_write_disable,
184 .read_enable = iofd_poll_read_enable,
185 .read_disable = iofd_poll_read_disable,
186};
187
188#endif /* defined(__linux__) */