test_reset.ino 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687
  1. /*
  2. * test_reset
  3. * May 2015 by Bill Westfield (WestfW)
  4. * Released to the public domain.
  5. *
  6. * This sketch demonstrates retrival of the Reset Cause register (MCUSR) of the AVR.
  7. * Normally, MCUSR itself is destroyed by the use of a bootloader, but Optiboot v4.6
  8. * and later save the contents in register r2, where it can be accessed by an
  9. * application.
  10. */
  11. #include <avr/wdt.h>
  12. /*
  13. * First, we need a variable to hold the reset cause that can be written before
  14. * early sketch initialization (that might change r2), and won't be reset by the
  15. * various initialization code.
  16. * avr-gcc provides for this via the ".noinit" section.
  17. */
  18. uint8_t resetFlags __attribute__ ((section(".noinit")));
  19. /*
  20. * Next, we need to put some code to save reset cause from the bootload (in r2)
  21. * to the variable. Again, avr-gcc provides special code sections for this.
  22. * If compiled with link time optimization (-flto), as done by the Arduno
  23. * IDE version 1.6 and higher, we need the "used" attribute to prevent this
  24. * from being omitted.
  25. */
  26. void resetFlagsInit(void) __attribute__ ((naked))
  27. __attribute__ ((used))
  28. __attribute__ ((section (".init0")));
  29. void resetFlagsInit(void)
  30. {
  31. /*
  32. * save the reset flags passed from the bootloader
  33. * This is a "simple" matter of storing (STS) r2 in the special variable
  34. * that we have created. We use assembler to access the right variable.
  35. */
  36. __asm__ __volatile__ ("sts %0, r2\n" : "=m" (resetFlags) :);
  37. }
  38. void setup() {
  39. Serial.begin(9600); // Initialize serial port
  40. Serial.println("Reset flag test");
  41. Serial.print("Have reset flag value 0x");
  42. Serial.print(resetFlags, HEX);
  43. /*
  44. * check for the usual bits. Note that the symnbols defined in wdt.h are
  45. * bit numbers, so they have to be shifted before comparison.
  46. */
  47. if (resetFlags & (1<<WDRF))
  48. {
  49. Serial.print(" Watchdog");
  50. resetFlags &= ~(1<<WDRF);
  51. }
  52. if (resetFlags & (1<<BORF))
  53. {
  54. Serial.print(" Brownout");
  55. resetFlags &= ~(1<<BORF);
  56. }
  57. if (resetFlags & (1<<EXTRF))
  58. {
  59. Serial.print(" External");
  60. resetFlags &= ~(1<<EXTRF);
  61. }
  62. if (resetFlags & (1<<PORF))
  63. {
  64. Serial.print(" PowerOn");
  65. resetFlags &= ~(1<<PORF);
  66. }
  67. if (resetFlags != 0x00)
  68. {
  69. // It should never enter here
  70. Serial.print(" Unknown");
  71. }
  72. Serial.println("");
  73. }
  74. void loop() {
  75. Serial.println("Send something to reset through watchdog peripheral");
  76. while (Serial.read() < 0) ;
  77. wdt_enable(WDTO_15MS);
  78. while (1); // To prevent the loop to start again before WDT resets the board
  79. }