ArduPilot开源飞控之AP_Mount_Topotek

news2024/11/17 8:55:59

ArduPilot开源飞控之AP_Mount_Topotek

  • 1. 源由
  • 2. 框架设计
  • 3. 重要函数
    • 3.1 动态过程
      • 3.1.1 AP_Mount_Topotek::update
      • 3.1.2 AP_Mount_Backend::calculate_poi
    • 3.2 基础能力
      • 3.2.1 AP_Mount_Topotek::healthy
      • 3.2.2 AP_Mount_Topotek::has_pan_control
    • 3.3 设备功能
      • 3.3.1 AP_Mount_Topotek::take_picture
      • 3.3.2 AP_Mount_Topotek::record_video
      • 3.3.3 AP_Mount_Topotek::set_zoom
      • 3.3.4 AP_Mount_Topotek::set_focus
      • 3.3.5 AP_Mount_Topotek::set_tracking
      • 3.3.6 AP_Mount_Topotek::cancel_tracking
      • 3.3.7 AP_Mount_Topotek::set_lens
    • 3.4 测距功能
      • 3.4.1 AP_Mount_Topotek::get_rangefinder_distance
      • 3.4.2 AP_Mount_Topotek::set_rangefinder_enable
    • 3.5 辅助函数
      • 3.5.1 AP_Mount_Topotek::set_camera_source
      • 3.5.2 AP_Mount_Topotek::send_camera_information
      • 3.5.3 AP_Mount_Topotek::send_camera_settings
  • 4. 总结
  • 5. 参考资料

1. 源由

AP_Mount_Topotek是最近上传的代码,也是看下来最为独立且完善的云台设备(含摄像头、测距、ROI跟随等)后端代码。

  • AP_Mount: integrate topotek gimbal driver
  • AP_Mount: add topotek backend

在这里插入图片描述

2. 框架设计

  1. 构造函数:继承自 AP_Mount_Backend_Serial 的构造函数,使用 using 关键字。

  2. 禁止复制:使用 CLASS_NO_COPY 宏显式禁止 AP_Mount_Topotek 实例的复制。

  3. 重写方法

    • update():更新安装位置。
    • healthy() const:检查安装是否正常。
    • has_pan_control() const:如果安装可以控制平移,则返回 true。
    • 多个与摄像头控制相关的方法(如 take_picture()record_video()set_zoom()set_focus() 等)。
    • 发送摄像头信息和设置到地面控制站(GCS)的方法。
    • 与测距仪交互的方法。
  4. 枚举

    • HeaderTypeAddressByteControlByteParseState:用于数据包解析和通信协议的枚举类型。
  5. 私有成员

    • 各种布尔标志和计数器(如 _recording_is_tracking_sdcard_status 等),用于管理内部状态。
    • 缓冲区(_msg_buff)和结构体(_parser),用于消息处理和解析。
  6. 私有方法

    • 用于读取传入数据包、请求云台信息、向云台发送命令(如 send_angle_target()send_rate_target())以及分析云台响应(如 gimbal_angle_analyse()gimbal_record_analyse() 等)的方法。
    • 用于计算校验和、十六进制转换和处理数据包传输的实用方法。
  7. 数据结构

    • Identifier:用于表示标识符的固定大小数组的 typedef
    • UartCmdFunctionHandler:用于将 UART 命令键映射到成员函数以进行消息处理的结构体。
  8. 实例变量

    • 各种实例变量(如 _last_tracking_state_last_mode_firmware_ver 等),用于存储云台的状态和接收到的信息。

3. 重要函数

3.1 动态过程

3.1.1 AP_Mount_Topotek::update

