urb.c

来自「linux 内核源代码」· C语言 代码 · 共 606 行 · 第 1/2 页

C
606
字号
	 * and don't need to duplicate tests	 */	xfertype = usb_endpoint_type(&ep->desc);	if (xfertype == USB_ENDPOINT_XFER_CONTROL) {		struct usb_ctrlrequest *setup =				(struct usb_ctrlrequest *) urb->setup_packet;		if (!setup)			return -ENOEXEC;		is_out = !(setup->bRequestType & USB_DIR_IN) ||				!setup->wLength;	} else {		is_out = usb_endpoint_dir_out(&ep->desc);	}	/* Cache the direction for later use */	urb->transfer_flags = (urb->transfer_flags & ~URB_DIR_MASK) |			(is_out ? URB_DIR_OUT : URB_DIR_IN);	if (xfertype != USB_ENDPOINT_XFER_CONTROL &&			dev->state < USB_STATE_CONFIGURED)		return -ENODEV;	max = le16_to_cpu(ep->desc.wMaxPacketSize);	if (max <= 0) {		dev_dbg(&dev->dev,			"bogus endpoint ep%d%s in %s (bad maxpacket %d)\n",			usb_endpoint_num(&ep->desc), is_out ? "out" : "in",			__FUNCTION__, max);		return -EMSGSIZE;	}	/* periodic transfers limit size per frame/uframe,	 * but drivers only control those sizes for ISO.	 * while we're checking, initialize return status.	 */	if (xfertype == USB_ENDPOINT_XFER_ISOC) {		int	n, len;		/* "high bandwidth" mode, 1-3 packets/uframe? */		if (dev->speed == USB_SPEED_HIGH) {			int	mult = 1 + ((max >> 11) & 0x03);			max &= 0x07ff;			max *= mult;		}		if (urb->number_of_packets <= 0)		    			return -EINVAL;		for (n = 0; n < urb->number_of_packets; n++) {			len = urb->iso_frame_desc[n].length;			if (len < 0 || len > max) 				return -EMSGSIZE;			urb->iso_frame_desc[n].status = -EXDEV;			urb->iso_frame_desc[n].actual_length = 0;		}	}	/* the I/O buffer must be mapped/unmapped, except when length=0 */	if (urb->transfer_buffer_length < 0)		return -EMSGSIZE;#ifdef DEBUG	/* stuff that drivers shouldn't do, but which shouldn't	 * cause problems in HCDs if they get it wrong.	 */	{	unsigned int	orig_flags = urb->transfer_flags;	unsigned int	allowed;	/* enforce simple/standard policy */	allowed = (URB_NO_TRANSFER_DMA_MAP | URB_NO_SETUP_DMA_MAP |			URB_NO_INTERRUPT | URB_DIR_MASK | URB_FREE_BUFFER);	switch (xfertype) {	case USB_ENDPOINT_XFER_BULK:		if (is_out)			allowed |= URB_ZERO_PACKET;		/* FALLTHROUGH */	case USB_ENDPOINT_XFER_CONTROL:		allowed |= URB_NO_FSBR;	/* only affects UHCI */		/* FALLTHROUGH */	default:			/* all non-iso endpoints */		if (!is_out)			allowed |= URB_SHORT_NOT_OK;		break;	case USB_ENDPOINT_XFER_ISOC:		allowed |= URB_ISO_ASAP;		break;	}	urb->transfer_flags &= allowed;	/* fail if submitter gave bogus flags */	if (urb->transfer_flags != orig_flags) {		err("BOGUS urb flags, %x --> %x",			orig_flags, urb->transfer_flags);		return -EINVAL;	}	}#endif	/*	 * Force periodic transfer intervals to be legal values that are	 * a power of two (so HCDs don't need to).	 *	 * FIXME want bus->{intr,iso}_sched_horizon values here.  Each HC	 * supports different values... this uses EHCI/UHCI defaults (and	 * EHCI can use smaller non-default values).	 */	switch (xfertype) {	case USB_ENDPOINT_XFER_ISOC:	case USB_ENDPOINT_XFER_INT:		/* too small? */		if (urb->interval <= 0)			return -EINVAL;		/* too big? */		switch (dev->speed) {		case USB_SPEED_HIGH:	/* units are microframes */			// NOTE usb handles 2^15			if (urb->interval > (1024 * 8))				urb->interval = 1024 * 8;			max = 1024 * 8;			break;		case USB_SPEED_FULL:	/* units are frames/msec */		case USB_SPEED_LOW:			if (xfertype == USB_ENDPOINT_XFER_INT) {				if (urb->interval > 255)					return -EINVAL;				// NOTE ohci only handles up to 32				max = 128;			} else {				if (urb->interval > 1024)					urb->interval = 1024;				// NOTE usb and ohci handle up to 2^15				max = 1024;			}			break;		default:			return -EINVAL;		}		/* Round down to a power of 2, no more than max */		urb->interval = min(max, 1 << ilog2(urb->interval));	}	return usb_hcd_submit_urb(urb, mem_flags);}/*-------------------------------------------------------------------*//** * usb_unlink_urb - abort/cancel a transfer request for an endpoint * @urb: pointer to urb describing a previously submitted request, *	may be NULL * * This routine cancels an in-progress request.  URBs complete only once * per submission, and may be canceled only once per submission. * Successful cancellation means termination of @urb will be expedited * and the completion handler will be called with a status code * indicating that the request has been canceled (rather than any other * code). * * This request is always asynchronous.  Success is indicated by * returning -EINPROGRESS, at which time the URB will probably not yet * have been given back to the device driver.  When it is eventually * called, the completion function will see @urb->status == -ECONNRESET. * Failure is indicated by usb_unlink_urb() returning any other value. * Unlinking will fail when @urb is not currently "linked" (i.e., it was * never submitted, or it was unlinked before, or the hardware is already * finished with it), even if the completion handler has not yet run. * * Unlinking and Endpoint Queues: * * [The behaviors and guarantees described below do not apply to virtual * root hubs but only to endpoint queues for physical USB devices.] * * Host Controller Drivers (HCDs) place all the URBs for a particular * endpoint in a queue.  Normally the queue advances as the controller * hardware processes each request.  But when an URB terminates with an * error its queue generally stops (see below), at least until that URB's * completion routine returns.  It is guaranteed that a stopped queue * will not restart until all its unlinked URBs have been fully retired, * with their completion routines run, even if that's not until some time * after the original completion handler returns.  The same behavior and * guarantee apply when an URB terminates because it was unlinked. * * Bulk and interrupt endpoint queues are guaranteed to stop whenever an * URB terminates with any sort of error, including -ECONNRESET, -ENOENT, * and -EREMOTEIO.  Control endpoint queues behave the same way except * that they are not guaranteed to stop for -EREMOTEIO errors.  Queues * for isochronous endpoints are treated differently, because they must * advance at fixed rates.  Such queues do not stop when an URB * encounters an error or is unlinked.  An unlinked isochronous URB may * leave a gap in the stream of packets; it is undefined whether such * gaps can be filled in. * * Note that early termination of an URB because a short packet was * received will generate a -EREMOTEIO error if and only if the * URB_SHORT_NOT_OK flag is set.  By setting this flag, USB device * drivers can build deep queues for large or complex bulk transfers * and clean them up reliably after any sort of aborted transfer by * unlinking all pending URBs at the first fault. * * When a control URB terminates with an error other than -EREMOTEIO, it * is quite likely that the status stage of the transfer will not take * place. */int usb_unlink_urb(struct urb *urb){	if (!urb)		return -EINVAL;	if (!urb->dev)		return -ENODEV;	if (!urb->ep)		return -EIDRM;	return usb_hcd_unlink_urb(urb, -ECONNRESET);}/** * usb_kill_urb - cancel a transfer request and wait for it to finish * @urb: pointer to URB describing a previously submitted request, *	may be NULL * * This routine cancels an in-progress request.  It is guaranteed that * upon return all completion handlers will have finished and the URB * will be totally idle and available for reuse.  These features make * this an ideal way to stop I/O in a disconnect() callback or close() * function.  If the request has not already finished or been unlinked * the completion handler will see urb->status == -ENOENT. * * While the routine is running, attempts to resubmit the URB will fail * with error -EPERM.  Thus even if the URB's completion handler always * tries to resubmit, it will not succeed and the URB will become idle. * * This routine may not be used in an interrupt context (such as a bottom * half or a completion handler), or when holding a spinlock, or in other * situations where the caller can't schedule(). */void usb_kill_urb(struct urb *urb){	static DEFINE_MUTEX(reject_mutex);	might_sleep();	if (!(urb && urb->dev && urb->ep))		return;	mutex_lock(&reject_mutex);	++urb->reject;	mutex_unlock(&reject_mutex);	usb_hcd_unlink_urb(urb, -ENOENT);	wait_event(usb_kill_urb_queue, atomic_read(&urb->use_count) == 0);	mutex_lock(&reject_mutex);	--urb->reject;	mutex_unlock(&reject_mutex);}/** * usb_kill_anchored_urbs - cancel transfer requests en masse * @anchor: anchor the requests are bound to * * this allows all outstanding URBs to be killed starting * from the back of the queue */void usb_kill_anchored_urbs(struct usb_anchor *anchor){	struct urb *victim;	spin_lock_irq(&anchor->lock);	while (!list_empty(&anchor->urb_list)) {		victim = list_entry(anchor->urb_list.prev, struct urb, anchor_list);		/* we must make sure the URB isn't freed before we kill it*/		usb_get_urb(victim);		spin_unlock_irq(&anchor->lock);		/* this will unanchor the URB */		usb_kill_urb(victim);		usb_put_urb(victim);		spin_lock_irq(&anchor->lock);	}	spin_unlock_irq(&anchor->lock);}EXPORT_SYMBOL_GPL(usb_kill_anchored_urbs);/** * usb_wait_anchor_empty_timeout - wait for an anchor to be unused * @anchor: the anchor you want to become unused * @timeout: how long you are willing to wait in milliseconds * * Call this is you want to be sure all an anchor's * URBs have finished */int usb_wait_anchor_empty_timeout(struct usb_anchor *anchor,				  unsigned int timeout){	return wait_event_timeout(anchor->wait, list_empty(&anchor->urb_list),				  msecs_to_jiffies(timeout));}EXPORT_SYMBOL_GPL(usb_wait_anchor_empty_timeout);EXPORT_SYMBOL(usb_init_urb);EXPORT_SYMBOL(usb_alloc_urb);EXPORT_SYMBOL(usb_free_urb);EXPORT_SYMBOL(usb_get_urb);EXPORT_SYMBOL(usb_submit_urb);EXPORT_SYMBOL(usb_unlink_urb);EXPORT_SYMBOL(usb_kill_urb);

⌨️ 快捷键说明

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