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

📄 cdc_ether.c

📁 omap3 linux 2.6 用nocc去除了冗余代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * CDC Ethernet based networking peripherals * Copyright (C) 2003-2005 by David Brownell * Copyright (C) 2006 by Ole Andre Vadla Ravnas (ActiveSync) * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */// #define	DEBUG			// error path messages, extra info// #define	VERBOSE			// more; success messages#include <linux/module.h>#include <linux/init.h>#include <linux/netdevice.h>#include <linux/etherdevice.h>#include <linux/ctype.h>#include <linux/ethtool.h>#include <linux/workqueue.h>#include <linux/mii.h>#include <linux/usb.h>#include <linux/usb/cdc.h>#include "usbnet.h"#define is_rndis(desc)		0#define is_activesync(desc)	0/* * probes control interface, claims data interface, collects the bulk * endpoints, activates data interface (if needed), maybe sets MTU. * all pure cdc, except for certain firmware workarounds, and knowing * that rndis uses one different rule. */int usbnet_generic_cdc_bind(struct usbnet *dev, struct usb_interface *intf){	u8				*buf = intf->cur_altsetting->extra;	int				len = intf->cur_altsetting->extralen;	struct usb_interface_descriptor	*d;	struct cdc_state		*info = (void *) &dev->data;	int				status;	int				rndis;	struct usb_driver		*driver = driver_of(intf);	if (sizeof dev->data < sizeof *info)		return -EDOM;	/* expect strict spec conformance for the descriptors, but	 * cope with firmware which stores them in the wrong place	 */	if (len == 0 && dev->udev->actconfig->extralen) {		/* Motorola SB4100 (and others: Brad Hards says it's		 * from a Broadcom design) put CDC descriptors here		 */		buf = dev->udev->actconfig->extra;		len = dev->udev->actconfig->extralen;		if (len)			dev_dbg(&intf->dev,				"CDC descriptors on config\n");	}	/* Maybe CDC descriptors are after the endpoint?  This bug has	 * been seen on some 2Wire Inc RNDIS-ish products.	 */	if (len == 0) {		struct usb_host_endpoint	*hep;		hep = intf->cur_altsetting->endpoint;		if (hep) {			buf = hep->extra;			len = hep->extralen;		}		if (len)			dev_dbg(&intf->dev,				"CDC descriptors on endpoint\n");	}	/* this assumes that if there's a non-RNDIS vendor variant	 * of cdc-acm, it'll fail RNDIS requests cleanly.	 */	rndis = is_rndis(&intf->cur_altsetting->desc)		|| is_activesync(&intf->cur_altsetting->desc);	memset(info, 0, sizeof *info);	info->control = intf;	while (len > 3) {		if (buf [1] != USB_DT_CS_INTERFACE)			goto next_desc;		/* use bDescriptorSubType to identify the CDC descriptors.		 * We expect devices with CDC header and union descriptors.		 * For CDC Ethernet we need the ethernet descriptor.		 * For RNDIS, ignore two (pointless) CDC modem descriptors		 * in favor of a complicated OID-based RPC scheme doing what		 * CDC Ethernet achieves with a simple descriptor.		 */		switch (buf [2]) {		case USB_CDC_HEADER_TYPE:			if (info->header) {				dev_dbg(&intf->dev, "extra CDC header\n");				goto bad_desc;			}			info->header = (void *) buf;			if (info->header->bLength != sizeof *info->header) {				dev_dbg(&intf->dev, "CDC header len %u\n",					info->header->bLength);				goto bad_desc;			}			break;		case USB_CDC_ACM_TYPE:			/* paranoia:  disambiguate a "real" vendor-specific			 * modem interface from an RNDIS non-modem.			 */			if (rndis) {				struct usb_cdc_acm_descriptor *acm;				acm = (void *) buf;				if (acm->bmCapabilities) {					dev_dbg(&intf->dev,						"ACM capabilities %02x, "						"not really RNDIS?\n",						acm->bmCapabilities);					goto bad_desc;				}			}			break;		case USB_CDC_UNION_TYPE:			if (info->u) {				dev_dbg(&intf->dev, "extra CDC union\n");				goto bad_desc;			}			info->u = (void *) buf;			if (info->u->bLength != sizeof *info->u) {				dev_dbg(&intf->dev, "CDC union len %u\n",					info->u->bLength);				goto bad_desc;			}			/* we need a master/control interface (what we're			 * probed with) and a slave/data interface; union			 * descriptors sort this all out.			 */			info->control = usb_ifnum_to_if(dev->udev,						info->u->bMasterInterface0);			info->data = usb_ifnum_to_if(dev->udev,						info->u->bSlaveInterface0);			if (!info->control || !info->data) {				dev_dbg(&intf->dev,					"master #%u/%p slave #%u/%p\n",					info->u->bMasterInterface0,					info->control,					info->u->bSlaveInterface0,					info->data);				goto bad_desc;			}			if (info->control != intf) {				dev_dbg(&intf->dev, "bogus CDC Union\n");				/* Ambit USB Cable Modem (and maybe others)				 * interchanges master and slave interface.				 */				if (info->data == intf) {					info->data = info->control;					info->control = intf;				} else					goto bad_desc;			}			/* a data interface altsetting does the real i/o */			d = &info->data->cur_altsetting->desc;			if (d->bInterfaceClass != USB_CLASS_CDC_DATA) {				dev_dbg(&intf->dev, "slave class %u\n",					d->bInterfaceClass);				goto bad_desc;			}			break;		case USB_CDC_ETHERNET_TYPE:			if (info->ether) {				dev_dbg(&intf->dev, "extra CDC ether\n");				goto bad_desc;			}			info->ether = (void *) buf;			if (info->ether->bLength != sizeof *info->ether) {				dev_dbg(&intf->dev, "CDC ether len %u\n",					info->ether->bLength);				goto bad_desc;			}			dev->hard_mtu = le16_to_cpu(						info->ether->wMaxSegmentSize);			/* because of Zaurus, we may be ignoring the host			 * side link address we were given.			 */			break;		}next_desc:		len -= buf [0];	/* bLength */		buf += buf [0];	}	/* Microsoft ActiveSync based RNDIS devices lack the CDC descriptors,	 * so we'll hard-wire the interfaces and not check for descriptors.	 */	if (is_activesync(&intf->cur_altsetting->desc) && !info->u) {		info->control = usb_ifnum_to_if(dev->udev, 0);		info->data = usb_ifnum_to_if(dev->udev, 1);		if (!info->control || !info->data) {			dev_dbg(&intf->dev,				"activesync: master #0/%p slave #1/%p\n",				info->control,				info->data);			goto bad_desc;		}	} else if (!info->header || !info->u || (!rndis && !info->ether)) {		dev_dbg(&intf->dev, "missing cdc %s%s%sdescriptor\n",			info->header ? "" : "header ",			info->u ? "" : "union ",			info->ether ? "" : "ether ");		goto bad_desc;	}	/* claim data interface and set it up ... with side effects.	 * network traffic can't flow until an altsetting is enabled.	 */	status = usb_driver_claim_interface(driver, info->data, dev);	if (status < 0)		return status;	status = usbnet_get_endpoints(dev, info->data);	if (status < 0) {		/* ensure immediate exit from usbnet_disconnect */		usb_set_intfdata(info->data, NULL);		usb_driver_release_interface(driver, info->data);		return status;	}	/* status endpoint: optional for CDC Ethernet, not RNDIS (or ACM) */	dev->status = NULL;	if (info->control->cur_altsetting->desc.bNumEndpoints == 1) {		struct usb_endpoint_descriptor	*desc;		dev->status = &info->control->cur_altsetting->endpoint [0];		desc = &dev->status->desc;		if (!usb_endpoint_is_int_in(desc)				|| (le16_to_cpu(desc->wMaxPacketSize)					< sizeof(struct usb_cdc_notification))				|| !desc->bInterval) {			dev_dbg(&intf->dev, "bad notification endpoint\n");			dev->status = NULL;		}	}	if (rndis && !dev->status) {		dev_dbg(&intf->dev, "missing RNDIS status endpoint\n");		usb_set_intfdata(info->data, NULL);		usb_driver_release_interface(driver, info->data);		return -ENODEV;	}	return 0;bad_desc:	dev_info(&dev->udev->dev, "bad CDC descriptors\n");	return -ENODEV;}EXPORT_SYMBOL_GPL(usbnet_generic_cdc_bind);void usbnet_cdc_unbind(struct usbnet *dev, struct usb_interface *intf){	struct cdc_state		*info = (void *) &dev->data;	struct usb_driver		*driver = driver_of(intf);	/* disconnect master --> disconnect slave */	if (intf == info->control && info->data) {		/* ensure immediate exit from usbnet_disconnect */

⌨️ 快捷键说明

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