mcu_power_sleep.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. #include "mcu_power_sleep.h"
  2. #include "bsp/ml5238.h"
  3. #include "bsp/cs1180.h"
  4. #include "bsp/gpio.h"
  5. #include "bsp/spi.h"
  6. #include "bsp/uart.h"
  7. #include "bsp/i2c.h"
  8. #include "bsp/shark_rtc.h"
  9. #include "bsp/AT24CXX.h"
  10. #include "app/sox/iostate.h"
  11. #include "bsp/gd32_adc.h"
  12. extern void system_clock_24m_irc8m(void);
  13. extern void system_clock_config(void);
  14. extern void SystemCoreClockUpdate(void);
  15. extern void current_calibrate(void);
  16. extern void adc_init(void);
  17. extern void adc_deinit(void);
  18. static uint32_t _sleep_second_time = 0;
  19. static uint32_t _sleep_second_time_now = 0;
  20. static uint32_t _wakeup_source = 0;
  21. static void post_deepsleep(void);
  22. #define WDOG_TIME_FOR_SLEEP 15
  23. #define RTC_ALARM_FOR_SLEEP 10 //rtc alarm time MUST be small than wdog time!!!!
  24. uint32_t get_system_sleep_time(void){
  25. return _sleep_second_time;
  26. }
  27. static int enable_wakeup_irq(void){
  28. //hall_1_detect_irq_enable(1);
  29. //hall_2_detect_irq_enable(1);
  30. return shark_rtc_start_alarm(RTC_ALARM_FOR_SLEEP);
  31. }
  32. static void disable_wakeup_irq(void){
  33. shark_rtc_stop_alarm();
  34. //hall_1_detect_irq_enable(0);
  35. //hall_2_detect_irq_enable(0);
  36. }
  37. static int _is_wakeup_source(void){
  38. return (_wakeup_source & (~WAKEUP_SOURCE_RTC)) != 0;
  39. }
  40. void mcu_sleep_set_wakeup_source(uint32_t source){
  41. _wakeup_source |= source;
  42. }
  43. /* 在reload命令及deepsleep/standby模式命令中间插入(3个以上)IRC40K时钟间隔 */
  44. static void wait_for_enter_dsleep(void){
  45. volatile uint32_t wait = 10000 * 30;
  46. while(wait--> 0);
  47. }
  48. static void wait_dcdc_good(void) {
  49. volatile uint32_t wait_cycle = 2000;
  50. while(--wait_cycle > 0 && (!IS_DCDC_POWER_GOOD()));
  51. }
  52. static int pre_deepsleep(void){
  53. LED_ALL_ON(0);
  54. systick_close();
  55. shark_uart_deinit(SHARK_UART0);
  56. #if (CONFIG_BOARD_TYPE==SHARK_BOARD_SP700)
  57. shark_uart_deinit(SHARK_UART1);
  58. #endif
  59. wdog_reload();
  60. ml5238_power_save(1); //call, before spi0_deinit
  61. cs1180_adc_shutdown();
  62. AT24CXX_DeInit();
  63. gd32_adc_deinit();
  64. if (AUX_VOL_IS_OPEN()) {
  65. AUX_VOL_OPEN(0);//we should close small power, before dcdc close
  66. delay_us(1000);
  67. }
  68. wdog_reload();
  69. io_state()->aux_lock_detect = 0;
  70. DCDC_VOL_OPEN(0);
  71. delay_us(1000); // give 1s to wait small current short, when dcdc is closed
  72. AUX_VOL_OPEN(1);
  73. delay_us(5000); //give 5s to detect if the small current is short
  74. if (io_state()->aux_lock_detect){
  75. post_deepsleep();
  76. return -1;
  77. }
  78. wdog_set_timeout(WDOG_TIME_FOR_SLEEP);
  79. wait_for_enter_dsleep();
  80. if (_is_wakeup_source()) {
  81. _wakeup_source = 0;
  82. post_deepsleep();
  83. return -1;
  84. }
  85. return 0;
  86. }
  87. static void post_deepsleep(void){
  88. DCDC_VOL_OPEN(1);
  89. AUX_VOL_OPEN(0);//must close small power, cs1180 cali need small power close
  90. wait_dcdc_good();
  91. SystemInit();
  92. system_clock_config();
  93. SystemCoreClockUpdate();
  94. ml5238_power_save(0);
  95. gd32_adc_init();
  96. cs1180_adc_init();
  97. shark_uart_init(SHARK_UART0);
  98. #if (CONFIG_BOARD_TYPE==SHARK_BOARD_SP700)
  99. shark_uart_init(SHARK_UART1);
  100. #endif
  101. AT24CXX_Init();
  102. wdog_set_timeout(4);
  103. current_calibrate();
  104. wdog_reload();
  105. systick_open();
  106. AUX_VOL_OPEN(1);
  107. }
  108. void mcu_enter_deepsleep(void){
  109. _sleep_second_time_now = 0;
  110. if (pre_deepsleep()< 0){
  111. return;
  112. }
  113. _wakeup_source = 0;
  114. if (enable_wakeup_irq()< 0){
  115. disable_wakeup_irq();
  116. post_deepsleep();
  117. return;
  118. }
  119. do {
  120. u32 start_time = shark_rtc_get_second();
  121. pmu_to_deepsleepmode(PMU_LDO_LOWPOWER, WFI_CMD);
  122. u32 end_time = shark_rtc_get_second();
  123. if (end_time >= start_time) {
  124. _sleep_second_time_now += end_time - start_time;
  125. }else { //rtc second wrap
  126. _sleep_second_time_now += 60 - start_time + end_time;
  127. }
  128. if (_wakeup_source & WAKEUP_SOURCE_RTC) {
  129. wdog_reload();
  130. while (shark_rtc_start_alarm(RTC_ALARM_FOR_SLEEP) < 0){
  131. wdog_reload();
  132. if (_is_wakeup_source() || (_sleep_second_time_now >= (60))){
  133. break;
  134. }
  135. }
  136. wait_for_enter_dsleep();
  137. }
  138. }while(!_is_wakeup_source() && (_sleep_second_time_now < (60)));
  139. _sleep_second_time += _sleep_second_time_now;
  140. disable_wakeup_irq();
  141. post_deepsleep();
  142. }
  143. uint32_t mcu_get_sleeptime(void){
  144. return _sleep_second_time_now;
  145. }