HID 数据传输上位机READFILE不到数据的问题

[复制链接]
 楼主| coolcall 发表于 2013-3-17 14:37 | 显示全部楼层 |阅读模式
一个AVR和上位机的通讯实验,上位机VC2010
功能:上位机发送8位数据,HID设备返回相同数据
BUS HOUND 能检测出数据返回数据,可是READFILE()读不出数据,这是为什么?
找了两天的资料,说什么的都有呀,头快爆炸了



固件程序代码

  1. char usbDescriptorHidReport[33] = { // USB report descriptor
  2. 0x06,0x00,0xFF, //USAGE_PAGE (Vendor Defined Page 1)
  3. 0x09,0x01, //USAGE (Vendor Usage 1)
  4. 0xA1,0x01, //COLLECTION (Application)
  5. 0x19,0x01, //(Vendor Usage 1)
  6. 0x29,0x08, //(Vendor Usage 1)
  7. 0x85,0x01, //报告ID(1)
  8. 0x15,0x00, //LOGICAL_MINIMUM (0)
  9. 0x26,0xFF,0x00, //LOGICAL_MAXIMUM (255)
  10. 0x75,0x08, //REPORT_SIZE (8)
  11. 0x95,0x07, //REPORT_COUNT (7)
  12. 0x81,0x02, //INPUT (Data,Var,Abs)
  13. 0x19,0x01, //(Vendor Usage 1)
  14. 0x29,0x08, //(Vendor Usage 1)
  15. 0x85,0x02, //报告ID(2)上位机VC-WriteFile函数缓冲区0位必须设置此ID数
  16. 0x91,0x02, //OUTPUT (Data,Var,Abs)
  17. 0xC0 // END_COLLECTION // END_COLLECTION
  18. };
  19. static uchar rcbuf[8]={0};
  20. static uchar rpbuf[8]={0};
  21. void write()//AVRM16属低速设备,每次只能发8个字节
  22. {
  23. if(usbInterruptIsReady())
  24. {
  25. usbSetInterrupt(rcbuf, sizeof(rcbuf));
  26. }
  27. return;
  28. }

  29. uchar usbFunctionSetup(uchar data[8])
  30. {
  31. //LCDprintf(1,(void *)data);

  32. /* 这里你可以根据data[8]里面的参数做一些判断,决定返回哪些数据 */
  33. return USB_NO_MSG; //USB_NO_MSG=0xff
  34. }
  35. /*
  36. 上位机通过函数要求读数据(UsbRequestType.EndpointIn),
  37. vusb Driver则自动会调用usbFunctionRead()函数来把数据
  38. (rpbuf数组里的数据)返回给上位机,其中len参数就是的参
  39. 数bufferLen。
  40. */
  41. uchar usbFunctionRead(uchar *data, uchar len) {
  42. uchar i;
  43. for(i=0;i<len;i++){
  44. data[i]=rpbuf[i];
  45. }
  46. // LCDprintf(2,(void *)data);
  47. return len;
  48. }
  49. /*
  50. 如果上位机通过函数要求向单片机写数据(UsbRequestType.EndpointOut),
  51. vusb Driver自动会调用usbFunctionWrite()函数来接收上位机传来的数据,
  52. 并保存到rcbuf数组中。
  53. */
  54. uchar usbFunctionWrite(uchar *data, uchar len) {
  55. uchar i;
  56. for(i=0;i<len;i++){
  57. rcbuf[i]=data[i];
  58. }
  59. write();
  60. // LCDprintf(3,(void *)rcbuf);
  61. return len;
  62. }

  63. /* ------------------------------------------------------------------------- */
  64. int main(void)
  65. {
  66. DDRC = 0xFF;
  67. PORTC = 0xFF;
  68. init_devices();
  69. _delay_ms(50);
  70. LCD_init(); //LCD初始化
  71. LCD_clear(); //清屏
  72. clear_img();
  73. DDRC = 0xff;
  74. PORTC = 0xff;
  75. LCDprintf(1,"USBhid通讯实验");
  76. wdt_enable(WDTO_2S);
  77. // odDebugInit();
  78. usbInit();
  79. sei();
  80. // DBG1(0x00, 0, 0);
  81. for(;;){ /* main event loop */
  82. wdt_reset();
  83. usbPoll();
  84. }
  85. return 0;
  86. }



