Linux内核MMC里的轮询机制

简介: Linux内核MMC里的轮询机制

1 前言

最近遇到客户提的一个问题,大概意思是他们的SDIO Wi-Fi在卸载Wi-Fi驱动后再加载就检测不到Wi-Fi设备了。从他的问题看大概是热插拔有问题。

想要支持Wi-Fi复位后能重新扫描到Wi-Fi设备,需要host设置MMC_CAP_NEEDS_POLL

#define MMC_CAP_NEEDS_POLL  (1 << 5)  /* Needs polling for card-detection */

2 如何使用MMC里的轮询机制做探卡检测?

先说方法,后面再分析。

方法一:修改dts,在对应的节点增加字段broken-cd,同时,如果有non-removable字段,必须去掉该字段。

On Snow we've got sip WiFi and so can keep drive strengths low to* reduce EMI.*/
&mmc_3 {
status = "okay" num-slots = <1>? broken-cd; cap-sd1o-1rq;
keep-power-in-suspend;
card-detect-delay =<200>;
samsung,dw-mshc-ciu-div =<3>:
samsung,dw-mshc-sdr-timing = <2 3> samsung,dw-mshc-ddr-timing = <1 2> pinctrl-names ="default";
pinctr1-0 =<&sd3_clk &sd3_cmd &sd3_bus4 &wifi_en &wifi_rst>" bus-width = <4>" cap-sd-highspeed;
mmc-pwrseq =<&mmc3_pwrseq>; 哆哆jarvis  
};

方法二:通过其他手段设置host->caps |= MMC_CAP_NEEDS_POLL

3 MMC里的轮询机制剖析

3.1 在dts设置broken-cd字段,代码在哪里解析?

mmc_of_parse函数,路径是drivers\mmc\core\core.c,of_property_read_bool函数读broken-cd字段,如果读到,就给host设置MMC_CAP_NEEDS_POLL能力。

int mmc_of_parse(struct mmc_host *host)
{
    /* 省略 */
 /*
  * Configure CD and WP pins. They are both by default active low to
  * match the SDHCI spec. If GPIOs are provided for CD and / or WP, the
  * mmc-gpio helpers are used to attach, configure and use them. If
  * polarity inversion is specified in DT, one of MMC_CAP2_CD_ACTIVE_HIGH
  * and MMC_CAP2_RO_ACTIVE_HIGH capability-2 flags is set. If the
  * "broken-cd" property is provided, the MMC_CAP_NEEDS_POLL capability
  * is set. If the "non-removable" property is found, the
  * MMC_CAP_NONREMOVABLE capability is set and no card-detection
  * configuration is performed.
  */
 /* Parse Card Detection */
 if (of_property_read_bool(np, "non-removable")) {
  host->caps |= MMC_CAP_NONREMOVABLE;
  if (of_property_read_bool(np, "cd-post"))
   host->caps2 |= MMC_CAP2_CD_POST;
 } else {
  cd_cap_invert = of_property_read_bool(np, "cd-inverted");
  if (of_property_read_bool(np, "broken-cd"))
    host->caps |= MMC_CAP_NEEDS_POLL;
  ret = mmc_gpiod_request_cd(host, "cd", 0, true,
        0, &cd_gpio_invert);
  if (!ret)
   dev_info(host->parent, "Got CD GPIO\n");
  else if (ret != -ENOENT && ret != -ENOSYS)
   return ret;
  /*
   * There are two ways to flag that the CD line is inverted:
   * through the cd-inverted flag and by the GPIO line itself
   * being inverted from the GPIO subsystem. This is a leftover
   * from the times when the GPIO subsystem did not make it
   * possible to flag a line as inverted.
   *
   * If the capability on the host AND the GPIO line are
   * both inverted, the end result is that the CD line is
   * not inverted.
   */
  if (cd_cap_invert ^ cd_gpio_invert)
   host->caps2 |= MMC_CAP2_CD_ACTIVE_HIGH;
 }
    /* 省略 */
}

3.2 探卡检测流程

mmc_alloc_host函数会创建一个工作队列,mmc_rescan与host->detect绑定。

INIT_DELAYED_WORK(&host->detect, mmc_rescan);

mmc_rescan就是扫描卡的函数

void mmc_rescan(struct work_struct *work)
{
 struct mmc_host *host =
  container_of(work, struct mmc_host, detect.work);
 int i;
 if (host->trigger_card_event && host->ops->card_event) {
  host->ops->card_event(host);
  host->trigger_card_event = false;
 }
 if (host->rescan_disable)
  return;
 /* If there is a non-removable card registered, only scan once */
 if ((host->caps & MMC_CAP_NONREMOVABLE) && host->rescan_entered)
  return;
 host->rescan_entered = 1;
 mmc_bus_get(host);
 /*
  * if there is a _removable_ card registered, check whether it is
  * still present
  */
 if (host->bus_ops && !host->bus_dead
     && !(host->caps & MMC_CAP_NONREMOVABLE))
  host->bus_ops->detect(host);
 host->detect_change = 0;
 /*
  * Let mmc_bus_put() free the bus/bus_ops if we've found that
  * the card is no longer present.
  */
 mmc_bus_put(host);
 mmc_bus_get(host);
 /* if there still is a card present, stop here */
 if (host->bus_ops != NULL) {
  mmc_bus_put(host);
  goto out;
 }
 /*
  * Only we can add a new handler, so it's safe to
  * release the lock here.
  */
 mmc_bus_put(host);
 if (!(host->caps & MMC_CAP_NONREMOVABLE) && host->ops->get_cd &&
   host->ops->get_cd(host) == 0) {
  mmc_claim_host(host);
  mmc_power_off(host);
  mmc_release_host(host);
  goto out;
 }
 mmc_claim_host(host);
 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;
 }
 mmc_release_host(host);
