OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 301|回复: 3

关于梯形加减速控制脉冲控制电机的运行频率疑惑

[复制链接]

4

主题

18

帖子

0

精华

初级会员

Rank: 2

积分
141
金钱
141
注册时间
2019-4-11
在线时间
36 小时
发表于 2024-9-4 17:37:37 | 显示全部楼层 |阅读模式
5金钱
问题描述:主控芯片为AT32F407,主频200Mhz,移植原子的电机控制程序。定时器计数频率为40Mhz。当电机处于均速,即最大速度时,比较向前的步进加载 motor->min_delay = 99。那么 脉冲频率为: 40Mhz/100*2) = 200khz。但是用示波器测得的实际频率为178khz。怀疑是中断函数处理时间过程,汇编调试为55步,时间为 55* 0.005us = 0.275us。 中断函数加变量err,记录最大时间为 12个定时器计数。时间 = 20*0.025us = 0.5us。
问题诉求:请大师指导是什么原因导致,实际频率为178khz。


int err = 0;

/**
  * @brief  定时器比较中断
  * @param  htim:定时器句柄指针
  * @note   无
  * @retval 无
  */
void HAL_TIM_OC_DelayElapsedCallback(MotorData *motor ) // 200M 20M min0.5uS max 0.7uS
{

    __IO uint32_t tim_count = 0;
    __IO uint32_t tmp = 0;

                //tim_count = tmr_counter_value_get(motor->tmr_x);
          
                //tmp = tim_count + motor->step_delay/2;               /* 整个C值里边是需要翻转两次的所以需要除以2 */
                //tmr_channel_value_set(motor->tmr_x,motor->pwm_ch,tmp);
                tmp =  motor->tmr_x->cval;
                motor->tmr_x->c4dt = motor->tmr_x->cval+motor->step_delay/2;

       
                motor->i++;                                                /* 定时器中断次数计数值 */
                if(motor->i == 2)                                          /* 2次,说明已经输出一个完整脉冲 */
                {
                        motor->i = 0;                                          /* 清零定时器中断次数计数值 */
                        switch(motor->run_state)                               /* 加减速曲线阶段 */
                        {
                                case STOP:
                                        motor->step_count = 0;                             /* 清零步数计数器 */
                                        motor->rest = 0;                                   /* 清零余值 */
                                        /* 关闭通道*/
                                        tmr_channel_enable(motor->tmr_x,motor->pwm_ch,FALSE);                                               
                                        tmr_counter_enable(motor->tmr_x, FALSE);
                                        //ST1_EN(EN_OFF);
                                        motor->g_motion_sta = FALSE;                           /* 电机为停止状态  */
                                        break;

                                case ACCEL:
                                        motor->g_add_pulse_count++;                        /* 只用于记录相对位置转动了多少度 */
                                        motor->step_count++;                               /* 步数加1*/
                                        if(motor->dir == CW)
                                        {
                                                        motor->g_step_position++;                      /* 绝对位置加1  记录绝对位置转动多少度*/
                                        }
                                        else
                                        {
                                                        motor->g_step_position--;                      /* 绝对位置减1*/
                                        }
                                        motor->accel_count++;                        /* 加速计数值加1*/
                                        motor->new_step_delay = motor->step_delay - (((2 *motor->step_delay) + motor->rest)/(4 * motor->accel_count + 1));/* 计算新(下)一步脉冲周期(时间间隔) */
                                        motor->rest = ((2 * motor->step_delay)+motor->rest)%(4 * motor->accel_count + 1);                                /* 计算余数,下次计算补上余数,减少误差 */
                                        if(motor->step_count >= motor->decel_start)         /* 检查是否到了需要减速的步数 */
                                        {
                                                        motor->accel_count = motor->decel_val;    /* 加速计数值为减速阶段计数值的初始值 */
                                                        motor->run_state = DECEL;                /* 下个脉冲进入减速阶段 */
                                        }
                                        else if(motor->new_step_delay <= motor->min_delay)  /* 检查是否到达期望的最大速度 计数值越小速度越快,当你的速度和最大速度相等或更快就进入匀速*/
                                        {
                                                        motor->last_accel_delay = motor->new_step_delay;      /* 保存加速过程中最后一次延时(脉冲周期)*/
                                                        motor->new_step_delay = motor->min_delay;       /* 使用min_delay(对应最大速度speed)*/
                                                        motor->rest = 0;                               /* 清零余值 */
                                                        motor->run_state = RUN;                  /* 设置为匀速运行状态 */
                                        }
                                        break;

                                case RUN:
                                        motor->g_add_pulse_count++;
                                        motor->step_count++;                               /* 步数加1 */
                                        if(motor->dir == CW)
                                        {
                                                        motor->g_step_position++;                      /* 绝对位置加1 */
                                        }
                                        else
                                        {
                                                        motor->g_step_position--;                      /* 绝对位置减1*/
                                        }
                                        motor->new_step_delay = motor->min_delay;           /* 使用min_delay(对应最大速度speed)*/
                                        if(motor->step_count >= motor->decel_start && !motor-> isContinuous)         /* 需要开始减速 */
                                        {
                                                        motor->accel_count = motor->decel_val;    /* 减速步数做为加速计数值 */
                                                        motor->new_step_delay = motor->last_accel_delay;      /* 加阶段最后的延时做为减速阶段的起始延时(脉冲周期) */
                                                        motor->run_state = DECEL;                /* 状态改变为减速 */
                                        }
                                        break;

                                case DECEL:
                                        motor->step_count++;                               /* 步数加1 */
                                        motor->g_add_pulse_count++;
                                        if(motor->dir == CW)
                                        {
                                                        motor->g_step_position++;                      /* 绝对位置加1 */
                                        }
                                        else
                                        {
                                                        motor->g_step_position--;                      /* 绝对位置减1 */
                                        }
                                        motor->accel_count++;
                                        motor->new_step_delay = motor->step_delay - (((2 * motor->step_delay) + motor->rest)/(4 * motor->accel_count + 1));  /* 计算新(下)一步脉冲周期(时间间隔) */
                                        motor->rest = ((2 * motor->step_delay)+motor->rest)%(4 * motor->accel_count + 1);                                   /* 计算余数,下次计算补上余数,减少误差 */

                                        /* 检查是否为最后一步 */
                                        if(motor->accel_count >= 0)                  /* 判断减速步数是否从负值加到0是的话 减速完成 */
                                        {
                                                motor->run_state = STOP;
                                       
                                                motor->step_count = 0;                             /* 清零步数计数器 */
                                                motor->rest = 0;                                   /* 清零余值 */
                                                /* 关闭通道*/
                                                tmr_channel_enable(motor->tmr_x,motor->pwm_ch,FALSE);                                               
                                                tmr_counter_enable(motor->tmr_x, FALSE);
                                                //ST1_EN(EN_OFF);
                                                motor->g_motion_sta = FALSE;                           /* 电机为停止状态  */                                                                                                                       
                                        }
                                        break;
                                }
                                motor->step_delay = motor->new_step_delay;              /* 为下个(新的)延时(脉冲周期)赋值 */
                                if(motor->homeParas.homeFlag)
                                {
                                        Homing(motor);
                                }
                }
                err =  (motor->tmr_x->cval - tmp)> err ? motor->tmr_x->cval - tmp:err;
               
}

