整理sd卡驱动在linux 2.6.24上的实现简易心得
1.mmc_rescan 当GPIO8发生sd卡插入动作后,进入pxamci_detect_irq()中断,进而触发mmc_rescan检卡work queue工作队列
2.对于sdio设备 host->bus_ops = mmc_sdio_ops;
对于sd设备 host->bus_ops = mmc_sd_ops;
对于mmc设备 host->bus_ops = mmc_ops;
如果检测到插入sd口的设备不是上述3种设备,那么这次mmc_rescan将不能建立任何设备,关闭sd口电源,之后退出.
如果检测到的是sdio设备,那么card将被放到mmc_bus_type总线,同时sdio_funcs会被放到sdio_bus_type总线上,去匹配sido_func驱动
如果检测到的是sd设备,那么card将被放到mmc_bus_type总线,去匹配mmc_bus_type总线上的mmc驱动
如果检测到的是mmc设备,那么card也和sd设备一样将被放到mmc_bus_type总线上.
只有host不会挂载到总线上,插入的sd设备是一定会被登记到mmc_bus_type总线上,如果是sdio设备,那么同时sdio设备的sdio_funcs会被放到sdio_bus_type总线上,去匹配sido_func驱动
3.当host发送完数据之后,命令完成之后,以及sdio_int中断上来之后,cpu会触发pxamci_irq,去集中处理
4.对于主host控制器的同一读写函数为
mmc_host->ops = pxamci_ops
static const struct mmc_host_ops pxamci_ops = {
.request = pxamci_request,
.get_ro = pxamci_get_ro,
.set_ios = pxamci_set_ios,
.enable_sdio_irq= pxamci_enable_sdio_irq,
};
ps:
因为mmc主控制器只有一个,被众多驱动和设备共享,同时对于同一设备,因为数据发送是异步进行的,所以当前如果需要发送数据,那么之前的数据可能因为还没有发送完成,而仍然占用着mmc主控制器的物理寄存器,所以为了保证对物理设备的唯一共享,不扰乱物理通道上的数据序列,驱动中使用mmc_claim_host(host);来得知,当前mmc控制器是否被占用,当前mmc控制器如果被占用,那么host->claimed = 1;否则为0,如果为1,那么会在for(;;)循环中调用schedule切换出自己,当占用mmc控制器的操作完成之后,执行mmc_release_host()的时候,会激活登记到等待队列&host->wq中的其他程序获得mmc主控制器的物理使用权[gliethttp_20080630].
ps1:
不过在pxamci_irq实际中断处理过程中,也就是cmd,data等传输完毕之后,都是会调用pxamci_finish_request()来wake_up唤醒阻塞等待的mmc_wait_done()回调函数,这也是命令执行函数mmc_wait_for_cmd()使用的阻塞等待当前命令结束的方法,之所以这样,是因为有些操作需要驱动去直接操作mmc控制器的寄存器,如果不这样作,就有可能和正在传输中的data中断发生数据和cmd等的命令冲突,这样就使得mmc主控制器上的数据发生混乱,这是绝对不允许的,所以就只能使用这种阻塞方式,来一个一个的完成操作,肯定是不能并行的搞[gliethttp_20080630].