西门子 S7-1500 PLC,使用手轮控制伺服电机
本文描述了一种,1500PLC使用叠加定位的方法,实现手轮操作的方法。
手轮操作需要的功能
数控机床等设备上的电子手轮,起源于机械手轮。机械手轮是通过一个圆的摇柄,经过减速箱,带动机床上的滑台进行移动。
数控机床的进给轴、主轴已经实现了电气化。由伺服电机替代了人力来驱动滑台。相应的,电子手轮也替代了机械手轮作为伺服轴的操作部件。
![]
- 定量进给,根据“倍率”的选择,手轮每摇一格,滑台移动一个增量。
- 快速响应,机床(设备)操作者在摇动手轮时,不能有明显的之后。
- 立即停止,摇动停止后,滑台立刻停止运动。
- 平稳,电子手轮每圈分辨率为100,分辨率不高,并且人的操作有微观的抖动,但滑台的运动需要相对平稳。
1500PLC可以将伺服电机配置为轴工艺对象,可以将手轮配置为编码器工艺对象。然而,1500PLC支持相对齿轮同步。看似我们可以使用齿轮同步方式来使用手轮,然而配置时发现,编码器不能作为同步运动的主轴。
如果希望使用编码器作为主轴,需要选择1500T PLC,价格会更贵一些。
手轮和PLC,SIMOTION 的硬件连接方式
方式1
手轮连接到驱动侧,使用SMC30编码器接口模块连接
手轮作为驱动DO(Drive object),驱动和PLC(或simotion)之间通过81报文通讯,PLC侧建立编码器(TO)工艺对象
参考:《simotion 使用齿轮同步方式连接手轮》
方式2
手持型触摸屏MP277/MP377等上的电子手轮,手轮信号是两个byte的整数信号,通过PROFINET通讯方式连接到PLC/SIMOTION。可以通过累加计数方式控制伺服电机,也可以通过建立工艺对象,并将工艺对象
参考:《将变量值写入simotion encoder位置》
http://www.cnblogs.com/lion-zheng/p/7657811.html
方式3
手轮连接到PLC的位置输入模块上,例如TM PosInput 2, 6ES7551-1AB00-0AB0
PLC(或simotion)和位置输入模块之间通过81报文通讯,PLC侧建立编码器工艺对象
控制思路
本文讲的是方式3的配置和调试方法
,由于1500PLC不支持轴和编码器之间齿轮同步,我选择使用 叠加定位 的进行控制。
- 按照100ms周期执行程序
- 判断编码器位置是否发生改变,如果发生改变,进行增量定位
- 读取增量定位的余程(distance-to-go),本周期定位长度 = 余程 + 编码器增量
- 如果手轮位置没有改变,stop计数器加1
- stop计数器值大于阈值(2),轴执行
stop
指令 - 程序中的加速度为固定值,速度和
倍率开关 X1 X10 X100
的位置相关
函数接口和调用方式
1)程序接口
因为是一个近似定位功能,我把程序块(Function_block)的名字叫做stupid_handwheel
name | datatype | function |
---|---|---|
handwheel_position | lreal | encoder position |
TO_Axis | TO_PositioningAxis | Axis tech object |
enable | BOOL | 手轮功能使能 |
cycle_trig | BOOL | 周期信号 |
X1 | BOOL | 手轮倍率 X1 |
X10 | BOOL | 手轮倍率 X10 |
X100 | BOOL | 手轮倍率 X100 |
2)调用方式
手轮控制轴运行时的特性
图中可以看到,定位过程中,速度被限制为10mm/s,轴的运行比较平稳。
词汇
PLC -------------Programmable Logic Controller 可编程逻辑控制器
S7-1500 --------西门子在21世纪推出的中高端PLC,是S7-300/400的升级产品
DO ---------------Drive object S120驱动内部的对象
source code
stupid_handwheel
程序块的源程序
FUNCTION_BLOCK "Stupid_handwheel" { S7_Optimized_Access := 'TRUE' } VERSION : 0.1 //count handwheel numbers, and pos relative VAR_INPUT handwheel_position : LReal; TO_Axis {OriginalPartName := 'TO_PositioningAxis'; LibVersion := '3.0'} : TO_PositioningAxis; enable : Bool; cycle_trig : Bool; X1 : Bool; X10 : Bool; X100 : Bool; END_VAR VAR cycle_TRIG_Instance {OriginalPartName := 'R_TRIG_1500'; LibVersion := '1.0'} : R_TRIG; cycle_TrigQ : Bool; handwheel_position_old : LReal; diff : LReal; DistanceToGo : LReal; MC_MOVERELATIVE_Instance {OriginalPartName := 'MC_MOVERELATIVE'; LibVersion := '3.0'} : MC_MOVERELATIVE; MC_HALT_Instance {OriginalPartName := 'MC_HALT'; LibVersion := '3.0'} : MC_HALT; Pos_CMD : Bool; Pos_CMD_ID : Int; Stop_CMD : Bool; Pos_Velocity : LReal; Pos_Distance : LReal; Pos_Acc : LReal; Pos_Jerk : LReal; Length_Factor : LReal; PosTon {OriginalPartName := 'IEC_TIMER'; LibVersion := '1.0'} : TON_TIME; StopTon {OriginalPartName := 'IEC_TIMER'; LibVersion := '1.0'} : TON_TIME; StopCycleCounter : Int; END_VAR BEGIN #cycle_TRIG_Instance(CLK:=#cycle_trig); IF #X1 THEN #Length_Factor := 0.001; #Pos_Velocity := 1; ELSIF #X10 THEN #Length_Factor := 0.01; #Pos_Velocity := 2; ELSIF #X100 THEN #Length_Factor := 0.1; #Pos_Velocity := 10; END_IF; IF #cycle_TRIG_Instance.Q THEN //position difference in one cycle #diff := #handwheel_position - #handwheel_position_old; #handwheel_position_old := #handwheel_position; IF #diff > 50 THEN #diff := #diff - 100; END_IF; IF #diff < -50 THEN #diff := #diff + 100; END_IF; //Distance to go IF #TO_Axis.StatusPositioning.TargetPosition > #TO_Axis.Position THEN #DistanceToGo := #TO_Axis.StatusPositioning.Distance; ELSE #DistanceToGo := -#TO_Axis.StatusPositioning.Distance;; END_IF; //New command length IF ABS(#diff) > 0.5 AND #enable THEN #Pos_Distance := #diff * #Length_Factor + #DistanceToGo; #Pos_CMD := true; #Pos_CMD_ID := 99; END_IF; //stop cmd IF ABS(#diff) < 0.5 AND #Pos_CMD_ID = 99 THEN #StopCycleCounter := #StopCycleCounter + 1; END_IF; IF #StopCycleCounter > 1 THEN #Stop_CMD := TRUE; #Pos_CMD_ID := 0; #StopCycleCounter := 0; END_IF; END_IF; //IF #TO_Axis #MC_MOVERELATIVE_Instance(Axis := #TO_Axis, Execute := #Pos_CMD, Distance := #Pos_Distance, Velocity := #Pos_Velocity, Acceleration := 500, Deceleration := 500, Jerk := 8000.0); IF #MC_MOVERELATIVE_Instance.Done THEN #Pos_CMD_ID := 0; END_IF; #MC_HALT_Instance(Axis:=#TO_Axis, Execute:=#Stop_CMD, Deceleration:= 500, Jerk:= 10000, AbortAcceleration:= TRUE); //reset pos CMD #PosTon(IN:=#Pos_CMD,PT:=t#30ms); IF #PosTon.Q THEN #Pos_CMD := FALSE; END_IF; //reset stop CMD #StopTon(IN:=#Stop_CMD,PT:=t#30ms); IF #StopTon.Q THEN #Stop_CMD := FALSE; END_IF; END_FUNCTION_BLOCK
来源:https://www.cnblogs.com/lion-zheng/p/7847957.html