上位机VC代码
  1. #define USB_VID 0x16c0
  2. #define USB_PID 0x05e1
  3. HANDLE OpenMyHIDDevice(int overlapped);
  4. LPVOID lpMsgBuf;
  5. /*用FormatMessage()得到由GetLastError()返回的出错编码所对应错误信息*/
  6. #define error FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_IGNORE_INSERTS,NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPTSTR) &lpMsgBuf,0, NULL )
  7. /*---------------------------------------------*/
  8. BYTE lpBuf[8];

  9. bool write(HANDLE hComm)
  10. {
  11. OVERLAPPED osWrite = {0};
  12. DWORD dwWritten;
  13. BOOL fRes;
  14. memset(lpBuf, 0, 8);//数组清零
  15. lpBuf[0]=2;
  16. lpBuf[1]=4;
  17. // Create this writes OVERLAPPED structure hEvent.
  18. osWrite.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
  19. if (osWrite.hEvent == NULL)
  20. // Error creating overlapped event handle.
  21. return false;

  22. // Issue write.
  23. if (!WriteFile(hComm, lpBuf, 8, &dwWritten, &osWrite))
  24. {
  25. if (GetLastError() != ERROR_IO_PENDING)
  26. {
  27. // WriteFile failed, but it isn't delayed. Report error and abort.
  28. fRes = false;
  29. }
  30. else
  31. {
  32. // Write is pending.

  33. #define WRITE_TIMEOUT 500 //write timeout
  34. DWORD dwRes = WaitForSingleObject(osWrite.hEvent, WRITE_TIMEOUT);
  35. switch(dwRes)
  36. {
  37. // Read completed.
  38. case WAIT_OBJECT_0:
  39. if (!GetOverlappedResult(hComm, &osWrite, &dwWritten, TRUE))
  40. fRes = FALSE;
  41. else
  42. // Write operation completed successfully.
  43. fRes = true;
  44. case WAIT_TIMEOUT:
  45. // This is a good time to do some background work.
  46. break;

  47. default:
  48. // Error in the WaitForSingleObject; abort.
  49. // This indicates a problem with the OVERLAPPED structure's
  50. // event handle.
  51. break;
  52. }
  53. }
  54. }
  55. else
  56. // WriteFile completed immediately.
  57. fRes = true;

  58. CloseHandle(osWrite.hEvent);
  59. return fRes;
  60. }


  61. bool read(HANDLE hComm)
  62. {
  63. DWORD dwRead;
  64. BOOL fWaitingOnRead = FALSE;
  65. OVERLAPPED osReader = {0};
  66. memset(lpBuf, 0, 8);//数组清零
  67. // Create the overlapped event. Must be closed before exiting
  68. // to avoid a handle leak.
  69. osReader.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);

  70. if (osReader.hEvent == NULL)
  71. {
  72. // Error creating overlapped event; abort.
  73. return false;
  74. }

  75. if (!fWaitingOnRead)
  76. {
  77. // Issue read operation.
  78. if (!ReadFile(hComm, lpBuf, 8, &dwRead, &osReader))
  79. {
  80. // error;//错误查询
  81. if (GetLastError() != ERROR_IO_PENDING) // read not delayed?
  82. {
  83. // Error in communications; report it.
  84. }
  85. else
  86. {
  87. fWaitingOnRead = TRUE;
  88. }
  89. }
  90. else
  91. {
  92. // read completed immediately
  93. // HandleASuccessfulRead(lpBuf, dwRead);
  94. }
  95. }

  96. if (fWaitingOnRead)
  97. {
  98. #define READ_TIMEOUT 500 // milliseconds
  99. DWORD dwRes = WaitForSingleObject(osReader.hEvent, READ_TIMEOUT);
  100. switch(dwRes)
  101. {
  102. // Read completed.
  103. case WAIT_OBJECT_0:
  104. if (!GetOverlappedResult(hComm, &osReader, &dwRead, FALSE))
  105. {
  106. // Error in communications; report it.
  107. }
  108. else
  109. {
  110. // Read completed successfully.
  111. //HandleASuccessfulRead(lpBuf, dwRead);
  112. }
  113. // Reset flag so that another opertion can be issued.
  114. fWaitingOnRead = false;
  115. break;

  116. case WAIT_TIMEOUT:
  117. // Operation isn't complete yet. fWaitingOnRead flag isn't
  118. // changed since I'll loop back around, and I don't want
  119. // to issue another read until the first one finishes.
  120. //
  121. // This is a good time to do some background work.
  122. break;

  123. default:
  124. // Error in the WaitForSingleObject; abort.
  125. // This indicates a problem with the OVERLAPPED structure's
  126. // event handle.
  127. break;
  128. }
  129. }
  130. return true;
  131. }
  132. void HIDSampleFunc()
  133. {
  134. HANDLE hDev;



  135. hDev = OpenMyHIDDevice(1); //打开设备,使用重叠(异步)方式;
  136. printf("传递设备句柄:%x\n",hDev);
  137. if (hDev == INVALID_HANDLE_VALUE)
  138. return;
  139. /*//不使用异步方式的时候的代码
  140. // BYTE recvDataBuf[8];
  141. // BYTE reportBuf[8];
  142. // DWORD bytes;
  143. memset(reportBuf, 0, 8);//数组清零
  144. reportBuf[0] = 4; //输出报告的报告ID是4
  145. reportBuf[1] = 1;
  146. if (!WriteFile(hDev,// 设备句柄,即 CreateFile 的返回值
  147. reportBuf,// 存有待发送数据的 buffer
  148. 8, // 待发送数据的长度
  149. &bytes, // 实际收到的数据的字节数
  150. 0// 异步模式
  151. )) //写入数据到设备
  152. {

  153. return;}
  154. // Sleep(1500);
  155. printf("开始接收");
  156. ReadFile(hDev, recvDataBuf, 8, &bytes, NULL); //读取设备发给主机的数
  157. printf("结束");
  158. */


  159. for(int i=0;i<8;i++)
  160. {
  161. printf("开始写入\n");
  162. write(hDev);
  163. printf("开始接收\n");
  164. read(hDev);//error;//错误查询
  165. for(int i=0;i<8;i++){printf("%x ",lpBuf[i]);}printf("结束\n");}


  166. }


  167. HANDLE OpenMyHIDDevice(int overlapped)
  168. {
  169. HANDLE hidHandle;
  170. GUID hidGuid;
  171. HidD_GetHidGuid(&hidGuid);

  172. HDEVINFO hDevInfo = SetupDiGetClassDevs(
  173. &hidGuid,
  174. NULL,
  175. NULL,
  176. (DIGCF_PRESENT | DIGCF_DEVICEINTERFACE));
  177. if (hDevInfo == INVALID_HANDLE_VALUE)
  178. {
  179. return INVALID_HANDLE_VALUE;
  180. }

  181. SP_DEVICE_INTERFACE_DATA devInfoData;
  182. devInfoData.cbSize = sizeof (SP_DEVICE_INTERFACE_DATA);
  183. int deviceNo = 0;

  184. SetLastError(NO_ERROR);

  185. while (GetLastError() != ERROR_NO_MORE_ITEMS)
  186. {
  187. if (SetupDiEnumInterfaceDevice (hDevInfo,
  188. 0,
  189. &hidGuid,
  190. deviceNo,
  191. &devInfoData))
  192. {
  193. ULONG requiredLength = 0;
  194. SetupDiGetInterfaceDeviceDetail(hDevInfo,
  195. &devInfoData,
  196. NULL,
  197. 0,
  198. &requiredLength,
  199. NULL);

  200. PSP_INTERFACE_DEVICE_DETAIL_DATA devDetail =
  201. (SP_INTERFACE_DEVICE_DETAIL_DATA*) malloc (requiredLength);
  202. devDetail->cbSize = sizeof(SP_INTERFACE_DEVICE_DETAIL_DATA);

  203. if(!SetupDiGetInterfaceDeviceDetail(hDevInfo,
  204. &devInfoData,
  205. devDetail,
  206. requiredLength,
  207. NULL,
  208. NULL))
  209. {

  210. free(devDetail);
  211. SetupDiDestroyDeviceInfoList(hDevInfo);
  212. return INVALID_HANDLE_VALUE;
  213. }
  214. /*----------------------该处为vid-pid-guid的比较后作出标志位------------*/
  215. char vidpidguid[30];
  216. sprintf(vidpidguid,"%s","\\\\?\\hid#vid_16c0&pid_05e1#");//“\\”为转移字符,要改成这样
  217. int hidcheck=0;
  218. for(int i=0 ;i<25;i++){
  219. if(devDetail->DevicePath[i] == vidpidguid[i]){hidcheck++;}
  220. printf("%c",devDetail->DevicePath[i]);
  221. }
  222. printf("======%d\n",hidcheck);
  223. /*-------------------------------------------------------------------*/
  224. if (overlapped)
  225. {

  226. if(hidcheck == 25)//vid-pid-guid的比较正确,则使用GENERIC_READ | GENERIC_WRITE访问
  227. {
  228. hidHandle = CreateFile(devDetail->DevicePath,
  229. GENERIC_READ | GENERIC_WRITE,
  230. FILE_SHARE_READ | FILE_SHARE_WRITE,
  231. (LPSECURITY_ATTRIBUTES)NULL,
  232. OPEN_EXISTING,
  233. FILE_FLAG_OVERLAPPED,
  234. NULL);
  235. printf("使用读写方式打开\n");
  236. }
  237. else
  238. {//否则作为独占方式访问
  239. hidHandle = CreateFile(devDetail->DevicePath,
  240. 0,
  241. FILE_SHARE_READ | FILE_SHARE_WRITE,
  242. (LPSECURITY_ATTRIBUTES)NULL,
  243. OPEN_EXISTING,
  244. FILE_FLAG_OVERLAPPED,
  245. NULL);
  246. printf("系统独占方式打开\n");
  247. }

  248. }
  249. else
  250. {
  251. if(hidcheck == 25)//vid-pid-guid的比较正确,则使用GENERIC_READ | GENERIC_WRITE访问
  252. {
  253. hidHandle = CreateFile(devDetail->DevicePath, //访问路径
  254. GENERIC_READ | GENERIC_WRITE, //访问方式必须为零,由于鼠标键盘HID是系统独占,所以会出错
  255. FILE_SHARE_READ | FILE_SHARE_WRITE, //共享模式
  256. (LPSECURITY_ATTRIBUTES)NULL,
  257. OPEN_EXISTING, //文件不存在是返回失败
  258. 0, //若为FILE_FLAG_OVERLAPPED以重叠(异步)模式打开
  259. NULL
  260. );
  261. printf("使用读写方式打开\n");
  262. }
  263. else
  264. {//否则作为独占方式访问
  265. hidHandle = CreateFile(devDetail->DevicePath, //访问路径
  266. 0, //访问方式必须为零,由于鼠标键盘HID是系统独占,所以会出错
  267. FILE_SHARE_READ | FILE_SHARE_WRITE, //共享模式
  268. (LPSECURITY_ATTRIBUTES)NULL,
  269. OPEN_EXISTING, //文件不存在是返回失败
  270. 0, //若为FILE_FLAG_OVERLAPPED以重叠(异步)模式打开
  271. NULL
  272. );
  273. printf("系统独占方式打开\n");
  274. }
  275. }

  276. free(devDetail);

  277. if (hidHandle==INVALID_HANDLE_VALUE)
  278. {
  279. SetupDiDestroyDeviceInfoList(hDevInfo);
  280. free(devDetail);
  281. return INVALID_HANDLE_VALUE;
  282. }
  283. printf("打开成功\n");
  284. _HIDD_ATTRIBUTES hidAttributes;
  285. if(!HidD_GetAttributes(hidHandle, &hidAttributes))
  286. {
  287. CloseHandle(hidHandle);
  288. SetupDiDestroyDeviceInfoList(hDevInfo);


  289. return INVALID_HANDLE_VALUE;
  290. }

  291. if (USB_VID == hidAttributes.VendorID
  292. && USB_PID == hidAttributes.ProductID)
  293. {
  294. printf("找到设备!\n");
  295. break;
  296. }
  297. else
  298. {
  299. CloseHandle(hidHandle);
  300. ++deviceNo;
  301. }
  302. }
  303. }

  304. SetupDiDestroyDeviceInfoList(hDevInfo);

  305. return hidHandle;
  306. }



  307. int _tmain(int argc, _TCHAR* argv[])
  308. {
  309. HIDSampleFunc();


  310. // MessageBox(NULL,(LPCTSTR)lpMsgBuf, _T("Error"), MB_OK | MB_ICONINFORMATION );
  311. // Free the buffer.
  312. // LocalFree( lpMsgBuf );
  313. system("PAUSE");
  314. return 0;
  315. }


