AVR446_Linear speed control of stepper motor步进电机曲线分析

1.1.  单片机代码处理

// 定义定时器预分频,定时器实际时钟频率为:72MHz/(STEPMOTOR_TIMx_PRESCALER+1)

#define STEPMOTOR_TIM_PRESCALER          3  // 步进电机驱动器细分设置为:   32  细

// 定义定时器周期,输出比较模式周期设置为0xFFFF

#define STEPMOTOR_TIM_PERIOD                  0xFFFF

#define FALSE                                 0

#define TRUE                                  1

#define CW                                    0 // 顺时针

#define CCW                                   1 // 逆时针

#define STOP                                  0 // 加减速曲线状态:停止

#define ACCEL                                 1 // 加减速曲线状态:加速阶段

#define DECEL                                 2 // 加减速曲线状态:减速阶段

#define RUN                                   3 // 加减速曲线状态:匀速阶段

#define T1_FREQ                               (SystemCoreClock/(STEPMOTOR_TIM_PRESCALER+1)) // 频率ft值

#define FSPR                                  200         //步进电机单圈步数

#define MICRO_STEP                            32          // 步进电机驱动器细分数

#define SPR                                   (FSPR*MICRO_STEP)   // 旋转一圈需要的脉冲数

// 数学常数

#define ALPHA                                 ((float)(2*3.14159/SPR))       // α= 2*pi/spr//

#define A_T_x10                               ((float)(10*ALPHA*T1_FREQ))

#define T1_FREQ_148                           ((float)((T1_FREQ*0.676)/10)) // 0.676为误差修正值

#define A_SQ                                  ((float)(2*100000*ALPHA))

#define A_x200                                ((float)(200*ALPHA))

/**

* 函数功能: 相对位置运动:运动给定的步数

* 输入参数: step:移动的步数 (正数为顺时针,负数为逆时针).

accel  加速度,实际值为accel*0.1*rad/sec^2

decel  减速度,实际值为decel*0.1*rad/sec^2

speed  最大速度,实际值为speed*0.1*rad/sec

* 返 回 值: 无

* 说    明: 以给定的步数移动步进电机,先加速到最大速度,然后在合适位置开始

*           减速至停止,使得整个运动距离为指定的步数。如果加减速阶段很短并且

*           速度很慢,那还没达到最大速度就要开始减速

*/

void STEPMOTOR_AxisMoveRel(__IO int32_t step, __IO uint32_t accel, __IO uint32_t decel, __IO uint32_t speed)

{

__IO uint16_t tim_count;

// 达到最大速度时的步数

__IO uint32_t max_s_lim;

// 必须要开始减速的步数(如果加速没有达到最大速度)

__IO uint32_t accel_lim;

if(step < 0) // 步数为负数

{

srd.dir = CCW; // 逆时针方向旋转

STEPMOTOR_DIR_REVERSAL();

step =-step;   // 获取步数绝对值

}

else

{

srd.dir = CW; // 顺时针方向旋转

STEPMOTOR_DIR_FORWARD();

}

if(step == 1)    // 步数为1

{

srd.accel_count = -1;   // 只移动一步

srd.run_state = DECEL;  // 减速状态.

srd.step_delay = 1000;   // 短延时

}

else if(step != 0)  // 如果目标运动步数不为0

{

// 我们的驱动器用户手册有详细的计算及推导过程

// 设置最大速度极限, 计算得到min_delay用于定时器的计数器的值。

// min_delay = (alpha / tt)/ w

srd.min_delay = (int32_t)(A_T_x10/speed);

// 通过计算第一个(c0) 的步进延时来设定加速度,其中accel单位为0.1rad/sec^2

// step_delay = 1/tt * sqrt(2*alpha/accel)

// step_delay = ( tfreq*0.676/10 )*10 * sqrt( (2*alpha*100000) / (accel*10) )/100

srd.step_delay = (int32_t)((T1_FREQ_148 * sqrt(A_SQ / accel))/10);

// 计算多少步之后达到最大速度的限制

// max_s_lim = speed^2 / (2*alpha*accel)

max_s_lim = (uint32_t)(speed*speed/(A_x200*accel/10));

// 如果达到最大速度小于0.5步,我们将四舍五入为0

// 但实际我们必须移动至少一步才能达到想要的速度

if(max_s_lim == 0){

max_s_lim = 1;

}

// 计算多少步之后我们必须开始减速

// n1 = (n1+n2)decel / (accel + decel)

accel_lim = (uint32_t)(step*decel/(accel+decel));

// 我们必须加速至少1步才能才能开始减速.

if(accel_lim == 0){

accel_lim = 1;

}

// 使用限制条件我们可以计算出减速阶段步数

if(accel_lim <= max_s_lim){

srd.decel_val = accel_lim - step;

}

else{

srd.decel_val = -(max_s_lim*accel/decel);

}

// 当只剩下一步我们必须减速

if(srd.decel_val == 0){

srd.decel_val = -1;

}

// 计算开始减速时的步数

srd.decel_start = step + srd.decel_val;

// 如果最大速度很慢,我们就不需要进行加速运动

if(srd.step_delay <= srd.min_delay){

srd.step_delay = srd.min_delay;

srd.run_state = RUN;

}

else{

srd.run_state = ACCEL;

}

// 复位加速度计数值

srd.accel_count = 0;

}

MotionStatus = 1; // 电机为运动状态

tim_count=__HAL_TIM_GET_COUNTER(&htimx_STEPMOTOR);  __HAL_TIM_SET_COMPARE(&htimx_STEPMOTOR,STEPMOTOR_TIM_CHANNEL_x,tim_count+srd.step_delay); // 设置定时器比较值

TIM_CCxChannelCmd(STEPMOTOR_TIMx, STEPMOTOR_TIM_CHANNEL_x, TIM_CCx_ENABLE);// 使能定时器通道

STEPMOTOR_OUTPUT_ENABLE();

}

