1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 /* 22 * Copyright 2008 Sun Microsystems, Inc. All rights reserved. 23 * Use is subject to license terms. 24 */ 25 26 27 /* 28 * skeleton hub driver, the actual code is in hubdi.c 29 * as it is shared between the root hub and the other hub instances 30 */ 31 #if defined(lint) && !defined(DEBUG) 32 #define DEBUG 1 33 #endif 34 35 #include <sys/usb/usba/usbai_version.h> 36 #include <sys/usb/usba.h> 37 #include <sys/usb/usba/hubdi.h> 38 #include <sys/usb/hubd/hub.h> 39 #include <sys/usb/hubd/hubdvar.h> 40 41 static int hubd_open(dev_t *devp, int flags, int otyp, cred_t *credp); 42 static int hubd_close(dev_t dev, int flag, int otyp, cred_t *credp); 43 static int hubd_ioctl(dev_t dev, int cmd, intptr_t arg, int mode, 44 cred_t *credp, int *rvalp); 45 static int hubd_info(dev_info_t *dip, ddi_info_cmd_t infocmd, 46 void *arg, void **result); 47 extern int usba_hubdi_power(dev_info_t *dip, int comp, int level); 48 49 50 static struct cb_ops hubd_cb_ops = { 51 hubd_open, /* open */ 52 hubd_close, /* close */ 53 nodev, /* strategy */ 54 nodev, /* print */ 55 nodev, /* dump */ 56 nodev, /* read */ 57 nodev, /* write */ 58 hubd_ioctl, /* ioctl */ 59 nodev, /* devmap */ 60 nodev, /* mmap */ 61 nodev, /* segmap */ 62 nochpoll, /* poll */ 63 ddi_prop_op, /* cb_prop_op */ 64 NULL, /* streamtab */ 65 D_MP /* Driver compatibility flag */ 66 }; 67 68 static struct dev_ops hubd_ops = { 69 DEVO_REV, /* devo_rev, */ 70 0, /* refcnt */ 71 hubd_info, /* info */ 72 nulldev, /* identify */ 73 nulldev, /* probe */ 74 usba_hubdi_attach, /* attach */ 75 usba_hubdi_detach, /* detach */ 76 nodev, /* reset */ 77 &hubd_cb_ops, /* driver operations */ 78 &usba_hubdi_busops, /* bus operations */ 79 usba_hubdi_power, /* power */ 80 ddi_quiesce_not_needed, /* quiesce */ 81 }; 82 83 static struct modldrv modldrv = { 84 &mod_driverops, /* Type of module. This one is a driver */ 85 "USB Hub Driver", /* Name of the module. */ 86 &hubd_ops, /* driver ops */ 87 }; 88 89 static struct modlinkage modlinkage = { 90 MODREV_1, (void *)&modldrv, NULL 91 }; 92 93 94 extern void *hubd_statep; 95 96 int 97 _init(void) 98 { 99 int rval; 100 101 /* Initialize the soft state structures */ 102 if ((rval = ddi_soft_state_init(&hubd_statep, 103 sizeof (hubd_t), HUBD_INITIAL_SOFT_SPACE)) != 0) { 104 return (rval); 105 } 106 107 if ((rval = mod_install(&modlinkage)) != 0) { 108 ddi_soft_state_fini(&hubd_statep); 109 110 return (rval); 111 } 112 113 return (rval); 114 } 115 116 117 int 118 _fini(void) 119 { 120 int rval = mod_remove(&modlinkage); 121 if (rval == 0) { 122 ddi_soft_state_fini(&hubd_statep); 123 } 124 125 return (rval); 126 } 127 128 129 int 130 _info(struct modinfo *modinfop) 131 { 132 return (mod_info(&modlinkage, modinfop)); 133 } 134 135 136 static dev_info_t * 137 hubd_get_dip(dev_t dev) 138 { 139 minor_t minor = getminor(dev); 140 int instance = (int)minor & ~HUBD_IS_ROOT_HUB; 141 hubd_t *hubd = ddi_get_soft_state(hubd_statep, instance); 142 143 if (hubd) { 144 return (hubd->h_dip); 145 } else { 146 return (NULL); 147 } 148 } 149 150 /* 151 * info handler 152 */ 153 /*ARGSUSED*/ 154 int 155 hubd_info(dev_info_t *dip, ddi_info_cmd_t infocmd, 156 void *arg, void **result) 157 { 158 dev_t dev; 159 int instance; 160 int error = DDI_FAILURE; 161 162 switch (infocmd) { 163 case DDI_INFO_DEVT2DEVINFO: 164 *result = (void *)hubd_get_dip((dev_t)arg); 165 if (*result != NULL) { 166 error = DDI_SUCCESS; 167 } 168 break; 169 case DDI_INFO_DEVT2INSTANCE: 170 dev = (dev_t)arg; 171 instance = HUBD_UNIT(dev); 172 *result = (void *)(intptr_t)instance; 173 error = DDI_SUCCESS; 174 break; 175 default: 176 break; 177 } 178 return (error); 179 } 180 181 static int 182 hubd_open(dev_t *devp, int flags, int otyp, cred_t *credp) 183 { 184 dev_info_t *dip = hubd_get_dip(*devp); 185 186 return (usba_hubdi_open(dip, devp, flags, otyp, credp)); 187 } 188 189 190 static int 191 hubd_close(dev_t dev, int flag, int otyp, cred_t *credp) 192 { 193 dev_info_t *dip = hubd_get_dip(dev); 194 195 return (usba_hubdi_close(dip, dev, flag, otyp, credp)); 196 } 197 198 199 static int 200 hubd_ioctl(dev_t dev, int cmd, intptr_t arg, int mode, 201 cred_t *credp, int *rvalp) 202 { 203 dev_info_t *dip = hubd_get_dip(dev); 204 205 return (usba_hubdi_ioctl(dip, dev, cmd, arg, mode, 206 credp, rvalp)); 207 }