dm9k.c 23 KB

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