/* * Copyright (c) 2010 Vojtech Horky * Copyright (c) 2011 Jan Vesely * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * - The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ /** @addtogroup libdrv * @{ */ /** @file */ #include #include #include #include #include #include "usb_iface.h" #include "ddf/driver.h" usb_dev_session_t *usb_dev_connect(devman_handle_t handle) { return devman_device_connect(handle, IPC_FLAG_BLOCKING); } usb_dev_session_t *usb_dev_connect_to_self(ddf_dev_t *dev) { return devman_parent_device_connect(ddf_dev_get_handle(dev), IPC_FLAG_BLOCKING); } void usb_dev_disconnect(usb_dev_session_t *sess) { if (sess) async_hangup(sess); } typedef enum { IPC_M_USB_GET_MY_DESCRIPTION, } usb_iface_funcs_t; /** Tell interface number given device can use. * @param[in] exch IPC communication exchange * @param[in] handle Id of the device * @param[out] usb_iface Assigned USB interface * @return Error code. */ int usb_get_my_description(async_exch_t *exch, usb_device_desc_t *desc) { if (!exch) return EBADMEM; usb_device_desc_t tmp_desc; const int ret = async_req_1_4(exch, DEV_IFACE_ID(USB_DEV_IFACE), IPC_M_USB_GET_MY_DESCRIPTION, (sysarg_t *) &tmp_desc.address, (sysarg_t *) &tmp_desc.speed, &tmp_desc.handle, (sysarg_t *) &tmp_desc.iface); if (ret == EOK && desc) *desc = tmp_desc; return ret; } static void remote_usb_get_my_description(ddf_fun_t *, void *, ipc_callid_t, ipc_call_t *); /** Remote USB interface operations. */ static const remote_iface_func_ptr_t remote_usb_iface_ops [] = { [IPC_M_USB_GET_MY_DESCRIPTION] = remote_usb_get_my_description, }; /** Remote USB interface structure. */ const remote_iface_t remote_usb_iface = { .method_count = ARRAY_SIZE(remote_usb_iface_ops), .methods = remote_usb_iface_ops, }; void remote_usb_get_my_description(ddf_fun_t *fun, void *iface, ipc_callid_t callid, ipc_call_t *call) { const usb_iface_t *usb_iface = (usb_iface_t *) iface; if (usb_iface->get_my_description == NULL) { async_answer_0(callid, ENOTSUP); return; } usb_device_desc_t desc; const int ret = usb_iface->get_my_description(fun, &desc); if (ret != EOK) { async_answer_0(callid, ret); } else { async_answer_4(callid, EOK, (sysarg_t) desc.address, (sysarg_t) desc.speed, desc.handle, desc.iface); } } /** * @} */