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 <asm/io.h>
  11. #include <linux/compiler.h>
  12. #include <serial.h>
  13. struct altera_uart_regs {
  14. u32 rxdata; /* Rx data reg */
  15. u32 txdata; /* Tx data reg */
  16. u32 status; /* Status reg */
  17. u32 control; /* Control reg */
  18. u32 divisor; /* Baud rate divisor reg */
  19. u32 endofpacket; /* End-of-packet reg */
  20. };
  21. struct altera_uart_platdata {
  22. struct altera_uart_regs *regs;
  23. unsigned int uartclk;
  24. };
  25. /* status register */
  26. #define ALTERA_UART_TMT BIT(5) /* tx empty */
  27. #define ALTERA_UART_TRDY BIT(6) /* tx ready */
  28. #define ALTERA_UART_RRDY BIT(7) /* rx ready */
  29. DECLARE_GLOBAL_DATA_PTR;
  30. static int altera_uart_setbrg(struct udevice *dev, int baudrate)
  31. {
  32. struct altera_uart_platdata *plat = dev->platdata;
  33. struct altera_uart_regs *const regs = plat->regs;
  34. u32 div;
  35. div = (plat->uartclk / baudrate) - 1;
  36. writel(div, &regs->divisor);
  37. return 0;
  38. }
  39. static int altera_uart_putc(struct udevice *dev, const char ch)
  40. {
  41. struct altera_uart_platdata *plat = dev->platdata;
  42. struct altera_uart_regs *const regs = plat->regs;
  43. if (!(readl(&regs->status) & ALTERA_UART_TRDY))
  44. return -EAGAIN;
  45. writel(ch, &regs->txdata);
  46. return 0;
  47. }
  48. static int altera_uart_pending(struct udevice *dev, bool input)
  49. {
  50. struct altera_uart_platdata *plat = dev->platdata;
  51. struct altera_uart_regs *const regs = plat->regs;
  52. u32 st = readl(&regs->status);
  53. if (input)
  54. return st & ALTERA_UART_RRDY ? 1 : 0;
  55. else
  56. return !(st & ALTERA_UART_TMT);
  57. }
  58. static int altera_uart_getc(struct udevice *dev)
  59. {
  60. struct altera_uart_platdata *plat = dev->platdata;
  61. struct altera_uart_regs *const regs = plat->regs;
  62. if (!(readl(&regs->status) & ALTERA_UART_RRDY))
  63. return -EAGAIN;
  64. return readl(&regs->rxdata) & 0xff;
  65. }
  66. static int altera_uart_probe(struct udevice *dev)
  67. {
  68. return 0;
  69. }
  70. static int altera_uart_ofdata_to_platdata(struct udevice *dev)
  71. {
  72. struct altera_uart_platdata *plat = dev_get_platdata(dev);
  73. plat->regs = ioremap(dev_get_addr(dev),
  74. sizeof(struct altera_uart_regs));
  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. 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