AP_Mount_Topotek::update() // 更新云台位置 - 应定期调用
|
|-- if (!_initialised)
|   |-- return;  // 未初始化则退出
|
|-- read_incoming_packets()  // 读取来自云台的传入数据包
|
|-- uint32_t now_ms = AP_HAL::millis();  // 10Hz更新频率
|   |-- if ((now_ms - _last_req_current_info_ms) < 100)
|       |-- return;  // 控制更新频率,避免过于频繁
|   |-- _last_req_current_info_ms = now_ms;
|
|-- if (_last_zoom_stop)
|   |-- // 重发停止变焦命令第二次,以防止数据传输错误
|   |-- _last_zoom_stop = false;
|   |-- send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_ZOOM, true, 0);
|
|-- if (_last_focus_stop)
|   |-- // 重发停止对焦命令第二次,以防止数据传输错误
|   |-- _last_focus_stop = false;
|   |-- send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_FOCUS, true, 0);
|
|-- send_location_info()  // 发送与GPS相关的信息到云台
|
|-- _last_req_step++;  // 1Hz频率调用
|   |-- if (_last_req_step >= 10)
|       |-- _last_req_step = 0;
|
|-- switch (_last_req_step)
|   |-- case 0:
|       |-- // 获取云台版本
|       |-- if (!_got_gimbal_version)
|           |-- request_gimbal_version();
|       |-- break;
|   |-- case 2:
|       |-- // 请求云台姿态,1Hz
|       |-- request_gimbal_attitude();
|       |-- break;
|   |-- case 4:
|       |-- // 请求存储卡信息
|       |-- request_gimbal_sdcard_info();
|       |-- break;
|   |-- case 6:
|       |-- // 请求跟踪信息
|       |-- if (_is_tracking)
|           |-- request_track_status();
|       |-- break;
|
|-- set_rctargeting_on_rcinput_change()  // 若RC输入发生变化,则切换到RC_TARGETING模式
|
|-- if (_is_tracking)  // 处理跟踪状态
|   |-- if (_last_mode != _mode)
|       |-- // 若模式发生变化,则取消跟踪
|       |-- cancel_tracking();
|   |-- else
|       |-- return;  // 图像跟踪激活,不发送姿态目标
|
|-- _last_mode = _mode;
|
|-- switch (get_mode())  // 根据云台模式更新
|   |-- case MAV_MOUNT_MODE_RETRACT:
|       |-- // 将云台移动到“收起”位置
|       |-- const Vector3f &angle_bf_target = _params.retract_angles.get();
|       |-- mnt_target.target_type = MountTargetType::ANGLE;
|       |-- mnt_target.angle_rad.set(angle_bf_target * DEG_TO_RAD, false);
|       |-- break;
|   |-- case MAV_MOUNT_MODE_NEUTRAL:
|       |-- // 将云台移动到中性位置
|       |-- const Vector3f &angle_bf_target = _params.neutral_angles.get();
|       |-- mnt_target.target_type = MountTargetType::ANGLE;
|       |-- mnt_target.angle_rad.set(angle_bf_target * DEG_TO_RAD, false);
|       |-- break;
|   |-- case MAV_MOUNT_MODE_MAVLINK_TARGETING:
|       |-- // mavlink目标处理
|       |-- break;
|   |-- case MAV_MOUNT_MODE_RC_TARGETING:
|       |-- // RC_TARGETING模式,使用RC输入更新目标
|       |-- MountTarget rc_target;
|       |-- get_rc_target(mnt_target.target_type, rc_target);
|       |-- switch (mnt_target.target_type)
|           |-- case MountTargetType::ANGLE:
|               |-- mnt_target.angle_rad = rc_target;
|               |-- break;
|           |-- case MountTargetType::RATE:
|               |-- mnt_target.rate_rads = rc_target;
|               |-- break;
|       |-- break;
|   |-- case MAV_MOUNT_MODE_GPS_POINT:
|       |-- // 将云台指向GPS点
|       |-- if (get_angle_target_to_roi(mnt_target.angle_rad))
|           |-- mnt_target.target_type = MountTargetType::ANGLE;
|       |-- break;
|   |-- case MAV_MOUNT_MODE_HOME_LOCATION:
|       |-- // 将云台指向Home位置
|       |-- if (get_angle_target_to_home(mnt_target.angle_rad))
|           |-- mnt_target.target_type = MountTargetType::ANGLE;
|       |-- break;
|   |-- case MAV_MOUNT_MODE_SYSID_TARGET:
|       |-- // 将云台指向另一个车辆
|       |-- if (get_angle_target_to_sysid(mnt_target.angle_rad))
|           |-- mnt_target.target_type = MountTargetType::ANGLE;
|       |-- break;
|   |-- default:
|       |-- // 未知模式,引发内部错误
|       |-- INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
|       |-- break;
|
|-- switch (mnt_target.target_type) // 根据目标类型发送目标角度或速率
|   |-- case MountTargetType::ANGLE:
|       |-- send_angle_target(mnt_target.angle_rad);
|       |-- break;
|   |-- case MountTargetType::RATE:
|       |-- send_rate_target(mnt_target.rate_rads);
|       |-- break;

