source: mainline/uspace/drv/bus/usb/usbmid/usbmid.c@ 1a4ea01d

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 1a4ea01d was 5203e256, checked in by Martin Decky <martin@…>, 14 years ago

keep the drivers source tree tidy by using logical subdirectories

  • Property mode set to 100644
File size: 4.2 KB
Line 
1/*
2 * Copyright (c) 2011 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
29/** @addtogroup drvusbmid
30 * @{
31 */
32/**
33 * @file
34 * Helper functions.
35 */
36#include <errno.h>
37#include <str_error.h>
38#include <stdlib.h>
39#include <usb_iface.h>
40#include <usb/ddfiface.h>
41#include <usb/dev/pipes.h>
42#include <usb/classes/classes.h>
43#include <usb/dev/recognise.h>
44#include "usbmid.h"
45
46/** Callback for DDF USB interface. */
47static int usb_iface_get_address_impl(ddf_fun_t *fun, devman_handle_t handle,
48 usb_address_t *address)
49{
50 return usb_iface_get_address_hub_impl(fun, handle, address);
51}
52
53/** Callback for DDF USB interface. */
54static int usb_iface_get_interface_impl(ddf_fun_t *fun, devman_handle_t handle,
55 int *iface_no)
56{
57 assert(fun);
58
59 usbmid_interface_t *iface = fun->driver_data;
60 assert(iface);
61
62 if (iface_no != NULL) {
63 *iface_no = iface->interface_no;
64 }
65
66 return EOK;
67}
68
69/** DDF interface of the child - interface function. */
70static usb_iface_t child_usb_iface = {
71 .get_hc_handle = usb_iface_get_hc_handle_hub_child_impl,
72 .get_address = usb_iface_get_address_impl,
73 .get_interface = usb_iface_get_interface_impl
74};
75
76/** Operations for children - interface functions. */
77static ddf_dev_ops_t child_device_ops = {
78 .interfaces[USB_DEV_IFACE] = &child_usb_iface
79};
80
81
82/** Spawn new child device from one interface.
83 *
84 * @param parent Parent MID device.
85 * @param iface Interface information.
86 * @param device_descriptor Device descriptor.
87 * @param interface_descriptor Interface descriptor.
88 * @return Error code.
89 */
90int usbmid_spawn_interface_child(usb_device_t *parent,
91 usbmid_interface_t *iface,
92 const usb_standard_device_descriptor_t *device_descriptor,
93 const usb_standard_interface_descriptor_t *interface_descriptor)
94{
95 ddf_fun_t *child = NULL;
96 char *child_name = NULL;
97 int rc;
98
99 /*
100 * Name is class name followed by interface number.
101 * The interface number shall provide uniqueness while the
102 * class name something humanly understandable.
103 */
104 rc = asprintf(&child_name, "%s%d",
105 usb_str_class(interface_descriptor->interface_class),
106 (int) interface_descriptor->interface_number);
107 if (rc < 0) {
108 goto error_leave;
109 }
110
111 /* Create the device. */
112 child = ddf_fun_create(parent->ddf_dev, fun_inner, child_name);
113 if (child == NULL) {
114 rc = ENOMEM;
115 goto error_leave;
116 }
117
118 iface->fun = child;
119
120 child->driver_data = iface;
121 child->ops = &child_device_ops;
122
123 rc = usb_device_create_match_ids_from_interface(device_descriptor,
124 interface_descriptor,
125 &child->match_ids);
126 if (rc != EOK) {
127 goto error_leave;
128 }
129
130 rc = ddf_fun_bind(child);
131 if (rc != EOK) {
132 goto error_leave;
133 }
134
135 return EOK;
136
137error_leave:
138 if (child != NULL) {
139 child->name = NULL;
140 /* This takes care of match_id deallocation as well. */
141 ddf_fun_destroy(child);
142 }
143 if (child_name != NULL) {
144 free(child_name);
145 }
146
147 return rc;
148}
149
150/**
151 * @}
152 */
Note: See TracBrowser for help on using the repository browser.