source: mainline/uspace/lib/libdrv/generic/driver.c@ 5af21c5

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 5af21c5 was 5af21c5, checked in by Lenka Trochtova <trochtova.lenka@…>, 15 years ago

fixed a bug

  • Property mode set to 100644
File size: 8.1 KB
RevLine 
[c16cf62]1/*
2 * Copyright (c) 2010 Lenka Trochtova
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/**
30 * @defgroup libdrv generic device driver support.
31 * @brief HelenOS generic device driver support.
32 * @{
33 */
34
35/** @file
36 */
[52b7b1bb]37
[c16cf62]38#include <assert.h>
39#include <ipc/services.h>
40#include <ipc/ns.h>
41#include <async.h>
42#include <stdio.h>
43#include <errno.h>
44#include <bool.h>
45#include <fibril_synch.h>
46#include <stdlib.h>
47#include <string.h>
48#include <ctype.h>
[66babbd]49#include <errno.h>
[c16cf62]50
51#include <devman.h>
52#include <ipc/devman.h>
[084ff99]53#include <ipc/driver.h>
[c16cf62]54
55#include "driver.h"
56
57static driver_t *driver;
[084ff99]58LIST_INITIALIZE(devices);
[9a66bc2e]59FIBRIL_MUTEX_INITIALIZE(devices_mutex);
[084ff99]60
[9a66bc2e]61static void add_to_devices_list(device_t *dev)
62{
63 fibril_mutex_lock(&devices_mutex);
64 list_append(&dev->link, &devices);
65 fibril_mutex_unlock(&devices_mutex);
66}
67
68static void remove_from_devices_list(device_t *dev)
69{
70 fibril_mutex_lock(&devices_mutex);
[5af21c5]71 list_remove(&dev->link);
[9a66bc2e]72 fibril_mutex_unlock(&devices_mutex);
73}
74
[52b7b1bb]75static device_t * driver_get_device(link_t *devices, device_handle_t handle)
76{
[a1769ee]77 device_t *dev = NULL;
[9a66bc2e]78
79 fibril_mutex_lock(&devices_mutex);
[a1769ee]80 link_t *link = devices->next;
81 while (link != devices) {
82 dev = list_get_instance(link, device_t, link);
83 if (handle == dev->handle) {
[9a66bc2e]84 fibril_mutex_unlock(&devices_mutex);
[a1769ee]85 return dev;
86 }
[9a66bc2e]87 link = link->next;
[a1769ee]88 }
[9a66bc2e]89 fibril_mutex_unlock(&devices_mutex);
[52b7b1bb]90
[a1769ee]91 return NULL;
92}
93
[52b7b1bb]94static void driver_add_device(ipc_callid_t iid, ipc_call_t *icall)
[084ff99]95{
[df747b9c]96 char *dev_name = NULL;
97 int res = EOK;
[9a66bc2e]98
[52b7b1bb]99 device_handle_t dev_handle = IPC_GET_ARG1(*icall);
[5af21c5]100 device_t *dev = create_device();
[084ff99]101 dev->handle = dev_handle;
[df747b9c]102
103 async_string_receive(&dev_name, 0, NULL);
104 dev->name = dev_name;
105
106 add_to_devices_list(dev);
107 res = driver->driver_ops->add_device(dev);
108 if (0 == res) {
[9a66bc2e]109 printf("%s: new device with handle = %x was added.\n", driver->name, dev_handle);
110 } else {
[df747b9c]111 printf("%s: failed to add a new device with handle = %d.\n", driver->name, dev_handle);
[9a66bc2e]112 remove_from_devices_list(dev);
[5af21c5]113 delete_device(dev);
[084ff99]114 }
[df747b9c]115
116 ipc_answer_0(iid, res);
[084ff99]117}
[c16cf62]118
119static void driver_connection_devman(ipc_callid_t iid, ipc_call_t *icall)
120{
121 /* Accept connection */
122 ipc_answer_0(iid, EOK);
[52b7b1bb]123
[c16cf62]124 bool cont = true;
125 while (cont) {
126 ipc_call_t call;
127 ipc_callid_t callid = async_get_call(&call);
[52b7b1bb]128
[c16cf62]129 switch (IPC_GET_METHOD(call)) {
130 case IPC_M_PHONE_HUNGUP:
131 cont = false;
132 continue;
133 case DRIVER_ADD_DEVICE:
[084ff99]134 driver_add_device(callid, &call);
[c16cf62]135 break;
136 default:
137 if (!(callid & IPC_CALLID_NOTIFICATION))
138 ipc_answer_0(callid, ENOENT);
139 }
[52b7b1bb]140 }
[c16cf62]141}
142
[52b7b1bb]143/**
[a1769ee]144 * Generic client connection handler both for applications and drivers.
[52b7b1bb]145 *
[9a66bc2e]146 * @param drv true for driver client, false for other clients (applications, services etc.).
[a1769ee]147 */
[9a66bc2e]148static void driver_connection_gen(ipc_callid_t iid, ipc_call_t *icall, bool drv)
[52b7b1bb]149{
150 // Answer the first IPC_M_CONNECT_ME_TO call and remember the handle of the device to which the client connected.
[9a66bc2e]151 device_handle_t handle = IPC_GET_ARG2(*icall);
[a1769ee]152 device_t *dev = driver_get_device(&devices, handle);
[52b7b1bb]153
[a1769ee]154 if (dev == NULL) {
[9a66bc2e]155 printf("%s: driver_connection_gen error - no device with handle %x was found.\n", driver->name, handle);
[a1769ee]156 ipc_answer_0(iid, ENOENT);
157 return;
158 }
[5cd136ab]159
160
161 // TODO - if the client is not a driver, check whether it is allowed to use the device
[a1769ee]162
[52b7b1bb]163 // TODO open the device (introduce some callbacks for opening and closing devices registered by the driver)
[a1769ee]164
[9a66bc2e]165 ipc_answer_0(iid, EOK);
[52b7b1bb]166
[a1769ee]167 while (1) {
168 ipc_callid_t callid;
169 ipc_call_t call;
170 callid = async_get_call(&call);
171 ipcarg_t method = IPC_GET_METHOD(call);
[3843ecb]172 int iface_idx;
173
[a1769ee]174 switch (method) {
175 case IPC_M_PHONE_HUNGUP:
[52b7b1bb]176
177 // TODO close the device
178
[a1769ee]179 ipc_answer_0(callid, EOK);
180 return;
[3843ecb]181 default:
182 // convert ipc interface id to interface index
183
184 iface_idx = DEV_IFACE_IDX(method);
185
186 if (!is_valid_iface_idx(iface_idx)) {
[52b7b1bb]187 // this is not device's interface
[9a66bc2e]188 printf("%s: driver_connection_gen error - invalid interface id %x.", driver->name, method);
[52b7b1bb]189 ipc_answer_0(callid, ENOTSUP);
190 break;
191 }
192
193 // calling one of the device's interfaces
194
195 // get the device interface structure
[3843ecb]196 void *iface = device_get_iface(dev, iface_idx);
[52b7b1bb]197 if (NULL == iface) {
[9a66bc2e]198 printf("%s: driver_connection_gen error - ", driver->name);
199 printf("device with handle %x has no interface with id %x.\n", handle, method);
[a1769ee]200 ipc_answer_0(callid, ENOTSUP);
[52b7b1bb]201 break;
[a1769ee]202 }
[52b7b1bb]203
204 // get the corresponding interface for remote request handling ("remote interface")
[3843ecb]205 remote_iface_t* rem_iface = get_remote_iface(iface_idx);
[52b7b1bb]206 assert(NULL != rem_iface);
207
208 // get the method of the remote interface
209 ipcarg_t iface_method_idx = IPC_GET_ARG1(call);
210 remote_iface_func_ptr_t iface_method_ptr = get_remote_method(rem_iface, iface_method_idx);
211 if (NULL == iface_method_ptr) {
212 // the interface has not such method
[9a66bc2e]213 printf("%s: driver_connection_gen error - invalid interface method.", driver->name);
[52b7b1bb]214 ipc_answer_0(callid, ENOTSUP);
215 break;
216 }
217
218 // call the remote interface's method, which will receive parameters from the remote client
219 // and it will pass it to the corresponding local interface method associated with the device
220 // by its driver
221 (*iface_method_ptr)(dev, iface, callid, &call);
[a1769ee]222 break;
223 }
224 }
225}
226
[c16cf62]227static void driver_connection_driver(ipc_callid_t iid, ipc_call_t *icall)
228{
[52b7b1bb]229 driver_connection_gen(iid, icall, true);
[c16cf62]230}
231
232static void driver_connection_client(ipc_callid_t iid, ipc_call_t *icall)
233{
[52b7b1bb]234 driver_connection_gen(iid, icall, false);
[c16cf62]235}
236
237
238/** Function for handling connections to device driver.
239 *
240 */
241static void driver_connection(ipc_callid_t iid, ipc_call_t *icall)
242{
243 /* Select interface */
244 switch ((ipcarg_t) (IPC_GET_ARG1(*icall))) {
245 case DRIVER_DEVMAN:
246 // handle PnP events from device manager
247 driver_connection_devman(iid, icall);
248 break;
249 case DRIVER_DRIVER:
250 // handle request from drivers of child devices
251 driver_connection_driver(iid, icall);
252 break;
253 case DRIVER_CLIENT:
[52b7b1bb]254 // handle requests from client applications
[c16cf62]255 driver_connection_client(iid, icall);
256 break;
257
258 default:
[52b7b1bb]259 /* No such interface */
[c16cf62]260 ipc_answer_0(iid, ENOENT);
261 }
262}
263
[df747b9c]264int child_device_register(device_t *child, device_t *parent)
[7707954]265{
[2480e19]266 // printf("%s: child_device_register\n", driver->name);
[52b7b1bb]267
[bda60d9]268 assert(NULL != child->name);
[52b7b1bb]269
[df747b9c]270 int res;
271
[9a66bc2e]272 add_to_devices_list(child);
[df747b9c]273 if (EOK == (res = devman_child_device_register(child->name, &child->match_ids, parent->handle, &child->handle))) {
274 return res;
[7707954]275 }
[9a66bc2e]276 remove_from_devices_list(child);
[df747b9c]277 return res;
[7707954]278}
279
[52b7b1bb]280int driver_main(driver_t *drv)
[c16cf62]281{
282 // remember the driver structure - driver_ops will be called by generic handler for incoming connections
283 driver = drv;
[52b7b1bb]284
[c16cf62]285 // register driver by device manager with generic handler for incoming connections
[52b7b1bb]286 devman_driver_register(driver->name, driver_connection);
[c16cf62]287
288 async_manager();
289
290 // Never reached
[52b7b1bb]291 return 0;
[c16cf62]292}
293
294/**
295 * @}
[52b7b1bb]296 */
Note: See TracBrowser for help on using the repository browser.