xilinx_emaclite.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632
  1. /*
  2. * (C) Copyright 2007-2009 Michal Simek
  3. * (C) Copyright 2003 Xilinx Inc.
  4. *
  5. * Michal SIMEK <monstr@monstr.eu>
  6. *
  7. * SPDX-License-Identifier: GPL-2.0+
  8. */
  9. #include <common.h>
  10. #include <net.h>
  11. #include <config.h>
  12. #include <dm.h>
  13. #include <console.h>
  14. #include <malloc.h>
  15. #include <asm/io.h>
  16. #include <phy.h>
  17. #include <miiphy.h>
  18. #include <fdtdec.h>
  19. #include <asm-generic/errno.h>
  20. #include <linux/kernel.h>
  21. DECLARE_GLOBAL_DATA_PTR;
  22. #define ENET_ADDR_LENGTH 6
  23. #define ETH_FCS_LEN 4 /* Octets in the FCS */
  24. /* Xmit complete */
  25. #define XEL_TSR_XMIT_BUSY_MASK 0x00000001UL
  26. /* Xmit interrupt enable bit */
  27. #define XEL_TSR_XMIT_IE_MASK 0x00000008UL
  28. /* Program the MAC address */
  29. #define XEL_TSR_PROGRAM_MASK 0x00000002UL
  30. /* define for programming the MAC address into the EMAC Lite */
  31. #define XEL_TSR_PROG_MAC_ADDR (XEL_TSR_XMIT_BUSY_MASK | XEL_TSR_PROGRAM_MASK)
  32. /* Transmit packet length upper byte */
  33. #define XEL_TPLR_LENGTH_MASK_HI 0x0000FF00UL
  34. /* Transmit packet length lower byte */
  35. #define XEL_TPLR_LENGTH_MASK_LO 0x000000FFUL
  36. /* Recv complete */
  37. #define XEL_RSR_RECV_DONE_MASK 0x00000001UL
  38. /* Recv interrupt enable bit */
  39. #define XEL_RSR_RECV_IE_MASK 0x00000008UL
  40. /* MDIO Address Register Bit Masks */
  41. #define XEL_MDIOADDR_REGADR_MASK 0x0000001F /* Register Address */
  42. #define XEL_MDIOADDR_PHYADR_MASK 0x000003E0 /* PHY Address */
  43. #define XEL_MDIOADDR_PHYADR_SHIFT 5
  44. #define XEL_MDIOADDR_OP_MASK 0x00000400 /* RD/WR Operation */
  45. /* MDIO Write Data Register Bit Masks */
  46. #define XEL_MDIOWR_WRDATA_MASK 0x0000FFFF /* Data to be Written */
  47. /* MDIO Read Data Register Bit Masks */
  48. #define XEL_MDIORD_RDDATA_MASK 0x0000FFFF /* Data to be Read */
  49. /* MDIO Control Register Bit Masks */
  50. #define XEL_MDIOCTRL_MDIOSTS_MASK 0x00000001 /* MDIO Status Mask */
  51. #define XEL_MDIOCTRL_MDIOEN_MASK 0x00000008 /* MDIO Enable */
  52. struct emaclite_regs {
  53. u32 tx_ping; /* 0x0 - TX Ping buffer */
  54. u32 reserved1[504];
  55. u32 mdioaddr; /* 0x7e4 - MDIO Address Register */
  56. u32 mdiowr; /* 0x7e8 - MDIO Write Data Register */
  57. u32 mdiord;/* 0x7ec - MDIO Read Data Register */
  58. u32 mdioctrl; /* 0x7f0 - MDIO Control Register */
  59. u32 tx_ping_tplr; /* 0x7f4 - Tx packet length */
  60. u32 global_interrupt; /* 0x7f8 - Global interrupt enable */
  61. u32 tx_ping_tsr; /* 0x7fc - Tx status */
  62. u32 tx_pong; /* 0x800 - TX Pong buffer */
  63. u32 reserved2[508];
  64. u32 tx_pong_tplr; /* 0xff4 - Tx packet length */
  65. u32 reserved3; /* 0xff8 */
  66. u32 tx_pong_tsr; /* 0xffc - Tx status */
  67. u32 rx_ping; /* 0x1000 - Receive Buffer */
  68. u32 reserved4[510];
  69. u32 rx_ping_rsr; /* 0x17fc - Rx status */
  70. u32 rx_pong; /* 0x1800 - Receive Buffer */
  71. u32 reserved5[510];
  72. u32 rx_pong_rsr; /* 0x1ffc - Rx status */
  73. };
  74. struct xemaclite {
  75. bool use_rx_pong_buffer_next; /* Next RX buffer to read from */
  76. u32 txpp; /* TX ping pong buffer */
  77. u32 rxpp; /* RX ping pong buffer */
  78. int phyaddr;
  79. struct emaclite_regs *regs;
  80. struct phy_device *phydev;
  81. struct mii_dev *bus;
  82. };
  83. static u32 etherrxbuff[PKTSIZE_ALIGN/4]; /* Receive buffer */
  84. static void xemaclite_alignedread(u32 *srcptr, void *destptr, u32 bytecount)
  85. {
  86. u32 i;
  87. u32 alignbuffer;
  88. u32 *to32ptr;
  89. u32 *from32ptr;
  90. u8 *to8ptr;
  91. u8 *from8ptr;
  92. from32ptr = (u32 *) srcptr;
  93. /* Word aligned buffer, no correction needed. */
  94. to32ptr = (u32 *) destptr;
  95. while (bytecount > 3) {
  96. *to32ptr++ = *from32ptr++;
  97. bytecount -= 4;
  98. }
  99. to8ptr = (u8 *) to32ptr;
  100. alignbuffer = *from32ptr++;
  101. from8ptr = (u8 *) &alignbuffer;
  102. for (i = 0; i < bytecount; i++)
  103. *to8ptr++ = *from8ptr++;
  104. }
  105. static void xemaclite_alignedwrite(void *srcptr, u32 *destptr, u32 bytecount)
  106. {
  107. u32 i;
  108. u32 alignbuffer;
  109. u32 *to32ptr = (u32 *) destptr;
  110. u32 *from32ptr;
  111. u8 *to8ptr;
  112. u8 *from8ptr;
  113. from32ptr = (u32 *) srcptr;
  114. while (bytecount > 3) {
  115. *to32ptr++ = *from32ptr++;
  116. bytecount -= 4;
  117. }
  118. alignbuffer = 0;
  119. to8ptr = (u8 *) &alignbuffer;
  120. from8ptr = (u8 *) from32ptr;
  121. for (i = 0; i < bytecount; i++)
  122. *to8ptr++ = *from8ptr++;
  123. *to32ptr++ = alignbuffer;
  124. }
  125. static int wait_for_bit(const char *func, u32 *reg, const u32 mask,
  126. bool set, unsigned int timeout)
  127. {
  128. u32 val;
  129. unsigned long start = get_timer(0);
  130. while (1) {
  131. val = readl(reg);
  132. if (!set)
  133. val = ~val;
  134. if ((val & mask) == mask)
  135. return 0;
  136. if (get_timer(start) > timeout)
  137. break;
  138. if (ctrlc()) {
  139. puts("Abort\n");
  140. return -EINTR;
  141. }
  142. udelay(1);
  143. }
  144. debug("%s: Timeout (reg=%p mask=%08x wait_set=%i)\n",
  145. func, reg, mask, set);
  146. return -ETIMEDOUT;
  147. }
  148. static int mdio_wait(struct emaclite_regs *regs)
  149. {
  150. return wait_for_bit(__func__, &regs->mdioctrl,
  151. XEL_MDIOCTRL_MDIOSTS_MASK, false, 2000);
  152. }
  153. static u32 phyread(struct xemaclite *emaclite, u32 phyaddress, u32 registernum,
  154. u16 *data)
  155. {
  156. struct emaclite_regs *regs = emaclite->regs;
  157. if (mdio_wait(regs))
  158. return 1;
  159. u32 ctrl_reg = in_be32(&regs->mdioctrl);
  160. out_be32(&regs->mdioaddr, XEL_MDIOADDR_OP_MASK |
  161. ((phyaddress << XEL_MDIOADDR_PHYADR_SHIFT) | registernum));
  162. out_be32(&regs->mdioctrl, ctrl_reg | XEL_MDIOCTRL_MDIOSTS_MASK);
  163. if (mdio_wait(regs))
  164. return 1;
  165. /* Read data */
  166. *data = in_be32(&regs->mdiord);
  167. return 0;
  168. }
  169. static u32 phywrite(struct xemaclite *emaclite, u32 phyaddress, u32 registernum,
  170. u16 data)
  171. {
  172. struct emaclite_regs *regs = emaclite->regs;
  173. if (mdio_wait(regs))
  174. return 1;
  175. /*
  176. * Write the PHY address, register number and clear the OP bit in the
  177. * MDIO Address register and then write the value into the MDIO Write
  178. * Data register. Finally, set the Status bit in the MDIO Control
  179. * register to start a MDIO write transaction.
  180. */
  181. u32 ctrl_reg = in_be32(&regs->mdioctrl);
  182. out_be32(&regs->mdioaddr, ~XEL_MDIOADDR_OP_MASK &
  183. ((phyaddress << XEL_MDIOADDR_PHYADR_SHIFT) | registernum));
  184. out_be32(&regs->mdiowr, data);
  185. out_be32(&regs->mdioctrl, ctrl_reg | XEL_MDIOCTRL_MDIOSTS_MASK);
  186. if (mdio_wait(regs))
  187. return 1;
  188. return 0;
  189. }
  190. static void emaclite_halt(struct udevice *dev)
  191. {
  192. debug("eth_halt\n");
  193. }
  194. /* Use MII register 1 (MII status register) to detect PHY */
  195. #define PHY_DETECT_REG 1
  196. /* Mask used to verify certain PHY features (or register contents)
  197. * in the register above:
  198. * 0x1000: 10Mbps full duplex support
  199. * 0x0800: 10Mbps half duplex support
  200. * 0x0008: Auto-negotiation support
  201. */
  202. #define PHY_DETECT_MASK 0x1808
  203. static int setup_phy(struct udevice *dev)
  204. {
  205. int i;
  206. u16 phyreg;
  207. struct xemaclite *emaclite = dev_get_priv(dev);
  208. struct phy_device *phydev;
  209. u32 supported = SUPPORTED_10baseT_Half |
  210. SUPPORTED_10baseT_Full |
  211. SUPPORTED_100baseT_Half |
  212. SUPPORTED_100baseT_Full;
  213. if (emaclite->phyaddr != -1) {
  214. phyread(emaclite, emaclite->phyaddr, PHY_DETECT_REG, &phyreg);
  215. if ((phyreg != 0xFFFF) &&
  216. ((phyreg & PHY_DETECT_MASK) == PHY_DETECT_MASK)) {
  217. /* Found a valid PHY address */
  218. debug("Default phy address %d is valid\n",
  219. emaclite->phyaddr);
  220. } else {
  221. debug("PHY address is not setup correctly %d\n",
  222. emaclite->phyaddr);
  223. emaclite->phyaddr = -1;
  224. }
  225. }
  226. if (emaclite->phyaddr == -1) {
  227. /* detect the PHY address */
  228. for (i = 31; i >= 0; i--) {
  229. phyread(emaclite, i, PHY_DETECT_REG, &phyreg);
  230. if ((phyreg != 0xFFFF) &&
  231. ((phyreg & PHY_DETECT_MASK) == PHY_DETECT_MASK)) {
  232. /* Found a valid PHY address */
  233. emaclite->phyaddr = i;
  234. debug("emaclite: Found valid phy address, %d\n",
  235. i);
  236. break;
  237. }
  238. }
  239. }
  240. /* interface - look at tsec */
  241. phydev = phy_connect(emaclite->bus, emaclite->phyaddr, dev,
  242. PHY_INTERFACE_MODE_MII);
  243. /*
  244. * Phy can support 1000baseT but device NOT that's why phydev->supported
  245. * must be setup for 1000baseT. phydev->advertising setups what speeds
  246. * will be used for autonegotiation where 1000baseT must be disabled.
  247. */
  248. phydev->supported = supported | SUPPORTED_1000baseT_Half |
  249. SUPPORTED_1000baseT_Full;
  250. phydev->advertising = supported;
  251. emaclite->phydev = phydev;
  252. phy_config(phydev);
  253. phy_startup(phydev);
  254. if (!phydev->link) {
  255. printf("%s: No link.\n", phydev->dev->name);
  256. return 0;
  257. }
  258. /* Do not setup anything */
  259. return 1;
  260. }
  261. static int emaclite_init(struct udevice *dev)
  262. {
  263. struct xemaclite *emaclite = dev_get_priv(dev);
  264. struct eth_pdata *pdata = dev_get_platdata(dev);
  265. struct emaclite_regs *regs = emaclite->regs;
  266. debug("EmacLite Initialization Started\n");
  267. /*
  268. * TX - TX_PING & TX_PONG initialization
  269. */
  270. /* Restart PING TX */
  271. out_be32(&regs->tx_ping_tsr, 0);
  272. /* Copy MAC address */
  273. xemaclite_alignedwrite(pdata->enetaddr, &regs->tx_ping,
  274. ENET_ADDR_LENGTH);
  275. /* Set the length */
  276. out_be32(&regs->tx_ping_tplr, ENET_ADDR_LENGTH);
  277. /* Update the MAC address in the EMAC Lite */
  278. out_be32(&regs->tx_ping_tsr, XEL_TSR_PROG_MAC_ADDR);
  279. /* Wait for EMAC Lite to finish with the MAC address update */
  280. while ((in_be32 (&regs->tx_ping_tsr) &
  281. XEL_TSR_PROG_MAC_ADDR) != 0)
  282. ;
  283. if (emaclite->txpp) {
  284. /* The same operation with PONG TX */
  285. out_be32(&regs->tx_pong_tsr, 0);
  286. xemaclite_alignedwrite(pdata->enetaddr, &regs->tx_pong,
  287. ENET_ADDR_LENGTH);
  288. out_be32(&regs->tx_pong_tplr, ENET_ADDR_LENGTH);
  289. out_be32(&regs->tx_pong_tsr, XEL_TSR_PROG_MAC_ADDR);
  290. while ((in_be32(&regs->tx_pong_tsr) &
  291. XEL_TSR_PROG_MAC_ADDR) != 0)
  292. ;
  293. }
  294. /*
  295. * RX - RX_PING & RX_PONG initialization
  296. */
  297. /* Write out the value to flush the RX buffer */
  298. out_be32(&regs->rx_ping_rsr, XEL_RSR_RECV_IE_MASK);
  299. if (emaclite->rxpp)
  300. out_be32(&regs->rx_pong_rsr, XEL_RSR_RECV_IE_MASK);
  301. out_be32(&regs->mdioctrl, XEL_MDIOCTRL_MDIOEN_MASK);
  302. if (in_be32(&regs->mdioctrl) & XEL_MDIOCTRL_MDIOEN_MASK)
  303. if (!setup_phy(dev))
  304. return -1;
  305. debug("EmacLite Initialization complete\n");
  306. return 0;
  307. }
  308. static int xemaclite_txbufferavailable(struct xemaclite *emaclite)
  309. {
  310. u32 tmp;
  311. struct emaclite_regs *regs = emaclite->regs;
  312. /*
  313. * Read the other buffer register
  314. * and determine if the other buffer is available
  315. */
  316. tmp = ~in_be32(&regs->tx_ping_tsr);
  317. if (emaclite->txpp)
  318. tmp |= ~in_be32(&regs->tx_pong_tsr);
  319. return !(tmp & XEL_TSR_XMIT_BUSY_MASK);
  320. }
  321. static int emaclite_send(struct udevice *dev, void *ptr, int len)
  322. {
  323. u32 reg;
  324. struct xemaclite *emaclite = dev_get_priv(dev);
  325. struct emaclite_regs *regs = emaclite->regs;
  326. u32 maxtry = 1000;
  327. if (len > PKTSIZE)
  328. len = PKTSIZE;
  329. while (xemaclite_txbufferavailable(emaclite) && maxtry) {
  330. udelay(10);
  331. maxtry--;
  332. }
  333. if (!maxtry) {
  334. printf("Error: Timeout waiting for ethernet TX buffer\n");
  335. /* Restart PING TX */
  336. out_be32(&regs->tx_ping_tsr, 0);
  337. if (emaclite->txpp) {
  338. out_be32(&regs->tx_pong_tsr, 0);
  339. }
  340. return -1;
  341. }
  342. /* Determine if the expected buffer address is empty */
  343. reg = in_be32(&regs->tx_ping_tsr);
  344. if ((reg & XEL_TSR_XMIT_BUSY_MASK) == 0) {
  345. debug("Send packet from tx_ping buffer\n");
  346. /* Write the frame to the buffer */
  347. xemaclite_alignedwrite(ptr, &regs->tx_ping, len);
  348. out_be32(&regs->tx_ping_tplr, len &
  349. (XEL_TPLR_LENGTH_MASK_HI | XEL_TPLR_LENGTH_MASK_LO));
  350. reg = in_be32(&regs->tx_ping_tsr);
  351. reg |= XEL_TSR_XMIT_BUSY_MASK;
  352. out_be32(&regs->tx_ping_tsr, reg);
  353. return 0;
  354. }
  355. if (emaclite->txpp) {
  356. /* Determine if the expected buffer address is empty */
  357. reg = in_be32(&regs->tx_pong_tsr);
  358. if ((reg & XEL_TSR_XMIT_BUSY_MASK) == 0) {
  359. debug("Send packet from tx_pong buffer\n");
  360. /* Write the frame to the buffer */
  361. xemaclite_alignedwrite(ptr, &regs->tx_pong, len);
  362. out_be32(&regs->tx_pong_tplr, len &
  363. (XEL_TPLR_LENGTH_MASK_HI |
  364. XEL_TPLR_LENGTH_MASK_LO));
  365. reg = in_be32(&regs->tx_pong_tsr);
  366. reg |= XEL_TSR_XMIT_BUSY_MASK;
  367. out_be32(&regs->tx_pong_tsr, reg);
  368. return 0;
  369. }
  370. }
  371. puts("Error while sending frame\n");
  372. return -1;
  373. }
  374. static int emaclite_recv(struct udevice *dev, int flags, uchar **packetp)
  375. {
  376. u32 length, first_read, reg, attempt = 0;
  377. void *addr, *ack;
  378. struct xemaclite *emaclite = dev->priv;
  379. struct emaclite_regs *regs = emaclite->regs;
  380. struct ethernet_hdr *eth;
  381. struct ip_udp_hdr *ip;
  382. try_again:
  383. if (!emaclite->use_rx_pong_buffer_next) {
  384. reg = in_be32(&regs->rx_ping_rsr);
  385. debug("Testing data at rx_ping\n");
  386. if ((reg & XEL_RSR_RECV_DONE_MASK) == XEL_RSR_RECV_DONE_MASK) {
  387. debug("Data found in rx_ping buffer\n");
  388. addr = &regs->rx_ping;
  389. ack = &regs->rx_ping_rsr;
  390. } else {
  391. debug("Data not found in rx_ping buffer\n");
  392. /* Pong buffer is not available - return immediately */
  393. if (!emaclite->rxpp)
  394. return -1;
  395. /* Try pong buffer if this is first attempt */
  396. if (attempt++)
  397. return -1;
  398. emaclite->use_rx_pong_buffer_next =
  399. !emaclite->use_rx_pong_buffer_next;
  400. goto try_again;
  401. }
  402. } else {
  403. reg = in_be32(&regs->rx_pong_rsr);
  404. debug("Testing data at rx_pong\n");
  405. if ((reg & XEL_RSR_RECV_DONE_MASK) == XEL_RSR_RECV_DONE_MASK) {
  406. debug("Data found in rx_pong buffer\n");
  407. addr = &regs->rx_pong;
  408. ack = &regs->rx_pong_rsr;
  409. } else {
  410. debug("Data not found in rx_pong buffer\n");
  411. /* Try ping buffer if this is first attempt */
  412. if (attempt++)
  413. return -1;
  414. emaclite->use_rx_pong_buffer_next =
  415. !emaclite->use_rx_pong_buffer_next;
  416. goto try_again;
  417. }
  418. }
  419. /* Read all bytes for ARP packet with 32bit alignment - 48bytes */
  420. first_read = ALIGN(ETHER_HDR_SIZE + ARP_HDR_SIZE + ETH_FCS_LEN, 4);
  421. xemaclite_alignedread(addr, etherrxbuff, first_read);
  422. /* Detect real packet size */
  423. eth = (struct ethernet_hdr *)etherrxbuff;
  424. switch (ntohs(eth->et_protlen)) {
  425. case PROT_ARP:
  426. length = first_read;
  427. debug("ARP Packet %x\n", length);
  428. break;
  429. case PROT_IP:
  430. ip = (struct ip_udp_hdr *)(etherrxbuff + ETHER_HDR_SIZE);
  431. length = ntohs(ip->ip_len);
  432. length += ETHER_HDR_SIZE + ETH_FCS_LEN;
  433. debug("IP Packet %x\n", length);
  434. break;
  435. default:
  436. debug("Other Packet\n");
  437. length = PKTSIZE;
  438. break;
  439. }
  440. /* Read the rest of the packet which is longer then first read */
  441. if (length != first_read)
  442. xemaclite_alignedread(addr + first_read,
  443. etherrxbuff + first_read,
  444. length - first_read);
  445. /* Acknowledge the frame */
  446. reg = in_be32(ack);
  447. reg &= ~XEL_RSR_RECV_DONE_MASK;
  448. out_be32(ack, reg);
  449. debug("Packet receive from 0x%p, length %dB\n", addr, length);
  450. net_process_received_packet((uchar *)etherrxbuff, length);
  451. return 0;
  452. }
  453. static int emaclite_miiphy_read(struct mii_dev *bus, int addr,
  454. int devad, int reg)
  455. {
  456. u32 ret;
  457. u16 val = 0;
  458. ret = phyread(bus->priv, addr, reg, &val);
  459. debug("emaclite: Read MII 0x%x, 0x%x, 0x%x, %d\n", addr, reg, val, ret);
  460. return val;
  461. }
  462. static int emaclite_miiphy_write(struct mii_dev *bus, int addr, int devad,
  463. int reg, u16 value)
  464. {
  465. debug("emaclite: Write MII 0x%x, 0x%x, 0x%x\n", addr, reg, value);
  466. return phywrite(bus->priv, addr, reg, value);
  467. }
  468. static int emaclite_probe(struct udevice *dev)
  469. {
  470. struct xemaclite *emaclite = dev_get_priv(dev);
  471. int ret;
  472. emaclite->bus = mdio_alloc();
  473. emaclite->bus->read = emaclite_miiphy_read;
  474. emaclite->bus->write = emaclite_miiphy_write;
  475. emaclite->bus->priv = emaclite;
  476. strcpy(emaclite->bus->name, "emaclite");
  477. ret = mdio_register(emaclite->bus);
  478. if (ret)
  479. return ret;
  480. return 0;
  481. }
  482. static int emaclite_remove(struct udevice *dev)
  483. {
  484. struct xemaclite *emaclite = dev_get_priv(dev);
  485. free(emaclite->phydev);
  486. mdio_unregister(emaclite->bus);
  487. mdio_free(emaclite->bus);
  488. return 0;
  489. }
  490. static const struct eth_ops emaclite_ops = {
  491. .start = emaclite_init,
  492. .send = emaclite_send,
  493. .recv = emaclite_recv,
  494. .stop = emaclite_halt,
  495. };
  496. static int emaclite_ofdata_to_platdata(struct udevice *dev)
  497. {
  498. struct eth_pdata *pdata = dev_get_platdata(dev);
  499. struct xemaclite *emaclite = dev_get_priv(dev);
  500. int offset = 0;
  501. pdata->iobase = (phys_addr_t)dev_get_addr(dev);
  502. emaclite->regs = (struct emaclite_regs *)pdata->iobase;
  503. emaclite->phyaddr = -1;
  504. offset = fdtdec_lookup_phandle(gd->fdt_blob, dev->of_offset,
  505. "phy-handle");
  506. if (offset > 0)
  507. emaclite->phyaddr = fdtdec_get_int(gd->fdt_blob, offset,
  508. "reg", -1);
  509. emaclite->txpp = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  510. "xlnx,tx-ping-pong", 0);
  511. emaclite->rxpp = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  512. "xlnx,rx-ping-pong", 0);
  513. printf("EMACLITE: %lx, phyaddr %d, %d/%d\n", (ulong)emaclite->regs,
  514. emaclite->phyaddr, emaclite->txpp, emaclite->rxpp);
  515. return 0;
  516. }
  517. static const struct udevice_id emaclite_ids[] = {
  518. { .compatible = "xlnx,xps-ethernetlite-1.00.a" },
  519. { }
  520. };
  521. U_BOOT_DRIVER(emaclite) = {
  522. .name = "emaclite",
  523. .id = UCLASS_ETH,
  524. .of_match = emaclite_ids,
  525. .ofdata_to_platdata = emaclite_ofdata_to_platdata,
  526. .probe = emaclite_probe,
  527. .remove = emaclite_remove,
  528. .ops = &emaclite_ops,
  529. .priv_auto_alloc_size = sizeof(struct xemaclite),
  530. .platdata_auto_alloc_size = sizeof(struct eth_pdata),
  531. };