Android hid 数据传输(device 端 )

打印 上一主题 下一主题

主题 822|帖子 822|积分 2466

最近一直在处理hid 数据需求,简而言之就是两台装备直接可以通过usb 线相互通报数据。
项目架构


为什么Device 端要采用HID(人机接口装备)的方式发送和接收数据呢?
主要是速率快,举个例子,就是鼠标移动,屏幕可以及时相应,用的也是这种协议。
因为Host端底层我们控制不了,不能保证都支持Hid 协议,以是Host 端采用跨平台方案,libusb 协议。
Liusb 网上介绍的很多啦,可以运行在各个平台,windows ,android.linux,是一种理想中的跨平台数据传输方案。

项目主要功能

1,sensor数据传输
2,TP 数据传输(按键传输同理)


项目主要技术点

1,TP数据监听
2,sensor数据监听
3,hid 数据传输丢失问题
4,HID 节点天生监听
5,开机启动native 服务处理数据
6,selinux 权限问题

技术实现

代码结构


TP数据监听

驱动全部的滑动 和 按键 上报都是通过节点的方式,不同平台节点有所差异,需要和驱动沟通。我试验的平台节点是:
  1. #define INPUT_KEY_NODE "/dev/input/event1"
  2. #define INPUT_TP_NODE "/dev/input/event3"
复制代码
以是监听这两个就行了,我们这里采用的是poll 的方式,有数据的时候会回调,没有的话会阻塞
主要代码
  1. void HidReceiver::listenThread()
  2. {
  3.     struct pollfd fds[IN_FILES];
  4.     fds[0].events = POLLIN;
  5.     fds[1].events = POLLIN;
  6.     fds[2].events = POLLIN;
  7.     int result;
  8.     char buff[512];
  9. //    sleep(1);
  10.     LOGD("hid open");
  11.     hid_fd = open(DEVICE_NODE, O_RDWR | O_NONBLOCK);
  12.     int key_fd =  open(INPUT_KEY_NODE, O_RDWR | O_NONBLOCK);
  13.     int tp_fd =  open(INPUT_TP_NODE, O_RDWR | O_NONBLOCK);
  14.     LOGD("nod %d,%d,%d",hid_fd,key_fd,tp_fd);
  15.     fds[0].fd = hid_fd;
  16.     fds[1].fd = key_fd;
  17.     fds[2].fd = tp_fd;
  18.     unsigned char data[sizeof(input_event)];
  19.     input_event dev_data;
  20.     while(1){
  21.         result = poll(fds, IN_FILES, -1);
  22.         if (result == 0) {
  23.             LOGD("Poll timeout");
  24.         } else if(result > 0){
  25.             if ((fds[0].revents & POLLIN)){
  26.                 int size = read(fds[0].fd, buff, sizeof(buff));
  27.                 if(size > 0){
  28.                     process_event(buff);
  29.                 }
  30.             }
  31.             if ((fds[1].revents & POLLIN)){
  32.                 int size = read (fds[1].fd, (unsigned char*)data, sizeof(input_event));
  33.                 LOGD("size:%d", size);
  34.                 memcpy(&dev_data, data, sizeof(input_event));
  35.                 LOGD("Keyevent size:%d", dev_data.type);
  36.                 // sensordata.sensorType = 0x104;
  37.                 // sensordata.ievent = dev_data;
  38.                 // process_event(sensordata);
  39.             }
  40.             if ((fds[2].revents & POLLIN)){
  41.                 int size = read (fds[2].fd, (unsigned char*)data, sizeof(input_event));
  42.                 memcpy(&dev_data, data, sizeof(input_event));
  43.                 #if 0
  44.                 LOGD("abs size:%d", dev_data.type);
  45.                 if (dev_data.type == EV_ABS)
  46.                 {
  47.                     if (dev_data.code == ABS_MT_POSITION_X)
  48.                     {
  49.                             x = dev_data.value;
  50.                             if (x < 0) x = 0;
  51.                             LOGD("rel X:%d", dev_data.value);
  52.                     }
  53.                     else if(dev_data.code == ABS_MT_POSITION_Y)
  54.                     {
  55.                             LOGD("\nx=%d,y=%d,dev_data.code=%d\n", x, y,dev_data.code);
  56.                             y = dev_data.value;
  57.                             if (y < 0) y = 0;
  58.                             sensordata.sensorType = 0x104;
  59.                             sensordata.abs_x = x;
  60.                             sensordata.abs_y = y;
  61.                             process_event(sensordata);
  62.                             x = 0;
  63.                             y = 0;
  64.                     }
  65.                 }
  66.                 #endif
  67.                 if (dev_data.type == EV_KEY)
  68.                 {
  69.                     LOGD("EV_KEY %d",dev_data.code);
  70.                     switch(dev_data.code){
  71.                         case KEY_KP5://双击
  72.                         case KEY_DASHBOARD://单机
  73.                         case KEY_F17://左滑
  74.                         case KEY_ISO://右滑
  75.                         case KEY_F16://上滑
  76.                         case KEY_CONFIG://下滑
  77.                             sensordata.sensorType = 0x104;
  78.                             sensordata.type = dev_data.type;
  79.                             sensordata.code = dev_data.code;
  80.                             sensordata.value = dev_data.value;
  81.                             sensordata.priority = 3;
  82.                             process_event(sensordata);
  83.                             break;            
  84.                     }
  85.                 }
  86.             }
  87.         }   
  88.     }
  89. }
