一、工作流程

mmc驱动主要文件包括

drivers/mmc/card/block.c
drivers/mmc/card/queue.c
drivers/mmc/core/core.c
drivers/mmc/core/host.c
drivers/mmc/core/

内核启动时,首先执行core/core.c的mmc_init,注册mmc、sd总线,以及一个host class设备。接着执行card/block.c中,申请一个块设备。

二、数据结构:

这里涉及三种总线

[cpp] view plaincopy
  1. 1. platform bus //MMC host controller 作为一种 platform device, 它是需要注册到 platform bus上 的
  2. driver/base/platform.c
  3. struct bus_type platform_bus_type = {
  4. .name        = "platform",
  5. .dev_attrs    = platform_dev_attrs,
  6. .match        = platform_match,
  7. .uevent        = platform_uevent,
  8. .pm        = &platform_dev_pm_ops,
  9. };
  10. 2. mmc bus type  //在mmc_init()中被创建的.通过调用 mmc_register_bus() 来注册 MMC 总线
  11. drivers\mmc\core\bus.c
  12. static struct bus_type mmc_bus_type = {
  13. .name        = "mmc",
  14. .dev_attrs    = mmc_dev_attrs,
  15. .match        = mmc_bus_match,
  16. .uevent        = mmc_bus_uevent,
  17. .probe        = mmc_bus_probe,
  18. .remove        = mmc_bus_remove,
  19. .shutdown        = mmc_bus_shutdown,
  20. .pm        = &mmc_bus_pm_ops,
  21. };
  22. 3. sdio bus type    //在mmc_init()中被创建的.通过调用sdio_register_bus() 来注册 SDIO 总线
  23. drivers\mmc\core\sdio_bus.c
  24. static struct bus_type sdio_bus_type = {
  25. .name        = "sdio",
  26. .dev_attrs    = sdio_dev_attrs,
  27. .match        = sdio_bus_match,
  28. .uevent        = sdio_bus_uevent,
  29. .probe        = sdio_bus_probe,
  30. .remove        = sdio_bus_remove,
  31. .pm        = SDIO_PM_OPS_PTR,
  32. };

其中mmc总线操作相关函数,由于mmc卡支持多种总数据线,如SPI、SDIO、8LineMMC而不同的总线的操作控制方式不尽相同,所以通过此结构与相应的总线回调函数相关联。

[cpp] view plaincopy
  1. //总线操作结构
  2. struct mmc_bus_ops {
  3. void (*remove)(struct mmc_host *);
  4. void (*detect)(struct mmc_host *);
  5. int (*sysfs_add)(struct mmc_host *, struct mmc_card *card);
  6. void (*sysfs_remove)(struct mmc_host *, struct mmc_card *card);
  7. void (*suspend)(struct mmc_host *);
  8. void (*resume)(struct mmc_host *);
  9. };
  10. //  mmc卡的总线操作 core/mmc.c
  11. static const struct mmc_bus_ops mmc_ops = {
  12. .remove = mmc_remove,
  13. .detect = mmc_detect,
  14. .sysfs_add = mmc_sysfs_add,
  15. .sysfs_remove = mmc_sysfs_remove,
  16. .suspend = mmc_suspend,
  17. .resume = mmc_resume,
  18. };
  19. // sd卡的总线操作 core/sd.c
  20. static const struct mmc_bus_ops mmc_sd_ops = {
  21. .remove = mmc_sd_remove,
  22. .detect = mmc_sd_detect,
  23. .sysfs_add = mmc_sd_sysfs_add,
  24. .sysfs_remove = mmc_sd_sysfs_remove,
  25. .suspend = mmc_sd_suspend,
  26. .resume = mmc_sd_resume,
  27. };
  28. // sdio的总线操作 core/sdio.c
  29. static const struct mmc_bus_ops mmc_sdio_ops = {
  30. .remove = mmc_sdio_remove,
  31. .detect = mmc_sdio_detect,
  32. };

关于总线操作的函数:

.detect,驱动程序经常需要调用此函数去检测mmc卡的状态,具体实现是发送CMD13命令,并读回响应,如果响应错误,则依次调用.remove、detach_bus来移除卡及释放总线。

三、总体架构

1、kernel启动时,先后执行mmc_init()及mmc_blk_init(),以对mmc设备及mmc块模块进行初始化

[cpp] view plaincopy
  1. mmc/core/core.c
  2. static int __init mmc_init(void)
  3. workqueue = alloc_ordered_workqueue("kmmcd", 0);//建立了一个工作队列workqueue,这个工作队列的作用主要是用来支持热插拔
  4. ret = mmc_register_bus();//注册一个mmc总线
  5. ret = mmc_register_host_class();//注册了一个 mmc_host 类
  6. ret = sdio_register_bus();//注册了一个 sdio_bus_type
  7. *******
  8. mmc/card/block.c
  9. static int __init mmc_blk_init(void)
  10. res = register_blkdev(MMC_BLOCK_MAJOR, "mmc");//注册一个块设备
  11. res = mmc_register_driver(&mmc_driver);//注册一个mmc设备驱动
  12. static struct mmc_driver mmc_driver =
  13. .probe      = mmc_blk_probe,
  14. static int mmc_blk_probe(struct mmc_card *card)
  15. mmc_set_bus_resume_policy(card->host, 1);//*host 该指针指向一个mmc主机实例,块设备中的读写操作就是调用这个mmc主机的操作函数host->ops->request来实现对实际硬件的操作。

2、core部分会做两件事

a -- 取得总线

b -- 检查总线操作结构指针bus_ops,如果为空,则重新利用各总线对端口进行扫描,检测顺序依次为:SDIO、Normal SD、MMC。当检测到相应的卡类型后,就使用mmc_attach_bus()把相对应的总线操作与host连接起来

[cpp] view plaincopy
  1. void mmc_attach_bus(struct mmc_host *host, const struct mmc_bus_ops *ops)
  2. {
  3. ...
  4. host->bus_ops = ops;
  5. ...
  6. }

3、然后在挂载mmc设备驱动时,执行驱动程序中的xx_mmc_probe(),检测host设备中挂载的sd设备

[cpp] view plaincopy
  1. kernel\arch\arm\configs\msm9625_defconfig
  2. CONFIG_MMC_MSM=y
  3. kernel\drivers\mmc\host\Makefile
  4. obj-$(CONFIG_MMC_MSM)        += msm_sdcc.o
  5. msm_sdcc.c (drivers\mmc\host)
  6. //系统初始化时扫描 platform 总线上是否有名为该SD主控制器名字"msm_sdcc"的设备,如果有, 驱动程序将主控制器挂载到 platform 总线上,并注册该驱动程序
  7. static int __init msmsdcc_init(void)
  8. platform_driver_register(&msmsdcc_driver);    //注册 platform driver
  9. static struct platform_driver msmsdcc_driver = {
  10. .probe        = msmsdcc_probe,
  11. .remove        = msmsdcc_remove,
  12. .driver        = {
  13. .name    = "msm_sdcc",
  14. .pm    = &msmsdcc_dev_pm_ops,
  15. .of_match_table = msmsdcc_dt_match,
  16. },
  17. };
  18. //整个设备驱动的 probe()函数,其本质就是是为设备建立起数据结构并对其赋初值
  19. //msmsdcc_probe 所有赋值中,我们重点关注从 platform_device *pdev里得到的数据,即设备树里的数据
  20. //platform_device *pdev是在系统初始化的时候扫描 platform 总线发现SD主控制器后所得到的数据
  21. static int msmsdcc_probe(struct platform_device *pdev)
  22. {
  23. //初始化设备的数据结构
  24. if (pdev->dev.of_node) {
  25. plat = msmsdcc_populate_pdata(&pdev->dev);        //获取设备树信息
  26. of_property_read_u32((&pdev->dev)->of_node,"cell-index", &pdev->id);
  27. } else {
  28. plat = pdev->dev.platform_data;
  29. }
  30. //为主设备控制器建立数据结构,建立kobject,并初始化等待队列,工作队列,以及一些控制器的配置
  31. mmc = mmc_alloc_host(sizeof(struct msmsdcc_host), &pdev->dev);            ---- 1
  32. //实现设备驱动的功能函数,如mmc->ops = &pxamci_ops;
  33. mmc->ops = &msmsdcc_ops;
  34. //申请中断函数 request_irq()
  35. ret = request_irq(core_irqres->start, msmsdcc_irq, IRQF_SHARED,DRIVER_NAME " (cmd)", host);
  36. //注册设备,即注册kobject,建立sys文件,发送uevent等
  37. mmc_add_host(mmc);                                                        ---- 2
  38. //其他需求,如在/proc/driver下建立用户交互文件等
  39. ret = device_create_file(&pdev->dev, &host->auto_cmd21_attr);
  40. }

4、此时probe函数会创建一个host设备,然后开启一个延时任务mmc_rescan()

[cpp] view plaincopy
  1. 1:
  2. core/host.c
  3. //重要函数mmc_alloc_host , 用于分配mmc_host结构体指针的内存空间大小
  4. struct mmc_host *mmc_alloc_host(int extra, struct device *dev)----创建一个 mmc_host 和 mmc_spi_host ,且mmc_host的最后一个成员指针private指向mmc_spi_host
  5. //建立数据结构
  6. struct mmc_host *host;
  7. host = kzalloc(sizeof(struct mmc_host) + extra, GFP_KERNEL);
  8. //建立kobject
  9. host->parent = dev;
  10. host->class_dev.parent = dev;
  11. host->class_dev.class = &mmc_host_class;
  12. device_initialize(&host->class_dev);
  13. //初始化等待队列,工作队列
  14. init_waitqueue_head(&host->wq);
  15. INIT_DELAYED_WORK(&host->detect, mmc_rescan);    //建立了一个工作队列任务 structdelayed_work detect。工作队列任务执行的函数为mmc_rescan
  16. //配置控制器
  17. host->max_segs = 1;
  18. host->max_seg_size = PAGE_CACHE_SIZE;
  19. return host;

5、驱动挂载成功后,mmc_rescan()函数被执行,然后对卡进行初始化(步骤后面详细讲述)

[cpp] view plaincopy
  1. core/core.c
  2. //mmc_rescan 函数是需要重点关注的,因为SD卡协议中的检测,以及卡识别等都是在此函数中实现
  3. void mmc_rescan(struct work_struct *work)
  4. if (host->bus_ops && host->bus_ops->detect && !host->bus_dead && !(host->caps & MMC_CAP_NONREMOVABLE))    //存在热插拔卡,不包括emmc,调用探测函数
  5. host->bus_ops->detect(host);
  6. mmc_bus_put(host);    //减少引用技术,就释放
  7. mmc_bus_get(host);    //增加bus引用计数
  8. if (host->bus_ops != NULL) {
  9. mmc_bus_put(host);    //如果卡仍然存在,减少引用计数,不必探测了
  10. goto out;
  11. }
  12. if (host->ops->get_cd && host->ops->get_cd(host) == 0)  //有卡,退出
  13. goto out;
  14. mmc_claim_host(host);                   //用于检测host是否被占用,占用则退出,否则标记成占用
  15. if (!mmc_rescan_try_freq(host, host->f_min))

初始化卡接以下流程初始化:

a、发送CMD0使卡进入IDLE状态
b、发送CMD8,检查卡是否SD2.0。SD1.1是不支持CMD8的,因此在SD2.0 Spec中提出了先发送CMD8,如响应为无效命令,则卡为SD1.1,否则就是SD2.0(请参考SD2.0 Spec)。
c、发送CMD5读取OCR寄存器。
d、发送ACMD55、CMD41,使卡进入工作状态。MMC卡并不支持ACMD55、CMD41,如果这步通过了,则证明这张卡是SD卡。
e、如果d步骤错误,则发送CMD1判断卡是否为MMC。SD卡不支持CMD1,而MMC卡支持,这就是SD和MMC类型的判断依据。
f、如果ACMD41和CMD1都不能通过,那这张卡恐怕就是无效卡了,初始化失败。

假如扫描到总线上挂有有效的设备,就调用相对应的函数把设备装到系统中,mmc_attach_sdio()、mmc_attach_sd()、mmc_attach_mmc()这三个函数分别是装载sdio设备,sd卡和mmc卡的。

在 sd卡中,驱动循环发送ACMD41、CMD55给卡,读取OCR寄存器,成功后,依次发送CMD2(读CID)、CMD3(得到RCA)、CMD9(读 CSD)、CMD7(选择卡)。后面还有几个命令分别是ACMD41&CMD51,使用CMD6切换一些功能,如切换到高速模式。

经过上述步骤,已经确定当前插入的卡是一张有效、可识别的存储卡。然后调用mmc_add_card()把存储卡加到系统中。正式与系统驱动连接在一起

[cpp] view plaincopy
  1. static int mmc_rescan_try_freq(struct mmc_host *host, unsigned freq)
  2. host->f_init = freq;                 //设置某一个时钟频率
  3. mmc_power_up(host);                     //与 mmc_power_off 类似,不过设置了启动时需要的 ios
  4. mmc_go_idle(host);          ----1a      //CMD0 ,SD卡从 inactive 到 idle
  5. mmc_send_if_cond(host, host->ocr_avail);//检测SD卡是否支持SD2.0
  6. if (!mmc_attach_sd(host))   ----1b      //然后对mmc或者sd发送一些命令进行探测,这里以 sd 为例
  7. 1a:
  8. int mmc_go_idle(struct mmc_host *host)
  9. struct mmc_command cmd = {0};
  10. cmd.opcode = MMC_GO_IDLE_STATE; //即CMD0
  11. cmd.arg = 0;                    //此命令无参数
  12. err = mmc_wait_for_cmd(host, &cmd, 0)
  13. int mmc_wait_for_cmd(struct mmc_host *host, struct mmc_command *cmd, int retries)
  14. memset(cmd->resp, 0, sizeof(cmd->resp));  //调用了 mmc_start_request,
  15. cmd->retries = retries;
  16. mrq.cmd = cmd;
  17. mmc_wait_for_req(host, &mrq);
  18. void mmc_wait_for_req(struct mmc_host *host, struct mmc_request *mrq)   ----重要函数
  19. __mmc_start_req(host, mrq);
  20. static int __mmc_start_req(struct mmc_host *host, struct mmc_request *mrq)
  21. mmc_start_request(host, mrq);
  22. static void mmc_start_request(struct mmc_host *host, struct mmc_request *mrq)
  23. host->ops->request(host, mrq);    //即 msmsdcc_request, MMC 核心与核HOST 层握手了
  24. 1b:
  25. core/mmc.c
  26. int mmc_attach_sd(struct mmc_host *host)                    //完成匹配,和初始化卡的功能
  27. err = mmc_send_app_op_cond(host, 0, &ocr);      ----1b1 //检测是否是支持SD卡
  28. host->ocr = mmc_select_voltage(host, ocr);               //设置MMC电压
  29. err = mmc_init_card(host, host->ocr, NULL);              //对mmc卡进行初始化,主要是读取mmc卡里的一些寄存器信息,且对这些寄存器的值进行设置
  30. err = mmc_sd_init_card(host, host->ocr, NULL);   ----1b2
  31. err = mmc_add_card(host->card);                  ----1b3 //调用 mmc_add_card 来把 mmc_card 挂载到 mmc_bus_type 总线去
  32. 1b1:
  33. int mmc_send_app_op_cond(struct mmc_host *host, u32 ocr, u32 *rocr)
  34. cmd.opcode = SD_APP_OP_COND;    //ACMD41,获取 SDcard 的允许电压范围值,保存在 ocr 中. 所有发送它之前需要发送 CMD_55 命令。执行完后 card 状态变为 READY
  35. 1b2:
  36. static int mmc_sd_init_card(struct mmc_host *host, u32 ocr,struct mmc_card *oldcard)
  37. err = mmc_sd_get_cid(host, ocr, cid, &rocr);        //发送 CMD2 ,获取卡的身份信息,进入到身份状态
  38. card = mmc_alloc_card(host, &sd_type);              //分配一张 SD 类型的 card 结构
  39. err = mmc_send_relative_addr(host, &card->rca);      //获取卡的相对地址,注意一前卡和主机通信都采用默认地址,现在有了自己的地址了,进入到 stand_by 状态
  40. err = mmc_sd_get_csd(host, card); ----mmc_send_csd(card, card->raw_csd);//CMD9, 获取 CSD 寄存器的信息,包括 block 长度,卡容量等信息
  41. err = mmc_select_card(card);                        //发送 CMD7, 选中目前 RADD 地址上的卡,任何时候总线上只有一张卡被选中,进入了传输状态
  42. err = mmc_sd_setup_card(host, card, oldcard != NULL);
  43. int mmc_sd_setup_card(struct mmc_host *host, struct mmc_card *card,bool reinit)
  44. mmc_app_send_scr(card, card->raw_scr);   //发送命令 ACMD51 获取 SRC 寄存器的内容,进入到 SENDING-DATA 状态
  45. if (host->ops->get_ro(host) > 0 )      // get_ro(host) 即是 msmsdcc_get_ro
  46. mmc_card_set_readonly(card);        //是否写保护,如果是的,将 card 状态设置为只读状态
  47. 1b3:
  48. core/bus.c
  49. int mmc_add_card(struct mmc_card *card)     //  /sys/devices/msm_sdcc.2/mmc_host/mmc0
  50. ret = device_add(&card->dev);
  51. drivers/base/core.c
  52. int device_add(struct device *dev)
  53. dev_set_name(dev, "%s%u", dev->bus->dev_name, dev->id); //
  54. bus_probe_device(dev);
  55. void bus_probe_device(struct device *dev)
  56. if (bus->p->drivers_autoprobe)
  57. ret = device_attach(dev);           //这样,在总线 mmc_bus_type 中就有了 mmc 设备 mmc_card 了
  58. ***********
  59. 2:
  60. //完成kobject的注册,并调用 mmc_rescan,目的在于在系统初始化的时候就扫描SD总线查看是否存在SD卡
  61. int mmc_add_host(struct mmc_host *host)
  62. err = device_add(&host->class_dev);//将设备注册进linux设备模型,最终的结果就是在 sys/bus/platform/devices 目录下能见到 mmc 设备节点
  63. mmc_start_host(host);
  64. void mmc_start_host(struct mmc_host *host)
  65. mmc_power_off(host);                ----2a
  66. mmc_detect_change(host, 0);         ----2b
  67. 2a:
  68. void mmc_power_off(struct mmc_host *host)
  69. host->ios.power_mode = MMC_POWER_OFF;    //对 ios 进行了设置
  70. ...
  71. mmc_set_ios(host);
  72. void mmc_set_ios(struct mmc_host *host)
  73. host->ops->set_ios(host, ios);            // set_ios 实际上就是 mmc_host_ops 的 .set_ios  = msmsdcc_set_ios,
  74. 2b:
  75. void mmc_detect_change(struct mmc_host *host, unsigned long delay)
  76. mmc_schedule_delayed_work(&host->detect, delay); //实际上就是调用我们前面说的延时函数 mmc_rescan

6、卡设备加到系统中后,通知mmc块设备驱动。块设备驱动此时调用probe函数,即mmc_blk_probe()函数,mmc_blk_probe()首 先分配一个新的mmc_blk_data结构变量,然后调用mmc_init_queue,初始化blk队列。然后建立一个线程 mmc_queue_thread()

7、然后就可以进行传输命令和数据了

[cpp] view plaincopy
  1. struct mmc_host_ops {
  2. //用于SD卡命令的传输,比如发送和接收命令,CMD0,CMD8,ACMD41诸如此类的都是在这个函数去实现
  3. void    (*request)(struct mmc_host *host, struct mmc_request *req);
  4. }
  5. static const struct mmc_host_ops msmsdcc_ops = {
  6. .enable     = msmsdcc_enable,
  7. .disable    = msmsdcc_disable,
  8. .pre_req        = msmsdcc_pre_req,
  9. .post_req       = msmsdcc_post_req,
  10. .request    = msmsdcc_request,
  11. .set_ios    = msmsdcc_set_ios,
  12. .get_ro     = msmsdcc_get_ro,
  13. .enable_sdio_irq = msmsdcc_enable_sdio_irq,
  14. .start_signal_voltage_switch = msmsdcc_switch_io_voltage,
  15. .execute_tuning = msmsdcc_execute_tuning,
  16. .hw_reset = msmsdcc_hw_reset,
  17. .stop_request = msmsdcc_stop_request,
  18. .get_xfer_remain = msmsdcc_get_xfer_remain,
  19. .notify_load = msmsdcc_notify_load,
  20. };
  21. /*这个函数实现了命令和数据的发送和接收,
  22. 当 CORE 部分需要发送命令或者传输数据时,都会调用这个函数,并传递 mrq 请求*/
  23. static void msmsdcc_request(struct mmc_host *mmc, struct mmc_request *mrq)
  24. mmc_request_done(mmc, mrq);             // 如果卡不存在,就终止请求
  25. msmsdcc_request_start(host, mrq);
  26. static void msmsdcc_request_start (struct msmsdcc_host *host, struct mmc_request *mrq)
  27. if ((mrq->data->flags & MMC_DATA_READ) ||host->curr.use_wr_data_pend)      //判断发送数据还是命令
  28. msmsdcc_start_data(host, mrq->data,mrq->sbc ? mrq->sbc : mrq->cmd,0);   //发送数据
  29. else
  30. msmsdcc_start_command(host,mrq->sbc ? mrq->sbc : mrq->cmd,0);          //发送命令
  31. static void msmsdcc_start_data(struct msmsdcc_host *host, struct mmc_data *data,struct mmc_command *cmd, u32 c)
  32. //对某些 寄存器进行设置, 使能某些中断, 如 pio_irqmask
  33. ...
  34. if (is_dma_mode(host) && (datactrl & MCI_DPSM_DMAENABLE))   //采用 DMA 进行数据传输还是采用 FIFO 进行数据传输
  35. msmsdcc_start_command_deferred(host, cmd, &c);          //启动了数据传输模式
  36. else
  37. msmsdcc_start_command(host, cmd, c)
  38. static void msmsdcc_start_command(struct msmsdcc_host *host, struct mmc_command *cmd, u32 c)
  39. {
  40. msmsdcc_start_command_deferred(host, cmd, &c);
  41. msmsdcc_start_command_exec(host, cmd->arg, c);
  42. }
  43. static void msmsdcc_start_command_deferred(struct msmsdcc_host *host,struct mmc_command *cmd, u32 *c)
  44. cmd->opcode ----对应SD卡命令 ,如 CMD0:复位SD 卡

Linux SD卡驱动开发(六) —— SD卡启动过程总体分析相关推荐

  1. Linux SD卡驱动开发(五) —— SD 卡驱动分析Core补充篇

    Core层中有两个重要函数 mmc_alloc_host 用于构造host,前面已经学习过,这里不再阐述:另一个就是 mmc_add_host,用于注册host 前面探测函数s3cmci_probe, ...

  2. Linux SD卡驱动开发(二) —— SD 卡驱动分析HOST篇

    回顾一下前面的知识,MMC 子系统范围三个部分: HOST 部分是针对不同主机的驱动程序,这一部是驱动程序工程师需要根据自己的特点平台来完成的. CORE 部分: 这是整个MMC 的核心存,这部分完成 ...

  3. Linux SD卡驱动开发(四) —— SD 控制器之真正的硬件操作

    前面对SD卡控制器有了一个基本的介绍.其实SD控制器层更过的意义是为core层提供一种操作SD卡硬件的一种方法,当然不同的控制器对硬件控制的方法不尽相同,但是他们最终都能像core层提交一个统一的封装 ...

  4. linux sd卡驱动教程,Linux SD卡驱动开发(四) —— SD 控制器之真正的硬件操作

    前面对SD卡控制器有了一个基本的介绍.其实SD控制器层更过的意义是为core层提供一种操作SD卡硬件的一种方法,当然不同的控制器对硬件控制的方法不尽相同,但是他们最终都能像core层提交一个统一的封装 ...

  5. Linux 下wifi 驱动开发(三)—— SDIO接口WiFi驱动浅析

    SDIO-Wifi模块是基于SDIO接口的符合wifi无线网络标准的嵌入式模块,内置无线网络协议IEEE802.11协议栈以及TCP/IP协议栈,能够实现用户主平台数据通过SDIO口到无线网络之间的转 ...

  6. linux系统从开机到登陆界面的启动过程

    简述: 1.开机BIOS自检 2.MBR引导 3.grub引导菜单 4.加载内核kernel 5.启动init进程 6.读取inittab文件,执行rc.sysinit,rc等脚本 7.启动minge ...

  7. 【SD卡】关于DJYOS下SD卡驱动开发详解

    关于DJYOS下SD卡驱动开发详解 王建忠 2011/6/21 1      开发环境及说明 硬件平台:tq2440(CPU: s3c2440) 操作系统:DJYOS1.0.0 1.1    说明 T ...

  8. 2022年10月deepin 20.7.1 谈谈我们的N卡驱动开发心得

    deepin 20.7.1 即将到来之际,各位小伙伴在参与内测的过程中,有没有发现安装N卡驱动时不同寻常的变化呢? 20.7.1版本可以根据当前你机器的上NVIDIA显卡,自动匹配合适的闭源驱动版本进 ...

  9. Linux 中的驱动开发的初学者体会

    Linux 中的驱动开发的初学者体会 很多年前,心里就存下这样一个愿望.就是把Linux 的驱动开发搞清楚. 但是一开始上上这样的开发难度天大了,对着一堆的寄存器发愁. 于是就从简单的STM8,PIC ...

最新文章

  1. Linux命令sngre,linux gre隧道创建
  2. 浅谈Normalize.css
  3. 飘逸的python - 字典合并值相加
  4. 如何将网站前端如何添加登录密码访问_如何在Mac上查找保存的密码的所有信息...
  5. 人工智能+智能运维解决方案_人工智能驱动的解决方案可以提升您的项目管理水平
  6. kali linux安装谷歌拼音输入法(亲测可用)
  7. 推荐几个移动端前端UI框架
  8. Matlab矢量图导出PDF格式方式及LaTex图片排版技巧
  9. 扫描枪无限连服务器,无线扫描枪连接电脑的3个步骤
  10. java reflection singleton factorypattern
  11. 单反相机的传奇—佳能单反50年辉煌之路(连载十三)
  12. 技术债务_不要浪费时间跟踪技术债务
  13. MIT算法导论03-分治法
  14. 计算机网络设备网关属于固定资产,财政六大类常用固定资产分类及代码
  15. 用友安装时显示加密服务器,用友云主机指向加密服务器
  16. JOISC 2016 回转寿司
  17. 3_计算机网络_网路层-IP-子网划分-路由-ping
  18. 设计模式(个人理解)
  19. 用Python多线程实现低速处理器和高帧率摄像头的并行运行
  20. vmware workstation激活密钥和vmware workstation激活密钥。请单击确定关闭应用程序

热门文章

  1. 新RSS reader
  2. linux 系统下配置java环境变量
  3. kafka性能测试(转)KAFKA 0.8 PRODUCER PERFORMANCE
  4. 基于KVM的虚拟化研究及应用
  5. 数据可视化 信息可视化_更好的数据可视化的8个技巧
  6. python 数据科学 包_什么时候应该使用哪个Python数据科学软件包?
  7. opencv:卷积涉及的基础概念,Sobel边缘检测代码实现及卷积填充模式
  8. 程序编写经验教训_编写您永远都不会忘记的有效绩效评估的经验教训。
  9. 如何在iOS上运行React Native应用
  10. Linux Supervisor 守护进程基本配置