3.1.2 AP_Mount_Backend::calculate_poi

略,详见:ArduPilot开源飞控之AP_Mount_Backend

3.2 基础能力

3.2.1 AP_Mount_Topotek::healthy

// 如果健康则返回true
bool AP_Mount_Topotek::healthy() const
{
    // 如果未初始化,则立即退出
    if (!_initialised) {
        return false;
    }

    // 如果最近没有接收到姿态信息,则认为不健康
    const uint32_t last_current_angle_ms = _last_current_angle_ms;
    return (AP_HAL::millis() - last_current_angle_ms < AP_MOUNT_TOPOTEK_HEALTH_TIMEOUT_MS);
}

3.2.2 AP_Mount_Topotek::has_pan_control

// has_pan_control - 如果该云台可以控制其水平旋转(多旋翼飞行器所需),则返回 true
bool has_pan_control() const override { return yaw_range_valid(); };

3.3 设备功能

3.3.1 AP_Mount_Topotek::take_picture

// 拍摄照片。成功返回 true
bool AP_Mount_Topotek::take_picture()
{
    // 如果未初始化,立即退出
    if (!_initialised) {
        return false;
    }

    // 如果内存卡异常,立即退出
    if (!_sdcard_status) {
        GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "%s SD 卡错误", send_message_prefix);
        return false;
    }

    // 示例命令: #TPUD2wCAP01
    return send_fixedlen_packet(AddressByte::SYSTEM_AND_IMAGE, AP_MOUNT_TOPOTEK_ID3CHAR_CAPTURE, true, 1);
}

3.3.2 AP_Mount_Topotek::record_video

// 启动或停止视频录制。成功时返回 true
// 设置 start_recording = true 开始录制,设置为 false 停止录制
bool AP_Mount_Topotek::record_video(bool start_recording)
{
    // 如果未初始化,立即退出
    if (!_initialised) {
        return false;
    }

    // 如果存储卡异常,立即退出
    if (!_sdcard_status) {
        GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "%s SD 卡错误", send_message_prefix);
        return false;
    }

    // 示例命令: #TPUD2wREC01
    return send_fixedlen_packet(AddressByte::SYSTEM_AND_IMAGE, AP_MOUNT_TOPOTEK_ID3CHAR_RECORD_VIDEO, true, start_recording ? 1 : 0);
}

3.3.3 AP_Mount_Topotek::set_zoom

// 设置缩放指定为比例
bool AP_Mount_Topotek::set_zoom(ZoomType zoom_type, float zoom_value)
{
    // 如果没有初始化则立即退出
    if (!_initialised) {
        return false;
    }

    // 缩放比例
    if (zoom_type == ZoomType::RATE) {
        uint8_t zoom_cmd;
        if (is_zero(zoom_value)) {
            // 停止缩放
            zoom_cmd = 0;
            _last_zoom_stop = true;
        } else if (zoom_value < 0) {
            // 缩小
            zoom_cmd = 1;
        } else {
            // 放大
            zoom_cmd = 2;
        }
        // 示例命令: #TPUM2wZMC00
        return send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_ZOOM, true, zoom_cmd);
    }

    // 不支持的缩放类型
    return false;
}

3.3.4 AP_Mount_Topotek::set_focus

