altera_uart.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. /*
  2. * (C) Copyright 2004, Psyent Corporation <www.psyent.com>
  3. * Scott McNutt <smcnutt@psyent.com>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <errno.h>
  10. #include <serial.h>
  11. #include <asm/io.h>
  12. DECLARE_GLOBAL_DATA_PTR;
  13. /* status register */
  14. #define ALTERA_UART_TMT BIT(5) /* tx empty */
  15. #define ALTERA_UART_TRDY BIT(6) /* tx ready */
  16. #define ALTERA_UART_RRDY BIT(7) /* rx ready */
  17. struct altera_uart_regs {
  18. u32 rxdata; /* Rx data reg */
  19. u32 txdata; /* Tx data reg */
  20. u32 status; /* Status reg */
  21. u32 control; /* Control reg */
  22. u32 divisor; /* Baud rate divisor reg */
  23. u32 endofpacket; /* End-of-packet reg */
  24. };
  25. struct altera_uart_platdata {
  26. struct altera_uart_regs *regs;
  27. unsigned int uartclk;
  28. };
  29. static int altera_uart_setbrg(struct udevice *dev, int baudrate)
  30. {
  31. struct altera_uart_platdata *plat = dev->platdata;
  32. struct altera_uart_regs *const regs = plat->regs;
  33. u32 div;
  34. div = (plat->uartclk / baudrate) - 1;
  35. writel(div, &regs->divisor);
  36. return 0;
  37. }
  38. static int altera_uart_putc(struct udevice *dev, const char ch)
  39. {
  40. struct altera_uart_platdata *plat = dev->platdata;
  41. struct altera_uart_regs *const regs = plat->regs;
  42. if (!(readl(&regs->status) & ALTERA_UART_TRDY))
  43. return -EAGAIN;
  44. writel(ch, &regs->txdata);
  45. return 0;
  46. }
  47. static int altera_uart_pending(struct udevice *dev, bool input)
  48. {
  49. struct altera_uart_platdata *plat = dev->platdata;
  50. struct altera_uart_regs *const regs = plat->regs;
  51. u32 st = readl(&regs->status);
  52. if (input)
  53. return st & ALTERA_UART_RRDY ? 1 : 0;
  54. else
  55. return !(st & ALTERA_UART_TMT);
  56. }
  57. static int altera_uart_getc(struct udevice *dev)
  58. {
  59. struct altera_uart_platdata *plat = dev->platdata;
  60. struct altera_uart_regs *const regs = plat->regs;
  61. if (!(readl(&regs->status) & ALTERA_UART_RRDY))
  62. return -EAGAIN;
  63. return readl(&regs->rxdata) & 0xff;
  64. }
  65. static int altera_uart_probe(struct udevice *dev)
  66. {
  67. return 0;
  68. }
  69. static int altera_uart_ofdata_to_platdata(struct udevice *dev)
  70. {
  71. struct altera_uart_platdata *plat = dev_get_platdata(dev);
  72. plat->regs = ioremap(dev_get_addr(dev),
  73. sizeof(struct altera_uart_regs));
  74. plat->uartclk = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  75. "clock-frequency", 0);
  76. return 0;
  77. }
  78. static const struct dm_serial_ops altera_uart_ops = {
  79. .putc = altera_uart_putc,
  80. .pending = altera_uart_pending,
  81. .getc = altera_uart_getc,
  82. .setbrg = altera_uart_setbrg,
  83. };
  84. static const struct udevice_id altera_uart_ids[] = {
  85. { .compatible = "altr,uart-1.0" },
  86. {}
  87. };
  88. U_BOOT_DRIVER(altera_uart) = {
  89. .name = "altera_uart",
  90. .id = UCLASS_SERIAL,
  91. .of_match = altera_uart_ids,
  92. .ofdata_to_platdata = altera_uart_ofdata_to_platdata,
  93. .platdata_auto_alloc_size = sizeof(struct altera_uart_platdata),
  94. .probe = altera_uart_probe,
  95. .ops = &altera_uart_ops,
  96. .flags = DM_FLAG_PRE_RELOC,
  97. };
  98. #ifdef CONFIG_DEBUG_UART_ALTERA_UART
  99. #include <debug_uart.h>
  100. void debug_uart_init(void)
  101. {
  102. struct altera_uart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
  103. u32 div;
  104. div = (CONFIG_DEBUG_UART_CLOCK / CONFIG_BAUDRATE) - 1;
  105. writel(div, &regs->divisor);
  106. }
  107. static inline void _debug_uart_putc(int ch)
  108. {
  109. struct altera_uart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
  110. while (1) {
  111. u32 st = readl(&regs->status);
  112. if (st & ALTERA_UART_TRDY)
  113. break;
  114. }
  115. writel(ch, &regs->txdata);
  116. }
  117. DEBUG_UART_FUNCS
  118. #endif