2016-06-08 07:25:08 +02:00
|
|
|
/**
|
|
|
|
* @file lv_vdb.c
|
2018-06-19 09:49:58 +02:00
|
|
|
*
|
2016-06-08 07:25:08 +02:00
|
|
|
*/
|
2018-07-07 11:53:22 +02:00
|
|
|
|
2018-08-04 08:45:22 +02:00
|
|
|
/*********************
|
|
|
|
* INCLUDES
|
|
|
|
*********************/
|
2018-07-07 11:53:22 +02:00
|
|
|
#include "lv_vdb.h"
|
2016-06-08 07:25:08 +02:00
|
|
|
#if LV_VDB_SIZE != 0
|
|
|
|
|
2017-10-09 15:21:26 +02:00
|
|
|
#include "../lv_hal/lv_hal_disp.h"
|
2018-09-12 09:03:48 +02:00
|
|
|
#include "../lv_misc/lv_log.h"
|
2016-06-08 07:25:08 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
/*********************
|
|
|
|
* DEFINES
|
|
|
|
*********************/
|
2018-11-24 07:49:49 +01:00
|
|
|
#ifndef LV_ATTRIBUTE_FLUSH_READY
|
|
|
|
#define LV_ATTRIBUTE_FLUSH_READY
|
|
|
|
#endif
|
2016-06-08 07:25:08 +02:00
|
|
|
|
2019-02-12 05:39:44 +01:00
|
|
|
#ifndef LV_ATTRIBUTE_MEM_ALIGN
|
|
|
|
#define LV_ATTRIBUTE_MEM_ALIGN
|
|
|
|
#endif
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* TYPEDEFS
|
|
|
|
**********************/
|
2017-11-29 10:46:59 +01:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC PROTOTYPES
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* STATIC VARIABLES
|
|
|
|
**********************/
|
2017-11-30 09:53:26 +01:00
|
|
|
|
2018-06-19 09:49:58 +02:00
|
|
|
/*Simple VDB*/
|
2018-12-22 19:08:03 +01:00
|
|
|
#if LV_VDB_DOUBLE == 0
|
2017-11-30 09:53:26 +01:00
|
|
|
# if LV_VDB_ADR == 0
|
2018-06-19 09:49:58 +02:00
|
|
|
/*If the buffer address is not specified simply allocate it*/
|
2019-02-12 05:39:44 +01:00
|
|
|
static LV_ATTRIBUTE_MEM_ALIGN uint8_t vdb_buf[LV_VDB_SIZE_IN_BYTES];
|
2018-10-05 17:22:49 +02:00
|
|
|
static lv_vdb_t vdb = {.buf = (lv_color_t *)vdb_buf};
|
|
|
|
# else /*LV_VDB_ADR != 0*/
|
2018-06-19 09:49:58 +02:00
|
|
|
/*If the buffer address is specified use that address*/
|
|
|
|
static lv_vdb_t vdb = {.buf = (lv_color_t *)LV_VDB_ADR};
|
2017-11-30 09:53:26 +01:00
|
|
|
# endif
|
2018-12-22 19:08:03 +01:00
|
|
|
|
|
|
|
/*LV_VDB_DOUBLE != 0*/
|
|
|
|
#else
|
2018-06-19 09:49:58 +02:00
|
|
|
/*Double VDB*/
|
2018-12-22 19:08:03 +01:00
|
|
|
static uint8_t vdb_active = 0;
|
2017-11-30 09:53:26 +01:00
|
|
|
# if LV_VDB_ADR == 0
|
2018-06-19 09:49:58 +02:00
|
|
|
/*If the buffer address is not specified simply allocate it*/
|
2019-02-12 05:39:44 +01:00
|
|
|
static LV_ATTRIBUTE_MEM_ALIGN uint8_t vdb_buf1[LV_VDB_SIZE_IN_BYTES];
|
|
|
|
static LV_ATTRIBUTE_MEM_ALIGN uint8_t vdb_buf2[LV_VDB_SIZE_IN_BYTES];
|
2018-08-04 01:46:00 +02:00
|
|
|
static lv_vdb_t vdb[2] = {{.buf = (lv_color_t *) vdb_buf1}, {.buf = (lv_color_t *) vdb_buf2}};
|
2018-10-05 17:22:49 +02:00
|
|
|
# else /*LV_VDB_ADR != 0*/
|
2018-06-19 09:49:58 +02:00
|
|
|
/*If the buffer address is specified use that address*/
|
|
|
|
static lv_vdb_t vdb[2] = {{.buf = (lv_color_t *)LV_VDB_ADR}, {.buf = (lv_color_t *)LV_VDB2_ADR}};
|
2017-11-30 09:53:26 +01:00
|
|
|
# endif
|
2017-08-04 19:09:29 +02:00
|
|
|
#endif
|
2016-06-08 07:25:08 +02:00
|
|
|
|
2018-12-22 19:08:03 +01:00
|
|
|
static volatile bool vdb_flushing = false;
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* MACROS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* GLOBAL FUNCTIONS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**
|
2017-08-04 19:09:29 +02:00
|
|
|
* Get the 'vdb' variable or allocate one in LV_VDB_DOUBLE mode
|
2017-11-29 10:46:59 +01:00
|
|
|
* @return pointer to a 'vdb' variable
|
2016-06-08 07:25:08 +02:00
|
|
|
*/
|
|
|
|
lv_vdb_t * lv_vdb_get(void)
|
|
|
|
{
|
2017-08-04 19:09:29 +02:00
|
|
|
#if LV_VDB_DOUBLE == 0
|
2018-12-22 19:08:03 +01:00
|
|
|
/* Wait until VDB is flushing.
|
|
|
|
* (Until this user calls of 'lv_flush_ready()' in the display drivers's flush function*/
|
|
|
|
while(vdb_flushing);
|
2018-09-12 09:03:48 +02:00
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
return &vdb;
|
2017-08-04 19:09:29 +02:00
|
|
|
#else
|
|
|
|
/*If already there is an active do nothing*/
|
2018-12-22 19:08:03 +01:00
|
|
|
return &vdb[vdb_active];
|
2017-08-04 19:09:29 +02:00
|
|
|
#endif
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-04 19:09:29 +02:00
|
|
|
* Flush the content of the VDB
|
2016-06-08 07:25:08 +02:00
|
|
|
*/
|
|
|
|
void lv_vdb_flush(void)
|
|
|
|
{
|
2017-08-04 19:09:29 +02:00
|
|
|
lv_vdb_t * vdb_act = lv_vdb_get();
|
2018-09-12 09:03:48 +02:00
|
|
|
if(!vdb_act) {
|
2018-10-05 17:22:49 +02:00
|
|
|
LV_LOG_WARN("Invalid VDB pointer");
|
|
|
|
return;
|
|
|
|
}
|
2017-08-04 19:09:29 +02:00
|
|
|
|
2018-12-22 19:08:03 +01:00
|
|
|
/*Don't start a new flush while the previous is not finished*/
|
|
|
|
#if LV_VDB_DOUBLE
|
|
|
|
while(vdb_flushing);
|
|
|
|
#endif /*LV_VDB_DOUBLE*/
|
|
|
|
|
|
|
|
vdb_flushing = true;
|
2017-08-04 19:09:29 +02:00
|
|
|
|
2018-02-15 10:12:28 +01:00
|
|
|
/*Flush the rendered content to the display*/
|
2018-06-19 09:49:58 +02:00
|
|
|
lv_disp_flush(vdb_act->area.x1, vdb_act->area.y1, vdb_act->area.x2, vdb_act->area.y2, vdb_act->buf);
|
2018-02-15 10:12:28 +01:00
|
|
|
|
2018-12-29 09:12:32 +01:00
|
|
|
|
2018-12-22 19:08:03 +01:00
|
|
|
#if LV_VDB_DOUBLE
|
|
|
|
/*Make the other VDB active. The content of the current will be kept until the next flush*/
|
|
|
|
vdb_active++;
|
|
|
|
vdb_active &= 0x1;
|
2017-08-04 19:09:29 +02:00
|
|
|
|
2018-12-22 19:08:03 +01:00
|
|
|
/*If the screen is transparent initialize it when the new VDB is selected*/
|
|
|
|
# if LV_COLOR_SCREEN_TRANSP
|
|
|
|
memset(vdb[vdb_active].buf, 0x00, LV_VDB_SIZE_IN_BYTES);
|
|
|
|
# endif /*LV_COLOR_SCREEN_TRANSP*/
|
|
|
|
|
|
|
|
#endif /*#if LV_VDB_DOUBLE*/
|
|
|
|
|
|
|
|
}
|
2018-09-12 09:03:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the address of VDB buffer(s) manually. To use this set `LV_VDB_ADR` (and `LV_VDB2_ADR`) to `LV_VDB_ADR_INV` in `lv_conf.h`.
|
2018-09-14 07:33:22 +02:00
|
|
|
* It should be called before `lv_init()`. The size of the buffer should be: `LV_VDB_SIZE_IN_BYTES`
|
2018-09-12 09:03:48 +02:00
|
|
|
* @param buf1 address of the VDB.
|
|
|
|
* @param buf2 address of the second buffer. `NULL` if `LV_VDB_DOUBLE 0`
|
|
|
|
*/
|
|
|
|
void lv_vdb_set_adr(void * buf1, void * buf2)
|
|
|
|
{
|
|
|
|
#if LV_VDB_DOUBLE == 0
|
2018-09-25 11:16:37 +02:00
|
|
|
(void) buf2; /*unused*/
|
2018-09-12 09:03:48 +02:00
|
|
|
vdb.buf = buf1;
|
|
|
|
#else
|
|
|
|
vdb[0].buf = buf1;
|
|
|
|
vdb[1].buf = buf2;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-08-04 19:09:29 +02:00
|
|
|
/**
|
2017-11-29 10:46:59 +01:00
|
|
|
* Call in the display driver's 'disp_flush' function when the flushing is finished
|
2017-08-04 19:09:29 +02:00
|
|
|
*/
|
2018-11-22 19:39:01 +02:00
|
|
|
LV_ATTRIBUTE_FLUSH_READY void lv_flush_ready(void)
|
2017-08-04 19:09:29 +02:00
|
|
|
{
|
2018-12-22 19:08:03 +01:00
|
|
|
vdb_flushing = false;
|
2018-09-20 22:16:03 +02:00
|
|
|
|
2018-12-22 19:08:03 +01:00
|
|
|
/*If the screen is transparent initialize it when the flushing is ready*/
|
|
|
|
#if LV_VDB_DOUBLE == 0 && LV_COLOR_SCREEN_TRANSP
|
2018-09-20 22:16:03 +02:00
|
|
|
memset(vdb_buf, 0x00, LV_VDB_SIZE_IN_BYTES);
|
|
|
|
#endif
|
2016-06-08 07:25:08 +02:00
|
|
|
}
|
|
|
|
|
2019-01-04 08:33:19 +01:00
|
|
|
/**
|
|
|
|
* Get currently active VDB, where the drawing happens. Used with `LV_VDB_DOUBLE 1`
|
|
|
|
* @return pointer to the active VDB. If `LV_VDB_DOUBLE 0` give the single VDB
|
|
|
|
*/
|
|
|
|
lv_vdb_t * lv_vdb_get_active(void)
|
|
|
|
{
|
2019-01-04 20:47:36 -08:00
|
|
|
#if LV_VDB_DOUBLE == 0
|
|
|
|
return &vdb;
|
|
|
|
#else
|
2019-01-04 08:33:19 +01:00
|
|
|
return &vdb[vdb_active];
|
2019-01-04 20:47:36 -08:00
|
|
|
#endif
|
2019-01-04 08:33:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get currently inactive VDB, which is being displayed or being flushed. Used with `LV_VDB_DOUBLE 1`
|
|
|
|
* @return pointer to the inactive VDB. If `LV_VDB_DOUBLE 0` give the single VDB
|
|
|
|
*/
|
|
|
|
lv_vdb_t * lv_vdb_get_inactive(void)
|
|
|
|
{
|
2019-01-04 20:47:36 -08:00
|
|
|
#if LV_VDB_DOUBLE == 0
|
|
|
|
return &vdb;
|
|
|
|
#else
|
2019-01-04 08:33:19 +01:00
|
|
|
return &vdb[(vdb_active + 1) & 0x1];
|
2019-01-04 20:47:36 -08:00
|
|
|
#endif
|
2019-01-04 08:33:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the flushing is in progress or not
|
|
|
|
* @return true: flushing is in progress; false: flushing ready
|
|
|
|
*/
|
|
|
|
bool lv_vdb_is_flushing(void)
|
|
|
|
{
|
|
|
|
return vdb_flushing;
|
|
|
|
}
|
|
|
|
|
2016-06-08 07:25:08 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC FUNCTIONS
|
|
|
|
**********************/
|
|
|
|
|
2017-12-02 20:43:50 +01:00
|
|
|
#else
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Just for compatibility
|
|
|
|
*/
|
|
|
|
void lv_flush_ready(void)
|
|
|
|
{
|
|
|
|
/*Do nothing. It is used only for VDB*/
|
|
|
|
}
|
2016-06-08 07:25:08 +02:00
|
|
|
#endif
|