/* * Copyright (c) 2019 Jan Van Winkel * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include "lvgl_display.h" void lvgl_flush_cb_mono(lv_disp_drv_t *disp_drv, const lv_area_t *area, lv_color_t *color_p) { uint16_t w = area->x2 - area->x1 + 1; uint16_t h = area->y2 - area->y1 + 1; struct lvgl_disp_data *data = (struct lvgl_disp_data *)disp_drv->user_data; const struct device *display_dev = data->display_dev; const bool is_epd = data->cap.screen_info & SCREEN_INFO_EPD; const bool is_last = lv_disp_flush_is_last(disp_drv); if (is_epd && !data->blanking_on && !is_last) { /* * Turn on display blanking when using an EPD * display. This prevents updates and the associated * flicker if the screen is rendered in multiple * steps. */ display_blanking_on(display_dev); data->blanking_on = true; } struct display_buffer_descriptor desc = { .buf_size = (w * h) / 8U, .width = w, .pitch = w, .height = h, .frame_incomplete = !is_last, }; display_write(display_dev, area->x1, area->y1, &desc, (void *)color_p); if (data->cap.screen_info & SCREEN_INFO_DOUBLE_BUFFER) { display_write(display_dev, area->x1, area->y1, &desc, (void *)color_p); } if (is_epd && is_last && data->blanking_on) { /* * The entire screen has now been rendered. Update the * display by disabling blanking. */ display_blanking_off(display_dev); data->blanking_on = false; } lv_disp_flush_ready(disp_drv); } void lvgl_set_px_cb_mono(lv_disp_drv_t *disp_drv, uint8_t *buf, lv_coord_t buf_w, lv_coord_t x, lv_coord_t y, lv_color_t color, lv_opa_t opa) { struct lvgl_disp_data *data = (struct lvgl_disp_data *)disp_drv->user_data; uint8_t *buf_xy; uint8_t bit; if (data->cap.screen_info & SCREEN_INFO_MONO_VTILED) { buf_xy = buf + x + y / 8 * buf_w; if (data->cap.screen_info & SCREEN_INFO_MONO_MSB_FIRST) { bit = 7 - y % 8; } else { bit = y % 8; } } else { buf_xy = buf + x / 8 + y * buf_w / 8; if (data->cap.screen_info & SCREEN_INFO_MONO_MSB_FIRST) { bit = 7 - x % 8; } else { bit = x % 8; } } if (data->cap.current_pixel_format == PIXEL_FORMAT_MONO10) { if (color.full == 0) { *buf_xy &= ~BIT(bit); } else { *buf_xy |= BIT(bit); } } else { if (color.full == 0) { *buf_xy |= BIT(bit); } else { *buf_xy &= ~BIT(bit); } } } void lvgl_rounder_cb_mono(lv_disp_drv_t *disp_drv, lv_area_t *area) { struct lvgl_disp_data *data = (struct lvgl_disp_data *)disp_drv->user_data; if (data->cap.screen_info & SCREEN_INFO_X_ALIGNMENT_WIDTH) { area->x1 = 0; area->x2 = data->cap.x_resolution - 1; } else { if (data->cap.screen_info & SCREEN_INFO_MONO_VTILED) { area->y1 &= ~0x7; area->y2 |= 0x7; } else { area->x1 &= ~0x7; area->x2 |= 0x7; } } }