linux下的sdcard驱动学习

调一款加密ic ET300时,发现通信不上,报错-110,就是超时出错,检查了gpio和sdcard配置,也检查了sd供电,发现sdio的供电域是1.8v,而根据协议初始化时默认是3.3v,然后切换到1.8v,通过飞线,使用一个ldo来供电。修改后,还是通信异常。

加打印调试,在mmc_send_cid函数报错了,就是读不到卡的cid信息。默认读不到cid,代码就退出了。 检查了再检查,实在找不出问题,于是修改代码,强制往下跑,看log,居然读到了卡的信息。

也就是只有CMD2命令跑不过,其他命令都跑过了。

发送CMD2命令前会先切换到1.8v的电压,对应平台的sdio驱动

 也就是sdio host的名字是sdio_sd才会复位下平台的sdio host,而dts配置名字不是sdio_sd,导致没有复位平台的host,修改名字后,代码能正常跑过。

调是调完了,顺便把相关的代码逻辑整理下。

从mmc_rescan这个函数讲起,相关内容已在Linux下的eMMC_Android开发-优快云博客描述过。

//以不同频率去扫描卡,但实际上某些平台定义的最小频率是400k(只会以400k去扫)
void mmc_rescan(struct work_struct *work)
{
    ...
    for (i = 0; i < ARRAY_SIZE(freqs); i++) {
        if (!mmc_rescan_try_freq(host, max(freqs[i], host->f_min)))
            break;
        if (freqs[i] <= host->f_min)
            break;
    }
    ...
}
static int mmc_rescan_try_freq(struct mmc_host *host, unsigned freq)
{
    mmc_power_up(host, host->ocr_avail);
    if (!(host->caps2 & MMC_CAP2_NO_SDIO))
        sdio_reset(host);

    mmc_go_idle(host);
    mmc_attach_sd(host);
    mmc_power_off(host);
    return -EIO;
}

int mmc_attach_sd(struct mmc_host *host)
{
    ...
    err = mmc_send_app_op_cond(host, 0, &ocr);
    mmc_attach_bus(host, &mmc_sd_ops);
    if (host->ocr_avail_sd)
        host->ocr_avail = host->ocr_avail_sd;

    rocr = mmc_select_voltage(host, ocr);
    err = mmc_sd_init_card(host, rocr, NULL);
    mmc_release_host(host);
    err = mmc_add_card(host->card);
    mmc_claim_host(host);
    ...
}


比较关键了几个函数mmc_send_app_op_cond,mmc_sd_init_card仔细看的话,实际上是sd协议的代码实现。如果扫描到了sd卡,会在/sys/bus/mmc/devices

生成相应的设备节点。

说到sd协议,比较权威的资料就是官网文档了,下载地址Simplified Specifications | SD Association,下这一篇(Physical Layer Simplified Specification)速度慢点,还是能下载的(自己也上传了一份Part1_Physical_Layer_Simplified_Specification_Ver8.00.pdf-Android文档类资源-优快云下载)。

比较关键的一张图,然后按照代码慢慢看吧。

补上sdcard热插拔部分

dts配置检测脚,如

&sdio0 {
         cd-gpios = <&eic_sync 19 GPIO_ACTIVE_LOW>;
};
int mmc_gpiod_request_cd(struct mmc_host *host, const char *con_id,unsigned int idx, bool override_active_level,unsigned int debounce, bool *gpio_invert)
{
	desc = devm_gpiod_get_index(host->parent, con_id, idx, GPIOD_IN);
	ctx->cd_gpio = desc;
}

void mmc_gpiod_request_cd_irq(struct mmc_host *host)
{
	irq = gpiod_to_irq(ctx->cd_gpio);
	ret = devm_request_threaded_irq(host->parent, irq,NULL, mmc_gpio_cd_irqt,IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING | IRQF_ONESHOT,ctx->cd_label, host);
}

