【NextPilot日志移植】日志写入流程

news2025/5/13 12:21:27

📝 文件后端日志写入流程详解

当后端选择文件时,日志写入过程主要涉及 LogWriter 和 LogWriterFile 类的协作。以下是详细的日志写入过程解释及涉及的代码:

1. LogWriter 类初始化

LogWriter 类的构造函数中,如果配置的后端包含 BackendFile,则会创建一个 LogWriterFile 对象。

LogWriter::LogWriter(Backend configured_backend, size_t file_buffer_size) :
    _backend(configured_backend) {
    if (configured_backend & BackendFile) {
        _log_writer_file_for_write = _log_writer_file = new LogWriterFile(file_buffer_size);

        if (!_log_writer_file) {
            PX4_ERR("LogWriterFile allocation failed");
        }
    }
    // ... 其他代码
}

2. 启动文件日志

Logger 类中,调用 start_log_file 方法启动文件日志。该方法会获取日志文件名,设置加密参数,调用 LogWriterstart_log_file 方法启动日志,并写入日志头、版本信息、格式信息等。

void Logger::start_log_file(LogType type) {
    if (_writer.is_started(type, LogWriter::BackendFile) || (_writer.backend() & LogWriter::BackendFile) == 0) {
        return;
    }

    // ... 其他代码

    char file_name[LOG_DIR_LEN] = "";

    if (get_log_file_name(type, file_name, sizeof(file_name), type == LogType::Full)) {
        PX4_ERR("failed to get log file name");
        return;
    }

#if defined(PX4_CRYPTO)
    _writer.set_encryption_parameters((px4_crypto_algorithm_t)_param_sdlog_crypto_algorithm.get(),
                                      _param_sdlog_crypto_key.get(), _param_sdlog_crypto_exchange_key.get());
#endif

    _writer.start_log_file(type, file_name);
    _writer.select_write_backend(LogWriter::BackendFile);
    _writer.set_need_reliable_transfer(true);

    write_header(type);
    write_version(type);
    write_formats(type);

    // ... 其他代码

    _writer.set_need_reliable_transfer(false);
    _writer.unselect_write_backend();
    _writer.notify();

    // ... 其他代码
}

3. LogWriter 调用 LogWriterFile 启动日志

LogWriterstart_log_file 方法会调用 LogWriterFilestart_log 方法。

void LogWriter::start_log_file(LogType type, const char *filename) {
    if (_log_writer_file) {
        _log_writer_file->start_log(type, filename);
    }
}

4. LogWriterFile 启动日志

LogWriterFilestart_log 方法会打开文件并设置 _should_run 标志。

bool LogWriterFile::LogFileBuffer::start_log(const char *filename) {
    _fd = open(filename, O_CREAT | O_TRUNC | O_WRONLY, 0666);

    if (_fd < 0) {
        PX4_ERR("failed to open log file (%s)", filename);
        return false;
    }

    _should_run = true;
    return true;
}

5. 写入日志消息

Logger 类中,调用 write_message 方法写入日志消息。该方法会调用 LogWriterwrite_message 方法。

bool Logger::write_message(LogType type, void *ptr, size_t size) {
    _writer.lock();
    bool ret = _writer.write_message(type, ptr, size) == 0;
    _writer.unlock();
    return ret;
}

6. LogWriter 调用 LogWriterFile 写入消息

LogWriterwrite_message 方法会调用 LogWriterFilewrite_message 方法。

int LogWriter::write_message(LogType type, void *ptr, size_t size, uint64_t dropout_start) {
    int ret_file = 0;

    if (_log_writer_file_for_write) {
        ret_file = _log_writer_file_for_write->write_message(type, ptr, size, dropout_start);
    }

    // ... 其他代码

    return ret_file;
}

7. LogWriterFile 写入消息

LogWriterFilewrite_message 方法会将消息写入缓冲区。

int LogWriterFile::write_message(LogType type, void *ptr, size_t size, uint64_t dropout_start) {
    LogFileBuffer &buffer = _buffers[(int)type];

    if (buffer.available() < size) {
        return -1; // 缓冲区空间不足
    }

    buffer.write_no_check(ptr, size);
    notify(); // 通知写入线程有新数据

    return 0;
}

8. LogWriterFile 写入线程处理

LogWriterFilerun 方法是一个循环,会不断检查缓冲区是否有数据,并将数据写入文件。

