⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 hub.c

📁 优龙2410linux2.6.8内核源代码
💻 C
📖 第 1 页 / 共 4 页
字号:
	/* power budgeting mostly matters with bus-powered hubs,	 * and battery-powered root hubs (may provide just 8 mA).	 */	ret = usb_get_status(hdev, USB_RECIP_DEVICE, 0, &hubstatus);	if (ret < 0) {		message = "can't get hub status";		goto fail;	}	cpu_to_le16s(&hubstatus);	if ((hubstatus & (1 << USB_DEVICE_SELF_POWERED)) == 0) {		dev_dbg(hub_dev, "hub controller current requirement: %dmA\n",			hub->descriptor->bHubContrCurrent);		hub->power_budget = (501 - hub->descriptor->bHubContrCurrent)					/ 2;		dev_dbg(hub_dev, "%dmA bus power budget for children\n",			hub->power_budget * 2);	}	ret = hub_hub_status(hub, &hubstatus, &hubchange);	if (ret < 0) {		message = "can't get hub status";		goto fail;	}	/* local power status reports aren't always correct */	if (hdev->actconfig->desc.bmAttributes & USB_CONFIG_ATT_SELFPOWER)		dev_dbg(hub_dev, "local power source is %s\n",			(hubstatus & HUB_STATUS_LOCAL_POWER)			? "lost (inactive)" : "good");	if ((hub->descriptor->wHubCharacteristics & HUB_CHAR_OCPM) == 0)		dev_dbg(hub_dev, "%sover-current condition exists\n",			(hubstatus & HUB_STATUS_OVERCURRENT) ? "" : "no ");	/* Start the interrupt endpoint */	pipe = usb_rcvintpipe(hdev, endpoint->bEndpointAddress);	maxp = usb_maxpacket(hdev, pipe, usb_pipeout(pipe));	if (maxp > sizeof(*hub->buffer))		maxp = sizeof(*hub->buffer);	hub->urb = usb_alloc_urb(0, GFP_KERNEL);	if (!hub->urb) {		message = "couldn't allocate interrupt urb";		ret = -ENOMEM;		goto fail;	}	usb_fill_int_urb(hub->urb, hdev, pipe, *hub->buffer, maxp, hub_irq,		hub, endpoint->bInterval);	hub->urb->transfer_dma = hub->buffer_dma;	hub->urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;	ret = usb_submit_urb(hub->urb, GFP_KERNEL);	if (ret) {		message = "couldn't submit status urb";		goto fail;	}	/* Wake up khubd */	wake_up(&khubd_wait);	/* maybe start cycling the hub leds */	if (hub->has_indicators && blinkenlights) {		set_port_led(hdev, 1, HUB_LED_GREEN);		hub->indicator [0] = INDICATOR_CYCLE;		schedule_delayed_work(&hub->leds, LED_CYCLE_PERIOD);	}	hub_power_on(hub);	return 0;fail:	dev_err (hub_dev, "config failed, %s (err %d)\n",			message, ret);	/* hub_disconnect() frees urb and descriptor */	return ret;}static unsigned highspeed_hubs;static void hub_disconnect(struct usb_interface *intf){	struct usb_hub *hub = usb_get_intfdata (intf);	struct usb_device *hdev;	if (!hub)		return;	hdev = hub->hdev;	if (hdev->speed == USB_SPEED_HIGH)		highspeed_hubs--;	usb_set_intfdata (intf, NULL);	if (hub->urb) {		usb_kill_urb(hub->urb);		usb_free_urb(hub->urb);		hub->urb = NULL;	}	spin_lock_irq(&hub_event_lock);	list_del_init(&hub->event_list);	spin_unlock_irq(&hub_event_lock);	/* assuming we used keventd, it must quiesce too */	if (hub->has_indicators)		cancel_delayed_work (&hub->leds);	if (hub->has_indicators || hub->tt.hub)		flush_scheduled_work ();	if (hub->descriptor) {		kfree(hub->descriptor);		hub->descriptor = NULL;	}	if (hub->status) {		kfree(hub->status);		hub->status = NULL;	}	if (hub->buffer) {		usb_buffer_free(hdev, sizeof(*hub->buffer), hub->buffer,				hub->buffer_dma);		hub->buffer = NULL;	}	/* Free the memory */	kfree(hub);}static int hub_probe(struct usb_interface *intf, const struct usb_device_id *id){	struct usb_host_interface *desc;	struct usb_endpoint_descriptor *endpoint;	struct usb_device *hdev;	struct usb_hub *hub;	struct device *hub_dev;	desc = intf->cur_altsetting;	hdev = interface_to_usbdev(intf);	hub_dev = &intf->dev;	/* Some hubs have a subclass of 1, which AFAICT according to the */	/*  specs is not defined, but it works */	if ((desc->desc.bInterfaceSubClass != 0) &&	    (desc->desc.bInterfaceSubClass != 1)) {descriptor_error:		dev_err (hub_dev, "bad descriptor, ignoring hub\n");		return -EIO;	}	/* Multiple endpoints? What kind of mutant ninja-hub is this? */	if (desc->desc.bNumEndpoints != 1)		goto descriptor_error;	endpoint = &desc->endpoint[0].desc;	/* Output endpoint? Curiouser and curiouser.. */	if (!(endpoint->bEndpointAddress & USB_DIR_IN))		goto descriptor_error;	/* If it's not an interrupt endpoint, we'd better punt! */	if ((endpoint->bmAttributes & USB_ENDPOINT_XFERTYPE_MASK)			!= USB_ENDPOINT_XFER_INT)		goto descriptor_error;	/* We found a hub */	dev_info (hub_dev, "USB hub found\n");	hub = kmalloc(sizeof(*hub), GFP_KERNEL);	if (!hub) {		dev_dbg (hub_dev, "couldn't kmalloc hub struct\n");		return -ENOMEM;	}	memset(hub, 0, sizeof(*hub));	INIT_LIST_HEAD(&hub->event_list);	hub->intf = intf;	hub->hdev = hdev;	INIT_WORK(&hub->leds, led_work, hub);	usb_set_intfdata (intf, hub);	if (hdev->speed == USB_SPEED_HIGH)		highspeed_hubs++;	if (hub_configure(hub, endpoint) >= 0)		return 0;	hub_disconnect (intf);	return -ENODEV;}static inthub_ioctl(struct usb_interface *intf, unsigned int code, void *user_data){	struct usb_device *hdev = interface_to_usbdev (intf);	/* assert ifno == 0 (part of hub spec) */	switch (code) {	case USBDEVFS_HUB_PORTINFO: {		struct usbdevfs_hub_portinfo *info = user_data;		unsigned long flags;		int i;		spin_lock_irqsave(&hub_event_lock, flags);		if (hdev->devnum <= 0)			info->nports = 0;		else {			info->nports = hdev->maxchild;			for (i = 0; i < info->nports; i++) {				if (hdev->children[i] == NULL)					info->port[i] = 0;				else					info->port[i] =						hdev->children[i]->devnum;			}		}		spin_unlock_irqrestore(&hub_event_lock, flags);		return info->nports + 1;		}	default:		return -ENOSYS;	}}static int hub_reset(struct usb_hub *hub){	struct usb_device *hdev = hub->hdev;	int i;	/* Disconnect any attached devices */	for (i = 0; i < hub->descriptor->bNbrPorts; i++) {		if (hdev->children[i])			usb_disconnect(&hdev->children[i]);	}	/* Attempt to reset the hub */	if (hub->urb)		usb_kill_urb(hub->urb);	else		return -1;	if (usb_reset_device(hdev))		return -1;	hub->urb->dev = hdev;                                                    	if (usb_submit_urb(hub->urb, GFP_KERNEL))		return -1;	hub_power_on(hub);	return 0;}/* FIXME!  This routine should be subsumed into hub_reset */static void hub_start_disconnect(struct usb_device *hdev){	struct usb_device *parent = hdev->parent;	int i;	/* Find the device pointer to disconnect */	if (parent) {		for (i = 0; i < parent->maxchild; i++) {			if (parent->children[i] == hdev) {				usb_disconnect(&parent->children[i]);				return;			}		}	}	dev_err(&hdev->dev, "cannot disconnect hub!\n");}static void recursively_mark_NOTATTACHED(struct usb_device *udev){	int i;	for (i = 0; i < udev->maxchild; ++i) {		if (udev->children[i])			recursively_mark_NOTATTACHED(udev->children[i]);	}	udev->state = USB_STATE_NOTATTACHED;}/** * usb_set_device_state - change a device's current state (usbcore-internal) * @udev: pointer to device whose state should be changed * @new_state: new state value to be stored * * udev->state is _not_ protected by the udev->serialize semaphore.  This * is so that devices can be marked as disconnected as soon as possible, * without having to wait for the semaphore to be released.  Instead, * changes to the state must be protected by the device_state_lock spinlock. * * Once a device has been added to the device tree, all changes to its state * should be made using this routine.  The state should _not_ be set directly. * * If udev->state is already USB_STATE_NOTATTACHED then no change is made. * Otherwise udev->state is set to new_state, and if new_state is * USB_STATE_NOTATTACHED then all of udev's descendant's states are also set * to USB_STATE_NOTATTACHED. */void usb_set_device_state(struct usb_device *udev,		enum usb_device_state new_state){	unsigned long flags;	spin_lock_irqsave(&device_state_lock, flags);	if (udev->state == USB_STATE_NOTATTACHED)		;	/* do nothing */	else if (new_state != USB_STATE_NOTATTACHED)		udev->state = new_state;	else		recursively_mark_NOTATTACHED(udev);	spin_unlock_irqrestore(&device_state_lock, flags);}static void choose_address(struct usb_device *udev){	int		devnum;	struct usb_bus	*bus = udev->bus;	/* If khubd ever becomes multithreaded, this will need a lock */	/* Try to allocate the next devnum beginning at bus->devnum_next. */	devnum = find_next_zero_bit(bus->devmap.devicemap, 128,			bus->devnum_next);	if (devnum >= 128)		devnum = find_next_zero_bit(bus->devmap.devicemap, 128, 1);	bus->devnum_next = ( devnum >= 127 ? 1 : devnum + 1);	if (devnum < 128) {		set_bit(devnum, bus->devmap.devicemap);		udev->devnum = devnum;	}}static void release_address(struct usb_device *udev){	if (udev->devnum > 0) {		clear_bit(udev->devnum, udev->bus->devmap.devicemap);		udev->devnum = -1;	}}/** * usb_disconnect - disconnect a device (usbcore-internal) * @pdev: pointer to device being disconnected * Context: !in_interrupt () * * Something got disconnected. Get rid of it, and all of its children. * If *pdev is a normal device then the parent hub should be locked. * If *pdev is a root hub then this routine will acquire the * usb_bus_list_lock on behalf of the caller. * * Only hub drivers (including virtual root hub drivers for host * controllers) should ever call this. * * This call is synchronous, and may not be used in an interrupt context. */void usb_disconnect(struct usb_device **pdev){	struct usb_device	*udev = *pdev;	int			i;	if (!udev) {		pr_debug ("%s nodev\n", __FUNCTION__);		return;	}	/* mark the device as inactive, so any further urb submissions for	 * this device will fail.	 */	usb_set_device_state(udev, USB_STATE_NOTATTACHED);	/* lock the bus list on behalf of HCDs unregistering their root hubs */	if (!udev->parent)		down(&usb_bus_list_lock);	down(&udev->serialize);	dev_info (&udev->dev, "USB disconnect, address %d\n", udev->devnum);	/* Free up all the children before we remove this device */	for (i = 0; i < USB_MAXCHILDREN; i++) {		if (udev->children[i])			usb_disconnect(&udev->children[i]);	}	/* deallocate hcd/hardware state ... nuking all pending urbs and	 * cleaning up all state associated with the current configuration	 */	usb_disable_device(udev, 0);	/* Free the device number, remove the /proc/bus/usb entry and	 * the sysfs attributes, and delete the parent's children[]	 * (or root_hub) pointer.	 */	dev_dbg (&udev->dev, "unregistering device\n");	release_address(udev);	usbfs_remove_device(udev);	usb_remove_sysfs_dev_files(udev);	/* Avoid races with recursively_mark_NOTATTACHED() */	spin_lock_irq(&device_state_lock);	*pdev = NULL;	spin_unlock_irq(&device_state_lock);	up(&udev->serialize);	if (!udev->parent)		up(&usb_bus_list_lock);	device_unregister(&udev->dev);}static int choose_configuration(struct usb_device *udev){	int c, i;	/* NOTE: this should interact with hub power budgeting */	c = udev->config[0].desc.bConfigurationValue;	if (udev->descriptor.bNumConfigurations != 1) {		for (i = 0; i < udev->descriptor.bNumConfigurations; i++) {			struct usb_interface_descriptor	*desc;			/* heuristic:  Linux is more likely to have class			 * drivers, so avoid vendor-specific interfaces.			 */			desc = &udev->config[i].intf_cache[0]					->altsetting->desc;			if (desc->bInterfaceClass == USB_CLASS_VENDOR_SPEC)				continue;			/* COMM/2/all is CDC ACM, except 0xff is MSFT RNDIS */			if (desc->bInterfaceClass == USB_CLASS_COMM					&& desc->bInterfaceSubClass == 2					&& desc->bInterfaceProtocol == 0xff)				continue;			c = udev->config[i].desc.bConfigurationValue;			break;		}		dev_info(&udev->dev,			"configuration #%d chosen from %d choices\n",			c, udev->descriptor.bNumConfigurations);	}	return c;}#ifdef DEBUGstatic void show_string(struct usb_device *udev, char *id, int index){	char *buf;	if (!index)		return;	if (!(buf = kmalloc(256, GFP_KERNEL)))		return;	if (usb_string(udev, index, buf, 256) > 0)		dev_printk(KERN_INFO, &udev->dev, "%s: %s\n", id, buf);	kfree(buf);}#elsestatic inline void show_string(struct usb_device *udev, char *id, int index){}#endif/** * usb_new_device - perform initial device setup (usbcore-internal) * @udev: newly addressed device (in ADDRESS state) * * This is called with devices which have been enumerated, but not yet * configured.  The device descriptor is available, but not descriptors * for any device configuration.  The caller must have locked udev and * either the parent hub (if udev is a normal device) or else the * usb_bus_list_lock (if udev is a root hub).  The parent's pointer to * udev has already been installed, but udev is not yet visible through * sysfs or other filesystem code. * * Returns 0 for success (device is configured and listed, with its * interfaces, in sysfs); else a negative errno value. * * This call is synchronous, and may not be used in an interrupt context. * * Only the hub driver should ever call this; root hub registration * uses it indirectly. */int usb_new_device(struct usb_device *udev){	int err;	int c;	err = usb_get_configuration(udev);	if (err < 0) {		dev_err(&udev->dev, "can't read configurations, error %d\n",			err);		goto fail;	}	/* Tell the world! */	dev_dbg(&udev->dev, "new device strings: Mfr=%d, Product=%d, "			"SerialNumber=%d\n",			udev->descriptor.iManufacturer,			udev->descriptor.iProduct,			udev->descriptor.iSerialNumber);	if (udev->descriptor.iProduct)		show_string(udev, "Product",				udev->descriptor.iProduct);	if (udev->descriptor.iManufacturer)		show_string(udev, "Manufacturer",				udev->descriptor.iManufacturer);	if (udev->descriptor.iSerialNumber)		show_string(udev, "SerialNumber",				udev->descriptor.iSerialNumber);	/* put device-specific files into sysfs */	err = device_add (&udev->dev);	if (err) {		dev_err(&udev->dev, "can't device_add, error %d\n", err);		goto fail;	}	usb_create_sysfs_dev_files (udev);	/* choose and set the configuration. that registers the interfaces	 * with the driver core, and lets usb device drivers bind to them.	 */	c = choose_configuration(udev);	if (c < 0)		dev_warn(&udev->dev,				"can't choose an initial configuration\n");	else {		err = usb_set_configuration(udev, c);		if (err) {			dev_err(&udev->dev, "can't set config #%d, error %d\n",

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -