mcu_power_sleep.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  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. _wakeup_source = 0;
  55. systick_close();
  56. shark_rtc_set_backup(0xF0);
  57. shark_uart_deinit(SHARK_UART0);
  58. #if (CONFIG_BOARD_TYPE==SHARK_BOARD_SP700)
  59. shark_uart_deinit(SHARK_UART1);
  60. #endif
  61. wdog_reload();
  62. shark_rtc_set_backup(0xF1);
  63. ml5238_power_save(1); //call, before spi0_deinit
  64. cs1180_adc_shutdown();
  65. AT24CXX_DeInit();
  66. gd32_adc_deinit();
  67. if (AUX_VOL_IS_OPEN()) {
  68. AUX_VOL_OPEN(0);//we should close small power, before dcdc close
  69. delay_us(1000);
  70. }
  71. shark_rtc_set_backup(0xF2);
  72. wdog_reload();
  73. io_state()->aux_lock_detect = 0;
  74. DCDC_VOL_OPEN(0);
  75. delay_us(1000); // give 1s to wait small current short, when dcdc is closed
  76. AUX_VOL_OPEN(1);
  77. delay_us(5000); //give 5s to detect if the small current is short
  78. if (io_state()->aux_lock_detect){
  79. shark_rtc_set_backup(0xF3);
  80. post_deepsleep();
  81. return -1;
  82. }
  83. shark_rtc_set_backup(0xF4);
  84. if (wdog_set_timeout(WDOG_TIME_FOR_SLEEP) < 0){
  85. post_deepsleep();
  86. return -1;
  87. }
  88. shark_rtc_set_backup(0xF5);
  89. wait_for_enter_dsleep();
  90. shark_rtc_set_backup(0xF6);
  91. if (_is_wakeup_source()) {
  92. _wakeup_source = 0;
  93. post_deepsleep();
  94. return -1;
  95. }
  96. shark_rtc_set_backup(0xF7);
  97. return 0;
  98. }
  99. static void post_deepsleep(void){
  100. shark_rtc_set_backup(0xF8);
  101. DCDC_VOL_OPEN(1);
  102. AUX_VOL_OPEN(0);//must close small power, cs1180 cali need small power close
  103. wait_dcdc_good();
  104. shark_rtc_set_backup(0xF9);
  105. SystemInit();
  106. system_clock_config();
  107. SystemCoreClockUpdate();
  108. ml5238_power_save(0);
  109. gd32_adc_init();
  110. cs1180_adc_init();
  111. shark_rtc_set_backup(0xFA);
  112. shark_uart_init(SHARK_UART0);
  113. #if (CONFIG_BOARD_TYPE==SHARK_BOARD_SP700)
  114. shark_uart_init(SHARK_UART1);
  115. #endif
  116. AT24CXX_Init();
  117. shark_rtc_set_backup(0xFB);
  118. wdog_set_timeout(4);
  119. shark_rtc_set_backup(0xFC);
  120. current_calibrate();
  121. wdog_reload();
  122. systick_open();
  123. AUX_VOL_OPEN(1);
  124. shark_rtc_set_backup(0xFD);
  125. }
  126. void mcu_enter_deepsleep(void){
  127. _sleep_second_time_now = 0;
  128. if (pre_deepsleep()< 0){
  129. return;
  130. }
  131. _wakeup_source = 0;
  132. if (enable_wakeup_irq()< 0){
  133. disable_wakeup_irq();
  134. post_deepsleep();
  135. return;
  136. }
  137. shark_rtc_set_backup(0xFF0);
  138. do {
  139. u32 start_time = shark_rtc_get_second();
  140. pmu_to_deepsleepmode(PMU_LDO_LOWPOWER, WFI_CMD);
  141. shark_rtc_set_backup(0xFF1);
  142. u32 end_time = shark_rtc_get_second();
  143. if (end_time >= start_time) {
  144. _sleep_second_time_now += end_time - start_time;
  145. }else { //rtc second wrap
  146. _sleep_second_time_now += 60 - start_time + end_time;
  147. }
  148. if (_wakeup_source & WAKEUP_SOURCE_RTC) {
  149. wdog_reload();
  150. shark_rtc_set_backup(0xFF2);
  151. while (shark_rtc_start_alarm(RTC_ALARM_FOR_SLEEP) < 0){
  152. wdog_reload();
  153. shark_rtc_set_backup(0xFF3);
  154. if (_is_wakeup_source() || (_sleep_second_time_now >= (60))){
  155. break;
  156. }
  157. }
  158. shark_rtc_set_backup(0xFF4);
  159. wait_for_enter_dsleep();
  160. shark_rtc_set_backup(0xFF5);
  161. }
  162. }while(!_is_wakeup_source() && (_sleep_second_time_now < (60)));
  163. _sleep_second_time += _sleep_second_time_now;
  164. disable_wakeup_irq();
  165. shark_rtc_set_backup(0xFF6);
  166. post_deepsleep();
  167. }
  168. uint32_t mcu_get_sleeptime(void){
  169. return _sleep_second_time_now;
  170. }