/* * Copyright (c) 2011 Vojtech Horky * 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 drvusbmid * @{ */ /** * @file * Main routines of USB multi interface device driver. */ #include #include #include #include #include #include #include #include "usbmid.h" /** Callback when new MID device is attached to the host. * * @param gen_dev Generic DDF device representing the new device. * @return Error code. */ static int usbmid_device_add(usb_device_t *dev) { usb_log_info("Taking care of new MID `%s'.\n", dev->ddf_dev->name); usb_pipe_start_long_transfer(&dev->ctrl_pipe); bool accept = usbmid_explore_device(dev); usb_pipe_end_long_transfer(&dev->ctrl_pipe); if (!accept) { return ENOTSUP; } return EOK; } /*----------------------------------------------------------------------------*/ /** Callback when a MID device is about to be removed from the host. * * @param gen_dev Generic DDF device representing the removed device. * @return Error code. */ static int usbmid_device_remove(usb_device_t *dev) { assert(dev); int ret = ENOTSUP; usb_mid_t *usb_mid = dev->driver_data; assert(usb_mid); /* Signal all interface functions */ list_foreach(usb_mid->interface_list, item) { usbmid_interface_t *iface = usbmid_interface_from_link(item); usb_log_info("Signaling remove to child for interface " "%d (%s).\n", iface->interface_no, usb_str_class(iface->interface->interface_class)); // TODO cascade the call. } return ret; } /*----------------------------------------------------------------------------*/ /** Callback when a MID device was removed from the host. * * @param gen_dev Generic DDF device representing the removed device. * @return Error code. */ static int usbmid_device_gone(usb_device_t *dev) { assert(dev); usb_log_info("USB MID gone: `%s'.\n", dev->ddf_dev->name); /* Remove ctl function */ usb_mid_t *usb_mid = dev->driver_data; int ret = ddf_fun_unbind(usb_mid->ctl_fun); if (ret != EOK) { usb_log_error("Failed to unbind USB MID ctl function: %s.\n", str_error(ret)); return ret; } ddf_fun_destroy(usb_mid->ctl_fun); /* Now remove all other functions */ while (!list_empty(&usb_mid->interface_list)) { link_t *item = list_first(&usb_mid->interface_list); list_remove(item); usbmid_interface_t *iface = usbmid_interface_from_link(item); usb_log_info("Removing child for interface %d (%s).\n", iface->interface_no, usb_str_class(iface->interface->interface_class)); const int pret = usbmid_interface_destroy(iface); if (pret != EOK) { usb_log_error("Failed to remove child for interface " "%d (%s): %s\n", iface->interface_no, usb_str_class(iface->interface->interface_class), str_error(pret)); ret = pret; } } return ret; } /** USB MID driver ops. */ static const usb_driver_ops_t mid_driver_ops = { .device_add = usbmid_device_add, .device_rem = usbmid_device_remove, .device_gone = usbmid_device_gone, }; /** USB MID driver. */ static const usb_driver_t mid_driver = { .name = NAME, .ops = &mid_driver_ops, .endpoints = NULL }; int main(int argc, char *argv[]) { printf(NAME ": USB multi interface device driver.\n"); usb_log_enable(USB_LOG_LEVEL_DEFAULT, NAME); return usb_driver_main(&mid_driver); } /** * @} */