dm9000a框架原理图:

EEPROM Interface接口用于存放mac地址,InternalSRAM用于存放收发数据,MII部分把MAC部分与PHY部分连接起来通信,AUTO-MDIX用于自适应10/100M网络,在物理层上,MAC在PHY之下。

由dm9000a驱动可知,dm9000a驱动是用platform模型编写的,分析一个驱动源码都是从模块加载函数module_init()开始,而dm9000a加载函数是module_init(dm9000_init).

继而调用:

 
  1. static int __init

  2. dm9000_init(void)

  3. {

  4. printk(KERN_INFO "%s Ethernet Driver, V%s\n", CARDNAME, DRV_VERSION);

  5. return platform_driver_register(&dm9000_driver);

  6. }

dm9000_driver结构体:

 
  1. static struct platform_driver dm9000_driver = {

  2. .driver = {

  3. .name = "dm9000",//名字

  4. .owner = THIS_MODULE,

  5. },

  6. .probe = dm9000_probe,//模块加载后,调用probe函数

  7. .remove = __devexit_p(dm9000_drv_remove),

  8. .suspend = dm9000_drv_suspend,

  9. .resume = dm9000_drv_resume,

  10. };

模块加载之后,调用probe函数,如下:

 
  1. /*

  2. * Search DM9000 board, allocate space and register it

  3. */

  4. static int __devinit

  5. dm9000_probe(struct platform_device *pdev)

  6. {

  7. struct dm9000_plat_data *pdata = pdev->dev.platform_data;

  8. struct board_info *db; /* Point a board information structure */

  9. struct net_device *ndev;

  10. const unsigned char *mac_src;

  11. int ret = 0;

  12. int iosize;

  13. int i;

  14. u32 id_val;

  15. unsigned char ne_def_eth_mac_addr[]={0x00,0x12,0x34,0x56,0x80,0x49};

  16. /* ------------------------------------------------------------------------ */

  17. static void *bwscon;

  18. static void *gpfcon;

  19. static void *extint0;

  20. static void *intmsk;

  21. #define BWSCON (0x48000000)

  22. #define GPFCON (0x56000050)

  23. #define EXTINT0 (0x56000088)

  24. #define INTMSK (0x4A000008)

  25. bwscon=ioremap_nocache(BWSCON,0x0000004);

  26. gpfcon=ioremap_nocache(GPFCON,0x0000004);

  27. extint0=ioremap_nocache(EXTINT0,0x0000004);

  28. intmsk=ioremap_nocache(INTMSK,0x0000004);

  29. writel(readl(bwscon)|0xc0000,bwscon);

  30. writel( (readl(gpfcon) & ~(0x3 << 14)) | (0x2 << 14), gpfcon);

  31. writel( readl(gpfcon) | (0x1 << 7), gpfcon); // Disable pull-up

  32. writel( (readl(extint0) & ~(0xf << 28)) | (0x4 << 28), extint0); //rising edge

  33. writel( (readl(intmsk)) & ~0x80, intmsk);

  34. /* ------------------------------------------------------------------------ */

  35. /* Init network device */

  36. /* 分配eth网卡资源,私有数据区保存board_info*/

  37. ndev = alloc_etherdev(sizeof(struct board_info));

  38. if (!ndev) {

  39. dev_err(&pdev->dev, "could not allocate device.\n");

  40. return -ENOMEM;

  41. }

  42. SET_NETDEV_DEV(ndev, &pdev->dev);

  43. dev_dbg(&pdev->dev, "dm9000_probe()\n");

  44. /* setup board info structure 初始化为0*/

  45. db = ndev->priv;

  46. memset(db, 0, sizeof(*db));

  47. db->dev = &pdev->dev;

  48. db->ndev = ndev;

  49. /*初始化spinlock*/

  50. spin_lock_init(&db->lock);

  51. mutex_init(&db->addr_lock);

  52. /*提交一个任务给一个工作队列,你需要填充一个work_struct结构db->phy_poll*/

  53. INIT_DELAYED_WORK(&db->phy_poll, dm9000_poll_work);

  54. /*获取IO内存和中断资源*/

  55. db->addr_res = platform_get_resource(pdev, IORESOURCE_MEM, 0);

  56. db->data_res = platform_get_resource(pdev, IORESOURCE_MEM, 1);

  57. db->irq_res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);

  58. if (db->addr_res == NULL || db->data_res == NULL ||

  59. db->irq_res == NULL) {

  60. dev_err(db->dev, "insufficient resources\n");

  61. ret = -ENOENT;

  62. goto out;

  63. }

  64. /*映射到内核,并获得IO内存的虚拟地址,ioremap完成页表的建立,

  65. 不同于vmalloc,但是,它实际上不分配内存*/

  66. iosize = res_size(db->addr_res);

  67. db->addr_req = request_mem_region(db->addr_res->start, iosize,

  68. pdev->name);

  69. if (db->addr_req == NULL) {

  70. dev_err(db->dev, "cannot claim address reg area\n");

  71. ret = -EIO;

  72. goto out;

  73. }

  74. db->io_addr = ioremap(db->addr_res->start, iosize);

  75. if (db->io_addr == NULL) {

  76. dev_err(db->dev, "failed to ioremap address reg\n");

  77. ret = -EINVAL;

  78. goto out;

  79. }

  80. iosize = res_size(db->data_res);

  81. db->data_req = request_mem_region(db->data_res->start, iosize,

  82. pdev->name);

  83. if (db->data_req == NULL) {

  84. dev_err(db->dev, "cannot claim data reg area\n");

  85. ret = -EIO;

  86. goto out;

  87. }

  88. db->io_data = ioremap(db->data_res->start, iosize);

  89. if (db->io_data == NULL) {

  90. dev_err(db->dev, "failed to ioremap data reg\n");

  91. ret = -EINVAL;

  92. goto out;

  93. }

  94. /* fill in parameters for net-dev structure */

  95. /*获得网络设备的基地址*/

  96. ndev->base_addr = (unsigned long)db->io_addr;

  97. /*获得网络设备的中断号*/

  98. ndev->irq = db->irq_res->start;

  99. /* ensure at least we have a default set of IO routines */

  100. /*设置默认的IO函数*/

  101. dm9000_set_io(db, iosize);

  102. /*如果平台数据不为空 */

  103. if (pdata != NULL) {

  104. /* check to see if the driver wants to over-ride the

  105. * default IO width */

  106. if (pdata->flags & DM9000_PLATF_8BITONLY)

  107. dm9000_set_io(db, 1);

  108. if (pdata->flags & DM9000_PLATF_16BITONLY)

  109. dm9000_set_io(db, 2);

  110. if (pdata->flags & DM9000_PLATF_32BITONLY)

  111. dm9000_set_io(db, 4);

  112. /* check to see if there are any IO routine

  113. * over-rides */

  114. if (pdata->inblk != NULL)

  115. db->inblk = pdata->inblk;

  116. if (pdata->outblk != NULL)

  117. db->outblk = pdata->outblk;

  118. if (pdata->dumpblk != NULL)

  119. db->dumpblk = pdata->dumpblk;

  120. db->flags = pdata->flags;

  121. }

  122. #ifdef CONFIG_DM9000_FORCE_SIMPLE_PHY_POLL

  123. db->flags |= DM9000_PLATF_SIMPLE_PHY;

  124. #endif

  125. /*根据board info信息,复位DM9000芯片*/

  126. dm9000_reset(db);

  127. /*读取Vendor ID Register,Product ID Register中的值,与0x90000A46比较,如果相等,则说明是DM9000*/

  128. /* try multiple times, DM9000 sometimes gets the read wrong */

  129. for (i = 0; i < 8; i++) {

  130. id_val = ior(db, DM9000_VIDL);

  131. id_val |= (u32)ior(db, DM9000_VIDH) << 8;

  132. id_val |= (u32)ior(db, DM9000_PIDL) << 16;

  133. id_val |= (u32)ior(db, DM9000_PIDH) << 24;

  134. if (id_val == DM9000_ID)

  135. break;

  136. dev_err(db->dev, "read wrong id 0x%08x\n", id_val);

  137. }

  138. /*芯片的ID获取失败,驱动不匹配*/

  139. if (id_val != DM9000_ID) {

  140. dev_err(db->dev, "wrong id: 0x%08x\n", id_val);

  141. ret = -ENODEV;

  142. goto out;

  143. }

  144. /* Identify what type of DM9000 we are working on */

  145. /*读取Chip Revision Register中的值*/

  146. id_val = ior(db, DM9000_CHIPR);

  147. dev_dbg(db->dev, "dm9000 revision 0x%02x\n", id_val);

  148. switch (id_val) {

  149. case CHIPR_DM9000A:

  150. db->type = TYPE_DM9000A;

  151. break;

  152. case CHIPR_DM9000B:

  153. db->type = TYPE_DM9000B;

  154. break;

  155. default:

  156. dev_dbg(db->dev, "ID %02x => defaulting to DM9000E\n", id_val);

  157. db->type = TYPE_DM9000E;

  158. }

  159. /* from this point we assume that we have found a DM9000 */

  160. /* driver system function */

  161. /*设置部分net_device字段*/

  162. ether_setup(ndev);

  163. ndev->open = &dm9000_open;

  164. ndev->hard_start_xmit = &dm9000_start_xmit;

  165. ndev->tx_timeout = &dm9000_timeout;

  166. ndev->watchdog_timeo = msecs_to_jiffies(watchdog);

  167. ndev->stop = &dm9000_stop;

  168. ndev->set_multicast_list = &dm9000_hash_table;

  169. /*对ethtool支持的相关声明可在<linux/ethtool.h>中找到。

  170. 它的核心是一个ethtool_ops类型的结构,里边包含一个全部

  171. 的24个不同的方法来支持ethtool*/

  172. ndev->ethtool_ops = &dm9000_ethtool_ops;

  173. ndev->do_ioctl = &dm9000_ioctl;

  174. #ifdef CONFIG_NET_POLL_CONTROLLER

  175. ndev->poll_controller = &dm9000_poll_controller;

  176. #endif

  177. db->msg_enable = NETIF_MSG_LINK;

  178. db->mii.phy_id_mask = 0x1f;

  179. db->mii.reg_num_mask = 0x1f;

  180. db->mii.force_media = 0;

  181. db->mii.full_duplex = 0;

  182. db->mii.dev = ndev;

  183. db->mii.mdio_read = dm9000_phy_read;

  184. db->mii.mdio_write = dm9000_phy_write;

  185. /*MAC地址的源是eeprom*/

  186. mac_src = "eeprom";

  187. /* try reading the node address from the attached EEPROM */

  188. for (i = 0; i < 6; i += 2)

  189. dm9000_read_eeprom(db, i / 2, ndev->dev_addr+i);

  190. /*如果从eeprom中读取的地址无效,并且私有数据不为空,从platform_device的私有数据中获取dev_addr*/

  191. if (!is_valid_ether_addr(ndev->dev_addr) && pdata != NULL) {

  192. mac_src = "platform data";

  193. memcpy(ndev->dev_addr, pdata->dev_addr, 6);

  194. }

  195. /*如果地址依然无效,从PAR:物理地址(MAC)寄存器(Physical Address Register)中读取*/

  196. if (!is_valid_ether_addr(ndev->dev_addr)) {

  197. /* try reading from mac */

  198. mac_src = "chip";

  199. for (i = 0; i < 6; i++)

  200. //ndev->dev_addr[i] = ior(db, i+DM9000_PAR); // by bai

  201. ndev->dev_addr[i] = ne_def_eth_mac_addr[i];

  202. }

  203. /*查看以太网网卡设备地址是否有效*/

  204. if (!is_valid_ether_addr(ndev->dev_addr))

  205. dev_warn(db->dev, "%s: Invalid ethernet MAC address. Please "

  206. "set using ifconfig\n", ndev->name);

  207. /*将ndev保存到pdev->dev->driver_data中*/

  208. platform_set_drvdata(pdev, ndev);

  209. /*一切都初始化好后,注册网络设备*/

  210. ret = register_netdev(ndev);

  211. if (ret == 0) {

  212. DECLARE_MAC_BUF(mac);

  213. printk(KERN_INFO "%s: dm9000%c at %p,%p IRQ %d MAC: %s (%s)\n",

  214. ndev->name, dm9000_type_to_char(db->type),

  215. db->io_addr, db->io_data, ndev->irq,

  216. print_mac(mac, ndev->dev_addr), mac_src);

  217. }

  218. return 0;

  219. out:

  220. dev_err(db->dev, "not found (%d).\n", ret);

  221. /*失败时,释放资源*/

  222. dm9000_release_board(pdev, db);

  223. free_netdev(ndev);

  224. return ret;

  225. }

dm9000a驱动源码分析相关推荐

  1. (转)Linux设备驱动之HID驱动 源码分析

    //Linux设备驱动之HID驱动 源码分析 http://blog.chinaunix.net/uid-20543183-id-1930836.html HID是Human Interface De ...

  2. LCD驱动源码分析(s3cfb.c)

    1.驱动源码分析大致思路 (1)分析LCD驱动首先需要分析内核的帧缓冲子系统,因为LCD驱动就是按照帧缓冲子系统提供的注册接口来注册的: (2)内核帧缓冲子系统参考博客:<Linux 帧缓冲子系 ...

  3. 通用USB设备驱动源码分析

    通用USB设备驱动源码分析 Author:aaron 前段时间写了篇<qualcomm usb modem驱动小结>的文章, 描述了自己如何为高通的一个usb modem设备写驱动的过程, ...

  4. linux网卡驱动源码分析(一)

    linux网卡驱动源码分析(一) linux struct linux内核 网络 descriptor resources 转自http://blog.csdn.net/ustc_dylan/arti ...

  5. linux打印源码,Linux打印机驱动源码分析.pdf

    您所在位置:网站首页 > 海量文档 &nbsp>&nbsp计算机&nbsp>&nbsplinux/Unix相关 Linux打印机驱动源码分析.pdf1 ...

  6. hisi3516dv300芯片基于hwmon驱动框架的温度获取驱动源码分析

    1.内核hwmon驱动框架 参考博客:<内核hwmon驱动框架详解以及海思芯片温度驱动分析>: 2.驱动实现的效果 /sys/devices/virtual/hwmon/hwmon0 # ...

  7. 字符设备驱动基础篇1——简单的驱动源码分析

    以下内容源于朱有鹏嵌入式课程的学习,如有侵权,请告知删除. 参考资料:http://www.cnblogs.com/biaohc/p/6575074.html module_test.c代码 #inc ...

  8. 我的内核学习笔记10:Intel GPIO驱动源码分析

    本文对Intel e3800的GPIO驱动源码进行分析. 一.概述 1.1 内核配置 Intel e3800的GPIO在Linux内核中使用的驱动名为gpio_ich(为了行文方便,将对应的设备称为& ...

  9. java mysql 源码分析_JAVA JDBC(MySQL)驱动源码分析

    JAVA连接数据库是其众多功能中的一部分,主要有两种方式连接DataBase: 一种是采用JDBC-ODBC桥,另一种则是称之为纯驱动连接DataBase,第一种方式在大型项目中基本上不再使用,本系列 ...

  10. Linux kernel SPI源码分析之SPI设备驱动源码分析(linux kernel 5.18)

    SPI基础支持此处不再赘述,直接分析linux中的SPI驱动源码. 1.SPI设备驱动架构图 2.源码分析 本次分析基于kernel5.18,linux/drivers/spi/spidev.c 设备 ...

最新文章

  1. 哈尔滨工业大学计算机改专业课,哈尔滨工业大学计算机专业课 复试 2013HITCS
  2. Python安装hmmlearn
  3. 截取屏幕指定区域保存为BMP文件
  4. python 序列解包(解压缩)
  5. Visual Studio——理解多字节编码与Unicode码
  6. ADO.NET中在C/S模式中使用的连接池
  7. C语言的fgets 与 gets
  8. php生成wsdl文件,利用nusoap生成wsdl文件
  9. web跨域问题解决方案
  10. 程序设计基础基于C语言(第二版),程序设计基础_基于C语言(第2版)__课后习题参考答案.doc...
  11. CS231n课程笔记:神经网络笔记 2
  12. extjs 中formPanel提交到action后返回json数据到ext中,但是ext页面不进入sucess也不进入failure
  13. Python数据可视化之随机点图
  14. oracle经常考的题型是哪些,Oracle考试试题(带答案).doc
  15. aiml php,chatbot:基于 AIML 的 PHP 聊天机器人
  16. 2020安洵杯——EasyCM WriteUP
  17. 特征重要性排序--Permutation Importance
  18. Android中图片压缩方案详解
  19. HTTPS接口压测 --- hey工具
  20. java程序运行机制

热门文章

  1. 无涯社区分享:链网+DID,区块链应用破局的关键 | ArcBlock 活动
  2. Texlive安装宏包
  3. O2O电子商务盈利模式是什么 O2O电子商务模式遇到的困难有哪些?
  4. java 视频截图_Java Web 中使用ffmpeg实现视频转码、视频截图
  5. 小米node2红外_蓝牙,移动侦测,环境光,小米人体传感器2
  6. 开wifi微信定位服务器,企业微信wifi定位
  7. 论文笔记-Exploring Plain Vision Transformer Backbones for Object Detection
  8. 一款开源的支持离线的支持MarkDown的优秀笔记软件----思源笔记
  9. 安捷伦仪器仪表 - 程控总结
  10. 安川e1000中文说明书_安川(YASKAWA)变频器E1000说明书.pdf