ArduPilot开源代码之AP_Logger

news2025/1/10 20:51:10

ArduPilot开源代码之AP_Logger

  • 1. 源由
  • 2. Logger类
    • 2.1 Copter初始化
    • 2.2 Copter任务
  • 3. 实例理解
    • 3.1 Copter设备应用任务
      • 3.1.1 Copter::ten_hz_logging_loop
      • 3.1.2 Copter::twentyfive_hz_logging
    • 3.2 AP_Logger应用类任务
      • 3.2.1 AP_Logger::periodic_tasks
      • 3.2.2 AP_Logger::io_thread
  • 4. H743 Dual - W25N01GV
    • 4.1 AP_Logger_W25N01GV::probe
    • 4.2 AP_Logger_W25N01GV::Init
    • 4.4 AP_Logger_Block::Init
    • 4.5 AP_Logger_Block::io_timer
  • 5. 总结
  • 6. 参考资料

1. 源由

日志对于程序来说相辅相成,历史由来已久。

鉴于传感器数据高速采集和存储器的慢速存储问题,尤其在实际日志记录过程中,重点是针对问题或者关注的数据进行记录,不太可能将所有的数据都记录下来(也没有必要)。

ArduPilot的日志模块部分,这里从整体做个介绍,以便大家对日志有更像详细的了解。

2. Logger类

  • AP_Logger
  • AP_Logger_RateLimiter
  • AP_Logger_Backend
  • AP_Logger_File
  • AP_Logger_MAVLink
  • AP_Logger_Block
  • AP_Logger_DataFlash
  • AP_Logger_W25N01GV

在这里插入图片描述

2.1 Copter初始化

应用类的初始化总是有个init函数处理,启动过程可参考:ArduPilot飞控启动&运行过程简介。

AP_Vehicle::setup
 └──> init_ardupilot
     └──> Copter::log_init
         └──> AP_Logger::Init

初始化最终要根据硬件芯片进行特定的初始化逻辑,鉴于H743 Dual BIM270这块板子使用的是W25N01GV,详见配置文件。
在这里插入图片描述

AP_Logger::Init
 ├──> _params.file_bufsize.convert_parameter_width(AP_PARAM_INT8)  //convert from 8 bit to 16 bit LOG_FILE_BUFSIZE
 ├──> <hal.util->was_watchdog_armed()>
 │   ├──> GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Forcing logging for watchdog reset")
 │   └──> _params.log_disarmed.set(LogDisarmed::LOG_WHILE_DISARMED)
 ├──> <CONFIG_HAL_BOARD == HAL_BOARD_SITL> 
 │   ├──> validate_structures(structures, num_types)
 │   └──> dump_structures(structures, num_types)
 ├──> <Backend_Type::BLOCK> AP_Logger_W25N01GV::probe
 ├──> backends[i]->Init
 ├──> start_io_thread
 └──> EnableWrites(true)

注1:如果Backend_TypeFILESYSTEM则需要使用AP_Logger_File::probe;若Backend_TypeMAVLINK则需要使用AP_Logger_MAVLink::probe

// start the update thread
void AP_Logger::start_io_thread(void)
{
    WITH_SEMAPHORE(_log_send_sem);

    if (_io_thread_started) {
        return;
    }

    if (!hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&AP_Logger::io_thread, void), "log_io", HAL_LOGGING_STACK_SIZE, AP_HAL::Scheduler::PRIORITY_IO, 1)) {
        AP_HAL::panic("Failed to start Logger IO thread");
    }

    _io_thread_started = true;
    return;
}

注2:这里start_io_thread开启了一个io线程来处理存储任务AP_Logger::io_thread,后面我们会对其做简单的介绍。

2.2 Copter任务

  • Copter::ten_hz_logging_loop
  • Copter::twentyfive_hz_logging
  • AP_Logger::periodic_tasks

这里看出,实际日志记录使用有两处与设备应用紧密结合,而常规AP_Logger应用处理的是periodic_tasks

