jtag-console.c 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. /*
  2. * jtag-console.c - console driver over Blackfin JTAG
  3. *
  4. * Copyright (c) 2008 Analog Devices Inc.
  5. *
  6. * Licensed under the GPL-2 or later.
  7. */
  8. #include <common.h>
  9. #include <devices.h>
  10. #include <asm/blackfin.h>
  11. #ifndef CONFIG_JTAG_CONSOLE_TIMEOUT
  12. # define CONFIG_JTAG_CONSOLE_TIMEOUT 100
  13. #endif
  14. /* The Blackfin tends to be much much faster than the JTAG hardware. */
  15. static void jtag_write_emudat(uint32_t emudat)
  16. {
  17. static bool overflowed = false;
  18. ulong timeout = get_timer(0) + CONFIG_JTAG_CONSOLE_TIMEOUT;
  19. while (bfin_read_DBGSTAT() & 0x1) {
  20. if (overflowed)
  21. return;
  22. if (timeout < get_timer(0))
  23. overflowed = true;
  24. }
  25. overflowed = false;
  26. __asm__ __volatile__("emudat = %0;" : : "d"(emudat));
  27. }
  28. /* Transmit a buffer. The format is:
  29. * [32bit length][actual data]
  30. */
  31. static void jtag_send(const char *c, uint32_t len)
  32. {
  33. uint32_t i;
  34. if (len == 0)
  35. return;
  36. /* First send the length */
  37. jtag_write_emudat(len);
  38. /* Then send the data */
  39. for (i = 0; i < len; i += 4)
  40. jtag_write_emudat((c[i] << 0) | (c[i+1] << 8) | (c[i+2] << 16) | (c[i+3] << 24));
  41. }
  42. static void jtag_putc(const char c)
  43. {
  44. jtag_send(&c, 1);
  45. }
  46. static void jtag_puts(const char *s)
  47. {
  48. jtag_send(s, strlen(s));
  49. }
  50. static size_t inbound_len, leftovers_len;
  51. /* Lower layers want to know when jtag has data */
  52. static int jtag_tstc_dbg(void)
  53. {
  54. return (bfin_read_DBGSTAT() & 0x2);
  55. }
  56. /* Higher layers want to know when any data is available */
  57. static int jtag_tstc(void)
  58. {
  59. return jtag_tstc_dbg() || leftovers_len;
  60. }
  61. /* Receive a buffer. The format is:
  62. * [32bit length][actual data]
  63. */
  64. static uint32_t leftovers;
  65. static int jtag_getc(void)
  66. {
  67. int ret;
  68. uint32_t emudat;
  69. /* see if any data is left over */
  70. if (leftovers_len) {
  71. --leftovers_len;
  72. ret = leftovers & 0xff;
  73. leftovers >>= 8;
  74. return ret;
  75. }
  76. /* wait for new data ! */
  77. while (!jtag_tstc_dbg())
  78. continue;
  79. __asm__("%0 = emudat;" : "=d"(emudat));
  80. if (inbound_len == 0) {
  81. /* grab the length */
  82. inbound_len = emudat;
  83. } else {
  84. /* store the bytes */
  85. leftovers_len = min(4, inbound_len);
  86. inbound_len -= leftovers_len;
  87. leftovers = emudat;
  88. }
  89. return jtag_getc();
  90. }
  91. int drv_jtag_console_init(void)
  92. {
  93. device_t dev;
  94. int ret;
  95. memset(&dev, 0x00, sizeof(dev));
  96. strcpy(dev.name, "jtag");
  97. dev.flags = DEV_FLAGS_OUTPUT | DEV_FLAGS_INPUT | DEV_FLAGS_SYSTEM;
  98. dev.putc = jtag_putc;
  99. dev.puts = jtag_puts;
  100. dev.tstc = jtag_tstc;
  101. dev.getc = jtag_getc;
  102. ret = device_register(&dev);
  103. return (ret == 0 ? 1 : ret);
  104. }
  105. #ifdef CONFIG_UART_CONSOLE_IS_JTAG
  106. /* Since the JTAG is always available (at power on), allow it to fake a UART */
  107. void serial_set_baud(uint32_t baud) {}
  108. void serial_setbrg(void) {}
  109. int serial_init(void) { return 0; }
  110. void serial_putc(const char c) __attribute__((alias("jtag_putc")));
  111. void serial_puts(const char *s) __attribute__((alias("jtag_puts")));
  112. int serial_tstc(void) __attribute__((alias("jtag_tstc")));
  113. int serial_getc(void) __attribute__((alias("jtag_getc")));
  114. #endif