// 设置对焦类型,可以是速度、百分比或自动
// focus in = -1, focus hold = 0, focus out = 1
SetFocusResult AP_Mount_Topotek::set_focus(FocusType focus_type, float focus_value)
{
    // 如果没有初始化,立即退出
    if (!_initialised) {
        return SetFocusResult::FAILED;
    }

    switch (focus_type) {
    case FocusType::RATE: {
        // 停止对焦
        uint8_t focus_cmd;
        if (is_zero(focus_value)) {
            focus_cmd = 0;
            _last_focus_stop = true;
        } else if (focus_value < 0) {
            // 对焦-
            focus_cmd = 2;
        } else {
            // 对焦+
            focus_cmd = 1;
        }
        // 发送对焦命令并切换到手动对焦
        // 示例命令: #TPUM2wFCC00
        if (send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_FOCUS, true, focus_cmd) &&
            send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_FOCUS, true, 0x11)) {
                return SetFocusResult::ACCEPTED;
        }
        return SetFocusResult::FAILED;
    }
    case FocusType::PCT:
        // 不支持
        return SetFocusResult::INVALID_PARAMETERS;
    case FocusType::AUTO:
        // 自动对焦
        if (send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_CONTROL_FOCUS, true, 0x10)) {
            return SetFocusResult::ACCEPTED;
        }
        return SetFocusResult::FAILED;
    }

    // 不支持的对焦类型
    return SetFocusResult::INVALID_PARAMETERS;
}

3.3.5 AP_Mount_Topotek::set_tracking

// 设置跟踪模式为无、点或矩形(参见 TrackingType 枚举)
// 如果是 POINT 仅使用 p1,如果是 RECTANGLE 则 p1 是左上角,p2 是右下角
// p1、p2 的范围是 0 到 1。0 表示左或上,1 表示右或下
bool AP_Mount_Topotek::set_tracking(TrackingType tracking_type, const Vector2f& p1, const Vector2f& p2)
{
    // 如果未初始化则立即退出
    if (!_initialised) {
        return false;
    }

    // 局部变量保存跟踪中心和宽度
    int16_t track_center_x, track_center_y, track_width, track_height;
    bool send_tracking_cmd = false;

    switch (tracking_type) {

    case TrackingType::TRK_NONE:
        return cancel_tracking();

    case TrackingType::TRK_POINT: {
        // 计算跟踪中心、宽度和高度
        track_center_x = (int16_t)((p1.x*TRACK_TOTAL_WIDTH - 960) /  0.96);
        track_center_y = (int16_t)((p1.y*TRACK_TOTAL_HEIGHT - 540) /  0.54);
        track_width = (int16_t)(TRACK_RANGE / 0.96);
        track_height = (int16_t)(TRACK_RANGE / 0.54);
        send_tracking_cmd = true;
        break;
    }

    case TrackingType::TRK_RECTANGLE:
        // 计算左上角和右下角点
        // 处理 p1 和 p2 顺序意外的情况
        int16_t upper_leftx = (int16_t)(MIN(p1.x, p2.x)*TRACK_TOTAL_WIDTH);
        int16_t upper_lefty = (int16_t)(MIN(p1.y, p2.y)*TRACK_TOTAL_HEIGHT);
        int16_t bottom_rightx = (int16_t)(MAX(p1.x, p2.x)*TRACK_TOTAL_WIDTH);
        int16_t bottom_righty = (int16_t)(MAX(p1.y, p2.y)*TRACK_TOTAL_HEIGHT);

        // 计算宽度和高度并进行合理性检查 
        const int16_t frame_selection_width = bottom_rightx - upper_leftx;
        const int16_t frame_selection_height = bottom_righty - upper_lefty;
        if (frame_selection_width <= 0 或 frame_selection_height <= 0) {
            return false;
        }

        // 计算跟踪中心
        track_center_x = (int16_t)((((upper_leftx + bottom_rightx) * 0.5) - 960) / 0.96);
        track_center_y = (int16_t)((((upper_lefty + bottom_righty) * 0.5) - 540) / 0.54);

        // 转换后的跟踪范围
        track_width = (int16_t)(frame_selection_width / 0.96);
        track_height = (int16_t)(frame_selection_height / 0.54);

        send_tracking_cmd = true;
        break;
    }

    if (send_tracking_cmd) {
        // 准备数据字节
        uint8_t databuff[10];
        databuff[0] = HIGHBYTE(track_center_x);
        databuff[1] = LOWBYTE(track_center_x);
        databuff[2] = HIGHBYTE(track_center_y);
        databuff[3] = LOWBYTE(track_center_y);
        databuff[4] = HIGHBYTE(track_width);
        databuff[5] = LOWBYTE(track_width);
        databuff[6] = HIGHBYTE(track_height);
        databuff[7] = LOWBYTE(track_height);
        databuff[8] = 0;
        databuff[9] = 0;

        // 发送跟踪命令
        bool res = send_variablelen_packet(HeaderType::VARIABLE_LEN,
                                           AddressByte::SYSTEM_AND_IMAGE,
                                           AP_MOUNT_TOPOTEK_ID3CHAR_START_TRACKING,
                                           true,
                                           (uint8_t*)databuff, ARRAY_SIZE(databuff));
        _is_tracking |= res;
        return res;
    }

    // 不应该到达这里
    return false;
}

