Chinaunix首页 | 论坛 | 博客
  • 博客访问: 665682
  • 博文数量: 237
  • 博客积分: 4285
  • 博客等级: 上校
  • 技术积分: 2701
  • 用 户 组: 普通用户
  • 注册时间: 2009-11-15 14:05
文章分类

全部博文(237)

文章存档

2014年(2)

2013年(3)

2012年(47)

2011年(15)

2010年(68)

2009年(102)

我的朋友

分类: LINUX

2012-03-30 14:14:29

结束了get_device_info,我们继续沿着storage_probe一步一步地走下去。继续,这就是我们前面提到过的三个函 数,get_transport、get_protocol和get_pipes。一旦结束了这三个函数,我们就将进入本故事的高潮部分。而在这之前,我 们只能一个一个地来看。好在这几个函数虽然不短,但是真正有用的信息只有一点,所以可以很快看完。

993   /* Get the transport, protocol,and pipe settings */

994   result = get_transport(us);

995    if(result)

996         goto BadDevice;

997    result =get_protocol(us);

998    if(result)

999         goto BadDevice;

1000   result = get_pipes(us);

1001    if(result)

1002       goto BadDevice;

第1个,get_transport(us)。

557 static int get_transport(struct us_data *us)

558 {

559    switch(us->protocol) {

560    caseUS_PR_CB:

561        us->transport_name = "Control/Bulk";

562         us->transport = usb_stor_CB_transport;

563         us->transport_reset = usb_stor_CB_reset;

564         us->max_lun = 7;

565         break;

566

567  case US_PR_CBI:

568          us->transport_name ="Control/Bulk/Interrupt";

569         us->transport = usb_stor_CBI_transport;

570         us->transport_reset = usb_stor_CB_reset;

571          us->max_lun = 7;

572        break;

573

574   case US_PR_BULK:

575        us->transport_name = "Bulk";

576        us->transport = usb_stor_Bulk_transport;

577         us->transport_reset = usb_stor_Bulk_reset;

578          break;

579

580 #ifdef CONFIG_USB_STORAGE_USBAT

581    caseUS_PR_USBAT:

582        us->transport_name = "Shuttle USBAT";

583         us->transport = usbat_transport;

584         us->transport_reset = usb_stor_CB_reset;

585         us->max_lun = 1;

586         break;

587 #endif

588

589 #ifdef CONFIG_USB_STORAGE_SDDR09

590    caseUS_PR_EUSB_SDDR09:

591         us->transport_name ="EUSB/SDDR09";

592         us->transport = sddr09_transport;

593         us->transport_reset = usb_stor_CB_reset;

594         us->max_lun = 0;

595         break;

596 #endif

597

598 #ifdef CONFIG_USB_STORAGE_SDDR55

599   case US_PR_SDDR55:

600         us->transport_name = "SDDR55";

601        us->transport = sddr55_transport;

602         us->transport_reset = sddr55_reset;

603         us->max_lun = 0;

604         break;

605 #endif

606

607 #ifdef CONFIG_USB_STORAGE_DPCM

608    caseUS_PR_DPCM_USB:

609         us->transport_name ="Control/Bulk-EUSB/SDDR09";

610        us->transport = dpcm_transport;

611         us->transport_reset= usb_stor_CB_reset;

612         us->max_lun = 1;

613         break;

614 #endif

615

616 #ifdef CONFIG_USB_STORAGE_FREECOM

617   case US_PR_FREECOM:

618         us->transport_name = "Freecom";

619         us->transport = freecom_transport;

620          us->transport_reset =usb_stor_freecom_reset;

621        us->max_lun = 0;

622        break;

623 #endif

624

625 #ifdef CONFIG_USB_STORAGE_DATAFAB

626   case US_PR_DATAFAB:

627       us->transport_name  ="Datafab Bulk-Only";

628         us->transport = datafab_transport;

629         us->transport_reset = usb_stor_Bulk_reset;

630        us->max_lun = 1;

631        break;

632 #endif

633

634 #ifdef CONFIG_USB_STORAGE_JUMPSHOT

635   case US_PR_JUMPSHOT:

636        us->transport_name  ="Lexar Jumpshot Control/Bulk";

637         us->transport = jumpshot_transport;

638       us->transport_reset = usb_stor_Bulk_reset;

639        us->max_lun = 1;

640         break;

641 #endif

642

643 #ifdef CONFIG_USB_STORAGE_ALAUDA

644    case US_PR_ALAUDA:

645         us->transport_name  = "Alauda Control/Bulk";

646         us->transport = alauda_transport;

647         us->transport_reset = usb_stor_Bulk_reset;

648         us->max_lun = 1;

649        break;

650 #endif

651

652 #ifdef CONFIG_USB_STORAGE_KARMA

653    caseUS_PR_KARMA:

654        us->transport_name = "Rio Karma/Bulk";

655         us->transport = rio_karma_transport;

656         us->transport_reset = usb_stor_Bulk_reset;

657         break;

658 #endif

659

660   default:

661           return -EIO;

662   }

663     US_DEBUGP("Transport:%s\n", us->transport_name);

664

665  /* fix for single-lun devices */

666    if(us->flags & US_FL_SINGLE_LUN)

667        us->max_lun = 0;

668    return 0;

669 }