#if LOGGING_ENABLED == ENABLED
    SCHED_TASK(ten_hz_logging_loop,   10,    350, 114),
    SCHED_TASK(twentyfive_hz_logging, 25,    110, 117),
    SCHED_TASK_CLASS(AP_Logger,            &copter.logger,              periodic_tasks, 400, 300, 120),
#endif

3. 实例理解

根据日志应用的情况展开研读和讨论,希望从应用场景进而理解日志类的继承和实现。

3.1 Copter设备应用任务

3.1.1 Copter::ten_hz_logging_loop

AP_Scheduler保证10Hz遍历,并在标志位的作用下,记录相关飞行数据。

Copter::ten_hz_logging_loop
 ├──> <should_log(MASK_LOG_ATTITUDE_MED) && !should_log(MASK_LOG_ATTITUDE_FAST) && !copter.flightmode->logs_attitude()> 
 │   └──> Log_Write_Attitude
 ├──> <!should_log(MASK_LOG_ATTITUDE_FAST) && !copter.flightmode->logs_attitude()>
 │   └──> Log_Write_PIDS
 ├──> <!should_log(MASK_LOG_ATTITUDE_FAST)>
 │   └──> Log_Write_EKF_POS
 ├──> <should_log(MASK_LOG_MOTBATT)>
 │   └──> motors->Log_Write
 ├──> <should_log(MASK_LOG_RCIN)>
 │   ├──> logger.Write_RCIN
 │   └──> <rssi.enabled()>
 │       └──> logger.Write_RSSI
 ├──> <should_log(MASK_LOG_RCOUT)>
 │   └──> logger.Write_RCOUT
 ├──> <should_log(MASK_LOG_NTUN) && (flightmode->requires_GPS() || landing_with_GPS() || !flightmode->has_manual_throttle())>
 │   └──> pos_control->write_log
 ├──> <should_log(MASK_LOG_IMU) || should_log(MASK_LOG_IMU_FAST) || should_log(MASK_LOG_IMU_RAW)>
 │   └──> AP::ins().Write_Vibration
 ├──> <should_log(MASK_LOG_CTUN)>
 │   ├──> attitude_control->control_monitor_log
 │   ├──> <HAL_PROXIMITY_ENABLED> g2.proximity.log
 │   └──> <AP_BEACON_ENABLED> g2.beacon.log
 ├──> <FRAME_CONFIG == HELI_FRAME> Log_Write_Heli
 └──> <AP_WINCH_ENABLED> <should_log(MASK_LOG_ANY)> g2.winch.write_log

3.1.2 Copter::twentyfive_hz_logging

AP_Scheduler保证25Hz遍历,并在标志位的作用下,记录相关飞行数据。

Copter::twentyfive_hz_logging
 ├──> <should_log(MASK_LOG_ATTITUDE_FAST)> 
 │   └──> Log_Write_EKF_POS
 ├──> <should_log(MASK_LOG_IMU) && !(should_log(MASK_LOG_IMU_FAST))>
 │   └──> AP::ins().Write_IMU()
 ├──> <MODE_AUTOROTATE_ENABLED == ENABLED> <should_log(MASK_LOG_ATTITUDE_MED) || should_log(MASK_LOG_ATTITUDE_FAST)>
 │   └──> g2.arot.Log_Write_Autorotation
 └──> <HAL_GYROFFT_ENABLED> <should_log(MASK_LOG_FTN_FAST)> 
     └──> gyro_fft.write_log_messages

3.2 AP_Logger应用类任务

3.2.1 AP_Logger::periodic_tasks

AP_Scheduler保证400Hz遍历,并在标志位的作用下,记录相关飞行数据。

AP_Logger::periodic_tasks
 ├──> <HAL_BUILD_AP_PERIPH> handle_log_send
 └──> FOR_EACH_BACKEND(periodic_tasks())

3.2.2 AP_Logger::io_thread

使用OS线程进行1KHz的频率对存储系统进行相关任务操作。