void LogWriterFile::run() {
    while (!_exit_thread.load()) {
        // ... 其他代码

        while (true) {
            const hrt_abstime now = hrt_absolute_time();

            // ... 其他代码

            int i = (int)LogType::Count - 1;

            while (i >= 0) {
                void          *read_ptr;
                bool           is_part;
                LogFileBuffer &buffer    = _buffers[i];
                size_t         available = buffer.get_read_ptr(&read_ptr, &is_part);

                // ... 其他代码

                if (available >= min_available[i] || is_part || (!buffer._should_run && available > 0)) {
                    pthread_mutex_unlock(&_mtx);

                    // ... 其他代码

                    int written = buffer.write_to_file(read_ptr, available, call_fsync);

                    // ... 其他代码

                    if (written >= 0) {
                        pthread_mutex_lock(&_mtx);
                        buffer.mark_read(written);

                        // ... 其他代码
                    } else {
                        // ... 其他代码
                    }
                }

                // ... 其他代码
            }

            // ... 其他代码
        }

        // ... 其他代码
    }
}

9. LogWriterFile 写入文件

LogWriterFileLogFileBuffer 类的 write_to_file 方法会将数据写入文件。

ssize_t LogWriterFile::LogFileBuffer::write_to_file(const void *buffer, size_t size, bool call_fsync) const {
    perf_begin(_perf_write);
    ssize_t ret = ::write(_fd, buffer, size);
    perf_end(_perf_write);

    if (call_fsync) {
        fsync();
    }

    return ret;
}

总结

整个文件日志写入过程可以概括为:

  1. LogWriter 初始化时创建 LogWriterFile 对象。
  2. Logger 调用 start_log_file 方法启动文件日志。
  3. LogWriter 调用 LogWriterFilestart_log 方法打开文件。
  4. Logger 调用 write_message 方法写入日志消息。
  5. LogWriter 调用 LogWriterFilewrite_message 方法将消息写入缓冲区。
  6. LogWriterFile 的写入线程不断检查缓冲区,将数据写入文件。

通过以上步骤,实现了文件后端的日志写入功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2374649.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

二极管钳位电路——Multisim电路仿真

目录 二极管钳位电路 2.1 二极管正向钳位电路 二极管压降测试 2.1.1 二极管正向钳位电路图 2.1.2 二极管正向钳位工作原理 2.2 二极管负向钳位电路 2.2.1 二极管负向钳位电路图 2.2.2 二极管负向钳位工作原理 二极管正向反向钳位仿真电路实验结果 2.3 二极管顶部钳位…

suricata增加单元测试编译失败

一、环境 $ lsb_release -a No LSB modules are available. Distributor ID: Ubuntu Description: Ubuntu 22.04.5 LTS Release: 22.04 Codename: jammysuricata: suricata7.0.5 IDE: vscode 二、背景 在suricata中开发了某个功能后&#xff0c;增加unittest时&#xff0c;…

高并发场景下的BI架构设计:衡石分布式查询引擎与缓存分级策略

在电商大促、金融交易时段或IoT实时监控场景中&#xff0c;企业BI系统常面临瞬时万级并发查询的冲击——运营团队需要实时追踪GMV波动&#xff0c;风控部门需秒级响应欺诈检测&#xff0c;产线监控需毫秒级反馈设备状态。传统单体架构的BI系统在此类场景下极易崩溃&#xff0c;…

鱼眼摄像头(一)多平面格式 单缓冲读取图像并显示

鱼眼摄像头&#xff08;一&#xff09;多平面格式 单缓冲读取图像并显示 1.摄像头格式 1. 单平面格式&#xff08;Single Plane&#xff09;&#xff1a;各通道数据保存在同一个平面&#xff08;缓冲&#xff09;&#xff0c;图像数据按行连续存储a. mjpeg&#xff0c;yuyv等…

机器学习笔记——特征工程

大家好&#xff0c;这里是好评笔记&#xff0c;公主号&#xff1a;Goodnote&#xff0c;专栏文章私信限时Free。本笔记介绍机器学习中常见的特征工程方法、正则化方法和简要介绍强化学习。 文章目录 特征工程&#xff08;Fzeature Engineering&#xff09;1. 特征提取&#xff…

A Survey of Learning from Rewards:从训练到应用的全面剖析

A Survey of Learning from Rewards&#xff1a;从训练到应用的全面剖析 你知道大语言模型&#xff08;LLMs&#xff09;如何通过奖励学习变得更智能吗&#xff1f;这篇论文将带你深入探索。从克服预训练局限的新范式&#xff0c;到训练、推理各阶段的策略&#xff0c;再到广泛…

Python爬虫第20节-使用 Selenium 爬取小米商城空调商品

目录 前言 一、 本文目标 二、环境准备 2.1 安装依赖 2.2 配置 ChromeDriver 三、小米商城页面结构分析 3.1 商品列表结构 3.2 分页结构 四、Selenium 自动化爬虫实现 4.1 脚本整体结构 4.2 代码实现 五、关键技术详解 5.1 Selenium 启动与配置 5.2 页面等待与异…

Aware和InitializingBean接口以及@Autowired注解失效分析

Aware 接口用于注入一些与容器相关信息&#xff0c;例如&#xff1a; ​ a. BeanNameAware 注入 Bean 的名字 ​ b. BeanFactoryAware 注入 BeanFactory 容器 ​ c. ApplicationContextAware 注入 ApplicationContext 容器 ​ d. EmbeddedValueResolverAware 注入 解析器&a…