复制代码
代码中DEVICE_NODE  用于监听hid 数据的,这个后面说。
sensor数据监听

  1. void SensorTransfer::listenThread()
  2. {
  3.     int64_t stamp;
  4.     LOGD("listenThread");
  5.     while (m_bListening)
  6.     {
  7.         ASensorEvent event;
  8.         while (ASensorEventQueue_getEvents(m_pEvtQue, &event, 1) > 0)
  9.         {
  10.             stamp = event.timestamp;
  11.             switch (event.type)
  12.             {
  13.             // case ASENSOR_TYPE_GYROSCOPE:
  14.             //     printf("GYROSCOPE:(%llu, %f,%f,%f)\n", (unsigned long long)stamp, event.data[0], event.data[1], event.data[2]);
  15.             //     break;
  16.             case ASENSOR_TYPE_ACCELEROMETER:
  17.                 // printf("ACCELEROMETER: (%llu, %f,%f,%f)\n", (unsigned long long)stamp, event.data[0], event.data[1], event.data[2]);
  18.                 sensordata.stamp = stamp;
  19.                 sensordata.sensorType = 0x100;
  20.                 sensordata.xvalue = event.data[0];
  21.                 sensordata.yvalue = event.data[1];
  22.                 sensordata.zvalue = event.data[2];
  23.                 saveSensorData(sensordata);
  24.                 sensordata.priority = 1;
  25.                 break;
  26.             case ASENSOR_TYPE_GRAVITY:
  27.                 // printf("GRAVITY: (%llu, %f,%f,%f)\n", (unsigned long long)stamp, event.data[0], event.data[1], event.data[2]);
  28.                 sensordata.stamp = stamp;
  29.                 sensordata.sensorType = 0x101;
  30.                 sensordata.xvalue = event.data[0];
  31.                 sensordata.yvalue = event.data[1];
  32.                 sensordata.zvalue = event.data[2];
  33.                 sensordata.priority = 1;
  34.                 saveSensorData(sensordata);
  35.                 break;
  36.             case ASENSOR_TYPE_PROXIMITY:
  37.                 sensordata.stamp = stamp;
  38.                 sensordata.sensorType = 0x102;
  39.                 sensordata.lightvalue = event.data[0];
  40.                 sensordata.priority = 1;
  41.                 saveSensorData(sensordata);
  42.                 break;
  43.             default:
  44.                 break;
  45.             }
  46.         }
  47.         usleep(1000);
  48.     }
  49. }
复制代码
这个参考的一个博主的方案,主要是通过循环读取native sensor 数据。
监听HID节点删除添加

  1. void HidReceiver::nodWatch(){
  2.     int length, i = 0;
  3.     int fd;
  4.     int wd;
  5.     char buffer[BUF_LEN];
  6.     fd = inotify_init();
  7.     if (fd < 0) {
  8.         LOGD("inotify_init");
  9.     }
  10.     wd = inotify_add_watch(fd, DEV_NODE,
  11.                            IN_CREATE );
  12.     if (wd < 0) {
  13.         LOGD("inotify_add_watch");
  14.     }
  15.     LOGD("Monitoring directory: %s", DEV_NODE);
  16.     bool monitor = true;
  17.     while (monitor) {
  18.         LOGD("start monitor");
  19.         length = read(fd, buffer, BUF_LEN);  
  20.         if (length < 0) {
  21.             LOGD("read");
  22.         }  
  23.         i = 0;
  24.         LOGD("read %d",length);
  25.         while (i < length) {
  26.             struct inotify_event *event = (struct inotify_event *) &buffer[i];
  27.             LOGD("inotify_event %d",event->len);
  28.             if (event->len) {
  29.                 LOGD("ievent->mask %d",event->mask);
  30.                 if (event->mask & IN_CREATE) {
  31.                     LOGD("Created: %s", event->name);
  32.                     if(strcmp(event->name,"hidg0") == 0){
  33.                         LOGD("Created: hidg0");
  34.                         monitor = false;
  35.                         startListen();
  36.                         inotify_rm_watch(fd, wd);
  37.                         return;
  38.                     }
  39.                 } else if (event->mask & IN_DELETE) {
  40.                     LOGD("Deleted: %s", event->name);
  41.                 } else if (event->mask & IN_MODIFY) {
  42.                     LOGD("Modified: %s", event->name);
  43.                 } else if (event->mask & IN_MOVED_FROM) {
  44.                     LOGD("Moved from: %s", event->name);
  45.                 } else if (event->mask & IN_MOVED_TO) {
  46.                     LOGD("Moved to: %s", event->name);
  47.                 }
  48.             }
  49.             i += EVENT_SIZE + event->len;
  50.         }
  51.     }
  52. }