AP_Logger::io_thread
 └──> loop(1)
     ├──> [delay MAX(1000 - (now - last_run_us), 250)]
     ├──> FOR_EACH_BACKEND(io_timer())
     ├──> <now - last_stack_us > 100000U> hal.util->log_stack_info()
     ├──> <!done_crash_dump_save && now - last_crash_check_us > 5000000U> check_crash_dump_save
     └──> file_content_update

4. H743 Dual - W25N01GV

由于H743 Dual BIM270这块板子使用的是W25N01GV,接下来我们看下AP_Logger_W25N01GV重要的几个具体实现。

4.1 AP_Logger_W25N01GV::probe

C++对象new一个实例出来,把该设备对象创建出来,用于后续的硬件设备管理。

    static AP_Logger_Backend  *probe(AP_Logger &front,
                                     LoggerMessageWriter_DFLogStart *ls) {
        return new AP_Logger_W25N01GV(front, ls);
    }

4.2 AP_Logger_W25N01GV::Init

通过定义的SPI总线,对SPI-Flash芯片进行硬件检测和芯片初始化。

void AP_Logger_W25N01GV::Init()
{
    dev = hal.spi->get_device("dataflash");
    if (!dev) {
        AP_HAL::panic("PANIC: AP_Logger W25N01GV device not found");
        return;
    }

    dev_sem = dev->get_semaphore();

    if (!getSectorCount()) {
        flash_died = true;
        return;
    }

    flash_died = false;

    // reset the device
    WaitReady();
    {
        WITH_SEMAPHORE(dev_sem);
        uint8_t b = JEDEC_DEVICE_RESET;
        dev->transfer(&b, 1, nullptr, 0);
    }
    hal.scheduler->delay(W25N01G_TIMEOUT_RESET_MS);

    // disable write protection
    WriteStatusReg(W25N01G_PROT_REG, 0);
    // enable ECC and buffer mode
    WriteStatusReg(W25N01G_CONF_REG, W25N01G_CONFIG_ECC_ENABLE|W25N01G_CONFIG_BUFFER_READ_MODE);

    printf("W25N01GV status: SR-1=0x%x, SR-2=0x%x, SR-3=0x%x\n",
        ReadStatusRegBits(W25N01G_PROT_REG),
        ReadStatusRegBits(W25N01G_CONF_REG),
        ReadStatusRegBits(W25N01G_STATUS_REG));

    AP_Logger_Block::Init();
}

4.4 AP_Logger_Block::Init

SPI属于块设备,需要进行相应的一些块设备初始化。该函数在AP_Logger_W25N01GV::Init中调用。

void AP_Logger_Block::Init(void)
{
    // buffer is used for both reads and writes so access must always be within the semaphore
    buffer = (uint8_t *)hal.util->malloc_type(df_PageSize, AP_HAL::Util::MEM_DMA_SAFE);
    if (buffer == nullptr) {
        AP_HAL::panic("Out of DMA memory for logging");
    }

    //flash_test();

    if (CardInserted()) {
        // reserve space for version in last sector
        df_NumPages -= df_PagePerBlock;

        // determine and limit file backend buffersize
        uint32_t bufsize = _front._params.file_bufsize;
        if (bufsize > 64) {
            bufsize = 64;
        }
        bufsize *= 1024;

        // If we can't allocate the full size, try to reduce it until we can allocate it
        while (!writebuf.set_size(bufsize) && bufsize >= df_PageSize * df_PagePerBlock) {
            DEV_PRINTF("AP_Logger_Block: Couldn't set buffer size to=%u\n", (unsigned)bufsize);
            bufsize >>= 1;
        }

        if (!writebuf.get_size()) {
            DEV_PRINTF("Out of memory for logging\n");
            return;
        }

        DEV_PRINTF("AP_Logger_Block: buffer size=%u\n", (unsigned)bufsize);
        _initialised = true;
    }

    WITH_SEMAPHORE(sem);

    if (NeedErase()) {
        EraseAll();
    } else {
        validate_log_structure();
    }
}

4.5 AP_Logger_Block::io_timer

定时循环对块设备进行相应的擦除,写入操作。这里非常类似Linux驱动创建了一个内核任务。