本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?注册

×
一般首席 发表于 2013-3-18 22:17 | 显示全部楼层
楼主用的是AVRUSB吗
 楼主| coolcall 发表于 2013-3-19 10:24 | 显示全部楼层
对的,现在好像有点思路了,不过还没证实,先试好再说了
奇缘时间 发表于 2013-3-19 15:51 | 显示全部楼层
请问楼主,进行通讯的那个10ms所指的事务包必须包含要向PC发送的数据,近来我也在做个有关,可是现在往PC发送的数据在断电时总会多一个数据出来,请问给指点下嘛!
lzqxs89 发表于 2013-3-21 09:01 | 显示全部楼层
楼主是上位机程序是运用API来完成与HID的通信么?
 楼主| coolcall 发表于 2013-3-21 14:02 | 显示全部楼层
一般首席 发表于 2013-3-18 22:17
楼主用的是AVRUSB吗

是的
 楼主| coolcall 发表于 2013-3-21 14:03 | 显示全部楼层
奇缘时间 发表于 2013-3-19 15:51
请问楼主,进行通讯的那个10ms所指的事务包必须包含要向PC发送的数据,近来我也在做个有关,可是现在往PC发 ...

我是初学者,不是很清楚你说什么
 楼主| coolcall 发表于 2013-3-21 14:06 | 显示全部楼层
lzqxs89 发表于 2013-3-21 09:01
楼主是上位机程序是运用API来完成与HID的通信么?

对呀,writefile数据,用readfile读回显

现在怀疑是下位机参数配置不好,谁能给个AVRusb的代码看看?
lgnativs 发表于 2013-3-22 10:20 | 显示全部楼层
楼主你用非非异步的方式打开的设备,却用异步的模式来读?
 楼主| coolcall 发表于 2013-3-22 13:35 | 显示全部楼层
lgnativs 发表于 2013-3-22 10:20
楼主你用非非异步的方式打开的设备,却用异步的模式来读?

没有把,是异步方式打开的

CreateFile(devDetail->DevicePath,
GENERIC_READ | GENERIC_WRITE,
FILE_SHARE_READ | FILE_SHARE_WRITE,
(LPSECURITY_ATTRIBUTES)NULL,
OPEN_EXISTING,
FILE_FLAG_OVERLAPPED,
NULL);

 楼主| coolcall 发表于 2013-3-23 12:54 | 显示全部楼层
自己解决了,原来writefile和readfile的缓冲去不能共用

您需要登录后才可以回帖 登录 | 注册

本版积分规则

1

主题

8

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部

1

主题

8

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部