新增了教程和注释以及文档,增加了一键编译并打开ozone调试的脚本

This commit is contained in:
NeoZng 2023-02-14 11:13:32 +08:00
parent 8fa03012cf
commit 7c76852041
41 changed files with 201 additions and 45 deletions

View File

@ -14,6 +14,7 @@
# target # target
###################################### ######################################
TARGET = basic_framework TARGET = basic_framework
SHELL = cmd.exe
###################################### ######################################

View File

@ -11,18 +11,33 @@
> 更多关于发布-订阅的实现,请参考`modules/message_center`下的文档。 > 更多关于发布-订阅的实现,请参考`modules/message_center`下的文档。
## robot_def.h
这是机器人的参数配置文件,必须要针对每个机器人进行修改。包括机器人的尺寸参数和性能参数等。你还需要在这里设定软硬件配置:云台板/底盘板/单板等。这里定义的宏会作为条件编译的决断。
app层共用的状态变量和结构体等也应该定义在这里例如用于应用之间通信的数据。记得通信变量要用:
```c
#pragma pack(1)
typedef struct
{
// your struct
} your_struct;
#pragma pack()
```
包裹起来取消字节对齐以防止出现访问8byte地址而出现错误。
## robot_cmd ## robot_cmd
机器人命令模块是对整个机器人的抽象,对于单板控制整车的情况,该应用应该包含接收控制指令的模块,例如遥控器、视觉通信模块。该模块会处理接收到的控制数据,并将其转化为**具体的、定量的**控制信息,发送给其他模块。 机器人命令模块是对整个机器人的抽象,对于单板控制整车的情况,该应用应该包含接收控制指令的模块,例如遥控器、视觉通信模块。该模块会处理接收到的控制数据,并将其转化为**具体的、定量的**控制信息,发送给其他模块。同时cmd应用会处理模块和应用离线的情况出现紧急状况时停止所有执行机构的运行。
如从遥控器获知当前右侧摇杆拨向上方则将遥控器发来的数值转化为底盘前进的速度值然后发送给其他应用。同时robot_cmd还要从其他应用获取反馈信息做出其他决策。可以将其视为整个机器人的**大脑**。 如从遥控器获知当前右侧摇杆拨向上方则将遥控器发来的数值转化为底盘前进的速度值然后发送给其他应用。同时robot_cmd还要从其他应用获取反馈信息做出其他决策。可以将其视为整个机器人的**大脑**。
robot_cmd工作起来就像一个遥控数据的兼容层不论数据的来源是视觉上位机/遥控器/键鼠/图传通信链路/ps手柄最后都会被转化成真实参考输入提供给其他的app。它的任务是将其他来源的数据映射到控制输入上。
## gimbal ## gimbal
以步兵为例云台应用应当包含两个电机分别用于驱动yaw和pitch轴还有一个imu开发板一般放在云台上。gimbal模块会接收robot_cmd发来的控制信息云台的角度、转速等并通过电机提供的接口完成电机的参考值设定。gimbal还要把imu的数据反馈给cmd用于和视觉的通信以及云台状态的判断。 以步兵为例云台应用应当包含两个电机分别用于驱动yaw和pitch轴(除非你是一个三轴的云台)还有一个imu开发板一般放在云台上。gimbal模块会接收robot_cmd发来的控制信息云台的角度、转速等并通过电机提供的接口完成电机的参考值设定。gimbal还要把imu的数据反馈给cmd用于和视觉的通信以及云台状态的判断。
@ -46,5 +61,5 @@
## 双板兼容 ## 双板兼容
此框架对单开发板/双开发板/多开发板的情况都提供了支持(多板一般只在工程机器人上出现,需要自己编写),目前通过条件编译实现了对单双板的切换。使用双板时,主控板在云台上,连接遥控器和上位机;副板在底盘上,负责底盘的运动控制和与裁判系统的通信。 此框架对单开发板/双开发板/多开发板的情况都提供了支持(多板一般只在工程机器人上出现,需要自己在robot_cmd和robot_def增加相应的条件编译选项robot.c中也不要忘记增加初始化和任务运行函数),目前通过条件编译实现了对单双板的切换。使用双板时,主控板在云台上,连接遥控器和上位机;副板在底盘上,负责底盘的运动控制和与裁判系统的通信。

View File