1.2.  VS2008仿真曲线()

  1. 假设运动无限远距离,速度一般,计算达到最大速度的理论步数与实际步数
  2. 其中速度 加速度 减速度 都是放大10倍的
  3. 问题点: max_s_lim为理论加速步数,accel为实际加速步数,

/* 类型定义------------------------------------------------------------------*/

typedef struct {

uint8  run_state ;  // 电机旋转状态

uint8  dir ;        // 电机旋转方向

int32  step_delay;  // 下个脉冲周期(时间间隔),启动时为加速度

uint32 decel_start; // 启动减速位置

int32  decel_val;   // 减速阶段步数

int32  min_delay;   // 最小脉冲周期(最大速度,即匀速段速度)

int32  accel_count; // 加减速阶段计数值

}speedRampData;

// 定义定时器预分频,定时器实际时钟频率为:MHz/(STEPMOTOR_TIMx_PRESCALER+1)

#define STEPMOTOR_TIM_PRESCALER               3  // 步进电机驱动器细分设置为:  32  细分

//#define STEPMOTOR_TIM_PRESCALER               7  // 步进电机驱动器细分设置为:  16  细分

//#define STEPMOTOR_TIM_PRESCALER               15  // 步进电机驱动器细分设置为:  8  细分

//#define STEPMOTOR_TIM_PRESCALER               31  // 步进电机驱动器细分设置为:  4  细分

//#define STEPMOTOR_TIM_PRESCALER               63  // 步进电机驱动器细分设置为:  2  细分

//#define STEPMOTOR_TIM_PRESCALER               127  // 步进电机驱动器细分设置为:  1  细分

// 定义定时器周期,输出比较模式周期设置为xFFFF

#define STEPMOTOR_TIM_PERIOD                  0xFFFF

#define FALSE                                 0

#define TRUE                                  1

#define CW                                    0 // 顺时针

#define CCW                                   1 // 逆时针

#define STOP                                  0 // 加减速曲线状态:停止

#define ACCEL                                 1 // 加减速曲线状态:加速阶段

#define DECEL                                 2 // 加减速曲线状态:减速阶段

#define RUN                                   3 // 加减速曲线状态:匀速阶段

#define T1_FREQ                        (72000000/(STEPMOTOR_TIM_PRESCALER+1)) // 频率ft值

#define FSPR                                  200         //步进电机单圈步数

#define MICRO_STEP                            32          // 步进电机驱动器细分数

#define SPR                                  (FSPR*MICRO_STEP)   // 旋转一圈需要的脉冲数

// 数学常数

#define ALPHA                                 ((float)(2*3.14159/SPR))       // α= 2*pi/spr

#define A_T_x10                               ((float)(10*ALPHA*T1_FREQ))

#define T1_FREQ_148                           ((float)((T1_FREQ*0.676)/10)) // 0.676为误差修正值

#define A_SQ                                  ((float)(2*100000*ALPHA))

#define A_x200                                ((float)(200*ALPHA))

#include "stdafx.h"

#include "stdio.h"

#include "stdlib.h"

#include "string.h"

#include "motor.h"

#include "math.h"

MOTOR_DATA   MotorData;

int32  step=20000000;

uint32 accel=15000;//rad/s

uint32 decel=15000;//rad/s

uint32 speed=1200;//rad/s

char buff[30];

char buff1[30];

speedRampData srd;

//假设距离无限远,那么肯定可以达到最大速度,计算加速到最大速度的步数

void MotorInit(FILE *fp)

