英创信息技术ARM9工控板串口设备的COM组件应用

描述

  英创公司的ARM9工控主板产品均预装了Windows CE5.0操作系统,支持包括EVC、C#、VB、LabView等多种开发工具。作为工业控制领域的嵌入模块,客户的应用程序往往对系统的底层调用较多,相对于其它语言,C++具有强大的硬件控制能力和很高执行效率,因此我们提供的示例程序和软件方面的技术支持均集中在C++方面。而C#、.net VB等在图型界面开发、数据库方面的应用和易用性方面更具优势,我们很多客户也选用他们作为开发工具。为了结合各语言的优势,为了对客户提供更好的支持,我们将与主板密切相关的一些底层功能模块封装成COM组件,用户可以使用自己喜欢的语言来调用COM组件,不必关心低层调用的细节,而COM组件本身则采用C++来编写。本文将介绍基于英创工控主板串口应用的COM组件和组件调用方法。

  1、创建串口应用COM组件

  串口通讯在工业控制场合应用十分广泛,当串口作为RS485通信应用时,很多场合下,需要通过硬件RTS信号来控制数据收发的方向,以提高抗干扰能力。在C#这一类的高级语言中,尽管也包括了串口控件,但缺乏对RTS硬件的操作,因此在RS485应用中受到一定的限制。为了弥补这一缺陷,在我们所设计的串口应用COM组件中,通过对串口DCB结构参数的设置,并结合英创ARM9主板低层的串口驱动程序,实现对RTS信号的完整控制。本串口组件采用C++编写,在组件内部对RTS进行设置,而上层的开发工具,如C#等,可通过向组件接口函数传递参数来控制RTS信号,从而最终实现RS485的半双工通讯。

  本串口COM组件提供四个接口方法函数:打开串口,关闭串口,向串口写数据,读串口数据。客户方调用COM组件打开串口后,COM组件服务器便在组件内部创建一数据接收线程,接收线程里通过WaitCommEvent来等待串口事件发生,当串口收到数据后,将数据放入指定的接收数据缓存中,客户方可调用读串口数据方法函数将缓存中的数据读出。在实际应用中,客户可在接收线程中加入自己特定的协议转换代码,使得通过组件读取的数据为一个完整应用报文。

  为了跨语言调用组件,接口方法函数参数数据均采用VARIANT数据类型,这样ASP、vbscript等可方便的进行组件调用,从而轻松实现通过网页对串口进行操作。

  COM组件的创建过程请参考本网站相关文章或参考相应书籍。这里不再赘述。

  2、串口应用组件接口方法函数

  为了跨语言调用组件,组件接口方法函数参数数据均采用VARIANT数据类型。

  (1)OpenPort( VARIANT portNo, VARIANT baud, VARIANT parity, VARIANT dataBits, VARIANT stopBits, VARIANT rtsCtrl, VARIANT* pbool)

  功能描述:打开指定串口。

  输入参数:

  VARIANT portNo 要打开的串口号

  VARIANT baud 设置波特率

  VARIANT parity 设置奇偶较验

  VARIANT dataBits 设置数据位

  VARIANT stopBits 设置停止位

  VARIANT rtsCtrl RTS设置

  输出参数:

  VARIANT* pbool 串口打开成功失败标志

  (2)WritePort(VARIANT *var_inp, VARIANT *retLen)

  功能描述:向串口写数据

  输入参数:

  VARIANT *var_inp 发送数据缓存

  输出参数:

  VARIANT *retLen 发送数据个数

  (3)ReadPort(VARIANT *rxData)

  功能描述:读取串口数据

  输出参数:

  VARIANT *rxData 接收数据缓存

  (4)ClosePort( )

  功能描述:关闭串口

  3、串口组件调用

  下面是在EVC中调用串口组件接口函数的一些程序片段,主要说明在调用接口方法时,VARIANT参数的用法。

  // 从Program ID得到Class ID

  hr = CLSIDFromProgID( OLESTR( ‘ComSerial.CoSerial’ ), &clsid );

  if( FAILED( hr ) )

  {

  return -1;

  }

  // 从Class ID得到ICoSerial接口指针

  hr = CoCreateInstance( clsid, NULL, CLSCTX_INPROC_SERVER, __uuidof( ICoSerial ),

  ( void** )&pICoSerial );

  if( FAILED( hr ))

  {

  return -1;

  }

  //打开串口

  CComVariant portNo( 3 ); // 打开串口3

  CComVariant baud( 9600 ); // 波特率:9600

  CComVariant parity( ‘n’ ); // 无校验位

  CComVariant dataBits( 8 ); // 8位数据

  CComVariant stopBits( 1 ); // 1位停止位

  CComVariant rtsCtrl(RTS_CONTROL_TOGGLE); // RTS设置

  CComVariant pbool( FALSE ); // 串口打开成功标志

  pbool = pICoSerial-》OpenPort( portNo, baud, parity, dataBits, stopBits, rtsCtrl );

  // 向串口发送数据

  char strBuf[100];

  strcpy( strBuf, ‘1234567890!’ );

  long i=0, m=0;

  m = strlen( strBuf );

  SAFEARRAY FAR* pSafeArray;

  SAFEARRAYBOUND rgsabound[1];

  rgsabound[0].lLbound = 0;

  rgsabound[0].cElements = m;

  pSafeArray = SafeArrayCreate( VT_VARIANT, 1, rgsabound );

  VARIANT var;

  for( i; i 《 m; i++)

  {

  var.vt = VT_UI1;

  var.bVal = strBuf[i];

  SafeArrayPutElement( pSafeArray, &i, &var );

  }

  VARIANT tarray,retLen;

  tarray.parray = pSafeArray;

  retLen = pICoSerial-》WritePort( &tarray );

  // 接收数据

  VARIANT vinput, var;

  BYTE rxBuf[1500];

  vinput = pICoSerial-》ReadPort( );

  SafeArrayGetUBound( vinput.parray, 1, &lUbound );

  SafeArrayGetLBound( vinput.parray, 1, &lLbound );

  // m:串口接收到的数据个数

  m = lUbound - lLbound+1;

  // m=0表示串口没有收到数据

  if( 0 == m )

  return;

  for( i=0; i 《 m; i++)

  {

  Safe=ArrayGetElement( vinput.parray, &i, &var );

  rxBuf[i] = (BYTE)var.bVal;

  }

  在实际的应用中,通讯各方必须遵循统一的通讯规约。在发送数据前通常需要将数据按相应的协议打包,添加较验信息等,收到数据后,要进行帧完整性判断、数据解包、数据较验、协议转换等工作。现在可以将这些与协议密切相关的处理放在COM组件内部,客户方在使用串口时仅需要对客户感兴趣的数据进行处理,而协议转换等工作交由COM组件完成。这样程序具有更好结构,维护也更加方便。

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分