AzulSCSI_log.cpp 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. #include "AzulSCSI_log.h"
  2. #include "AzulSCSI_config.h"
  3. #include "AzulSCSI_platform.h"
  4. const char *g_azlog_firmwareversion = "1.0.1" " " __DATE__ " " __TIME__;
  5. bool g_azlog_debug = true;
  6. // This memory buffer can be read by debugger and is also saved to azullog.txt
  7. #define LOGBUFMASK (LOGBUFSIZE - 1)
  8. // The log buffer is in special uninitialized RAM section so that it is not reset
  9. // when soft rebooting or jumping from bootloader.
  10. uint32_t g_log_magic;
  11. char g_logbuffer[LOGBUFSIZE + 1];
  12. uint32_t g_logpos;
  13. void azlog_raw(const char *str)
  14. {
  15. // Keep log from reboot / bootloader if magic matches expected value
  16. if (g_log_magic != 0xAA55AA55)
  17. {
  18. g_log_magic = 0xAA55AA55;
  19. g_logpos = 0;
  20. }
  21. const char *p = str;
  22. while (*p)
  23. {
  24. g_logbuffer[g_logpos & LOGBUFMASK] = *p++;
  25. g_logpos++;
  26. }
  27. // Keep buffer null-terminated
  28. g_logbuffer[g_logpos & LOGBUFMASK] = '\0';
  29. azplatform_log(str);
  30. }
  31. // Log byte as hex
  32. void azlog_raw(uint8_t value)
  33. {
  34. const char *nibble = "0123456789ABCDEF";
  35. char hexbuf[5] = {
  36. '0', 'x',
  37. nibble[(value >> 4) & 0xF], nibble[(value >> 0) & 0xF],
  38. 0
  39. };
  40. azlog_raw(hexbuf);
  41. }
  42. // Log integer as hex
  43. void azlog_raw(uint32_t value)
  44. {
  45. const char *nibble = "0123456789ABCDEF";
  46. char hexbuf[11] = {
  47. '0', 'x',
  48. nibble[(value >> 28) & 0xF], nibble[(value >> 24) & 0xF],
  49. nibble[(value >> 20) & 0xF], nibble[(value >> 16) & 0xF],
  50. nibble[(value >> 12) & 0xF], nibble[(value >> 8) & 0xF],
  51. nibble[(value >> 4) & 0xF], nibble[(value >> 0) & 0xF],
  52. 0
  53. };
  54. azlog_raw(hexbuf);
  55. }
  56. // Log integer as decimal
  57. void azlog_raw(int value)
  58. {
  59. char decbuf[16] = {0};
  60. char *p = &decbuf[14];
  61. int remainder = (value < 0) ? -value : value;
  62. do
  63. {
  64. *--p = '0' + (remainder % 10);
  65. remainder /= 10;
  66. } while (remainder > 0);
  67. if (value < 0)
  68. {
  69. *--p = '-';
  70. }
  71. azlog_raw(p);
  72. }
  73. void azlog_raw(bytearray array)
  74. {
  75. for (size_t i = 0; i < array.len; i++)
  76. {
  77. azlog_raw(array.data[i]);
  78. azlog_raw(" ");
  79. if (i > 32)
  80. {
  81. azlog_raw("... (total ", (int)array.len, ")");
  82. break;
  83. }
  84. }
  85. }
  86. uint32_t azlog_get_buffer_len()
  87. {
  88. return g_logpos;
  89. }
  90. const char *azlog_get_buffer(uint32_t *startpos)
  91. {
  92. uint32_t default_pos = 0;
  93. if (startpos == NULL)
  94. {
  95. startpos = &default_pos;
  96. }
  97. // Check oldest data available in buffer
  98. uint32_t margin = 16;
  99. if (g_logpos + margin > LOGBUFSIZE)
  100. {
  101. uint32_t oldest = g_logpos + margin - LOGBUFSIZE;
  102. if (*startpos < oldest)
  103. {
  104. *startpos = oldest;
  105. }
  106. }
  107. const char *result = &g_logbuffer[*startpos & LOGBUFMASK];
  108. if ((g_logpos & LOGBUFMASK) >= (*startpos & LOGBUFMASK))
  109. {
  110. // Ok, everything has been read now
  111. *startpos = g_logpos;
  112. }
  113. else
  114. {
  115. // Buffer wraps, read to end of buffer now and start from beginning on next call.
  116. *startpos = g_logpos & (~LOGBUFMASK);
  117. }
  118. return result;
  119. }