rtt_console.c 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. /* rtt_console.c - Console messages to a RAM buffer that is then read by
  2. * the Segger J-Link debugger
  3. */
  4. /*
  5. * Copyright (c) 2016 Nordic Semiconductor ASA
  6. *
  7. * SPDX-License-Identifier: Apache-2.0
  8. */
  9. #include <kernel.h>
  10. #include <sys/printk.h>
  11. #include <device.h>
  12. #include <init.h>
  13. #include <SEGGER_RTT.h>
  14. extern void __printk_hook_install(int (*fn)(int));
  15. extern void __stdout_hook_install(int (*fn)(int));
  16. static bool host_present;
  17. /** @brief Wait for fixed period.
  18. *
  19. */
  20. static void wait(void)
  21. {
  22. if (!IS_ENABLED(CONFIG_MULTITHREADING) || k_is_in_isr()) {
  23. if (IS_ENABLED(CONFIG_RTT_TX_RETRY_IN_INTERRUPT)) {
  24. k_busy_wait(1000*CONFIG_RTT_TX_RETRY_DELAY_MS);
  25. }
  26. } else {
  27. k_msleep(CONFIG_RTT_TX_RETRY_DELAY_MS);
  28. }
  29. }
  30. static int rtt_console_out(int character)
  31. {
  32. char c = (char)character;
  33. unsigned int cnt;
  34. int max_cnt = CONFIG_RTT_TX_RETRY_CNT;
  35. uint32_t key;
  36. do {
  37. /* bug fix: can not using mutex in pre-kernel phase */
  38. key = irq_lock();
  39. cnt = SEGGER_RTT_WriteNoLock(0, &c, 1);
  40. irq_unlock(key);
  41. /* There are two possible reasons for not writing any data to
  42. * RTT:
  43. * - The host is not connected and not reading the data.
  44. * - The buffer got full and will be read by the host.
  45. * These two situations are distinguished using the following
  46. * algorithm:
  47. * At the beginning, the module assumes that the host is active,
  48. * so when no data is read, it busy waits and retries.
  49. * If, after retrying, the host reads the data, the module
  50. * assumes that the host is active. If it fails, the module
  51. * assumes that the host is inactive and stores that
  52. * information. On next call, only one attempt takes place.
  53. * The host is marked as active if the attempt is successful.
  54. */
  55. if (cnt) {
  56. /* byte processed - host is present. */
  57. host_present = true;
  58. } else if (host_present) {
  59. if (max_cnt) {
  60. /* bug fix: can not sleep in pre-kernel phase */
  61. //wait();
  62. max_cnt--;
  63. continue;
  64. } else {
  65. host_present = false;
  66. }
  67. }
  68. break;
  69. } while (1);
  70. return character;
  71. }
  72. static int rtt_console_init(const struct device *d)
  73. {
  74. ARG_UNUSED(d);
  75. __printk_hook_install(rtt_console_out);
  76. __stdout_hook_install(rtt_console_out);
  77. return 0;
  78. }
  79. SYS_INIT(rtt_console_init, PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);