351 lines
10 KiB
C
351 lines
10 KiB
C
/**
|
|
* @file lv_draw_buf.c
|
|
*
|
|
*/
|
|
|
|
/*********************
|
|
* INCLUDES
|
|
*********************/
|
|
#include "../misc/lv_types.h"
|
|
#include "lv_draw_buf.h"
|
|
#include "../stdlib/lv_string.h"
|
|
#include "../core/lv_global.h"
|
|
|
|
/*********************
|
|
* DEFINES
|
|
*********************/
|
|
#define handlers LV_GLOBAL_DEFAULT()->draw_buf_handlers
|
|
|
|
/**********************
|
|
* TYPEDEFS
|
|
**********************/
|
|
|
|
/**********************
|
|
* STATIC PROTOTYPES
|
|
**********************/
|
|
static void * buf_malloc(size_t size, lv_color_format_t color_format);
|
|
static void buf_free(void * buf);
|
|
static void * buf_align(void * buf, lv_color_format_t color_format);
|
|
static uint32_t width_to_stride(uint32_t w, lv_color_format_t color_format);
|
|
|
|
/**********************
|
|
* STATIC VARIABLES
|
|
**********************/
|
|
|
|
/**********************
|
|
* MACROS
|
|
**********************/
|
|
|
|
/**********************
|
|
* GLOBAL FUNCTIONS
|
|
**********************/
|
|
|
|
void _lv_draw_buf_init_handlers(void)
|
|
{
|
|
lv_memzero(&handlers, sizeof(lv_draw_buf_handlers_t));
|
|
handlers.buf_malloc_cb = buf_malloc;
|
|
handlers.buf_free_cb = buf_free;
|
|
handlers.align_pointer_cb = buf_align;
|
|
handlers.invalidate_cache_cb = NULL;
|
|
handlers.width_to_stride_cb = width_to_stride;
|
|
}
|
|
|
|
lv_draw_buf_handlers_t * lv_draw_buf_get_handlers(void)
|
|
{
|
|
return &handlers;
|
|
}
|
|
|
|
uint32_t lv_draw_buf_width_to_stride(uint32_t w, lv_color_format_t color_format)
|
|
{
|
|
if(handlers.width_to_stride_cb) return handlers.width_to_stride_cb(w, color_format);
|
|
else return 0;
|
|
}
|
|
|
|
void * lv_draw_buf_malloc(size_t size_bytes, lv_color_format_t color_format)
|
|
{
|
|
if(handlers.buf_malloc_cb) return handlers.buf_malloc_cb(size_bytes, color_format);
|
|
else return NULL;
|
|
}
|
|
|
|
void lv_draw_buf_free(void * buf)
|
|
{
|
|
if(handlers.buf_free_cb) handlers.buf_free_cb(buf);
|
|
}
|
|
|
|
void * lv_draw_buf_align(void * data, lv_color_format_t color_format)
|
|
{
|
|
if(handlers.align_pointer_cb) return handlers.align_pointer_cb(data, color_format);
|
|
else return NULL;
|
|
}
|
|
|
|
void lv_draw_buf_invalidate_cache(void * buf, uint32_t stride, lv_color_format_t color_format, const lv_area_t * area)
|
|
{
|
|
if(handlers.invalidate_cache_cb) handlers.invalidate_cache_cb(buf, stride, color_format, area);
|
|
}
|
|
|
|
void lv_draw_buf_clear(void * buf, uint32_t w, uint32_t h, lv_color_format_t color_format, const lv_area_t * a)
|
|
{
|
|
LV_UNUSED(h);
|
|
|
|
uint8_t px_size = lv_color_format_get_size(color_format);
|
|
uint32_t stride = lv_draw_buf_width_to_stride(w, color_format);
|
|
uint8_t * bufc = buf;
|
|
|
|
/*Got the first pixel of each buffer*/
|
|
bufc += stride * a->y1;
|
|
bufc += a->x1 * px_size;
|
|
|
|
uint32_t line_length = lv_area_get_width(a) * px_size;
|
|
int32_t y;
|
|
for(y = a->y1; y <= a->y2; y++) {
|
|
lv_memzero(bufc, line_length);
|
|
bufc += stride;
|
|
}
|
|
|
|
}
|
|
|
|
void lv_draw_buf_copy(void * dest_buf, uint32_t dest_w, uint32_t dest_h, const lv_area_t * dest_area_to_copy,
|
|
void * src_buf, uint32_t src_w, uint32_t src_h, const lv_area_t * src_area_to_copy,
|
|
lv_color_format_t color_format)
|
|
{
|
|
LV_UNUSED(dest_h);
|
|
LV_UNUSED(src_h);
|
|
|
|
uint8_t px_size = lv_color_format_get_size(color_format);
|
|
uint8_t * dest_bufc = dest_buf;
|
|
uint8_t * src_bufc = src_buf;
|
|
|
|
uint32_t dest_stride = lv_draw_buf_width_to_stride(dest_w, color_format);
|
|
uint32_t src_stride = lv_draw_buf_width_to_stride(src_w, color_format);
|
|
|
|
/*Got the first pixel of each buffer*/
|
|
dest_bufc += dest_stride * dest_area_to_copy->y1;
|
|
dest_bufc += dest_area_to_copy->x1 * px_size;
|
|
|
|
src_bufc += src_stride * src_area_to_copy->y1;
|
|
src_bufc += src_area_to_copy->x1 * px_size;
|
|
|
|
uint32_t line_length = lv_area_get_width(dest_area_to_copy) * px_size;
|
|
int32_t y;
|
|
for(y = dest_area_to_copy->y1; y <= dest_area_to_copy->y2; y++) {
|
|
lv_memcpy(dest_bufc, src_bufc, line_length);
|
|
dest_bufc += dest_stride;
|
|
src_bufc += src_stride;
|
|
}
|
|
}
|
|
|
|
lv_draw_buf_t * lv_draw_buf_create(uint32_t w, uint32_t h, lv_color_format_t cf, uint32_t stride)
|
|
{
|
|
uint32_t size;
|
|
lv_draw_buf_t * draw_buf = lv_malloc_zeroed(sizeof(lv_draw_buf_t));
|
|
LV_ASSERT_MALLOC(draw_buf);
|
|
if(draw_buf == NULL) return NULL;
|
|
if(stride == 0) stride = lv_draw_buf_width_to_stride(w, cf);
|
|
|
|
size = stride * h;
|
|
if(cf == LV_COLOR_FORMAT_RGB565A8) {
|
|
size += (stride / 2) * h; /*A8 mask*/
|
|
}
|
|
else if(LV_COLOR_FORMAT_IS_INDEXED(cf)) {
|
|
/*@todo we have to include palette right before image data*/
|
|
size += LV_COLOR_INDEXED_PALETTE_SIZE(cf) * 4;
|
|
}
|
|
|
|
/*RLE decompression operates on pixel unit, thus add padding to make sure memory is enough*/
|
|
uint8_t bpp = lv_color_format_get_bpp(cf);
|
|
bpp = (bpp + 7) >> 3;
|
|
size += bpp;
|
|
|
|
void * buf = lv_draw_buf_malloc(size, cf);
|
|
LV_ASSERT_MALLOC(buf);
|
|
if(buf == NULL) {
|
|
lv_free(draw_buf);
|
|
return NULL;
|
|
}
|
|
|
|
draw_buf->header.w = w;
|
|
draw_buf->header.h = h;
|
|
draw_buf->header.cf = cf;
|
|
draw_buf->header.flags = LV_IMAGE_FLAGS_MODIFIABLE | LV_IMAGE_FLAGS_ALLOCATED;
|
|
draw_buf->header.stride = stride;
|
|
draw_buf->data = lv_draw_buf_align(buf, cf);
|
|
draw_buf->unaligned_data = buf;
|
|
draw_buf->data_size = size;
|
|
return draw_buf;
|
|
}
|
|
|
|
lv_draw_buf_t * lv_draw_buf_dup(const lv_draw_buf_t * draw_buf)
|
|
{
|
|
const lv_image_header_t * header = &draw_buf->header;
|
|
lv_draw_buf_t * new_buf = lv_draw_buf_create(header->w, header->h, header->cf, header->stride);
|
|
if(new_buf == NULL) return NULL;
|
|
|
|
/*Choose the smaller size to copy*/
|
|
uint32_t size = LV_MIN(draw_buf->data_size, new_buf->data_size);
|
|
|
|
/*Copy image data*/
|
|
lv_memcpy(new_buf->data, draw_buf->data, size);
|
|
return new_buf;
|
|
}
|
|
|
|
void lv_draw_buf_destroy(lv_draw_buf_t * buf)
|
|
{
|
|
LV_ASSERT_NULL(buf);
|
|
if(buf == NULL) return;
|
|
|
|
if(buf->header.flags & LV_IMAGE_FLAGS_ALLOCATED) {
|
|
lv_draw_buf_free(buf->unaligned_data);
|
|
lv_free(buf);
|
|
}
|
|
else {
|
|
LV_LOG_ERROR("draw buffer is not allocated, ignored");
|
|
}
|
|
}
|
|
|
|
void * lv_draw_buf_goto_xy(lv_draw_buf_t * buf, uint32_t x, uint32_t y)
|
|
{
|
|
LV_ASSERT_NULL(buf);
|
|
if(buf == NULL) return NULL;
|
|
|
|
uint8_t * data = buf->data;
|
|
data += buf->header.stride * y;
|
|
|
|
if(x == 0)
|
|
return data;
|
|
|
|
return data + x * lv_color_format_get_size(buf->header.cf);
|
|
}
|
|
|
|
lv_draw_buf_t * lv_draw_buf_adjust_stride(const lv_draw_buf_t * src, uint32_t stride)
|
|
{
|
|
LV_ASSERT_NULL(src);
|
|
LV_ASSERT_NULL(src->data);
|
|
if(src == NULL) return NULL;
|
|
if(src->data == NULL) return NULL;
|
|
|
|
/*Check if stride already match*/
|
|
if(src->header.stride == stride) return NULL;
|
|
|
|
/*Calculate the minimal stride allowed from bpp*/
|
|
uint32_t bpp = lv_color_format_get_bpp(src->header.cf);
|
|
uint32_t min_stride = (src->header.w * bpp + 7) >> 3;
|
|
if(stride < min_stride) {
|
|
LV_LOG_WARN("New stride is too small. min: %" LV_PRId32, min_stride);
|
|
return NULL;
|
|
}
|
|
|
|
lv_draw_buf_t * dst = lv_draw_buf_create(src->header.w, src->header.h, src->header.cf, stride);
|
|
if(dst == NULL) return NULL;
|
|
|
|
uint8_t * dst_data = dst->data;
|
|
const uint8_t * src_data = src->data;
|
|
for(int32_t y = 0; y < src->header.h; y++) {
|
|
lv_memcpy(dst_data, src_data, min_stride);
|
|
src_data += src->header.stride;
|
|
dst_data += dst->header.stride;
|
|
}
|
|
|
|
return dst;
|
|
}
|
|
|
|
lv_result_t lv_draw_buf_premultiply(lv_draw_buf_t * draw_buf)
|
|
{
|
|
LV_ASSERT_NULL(draw_buf);
|
|
if(draw_buf == NULL) return LV_RESULT_INVALID;
|
|
|
|
if(draw_buf->header.flags & LV_IMAGE_FLAGS_PREMULTIPLIED) return LV_RESULT_INVALID;
|
|
if((draw_buf->header.flags & LV_IMAGE_FLAGS_MODIFIABLE) == 0) {
|
|
LV_LOG_WARN("draw buf is not modifiable: 0x%04x", draw_buf->header.flags);
|
|
return LV_RESULT_INVALID;
|
|
}
|
|
|
|
/*Premultiply color with alpha, do case by case by judging color format*/
|
|
lv_color_format_t cf = draw_buf->header.cf;
|
|
if(LV_COLOR_FORMAT_IS_INDEXED(cf)) {
|
|
int size = LV_COLOR_INDEXED_PALETTE_SIZE(cf);
|
|
lv_color32_t * palette = (lv_color32_t *)draw_buf->data;
|
|
for(int i = 0; i < size; i++) {
|
|
lv_color_premultiply(&palette[i]);
|
|
}
|
|
}
|
|
else if(cf == LV_COLOR_FORMAT_ARGB8888) {
|
|
uint32_t h = draw_buf->header.h;
|
|
uint32_t w = draw_buf->header.w;
|
|
uint32_t stride = draw_buf->header.stride;
|
|
uint8_t * line = (uint8_t *)draw_buf->data;
|
|
for(uint32_t y = 0; y < h; y++) {
|
|
lv_color32_t * pixel = (lv_color32_t *)line;
|
|
for(uint32_t x = 0; x < w; x++) {
|
|
lv_color_premultiply(pixel);
|
|
pixel++;
|
|
}
|
|
line += stride;
|
|
}
|
|
}
|
|
else if(cf == LV_COLOR_FORMAT_RGB565A8) {
|
|
uint32_t h = draw_buf->header.h;
|
|
uint32_t w = draw_buf->header.w;
|
|
uint32_t stride = draw_buf->header.stride;
|
|
uint32_t alpha_stride = stride / 2;
|
|
uint8_t * line = (uint8_t *)draw_buf->data;
|
|
lv_opa_t * alpha = (lv_opa_t *)(line + stride * h);
|
|
for(uint32_t y = 0; y < h; y++) {
|
|
lv_color16_t * pixel = (lv_color16_t *)line;
|
|
for(uint32_t x = 0; x < w; x++) {
|
|
lv_color16_premultiply(pixel, alpha[x]);
|
|
pixel++;
|
|
}
|
|
line += stride;
|
|
alpha += alpha_stride;
|
|
}
|
|
}
|
|
else if(LV_COLOR_FORMAT_IS_ALPHA_ONLY(cf)) {
|
|
/*Pass*/
|
|
}
|
|
else {
|
|
LV_LOG_WARN("draw buf has no alpha, cf: %d", cf);
|
|
}
|
|
|
|
draw_buf->header.flags |= LV_IMAGE_FLAGS_PREMULTIPLIED;
|
|
|
|
return LV_RESULT_OK;
|
|
}
|
|
|
|
/**********************
|
|
* STATIC FUNCTIONS
|
|
**********************/
|
|
|
|
static void * buf_malloc(size_t size_bytes, lv_color_format_t color_format)
|
|
{
|
|
LV_UNUSED(color_format);
|
|
|
|
/*Allocate larger memory to be sure it can be aligned as needed*/
|
|
size_bytes += LV_DRAW_BUF_ALIGN - 1;
|
|
return lv_malloc(size_bytes);
|
|
}
|
|
|
|
static void buf_free(void * buf)
|
|
{
|
|
lv_free(buf);
|
|
}
|
|
|
|
static void * buf_align(void * buf, lv_color_format_t color_format)
|
|
{
|
|
LV_UNUSED(color_format);
|
|
|
|
uint8_t * buf_u8 = buf;
|
|
if(buf_u8) {
|
|
buf_u8 += LV_DRAW_BUF_ALIGN - 1;
|
|
buf_u8 = (uint8_t *)((lv_uintptr_t) buf_u8 & ~(LV_DRAW_BUF_ALIGN - 1));
|
|
}
|
|
return buf_u8;
|
|
}
|
|
|
|
static uint32_t width_to_stride(uint32_t w, lv_color_format_t color_format)
|
|
{
|
|
uint32_t width_byte;
|
|
width_byte = w * lv_color_format_get_bpp(color_format);
|
|
width_byte = (width_byte + 7) >> 3; /*Round up*/
|
|
return (width_byte + LV_DRAW_BUF_STRIDE_ALIGN - 1) & ~(LV_DRAW_BUF_STRIDE_ALIGN - 1);
|
|
}
|