@ -24,6 +24,8 @@ Robot.c是整个机器人的抽象其下有4个应用robot_cmdgimbal
为了进一步解耦应用之间的关系,这里并没有采用层级结构(或设计模式中所谓的**工厂模式**即robot_cmd包含其他三个模块而采用了应用并列的**发布-订阅**机制四个应用之间没有任何相互包含关系他们之间的通信通过module层提供的`message_center`实现。每个应用会通过该模块向一些话题事件发布一些消息同时从一些话题订阅消息。如robot_cmd应用会发布其他三个模块的控制信息同时订阅其他三个模块的反馈信息。其他三个模块会订阅robot_cmd发布的控制信息同时发布反馈给robot_cmd的信息他们不需要知道彼此的存在只是从`message_center`处获取其他应用发布的消息或向自己发布的话题推送消息。 为了进一步解耦应用之间的关系,这里并没有采用层级结构(或设计模式中所谓的**工厂模式**即robot_cmd包含其他三个模块而采用了应用并列的**发布-订阅**机制四个应用之间没有任何相互包含关系他们之间的通信通过module层提供的`message_center`实现。每个应用会通过该模块向一些话题事件发布一些消息同时从一些话题订阅消息。如robot_cmd应用会发布其他三个模块的控制信息同时订阅其他三个模块的反馈信息。其他三个模块会订阅robot_cmd发布的控制信息同时发布反馈给robot_cmd的信息他们不需要知道彼此的存在只是从`message_center`处获取其他应用发布的消息或向自己发布的话题推送消息。
application在初始化module的时候初始化参数会包含部分bsp的内容但仅仅是外设和引脚的选择以及id设置用于通信的外设需要id设置。实际上当前框架的app层和cubemx初始化部分耦合在配置的时候就必须确定每个外设的作用和归属权一旦cubemx完成设置app层必须按照对应参数设置引脚和并分配module的外设.后续考虑将cubemx和bsp耦合去除顶层代码和底层的关系
## 整车程序流程 ## 整车程序流程

View File

@ -8,7 +8,7 @@
* @copyright Copyright (c) HNU YueLu EC 2022 all rights reserved * @copyright Copyright (c) HNU YueLu EC 2022 all rights reserved
* *
*/ */
#pragma once #pragma once // 可以用#pragma once代替#ifndef ROBOT_DEF_H(header guard)
#ifndef ROBOT_DEF_H #ifndef ROBOT_DEF_H
#define ROBOT_DEF_H #define ROBOT_DEF_H
@ -16,6 +16,7 @@
#include "master_process.h" #include "master_process.h"
#include "stdint.h" #include "stdint.h"
// 编译warning,提醒开发者修改机器人参数
#ifndef ROBOT_DEF_PARAM_WARNING #ifndef ROBOT_DEF_PARAM_WARNING
#define ROBOT_DEF_PARAM_WARNING #define ROBOT_DEF_PARAM_WARNING
#warning BE SURED THAT YOU HAVE ALREADY MODIFIED THESE PARAMETER TO FIT THE ROBOT #warning BE SURED THAT YOU HAVE ALREADY MODIFIED THESE PARAMETER TO FIT THE ROBOT
@ -27,6 +28,7 @@
// #define GIMBAL_BOARD //云台板 // #define GIMBAL_BOARD //云台板
// @todo: 增加机器人类型定义,后续是否要兼容所有机器人?(只兼容步兵英雄哨兵似乎就够了) // @todo: 增加机器人类型定义,后续是否要兼容所有机器人?(只兼容步兵英雄哨兵似乎就够了)
// 通过该宏,你可以直接将所有机器人的参数保存在一处,然后每次只需要修改这个宏就可以替换所有参数
/* 机器人类型定义 */ /* 机器人类型定义 */
// #define ROBOT_HERO 1 // 英雄机器人 // #define ROBOT_HERO 1 // 英雄机器人
// #define ROBOT_ENINEER 2 // 工程机器人 // #define ROBOT_ENINEER 2 // 工程机器人
@ -52,7 +54,7 @@
#define RADIUS_WHEEL 60 // 轮子半径 #define RADIUS_WHEEL 60 // 轮子半径
#define REDUCTION_RATIO_WHEEL 19.0f // 电机减速比,因为编码器量测的是转子的速度而不是输出轴的速度故需进行转换 #define REDUCTION_RATIO_WHEEL 19.0f // 电机减速比,因为编码器量测的是转子的速度而不是输出轴的速度故需进行转换
// 检查是否出现定义冲突,只允许一个开发板定义存在,否则编译会自动报错 // 检查是否出现主控板定义冲突,只允许一个开发板定义存在,否则编译会自动报错
#if (defined(ONE_BOARD) && defined(CHASSIS_BOARD)) || \ #if (defined(ONE_BOARD) && defined(CHASSIS_BOARD)) || \
(defined(ONE_BOARD) && defined(GIMBAL_BOARD)) || \ (defined(ONE_BOARD) && defined(GIMBAL_BOARD)) || \
(defined(CHASSIS_BOARD) && defined(GIMBAL_BOARD)) (defined(CHASSIS_BOARD) && defined(GIMBAL_BOARD))

View File

@ -1,3 +1,5 @@
待添加adc的bsp支持,应该提供阻塞/IT/DMA的量测接口 待添加adc的bsp支持,应该提供阻塞/IT/DMA的量测接口
是否需要bsp_adc?由于功能非常简单,似乎可以直接使用HAL的接口,没有必要多此一举进行封装? 是否需要bsp_adc?由于功能非常简单,似乎可以直接使用HAL的接口,没有必要多此一举进行封装?
回调函数.

View File

@ -3,12 +3,9 @@ BSP
这是BSP层的说明文档。 这是BSP层的说明文档。
> TODO: > TODO:
> 1. 增加SPI和I2C支持 > 1. 增加软件中断支持
> 2. 增加外部中断支持 > 2. 增加硬件CRC支持
> 3. 增加软件中断支持
> 4. 增加硬件CRC支持
> 5. 增加USB和虚拟串口支持
bsp的功能是提供对片上外设的封装即单片机芯片内部拥有的功能的封装。在开发板pcb上集成的模块应该放在module层而不是这里。 bsp的功能是提供对片上外设的封装即单片机芯片内部拥有的功能的封装。在开发板pcb上集成的模块应该放在module层而不是这里。
bsp应该提供几种接口。包括初始化接口一般命名为`XXXRegister()`调用此模块实现的必要功能如通信型外设CubeMX下的connectivity提供接收和发送的接口以及接收完成的数据回调函数。 bsp应该提供几种接口。包括初始化接口一般命名为`XXXRegister()`(对于只有一个instance的可以叫`XXXInit()`,但建议统一风格都叫register)调用此模块实现的必要功能如通信型外设CubeMX下的connectivity提供接收和发送的接口以及接收完成的数据回调函数。

View File

@ -11,9 +11,10 @@ void BSPInit()
{ {
DWT_Init(168); DWT_Init(168);
BSPLogInit(); BSPLogInit();
USBInit(); // 务必在进入操作系统之前执行USB初始化 USBInit(); // 务必在进入操作系统之前执行USBInit
// legacy support待删除,将在实现了led/tempctrl/buzzer的module之后移动到app层
// legacy support待删除,将在实现了led/tempctrl/buzzer的module之后移动到app层进行XXXRegister()
LEDInit(); LEDInit();
IMUTempInit(); IMUTempInit();
BuzzerInit(); BuzzerInit();

View File

@ -1,3 +1,5 @@
# legacy bsp # legacy bsp
这些bsp实现将在v0.1删除,因为他们实际上都是用pwm实现的,应当放在module层,以彻底隔离bsp和CubeMX的初始化代码.之后在修改CubeMX的初始化配置之后就不再需要修改bsp的内容了,所有的修改都会通过app层的初始化配置`xxx_Init_Config_s`来实现. 这些bsp实现将在v0.1删除,因为他们实际上都是用pwm实现的,应当放在module层,以彻底隔离bsp和CubeMX的初始化代码.现已提供了bsp_pwm的实现.
之后在修改CubeMX的初始化配置之后就不再需要修改bsp的内容了,所有的修改都会通过app层的初始化配置`xxx_Init_Config_s`来实现.

View File

@ -140,7 +140,7 @@ static void CANFIFOxCallback(CAN_HandleTypeDef *_hcan, uint32_t fifox)
/** /**
* @brief ,STM32的两个CAN设备共享两个FIFO * @brief ,STM32的两个CAN设备共享两个FIFO
* HAL库中的回调函数,__weak,() * HAL库中的回调函数,HAL声明为__weak,()
* FIFO0或FIFO1溢出时会调用这两个函数 * FIFO0或FIFO1溢出时会调用这两个函数
*/ */
// 下面的函数会调用CANFIFOxCallback()来进一步处理来自特定CAN设备的消息 // 下面的函数会调用CANFIFOxCallback()来进一步处理来自特定CAN设备的消息

View File

@ -8,6 +8,7 @@
#define CAN_MX_REGISTER_CNT 16 // 这个数量取决于CAN总线的负载 #define CAN_MX_REGISTER_CNT 16 // 这个数量取决于CAN总线的负载
#define MX_CAN_FILTER_CNT (2 * 14) // 最多可以使用的CAN过滤器数量,目前远不会用到这么多 #define MX_CAN_FILTER_CNT (2 * 14) // 最多可以使用的CAN过滤器数量,目前远不会用到这么多
#define DEVICE_CAN_CNT 2 // 根据板子设定,F407IG有CAN1,CAN2,因此为2;F334只有一个,则设为1 #define DEVICE_CAN_CNT 2 // 根据板子设定,F407IG有CAN1,CAN2,因此为2;F334只有一个,则设为1
// 如果只有1个CAN,还需要把bsp_can.c中所有的hcan2变量改为hcan1(别担心,主要是总线和FIFO的负载均衡,不影响功能)
/* can instance typedef, every module registered to CAN should have this variable */ /* can instance typedef, every module registered to CAN should have this variable */
#pragma pack(1) #pragma pack(1)

View File

@ -6,18 +6,15 @@
* @version V1.1.0 * @version V1.1.0
* @date 2022/3/8 * @date 2022/3/8
* @brief * @brief
******************************************************************************
* @attention
*
******************************************************************************
*/ */
#include "bsp_dwt.h" #include "bsp_dwt.h"
DWT_Time_t SysTime; static DWT_Time_t SysTime;
static uint32_t CPU_FREQ_Hz, CPU_FREQ_Hz_ms, CPU_FREQ_Hz_us; static uint32_t CPU_FREQ_Hz, CPU_FREQ_Hz_ms, CPU_FREQ_Hz_us;
static uint32_t CYCCNT_RountCount; static uint32_t CYCCNT_RountCount;
static uint32_t CYCCNT_LAST; static uint32_t CYCCNT_LAST;
uint64_t CYCCNT64; static uint64_t CYCCNT64;
/** /**
* @brief ,DWT CYCCNT寄存器是否溢出,CYCCNT_RountCount * @brief ,DWT CYCCNT寄存器是否溢出,CYCCNT_RountCount

View File

@ -23,7 +23,6 @@ typedef struct
uint16_t us; uint16_t us;
} DWT_Time_t; } DWT_Time_t;
extern DWT_Time_t SysTime;
/** /**
* @brief DWT,CPU频率,MHz * @brief DWT,CPU频率,MHz

View File

@ -1,3 +1,3 @@
# bsp_dwt # bsp_dwt
DWT是stm32内部的一个"隐藏资源",他的用途是给下载器提供准确的定时,从而为调试信息加上时间戳. DWT是stm32内部的一个"隐藏资源",他的用途是给下载器提供准确的定时,从而为调试信息加上时间戳.并在固定的时间间隔将调试数据发送到你的xxlink上.

View File

@ -6,9 +6,11 @@
![img](../../assets/00937839b59a4c039ee8ecb8a5136e3c.png) ![img](../../assets/00937839b59a4c039ee8ecb8a5136e3c.png)
使用示例 使用示例
```c ```c
//在app层只需要设置前三个,callback由module自动设置
GPIO_Init_Config_s gpio_init = { GPIO_Init_Config_s gpio_init = {
.exti_mode = GPIO_EXTI_MODE_FALLING, // 注意和CUBEMX的配置一致 .exti_mode = GPIO_EXTI_MODE_FALLING, // 注意和CUBEMX的配置一致
@ -19,4 +21,5 @@ GPIO_Init_Config_s gpio_init = {
GPIOInstance* test_example = GPIORegister(&gpio_init); GPIOInstance* test_example = GPIORegister(&gpio_init);
GPIOSet(test_example); GPIOSet(test_example);
// GPIOxxx(test_exmaple, ...);
``` ```

View File

@ -70,7 +70,7 @@ void IICReceive(IICInstance *iic, uint8_t *data, uint16_t size, IIC_Seq_Mode_e s
{ {
if (seq_mode != IIC_RELEASE && seq_mode != IIC_HOLD_ON) if (seq_mode != IIC_RELEASE && seq_mode != IIC_HOLD_ON)
while (1) while (1)
; // 未知传输模式, 程序停止 ; // 未知传输模式, 程序停止,请检查指针越界
// 初始化接收缓冲区地址以及接受长度, 用于中断回调函数 // 初始化接收缓冲区地址以及接受长度, 用于中断回调函数
iic->rx_buffer = data; iic->rx_buffer = data;
@ -140,7 +140,7 @@ void HAL_I2C_MasterRxCpltCallback(I2C_HandleTypeDef *hi2c)
} }
/** /**
* @brief ,使HAL_I2C_MasterRxCpltCallback * @brief 访,,使HAL_I2C_MasterRxCpltCallback
* *
* @param hi2c handle * @param hi2c handle
*/ */

View File

@ -5,3 +5,6 @@
https://blog.csdn.net/NeoZng/article/details/128496694 https://blog.csdn.net/NeoZng/article/details/128496694
https://blog.csdn.net/NeoZng/article/details/128486366 https://blog.csdn.net/NeoZng/article/details/128486366
使用序列通信则在单次通信后不会释放总线,继续占用直到调用传输函数时传入`IIC_RELEASE`参数. 这个功能只在一条总线上挂载多个主机的时候有用.

View File

@ -6,6 +6,11 @@
static uint8_t idx; static uint8_t idx;
static PWMInstance *pwm_instance[PWM_DEVICE_CNT] = {NULL}; // 所有的pwm instance保存于此,用于callback时判断中断来源 static PWMInstance *pwm_instance[PWM_DEVICE_CNT] = {NULL}; // 所有的pwm instance保存于此,用于callback时判断中断来源
/**
* @brief pwm dma传输完成回调函数
*
* @param htim
*/
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim) void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim)
{ {
for (uint8_t i = 0; i < idx; i++) for (uint8_t i = 0; i < idx; i++)

View File

@ -1,3 +1,15 @@
/**
* @file bsp_pwm.h
* @author your name (you@domain.com)
* @brief
* @version 0.1
* @date 2023-02-14
* @todo ,module修改tim的分频器和自动重装载寄存器?
*
* @copyright Copyright (c) 2023
*
*/
#ifndef BSP_PWM_H #ifndef BSP_PWM_H
#define BSP_PWM_H #define BSP_PWM_H
@ -52,7 +64,8 @@ void PWMStart(PWMInstance *pwm);
*/ */
void PWMStop(PWMInstance *pwm); void PWMStop(PWMInstance *pwm);
// @todo 这三个函数还需要进一步封装,协调好三者之间的关系
// @todo 这三个函数还需要进一步封装,务必协调好三者之间的关系
/** /**
* @brief pwm脉宽 * @brief pwm脉宽
* *

View File

@ -32,6 +32,7 @@ typedef struct spi_ins_temp
/* 接收回调函数定义,包含SPI的module按照此格式构建回调函数 */ /* 接收回调函数定义,包含SPI的module按照此格式构建回调函数 */
typedef void (*spi_rx_callback)(SPIInstance *); typedef void (*spi_rx_callback)(SPIInstance *);
// @todo: 这里可以将GPIO_TypeDef *GPIO_cs; uint16_t cs_pin合并为bsp_gpio以简化代码实现
/* SPI初始化配置,其实基本和SPIIstance一模一样,为了代码风格统一因此再次定义 */ /* SPI初始化配置,其实基本和SPIIstance一模一样,为了代码风格统一因此再次定义 */
typedef struct typedef struct
{ {

View File

@ -17,7 +17,8 @@ typedef enum
USART_TRANSFER_RX, USART_TRANSFER_RX,
} USART_TRANSFER_MODE; } USART_TRANSFER_MODE;
// 串口实例结构体,每个module都要包含一个实例 // 串口实例结构体,每个module都要包含一个实例.
// 由于串口是独占的点对点通信,所以不需要考虑多个module同时使用一个串口的情况,因此不用加入id;当然也可以选择加入,这样在bsp层可以访问到module的其他信息
typedef struct typedef struct
{ {
uint8_t recv_buff[USART_RXBUFF_LIMIT]; // 预先定义的最大buff大小,如果太小请修改USART_RXBUFF_LIMIT uint8_t recv_buff[USART_RXBUFF_LIMIT]; // 预先定义的最大buff大小,如果太小请修改USART_RXBUFF_LIMIT
@ -44,6 +45,8 @@ USARTInstance *USARTRegister(USART_Init_Config_s *init_config);
/** /**
* @brief ,usart实例,buff以及这一帧的长度 * @brief ,usart实例,buff以及这一帧的长度
* DMA发送, * DMA发送,
* @todo DMA发送,
* ,
* *
* @param _instance * @param _instance
* @param send_buf buffer * @param send_buf buffer

View File

@ -2,7 +2,8 @@
<p align='right'>neozng1@hnu.edu.cn</p> <p align='right'>neozng1@hnu.edu.cn</p>
> TODO:为初始化定义一个结构体`usart_init_config`用于保存初始化所需的参数从而避免单独赋值,使得整体风格统一。 > TODO: 增加发送队列以解决短时间内调用`USARTSend()`发生丢包的问题,并提供阻塞和IT模式以供选择
> 可以直接在发送函数的参数列表添加发送模式选择,或增加instance成员变量,并提供设置模式接口,两者各有优劣
## 使用说明 ## 使用说明

View File

@ -3,7 +3,7 @@
* @author your name (you@domain.com) * @author your name (you@domain.com)
* @brief usb vpc(virtal com port),hid和msf考虑后续添加 * @brief usb vpc(virtal com port),hid和msf考虑后续添加
* @todo usart的接口统一完成兼容 * @todo usart的接口统一完成兼容
* @attention usbd_cdc_if.c中的CDC_Receive_FS函数,cube生成后会被覆盖.usbcdciftemplate创建一套自己的模板 * @attention usb修改了usbd_cdc_if.c中的CDC_Receive_FS函数,使cube生成后会被覆盖.usbcdciftemplate创建一套自己的模板
* @version 0.1 * @version 0.1
* @date 2023-02-09 * @date 2023-02-09
* *

View File

@ -379,6 +379,7 @@ void BMI088CalibrateIMU(BMI088Instance *_bmi088)
fabsf(_bmi088->gyro_offset[2]) > 0.01f); // 满足条件说明标定环境不好 fabsf(_bmi088->gyro_offset[2]) > 0.01f); // 满足条件说明标定环境不好
} }
// 离线标定
if (_bmi088->cali_mode == BMI088_LOAD_PRE_CALI_MODE) // 如果标定失败也会进来,直接使用离线数据 if (_bmi088->cali_mode == BMI088_LOAD_PRE_CALI_MODE) // 如果标定失败也会进来,直接使用离线数据
{ {
// 读取标定数据 // 读取标定数据
@ -408,6 +409,7 @@ BMI088Instance *BMI088Register(BMI088_Init_Config_s *config)
config->spi_gyro_config.id = config->spi_gyro_config.id =
config->heat_pwm_config.id = bmi088_instance; config->heat_pwm_config.id = bmi088_instance;
// @todo:
// 目前只实现了!!!阻塞读取模式!!!.如果需要使用IT模式,则需要修改这里的代码,为spi和gpio注册callback(默认为NULL) // 目前只实现了!!!阻塞读取模式!!!.如果需要使用IT模式,则需要修改这里的代码,为spi和gpio注册callback(默认为NULL)
// 还需要设置SPI的传输模式为DMA模式或IT模式(默认为blocking) // 还需要设置SPI的传输模式为DMA模式或IT模式(默认为blocking)
// 可以通过conditional compilation或者runtime参数判断 // 可以通过conditional compilation或者runtime参数判断

View File

@ -29,8 +29,8 @@ typedef struct
float acc[3]; // 加速度计数据,xyz float acc[3]; // 加速度计数据,xyz
float temperature; // 温度 float temperature; // 温度
// uint32_t timestamp; // 时间戳 // float timestamp; // 时间戳,单位为ms,用于计算两次采样的时间间隔,同时给视觉提供timeline
// uint32_t count; // uint32_t count; // 第count次采样,用于对齐时间戳
} BMI088_Data_t; } BMI088_Data_t;
#pragma pack() // 恢复默认对齐,需要传输的结构体务必开启1字节对齐 #pragma pack() // 恢复默认对齐,需要传输的结构体务必开启1字节对齐
@ -107,7 +107,8 @@ BMI088Instance *BMI088Register(BMI088_Init_Config_s *config);
BMI088_Data_t BMI088Acquire(BMI088Instance *bmi088); BMI088_Data_t BMI088Acquire(BMI088Instance *bmi088);
/** /**
* @brief * @brief .BMI088在初始化的时候会调用此函数. 便线
* @attention @todo ,,INS_Task冲突.,
* *
* @param _bmi088 * @param _bmi088
*/ */

View File

@ -47,6 +47,7 @@ write写入:
温度数据不需要和accel和gyro同步,它不参与姿态解算,可以不用管. 温度数据不需要和accel和gyro同步,它不参与姿态解算,可以不用管.
当加速度数据和陀螺仪数据都准备完成之后,唤醒姿态解算任务INs_taSk,进行一次解算.唤醒可以通过软件开启EXTI中断,在中断中调用实时系统的vTaskNotifyFromISR()完成,也可以将任务ready标志位置位,当运行到对应位置时检查标志位判断是否要进行任务.时间间隔不是大问题,inS_TAsK中有dwt用于计算两次任务执行的间隔,它将会自动处理好bias的大小 当加速度数据和陀螺仪数据都准备完成之后,唤醒姿态解算任务INs_taSk,进行一次解算.唤醒可以通过软件开启EXTI中断,在中断中调用实时系统的vTaskNotifyFromISR()完成,也可以将任务ready标志位置位,当运行到对应位置时检查标志位判断是否要进行任务.时间间隔不是大问题,inS_TAsK中有dwt用于计算两次任务执行的间隔,它将会自动处理好bias的大小
`__HAL_GPIO_EXTI_GENERATE_SWIT()` `HAL_EXTI_GENERATE_SWI()`
`__HAL_GPIO_EXTI_GENERATE_SWIT()` `HAL_EXTI_GENERATE_SWI()` 可以触发软件中断
of course,两者的数据更新实际上可以异步进行,这里为了方便起见当两者数据都准备好以后再行融合 of course,两者的数据更新实际上可以异步进行,这里为了方便起见当两者数据都准备好以后再行融合

View File

@ -1,7 +1,7 @@
#ifndef BMI088REG_H #ifndef BMI088REG_H
#define BMI088REG_H #define BMI088REG_H
/*------- BMI088寄存器地址内容-------*/ /*------- BMI088寄存器地址及其存放的内容-------*/
#define BMI088_ACC_CHIP_ID 0x00 // the register is " Who am I " #define BMI088_ACC_CHIP_ID 0x00 // the register is " Who am I "
#define BMI088_ACC_CHIP_ID_VALUE 0x1E #define BMI088_ACC_CHIP_ID_VALUE 0x1E

View File

@ -0,0 +1 @@
#include "LQR.h"

View File

@ -0,0 +1,12 @@
/**
* @file LQR.h
* @author your name (you@domain.com)
* @brief arm math库实现矩阵运算功能
* @version 0.1
* @date 2023-02-14
*
* @copyright Copyright (c) 2023
*
*/
#pragma once

View File

@ -71,7 +71,7 @@ void CANCommSend(CANCommInstance *instance, uint8_t *data);
* @return void* * @return void*
* @attention 访,union或struct,使pack(n) * @attention 访,union或struct,使pack(n)
* CANComm接收到的数据可以看作是pack(1),. * CANComm接收到的数据可以看作是pack(1),.
* 使pack(n),使访,. * 使pack(n),使memcpy直接访问,.
* CANComm传输的数据使用pack(1) * CANComm传输的数据使用pack(1)
*/ */
void *CANCommGet(CANCommInstance *instance); void *CANCommGet(CANCommInstance *instance);

View File

@ -1,6 +1,9 @@
#ifndef GENERAL_DEF_H #ifndef GENERAL_DEF_H
#define GENERAL_DEF_H #define GENERAL_DEF_H
// 一些module的通用数值型定义,注意条件macro兼容,一些宏可能在math.h中已经定义过了
#ifndef PI #ifndef PI
#define PI 3.1415926535f #define PI 3.1415926535f
#endif // !PI #endif // !PI

View File

@ -12,3 +12,4 @@
## 模块移植 ## 模块移植
由于历史遗留问题,IMU模块耦合程度高.后续实现BSP_SPI,将bmi088 middleware删除.仅保留BMI088读取的协议和寄存器定义等,单独实现IMU模块. 由于历史遗留问题,IMU模块耦合程度高.后续实现BSP_SPI,将bmi088 middleware删除.仅保留BMI088读取的协议和寄存器定义等,单独实现IMU模块.
> 移植已经完成,请转而使用module/BMI088的模块. 当前文件夹将在beta1.2停止支持, 1.5之后删除. INS_Task届时会被放到algorithm中,以提供对不同IMU的兼容.

View File

@ -21,6 +21,7 @@ IST8310_Init_Config_s ist8310_conf = {
.work_mode = IIC_BLOCK_MODE, .work_mode = IIC_BLOCK_MODE,
}, },
}; };
IST8310Instance *asdf = IST8310Init(&ist8310_conf); IST8310Instance *asdf = IST8310Init(&ist8310_conf);
// 随后数据会被放到asdf.mag[i]中 // 随后数据会被放到asdf.mag[i]中

View File

@ -1,3 +1,15 @@
/**
* @file oled.h
* @author your name (you@domain.com)
* @brief
* @version 0.1
* @date 2023-02-14
* @todo show string/init/clean/update buffer等的实现
*
* @copyright Copyright (c) 2023
*
*/
#ifndef OLED_H #ifndef OLED_H
#define OLED_H #define OLED_H
#include <stdint.h> #include <stdint.h>

View File

@ -0,0 +1,5 @@
# oled
当前oled支持不完整,api较为混乱. 需要使用bsp_iic进行实现的重构,并提供统一方便的接口
请使用字库软件制作自己的图标和不同大小的ascii码.

View File

@ -10,6 +10,7 @@
#ifndef __OLED__FONT__H #ifndef __OLED__FONT__H
#define __OLED__FONT__H #define __OLED__FONT__H
//the common ascii character //the common ascii character
const unsigned char asc2_1206[95][12]={ const unsigned char asc2_1206[95][12]={
{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*" ",0*/ {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*" ",0*/
@ -109,6 +110,7 @@ const unsigned char asc2_1206[95][12]={
{0x40,0x00,0x80,0x00,0x40,0x00,0x20,0x00,0x20,0x00,0x40,0x00},/*"~",94*/ {0x40,0x00,0x80,0x00,0x40,0x00,0x20,0x00,0x20,0x00,0x40,0x00},/*"~",94*/
}; };
// 你可以使用图标字库软件生成自己的pattern,软件会输出一个数组,将数组复制到这里即可.
//the logo of robomaster //the logo of robomaster
const unsigned char LOGO_BMP[128][8] = { const unsigned char LOGO_BMP[128][8] = {

View File

@ -0,0 +1,3 @@
# ps_handle
提供ps手柄的支持,其通信协议和spi类似,可以使用bsp_spi来兼容.待编写.

View File

@ -1,6 +1,6 @@
#include "crc_ref.h" #include "crc_ref.h"
#include <stdio.h> #include <stdio.h>
//裁判系统官方CRC校验
const uint8_t CRC8_INIT = 0xff; const uint8_t CRC8_INIT = 0xff;
const uint8_t CRC8_TAB[256] = const uint8_t CRC8_TAB[256] =

View File

@ -1,6 +1,8 @@
#ifndef __CRC_H_ #ifndef __CRC_H_
#define __CRC_H_ #define __CRC_H_
//裁判系统官方CRC校验,LUT和module/algorithms中的不同,后续需要统一实现crc,提供8/16/32的支持
#include <stdint.h> #include <stdint.h>
#define TRUE 1 #define TRUE 1

View File

@ -70,8 +70,9 @@ remote_control
``` ```
目前的映射:
拨轮向下打到底进入紧急停止模式(后续改为关闭遥控器停止,利用daemon);拨轮向上打开启摩擦轮,超过一半开始发射(速度环,连发) 拨轮向下打到底进入紧急停止模式(后续改为关闭遥控器停止,利用daemon; daemon已经模块完成);拨轮向上打开启摩擦轮,超过一半开始发射(速度环,连发)
左侧开关: 左侧开关:
- 上:键鼠控制 - 上:键鼠控制

View File

@ -9,7 +9,7 @@
#include "memory.h" #include "memory.h"
#include "stdlib.h" #include "stdlib.h"
static SuperCapInstance *super_cap_instance = NULL; static SuperCapInstance *super_cap_instance = NULL; // 可以由app保存此指针
static void SuperCapRxCallback(CANInstance *_instance) static void SuperCapRxCallback(CANInstance *_instance)
{ {

61
task.ps1 Normal file
View File

@ -0,0 +1,61 @@
JLinkGDBServer:
$(Q)JLinkGDBServer -select USB -device $(CHIP) \
-endian little -if SWD -speed 4000 -noir -LocalhostOnly
debug:
$(Q)make
$(Q)echo target remote localhost\:2331 > gdb.gdb
$(Q)echo monitor reset >> gdb.gdb
$(Q)echo monitor halt >> gdb.gdb
$(Q)echo load >> gdb.gdb
$(Q)echo b main >> gdb.gdb
$(Q)echo - >> gdb.gdb
$(Q)echo c >> gdb.gdb
$(Q)-$(GDB) $(BUILD)/$(TARGET).elf --command=gdb.gdb
$(Q)$(RM) gdb.gdb
debug_ozone:
$(Q)ozone ./debug_ozone.jdebug
download:
$(Q)make
$(Q)echo "h" > jlink.jlink
$(Q)echo "loadfile" $(BUILD)/$(TARGET).hex >> jlink.jlink
$(Q)echo "r" >> jlink.jlink
$(Q)echo "qc" >> jlink.jlink
$(Q)$(JLINKEXE) -device $(CHIP) -Speed 4000 -IF SWD -CommanderScript jlink.jlink
$(Q)$(RM) jlink.jlink
reset:
$(Q)echo "r" >> jlink.jlink
$(Q)echo "qc" >> jlink.jlink
$(Q)$(JLINKEXE) -device $(CHIP) -Speed 4000 -IF SWD -CommanderScript jlink.jlink
$(Q)$(RM) jlink.jlink