技术频道

娓娓工业
您现在的位置: 中国传动网 > 技术频道 > 技术百科 > 简单易用的运动控制卡(五):IO配置与回零运动

简单易用的运动控制卡(五):IO配置与回零运动

时间:2021-10-26 10:16:25来源:深圳市正运动技术有限公司

导语:​今天,正运动小助手给大家分享一下运动控制卡之ECI3808如何使用C++编写控制器回零运动并对应配置IO。

  今天,正运动小助手给大家分享一下运动控制卡之ECI3808如何使用C++编写控制器回零运动并对应配置IO。

  一ECI3808硬件介绍

  1.功能介绍

  ECI3808系列控制卡支持最多达12轴直线插补、任意圆弧插补、空间圆弧、螺旋插补、电子凸轮、电子齿轮、同步跟随、虚拟轴、机械手指令等;采用优化的网络通讯协议可以实现实时的运动控制。

  ECI3808系列运动控制卡支持以太网,RS232 通讯接口和电脑相连,接收电脑的指令运行,可以通过CAN总线去连接各个扩展模块,从而扩展输入输出点数或运动轴。

  ECI3808系列运动控制卡的应用程序可以使用 VC,VB,VS,C++,C#等软件来开发,程序运行时需要动态库 zmotion.dll。调试时可以把ZDevelop软件同时连接到控制器,从而方便调试和观察。

现场总线

  2.硬件接口

现场总线

  通用输入口电路

运动控制

  通用输入口电路

运动控制

  AD/DA接口说明

运动控制

  本地脉冲轴说明

      3.控制器基本信息

运动控制

  二、C++进行运动控制开发

  1.新建MFC项目并添加函数库

  (1)在VS2015菜单“文件”→“新建”→ “项目”,启动创建项目向导。

运动控制

       (2)选择开发语言为“Visual C++”和程序类型“MFC应用程序”。

运动控制

  (3)点击下一步即可。

运动控制

  (4)选择类型为“基于对话框”,下一步或者完成。

运动控制

       (5)找到厂家提供的光盘资料,路径如下(64位库为例)。

  A.进入厂商提供的光盘资料找到“8.PC函数”文件夹,并点击进入。

运动控制

  B.选择“函数库2.1”文件夹。

运动控制

  C.选择“Windows平台”文件夹。

运动控制

  D.根据需要选择对应的函数库这里选择64位库。

运动控制

  E.解压C++的压缩包,里面有C++对应的函数库。

运动控制

  F.函数库具体路径如下。

运动控制

  (6)将厂商提供的C++的库文件和相关头文件复制到新建的项目里面。

运动控制

       (7)在项目中添加静态库和相关头文件。

  A.先右击项目文件,接着依次选择:“添加”→“现有项”。

运动控制

  B.在弹出的窗口中依次添加静态库和相关头文件。

  (8)声明用到的头文件和定义控制器连接句柄。

运动控制

  至此项目新建完成,可进行MFC项目开发。

  2.查看PC函数手册,熟悉相关函数接口。

       (1)PC函数手册也在光盘资料里面,具体路径如下:“光盘资料\8.PC函数\函数库2.1\ZMotion函数库编程手册 V2.1.pdf”

运动控制

  (2)链接控制器,获取链接句柄。

  ZAux_OpenEth()接口说明:

运动控制

  (3) 配置IO信号点对应函数接口如下。

运动控制

  对接口的详细说明可查看PC函数手册。

  以下为回零运动调用接口,以及对回零模式的详细说明,加10表示碰到限位后反找, 不会碰到限位停止,例如13=模式3+限位反找10,用于原点在正中间的情况。

运动控制

  3.MFC开发控制器硬件外设读写例程。

  (1)例程界面如下。