乍一看,这么长一段,不过明眼人一看就知道了,主要就是一个switch,选择语句,语法上来说很简单,所以我们看懂这段代码不难。只是,我想说的 是,虽然这里做出一个选择不难,但是不同选择就意味着后来整个故事会有千差万别的结局,当鸟儿选择在两翼上系上黄金,就意味着它放弃展翅高飞;选择云天搏 击,就意味着放弃身外的负累。

所以,此处,我们需要仔细地看清楚我们究竟选择了怎样一条路。很显然,前面我们已经说过,对于U盘,spec规定了,它就属于Bulk-only的 传输方式,即它的us->protocol就是US_PR_BULK。这是我们刚刚在get_device_info中确定下来的。于是,在整个 switch段落中,我们所执行的只是US_PR_BULK这一段,即us的transport_name被赋值为“Bulk”,transport被赋 值为usb_stor_Bulk_transport,transport_reset被赋值为usb_stor_Bulk_reset。其中我们最需要 记住的是,us的成员transport和transport_reset是两个函数指针。程序员们把这个称作“钩子”。这两个赋值我们需要牢记,日后我 们一定会用到它们,因为这正是我们真正的数据传输时调用的东西。关于usb_stor_Bulk_*的这两个函数,到时候调用了再来看。现在只需知道,日 后我们一定会回过来看这个赋值的。

580行到658行,不用多说了,这里全是与各种特定产品相关的一些编译开关,它们有一些自己定义一些传输函数,有些则共用通用的函数。

666行,判断us->flags,还记得我们在讲unusual_devs.h文件时说的flags吧,这里第一次用上了。有些设备设置了 US_FL_SINGLE_LUN这个flag,就表明它是只有一个LUN的。像这样的设备挺多的,随便从unusual_devs.h中抓一个出来:

596 UNUSUAL_DEV(  0x054c, 0x002d, 0x0100, 0x0100,

597                "Sony",

598                "Memorystick MSAC-US1",

599                US_SC_DEVICE, US_PR_DEVICE, NULL,

600                US_FL_SINGLE_LUN ),

比如这个Sony的Memorystick。中文名叫“记忆棒”,大小就与口香糖一样,也是一种存储芯片。它是Sony公司推出的,广泛用于Sony的各种数码产品中,比如数码照相机、数码摄影机。

LUN就是Logical Unit Number。通常在谈到SCSI设备时不可避免要说起LUN。关于LUN,曾几何时,一位来自Novell的参与开发Linux内核中USB子系统的工 程师这样对我说,一个LUN就是设备中的一个驱动。下面举例来说一下USB中引入LUN的目的。有些读卡器可以有多个插槽,比如有两个,其中一个支持CF 卡,另一个支持SD卡,那么这种情况要区分这两个插槽里的设备,就得引入LUN有这个概念,即逻辑单元。很显然,像U盘这样简单的设备其LUN必然是一 个。有时候,人们常把U盘中一个分区当做一个LUN,但是不应该这么理解。

知道了LUN以后,自然就可以知道US_FL_SINGLE_LUN是做什么了,这个flag的意义很明显,直截了当地告诉你,这个设备只有一个 LUN,它不支持多个LUN。而max_lun又是什么意思?us中的成员max_lun等于一个设备所支持的最大LUN号。即如果一个设备支持四个 LUNs,那么这四个LUN的编号就是0,1,2,3,而max_lun就是3。如果一个设备不用支持多个LUN,那么它的max_lun就是0。所以这 里max_lun就是设为0。

另外一个需要注意的地方是,比较一下各个case语句会发现US_PR_BULK和其他的case不一样,其他的case下面都设置了 us->max_lun,而对应于Bulk-Only协议的这个case,它没有设置us->max_lun,之所以不设,是因为这个值由设 备说了算,必须向设备查询,这是Bulk-Only协议规定的。在drivers/usb/storage/transport.c中定义了一个 usb_stor_Bulk_max_lun()函数,它将负责获取这个max lun。而我依然要声明一次,这个函数对我们U盘没有什么意义,这个值肯定是0,所以这个函数咱们就不去理睬了。

至此,get_transport()也结束了,和get_device_info()一样。我们目前所看到的这些函数都不得不面对现实,对它们来说,凋 谢是最终的结果,盛开只是一个过程。而对我们来说,要到达终点,那么和这些函数狭路相逢,终不能幸免。然而,不管这部分代码有多么重要,也不过是我们整个 长途旅程中,来去匆匆的转机站,无论停留多久,始终要离去坐另一班机。
阅读(567) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~