From a1509ee665acd004484cfca4167f8f4104f3e2be Mon Sep 17 00:00:00 2001
From: NeoZng
Date: Fri, 11 Nov 2022 21:00:02 +0800
Subject: [PATCH] add doc for bsp_usart, bsp_can and dji_motor
---
.vscode/settings.json | 3 +-
HAL_N_Middlewares/Src/main.c | 2 +-
Makefile | 6 +-
bsp/bsp_can.c | 2 +-
bsp/bsp_can.h | 2 +-
bsp/bsp_can.md | 98 ++++++++++
bsp/bsp_usart.c | 2 +-
bsp/bsp_usart.md | 69 +++++++
modules/algorithm/controller.h | 11 +-
modules/motor/dji_motor.h | 8 +-
modules/motor/dji_motor.md | 332 +++++++++++++++++++++++++++++++++
modules/motor/motor_def.h | 3 +-
12 files changed, 519 insertions(+), 19 deletions(-)
create mode 100644 bsp/bsp_can.md
create mode 100644 bsp/bsp_usart.md
create mode 100644 modules/motor/dji_motor.md
diff --git a/.vscode/settings.json b/.vscode/settings.json
index 46e315a..7bbc271 100644
--- a/.vscode/settings.json
+++ b/.vscode/settings.json
@@ -43,6 +43,7 @@
"vofa_protocol.h": "c",
"master_process.h": "c",
"stdint-gcc.h": "c",
- "string.h": "c"
+ "string.h": "c",
+ "motor_def.h": "c"
}
}
\ No newline at end of file
diff --git a/HAL_N_Middlewares/Src/main.c b/HAL_N_Middlewares/Src/main.c
index c276a02..fd640c3 100644
--- a/HAL_N_Middlewares/Src/main.c
+++ b/HAL_N_Middlewares/Src/main.c
@@ -158,7 +158,7 @@ int main(void)
DJIMotorSetRef(djimotor, get_remote_control_point()->rc.ch[0]);
MotorControlTask();
- HAL_Delay(10);
+ HAL_Delay(100);
/* USER CODE BEGIN 3 */
}
/* USER CODE END 3 */
diff --git a/Makefile b/Makefile
index 91bc6ec..434ef8e 100644
--- a/Makefile
+++ b/Makefile
@@ -255,7 +255,7 @@ $(BUILD_DIR)/%.o: %.s Makefile | $(BUILD_DIR)
$(BUILD_DIR)/$(TARGET).elf: $(OBJECTS) Makefile
@$(CC) $(OBJECTS) $(LDFLAGS) -o $@
- $(SZ) $@
+ @$(SZ) $@
$(BUILD_DIR)/%.hex: $(BUILD_DIR)/%.elf | $(BUILD_DIR)
$(HEX) $< $@
@@ -264,13 +264,13 @@ $(BUILD_DIR)/%.bin: $(BUILD_DIR)/%.elf | $(BUILD_DIR)
$(BIN) $< $@
$(BUILD_DIR):
- @mkdir $@
+ mkdir $@
#######################################
# clean up
#######################################
clean:
- -rm -fR $(BUILD_DIR)
+ -rm -r $(BUILD_DIR)
#######################################
# dependencies
diff --git a/bsp/bsp_can.c b/bsp/bsp_can.c
index 7cbd0b8..35d30e0 100644
--- a/bsp/bsp_can.c
+++ b/bsp/bsp_can.c
@@ -4,7 +4,7 @@
#include "memory.h"
/* can instance ptrs storage, used for recv callback */
-static can_instance *instance[MX_REGISTER_DEVICE_CNT];
+static can_instance *instance[MX_REGISTER_DEVICE_CNT]={NULL};
/* ----------------two static function called by CANRegister()-------------------- */
diff --git a/bsp/bsp_can.h b/bsp/bsp_can.h
index 5627a2d..35f1457 100644
--- a/bsp/bsp_can.h
+++ b/bsp/bsp_can.h
@@ -6,7 +6,7 @@
#define MX_REGISTER_DEVICE_CNT 12 // maximum number of device can be registered to CAN service
// this number depends on the load of CAN bus.
-#define MX_CAN_FILTER_CNT (4 * 14) // temporarily useless
+#define MX_CAN_FILTER_CNT (2 * 14) // temporarily useless
#define DEVICE_CAN_CNT 2 // CAN1,CAN2
diff --git a/bsp/bsp_can.md b/bsp/bsp_can.md
new file mode 100644
index 0000000..6e83cb1
--- /dev/null
+++ b/bsp/bsp_can.md
@@ -0,0 +1,98 @@
+# bsp_can
+
+neozng1@hnu.edu.cn
+
+> TODO:
+>
+> 1. 增加数据帧的长度定义,使得收发更加灵活,而不是固定的8 bytes
+> 2. 增加自动检测ID冲突的log输出。
+
+## 代码结构
+
+.h文件内包括了外部接口和类型定义,以及模块对应的宏。c文件内为私有函数和外部接口的定义。
+
+### 类型定义
+
+```c
+
+#define MX_REGISTER_DEVICE_CNT 12 // maximum number of device can be registered to CAN service, this number depends on the load of CAN bus.
+#define MX_CAN_FILTER_CNT (4 * 14) // temporarily useless
+#define DEVICE_CAN_CNT 2 // CAN1,CAN2
+
+/* can instance typedef, every module registered to CAN should have this variable */
+typedef struct _
+{
+ CAN_HandleTypeDef* can_handle;
+ CAN_TxHeaderTypeDef txconf;
+ uint32_t tx_id;
+ uint32_t tx_mailbox;
+ uint8_t tx_buff[8];
+ uint8_t rx_buff[8];
+ uint32_t rx_id;
+ void (*can_module_callback)(struct _*);
+} can_instance;
+
+typedef struct
+{
+ CAN_HandleTypeDef* can_handle;
+ uint32_t tx_id;
+ uint32_t rx_id;
+ void (*can_module_callback)(can_instance*);
+} can_instance_config;
+
+typedef void (*can_callback)(can_instance*);
+```
+
+- `MX_REGISTER_DEVICE_CNT`是最大的CAN设备注册数量,当每个设备的发送频率都较高时,设备过多会产生总线拥塞从而出现丢包和数据错误的情况。
+- `MX_CAN_FILTER_CNT`是最大的CAN接收过滤器数量,两个CAN共享标号0~27共28个过滤器。这部分内容比较繁杂,暂时不用理解,有兴趣自行参考MCU的数据手册。当前为简单起见,每个过滤器只设置一组规则用于控制一个id的过滤。
+- `DEVICE_CAN_CNT`是MCU拥有的CAN硬件数量。
+
+- `can_instance`是一个CAN实例。注意,CAN作为一个总线设备,一条总线上可以挂载多个设备,因此多个设备可以共享同一个CAN硬件。其成员变量包括发送id,发送邮箱(不需要管,只是一个32位变量,CAN收发器会自动设置其值),发送buff以及接收buff,还有接收id和接收协议解析回调函数。**由于目前使用的设备每个数据帧的长度都是8,因此收发buff长度暂时固定为8**。定义该结构体的时候使用了一个技巧,使得在结构体内部可以用结构体自身的指针作为成员,即`can_module_callback`的定义。
+
+- `can_instance_config`是用于初始化CAN实例的结构,在调用CAN实例的初始化函数时传入(下面介绍函数时详细介绍)。
+
+- `can_module_callback()`是模块提供给CAN接收中断回调函数使用的协议解析函数指针。对于每个需要CAN的模块,需要定义一个这样的函数用于解包数据。
+- 每个使用CAN外设的module,都需要在其内部定义一个`can_instance`。
+
+
+### 外部接口
+
+```c
+void CANRegister(can_instance* instance, can_instance_config config);
+void CANTransmit(can_instance* _instance);
+```
+
+`CANRegister`是用于初始化CAN实例的接口,module层的模块对象(也应当为一个结构体)内要包含一个`usart_instance`。调用时传入实例指针,以及用于初始化的config。`CANRegister`应当在module的初始化函数内被调用,推荐config采用以下的方式定义,更加直观明了:
+
+```c
+can_instance_config config={.can_handle=&hcan1,
+ .tx_id=0x005,
+ .rx_id=0x200,
+ can_module_callback=MotorCallback}
+```
+
+`CANTransmit()`是通过模块通过其拥有的CAN实例发送数据的接口,调用时传入对应的instance。在发送之前,应当给instance内的`send_buff`赋值。
+
+### 私有函数和变量
+
+在.c文件内设为static的函数和变量
+
+```c
+static can_instance *instance[MX_REGISTER_DEVICE_CNT]={NULL};
+```
+
+这是bsp层管理所有CAN实例的入口。
+
+```c
+static void CANServiceInit()
+static void CANAddFilter(can_instance *_instance)
+static void CANFIFOxCallback(CAN_HandleTypeDef *_hcan, uint32_t fifox)
+void HAL_CAN_RxFifo0MsgPendingCallback(CAN_HandleTypeDef *hcan)
+void HAL_CAN_RxFifo1MsgPendingCallback(CAN_HandleTypeDef *hcan)
+```
+
+- `CANServiceInit()`会被`CANRegister()`调用,对CAN外设进行硬件初始化并开启接收中断和消息提醒。
+
+- `CANAddFilter()`在每次使用`CANRegister()`的时候被调用,用于给当前注册的实例添加过滤器规则并设定处理对应`rx_id`的接收FIFO。过滤器的作用是减小CAN收发器的压力,只接收符合过滤器规则的报文(否则不会产生接收中断)。
+
+- `HAL_CAN_RxFifo0MsgPendingCallback()`和`HAL_CAN_RxFifo1MsgPendingCallback()`都是对HAL的CAN回调函数的重定义(原本的callback是`__week`修饰的弱定义),当发生FIFO0或FIFO1有新消息到达的时候,对应的callback会被调用。`CANFIFOxCallback()`随后被前两者调用,并根据接收id和硬件中断来源(哪一个CAN硬件,CAN1还是CAN2)调用对应的instance的回调函数进行协议解析。
\ No newline at end of file
diff --git a/bsp/bsp_usart.c b/bsp/bsp_usart.c
index 030ab81..765bd20 100644
--- a/bsp/bsp_usart.c
+++ b/bsp/bsp_usart.c
@@ -13,7 +13,7 @@
/* usart service instance, modules' info would be recoreded here using USARTRegister() */
/* usart服务实例,所有注册了usart的模块信息会被保存在这里 */
-static usart_instance *instance[DEVICE_USART_CNT];
+static usart_instance *instance[DEVICE_USART_CNT]={NULL};
/**
* @brief usart service will start automatically, after each module registered
diff --git a/bsp/bsp_usart.md b/bsp/bsp_usart.md
new file mode 100644
index 0000000..da7f98a
--- /dev/null
+++ b/bsp/bsp_usart.md
@@ -0,0 +1,69 @@
+# bsp_usart
+
+neozng1@hnu.edu.cn
+
+> TODO:为初始化定义一个结构体`usart_init_config`用于保存初始化所需的参数从而避免单独赋值,使得整体风格统一。
+
+## 代码结构
+
+.h文件内包括了外部接口和类型定义,以及模块对应的宏。c文件内为私有函数和外部接口的定义。
+
+### 类型定义
+
+```c
+#define DEVICE_USART_CNT 3 // C板至多分配3个串口
+#define USART_RXBUFF_LIMIT 128 // if your protocol needs bigger buff, modify here
+
+typedef void (*usart_module_callback)();
+
+/* usart_instance struct,each app would have one instance */
+typedef struct
+{
+ uint8_t recv_buff[USART_RXBUFF_LIMIT]; // 预先定义的最大buff大小,如果太小请修改USART_RXBUFF_LIMIT
+ uint8_t recv_buff_size; // 模块接收一包数据的大小
+ UART_HandleTypeDef *usart_handle; // 实例对应的usart_handle
+ usart_module_callback module_callback; // 解析收到的数据的回调函数
+} usart_instance;
+```
+
+- `DEVICE_USART_CNT`是开发板上可用的串口数量。
+
+- `USART_RXBUFF_LIMIT`是串口单次接收的数据长度上限,暂时设为128,如果需要更大的buffer容量,修改该值。
+
+- `usart_module_callback()`是模块提供给串口接收中断回调函数使用的协议解析函数指针。对于每个需要串口的模块,需要定义一个这样的函数用于解包数据。
+
+- 每定义一个`usart_instance`,就代表一个串口的**实例**(对象)。一个串口实例内有接收buffer,单个数据包的大小,该串口对应的`HAL handle`(代表其使用的串口硬件具体是哪一个)以及用于解包数据的回调函数。
+
+
+### 外部接口
+
+```c
+void USARTRegister(usart_instance *_instance);
+void USARTSend(usart_instance *_instance, uint8_t *send_buf, uint16_t send_size);
+```
+
+- `USARTRegister`是用于初始化串口对象的接口,module层的模块对象(也应当为一个结构体)内要包含一个`usart_instance`。
+
+ **在调用该函数之前,需要先对其成员变量`*usart_handle`,`module_callback()`以及`recv_buff_size`进行赋值。**
+
+- `USARTSend()`是通过模块通过其拥有的串口对象发送数据的接口,调用时传入的参数为串口实例指针,发送缓存以及此次要发送的数据长度(8-bit\*n)。
+
+### 私有函数和变量
+
+在.c文件内设为static的函数和变量
+
+```c
+static usart_instance *instance[DEVICE_USART_CNT];
+```
+
+这是bsp层管理所有串口实例的入口。
+
+```c
+static void USARTServiceInit(usart_instance *_instance)
+void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
+void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart)
+```
+
+- `USARTServiceInit()`会被`USARTRegister()`调用,开启接收中断
+
+- `HAL_UARTEx_RxEventCallback()`和`HAL_UART_ErrorCallback()`都是对HAL的回调函数的重定义(原本的callback是`__week`修饰的弱定义),前者在发生**IDLE中断**或**单次DMA传输中断**后会被调用(说明收到了完整的一包数据),随后在里面根据中断来源,调用拥有产生了该中断的模块的协议解析函数进行数据解包;后者在串口传输出错的时候会被调用,重新开启接收。
\ No newline at end of file
diff --git a/modules/algorithm/controller.h b/modules/algorithm/controller.h
index bf51b1f..578a735 100644
--- a/modules/algorithm/controller.h
+++ b/modules/algorithm/controller.h
@@ -26,7 +26,7 @@
#endif
// PID 优化环节使能标志位
-typedef enum pid_Improvement_e
+typedef enum
{
NONE = 0b00000000, // 0000 0000
Integral_Limit = 0b00000001, // 0000 0001
@@ -103,16 +103,15 @@ typedef struct
float Ki;
float Kd;
- float MaxOut;
- float IntegralLimit;
- float DeadBand;
+ float MaxOut; // 输出限幅
+ float IntegralLimit; // 积分限幅
+ float DeadBand; // 死区
float CoefA; // For Changing Integral
float CoefB; // ITerm = Err*((A-abs(err)+B)/A) when B<|err|neozng1@hnu.edu.cn
+
+> TODO:
+>
+> 1. 给不同的电机设置不同的低通滤波器惯性系数而不是统一使用宏
+> 2. 当前电机初始化函数`DJIMotorInit()`稍显凌乱,应设置一个`dji_motor_init_config_s`结构体用于电机初始化,使得风格统一,提高可读性
+
+
+
+## 总览和封装说明
+
+> 如果你不需要理解该模块的工作原理,你只需要查看这一小节。
+
+dji_motor模块对DJI智能电机,包括M2006,M3508以及GM6020进行了详尽的封装。你不再需要关心PID的计算以及CAN报文的发送和接收解析,你只需要专注于根据应用层的需求,设定合理的期望值,并通过`DJIMotorSetRef()`设置对应电机的输入参考即可。如果你希望更改电机的反馈来源,比如进入小陀螺模式(这时候你想要云台保持静止,使用IMU的yaw角度值作为反馈来源),只需要调用`DJIMotorChangeFeed()`,电机便可立刻切换反馈数据来源至IMU。
+
+要获得一个电机,请通过`DJIMotorInit()`并传入一些参数,他就会返回一个电机的指针。你也不再需要查看这些电机和电调的说明书,**只需要设置其电机id**(6020为拨码开关值,2006和3508为电调的闪动次数),该模块会自动为你计算CAN发送和接收ID并搞定所有硬件层的琐事。
+
+初始化电机时,你需要传入的参数包括:
+
+- 电机挂载的总线设置:CAN1 or CAN2,以及电机的id
+
+- 电机类型:
+
+ ```c
+ GM6020 = 0
+ M3508 = 1
+ M2006 = 2
+ ```
+
+- 闭环类型
+
+ ```c
+ OPEN_LOOP
+ CURRENT_LOOP
+ SPEED_LOOP
+ ANGLE_LOOP
+ CURRENT_LOOP| SPEED_LOOP // 同时对电流和速度闭环
+ SPEED_LOOP | ANGLE_LOOP // 同时对速度和位置闭环
+ CURRENT_LOOP| SPEED_LOOP |ANGLE_LOOP // 三环全开
+ ```
+
+- 是否反转
+
+ ```c
+ MOTOR_DIRECTION_NORMAL
+ MOTOR_DIRECTION_REVERSE
+ ```
+
+- 是否其他反馈来源,以及他们对应的数据指针(如果有的话)
+
+ ```c
+ MOTOR_FEED = 0
+ OTHER_FEED = 1
+ ---
+ float *other_angle_feedback_ptr
+ float *other_speed_feedback_ptr
+ // 电流只能从电机传感器获得所以无法设置其他来源
+ ```
+
+- 每个环的PID参数以及是否使用改进功能
+
+ ```c
+ typedef struct // config parameter
+ {
+ float Kp;
+ float Ki;
+ float Kd;
+
+ float MaxOut; // 输出限幅
+ // 以下是优化参数
+ float IntegralLimit; // 积分限幅
+ float DeadBand; // 死区
+ float CoefA; // For Changing Integral
+ float CoefB; // ITerm = Err*((A-abs(err)+B)/A) when B<|err|