运动控制

       (2)链接按钮的事件处理函数中调用链接控制器的接口函数ZAux_OpenEth(),与控制器进行链接,链接成功后启动定时器1监控控制器状态。

  //网口链接控制器void CSingle_move_Dlg::OnOpen(){ char buffer[256]; int32 iresult; //如果已经链接,则先断开链接 if(NULL != g_handle) { ZAux_Close(g_handle); g_handle = NULL; } //从IP下拉框中选择获取IP地址 GetDlgItemText(IDC_IPLIST,buffer,255); buffer[255] = '\0'; //开始链接控制器 iresult = ZAux_OpenEth(buffer, &g_handle); if(ERR_SUCCESS != iresult) { g_handle = NULL; MessageBox(_T("链接失败")); SetWindowText("未链接"); return; } //链接成功开启定时器1 SetWindowText("已链接"); SetTimer( 1, 100, NULL ); }

  (3)通过定时器监控控制器状态。

  void CFuncWrapperDlg::OnTimer(UINT_PTR nIDEvent){ // TODO: 在此添加消息处理程序代码和/或调用默认值 if(NULL == g_handle) { MessageBox(_T("链接断开")); return ; } if(1 == nIDEvent) { CString string; float position[4] = {0}; int status[4] = {0}; int nAxisStatus[4] = {0}; for (int i = 0;i< 4; i++) { ZAux_Direct_GetDpos( g_handle,i,&position[i]); //获取当前轴位置 ZAux_Direct_GetIfIdle(g_handle,i,&status[i]); //判断当前轴状态 ZAux_Direct_GetAxisStatus(g_handle, i, &nAxisStatus[i]); } if (status[0] == -1) { if (nAxisStatus[0] == 0 || (nAxisStatus[0] & 0x000040) == 64) { string.Format("X 停 止 %.2f 轴状态正常", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } if ((nAxisStatus[0] & 0x000010) == 16) { string.Format("X 停 止 %.2f 轴正向硬限位报警", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } if ((nAxisStatus[0] & 0x000020) == 32) { string.Format("X 停 止 %.2f 轴反向硬限位报警", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } } else { if (nAxisStatus[0] == 0 || (nAxisStatus[0] & 0x000040) == 64) { string.Format("X 运行中 %.2f 轴状态正常", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } if ((nAxisStatus[0] & 0x000010) == 16) { string.Format("X 运行中 %.2f 轴正向硬限位报警", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } if ((nAxisStatus[0] & 0x000020) == 32) { string.Format("X 运行中 %.2f 轴反向硬限位报警", position[0]); GetDlgItem(IDC_STATE_X)->SetWindowText(string); } } if (status[1] == -1) { if (nAxisStatus[1] == 0 || (nAxisStatus[1] & 0x000040) == 64) { string.Format("Y 停 止 %.2f 轴状态正常", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } if ((nAxisStatus[1] & 0x000010) == 16) { string.Format("Y 停 止 %.2f 轴正向硬限位报警", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } if ((nAxisStatus[1] & 0x000020) == 32) { string.Format("Y 停 止 %.2f 轴反向硬限位报警", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } } else { if (nAxisStatus[1] == 0 || (nAxisStatus[1] & 0x000040) == 64) { string.Format("Y 运行中 %.2f 轴状态正常", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } if ((nAxisStatus[1] & 0x000010) == 16) { string.Format("Y 运行中 %.2f 轴正向硬限位报警", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } if ((nAxisStatus[1] & 0x000020) == 32) { string.Format("Y 运行中 %.2f 轴反向硬限位报警", position[1]); GetDlgItem(IDC_STATE_Y)->SetWindowText(string); } } if (status[2] == -1) { if (nAxisStatus[2] == 0 || (nAxisStatus[2] & 0x000040) == 64) { string.Format("Z 停 止 %.2f 轴状态正常", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } if ((nAxisStatus[2] & 0x000010) == 16) { string.Format("Z 停 止 %.2f 轴正向硬限位报警", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } if ((nAxisStatus[2] & 0x000020) == 32) { string.Format("Z 停 止 %.2f 轴反向硬限位报警", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } } else { if (nAxisStatus[2] == 0 || (nAxisStatus[2] & 0x000040) == 64) { string.Format("Z 运行中 %.2f 轴状态正常", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } if ((nAxisStatus[2] & 0x000010) == 16) { string.Format("Z 运行中 %.2f 轴正向硬限位报警", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } if ((nAxisStatus[2] & 0x000020) == 32) { string.Format("Z 运行中 %.2f 轴反向硬限位报警", position[2]); GetDlgItem(IDC_STATE_Z)->SetWindowText(string); } } if (status[3] == -1) { if (nAxisStatus[3] == 0 || (nAxisStatus[3] & 0x000040) == 64) { string.Format("R 停 止 %.2f 轴状态正常", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } if ((nAxisStatus[3] & 0x000010) == 16) { string.Format("R 停 止 %.2f 轴正向硬限位报警", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } if ((nAxisStatus[3] & 0x000020) == 32) { string.Format("R 停 止 %.2f 轴反向硬限位报警", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } } else { if (nAxisStatus[3] == 0 || (nAxisStatus[3] & 0x000040) == 64) { string.Format("R 运行中 %.2f 轴状态正常", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } if ((nAxisStatus[3] & 0x000010) == 16) { string.Format("R 运行中 %.2f 轴正向硬限位报警", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } if ((nAxisStatus[3] & 0x000020) == 32) { string.Format("R 运行中 %.2f 轴反向硬限位报警", position[3]); GetDlgItem(IDC_STATE_R)->SetWindowText(string); } } } CDialog::OnTimer(nIDEvent);}

  (4)使用回零按钮的事件处理函数对回零运动前的参数进行初始化以及调用对应的回零模式操作回零运动。

  void CSingle_homeDlg::OnHome() //回零运动{ // TODO: Add your control notification handler code here UpdateData(true);//刷新参数 int status = 0; ZAux_Direct_GetIfIdle(g_handle, m_nAxis,&status); //判断当前轴状态 if (status == 0) //已经在运动中 { return; } //设定轴类型 7- 脉冲轴类型 + 编码器Z信号 不用EZ回零也可以设置为1 ZAux_Direct_SetAtype(g_handle, m_nAxis, 7); //设定脉冲模式及逻辑方向(脉冲+方向) ZAux_Direct_SetInvertStep(g_handle, m_nAxis, 0); //设置脉冲当量 1表示以一个脉冲为单位 ,设置为1MM的脉冲个数,这度量单位为MM ZAux_Direct_SetUnits(g_handle, m_nAxis, m_units); //设定速度,加减速 ZAux_Direct_SetLspeed(g_handle, m_nAxis, m_lspeed); ZAux_Direct_SetSpeed(g_handle, m_nAxis, m_speed); ZAux_Direct_SetAccel(g_handle, m_nAxis, m_acc); ZAux_Direct_SetDecel(g_handle, m_nAxis, m_dec); ZAux_Direct_SetCreep(g_handle, m_nAxis, m_creep); //设定对应轴的原点输入口信号 ZAux_Direct_SetDatumIn(g_handle, m_nAxis, m_datumin); //ZMC系列认为OFF时碰到了原点信号(常闭) ,如果是常开传感器则需要反转输入口,ECI系列的不需要反转 ZAux_Direct_SetInvertIn(g_handle, m_datumin, 1); //设定对应轴的原点输入口信号 ZAux_Direct_SetFwdIn(g_handle, m_nAxis, m_FwdIn); //ZMC系列认为OFF时碰到了原点信号(常闭) ,如果是常开传感器则需要反转输入口,ECI系列的不需要反转 ZAux_Direct_SetInvertIn(g_handle, m_FwdIn, 1); //设定对应轴的原点输入口信号 ZAux_Direct_SetRevIn(g_handle, m_nAxis, m_RevIn); //ZMC系列认为OFF时碰到了原点信号(常闭) ,如果是常开传感器则需要反转输入口,ECI系列的不需要反转 ZAux_Direct_SetInvertIn(g_handle, m_RevIn, 1); //回零 if( m_datummode < 4) { ZAux_Direct_Single_Datum(g_handle, m_nAxis, m_datummode + 1); //模式1-4 } else { ZAux_Direct_Single_Datum(g_handle, m_nAxis, m_datummode + 4 ); //模式8,9 } UpdateData(false); }

  (5)通过停止运动按钮的事件处理函数来停止当前的运动。

  void CSingle_homeDlg::OnStop() //停止运动{ // TODO: Add your control notification handler code here if(NULL == g_handle) { MessageBox(_T("链接断开状态")); return ; } ZAux_Direct_Single_Cancel(g_handle,m_nAxis,2); //}

  (6)通过坐标清零按钮的事件处理函数来对当前轴的坐标进行对应清零。

  void CSingle_homeDlg::OnZero() //坐标清零{ if(NULL == g_handle) { MessageBox(_T("链接断开状态")); return ; } // TODO: Add your control notification handler code here for (int i=0;i<4;i++) { ZAux_Direct_SetDpos(g_handle,i,0); //设置零点 }}

完整代码获取地址

运动控制

  三调试与监控

  编译运行例程,同时通过ZDevelop软件连接控制器对控制器状态进行监控。

  1.连接ZDevelop软件,并点击“视图”→“示波器”打开示波器对轴运动情况进行监控。

运动控制

运动控制

运动控制

  2.ZDevelop软件调试视频。

运动控制

  本次,正运动技术简单易用的运动控制卡(五):IO配置与回零运动,就分享到这里。更多精彩内容请关注“正运动小助手”公众号,需要相关开发环境与例程代码,请咨询正运动技术销售工程师:400-089-8936。

  本文由正运动技术原创,欢迎大家转载,共同学习,一起提高中国智能制造水平。文章版权归正运动技术所有,如有转载请注明文章来源。


标签: 运动控制工业以太网现场总线

点赞

分享到:

上一篇:预精焊机组生产对钢管划伤的...

下一篇:供应链从物联网中受益的6种突...

中国传动网版权与免责声明:凡本网注明[来源:中国传动网]的所有文字、图片、音视和视频文件,版权均为中国传动网(www.chuandong.com)独家所有。如需转载请与0755-82949061联系。任何媒体、网站或个人转载使用时须注明来源“中国传动网”,违反者本网将追究其法律责任。

本网转载并注明其他来源的稿件,均来自互联网或业内投稿人士,版权属于原版权人。转载请保留稿件来源及作者,禁止擅自篡改,违者自负版权法律责任。

网站简介|会员服务|联系方式|帮助信息|版权信息|网站地图|友情链接|法律支持|意见反馈|sitemap

中国传动网-工业自动化与智能制造的全媒体“互联网+”创新服务平台

网站客服服务咨询采购咨询媒体合作

Chuandong.com Copyright ©2005 - 2021 ,All Rights Reserved 版权所有 粤ICP备 14004826号 | 营业执照证书 | 不良信息举报中心 | 粤公网安备 44030402000946号