{

//uint16 tim_count;

//达到最大速度时的步数

uint32 max_s_lim;

//必须要开始减速的步数

uint32 accel_lim;

float  step_delay;

//float  min_delay;

// min_delay = (alpha / tt)/ w

srd.min_delay = (int32_t)(A_T_x10/speed);

//min_delay=(A_T_x10/speed);

// 通过计算第一个(c0) 的步进延时来设定加速度,其中accel单位为0.1rad/sec^2

// step_delay = 1/tt * sqrt(2*alpha/accel)

// step_delay = ( tfreq*0.676/10 )*10 * sqrt( (2*alpha*100000) / (accel*10) )/100

srd.step_delay = (int32_t)((T1_FREQ_148 * sqrt(A_SQ / accel))/10);

step_delay = ((T1_FREQ_148 * sqrt(A_SQ / accel))/10);

// 计算多少步之后达到最大速度的限制

// max_s_lim = speed^2 / (2*alpha*accel)

max_s_lim = (uint32_t)(speed*speed/(A_x200*accel/10));

accel_lim = (uint32_t)(step*decel/(accel+decel));

sprintf(buff,"max_s_lim=%d accel_lim=%d\r",max_s_lim,accel_lim);

fwrite(buff,sizeof(buff),1,fp);

srd.accel_count = 0;

}

void Timer_Scan(FILE *fp)

{

uint16_t tim_count=0;

// 保存新(下)一个延时周期

uint16_t new_step_delay=0;

// 加速过程中最后一次延时(脉冲周期).

static uint16_t last_accel_delay=0;

// 总移动步数计数器

static uint32_t step_count = 0;

// 记录new_step_delay中的余数,提高下一步计算的精度

static int32_t rest = 0;

//定时器使用翻转模式,需要进入两次中断才输出一个完整脉冲

static uint8_t i=0;

srd.accel_count++;// 加速计数值加1

////计算新(下)一步脉冲周期(时间间隔)

new_step_delay = srd.step_delay - (((2 *srd.step_delay) + rest)/(4 * srd.accel_count + 1));

// 计算余数,下次计算补上余数,减少误差

rest = ((2 * srd.step_delay)+rest)%(4 * srd.accel_count + 1);

srd.step_delay=new_step_delay;

while(new_step_delay >= srd.min_delay)//超过当前速度时退出计算

{

srd.accel_count++;

new_step_delay = srd.step_delay - (((2 *srd.step_delay) + rest)/(4 * srd.accel_count + 1));

rest = ((2 * srd.step_delay)+rest)%(4 * srd.accel_count + 1);

srd.step_delay = new_step_delay;

sprintf(buff,"new_step_delay=%d count=%d\r",new_step_delay,srd.accel_count);

fwrite(buff,sizeof(buff),1,fp);

printf("%s %s \r",buff,buff1);

memset(buff,0x00,sizeof(buff));

}

}

int main()

{

FILE *fp;

fp=fopen("Timer.txt","w");

MotorInit(fp);//参数初始化

Timer_Scan(fp);

fclose(fp);

system("pause");

return 0;

}

void Timer_Scan(FILE *fp)

{

uint16_t tim_count=0;

// 保存新(下)一个延时周期

uint16_t new_step_delay=0;

// 加速过程中最后一次延时(脉冲周期).

static uint16_t last_accel_delay=0;

// 总移动步数计数器

static uint32_t step_count = 0;

// 记录new_step_delay中的余数,提高下一步计算的精度

static int32_t rest = 0;

//定时器使用翻转模式,需要进入两次中断才输出一个完整脉冲

static uint8_t i=0;

double a;

//srd.accel_count++;// 加速计数值加1

new_step_delay=srd.min_delay;//

while(new_step_delay >= srd.min_delay)//超过当前速度时退出计算

{

srd.accel_count++;

//new_step_delay=(int32)(srd.step_delay*sqrt((double)(srd.accel_count+1));

new_step_delay=(int32)(srd.step_delay*(sqrt((double)(srd.accel_count+1))-sqrt((double)(srd.accel_count))));

sprintf(buff,"new_step_delay=%d count=%d\r",new_step_delay,srd.accel_count);

fwrite(buff,sizeof(buff),1,fp);

printf("%s %s \r",buff,buff1);

memset(buff,0x00,sizeof(buff));

}

}

原文地址:https://www.cnblogs.com/csczq54/p/10799347.html

时间: 2024-10-29 17:42:00

AVR446_Linear speed control of stepper motor步进电机曲线分析的相关文章

股票主动资金变化曲线分析交易意向和趋势

