当前位置: 首页 > news >正文

RT-Thread 在SD卡实现ulog+时间戳保存不同日志方法

1.问题背景:

在项目开发和维护中,总是会遇到一些问题:

为什么我的代码烧录进去没反应?

遇到这个问题,我们可以通过打印一下串口信息来快速定位问题。

但如果是产品出售阶段,我的代码怎么跑着跑着就死机了?

你不可能让串口一直连着控制台吧?

或者我们知道了某一个模块有问题,我们需要找那个模块相对应的代码。

你指望从串口猛猛输出的众多log信息中得到想要的模块数据吗?

因此我们需要一个保存log的文件,并且需要根据不同的日志分类保存到不同的log文件中。

以前RT-Thread的开发者们根据这些问题设计了一个软件包——ulog

后面发现这个ulog使用的效果很不错,就把它吸收到内核中了。

2.ulog简介

可以直接查看RT-Thread手册对ulog的描述ulog 日志

我们可以打开rt-thread\components\utilities\ulog\backend\console_be.c文件,就能看到以下代码

/** Copyright (c) 2006-2022, RT-Thread Development Team** SPDX-License-Identifier: Apache-2.0** Change Logs:* Date           Author       Notes* 2018-09-04     armink       the first version*/#include <rthw.h>
#include <ulog.h>#ifdef ULOG_BACKEND_USING_CONSOLE#if defined(ULOG_ASYNC_OUTPUT_BY_THREAD) && ULOG_ASYNC_OUTPUT_THREAD_STACK < 384
#error "The thread stack size must more than 384 when using async output by thread (ULOG_ASYNC_OUTPUT_BY_THREAD)"
#endifstatic struct ulog_backend console = { 0 };void ulog_console_backend_output(struct ulog_backend *backend, rt_uint32_t level, const char *tag, rt_bool_t is_raw,const char *log, rt_size_t len)
{
#ifdef RT_USING_DEVICErt_device_t dev = rt_console_get_device();if (dev == RT_NULL){rt_hw_console_output(log);}else{rt_uint16_t old_flag = dev->open_flag;dev->open_flag |= RT_DEVICE_FLAG_STREAM;rt_device_write(dev, 0, log, len);dev->open_flag = old_flag;}
#elsert_hw_console_output(log);
#endif}int ulog_console_backend_init(void)
{ulog_init();console.output = ulog_console_backend_output;ulog_backend_register(&console, "console", RT_TRUE);return 0;
}
INIT_PREV_EXPORT(ulog_console_backend_init);#endif /* ULOG_BACKEND_USING_CONSOLE */

我们的LOG为什么可以上传到控制台msh?就是因为项目生成的时候就帮我们注册了一个ulog后端。

这个后端是最简单的一种

因为它不需要过滤各种信息,只需要将各类log像rt_kprintf一样打印出来就可以了。

并且它还不需要保存成文件供我们查阅。

因此如果我们需要实现类似的log输出功能,我们不仅需要和console一样注册一个后端,还需要初始化文件进行log的保存。

然后我们的文件肯定是有最大值的,超过了最大值之后ulog就会新开一个log文件进行保存,我们可以打开rt-thread\components\utilities\ulog\backend\file_be.c查看具体的转文件保存的函数

/* rotate the log file xxx_n-1.log => xxx_n.log, and xxx.log => xxx_0.log */
static rt_bool_t ulog_file_rotate(struct ulog_file_be *be)
{
#define SUFFIX_LEN          10 //保存文件的最大值/* mv xxx_n-1.log => xxx_n.log, and xxx.log => xxx_0.log */static char old_path[ULOG_FILE_PATH_LEN], new_path[ULOG_FILE_PATH_LEN];int index = 0, err = 0, file_fd = 0;rt_bool_t result = RT_FALSE;size_t base_len = 0;rt_snprintf(old_path, ULOG_FILE_PATH_LEN, "%s/%s", be->cur_log_dir_path, be->parent.name);rt_snprintf(new_path, ULOG_FILE_PATH_LEN, "%s/%s", be->cur_log_dir_path, be->parent.name);base_len = rt_strlen(be->cur_log_dir_path) + rt_strlen(be->parent.name) + 1;if (be->cur_log_file_fd >= 0){close(be->cur_log_file_fd);}for (index = be->file_max_num - 2; index >= 0; --index){rt_snprintf(old_path + base_len, SUFFIX_LEN, index ? "_%d.log" : ".log", index - 1);rt_snprintf(new_path + base_len, SUFFIX_LEN, "_%d.log", index);/* remove the old file */if ((file_fd = open(new_path, O_RDONLY)) >= 0){close(file_fd);unlink(new_path);}/* change the new log file to old file name */if ((file_fd = open(old_path , O_RDONLY)) >= 0){close(file_fd);err = dfs_file_rename(old_path, new_path);}if (err < 0){result = RT_FALSE;goto __exit;}result = RT_TRUE;}__exit:/* reopen the file */be->cur_log_file_fd = open(be->cur_log_file_path, O_CREAT | O_RDWR | O_APPEND);return result;
}

3.具体步骤

3.1 配置RT-Thread setting

 SD卡和RTC的初始化可以看我之前的文章,这边不再赘述,有点费篇幅了

STM32H750 + RT-Thread studio实现SDMMC功能(二)功能配置——SDMMC功能实现-CSDN博客

使用cubemx和rt-thread studio联合配置(以RTC时钟为例)_rt-thread studio cubemx-CSDN博客

3.2 注册后端设备

本章我们注册两个后端设备进行比较

/** Copyright (c) 2006-2021, RT-Thread Development Team** SPDX-License-Identifier: Apache-2.0** Change Logs:* Date           Author       Notes* 2025-11-02     Administrator       the first version*/#include <rtthread.h>
#include <ulog_be.h>/*
* 后端注册表
*/
struct _log_file
{const char *name;                 //文件名ulog_backend_t backend;struct ulog_file_be *file_be;const char *dir_path;             //保存路径rt_size_t max_num;                //保存最大文件数量rt_size_t max_size;                //保存最大文件大小rt_size_t buf_size;                //文件保存缓存大小
};
/*
* 文件后端标识
*/
typedef enum
{sys_id = 0,dht_id = 1,
}ulog_file_be_name;#define ROOT_PATH "/sdcard/log" //设置保存路径
#define FILE_SIZE 512 * 1024   //设置单个文件大小
#define BUFF_SIZE 512           //设备缓存区大小static struct ulog_backend sys_log_backend;
static struct ulog_file_be sys_log_file;static struct ulog_backend dht_log_backend;
static struct ulog_file_be dht_log_file;static struct _log_file table[] =
{{"sys"      ,&sys_log_backend,&sys_log_file,ROOT_PATH,10,FILE_SIZE,BUFF_SIZE},{"dht"      ,&dht_log_backend,&dht_log_file,ROOT_PATH,5 ,FILE_SIZE,BUFF_SIZE},
};#define SYS_TAG  "main"
static rt_bool_t sys_log_file_backend_filter(struct ulog_backend *backend, rt_uint32_t level, const char *tag, rt_bool_t is_raw,const char *log, rt_size_t len){if (rt_strncmp(tag,SYS_TAG, sizeof(SYS_TAG)) == 0)//排除带有"main"标签输出return RT_TRUE;elsereturn RT_FALSE;
}#define DHT_TAG  "main"
static rt_bool_t dht_log_file_backend_filter(struct ulog_backend *backend, rt_uint32_t level, const char *tag, rt_bool_t is_raw,const char *log, rt_size_t len){if (rt_strncmp(tag,DHT_TAG, sizeof(DHT_TAG)) == 0)//排除除了"main"的标签输出return RT_FALSE;elsereturn RT_TRUE;
}/* Private function prototypes -----------------------------------------------*/
/*** @brief  系统日志文件后端初始化.* @param  None.* @retval None.* @note   None.
*/
int sys_log_file_backend_init(void)
{struct ulog_file_be *file_be = &sys_log_file;uint8_t id = sys_id;file_be->parent = sys_log_backend;//滤波函数设置ulog_backend_filter_t filter = sys_log_file_backend_filter;ulog_backend_set_filter(&file_be->parent,filter);ulog_file_backend_init( file_be,table[id].name,table[id].dir_path,table[id].max_num,table[id].max_size,table[id].buf_size);ulog_file_backend_enable(file_be); //必须使能才能有效return RT_EOK;
}
INIT_APP_EXPORT(sys_log_file_backend_init);int dht_log_file_backend_init(void)
{struct ulog_file_be *file_be = &dht_log_file;uint8_t id = dht_id;file_be->parent = dht_log_backend;//滤波函数设置ulog_backend_filter_t filter = dht_log_file_backend_filter;ulog_backend_set_filter(&file_be->parent,filter);ulog_file_backend_init( file_be,table[id].name,table[id].dir_path,table[id].max_num,table[id].max_size,table[id].buf_size);ulog_file_backend_enable(file_be);return RT_EOK;
}
INIT_APP_EXPORT(dht_log_file_backend_init);

4.功能展示

http://www.dtcms.com/a/570474.html

相关文章:

  • 移动网站与pc网站wordpress 页面文章列表
  • 山西响应式网站建设设计软件技术是干什么的
  • 网站被k怎么恢复南京那些公司做网站
  • 网站建设2种账号体系专业h5网站建设教程
  • 什么网站发布找做效果图的装饰公司排名
  • 广州网页设计网站网站建设毕业设计开题报告
  • 天津七七一网站建设有限公司怎么样昆明网站建设公司排行
  • 零基础入门C语言之深入结构体
  • 成都销售型网站北京vi设计公司价格
  • 个人网站建设方案书怎么写呼伦贝尔哪里做网站
  • 网站建设公司 销量怎么在各大网站做推广
  • CDN安全加速:高速访问与智能防护的完美结合
  • 买域名的网站有哪些有哪些比较好的外贸网站
  • seo网站的优化方案wordpress子页面内容
  • 公众号怎么做微网站网站时间轴
  • 东南亚电商新策略:智能测评驱动Shopee与Lazada增长新引擎
  • 备案需要写网站建设方案书郑州整形网站建设
  • 做网站的计划书营销网站建设技术
  • kubectl 的taint和cordon命令区别
  • 建设摩托车官网的网站首页荣耀手机商城
  • 有域名有空间如何做网站虚拟主机网站建设
  • 网站快速排名优化方法常德网
  • 集合推导式练习题
  • 怎样做支付网站wordpress搜索参数
  • 企业网站推广哪家好跨境电商平台有哪些及其特点
  • 能源业人工智能技术应用现状2025:技术落地、案例数据与未来趋势
  • 19、docker跨主机网络 Overlay\Underlay
  • 网站内容方向网站网页怎么设计
  • dw网页制作教程装修手册潍坊网站搜索引擎优化
  • 做设计必须知道的几个网站wordpress+网速