3.3.6 AP_Mount_Topotek::cancel_tracking

// 向云台发送取消跟踪命令(如果有必要)
// 成功返回 true,发送消息失败返回 false
bool AP_Mount_Topotek::cancel_tracking()
{
    // 如果未初始化则立即退出
    if (!_initialised) {
        return false;
    }

    // 发送跟踪命令
    if (send_fixedlen_packet(AddressByte::SYSTEM_AND_IMAGE, AP_MOUNT_TOPOTEK_ID3CHAR_TRACKING, true, 1)) {
        return true;
    }
    return false;
}

3.3.7 AP_Mount_Topotek::set_lens

// 设置摄像头画中画模式
bool AP_Mount_Topotek::set_lens(uint8_t lens)
{
    // 如果未初始化,立即退出
    if (!_initialised) {
        return false;
    }

    // 检查镜头编号的有效性
    // 00: 仅主镜头, 01: 主镜头+副镜头, 02: 副镜头+主镜头, 03: 仅副镜头, 0A: 下一个
    // 示例命令: #TPUD2wPIP0A
    if (lens > 3) {
        return false;
    }

    // 发送画中画命令
    return send_fixedlen_packet(AddressByte::SYSTEM_AND_IMAGE, AP_MOUNT_TOPOTEK_ID3CHAR_PIP, true, lens);
}

3.4 测距功能

3.4.1 AP_Mount_Topotek::get_rangefinder_distance

// 获取测距仪距离。成功时返回 true
bool AP_Mount_Topotek::get_rangefinder_distance(float& distance_m) const
{
    // 如果不健康或距离为负则返回 false
    // healthy() 检查姿态超时,该超时与测距仪距离在同一消息中
    if (!healthy() || (_measure_dist_m < 0)) {
        return false;
    }

    distance_m = _measure_dist_m;
    return true;
}

3.4.2 AP_Mount_Topotek::set_rangefinder_enable

// enable/disable rangefinder.  Returns true on success
// 启用/禁用测距仪。成功时返回true
bool AP_Mount_Topotek::set_rangefinder_enable(bool enable)
{
    // exit immediately if not initialised
    // 如果未初始化,则立即退出
    if (!_initialised) {
        return false;
    }

    // 00:ranging stop, 01:ranging start, 02:single measurement, 03:continuous measurement
    // 00:停止测距,01:开始测距,02:单次测量,03:连续测量
    // sample command: #TPUM2wLRF00
    // 示例命令:#TPUM2wLRF00
    return send_fixedlen_packet(AddressByte::LENS, AP_MOUNT_TOPOTEK_ID3CHAR_LRF, true, enable ? 3 : 0);
}

3.5 辅助函数

3.5.1 AP_Mount_Topotek::set_camera_source