void AP_Logger_Block::io_timer(void)
{
    uint32_t tnow = AP_HAL::millis();
    io_timer_heartbeat = tnow;

    // don't write anything for the first 2s to give the dataflash chip a chance to be ready
    if (!_initialised || tnow < 2000) {
        return;
    }

    if (erase_started) {
        WITH_SEMAPHORE(sem);

        if (InErase()) {
            return;
        }
        // write the logging format in the last page
        StartWrite(df_NumPages+1);
        uint32_t version = DF_LOGGING_FORMAT;
        memset(buffer, 0, df_PageSize);
        memcpy(buffer, &version, sizeof(version));
        FinishWrite();
        erase_started = false;
        chip_full = false;
        status_msg = StatusMessage::ERASE_COMPLETE;
        return;
    }

    if (df_EraseFrom > 0) {
        WITH_SEMAPHORE(sem);

        const uint32_t sectors = df_NumPages / df_PagePerSector;
        const uint32_t block_size = df_PagePerBlock * df_PageSize;
        const uint32_t sectors_in_block = block_size / (df_PagePerSector * df_PageSize);
        uint32_t next_sector = get_sector(df_EraseFrom);
        const uint32_t aligned_sector = sectors - (((df_NumPages - df_EraseFrom + 1) / df_PagePerSector) / sectors_in_block) * sectors_in_block;
        while (next_sector < aligned_sector) {
            Sector4kErase(next_sector);
            io_timer_heartbeat = AP_HAL::millis();
            next_sector++;
        }
        uint16_t blocks_erased = 0;
        while (next_sector < sectors) {
            blocks_erased++;
            SectorErase(next_sector / sectors_in_block);
            io_timer_heartbeat = AP_HAL::millis();
            next_sector += sectors_in_block;
        }
        status_msg = StatusMessage::RECOVERY_COMPLETE;
        df_EraseFrom = 0;
    }

    if (!CardInserted() || new_log_pending || chip_full) {
        return;
    }

    // we have been asked to stop logging, flush everything
    if (stop_log_pending) {
        WITH_SEMAPHORE(sem);

        log_write_started = false;

        // complete writing any previous log, a page at a time to avoid holding the lock for too long
        if (writebuf.available()) {
            write_log_page();
        } else {
            writebuf.clear();
            stop_log_pending = false;
        }

    // write at most one page
    } else if (writebuf.available() >= df_PageSize - sizeof(struct PageHeader)) {
        WITH_SEMAPHORE(sem);

        write_log_page();
    }
}

5. 总结

通过AP_Logger模块的代码初步研读,整个日志的应用是如何初始化?如何日志记录?如何驱动更新数据?应该都能很好的找到入口点。

通过研读,希望能够从代码框架的角度,更好的理解设计。当我们需要新增一些内容的时候,可以有的放矢。

比如:

  1. 移植新的SPI-Flash芯片驱动
  2. 新增应用日志记录
  3. 调整日志记录频率
  4. 定位日志应用的异常问题
    等等。。。 。。。

6. 参考资料

【1】ArduPilot开源飞控系统之简单介绍
【2】ArduPilot之开源代码框架
【3】ArduPilot飞控之ubuntu22.04-SITL安装
【4】ArduPilot飞控之ubuntu22.04-Gazebo模拟
【5】ArduPilot飞控之Mission Planner模拟
【6】ArduPilot飞控AOCODARC-H7DUAL固件编译
【7】ArduPilot之开源代码Library&Sketches设计
【8】ArduPilot之开源代码Sensor Drivers设计
【9】ArduPilot之开源代码基础知识&Threading概念
【10】ArduPilot之开源代码UARTs and the Console使用
【11】ArduPilot飞控启动&运行过程简介
【11】ArduPilot之开源代码Task介绍
【12】ArduPilot开源代码之AP_Param
【13】ArduPilot开源代码之AP_Scheduler
【14】ArduPilot开源代码之AP_VideoTX
【15】ArduPilot开源代码之AP_InertialSensor_Backend
【16】ArduPilot开源代码之AP_InertialSensor

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

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

相关文章

