255 lines
8.3 KiB
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;
|
|
}
|
|
|
|
|
|
|