yiffer的个人空间 https://blog.eetop.cn/edesign [收藏] [复制] [分享] [RSS]

空间首页 动态 记录 日志 相册 主题 分享 留言板 个人资料

日志

PC与单片机USB(D12)数据通讯过程描述

已有 725 次阅读| 2009-6-4 21:39

 

在与网友交流D12开发心得时,发现有些网友对与PC应用软件与单片机之间数据交换的过程有些困惑,不明白PC应用软件是怎么将数据发给单片机以及单片机是怎样通过D12将数据传给PC应用软件的。在此,谈谈个人对这一过程的理解,希望对大家有些帮助。

用户开发的USB设备一般不是windows开发的标准设备,而在VC软件中要对一个设备进行操作,必须先用CreateFile函数打开设备才能对其进行读写操作。当我们采用driverstudio开发驱动时,框架会产生一个OpenByInterface函数,它将CreateFile函数封装在了里面,其原型如下:

HANDLE OpenByInterface(

              GUID* pClassGuid,       // points to the GUID that identifies the interface class

DWORD instance,         // specifies which instance of the enumerated devices to open

              PDWORD pError          // address of variable to receive error status

              )

当我需要打开一个USB设备时只需要知道该设备的Guid就行了。这个所谓的Guid就是windows里面唯一标记硬件设备的标志,可由driverstudio自动产生,不需要人工干预。

在打开设备以后,我们就可以调用读写函数对设备进行读写了。VC中与驱动交流的函数主要是DeviceIoControl函数。该函数定义如下:

BOOL DeviceIoControl(
  HANDLE hDevice,              // handle to device,设备句柄
  DWORD dwIoControlCode,       // operationIOCTL操作码
  LPVOID lpInBuffer,           // input data buffer,输入数据缓冲//
  DWORD nInBufferSize,         // size of input data buffer,输入//数据缓冲区大小
  LPVOID lpOutBuffer,          // output data buffer,输出数据缓冲//
DWORD nOutBufferSize,        // size of output data buffer,输出
//数据缓冲区大小
  LPDWORD lpBytesReturned,     // byte count,通讯字节计数
  LPOVERLAPPED lpOverlapped    // overlapped information,异步通讯//信息
);

参数中的dwIoControlCode对应着驱动中定义的IOCTL操作码。该操作码唯一定义了驱动的各项操作,比如读写端点1,读写端点2等。其他参数请参考msdn

利用上述函数,分别编写VC中读写各端点的函数。在本人提供的应用程序实例里定义了以下几个函数:

DWORD CTestDevice::Endpoint1ReadPipes(UINT Length, void *pBuffer)

DWORD CTestDevice::Endpoint1WritePipes(UINT Length, void *pBuffer)

DWORD CTestDevice::ReadBulkPipes(UINT Length,void* pBuffer,DWORD* dwBytesTransferred)

DWORD CTestDevice::WriteBulkPipes(UINT Length,void* pBuffer,DWORD* dwBytesTransferred)

在程序中,定义一个CTestDevice类,然后调用上面的函数即实现了对4D12端点的同步读写操作。由于异步读写需要更深层次的驱动的知识,所以不做探讨。

PC应用软件希望发送数据给单片机时,只需调用Endpoint1WritePipes或者WriteBulkPipes(针对不同端点,下同)函数,剩下的USB底层数据传送则交给了驱动与D12。当数据传送过来后,D12便触发中断,单片机在查询了中断寄存器后便知道PC通过哪个端点发送数据过来,随后读出该端点缓冲区的数据,进行操作。

当单片机需要发送数据给PC应用软件时,只要调用D12_WriteEndpoint函数即可将数据通过D12传送到PC端。那么PC应用软件怎么知道数据已经过来了呢?在同步数据读写方式下,PC应用软件一般采用查询的方式。大家可以看到DeviceIoControl函数中定义了输出缓冲区和输入缓冲区。PC应用软件在得到单片机发送过来的数据前,一直查询输入缓冲区的数据有没变化。一旦数据有变化,表明单片机已经发送数据过来,然后读出缓冲区的数据进行操作。当然这有个很大的缺点,就是PC应用软件进行查询时,就不能再做别的事情了,线程被阻塞。这个可以通过多线程的方式解决。

当采用异步读写的时候,就可以避免上面的问题,它类似与一种中断机制,即当数据传送过来时,驱动会发送一个IRP包通知应用程序。在这之前,应用程序完全可以处理别的事情,而不需要等待。当然这种方式是以增加驱动程序难度为代价的,对于初学者来说还是过于复杂了。


点赞

评论 (0 个评论)

facelist

您需要登录后才可以评论 登录 | 注册

  • 关注TA
  • 加好友
  • 联系TA
  • 0

    周排名
  • 0

    月排名
  • 0

    总排名
  • 0

    关注
  • 3

    粉丝
  • 0

    好友
  • 20

    获赞
  • 69

    评论
  • 3705

    访问数
关闭

站长推荐 上一条 /1 下一条

小黑屋| 关于我们| 联系我们| 在线咨询| 隐私声明| EETOP 创芯网
( 京ICP备:10050787号 京公网安备:11010502037710 )

GMT+8, 2024-4-25 14:20 , Processed in 0.017429 second(s), 6 queries , Gzip On, Redis On.

eetop公众号 创芯大讲堂 创芯人才网
返回顶部