out:
  if (host->caps & MMC_CAP_NEEDS_POLL)
  mmc_schedule_delayed_work(&host->detect, HZ);
}

看到最后两行,判断host的能力,如果设置了MMC_CAP_NEEDS_POLL,也就是轮询机制,就会每隔HZ(这是个宏)时间执行一次host->detect,也就是mmc_rescan。

总结

对于探卡检测,通过在dts里面配置broken-cd就可以实现轮询探卡检测。

号主:一枚机械专业本科生,经历了转行,从外包逆袭到芯片原厂的Linux驱动开发工程师,深入操作系统的世界,贯彻终身学习、终身成长的理念。平时喜欢折腾,寒冬之下,抱团取暖,期待你来一起探讨技术、搞自媒体副业,程序员接单和投资理财。【对了,不定期送闲置开发板、书籍、键盘等等】。

如果你想了解我的转行经验,欢迎找我交流~gongzhong号【哆哆jarvis】

一起不断探索自我、走出迷茫、找到热爱,希望和你成为朋友,一起成长~

相关文章
|
5天前
|
Ubuntu Linux 开发者
Ubuntu20.04搭建嵌入式linux网络加载内核、设备树和根文件系统
使用上述U-Boot命令配置并启动嵌入式设备。如果配置正确,设备将通过TFTP加载内核和设备树,并通过NFS挂载根文件系统。
33 15
|
24天前
|
存储 编译器 Linux
动态链接的魔法:Linux下动态链接库机制探讨
本文将深入探讨Linux系统中的动态链接库机制,这其中包括但不限于全局符号介入、延迟绑定以及地址无关代码等内容。
323 22
|
30天前
|
算法 Linux
深入探索Linux内核的内存管理机制
本文旨在为读者提供对Linux操作系统内核中内存管理机制的深入理解。通过探讨Linux内核如何高效地分配、回收和优化内存资源,我们揭示了这一复杂系统背后的原理及其对系统性能的影响。不同于常规的摘要,本文将直接进入主题,不包含背景信息或研究目的等标准部分,而是专注于技术细节和实际操作。
|
30天前
|
存储 缓存 网络协议
Linux操作系统的内核优化与性能调优####
本文深入探讨了Linux操作系统内核的优化策略与性能调优方法,旨在为系统管理员和高级用户提供一套实用的指南。通过分析内核参数调整、文件系统选择、内存管理及网络配置等关键方面,本文揭示了如何有效提升Linux系统的稳定性和运行效率。不同于常规摘要仅概述内容的做法,本摘要直接指出文章的核心价值——提供具体可行的优化措施,助力读者实现系统性能的飞跃。 ####
|
1月前
|
监控 算法 Linux
Linux内核锁机制深度剖析与实践优化####
本文作为一篇技术性文章,深入探讨了Linux操作系统内核中锁机制的工作原理、类型及其在并发控制中的应用,旨在为开发者提供关于如何有效利用这些工具来提升系统性能和稳定性的见解。不同于常规摘要的概述性质,本文将直接通过具体案例分析,展示在不同场景下选择合适的锁策略对于解决竞争条件、死锁问题的重要性,以及如何根据实际需求调整锁的粒度以达到最佳效果,为读者呈现一份实用性强的实践指南。 ####
|
1月前
|
缓存 监控 网络协议
Linux操作系统的内核优化与实践####
本文旨在探讨Linux操作系统内核的优化策略与实际应用案例,深入分析内核参数调优、编译选项配置及实时性能监控的方法。通过具体实例讲解如何根据不同应用场景调整内核设置,以提升系统性能和稳定性,为系统管理员和技术爱好者提供实用的优化指南。 ####
|
1月前
|
负载均衡 算法 Linux
深入探索Linux内核调度机制:公平与效率的平衡####
本文旨在剖析Linux操作系统内核中的进程调度机制,特别是其如何通过CFS(完全公平调度器)算法实现多任务环境下资源分配的公平性与系统响应速度之间的微妙平衡。不同于传统摘要的概览性质,本文摘要将直接聚焦于CFS的核心原理、设计目标及面临的挑战,为读者揭开Linux高效调度的秘密。 ####
37 3
|
1月前
|
消息中间件 安全 Linux
深入探索Linux操作系统的内核机制
本文旨在为读者提供一个关于Linux操作系统内核机制的全面解析。通过探讨Linux内核的设计哲学、核心组件、以及其如何高效地管理硬件资源和系统操作,本文揭示了Linux之所以成为众多开发者和组织首选操作系统的原因。不同于常规摘要,此处我们不涉及具体代码或技术细节,而是从宏观的角度审视Linux内核的架构和功能,为对Linux感兴趣的读者提供一个高层次的理解框架。
|
2月前
|
Linux 网络安全 数据安全/隐私保护
Linux 超级强大的十六进制 dump 工具:XXD 命令,我教你应该如何使用!
在 Linux 系统中,xxd 命令是一个强大的十六进制 dump 工具,可以将文件或数据以十六进制和 ASCII 字符形式显示,帮助用户深入了解和分析数据。本文详细介绍了 xxd 命令的基本用法、高级功能及实际应用案例,包括查看文件内容、指定输出格式、写入文件、数据比较、数据提取、数据转换和数据加密解密等。通过掌握这些技巧,用户可以更高效地处理各种数据问题。
141 8