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

📄 hpusbscsi.c

📁 S3C2440ARM9开发板的USB驱动程序
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * hpusbscsi * (C) Copyright 2001 Oliver Neukum  * Sponsored by the Linux Usb Project * Large parts based on or taken from code by John Fremlin and Matt Dharm *  * This driver is known to work with the following scanners (VID, PID) *    (0x03f0, 0x0701)  HP 53xx  *    (0x03f0, 0x0801)  HP 7400  *    (0x0638, 0x026a)  Minolta Scan Dual II *    (0x0686, 0x4004)  Minolta Elite II * To load with full debugging load with "insmod hpusbscsi debug=2" *  * 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., 675 Mass Ave, Cambridge, MA 02139, USA. * * Contributors: *   Oliver Neukum *   John Fremlin *   Matt Dharm *   . *   . *   Timothy Jedlicka <bonzo@lucent.com> * * History * * 22-Apr-2002 * * - Added Elite II scanner - bonzo * - Cleaned up the debug statements and made them optional at load time - bonzo * * 20020618 * * - Confirm to stupid 2.4 rules on io_request_lock * */#include <linux/module.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/signal.h>#include <linux/errno.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/spinlock.h>#include <linux/smp_lock.h>#include <linux/usb.h>#include <asm/atomic.h>#include <linux/blk.h>#include "../scsi/scsi.h"#include "../scsi/hosts.h"#include "../scsi/sd.h"#include "hpusbscsi.h"static char *states[]={"FREE", "BEGINNING", "WORKING", "ERROR", "WAIT", "PREMATURE"};/* DEBUG related parts */#define HPUSBSCSI_DEBUG#ifdef HPUSBSCSI_DEBUG#  define PDEBUG(level, fmt, args...) \          if (debug >= (level)) info("[" __PRETTY_FUNCTION__ ":%d] " fmt, __LINE__ , \                 ## args)#else#  define PDEBUG(level, fmt, args...) do {} while(0)#endif/* 0=no debug messages * 1=everything but trace states * 2=trace states */static int debug; /* = 0 */MODULE_PARM(debug, "i");MODULE_PARM_DESC(debug, "Debug level: 0=none, 1=no trace states, 2=trace states");/* global variables */struct list_head hpusbscsi_devices;//LIST_HEAD(hpusbscsi_devices);/* USB related parts */static void *hpusbscsi_usb_probe (struct usb_device *dev, unsigned int interface,		     const struct usb_device_id *id){	struct hpusbscsi *new;	struct usb_interface_descriptor *altsetting =		&(dev->actconfig->interface[interface].altsetting[0]);	int i, result;	/* basic check */	if (altsetting->bNumEndpoints != 3) {		printk (KERN_ERR "Wrong number of endpoints\n");		return NULL;	}	/* descriptor allocation */	new =		(struct hpusbscsi *) kmalloc (sizeof (struct hpusbscsi),					      GFP_KERNEL);	if (new == NULL)		return NULL;	PDEBUG (1, "Allocated memory");	memset (new, 0, sizeof (struct hpusbscsi));	spin_lock_init (&new->dataurb.lock);	spin_lock_init (&new->controlurb.lock);	new->dev = dev;	init_waitqueue_head (&new->pending);	init_waitqueue_head (&new->deathrow);	INIT_LIST_HEAD (&new->lh);	/* finding endpoints */	for (i = 0; i < altsetting->bNumEndpoints; i++) {		if (		    (altsetting->endpoint[i].		     bmAttributes & USB_ENDPOINT_XFERTYPE_MASK) ==		    USB_ENDPOINT_XFER_BULK) {			if (altsetting->endpoint[i].			    bEndpointAddress & USB_DIR_IN) {				new->ep_in =					altsetting->endpoint[i].					bEndpointAddress &					USB_ENDPOINT_NUMBER_MASK;			} else {				new->ep_out =					altsetting->endpoint[i].					bEndpointAddress &					USB_ENDPOINT_NUMBER_MASK;			}		} else {			new->ep_int =				altsetting->endpoint[i].				bEndpointAddress & USB_ENDPOINT_NUMBER_MASK;			new->interrupt_interval= altsetting->endpoint[i].bInterval;		}	}	/* USB initialisation magic for the simple case */	result = usb_set_interface (dev, altsetting->bInterfaceNumber, 0);	switch (result) {	case 0:		/* no error */		break;	case -EPIPE:		usb_clear_halt (dev, usb_sndctrlpipe (dev, 0));		break;	default:		printk (KERN_ERR "unknown error %d from usb_set_interface\n",			 result);		goto err_out;	}	/* making a template for the scsi layer to fake detection of a scsi device */	memcpy (&(new->ctempl), &hpusbscsi_scsi_host_template,		sizeof (hpusbscsi_scsi_host_template));	(struct hpusbscsi *) new->ctempl.proc_dir = new;	new->ctempl.module = THIS_MODULE;	if (scsi_register_module (MODULE_SCSI_HA, &(new->ctempl)))		goto err_out;	new->sense_command[0] = REQUEST_SENSE;	new->sense_command[4] = HPUSBSCSI_SENSE_LENGTH;	/* adding to list for module unload */	list_add (&hpusbscsi_devices, &new->lh);	return new;      err_out:	kfree (new);	return NULL;}static voidhpusbscsi_usb_disconnect (struct usb_device *dev, void *ptr){	struct hpusbscsi *hp = (struct hpusbscsi *)ptr;	usb_unlink_urb(&hp->controlurb);	usb_unlink_urb(&hp->dataurb);	spin_lock_irq(&io_request_lock);	hp->dev = NULL;	spin_unlock_irq(&io_request_lock);}static struct usb_device_id hpusbscsi_usb_ids[] = {	{USB_DEVICE (0x03f0, 0x0701)},	/* HP 53xx */	{USB_DEVICE (0x03f0, 0x0801)},	/* HP 7400 */	{USB_DEVICE (0x0638, 0x0268)},  /*iVina 1200U */	{USB_DEVICE (0x0638, 0x026a)},	/*Scan Dual II */	{USB_DEVICE (0x0638, 0x0A13)},  /*Avision AV600U */	{USB_DEVICE (0x0638, 0x0A16)},  /*Avision DS610CU Scancopier */	{USB_DEVICE (0x0638, 0x0A18)},  /*Avision AV600U Plus */	{USB_DEVICE (0x0638, 0x0A23)},  /*Avision AV220 */	{USB_DEVICE (0x0638, 0x0A24)},  /*Avision AV210 */	{USB_DEVICE (0x0686, 0x4004)},  /*Minolta Elite II */	{}			/* Terminating entry */};MODULE_DEVICE_TABLE (usb, hpusbscsi_usb_ids);MODULE_LICENSE("GPL");static struct usb_driver hpusbscsi_usb_driver = {	name:"hpusbscsi",	probe:hpusbscsi_usb_probe,	disconnect:hpusbscsi_usb_disconnect,	id_table:hpusbscsi_usb_ids,};/* module initialisation */int __inithpusbscsi_init (void){	int result;	INIT_LIST_HEAD (&hpusbscsi_devices);	PDEBUG(0, "driver loaded, DebugLvel=%d", debug); 	if ((result = usb_register (&hpusbscsi_usb_driver)) < 0) {		printk (KERN_ERR "hpusbscsi: driver registration failed\n");		return -1;	} else {		return 0;	}}void __exithpusbscsi_exit (void){	struct list_head *tmp;	struct list_head *old;	struct hpusbscsi * o;	for (tmp = hpusbscsi_devices.next; tmp != &hpusbscsi_devices;/*nothing */) {		old = tmp;		tmp = tmp->next;		o = (struct hpusbscsi *)old;		usb_unlink_urb(&o->controlurb);		if(scsi_unregister_module(MODULE_SCSI_HA,&o->ctempl)<0)			printk(KERN_CRIT"Deregistering failed!\n");		kfree(old);	}	usb_deregister (&hpusbscsi_usb_driver);}module_init (hpusbscsi_init);module_exit (hpusbscsi_exit);/* interface to the scsi layer */static inthpusbscsi_scsi_detect (struct SHT *sht){	/* Whole function stolen from usb-storage */	struct hpusbscsi *desc = (struct hpusbscsi *) sht->proc_dir;	/* What a hideous hack! */	char local_name[48];	spin_unlock_irq(&io_request_lock);	/* set up the name of our subdirectory under /proc/scsi/ */	sprintf (local_name, "hpusbscsi-%d", desc->number);	sht->proc_name = kmalloc (strlen (local_name) + 1, GFP_KERNEL);	/* FIXME: where is this freed ? */	if (!sht->proc_name) {		spin_lock_irq(&io_request_lock);		return 0;	}	strcpy (sht->proc_name, local_name);	sht->proc_dir = NULL;	/* build and submit an interrupt URB for status byte handling */ 	FILL_INT_URB(&desc->controlurb,			desc->dev,			usb_rcvintpipe(desc->dev,desc->ep_int),			&desc->scsi_state_byte,			1,			control_interrupt_callback,			desc,			desc->interrupt_interval	);	if ( 0  >  usb_submit_urb(&desc->controlurb)) {		kfree(sht->proc_name);		spin_lock_irq(&io_request_lock);		return 0;	}	/* In host->hostdata we store a pointer to desc */	desc->host = scsi_register (sht, sizeof (desc));	if (desc->host == NULL) {		kfree (sht->proc_name);		usb_unlink_urb(&desc->controlurb);		spin_lock_irq(&io_request_lock);		return 0;	}	desc->host->hostdata[0] = (unsigned long) desc;	spin_lock_irq(&io_request_lock);	return 1;}static int hpusbscsi_scsi_queuecommand (Scsi_Cmnd *srb, scsi_callback callback){

⌨️ 快捷键说明

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