我希望我的 Android 应用程序创建一个假的虚拟设备,要实现这一点,设备需要 root 并且需要 uinput 模块。
我正在使用以下代码创建设备,调用
static{ System.loadLibrary("myModule"); }
CreateVirtualDevice("Devname",0x123,0x123);
在我的java代码里面。这里是本机代码:
#include <string.h>
#include <jni.h>
#include <fcntl.h>
#include <linux/input.h>
#include <linux/uinput.h>
static int fd;
static struct uinput_user_dev dev;
short int analog_axis_list[] = { ABS_X,ABS_Y,ABS_RX,ABS_RY, -1};
jint Java_com_example_app_MyClass_CreateVirtualDevice(
JNIEnv* env, jobject thiz, jstring param, jint param2, jint param3) {
int i;
memset(&dev, 0, sizeof(dev));
fd = open("/dev/uinput", O_WRONLY | O_NONBLOCK);
if (fd < 0)
return -1;
if(ioctl(fd, UI_SET_EVBIT, EV_ABS)<0) return -4;
for(i=0;analog_axis_list[i]>=0;i++){
if(ioctl(fd,UI_SET_ABSBIT,analog_axis_list[i])<0) return -5;
dev.absmax[analog_axis_list[i]]=32767;
dev.absmin[analog_axis_list[i]]=-32768;
}
const char *cparam = (*env)->GetStringUTFChars(env, param, 0);
snprintf(dev.name, UINPUT_MAX_NAME_SIZE, cparam);
(*env)->ReleaseStringUTFChars(env, param, cparam);
dev.id.bustype = BUS_VIRTUAL;
dev.id.vendor = param2;
dev.id.product = param3;
dev.id.version = 1;
if (write(fd, &dev, sizeof(dev)) < 0)
return -7;
if (ioctl(fd, UI_DEV_CREATE) < 0)
return -8;
return 0;
}
设备创建成功,返回值为0。里面input.h
的ABS
值是这么定义的:
#define ABS_X 0x00
#define ABS_Y 0x01
#define ABS_RX 0x03
#define ABS_RY 0x04
但是在 android 上检查轴时,我得到了正确的 AXIS_X 和 AXIS_Y 值,但是 ABS_RX 和 ABS_RY 的值错误。我使用此代码检查轴值:
InputDevice device = InputDevice.getDevice(ids[position]);
List<InputDevice.MotionRange> ranges = device.getMotionRanges();
StringBuilder sb = new StringBuilder("");
if(ranges.size()==0){
sb.append("NO_MOTION_RANGES");
}
else{
int i = 0;
for(InputDevice.MotionRange range:ranges) {
if(i>0) {
sb.append(",");
}
sb.append(MotionEvent.axisToString(range.getAxis()));
sb.append("(").append(range.getAxis()).append(")");
i++;
}
}
return sb.toString();
结果是:
AXIS_X(0),AXIS_Y(1),AXIS_Z(11),AXIS_RZ(14)
我使用的是最新的 NDK 版本 (r10d),没有启用任何特定设置。什么会导致这些错误?
我想指出这是我的代码有问题,因为对于实际控制器,轴号是正确的。
编辑1:
我试图返回analog_axis_list [2],即ABS_RX
在我的函数末尾而不是0
它返回3
,所以我认为我将错误的类型传递给ioctl
调用。我应该选择哪种类型?