Add IPanelDevice.
This commit is contained in:
		
							parent
							
								
									670a523a16
								
							
						
					
					
						commit
						b3d830cdeb
					
				| @ -17,8 +17,8 @@ | ||||
| // We must use a mutex to protect it.
 | ||||
| _lock_t ScopedLock::lock_; | ||||
| 
 | ||||
| Display::Display(const I2C &i2c) : | ||||
|     panel_(i2c) | ||||
| Display::Display(IPanelDevice *device) : | ||||
|     panel_(device) | ||||
| { | ||||
|   if (!lv_is_initialized()) { | ||||
|     ESP_LOGI(TAG, "Initialize LVGL"); | ||||
| @ -26,26 +26,25 @@ Display::Display(const I2C &i2c) : | ||||
|   } | ||||
| 
 | ||||
|   // Create a lvgl display.
 | ||||
|   // TODO: Use PanelDevice functions to get SSD1306 specs
 | ||||
|   display_ = lv_display_create(LCD_H_RES, LCD_V_RES); | ||||
|   lv_display_ = panel_.device_->create_display(); | ||||
|   // associate the i2c panel handle to the display
 | ||||
|   lv_display_set_user_data(display_, panel_.get()); | ||||
|   lv_display_set_user_data(lv_display_, panel_.esp_panel_); | ||||
| 
 | ||||
|   // Create draw buffer.
 | ||||
|   ESP_LOGI(TAG, "Allocate separate LVGL draw buffers"); | ||||
|   // TODO: Use PanelDevice functions to get SSD1306 specs
 | ||||
|   buf_size_ = LCD_H_RES * LCD_V_RES / 8 + LVGL_PALETTE_SIZE; | ||||
|   buf_ = heap_caps_calloc(1, buf_size_, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); | ||||
|   assert(buf_); | ||||
|   lv_buf_ = heap_caps_calloc(1, panel_.device_->lv_buf_size_, | ||||
|                              MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); | ||||
|   assert(lv_buf_); | ||||
| 
 | ||||
|   // LVGL9 suooprt new monochromatic format.
 | ||||
|   lv_display_set_color_format(display_, LV_COLOR_FORMAT_I1); | ||||
|   lv_display_set_color_format(lv_display_, LV_COLOR_FORMAT_I1); | ||||
|   // Initialize LVGL draw buffers.
 | ||||
|   lv_display_set_buffers(display_, buf_, nullptr, buf_size_, | ||||
|   lv_display_set_buffers(lv_display_, lv_buf_, nullptr, | ||||
|                          panel_.device_->lv_buf_size_, | ||||
|                          LV_DISPLAY_RENDER_MODE_FULL); | ||||
|   lv_display_set_rotation(display_, LV_DISPLAY_ROTATION_0); | ||||
|   lv_display_set_rotation(lv_display_, LV_DISPLAY_ROTATION_0); | ||||
|   // Set callback which can copy the rendered image to an area of the display.
 | ||||
|   lv_display_set_flush_cb(display_, Display::lvgl_flush_cb); | ||||
|   lv_display_set_flush_cb(lv_display_, Display::lvgl_flush_cb); | ||||
| 
 | ||||
|   ESP_LOGI(TAG, "Register io panel callback for LVGL flush ready notification"); | ||||
|   const esp_lcd_panel_io_callbacks_t cbs = { | ||||
| @ -54,7 +53,7 @@ Display::Display(const I2C &i2c) : | ||||
|   /* Register done callback */ | ||||
|   ESP_ERROR_CHECK( | ||||
|       esp_lcd_panel_io_register_event_callbacks(panel_.io_handle_, &cbs, | ||||
|                                                 display_)); | ||||
|                                                 lv_display_)); | ||||
| 
 | ||||
|   // TODO: What is this
 | ||||
|   ESP_LOGI(TAG, "Use esp_timer as LVGL tick timer"); | ||||
| @ -81,7 +80,7 @@ void Display::set_text(const char *text, | ||||
|   ScopedLock lock; | ||||
| 
 | ||||
|   ESP_LOGI(TAG, "Display LVGL Scroll Text"); | ||||
|   lv_obj_t *scr = lv_display_get_screen_active(display_); | ||||
|   lv_obj_t *scr = lv_display_get_screen_active(lv_display_); | ||||
|   objects_[name] = lv_label_create(scr); | ||||
| 
 | ||||
|   // Circular scroll.
 | ||||
| @ -91,7 +90,7 @@ void Display::set_text(const char *text, | ||||
| 
 | ||||
|   // Set the size of the screen.
 | ||||
|   // If you use rotation 90 or 270 use lv_display_get_vertical_resolution.
 | ||||
|   lv_obj_set_width(obj, lv_display_get_horizontal_resolution(display_)); | ||||
|   lv_obj_set_width(obj, lv_display_get_horizontal_resolution(lv_display_)); | ||||
|   lv_obj_align(obj, align, 0, 0); | ||||
| } | ||||
| 
 | ||||
| @ -166,13 +165,13 @@ void Display::lvgl_increase_tick(void *) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| I2C::I2C() : | ||||
| I2C::I2C(gpio_num_t sda, gpio_num_t scl) : | ||||
|     i2c_bus_(nullptr), | ||||
|     bus_config_( | ||||
|         (i2c_master_bus_config_t) { | ||||
|             .i2c_port = I2C_BUS_PORT, | ||||
|             .sda_io_num = PIN_SDA, | ||||
|             .scl_io_num = PIN_SCL, | ||||
|             .sda_io_num = sda, | ||||
|             .scl_io_num = scl, | ||||
|             .clk_source = I2C_CLK_SRC_DEFAULT, | ||||
|             .glitch_ignore_cnt = 7, | ||||
|             .flags { | ||||
| @ -185,43 +184,29 @@ I2C::I2C() : | ||||
|   ESP_ERROR_CHECK(i2c_new_master_bus(&bus_config_, &i2c_bus_)); | ||||
| } | ||||
| 
 | ||||
| Panel::Panel(i2c_master_bus_handle_t i2c) : | ||||
| Panel::Panel(IPanelDevice *device) : | ||||
|     device_(device), | ||||
|     io_handle_(nullptr), | ||||
|     panel_(nullptr), | ||||
|     esp_panel_(nullptr), | ||||
|     // According to SSD1306 datasheet
 | ||||
|     panel_config_( | ||||
|         (esp_lcd_panel_dev_config_t) { | ||||
|             .reset_gpio_num = PIN_RST, | ||||
|             .reset_gpio_num = device_->reset_gpio_num_, | ||||
|             .bits_per_pixel = 1, | ||||
|         } | ||||
|     ), | ||||
|     // According to SSD1306 datasheet
 | ||||
|     io_config_( | ||||
|         (esp_lcd_panel_io_i2c_config_t) { | ||||
|             .dev_addr = I2C_HW_ADDR, | ||||
|             .control_phase_bytes = 1, | ||||
|             .dc_bit_offset = 6, | ||||
|             .lcd_cmd_bits = LCD_CMD_BITS, | ||||
|             .lcd_param_bits = LCD_CMD_BITS, | ||||
|             .scl_speed_hz = LCD_PIXEL_CLOCK_HZ, | ||||
|             // .vendor_config should be set in IPanelDevice::init_panel override
 | ||||
|         } | ||||
|     ) | ||||
| { | ||||
|   ESP_LOGI(TAG, "Install panel IO"); | ||||
|   ESP_ERROR_CHECK(esp_lcd_new_panel_io_i2c(i2c, &io_config_, &io_handle_)); | ||||
|   ESP_ERROR_CHECK(esp_lcd_new_panel_io_i2c( | ||||
|       device_->i2c_bus_, &device_->io_config_, &io_handle_)); | ||||
| 
 | ||||
|   ESP_LOGI(TAG, "Install SSD1306 panel driver"); | ||||
|   SSD1306 ssd1306( | ||||
|       (esp_lcd_panel_ssd1306_config_t) { | ||||
|           .height = LCD_V_RES, | ||||
|       } | ||||
|   ); | ||||
|   ssd1306.create_panel(panel_config_, io_handle_, panel_); | ||||
|   device_->create_panel(panel_config_, io_handle_, esp_panel_); | ||||
| 
 | ||||
|   ESP_LOGI(TAG, "Resetting panel display"); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_reset(panel_)); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_reset(esp_panel_)); | ||||
|   ESP_LOGI(TAG, "Initializing panel display"); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_init(panel_)); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_init(esp_panel_)); | ||||
|   ESP_LOGI(TAG, "Turning on panel display"); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_disp_on_off(panel_, true)); | ||||
|   ESP_ERROR_CHECK(esp_lcd_panel_disp_on_off(esp_panel_, true)); | ||||
| } | ||||
|  | ||||
| @ -20,10 +20,8 @@ | ||||
| #define LVGL_TASK_STACK_SIZE   (4 * 1024) | ||||
| #define LVGL_TASK_PRIORITY     2 | ||||
| 
 | ||||
