dm9k.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734
  1. /*
  2. *********************************************************************************************************
  3. *
  4. * 模块名称 : DM9KAEP 底层驱动模块(For STM32F4XX, uip)
  5. * 文件名称 : dm9k_uip.c
  6. * 版 本 : V1.1
  7. * 说 明 : 这是硬件底层驱动程序的主文件。每个c文件可以 #include "bsp.h" 来包含所有的外设驱动模块。
  8. * bsp = Borad surport packet 板级支持包
  9. * 修改记录 :
  10. * 版本号 日期 作者 说明
  11. * V1.0 2013-03-01 armfly 正式发布
  12. * V1.1 2013-06-20 armfly 规范注释,添加必要说明
  13. *
  14. * Copyright (C), 2013-2014, 安富莱电子 www.armfly.com
  15. *
  16. *********************************************************************************************************
  17. */
  18. #include "dm9k.h"
  19. uint8_t SendPackOk = 0;
  20. uint8_t is_fsmc_ok = 0; /* 用于指示FSMC是否初始化 */
  21. void dm9k_debug_test(void);
  22. static void dm9k_fsmc(void);
  23. static void dm9k_initnic(void);
  24. /*
  25. *********************************************************************************************************
  26. * 函 数 名: dm9k_init
  27. * 功能说明: uIP 接口函数,初始化网卡. uIP 接口函数.
  28. * 形 参: 无
  29. * 返 回 值: 无
  30. *********************************************************************************************************
  31. */
  32. void dm9k_init(void)
  33. {
  34. dm9k_fsmc();
  35. is_fsmc_ok = 1;
  36. dm9k_initnic(); /* 配置DM9K */
  37. }
  38. /*
  39. *********************************************************************************************************
  40. * 函 数 名: dm9k_fsmc
  41. * 功能说明: 配置FSMC并口访问时序
  42. * 形 参: 无
  43. * 返 回 值: 无
  44. *********************************************************************************************************
  45. */
  46. static void dm9k_fsmc(void)
  47. {
  48. FSMC_NORSRAMInitTypeDef FSMC_NORSRAMInitStructure;
  49. FSMC_NORSRAMTimingInitTypeDef p;
  50. /*-- FSMC Configuration ------------------------------------------------------*/
  51. /*----------------------- SRAM Bank 3 ----------------------------------------*/
  52. /*-- FSMC Configuration ------------------------------------------------------*/
  53. p.FSMC_AddressSetupTime = 6; /* 设置为2会出错; 3正常 */
  54. p.FSMC_AddressHoldTime = 2;
  55. p.FSMC_DataSetupTime = 4; /* 设置为1出错,2正常 */
  56. p.FSMC_BusTurnAroundDuration = 1;
  57. p.FSMC_CLKDivision = 0;
  58. p.FSMC_DataLatency = 0;
  59. p.FSMC_AccessMode = FSMC_AccessMode_A;
  60. FSMC_NORSRAMInitStructure.FSMC_Bank = FSMC_Bank1_NORSRAM3;
  61. FSMC_NORSRAMInitStructure.FSMC_DataAddressMux = FSMC_DataAddressMux_Disable;
  62. FSMC_NORSRAMInitStructure.FSMC_MemoryType = FSMC_MemoryType_SRAM; // FSMC_MemoryType_PSRAM;
  63. FSMC_NORSRAMInitStructure.FSMC_MemoryDataWidth = FSMC_MemoryDataWidth_16b;
  64. FSMC_NORSRAMInitStructure.FSMC_BurstAccessMode = FSMC_BurstAccessMode_Disable;
  65. FSMC_NORSRAMInitStructure.FSMC_AsynchronousWait = FSMC_AsynchronousWait_Disable;
  66. FSMC_NORSRAMInitStructure.FSMC_WaitSignalPolarity = FSMC_WaitSignalPolarity_Low;
  67. FSMC_NORSRAMInitStructure.FSMC_WrapMode = FSMC_WrapMode_Disable;
  68. FSMC_NORSRAMInitStructure.FSMC_WaitSignalActive = FSMC_WaitSignalActive_BeforeWaitState;
  69. FSMC_NORSRAMInitStructure.FSMC_WriteOperation = FSMC_WriteOperation_Enable;
  70. FSMC_NORSRAMInitStructure.FSMC_WaitSignal = FSMC_WaitSignal_Disable;
  71. FSMC_NORSRAMInitStructure.FSMC_ExtendedMode = FSMC_ExtendedMode_Disable;
  72. FSMC_NORSRAMInitStructure.FSMC_WriteBurst = FSMC_WriteBurst_Disable;
  73. FSMC_NORSRAMInitStructure.FSMC_ReadWriteTimingStruct = &p;
  74. FSMC_NORSRAMInitStructure.FSMC_WriteTimingStruct = &p;
  75. FSMC_NORSRAMInit(&FSMC_NORSRAMInitStructure);
  76. /*!< Enable FSMC Bank1_SRAM3 Bank */
  77. FSMC_NORSRAMCmd(FSMC_Bank1_NORSRAM3, ENABLE);
  78. }
  79. /*
  80. *********************************************************************************************************
  81. * 函 数 名: dm9k_ReadReg
  82. * 功能说明: 读出DM9K指定寄存器的值
  83. * 形 参: reg 寄存器地址
  84. * 返 回 值: 寄存器值
  85. *********************************************************************************************************
  86. */
  87. uint8_t dm9k_ReadReg(uint8_t reg)
  88. {
  89. NET_REG_ADDR = reg;
  90. return (NET_REG_DATA);
  91. }
  92. /*
  93. *********************************************************************************************************
  94. * 函 数 名: dm9k_WriteReg
  95. * 功能说明: 读出DM9K指定寄存器的值
  96. * 形 参: reg :寄存器地址
  97. * writedata : 写入的数据
  98. * 返 回 值: 无
  99. *********************************************************************************************************
  100. */
  101. void dm9k_WriteReg(uint8_t reg, uint8_t writedata)
  102. {
  103. NET_REG_ADDR = reg;
  104. NET_REG_DATA = writedata;
  105. }
  106. /*
  107. *********************************************************************************************************
  108. * 函 数 名: dm9k_hash_table
  109. * 功能说明: 设置 DM9KA MAC 、 广播 、 多播 寄存器
  110. * 形 参: 无
  111. * 返 回 值: 无
  112. *********************************************************************************************************
  113. */
  114. void dm9k_hash_table()
  115. {
  116. uint8_t i;
  117. /* 设置 网卡 MAC 位置,来自於 MyHardware */
  118. dm9k_WriteReg(DM9K_REG_PAR, DM9K_MAC_ADDR0);
  119. dm9k_WriteReg(DM9K_REG_PAR + 1, DM9K_MAC_ADDR1);
  120. dm9k_WriteReg(DM9K_REG_PAR + 2, DM9K_MAC_ADDR2);
  121. dm9k_WriteReg(DM9K_REG_PAR + 3, DM9K_MAC_ADDR3);
  122. dm9k_WriteReg(DM9K_REG_PAR + 4, DM9K_MAC_ADDR4);
  123. dm9k_WriteReg(DM9K_REG_PAR + 5, DM9K_MAC_ADDR5);
  124. /* 清除 网卡多播设置 */
  125. for (i = 0; i < 8; i++)
  126. {
  127. dm9k_WriteReg(DM9K_REG_MAR + i, 0x00);
  128. }
  129. /* 设置 广播包 设置 */
  130. dm9k_WriteReg(DM9K_REG_MAR + 7, 0x80);
  131. }
  132. /*
  133. *********************************************************************************************************
  134. * 函 数 名: dm9k_reset
  135. * 功能说明: 软件复位DM9KAE
  136. * 形 参: 无
  137. * 返 回 值: 无
  138. *********************************************************************************************************
  139. */
  140. void dm9k_reset(void)
  141. {
  142. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET); /* 对 DM9KA 进行软件重置 */
  143. OSTimeDly(10); /* delay 10us */
  144. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET); /* 对 DM9KA 进行软件重置 */
  145. OSTimeDly(10); /* delay 10us */
  146. /* 基本记存器相关设置 */
  147. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_OFF); /* 开启内存自环模式 */
  148. dm9k_WriteReg(DM9K_REG_TCR2, DM9K_TCR2_SET); /* 设置 LED 显示模式1:全双工亮,半双工灭 */
  149. /* 清除多余资讯 */
  150. dm9k_WriteReg(DM9K_REG_NSR, 0x2c);
  151. dm9k_WriteReg(DM9K_REG_TCR, 0x00);
  152. dm9k_WriteReg(DM9K_REG_ISR, 0x3f);
  153. #ifdef DM9KA_FLOW_CONTROL
  154. dm9k_WriteReg(DM9K_REG_BPTR, DM9K_BPTR_SET); /* 半双工流控设置 */
  155. dm9k_WriteReg(DM9K_REG_FCTR, DM9K_FCTR_SET); /* 全双工流控设置 */
  156. dm9k_WriteReg(DM9K_REG_FCR, DM9K_FCR_SET); /* 开启流控设置 */
  157. #endif
  158. #ifdef DM9KA_UPTO_100M
  159. /* DM9KA无此寄存器 */
  160. dm9k_WriteReg(DM9K_REG_OTCR, DM9K_OTCR_SET); /* 工作频率到 100Mhz 设置 */
  161. #endif
  162. #ifdef Rx_Int_enable
  163. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_SET); /* 开启 中断模式 */
  164. #else
  165. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_OFF); /* 关闭 中断模式 */
  166. #endif
  167. dm9k_WriteReg(DM9K_REG_RCR, DM9K_RCR_SET); /* 开启 接收工能 */
  168. SendPackOk = 0;
  169. }
  170. /*
  171. *********************************************************************************************************
  172. * 函 数 名: dm9k_phy_write
  173. * 功能说明: 软件复位DM9KAE
  174. * 形 参: phy_reg : PHY寄存器地址
  175. * writedata : 写入的数据
  176. * 返 回 值: 无
  177. *********************************************************************************************************
  178. */
  179. void dm9k_phy_write(uint8_t phy_reg, uint16_t writedata)
  180. {
  181. /* 设置写入 PHY 寄存器的位置 */
  182. dm9k_WriteReg(DM9K_REG_EPAR, phy_reg | DM9K_PHY);
  183. /* 设置写入 PHY 寄存器的值 */
  184. dm9k_WriteReg(DM9K_REG_EPDRH, (writedata >> 8) & 0xff);
  185. dm9k_WriteReg(DM9K_REG_EPDRL, writedata & 0xff);
  186. dm9k_WriteReg(DM9K_REG_EPCR, 0x0a); /* 将资料写入 PHY 寄存器 */
  187. while (dm9k_ReadReg(DM9K_REG_EPCR) & 0x01)
  188. ; /* 查寻是否执行结束 */
  189. dm9k_WriteReg(DM9K_REG_EPCR, 0x08); /* 清除写入命令 */
  190. }
  191. /*
  192. *********************************************************************************************************
  193. * 函 数 名: dm9k_initnic
  194. * 功能说明: 配置DM9KAE芯片(初始化)
  195. * 形 参: 无
  196. * 返 回 值: 无
  197. *********************************************************************************************************
  198. */
  199. static void dm9k_initnic(void)
  200. {
  201. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET); /* 对 DM9KA 进行软件重置 */
  202. OSTimeDly(10); /* delay 10us */
  203. dm9k_hash_table(); /* 设置 DM9KA MAC 及 多播*/
  204. dm9k_reset(); /* 进行 DM9KA 软件设置 */
  205. dm9k_WriteReg(DM9K_REG_GPR, DM9K_PHY_OFF); /* 关闭 PHY ,进行 PHY 设置*/
  206. dm9k_phy_write(0x00, 0x8000); /* 重置 PHY 的寄存器 */
  207. #ifdef DM9KA_FLOW_CONTROL
  208. dm9k_phy_write(0x04, 0x01e1 | 0x0400); /* 设置 自适应模式相容表 */
  209. #else
  210. dm9k_phy_write(0x04, 0x01e1); /* 设置 自适应模式相容表 */
  211. #endif
  212. // dm9k_phy_write(0x00, 0x1000); /* 设置 基本连接模式 */
  213. /* 连接模式设置
  214. 0x0000 : 固定10M半双工
  215. 0x0100 : 固定10M全双工
  216. 0x2000 : 固定100M半双工
  217. 0x2100 : 固定100M全双工
  218. 0x1000 : 自适应模式
  219. */
  220. dm9k_phy_write(0x00, 0x1000); /* 设置 基本连接模式 */
  221. dm9k_WriteReg(DM9K_REG_GPR, DM9K_PHY_ON); /* 结束 PHY 设置, 开启 PHY */
  222. // dm9k_debug_test();
  223. }
  224. /*
  225. *********************************************************************************************************
  226. * 函 数 名: dm9k_receive_packet
  227. * 功能说明: 配置DM9KAE芯片(初始化)
  228. * 形 参: _uip_buf : 接收结果存放的缓冲区指针
  229. * 返 回 值: > 0 表示接收的数据长度, 0表示没有数据
  230. *********************************************************************************************************
  231. */
  232. uint16_t dm9k_receive_packet(uint8_t *_uip_buf)
  233. {
  234. uint16_t ReceiveLength;
  235. uint16_t *ReceiveData;
  236. uint8_t rx_int_count = 0;
  237. uint8_t rx_checkbyte;
  238. uint16_t rx_status, rx_length;
  239. uint8_t jump_packet;
  240. uint16_t i;
  241. uint16_t calc_len;
  242. uint16_t calc_MRR;
  243. do
  244. {
  245. ReceiveLength = 0; /* 清除接收的长度 */
  246. ReceiveData = (uint16_t *)_uip_buf;
  247. jump_packet = 0; /* 清除跳包动作 */
  248. dm9k_ReadReg(DM9K_REG_MRCMDX); /* 读取内存数据,地址不增加 */
  249. /* 计算内存数据位置 */
  250. calc_MRR = (dm9k_ReadReg(DM9K_REG_MRRH) << 8) + dm9k_ReadReg(DM9K_REG_MRRL);
  251. rx_checkbyte = dm9k_ReadReg(DM9K_REG_MRCMDX); /* */
  252. if (rx_checkbyte == DM9K_PKT_RDY) /* 取 */
  253. {
  254. /* 读取封包相关资讯 及 长度 */
  255. NET_REG_ADDR = DM9K_REG_MRCMD;
  256. rx_status = NET_REG_DATA;
  257. rx_length = NET_REG_DATA;
  258. /* 若收到超过系统可承受的封包,此包跳过 */
  259. if (rx_length > Max_Ethernet_Lenth)
  260. jump_packet = 1;
  261. #ifdef Broadcast_Jump
  262. /* 若收到的广播或多播包超过特定长度,此包跳过 */
  263. if (rx_status & 0x4000)
  264. {
  265. if (rx_length > Max_Broadcast_Lenth)
  266. jump_packet = 1;
  267. }
  268. #endif
  269. /* 计算下一个包的指针位 , 若接收长度为奇数,需加一对齐偶字节。*/
  270. /* 若是超过 0x3fff ,则需回归绕到 0x0c00 起始位置 */
  271. calc_MRR += (rx_length + 4);
  272. if (rx_length & 0x01)
  273. calc_MRR++;
  274. if (calc_MRR > 0x3fff)
  275. calc_MRR -= 0x3400;
  276. if (jump_packet == 0x01)
  277. {
  278. /* 将指针移到下一个包的包头位置 */
  279. dm9k_WriteReg(DM9K_REG_MRRH, (calc_MRR >> 8) & 0xff);
  280. dm9k_WriteReg(DM9K_REG_MRRL, calc_MRR & 0xff);
  281. continue;
  282. }
  283. /* 开始将内存的资料搬到到系统中,每次移动一个 word */
  284. calc_len = (rx_length + 1) >> 1;
  285. for (i = 0; i < calc_len; i++)
  286. ReceiveData[i] = NET_REG_DATA;
  287. /* 将包长回报给 TCP/IP 上层,并减去最後 4 BYTE 的 CRC-32 检核码 */
  288. ReceiveLength = rx_length - 4;
  289. rx_int_count++; /* 累计收包次数 */
  290. #ifdef FifoPointCheck
  291. if (calc_MRR != ((dm9k_ReadReg(DM9K_REG_MRRH) << 8) + dm9k_ReadReg(DM9K_REG_MRRL)))
  292. {
  293. #ifdef Point_Error_Reset
  294. dm9k_reset(); /* 若是指针出错,重置 */
  295. return ReceiveLength;
  296. #endif
  297. /*若是指针出错,将指针移到下一个包的包头位置 */
  298. dm9k_WriteReg(DM9K_REG_MRRH, (calc_MRR >> 8) & 0xff);
  299. dm9k_WriteReg(DM9K_REG_MRRL, calc_MRR & 0xff);
  300. }
  301. #endif
  302. return ReceiveLength;
  303. }
  304. else
  305. {
  306. if (rx_checkbyte == DM9K_PKT_NORDY) /* 未收到包 */
  307. {
  308. dm9k_WriteReg(DM9K_REG_ISR, 0x3f); /* */
  309. }
  310. else
  311. {
  312. dm9k_reset(); /* 接收指针出错,重置 */
  313. }
  314. return (0);
  315. }
  316. } while (rx_int_count < Max_Int_Count); /* 是否超过最多接收封包计数 */
  317. return 0;
  318. }
  319. /*
  320. *********************************************************************************************************
  321. * 函 数 名: dm9k_send_packet
  322. * 功能说明: 发送一包数据
  323. * 形 参: p_char : 发送数据缓冲区
  324. * 返 回 值: length : 数据长度
  325. *********************************************************************************************************
  326. */
  327. void dm9k_send_packet(uint8_t *p_char, uint16_t length)
  328. {
  329. uint16_t SendLength = length;
  330. uint16_t *SendData = (uint16_t *)p_char;
  331. uint16_t i;
  332. uint16_t calc_len;
  333. __IO uint16_t calc_MWR;
  334. /* 检查 DM9KA 是否还在传送中!若是等待直到传送结束 */
  335. if (SendPackOk == Max_Send_Pack)
  336. {
  337. while (dm9k_ReadReg(DM9K_REG_TCR) & DM9K_TCR_SET)
  338. {
  339. OSTimeDly(5);
  340. }
  341. SendPackOk = 0;
  342. }
  343. SendPackOk++; /* 设置传送计数 */
  344. #ifdef FifoPointCheck
  345. /* 计算下一个传送的指针位 , 若接收长度为奇数,需加一对齐偶字节。*/
  346. /* 若是超过 0x0bff ,则需回归绕到 0x0000 起始位置 */
  347. calc_MWR = (dm9k_ReadReg(DM9K_REG_MWRH) << 8) + dm9k_ReadReg(DM9K_REG_MWRL);
  348. calc_MWR += SendLength;
  349. if (SendLength & 0x01)
  350. calc_MWR++;
  351. if (calc_MWR > 0x0bff)
  352. calc_MWR -= 0x0c00;
  353. #endif
  354. dm9k_WriteReg(DM9K_REG_TXPLH, (SendLength >> 8) & 0xff); /* 设置传送封包的长度 */
  355. dm9k_WriteReg(DM9K_REG_TXPLL, SendLength & 0xff);
  356. /* 开始将系统的资料搬到到内存中,每次移动一个 word */
  357. NET_REG_ADDR = DM9K_REG_MWCMD;
  358. calc_len = (SendLength + 1) >> 1;
  359. for (i = 0; i < calc_len; i++)
  360. NET_REG_DATA = SendData[i];
  361. dm9k_WriteReg(DM9K_REG_TCR, DM9K_TCR_SET); /* 进行传送 */
  362. #ifdef FifoPointCheck
  363. if (calc_MWR != ((dm9k_ReadReg(DM9K_REG_MWRH) << 8) + dm9k_ReadReg(DM9K_REG_MWRL)))
  364. {
  365. #ifdef Point_Error_Reset
  366. /* 若是指针出错,等待此一封包送完 , 之後进行重置 */
  367. while (dm9k_ReadReg(DM9K_REG_TCR) & DM9K_TCR_SET)
  368. bsp_DelayUS(5);
  369. dm9k_reset();
  370. return;
  371. #endif
  372. /*若是指针出错,将指针移到下一个传送包的包头位置 */
  373. dm9k_WriteReg(DM9K_REG_MWRH, (calc_MWR >> 8) & 0xff);
  374. dm9k_WriteReg(DM9K_REG_MWRL, calc_MWR & 0xff);
  375. }
  376. #endif
  377. return;
  378. }
  379. /*******************************************************************************
  380. * 函数名: etherdev_send
  381. * 参 数: p_char : 数据缓冲区
  382. * length : 数据长度
  383. * 返 回: 无
  384. * 功 能: uIP 接口函数,发送一包数据
  385. */
  386. void etherdev_send(uint8_t *p_char, uint16_t length)
  387. {
  388. dm9k_send_packet(p_char, length);
  389. }
  390. uint16_t etherdev_read(uint8_t *p_char)
  391. {
  392. return dm9k_receive_packet(p_char);
  393. }
  394. /*******************************************************************************
  395. * 函数名: etherdev_chkmedia
  396. * 参 数: p_char : 数据缓冲区
  397. * length : 数据长度
  398. * 返 回: 无
  399. * 功 能: uIP 接口函数, 检测网络连接状态
  400. */
  401. void etherdev_chkmedia(void)
  402. {
  403. // uint8_t status;
  404. while (!(dm9k_ReadReg(DM9K_REG_NSR) & DM9K_PHY))
  405. {
  406. OSTimeDly(2000);
  407. }
  408. }
  409. /*
  410. *********************************************************************************************************
  411. * 函 数 名: dm9k_interrupt
  412. * 功能说明: 中断处理函数 (webserver例程未使用中断)
  413. * 形 参: 无
  414. * 返 回 值: 无
  415. *********************************************************************************************************
  416. */
  417. void dm9k_interrupt(void)
  418. {
  419. uint8_t save_reg;
  420. uint16_t isr_status;
  421. save_reg = NET_REG_ADDR; /* 暂存所使用的位置 */
  422. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_OFF); /* 关闭 DM9KA 中断 */
  423. isr_status = dm9k_ReadReg(DM9K_REG_ISR); /* 取得中断产生值 */
  424. if (isr_status & DM9K_RX_INTR)
  425. { /* 检查是否为接收中断 */
  426. // dm9k_receive_packet(); /* 执行接收处理程序 */
  427. }
  428. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_SET); /* 开启 DM9KA 中断 */
  429. NET_REG_ADDR = save_reg; /* 回复所使用的位置 */
  430. }
  431. /*******************************************************************************
  432. * 函数名: dm9k_ReadID
  433. * 参 数: 无
  434. * 返 回: 无
  435. * 功 能: 读取芯片ID
  436. */
  437. uint32_t dm9k_ReadID(void)
  438. {
  439. uint8_t vid1, vid2, pid1, pid2;
  440. if (is_fsmc_ok == 0)
  441. {
  442. dm9k_fsmc();
  443. is_fsmc_ok = 1;
  444. }
  445. vid1 = dm9k_ReadReg(DM9K_REG_VID_L) & 0xFF;
  446. vid2 = dm9k_ReadReg(DM9K_REG_VID_H) & 0xFF;
  447. pid1 = dm9k_ReadReg(DM9K_REG_PID_L) & 0xFF;
  448. pid2 = dm9k_ReadReg(DM9K_REG_PID_H) & 0xFF;
  449. return (vid2 << 24) | (vid1 << 16) | (pid2 << 8) | pid1;
  450. }
  451. uint8_t dm9k_linkstat(void)
  452. {
  453. uint8_t linkchanged = 0;
  454. linkchanged = (dm9k_ReadReg(DM9K_PHY_BMSR) >> 6) & 0x01;
  455. return linkchanged;
  456. }
  457. #if 0
  458. /*
  459. *********************************************************************************************************
  460. * 函 数 名: dm9k_debug_test
  461. * 功能说明: 测试DM9KAE的函数,用于排错
  462. * 形 参: 无
  463. * 返 回 值: 无
  464. *********************************************************************************************************
  465. */
  466. void dm9k_debug_test(void)
  467. {
  468. uint32_t check_device;
  469. uint8_t check_iomode;
  470. uint8_t check_reg_fail = 0;
  471. uint8_t check_fifo_fail = 0;
  472. uint16_t i;
  473. uint16_t j;
  474. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET); /* 对 DM9KA 进行软件重置 */
  475. OSTimeDly(10); /* delay 10us */
  476. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET); /* 对 DM9KA 进行软件重置 */
  477. OSTimeDly(10); /* delay 10us */
  478. check_device = dm9k_ReadReg(DM9K_REG_VID_L);
  479. check_device |= dm9k_ReadReg(DM9K_REG_VID_H) << 8;
  480. check_device |= dm9k_ReadReg(DM9K_REG_PID_L) << 16;
  481. check_device |= dm9k_ReadReg(DM9K_REG_PID_H) << 24;
  482. if(check_device != 0x90000A46)
  483. {
  484. printk("DM9K_DEBUG ==> DEIVCE NOT FOUND, SYSTEM HOLD !!\n");
  485. while(1);
  486. }
  487. else
  488. {
  489. printk("DM9K_DEBUG ==> DEIVCE FOUND !!\n");
  490. }
  491. check_iomode = dm9k_ReadReg(DM9K_REG_ISR) >> 6;
  492. if(check_iomode != DM9K_WORD_MODE)
  493. {
  494. printk("DM9K_DEBUG ==> DEIVCE NOT WORD MODE, SYSTEM HOLD !!\n");
  495. while(1);
  496. }
  497. else
  498. {
  499. printk("DM9K_DEBUG ==> DEIVCE IS WORD MODE !!\n");
  500. }
  501. printk("DM9K_DEBUG ==> REGISTER R/W TEST !!\n");
  502. NET_REG_ADDR = DM9K_REG_MAR;
  503. for(i = 0; i < 0x0100; i++)
  504. {
  505. NET_REG_DATA = i;
  506. if(i != (NET_REG_DATA & 0xff))
  507. {
  508. printk(" > error W %02x , R %02x \n", i , NET_REG_DATA);
  509. check_reg_fail = 1;
  510. }
  511. }
  512. if(check_reg_fail)
  513. {
  514. printk("DM9K_DEBUG ==> REGISTER R/W FAIL, SYSTEM HOLD !!\n");
  515. while(1);
  516. }
  517. printk("DM9K_DEBUG ==> FIFO R/W TEST !!\n");
  518. printk("DM9K_DEBUG ==> FIFO WRITE START POINT 0x%02x%02x \n",
  519. dm9k_ReadReg(DM9K_REG_MWRH), dm9k_ReadReg(DM9K_REG_MWRL));
  520. NET_REG_ADDR = DM9K_REG_MWCMD;
  521. for(i = 0; i < 0x1000; i++)
  522. NET_REG_DATA = ((i & 0xff) * 0x0101);
  523. printk("DM9K_DEBUG ==> FIFO WRITE END POINT 0x%02x%02x \n",
  524. dm9k_ReadReg(DM9K_REG_MWRH), dm9k_ReadReg(DM9K_REG_MWRL));
  525. if((dm9k_ReadReg(DM9K_REG_MWRH) != 0x20) || (dm9k_ReadReg(DM9K_REG_MWRL) != 0x00))
  526. {
  527. printk("DM9K_DEBUG ==> FIFO WRITE FAIL, SYSTEM HOLD !!\n");
  528. while(1);
  529. }
  530. dm9k_ReadReg(DM9K_REG_MRCMDX);
  531. printk("DM9K_DEBUG ==> FIFO READ START POINT 0x%02x%02x \n",
  532. dm9k_ReadReg(DM9K_REG_MRRH), dm9k_ReadReg(DM9K_REG_MRRL));
  533. dm9k_ReadReg(DM9K_REG_MRCMDX);
  534. NET_REG_ADDR = DM9K_REG_MRCMD;
  535. for(i = 0; i < 0x1000; i++)
  536. {
  537. j = NET_REG_DATA;
  538. if(((i & 0xff) * 0x0101) != j)
  539. {
  540. //printk(" > error W %04x , R %04x \n",
  541. // ((i & 0xff) * 0x0101) , j);
  542. check_fifo_fail = 1;
  543. }
  544. }
  545. printk("DM9K_DEBUG ==> FIFO READ END POINT 0x%02x%02x \n",
  546. dm9k_ReadReg(DM9K_REG_MRRH), dm9k_ReadReg(DM9K_REG_MRRL));
  547. if((dm9k_ReadReg(DM9K_REG_MRRH) != 0x20) || (dm9k_ReadReg(DM9K_REG_MRRL) != 0x00))
  548. {
  549. printk("DM9K_DEBUG ==> FIFO WRITE FAIL, SYSTEM HOLD !!\n");
  550. while(1);
  551. }
  552. if(check_fifo_fail)
  553. {
  554. printk("DM9K_DEBUG ==> FIFO R/W DATA FAIL, SYSTEM HOLD !!\n");
  555. while(1);
  556. }
  557. printk("DM9K_DEBUG ==> PACKET SEND & INT TEST !! \n");
  558. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET);
  559. OSTimeDly(10);
  560. dm9k_WriteReg(DM9K_REG_NCR, DM9K_REG_RESET);
  561. OSTimeDly(10);
  562. dm9k_WriteReg(DM9K_REG_IMR, DM9K_IMR_OFF | DM9K_TX_INTR);
  563. dm9k_WriteReg(DM9K_REG_TXPLH, 0x01);
  564. dm9k_WriteReg(DM9K_REG_TXPLL, 0x00);
  565. do
  566. {
  567. dm9k_WriteReg(DM9K_REG_ISR, DM9K_TX_INTR);
  568. printk("DM9K_DEBUG ==> INT PIN IS OFF\n");
  569. NET_REG_ADDR = DM9K_REG_MWCMD;
  570. for(i = 0; i < (0x0100 / 2); i++)
  571. {
  572. if(i < 3)
  573. NET_REG_DATA = 0xffff;
  574. else
  575. NET_REG_DATA = i * 0x0101;
  576. }
  577. printk("DM9K_DEBUG ==> PACKET IS SEND \n");
  578. dm9k_WriteReg(DM9K_REG_TCR, DM9K_TCR_SET);
  579. while(dm9k_ReadReg(DM9K_REG_TCR) & DM9K_TCR_SET) bsp_DelayUS (5);
  580. if(dm9k_ReadReg(DM9K_REG_ISR) & DM9K_TX_INTR)
  581. printk("DM9K_DEBUG ==> INT PIN IS ACTIVE \n");
  582. else
  583. printk("DM9K_DEBUG ==> INT PIN IS NOT ACTIVE \n");
  584. for(i = 0; i < 10; i++)
  585. bsp_DelayUS(1000);
  586. }while(1);
  587. }
  588. /*
  589. *********************************************************************************************************
  590. * 函 数 名: etherdev_chkmedia
  591. * 功能说明: 检测网络连接状态
  592. * 形 参: 无
  593. * 返 回 值: 无
  594. *********************************************************************************************************
  595. */
  596. void etherdev_chkmedia(void)
  597. {
  598. // uint8_t status;
  599. while(!(dm9k_ReadReg(DM9K_REG_NSR) & DM9K_PHY))
  600. {
  601. bsp_DelayUS(2000);
  602. }
  603. }
  604. /*******************************************************************************
  605. * 函数名: etherdev_poll
  606. * 参 数: 无
  607. * 返 回: 无
  608. * 功 能: uIP 接口函数, 采用查询方式接收一个IP包
  609. */
  610. /*
  611. etherdev_poll()
  612. This function will read an entire IP packet into the uip_buf.
  613. If it must wait for more than 0.5 seconds, it will return with
  614. the return value 0. Otherwise, when a full packet has been read
  615. into the uip_buf buffer, the length of the packet is returned.
  616. */
  617. uint16_t etherdev_poll(void)
  618. {
  619. uint16_t bytes_read = 0;
  620. #if 0
  621. /* tick_count threshold should be 12 for 0.5 sec bail-out
  622. One second (24) worked better for me, but socket recycling
  623. is then slower. I set UIP_TIME_WAIT_TIMEOUT 60 in uipopt.h
  624. to counter this. Retransmission timing etc. is affected also. */
  625. while ((!(bytes_read = etherdev_read())) && (timer0_tick() < 12)) continue;
  626. timer0_reset();
  627. #endif
  628. return bytes_read;
  629. }
  630. #endif