Unity3D仿星露谷物语开发41之创建池管理器

1、目标 在PersistentScene中创建池管理器&#xff08;Pool Manager&#xff09;。这将允许一个预制对象池被创建和重用。 在游戏中当鼠标点击地面时&#xff0c;便会启用某一个对象。比如点击地面&#xff0c;就创建了一棵树&#xff0c;而这棵树是从预制体对象池中获取的&a…

Modbus协议介绍

Modbus是一种串行通信协议&#xff0c;由Modicon公司&#xff08;现为施耐德电气&#xff09;在1979年为可编程逻辑控制器&#xff08;PLC&#xff09;通信而开发。它是工业自动化领域最常用的通信协议之一&#xff0c;具有开放性、简单性和跨平台兼容性&#xff0c;广泛应用于…

I/O多路复用(select/poll/epoll)

通过一个进程来维护多个Socket&#xff0c;也就是I/O多路复用&#xff0c;是一种常见的并发编程技术&#xff0c;它允许单个线程或进程同时监视多个输入/输出&#xff08;I/O&#xff09;流&#xff08;例如网络连接、文件描述符&#xff09;。当任何一个I/O流准备好进行读写操…

Westlake-Omni 情感端音频生成式输出模型

简述 github地址在 GitHub - xinchen-ai/Westlake-OmniContribute to xinchen-ai/Westlake-Omni development by creating an account on GitHub.https://github.com/xinchen-ai/Westlake-Omni Westlake-Omni 是由西湖心辰&#xff08;xinchen-ai&#xff09;开发的一个开源…

随手记录5

一些顶级思维&#xff1a; ​ 顶级思维 1、永远不要自卑。 也永远不要感觉自己比别人差&#xff0c;这个人有没有钱&#xff0c;有多少钱&#xff0c;其实跟你都没有关系。有很多人就是那个奴性太强&#xff0c;看到比自己优秀的人&#xff0c;甚至一些装逼的人&#xff0c;这…

Linux驱动:驱动编译流程了解

要求 1、开发板中的linux的zImage必须是自己编译的 2、内核源码树,其实就是一个经过了配置编译之后的内核源码。 3、nfs挂载的rootfs,主机ubuntu中必须搭建一个nfs服务器。 内核源码树 解压 tar -jxvf x210kernel.tar.bz2 编译 make x210ii_qt_defconfigmakeCan’t use ‘…

使用 Flowise 构建基于私有知识库的智能客服 Agent(图文教程)

使用 Flowise 构建基于私有知识库的智能客服 Agent(图文教程) 在构建 AI 客服时,常见的需求是让机器人基于企业自身的知识文档,提供准确可靠的答案。本文将手把手教你如何使用 Flowise + 向量数据库(如 Pinecone),构建一个结合 RAG(Retrieval-Augmented Generation)检…

RabbitMQ ③-Spring使用RabbitMQ

Spring使用RabbitMQ 创建 Spring 项目后&#xff0c;引入依赖&#xff1a; <!-- https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-amqp --> <dependency><groupId>org.springframework.boot</groupId><artifac…

linux中常用的命令(四)

目录 1-cat查看文件内容 2-more命令 3-less命令 4-head命令 5-tail命令 1-cat查看文件内容 cat中的一些操作 -b : 列出行号&#xff08;不含空白行&#xff09;-E : 将结尾的断行以 $ 的形式展示出来-n : 列出行号&#xff08;含空白行&#xff09;-T : 将 tab 键 以 ^I 显示…

利用SSRF击穿内网!kali靶机实验

目录 1. 靶场拓扑图 2. 判断SSRF的存在 3. SSRF获取本地信息 3.1. SSRF常用协议 3.2. 使用file协议 4. 172.150.23.1/24探测端口 5. 172.150.23.22 - 代码注入 6. 172.150.23.23 SQL注入 7. 172.150.23.24 命令执行 7.1. 实验步骤 8. 172.150.23.27:6379 Redis未授权…

DVWA在线靶场-xss部分

目录 1. xxs&#xff08;dom&#xff09; 1.1 low 1.2 medium 1.3 high 1.4 impossible 2. xss&#xff08;reflected&#xff09; 反射型 2.1 low 2.2 medium 2.3 high 2.4 impossible 3. xss&#xff08;stored&#xff09;存储型 --留言板 3.1 low 3.2 medium 3.3 high 3.…

Go 语言 slice(切片) 的使用

序言 在许多开发语言中&#xff0c;动态数组是必不可少的一个组成部分。在实际的开发中很少会使用到数组&#xff0c;因为对于数组的大小大多数情况下我们是不能事先就确定好的&#xff0c;所以他不够灵活。动态数组通过提供自动扩容的机制&#xff0c;极大地提升了开发效率。这…