basic_framework/application/shoot/shoot.c

243 lines
10 KiB
C
Raw Permalink Normal View History

2023-10-02 18:00:54 +08:00
#include "shoot.h"
#include "robot_def.h"
#include "dji_motor.h"
#include "message_center.h"
#include "bsp_dwt.h"
2024-03-14 15:41:33 +08:00
#include "referee_task.h"
2023-10-02 18:00:54 +08:00
#include "general_def.h"
#include "servo_motor.h"
2023-10-02 18:00:54 +08:00
2024-01-16 19:51:18 +08:00
2023-10-02 18:00:54 +08:00
/* 对于双发射机构的机器人,将下面的数据封装成结构体即可,生成两份shoot应用实例 */
static DJIMotorInstance *friction_l, *friction_r,*friction_z, *loader; // 拨盘电机
static ServoInstance *telescope; //需要增加弹舱盖
2023-10-02 18:00:54 +08:00
static Publisher_t *shoot_pub;
static Shoot_Ctrl_Cmd_s shoot_cmd_recv; // 来自cmd的发射控制信息
static Subscriber_t *shoot_sub;
static Shoot_Upload_Data_s shoot_feedback_data; // 来自cmd的发射控制信息
float Kp=2;
float Ki=1;
float Kd=0;
static float stop_location;
2023-10-02 18:00:54 +08:00
// dwt定时,计算冷却用
static float hibernate_time = 0, dead_time = 0;
uint16_t angle=2;
2024-01-26 17:25:52 +08:00
static uint16_t locked_time;
2023-10-02 18:00:54 +08:00
void ShootInit()
{
// 摩擦轮
2023-10-02 18:00:54 +08:00
Motor_Init_Config_s friction_config = {
.can_init_config = {
.can_handle = &hcan2,
},
.controller_param_init_config = {
.speed_PID = {
.Kp = 6, // 20
.Ki = 0, // 1
2023-10-02 18:00:54 +08:00
.Kd = 0,
.Improve = PID_Integral_Limit,
.IntegralLimit = 10000,
.MaxOut = 28000,
2023-10-02 18:00:54 +08:00
},
.current_PID = {
.Kp = 0, // 0.7
.Ki = 0, // 0.1
2023-10-02 18:00:54 +08:00
.Kd = 0,
.Improve = PID_Integral_Limit,
.IntegralLimit = 10000,
.MaxOut = 15000,
},
},
.controller_setting_init_config = {
.angle_feedback_source = MOTOR_FEED,
.speed_feedback_source = MOTOR_FEED,
.outer_loop_type = SPEED_LOOP,
.close_loop_type = SPEED_LOOP ,
2024-04-17 15:26:00 +08:00
.motor_reverse_flag = MOTOR_DIRECTION_REVERSE,
2023-10-02 18:00:54 +08:00
},
.motor_type = M3508};
2024-04-17 15:26:00 +08:00
// friction_config.can_init_config.tx_id = 3,
// friction_l = DJIMotorInit(&friction_config);
2023-10-02 18:00:54 +08:00
2024-04-17 15:26:00 +08:00
friction_config.can_init_config.tx_id = 4; // 右摩擦轮,改txid和方向就行
friction_config.controller_setting_init_config.motor_reverse_flag = MOTOR_DIRECTION_NORMAL;
friction_r = DJIMotorInit(&friction_config);
2023-10-02 18:00:54 +08:00
2024-02-29 20:39:18 +08:00
friction_config.can_init_config.tx_id = 6; // 上摩擦轮,改txid和方向就行
2024-03-09 22:06:29 +08:00
friction_config.controller_setting_init_config.motor_reverse_flag = MOTOR_DIRECTION_REVERSE;
2024-02-29 20:39:18 +08:00
friction_z = DJIMotorInit(&friction_config);
2023-10-02 18:00:54 +08:00
// 拨盘电机
Motor_Init_Config_s loader_config = {
.can_init_config = {
2024-01-16 19:51:18 +08:00
.can_handle = &hcan1,
.tx_id = 5,
2023-10-02 18:00:54 +08:00
},
.controller_param_init_config = {
.angle_PID = {
// 如果启用位置环来控制发弹,需要较大的I值保证输出力矩的线性度否则出现接近拨出的力矩大幅下降
.Kp = 70, // 10
2024-01-26 17:25:52 +08:00
.Ki = 0,
.Kd = 0.01f,
2024-01-26 17:25:52 +08:00
.MaxOut = 15000,
2023-10-02 18:00:54 +08:00
},
.speed_PID = {
.Kp = 3.8f, // 10
.Ki = 0, // 1
2023-10-02 18:00:54 +08:00
.Kd = 0,
.Improve = PID_Integral_Limit,
.IntegralLimit = 5000,
2024-04-03 01:12:50 +08:00
.MaxOut = 30000,
2023-10-02 18:00:54 +08:00
},
.current_PID = {
2024-03-09 22:06:29 +08:00
.Kp = 0, // 0.7
2024-01-26 17:25:52 +08:00
.Ki = 0, // 0.1
2023-10-02 18:00:54 +08:00
.Kd = 0,
.Improve = PID_Integral_Limit,
.IntegralLimit = 5000,
2024-01-16 19:51:18 +08:00
.MaxOut = 15000,
2023-10-02 18:00:54 +08:00
},
},
.controller_setting_init_config = {
2024-01-16 19:51:18 +08:00
.angle_feedback_source = MOTOR_FEED,
.speed_feedback_source = MOTOR_FEED,
2023-10-02 18:00:54 +08:00
.outer_loop_type = SPEED_LOOP, // 初始化成SPEED_LOOP,让拨盘停在原地,防止拨盘上电时乱转
.close_loop_type = SPEED_LOOP | ANGLE_LOOP,
2023-10-02 18:00:54 +08:00
.motor_reverse_flag = MOTOR_DIRECTION_NORMAL, // 注意方向设置为拨盘的拨出的击发方向
},
2024-01-16 19:51:18 +08:00
.motor_type = M3508 // 英雄使用m3508
2023-10-02 18:00:54 +08:00
};
loader = DJIMotorInit(&loader_config);
// 倍镜舵机
Servo_Init_Config_s tele_config = {
.htim = &htim1,
.Channel = TIM_CHANNEL_1,
.Servo_type = Servo270,
.Servo_Angle_Type = Free_Angle_mode,
};
telescope = ServoInit(&tele_config);
2023-10-02 18:00:54 +08:00
shoot_pub = PubRegister("shoot_feed", sizeof(Shoot_Upload_Data_s));
shoot_sub = SubRegister("shoot_cmd", sizeof(Shoot_Ctrl_Cmd_s));
}
static void Stop_Check()
{
if(loader->measure.total_angle < stop_location*0.9)
{
DJIMotorOuterLoop(loader, ANGLE_LOOP);
DJIMotorSetRef(loader,stop_location);
hibernate_time = DWT_GetTimeline_ms(); // 记录触发指令的时间
dead_time = 1000;
}
}
2023-10-02 18:00:54 +08:00
/* 机器人发射机构控制核心任务 */
void ShootTask()
{
// 从cmd获取控制数据
SubGetMessage(shoot_sub, &shoot_cmd_recv);
friction_z->motor_controller.speed_PID.Kp=friction_r->motor_controller.speed_PID.Kp = Kp;
friction_z->motor_controller.speed_PID.Ki=friction_r->motor_controller.speed_PID.Ki = Ki;
friction_z->motor_controller.speed_PID.Kd=friction_r->motor_controller.speed_PID.Kd = Kd;
2023-10-02 18:00:54 +08:00
// 对shoot mode等于SHOOT_STOP的情况特殊处理,直接停止所有电机(紧急停止)
if (shoot_cmd_recv.shoot_mode == SHOOT_OFF)
{
2024-04-17 15:26:00 +08:00
// DJIMotorStop(friction_l);
DJIMotorStop(friction_r);
2024-02-29 20:39:18 +08:00
DJIMotorStop(friction_z);
2023-10-02 18:00:54 +08:00
DJIMotorStop(loader);
}
else // 恢复运行
{
2024-04-17 15:26:00 +08:00
// DJIMotorEnable(friction_l);
DJIMotorEnable(friction_r);
2024-02-29 20:39:18 +08:00
DJIMotorEnable(friction_z);
2023-10-02 18:00:54 +08:00
DJIMotorEnable(loader);
}
// 如果上一次触发单发或3发指令的时间加上不应期仍然大于当前时间(尚未休眠完毕),直接返回即可
// 单发模式主要提供给能量机关激活使用(以及英雄的射击大部分处于单发)
2024-01-16 19:51:18 +08:00
if (hibernate_time + dead_time > DWT_GetTimeline_ms())
return;
2023-10-02 18:00:54 +08:00
// 若不在休眠状态,根据robotCMD传来的控制模式进行拨盘电机参考值设定和模式切换
switch (shoot_cmd_recv.load_mode)
{
// 停止拨盘
case LOAD_STOP:
DJIMotorOuterLoop(loader, SPEED_LOOP); // 切换到速度环
DJIMotorSetRef(loader, 0); // 同时设定参考值为0,这样停止的速度最快
2023-10-02 18:00:54 +08:00
break;
// 单发模式,根据鼠标按下的时间,触发一次之后需要进入不响应输入的状态(否则按下的时间内可能多次进入,导致多次发射)
case LOAD_1_BULLET: // 激活能量机关/干扰对方用,英雄用.
2024-04-17 15:26:00 +08:00
DJIMotorOuterLoop(loader, ANGLE_LOOP); // 切换到角度环v
2023-10-02 18:00:54 +08:00
DJIMotorSetRef(loader, loader->measure.total_angle + ONE_BULLET_DELTA_ANGLE); // 控制量增加一发弹丸的角度
2024-01-16 19:51:18 +08:00
hibernate_time = DWT_GetTimeline_ms(); // 记录触发指令的时间
2024-04-03 01:12:50 +08:00
dead_time = shoot_cmd_recv.shoot_rate; // 完成1发弹丸发射的时间
2023-10-02 18:00:54 +08:00
break;
2024-04-03 01:12:50 +08:00
// 自爆连发
2023-10-02 18:00:54 +08:00
case LOAD_3_BULLET:
DJIMotorOuterLoop(loader, ANGLE_LOOP); // 切换到速度环
DJIMotorSetRef(loader, loader->measure.total_angle + 3 * ONE_BULLET_DELTA_ANGLE); // 增加3发
hibernate_time = DWT_GetTimeline_ms(); // 记录触发指令的时间
dead_time = 300; // 完成3发弹丸发射的时间
break;
// 连发模式,对速度闭环,射频后续修改为可变,目前固定为1Hz
case LOAD_BURSTFIRE:
DJIMotorOuterLoop(loader, SPEED_LOOP);
DJIMotorSetRef(loader, shoot_cmd_recv.shoot_rate * 360 * REDUCTION_RATIO_LOADER);
2024-01-16 19:51:18 +08:00
// x颗/秒换算成速度: 已知一圈的载弹量,由此计算出1s需要转的角度,注意换算角速度(DJIMotor的速度单位是)
2023-10-02 18:00:54 +08:00
break;
// 拨盘反转,对速度闭环,后续增加卡弹检测(通过裁判系统剩余热量反馈和电机电流)
// 也有可能需要从switch-case中独立出来
case LOAD_REVERSE:
2024-04-03 01:12:50 +08:00
DJIMotorOuterLoop(loader, ANGLE_LOOP); // 切换到角度环
DJIMotorSetRef(loader, loader->measure.total_angle - ONE_BULLET_DELTA_ANGLE); // 控制量增加一发弹丸的角度
hibernate_time = DWT_GetTimeline_ms(); // 记录触发指令的时间
dead_time = 300; // 完成1发弹丸发射的时间
break;
2023-10-02 18:00:54 +08:00
break;
default:
while (1)
; // 未知模式,停止运行,检查指针越界,内存溢出等问题
2024-01-26 17:25:52 +08:00
2023-10-02 18:00:54 +08:00
}
// 确定是否开启摩擦轮,后续可能修改为键鼠模式下始终开启摩擦轮(上场时建议一直开启)
if (shoot_cmd_recv.friction_mode == FRICTION_ON)
{
shoot_cmd_recv.motor_speed = LIMIT_MIN_MAX(shoot_cmd_recv.motor_speed,4000,8500);
DJIMotorSetRef(friction_r, (float)(6*shoot_cmd_recv.motor_speed));
DJIMotorSetRef(friction_z, (float)(6*shoot_cmd_recv.motor_speed));
// DJIMotorSetRef(friction_l, (float)(6*shoot_cmd_recv.motor_speed));
2023-10-02 18:00:54 +08:00
}
else // 关闭摩擦轮
{
2024-04-17 15:26:00 +08:00
// DJIMotorSetRef(friction_l, 0);
DJIMotorSetRef(friction_r, 0);
2024-02-29 20:39:18 +08:00
DJIMotorSetRef(friction_z, 0);
2023-10-02 18:00:54 +08:00
}
// 开关弹舱盖
if (shoot_cmd_recv.tele_mode == TELE_CLOSE)
2023-10-02 18:00:54 +08:00
{
Servo_Motor_FreeAngle_Set(telescope,angle);
2023-10-02 18:00:54 +08:00
}
else if (shoot_cmd_recv.tele_mode == TELE_OPEN)
2023-10-02 18:00:54 +08:00
{
Servo_Motor_FreeAngle_Set(telescope,60);
2023-10-02 18:00:54 +08:00
}
// 反馈数据,目前暂时没有要设定的反馈数据,后续可能增加应用离线监测以及卡弹反馈
shoot_feedback_data.Motor_Temperature = friction_r->measure.temperature;
2024-01-26 17:25:52 +08:00
//推送消息
2023-10-02 18:00:54 +08:00
PubPushMessage(shoot_pub, (void *)&shoot_feedback_data);
2024-01-16 19:51:18 +08:00
2023-10-02 18:00:54 +08:00
}