屏幕截图 2024-09-04 173338.jpg

最佳答案

查看完整内容[请看2#楼]

原因是当响应中断后再读取定时器的计数值,此时定时器的值已经已经向前计数了一小段时间。将程序改为读取比较值,或者直接把比较值向上加,就是标准的频率。但要注意,频率不能高到小于中断程序执行时间。 结帖。
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

4

主题

18

帖子

0

精华

初级会员

Rank: 2

积分
141
金钱
141
注册时间
2019-4-11
在线时间
36 小时
 楼主| 发表于 2024-9-4 17:37:38 | 显示全部楼层
本帖最后由 huqinglun 于 2024-9-6 09:46 编辑

原因是当响应中断后再读取定时器的计数值,此时定时器的值已经已经向前计数了一小段时间。将程序改为读取比较值,或者直接把比较值向上加,就是标准的频率。但要注意,频率不能高到小于中断程序执行时间。
结帖。
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165309
金钱
165309
注册时间
2010-12-1
在线时间
2108 小时
发表于 2024-9-6 00:56:38 | 显示全部楼层
帮顶
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165309
金钱
165309
注册时间
2010-12-1
在线时间
2108 小时
发表于 2024-9-7 21:30:48 | 显示全部楼层
huqinglun 发表于 2024-9-6 09:44
原因是当响应中断后再读取定时器的计数值,此时定时器的值已经已经向前计数了一小段时间。将程序改为读取比 ...

谢谢分享
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2024-11-22 06:27

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表