| [774afaae] | 1 | /* | 
|---|
|  | 2 | * Copyright (c) 2010 Vojtech Horky | 
|---|
|  | 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 |  | 
|---|
| [bd8c753d] | 29 | /** @addtogroup drvusbvhc | 
|---|
| [774afaae] | 30 | * @{ | 
|---|
|  | 31 | */ | 
|---|
|  | 32 | /** @file | 
|---|
|  | 33 | * @brief | 
|---|
|  | 34 | */ | 
|---|
|  | 35 | #include <usb/classes/classes.h> | 
|---|
|  | 36 | #include <usbvirt/device.h> | 
|---|
|  | 37 | #include <assert.h> | 
|---|
|  | 38 | #include <errno.h> | 
|---|
|  | 39 | #include <str_error.h> | 
|---|
| [eb1a2f4] | 40 | #include <assert.h> | 
|---|
| [774afaae] | 41 | #include <stdlib.h> | 
|---|
| [eb1a2f4] | 42 | #include <ddf/driver.h> | 
|---|
| [774afaae] | 43 |  | 
|---|
|  | 44 | #include "virthub.h" | 
|---|
|  | 45 | #include "hub.h" | 
|---|
|  | 46 |  | 
|---|
|  | 47 |  | 
|---|
|  | 48 | /** Standard device descriptor. */ | 
|---|
|  | 49 | usb_standard_device_descriptor_t std_device_descriptor = { | 
|---|
|  | 50 | .length = sizeof(usb_standard_device_descriptor_t), | 
|---|
|  | 51 | .descriptor_type = USB_DESCTYPE_DEVICE, | 
|---|
|  | 52 | .usb_spec_version = 0x110, | 
|---|
|  | 53 | .device_class = USB_CLASS_HUB, | 
|---|
|  | 54 | .device_subclass = 0, | 
|---|
|  | 55 | .device_protocol = 0, | 
|---|
|  | 56 | .max_packet_size = 64, | 
|---|
|  | 57 | .configuration_count = 1 | 
|---|
|  | 58 | }; | 
|---|
|  | 59 |  | 
|---|
|  | 60 | /** Standard interface descriptor. */ | 
|---|
|  | 61 | usb_standard_interface_descriptor_t std_interface_descriptor = { | 
|---|
|  | 62 | .length = sizeof(usb_standard_interface_descriptor_t), | 
|---|
|  | 63 | .descriptor_type = USB_DESCTYPE_INTERFACE, | 
|---|
|  | 64 | .interface_number = 0, | 
|---|
|  | 65 | .alternate_setting = 0, | 
|---|
|  | 66 | .endpoint_count = 1, | 
|---|
|  | 67 | .interface_class = USB_CLASS_HUB, | 
|---|
|  | 68 | .interface_subclass = 0, | 
|---|
|  | 69 | .interface_protocol = 0, | 
|---|
|  | 70 | .str_interface = 0 | 
|---|
|  | 71 | }; | 
|---|
|  | 72 |  | 
|---|
| [70e5ad5] | 73 | /** Hub descriptor. */ | 
|---|
| [774afaae] | 74 | hub_descriptor_t hub_descriptor = { | 
|---|
|  | 75 | .length = sizeof(hub_descriptor_t), | 
|---|
|  | 76 | .type = USB_DESCTYPE_HUB, | 
|---|
|  | 77 | .port_count = HUB_PORT_COUNT, | 
|---|
|  | 78 | .characteristics = 0, | 
|---|
|  | 79 | .power_on_warm_up = 50, /* Huh? */ | 
|---|
|  | 80 | .max_current = 100, /* Huh again. */ | 
|---|
|  | 81 | .removable_device = { 0 }, | 
|---|
|  | 82 | .port_power = { 0xFF } | 
|---|
|  | 83 | }; | 
|---|
|  | 84 |  | 
|---|
|  | 85 | /** Endpoint descriptor. */ | 
|---|
|  | 86 | usb_standard_endpoint_descriptor_t endpoint_descriptor = { | 
|---|
|  | 87 | .length = sizeof(usb_standard_endpoint_descriptor_t), | 
|---|
|  | 88 | .descriptor_type = USB_DESCTYPE_ENDPOINT, | 
|---|
|  | 89 | .endpoint_address = HUB_STATUS_CHANGE_PIPE | 128, | 
|---|
|  | 90 | .attributes = USB_TRANSFER_INTERRUPT, | 
|---|
|  | 91 | .max_packet_size = 8, | 
|---|
|  | 92 | .poll_interval = 0xFF | 
|---|
|  | 93 | }; | 
|---|
|  | 94 |  | 
|---|
|  | 95 | /** Standard configuration descriptor. */ | 
|---|
|  | 96 | usb_standard_configuration_descriptor_t std_configuration_descriptor = { | 
|---|
|  | 97 | .length = sizeof(usb_standard_configuration_descriptor_t), | 
|---|
|  | 98 | .descriptor_type = USB_DESCTYPE_CONFIGURATION, | 
|---|
|  | 99 | .total_length = | 
|---|
|  | 100 | sizeof(usb_standard_configuration_descriptor_t) | 
|---|
|  | 101 | + sizeof(std_interface_descriptor) | 
|---|
|  | 102 | + sizeof(hub_descriptor) | 
|---|
|  | 103 | + sizeof(endpoint_descriptor) | 
|---|
|  | 104 | , | 
|---|
|  | 105 | .interface_count = 1, | 
|---|
|  | 106 | .configuration_number = HUB_CONFIGURATION_ID, | 
|---|
|  | 107 | .str_configuration = 0, | 
|---|
|  | 108 | .attributes = 128, /* denotes bus-powered device */ | 
|---|
|  | 109 | .max_power = 50 | 
|---|
|  | 110 | }; | 
|---|
|  | 111 |  | 
|---|
|  | 112 | /** All hub configuration descriptors. */ | 
|---|
|  | 113 | static usbvirt_device_configuration_extras_t extra_descriptors[] = { | 
|---|
|  | 114 | { | 
|---|
|  | 115 | .data = (uint8_t *) &std_interface_descriptor, | 
|---|
|  | 116 | .length = sizeof(std_interface_descriptor) | 
|---|
|  | 117 | }, | 
|---|
|  | 118 | { | 
|---|
|  | 119 | .data = (uint8_t *) &hub_descriptor, | 
|---|
|  | 120 | .length = sizeof(hub_descriptor) | 
|---|
|  | 121 | }, | 
|---|
|  | 122 | { | 
|---|
|  | 123 | .data = (uint8_t *) &endpoint_descriptor, | 
|---|
|  | 124 | .length = sizeof(endpoint_descriptor) | 
|---|
|  | 125 | } | 
|---|
|  | 126 | }; | 
|---|
|  | 127 |  | 
|---|
|  | 128 | /** Hub configuration. */ | 
|---|
|  | 129 | usbvirt_device_configuration_t configuration = { | 
|---|
|  | 130 | .descriptor = &std_configuration_descriptor, | 
|---|
|  | 131 | .extra = extra_descriptors, | 
|---|
|  | 132 | .extra_count = sizeof(extra_descriptors)/sizeof(extra_descriptors[0]) | 
|---|
|  | 133 | }; | 
|---|
|  | 134 |  | 
|---|
|  | 135 | /** Hub standard descriptors. */ | 
|---|
|  | 136 | usbvirt_descriptors_t descriptors = { | 
|---|
|  | 137 | .device = &std_device_descriptor, | 
|---|
|  | 138 | .configuration = &configuration, | 
|---|
|  | 139 | .configuration_count = 1, | 
|---|
|  | 140 | }; | 
|---|
|  | 141 |  | 
|---|
| [70e5ad5] | 142 | /** Initializes virtual hub device. | 
|---|
|  | 143 | * | 
|---|
|  | 144 | * @param dev Virtual USB device backend. | 
|---|
|  | 145 | * @return Error code. | 
|---|
|  | 146 | */ | 
|---|
| [774afaae] | 147 | int virthub_init(usbvirt_device_t *dev) | 
|---|
|  | 148 | { | 
|---|
|  | 149 | if (dev == NULL) { | 
|---|
|  | 150 | return EBADMEM; | 
|---|
|  | 151 | } | 
|---|
|  | 152 | dev->ops = &hub_ops; | 
|---|
|  | 153 | dev->descriptors = &descriptors; | 
|---|
|  | 154 |  | 
|---|
|  | 155 | hub_t *hub = malloc(sizeof(hub_t)); | 
|---|
|  | 156 | if (hub == NULL) { | 
|---|
|  | 157 | return ENOMEM; | 
|---|
|  | 158 | } | 
|---|
|  | 159 |  | 
|---|
|  | 160 | hub_init(hub); | 
|---|
|  | 161 | dev->device_data = hub; | 
|---|
|  | 162 |  | 
|---|
| [6cb58e6] | 163 | return EOK; | 
|---|
| [774afaae] | 164 | } | 
|---|
|  | 165 |  | 
|---|
| [70e5ad5] | 166 | /** Connect a device to a virtual hub. | 
|---|
|  | 167 | * | 
|---|
|  | 168 | * @param dev Virtual device representing the hub. | 
|---|
|  | 169 | * @param conn Device to be connected. | 
|---|
|  | 170 | * @return Port device was connected to. | 
|---|
|  | 171 | */ | 
|---|
| [6cb58e6] | 172 | int virthub_connect_device(usbvirt_device_t *dev, vhc_virtdev_t *conn) | 
|---|
| [774afaae] | 173 | { | 
|---|
|  | 174 | assert(dev != NULL); | 
|---|
|  | 175 | assert(conn != NULL); | 
|---|
|  | 176 |  | 
|---|
|  | 177 | hub_t *hub = (hub_t *) dev->device_data; | 
|---|
|  | 178 |  | 
|---|
|  | 179 | hub_acquire(hub); | 
|---|
|  | 180 | size_t port = hub_connect_device(hub, conn); | 
|---|
|  | 181 | hub_release(hub); | 
|---|
|  | 182 |  | 
|---|
|  | 183 | return port; | 
|---|
|  | 184 | } | 
|---|
|  | 185 |  | 
|---|
| [70e5ad5] | 186 | /** Disconnect a device from a virtual hub. | 
|---|
|  | 187 | * | 
|---|
|  | 188 | * @param dev Virtual device representing the hub. | 
|---|
|  | 189 | * @param conn Device to be disconnected. | 
|---|
|  | 190 | * @return Error code. | 
|---|
|  | 191 | */ | 
|---|
| [6cb58e6] | 192 | int virthub_disconnect_device(usbvirt_device_t *dev, vhc_virtdev_t *conn) | 
|---|
| [774afaae] | 193 | { | 
|---|
|  | 194 | assert(dev != NULL); | 
|---|
|  | 195 | assert(conn != NULL); | 
|---|
|  | 196 |  | 
|---|
|  | 197 | hub_t *hub = (hub_t *) dev->device_data; | 
|---|
|  | 198 |  | 
|---|
|  | 199 | hub_acquire(hub); | 
|---|
| [68a68705] | 200 | hub_disconnect_device(hub, conn); | 
|---|
| [774afaae] | 201 | hub_release(hub); | 
|---|
|  | 202 |  | 
|---|
| [6cb58e6] | 203 | return EOK; | 
|---|
| [774afaae] | 204 | } | 
|---|
|  | 205 |  | 
|---|
| [70e5ad5] | 206 | /** Whether trafic is propagated to given device. | 
|---|
|  | 207 | * | 
|---|
|  | 208 | * @param dev Virtual device representing the hub. | 
|---|
|  | 209 | * @param conn Connected device. | 
|---|
|  | 210 | * @return Whether port is signalling to the device. | 
|---|
|  | 211 | */ | 
|---|
| [6cb58e6] | 212 | bool virthub_is_device_enabled(usbvirt_device_t *dev, vhc_virtdev_t *conn) | 
|---|
| [774afaae] | 213 | { | 
|---|
|  | 214 | assert(dev != NULL); | 
|---|
|  | 215 | assert(conn != NULL); | 
|---|
|  | 216 |  | 
|---|
|  | 217 | hub_t *hub = (hub_t *) dev->device_data; | 
|---|
|  | 218 |  | 
|---|
|  | 219 | hub_acquire(hub); | 
|---|
|  | 220 |  | 
|---|
|  | 221 | hub_port_state_t state = HUB_PORT_STATE_UNKNOWN; | 
|---|
|  | 222 | size_t port = hub_find_device(hub, conn); | 
|---|
|  | 223 | if (port != (size_t) -1) { | 
|---|
|  | 224 | state = hub_get_port_state(hub, port); | 
|---|
|  | 225 | } | 
|---|
|  | 226 | hub_release(hub); | 
|---|
|  | 227 |  | 
|---|
|  | 228 | return state == HUB_PORT_STATE_ENABLED; | 
|---|
|  | 229 | } | 
|---|
|  | 230 |  | 
|---|
| [70e5ad5] | 231 | /** Format status of a virtual hub. | 
|---|
|  | 232 | * | 
|---|
|  | 233 | * @param dev Virtual device representing the hub. | 
|---|
|  | 234 | * @param[out] status Hub status information. | 
|---|
|  | 235 | * @param[in] len Size of the @p status buffer. | 
|---|
|  | 236 | */ | 
|---|
| [774afaae] | 237 | void virthub_get_status(usbvirt_device_t *dev, char *status, size_t len) | 
|---|
|  | 238 | { | 
|---|
|  | 239 | assert(dev != NULL); | 
|---|
|  | 240 | if (len == 0) { | 
|---|
|  | 241 | return; | 
|---|
|  | 242 | } | 
|---|
|  | 243 |  | 
|---|
|  | 244 | hub_t *hub = (hub_t *) dev->device_data; | 
|---|
|  | 245 |  | 
|---|
|  | 246 | char port_status[HUB_PORT_COUNT + 1]; | 
|---|
|  | 247 |  | 
|---|
|  | 248 | size_t i; | 
|---|
|  | 249 | for (i = 0; i < HUB_PORT_COUNT; i++) { | 
|---|
|  | 250 | port_status[i] = hub_port_state_to_char( | 
|---|
|  | 251 | hub_get_port_state(hub, i)); | 
|---|
|  | 252 | } | 
|---|
|  | 253 | port_status[HUB_PORT_COUNT] = 0; | 
|---|
|  | 254 |  | 
|---|
|  | 255 | snprintf(status, len, "vhub:%s", port_status); | 
|---|
|  | 256 | } | 
|---|
|  | 257 |  | 
|---|
|  | 258 | /** | 
|---|
|  | 259 | * @} | 
|---|
|  | 260 | */ | 
|---|