🤖Have you ever tried Chat.M5Stack.com before asking??😎
    M5Stack Community
    • Categories
    • Recent
    • Tags
    • Popular
    • Users
    • Groups
    • Search
    • Register
    • Login

    Display on during deep sleep?

    Cores
    7
    29
    110.8k
    Loading More Posts
    • Oldest to Newest
    • Newest to Oldest
    • Most Votes
    Reply
    • Reply as topic
    Log in to reply
    This topic has been deleted. Only users with topic management privileges can see it.
    • felmueF
      felmue
      last edited by felmue

      Hello all

      the reason nothing can be seen on the LCD during deep sleep is that the backlight is off. The LCD backlight is driven by GPIO32 (PWM) which is stopped when the ESP32 goes into deep sleep. Luckily GPIO32 can be controlled by the Ultra Low Processor (ULP) too. Below is a small test program which implements PWM for GPIO32 with ULP C macros / assembler.

      Cheers
      Felix

      #include <M5Stack.h>
      #include "esp32/ulp.h"
      #include "driver/rtc_io.h"
      
      const int lcdBrightness = 1; // (1-255)
      
      void ulp_start(void) {
        // Slow memory initialization
        memset(RTC_SLOW_MEM, 0, 8192);
        // M5Stack LCD backlight is connected to GPIO32 (specify by +14)
        const gpio_num_t lcdPWMPin = GPIO_NUM_32;
        const int lcdPWMBit = RTCIO_GPIO32_CHANNEL + 14;
        // GPIO32 initialization (set to output and initial value is 0)
        rtc_gpio_init(lcdPWMPin);
        rtc_gpio_set_direction(lcdPWMPin, RTC_GPIO_MODE_OUTPUT_ONLY);
        rtc_gpio_set_level(lcdPWMPin, 0);
        // Define ULP program
        const ulp_insn_t  ulp_prog[] = {
          M_LABEL(1),
          I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 1), // on
          I_DELAY(lcdBrightness * 100),
          I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 0), // off
          I_DELAY(25500),
          M_BX(1),
        };
        // Run ULP program
        size_t size = sizeof(ulp_prog) / sizeof(ulp_insn_t);
        ulp_process_macros_and_load(0, ulp_prog, &size);
        ulp_run(0);
      }
      
      void setup() {
        M5.begin(true, false, true, false);
        M5.Power.begin();
        M5.Lcd.println("Put something onto the LCD...");
        ulp_start();
        M5.Lcd.println("Going to deep sleep");
        esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
        esp_deep_sleep_start();
      }
      
      void loop() {
      }
      

      GPIO translation table M5Stack / M5Core2
      Information about various M5Stack products.
      Code examples

      A 1 Reply Last reply Reply Quote 1
      • A
        AndyT @felmue
        last edited by

        @felmue Thanks for the above routine. Works well. However i was storing a RTC_bootCount so i could display a sequence of images on the screen after waking from sleep. However it seem the ulp over rights the bootCounter stored there. How do i need to change the above code to preserve the RTC_bootCounter?
        i tried this but never worked, I am new a ULP coding, so not sure how to locate a free memory space in ULP memory. and retrive it back into the main routine.
        // Define ULP program
        const ulp_insn_t ulp_prog[] = {
        I_ST(ulp_bootCount, R0, 0), // trying to store my bootCounter
        M_LABEL(1),
        I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 1), // on
        I_DELAY(lcdBrightness * 100),
        I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 0), // off
        I_DELAY(25500),
        M_BX(1),
        };
        and in my main code use ++ulp_bootCounter;
        please help . Thanks AndyT

        1 Reply Last reply Reply Quote 0
        • felmueF
          felmue
          last edited by felmue

          Hello @AndyT

          please have a look at the following example which stores variables after the ULP code: deep sleep example

          Cheers
          Felix

          GPIO translation table M5Stack / M5Core2
          Information about various M5Stack products.
          Code examples

          A 1 Reply Last reply Reply Quote 0
          • A
            AndyT
            last edited by

            This post is deleted!
            1 Reply Last reply Reply Quote 0
            • A
              AndyT @felmue
              last edited by AndyT

              @felmue Thanks for the link. Lot of code to sift through , spent hours trying to get clues from it. Tried snipets of code out of it, But still can't fully understand what' the problem is. I think i am missing a fundamental principle, how after a SLEEP the esp32 loads all it's variables which would include
              RTC_DATA_ATTR int bootCount = 0;
              So no wonder it is 0 when the main loop runs, as it is always being set to 0 on wake up. So confused about this from the outset the code even worked in the first place.
              But it did work until i introduced the ULP program to PWM a GPIO while sleeping to keep my esp32 M5Stack LCD back light pin operating.
              if I don't call the ulp_start(void) routine before sleeping then the ++bootCount increments on each wake just fine.
              Why is the below code effecting the RTC_DATA_ATTR int bootCount = 0; ???
              what do i need to add to fix it?
              I am using the Arduino IDE. So maybe there is an issue on where the Config is assigning the RTC_ to fast or slow ULP memory. I really can't fathom it out. why calling ulp_start(void) resets my counter to 0. Please give me an explanation and solution. Cheers Andy
              void ulp_start(void) {
              // Slow memory initialization
              memset(RTC_SLOW_MEM, 0, 8192);
              // M5Stack LCD backlight is connected to GPIO32 (specify by +14)
              const gpio_num_t lcdPWMPin = GPIO_NUM_32;
              const int lcdPWMBit = RTCIO_GPIO32_CHANNEL + 14;
              // GPIO32 initialization (set to output and initial value is 0)
              rtc_gpio_init(lcdPWMPin);
              rtc_gpio_set_direction(lcdPWMPin, RTC_GPIO_MODE_OUTPUT_ONLY);
              rtc_gpio_set_level(lcdPWMPin, 0);
              // Define ULP program
              const ulp_insn_t ulp_prog[] = {
              M_LABEL(1),
              I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 1), // on
              I_DELAY(lcdBrightness * 100),
              I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 0), // off
              I_DELAY(25500),
              M_BX(1),
              };
              // Run ULP program
              size_t size = sizeof(ulp_prog) / sizeof(ulp_insn_t);
              ulp_process_macros_and_load(0, ulp_prog, &size);
              ulp_run(00);
              }

              A 1 Reply Last reply Reply Quote 0
              • A
                AndyT @AndyT
                last edited by AndyT

                @andyt coming From a background of coding 8bit PIC's etc, I have a fare grip on assembler language. So once i get to grips with this ULP coding methods . My next step will be to add a routine to check battery level% change and wake from sleep to update the LCD battery level icon. As at the moment while it is sleeping with the LCD on the M5stack could run out of power and shut down. Or stay displaying the wrong battery level while it sleeps as at the moment it only updates while awake. Just need to get my head around the fundamentals of data handling and storage and setup of ULP. Good that there is more info on this topic these days. Exciting times!!![
                My M5stack waking up and quickly turning on wifi, grabbing my local environment agency river levels .and showing the levels and if they are rising or falling triggering speaker alarms and displaying waring icons when reaching or above flood levels. To conserve battery it is asleep while the backlight remains on at a duty cycle dependant on battery level. Getting dimmer as the battery % lowers for 10 minutes minimum and then display the weather forecast from a weather API. cycling through these screens quicker or slower depending on the level of importance to the viewer. So while the river levels are low the weather will be displayed longer. pressing one of the front buttons wakes the device and scrolls to the next page, so that the user can look through the pages at a glance. It's like doing a painting. Always improving to the code to finished slick working device. next stage is to right a nice routine for a user interface to input their own wifi and postcode location. So the device can select local levels and weather.

                1 Reply Last reply Reply Quote 0
                • felmueF
                  felmue
                  last edited by felmue

                  Hello Andy

                  ULP code and ULP variables share the same area of RTC_SLOW_MEM which is about 8 kB. Right now your ULP variable is reset to 0 at every boot since all of that area is reset to 0 by this command: memset(RTC_SLOW_MEM, 0, 8192);. In addition the ULP code and your ULP variable both are located at the same offset 0 in that area. ulp_process_macros_and_load(0, ulp_prog, &size);

                  In order to fix this the RTC_SLOW_MEM needs to be orgnanised, e.g. the ULP variables need to be located after (or before) the ULP code and the area used for ULP variables needs to be excluded from being cleared to 0 at every boot.

                  There are many ways to solve this. In the example I've linked earlier ULP variables are located shortly after the ULP code (offset 36). The solution I've chosen puts the ULP variables into the last of the eight 1 kB blocks of RTC_SLOW_MEM. That leaves more than enough space (the first 7 kB) for ULP code.

                  Note: Next to the boot count ULP variable I've added a second ULP dummy variable to illustrate how multiple ULP variables could be handled.

                  Cheers
                  Felix

                  #include <M5Stack.h>
                  #include "esp32/ulp.h"
                  #include "driver/rtc_io.h"
                  
                  const int lcdBrightness = 50; // (1-255)
                  
                  // RTC slow memory map (vars are in the last 1 kB of total 8 kB)
                  // Note: all ULP commands / vars are 4 bytes in size
                  const size_t rtc_slow_mem_total_size = 2048; // 8 kB = 4 * 2048
                  const size_t rtc_slow_mem_vars_space = 256;  // 1 kB = 4 * 256
                  const size_t rtc_slow_mem_vars_start = rtc_slow_mem_total_size - rtc_slow_mem_vars_space;
                  const size_t rtc_slow_mem_prog_start = 0;
                  const size_t rtc_slow_mem_prog_space = rtc_slow_mem_total_size - rtc_slow_mem_vars_space; // 7 kB
                  
                  const size_t ulp_boot_count_var_offset = 0;
                  const size_t ulp_dummy_var_offset = 1;
                  
                  void ulp_start(void) {
                    // RTC slow mem init (except space allocated for vars)
                    memset(RTC_SLOW_MEM, 0, rtc_slow_mem_prog_space * sizeof(uint32_t));
                    // M5Stack LCD backlight is connected to GPIO32 (specify by +14)
                    const gpio_num_t lcdPWMPin = GPIO_NUM_32;
                    const int lcdPWMBit = RTCIO_GPIO32_CHANNEL + 14;
                    // GPIO32 initialization (set to output and initial value is 0)
                    rtc_gpio_init(lcdPWMPin);
                    rtc_gpio_set_direction(lcdPWMPin, RTC_GPIO_MODE_OUTPUT_ONLY);
                    rtc_gpio_set_level(lcdPWMPin, 0);
                    // Define ULP program
                    const ulp_insn_t ulp_prog[] = {
                      M_LABEL(1),
                      I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 1), // on
                      I_DELAY(lcdBrightness * 100),
                      I_WR_REG(RTC_GPIO_OUT_REG, lcdPWMBit, lcdPWMBit, 0), // off
                      I_DELAY(25500),
                      M_BX(1),
                    };
                    // Run ULP program
                    size_t size = sizeof(ulp_prog) / sizeof(ulp_insn_t);
                  
                    ulp_process_macros_and_load(rtc_slow_mem_prog_start, ulp_prog, &size);
                    ulp_run(rtc_slow_mem_prog_start);
                  }
                  
                  void setup() {
                    M5.begin(true, false, true, false);
                    M5.Power.begin();
                    M5.Lcd.println("Put something onto the LCD...");
                  
                    if(esp_sleep_get_wakeup_cause() == ESP_SLEEP_WAKEUP_TIMER)
                    {
                      M5.Lcd.println("Woken from timer");
                      RTC_SLOW_MEM[rtc_slow_mem_vars_start + ulp_boot_count_var_offset]++;
                    }
                    else
                    {
                      M5.Lcd.println("Woken after reset");
                      RTC_SLOW_MEM[rtc_slow_mem_vars_start + ulp_boot_count_var_offset] = 0;
                      RTC_SLOW_MEM[rtc_slow_mem_vars_start + ulp_dummy_var_offset] = 4711;
                    }
                    M5.Lcd.println(RTC_SLOW_MEM[rtc_slow_mem_vars_start + ulp_boot_count_var_offset]);
                    M5.Lcd.println(RTC_SLOW_MEM[rtc_slow_mem_vars_start + ulp_dummy_var_offset]);
                  
                    ulp_start();
                  
                    M5.Lcd.println("Going to deep sleep");
                    esp_sleep_enable_timer_wakeup(5000);
                    esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
                    esp_deep_sleep_start();
                  }
                  
                  void loop() {
                  }
                  

                  GPIO translation table M5Stack / M5Core2
                  Information about various M5Stack products.
                  Code examples

                  A 1 Reply Last reply Reply Quote 0
                  • A
                    AndyT @felmue
                    last edited by AndyT

                    @felmue Yes!! Thank you. You are a Genius. Saved me hours. I get it !. It's working in my code.
                    0_1604174249013_be97bcb0-73ec-4ef4-aaf0-ad7a9a5c5df7-image.png

                    1 Reply Last reply Reply Quote 0
                    • felmueF
                      felmue
                      last edited by

                      Hello Andy

                      thank you very much. I am happy to hear it works to your liking.

                      Happy Stacking!
                      Felix

                      GPIO translation table M5Stack / M5Core2
                      Information about various M5Stack products.
                      Code examples

                      A 1 Reply Last reply Reply Quote 0
                      • A
                        AndyT @felmue
                        last edited by

                        @felmue Seems my Arduino IDE ulp.h does not have a macro I_i2c_RD. And i can't find a way to assemble the raw assembly instruction set i2C_RD. Am i missing something? As i want to take advantage of the the assembly instructions and also the latest set of Macros. Just spent a day going around the internet and came up with nothing. Trying to write ULP code to wake the M5stack when the battery % changes. Sorry again to trouble you.

                        1 Reply Last reply Reply Quote 0
                        • felmueF
                          felmue
                          last edited by

                          Hello Andy

                          I am afraid you're out of luck. The arduino-esp32 library is based on IDF release 3.3 and the ULP macros for I2C seem to be missing / not implemented yet.

                          #define OPCODE_I2C 3 /*!< Instruction: read/write I2C (not implemented yet) */

                          But even with the macros in place it would probably still not work. According to this info it seems that the ULP only allows for two predefined sets of GPIOs to be used for SDA and SDL, none of which are GPIO21/22 used in M5Stack.

                          Felix

                          GPIO translation table M5Stack / M5Core2
                          Information about various M5Stack products.
                          Code examples

                          A 1 Reply Last reply Reply Quote 0
                          • A
                            AndyT @felmue
                            last edited by

                            @felmue Thank you again for the prompt replay. Thank you for the info. I drew a blank at the link you provide as well . Also a good find regarding the pin assignments of the hard coded SDA and SDL for channels 0,1. Did find this link for ulp.h using I_i2C_ READ I_I2C_WRITE etc. But could not get the ZIPed Library to install into my Arduino IDE to try it out.
                            https://github.com/espressif/esp-idf/blob/master/components/ulp/include/esp32/ulp.h
                            Maybe start a new thread soon on this as i am getting off the original topic here now that your ULP code above fixed the display going off while sleeping. Just for the record now getting 24hrs off two 700mh add on battery modules at well bright 200 brightness level while the 5Stack is sleeping .

                            A 1 Reply Last reply Reply Quote 0
                            • A
                              AndyT @AndyT
                              last edited by

                              @andyt I now have the M5Stack black top and a grey top. The ulp program works well to keep the Black top version backlight running at varying levels of brightness. But if i set the PWM to less than 60 on the grey top version. The back light switches off completely after 15 seconds of sleep. which limits the available brightness levels between 60 and 255. which makes the brightness change hardly noticeable Not sure what is going on with the differences between the two M5stacks?

                              1 Reply Last reply Reply Quote 0
                              • felmueF
                                felmue
                                last edited by

                                Hello @AndyT

                                hmm, strange, if anything I'd expected the M5Stack Gray to work with low brightness level and the M5Stack Black to not behave due to the fact that the M5Stack Gray has additional hardware (e.g. IMU) consuming power to keep the device powered on.

                                That said, I suspect the IP5306 switching off due to too light load. Try to add the following statement before going into deep sleep. It should prevent the IP5306 from shutting down under light load.

                                M5.Power.setPowerBoostKeepOn(true);
                                

                                Happy Stacking!
                                Felix

                                GPIO translation table M5Stack / M5Core2
                                Information about various M5Stack products.
                                Code examples

                                1 Reply Last reply Reply Quote 0
                                • First post
                                  Last post