wheel_legged/application/chassis/leg.c

255 lines
8.3 KiB
C

//
// Created by SJQ on 2023/12/25.
//
#include "leg.h"
#include "math.h"
#include "main.h"
#include "arm_math.h"
#include "cal_phi0_dot.h"
static const float l[5]={0.15f,0.27f,0.27f,0.15f,0.15f};
static void calculate_leg_pos(Leg_State_t* leg_state);
static void VMC_jacobian(float L0, float phi0, float phi1, float phi2, float phi3,
float phi4, float F, float Tp, float T12[2]);
void VMC_getT(LegInstance* legInstance);
void Leg_Init(LegInstance* legInstance, Leg_init_config_s* legInitConfig)
{
PIDInit(&legInstance->Length_PID,&legInitConfig->length_PID_conf);
PIDInit(&legInstance->Phi0_PID,&legInitConfig->phi0_PID_conf);
legInstance->target_L0 = legInitConfig->init_target_L0;
legInstance->F_feedforward = legInitConfig->F_feedforward;
arm_mat_init_f32(&legInstance->VMC_Jacobi,2,2,legInstance->VMC_Jacobi_data);
arm_mat_init_f32(&legInstance->inv_VMC_Jacobi,2,2,legInstance->inv_VMC_Jacobi_data);
// set rest of memory to 0
DWT_GetDeltaT(&legInstance->legState.DWT_CNT);
}
void Leg_length_control_loop(LegInstance* legInstance)
{
float F_pid_out = PIDCalculate(&legInstance->Length_PID,legInstance->legState.L0,legInstance->target_L0);
float Tp_pid_out = PIDCalculate(&legInstance->Phi0_PID,legInstance->legState.phi0,legInstance->target_phi0);
Leg_Input_update(&legInstance->legInput,F_pid_out-2.73f,Tp_pid_out);
//Leg_calc_output(&legInstance->legState,&legInstance->legInput,&legInstance->legOutput);
VMC_getT(legInstance);
}
void Leg_State_update(LegInstance* legInstance, float phi1,float phi4 ,float phi1_dot,float phi4_dot)
{
Leg_State_t* leg_state = &legInstance->legState;
if(phi1<=PI/2)
phi1 = PI/2;
if(phi4>=PI/2)
phi4 = PI/2;
leg_state->phi1 = phi1;
leg_state->phi4 = phi4;
float leg_pos[4];
//calculate_leg_pos(leg_state->phi1,leg_state->phi4,leg_pos);
calculate_leg_pos(leg_state);
//对phi0求导
// leg_state->dt = DWT_GetDeltaT(&leg_state->DWT_CNT);
// if(leg_state->last_phi0>=0)
// leg_state->phi0_dot = (leg_state->phi0-leg_state->last_phi0)/leg_state->dt;
// else
// leg_state->phi0_dot = 0;
// leg_state->last_phi0 = leg_state->phi0;
leg_state->phi0_dot = get_dphi0(phi1,phi4,phi1_dot,phi4_dot);
leg_state->L0_dot = get_dL0(phi1,phi4,phi1_dot,phi4_dot);
}
void Leg_Input_update(Leg_Input_t* leg_Input, float F,float Tp)
{
leg_Input->F = F;
leg_Input->Tp = Tp;
}
void Leg_calc_output(Leg_State_t* leg_state,Leg_Input_t* leg_input,Leg_Output_t* leg_output)
{
// float T12[2] = {0,0};
// vmcJacobian(leg_state->L0,leg_state->phi0,leg_state->phi1,
// leg_state->phi2,leg_state->phi3,leg_state->phi4,
// leg_input->F,leg_input->Tp,T12);
// leg_output->T1 = T12[0];
// leg_output->T2 = T12[1];
}
void calculate_leg_pos(Leg_State_t* leg_state)
{
//以机体中点建立坐标系
float xb = -l[4]/2+l[0]* arm_cos_f32(leg_state->phi1);
float yb = l[0]* arm_sin_f32(leg_state->phi1);
float xd = l[4]/2+l[3]* arm_cos_f32(leg_state->phi4);
float yd = l[3]* arm_sin_f32(leg_state->phi4);
float A0 = 2*l[1]*(xd-xb);
float B0 = 2*l[1]*(yd-yb);
float lbd2 = (xd-xb)*(xd-xb)+(yd-yb)*(yd-yb);
float C0 = l[1]*l[1]+lbd2-l[2]*l[2];
float tanPhi = 0;
arm_sqrt_f32((A0*A0 + B0*B0 - C0*C0),&tanPhi);
//tanPhi = (B0+tanPhi)/(A0+C0);
float phi2 = 2* atan2f(B0+tanPhi,A0+C0);
float xc = xb + l[1] * arm_cos_f32(phi2);
float yc = yb + l[1] * arm_sin_f32(phi2);
float phi3 = acosf(xc-xd/l[2]);
float L0 = 0;
arm_sqrt_f32((xc*xc + yc*yc),&L0);
float phi0 = atan2f(yc,xc);
//赋值
leg_state->L0 = L0;
leg_state->phi0 = phi0;
leg_state->phi2 = phi2;
leg_state->phi3 = phi3;
}
void VMC_getT(LegInstance* legInstance)
{
//arm_matrix_instance_f32 T_matrix;
arm_matrix_instance_f32 F_Tp_matrix;
// float T_array[4] = {0};
float F_Tp_array[2] = {0};
F_Tp_array[0] = legInstance->legInput.F;
F_Tp_array[1] = legInstance->legInput.Tp;
float sinPhi23 = arm_sin_f32(legInstance->legState.phi2 - legInstance->legState.phi3);
float sinPhi12 = arm_sin_f32(legInstance->legState.phi1 - legInstance->legState.phi2);
float sinPhi34 = arm_sin_f32(legInstance->legState.phi3 - legInstance->legState.phi4);
float sinPhi03 = arm_sin_f32(legInstance->legState.phi0 - legInstance->legState.phi3);
float sinPhi02 = arm_sin_f32(legInstance->legState.phi0 - legInstance->legState.phi2);
float cosPhi03 = arm_cos_f32(legInstance->legState.phi0 - legInstance->legState.phi3);
float cosPhi02 = arm_cos_f32(legInstance->legState.phi0 - legInstance->legState.phi2);
// T_array[0] = -(l[0]* sinPhi03*sinPhi12)/sinPhi23;
// T_array[1] = -(l[0]* cosPhi03*sinPhi12)/(legInstance->legState.L0 * sinPhi23);
//
// T_array[2] = -(l[3]* sinPhi02*sinPhi34)/sinPhi23;
// T_array[3] = -(l[3]* cosPhi02*sinPhi34)/(legInstance->legState.L0 * sinPhi23);
legInstance->VMC_Jacobi_data[0] = -(l[0]* sinPhi03*sinPhi12)/sinPhi23;
legInstance->VMC_Jacobi_data[1] = -(l[0]* cosPhi03*sinPhi12)/(legInstance->legState.L0 * sinPhi23);
legInstance->VMC_Jacobi_data[2] = -(l[3]* sinPhi02*sinPhi34)/sinPhi23;
legInstance->VMC_Jacobi_data[3] = -(l[3]* cosPhi02*sinPhi34)/(legInstance->legState.L0 * sinPhi23);
//arm_mat_init_f32(&T_matrix,2,2,T_array);
arm_mat_init_f32(&F_Tp_matrix,2,1,F_Tp_array);
arm_matrix_instance_f32 T12_matrix;
float T12[2] = {0};
arm_mat_init_f32(&T12_matrix,2,1,T12);
arm_mat_mult_f32(&legInstance->VMC_Jacobi,&F_Tp_matrix,&T12_matrix);
legInstance->legOutput.T1 = T12_matrix.pData[0];
legInstance->legOutput.T2 = T12_matrix.pData[1];
}
void Leg_feedback_update(LegInstance* legInstance,float T1_fb,float T2_fb)
{
arm_mat_inverse_f32(&legInstance->VMC_Jacobi,&legInstance->inv_VMC_Jacobi);
arm_matrix_instance_f32 F_Tp_matrix;
float F_Tp_array[2] = {0};
arm_mat_init_f32(&F_Tp_matrix,2,1,F_Tp_array);
arm_matrix_instance_f32 T12_matrix;
float T12[2] = {0};
arm_mat_init_f32(&T12_matrix,2,1,T12);
T12[0] = T1_fb; T12[1] = T2_fb;
arm_mat_mult_f32(&legInstance->inv_VMC_Jacobi,&T12_matrix,&F_Tp_matrix);
legInstance->legFeedback.F = F_Tp_array[0];
legInstance->legFeedback.Tp = F_Tp_array[1];
}
////此段代码为matlab生成
//static void calculate_leg_pos(const float phi1, const float phi4, float leg_pos[4])
//{
// float A0;
// float B0;
// float C0;
// float L0;
// float phi2;
// float scale;
// float xc;
// float xd;
// float yb_tmp;
// /* 以机体中点建立坐标系 */
// L0 = 0.15F * cosf(phi1);
// yb_tmp = 0.15F * sinf(phi1);
// xd = 0.15F * cosf(phi4) + 0.075F;
// A0 = 0.54F * (xd - (L0 - 0.075F));
// C0 = 0.15F * sinf(phi4) - yb_tmp;
// B0 = 0.54F * C0;
// scale = (0.15F * cosf(phi4) + 0.075F) - (L0 - 0.075F);
// C0 = ((scale * scale + C0 * C0) + 0.0729000047F) - 0.0729000047F;
// phi2 = 2.0F * atanf((B0 + sqrtf((A0 * A0 + B0 * B0) - C0 * C0)) / (A0 + C0));
// xc = L0 + 0.27F * cosf(phi2);
// C0 = yb_tmp + 0.27F * sinf(phi2);
// scale = 1.29246971E-26F;
// A0 = fabsf(xc);
// if (A0 > 1.29246971E-26F) {
// L0 = 1.0F;
// scale = A0;
// } else {
// B0 = A0 / 1.29246971E-26F;
// L0 = B0 * B0;
// }
// A0 = fabsf(C0);
// if (A0 > scale) {
// B0 = scale / A0;
// L0 = L0 * B0 * B0 + 1.0F;
// scale = A0;
// } else {
// B0 = A0 / scale;
// L0 += B0 * B0;
// }
// L0 = scale * sqrtf(L0);
// leg_pos[0] = L0;
// leg_pos[1] = atanf(C0 / xc);
// leg_pos[2] = phi2;
// leg_pos[3] = acosf(xc - xd / 0.27F);
//}
//
void VMC_jacobian(float L0, float phi0, float phi1, float phi2, float phi3,
float phi4, float F, float Tp, float T12[2])
{
float T_tmp;
float b_T_tmp;
float c_T_tmp;
float d_T_tmp;
float e_T_tmp;
float f_T_tmp;
T_tmp = phi0 - phi3;
b_T_tmp = arm_sin_f32(phi1 - phi2);
c_T_tmp = arm_sin_f32(phi2 - phi3);
d_T_tmp = phi0 - phi2;
e_T_tmp = arm_sin_f32(phi3 - phi4);
f_T_tmp = L0 * c_T_tmp;
T12[0] = -(0.15F * arm_sin_f32(T_tmp) * b_T_tmp) / c_T_tmp * F +
-(0.15F * arm_cos_f32(T_tmp) * b_T_tmp) / f_T_tmp * Tp;
T12[1] = -(0.15F * arm_sin_f32(d_T_tmp) * e_T_tmp) / c_T_tmp * F +
-(0.15F * arm_cos_f32(d_T_tmp) * e_T_tmp) / f_T_tmp * Tp;
}