ORB_SLAM3 IMU预积分PreintegrateIMU

这篇博文主要分享ORB_SLAM3中Tracking::PreintegrateIMU()&#xff0c;其主要包括几个部分&#xff1a; 获得两帧之间的IMU数据中值积分IMU状态更新 关于IMU的理论推导参考&#xff1a; ORB_SLAM3_IMU预积分理论推导(预积分项)ORB_SLAM3_IMU预积分理论推导(噪声分析)ORB_SLA…

第五章 Linux的文件权限与目录配置

Linux最好的地方之一就在于他的多用户多任务环境。为了让各个使用者具有较保密的文件数据&#xff0c;因此文件的权限管理就变得很重要了。Linux一般将文件可存取的身份分为三个类别&#xff0c;分别是owner/group/others,且三种身份各有read/write/execute等权限。 5.1 使用者…

【Servlet学习一】认识Servlet 创建第一个Servlet项目

目录 &#x1f31f;需要知道&#xff1a; &#x1f308;1、Tomcat是什么&#xff1f; &#x1f308; 2、Maven &#x1f31f;一、认识Servlet &#x1f308;1、Servlet是什么&#xff1f; &#x1f308;2、实现第一个Servlet项目。 &#x1f308;3、简单了解Postman工具…

Python3 模块与输入与输出 | 菜鸟教程(十二)

目录 一、Python3 模块 &#xff08;一&#xff09;import 语句 1、想使用 Python 源文件&#xff0c;只需在另一个源文件里执行 import 语句&#xff0c;语法如下&#xff1a; 2、当解释器遇到 import 语句&#xff0c;如果模块在当前的搜索路径就会被导入。 3、搜索路径是…

PID控制算法 – 1、Sample Time(采样时间)

前面介绍的PID代码虽然能跑起来&#xff0c;但是还存在一些问题。 PID控制算法 – 0、PID原理_资深流水灯工程师的博客-CSDN博客 对应的代码也重新贴一下&#xff0c;方便比较 /*工作变量*/ unsigned long lastTime; double Input, Output, Setpoint; double errSum, lastEr…

【初识C语言(2)】字符串+转义字符+注释

文章目录 1. 字符串2. 转义字符转义字符表常见转义字符 3. 注释 1. 字符串 “hello world.\n” 上面这种由双引号引起的一串字符就被称为字符串&#xff1b; 字符串的存储 C 语言当中没有字符串类型&#xff0c;如果想要将字符串存储起来的话就需要用到字符串数组。 #include…

LLaMA及其子孙模型概述

文章目录 LLaMAAlpacaVicunaKoalaBaize (白泽)骆驼(Luotuo)BELLEGuanaco LLaMA 与原始transformer的区别: 预归一化[GPT3]。为了提高训练稳定性&#xff0c;对每个Transformer子层的输入进行归一化&#xff0c;而不是对输出进行归一化。使用了Zhang和Sennrich&#xff08;201…

Redis进阶篇(附面试快速答法)

文章目录 Redis使用场景1、缓存穿透布隆过滤器小总结面试快速答法 2、缓存击穿小总结面试快速答法 3、缓存雪崩面试快速答法 4、双写一致性小总结面试快速答法 5、持久化机制面试快速答法 6、数据过期策略小总结面试快速答法 7、数据淘汰策略小总结面试快速答法 8、分布式锁小总…

Pytest集成Allure Report

目录 安装 用法 基本报告 支持 Pytest features Xfail 条件标记 Fixtures and Finalizers 参数化 Allure Features Steps 附件 描述 标题 链接 重试 Tags BDD 标签 严重性标记 Behave 安装 使用 Features 严重性 步骤和场景状态 步骤数据 安装 Pytest可从…

SpringBoot 如何使用 @RequestBody 进行数据校验

SpringBoot 如何使用 RequestBody 进行数据校验 在 Web 开发中&#xff0c;前台向后台发送数据是非常常见的场景。而在 SpringBoot 框架中&#xff0c;我们通常使用 RequestBody 注解来接收前台发送的 JSON 数据&#xff0c;并将其转化为 Java 对象。但是&#xff0c;接收到的…