// set_camera_source功能上与set_lens相同,只是通过类型指定主要和次要镜头
// 主要和次要源使用AP_Camera::CameraSource枚举转换为uint8_t
bool AP_Mount_Topotek::set_camera_source(uint8_t primary_source, uint8_t secondary_source)
{
    // 如果未初始化,则立即退出
    if (!_initialised) {
        return false;
    }

    // 将主要和次要源映射到画中画(PiP)设置
    // PiP设置 00:仅主,01:主+次,02:次+主,03:仅次,0A:下一个
    // 示例命令:#TPUD2wPIP0A
    uint8_t pip_setting = 0;
    switch (primary_source) {
    case 0: // 默认(RGB)
        FALLTHROUGH;  // 继续执行下一个case
    case 1: // RGB
        switch (secondary_source) {
        case 0: // RGB + 默认(无)
            pip_setting = 0;    // 仅主
            break;
        case 2: // PIP RGB+IR
            pip_setting = 1;    // 主+次
            break;
        default:
            return false;
        }
        break;
    case 2: // IR
        switch (secondary_source) {
        case 0: // IR + 默认(无)
            pip_setting = 3;    // 仅次
            break;
        case 1: // IR+RGB
            pip_setting = 2;    // 次+主
            break;
        default:
            return false;
        }
        break;
    default:
        return false;
    }

    // 发送PiP命令
    return send_fixedlen_packet(AddressByte::SYSTEM_AND_IMAGE, AP_MOUNT_TOPOTEK_ID3CHAR_PIP, true, pip_setting);
}

3.5.2 AP_Mount_Topotek::send_camera_information

// 发送相机信息消息到地面控制站
void AP_Mount_Topotek::send_camera_information(mavlink_channel_t chan) const
{
    // 如果未初始化,则立即退出
    if (!_initialised) {
        return;
    }

    static const uint8_t vendor_name[32] = "Topotek";  // 厂商名称
    static uint8_t model_name[32] {};  // 模型名称
    const char cam_definition_uri[140] {};  // 相机定义URI

    // 能力标志
    const uint32_t flags = CAMERA_CAP_FLAGS_CAPTURE_VIDEO |
                           CAMERA_CAP_FLAGS_CAPTURE_IMAGE |
                           CAMERA_CAP_FLAGS_HAS_BASIC_ZOOM |
                           CAMERA_CAP_FLAGS_HAS_BASIC_FOCUS |
                           CAMERA_CAP_FLAGS_HAS_TRACKING_POINT |
                           CAMERA_CAP_FLAGS_HAS_TRACKING_RECTANGLE;

    // 发送相机信息消息
    mavlink_msg_camera_information_send(
        chan,
        AP_HAL::millis(),       // time_boot_ms,引导系统时间(毫秒)
        vendor_name,            // 厂商名称 uint8_t[32]
        model_name,             // 模型名称 uint8_t[32]
        _firmware_ver,          // 固件版本 uint32_t
        0,                      // 焦距 float (mm)
        0,                      // 水平传感器尺寸 float (mm)
        0,                      // 垂直传感器尺寸 float (mm)
        0,                      // 水平分辨率 uint16_t (像素)
        0,                      // 垂直分辨率 uint16_t (像素)
        0,                      // 镜头 ID uint8_t
        flags,                  // 标志 uint32_t (相机能力标志)
        0,                      // 相机定义版本 uint16_t
        cam_definition_uri,     // 相机定义URI char[140]
        _instance + 1);         // 云台设备 ID uint8_t
}

3.5.3 AP_Mount_Topotek::send_camera_settings

// 向 GCS 发送相机设置消息
void AP_Mount_Topotek::send_camera_settings(mavlink_channel_t chan) const
{
    // 如果未初始化,则立即退出
    if (!_initialised) {
        return;
    }

    const float NaN = nanf("0x4152");

    // 发送 CAMERA_SETTINGS 消息
    mavlink_msg_camera_settings_send(
        chan,
        AP_HAL::millis(),   // time_boot_ms,系统启动时间(毫秒)
        _recording ? CAMERA_MODE_VIDEO : CAMERA_MODE_IMAGE, // 相机模式(0: 图像, 1: 视频, 2: 图像勘测)
        NaN,                // zoomLevel 浮点数,百分比从 0 到 100,如果未知则为 NaN
        NaN);               // focusLevel 浮点数,百分比从 0 到 100,如果未知则为 NaN
}