| static const char *TAG = "lcd-panel"; | ||||
| 
 | ||||
| struct I2C { | ||||
|   I2C(); | ||||
|   I2C(gpio_num_t sda, gpio_num_t scl); | ||||
| 
 | ||||
|   ~I2C() = default; | ||||
| 
 | ||||
| @ -45,40 +43,29 @@ struct ScopedLock { | ||||
| 
 | ||||
| class Panel { | ||||
| public: | ||||
|   explicit Panel(i2c_master_bus_handle_t i2c); | ||||
| 
 | ||||
|   explicit Panel(const I2C &i2c) : Panel(i2c.i2c_bus_) { } | ||||
|   explicit Panel(IPanelDevice *device); | ||||
| 
 | ||||
|   ~Panel() = default; | ||||
| 
 | ||||
|   [[nodiscard]] inline const esp_lcd_panel_t *get() const { return panel_; } | ||||
| 
 | ||||
|   [[nodiscard]] inline esp_lcd_panel_t *get() { return panel_; } | ||||
| 
 | ||||
|   [[nodiscard]] inline const esp_lcd_panel_t * | ||||
|   operator*() const { return get(); } | ||||
| 
 | ||||
|   [[nodiscard]] inline esp_lcd_panel_t *operator*() { return get(); } | ||||
|   IPanelDevice *device_; | ||||
| 
 | ||||
|   esp_lcd_panel_io_handle_t io_handle_; | ||||
| 
 | ||||
|   esp_lcd_panel_handle_t panel_; | ||||
|   esp_lcd_panel_handle_t esp_panel_; | ||||
| 
 | ||||
| private: | ||||
|   esp_lcd_panel_dev_config_t panel_config_; | ||||
| 
 | ||||
|   esp_lcd_panel_io_i2c_config_t io_config_; | ||||
| }; | ||||
| 
 | ||||
| class Display { | ||||
| public: | ||||
|   explicit Display(const I2C &i2c); | ||||
|   explicit Display(IPanelDevice *device); | ||||
| 
 | ||||
|   ~Display() = default; | ||||
| 
 | ||||
|   [[nodiscard]] inline const lv_display_t *get() const { return display_; } | ||||
|   [[nodiscard]] inline const lv_display_t *get() const { return lv_display_; } | ||||
| 
 | ||||
|   [[nodiscard]] inline lv_display_t *get() { return display_; } | ||||
|   [[nodiscard]] inline lv_display_t *get() { return lv_display_; } | ||||
| 
 | ||||
|   [[nodiscard]] inline const lv_display_t *operator*() const { return get(); } | ||||
| 
 | ||||
| @ -104,13 +91,10 @@ public: | ||||
| private: | ||||
|   Panel panel_; | ||||
| 
 | ||||
|   lv_display_t *display_; | ||||
| 
 | ||||
|   // Size of the draw buffer associated with the lv_display_t.
 | ||||
|   size_t buf_size_; | ||||
|   lv_display_t *lv_display_; | ||||
| 
 | ||||
|   // Draw buffer associated with the lv_display_t.
 | ||||
|   void *buf_; | ||||
|   void *lv_buf_; | ||||
| 
 | ||||
|   // Objects stored in the screen associated with this display.
 | ||||
|   // @sa Display::set_text
 | ||||
|  | ||||
| @ -1,11 +1,19 @@ | ||||
| #include "display.h" | ||||
| #include "ssd1306.h" | ||||
| 
 | ||||
| // Pin may vary based on your schematic.
 | ||||
| #define PIN_SDA                GPIO_NUM_21 | ||||
| #define PIN_SCL                GPIO_NUM_22 | ||||
| #define PIN_RST                -1 | ||||
| 
 | ||||
| // TODO: Can this be static since there can only be one initialization?
 | ||||
| I2C i2c; | ||||
| // TODO: Store RST in I2C and retrieve within SSD instead of the #define
 | ||||
| I2C i2c(PIN_SDA, PIN_SCL); | ||||
| 
 | ||||
| void setup() | ||||
| { | ||||
|   Display d(i2c); | ||||
|   SSD1306 ssd1306(i2c.i2c_bus_); | ||||
|   Display d(&ssd1306); | ||||
| 
 | ||||
|   d.set_text("Test test 12345678910", | ||||
|              "test-text1", | ||||
|  | ||||
| @ -4,62 +4,62 @@ | ||||
| #include <esp_lcd_panel_dev.h> | ||||
| #include <esp_lcd_panel_ops.h> | ||||
| #include <esp_lcd_panel_ssd1306.h> | ||||
| #include <driver/i2c_types.h> | ||||
| #include <esp_lcd_panel_io.h> | ||||
| #include <esp_log.h> | ||||
| #include "display/lv_display.h" | ||||
| 
 | ||||
| #define LVGL_PALETTE_SIZE      8 | ||||
| 
 | ||||
| template<typename T> | ||||
| class PanelDeviceInterface { | ||||
| static const char *TAG = "lcd-panel"; | ||||
| 
 | ||||
| class IPanelDevice { | ||||
| public: | ||||
|   PanelDeviceInterface(size_t width, size_t height) : width_(width), height_(height) { } | ||||
|   explicit IPanelDevice(i2c_master_bus_handle_t i2c, | ||||
|                         int reset_gpio_num, | ||||
|                         esp_lcd_panel_io_i2c_config_t io_config) : | ||||
|       reset_gpio_num_(reset_gpio_num), | ||||
|       i2c_bus_(i2c), | ||||
|       io_config_(io_config) { } | ||||
| 
 | ||||
|   ~PanelDeviceInterface() = default; | ||||
| 
 | ||||
|   [[nodiscard]] size_t get_width() const { return width_; } | ||||
| 
 | ||||
|   [[nodiscard]] size_t get_height() const { return height_; } | ||||
| 
 | ||||
|   [[nodiscard]] size_t get_area() const { return width_ * height_; } | ||||
| 
 | ||||
|   [[nodiscard]] size_t get_draw_buffer_size() const | ||||
|   { | ||||
|     // LVGL reserves 2x4 bytes in the buffer to be used as a palette.
 | ||||
|     return width_ * height_ / 8 + LVGL_PALETTE_SIZE; | ||||
|   } | ||||
|   virtual ~IPanelDevice() = default; | ||||
| 
 | ||||
|   [[nodiscard]] lv_display_t *create_display() const | ||||
|   { | ||||
|     return lv_display_create(width_, height_); | ||||
|     auto display = lv_display_create(width_, height_); | ||||
|     assert(display); | ||||
|     return display; | ||||
|   } | ||||
| 
 | ||||
|   [[nodiscard]] T &get_vendor_config() const { return vendor_config_; } | ||||
|   void create_panel(esp_lcd_panel_dev_config_t &config, | ||||
|                     esp_lcd_panel_io_handle_t io, | ||||
|                     esp_lcd_panel_handle_t &panel) | ||||
|   { | ||||
|     // If the passed handle is already allocated, delete it.
 | ||||
|     if (panel != nullptr) { | ||||
|       ESP_LOGI(TAG, "Removing unused panel"); | ||||
|       esp_lcd_panel_del(panel); | ||||
|     } | ||||
| 
 | ||||
|   void create_panel(esp_lcd_panel_dev_config_t &panel_config, | ||||
|                     esp_lcd_panel_io_handle_t io_handle, | ||||
|                     esp_lcd_panel_handle_t &panel_handle); | ||||
|     ESP_LOGI(TAG, "Install SSD1306 panel driver"); | ||||
|     init_panel(config, io, panel); | ||||
|   } | ||||
| 
 | ||||
|   int32_t width_; | ||||
|   int32_t height_; | ||||
|   int reset_gpio_num_; | ||||
| 
 | ||||
|   // LVGL reserves 2x4 bytes in the buffer to be used as a palette.
 | ||||
|   size_t lv_buf_size_; | ||||
|   // TODO: Can we use a static accessor in I2C instead?
 | ||||
|   i2c_master_bus_handle_t i2c_bus_; | ||||
| 
 | ||||
|   esp_lcd_panel_io_i2c_config_t io_config_; | ||||
| 
 | ||||
| private: | ||||
|   size_t width_; | ||||
|   size_t height_; | ||||
| 
 | ||||
| protected: | ||||
|   T vendor_config_; | ||||
|   virtual void init_panel(esp_lcd_panel_dev_config_t &config, | ||||
|                           esp_lcd_panel_io_handle_t io, | ||||
|                           esp_lcd_panel_handle_t &panel) = 0; | ||||
| }; | ||||
| 
 | ||||
| template<typename T> | ||||
| void PanelDeviceInterface<T>::create_panel(esp_lcd_panel_dev_config_t &config, | ||||
|                                            esp_lcd_panel_io_handle_t io, | ||||
|                                            esp_lcd_panel_handle_t &panel) | ||||
| { | ||||
|   // If the passed handle is already allocated, delete it.
 | ||||
|   if (panel != nullptr) { | ||||
|     esp_lcd_panel_del(panel); | ||||
|   } | ||||
| 
 | ||||
|   // Set the vendor config and allocate a new panel handle.
 | ||||
|   config.vendor_config = &vendor_config_; | ||||
|   // TODO: This needs moved to SSD1306
 | ||||
|   ESP_ERROR_CHECK(esp_lcd_new_panel_ssd1306(io, &config, &panel)); | ||||
| } | ||||
| 
 | ||||
| #endif // PANEL_DEVICE_H
 | ||||
|  | ||||
| @ -4,10 +4,34 @@ | ||||
| // To use LV_COLOR_FORMAT_I1 we need an extra buffer to hold the converted data.
 | ||||
| uint8_t SSD1306::oled_buffer_[LCD_H_RES * LCD_V_RES / 8]; | ||||
| 
 | ||||
| SSD1306::SSD1306(esp_lcd_panel_ssd1306_config_t ssd_config, | ||||
| SSD1306::SSD1306(i2c_master_bus_handle_t i2c, | ||||
|                  esp_lcd_panel_ssd1306_config_t config, | ||||
|                  int width, | ||||
|                  int height) : | ||||
|     PanelDeviceInterface<esp_lcd_panel_ssd1306_config_t>(width, height) | ||||
|     IPanelDevice(i2c, | ||||
|                  PIN_RST, | ||||
|                  (esp_lcd_panel_io_i2c_config_t) { | ||||
|                      .dev_addr = I2C_HW_ADDR, | ||||
|                      .control_phase_bytes = 1, | ||||
|                      .dc_bit_offset = 6, | ||||
|                      .lcd_cmd_bits = LCD_CMD_BITS, | ||||
|                      .lcd_param_bits = LCD_CMD_BITS, | ||||
|                      .scl_speed_hz = LCD_PIXEL_CLOCK_HZ, | ||||
|                  } | ||||
|     ), | ||||
|     ssd1306_config_(config) | ||||
| { | ||||
|   vendor_config_ = ssd_config; | ||||
|   this->width_ = width; | ||||
|   this->height_ = height; | ||||
|   this->reset_gpio_num_ = PIN_RST; | ||||
|   this->lv_buf_size_ = width_ * height_ / 8 + LVGL_PALETTE_SIZE; | ||||
| } | ||||
| 
 | ||||
| void SSD1306::init_panel(esp_lcd_panel_dev_config_t &config, | ||||
|                          esp_lcd_panel_io_handle_t io, | ||||
|                          esp_lcd_panel_handle_t &panel) | ||||
| { | ||||
|   // Allocate SSD1306 panel handle.
 | ||||
|   config.vendor_config = &ssd1306_config_; | ||||
|   ESP_ERROR_CHECK(esp_lcd_new_panel_ssd1306(io, &config, &panel)); | ||||
| } | ||||
|  | ||||
| @ -17,20 +17,32 @@ | ||||
| #define LCD_CMD_BITS           8 | ||||
| #define LCD_PARAM_BITS         8 | ||||
| 
 | ||||
| // Pin may vary based on your schematic.
 | ||||
| #define PIN_SDA                GPIO_NUM_21 | ||||
| #define PIN_SCL                GPIO_NUM_22 | ||||
| #define PIN_RST                -1 | ||||
| 
 | ||||
| class SSD1306 : public PanelDeviceInterface<esp_lcd_panel_ssd1306_config_t> { | ||||
| class SSD1306 : public IPanelDevice { | ||||
| public: | ||||
|   explicit SSD1306(esp_lcd_panel_ssd1306_config_t ssd_config, | ||||
|                    int width = SCREEN_WIDTH, int height = SCREEN_HEIGHT); | ||||
|   // Constructors allow overriding ssd1306 config.
 | ||||
|   explicit SSD1306(i2c_master_bus_handle_t i2c) : | ||||
|       SSD1306(i2c, {.height = SCREEN_HEIGHT}) { } | ||||
| 
 | ||||
|   ~SSD1306() = default; | ||||
|   explicit SSD1306(i2c_master_bus_handle_t i2c, | ||||
|                    esp_lcd_panel_ssd1306_config_t config, | ||||
|                    int width = SCREEN_WIDTH, | ||||
|                    int height = SCREEN_HEIGHT); | ||||
| 
 | ||||
|   virtual ~SSD1306() = default; | ||||
| 
 | ||||
|   // The configuration structure specific to the SSD1306.
 | ||||
|   esp_lcd_panel_ssd1306_config_t ssd1306_config_; | ||||
| 
 | ||||
|   // For LV_COLOR_FORMAT_I1 we need an extra buffer to hold the converted data.
 | ||||
|   static uint8_t oled_buffer_[LCD_H_RES * LCD_V_RES / 8]; | ||||
| 
 | ||||
| private: | ||||
|   void init_panel(esp_lcd_panel_dev_config_t &config, | ||||
|                   esp_lcd_panel_io_handle_t io, | ||||
|                   esp_lcd_panel_handle_t &panel) override; | ||||
| 
 | ||||
| }; | ||||
| 
 | ||||
| #endif // SSD1306_H
 | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user