altera_uart.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  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 = map_physmem(dev_get_addr(dev),
  73. sizeof(struct altera_uart_regs),
  74. MAP_NOCACHE);
  75. plat->uartclk = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
  76. "clock-frequency", 0);
  77. return 0;
  78. }
  79. static const struct dm_serial_ops altera_uart_ops = {
  80. .putc = altera_uart_putc,
  81. .pending = altera_uart_pending,
  82. .getc = altera_uart_getc,
  83. .setbrg = altera_uart_setbrg,
  84. };
  85. static const struct udevice_id altera_uart_ids[] = {
  86. { .compatible = "altr,uart-1.0" },
  87. {}
  88. };
  89. U_BOOT_DRIVER(altera_uart) = {
  90. .name = "altera_uart",
  91. .id = UCLASS_SERIAL,
  92. .of_match = altera_uart_ids,
  93. .ofdata_to_platdata = altera_uart_ofdata_to_platdata,
  94. .platdata_auto_alloc_size = sizeof(struct altera_uart_platdata),
  95. .probe = altera_uart_probe,
  96. .ops = &altera_uart_ops,
  97. .flags = DM_FLAG_PRE_RELOC,
  98. };
  99. #ifdef CONFIG_DEBUG_UART_ALTERA_UART
  100. #include <debug_uart.h>
  101. static inline void _debug_uart_init(void)
  102. {
  103. struct altera_uart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
  104. u32 div;
  105. div = (CONFIG_DEBUG_UART_CLOCK / CONFIG_BAUDRATE) - 1;
  106. writel(div, &regs->divisor);
  107. }
  108. static inline void _debug_uart_putc(int ch)
  109. {
  110. struct altera_uart_regs *regs = (void *)CONFIG_DEBUG_UART_BASE;
  111. while (1) {
  112. u32 st = readl(&regs->status);
  113. if (st & ALTERA_UART_TRDY)
  114. break;
  115. }
  116. writel(ch, &regs->txdata);
  117. }
  118. DEBUG_UART_FUNCS
  119. #endif