mcu_power_sleep.c 4.6 KB

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