services.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  1. /*
  2. This example code is in the Public Domain (or CC0 licensed, at your option.)
  3. Unless required by applicable law or agreed to in writing, this
  4. software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
  5. CONDITIONS OF ANY KIND, either express or implied.
  6. */
  7. #include <stdio.h>
  8. #include "freertos/FreeRTOS.h"
  9. #include "freertos/timers.h"
  10. #include "esp_log.h"
  11. #include "esp_sleep.h"
  12. #include "driver/rtc_io.h"
  13. #include "driver/ledc.h"
  14. #include "driver/i2c.h"
  15. #include "driver/rmt.h"
  16. #include "Configurator.h"
  17. #include "gpio_exp.h"
  18. #include "battery.h"
  19. #include "led.h"
  20. #include "monitor.h"
  21. #include "globdefs.h"
  22. #include "accessors.h"
  23. #include "messaging.h"
  24. #include "buttons.h"
  25. #include "services.h"
  26. extern void battery_svc_init(void);
  27. extern void monitor_svc_init(void);
  28. extern void led_svc_init(void);
  29. int i2c_system_port = I2C_SYSTEM_PORT;
  30. int i2c_system_speed = 400000;
  31. int spi_system_host = SPI_SYSTEM_HOST;
  32. int spi_system_dc_gpio = -1;
  33. int rmt_system_base_tx_channel = RMT_CHANNEL_0;
  34. int rmt_system_base_rx_channel = RMT_CHANNEL_MAX-1;
  35. pwm_system_t pwm_system = {
  36. .timer = LEDC_TIMER_0,
  37. .base_channel = LEDC_CHANNEL_0,
  38. .max = (1 << LEDC_TIMER_13_BIT),
  39. };
  40. static sys_SleepService * sleep_config;
  41. static EXT_RAM_ATTR struct {
  42. uint64_t wake_gpio, wake_level;
  43. uint64_t rtc_gpio, rtc_level;
  44. uint32_t delay, spurious;
  45. float battery_level;
  46. int battery_count;
  47. void (*idle_chain)(uint32_t now);
  48. void (*battery_chain)(float level, int cells);
  49. void (*suspend[10])(void);
  50. uint32_t (*sleeper[10])(void);
  51. } sleep_context;
  52. static const char *TAG = "services";
  53. void set_gpio_level(sys_GPIO*gpio,const char * name, gpio_mode_t mode){
  54. gpio_pad_select_gpio(gpio->pin);
  55. gpio_set_direction(gpio->pin, mode);
  56. gpio_set_level(gpio->pin, gpio->level);
  57. ESP_LOGI(TAG, "set GPIO %u to %s, level %d", gpio->pin,name, gpio->level);
  58. }
  59. void set_chip_power_gpio(sys_Gpios*gpios) {
  60. if(gpios->has_power){
  61. gpios->power.level = 1;
  62. set_gpio_level(&gpios->power,"vcc", GPIO_MODE_OUTPUT);
  63. }
  64. if(gpios->has_GND){
  65. gpios->GND.level = 0;
  66. set_gpio_level(&gpios->GND,"gnd", GPIO_MODE_OUTPUT);
  67. }
  68. }
  69. /****************************************************************************************
  70. *
  71. */
  72. static void sleep_gpio_handler(void *id, button_event_e event, button_press_e mode, bool long_press) {
  73. if (event == BUTTON_PRESSED) services_sleep_activate(SLEEP_ONGPIO);
  74. }
  75. /****************************************************************************************
  76. *
  77. */
  78. static void sleep_timer(uint32_t now) {
  79. static EXT_RAM_ATTR uint32_t last, first;
  80. // first chain the calls to pseudo_idle function
  81. if (sleep_context.idle_chain) sleep_context.idle_chain(now);
  82. // we need boot time for spurious timeout calculation
  83. if (!first) first = now;
  84. // only query callbacks every 30s if we have at least one sleeper
  85. if (!*sleep_context.sleeper || now < last + 30*1000) return;
  86. last = now;
  87. // time to evaluate if we had spurious wake-up
  88. if (sleep_context.spurious && now > sleep_context.spurious + first) {
  89. bool spurious = true;
  90. // see if at least one sleeper has been awake since we started
  91. for (uint32_t (**sleeper)(void) = sleep_context.sleeper; *sleeper && spurious; sleeper++) {
  92. spurious &= (*sleeper)() >= now - first;
  93. }
  94. // no activity since we woke-up, this was a spurious one
  95. if (spurious) {
  96. ESP_LOGI(TAG, "spurious wake of %d sec, going back to sleep", (now - first) / 1000);
  97. services_sleep_activate(SLEEP_ONTIMER);
  98. }
  99. // resume normal work but we might have no "regular" inactivity delay
  100. sleep_context.spurious = 0;
  101. if (!sleep_context.delay) *sleep_context.sleeper = NULL;
  102. ESP_LOGI(TAG, "wake-up was not spurious after %d sec", (now - first) / 1000);
  103. }
  104. // we might be here because we are waiting for spurious
  105. if (sleep_context.delay) {
  106. // call all sleepers to know how long for how long they have been inactive
  107. for (uint32_t (**sleeper)(void) = sleep_context.sleeper; sleep_context.delay && *sleeper; sleeper++) {
  108. if ((*sleeper)() < sleep_context.delay) return;
  109. }
  110. // if we are here, we are ready to sleep;
  111. services_sleep_activate(SLEEP_ONTIMER);
  112. }
  113. }
  114. /****************************************************************************************
  115. *
  116. */
  117. static void sleep_battery(float level, int cells) {
  118. // chain if any
  119. if (sleep_context.battery_chain) sleep_context.battery_chain(level, cells);
  120. // then assess if we have to stop because of low batt
  121. if (level < sleep_context.battery_level) {
  122. if (sleep_context.battery_count++ == 2) services_sleep_activate(SLEEP_ONBATTERY);
  123. } else {
  124. sleep_context.battery_count = 0;
  125. }
  126. }
  127. /****************************************************************************************
  128. *
  129. */
  130. void services_sleep_init(void) {
  131. ESP_LOGD(TAG,"Initializing sleep services");
  132. if(!SYS_SERVICES_SLEEP(sleep_config)){
  133. ESP_LOGD(TAG,"No sleep service configured") ;
  134. }
  135. // get the wake criteria
  136. for(int i=0;i<sleep_config->wake_count;i++){
  137. if (!rtc_gpio_is_valid_gpio(sleep_config->wake[i].pin)) {
  138. ESP_LOGE(TAG, "invalid wake GPIO %d (not in RTC domain)", sleep_config->wake[i].pin);
  139. } else {
  140. sleep_context.wake_gpio |= 1LL << sleep_config->wake[i].pin;
  141. sleep_context.wake_gpio |= 1LL << sleep_config->wake[i].pin;
  142. sleep_context.wake_level |= sleep_config->wake[i].level << sleep_config->wake[i].pin;
  143. }
  144. }
  145. // when moving to esp-idf more recent than 4.4.x, multiple gpio wake-up with level specific can be done
  146. if (sleep_context.wake_gpio) {
  147. ESP_LOGI(TAG, "Sleep wake-up gpio bitmap 0x%llx (active 0x%llx)", sleep_context.wake_gpio, sleep_context.wake_level);
  148. }
  149. // do we want battery safety
  150. sleep_context.battery_level = sleep_config->batt;
  151. if (sleep_context.battery_level != 0.0) {
  152. sleep_context.battery_chain = battery_handler_svc;
  153. battery_handler_svc = sleep_battery;
  154. ESP_LOGI(TAG, "Sleep on battery level of %.2f", sleep_context.battery_level);
  155. }
  156. for(int i = 0;i<sleep_config->rtc_count;i++){
  157. if (!rtc_gpio_is_valid_gpio(sleep_config->rtc[i].pin)) {
  158. ESP_LOGE(TAG, "invalid rtc GPIO %d", sleep_config->rtc[i].pin);
  159. } else {
  160. sleep_context.rtc_gpio |= 1LL << sleep_config->rtc[i].pin;
  161. sleep_context.rtc_level |= sleep_config->rtc[i].level << sleep_config->rtc[i].pin;
  162. }
  163. }
  164. // when moving to esp-idf more recent than 4.4.x, multiple gpio wake-up with level specific can be done
  165. if (sleep_context.rtc_gpio) {
  166. ESP_LOGI(TAG, "RTC forced gpio bitmap 0x%llx (active 0x%llx)", sleep_context.rtc_gpio, sleep_context.rtc_level);
  167. }
  168. // get the GPIOs that activate sleep (we could check that we have a valid wake)
  169. if(sleep_config->has_sleep && sleep_config->sleep.pin >=0 ){
  170. ESP_LOGI(TAG, "Sleep activation gpio %d (active %d)", sleep_config->sleep.pin, sleep_config->sleep.level);
  171. button_create(NULL, sleep_config->sleep.pin, sleep_config->sleep.level ? BUTTON_HIGH : BUTTON_LOW, true, 0, sleep_gpio_handler, 0, -1);
  172. }
  173. // do we want delay sleep
  174. sleep_context.delay = sleep_config->delay*60*1000;
  175. // now check why we woke-up
  176. esp_sleep_wakeup_cause_t cause = esp_sleep_get_wakeup_cause();
  177. if (cause == ESP_SLEEP_WAKEUP_EXT0 || cause == ESP_SLEEP_WAKEUP_EXT1) {
  178. ESP_LOGI(TAG, "waking-up from deep sleep with cause %d", cause);
  179. // find the type of wake-up
  180. uint64_t wake_gpio;
  181. if (cause == ESP_SLEEP_WAKEUP_EXT0) wake_gpio = sleep_context.wake_gpio;
  182. else wake_gpio = esp_sleep_get_ext1_wakeup_status();
  183. // we might be woken up by infrared in which case we want a short sleep
  184. if (infrared_gpio() >= 0 && ((1LL << infrared_gpio()) & wake_gpio)) {
  185. sleep_context.spurious = 1;
  186. if(sleep_config->spurious>0){
  187. sleep_context.spurious = sleep_config->spurious;
  188. }
  189. sleep_context.spurious *= 60*1000;
  190. ESP_LOGI(TAG, "spurious wake-up detection during %d sec", sleep_context.spurious / 1000);
  191. }
  192. }
  193. // if we have inactivity timer (user-set or because of IR wake) then active counters
  194. if (sleep_context.delay || sleep_context.spurious) {
  195. sleep_context.idle_chain = pseudo_idle_svc;
  196. pseudo_idle_svc = sleep_timer;
  197. if (sleep_context.delay) ESP_LOGI(TAG, "inactivity timer of %d minute(s)", sleep_context.delay / (60*1000));
  198. }
  199. }
  200. /****************************************************************************************
  201. *
  202. */
  203. void services_sleep_activate(sleep_cause_e cause) {
  204. // call all sleep hooks that might want to do something
  205. for (void (**suspend)(void) = sleep_context.suspend; *suspend; suspend++) (*suspend)();
  206. // isolate all possible GPIOs, except the wake-up and RTC-maintaines ones
  207. esp_sleep_config_gpio_isolate();
  208. // keep RTC domain up if we need to maintain pull-up/down of some GPIO from RTC
  209. if (sleep_context.rtc_gpio) esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
  210. for (int i = 0; i < GPIO_NUM_MAX; i++) {
  211. // must be a RTC GPIO
  212. if (!rtc_gpio_is_valid_gpio(i)) continue;
  213. // do we need to maintain a pull-up or down of that GPIO
  214. if ((1LL << i) & sleep_context.rtc_gpio) {
  215. if ((sleep_context.rtc_level >> i) & 0x01) rtc_gpio_pullup_en(i);
  216. else rtc_gpio_pulldown_en(i);
  217. // or is this not wake-up GPIO, just isolate it
  218. } else if (!((1LL << i) & sleep_context.wake_gpio)) {
  219. rtc_gpio_isolate(i);
  220. }
  221. }
  222. // is there just one GPIO
  223. if (sleep_context.wake_gpio & (sleep_context.wake_gpio - 1)) {
  224. ESP_LOGI(TAG, "going to sleep cause %d, wake-up on multiple GPIO, any '1' wakes up 0x%llx", cause, sleep_context.wake_gpio);
  225. #if defined(CONFIG_IDF_TARGET_ESP32S3) && ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 2, 0)
  226. if (!sleep_context.wake_level) esp_sleep_enable_ext1_wakeup(sleep_context.wake_gpio, ESP_EXT1_WAKEUP_ANY_LOW);
  227. else
  228. #endif
  229. esp_sleep_enable_ext1_wakeup(sleep_context.wake_gpio, ESP_EXT1_WAKEUP_ANY_HIGH);
  230. } else if (sleep_context.wake_gpio) {
  231. int gpio = __builtin_ctzll(sleep_context.wake_gpio);
  232. int level = (sleep_context.wake_level >> gpio) & 0x01;
  233. ESP_LOGI(TAG, "going to sleep cause %d, wake-up on GPIO %d level %d", cause, gpio, level);
  234. esp_sleep_enable_ext0_wakeup(gpio, level);
  235. } else {
  236. ESP_LOGW(TAG, "going to sleep cause %d, no wake-up option", cause);
  237. }
  238. // we need to use a timer in case the same button is used for sleep and wake-up and it's "pressed" vs "released" selected
  239. if (cause == SLEEP_ONKEY) xTimerStart(xTimerCreate("sleepTimer", pdMS_TO_TICKS(1000), pdFALSE, NULL, (void (*)(void*)) esp_deep_sleep_start), 0);
  240. else esp_deep_sleep_start();
  241. }
  242. /****************************************************************************************
  243. *
  244. */
  245. static void register_method(void **store, size_t size, void *method) {
  246. for (int i = 0; i < size; i++, *store++) if (!*store) {
  247. *store = method;
  248. return;
  249. }
  250. }
  251. /****************************************************************************************
  252. *
  253. */
  254. void services_sleep_setsuspend(void (*hook)(void)) {
  255. register_method((void**) sleep_context.suspend, sizeof(sleep_context.suspend)/sizeof(*sleep_context.suspend), (void*) hook);
  256. }
  257. /****************************************************************************************
  258. *
  259. */
  260. void services_sleep_setsleeper(uint32_t (*sleeper)(void)) {
  261. register_method((void**) sleep_context.sleeper, sizeof(sleep_context.sleeper)/sizeof(*sleep_context.sleeper), (void*) sleeper);
  262. }
  263. /****************************************************************************************
  264. *
  265. */
  266. void services_init(void) {
  267. messaging_service_init();
  268. gpio_install_isr_service(0);
  269. // todo: untangle i2c stuff
  270. #ifdef CONFIG_I2C_LOCKED
  271. if (i2c_system_port == 0) {
  272. i2c_system_port = 1;
  273. ESP_LOGE(TAG, "Port 0 is reserved for internal DAC use");
  274. }
  275. #endif
  276. // set potential power GPIO on chip first in case expanders are power using these
  277. set_chip_power_gpio(&platform->gpios);
  278. // shared I2C bus
  279. const i2c_config_t * i2c_config = config_i2c_get(&i2c_system_port);
  280. ESP_LOGI(TAG,"Configuring I2C sda:%d scl:%d port:%u speed:%u", i2c_config->sda_io_num, i2c_config->scl_io_num, i2c_system_port, i2c_config->master.clk_speed);
  281. if (i2c_config->sda_io_num != -1 && i2c_config->scl_io_num != -1) {
  282. i2c_param_config(i2c_system_port, i2c_config);
  283. i2c_driver_install(i2c_system_port, i2c_config->mode, 0, 0, 0 );
  284. } else {
  285. i2c_system_port = -1;
  286. ESP_LOGW(TAG, "no I2C configured");
  287. }
  288. const spi_bus_config_t * spi_config = config_spi_get((spi_host_device_t*) &spi_system_host);
  289. ESP_LOGI(TAG,"Configuring SPI mosi:%d miso:%d clk:%d host:%u dc:%d", spi_config->mosi_io_num, spi_config->miso_io_num, spi_config->sclk_io_num, spi_system_host, spi_system_dc_gpio);
  290. if (spi_config->mosi_io_num != -1 && spi_config->sclk_io_num != -1) {
  291. spi_bus_initialize( spi_system_host, spi_config, SPI_DMA_CH_AUTO );
  292. if (spi_system_dc_gpio != -1) {
  293. gpio_reset_pin(spi_system_dc_gpio);
  294. gpio_set_direction( spi_system_dc_gpio, GPIO_MODE_OUTPUT );
  295. gpio_set_level( spi_system_dc_gpio, 0 );
  296. } else {
  297. ESP_LOGW(TAG, "No DC GPIO set, SPI display will not work");
  298. }
  299. } else {
  300. spi_system_host = -1;
  301. ESP_LOGW(TAG, "no SPI configured");
  302. }
  303. // create GPIO expanders
  304. gpio_exp_config_t gpio_exp_config;
  305. if(platform->has_dev && platform->dev.gpio_exp_count>0){
  306. for(int count = 0;count<platform->dev.gpio_exp_count;count++){
  307. sys_GPIOExp * exp = &platform->dev.gpio_exp[count];
  308. gpio_exp_config.base = exp->base;
  309. gpio_exp_config.count = exp->count;
  310. gpio_exp_config.phy.addr = exp->addr;
  311. if(exp->has_intr){
  312. gpio_exp_config.intr = exp->intr.pin;
  313. }
  314. else {
  315. ESP_LOGW(TAG,"Expander doesn't have intr pin");
  316. }
  317. if(exp->which_ExpType == sys_GPIOExp_spi_tag){
  318. gpio_exp_config.phy.cs_pin= exp->ExpType.spi.cs.pin;
  319. gpio_exp_config.phy.host = exp->ExpType.spi.host == sys_HostEnum_UNSPECIFIED_HOST ?sys_HostEnum_Host0:exp->ExpType.spi.host -1;
  320. gpio_exp_config.phy.speed = exp->ExpType.spi.speed>0?exp->ExpType.spi.speed:0;
  321. }
  322. else {
  323. gpio_exp_config.phy.port = exp->ExpType.i2c.port == sys_PortEnum_UNSPECIFIED_SYSTPORT?sys_PortEnum_SYSTEM:exp->ExpType.i2c.port -1;
  324. }
  325. strncpy(gpio_exp_config.model,sys_GPIOExpModelEnum_name(exp->model),sizeof(gpio_exp_config.model)-1);
  326. gpio_exp_create(&gpio_exp_config);
  327. }
  328. }
  329. if(platform->has_gpios ){
  330. // if(platform->gpios.has_GND){
  331. // platform->gpios.GND.level = 0;
  332. // set_gpio_level(&platform->gpios.GND,"GND", GPIO_MODE_OUTPUT);
  333. // }
  334. // if(platform->gpios.has_Vcc){
  335. // platform->gpios.Vcc.level = 1;
  336. // set_gpio_level(&platform->gpios.Vcc,"VCC", GPIO_MODE_OUTPUT);
  337. // }
  338. set_chip_power_gpio(&platform->gpios);
  339. }
  340. // system-wide PWM timer configuration
  341. ledc_timer_config_t pwm_timer = {
  342. .duty_resolution = LEDC_TIMER_13_BIT,
  343. .freq_hz = 5000,
  344. #ifdef CONFIG_IDF_TARGET_ESP32S3
  345. .speed_mode = LEDC_LOW_SPEED_MODE,
  346. #else
  347. .speed_mode = LEDC_HIGH_SPEED_MODE,
  348. #endif
  349. .timer_num = pwm_system.timer,
  350. };
  351. ledc_timer_config(&pwm_timer);
  352. led_svc_init();
  353. battery_svc_init();
  354. monitor_svc_init();
  355. }