浅谈 linux mmc的sd的probe函数

linux的host层:即是芯片主机控制器层,通过去操作core层的接口函数去操作硬件层,通常各个厂商会封装自己的host结构体,通常为sdxx_host *host,该host内部有一个                                           mmc_hos(即mmc结构体)t的结构体,通过操作mmc_host(mmc结构体)间接去操作card层

            core层:即是核心层,向上给host层提供接口函数,向下去操作card层(硬件层),通常使用核心层有自己的mmc结构体,通常为mmc_host *host

                            发送cmd也是在核心层进行,

            card层:硬件层:sd卡,mmc卡,sdio卡等,有自己的card结构体,通过与core层的mmc结构体通信。


linux mmc 总线模型:mmc子系统主要涉及三条总线

             (1)   host驱动对应的driver和device挂载在linux内置的虚拟总线platform_bus_type(其实虚拟总线的作用其实就是为了将驱动和硬件分离),两者的匹配用名称匹配                              的方式,有4种匹配方式(设备树名称匹配,device 和driver 的name匹配,platform的id table匹配,还有ACHI的name匹配),即driver.c和device.c两者的                                    name相同, 则认为device匹配driver。

               (2)card 驱动相对应的driver和device挂载在mmc子系统自己创建的mmc_bus_type总线上,直接匹配。

                        匹配函数为mmc_bus_match:   原型----->          static int mmc_bus_match(struct device* dev,struct device_driver* drv)

                                                                                                            {

                                               return 1;

                                                                                                            }

                         这个很直接,返回的都是1,也就是任何挂在mmc总线上的driver和device都会匹配。

                 (3)sd驱动对应的driver和device挂载在mmc自己创建的虚拟总线sd_bus_type下,根据设备和驱动的id table来匹配。

mmc协议的实现:实现在sd卡的初始化:mmc_attach_sd

                  接下来进入正题host层的probe函数:

                 (1) mmc_alloc_host::kzmalloc (申请mmc_host空间即mmc结构体空间) +  INIT_DELAYED_WORK(host,mmc_rescan);其中INIT_DELAYED_WORK是一个                         延时函数,

               (2)初始化mmc_host结构体                                             

                (3)获得一些资源:如platform_get_resource()获得从设备树解析过来的硬件信息,保存在struct  resource res结构体

                         regulate_get (获取电压)+ clk (设置时钟) +  request_irq(设置sd中断)

                         中断上半部:request_irq(即中断处理程序,接收sd中断)

                         中断下半部 :tasket (任务队列) +  timer  (定时器)    :处理中断任务                                                                       

                (4)初始化mmc的callback函数,即core层mmc结构体的一些回调函数接口,mmc->ops = sd_host_ops;

                (5)探测sd卡:sdhost_add_host                                                                          

                                                       ||调用  

                                            mmc_add_host()-------> (5.1)host->rescan_disable = 0; //可以扫卡    

                                                                      (5.2)mmc_power_up();  //上电

                                                                      (5.3)mmc_detect_change(); -------> host->detect_change = 1;/任务延时状态改变,可以开启延时任    务


                   mmc_schedule_delayed_work(&host->detect,delay);

                  //开启延时任务,                                                                 //开启的任务为:        //INIT_DELAYED_WORK(host,mmc_rescan)                                                                                                                                                               //唤醒回调函数mmc_rescan进行扫卡


                    接下来进入core层:

                       (1) mmc_rescan()

                             ||调用

                        mmc_rescan_freq();//循环遍历扫卡

                、             ||调用

                        mmc_attch_sd(host);//扫sd卡,扫卡不成功记得断电

                                ||调用

                        mmc_sd_attch_bus() ;//当检测到相应的sd卡后,使用mmc总线将对应mmc总线操作与host连接起来,使用的是struct mmc_bus_ops

                                ||调用

                       mmc_attch_bus(host,bus_ops);//将总线与对应的host芯片控制器连接起来:host->bus_ops = ops

                                                                                                                                                  host->bus_refs = 1;

                           (2)mmc_select_voltage();//选择mmc设备工作电压     

                              

                           (3)mmc_sd_init_card();//初始化sd卡

                                              ||

                                   mmc_sd_get_cid();//获取sd卡的CID信息


                                              ||

                                  mmc_alloc_card();//分配卡的空间,确定卡的类型sd卡:card->card_type = mmc_type_sd;

                                              ||

                                  mmc_send_relative_addr(host,&card->rca);//发生命令CMD3获取RCA,得到卡的地址,流程用到这里就结束了

                                             ||

                                  mmc_sd_get_csd();//发生CMD9获取CSD并解析,填充到card,CSD寄存器保存大量卡的信息

                                             || 

                                  mmc_decode_cid(card);//解析前面的CID兵填充到card,因为sd协议有不同的版本信息放在csd,所以得到csd,获得版本号,

                                                                          再根据版本号去解析CID的数据,获取CID寄存器的值来填充struct mmc_card* card结构体

                                            ||

                                 mmc_select_card();//发生CMD7,使用上面获得的地址选择卡

                                           ||

                                 mmc_sd_setup_card();//发生ACMD51获取SCR寄存器的值,发生ACMD13获取SD卡的状态信息,解析并填充到card结构,

                                            ||                              SCR寄存器就是对CSD的补充

                                 mmc_sd_switch_hs(card);//支持高速卡就发命令设置卡为高速卡

                                           ||

                                  mmc_set_clock();//设置sd卡的时钟              

                                          ||

                                  mmc_app_set_bus_width();//使用命令ACMD6设置卡的总线宽度


总结:设备通过解析dts文件吧mmc_host_dev(platform平台设备)挂在platform虚拟平台总线上,然后注册mmc总线和sd总线,紧接着注册card层的card driver,挂在mmc总线上。其后,在host层的目录下,我们有自己的platform_driver文件(即platform 驱动),调用platform_driver_register注册platform平台驱动,即host 层的 card  device ,挂载到mmc总线上。最后,mmc总线上自己会匹配card driver 和card dev,执行card层的card 的driver文件的probe函数,生产block设备或者mmc_test相关的属性文件。



注:本文是自己个人理解,如有不对请各位大牛批评指正。































你可能感兴趣的:(linux)