test_reset.ino 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. /*
  2. test_reset
  3. May 2015 by Bill Westfield (WestfW)
  4. Released to the public domain.
  5. This sketch demonstrates retrival of the Reset Cause register (MCUSR) of the AVR.
  6. Normally, MCUSR itself is destroyed by the use of a bootloader, but Optiboot v4.6
  7. and later save the contents in register r2, where it can be accessed by an
  8. application.
  9. */
  10. #include <avr/wdt.h>
  11. #if defined (__AVR_ATmega328P__) || defined (__AVR_ATmega168__)
  12. #include <LowPower.h>
  13. #endif
  14. /*
  15. First, we need a variable to hold the reset cause that can be written before
  16. early sketch initialization (that might change r2), and won't be reset by the
  17. various initialization code.
  18. avr-gcc provides for this via the ".noinit" section.
  19. */
  20. uint8_t resetFlag __attribute__ ((section(".noinit")));
  21. /*
  22. Next, we need to put some code to save reset cause from the bootload (in r2)
  23. to the variable. Again, avr-gcc provides special code sections for this.
  24. If compiled with link time optimization (-flto), as done by the Arduno
  25. IDE version 1.6 and higher, we need the "used" attribute to prevent this
  26. from being omitted.
  27. */
  28. void resetFlagsInit(void) __attribute__ ((naked))
  29. __attribute__ ((used))
  30. __attribute__ ((section (".init0")));
  31. void resetFlagsInit(void)
  32. {
  33. /*
  34. save the reset flags passed from the bootloader
  35. This is a "simple" matter of storing (STS) r2 in the special variable
  36. that we have created. We use assembler to access the right variable.
  37. */
  38. __asm__ __volatile__ ("sts %0, r2\n" : "=m" (resetFlag) :);
  39. }
  40. void printReset(const char *label, uint8_t resetFlags)
  41. {
  42. Serial.print(label);
  43. Serial.print(resetFlags, HEX);
  44. /*
  45. check for the usual bits. Note that the symnbols defined in wdt.h are
  46. bit numbers, so they have to be shifted before comparison.
  47. */
  48. if (resetFlags & (1 << WDRF))
  49. {
  50. Serial.print(F(" Watchdog"));
  51. resetFlags &= ~(1 << WDRF);
  52. }
  53. if (resetFlags & (1 << BORF))
  54. {
  55. Serial.print(F(" Brownout"));
  56. resetFlags &= ~(1 << BORF);
  57. }
  58. if (resetFlags & (1 << EXTRF))
  59. {
  60. Serial.print(F(" External"));
  61. resetFlags &= ~(1 << EXTRF);
  62. }
  63. if (resetFlags & (1 << PORF))
  64. {
  65. Serial.print(F(" PowerOn"));
  66. resetFlags &= ~(1 << PORF);
  67. }
  68. if (resetFlags != 0x00)
  69. {
  70. // It should never enter here
  71. Serial.print(" Unknown");
  72. }
  73. Serial.println("");
  74. }
  75. void setup() {
  76. #if 1
  77. /*
  78. * if we're going to allow the bootloader to be called as a service ('j" command),
  79. * then we need to disable the WDT when the sketch starts. Essentially, there is no
  80. * way to tell a watchdog that occurs during the sketch from the watchdog that will
  81. * re-start the sketch when the bootloader is called from the application.
  82. */
  83. MCUSR = ~(1 << WDRF); // allow us to disable WD
  84. wdt_disable();
  85. #endif
  86. Serial.begin(9600); // Initialize serial port
  87. Serial.println(F("Reset flag test\n"));
  88. printReset("Actual MCUSR content: 0x", MCUSR);
  89. printReset("Passed in R2: 0x", resetFlag);
  90. #ifdef GPIOR0
  91. printReset("Passed in GPIOR0: 0x", GPIOR0);
  92. #endif
  93. }
  94. void loop() {
  95. int ch;
  96. Serial.println(F("\nSend 0-9 to set MCUSR, (W)atchdog enable, (J)mp to bootload, (S)leep"));
  97. while ((ch = Serial.read()) < 0) ;
  98. if (ch >= '0' && ch <= '9') {
  99. MCUSR = ch & 0xF;
  100. printReset("\nNew MCUSR content: 0x", MCUSR);
  101. } else
  102. switch (ch & ~('a' - 'A')) {
  103. case 'W':
  104. wdt_enable(WDTO_15MS);
  105. while (1); // To prevent the loop to start again before WDT resets the board
  106. break;
  107. case 'S':
  108. #if defined (__AVR_ATmega328P__) || defined (__AVR_ATmega168__)
  109. // Enter power down state for 8 s with ADC and BOD module disabled
  110. Serial.println(F("Low Power Sleep Mode"));
  111. Serial.flush();
  112. LowPower.powerDown(SLEEP_8S, ADC_OFF, BOD_OFF);
  113. Serial.println(F("Sleep Wakeup"));
  114. Serial.flush();
  115. #else
  116. Serial.print(F("\nPower down not supported on this CPU\n"));
  117. #endif
  118. break;
  119. case 'J':
  120. /* Figure out where the bootloader starts. */
  121. #if FLASHEND > 140000
  122. Serial.println(F("Jump to bootloader not supported on chips with >128k memory"));
  123. #else
  124. typedef void (*do_reboot_t)(void);
  125. const do_reboot_t do_reboot = (do_reboot_t)((FLASHEND - 511) >> 1);
  126. Serial.print("bootstart = ");
  127. Serial.println((unsigned int)do_reboot, HEX);
  128. Serial.flush();
  129. cli(); TCCR0A = TCCR1A = TCCR2A = 0; // make sure interrupts are off and timers are reset.
  130. do_reboot();
  131. #endif
  132. break;
  133. }
  134. }