复制代码

Hid 数据传输和数据丢失问题

hid 数据怎么传,其实很简单,写节点就可以了,但是数据量太大的时候,会出现写节点失败,同时,按键或者TP 等数据,也会丢失,sensor 数据丢失感知倒不是很大,但是按键和触摸这些传输失败,Host端就无法相应,体验会很差。

目前采用的方案是
Bufferqueue + 延时 办理HID 数据丢失的问题(生产者斲丧者模式)
priority_queue  办理用户自动利用的数据优先级问题,主要是TP 和 按键,保证优先相应
主要代码:
斲丧者
  1. // 消费者线程,读取队列中的数据并发送
  2. void SensorTransfer::consumer() {
  3.     while (true) {
  4.         std::unique_lock<std::mutex> lock(queueMutex);
  5.         dataCondition.wait(lock, [this] { return !bufferQueue.empty(); });
  6.         // if (!bufferQueue.empty())
  7.         //     break; // 程序结束
  8.         if(!bufferQueue.empty()){
  9.             sensor_data data = bufferQueue.top();
  10.             // 发送数据到 HID 设备
  11.             int written = write(hid_fd,&data,sizeof(struct sensor_data));
  12.             if(written >=0){
  13.                 bufferQueue.pop();
  14.             }else{
  15.                 //LOGD("rewrite data result fail");
  16.                 std::this_thread::sleep_for(std::chrono::milliseconds(1)); // 控制发送速率
  17.             }
  18.             if(data.sensorType == 259){
  19.                 LOGD("consumer sn");
  20.                 std::this_thread::sleep_for(std::chrono::milliseconds(5));
  21.             }
  22.             if(data.sensorType == 260){
  23.                 LOGD("consumer KEY");
  24.                 std::this_thread::sleep_for(std::chrono::milliseconds(5));
  25.             }
  26.         }else{
  27.             LOGD("consumer 等待");
  28.         }
  29.     }
  30. }
复制代码
生产者
  1. int SensorTransfer::saveSensorData(sensor_data data) {
  2.     //std::lock_guard<std::mutex> lock(queueMutex);
  3.     if (bufferQueue.size() < MAX_QUEUE) { // 限制队列最大长度
  4.         bufferQueue.push(data);
  5.         sensor_data topdata = bufferQueue.top();
  6.         if(topdata.sensorType != 259&&topdata.sensorType != 260){
  7.             dataCondition.notify_one(); // 通知消费者线程
  8.         }
  9.     }else{
  10.         LOGD("buffer is full");
  11.     }
  12.     return 0;
  13. }
复制代码

selinux 添加
这个是老一套了,之前也写过文章,可以参考这里直接贴上主要权限
新增hidtransfer.te
  1. type hidtransfer, domain,mlstrustedsubject;
  2. typeattribute hidtransfer coredomain;
  3. type hidtransfer_exec, system_file_type, exec_type, file_type;
  4. binder_use(hidtransfer)
  5. init_daemon_domain(hidtransfer)
  6. allow hidtransfer system_server:unix_stream_socket {read write};
  7. allow hidtransfer tty_device:chr_file {write read getattr};
  8. allow hidtransfer hid_device:chr_file { read getattr open ioctl write};
  9. allow hidtransfer device:dir read;
  10. allow hidtransfer system_server:binder call;
  11. allow hidtransfer tty_device:chr_file ioctl;
  12. allow hidtransfer serialno_prop:file { map getattr open read};
  13. allow hidtransfer permission_service:service_manager find;
  14. allow hidtransfer sensorservice_service:service_manager find;
  15. allow hidtransfer input_device:chr_file { read write open };
  16. allow hidtransfer input_device:dir { search };
  17. allow hidtransfer device:dir watch;
  18. allow hidtransfer system_server:fd use;
复制代码
file_contexts
  1. /system/bin/hidtransfer u:object_r:hidtransfer_exec:s0
  2. /dev/hidg0 u:object_r:hid_device:s0
复制代码
device.te
  1. type hid_device,dev_type;
复制代码

参考:
1.Android Native Sensor(C++)实例_sensor hal 陀螺仪读取数据实当代码-CSDN博客


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

正序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

南飓风

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表