From a2afb2daa983a7142f6d13e9cfee91208e61cb02 Mon Sep 17 00:00:00 2001 From: raiden00pl Date: Mon, 1 Mar 2021 20:17:56 +0100 Subject: [PATCH] libdsp: add Permanent Magnet Synchronous Motor (PMSM) model --- include/dsp.h | 112 +++++++++++--- libs/libdsp/Makefile | 1 + libs/libdsp/lib_motor.c | 44 ++++++ libs/libdsp/lib_pmsm_model.c | 283 +++++++++++++++++++++++++++++++++++ 4 files changed, 421 insertions(+), 19 deletions(-) create mode 100644 libs/libdsp/lib_pmsm_model.c diff --git a/include/dsp.h b/include/dsp.h index 9e439715b3..c8d9ef2660 100644 --- a/include/dsp.h +++ b/include/dsp.h @@ -26,6 +26,7 @@ ****************************************************************************/ #include + #include #include #include @@ -312,22 +313,6 @@ struct motor_observer_smo_f32_s ab_frame_f32_t sign; /* Bang-bang controller sign */ }; -/* Motor physical parameters. - * This data structure was designed to work with BLDC/PMSM motors, - * but probably can be used to describe different types of motors. - */ - -struct motor_phy_params_f32_s -{ - uint8_t p; /* Number of the motor pole pairs */ - - float res; /* Phase-to-neutral temperature compensated - * resistance - */ - float ind; /* Average phase-to-neutral inductance */ - float one_by_ind; /* Inverse phase-to-neutral inductance */ -}; - /* FOC initialize data */ struct foc_initdata_f32_s @@ -367,6 +352,81 @@ struct foc_data_f32_s phase_angle_f32_t angle; /* Phase angle */ }; +/* Motor physical parameters. + * This data structure was designed to work with BLDC/PMSM motors, + * but probably can be used to describe different types of motors. + */ + +struct motor_phy_params_f32_s +{ + uint8_t p; /* Number of the motor pole pairs */ + + float res; /* Phase-to-neutral temperature compensated + * resistance + */ + float ind; /* Average phase-to-neutral inductance */ + float one_by_ind; /* Inverse phase-to-neutral inductance */ +}; + +/* PMSM motor physcial parameters */ + +struct pmsm_phy_params_f32_s +{ + struct motor_phy_params_f32_s motor; /* Motor common PHY */ + float iner; /* Rotor inertia */ + float flux_link; /* Flux linkage */ + float ind_d; /* d-inductance */ + float ind_q; /* q-inductance */ + float one_by_iner; /* One by intertia */ + float one_by_indd; /* One by Ld */ + float one_by_indq; /* One by Lq */ +}; + +/* PMSM motor model state */ + +struct pmsm_model_state_f32_s +{ + /* Motor model phase current */ + + abc_frame_f32_t i_abc; + ab_frame_f32_t i_ab; + dq_frame_f32_t i_dq; + + /* Motor model phase voltage */ + + abc_frame_f32_t v_abc; + ab_frame_f32_t v_ab; + dq_frame_f32_t v_dq; + + /* Motor model angle */ + + struct motor_angle_f32_s angle; + + /* Angular speed */ + + float omega_e; + float omega_m; +}; + +/* PMSM motor model external conditions */ + +struct pmsm_model_ext_f32_s +{ + float load; /* Motor model load torque */ +}; + +/* PMSM motor model */ + +struct pmsm_model_f32_s +{ + struct pmsm_phy_params_f32_s phy; /* Motor model physical parameters */ + struct pmsm_model_state_f32_s state; /* Motor model state */ + struct pmsm_model_ext_f32_s ext; /* Motor model external conditions */ + float per; /* Control period */ + float id_int; /* Id integral part */ + float iq_int; /* Iq integral part */ +}; + /**************************************************************************** * Public Functions Prototypes ****************************************************************************/ @@ -488,12 +548,26 @@ void motor_angle_m_update(FAR struct motor_angle_f32_s *angle, float motor_angle_m_get(FAR struct motor_angle_f32_s *angle); float motor_angle_e_get(FAR struct motor_angle_f32_s *angle); -/* Motor physical parameters functions */ +/* Motor physical parameters */ void motor_phy_params_init(FAR struct motor_phy_params_f32_s *phy, uint8_t poles, float res, float ind); -void motor_phy_params_temp_set(FAR struct motor_phy_params_f32_s *phy, - float res_alpha, float res_temp_ref); + +/* PMSM physical parameters functions */ + +void pmsm_phy_params_init(FAR struct pmsm_phy_params_f32_s *phy, + uint8_t poles, float res, float ind, + float iner, float flux, + float ind_d, float ind_q); + +/* PMSM motor model */ + +int pmsm_model_initialize(FAR struct pmsm_model_f32_s *model, + FAR struct pmsm_phy_params_f32_s *phy, + float per); +int pmsm_model_elec(FAR struct pmsm_model_f32_s *model, + FAR ab_frame_f32_t *vab); +int pmsm_model_mech(FAR struct pmsm_model_f32_s *model, float load); #undef EXTERN #if defined(__cplusplus) diff --git a/libs/libdsp/Makefile b/libs/libdsp/Makefile index e51ed78cbe..7b0ec47974 100644 --- a/libs/libdsp/Makefile +++ b/libs/libdsp/Makefile @@ -28,6 +28,7 @@ CSRCS += lib_observer.c CSRCS += lib_foc.c CSRCS += lib_misc.c CSRCS += lib_motor.c +CSRCS += lib_pmsm_model.c endif AOBJS = $(ASRCS:.S=$(OBJEXT)) diff --git a/libs/libdsp/lib_motor.c b/libs/libdsp/lib_motor.c index 2db21788bc..47c3cb9e7b 100644 --- a/libs/libdsp/lib_motor.c +++ b/libs/libdsp/lib_motor.c @@ -372,3 +372,47 @@ void motor_phy_params_init(FAR struct motor_phy_params_f32_s *phy, phy->ind = ind; phy->one_by_ind = (1.0f / ind); } + +/**************************************************************************** + * Name: pmsm_phy_params_init + * + * Description: + * Initialize PMSM physical parameters + * + * Input Parameters: + * phy - (in/out) pointer to the PMSM physical parameters + * poles - (in) number of the motor pole pairs + * res - (in) average phase-to-neutral base motor resistance + * (without temperature compensation) + * ind - (in) average phase-to-neutral motor inductance + * iner - (in) rotor inertia (J) + * flux - (in) flux linkage + * ind_d - (in) d-inductance + * ind_q - (in) q-inductance + * + * Returned Value: + * None + * + ****************************************************************************/ + +void pmsm_phy_params_init(FAR struct pmsm_phy_params_f32_s *phy, + uint8_t poles, float res, float ind, + float iner, float flux, + float ind_d, float ind_q) +{ + LIBDSP_DEBUGASSERT(phy != NULL); + + /* Initialize motor phy */ + + motor_phy_params_init(&phy->motor, poles, res, ind); + + /* Iniitalize PMSM specific data */ + + phy->iner = iner; + phy->flux_link = flux; + phy->ind_d = ind_d; + phy->ind_q = ind_q; + phy->one_by_iner = (1.0f / iner); + phy->one_by_indd = (1.0f / ind_d); + phy->one_by_indq = (1.0f / ind_q); +} diff --git a/libs/libdsp/lib_pmsm_model.c b/libs/libdsp/lib_pmsm_model.c new file mode 100644 index 0000000000..839e79900f --- /dev/null +++ b/libs/libdsp/lib_pmsm_model.c @@ -0,0 +1,283 @@ +/**************************************************************************** + * libs/libdsp/lib_pmsm_model.c + * + * Licensed to the Apache Software Foundation (ASF) under one or more + * contributor license agreements. See the NOTICE file distributed with + * this work for additional information regarding copyright ownership. The + * ASF licenses this file to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance with the + * License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + ****************************************************************************/ + +/* Permanent magnet synchronous motor model (PMSM) + * + * 1) Flux equation: + * + * lambda_q = L_q * i_q + * lambda_d = L_d * i_d + lampda_m + * + * 2) Voltage equation: + * + * a) D-part: + * + * id Rs Ld + * o-->----/\/\/\/\----mmmmmm----+ + * ^ | + * . | + * . .-----. + * . | ^ | + * vd . | | | + * . .-----. + * . | ed + * . | + * o-----------------------------+ + * + * ed = -we * lamda_q + * ed = -we * (Lq * iq) + * + * vd = (Rs * id) + (d/dt * (Ld * id)) - (Lq * we * iq) + * Ld * (d/dt * id) = vd - (Rs * id) + (we * Lq * iq) + * (d/dt * id) = (vd - (Rs * id) + (we * Lq * iq)) / Ld + * + * b) Q-part: + * + * iq Rs Lq + * o-->----/\/\/\/\----mmmmmm----+ + * ^ | + * . | + * . .-----. + * . | ^ | + * vq . | | | + * . .-----. + * . | eq + * . | + * o-----------------------------+ + * + * eq = we * lamda_d + * eq = we * (Lq * iq + lamda_m) + * + * vq = (Rs * iq) + (d/dt * (Lq * iq)) + (we * (Ld * id + lambda_m)) + * Lq * (d/dt * iq) = vq - (Rs * iq) - (we * (Ld * id + lambda_m)) + * (d/dt * iq) = (vq - (Rs * iq) - (we * (Ld * id + lambda_m))) / Lq + * + * 3) Torque equation: + * + * Te = (3/2) * p * (lambda_d * i_q - lambda_q * i_d) + * Te = (3/2) * p * (lambda_m * i_q + (L_d - L_q) * i_q * i_d) + * Te = (3/2) * p * i_q * (lambda_m + (L_d - L_q) * i_d) + * + * 4) Electromechanical power equation: + * + * Pem = wm * Te + * Pem = (3/2) * wm * (lamda_d * i_q - lamda_q * i_d) + * + * 5) The general mechanical equation for the motor: + * + * Te = Tl + Td + B * wm + J * (d/dt) * wm + * we = p * wm = (P/2) * wm + * p = (P/2) + * + * assume no friction: + * + * Te = Tl + J * (d/dt) * wm + * (d/dt) * wm = (Te - Tl) / J + * + * where: + * B - viscous frictions coefficient + * J - interia of the shaft and the load system + * Td - dry friction + * Tl - load torque + * Te - electromagnetic torque + * Pe - electromagnetical power + * we - electrical velociti of the motor + * wm - mechanical velocity of the rotor + * lambda_m - flux linkage + * P - Number of poles + * p - pole pairs + */ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include + +#include +#include + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: pmsm_model_initialize + * + * Description: + * Initialzie FOC model + * + ****************************************************************************/ + +int pmsm_model_initialize(FAR struct pmsm_model_f32_s *model, + FAR struct pmsm_phy_params_f32_s *phy, + float per) +{ + DEBUGASSERT(model); + DEBUGASSERT(phy); + DEBUGASSERT(per > 0.0f); + + /* Copy motor model parameters */ + + memcpy(&model->phy, phy, sizeof(struct pmsm_phy_params_f32_s)); + + /* Initialize controller period */ + + model->per = per; + + return OK; +} + +/**************************************************************************** + * Name: pmsm_model_elec + * + * Description: + * Update motor model electrical state + * + ****************************************************************************/ + +int pmsm_model_elec(FAR struct pmsm_model_f32_s *model, + FAR ab_frame_f32_t *vab) +{ + float tmp1 = 0.0f; + float tmp2 = 0.0f; + float tmp3 = 0.0f; + float tmp4 = 0.0f; + float tmp5 = 0.0f; + float tmp6 = 0.0f; + + DEBUGASSERT(model); + DEBUGASSERT(vab); + + /* Copy alpha-beta voltage */ + + model->state.v_ab.a = vab->a; + model->state.v_ab.b = vab->b; + + /* Inverse Clarke transform - get abc voltage */ + + inv_clarke_transform(&model->state.v_ab, + &model->state.v_abc); + + /* Park transform - get DQ voltage */ + + park_transform(&model->state.angle.angle_el, + &model->state.v_ab, + &model->state.v_dq); + + /* q current */ + + tmp1 = model->phy.motor.res * model->state.i_dq.q; + tmp2 = model->phy.ind_d * model->state.i_dq.d; + tmp3 = tmp2 + model->phy.flux_link; + tmp4 = model->state.omega_e * tmp3; + tmp5 = model->state.v_dq.q - tmp1 - tmp4; + tmp6 = model->per * tmp5; + + model->iq_int += (tmp6 * model->phy.one_by_indq); + + /* d current */ + + tmp1 = model->phy.motor.res * model->state.i_dq.d; + tmp2 = model->phy.ind_q * model->state.i_dq.q; + tmp3 = tmp2 * model->state.omega_e; + tmp4 = model->state.v_dq.d - tmp1 + tmp3; + tmp5 = model->per * tmp4; + + model->id_int += (tmp5 * model->phy.one_by_indd); + + /* Store currents */ + + model->state.i_dq.q = model->iq_int; + model->state.i_dq.d = model->id_int; + + /* Inverse Park transform - get alpha-beta current */ + + inv_park_transform(&model->state.angle.angle_el, + &model->state.i_dq, + &model->state.i_ab); + + /* Inverse Clarke transform - get abc current */ + + inv_clarke_transform(&model->state.i_ab, + &model->state.i_abc); + + return OK; +} + +/**************************************************************************** + * Name: pmsm_model_mech + * + * Description: + * Update motor model mechanical state + * + ****************************************************************************/ + +int pmsm_model_mech(FAR struct pmsm_model_f32_s *model, float load) +{ + float angle = 0.0f; + float dir = 0.0f; + float te = 0.0f; + float tmp1 = 0.0f; + float tmp2 = 0.0f; + float tmp3 = 0.0f; + float tmp4 = 0.0f; + float tmp5 = 0.0f; + + DEBUGASSERT(model); + + /* Get electrical torque developed by the motor */ + + tmp1 = model->phy.ind_d - model->phy.ind_q; + tmp2 = tmp1 * model->state.i_dq.d; + tmp3 = model->phy.flux_link - tmp2; + tmp4 = 1.5f * model->phy.motor.p; + tmp5 = tmp4 * model->state.i_dq.q; + + te = tmp5 * tmp3; + + /* Get new mechanical velocity */ + + tmp1 = te - load; + tmp2 = model->per * tmp1 ; + tmp3 = tmp2 * model->phy.one_by_iner; + + model->state.omega_m = (model->state.omega_m + tmp3); + + /* Get new electrical velocity */ + + model->state.omega_e = model->state.omega_m * model->phy.motor.p; + + /* Get rotation direction */ + + dir = (model->state.omega_e > 0 ? DIR_CW : DIR_CCW); + + /* Update electrical angle */ + + tmp1 = model->state.omega_e * model->per; + + angle = model->state.angle.angle_el.angle + tmp1; + + /* Update with mechanical angel */ + + motor_angle_e_update(&model->state.angle, angle, dir); + + return OK; +}