QStockView_1.0.8 新增主动资金变化曲线分析,主动资金是指主动买入的交易金额减去主动卖出的交易金额,分为全部净资金和主力净资金.单位为百万.主动资金为正时,表示主动买入的资金比主动卖出的资金要多,股票会呈上升趋势,同理主动资金为负时,表明主动买入的资金比主动卖出的资金少,股票会下跌:可以根据主动资金曲线,了解交易意向和趋势: 如下图所示,恒生电子的主力资金为负,但是总的主动资金为正.说明只是散户买入意向比较高,主力意向并不高.而海康威视都为正数,且主力净资金大于总的净资金,所以主力

apache config 1 Bandwidth Mod (apache speed control)

调试网络的时候进行  HTTP模拟网速 快慢  使用apache 插件 Bandwidth Mod 1.安装mod_bw 下载http://bwmod.sourceforge.net/# 解压 进入目录执行 sudo apxs2 -i -a -c mod_bw.c 2.配置 在配置文件/etc/apache2/httpd.conf 加入如下 配置为低速模式 1KB/S <location /modbw> SetHandler modbw-handler </location> &l

nvidia gpu fan speed control

nvidia在linux上控制风扇速度有点坑爹,需要起X,这里汇总一下整个流程 1.nvidia-xconfig --enable-all-gpus --separate-x-screens --cool-bits=4 生成xorg.conf 2.在X已启动的前提下,用nvidia-settings -q all | grep -i fan 列出关于风扇的所有属性,其中应该有 GPUFanControlState,把它设置为 1,然后再根据需要设置 GPUFanTarget 3.如果X没有启动,

步进电机库函数

This library allows you to control unipolar or bipolar stepper motors. To use it you will need a stepper motor, and the appropriate hardware to control it. For more on that, see Tom Igoe's notes on steppers. 函数 Stepper(steps, pin1, pin2)            步

AN2820 Driving bipolar stepper motors using a medium-density STM32F103xx microcontroller

AN2820 Driving bipolar stepper motors using a medium-density STM32F103xx microcontroller Introduction This application note describes how to achieve compact size, high speed and low cost with less resources when driving bipolar stepper motors using t

[PCB制作] 1、记录一个简单的电路板的制作过程——四线二项步进电机驱动模块(L6219)

前言 现在,很多人手上都有一两个电子设备,但是却很少有人清楚其中比较关键的部分(PCB电路板)是如何制作出来的.我虽然懂点硬件,但是之前设计的简单系统都是自己在万能板上用导线自己焊接的(如下图左),复杂的都是模块拼接的(如下图右):      工作中原理图和PCB也有专门的工程师来制作,因此我对这一块了解比较少.而最近闲来无事,又因为手头上确实少一个四线二项步进电机驱动模块.起初是在淘宝上找了很久才找到一个适合的,结果实验了一下午还是不行:又考虑自己在万能板上焊接,可是发现该模块外围需要10个左

Android 利用二次贝塞尔曲线模仿购物车添加物品抛物线动画

0.首先,先给出一张效果gif图. 1.贝塞尔曲线原理及相关公式参考:http://www.jianshu.com/p/c0d7ad796cee 作者:许方镇. 2.原理:计算被点击 view.购物车view 以及他们所在父容器相对于屏幕的坐标. 3.在呗点击View坐标位置 父容器通过addView 增加需要完成动画的imgview. 4.自定义估值器 通过二次贝塞尔曲线公式(2个数据点,一个控制点)完成抛物线路径上的点xy坐标计算. 5.利用属性动画 +自定义估值器 完成imgview在父容

Android自定义View进阶 - 贝塞尔曲线

Path之贝塞尔曲线 作者微博: @GcsSloop [本系列相关文章] 在上一篇文章Path之基本图形中我们了解了Path的基本使用方法,本次了解Path中非常非常非常重要的内容-贝塞尔曲线. 一.Path常用方法表 为了兼容性(偷懒) 本表格中去除了在API21(即安卓版本5.0)以上才添加的方法.忍不住吐槽一下,为啥看起来有些顺手就能写的重载方法要等到API21才添加上啊.宝宝此刻内心也是崩溃的. 作用 相关方法 备注 移动起点 moveTo 移动下一次操作的起点位置 设置终点 setLa

命令行界面下使用emca安装配置Oracle Database Control实战

作为命令行忠实用户,服务器端软件的运维都倾向于使用命令或 脚本完成,非常讨厌资源占用很大的GUI.Oracle数据库作为重要的服务器端软件,其安装运维自然也完全支持纯命令行方式,尽管同时提供了OUI这个GUI安装程序.至于纯命令行下静默安装Oracle11g,本人之前写过详细的博文,有需要的请参考之,博文地址http://blog.csdn.net/smstong/article/details/7352036.按照这篇博文安装的数据库完全可用.本文算是对前面这篇博文的一个补充,就是在数据库安装