mirror of
https://github.com/lvgl/lvgl.git
synced 2025-01-21 06:53:01 +08:00
213 lines
4.6 KiB
C
213 lines
4.6 KiB
C
/**
|
|
* @file lv_debug.c
|
|
*
|
|
*/
|
|
|
|
/*********************
|
|
* INCLUDES
|
|
*********************/
|
|
#include "lv_obj.h"
|
|
#include "lv_debug.h"
|
|
|
|
#if LV_USE_DEBUG
|
|
|
|
/*********************
|
|
* DEFINES
|
|
*********************/
|
|
#ifndef LV_DEBUG_STR_MAX_LENGTH
|
|
#define LV_DEBUG_STR_MAX_LENGTH (1024 * 8)
|
|
#endif
|
|
|
|
#ifndef LV_DEBUG_STR_MAX_REPEAT
|
|
#define LV_DEBUG_STR_MAX_REPEAT 8
|
|
#endif
|
|
/**********************
|
|
* TYPEDEFS
|
|
**********************/
|
|
|
|
/**********************
|
|
* STATIC PROTOTYPES
|
|
**********************/
|
|
static bool obj_valid_child(const lv_obj_t * parent, const lv_obj_t * obj_to_find);
|
|
|
|
/**********************
|
|
* STATIC VARIABLES
|
|
**********************/
|
|
|
|
/**********************
|
|
* MACROS
|
|
**********************/
|
|
|
|
/**********************
|
|
* GLOBAL FUNCTIONS
|
|
**********************/
|
|
|
|
bool lv_debug_check_null(const void * p)
|
|
{
|
|
if(p) return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
bool lv_debug_check_mem_integrity(void)
|
|
{
|
|
return lv_mem_test() == LV_RES_OK ? true : false;
|
|
}
|
|
|
|
bool lv_debug_check_obj_type(const lv_obj_t * obj, const char * obj_type)
|
|
{
|
|
if(obj_type[0] == '\0') return true;
|
|
|
|
lv_obj_type_t types;
|
|
lv_obj_get_type((lv_obj_t *)obj, &types);
|
|
|
|
uint8_t i;
|
|
for(i = 0; i < LV_MAX_ANCESTOR_NUM; i++) {
|
|
if(strcmp(types.type[i], obj_type) == 0) return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool lv_debug_check_obj_valid(const lv_obj_t * obj)
|
|
{
|
|
lv_disp_t * disp = lv_disp_get_next(NULL);
|
|
while(disp) {
|
|
lv_obj_t * scr;
|
|
LV_LL_READ(disp->scr_ll, scr) {
|
|
|
|
if(scr == obj) return true;
|
|
bool found = obj_valid_child(scr, obj);
|
|
if(found) return true;
|
|
}
|
|
|
|
disp = lv_disp_get_next(disp);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool lv_debug_check_style(const lv_style_t * style)
|
|
{
|
|
if(style == NULL) return true; /*NULL style is still valid*/
|
|
|
|
#if LV_USE_ASSERT_STYLE
|
|
if(style->sentinel != LV_DEBUG_STYLE_SENTINEL_VALUE) {
|
|
LV_LOG_WARN("Invalid style (local variable or not initialized?)");
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
bool lv_debug_check_style_list(const lv_style_list_t * list)
|
|
{
|
|
if(list == NULL) return true; /*NULL list is still valid*/
|
|
|
|
#if LV_USE_ASSERT_STYLE
|
|
if(list->sentinel != LV_DEBUG_STYLE_LIST_SENTINEL_VALUE) {
|
|
LV_LOG_WARN("Invalid style (local variable or not initialized?)");
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
return true;
|
|
}
|
|
|
|
bool lv_debug_check_str(const void * str)
|
|
{
|
|
const uint8_t * s = (const uint8_t *)str;
|
|
uint8_t last_byte = 0;
|
|
uint32_t rep = 0;
|
|
uint32_t i;
|
|
|
|
for(i = 0; s[i] != '\0' && i < LV_DEBUG_STR_MAX_LENGTH; i++) {
|
|
if(s[i] != last_byte) {
|
|
last_byte = s[i];
|
|
rep = 1;
|
|
} else if(s[i] > 0x7F){
|
|
rep++;
|
|
if(rep > LV_DEBUG_STR_MAX_REPEAT) {
|
|
LV_LOG_WARN("lv_debug_check_str: a non-ASCII char has repeated more than LV_DEBUG_STR_MAX_REPEAT times)");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if(s[i] < 10) {
|
|
LV_LOG_WARN("lv_debug_check_str: invalid char in the string (< 10 value)");
|
|
return false; /*Shouldn't occur in strings*/
|
|
}
|
|
}
|
|
|
|
if(s[i] == '\0') return true;
|
|
|
|
LV_LOG_WARN("lv_debug_check_str: string is longer than LV_DEBUG_STR_MAX_LENGTH");
|
|
return false;
|
|
}
|
|
|
|
void lv_debug_log_error(const char * msg, uint64_t value)
|
|
{
|
|
static const char hex[] = "0123456789ABCDEF";
|
|
|
|
size_t msg_len = strlen(msg);
|
|
uint32_t value_len = sizeof(unsigned long int);
|
|
|
|
if(msg_len < 230) {
|
|
char buf[255];
|
|
char * bufp = buf;
|
|
|
|
/*Add the function name*/
|
|
memcpy(bufp, msg, msg_len);
|
|
bufp += msg_len;
|
|
|
|
/*Add value in hey*/
|
|
*bufp = ' ';
|
|
bufp ++;
|
|
*bufp = '(';
|
|
bufp ++;
|
|
*bufp = '0';
|
|
bufp ++;
|
|
*bufp = 'x';
|
|
bufp ++;
|
|
|
|
int8_t i;
|
|
for(i = value_len * 2 - 1; i >= 0; i--) {
|
|
uint8_t x = (unsigned long int)((unsigned long int)value >> (i * 4)) & 0xF;
|
|
|
|
*bufp = hex[x];
|
|
bufp++;
|
|
}
|
|
|
|
*bufp = ')';
|
|
bufp ++;
|
|
|
|
*bufp = '\0';
|
|
LV_LOG_ERROR(buf);
|
|
} else {
|
|
LV_LOG_ERROR(msg);
|
|
}
|
|
}
|
|
|
|
/**********************
|
|
* STATIC FUNCTIONS
|
|
**********************/
|
|
|
|
static bool obj_valid_child(const lv_obj_t * parent, const lv_obj_t * obj_to_find)
|
|
{
|
|
/*Check all children of `parent`*/
|
|
lv_obj_t * child;
|
|
LV_LL_READ(parent->child_ll, child) {
|
|
if(child == obj_to_find) return true;
|
|
|
|
/*Check the children*/
|
|
bool found = obj_valid_child(child, obj_to_find);
|
|
if(found) return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
#endif /*LV_USE_DEBUG*/
|
|
|