source: mainline/uspace/srv/net/inetsrv/inetping.c@ 984a9ba

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 984a9ba was 984a9ba, checked in by Martin Decky <martin@…>, 7 years ago

do not expose the call capability handler from the async framework

Keep the call capability handler encapsulated within the async framework
and do not expose it explicitly via its API. Use the pointer to
ipc_call_t as the sole object identifying an IPC call in the code that
uses the async framework.

This plugs a major leak in the abstraction and also simplifies both the
async framework (slightly) and all IPC servers.

  • Property mode set to 100644
File size: 7.4 KB
Line 
1/*
2 * Copyright (c) 2013 Jiri Svoboda
3 * Copyright (c) 2013 Martin Decky
4 * All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 *
10 * - Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * - Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * - The name of the author may not be used to endorse or promote products
16 * derived from this software without specific prior written permission.
17 *
18 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
30/** @addtogroup inet
31 * @{
32 */
33/**
34 * @file
35 * @brief
36 */
37
38#include <async.h>
39#include <errno.h>
40#include <fibril_synch.h>
41#include <io/log.h>
42#include <ipc/inet.h>
43#include <loc.h>
44#include <stdlib.h>
45#include <stddef.h>
46#include <stdint.h>
47#include <types/inetping.h>
48#include "icmp.h"
49#include "icmpv6.h"
50#include "icmp_std.h"
51#include "inetsrv.h"
52#include "inetping.h"
53
54static FIBRIL_MUTEX_INITIALIZE(client_list_lock);
55static LIST_INITIALIZE(client_list);
56
57/** Last used session identifier. Protected by @c client_list_lock */
58static uint16_t inetping_ident = 0;
59
60static errno_t inetping_send(inetping_client_t *client, inetping_sdu_t *sdu)
61{
62 if (sdu->src.version != sdu->dest.version)
63 return EINVAL;
64
65 switch (sdu->src.version) {
66 case ip_v4:
67 return icmp_ping_send(client->ident, sdu);
68 case ip_v6:
69 return icmpv6_ping_send(client->ident, sdu);
70 default:
71 return EINVAL;
72 }
73}
74
75static errno_t inetping_get_srcaddr(inetping_client_t *client,
76 inet_addr_t *remote, inet_addr_t *local)
77{
78 return inet_get_srcaddr(remote, ICMP_TOS, local);
79}
80
81static inetping_client_t *inetping_client_find(uint16_t ident)
82{
83 fibril_mutex_lock(&client_list_lock);
84
85 list_foreach(client_list, client_list, inetping_client_t, client) {
86 if (client->ident == ident) {
87 fibril_mutex_unlock(&client_list_lock);
88 return client;
89 }
90 }
91
92 fibril_mutex_unlock(&client_list_lock);
93 return NULL;
94}
95
96errno_t inetping_recv(uint16_t ident, inetping_sdu_t *sdu)
97{
98 inetping_client_t *client = inetping_client_find(ident);
99 if (client == NULL) {
100 log_msg(LOG_DEFAULT, LVL_DEBUG, "Unknown ICMP ident. Dropping.");
101 return ENOENT;
102 }
103
104 async_exch_t *exch = async_exchange_begin(client->sess);
105
106 ipc_call_t answer;
107 aid_t req = async_send_1(exch, INETPING_EV_RECV, sdu->seq_no, &answer);
108
109 errno_t rc = async_data_write_start(exch, &sdu->src, sizeof(sdu->src));
110 if (rc != EOK) {
111 async_exchange_end(exch);
112 async_forget(req);
113 return rc;
114 }
115
116 rc = async_data_write_start(exch, &sdu->dest, sizeof(sdu->dest));
117 if (rc != EOK) {
118 async_exchange_end(exch);
119 async_forget(req);
120 return rc;
121 }
122
123 rc = async_data_write_start(exch, sdu->data, sdu->size);
124
125 async_exchange_end(exch);
126
127 if (rc != EOK) {
128 async_forget(req);
129 return rc;
130 }
131
132 errno_t retval;
133 async_wait_for(req, &retval);
134
135 return retval;
136}
137
138static void inetping_send_srv(inetping_client_t *client, ipc_call_t *icall)
139{
140 log_msg(LOG_DEFAULT, LVL_DEBUG, "inetping_send_srv()");
141
142 inetping_sdu_t sdu;
143 errno_t rc;
144
145 sdu.seq_no = IPC_GET_ARG1(*icall);
146
147 ipc_call_t call;
148 size_t size;
149 if (!async_data_write_receive(&call, &size)) {
150 async_answer_0(&call, EREFUSED);
151 async_answer_0(icall, EREFUSED);
152 return;
153 }
154
155 if (size != sizeof(sdu.src)) {
156 async_answer_0(&call, EINVAL);
157 async_answer_0(icall, EINVAL);
158 return;
159 }
160
161 rc = async_data_write_finalize(&call, &sdu.src, size);
162 if (rc != EOK) {
163 async_answer_0(&call, rc);
164 async_answer_0(icall, rc);
165 return;
166 }
167
168 if (!async_data_write_receive(&call, &size)) {
169 async_answer_0(&call, EREFUSED);
170 async_answer_0(icall, EREFUSED);
171 return;
172 }
173
174 if (size != sizeof(sdu.dest)) {
175 async_answer_0(&call, EINVAL);
176 async_answer_0(icall, EINVAL);
177 return;
178 }
179
180 rc = async_data_write_finalize(&call, &sdu.dest, size);
181 if (rc != EOK) {
182 async_answer_0(&call, rc);
183 async_answer_0(icall, rc);
184 return;
185 }
186
187 rc = async_data_write_accept((void **) &sdu.data, false, 0, 0, 0,
188 &sdu.size);
189 if (rc != EOK) {
190 async_answer_0(icall, rc);
191 return;
192 }
193
194 rc = inetping_send(client, &sdu);
195 free(sdu.data);
196
197 async_answer_0(icall, rc);
198}
199
200static void inetping_get_srcaddr_srv(inetping_client_t *client, ipc_call_t *icall)
201{
202 log_msg(LOG_DEFAULT, LVL_DEBUG, "inetping_get_srcaddr_srv()");
203
204 ipc_call_t call;
205 size_t size;
206
207 inet_addr_t local;
208 inet_addr_t remote;
209
210 if (!async_data_write_receive(&call, &size)) {
211 async_answer_0(&call, EREFUSED);
212 async_answer_0(icall, EREFUSED);
213 return;
214 }
215
216 if (size != sizeof(remote)) {
217 async_answer_0(&call, EINVAL);
218 async_answer_0(icall, EINVAL);
219 return;
220 }
221
222 errno_t rc = async_data_write_finalize(&call, &remote, size);
223 if (rc != EOK) {
224 async_answer_0(&call, rc);
225 async_answer_0(icall, rc);
226 return;
227 }
228
229 rc = inetping_get_srcaddr(client, &remote, &local);
230 if (rc != EOK) {
231 async_answer_0(icall, rc);
232 return;
233 }
234
235 if (!async_data_read_receive(&call, &size)) {
236 async_answer_0(&call, EREFUSED);
237 async_answer_0(icall, EREFUSED);
238 return;
239 }
240
241 if (size != sizeof(local)) {
242 async_answer_0(&call, EINVAL);
243 async_answer_0(icall, EINVAL);
244 return;
245 }
246
247 rc = async_data_read_finalize(&call, &local, size);
248 if (rc != EOK)
249 async_answer_0(&call, rc);
250
251 async_answer_0(icall, rc);
252}
253
254static errno_t inetping_client_init(inetping_client_t *client)
255{
256 async_sess_t *sess = async_callback_receive(EXCHANGE_SERIALIZE);
257 if (sess == NULL)
258 return ENOMEM;
259
260 client->sess = sess;
261 link_initialize(&client->client_list);
262
263 fibril_mutex_lock(&client_list_lock);
264 client->ident = ++inetping_ident;
265 list_append(&client->client_list, &client_list);
266 fibril_mutex_unlock(&client_list_lock);
267
268 return EOK;
269}
270
271static void inetping_client_fini(inetping_client_t *client)
272{
273 async_hangup(client->sess);
274 client->sess = NULL;
275
276 fibril_mutex_lock(&client_list_lock);
277 list_remove(&client->client_list);
278 fibril_mutex_unlock(&client_list_lock);
279}
280
281void inetping_conn(ipc_call_t *icall, void *arg)
282{
283 log_msg(LOG_DEFAULT, LVL_DEBUG, "inetping_conn()");
284
285 /* Accept the connection */
286 async_answer_0(icall, EOK);
287
288 inetping_client_t client;
289 errno_t rc = inetping_client_init(&client);
290 if (rc != EOK)
291 return;
292
293 while (true) {
294 ipc_call_t call;
295 async_get_call(&call);
296 sysarg_t method = IPC_GET_IMETHOD(call);
297
298 if (!method) {
299 /* The other side has hung up */
300 async_answer_0(&call, EOK);
301 break;
302 }
303
304 switch (method) {
305 case INETPING_SEND:
306 inetping_send_srv(&client, &call);
307 break;
308 case INETPING_GET_SRCADDR:
309 inetping_get_srcaddr_srv(&client, &call);
310 break;
311 default:
312 async_answer_0(&call, EINVAL);
313 }
314 }
315
316 inetping_client_fini(&client);
317}
318
319/** @}
320 */
Note: See TracBrowser for help on using the repository browser.