def c2v_design(R, L, n_pp, J_s, KE, B=0, CLBW_Hz=1000, CL_TS=1 / 20e3, fignum=5): currentKp, currentKi = get_coeffs_dc_motor_current_regulator(R, L, CLBW_Hz) currentKiCode = currentKi * currentKp * CL_TS if True: # 这里打印的用于实验中CCS的debug窗口检查电流环PI系数 上位机电流KP = CLBW_Hz 上位机电流KI = 1000 iSMC_currentKp = 上位机电流KP * L * 2 * np.pi iSMC_currentKi = 上位机电流KI / 1000 * R / L iSMC_currentKiCode = iSMC_currentKi * CL_TS * iSMC_currentKp print(f'\tiSMC_currentKp={iSMC_currentKp:g}, \ iSMC_currentKi={iSMC_currentKi:g}, \ iSMC_currentKiCode={iSMC_currentKiCode:g}') print(f'\tSimC_currentKp={currentKp:g}, \ SimC_currentKi={currentKi:g}, \ SimC_currentKiCode={currentKiCode:g}') print(f'\t上位机电流KP={上位机电流KP:g}, \ 上位机电流KI={上位机电流KI:g}') Gi_closed = control.tf( [1], [L / currentKp, 1]) # current loop zero-pole cancelled already currentBandwidth_radPerSec = currentKp / L KT = 1.5 * n_pp * KE dc_motor_motion = control.tf([KT], [J_s / n_pp, B]) # [Apk] to [elec.rad/s] print(dc_motor_motion) # quit() # 注意,我们研究的开环传递函数是以电流给定为输入的,而不是以转速控制误差为输入,这样仿真和实验容易实现一点。 # Gw_open = dc_motor_motion * Gi_closed * speedPI c2v_tf = dc_motor_motion * Gi_closed # fig5 = plt.figure(fignum) # plt.title('Designed Current Ref. to Velocity Meas. Transfer Function') mag, phase, omega = control.bode_plot(c2v_tf, 2 * np.pi * np.logspace(0, 4, 500), dB=1, Hz=1, deg=1, lw='0.5', label=f'{CLBW_Hz:g} Hz') open_cutoff_frequency_HZ = omega[(np.abs(mag - 0.0)).argmin()] / 2 / np.pi # print('\tCut-off frequency (without speed PI regulator):', open_cutoff_frequency_HZ, 'Hz') return (currentKp, currentKi), \ (上位机电流KP, 上位机电流KI), \ (mag, phase, omega)
def iterate_for_desired_bandwidth(delta, desired_VLBW_Hz, motor_dict, CLBW_Hz_initial=1000, CLBW_Hz_stepSize=100): # print('DEBUG tuner.py', motor_dict) R = motor_dict['Rs'] L = motor_dict['Ls'] J_s = motor_dict['J_s'] JLoadRatio = motor_dict['JLoadRatio'] n_pp = motor_dict['n_pp'] KE = motor_dict['KE'] CL_TS = motor_dict['CL_TS'] VL_TS = motor_dict['VL_TS'] J_total = J_s * (1 + JLoadRatio) CLBW_Hz = CLBW_Hz_initial #100 # Hz (initial) VLBW_Hz = -10 # Hz (initial) count = 0 while True: count += 1 if count > 20: msg = f'Loop count 20 is reached. Step size is {CLBW_Hz_stepSize} Hz.' print(msg) # raise Exception() break # Current loop (Tune its bandwidth to support required speed response) if abs(VLBW_Hz - desired_VLBW_Hz) <= 10: # Hz break else: if VLBW_Hz > desired_VLBW_Hz: CLBW_Hz -= CLBW_Hz_stepSize # Hz if CLBW_Hz <= 0: raise Exception( f'Negative CLBW_Hz. Maybe change the step size of "CLBW_Hz" ({CLBW_Hz_stepSize} Hz) and try again.' ) break else: CLBW_Hz += CLBW_Hz_stepSize # Hz # print(f'CLBW_Hz = {CLBW_Hz}') currentKp, currentKi = get_coeffs_dc_motor_current_regulator( R, L, CLBW_Hz) currentKiCode = currentKi * currentKp * CL_TS if True: # 这里打印的用于实验中CCS的debug窗口检查电流环PI系数 上位机电流KP = CLBW_Hz 上位机电流KI = 1000 iSMC_currentKp = 上位机电流KP * L * 2 * np.pi iSMC_currentKi = 上位机电流KI / 1000 * R / L iSMC_currentKiCode = iSMC_currentKi * CL_TS * iSMC_currentKp # print(f'\tiSMC_currentKp={iSMC_currentKp:g}, \ # iSMC_currentKi={iSMC_currentKi:g}, \ # iSMC_currentKiCode={iSMC_currentKiCode:g}') # print(f'\tSimC_currentKp={currentKp:g}, \ # SimC_currentKi={currentKi:g}, \ # SimC_currentKiCode={currentKiCode:g}') # print(f'\t上位机电流KP={上位机电流KP:g}, \ # 上位机电流KI={上位机电流KI:g}') Gi_closed = control.tf( [1], [L / currentKp, 1]) # current loop zero-pole cancelled already currentBandwidth_radPerSec = currentKp / L # Speed loop KT = 1.5 * n_pp * KE dc_motor_motion = control.tf([KT * n_pp / J_total], [1, 0]) speedKp, speedKi = get_coeffs_dc_motor_SPEED_regulator( J_total, n_pp, KE, delta, currentBandwidth_radPerSec) speedKiCode = speedKi * speedKp * VL_TS if True: # 这里打印的用于实验中TI的debug窗口检查系数 上位机速度KP, 上位机速度KI = 逆上位机速度PI系数转换CODE(speedKp, speedKiCode, VL_TS, J_total) iSMC_speedKp, iSMC_speedKi, iSMC_speedKiCode = 上位机速度PI系数转换CODE( 上位机速度KP, 上位机速度KI, VL_TS, J_total) # print(f'\tiSMC_speedKp={iSMC_speedKp:g}, \ # iSMC_speedKi={iSMC_speedKi:g}, \ # iSMC_speedKiCode={iSMC_speedKiCode:g}') # print(f'\tSimC_speedKp={speedKp:g}, \ # SimC_speedKi={speedKi:g}, \ # SimC_speedKiCode={speedKiCode:g}') # print(f'\t上位机速度KP={上位机速度KP:g}, \ # 上位机速度KI={上位机速度KI:g}') # 下面打印的用于仿真 # print(f'\tspeedKp = {speedKp:g}', f'speedKi = {speedKi:g}', \ # f'wzero = {speedKi/2/np.pi:g} Hz', \ # f'cutoff = {delta*speedKi/2/np.pi:g} Hz', \ # f'ipole = {currentKp/L/2/np.pi:g} Hz', sep=' | ') speedPI = control.tf([speedKp, speedKp * speedKi], [1, 0]) Gw_open = dc_motor_motion * Gi_closed * speedPI # C2C c2c_tf = Gi_closed mag, phase, omega = control.bode_plot(c2c_tf, 2 * np.pi * np.logspace(0, 4, 500), dB=1, Hz=1, deg=1, lw='0.5', label=f'{CLBW_Hz:g} Hz') CLBW_Hz = omega[(np.abs(mag - 0.707)).argmin()] / 2 / np.pi C2C_designedMagPhaseOmega = mag, phase, omega # C2V c2v_tf = dc_motor_motion * Gi_closed mag, phase, omega = control.bode_plot(c2v_tf, 2 * np.pi * np.logspace(0, 4, 500), dB=1, Hz=1, deg=1, lw='0.5', label=f'{CLBW_Hz:g} Hz') open_cutoff_frequency_HZ = omega[(np.abs(mag - 0.0)).argmin()] / 2 / np.pi C2V_designedMagPhaseOmega = mag, phase, omega # V2V Gw_closed = Gw_open / (1 + Gw_open) mag, phase, omega = control.bode_plot(Gw_closed, 2 * np.pi * np.logspace(0, 4, 500), dB=1, Hz=1, deg=1, lw='0.5', label=f'{delta:g}') VLBW_Hz = omega[(np.abs(mag - 0.707)).argmin()] / 2 / np.pi V2V_designedMagPhaseOmega = mag, phase, omega # print(Gw_closed) # print('\tSpeed loop bandwidth:', VLBW_Hz, 'Hz') return (currentKp, currentKi), \ (speedKp, speedKi), \ (上位机电流KP, 上位机电流KI), \ (上位机速度KP, 上位机速度KI), \ (C2C_designedMagPhaseOmega, C2V_designedMagPhaseOmega, V2V_designedMagPhaseOmega), \ (CLBW_Hz, VLBW_Hz, open_cutoff_frequency_HZ)
display(pi_regulator) # figure() # mag, phase, omega = bode_plot(dc_motor, 2*np.pi*np.logspace(-2,4,1000), dB=1, Hz=1, deg=1) # figure() # mag, phase, omega = bode_plot(pi_regulator, 2*np.pi*np.logspace(-2,4,1000), dB=1, Hz=1, deg=1) open_sys = control.series(pi_regulator, dc_motor) closed_sys = control.feedback(dc_motor, pi_regulator, sign=-1) # open_sys = pi_regulator * dc_motor closed_sys = open_sys / (1 + open_sys) display(open_sys) display(control.minreal(closed_sys)) closed_sys = control.minreal(closed_sys) # figure() # mag, phase, omega = bode_plot(open_sys, 2*np.pi*np.logspace(-2,4,1000), dB=1, Hz=1, deg=1) plt.figure() mag, phase, omega = bode_plot(closed_sys, 2 * np.pi * np.logspace(-2, 4, 1000), dB=1, Hz=1, deg=1) T, yout = control.step_response(closed_sys, np.arange(0, 2, 1e-4)) plt.figure() plt.plot(T, yout) plt.show()