source: mainline/uspace/lib/drv/generic/remote_ahci.c@ 8bf9058

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 8bf9058 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: 8.6 KB
Line 
1/*
2 * Copyright (c) 2012 Petr Jerman
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup libdrv
30 * @{
31 */
32/** @file
33 */
34
35#include <as.h>
36#include <async.h>
37#include <devman.h>
38#include <errno.h>
39#include <stdio.h>
40#include <macros.h>
41#include <str.h>
42#include "ahci_iface.h"
43#include "ddf/driver.h"
44
45typedef enum {
46 IPC_M_AHCI_GET_SATA_DEVICE_NAME,
47 IPC_M_AHCI_GET_NUM_BLOCKS,
48 IPC_M_AHCI_GET_BLOCK_SIZE,
49 IPC_M_AHCI_READ_BLOCKS,
50 IPC_M_AHCI_WRITE_BLOCKS
51} ahci_iface_funcs_t;
52
53#define MAX_NAME_LENGTH 1024
54
55#define LO(ptr) \
56 ((uint32_t) (((uint64_t) ((uintptr_t) (ptr))) & 0xffffffff))
57
58#define HI(ptr) \
59 ((uint32_t) (((uint64_t) ((uintptr_t) (ptr))) >> 32))
60
61async_sess_t *ahci_get_sess(devman_handle_t funh, char **name)
62{
63 // FIXME: Use a better way than substring match
64
65 *name = NULL;
66
67 char devn[MAX_NAME_LENGTH];
68 errno_t rc = devman_fun_get_name(funh, devn, MAX_NAME_LENGTH);
69 if (rc != EOK)
70 return NULL;
71
72 size_t devn_size = str_size(devn);
73
74 if ((devn_size > 5) && (str_lcmp(devn, "ahci_", 5) == 0)) {
75 async_sess_t *sess = devman_device_connect(funh, IPC_FLAG_BLOCKING);
76
77 if (sess) {
78 *name = str_dup(devn);
79 return sess;
80 }
81 }
82
83 return NULL;
84}
85
86errno_t ahci_get_sata_device_name(async_sess_t *sess, size_t sata_dev_name_length,
87 char *sata_dev_name)
88{
89 async_exch_t *exch = async_exchange_begin(sess);
90 if (!exch)
91 return EINVAL;
92
93 aid_t req = async_send_2(exch, DEV_IFACE_ID(AHCI_DEV_IFACE),
94 IPC_M_AHCI_GET_SATA_DEVICE_NAME, sata_dev_name_length, NULL);
95
96 async_data_read_start(exch, sata_dev_name, sata_dev_name_length);
97
98 errno_t rc;
99 async_wait_for(req, &rc);
100
101 return rc;
102}
103
104errno_t ahci_get_num_blocks(async_sess_t *sess, uint64_t *blocks)
105{
106 async_exch_t *exch = async_exchange_begin(sess);
107 if (!exch)
108 return EINVAL;
109
110 sysarg_t blocks_hi;
111 sysarg_t blocks_lo;
112 errno_t rc = async_req_1_2(exch, DEV_IFACE_ID(AHCI_DEV_IFACE),
113 IPC_M_AHCI_GET_NUM_BLOCKS, &blocks_hi, &blocks_lo);
114
115 async_exchange_end(exch);
116
117 if (rc == EOK) {
118 *blocks = (((uint64_t) blocks_hi) << 32) |
119 (((uint64_t) blocks_lo) & 0xffffffff);
120 }
121
122 return rc;
123}
124
125errno_t ahci_get_block_size(async_sess_t *sess, size_t *blocks_size)
126{
127 async_exch_t *exch = async_exchange_begin(sess);
128 if (!exch)
129 return EINVAL;
130
131 sysarg_t bs;
132 errno_t rc = async_req_1_1(exch, DEV_IFACE_ID(AHCI_DEV_IFACE),
133 IPC_M_AHCI_GET_BLOCK_SIZE, &bs);
134
135 async_exchange_end(exch);
136
137 if (rc == EOK)
138 *blocks_size = (size_t) bs;
139
140 return rc;
141}
142
143errno_t ahci_read_blocks(async_sess_t *sess, uint64_t blocknum, size_t count,
144 void *buf)
145{
146 async_exch_t *exch = async_exchange_begin(sess);
147 if (!exch)
148 return EINVAL;
149
150 aid_t req;
151 req = async_send_4(exch, DEV_IFACE_ID(AHCI_DEV_IFACE),
152 IPC_M_AHCI_READ_BLOCKS, HI(blocknum), LO(blocknum), count, NULL);
153
154 async_share_out_start(exch, buf, AS_AREA_READ | AS_AREA_WRITE);
155
156 async_exchange_end(exch);
157
158 errno_t rc;
159 async_wait_for(req, &rc);
160
161 return rc;
162}
163
164errno_t ahci_write_blocks(async_sess_t *sess, uint64_t blocknum, size_t count,
165 void *buf)
166{
167 async_exch_t *exch = async_exchange_begin(sess);
168 if (!exch)
169 return EINVAL;
170
171 aid_t req = async_send_4(exch, DEV_IFACE_ID(AHCI_DEV_IFACE),
172 IPC_M_AHCI_WRITE_BLOCKS, HI(blocknum), LO(blocknum), count, NULL);
173
174 async_share_out_start(exch, buf, AS_AREA_READ | AS_AREA_WRITE);
175
176 async_exchange_end(exch);
177
178 errno_t rc;
179 async_wait_for(req, &rc);
180
181 return rc;
182}
183
184static void remote_ahci_get_sata_device_name(ddf_fun_t *, void *, ipc_call_t *);
185static void remote_ahci_get_num_blocks(ddf_fun_t *, void *, ipc_call_t *);
186static void remote_ahci_get_block_size(ddf_fun_t *, void *, ipc_call_t *);
187static void remote_ahci_read_blocks(ddf_fun_t *, void *, ipc_call_t *);
188static void remote_ahci_write_blocks(ddf_fun_t *, void *, ipc_call_t *);
189
190/** Remote AHCI interface operations. */
191static const remote_iface_func_ptr_t remote_ahci_iface_ops [] = {
192 [IPC_M_AHCI_GET_SATA_DEVICE_NAME] = remote_ahci_get_sata_device_name,
193 [IPC_M_AHCI_GET_NUM_BLOCKS] = remote_ahci_get_num_blocks,
194 [IPC_M_AHCI_GET_BLOCK_SIZE] = remote_ahci_get_block_size,
195 [IPC_M_AHCI_READ_BLOCKS] = remote_ahci_read_blocks,
196 [IPC_M_AHCI_WRITE_BLOCKS] = remote_ahci_write_blocks
197};
198
199/** Remote AHCI interface structure.
200 */
201const remote_iface_t remote_ahci_iface = {
202 .method_count = ARRAY_SIZE(remote_ahci_iface_ops),
203 .methods = remote_ahci_iface_ops
204};
205
206void remote_ahci_get_sata_device_name(ddf_fun_t *fun, void *iface,
207 ipc_call_t *call)
208{
209 const ahci_iface_t *ahci_iface = (ahci_iface_t *) iface;
210
211 if (ahci_iface->get_sata_device_name == NULL) {
212 async_answer_0(call, ENOTSUP);
213 return;
214 }
215
216 const size_t sata_dev_name_length =
217 (size_t) DEV_IPC_GET_ARG1(*call);
218
219 char *sata_dev_name = malloc(sata_dev_name_length);
220 if (sata_dev_name == NULL) {
221 async_answer_0(call, ENOMEM);
222 return;
223 }
224
225 const errno_t ret = ahci_iface->get_sata_device_name(fun,
226 sata_dev_name_length, sata_dev_name);
227
228 ipc_call_t data;
229 size_t real_size;
230 if ((async_data_read_receive(&data, &real_size)) &&
231 (real_size == sata_dev_name_length))
232 async_data_read_finalize(&data, sata_dev_name,
233 sata_dev_name_length);
234
235 free(sata_dev_name);
236 async_answer_0(call, ret);
237}
238
239static void remote_ahci_get_num_blocks(ddf_fun_t *fun, void *iface,
240 ipc_call_t *call)
241{
242 const ahci_iface_t *ahci_iface = (ahci_iface_t *) iface;
243
244 if (ahci_iface->get_num_blocks == NULL) {
245 async_answer_0(call, ENOTSUP);
246 return;
247 }
248
249 uint64_t blocks;
250 const errno_t ret = ahci_iface->get_num_blocks(fun, &blocks);
251
252 if (ret != EOK)
253 async_answer_0(call, ret);
254 else
255 async_answer_2(call, EOK, HI(blocks), LO(blocks));
256}
257
258static void remote_ahci_get_block_size(ddf_fun_t *fun, void *iface,
259 ipc_call_t *call)
260{
261 const ahci_iface_t *ahci_iface = (ahci_iface_t *) iface;
262
263 if (ahci_iface->get_block_size == NULL) {
264 async_answer_0(call, ENOTSUP);
265 return;
266 }
267
268 size_t blocks;
269 const errno_t ret = ahci_iface->get_block_size(fun, &blocks);
270
271 if (ret != EOK)
272 async_answer_0(call, ret);
273 else
274 async_answer_1(call, EOK, blocks);
275}
276
277void remote_ahci_read_blocks(ddf_fun_t *fun, void *iface, ipc_call_t *call)
278{
279 const ahci_iface_t *ahci_iface = (ahci_iface_t *) iface;
280
281 if (ahci_iface->read_blocks == NULL) {
282 async_answer_0(call, ENOTSUP);
283 return;
284 }
285
286 ipc_call_t data;
287 size_t maxblock_size;
288 unsigned int flags;
289 async_share_out_receive(&data, &maxblock_size, &flags);
290
291 void *buf;
292 async_share_out_finalize(&data, &buf);
293
294 const uint64_t blocknum =
295 (((uint64_t) (DEV_IPC_GET_ARG1(*call))) << 32) |
296 (((uint64_t) (DEV_IPC_GET_ARG2(*call))) & 0xffffffff);
297 const size_t cnt = (size_t) DEV_IPC_GET_ARG3(*call);
298
299 const errno_t ret = ahci_iface->read_blocks(fun, blocknum, cnt, buf);
300
301 async_answer_0(call, ret);
302}
303
304void remote_ahci_write_blocks(ddf_fun_t *fun, void *iface, ipc_call_t *call)
305{
306 const ahci_iface_t *ahci_iface = (ahci_iface_t *) iface;
307
308 if (ahci_iface->read_blocks == NULL) {
309 async_answer_0(call, ENOTSUP);
310 return;
311 }
312
313 ipc_call_t data;
314 size_t maxblock_size;
315 unsigned int flags;
316 async_share_out_receive(&data, &maxblock_size, &flags);
317
318 void *buf;
319 async_share_out_finalize(&data, &buf);
320
321 const uint64_t blocknum =
322 (((uint64_t)(DEV_IPC_GET_ARG1(*call))) << 32) |
323 (((uint64_t)(DEV_IPC_GET_ARG2(*call))) & 0xffffffff);
324 const size_t cnt = (size_t) DEV_IPC_GET_ARG3(*call);
325
326 const errno_t ret = ahci_iface->write_blocks(fun, blocknum, cnt, buf);
327
328 async_answer_0(call, ret);
329}
330
331/**
332 * @}
333 */
Note: See TracBrowser for help on using the repository browser.