你一定想知道的 如何进行动态内存管理?

文章目录 引言malloc函数calloc函数realloc函数free函数-避免内存泄漏常见的动态内存错误 引言 如果我们被问道&#xff1a;如何创建一个可以根据用户需求来开辟大小的数组&#xff1f; 可能有些博友会写出如下代码&#xff1a; #include <stdio.h> int main() {int n…

c++11 标准模板(STL)(std::basic_streambuf)(二)

定义于头文件 <streambuf> template< class CharT, class Traits std::char_traits<CharT> > class basic_streambuf; 类 basic_streambuf 控制字符序列的输入与输出。它包含下列内容并提供到它们的访问&#xff1a; 1) 受控制字符序列&#xff…

专项练习9

目录 一、选择题 1、在 JavaScript 中&#xff0c;用于阻止默认事件的默认操作的方法是 2、以下代码执行后&#xff0c;result 的值为&#xff08;&#xff09; 3、不能从字符串 const str qwbewrbbeqqbbbweebbbbqee;中能得到结果 ["b", "bb", "bbb…

实时在线云消费机、考勤门禁控制器、网络读卡器服务端C# Socket源码

消费机UDP通讯协议介绍&#xff1a; 设备向服务器发送的指令格式&#xff0c;每个字段用半角逗号(,)分隔。序号指令名称指令格式指令说明示例1响应服务器的搜索100,包序列号,终端IP,子网掩码,网关IP,远程电脑主机IP,端口号,终端硬件号响应电脑发出的搜寻局域网内所有终端设备指…

【Python 基础篇】Python 异常处理

文章目录 引言一、Python异常概述二、常见的内置异常三、异常处理语句四、异常捕获和处理五、实例演示六、总结 引言 在软件开发中&#xff0c;错误和异常是难以避免的。当我们编写Python代码时&#xff0c;有时候会遇到各种各样的问题&#xff0c;例如无效的输入、文件不存在…

hello算法笔记之树

一、二叉树 与链表类似&#xff0c;二叉树的基本单元是节点&#xff0c;每个节点包含一个「值」和两个「指针」。 在二叉树中&#xff0c;除叶节点外&#xff0c;其他所有节点都包含子节点和非空子树。 一些术语&#xff1a; 「根节点 Root Node」&#xff1a;位于二叉树顶…

VNC虚拟网络控制台(概述、windows系统连接linux系统演示)

第三阶段基础 时 间&#xff1a;2023年6月22日 参加人&#xff1a;全班人员 内 容&#xff1a; VNC虚拟网络控制台 目录 一、VNC概述 二、VNC基本上是由两部分组成 三、VNC特点 四、工作流程 五、安装 六、操作演示Windiws10系统远程控制linux 服务端&#xff1a;…

window版安装kafka并提供启动快捷脚本

kafka下载地址&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1DpcGXvpTYAcG_fvS-p9-3g?pwd1234 提取码&#xff1a;1234 官网&#xff1a;https://kafka.apache.org/downloads 注意不需要单独安装zk&#xff0c;里面包括zk Kafka解压包目录不要太深了&#xff0c…

养老院人员跌倒检测识别算法

养老院人员跌倒检测识别预警系统通过yolov5python网络模型技术&#xff0c;养老院人员跌倒检测识别预警算法对跌倒事件进行识别和分析&#xff0c;当检测到有人员跌倒时&#xff0c;将自动发出警报提示相关人员及时采取措施。YOLOv5是一种单阶段目标检测算法&#xff0c;该算法…

CTF-Show密码学【Base64、栅栏密码、16进制】

题目内容 密文&#xff1a;53316C6B5A6A42684D3256695A44566A4E47526A4D5459774C5556375A6D49324D32566C4D4449354F4749345A6A526B4F48303D 提交格式&#xff1a;KEY{XXXXXXXXXXXXXX}工具下载&#xff1a;https://www.lanzoui.com/i9fn2aj萌新_密码13 分析和解决过程 初步分析…