ftmac100.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. /*
  2. * Faraday FTMAC100 Ethernet
  3. *
  4. * (C) Copyright 2009 Faraday Technology
  5. * Po-Yu Chuang <ratbert@faraday-tech.com>
  6. *
  7. * SPDX-License-Identifier: GPL-2.0+
  8. */
  9. #include <config.h>
  10. #include <common.h>
  11. #include <malloc.h>
  12. #include <net.h>
  13. #include <linux/io.h>
  14. #include "ftmac100.h"
  15. #ifdef CONFIG_DM_ETH
  16. #include <dm.h>
  17. DECLARE_GLOBAL_DATA_PTR;
  18. #endif
  19. #define ETH_ZLEN 60
  20. struct ftmac100_data {
  21. struct ftmac100_txdes txdes[1];
  22. struct ftmac100_rxdes rxdes[PKTBUFSRX];
  23. int rx_index;
  24. const char *name;
  25. phys_addr_t iobase;
  26. };
  27. /*
  28. * Reset MAC
  29. */
  30. static void ftmac100_reset(struct ftmac100_data *priv)
  31. {
  32. struct ftmac100 *ftmac100 = (struct ftmac100 *)priv->iobase;
  33. debug ("%s()\n", __func__);
  34. writel (FTMAC100_MACCR_SW_RST, &ftmac100->maccr);
  35. while (readl (&ftmac100->maccr) & FTMAC100_MACCR_SW_RST)
  36. ;
  37. }
  38. /*
  39. * Set MAC address
  40. */
  41. static void ftmac100_set_mac(struct ftmac100_data *priv ,
  42. const unsigned char *mac)
  43. {
  44. struct ftmac100 *ftmac100 = (struct ftmac100 *)priv->iobase;
  45. unsigned int maddr = mac[0] << 8 | mac[1];
  46. unsigned int laddr = mac[2] << 24 | mac[3] << 16 | mac[4] << 8 | mac[5];
  47. debug ("%s(%x %x)\n", __func__, maddr, laddr);
  48. writel (maddr, &ftmac100->mac_madr);
  49. writel (laddr, &ftmac100->mac_ladr);
  50. }
  51. /*
  52. * Disable MAC
  53. */
  54. static void _ftmac100_halt(struct ftmac100_data *priv)
  55. {
  56. struct ftmac100 *ftmac100 = (struct ftmac100 *)priv->iobase;
  57. debug ("%s()\n", __func__);
  58. writel (0, &ftmac100->maccr);
  59. }
  60. /*
  61. * Initialize MAC
  62. */
  63. static int _ftmac100_init(struct ftmac100_data *priv, unsigned char enetaddr[6])
  64. {
  65. struct ftmac100 *ftmac100 = (struct ftmac100 *)priv->iobase;
  66. struct ftmac100_txdes *txdes = priv->txdes;
  67. struct ftmac100_rxdes *rxdes = priv->rxdes;
  68. unsigned int maccr;
  69. int i;
  70. debug ("%s()\n", __func__);
  71. ftmac100_reset(priv);
  72. /* set the ethernet address */
  73. ftmac100_set_mac(priv, enetaddr);
  74. /* disable all interrupts */
  75. writel (0, &ftmac100->imr);
  76. /* initialize descriptors */
  77. priv->rx_index = 0;
  78. txdes[0].txdes1 = FTMAC100_TXDES1_EDOTR;
  79. rxdes[PKTBUFSRX - 1].rxdes1 = FTMAC100_RXDES1_EDORR;
  80. for (i = 0; i < PKTBUFSRX; i++) {
  81. /* RXBUF_BADR */
  82. rxdes[i].rxdes2 = (unsigned int)net_rx_packets[i];
  83. rxdes[i].rxdes1 |= FTMAC100_RXDES1_RXBUF_SIZE (PKTSIZE_ALIGN);
  84. rxdes[i].rxdes0 = FTMAC100_RXDES0_RXDMA_OWN;
  85. }
  86. /* transmit ring */
  87. writel ((unsigned int)txdes, &ftmac100->txr_badr);
  88. /* receive ring */
  89. writel ((unsigned int)rxdes, &ftmac100->rxr_badr);
  90. /* poll receive descriptor automatically */
  91. writel (FTMAC100_APTC_RXPOLL_CNT (1), &ftmac100->aptc);
  92. /* enable transmitter, receiver */
  93. maccr = FTMAC100_MACCR_XMT_EN |
  94. FTMAC100_MACCR_RCV_EN |
  95. FTMAC100_MACCR_XDMA_EN |
  96. FTMAC100_MACCR_RDMA_EN |
  97. FTMAC100_MACCR_CRC_APD |
  98. FTMAC100_MACCR_ENRX_IN_HALFTX |
  99. FTMAC100_MACCR_RX_RUNT |
  100. FTMAC100_MACCR_RX_BROADPKT;
  101. writel (maccr, &ftmac100->maccr);
  102. return 0;
  103. }
  104. /*
  105. * Free receiving buffer
  106. */
  107. static int _ftmac100_free_pkt(struct ftmac100_data *priv)
  108. {
  109. struct ftmac100_rxdes *curr_des;
  110. curr_des = &priv->rxdes[priv->rx_index];
  111. /* release buffer to DMA */
  112. curr_des->rxdes0 |= FTMAC100_RXDES0_RXDMA_OWN;
  113. priv->rx_index = (priv->rx_index + 1) % PKTBUFSRX;
  114. return 0;
  115. }
  116. /*
  117. * Receive a data block via Ethernet
  118. */
  119. static int __ftmac100_recv(struct ftmac100_data *priv)
  120. {
  121. struct ftmac100_rxdes *curr_des;
  122. unsigned short rxlen;
  123. curr_des = &priv->rxdes[priv->rx_index];
  124. if (curr_des->rxdes0 & FTMAC100_RXDES0_RXDMA_OWN)
  125. return 0;
  126. if (curr_des->rxdes0 & (FTMAC100_RXDES0_RX_ERR |
  127. FTMAC100_RXDES0_CRC_ERR |
  128. FTMAC100_RXDES0_FTL |
  129. FTMAC100_RXDES0_RUNT |
  130. FTMAC100_RXDES0_RX_ODD_NB)) {
  131. return 0;
  132. }
  133. rxlen = FTMAC100_RXDES0_RFL (curr_des->rxdes0);
  134. debug ("%s(): RX buffer %d, %x received\n",
  135. __func__, priv->rx_index, rxlen);
  136. return rxlen;
  137. }
  138. /*
  139. * Send a data block via Ethernet
  140. */
  141. static int _ftmac100_send(struct ftmac100_data *priv, void *packet, int length)
  142. {
  143. struct ftmac100 *ftmac100 = (struct ftmac100 *)priv->iobase;
  144. struct ftmac100_txdes *curr_des = priv->txdes;
  145. ulong start;
  146. if (curr_des->txdes0 & FTMAC100_TXDES0_TXDMA_OWN) {
  147. debug ("%s(): no TX descriptor available\n", __func__);
  148. return -1;
  149. }
  150. debug ("%s(%x, %x)\n", __func__, (int)packet, length);
  151. length = (length < ETH_ZLEN) ? ETH_ZLEN : length;
  152. /* initiate a transmit sequence */
  153. curr_des->txdes2 = (unsigned int)packet; /* TXBUF_BADR */
  154. curr_des->txdes1 &= FTMAC100_TXDES1_EDOTR;
  155. curr_des->txdes1 |= FTMAC100_TXDES1_FTS |
  156. FTMAC100_TXDES1_LTS |
  157. FTMAC100_TXDES1_TXBUF_SIZE (length);
  158. curr_des->txdes0 = FTMAC100_TXDES0_TXDMA_OWN;
  159. /* start transmit */
  160. writel (1, &ftmac100->txpd);
  161. /* wait for transfer to succeed */
  162. start = get_timer(0);
  163. while (curr_des->txdes0 & FTMAC100_TXDES0_TXDMA_OWN) {
  164. if (get_timer(start) >= 5) {
  165. debug ("%s(): timed out\n", __func__);
  166. return -1;
  167. }
  168. }
  169. debug ("%s(): packet sent\n", __func__);
  170. return 0;
  171. }
  172. #ifndef CONFIG_DM_ETH
  173. /*
  174. * disable transmitter, receiver
  175. */
  176. static void ftmac100_halt(struct eth_device *dev)
  177. {
  178. struct ftmac100_data *priv = dev->priv;
  179. return _ftmac100_halt(priv);
  180. }
  181. static int ftmac100_init(struct eth_device *dev, bd_t *bd)
  182. {
  183. struct ftmac100_data *priv = dev->priv;
  184. return _ftmac100_init(priv , dev->enetaddr);
  185. }
  186. static int _ftmac100_recv(struct ftmac100_data *priv)
  187. {
  188. struct ftmac100_rxdes *curr_des;
  189. unsigned short len;
  190. curr_des = &priv->rxdes[priv->rx_index];
  191. len = __ftmac100_recv(priv);
  192. if (len) {
  193. /* pass the packet up to the protocol layers. */
  194. net_process_received_packet((void *)curr_des->rxdes2, len);
  195. _ftmac100_free_pkt(priv);
  196. }
  197. return len ? 1 : 0;
  198. }
  199. /*
  200. * Get a data block via Ethernet
  201. */
  202. static int ftmac100_recv(struct eth_device *dev)
  203. {
  204. struct ftmac100_data *priv = dev->priv;
  205. return _ftmac100_recv(priv);
  206. }
  207. /*
  208. * Send a data block via Ethernet
  209. */
  210. static int ftmac100_send(struct eth_device *dev, void *packet, int length)
  211. {
  212. struct ftmac100_data *priv = dev->priv;
  213. return _ftmac100_send(priv , packet , length);
  214. }
  215. int ftmac100_initialize (bd_t *bd)
  216. {
  217. struct eth_device *dev;
  218. struct ftmac100_data *priv;
  219. dev = malloc (sizeof *dev);
  220. if (!dev) {
  221. printf ("%s(): failed to allocate dev\n", __func__);
  222. goto out;
  223. }
  224. /* Transmit and receive descriptors should align to 16 bytes */
  225. priv = memalign (16, sizeof (struct ftmac100_data));
  226. if (!priv) {
  227. printf ("%s(): failed to allocate priv\n", __func__);
  228. goto free_dev;
  229. }
  230. memset (dev, 0, sizeof (*dev));
  231. memset (priv, 0, sizeof (*priv));
  232. strcpy(dev->name, "FTMAC100");
  233. dev->iobase = CONFIG_FTMAC100_BASE;
  234. dev->init = ftmac100_init;
  235. dev->halt = ftmac100_halt;
  236. dev->send = ftmac100_send;
  237. dev->recv = ftmac100_recv;
  238. dev->priv = priv;
  239. priv->iobase = dev->iobase;
  240. eth_register (dev);
  241. return 1;
  242. free_dev:
  243. free (dev);
  244. out:
  245. return 0;
  246. }
  247. #endif
  248. #ifdef CONFIG_DM_ETH
  249. static int ftmac100_start(struct udevice *dev)
  250. {
  251. struct eth_pdata *plat = dev_get_platdata(dev);
  252. struct ftmac100_data *priv = dev_get_priv(dev);
  253. return _ftmac100_init(priv, plat->enetaddr);
  254. }
  255. static void ftmac100_stop(struct udevice *dev)
  256. {
  257. struct ftmac100_data *priv = dev_get_priv(dev);
  258. _ftmac100_halt(priv);
  259. }
  260. static int ftmac100_send(struct udevice *dev, void *packet, int length)
  261. {
  262. struct ftmac100_data *priv = dev_get_priv(dev);
  263. int ret;
  264. ret = _ftmac100_send(priv , packet , length);
  265. return ret ? 0 : -ETIMEDOUT;
  266. }
  267. static int ftmac100_recv(struct udevice *dev, int flags, uchar **packetp)
  268. {
  269. struct ftmac100_data *priv = dev_get_priv(dev);
  270. struct ftmac100_rxdes *curr_des;
  271. curr_des = &priv->rxdes[priv->rx_index];
  272. int len;
  273. len = __ftmac100_recv(priv);
  274. if (len)
  275. *packetp = (void *)curr_des->rxdes2;
  276. return len ? len : -EAGAIN;
  277. }
  278. static int ftmac100_free_pkt(struct udevice *dev, uchar *packet, int length)
  279. {
  280. struct ftmac100_data *priv = dev_get_priv(dev);
  281. _ftmac100_free_pkt(priv);
  282. return 0;
  283. }
  284. int ftmac100_read_rom_hwaddr(struct udevice *dev)
  285. {
  286. struct eth_pdata *pdata = dev_get_platdata(dev);
  287. eth_getenv_enetaddr("ethaddr", pdata->enetaddr);
  288. return 0;
  289. }
  290. static const char *dtbmacaddr(u32 ifno)
  291. {
  292. int node, len;
  293. char enet[16];
  294. const char *mac;
  295. const char *path;
  296. if (gd->fdt_blob == NULL) {
  297. printf("%s: don't have a valid gd->fdt_blob!\n", __func__);
  298. return NULL;
  299. }
  300. node = fdt_path_offset(gd->fdt_blob, "/aliases");
  301. if (node < 0)
  302. return NULL;
  303. sprintf(enet, "ethernet%d", ifno);
  304. path = fdt_getprop(gd->fdt_blob, node, enet, NULL);
  305. if (!path) {
  306. printf("no alias for %s\n", enet);
  307. return NULL;
  308. }
  309. node = fdt_path_offset(gd->fdt_blob, path);
  310. mac = fdt_getprop(gd->fdt_blob, node, "mac-address", &len);
  311. if (mac && is_valid_ethaddr((u8 *)mac))
  312. return mac;
  313. return NULL;
  314. }
  315. static int ftmac100_ofdata_to_platdata(struct udevice *dev)
  316. {
  317. struct ftmac100_data *priv = dev_get_priv(dev);
  318. struct eth_pdata *pdata = dev_get_platdata(dev);
  319. const char *mac;
  320. pdata->iobase = dev_get_addr(dev);
  321. priv->iobase = pdata->iobase;
  322. mac = dtbmacaddr(0);
  323. if (mac)
  324. memcpy(pdata->enetaddr , mac , 6);
  325. return 0;
  326. }
  327. static int ftmac100_probe(struct udevice *dev)
  328. {
  329. struct ftmac100_data *priv = dev_get_priv(dev);
  330. priv->name = dev->name;
  331. return 0;
  332. }
  333. static int ftmac100_bind(struct udevice *dev)
  334. {
  335. return device_set_name(dev, dev->name);
  336. }
  337. static const struct eth_ops ftmac100_ops = {
  338. .start = ftmac100_start,
  339. .send = ftmac100_send,
  340. .recv = ftmac100_recv,
  341. .stop = ftmac100_stop,
  342. .free_pkt = ftmac100_free_pkt,
  343. };
  344. static const struct udevice_id ftmac100_ids[] = {
  345. { .compatible = "andestech,atmac100" },
  346. { }
  347. };
  348. U_BOOT_DRIVER(ftmac100) = {
  349. .name = "nds32_mac",
  350. .id = UCLASS_ETH,
  351. .of_match = ftmac100_ids,
  352. .bind = ftmac100_bind,
  353. .ofdata_to_platdata = ftmac100_ofdata_to_platdata,
  354. .probe = ftmac100_probe,
  355. .ops = &ftmac100_ops,
  356. .priv_auto_alloc_size = sizeof(struct ftmac100_data),
  357. .platdata_auto_alloc_size = sizeof(struct eth_pdata),
  358. .flags = DM_FLAG_ALLOC_PRIV_DMA,
  359. };
  360. #endif