4. 总结

该协议支持来源于拓扑联创,其英文网站Topotek。

由于开源社区体系的完善,设备提供方为了更好的服务客服,融入社区,就必须提供已有后端驱动接口或者提供上述对接源代码。

从设计的角度,这就是一个类似灰盒的接口暴露在外,供三方应用更好的集成和测试。这是一种非常好的设计模式,很高兴看到这么多Ardupilot Partner的设备源源不断的进入社区。

5. 参考资料

【1】ArduPilot开源飞控系统之简单介绍
【2】ArduPilot之开源代码Task介绍
【3】ArduPilot飞控启动&运行过程简介
【4】ArduPilot之开源代码Library&Sketches设计
【5】ArduPilot之开源代码Sensor Drivers设计
【6】ArduPilot开源飞控之AP_Mount
【7】ArduPilot开源飞控之AP_Mount_Backend

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

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

相关文章

【python】Python报错分析:深入探索`AttributeError`及其解决方案

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

论文翻译:Large Language Models for Education: A Survey

目录 大型语言模型在教育领域的应用&#xff1a;一项综述摘要1 引言2. 教育中的LLM特征2.1. LLMs的特征2.2 教育的特征2.2.1 教育发展过程 低进入门槛。2.2.2. 对教师的影响2.2.3 教育挑战 2.3 LLMEdu的特征2.3.1 "LLMs 教育"的具体体现2.3.2 "LLMs 教育"…

Kafka接收消息

文章目录 Acknowledgment读消息指定分区批量消费消息拦截 // 采用监听得方式接收 Payload标记消息体内容. KafkaListener(topics {"test"},groupId "hello") public void onEvent(Payload String event,Header(value KafkaHeaders.RECEIVED_TOPIC) Stri…

运算放大器(2)

&#xff08;1&#xff09;反向放大器 Vout(-R2/R1)*Vi 图一运放的同向端接地0V&#xff0c;反向端和同向端虚短&#xff0c;所以也是0V 反向输入端输入电阻很高&#xff0c;虚断&#xff0c;几乎没有电流注入和流出&#xff0c;那么R1和R2相当于是串联的&#xff0c;流过一个…

安装图形库后 帮助文档显示空白

EasyX的帮助文档显示空白 点击属性&#xff0c;选择解除锁定即可 这样就能浏览帮助文档了 电脑文件的属性中的"解除锁定"通常指的是文件或文件夹被锁定&#xff0c;导致用户无法对其进行编辑、移动或删除等操作。这种锁定可能是由于文件被其他程序占用、权限设置不当…

Spring源码十九:Bean实例化流程二

上一篇我们在Spring源码十八&#xff1a;Bean实例化流程一 中&#xff0c;主要讨论了Spring在实例化前的两重要准备工作&#xff0c;1、获取我们前面注册好的BeanDefinition&#xff0c;将GenericBeanDefinition封装为RootBeanDefinition如果Bean Definition只存在父容器中&…

“好物”推荐+Xshell连接实例+使用Conda创建独立的Python环境

目录 主题&#xff1a;好易智算平台推荐RTX 4090DGPU实例租用演示安装配置torch1.9.1cuda11.1.1环境引言&#xff1a;算力的新时代平台介绍&#xff1a;技术与信任的结晶使用案例&#xff1a;实际使用展示创建实例开始使用连接实例&#xff08;下文演示使用Xshell连接&#xff…

阿里云ECS服务器安装jdk并运行jar包,访问成功详解

安装 OpenJDK 8 使用 yum 包管理器安装 OpenJDK 8 sudo yum install -y java-1.8.0-openjdk-devel 验证安装 安装完成后&#xff0c;验证 JDK 是否安装成功&#xff1a; java -version设置 JAVA_HOME 环境变量&#xff1a; 为了确保系统中的其他应用程序可以找到 JDK&…

智能优化算法之蚁群算法ACO

