Beispiel #1
0
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)
Beispiel #2
0
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)
Beispiel #3
0
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()