int mmc_of_parse(struct mmc_host *host){
{
	mmc_gpiod_request_cd(host, "cd", 0, true,0, &cd_gpio_invert);//获取热插拔的gpio
}
int mmc_add_host(struct mmc_host *host)
{
	mmc_start_host-->mmc_gpiod_request_cd_irq/申请中断
}
static irqreturn_t mmc_gpio_cd_irqt(int irq, void *dev_id)
{
	/* Schedule a card detection after a debounce timeout */
	struct mmc_host *host = dev_id;

	host->trigger_card_event = true;
	mmc_detect_change(host, msecs_to_jiffies(200));-->mmc_schedule_delayed_work(&host->detect, delay);

	return IRQ_HANDLED;
}


INIT_DELAYED_WORK(&host->detect, mmc_rescan);//进行重新扫描

 

### 回答1: 在Linux内核中,SD卡、eMMC和UFS的驱动代码位于不同的目录中。 1. SD卡驱动代码位于`drivers/mmc`目录下。在该目录中,可以找到与SD卡相关的驱动文件和子目录,如`mmc_core.c`、`mmc_block.c`等。此外,还有一些特定厂商的SD卡驱动,例如`mmc/sdhci-s3c.c`。 2. eMMC驱动代码同样位于`drivers/mmc`目录下。在该目录中,也可以找到与eMMC相关的驱动文件和子目录,如`mmc_core.c`、`mmc_block.c`等。与SD卡相比,eMMC的驱动代码可能会有一些特定的文件,但大部分代码都是共用的。 3. UFS驱动代码位于`drivers/scsi/ufs`目录下。在该目录中,可以找到与UFS相关的驱动文件和子目录,如`ufs.c`、`ufs-sysfs.c`和`ufs-exynos.c`等。这些文件包含了UFS设备的驱动代码。 此外,还需要注意,以上目录可能会随着不同内核版本的变化而有所不同,因此在具体的内核版本中可能会有一些细微的差异。 ### 回答2: 在Linux内核中,SD卡、eMMC和UFS存储设备的驱动程序代码位于不同的目录下面。 1. SD卡驱动程序的代码位于`drivers/mmc`目录下。具体而言,SD卡的核心驱动程序是`mmc_core.c`,而与SD卡相关的SDIO(Secure Digital Input Output)功能驱动程序位于`sdio`目录下。 2. eMMC驱动程序的代码位于`drivers/mmc`目录下的`mmc_core.c`中。eMMC是一种闪存存储设备,因此其驱动程序与SD卡驱动程序共享。 3. UFS驱动程序的代码位于`drivers/scsi/ufs`目录下。UFS(Universal Flash Storage)是一种新的高性能存储标准,可以代替eMMC。UFS驱动程序的核心文件是`ufs.c`。此外,与UFS相关的Host控制器驱动程序代码位于`drivers/scsi/ufs/ufshcd.c`中。 需要注意的是,以上目录是指Linux内核的主线驱动程序目录结构。对于特定的内核版本或特定的Linux发行版,可能会存在一些小的差异。因此,在查找驱动程序代码时,最好根据具体的内核版本或系统来进行查找。 ### 回答3: 在Linux内核中,SD卡、eMMC和UFS的驱动代码可以在以下几个目录下找到: 1. SD卡驱动代码:可以在目录/drivers/mmc/host/下找到SD卡的驱动代码。主要文件包括: - sdhci.c: SD卡主机控制器(SD Host Controller)的驱动代码。 - mmc_block.c: SD卡块设备驱动代码,用于提供SD卡的块设备访问接口。 2. eMMC驱动代码:可以在目录/drivers/mmc/host/下找到eMMC驱动代码。主要文件包括: - sdhci.c: eMMC主机控制器(eMMC Host Controller)的驱动代码。 - mmc_block.c: eMMC块设备驱动代码,用于提供eMMC的块设备访问接口。 3. UFS驱动代码:可以在目录/drivers/scsi/ufs/下找到UFS驱动代码。主要文件包括: - ufs.h: UFS驱动的头文件。 - ufs-scsi.c: UFS SCSI驱动代码。 - ufs-qp.c: UFS队列处理驱动代码。 这些驱动代码文件包含了与SD卡、eMMC和UFS硬件交互的相关函数和数据结构,用于在Linux内核中实现对这些存储设备的访问和控制。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值