蚁群算法&#xff08;Ant Colony Optimization, ACO&#xff09;由意大利学者马尔科多里戈&#xff08;Marco Dorigo&#xff09;于1992年在其博士论文中首次提出。灵感来自于自然界中的蚂蚁群体行为&#xff0c;特别是蚂蚁在寻找食物过程中所展示的路径优化能力。蚁群算法属于…

Snap Video:用于文本到视频合成的扩展时空变换器

图像生成模型的质量和多功能性的显著提升&#xff0c;研究界开始将其应用于视频生成领域。但是视频内容高度冗余&#xff0c;直接将图像模型技术应用于视频生成可能会降低运动的保真度和视觉质量&#xff0c;并影响可扩展性。来自 Snap 的研究团队及其合作者提出了 "Snap …

JAVA从入门到精通之入门初阶(二)

1. 自动类型转换 自动类型转换&#xff1a;类型范围小的变量可以赋值给类型范围大的变量 byte->int public class java_7_10 {public static void main(String[] args) {//自动类型转换//类型范围小的变量可以赋值给类型范围大的变量 byte->intbyte a 12;int b a;//自动…

MT6985(天玑9200)芯片性能参数_MTK联发科旗舰5G移动平台处理器

MT6985天玑 9200 旗舰移动平台拥有专业级影像、沉浸式游戏和先进移动显示技术&#xff0c;以及更快捷、覆盖更广的 5G 和 支持 Wi-Fi 7 连接&#xff0c;具有高性能、高能效、低功耗表现。率先采用 Armv9 性能核&#xff0c;全部支持纯 64 位应用&#xff0c;开启高能效架构设计…

华为OD机试 - 堆内存申请(Java 2024 D卷 100分)

华为OD机试 2024D卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;D卷C卷A卷B卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;每一题都有详细的答题思路、详细的代码注释、样例测…

nginx正向代理和反向代理

nginx正向代理和反向代理 正向代理以及缓存配置 代理&#xff1a;客户端不再是直接访问服务器&#xff0c;通过代理服务器访问服务端。 正向代理&#xff1a;面向客户端&#xff0c;我们通过代理服务器的IP地址访问目标服务端。 服务端只知道代理服务器的地址&#xff0c;真…

【RHCE】基于密钥的身份验证(Win-Linux)

目的&#xff1a;要提⾼系统安全性&#xff0c;通过在 OpenSSH 服务器上禁⽤密码⾝份验证来强制进⾏基于密钥的⾝份验证。 1、一台虚拟机无需密码连接另一台虚拟机 .ssh目录 > 保存了ssh相关的key和一些记录文件 &#xff08;1&#xff09;生成密钥对 使⽤这个流程在本地…

智慧港口可视化:提高运营效率与安全性

智慧港口通过图扑可视化技术&#xff0c;实时展示船舶停泊、货物装卸等关键数据&#xff0c;提高运营效率&#xff0c;保障港口的整体安全性与可靠性。

IT资产管理专题丨一文读懂什么是企业IT资产管理系统

在现代企业管理中&#xff0c;IT资产的管理变得越来越重要。随着信息技术的飞速发展&#xff0c;企业IT资产种类繁多&#xff0c;包括硬件设备、软件应用、许可证、合同等。 如何有效管理和利用这些资产成为企业面临的一大挑战。本文将通过人物对话的形式&#xff0c;详细解读企…

离线语音识别芯片在智能生活中的应用

离线语音识别芯片&#xff0c;这一技术正逐渐渗透到我们日常生活的每一个角落&#xff0c;为众多产品带来前所未有的智能体验。它能够应用到多种产品中&#xff0c;‌包括但不限于&#xff1a;‌ 1、智能音箱&#xff1a;‌语音识别芯片作为智能音箱的核心&#xff0c;‌使用户…

中霖教育:2024年中级经济师备考还来得及吗?

【中霖教育怎么样】【中霖教育口碑】 2024年的中级经济师考试还未开始报名&#xff0c;考试时间在11月16日-11月17日进行&#xff0c;考生目前距离考试还有半年的准备时间。不同的考生人群针对性的复习方法不同&#xff0c;以下内容可以作为大家的参考。 1、零基础考生&#…

【MySQL系列】VARCHAR的底层存储

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…